语言切换快捷键 Ctrl + Shift + S
// ==UserScript== // @name 语言切换快捷键|适配大部分在线翻译网站 // @namespace https://github.com/CandyTek // @version 1.1 // @license MIT // @description 语言切换快捷键 Ctrl + Shift + S // @author CandyTek // @match *://translate.yandex.com/?* // @match *://zh.pons.com/* // @match *://fanyi.so.com/* // @match *://fanyi.xfyun.cn/console/trans/* // @match *://translate.volcengine.com/* // @match *://translation.imtranslator.net/* // @match *://www.iciba.com/translate* // @match *://fanyi.baidu.com/* // @match *://dictionary.cambridge.org/* // @match *://www.baidu.com/s?* // @match *://fanyi.youdao.com/* // @match *://www.deepl.com/* // @match *://translation2.paralink.com/* // @match *://cn.bing.com/search?* // @match *://cn.bing.com/translator?* // @match *://fanyi.sogou.com/text?* // @match *://www.amz123.com/tools-translate/sougou // @match *://fanyi.caiyunapp.com/* // @match *://niutrans.com/trans?* // @match *://translate.alibaba.com/* // @match *://dict.eudic.net/home/translation // @match *://www.fanyi1234.com/lang/* // @icon  // @grant GM_addStyle // @grant GM_registerMenuCommand // @grant GM_getValue // @grant GM_setValue // @run-at document-end // ==/UserScript== // 网站规则列表 const myMatchWebsites = [ { // https://translate.yandex.com/?source_lang=en&target_lang=zh match: ["*://translate.yandex.com/?*"], button:"[aria-label=\"Switch direction\"]", }, { // https://zh.pons.com/%E7%BF%BB%E8%AF%91/ match: ["*://zh.pons.com/*"], button:"[data-e2e=\"switch-language\"]", }, { // https://fanyi.so.com/# match: ["*://fanyi.so.com/*"], button:".exchange", }, { // https://fanyi.xfyun.cn/console/trans/text match: ["*://fanyi.xfyun.cn/console/trans/*"], button:".anticon-swap", }, { // https://translate.volcengine.com/ match: ["*://translate.volcengine.com/*"], button:".reverse-img", }, { // https://translation.imtranslator.net/ match: ["*://translation.imtranslator.net/*"], button:"[title=\"Change translation direction of a selected language pair\"]", }, { // https://www.iciba.com/translate match: ["*://www.iciba.com/translate*"], button:".select_exchange__AfzH4", }, { // fanyi.baidu.com match: ["*://fanyi.baidu.com/*"], button:"svg.mg3bUrpQ", }, { // https://dictionary.cambridge.org/zhs/%E8%AF%8D%E5%85%B8/%E8%8B%B1%E8%AF%AD-%E6%B1%89%E8%AF%AD-%E7%B9%81%E4%BD%93/exchange match: ["*://dictionary.cambridge.org/*"], button:".i-exchange", }, { // https://www.baidu.com/s?tn=68018901_3_dg&ie=UTF-8&wd=%E7%BF%BB%E8%AF%91%E7%BD%91%E7%AB%99 match: ["*://www.baidu.com/s?*"], button:".op_translation_exchange_img", }, { // https://fanyi.youdao.com/#/ match: ["*://fanyi.youdao.com/*"], button:".ic_language_exchange", }, { // https://fanyi.sogou.com/text?keyword=&transfrom=en&transto=zh-CHS&model=general&exchange=true match: ["*://fanyi.sogou.com/*"], button:".btn-switch", }, { // https://www.deepl.com/zh/translator match: ["*://www.deepl.com/*"], button:"[data-testid=\"lmt_language_switch\"]", }, { // https://translation2.paralink.com/ match: ["*://translation2.paralink.com/*"], button:"#switch", }, { // https://cn.bing.com/search?q=%E4%BD%A0%E5%A5%BD%20%E8%8B%B1%E8%AF%AD%E7%BF%BB%E8%AF%91 match: ["*://cn.bing.com/search?*","*://cn.bing.com/translator?*"], button:"#tta_revIcon", }, // { // // https://www.amz123.com/tools-translate/sougou // match: ["*://www.amz123.com/tools-translate/sougou"], // button:".btn-switch", // iframe:"#translate", // }, { // https://fanyi.caiyunapp.com/ match: ["*://fanyi.caiyunapp.com/*"], button:".changeImg", }, { // https://niutrans.com/trans?type=text match: ["*://niutrans.com/trans?*"], button:".nt-icon-qiehuan", }, { // https://translate.alibaba.com/ match: ["*://translate.alibaba.com/*"], button:".anticon-swap", }, { // https://dict.eudic.net/home/translation match: ["*://dict.eudic.net/home/translation"], button:".switchBtn", }, { // https://www.fanyi1234.com/lang/ match: ["*://www.fanyi1234.com/lang/*"], button:"[alt=\"转换\"]", }, ]; /** 设置工具类 */ class CandyTekPreferenceUtil { /** 是否已向网页添加过设置面板了 */ isAlreadyAddSettingPanel = false; /** 设置面板根元素 */ rootShadow = null; /** 存放设置值的地方。获取 prefValues[key] */ prefValues; /** 源 pref 配置数组 */ preferenceList; constructor(preferenceList) { this.preferenceList = preferenceList; this.refreshPrefValues(); } /** 刷新设置值 */ refreshPrefValues() { this.prefValues = this.preferenceList.reduce((list, curr) => { list[curr.preference] = GM_getValue(curr.preference, curr.defaultValue); return list; }, {}); } /** 获取设置值 */ get(key) { return this.prefValues.hasOwnProperty(key) ? this.prefValues[key] : GM_getValue(key, ""); } /** 写入设置值,未适配 boolean */ set(key, value) { GM_setValue(key, value); this.prefValues[key] = value; } /** 显示设置面板在网页右上角 */ show() { if (this.isAlreadyAddSettingPanel) { this.rootShadow.querySelector(".setting_panel").style.display = "block"; return; } if (!document.body.createShadowRoot) { console.warn("可能不能创建 ShadowRoot"); //return; } // 创建设置面板 const host = document.createElement('div'); host.id = "simplify_article_settings_panel"; document.body.appendChild(host); const root = host.attachShadow({ mode: 'open' }); this.rootShadow = root; this.isAlreadyAddSettingPanel = true; root.innerHTML = ` <style> .preference_title { width: fit-content; height: 40px; font-size: 20px; margin: 0px; line-height: 40px; padding-left: 16px; font-weight: bold; } .preference_item { display: flex; padding: 12px 8px; } .preference_item_title { padding: 0px 0px 0px 10px; margin: 0px; font-size: 15px; line-height: 40px; letter-spacing: 2px; height: 40px; width: 140px; } .preference_item_edittext { font-size: 14px; margin-left: auto; line-height: 36px; height: 36px; padding: 0px; border: 2px solid #c4c7ce; border-radius: 6px; text-align: center; width: 138px; } .preference_item_textarea { text-align: unset; line-height: 20px; } .preference_item_edittext_color { width: 100px; border-radius: 6px 0px 0px 6px; border-right: 0; } .hoverbutton { background: none; } .hoverbutton:hover { background: #CCC; background-size: 80% 80%; border-radius: 4px; } .input_select_color { width: 40px; height: 40px; margin: 0px; padding:0px 2px 0px 4px; box-sizing: border-box; background-color:#ffffff; border-width: 2px; border-radius: 0px 6px 6px 0px; border-left: 0px; border-color: #c4c7ce; } .checkbox_input { width: 24px; height: 40px; margin: 0px 0px 0px auto; } .setting_panel { position: fixed; right: 20px; top: 20px; width: fit-content; height: fit-content; border-radius: 8px; background: #FFFFFF; padding: 8px; box-shadow: 0 10px 20px rgb(0 0 0 / 15%); z-index:9999; } .container { background: #F0F0F0; border-radius: 8px; margin-top: 0px; padding-top: 8px; padding-right: 8px; } </style> <div class="setting_panel"> <div class="preference_item" style="padding-top: 0px;"> <button id="close" title="关闭并保存" class="hoverbutton" type="submit" style="width: 40px;height: 40px;display: flex;align-items: center; justify-content: center; border: unset;"> <svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" fill="#5f6368" viewBox="0 -960 960 960"> <path d="m256-200-56-56 224-224-224-224 56-56 224 224 224-224 56 56-224 224 224 224-56 56-224-224-224 224Z" /> </svg> </button> <p class="preference_title">设置</p> </div> <div class="container" id="container"> </div> </div> `; const container = root.querySelector("#container"); // 动态创建设置项 for (const index in this.preferenceList) { const item = this.preferenceList[index]; const itemDiv = document.createElement("div"); itemDiv.className = "preference_item"; const itemTitle = document.createElement("p"); itemTitle.className = "preference_item_title"; itemTitle.innerText = item.text; itemDiv.appendChild(itemTitle); if (item.type == "number") { const input = document.createElement("input"); input.type = "number"; input.className = "preference_item_edittext"; input.id = item.preference; input.value = GM_getValue(item.preference, item.defaultValue); itemDiv.appendChild(input); } else if (item.type == "color") { const inputText = document.createElement("input"); inputText.type = "text"; inputText.className = "preference_item_edittext preference_item_edittext_color"; inputText.id = item.preference; inputText.value = GM_getValue(item.preference, item.defaultValue); inputText.maxLength = 50; itemDiv.appendChild(inputText); const inputColor = document.createElement("input"); inputColor.type = "color"; inputColor.className = "input_select_color"; if (this.isValidHexColor(inputText.value)) { inputColor.value = inputText.value; } itemDiv.appendChild(inputColor); inputText.addEventListener('input', () => this.inputTextAndChangeDisplayColor(inputText, inputColor)); inputColor.addEventListener('input', () => this.selectColorAndChangeText(inputText, inputColor)); } else if (item.type == "checkbox") { const input = document.createElement("input"); input.type = "checkbox"; input.id = item.preference; const checkValue = GM_getValue(item.preference, item.defaultValue); input.checked = checkValue; input.className = "checkbox_input"; itemDiv.appendChild(input); } else if (item.type == "textarea") { const input = document.createElement("textarea"); input.id = item.preference; input.value = GM_getValue(item.preference, item.defaultValue); input.className = "preference_item_edittext preference_item_textarea"; itemDiv.appendChild(input); } container.appendChild(itemDiv); } root.querySelector("#close").onclick = () => { root.querySelector(".setting_panel").style.display = "none"; // 动态创建设置项 for (const index in this.preferenceList) { const item = this.preferenceList[index]; if (item.type == "color" || item.type == "textarea") { try { GM_setValue(item.preference, root.querySelector(`#${item.preference}`).value); } catch (error) { console.error(`保存配置失败:${item.preference}`); } } else if (item.type == "number") { try { GM_setValue(item.preference, parseFloat(root.querySelector(`#${item.preference}`).value)); } catch (error) { console.error(`保存配置失败:${item.preference}`); } } else if (item.type == "checkbox") { try { GM_setValue(item.preference, root.querySelector(`#${item.preference}`).checked); } catch (error) { console.error(`保存配置失败:${item.preference}`); } } } this.refreshPrefValues(); }; } /** input 颜色选择器更改颜色时,同时更改文本框 */ selectColorAndChangeText(inputText, inputColor) { inputText.value = inputColor.value; }; /** 文本框更改值时,同时更改颜色显示 */ inputTextAndChangeDisplayColor(inputText, inputColor) { const color = inputText.value; if (this.isValidHexColor(color)) { inputColor.value = color; } }; /** 用于校验 6 位的十六进制颜色值 */ isValidHexColor(hex) { try { const hexPattern = /^#?([a-fA-F0-9]{6})$/; return hexPattern.test(hex); } catch (error) { return false; } } } (() => { let p; // 开始匹配网站 for (const website of myMatchWebsites) { let hit = false; hit = Array.isArray(website.match) ? website.match.some((s) => matchRule(window.location.href, s)) : matchRule(window.location.href, website.match); if (hit) { //p = new CandyTekPreferenceUtil(myPreferenceList); // 添加设置菜单 // GM_registerMenuCommand("快捷键设置", () => { // p.show(); // }); // 添加语音调转快捷键 document.addEventListener("keydown", function(event) { if (event.ctrlKey && event.shiftKey && !event.altKey && event.key === "S") { let el = document.querySelector(website.button); if(website.iframe){ const iframe = document.querySelector(website.iframe); const iframeDocument = iframe.contentDocument || iframe.contentWindow.document; el=iframeDocument.querySelector(website.button); } customClick(el); if(website.secondButton){ let el2 = document.querySelector(website.secondButton); customClick(el2); } } }); console.info(`匹配成功 ${website.match}`); break; } }; function customClick(el){ if(el){ try{ el.click(); }catch{ el.dispatchEvent(new MouseEvent('click', {bubbles: true,cancelable: true,})); } } } /** match匹配方法 */ function matchRule(str, rule) { const escapeRegex = (str) => str.replace(/([.*+?^=!:${}()|\[\]\/\\])/g, "\\$1"); return new RegExp(`^${rule.split("*").map(escapeRegex).join(".*")}$`).test(str); } // todo:https://www.reverso.net/text-translation // todo:google // node:http://www.sowang.com/sogou/fanyi.htm // todo:https://www.amz123.com/tools-translate/tenxunjiaohu // todo:https://fanyi.pdf365.cn/free // todo:https://www.fanyi1234.com/ // todo:https://dict.cnki.net/ // todo:https://www.ichacha.net/ // todo:https://www.tangpafanyi.com/text.html // todo:https://tran.httpcn.com/FanyiWeb/ // todo:https://fanyi.zou.la/ // todo:https://fanyi.dict.cn/ // todo:https://www.99yee.cn/ // todo:https://transmart.qq.com/zh-CN/index // todo:https://fanyi.atman360.com/index // todo:https://fanyi.qq.com/ // todo:https://www.medsci.cn/sci/translation.do // todo:https://www.worldlingo.com/en/products/text_translator.html })();