/* global $ $create openURL showHelp t tryRegExp URLS */ /* exported regExpTester */ 'use strict'; const regExpTester = (() => { const GET_FAVICON_URL = 'https://www.google.com/s2/favicons?domain='; const OWN_ICON = chrome.runtime.getManifest().icons['16']; const cachedRegexps = new Map(); let currentRegexps = []; let isInit = false; function init() { isInit = true; chrome.tabs.onUpdated.addListener(onTabUpdate); } function uninit() { chrome.tabs.onUpdated.removeListener(onTabUpdate); isInit = false; } function onTabUpdate(tabId, info) { if (info.url) { update(); } } function isShown() { return Boolean($('.regexp-report')); } function toggle(state = !isShown()) { if (state && !isShown()) { if (!isInit) { init(); } showHelp('', $create('.regexp-report')); } else if (!state && isShown()) { if (isInit) { uninit(); } // TODO: need a closeHelp function $('#help-popup .dismiss').onclick(); } } function update(newRegexps) { if (!isShown()) { if (isInit) { uninit(); } return; } if (newRegexps) { currentRegexps = newRegexps; } const regexps = currentRegexps.map(text => { const rxData = Object.assign({text}, cachedRegexps.get(text)); if (!rxData.urls) { cachedRegexps.set(text, Object.assign(rxData, { // imitate buggy Stylish-for-chrome rx: tryRegExp('^' + text + '$'), urls: new Map(), })); } return rxData; }); const getMatchInfo = m => m && {text: m[0], pos: m.index}; browser.tabs.query({}).then(tabs => { const supported = tabs.map(tab => tab.pendingUrl || tab.url).filter(URLS.supported); const unique = [...new Set(supported).values()]; for (const rxData of regexps) { const {rx, urls} = rxData; if (rx) { const urlsNow = new Map(); for (const url of unique) { const match = urls.get(url) || getMatchInfo(url.match(rx)); if (match) { urlsNow.set(url, match); } } rxData.urls = urlsNow; } } const stats = { full: {data: [], label: t('styleRegexpTestFull')}, partial: {data: [], label: [ t('styleRegexpTestPartial'), t.template.regexpTestPartial.cloneNode(true), ]}, none: {data: [], label: t('styleRegexpTestNone')}, invalid: {data: [], label: t('styleRegexpTestInvalid')}, }; // collect stats for (const {text, rx, urls} of regexps) { if (!rx) { stats.invalid.data.push({text}); continue; } if (!urls.size) { stats.none.data.push({text}); continue; } const full = []; const partial = []; for (const [url, match] of urls.entries()) { const faviconUrl = url.startsWith(URLS.ownOrigin) ? OWN_ICON : GET_FAVICON_URL + new URL(url).hostname; const icon = $create('img', {src: faviconUrl}); if (match.text.length === url.length) { full.push($create('a', {href: '#'}, [ icon, url, ])); } else { partial.push($create('a', {href: '#'}, [ icon, url.substr(0, match.pos), $create('mark', match.text), url.substr(match.pos + match.text.length), ])); } } if (full.length) { stats.full.data.push({text, urls: full}); } if (partial.length) { stats.partial.data.push({text, urls: partial}); } } // render stats const report = $create('.regexp-report'); const br = $create('br'); for (const type in stats) { // top level groups: full, partial, none, invalid const {label, data} = stats[type]; if (!data.length) { continue; } const block = report.appendChild( $create('details', {open: true, dataset: {type}}, [ $create('summary', label), ])); // 2nd level: regexp text for (const {text, urls} of data) { if (urls) { // type is partial or full block.appendChild( $create('details', {open: true}, [ $create('summary', text), // 3rd level: tab urls ...urls, ])); } else { // type is none or invalid block.appendChild(document.createTextNode(text)); block.appendChild(br.cloneNode()); } } } showHelp(t('styleRegexpTestTitle'), report); report.onclick = onClick; const note = $create('p.regexp-report-note', t('styleRegexpTestNote') .split(/(\\+)/) .map(s => (s.startsWith('\\') ? $create('code', s) : s))); report.appendChild(note); adjustNote(report, note); }); function onClick(event) { const a = event.target.closest('a'); if (a) { event.preventDefault(); openURL({ url: a.href && a.getAttribute('href') !== '#' && a.href || a.textContent, currentWindow: null, }); } else if (event.target.closest('details')) { setTimeout(adjustNote); } } function adjustNote(report, note) { report.style.paddingBottom = note.offsetHeight + 'px'; } } return {toggle, update}; })();