/** * @description API URL * @type {string} */ const apiUrl = 'https://api.cookie-dialog-monster.com/rest/v1'; /** * @description Base data URL * @type {string} */ const baseDataUrl = 'https://raw.githubusercontent.com/wanhose/cookie-dialog-monster/main/data'; /** * @description Context menu identifier * @type {string} */ const contextMenuId = 'CDM_MENU'; /** * @description Cache initial state * @type {{ enabled: boolean }} */ const initial = { enabled: true }; /** * @description Disables icon * @param {string} tabId */ const disableIcon = (tabId) => chrome.browserAction.setIcon({ path: 'assets/icons/disabled.png', tabId }); /** * @description Enables icon * @param {string} tabId */ const enableIcon = (tabId) => chrome.browserAction.setIcon({ path: 'assets/icons/enabled.png', tabId }); /** * @description Enables popup * @param {string} tabId */ const enablePopup = (tabId) => chrome.browserAction.setPopup({ popup: 'popup.html', tabId }); /** * @description Retrieves cache state * @param {string} hostname * @param {void} callback * @returns {{ enabled: boolean }} */ const getCache = (hostname, callback) => { chrome.storage.local.get(null, (store) => { callback(store[hostname] ?? initial); }); }; /** * @async * @description Retrieves data from GitHub * @param {string} key * @param {void} callback * @returns {Promise<{ any: string[] }>} */ const query = async (key, callback) => { try { const url = `${baseDataUrl}/${key}.txt`; const response = await fetch(url); const data = await response.text(); if (response.status !== 200) throw new Error(); callback({ [key]: data.split('\n') }); } catch { callback({ [key]: [] }); } }; /** * @description Retrieves current tab information * @param {void} [callback] * @returns {Promise<{ id: string, location: string }>} */ const queryTab = (callback) => { chrome.tabs.query({ active: true, currentWindow: true }, (tabs) => { callback({ id: tabs[0]?.id, hostname: new URL(tabs[0].url).hostname.split('.').slice(-2).join('.'), }); }); }; /** * @description Reports active tab URL */ const report = () => { chrome.tabs.query({ active: true, currentWindow: true }, (tabs) => { const tab = tabs[0]; const userAgent = window.navigator.userAgent; const version = chrome.runtime.getManifest().version; if (tab) { fetch(`${apiUrl}/report/`, { body: JSON.stringify({ html: `Browser: ${userAgent}
Site: ${tab.url}
Version: ${version}`, to: 'wanhose.development@gmail.com', subject: 'Cookie Dialog Monster Report', }), headers: { 'Content-type': 'application/json', }, method: 'POST', }); } }); }; /** * @description Update cache state * @param {string} [hostname] * @param {object} [state] */ const updateCache = (hostname, state) => { chrome.storage.local.get(null, (cache) => { const current = cache[hostname]; chrome.storage.local.set({ [hostname]: { enabled: typeof state.enabled === 'undefined' ? current.enabled : state.enabled, }, }); }); }; /** * @description Listens to messages */ chrome.runtime.onMessage.addListener((request, sender, callback) => { const hostname = request.hostname; const state = request.state; const tabId = sender.tab?.id; switch (request.type) { case 'DISABLE_ICON': if (tabId) disableIcon(tabId); break; case 'ENABLE_ICON': if (tabId) enableIcon(tabId); break; case 'ENABLE_POPUP': if (tabId) enablePopup(tabId); break; case 'GET_CACHE': getCache(hostname, callback); break; case 'GET_CLASSES': query('classes', callback); break; case 'GET_SKIPS': query('skips', callback); break; case 'GET_FIXES': query('fixes', callback); break; case 'GET_SELECTORS': query('elements', callback); break; case 'GET_TAB': queryTab(callback); break; case 'UPDATE_CACHE': updateCache(hostname, state); break; default: break; } return true; }); /** * @description Creates context menu */ chrome.contextMenus.create({ contexts: ['all'], id: contextMenuId, title: chrome.i18n.getMessage('contextMenuText'), }); /** * @description Listens to context menus */ chrome.contextMenus.onClicked.addListener((info) => { if (info.menuItemId !== contextMenuId) return; report(); });