Windows 11 国旗 Emoji 修复器

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

// ==UserScript==
// @name         Windows 11 国旗 Emoji 修复器
// @name:zh      Windows 11 国旗 Emoji 修复器
// @name:en      Windows 11 Flag Emoji Fixer
// @namespace    http://tampermonkey.net/
// @version      2.3
// @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/gh/mozilla/twemoji-colr@latest/assets/TwemojiMozilla-colr.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/gh/mozilla/twemoji-colr@latest/assets/TwemojiMozilla-colr.woff2') format('woff2'),
                 url('https://cdn.jsdelivr.net/gh/mozilla/twemoji-colr@latest/fonts/TwemojiMozilla.ttf') format('truetype');
            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);
    });

})();