├── .eslintrc.json ├── .github ├── issue_template.md └── stale.yml ├── .gitignore ├── .prettierrc ├── Gruntfile.js ├── LICENSE ├── NOTICE ├── README.md ├── TODO ├── pack_xpi.sh ├── package-lock.json ├── package.json └── src ├── _locales ├── de │ └── messages.json ├── en │ └── messages.json ├── pt_BR │ └── messages.json ├── pt_PT │ └── messages.json ├── ru │ └── messages.json ├── zh_CN │ └── messages.json └── zh_TW │ └── messages.json ├── about.html ├── broken.html ├── css ├── debug.css ├── fontello.css ├── popup.css ├── style.css └── suspended.css ├── debug.html ├── font ├── fontello.woff └── fontello.woff2 ├── history.html ├── img ├── chromeDefaultFavicon.png ├── chromeDefaultFaviconSml.png ├── chromeDevDefaultFavicon.png ├── chromeDevDefaultFaviconSml.png ├── ic_suspendy_128x128.png ├── ic_suspendy_16x16.png ├── ic_suspendy_16x16_grey.png ├── ic_suspendy_32x32.png ├── ic_suspendy_32x32_grey.png ├── ic_suspendy_48x48.png ├── snoozy_tab.svg ├── snoozy_tab_awake.svg ├── suspendy-guy-alt.png ├── suspendy-guy-oops.png ├── suspendy-guy-uh-oh.png └── suspendy-guy.png ├── js ├── about.js ├── background.js ├── broken.js ├── contentscript.js ├── db.js ├── debug.js ├── dom-to-image-more.js ├── gsChrome.js ├── gsFavicon.js ├── gsIndexedDb.js ├── gsMessages.js ├── gsSession.js ├── gsStorage.js ├── gsSuspendedTab.js ├── gsTabCheckManager.js ├── gsTabDiscardManager.js ├── gsTabQueue.js ├── gsTabSuspendManager.js ├── gsUtils.js ├── history.js ├── historyItems.js ├── historyUtils.js ├── html2canvas.js ├── html2canvas.min.js ├── notice.js ├── options.js ├── permissions.js ├── popup.js ├── recovery.js ├── restoring-window.js ├── shortcuts.js ├── tests │ ├── fixture_currentSessions.json │ ├── fixture_previewUrls.json │ ├── fixture_savedSessions.json │ ├── test_createAndUpdateSessionRestorePoint.js │ ├── test_currentSessions.js │ ├── test_gsChrome.js │ ├── test_gsTabQueue.js │ ├── test_gsUtils.js │ ├── test_savedSessions.js │ ├── test_suspendTab.js │ ├── test_trimDbItems.js │ ├── test_updateCurrentSession.js │ └── tests.js ├── update.js └── updated.js ├── manifest.json ├── notice.html ├── options.html ├── permissions.html ├── popup.html ├── recovery.html ├── restoring-window.html ├── shortcuts.html ├── suspended.html ├── tests.html ├── update.html └── updated.html /.eslintrc.json: -------------------------------------------------------------------------------- 1 | { 2 | "root": true, 3 | "parserOptions": { 4 | "ecmaVersion": 2017 5 | }, 6 | "env": { 7 | "browser": true, 8 | "es6": true, 9 | "webextensions": true 10 | }, 11 | "extends": [ 12 | "eslint:recommended", 13 | "prettier" 14 | ], 15 | "rules": { 16 | "no-console": 0, 17 | "no-unused-vars": [ 18 | "error", 19 | { "vars": "all", "args": "none", "ignoreRestSiblings": false } 20 | ], 21 | "no-undef": ["error"], 22 | "no-proto": ["error"], 23 | // "prefer-arrow-callback": ["warn"], TODO: refactor to use arrow functions 24 | // "no-var": ["error"], TODO: refactor to use let and const 25 | "prefer-spread": ["warn"], 26 | // "semi": ["error", "always"], 27 | "padded-blocks": ["off", { "blocks": "never" }], 28 | // "indent": ["error", 2], 29 | "one-var": ["off", "never"], 30 | "spaced-comment": ["off", "always"] 31 | // "space-before-function-paren": [ 32 | // "error", 33 | // { 34 | // "anonymous": "always", 35 | // "named": "never", 36 | // "asyncArrow": "always" 37 | // } 38 | // ] 39 | } 40 | } 41 | -------------------------------------------------------------------------------- /.github/issue_template.md: -------------------------------------------------------------------------------- 1 | Please complete the following information when submitting a feature request or bug report. 2 | * Extension version: 3 | * Browser name & version: 4 | * Operating system & version: 5 | 6 | And please also do a search for your request/bug before create a new one thanks! 7 | -------------------------------------------------------------------------------- /.github/stale.yml: -------------------------------------------------------------------------------- 1 | # Number of days of inactivity before an issue becomes stale 2 | daysUntilStale: 180 3 | # Number of days of inactivity before a stale issue is closed 4 | daysUntilClose: 30 5 | # Issues with these labels will never be considered stale 6 | exemptLabels: 7 | - blocked 8 | - bug-confirmed 9 | - feature 10 | - pinned 11 | - prioritised 12 | - released 13 | - security 14 | - waiting-on-release 15 | # Label to use when marking an issue as stale 16 | staleLabel: stale 17 | # Comment to post when marking an issue as stale. Set to `false` to disable 18 | markComment: > 19 | This issue has been automatically marked as stale because it has not had any new 20 | activity for 180 days. It will be closed in 30 days if no further activity occurs. 21 | Thank you for your contributions. 22 | # Comment to post when closing a stale issue. Set to `false` to disable 23 | closeComment: false 24 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | **/Thumbs.db 2 | **/*.pem 3 | /node_modules 4 | /assets/* 5 | /build/* 6 | /.idea/* 7 | /.debris/* 8 | build/zip/thegreatsuspender-6.30-dev/welcome.html 9 | .DS_Store 10 | -------------------------------------------------------------------------------- /.prettierrc: -------------------------------------------------------------------------------- 1 | { 2 | "_COMMENT1": "THE BELOW SHOULD BE THE DEFAULTS BUT IT'S NICE TO SEE WHAT THEY ARE", 3 | "printWidth": 80, 4 | "tabWidth": 2, 5 | 6 | 7 | "_COMMENT2": "THE BELOW OVERRIDE PRETTIER'S DEFAULTS", 8 | "singleQuote": true, 9 | "trailingComma": "es5", 10 | } 11 | -------------------------------------------------------------------------------- /Gruntfile.js: -------------------------------------------------------------------------------- 1 | module.exports = function(grunt) { 2 | require('time-grunt')(grunt); 3 | 4 | grunt.initConfig({ 5 | pkg: grunt.file.readJSON('package.json'), 6 | manifest: grunt.file.readJSON('src/manifest.json'), 7 | config: { 8 | tempDir: 9 | grunt.cli.tasks[0] === 'tgut' ? 'build/tgut-temp/' : 'build/tgs-temp/', 10 | buildName: 11 | grunt.cli.tasks[0] === 'tgut' 12 | ? 'tgut-<%= manifest.version %>' 13 | : 'tgs-<%= manifest.version %>', 14 | }, 15 | copy: { 16 | main: { 17 | expand: true, 18 | src: ['src/**', '!src/tests.html', '!src/js/tests/**'], 19 | dest: '<%= config.tempDir %>', 20 | }, 21 | }, 22 | 'string-replace': { 23 | debugoff: { 24 | files: { 25 | '<%= config.tempDir %>src/js/': 26 | '<%= config.tempDir %>src/js/gsUtils.js', 27 | }, 28 | options: { 29 | replacements: [ 30 | { 31 | pattern: /debugInfo\s*=\s*true/, 32 | replacement: 'debugInfo = false', 33 | }, 34 | { 35 | pattern: /debugError\s*=\s*true/, 36 | replacement: 'debugError = false', 37 | }, 38 | ], 39 | }, 40 | }, 41 | debugon: { 42 | files: { 43 | '<%= config.tempDir %>src/js/': 44 | '<%= config.tempDir %>src/js/gsUtils.js', 45 | }, 46 | options: { 47 | replacements: [ 48 | { 49 | pattern: /debugInfo\s*=\s*false/, 50 | replacement: 'debugInfo = true', 51 | }, 52 | { 53 | pattern: /debugError\s*=\s*false/, 54 | replacement: 'debugError = true', 55 | }, 56 | ], 57 | }, 58 | }, 59 | localesTgut: { 60 | files: { 61 | '<%= config.tempDir %>src/_locales/': 62 | '<%= config.tempDir %>src/_locales/**', 63 | }, 64 | options: { 65 | replacements: [ 66 | { 67 | pattern: /The Great Suspender/gi, 68 | replacement: 'The Great Tester', 69 | }, 70 | ], 71 | }, 72 | }, 73 | }, 74 | crx: { 75 | public: { 76 | src: [ 77 | '<%= config.tempDir %>src/**/*', 78 | '!**/html2canvas.js', 79 | '!**/Thumbs.db', 80 | ], 81 | dest: 'build/zip/<%= config.buildName %>.zip', 82 | }, 83 | private: { 84 | src: [ 85 | '<%= config.tempDir %>src/**/*', 86 | '!**/html2canvas.js', 87 | '!**/Thumbs.db', 88 | ], 89 | dest: 'build/crx/<%= config.buildName %>.crx', 90 | options: { 91 | privateKey: 'key.pem', 92 | }, 93 | }, 94 | }, 95 | clean: ['<%= config.tempDir %>'], 96 | }); 97 | 98 | grunt.loadNpmTasks('grunt-contrib-copy'); 99 | grunt.loadNpmTasks('grunt-string-replace'); 100 | grunt.loadNpmTasks('grunt-crx'); 101 | grunt.loadNpmTasks('grunt-contrib-clean'); 102 | grunt.registerTask('default', [ 103 | 'copy', 104 | 'string-replace:debugoff', 105 | 'crx:public', 106 | 'crx:private', 107 | 'clean', 108 | ]); 109 | grunt.registerTask('tgut', [ 110 | 'copy', 111 | 'string-replace:debugon', 112 | 'string-replace:localesTgut', 113 | 'crx:public', 114 | 'crx:private', 115 | 'clean', 116 | ]); 117 | }; 118 | -------------------------------------------------------------------------------- /NOTICE: -------------------------------------------------------------------------------- 1 | src/js/html2canvas.js 2 | src/js/html2canvas.min.js 3 | -- 4 | https://github.com/niklasvh/html2canvas 5 | -- 6 | 7 | Copyright (c) 2012 Niklas von Hertzen 8 | 9 | Permission is hereby granted, free of charge, to any person 10 | obtaining a copy of this software and associated documentation 11 | files (the "Software"), to deal in the Software without 12 | restriction, including without limitation the rights to use, 13 | copy, modify, merge, publish, distribute, sublicense, and/or sell 14 | copies of the Software, and to permit persons to whom the 15 | Software is furnished to do so, subject to the following 16 | conditions: 17 | 18 | The above copyright notice and this permission notice shall be 19 | included in all copies or substantial portions of the Software. 20 | 21 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, 22 | EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES 23 | OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND 24 | NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT 25 | HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, 26 | WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING 27 | FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR 28 | OTHER DEALINGS IN THE SOFTWARE. 29 | 30 | ------------------------------------------------------------- 31 | 32 | src/js/db.js 33 | -- 34 | https://github.com/aaronpowell/db.js 35 | -- 36 | 37 | The MIT License 38 | 39 | Copyright (c) 2012 Aaron Powell 40 | 41 | Permission is hereby granted, free of charge, to any person 42 | obtaining a copy of this software and associated documentation 43 | files (the "Software"), to deal in the Software without 44 | restriction, including without limitation the rights to use, 45 | copy, modify, merge, publish, distribute, sublicense, and/or sell 46 | copies of the Software, and to permit persons to whom the 47 | Software is furnished to do so, subject to the following 48 | conditions: 49 | 50 | The above copyright notice and this permission notice shall be 51 | included in all copies or substantial portions of the Software. 52 | 53 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, 54 | EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES 55 | OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND 56 | NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT 57 | HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, 58 | WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING 59 | FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR 60 | OTHER DEALINGS IN THE SOFTWARE. 61 | 62 | ------------------------------------------------------------- 63 | 64 | src/js/dom-to-image.js 65 | src/js/dom-to-image-more.js 66 | -- 67 | https://github.com/tsayen/dom-to-image 68 | https://github.com/1904labs/dom-to-image-more 69 | -- 70 | 71 | The MIT License (MIT) 72 | 73 | Copyright 2018 Marc Brooks 74 | https://about.me/idisposable 75 | 76 | Copyright 2015 Anatolii Saienko 77 | https://github.com/tsayen 78 | 79 | Copyright 2012 Paul Bakaus 80 | http://paulbakaus.com/ 81 | 82 | Permission is hereby granted, free of charge, to any person obtaining 83 | a copy of this software and associated documentation files (the 84 | "Software"), to deal in the Software without restriction, including 85 | without limitation the rights to use, copy, modify, merge, publish, 86 | distribute, sublicense, and/or sell copies of the Software, and to 87 | permit persons to whom the Software is furnished to do so, subject to 88 | the following conditions: 89 | 90 | The above copyright notice and this permission notice shall be 91 | included in all copies or substantial portions of the Software. 92 | 93 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, 94 | EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF 95 | MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND 96 | NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE 97 | LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION 98 | OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION 99 | WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. 100 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # Unofficial The Great Suspender for Firefox 2 | 3 | This is a work-in-progress port of "The Great Suspender" WebExtension for Mozilla Firefox and compatible browsers. 4 | Current TODO could be found in [TODO](TODO). 5 | You may also submit a bug or pull request via Github. 6 | 7 | ## Debugging 8 | 9 | You may load it as a temporary extension at `about:debugging` 10 | 11 | ## Building from source 12 | 13 | There are two options 14 | 1. Use npm as described in the original README and rename zip/crx into xpi 15 | 2. Use `pack_xpi.sh` or do the same manually 16 | 17 | **NOTE**: to install an unsigned xpi you should use Developer or custom build of Firefox with disabled extension signing enforcement. It may also work on some GNU/Linux distros. *Search for it yourself if required*. 18 | 19 | 20 | The rest of README file carried from the original project without change. However **do not** submit bug reports about *this* project to *original author's repository*. 21 | 22 | # The Great Suspender 23 | 24 | 25 | 26 | "The Great Suspender" is a free and open-source Google Chrome extension for people who find that chrome is consuming too much system resource or suffer from frequent chrome crashing. Once installed and enabled, this extension will automatically *suspend* tabs that have not been used for a while, freeing up memory and cpu that the tab was consuming. 27 | 28 | If you have suggestions or problems using the extension, please [submit a bug or a feature request](https://github.com/deanoemcke/thegreatsuspender/issues/). For other enquiries you can email me at greatsuspender@gmail.com. 29 | 30 | ### New release! 31 | I am currently rolling out a new chrome webstore release. 32 | 33 | **If you have lost tabs from your browser:** I have written a guide for how to recover your lost tabs [here](https://github.com/deanoemcke/thegreatsuspender/issues/526 34 | ). 35 | 36 | ### Chrome Web Store 37 | 38 | The Great Suspender is [available via the official Chrome Web Store](https://chrome.google.com/webstore/detail/the-great-suspender/klbibkeccnjlkjkiokjodocebajanakg). 39 | 40 | Please note that the webstore version may be behind the latest version here. That is because I try to keep webstore updates down to a minimum due to their [disruptive effect](https://github.com/deanoemcke/thegreatsuspender/issues/526). 41 | 42 | For more information on the permissions required for the extension, please refer to this gitHub issue: (https://github.com/deanoemcke/thegreatsuspender/issues/213) 43 | 44 | ### Install as an extension from source 45 | 46 | 1. Download the **[latest available version](https://github.com/deanoemcke/thegreatsuspender/releases)** and unarchive to your preferred location (whichever suits you). 47 | 2. Using **Google Chrome** browser, navigate to chrome://extensions/ and enable "Developer mode" in the upper right corner. 48 | 3. Click on the Load unpacked extension... button. 49 | 4. Browse to the src directory of the unarchived folder and confirm. 50 | 51 | If you have completed the above steps, the "welcome" page will open indicating successful installation of the extension. 52 | 53 | Be sure to unsuspend all suspended tabs before removing any other version of the extension or they will disappear forever! 54 | 55 | ### Build from github 56 | 57 | Dependencies: openssl, npm. 58 | 59 | Clone the repository and run these commands: 60 | ``` 61 | npm install 62 | npm run generate-key 63 | npm run build 64 | ``` 65 | 66 | It should say: 67 | ``` 68 | Done, without errors. 69 | ``` 70 | 71 | The extension in crx format will be inside the build/crx/ directory. You can drag it into [extensions] (chrome://extensions) to install locally. 72 | 73 | ### Integrating with another Chrome extension or app 74 | 75 | This extension has a small external api to allow other extensions to request the suspension of a tab. See [this issue](https://github.com/deanoemcke/thegreatsuspender/issues/276) for more information. And please let me know about it so that I can try it out! 76 | 77 | ### Contributing to this extension 78 | 79 | Contributions are very welcome. Feel free to submit pull requests for new features and bug fixes. For new features, ideally you would raise an issue for the proposed change first so that we can discuss ideas. This will go a long way to ensuring your pull request is accepted. 80 | 81 | ### License 82 | 83 | This work is licensed under a GNU GENERAL PUBLIC LICENSE (v2) 84 | 85 | ### Shoutouts 86 | 87 | This package uses the [html2canvas](https://github.com/niklasvh/html2canvas) library written by Niklas von Hertzen. 88 | It also uses the indexedDb wrapper [db.js](https://github.com/aaronpowell/db.js) written by Aaron Powell. 89 | Thank you also to [BrowserStack](https://www.browserstack.com) for providing free chrome testing tools. 90 | -------------------------------------------------------------------------------- /TODO: -------------------------------------------------------------------------------- 1 | + Done 2 | - Discarded 3 | * New 4 | H On hold 5 | 6 | + Remove autodicardable to avoid code breaking in Firefox 7 | H Keep up with Firefox changes on autodiscardable: unsupported as of 2019.08 8 | + Fix screenshots: check Firefox Screenshots 9 | + Fix Firefox shortcut: Ctrl+Shift+S confilicts with Firefox Screenshots 10 | * Rebrand: name 11 | * Rebrand: logo 12 | * Migrate to building with Makefile 13 | * Publish to AMO 14 | -------------------------------------------------------------------------------- /pack_xpi.sh: -------------------------------------------------------------------------------- 1 | #!/bin/bash 2 | mkdir -p build/xpi 3 | cd src 4 | find -not -name 'html2canvas.min*' -not -name 'Thumbs.db' | xargs zip -q ../build/xpi/tgs-firefox.xpi 5 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "thegreatsuspender", 3 | "version": "0.0.0", 4 | "description": "A chrome extension for suspending all tabs to free up memory", 5 | "main": "", 6 | "scripts": { 7 | "build": "grunt", 8 | "generate-key": "openssl genrsa -out key.pem", 9 | "test": "echo \"Error: no test specified\" && exit 1", 10 | "eslint-check": "eslint --print-config .eslintrc.js | eslint-config-prettier-check" 11 | }, 12 | "repository": { 13 | "type": "git", 14 | "url": "git://github.com/deanoemcke/thegreatsuspender.git" 15 | }, 16 | "keywords": [ 17 | "chrome", 18 | "extension", 19 | "addon", 20 | "memory", 21 | "suspend", 22 | "tab" 23 | ], 24 | "author": "deanoemcke", 25 | "license": "GPLv2", 26 | "bugs": { 27 | "url": "https://github.com/deanoemcke/thegreatsuspender/issues" 28 | }, 29 | "devDependencies": { 30 | "eslint": "^4.19.1", 31 | "eslint-config-prettier": "^2.9.0", 32 | "eslint-config-standard": "^10.2.1", 33 | "eslint-plugin-import": "^2.7.0", 34 | "eslint-plugin-node": "^5.1.1", 35 | "eslint-plugin-promise": "^3.5.0", 36 | "eslint-plugin-standard": "^3.0.1", 37 | "grunt": "~0.4.5", 38 | "grunt-cli": "^1.2.0", 39 | "grunt-contrib-clean": "^1.1.0", 40 | "grunt-contrib-copy": "^1.0.0", 41 | "grunt-crx": "~1.0.5", 42 | "grunt-string-replace": "^1.3.1", 43 | "prettier": "1.13.7", 44 | "time-grunt": "~1.2.1" 45 | }, 46 | "dependencies": { 47 | "db.js": "^0.15.0" 48 | } 49 | } 50 | -------------------------------------------------------------------------------- /src/about.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 |
15 | 16 | 19 | 20 |
21 | 35 |
36 | 37 |
38 | 39 |

40 | 41 |

42 | 43 | github.com/dvalter/thegreatsuspender 44 |
45 |

46 | 47 |

48 | 49 | 50 | 51 |

52 | 53 |

54 | 55 | 56 |

57 | 58 |
59 |
60 |

61 | 62 | 63 | 64 |
65 | 66 | 67 | 68 |
69 | 70 | 71 | 72 |
73 | 78 |

79 |
80 | 81 |
82 | 83 |
84 | 85 | 86 | 87 | 88 | -------------------------------------------------------------------------------- /src/broken.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | The Great Suspender is broken 8 | 9 | 10 | 11 | 12 | 13 |
14 | 15 | 16 | 17 |
18 |

Ruh Roh!

19 |

The Great Suspender failed to start. Perhaps you are using an incompatible version of chrome?

20 |

Try to restart the extension. If the problem persists, ask for help on the GitHub project page.

21 |

You can recover lost tabs from the session management page.

22 |
23 | 24 |
25 |
26 | 27 | 28 | 29 | 30 | -------------------------------------------------------------------------------- /src/css/debug.css: -------------------------------------------------------------------------------- 1 | body { 2 | padding: 20px; 3 | font-size: 1.3rem; 4 | } 5 | 6 | td { 7 | max-width:700px; 8 | white-space:nowrap; 9 | overflow:hidden; 10 | text-overflow:ellipsis; 11 | } 12 | 13 | img { 14 | height: 16px; 15 | width: 16px; 16 | } -------------------------------------------------------------------------------- /src/css/fontello.css: -------------------------------------------------------------------------------- 1 | @font-face { 2 | font-family: 'fontello'; 3 | src: url('../font/fontello.woff2?80430491') format('woff2'), 4 | url('../font/fontello.woff?80430491') format('woff'); 5 | font-weight: normal; 6 | font-style: normal; 7 | } 8 | /* Chrome hack: SVG is rendered more smooth in Windozze. 100% magic, uncomment if you need it. */ 9 | /* Note, that will break hinting! In other OS-es font will be not as sharp as it could be */ 10 | /* 11 | @media screen and (-webkit-min-device-pixel-ratio:0) { 12 | @font-face { 13 | font-family: 'fontello'; 14 | src: url('../font/fontello.svg?80430491#fontello') format('svg'); 15 | } 16 | } 17 | */ 18 | 19 | [class^='icon-']:before, 20 | [class*=' icon-']:before { 21 | font-family: 'fontello'; 22 | font-style: normal; 23 | font-weight: normal; 24 | speak: none; 25 | 26 | display: inline-block; 27 | text-decoration: inherit; 28 | width: 1em; 29 | margin-right: 0.2em; 30 | text-align: center; 31 | /* opacity: .8; */ 32 | 33 | /* For safety - reset parent styles, that can break glyph codes*/ 34 | font-variant: normal; 35 | text-transform: none; 36 | 37 | /* fix buttons height, for twitter bootstrap */ 38 | line-height: 1em; 39 | 40 | /* Animation center compensation - margins should be symmetric */ 41 | /* remove if not needed */ 42 | margin-left: 0.2em; 43 | 44 | /* you can be more comfortable with increased icons size */ 45 | /* font-size: 120%; */ 46 | 47 | /* Font smoothing. That was taken from TWBS */ 48 | -webkit-font-smoothing: antialiased; 49 | -moz-osx-font-smoothing: grayscale; 50 | 51 | /* Uncomment for 3D effect */ 52 | /* text-shadow: 1px 1px 1px rgba(127, 127, 127, 0.3); */ 53 | } 54 | 55 | .icon-help-circled:before { 56 | content: '\e80a'; 57 | } /* '' */ 58 | .icon-check-empty:before { 59 | content: '\f096'; 60 | } /* '' */ 61 | .icon-sort-down:before { 62 | content: '\f0dd'; 63 | } /* '' */ 64 | .icon-minus-squared-alt:before { 65 | content: '\f147'; 66 | } /* '' */ 67 | .icon-ok-squared:before { 68 | content: '\f14a'; 69 | } /* '' */ 70 | .icon-plus-squared-alt:before { 71 | content: '\f196'; 72 | } /* '' */ 73 | -------------------------------------------------------------------------------- /src/css/popup.css: -------------------------------------------------------------------------------- 1 | /* 2 | 3 | - - /|_/| .-------------------. 4 | - _______________| @.@| / Styles by ) 5 | -- (______ >\_C/< ---/ Liam Johnston / 6 | - - / ______ _/____) ( @liamjohnstonnz / 7 | - - / /\ \ \ \ `-------------------' 8 | - (_/ \_) - \_) 9 | 10 | */ 11 | html, 12 | body { 13 | margin: 0; 14 | padding: 0; 15 | } 16 | html { 17 | box-sizing: border-box; 18 | } 19 | *, 20 | *::before, 21 | *::after { 22 | box-sizing: inherit; 23 | } 24 | body { 25 | font-family: system-ui, -apple-system, BlinkMacSystemFont, 'Segoe UI', 26 | Helvetica, Arial, sans-serif, 'Apple Color Emoji', 'Segoe UI Emoji', 27 | 'Segoe UI Symbol'; 28 | color: #444; 29 | padding: 0; 30 | margin: 0; 31 | min-width: 325px; 32 | position: relative; 33 | font-size: 15px; 34 | background: #fff; 35 | } 36 | @media (min-resolution: 192dpi) { 37 | body { 38 | font-weight: 300; 39 | } 40 | } 41 | 42 | a { 43 | outline: none; 44 | color: #3477db; 45 | } 46 | #popupContent { 47 | opacity: 0; 48 | transition: opacity 200ms ease; 49 | } 50 | #statusDetail a { 51 | color: #ffffff; 52 | } 53 | #header { 54 | padding: 15px; 55 | background: #777; 56 | color: #fff; 57 | } 58 | #header.willSuspend { 59 | background: #3477db; 60 | } 61 | #header.blockedFile { 62 | background: #a70707; 63 | } 64 | 65 | .group { 66 | padding: 10px 0; 67 | } 68 | .group:not(:last-of-type) { 69 | border-bottom: 1px solid #ccc; 70 | } 71 | 72 | .menuOption { 73 | line-height: 2em; 74 | font-size: 16px; 75 | white-space: nowrap; 76 | cursor: pointer; 77 | padding: 0 15px; 78 | } 79 | .menuOption:hover .optionText { 80 | text-decoration: underline; 81 | } 82 | /* dark theme for night lurkers */ 83 | .dark { 84 | background: #353535; 85 | color: #b8b8b8; 86 | } 87 | .dark .group { 88 | border-color: #555; 89 | } 90 | -------------------------------------------------------------------------------- /src/css/style.css: -------------------------------------------------------------------------------- 1 | /* 2 | 3 | - - /|_/| .-------------------. 4 | - _______________| @.@| / Styles by ) 5 | -- (______ >\_C/< ---/ Liam Johnston / 6 | - - / ______ _/____) ( @liamjohnstonnz / 7 | - - / /\ \ \ \ `-------------------' 8 | - (_/ \_) - \_) 9 | 10 | */ 11 | html, 12 | body { 13 | margin: 0; 14 | padding: 0; 15 | min-height: 100%; 16 | font-size: 10px; 17 | } 18 | html { 19 | box-sizing: border-box; 20 | height: 100vh; 21 | } 22 | *, 23 | *::before, 24 | *::after { 25 | box-sizing: inherit; 26 | } 27 | body { 28 | min-height: 100%; 29 | font-family: system-ui, -apple-system, BlinkMacSystemFont, 'Segoe UI', 30 | Helvetica, Arial, sans-serif, 'Apple Color Emoji', 'Segoe UI Emoji', 31 | 'Segoe UI Symbol'; 32 | font-size: 1.5rem; 33 | line-height: 1.5em; 34 | background: #fafafa; 35 | color: #444; 36 | } 37 | @media (min-resolution: 192dpi) { 38 | body { 39 | font-weight: 300; 40 | } 41 | } 42 | /* 'content' pages alignment (suspended tabs have their own overrides) */ 43 | /* apply to body tag */ 44 | .splash-wrap { 45 | display: flex; 46 | justify-content: center; 47 | /* 48 | vertical center usually looks a little better but some pages have content 49 | which changes height (accordion etc) so this doens't really work 50 | */ 51 | /* align-items: center; */ 52 | } 53 | strong { 54 | font-weight: 600; 55 | } 56 | small { 57 | font-size: smaller; 58 | } 59 | h1 { 60 | font-size: 26px; 61 | margin-bottom: 20px; 62 | } 63 | p { 64 | margin: 0 0 15px; 65 | } 66 | a { 67 | color: #3477db; 68 | text-decoration: none; 69 | cursor: pointer; 70 | } 71 | a:hover { 72 | text-decoration: underline; 73 | } 74 | a.active { 75 | cursor: default; 76 | color: #444; 77 | pointer-events: none; 78 | } 79 | hr { 80 | border-width: 0; 81 | border-top: 1px solid #e7e7e7; 82 | height: 1px; 83 | margin: 23px 0; 84 | } 85 | .fl { 86 | float: left; 87 | margin-right: 10px; 88 | } 89 | .fr { 90 | float: right; 91 | margin-left: 10px; 92 | } 93 | .topLabel { 94 | display: inline-block; 95 | margin: 0 0 8px; 96 | } 97 | select { 98 | min-width: 150px; 99 | color: #444; 100 | background: #fff; 101 | border-radius: 3px; 102 | -webkit-appearance: none; 103 | appearance: none; 104 | margin: 0; 105 | padding: 10px 30px 10px 15px; 106 | font-size: 14px; 107 | border-color: #ccc; 108 | position: relative; 109 | cursor: pointer; 110 | } 111 | .select-wrapper { 112 | background-color: #fff; 113 | display: inline-block; 114 | position: relative; 115 | } 116 | .select-wrapper::after { 117 | position: absolute; 118 | right: 15px; 119 | top: 6px; 120 | font-family: 'fontello'; 121 | text-decoration: none; 122 | content: '\f0dd'; 123 | color: #3477db; 124 | pointer-events: none; 125 | } 126 | textarea { 127 | color: #444; 128 | width: 100%; 129 | border-color: #c3c3c3; 130 | font-size: 14px; 131 | padding: 3px 5px; 132 | white-space: nowrap; 133 | margin-top: 9px; 134 | } 135 | .formRow { 136 | margin: 0 0 20px; 137 | } 138 | ul.unorderedList, 139 | ol.orderedList { 140 | padding-left: 20px; 141 | margin: 7px 0 10px 0; 142 | } 143 | ul.unorderedList li { 144 | margin: 0 0 3 10px; 145 | list-style: disc; 146 | } 147 | .splash { 148 | padding: 0 20px; 149 | margin: 50px auto; 150 | display: grid; 151 | grid-template-columns: 150px 1fr; 152 | grid-gap: 30px; 153 | } 154 | .splash:not(.welcome-message) { 155 | width: 900px; 156 | } 157 | .suspendy-guy { 158 | width: 100%; 159 | } 160 | #suspendy-guy-inprogress { 161 | max-height: 220px; 162 | max-width: 137px; 163 | } 164 | #suspendy-guy-complete { 165 | max-height: 220px; 166 | } 167 | .btn { 168 | background: #3477db; 169 | color: #fff; 170 | border-radius: 3px; 171 | height: 40px; 172 | line-height: 40px; 173 | padding: 0 20px; 174 | display: inline-block; 175 | border: 0; 176 | font-size: 14px; 177 | font-weight: 500; 178 | cursor: pointer; 179 | 180 | min-width: 80px; 181 | text-align: center; 182 | } 183 | .btn:hover { 184 | background: #5c9dfe; 185 | text-decoration: none; 186 | } 187 | .btn.btnNeg { 188 | background: #ddd; 189 | color: #333; 190 | } 191 | .btn.btnNeg:hover { 192 | background: #ccc; 193 | } 194 | 195 | .btnDisabled { 196 | background: #f1f1f1; 197 | color: #888; 198 | cursor: default; 199 | } 200 | .btnDisabled:hover { 201 | background: #f1f1f1; 202 | color: #888; 203 | } 204 | .lesserText { 205 | color: #999; 206 | } 207 | /* oh dear. well, too late now... */ 208 | .hidden { 209 | visibility: hidden; 210 | } 211 | .reallyHidden { 212 | display: none; 213 | } 214 | .mainContent { 215 | width: 750px; 216 | margin: 50px auto; 217 | display: grid; 218 | /* defining first col width because 'auto' could change due to bold text on active item */ 219 | grid-template-columns: 200px auto; 220 | grid-column-gap: 40px; 221 | grid-template-areas: 222 | 'h h' 223 | 'n m'; 224 | } 225 | .pageHeader { 226 | grid-area: h; 227 | margin: 0 0 50px; 228 | font-size: 20px; 229 | } 230 | .contentNav { 231 | grid-area: n; 232 | border-right: 1px solid #ddd; 233 | padding-right: 15px; 234 | align-self: start; 235 | } 236 | .contentNav ul { 237 | padding: 0; 238 | margin: 0; 239 | list-style-type: none; 240 | } 241 | .contentNav a { 242 | font-size: 16px; 243 | margin-bottom: 20px; 244 | color: #888; 245 | display: block; 246 | } 247 | .contentNav .active { 248 | color: #3477db; 249 | font-weight: bold; 250 | } 251 | .content { 252 | grid-area: m; 253 | font-size: 14px; 254 | } 255 | .content h2 { 256 | font-size: 16px; 257 | font-weight: bold; 258 | margin: 0 0 30px; 259 | } 260 | .heading-note { 261 | font-style: italic; 262 | margin-top: -25px; 263 | margin-bottom: 25px; 264 | } 265 | .welcome-message { 266 | border-radius: 6px; 267 | background: #fff; 268 | padding: 30px; 269 | margin-bottom: 20px; 270 | align-items: center; 271 | grid-gap: 40px; 272 | box-shadow: 0 1px 3px rgba(0,0,0,0.12), 0 1px 2px rgba(0,0,0,0.24); 273 | } 274 | .gsNote { 275 | font-style: italic; 276 | font-size: 12px; 277 | background: rgb(255, 247, 202); 278 | padding: 5px; 279 | margin-top: 5px; 280 | } 281 | .sessionLink { 282 | cursor: pointer; 283 | } 284 | .sessionLink:hover { 285 | text-decoration: underline; 286 | } 287 | .tabContainer { 288 | padding: 0px; 289 | margin: 0 0 5px 0; 290 | } 291 | .tabContainer > * { 292 | vertical-align: middle; 293 | } 294 | a.historyLink { 295 | padding-left: 5px; 296 | color: #333; 297 | font-size: 13px; 298 | text-decoration: none; 299 | max-width: 90%; 300 | overflow: hidden; 301 | display: inline-block; 302 | white-space: nowrap; 303 | text-overflow: ellipsis; 304 | } 305 | a.historyLink:hover { 306 | text-decoration: underline; 307 | } 308 | .windowContainer { 309 | font-weight: bold; 310 | margin: 20px 0 10px; 311 | } 312 | 313 | .groupLink { 314 | margin: 0 0 0 10px; 315 | font-size: 11px; 316 | font-weight: normal; 317 | } 318 | .sessionContainer { 319 | white-space: nowrap; 320 | margin: 0 0 5px; 321 | } 322 | .sessionContainer .groupLink { 323 | visibility: hidden; 324 | } 325 | .sessionContainer:hover .groupLink { 326 | visibility: visible; 327 | } 328 | .sessionContents { 329 | margin-left: 28px; 330 | } 331 | /* conatiner for a group of sessions */ 332 | .sessionsContainer { 333 | margin-bottom: 50px; 334 | } 335 | .sessionIcon { 336 | cursor: pointer; 337 | margin: 0 10px 0 -3px; 338 | } 339 | .sessionContents div:last-child { 340 | padding-bottom: 10px; 341 | } 342 | .tabContainer .itemHover { 343 | visibility: hidden; 344 | margin: 0; 345 | padding: 0 0 3px; 346 | cursor: pointer; 347 | } 348 | .tabContainer:hover .itemHover { 349 | visibility: visible; 350 | } 351 | .tabContainer .removeLink { 352 | margin-left: -20px; 353 | color: #888; 354 | } 355 | #screenCaptureNotice { 356 | display: none; 357 | clear: left; 358 | margin-top: 30px; 359 | background: rgb(255, 247, 202); 360 | padding: 10px 0 10px 10px; 361 | } 362 | .keyboardShortcuts { 363 | display: grid; 364 | grid-template-columns: auto auto; 365 | } 366 | .keyboardShortcuts div { 367 | margin: 0 0 2px; 368 | } 369 | .bottomMargin { 370 | margin-bottom: 20px !important; 371 | } 372 | .hotkeyCommand { 373 | word-spacing: -1px; 374 | } 375 | 376 | /* custom checkboxes */ 377 | input[type='checkbox'] { 378 | position: absolute; 379 | opacity: 0; 380 | } 381 | input[type='checkbox'] + label { 382 | position: relative; 383 | padding-left: 30px; 384 | cursor: pointer; 385 | } 386 | input[type='checkbox'] + label:before { 387 | position: absolute; 388 | left: 0; 389 | top: -2px; 390 | font-family: 'fontello'; 391 | text-decoration: none; 392 | content: '\f096'; 393 | color: #888; 394 | font-size: 24px; 395 | cursor: pointer; 396 | } 397 | input[type='checkbox']:checked + label:before { 398 | content: '\f14a'; 399 | color: #3477db; 400 | } 401 | 402 | /* custom radio btns */ 403 | .radio { 404 | position: relative; 405 | display: block; 406 | margin-top: 15px; 407 | margin-bottom: 15px; 408 | } 409 | .radio input[type='radio'] { 410 | opacity: 0; 411 | z-index: 1; 412 | } 413 | .radio input[type='radio']:checked + label::before { 414 | border-color: #3477db; 415 | } 416 | .radio input[type='radio']:checked + label::after { 417 | transform: scale(1, 1); 418 | } 419 | .radio input[type='radio']:checked + label::after { 420 | background-color: #3477db; 421 | } 422 | .radio label { 423 | display: inline-block; 424 | vertical-align: middle; 425 | position: relative; 426 | padding-left: 5px; 427 | cursor: pointer; 428 | } 429 | .radio label::before { 430 | cursor: pointer; 431 | content: ''; 432 | display: inline-block; 433 | position: absolute; 434 | width: 17px; 435 | height: 17px; 436 | left: 0; 437 | top: 4px; 438 | margin-left: -20px; 439 | border: 1px solid #ccc; 440 | border-radius: 50%; 441 | background-color: #fff; 442 | } 443 | .radio label::after { 444 | display: inline-block; 445 | position: absolute; 446 | content: ' '; 447 | width: 13px; 448 | height: 13px; 449 | left: 2px; 450 | top: 6px; 451 | margin-left: -20px; 452 | border-radius: 50%; 453 | background-color: #555; 454 | transform: scale(0, 0); 455 | } 456 | .sub-section { 457 | margin-bottom: 35px; 458 | } 459 | 460 | .tooltipIcon { 461 | color: #444; 462 | } 463 | /* Tooltips */ 464 | [data-i18n-tooltip] { 465 | position: relative; 466 | } 467 | [data-i18n-tooltip]::after { 468 | content: attr(data-i18n-tooltip); 469 | position: absolute; 470 | left: 50%; 471 | top: -6px; 472 | transform: translateX(-50%) translateY(-100%); 473 | background: #fff; 474 | line-height: 20px; 475 | color: #444; 476 | padding: 4px 2px; 477 | min-width: 500px; 478 | border-radius: 4px; 479 | border: 1px solid #e8e8e8; 480 | pointer-events: none; 481 | padding: 24px 24px 32px; 482 | z-index: 99; 483 | opacity: 0; 484 | white-space: pre; 485 | } 486 | [data-i18n-tooltip]:hover::after { 487 | opacity: 1; 488 | } 489 | @keyframes spinner { 490 | to {transform: rotate(360deg);} 491 | } 492 | .faviconSpinner { 493 | position: relative; 494 | display: inline-block; 495 | min-width: 16px; 496 | min-height: 16px; 497 | margin-top: 4px; 498 | } 499 | .faviconSpinner:before { 500 | content: ''; 501 | box-sizing: border-box; 502 | position: absolute; 503 | top: 50%; 504 | left: 50%; 505 | width: 16px; 506 | height: 16px; 507 | margin-top: -10px; 508 | margin-left: -10px; 509 | border-radius: 50%; 510 | border: 2px solid #ccc; 511 | border-top-color: #333; 512 | animation: spinner .6s linear infinite; 513 | } -------------------------------------------------------------------------------- /src/css/suspended.css: -------------------------------------------------------------------------------- 1 | /* 2 | 3 | - - /|_/| .-------------------. 4 | - _______________| @.@| / Styles by ) 5 | -- (______ >\_C/< ---/ Liam Johnston / 6 | - - / ______ _/____) ( @liamjohnstonnz / 7 | - - / /\ \ \ \ `-------------------' 8 | - (_/ \_) - \_) 9 | 10 | */ 11 | html, 12 | body { 13 | margin: 0; 14 | padding: 0; 15 | min-height: 100%; 16 | font-size: 10px; 17 | } 18 | html { 19 | box-sizing: border-box; 20 | height: 100vh; 21 | } 22 | *, 23 | *::before, 24 | *::after { 25 | box-sizing: inherit; 26 | } 27 | body { 28 | min-height: 100%; 29 | font-family: system-ui, -apple-system, BlinkMacSystemFont, 'Segoe UI', 30 | Helvetica, Arial, sans-serif, 'Apple Color Emoji', 'Segoe UI Emoji', 31 | 'Segoe UI Symbol'; 32 | font-size: 1.5rem; 33 | line-height: 1.5em; 34 | background: #fafafa; 35 | color: #444; 36 | } 37 | @media (min-resolution: 192dpi) { 38 | body { 39 | font-weight: 300; 40 | } 41 | } 42 | h1 { 43 | font-size: 26px; 44 | margin-bottom: 20px; 45 | } 46 | p { 47 | margin: 0 0 15px; 48 | } 49 | a { 50 | color: #3477db; 51 | text-decoration: none; 52 | cursor: pointer; 53 | } 54 | a:hover { 55 | text-decoration: underline; 56 | } 57 | a.active { 58 | cursor: default; 59 | color: #444; 60 | pointer-events: none; 61 | } 62 | body.hide-initially { 63 | display: none !important; 64 | } 65 | body.suspended-page:not(.img-preview-mode) { 66 | display: flex; 67 | align-items: center; 68 | justify-content: center; 69 | flex-direction: column; 70 | position: absolute; 71 | height: 100%; 72 | width: 100%; 73 | overflow-x: hidden; 74 | overflow-y: hidden; 75 | } 76 | .watermark { 77 | display: none; /* show only if not image preview mode */ 78 | position: absolute; 79 | bottom: 30px; 80 | right: 30px; 81 | font-weight: bold; 82 | color: #b6b6b6; 83 | cursor: pointer; 84 | } 85 | .suspended-page:not(.img-preview-mode) .watermark { 86 | display: block; 87 | } 88 | .gsTopBar { 89 | position: fixed; 90 | background: #fafafa; 91 | top: 0; 92 | padding: 30px 40px 20px; 93 | width: 100%; 94 | text-align: center; 95 | z-index: 101; 96 | } 97 | .faviconWrap { 98 | display: inline-block; 99 | vertical-align: bottom; 100 | margin-bottom: -1px; 101 | margin-right: 10px; 102 | } 103 | .dark .faviconWrapLowContrast { 104 | filter: invert(1) grayscale(1); 105 | } 106 | .gsTopBarImg { 107 | height: 16px; 108 | width: 16px; 109 | } 110 | .gsTopBarTitle { 111 | color: #444; 112 | font-size: 20px; 113 | cursor: default; 114 | } 115 | .gsTopBarUrl { 116 | color: #444; 117 | cursor: pointer; 118 | padding: 0 20px; 119 | } 120 | .gsTopBarUrl, 121 | .gsTopBarTitleWrap { 122 | max-width: 100%; 123 | overflow: hidden; 124 | white-space: nowrap; 125 | text-overflow: ellipsis; 126 | display: inline-block; 127 | margin-bottom: 8px; 128 | } 129 | .hideOverflow { 130 | overflow: hidden; 131 | } 132 | .gsPreviewContainer { 133 | width: 90%; 134 | margin: 0 auto; 135 | padding: 30px 0; 136 | } 137 | .gsPreviewImg { 138 | display: block; 139 | max-width: 100%; 140 | margin: 0 auto; 141 | border-radius: 20px; 142 | overflow: hidden; 143 | box-shadow: 0 3px 6px rgba(0, 0, 0, 0.16), 0 3px 6px rgba(0, 0, 0, 0.23); 144 | transition: all 0.2s ease; 145 | } 146 | .gsPreviewImg:hover { 147 | box-shadow: 0 10px 20px rgba(0, 0, 0, 0.19), 0 6px 6px rgba(0, 0, 0, 0.23); 148 | } 149 | .dark .gsPreviewImg { 150 | filter: brightness(70%); 151 | opacity: 0.7; 152 | } 153 | .dark .gsPreviewImg:hover { 154 | opacity: 1; 155 | } 156 | body.img-preview-mode .gsTopBar { 157 | position: relative; 158 | } 159 | .suspended-page { 160 | cursor: pointer; 161 | } 162 | .suspendedMsg { 163 | width: 100vw; 164 | height: 100vh; 165 | line-height: 30px; 166 | display: flex; 167 | align-items: center; 168 | justify-content: center; 169 | flex-direction: column; 170 | text-align: center; 171 | } 172 | .hotkeyCommand { 173 | word-spacing: -1px; 174 | } 175 | .reasonMsg { 176 | font-size: 15px; 177 | margin-bottom: 25px; 178 | } 179 | .suspendedMsg img { 180 | height: 180px; 181 | margin-bottom: 30px; 182 | } 183 | .suspendedMsg-instr { 184 | font-size: 20px; 185 | } 186 | .suspendedMsg-shortcut { 187 | font-size: 15px; 188 | } 189 | 190 | .spinner:before { 191 | content: ''; 192 | box-sizing: border-box; 193 | position: fixed; 194 | top: 50%; 195 | left: 50%; 196 | width: 80px; 197 | height: 80px; 198 | margin-top: -40px; 199 | margin-left: -40px; 200 | border-radius: 50%; 201 | border: 8px solid transparent; 202 | border-top-color: #3477db; 203 | animation: spinner 0.6s linear infinite; 204 | z-index: 100; 205 | } 206 | .snoozyWrapper { 207 | position: relative; 208 | } 209 | #snoozySpinner.spinner:before { 210 | border: 2px solid transparent; 211 | border-right-color: #4a4a4a; 212 | border-top-color: #4a4a4a; 213 | animation: spinner 0.6s linear infinite; 214 | margin: 0; 215 | position: absolute; 216 | top: 49px; 217 | right: 7px; 218 | left: auto; 219 | width: 12px; 220 | height: 12px; 221 | } 222 | .suspendedTextWrap { 223 | height: 60px; /* stops slight jump when suspending */ 224 | } 225 | .waking .suspendedTextWrap { 226 | opacity: 0; 227 | } 228 | @keyframes spinner { 229 | to { 230 | transform: rotate(360deg); 231 | } 232 | } 233 | 234 | /* dark theme for night lurkers */ 235 | .dark, 236 | .dark .gsTopBar, 237 | .dark .suspendedMsg { 238 | background: #222; 239 | } 240 | .dark .suspendedMsg img { 241 | filter: brightness(150%); 242 | } 243 | .dark .gsTopBar, 244 | .dark .gsTopBarTitle, 245 | .dark .gsTopBar a, 246 | .dark .suspendedMsg, 247 | .dark .watermark { 248 | color: #b8b8b8; 249 | } 250 | .dark #setKeyboardShortcut { 251 | text-decoration: underline; 252 | color: #b8b8b8; 253 | } 254 | 255 | /* end suspended tab styles */ 256 | 257 | .toast-wrapper { 258 | display: none; 259 | text-align: center; 260 | position: fixed; 261 | z-index: 9999999; 262 | left: 0; 263 | top: 0; 264 | width: 100%; 265 | height: 100%; 266 | opacity: 0; 267 | animation: fadeinout 4s linear forwards; 268 | } 269 | .toast-content { 270 | display: inline-block; 271 | position: relative; 272 | top: 50%; 273 | transform: translateY(-50%); 274 | background-color: #fefefe; 275 | margin: auto; 276 | padding: 10px 20px 20px 20px; 277 | border: 1px solid #888; 278 | border-radius: 5px; 279 | box-shadow: 0 4px 8px 0 rgba(0, 0, 0, 0.2), 0 6px 20px 0 rgba(0, 0, 0, 0.19); 280 | } 281 | 282 | .toast-content p { 283 | font-size: 16px; 284 | } 285 | 286 | @keyframes fadeinout { 287 | 0%, 288 | 100% { 289 | opacity: 0; 290 | visibility: hidden; 291 | } 292 | 5%, 293 | 90% { 294 | opacity: 1; 295 | visibility: visible; 296 | } 297 | } 298 | 299 | @keyframes fadein { 300 | from { 301 | opacity: 0; 302 | } 303 | to { 304 | opacity: 1; 305 | } 306 | } 307 | -------------------------------------------------------------------------------- /src/debug.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | The Great Suspender - Debugger 9 | 10 | 11 | 12 | 13 |
14 | debugErrors: 15 |  |  16 | debugInfo: 17 |  |  18 | discardInPlaceOfSuspend: 19 |  |  20 | useAlternateScreenCaptureLib: 21 |  |  22 | claim all suspended tabs 23 |
24 |
25 |
26 | To view debug messages, go to the 27 |
28 | Ensure "developer mode" is checked at the top of the page, then click on the "Inspect views: background page" link for this extension. 29 |
30 | 31 |
32 | 33 | 34 | 35 | 36 | 37 | 38 | 39 | 40 | 41 | 42 | 43 | 44 | 45 |
WinIdTabIdIndexTitleTime to suspendStatus
46 | 47 | 48 | 49 | -------------------------------------------------------------------------------- /src/font/fontello.woff: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/dvalter/ff-thegreatsuspender/81bfe8395d024814af55db4af64a1a701a6f75eb/src/font/fontello.woff -------------------------------------------------------------------------------- /src/font/fontello.woff2: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/dvalter/ff-thegreatsuspender/81bfe8395d024814af55db4af64a1a701a6f75eb/src/font/fontello.woff2 -------------------------------------------------------------------------------- /src/history.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 | 15 | 16 |
17 | 18 | 21 | 22 |
23 | 37 |
38 | 39 |
40 |

41 |
42 | 43 |

44 |
45 | 46 |

47 |
48 | 49 | 50 | 51 |
52 | 53 |
54 | 55 | 56 | 57 | -------------------------------------------------------------------------------- /src/img/chromeDefaultFavicon.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/dvalter/ff-thegreatsuspender/81bfe8395d024814af55db4af64a1a701a6f75eb/src/img/chromeDefaultFavicon.png -------------------------------------------------------------------------------- /src/img/chromeDefaultFaviconSml.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/dvalter/ff-thegreatsuspender/81bfe8395d024814af55db4af64a1a701a6f75eb/src/img/chromeDefaultFaviconSml.png -------------------------------------------------------------------------------- /src/img/chromeDevDefaultFavicon.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/dvalter/ff-thegreatsuspender/81bfe8395d024814af55db4af64a1a701a6f75eb/src/img/chromeDevDefaultFavicon.png -------------------------------------------------------------------------------- /src/img/chromeDevDefaultFaviconSml.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/dvalter/ff-thegreatsuspender/81bfe8395d024814af55db4af64a1a701a6f75eb/src/img/chromeDevDefaultFaviconSml.png -------------------------------------------------------------------------------- /src/img/ic_suspendy_128x128.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/dvalter/ff-thegreatsuspender/81bfe8395d024814af55db4af64a1a701a6f75eb/src/img/ic_suspendy_128x128.png -------------------------------------------------------------------------------- /src/img/ic_suspendy_16x16.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/dvalter/ff-thegreatsuspender/81bfe8395d024814af55db4af64a1a701a6f75eb/src/img/ic_suspendy_16x16.png -------------------------------------------------------------------------------- /src/img/ic_suspendy_16x16_grey.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/dvalter/ff-thegreatsuspender/81bfe8395d024814af55db4af64a1a701a6f75eb/src/img/ic_suspendy_16x16_grey.png -------------------------------------------------------------------------------- /src/img/ic_suspendy_32x32.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/dvalter/ff-thegreatsuspender/81bfe8395d024814af55db4af64a1a701a6f75eb/src/img/ic_suspendy_32x32.png -------------------------------------------------------------------------------- /src/img/ic_suspendy_32x32_grey.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/dvalter/ff-thegreatsuspender/81bfe8395d024814af55db4af64a1a701a6f75eb/src/img/ic_suspendy_32x32_grey.png -------------------------------------------------------------------------------- /src/img/ic_suspendy_48x48.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/dvalter/ff-thegreatsuspender/81bfe8395d024814af55db4af64a1a701a6f75eb/src/img/ic_suspendy_48x48.png -------------------------------------------------------------------------------- /src/img/snoozy_tab.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 | 15 | -------------------------------------------------------------------------------- /src/img/snoozy_tab_awake.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | -------------------------------------------------------------------------------- /src/img/suspendy-guy-alt.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/dvalter/ff-thegreatsuspender/81bfe8395d024814af55db4af64a1a701a6f75eb/src/img/suspendy-guy-alt.png -------------------------------------------------------------------------------- /src/img/suspendy-guy-oops.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/dvalter/ff-thegreatsuspender/81bfe8395d024814af55db4af64a1a701a6f75eb/src/img/suspendy-guy-oops.png -------------------------------------------------------------------------------- /src/img/suspendy-guy-uh-oh.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/dvalter/ff-thegreatsuspender/81bfe8395d024814af55db4af64a1a701a6f75eb/src/img/suspendy-guy-uh-oh.png -------------------------------------------------------------------------------- /src/img/suspendy-guy.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/dvalter/ff-thegreatsuspender/81bfe8395d024814af55db4af64a1a701a6f75eb/src/img/suspendy-guy.png -------------------------------------------------------------------------------- /src/js/about.js: -------------------------------------------------------------------------------- 1 | /* global chrome, XMLHttpRequest, gsStorage, gsUtils */ 2 | (function(global) { 3 | 'use strict'; 4 | 5 | try { 6 | chrome.extension.getBackgroundPage().tgs.setViewGlobals(global); 7 | } catch (e) { 8 | window.setTimeout(() => window.location.reload(), 1000); 9 | return; 10 | } 11 | 12 | gsUtils.documentReadyAndLocalisedAsPromsied(document).then(function() { 13 | var versionEl = document.getElementById('aboutVersion'); 14 | versionEl.innerHTML = 'v' + chrome.runtime.getManifest().version; 15 | 16 | //hide incompatible sidebar items if in incognito mode 17 | if (chrome.extension.inIncognitoContext) { 18 | Array.prototype.forEach.call( 19 | document.getElementsByClassName('noIncognito'), 20 | function(el) { 21 | el.style.display = 'none'; 22 | } 23 | ); 24 | } 25 | }); 26 | })(this); 27 | -------------------------------------------------------------------------------- /src/js/broken.js: -------------------------------------------------------------------------------- 1 | /*global chrome */ 2 | (function(global) { 3 | 'use strict'; 4 | 5 | try { 6 | chrome.extension.getBackgroundPage().tgs.setViewGlobals(global); 7 | } catch (e) { 8 | window.setTimeout(() => window.location.reload(), 1000); 9 | return; 10 | } 11 | 12 | function init() { 13 | document 14 | .getElementById('restartExtension') 15 | .addEventListener('click', function() { 16 | chrome.runtime.reload(); 17 | }); 18 | document 19 | .getElementById('sessionManagementLink') 20 | .addEventListener('click', function() { 21 | chrome.tabs.create({ url: chrome.extension.getURL('history.html') }); 22 | }); 23 | } 24 | if (document.readyState !== 'loading') { 25 | init(); 26 | } else { 27 | document.addEventListener('DOMContentLoaded', function() { 28 | init(); 29 | }); 30 | } 31 | })(this); 32 | -------------------------------------------------------------------------------- /src/js/contentscript.js: -------------------------------------------------------------------------------- 1 | /*global chrome */ 2 | /* 3 | * The Great Suspender 4 | * Copyright (C) 2017 Dean Oemcke 5 | * Available under GNU GENERAL PUBLIC LICENSE v2 6 | * http://github.com/deanoemcke/thegreatsuspender 7 | * ლ(ಠ益ಠლ) 8 | */ 9 | (function() { 10 | 'use strict'; 11 | 12 | let isFormListenerInitialised = false; 13 | let isReceivingFormInput = false; 14 | let isIgnoreForms = false; 15 | let tempWhitelist = false; 16 | 17 | function formInputListener(e) { 18 | if (!isReceivingFormInput && !tempWhitelist) { 19 | if (event.keyCode >= 48 && event.keyCode <= 90 && event.target.tagName) { 20 | if ( 21 | event.target.tagName.toUpperCase() === 'INPUT' || 22 | event.target.tagName.toUpperCase() === 'TEXTAREA' || 23 | event.target.tagName.toUpperCase() === 'FORM' || 24 | event.target.isContentEditable === true || 25 | event.target.type === "application/pdf" 26 | ) { 27 | isReceivingFormInput = true; 28 | if (!isBackgroundConnectable()) { 29 | return false; 30 | } 31 | chrome.runtime.sendMessage(buildReportTabStatePayload()); 32 | } 33 | } 34 | } 35 | } 36 | 37 | function initFormInputListener() { 38 | if (isFormListenerInitialised) { 39 | return; 40 | } 41 | window.addEventListener('keydown', formInputListener); 42 | isFormListenerInitialised = true; 43 | } 44 | 45 | function init() { 46 | //listen for background events 47 | chrome.runtime.onMessage.addListener(function( 48 | request, 49 | sender, 50 | sendResponse 51 | ) { 52 | if (request.hasOwnProperty('action')) { 53 | if (request.action === 'requestInfo') { 54 | sendResponse(buildReportTabStatePayload()); 55 | return false; 56 | } 57 | } 58 | 59 | if (request.hasOwnProperty('scrollPos')) { 60 | if (request.scrollPos !== '' && request.scrollPos !== '0') { 61 | document.body.scrollTop = request.scrollPos; 62 | document.documentElement.scrollTop = request.scrollPos; 63 | } 64 | } 65 | if (request.hasOwnProperty('ignoreForms')) { 66 | isIgnoreForms = request.ignoreForms; 67 | if (isIgnoreForms) { 68 | initFormInputListener(); 69 | } 70 | isReceivingFormInput = isReceivingFormInput && isIgnoreForms; 71 | } 72 | if (request.hasOwnProperty('tempWhitelist')) { 73 | if (isReceivingFormInput && !request.tempWhitelist) { 74 | isReceivingFormInput = false; 75 | } 76 | tempWhitelist = request.tempWhitelist; 77 | } 78 | sendResponse(buildReportTabStatePayload()); 79 | return false; 80 | }); 81 | } 82 | 83 | function waitForRuntimeReady(retries) { 84 | retries = retries || 0; 85 | return new Promise(r => r(chrome.runtime)).then(chromeRuntime => { 86 | if (chromeRuntime) { 87 | return Promise.resolve(); 88 | } 89 | if (retries > 3) { 90 | return Promise.reject('Failed waiting for chrome.runtime'); 91 | } 92 | retries += 1; 93 | return new Promise(r => window.setTimeout(r, 500)).then(() => 94 | waitForRuntimeReady(retries) 95 | ); 96 | }); 97 | } 98 | 99 | function isBackgroundConnectable() { 100 | try { 101 | var port = chrome.runtime.connect(); 102 | if (port) { 103 | port.disconnect(); 104 | return true; 105 | } 106 | return false; 107 | } catch (e) { 108 | return false; 109 | } 110 | } 111 | 112 | function buildReportTabStatePayload() { 113 | return { 114 | action: 'reportTabState', 115 | status: 116 | isIgnoreForms && isReceivingFormInput 117 | ? 'formInput' 118 | : tempWhitelist 119 | ? 'tempWhitelist' 120 | : 'normal', 121 | scrollPos: 122 | document.body.scrollTop || document.documentElement.scrollTop || 0, 123 | }; 124 | } 125 | 126 | waitForRuntimeReady() 127 | .then(init) 128 | .catch(e => { 129 | console.error(e); 130 | setTimeout(() => { 131 | init(); 132 | }, 200); 133 | }); 134 | })(); 135 | -------------------------------------------------------------------------------- /src/js/debug.js: -------------------------------------------------------------------------------- 1 | /*global chrome, tgs, gsUtils, gsFavicon, gsStorage, gsChrome */ 2 | (function(global) { 3 | 'use strict'; 4 | 5 | try { 6 | chrome.extension.getBackgroundPage().tgs.setViewGlobals(global); 7 | } catch (e) { 8 | window.setTimeout(() => window.location.reload(), 1000); 9 | return; 10 | } 11 | 12 | var currentTabs = {}; 13 | 14 | function generateTabInfo(info) { 15 | // console.log(info.tabId, info); 16 | var timerStr = 17 | info && info.timerUp && info && info.timerUp !== '-' 18 | ? new Date(info.timerUp).toLocaleString() 19 | : '-'; 20 | var html = '', 21 | windowId = info && info.windowId ? info.windowId : '?', 22 | tabId = info && info.tabId ? info.tabId : '?', 23 | tabIndex = info && info.tab ? info.tab.index : '?', 24 | favicon = info && info.tab ? info.tab.favIconUrl : '', 25 | tabTitle = info && info.tab ? gsUtils.htmlEncode(info.tab.title) : '?', 26 | tabTimer = timerStr, 27 | tabStatus = info ? info.status : '?'; 28 | 29 | favicon = 30 | favicon && favicon.indexOf('data') === 0 31 | ? favicon 32 | : chrome.extension.getURL('img/chromeDefaultFavicon.png'); 33 | 34 | html += ''; 35 | html += '' + windowId + ''; 36 | html += '' + tabId + ''; 37 | html += '' + tabIndex + ''; 38 | html += ''; 39 | html += '' + tabTitle + ''; 40 | html += '' + tabTimer + ''; 41 | html += '' + tabStatus + ''; 42 | html += ''; 43 | 44 | return html; 45 | } 46 | 47 | async function fetchInfo() { 48 | const tabs = await gsChrome.tabsQuery(); 49 | const debugInfoPromises = []; 50 | for (const [i, curTab] of tabs.entries()) { 51 | currentTabs[tabs[i].id] = tabs[i]; 52 | debugInfoPromises.push( 53 | new Promise(r => 54 | tgs.getDebugInfo(curTab.id, o => { 55 | o.tab = curTab; 56 | r(o); 57 | }) 58 | ) 59 | ); 60 | } 61 | const debugInfos = await Promise.all(debugInfoPromises); 62 | for (const debugInfo of debugInfos) { 63 | var html, 64 | tableEl = document.getElementById('gsProfilerBody'); 65 | html = generateTabInfo(debugInfo); 66 | tableEl.innerHTML = tableEl.innerHTML + html; 67 | } 68 | } 69 | 70 | function addFlagHtml(elementId, getterFn, setterFn) { 71 | document.getElementById(elementId).innerHTML = getterFn(); 72 | document.getElementById(elementId).onclick = function(e) { 73 | const newVal = !getterFn(); 74 | setterFn(newVal); 75 | document.getElementById(elementId).innerHTML = newVal; 76 | }; 77 | } 78 | 79 | gsUtils.documentReadyAndLocalisedAsPromsied(document).then(async function() { 80 | await fetchInfo(); 81 | addFlagHtml( 82 | 'toggleDebugInfo', 83 | () => gsUtils.isDebugInfo(), 84 | newVal => gsUtils.setDebugInfo(newVal) 85 | ); 86 | addFlagHtml( 87 | 'toggleDebugError', 88 | () => gsUtils.isDebugError(), 89 | newVal => gsUtils.setDebugError(newVal) 90 | ); 91 | addFlagHtml( 92 | 'toggleDiscardInPlaceOfSuspend', 93 | () => gsStorage.getOption(gsStorage.DISCARD_IN_PLACE_OF_SUSPEND), 94 | newVal => { 95 | gsStorage.setOptionAndSync( 96 | gsStorage.DISCARD_IN_PLACE_OF_SUSPEND, 97 | newVal 98 | ); 99 | } 100 | ); 101 | addFlagHtml( 102 | 'toggleUseAlternateScreenCaptureLib', 103 | () => gsStorage.getOption(gsStorage.USE_ALT_SCREEN_CAPTURE_LIB), 104 | newVal => { 105 | gsStorage.setOptionAndSync( 106 | gsStorage.USE_ALT_SCREEN_CAPTURE_LIB, 107 | newVal 108 | ); 109 | } 110 | ); 111 | document.getElementById('claimSuspendedTabs').onclick = async function(e) { 112 | const tabs = await gsChrome.tabsQuery(); 113 | for (const tab of tabs) { 114 | if ( 115 | gsUtils.isSuspendedTab(tab, true) && 116 | tab.url.indexOf(chrome.runtime.id) < 0 117 | ) { 118 | const newUrl = tab.url.replace( 119 | gsUtils.getRootUrl(tab.url), 120 | chrome.runtime.id 121 | ); 122 | await gsChrome.tabsUpdate(tab.id, { url: newUrl }); 123 | } 124 | } 125 | }; 126 | 127 | var extensionsUrl = `about:devtools-toolbox?type=extension&id=${chrome.runtime.id}`; 128 | document 129 | .getElementById('backgroundPage') 130 | .setAttribute('href', extensionsUrl); 131 | document.getElementById('backgroundPage').onclick = function() { 132 | chrome.tabs.create({ url: extensionsUrl }); 133 | }; 134 | 135 | /* 136 | chrome.processes.onUpdatedWithMemory.addListener(function (processes) { 137 | chrome.tabs.query({}, function (tabs) { 138 | var html = ''; 139 | html += generateMemStats(processes); 140 | html += '
'; 141 | html += generateTabStats(tabs); 142 | document.getElementById('gsProfiler').innerHTML = html; 143 | }); 144 | }); 145 | */ 146 | }); 147 | 148 | window.onload = function() { 149 | document.getElementById('inspect-url').innerHTML = `about:devtools-toolbox?type=extension&id=${chrome.runtime.id}`; 150 | } 151 | })(this); 152 | -------------------------------------------------------------------------------- /src/js/gsChrome.js: -------------------------------------------------------------------------------- 1 | /*global chrome, gsUtils */ 2 | 'use strict'; 3 | // eslint-disable-next-line no-unused-vars 4 | var gsChrome = { 5 | cookiesGetAll: function() { 6 | return new Promise(resolve => { 7 | chrome.cookies.getAll({}, cookies => { 8 | if (chrome.runtime.lastError) { 9 | gsUtils.warning('chromeCookies', chrome.runtime.lastError); 10 | cookies = []; 11 | } 12 | resolve(cookies); 13 | }); 14 | }); 15 | }, 16 | cookiesRemove: function(url, name) { 17 | return new Promise(resolve => { 18 | if (!url || !name) { 19 | gsUtils.warning('chromeCookies', 'url or name not specified'); 20 | resolve(null); 21 | return; 22 | } 23 | chrome.cookies.remove({ url, name }, details => { 24 | if (chrome.runtime.lastError) { 25 | gsUtils.warning('chromeCookies', chrome.runtime.lastError); 26 | details = null; 27 | } 28 | resolve(details); 29 | }); 30 | }); 31 | }, 32 | 33 | tabsCreate: function(details) { 34 | return new Promise(resolve => { 35 | if ( 36 | !details || 37 | (typeof details !== 'string' && typeof details.url !== 'string') 38 | ) { 39 | gsUtils.warning('chromeTabs', 'url not specified'); 40 | resolve(null); 41 | return; 42 | } 43 | details = typeof details === 'string' ? { url: details } : details; 44 | chrome.tabs.create(details, tab => { 45 | if (chrome.runtime.lastError) { 46 | gsUtils.warning('chromeTabs', chrome.runtime.lastError); 47 | tab = null; 48 | } 49 | resolve(tab); 50 | }); 51 | }); 52 | }, 53 | tabsReload: function(tabId) { 54 | return new Promise(resolve => { 55 | if (!tabId) { 56 | gsUtils.warning('chromeTabs', 'tabId not specified'); 57 | resolve(false); 58 | return; 59 | } 60 | chrome.tabs.reload(tabId, () => { 61 | if (chrome.runtime.lastError) { 62 | gsUtils.warning('chromeTabs', chrome.runtime.lastError); 63 | resolve(false); 64 | return; 65 | } 66 | resolve(true); 67 | }); 68 | }); 69 | }, 70 | tabsUpdate: function(tabId, updateProperties) { 71 | return new Promise(resolve => { 72 | if (!tabId || !updateProperties) { 73 | gsUtils.warning( 74 | 'chromeTabs', 75 | 'tabId or updateProperties not specified' 76 | ); 77 | resolve(null); 78 | return; 79 | } 80 | chrome.tabs.update(tabId, updateProperties, tab => { 81 | if (chrome.runtime.lastError) { 82 | gsUtils.warning('chromeTabs', chrome.runtime.lastError); 83 | tab = null; 84 | } 85 | resolve(tab); 86 | }); 87 | }); 88 | }, 89 | tabsGet: function(tabId) { 90 | return new Promise(resolve => { 91 | if (!tabId) { 92 | gsUtils.warning('chromeTabs', 'tabId not specified'); 93 | resolve(null); 94 | return; 95 | } 96 | chrome.tabs.get(tabId, tab => { 97 | if (chrome.runtime.lastError) { 98 | gsUtils.warning('chromeTabs', chrome.runtime.lastError); 99 | tab = null; 100 | } 101 | resolve(tab); 102 | }); 103 | }); 104 | }, 105 | tabsQuery: function(queryInfo) { 106 | queryInfo = queryInfo || {}; 107 | return new Promise(resolve => { 108 | chrome.tabs.query(queryInfo, tabs => { 109 | if (chrome.runtime.lastError) { 110 | gsUtils.warning('chromeTabs', chrome.runtime.lastError); 111 | tabs = []; 112 | } 113 | resolve(tabs); 114 | }); 115 | }); 116 | }, 117 | tabsRemove: function(tabId) { 118 | return new Promise(resolve => { 119 | if (!tabId) { 120 | gsUtils.warning('chromeTabs', 'tabId not specified'); 121 | resolve(null); 122 | return; 123 | } 124 | chrome.tabs.remove(tabId, () => { 125 | if (chrome.runtime.lastError) { 126 | gsUtils.warning('chromeTabs', chrome.runtime.lastError); 127 | } 128 | resolve(); 129 | }); 130 | }); 131 | }, 132 | 133 | windowsGetLastFocused: function() { 134 | return new Promise(resolve => { 135 | chrome.windows.getLastFocused({}, window => { 136 | if (chrome.runtime.lastError) { 137 | gsUtils.warning('chromeWindows', chrome.runtime.lastError); 138 | window = null; 139 | } 140 | resolve(window); 141 | }); 142 | }); 143 | }, 144 | windowsGet: function(windowId) { 145 | return new Promise(resolve => { 146 | if (!windowId) { 147 | gsUtils.warning('chromeWindows', 'windowId not specified'); 148 | resolve(null); 149 | return; 150 | } 151 | chrome.windows.get(windowId, { populate: true }, window => { 152 | if (chrome.runtime.lastError) { 153 | gsUtils.warning('chromeWindows', chrome.runtime.lastError); 154 | window = null; 155 | } 156 | resolve(window); 157 | }); 158 | }); 159 | }, 160 | windowsGetAll: function() { 161 | return new Promise(resolve => { 162 | chrome.windows.getAll({ populate: true }, windows => { 163 | if (chrome.runtime.lastError) { 164 | gsUtils.warning('chromeWindows', chrome.runtime.lastError); 165 | windows = []; 166 | } 167 | resolve(windows); 168 | }); 169 | }); 170 | }, 171 | windowsCreate: function(createData) { 172 | createData = createData || {}; 173 | return new Promise(resolve => { 174 | chrome.windows.create(createData, window => { 175 | if (chrome.runtime.lastError) { 176 | gsUtils.warning('chromeWindows', chrome.runtime.lastError); 177 | window = null; 178 | } 179 | resolve(window); 180 | }); 181 | }); 182 | }, 183 | windowsUpdate: function(windowId, updateInfo) { 184 | return new Promise(resolve => { 185 | if (!windowId || !updateInfo) { 186 | gsUtils.warning('chromeTabs', 'windowId or updateInfo not specified'); 187 | resolve(null); 188 | return; 189 | } 190 | chrome.windows.update(windowId, updateInfo, window => { 191 | if (chrome.runtime.lastError) { 192 | gsUtils.warning('chromeWindows', chrome.runtime.lastError); 193 | window = null; 194 | } 195 | resolve(window); 196 | }); 197 | }); 198 | }, 199 | }; 200 | -------------------------------------------------------------------------------- /src/js/gsMessages.js: -------------------------------------------------------------------------------- 1 | /*global gsUtils, gsStorage */ 2 | // eslint-disable-next-line no-unused-vars 3 | var gsMessages = { 4 | INFO: 'info', 5 | WARNING: 'warning', 6 | ERROR: 'error', 7 | 8 | sendInitTabToContentScript( 9 | tabId, 10 | ignoreForms, 11 | tempWhitelist, 12 | scrollPos, 13 | callback 14 | ) { 15 | var payload = { 16 | ignoreForms: ignoreForms, 17 | tempWhitelist: tempWhitelist, 18 | }; 19 | if (scrollPos) { 20 | payload.scrollPos = scrollPos; 21 | } 22 | gsMessages.sendMessageToContentScript( 23 | tabId, 24 | payload, 25 | gsMessages.ERROR, 26 | callback 27 | ); 28 | }, 29 | 30 | sendUpdateToContentScriptOfTab: function(tab) { 31 | if ( 32 | gsUtils.isSpecialTab(tab) || 33 | gsUtils.isSuspendedTab(tab, true) || 34 | gsUtils.isDiscardedTab(tab) 35 | ) { 36 | return; 37 | } 38 | 39 | const ignoreForms = gsStorage.getOption(gsStorage.IGNORE_FORMS); 40 | gsMessages.sendMessageToContentScript( 41 | tab.id, 42 | { ignoreForms }, 43 | gsMessages.WARNING 44 | ); 45 | }, 46 | 47 | sendTemporaryWhitelistToContentScript: function(tabId, callback) { 48 | gsMessages.sendMessageToContentScript( 49 | tabId, 50 | { 51 | tempWhitelist: true, 52 | }, 53 | gsMessages.WARNING, 54 | callback 55 | ); 56 | }, 57 | 58 | sendUndoTemporaryWhitelistToContentScript: function(tabId, callback) { 59 | gsMessages.sendMessageToContentScript( 60 | tabId, 61 | { 62 | tempWhitelist: false, 63 | }, 64 | gsMessages.WARNING, 65 | callback 66 | ); 67 | }, 68 | 69 | sendRequestInfoToContentScript(tabId, callback) { 70 | gsMessages.sendMessageToContentScript( 71 | tabId, 72 | { 73 | action: 'requestInfo', 74 | }, 75 | gsMessages.WARNING, 76 | callback 77 | ); 78 | }, 79 | 80 | sendMessageToContentScript: function(tabId, message, severity, callback) { 81 | gsMessages.sendMessageToTab(tabId, message, severity, function( 82 | error, 83 | response 84 | ) { 85 | if (error) { 86 | if (callback) callback(error); 87 | } else { 88 | if (callback) callback(null, response); 89 | } 90 | }); 91 | }, 92 | 93 | sendPingToTab: function(tabId, callback) { 94 | gsMessages.sendMessageToTab( 95 | tabId, 96 | { 97 | action: 'ping', 98 | }, 99 | gsMessages.INFO, 100 | callback 101 | ); 102 | }, 103 | 104 | sendMessageToTab: function(tabId, message, severity, callback) { 105 | if (!tabId) { 106 | if (callback) callback('tabId not specified'); 107 | return; 108 | } 109 | var responseHandler = function(response) { 110 | gsUtils.log(tabId, 'response from tab', response); 111 | if (chrome.runtime.lastError) { 112 | if (callback) callback(chrome.runtime.lastError); 113 | } else { 114 | if (callback) callback(null, response); 115 | } 116 | }; 117 | 118 | message.tabId = tabId; 119 | try { 120 | gsUtils.log(tabId, 'send message to tab', message); 121 | chrome.tabs.sendMessage(tabId, message, { frameId: 0 }, responseHandler); 122 | } catch (e) { 123 | // gsUtils.error(tabId, e); 124 | chrome.tabs.sendMessage(tabId, message, responseHandler); 125 | } 126 | }, 127 | 128 | executeScriptOnTab: function(tabId, scriptPath, callback) { 129 | if (!tabId) { 130 | if (callback) callback('tabId not specified'); 131 | return; 132 | } 133 | gsUtils.log(`executing script ${scriptPath}`) 134 | 135 | var executing = browser.tabs.executeScript(tabId, { file: scriptPath }); 136 | 137 | if (executing) { 138 | executing.then(function(response) { 139 | if (callback) callback(null, response); 140 | }, function(error) { 141 | if (callback) callback(error); 142 | }) 143 | } 144 | }, 145 | 146 | executeCodeOnTab: function(tabId, codeString, callback) { 147 | if (!tabId) { 148 | if (callback) callback('tabId not specified'); 149 | return; 150 | } 151 | gsUtils.log(`executing code ${codeString}`) 152 | 153 | var executing = chrome.tabs.executeScript(tabId, { code: codeString }); 154 | 155 | if (executing) { 156 | 157 | executing.then(function(response) { 158 | if (callback) callback(null, response); 159 | }, function(error) { 160 | if (callback) callback(error); 161 | }) 162 | } 163 | }, 164 | }; 165 | -------------------------------------------------------------------------------- /src/js/gsTabDiscardManager.js: -------------------------------------------------------------------------------- 1 | /*global chrome, localStorage, tgs, gsUtils, gsChrome, GsTabQueue, gsStorage, gsTabSuspendManager */ 2 | // eslint-disable-next-line no-unused-vars 3 | var gsTabDiscardManager = (function() { 4 | 'use strict'; 5 | 6 | const DEFAULT_CONCURRENT_DISCARDS = 5; 7 | const DEFAULT_DISCARD_TIMEOUT = 5 * 1000; 8 | 9 | const QUEUE_ID = '_discardQueue'; 10 | 11 | let _discardQueue; 12 | 13 | function initAsPromised() { 14 | return new Promise(resolve => { 15 | const queueProps = { 16 | concurrentExecutors: DEFAULT_CONCURRENT_DISCARDS, 17 | jobTimeout: DEFAULT_DISCARD_TIMEOUT, 18 | executorFn: performDiscard, 19 | exceptionFn: handleDiscardException, 20 | }; 21 | _discardQueue = GsTabQueue(QUEUE_ID, queueProps); 22 | gsUtils.log(QUEUE_ID, 'init successful'); 23 | resolve(); 24 | }); 25 | } 26 | 27 | function queueTabForDiscard(tab, executionProps, processingDelay) { 28 | queueTabForDiscardAsPromise(tab, executionProps, processingDelay).catch( 29 | e => { 30 | gsUtils.log(tab.id, QUEUE_ID, e); 31 | } 32 | ); 33 | } 34 | 35 | function queueTabForDiscardAsPromise(tab, executionProps, processingDelay) { 36 | gsUtils.log(tab.id, QUEUE_ID, `Queueing tab for discarding.`); 37 | executionProps = executionProps || {}; 38 | return _discardQueue.queueTabAsPromise( 39 | tab, 40 | executionProps, 41 | processingDelay 42 | ); 43 | } 44 | 45 | function unqueueTabForDiscard(tab) { 46 | const removed = _discardQueue.unqueueTab(tab); 47 | if (removed) { 48 | gsUtils.log(tab.id, QUEUE_ID, 'Removed tab from discard queue'); 49 | } 50 | } 51 | 52 | // This is called remotely by the _discardQueue 53 | // So we must first re-fetch the tab in case it has changed 54 | async function performDiscard(tab, executionProps, resolve, reject, requeue) { 55 | let _tab = null; 56 | try { 57 | _tab = await gsChrome.tabsGet(tab.id); 58 | } catch (error) { 59 | // assume tab has been discarded 60 | } 61 | if (!_tab) { 62 | gsUtils.warning( 63 | tab.id, 64 | QUEUE_ID, 65 | `Failed to discard tab. Tab may have already been discarded or removed.` 66 | ); 67 | resolve(false); 68 | return; 69 | } 70 | tab = _tab; 71 | 72 | if (gsUtils.isSuspendedTab(tab) && tab.status === 'loading') { 73 | gsUtils.log(tab.id, QUEUE_ID, 'Tab is still loading'); 74 | requeue(); 75 | return; 76 | } 77 | if (tgs.isCurrentActiveTab(tab)) { 78 | const discardInPlaceOfSuspend = gsStorage.getOption( 79 | gsStorage.DISCARD_IN_PLACE_OF_SUSPEND 80 | ); 81 | if (!discardInPlaceOfSuspend) { 82 | gsUtils.log(tab.id, QUEUE_ID, 'Tab is active. Aborting discard.'); 83 | resolve(false); 84 | return; 85 | } 86 | } 87 | if (gsUtils.isDiscardedTab(tab)) { 88 | gsUtils.log(tab.id, QUEUE_ID, 'Tab already discarded'); 89 | resolve(false); 90 | return; 91 | } 92 | gsUtils.log(tab.id, QUEUE_ID, 'Forcing discarding of tab.'); 93 | browser.tabs.discard(tab.id).then(() => { 94 | resolve(true); 95 | }).catch((err) => { 96 | gsUtils.warning(tab.id, QUEUE_ID, err); 97 | resolve(false); 98 | }); 99 | } 100 | 101 | function handleDiscardException( 102 | tab, 103 | executionProps, 104 | exceptionType, 105 | resolve, 106 | reject, 107 | requeue 108 | ) { 109 | gsUtils.warning( 110 | tab.id, 111 | QUEUE_ID, 112 | `Failed to discard tab: ${exceptionType}` 113 | ); 114 | resolve(false); 115 | } 116 | 117 | async function handleDiscardedUnsuspendedTab(tab) { 118 | if ( 119 | gsUtils.shouldSuspendDiscardedTabs() && 120 | gsTabSuspendManager.checkTabEligibilityForSuspension(tab, 3) 121 | ) { 122 | tgs.setTabStatePropForTabId(tab.id, tgs.STATE_SUSPEND_REASON, 3); 123 | const suspendedUrl = gsUtils.generateSuspendedUrl(tab.url, tab.title, 0); 124 | gsUtils.log(tab.id, QUEUE_ID, 'Suspending discarded unsuspended tab'); 125 | 126 | // Note: This bypasses the suspension tab queue and also prevents screenshots from being taken 127 | await gsTabSuspendManager.executeTabSuspension(tab, suspendedUrl); 128 | return; 129 | } 130 | } 131 | 132 | return { 133 | initAsPromised, 134 | queueTabForDiscard, 135 | queueTabForDiscardAsPromise, 136 | unqueueTabForDiscard, 137 | handleDiscardedUnsuspendedTab, 138 | }; 139 | })(); 140 | -------------------------------------------------------------------------------- /src/js/gsTabQueue.js: -------------------------------------------------------------------------------- 1 | /*global gsUtils */ 2 | // eslint-disable-next-line no-unused-vars 3 | function GsTabQueue(queueId, queueProps) { 4 | return (function() { 5 | 'use strict'; 6 | 7 | const STATUS_QUEUED = 'queued'; 8 | const STATUS_IN_PROGRESS = 'inProgress'; 9 | const STATUS_SLEEPING = 'sleeping'; 10 | 11 | const EXCEPTION_TIMEOUT = 'timeout'; 12 | 13 | const DEFAULT_CONCURRENT_EXECUTORS = 1; 14 | const DEFAULT_JOB_TIMEOUT = 1000; 15 | const DEFAULT_PROCESSING_DELAY = 500; 16 | const DEFAULT_REQUEUE_DELAY = 5000; 17 | const PROCESSING_QUEUE_CHECK_INTERVAL = 50; 18 | 19 | const _queueProperties = { 20 | concurrentExecutors: DEFAULT_CONCURRENT_EXECUTORS, 21 | jobTimeout: DEFAULT_JOB_TIMEOUT, 22 | processingDelay: DEFAULT_PROCESSING_DELAY, 23 | executorFn: (tab, resolve, reject, requeue) => resolve(true), 24 | exceptionFn: (tab, resolve, reject, requeue) => resolve(false), 25 | }; 26 | const _tabDetailsByTabId = {}; 27 | const _queuedTabIds = []; 28 | let _processingQueueBufferTimer = null; 29 | let _queueId = queueId; 30 | 31 | setQueueProperties(queueProps); 32 | 33 | function setQueueProperties(queueProps) { 34 | for (const propName of Object.keys(queueProps)) { 35 | _queueProperties[propName] = queueProps[propName]; 36 | } 37 | if (!isValidInteger(_queueProperties.concurrentExecutors, 1)) { 38 | throw new Error( 39 | 'concurrentExecutors must be an integer greater than 0' 40 | ); 41 | } 42 | if (!isValidInteger(_queueProperties.jobTimeout, 1)) { 43 | throw new Error('jobTimeout must be an integer greater than 0'); 44 | } 45 | if (!isValidInteger(_queueProperties.processingDelay, 0)) { 46 | throw new Error('processingDelay must be an integer of at least 0'); 47 | } 48 | if (!(typeof _queueProperties.executorFn === 'function')) { 49 | throw new Error('executorFn must be a function'); 50 | } 51 | if (!(typeof _queueProperties.exceptionFn === 'function')) { 52 | throw new Error('executorFn must be a function'); 53 | } 54 | } 55 | 56 | function getQueueProperties() { 57 | return _queueProperties; 58 | } 59 | 60 | function isValidInteger(value, minimum) { 61 | return value !== null && !isNaN(Number(value) && value >= minimum); 62 | } 63 | 64 | function getTotalQueueSize() { 65 | return Object.keys(_tabDetailsByTabId).length; 66 | } 67 | 68 | function queueTabAsPromise(tab, executionProps, delay) { 69 | executionProps = executionProps || {}; 70 | let tabDetails = _tabDetailsByTabId[tab.id]; 71 | if (!tabDetails) { 72 | // gsUtils.log(tab.id, _queueId, 'Queueing new tab.'); 73 | tabDetails = { 74 | tab, 75 | executionProps, 76 | deferredPromise: createDeferredPromise(), 77 | status: STATUS_QUEUED, 78 | requeues: 0, 79 | }; 80 | addTabToQueue(tabDetails); 81 | } else { 82 | tabDetails.tab = tab; 83 | applyExecutionProps(tabDetails, executionProps); 84 | gsUtils.log(tab.id, _queueId, 'Tab already queued.'); 85 | } 86 | 87 | if (delay && isValidInteger(delay, 1)) { 88 | gsUtils.log(tab.id, _queueId, `Sleeping tab for ${delay}ms`); 89 | sleepTab(tabDetails, delay); 90 | } else { 91 | // If tab is already marked as sleeping then wake it up 92 | if (tabDetails.sleepTimer) { 93 | gsUtils.log(tab.id, _queueId, 'Removing tab from sleep'); 94 | clearTimeout(tabDetails.sleepTimer); 95 | delete tabDetails.sleepTimer; 96 | tabDetails.status = STATUS_QUEUED; 97 | } 98 | requestProcessQueue(0); 99 | } 100 | return tabDetails.deferredPromise; 101 | } 102 | 103 | function applyExecutionProps(tabDetails, executionProps) { 104 | executionProps = executionProps || {}; 105 | for (const prop in executionProps) { 106 | tabDetails.executionProps[prop] = executionProps[prop]; 107 | } 108 | } 109 | 110 | function unqueueTab(tab) { 111 | const tabDetails = _tabDetailsByTabId[tab.id]; 112 | if (tabDetails) { 113 | // gsUtils.log(tab.id, _queueId, 'Unqueueing tab.'); 114 | clearTimeout(tabDetails.timeoutTimer); 115 | removeTabFromQueue(tabDetails); 116 | rejectTabPromise(tabDetails, 'Queued tab job cancelled externally'); 117 | return true; 118 | } else { 119 | return false; 120 | } 121 | } 122 | 123 | function addTabToQueue(tabDetails) { 124 | const tab = tabDetails.tab; 125 | _tabDetailsByTabId[tab.id] = tabDetails; 126 | _queuedTabIds.push(tab.id); 127 | } 128 | 129 | function removeTabFromQueue(tabDetails) { 130 | const tab = tabDetails.tab; 131 | delete _tabDetailsByTabId[tab.id]; 132 | for (const [i, tabId] of _queuedTabIds.entries()) { 133 | if (tabId === tab.id) { 134 | _queuedTabIds.splice(i, 1); 135 | break; 136 | } 137 | } 138 | gsUtils.log(_queueId, `total queue size: ${_queuedTabIds.length}`); 139 | } 140 | 141 | // eslint-disable-next-line no-unused-vars 142 | function moveTabToEndOfQueue(tabDetails) { 143 | const tab = tabDetails.tab; 144 | for (const [i, tabId] of _queuedTabIds.entries()) { 145 | if (tabId === tab.id) { 146 | _queuedTabIds.push(_queuedTabIds.splice(i, 1)[0]); 147 | break; 148 | } 149 | } 150 | } 151 | 152 | function getQueuedTabDetails(tab) { 153 | return _tabDetailsByTabId[tab.id]; 154 | } 155 | 156 | function createDeferredPromise() { 157 | let res; 158 | let rej; 159 | const promise = new Promise((resolve, reject) => { 160 | res = resolve; 161 | rej = reject; 162 | }); 163 | promise.resolve = o => { 164 | res(o); 165 | return promise; 166 | }; 167 | promise.reject = o => { 168 | rej(o); 169 | return promise; 170 | }; 171 | return promise; 172 | } 173 | 174 | function requestProcessQueue(processingDelay) { 175 | setTimeout(() => { 176 | startProcessQueueBufferTimer(); 177 | }, processingDelay); 178 | } 179 | 180 | function startProcessQueueBufferTimer() { 181 | if (_processingQueueBufferTimer === null) { 182 | _processingQueueBufferTimer = setTimeout(() => { 183 | _processingQueueBufferTimer = null; 184 | processQueue(); 185 | }, PROCESSING_QUEUE_CHECK_INTERVAL); 186 | } 187 | } 188 | 189 | function processQueue() { 190 | let inProgressCount = 0; 191 | for (const tabId of _queuedTabIds) { 192 | const tabDetails = _tabDetailsByTabId[tabId]; 193 | if (tabDetails.status === STATUS_IN_PROGRESS) { 194 | inProgressCount += 1; 195 | } else if (tabDetails.status === STATUS_QUEUED) { 196 | processTab(tabDetails); 197 | inProgressCount += 1; 198 | } else if (tabDetails.status === STATUS_SLEEPING) { 199 | // ignore 200 | } 201 | if (inProgressCount >= _queueProperties.concurrentExecutors) { 202 | break; 203 | } 204 | } 205 | } 206 | 207 | function processTab(tabDetails) { 208 | tabDetails.status = STATUS_IN_PROGRESS; 209 | gsUtils.log( 210 | tabDetails.tab.id, 211 | _queueId, 212 | 'Executing executorFn for tab.' 213 | // tabDetails 214 | ); 215 | 216 | const _resolveTabPromise = r => resolveTabPromise(tabDetails, r); 217 | const _rejectTabPromise = e => rejectTabPromise(tabDetails, e); 218 | const _requeueTab = (requeueDelay, executionProps) => { 219 | requeueTab(tabDetails, requeueDelay, executionProps); 220 | }; 221 | 222 | // If timeout timer has not yet been initiated, then start it now 223 | if (!tabDetails.hasOwnProperty('timeoutTimer')) { 224 | tabDetails.timeoutTimer = setTimeout(() => { 225 | gsUtils.log(tabDetails.tab.id, _queueId, 'Tab job timed out'); 226 | _queueProperties.exceptionFn( 227 | tabDetails.tab, 228 | tabDetails.executionProps, 229 | EXCEPTION_TIMEOUT, 230 | _resolveTabPromise, 231 | _rejectTabPromise, 232 | _requeueTab 233 | ); //async. unhandled promise 234 | }, _queueProperties.jobTimeout); 235 | } 236 | 237 | _queueProperties.executorFn( 238 | tabDetails.tab, 239 | tabDetails.executionProps, 240 | _resolveTabPromise, 241 | _rejectTabPromise, 242 | _requeueTab 243 | ); //async. unhandled promise 244 | } 245 | 246 | function resolveTabPromise(tabDetails, result) { 247 | if (!_tabDetailsByTabId[tabDetails.tab.id]) { 248 | return; 249 | } 250 | gsUtils.log( 251 | tabDetails.tab.id, 252 | _queueId, 253 | 'Queued tab resolved. Result: ', 254 | result 255 | ); 256 | clearTimeout(tabDetails.timeoutTimer); 257 | removeTabFromQueue(tabDetails); 258 | tabDetails.deferredPromise.resolve(result); 259 | requestProcessQueue(_queueProperties.processingDelay); 260 | } 261 | 262 | function rejectTabPromise(tabDetails, error) { 263 | if (!_tabDetailsByTabId[tabDetails.tab.id]) { 264 | return; 265 | } 266 | gsUtils.log( 267 | tabDetails.tab.id, 268 | _queueId, 269 | 'Queued tab rejected. Error: ', 270 | error 271 | ); 272 | clearTimeout(tabDetails.timeoutTimer); 273 | removeTabFromQueue(tabDetails); 274 | tabDetails.deferredPromise.reject(error); 275 | requestProcessQueue(_queueProperties.processingDelay); 276 | } 277 | 278 | function requeueTab(tabDetails, requeueDelay, executionProps) { 279 | requeueDelay = requeueDelay || DEFAULT_REQUEUE_DELAY; 280 | if (executionProps) { 281 | applyExecutionProps(tabDetails, executionProps); 282 | } 283 | tabDetails.requeues += 1; 284 | gsUtils.log( 285 | tabDetails.tab.id, 286 | _queueId, 287 | `Requeueing tab. Requeues: ${tabDetails.requeues}` 288 | ); 289 | // moveTabToEndOfQueue(tabDetails); 290 | sleepTab(tabDetails, requeueDelay); 291 | requestProcessQueue(_queueProperties.processingDelay); 292 | } 293 | 294 | function sleepTab(tabDetails, delay) { 295 | tabDetails.status = STATUS_SLEEPING; 296 | if (tabDetails.sleepTimer) { 297 | clearTimeout(tabDetails.sleepTimer); 298 | } 299 | tabDetails.sleepTimer = window.setTimeout(() => { 300 | delete tabDetails.sleepTimer; 301 | tabDetails.status = STATUS_QUEUED; 302 | requestProcessQueue(0); 303 | }, delay); 304 | } 305 | 306 | return { 307 | EXCEPTION_TIMEOUT, 308 | setQueueProperties, 309 | getQueueProperties, 310 | getTotalQueueSize, 311 | queueTabAsPromise, 312 | unqueueTab, 313 | getQueuedTabDetails, 314 | }; 315 | })(); 316 | } 317 | -------------------------------------------------------------------------------- /src/js/history.js: -------------------------------------------------------------------------------- 1 | /*global chrome, historyItems, historyUtils, gsSession, gsIndexedDb, gsUtils */ 2 | (function(global) { 3 | 'use strict'; 4 | 5 | try { 6 | chrome.extension.getBackgroundPage().tgs.setViewGlobals(global); 7 | } catch (e) { 8 | window.setTimeout(() => window.location.reload(), 1000); 9 | return; 10 | } 11 | 12 | async function reloadTabs(sessionId, windowId, openTabsAsSuspended) { 13 | const session = await gsIndexedDb.fetchSessionBySessionId(sessionId); 14 | if (!session || !session.windows) { 15 | return; 16 | } 17 | 18 | gsUtils.removeInternalUrlsFromSession(session); 19 | 20 | //if loading a specific window 21 | let sessionWindows = []; 22 | if (windowId) { 23 | sessionWindows.push(gsUtils.getWindowFromSession(windowId, session)); 24 | //else load all windows from session 25 | } else { 26 | sessionWindows = session.windows; 27 | } 28 | 29 | for (let sessionWindow of sessionWindows) { 30 | const suspendMode = openTabsAsSuspended ? 1 : 2; 31 | await gsSession.restoreSessionWindow(sessionWindow, null, suspendMode); 32 | } 33 | } 34 | 35 | function deleteSession(sessionId) { 36 | var result = window.confirm( 37 | chrome.i18n.getMessage('js_history_confirm_delete') 38 | ); 39 | if (result) { 40 | gsIndexedDb.removeSessionFromHistory(sessionId).then(function() { 41 | window.location.reload(); 42 | }); 43 | } 44 | } 45 | 46 | function removeTab(element, sessionId, windowId, tabId) { 47 | var sessionEl, newSessionEl; 48 | 49 | gsIndexedDb 50 | .removeTabFromSessionHistory(sessionId, windowId, tabId) 51 | .then(function(session) { 52 | gsUtils.removeInternalUrlsFromSession(session); 53 | //if we have a valid session returned 54 | if (session) { 55 | sessionEl = element.parentElement.parentElement; 56 | newSessionEl = createSessionElement(session); 57 | sessionEl.parentElement.replaceChild(newSessionEl, sessionEl); 58 | toggleSession(newSessionEl, session.sessionId); //async. unhandled promise 59 | 60 | //otherwise assume it was the last tab in session and session has been removed 61 | } else { 62 | window.location.reload(); 63 | } 64 | }); 65 | } 66 | 67 | async function toggleSession(element, sessionId) { 68 | var sessionContentsEl = element.getElementsByClassName( 69 | 'sessionContents' 70 | )[0]; 71 | var sessionIcon = element.getElementsByClassName('sessionIcon')[0]; 72 | if (sessionIcon.classList.contains('icon-plus-squared-alt')) { 73 | sessionIcon.classList.remove('icon-plus-squared-alt'); 74 | sessionIcon.classList.add('icon-minus-squared-alt'); 75 | } else { 76 | sessionIcon.classList.remove('icon-minus-squared-alt'); 77 | sessionIcon.classList.add('icon-plus-squared-alt'); 78 | } 79 | 80 | //if toggled on already, then toggle off 81 | if (sessionContentsEl.childElementCount > 0) { 82 | sessionContentsEl.innerHTML = ''; 83 | return; 84 | } 85 | 86 | gsIndexedDb 87 | .fetchSessionBySessionId(sessionId) 88 | .then(async function(curSession) { 89 | if (!curSession || !curSession.windows) { 90 | return; 91 | } 92 | gsUtils.removeInternalUrlsFromSession(curSession); 93 | 94 | for (const [i, curWindow] of curSession.windows.entries()) { 95 | curWindow.sessionId = curSession.sessionId; 96 | sessionContentsEl.appendChild( 97 | createWindowElement(curSession, curWindow, i) 98 | ); 99 | 100 | const tabPromises = []; 101 | for (const curTab of curWindow.tabs) { 102 | curTab.windowId = curWindow.id; 103 | curTab.sessionId = curSession.sessionId; 104 | curTab.title = gsUtils.getCleanTabTitle(curTab); 105 | if (gsUtils.isSuspendedTab(curTab)) { 106 | curTab.url = gsUtils.getOriginalUrl(curTab.url); 107 | } 108 | tabPromises.push(createTabElement(curSession, curWindow, curTab)); 109 | } 110 | const tabEls = await Promise.all(tabPromises); 111 | for (const tabEl of tabEls) { 112 | sessionContentsEl.appendChild(tabEl); 113 | } 114 | } 115 | }); 116 | } 117 | 118 | function addClickListenerToElement(element, func) { 119 | if (element) { 120 | element.onclick = func; 121 | } 122 | } 123 | 124 | function createSessionElement(session) { 125 | var sessionEl = historyItems.createSessionHtml(session, true); 126 | 127 | addClickListenerToElement( 128 | sessionEl.getElementsByClassName('sessionIcon')[0], 129 | function() { 130 | toggleSession(sessionEl, session.sessionId); //async. unhandled promise 131 | } 132 | ); 133 | addClickListenerToElement( 134 | sessionEl.getElementsByClassName('sessionLink')[0], 135 | function() { 136 | toggleSession(sessionEl, session.sessionId); //async. unhandled promise 137 | } 138 | ); 139 | addClickListenerToElement( 140 | sessionEl.getElementsByClassName('exportLink')[0], 141 | function() { 142 | historyUtils.exportSessionWithId(session.sessionId); 143 | } 144 | ); 145 | addClickListenerToElement( 146 | sessionEl.getElementsByClassName('resuspendLink')[0], 147 | function() { 148 | reloadTabs(session.sessionId, null, true); // async 149 | } 150 | ); 151 | addClickListenerToElement( 152 | sessionEl.getElementsByClassName('reloadLink')[0], 153 | function() { 154 | reloadTabs(session.sessionId, null, false); // async 155 | } 156 | ); 157 | addClickListenerToElement( 158 | sessionEl.getElementsByClassName('saveLink')[0], 159 | function() { 160 | historyUtils.saveSession(session.sessionId); 161 | } 162 | ); 163 | addClickListenerToElement( 164 | sessionEl.getElementsByClassName('deleteLink')[0], 165 | function() { 166 | deleteSession(session.sessionId); 167 | } 168 | ); 169 | return sessionEl; 170 | } 171 | 172 | function createWindowElement(session, window, index) { 173 | var allowReload = session.sessionId !== gsSession.getSessionId(); 174 | var windowEl = historyItems.createWindowHtml(window, index, allowReload); 175 | 176 | addClickListenerToElement( 177 | windowEl.getElementsByClassName('resuspendLink')[0], 178 | function() { 179 | reloadTabs(session.sessionId, window.id, true); // async 180 | } 181 | ); 182 | addClickListenerToElement( 183 | windowEl.getElementsByClassName('reloadLink')[0], 184 | function() { 185 | reloadTabs(session.sessionId, window.id, false); // async 186 | } 187 | ); 188 | return windowEl; 189 | } 190 | 191 | async function createTabElement(session, window, tab) { 192 | var allowDelete = session.sessionId !== gsSession.getSessionId(); 193 | var tabEl = await historyItems.createTabHtml(tab, allowDelete); 194 | 195 | addClickListenerToElement( 196 | tabEl.getElementsByClassName('removeLink')[0], 197 | function() { 198 | removeTab(tabEl, session.sessionId, window.id, tab.id); 199 | } 200 | ); 201 | return tabEl; 202 | } 203 | 204 | function render() { 205 | var currentDiv = document.getElementById('currentSessions'), 206 | sessionsDiv = document.getElementById('recoverySessions'), 207 | historyDiv = document.getElementById('historySessions'), 208 | importSessionEl = document.getElementById('importSession'), 209 | importSessionActionEl = document.getElementById('importSessionAction'), 210 | firstSession = true; 211 | 212 | currentDiv.innerHTML = ''; 213 | sessionsDiv.innerHTML = ''; 214 | historyDiv.innerHTML = ''; 215 | 216 | gsIndexedDb.fetchCurrentSessions().then(function(currentSessions) { 217 | currentSessions.forEach(function(session, index) { 218 | gsUtils.removeInternalUrlsFromSession(session); 219 | var sessionEl = createSessionElement(session); 220 | if (firstSession) { 221 | currentDiv.appendChild(sessionEl); 222 | firstSession = false; 223 | } else { 224 | sessionsDiv.appendChild(sessionEl); 225 | } 226 | }); 227 | }); 228 | 229 | gsIndexedDb.fetchSavedSessions().then(function(savedSessions) { 230 | savedSessions.forEach(function(session, index) { 231 | gsUtils.removeInternalUrlsFromSession(session); 232 | var sessionEl = createSessionElement(session); 233 | historyDiv.appendChild(sessionEl); 234 | }); 235 | }); 236 | 237 | importSessionActionEl.addEventListener( 238 | 'change', 239 | historyUtils.importSession, 240 | false 241 | ); 242 | importSessionEl.onclick = function() { 243 | importSessionActionEl.click(); 244 | }; 245 | 246 | //hide incompatible sidebar items if in incognito mode 247 | if (chrome.extension.inIncognitoContext) { 248 | Array.prototype.forEach.call( 249 | document.getElementsByClassName('noIncognito'), 250 | function(el) { 251 | el.style.display = 'none'; 252 | } 253 | ); 254 | } 255 | } 256 | 257 | gsUtils.documentReadyAndLocalisedAsPromsied(document).then(function() { 258 | render(); 259 | }); 260 | })(this); 261 | -------------------------------------------------------------------------------- /src/js/historyItems.js: -------------------------------------------------------------------------------- 1 | /*global chrome, gsSession, gsUtils, gsFavicon */ 2 | // eslint-disable-next-line no-unused-vars 3 | var historyItems = (function(global) { 4 | 'use strict'; 5 | 6 | if ( 7 | !chrome.extension.getBackgroundPage() || 8 | !chrome.extension.getBackgroundPage().tgs 9 | ) { 10 | return; 11 | } 12 | chrome.extension.getBackgroundPage().tgs.setViewGlobals(global); 13 | 14 | function createSessionHtml(session, showLinks) { 15 | session.windows = session.windows || []; 16 | 17 | var sessionType = 18 | session.sessionId === gsSession.getSessionId() 19 | ? 'current' 20 | : session.name 21 | ? 'saved' 22 | : 'recent', 23 | sessionContainer, 24 | sessionTitle, 25 | sessionSave, 26 | sessionDelete, 27 | sessionExport, 28 | sessionDiv, 29 | sessionIcon, 30 | windowResuspend, 31 | windowReload, 32 | titleText, 33 | winCnt = session.windows.length, 34 | tabCnt = session.windows.reduce(function(a, b) { 35 | return a + b.tabs.length; 36 | }, 0); 37 | 38 | if (sessionType === 'saved') { 39 | titleText = session.name; 40 | } else { 41 | titleText = gsUtils.getHumanDate(session.date); 42 | } 43 | titleText += 44 | '  (' + 45 | winCnt + 46 | pluralise( 47 | ' ' + chrome.i18n.getMessage('js_history_window').toLowerCase(), 48 | winCnt 49 | ) + 50 | ', ' + 51 | tabCnt + 52 | pluralise( 53 | ' ' + chrome.i18n.getMessage('js_history_tab').toLowerCase(), 54 | tabCnt 55 | ) + 56 | ')'; 57 | 58 | sessionIcon = createEl('i', { 59 | class: 'sessionIcon icon icon-plus-squared-alt', 60 | }); 61 | 62 | sessionDiv = createEl('div', { 63 | class: 'sessionContents', 64 | }); 65 | 66 | sessionTitle = createEl('span', { 67 | class: 'sessionLink', 68 | }); 69 | sessionTitle.innerHTML = titleText; 70 | 71 | sessionSave = createEl( 72 | 'a', 73 | { 74 | class: 'groupLink saveLink', 75 | href: '#', 76 | }, 77 | chrome.i18n.getMessage('js_history_save') 78 | ); 79 | 80 | sessionDelete = createEl( 81 | 'a', 82 | { 83 | class: 'groupLink deleteLink', 84 | href: '#', 85 | }, 86 | chrome.i18n.getMessage('js_history_delete') 87 | ); 88 | 89 | windowResuspend = createEl( 90 | 'a', 91 | { 92 | class: 'groupLink resuspendLink', 93 | href: '#', 94 | }, 95 | chrome.i18n.getMessage('js_history_resuspend') 96 | ); 97 | 98 | windowReload = createEl( 99 | 'a', 100 | { 101 | class: 'groupLink reloadLink', 102 | href: '#', 103 | }, 104 | chrome.i18n.getMessage('js_history_reload') 105 | ); 106 | 107 | sessionExport = createEl( 108 | 'a', 109 | { 110 | class: 'groupLink exportLink', 111 | href: '#', 112 | }, 113 | chrome.i18n.getMessage('js_history_export') 114 | ); 115 | 116 | sessionContainer = createEl('div', { 117 | class: 'sessionContainer', 118 | }); 119 | sessionContainer.appendChild(sessionIcon); 120 | sessionContainer.appendChild(sessionTitle); 121 | if (showLinks && sessionType !== 'current') { 122 | sessionContainer.appendChild(windowResuspend); 123 | sessionContainer.appendChild(windowReload); 124 | } 125 | if (showLinks) { 126 | sessionContainer.appendChild(sessionExport); 127 | } 128 | if (showLinks && sessionType !== 'saved') { 129 | sessionContainer.appendChild(sessionSave); 130 | } 131 | if (showLinks && sessionType !== 'current') { 132 | sessionContainer.appendChild(sessionDelete); 133 | } 134 | 135 | sessionContainer.appendChild(sessionDiv); 136 | 137 | return sessionContainer; 138 | } 139 | 140 | function createWindowHtml(window, index, showLinks) { 141 | var groupHeading, windowContainer, groupUnsuspendCurrent, groupUnsuspendNew; 142 | 143 | groupHeading = createEl('div', { 144 | class: 'windowContainer', 145 | }); 146 | 147 | var windowString = chrome.i18n.getMessage('js_history_window'); 148 | windowContainer = createEl( 149 | 'span', 150 | {}, 151 | windowString + ' ' + (index + 1) + ':\u00A0' 152 | ); 153 | 154 | groupUnsuspendCurrent = createEl( 155 | 'a', 156 | { 157 | class: 'groupLink resuspendLink ', 158 | href: '#', 159 | }, 160 | chrome.i18n.getMessage('js_history_resuspend') 161 | ); 162 | 163 | groupUnsuspendNew = createEl( 164 | 'a', 165 | { 166 | class: 'groupLink reloadLink', 167 | href: '#', 168 | }, 169 | chrome.i18n.getMessage('js_history_reload') 170 | ); 171 | 172 | groupHeading.appendChild(windowContainer); 173 | if (showLinks) { 174 | groupHeading.appendChild(groupUnsuspendCurrent); 175 | groupHeading.appendChild(groupUnsuspendNew); 176 | } 177 | 178 | return groupHeading; 179 | } 180 | 181 | async function createTabHtml(tab, showLinks) { 182 | var linksSpan, listImg, listLink, listHover; 183 | 184 | if (tab.sessionId) { 185 | linksSpan = createEl('div', { 186 | class: 'tabContainer', 187 | 'data-tabId': tab.id || tab.url, 188 | 'data-url': tab.url, 189 | }); 190 | } else { 191 | linksSpan = createEl('div', { 192 | class: 'tabContainer', 193 | 'data-url': tab.url, 194 | }); 195 | } 196 | 197 | listHover = createEl( 198 | 'span', 199 | { 200 | class: 'itemHover removeLink', 201 | }, 202 | '\u2716' 203 | ); 204 | 205 | const faviconMeta = await gsFavicon.getFaviconMetaData(tab); 206 | const favIconUrl = faviconMeta.normalisedDataUrl; 207 | listImg = createEl('img', { 208 | src: favIconUrl, 209 | height: '16px', 210 | width: '16px', 211 | }); 212 | 213 | listLink = createEl( 214 | 'a', 215 | { 216 | class: 'historyLink', 217 | href: tab.url, 218 | target: '_blank', 219 | }, 220 | tab.title && tab.title.length > 1 ? tab.title : tab.url 221 | ); 222 | 223 | if (showLinks) { 224 | linksSpan.appendChild(listHover); 225 | } 226 | linksSpan.appendChild(listImg); 227 | linksSpan.appendChild(listLink); 228 | linksSpan.appendChild(createEl('br')); 229 | 230 | return linksSpan; 231 | } 232 | 233 | function createEl(elType, attributes, text) { 234 | var el = document.createElement(elType); 235 | attributes = attributes || {}; 236 | el = setElAttributes(el, attributes); 237 | el.innerHTML = gsUtils.htmlEncode(text || ''); 238 | return el; 239 | } 240 | function setElAttributes(el, attributes) { 241 | for (var key in attributes) { 242 | if (attributes.hasOwnProperty(key)) { 243 | el.setAttribute(key, attributes[key]); 244 | } 245 | } 246 | return el; 247 | } 248 | 249 | function pluralise(text, count) { 250 | return ( 251 | text + (count > 1 ? chrome.i18n.getMessage('js_history_plural') : '') 252 | ); 253 | } 254 | 255 | return { 256 | createSessionHtml: createSessionHtml, 257 | createWindowHtml: createWindowHtml, 258 | createTabHtml: createTabHtml, 259 | }; 260 | })(this); 261 | -------------------------------------------------------------------------------- /src/js/historyUtils.js: -------------------------------------------------------------------------------- 1 | /* global chrome, gsIndexedDb, gsUtils */ 2 | // eslint-disable-next-line no-unused-vars 3 | var historyUtils = (function(global) { 4 | 'use strict'; 5 | 6 | if ( 7 | !chrome.extension.getBackgroundPage() || 8 | !chrome.extension.getBackgroundPage().tgs 9 | ) { 10 | return; 11 | } 12 | chrome.extension.getBackgroundPage().tgs.setViewGlobals(global); 13 | 14 | var noop = function() {}; 15 | 16 | function importSession(e) { 17 | var f = e.target.files[0]; 18 | if (f) { 19 | var r = new FileReader(); 20 | r.onload = function(e) { 21 | var contents = e.target.result; 22 | if (f.type !== 'text/plain') { 23 | alert(chrome.i18n.getMessage('js_history_import_fail')); 24 | } else { 25 | handleImport(f.name, contents).then(function() { 26 | window.location.reload(); 27 | }); 28 | } 29 | }; 30 | r.readAsText(f); 31 | } else { 32 | alert(chrome.i18n.getMessage('js_history_import_fail')); 33 | } 34 | } 35 | 36 | async function handleImport(sessionName, textContents) { 37 | sessionName = window.prompt( 38 | chrome.i18n.getMessage('js_history_enter_name_for_session'), 39 | sessionName 40 | ); 41 | if (sessionName) { 42 | const shouldSave = await new Promise(resolve => { 43 | validateNewSessionName(sessionName, function(result) { 44 | resolve(result); 45 | }); 46 | }); 47 | if (!shouldSave) { 48 | return; 49 | } 50 | 51 | var sessionId = '_' + gsUtils.generateHashCode(sessionName); 52 | var windows = []; 53 | 54 | var createNextWindow = function() { 55 | return { 56 | id: sessionId + '_' + windows.length, 57 | tabs: [], 58 | }; 59 | }; 60 | var curWindow = createNextWindow(); 61 | 62 | for (const line of textContents.split('\n')) { 63 | if (typeof line !== 'string') { 64 | continue; 65 | } 66 | if (line === '') { 67 | if (curWindow.tabs.length > 0) { 68 | windows.push(curWindow); 69 | curWindow = createNextWindow(); 70 | } 71 | continue; 72 | } 73 | if (line.indexOf('://') < 0) { 74 | continue; 75 | } 76 | const tabInfo = { 77 | windowId: curWindow.id, 78 | sessionId: sessionId, 79 | id: curWindow.id + '_' + curWindow.tabs.length, 80 | url: line, 81 | title: line, 82 | index: curWindow.tabs.length, 83 | pinned: false, 84 | }; 85 | const savedTabInfo = await gsIndexedDb.fetchTabInfo(line); 86 | if (savedTabInfo) { 87 | tabInfo.title = savedTabInfo.title; 88 | tabInfo.favIconUrl = savedTabInfo.favIconUrl; 89 | } 90 | curWindow.tabs.push(tabInfo); 91 | } 92 | if (curWindow.tabs.length > 0) { 93 | windows.push(curWindow); 94 | } 95 | 96 | var session = { 97 | name: sessionName, 98 | sessionId: sessionId, 99 | windows: windows, 100 | date: new Date().toISOString(), 101 | }; 102 | await gsIndexedDb.updateSession(session); 103 | } 104 | } 105 | 106 | function exportSessionWithId(sessionId, callback) { 107 | callback = typeof callback !== 'function' ? noop : callback; 108 | 109 | gsIndexedDb.fetchSessionBySessionId(sessionId).then(function(session) { 110 | if (!session || !session.windows) { 111 | callback(); 112 | } else { 113 | exportSession(session, callback); 114 | } 115 | }); 116 | } 117 | 118 | function exportSession(session, callback) { 119 | let sessionString = ''; 120 | 121 | session.windows.forEach(function(curWindow, index) { 122 | curWindow.tabs.forEach(function(curTab, tabIndex) { 123 | if (gsUtils.isSuspendedTab(curTab)) { 124 | sessionString += gsUtils.getOriginalUrl(curTab.url) + '\n'; 125 | } else { 126 | sessionString += curTab.url + '\n'; 127 | } 128 | }); 129 | //add an extra newline to separate windows 130 | sessionString += '\n'; 131 | }); 132 | 133 | const blob = new Blob([sessionString], { type: 'text/plain' }); 134 | const blobUrl = URL.createObjectURL(blob); 135 | const link = document.createElement('a'); 136 | link.setAttribute('href', blobUrl); 137 | link.setAttribute('download', 'session.txt'); 138 | link.click(); 139 | 140 | callback(); 141 | } 142 | 143 | function validateNewSessionName(sessionName, callback) { 144 | gsIndexedDb.fetchSavedSessions().then(function(savedSessions) { 145 | var nameExists = savedSessions.some(function(savedSession, index) { 146 | return savedSession.name === sessionName; 147 | }); 148 | if (nameExists) { 149 | var overwrite = window.confirm( 150 | chrome.i18n.getMessage('js_history_confirm_session_overwrite') 151 | ); 152 | if (!overwrite) { 153 | callback(false); 154 | return; 155 | } 156 | } 157 | callback(true); 158 | }); 159 | } 160 | 161 | function saveSession(sessionId) { 162 | gsIndexedDb.fetchSessionBySessionId(sessionId).then(function(session) { 163 | if (!session) { 164 | gsUtils.warning( 165 | 'historyUtils', 166 | 'Could not find session with sessionId: ' + 167 | sessionId + 168 | '. Save aborted' 169 | ); 170 | return; 171 | } 172 | var sessionName = window.prompt( 173 | chrome.i18n.getMessage('js_history_enter_name_for_session') 174 | ); 175 | if (sessionName) { 176 | historyUtils.validateNewSessionName(sessionName, function(shouldSave) { 177 | if (shouldSave) { 178 | session.name = sessionName; 179 | gsIndexedDb.addToSavedSessions(session).then(function() { 180 | window.location.reload(); 181 | }); 182 | } 183 | }); 184 | } 185 | }); 186 | } 187 | 188 | return { 189 | importSession, 190 | exportSession, 191 | exportSessionWithId, 192 | validateNewSessionName, 193 | saveSession, 194 | }; 195 | })(this); 196 | -------------------------------------------------------------------------------- /src/js/notice.js: -------------------------------------------------------------------------------- 1 | /*global chrome, tgs, gsStorage, gsUtils */ 2 | (function(global) { 3 | 'use strict'; 4 | 5 | try { 6 | chrome.extension.getBackgroundPage().tgs.setViewGlobals(global); 7 | } catch (e) { 8 | window.setTimeout(() => window.location.reload(), 1000); 9 | return; 10 | } 11 | 12 | gsUtils.documentReadyAndLocalisedAsPromsied(document).then(function() { 13 | var notice = tgs.requestNotice(); 14 | if ( 15 | notice && 16 | notice.hasOwnProperty('text') && 17 | notice.hasOwnProperty('version') 18 | ) { 19 | var noticeContentEl = document.getElementById('gsNotice'); 20 | noticeContentEl.innerHTML = notice.text; 21 | //update local notice version 22 | gsStorage.setNoticeVersion(notice.version); 23 | } 24 | 25 | //clear notice (to prevent it showing again) 26 | tgs.clearNotice(); 27 | }); 28 | })(this); 29 | -------------------------------------------------------------------------------- /src/js/options.js: -------------------------------------------------------------------------------- 1 | /*global chrome, gsStorage, gsChrome, gsUtils */ 2 | (function(global) { 3 | try { 4 | chrome.extension.getBackgroundPage().tgs.setViewGlobals(global); 5 | } catch (e) { 6 | window.setTimeout(() => window.location.reload(), 1000); 7 | return; 8 | } 9 | 10 | var elementPrefMap = { 11 | preview: gsStorage.SCREEN_CAPTURE, 12 | forceScreenCapture: gsStorage.SCREEN_CAPTURE_FORCE, 13 | suspendInPlaceOfDiscard: gsStorage.SUSPEND_IN_PLACE_OF_DISCARD, 14 | discardInPlaceOfSuspend: gsStorage.DISCARD_IN_PLACE_OF_SUSPEND, 15 | onlineCheck: gsStorage.IGNORE_WHEN_OFFLINE, 16 | batteryCheck: gsStorage.IGNORE_WHEN_CHARGING, 17 | unsuspendOnFocus: gsStorage.UNSUSPEND_ON_FOCUS, 18 | discardAfterSuspend: gsStorage.DISCARD_AFTER_SUSPEND, 19 | dontSuspendPinned: gsStorage.IGNORE_PINNED, 20 | dontSuspendForms: gsStorage.IGNORE_FORMS, 21 | dontSuspendAudio: gsStorage.IGNORE_AUDIO, 22 | dontSuspendActiveTabs: gsStorage.IGNORE_ACTIVE_TABS, 23 | ignoreCache: gsStorage.IGNORE_CACHE, 24 | addContextMenu: gsStorage.ADD_CONTEXT, 25 | syncSettings: gsStorage.SYNC_SETTINGS, 26 | timeToSuspend: gsStorage.SUSPEND_TIME, 27 | theme: gsStorage.THEME, 28 | whitelist: gsStorage.WHITELIST, 29 | }; 30 | 31 | var hideWhenDiscardSet = [ 32 | 'previewContainer', 33 | 'suspendedOptionsContainer', 34 | 'suspendInPlaceOfDiscard', 35 | ] 36 | 37 | function selectComboBox(element, key) { 38 | console.log("conbobox", element, key); 39 | var i, child; 40 | 41 | for (i = 0; i < element.children.length; i += 1) { 42 | child = element.children[i]; 43 | if (child.value === key) { 44 | child.selected = 'true'; 45 | break; 46 | } 47 | } 48 | } 49 | 50 | //populate settings from synced storage 51 | function initSettings() { 52 | var optionEls = document.getElementsByClassName('option'), 53 | pref, 54 | element, 55 | i; 56 | 57 | for (i = 0; i < optionEls.length; i++) { 58 | element = optionEls[i]; 59 | pref = elementPrefMap[element.id]; 60 | populateOption(element, gsStorage.getOption(pref)); 61 | } 62 | 63 | setForceScreenCaptureVisibility( 64 | gsStorage.getOption(gsStorage.SCREEN_CAPTURE) !== '0' 65 | ); 66 | setAutoSuspendOptionsVisibility( 67 | parseFloat(gsStorage.getOption(gsStorage.SUSPEND_TIME)) > 0 68 | ); 69 | setSyncNoteVisibility(!gsStorage.getOption(gsStorage.SYNC_SETTINGS)); 70 | 71 | let searchParams = new URL(location.href).searchParams; 72 | if (searchParams.has('firstTime')) { 73 | document 74 | .querySelector('.welcome-message') 75 | .classList.remove('reallyHidden'); 76 | document.querySelector('#options-heading').classList.add('reallyHidden'); 77 | } 78 | 79 | var discardSet = gsStorage.getOption(gsStorage.DISCARD_IN_PLACE_OF_SUSPEND); 80 | showHideSuspendOnlyOptions(discardSet); 81 | } 82 | 83 | function populateOption(element, value) { 84 | if ( 85 | element.tagName === 'INPUT' && 86 | element.hasAttribute('type') && 87 | element.getAttribute('type') === 'checkbox' 88 | ) { 89 | element.checked = value; 90 | } else if (element.tagName === 'SELECT') { 91 | selectComboBox(element, value); 92 | } else if (element.tagName === 'TEXTAREA') { 93 | element.value = value; 94 | } 95 | } 96 | 97 | function getOptionValue(element) { 98 | if ( 99 | element.tagName === 'INPUT' && 100 | element.hasAttribute('type') && 101 | element.getAttribute('type') === 'checkbox' 102 | ) { 103 | return element.checked; 104 | } 105 | if (element.tagName === 'SELECT') { 106 | return element.children[element.selectedIndex].value; 107 | } 108 | if (element.tagName === 'TEXTAREA') { 109 | return element.value; 110 | } 111 | } 112 | 113 | function setForceScreenCaptureVisibility(visible) { 114 | if (visible) { 115 | document.getElementById('forceScreenCaptureContainer').style.display = 116 | 'block'; 117 | } else { 118 | document.getElementById('forceScreenCaptureContainer').style.display = 119 | 'none'; 120 | } 121 | } 122 | 123 | function setSyncNoteVisibility(visible) { 124 | if (visible) { 125 | document.getElementById('syncNote').style.display = 'block'; 126 | } else { 127 | document.getElementById('syncNote').style.display = 'none'; 128 | } 129 | } 130 | 131 | function setAutoSuspendOptionsVisibility(visible) { 132 | Array.prototype.forEach.call( 133 | document.getElementsByClassName('autoSuspendOption'), 134 | function(el) { 135 | if (visible) { 136 | el.style.display = 'block'; 137 | } else { 138 | el.style.display = 'none'; 139 | } 140 | } 141 | ); 142 | } 143 | 144 | function showHideSuspendOnlyOptions(hide) { 145 | hideWhenDiscardSet.forEach(elementId => { 146 | console.log(elementId); 147 | element = document.getElementById(elementId); 148 | console.log(element); 149 | element.style.visibility = hide ? 'hidden' : 'visible'; 150 | }); 151 | } 152 | 153 | function handleChange(element) { 154 | return function() { 155 | console.log(element); 156 | var pref = elementPrefMap[element.id], 157 | interval; 158 | 159 | //add specific screen element listeners 160 | if (pref === gsStorage.SCREEN_CAPTURE) { 161 | setForceScreenCaptureVisibility(getOptionValue(element) !== '0'); 162 | } else if (pref === gsStorage.SUSPEND_TIME) { 163 | interval = getOptionValue(element); 164 | setAutoSuspendOptionsVisibility(interval > 0); 165 | } else if (pref === gsStorage.SYNC_SETTINGS) { 166 | // we only really want to show this on load. not on toggle 167 | if (getOptionValue(element)) { 168 | setSyncNoteVisibility(false); 169 | } 170 | } else if (pref == gsStorage.DISCARD_IN_PLACE_OF_SUSPEND) { 171 | var discardSet = getOptionValue(element); 172 | showHideSuspendOnlyOptions(discardSet); 173 | } 174 | 175 | var [oldValue, newValue] = saveChange(element); 176 | if (oldValue !== newValue) { 177 | var prefKey = elementPrefMap[element.id]; 178 | gsUtils.performPostSaveUpdates( 179 | [prefKey], 180 | { [prefKey]: oldValue }, 181 | { [prefKey]: newValue } 182 | ); 183 | } 184 | }; 185 | } 186 | 187 | function saveChange(element) { 188 | var pref = elementPrefMap[element.id], 189 | oldValue = gsStorage.getOption(pref), 190 | newValue = getOptionValue(element); 191 | 192 | //clean up whitelist before saving 193 | if (pref === gsStorage.WHITELIST) { 194 | newValue = gsUtils.cleanupWhitelist(newValue); 195 | } 196 | 197 | //save option 198 | if (oldValue !== newValue) { 199 | gsStorage.setOptionAndSync(elementPrefMap[element.id], newValue); 200 | } 201 | 202 | return [oldValue, newValue]; 203 | } 204 | 205 | gsUtils.documentReadyAndLocalisedAsPromsied(document).then(function() { 206 | initSettings(); 207 | 208 | var optionEls = document.getElementsByClassName('option'), 209 | element, 210 | i; 211 | 212 | //add change listeners for all 'option' elements 213 | for (i = 0; i < optionEls.length; i++) { 214 | element = optionEls[i]; 215 | if (element.tagName === 'TEXTAREA') { 216 | element.addEventListener( 217 | 'input', 218 | gsUtils.debounce(handleChange(element), 200), 219 | false 220 | ); 221 | } else { 222 | element.onchange = handleChange(element); 223 | } 224 | } 225 | 226 | document.getElementById('testWhitelistBtn').onclick = async e => { 227 | e.preventDefault(); 228 | const tabs = await gsChrome.tabsQuery(); 229 | const tabUrls = tabs 230 | .map( 231 | tab => 232 | gsUtils.isSuspendedTab(tab) 233 | ? gsUtils.getOriginalUrl(tab.url) 234 | : tab.url 235 | ) 236 | .filter( 237 | url => !gsUtils.isSuspendedUrl(url) && gsUtils.checkWhiteList(url) 238 | ) 239 | .map(url => (url.length > 55 ? url.substr(0, 52) + '...' : url)); 240 | if (tabUrls.length === 0) { 241 | alert(chrome.i18n.getMessage('js_options_whitelist_no_matches')); 242 | return; 243 | } 244 | const firstUrls = tabUrls.splice(0, 22); 245 | let alertString = `${chrome.i18n.getMessage( 246 | 'js_options_whitelist_matches_heading' 247 | )}\n${firstUrls.join('\n')}`; 248 | 249 | if (tabUrls.length > 0) { 250 | alertString += `\n${chrome.i18n.getMessage( 251 | 'js_options_whitelist_matches_overflow_prefix' 252 | )} ${tabUrls.length} ${chrome.i18n.getMessage( 253 | 'js_options_whitelist_matches_overflow_suffix' 254 | )}`; 255 | } 256 | alert(alertString); 257 | }; 258 | 259 | //hide incompatible sidebar items if in incognito mode 260 | if (chrome.extension.inIncognitoContext) { 261 | Array.prototype.forEach.call( 262 | document.getElementsByClassName('noIncognito'), 263 | function(el) { 264 | el.style.display = 'none'; 265 | } 266 | ); 267 | window.alert(chrome.i18n.getMessage('js_options_incognito_warning')); 268 | } 269 | }); 270 | 271 | global.exports = { 272 | initSettings, 273 | }; 274 | })(this); 275 | -------------------------------------------------------------------------------- /src/js/permissions.js: -------------------------------------------------------------------------------- 1 | (function(global) { 2 | 'use strict'; 3 | 4 | try { 5 | chrome.extension.getBackgroundPage().tgs.setViewGlobals(global); 6 | } catch (e) { 7 | window.setTimeout(() => window.location.reload(), 1000); 8 | return; 9 | } 10 | 11 | gsUtils.documentReadyAndLocalisedAsPromsied(document).then(function() { 12 | document.getElementById('exportBackupBtn').onclick = async function(e) { 13 | const currentSession = await gsSession.buildCurrentSession(); 14 | historyUtils.exportSession(currentSession, function() { 15 | document.getElementById('exportBackupBtn').style.display = 'none'; 16 | }); 17 | }; 18 | document.getElementById('setFilePermissiosnBtn').onclick = async function( 19 | e 20 | ) { 21 | await gsChrome.tabsCreate({ 22 | url: 'about:addons?id=' + browser.extension.getURL ('.').replace('moz-extension:', '').replace('/', ''), 23 | }); 24 | }; 25 | }); 26 | })(this); 27 | -------------------------------------------------------------------------------- /src/js/recovery.js: -------------------------------------------------------------------------------- 1 | /*global chrome, historyItems, gsMessages, gsSession, gsStorage, gsIndexedDb, gsChrome, gsUtils */ 2 | (function(global) { 3 | 'use strict'; 4 | 5 | try { 6 | chrome.extension.getBackgroundPage().tgs.setViewGlobals(global); 7 | } catch (e) { 8 | window.setTimeout(() => window.location.reload(), 1000); 9 | return; 10 | } 11 | 12 | var restoreAttempted = false; 13 | var tabsToRecover = []; 14 | 15 | async function getRecoverableTabs(currentTabs) { 16 | const lastSession = await gsIndexedDb.fetchLastSession(); 17 | //check to see if they still exist in current session 18 | if (lastSession) { 19 | gsUtils.removeInternalUrlsFromSession(lastSession); 20 | for (const window of lastSession.windows) { 21 | for (const tabProperties of window.tabs) { 22 | if (gsUtils.isSuspendedTab(tabProperties)) { 23 | var originalUrl = gsUtils.getOriginalUrl(tabProperties.url); 24 | // Ignore suspended tabs from previous session that exist unsuspended now 25 | const originalTab = currentTabs.find(o => o.url === originalUrl); 26 | if (!originalTab) { 27 | tabProperties.windowId = window.id; 28 | tabProperties.sessionId = lastSession.sessionId; 29 | tabsToRecover.push(tabProperties); 30 | } 31 | } 32 | } 33 | } 34 | return tabsToRecover; 35 | } 36 | } 37 | 38 | function removeTabFromList(tabToRemove) { 39 | const recoveryTabsEl = document.getElementById('recoveryTabs'); 40 | const childLinks = recoveryTabsEl.children; 41 | 42 | for (var i = 0; i < childLinks.length; i++) { 43 | const element = childLinks[i]; 44 | const url = gsUtils.isSuspendedTab(tabToRemove) 45 | ? gsUtils.getOriginalUrl(tabToRemove.url) 46 | : tabToRemove.url; 47 | 48 | if ( 49 | element.getAttribute('data-url') === url || 50 | element.getAttribute('data-tabId') == tabToRemove.id 51 | ) { 52 | // eslint-disable-line eqeqeq 53 | recoveryTabsEl.removeChild(element); 54 | } 55 | } 56 | 57 | //if removing the last element.. (re-get the element this function gets called asynchronously 58 | if (document.getElementById('recoveryTabs').children.length === 0) { 59 | //if we have already clicked the restore button then redirect to success page 60 | if (restoreAttempted) { 61 | document.getElementById('suspendy-guy-inprogress').style.display = 62 | 'none'; 63 | document.getElementById('recovery-inprogress').style.display = 'none'; 64 | document.getElementById('suspendy-guy-complete').style.display = 65 | 'inline-block'; 66 | document.getElementById('recovery-complete').style.display = 67 | 'inline-block'; 68 | 69 | //otherwise we have no tabs to recover so just hide references to recovery 70 | } else { 71 | hideRecoverySection(); 72 | } 73 | } 74 | } 75 | 76 | function showTabSpinners() { 77 | var recoveryTabsEl = document.getElementById('recoveryTabs'), 78 | childLinks = recoveryTabsEl.children; 79 | 80 | for (var i = 0; i < childLinks.length; i++) { 81 | var tabContainerEl = childLinks[i]; 82 | tabContainerEl.removeChild(tabContainerEl.firstChild); 83 | var spinnerEl = document.createElement('span'); 84 | spinnerEl.classList.add('faviconSpinner'); 85 | tabContainerEl.insertBefore(spinnerEl, tabContainerEl.firstChild); 86 | } 87 | } 88 | 89 | function hideRecoverySection() { 90 | var recoverySectionEls = document.getElementsByClassName('recoverySection'); 91 | for (var i = 0; i < recoverySectionEls.length; i++) { 92 | recoverySectionEls[i].style.display = 'none'; 93 | } 94 | document.getElementById('restoreSession').style.display = 'none'; 95 | } 96 | 97 | gsUtils.documentReadyAndLocalisedAsPromsied(document).then(async function() { 98 | var restoreEl = document.getElementById('restoreSession'), 99 | manageEl = document.getElementById('manageManuallyLink'), 100 | previewsEl = document.getElementById('previewsOffBtn'), 101 | recoveryEl = document.getElementById('recoveryTabs'), 102 | warningEl = document.getElementById('screenCaptureNotice'), 103 | tabEl; 104 | 105 | manageEl.onclick = function(e) { 106 | e.preventDefault(); 107 | chrome.tabs.create({ url: chrome.extension.getURL('history.html') }); 108 | }; 109 | 110 | if (previewsEl) { 111 | previewsEl.onclick = function(e) { 112 | gsStorage.setOptionAndSync(gsStorage.SCREEN_CAPTURE, '0'); 113 | window.location.reload(); 114 | }; 115 | 116 | //show warning if screen capturing turned on 117 | if (gsStorage.getOption(gsStorage.SCREEN_CAPTURE) !== '0') { 118 | warningEl.style.display = 'block'; 119 | } 120 | } 121 | 122 | var performRestore = async function() { 123 | restoreAttempted = true; 124 | restoreEl.className += ' btnDisabled'; 125 | restoreEl.removeEventListener('click', performRestore); 126 | showTabSpinners(); 127 | while (gsSession.isInitialising()) { 128 | await gsUtils.setTimeout(200); 129 | } 130 | await gsSession.recoverLostTabs(); 131 | }; 132 | 133 | restoreEl.addEventListener('click', performRestore); 134 | 135 | const currentTabs = await gsChrome.tabsQuery(); 136 | const tabsToRecover = await getRecoverableTabs(currentTabs); 137 | if (tabsToRecover.length === 0) { 138 | hideRecoverySection(); 139 | return; 140 | } 141 | 142 | for (var tabToRecover of tabsToRecover) { 143 | tabToRecover.title = gsUtils.getCleanTabTitle(tabToRecover); 144 | tabToRecover.url = gsUtils.getOriginalUrl(tabToRecover.url); 145 | tabEl = await historyItems.createTabHtml(tabToRecover, false); 146 | tabEl.onclick = function() { 147 | return function(e) { 148 | e.preventDefault(); 149 | chrome.tabs.create({ url: tabToRecover.url, active: false }); 150 | removeTabFromList(tabToRecover); 151 | }; 152 | }; 153 | recoveryEl.appendChild(tabEl); 154 | } 155 | 156 | var currentSuspendedTabs = currentTabs.filter(o => 157 | gsUtils.isSuspendedTab(o) 158 | ); 159 | for (const suspendedTab of currentSuspendedTabs) { 160 | gsMessages.sendPingToTab(suspendedTab.id, function(error) { 161 | if (error) { 162 | gsUtils.warning(suspendedTab.id, 'Failed to sendPingToTab', error); 163 | } else { 164 | removeTabFromList(suspendedTab); 165 | } 166 | }); 167 | } 168 | }); 169 | 170 | global.exports = { 171 | removeTabFromList, 172 | }; 173 | })(this); 174 | -------------------------------------------------------------------------------- /src/js/restoring-window.js: -------------------------------------------------------------------------------- 1 | /*global chrome, gsUtils */ 2 | (function(global) { 3 | 'use strict'; 4 | 5 | try { 6 | chrome.extension.getBackgroundPage().tgs.setViewGlobals(global); 7 | } catch (e) { 8 | window.setTimeout(() => window.location.reload(), 1000); 9 | return; 10 | } 11 | 12 | gsUtils.documentReadyAndLocalisedAsPromsied(document).then(function() { 13 | //do nothing 14 | }); 15 | })(this); 16 | -------------------------------------------------------------------------------- /src/js/shortcuts.js: -------------------------------------------------------------------------------- 1 | /*global chrome, gsUtils */ 2 | (function(global) { 3 | 'use strict'; 4 | 5 | try { 6 | chrome.extension.getBackgroundPage().tgs.setViewGlobals(global); 7 | } catch (e) { 8 | window.setTimeout(() => window.location.reload(), 1000); 9 | return; 10 | } 11 | 12 | gsUtils.documentReadyAndLocalisedAsPromsied(document).then(function() { 13 | var shortcutsEl = document.getElementById('keyboardShortcuts'); 14 | var configureShortcutsEl = document.getElementById('configureShortcuts'); 15 | 16 | var notSetMessage = chrome.i18n.getMessage('js_shortcuts_not_set'); 17 | var groupingKeys = [ 18 | '2-toggle-temp-whitelist-tab', 19 | '2b-unsuspend-selected-tabs', 20 | '4-unsuspend-active-window', 21 | ]; 22 | 23 | //populate keyboard shortcuts 24 | chrome.commands.getAll(commands => { 25 | commands.forEach(command => { 26 | if (command.name !== '_execute_browser_action') { 27 | const shortcut = 28 | (command.shortcut !== '' && command.shortcut !== null) 29 | ? gsUtils.formatHotkeyString(command.shortcut) 30 | : '(' + notSetMessage + ')'; 31 | var addMarginBottom = groupingKeys.includes(command.name); 32 | shortcutsEl.innerHTML += `
${command.description}
35 |
${shortcut}
`; 38 | } 39 | }); 40 | }); 41 | 42 | //listener for configureShortcuts 43 | configureShortcutsEl.onclick = function(e) { 44 | chrome.tabs.update({ url: 'chrome://extensions/shortcuts' }); 45 | }; 46 | }); 47 | 48 | })(this); 49 | -------------------------------------------------------------------------------- /src/js/tests/fixture_currentSessions.json: -------------------------------------------------------------------------------- 1 | { 2 | "currentSession1": { 3 | "sessionId": "111111", 4 | "windows": [ 5 | { 6 | "alwaysOnTop": false, 7 | "focused": true, 8 | "height": 1027, 9 | "id": 3628, 10 | "incognito": false, 11 | "left": 0, 12 | "state": "maximized", 13 | "tabs": [ 14 | { 15 | "active": true, 16 | "audible": false, 17 | "autoDiscardable": true, 18 | "discarded": false, 19 | "favIconUrl": "https://ssl.gstatic.com/ui/v1/icons/mail/images/favicon5.ico", 20 | "height": 953, 21 | "highlighted": true, 22 | "id": 3630, 23 | "incognito": false, 24 | "index": 0, 25 | "mutedInfo": { 26 | "muted": false 27 | }, 28 | "pinned": false, 29 | "selected": true, 30 | "status": "loading", 31 | "title": "Gmail", 32 | "url": "https://mail.google.com/mail/u/0/#inbox", 33 | "width": 1680, 34 | "windowId": 3628 35 | }, 36 | { 37 | "active": false, 38 | "audible": false, 39 | "autoDiscardable": true, 40 | "discarded": false, 41 | "favIconUrl": "", 42 | "height": 914, 43 | "highlighted": false, 44 | "id": 3631, 45 | "incognito": false, 46 | "index": 1, 47 | "mutedInfo": { 48 | "muted": false 49 | }, 50 | "pinned": false, 51 | "selected": false, 52 | "status": "complete", 53 | "title": "Trello", 54 | "url": "chrome-extension://jmhiginckkdplecbbnnfcapnpofhmgbb/suspended.html#ttl=https%3A%2F%2Ftrello.com&pos=0&uri=https://trello.com", 55 | "width": 1680, 56 | "windowId": 3628 57 | }, 58 | { 59 | "active": false, 60 | "audible": false, 61 | "autoDiscardable": true, 62 | "discarded": false, 63 | "favIconUrl": "", 64 | "height": 914, 65 | "highlighted": false, 66 | "id": 3632, 67 | "incognito": false, 68 | "index": 2, 69 | "mutedInfo": { 70 | "muted": false 71 | }, 72 | "pinned": false, 73 | "selected": false, 74 | "status": "complete", 75 | "title": "https://github.com/deanoemcke/thegreatsuspender/issues/266", 76 | "url": "chrome-extension://jmhiginckkdplecbbnnfcapnpofhmgbb/suspended.html#ttl=https%3A%2F%2Fgithub.com%2Fdeanoemcke%2Fthegreatsuspender%2Fissues%2F266&pos=0&uri=https://github.com/deanoemcke/thegreatsuspender/issues/266", 77 | "width": 1680, 78 | "windowId": 3628 79 | }, 80 | { 81 | "active": false, 82 | "audible": false, 83 | "autoDiscardable": true, 84 | "discarded": false, 85 | "favIconUrl": "", 86 | "height": 914, 87 | "highlighted": false, 88 | "id": 3633, 89 | "incognito": false, 90 | "index": 3, 91 | "mutedInfo": { 92 | "muted": false 93 | }, 94 | "pinned": false, 95 | "selected": false, 96 | "status": "complete", 97 | "title": "chrome.i18n - Google Chrome", 98 | "url": "chrome-extension://jmhiginckkdplecbbnnfcapnpofhmgbb/suspended.html#ttl=https%3A%2F%2Fdeveloper.chrome.com%2Fextensions%2Fi18n&pos=0&uri=https://developer.chrome.com/extensions/i18n", 99 | "width": 1680, 100 | "windowId": 3628 101 | }, 102 | { 103 | "active": false, 104 | "audible": false, 105 | "autoDiscardable": true, 106 | "discarded": false, 107 | "favIconUrl": "", 108 | "height": 914, 109 | "highlighted": false, 110 | "id": 3634, 111 | "incognito": false, 112 | "index": 4, 113 | "mutedInfo": { 114 | "muted": false 115 | }, 116 | "pinned": false, 117 | "selected": false, 118 | "status": "complete", 119 | "title": "447856 - Feature Request: Ability for an extension to change the URL in the omnibox - chromium - Monorail", 120 | "url": "chrome-extension://jmhiginckkdplecbbnnfcapnpofhmgbb/suspended.html#ttl=https%3A%2F%2Fbugs.chromium.org%2Fp%2Fchromium%2Fissues%2Fdetail%3Fid%3D447856&pos=0&uri=https://bugs.chromium.org/p/chromium/issues/detail?id=447856", 121 | "width": 1680, 122 | "windowId": 3628 123 | } 124 | ], 125 | "top": 23, 126 | "type": "normal", 127 | "width": 1680 128 | } 129 | ], 130 | "date": "2018-08-29T14:07:04.041Z", 131 | "name": "currentSession1", 132 | "id": 1 133 | } 134 | } 135 | -------------------------------------------------------------------------------- /src/js/tests/fixture_savedSessions.json: -------------------------------------------------------------------------------- 1 | { 2 | "savedSession1": { 3 | "sessionId": "_111111", 4 | "windows": [ 5 | { 6 | "alwaysOnTop": false, 7 | "focused": true, 8 | "height": 1027, 9 | "id": 3628, 10 | "incognito": false, 11 | "left": 0, 12 | "state": "maximized", 13 | "tabs": [ 14 | { 15 | "active": true, 16 | "audible": false, 17 | "autoDiscardable": true, 18 | "discarded": false, 19 | "favIconUrl": "https://ssl.gstatic.com/ui/v1/icons/mail/images/favicon5.ico", 20 | "height": 953, 21 | "highlighted": true, 22 | "id": 3630, 23 | "incognito": false, 24 | "index": 0, 25 | "mutedInfo": { 26 | "muted": false 27 | }, 28 | "pinned": false, 29 | "selected": true, 30 | "status": "loading", 31 | "title": "Gmail", 32 | "url": "https://mail.google.com/mail/u/0/#inbox", 33 | "width": 1680, 34 | "windowId": 3628 35 | }, 36 | { 37 | "active": false, 38 | "audible": false, 39 | "autoDiscardable": true, 40 | "discarded": false, 41 | "favIconUrl": "", 42 | "height": 914, 43 | "highlighted": false, 44 | "id": 3631, 45 | "incognito": false, 46 | "index": 1, 47 | "mutedInfo": { 48 | "muted": false 49 | }, 50 | "pinned": false, 51 | "selected": false, 52 | "status": "complete", 53 | "title": "the great suspender | Trello", 54 | "url": "chrome-extension://jmhiginckkdplecbbnnfcapnpofhmgbb/suspended.html#ttl=https%3A%2F%2Ftrello.com%2Fb%2FCNT09XJg%2Fthe-great-suspender&pos=0&uri=https://trello.com/b/CNT09XJg/the-great-suspender", 55 | "width": 1680, 56 | "windowId": 3628 57 | }, 58 | { 59 | "active": false, 60 | "audible": false, 61 | "autoDiscardable": true, 62 | "discarded": false, 63 | "favIconUrl": "", 64 | "height": 914, 65 | "highlighted": false, 66 | "id": 3632, 67 | "incognito": false, 68 | "index": 2, 69 | "mutedInfo": { 70 | "muted": false 71 | }, 72 | "pinned": false, 73 | "selected": false, 74 | "status": "complete", 75 | "title": "https://github.com/deanoemcke/thegreatsuspender/issues/266", 76 | "url": "chrome-extension://jmhiginckkdplecbbnnfcapnpofhmgbb/suspended.html#ttl=https%3A%2F%2Fgithub.com%2Fdeanoemcke%2Fthegreatsuspender%2Fissues%2F266&pos=0&uri=https://github.com/deanoemcke/thegreatsuspender/issues/266", 77 | "width": 1680, 78 | "windowId": 3628 79 | }, 80 | { 81 | "active": false, 82 | "audible": false, 83 | "autoDiscardable": true, 84 | "discarded": false, 85 | "favIconUrl": "", 86 | "height": 914, 87 | "highlighted": false, 88 | "id": 3633, 89 | "incognito": false, 90 | "index": 3, 91 | "mutedInfo": { 92 | "muted": false 93 | }, 94 | "pinned": false, 95 | "selected": false, 96 | "status": "complete", 97 | "title": "chrome.i18n - Google Chrome", 98 | "url": "chrome-extension://jmhiginckkdplecbbnnfcapnpofhmgbb/suspended.html#ttl=https%3A%2F%2Fdeveloper.chrome.com%2Fextensions%2Fi18n&pos=0&uri=https://developer.chrome.com/extensions/i18n", 99 | "width": 1680, 100 | "windowId": 3628 101 | }, 102 | { 103 | "active": false, 104 | "audible": false, 105 | "autoDiscardable": true, 106 | "discarded": false, 107 | "favIconUrl": "", 108 | "height": 914, 109 | "highlighted": false, 110 | "id": 3634, 111 | "incognito": false, 112 | "index": 4, 113 | "mutedInfo": { 114 | "muted": false 115 | }, 116 | "pinned": false, 117 | "selected": false, 118 | "status": "complete", 119 | "title": "447856 - Feature Request: Ability for an extension to change the URL in the omnibox - chromium - Monorail", 120 | "url": "chrome-extension://jmhiginckkdplecbbnnfcapnpofhmgbb/suspended.html#ttl=https%3A%2F%2Fbugs.chromium.org%2Fp%2Fchromium%2Fissues%2Fdetail%3Fid%3D447856&pos=0&uri=https://bugs.chromium.org/p/chromium/issues/detail?id=447856", 121 | "width": 1680, 122 | "windowId": 3628 123 | } 124 | ], 125 | "top": 23, 126 | "type": "normal", 127 | "width": 1680 128 | } 129 | ], 130 | "date": "2018-08-29T14:07:04.041Z", 131 | "name": "savedSession1", 132 | "id": 33 133 | } 134 | } 135 | -------------------------------------------------------------------------------- /src/js/tests/test_createAndUpdateSessionRestorePoint.js: -------------------------------------------------------------------------------- 1 | /*global chrome, gsIndexedDb, gsSession, getFixture, loadJsFile, assertTrue, FIXTURE_CURRENT_SESSIONS */ 2 | var testSuites = typeof testSuites === 'undefined' ? [] : testSuites; 3 | testSuites.push( 4 | (function() { 5 | 'use strict'; 6 | 7 | const oldVersion = '1.2.34'; 8 | 9 | const tests = [ 10 | // Test create session restore point. Should create a session from the currently open windows 11 | async () => { 12 | // Simulate gsSession.prepareForUpdate 13 | const session1 = await getFixture( 14 | FIXTURE_CURRENT_SESSIONS, 15 | 'currentSession1' 16 | ); 17 | const currentSession = await gsSession.buildCurrentSession(); 18 | currentSession.windows = session1.windows; 19 | const sessionRestorePointAfter = await gsIndexedDb.createOrUpdateSessionRestorePoint( 20 | currentSession, 21 | oldVersion 22 | ); 23 | const isSessionRestorePointValid = 24 | sessionRestorePointAfter.windows[0].tabs.length === 5; 25 | return assertTrue(isSessionRestorePointValid); 26 | }, 27 | 28 | // Test create session restore point when session restore point already exists from same session 29 | async () => { 30 | // Create a session restore point 31 | const session1 = await getFixture( 32 | FIXTURE_CURRENT_SESSIONS, 33 | 'currentSession1' 34 | ); 35 | const currentSession1 = await gsSession.buildCurrentSession(); 36 | currentSession1.windows = session1.windows; 37 | await gsIndexedDb.createOrUpdateSessionRestorePoint( 38 | currentSession1, 39 | oldVersion 40 | ); 41 | const newSessionRestorePointBefore = await gsIndexedDb.fetchSessionBySessionId( 42 | currentSession1.sessionId 43 | ); 44 | const isSessionRestorePointBeforeValid = 45 | newSessionRestorePointBefore.windows[0].tabs.length === 5; 46 | 47 | const session2 = await getFixture( 48 | FIXTURE_CURRENT_SESSIONS, 49 | 'currentSession1' 50 | ); 51 | const currentSession2 = await gsSession.buildCurrentSession(); 52 | currentSession2.windows = session2.windows; 53 | currentSession2.windows[0].tabs.push({ 54 | id: 7777, 55 | title: 'testTab', 56 | url: 'https://test.com', 57 | }); 58 | const sessionRestorePointAfter = await gsIndexedDb.createOrUpdateSessionRestorePoint( 59 | currentSession2, 60 | oldVersion 61 | ); 62 | const sessionRestorePointAfterValid = 63 | sessionRestorePointAfter.windows[0].tabs.length === 6; 64 | 65 | const gsTestDb = await gsIndexedDb.getDb(); 66 | const sessionRestoreCount = await gsTestDb 67 | .query(gsIndexedDb.DB_SAVED_SESSIONS) 68 | .filter(gsIndexedDb.DB_SESSION_PRE_UPGRADE_KEY, oldVersion) 69 | .execute() 70 | .then(o => o.length); 71 | 72 | return assertTrue( 73 | isSessionRestorePointBeforeValid && 74 | sessionRestorePointAfterValid && 75 | sessionRestoreCount === 1 76 | ); 77 | }, 78 | 79 | // Test create session restore point when session restore point already exists from another session 80 | async () => { 81 | // Create a session restore point (uses current session based on gsSession.getSessionId) 82 | const session1 = await getFixture( 83 | FIXTURE_CURRENT_SESSIONS, 84 | 'currentSession1' 85 | ); 86 | const currentSession1 = await gsSession.buildCurrentSession(); 87 | currentSession1.windows = session1.windows; 88 | const oldCurrentSessionId = currentSession1.sessionId; 89 | const sessionRestorePointBefore = await gsIndexedDb.createOrUpdateSessionRestorePoint( 90 | currentSession1, 91 | oldVersion 92 | ); 93 | const isSessionRestorePointBeforeValid = 94 | sessionRestorePointBefore.windows[0].tabs.length === 5; 95 | 96 | // Simulate an extension restart by resetting gsSession sessionId and saving a new 'current session' 97 | await loadJsFile('gsSession'); 98 | const session2 = await getFixture( 99 | FIXTURE_CURRENT_SESSIONS, 100 | 'currentSession1' 101 | ); 102 | const currentSession2 = await gsSession.buildCurrentSession(); 103 | const newCurrentSessionId = currentSession2.sessionId; 104 | const isCurrentSessionIdChanged = 105 | oldCurrentSessionId !== newCurrentSessionId; 106 | 107 | currentSession2.windows = session2.windows; 108 | currentSession2.windows[0].tabs.push({ 109 | id: 7777, 110 | title: 'testTab', 111 | url: 'https://test.com', 112 | }); 113 | const sessionRestorePointAfter = await gsIndexedDb.createOrUpdateSessionRestorePoint( 114 | currentSession2, 115 | oldVersion 116 | ); 117 | const sessionRestorePointAfterValid = 118 | sessionRestorePointAfter.windows[0].tabs.length === 6; 119 | 120 | const gsTestDb = await gsIndexedDb.getDb(); 121 | const sessionRestoreCount = await gsTestDb 122 | .query(gsIndexedDb.DB_SAVED_SESSIONS) 123 | .filter(gsIndexedDb.DB_SESSION_PRE_UPGRADE_KEY, oldVersion) 124 | .execute() 125 | .then(o => o.length); 126 | 127 | return assertTrue( 128 | isSessionRestorePointBeforeValid && 129 | isCurrentSessionIdChanged && 130 | sessionRestorePointAfterValid && 131 | sessionRestoreCount === 1 132 | ); 133 | }, 134 | ]; 135 | 136 | return { 137 | name: 'Session Restore Points', 138 | tests, 139 | }; 140 | })() 141 | ); 142 | -------------------------------------------------------------------------------- /src/js/tests/test_currentSessions.js: -------------------------------------------------------------------------------- 1 | /*global chrome, gsIndexedDb, gsSession, getFixture, assertTrue, FIXTURE_CURRENT_SESSIONS */ 2 | var testSuites = typeof testSuites === 'undefined' ? [] : testSuites; 3 | testSuites.push( 4 | (function() { 5 | 'use strict'; 6 | 7 | const tests = [ 8 | // Test saving new currentSession 9 | async () => { 10 | const currentSessionId = gsSession.getSessionId(); 11 | const currentSessionsBefore = await gsIndexedDb.fetchCurrentSessions(); 12 | const wasCurrentSessionsEmpty = currentSessionsBefore.length === 0; 13 | 14 | // Simulate gsSession.updateCurrentSession() 15 | const session1 = await getFixture( 16 | FIXTURE_CURRENT_SESSIONS, 17 | 'currentSession1' 18 | ); 19 | const currentSession = await gsSession.buildCurrentSession(); 20 | currentSession.windows = session1.windows; 21 | await gsIndexedDb.updateSession(currentSession); 22 | const savedCurrentSession = await gsIndexedDb.fetchSessionBySessionId( 23 | currentSessionId 24 | ); 25 | 26 | const isSessionValid = 27 | savedCurrentSession.sessionId === currentSessionId && 28 | savedCurrentSession.windows.length === 1 && 29 | savedCurrentSession.windows[0].tabs.length === 5 && 30 | savedCurrentSession.windows[0].tabs[0].id === 3630; 31 | 32 | const currentSessionsAfter = await gsIndexedDb.fetchCurrentSessions(); 33 | const isCurrentSessionsPopulated = currentSessionsAfter.length === 1; 34 | 35 | return assertTrue( 36 | wasCurrentSessionsEmpty && 37 | isCurrentSessionsPopulated && 38 | isSessionValid 39 | ); 40 | }, 41 | 42 | // Test updating existing currentSession 43 | async () => { 44 | const currentSessionId = gsSession.getSessionId(); 45 | const session1 = await getFixture( 46 | FIXTURE_CURRENT_SESSIONS, 47 | 'currentSession1' 48 | ); 49 | const currentSession1 = await gsSession.buildCurrentSession(); 50 | currentSession1.windows = session1.windows; 51 | await gsIndexedDb.updateSession(currentSession1); 52 | const dbCurrentSession1 = await gsIndexedDb.fetchSessionBySessionId( 53 | currentSessionId 54 | ); 55 | const isSession1Valid = 56 | dbCurrentSession1.sessionId === currentSessionId && 57 | dbCurrentSession1.windows[0].tabs.length === 5; 58 | 59 | const session2 = await getFixture( 60 | FIXTURE_CURRENT_SESSIONS, 61 | 'currentSession1' 62 | ); 63 | const currentSession2 = await gsSession.buildCurrentSession(); 64 | currentSession2.windows = session2.windows; 65 | currentSession2.windows[0].tabs.push({ 66 | id: 7777, 67 | title: 'testTab', 68 | url: 'https://test.com', 69 | }); 70 | await gsIndexedDb.updateSession(currentSession2); 71 | 72 | const dbCurrentSession2 = await gsIndexedDb.fetchSessionBySessionId( 73 | currentSessionId 74 | ); 75 | const isSession2Valid = 76 | dbCurrentSession2.sessionId === currentSessionId && 77 | dbCurrentSession2.windows.length === 1 && 78 | dbCurrentSession2.windows[0].tabs.length === 6 && 79 | dbCurrentSession2.windows[0].tabs[5].id === 7777 && 80 | dbCurrentSession1.id === dbCurrentSession2.id && 81 | dbCurrentSession1.date < dbCurrentSession2.date; 82 | 83 | const currentSessionsAfter = await gsIndexedDb.fetchCurrentSessions(); 84 | const isCurrentSessionsAfterValid = currentSessionsAfter.length === 1; 85 | 86 | return assertTrue( 87 | isSession1Valid && isSession2Valid && isCurrentSessionsAfterValid 88 | ); 89 | }, 90 | ]; 91 | 92 | return { 93 | name: 'Current Sessions', 94 | tests, 95 | }; 96 | })() 97 | ); 98 | -------------------------------------------------------------------------------- /src/js/tests/test_gsTabQueue.js: -------------------------------------------------------------------------------- 1 | /*global chrome, GsTabQueue, gsUtils, assertTrue */ 2 | var testSuites = typeof testSuites === 'undefined' ? [] : testSuites; 3 | testSuites.push( 4 | (function() { 5 | 'use strict'; 6 | 7 | const MAX_REQUEUES = 2; 8 | 9 | function buildExecutorResolveTrue(executorDelay) { 10 | return async (tab, executionProps, resolve, reject, requeue) => { 11 | await gsUtils.setTimeout(executorDelay); 12 | resolve(true); 13 | }; 14 | } 15 | 16 | function buildExecutorRequeue(executorDelay, requeueDelay) { 17 | return async (tab, executionProps, resolve, reject, requeue) => { 18 | executionProps.requeues = executionProps.requeues || 0; 19 | await gsUtils.setTimeout(executorDelay); 20 | if (executionProps.requeues !== MAX_REQUEUES) { 21 | executionProps.requeues += 1; 22 | requeue(requeueDelay); 23 | } else { 24 | resolve(true); 25 | } 26 | }; 27 | } 28 | 29 | function buildExceptionResolvesFalse() { 30 | return (tab, executionProps, exceptionType, resolve, reject, requeue) => { 31 | resolve(false); 32 | }; 33 | } 34 | function buildExceptionRejects() { 35 | return (tab, executionProps, exceptionType, resolve, reject, requeue) => { 36 | reject('Test error'); 37 | }; 38 | } 39 | 40 | async function runQueueTest(tabCount, gsTabQueue) { 41 | const tabCheckPromises = []; 42 | for (let tabId = 1; tabId <= tabCount; tabId += 1) { 43 | const tabCheckPromise = gsTabQueue.queueTabAsPromise({ 44 | id: tabId, 45 | }); 46 | tabCheckPromises.push(tabCheckPromise); 47 | } 48 | 49 | let results; 50 | try { 51 | results = await Promise.all(tabCheckPromises); 52 | } catch (e) { 53 | console.log('Error!', e); 54 | } 55 | 56 | // Wait for queue to finish 57 | while (gsTabQueue.getTotalQueueSize() > 0) { 58 | await gsUtils.setTimeout(10); 59 | } 60 | return results; 61 | } 62 | 63 | async function makeTest( 64 | tabCount, 65 | executorDelay, 66 | requeueDelay, 67 | concurrentExecutors, 68 | jobTimeout, 69 | executorFnType, 70 | exceptionFnType, 71 | shouldGenerateException, 72 | expectedTimeTaken 73 | ) { 74 | let executorFn; 75 | if (executorFnType === 'resolveTrue') { 76 | executorFn = buildExecutorResolveTrue(executorDelay); 77 | } else if (executorFnType === 'requeue') { 78 | executorFn = buildExecutorRequeue(executorDelay, requeueDelay); 79 | } 80 | let exceptionFn; 81 | if (exceptionFnType === 'resolveFalse') { 82 | exceptionFn = buildExceptionResolvesFalse(); 83 | } else if (exceptionFnType === 'reject') { 84 | exceptionFn = buildExceptionRejects(); 85 | } 86 | const queueProps = { 87 | concurrentExecutors, 88 | jobTimeout, 89 | executorFn, 90 | exceptionFn, 91 | processingDelay: 0, 92 | }; 93 | 94 | const startTime = Date.now(); 95 | const gsTabQueue = GsTabQueue('testQueue', queueProps); 96 | const results = await runQueueTest(tabCount, gsTabQueue); 97 | const timeTaken = Date.now() - startTime; 98 | console.log( 99 | `timers. timeTaken: ${timeTaken}. expected: ${expectedTimeTaken}` 100 | ); 101 | 102 | let isResultsValid = false; 103 | if (shouldGenerateException && exceptionFnType === 'resolveFalse') { 104 | isResultsValid = 105 | results.length === tabCount && results.every(o => o === false); 106 | } else if (shouldGenerateException && exceptionFnType === 'reject') { 107 | isResultsValid = typeof results === 'undefined'; 108 | } else { 109 | isResultsValid = 110 | results.length === tabCount && results.every(o => o === true); 111 | } 112 | 113 | // Nasty hack here 114 | const allowedTimingVariation = 150; 115 | 116 | let isTimeValid = 117 | timeTaken > expectedTimeTaken && 118 | timeTaken < expectedTimeTaken + allowedTimingVariation; 119 | 120 | return assertTrue(isResultsValid && isTimeValid); 121 | } 122 | 123 | const tests = [ 124 | async () => { 125 | // Test: 5 tabs. 100ms per tab. No requeue delay. 1 at a time. 1000ms timeout. 126 | // Executor function resolvesTrue. Exception function resolvesFalse. 127 | // Should resolveTrue. 128 | // Expected time taken: 5 * 100 + 5 * 50 129 | return await makeTest( 130 | 5, 131 | 100, 132 | 0, 133 | 1, 134 | 1000, 135 | 'resolveTrue', 136 | 'resolveFalse', 137 | false, 138 | 750 139 | ); 140 | }, 141 | 142 | async () => { 143 | // Test: 5 tabs. 100ms per tab. No requeue delay. 2 at a time. 1000ms timeout. 144 | // Executor function resolvesTrue. Exception function resolvesFalse. 145 | // Should resolveTrue. 146 | // Expected time taken: 3 * 100 + 3 * 50 147 | return await makeTest( 148 | 5, 149 | 100, 150 | 0, 151 | 2, 152 | 1000, 153 | 'resolveTrue', 154 | 'resolveFalse', 155 | false, 156 | 450 157 | ); 158 | }, 159 | 160 | async () => { 161 | // Test: 5 tabs. 100ms per tab. No requeue delay. 50 at a time. 1000ms timeout. 162 | // Executor function resolvesTrue. Exception function resolvesFalse. 163 | // Should resolveTrue. 164 | // Expected time taken: 1 * 100 + 1 * 50 165 | return await makeTest( 166 | 5, 167 | 100, 168 | 0, 169 | 50, 170 | 1000, 171 | 'resolveTrue', 172 | 'resolveFalse', 173 | false, 174 | 150 175 | ); 176 | }, 177 | 178 | async () => { 179 | // Test: 50 tabs. 100ms per tab. No requeue delay. 20 at a time. 1000ms timeout. 180 | // Executor function resolvesTrue. Exception function resolvesFalse. 181 | // Should resolveTrue. 182 | // Expected time taken: 3 * 100 + 3 * 50 183 | return await makeTest( 184 | 50, 185 | 100, 186 | 0, 187 | 20, 188 | 1000, 189 | 'resolveTrue', 190 | 'resolveFalse', 191 | false, 192 | 450 193 | ); 194 | }, 195 | 196 | async () => { 197 | // Test: 5 tabs. 100ms per tab. No requeue delay. 1 at a time. 10ms timeout. 198 | // Executor function resolvesTrue. Exception function resolvesFalse. 199 | // Should timeout on each execution. 200 | // Should resolveFalse. 201 | // Expected time taken: 5 * 10 + 5 * 50 202 | return await makeTest( 203 | 5, 204 | 100, 205 | 0, 206 | 1, 207 | 10, 208 | 'resolveTrue', 209 | 'resolveFalse', 210 | true, 211 | 300 212 | ); 213 | }, 214 | 215 | async () => { 216 | // Test: 5 tabs. 100ms per tab. No requeue delay. 1 at a time. 10ms timeout. 217 | // Executor function resolvesTrue. Exception function rejects. 218 | // Results should be undefined as Promises.all rejects. 219 | // Should reject. 220 | // Expected time taken: 5 * 10 + 5 * 50 221 | return await makeTest(5, 100, 0, 1, 10, 'resolveTrue', 'reject', true, 300); 222 | }, 223 | 224 | async () => { 225 | // Test: 1 tab. 100ms per tab. 100ms requeue delay, 1 at a time. 1000ms timeout. 226 | // Executor function requeues (up to 2 times). 227 | // Exception function resolvesFalse. 228 | // Should requeue 2 times then resolveTrue. 229 | // Expected time taken: 3 * 1 * 100 + 3 * 100 + 1 * 50 230 | return await makeTest( 231 | 1, 232 | 100, 233 | 100, 234 | 1, 235 | 1000, 236 | 'requeue', 237 | 'resolveFalse', 238 | false, 239 | 650 240 | ); 241 | }, 242 | 243 | async () => { 244 | // Test: 1 tab. 100ms per tab. 100ms requeue delay, 1 at a time. 250ms timeout. 245 | // Executor function requeues (up to 2 times). 246 | // Exception function resolvesFalse. 247 | // Should requeue up to 2 times then timeout. 248 | // Expected time taken: 1 * 250 + 1 * 50 249 | return await makeTest( 250 | 1, 251 | 100, 252 | 100, 253 | 1, 254 | 250, 255 | 'requeue', 256 | 'resolveFalse', 257 | true, 258 | 300 259 | ); 260 | }, 261 | 262 | async () => { 263 | // Test: 1 tab. 100ms per tab. 100ms requeue delay. 1 at a time. 250ms timeout. 264 | // Executor function requeues (up to 2 times). 265 | // Exception function rejects. 266 | // Should requeue up to 2 times then timeout. 267 | // Expected time taken: 1 * 250 + 1 * 50 268 | return await makeTest(1, 100, 100, 1, 250, 'requeue', 'reject', true, 300); 269 | }, 270 | ]; 271 | 272 | return { 273 | name: 'gsTabQueue Library', 274 | tests, 275 | }; 276 | })() 277 | ); 278 | -------------------------------------------------------------------------------- /src/js/tests/test_gsUtils.js: -------------------------------------------------------------------------------- 1 | /*global chrome, gsUtils, assertTrue */ 2 | var testSuites = typeof testSuites === 'undefined' ? [] : testSuites; 3 | testSuites.push( 4 | (function() { 5 | 'use strict'; 6 | 7 | const tests = [ 8 | // Test gsUtils.setTimeout 9 | async () => { 10 | const timeout = 500; 11 | const timeBefore = new Date().getTime(); 12 | await gsUtils.setTimeout(timeout); 13 | const timeAfter = new Date().getTime(); 14 | const isTimeAfterValid = 15 | timeAfter > timeBefore + timeout && 16 | timeAfter < timeBefore + timeout + 200; 17 | 18 | return assertTrue(isTimeAfterValid); 19 | }, 20 | 21 | // Test gsUtils.getRootUrl 22 | async () => { 23 | const rawUrl1 = 'https://searchengine.site'; 24 | const isUrl1Valid = 25 | gsUtils.getRootUrl(rawUrl1, false, false) === 'searchengine.site' && 26 | gsUtils.getRootUrl(rawUrl1, true, false) === 'searchengine.site' && 27 | gsUtils.getRootUrl(rawUrl1, false, true) === 'https://searchengine.site' && 28 | gsUtils.getRootUrl(rawUrl1, true, true) === 'https://searchengine.site'; 29 | 30 | const rawUrl2 = 'https://searchengine.site/'; 31 | const isUrl2Valid = 32 | gsUtils.getRootUrl(rawUrl2, false, false) === 'searchengine.site' && 33 | gsUtils.getRootUrl(rawUrl2, true, false) === 'searchengine.site' && 34 | gsUtils.getRootUrl(rawUrl2, false, true) === 'https://searchengine.site' && 35 | gsUtils.getRootUrl(rawUrl2, true, true) === 'https://searchengine.site'; 36 | 37 | const rawUrl3 = 38 | 'https://searchengine.site/search?source=hp&ei=T2HWW9jfGoWJ5wKzt4WgBw&q=rabbits&oq=rabbits&gs_l=psy-ab.3..35i39k1l2j0i67k1l6j0i10k1j0i67k1.1353.2316.0.2448.9.7.0.0.0.0.120.704.4j3.7.0....0...1c.1.64.psy-ab..2.7.701.0..0.0.dk-gx_j1MUI'; 39 | const isUrl3Valid = 40 | gsUtils.getRootUrl(rawUrl3, false, false) === 'searchengine.site' && 41 | gsUtils.getRootUrl(rawUrl3, true, false) === 'searchengine.site/search' && 42 | gsUtils.getRootUrl(rawUrl3, false, true) === 'https://searchengine.site' && 43 | gsUtils.getRootUrl(rawUrl3, true, true) === 44 | 'https://searchengine.site/search'; 45 | 46 | const rawUrl4 = 'www.searchengine.site'; 47 | const isUrl4Valid = 48 | gsUtils.getRootUrl(rawUrl4, false, false) === 'www.searchengine.site' && 49 | gsUtils.getRootUrl(rawUrl4, true, false) === 'www.searchengine.site' && 50 | gsUtils.getRootUrl(rawUrl4, false, true) === 'www.searchengine.site' && 51 | gsUtils.getRootUrl(rawUrl4, true, true) === 'www.searchengine.site'; 52 | 53 | const rawUrl5 = 54 | 'https://github.com/torvalds/linux/issues/478#issuecomment-430780678'; 55 | const isUrl5Valid = 56 | gsUtils.getRootUrl(rawUrl5, false, false) === 'github.com' && 57 | gsUtils.getRootUrl(rawUrl5, true, false) === 58 | 'github.com/torvalds/linux/issues/478' && 59 | gsUtils.getRootUrl(rawUrl5, false, true) === 'https://github.com' && 60 | gsUtils.getRootUrl(rawUrl5, true, true) === 61 | 'https://github.com/torvalds/linux/issues/478'; 62 | 63 | const rawUrl6 = 'file:///Users/username/Downloads/session%20(63).txt'; 64 | const isUrl6Valid = 65 | gsUtils.getRootUrl(rawUrl6, false, false) === 66 | '/Users/username/Downloads' && 67 | gsUtils.getRootUrl(rawUrl6, true, false) === 68 | '/Users/username/Downloads/session%20(63).txt' && 69 | gsUtils.getRootUrl(rawUrl6, false, true) === 70 | 'file:///Users/username/Downloads' && 71 | gsUtils.getRootUrl(rawUrl6, true, true) === 72 | 'file:///Users/username/Downloads/session%20(63).txt'; 73 | 74 | const rawUrl7 = 75 | 'https://subdomain.domain.org/serverdir/web/#/report-home/a52338347w84781065p87884368'; 76 | const isUrl7Valid = 77 | gsUtils.getRootUrl(rawUrl7, false, false) === 78 | 'subdomain.domain.org' && 79 | gsUtils.getRootUrl(rawUrl7, true, false) === 80 | 'subdomain.domain.org/serverdir/web' && 81 | gsUtils.getRootUrl(rawUrl7, false, true) === 82 | 'https://subdomain.domain.org' && 83 | gsUtils.getRootUrl(rawUrl7, true, true) === 84 | 'https://subdomain.domain.org/serverdir/web'; 85 | 86 | return assertTrue( 87 | isUrl1Valid && 88 | isUrl2Valid && 89 | isUrl3Valid && 90 | isUrl4Valid && 91 | isUrl5Valid && 92 | isUrl6Valid && 93 | isUrl7Valid 94 | ); 95 | }, 96 | 97 | // Test gsUtils.executeWithRetries 98 | async () => { 99 | const successPromiseFn = val => new Promise((r, j) => r(val)); 100 | let result1; 101 | const timeBefore1 = new Date().getTime(); 102 | try { 103 | result1 = await gsUtils.executeWithRetries( 104 | successPromiseFn, 105 | 'a', 106 | 3, 107 | 500 108 | ); 109 | } catch (e) { 110 | // do nothing 111 | } 112 | const timeAfter1 = new Date().getTime(); 113 | const timeTaken1 = timeAfter1 - timeBefore1; 114 | const isTime1Valid = timeTaken1 >= 0 && timeTaken1 < 100; 115 | const isResult1Valid = result1 === 'a'; 116 | 117 | const errorPromiseFn = val => new Promise((r, j) => j()); 118 | let result2; 119 | const timeBefore2 = new Date().getTime(); 120 | try { 121 | result2 = await gsUtils.executeWithRetries( 122 | errorPromiseFn, 123 | 'b', 124 | 3, 125 | 500 126 | ); 127 | } catch (e) { 128 | // do nothing 129 | } 130 | const timeAfter2 = new Date().getTime(); 131 | const timeTaken2 = timeAfter2 - timeBefore2; 132 | const isTime2Valid = timeTaken2 >= 1500 && timeTaken2 < 1600; 133 | const isResult2Valid = result2 === undefined; 134 | 135 | return assertTrue( 136 | isResult1Valid && isTime1Valid && isResult2Valid && isTime2Valid 137 | ); 138 | }, 139 | ]; 140 | 141 | return { 142 | name: 'gsUtils Library', 143 | tests, 144 | }; 145 | })() 146 | ); 147 | -------------------------------------------------------------------------------- /src/js/tests/test_savedSessions.js: -------------------------------------------------------------------------------- 1 | /*global chrome, gsIndexedDb, getFixture, assertTrue, FIXTURE_SAVED_SESSIONS */ 2 | var testSuites = typeof testSuites === 'undefined' ? [] : testSuites; 3 | testSuites.push( 4 | (function() { 5 | 'use strict'; 6 | 7 | const tests = [ 8 | // Test saving new savedSession 9 | async () => { 10 | const currentSessionsBefore = await gsIndexedDb.fetchCurrentSessions(); 11 | const wasCurrentSessionsEmpty = currentSessionsBefore.length === 0; 12 | const savedSessionsBefore = await gsIndexedDb.fetchSavedSessions(); 13 | const wasSavedSessionsEmpty = savedSessionsBefore.length === 0; 14 | 15 | const session1 = await getFixture( 16 | FIXTURE_SAVED_SESSIONS, 17 | 'savedSession1' 18 | ); 19 | await gsIndexedDb.updateSession(session1); 20 | const dbSavedSession1 = await gsIndexedDb.fetchSessionBySessionId( 21 | session1.sessionId 22 | ); 23 | const isSessionValid = 24 | dbSavedSession1.id === session1.id && 25 | dbSavedSession1.sessionId === session1.sessionId && 26 | dbSavedSession1.sessionId.indexOf('_') === 0 && 27 | dbSavedSession1.windows.length === 1 && 28 | dbSavedSession1.windows[0].tabs.length === 5 && 29 | dbSavedSession1.windows[0].tabs[0].id === 3630; 30 | 31 | const currentSessionsAfter = await gsIndexedDb.fetchCurrentSessions(); 32 | const isCurrentSessionsEmpty = currentSessionsAfter.length === 0; 33 | 34 | const savedSessionsAfter = await gsIndexedDb.fetchSavedSessions(); 35 | const isSavedSessionsPopulated = savedSessionsAfter.length === 1; 36 | 37 | return assertTrue( 38 | wasCurrentSessionsEmpty && 39 | wasSavedSessionsEmpty && 40 | isCurrentSessionsEmpty && 41 | isSavedSessionsPopulated && 42 | isSessionValid 43 | ); 44 | }, 45 | 46 | // Test removing savedSession 47 | async () => { 48 | const session1 = await getFixture( 49 | FIXTURE_SAVED_SESSIONS, 50 | 'savedSession1' 51 | ); 52 | await gsIndexedDb.updateSession(session1); 53 | const savedSessionsBefore = await gsIndexedDb.fetchSavedSessions(); 54 | const isSavedSessionsBeforeValid = savedSessionsBefore.length === 1; 55 | 56 | await gsIndexedDb.removeSessionFromHistory( 57 | savedSessionsBefore[0].sessionId 58 | ); 59 | 60 | const savedSessionsAfter = await gsIndexedDb.fetchSavedSessions(); 61 | const isSavedSessionsAfterValid = savedSessionsAfter.length === 0; 62 | 63 | return assertTrue( 64 | isSavedSessionsBeforeValid && isSavedSessionsAfterValid 65 | ); 66 | }, 67 | 68 | // Test saving a lot of large sessions 69 | // async () => { 70 | // const largeSessionTemplate = getFixture(FIXTURE_SAVED_SESSIONS, 'savedSession1'); 71 | // delete largeSessionTemplate.id; 72 | // const tabsTemplate = JSON.parse( 73 | // JSON.stringify(largeSessionTemplate.windows[0].tabs) 74 | // ); 75 | // for (let i = 0; i < 500; i++) { 76 | // largeSessionTemplate.windows[0].tabs = largeSessionTemplate.windows[0].tabs.concat( 77 | // JSON.parse(JSON.stringify(tabsTemplate)) 78 | // ); 79 | // } 80 | // for (let j = 0; j < 50; j++) { 81 | // const largeSession = JSON.parse(JSON.stringify(largeSessionTemplate)); 82 | // largeSession.sessionId = '_' + j; 83 | // const dbSession = await gsStorage.updateSession(largeSession); 84 | // } 85 | // 86 | // const savedSessionsAfter = await gsStorage.fetchSavedSessions(); 87 | // const isSavedSessionsPopulated = savedSessionsAfter.length === 101; 88 | // 89 | // return assertTrue( 90 | // isSavedSessionsPopulated 91 | // ); 92 | // }, 93 | ]; 94 | 95 | return { 96 | name: 'Saved Sessions', 97 | tests, 98 | }; 99 | })() 100 | ); 101 | -------------------------------------------------------------------------------- /src/js/tests/test_suspendTab.js: -------------------------------------------------------------------------------- 1 | /*global chrome, gsIndexedDb, gsTabSuspendManager, getFixture, assertTrue, FIXTURE_CURRENT_SESSIONS, FIXTURE_PREVIEW_URLS */ 2 | var testSuites = typeof testSuites === 'undefined' ? [] : testSuites; 3 | testSuites.push( 4 | (function() { 5 | 'use strict'; 6 | 7 | const tests = [ 8 | // Test functions associated with suspending a tab 9 | async () => { 10 | const session1 = await getFixture( 11 | FIXTURE_CURRENT_SESSIONS, 12 | 'currentSession1' 13 | ); 14 | const tab = session1.windows[0].tabs[0]; 15 | const previewUrl = await getFixture( 16 | FIXTURE_PREVIEW_URLS, 17 | 'previewUrl1' 18 | ); 19 | 20 | await gsTabSuspendManager.saveSuspendData(tab); 21 | const tabProperties = await gsIndexedDb.fetchTabInfo(tab.url); 22 | const isTabPropertiesValid = 23 | tabProperties.url === tab.url && 24 | tabProperties.title === tab.title && 25 | tabProperties.favIconUrl === tab.favIconUrl; 26 | 27 | await gsIndexedDb.addPreviewImage(tab.url, previewUrl); 28 | const preview = await gsIndexedDb.fetchPreviewImage(tab.url); 29 | const isPreviewValid = preview.img === previewUrl; 30 | 31 | return assertTrue(isTabPropertiesValid && isPreviewValid); 32 | }, 33 | ]; 34 | 35 | return { 36 | name: 'Suspend Tab', 37 | tests, 38 | }; 39 | })() 40 | ); 41 | -------------------------------------------------------------------------------- /src/js/tests/test_trimDbItems.js: -------------------------------------------------------------------------------- 1 | /*global chrome, gsIndexedDb, gsSession, getFixture, assertTrue, FIXTURE_CURRENT_SESSIONS */ 2 | var testSuites = typeof testSuites === 'undefined' ? [] : testSuites; 3 | testSuites.push( 4 | (function() { 5 | 'use strict'; 6 | 7 | const tests = [ 8 | // Test trim currentSessions 9 | async () => { 10 | const currentSessionId = gsSession.getSessionId(); 11 | // Simulate adding 10 older sessions in DB_CURRENT_SESSIONS 12 | for (let i = 10; i > 0; i--) { 13 | const oldSession = await getFixture( 14 | FIXTURE_CURRENT_SESSIONS, 15 | 'currentSession1' 16 | ); 17 | delete oldSession.id; 18 | oldSession.sessionId = i + ''; 19 | const previousDateInMs = Date.now() - 1000 * 60 * 60 * i; 20 | oldSession.date = new Date(previousDateInMs).toISOString(); 21 | await gsIndexedDb.updateSession(oldSession); 22 | } 23 | 24 | // Add a current session 25 | const session1 = await getFixture( 26 | FIXTURE_CURRENT_SESSIONS, 27 | 'currentSession1' 28 | ); 29 | const currentSession1 = await gsSession.buildCurrentSession(); 30 | currentSession1.windows = session1.windows; 31 | await gsIndexedDb.updateSession(currentSession1); 32 | 33 | const currentSessionsBefore = await gsIndexedDb.fetchCurrentSessions(); 34 | const areCurrentSessionsBeforeValid = 35 | currentSessionsBefore.length === 11; 36 | 37 | const lastSessionBefore = await gsIndexedDb.fetchLastSession(); 38 | const isLastSessionBeforeValid = lastSessionBefore.sessionId === '1'; 39 | 40 | await gsIndexedDb.trimDbItems(); 41 | 42 | // Ensure current session still exists 43 | const currentSession = await gsIndexedDb.fetchSessionBySessionId( 44 | currentSessionId 45 | ); 46 | const isCurrentSessionValid = currentSession !== null; 47 | 48 | // Ensure correct DB_CURRENT_SESSIONS items were trimmed 49 | const currentSessionsAfter = await gsIndexedDb.fetchCurrentSessions(); 50 | const areCurrentSessionsAfterValid = currentSessionsAfter.length === 5; 51 | 52 | // Ensure fetchLastSession returns correct session 53 | const lastSessionAfter = await gsIndexedDb.fetchLastSession(); 54 | const isLastSessionAfterValid = lastSessionAfter.sessionId === '1'; 55 | 56 | return assertTrue( 57 | areCurrentSessionsBeforeValid && 58 | isLastSessionBeforeValid && 59 | isCurrentSessionValid && 60 | areCurrentSessionsAfterValid && 61 | isLastSessionAfterValid 62 | ); 63 | }, 64 | ]; 65 | 66 | return { 67 | name: 'Trim Db Items', 68 | tests, 69 | }; 70 | })() 71 | ); 72 | -------------------------------------------------------------------------------- /src/js/tests/test_updateCurrentSession.js: -------------------------------------------------------------------------------- 1 | /*global chrome, gsSession, gsIndexedDb, gsUtils, getFixture, assertTrue, FIXTURE_CURRENT_SESSIONS */ 2 | var testSuites = typeof testSuites === 'undefined' ? [] : testSuites; 3 | testSuites.push( 4 | (function() { 5 | 'use strict'; 6 | 7 | const tests = [ 8 | // Test hammering current session with updates 9 | async () => { 10 | const currentSessionWindows = []; 11 | const currentSessionId = gsSession.getSessionId(); 12 | 13 | for (let i = 0; i < 100; i++) { 14 | const session1 = await getFixture( 15 | FIXTURE_CURRENT_SESSIONS, 16 | 'currentSession1' 17 | ); 18 | let windowTemplate = session1.windows[0]; 19 | windowTemplate.id = i; 20 | currentSessionWindows.push(windowTemplate); 21 | 22 | const currentSession = await gsSession.buildCurrentSession(); 23 | currentSession.windows = currentSessionWindows; 24 | 25 | // Purposely don't await on this call 26 | gsIndexedDb.updateSession(currentSession); 27 | await gsUtils.setTimeout(1); 28 | } 29 | 30 | //if it's a saved session (prefixed with an underscore) 31 | const gsTestDb = await gsIndexedDb.getDb(); 32 | const results = await gsTestDb 33 | .query(gsIndexedDb.DB_CURRENT_SESSIONS, 'sessionId') 34 | .only(currentSessionId) 35 | .desc() 36 | .execute(); 37 | const onlySingleSessionForIdExists = results.length === 1; 38 | 39 | const currentSessionsAfter = await gsIndexedDb.fetchCurrentSessions(); 40 | const isCurrentSessionsPopulated = currentSessionsAfter.length === 1; 41 | const isCurrentSessionValid = 42 | currentSessionsAfter[0].windows.length === 100; 43 | 44 | return assertTrue( 45 | onlySingleSessionForIdExists && 46 | isCurrentSessionsPopulated && 47 | isCurrentSessionValid 48 | ); 49 | }, 50 | ]; 51 | 52 | return { 53 | name: 'Update current session', 54 | tests, 55 | }; 56 | })() 57 | ); 58 | -------------------------------------------------------------------------------- /src/js/tests/tests.js: -------------------------------------------------------------------------------- 1 | /* global gsIndexedDb, testSuites */ 2 | /* eslint-disable no-unused-vars */ 3 | 4 | const FIXTURE_CURRENT_SESSIONS = 'currentSessions'; 5 | const FIXTURE_SAVED_SESSIONS = 'savedSessions'; 6 | const FIXTURE_PREVIEW_URLS = 'previewUrls'; 7 | 8 | const requiredLibs = [ 9 | 'db', 10 | 'gsSession', 11 | 'gsStorage', 12 | 'gsUtils', 13 | 'gsChrome', 14 | 'gsTabSuspendManager', 15 | 'gsIndexedDb', 16 | 'gsTabQueue', 17 | 'gsFavicon', 18 | ]; 19 | 20 | function loadJsFile(fileName) { 21 | return new Promise(resolve => { 22 | const script = document.createElement('script'); 23 | script.onload = resolve; 24 | script.src = chrome.extension.getURL(`js/${fileName}.js`); 25 | document.head.appendChild(script); 26 | }); 27 | } 28 | 29 | function loadJsonFixture(fileName) { 30 | return new Promise(resolve => { 31 | const request = new XMLHttpRequest(); 32 | request.open( 33 | 'GET', 34 | chrome.extension.getURL(`js/tests/fixture_${fileName}.json`), 35 | true 36 | ); 37 | request.onload = () => { 38 | return resolve(JSON.parse(request.responseText)); 39 | }; 40 | request.send(); 41 | }); 42 | } 43 | 44 | function assertTrue(testResult) { 45 | if (testResult) { 46 | return Promise.resolve(true); 47 | } else { 48 | return Promise.reject(new Error(Error.captureStackTrace({}))); 49 | } 50 | } 51 | 52 | async function getFixture(fixtureName, itemName) { 53 | const fixtures = await loadJsonFixture(fixtureName); 54 | return JSON.parse(JSON.stringify(fixtures[itemName])); 55 | } 56 | 57 | async function runTests() { 58 | for (let testSuite of testSuites) { 59 | const resultEl = document.createElement('div'); 60 | resultEl.innerHTML = `Testing ${testSuite.name}...`; 61 | document.getElementById('results').appendChild(resultEl); 62 | 63 | let allTestsPassed = true; 64 | console.log(`Running testSuite: ${testSuite.name}..`); 65 | for (let [j, test] of testSuite.tests.entries()) { 66 | console.log(` Running test ${j + 1}..`); 67 | 68 | // loads/reset required libs 69 | await Promise.all(requiredLibs.map(loadJsFile)); 70 | 71 | // clear indexedDb contents 72 | gsIndexedDb.DB_SERVER = 'tgsTest'; 73 | await gsIndexedDb.clearGsDatabase(); 74 | 75 | // run test 76 | try { 77 | const result = await test(); 78 | console.log(` ${result}`); 79 | allTestsPassed = allTestsPassed && result; 80 | } catch (e) { 81 | allTestsPassed = false; 82 | console.error(e); 83 | } 84 | } 85 | 86 | //update test.html with testSuite result 87 | if (allTestsPassed) { 88 | resultEl.innerHTML = `Testing ${testSuite.name}: PASSED`; 89 | resultEl.style = 'color: green;'; 90 | } else { 91 | resultEl.innerHTML = `Testing ${testSuite.name}: FAILED`; 92 | resultEl.style = 'color: red;'; 93 | } 94 | } 95 | document.getElementById('suspendy-guy-inprogress').style.display = 'none'; 96 | document.getElementById('suspendy-guy-complete').style.display = 97 | 'inline-block'; 98 | } 99 | 100 | if (document.readyState !== 'loading') { 101 | runTests(); 102 | } else { 103 | document.addEventListener('DOMContentLoaded', function() { 104 | runTests(); 105 | }); 106 | } 107 | -------------------------------------------------------------------------------- /src/js/update.js: -------------------------------------------------------------------------------- 1 | /*global chrome, historyUtils, gsSession, gsIndexedDb, gsUtils */ 2 | (function(global) { 3 | 'use strict'; 4 | 5 | try { 6 | chrome.extension.getBackgroundPage().tgs.setViewGlobals(global); 7 | } catch (e) { 8 | window.setTimeout(() => window.location.reload(), 1000); 9 | return; 10 | } 11 | 12 | function setRestartExtensionClickHandler(warnFirst) { 13 | document.getElementById('restartExtensionBtn').onclick = async function(e) { 14 | // var result = true; 15 | // if (warnFirst) { 16 | // result = window.confirm(chrome.i18n.getMessage('js_update_confirm')); 17 | // } 18 | // if (result) { 19 | 20 | document.getElementById('restartExtensionBtn').className += ' btnDisabled'; 21 | document.getElementById('restartExtensionBtn').onclick = null; 22 | 23 | const currentSession = await gsSession.buildCurrentSession(); 24 | if (currentSession) { 25 | var currentVersion = chrome.runtime.getManifest().version; 26 | await gsIndexedDb.createOrUpdateSessionRestorePoint( 27 | currentSession, 28 | currentVersion 29 | ); 30 | } 31 | 32 | //ensure we don't leave any windows with no unsuspended tabs 33 | await gsSession.unsuspendActiveTabInEachWindow(); 34 | 35 | //update current session to ensure the new tab ids are saved before 36 | //we restart the extension 37 | await gsSession.updateCurrentSession(); 38 | 39 | chrome.runtime.reload(); 40 | // } 41 | }; 42 | } 43 | 44 | function setExportBackupClickHandler() { 45 | document.getElementById('exportBackupBtn').onclick = async function(e) { 46 | const currentSession = await gsSession.buildCurrentSession(); 47 | historyUtils.exportSession(currentSession, function() { 48 | document.getElementById('exportBackupBtn').style.display = 'none'; 49 | setRestartExtensionClickHandler(false); 50 | }); 51 | }; 52 | } 53 | 54 | function setSessionManagerClickHandler() { 55 | document.getElementById('sessionManagerLink').onclick = function(e) { 56 | e.preventDefault(); 57 | chrome.tabs.create({ url: chrome.extension.getURL('history.html') }); 58 | setRestartExtensionClickHandler(false); 59 | }; 60 | } 61 | 62 | gsUtils.documentReadyAndLocalisedAsPromsied(document).then(function() { 63 | setSessionManagerClickHandler(); 64 | setRestartExtensionClickHandler(true); 65 | setExportBackupClickHandler(); 66 | 67 | var currentVersion = chrome.runtime.getManifest().version; 68 | gsIndexedDb 69 | .fetchSessionRestorePoint(currentVersion) 70 | .then(function(sessionRestorePoint) { 71 | if (!sessionRestorePoint) { 72 | gsUtils.warning( 73 | 'update', 74 | 'Couldnt find session restore point. Something has gone horribly wrong!!' 75 | ); 76 | document.getElementById('noBackupInfo').style.display = 'block'; 77 | document.getElementById('backupInfo').style.display = 'none'; 78 | document.getElementById('exportBackupBtn').style.display = 'none'; 79 | } 80 | }); 81 | }); 82 | })(this); 83 | -------------------------------------------------------------------------------- /src/js/updated.js: -------------------------------------------------------------------------------- 1 | /*global chrome, gsSession, gsUtils */ 2 | (function(global) { 3 | 'use strict'; 4 | 5 | try { 6 | chrome.extension.getBackgroundPage().tgs.setViewGlobals(global); 7 | } catch (e) { 8 | window.setTimeout(() => window.location.reload(), 1000); 9 | return; 10 | } 11 | 12 | function toggleUpdated() { 13 | document.getElementById('updating').style.display = 'none'; 14 | document.getElementById('updated').style.display = 'block'; 15 | } 16 | 17 | gsUtils.documentReadyAndLocalisedAsPromsied(document).then(function() { 18 | var versionEl = document.getElementById('updatedVersion'); 19 | versionEl.innerHTML = 'v' + chrome.runtime.getManifest().version; 20 | 21 | document.getElementById('sessionManagerLink').onclick = function(e) { 22 | e.preventDefault(); 23 | chrome.tabs.create({ url: chrome.extension.getURL('history.html') }); 24 | }; 25 | 26 | var updateType = gsSession.getUpdateType(); 27 | if (updateType === 'major') { 28 | document.getElementById('patchMessage').style.display = 'none'; 29 | document.getElementById('minorUpdateDetail').style.display = 'none'; 30 | } else if (updateType === 'minor') { 31 | document.getElementById('patchMessage').style.display = 'none'; 32 | document.getElementById('majorUpdateDetail').style.display = 'none'; 33 | } else { 34 | document.getElementById('updateDetail').style.display = 'none'; 35 | } 36 | 37 | if (gsSession.isUpdated()) { 38 | toggleUpdated(); 39 | } 40 | }); 41 | 42 | global.exports = { 43 | toggleUpdated, 44 | }; 45 | })(this); 46 | -------------------------------------------------------------------------------- /src/manifest.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "Unofficial The Great Suspender", 3 | "description": "__MSG_ext_extension_description__", 4 | "version": "7.1.5", 5 | "default_locale": "en", 6 | "permissions": [ 7 | "tabs", 8 | "storage", 9 | "history", 10 | "unlimitedStorage", 11 | "http://*/*", 12 | "https://*/*", 13 | "file://*/*", 14 | "contextMenus", 15 | "", 16 | "activeTab", 17 | "cookies" 18 | ], 19 | "background": { 20 | "scripts": ["js/gsUtils.js", "js/gsChrome.js", "js/gsStorage.js", "js/db.js", "js/gsIndexedDb.js", 21 | "js/gsMessages.js", "js/gsSession.js", "js/gsTabQueue.js", "js/gsTabCheckManager.js", "js/gsFavicon.js", 22 | "js/gsTabSuspendManager.js", "js/gsTabDiscardManager.js", "js/gsSuspendedTab.js", "js/background.js"], 23 | "persistent": true 24 | }, 25 | "content_scripts": [ 26 | { 27 | "matches": ["http://*/*", "https://*/*", "file://*/*"], 28 | "js": ["js/contentscript.js"] 29 | } 30 | ], 31 | "browser_action": { 32 | "default_title": "__MSG_ext_default_title__", 33 | "default_icon": { 34 | "16": "img/ic_suspendy_16x16.png", 35 | "32": "img/ic_suspendy_32x32.png" 36 | }, 37 | "default_popup": "popup.html" 38 | }, 39 | "options_ui": { 40 | "page": "options.html", 41 | "open_in_tab": true 42 | }, 43 | "icons": { 44 | "16": "img/ic_suspendy_16x16.png", 45 | "32": "img/ic_suspendy_32x32.png", 46 | "48": "img/ic_suspendy_48x48.png", 47 | "128": "img/ic_suspendy_128x128.png" 48 | }, 49 | "content_security_policy": "script-src 'self'; object-src 'self';", 50 | "incognito": "not_allowed", 51 | "manifest_version": 2, 52 | "minimum_chrome_version": "55", 53 | 54 | "commands": { 55 | "1-suspend-tab": { 56 | "description": "__MSG_ext_cmd_toggle_tab_suspension_description__", 57 | "suggested_key": { "default": "Ctrl+Shift+U" } 58 | }, 59 | "2-toggle-temp-whitelist-tab": { 60 | "description": "__MSG_ext_cmd_toggle_tab_pause_description__" 61 | }, 62 | "2a-suspend-selected-tabs": { 63 | "description": "__MSG_ext_cmd_suspend_selected_tabs_description__" 64 | }, 65 | "2b-unsuspend-selected-tabs": { 66 | "description": "__MSG_ext_cmd_unsuspend_selected_tabs_description__" 67 | }, 68 | "3-suspend-active-window": { 69 | "description": "__MSG_ext_cmd_soft_suspend_active_window_description__" 70 | }, 71 | "3b-force-suspend-active-window": { 72 | "description": "__MSG_ext_cmd_force_suspend_active_window_description__" 73 | }, 74 | "4-unsuspend-active-window": { 75 | "description": "__MSG_ext_cmd_unsuspend_active_window_description__" 76 | }, 77 | "4b-soft-suspend-all-windows": { 78 | "description": "__MSG_ext_cmd_soft_suspend_all_windows_description__" 79 | }, 80 | "5-suspend-all-windows": { 81 | "description": "__MSG_ext_cmd_force_suspend_all_windows_description__" 82 | }, 83 | "6-unsuspend-all-windows": { 84 | "description": "__MSG_ext_cmd_unsuspend_all_windows_description__" 85 | } 86 | }, 87 | "applications": { 88 | "gecko": { 89 | "id": "thegreatsuspender@dvalter" 90 | } 91 | } 92 | } 93 | -------------------------------------------------------------------------------- /src/notice.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 |
14 |
15 | 16 | 17 | 18 | 19 | -------------------------------------------------------------------------------- /src/permissions.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 |
15 | 16 | 17 | 18 |
19 |

20 | 21 |

22 |

23 | 24 |
25 |
26 | 27 | 28 |
29 |
30 | 31 |
32 | 33 | 34 | 35 | 36 | -------------------------------------------------------------------------------- /src/popup.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 14 |
15 |
16 | 19 | 22 | 25 | 28 |
29 | 30 |
31 | 34 | 37 |
38 | 39 |
40 | 43 | 46 |
47 | 48 |
49 | 52 |
53 |
54 | 55 | 56 | 57 | -------------------------------------------------------------------------------- /src/recovery.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 |
15 | 16 |
17 | 18 | 19 |
20 |
21 |
22 |

23 |

24 | 25 |
26 | 27 |

28 |
29 |

30 | 31 |
32 |
33 | 34 | 35 | 36 |
37 |
38 |
39 |
40 |
41 |
42 |

43 | 44 |

45 |

46 | 47 | 48 | 49 |

50 |
51 |
52 |
53 | 54 | 55 | 56 | 57 | -------------------------------------------------------------------------------- /src/restoring-window.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 |
14 | 15 | 16 |
17 |

18 |
19 | 20 |
21 |
22 |
23 | 24 | 25 | 26 | 27 | -------------------------------------------------------------------------------- /src/shortcuts.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 | 15 |
16 | 17 | 20 |
21 |
    22 |
  • 23 | 24 |
  • 25 |
  • 26 | 27 |
  • 28 |
  • 29 | 30 |
  • 31 |
  • 32 | 33 |
  • 34 |
35 |
36 | 37 |
38 | 39 |

40 |
41 | 42 |
43 |
44 | 45 | 46 |
47 |
48 | 49 |
50 | 51 | 52 | 53 | 54 | -------------------------------------------------------------------------------- /src/suspended.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | ... 12 | 13 | 14 | 15 | 16 | 17 | 18 | 21 | 22 | 28 | 29 |
30 |
31 |
32 | 33 |
34 | 35 |
36 |
37 | 38 |
39 |
40 | 41 |
42 |
43 | 44 |
45 |
46 |
47 |
48 |
49 |
50 |
51 | 52 |
53 |
54 |
55 |
56 | The Great Suspender 57 |
58 | 59 |
60 | 61 | 62 | 63 | -------------------------------------------------------------------------------- /src/tests.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 | 15 | 16 | The Great Suspender is testing 17 | 18 | 19 | 20 | 21 |
22 | 23 |
24 | 25 | 26 |
27 |
28 |
29 |
30 |
31 | 32 | 33 | 34 | 35 | -------------------------------------------------------------------------------- /src/update.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 |
15 | 16 | 17 | 18 |
19 |

20 | 21 | 22 |

23 |
24 |

25 |
26 |
27 |

28 | 29 |

30 |
31 |
32 |
33 | 34 | 35 |
36 |
37 | 38 |
39 | 40 | 41 | 42 | 43 | -------------------------------------------------------------------------------- /src/updated.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 | 15 |
16 | 17 | 18 |
19 |

20 |
21 | 22 |
23 |
24 | 25 |
26 |

27 |
28 | 29 |
30 |
31 |
32 |

33 |

34 | 35 |

36 |
37 | 38 |
39 |
40 |

41 | 42 |

43 | 44 |
45 |
    46 |
  • 47 | New option to enable Chrome's built-in memory-saving for suspended tabs.
    48 | Can boost memory savings by up to 5x! 49 |
  • 50 |
  • New right-click context menu options and keyboard shortcuts.
  • 51 |
  • Ability to suspend local file urls.
  • 52 |
  • Improved session saving, export and recovery.
  • 53 |
  • Fixed issues with chrome stealing focus when a tab suspends in the background.
  • 54 |
  • Many other minor bug fixes and performance improvements.
  • 55 |
56 |
57 | 58 |
59 |
60 | 61 |
62 |
    63 |
  • New look UI, themes and icons.
  • 64 |
  • Extension settings can now be synced across computers.
  • 65 |
  • 66 | New option to enable Chrome's built-in memory-saving for suspended tabs.
    67 | Can boost memory savings by up to 5x! 68 |
  • 69 |
  • New right-click context menu options and keyboard shortcuts.
  • 70 |
  • Ability to suspend local file urls.
  • 71 |
  • Improved "Open link in new suspended tab" right-click menu option.
  • 72 |
  • Support for retina favicons and high resolution screen captures.
  • 73 |
  • Suspended youtube tabs now remember current play position.
  • 74 |
  • Session management now supports importing of saved sessions.
  • 75 |
  • Added 1 week & 2 week suspend timer options.
  • 76 |
77 |
78 | 79 |
80 |
    81 |
  • Fixed issue with cookies breaking websites like YouTube.
  • 82 |
  • Fixed issue where suspended tabs showed wrong favicon on restart.
  • 83 |
  • Fixed issues with chrome stealing focus when a tab suspends in the background.
  • 84 |
  • Improved behaviour of automatic tab unsuspending.
  • 85 |
  • Fixed issue with setting the correct scroll location on suspend and unsuspend.
  • 86 |
  • Fixed issue with right-click context menu.
  • 87 |
  • Fixed issue with session exporting.
  • 88 |
  • Many other minor bug fixes and performance improvements.
  • 89 |
90 |
91 |
92 |

93 | github.com/deanoemcke/thegreatsuspender/releases 94 |
95 | Thanks to all the people who have helped with these features and fixes! 96 |

97 |
98 |
99 |
100 | 101 | 102 | 103 | 104 | --------------------------------------------------------------------------------