// ==UserScript==
// @namespace ATGT
// @name Bing Dict, Translate by selecting words, with pronunciation
// @name:zh-CN 必应词典,划词翻译,带英语发音
// @description Translate selected words by Bing Dict(Dictionary support EN to CN, CN to EN), with EN pronunciation, with CN pinyin. Translation is enabled by default, click the 'Bing Dict' icon at bottom left to toggle translation. Auto play pronunciation can be enabled in menu.
// @description:zh-CN 划词翻译,使用必应词典(支持英汉、汉英),带英语发音,带中文拼音。默认开启翻译,点击左下角的'Bing Dict'图标来开启/关闭翻译。自动发音可以通过菜单启用。
// @version 1.4.31
// @author StrongOp
// @license MIT
// @supportURL https://github.com/strongop/user-scripts/issues
// @match http://*/*
// @match https://*/*
// -match https://github.com/*
// @require https://greasemonkey.github.io/gm4-polyfill/gm4-polyfill.js
// @grant GM.xmlHttpRequest
// @grant GM_xmlhttpRequest
// @grant GM.setValue
// @grant GM.getValue
// @grant GM_setValue
// @grant GM_getValue
// @grant GM.registerMenuCommand
// @grant GM_registerMenuCommand
// @connect www.bing.com
// @connect cn.bing.com
// @connect dict.bing.com
// @icon https://www.bing.com/favicon.ico
// @run-at document-end
// @downloadURL https://update.greasyfork.icu/scripts/37358/Bing%20Dict%2C%20Translate%20by%20selecting%20words%2C%20with%20pronunciation.user.js
// @updateURL https://update.greasyfork.icu/scripts/37358/Bing%20Dict%2C%20Translate%20by%20selecting%20words%2C%20with%20pronunciation.meta.js
// ==/UserScript==
/* eslint: */
/* global GM GM_xmlhttpRequest GM_setValue GM_getValue GM_registerMenuCommand */
/* eslint curly: ["off", "multi", "consistent"] */
/* eslint no-empty: "off" */
/*
Change Log:
1.4.31:
3 Sep 2024, Fix parsing `octave` failure.
1.4.30:
22 Mar 2024, Fix for bing dict web chagne.
1.4.29:
7 Dec 2023, Fix mobile browser support.
v1.4.28:
6 Dec 2023, Fix pronunciation.
v1.4.23:
15 Oct 2019, Fix view removed by other script, create every time. rename names realted to enableTrans.
v1.4.21:
29 Mar 2019, Icon not show correctly if auto trans enabled and previously explicit disabled.
CSS sucks.
v1.4.20:
28 Mar 2019, Add menu for auto translation, tune icon style.
v1.4.15:
16 Feb 2019, Use int max as z-index to fix overlayed in vgr.com
v1.4.13:
13 Feb 2019, Add autoplay pronunciation menu
v1.4.5:
2 Feb 2019, Use more reliable CSS strategy.
v1.4.4:
2 Feb 2019, Fix click headword, fix enable/disable translate, fix translate selected on result view.
v1.4.1:
31 Jan 2019, Refactor with class, Add pronunciation support.
v1.3.8:
30 Jan 2019, Need user click on checkbox to enable translate.
v1.3.5:
21 Dec 2018, Fix GM_xmlhttpRequest not def in Greasemonkey, GM not define in Tampermonkey
v1.3.4:
28 Nov 2018, Fix GM.xmlHttpRequest not def in chrome.
v1.3.3:
28 Jan 2018, Fix dict provider overlap with result.
Add test cases.
v1.3.2:
27 Jan 2018, Add dict provider name: Bing Dict.
v1.3.1:
19 Jan 2018, Escape suggested words.
v1.3:
19 Jan 2018, refactor & parse search suggestion.
v1.2:
14 Jan 2018, Escape search word and result.
v1.1:
13 Jan 2018, Reset style for result div.
v1.0:
12 Jan 2018, Initial version.
*/
console.log(`=== bing-dict on '${location.href}' ===`);
let dict_result_id = 'ATGT-bing-dict-result-wrapper';
const DICT_RESULT_CSS = `
div#${dict_result_id}-reset {
all: initial;
}
div#${dict_result_id}-reset * {
all: initial;
display: block;
font-family: sans-serif;
font-size: small;
font-weight: normal;
white-space: normal;
margin: 0 0;
padding: 0 0;
}
div#${dict_result_id}-reset audio,
div#${dict_result_id}-reset img,
div#${dict_result_id}-reset input,
div#${dict_result_id}-reset label {
display: inline-block;
}
div#${dict_result_id}-reset a,
div#${dict_result_id}-reset span {
display: inline;
}
div#${dict_result_id} {
display: block;
position: fixed;
left: 2px;
bottom: 2px;
max-width: 32%;
z-index: 2147483647;
padding: 2px 2px;
margin: 0px 0px;
color: silver !important;
background-color: rgba(255,255,255,0.9) !important;
border-radius: 4px;
}
div#${dict_result_id} .margin-for-badget {
margin-right: 20px;
}
div#${dict_result_id} input.dict-provider {
display: none;
}
div#${dict_result_id} input.dict-provider ~ label img {
border-radius: 3px;
transition: all 0.2s ease-in-out;
width: 16px;
vertical-align: bottom;
}
div#${dict_result_id} input.dict-provider:not(:checked) ~ label img {
filter: gray; /* IE */
-webkit-filter: grayscale(1); /* Old WebKit */
-webkit-filter: grayscale(100%); /* New WebKit */
filter: url(resources.svg#desaturate); /* older Firefox */
filter: grayscale(100%); /* Current draft standard */
}
div#${dict_result_id}[data-display-mode="Result"] input.dict-provider + label img {
position: absolute;
top: 2px;
right: 2px;
}
div#${dict_result_id} input.dict-provider ~ label img:hover {
width: 32px;
}
div#${dict_result_id} .search_suggest_area ul li * {
font-size: x-small;
}
div#${dict_result_id} .error {
color: red;
}
div#${dict_result_id} .headword {
display: inline-block;
margin-right: 20px;
}
div#${dict_result_id} .headword a {
font-weight: bold;
font-size: medium;
}
div#${dict_result_id} .div_title {
font-weight: bold;
}
div#${dict_result_id} .suggest_word {
margin-right: 5px;
}
div#${dict_result_id} .mach_trans {
display: inline-block;
font-style: italic;
font-size: x-small;
}
/* a: link visited hover active, the order matters */
div#${dict_result_id} a:link {
color: #37a;
background-color: rgba(255,255,255,0.9);
text-decoration: none;
}
div#${dict_result_id} a:visited {
color: #37a;
}
div#${dict_result_id} a:hover {
color: white;
background-color: #37a;
cursor: pointer;
}
div#${dict_result_id} .pronuce {
display: block
}
div#${dict_result_id} .pronuce * {
color: gray;
}
div#${dict_result_id} audio {
width: 0;
height: 0;
}
div#${dict_result_id} .pronuce a:hover {
color: white;
background-color: rgba(255,255,255,0.9);
}
div#${dict_result_id} .mach_trans_result {
color: gray;
}
div#${dict_result_id} ul {
list-style-type: none;
padding: 1px;
margin: 0px;
}
div#${dict_result_id} ul li{
margin-top: 1px;
color: gray;
}
div#${dict_result_id} ul li span.def-category {
float:left;
color: white;
background-color: gray;
text-align: center;
padding: 0 2px;
margin-right: 3px;
border-radius: 3px;
}
div#${dict_result_id} a img.audioPlayer:hover {
opacity: 0.8;
}
div#${dict_result_id} img.audioPlayer {
width: 1em;
height: 1em;
}
/*
body {
background: gray;
}
*/
`;
class DictResultView {
//dictResultDiv;
constructor(prefs) {
this.prefs = prefs;
this.initView();
}
initView() {
this.addStyleSheet();
this.createDictResultDiv();
}
addStyleSheet() {
let cssid = `${dict_result_id}-css`;
if (document.querySelector(`#${cssid}`))
return;
let style = document.createElement('STYLE');
style.type = 'text/css';
style.id = cssid;
style.appendChild(document.createTextNode(DICT_RESULT_CSS));
document.head.appendChild(style);
}
createDictResultDiv() {
let div_wrapper_reset = document.createElement('DIV');
div_wrapper_reset.id = `${dict_result_id}-reset`;
let div = document.createElement('DIV');
div.id = `${dict_result_id}`;
div_wrapper_reset.appendChild(div);
document.body.appendChild(div_wrapper_reset);
this.dictResultDiv = div;
}
setProvider(provider) {
this.dictProvider = provider;
}
setResult(defs) {
this.dictResultDiv.innerHTML = this.dictProvider.dictProviderDesc + defs;
this.dictResultDiv.style.display = 'block';
this.showEnableTransBtn();
if (this.prefs.isTransEnabled()) {
this.dictResultDiv.style.minWidth = '200px';
this.dictResultDiv.style.background = 'rgba(255, 255, 255, 0.9) !important';
} else {
this.dictResultDiv.style.minWidth = 'unset';
this.dictResultDiv.style.background = 'rgba(255, 255, 255, 0.6) !important';
}
// DO NOT DELETE, set mode to use different css rules
this.dictResultDiv.dataset['displayMode'] = (defs.length == 0) ? 'IconOnly' : 'Result';
}
hideResult() {
this.dictResultDiv.style.display = 'none';
this.enableTransBtnVisibility = false;
let wrapper = document.querySelector(`#${dict_result_id}-reset`);
if (wrapper)
wrapper.remove();
}
mouseEventInView(event) {
if (!/touch|mouse/.test(event.type) && !event.clientX)
return;
let lastPos;
if (/touch/.test(event.type)) {
let touches = event.touches;
lastPos = touches.item(touches.length - 1) || {}
this.lastTouch = lastPos;
this.lastTouchTime = Date.now();
} else {
lastPos = event;
}
// if Mouse is inside result element
let divRect = this.dictResultDiv.getBoundingClientRect();
let isInView = (lastPos.clientX >= divRect.left && lastPos.clientX <= divRect.right &&
lastPos.clientY >= divRect.top && lastPos.clientY <= divRect.bottom);
// console.log(`InView ${isInView} lastpos ${lastPos.clientX},${lastPos.clientY}, divRect`, JSON.stringify(divRect));
return isInView;
}
mouseEventInDictProviderBanner(event) {
if (!/touch|mouse/.test(event.type))
return;
let lastPos;
if (/touch/.test(event.type)) {
lastPos = event.touches.item(event.touches.length - 1) || {};
} else {
lastPos = event;
}
// if Mouse is inside dict provider to enable/disable tranlation
try {
let divRect = this.dictResultDiv.querySelector(`.dict-provider ~ label img`).getBoundingClientRect();
let isInView = (lastPos.clientX >= divRect.left && lastPos.clientX <= divRect.right &&
lastPos.clientY >= divRect.top && lastPos.clientY <= divRect.bottom);
return isInView;
} catch (e) {
return false;
}
}
showEnableTransBtn() {
this.enableTransBtnVisibility = true;
let prefs = this.prefs;
let view = this;
function enableTransBtnHandler(event) {
//console.log('enableTransChoiceHandler called, translate enable ', event.target.checked);
prefs.transEnabledOnPage = event.target.checked ? TRANS_EXPLICIT_ENABLE : TRANS_EXPLICIT_DISABLE;
prefs.updateTransEnabledList(location.host, prefs.transEnabledOnPage);
if (prefs.isTransEnabled() && CurrentSelWord.length > 0)
setTimeout(view.dictProvider.search.bind(view.dictProvider), 0, CurrentSelWord);
else
view.hideResult();
}
let enableCheckbox = document.querySelector(`div#${dict_result_id} input#enableTrans`);
enableCheckbox.checked = prefs.isTransEnabled();
enableCheckbox.onclick = enableTransBtnHandler;
}
}
var dictCache = {};
var entityMap = {
'&': '&',
'<': '<',
'>': '>',
'"': '"',
"'": ''',
'/': '/',
'`': '`',
'=': '='
};
function escapeHtml(string) {
return String(string).replace(/[&<>"'`=/]/g, function (s) {
return entityMap[s];
});
}
class DictProvider {
//resultView;
constructor(prefs, resultView) {
this.prefs = prefs;
this.resultView = resultView;
this.dictProviderDesc = `
`;
this.resultView.setProvider(this);
}
search(word) {
console.log(`base DictProvider::search ${word}`);
return 'not implement yet.';
}
}
class BingDictProvider extends DictProvider {
constructor(prefs, resultView) {
super(prefs, resultView);
let bingIcon = '';
this.dictProviderDesc = `
`;
this.resultView.setProvider(this);
this.baseDomain = 'dict.bing.com';
this.webDomain = 'www.bing.com';
this.baseURL = `https://${this.baseDomain}`;
this.dictMarketIsCN = false;
}
search(word) {
//console.log(`>>> do search ${word}`);
let self = this;
if (!this.dictMarketIsCN) {
let changeMarketURL = `https://www.bing.com/?mkt=zh-CN`;
(typeof GM_xmlhttpRequest != 'undefined' && GM_xmlhttpRequest || GM.xmlHttpRequest)({
url: changeMarketURL,
method: 'GET',
onload: (response) => { self.dictMarketIsCN = true; },
onerror: (response) => { console.log('Change market failed. Change to CN what so ever!'); self.dictMarketIsCN = true; },
});
}
function limitedSearchString(headword) {
return `${headword.substring(0, 77)}${(headword.length >= 77) ? '...' : ''}`;
}
/*
function playAudio(audioLink) {
let audio = new Audio(audioLink);
audio.play();
}*/
function parseVoiceLink(elem, id_prefix) {
let voiceLink;
let lang = id_prefix.includes('US') ? 'US' : 'UK';
//let voiceIcon;
try {
let linkElem = elem.childNodes[0];
//console.log('PronuceLink', linkElem);
voiceLink = linkElem.dataset['mp3link'];
//let matches = handler.match(/'(https?:\/\/[^ ']*)'\s*,\s*'([^']*)'/m);
//console.log('matches', matches);
//voiceLink = matches[1];
} catch (e) {
console.log('parseVoiceLink', e);
return '';
}
let id = `${id_prefix}_${Math.random()}`;
let speakerIcon = '';
let voiceHTML = `
`;
return voiceHTML;
}
function parsePronuce(elem) {
let prUS = elem.querySelector('.hd_prUS');
let prUK = elem.querySelector('.hd_pr');
let pronText = '';
if (prUS)
pronText = `${escapeHtml(prUS.innerText)}${parseVoiceLink(prUS.nextElementSibling, 'voiceUS')}
${escapeHtml(prUK.innerText)}${parseVoiceLink(prUK.nextElementSibling, 'voiceUK')}`;
else
pronText = `${escapeHtml(elem.innerText)}`;
return pronText;
}
function parseDefinition(page, url) {
//console.log('parseDefinition');
let qdef = page.querySelector('.qdef');
//console.log('qdef ', qdef);
let hd_area = qdef.childNodes[0];
let headword = '';
let pronuce = '';
try {
headword = escapeHtml(hd_area.querySelector('#headword').innerText);
pronuce = parsePronuce(hd_area.querySelector('.hd_tf_lh'));
} catch (e) {
console.log('parse headword/pronunce fail', e);
}
headword = `