🎉 欢迎访问GreasyFork.Org 镜像站!本镜像站由公众号【爱吃馍】搭建,用于分享脚本。联系邮箱📮

Greasy fork 爱吃馍镜像

Twitter - Add notes to the user

Add notes (aliases/tags) for users to help identify and search

Per 27-02-2023. Zie de nieuwste versie.

Voor het installeren van scripts heb je een extensie nodig, zoals Tampermonkey, Greasemonkey of Violentmonkey.

Voor het installeren van scripts heb je een extensie nodig, zoals {tampermonkey_link:Tampermonkey}.

Voor het installeren van scripts heb je een extensie nodig, zoals Tampermonkey of Violentmonkey.

Voor het installeren van scripts heb je een extensie nodig, zoals Tampermonkey of Userscripts.

Voor het installeren van scripts heb je een extensie nodig, zoals {tampermonkey_link:Tampermonkey}.

Voor het installeren van scripts heb je een gebruikersscriptbeheerder nodig.

(Ik heb al een user script manager, laat me het downloaden!)

🚀 安装遇到问题?关注公众号获取帮助

公众号二维码

扫码关注【爱吃馍】

回复【脚本】获取最新教程和防失联地址

Voor het installeren van gebruikersstijlen heb je een extensie nodig, zoals {stylus_link:Stylus}.

Voor het installeren van gebruikersstijlen heb je een extensie nodig, zoals {stylus_link:Stylus}.

Voor het installeren van gebruikersstijlen heb je een extensie nodig, zoals {stylus_link:Stylus}.

Voor het installeren van gebruikersstijlen heb je een gebruikersstijlbeheerder nodig.

Voor het installeren van gebruikersstijlen heb je een gebruikersstijlbeheerder nodig.

Voor het installeren van gebruikersstijlen heb je een gebruikersstijlbeheerder nodig.

(Ik heb al een beheerder - laat me doorgaan met de installatie!)

🚀 安装遇到问题?关注公众号获取帮助

公众号二维码

扫码关注【爱吃馍】

回复【脚本】获取最新教程和防失联地址

// ==UserScript==
// @name                Twitter - Add notes to the user
// @name:zh-CN          Twitter - 为用户添加备注(别名/标签)
// @name:zh-TW          Twitter - 為使用者新增備註(別名/標籤)
// @namespace           https://greasyfork.org/zh-CN/users/193133-pana
// @homepage            https://greasyfork.org/zh-CN/users/193133-pana
// @icon                
// @version             6.0.4
// @description         Add notes (aliases/tags) for users to help identify and search
// @description:zh-CN   为用户添加备注(别名/标签)功能,以帮助识别和搜索
// @description:zh-TW   為使用者新增備註(別名/標籤)功能,以幫助識別和搜尋
// @author              pana
// @license             GNU General Public License v3.0 or later
// @compatible          chrome
// @compatible          firefox
// @match               *://*twitter.com/*
// @require             https://gcore.jsdelivr.net/npm/[email protected]/minified/arrive.min.js
// @require             https://gcore.jsdelivr.net/gh/LightAPIs/greasy-fork-library@1b596ab3b97d13e3aa41dcdad1870b65944fda4d/Note_Obj.js
// @noframes
// @grant               GM_info
// @grant               GM_getValue
// @grant               GM_setValue
// @grant               GM_deleteValue
// @grant               GM_listValues
// @grant               GM_openInTab
// @grant               GM_addStyle
// @grant               GM_registerMenuCommand
// @grant               GM_unregisterMenuCommand
// @grant               GM_addValueChangeListener
// @grant               GM_removeValueChangeListener
// ==/UserScript==

