翻译小工具 | 翻译任何语言

任何语言翻译及快捷一键翻译想要语言 | 页面翻译 | 选中文字( 按Ctrl )| 提供快捷方式,只需按一个键即可快速翻译想要语言 |英文学习 | 翻译文可设置,支持全球多数通用语言 | 有什么问题都可以反馈

当前为 2025-03-26 提交的版本,查看 最新版本

您需要先安装一个扩展,例如 篡改猴Greasemonkey暴力猴,之后才能安装此脚本。

You will need to install an extension such as Tampermonkey to install this script.

您需要先安装一个扩展,例如 篡改猴暴力猴,之后才能安装此脚本。

您需要先安装一个扩展,例如 篡改猴Userscripts ,之后才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。

您需要先安装用户脚本管理器扩展后才能安装此脚本。

(我已经安装了用户脚本管理器,让我安装!)

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

(我已经安装了用户样式管理器,让我安装!)

// ==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            [email protected],人民的勤务员 <[email protected]>
// @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}&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; <img title="复制" class="icon_style copy_icon" value="${toText}" alt="复制" src="${copyImgData}"></div>
            <div class="transText_node_from">${formText}&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; <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 = ``