Windows 11 国旗 Emoji 修复器

通过精确包裹旗帜 Emoji 的方式,修复 Windows 11 上的显示问题,不影响任何网站原有字体。

您需要先安装一个扩展,例如 篡改猴Greasemonkey暴力猴,之后才能安装此脚本。

您需要先安装一个扩展,例如 篡改猴暴力猴,之后才能安装此脚本。

您需要先安装一个扩展,例如 篡改猴暴力猴,之后才能安装此脚本。

您需要先安装一个扩展,例如 篡改猴Userscripts ,之后才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。

您需要先安装用户脚本管理器扩展后才能安装此脚本。

(我已经安装了用户脚本管理器,让我安装!)

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

(我已经安装了用户样式管理器,让我安装!)

// ==UserScript==
// @name         Windows 11 国旗 Emoji 修复器
// @name:zh      Windows 11 国旗 Emoji 修复器
// @name:en      Windows 11 Flag Emoji Fixer
// @namespace    http://tampermonkey.net/
// @version      2.4
// @icon         
// @description  通过精确包裹旗帜 Emoji 的方式,修复 Windows 11 上的显示问题,不影响任何网站原有字体。
// @description:zh 通过精确包裹旗帜 Emoji 的方式,修复 Windows 11 上的显示问题,不影响任何网站原有字体。
// @description:en Fixes Windows 11 flag emoji display issues by precisely wrapping flag characters, without affecting website fonts.
// @author       DeepSeek 3.2 & Gemini 2.5 pro
// @match        *://*/*
// @grant        GM_addStyle
// @license      MIT
// ==/UserScript==

