- // ==UserScript==
- // @name Smart Translator Tool
- // @name:ar أداة الترجمة |
- // @name:bg Превод на джаджа |
- // @name:cs Překlad Widget |
- // @name:da Oversættelseswidget |
- // @name:de Übersetzungs -Widget |
- // @name:el Μεταφραστικό widget | Μεταφράστε οποιαδήποτε γλώσσα
- // @name:en Translation widget | Translate any language
- // @name:eo Traduko -fenestraĵo |
- // @name:es Widget de traducción |
- // @name:fi Käännöswidget | Käännä kaikki kielet
- // @name:fr Widget de traduction | Traduire n’importe quelle langue
- // @name:fr-CA Widget de traduction | Traduire n’importe quelle langue
- // @name:he יישומון תרגום |
- // @name:hr Prijevod Widget |
- // @name:hu A Widget fordítása |
- // @name:id Widget Terjemahan | Menerjemahkan bahasa apa pun
- // @name:it Widget di traduzione |
- // @name:ja 翻訳ウィジェット|
- // @name:ka თარგმანის ვიჯეტი | თარგმნეთ ნებისმიერი ენა
- // @name:ko 번역 위젯 |. 모든 언어 번역
- // @name:nb Oversettelseswidget |
- // @name:nl Vertaalwidget |
- // @name:pl Widżet tłumaczenia |
- // @name:pt-BR Widget de tradução |
- // @name:ro Widget de traducere |
- // @name:ru Перевод виджет |
- // @name:sk Preklad miniaplikácie
- // @name:sr Превод Видгет | Преведи било који језик
- // @name:sv Översätt widget | Översätt något språk
- // @name:th วิดเจ็ตแปล
- // @name:tr Çeviri widget’ı |
- // @name:ug تەرجىمە كىچىك قورالى | ھەر قانداق تىلنى تەرجىمە قىلىڭ
- // @name:uk Віджет перекладу | Перекладіть будь -яку мову
- // @name:vi Tiện ích dịch |
- // @name:zh 翻译小工具 | 翻译任何语言
- // @name:zh-CN 翻译小工具 | 翻译任何语言
- // @name:zh-HK 翻譯小工具 | 翻譯任何語言
- // @name:zh-SG 翻译小工具 | 翻译任何语言
- // @name:zh-TW 翻譯小工具 | 翻譯任何語言
- // @description Translate any language and quickly translate the desired language with one key | Page translation | Select text (Press Ctrl) | Provides shortcuts, allowing quick translation of the desired language with just one key | English learning | Translation settings available, supporting most common languages worldwide | Feel free to provide feedback for any issues
- // @description:ar ترجمة أي لغة واختصار للترجمة |
- // @description:bg Превод на всеки език и пряк път с едно щракване превод на езика, който искате | Превод на страница | Изберете текст (натиснете Ctrl) | Осигурете преки пътища, просто натиснете един клавиш, за да преведете бързо желания език | Английско обучение | Преводът може да бъде зададен и поддържа най -често срещаните езици по света | Обратна връзка, ако имате въпроси
- // @description:cs Překlad jazyka a překladu jazyka, který chcete, překlad |
- // @description:da Oversættelse af ethvert sprog og genvej med et-klik på det sprog, du ønsker |
- // @description:de Übersetzung von Sprach- und Verknüpfungsverknüpfungen Ein-Klick-Übersetzung der gewünschten Sprache | Seitenübersetzung | Text auswählen (drücken Sie Strg) | Stellen Sie Verknüpfungen an, drücken Sie einfach eine Taste, um schnell die gewünschte Sprache zu übersetzen | Englisches Lernen | Die Übersetzung kann festgelegt werden und unterstützt die häufigsten Sprachen auf der ganzen Welt | Feedback, wenn Sie Fragen haben
- // @description:el Μετάφραση οποιασδήποτε γλώσσας και συντόμευσης με ένα κλικ της γλώσσας που θέλετε |
- // @description:en Translation of any language and shortcut one-click translation of the language you want | Page Translation | Select text (press Ctrl) | Provide shortcuts, just press one key to quickly translate the language you want | English learning | Translation can be set and supports most common languages around the world | Feedback if you have any questions
- // @description:eo Traduko de iu ajn lingvo kaj ŝparvojo unu-klaka traduko de la lingvo, kiun vi volas
- // @description:es Traducción de cualquier idioma y atajo Traducción del idioma que desee |
- // @description:fi Minkä tahansa kielen käännös ja pikakuvake yhden napsautuksen käännös haluamastasi kielestä | Sivun käännös | Valitse teksti (paina Ctrl) | Tarjoa pikakuvakkeita, painamalla vain yhtä näppäintä kääntääksesi haluamasi kielen nopeasti | Englannin oppiminen | Käännös voidaan asettaa ja tukee yleisimpiä kieliä ympäri maailmaa | Palautetta, jos sinulla on kysyttävää
- // @description:fr Traduction de toute langue et de la traduction en un seul clic de la langue que vous voulez |
- // @description:fr-CA Traduction de toute langue et de la traduction en un seul clic de la langue que vous voulez |
- // @description:he תרגום של כל שפה וקיצור של לחיצה אחת על השפה שאתה רוצה |
- // @description:hr Prijevod i prečac prijevoda jezika koji želite prijevod |
- // @description:hu Bármely nyelv és a parancsikon a kívánt nyelv fordítása |
- // @description:id Terjemahan dari bahasa apa pun yang Anda inginkan |
- // @description:it Traduzione di qualsiasi linguaggio e premi per la traduzione della lingua |
- // @description:ja 任意のショートカットのページ翻訳|
- // @description:ka ნებისმიერი ენის თარგმანი და მალსახმობი ერთ-
- // @description:ko 모든 언어의 번역은 원하는 텍스트를 선택하십시오
- // @description:nb Oversettelse av noe språk og snarvei enklikk på oversettelsen av språket du vil ha | Sideoversettelse |.
- // @description:nl Vertaling van elke taal- en snelkoppelingsklikvertaling van de taal die u wilt | Pagina-vertaling | Selecteer tekst (druk op CTRL) |
- // @description:pl Tłumaczenie dowolnego języka i skrót tłumaczenie jednego kliknięcia języka, który chcesz | Tłumaczenie strony | Wybierz tekst (naciśnij Ctrl) | Podaj skróty, wystarczy nacisnąć jeden klawisz, aby szybko przetłumaczyć żądany język | Uczenie się angielskiego | Tłumaczenie może być ustawione i obsługuje najczęstsze języki na całym świecie | Informacje zwrotne, jeśli masz jakieś pytania
- // @description:pt-BR Tradução de qualquer idioma e atalho com um clique de tradução do idioma que você deseja | Tradução da página | Selecione Texto (pressione Ctrl) | Forneça atalhos, basta pressionar uma tecla para traduzir rapidamente o idioma que deseja | Aprendizagem de inglês | A tradução pode ser definida e suporta idiomas mais comuns em todo o mundo | Feedback se você tiver alguma dúvida
- // @description:ro Traducere a oricărei limbi și scurte de traducere cu un singur clic pe un singur clip
- // @description:ru Перевод любого языка и ярлык.
- // @description:sk Preklad akéhokoľvek jazyka a skratkou prekladu, ktorý chcete, preklad s jedným kliknutím |
- // @description:sr Превод било којег језика и пречице Превод једног клика Један превод језика | Изаберите текст (Притисните ЦТРЛ) | Обезбедите пречице, само притисните један тастер да бисте брзо превели језик у којем се можете подесити и подржати најчешћи језици широм света | ако имате било каква питања.
- // @description:sv Översättning av alla språk och genvägar en klick översättning av det språk du vill ha |
- // @description:th การแปลภาษาใด ๆ และการแปลภาษาหนึ่งคลิกของภาษาที่คุณต้องการ
- // @description:tr Herhangi bir dil ve kısayol, metin seçin |
- // @description:ug سىز تاللىغان تىل ۋە «قىسقا ئۇچۇرنى تاللاش تەرجىمىسى | قىسقا ئۇچۇر بىلەن تەمىنلەڭ | پەقەت بىر ئاچقۇچنى باسسىڭىز بىر ئاچقۇچنى باسسىڭىز بولىدۇ | ئىنگلىزچە ئۆگىنىشنى | سوئاللىرىڭىز بولسا تەكلىپ-پىكىرلەرنى تىزىڭ ۋە قوللايدۇ
- // @description:uk Переклад будь-якої мови та ярлика клацання перекладу мови, яку ви хочете | Переклад сторінки | Виберіть текст (натисніть Ctrl) | Надайте ярлики, просто натисніть одну клавішу, щоб швидко перекласти потрібну мову | Англійське навчання | Переклад може бути встановлений та підтримує найпоширеніші мови у всьому світі | Відгук Якщо у вас є якісь питання
- // @description:vi Bản dịch của bất kỳ ngôn ngữ nào của Ngôn ngữ bạn muốn |
- // @description:zh 任何语言翻译及快捷一键翻译想要语言 | 页面翻译 | 选中文字( 按Ctrl )| 提供快捷方式,只需按一个键即可快速翻译想要语言 |英文学习 | 翻译文可设置,支持全球多数通用语言 | 有什么问题都可以反馈
- // @description:zh-CN 任何语言翻译及快捷一键翻译想要语言 | 页面翻译 | 选中文字( 按Ctrl )| 提供快捷方式,只需按一个键即可快速翻译想要语言 |英文学习 | 翻译文可设置,支持全球多数通用语言 | 有什么问题都可以反馈
- // @description:zh-HK 任何語言翻譯及快捷一鍵翻譯想要語言 | 頁面翻譯 | 選中文字( 按Ctrl )| 提供快捷方式,只需按一個鍵即可快速翻譯想要語言 |英文學習 | 翻譯文可設置,支持全球多數通用語言 | 有什麼問題都可以反饋
- // @description:zh-SG 任何语言翻译及快捷一键翻译想要语言 | 页面翻译 | 选中文字( 按Ctrl )| 提供快捷方式,只需按一个键即可快速翻译想要语言 |英文学习 | 翻译文可设置,支持全球多数通用语言 | 有什么问题都可以反馈
- // @description:zh-TW 任何語言翻譯及快捷一鍵翻譯想要語言 | 頁面翻譯 | 選中文字( 按Ctrl )| 提供快捷方式,只需按一個鍵即可快速翻譯想要語言 |英文學習 | 翻譯文可設置,支持全球多數通用語言 | 有什麼問題都可以反饋
- // @author shing0727@foxmail.com,人民的勤务员 <china.qinwuyuan@gmail.com>
- // @namespace https://github.com/ChinaGodMan/UserScripts
- // @supportURL https://github.com/ChinaGodMan/UserScripts/issues
- // @homepageURL https://github.com/ChinaGodMan/UserScripts
- // @license MIT
- // @match *://*/*
- // @icon https://s21.ax1x.com/2024/05/17/pkuVzUH.png
- // @require https://code.jquery.com/jquery-3.6.0.min.js
- // @require https://cdnjs.cloudflare.com/ajax/libs/crypto-js/4.1.1/crypto-js.min.js
- // @compatible chrome
- // @compatible firefox
- // @compatible edge
- // @compatible opera
- // @compatible safari
- // @compatible kiwi
- // @compatible qq
- // @compatible via
- // @compatible brave
- // @version 2025.03.21.1227
- // @grant GM_xmlhttpRequest
- // @created 2025-03-21 12:27:54
- // @modified 2025-03-21 12:27:54
- // ==/UserScript==
-
- // 百度logo
- var baiduImgData = ``
-
- // 谷歌logo
- var googleImgData = ``
- var cssContent = `
- .fy_btn_box{
- position: fixed;
- top: 50px;
- right: 50px;
- z-index: 9999;
- background-color: rgba(255, 255, 255, 0.5);
- border-radius: 10px;
- padding: 10px;
- box-shadow: 0 0 10px rgba(0, 0, 0, 0.5);
- color: #333;
- }
- #fy_transContainer{
- display: none;
- position: fixed;
- top: 50px;
- left: 50px;
- max-width: 500px;
- min-width: 300px;
- padding: 10px;
- padding-top: 24px;
- border-radius: 4px;
- flex-direction: column;
- justify-content: center;
- align-items: flex-start;
- box-shadow: 0 3px 10px 1px rgba(0, 0, 0, 0.1);
- background-color: rgba(255,255,255,0.7);
- backdrop-filter: saturate(420%) blur(50px);
- -webkit-backdrop-filter: saturate(420%) blur(50px);
- z-index: 9999;
- font-size: 14px;
- border-bottom-right-radius: 0;
- overflow: hidden;
- }
- #fy_dragBar{
- position: absolute;
- top: 0;
- left: 0;
- width: 100%;
- height: 20px;
- cursor: grab;
- // background-color: #F9E79F;
- background-color: rgba(0,0,0,0.1);
- backdrop-filter: saturate(420%) blur(50px);
- -webkit-backdrop-filter: saturate(420%) blur(50px);
- }
- #fy_Scale_rb, #fy_Scale_lb{
- position:absolute;
- bottom:0;
- width:9px;
- height:9px;
- // background: #ddd;
- // clip-path: polygon(100% 0%, 100% 100%, 0% 100%);
- }
- #fy_Scale_rb{
- right:0;
- cursor: nw-resize;
- }
- #fy_Scale_lb{
- left:0;
- cursor: ne-resize;
- }
-
- #fy_contentBox{
- width: 100%;
- overflow: auto;
- line-height: 1.3em;
- letter-spacing: 0.5px;
- }
- #fy_contentBox .textRight{
- text-align: right;
- }
- .transText_node{
- width: 100%;
- padding: 7px;
- box-sizing: border-box;
- }
- .transText_node:hover{
- background-color: rgba(0,0,0,0.04);
- border-radius: 6px;
- }
- .transText_node_to{
- position: relative;
- transition: all 0.2s;
- }
- .transText_node_from{
- position: relative;
- height: 0;
- overflow: hidden;
- transition: all 0.2s;
- }
-
- #fy_contentBox .fy_node_expand{
- background-color: rgba(0,0,0,0.04);
- border-radius: 6px;
- margin: 5px 0;
- }
-
- .fy_node_expand .transText_node_from, .fy_node_expand .transText_node_to{
- padding: 6px 8px;
- }
- .fy_node_expand .transText_node_to{
- background-color: rgb(209, 255, 240);
- border-top-left-radius: 4px;
- border-top-right-radius: 4px;
- }
- .fy_node_expand .transText_node_from{
- background-color: rgb(254, 234, 242);
- height: auto;
- border-bottom-left-radius: 4px;
- border-bottom-right-radius: 4px;
- }
-
- .icon_style{
- position: absolute;
- right: 0;
- bottom: 0;
- cursor: pointer;
- opacity: 0;
- transition: all 0.3s ease;
- width: 20px;
- z-index: 9999;
- opacity: 0.5;
- }
- .transText_node_from:hover .copy_icon{
- opacity: 1;
- }
- .transText_node_to:hover .copy_icon{
- opacity: 1;
- }
- .fy_node_expand .copy_icon{
- right: 5px;
- bottom: 5px;
- }
-
- #fy_ctrl_ber{
- position: absolute;
- top: 2px;
- right: 0;
- }
- #fy_select, #fy_api_select{
- position: relative;
- background-color: transparent;
- border: none; /* 去除默认边框 */
- box-shadow: none; /* 去除默认的阴影 */
- outline: none; /* 去除可能的轮廓线 */
- font: inherit;
- line-height: 1.2em;
- height: auto;
- text-align: center;
- width: auto;
- min-width: auto;
- min-height: auto;
- }
- #fy_api_select{
- }
-
- #fy_loading{
- display: none;
- width: 100%;
- padding-top: 10px;
- display: flex;
- align-items: center;
- justify-content: center;
- }
- #fy_loading svg {
- width: 2.25em;
- transform-origin: center;
- animation: rotate4 2s linear infinite;
- }
-
- #fy_loading circle {
- fill: none;
- stroke: hsl(214, 97%, 59%);
- stroke-width: 2;
- stroke-dasharray: 1, 200;
- stroke-dashoffset: 0;
- stroke-linecap: round;
- animation: dash4 1.5s ease-in-out infinite;
- }
-
- @keyframes rotate4 {
- 100% {
- transform: rotate(360deg);
- }
- }
-
- @keyframes dash4 {
- 0% {
- stroke-dasharray: 1, 200;
- stroke-dashoffset: 0;
- }
-
- 50% {
- stroke-dasharray: 90, 200;
- stroke-dashoffset: -35px;
- }
-
- 100% {
- stroke-dashoffset: -125px;
- }
- }
-
- #fy_entry_container {
- position: fixed;
- background-color: aqua;
- top: 500px;
- right: 0;
- border-top-left-radius: 16px;
- border-bottom-left-radius: 16px;
- width: 35px;
- height: 34px;
- display: flex;
- align-items: center;
- padding-left: 12px;
- box-sizing: border-box;
- transition: width 0.2s;
- background-color: #1890ff;
- opacity: 0.5;
- cursor: pointer;
-
- -webkit-user-select: none;
- -moz-user-select: none;
- -ms-user-select: none;
- user-select: none;
-
- display:none;
- }
-
- #fy_entry_container:hover {
- width: 55px;
- opacity: 1;
- }
-
- #fy_entry_container img {
- width: 20px;
- height: 20px;
- pointer-events: none;
- }
-
- #fy_translate_tools_container {
- position: fixed;
- top: 45vh;
- left: 100%;
- display: flex;
- flex-wrap: wrap;
- z-index: 9980;
- width: 300px;
- box-shadow: 0 0 5px 1px rgba(0, 0, 0, 0.1);
- background-color: #f2f4fb;
- padding: 6px 8px 10px;
- border-radius: 4px;
- color: #333;
- cursor: grab;
- transition: left 0.5s;
- }
-
- .fy_tools_top,
- .fy_tools_main,
- .fy_tools_footer {
- position: relative;
- width: 100%;
- }
-
- .fy_tools_top {
- display: flex;
- justify-content: space-between;
- align-items: center;
- padding-bottom: 6px;
- }
-
- .fy_tools_top .fy_title {
- font-size: 13px;
- font-weight: 600;
- letter-spacing: 1px;
- }
-
- .fy_tools_ctrl_box {
- display: flex;
- align-items: center;
- }
-
- #fy_tools_close {
- font-size: 10px;
- padding: 4px;
- -webkit-user-select: none;
- -moz-user-select: none;
- -ms-user-select: none;
- user-select: none;
- filter: grayscale(100%);
- }
-
- #fy_tools_close:hover {
- filter: grayscale(0%);
- cursor: pointer;
- }
-
- .fy_tools_select_container {
- display: flex;
- align-items: center;
- font-size: 13px;
- margin-right: 5px;
- border-radius: 4px;
- }
-
- .fy_tools_select_container img {
- display: inline-block;
- margin: auto 2px;
- width: 16px;
-
- -webkit-user-drag: none;
- -khtml-user-drag: none;
- -moz-user-drag: none;
- -o-user-drag: none;
- user-drag: none;
- }
-
- .fy_tools_select_from {
- position: relative;
- letter-spacing: 0;
- background-color: rgba(169, 173, 204, .1);
- height: 20px;
- line-height: 20px;
- padding: 0 6px;
- border-radius: 4px;
- }
-
- .fy_tools_select_to {
- position: relative;
- }
-
- #fy_tools_selected_active {
- background-color: #ffffff;
- font-size: 12px;
- width: 30px;
- height: 20px;
- line-height: 20px;
- border-radius: 4px;
- text-align: center;
- cursor: pointer;
- }
-
- .fy_tools_ul {
- list-style-type: none;
- margin: 0;
- padding: 4px 0;
- position: absolute;
- top: 0;
- left: 50%;
- transform: translate(-50%, -110%);
- color: #333333;
- z-index: 9990;
- text-align: center;
- background-color: #fff;
- box-shadow: 0 0 5px 1px rgba(0, 0, 0, 0.1);
- font-size: 12px;
- border-radius: 4px;
- width: 300px;
- justify-content: center;
- flex-wrap: wrap;
- display: none;
- overflow: hidden;
- }
-
- .fy_tools_ul li {
- -webkit-user-select: none;
- -moz-user-select: none;
- -ms-user-select: none;
- user-select: none;
- line-height: 20px;
- cursor: pointer;
- width: 40px;
- }
- .fy_tools_ul li:hover {
- background-color: rgba(169, 173, 204, .3);
- }
-
- .fy_translate_to{
- position: relative;
- overflow: auto;
- }
-
- .fy_tools_text_content {
- box-sizing: border-box;
- max-width: 100%;
- width: 100%;
- min-height: 100px;
- padding: 6px 8px;
- font-size: 14px;
- border: none;
- outline: none;
- overflow: auto;
- border-radius: 8px;
- margin: 0;
- }
-
- #fy_translate_result {
- margin-top: 5px;
- width: 100%;
- background-color: rgba(169, 173, 204, .1);
- display: none;
- cursor: auto;
- }
- .tools_ctrl_box{
- position: absolute;
- bottom: 0;
- right: 0;
- padding: 5px;
- display: flex;
- z-index: 9999;
- }
- .tools_ctrl_box img{
- width: 20px;
- transition: all 0.3s ease;
- margin-left: 5px;
- cursor: pointer;
- opacity: 0.7;
- }
- .tools_ctrl_box img:hover{
- opacity: 1;
- }
-
- #fy_tools_api_selected{
- border-radius: 4px;
- text-align: center;
- cursor: pointer;
- display: flex;
- justify-content: center;
- items-align: center;
- margin-right: 5px;
- }
- #fy_tools_api_selected img{
- display: block;
- }
- #fy_tools_api_ul img{
- width: 17px;
- margin: 0;
- }
- #fy_tools_api_active img{
- width: 17px;
- }
- #fy_tools_api_selected .fy_tools_ul{
- width: auto;
- padding: 0;
- }
- #fy_tools_api_ul{
- left: auto;
- transform: translate(0%, -100%);
- }
- #fy_tools_api_ul li{
- width: auto;
- padding: 5px 7px;
- }
-
- /* 滚动条整体样式 */
- ::-webkit-scrollbar {
- width: 6px; /* 宽度 */
- height: 6px; /* 高度(对于垂直滚动条) */
- }
-
- /* 滚动条滑块 */
- ::-webkit-scrollbar-thumb {
- background: #aaa;
- border-radius: 6px;
- }
-
- /* 滚动条滑块:hover状态样式 */
- ::-webkit-scrollbar-thumb:hover {
- background: #888;
- }
-
- /* 滚动条轨道 */
- ::-webkit-scrollbar-track {
- background: #f1f1f1;
- border-radius: 6px;
- }
-
- /* 滚动条轨道:hover状态样式 */
- ::-webkit-scrollbar-track:hover {
- background: #ddd;
- }
-
- /* 滚动条轨道:active状态样式 */
- ::-webkit-scrollbar-track-piece:active {
- background: #eee;
- }
-
- /* 滚动条:角落样式(即两个滚动条交汇处) */
- ::-webkit-scrollbar-corner {
- background: #535353;
- }
-
- `
-
- // ------------全局----------------
- // 显示总容器
- var transContainerDOM = null
- // 翻译内容容器
- var fyContentDOM = null
- // 拖动条
- var fyDragBarDOM = null
- // 待翻译文本
- var fromTransTextArray = []
- // 翻译结果对象
- var transRes = {}
- // 时间戳
- var salt = Date.now()
- // 当前翻译对象数组
- var currTransToObjs = []
- // --------------------------
- // 语言选择类型
- var selectTypes = [
- { type: 'zh', valueName: '中', name: '中文' },
- { type: 'en', valueName: '英', name: '英文' },
- { type: 'jp', valueName: '日', name: '日文' },
- { type: 'kor', valueName: '韩', name: '韩文' },
- { type: 'fra', valueName: '法', name: '法文' },
- { type: 'spa', valueName: '西', name: '西班牙文' },
- { type: 'ru', valueName: '俄', name: '俄文' },
- { type: 'de', valueName: '德', name: '德文' },
- { type: 'it', valueName: '意', name: '意大利文' },
- { type: 'th', valueName: '泰', name: '泰文' },
- { type: 'vie', valueName: '越', name: '越南文' },
- { type: 'pt', valueName: '葡', name: '葡萄牙文' },
- { type: 'ara', valueName: '阿', name: '阿拉伯文' },
- { type: 'cht', valueName: '中(繁)', name: '中文繁体' },
- { type: 'yue', valueName: '中(粤)', name: '粤语' }
- ]
- // 翻译api列表
- var apiMap = [
- { name: 'Baidu', logo: baiduImgData, descript: '百度翻译' },
- { name: 'Google', logo: googleImgData, descript: '谷歌翻译(需要外网)' }
- ]
- // 默认百度翻译
- var currFromApi = 'Baidu'
- // 默认翻译语言
- var fyToType = '中'
-
- // 初始加载样式
- const loadStyle = () => {
- var style = document.createElement('style')
- if (style.styleSheet) {
- // 对于老版本的IE浏览器
- style.styleSheet.cssText = cssContent
- } else {
- style.appendChild(document.createTextNode(cssContent))
- }
- var head = document.head || document.getElementsByTagName('head')[0]
- head.appendChild(style)
- }
- /**
- * 传入配置信息创建元素并返回DOM对象
- */
- const myCreateEle = (option, mountE) => {
- let e = document.createElement(option.el || 'div')
- option.className && e.classList.add(option.className)
- for (let p in (option.props || {})) {
- e.setAttribute(p, option.props[p])
- }
- e.textContent = option.text || ''
- e.style.cssText = option.style || ''
- mountE && mountE.appendChild(e)
- return e
- }
-
- // 初始生成元素
- const initLoadElement = () => {
- transContainerDOM = myCreateEle({
- props: { id: 'fy_transContainer' }
- }, document.body)
- // 内容容器
- fyContentDOM = myCreateEle({ props: { id: 'fy_contentBox' } }, transContainerDOM)
-
- // loading
- $(transContainerDOM).append(`
- <div id="fy_loading">
- <svg viewBox="25 25 50 50">
- <circle r="20" cy="50" cx="50"></circle>
- </svg>
- </div>
- `)
-
- // 拖动条
- fyDragBarDOM = myCreateEle({
- props: { id: 'fy_dragBar' }
- }, transContainerDOM)
-
- // 缩放点
- myCreateEle({
- props: { id: 'fy_Scale_rb' }
- }, transContainerDOM)
- myCreateEle({
- props: { id: 'fy_Scale_lb' }
- }, transContainerDOM)
-
- // 控制条
- let fyCtrlBarDom = myCreateEle({
- props: { id: 'fy_ctrl_ber' }
- }, transContainerDOM)
- // 选择翻译api源
- let fyApiSelectDom = myCreateEle({
- el: 'select',
- props: { id: 'fy_api_select' }
- }, fyCtrlBarDom)
- apiMap.forEach(item => {
- $(fyApiSelectDom).append(`
- <option value="${item.name}" title="${item.descript}">
- <span>${item.name}</span>
- </option>
- `)
- })
- // 语言选择
- let fyLangSelectDom = myCreateEle({
- el: 'select',
- props: { id: 'fy_select' }
- }, fyCtrlBarDom)
- selectTypes.forEach(item => {
- $(fyLangSelectDom).append(`
- <option value="${item.valueName}" title="${item.name}">
- <span>${item.valueName}</span>
- </option>
- `)
- })
-
- $(document.body).append(`
- <div id="fy_entry_container">
- <img src="${transImgData}" alt="翻译" />
-
- </div>
- `)
-
- let currlogo = (apiMap.find(item => item.name == currFromApi) || {})?.logo
- $(document.body).append(`
- <div id="fy_translate_tools_container">
- <div class="fy_tools_top">
- <div class="fy_title">翻译工具</div>
- <div class="fy_tools_ctrl_box">
- <div class="fy_tools_select_container">
- <div id="fy_tools_api_selected">
- <div id="fy_tools_api_active">
- <img src="${currlogo}" alt="">
- </div>
- <ul class="fy_tools_ul" id="fy_tools_api_ul">
- </ul>
- </div>
- <div class="fy_tools_select_from">auto</div>
- <img src="${zhuanhuangImgData}" alt="">
- <div class="fy_tools_select_to">
- <div id="fy_tools_selected_active">中</div>
- <ul class="fy_tools_ul" id="fy_tools_transTo_ul">
- </ul>
- </div>
- </div>
- <div id="fy_tools_close">❌</div>
- </div>
- </div>
- <div class="fy_tools_main">
- <div class="fy_translate_from">
- <textarea id="fy_translate_input" class="fy_tools_text_content" placeholder="输入翻译文字"></textarea>
- </div>
- <div class="fy_translate_to">
- <div id='fy_translate_result' class="fy_tools_text_content"></div>
- <div class="tools_ctrl_box">
- <img title="复制" class="tools_result_copy" alt="复制" src="${copyImgData}">
- <img title="发音" class="tools_result_audio" alt="发音" src="${audioImgData}">
- </div>
- </div>
- </div>
- <div class="fy_tools_footer">
- </div>
- </div>
- `)
-
- selectTypes.forEach(item => {
- $('#fy_tools_transTo_ul').append(`
- <li value="${item.valueName}">${item.valueName}</li>
- `)
- })
-
- apiMap.forEach(item => {
- $('#fy_tools_api_ul').append(`
- <li value="${item.name}"><img value="${item.name}" src="${item.logo}" alt="" /></li>
- `)
- })
-
- // ❌➖📌💡🎯📝✔️❓❗️📅🚫🔄✅📖📘
- // filter: grayscale(100%); 置灰
- }
-
- // 生成MD5值
- const calculateMD5 = (input) => {
- return CryptoJS.MD5(input).toString()
- }
-
- // 百度翻译
- const baiduTranslate = async (fromTransText, isTools = false) => {
- let targetLang = (baiduOptions.find(item => item.valueName == fyToType) || {})?.type
- let appid = '20240513002050392'
- let sign = calculateMD5(appid + fromTransText + salt + 'evAKKTnaxMEpHrnCxwDC')
- let param = `?q=${fromTransText}&from=auto&to=${targetLang}&appid=${appid}&salt=${salt}&sign=${sign}`
- console.log('baiduRequesy ', param)
- await GM_xmlhttpRequest({
- url: 'https://fanyi-api.baidu.com/api/trans/vip/translate' + param,
- method: 'GET',
- onload: function (response) {
- if (response.status === 200) {
- let res = JSON.parse((response.responseText || ''))
- if (!(res.trans_result && res.trans_result.length > 0)) return
- transRes = {
- apiType: 'baidu',
- formText: res.trans_result[0].src,
- toText: res.trans_result[0].dst
- }
- currTransToObjs.push({ ...transRes })
- renderRes(res.trans_result[0].src, res.trans_result[0].dst)
- $('#fy_loading').hide()
- // !reUpdate && computedContainer()
- // 复制翻译后文字
-
- copyText(res.trans_result[0].dst)
-
- isTools && toolsResult(res.trans_result[0].dst)
-
- } else {
- console.error('百度翻译请求失败,状态码: ' + response.status)
- }
- },
- onerror: function (e) {
- handleError(e)
- }
- })
- }
-
- /* -------------- 修复google翻译 @ChinaGodMan 2025-03-21 @ 12:25:06 Friday +0800 -------------- */
- const googleTranslate = async (texts, isTools = false) => {
- const buildQueryString = (params) => {
- return '?' + Object.keys(params).map(key => `${key}=${encodeURIComponent(params[key])}`).join('&')
- }
- isTools && (texts = [texts])
- let targetLang = (googleOptions.find(item => item.valueName == fyToType)).type
- const translateText = async (text) => {
- return new Promise((resolve, reject) => {
- const api = 'https://translate.googleapis.com/translate_a/single'
- const params = { client: 'gtx', dt: 't', sl: 'auto', tl: targetLang, q: text }
- GM_xmlhttpRequest({
- method: 'GET',
- url: api + buildQueryString(params),
- onload: function (response) {
- try {
- const data = JSON.parse(response.responseText.replace('\'', '\u2019'))
- const translatedText = data[0].reduce((acc, item) => acc + item[0], '')
- resolve(translatedText)
- } catch (error) {
- console.error('翻译失败:', error)
- reject('翻译失败')
- }
- },
- onerror: function (response) {
- console.error('请求翻译失败:', response.statusText)
- reject('请求翻译失败')
- }
- })
- })
- }
-
- try {
- const translatedResults = await Promise.all(texts.map(text => translateText(text)))
- transRes = {
- apiType: 'google',
- formTextArray: texts,
- toTextsArray: translatedResults
- }
- translatedResults.forEach((translatedText, i) => {
- renderRes(texts[i], translatedText)
- })
- $('#fy_loading').hide()
- //! 禁止复制翻译后的文本到剪辑版.2025-03-21 @ 12:33:38 Friday +0800
- //copyText(transRes.toTextsArray[0])
- isTools && toolsResult(transRes.toTextsArray[0])
- } catch (error) {
- console.error('翻译失败:', error)
- }
- }
-
- /* ------------------------------ 谷歌翻译构建URL请求参数 ----------------------------- */
- function buildQueryString(params) {
- return '?' + Object.keys(params).map(function (key) {
- return encodeURIComponent(key) + '=' + encodeURIComponent(params[key])
- }).join('&')
- }
- const renderRes = (formText, toText) => {
- $(fyContentDOM).append(`
- <div class="transText_node">
- <div class="transText_node_to" title="${formText}">${toText} <img title="复制" class="icon_style copy_icon" value="${toText}" alt="复制" src="${copyImgData}"></div>
- <div class="transText_node_from">${formText} <img class="icon_style copy_icon" title="复制" value="${formText}" alt="复制" src="${copyImgData}" ></div>
- </div>
- `)
- }
- const handleError = (e) => {
- // <div>或是否在用VPN代理❗️(百度的api😅)</div>
- $(fyContentDOM).append(`
- <div class="transText_node" style="text-align: center;">
- <div>请求失败❗️☹️</div>
- <div>请检查网络连接❗️</div>
- </div>
- `)
- $('#fy_loading').hide()
- }
-
- var currX = 0 // 当前鼠标位置
- var currY = 0 // 当前鼠标位置
- var isContainer = false // 容器是否出现
- var isCtrl = false // 是否处于可翻译状态
- // 绑定事件
- const bingEvents = () => {
- // // transContainerDOM 翻译容器
-
- // 绑定Ctrl+右键点击翻译
- // bindCtrlRightClick()
- // 绑定翻译键
- bingTransKeys()
-
- // 点击页面
- document.body.onclick = function (event) {
- if (isContainer) {
- clearTransContainer()
- }
- }
- // 清除翻译容器
- const clearTransContainer = () => {
- isContainer = false
- transContainerDOM.style.display = 'none'
- transContainerDOM.style.maxWidth = '500px'
- transContainerDOM.style.minWidth = '300px'
- transContainerDOM.style.width = 'auto'
- transContainerDOM.style.height = 'auto'
- fyContentDOM.textContent = ''
- fromTransTextArray = []
- }
-
- transContainerDOM.onclick = function (e) {
- e.stopPropagation() // 阻止事件冒泡
- }
- // 上下文菜单
- document.addEventListener('contextmenu', function (event) {
- if (isCtrl) {
- // 取消默认行为(阻止上下文菜单出现)
- event.preventDefault()
- }
- })
- bindHandleDrag() // 绑定拖动模块事件
- bindHandleScale() // 绑定缩放模块事件
- bindHandleSelectLang() // 绑定切换翻译事件
- bindHandleSelectApi() // 绑定切换翻译请求的api事件
- bindTextClick() // 点击翻译文本事件
-
- // 鼠标按键抬起事件
- document.addEventListener('mouseup', function (event) {
- if (isContainer) {
- const rect = transContainerDOM.getBoundingClientRect()
- currX = rect.left
- currY = rect.top
- } else {
- currX = event.clientX
- currY = event.clientY
- }
- })
-
- }
-
- // Ctrl + 鼠标右键点击事件
- // const bindCtrlRightClick = () => {
- // document.addEventListener('keydown', (e) => {
- // if (e.key === 'Control') {
- // isCtrl = true;
- // }
- // });
- // document.addEventListener('keyup', (e) => {
- // if (e.key === 'Control') {
- // isCtrl = false;
- // }
- // });
- // // 鼠标按下事件
- // document.addEventListener("mousedown", function (event) {
- // currX = event.clientX;
- // currY = event.clientY;
- // if (isCtrl && event.button === 2) {
- // // 获取Selection对象,选中的文本
- // let textAll = window.getSelection().toString();
- // if (!textAll) return
- // startTrans()
- // }
- // })
- // }
-
- // 绑定翻译事件按键
- const bingTransKeys = () => {
- var inOnlyKeyVal = ''
- document.addEventListener('keydown', function (event) {
- inOnlyKeyVal = event.key
- })
- document.addEventListener('keyup', function (event) {
- let textAll = window.getSelection().toString()
- if (!inOnlyKeyVal) return
- if (textAll) {
- // if ([event.key, inOnlyKeyVal].every(key => key === 'Control')) {
- // startTrans()
- // }
- if ([event.key, inOnlyKeyVal].every(key => key === '`')) {
- startTrans()
- }
- }
- if (textAll || fromTransTextArray.length) {
- // 按键必须为数字键,且都为正整数
- if ([event.key, inOnlyKeyVal].every(key => !isNaN(parseInt(key)))) {
- let selectedValue = selectTypes[parseInt(event.key) - 1].valueName || ''
- if (!selectedValue) return
- $('#fy_select').val(selectedValue)
- setFyToType(selectedValue)
- startTrans()
- }
- }
- inOnlyKeyVal = ''
- })
- }
-
- // 防止重复请求锁
- var isLockTrans = false
- // 开始执行翻译请求
- const startTrans = () => {
- if (isLockTrans) {
- showMessage({ message: '操作频繁,请稍后~', type: 'warning', time: 1000 })
- return
- }
- isLockTrans = true
- let textAll = window.getSelection().toString()
- let arrFroms = formatTrans(textAll)
- if (arrFroms && arrFroms.length) {
- fromTransTextArray = arrFroms
- }
- // 判断是否有选中翻译的文本
- isContainer = true // 更改容器状态
- transContainerDOM.style.display = 'flex'
- $('#fy_loading').show()
- fyContentDOM.textContent = ''
- computedContainer() // 计算容器位置
-
- const methodMap = {
- Baidu: (texts) => {
- texts.filter(text => text).forEach(text => {
- baiduTranslate(text)
- })
- },
- Google: googleTranslate
- }
- methodMap[currFromApi](fromTransTextArray)
- setTimeout(() => {
- isLockTrans = false
- }, 500)
- }
-
- // 切换翻译语言
- const bindHandleSelectLang = () => {
- document.getElementById('fy_select').onchange = function (e) {
- setFyToType(this.value)
- fyContentDOM.textContent = ''
- $('#fy_loading').show()
- startTrans()
- // googleTranslate(fromTransTextArray, true)
- }
-
- document.addEventListener('keyup', (e) => {
- if (isContainer && e.code === 'KeyB') {
- setCurrFromApi('Baidu')
- setTimeout(() => {
- startTrans()
- }, 10)
- }
- if (isContainer && e.code === 'KeyG') {
- setCurrFromApi('Google')
- setTimeout(() => {
- startTrans()
- }, 10)
- }
- })
-
- }
- // 切换api
- const bindHandleSelectApi = () => {
- document.getElementById('fy_api_select').onchange = function (e) {
- setCurrFromApi(this.value)
- fyContentDOM.textContent = ''
- $('#fy_loading').show()
- startTrans()
- // googleTranslate(fromTransTextArray, true)
- }
- }
-
- // 窗口拖动事件
- const bindHandleDrag = () => {
- let isMove = false
- let mouseToEleX
- let mouseToEleY
- // 拖动处理
- fyDragBarDOM.addEventListener('mousedown', function (e) {
- if (!isCtrl) {
- isMove = true
- fyDragBarDOM.style.cursor = 'grabbing'
- // 获取鼠标相对于元素的位置
- mouseToEleX = e.clientX - transContainerDOM.getBoundingClientRect().left
- mouseToEleY = e.clientY - transContainerDOM.getBoundingClientRect().top
- }
- })
- // 当鼠标移动时
- window.addEventListener('mousemove', (e) => {
- if (!isMove) return
- // 防止默认的拖动选择文本行为
- e.preventDefault()
- let t = (e.clientY - mouseToEleY) < 0 ? 0 : e.clientY - mouseToEleY
- // 更新元素的位置
- transContainerDOM.style.left = (e.clientX - mouseToEleX) + 'px'
- transContainerDOM.style.top = t + 'px'
- })
- // 当鼠标松开时
- window.addEventListener('mouseup', () => {
- isMove = false
- fyDragBarDOM.style.cursor = 'grab'
- })
- }
- // 改变窗口大小事件
- const bindHandleScale = () => {
- let mainCurrWidth
- let mainCurrHeight
- let cX, cY
- let isScale = false
- let scaleType = ''
-
- const scaleFun = (e, type) => {
- isScale = true
- mainCurrWidth = transContainerDOM.offsetWidth
- mainCurrHeight = transContainerDOM.offsetHeight
- cX = e.clientX
- cY = e.clientY
- scaleType = type
- let rect = transContainerDOM.getBoundingClientRect()
- if (scaleType == 'rb') {
- transContainerDOM.style.left = rect.left + 'px'
- transContainerDOM.style.right = 'auto'
- }
- if (scaleType == 'lb') {
- let scrollBarWidth = window.innerWidth - document.documentElement.clientWidth
- transContainerDOM.style.right = (window.innerWidth - rect.right - scrollBarWidth) + 'px'
- transContainerDOM.style.left = 'auto'
- }
- }
- fy_Scale_rb.addEventListener('mousedown', (e) => {
- scaleFun(e, 'rb')
- })
- fy_Scale_lb.addEventListener('mousedown', (e) => {
- scaleFun(e, 'lb')
- })
- // 当鼠标移动时
- window.addEventListener('mousemove', (e) => {
- if (!isScale) return
- // 防止默认的拖动选择文本行为
- e.preventDefault()
- transContainerDOM.style.maxWidth = 'none'
- let newHeight = mainCurrHeight + (e.clientY - cY)
- let newWidth = mainCurrWidth
-
- if (scaleType == 'rb') {
- newWidth = mainCurrWidth + (e.clientX - cX)
- }
- if (scaleType == 'lb') {
- newWidth = mainCurrWidth + (cX - e.clientX)
- }
- // 更新元素的位置
- transContainerDOM.style.width = Math.max(10, newWidth) + 'px'
- transContainerDOM.style.height = Math.max(10, newHeight) + 'px'
- })
- // 当鼠标松开时
- window.addEventListener('mouseup', () => {
- isScale = false
- })
- }
-
- // 点击译文事件
- var isClickLock = true
- const bindTextClick = () => {
- // fyContentDOM.addEventListener('click', function (event) {
- // if (!isClickLock) return;
- // isClickLock = false
- // setTimeout(() => {
- // isClickLock = true;
- // }, 300); // 双击事件的间隔时间通常是300毫秒左右
- // let textAll = window.getSelection().toString();
- // if (textAll) return;
- // let targetEle = event.target
- // if (!targetEle.classList.contains('transText_node')) {
- // targetEle = targetEle.parentNode
- // }
- // if (!targetEle.classList.contains('transText_node')) return;
-
- // if (targetEle.classList.contains('fy_node_expand')) {
- // targetEle.classList.remove('fy_node_expand');
- // } else {
- // targetEle.classList.add('fy_node_expand')
- // }
-
- // var rect = transContainerDOM.getBoundingClientRect();
- // // 获取视口的高度
- // var viewportHeight = window.innerHeight || document.documentElement.clientHeight;
- // // 计算元素底部到视口底部的距离
- // if ((viewportHeight - rect.bottom) < 30) {
- // transContainerDOM.style.height = (viewportHeight - rect.top - 50) + 'px'
- // }
- // });
-
- $(fyContentDOM).on('click', '.transText_node_to', function (e) {
- if (!isClickLock) return
- isClickLock = false
- setTimeout(() => {
- isClickLock = true
- }, 300) // 双击事件的间隔时间通常是300毫秒左右
- let textAll = window.getSelection().toString()
- if (textAll) return
- var parent = $(e.target).parent()[0]
- if ($(parent).hasClass('fy_node_expand')) {
- $(parent).removeClass('fy_node_expand')
- } else {
- $(parent).addClass('fy_node_expand')
- }
- var rect = transContainerDOM.getBoundingClientRect()
- // 获取视口的高度
- var viewportHeight = window.innerHeight || document.documentElement.clientHeight
- // 计算元素底部到视口底部的距离
- if ((viewportHeight - rect.bottom) < 30) {
- transContainerDOM.style.height = (viewportHeight - rect.top - 50) + 'px'
- }
-
- })
-
- $(fyContentDOM).on('click', '.copy_icon', function (e) {
- e.stopPropagation() // 阻止事件冒泡
- copyText(this.getAttribute('value'))
- showMessage({
- message: '复制成功',
- time: 800
- })
- })
-
- // 翻译工具-结果复制
- $('body').on('click', '.tools_result_copy', function (e) {
- e.stopPropagation() // 阻止事件冒泡
- copyText($('#fy_translate_result').text())
- showMessage({
- message: '复制成功',
- time: 800,
- mainDOM: document.getElementById('fy_translate_tools_container')
- })
- })
- // 翻译工具-结果发音
- $('body').on('click', '.tools_result_audio', function (e) {
- e.stopPropagation() // 阻止事件冒泡
- playAudioText($('#fy_translate_result').text())
- })
-
- }
-
- // 计算渲染容器高度位置
- const computedContainer = () => {
- let scrollBarWidth = window.innerWidth - document.documentElement.clientWidth
- let distance_right = (window.innerWidth - transContainerDOM.getBoundingClientRect().right - scrollBarWidth)
- if (distance_right < 5) {
- transContainerDOM.style.left = 'auto'
- transContainerDOM.style.right = '5px'
- } else {
- transContainerDOM.style.right = 'auto'
- transContainerDOM.style.left = currX + 'px'
- }
- transContainerDOM.style.top = currY + 'px'
-
- let topToBotton = window.innerHeight - currY
- if (transContainerDOM.offsetHeight > topToBotton) {
- transContainerDOM.style.height = topToBotton + 'px'
- }
- }
-
- // 语音播放文本
- const playAudioText = (text) => {
- // 创建一个新的 SpeechSynthesisUtterance 对象
- console.log('阅读~')
- const utterance = new SpeechSynthesisUtterance(text)
- // 设置一些可选的属性(例如音量、语速和音调)
- utterance.volume = 1 // 0到1之间的值
- utterance.rate = 1 // 0.1到10之间的值
- utterance.pitch = 1 // 0到2之间的值
- // 朗读文本
- window.speechSynthesis.speak(utterance)
-
- // // 检查浏览器是否支持语音合成
- // if ('speechSynthesis' in window) {
- // // 创建语音合成实例
- // var synthesis = window.speechSynthesis;
- // var textToSpeak = text;
- // // 创建语音合成的配置
- // var utterance = new SpeechSynthesisUtterance(textToSpeak);
- // // 使用默认语音
- // utterance.voice = speechSynthesis.getVoices()[0];
- // // 播放文本
- // synthesis.speak(utterance);
- // } else {
- // console.log("抱歉,您的浏览器不支持语音合成功能。");
- // }
- }
-
- const init = (e) => {
- initLoadElement()
- bingEvents()
- bindToolsEvent()
- }
-
- // 入口程序------------------------------------------------
- (function () {
- if (window.self !== window.top) return
- console.log('S translate ~')
- loadStyle()
- this.setTimeout(() => {
- init()
- if (localStorage.getItem('FY_TRANSLATE_TO')) {
- fyToType = localStorage.getItem('FY_TRANSLATE_TO')
- $('#fy_select').val(fyToType)
- fy_tools_selected_active.setAttribute('value', fyToType)
- fy_tools_selected_active.textContent = fyToType
- }
- if (localStorage.getItem('FY_TRANSLATE_API_TYPE')) {
- currFromApi = localStorage.getItem('FY_TRANSLATE_API_TYPE')
- $('#fy_api_select').val(currFromApi)
- }
- }, 200)
- })()
-
- const setFyToType = (type) => {
- localStorage.setItem('FY_TRANSLATE_TO', type)
- fyToType = type
- fy_tools_selected_active.setAttribute('value', type)
- fy_tools_selected_active.textContent = type
- }
- const setCurrFromApi = (type) => {
- localStorage.setItem('FY_TRANSLATE_API_TYPE', type)
- currFromApi = type
- $('#fy_api_select').val(type)
- }
-
- // 百度语言标识符列表
- var baiduOptions = [
- { type: 'zh', valueName: '中' },
- { type: 'en', valueName: '英' },
- { type: 'jp', valueName: '日' },
- { type: 'kor', valueName: '韩' },
- { type: 'fra', valueName: '法' },
- { type: 'spa', valueName: '西' },
- { type: 'ru', valueName: '俄' },
- { type: 'de', valueName: '德' },
- { type: 'it', valueName: '意' },
- { type: 'th', valueName: '泰' },
- { type: 'vie', valueName: '越' },
- { type: 'pt', valueName: '葡' },
- { type: 'ara', valueName: '阿' },
- { type: 'cht', valueName: '中(繁)' },
- { type: 'yue', valueName: '中(粤)' }
- ]
- // google语言标识符列表
- var googleOptions = [
- { type: 'zh-CN', valueName: '中' },
- { type: 'en', valueName: '英' },
- { type: 'ja', valueName: '日' },
- { type: 'ko', valueName: '韩' },
- { type: 'fr', valueName: '法' },
- { type: 'es', valueName: '西' },
- { type: 'ru', valueName: '俄' },
- { type: 'de', valueName: '德' },
- { type: 'it', valueName: '意' },
- { type: 'th', valueName: '泰' },
- { type: 'vi', valueName: '越' },
- { type: 'pt', valueName: '葡' },
- { type: 'ar', valueName: '阿' },
- { type: 'zh-TW', valueName: '中(繁)' },
- { type: 'zh-TW', valueName: '中(粤)' }
- ]
-
- // 唤起提示
- const showMessage = (options) => {
- let { type = 'success', message, time, mainDOM = transContainerDOM } = options
- let tipsDOM = myCreateEle({ text: message, type: type ?? 'success', style: 'position: absolute; top: 30px; left: 50%; transform: translate(-50%, 0%); padding: 2px 6px; border-radius: 2px; color:#fff; background-color: #67c23a; font-size: 10px;' }, mainDOM)
- const colorMap = {
- success: '#67c23a',
- warning: '#e6a23c',
- error: '#f56c6c',
- info: '#909399'
- }
- tipsDOM.style.backgroundColor = colorMap[type]
- setTimeout(() => {
- tipsDOM.remove()
- }, time ?? 2000)
- }
-
- // 格式化页面划选的文本,拆分成数组
- const formatTrans = (texts = '') => {
- return (texts.split(/[\n\t]+/) || []).filter(text => text)
- }
-
- // copy 文本
- const copyText = (text) => {
- navigator.clipboard.writeText(text)
- .then(() => {
- })
- .catch(err => {
- // 某些浏览器可能不支持或需要用户交互
- // console.error('无法复制文本: ', err);
- })
- }
-
- // -------------------------------------
-
- var isTools = false
- // 翻译小球-拖动事件
- function bindEntryMove() {
- let isMoveEntry = false
- let topMoveY
- fy_entry_container.addEventListener('mousedown', (e) => {
- isMoveEntry = true
- topMoveY = e.clientY - fy_entry_container.getBoundingClientRect().top
- })
- window.addEventListener('mousemove', (e) => {
- if (isMoveEntry) {
- e.preventDefault()
- fy_entry_container.style.top = (e.clientY - topMoveY) + 'px'
- }
- })
- // 当鼠标松开时
- window.addEventListener('mouseup', () => {
- isMoveEntry = false
- })
-
- // 翻译tools出现及隐藏
- fy_entry_container.addEventListener('mouseup', (e) => {
- if (!isTools) {
- showTools()
- }
- })
- fy_tools_close.onclick = function (e) {
- closeTools()
- }
-
- let rect = fy_translate_tools_container.getBoundingClientRect()
- var cacheLetf = rect.left - rect.width
- const showTools = () => {
- fy_translate_tools_container.style.left = (cacheLetf) + 'px'
- isTools = true
- setTimeout(() => {
- fy_translate_tools_container.style.transition = 'none'
- }, 600)
- }
- const closeTools = () => {
- isTools = false
- fy_translate_tools_container.style.transition = 'left 0.5s'
- cacheLetf = fy_translate_tools_container.getBoundingClientRect().left
- fy_translate_tools_container.style.left = '100%'
- }
-
- // 连续点击 两下``
- let isBackquote = false
- window.addEventListener('keydown', (e) => {
- if (e.code === 'Backquote' && isBackquote) {
- isTools ? closeTools() : showTools()
- isBackquote = false
- return
- }
- e.code === 'Backquote' && (isBackquote = true)
- setTimeout(() => {
- isBackquote = false
- }, 250)
- })
-
- }
-
- // 输入翻译小工具框-拖动事件
- function bingTranslateTools() {
- let isMoveTools = false
- let mouseToEleX
- let mouseToEleY
- fy_translate_tools_container.addEventListener('mousedown', (e) => {
- if (e.target.classList.contains('fy_tools_text_content') || e.target.id === 'fy_tools_close') return
- isMoveTools = true
- let rect = fy_translate_tools_container.getBoundingClientRect()
- mouseToEleX = e.clientX - rect.left
- mouseToEleY = e.clientY - rect.top
- fy_translate_tools_container.style.cursor = 'grabbing'
- })
- window.addEventListener('mousemove', (e) => {
- if (isMoveTools) {
- e.preventDefault()
- let t = (e.clientY - mouseToEleY) < 0 ? 0 : e.clientY - mouseToEleY
- fy_translate_tools_container.style.top = t + 'px'
- fy_translate_tools_container.style.left = (e.clientX - mouseToEleX) + 'px'
- }
- })
- // 当鼠标松开时
- window.addEventListener('mouseup', () => {
- isMoveTools = false
- fy_translate_tools_container.style.cursor = 'grab'
- })
- }
-
- // 翻译输入触发事件
- function bindInputChange() {
- let timer = null
- fy_translate_input.addEventListener('input', function (e) {
- if (timer) {
- clearTimeout(timer)
- }
- timer = setTimeout(() => {
- toolsTranslateRequest(this.value)
- }, 500)
- })
- }
-
- // tools切换翻译语言
- function bindChangeTranslateTo() {
- fy_tools_transTo_ul.onclick = function (e) {
- e.stopPropagation() // 阻止事件冒泡
- let value = e.target.getAttribute('value')
- if (!value) return
- setFyToType(value)
- fy_tools_selected_active.setAttribute('value', value)
- fy_tools_selected_active.textContent = e.target.textContent
- fy_tools_transTo_ul.style.display = 'none'
- toolsTranslateRequest($('#fy_translate_input').val())
- }
- fy_tools_selected_active.onclick = function (e) {
- e.stopPropagation() // 阻止事件冒泡
- if (window.getComputedStyle(fy_tools_transTo_ul).display === 'none') {
- fy_tools_transTo_ul.style.display = 'flex'
- } else {
- fy_tools_transTo_ul.style.display = 'none'
- }
- }
-
- fy_tools_api_active.onclick = function (e) {
- e.stopPropagation() // 阻止事件冒泡
- if (window.getComputedStyle(fy_tools_api_ul).display === 'none') {
- fy_tools_api_ul.style.display = 'flex'
- } else {
- fy_tools_api_ul.style.display = 'none'
- }
- }
- fy_tools_api_ul.onclick = function (e) {
- e.stopPropagation() // 阻止事件冒泡
- let value = e.target.getAttribute('value')
- if (!value) return
- setCurrFromApi(value)
- fy_tools_api_active.setAttribute('value', value)
- let currlogo = (apiMap.find(item => item.name == value) || {})?.logo
- $('#fy_tools_api_active img').attr('src', currlogo)
- fy_tools_api_ul.style.display = 'none'
- toolsTranslateRequest($('#fy_translate_input').val())
- }
-
- $('body').click(function (e) {
- fy_tools_transTo_ul.style.display = 'none'
- fy_tools_api_ul.style.display = 'none'
- })
- }
-
- const toolsTranslateRequest = (fromText) => {
- const methodMap = {
- Baidu: baiduTranslate,
- Google: googleTranslate
- }
- $('#fy_translate_result').text('')
- methodMap[currFromApi](fromText, true)
- }
- var toolsTranslateResultText = ''
- const toolsResult = (text) => {
- $('#fy_translate_result').text(text)
- toolsTranslateResultText = text
- if (text) {
- $('#fy_translate_result').show()
- } else {
- // $('#fy_translate_result').hide()
- }
- }
-
- const bindToolsEvent = () => {
- // 入口小球事件
- bindEntryMove()
- // 翻译工具事件
- bingTranslateTools()
- // 输入改变事件
- bindInputChange()
- // 切换翻译语言事件
- bindChangeTranslateTo()
- }
-
- // 转换
- var zhuanhuangImgData = ``
-
- // 复制
- var copyImgData = ``
-
- var audioImgData = ``
-
- // 翻译-icon
- var transImgData = ``