Google Search Region

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

当前为 2021-01-22 提交的版本,查看 最新版本

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