/******************************************************************************* µBlock - a Chromium browser extension to block requests. Copyright (C) 2014 The µBlock authors This program is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program. If not, see {http://www.gnu.org/licenses/}. Home: https://github.com/gorhill/uBlock */ // For background page /* global SafariBrowserTab, Services, XPCOMUtils */ /******************************************************************************/ (function() { 'use strict'; /******************************************************************************/ self.vAPI = self.vAPI || {}; vAPI.safari = true; /******************************************************************************/ // addContentScriptFromURL allows whitelisting, // so load sitepaching this way, instead of adding it to the Info.plist safari.extension.addContentScriptFromURL( safari.extension.baseURI + 'js/sitepatch-safari.js', [ 'http://www.youtube.com/*', 'https://www.youtube.com/*', 'http://www.youtube-nocookie.com/*', 'https://www.youtube-nocookie.com/*' ] ); /******************************************************************************/ safari.extension.settings.addEventListener('change', function(e) { if (e.key === 'open_prefs') { vAPI.tabs.open({url: 'dashboard.html', active: true}); } }, false); /******************************************************************************/ vAPI.storage = { _storage: safari.extension.settings, QUOTA_BYTES: 52428800, // copied from Info.plist get: function(keys, callback) { if (typeof callback !== 'function') { return; } var i, value, result = {}; if (keys === null) { for (i in this._storage) { value = this._storage[i]; if (typeof value === 'string') { result[i] = JSON.parse(value); } } } else if (typeof keys === 'string') { value = this._storage[keys]; if (typeof value === 'string') { result[keys] = JSON.parse(value); } } else if (Array.isArray(keys)) { for ( i = 0; i < keys.length; ++i) { value = this._storage[i]; if (typeof value === 'string') { result[keys[i]] = JSON.parse(value); } } } else if (typeof keys === 'object') { for (i in keys) { value = this._storage[i]; if (typeof value === 'string') { result[i] = JSON.parse(value); } else { result[i] = keys[i]; } } } callback(result); }, set: function(details, callback) { for (var key in details) { this._storage.setItem(key, JSON.stringify(details[key])); } if (typeof callback === 'function') { callback(); } }, remove: function(keys) { if (typeof keys === 'string') { keys = [keys]; } for (var i = 0; i < keys.length; ++i) { this._storage.removeItem(keys[i]); } }, clear: function(callback) { this._storage.clear(); callback(); }, getBytesInUse: function(keys, callback) { var key, size = 0; if (keys === null) { for (key in this._storage) { size += (this._storage[key] || '').length; } } else { if (typeof keys === 'string') { keys = [keys]; } for (key = 0; key < keys.length; ++key) { size += (this._storage[keys[key]] || '').length; } } callback(size); } }; /******************************************************************************/ vAPI.tabs = { stack: {}, stackID: 1, registerListeners: function() { var onNavigation = this.onNavigation; if (typeof onNavigation === 'function') { this.onNavigation = function(e) { // e.url is not present for local files or data URIs, // or probably for those URLs which we don't have access to if (!e.target || !e.target.url) { return; } onNavigation({ frameId: 0, tabId: vAPI.tabs.getTabId(e.target), url: e.target.url }); }; safari.application.addEventListener('navigate', this.onNavigation, true); } // ?? /* if (typeof this.onUpdated === 'function') { } */ // onClosed handled in the main tab-close event // onPopup is handled in window.open on web-pages? /* if (typeof onPopup === 'function') { } */ }, getTabId: function(tab) { for (var i in vAPI.tabs.stack) { if (vAPI.tabs.stack[i] === tab) { return +i; } } return -1; }, get: function(tabId, callback) { var tab; if (tabId === null) { tab = safari.application.activeBrowserWindow.activeTab; tabId = this.getTabId(tab); } else { tab = this.stack[tabId]; } if (!tab) { callback(); return; } callback({ id: tabId, index: tab.browserWindow.tabs.indexOf(tab), windowId: safari.application.browserWindows.indexOf(tab.browserWindow), active: tab === tab.browserWindow.activeTab, url: tab.url, title: tab.title }); }, open: function(details) { if (!details.url) { return null; } // extension pages else if (!details.url.match(/^\w{2,20}:/)) { details.url = vAPI.getURL(details.url); } // properties of the details object: // url: 'URL', // the address that will be opened // tabId: 1, // the tab is used if set, instead of creating a new one // index: -1, // undefined: end of the list, -1: following tab, or after index // active: false, // opens the tab in background - true and undefined: foreground // select: true // if a tab is already opened with that url, then select it instead of opening a new one var curWin, tab; if (details.select) { tab = safari.application.browserWindows.some(function(win) { var rgxHash = /#.*/; // this is questionable var url = details.url.replace(rgxHash, ''); for (var i = 0; i < win.tabs.length; ++i) { if (win.tabs[i].url.replace(rgxHash, '') === url) { win.tabs[i].activate(); return true; } } }); if (tab) { return; } } if (details.active === undefined) { details.active = true; } curWin = safari.application.activeBrowserWindow; // it must be calculated before opening a new tab, // otherwise the new tab will be the active tab here if (details.index === -1) { details.index = curWin.tabs.indexOf(curWin.activeTab) + 1; } tab = details.tabId && this.stack[details.tabId] || curWin.openTab(details.active ? 'foreground' : 'background'); if (details.index !== undefined) { curWin.insertTab(tab, details.index); } tab.url = details.url; }, close: function(tab) { if (!(tab instanceof SafariBrowserTab)) { tab = this.stack[tab]; } if (tab) { tab.close(); } }, injectScript: function(tabId, details, callback) { var tab = tabId ? this.stack[tabId] : safari.application.activeBrowserWindow.activeTab; if (details.file) { var xhr = new XMLHttpRequest; xhr.overrideMimeType('application/x-javascript;charset=utf-8'); xhr.open('GET', details.file, false); xhr.send(); details.code = xhr.responseText; } tab.page.dispatchMessage('broadcast', { portName: 'vAPI', msg: { cmd: 'runScript', details: details } }); if (typeof callback === 'function') { setTimeout(callback, 13); } } }; /******************************************************************************/ // bind tabs to unique IDs (function() { var wins = safari.application.browserWindows, i = wins.length, j; var tabs = []; while (i--) { j = wins[i].tabs.length; while (j--) { tabs.push(wins[i].tabs[j]); } } return tabs; })().forEach(function(tab) { vAPI.tabs.stack[vAPI.tabs.stackID++] = tab; }); /******************************************************************************/ safari.application.addEventListener('open', function(e) { // ignore windows if (e.target instanceof SafariBrowserTab) { vAPI.tabs.stack[vAPI.tabs.stackID++] = e.target; } }, true); /******************************************************************************/ safari.application.addEventListener('close', function(e) { // ignore windows if (!(e.target instanceof SafariBrowserTab)) { return; } var tabId = vAPI.tabs.getTabId(e.target); if (tabId > -1) { // to not add another listener, put this here // instead of vAPI.tabs.registerListeners if (typeof vAPI.tabs.onClosed === 'function') { vAPI.tabs.onClosed(tabId); } delete vAPI.tabIcons[tabId]; delete vAPI.tabs.stack[tabId]; } }, true); /******************************************************************************/ // update badge when tab is activated safari.application.addEventListener('activate', function(e) { // hide popover, since in some cases won't close by itself var items = safari.extension.toolbarItems; for (var i = 0; i < items.length; ++i) { if (items[i].browserWindow === safari.application.activeBrowserWindow) { if (items[i].popover) { items[i].popover.hide(); } break; } } // ignore windows if (!(e.target instanceof SafariBrowserTab)) { return; } // update the badge, when tab is selected vAPI.setIcon(); }, true); /******************************************************************************/ // reload the popup when that is opened safari.application.addEventListener('popover', function(e) { e.target.contentWindow.document.body.textContent = ''; e.target.contentWindow.location.reload(); }, true); /******************************************************************************/ vAPI.tabIcons = { /*tabId: {badge: 0, img: dict}*/ }; vAPI.setIcon = function(tabId, img, badge) { var curTabId = vAPI.tabs.getTabId(safari.application.activeBrowserWindow.activeTab); // from 'activate' event if (tabId === undefined) { tabId = curTabId; } else { vAPI.tabIcons[tabId] = { badge: badge || 0/*, img: img*/ }; } // if the selected tab has the same ID, then update the badge too, // or always update it when changing tabs ('activate' event) if (tabId === curTabId) { var items = safari.extension.toolbarItems, i = items.length; while (i--) { if (items[i].browserWindow === safari.application.activeBrowserWindow) { if (vAPI.tabIcons[tabId]) { items[i].badge = vAPI.tabIcons[tabId].badge; // items[i].img = vAPI.tabIcons[tabId].img; } else { items[i].badge = 0; } return; } } } }; /******************************************************************************/ vAPI.messaging = { listeners: {}, listen: function(listenerName, callback) { this.listeners[listenerName] = callback; }, setup: function(connector) { if (this.connector) { return; } this.connector = function(request) { var callback = function(response) { if (response !== undefined) { request.target.page.dispatchMessage( request.name, { requestId: request.message.requestId, portName: request.message.portName, msg: response } ); } }; var sender = { tab: { id: vAPI.tabs.getTabId(request.target) } }; var listener = connector(request.message.msg, sender, callback); if (listener === vAPI.messaging.UNHANDLED) { listener = vAPI.messaging.listeners[request.message.portName]; if (typeof listener === 'function') { listener(request.message.msg, sender, callback); } else { console.error('µBlock> messaging > unknown request: %o', request.message); } } }; // the third parameter must stay false (bubbling), so later // onBeforeRequest will use true (capturing), where we can invoke // stopPropagation() (this way this.connector won't be fired) safari.application.addEventListener('message', this.connector, false); }, broadcast: function(message) { message = { broadcast: true, msg: message }; for (var tabId in vAPI.tabs.stack) { vAPI.tabs.stack[tabId].page.dispatchMessage('broadcast', message); } } }; /******************************************************************************/ vAPI.net = { registerListeners: function() { var onBeforeRequest = this.onBeforeRequest; if (typeof onBeforeRequest.callback === 'function') { if (!Array.isArray(onBeforeRequest.types)) { onBeforeRequest.types = []; } onBeforeRequest = onBeforeRequest.callback; this.onBeforeRequest.callback = function(e) { var block; if (e.name !== 'canLoad') { return; } // no stopPropagation if it was called from beforeNavigate event if (e.stopPropagation) { e.stopPropagation(); } if (e.message.isWhiteListed) { block = µBlock.URI.hostnameFromURI(e.message.isWhiteListed); block = µBlock.URI.domainFromHostname(block) || block; e.message = !!µBlock.netWhitelist[block]; return e.message; } // blocking unwanted pop-ups if (e.message.type === 'popup') { if (typeof vAPI.tabs.onPopup === 'function') { e.message.type = 'main_frame'; e.message.sourceTabId = vAPI.tabs.getTabId(e.target); if (vAPI.tabs.onPopup(e.message)) { e.message = false; return; } } e.message = true; return; } block = vAPI.net.onBeforeRequest; if (block.types.indexOf(e.message.type) < 0) { return true; } e.message.tabId = vAPI.tabs.getTabId(e.target); block = onBeforeRequest(e.message); // truthy return value will allow the request, // except when redirectUrl is present if (block && typeof block === 'object') { if (block.cancel) { e.message = false; } else if (e.message.type === 'script' && typeof block.redirectUrl === "string") { e.message = block.redirectUrl; } else { e.message = true; } } else { e.message = true; } return e.message; }; safari.application.addEventListener('message', this.onBeforeRequest.callback, true); } } }; /******************************************************************************/ vAPI.contextMenu = { create: function(details, callback) { var contexts = details.contexts; var menuItemId = details.id; var menuTitle = details.title; if (Array.isArray(contexts) && contexts.length) { contexts = contexts.indexOf('all') === -1 ? contexts : null; } else { // default in Chrome contexts = ['page']; } this.onContextMenu = function(e) { var uI = e.userInfo; if (uI && /^https?:\/\//i.test(uI.pageUrl)) { if (contexts) { var invalidContext = true; for (var i = 0; i < contexts.length; ++i) { if (contexts[i] === 'frame') { if (uI.insideFrame) { invalidContext = false; break; } } else if (contexts[i] === 'link') { if (uI.linkHref) { invalidContext = false; break; } } else if (contexts[i] === 'image') { if (uI.srcUrl) { invalidContext = false; break; } } else if (contexts[i] === 'audio' || contexts[i] === 'video') { if (uI.srcUrl && uI.tagName === contexts[i]) { invalidContext = false; break; } } else if (contexts[i] === 'editable') { if (uI.editable) { invalidContext = false; break; } } else if (contexts[i] === 'page') { if (!(uI.insideFrame || uI.linkHref || uI.mediaType || uI.editable)) { invalidContext = false; break; } } } if (invalidContext) { return; } } e.contextMenu.appendContextMenuItem(menuItemId, menuTitle); } }; this.onContextMenuCommand = function(e) { if (e.command === menuItemId) { var tab = e.currentTarget.activeBrowserWindow.activeTab; e.userInfo.menuItemId = menuItemId; callback(e.userInfo, tab ? { id: vAPI.tabs.getTabId(tab), url: tab.url } : undefined); } }; safari.application.addEventListener('contextmenu', this.onContextMenu); safari.application.addEventListener("command", this.onContextMenuCommand); }, remove: function() { safari.application.removeEventListener('contextmenu', this.onContextMenu); safari.application.removeEventListener("command", this.onContextMenuCommand); this.onContextMenu = null; this.onContextMenuCommand = null; } }; /******************************************************************************/ vAPI.lastError = { return null; }; /******************************************************************************/ })();