mirror of
https://github.com/XFox111/TabsAsideExtension.git
synced 2026-07-02 19:52:47 +03:00
Compare commits
10 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| d8c31e3da5 | |||
| 8d2c83eea6 | |||
| 3e60583427 | |||
| 10edf4f975 | |||
| cc854fbd25 | |||
| ad9acb6208 | |||
| f74eeb5759 | |||
| 80c9dbe1a5 | |||
| c777f1a60a | |||
| 1b7b419168 |
@@ -0,0 +1,73 @@
|
|||||||
|
name: CI
|
||||||
|
|
||||||
|
on:
|
||||||
|
workflow_dispatch:
|
||||||
|
push:
|
||||||
|
branches: [ master ]
|
||||||
|
paths:
|
||||||
|
# Trigger deploy on manifest change
|
||||||
|
- 'manifest.json'
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
Firefox:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v2
|
||||||
|
|
||||||
|
- name: Build Extension for Firefox
|
||||||
|
id: web-ext-build
|
||||||
|
uses: kewisch/action-web-ext@v1
|
||||||
|
with:
|
||||||
|
cmd: build
|
||||||
|
|
||||||
|
- name: Sign build
|
||||||
|
id: web-ext-sign
|
||||||
|
uses: kewisch/action-web-ext@v1
|
||||||
|
with:
|
||||||
|
cmd: sign
|
||||||
|
source: ${{ steps.web-ext-build.outputs.target }}
|
||||||
|
apiKey: ${{ secrets.FIREFOX_API_KEY }}
|
||||||
|
apiSecret: ${{ secrets.FIREFOX_CLIENT_SECRET }}
|
||||||
|
|
||||||
|
- name: Publish to Firefox Webstore
|
||||||
|
uses: trmcnvn/firefox-addon@v1
|
||||||
|
with:
|
||||||
|
uuid: tabsaside@xfox111.net
|
||||||
|
xpi: ${{ steps.web-ext-sign.outputs.target }}
|
||||||
|
manifest: ./manifest.json
|
||||||
|
api-key: ${{ secrets.FIREFOX_API_KEY }}
|
||||||
|
api-secret: ${{ secrets.FIREFOX_CLIENT_SECRET }}
|
||||||
|
|
||||||
|
- name: Drop artifacts
|
||||||
|
uses: actions/upload-artifact@v2
|
||||||
|
with:
|
||||||
|
name: 'Firefox Artefacts'
|
||||||
|
path: ${{ steps.web-ext-sign.outputs.target }}
|
||||||
|
|
||||||
|
Chrome:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v2
|
||||||
|
|
||||||
|
- name: Pack extension
|
||||||
|
uses: TheDoctor0/zip-release@0.4.1
|
||||||
|
with:
|
||||||
|
filename: ./TabsAside.zip
|
||||||
|
exclusions: '.git/* .vscode/* .github/* *.md'
|
||||||
|
|
||||||
|
- name: Publish to Chrome Webstore
|
||||||
|
uses: SebastienGllmt/chrome-addon@v3
|
||||||
|
with:
|
||||||
|
extension: mgmjbodjgijnebfgohlnjkegdpbdjgin
|
||||||
|
zip: ./TabsAside.zip
|
||||||
|
client-id: ${{ secrets.CHROME_CLIENT_ID }}
|
||||||
|
client-secret: ${{ secrets.CHROME_CLIENT_SECRET }}
|
||||||
|
refresh-token: ${{ secrets.CHROME_REFRESH_TOKEN }}
|
||||||
|
|
||||||
|
- name: Drop artifacts
|
||||||
|
uses: actions/upload-artifact@v2
|
||||||
|
with:
|
||||||
|
name: 'Chrome Artifacts'
|
||||||
|
path: ./TabsAside.zip
|
||||||
Vendored
+13
@@ -0,0 +1,13 @@
|
|||||||
|
{
|
||||||
|
"version": "0.2.0",
|
||||||
|
"configurations":
|
||||||
|
[
|
||||||
|
{
|
||||||
|
"name": "Debug",
|
||||||
|
"type": "firefox",
|
||||||
|
"request": "launch",
|
||||||
|
"reAttach": true,
|
||||||
|
"addonPath": "${workspaceFolder}"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
+6
-6
@@ -4,7 +4,7 @@ Welcome, and thank you for your interest in contributing to my project!
|
|||||||
There are many ways in which you can contribute, beyond writing code. The goal of this document is to provide a high-level overview of how you can get involved.
|
There are many ways in which you can contribute, beyond writing code. The goal of this document is to provide a high-level overview of how you can get involved.
|
||||||
|
|
||||||
## Table of Contents
|
## Table of Contents
|
||||||
- [Contribution Guidelines](#gutschedule-contribution-guidelines)
|
- [Contribution Guidelines](#contribution-guidelines)
|
||||||
- [Table of Contents](#table-of-contents)
|
- [Table of Contents](#table-of-contents)
|
||||||
- [Asking Questions](#asking-questions)
|
- [Asking Questions](#asking-questions)
|
||||||
- [Providing Feedback](#providing-feedback)
|
- [Providing Feedback](#providing-feedback)
|
||||||
@@ -35,15 +35,15 @@ Have a question? Rather than opening an issue, please ask me directly on opensou
|
|||||||
|
|
||||||
## Providing Feedback
|
## Providing Feedback
|
||||||
Your comments and feedback are welcome.
|
Your comments and feedback are welcome.
|
||||||
You can leave your feedbak on feedback@xfox111.net or do it on [Microsoft Edge Add-ons Webstore](https://microsoftedge.microsoft.com/addons/detail/tabs-aside/kmnblllmalkiapkfknnlpobmjjdnlhnd) or [Chrome Webstore](https://chrome.google.com/webstore/detail/tabs-aside/mgmjbodjgijnebfgohlnjkegdpbdjgin)
|
You can leave your feedbak on feedback@xfox111.net or do it on [Microsoft Edge Add-ons Webstore](https://microsoftedge.microsoft.com/addons/detail/tabs-aside/kmnblllmalkiapkfknnlpobmjjdnlhnd), [Chrome Extensions Webstore](https://chrome.google.com/webstore/detail/tabs-aside/mgmjbodjgijnebfgohlnjkegdpbdjgin) or [Mozilla Add-ons Webstore](https://addons.mozilla.org/firefox/addon/ms-edge-tabs-aside/)
|
||||||
|
|
||||||
## Reporting Issues
|
## Reporting Issues
|
||||||
Have you identified a reproducible problem in the extension? Have a feature request? I'd like to hear it! Here's how you can make reporting your issue as effective as possible.
|
Have you identified a reproducible problem in the extension? Have a feature request? I'd like to hear it! Here's how you can make reporting your issue as effective as possible.
|
||||||
|
|
||||||
### Look For an Existing Issue
|
### Look For an Existing Issue
|
||||||
Before you create a new issue, please do a search in [open issues](https://github.com/xfox111/ChromiumTabsAside/issues) to see if the issue or feature request has already been filed.
|
Before you create a new issue, please do a search in [open issues](https://github.com/xfox111/TabsAsideExtension/issues) to see if the issue or feature request has already been filed.
|
||||||
|
|
||||||
Be sure to scan through the [feature requests](https://github.com/XFox111/ChromiumTabsAside/issues?q=is%3Aissue+is%3Aopen+label%3Aenhancement).
|
Be sure to scan through the [feature requests](https://github.com/XFox111/TabsAsideExtension/issues?q=is%3Aissue+is%3Aopen+label%3Aenhancement).
|
||||||
|
|
||||||
If you find your issue already exists, make relevant comments and add your [reaction](https://github.com/blog/2119-add-reactions-to-pull-requests-issues-and-comments). Use a reaction in place of a "+1" comment:
|
If you find your issue already exists, make relevant comments and add your [reaction](https://github.com/blog/2119-add-reactions-to-pull-requests-issues-and-comments). Use a reaction in place of a "+1" comment:
|
||||||
|
|
||||||
@@ -83,7 +83,7 @@ If you are interested in writing code to fix issues or implement new awesome fea
|
|||||||
### Deploy test version on your browser
|
### Deploy test version on your browser
|
||||||
1. Clone repository to local storage using [Git](https://guides.github.com/introduction/git-handbook/)
|
1. Clone repository to local storage using [Git](https://guides.github.com/introduction/git-handbook/)
|
||||||
```
|
```
|
||||||
git clone https://github.com/xfox111/ChromiumTabsAside.git
|
git clone https://github.com/xfox111/TabsAsideExtension.git
|
||||||
```
|
```
|
||||||
2. Enable Developers mode on your browser extensions page
|
2. Enable Developers mode on your browser extensions page
|
||||||
3. Click "Load unpacked" button and navigate to the extension root folder (contains `manifest.json`)
|
3. Click "Load unpacked" button and navigate to the extension root folder (contains `manifest.json`)
|
||||||
@@ -175,7 +175,7 @@ Use "double quotes" wherever it's possible
|
|||||||
```
|
```
|
||||||
|
|
||||||
### Finding an issue to work on
|
### Finding an issue to work on
|
||||||
Check out the [full issues list](https://github.com/XFox111/ChromiumTabsAside/issues?utf8=%E2%9C%93&q=is%3Aopen+is%3Aissue) for a list of all potential areas for contributions. **Note** that just because an issue exists in the repository does not mean we will accept a contribution. There are several reasons we may not accept a pull request like:
|
Check out the [full issues list](https://github.com/XFox111/TabsAsideExtension/issues?utf8=%E2%9C%93&q=is%3Aopen+is%3Aissue) for a list of all potential areas for contributions. **Note** that just because an issue exists in the repository does not mean we will accept a contribution. There are several reasons we may not accept a pull request like:
|
||||||
|
|
||||||
- Performance - One of Tabs Aside core values is to deliver a lightweight extension, that means it should perform well in both real and test environments.
|
- Performance - One of Tabs Aside core values is to deliver a lightweight extension, that means it should perform well in both real and test environments.
|
||||||
- User experience - Since we want to deliver a lightweight extension, the UX should feel lightweight as well and not be cluttered. Most changes to the UI should go through the issue owner and project owner (@XFox111).
|
- User experience - Since we want to deliver a lightweight extension, the UX should feel lightweight as well and not be cluttered. Most changes to the UI should go through the issue owner and project owner (@XFox111).
|
||||||
|
|||||||
@@ -1,56 +1,64 @@
|
|||||||
# Tabs aside for Google Chrome
|
# Tabs aside
|
||||||
|
[](https://github.com/xfox111/TabsAsideExtension/releases/latest)
|
||||||
|

|
||||||
|
|
||||||
[](https://chrome.google.com/webstore/detail/tabs-aside/mgmjbodjgijnebfgohlnjkegdpbdjgin)
|
[](https://chrome.google.com/webstore/detail/tabs-aside/mgmjbodjgijnebfgohlnjkegdpbdjgin)
|
||||||
[](https://chrome.google.com/webstore/detail/tabs-aside/mgmjbodjgijnebfgohlnjkegdpbdjgin)
|
[](https://chrome.google.com/webstore/detail/tabs-aside/mgmjbodjgijnebfgohlnjkegdpbdjgin)
|
||||||
[](https://github.com/xfox111/chromiumtabsaside/releases/latest)
|
|
||||||
|
|
||||||
[](https://github.com/xfox111/ChromiumTabsAside/issues)
|
[](https://addons.mozilla.org/firefox/addon/ms-edge-tabs-aside/)
|
||||||
[](https://github.com/xfox111/ChromiumTabsAside/commits/master)
|
[](https://addons.mozilla.org/firefox/addon/ms-edge-tabs-aside/)
|
||||||
[](https://github.com/xfox111/ChromiumTabsAside)
|
[](https://addons.mozilla.org/firefox/addon/ms-edge-tabs-aside/)
|
||||||
[](https://opensource.org/licenses/MIT)
|
|
||||||
|
[](https://github.com/xfox111/TabsAsideExtension/issues)
|
||||||
|
[](https://github.com/xfox111/TabsAsideExtension/commits/master)
|
||||||
|
[](https://github.com/xfox111/TabsAsideExtension)
|
||||||
|
[](https://opensource.org/licenses/MIT)
|
||||||
|
|
||||||
[](https://twitter.com/xfox111)
|
[](https://twitter.com/xfox111)
|
||||||
[](https://github.com/xfox111)
|
[](https://github.com/xfox111)
|
||||||
[](https://buymeacoffee.com/xfox111)
|
[](https://buymeacoffee.com/xfox111)
|
||||||
|
|
||||||

|

|
||||||
|
|
||||||
If you’re like me, you often find yourself with a bunch of open tabs. You’d like to get those tabs out of the way sometimes, but they’re maybe not worth saving as actual bookmarks.
|
If you’re like me, you often find yourself with a bunch of open tabs. You’d like to get those tabs out of the way sometimes, but they’re maybe not worth saving as actual bookmarks.
|
||||||
|
|
||||||
In the Edge browser, Microsoft has introduced a new feature called "Tabs aside" (or Tab groups) which lets you set tabs aside in a sort of temporary workspace so that you can call them back up later.
|
In the Edge browser, Microsoft has introduced a new feature called "Tabs aside" (or Tab groups) which lets you set tabs aside in a sort of temporary workspace so that you can call them back up later.
|
||||||
|
|
||||||
Unfortunately, in new Chromium-based Microsoft Edge, the devs decided not to implement this must-have-feature. So I've decided to create a browser extension which replicates this awesome feature in Chromium-based browsers
|
Unfortunately, in new Chromium-based Microsoft Edge, the devs decided not to implement this must-have-feature. So I've decided to create a browser extension which replicates this awesome feature
|
||||||
|
|
||||||
## Features
|
## Features
|
||||||
- Familiar UI inherited from legacy Microsoft Edge with some improvements
|
- Familiar UI inherited from legacy Microsoft Edge with some improvements
|
||||||
- Auto Dark mode
|
- Auto Dark mode
|
||||||
- Now you can restore one tab from collection without removing
|
- Now you can restore one tab from collection without removing
|
||||||
- Now you can choose if you want to load restored tabs only when you're navigating onto them
|
- Now you can choose if you want to load restored tabs only when you're navigating onto them
|
||||||
|
- Set tabs you've selected aside
|
||||||
|
- **Now available for Firefox!**
|
||||||
|
|
||||||
## Download
|
## Download
|
||||||
- [Google Chrome Webstore](https://chrome.google.com/webstore/detail/tabs-aside/mgmjbodjgijnebfgohlnjkegdpbdjgin)
|
- [Google Chrome Webstore](https://chrome.google.com/webstore/detail/tabs-aside/mgmjbodjgijnebfgohlnjkegdpbdjgin)
|
||||||
- [Microsoft Edge Add-ons Webstore](https://microsoftedge.microsoft.com/addons/detail/kmnblllmalkiapkfknnlpobmjjdnlhnd)
|
- [Microsoft Edge Add-ons Webstore](https://microsoftedge.microsoft.com/addons/detail/kmnblllmalkiapkfknnlpobmjjdnlhnd)
|
||||||
- [GitHub Releases](https://github.com/xfox111/chromiumtabsaside/releases/latest)
|
- [Firefox Add-ons](https://addons.mozilla.org/firefox/addon/ms-edge-tabs-aside/)
|
||||||
|
- [GitHub Releases](https://github.com/xfox111/TabsAsideExtension/releases/latest)
|
||||||
|
|
||||||
## Project roadmap
|
## Project roadmap
|
||||||
You can go to the project's [roadmap kanban board](https://github.com/XFox111/ChromiumTabsAside/projects/1) and see what have we planned and watch our progress in realtime
|
You can go to the project's [roadmap kanban board](https://github.com/XFox111/TabsAsideExtension/projects/1) and see what have we planned and watch our progress in realtime
|
||||||
|
|
||||||
## Contributing
|
## Contributing
|
||||||
There are many ways in which you can participate in the project, for example:
|
There are many ways in which you can participate in the project, for example:
|
||||||
- [Submit bugs and feature requests](https://github.com/xfox111/gutschedule/issues), and help us verify as they are checked in
|
- [Submit bugs and feature requests](https://github.com/xfox111/TabsAsideExtension/issues), and help us verify as they are checked in
|
||||||
- Review [source code changes](https://github.com/xfox111/gutschedule/pulls)
|
- Review [source code changes](https://github.com/xfox111/TabsAsideExtension/pulls)
|
||||||
- Review documentation and make pull requests for anything from typos to new content
|
- Review documentation and make pull requests for anything from typos to new content
|
||||||
|
|
||||||
If you are interested in fixing issues and contributing directly to the code base, please see the [Contribution Guidelines](https://github.com/XFox111/ChromiumTabsAside/blob/master/CONTRIBUTING.md), which covers the following:
|
If you are interested in fixing issues and contributing directly to the code base, please see the [Contribution Guidelines](https://github.com/XFox111/TabsAsideExtension/blob/master/CONTRIBUTING.md), which covers the following:
|
||||||
- [How to deploy the extension on your browser](https://github.com/XFox111/ChromiumTabsAside/blob/master/CONTRIBUTING.md#deploy-test-version-on-your-browser)
|
- [How to deploy the extension on your browser](https://github.com/XFox111/TabsAsideExtension/blob/master/CONTRIBUTING.md#deploy-test-version-on-your-browser)
|
||||||
- [The development workflow](https://github.com/XFox111/ChromiumTabsAside/blob/master/CONTRIBUTING.md#development-workflow), including debugging and running tests
|
- [The development workflow](https://github.com/XFox111/TabsAsideExtension/blob/master/CONTRIBUTING.md#development-workflow), including debugging and running tests
|
||||||
- [Coding guidelines](https://github.com/XFox111/ChromiumTabsAside/blob/master/CONTRIBUTING.md#coding-guidelines)
|
- [Coding guidelines](https://github.com/XFox111/TabsAsideExtension/blob/master/CONTRIBUTING.md#coding-guidelines)
|
||||||
- [Submitting pull requests](https://github.com/XFox111/ChromiumTabsAside/blob/master/CONTRIBUTING.md#submitting-pull-requests)
|
- [Submitting pull requests](https://github.com/XFox111/TabsAsideExtension/blob/master/CONTRIBUTING.md#submitting-pull-requests)
|
||||||
- [Finding an issue to work on](https://github.com/XFox111/ChromiumTabsAside/blob/master/CONTRIBUTING.md#finding-an-issue-to-work-on)
|
- [Finding an issue to work on](https://github.com/XFox111/TabsAsideExtension/blob/master/CONTRIBUTING.md#finding-an-issue-to-work-on)
|
||||||
- [Contributing to translations](https://github.com/XFox111/ChromiumTabsAside/blob/master/CONTRIBUTING.md#contributing-to-translations)
|
- [Contributing to translations](https://github.com/XFox111/TabsAsideExtension/blob/master/CONTRIBUTING.md#contributing-to-translations)
|
||||||
|
|
||||||
## Code of Conduct
|
## Code of Conduct
|
||||||
This project has adopted the Contributor Covenant. For more information see the [Code of Conduct](https://github.com/XFox111/ChromiumTabsAside/blob/master/CODE_OF_CONDUCT.md)
|
This project has adopted the Contributor Covenant. For more information see the [Code of Conduct](https://github.com/XFox111/TabsAsideExtension/blob/master/CODE_OF_CONDUCT.md)
|
||||||
|
|
||||||
## Copyrights
|
## Copyrights
|
||||||
> ©2020 Michael "XFox" Gordeev
|
> ©2020 Michael "XFox" Gordeev
|
||||||
|
|||||||
+27
-7
@@ -8,13 +8,16 @@
|
|||||||
<link rel="stylesheet" type="text/css" href="css/style.css" />
|
<link rel="stylesheet" type="text/css" href="css/style.css" />
|
||||||
<link rel="stylesheet" type="text/css" href="css/style.generic.css" />
|
<link rel="stylesheet" type="text/css" href="css/style.generic.css" />
|
||||||
<link rel="stylesheet" type="text/css" href="css/style.dark.css" />
|
<link rel="stylesheet" type="text/css" href="css/style.dark.css" />
|
||||||
|
<link rel="stylesheet" type="text/css" href="css/style.listview.css" />
|
||||||
|
|
||||||
|
<meta charset="utf-8" />
|
||||||
</head>
|
</head>
|
||||||
|
|
||||||
<body>
|
<body>
|
||||||
<div class="tabsAside background">
|
<div class="tabsAside background">
|
||||||
|
<div class="tabsAside closeArea"></div>
|
||||||
<aside class="tabsAside pane">
|
<aside class="tabsAside pane">
|
||||||
<header>
|
<header>
|
||||||
<div>
|
|
||||||
<h1 loc="name">Tabs aside</h1>
|
<h1 loc="name">Tabs aside</h1>
|
||||||
<button loc_alt="options" class="btn more" title="Options"></button>
|
<button loc_alt="options" class="btn more" title="Options"></button>
|
||||||
|
|
||||||
@@ -27,21 +30,38 @@
|
|||||||
<input type="checkbox" id="swapIconAction" />
|
<input type="checkbox" id="swapIconAction" />
|
||||||
<label loc="swapIconAction" for="swapIconAction">Set tabs aside on extension icon click (Alt+P or right-click to open the pane)</label>
|
<label loc="swapIconAction" for="swapIconAction">Set tabs aside on extension icon click (Alt+P or right-click to open the pane)</label>
|
||||||
</p>
|
</p>
|
||||||
|
<p>
|
||||||
|
<input type="checkbox" id="showDeleteDialog" />
|
||||||
|
<label loc="showDeleteDialog" for="showDeleteDialog">Show confirmation dialog before deleting an item</label>
|
||||||
|
</p>
|
||||||
|
<hr />
|
||||||
<div>
|
<div>
|
||||||
<button loc="github" value="https://github.com/xfox111/ChromiumTabsAside">Visit GitHub page</button>
|
<button loc="github" value="https://github.com/xfox111/TabsAsideExtension">Visit GitHub page</button>
|
||||||
<button loc="contributors" value="https://github.com/XFox111/ChromiumTabsAside/graphs/contributors">Project contributors</button>
|
<button loc="contributors" value="https://github.com/XFox111/TabsAsideExtension/graphs/contributors">Project contributors</button>
|
||||||
<button loc="feedback" feedback-button>Leave feedback</button>
|
<button loc="feedback" feedback-button>Leave feedback</button>
|
||||||
<button loc="buyMeACoffee" value="https://buymeacoffee.com/xfox111">Buy me a coffee!</button>
|
<button loc="buyMeACoffee" value="https://buymeacoffee.com/xfox111">Buy me a coffee!</button>
|
||||||
</div>
|
</div>
|
||||||
|
<hr />
|
||||||
<p>
|
<p>
|
||||||
<small>v1.0</small><br />
|
<small>v1.0</small><br />
|
||||||
<span loc="credits">Developed by Michael Gordeev</span> (<a href="https://twitter.com/xfox111"
|
<span loc="credits">Developed by Michael Gordeev</span> (<a href="https://twitter.com/xfox111" target="_blank">@xfox111</a>)
|
||||||
target="_blank">@xfox111</a>)
|
|
||||||
</p>
|
</p>
|
||||||
</nav>
|
</nav>
|
||||||
</div>
|
|
||||||
|
<button loc_alt="closePanel" class="btn remove" title="Close panel"></button>
|
||||||
<a class="saveTabs"><span class="iconArrowRight"></span> <span loc="setAside">Set current tabs aside</span></a>
|
<a class="saveTabs"><span class="iconArrowRight"></span> <span loc="setAside">Set current tabs aside</span></a>
|
||||||
<hr />
|
|
||||||
|
<div class="listviewSwitch tile">
|
||||||
|
<div></div>
|
||||||
|
<div></div>
|
||||||
|
<div></div>
|
||||||
|
<div></div>
|
||||||
|
</div>
|
||||||
|
<div class="listviewSwitch list">
|
||||||
|
<div></div>
|
||||||
|
<div></div>
|
||||||
|
<div></div>
|
||||||
|
</div>
|
||||||
</header>
|
</header>
|
||||||
|
|
||||||
<section>
|
<section>
|
||||||
|
|||||||
+14
-44
@@ -6,7 +6,7 @@
|
|||||||
},
|
},
|
||||||
"description":
|
"description":
|
||||||
{
|
{
|
||||||
"message": "Classic Microsoft Edge \"Tabs Aside\" feature for Chromium browsers",
|
"message": "Classic Microsoft Edge \"Tabs Aside\" feature",
|
||||||
"description": "Extension description"
|
"description": "Extension description"
|
||||||
},
|
},
|
||||||
"author":
|
"author":
|
||||||
@@ -19,14 +19,24 @@
|
|||||||
"message": "Options",
|
"message": "Options",
|
||||||
"description": "Alternative text for options button in the pane"
|
"description": "Alternative text for options button in the pane"
|
||||||
},
|
},
|
||||||
|
"closePanel":
|
||||||
|
{
|
||||||
|
"message": "Close",
|
||||||
|
"description": "Alternative text for close panel button"
|
||||||
|
},
|
||||||
"loadOnRestore":
|
"loadOnRestore":
|
||||||
{
|
{
|
||||||
"message": "Load tabs on restore",
|
"message": "Load tabs on restore",
|
||||||
"description": "Label for option"
|
"description": "Label for option"
|
||||||
},
|
},
|
||||||
|
"showDeleteDialog":
|
||||||
|
{
|
||||||
|
"message": "Show confirmation dialog before deleting an item",
|
||||||
|
"description": "Label for option"
|
||||||
|
},
|
||||||
"swapIconAction":
|
"swapIconAction":
|
||||||
{
|
{
|
||||||
"message": "Set tabs aside on extension icon click (Alt+P or right-click to open the pane)",
|
"message": "Set tabs aside on extension icon click (%TOGGLE_SHORTCUT% or right-click to open the pane)",
|
||||||
"description": "Label for option"
|
"description": "Label for option"
|
||||||
},
|
},
|
||||||
"github":
|
"github":
|
||||||
@@ -111,47 +121,7 @@
|
|||||||
},
|
},
|
||||||
"tabs":
|
"tabs":
|
||||||
{
|
{
|
||||||
"message": "Tabs",
|
"message": "items",
|
||||||
"description": "Collection tabs counter label"
|
"description": "Collection tabs counter label (e.g. 8 items)"
|
||||||
},
|
|
||||||
"ago":
|
|
||||||
{
|
|
||||||
"message": "ago",
|
|
||||||
"description": "Human friendly timestamp part (e.g. 15 hour(s) ago)"
|
|
||||||
},
|
|
||||||
"minutes":
|
|
||||||
{
|
|
||||||
"message": "minute(s)",
|
|
||||||
"description": "Human friendly timestamp part (e.g. 15 minute(s) ago)"
|
|
||||||
},
|
|
||||||
"hours":
|
|
||||||
{
|
|
||||||
"message": "hour(s)",
|
|
||||||
"description": "Human friendly timestamp part (e.g. 15 hour(s) ago)"
|
|
||||||
},
|
|
||||||
"days":
|
|
||||||
{
|
|
||||||
"message": "day(s)",
|
|
||||||
"description": "Human friendly timestamp part (e.g. 15 day(s) ago)"
|
|
||||||
},
|
|
||||||
"weeks":
|
|
||||||
{
|
|
||||||
"message": "week(s)",
|
|
||||||
"description": "Human friendly timestamp part (e.g. 15 week(s) ago)"
|
|
||||||
},
|
|
||||||
"months":
|
|
||||||
{
|
|
||||||
"message": "month(s)",
|
|
||||||
"description": "Human friendly timestamp part (e.g. 15 month(s) ago)"
|
|
||||||
},
|
|
||||||
"years":
|
|
||||||
{
|
|
||||||
"message": "years(s)",
|
|
||||||
"description": "Human friendly timestamp part (e.g. 15 years(s) ago)"
|
|
||||||
},
|
|
||||||
"justNow":
|
|
||||||
{
|
|
||||||
"message": "Just now",
|
|
||||||
"description": "Human friendly timestamp part"
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -0,0 +1,127 @@
|
|||||||
|
{
|
||||||
|
"name":
|
||||||
|
{
|
||||||
|
"message": "Pestañas que has reservado",
|
||||||
|
"description": "Extension name. Displayed in the manifest and pane header"
|
||||||
|
},
|
||||||
|
"description":
|
||||||
|
{
|
||||||
|
"message": "Función \"Reservar pestañas\" de Microsoft Edge clásico",
|
||||||
|
"description": "Extension description"
|
||||||
|
},
|
||||||
|
"author":
|
||||||
|
{
|
||||||
|
"message": "Michael \"XFox\" Gordeev",
|
||||||
|
"description": "Author name"
|
||||||
|
},
|
||||||
|
"options":
|
||||||
|
{
|
||||||
|
"message": "Opciones",
|
||||||
|
"description": "Alternative text for options button in the pane"
|
||||||
|
},
|
||||||
|
"closePanel":
|
||||||
|
{
|
||||||
|
"message": "Cerrar",
|
||||||
|
"description": "Alternative text for close panel button"
|
||||||
|
},
|
||||||
|
"loadOnRestore":
|
||||||
|
{
|
||||||
|
"message": "Cargar pestañas al restaurar",
|
||||||
|
"description": "Label for option"
|
||||||
|
},
|
||||||
|
"showDeleteDialog":
|
||||||
|
{
|
||||||
|
"message": "Mostrar diálogo de confirmacion antes de eliminar un item",
|
||||||
|
"description": "Label for option"
|
||||||
|
},
|
||||||
|
"swapIconAction":
|
||||||
|
{
|
||||||
|
"message": "Reservar pestañas al apretar el icono de la extensión (%TOGGLE_SHORTCUT% o click-derecho para abrir el panel)",
|
||||||
|
"description": "Label for option"
|
||||||
|
},
|
||||||
|
"github":
|
||||||
|
{
|
||||||
|
"message": "Visitar la pagina de GitHub",
|
||||||
|
"description": "Link title"
|
||||||
|
},
|
||||||
|
"contributors":
|
||||||
|
{
|
||||||
|
"message": "Contribuidores del proyecto",
|
||||||
|
"description": "Link title"
|
||||||
|
},
|
||||||
|
"feedback":
|
||||||
|
{
|
||||||
|
"message": "Deja un comentario",
|
||||||
|
"description": "Link title"
|
||||||
|
},
|
||||||
|
"buyMeACoffee":
|
||||||
|
{
|
||||||
|
"message": "¡Comprame un cafe!",
|
||||||
|
"description": "Link title"
|
||||||
|
},
|
||||||
|
"credits":
|
||||||
|
{
|
||||||
|
"message": "Desarrollado por Michael 'XFox' Gordeev",
|
||||||
|
"description": "Options menu credits"
|
||||||
|
},
|
||||||
|
"setAside":
|
||||||
|
{
|
||||||
|
"message": "Reservar las pestañas actuales",
|
||||||
|
"description": "Save collection action name. Used in the pane, extension context menu and manifest shortcuts"
|
||||||
|
},
|
||||||
|
"nothingSaved":
|
||||||
|
{
|
||||||
|
"message": "No tienes pestañas reservadas",
|
||||||
|
"description": "Placeholder for empty pane"
|
||||||
|
},
|
||||||
|
"removeTab":
|
||||||
|
{
|
||||||
|
"message": "Eliminar pestañas de la colección",
|
||||||
|
"description": "Button hint on a tab card"
|
||||||
|
},
|
||||||
|
"restoreTabs":
|
||||||
|
{
|
||||||
|
"message": "Restaurar pestañas",
|
||||||
|
"description": "Collection restore action link name"
|
||||||
|
},
|
||||||
|
"more":
|
||||||
|
{
|
||||||
|
"message": "Más...",
|
||||||
|
"description": "Collections' more button title"
|
||||||
|
},
|
||||||
|
"restoreNoRemove":
|
||||||
|
{
|
||||||
|
"message": "Restaurar sin eliminar",
|
||||||
|
"description": "Context action item name"
|
||||||
|
},
|
||||||
|
"removeCollection":
|
||||||
|
{
|
||||||
|
"message": "Eliminar la colección",
|
||||||
|
"description": "Collection remove action name"
|
||||||
|
},
|
||||||
|
"removeCollectionConfirm":
|
||||||
|
{
|
||||||
|
"message": "¿Está seguro que quiere eliminar esta colección?",
|
||||||
|
"description": "Prompt dialog content on collection deletion"
|
||||||
|
},
|
||||||
|
"removeTabConfirm":
|
||||||
|
{
|
||||||
|
"message": "¿Está seguro que quiere eliminar esta pestaña?",
|
||||||
|
"description": "Prompt dialog content on one tab deletion"
|
||||||
|
},
|
||||||
|
"togglePaneContext":
|
||||||
|
{
|
||||||
|
"message": "Abrir panel de pestañas",
|
||||||
|
"description": "Context action name. Used in extension context menu and manifest shortcuts"
|
||||||
|
},
|
||||||
|
"noTabsToSave":
|
||||||
|
{
|
||||||
|
"message": "No hay pestañas disponibles para reservar",
|
||||||
|
"description": "Alert dialog message when there's no tabs to save"
|
||||||
|
},
|
||||||
|
"tabs":
|
||||||
|
{
|
||||||
|
"message": "items",
|
||||||
|
"description": "Collection tabs counter label (e.g. 8 items)"
|
||||||
|
}
|
||||||
|
}
|
||||||
+14
-44
@@ -6,7 +6,7 @@
|
|||||||
},
|
},
|
||||||
"description":
|
"description":
|
||||||
{
|
{
|
||||||
"message": "Функиця классического Microsoft Edge на браузерах Chromium",
|
"message": "Функиця отложенных вкладок классического Microsoft Edge",
|
||||||
"description": "Extension description"
|
"description": "Extension description"
|
||||||
},
|
},
|
||||||
"author":
|
"author":
|
||||||
@@ -19,14 +19,24 @@
|
|||||||
"message": "Настройки",
|
"message": "Настройки",
|
||||||
"description": "Alternative text for options button in the pane"
|
"description": "Alternative text for options button in the pane"
|
||||||
},
|
},
|
||||||
|
"closePanel":
|
||||||
|
{
|
||||||
|
"message": "Закрыть",
|
||||||
|
"description": "Alternative text for close panel button"
|
||||||
|
},
|
||||||
"loadOnRestore":
|
"loadOnRestore":
|
||||||
{
|
{
|
||||||
"message": "Загружать вкладки после открытия",
|
"message": "Загружать вкладки после открытия",
|
||||||
"description": "Label for option"
|
"description": "Label for option"
|
||||||
},
|
},
|
||||||
|
"showDeleteDialog":
|
||||||
|
{
|
||||||
|
"message": "Показывать окно подтверждения перед удалением элемента",
|
||||||
|
"description": "Label for option"
|
||||||
|
},
|
||||||
"swapIconAction":
|
"swapIconAction":
|
||||||
{
|
{
|
||||||
"message": "Откладывать вкладки при нажатии на иконку расширения (Alt+P или правая кнопка мыши для открытия панели)",
|
"message": "Откладывать вкладки при нажатии на иконку расширения (%TOGGLE_SHORTCUT% или правая кнопка мыши для открытия панели)",
|
||||||
"description": "Label for option"
|
"description": "Label for option"
|
||||||
},
|
},
|
||||||
"github":
|
"github":
|
||||||
@@ -111,47 +121,7 @@
|
|||||||
},
|
},
|
||||||
"tabs":
|
"tabs":
|
||||||
{
|
{
|
||||||
"message": "Вкладок",
|
"message": "вкладок",
|
||||||
"description": "Collection tabs counter label"
|
"description": "Collection tabs counter label (e.g. 8 items)"
|
||||||
},
|
|
||||||
"ago":
|
|
||||||
{
|
|
||||||
"message": "назад",
|
|
||||||
"description": "Human friendly timestamp part (e.g. 15 hour(s) ago)"
|
|
||||||
},
|
|
||||||
"minutes":
|
|
||||||
{
|
|
||||||
"message": "минут(ы)",
|
|
||||||
"description": "Human friendly timestamp part (e.g. 15 minute(s) ago)"
|
|
||||||
},
|
|
||||||
"hours":
|
|
||||||
{
|
|
||||||
"message": "час(ов)",
|
|
||||||
"description": "Human friendly timestamp part (e.g. 15 hour(s) ago)"
|
|
||||||
},
|
|
||||||
"days":
|
|
||||||
{
|
|
||||||
"message": "дней",
|
|
||||||
"description": "Human friendly timestamp part (e.g. 15 day(s) ago)"
|
|
||||||
},
|
|
||||||
"weeks":
|
|
||||||
{
|
|
||||||
"message": "недель",
|
|
||||||
"description": "Human friendly timestamp part (e.g. 15 week(s) ago)"
|
|
||||||
},
|
|
||||||
"months":
|
|
||||||
{
|
|
||||||
"message": "месяц(ев)",
|
|
||||||
"description": "Human friendly timestamp part (e.g. 15 month(s) ago)"
|
|
||||||
},
|
|
||||||
"years":
|
|
||||||
{
|
|
||||||
"message": "лет",
|
|
||||||
"description": "Human friendly timestamp part (e.g. 15 years(s) ago)"
|
|
||||||
},
|
|
||||||
"justNow":
|
|
||||||
{
|
|
||||||
"message": "Только что",
|
|
||||||
"description": "Human friendly timestamp part"
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -0,0 +1,127 @@
|
|||||||
|
{
|
||||||
|
"name":
|
||||||
|
{
|
||||||
|
"message": "搁置的标签页",
|
||||||
|
"description": "Extension name. Displayed in the manifest and pane header"
|
||||||
|
},
|
||||||
|
"description":
|
||||||
|
{
|
||||||
|
"message": "为Chromium浏览器提供旧版 Microsoft Edge 中的\"搁置标签页\" 功能",
|
||||||
|
"description": "Extension description"
|
||||||
|
},
|
||||||
|
"author":
|
||||||
|
{
|
||||||
|
"message": "Michael \"XFox\" Gordeev",
|
||||||
|
"description": "Author name"
|
||||||
|
},
|
||||||
|
"options":
|
||||||
|
{
|
||||||
|
"message": "设置",
|
||||||
|
"description": "Alternative text for options button in the pane"
|
||||||
|
},
|
||||||
|
"closePanel":
|
||||||
|
{
|
||||||
|
"message": "关闭",
|
||||||
|
"description": "Alternative text for close panel button"
|
||||||
|
},
|
||||||
|
"loadOnRestore":
|
||||||
|
{
|
||||||
|
"message": "在重新打开时加载页面",
|
||||||
|
"description": "Label for option"
|
||||||
|
},
|
||||||
|
"showDeleteDialog":
|
||||||
|
{
|
||||||
|
"message": "在删除项目之前显示确认对话框",
|
||||||
|
"description": "Label for option"
|
||||||
|
},
|
||||||
|
"swapIconAction":
|
||||||
|
{
|
||||||
|
"message": "点击拓展图标来搁置所有标签页 (按%TOGGLE_SHORTCUT%或右键来打开侧栏)",
|
||||||
|
"description": "Label for option"
|
||||||
|
},
|
||||||
|
"github":
|
||||||
|
{
|
||||||
|
"message": "查看GitHub页面",
|
||||||
|
"description": "Link title"
|
||||||
|
},
|
||||||
|
"contributors":
|
||||||
|
{
|
||||||
|
"message": "项目贡献者",
|
||||||
|
"description": "Link title"
|
||||||
|
},
|
||||||
|
"feedback":
|
||||||
|
{
|
||||||
|
"message": "给我们反馈",
|
||||||
|
"description": "Link title"
|
||||||
|
},
|
||||||
|
"buyMeACoffee":
|
||||||
|
{
|
||||||
|
"message": "给我买杯咖啡!",
|
||||||
|
"description": "Link title"
|
||||||
|
},
|
||||||
|
"credits":
|
||||||
|
{
|
||||||
|
"message": "由Michael 'XFox' Gordeev开发",
|
||||||
|
"description": "Options menu credits"
|
||||||
|
},
|
||||||
|
"setAside":
|
||||||
|
{
|
||||||
|
"message": "搁置当前的所有标签页",
|
||||||
|
"description": "Save collection action name. Used in the pane, extension context menu and manifest shortcuts"
|
||||||
|
},
|
||||||
|
"nothingSaved":
|
||||||
|
{
|
||||||
|
"message": "你目前没有搁置的标签页",
|
||||||
|
"description": "Placeholder for empty pane"
|
||||||
|
},
|
||||||
|
"removeTab":
|
||||||
|
{
|
||||||
|
"message": "从标签页集中移除标签页",
|
||||||
|
"description": "Button hint on a tab card"
|
||||||
|
},
|
||||||
|
"restoreTabs":
|
||||||
|
{
|
||||||
|
"message": "恢复标签页",
|
||||||
|
"description": "Collection restore action link name"
|
||||||
|
},
|
||||||
|
"more":
|
||||||
|
{
|
||||||
|
"message": "更多...",
|
||||||
|
"description": "Collections' more button title"
|
||||||
|
},
|
||||||
|
"restoreNoRemove":
|
||||||
|
{
|
||||||
|
"message": "恢复但不移除标签页",
|
||||||
|
"description": "Context action item name"
|
||||||
|
},
|
||||||
|
"removeCollection":
|
||||||
|
{
|
||||||
|
"message": "移除标签页集",
|
||||||
|
"description": "Collection remove action name"
|
||||||
|
},
|
||||||
|
"removeCollectionConfirm":
|
||||||
|
{
|
||||||
|
"message": "你确定要移除这个标签页集吗?",
|
||||||
|
"description": "Prompt dialog content on collection deletion"
|
||||||
|
},
|
||||||
|
"removeTabConfirm":
|
||||||
|
{
|
||||||
|
"message": "你确定要移除这个标签页吗?",
|
||||||
|
"description": "Prompt dialog content on one tab deletion"
|
||||||
|
},
|
||||||
|
"togglePaneContext":
|
||||||
|
{
|
||||||
|
"message": "打开或关闭侧栏",
|
||||||
|
"description": "Context action name. Used in extension context menu and manifest shortcuts"
|
||||||
|
},
|
||||||
|
"noTabsToSave":
|
||||||
|
{
|
||||||
|
"message": "没有可以搁置的标签页",
|
||||||
|
"description": "Alert dialog message when there's no tabs to save"
|
||||||
|
},
|
||||||
|
"tabs":
|
||||||
|
{
|
||||||
|
"message": "项",
|
||||||
|
"description": "Collection tabs counter label (e.g. 8 items)"
|
||||||
|
}
|
||||||
|
}
|
||||||
+99
-47
@@ -10,6 +10,17 @@
|
|||||||
color: black;
|
color: black;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.tabsAside.closeArea
|
||||||
|
{
|
||||||
|
position: fixed;
|
||||||
|
top: 0;
|
||||||
|
bottom: 0;
|
||||||
|
right: 0;
|
||||||
|
left: 0;
|
||||||
|
|
||||||
|
background-color: transparent;
|
||||||
|
}
|
||||||
|
|
||||||
.tabsAside.pane
|
.tabsAside.pane
|
||||||
{
|
{
|
||||||
user-select: none;
|
user-select: none;
|
||||||
@@ -18,7 +29,9 @@
|
|||||||
right: 0px;
|
right: 0px;
|
||||||
top: 0px;
|
top: 0px;
|
||||||
bottom: 0px;
|
bottom: 0px;
|
||||||
overflow: auto;
|
overflow: hidden;
|
||||||
|
display: grid;
|
||||||
|
grid-template-rows: auto 1fr;
|
||||||
|
|
||||||
width: 100%;
|
width: 100%;
|
||||||
min-width: 500px;
|
min-width: 500px;
|
||||||
@@ -36,7 +49,7 @@
|
|||||||
|
|
||||||
aside[embedded]
|
aside[embedded]
|
||||||
{
|
{
|
||||||
width: 40% !important;
|
width: 500px !important;
|
||||||
}
|
}
|
||||||
|
|
||||||
.tabsAside.pane[opened]
|
.tabsAside.pane[opened]
|
||||||
@@ -47,35 +60,27 @@
|
|||||||
/* Pane header*/
|
/* Pane header*/
|
||||||
.tabsAside.pane > header
|
.tabsAside.pane > header
|
||||||
{
|
{
|
||||||
margin: 20px 40px;
|
z-index: 1;
|
||||||
}
|
padding: 14px 20px 16px 20px;
|
||||||
.tabsAside.pane > header > div
|
box-shadow: 0px 0px 5px rgba(0,0,0,.5);
|
||||||
{
|
background-color: white;
|
||||||
display: grid;
|
display: grid;
|
||||||
grid-template-columns: 1fr auto;
|
grid-template-columns: 1fr auto auto;
|
||||||
|
grid-column-gap: 10px;
|
||||||
|
grid-row-gap: 30px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.tabsAside.pane > header > div > h1
|
.tabsAside.pane > header > h1
|
||||||
{
|
{
|
||||||
margin: 10px 0px;
|
margin: 0px 5px;
|
||||||
font-weight: normal;
|
font-weight: 500;
|
||||||
font-size: 21pt;
|
font-size: 15pt;
|
||||||
}
|
}
|
||||||
|
|
||||||
.tabsAside.pane > header > div > button
|
.tabsAside.pane > header > nav
|
||||||
{
|
{
|
||||||
margin: auto;
|
top: 45px;
|
||||||
}
|
right: 55px;
|
||||||
|
|
||||||
.tabsAside.pane > header > div > nav
|
|
||||||
{
|
|
||||||
top: 70px;
|
|
||||||
right: 40px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.tabsAside.pane > header nav > div
|
|
||||||
{
|
|
||||||
box-shadow: 0px 4px 5px -2px rgba(100, 100, 100, .5);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.tabsAside.pane > header nav > p
|
.tabsAside.pane > header nav > p
|
||||||
@@ -88,25 +93,37 @@
|
|||||||
text-decoration: none;
|
text-decoration: none;
|
||||||
}
|
}
|
||||||
|
|
||||||
.iconArrowRight
|
.saveTabs
|
||||||
{
|
{
|
||||||
width: 13px;
|
display: inline-grid;
|
||||||
height: 13px;
|
grid-template-columns: 16px auto;
|
||||||
display: inline-block;
|
grid-column-gap: 15px;
|
||||||
background-repeat: no-repeat;
|
|
||||||
background-size: 13px;
|
font-weight: 600;
|
||||||
background-position: center;
|
margin-right: auto;
|
||||||
background-image: url("chrome-extension://__MSG_@@extension_id__/icons/arrowRight.svg");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.tabsAside.pane > header > hr
|
.iconArrowRight
|
||||||
{
|
{
|
||||||
border: 1px solid #8a8a8a;
|
width: 16px;
|
||||||
|
height: 16px;
|
||||||
|
display: inline-block;
|
||||||
|
background-repeat: no-repeat;
|
||||||
|
background-size: 16px;
|
||||||
|
background-position: center;
|
||||||
|
background-image: url("chrome-extension://__MSG_@@extension_id__/icons/arrowRight.svg"),
|
||||||
|
url("moz-extension://__MSG_@@extension_id__/icons/arrowRight.svg");
|
||||||
|
margin: 2px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tabsAside.pane section
|
||||||
|
{
|
||||||
|
overflow: auto;
|
||||||
}
|
}
|
||||||
|
|
||||||
.tabsAside.pane > section > h2
|
.tabsAside.pane > section > h2
|
||||||
{
|
{
|
||||||
margin: 0px 40px;
|
margin: 20px;
|
||||||
font-weight: normal;
|
font-weight: normal;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -116,11 +133,34 @@
|
|||||||
transition: .2s;
|
transition: .2s;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.collectionSet
|
||||||
|
{
|
||||||
|
background-color: white;
|
||||||
|
margin: 10px;
|
||||||
|
border-radius: 5px;
|
||||||
|
border: 1px solid #eee;
|
||||||
|
}
|
||||||
|
|
||||||
|
.collectionSet:hover
|
||||||
|
{
|
||||||
|
box-shadow: 0px 0px 5px rgba(0, 0, 0, .25);
|
||||||
|
}
|
||||||
|
|
||||||
|
.collectionSet .header > *
|
||||||
|
{
|
||||||
|
visibility: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
.collectionSet:hover .header > *
|
||||||
|
{
|
||||||
|
visibility: visible;
|
||||||
|
}
|
||||||
|
|
||||||
.collectionSet > .header
|
.collectionSet > .header
|
||||||
{
|
{
|
||||||
margin: 0px 20px;
|
margin: 10px 10px 0px 20px;
|
||||||
display: grid;
|
display: grid;
|
||||||
grid-template-columns: auto 1fr auto auto auto;
|
grid-template-columns: 1fr auto auto auto;
|
||||||
grid-column-gap: 10px;
|
grid-column-gap: 10px;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
}
|
}
|
||||||
@@ -128,11 +168,21 @@
|
|||||||
.collectionSet > .header > small
|
.collectionSet > .header > small
|
||||||
{
|
{
|
||||||
color: gray;
|
color: gray;
|
||||||
|
visibility: visible !important;
|
||||||
}
|
}
|
||||||
|
|
||||||
.collectionSet > .header > span
|
.collectionSet > .header > input
|
||||||
{
|
{
|
||||||
font-weight: 600;
|
margin: 0px;
|
||||||
|
visibility: visible !important;
|
||||||
|
font-weight: 500;
|
||||||
|
border: none;
|
||||||
|
background: transparent;
|
||||||
|
}
|
||||||
|
|
||||||
|
.collectionSet > .header > input:hover
|
||||||
|
{
|
||||||
|
border: 1px solid black;
|
||||||
}
|
}
|
||||||
|
|
||||||
.collectionSet > .header > a
|
.collectionSet > .header > a
|
||||||
@@ -149,8 +199,7 @@
|
|||||||
/* Tabs collection */
|
/* Tabs collection */
|
||||||
.collectionSet > .set
|
.collectionSet > .set
|
||||||
{
|
{
|
||||||
margin: 0px 0px 0px 20px;
|
padding: 5px 10px;
|
||||||
padding: 10px 40px 10px 20px;
|
|
||||||
white-space: nowrap;
|
white-space: nowrap;
|
||||||
overflow: auto;
|
overflow: auto;
|
||||||
}
|
}
|
||||||
@@ -172,21 +221,23 @@
|
|||||||
margin: 5px;
|
margin: 5px;
|
||||||
|
|
||||||
background-color: #c2c2c2;
|
background-color: #c2c2c2;
|
||||||
background-image: url("chrome-extension://__MSG_@@extension_id__/images/tab_thumbnail.png");
|
background-image: url("chrome-extension://__MSG_@@extension_id__/images/tab_thumbnail.png"),
|
||||||
|
url("chrome-extension://__MSG_@@extension_id__/images/tab_thumbnail.png");
|
||||||
background-size: cover;
|
background-size: cover;
|
||||||
background-position-x: center;
|
background-position-x: center;
|
||||||
|
|
||||||
display: inline-grid;
|
display: inline-grid;
|
||||||
grid-template-rows: 1fr auto;
|
grid-template-rows: 1fr auto;
|
||||||
|
|
||||||
box-shadow: 0px 0px 5px rgba(100, 100, 100, .5);
|
|
||||||
transition: .25s;
|
transition: .25s;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
|
|
||||||
|
border: 1px solid #eee;
|
||||||
|
border-radius: 5px;
|
||||||
}
|
}
|
||||||
.collectionSet > .set > div:hover
|
.collectionSet > .set > div:hover
|
||||||
{
|
{
|
||||||
filter: brightness(120%);
|
box-shadow: 0px 0px 5px rgba(100, 100, 100, .5);
|
||||||
box-shadow: 0px 0px 15px rgba(100, 100, 100, .5);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.collectionSet > .set > div > div
|
.collectionSet > .set > div > div
|
||||||
@@ -213,9 +264,10 @@
|
|||||||
{
|
{
|
||||||
width: 20px;
|
width: 20px;
|
||||||
height: 20px;
|
height: 20px;
|
||||||
margin: 10px;
|
margin: 8px;
|
||||||
|
|
||||||
background-image: url("chrome-extension://__MSG_@@extension_id__/images/tab_icon.png");
|
background-image: url("chrome-extension://__MSG_@@extension_id__/images/tab_icon.png"),
|
||||||
|
url("moz-extension://__MSG_@@extension_id__/images/tab_icon.png");
|
||||||
background-size: 20px;
|
background-size: 20px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
+37
-2
@@ -9,6 +9,16 @@
|
|||||||
color: white;
|
color: white;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.tabsAside[darkmode] .pane header .iconArrowRight
|
||||||
|
{
|
||||||
|
filter: invert();
|
||||||
|
}
|
||||||
|
|
||||||
|
.tabsAside[darkmode] .pane header
|
||||||
|
{
|
||||||
|
background-color: #3b3b3b;
|
||||||
|
}
|
||||||
|
|
||||||
.tabsAside[darkmode] .saveTabs > div
|
.tabsAside[darkmode] .saveTabs > div
|
||||||
{
|
{
|
||||||
filter: invert();
|
filter: invert();
|
||||||
@@ -28,6 +38,22 @@
|
|||||||
background-color: dimgray;
|
background-color: dimgray;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.tabsAside[darkmode] .collectionSet > .header > input
|
||||||
|
{
|
||||||
|
color: white;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tabsAside[darkmode] .collectionSet > .header > input:hover
|
||||||
|
{
|
||||||
|
border: 1px solid dimgray;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tabsAside[darkmode] .collectionSet > .header > input:focus
|
||||||
|
{
|
||||||
|
background: white;
|
||||||
|
color: black;
|
||||||
|
}
|
||||||
|
|
||||||
.tabsAside[darkmode] a
|
.tabsAside[darkmode] a
|
||||||
{
|
{
|
||||||
color: #48adff;
|
color: #48adff;
|
||||||
@@ -50,11 +76,19 @@
|
|||||||
background: dimgray;
|
background: dimgray;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.tabsAside[darkmode] .pane .collectionSet
|
||||||
|
{
|
||||||
|
background-color: #3b3b3b;
|
||||||
|
border-color: #444;
|
||||||
|
}
|
||||||
|
|
||||||
/* Tab style */
|
/* Tab style */
|
||||||
.tabsAside[darkmode] .pane .collectionSet > .set > div
|
.tabsAside[darkmode] .pane .collectionSet > .set > div
|
||||||
{
|
{
|
||||||
background-color: #0c0c0c;
|
background-color: #0c0c0c;
|
||||||
background-image: url("chrome-extension://__MSG_@@extension_id__/images/tab_thumbnail_dark.png");
|
background-image: url("chrome-extension://__MSG_@@extension_id__/images/tab_thumbnail_dark.png"),
|
||||||
|
url("moz-extension://__MSG_@@extension_id__/images/tab_thumbnail_dark.png");
|
||||||
|
border-color: #444;
|
||||||
}
|
}
|
||||||
|
|
||||||
.tabsAside[darkmode] .pane .collectionSet > .set > div > div
|
.tabsAside[darkmode] .pane .collectionSet > .set > div > div
|
||||||
@@ -64,7 +98,8 @@
|
|||||||
|
|
||||||
.tabsAside[darkmode] .pane .collectionSet > .set > div > div > div
|
.tabsAside[darkmode] .pane .collectionSet > .set > div > div > div
|
||||||
{
|
{
|
||||||
background-image: url("chrome-extension://__MSG_@@extension_id__/images/tab_icon_dark.png");
|
background-image: url("chrome-extension://__MSG_@@extension_id__/images/tab_icon_dark.png"),
|
||||||
|
url("moz-extension://__MSG_@@extension_id__/images/tab_icon_dark.png");
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Context menu style */
|
/* Context menu style */
|
||||||
|
|||||||
+32
-10
@@ -17,7 +17,7 @@
|
|||||||
|
|
||||||
.tabsAside
|
.tabsAside
|
||||||
{
|
{
|
||||||
font-family: 'DefaultFont';
|
font-family: 'Segoe UI' ,'DefaultFont';
|
||||||
font-size: 14px;
|
font-size: 14px;
|
||||||
user-select: none;
|
user-select: none;
|
||||||
}
|
}
|
||||||
@@ -41,15 +41,15 @@
|
|||||||
/* Buttons style */
|
/* Buttons style */
|
||||||
.tabsAside button
|
.tabsAside button
|
||||||
{
|
{
|
||||||
width: 32px;
|
width: 28px;
|
||||||
height: 32px;
|
height: 28px;
|
||||||
background-color: transparent;
|
background-color: transparent;
|
||||||
border: none;
|
border: none;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
}
|
}
|
||||||
.tabsAside button:hover
|
.tabsAside button:hover
|
||||||
{
|
{
|
||||||
background-color: #c6c6c6;
|
background-color: #f2f2f2;
|
||||||
}
|
}
|
||||||
.tabsAside button:active
|
.tabsAside button:active
|
||||||
{
|
{
|
||||||
@@ -62,15 +62,23 @@
|
|||||||
user-select: none;
|
user-select: none;
|
||||||
|
|
||||||
position: absolute;
|
position: absolute;
|
||||||
width: 250px;
|
width: 290px;
|
||||||
|
|
||||||
box-shadow: 0px 0px 10px black;
|
box-shadow: 0px 0px 10px rgba(0,0,0,.5);
|
||||||
background-color: white;
|
background-color: white;
|
||||||
border-radius: 5px;
|
border-radius: 5px;
|
||||||
|
|
||||||
z-index: 10;
|
z-index: 10;
|
||||||
|
|
||||||
visibility: hidden;
|
visibility: hidden;
|
||||||
|
padding: 4px 0px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tabsAside nav hr
|
||||||
|
{
|
||||||
|
border: none;
|
||||||
|
height: 1px;
|
||||||
|
background-color: lightgray;
|
||||||
}
|
}
|
||||||
|
|
||||||
.tabsAside nav button
|
.tabsAside nav button
|
||||||
@@ -79,6 +87,14 @@
|
|||||||
|
|
||||||
padding: 0px 10px;
|
padding: 0px 10px;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
|
|
||||||
|
height: 32px;
|
||||||
|
font-family: 'Segoe UI' ,'DefaultFont';
|
||||||
|
}
|
||||||
|
|
||||||
|
.tabsAside nav button:hover
|
||||||
|
{
|
||||||
|
background-color: #eeee;
|
||||||
}
|
}
|
||||||
|
|
||||||
.tabsAside button + nav:active,
|
.tabsAside button + nav:active,
|
||||||
@@ -91,18 +107,20 @@
|
|||||||
.btn
|
.btn
|
||||||
{
|
{
|
||||||
background-repeat: no-repeat;
|
background-repeat: no-repeat;
|
||||||
background-size: 15px;
|
background-size: 12px;
|
||||||
background-position: center;
|
background-position: center;
|
||||||
}
|
}
|
||||||
|
|
||||||
.btn.more
|
.btn.more
|
||||||
{
|
{
|
||||||
background-image: url("chrome-extension://__MSG_@@extension_id__/icons/more.svg");
|
background-image: url("chrome-extension://__MSG_@@extension_id__/icons/more.svg"),
|
||||||
|
url("moz-extension://__MSG_@@extension_id__/icons/more.svg");
|
||||||
}
|
}
|
||||||
|
|
||||||
.btn.remove
|
.btn.remove
|
||||||
{
|
{
|
||||||
background-image: url("chrome-extension://__MSG_@@extension_id__/icons/cancel.svg");
|
background-image: url("chrome-extension://__MSG_@@extension_id__/icons/cancel.svg"),
|
||||||
|
url("moz-extension://__MSG_@@extension_id__/icons/cancel.svg");
|
||||||
}
|
}
|
||||||
|
|
||||||
@font-face
|
@font-face
|
||||||
@@ -111,5 +129,9 @@
|
|||||||
src: local("Segoe UI"),
|
src: local("Segoe UI"),
|
||||||
url("chrome-extension://__MSG_@@extension_id__/fonts/WeblySleekUI/weblysleekuisemilight.ttf") format("truetype"),
|
url("chrome-extension://__MSG_@@extension_id__/fonts/WeblySleekUI/weblysleekuisemilight.ttf") format("truetype"),
|
||||||
url("chrome-extension://__MSG_@@extension_id__/fonts/WeblySleekUI/weblysleekuisemilight.woff") format("woff"),
|
url("chrome-extension://__MSG_@@extension_id__/fonts/WeblySleekUI/weblysleekuisemilight.woff") format("woff"),
|
||||||
url("chrome-extension://__MSG_@@extension_id__/fonts/WeblySleekUI/weblysleekuisemilight.woff2") format("woff2");
|
url("chrome-extension://__MSG_@@extension_id__/fonts/WeblySleekUI/weblysleekuisemilight.woff2") format("woff2"),
|
||||||
|
|
||||||
|
url("moz-extension://__MSG_@@extension_id__/fonts/WeblySleekUI/weblysleekuisemilight.ttf") format("truetype"),
|
||||||
|
url("moz-extension://__MSG_@@extension_id__/fonts/WeblySleekUI/weblysleekuisemilight.woff") format("woff"),
|
||||||
|
url("moz-extension://__MSG_@@extension_id__/fonts/WeblySleekUI/weblysleekuisemilight.woff2") format("woff2");
|
||||||
}
|
}
|
||||||
@@ -0,0 +1,58 @@
|
|||||||
|
.tabsAside[listview] .collectionSet > .header
|
||||||
|
{
|
||||||
|
margin-bottom: 5px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tabsAside[listview] .collectionSet > .set
|
||||||
|
{
|
||||||
|
max-height: 250px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tabsAside[listview] .collectionSet > .set > div
|
||||||
|
{
|
||||||
|
width: initial;
|
||||||
|
height: initial;
|
||||||
|
background-image: none !important;
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
|
||||||
|
.listviewSwitch
|
||||||
|
{
|
||||||
|
width: 20px;
|
||||||
|
height: 20px;
|
||||||
|
display: grid;
|
||||||
|
grid-row-gap: 2px;
|
||||||
|
grid-column-gap: 2px;
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
|
||||||
|
.listviewSwitch.tile
|
||||||
|
{
|
||||||
|
grid-template-columns: 1fr 1fr;
|
||||||
|
}
|
||||||
|
|
||||||
|
.listviewSwitch > div
|
||||||
|
{
|
||||||
|
border-radius: 1px;
|
||||||
|
background-color: #c2c2c2;
|
||||||
|
}
|
||||||
|
|
||||||
|
.listviewSwitch:hover > div
|
||||||
|
{
|
||||||
|
background-color: #a0a0aa;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tabsAside .listviewSwitch.tile > div
|
||||||
|
{
|
||||||
|
background-color: gray;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tabsAside[listview] .listviewSwitch.tile > div
|
||||||
|
{
|
||||||
|
background-color: #c2c2c2;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tabsAside[listview] .listviewSwitch.list > div
|
||||||
|
{
|
||||||
|
background-color: gray;
|
||||||
|
}
|
||||||
+110
-40
@@ -1,4 +1,4 @@
|
|||||||
if (window.location === window.parent.location && window.location.protocol != "chrome-extension:") // For open/close call
|
if (window.location === window.parent.location && !window.location.protocol.includes("-extension:")) // For open/close call
|
||||||
{
|
{
|
||||||
var iframe = document.querySelector("iframe.tabsAsideIframe");
|
var iframe = document.querySelector("iframe.tabsAsideIframe");
|
||||||
if (!iframe)
|
if (!iframe)
|
||||||
@@ -71,14 +71,20 @@ function Initialize()
|
|||||||
}
|
}
|
||||||
|
|
||||||
document.querySelector(".tabsAside .saveTabs").onclick = SetTabsAside;
|
document.querySelector(".tabsAside .saveTabs").onclick = SetTabsAside;
|
||||||
|
document.querySelector(".tabsAside header .btn.remove").addEventListener("click", () =>
|
||||||
|
chrome.runtime.sendMessage({ command: "togglePane" })
|
||||||
|
);
|
||||||
|
document.querySelector(".tabsAside.closeArea").addEventListener("click", () =>
|
||||||
|
chrome.runtime.sendMessage({ command: "togglePane" })
|
||||||
|
);
|
||||||
|
|
||||||
document.querySelector("nav > p > small").textContent = chrome.runtime.getManifest()["version"];
|
document.querySelector("nav > p > small").textContent = chrome.runtime.getManifest()["version"];
|
||||||
|
|
||||||
// Tabs dismiss option
|
// Tabs dismiss option
|
||||||
var loadOnRestoreCheckbox = document.querySelector("#loadOnRestore");
|
var loadOnRestoreCheckbox = document.querySelector("#loadOnRestore");
|
||||||
chrome.storage.sync.get(
|
chrome.storage.sync.get(
|
||||||
{ "loadOnRestore": false },
|
{ "loadOnRestore": true },
|
||||||
values => loadOnRestoreCheckbox.checked = values.loadOnRestore
|
values => loadOnRestoreCheckbox.checked = values?.loadOnRestore ?? true
|
||||||
);
|
);
|
||||||
chrome.storage.onChanged.addListener((changes, namespace) =>
|
chrome.storage.onChanged.addListener((changes, namespace) =>
|
||||||
{
|
{
|
||||||
@@ -98,7 +104,7 @@ function Initialize()
|
|||||||
var swapIconAction = document.querySelector("#swapIconAction");
|
var swapIconAction = document.querySelector("#swapIconAction");
|
||||||
chrome.storage.sync.get(
|
chrome.storage.sync.get(
|
||||||
{ "setAsideOnClick": false },
|
{ "setAsideOnClick": false },
|
||||||
values => swapIconAction.checked = values.setAsideOnClick
|
values => swapIconAction.checked = values?.setAsideOnClick ?? false
|
||||||
);
|
);
|
||||||
chrome.storage.onChanged.addListener((changes, namespace) =>
|
chrome.storage.onChanged.addListener((changes, namespace) =>
|
||||||
{
|
{
|
||||||
@@ -114,6 +120,62 @@ function Initialize()
|
|||||||
})
|
})
|
||||||
);
|
);
|
||||||
|
|
||||||
|
// Deletion confirmation dialog
|
||||||
|
var showDeleteDialog = document.querySelector("#showDeleteDialog");
|
||||||
|
chrome.storage.sync.get(
|
||||||
|
{ "showDeleteDialog": true },
|
||||||
|
values => showDeleteDialog.checked = values.showDeleteDialog
|
||||||
|
);
|
||||||
|
chrome.storage.onChanged.addListener((changes, namespace) =>
|
||||||
|
{
|
||||||
|
if (namespace == 'sync')
|
||||||
|
for (key in changes)
|
||||||
|
if (key === 'showDeleteDialog')
|
||||||
|
showDeleteDialog.checked = changes[key].newValue
|
||||||
|
});
|
||||||
|
showDeleteDialog.addEventListener("click", () =>
|
||||||
|
chrome.storage.sync.set(
|
||||||
|
{
|
||||||
|
"showDeleteDialog": showDeleteDialog.checked
|
||||||
|
})
|
||||||
|
);
|
||||||
|
|
||||||
|
// Collections view switch
|
||||||
|
chrome.storage.sync.get(
|
||||||
|
{ "listview": false },
|
||||||
|
values =>
|
||||||
|
{
|
||||||
|
if (values?.listview)
|
||||||
|
pane.setAttribute("listview", "");
|
||||||
|
}
|
||||||
|
);
|
||||||
|
document.querySelectorAll(".listviewSwitch").forEach(i =>
|
||||||
|
{
|
||||||
|
i.onclick = (args) =>
|
||||||
|
{
|
||||||
|
if (args.currentTarget.classList[1] == "list")
|
||||||
|
{
|
||||||
|
pane.setAttribute("listview", "");
|
||||||
|
chrome.storage.sync.set({ "listview": true });
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
pane.removeAttribute("listview");
|
||||||
|
chrome.storage.sync.set({ "listview": false });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
chrome.storage.onChanged.addListener((changes, namespace) =>
|
||||||
|
{
|
||||||
|
if (namespace == 'sync')
|
||||||
|
for (key in changes)
|
||||||
|
if (key === 'listview')
|
||||||
|
if (changes[key].newValue)
|
||||||
|
pane.setAttribute("listview", "");
|
||||||
|
else
|
||||||
|
pane.removeAttribute("listview");
|
||||||
|
});
|
||||||
|
|
||||||
document.querySelectorAll(".tabsAside.pane > header nav button").forEach(i =>
|
document.querySelectorAll(".tabsAside.pane > header nav button").forEach(i =>
|
||||||
i.onclick = () =>
|
i.onclick = () =>
|
||||||
{
|
{
|
||||||
@@ -142,6 +204,11 @@ function UpdateLocale()
|
|||||||
{
|
{
|
||||||
document.querySelectorAll("*[loc]").forEach(i => i.textContent = chrome.i18n.getMessage(i.getAttribute("loc")));
|
document.querySelectorAll("*[loc]").forEach(i => i.textContent = chrome.i18n.getMessage(i.getAttribute("loc")));
|
||||||
document.querySelectorAll("*[loc_alt]").forEach(i => i.title = chrome.i18n.getMessage(i.getAttribute("loc_alt")));
|
document.querySelectorAll("*[loc_alt]").forEach(i => i.title = chrome.i18n.getMessage(i.getAttribute("loc_alt")));
|
||||||
|
|
||||||
|
var swapActionsLabel = document.querySelector("label[loc=swapIconAction]");
|
||||||
|
chrome.runtime.sendMessage({ command: "getShortcuts" }, (shortcuts) =>
|
||||||
|
swapActionsLabel.textContent = swapActionsLabel.textContent.replace("%TOGGLE_SHORTCUT%", shortcuts.filter(i => i.name == "toggle-pane")[0].shortcut)
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
function AddCollection(collection)
|
function AddCollection(collection)
|
||||||
@@ -154,12 +221,12 @@ function AddCollection(collection)
|
|||||||
for (var i = 0; i < collection.links.length; i++)
|
for (var i = 0; i < collection.links.length; i++)
|
||||||
{
|
{
|
||||||
rawTabs +=
|
rawTabs +=
|
||||||
"<div title='" + collection.titles[i] + "'" + ((collection.thumbnails && collection.thumbnails[i]) ? " style='background-image: url(" + collection.thumbnails[i] + ")'" : "") + ">" +
|
"<div title='" + collection.titles[i] + "'" + ((collection.thumbnails && collection.thumbnails[i]) ? " style='background-image: url(" + collection.thumbnails[i] + ")'" : "") + " value='" + collection.links[i] + "'>" +
|
||||||
"<span class='openTab' value='" + collection.links[i] + "'></span>" +
|
//"<span class='openTab' value='" + collection.links[i] + "'></span>" +
|
||||||
"<div>" +
|
"<div>" +
|
||||||
"<div" + ((collection.icons[i] == 0 || collection.icons[i] == null) ? "" : " style='background-image: url(\"" + collection.icons[i] + "\")'") + "></div>" +
|
"<div" + ((collection.icons[i] == 0 || collection.icons[i] == null) ? "" : " style='background-image: url(\"" + collection.icons[i] + "\")'") + "></div>" +
|
||||||
"<span>" + collection.titles[i] + "</span>" +
|
"<span>" + collection.titles[i] + "</span>" +
|
||||||
"<button loc_alt='name' class='btn remove' title='Remove tab from collection'></button>" +
|
"<button loc_alt='removeTab' class='btn remove' title='Remove tab from collection'></button>" +
|
||||||
"</div>" +
|
"</div>" +
|
||||||
"</div>";
|
"</div>";
|
||||||
}
|
}
|
||||||
@@ -167,8 +234,7 @@ function AddCollection(collection)
|
|||||||
list.innerHTML +=
|
list.innerHTML +=
|
||||||
"<div class='collectionSet'>" +
|
"<div class='collectionSet'>" +
|
||||||
"<div class='header'>" +
|
"<div class='header'>" +
|
||||||
"<span>" + chrome.i18n.getMessage("tabs") + ": " + collection.links.length + "</span>" +
|
"<input type='text' value='" + (collection.name ?? new Date(collection.timestamp).toDateString()) + "'/>" +
|
||||||
"<small>" + GetAgo(collection.timestamp) + "</small>" +
|
|
||||||
"<a loc='restoreTabs' class='restoreCollection'>Restore tabs</a>" +
|
"<a loc='restoreTabs' class='restoreCollection'>Restore tabs</a>" +
|
||||||
"<div>" +
|
"<div>" +
|
||||||
"<button loc_alt='more' class='btn more' title='More...'></button>" +
|
"<button loc_alt='more' class='btn more' title='More...'></button>" +
|
||||||
@@ -177,6 +243,7 @@ function AddCollection(collection)
|
|||||||
"</nav>" +
|
"</nav>" +
|
||||||
"</div>" +
|
"</div>" +
|
||||||
"<button loc_alt='removeCollection' class='btn remove' title='Remove collection'></button>" +
|
"<button loc_alt='removeCollection' class='btn remove' title='Remove collection'></button>" +
|
||||||
|
"<small>" + collection.links.length + " " + chrome.i18n.getMessage("tabs") +"</small>" +
|
||||||
"</div>" +
|
"</div>" +
|
||||||
|
|
||||||
"<div class='set' class='tabsList'>" + rawTabs + "</div>" +
|
"<div class='set' class='tabsList'>" + rawTabs + "</div>" +
|
||||||
@@ -184,26 +251,33 @@ function AddCollection(collection)
|
|||||||
|
|
||||||
UpdateLocale();
|
UpdateLocale();
|
||||||
|
|
||||||
|
list.querySelectorAll("input").forEach(i =>
|
||||||
|
i.oninput = (event) => RenameCollection(i.parentElement.parentElement, event.target.value));
|
||||||
|
|
||||||
list.querySelectorAll(".restoreCollection").forEach(i =>
|
list.querySelectorAll(".restoreCollection").forEach(i =>
|
||||||
i.onclick = () => RestoreTabs(i.parentElement.parentElement));
|
i.onclick = () => RestoreTabs(i.parentElement.parentElement));
|
||||||
|
|
||||||
list.querySelectorAll(".restoreCollection.noDelete").forEach(i =>
|
list.querySelectorAll(".restoreCollection.noDelete").forEach(i =>
|
||||||
i.onclick = () => RestoreTabs(i.parentElement.parentElement.parentElement.parentElement, false));
|
i.onclick = () => RestoreTabs(i.parentElement.parentElement.parentElement.parentElement, false));
|
||||||
|
|
||||||
list.querySelectorAll(".openTab").forEach(i =>
|
list.querySelectorAll(".set > div").forEach(i =>
|
||||||
i.onclick = () =>
|
i.onclick = (args) =>
|
||||||
|
{
|
||||||
|
if (args.target.localName != "button")
|
||||||
chrome.runtime.sendMessage(
|
chrome.runtime.sendMessage(
|
||||||
{
|
{
|
||||||
command: "openTab",
|
command: "openTab",
|
||||||
url: i.getAttribute("value")
|
url: i.getAttribute("value")
|
||||||
}
|
}
|
||||||
));
|
);
|
||||||
|
});
|
||||||
|
|
||||||
document.querySelectorAll(".btn.remove").forEach(i =>
|
document.querySelectorAll(".header .btn.remove").forEach(i =>
|
||||||
i.onclick = () => RemoveTabs(i.parentElement.parentElement));
|
i.onclick = () => RemoveTabs(i.parentElement.parentElement));
|
||||||
|
|
||||||
document.querySelectorAll(".tabsList .btn.remove").forEach(i =>
|
document.querySelectorAll(".set .btn.remove").forEach(i =>
|
||||||
i.onclick = () => RemoveOneTab(i.parentElement.parentElement));
|
i.onclick = (args) =>
|
||||||
|
RemoveOneTab(i.parentElement.parentElement));
|
||||||
}
|
}
|
||||||
|
|
||||||
function SetTabsAside()
|
function SetTabsAside()
|
||||||
@@ -211,6 +285,16 @@ function SetTabsAside()
|
|||||||
chrome.runtime.sendMessage({ command: "saveTabs" });
|
chrome.runtime.sendMessage({ command: "saveTabs" });
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function RenameCollection(collectionData, name)
|
||||||
|
{
|
||||||
|
chrome.runtime.sendMessage(
|
||||||
|
{
|
||||||
|
command: "renameCollection",
|
||||||
|
newName: name,
|
||||||
|
collectionIndex: Array.prototype.slice.call(collectionData.parentElement.children).indexOf(collectionData) - 1
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
function RestoreTabs(collectionData, removeCollection = true)
|
function RestoreTabs(collectionData, removeCollection = true)
|
||||||
{
|
{
|
||||||
chrome.runtime.sendMessage(
|
chrome.runtime.sendMessage(
|
||||||
@@ -229,7 +313,9 @@ function RestoreTabs(collectionData, removeCollection = true)
|
|||||||
|
|
||||||
function RemoveTabs(collectionData)
|
function RemoveTabs(collectionData)
|
||||||
{
|
{
|
||||||
if (!confirm(chrome.i18n.getMessage("removeCollectionConfirm")))
|
chrome.storage.sync.get({ "showDeleteDialog": true }, values =>
|
||||||
|
{
|
||||||
|
if (values.showDeleteDialog && !confirm(chrome.i18n.getMessage("removeCollectionConfirm")))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
chrome.runtime.sendMessage(
|
chrome.runtime.sendMessage(
|
||||||
@@ -239,11 +325,14 @@ function RemoveTabs(collectionData)
|
|||||||
},
|
},
|
||||||
() => RemoveCollectionElement(collectionData)
|
() => RemoveCollectionElement(collectionData)
|
||||||
);
|
);
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
function RemoveOneTab(tabData)
|
function RemoveOneTab(tabData)
|
||||||
{
|
{
|
||||||
if (!confirm(chrome.i18n.getMessage("removeTabConfirm")))
|
chrome.storage.sync.get({ "showDeleteDialog": true }, values =>
|
||||||
|
{
|
||||||
|
if (values.showDeleteDialog && !confirm(chrome.i18n.getMessage("removeTabConfirm")))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
chrome.runtime.sendMessage(
|
chrome.runtime.sendMessage(
|
||||||
@@ -254,7 +343,7 @@ function RemoveOneTab(tabData)
|
|||||||
},
|
},
|
||||||
() =>
|
() =>
|
||||||
{
|
{
|
||||||
tabData.parentElement.previousElementSibling.children[0].textContent = chrome.i18n.getMessage("tabs") + ": " + (tabData.parentElement.children.length - 1);
|
tabData.parentElement.previousElementSibling.querySelector("small").textContent = (tabData.parentElement.children.length - 1) + " " + chrome.i18n.getMessage("tabs");
|
||||||
if (tabData.parentElement.children.length < 2)
|
if (tabData.parentElement.children.length < 2)
|
||||||
{
|
{
|
||||||
RemoveElement(tabData.parentElement.parentElement);
|
RemoveElement(tabData.parentElement.parentElement);
|
||||||
@@ -264,26 +353,7 @@ function RemoveOneTab(tabData)
|
|||||||
else
|
else
|
||||||
RemoveElement(tabData);
|
RemoveElement(tabData);
|
||||||
});
|
});
|
||||||
}
|
});
|
||||||
|
|
||||||
function GetAgo(timestamp)
|
|
||||||
{
|
|
||||||
var minutes = (Date.now() - timestamp) / 60000;
|
|
||||||
|
|
||||||
if (minutes < 1)
|
|
||||||
return chrome.i18n.getMessage("justNow");
|
|
||||||
else if (minutes < 60)
|
|
||||||
return Math.floor(minutes) + " " + chrome.i18n.getMessage("minutes") + " " + chrome.i18n.getMessage("ago");
|
|
||||||
else if (minutes < 24 * 60)
|
|
||||||
return Math.floor(minutes / 60) + " " + chrome.i18n.getMessage("hours") + " " + chrome.i18n.getMessage("ago");
|
|
||||||
else if (minutes < 7 * 24 * 60)
|
|
||||||
return Math.floor(minutes / 24 / 60) + " " + chrome.i18n.getMessage("days") + " " + chrome.i18n.getMessage("ago");
|
|
||||||
else if (minutes < 30 * 24 * 60)
|
|
||||||
return Math.floor(minutes / 7 / 24 / 60) + " " + chrome.i18n.getMessage("weeks") + " " + chrome.i18n.getMessage("ago");
|
|
||||||
else if (minutes < 365 * 24 * 60)
|
|
||||||
return Math.floor(minutes / 30 / 24 / 60) + " " + chrome.i18n.getMessage("months") + " " + chrome.i18n.getMessage("ago");
|
|
||||||
else
|
|
||||||
return Math.floor(minutes / 365 / 24 / 60) + " " + chrome.i18n.getMessage("years") + " " + chrome.i18n.getMessage("ago");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function RemoveElement(el)
|
function RemoveElement(el)
|
||||||
@@ -294,7 +364,7 @@ function RemoveElement(el)
|
|||||||
|
|
||||||
function RemoveCollectionElement(el)
|
function RemoveCollectionElement(el)
|
||||||
{
|
{
|
||||||
RemoveElement(el);
|
if (el.parentElement.children.length < 3)
|
||||||
if (el.parentElement.children.length < 2)
|
|
||||||
setTimeout(() => document.querySelector(".tabsAside.pane > section > h2").removeAttribute("hidden"), 250);
|
setTimeout(() => document.querySelector(".tabsAside.pane > section > h2").removeAttribute("hidden"), 250);
|
||||||
|
RemoveElement(el);
|
||||||
}
|
}
|
||||||
+73
-21
@@ -1,7 +1,26 @@
|
|||||||
|
//This variable is populated when the browser action icon is clicked, or a command is called (with a shortcut for example).
|
||||||
|
//We can't populate it later, as selected tabs get deselected on a click inside a tab.
|
||||||
|
var tabsToSave = [];
|
||||||
|
|
||||||
|
|
||||||
|
//Get the tabs to save, either all the window or the selected tabs only, and pass them through a callback.
|
||||||
|
function GetTabsToSave(callback)
|
||||||
|
{
|
||||||
|
chrome.tabs.query({ currentWindow: true }, (windowTabs) =>
|
||||||
|
{
|
||||||
|
var highlightedTabs = windowTabs.filter(item => item.highlighted);
|
||||||
|
//If there are more than one selected tab in the window, we set only those aside.
|
||||||
|
// Otherwise, all the window's tabs get saved.
|
||||||
|
return callback((highlightedTabs.length > 1 ? highlightedTabs : windowTabs));
|
||||||
|
});
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
function TogglePane(tab)
|
function TogglePane(tab)
|
||||||
{
|
{
|
||||||
if (tab.url.startsWith("http")
|
if (tab.url.startsWith("http")
|
||||||
&& !tab.url.includes("chrome.google.com")
|
&& !tab.url.includes("chrome.google.com")
|
||||||
|
&& !tab.url.includes("addons.mozilla.org")
|
||||||
&& !tab.url.includes("microsoftedge.microsoft.com"))
|
&& !tab.url.includes("microsoftedge.microsoft.com"))
|
||||||
{
|
{
|
||||||
chrome.tabs.executeScript(tab.id,
|
chrome.tabs.executeScript(tab.id,
|
||||||
@@ -42,6 +61,9 @@ function TogglePane(tab)
|
|||||||
|
|
||||||
function ProcessCommand(command)
|
function ProcessCommand(command)
|
||||||
{
|
{
|
||||||
|
GetTabsToSave((returnedTabs) =>
|
||||||
|
{
|
||||||
|
tabsToSave = returnedTabs;
|
||||||
switch(command)
|
switch(command)
|
||||||
{
|
{
|
||||||
case "set-aside":
|
case "set-aside":
|
||||||
@@ -57,40 +79,53 @@ function ProcessCommand(command)
|
|||||||
)
|
)
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
chrome.browserAction.onClicked.addListener((tab) =>
|
chrome.browserAction.onClicked.addListener((tab) =>
|
||||||
{
|
{
|
||||||
|
GetTabsToSave((returnedTabs) =>
|
||||||
|
{
|
||||||
|
tabsToSave = returnedTabs;
|
||||||
|
|
||||||
chrome.storage.sync.get({ "setAsideOnClick": false }, values =>
|
chrome.storage.sync.get({ "setAsideOnClick": false }, values =>
|
||||||
{
|
{
|
||||||
if (values.setAsideOnClick)
|
if (values?.setAsideOnClick)
|
||||||
SaveCollection();
|
SaveCollection();
|
||||||
else
|
else
|
||||||
TogglePane(tab);
|
TogglePane(tab);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
var collections = JSON.parse(localStorage.getItem("sets")) || [];
|
||||||
|
var shortcuts;
|
||||||
|
chrome.commands.getAll((commands) => shortcuts = commands);
|
||||||
|
|
||||||
|
chrome.commands.onCommand.addListener(ProcessCommand);
|
||||||
|
chrome.contextMenus.onClicked.addListener((info) => ProcessCommand(info.menuItemId));
|
||||||
|
|
||||||
|
chrome.runtime.onInstalled.addListener((reason) =>
|
||||||
|
{
|
||||||
|
chrome.tabs.create({ url: "https://github.com/XFox111/TabsAsideExtension/releases/latest" });
|
||||||
// Adding context menu options
|
// Adding context menu options
|
||||||
chrome.contextMenus.create(
|
chrome.contextMenus.create(
|
||||||
{
|
{
|
||||||
id: "toggle-pane",
|
id: "toggle-pane",
|
||||||
contexts: ['all'],
|
contexts: ["browser_action"],
|
||||||
title: chrome.i18n.getMessage("togglePaneContext")
|
title: chrome.i18n.getMessage("togglePaneContext")
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
chrome.contextMenus.create(
|
chrome.contextMenus.create(
|
||||||
{
|
{
|
||||||
id: "set-aside",
|
id: "set-aside",
|
||||||
contexts: ['all'],
|
contexts: ["browser_action"],
|
||||||
title: chrome.i18n.getMessage("setAside")
|
title: chrome.i18n.getMessage("setAside")
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
});
|
||||||
|
|
||||||
var collections = JSON.parse(localStorage.getItem("sets")) || [];
|
//We receive a message from the pane aside-script, which means the tabsToSave are already assigned on message reception.
|
||||||
|
|
||||||
chrome.commands.onCommand.addListener(ProcessCommand);
|
|
||||||
chrome.contextMenus.onClicked.addListener((info) => ProcessCommand(info.menuItemId));
|
|
||||||
|
|
||||||
chrome.runtime.onMessage.addListener((message, sender, sendResponse) =>
|
chrome.runtime.onMessage.addListener((message, sender, sendResponse) =>
|
||||||
{
|
{
|
||||||
switch (message.command)
|
switch (message.command)
|
||||||
@@ -116,12 +151,34 @@ chrome.runtime.onMessage.addListener((message, sender, sendResponse) =>
|
|||||||
RemoveTab(message.collectionIndex, message.tabIndex);
|
RemoveTab(message.collectionIndex, message.tabIndex);
|
||||||
sendResponse();
|
sendResponse();
|
||||||
break;
|
break;
|
||||||
|
case "renameCollection":
|
||||||
|
collections[message.collectionIndex].name = message.newName;
|
||||||
|
localStorage.setItem("sets", JSON.stringify(collections));
|
||||||
|
break;
|
||||||
|
case "togglePane":
|
||||||
|
chrome.tabs.query(
|
||||||
|
{
|
||||||
|
active: true,
|
||||||
|
currentWindow: true
|
||||||
|
},
|
||||||
|
(tabs) => TogglePane(tabs[0])
|
||||||
|
)
|
||||||
|
break;
|
||||||
|
case "getShortcuts":
|
||||||
|
sendResponse(shortcuts);
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
// This function updates the extension's toolbar icon
|
// This function updates the extension's toolbar icon
|
||||||
function UpdateTheme()
|
function UpdateTheme()
|
||||||
{
|
{
|
||||||
|
// Updating badge counter
|
||||||
|
chrome.browserAction.setBadgeText({ text: collections.length < 1 ? "" : collections.length.toString() });
|
||||||
|
|
||||||
|
if (chrome.theme) // Firefox sets theme automatically
|
||||||
|
return;
|
||||||
|
|
||||||
var theme = window.matchMedia("(prefers-color-scheme: dark)").matches ? "dark" : "light";
|
var theme = window.matchMedia("(prefers-color-scheme: dark)").matches ? "dark" : "light";
|
||||||
var iconStatus = collections.length ? "full" : "empty";
|
var iconStatus = collections.length ? "full" : "empty";
|
||||||
|
|
||||||
@@ -137,12 +194,6 @@ function UpdateTheme()
|
|||||||
"16": basePath + "16.png"
|
"16": basePath + "16.png"
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
// Updating badge counter
|
|
||||||
if (collections.length < 1)
|
|
||||||
chrome.browserAction.setBadgeText({ });
|
|
||||||
else
|
|
||||||
chrome.browserAction.setBadgeText({ text: collections.length.toString() });
|
|
||||||
}
|
}
|
||||||
|
|
||||||
UpdateTheme();
|
UpdateTheme();
|
||||||
@@ -153,9 +204,7 @@ chrome.tabs.onActivated.addListener(UpdateTheme);
|
|||||||
// Set current tabs aside
|
// Set current tabs aside
|
||||||
function SaveCollection()
|
function SaveCollection()
|
||||||
{
|
{
|
||||||
chrome.tabs.query({ currentWindow: true }, (rawTabs) =>
|
var tabs = tabsToSave.filter(i => i.url != chrome.runtime.getURL("TabsAside.html") && !i.pinned && !i.url.includes("//newtab") && !i.url.includes("about:blank") && !i.url.includes("about:home"));
|
||||||
{
|
|
||||||
var tabs = rawTabs.filter(i => i.url != chrome.runtime.getURL("TabsAside.html") && !i.pinned && !i.url.includes("//newtab"));
|
|
||||||
|
|
||||||
if (tabs.length < 1)
|
if (tabs.length < 1)
|
||||||
{
|
{
|
||||||
@@ -190,11 +239,10 @@ function SaveCollection()
|
|||||||
chrome.tabs.create({}, (tab) =>
|
chrome.tabs.create({}, (tab) =>
|
||||||
{
|
{
|
||||||
newTabId = tab.id;
|
newTabId = tab.id;
|
||||||
chrome.tabs.remove(rawTabs.filter(i => !i.pinned && i.id != newTabId).map(tab => tab.id));
|
chrome.tabs.remove(tabsToSave.filter(i => !i.pinned && i.id != newTabId).map(tab => tab.id));
|
||||||
});
|
});
|
||||||
|
|
||||||
UpdateTheme();
|
UpdateTheme();
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function DeleteCollection(collectionIndex)
|
function DeleteCollection(collectionIndex)
|
||||||
@@ -216,9 +264,9 @@ function RestoreCollection(collectionIndex, removeCollection)
|
|||||||
},
|
},
|
||||||
(createdTab) =>
|
(createdTab) =>
|
||||||
{
|
{
|
||||||
chrome.storage.sync.get({ "loadOnRestore" : false }, values =>
|
chrome.storage.sync.get({ "loadOnRestore" : true }, values =>
|
||||||
{
|
{
|
||||||
if (!values.loadOnRestore)
|
if (!(values?.loadOnRestore))
|
||||||
chrome.tabs.onUpdated.addListener(function DiscardTab(updatedTabId, changeInfo, updatedTab)
|
chrome.tabs.onUpdated.addListener(function DiscardTab(updatedTabId, changeInfo, updatedTab)
|
||||||
{
|
{
|
||||||
if (updatedTabId === createdTab.id) {
|
if (updatedTabId === createdTab.id) {
|
||||||
@@ -232,6 +280,10 @@ function RestoreCollection(collectionIndex, removeCollection)
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
//We added new tabs by restoring a collection, so we refresh the array of tabs ready to be saved.
|
||||||
|
GetTabsToSave((returnedTabs) =>
|
||||||
|
tabsToSave = returnedTabs)
|
||||||
|
|
||||||
if (!removeCollection)
|
if (!removeCollection)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
|||||||
+41
-5
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "__MSG_name__",
|
"name": "__MSG_name__",
|
||||||
"version": "1.6",
|
"version": "1.9",
|
||||||
"manifest_version": 2,
|
"manifest_version": 2,
|
||||||
"description": "__MSG_description__",
|
"description": "__MSG_description__",
|
||||||
"author": "__MSG_author__",
|
"author": "__MSG_author__",
|
||||||
@@ -22,7 +22,33 @@
|
|||||||
"32": "icons/light/empty/32.png",
|
"32": "icons/light/empty/32.png",
|
||||||
"16": "icons/light/empty/16.png"
|
"16": "icons/light/empty/16.png"
|
||||||
},
|
},
|
||||||
"browser_action": { "default_icon": "icons/light/empty/32.png" },
|
"browser_action":
|
||||||
|
{
|
||||||
|
"default_icon": "icons/light/empty/32.png",
|
||||||
|
"theme_icons":
|
||||||
|
[
|
||||||
|
{
|
||||||
|
"size": 128,
|
||||||
|
"dark": "icons/light/full/128.png",
|
||||||
|
"light": "icons/dark/full/128.png"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"size": 48,
|
||||||
|
"dark": "icons/light/full/48.png",
|
||||||
|
"light": "icons/dark/full/48.png"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"size": 32,
|
||||||
|
"dark": "icons/light/full/32.png",
|
||||||
|
"light": "icons/dark/full/32.png"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"size": 16,
|
||||||
|
"dark": "icons/light/full/16.png",
|
||||||
|
"light": "icons/dark/full/16.png"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
"web_accessible_resources": [ "*" ],
|
"web_accessible_resources": [ "*" ],
|
||||||
|
|
||||||
"background":
|
"background":
|
||||||
@@ -38,7 +64,8 @@
|
|||||||
"description": "__MSG_setAside__",
|
"description": "__MSG_setAside__",
|
||||||
"suggested_key":
|
"suggested_key":
|
||||||
{
|
{
|
||||||
"default": "Alt+Left"
|
"default": "Alt+Left",
|
||||||
|
"mac": "MacCtrl+T"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"toggle-pane":
|
"toggle-pane":
|
||||||
@@ -46,8 +73,17 @@
|
|||||||
"description": "__MSG_togglePaneContext__",
|
"description": "__MSG_togglePaneContext__",
|
||||||
"suggested_key":
|
"suggested_key":
|
||||||
{
|
{
|
||||||
"default": "Alt+P"
|
"default": "Alt+P",
|
||||||
}
|
"mac": "Command+Shift+P"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"browser_specific_settings":
|
||||||
|
{
|
||||||
|
"gecko":
|
||||||
|
{
|
||||||
|
"id": "tabsaside@xfox111.net",
|
||||||
|
"strict_min_version": "58.0"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Reference in New Issue
Block a user