(function () {
    'use strict';
    const UPDATED = '2023-02-27';
    const TWITTER_ICON = {
        NOTE_GRAY: 'url()',
        NOTE_BLUE: 'url()',
    };
    const TWITTER_STYLE = `
    .note-obj-twitter-blue-tag {
        background-color: #3c81df;
        color: #fff;
        display: inline-flex;
        align-items: center;
        padding: 2px 10px;
        line-height: 100%;
        border-radius: 50px;
    }
    .note-obj-twitter-note-btn {
        background-image: ${TWITTER_ICON.NOTE_GRAY};
        background-repeat: no-repeat;
        background-position: center;
        background-color: rgba(0, 0, 0, 0);
        border-bottom-left-radius: 9999px;
        border-bottom-right-radius: 9999px;
        border-top-left-radius: 9999px;
        border-top-right-radius: 9999px;
        transition-property: background-color, box-shadow;
        transition-duration: 0.2s;
    }
    .note-obj-twitter-note-btn:hover {
        background-image: ${TWITTER_ICON.NOTE_BLUE};
        background-color: rgba(29, 161, 242, .1);
    }
    .note-obj-twitter-panel-btn {
        height: 32px;
        width: 32px;
        margin: 5px 0px 0px 0px;
        background-size: 28px auto;
        cursor: pointer !important;
        border-radius: 0px;
    }
    .note-obj-twitter-panel-btn:hover::after {
        content: "";
        display: flex;
        position: relative;
        background-color: rgba(29, 161, 242, .1);
        width: 48px;
        height: 48px;
        top: -8px;
        left: -8px;
        border-radius: 99px;
    }
    .note-obj-twitter-before-follow-note-btn {
        height: 36px;
        width: 36px;
        background-image: ${TWITTER_ICON.NOTE_BLUE};
        background-repeat: no-repeat;
        background-size: 19px auto;
        background-position: center;
        margin-bottom: 12px;
        margin-right: 12px;
        cursor: pointer;
        border: 1px solid rgba(29, 161, 242, 1);
        border-bottom-left-radius: 9999px;
        border-bottom-right-radius: 9999px;
        border-top-left-radius: 9999px;
        border-top-right-radius: 9999px;
        background-color: rgba(0, 0, 0, 0);
        transition-property: background-color, box-shadow;
        transition-duration: 0.2s;
    }
    .note-obj-twitter-before-follow-note-btn:hover {
        background-color: rgba(29, 161, 242, .1);
    }
    .note-obj-twitter-base-tool-bar-btn {
        height: 18px;
        width: 18px;
        margin: 0px -40px 0px 0px;
        background-size: 20px auto;
        border-radius: 0px;
    }
    .note-obj-twitter-base-tool-bar-btn:hover::after {
        content: "";
        position: absolute;
        background-color: rgba(29, 161, 242, .1);
        width: 34px;
        height: 34px;
        top: -8px;
        left: -8px;
        border-radius: 99px;
    }
    .note-obj-twitter-comment-tool-bar-btn {
        height: 24px;
        width: 24px;
        margin: 12px 0px 0px 0px;
        background-size: 24px auto;
        border-radius: 0px;
        cursor: pointer;
    }
    .note-obj-twitter-comment-tool-bar-btn:hover::after {
        content: "";
        position: absolute;
        background-color: rgba(29, 161, 242, .1);
        width: 38px;
        height: 38px;
        top: -8px;
        left: -8px;
        border-radius: 99px;
    }`;
    const selector = {
        root: '#react-root',
        homepage: {
            id: 'div[data-testid="User-Names"] a[role="link"] > div[dir] > span',
            article: 'article',
            toolBar: '.css-1dbjc4n.r-18u37iz.r-1wtj0ep.r-1mdbhws',
            showName: 'div[data-testid="User-Names"] a[role="link"] > div > div[dir] > span',
            reprintA: '.css-1dbjc4n.r-1habvwh.r-16y2uox a',
            reprintName: ':scope > span:first-of-type > span',
            at: 'a.css-4rbku5.css-18t94o4.css-901oao.css-16my406.r-1loqt21.r-bcqeeo.r-qvutc0',
            userFrame: '.css-18t94o4.css-1dbjc4n.r-1loqt21.r-1wbh5a2.r-dnmrzs.r-1ny4l3l',
            blockquote: 'div[aria-labelledby][id] > .css-1dbjc4n > div[role="link"]',
            blockquoteId: ':scope div[tabindex] .css-1dbjc4n > div[dir] > span',
            blockquoteShowName: 'div[data-testid="UserAvatar-Container-unknown"]',
        },
        userpage: {
            main: '.css-1dbjc4n.r-1ifxtd0.r-ymttw5.r-ttdzmv',
            id: '[data-testid="UserName"] .css-1dbjc4n.r-18u37iz.r-1wbh5a2 span',
            showName: '[data-testid="UserName"] .css-901oao.r-1vr29t4.r-bcqeeo.r-qvutc0 > span',
            follow: '.css-1dbjc4n.r-obd0qt.r-18u37iz.r-1w6e6rj.r-1h0z5md.r-dnmrzs',
        },
        comment: {
            toolBar: '.css-1dbjc4n.r-1oszu61.r-1efd50x.r-5kkj8d.r-18u37iz.r-a2tzq0',
        },
        hover: {
            panel: 'div[data-testid="HoverCard"] .r-nsbfu8',
            followBtn: '.css-1dbjc4n.r-bcqeeo',
            id: 'a[role="link"]',
            showName: 'a[role="link"] > div > [dir] > span',
        },
        modal: {
            cell: '[aria-labelledby="modal-header"] [data-testid="UserCell"]',
            id: 'a[role="link"]',
            showName: 'a[role="link"] > div > [dir] > span',
        },
        follow: {
            cell: '[data-testid="cellInnerDiv"] [data-testid="UserCell"]',
            id: 'a[role="link"]',
            showName: 'a[role="link"] > div > [dir] > span',
        },
        rightRecommended: {
            cell: '[role="complementary"] [data-testid="UserCell"]',
            id: 'a[role="link"]',
            showName: 'a[role="link"] > div > [dir]',
        },
    };
    const noteObj = new Note_Obj({
        id: 'myTwitterNote',
        script: {
            author: {
                name: 'pana',
                homepage: 'https://greasyfork.org/zh-CN/users/193133-pana',
            },
            url: 'https://greasyfork.org/scripts/404587',
            updated: UPDATED,
            library: [
                {
                    name: 'arrive.js',
                    version: '2.4.1',
                    url: 'https://github.com/uzairfarooq/arrive',
                },
            ],
        },
        style: selector.homepage.showName + ', ' + selector.modal.showName + ' { white-space: normal; }\n' + TWITTER_STYLE,
        changeEvent: changeEvent,
        settings: {
            showToolbarButton: {
                type: 'checkbox',
                lang: {
                    en: 'Display the "Note" button in the toolbar below each tweet (if there is no such button in the user\'s hover information panel, this option can be turned on)',
                    zhHans: '在每条推特下方的工具栏里显示"备注"按钮 (如果在用户的悬停信息面板里没有此按钮时,可以打开此选项)',
                    zhHant: '在每條推特下方的工具欄裡顯示"備註"按鈕 (如果在使用者的懸停資訊面板裡沒有此按鈕時,可以開啟此選項)',
                },
                default: false,
                event: insertToolbarButtonEvent,
            },
            disableInTweets: {
                type: 'checkbox',
                lang: {
                    en: 'Disable replacing @user with @note in tweets',
                    zhHans: '禁用将推文中的 @user 替换为 @note',
                    zhHant: '禁用將推文中的 @user 替換為 @note',
                },
                default: false,
                event: disableInTweetsEvent,
            },
        },
    });
    function atFilter(text) {
        return text.replace(/^@/, '');
    }
    function hrefComparator(href) {
        return /^\/[^/]+$/i.test(href);
    }
    function toolBarNoteButton(ele, state) {
        const eleId = Note_Obj.fn.getText(ele, selector.homepage.id, 'error', atFilter);
        if (eleId) {
            const eleName = Note_Obj.fn.getText(ele, selector.homepage.showName, 'info');
            const homepageToolBar = Note_Obj.fn.query(ele, selector.homepage.toolBar, 'info');
            const commentToolBar = Note_Obj.fn.query(ele, selector.comment.toolBar, 'info');
            if (homepageToolBar) {
                const homepageToolBarBtn = Note_Obj.fn.query(homepageToolBar, '.' + Note_Obj.btnClassName, 'none');
                if (state) {
                    !homepageToolBarBtn &&
                        homepageToolBar.appendChild(noteObj.createNoteBtn(eleId, eleName, ['note-obj-twitter-note-btn', 'note-obj-twitter-base-tool-bar-btn', 'css-1dbjc4n']));
                }
                else {
                    homepageToolBarBtn && homepageToolBarBtn.remove();
                }
            }
            if (commentToolBar) {
                const commentToolBarBtn = Note_Obj.fn.query(commentToolBar, '.' + Note_Obj.btnClassName, 'none');
                if (state) {
                    !commentToolBarBtn &&
                        commentToolBar.appendChild(noteObj.createNoteBtn(eleId, eleName, ['note-obj-twitter-note-btn', 'note-obj-twitter-comment-tool-bar-btn', 'css-1dbjc4n']));
                }
                else {
                    commentToolBarBtn && commentToolBarBtn.remove();
                }
            }
        }
    }
    function homepageNote(ele, changeId) {
        const eleId = Note_Obj.fn.getText(ele, selector.homepage.id, 'error', atFilter);
        if (eleId) {
            if (changeId) {
                changeId === eleId &&
                    noteObj.handler(eleId, ele, selector.homepage.showName, {
                        add: 'span',
                        className: ['note-obj-twitter-blue-tag'],
                    });
            }
            else {
                const eleName = Note_Obj.fn.getText(ele, selector.homepage.showName, 'info');
                noteObj.handler(eleId, ele, selector.homepage.showName, {
                    add: 'span',
                    className: ['note-obj-twitter-blue-tag'],
                }, eleName);
            }
        }
    }
    function reprintANote(ele, changeId) {
        const reprintA = Note_Obj.fn.queryAnchor(ele, selector.homepage.reprintA, 'info');
        if (reprintA) {
            const eleId = Note_Obj.fn.getIdFromUrl(reprintA.href);
            if (!changeId || changeId === eleId) {
                noteObj.handler(eleId, reprintA, selector.homepage.reprintName, {
                    add: 'span',
                    className: ['note-obj-twitter-blue-tag'],
                    offsetWidth: 30,
                });
            }
        }
    }
    function blockquoteNote(ele, changeId) {
        const blockquote = Note_Obj.fn.query(ele, selector.homepage.blockquote, 'info');
        if (blockquote) {
            const blockquoteUser = Note_Obj.fn.query(blockquote, selector.homepage.blockquoteShowName);
            if (blockquoteUser) {
                const bn = blockquoteUser.nextElementSibling;
                if (bn) {
                    const eleId = Note_Obj.fn.getText(blockquote, selector.homepage.blockquoteId, 'error', atFilter);
                    if (!changeId || changeId === eleId) {
                        noteObj.handler(eleId, bn, undefined, {
                            add: 'span',
                            className: ['note-obj-twitter-blue-tag'],
                        });
                    }
                }
            }
        }
    }
    function homepageAtNote(ele, state, changeId) {
        for (const atUser of Note_Obj.fn.queryAllAnchor(ele, selector.homepage.at, 'info')) {
            if (hrefComparator(atUser.getAttribute('href') || '')) {
                const atUserId = Note_Obj.fn.getIdFromUrl(atUser.href);
                if (!changeId || changeId === atUserId) {
                    noteObj.handler(atUserId, atUser, undefined, {
                        prefix: '@',
                        restore: state,
                    });
                }
            }
        }
    }
    function userpageNote(ele, changeId) {
        const eleId = Note_Obj.fn.getText(ele, selector.userpage.id, 'error', atFilter);
        if (changeId) {
            changeId === eleId &&
                noteObj.handler(eleId, ele, selector.userpage.showName, {
                    add: 'span',
                    className: ['note-obj-twitter-blue-tag'],
                });
        }
        else {
            const eleName = Note_Obj.fn.getText(ele, selector.userpage.showName, 'info');
            noteObj.handler(eleId, ele, selector.userpage.showName, {
                add: 'span',
                className: ['note-obj-twitter-blue-tag'],
            }, eleName);
        }
    }
    function followNote(ele, changeId) {
        spanItemNote(ele, selector.follow.id, selector.follow.showName, changeId);
    }
    function rightRecommendedNote(ele, changeId) {
        spanItemNote(ele, selector.rightRecommended.id, selector.rightRecommended.showName, changeId);
    }
    function modalNote(ele, changeId) {
        spanItemNote(ele, selector.modal.id, selector.modal.showName, changeId);
    }
    function spanItemNote(ele, idSelector, nameSelector, changeId) {
        const eleId = Note_Obj.fn.getUrlId(ele, idSelector);
        if (!changeId || changeId === eleId) {
            noteObj.handler(eleId, ele, nameSelector, {
                add: 'span',
                className: ['note-obj-twitter-blue-tag'],
            });
        }
    }
    function disableInTweetsEvent(status) {
        Note_Obj.fn.docQueryAll(selector.homepage.article, 'none').forEach(ele => {
            homepageAtNote(ele, status);
        });
    }
    function insertToolbarButtonEvent(status) {
        Note_Obj.fn.docQueryAll(selector.homepage.article, 'none').forEach(ele => {
            toolBarNoteButton(ele, status);
        });
    }
    function changeEvent(changeId) {
        Note_Obj.fn.docQueryAll(selector.homepage.article, 'none').forEach(ele => {
            homepageNote(ele, changeId);
            reprintANote(ele, changeId);
            blockquoteNote(ele, changeId);
            homepageAtNote(ele, noteObj.getOtherConfig().disableInTweets === true, changeId);
        });
        Note_Obj.fn.docQueryAll(selector.userpage.main).forEach(ele => {
            userpageNote(ele, changeId);
        });
        Note_Obj.fn.docQueryAll(selector.follow.cell, 'info').forEach(ele => {
            followNote(ele, changeId);
        });
        Note_Obj.fn.docQueryAll(selector.rightRecommended.cell).forEach(ele => {
            rightRecommendedNote(ele, changeId);
        });
        Note_Obj.fn.docQueryAll(selector.modal.cell, 'info').forEach(ele => {
            modalNote(ele, changeId);
        });
    }
    function init() {
        const arriveOption = {
            fireOnAttributesModification: true,
            existing: true,
        };
        const rootDom = Note_Obj.fn.docQuery(selector.root);
        if (rootDom === null) {
            return;
        }
        rootDom.arrive(selector.homepage.article, arriveOption, ele => {
            toolBarNoteButton(ele, noteObj.getOtherConfig().showToolbarButton === true);
            homepageNote(ele);
            reprintANote(ele);
            blockquoteNote(ele);
            const disableInTweets = noteObj.getOtherConfig().disableInTweets === true;
            if (!disableInTweets) {
                homepageAtNote(ele, disableInTweets);
            }
        });
        rootDom.arrive(selector.userpage.main, arriveOption, ele => {
            const eleId = Note_Obj.fn.getText(ele, selector.userpage.id, 'error', atFilter);
            if (eleId) {
                const eleName = Note_Obj.fn.getText(ele, selector.userpage.showName, 'info');
                let followNoteBtn;
                const userpageFollow = Note_Obj.fn.query(ele, selector.userpage.follow);
                if (userpageFollow) {
                    followNoteBtn = noteObj.createNoteBtn(eleId, eleName, ['note-obj-twitter-before-follow-note-btn', 'css-901oao']);
                    userpageFollow.insertAdjacentElement('afterbegin', followNoteBtn);
                }
                const userIdChange = new MutationObserver(() => {
                    const newUserId = Note_Obj.fn.getText(ele, selector.userpage.id, 'error', atFilter);
                    if (newUserId) {
                        noteObj.handler('', ele, selector.userpage.showName, {
                            add: 'span',
                            className: ['note-obj-twitter-blue-tag'],
                        });
                        const newUserName = Note_Obj.fn.getText(ele, selector.userpage.showName, 'info');
                        if (followNoteBtn) {
                            followNoteBtn.remove();
                            followNoteBtn = noteObj.createNoteBtn(newUserId, newUserName, ['note-obj-twitter-before-follow-note-btn', 'css-901oao']);
                            userpageFollow && userpageFollow.insertAdjacentElement('afterbegin', followNoteBtn);
                        }
                        noteObj.handler(newUserId, ele, selector.userpage.showName, {
                            add: 'span',
                            className: ['note-obj-twitter-blue-tag'],
                        }, newUserName);
                    }
                });
                const obId = Note_Obj.fn.query(ele, selector.userpage.id);
                obId &&
                    userIdChange.observe(obId, {
                        subtree: true,
                        characterData: true,
                    });
            }
            userpageNote(ele);
        });
        rootDom.arrive(selector.follow.cell, arriveOption, ele => {
            followNote(ele);
        });
        rootDom.arrive(selector.rightRecommended.cell, arriveOption, ele => {
            rightRecommendedNote(ele);
        });
        rootDom.arrive(selector.modal.cell, arriveOption, ele => {
            modalNote(ele);
        });
        rootDom.arrive(selector.hover.panel, arriveOption, ele => {
            const eleId = Note_Obj.fn.getUrlId(ele, selector.hover.id);
            if (eleId) {
                const userShowNameText = Note_Obj.fn.getText(ele, selector.hover.showName, 'info');
                const hoverFollowBtn = Note_Obj.fn.query(ele, selector.hover.followBtn);
                hoverFollowBtn &&
                    hoverFollowBtn.insertAdjacentElement('beforebegin', noteObj.createNoteBtn(eleId, userShowNameText, ['note-obj-twitter-note-btn', 'note-obj-twitter-panel-btn']));
                noteObj.handler(eleId, ele, selector.hover.showName, {
                    add: 'span',
                    className: ['note-obj-twitter-blue-tag'],
                }, userShowNameText);
            }
        });
    }
    init();
})();