// ==UserScript==
// @name X(Twitter) - Indexed DB Version Add notes to the user
// @name:zh-CN X(Twitter) - Indexed DB 增强版为用户添加备注(别名/标签)
// @name:zh-TW X(Twitter) - Indexed DB 增强版為使用者新增備註(別名/標籤)
// @namespace https://greasyfork.org/zh-CN/users/193133-pana
// @homepage https://greasyfork.org/zh-CN/users/193133-pana
// @icon 
// @version 6.1.15
// @description Add notes (aliases/tags) for users to help identify and search, optimized for large datasets
// @description:zh-CN 为用户添加备注(别名/标签)功能,针对大数据量优化,帮助识别和搜索
// @description:zh-TW 為使用者新增備註(別名/標籤)功能,針對大數據量最佳化,幫助識別和搜尋
// @author pana
// @license GNU General Public License v3.0
// @compatible chrome
// @compatible firefox
// @match *://x.com/*
// @match *://*twitter.com/*
// @require https://gcore.jsdelivr.net/gh/LightAPIs/greasy-fork-library@47d998f5f1e438fe137647b8735b1e17a77e4b69/Note_Obj.js
// @connect *
// @noframes
// @grant GM_info
// @grant GM_getValue
// @grant GM_setValue
// @grant GM_deleteValue
// @grant GM_listValues
// @grant GM_openInTab
// @grant GM_addStyle
// @grant GM_xmlhttpRequest
// @grant GM_registerMenuCommand
// @grant GM_unregisterMenuCommand
// @grant GM_addValueChangeListener
// @grant GM_removeValueChangeListener
// ==/UserScript==
(function () {
"use strict";
const UPDATED = "2024-05-15";
const TWITTER_ICON = {
NOTE_GRAY:
"url()",
NOTE_BLUE:
"url()",
};
const selector = {
root: "#react-root",
homepage: {
id: 'div[data-testid="User-Name"] a[role="link"] > div[dir] > span',
article: "article",
toolBar: '[tabindex="0"]:scope [role="group"][id]',
showName:
'div[data-testid="User-Name"] a[role="link"] > div > div[dir] > span',
reprintA: "a[role][dir][id]",
reprintName: '[data-testid="socialContext"] [dir]',
at: '[data-testid="tweetText"] a[dir][role="link"]',
blockquote: 'div[aria-labelledby][id] div[id] div[role="link"]',
blockquoteId: 'div[data-testid="User-Name"] div[tabindex] div[dir]',
blockquoteShowName: 'div[data-testid="User-Name"] div[dir]',
},
userpage: {
main: ".css-175oi2r.r-ttdzmv.r-1ifxtd0",
id: '[data-testid="UserName"] div[tabindex] div[dir] > span',
showName: '[data-testid="UserName"] div[dir] > span',
follow: ".css-175oi2r.r-obd0qt.r-18u37iz.r-1w6e6rj.r-1h0z5md.r-dnmrzs",
},
comment: {
toolBar: '[tabindex="-1"]:scope [role="group"][id]',
},
hover: {
panel: 'div[data-testid="HoverCard"] > div > div',
userAvatar: '[data-testid^="UserAvatar-Container-"]',
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 nameSet = {
blueTag: "note-obj-twitter-blue-tag",
noteBtn: "note-obj-twitter-note-btn",
panelBtn: "note-obj-twitter-panel-btn",
beforeFollowNoteBtn: "note-obj-twitter-before-follow-note-btn",
baseToolBarBtn: "note-obj-twitter-base-tool-bar-btn",
commentToolBarBtn: "note-obj-twitter-comment-tool-bar-btn",
};
const style = `
.${nameSet.blueTag} {
background-color: #3c81df;
color: #fff;
display: inline-flex;
align-items: center;
padding: 2px 10px;
line-height: 100%;
border-radius: 50px;
}
.${nameSet.noteBtn} {
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;
}
.${nameSet.noteBtn}:hover {
background-image: ${TWITTER_ICON.NOTE_BLUE};
background-color: rgba(29, 161, 242, .1);
}
.${nameSet.panelBtn} {
height: 32px;
width: 32px;
margin: 5px 0px 0px 0px;
background-size: 28px auto;
cursor: pointer !important;
border-radius: 0px;
}
.${nameSet.panelBtn}: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;
}
.${nameSet.beforeFollowNoteBtn} {
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;
}
.${nameSet.beforeFollowNoteBtn}:hover {
background-color: rgba(29, 161, 242, .1);
}
.${nameSet.baseToolBarBtn} {
height: 18px;
width: 18px;
margin: 0px -40px 0px 0px;
background-size: 20px auto;
border-radius: 0px;
margin: 0 12px;
}
.${nameSet.baseToolBarBtn}:hover::after {
content: "";
position: absolute;
background-color: rgba(29, 161, 242, .1);
width: 34px;
height: 34px;
top: -8px;
right: 5px;
border-radius: 99px;
}
.${nameSet.commentToolBarBtn} {
height: 24px;
width: 24px;
margin: 10px 0px 0px 0px;
background-size: 24px auto;
border-radius: 0px;
cursor: pointer;
margin-left: 12px;
}
.${nameSet.commentToolBarBtn}:hover::after {
content: "";
position: absolute;
background-color: rgba(29, 161, 242, .1);
width: 38px;
height: 38px;
top: 3px;
right: -2px;
border-radius: 99px;
}
${selector.homepage.showName}, ${selector.modal.showName} {
white-space: normal;
}
.note-obj-add-frame-dialog button {
text-align: center;
}
.note-obj-management-frame-save-content,
.note-obj-management-frame-cancel-content,
.note-obj-group-frame-save-content,
.note-obj-group-frame-cancel-content {
font-size: 12px;
}`;
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,
},
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 = noteObj.fn.getText(
ele,
selector.homepage.id,
"error",
atFilter
);
if (eleId) {
const eleName = noteObj.fn.getText(
ele,
selector.homepage.showName,
"info"
);
const homepageToolBar = noteObj.fn.query(
ele,
selector.homepage.toolBar,
"info"
);
const commentToolBar = noteObj.fn.query(
ele,
selector.comment.toolBar,
"info"
);
if (homepageToolBar) {
const homepageToolBarBtn = noteObj.fn.query(
homepageToolBar,
"." + Note_Obj.btnClassName,
"none"
);
if (state) {
!homepageToolBarBtn &&
homepageToolBar.appendChild(
noteObj.createNoteBtn(eleId, eleName, [
nameSet.noteBtn,
nameSet.baseToolBarBtn,
])
);
} else {
homepageToolBarBtn && homepageToolBarBtn.remove();
}
}
if (commentToolBar) {
const commentToolBarBtn = noteObj.fn.query(
commentToolBar,
"." + Note_Obj.btnClassName,
"none"
);
if (state) {
!commentToolBarBtn &&
commentToolBar.appendChild(
noteObj.createNoteBtn(eleId, eleName, [
nameSet.noteBtn,
nameSet.commentToolBarBtn,
])
);
} else {
commentToolBarBtn && commentToolBarBtn.remove();
}
}
}
}
function homepageNote(ele, changeId) {
const eleId = noteObj.fn.getText(
ele,
selector.homepage.id,
"error",
atFilter
);
if (eleId) {
if (changeId) {
changeId === eleId &&
noteObj.handler(eleId, ele, selector.homepage.showName, {
add: "span",
className: [nameSet.blueTag],
});
} else {
const eleName = noteObj.fn.getText(
ele,
selector.homepage.showName,
"info"
);
noteObj.handler(
eleId,
ele,
selector.homepage.showName,
{
add: "span",
className: [nameSet.blueTag],
},
eleName
);
}
}
}
function reprintANote(ele, changeId) {
const reprintA = noteObj.fn.queryAnchor(
ele,
selector.homepage.reprintA,
"info"
);
if (reprintA) {
const eleId = noteObj.fn.getIdFromUrl(reprintA.href);
if (!changeId || changeId === eleId) {
noteObj.handler(eleId, reprintA, selector.homepage.reprintName, {
add: "span",
className: [nameSet.blueTag],
offsetWidth: 30,
});
}
}
}
function blockquoteNote(ele, changeId) {
const blockquote = noteObj.fn.query(
ele,
selector.homepage.blockquote,
"info"
);
if (blockquote) {
const blockquoteUser = noteObj.fn.query(
blockquote,
selector.homepage.blockquoteShowName
);
if (blockquoteUser) {
const eleId = noteObj.fn.getText(
blockquote,
selector.homepage.blockquoteId,
"error",
atFilter
);
if (!changeId || changeId === eleId) {
noteObj.handler(eleId, blockquoteUser, undefined, {
add: "span",
className: [nameSet.blueTag],
});
}
}
}
}
function homepageAtNote(ele, state, changeId) {
for (const atUser of noteObj.fn.queryAllAnchor(
ele,
selector.homepage.at,
"info"
)) {
if (hrefComparator(atUser.getAttribute("href") || "")) {
const atUserId = noteObj.fn.getIdFromUrl(atUser.href);
if (!changeId || changeId === atUserId) {
noteObj.handler(atUserId, atUser, undefined, {
prefix: "@",
restore: state,
});
}
}
}
}
function userpageNote(ele, changeId) {
const eleId = noteObj.fn.getText(
ele,
selector.userpage.id,
"error",
atFilter
);
if (changeId) {
changeId === eleId &&
noteObj.handler(eleId, ele, selector.userpage.showName, {
add: "span",
className: [nameSet.blueTag],
});
} else {
const eleName = noteObj.fn.getText(
ele,
selector.userpage.showName,
"info"
);
noteObj.handler(
eleId,
ele,
selector.userpage.showName,
{
add: "span",
className: [nameSet.blueTag],
},
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 = noteObj.fn.getUrlId(ele, idSelector);
if (!changeId || changeId === eleId) {
noteObj.handler(eleId, ele, nameSelector, {
add: "span",
className: [nameSet.blueTag],
});
}
}
function disableInTweetsEvent(status) {
noteObj.fn.queryAll(selector.homepage.article, "none").forEach((ele) => {
homepageAtNote(ele, status);
});
}
function insertToolbarButtonEvent(status) {
noteObj.fn.queryAll(selector.homepage.article, "none").forEach((ele) => {
toolBarNoteButton(ele, status);
});
}
function changeEvent(changeId) {
const articles = noteObj.fn.queryAll(selector.homepage.article, "none");
batchProcess(articles, (ele) => {
try {
homepageNote(ele, changeId);
reprintANote(ele, changeId);
blockquoteNote(ele, changeId);
homepageAtNote(
ele,
noteObj.getOtherConfig().disableInTweets === true,
changeId
);
} catch (e) {
console.error("Process element error:", e);
}
});
noteObj.fn.queryAll(selector.userpage.main).forEach((ele) => {
userpageNote(ele, changeId);
});
noteObj.fn.queryAll(selector.follow.cell, "info").forEach((ele) => {
followNote(ele, changeId);
});
noteObj.fn.queryAll(selector.rightRecommended.cell).forEach((ele) => {
rightRecommendedNote(ele, changeId);
});
noteObj.fn.queryAll(selector.modal.cell, "info").forEach((ele) => {
modalNote(ele, changeId);
});
}
function init() {
try {
const arriveOption = {
fireOnAttributesModification: true,
existing: true,
};
const rootDom = noteObj.fn.query(selector.root);
if (!rootDom) {
console.warn("Root element not found");
return;
}
const throttledCallback = throttle((ele) => {
try {
toolBarNoteButton(
ele,
noteObj.getOtherConfig().showToolbarButton === true
);
homepageNote(ele);
reprintANote(ele);
blockquoteNote(ele);
const disableInTweets =
noteObj.getOtherConfig().disableInTweets === true;
if (!disableInTweets) {
homepageAtNote(ele, disableInTweets);
}
} catch (e) {
console.error("Article processing error:", e);
}
}, 100);
noteObj.arrive(
rootDom,
selector.homepage.article,
arriveOption,
throttledCallback
);
noteObj.arrive(rootDom, selector.userpage.main, arriveOption, (ele) => {
const eleId = noteObj.fn.getText(
ele,
selector.userpage.id,
"error",
atFilter
);
if (eleId) {
const eleName = noteObj.fn.getText(
ele,
selector.userpage.showName,
"info"
);
let followNoteBtn;
const userpageFollow = noteObj.fn.query(
ele,
selector.userpage.follow
);
if (userpageFollow) {
followNoteBtn = noteObj.createNoteBtn(eleId, eleName, [
nameSet.beforeFollowNoteBtn,
"css-901oao",
]);
userpageFollow.insertAdjacentElement("afterbegin", followNoteBtn);
}
const userIdChange = new MutationObserver(() => {
const newUserId = noteObj.fn.getText(
ele,
selector.userpage.id,
"error",
atFilter
);
if (newUserId) {
noteObj.handler("", ele, selector.userpage.showName, {
add: "span",
className: [nameSet.blueTag],
});
const newUserName = noteObj.fn.getText(
ele,
selector.userpage.showName,
"info"
);
if (followNoteBtn) {
followNoteBtn.remove();
followNoteBtn = noteObj.createNoteBtn(newUserId, newUserName, [
nameSet.beforeFollowNoteBtn,
"css-901oao",
]);
userpageFollow &&
userpageFollow.insertAdjacentElement(
"afterbegin",
followNoteBtn
);
}
noteObj.handler(
newUserId,
ele,
selector.userpage.showName,
{
add: "span",
className: [nameSet.blueTag],
},
newUserName
);
}
});
const obId = noteObj.fn.query(ele, selector.userpage.id);
obId &&
userIdChange.observe(obId, {
subtree: true,
characterData: true,
});
}
userpageNote(ele);
});
noteObj.arrive(rootDom, selector.follow.cell, arriveOption, (ele) => {
followNote(ele);
});
noteObj.arrive(
rootDom,
selector.rightRecommended.cell,
arriveOption,
(ele) => {
rightRecommendedNote(ele);
}
);
noteObj.arrive(rootDom, selector.modal.cell, arriveOption, (ele) => {
modalNote(ele);
});
noteObj.arrive(rootDom, selector.hover.panel, arriveOption, (ele) => {
const eleId = noteObj.fn.getUrlId(ele, selector.hover.id);
if (eleId) {
const userShowNameText = noteObj.fn.getText(
ele,
selector.hover.showName,
"info"
);
const userAvatar = noteObj.fn.query(ele, selector.hover.userAvatar);
userAvatar &&
userAvatar.after(
noteObj.createNoteBtn(eleId, userShowNameText, [
nameSet.noteBtn,
nameSet.panelBtn,
])
);
noteObj.handler(
eleId,
ele,
selector.hover.showName,
{
add: "span",
className: [nameSet.blueTag],
},
userShowNameText
);
}
});
} catch (error) {
console.error("Initialization failed:", error);
}
}
init();
})();