/* global SafariBrowserTab, Services, XPCOMUtils */
// for background page only

(function() {
'use strict';

self.vAPI = self.vAPI || {};

if (self.chrome) {
    var chrome = self.chrome;

    vAPI.chrome = true;

    vAPI.storage = chrome.storage.local;

    vAPI.tabs = {
        registerListeners: function() {
            if (typeof this.onNavigation === 'function') {
                chrome.webNavigation.onCommitted.addListener(this.onNavigation);
            }

            if (typeof this.onUpdated === 'function') {
                chrome.tabs.onUpdated.addListener(this.onUpdated);
            }

            if (typeof this.onClosed === 'function') {
                chrome.tabs.onRemoved.addListener(this.onClosed);
            }

            if (typeof this.onPopup === 'function') {
                chrome.webNavigation.onCreatedNavigationTarget.addListener(this.onPopup);
            }
        },
        get: function(tabId, callback) {
            if (tabId === null) {
                chrome.tabs.query(
                    {
                        active: true,
                        currentWindow: true
                    },
                    function(tabs) {
                        callback(tabs[0]);
                    }
                );
            }
            else {
                chrome.tabs.get(tabId, callback);
            }
        },
        /*open: function(details) {
            // to keep incognito context?
            chrome.windows.getCurrent(function(win) {
                details.windowId = win.windowId;
                chrome.tabs.create(details);
            });
        },*/
        open: function(details) {
            if (!details.url) {
                return null;
            }
            // extension pages
            else if (!details.url.match(/^\w{2,20}:/)) {
                details.url = vAPI.getURL(details.url);
            }

            // dealing with Chrome's asynhronous API
            var wrapper = function() {
                if (details.active === undefined) {
                    details.active = true;
                }

                var subWrapper = function() {
                    var _details = {
                        url: details.url,
                        active: !!details.active
                    };

                    if (details.tabId) {
                        // update doesn't accept index, must use move
                        chrome.tabs.update(details.tabId, _details, function(tab) {
                            // if the tab doesn't exist
                            if (chrome.runtime.lastError) {
                                chrome.tabs.create(_details);
                            }
                            else if (details.index !== undefined) {
                                chrome.tabs.move(tab.id, {index: details.index});
                            }
                        });
                    }
                    else {
                        if (details.index !== undefined) {
                            _details.index = details.index;
                        }

                        chrome.tabs.create(_details);
                    }
                };

                if (details.index === -1) {
                    vAPI.tabs.get(null, function(tab) {
                        if (tab) {
                            details.index = tab.index + 1;
                        }
                        else {
                            delete details.index;
                        }

                        subWrapper();
                    });
                }
                else {
                    subWrapper();
                }
            };

            if (details.select) {
                // note that currentWindow may be even the window of Developer Tools
                // so, test with setTimeout...
                chrome.tabs.query({currentWindow: true}, function(tabs) {
                    var url = details.url.replace(rgxHash, '');
                    // this is questionable
                    var rgxHash = /#.*/;

                    tabs = tabs.some(function(tab) {
                        if (tab.url.replace(rgxHash, '') === url) {
                            chrome.tabs.update(tab.id, {active: true});
                            return true;
                        }
                    });

                    if (!tabs) {
                        wrapper();
                    }
                });
            }
            else {
                wrapper();
            }
        },
        close: chrome.tabs.remove.bind(chrome.tabs),
        injectScript: function(tabId, details, callback) {
            if (!callback) {
                callback = function(){};
            }

            if (tabId) {
                chrome.tabs.executeScript(tabId, details, callback);
            }
            else {
                chrome.tabs.executeScript(details, callback);
            }
        }
    };

    // Must read: https://code.google.com/p/chromium/issues/detail?id=410868#c8

    // https://github.com/gorhill/uBlock/issues/19
    // https://github.com/gorhill/uBlock/issues/207
    // Since we may be called asynchronously, the tab id may not exist
    // anymore, so this ensures it does still exist.

    vAPI.setIcon = function(tabId, img, badge) {
        var onIconReady = function() {
            if ( chrome.runtime.lastError ) {
                return;
            }

            chrome.browserAction.setBadgeText({ tabId: tabId, text: badge });

            if ( badge !== '' ) {
                chrome.browserAction.setBadgeBackgroundColor({ tabId: tabId, color: '#666' });
            }
        };
        chrome.browserAction.setIcon({ tabId: tabId, path: img }, onIconReady);
    };

    vAPI.messaging = {
        ports: {},
        listeners: {},
        listen: function(listenerName, callback) {
            this.listeners[listenerName] = callback;
        },
        setup: function(connector) {
            if (this.connector) {
                return;
            }

            this.connector = function(port) {
                var onMessage = function(request) {
                    var callback = function(response) {
                        if (chrome.runtime.lastError || response === undefined) {
                            return;
                        }

                        if (request.requestId) {
                            port.postMessage({
                                requestId: request.requestId,
                                portName: request.portName,
                                msg: response
                            });
                        }
                    };

                    var listener = connector(request.msg, port.sender, callback);

                    if (listener === null) {
                        listener = vAPI.messaging.listeners[request.portName];

                        if (typeof listener === 'function') {
                            listener(request.msg, port.sender, callback);
                        } else {
                            console.error('µBlock> messaging > unknown request: %o', request);
                        }
                    }
                };

                var onDisconnect = function(port) {
                    port.onDisconnect.removeListener(onDisconnect);
                    port.onMessage.removeListener(onMessage);
                    delete vAPI.messaging.ports[port.name];
                };

                port.onDisconnect.addListener(onDisconnect);
                port.onMessage.addListener(onMessage);
                vAPI.messaging.ports[port.name] = port;
            };

            chrome.runtime.onConnect.addListener(this.connector);
        },
        broadcast: function(message) {
            message = {
                broadcast: true,
                msg: message
            };

            for (var portName in this.ports) {
                this.ports[portName].postMessage(message);
            }
        }
    };

    vAPI.net = {
        registerListeners: function() {
            var listeners = [
                'onBeforeRequest',
                'onBeforeSendHeaders',
                'onHeadersReceived'
            ];

            for (var i = 0; i < listeners.length; ++i) {
                chrome.webRequest[listeners[i]].addListener(
                    this[listeners[i]].callback,
                    {
                        'urls': this[listeners[i]].urls || ['<all_urls>'],
                        'types': this[listeners[i]].types || []
                    },
                    this[listeners[i]].extra
                );
            }
        }
    };

    vAPI.contextMenu = {
        create: function(details, callback) {
            this.menuId = details.id;
            this.callback = callback;
            chrome.contextMenus.create(details);
            chrome.contextMenus.onClicked.addListener(this.callback);
        },
        remove: function() {
            chrome.contextMenus.onClicked.removeListener(this.callback);
            chrome.contextMenus.remove(this.menuId);
        }
    };
} else if (self.safari) {
    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 (request.message.requestId && 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 === null) {
                    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;
        }
    };
}

if (!self.chrome) {
    self.chrome = { runtime: { lastError: null } };
}
})();