File: extension-tabsrunner.js

package info (click to toggle)
plasma-browser-integration 6.3.4-2
  • links: PTS, VCS
  • area: main
  • in suites: trixie
  • size: 2,044 kB
  • sloc: cpp: 2,989; javascript: 2,444; xml: 142; python: 77; sh: 46; makefile: 16
file content (163 lines) | stat: -rw-r--r-- 5,234 bytes parent folder | download | duplicates (3)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
/*
    Copyright (C) 2017 Kai Uwe Broulik <kde@privat.broulik.de>

    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/>.
 */

// URL - data URL
let favIconDataForUrl = {};
let clearFavIconDataTimeoutId = 0;
let runningGetTabsQueries = 0;

addCallback("tabsrunner", "activate", function (message) {
    var tabId = message.tabId;

    console.log("Tabs Runner requested to activate tab with id", tabId);

    raiseTab(tabId);
});

addCallback("tabsrunner", "setMuted", function (message) {

    var tabId = message.tabId;
    var muted = message.muted;

    chrome.tabs.update(tabId, {muted: muted}, function (tab) {

        if (chrome.runtime.lastError || !tab) { // this "lastError" stuff feels so archaic
            // failed to mute/unmute
            return;
        }
    });

});

// only forward certain tab properties back to our host
var whitelistedTabProperties = [
    "id", "active", "audible", "favIconUrl", "incognito", "title", "url", "mutedInfo"
];

// FIXME We really should enforce some kind of security policy, so only e.g. plasmashell and krunner
// may access your tabs
addCallback("tabsrunner", "getTabs", function (message) {
    ++runningGetTabsQueries;

    chrome.tabs.query({
        windowType: "normal"
    }, (tabs) => {
        if (clearFavIconDataTimeoutId) {
            clearTimeout(clearFavIconDataTimeoutId);
            clearFavIconDataTimeoutId = 0;
        }

        // remove incognito tabs and properties not in whitelist
        let filteredTabs = tabs;

        // Firefox before 67 runs extensions in incognito by default
        // but we keep running after an update, so exclude those tabs for it
        if (IS_FIREFOX) {
            filteredTabs = filteredTabs.filter(function (tab) {
                return !tab.incognito;
            });
        }

        filteredTabs = filterArrayObjects(filteredTabs, whitelistedTabProperties);

        let favIconUrlsToFetch = new Set();

        // Collect a set of fav icons to be requested
        filteredTabs.forEach((tab) => {
            const url = tab.favIconUrl;
            if (!url) {
                return;
            }

            // Already a data URL
            if (url.match(/^data:image/)) {
                return;
            }

            // Already in cache
            if (favIconDataForUrl[url]) {
                return;
            }

            favIconUrlsToFetch.add(url);
        });

        // Prepare the download requests for all fav icons
        let requests = [];
        favIconUrlsToFetch.forEach((url) => {
            requests.push(new Promise((resolve) => {
                fetch(url, {
                    cache: "force-cache"
                }).then((response) => {
                    if (!response.ok) {
                        return resolve();
                    }

                    response.blob().then((blob) => {
                        let reader = new FileReader();
                        reader.onloadend = function() {
                            favIconDataForUrl[url] = reader.result;
                            return resolve();
                        }
                        reader.readAsDataURL(blob);
                    }, (err) => {
                        console.warn("Failed to read response of", url, "as blob", err);
                        resolve();
                    });
                }, (err) => {
                    console.warn("Failed to get favicon from", url, err);
                    resolve();
                });
            }));
        });

        // Download all favicons and send them out
        Promise.all(requests).then(() => {
            filteredTabs = filteredTabs.map((tab) => {
                const favIconUrl = tab.favIconUrl;
                if (!favIconUrl) {
                    return tab;
                }

                if (favIconUrl.match(/^data:image/)) {
                    tab.favIconData = favIconUrl;
                    return tab
                }

                const data = favIconDataForUrl[favIconUrl];
                if (data) {
                    tab.favIconData = data;
                }
                return tab;
            });

            --runningGetTabsQueries;
            if (runningGetTabsQueries === 0) {
                clearFavIconDataTimeoutId = setTimeout(() => {
                    favIconDataForUrl = {};
                    clearFavIconDataTimeoutId = 0;
                }, 60000);
            }

            port.postMessage({
                subsystem: "tabsrunner",
                event: "gotTabs",
                tabs: filteredTabs
            });
        });
    });
});