// ==UserScript==
// @namespace ATGT
// @name Bing Dict, 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 disabled by default, check the 'Bing Dict' at bottom left to enable tranlation. Auto play pronunciation can be enabled in menu.
// @description:zh-CN 划词翻译,使用必应词典(支持英汉、汉英),带英语发音,带中文拼音,默认不开启翻译,勾选左下角的'Bing Dict'开启翻译。自动播放发音可以通过菜单启用。
// @version 1.4.21
// @author StrongOp
// @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
// @connect www.bing.com
// @icon https://www.bing.com/favicon.ico
// @run-at document-end
// ==/UserScript==
/* eslint: */
/* global GM GM_xmlhttpRequest GM_setValue GM_getValue GM_registerMenuCommand */
/* eslint curly: ["off", "multi", "consistent"] */
/* eslint no-empty: "off" */
/*
Change Log:
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.
*/
/*
if (typeof GM_xmlhttpRequest !== 'undefined')
var GM = {
setValue: GM_setValue,
getValue: GM_getValue
};
*/
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: black;
background-color: rgba(255,255,255,0.9);
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} .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;
}
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.addStyleSheet();
this.createDictResultDiv();
this.hideResult();
}
addStyleSheet() {
let style = document.createElement('STYLE');
style.type = 'text/css';
style.appendChild(document.createTextNode(DICT_RESULT_CSS));
document.head.appendChild(style);
}
createDictResultDiv() {
//this.addStyleSheet();
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 + defs;
this.dictResultDiv.style.display = 'block';
this.showEnableTransChoice();
if (this.prefs.isTransEnabled()) {
this.dictResultDiv.style.minWidth = '200px';
this.dictResultDiv.style.background = 'rgba(255, 255, 255, 0.9)';
} else {
this.dictResultDiv.style.minWidth = 'unset';
this.dictResultDiv.style.background = 'rgba(255, 255, 255, 0.6)';
}
this.dictResultDiv.dataset['displayMode'] = defs === '' ? 'IconOnly' : 'Result';
}
hideResult() {
this.dictResultDiv.style.display = 'none';
this.transEnableShown = false;
}
mouseEventInView(event) {
// if Mouse is inside result element
let divRect = this.dictResultDiv.getBoundingClientRect();
let isInView = (event.clientX >= divRect.left && event.clientX <= divRect.right &&
event.clientY >= divRect.top && event.clientY <= divRect.bottom);
return isInView;
}
mouseEventInDictProviderBanner(event) {
// if Mouse is inside dict provider to enable/disable tranlation
try {
let divRect = this.dictResultDiv.querySelector(`.dict-provider ~ label img`).getBoundingClientRect();
let isInView = (event.clientX >= divRect.left && event.clientX <= divRect.right &&
event.clientY >= divRect.top && event.clientY <= divRect.bottom);
return isInView;
} catch (e) {
return false;
}
}
showEnableTransChoice() {
this.transEnableShown = true;
let prefs = this.prefs;
let view = this;
function enableTransChoiceHandler(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(bingDict.search.bind(bingDict), 0, CurrentSelWord);
else
view.hideResult();
}
let enableCheckbox = document.querySelector(`div#${dict_result_id} input#enableTrans`);
enableCheckbox.checked = this.prefs.transEnabledOnPage === TRANS_EXPLICIT_ENABLE;
enableCheckbox.onclick = enableTransChoiceHandler;
}
}
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;
let dictProvider = `
<input type="checkbox" id="enableTrans" class="dict-provider" name="enableTrans">
<label for="enableTrans">No Dict Provider</label>
`;
this.resultView.setProvider(dictProvider);
}
search(word) {
console.log(`base DictProvider::search ${word}`);
return 'not implement yet.';
}
}
class BingDictProvider extends DictProvider {
constructor(prefs, resultView) {
super(prefs, resultView);
let bingIcon = '';
let dictProvider = `
<input type="checkbox" id="enableTrans" class="dict-provider" name="enableTrans"
title="Click to enable/disable translation with Bing Dict" ><label for="enableTrans"><img src='${bingIcon}' alt='Bing Dict' title="Click to enable/disable translation with Bing Dict"></label>
`;
this.resultView.setProvider(dictProvider);
this.baseURL = 'https://www.bing.com/';
}
search(word) {
console.log(`>>> do search ${word}`);
let self = this;
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);
let handler = linkElem.attributes['onclick'].value;
let matches = handler.match(/'(https?:\/\/[^ ']*)'\s*,\s*'([^']*)'/m);
//console.log('matches', matches);
voiceLink = matches[1];
//voiceIcon = matches[2];
} catch (e) {
console.log('parseVoiceLink', e);
}
let id = `${id_prefix}_${Math.random()}`;
let speakerIcon = '';
let voiceHTML = `<audio id="${id}" src="${voiceLink}" preload="auto" ${self.prefs.autoplayPronuce[lang] ? 'autoplay' : ''}></audio>
<a onclick="javascript: document.getElementById('${id}').play()"
onmouseover="javascript: document.getElementById('${id}').play()">
<img class="audioPlayer" src="${speakerIcon}"></img>
</a>`;
return voiceHTML;
}
function parsePronuce(elem) {
let prUS = elem.querySelector('.hd_prUS');
let prUK = elem.querySelector('.hd_pr');
let pronText = '';
if (prUS)
pronText = `<span>${escapeHtml(prUS.innerText)}</span>${parseVoiceLink(prUS.nextElementSibling, 'voiceUS')}
 <span>${escapeHtml(prUK.innerText)}</span>${parseVoiceLink(prUK.nextElementSibling, 'voiceUK')}`;
else
pronText = `<span>${escapeHtml(elem.innerText)}</span>`;
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) {
}
headword = `<div class='headword'>
<a href='${url}' target='_blank'>${headword ? headword : word}</a>
</div>`;
pronuce = `<div class='pronuce'>${pronuce}</div>`;
let defs = '';
try {
let def_area = qdef.childNodes[1];
let def_list = def_area.querySelectorAll('li');
defs = '<ul>';
for (let def of def_list) {
defs += `<li><span class='def-category'>${escapeHtml(def.childNodes[0].innerText)}</span>
${escapeHtml(def.childNodes[1].innerText)}</li>`;
}
defs += '</ul>';
} catch (e) {
defs = '';
}
return headword + pronuce + defs;
}
function parseMachTrans(page, url) {
//console.log('parseMachTrans');
try {
let trans_area = page.querySelector('.lf_area');
let smt_hw_elem = trans_area.querySelector('.smt_hw');
let smt_hw = escapeHtml(smt_hw_elem.innerText);
let headword = escapeHtml(smt_hw_elem.nextElementSibling.innerText);
let trans_result = escapeHtml(smt_hw_elem.nextElementSibling.nextElementSibling.innerText);
smt_hw = `<div class='mach_trans margin-for-badget'>${smt_hw}</div>`;
headword = `<div class='headsentence'>
<a href='${url}' target='_blank'>${limitedSearchString(headword)}</a>
</div>`;
trans_result = `<div class='mach_trans_result'>${trans_result}</div>`;
return smt_hw + headword + trans_result;
} catch (e) {
console.error('parseMachTrans error');
return '';
}
}
function parseSearchSuggestDetail(detail) {
let suggest = escapeHtml(detail.querySelector('.df_wb_a').innerText);
suggest = `<div class='div_title'>${suggest}</div>`;
let defs = '<ul>';
for (let s of detail.querySelectorAll('.df_wb_c')) {
let r0 = s.childNodes[0];
let r1 = s.childNodes[1];
defs += `<li>
<a class='suggest_word' href='//www.bing.com${r0.pathname}${r0.search}' target='_blank'>
${escapeHtml(r0.innerText)}</a>
<span>${escapeHtml(r1.innerText)}<span>
</li>`;
}
defs += '</ul>';
return suggest + defs;
}
function parseSearchSuggest(page, url) {
//console.log('parseSearchSuggest');
let trans_area = page.querySelector('.lf_area');
let headword = escapeHtml(trans_area.querySelector('.dym_p').innerText);
let suggest = escapeHtml(trans_area.querySelector('.p2-2').innerText);
headword = `<div class='headword'><a href='${url}' target='_blank'>${headword}</a></div>`;
suggest = `<div class='div_title'>${suggest}</div>`;
let defs = '';
for (let detail of trans_area.querySelectorAll('.dym_area')) {
defs += parseSearchSuggestDetail(detail);
}
return `<div class='search_suggest_area'>${headword}${suggest}${defs}</div>`;
}
function parseDictResultDom(page, url) {
//console.log('page ', page);
let qdef = page.querySelector('.qdef');
let smt_hw = page.querySelector('.smt_hw');
let search_suggest = page.querySelector('.dym_area') && page.querySelector('.df_wb_c');
//let no_result = page.querySelector('.no_results');
if (qdef)
return parseDefinition(page, url);
else if (smt_hw)
return parseMachTrans(page, url);
else if (search_suggest)
return parseSearchSuggest(page, url);
else
return '';
}
const FAILURE_MSG = `<span class='margin-for-badget'>No result for '${escapeHtml(limitedSearchString(word))}'.<span><br />
Try <a href='https://www.bing.com/translator' target='_blank'>Microsoft Translator</a>.`;
function parseDictResult(word, response) {
//console.log('search dict ok', response);
let defs = '';
let url = response.finalUrl;
try {
let doc = (new DOMParser()).parseFromString(response.responseText, 'text/html');
defs = parseDictResultDom(doc, url);
if (!defs)
defs = FAILURE_MSG;
dictCache[word] = defs;
} catch (e) {
console.log('parseDictResult failed:\n ', e, e.stack);
defs = `<span class='error'>Error</span> parsing result of
<a href='${url}' target='_blank'>${escapeHtml(limitedSearchString(word))}</a>, <br />
${FAILURE_MSG}`;
}
self.resultView.setResult(defs);
}
function searchDictFail(word, response) {
//console.log('search dict fail ', response);
let url = response.finalUrl;
let status = (response.status ? response.status : '')
+ ' ' + (response.statusText ? response.statusText : '');
self.resultView.setResult(`<span class='error'>Error</span>
searching <a href='${url}' target='_blank'>${escapeHtml(limitedSearchString(word))}</a>, ${status}<br />
${FAILURE_MSG}`);
}
function searchBingDict(word) {
if (word in dictCache && dictCache[word]) {
//console.log(`cache hit '${word}'`);
self.resultView.setResult(dictCache[word]);
return;
} else {
//console.log('cache miss');
}
let url = 'http://www.bing.com/dict/search?q=' + encodeURIComponent(word);
console.log(url);
self.resultView.setResult(`Searching <span class='headword'>
<a href='${url}' target='_blank'>${escapeHtml(limitedSearchString(word))}</a>
</span>`);
(typeof GM_xmlhttpRequest != 'undefined' && GM_xmlhttpRequest || GM.xmlHttpRequest)({
url: url,
method: 'GET',
onload: (response) => parseDictResult(word, response),
onerror: (response) => searchDictFail(word, response),
});
}
searchBingDict(word);
}
}
var CurrentSelWord = '';
const TRANS_EXPLICIT_NONE = 0;
const TRANS_EXPLICIT_DISABLE = 1;
const TRANS_EXPLICIT_ENABLE = 2;
class DictPrefs {
constructor() {
this.transEnabledOnPage = TRANS_EXPLICIT_NONE;
this.globallyEnableAutoTrans = true;
this.autoplayPronuce = {
US: false,
UK: false
};
this.checkEnableTrans();
this.checkAutoPlay();
}
isTransEnabled() {
return (this.transEnabledOnPage !== TRANS_EXPLICIT_DISABLE) &&
(this.transEnabledOnPage === TRANS_EXPLICIT_ENABLE || this.globallyEnableAutoTrans);
}
updateTransEnabledList(key, value) {
GM.getValue('transEnabledList', {}).then((transEnabledList) => {
console.log(`update ${key} => ${value} into transEnabledList`, transEnabledList);
transEnabledList[key] = value;
GM.setValue('transEnabledList', transEnabledList);
//console.log(`updated ${key} => ${value} into transEnabledList`, transEnabledList);
});
}
checkEnableTrans() {
GM.getValue('globallyEnableAutoTrans', {}).then((globallyEnableAutoTrans) => {
console.log('globallyEnableAutoTrans', globallyEnableAutoTrans);
if (typeof globallyEnableAutoTrans === 'undefined')
return;
if (typeof globallyEnableAutoTrans === 'string')
GM.setValue('globallyEnableAutoTrans', true);
this.globallyEnableAutoTrans = globallyEnableAutoTrans;
GM_registerMenuCommand(`Globally ${(!this.globallyEnableAutoTrans) ? 'En' : 'Dis'}able Auto Translation. (Currently ${this.globallyEnableAutoTrans ? 'En' : 'Dis'}abled)`, () => { this.setAutoTranslate(!this.globallyEnableAutoTrans); });
});
GM.getValue('transEnabledList', {}).then((transEnabledList) => {
console.log('transEnabledList', transEnabledList);
if (typeof transEnabledList === 'undefined')
return;
if (typeof transEnabledList === 'string')
GM.setValue('transEnabledList', {});
if (location.host in transEnabledList) {
this.transEnabledOnPage = transEnabledList[location.host];
if (typeof this.transEnabledOnPage === 'boolean') {
this.transEnabledOnPage = this.transEnabledOnPage === true ? TRANS_EXPLICIT_ENABLE : TRANS_EXPLICIT_DISABLE;
this.updateTransEnabledList(location.host, this.transEnabledOnPage);
}
}
});
}
setAutoTranslate(en) {
this.globallyEnableAutoTrans = en;
GM.setValue('globallyEnableAutoTrans', en);
location.reload();
}
checkAutoPlay() {
GM.getValue('autoplayPronuce', {}).then((autoplayPronuce) => {
console.log('autoplay', autoplayPronuce);
if (!autoplayPronuce)
return;
if (typeof autoplayPronuce === 'string')
GM.setValue('autoplayPronuce', {});
this.autoplayPronuce.US = autoplayPronuce.US;
this.autoplayPronuce.UK = autoplayPronuce.UK;
if (self == top) {
GM_registerMenuCommand(`Globally ${(!this.autoplayPronuce.US) ? 'En' : 'Dis'}able Autoplay US pronunciation. (Currently ${this.autoplayPronuce.US ? 'En' : 'Dis'}abled)`, () => { this.setAutoplay('US', !this.autoplayPronuce.US); });
GM_registerMenuCommand(`Globally ${(!this.autoplayPronuce.UK) ? 'En' : 'Dis'}able Autoplay UK pronunciation. (Currently ${this.autoplayPronuce.UK ? 'En' : 'Dis'}abled)`, () => { this.setAutoplay('UK', !this.autoplayPronuce.UK); });
}
});
}
setAutoplay(lang, on) {
console.log(`${on ? 'en' : 'dis'}able auto play of ${lang} pronunciation`);
this.autoplayPronuce[lang] = on;
GM.setValue('autoplayPronuce', this.autoplayPronuce);
location.reload();
}
}
var dictPrefs = new DictPrefs();
//dictPrefs.checkEnableTrans();
var dictResultView = new DictResultView(dictPrefs);
var bingDict = new BingDictProvider(dictPrefs, dictResultView);
document.addEventListener('mouseup', function (event) {
// click on enable/disable translation area, return, so the checkbox handler can be called
if (dictResultView.mouseEventInDictProviderBanner(event))
return;
// get selected word/sentense
CurrentSelWord = window.getSelection().toString().replace(/^\s*|\s*$/g, '');
console.log(`selected: '${CurrentSelWord}', length ${CurrentSelWord.length}`);
// click on page other than result view hides the result view
if (!dictResultView.mouseEventInView(event) && CurrentSelWord.length == 0) {
dictResultView.hideResult();
return;
}
// show enable translate option if not enabled
if (!dictPrefs.isTransEnabled()) {
if (!dictResultView.transEnableShown)
dictResultView.setResult('');
console.log('translate not enabled.');
return;
}
// return if single click on result view
// return if click headword to open new tab
// go on translate if word selected in page/result_view
if (dictResultView.mouseEventInView(event) &&
(CurrentSelWord.length == 0 || event.target.nodeName == 'A' ||
event.target.parentNode.nodeName == 'A')) {
return;
}
bingDict.search(CurrentSelWord);
});
function dictTest() {
let testWords = [
'tunnel',
'hello', // word definition
'browsing experience', // long sentense
'你好',
'hello, this is world', //machine translation
'ndalo', // ambigous
'DNS queries', // example sentence only
'<script>', // html escape
'',
'overrideMimeType', // no result
];
for (let i = 0; i < testWords.length; i++) {
setTimeout(bingDict.search.bind(bingDict), (i + 1) * 3000, testWords[i]);
}
dictResultView.setResult(
`<pre>
/*TODO:
* 1. Translation enable/disable test.
* 2. Audio/voice test.
* 3. Click on headword to open new tab of dict.bing.com
* 4. Select word on result view to translate
*/
</pre>`);
}
//dictTest();
console.log(`=== /bing-dict on '${location.href}' ===`);