Google Search Region

A user script that lets you quickly switch Google search to different region.

当前为 2018-09-12 提交的版本,查看 最新版本

  1. // ==UserScript==
  2. // @name Google Search Region
  3. // @namespace jmln.tw
  4. // @version 0.2.1
  5. // @description A user script that lets you quickly switch Google search to different region.
  6. // @author Jimmy Lin
  7. // @license MIT
  8. // @homepage https://github.com/jmlntw/google-search-region
  9. // @supportURL https://github.com/jmlntw/google-search-region/issues
  10. // @include https://www.google.*/search?*
  11. // @include https://www.google.*/webhp?*
  12. // @include https://encrypted.google.com/search?*
  13. // @include https://encrypted.google.com/webhp?*
  14. // @compatible firefox
  15. // @compatible chrome
  16. // @compatible opera
  17. // @run-at document-end
  18. // @grant GM_getValue
  19. // @grant GM_setValue
  20. // @grant GM.getValue
  21. // @grant GM.setValue
  22. // ==/UserScript==
  23.  
  24. // =============================================================================
  25. // Add compatibility between the Greasemonkey 4 APIs and existing/legacy APIs.
  26. // =============================================================================
  27.  
  28. if (typeof GM === 'undefined') {
  29. // eslint-disable-next-line no-global-assign
  30. GM = {
  31. getValue: (...args) => Promise.resolve(GM_getValue.apply(this, args)),
  32. setValue: (...args) => Promise.resolve(GM_setValue.apply(this, args))
  33. }
  34. }
  35.  
  36. function GM_addStyle (css) {
  37. const style = document.createElement('style')
  38. style.type = 'text/css'
  39. style.textContent = css
  40. document.head.appendChild(style)
  41. return style
  42. }
  43. GM.addStyle = GM_addStyle
  44.  
  45. // =============================================================================
  46. // Helper Functions
  47. // =============================================================================
  48.  
  49. /**
  50. * @param {string} selector
  51. * @param {Element} [context]
  52. * @return {Element}
  53. */
  54. function $ (selector, context) {
  55. return (context || document).querySelector(selector)
  56. }
  57.  
  58. /**
  59. * @param {string} selector
  60. * @param {Element} [context]
  61. * @return {NodeListOf<Element>}
  62. */
  63. function $$ (selector, context) {
  64. return (context || document).querySelectorAll(selector)
  65. }
  66.  
  67. /**
  68. * @param {Element} target
  69. * @param {string} type
  70. * @param {EventListener} callback
  71. * @param {boolean} [useCapture]
  72. */
  73. function $on (target, type, callback, useCapture) {
  74. target.addEventListener(type, callback, !!useCapture)
  75. }
  76.  
  77. /**
  78. * @param {Element} target
  79. * @param {string} selector
  80. * @param {string} type
  81. * @param {EventListener} callback
  82. */
  83. function $delegate (target, selector, type, callback) {
  84. const useCapture = (type === 'blur') || (type === 'focus')
  85. const dispatchEvent = function dispatchEvent (event) {
  86. if (event.target.matches(selector)) { callback.call(event.target, event) }
  87. }
  88.  
  89. $on(target, type, dispatchEvent, useCapture)
  90. }
  91.  
  92. if (window.NodeList && !window.NodeList.prototype.forEach) {
  93. window.NodeList.prototype.forEach = Array.prototype.forEach
  94. }
  95.  
  96. // =============================================================================
  97. // Template Engine
  98. // =============================================================================
  99.  
  100. /**
  101. * @param {string} text
  102. * @param {Object} data
  103. * @return {string}
  104. */
  105. function renderTemplate (text, data) {
  106. const matcher = /<%-([\s\S]+?)%>|<%=([\s\S]+?)%>|<%([\s\S]+?)%>|$/g
  107. const escapeChar = function escapeChar (text) {
  108. return text
  109. .replace(/\\/g, '\\\\')
  110. .replace(/'/g, "\\'")
  111. .replace(/\r/g, '\\r')
  112. .replace(/\n/g, '\\n')
  113. .replace(/\u2028/g, '\\u2028')
  114. .replace(/\u2029/g, '\\u2029')
  115. }
  116. const escape = function escape (text) {
  117. return ('' + text)
  118. .replace(/&/g, '&amp;')
  119. .replace(/</g, '&lt;')
  120. .replace(/>/g, '&gt;')
  121. .replace(/"/g, '&quot;')
  122. .replace(/'/g, '&#x27;')
  123. .replace(/`/g, '&#x60;')
  124. }
  125.  
  126. let index = 0
  127. let source = "__p += '"
  128.  
  129. text.replace(matcher, (match, escape, interpolate, evaluate, offset) => {
  130. source += escapeChar(text.slice(index, offset))
  131. index = offset + match.length
  132. if (escape) {
  133. source += `' + ((__t = (${escape})) == null ? '' : escape(__t)) + '`
  134. } else if (interpolate) {
  135. source += `' + ((__t = (${interpolate})) == null ? '' : __t) + '`
  136. } else if (evaluate) {
  137. source += `'; ${evaluate} __p += '`
  138. }
  139. return match
  140. })
  141.  
  142. source += "';"
  143. source = `
  144. let __t, __p = '';
  145. const __j = Array.prototype.join;
  146. const print = function print () { __p += __j.call(arguments, ''); };
  147. with (data || {}) { ${source} }
  148. return __p;
  149. `
  150.  
  151. try {
  152. // eslint-disable-next-line no-new-func
  153. return new Function('data', 'escape', source).call(this, data, escape)
  154. } catch (err) {
  155. err.source = source
  156. throw err
  157. }
  158. }
  159.  
  160. // =============================================================================
  161. // User Script Configuration
  162. // =============================================================================
  163.  
  164. /**
  165. * @typedef {Object} Config
  166. * @property {boolean} setTLD
  167. * @property {boolean} setHl
  168. * @property {boolean} setGl
  169. * @property {boolean} setCr
  170. * @property {boolean} setLr
  171. * @property {boolean} showFlags
  172. * @property {Array<string>} userRegions
  173. */
  174.  
  175. /**
  176. * @type {Config}
  177. */
  178. const config = Object.seal({
  179. setTLD: true,
  180. setHl: true,
  181. setGl: true,
  182. setCr: false,
  183. setLr: false,
  184. showFlags: true,
  185. userRegions: ['wt-wt', 'jp-ja', 'tw-zh', 'us-en']
  186. })
  187.  
  188. /**
  189. * @return {Promise<Config>}
  190. */
  191. function loadConfig () {
  192. return GM.getValue('config')
  193. .then(value => {
  194. try { return JSON.parse(value) } catch (err) { return {} }
  195. })
  196. .then(value => {
  197. return Object.assign(config, value)
  198. })
  199. }
  200.  
  201. /**
  202. * @return {Promise<Config>}
  203. */
  204. function saveConfig () {
  205. return GM.setValue('config', JSON.stringify(config))
  206. }
  207.  
  208. // =============================================================================
  209. // Search Regions
  210. // =============================================================================
  211.  
  212. /**
  213. * @typedef {Object} Region
  214. * @property {string} id
  215. * @property {string} name
  216. * @property {string} [tld]
  217. * @property {string} [country]
  218. * @property {string} [lang]
  219. */
  220.  
  221. /**
  222. * @type {ReadonlyArray<Region>}
  223. */
  224. const regions = Object.freeze([
  225. {id: 'wt-wt', name: 'All Regions', tld: 'com'},
  226. {id: 'ar-es', name: 'Argentina', tld: 'com.ar', country: 'ar', lang: 'es'},
  227. {id: 'au-en', name: 'Australia', tld: 'com.au', country: 'au', lang: 'en'},
  228. {id: 'at-de', name: 'Austria', tld: 'at', country: 'at', lang: 'de'},
  229. {id: 'be-fr', name: 'Belgium (fr)', tld: 'be', country: 'be', lang: 'fr'},
  230. {id: 'be-nl', name: 'Belgium (nl)', tld: 'be', country: 'be', lang: 'nl'},
  231. {id: 'br-pt', name: 'Brazil', tld: 'com.br', country: 'br', lang: 'pt'},
  232. {id: 'bg-bg', name: 'Bulgaria', tld: 'bg', country: 'bg', lang: 'bg'},
  233. {id: 'ca-en', name: 'Canada', tld: 'ca', country: 'ca', lang: 'en'},
  234. {id: 'ca-fr', name: 'Canada (fr)', tld: 'ca', country: 'ca', lang: 'fr'},
  235. {id: 'ct-ca', name: 'Catalonia', tld: 'cat', country: 'ct', lang: 'ca'},
  236. {id: 'cl-es', name: 'Chile', tld: 'cl', country: 'cl', lang: 'es'},
  237. {id: 'cn-zh', name: 'China', tld: 'com.hk', country: 'cn', lang: 'zh-cn'},
  238. {id: 'co-es', name: 'Colombia', tld: 'com.co', country: 'co', lang: 'es'},
  239. {id: 'hr-hr', name: 'Croatia', tld: 'hr', country: 'hr', lang: 'hr'},
  240. {id: 'cz-cs', name: 'Czech Republic', tld: 'cz', country: 'cz', lang: 'cs'},
  241. {id: 'dk-da', name: 'Denmark', tld: 'dk', country: 'dk', lang: 'da'},
  242. {id: 'ee-et', name: 'Estonia', tld: 'ee', country: 'ee', lang: 'et'},
  243. {id: 'fi-fi', name: 'Finland', tld: 'fi', country: 'fi', lang: 'fi'},
  244. {id: 'fr-fr', name: 'France', tld: 'fr', country: 'fr', lang: 'fr'},
  245. {id: 'de-de', name: 'Germany', tld: 'de', country: 'de', lang: 'de'},
  246. {id: 'gr-el', name: 'Greece', tld: 'gr', country: 'gr', lang: 'el'},
  247. {id: 'hk-zh', name: 'Hong Kong', tld: 'com.hk', country: 'hk', lang: 'zh-hk'},
  248. {id: 'hu-hu', name: 'Hungary', tld: 'hu', country: 'hu', lang: 'hu'},
  249. {id: 'in-en', name: 'India', tld: 'co.in', country: 'in', lang: 'en'},
  250. {id: 'id-id', name: 'Indonesia', tld: 'co.id', country: 'id', lang: 'id'},
  251. {id: 'id-en', name: 'Indonesia (en)', tld: 'co.id', country: 'id', lang: 'en'},
  252. {id: 'ie-en', name: 'Ireland', tld: 'ie', country: 'ie', lang: 'en'},
  253. {id: 'il-he', name: 'Israel', tld: 'co.il', country: 'il', lang: 'he'},
  254. {id: 'it-it', name: 'Italy', tld: 'it', country: 'it', lang: 'it'},
  255. {id: 'jp-ja', name: 'Japan', tld: 'co.jp', country: 'jp', lang: 'ja'},
  256. {id: 'kr-ko', name: 'Korea', tld: 'co.kr', country: 'kr', lang: 'ko'},
  257. {id: 'lv-lv', name: 'Latvia', tld: 'lv', country: 'lv', lang: 'lv'},
  258. {id: 'lt-lt', name: 'Lithuania', tld: 'lt', country: 'lt', lang: 'lt'},
  259. {id: 'my-ms', name: 'Malaysia', tld: 'com.my', country: 'my', lang: 'ms'},
  260. {id: 'my-en', name: 'Malaysia (en)', tld: 'com.my', country: 'my', lang: 'en'},
  261. {id: 'mx-es', name: 'Mexico', tld: 'mx', country: 'mx', lang: 'es'},
  262. {id: 'nl-nl', name: 'Netherlands', tld: 'nl', country: 'nl', lang: 'nl'},
  263. {id: 'nz-en', name: 'New Zealand', tld: 'co.nz', country: 'nz', lang: 'en'},
  264. {id: 'no-no', name: 'Norway', tld: 'no', country: 'no', lang: 'no'},
  265. {id: 'pe-es', name: 'Peru', tld: 'com.pe', country: 'pe', lang: 'es'},
  266. {id: 'ph-en', name: 'Philippines', tld: 'com.ph', country: 'ph', lang: 'en'},
  267. {id: 'ph-tl', name: 'Philippines (tl)', tld: 'com.ph', country: 'ph', lang: 'tl'},
  268. {id: 'pl-pl', name: 'Poland', tld: 'pl', country: 'pl', lang: 'pl'},
  269. {id: 'pt-pt', name: 'Portugal', tld: 'pt', country: 'pt', lang: 'pt'},
  270. {id: 'ro-ro', name: 'Romania', tld: 'ro', country: 'ro', lang: 'ro'},
  271. {id: 'ru-ru', name: 'Russia', tld: 'ru', country: 'ru', lang: 'ru'},
  272. {id: 'sa-ar', name: 'Saudi Arabia', tld: 'com.sa', country: 'sa', lang: 'ar'},
  273. {id: 'sg-en', name: 'Singapore', tld: 'com.sg', country: 'sg', lang: 'en'},
  274. {id: 'sk-sk', name: 'Slovakia', tld: 'sk', country: 'sk', lang: 'sk'},
  275. {id: 'sl-sl', name: 'Slovenia', tld: 'si', country: 'sl', lang: 'sl'},
  276. {id: 'za-en', name: 'South Africa', tld: 'co.za', country: 'za', lang: 'en'},
  277. {id: 'es-es', name: 'Spain', tld: 'es', country: 'es', lang: 'es'},
  278. {id: 'es-ca', name: 'Spain (ca)', tld: 'es', country: 'es', lang: 'ca'},
  279. {id: 'se-sv', name: 'Sweden', tld: 'se', country: 'se', lang: 'sv'},
  280. {id: 'ch-de', name: 'Switzerland (de)', tld: 'ch', country: 'ch', lang: 'de'},
  281. {id: 'ch-fr', name: 'Switzerland (fr)', tld: 'ch', country: 'ch', lang: 'fr'},
  282. {id: 'ch-it', name: 'Switzerland (it)', tld: 'ch', country: 'ch', lang: 'it'},
  283. {id: 'tw-zh', name: 'Taiwan', tld: 'com.tw', country: 'tw', lang: 'zh-tw'},
  284. {id: 'th-th', name: 'Thailand', tld: 'co.th', country: 'th', lang: 'th'},
  285. {id: 'tr-tr', name: 'Turkey', tld: 'com.tr', country: 'tr', lang: 'tr'},
  286. {id: 'gb-en', name: 'United Kingdom', tld: 'co.uk', country: 'gb', lang: 'en'},
  287. {id: 'us-en', name: 'United States', tld: 'com', country: 'us', lang: 'en'},
  288. {id: 'us-es', name: 'United States (es)', tld: 'com', country: 'us', lang: 'es'},
  289. {id: 'vn-vi', name: 'Vietnam', tld: 'com.vn', country: 'vn', lang: 'vi'}
  290. ])
  291.  
  292. /**
  293. * @param {Object} predicate
  294. * @return {Region}
  295. */
  296. function findRegion (predicate) {
  297. return regions.find(region => {
  298. return Object.keys(predicate).every(key => {
  299. return predicate[key] === region[key]
  300. })
  301. })
  302. }
  303.  
  304. /**
  305. * @param {string} regionID
  306. * @return {Region}
  307. */
  308. function getRegionByID (regionID) {
  309. return findRegion({ id: regionID })
  310. }
  311.  
  312. const urlRegExp = Object.freeze({
  313. tld: /^www\.google\.([\w.]+)$/i,
  314. cr: /^country(\w+)$/i,
  315. lr: /^lang_([\w-]+)$/i,
  316. lang: /-\w+$/i
  317. })
  318.  
  319. /**
  320. * @return {Region}
  321. */
  322. function getCurrentRegion () {
  323. const { hostname, searchParams } = new window.URL(window.location.href)
  324. const { setTLD, setHl, setGl, setCr, setLr } = config
  325. const predicate = {}
  326.  
  327. if (setTLD && urlRegExp.tld.test(hostname)) {
  328. predicate.tld = hostname.replace(urlRegExp.tld, '$1')
  329. }
  330. if (setHl && searchParams.has('hl')) {
  331. predicate.lang = searchParams.get('hl')
  332. }
  333. if (setGl && searchParams.has('gl')) {
  334. predicate.country = searchParams.get('gl')
  335. }
  336. if (setCr && searchParams.has('cr')) {
  337. predicate.country = searchParams.get('cr').replace(urlRegExp.cr, '$1')
  338. }
  339. if (setLr && searchParams.has('lr')) {
  340. predicate.lang = searchParams.get('lr').replace(urlRegExp.lr, '$1')
  341. }
  342.  
  343. for (let prop in predicate) {
  344. predicate[prop] = predicate[prop].toLowerCase()
  345. }
  346.  
  347. return findRegion(predicate)
  348. }
  349.  
  350. /**
  351. * @type {ReadonlyArray<string>}
  352. */
  353. const delParams = Object.freeze([
  354. 'aqs',
  355. 'bav',
  356. 'bih',
  357. 'biw',
  358. 'bvm',
  359. 'client',
  360. 'cp',
  361. 'dcr',
  362. 'dpr',
  363. 'dq',
  364. 'ech',
  365. 'ei',
  366. 'gfe_rd',
  367. 'gs_gbg',
  368. 'gs_l',
  369. 'gs_mss',
  370. 'gs_rn',
  371. 'gws_rd',
  372. 'oq',
  373. 'pbx',
  374. 'pf',
  375. 'pq',
  376. 'prds',
  377. 'psi',
  378. 'sa',
  379. 'safe',
  380. 'sclient',
  381. 'source',
  382. 'stick',
  383. 'ved'
  384. ])
  385.  
  386. /**
  387. * @param {Region} region
  388. * @return {string}
  389. */
  390. function getSearchURL (region) {
  391. const url = new window.URL(window.location.href)
  392. const { hostname, searchParams } = url
  393. const { setTLD, setHl, setGl, setCr, setLr } = config
  394. const { tld, country, lang } = region
  395.  
  396. if (setTLD && tld) {
  397. url.hostname = hostname.replace(urlRegExp.tld, `www.google.${tld}`)
  398. } else if (urlRegExp.tld.test(url.hostname)) {
  399. url.hostname = 'www.google.com'
  400. }
  401. if (setHl && lang) {
  402. searchParams.set('hl', lang)
  403. } else {
  404. searchParams.delete('hl')
  405. }
  406. if (setGl && country) {
  407. searchParams.set('gl', country)
  408. } else {
  409. searchParams.delete('gl')
  410. }
  411. if (setCr && country) {
  412. searchParams.set('cr', `country${country.toUpperCase()}`)
  413. } else {
  414. searchParams.delete('cr')
  415. }
  416. if (setLr && lang) {
  417. const lr = `lang_${lang.replace(urlRegExp.lang, m => m.toUpperCase())}`
  418. searchParams.set('lr', lr)
  419. } else {
  420. searchParams.delete('lr')
  421. }
  422.  
  423. delParams.forEach(param => {
  424. searchParams.delete(param)
  425. })
  426.  
  427. return url.toString()
  428. }
  429.  
  430. // =============================================================================
  431. // User Interface
  432. // =============================================================================
  433.  
  434. /**
  435. * @param {Element} target
  436. */
  437. function createMenu (target) {
  438. const currentRegion = getCurrentRegion()
  439. const data = { config, regions, getRegionByID, getSearchURL, currentRegion }
  440. const template = `
  441. <% const { showFlags, userRegions } = config; %>
  442.  
  443. <!-- Menu Dropdown Toggle -->
  444. <div class="hdtb-mn-hd gm-region-menu-toggle <%- currentRegion ? 'hdtb-sel' : '' %>" role="button">
  445. <div class="mn-hd-txt">
  446. <% if (currentRegion) { %>
  447. <% let { name, country } = currentRegion; %>
  448. <% if (country && showFlags) { %> <span class="flag flag-<%- country %>"></span> <% } %>
  449. <%- name %>
  450. <% } else { %>
  451. Regions
  452. <% } %>
  453. </div>
  454. <span class="mn-dwn-arw"></span>
  455. </div>
  456.  
  457. <!-- Menu Dropdown -->
  458. <ul class="hdtbU hdtb-mn-c gm-region-menu-dropdown">
  459. <!-- User Regions List -->
  460. <% userRegions.map(getRegionByID).forEach(region => { %>
  461. <% if (!region) { return; } %>
  462. <% let { id, name, country } = region; %>
  463. <% let isCurrent = currentRegion && currentRegion.id === id; %>
  464. <% let url = getSearchURL(region); %>
  465. <li class="hdtbItm <%- isCurrent ? 'hdtbSel' : '' %>">
  466. <a class="q qs" href="<%- url %>">
  467. <% if (country && showFlags) { %> <span class="flag flag-<%- country %>"></span> <% } %>
  468. <%- name %>
  469. </a>
  470. </li>
  471. <% }); %>
  472.  
  473. <!-- Configuration Modal Toggle -->
  474. <li class="hdtbItm">
  475. <div class="cdr_sep"></div>
  476. <a class="q qs gm-region-menu-config" data-gm-region-onclick="showModal" title="Google Search Region">...</a>
  477. </li>
  478. </ul>
  479. `
  480. const html = renderTemplate(template, data)
  481.  
  482. target.insertAdjacentHTML('afterend', html)
  483. }
  484.  
  485. /**
  486. * @param {Element} target
  487. */
  488. function createModal (target) {
  489. const data = { config, regions }
  490. const template = `
  491. <% const { setTLD, setHl, setGl, setCr, setLr, showFlags, userRegions } = config; %>
  492.  
  493. <!-- Configuration Modal -->
  494. <div class="gm-region-modal" data-gm-region-onclick="hideModal">
  495. <!-- Modal Dialog -->
  496. <div class="gm-region-modal-dialog">
  497. <!-- Modal Header -->
  498. <div class="gm-region-modal-header">
  499. <div class="gm-region-modal-title">Google Search Region</div>
  500. <div class="gm-region-modal-close" role="button" aria-label="Close" data-gm-region-onclick="hideModal"></div>
  501. </div>
  502.  
  503. <!-- Modal Body -->
  504. <div class="gm-region-modal-body">
  505. <!-- Menu Configuration -->
  506. <div class="gm-region-modal-subtitle">Menu</div>
  507. <!-- config.showFlags -->
  508. <label class="gm-region-control">
  509. <input class="gm-region-control-input" type="checkbox" data-gm-region-config="showFlags" <%- showFlags ? 'checked' : '' %>>
  510. <span class="gm-region-control-indicator"></span>
  511. <span class="gm-region-control-description">Show country flags</span>
  512. </label>
  513.  
  514. <!-- URL Configuration -->
  515. <div class="gm-region-modal-subtitle">URL</div>
  516. <!-- config.setTLD -->
  517. <label class="gm-region-control">
  518. <input class="gm-region-control-input" type="checkbox" data-gm-region-config="setTLD" <%- setTLD ? 'checked' : '' %>>
  519. <span class="gm-region-control-indicator"></span>
  520. <span class="gm-region-control-description">Set top level domain</span>
  521. </label>
  522. <!-- config.setHl -->
  523. <label class="gm-region-control">
  524. <input class="gm-region-control-input" type="checkbox" data-gm-region-config="setHl" <%- setHl ? 'checked' : '' %>>
  525. <span class="gm-region-control-indicator"></span>
  526. <span class="gm-region-control-description">Set host language (hl)</span>
  527. </label>
  528. <!-- config.setGl -->
  529. <label class="gm-region-control">
  530. <input class="gm-region-control-input" type="checkbox" data-gm-region-config="setGl" <%- setGl ? 'checked' : '' %>>
  531. <span class="gm-region-control-indicator"></span>
  532. <span class="gm-region-control-description">Set region (gl)</span>
  533. </label>
  534. <!-- config.setCr -->
  535. <label class="gm-region-control">
  536. <input class="gm-region-control-input" type="checkbox" data-gm-region-config="setCr" <%- setCr ? 'checked' : '' %>>
  537. <span class="gm-region-control-indicator"></span>
  538. <span class="gm-region-control-description">Set country filter (cr)</span>
  539. </label>
  540. <!-- config.setLr -->
  541. <label class="gm-region-control">
  542. <input class="gm-region-control-input" type="checkbox" data-gm-region-config="setLr" <%- setLr ? 'checked' : '' %>>
  543. <span class="gm-region-control-indicator"></span>
  544. <span class="gm-region-control-description">Set language filter (lr)</span>
  545. </label>
  546.  
  547. <!-- Regions Configuration -->
  548. <div class="gm-region-modal-subtitle">Regions</div>
  549. <div class="gm-region-columns">
  550. <!-- config.userRegions -->
  551. <% regions.forEach(region => { %>
  552. <% let { id, name, country } = region; %>
  553. <% let isChecked = userRegions.includes(id); %>
  554. <label class="gm-region-control" title="<%- name %>">
  555. <input class="gm-region-control-input" type="checkbox"
  556. data-gm-region-config="userRegions:<%- id %>" <%- isChecked ? 'checked' : '' %>>
  557. <span class="gm-region-control-indicator"></span>
  558. <span class="gm-region-control-description">
  559. <% if (country) { %> <span class="flag flag-<%- country %>"></span> <% } %>
  560. <%- name %>
  561. </span>
  562. </label>
  563. <% }); %>
  564. </div>
  565. </div>
  566.  
  567. <!-- Modal Footer -->
  568. <div class="gm-region-modal-footer">
  569. <button class="gm-region-btn gm-region-btn-primary" data-gm-region-onclick="save">Save</button>
  570. <button class="gm-region-btn gm-region-btn-default" data-gm-region-onclick="hideModal">Cancel</button>
  571. </div>
  572. </div>
  573. </div>
  574. `
  575. const html = renderTemplate(template, data)
  576.  
  577. target.insertAdjacentHTML('beforeend', html)
  578. }
  579.  
  580. /**
  581. * @return {Promise<void>}
  582. */
  583. function delegateEvents () {
  584. const body = document.body
  585. const events = {}
  586.  
  587. events.showModal = function showModal () {
  588. const modal = $('.gm-region-modal')
  589. if (modal) { modal.style.display = null } else { createModal(body) }
  590. }
  591.  
  592. events.hideModal = function hideModal () {
  593. const modal = $('.gm-region-modal')
  594. if (modal) { modal.style.display = 'none' }
  595. }
  596.  
  597. events.save = function save () {
  598. const modal = $('.gm-region-modal')
  599. const controls = $$('[data-gm-region-config]', modal)
  600. const pending = {}
  601.  
  602. controls.forEach(control => {
  603. const attr = control.getAttribute('data-gm-region-config').split(':')
  604. const [name, value = control.value] = attr
  605.  
  606. if (typeof config[name] === 'boolean') {
  607. pending[name] = control.checked
  608. }
  609. if (Array.isArray(config[name])) {
  610. if (!Array.isArray(pending[name])) { pending[name] = [] }
  611. if (control.checked) { pending[name].push(value) }
  612. }
  613. })
  614.  
  615. Object.assign(config, pending)
  616.  
  617. saveConfig().then(() => {
  618. window.location.reload()
  619. })
  620. }
  621.  
  622. $delegate(body, '[data-gm-region-onclick]', 'click', event => {
  623. const name = event.target.getAttribute('data-gm-region-onclick')
  624. const callback = events[name]
  625. if (callback) { callback.call(event.target, event) }
  626. })
  627.  
  628. return Promise.resolve()
  629. }
  630.  
  631. /**
  632. * @return {Promise<HTMLStyleElement>}
  633. */
  634. function addStyles () {
  635. const style = GM_addStyle(`
  636. /*!
  637. * Region Menu Dropdown CSS
  638. */
  639. .hdtb-sel{font-weight:700}
  640. .gm-region-menu-dropdown{max-height:80vh;overflow-y:auto}
  641. .gm-region-menu-dropdown .hdtbItm.hdtbSel{padding:0}
  642. .gm-region-menu-dropdown .hdtbItm.hdtbSel a{background-color:transparent}
  643. .gm-region-menu-config{cursor:pointer}
  644. /*!
  645. * Configuration Modal CSS
  646. */
  647. .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)}
  648. .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}
  649. .gm-region-modal-header{display:flex;justify-content:space-between}
  650. .gm-region-modal-footer{text-align:right}
  651. .gm-region-modal-body{margin:16px 0}
  652. .gm-region-modal-title{font-size:16px;font-weight:thin}
  653. .gm-region-modal-subtitle{margin:16px 0;font-size:13px;font-weight:700}
  654. .gm-region-modal-close{display:inline-block;width:10px;height:10px;background-image:url();background-repeat:no-repeat;cursor:pointer}
  655. .gm-region-columns{max-height:300px;overflow-x:auto;-webkit-column-count:5;-moz-column-count:5;column-count:5}
  656. .gm-region-control{display:block;margin:4px 0;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}
  657. .gm-region-control-input{display:none}
  658. .gm-region-control-indicator{display:inline-block;margin:0 4px;width:10px;height:10px;border:1px solid #c6c6c6;border-radius:1px;vertical-align:middle}
  659. .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}
  660. .gm-region-control:hover .gm-region-control-indicator{border-color:#b2b2b2;box-shadow:inset 0 1px 1px rgba(0,0,0,.1)}
  661. .gm-region-control-input:checked~.gm-region-control-indicator::after{display:inline-block}
  662. .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}
  663. .gm-region-btn-default{border-color:rgba(0,0,0,.1);background-image:linear-gradient(#f5f5f5,#f1f1f1);color:#444}
  664. .gm-region-btn-default:hover{border-color:#c6c6c6;background-image:linear-gradient(#f8f8f8,#f1f1f1);color:#333}
  665. .gm-region-btn-default:focus{border-color:#4d90fe}
  666. .gm-region-btn-primary{border-color:#3079ed;background-image:linear-gradient(#4d90fe,#4787ed);color:#fff}
  667. .gm-region-btn-primary:hover{border-color:#2f5bb7;background-image:linear-gradient(#4d90fe,#357ae8);color:#fff}
  668. .gm-region-btn-primary:focus{border-color:transparent;box-shadow:inset 0 0 0 1px #fff}
  669. /*!
  670. * Generated with CSS Flag Sprite Generator <https://www.flag-sprites.com/>
  671. *
  672. * FAMFAMFAM Flag Icons <http://www.famfamfam.com/lab/icons/flags/>
  673. * These flag icons are available for free use for any purpose with no
  674. * requirement for attribution.
  675. */
  676. .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}
  677. .flag.flag-ar{background-position:0 0}
  678. .flag.flag-at{background-position:-16px 0}
  679. .flag.flag-au{background-position:-32px 0}
  680. .flag.flag-be{background-position:-48px 0}
  681. .flag.flag-bg{background-position:-64px 0}
  682. .flag.flag-br{background-position:-80px 0}
  683. .flag.flag-ca{background-position:-96px 0}
  684. .flag.flag-ct{background-position:-112px 0}
  685. .flag.flag-ch{background-position:0 -11px}
  686. .flag.flag-cl{background-position:-16px -11px}
  687. .flag.flag-cn{background-position:-32px -11px}
  688. .flag.flag-co{background-position:-48px -11px}
  689. .flag.flag-cz{background-position:-64px -11px}
  690. .flag.flag-de{background-position:-80px -11px}
  691. .flag.flag-dk{background-position:-96px -11px}
  692. .flag.flag-ee{background-position:-112px -11px}
  693. .flag.flag-es{background-position:0 -22px}
  694. .flag.flag-fi{background-position:-16px -22px}
  695. .flag.flag-fr{background-position:-32px -22px}
  696. .flag.flag-gb{background-position:-48px -22px}
  697. .flag.flag-gr{background-position:-64px -22px}
  698. .flag.flag-hk{background-position:-80px -22px}
  699. .flag.flag-hr{background-position:-96px -22px}
  700. .flag.flag-hu{background-position:-112px -22px}
  701. .flag.flag-id{background-position:0 -33px}
  702. .flag.flag-ie{background-position:-16px -33px}
  703. .flag.flag-il{background-position:-32px -33px}
  704. .flag.flag-in{background-position:-48px -33px}
  705. .flag.flag-it{background-position:-64px -33px}
  706. .flag.flag-jp{background-position:-80px -33px}
  707. .flag.flag-kr{background-position:-96px -33px}
  708. .flag.flag-lt{background-position:-112px -33px}
  709. .flag.flag-lv{background-position:0 -44px}
  710. .flag.flag-mx{background-position:-16px -44px}
  711. .flag.flag-my{background-position:-32px -44px}
  712. .flag.flag-nl{background-position:-48px -44px}
  713. .flag.flag-no{background-position:-64px -44px}
  714. .flag.flag-nz{background-position:-80px -44px}
  715. .flag.flag-pe{background-position:-96px -44px}
  716. .flag.flag-ph{background-position:-112px -44px}
  717. .flag.flag-pl{background-position:0 -55px}
  718. .flag.flag-pt{background-position:-16px -55px}
  719. .flag.flag-ro{background-position:-32px -55px}
  720. .flag.flag-ru{background-position:-48px -55px}
  721. .flag.flag-sa{background-position:-64px -55px}
  722. .flag.flag-se{background-position:-80px -55px}
  723. .flag.flag-sg{background-position:-96px -55px}
  724. .flag.flag-sk{background-position:-112px -55px}
  725. .flag.flag-sl{background-position:0 -66px}
  726. .flag.flag-th{background-position:-16px -66px}
  727. .flag.flag-tr{background-position:-32px -66px}
  728. .flag.flag-tw{background-position:-48px -66px}
  729. .flag.flag-us{background-position:-64px -66px}
  730. .flag.flag-vn{background-position:-80px -66px}
  731. .flag.flag-za{background-position:-96px -66px}
  732. `)
  733.  
  734. return Promise.resolve(style)
  735. }
  736.  
  737. // =============================================================================
  738. // Initialization
  739. // =============================================================================
  740.  
  741. /**
  742. * @return {Promise<Element>}
  743. */
  744. function waitForPageReady () {
  745. return new Promise(resolve => {
  746. const observee = $('#hdtb')
  747. const observer = new MutationObserver(() => {
  748. const target = $('#hdtb-mn-gp')
  749. if (target) { resolve(target) }
  750. })
  751.  
  752. observer.observe(observee, { childList: true, subtree: true })
  753. })
  754. }
  755.  
  756. Promise.all([
  757. waitForPageReady(),
  758. loadConfig(),
  759. delegateEvents(),
  760. addStyles()
  761. ]).then(values => createMenu(values[0]))