[gnome-browser-extension/wip/refactor: 8/9] Format code




commit 27851db677c66852f14cc1ea45fa8e929be7bce0
Author: Yuri Konotopov <ykonotopov gnome org>
Date:   Thu Aug 4 22:40:52 2022 +0400

    Format code
    
    1. Use spaces instead of tabs
    2. Use open curly bracket on same line
    3. Use space after operator before parenthesis

 extension/content-script-start.js          |  53 +-
 extension/css/options.css                  | 143 ++---
 extension/extension.js                     | 320 ++++++-----
 extension/include/compat-common.js         | 160 +++---
 extension/include/compat-content-script.js |  68 ++-
 extension/include/constants.js             |  96 ++--
 extension/include/gsc.js                   | 260 +++++----
 extension/include/i18n.js                  |  43 +-
 extension/include/notifications.js         | 396 +++++++-------
 extension/include/sweettooth-api.js        | 397 +++++++-------
 extension/include/sync.js                  | 832 ++++++++++++++---------------
 extension/include/toolbar.js               |  90 ++--
 extension/include/update.js                | 360 ++++++-------
 extension/options.html                     | 229 ++++----
 extension/options.js                       | 703 +++++++++++-------------
 15 files changed, 1981 insertions(+), 2169 deletions(-)
---
diff --git a/extension/content-script-start.js b/extension/content-script-start.js
index 2772662..dfa820d 100644
--- a/extension/content-script-start.js
+++ b/extension/content-script-start.js
@@ -6,51 +6,48 @@
  */
 
 var gs_require_inject = function () {
-       GS_CHROME_ID            = "${GS_CHROME_ID}";
-       GS_CHROME_VERSION       = "${GS_CHROME_VERSION}";
+    GS_CHROME_ID        = "${GS_CHROME_ID}";
+    GS_CHROME_VERSION   = "${GS_CHROME_VERSION}";
 };
 
 var siteMessages = {};
-if(EXTERNAL_MESSAGES)
-{
-       for(var key in EXTERNAL_MESSAGES)
-       {
-               siteMessages[EXTERNAL_MESSAGES[key]] = chrome.i18n.getMessage(EXTERNAL_MESSAGES[key]);
-       }
+if (EXTERNAL_MESSAGES) {
+    for (var key in EXTERNAL_MESSAGES) {
+        siteMessages[EXTERNAL_MESSAGES[key]] = chrome.i18n.getMessage(EXTERNAL_MESSAGES[key]);
+    }
 }
 
 var s = document.createElement('script');
 
 s.type = "text/javascript";
 s.textContent = '(' +
-       gs_require_inject.toString()
-               .replace("${GS_CHROME_ID}", GS_CHROME_ID)
-               .replace("${GS_CHROME_VERSION}", chrome.runtime.getManifest().version)
-       + ")(); GSC = {i18n: JSON.parse('" + JSON.stringify(siteMessages).replace(/'/g, "\\'") + "')};";
-(document.head||document.documentElement).appendChild(s);
+    gs_require_inject.toString()
+        .replace("${GS_CHROME_ID}", GS_CHROME_ID)
+        .replace("${GS_CHROME_VERSION}", chrome.runtime.getManifest().version)
+    + ")(); GSC = {i18n: JSON.parse('" + JSON.stringify(siteMessages).replace(/'/g, "\\'") + "')};";
+(document.head || document.documentElement).appendChild(s);
 s.parentNode.removeChild(s);
 
 chrome.runtime.onMessage.addListener(
-       function (request, sender, sendResponse) {
-               if(
-                       sender.id && sender.id === GS_CHROME_ID &&
-                       request && request.signal &&
-                       [SIGNAL_EXTENSION_CHANGED, SIGNAL_SHELL_APPEARED, 
SIGNAL_SHELL_SETTING_CHANGED].indexOf(request.signal) !== -1)
-               {
-                       window.postMessage(
-                               {
-                                       type: "gs-chrome",
-                                       request: request
-                               }, "*"
-                       );
-               }
-       }
+    function (request, sender, sendResponse) {
+        if (
+            sender.id && sender.id === GS_CHROME_ID &&
+            request && request.signal &&
+            [SIGNAL_EXTENSION_CHANGED, SIGNAL_SHELL_APPEARED, 
SIGNAL_SHELL_SETTING_CHANGED].indexOf(request.signal) !== -1) {
+            window.postMessage(
+                {
+                    type: "gs-chrome",
+                    request: request
+                }, "*"
+            );
+        }
+    }
 );
 
 s = document.createElement('script');
 
 s.src = chrome.extension.getURL('include/sweettooth-api.js');
-s.onload = function() {
+s.onload = function () {
     this.parentNode.removeChild(this);
 };
 (document.head || document.documentElement).appendChild(s);
diff --git a/extension/css/options.css b/extension/css/options.css
index 3c7ba6e..456bb04 100644
--- a/extension/css/options.css
+++ b/extension/css/options.css
@@ -1,125 +1,144 @@
 body {
-       font-family: Cantarell, Arial, sans-serif;
-       padding: 10px;
-       min-width: 550px;
+    font-family: Cantarell, Arial, sans-serif;
+    padding: 10px;
+    min-width: 550px;
+}
+
+#update_check_period {
+    width: 5em;
 }
 
-#update_check_period { width: 5em; }
 #status {
-       visibility: hidden;
-       opacity: 0;
-       color: green;
-       text-align: center;
+    visibility: hidden;
+    opacity: 0;
+    color: green;
+    text-align: center;
 }
 
 #error {
-       visibility: hidden;
-       opacity: 0;
-       color: red;
-       text-align: center;
+    visibility: hidden;
+    opacity: 0;
+    color: red;
+    text-align: center;
 }
 
 .show {
-       visibility: visible !important;
-       opacity: 1 !important;
-       transition: visibility 0.25s, opacity 0.25s ease-in-out;
+    visibility: visible !important;
+    opacity: 1 !important;
+    transition: visibility 0.25s, opacity 0.25s ease-in-out;
 }
 
 .hide {
-       display: none;
+    display: none;
 }
 
-.buttons { text-align: center; margin-top: 1em; }
-.notice { font-size: 90%; font-weight: normal; margin-top: 0.5em; white-space: nowrap; }
-.update-notice { display: none; }
-.wrapped { white-space: normal; }
+.buttons {
+    text-align: center;
+    margin-top: 1em;
+}
+
+.notice {
+    font-size: 90%;
+    font-weight: normal;
+    margin-top: 0.5em;
+    white-space: nowrap;
+}
+
+.update-notice {
+    display: none;
+}
+
+.wrapped {
+    white-space: normal;
+}
 
 fieldset {
-       border: 0;
-       margin: 0;
-       padding: 0;
-       width: 100%;
+    border: 0;
+    margin: 0;
+    padding: 0;
+    width: 100%;
 }
 
 dl {
-       padding: 0.2em 0;
-       clear: both;
+    padding: 0.2em 0;
+    clear: both;
 }
 
 dt {
-       display: block;
-       float: left;
-       width: 50%;
-       text-align: left;
-       font-weight: bold;
+    display: block;
+    float: left;
+    width: 50%;
+    text-align: left;
+    font-weight: bold;
 }
 
 dd {
-       margin-left: 51%;
-       vertical-align: top;
-       margin-bottom: 3px;
+    margin-left: 51%;
+    vertical-align: top;
+    margin-bottom: 3px;
 }
 
 ul.tabs {
-       float: right;
+    float: right;
 }
 
 ul.tabs li {
-       list-style: none;
+    list-style: none;
 }
 
-ul.tabs li, ul.tabs li a {
-       display: inline-block;
+ul.tabs li,
+ul.tabs li a {
+    display: inline-block;
 }
 
 ul.tabs li a.active {
-       color: inherit;
-       cursor: default;
-       text-decoration: none;
+    color: inherit;
+    cursor: default;
+    text-decoration: none;
 }
 
 ul.tabs li:not(:last-child) {
-       border-right: 1px solid #000;
-       margin-right: 5px;
-       padding-right: 5px;
+    border-right: 1px solid #000;
+    margin-right: 5px;
+    padding-right: 5px;
 }
 
 div.tabs-pane {
-       clear: both;
+    clear: both;
 }
 
 #synchronization table {
-       width: 100%;
-       border-collapse: collapse;
-       vertical-align: middle;
+    width: 100%;
+    border-collapse: collapse;
+    vertical-align: middle;
 }
 
 #synchronization table th {
-       border-bottom: 1px solid #000;
+    border-bottom: 1px solid #000;
 }
 
-#synchronization table tbody td.ok, #synchronization table td.fail {
-       text-align: center;
-       font-size: 150%;
+#synchronization table tbody td.ok,
+#synchronization table td.fail {
+    text-align: center;
+    font-size: 150%;
 }
 
 #synchronization table tbody td.ok:after {
-       content: '\2713';
-       color: green;
+    content: '\2713';
+    color: green;
 }
 
 #synchronization table td.fail:after {
-       content: '\2717';
-       color: red;
+    content: '\2717';
+    color: red;
 }
 
 dialog#syncChoice button {
-       display: block;
-       width: 90%;
-       margin: 0.3em auto;
+    display: block;
+    width: 90%;
+    margin: 0.3em auto;
 }
 
-div#translation_credits div
-{
-       white-space: pre-line;
+div#translation_credits div {
+    white-space: pre-line;
 }
diff --git a/extension/extension.js b/extension/extension.js
index 0b752db..f1032dd 100644
--- a/extension/extension.js
+++ b/extension/extension.js
@@ -1,196 +1,178 @@
 // SPDX-License-Identifer: GPL-3.0-or-later
 
