// ==UserScript==
// @name tradingview screener assistant
// @namespace http://tampermonkey.net/
// @version 2024-09-20.3
// @description insert batch copy button, chart copy button ,chart button and blacklist button in tradingview screener
// @author goodzhuwang
// @match https://*.tradingview.com/screener/*
// @icon 
// @grant none
// @license MIT
// ==/UserScript==
(function () {
"use strict";
const ext_name = "tv_assistant";
console.debug(`${ext_name} running`);
let batch_button_id = "__batch_copybtn";
// 代码黑名单
let blacklist = [];
// 最新的中概股名单 : https://stockanalysis.com/list/chinese-stocks-us/
let chinese_stocks = [
"BABA",
"NTES",
"JD",
"BIDU",
"PUK",
"LI",
"BEKE",
"ZTO",
"TME",
"YUMC",
"NIO",
"EDU",
"HTHT",
"FUTU",
"XPEV",
"YMM",
"VIPS",
"BILI",
"BZ",
"MNSO",
"TAL",
"ZK",
"QFIN",
"GDS",
"LOT",
"ATHM",
"ATAT",
"HCM",
"MLCO",
"RLX",
"ZLAB",
"IQ",
"LU",
"WB",
"SIMO",
"FINV",
"DQ",
"MOMO",
"MSC",
"JKS",
"SDA",
"HUYA",
"VNET",
"TUYA",
"XCH",
"EH",
"GOTU",
"TIGR",
"ECX",
"DDL",
"NOAH",
"HSAI",
"KC",
"ZKH",
"SOHU",
"RERE",
"ICG",
"ZJYL",
"YRD",
"DOGZ",
"WDH",
"PGHL",
"DAO",
"QD",
"DADA",
"YSG",
"ZH",
"TROO",
"JFIN",
"UXIN",
"YXT",
"LX",
"DOYU",
"RITR",
"GHG",
"DSY",
"XYF",
"LANV",
"AHG",
"BSII",
"HPH",
"AGBA",
"CANG",
"RTC",
"BZUN",
"NIU",
"KNDI",
"LSB",
"EM",
"YIBO",
"ZBAO",
"GSIW",
"SRL",
"AZI",
"SFWL",
"CMCM",
"CAAS",
"TOUR",
"XNET",
"THCH",
"ADAG",
"CASI",
"QMMM",
"LGCL",
"QH",
"VIOT",
"GLAC",
"CBAT",
"QSG",
"STG",
"GLXG",
"IH",
"RGC",
"WIMI",
"JVSA",
"SY",
"NHTC",
"LICN",
"DIST",
"XHG",
"CNF",
"FEBO",
"FANH",
"WOK",
"JUNE",
"CCG",
"PTHL",
"YHNA",
"NCTY",
"CHSN",
"BEDU",
"OCFT",
"BGM",
"NISN",
"TOP",
"PMAX",
"MTC",
"PRE",
"YI",
"FVN",
"BEST",
"BYU",
"ZEPP",
"UCL",
"EHGO",
"RCON",
"NAAS",
"HKIT",
"MATH",
"SWIN",
"TWG",
"EBON",
"CDTG",
"ABLV",
"CLPS",
"AGMH",
"FENG",
"SJ",
"BNR",
"KUKE",
"HUIZ",
"RAY",
"AIXI",
"JG",
"ICLK",
"MTEN",
"HUDI",
"IZM",
"UTSI",
"GMM",
"CGA",
"MHUA",
"RETO",
"HAO",
"EPOW",
"CHR",
"CCM",
"MI",
"HLP",
"UCAR",
"PSIG",
"WETH",
"CCTG",
"WLGS",
"NA",
"FEDU",
"UBXG",
"BHAT",
"ATGL",
"HOLO",
"MGIH",
"ILAG",
"ABTS",
"JFU",
"JYD",
"JZ",
"CPOP",
"NCI",
"SEED",
"SUGP",
"PT",
"AACG",
"EDTK",
"MOGU",
"MMV",
"JZXN",
"ZKIN",
"XIN",
"JDZG",
"GSUN",
"CLWT",
"CJET",
"INTJ",
"LOBO",
"GDHG",
"AIHS",
"STEC",
"JL",
"PETZ",
"SOS",
"YJ",
"YQ",
"CJJD",
"GURE",
"JXJT",
"WTO",
"MFI",
"GRFX",
"MEGL",
"NXTT",
"TCTM",
"ROMA",
"HIHO",
"EJH",
"UPC",
"KRKR",
"WAFU",
"CLEU",
"DTSS",
"RAYA",
"BON",
"CREG",
"PWM",
"DDC",
"SNTG",
"LKCO",
"CHNR",
"YGMZ",
"OST",
"TCJH",
"OCG",
"TIRX",
"ANTE",
"IFBD",
"TAOP",
"CPHI",
"CNET",
"BTCT",
"EZGO",
"SISI",
"BAOS",
"KXIN",
"ZCMD",
"ATXG",
"JWEL",
"TC",
"WNW",
"LXEH",
"DUO",
"ITP",
"VSME",
"FAMI",
"SXTC",
"BQ",
"MLGO",
"TANH",
"UK",
"CNEY",
];
const blacklist_icon_svgstr =
'<svg class="_LC-finviz-blacklist-icon" style="position: absolute; left: 80px; top: 0;" width="80" height="36" viewport="0 0 80 36" xmlns="http://www.w3.org/2000/svg"> <g> <title>Layer 1</title> <text stroke-width="4" font-weight="bold" xml:space="preserve" text-anchor="start" font-family="\'Bitter\'" font-size="24" id="svg_1" y="27" x="4.5" fill="#bf0000">黑名单</text> <rect stroke="#bf0000" rx="5" fill-opacity="0" id="svg_4" height="30" width="76" y="3" x="2" stroke-width="4" fill="#000000"/> </g> </svg>';
/**
* Returns an array of elements which are the symbol items in the list.
* These are the elements which contain the symbol name and code.
*
* @return {HTMLCollectionOf<Element>} the symbol items
*/
function getSymbolItems() {
// 列表模式下的选择器: 正则比配: tickerName-
// 图表模式下的选择器,正则比配:symbolNameBox-. @todo 需要优化,图表模式下,tv会自动删除不必要的图表项目,导致全选无法选中所有的图表。暂时没有办法解决
let list_mode_items = findElementsByClassRegex(/tickerName-/);
let chart_mode_items = findElementsByClassRegex(/symbolNameBox-/);
if (checkIsChartMode()) {
return chart_mode_items;
} else {
return list_mode_items;
}
}
function copyToClipboard(text) {
// 将文本复制到剪贴板
navigator.clipboard
.writeText(text)
.then(function () {
console.debug("Text copied to clipboard");
})
.catch(function (err) {
console.error("Failed to copy text to clipboard: ", err);
});
}
/**
* Finds all elements in the document whose class attribute starts with the
* given regex pattern.
* @param {RegExp} classNameRegex the regex pattern to match against the
* element's class attribute
* 注意:这个选择器是根据class属性的值匹配的。如果元素有多个class选择器,但是class属性只有一个哦
* @return {!Array.<!Element>} an array of elements that match the given regex
* pattern
* @see https://developer.mozilla.org/en-US/docs/Web/API/Document/querySelector
* 不知道为什么 class~=name,会找不到
*/
function findElementsByClassRegex(classNameRegex) {
const selector = `*[class*="${classNameRegex.source}"]`;
return document.querySelectorAll(selector);
}
function checkIsChartMode() {
let el = findElementsByClassRegex(/chartsContent/);
return el && el.length > 0;
}
// 显示Toast消息
function showToast(message) {
// 创建一个div元素作为Toast消息容器
var toast = document.createElement("div");
toast.style.position = "fixed";
toast.style.top = "5%";
toast.style.left = "50%";
toast.style.transform = "translate(-50%, -50%)";
toast.style.backgroundColor = "rgba(0, 0, 0, 0.7)";
toast.style.color = "#fff";
toast.style.padding = "10px";
toast.style.borderRadius = "5px";
toast.style.zIndex = "9999";
toast.textContent = message;
document.body.appendChild(toast);
setTimeout(function () {
document.body.removeChild(toast);
}, 2000);
}
function insertStyleNode() {
let style_node_id = "_my_button_style";
// 获取具有ID为myDiv的节点
var el = document.getElementById(style_node_id);
// 从body中删除myDiv节点
if (el) {
document.body.removeChild(el);
}
// 创建一个style元素
var style = document.createElement("style");
style.id = style_node_id;
// 设置style元素的内容为CSS代码
style.innerHTML = `
._LC-button {
border: none;
padding: 3px 5px;
background-color: #007bff4f;
color: #fff;
border-radius: 2px;
cursor: pointer;
font-size: 10px;
}
._LC-button:hover {
background-color: #f0f3fa;
}
._LC-button:active {
background-color: #0033664f;
}
._LC-batch-copy-button{
border-radius: 6px;
font-size: 12px;
min-width: 34px;
--ui-lib-light-button-default-color-bg: #0000;
--ui-lib-light-button-default-color-content: #131722;
--ui-lib-light-button-default-color-border: #e0e3eb;
align-items: center;
background-color: var(--ui-lib-light-button-color-bg, var(--ui-lib-light-button-default-color-bg));
border-color: var(--ui-lib-light-button-color-border, var(--ui-lib-light-button-default-color-border));
border-style: solid;
border-width: 1px;
box-sizing: border-box;
color: var(--ui-lib-light-button-color-content, var(--ui-lib-light-button-default-color-content));
cursor: default;
display: flex;
justify-content: center;
min-width: 36px;
outline: none;
padding: 0;
margin-right: 0;
}
._LC-chart-link,._LC-copy-button,._LC-finviz-add-remove-blacklist-button{
font-family: -apple-system, BlinkMacSystemFont, Trebuchet MS, Roboto, Ubuntu, sans-serif;
font-feature-settings: "tnum" on, "lnum" on;
--ui-lib-typography-line-height: 16px;
line-height: var(--ui-lib-typography-line-height);
--ui-lib-typography-font-size: 12px;
background-color: #f0f3fa;
border:none;
border-radius: 6px;
margin-left:3px;
box-sizing: border-box;
color: #131722;
display: block;
font-size: var(--ui-lib-typography-font-size);
font-style: normal;
font-weight: 600;
max-width: 96px;
min-width: 36px;
overflow: hidden;
padding: 4px 8px;
text-align: center;
text-overflow: ellipsis;
text-transform: uppercase;
white-space: nowrap;
cursor:pointer;
}
`;
// 将style元素添加到body中
document.body.appendChild(style);
}
// 插入“复制全部”按钮到页面
function insertCopyAllButton() {
let buttonWrapper = document.querySelector(
".innerControlContainer-k3vjdDEs"
);
if (!buttonWrapper) {
buttonWrapper = document.getElementById("js-screener-container");
}
if (!buttonWrapper) {
buttonWrapper = document.body;
}
// 获取具有ID为myDiv的节点
var batch_copy_button = document.getElementById(batch_button_id);
// 从body中删除myDiv节点
if (batch_copy_button) {
buttonWrapper.removeChild(batch_copy_button);
}
// 创建一个div元素
batch_copy_button = document.createElement("button");
// 设置div元素的属性和样式
batch_copy_button.id = batch_button_id;
batch_copy_button.classList.add("_LC-button");
batch_copy_button.classList.add("_LC-batch-copy-button");
batch_copy_button.innerHTML = `<svg width="28" height="28" viewBox="0 0 64 64" xmlns="http://www.w3.org/2000/svg" stroke-width="3" stroke="#000000" fill="none"><rect x="11.13" y="17.72" width="33.92" height="36.85" rx="2.5"/><path d="M19.35,14.23V13.09a3.51,3.51,0,0,1,3.33-3.66H49.54a3.51,3.51,0,0,1,3.33,3.66V42.62a3.51,3.51,0,0,1-3.33,3.66H48.39"/></svg>`;
// 将div元素插入到body中
batch_copy_button.title = "批量复制股票代码";
buttonWrapper.appendChild(batch_copy_button);
batch_copy_button.addEventListener("click", function (event) {
const elements = getSymbolItems();
const texts = [];
elements.forEach((el) => {
texts.push(el.innerText);
});
if (texts.length === 0) {
showToast("页面中没有找到代码,请检查脚本选择器");
} else {
const res = texts.join(", ");
copyToClipboard(res);
// 示例用法
if (checkIsChartMode()) {
showToast(
`已复制 ${texts.length} 个代码到剪贴板。图表模式下可能无法全选复制,请切换到列表模式下复制全部。`
);
} else {
showToast(`已复制 ${texts.length} 个代码到剪贴板`);
}
}
});
}
function insertItemButtons() {
let nodes = findElementsByClassRegex(/symbolNameBox-/);
if (!nodes.length) {
console.debug(`${ext_name} not found chart items`);
return;
}
nodes.forEach((e) => {
let target_el = e;
let item_wrapper = e.parentNode;
let href = e.getAttribute("href") || "";
let exchange_symbol = href
.replaceAll(/^\/symbols\//g, "")
.replaceAll(/\/$/g, "")
.replace("-", ":");
let symbol = exchange_symbol.replace(/^\w+\:/, "");
// 添加移除黑名单
let add_remove_blacklist_button = item_wrapper.querySelector(
"._LC-finviz-add-remove-blacklist-button"
);
if (!add_remove_blacklist_button) {
add_remove_blacklist_button = document.createElement("button");
add_remove_blacklist_button.classList.add("_LC-button");
add_remove_blacklist_button.classList.add(
"_LC-finviz-add-remove-blacklist-button"
);
if (blacklist.includes(symbol)) {
add_remove_blacklist_button.innerText = "-BL";
} else {
add_remove_blacklist_button.innerText = "+BL";
}
add_remove_blacklist_button.title = "添加/移除黑名单";
item_wrapper.appendChild(add_remove_blacklist_button);
add_remove_blacklist_button.addEventListener("click", function (event) {
if (blacklist.includes(symbol)) {
let index = blacklist.indexOf(symbol);
if (index !== -1) {
blacklist.splice(index, 1);
}
add_remove_blacklist_button.innerText = "+BL";
} else {
add_remove_blacklist_button.innerText = "-BL";
blacklist.push(symbol);
}
add_or_remove_blacklist_icon(target_el, symbol);
localStorage.setItem("blacklist", JSON.stringify(blacklist));
});
}
// 复制按钮
let copybtn = item_wrapper.querySelector("._LC-copy-button");
if (!copybtn) {
copybtn = document.createElement("button");
copybtn.title = "复制股票代码";
copybtn.innerText = "复制";
copybtn.classList.add("_LC-button");
copybtn.classList.add("_LC-copy-button");
copybtn.addEventListener("click", function (event) {
copyToClipboard(symbol);
showToast(`已复制代码:${symbol}`);
});
item_wrapper.appendChild(copybtn);
}
// 跳转tv图表按钮。
let tv_link = item_wrapper.querySelector("._LC-tv-chart-link");
if (!tv_link) {
let uri_symbol = encodeURIComponent(exchange_symbol);
tv_link = document.createElement("a");
tv_link.classList.add("_LC-button");
tv_link.classList.add("_LC-chart-link");
tv_link.classList.add("_LC-tv-chart-link");
tv_link.href = `https://cn.tradingview.com/chart/700qUKjc/?symbol=${uri_symbol}`;
tv_link.target = "_blank";
tv_link.innerText = "图表";
tv_link.title = "查看tradingview图表";
item_wrapper.appendChild(tv_link);
}
// 跳转finviz图表按钮。
let finviz_link = item_wrapper.querySelector("._LC-finviz-chart-link");
if (!finviz_link) {
finviz_link = document.createElement("a");
finviz_link.classList.add("_LC-button");
finviz_link.classList.add("_LC-chart-link");
finviz_link.classList.add("_LC-finviz-chart-link");
finviz_link.href = `https://finviz.com/quote.ashx?t=${symbol}&p=d`;
finviz_link.target = "_blank";
finviz_link.innerText = "finviz";
finviz_link.title = "查看finviz图表";
item_wrapper.appendChild(finviz_link);
}
// 插入黑名单标志
add_or_remove_blacklist_icon(target_el, symbol);
});
}
// 添加或移除黑名单标志
function add_or_remove_blacklist_icon(target_el, symbol) {
let item_wrapper = target_el.parentNode;
let blacklist_icon = item_wrapper.querySelector(
"._LC-finviz-blacklist-icon"
);
if (blacklist.includes(symbol)) {
if (!blacklist_icon) {
target_el.insertAdjacentHTML("afterend", blacklist_icon_svgstr);
}
} else {
let blacklist_icon = item_wrapper.querySelector(
"._LC-finviz-blacklist-icon"
);
blacklist_icon && blacklist_icon.remove();
}
}
function init_blacklist() {
let blacklist_str = localStorage.getItem("blacklist");
console.debug("初始化blacklist", blacklist);
if (blacklist_str) {
try {
let list = JSON.parse(blacklist_str);
if (Array.isArray(list)) {
blacklist = [...chinese_stocks, ...list];
}
} catch (error) {}
} else {
blacklist = [...chinese_stocks];
}
}
// 初始化blacklist
init_blacklist();
insertStyleNode();
insertCopyAllButton();
let _interval = setInterval(function () {
console.debug(`${ext_name}定时检测item按钮是否创建`);
let nodes = document.querySelectorAll("._LC-chart-link");
// if (nodes && nodes.length > 0) {
// console.debug(`${ext_name}item按钮创建完成`)
// // if (_interval) {
// // clearInterval(_interval)
// // _interval = null
// // }
// } else {
// }
let items = findElementsByClassRegex(/chartContainer/);
if (items && items.length) {
insertItemButtons();
} else {
console.debug(
`${ext_name}没有找到需要添加按钮的item,请检查class属性是否正确: .chartContainer `
);
}
}, 5000);
})();