// ==UserScript==
// @name Google Search Region
// @namespace jmln.tw
// @version 0.4.1
// @description A user script that lets you quickly switch Google search to different region.
// @author Jimmy Lin
// @license MIT
// @homepageURL https://github.com/jmlntw/google-search-region
// @supportURL https://github.com/jmlntw/google-search-region/issues
// @include /^https:\/\/(?:ipv4|ipv6|www)\.google\.(?:[a-z\.]+)\/search\?(?:.+&)?q=[^&]+(?:&.+)?$/
// @exclude /^https:\/\/(?:ipv4|ipv6|www)\.google\.(?:[a-z\.]+)\/search\?(?:.+&)?tbm=lcl(?:&.+)?$/
// @compatible firefox
// @compatible chrome
// @compatible edge
// @compatible opera
// @run-at document-end
// @grant GM_getValue
// @grant GM_setValue
// @grant GM.getValue
// @grant GM.setValue
// ==/UserScript==
// =============================================================================
// Add compatibility between the Greasemonkey 4 APIs and existing/legacy APIs.
// =============================================================================
if (typeof GM === 'undefined') {
// eslint-disable-next-line no-global-assign
GM = {
getValue: (...args) => Promise.resolve(GM_getValue.apply(this, args)),
setValue: (...args) => Promise.resolve(GM_setValue.apply(this, args))
}
}
function addStyle (css) {
const style = document.createElement('style')
style.type = 'text/css'
style.textContent = css
document.head.appendChild(style)
return style
}
GM.addStyle = addStyle
// =============================================================================
// Helper Functions
// =============================================================================
/**
* @param {string} selector
* @param {Element} [context]
* @return {Element}
*/
function $ (selector, context) {
return (context || document).querySelector(selector)
}
/**
* @param {string} selector
* @param {Element} [context]
* @return {NodeListOf<Element>}
*/
function $$ (selector, context) {
return (context || document).querySelectorAll(selector)
}
/**
* @param {Element} target
* @param {string} type
* @param {EventListener} callback
* @param {boolean} [useCapture]
*/
function $on (target, type, callback, useCapture) {
target.addEventListener(type, callback, !!useCapture)
}
/**
* @param {Element} target
* @param {string} selector
* @param {string} type
* @param {EventListener} callback
*/
function $delegate (target, selector, type, callback) {
const useCapture = (type === 'blur') || (type === 'focus')
const dispatchEvent = function dispatchEvent (event) {
if (event.target.matches(selector)) { callback.call(event.target, event) }
}
$on(target, type, dispatchEvent, useCapture)
}
if (window.NodeList && !window.NodeList.prototype.forEach) {
window.NodeList.prototype.forEach = Array.prototype.forEach
}
// =============================================================================
// Template Engine
// =============================================================================
/**
* @param {string} text
* @param {Object} data
* @return {string}
*/
function renderTemplate (text, data) {
const matcher = /<%-([\s\S]+?)%>|<%=([\s\S]+?)%>|<%([\s\S]+?)%>|$/g
const escapeChar = function escapeChar (text) {
return text
.replace(/\\/g, '\\\\')
.replace(/'/g, "\\'")
.replace(/\r/g, '\\r')
.replace(/\n/g, '\\n')
.replace(/\u2028/g, '\\u2028')
.replace(/\u2029/g, '\\u2029')
}
const escape = function escape (text) {
return ('' + text)
.replace(/&/g, '&')
.replace(/</g, '<')
.replace(/>/g, '>')
.replace(/"/g, '"')
.replace(/'/g, ''')
.replace(/`/g, '`')
}
let index = 0
let source = "__p += '"
text.replace(matcher, (match, escape, interpolate, evaluate, offset) => {
source += escapeChar(text.slice(index, offset))
index = offset + match.length
if (escape) {
source += `' + ((__t = (${escape})) == null ? '' : escape(__t)) + '`
} else if (interpolate) {
source += `' + ((__t = (${interpolate})) == null ? '' : __t) + '`
} else if (evaluate) {
source += `'; ${evaluate} __p += '`
}
return match
})
source += "';"
source = `
let __t, __p = '';
const __j = Array.prototype.join;
const print = function print () { __p += __j.call(arguments, ''); };
with (data || {}) { ${source} }
return __p;
`
try {
// eslint-disable-next-line no-new-func
return new Function('data', 'escape', source).call(this, data, escape)
} catch (err) {
err.source = source
throw err
}
}
// =============================================================================
// User Script Configuration
// =============================================================================
/**
* @typedef {Object} Config
* @property {boolean} setTLD
* @property {boolean} setHl
* @property {boolean} setGl
* @property {boolean} setCr
* @property {boolean} setLr
* @property {boolean} showFlags
* @property {Array<string>} userRegions
*/
/**
* @type {Config}
*/
const config = Object.seal({
setTLD: true,
setHl: true,
setGl: true,
setCr: false,
setLr: false,
showFlags: true,
userRegions: ['wt-wt', 'jp-ja', 'tw-zh', 'us-en']
})
/**
* @return {Promise<Config>}
*/
function loadConfig () {
return GM.getValue('config')
.then(value => {
try { return JSON.parse(value) } catch (err) { return {} }
})
.then(value => {
return Object.assign(config, value)
})
}
/**
* @return {Promise<Config>}
*/
function saveConfig () {
return GM.setValue('config', JSON.stringify(config))
}
// =============================================================================
// Search Regions
// =============================================================================
/**
* @typedef {Object} Region
* @property {string} id
* @property {string} name
* @property {string} [tld]
* @property {string} [country]
* @property {string} [lang]
*/
/**
* @type {ReadonlyArray<Region>}
*/
const regions = Object.freeze([
{id: 'wt-wt', name: 'All Regions', tld: 'com'},
{id: 'ar-es', name: 'Argentina', tld: 'com.ar', country: 'ar', lang: 'es'},
{id: 'au-en', name: 'Australia', tld: 'com.au', country: 'au', lang: 'en'},
{id: 'at-de', name: 'Austria', tld: 'at', country: 'at', lang: 'de'},
{id: 'be-fr', name: 'Belgium (fr)', tld: 'be', country: 'be', lang: 'fr'},
{id: 'be-nl', name: 'Belgium (nl)', tld: 'be', country: 'be', lang: 'nl'},
{id: 'br-pt', name: 'Brazil', tld: 'com.br', country: 'br', lang: 'pt'},
{id: 'bg-bg', name: 'Bulgaria', tld: 'bg', country: 'bg', lang: 'bg'},
{id: 'ca-en', name: 'Canada', tld: 'ca', country: 'ca', lang: 'en'},
{id: 'ca-fr', name: 'Canada (fr)', tld: 'ca', country: 'ca', lang: 'fr'},
{id: 'ct-ca', name: 'Catalonia', tld: 'cat', country: 'ct', lang: 'ca'},
{id: 'cl-es', name: 'Chile', tld: 'cl', country: 'cl', lang: 'es'},
{id: 'cn-zh', name: 'China', tld: 'com.hk', country: 'cn', lang: 'zh-cn'},
{id: 'co-es', name: 'Colombia', tld: 'com.co', country: 'co', lang: 'es'},
{id: 'hr-hr', name: 'Croatia', tld: 'hr', country: 'hr', lang: 'hr'},
{id: 'cz-cs', name: 'Czech Republic', tld: 'cz', country: 'cz', lang: 'cs'},
{id: 'dk-da', name: 'Denmark', tld: 'dk', country: 'dk', lang: 'da'},
{id: 'ee-et', name: 'Estonia', tld: 'ee', country: 'ee', lang: 'et'},
{id: 'fi-fi', name: 'Finland', tld: 'fi', country: 'fi', lang: 'fi'},
{id: 'fr-fr', name: 'France', tld: 'fr', country: 'fr', lang: 'fr'},
{id: 'de-de', name: 'Germany', tld: 'de', country: 'de', lang: 'de'},
{id: 'gr-el', name: 'Greece', tld: 'gr', country: 'gr', lang: 'el'},
{id: 'hk-zh', name: 'Hong Kong', tld: 'com.hk', country: 'hk', lang: 'zh-hk'},
{id: 'hu-hu', name: 'Hungary', tld: 'hu', country: 'hu', lang: 'hu'},
{id: 'in-en', name: 'India', tld: 'co.in', country: 'in', lang: 'en'},
{id: 'id-id', name: 'Indonesia', tld: 'co.id', country: 'id', lang: 'id'},
{id: 'id-en', name: 'Indonesia (en)', tld: 'co.id', country: 'id', lang: 'en'},
{id: 'ie-en', name: 'Ireland', tld: 'ie', country: 'ie', lang: 'en'},
{id: 'il-he', name: 'Israel', tld: 'co.il', country: 'il', lang: 'he'},
{id: 'it-it', name: 'Italy', tld: 'it', country: 'it', lang: 'it'},
{id: 'jp-ja', name: 'Japan', tld: 'co.jp', country: 'jp', lang: 'ja'},
{id: 'kr-ko', name: 'Korea', tld: 'co.kr', country: 'kr', lang: 'ko'},
{id: 'lv-lv', name: 'Latvia', tld: 'lv', country: 'lv', lang: 'lv'},
{id: 'lt-lt', name: 'Lithuania', tld: 'lt', country: 'lt', lang: 'lt'},
{id: 'my-ms', name: 'Malaysia', tld: 'com.my', country: 'my', lang: 'ms'},
{id: 'my-en', name: 'Malaysia (en)', tld: 'com.my', country: 'my', lang: 'en'},
{id: 'mx-es', name: 'Mexico', tld: 'mx', country: 'mx', lang: 'es'},
{id: 'nl-nl', name: 'Netherlands', tld: 'nl', country: 'nl', lang: 'nl'},
{id: 'nz-en', name: 'New Zealand', tld: 'co.nz', country: 'nz', lang: 'en'},
{id: 'no-no', name: 'Norway', tld: 'no', country: 'no', lang: 'no'},
{id: 'pe-es', name: 'Peru', tld: 'com.pe', country: 'pe', lang: 'es'},
{id: 'ph-en', name: 'Philippines', tld: 'com.ph', country: 'ph', lang: 'en'},
{id: 'ph-tl', name: 'Philippines (tl)', tld: 'com.ph', country: 'ph', lang: 'tl'},
{id: 'pl-pl', name: 'Poland', tld: 'pl', country: 'pl', lang: 'pl'},
{id: 'pt-pt', name: 'Portugal', tld: 'pt', country: 'pt', lang: 'pt'},
{id: 'ro-ro', name: 'Romania', tld: 'ro', country: 'ro', lang: 'ro'},
{id: 'ru-ru', name: 'Russia', tld: 'ru', country: 'ru', lang: 'ru'},
{id: 'sa-ar', name: 'Saudi Arabia', tld: 'com.sa', country: 'sa', lang: 'ar'},
{id: 'sg-en', name: 'Singapore', tld: 'com.sg', country: 'sg', lang: 'en'},
{id: 'sk-sk', name: 'Slovakia', tld: 'sk', country: 'sk', lang: 'sk'},
{id: 'sl-sl', name: 'Slovenia', tld: 'si', country: 'sl', lang: 'sl'},
{id: 'za-en', name: 'South Africa', tld: 'co.za', country: 'za', lang: 'en'},
{id: 'es-es', name: 'Spain', tld: 'es', country: 'es', lang: 'es'},
{id: 'es-ca', name: 'Spain (ca)', tld: 'es', country: 'es', lang: 'ca'},
{id: 'se-sv', name: 'Sweden', tld: 'se', country: 'se', lang: 'sv'},
{id: 'ch-de', name: 'Switzerland (de)', tld: 'ch', country: 'ch', lang: 'de'},
{id: 'ch-fr', name: 'Switzerland (fr)', tld: 'ch', country: 'ch', lang: 'fr'},
{id: 'ch-it', name: 'Switzerland (it)', tld: 'ch', country: 'ch', lang: 'it'},
{id: 'tw-zh', name: 'Taiwan', tld: 'com.tw', country: 'tw', lang: 'zh-tw'},
{id: 'th-th', name: 'Thailand', tld: 'co.th', country: 'th', lang: 'th'},
{id: 'tr-tr', name: 'Turkey', tld: 'com.tr', country: 'tr', lang: 'tr'},
{id: 'gb-en', name: 'United Kingdom', tld: 'co.uk', country: 'gb', lang: 'en'},
{id: 'us-en', name: 'United States', tld: 'com', country: 'us', lang: 'en'},
{id: 'us-es', name: 'United States (es)', tld: 'com', country: 'us', lang: 'es'},
{id: 'vn-vi', name: 'Vietnam', tld: 'com.vn', country: 'vn', lang: 'vi'}
])
/**
* @param {Object} predicate
* @return {Region}
*/
function findRegion (predicate) {
return regions.find(region => {
return Object.keys(predicate).every(key => {
return predicate[key] === region[key]
})
})
}
/**
* @param {string} regionID
* @return {Region}
*/
function getRegionByID (regionID) {
return findRegion({ id: regionID })
}
const urlRegExp = Object.freeze({
tld: /^www\.google\.([\w.]+)$/i,
cr: /^country(\w+)$/i,
lr: /^lang_([\w-]+)$/i,
lang: /-\w+$/i
})
/**
* @return {Region}
*/
function getCurrentRegion () {
const { hostname, searchParams } = new window.URL(window.location.href)
const { setTLD, setHl, setGl, setCr, setLr } = config
const predicate = {}
if (setTLD && urlRegExp.tld.test(hostname)) {
predicate.tld = hostname.replace(urlRegExp.tld, '$1')
}
if (setHl && searchParams.has('hl')) {
predicate.lang = searchParams.get('hl')
}
if (setGl && searchParams.has('gl')) {
predicate.country = searchParams.get('gl')
}
if (setCr && searchParams.has('cr')) {
predicate.country = searchParams.get('cr').replace(urlRegExp.cr, '$1')
}
if (setLr && searchParams.has('lr')) {
predicate.lang = searchParams.get('lr').replace(urlRegExp.lr, '$1')
}
for (let prop in predicate) {
predicate[prop] = predicate[prop].toLowerCase()
}
return findRegion(predicate)
}
/**
* @type {ReadonlyArray<string>}
*/
const delParams = Object.freeze([
'aqs',
'bav',
'bih',
'biw',
'bvm',
'client',
'cp',
'dcr',
'dpr',
'dq',
'ech',
'ei',
'gfe_rd',
'gs_gbg',
'gs_l',
'gs_mss',
'gs_rn',
'gws_rd',
'oq',
'pbx',
'pf',
'pq',
'prds',
'psi',
'sa',
'safe',
'sclient',
'source',
'stick',
'ved'
])
/**
* @param {Region} region
* @return {string}
*/
function getSearchURL (region) {
const url = new window.URL(window.location.href)
const { hostname, searchParams } = url
const { setTLD, setHl, setGl, setCr, setLr } = config
const { tld, country, lang } = region
if (setTLD && tld) {
url.hostname = hostname.replace(urlRegExp.tld, `www.google.${tld}`)
} else if (urlRegExp.tld.test(url.hostname)) {
url.hostname = 'www.google.com'
}
if (setHl && lang) {
searchParams.set('hl', lang)
} else {
searchParams.delete('hl')
}
if (setGl && country) {
searchParams.set('gl', country)
} else {
searchParams.delete('gl')
}
if (setCr && country) {
searchParams.set('cr', `country${country.toUpperCase()}`)
} else {
searchParams.delete('cr')
}
if (setLr && lang) {
const lr = `lang_${lang.replace(urlRegExp.lang, m => m.toUpperCase())}`
searchParams.set('lr', lr)
} else {
searchParams.delete('lr')
}
delParams.forEach(param => {
searchParams.delete(param)
})
return url.toString()
}
// =============================================================================
// User Interface
// =============================================================================
/**
* @param {Element} target
*/
function createMenu (target) {
const currentRegion = getCurrentRegion()
const data = { config, regions, getRegionByID, getSearchURL, currentRegion }
const template = `
<% const { showFlags, userRegions } = config; %>
<span>
<g-popup>
<!-- Menu Dropdown Toggle -->
<div class="CcNe6e hide-focus-ring" aria-haspopup="true" role="button" tabindex="0">
<div class="hdtb-mn-hd gm-region-menu-toggle <%- currentRegion ? 'hdtb-sel' : '' %>" data-gm-region-onclick="toggleMenu">
<div class="mn-hd-txt KTBKoe" data-gm-region-onclick="toggleMenu">
<% if (currentRegion) { %>
<% let { name, country } = currentRegion; %>
<% if (country && showFlags) { %> <span class="flag flag-<%- country %>" data-gm-region-onclick="toggleMenu"></span> <% } %>
<%- name %>
<% } else { %>
Regions
<% } %>
</div>
<span class="mn-dwn-arw gTl8xb"></span>
</div>
</div>
<!-- Menu Dropdown -->
<div class="UjBGL pkWBse iRQHZe gm-region-menu-dropdown" style="display: none">
<g-menu class="cF4V5c Tlae9d yTik0 wplJBd PBn44e" role="menu" tabindex="-1">
<!-- User Regions List -->
<% userRegions.map(getRegionByID).forEach(region => { %>
<% if (!region) { return; } %>
<% let { id, name, country } = region; %>
<% let isCurrent = currentRegion && currentRegion.id === id; %>
<% let url = getSearchURL(region); %>
<g-menu-item class="EpPYLd hide-focus-ring <%- isCurrent ? 'nvELY' : '' %>">
<div class="YpcDnf OSrXXb HG1dvd">
<a href="<%- url %>" role="menuitem">
<% if (country && showFlags) { %> <span class="flag flag-<%- country %>"></span> <% } %>
<%- name %>
</a>
</div>
</g-menu-item>
<% }); %>
<!-- Configuration Modal Toggle -->
<g-menu-item class="ErsxPb hide-focus-ring">
<div class="znKVS tnhqA">
<a class="gm-region-menu-config" data-gm-region-onclick="showModal" title="Google Search Region">...</a>
</div>
</g-menu-item>
</g-menu>
</div>
</g-popup>
</span>
`
const html = renderTemplate(template, data)
target.insertAdjacentHTML('afterend', html)
}
/**
* @param {Element} target
*/
function createModal (target) {
const data = { config, regions }
const template = `
<% const { setTLD, setHl, setGl, setCr, setLr, showFlags, userRegions } = config; %>
<!-- Configuration Modal -->
<div class="gm-region-modal" data-gm-region-onclick="hideModal">
<!-- Modal Dialog -->
<div class="gm-region-modal-dialog">
<!-- Modal Header -->
<div class="gm-region-modal-header">
<div class="gm-region-modal-title">Google Search Region</div>
<div class="gm-region-modal-close" role="button" aria-label="Close" data-gm-region-onclick="hideModal"></div>
</div>
<!-- Modal Body -->
<div class="gm-region-modal-body">
<!-- Menu Configuration -->
<div class="gm-region-modal-subtitle">Menu</div>
<!-- config.showFlags -->
<label class="gm-region-control">
<input class="gm-region-control-input" type="checkbox" data-gm-region-config="showFlags" <%- showFlags ? 'checked' : '' %>>
<span class="gm-region-control-indicator"></span>
<span class="gm-region-control-description">Show country flags</span>
</label>
<!-- URL Configuration -->
<div class="gm-region-modal-subtitle">URL</div>
<!-- config.setTLD -->
<label class="gm-region-control">
<input class="gm-region-control-input" type="checkbox" data-gm-region-config="setTLD" <%- setTLD ? 'checked' : '' %>>
<span class="gm-region-control-indicator"></span>
<span class="gm-region-control-description">Set top level domain</span>
</label>
<!-- config.setHl -->
<label class="gm-region-control">
<input class="gm-region-control-input" type="checkbox" data-gm-region-config="setHl" <%- setHl ? 'checked' : '' %>>
<span class="gm-region-control-indicator"></span>
<span class="gm-region-control-description">Set host language (hl)</span>
</label>
<!-- config.setGl -->
<label class="gm-region-control">
<input class="gm-region-control-input" type="checkbox" data-gm-region-config="setGl" <%- setGl ? 'checked' : '' %>>
<span class="gm-region-control-indicator"></span>
<span class="gm-region-control-description">Set region (gl)</span>
</label>
<!-- config.setCr -->
<label class="gm-region-control">
<input class="gm-region-control-input" type="checkbox" data-gm-region-config="setCr" <%- setCr ? 'checked' : '' %>>
<span class="gm-region-control-indicator"></span>
<span class="gm-region-control-description">Set country filter (cr)</span>
</label>
<!-- config.setLr -->
<label class="gm-region-control">
<input class="gm-region-control-input" type="checkbox" data-gm-region-config="setLr" <%- setLr ? 'checked' : '' %>>
<span class="gm-region-control-indicator"></span>
<span class="gm-region-control-description">Set language filter (lr)</span>
</label>
<!-- Regions Configuration -->
<div class="gm-region-modal-subtitle">Regions</div>
<div class="gm-region-columns">
<!-- config.userRegions -->
<% regions.forEach(region => { %>
<% let { id, name, country } = region; %>
<% let isChecked = userRegions.includes(id); %>
<label class="gm-region-control" title="<%- name %>">
<input class="gm-region-control-input" type="checkbox"
data-gm-region-config="userRegions:<%- id %>" <%- isChecked ? 'checked' : '' %>>
<span class="gm-region-control-indicator"></span>
<span class="gm-region-control-description">
<% if (country) { %> <span class="flag flag-<%- country %>"></span> <% } %>
<%- name %>
</span>
</label>
<% }); %>
</div>
</div>
<!-- Modal Footer -->
<div class="gm-region-modal-footer">
<button class="gm-region-btn gm-region-btn-primary" data-gm-region-onclick="save">Save</button>
<button class="gm-region-btn gm-region-btn-default" data-gm-region-onclick="hideModal">Cancel</button>
</div>
</div>
</div>
`
const html = renderTemplate(template, data)
target.insertAdjacentHTML('beforeend', html)
}
/**
* @return {Promise<void>}
*/
function delegateEvents () {
const body = document.body
const events = {}
events.showModal = function showModal () {
const modal = $('.gm-region-modal')
if (modal) { modal.style.display = null } else { createModal(body) }
}
events.hideModal = function hideModal () {
const modal = $('.gm-region-modal')
if (modal) { modal.style.display = 'none' }
}
events.toggleMenu = function toggleMenu () {
const menu = $('.gm-region-menu-dropdown')
const toggle = $('.gm-region-menu-toggle')
if (menu) {
if (menu.style.display === 'none') {
menu.style.display = 'block'
menu.style.left = toggle.getBoundingClientRect().left + 'px'
} else {
menu.style.display = 'none'
}
}
}
events.save = function save () {
const modal = $('.gm-region-modal')
const controls = $$('[data-gm-region-config]', modal)
const pending = {}
controls.forEach(control => {
const attr = control.getAttribute('data-gm-region-config').split(':')
const [name, value = control.value] = attr
if (typeof config[name] === 'boolean') {
pending[name] = control.checked
}
if (Array.isArray(config[name])) {
if (!Array.isArray(pending[name])) { pending[name] = [] }
if (control.checked) { pending[name].push(value) }
}
})
Object.assign(config, pending)
saveConfig().then(() => {
window.location.reload()
})
}
$delegate(body, '[data-gm-region-onclick]', 'click', event => {
const name = event.target.getAttribute('data-gm-region-onclick')
const callback = events[name]
if (callback) { callback.call(event.target, event) }
})
return Promise.resolve()
}
/**
* @return {Promise<HTMLStyleElement>}
*/
function addStyles () {
addStyle(`
/*!
* Region Menu Dropdown CSS
*/
.hdtb-sel{font-weight:700}
.gm-region-menu-toggle{margin-inline-end:32px}
.gm-region-menu-dropdown{display:none;position:absolute;max-height:80vh;overflow-y:auto;z-index:200}
.gm-region-menu-dropdown-show{display:block}
.gm-region-menu-dropdown .hdtbItm.hdtbSel{padding:0}
.gm-region-menu-dropdown .hdtbItm.hdtbSel a{background-color:transparent}
.gm-region-menu-config{cursor:pointer}
.gm-region-menu-dropdown g-menu-item:hover{background-color:rgba(0,0,0,.1)}
/*!
* Configuration Modal CSS
*/
.gm-region-modal{display:flex;align-items:center;justify-content:center;position:fixed;z-index:10000;top:0;left:0;width:100%;height:100%;background-color:rgba(255,255,255,.75);color:#4d5156}
.gm-region-modal-dialog{display:block;width:800px;max-width:80vw;max-height:80vh;overflow:auto;margin:32px;padding:32px;border:1px solid #c5c5c5;box-shadow:0 4px 16px rgba(0,0,0,.2);background-color:#fff;font-size:13px}
.gm-region-modal-header{display:flex;justify-content:space-between}
.gm-region-modal-footer{text-align:right}
.gm-region-modal-body{margin:16px 0}
.gm-region-modal-title{font-size:16px;font-weight:thin}
.gm-region-modal-subtitle{margin:16px 0;font-size:13px;font-weight:700}
.gm-region-modal-close{display:inline-block;width:10px;height:10px;background-image:url('');background-repeat:no-repeat;cursor:pointer}
.gm-region-columns{max-height:300px;overflow-x:auto;-webkit-column-count:5;-moz-column-count:5;column-count:5}
.gm-region-control{display:block;margin:4px 0;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}
.gm-region-control-input{display:none}
.gm-region-control-indicator{display:inline-block;margin:0 4px;width:10px;height:10px;border:1px solid #c6c6c6;border-radius:1px;vertical-align:middle}
.gm-region-control-indicator::after{content:" ";display:none;position:relative;top:-3px;width:15px;height:15px;background-image:url('');background-repeat:no-repeat;background-position:-5px -3px}
.gm-region-control:hover .gm-region-control-indicator{border-color:#b2b2b2;box-shadow:inset 0 1px 1px rgba(0,0,0,.1)}
.gm-region-control-input:checked~.gm-region-control-indicator::after{display:inline-block}
.gm-region-btn{display:inline-block;min-width:70px;height:27px;padding:0 8px;border:1px solid;border-radius:2px;font-family:inherit;font-size:11px;font-weight:700;outline:0}
.gm-region-btn-default{border-color:rgba(0,0,0,.1);background-image:linear-gradient(#f5f5f5,#f1f1f1);color:#444}
.gm-region-btn-default:hover{border-color:#c6c6c6;background-image:linear-gradient(#f8f8f8,#f1f1f1);color:#333}
.gm-region-btn-default:focus{border-color:#4d90fe}
.gm-region-btn-primary{border-color:#3079ed;background-image:linear-gradient(#4d90fe,#4787ed);color:#fff}
.gm-region-btn-primary:hover{border-color:#2f5bb7;background-image:linear-gradient(#4d90fe,#357ae8);color:#fff}
.gm-region-btn-primary:focus{border-color:transparent;box-shadow:inset 0 0 0 1px #fff}
/*!
* Generated with CSS Flag Sprite Generator <https://www.flag-sprites.com/>
*
* FAMFAMFAM Flag Icons <http://www.famfamfam.com/lab/icons/flags/>
* These flag icons are available for free use for any purpose with no
* requirement for attribution.
*/
.flag{display:inline-block;width:16px;height:11px;background:url('') no-repeat;image-rendering:-moz-crisp-edges;image-rendering:crisp-edges;image-rendering:pixelated;vertical-align:middle}
.flag.flag-ar{background-position:0 0}
.flag.flag-at{background-position:-16px 0}
.flag.flag-au{background-position:-32px 0}
.flag.flag-be{background-position:-48px 0}
.flag.flag-bg{background-position:-64px 0}
.flag.flag-br{background-position:-80px 0}
.flag.flag-ca{background-position:-96px 0}
.flag.flag-ct{background-position:-112px 0}
.flag.flag-ch{background-position:0 -11px}
.flag.flag-cl{background-position:-16px -11px}
.flag.flag-cn{background-position:-32px -11px}
.flag.flag-co{background-position:-48px -11px}
.flag.flag-cz{background-position:-64px -11px}
.flag.flag-de{background-position:-80px -11px}
.flag.flag-dk{background-position:-96px -11px}
.flag.flag-ee{background-position:-112px -11px}
.flag.flag-es{background-position:0 -22px}
.flag.flag-fi{background-position:-16px -22px}
.flag.flag-fr{background-position:-32px -22px}
.flag.flag-gb{background-position:-48px -22px}
.flag.flag-gr{background-position:-64px -22px}
.flag.flag-hk{background-position:-80px -22px}
.flag.flag-hr{background-position:-96px -22px}
.flag.flag-hu{background-position:-112px -22px}
.flag.flag-id{background-position:0 -33px}
.flag.flag-ie{background-position:-16px -33px}
.flag.flag-il{background-position:-32px -33px}
.flag.flag-in{background-position:-48px -33px}
.flag.flag-it{background-position:-64px -33px}
.flag.flag-jp{background-position:-80px -33px}
.flag.flag-kr{background-position:-96px -33px}
.flag.flag-lt{background-position:-112px -33px}
.flag.flag-lv{background-position:0 -44px}
.flag.flag-mx{background-position:-16px -44px}
.flag.flag-my{background-position:-32px -44px}
.flag.flag-nl{background-position:-48px -44px}
.flag.flag-no{background-position:-64px -44px}
.flag.flag-nz{background-position:-80px -44px}
.flag.flag-pe{background-position:-96px -44px}
.flag.flag-ph{background-position:-112px -44px}
.flag.flag-pl{background-position:0 -55px}
.flag.flag-pt{background-position:-16px -55px}
.flag.flag-ro{background-position:-32px -55px}
.flag.flag-ru{background-position:-48px -55px}
.flag.flag-sa{background-position:-64px -55px}
.flag.flag-se{background-position:-80px -55px}
.flag.flag-sg{background-position:-96px -55px}
.flag.flag-sk{background-position:-112px -55px}
.flag.flag-sl{background-position:0 -66px}
.flag.flag-th{background-position:-16px -66px}
.flag.flag-tr{background-position:-32px -66px}
.flag.flag-tw{background-position:-48px -66px}
.flag.flag-us{background-position:-64px -66px}
.flag.flag-vn{background-position:-80px -66px}
.flag.flag-za{background-position:-96px -66px}
`)
if (
window
.getComputedStyle(document.body)
.getPropertyValue('background-color') !== 'rgb(255, 255, 255)'
) {
addStyle(`
/*!
* Configuration Modal CSS (Dark Theme)
*/
.gm-region-modal{background-color:rgba(32,33,36,.75);color:#bdc1c6}
.gm-region-modal-dialog{box-shadow:0 4px 16px rgba(0,0,0,.8);border:1px solid #313437;background:#202124}
.gm-region-modal-close{filter:invert(100%)}
.gm-region-control-indicator{border:1px solid #5f6368}
.gm-region-control-indicator::after{filter:invert(100%)}
`)
}
return Promise.resolve(true)
}
// =============================================================================
// Initialization
// =============================================================================
/**
* @return {Promise<Element>}
*/
function waitForPageReady () {
return new Promise(resolve => {
const observee = $('#hdtb')
const observer = new MutationObserver(() => {
const target = $('#hdtbMenus > #tn_1 > div:first-child')
if (target) {
resolve(target)
observer.disconnect()
}
})
observer.observe(observee, { childList: true, subtree: true })
})
}
Promise.all([
waitForPageReady(),
loadConfig(),
delegateEvents(),
addStyles()
]).then(values => createMenu(values[0]))