-chrome.runtime.onInstalled.addListener(function(details) {
-       var version = chrome.runtime.getManifest().version;
-
-       if(details.reason == chrome.runtime.OnInstalledReason.UPDATE && details.previousVersion != version)
-       {
-               chrome.storage.sync.get(DEFAULT_SYNC_OPTIONS, function (options) {
-                       if(options.showReleaseNotes)
-                       {
-                               chrome.tabs.create({
-                                       url: 
'https://wiki.gnome.org/Projects/GnomeShellIntegrationForChrome/ReleaseNotes/' + version,
-                                       active: true
-                               });
-                       }
-               });
-       }
+chrome.runtime.onInstalled.addListener(function (details) {
+    var version = chrome.runtime.getManifest().version;
+
+    if (details.reason == chrome.runtime.OnInstalledReason.UPDATE && details.previousVersion != version) {
+        chrome.storage.sync.get(DEFAULT_SYNC_OPTIONS, function (options) {
+            if (options.showReleaseNotes) {
+                chrome.tabs.create({
+                    url: 'https://wiki.gnome.org/Projects/GnomeShellIntegrationForChrome/ReleaseNotes/' + 
version,
+                    active: true
+                });
+            }
+        });
+    }
 });
 
 chrome.runtime.onMessageExternal.addListener(function (request, sender, sendResponse) {
-       if (
-               EXTENSIONS_WEBSITE.reduce(
-                       (accumulator, url) => accumulator + sender.url.startsWith(url),
-                       0)
-       )
-       {
-               if (request && request.execute)
-               {
-                       if (request.uuid && !GSC.isUUID(request.uuid))
-                       {
-                               return;
-                       }
-
-                       switch (request.execute)
-                       {
-                               case 'initialize':
-                               case 'listExtensions':
-                                       GSC.sendNativeRequest({execute: request.execute}, sendResponse);
-                                       return true;
-
-                               case 'launchExtensionPrefs':
-                                       GSC.sendNativeRequest({execute: request.execute, uuid: request.uuid});
-                                       break;
-
-                               case 'getExtensionErrors':
-                               case 'getExtensionInfo':
-                               case 'installExtension':
-                               case 'uninstallExtension':
-                                       GSC.sendNativeRequest({execute: request.execute, uuid: request.uuid}, 
sendResponse);
-                                       return true;
-
-                               case 'enableExtension':
-                                       GSC.sendNativeRequest({
-                                                       execute: request.execute,
-                                                       uuid: request.uuid,
-                                                       enable: request.enable
-                                               },
-                                               sendResponse
-                                       );
-                                       return true;
-
-                               case 'setUserExtensionsDisabled':
-                               case 'setVersionValidationDisabled':
-                                       GSC.sendNativeRequest({execute: request.execute, disable: 
request.disable ? true : false}, sendResponse);
-                                       return true;
-                       }
-               }
-       }
+    if (
+        EXTENSIONS_WEBSITE.reduce(
+            (accumulator, url) => accumulator + sender.url.startsWith(url),
+            0)
+    ) {
+        if (request && request.execute) {
+            if (request.uuid && !GSC.isUUID(request.uuid)) {
+                return;
+            }
+
+            switch (request.execute) {
+                case 'initialize':
+                case 'listExtensions':
+                    GSC.sendNativeRequest({ execute: request.execute }, sendResponse);
+                    return true;
+
+                case 'launchExtensionPrefs':
+                    GSC.sendNativeRequest({ execute: request.execute, uuid: request.uuid });
+                    break;
+
+                case 'getExtensionErrors':
+                case 'getExtensionInfo':
+                case 'installExtension':
+                case 'uninstallExtension':
+                    GSC.sendNativeRequest({ execute: request.execute, uuid: request.uuid }, sendResponse);
+                    return true;
+
+                case 'enableExtension':
+                    GSC.sendNativeRequest({
+                        execute: request.execute,
+                        uuid: request.uuid,
+                        enable: request.enable
+                    },
+                        sendResponse
+                    );
+                    return true;
+
+                case 'setUserExtensionsDisabled':
+                case 'setVersionValidationDisabled':
+                    GSC.sendNativeRequest({ execute: request.execute, disable: request.disable ? true : 
false }, sendResponse);
+                    return true;
+            }
+        }
+    }
 });
 
 EXTENSIONS_WEBSITE.forEach(url => {
-       chrome.browserAction.onClicked.addListener(function () {
-               chrome.tabs.create({
-                       url: url,
-                       active: true
-               });
-       });
+    chrome.browserAction.onClicked.addListener(function () {
+        chrome.tabs.create({
+            url: url,
+            active: true
+        });
+    });
 });
 
 var disabledExtensionTimeout = null;
-var lastPortMessage = {message: null, date: 0};
+var lastPortMessage = { message: null, date: 0 };
 var port = chrome.runtime.connectNative(NATIVE_HOST);
 /*
  * Native host messaging events handler.
  */
 port.onMessage.addListener(function (message) {
-       if (message && message.signal)
-       {
-               if([SIGNAL_EXTENSION_CHANGED, SIGNAL_SHELL_APPEARED, 
SIGNAL_SHELL_SETTING_CHANGED].indexOf(message.signal) !== -1)
-               {
-                       /*
-                        * Skip duplicate events. This is happens eg when extension is installed.
-                        */
-                       if (
-                                       message.signal != SIGNAL_SHELL_SETTING_CHANGED &&
-                                       (new Date().getTime()) - lastPortMessage.date < 1000 && 
GSC.isSignalsEqual(message, lastPortMessage.message)
-                       )
-                       {
-                               lastPortMessage.date = new Date().getTime();
-                               return;
-                       }
-
-                       /*
-                        * Send events to opened extensions.gnome.org tabs
-                        */
-                       EXTENSIONS_WEBSITE.forEach(url => {
-                               chrome.tabs.query({
-                                       url: url + '*'
-                               },
-                               function (tabs) {
-                                       for (k in tabs)
-                                       {
-                                               chrome.tabs.sendMessage(tabs[k].id, message);
-                                       }
-                               });
-                       });
-
-                       /*
-                        * Route message to Options page.
-                        */
-                       chrome.runtime.sendMessage(GS_CHROME_ID, message);
-                       if (message.signal === SIGNAL_EXTENSION_CHANGED)
-                       {
-                               /*
-                                * GNOME Shell sends 2 events when extension is uninstalled:
-                                * "disabled" event and then "uninstalled" event.
-                                * Let's delay any "disabled" event and drop it if
-                                * "uninstalled" event received within 1,5 secs.
-                                */
-                               if (message.parameters[EXTENSION_CHANGED_STATE] === EXTENSION_STATE.DISABLED)
-                               {
-                                       disabledExtensionTimeout = setTimeout(function () {
-                                               disabledExtensionTimeout = null;
-                                               GSC.sync.onExtensionChanged(message);
-                                       }, 1500);
-                               }
-                               else if (
-                                       disabledExtensionTimeout &&
-                                       message.parameters[EXTENSION_CHANGED_STATE] === 
EXTENSION_STATE.UNINSTALLED &&
-                                       lastPortMessage.message.signal === SIGNAL_EXTENSION_CHANGED &&
-                                       lastPortMessage.message.parameters[EXTENSION_CHANGED_UUID] === 
message.parameters[EXTENSION_CHANGED_UUID] &&
-                                       lastPortMessage.message.parameters[EXTENSION_CHANGED_STATE] === 
EXTENSION_STATE.DISABLED
-                               )
-                               {
-                                       clearTimeout(disabledExtensionTimeout);
-                                       disabledExtensionTimeout = null;
-                                       GSC.sync.onExtensionChanged(message);
-                               }
-                               else
-                               {
-                                       GSC.sync.onExtensionChanged(message);
-                               }
-                       }
-
-                       lastPortMessage = {
-                               message: message,
-                               date: new Date().getTime()
-                       };
-               }
-               else if([SIGNAL_NOTIFICATION_ACTION, SIGNAL_NOTIFICATION_CLICKED].indexOf(message.signal) != 
-1)
-               {
-                       window.postMessage(message, "*");
-               }
-       }
+    if (message && message.signal) {
+        if ([SIGNAL_EXTENSION_CHANGED, SIGNAL_SHELL_APPEARED, 
SIGNAL_SHELL_SETTING_CHANGED].indexOf(message.signal) !== -1) {
+            /*
+             * Skip duplicate events. This is happens eg when extension is installed.
+             */
+            if (
+                message.signal != SIGNAL_SHELL_SETTING_CHANGED &&
+                (new Date().getTime()) - lastPortMessage.date < 1000 && GSC.isSignalsEqual(message, 
lastPortMessage.message)
+            ) {
+                lastPortMessage.date = new Date().getTime();
+                return;
+            }
+
+            /*
+             * Send events to opened extensions.gnome.org tabs
+             */
+            EXTENSIONS_WEBSITE.forEach(url => {
+                chrome.tabs.query({
+                    url: url + '*'
+                },
+                    function (tabs) {
+                        for (k in tabs) {
+                            chrome.tabs.sendMessage(tabs[k].id, message);
+                        }
+                    });
+            });
+
+            /*
+             * Route message to Options page.
+             */
+            chrome.runtime.sendMessage(GS_CHROME_ID, message);
+            if (message.signal === SIGNAL_EXTENSION_CHANGED) {
+                /*
+                 * GNOME Shell sends 2 events when extension is uninstalled:
+                 * "disabled" event and then "uninstalled" event.
+                 * Let's delay any "disabled" event and drop it if
+                 * "uninstalled" event received within 1,5 secs.
+                 */
+                if (message.parameters[EXTENSION_CHANGED_STATE] === EXTENSION_STATE.DISABLED) {
+                    disabledExtensionTimeout = setTimeout(function () {
+                        disabledExtensionTimeout = null;
+                        GSC.sync.onExtensionChanged(message);
+                    }, 1500);
+                }
+                else if (
+                    disabledExtensionTimeout &&
+                    message.parameters[EXTENSION_CHANGED_STATE] === EXTENSION_STATE.UNINSTALLED &&
+                    lastPortMessage.message.signal === SIGNAL_EXTENSION_CHANGED &&
+                    lastPortMessage.message.parameters[EXTENSION_CHANGED_UUID] === 
message.parameters[EXTENSION_CHANGED_UUID] &&
+                    lastPortMessage.message.parameters[EXTENSION_CHANGED_STATE] === EXTENSION_STATE.DISABLED
+                ) {
+                    clearTimeout(disabledExtensionTimeout);
+                    disabledExtensionTimeout = null;
+                    GSC.sync.onExtensionChanged(message);
+                }
+                else {
+                    GSC.sync.onExtensionChanged(message);
+                }
+            }
+
+            lastPortMessage = {
+                message: message,
+                date: new Date().getTime()
+            };
+        }
+        else if ([SIGNAL_NOTIFICATION_ACTION, SIGNAL_NOTIFICATION_CLICKED].indexOf(message.signal) != -1) {
+            window.postMessage(message, "*");
+        }
+    }
 });
 /*
  * Subscribe to GNOME Shell signals
  */
-port.postMessage({execute: 'subscribeSignals'});
+port.postMessage({ execute: 'subscribeSignals' });
 
 window.addEventListener("message", function (event) {
-               // We only accept messages from ourselves
-               if (event.source == window && event.data && event.data.execute)
-               {
-                       switch (event.data.execute)
-                       {
-                               case 'createNotification':
-                                       port.postMessage(event.data);
-                                       break;
-                               case 'removeNotification':
-                                       port.postMessage(event.data);
-                                       break;
-                       }
-               }
-       }
+    // We only accept messages from ourselves
+    if (event.source == window && event.data && event.data.execute) {
+        switch (event.data.execute) {
+            case 'createNotification':
+                port.postMessage(event.data);
+                break;
+            case 'removeNotification':
+                port.postMessage(event.data);
+                break;
+        }
+    }
+}
 );
 
-chrome.runtime.getPlatformInfo(function(info) {
-       if (PLATFORMS_WHITELIST.indexOf(info.os) !== -1)
-       {
-               GSC.update.init();
-               GSC.sync.init();
-       }
+chrome.runtime.getPlatformInfo(function (info) {
+    if (PLATFORMS_WHITELIST.indexOf(info.os) !== -1) {
+        GSC.update.init();
+        GSC.sync.init();
+    }
 });
 
 GSC.toolbar.init();
diff --git a/extension/include/compat-common.js b/extension/include/compat-common.js
index a0d17a5..0ad3cd3 100644
--- a/extension/include/compat-common.js
+++ b/extension/include/compat-common.js
@@ -6,106 +6,98 @@ $ = (...args) => document.querySelector(...args);
 $$ = (...args) => document.querySelectorAll(...args);
 
 function empty(element) {
-       while(element.firstChild) element.removeChild(element.firstChild);
+    while (element.firstChild) element.removeChild(element.firstChild);
 }
 
 function isEmptyObject(object) {
-       for (const k in object) return false;
-       return true;
+    for (const k in object) return false;
+    return true;
 }
 
 function showWithDelay(element, delay, message) {
-       if (message) {
-               element.innerHtml = message;
-       }
+    if (message) {
+        element.innerHtml = message;
+    }
 
-       element.classList.remove('hide');
-       element.classList.add('show');
-       setTimeout(() => {
-               element.classList.remove('show');
-               setTimeout(() => {
-                       element.classList.add('hide');
-               }, 250);
-       }, delay);
+    element.classList.remove('hide');
+    element.classList.add('show');
+    setTimeout(() => {
+        element.classList.remove('show');
+        setTimeout(() => {
+            element.classList.add('hide');
+        }, 250);
+    }, delay);
 }
 
-COMPAT.PERMISSIONS_CONTAINS            = true;
-COMPAT.PERMISSIONS_EVENTS              = true;
-COMPAT.SYNC_STORAGE                            = true;
-COMPAT.NOTIFICATIONS_BUTTONS   = (!COMPAT.IS_FIREFOX || false);
+COMPAT.PERMISSIONS_CONTAINS     = true;
+COMPAT.PERMISSIONS_EVENTS       = true;
+COMPAT.SYNC_STORAGE             = true;
+COMPAT.NOTIFICATIONS_BUTTONS    = (!COMPAT.IS_FIREFOX || false);
 
-if(COMPAT.IS_FIREFOX)
-{
-       chrome.runtime.onMessageExternal = {
-               addListener: chrome.runtime.onMessage.addListener
-       };
+if (COMPAT.IS_FIREFOX) {
+    chrome.runtime.onMessageExternal = {
+        addListener: chrome.runtime.onMessage.addListener
+    };
 }
 
-if(typeof(chrome.permissions) === 'undefined')
-{
-       chrome.permissions = {
-               contains: function(permissions, callback) {
-                       callback(false);
-               }
-       };
-       COMPAT.PERMISSIONS_CONTAINS = false;
+if (typeof (chrome.permissions) === 'undefined') {
+    chrome.permissions = {
+        contains: function (permissions, callback) {
+            callback(false);
+        }
+    };
+    COMPAT.PERMISSIONS_CONTAINS = false;
 }
 
-if(typeof(chrome.permissions.onAdded) === 'undefined' || typeof(chrome.permissions.onRemoved) === 
'undefined')
-{
-       chrome.permissions.onAdded = {
-               addListener: function(callback) {
-                       chrome.runtime.onMessage.addListener(
-                               function (request, sender, sendResponse) {
-                                       if (sender.id && sender.id === GS_CHROME_ID && request)
-                                       {
-                                               if (request === MESSAGE_IDLE_PERMISSION_ADDED)
-                                               {
-                                                       callback({
-                                                               permissions: ['idle']
-                                                       });
-                                               }
-                                       }
-                               }
-                       );
-               },
-               removeListener: function(callback) {
-                       chrome.runtime.onMessage.removeListener(callback);
-               },
-               hasListener: function(callback) {
-                       return chrome.runtime.onMessage.hasListener(callback);
-               }
-       }
+if (typeof (chrome.permissions.onAdded) === 'undefined' || typeof (chrome.permissions.onRemoved) === 
'undefined') {
+    chrome.permissions.onAdded = {
+        addListener: function (callback) {
+            chrome.runtime.onMessage.addListener(
+                function (request, sender, sendResponse) {
+                    if (sender.id && sender.id === GS_CHROME_ID && request) {
+                        if (request === MESSAGE_IDLE_PERMISSION_ADDED) {
+                            callback({
+                                permissions: ['idle']
+                            });
+                        }
+                    }
+                }
+            );
+        },
+        removeListener: function (callback) {
+            chrome.runtime.onMessage.removeListener(callback);
+        },
+        hasListener: function (callback) {
+            return chrome.runtime.onMessage.hasListener(callback);
+        }
+    }
 
-       chrome.permissions.onRemoved = {
-               addListener: function(callback) {
-                       chrome.runtime.onMessage.addListener(
-                               function (request, sender, sendResponse) {
-                                       if (sender.id && sender.id === GS_CHROME_ID && request)
-                                       {
-                                               if (request === MESSAGE_IDLE_PERMISSION_REMOVED)
-                                               {
-                                                       callback({
-                                                               permissions: ['idle']
-                                                       });
-                                               }
-                                       }
-                               }
-                       );
-               },
-               removeListener: function(callback) {
-                       chrome.runtime.onMessage.removeListener(callback);
-               },
-               hasListener: function(callback) {
-                       return chrome.runtime.onMessage.hasListener(callback);
-               }
-       }
+    chrome.permissions.onRemoved = {
+        addListener: function (callback) {
+            chrome.runtime.onMessage.addListener(
+                function (request, sender, sendResponse) {
+                    if (sender.id && sender.id === GS_CHROME_ID && request) {
+                        if (request === MESSAGE_IDLE_PERMISSION_REMOVED) {
+                            callback({
+                                permissions: ['idle']
+                            });
+                        }
+                    }
+                }
+            );
+        },
+        removeListener: function (callback) {
+            chrome.runtime.onMessage.removeListener(callback);
+        },
+        hasListener: function (callback) {
+            return chrome.runtime.onMessage.hasListener(callback);
+        }
+    }
 
-       COMPAT.PERMISSIONS_EVENTS = false;
+    COMPAT.PERMISSIONS_EVENTS = false;
 }
 
-if(typeof(chrome.storage.sync) === 'undefined')
-{
-       chrome.storage.sync = chrome.storage.local;
-       COMPAT.SYNC_STORAGE = false;
+if (typeof (chrome.storage.sync) === 'undefined') {
+    chrome.storage.sync = chrome.storage.local;
+    COMPAT.SYNC_STORAGE = false;
 }
diff --git a/extension/include/compat-content-script.js b/extension/include/compat-content-script.js
index f6870d2..448f55f 100644
--- a/extension/include/compat-content-script.js
+++ b/extension/include/compat-content-script.js
@@ -1,42 +1,38 @@
 // SPDX-License-Identifer: GPL-3.0-or-later
 
-if(!window.chrome)
-{
-       (function() {
-               // Define the API subset provided to the webpage
-               var externalMessaging = {
-                       runtime: {
-                               sendMessage: function (extensionId, message, options, responseCallback) {
-                                       if(extensionId !== chrome.runtime.id)
-                                       {
-                                               console.error('Wrong extension id provided.')
-                                               return;
-                                       }
+if (!window.chrome) {
+    (function () {
+        // Define the API subset provided to the webpage
+        var externalMessaging = {
+            runtime: {
+                sendMessage: function (extensionId, message, options, responseCallback) {
+                    if (extensionId !== chrome.runtime.id) {
+                        console.error('Wrong extension id provided.')
+                        return;
+                    }
 
-                                       if(typeof(options) === 'function')
-                                       {
-                                               responseCallback = options;
-                                               options = undefined;
-                                       }
+                    if (typeof (options) === 'function') {
+                        responseCallback = options;
+                        options = undefined;
+                    }
 
-                                       chrome.runtime.sendMessage(extensionId, message, options)
-                                               .then(result => {
-                                                       if(typeof(responseCallback) == 'function')
-                                                       {
-                                                               responseCallback(cloneInto(result, window));
-                                                       }
-                                               })
-                                               .catch(err => {
-                                                       console.error("firefox-external-messaging: 
runtime.sendMessage error", err);
-                                               });
-                               }
-                       }
-               };
+                    chrome.runtime.sendMessage(extensionId, message, options)
+                        .then(result => {
+                            if (typeof (responseCallback) == 'function') {
+                                responseCallback(cloneInto(result, window));
+                            }
+                        })
+                        .catch(err => {
+                            console.error("firefox-external-messaging: runtime.sendMessage error", err);
+                        });
+                }
+            }
+        };
 
-               // Inject the API in the webpage wrapped by this content script
-               // (exposed as `chrome.runtime.sendMessage({anyProp: "anyValue"}).then(reply => ..., err => 
...)`)
-               window.wrappedJSObject.chrome = cloneInto(externalMessaging, window, {
-                       cloneFunctions: true,
-               });
-       })();
+        // Inject the API in the webpage wrapped by this content script
+        // (exposed as `chrome.runtime.sendMessage({anyProp: "anyValue"}).then(reply => ..., err => ...)`)
+        window.wrappedJSObject.chrome = cloneInto(externalMessaging, window, {
+            cloneFunctions: true,
+        });
+    })();
 }
diff --git a/extension/include/constants.js b/extension/include/constants.js
index 87adf55..976a63b 100644
--- a/extension/include/constants.js
+++ b/extension/include/constants.js
@@ -1,73 +1,73 @@
 // SPDX-License-Identifer: GPL-3.0-or-later
 
-GS_CHROME_ID                           = chrome.runtime.id;
-PLATFORMS_WHITELIST                    = ["freebsd", "linux", "openbsd"];
+GS_CHROME_ID = chrome.runtime.id;
+PLATFORMS_WHITELIST = ["freebsd", "linux", "openbsd"];
 
 COMPAT = {
-       IS_FIREFOX:      CSS.supports("-moz-appearance: none"),
+    IS_FIREFOX: CSS.supports("-moz-appearance: none"),
 };
 
-NOTIFICATION_SYNC_FAILED               = 'gs-chrome-sync-fail';
-NOTIFICATION_UPDATE_AVAILABLE          = 'gs-chrome-update';
-ALARM_UPDATE_CHECK                     = 'gs-chrome-update-check';
+NOTIFICATION_SYNC_FAILED        = 'gs-chrome-sync-fail';
+NOTIFICATION_UPDATE_AVAILABLE   = 'gs-chrome-update';
+ALARM_UPDATE_CHECK              = 'gs-chrome-update-check';
 
-MESSAGE_NEXT_UPDATE_CHANGED            = 'gs-next-update-changed';
-MESSAGE_SYNC_FROM_REMOTE               = 'gs-sync-from-remote';
-MESSAGE_IDLE_PERMISSION_ADDED  = 'gs-idle-added';
-MESSAGE_IDLE_PERMISSION_REMOVED        = 'gs-idle-removed';
+MESSAGE_NEXT_UPDATE_CHANGED     = 'gs-next-update-changed';
+MESSAGE_SYNC_FROM_REMOTE        = 'gs-sync-from-remote';
+MESSAGE_IDLE_PERMISSION_ADDED   = 'gs-idle-added';
+MESSAGE_IDLE_PERMISSION_REMOVED = 'gs-idle-removed';
 
-SIGNAL_EXTENSION_CHANGED               = 'ExtensionStatusChanged';
-SIGNAL_NOTIFICATION_ACTION             = 'NotificationAction';
-SIGNAL_NOTIFICATION_CLICKED            = 'NotificationClicked';
-SIGNAL_SHELL_SETTING_CHANGED   = 'ShellSettingsChanged';
-SIGNAL_SHELL_APPEARED                  = 'org.gnome.Shell';
+SIGNAL_EXTENSION_CHANGED        = 'ExtensionStatusChanged';
+SIGNAL_NOTIFICATION_ACTION      = 'NotificationAction';
+SIGNAL_NOTIFICATION_CLICKED     = 'NotificationClicked';
+SIGNAL_SHELL_SETTING_CHANGED    = 'ShellSettingsChanged';
+SIGNAL_SHELL_APPEARED           = 'org.gnome.Shell';
 
-EXTENSION_CHANGED_UUID                 = 0;
-EXTENSION_CHANGED_STATE                        = 1;
-EXTENSION_CHANGED_ERROR                        = 2;
+EXTENSION_CHANGED_UUID          = 0;
+EXTENSION_CHANGED_STATE         = 1;
+EXTENSION_CHANGED_ERROR         = 2;
 
-NATIVE_HOST                            = 'org.gnome.chrome_gnome_shell';
+NATIVE_HOST = 'org.gnome.chrome_gnome_shell';
 
-EXTENSIONS_WEBSITE                     = [
-       'https://extensions.gnome.org/',
+EXTENSIONS_WEBSITE = [
+    'https://extensions.gnome.org/',
 ];
-UPDATE_URL                             = EXTENSIONS_WEBSITE + 'update-info/';
+UPDATE_URL = EXTENSIONS_WEBSITE + 'update-info/';
 
-DEFAULT_SYNC_OPTIONS                   = {
-       showReleaseNotes:       true,
-       updateCheck:            true,
-       updateCheckEnabledOnly: true,
-       updateCheckPeriod:      6
+DEFAULT_SYNC_OPTIONS = {
+    showReleaseNotes: true,
+    updateCheck: true,
+    updateCheckEnabledOnly: true,
+    updateCheckPeriod: 6
 };
 
-DEFAULT_LOCAL_OPTIONS                  = {
-       syncExtensions:         false,
-       useLightIcon:           false
+DEFAULT_LOCAL_OPTIONS = {
+    syncExtensions: false,
+    useLightIcon: false
 };
 
-EXTERNAL_MESSAGES                      = [
-       "error_connector_response",
-       "no_gnome_shell",
-       "no_host_connector",
-       "warning_apis_missing"
+EXTERNAL_MESSAGES = [
+    "error_connector_response",
+    "no_gnome_shell",
+    "no_host_connector",
+    "warning_apis_missing"
 ];
 
 // gnome-shell/js/ui/extensionSystem.js
-EXTENSION_STATE                        = {
-       ENABLED:        1,
-       DISABLED:       2,
-       ERROR:          3,
-       OUT_OF_DATE:    4,
-       DOWNLOADING:    5,
-       INITIALIZED:    6,
+EXTENSION_STATE = {
+    ENABLED:        1,
+    DISABLED:       2,
+    ERROR:          3,
+    OUT_OF_DATE:    4,
+    DOWNLOADING:    5,
+    INITIALIZED:    6,
 
-       // Used as an error state for operations on unknown extensions,
-       // should never be in a real extensionMeta object.
-       UNINSTALLED:    99
+    // Used as an error state for operations on unknown extensions,
+    // should never be in a real extensionMeta object.
+    UNINSTALLED:    99
 };
 
 // gnome-shell/js/misc/extensionUtils.js
-EXTENSION_TYPE                 = {
-    SYSTEM:            1,
-    PER_USER:  2
+EXTENSION_TYPE = {
+    SYSTEM:     1,
+    PER_USER:   2
 };
diff --git a/extension/include/gsc.js b/extension/include/gsc.js
index 7f52885..8e00907 100644
--- a/extension/include/gsc.js
+++ b/extension/include/gsc.js
@@ -1,140 +1,126 @@
 // SPDX-License-Identifer: GPL-3.0-or-later
 
-GSC = (function() {
-       var ready = new Promise(function(resolve, reject) {
-               chrome.runtime.getPlatformInfo(function(info) {
-                       if (PLATFORMS_WHITELIST.indexOf(info.os) === -1)
-                       {
-                               reject();
-                       }
-                       else
-                       {
-                               resolve();
-                       }
-               });
-       });
-       ready.catch(function() {});
-
-       var onInitialize = new Promise((resolve, reject) => {
-               sendNativeRequest({ execute: "initialize" }, (response) => {
-                       if(response && response.success)
-                       {
-                               resolve(response);
-                       }
-                       else
-                       {
-                               reject(response);
-                       }
-               });
-       });
-
-       function sendNativeRequest(request, sendResponse) {
-               ready.then(function () {
-                       if (sendResponse)
-                       {
-                               chrome.runtime.sendNativeMessage(
-                                       NATIVE_HOST,
-                                       request,
-                                       function (response) {
-                                               if (response)
-                                               {
-                                                       sendResponse(response);
-                                               }
-                                               else
-                                               {
-                                                       var message = m('no_host_connector');
-                                                       if (
-                                                               chrome.runtime.lastError &&
-                                                               chrome.runtime.lastError.message &&
-                                                               
chrome.runtime.lastError.message.indexOf("host not found") === -1 && // Chrome
-                                                               
chrome.runtime.lastError.message.indexOf("disconnected port") === -1 // Firefox
-                                                       )
-                                                       {
-                                                               // Some error occured. Show to user
-                                                               message = chrome.runtime.lastError.message;
-                                                       }
-
-                                                       sendResponse({
-                                                               success: false,
-                                                               message: message
-                                                       });
-                                               }
-                                       }
-                               );
-                       }
-                       else
-                       {
-                               chrome.runtime.sendNativeMessage(NATIVE_HOST, request);
-                       }
-               }, function () {
-                       if (sendResponse)
-                       {
-                               sendResponse({
-                                       success: false,
-                                       message: m('platform_not_supported')
-                               });
-                       }
-               });
-       }
-
-       function isSupported(feature, response) {
-               return response.properties &&
-                               response.properties.supports && response.properties.supports.indexOf(feature) 
!== -1;
-       }
-
-       return {
-               // https://wiki.gnome.org/Projects/GnomeShell/Extensions/UUIDGuidelines
-               isUUID: function(uuid) {
-                       return uuid && uuid.match('^[-a-zA-Z0-9@._]+$');
-               },
-
-               sendNativeRequest: sendNativeRequest,
-
-               isSignalsEqual: function(newSignal, oldSignal) {
-                       if(!oldSignal || !newSignal)
-                               return false;
-
-                       if(!newSignal.signal || !oldSignal.signal || newSignal.signal !== oldSignal.signal)
-                               return false;
-
-                       if(newSignal.parameters)
-                       {
-                               if(!oldSignal.parameters)
-                                       return false;
-
-                               if(newSignal.parameters.length !== oldSignal.parameters.length)
-                                       return false;
-
-                               for(var i = 0; i < newSignal.parameters.length; i++)
-                               {
-                                       if(newSignal.parameters[i] !== oldSignal.parameters[i])
-                                       {
-                                               return false;
-                                       }
-                               }
-                       }
-                       else if (oldSignal.parameters)
-                       {
-                               return false;
-                       }
-
-                       return true;
-               },
-
-               onInitialize: function() {
-                       return onInitialize;
-               },
-
-               nativeNotificationsSupported: function (response) {
-                       return isSupported('notifications', response);
-               },
-
-               nativeUpdateCheckSupported: function (response) {
-                       return isSupported('update-check', response);
-               },
-
-               nativeUpdateCheckEnabledOnlySupported: function (response) {
-                       return isSupported('update-enabled', response);
-               }
-       };
+GSC = (function () {
+    var ready = new Promise(function (resolve, reject) {
+        chrome.runtime.getPlatformInfo(function (info) {
+            if (PLATFORMS_WHITELIST.indexOf(info.os) === -1) {
+                reject();
+            }
+            else {
+                resolve();
+            }
+        });
+    });
+    ready.catch(function () { });
+
+    var onInitialize = new Promise((resolve, reject) => {
+        sendNativeRequest({ execute: "initialize" }, (response) => {
+            if (response && response.success) {
+                resolve(response);
+            }
+            else {
+                reject(response);
+            }
+        });
+    });
+
+    function sendNativeRequest(request, sendResponse) {
+        ready.then(function () {
+            if (sendResponse) {
+                chrome.runtime.sendNativeMessage(
+                    NATIVE_HOST,
+                    request,
+                    function (response) {
+                        if (response) {
+                            sendResponse(response);
+                        }
+                        else {
+                            var message = m('no_host_connector');
+                            if (
+                                chrome.runtime.lastError &&
+                                chrome.runtime.lastError.message &&
+                                chrome.runtime.lastError.message.indexOf("host not found") === -1 && // 
Chrome
+                                chrome.runtime.lastError.message.indexOf("disconnected port") === -1 // 
Firefox
+                            ) {
+                                // Some error occured. Show to user
+                                message = chrome.runtime.lastError.message;
+                            }
+
+                            sendResponse({
+                                success: false,
+                                message: message
+                            });
+                        }
+                    }
+                );
+            }
+            else {
+                chrome.runtime.sendNativeMessage(NATIVE_HOST, request);
+            }
+        }, function () {
+            if (sendResponse) {
+                sendResponse({
+                    success: false,
+                    message: m('platform_not_supported')
+                });
+            }
+        });
+    }
+
+    function isSupported(feature, response) {
+        return response.properties &&
+            response.properties.supports && response.properties.supports.indexOf(feature) !== -1;
+    }
+
+    return {
+        // https://wiki.gnome.org/Projects/GnomeShell/Extensions/UUIDGuidelines
+        isUUID: function (uuid) {
+            return uuid && uuid.match('^[-a-zA-Z0-9@._]+$');
+        },
+
+        sendNativeRequest: sendNativeRequest,
+
+        isSignalsEqual: function (newSignal, oldSignal) {
+            if (!oldSignal || !newSignal)
+                return false;
+
+            if (!newSignal.signal || !oldSignal.signal || newSignal.signal !== oldSignal.signal)
+                return false;
+
+            if (newSignal.parameters) {
+                if (!oldSignal.parameters)
+                    return false;
+
+                if (newSignal.parameters.length !== oldSignal.parameters.length)
+                    return false;
+
+                for (var i = 0; i < newSignal.parameters.length; i++) {
+                    if (newSignal.parameters[i] !== oldSignal.parameters[i]) {
+                        return false;
+                    }
+                }
+            }
+            else if (oldSignal.parameters) {
+                return false;
+            }
+
+            return true;
+        },
+
+        onInitialize: function () {
+            return onInitialize;
+        },
+
+        nativeNotificationsSupported: function (response) {
+            return isSupported('notifications', response);
+        },
+
+        nativeUpdateCheckSupported: function (response) {
+            return isSupported('update-check', response);
+        },
+
+        nativeUpdateCheckEnabledOnlySupported: function (response) {
+            return isSupported('update-enabled', response);
+        }
+    };
 })();
diff --git a/extension/include/i18n.js b/extension/include/i18n.js
index ba9ac4a..b32fbd3 100644
--- a/extension/include/i18n.js
+++ b/extension/include/i18n.js
@@ -2,31 +2,26 @@
 
 m = chrome.i18n.getMessage;
 i18n = (() => {
-       $$('[data-i18n]').forEach((element) => {
-               let data = element.dataset.i18n.split(',').map((value) => {
-                       value = value.trim();
+    $$('[data-i18n]').forEach((element) => {
+        let data = element.dataset.i18n.split(',').map((value) => {
+            value = value.trim();
 
-                       if(value.startsWith('__MSG_'))
-                       {
-                               return value.replace(/__MSG_(\w+)__/g, function(match, key)
-                               {
-                                       return key ? m(key) : "";
-                               });
-                       }
+            if (value.startsWith('__MSG_')) {
+                return value.replace(/__MSG_(\w+)__/g, function (match, key) {
+                    return key ? m(key) : "";
+                });
+            }
 
-                       return value;
-               });
+            return value;
+        });
 
-               if(data)
-               {
-                       if(element.dataset.i18nHtml)
-                       {
-                               element.innerHtml = m(data[0], data.slice(1));
-                       }
-                       else
-                       {
-                               element.innerText = m(data[0], data.slice(1));
-                       }
-               }
-       });
+        if (data) {
+            if (element.dataset.i18nHtml) {
+                element.innerHtml = m(data[0], data.slice(1));
+            }
+            else {
+                element.innerText = m(data[0], data.slice(1));
+            }
+        }
+    });
 });
diff --git a/extension/include/notifications.js b/extension/include/notifications.js
index d2631cc..05fb445 100644
--- a/extension/include/notifications.js
+++ b/extension/include/notifications.js
@@ -1,210 +1,192 @@
 // SPDX-License-Identifer: GPL-3.0-or-later
 
-GSC.notifications = (function() {
-       var DEFAULT_NOTIFICATION_OPTIONS = {
-               type: chrome.notifications.TemplateType.BASIC,
-               iconUrl: 'icons/GnomeLogo-128.png',
-               title: m('gs_chrome'),
-               buttons: [
-                       {title: m('close')}
-               ],
-               priority: 2,
-               isClickable: true,
-               requireInteraction: true
-       };
-
-       function remove_list(options) {
-               if(options.items)
-               {
-                       var items = [];
-                       for (k in options.items)
-                       {
-                               if (options.items.hasOwnProperty(k))
-                               {
-                                       items.push(options.items[k].title + ' ' + options.items[k].message);
-                               }
-                       }
-
-                       if(options.message && items)
-                       {
-                               options.message += "\n";
-                       }
-
-                       options.message += items.join("\n");
-
-                       options.type = chrome.notifications.TemplateType.BASIC;
-                       delete options.items;
-               }
-
-               return options;
-       }
-
-       /*
-               @Deprecated: remove browser notifications in version 9
-        */
-       var browser = (function() {
-               function init() {
-                       chrome.runtime.onStartup.addListener(function() {
-                               // Do nothing. We just need this callback to restore notifications
-                       });
-
-                       chrome.notifications.onClosed.addListener(function (notificationId, byUser) {
-                               if (!byUser)
-                               {
-                                       update(notificationId);
-                               }
-                               else
-                               {
-                                       remove(notificationId);
-                               }
-                       });
-
-                       chrome.notifications.onClicked.addListener(function (notificationId) {
-                               GSC.notifications.remove(notificationId);
-                       });
-
-                       restore();
-               }
-
-               function create(name, options) {
-                       chrome.storage.local.get({
-                               notifications: {}
-                       }, function (items) {
-                               var notifications = items.notifications;
-
-                               notifications[name] = {...DEFAULT_NOTIFICATION_OPTIONS, ...options};
-
-                               _create(name, notifications[name], function (notificationId) {
-                                       chrome.storage.local.set({
-                                               notifications: notifications
-                                       });
-                               });
-                       });
-               }
-
-               function _create(name, options, callback)
-               {
-                       if(!COMPAT.NOTIFICATIONS_BUTTONS && options.buttons)
-                       {
-                               delete options.buttons;
-                       }
-
-                       if (callback)
-                       {
-                               chrome.notifications.create(name, options, callback);
-                       }
-                       else
-                       {
-                               chrome.notifications.create(name, options);
-                       }
-               }
-
-               function update(notificationId) {
-                       chrome.storage.local.get({
-                               notifications: {}
-                       }, function (items) {
-                               var notifications = items.notifications;
-
-                               if (notifications[notificationId])
-                               {
-                                       _create(notificationId, notifications[notificationId]);
-                               }
-                       });
-               }
-
-               function remove(notificationId) {
-                       chrome.storage.local.get({
-                               notifications: {}
-                       }, function (items) {
-                               var notifications = items.notifications;
-
-                               if (notifications[notificationId])
-                               {
-                                       delete notifications[notificationId];
-                                       chrome.storage.local.set({
-                                               notifications: notifications
-                                       });
-                               }
-
-                               chrome.notifications.clear(notificationId);
-                       });
-               }
-
-               function restore() {
-                       chrome.storage.local.get({
-                               notifications: {}
-                       }, function (items) {
-                               var notifications = items.notifications;
-
-                               for (notificationId in notifications)
-                               {
-                                       update(notificationId);
-                               }
-                       });
-               }
-
-               return {
-                       create: create,
-                       remove: remove,
-                       init: init
-               };
-       })();
-
-       var native = (function() {
-               function create(name, options) {
-                       options = remove_list(options);
-
-                       window.postMessage({
-                               execute: 'createNotification',
-                               name: name,
-                               options: {...DEFAULT_NOTIFICATION_OPTIONS, ...options}
-                       }, "*");
-               }
-
-               function remove(notificationId) {
-                       chrome.runtime.sendMessage({
-                               execute: 'removeNotification',
-                               name: notificationId
-                       });
-               }
-
-               return {
-                       create: create,
-                       remove: remove
-               };
-       })();
-
-       GSC.onInitialize().then(response => {
-               if (!GSC.nativeNotificationsSupported(response))
-               {
-                       browser.init();
-               }
-       });
-
-       return {
-               create: function() {
-                       GSC.onInitialize().then(response => {
-                               if(GSC.nativeNotificationsSupported(response))
-                               {
-                                       native.create.apply(this, arguments);
-                               }
-                               else
-                               {
-                                       browser.create.apply(this, arguments);
-                               }
-                       });
-               },
-               remove: function() {
-                       GSC.onInitialize().then(response => {
-                               if(GSC.nativeNotificationsSupported(response))
-                               {
-                                       native.remove.apply(this, arguments);
-                               }
-                               else
-                               {
-                                       browser.remove.apply(this, arguments);
-                               }
-                       });
-               }
-       };
+GSC.notifications = (function () {
+    var DEFAULT_NOTIFICATION_OPTIONS = {
+        type: chrome.notifications.TemplateType.BASIC,
+        iconUrl: 'icons/GnomeLogo-128.png',
+        title: m('gs_chrome'),
+        buttons: [
+            { title: m('close') }
+        ],
+        priority: 2,
+        isClickable: true,
+        requireInteraction: true
+    };
+
+    function remove_list(options) {
+        if (options.items) {
+            var items = [];
+            for (k in options.items) {
+                if (options.items.hasOwnProperty(k)) {
+                    items.push(options.items[k].title + ' ' + options.items[k].message);
+                }
+            }
+
+            if (options.message && items) {
+                options.message += "\n";
+            }
+
+            options.message += items.join("\n");
+
+            options.type = chrome.notifications.TemplateType.BASIC;
+            delete options.items;
+        }
+
+        return options;
+    }
+
+    /*
+        @Deprecated: remove browser notifications in version 9
+     */
+    var browser = (function () {
+        function init() {
+            chrome.runtime.onStartup.addListener(function () {
+                // Do nothing. We just need this callback to restore notifications
+            });
+
+            chrome.notifications.onClosed.addListener(function (notificationId, byUser) {
+                if (!byUser) {
+                    update(notificationId);
+                }
+                else {
+                    remove(notificationId);
+                }
+            });
+
+            chrome.notifications.onClicked.addListener(function (notificationId) {
+                GSC.notifications.remove(notificationId);
+            });
+
+            restore();
+        }
+
+        function create(name, options) {
+            chrome.storage.local.get({
+                notifications: {}
+            }, function (items) {
+                var notifications = items.notifications;
+
+                notifications[name] = { ...DEFAULT_NOTIFICATION_OPTIONS, ...options };
+
+                _create(name, notifications[name], function (notificationId) {
+                    chrome.storage.local.set({
+                        notifications: notifications
+                    });
+                });
+            });
+        }
+
+        function _create(name, options, callback) {
+            if (!COMPAT.NOTIFICATIONS_BUTTONS && options.buttons) {
+                delete options.buttons;
+            }
+
+            if (callback) {
+                chrome.notifications.create(name, options, callback);
+            }
+            else {
+                chrome.notifications.create(name, options);
+            }
+        }
+
+        function update(notificationId) {
+            chrome.storage.local.get({
+                notifications: {}
+            }, function (items) {
+                var notifications = items.notifications;
+
+                if (notifications[notificationId]) {
+                    _create(notificationId, notifications[notificationId]);
+                }
+            });
+        }
+
+        function remove(notificationId) {
+            chrome.storage.local.get({
+                notifications: {}
+            }, function (items) {
+                var notifications = items.notifications;
+
+                if (notifications[notificationId]) {
+                    delete notifications[notificationId];
+                    chrome.storage.local.set({
+                        notifications: notifications
+                    });
+                }
+
+                chrome.notifications.clear(notificationId);
+            });
+        }
+
+        function restore() {
+            chrome.storage.local.get({
+                notifications: {}
+            }, function (items) {
+                var notifications = items.notifications;
+
+                for (notificationId in notifications) {
+                    update(notificationId);
+                }
+            });
+        }
+
+        return {
+            create: create,
+            remove: remove,
+            init: init
+        };
+    })();
+
+    var native = (function () {
+        function create(name, options) {
+            options = remove_list(options);
+
+            window.postMessage({
+                execute: 'createNotification',
+                name: name,
+                options: { ...DEFAULT_NOTIFICATION_OPTIONS, ...options }
+            }, "*");
+        }
+
+        function remove(notificationId) {
+            chrome.runtime.sendMessage({
+                execute: 'removeNotification',
+                name: notificationId
+            });
+        }
+
+        return {
+            create: create,
+            remove: remove
+        };
+    })();
+
+    GSC.onInitialize().then(response => {
+        if (!GSC.nativeNotificationsSupported(response)) {
+            browser.init();
+        }
+    });
+
+    return {
+        create: function () {
+            GSC.onInitialize().then(response => {
+                if (GSC.nativeNotificationsSupported(response)) {
+                    native.create.apply(this, arguments);
+                }
+                else {
+                    browser.create.apply(this, arguments);
+                }
+            });
+        },
+        remove: function () {
+            GSC.onInitialize().then(response => {
+                if (GSC.nativeNotificationsSupported(response)) {
+                    native.remove.apply(this, arguments);
+                }
+                else {
+                    browser.remove.apply(this, arguments);
+                }
+            });
+        }
+    };
 })();
diff --git a/extension/include/sweettooth-api.js b/extension/include/sweettooth-api.js
index 1929d00..b7c197e 100644
--- a/extension/include/sweettooth-api.js
+++ b/extension/include/sweettooth-api.js
@@ -3,220 +3,197 @@
 "use strict";
 
 GSC.getMessage = function (key) {
-       if (GSC && GSC.i18n && GSC.i18n[key])
-       {
-               var message = GSC.i18n[key];
+    if (GSC && GSC.i18n && GSC.i18n[key]) {
+        var message = GSC.i18n[key];
 
-               for (var i = 1; i < arguments.length; i++)
-               {
-                       message = message.replace('$' + i, arguments[i]);
-               }
+        for (var i = 1; i < arguments.length; i++) {
+            message = message.replace('$' + i, arguments[i]);
+        }
 
-               return message;
-       }
+        return message;
+    }
 
-       return key;
+    return key;
 };
 
 window.SweetTooth = function () {
-       var apiObject = {
-               apiVersion: 5,
-               shellVersion: '-1',
-               versionValidationEnabled: true,
-               userExtensionsDisabled: false,
-
-               getChromeExtensionId: function () {
-                       return GS_CHROME_ID;
-               },
-
-               getExtensionErrors: function (uuid) {
-                       return sendResolveExtensionMessage("getExtensionErrors", "extensionErrors", {uuid: 
uuid});
-               },
-
-               getExtensionInfo: function (uuid) {
-                       return sendResolveExtensionMessage("getExtensionInfo", "extensionInfo", {uuid: uuid});
-               },
-
-               installExtension: function (uuid) {
-                       return sendResolveExtensionMessage("installExtension", "status", {uuid: uuid});
-               },
-
-               launchExtensionPrefs: function (uuid) {
-                       sendExtensionMessage("launchExtensionPrefs", null, {uuid: uuid});
-               },
-
-               listExtensions: function () {
-                       return sendResolveExtensionMessage("listExtensions", "extensions");
-               },
-
-               setExtensionEnabled: function (uuid, enable) {
-                       return sendResolveExtensionMessage("enableExtension", "success", {uuid: uuid, enable: 
enable});
-               },
-
-               uninstallExtension: function (uuid) {
-                       return sendResolveExtensionMessage("uninstallExtension", "status", {uuid: uuid});
-               },
-
-               setUserExtensionsDisabled: function (disable) {
-                       return sendResolveExtensionMessage("setUserExtensionsDisabled", "success", {disable: 
disable})
-                               .then(success => {
-                                       if(success)
-                                       {
-                                               apiObject.userExtensionsDisabled = disable;
-                                       }
-
-                                       return success;
-                               });
-               },
-
-               setVersionValidationDisabled: function (disable) {
-                       return sendResolveExtensionMessage("setVersionValidationDisabled", "success", 
{disable: disable})
-                               .then(success => {
-                                       if(success)
-                                       {
-                                               apiObject.versionValidationEnabled = !disable;
-                                       }
-
-                                       return success;
-                               });
-               },
-
-               initialize: function () {
-                       if (SweetTooth.shellVersion !== '-1')
-                       {
-                               return Promise.resolve(apiObject);
-                       }
-
-                       var ready = new Promise(function(resolve, reject) {
-                               sendExtensionMessage("initialize", response => {
-                                       if (response && response.success && response.properties && 
response.properties.shellVersion)
-                                       {
-                                               resolve(response.properties);
-                                       }
-                                       else
-                                       {
-                                               var message = response && response.message ? 
GSC.getMessage(response.message) : GSC.getMessage('error_connector_response');
-                                               reject(message);
-                                       }
-                               });
-                       });
-
-                       ready.then(function (response) {
-                               apiObject.shellVersion = response.shellVersion;
-                               apiObject.versionValidationEnabled = response.versionValidationEnabled;
-                               apiObject.userExtensionsDisabled = response.userExtensionsDisabled;
-
-                               let REQUIRED_APIS = [
-                                       "notifications",
-                                       "v6"
-                               ];
-
-                               if(response.supports)
-                               {
-                                       for(let api of response.supports)
-                                       {
-                                               let api_index;
-                                               if((api_index = REQUIRED_APIS.indexOf(api)) != -1)
-                                               {
-                                                       REQUIRED_APIS.splice(api_index, 1);
-                                               }
-
-                                               if(api === 'v6')
-                                               {
-                                                       apiObject.apiVersion = 6;
-                                               }
-                                       }
-                               }
-
-                               if (REQUIRED_APIS.length > 0)
-                               {
-                                       require(['messages'], function (messages) {
-                                               messages.addWarning(GSC.getMessage('warning_apis_missing', 
REQUIRED_APIS.join(", ")));
-                                       });
-                               }
-                       }, function (message) {
-                               apiObject.apiVersion = null;
-
-                               require(['messages'], function (messages) {
-                                       messages.addError(message ? message : 
GSC.getMessage('no_host_connector'));
-                               })
-                       });
-
-                       return ready;
-               }
-       };
-
-       window.addEventListener("message", function (event) {
-               // We only accept messages from ourselves
-               if (event.source != window)
-               {
-                       return;
-               }
-
-               if (event.data.type)
-               {
-                       if (event.data.type == "gs-chrome")
-                       {
-                               if (event.data.request.signal == 'ExtensionStatusChanged' && 
apiObject.onchange)
-                               {
-                                       apiObject.onchange(
-                                               event.data.request.parameters[0],
-                                               event.data.request.parameters[1],
-                                               event.data.request.parameters[2]
-                                       );
-                               }
-                               else if (event.data.request.signal == 'org.gnome.Shell' && 
apiObject.onshellrestart)
-                               {
-                                       apiObject.onshellrestart();
-                               }
-                               else if (event.data.request.signal == 'ShellSettingsChanged' && 
apiObject.onShellSettingChanged)
-                               {
-                                       if(event.data.request.key === 'disable-user-extensions')
-                                       {
-                                               apiObject.userExtensionsDisabled = event.data.request.value;
-                                       }
-                                       else if(event.data.request.key === 
'disable-extension-version-validation')
-                                       {
-                                               apiObject.versionValidationEnabled = 
!event.data.request.value;
-                                       }
-
-                                       apiObject.onShellSettingChanged(event.data.request.key, 
event.data.request.value);
-                               }
-                       }
-               }
-       }, false);
-
-       function sendResolveExtensionMessage(method, resolveProperty, parameters) {
-               return new Promise(function (resolve, reject) {
-                       sendExtensionMessage(method, function (response) {
-                                       if (response && response.success)
-                                       {
-                                               resolve(response[resolveProperty]);
-                                       }
-                                       else
-                                       {
-                                               var message = response && response.message ? response.message 
: GSC.getMessage('error_connector_response');
-                                               reject(message);
-                                       }
-                               },
-                               parameters
-                       );
-               });
-       }
-
-       function sendExtensionMessage(method, callback, parameters) {
-               var request = {execute: method};
-               if (parameters)
-               {
-                       request = Object.assign(parameters, request);
-               }
-
-               chrome.runtime.sendMessage(
-                       apiObject.getChromeExtensionId(),
-                       request,
-                       callback
-               );
-       }
-
-       return apiObject;
+    var apiObject = {
+        apiVersion: 5,
+        shellVersion: '-1',
+        versionValidationEnabled: true,
+        userExtensionsDisabled: false,
+
+        getChromeExtensionId: function () {
+            return GS_CHROME_ID;
+        },
+
+        getExtensionErrors: function (uuid) {
+            return sendResolveExtensionMessage("getExtensionErrors", "extensionErrors", { uuid: uuid });
+        },
+
+        getExtensionInfo: function (uuid) {
+            return sendResolveExtensionMessage("getExtensionInfo", "extensionInfo", { uuid: uuid });
+        },
+
+        installExtension: function (uuid) {
+            return sendResolveExtensionMessage("installExtension", "status", { uuid: uuid });
+        },
+
+        launchExtensionPrefs: function (uuid) {
+            sendExtensionMessage("launchExtensionPrefs", null, { uuid: uuid });
+        },
+
+        listExtensions: function () {
+            return sendResolveExtensionMessage("listExtensions", "extensions");
+        },
+
+        setExtensionEnabled: function (uuid, enable) {
+            return sendResolveExtensionMessage("enableExtension", "success", { uuid: uuid, enable: enable });
+        },
+
+        uninstallExtension: function (uuid) {
+            return sendResolveExtensionMessage("uninstallExtension", "status", { uuid: uuid });
+        },
+
+        setUserExtensionsDisabled: function (disable) {
+            return sendResolveExtensionMessage("setUserExtensionsDisabled", "success", { disable: disable })
+                .then(success => {
+                    if (success) {
+                        apiObject.userExtensionsDisabled = disable;
+                    }
+
+                    return success;
+                });
+        },
+
+        setVersionValidationDisabled: function (disable) {
+            return sendResolveExtensionMessage("setVersionValidationDisabled", "success", { disable: disable 
})
+                .then(success => {
+                    if (success) {
+                        apiObject.versionValidationEnabled = !disable;
+                    }
+
+                    return success;
+                });
+        },
+
+        initialize: function () {
+            if (SweetTooth.shellVersion !== '-1') {
+                return Promise.resolve(apiObject);
+            }
+
+            var ready = new Promise(function (resolve, reject) {
+                sendExtensionMessage("initialize", response => {
+                    if (response && response.success && response.properties && 
response.properties.shellVersion) {
+                        resolve(response.properties);
+                    }
+                    else {
+                        var message = response && response.message ? GSC.getMessage(response.message) : 
GSC.getMessage('error_connector_response');
+                        reject(message);
+                    }
+                });
+            });
+
+            ready.then(function (response) {
+                apiObject.shellVersion = response.shellVersion;
+                apiObject.versionValidationEnabled = response.versionValidationEnabled;
+                apiObject.userExtensionsDisabled = response.userExtensionsDisabled;
+
+                let REQUIRED_APIS = [
+                    "notifications",
+                    "v6"
+                ];
+
+                if (response.supports) {
+                    for (let api of response.supports) {
+                        let api_index;
+                        if ((api_index = REQUIRED_APIS.indexOf(api)) != -1) {
+                            REQUIRED_APIS.splice(api_index, 1);
+                        }
+
+                        if (api === 'v6') {
+                            apiObject.apiVersion = 6;
+                        }
+                    }
+                }
+
+                if (REQUIRED_APIS.length > 0) {
+                    require(['messages'], function (messages) {
+                        messages.addWarning(GSC.getMessage('warning_apis_missing', REQUIRED_APIS.join(", 
")));
+                    });
+                }
+            }, function (message) {
+                apiObject.apiVersion = null;
+
+                require(['messages'], function (messages) {
+                    messages.addError(message ? message : GSC.getMessage('no_host_connector'));
+                })
+            });
+
+            return ready;
+        }
+    };
+
+    window.addEventListener("message", function (event) {
+        // We only accept messages from ourselves
+        if (event.source != window) {
+            return;
+        }
+
+        if (event.data.type) {
+            if (event.data.type == "gs-chrome") {
+                if (event.data.request.signal == 'ExtensionStatusChanged' && apiObject.onchange) {
+                    apiObject.onchange(
+                        event.data.request.parameters[0],
+                        event.data.request.parameters[1],
+                        event.data.request.parameters[2]
+                    );
+                }
+                else if (event.data.request.signal == 'org.gnome.Shell' && apiObject.onshellrestart) {
+                    apiObject.onshellrestart();
+                }
+                else if (event.data.request.signal == 'ShellSettingsChanged' && 
apiObject.onShellSettingChanged) {
+                    if (event.data.request.key === 'disable-user-extensions') {
+                        apiObject.userExtensionsDisabled = event.data.request.value;
+                    }
+                    else if (event.data.request.key === 'disable-extension-version-validation') {
+                        apiObject.versionValidationEnabled = !event.data.request.value;
+                    }
+
+                    apiObject.onShellSettingChanged(event.data.request.key, event.data.request.value);
+                }
+            }
+        }
+    }, false);
+
+    function sendResolveExtensionMessage(method, resolveProperty, parameters) {
+        return new Promise(function (resolve, reject) {
+            sendExtensionMessage(method, function (response) {
+                if (response && response.success) {
+                    resolve(response[resolveProperty]);
+                }
+                else {
+                    var message = response && response.message ? response.message : 
GSC.getMessage('error_connector_response');
+                    reject(message);
+                }
+            },
+                parameters
+            );
+        });
+    }
+
+    function sendExtensionMessage(method, callback, parameters) {
+        var request = { execute: method };
+        if (parameters) {
+            request = Object.assign(parameters, request);
+        }
+
+        chrome.runtime.sendMessage(
+            apiObject.getChromeExtensionId(),
+            request,
+            callback
+        );
+    }
+
+    return apiObject;
 }();
diff --git a/extension/include/sync.js b/extension/include/sync.js
index 37ff0f5..dc390f9 100644
--- a/extension/include/sync.js
+++ b/extension/include/sync.js
@@ -3,445 +3,395 @@
 /*
  * Main object that handles extensions synchronization with remote storage.
  */
-GSC.sync = (function($) {
-       var enabled = true;
-       var extensionChangedTimeout = false;
-       var extensionChangedQueue = {};
-
-       const SYNC_QUEUE_TIMEOUT = 7000;
-
-       /*
-        * Initialization rutines.
-        */
-       function init() {
-               if(!COMPAT.SYNC_STORAGE)
-               {
-                       enabled = false;
-                       return;
-               }
-
-               function onIdleStateChanged(state) {
-                       if (state === 'locked')
-                       {
-                               enabled = false;
-
-                               // Remove all disabled extensions from queue
-                               for (const [extensionId, extension] of Object.entries(extensionChangedQueue)) 
{
-                                       if(extension.state == EXTENSION_STATE.DISABLED)
-                                       {
-                                               delete extensionChangedQueue[extensionId];
-                                       }
-                               };
-                       }
-                       else if (state === 'active')
-                       {
-                               enabled = true;
-                       }
-               }
-
-               chrome.permissions.contains({
-                       permissions: ["idle"]
-               }, function (result) {
-                       if(result)
-                       {
-                               chrome.idle.onStateChanged.addListener(onIdleStateChanged);
-                       }
-                       else
-                       {
-                               enabled = false;
-                       }
-               });
-
-               chrome.permissions.onAdded.addListener(function(permissions) {
-                       if(permissions.permissions && permissions.permissions.indexOf('idle') !== -1)
-                       {
-                               enabled = true;
-                               chrome.idle.onStateChanged.addListener(onIdleStateChanged);
-                       }
-               });
-
-               chrome.permissions.onRemoved.addListener(function(permissions) {
-                       if(permissions.permissions && permissions.permissions.indexOf('idle') !== -1)
-                       {
-                               enabled = false;
-                               chrome.idle.onStateChanged.removeListener(onIdleStateChanged);
-                       }
-               });
-
-               function onNotificationAction(notificationId, buttonIndex) {
-                       if (notificationId !== NOTIFICATION_SYNC_FAILED)
-                       {
-                               return;
-                       }
-
-                       GSC.notifications.remove(notificationId);
-               }
-
-               onSyncFromRemote();
-               chrome.storage.onChanged.addListener(function(changes, areaName) {
-                       if(areaName === 'sync' && changes.extensions)
-                       {
-                               onSyncFromRemote(changes.extensions.newValue);
-                       }
-               });
-
-               chrome.runtime.onMessage.addListener(
-                       function (request, sender, sendResponse) {
-                               if (sender.id && sender.id === GS_CHROME_ID && request)
-                               {
-                                       if (request === MESSAGE_SYNC_FROM_REMOTE)
-                                       {
-                                               onSyncFromRemote();
-                                       }
-                               }
-                       }
-               );
-
-               GSC.onInitialize().then(response => {
-                       /*
-                               @Deprecated: remove browser notifications in version 9
-                        */
-                       if (!GSC.nativeNotificationsSupported(response))
-                       {
-                               chrome.notifications.onButtonClicked.addListener(onNotificationAction);
-                       }
-                       else
-                       {
-                               chrome.runtime.onMessage.addListener(
-                                       function (request, sender, sendResponse) {
-                                               if(
-                                                       sender.id && sender.id === GS_CHROME_ID &&
-                                                       request && request.signal)
-                                               {
-                                                       if(request.signal == SIGNAL_NOTIFICATION_ACTION)
-                                                       {
-                                                               onNotificationAction(request.name, 
request.button_id);
-                                                       }
-                                               }
-                                       }
-                               );
-                       }
-               });
-       }
-
-       /*
-        * Returns array of all local and remote extensions with structure:
-        * [
-        *      $extension_uuid: {
-        *              uuid:           extension uuid,
-        *              name:           extension name,
-        *              remoteState:    extension state in remote storage,
-        *              localState:     extension state in current GNOME Shell,
-        *              remote:         true if extensions is in remote storage,
-        *              local:          true if extension installed localy
-        *      },
-        *      ...
-        * ]
-        */
-       function getExtensions(remoteExtensions) {
-               return new Promise((resolve, reject) => {
-                       GSC.sendNativeRequest({
-                               execute: 'listExtensions'
-                       }, function(response) {
-                               if(response && response.success)
-                               {
-                                       if(remoteExtensions)
-                                       {
-                                               resolve(mergeExtensions(remoteExtensions, 
response.extensions));
-                                       }
-                                       else
-                                       {
-                                               chrome.storage.sync.get({
-                                                       extensions: {}
-                                               }, function(options) {
-                                                       if(chrome.runtime.lastError)
-                                                       {
-                                                               reject(chrome.runtime.lastError.message);
-                                                       }
-                                                       else
-                                                       {
-                                                               resolve(mergeExtensions(options.extensions, 
response.extensions));
-                                                       }
-                                               });
-                                       }
-                               }
-                               else
-                               {
-                                       var message = response && response.message ? response.message : 
m('error_connector_response');
-                                       reject(message);
-                               }
-                       });
-               });
-       }
-
-       /*
-        * Returns merged list of extensions list in remote storage and
-        * locally installed extensions.
-        * 
-        * Both parameters should be in form:
-        * {
-        *      $extension_uuid: {
-        *              uuid:   ,
-        *              name:   ,
-        *              state:  
-        *      },
-        *      ...
-        * }
-        */
-       function mergeExtensions(remoteExtensions, localExtensions)
-       {
-               var extensions = {};
-
-               for (const [key, extension] of Object.entries(remoteExtensions)) {
-                       if(extension.uuid && extension.name && extension.state)
-                       {
-                               extensions[extension.uuid] = {
-                                       uuid:           extension.uuid,
-                                       name:           extension.name,
-                                       remoteState:    extension.state,
-                                       remote:         true,
-                                       local:          false
-                               };
-                       }
-               };
-
-               for (const [key, extension] of Object.entries(localExtensions)) {
-                       if(extensions[extension.uuid])
-                       {
-                               extensions[extension.uuid].name = extension.name;
-                               extensions[extension.uuid].localState = extension.state;
-                               extensions[extension.uuid].local = true;
-                       }
-                       else
-                       {
-                               extensions[extension.uuid] = {
-                                       uuid:           extension.uuid,
-                                       name:           extension.name,
-                                       remoteState:    EXTENSION_STATE.UNINSTALLED,
-                                       localState:     extension.state,
-                                       remote:         false,
-                                       local:          true
-                               };
-                       }
-               };
-
-               return extensions;
-       }
-
-       /*
-        * Synchronize local changed extensions to remote list.
-        */
-       function localExtensionsChanged() {
-               extensionChangedTimeout = false;
-
-               if (!isEmptyObject(extensionChangedQueue))
-               {
-                       GSC.sendNativeRequest({
-                               execute: 'listExtensions'
-                       }, function (response) {
-                               if (response && response.success && response.extensions)
-                               {
-                                       chrome.storage.sync.get({
-                                               extensions: {}
-                                       }, function (options) {
-                                               for (const [extensionId, extension] of 
Object.entries(extensionChangedQueue)) {
-                                                       if ([EXTENSION_STATE.ENABLED, 
EXTENSION_STATE.DISABLED, EXTENSION_STATE.UNINSTALLED].includes(extension.state))
-                                                       {
-                                                               // Extension can be uninstalled already
-                                                               if (response.extensions[extensionId] && 
!isEmptyObject(response.extensions[extensionId]))
-                                                               {
-                                                                       extension = 
response.extensions[extensionId];
-                                                               }
-
-                                                               if (extension.state === 
EXTENSION_STATE.UNINSTALLED && options.extensions[extension.uuid])
-                                                               {
-                                                                       delete 
options.extensions[extension.uuid];
-                                                               }
-                                                               else
-                                                               {
-                                                                       options.extensions[extension.uuid] = {
-                                                                               uuid: extension.uuid,
-                                                                               name: extension.name,
-                                                                               state: extension.state
-                                                                       };
-                                                               }
-                                                       }
-                                               };
-
-                                               chrome.storage.sync.set({
-                                                       extensions: options.extensions
-                                               });
-
-                                               extensionChangedQueue = {};
-                                       });
-                               }
-                               else
-                               {
-                                       createSyncFailedNotification();
-                               }
-                       });
-               }
-       }
-
-       /*
-        * Synchronize remote changes with local GNOME Shell.
-        * 
-        * @param remoteExtensions - (optional) remote extensions list
-        */
-       function remoteExtensionsChanged(remoteExtensions) {
-               getExtensions(remoteExtensions).then((extensions) => {
-                       var enableExtensions = [];
-                       for ([uuid, extension] of Object.entries(extensions)) {
-                               if(extension.remote)
-                               {
-                                       if(!extension.local)
-                                       {
-                                               GSC.sendNativeRequest({
-                                                       execute: "installExtension",
-                                                       uuid: extension.uuid
-                                               }, onInstallUninstall);
-                                       }
-                                       else if (extension.remoteState !== extension.localState)
-                                       {
-                                               if(extension.remoteState === EXTENSION_STATE.ENABLED)
-                                               {
-                                                       enableExtensions.push({
-                                                               uuid: extension.uuid,
-                                                               enable: true
-                                                       });
-                                               }
-                                               else
-                                               {
-                                                       enableExtensions.push({
-                                                               uuid: extension.uuid,
-                                                               enable: false
-                                                       });
-                                               }
-                                       }
-                               }
-                               else if(extension.local)
-                               {
-                                       GSC.sendNativeRequest({
-                                               execute: "uninstallExtension",
-                                               uuid: extension.uuid
-                                       }, onInstallUninstall);
-                               }
-                       };
-
-                       if(enableExtensions.length > 0)
-                       {
-                               GSC.sendNativeRequest({
-                                       execute: "enableExtension",
-                                       extensions: enableExtensions
-                               });
-                       }
-               }).catch((message) => {
-                       createSyncFailedNotification(message);
-               });
-       }
-
-       /*
-        * Callback called when extension is installed or uninstalled as part
-        * of synchronization process.
-        */
-       function onInstallUninstall(response) {
-               if(response)
-               {
-                       if(!response.success)
-                       {
-                               createSyncFailedNotification(response.message);
-                       }
-               }
-               else
-               {
-                       createSyncFailedNotification();
-               }
-       }
-
-       /*
-        * Wrapper for localExtensionChanged that checks if synchronization is
-        * enabled.
-        */
-       function onExtensionChanged(request)
-       {
-               if(!COMPAT.SYNC_STORAGE || !enabled)
-               {
-                       return;
-               }
-
-               runIfSyncEnabled(() => {
-                       if (extensionChangedTimeout)
-                       {
-                               clearTimeout(extensionChangedTimeout);
-                       }
-
-                       extensionChangedQueue[request.parameters[EXTENSION_CHANGED_UUID]] = {
-                               uuid: request.parameters[EXTENSION_CHANGED_UUID],
-                               state: request.parameters[EXTENSION_CHANGED_STATE],
-                               error: request.parameters[EXTENSION_CHANGED_ERROR]
-                       };
-
-                       extensionChangedTimeout = setTimeout(function () {
-                               localExtensionsChanged();
-                       }, SYNC_QUEUE_TIMEOUT);
-               });
-       }
-
-       /*
-        * Wrapper for remoteExtensionsChanged that checks if synchronization is
-        * enabled.
-        */
-       function onSyncFromRemote(remoteExtensions)
-       {
-               if(!COMPAT.SYNC_STORAGE)
-               {
-                       return;
-               }
-
-               runIfSyncEnabled(function() {
-                       remoteExtensionsChanged(remoteExtensions);
-               });
-       }
-
-       /*
-        * Runs callback function if synchronyzation is enabled.
-        * 
-        * @param callback - callback function
-        */
-       function runIfSyncEnabled(callback) {
-               chrome.storage.local.get({
-                       syncExtensions: false
-               }, function (options) {
-                       if (options.syncExtensions)
-                       {
-                               chrome.permissions.contains({
-                                       permissions: ["idle"]
-                               }, function (result) {
-                                       if (result)
-                                       {
-                                               callback();
-                                       }
-                               });
-
-                       }
-               });
-       }
-
-       /*
-        * Create notification when synchronization failed.
-        */
-       function createSyncFailedNotification(cause) {
-               GSC.notifications.create(NOTIFICATION_SYNC_FAILED, {
-                       message: m('synchronization_failed', cause ? cause : m('unknown_error'))
-               });
-       }
-
-       /*
-        * Public methods.
-        */
-       return {
-               init: init,
-               getExtensions: getExtensions,
-               onExtensionChanged: onExtensionChanged
-       };
+GSC.sync = (function ($) {
+    var enabled = true;
+    var extensionChangedTimeout = false;
+    var extensionChangedQueue = {};
+
+    const SYNC_QUEUE_TIMEOUT = 7000;
+
+    /*
+     * Initialization rutines.
+     */
+    function init() {
+        if (!COMPAT.SYNC_STORAGE) {
+            enabled = false;
+            return;
+        }
+
+        function onIdleStateChanged(state) {
+            if (state === 'locked') {
+                enabled = false;
+
+                // Remove all disabled extensions from queue
+                for (const [extensionId, extension] of Object.entries(extensionChangedQueue)) {
+                    if (extension.state == EXTENSION_STATE.DISABLED) {
+                        delete extensionChangedQueue[extensionId];
+                    }
+                };
+            }
+            else if (state === 'active') {
+                enabled = true;
+            }
+        }
+
+        chrome.permissions.contains({
+            permissions: ["idle"]
+        }, function (result) {
+            if (result) {
+                chrome.idle.onStateChanged.addListener(onIdleStateChanged);
+            }
+            else {
+                enabled = false;
+            }
+        });
+
+        chrome.permissions.onAdded.addListener(function (permissions) {
+            if (permissions.permissions && permissions.permissions.indexOf('idle') !== -1) {
+                enabled = true;
+                chrome.idle.onStateChanged.addListener(onIdleStateChanged);
+            }
+        });
+
+        chrome.permissions.onRemoved.addListener(function (permissions) {
+            if (permissions.permissions && permissions.permissions.indexOf('idle') !== -1) {
+                enabled = false;
+                chrome.idle.onStateChanged.removeListener(onIdleStateChanged);
+            }
+        });
+
+        function onNotificationAction(notificationId, buttonIndex) {
+            if (notificationId !== NOTIFICATION_SYNC_FAILED) {
+                return;
+            }
+
+            GSC.notifications.remove(notificationId);
+        }
+
+        onSyncFromRemote();
+        chrome.storage.onChanged.addListener(function (changes, areaName) {
+            if (areaName === 'sync' && changes.extensions) {
+                onSyncFromRemote(changes.extensions.newValue);
+            }
+        });
+
+        chrome.runtime.onMessage.addListener(
+            function (request, sender, sendResponse) {
+                if (sender.id && sender.id === GS_CHROME_ID && request) {
+                    if (request === MESSAGE_SYNC_FROM_REMOTE) {
+                        onSyncFromRemote();
+                    }
+                }
+            }
+        );
+
+        GSC.onInitialize().then(response => {
+            /*
+                @Deprecated: remove browser notifications in version 9
+             */
+            if (!GSC.nativeNotificationsSupported(response)) {
+                chrome.notifications.onButtonClicked.addListener(onNotificationAction);
+            }
+            else {
+                chrome.runtime.onMessage.addListener(
+                    function (request, sender, sendResponse) {
+                        if (
+                            sender.id && sender.id === GS_CHROME_ID &&
+                            request && request.signal) {
+                            if (request.signal == SIGNAL_NOTIFICATION_ACTION) {
+                                onNotificationAction(request.name, request.button_id);
+                            }
+                        }
+                    }
+                );
+            }
+        });
+    }
+
+    /*
+     * Returns array of all local and remote extensions with structure:
+     * [
+     * $extension_uuid: {
+     *         uuid:           extension uuid,
+     *         name:           extension name,
+     *         remoteState:    extension state in remote storage,
+     *         localState:     extension state in current GNOME Shell,
+     *         remote:         true if extensions is in remote storage,
+     *         local:          true if extension installed localy
+     * },
+     * ...
+     * ]
+     */
+    function getExtensions(remoteExtensions) {
+        return new Promise((resolve, reject) => {
+            GSC.sendNativeRequest({
+                execute: 'listExtensions'
+            }, function (response) {
+                if (response && response.success) {
+                    if (remoteExtensions) {
+                        resolve(mergeExtensions(remoteExtensions, response.extensions));
+                    }
+                    else {
+                        chrome.storage.sync.get({
+                            extensions: {}
+                        }, function (options) {
+                            if (chrome.runtime.lastError) {
+                                reject(chrome.runtime.lastError.message);
+                            }
+                            else {
+                                resolve(mergeExtensions(options.extensions, response.extensions));
+                            }
+                        });
+                    }
+                }
+                else {
+                    var message = response && response.message ? response.message : 
m('error_connector_response');
+                    reject(message);
+                }
+            });
+        });
+    }
+
+    /*
+     * Returns merged list of extensions list in remote storage and
+     * locally installed extensions.
+     *
+     * Both parameters should be in form:
+     * {
+     * $extension_uuid: {
+     *         uuid:   ,
+     *         name:   ,
+     *         state:
+     * },
+     * ...
+     * }
+     */
+    function mergeExtensions(remoteExtensions, localExtensions) {
+        var extensions = {};
+
+        for (const [key, extension] of Object.entries(remoteExtensions)) {
+            if (extension.uuid && extension.name && extension.state) {
+                extensions[extension.uuid] = {
+                    uuid: extension.uuid,
+                    name: extension.name,
+                    remoteState: extension.state,
+                    remote: true,
+                    local: false
+                };
+            }
+        };
+
+        for (const [key, extension] of Object.entries(localExtensions)) {
+            if (extensions[extension.uuid]) {
+                extensions[extension.uuid].name = extension.name;
+                extensions[extension.uuid].localState = extension.state;
+                extensions[extension.uuid].local = true;
+            }
+            else {
+                extensions[extension.uuid] = {
+                    uuid: extension.uuid,
+                    name: extension.name,
+                    remoteState: EXTENSION_STATE.UNINSTALLED,
+                    localState: extension.state,
+                    remote: false,
+                    local: true
+                };
+            }
+        };
+
+        return extensions;
+    }
+
+    /*
+     * Synchronize local changed extensions to remote list.
+     */
+    function localExtensionsChanged() {
+        extensionChangedTimeout = false;
+
+        if (!isEmptyObject(extensionChangedQueue)) {
+            GSC.sendNativeRequest({
+                execute: 'listExtensions'
+            }, function (response) {
+                if (response && response.success && response.extensions) {
+                    chrome.storage.sync.get({
+                        extensions: {}
+                    }, function (options) {
+                        for (const [extensionId, extension] of Object.entries(extensionChangedQueue)) {
+                            if ([EXTENSION_STATE.ENABLED, EXTENSION_STATE.DISABLED, 
EXTENSION_STATE.UNINSTALLED].includes(extension.state)) {
+                                // Extension can be uninstalled already
+                                if (response.extensions[extensionId] && 
!isEmptyObject(response.extensions[extensionId])) {
+                                    extension = response.extensions[extensionId];
+                                }
+
+                                if (extension.state === EXTENSION_STATE.UNINSTALLED && 
options.extensions[extension.uuid]) {
+                                    delete options.extensions[extension.uuid];
+                                }
+                                else {
+                                    options.extensions[extension.uuid] = {
+                                        uuid: extension.uuid,
+                                        name: extension.name,
+                                        state: extension.state
+                                    };
+                                }
+                            }
+                        };
+
+                        chrome.storage.sync.set({
+                            extensions: options.extensions
+                        });
+
+                        extensionChangedQueue = {};
+                    });
+                }
+                else {
+                    createSyncFailedNotification();
+                }
+            });
+        }
+    }
+
+    /*
+     * Synchronize remote changes with local GNOME Shell.
+     *
+     * @param remoteExtensions - (optional) remote extensions list
+     */
+    function remoteExtensionsChanged(remoteExtensions) {
+        getExtensions(remoteExtensions).then((extensions) => {
+            var enableExtensions = [];
+            for ([uuid, extension] of Object.entries(extensions)) {
+                if (extension.remote) {
+                    if (!extension.local) {
+                        GSC.sendNativeRequest({
+                            execute: "installExtension",
+                            uuid: extension.uuid
+                        }, onInstallUninstall);
+                    }
+                    else if (extension.remoteState !== extension.localState) {
+                        if (extension.remoteState === EXTENSION_STATE.ENABLED) {
+                            enableExtensions.push({
+                                uuid: extension.uuid,
+                                enable: true
+                            });
+                        }
+                        else {
+                            enableExtensions.push({
+                                uuid: extension.uuid,
+                                enable: false
+                            });
+                        }
+                    }
+                }
+                else if (extension.local) {
+                    GSC.sendNativeRequest({
+                        execute: "uninstallExtension",
+                        uuid: extension.uuid
+                    }, onInstallUninstall);
+                }
+            };
+
+            if (enableExtensions.length > 0) {
+                GSC.sendNativeRequest({
+                    execute: "enableExtension",
+                    extensions: enableExtensions
+                });
+            }
+        }).catch((message) => {
+            createSyncFailedNotification(message);
+        });
+    }
+
+    /*
+     * Callback called when extension is installed or uninstalled as part
+     * of synchronization process.
+     */
+    function onInstallUninstall(response) {
+        if (response) {
+            if (!response.success) {
+                createSyncFailedNotification(response.message);
+            }
+        }
+        else {
+            createSyncFailedNotification();
+        }
+    }
+
+    /*
+     * Wrapper for localExtensionChanged that checks if synchronization is
+     * enabled.
+     */
+    function onExtensionChanged(request) {
+        if (!COMPAT.SYNC_STORAGE || !enabled) {
+            return;
+        }
+
+        runIfSyncEnabled(() => {
+            if (extensionChangedTimeout) {
+                clearTimeout(extensionChangedTimeout);
+            }
+
+            extensionChangedQueue[request.parameters[EXTENSION_CHANGED_UUID]] = {
+                uuid: request.parameters[EXTENSION_CHANGED_UUID],
+                state: request.parameters[EXTENSION_CHANGED_STATE],
+                error: request.parameters[EXTENSION_CHANGED_ERROR]
+            };
+
+            extensionChangedTimeout = setTimeout(function () {
+                localExtensionsChanged();
+            }, SYNC_QUEUE_TIMEOUT);
+        });
+    }
+
+    /*
+     * Wrapper for remoteExtensionsChanged that checks if synchronization is
+     * enabled.
+     */
+    function onSyncFromRemote(remoteExtensions) {
+        if (!COMPAT.SYNC_STORAGE) {
+            return;
+        }
+
+        runIfSyncEnabled(function () {
+            remoteExtensionsChanged(remoteExtensions);
+        });
+    }
+
+    /*
+     * Runs callback function if synchronyzation is enabled.
+     *
+     * @param callback - callback function
+     */
+    function runIfSyncEnabled(callback) {
+        chrome.storage.local.get({
+            syncExtensions: false
+        }, function (options) {
+            if (options.syncExtensions) {
+                chrome.permissions.contains({
+                    permissions: ["idle"]
+                }, function (result) {
+                    if (result) {
+                        callback();
+                    }
+                });
+
+            }
+        });
+    }
+
+    /*
+     * Create notification when synchronization failed.
+     */
+    function createSyncFailedNotification(cause) {
+        GSC.notifications.create(NOTIFICATION_SYNC_FAILED, {
+            message: m('synchronization_failed', cause ? cause : m('unknown_error'))
+        });
+    }
+
+    /*
+     * Public methods.
+     */
+    return {
+        init: init,
+        getExtensions: getExtensions,
+        onExtensionChanged: onExtensionChanged
+    };
 })();
diff --git a/extension/include/toolbar.js b/extension/include/toolbar.js
index d918b43..24a7fda 100644
--- a/extension/include/toolbar.js
+++ b/extension/include/toolbar.js
@@ -3,55 +3,51 @@
 /*
  * Main object that handles toolbar icon.
  */
-GSC.toolbar = (function() {
-       /*
-        * Initialization rutines.
-        */
-       function init() {
-               chrome.storage.local.get(DEFAULT_LOCAL_OPTIONS, function (options) {
-                       if (options.useLightIcon)
-                       {
-                               setLightIcon();
-                       }
-               });
+GSC.toolbar = (function () {
+    /*
+     * Initialization rutines.
+     */
+    function init() {
+        chrome.storage.local.get(DEFAULT_LOCAL_OPTIONS, function (options) {
+            if (options.useLightIcon) {
+                setLightIcon();
+            }
+        });
 
-               chrome.storage.onChanged.addListener(function(changes, areaName) {
-                       if(areaName === 'local' && changes.useLightIcon)
-                       {
-                               if(changes.useLightIcon.newValue)
-                               {
-                                       setLightIcon();
-                               }
-                               else
-                               {
-                                       setDarkicon();
-                               }
-                       }
-               });
-       }
+        chrome.storage.onChanged.addListener(function (changes, areaName) {
+            if (areaName === 'local' && changes.useLightIcon) {
+                if (changes.useLightIcon.newValue) {
+                    setLightIcon();
+                }
+                else {
+                    setDarkicon();
+                }
+            }
+        });
+    }
 
-       function setLightIcon() {
-               chrome.browserAction.setIcon({
-                       path: {
-                               "16": "icons/GnomeLogo-light-16.png",
-                               "32": "icons/GnomeLogo-light-32.png"
-                       }
-               });
-       }
+    function setLightIcon() {
+        chrome.browserAction.setIcon({
+            path: {
+                "16": "icons/GnomeLogo-light-16.png",
+                "32": "icons/GnomeLogo-light-32.png"
+            }
+        });
+    }
 
-       function setDarkicon() {
-               chrome.browserAction.setIcon({
-                       path: {
-                               "16": "icons/GnomeLogo-16.png",
-                           "32": "icons/GnomeLogo-32.png"
-                       }
-               });
-       }
+    function setDarkicon() {
+        chrome.browserAction.setIcon({
+            path: {
+                "16": "icons/GnomeLogo-16.png",
+                "32": "icons/GnomeLogo-32.png"
+            }
+        });
+    }
 
-       /*
-        * Public methods.
-        */
-       return {
-               init: init
-       };
+    /*
+     * Public methods.
+     */
+    return {
+        init: init
+    };
 })();
diff --git a/extension/include/update.js b/extension/include/update.js
index db1a948..53050ff 100644
--- a/extension/include/update.js
+++ b/extension/include/update.js
@@ -1,196 +1,170 @@
 // SPDX-License-Identifer: GPL-3.0-or-later
 
-GSC.update = (function() {
-       function schedule(updateCheckPeriod, skipCheck) {
-               if(!skipCheck)
-               {
-                       check();
-               }
-
-               chrome.alarms.create(
-                       ALARM_UPDATE_CHECK,
-                       {
-                               delayInMinutes: updateCheckPeriod * 60,
-                               periodInMinutes: updateCheckPeriod * 60
-                       }
-               );
-
-               chrome.runtime.sendMessage(GS_CHROME_ID, MESSAGE_NEXT_UPDATE_CHANGED);
-       }
-
-       function check() {
-               GSC.onInitialize().then(response => {
-                       if (response.success)
-                       {
-                               if(GSC.nativeUpdateCheckSupported(response))
-                               {
-                                       chrome.storage.sync.get(DEFAULT_SYNC_OPTIONS, function (options) {
-                                               GSC.sendNativeRequest(
-                                                       {
-                                                               execute: 'checkUpdate',
-                                                               url: UPDATE_URL,
-                                                               enabledOnly: options.updateCheckEnabledOnly
-                                                       }, function (response) {
-                                                       if (response.success)
-                                                       {
-                                                               onSweetToothResponse(response.upgrade, 
response.extensions);
-                                                       }
-                                                       else if(console)
-                                                       {
-                                                               console.error(response.message ? 
response.message : m('native_request_failed', 'checkUpdate'));
-                                                       }
-                                               });
-                                       });
-                               }
-                               else
-                               {
-                                       chrome.storage.sync.set({
-                                               updateCheck: false
-                                       });
-                               }
-                       }
-                       else if(console)
-                       {
-                               console.error(response.message ? response.message : 
m('native_request_failed', 'initialize'));
-                       }
-               });
-       }
-
-       function onSweetToothResponse(data, installedExtensions) {
-               var toUpgrade = [];
-               for (uuid in data)
-               {
-                       if (installedExtensions[uuid] && ['upgrade', 'downgrade'].includes(data[uuid]))
-                       {
-                               toUpgrade.push({
-                                       title: installedExtensions[uuid].name,
-                                       message: m('extension_status_' + data[uuid])
-                               });
-                       }
-               }
-
-               if (toUpgrade.length > 0)
-               {
-                       GSC.notifications.create(NOTIFICATION_UPDATE_AVAILABLE, {
-                               type: chrome.notifications.TemplateType.LIST,
-                               title: m('update_available'),
-                               message: '',
-                               items: toUpgrade
-                       });
-               }
-
-               chrome.storage.local.set({
-                       lastUpdateCheck: new Date().toLocaleString()
-               });
-       }
-
-       function init() {
-               function onNotificationAction(notificationId, buttonIndex) {
-                       if (NOTIFICATION_UPDATE_AVAILABLE == notificationId)
-                               return;
-
-                       GSC.notifications.remove(notificationId);
-               }
-
-               function onNotificationClicked(notificationId) {
-                       if (notificationId === NOTIFICATION_UPDATE_AVAILABLE)
-                       {
-                               chrome.tabs.create({
-                                       url: EXTENSIONS_WEBSITE + 'local/',
-                                       active: true
-                               });
-                       }
-               }
-
-               chrome.alarms.onAlarm.addListener(function (alarm) {
-                       if (alarm.name === ALARM_UPDATE_CHECK)
-                       {
-                               check();
-
-                               chrome.alarms.get(ALARM_UPDATE_CHECK, function (alarm) {
-                                       if (alarm && alarm.periodInMinutes && ((alarm.scheduledTime - 
Date.now()) / 1000 / 60 < alarm.periodInMinutes * 0.9))
-                                       {
-                                               schedule(alarm.periodInMinutes / 60, true);
-                                       }
-                                       else
-                                       {
-                                               chrome.runtime.sendMessage(GS_CHROME_ID, 
MESSAGE_NEXT_UPDATE_CHANGED);
-                                       }
-                               });
-                       }
-               });
-
-               GSC.onInitialize().then(response => {
-                       /*
-                               @Deprecated: remove browser notifications in version 9
-                        */
-                       if (!GSC.nativeNotificationsSupported(response))
-                       {
-                               chrome.notifications.onClicked.addListener(function (notificationId) {
-                                       onNotificationClicked(notificationId);
-                               });
-
-                               chrome.notifications.onButtonClicked.addListener(onNotificationAction);
-                       }
-                       else
-                       {
-                               window.addEventListener("message", function (event) {
-                                       if (event.source == window && event.data && event.data.signal)
-                                       {
-                                               if(event.data.signal == SIGNAL_NOTIFICATION_ACTION)
-                                               {
-                                                       onNotificationAction(event.data.name, 
event.data.button_id);
-                                               }
-                                               else if(event.data.signal == SIGNAL_NOTIFICATION_CLICKED)
-                                               {
-                                                       onNotificationClicked(event.data.name);
-                                               }
-                                       }
-                               });
-                       }
-               });
-
-               chrome.storage.onChanged.addListener(function (changes, areaName) {
-                       if (changes.updateCheck)
-                       {
-                               if (!changes.updateCheck.newValue)
-                               {
-                                       chrome.alarms.clear(ALARM_UPDATE_CHECK);
-                               }
-                               else
-                               {
-                                       chrome.storage.sync.get(DEFAULT_SYNC_OPTIONS, function (options) {
-                                               schedule(options.updateCheckPeriod);
-                                       });
-                               }
-                       }
-                       else if (changes.updateCheckPeriod)
-                       {
-                               chrome.storage.sync.get(DEFAULT_SYNC_OPTIONS, function (options) {
-                                       if (options.updateCheck)
-                                       {
-                                               schedule(options.updateCheckPeriod);
-                                       }
-                               });
-                       }
-               });
-
-               chrome.storage.sync.get(DEFAULT_SYNC_OPTIONS, function (options) {
-                       if (options.updateCheck)
-                       {
-                               chrome.alarms.get(ALARM_UPDATE_CHECK, function (alarm) {
-                                       if (!alarm || !alarm.periodInMinutes || alarm.periodInMinutes !== 
options.updateCheckPeriod * 60)
-                                       {
-                                               schedule(options.updateCheckPeriod);
-                                       }
-                               });
-                       }
-               });
-       }
-
-       return {
-               init: init,
-               check: check,
-               schedule: schedule
-       };
+GSC.update = (function () {
+    function schedule(updateCheckPeriod, skipCheck) {
+        if (!skipCheck) {
+            check();
+        }
+
+        chrome.alarms.create(
+            ALARM_UPDATE_CHECK,
+            {
+                delayInMinutes: updateCheckPeriod * 60,
+                periodInMinutes: updateCheckPeriod * 60
+            }
+        );
+
+        chrome.runtime.sendMessage(GS_CHROME_ID, MESSAGE_NEXT_UPDATE_CHANGED);
+    }
+
+    function check() {
+        GSC.onInitialize().then(response => {
+            if (response.success) {
+                if (GSC.nativeUpdateCheckSupported(response)) {
+                    chrome.storage.sync.get(DEFAULT_SYNC_OPTIONS, function (options) {
+                        GSC.sendNativeRequest(
+                            {
+                                execute: 'checkUpdate',
+                                url: UPDATE_URL,
+                                enabledOnly: options.updateCheckEnabledOnly
+                            }, function (response) {
+                                if (response.success) {
+                                    onSweetToothResponse(response.upgrade, response.extensions);
+                                }
+                                else if (console) {
+                                    console.error(response.message ? response.message : 
m('native_request_failed', 'checkUpdate'));
+                                }
+                            });
+                    });
+                }
+                else {
+                    chrome.storage.sync.set({
+                        updateCheck: false
+                    });
+                }
+            }
+            else if (console) {
+                console.error(response.message ? response.message : m('native_request_failed', 
'initialize'));
+            }
+        });
+    }
+
+    function onSweetToothResponse(data, installedExtensions) {
+        var toUpgrade = [];
+        for (uuid in data) {
+            if (installedExtensions[uuid] && ['upgrade', 'downgrade'].includes(data[uuid])) {
+                toUpgrade.push({
+                    title: installedExtensions[uuid].name,
+                    message: m('extension_status_' + data[uuid])
+                });
+            }
+        }
+
+        if (toUpgrade.length > 0) {
+            GSC.notifications.create(NOTIFICATION_UPDATE_AVAILABLE, {
+                type: chrome.notifications.TemplateType.LIST,
+                title: m('update_available'),
+                message: '',
+                items: toUpgrade
+            });
+        }
+
+        chrome.storage.local.set({
+            lastUpdateCheck: new Date().toLocaleString()
+        });
+    }
+
+    function init() {
+        function onNotificationAction(notificationId, buttonIndex) {
+            if (NOTIFICATION_UPDATE_AVAILABLE == notificationId)
+                return;
+
+            GSC.notifications.remove(notificationId);
+        }
+
+        function onNotificationClicked(notificationId) {
+            if (notificationId === NOTIFICATION_UPDATE_AVAILABLE) {
+                chrome.tabs.create({
+                    url: EXTENSIONS_WEBSITE + 'local/',
+                    active: true
+                });
+            }
+        }
+
+        chrome.alarms.onAlarm.addListener(function (alarm) {
+            if (alarm.name === ALARM_UPDATE_CHECK) {
+                check();
+
+                chrome.alarms.get(ALARM_UPDATE_CHECK, function (alarm) {
+                    if (alarm && alarm.periodInMinutes && ((alarm.scheduledTime - Date.now()) / 1000 / 60 < 
alarm.periodInMinutes * 0.9)) {
+                        schedule(alarm.periodInMinutes / 60, true);
+                    }
+                    else {
+                        chrome.runtime.sendMessage(GS_CHROME_ID, MESSAGE_NEXT_UPDATE_CHANGED);
+                    }
+                });
+            }
+        });
+
+        GSC.onInitialize().then(response => {
+            /*
+                @Deprecated: remove browser notifications in version 9
+             */
+            if (!GSC.nativeNotificationsSupported(response)) {
+                chrome.notifications.onClicked.addListener(function (notificationId) {
+                    onNotificationClicked(notificationId);
+                });
+
+                chrome.notifications.onButtonClicked.addListener(onNotificationAction);
+            }
+            else {
+                window.addEventListener("message", function (event) {
+                    if (event.source == window && event.data && event.data.signal) {
+                        if (event.data.signal == SIGNAL_NOTIFICATION_ACTION) {
+                            onNotificationAction(event.data.name, event.data.button_id);
+                        }
+                        else if (event.data.signal == SIGNAL_NOTIFICATION_CLICKED) {
+                            onNotificationClicked(event.data.name);
+                        }
+                    }
+                });
+            }
+        });
+
+        chrome.storage.onChanged.addListener(function (changes, areaName) {
+            if (changes.updateCheck) {
+                if (!changes.updateCheck.newValue) {
+                    chrome.alarms.clear(ALARM_UPDATE_CHECK);
+                }
+                else {
+                    chrome.storage.sync.get(DEFAULT_SYNC_OPTIONS, function (options) {
+                        schedule(options.updateCheckPeriod);
+                    });
+                }
+            }
+            else if (changes.updateCheckPeriod) {
+                chrome.storage.sync.get(DEFAULT_SYNC_OPTIONS, function (options) {
+                    if (options.updateCheck) {
+                        schedule(options.updateCheckPeriod);
+                    }
+                });
+            }
+        });
+
+        chrome.storage.sync.get(DEFAULT_SYNC_OPTIONS, function (options) {
+            if (options.updateCheck) {
+                chrome.alarms.get(ALARM_UPDATE_CHECK, function (alarm) {
+                    if (!alarm || !alarm.periodInMinutes || alarm.periodInMinutes !== 
options.updateCheckPeriod * 60) {
+                        schedule(options.updateCheckPeriod);
+                    }
+                });
+            }
+        });
+    }
+
+    return {
+        init: init,
+        check: check,
+        schedule: schedule
+    };
 })();
diff --git a/extension/options.html b/extension/options.html
index c8f809a..61e3337 100644
--- a/extension/options.html
+++ b/extension/options.html
@@ -1,113 +1,132 @@
 <!DOCTYPE html>
 <html>
-       <head>
-               <meta http-equiv="Content-Type" content="text/html; charset=UTF-8" />
-               <title data-i18n="options_title, __MSG_gs_chrome__"></title>
-               <link href="css/options.css" rel="stylesheet" type="text/css" />
-               <script type="text/javascript" src="include/i18n.js"></script>
-               <script type="text/javascript" src="include/constants.js"></script>
-               <script type="text/javascript" src="include/compat-common.js"></script>
-               <script type="text/javascript" src="include/gsc.js"></script>
-               <script type="text/javascript" src="include/sync.js"></script>
-       </head>
-       <body>
-               <ul data-tabs class="tabs">
-                       <li><a href="#options" data-tabs-active data-i18n='options_link' 
data-tab='tab'></a></li>
-                       <li><a href="#synchronization" data-i18n='synchronization' data-tab='tab'></a></li>
-                       <li class='translation_credits_container'><a href="#translation_credits" 
data-i18n='translation_credits_title' data-tab='tab'></a></li>
-               </ul>
-               <div>
-                       <div id="options" class="tabs-pane active" data-tabs-pane='tabs-pane'>
-                               <div id="status" class="hide" data-i18n="options_saved"></div>
-                               <div id="error" class="hide"></div>
 
-                               <fieldset>
-                                       <dl>
-                                               <dt>
-                                                       <span data-i18n="options_update_check"></span>:<br 
class="update-notice"/>
-                                                       <span class="notice wrapped update-notice" 
data-i18n="options_update_check_notice"></span>
-                                               </dt>
-                                               <dd>
-                                                       <label for='update_check_yes' 
data-i18n="yes"></label> <input type='radio' id='update_check_yes' name='update_check' disabled='disabled' />
-                                                       <label for='update_check_no' data-i18n="no"></label> 
<input type='radio' id='update_check_no' name='update_check' disabled='disabled' />
+<head>
+    <meta http-equiv="Content-Type" content="text/html; charset=UTF-8" />
+    <title data-i18n="options_title, __MSG_gs_chrome__"></title>
+    <link href="css/options.css" rel="stylesheet" type="text/css" />
+    <script type="text/javascript" src="include/i18n.js"></script>
+    <script type="text/javascript" src="include/constants.js"></script>
+    <script type="text/javascript" src="include/compat-common.js"></script>
+    <script type="text/javascript" src="include/gsc.js"></script>
+    <script type="text/javascript" src="include/sync.js"></script>
+</head>
 
-                                                       <div class='notice'><span 
data-i18n="options_last_check"></span>: <span id="last_update_check" data-i18n="never"></span></div>
-                                               </dd>
-                                       </dl>
-                                       <dl>
-                                               <dt>
-                                                       <span 
data-i18n="options_update_check_enabled"></span>:<br class="update-notice"/>
-                                                       <span class="notice wrapped update-notice" 
data-i18n="options_update_check_enabled_notice"></span>
-                                               </dt>
-                                               <dd>
-                                                       <label for='update_check_enabled_yes' 
data-i18n="yes"></label> <input type='radio' id='update_check_enabled_yes' name='update_check_enabled' 
disabled='disabled' />
-                                                       <label for='update_check_enabled_no' 
data-i18n="no"></label> <input type='radio' id='update_check_enabled_no' name='update_check_enabled' 
disabled='disabled' />
-                                               </dd>
-                                       </dl>
-                                       <dl>
-                                               <dt><label for='update_check_period' 
data-i18n="options_check_period"></label>:</dt>
-                                               <dd>
-                                                       <input id='update_check_period' type='number' min="3" 
disabled='disabled' /> <span data-i18n="hours"></span>
+<body>
+    <ul data-tabs class="tabs">
+        <li><a href="#options" data-tabs-active data-i18n='options_link' data-tab='tab'></a></li>
+        <li><a href="#synchronization" data-i18n='synchronization' data-tab='tab'></a></li>
+        <li class='translation_credits_container'><a href="#translation_credits" 
data-i18n='translation_credits_title'
+                data-tab='tab'></a></li>
+    </ul>
+    <div>
+        <div id="options" class="tabs-pane active" data-tabs-pane='tabs-pane'>
+            <div id="status" class="hide" data-i18n="options_saved"></div>
+            <div id="error" class="hide"></div>
 
-                                                       <div class='notice'><span 
data-i18n="options_next_check"></span>: <span id="next_update_check" data-i18n="never"></span></div>
-                                               </dd>
-                                       </dl>
-                                       <dl>
-                                               <dt>
-                                                       <label for='synchronize_extensions_yes' 
data-i18n="options_synchronize_extensions"></label>:<br />
-                                                       <span class="notice wrapped" 
data-i18n="options_synchronize_extensions_notice"></span>
-                                               </dt>
-                                               <dd>
-                                                       <label for='synchronize_extensions_yes' 
data-i18n="yes"></label> <input type='radio' id='synchronize_extensions_yes' name='synchronize_extensions' />
-                                                       <label for='synchronize_extensions_no' 
data-i18n="no"></label> <input type='radio' id='synchronize_extensions_no' name='synchronize_extensions' />
-                                               </dd>
-                                       </dl>
-                                       <dl>
-                                               <dt><span data-i18n="options_use_light_icon"></span>:</dt>
-                                               <dd>
-                                                       <label for='use_light_icon_yes' 
data-i18n="yes"></label> <input type='radio' id='use_light_icon_yes' name='use_light_icon' />
-                                                       <label for='use_light_icon_no' 
data-i18n="no"></label> <input type='radio' id='use_light_icon_no' name='use_light_icon' />
-                                               </dd>
-                                       </dl>
-                                       <dl>
-                                               <dt><span data-i18n="options_show_release_notes"></span>:</dt>
-                                               <dd>
-                                                       <label for='show_release_notes_yes' 
data-i18n="yes"></label> <input type='radio' id='show_release_notes_yes' name='show_release_notes' />
-                                                       <label for='show_release_notes_no' 
data-i18n="no"></label> <input type='radio' id='show_release_notes_no' name='show_release_notes' />
-                                               </dd>
-                                       </dl>
-                               </fieldset>
+            <fieldset>
+                <dl>
+                    <dt>
+                        <span data-i18n="options_update_check"></span>:<br class="update-notice" />
+                        <span class="notice wrapped update-notice" 
data-i18n="options_update_check_notice"></span>
+                    </dt>
+                    <dd>
+                        <label for='update_check_yes' data-i18n="yes"></label> <input type='radio' 
id='update_check_yes'
+                            name='update_check' disabled='disabled' />
+                        <label for='update_check_no' data-i18n="no"></label> <input type='radio' 
id='update_check_no'
+                            name='update_check' disabled='disabled' />
 
-                               <div class='buttons'>
-                                       <button id="save" data-i18n="save"></button>
-                               </div>
-                       </div>
-                       <div id="synchronization" class="tabs-pane" data-tabs-pane='tabs-pane'>
-                               <table>
-                                       <thead>
-                                               <tr>
-                                                       <th data-i18n="extension_name"></th>
-                                                       <th data-i18n="extension_installed"></th>
-                                                       <th data-i18n="extension_enabled"></th>
-                                                       <th data-i18n="extension_synchronized"></th>
-                                               </tr>
-                                       </thead>
-                                       <tbody></tbody>
-                               </table>
-                       </div>
-                       <div id="translation_credits" class="tabs-pane translation_credits_container" 
data-tabs-pane='tabs-pane'>
-                               <div data-i18n='translation_credits' data-i18n-html='true'></div>
-                       </div>
-               </div>
+                        <div class='notice'><span data-i18n="options_last_check"></span>: <span 
id="last_update_check"
+                                data-i18n="never"></span></div>
+                    </dd>
+                </dl>
+                <dl>
+                    <dt>
+                        <span data-i18n="options_update_check_enabled"></span>:<br class="update-notice" />
+                        <span class="notice wrapped update-notice"
+                            data-i18n="options_update_check_enabled_notice"></span>
+                    </dt>
+                    <dd>
+                        <label for='update_check_enabled_yes' data-i18n="yes"></label> <input type='radio'
+                            id='update_check_enabled_yes' name='update_check_enabled' disabled='disabled' />
+                        <label for='update_check_enabled_no' data-i18n="no"></label> <input type='radio'
+                            id='update_check_enabled_no' name='update_check_enabled' disabled='disabled' />
+                    </dd>
+                </dl>
+                <dl>
+                    <dt><label for='update_check_period' data-i18n="options_check_period"></label>:</dt>
+                    <dd>
+                        <input id='update_check_period' type='number' min="3" disabled='disabled' /> <span
+                            data-i18n="hours"></span>
+
+                        <div class='notice'><span data-i18n="options_next_check"></span>: <span 
id="next_update_check"
+                                data-i18n="never"></span></div>
+                    </dd>
+                </dl>
+                <dl>
+                    <dt>
+                        <label for='synchronize_extensions_yes'
+                            data-i18n="options_synchronize_extensions"></label>:<br />
+                        <span class="notice wrapped" 
data-i18n="options_synchronize_extensions_notice"></span>
+                    </dt>
+                    <dd>
+                        <label for='synchronize_extensions_yes' data-i18n="yes"></label> <input type='radio'
+                            id='synchronize_extensions_yes' name='synchronize_extensions' />
+                        <label for='synchronize_extensions_no' data-i18n="no"></label> <input type='radio'
+                            id='synchronize_extensions_no' name='synchronize_extensions' />
+                    </dd>
+                </dl>
+                <dl>
+                    <dt><span data-i18n="options_use_light_icon"></span>:</dt>
+                    <dd>
+                        <label for='use_light_icon_yes' data-i18n="yes"></label> <input type='radio'
+                            id='use_light_icon_yes' name='use_light_icon' />
+                        <label for='use_light_icon_no' data-i18n="no"></label> <input type='radio'
+                            id='use_light_icon_no' name='use_light_icon' />
+                    </dd>
+                </dl>
+                <dl>
+                    <dt><span data-i18n="options_show_release_notes"></span>:</dt>
+                    <dd>
+                        <label for='show_release_notes_yes' data-i18n="yes"></label> <input type='radio'
+                            id='show_release_notes_yes' name='show_release_notes' />
+                        <label for='show_release_notes_no' data-i18n="no"></label> <input type='radio'
+                            id='show_release_notes_no' name='show_release_notes' />
+                    </dd>
+                </dl>
+            </fieldset>
+
+            <div class='buttons'>
+                <button id="save" data-i18n="save"></button>
+            </div>
+        </div>
+        <div id="synchronization" class="tabs-pane" data-tabs-pane='tabs-pane'>
+            <table>
+                <thead>
+                    <tr>
+                        <th data-i18n="extension_name"></th>
+                        <th data-i18n="extension_installed"></th>
+                        <th data-i18n="extension_enabled"></th>
+                        <th data-i18n="extension_synchronized"></th>
+                    </tr>
+                </thead>
+                <tbody></tbody>
+            </table>
+        </div>
+        <div id="translation_credits" class="tabs-pane translation_credits_container" 
data-tabs-pane='tabs-pane'>
+            <div data-i18n='translation_credits' data-i18n-html='true'></div>
+        </div>
+    </div>
+
+    <dialog id='syncChoice'>
+        <h2 data-i18n='synchronization_storage_exists'></h2>
+        <form method="dialog">
+            <button type='submit' value='local' data-i18n='synchronization_use_local'></button>
+            <button type='submit' value='remote' data-i18n='synchronization_use_remote'></button>
+            <button type='submit' value='cancel' data-i18n='synchronization_cancel'></button>
+        </form>
+    </dialog>
+    <script type="text/javascript" src="options.js"></script>
+</body>
 
-               <dialog id='syncChoice'>
-                       <h2 data-i18n='synchronization_storage_exists'></h2>
-                       <form method="dialog">
-                               <button type='submit' value='local' 
data-i18n='synchronization_use_local'></button>
-                               <button type='submit' value='remote' 
data-i18n='synchronization_use_remote'></button>
-                               <button type='submit' value='cancel' 
data-i18n='synchronization_cancel'></button>
-                       </form>
-               </dialog>
-               <script type="text/javascript" src="options.js"></script>
-       </body>
 </html>
diff --git a/extension/options.js b/extension/options.js
index f0f67f2..5aa392e 100644
--- a/extension/options.js
+++ b/extension/options.js
@@ -1,369 +1,325 @@
 // SPDX-License-Identifer: GPL-3.0-or-later
 
-function init_tabs()
-{
-       let tabLinks = $("[data-tabs]").querySelectorAll('a');
-       let updateTabs = (active = null) => {
-               tabLinks.forEach((link) => {
-                       let tab = $(link.getAttribute("href"));
-
-                       if((!active && link.hasAttribute('data-tabs-active')) || link == active) {
-                               tab.classList.remove('hide');
-                       }
-                       else {
-                               tab.classList.add('hide');
-                       }
-               });
-       };
-
-       tabLinks.forEach((link) => {
-               link.addEventListener('click', (event) => {
-                       updateTabs(event.target);
-               });
-       });
-
-       updateTabs();
+function init_tabs() {
+    let tabLinks = $("[data-tabs]").querySelectorAll('a');
+    let updateTabs = (active = null) => {
+        tabLinks.forEach((link) => {
+            let tab = $(link.getAttribute("href"));
+
+            if ((!active && link.hasAttribute('data-tabs-active')) || link == active) {
+                tab.classList.remove('hide');
+            }
+            else {
+                tab.classList.add('hide');
+            }
+        });
+    };
+
+    tabLinks.forEach((link) => {
+        link.addEventListener('click', (event) => {
+            updateTabs(event.target);
+        });
+    });
+
+    updateTabs();
 }
 
-function save_options()
-{
-       var showReleaseNotes = $('#show_release_notes_yes').checked;
-       var syncExtensions = $('#synchronize_extensions_yes').checked;
-       var updateCheck = $('#update_check_yes').checked;
-       var updateCheckEnabledOnly = $('#update_check_enabled_yes').checked;
-       var updateCheckPeriod = $('#update_check_period').value;
-       var useLightIcon = $('#use_light_icon_yes').checked;
-       updateCheckPeriod = Math.max(3, updateCheckPeriod);
-
-       chrome.storage.sync.set({
-               showReleaseNotes:       showReleaseNotes,
-               updateCheck:            updateCheck,
-               updateCheckEnabledOnly: updateCheckEnabledOnly,
-               updateCheckPeriod:      updateCheckPeriod
-       }, function () {
-               chrome.storage.local.set({
-                       syncExtensions: syncExtensions,
-                       useLightIcon:   useLightIcon
-               }, function() {
-                       if(syncExtensions)
-                       {
-                               syncType = document.getElementById('syncChoice').returnValue;
-                               if(!syncType || syncType === 'local')
-                               {
-                                       GSC.sync.getExtensions().then((extensions) => {
-                                               var localExtensions = {};
-                                               for (const [uuid, extension] of Object.entries(extensions)) {
-                                                       if(extension.local && extension.localState != 
EXTENSION_STATE.UNINSTALLED)
-                                                       {
-                                                               localExtensions[extension.uuid] = {
-                                                                       uuid:   extension.uuid,
-                                                                       name:   extension.name,
-                                                                       state:  extension.localState
-                                                               };
-                                                       }
-                                               };
-
-                                               chrome.storage.sync.set({
-                                                       extensions: localExtensions
-                                               }, function() {
-                                                       showSuccessStatus();
-                                               });
-                                       }).catch((message) => {
-                                               showWithDelay($('#error'), 15000, message);
-                                       });
-                               }
-                               else if(syncType === 'remote')
-                               {
-                                       chrome.runtime.sendMessage(GS_CHROME_ID, MESSAGE_SYNC_FROM_REMOTE);
-                                       showSuccessStatus();
-                               }
-                       }
-                       else
-                       {
-                               showSuccessStatus();
-                       }
-               });
-       });
+function save_options() {
+    var showReleaseNotes = $('#show_release_notes_yes').checked;
+    var syncExtensions = $('#synchronize_extensions_yes').checked;
+    var updateCheck = $('#update_check_yes').checked;
+    var updateCheckEnabledOnly = $('#update_check_enabled_yes').checked;
+    var updateCheckPeriod = $('#update_check_period').value;
+    var useLightIcon = $('#use_light_icon_yes').checked;
+    updateCheckPeriod = Math.max(3, updateCheckPeriod);
+
+    chrome.storage.sync.set({
+        showReleaseNotes: showReleaseNotes,
+        updateCheck: updateCheck,
+        updateCheckEnabledOnly: updateCheckEnabledOnly,
+        updateCheckPeriod: updateCheckPeriod
+    }, function () {
+        chrome.storage.local.set({
+            syncExtensions: syncExtensions,
+            useLightIcon: useLightIcon
+        }, function () {
+            if (syncExtensions) {
+                syncType = document.getElementById('syncChoice').returnValue;
+                if (!syncType || syncType === 'local') {
+                    GSC.sync.getExtensions().then((extensions) => {
+                        var localExtensions = {};
+                        for (const [uuid, extension] of Object.entries(extensions)) {
+                            if (extension.local && extension.localState != EXTENSION_STATE.UNINSTALLED) {
+                                localExtensions[extension.uuid] = {
+                                    uuid: extension.uuid,
+                                    name: extension.name,
+                                    state: extension.localState
+                                };
+                            }
+                        };
+
+                        chrome.storage.sync.set({
+                            extensions: localExtensions
+                        }, function () {
+                            showSuccessStatus();
+                        });
+                    }).catch((message) => {
+                        showWithDelay($('#error'), 15000, message);
+                    });
+                }
+                else if (syncType === 'remote') {
+                    chrome.runtime.sendMessage(GS_CHROME_ID, MESSAGE_SYNC_FROM_REMOTE);
+                    showSuccessStatus();
+                }
+            }
+            else {
+                showSuccessStatus();
+            }
+        });
+    });
 }
 
-function showSuccessStatus()
-{
-       // Update status to let user know options were saved.
-       showWithDelay($('#status'), 750);
+function showSuccessStatus() {
+    // Update status to let user know options were saved.
+    showWithDelay($('#status'), 750);
 }
 
-function restore_options()
-{
-       init_tabs();
-
-       chrome.storage.sync.get(DEFAULT_SYNC_OPTIONS, function (items) {
-               function toggle_notice(show, id) {
-                       let notice = $('#' + id)
-                               .closest('dl')
-                               .querySelector('dt br, dt span.notice');
-
-                       if (show)
-                       {
-                               notice.style.display = 'block';
-                       }
-                       else
-                       {
-                               notice.style.display = 'none';
-                       }
-               }
-
-               function toggle_update_notice(show) {
-                       toggle_notice(show, "update_check_yes");
-               }
-
-               function toggle_update_enable_notice(show) {
-                       toggle_notice(show, "update_check_enabled_yes");
-               }
-
-               function disable_update_check() {
-                       if (items.updateCheck)
-                       {
-                               items.updateCheck = false;
-
-                               chrome.storage.sync.set({
-                                       updateCheck: items.updateCheck
-                               });
-                       }
-
-                       toggle_update_notice(true);
-               }
-
-               function disable_update_enabled_only() {
-                       if (items.updateCheckEnabledOnly)
-                       {
-                               items.updateCheckEnabledOnly = false;
-
-                               chrome.storage.sync.set({
-                                       updateCheckEnabledOnly: items.updateCheckEnabledOnly
-                               });
-                       }
-
-                       toggle_update_enable_notice(true);
-               }
-
-               GSC.onInitialize().then(function (response) {
-                       if (!GSC.nativeUpdateCheckSupported(response))
-                       {
-                               disable_update_check();
-                       }
-                       else
-                       {
-                               $$("input[name='update_check'], #update_check_period").forEach((input) => 
input.disabled = false);
-                               $('#update_check_period').value = items.updateCheckPeriod;
-                               toggle_update_notice(false);
-                               retrieveUpdateTimes();
-                       }
-
-                       if (!GSC.nativeUpdateCheckEnabledOnlySupported(response))
-                       {
-                               disable_update_enabled_only();
-                       }
-                       else
-                       {
-                               $("input[name='update_check_enabled']").disabled = false;
-                               toggle_update_enable_notice(false);
-                       }
-
-                       setCheckUpdate(items.updateCheck);
-                       setCheckUpdateEnabledOnly(items.updateCheckEnabledOnly);
-               }, function(response) {
-                       disable_update_check();
-               });
-
-               setReleaseNotes(items.showReleaseNotes);
-       });
-
-       if(COMPAT.SYNC_STORAGE)
-       {
-               updateSynchronizationStatus();
-       }
-       else
-       {
-               $('a[data-i18n="synchronization"]').parentNode.remove();
-               $('#synchronize_extensions_yes').closest('dl').style.display = 'none';
-       }
-
-       chrome.storage.local.get(DEFAULT_LOCAL_OPTIONS, function (items) {
-               if(items.syncExtensions)
-               {
-                       chrome.permissions.contains({
-                               permissions: ["idle"]
-                       }, function (result) {
-                               setSyncExtensions(result);
-                       });
-               }
-               else if(COMPAT.SYNC_STORAGE)
-               {
-                       setSyncExtensions(false);
-               }
-
-               setLightIcon(items.useLightIcon);
-       });
+function restore_options() {
+    init_tabs();
+
+    chrome.storage.sync.get(DEFAULT_SYNC_OPTIONS, function (items) {
+        function toggle_notice(show, id) {
+            let notice = $('#' + id)
+                .closest('dl')
+                .querySelector('dt br, dt span.notice');
+
+            if (show) {
+                notice.style.display = 'block';
+            }
+            else {
+                notice.style.display = 'none';
+            }
+        }
+
+        function toggle_update_notice(show) {
+            toggle_notice(show, "update_check_yes");
+        }
+
+        function toggle_update_enable_notice(show) {
+            toggle_notice(show, "update_check_enabled_yes");
+        }
+
+        function disable_update_check() {
+            if (items.updateCheck) {
+                items.updateCheck = false;
+
+                chrome.storage.sync.set({
+                    updateCheck: items.updateCheck
+                });
+            }
+
+            toggle_update_notice(true);
+        }
+
+        function disable_update_enabled_only() {
+            if (items.updateCheckEnabledOnly) {
+                items.updateCheckEnabledOnly = false;
+
+                chrome.storage.sync.set({
+                    updateCheckEnabledOnly: items.updateCheckEnabledOnly
+                });
+            }
+
+            toggle_update_enable_notice(true);
+        }
+
+        GSC.onInitialize().then(function (response) {
+            if (!GSC.nativeUpdateCheckSupported(response)) {
+                disable_update_check();
+            }
+            else {
+                $$("input[name='update_check'], #update_check_period").forEach((input) => input.disabled = 
false);
+                $('#update_check_period').value = items.updateCheckPeriod;
+                toggle_update_notice(false);
+                retrieveUpdateTimes();
+            }
+
+            if (!GSC.nativeUpdateCheckEnabledOnlySupported(response)) {
+                disable_update_enabled_only();
+            }
+            else {
+                $("input[name='update_check_enabled']").disabled = false;
+                toggle_update_enable_notice(false);
+            }
+
+            setCheckUpdate(items.updateCheck);
+            setCheckUpdateEnabledOnly(items.updateCheckEnabledOnly);
+        }, function (response) {
+            disable_update_check();
+        });
+
+        setReleaseNotes(items.showReleaseNotes);
+    });
+
+    if (COMPAT.SYNC_STORAGE) {
+        updateSynchronizationStatus();
+    }
+    else {
+        $('a[data-i18n="synchronization"]').parentNode.remove();
+        $('#synchronize_extensions_yes').closest('dl').style.display = 'none';
+    }
+
+    chrome.storage.local.get(DEFAULT_LOCAL_OPTIONS, function (items) {
+        if (items.syncExtensions) {
+            chrome.permissions.contains({
+                permissions: ["idle"]
+            }, function (result) {
+                setSyncExtensions(result);
+            });
+        }
+        else if (COMPAT.SYNC_STORAGE) {
+            setSyncExtensions(false);
+        }
+
+        setLightIcon(items.useLightIcon);
+    });
 }
 
-function retrieveUpdateTimes()
-{
-       chrome.storage.local.get({
-               lastUpdateCheck: null
-       }, function (items) {
-               if(items.lastUpdateCheck)
-               {
-                       $('#last_update_check').innerText = items.lastUpdateCheck;
-               }
-               else
-               {
-                       $('#last_update_check').innerText = m('never');
-               }
-       });
-
-       retrieveNextUpdateTime();
+function retrieveUpdateTimes() {
+    chrome.storage.local.get({
+        lastUpdateCheck: null
+    }, function (items) {
+        if (items.lastUpdateCheck) {
+            $('#last_update_check').innerText = items.lastUpdateCheck;
+        }
+        else {
+            $('#last_update_check').innerText = m('never');
+        }
+    });
+
+    retrieveNextUpdateTime();
 }
 
-function retrieveNextUpdateTime()
-{
-       chrome.alarms.get(ALARM_UPDATE_CHECK, function (alarm) {
-               if (alarm)
-               {
-                       $('#next_update_check').innerText = new Date(alarm.scheduledTime).toLocaleString();
-               }
-               else
-               {
-                       $('#next_update_check').innerText = m('never');
-               }
-       });
+function retrieveNextUpdateTime() {
+    chrome.alarms.get(ALARM_UPDATE_CHECK, function (alarm) {
+        if (alarm) {
+            $('#next_update_check').innerText = new Date(alarm.scheduledTime).toLocaleString();
+        }
+        else {
+            $('#next_update_check').innerText = m('never');
+        }
+    });
 }
 
-function setCheckUpdate(result)
-{
-       if(result)
-               $('#update_check_yes').checked = true;
-       else
-               $('#update_check_no').checked = true;
+function setCheckUpdate(result) {
+    if (result)
+        $('#update_check_yes').checked = true;
+    else
+        $('#update_check_no').checked = true;
 }
 
-function setCheckUpdateEnabledOnly(result)
-{
-       if(result)
-               $('#update_check_enabled_yes').checked = true;
-       else
-               $('#update_check_enabled_no').checked = true;
+function setCheckUpdateEnabledOnly(result) {
+    if (result)
+        $('#update_check_enabled_yes').checked = true;
+    else
+        $('#update_check_enabled_no').checked = true;
 }
 
-function setLightIcon(result)
-{
-       if(result)
-               $('#use_light_icon_yes').checked = true;
-       else
-               $('#use_light_icon_no').checked = true;
+function setLightIcon(result) {
+    if (result)
+        $('#use_light_icon_yes').checked = true;
+    else
+        $('#use_light_icon_no').checked = true;
 }
 
-function setReleaseNotes(result)
-{
-       if(result)
-               $('#show_release_notes_yes').checked = true;
-       else
-               $('#show_release_notes_no').checked = true;
+function setReleaseNotes(result) {
+    if (result)
+        $('#show_release_notes_yes').checked = true;
+    else
+        $('#show_release_notes_no').checked = true;
 }
 
-function setSyncExtensions(result)
-{
-       if(result)
-               $('#synchronize_extensions_yes').checked = true;
-       else
-               $('#synchronize_extensions_no').checked = true;
+function setSyncExtensions(result) {
+    if (result)
+        $('#synchronize_extensions_yes').checked = true;
+    else
+        $('#synchronize_extensions_no').checked = true;
 }
 
-function handleSynchronize()
-{
-       if($('#synchronize_extensions_yes').checked)
-       {
-               chrome.permissions.request({
-                       permissions: ["idle"]
-               }, function(granted) {
-                       if(granted)
-                       {
-                               if(!COMPAT.PERMISSIONS_EVENTS)
-                               {
-                                       chrome.runtime.sendMessage(GS_CHROME_ID, 
MESSAGE_IDLE_PERMISSION_ADDED);
-                               }
-
-                               chrome.storage.sync.get({
-                                       extensions: {}
-                               }, function (options) {
-                                       if(!isEmptyObject(options.extensions))
-                                       {
-                                               document.getElementById('syncChoice').showModal();
-                                       }
-                               });
-                       }
-                       else
-                       {
-                               if(!COMPAT.PERMISSIONS_EVENTS)
-                               {
-                                       chrome.runtime.sendMessage(GS_CHROME_ID, 
MESSAGE_IDLE_PERMISSION_REMOVED);
-                               }
-
-                               setSyncExtensions(false);
-                       }
-               });
-       }
-       else
-       {
-               chrome.permissions.remove({
-                       permissions: ["idle"]
-               }, function(removed) {
-                       if(removed && !COMPAT.PERMISSIONS_EVENTS)
-                       {
-                               chrome.runtime.sendMessage(GS_CHROME_ID, MESSAGE_IDLE_PERMISSION_REMOVED);
-                       }
-
-                       setSyncExtensions(!removed);
-               });
-       }
+function handleSynchronize() {
+    if ($('#synchronize_extensions_yes').checked) {
+        chrome.permissions.request({
+            permissions: ["idle"]
+        }, function (granted) {
+            if (granted) {
+                if (!COMPAT.PERMISSIONS_EVENTS) {
+                    chrome.runtime.sendMessage(GS_CHROME_ID, MESSAGE_IDLE_PERMISSION_ADDED);
+                }
+
+                chrome.storage.sync.get({
+                    extensions: {}
+                }, function (options) {
+                    if (!isEmptyObject(options.extensions)) {
+                        document.getElementById('syncChoice').showModal();
+                    }
+                });
+            }
+            else {
+                if (!COMPAT.PERMISSIONS_EVENTS) {
+                    chrome.runtime.sendMessage(GS_CHROME_ID, MESSAGE_IDLE_PERMISSION_REMOVED);
+                }
+
+                setSyncExtensions(false);
+            }
+        });
+    }
+    else {
+        chrome.permissions.remove({
+            permissions: ["idle"]
+        }, function (removed) {
+            if (removed && !COMPAT.PERMISSIONS_EVENTS) {
+                chrome.runtime.sendMessage(GS_CHROME_ID, MESSAGE_IDLE_PERMISSION_REMOVED);
+            }
+
+            setSyncExtensions(!removed);
+        });
+    }
 }
 
-function updateSynchronizationStatus()
-{
-       GSC.sync.getExtensions().then((extensions) => {
-               var keys = Object.keys(extensions).sort(function (a, b) {
-                       var nameA = extensions[a].name.toLowerCase();
-                       var nameB = extensions[b].name.toLowerCase();
-
-                       if (nameA < nameB)
-                       {
-                               return -1;
-                       }
-
-                       if (nameA > nameB)
-                       {
-                               return 1;
-                       }
-
-                       return 0;
-               });
-
-               empty($('#synchronization table tbody'));
-               for (const [key, uuid] of Object.entries(keys)) {
-                       var extension = extensions[uuid];
-
-                       $('#synchronization table tbody').insertAdjacentHTML(
-                               'beforeEnd',
-                               `<tr>
+function updateSynchronizationStatus() {
+    GSC.sync.getExtensions().then((extensions) => {
+        var keys = Object.keys(extensions).sort(function (a, b) {
+            var nameA = extensions[a].name.toLowerCase();
+            var nameB = extensions[b].name.toLowerCase();
+
+            if (nameA < nameB) {
+                return -1;
+            }
+
+            if (nameA > nameB) {
+                return 1;
+            }
+
+            return 0;
+        });
+
+        empty($('#synchronization table tbody'));
+        for (const [key, uuid] of Object.entries(keys)) {
+            var extension = extensions[uuid];
+
+            $('#synchronization table tbody').insertAdjacentHTML(
+                'beforeEnd',
+                `<tr>
                                        <td>${extension.name}</td>
                                        <td class='${extension.local && 'ok' || 'fail'}'></td>
                                        <td class='${extension.localState == EXTENSION_STATE.ENABLED && 'ok' 
|| 'fail'}'></td>
                                        <td class='${extension.remote && 'ok' || 'fail'}'></td>
                                </tr>`
-                       );
-               };
-       }).catch((message) => {
-               showWithDelay($('#error'), 15000, message);
-       });
+            );
+        };
+    }).catch((message) => {
+        showWithDelay($('#error'), 15000, message);
+    });
 }
 
 i18n();
@@ -372,58 +328,49 @@ document.addEventListener('DOMContentLoaded', restore_options);
 document.getElementById('save').addEventListener('click', save_options);
 
 document.getElementsByName('synchronize_extensions').forEach((control) => {
-       control.addEventListener('change', handleSynchronize);
+    control.addEventListener('change', handleSynchronize);
 });
 
 
-document.getElementById('syncChoice').addEventListener('close', function() {
-       if(document.getElementById('syncChoice').returnValue === 'cancel')
-       {
-               chrome.permissions.remove({
-                       permissions: ["idle"]
-               }, function(removed) {
-                       if(removed && !COMPAT.PERMISSIONS_EVENTS)
-                       {
-                               chrome.runtime.sendMessage(GS_CHROME_ID, MESSAGE_IDLE_PERMISSION_REMOVED);
-                       }
-
-                       setSyncExtensions(!removed);
-               });
-       }
+document.getElementById('syncChoice').addEventListener('close', function () {
+    if (document.getElementById('syncChoice').returnValue === 'cancel') {
+        chrome.permissions.remove({
+            permissions: ["idle"]
+        }, function (removed) {
+            if (removed && !COMPAT.PERMISSIONS_EVENTS) {
+                chrome.runtime.sendMessage(GS_CHROME_ID, MESSAGE_IDLE_PERMISSION_REMOVED);
+            }
+
+            setSyncExtensions(!removed);
+        });
+    }
 });
 
-if(!$('#translation_credits div').firstChild)
-{
-       $('.translation_credits_container').remove();
+if (!$('#translation_credits div').firstChild) {
+    $('.translation_credits_container').remove();
 }
 
 chrome.storage.onChanged.addListener(function (changes, areaName) {
-       if (areaName === 'local')
-       {
-               if(changes.lastUpdateCheck && changes.lastUpdateCheck.newValue)
-               {
-                       $('#last_update_check').innerText = changes.lastUpdateCheck.newValue;
-               }
-       }
-
-       if (areaName === 'sync' && changes.extensions)
-       {
-               updateSynchronizationStatus();
-       }
+    if (areaName === 'local') {
+        if (changes.lastUpdateCheck && changes.lastUpdateCheck.newValue) {
+            $('#last_update_check').innerText = changes.lastUpdateCheck.newValue;
+        }
+    }
+
+    if (areaName === 'sync' && changes.extensions) {
+        updateSynchronizationStatus();
+    }
 });
 
 chrome.runtime.onMessage.addListener(
-       function (request, sender, sendResponse) {
-               if(sender.id && sender.id === GS_CHROME_ID && request)
-               {
-                       if(request === MESSAGE_NEXT_UPDATE_CHANGED)
-                       {
-                               retrieveNextUpdateTime();
-                       }
-                       else if(request.signal && request.signal === SIGNAL_EXTENSION_CHANGED)
-                       {
-                               updateSynchronizationStatus();
-                       }
-               }
-       }
+    function (request, sender, sendResponse) {
+        if (sender.id && sender.id === GS_CHROME_ID && request) {
+            if (request === MESSAGE_NEXT_UPDATE_CHANGED) {
+                retrieveNextUpdateTime();
+            }
+            else if (request.signal && request.signal === SIGNAL_EXTENSION_CHANGED) {
+                updateSynchronizationStatus();
+            }
+        }
+    }
 );


[Date Prev][Date Next]   [Thread Prev][Thread Next]   [Thread Index] [Date Index] [Author Index]