// ==UserScript== // @name Twitter - Add notes to the user // @name:zh-CN Twitter - 为用户添加备注(别名/标签) // @name:zh-TW Twitter - 為用戶添加備註(別名/標籤) // @name:ja Twitter - ユーザーへのメモの追加(エイリアス/ラベル) // @name:ko Twitter - 사용자에게 메모 추가 (별칭/라벨) // @name:fr Twitter - ajouter des notes aux utilisateurs (alias/tag) // @namespace https://greasyfork.org/zh-CN/users/193133-pana // @homepage https://greasyfork.org/zh-CN/users/193133-pana // @icon  // @version 4.3.1 // @description Add a note(alias/tag) for users to help identify and search // @description:zh-CN 为用户添加备注(别名/标签)功能,以帮助识别和搜索 // @description:zh-TW 為用戶添加備註(別名/標籤)功能,以幫助識別和搜尋 // @description:ja ユーザーが識別と検索に役立つメモ(エイリアス/タグ)機能を追加する // @description:ko 사용자 식별 및 검색에 도움이되는 메모 (별칭/태그) 기능 추가 // @description:fr Ajouter une fonction de notes (alias/tag) pour les utilisateurs pour aider à identifier et rechercher // @author pana // @license GNU General Public License v3.0 or later // @compatible chrome // @compatible firefox // @match *://*twitter.com/* // @require https://cdn.jsdelivr.net/npm/arrive@2.4.1/minified/arrive.min.js // @require https://cdn.jsdelivr.net/npm/vue@2.6.14/dist/vue.min.js // @require https://cdn.jsdelivr.net/gh/LightAPIs/greasy-fork-library@c5961e56c4461790de3a52f1502e6c007ff64b4a/Note_Obj.js // @noframes // @grant GM_info // @grant GM.info // @grant GM_getValue // @grant GM.getValue // @grant GM_setValue // @grant GM.setValue // @grant GM_deleteValue // @grant GM.deleteValue // @grant GM_listValues // @grant GM.listValues // @grant GM_openInTab // @grant GM.openInTab // @grant GM_registerMenuCommand // @grant GM_unregisterMenuCommand // @grant GM_addValueChangeListener // @grant GM_removeValueChangeListener // @downloadURL none // ==/UserScript== (async function () { 'use strict'; if (typeof Note_Obj !== 'function') { alert('Note_Obj.js was not loaded successfully!'); } const UPDATED = '2022-06-04'; 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; } .note-obj-twitter-left-box { height: 50%; } `; const selector = { root: '#react-root div .r-13awgt0.r-12vffkv', homepage: { id: '.css-901oao.css-bfa6kz.r-18u37iz.r-37j5jr.r-16dba41.r-bcqeeo.r-qvutc0 > span', article: 'article', toolBar: '.css-1dbjc4n.r-18u37iz.r-1wtj0ep.r-1mdbhws', showName: '.css-901oao.css-bfa6kz.r-bcqeeo.r-poiln3.r-qvutc0 > 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[role="blockquote"]', }, 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.css-1dbjc4n.r-nsbfu8', followBtn: '.css-1dbjc4n.r-bcqeeo', id: '.css-901oao.css-bfa6kz.r-18u37iz.r-37j5jr.r-16dba41.r-bcqeeo.r-qvutc0 > span', showName: '.css-1dbjc4n.r-1awozwy.r-18u37iz .css-901oao.css-bfa6kz.r-bcqeeo.r-poiln3.r-qvutc0 > span', }, modal: { cell: '[aria-labelledby="modal-header"] [data-testid="UserCell"]', }, follow: { cell: '[data-testid="cellInnerDiv"] [data-testid="UserCell"]', }, rightRecommended: { cell: '[role="complementary"] [data-testid="UserCell"]', }, }; const noteObj = new Note_Obj('myTwitterNote'); await noteObj.init({ style: selector.homepage.showName + ' { white-space: normal; }\n' + TWITTER_STYLE, changeEvent: changeEvent, settings: { showToolbarButton: { type: 'checkbox', lang: { en: 'Display the "Add 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)', zh_cn: '在每条推特下方的工具栏里显示"添加备注"按钮 (如果在用户的悬停信息面板里没有此按钮时,可以打开此选项)', zh_tw: '在每條推特下方的工具欄裡顯示"添加備註"按鈕 (如果在用戶的懸停資訊面板裡沒有此按鈕時,可以打開此選項)', ja: '各Twitterの下のツールバーに“備考追加”ボタンが表示されます(ユーザのホバリング情報パネルにこのボタンがない場合は、このオプションを開くことができます)', ko: '각 트위터 아래의 도구 모음에 "메모 추가" 단추가 표시됩니다(사용자의 롤오버 정보 패널에 이 단추가 없는 경우 이 옵션을 설정할 수 있음)', fr: "Afficher le bouton \"Ajouter une note\" dans la barre d'outils sous chaque tweet (S'il n'y a pas de bouton de ce type dans le panneau d'informations de survol de l'utilisateur, vous pouvez activer cette option)", }, default: false, event: insertToolbarButtonEvent, }, disableInTweets: { type: 'checkbox', lang: { en: 'Disable replaces peoples @username to @note in tweets', zh_cn: '禁止将推文中的用户 @username 替换为 @note', zh_tw: '禁止將推文中的使用者 @username 替换为 @note', ja: '無効にすると、ツイート内のユーザーの@usernameが@noteに置き換えられます', ko: '비활성화는 트윗에서 @username을 @note로 대체합니다', fr: "Désactiver remplace le @nom d'utilisateur par @note dans les tweets", }, default: false, event: disableInTweetsEvent, }, }, script: { author: { name: 'pana', homepage: 'https://greasyfork.org/zh-CN/users/193133-pana', }, address: 'https://greasyfork.org/scripts/404587', updated: UPDATED, library: [ { name: 'arrive.js', version: '2.4.1', url: 'https://github.com/uzairfarooq/arrive', }, ], }, leftBtnBoxClassName: 'note-obj-twitter-left-box', }); function disableInTweetsEvent(status) { for (const ele of document.querySelectorAll(selector.homepage.article)) { for (const atUser of ele.querySelectorAll(selector.homepage.at)) { const atUserId = Note_Obj.fn.getUserIdFromLink(atUser.href, value => /^[^/]+$/i.test(value)); noteObj.judgeUsers(atUserId) && noteObj.handler(atUserId, atUser, null, { symbol: { prefix: '@', }, restore: status, }); } } } function insertToolbarButtonEvent(status) { document.querySelectorAll(selector.homepage.article).forEach(ele => { if (ele.querySelector(selector.homepage.id)) { const eleId = ele.querySelector(selector.homepage.id).textContent.replace(/^@/, ''); const eleName = ele.querySelector(selector.homepage.showName).textContent; const toolBar = ele.querySelector(selector.homepage.toolBar); const commentToolBar = ele.querySelector(selector.comment.toolBar); if (status) { toolBar && !toolBar.querySelector('.note-obj-add-note-btn') && toolBar.appendChild( noteObj.createNoteBtn(eleId, eleName, ['note-obj-twitter-note-btn', 'note-obj-twitter-base-tool-bar-btn', 'css-1dbjc4n']) ); commentToolBar && !commentToolBar.querySelector('.note-obj-add-note-btn') && commentToolBar.appendChild( noteObj.createNoteBtn(eleId, eleName, ['note-obj-twitter-note-btn', 'note-obj-twitter-comment-tool-bar-btn', 'css-1dbjc4n']) ); } else { toolBar && toolBar.querySelector('.note-obj-add-note-btn') && toolBar.querySelector('.note-obj-add-note-btn').remove(); commentToolBar && commentToolBar.querySelector('.note-obj-add-note-btn') && commentToolBar.querySelector('.note-obj-add-note-btn').remove(); } } }); } function changeEvent(noteObj, userId = null) { for (const ele of document.querySelectorAll(selector.homepage.article)) { if (ele.querySelector(selector.homepage.id)) { const eleId = ele.querySelector(selector.homepage.id).textContent.replace(/^@/, ''); (!userId || userId == eleId) && noteObj.handler(eleId, ele, selector.homepage.showName, { add: 'span', classname: 'note-obj-twitter-blue-tag', }); } const reprintA = ele.querySelector(selector.homepage.reprintA); if (reprintA) { const reprintId = Note_Obj.fn.getUserIdFromLink(reprintA.href); (!userId || userId == reprintId) && noteObj.handler(reprintId, reprintA, selector.homepage.reprintName, { add: 'span', classname: 'note-obj-twitter-blue-tag', symbol: { offsetWidth: 30, }, }); } const blockquoteUser = ele.querySelector(selector.homepage.blockquote); if (blockquoteUser) { const blockquoteUserId = blockquoteUser.querySelector(selector.homepage.id).textContent.replace(/^@/, ''); if (blockquoteUserId == userId) { noteObj.handler(userId, blockquoteUser, selector.homepage.showName); } (!userId || userId == blockquoteUserId) && noteObj.handler(blockquoteUserId, blockquoteUser, selector.homepage.showName, { add: 'span', classname: 'note-obj-twitter-blue-tag', }); } for (const atUser of ele.querySelectorAll(selector.homepage.at)) { const atUserId = Note_Obj.fn.getUserIdFromLink(atUser.href, value => /^[^/]+$/i.test(value)); (!userId || userId == atUserId) && noteObj.judgeUsers(atUserId) && noteObj.handler(atUserId, atUser, null, { symbol: { prefix: '@', }, restore: noteObj.getConfig().other.disableInTweets, }); } } for (const ele of document.querySelectorAll(selector.userpage.main)) { const user = ele.querySelector(selector.userpage.id); if (user) { const eleId = user.textContent.replace(/^@/, ''); (!userId || userId == eleId) && noteObj.handler(eleId, ele, selector.userpage.showName, { add: 'span', classname: 'note-obj-twitter-blue-tag', }); } } for (const ele of document.querySelectorAll(selector.follow.cell)) { const user = ele.querySelector(selector.homepage.id); if (user) { const eleId = user.textContent.replace(/^@/, ''); (!userId || userId == eleId) && noteObj.handler(eleId, ele, selector.homepage.showName, { add: 'span', class: 'note-obj-twitter-blue-tag', }); } } for (const ele of document.querySelectorAll(selector.rightRecommended.cell)) { const user = ele.querySelector(selector.homepage.id); if (user) { const eleId = user.textContent.replace(/^@/, ''); (!userId || userId == eleId) && noteObj.handler(eleId, ele, selector.homepage.showName, { add: 'span', class: 'note-obj-twitter-blue-tag', }); } } for (const ele of document.querySelectorAll(selector.modal.cell)) { const user = ele.querySelector(selector.homepage.id); if (user) { const eleId = user.textContent.replace(/^@/, ''); (!userId || userId == eleId) && noteObj.handler(eleId, ele, selector.homepage.showName, { add: 'span', class: 'note-obj-twitter-blue-tag', }); } } } function init() { const arriveOption = { fireOnAttributesModification: true, existing: true, }; document.querySelector(selector.root).arrive(selector.homepage.article, arriveOption, ele => { if (ele.querySelector(selector.homepage.id)) { const eleId = ele.querySelector(selector.homepage.id).textContent.replace(/^@/, ''); const eleName = ele.querySelector(selector.homepage.showName).textContent; noteObj.getConfig().other.showToolbarButton && ele.querySelector(selector.homepage.toolBar) && ele .querySelector(selector.homepage.toolBar) .appendChild( noteObj.createNoteBtn(eleId, eleName, ['note-obj-twitter-note-btn', 'note-obj-twitter-base-tool-bar-btn', 'css-1dbjc4n']) ); noteObj.getConfig().other.showToolbarButton && ele.querySelector(selector.comment.toolBar) && ele .querySelector(selector.comment.toolBar) .appendChild( noteObj.createNoteBtn(eleId, eleName, ['note-obj-twitter-note-btn', 'note-obj-twitter-comment-tool-bar-btn', 'css-1dbjc4n']) ); noteObj.judgeUsers(eleId) && noteObj.handler( eleId, ele, selector.homepage.showName, { add: 'span', classname: 'note-obj-twitter-blue-tag', }, eleName ); } const reprintA = ele.querySelector(selector.homepage.reprintA); if (reprintA) { const reprintId = Note_Obj.fn.getUserIdFromLink(reprintA.href); noteObj.judgeUsers(reprintId) && noteObj.handler(reprintId, reprintA, selector.homepage.reprintName, { add: 'span', classname: 'note-obj-twitter-blue-tag', symbol: { offsetWidth: 30, }, }); } const blockquoteUser = ele.querySelector(selector.homepage.blockquote); if (blockquoteUser) { const blockquoteUserId = blockquoteUser.querySelector(selector.homepage.id).textContent.replace(/^@/, ''); noteObj.judgeUsers(blockquoteUserId) && noteObj.handler(blockquoteUserId, blockquoteUser, selector.homepage.showName, { add: 'span', classname: 'note-obj-twitter-blue-tag', }); } if (!noteObj.getConfig().other.disableInTweets) { for (const atUser of ele.querySelectorAll(selector.homepage.at)) { const atUserId = Note_Obj.fn.getUserIdFromLink(atUser.href, value => /^[^/]+$/i.test(value)); noteObj.judgeUsers(atUserId) && noteObj.handler(atUserId, atUser, null, { symbol: { prefix: '@', }, }); } } }); document.querySelector(selector.root).arrive(selector.userpage.main, arriveOption, ele => { const eleId = ele.querySelector(selector.userpage.id).textContent.replace(/^@/, ''); const eleName = ele.querySelector(selector.userpage.showName).textContent; let followNoteBtn; if (ele.querySelector(selector.userpage.follow)) { followNoteBtn = noteObj.createNoteBtn(eleId, eleName, ['note-obj-twitter-before-follow-note-btn', 'css-901oao']); ele.querySelector(selector.userpage.follow).insertAdjacentElement('afterbegin', followNoteBtn); } noteObj.judgeUsers(eleId) && noteObj.handler( eleId, ele, selector.userpage.showName, { add: 'span', classname: 'note-obj-twitter-blue-tag', }, eleName ); const userIdChange = new MutationObserver(() => { const newUserId = ele.querySelector(selector.userpage.id).textContent.replace(/^@/, ''); noteObj.handler('', ele, selector.userpage.showName, { add: 'span', classname: 'note-obj-twitter-blue-tag', }); const newUserName = ele.querySelector(selector.userpage.showName).textContent; if (followNoteBtn) { followNoteBtn.remove(); followNoteBtn = noteObj.createNoteBtn(newUserId, newUserName, ['note-obj-twitter-before-follow-note-btn', 'css-901oao']); ele.querySelector(selector.userpage.follow).insertAdjacentElement('afterbegin', followNoteBtn); } noteObj.judgeUsers(newUserId) && noteObj.handler( newUserId, ele, selector.userpage.showName, { add: 'span', classname: 'note-obj-twitter-blue-tag', }, newUserName ); }); userIdChange.observe(ele.querySelector(selector.userpage.id), { subtree: true, characterData: true, }); }); document.querySelector(selector.root).arrive(selector.follow.cell, arriveOption, ele => { const eleId = ele.querySelector(selector.homepage.id) ? ele.querySelector(selector.homepage.id).textContent.replace(/^@/, '') : null; noteObj.judgeUsers(eleId) && noteObj.handler(eleId, ele, selector.homepage.showName, { add: 'span', class: 'note-obj-twitter-blue-tag', }); }); document.querySelector(selector.root).arrive(selector.rightRecommended.cell, arriveOption, ele => { const eleId = ele.querySelector(selector.homepage.id) ? ele.querySelector(selector.homepage.id).textContent.replace(/^@/, '') : null; noteObj.judgeUsers(eleId) && noteObj.handler(eleId, ele, selector.homepage.showName, { add: 'span', class: 'note-obj-twitter-blue-tag', }); }); document.querySelector(selector.root).arrive(selector.hover.panel, arriveOption, ele => { const user = ele.querySelector(selector.hover.id); if (user) { const eleId = user.textContent.replace(/^@/, ''); const userShowName = ele.querySelector(selector.hover.showName); if (userShowName) { const userShowNameText = userShowName.textContent; ele.querySelector(selector.hover.followBtn) && ele .querySelector(selector.hover.followBtn) .insertAdjacentElement( 'beforebegin', noteObj.createNoteBtn(eleId, userShowNameText, ['note-obj-twitter-note-btn', 'note-obj-twitter-panel-btn']) ); noteObj.judgeUsers(eleId) && noteObj.handler( eleId, ele, selector.hover.showName, { add: 'span', class: 'note-obj-twitter-blue-tag', }, userShowNameText ); } } }); document.querySelector(selector.root).arrive(selector.modal.cell, arriveOption, ele => { const eleId = ele.querySelector(selector.homepage.id) ? ele.querySelector(selector.homepage.id).textContent.replace(/^@/, '') : null; noteObj.judgeUsers(eleId) && noteObj.handler(eleId, ele, selector.homepage.showName, { add: 'span', class: 'note-obj-twitter-blue-tag', }); }); } init(); })();