(function() {
    'use strict';

    // 预加载字体
    const preloadLink = document.createElement('link');
    preloadLink.rel = 'preload';
    preloadLink.href = 'https://cdn.jsdelivr.net/npm/twemoji-colr-font@latest/twemoji.woff2';
    preloadLink.as = 'font';
    preloadLink.type = 'font/woff2';
    preloadLink.crossOrigin = 'anonymous';

    // 字体加载失败处理
    preloadLink.onerror = function() {
        console.warn('Windows 11 Flag Emoji Fixer: 字体加载失败,使用备用方案');
    };

    document.head.appendChild(preloadLink);

    // CSS保持精简
    const css = `
        @font-face {
            font-family: 'Windows11FlagFix';
            src: local('Twemoji Mozilla'),
                 local('TwemojiMozilla'),
                 url('https://cdn.jsdelivr.net/npm/twemoji-colr-font@latest/twemoji.woff2') format('woff2');
            unicode-range: U+1F1E6-1F1FF;
            font-display: swap;
        }

        .flag-emoji-fix-wrapper {
            font-family: 'Windows11FlagFix', 'Segoe UI Emoji', 'Apple Color Emoji', 'Noto Color Emoji', sans-serif !important;
            /* 确保包裹元素不会破坏布局 */
            display: inline !important;
            font-size: inherit !important;
            font-weight: inherit !important;
            font-style: inherit !important;
            line-height: inherit !important;
            letter-spacing: inherit !important;
            word-spacing: inherit !important;
        }
    `;

    GM_addStyle(css);

    // --- 优化后的 JavaScript 逻辑 ---

    // 将测试用的正则提到外部,避免重复创建
    const flagTestRegex = /[\u{1F1E6}-\u{1F1FF}]{2}/u;

    function fixFlagEmojisInNode(contextNode) {
        // 用于匹配两个连续的区域指示符号的正则表达式
        const flagRegex = /([\u{1F1E6}-\u{1F1FF}]{2})/gu;

        // 创建 TreeWalker 来遍历文本节点
        const walker = document.createTreeWalker(
            contextNode || document.body,
            NodeFilter.SHOW_TEXT,
            {
                acceptNode: function(node) {
                    // 过滤掉不可见、脚本、样式或已处理的节点
                    if (!node.textContent ||
                        node.parentElement.offsetParent === null ||
                        ['SCRIPT', 'STYLE', 'TEXTAREA', 'NOSCRIPT'].includes(node.parentElement.tagName) ||
                        node.parentElement.isContentEditable ||
                        node.parentElement.closest('.flag-emoji-fix-wrapper')) {
                        return NodeFilter.FILTER_REJECT;
                    }

                    // 使用外部定义的正则表达式
                    return flagTestRegex.test(node.textContent) ?
                        NodeFilter.FILTER_ACCEPT : NodeFilter.FILTER_REJECT;
                }
            }
        );

        const nodesToProcess = [];
        let currentNode;
        while ((currentNode = walker.nextNode())) {
            nodesToProcess.push(currentNode);
        }

        nodesToProcess.forEach(node => {
            // 再次检查节点是否已被处理
            if (!node.parentNode || node.parentNode.closest('.flag-emoji-fix-wrapper')) {
                return;
            }

            const parent = node.parentNode;
            const fragment = document.createDocumentFragment();
            let lastIndex = 0;
            let match;

            // 重置正则表达式的状态
            flagRegex.lastIndex = 0;

            while ((match = flagRegex.exec(node.textContent)) !== null) {
                // 添加旗帜前的文本
                if (match.index > lastIndex) {
                    fragment.appendChild(document.createTextNode(node.textContent.slice(lastIndex, match.index)));
                }

                // 创建并添加包裹旗帜的 span
                const wrapper = document.createElement('span');
                wrapper.className = 'flag-emoji-fix-wrapper';
                wrapper.textContent = match[0];
                fragment.appendChild(wrapper);

                lastIndex = flagRegex.lastIndex;
            }

            // 仅当正则表达式至少匹配到一次时才执行替换
            if (lastIndex > 0) {
                // 如果文本末尾还有内容,则追加
                if (lastIndex < node.textContent.length) {
                    fragment.appendChild(document.createTextNode(node.textContent.slice(lastIndex)));
                }
                parent.replaceChild(fragment, node);
            }
        });
    }

    // 初始执行
    if (document.readyState === 'loading') {
        document.addEventListener('DOMContentLoaded', () => fixFlagEmojisInNode(document.body));
    } else {
        fixFlagEmojisInNode(document.body);
    }

    // 使用防抖优化性能
    let fixTimeout;
    function debouncedFix(mutations) {
        clearTimeout(fixTimeout);
        fixTimeout = setTimeout(() => {
            // 使用 Set 存储待处理的根节点,自动去重
            const nodesToScan = new Set();

            for (const mutation of mutations) {
                if (mutation.type === 'childList') {
                    for (const addedNode of mutation.addedNodes) {
                        if (addedNode.nodeType === Node.ELEMENT_NODE) {
                            nodesToScan.add(addedNode);
                        } else if (addedNode.nodeType === Node.TEXT_NODE && addedNode.parentElement) {
                            nodesToScan.add(addedNode.parentElement);
                        }
                    }
                } else if (mutation.type === 'characterData') {
                    if (mutation.target && mutation.target.parentElement) {
                        nodesToScan.add(mutation.target.parentElement);
                    }
                }
            }

            // 遍历去重后的节点集合,执行修复
            for (const node of nodesToScan) {
                // 检查节点是否仍然在 DOM 中
                if (document.body.contains(node)) {
                   fixFlagEmojisInNode(node);
                }
            }
        }, 100);
    }

    // 监听动态内容变化 - 增加 characterData 监听
    const observer = new MutationObserver(debouncedFix);

    observer.observe(document.body, {
        childList: true,
        subtree: true,
        characterData: true
    });

    // 处理AJAX加载的内容
    const originalPushState = history.pushState;
    const originalReplaceState = history.replaceState;

    history.pushState = function() {
        originalPushState.apply(this, arguments);
        setTimeout(() => fixFlagEmojisInNode(document.body), 500);
    };

    history.replaceState = function() {
        originalReplaceState.apply(this, arguments);
        setTimeout(() => fixFlagEmojisInNode(document.body), 500);
    };

    // 监听hashchange事件
    window.addEventListener('hashchange', () => {
        setTimeout(() => fixFlagEmojisInNode(document.body), 500);
    });

})();