Compare commits
79 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
cfedcd724c | ||
|
|
fd28eaad06 | ||
|
|
8ce06facbd | ||
|
|
1a9db31c77 | ||
|
|
ac70b063b8 | ||
|
|
d0019c2c9b | ||
|
|
4c68408f1e | ||
|
|
1072505219 | ||
|
|
07fc555324 | ||
|
|
50fb29038a | ||
|
|
3785c623f8 | ||
|
|
14be3017c5 | ||
|
|
978bab0b81 | ||
|
|
d9d7f62d81 | ||
|
|
11e0613165 | ||
|
|
77c3d2bbb8 | ||
|
|
784a67afff | ||
|
|
19a092dd31 | ||
|
|
4c2f74949b | ||
|
|
a3fc3782ef | ||
|
|
cdc93f032a | ||
|
|
7dd15ed064 | ||
|
|
d169434157 | ||
|
|
6efed72bf2 | ||
|
|
15db311801 | ||
|
|
82d509129b | ||
|
|
939d838e35 | ||
|
|
6de4ce58c4 | ||
|
|
ee66196d9a | ||
|
|
3c13a9e837 | ||
|
|
22b23a9ece | ||
|
|
4ae385b7fc | ||
|
|
d9abc7f0b2 | ||
|
|
1f19b2528a | ||
|
|
1f11c39773 | ||
|
|
b6ccb5927a | ||
|
|
1d465ee318 | ||
|
|
3ab48b1cf5 | ||
|
|
de4ac978bd | ||
|
|
8df5faf4ef | ||
|
|
99a2035fdb | ||
|
|
a0d61728ea | ||
|
|
312bf6102c | ||
|
|
f4885b6f1c | ||
|
|
ca63a0979e | ||
|
|
1f266c3353 | ||
|
|
c7b2a70312 | ||
|
|
255133af4d | ||
|
|
db74531cd4 | ||
|
|
f36d8bfc78 | ||
|
|
3b31ace726 | ||
|
|
6badfef9da | ||
|
|
8c38dd73d8 | ||
|
|
b8682a852a | ||
|
|
8c17929151 | ||
|
|
bb92c9ec28 | ||
|
|
b4aacea2a3 | ||
|
|
8bbe6ebf75 | ||
|
|
a23d28f2cf | ||
|
|
f51e8859ee | ||
|
|
d235dee955 | ||
|
|
2afef4ea3d | ||
|
|
f709d6a566 | ||
|
|
532915b95d | ||
|
|
456e40bedf | ||
|
|
018b7e270f | ||
|
|
f8b8b4c97a | ||
|
|
978d7d0863 | ||
|
|
594a14d6ca | ||
|
|
c628aa9aef | ||
|
|
3060ebf79c | ||
|
|
1a34c7374b | ||
|
|
c262681011 | ||
|
|
f8ba5c46e4 | ||
|
|
b789250cb8 | ||
|
|
5df458739d | ||
|
|
d6fa9a2a03 | ||
|
|
ec15af01dd | ||
|
|
7073336f81 |
2
.github/workflows/quality-checks.yaml
vendored
2
.github/workflows/quality-checks.yaml
vendored
@@ -1,6 +1,6 @@
|
|||||||
name: Quality checks
|
name: Quality checks
|
||||||
|
|
||||||
on: push
|
on: [ push, pull_request ]
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
lint:
|
lint:
|
||||||
|
|||||||
1
.github/workflows/security-checks.yaml
vendored
1
.github/workflows/security-checks.yaml
vendored
@@ -2,6 +2,7 @@ name: Security checks
|
|||||||
|
|
||||||
on:
|
on:
|
||||||
push:
|
push:
|
||||||
|
pull_request:
|
||||||
schedule:
|
schedule:
|
||||||
- cron: '0 0 * * 0'
|
- cron: '0 0 * * 0'
|
||||||
|
|
||||||
|
|||||||
7
.github/workflows/test.yaml
vendored
7
.github/workflows/test.yaml
vendored
@@ -1,10 +1,13 @@
|
|||||||
name: Test
|
name: Test
|
||||||
|
|
||||||
on: push
|
on: [ push, pull_request ]
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
run-tests:
|
run-tests:
|
||||||
runs-on: ubuntu-latest
|
strategy:
|
||||||
|
matrix:
|
||||||
|
os: [macos, ubuntu, windows]
|
||||||
|
runs-on: ${{ matrix.os }}-latest
|
||||||
steps:
|
steps:
|
||||||
-
|
-
|
||||||
name: Checkout
|
name: Checkout
|
||||||
|
|||||||
2
.gitignore
vendored
2
.gitignore
vendored
@@ -1,5 +1,5 @@
|
|||||||
node_modules
|
node_modules
|
||||||
/dist
|
dist/
|
||||||
.vs
|
.vs
|
||||||
.vscode
|
.vscode
|
||||||
#Electron-builder output
|
#Electron-builder output
|
||||||
|
|||||||
106
CHANGELOG.md
106
CHANGELOG.md
@@ -1,5 +1,111 @@
|
|||||||
# Changelog
|
# Changelog
|
||||||
|
|
||||||
|
## 0.7.6 (2020-10-18)
|
||||||
|
|
||||||
|
* add docs for default0 pointing to github discussion (#30) | [commit](https://github.com/undergroundwires/privacy.sexy/commit/a3fc3782efd346b4c99d2a0b40df2eb0229f5b36)
|
||||||
|
* add robots.txt to explicitly allow indexing | [commit](https://github.com/undergroundwires/privacy.sexy/commit/4c2f74949b0758d33049bdfa4f0124a28958f8ea)
|
||||||
|
* add more reversibility | [commit](https://github.com/undergroundwires/privacy.sexy/commit/19a092dd31fb3588277f1ab3120b409d98506752)
|
||||||
|
* refactor to read more from package.json | [commit](https://github.com/undergroundwires/privacy.sexy/commit/784a67afff681bc19147d03c947de0e165d97e87)
|
||||||
|
* simplify "why" section | [commit](https://github.com/undergroundwires/privacy.sexy/commit/77c3d2bbb8d13db86bb82ed0b5cbeaacfdea3db9)
|
||||||
|
* update dependencies to latest | [commit](https://github.com/undergroundwires/privacy.sexy/commit/11e06131655398db08faeeacff62062e46e0dddd)
|
||||||
|
* run tests on all operating systems: macos, ubuntu, windows | [commit](https://github.com/undergroundwires/privacy.sexy/commit/d9d7f62d81d4d8f95104d33211e82641884d711f)
|
||||||
|
|
||||||
|
[compare](https://github.com/undergroundwires/privacy.sexy/compare/0.7.5...0.7.6)
|
||||||
|
|
||||||
|
## 0.7.5 (2020-09-14)
|
||||||
|
|
||||||
|
* fix reverting (reinstalling) capabilities not working | [commit](https://github.com/undergroundwires/privacy.sexy/commit/939d838e3535bb1c9b00c8ea9dacb735ae41d700)
|
||||||
|
* fix tests and checks are not running on PRs | [commit](https://github.com/undergroundwires/privacy.sexy/commit/82d509129b4e4a5df4b84786a0d6842a7d26e888)
|
||||||
|
* fix the recycling bin option (#32) | [commit](https://github.com/undergroundwires/privacy.sexy/commit/15db3118012a172a2191a2afad57084a65b34642)
|
||||||
|
* fix rendering issue in older edge/IE | [commit](https://github.com/undergroundwires/privacy.sexy/commit/6efed72bf25c2ddf0901caab7f22966ca13cd47a)
|
||||||
|
* fix pasting in search bar after page load showing no results | [commit](https://github.com/undergroundwires/privacy.sexy/commit/d1694341578288eeaf8b80caf9296a38d76789f0)
|
||||||
|
* fix typo | [commit](https://github.com/undergroundwires/privacy.sexy/commit/7dd15ed06433e0e6583ab0fa46a683ce6554bbea)
|
||||||
|
|
||||||
|
[compare](https://github.com/undergroundwires/privacy.sexy/compare/0.7.4...0.7.5)
|
||||||
|
|
||||||
|
## 0.7.4 (2020-09-12)
|
||||||
|
|
||||||
|
* fix checked checkbox has blue border | [commit](https://github.com/undergroundwires/privacy.sexy/commit/4ae385b7fcea9014a68442714b7d99e2ee7df7d0)
|
||||||
|
* fix spectre protection getting single lined #31 | [commit](https://github.com/undergroundwires/privacy.sexy/commit/22b23a9ece446c7f9abd4ede293051eb616ad50a)
|
||||||
|
* fix missing reg value in denying app access to account | [commit](https://github.com/undergroundwires/privacy.sexy/commit/3c13a9e837e06e097450b31d7eb0c0e6bf20cefb)
|
||||||
|
* fix wrong path in clear all firefox user profile settings | [commit](https://github.com/undergroundwires/privacy.sexy/commit/ee66196d9a60f27d17ae7f62d02b4f119a47e6e0)
|
||||||
|
|
||||||
|
[compare](https://github.com/undergroundwires/privacy.sexy/compare/0.7.3...0.7.4)
|
||||||
|
|
||||||
|
## 0.7.3 (2020-09-12)
|
||||||
|
|
||||||
|
* fix vscode settings file override and add more configs | [commit](https://github.com/undergroundwires/privacy.sexy/commit/a0d61728ead04b4455437f85820121a848db9e00)
|
||||||
|
* fix nvidia tweak error message, categorize and add reversibility | [commit](https://github.com/undergroundwires/privacy.sexy/commit/99a2035fdb0766a4dfc2753133eab0d7666516cd)
|
||||||
|
* improve CPU specific tweaks by conditional platform checks and reversibility | [commit](https://github.com/undergroundwires/privacy.sexy/commit/8df5faf4ef05a49da63973bd0fbb5c5d07d5bd93)
|
||||||
|
* fix wrong path to the main telemetry file | [commit](https://github.com/undergroundwires/privacy.sexy/commit/de4ac978bdda79573b36d355697b8a028d2c0beb)
|
||||||
|
* fix naming of firefox cleanup to mention profiles | [commit](https://github.com/undergroundwires/privacy.sexy/commit/3ab48b1cf5f7f934f07e468ef2318ccee07f530c)
|
||||||
|
* add reversibility and more scripts to denying app access with better structure | [commit](https://github.com/undergroundwires/privacy.sexy/commit/1d465ee3189d0e5a827453b3f0eb4361efe23770)
|
||||||
|
* fix comment lines are being detected as duplicate in validation | [commit](https://github.com/undergroundwires/privacy.sexy/commit/b6ccb5927a20412976a54fd2215eb645092f98a8)
|
||||||
|
* add more detailed error message | [commit](https://github.com/undergroundwires/privacy.sexy/commit/1f11c39773c12eccfb3efb898b58c2f6f37ab9ca)
|
||||||
|
* fix typo in a test | [commit](https://github.com/undergroundwires/privacy.sexy/commit/1f19b2528a69383e63e579d2885f01cd804abf6c)
|
||||||
|
|
||||||
|
[compare](https://github.com/undergroundwires/privacy.sexy/compare/0.7.2...0.7.3)
|
||||||
|
|
||||||
|
## 0.7.2 (2020-09-06)
|
||||||
|
|
||||||
|
* update onesync documentation and do not recommend it as it breaks other apps | [commit](https://github.com/undergroundwires/privacy.sexy/commit/f36d8bfc7848bb65ac0c641e318a689bf3816ccf)
|
||||||
|
* add reversibility for biometric disabling and do not recommend it | [commit](https://github.com/undergroundwires/privacy.sexy/commit/db74531cd4139615c6d595959217d3651f099019)
|
||||||
|
* fix bad highlighting of selected nodes when using keyboard navigation | [commit](https://github.com/undergroundwires/privacy.sexy/commit/255133af4dfae40171406648a3e2920f16d71cb3)
|
||||||
|
* add reversibility to removing bloatware | [commit](https://github.com/undergroundwires/privacy.sexy/commit/c7b2a703128470a05f12c9c6e8002444def37ef8)
|
||||||
|
* fix indeterminate state being lost | [commit](https://github.com/undergroundwires/privacy.sexy/commit/1f266c33535f72b69c65985bf2eff27cd2c5a104)
|
||||||
|
* fix wording in default text in text area | [commit](https://github.com/undergroundwires/privacy.sexy/commit/ca63a0979ef55d07d09d9443e5cea9aa888870a5)
|
||||||
|
* add best practice suggestion to come back | [commit](https://github.com/undergroundwires/privacy.sexy/commit/f4885b6f1c82752f2143934e336d6d1b1af03015)
|
||||||
|
|
||||||
|
[compare](https://github.com/undergroundwires/privacy.sexy/compare/0.7.1...0.7.2)
|
||||||
|
|
||||||
|
## 0.7.1 (2020-09-04)
|
||||||
|
|
||||||
|
* fix some browsers (including firefox) downloading the script as a text file | [commit](https://github.com/undergroundwires/privacy.sexy/commit/8c17929151f9c4fa5f48564492bbf400ced95eea)
|
||||||
|
* rename screenshot image file | [commit](https://github.com/undergroundwires/privacy.sexy/commit/b8682a852a14ed6cf49986695d9510b840ac9d3d)
|
||||||
|
* fix new/changed script higlighting not working on production builds | [commit](https://github.com/undergroundwires/privacy.sexy/commit/8c38dd73d8c7b77d8d341c0389f4d7229f9b97fd)
|
||||||
|
* refactor unused imports | [commit](https://github.com/undergroundwires/privacy.sexy/commit/6badfef9daace0c5de3fd33652a82bfe22261b11)
|
||||||
|
|
||||||
|
[compare](https://github.com/undergroundwires/privacy.sexy/compare/0.7.0...0.7.1)
|
||||||
|
|
||||||
|
## 0.7.0 (2020-09-02)
|
||||||
|
|
||||||
|
* [search] better (multilined) message when there are no results | [commit](https://github.com/undergroundwires/privacy.sexy/commit/ec15af01dd020b364c2174fe562fd66227c2320c)
|
||||||
|
* [search] added clear/close button | [commit](https://github.com/undergroundwires/privacy.sexy/commit/d6fa9a2a03c0ebe68b94f0b80cc52b4e200c9213)
|
||||||
|
* move script generation to /generation | [commit](https://github.com/undergroundwires/privacy.sexy/commit/5df458739d076719e350ba194c4f3f772884fcdb)
|
||||||
|
* add auto-highlighting of selected/updated code | [commit](https://github.com/undergroundwires/privacy.sexy/commit/b789250cb89e2130b08e1a927df8181cf945dfeb)
|
||||||
|
* prompt admin priviliges automatically | [commit](https://github.com/undergroundwires/privacy.sexy/commit/f8ba5c46e4923d9c35f200f8a08aa6437f7c0ecc)
|
||||||
|
* add removal of ghost (default0) telemetry user | [commit](https://github.com/undergroundwires/privacy.sexy/commit/c262681011f39b4412669b6cf233476f676ca550)
|
||||||
|
* add more windows defender tweaks, categorization and reversibility | [commit](https://github.com/undergroundwires/privacy.sexy/commit/1a34c7374ba56bafa0209bbb55c81b233bb419ed)
|
||||||
|
* fix NTP script documentation is on wrong place | [commit](https://github.com/undergroundwires/privacy.sexy/commit/3060ebf79cf242370433495cc3e1878b7581b202)
|
||||||
|
* updated dependencies to latest and audit fixes (#25) | [commit](https://github.com/undergroundwires/privacy.sexy/commit/c628aa9aef8ab7c815661d3c1711e7fbc65c69a2)
|
||||||
|
* categorize, fix and extend windows log files cleanup | [commit](https://github.com/undergroundwires/privacy.sexy/commit/594a14d6ca76cbd27a21877b8c373c1930589ca6)
|
||||||
|
* add more OneDrive cleanup scripts and categorize them | [commit](https://github.com/undergroundwires/privacy.sexy/commit/978d7d08638dd161082f239ed088b12302f29458)
|
||||||
|
* add disabling firefox telemetry | [commit](https://github.com/undergroundwires/privacy.sexy/commit/f8b8b4c97ab734d5ba7370894b694993924388da)
|
||||||
|
* add disabling ccleaner telemetry | [commit](https://github.com/undergroundwires/privacy.sexy/commit/018b7e270f207aac926cb12f8069ebfcdce193ce)
|
||||||
|
* Add disabling of PowerShell 7+ telemetry (#29) | [commit](https://github.com/undergroundwires/privacy.sexy/commit/456e40bedf9afcc846f9b13f1ea144cef6115cf6)
|
||||||
|
* categorize, fix, make scripts reversible in "UI for privacy", "security improvements" and "configure browsers" | [commit](https://github.com/undergroundwires/privacy.sexy/commit/532915b95da9fecd6b981d91bf489359e4e53caa)
|
||||||
|
* fix "Configure Defender" being in wrong category #28 | [commit](https://github.com/undergroundwires/privacy.sexy/commit/f709d6a566ed7846b677b383863deda9680a2a9c)
|
||||||
|
* do not hardcode capability versions and make them reversible | [commit](https://github.com/undergroundwires/privacy.sexy/commit/2afef4ea3d0d3d09aa1fa1eedba8493680bd8f10)
|
||||||
|
* exclude paint, wordpad and notepad from bloatware removal | [commit](https://github.com/undergroundwires/privacy.sexy/commit/d235dee95514a01745aef9479d07f88ffb4b40b8)
|
||||||
|
* add reversibility on category level | [commit](https://github.com/undergroundwires/privacy.sexy/commit/f51e8859eeb32c944126d692cfe03a0320c8b568)
|
||||||
|
* refactor unused imports & variables | [commit](https://github.com/undergroundwires/privacy.sexy/commit/a23d28f2cfa2d64d45460697cf5ee9d6b5920752)
|
||||||
|
* fix search (got broken in b789250) with tests and refactorings | [commit](https://github.com/undergroundwires/privacy.sexy/commit/8bbe6ebf750f1a1cbab493fb99b5ea91f4e21609)
|
||||||
|
* update the screenshot to show off highlighting | [commit](https://github.com/undergroundwires/privacy.sexy/commit/b4aacea2a3e0bbcf2d8a79ff67f51c0f19e888a6)
|
||||||
|
|
||||||
|
[compare](https://github.com/undergroundwires/privacy.sexy/compare/0.6.2...0.7.0)
|
||||||
|
|
||||||
|
## 0.6.2 (2020-08-16)
|
||||||
|
|
||||||
|
* 🐛 fixed disabling error reporting for november 2019 update | [commit](https://github.com/undergroundwires/privacy.sexy/commit/5967347b80976a519f6f4eb1972a62f3e600df2b)
|
||||||
|
* 🐛 fixed blank screen and icons on mac | [commit](https://github.com/undergroundwires/privacy.sexy/commit/7fac0fe79f252e8f9dda4f6f83cd6fa4ba2b539f)
|
||||||
|
* 🐛 fixed removing onedrive does not delete scheduled tasks | [commit](https://github.com/undergroundwires/privacy.sexy/commit/b6bfc2572740c0cd46d3bc0058fa767dd5fa862e)
|
||||||
|
* ⚙️ enhanced tweak to disable for office telemetry | [commit](https://github.com/undergroundwires/privacy.sexy/commit/afc3bfb3b8896f332c9a196973ded3dce8fd21e4)
|
||||||
|
* ✨ added script to clear dotnet telemery | [commit](https://github.com/undergroundwires/privacy.sexy/commit/1663bfeac7b6580b1335ca5fcf3587b69c080c72)
|
||||||
|
* 🐛 fixed changing time server not working | [commit](https://github.com/undergroundwires/privacy.sexy/commit/c69998c7cb29ffcf40f0af03b73150736581da69)
|
||||||
|
* 🔥 removed disabling ClickToRun as it breaks office | [commit](https://github.com/undergroundwires/privacy.sexy/commit/3d3380f27ebeea53f17f49974aaa89300ffaf2dd)
|
||||||
|
|
||||||
|
[compare](https://github.com/undergroundwires/privacy.sexy/compare/0.6.1...0.6.2)
|
||||||
|
|
||||||
## 0.6.1 (2020-08-09)
|
## 0.6.1 (2020-08-09)
|
||||||
|
|
||||||
* updated documentation | [commit](https://github.com/undergroundwires/privacy.sexy/commit/5963d2bac551083f9d16cce6b851abf0e8b88ce7)
|
* updated documentation | [commit](https://github.com/undergroundwires/privacy.sexy/commit/5963d2bac551083f9d16cce6b851abf0e8b88ce7)
|
||||||
|
|||||||
@@ -25,12 +25,6 @@
|
|||||||
|
|
||||||
## Guidelines
|
## Guidelines
|
||||||
|
|
||||||
### Extend scripts
|
|
||||||
|
|
||||||
- Create a [pull request](#Pull-Request-Process) for [application.yaml](./src/application/application.yaml)
|
|
||||||
- 🙏 For any new script, try to add `revertCode` that'll revert the changes caused by the script.
|
|
||||||
- See [typings](./src/application/application.yaml.d.ts) for documentation as code.
|
|
||||||
|
|
||||||
### Handle the state in presentation layer
|
### Handle the state in presentation layer
|
||||||
|
|
||||||
- There are two types of components:
|
- There are two types of components:
|
||||||
|
|||||||
25
README.md
25
README.md
@@ -15,25 +15,26 @@
|
|||||||
## Get started
|
## Get started
|
||||||
|
|
||||||
- Online version: [https://privacy.sexy](https://privacy.sexy)
|
- Online version: [https://privacy.sexy](https://privacy.sexy)
|
||||||
- or download latest desktop version for [Windows](https://github.com/undergroundwires/privacy.sexy/releases/download/0.6.1/privacy.sexy-Setup-0.6.1.exe), [Linux](https://github.com/undergroundwires/privacy.sexy/releases/download/0.6.1/privacy.sexy-0.6.1.AppImage), [macOS](https://github.com/undergroundwires/privacy.sexy/releases/download/0.6.1/privacy.sexy-0.6.1.dmg)
|
- or download latest desktop version for [Windows](https://github.com/undergroundwires/privacy.sexy/releases/download/0.7.6/privacy.sexy-Setup-0.7.6.exe), [Linux](https://github.com/undergroundwires/privacy.sexy/releases/download/0.7.6/privacy.sexy-0.7.6.AppImage), [macOS](https://github.com/undergroundwires/privacy.sexy/releases/download/0.7.6/privacy.sexy-0.7.6.dmg)
|
||||||
|
- 💡 Come back regularly to apply latest version for stronger privacy and security.
|
||||||
|
|
||||||

|
[](https://privacy.sexy)
|
||||||
|
|
||||||
## Why
|
## Why
|
||||||
|
|
||||||
- You don't need to run any compiled software that has access to your system, just run the generated scripts.
|
- Rich tweak pool to harden security & privacy of the OS and other software on it
|
||||||
- Have full visibility into what the tweaks do as you enable them.
|
- Free (both free as in beer and free as in speech)
|
||||||
- Ability to revert applied scripts
|
- No need to run any compiled software that has access to your system, just run the generated scripts
|
||||||
|
- Have full visibility into what the tweaks do as you enable them
|
||||||
|
- Ability to revert (undo) applied scripts
|
||||||
|
- Everything is transparent: both application and its infrastructure are open-source and automated
|
||||||
- Easily extendable
|
- Easily extendable
|
||||||
- Everything is open-sourced including both application and infrastructure
|
|
||||||
- Fully automated CI/CD pipeline using GitHub actions
|
|
||||||
- to AWS for provisioning serverless infrastructure
|
|
||||||
- for building and sharing the desktop applications
|
|
||||||
|
|
||||||
## Extend scripts
|
## Extend scripts
|
||||||
|
|
||||||
- Fork it & add more scripts in [application.yaml](src/application/application.yaml) and send a pull request 👌
|
- Fork it & add more scripts in [application.yaml](src/application/application.yaml) and send a pull request 👌
|
||||||
- 📖 More: [extend scripts | CONTRIBUTING.md](./CONTRIBUTING.md#extend-scripts)
|
- 📖 If you're unsure about the syntax you can refer to the [application file | documentation](docs/application-file.md).
|
||||||
|
- 🙏 For any new script, please add `revertCode` and `docs` values if possible.
|
||||||
|
|
||||||
## Commands
|
## Commands
|
||||||
|
|
||||||
@@ -48,8 +49,8 @@
|
|||||||
- Development: `npm run serve` to compile & hot-reload for development.
|
- Development: `npm run serve` to compile & hot-reload for development.
|
||||||
- Production: `npm run build` to prepare files for distribution.
|
- Production: `npm run build` to prepare files for distribution.
|
||||||
- Or run using Docker:
|
- Or run using Docker:
|
||||||
1. Build: `docker build -t undergroundwires/privacy.sexy:0.6.1 .`
|
1. Build: `docker build -t undergroundwires/privacy.sexy:0.7.6 .`
|
||||||
2. Run: `docker run -it -p 8080:80 --rm --name privacy.sexy-0.6.1 undergroundwires/privacy.sexy:0.6.1`
|
2. Run: `docker run -it -p 8080:80 --rm --name privacy.sexy-0.7.6 undergroundwires/privacy.sexy:0.7.6`
|
||||||
|
|
||||||
## Architecture
|
## Architecture
|
||||||
|
|
||||||
|
|||||||
139
docs/application-file.md
Normal file
139
docs/application-file.md
Normal file
@@ -0,0 +1,139 @@
|
|||||||
|
# Application file
|
||||||
|
|
||||||
|
- privacy.sexy is a data-driven application where it reads the necessary OS-specific logic from [`application.yaml`](./../src/application/application.yaml)
|
||||||
|
- 💡 Best practices
|
||||||
|
- If you repeat yourself, try to utilize [YAML-defined functions](#function)
|
||||||
|
- Always try to add documentation and a way to revert a tweak in [scripts](#script)
|
||||||
|
- 📖 Types in code: [`application.d.ts`](./../src/application/application.yaml.d.ts)
|
||||||
|
|
||||||
|
## Objects
|
||||||
|
|
||||||
|
### `Application`
|
||||||
|
|
||||||
|
- Application file simply defines different categories and their scripts in a tree structure.
|
||||||
|
- Application file also allows defining common [function](#function)s to be used throughout the application if you'd like different scripts to share same code.
|
||||||
|
|
||||||
|
#### `Application` syntax
|
||||||
|
|
||||||
|
- `actions: [` ***[`Category`](#Category)*** `, ... ]` **(required)**
|
||||||
|
- Each [category](#category) is rendered as different cards in card presentation.
|
||||||
|
- ❗ Application must consist of at least one category.
|
||||||
|
- `functions: [` ***[`Function`](#Function)*** `, ... ]`
|
||||||
|
- Functions are optionally defined to re-use the same code throughout different scripts.
|
||||||
|
|
||||||
|
### `Category`
|
||||||
|
|
||||||
|
- Category has a parent that has tree-like structure where it can have subcategories or subscripts.
|
||||||
|
- It's a logical grouping of different scripts and other categories.
|
||||||
|
|
||||||
|
#### `Category` syntax
|
||||||
|
|
||||||
|
- `category:` *`string`* (**required**)
|
||||||
|
- Name of the category
|
||||||
|
- ❗ Must be unique throughout the application
|
||||||
|
- `children: [` ***[`Category`](#category)*** `|` [***`Script`***](#Script) `, ... ]` (**required**)
|
||||||
|
- ❗ Category must consist of at least one subcategory or script.
|
||||||
|
- Children can be combination of scripts and subcategories.
|
||||||
|
|
||||||
|
### `Script`
|
||||||
|
|
||||||
|
- Script represents a single tweak.
|
||||||
|
- A script must include either:
|
||||||
|
- A `code` and `revertCode`
|
||||||
|
- Or `call` to call YAML-defined functions
|
||||||
|
- 🙏 For any new script, please add `revertCode` and `docs` values if possible.
|
||||||
|
|
||||||
|
#### `Script` syntax
|
||||||
|
|
||||||
|
- `name`: *`string`* (**required**)
|
||||||
|
- Name of the script
|
||||||
|
- ❗ Must be unique throughout the application
|
||||||
|
- E.g. `Disable targeted ads`
|
||||||
|
- `code`: *`string`* (may be **required**)
|
||||||
|
- Batch file commands that will be executed
|
||||||
|
- 💡 If defined, best practice to also define `revertCode`
|
||||||
|
- ❗ If not defined `call` must be defined, do not define if `call` is defined.
|
||||||
|
- `revertCode`: `string`
|
||||||
|
- Code that'll undo the change done by `code` property.
|
||||||
|
- E.g. let's say `code` sets an environment variable as `setx POWERSHELL_TELEMETRY_OPTOUT 1`
|
||||||
|
- then `revertCode` should be doing `setx POWERSHELL_TELEMETRY_OPTOUT 0`
|
||||||
|
- ❗ Do not define if `call` is defined.
|
||||||
|
- `call`: ***[`FunctionCall`](#FunctionCall)*** | `[` ***[`FunctionCall`](#FunctionCall)*** `, ... ]` (may be **required**)
|
||||||
|
- A shared function or sequence of functions to call (called in order)
|
||||||
|
- ❗ If not defined `code` must be defined
|
||||||
|
- `docs`: *`string`* | `[`*`string`*`, ... ]`
|
||||||
|
- Single documentation URL or list of URLs for those who wants to learn more about the script
|
||||||
|
- E.g. `https://docs.microsoft.com/en-us/windows-server/`
|
||||||
|
- `recommend`: `"standard"` | `"strict"` | `undefined` (default)
|
||||||
|
- If not defined then the script will not be recommended
|
||||||
|
- If defined it can be either
|
||||||
|
- `standard`: Only non-breaking scripts without limiting OS functionality
|
||||||
|
- `strict`: Scripts that can break certain functionality in favor of privacy and security
|
||||||
|
|
||||||
|
### `FunctionCall`
|
||||||
|
|
||||||
|
- Describes a single call to a function by optionally providing values to its parameters.
|
||||||
|
- 👀 See [parameter substitution](#parameter-substitution) for an example usage
|
||||||
|
|
||||||
|
#### `FunctionCall` syntax
|
||||||
|
|
||||||
|
- `function`: *`string`* (**required**)
|
||||||
|
- Name of the function to call.
|
||||||
|
- ❗ Function with same name must defined in `functions` property of [Application](#application)
|
||||||
|
- `parameters`: `[ parameterName:` *`parameterValue`*`, ... ]`
|
||||||
|
- Defines key value dictionary for each parameter and its value
|
||||||
|
- E.g.
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
parameters:
|
||||||
|
userDefinedParameterName: parameterValue
|
||||||
|
# ...
|
||||||
|
appName: Microsoft.WindowsFeedbackHub
|
||||||
|
```
|
||||||
|
|
||||||
|
### `Function`
|
||||||
|
|
||||||
|
- Functions allow re-usable code throughout the defined scripts.
|
||||||
|
- Functions are templates compiled by privacy.sexy and uses special expressions.
|
||||||
|
- Expressions are defined inside mustaches (double brackets, `{{` and `}}`)
|
||||||
|
- 👀 See [parameter substitution](#parameter-substitution) for an example usage
|
||||||
|
|
||||||
|
#### Parameter substitution
|
||||||
|
|
||||||
|
A simple function example
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
function: EchoArgument
|
||||||
|
parameters: [ 'argument' ]
|
||||||
|
code: Hello {{ $argument }} !
|
||||||
|
```
|
||||||
|
|
||||||
|
It would print "Hello world" if it's called in a [script](#script) as following:
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
script: Echo script
|
||||||
|
call:
|
||||||
|
function: EchoArgument
|
||||||
|
parameters:
|
||||||
|
argument: World
|
||||||
|
```
|
||||||
|
|
||||||
|
#### `Function` syntax
|
||||||
|
|
||||||
|
- `name`: *`string`* (**required**)
|
||||||
|
- Name of the function that scripts will use.
|
||||||
|
- Convention is to use camelCase, and be verbs.
|
||||||
|
- E.g. `uninstallStoreApp`
|
||||||
|
- ❗ Function names must be unique
|
||||||
|
- `parameters`: `[` *`string`* `, ... ]`
|
||||||
|
- Name of the parameters that the function has.
|
||||||
|
- Parameter values are provided by a [Script](#script) through a [FunctionCall](#functioncall)
|
||||||
|
- Parameter names must be defined to be used in expressions such as [parameter substitution](#parameter-substitution)
|
||||||
|
- ❗ Parameter names must be unique
|
||||||
|
`code`: *`string`* (**required**)
|
||||||
|
- Batch file commands that will be executed
|
||||||
|
- 💡 If defined, best practice to also define `revertCode`
|
||||||
|
- `revertCode`: *`string`*
|
||||||
|
- Code that'll undo the change done by `code` property.
|
||||||
|
- E.g. let's say `code` sets an environment variable as `setx POWERSHELL_TELEMETRY_OPTOUT 1`
|
||||||
|
- then `revertCode` should be doing `setx POWERSHELL_TELEMETRY_OPTOUT 0`
|
||||||
BIN
img/app.png
BIN
img/app.png
Binary file not shown.
|
Before Width: | Height: | Size: 66 KiB |
BIN
img/screenshot.png
Normal file
BIN
img/screenshot.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 52 KiB |
2092
package-lock.json
generated
2092
package-lock.json
generated
File diff suppressed because it is too large
Load Diff
60
package.json
60
package.json
@@ -1,9 +1,14 @@
|
|||||||
{
|
{
|
||||||
"name": "privacy.sexy",
|
"name": "privacy.sexy",
|
||||||
"version": "0.6.1",
|
"version": "0.7.6",
|
||||||
"author": "undergroundwires",
|
"author": "undergroundwires",
|
||||||
"description": "Enforce privacy & security best-practices on Windows, because privacy is sexy 🍑🍆",
|
"description": "Enforce privacy & security best-practices on Windows, because privacy is sexy 🍑🍆",
|
||||||
|
"homepage": "https://privacy.sexy",
|
||||||
"private": true,
|
"private": true,
|
||||||
|
"repository": {
|
||||||
|
"type": "git",
|
||||||
|
"url": "https://github.com/undergroundwires/privacy.sexy.git"
|
||||||
|
},
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"serve": "vue-cli-service serve",
|
"serve": "vue-cli-service serve",
|
||||||
"build": "vue-cli-service build",
|
"build": "vue-cli-service build",
|
||||||
@@ -21,47 +26,46 @@
|
|||||||
},
|
},
|
||||||
"main": "background.js",
|
"main": "background.js",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@fortawesome/fontawesome-svg-core": "^1.2.30",
|
"@fortawesome/fontawesome-svg-core": "^1.2.32",
|
||||||
"@fortawesome/free-brands-svg-icons": "^5.14.0",
|
"@fortawesome/free-brands-svg-icons": "^5.15.1",
|
||||||
"@fortawesome/free-regular-svg-icons": "^5.14.0",
|
"@fortawesome/free-regular-svg-icons": "^5.15.1",
|
||||||
"@fortawesome/free-solid-svg-icons": "^5.14.0",
|
"@fortawesome/free-solid-svg-icons": "^5.15.1",
|
||||||
"@fortawesome/vue-fontawesome": "^0.1.10",
|
"@fortawesome/vue-fontawesome": "^2.0.0",
|
||||||
"ace-builds": "^1.4.12",
|
"ace-builds": "^1.4.12",
|
||||||
"file-saver": "^2.0.2",
|
"file-saver": "^2.0.2",
|
||||||
"inversify": "^5.0.1",
|
"inversify": "^5.0.1",
|
||||||
"liquor-tree": "^0.2.70",
|
"liquor-tree": "^0.2.70",
|
||||||
"v-tooltip": "^2.0.2",
|
"v-tooltip": "2.0.2",
|
||||||
"vue": "^2.6.11",
|
"vue": "^2.6.12",
|
||||||
"vue-class-component": "^7.2.5",
|
"vue-class-component": "^7.2.6",
|
||||||
"vue-js-modal": "^2.0.0-rc.6",
|
"vue-js-modal": "^2.0.0-rc.6",
|
||||||
"vue-property-decorator": "^9.0.0"
|
"vue-property-decorator": "^9.0.2"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@types/ace": "0.0.43",
|
"@types/ace": "0.0.44",
|
||||||
"@types/chai": "^4.2.12",
|
"@types/chai": "^4.2.14",
|
||||||
"@types/file-saver": "^2.0.1",
|
"@types/file-saver": "^2.0.1",
|
||||||
"@types/mocha": "^8.0.0",
|
"@types/mocha": "^8.0.3",
|
||||||
"@types/node": "12.0.0",
|
"@vue/cli-plugin-typescript": "^4.5.7",
|
||||||
"@vue/cli-plugin-typescript": "^4.4.6",
|
"@vue/cli-plugin-unit-mocha": "^4.5.7",
|
||||||
"@vue/cli-plugin-unit-mocha": "^4.4.6",
|
"@vue/cli-service": "^4.5.7",
|
||||||
"@vue/cli-service": "^4.4.6",
|
"@vue/test-utils": "1.1.0",
|
||||||
"@vue/test-utils": "1.0.3",
|
|
||||||
"chai": "^4.2.0",
|
"chai": "^4.2.0",
|
||||||
"electron": "^9.1.1",
|
"electron": "^10.1.3",
|
||||||
"electron-devtools-installer": "^3.1.1",
|
"electron-devtools-installer": "^3.1.1",
|
||||||
"electron-log": "^4.2.2",
|
"electron-log": "^4.2.4",
|
||||||
"electron-updater": "^4.3.4",
|
"electron-updater": "^4.3.5",
|
||||||
"js-yaml-loader": "^1.2.2",
|
"js-yaml-loader": "^1.2.2",
|
||||||
"markdownlint-cli": "^0.23.2",
|
"markdownlint-cli": "^0.24.0",
|
||||||
"remark-cli": "^8.0.1",
|
"remark-cli": "^9.0.0",
|
||||||
"remark-lint-no-dead-urls": "^1.1.0",
|
"remark-lint-no-dead-urls": "^1.1.0",
|
||||||
"remark-preset-lint-consistent": "^3.0.1",
|
"remark-preset-lint-consistent": "^4.0.0",
|
||||||
"remark-validate-links": "^10.0.2",
|
"remark-validate-links": "^10.0.2",
|
||||||
"sass": "^1.26.10",
|
"sass": "^1.27.0",
|
||||||
"sass-loader": "^9.0.2",
|
"sass-loader": "^10.0.3",
|
||||||
"typescript": "^3.9.7",
|
"typescript": "^4.0.3",
|
||||||
"vue-cli-plugin-electron-builder": "^2.0.0-rc.4",
|
"vue-cli-plugin-electron-builder": "^2.0.0-rc.4",
|
||||||
"vue-template-compiler": "^2.6.11",
|
"vue-template-compiler": "^2.6.12",
|
||||||
"yaml-lint": "^1.2.4"
|
"yaml-lint": "^1.2.4"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
2
public/robots.txt
Normal file
2
public/robots.txt
Normal file
@@ -0,0 +1,2 @@
|
|||||||
|
User-agent: *
|
||||||
|
Disallow:
|
||||||
@@ -12,8 +12,7 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { Component, Vue, Prop } from 'vue-property-decorator';
|
import { Component, Vue } from 'vue-property-decorator';
|
||||||
import { ApplicationState } from '@/application/State/ApplicationState';
|
|
||||||
import TheHeader from '@/presentation/TheHeader.vue';
|
import TheHeader from '@/presentation/TheHeader.vue';
|
||||||
import TheFooter from '@/presentation/TheFooter/TheFooter.vue';
|
import TheFooter from '@/presentation/TheFooter/TheFooter.vue';
|
||||||
import TheCodeArea from '@/presentation/TheCodeArea.vue';
|
import TheCodeArea from '@/presentation/TheCodeArea.vue';
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { OperatingSystem } from '../OperatingSystem';
|
import { OperatingSystem } from '@/domain/OperatingSystem';
|
||||||
import { DetectorBuilder } from './DetectorBuilder';
|
import { DetectorBuilder } from './DetectorBuilder';
|
||||||
import { IBrowserOsDetector } from './IBrowserOsDetector';
|
import { IBrowserOsDetector } from './IBrowserOsDetector';
|
||||||
|
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import { IBrowserOsDetector } from './IBrowserOsDetector';
|
import { IBrowserOsDetector } from './IBrowserOsDetector';
|
||||||
import { OperatingSystem } from '../OperatingSystem';
|
import { OperatingSystem } from '@/domain/OperatingSystem';
|
||||||
|
|
||||||
export class DetectorBuilder {
|
export class DetectorBuilder {
|
||||||
private readonly existingPartsInUserAgent = new Array<string>();
|
private readonly existingPartsInUserAgent = new Array<string>();
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { OperatingSystem } from '../OperatingSystem';
|
import { OperatingSystem } from '@/domain/OperatingSystem';
|
||||||
|
|
||||||
export interface IBrowserOsDetector {
|
export interface IBrowserOsDetector {
|
||||||
detect(userAgent: string): OperatingSystem;
|
detect(userAgent: string): OperatingSystem;
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
import { BrowserOsDetector } from './BrowserOs/BrowserOsDetector';
|
import { BrowserOsDetector } from './BrowserOs/BrowserOsDetector';
|
||||||
import { IBrowserOsDetector } from './BrowserOs/IBrowserOsDetector';
|
import { IBrowserOsDetector } from './BrowserOs/IBrowserOsDetector';
|
||||||
import { IEnvironment } from './IEnvironment';
|
import { IEnvironment } from './IEnvironment';
|
||||||
import { OperatingSystem } from './OperatingSystem';
|
import { OperatingSystem } from '@/domain/OperatingSystem';
|
||||||
|
|
||||||
interface IEnvironmentVariables {
|
interface IEnvironmentVariables {
|
||||||
readonly window: Window & typeof globalThis;
|
readonly window: Window & typeof globalThis;
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { OperatingSystem } from './OperatingSystem';
|
import { OperatingSystem } from '@/domain/OperatingSystem';
|
||||||
|
|
||||||
export interface IEnvironment {
|
export interface IEnvironment {
|
||||||
isDesktop: boolean;
|
isDesktop: boolean;
|
||||||
|
|||||||
@@ -1,24 +1,37 @@
|
|||||||
import { Category } from '@/domain/Category';
|
import { Category } from '@/domain/Category';
|
||||||
import { Application } from '@/domain/Application';
|
import { Application } from '@/domain/Application';
|
||||||
import { IApplication } from '@/domain/IApplication';
|
import { IApplication } from '@/domain/IApplication';
|
||||||
|
import { IProjectInformation } from '@/domain/IProjectInformation';
|
||||||
import { ApplicationYaml } from 'js-yaml-loader!./../application.yaml';
|
import { ApplicationYaml } from 'js-yaml-loader!./../application.yaml';
|
||||||
import { parseCategory } from './CategoryParser';
|
import { parseCategory } from './CategoryParser';
|
||||||
|
import { ProjectInformation } from '@/domain/ProjectInformation';
|
||||||
|
import { ScriptCompiler } from './Compiler/ScriptCompiler';
|
||||||
|
|
||||||
export function parseApplication(content: ApplicationYaml): IApplication {
|
|
||||||
|
export function parseApplication(content: ApplicationYaml, env: NodeJS.ProcessEnv = process.env): IApplication {
|
||||||
validate(content);
|
validate(content);
|
||||||
|
const compiler = new ScriptCompiler(content.functions);
|
||||||
const categories = new Array<Category>();
|
const categories = new Array<Category>();
|
||||||
for (const action of content.actions) {
|
for (const action of content.actions) {
|
||||||
const category = parseCategory(action);
|
const category = parseCategory(action, compiler);
|
||||||
categories.push(category);
|
categories.push(category);
|
||||||
}
|
}
|
||||||
|
const info = readAppInformation(env);
|
||||||
const app = new Application(
|
const app = new Application(
|
||||||
content.name,
|
info,
|
||||||
content.repositoryUrl,
|
|
||||||
process.env.VUE_APP_VERSION,
|
|
||||||
categories);
|
categories);
|
||||||
return app;
|
return app;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function readAppInformation(environment: NodeJS.ProcessEnv): IProjectInformation {
|
||||||
|
return new ProjectInformation(
|
||||||
|
environment.VUE_APP_NAME,
|
||||||
|
environment.VUE_APP_VERSION,
|
||||||
|
environment.VUE_APP_REPOSITORY_URL,
|
||||||
|
environment.VUE_APP_HOMEPAGE_URL,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
function validate(content: ApplicationYaml): void {
|
function validate(content: ApplicationYaml): void {
|
||||||
if (!content) {
|
if (!content) {
|
||||||
throw new Error('application is null or undefined');
|
throw new Error('application is null or undefined');
|
||||||
|
|||||||
@@ -3,6 +3,7 @@ import { Script } from '@/domain/Script';
|
|||||||
import { Category } from '@/domain/Category';
|
import { Category } from '@/domain/Category';
|
||||||
import { parseDocUrls } from './DocumentationParser';
|
import { parseDocUrls } from './DocumentationParser';
|
||||||
import { parseScript } from './ScriptParser';
|
import { parseScript } from './ScriptParser';
|
||||||
|
import { IScriptCompiler } from './Compiler/IScriptCompiler';
|
||||||
|
|
||||||
let categoryIdCounter: number = 0;
|
let categoryIdCounter: number = 0;
|
||||||
|
|
||||||
@@ -11,14 +12,17 @@ interface ICategoryChildren {
|
|||||||
subScripts: Script[];
|
subScripts: Script[];
|
||||||
}
|
}
|
||||||
|
|
||||||
export function parseCategory(category: YamlCategory): Category {
|
export function parseCategory(category: YamlCategory, compiler: IScriptCompiler): Category {
|
||||||
|
if (!compiler) {
|
||||||
|
throw new Error('undefined compiler');
|
||||||
|
}
|
||||||
ensureValid(category);
|
ensureValid(category);
|
||||||
const children: ICategoryChildren = {
|
const children: ICategoryChildren = {
|
||||||
subCategories: new Array<Category>(),
|
subCategories: new Array<Category>(),
|
||||||
subScripts: new Array<Script>(),
|
subScripts: new Array<Script>(),
|
||||||
};
|
};
|
||||||
for (const categoryOrScript of category.children) {
|
for (const categoryOrScript of category.children) {
|
||||||
parseCategoryChild(categoryOrScript, children, category);
|
parseCategoryChild(categoryOrScript, children, category, compiler);
|
||||||
}
|
}
|
||||||
return new Category(
|
return new Category(
|
||||||
/*id*/ categoryIdCounter++,
|
/*id*/ categoryIdCounter++,
|
||||||
@@ -42,22 +46,26 @@ function ensureValid(category: YamlCategory) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function parseCategoryChild(
|
function parseCategoryChild(
|
||||||
categoryOrScript: any, children: ICategoryChildren, parent: YamlCategory) {
|
categoryOrScript: any,
|
||||||
|
children: ICategoryChildren,
|
||||||
|
parent: YamlCategory,
|
||||||
|
compiler: IScriptCompiler) {
|
||||||
if (isCategory(categoryOrScript)) {
|
if (isCategory(categoryOrScript)) {
|
||||||
const subCategory = parseCategory(categoryOrScript as YamlCategory);
|
const subCategory = parseCategory(categoryOrScript as YamlCategory, compiler);
|
||||||
children.subCategories.push(subCategory);
|
children.subCategories.push(subCategory);
|
||||||
} else if (isScript(categoryOrScript)) {
|
} else if (isScript(categoryOrScript)) {
|
||||||
const yamlScript = categoryOrScript as YamlScript;
|
const yamlScript = categoryOrScript as YamlScript;
|
||||||
const script = parseScript(yamlScript);
|
const script = parseScript(yamlScript, compiler);
|
||||||
children.subScripts.push(script);
|
children.subScripts.push(script);
|
||||||
} else {
|
} else {
|
||||||
throw new Error(`Child element is neither a category or a script.
|
throw new Error(`Child element is neither a category or a script.
|
||||||
Parent: ${parent.category}, element: ${categoryOrScript}`);
|
Parent: ${parent.category}, element: ${JSON.stringify(categoryOrScript)}`);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function isScript(categoryOrScript: any): boolean {
|
function isScript(categoryOrScript: any): boolean {
|
||||||
return categoryOrScript.code && categoryOrScript.code.length > 0;
|
return (categoryOrScript.code && categoryOrScript.code.length > 0)
|
||||||
|
|| categoryOrScript.call;
|
||||||
}
|
}
|
||||||
|
|
||||||
function isCategory(categoryOrScript: any): boolean {
|
function isCategory(categoryOrScript: any): boolean {
|
||||||
|
|||||||
7
src/application/Parser/Compiler/IScriptCompiler.ts
Normal file
7
src/application/Parser/Compiler/IScriptCompiler.ts
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
import { IScriptCode } from '@/domain/IScriptCode';
|
||||||
|
import { YamlScript } from 'js-yaml-loader!./application.yaml';
|
||||||
|
|
||||||
|
export interface IScriptCompiler {
|
||||||
|
canCompile(script: YamlScript): boolean;
|
||||||
|
compile(script: YamlScript): IScriptCode;
|
||||||
|
}
|
||||||
200
src/application/Parser/Compiler/ScriptCompiler.ts
Normal file
200
src/application/Parser/Compiler/ScriptCompiler.ts
Normal file
@@ -0,0 +1,200 @@
|
|||||||
|
import { IScriptCode } from '@/domain/IScriptCode';
|
||||||
|
import { ScriptCode } from '@/domain/ScriptCode';
|
||||||
|
import { YamlScript, YamlFunction, FunctionCall, ScriptFunctionCall, FunctionCallParameters } from 'js-yaml-loader!./application.yaml';
|
||||||
|
import { IScriptCompiler } from './IScriptCompiler';
|
||||||
|
|
||||||
|
interface ICompiledCode {
|
||||||
|
readonly code: string;
|
||||||
|
readonly revertCode: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export class ScriptCompiler implements IScriptCompiler {
|
||||||
|
constructor(private readonly functions: readonly YamlFunction[]) {
|
||||||
|
ensureValidFunctions(functions);
|
||||||
|
}
|
||||||
|
public canCompile(script: YamlScript): boolean {
|
||||||
|
if (!script.call) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
public compile(script: YamlScript): IScriptCode {
|
||||||
|
this.ensureCompilable(script.call);
|
||||||
|
const compiledCodes = new Array<ICompiledCode>();
|
||||||
|
const calls = getCallSequence(script.call);
|
||||||
|
calls.forEach((currentCall, currentCallIndex) => {
|
||||||
|
ensureValidCall(currentCall, script.name);
|
||||||
|
const commonFunction = this.getFunctionByName(currentCall.function);
|
||||||
|
let functionCode = compileCode(commonFunction, currentCall.parameters);
|
||||||
|
if (currentCallIndex !== calls.length - 1) {
|
||||||
|
functionCode = appendLine(functionCode);
|
||||||
|
}
|
||||||
|
compiledCodes.push(functionCode);
|
||||||
|
});
|
||||||
|
const scriptCode = merge(compiledCodes);
|
||||||
|
return new ScriptCode(script.name, scriptCode.code, scriptCode.revertCode);
|
||||||
|
}
|
||||||
|
|
||||||
|
private getFunctionByName(name: string): YamlFunction {
|
||||||
|
const func = this.functions.find((f) => f.name === name);
|
||||||
|
if (!func) {
|
||||||
|
throw new Error(`called function is not defined "${name}"`);
|
||||||
|
}
|
||||||
|
return func;
|
||||||
|
}
|
||||||
|
|
||||||
|
private ensureCompilable(call: ScriptFunctionCall) {
|
||||||
|
if (!this.functions || this.functions.length === 0) {
|
||||||
|
throw new Error('cannot compile without shared functions');
|
||||||
|
}
|
||||||
|
if (typeof call !== 'object') {
|
||||||
|
throw new Error('called function(s) must be an object');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function getDuplicates(texts: readonly string[]): string[] {
|
||||||
|
return texts.filter((item, index) => texts.indexOf(item) !== index);
|
||||||
|
}
|
||||||
|
|
||||||
|
function printList(list: readonly string[]): string {
|
||||||
|
return `"${list.join('","')}"`;
|
||||||
|
}
|
||||||
|
|
||||||
|
function ensureNoDuplicatesInFunctionNames(functions: readonly YamlFunction[]) {
|
||||||
|
const duplicateFunctionNames = getDuplicates(functions
|
||||||
|
.map((func) => func.name.toLowerCase()));
|
||||||
|
if (duplicateFunctionNames.length) {
|
||||||
|
throw new Error(`duplicate function name: ${printList(duplicateFunctionNames)}`);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function ensureNoDuplicatesInParameterNames(functions: readonly YamlFunction[]) {
|
||||||
|
const functionsWithParameters = functions
|
||||||
|
.filter((func) => func.parameters && func.parameters.length > 0);
|
||||||
|
for (const func of functionsWithParameters) {
|
||||||
|
const duplicateParameterNames = getDuplicates(func.parameters);
|
||||||
|
if (duplicateParameterNames.length) {
|
||||||
|
throw new Error(`"${func.name}": duplicate parameter name: ${printList(duplicateParameterNames)}`);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function ensureNoDuplicateCode(functions: readonly YamlFunction[]) {
|
||||||
|
const duplicateCodes = getDuplicates(functions.map((func) => func.code));
|
||||||
|
if (duplicateCodes.length > 0) {
|
||||||
|
throw new Error(`duplicate "code" in functions: ${printList(duplicateCodes)}`);
|
||||||
|
}
|
||||||
|
const duplicateRevertCodes = getDuplicates(functions
|
||||||
|
.filter((func) => func.revertCode)
|
||||||
|
.map((func) => func.revertCode));
|
||||||
|
if (duplicateRevertCodes.length > 0) {
|
||||||
|
throw new Error(`duplicate "revertCode" in functions: ${printList(duplicateRevertCodes)}`);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function ensureValidFunctions(functions: readonly YamlFunction[]) {
|
||||||
|
if (!functions) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
ensureNoDuplicatesInFunctionNames(functions);
|
||||||
|
ensureNoDuplicatesInParameterNames(functions);
|
||||||
|
ensureNoDuplicateCode(functions);
|
||||||
|
}
|
||||||
|
|
||||||
|
function appendLine(code: ICompiledCode): ICompiledCode {
|
||||||
|
const appendLineIfNotEmpty = (str: string) => str ? `${str}\n` : str;
|
||||||
|
return {
|
||||||
|
code: appendLineIfNotEmpty(code.code),
|
||||||
|
revertCode: appendLineIfNotEmpty(code.revertCode),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
function merge(codes: readonly ICompiledCode[]): ICompiledCode {
|
||||||
|
return {
|
||||||
|
code: codes.map((code) => code.code).join(''),
|
||||||
|
revertCode: codes.map((code) => code.revertCode).join(''),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
function compileCode(func: YamlFunction, parameters: FunctionCallParameters): ICompiledCode {
|
||||||
|
return {
|
||||||
|
code: compileExpressions(func.code, parameters),
|
||||||
|
revertCode: compileExpressions(func.revertCode, parameters),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
function compileExpressions(code: string, parameters: FunctionCallParameters): string {
|
||||||
|
let intermediateCode = compileToIL(code);
|
||||||
|
intermediateCode = substituteParameters(intermediateCode, parameters);
|
||||||
|
ensureNoExpressionLeft(intermediateCode);
|
||||||
|
return intermediateCode;
|
||||||
|
}
|
||||||
|
|
||||||
|
function substituteParameters(intermediateCode: string, parameters: FunctionCallParameters): string {
|
||||||
|
const parameterNames = getUniqueParameterNamesFromIL(intermediateCode);
|
||||||
|
if (parameterNames.length && !parameters) {
|
||||||
|
throw new Error(`no parameters defined, expected: ${printList(parameterNames)}`);
|
||||||
|
}
|
||||||
|
for (const parameterName of parameterNames) {
|
||||||
|
const parameterValue = parameters[parameterName];
|
||||||
|
intermediateCode = substituteParameter(intermediateCode, parameterName, parameterValue);
|
||||||
|
}
|
||||||
|
return intermediateCode;
|
||||||
|
}
|
||||||
|
|
||||||
|
function ensureValidCall(call: FunctionCall, scriptName: string) {
|
||||||
|
if (!call) {
|
||||||
|
throw new Error(`undefined function call in script "${scriptName}"`);
|
||||||
|
}
|
||||||
|
if (!call.function) {
|
||||||
|
throw new Error(`empty function name called in script "${scriptName}"`);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function getCallSequence(call: ScriptFunctionCall): FunctionCall[] {
|
||||||
|
if (call instanceof Array) {
|
||||||
|
return call as FunctionCall[];
|
||||||
|
}
|
||||||
|
return [ call as FunctionCall ];
|
||||||
|
}
|
||||||
|
|
||||||
|
function getDistinctValues(values: readonly string[]): string[] {
|
||||||
|
return values.filter((value, index, self) => {
|
||||||
|
return self.indexOf(value) === index;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// Trim each expression and put them inside "{{exp|}}" e.g. "{{ $hello }}" becomes "{{exp|$hello}}"
|
||||||
|
function compileToIL(code: string) {
|
||||||
|
return code.replace(/\{\{([\s]*[^;\s\{]+[\s]*)\}\}/g, (_, match) => {
|
||||||
|
return `\{\{exp|${match.trim()}\}\}`;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// Parses all distinct usages of {{exp|$parameterName}}
|
||||||
|
function getUniqueParameterNamesFromIL(ilCode: string) {
|
||||||
|
const allSubstitutions = ilCode.matchAll(/\{\{exp\|\$([^;\s\{]+[\s]*)\}\}/g);
|
||||||
|
const allParameters = Array.from(allSubstitutions, (match) => match[1]);
|
||||||
|
const uniqueParameterNames = getDistinctValues(allParameters);
|
||||||
|
return uniqueParameterNames;
|
||||||
|
}
|
||||||
|
|
||||||
|
// substitutes {{exp|$parameterName}} to value of the parameter
|
||||||
|
function substituteParameter(ilCode: string, parameterName: string, parameterValue: string) {
|
||||||
|
if (!parameterValue) {
|
||||||
|
throw Error(`parameter value is not provided for "${parameterName}" in function call`);
|
||||||
|
}
|
||||||
|
const pattern = `{{exp|$${parameterName}}}`;
|
||||||
|
return ilCode.split(pattern).join(parameterValue); // as .replaceAll() is not yet supported by TS
|
||||||
|
}
|
||||||
|
|
||||||
|
// finds all "{{exp|..}} left"
|
||||||
|
function ensureNoExpressionLeft(ilCode: string) {
|
||||||
|
const allSubstitutions = ilCode.matchAll(/\{\{exp\|(.*?)\}\}/g);
|
||||||
|
const allMatches = Array.from(allSubstitutions, (match) => match[1]);
|
||||||
|
const uniqueExpressions = getDistinctValues(allMatches);
|
||||||
|
if (uniqueExpressions.length > 0) {
|
||||||
|
throw new Error(`unknown expression: ${printList(uniqueExpressions)}`);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,16 +1,61 @@
|
|||||||
import { Script } from '@/domain/Script';
|
import { Script } from '@/domain/Script';
|
||||||
import { YamlScript } from 'js-yaml-loader!./application.yaml';
|
import { YamlScript } from 'js-yaml-loader!./application.yaml';
|
||||||
import { parseDocUrls } from './DocumentationParser';
|
import { parseDocUrls } from './DocumentationParser';
|
||||||
|
import { RecommendationLevelNames, RecommendationLevel } from '@/domain/RecommendationLevel';
|
||||||
|
import { IScriptCompiler } from './Compiler/IScriptCompiler';
|
||||||
|
import { IScriptCode } from '@/domain/IScriptCode';
|
||||||
|
import { ScriptCode } from '@/domain/ScriptCode';
|
||||||
|
|
||||||
export function parseScript(yamlScript: YamlScript): Script {
|
export function parseScript(yamlScript: YamlScript, compiler: IScriptCompiler): Script {
|
||||||
if (!yamlScript) {
|
validateScript(yamlScript);
|
||||||
throw new Error('script is null or undefined');
|
if (!compiler) {
|
||||||
|
throw new Error('undefined compiler');
|
||||||
}
|
}
|
||||||
const script = new Script(
|
const script = new Script(
|
||||||
/* name */ yamlScript.name,
|
/* name */ yamlScript.name,
|
||||||
/* code */ yamlScript.code,
|
/* code */ parseCode(yamlScript, compiler),
|
||||||
/* revertCode */ yamlScript.revertCode,
|
|
||||||
/* docs */ parseDocUrls(yamlScript),
|
/* docs */ parseDocUrls(yamlScript),
|
||||||
/* isRecommended */ yamlScript.recommend);
|
/* level */ getLevel(yamlScript.recommend));
|
||||||
return script;
|
return script;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function getLevel(level: string): RecommendationLevel | undefined {
|
||||||
|
if (!level) {
|
||||||
|
return undefined;
|
||||||
|
}
|
||||||
|
if (typeof level !== 'string') {
|
||||||
|
throw new Error(`level must be a string but it was ${typeof level}`);
|
||||||
|
}
|
||||||
|
const typedLevel = RecommendationLevelNames
|
||||||
|
.find((l) => l.toLowerCase() === level.toLowerCase());
|
||||||
|
if (!typedLevel) {
|
||||||
|
throw new Error(`unknown level: \"${level}\"`);
|
||||||
|
}
|
||||||
|
return RecommendationLevel[typedLevel as keyof typeof RecommendationLevel];
|
||||||
|
}
|
||||||
|
|
||||||
|
function parseCode(yamlScript: YamlScript, compiler: IScriptCompiler): IScriptCode {
|
||||||
|
if (compiler.canCompile(yamlScript)) {
|
||||||
|
return compiler.compile(yamlScript);
|
||||||
|
}
|
||||||
|
return new ScriptCode(yamlScript.name, yamlScript.code, yamlScript.revertCode);
|
||||||
|
}
|
||||||
|
|
||||||
|
function ensureNotBothCallAndCode(yamlScript: YamlScript) {
|
||||||
|
if (yamlScript.code && yamlScript.call) {
|
||||||
|
throw new Error('cannot define both "call" and "code"');
|
||||||
|
}
|
||||||
|
if (yamlScript.revertCode && yamlScript.call) {
|
||||||
|
throw new Error('cannot define "revertCode" if "call" is defined');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function validateScript(yamlScript: YamlScript) {
|
||||||
|
if (!yamlScript) {
|
||||||
|
throw new Error('undefined script');
|
||||||
|
}
|
||||||
|
if (!yamlScript.code && !yamlScript.call) {
|
||||||
|
throw new Error('must define either "call" or "code"');
|
||||||
|
}
|
||||||
|
ensureNotBothCallAndCode(yamlScript);
|
||||||
|
}
|
||||||
|
|||||||
@@ -11,6 +11,7 @@ import { Script } from '@/domain/Script';
|
|||||||
import { IApplication } from '@/domain/IApplication';
|
import { IApplication } from '@/domain/IApplication';
|
||||||
import { IApplicationCode } from './Code/IApplicationCode';
|
import { IApplicationCode } from './Code/IApplicationCode';
|
||||||
import applicationFile from 'js-yaml-loader!@/application/application.yaml';
|
import applicationFile from 'js-yaml-loader!@/application/application.yaml';
|
||||||
|
import { SelectedScript } from '@/application/State/Selection/SelectedScript';
|
||||||
|
|
||||||
/** Mutatable singleton application state that's the single source of truth throughout the application */
|
/** Mutatable singleton application state that's the single source of truth throughout the application */
|
||||||
export class ApplicationState implements IApplicationState {
|
export class ApplicationState implements IApplicationState {
|
||||||
@@ -37,8 +38,8 @@ export class ApplicationState implements IApplicationState {
|
|||||||
public readonly app: IApplication,
|
public readonly app: IApplication,
|
||||||
/** Initially selected scripts */
|
/** Initially selected scripts */
|
||||||
public readonly defaultScripts: Script[]) {
|
public readonly defaultScripts: Script[]) {
|
||||||
this.selection = new UserSelection(app, defaultScripts);
|
this.selection = new UserSelection(app, defaultScripts.map((script) => new SelectedScript(script, false)));
|
||||||
this.code = new ApplicationCode(this.selection, app.version);
|
this.code = new ApplicationCode(this.selection, app.info.version);
|
||||||
this.filter = new UserFilter(app);
|
this.filter = new UserFilter(app);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,30 +1,38 @@
|
|||||||
|
import { CodeChangedEvent } from './Event/CodeChangedEvent';
|
||||||
|
import { CodePosition } from './Position/CodePosition';
|
||||||
|
import { ICodeChangedEvent } from './Event/ICodeChangedEvent';
|
||||||
import { SelectedScript } from '@/application/State/Selection/SelectedScript';
|
import { SelectedScript } from '@/application/State/Selection/SelectedScript';
|
||||||
import { IUserSelection } from '@/application/State/Selection/IUserSelection';
|
import { IUserSelection } from '@/application/State/Selection/IUserSelection';
|
||||||
import { UserScriptGenerator } from './UserScriptGenerator';
|
import { UserScriptGenerator } from './Generation/UserScriptGenerator';
|
||||||
import { Signal } from '@/infrastructure/Events/Signal';
|
import { Signal } from '@/infrastructure/Events/Signal';
|
||||||
import { IApplicationCode } from './IApplicationCode';
|
import { IApplicationCode } from './IApplicationCode';
|
||||||
import { IUserScriptGenerator } from './IUserScriptGenerator';
|
import { IUserScriptGenerator } from './Generation/IUserScriptGenerator';
|
||||||
|
|
||||||
export class ApplicationCode implements IApplicationCode {
|
export class ApplicationCode implements IApplicationCode {
|
||||||
public readonly changed = new Signal<string>();
|
public readonly changed = new Signal<ICodeChangedEvent>();
|
||||||
public current: string;
|
public current: string;
|
||||||
|
|
||||||
private readonly generator: IUserScriptGenerator = new UserScriptGenerator();
|
private scriptPositions = new Map<SelectedScript, CodePosition>();
|
||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
userSelection: IUserSelection,
|
userSelection: IUserSelection,
|
||||||
private readonly version: string) {
|
private readonly version: string,
|
||||||
|
private readonly generator: IUserScriptGenerator = new UserScriptGenerator()) {
|
||||||
if (!userSelection) { throw new Error('userSelection is null or undefined'); }
|
if (!userSelection) { throw new Error('userSelection is null or undefined'); }
|
||||||
if (!version) { throw new Error('version is null or undefined'); }
|
if (!version) { throw new Error('version is null or undefined'); }
|
||||||
this.generator = new UserScriptGenerator();
|
if (!generator) { throw new Error('generator is null or undefined'); }
|
||||||
this.setCode(userSelection.selectedScripts);
|
this.setCode(userSelection.selectedScripts);
|
||||||
userSelection.changed.on((scripts) => {
|
userSelection.changed.on((scripts) => {
|
||||||
this.setCode(scripts);
|
this.setCode(scripts);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
private setCode(scripts: ReadonlyArray<SelectedScript>) {
|
private setCode(scripts: ReadonlyArray<SelectedScript>): void {
|
||||||
this.current = scripts.length === 0 ? '' : this.generator.buildCode(scripts, this.version);
|
const oldScripts = Array.from(this.scriptPositions.keys());
|
||||||
this.changed.notify(this.current);
|
const code = this.generator.buildCode(scripts, this.version);
|
||||||
|
this.current = code.code;
|
||||||
|
this.scriptPositions = code.scriptPositions;
|
||||||
|
const event = new CodeChangedEvent(code.code, oldScripts, code.scriptPositions);
|
||||||
|
this.changed.notify(event);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
64
src/application/State/Code/Event/CodeChangedEvent.ts
Normal file
64
src/application/State/Code/Event/CodeChangedEvent.ts
Normal file
@@ -0,0 +1,64 @@
|
|||||||
|
import { ICodeChangedEvent } from './ICodeChangedEvent';
|
||||||
|
import { SelectedScript } from '../../Selection/SelectedScript';
|
||||||
|
import { IScript } from '@/domain/IScript';
|
||||||
|
import { ICodePosition } from '@/application/State/Code/Position/ICodePosition';
|
||||||
|
|
||||||
|
export class CodeChangedEvent implements ICodeChangedEvent {
|
||||||
|
public readonly code: string;
|
||||||
|
public readonly addedScripts: ReadonlyArray<IScript>;
|
||||||
|
public readonly removedScripts: ReadonlyArray<IScript>;
|
||||||
|
public readonly changedScripts: ReadonlyArray<IScript>;
|
||||||
|
|
||||||
|
private readonly scripts: Map<IScript, ICodePosition>;
|
||||||
|
|
||||||
|
constructor(
|
||||||
|
code: string,
|
||||||
|
oldScripts: ReadonlyArray<SelectedScript>,
|
||||||
|
scripts: Map<SelectedScript, ICodePosition>) {
|
||||||
|
ensureAllPositionsExist(code, Array.from(scripts.values()));
|
||||||
|
this.code = code;
|
||||||
|
const newScripts = Array.from(scripts.keys());
|
||||||
|
this.addedScripts = selectIfNotExists(newScripts, oldScripts);
|
||||||
|
this.removedScripts = selectIfNotExists(oldScripts, newScripts);
|
||||||
|
this.changedScripts = getChangedScripts(oldScripts, newScripts);
|
||||||
|
this.scripts = new Map<IScript, ICodePosition>();
|
||||||
|
scripts.forEach((position, selection) => {
|
||||||
|
this.scripts.set(selection.script, position);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public isEmpty(): boolean {
|
||||||
|
return this.scripts.size === 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
public getScriptPositionInCode(script: IScript): ICodePosition {
|
||||||
|
return this.scripts.get(script);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function ensureAllPositionsExist(script: string, positions: ReadonlyArray<ICodePosition>) {
|
||||||
|
const totalLines = script.split(/\r\n|\r|\n/).length;
|
||||||
|
for (const position of positions) {
|
||||||
|
if (position.endLine > totalLines) {
|
||||||
|
throw new Error(`script end line (${position.endLine}) is out of range.` +
|
||||||
|
`(total code lines: ${totalLines}`);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function getChangedScripts(
|
||||||
|
oldScripts: ReadonlyArray<SelectedScript>,
|
||||||
|
newScripts: ReadonlyArray<SelectedScript>): ReadonlyArray<IScript> {
|
||||||
|
return newScripts
|
||||||
|
.filter((newScript) => oldScripts.find((oldScript) => oldScript.id === newScript.id
|
||||||
|
&& oldScript.revert !== newScript.revert ))
|
||||||
|
.map((selection) => selection.script);
|
||||||
|
}
|
||||||
|
|
||||||
|
function selectIfNotExists(
|
||||||
|
selectableContainer: ReadonlyArray<SelectedScript>,
|
||||||
|
test: ReadonlyArray<SelectedScript>) {
|
||||||
|
return selectableContainer
|
||||||
|
.filter((script) => !test.find((oldScript) => oldScript.id === script.id))
|
||||||
|
.map((selection) => selection.script);
|
||||||
|
}
|
||||||
11
src/application/State/Code/Event/ICodeChangedEvent.ts
Normal file
11
src/application/State/Code/Event/ICodeChangedEvent.ts
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
import { IScript } from '@/domain/IScript';
|
||||||
|
import { ICodePosition } from '@/application/State/Code/Position/ICodePosition';
|
||||||
|
|
||||||
|
export interface ICodeChangedEvent {
|
||||||
|
readonly code: string;
|
||||||
|
addedScripts: ReadonlyArray<IScript>;
|
||||||
|
removedScripts: ReadonlyArray<IScript>;
|
||||||
|
changedScripts: ReadonlyArray<IScript>;
|
||||||
|
isEmpty(): boolean;
|
||||||
|
getScriptPositionInCode(script: IScript): ICodePosition;
|
||||||
|
}
|
||||||
@@ -4,8 +4,20 @@ const TotalFunctionSeparatorChars = 58;
|
|||||||
export class CodeBuilder {
|
export class CodeBuilder {
|
||||||
private readonly lines = new Array<string>();
|
private readonly lines = new Array<string>();
|
||||||
|
|
||||||
|
// Returns current line starting from 0 (no lines), or 1 (have single line)
|
||||||
|
public get currentLine(): number {
|
||||||
|
return this.lines.length;
|
||||||
|
}
|
||||||
|
|
||||||
public appendLine(code?: string): CodeBuilder {
|
public appendLine(code?: string): CodeBuilder {
|
||||||
this.lines.push(code);
|
if (!code) {
|
||||||
|
this.lines.push('');
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
const lines = code.match(/[^\r\n]+/g);
|
||||||
|
for (const line of lines) {
|
||||||
|
this.lines.push(line);
|
||||||
|
}
|
||||||
return this;
|
return this;
|
||||||
}
|
}
|
||||||
|
|
||||||
7
src/application/State/Code/Generation/IUserScript.ts
Normal file
7
src/application/State/Code/Generation/IUserScript.ts
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
import { SelectedScript } from '@/application/State/Selection/SelectedScript';
|
||||||
|
import { ICodePosition } from '@/application/State/Code/Position/ICodePosition';
|
||||||
|
|
||||||
|
export interface IUserScript {
|
||||||
|
code: string;
|
||||||
|
scriptPositions: Map<SelectedScript, ICodePosition>;
|
||||||
|
}
|
||||||
@@ -0,0 +1,7 @@
|
|||||||
|
import { SelectedScript } from '@/application/State/Selection/SelectedScript';
|
||||||
|
import { IUserScript } from './IUserScript';
|
||||||
|
export interface IUserScriptGenerator {
|
||||||
|
buildCode(
|
||||||
|
selectedScripts: ReadonlyArray<SelectedScript>,
|
||||||
|
version: string): IUserScript;
|
||||||
|
}
|
||||||
68
src/application/State/Code/Generation/UserScriptGenerator.ts
Normal file
68
src/application/State/Code/Generation/UserScriptGenerator.ts
Normal file
@@ -0,0 +1,68 @@
|
|||||||
|
import { SelectedScript } from '@/application/State/Selection/SelectedScript';
|
||||||
|
import { IUserScriptGenerator } from './IUserScriptGenerator';
|
||||||
|
import { CodeBuilder } from './CodeBuilder';
|
||||||
|
import { ICodePosition } from '@/application/State/Code/Position/ICodePosition';
|
||||||
|
import { CodePosition } from '../Position/CodePosition';
|
||||||
|
import { IUserScript } from './IUserScript';
|
||||||
|
|
||||||
|
export const adminRightsScript = {
|
||||||
|
name: 'Ensure admin privileges',
|
||||||
|
code: 'fltmc >nul 2>&1 || (\n' +
|
||||||
|
' echo Administrator privileges are required.\n' +
|
||||||
|
' PowerShell Start -Verb RunAs \'%0\' 2> nul || (\n' +
|
||||||
|
' echo Right-click on the script and select "Run as administrator".\n' +
|
||||||
|
' pause & exit 1\n' +
|
||||||
|
' )\n' +
|
||||||
|
' exit 0\n' +
|
||||||
|
')',
|
||||||
|
};
|
||||||
|
|
||||||
|
export class UserScriptGenerator implements IUserScriptGenerator {
|
||||||
|
public buildCode(selectedScripts: ReadonlyArray<SelectedScript>, version: string): IUserScript {
|
||||||
|
if (!selectedScripts) { throw new Error('scripts is undefined'); }
|
||||||
|
if (!version) { throw new Error('version is undefined'); }
|
||||||
|
let scriptPositions = new Map<SelectedScript, ICodePosition>();
|
||||||
|
if (!selectedScripts.length) {
|
||||||
|
return { code: '', scriptPositions };
|
||||||
|
}
|
||||||
|
const builder = initializeCode(version);
|
||||||
|
for (const selection of selectedScripts) {
|
||||||
|
scriptPositions = appendSelection(selection, scriptPositions, builder);
|
||||||
|
}
|
||||||
|
const code = finalizeCode(builder);
|
||||||
|
return { code, scriptPositions };
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function initializeCode(version: string): CodeBuilder {
|
||||||
|
return new CodeBuilder()
|
||||||
|
.appendLine('@echo off')
|
||||||
|
.appendCommentLine(`https://privacy.sexy — v${version} — ${new Date().toUTCString()}`)
|
||||||
|
.appendFunction(adminRightsScript.name, adminRightsScript.code)
|
||||||
|
.appendLine();
|
||||||
|
}
|
||||||
|
|
||||||
|
function finalizeCode(builder: CodeBuilder): string {
|
||||||
|
return builder.appendLine()
|
||||||
|
.appendLine('pause')
|
||||||
|
.appendLine('exit /b 0')
|
||||||
|
.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
function appendSelection(
|
||||||
|
selection: SelectedScript,
|
||||||
|
scriptPositions: Map<SelectedScript, ICodePosition>,
|
||||||
|
builder: CodeBuilder): Map<SelectedScript, ICodePosition> {
|
||||||
|
const startPosition = builder.currentLine + 1;
|
||||||
|
appendCode(selection, builder);
|
||||||
|
const endPosition = builder.currentLine - 1;
|
||||||
|
builder.appendLine();
|
||||||
|
scriptPositions.set(selection, new CodePosition(startPosition, endPosition));
|
||||||
|
return scriptPositions;
|
||||||
|
}
|
||||||
|
|
||||||
|
function appendCode(selection: SelectedScript, builder: CodeBuilder) {
|
||||||
|
const name = selection.revert ? `${selection.script.name} (revert)` : selection.script.name;
|
||||||
|
const scriptCode = selection.revert ? selection.script.code.revert : selection.script.code.execute;
|
||||||
|
builder.appendFunction(name, scriptCode);
|
||||||
|
}
|
||||||
@@ -1,6 +1,7 @@
|
|||||||
|
import { ICodeChangedEvent } from './Event/ICodeChangedEvent';
|
||||||
import { ISignal } from '@/infrastructure/Events/ISignal';
|
import { ISignal } from '@/infrastructure/Events/ISignal';
|
||||||
|
|
||||||
export interface IApplicationCode {
|
export interface IApplicationCode {
|
||||||
readonly changed: ISignal<string>;
|
readonly changed: ISignal<ICodeChangedEvent>;
|
||||||
readonly current: string;
|
readonly current: string;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,5 +0,0 @@
|
|||||||
import { SelectedScript } from '@/application/State/Selection/SelectedScript';
|
|
||||||
|
|
||||||
export interface IUserScriptGenerator {
|
|
||||||
buildCode(selectedScripts: ReadonlyArray<SelectedScript>, version: string): string;
|
|
||||||
}
|
|
||||||
24
src/application/State/Code/Position/CodePosition.ts
Normal file
24
src/application/State/Code/Position/CodePosition.ts
Normal file
@@ -0,0 +1,24 @@
|
|||||||
|
import { ICodePosition } from './ICodePosition';
|
||||||
|
export class CodePosition implements ICodePosition {
|
||||||
|
|
||||||
|
public get totalLines(): number {
|
||||||
|
return this.endLine - this.startLine;
|
||||||
|
}
|
||||||
|
|
||||||
|
constructor(
|
||||||
|
public readonly startLine: number,
|
||||||
|
public readonly endLine: number) {
|
||||||
|
if (startLine < 0) {
|
||||||
|
throw new Error('Code cannot start in a negative line');
|
||||||
|
}
|
||||||
|
if (endLine < 0) {
|
||||||
|
throw new Error('Code cannot end in a negative line');
|
||||||
|
}
|
||||||
|
if (endLine === startLine) {
|
||||||
|
throw new Error('Empty code');
|
||||||
|
}
|
||||||
|
if (endLine < startLine) {
|
||||||
|
throw new Error('End line cannot be less than start line');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
5
src/application/State/Code/Position/ICodePosition.ts
Normal file
5
src/application/State/Code/Position/ICodePosition.ts
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
export interface ICodePosition {
|
||||||
|
readonly startLine: number;
|
||||||
|
readonly endLine: number;
|
||||||
|
readonly totalLines: number;
|
||||||
|
}
|
||||||
@@ -1,34 +0,0 @@
|
|||||||
import { SelectedScript } from '@/application/State/Selection/SelectedScript';
|
|
||||||
import { IUserScriptGenerator } from './IUserScriptGenerator';
|
|
||||||
import { CodeBuilder } from './CodeBuilder';
|
|
||||||
|
|
||||||
export const adminRightsScript = {
|
|
||||||
name: 'Ensure admin privileges',
|
|
||||||
code: 'fltmc >nul 2>&1 || (\n' +
|
|
||||||
' echo This batch script requires administrator privileges. Right-click on\n' +
|
|
||||||
' echo the script and select "Run as administrator".\n' +
|
|
||||||
' pause\n' +
|
|
||||||
' exit 1\n' +
|
|
||||||
')',
|
|
||||||
};
|
|
||||||
|
|
||||||
export class UserScriptGenerator implements IUserScriptGenerator {
|
|
||||||
public buildCode(selectedScripts: ReadonlyArray<SelectedScript>, version: string): string {
|
|
||||||
if (!selectedScripts) { throw new Error('scripts is undefined'); }
|
|
||||||
if (!selectedScripts.length) { throw new Error('scripts are empty'); }
|
|
||||||
if (!version) { throw new Error('version is undefined'); }
|
|
||||||
const builder = new CodeBuilder()
|
|
||||||
.appendLine('@echo off')
|
|
||||||
.appendCommentLine(`https://privacy.sexy — v${version} — ${new Date().toUTCString()}`)
|
|
||||||
.appendFunction(adminRightsScript.name, adminRightsScript.code).appendLine();
|
|
||||||
for (const selection of selectedScripts) {
|
|
||||||
const name = selection.revert ? `${selection.script.name} (revert)` : selection.script.name;
|
|
||||||
const code = selection.revert ? selection.script.revertCode : selection.script.code;
|
|
||||||
builder.appendFunction(name, code).appendLine();
|
|
||||||
}
|
|
||||||
return builder.appendLine()
|
|
||||||
.appendLine('pause')
|
|
||||||
.appendLine('exit /b 0')
|
|
||||||
.toString();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -2,6 +2,7 @@ import { IFilterResult } from './IFilterResult';
|
|||||||
import { ISignal } from '@/infrastructure/Events/Signal';
|
import { ISignal } from '@/infrastructure/Events/Signal';
|
||||||
|
|
||||||
export interface IUserFilter {
|
export interface IUserFilter {
|
||||||
|
readonly currentFilter: IFilterResult | undefined;
|
||||||
readonly filtered: ISignal<IFilterResult>;
|
readonly filtered: ISignal<IFilterResult>;
|
||||||
readonly filterRemoved: ISignal<void>;
|
readonly filterRemoved: ISignal<void>;
|
||||||
setFilter(filter: string): void;
|
setFilter(filter: string): void;
|
||||||
|
|||||||
@@ -8,6 +8,7 @@ import { Signal } from '@/infrastructure/Events/Signal';
|
|||||||
export class UserFilter implements IUserFilter {
|
export class UserFilter implements IUserFilter {
|
||||||
public readonly filtered = new Signal<IFilterResult>();
|
public readonly filtered = new Signal<IFilterResult>();
|
||||||
public readonly filterRemoved = new Signal<void>();
|
public readonly filterRemoved = new Signal<void>();
|
||||||
|
public currentFilter: IFilterResult | undefined;
|
||||||
|
|
||||||
constructor(private application: IApplication) {
|
constructor(private application: IApplication) {
|
||||||
|
|
||||||
@@ -22,17 +23,17 @@ export class UserFilter implements IUserFilter {
|
|||||||
(script) => isScriptAMatch(script, filterLowercase));
|
(script) => isScriptAMatch(script, filterLowercase));
|
||||||
const filteredCategories = this.application.getAllCategories().filter(
|
const filteredCategories = this.application.getAllCategories().filter(
|
||||||
(category) => category.name.toLowerCase().includes(filterLowercase));
|
(category) => category.name.toLowerCase().includes(filterLowercase));
|
||||||
|
|
||||||
const matches = new FilterResult(
|
const matches = new FilterResult(
|
||||||
filteredScripts,
|
filteredScripts,
|
||||||
filteredCategories,
|
filteredCategories,
|
||||||
filter,
|
filter,
|
||||||
);
|
);
|
||||||
|
this.currentFilter = matches;
|
||||||
this.filtered.notify(matches);
|
this.filtered.notify(matches);
|
||||||
}
|
}
|
||||||
|
|
||||||
public removeFilter(): void {
|
public removeFilter(): void {
|
||||||
|
this.currentFilter = undefined;
|
||||||
this.filterRemoved.notify();
|
this.filterRemoved.notify();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -41,11 +42,11 @@ function isScriptAMatch(script: IScript, filterLowercase: string) {
|
|||||||
if (script.name.toLowerCase().includes(filterLowercase)) {
|
if (script.name.toLowerCase().includes(filterLowercase)) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
if (script.code.toLowerCase().includes(filterLowercase)) {
|
if (script.code.execute.toLowerCase().includes(filterLowercase)) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
if (script.revertCode) {
|
if (script.code.revert) {
|
||||||
return script.revertCode.toLowerCase().includes(filterLowercase);
|
return script.code.revert.toLowerCase().includes(filterLowercase);
|
||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,16 +1,21 @@
|
|||||||
import { SelectedScript } from './SelectedScript';
|
import { SelectedScript } from './SelectedScript';
|
||||||
import { ISignal } from '@/infrastructure/Events/Signal';
|
import { ISignal } from '@/infrastructure/Events/Signal';
|
||||||
import { IScript } from '@/domain/IScript';
|
import { IScript } from '@/domain/IScript';
|
||||||
|
import { ICategory } from '@/domain/ICategory';
|
||||||
|
|
||||||
export interface IUserSelection {
|
export interface IUserSelection {
|
||||||
readonly changed: ISignal<ReadonlyArray<SelectedScript>>;
|
readonly changed: ISignal<ReadonlyArray<SelectedScript>>;
|
||||||
readonly selectedScripts: ReadonlyArray<SelectedScript>;
|
readonly selectedScripts: ReadonlyArray<SelectedScript>;
|
||||||
readonly totalSelected: number;
|
readonly totalSelected: number;
|
||||||
|
areAllSelected(category: ICategory): boolean;
|
||||||
|
isAnySelected(category: ICategory): boolean;
|
||||||
|
removeAllInCategory(categoryId: number): void;
|
||||||
|
addOrUpdateAllInCategory(categoryId: number, revert: boolean): void;
|
||||||
addSelectedScript(scriptId: string, revert: boolean): void;
|
addSelectedScript(scriptId: string, revert: boolean): void;
|
||||||
addOrUpdateSelectedScript(scriptId: string, revert: boolean): void;
|
addOrUpdateSelectedScript(scriptId: string, revert: boolean): void;
|
||||||
removeSelectedScript(scriptId: string): void;
|
removeSelectedScript(scriptId: string): void;
|
||||||
selectOnly(scripts: ReadonlyArray<IScript>): void;
|
selectOnly(scripts: ReadonlyArray<IScript>): void;
|
||||||
isSelected(script: IScript): boolean;
|
isSelected(scriptId: string): boolean;
|
||||||
selectAll(): void;
|
selectAll(): void;
|
||||||
deselectAll(): void;
|
deselectAll(): void;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import { SelectedScript } from './SelectedScript';
|
import { SelectedScript } from './SelectedScript';
|
||||||
import { IApplication } from '@/domain/IApplication';
|
import { IApplication, ICategory } from '@/domain/IApplication';
|
||||||
import { IUserSelection } from './IUserSelection';
|
import { IUserSelection } from './IUserSelection';
|
||||||
import { InMemoryRepository } from '@/infrastructure/Repository/InMemoryRepository';
|
import { InMemoryRepository } from '@/infrastructure/Repository/InMemoryRepository';
|
||||||
import { IScript } from '@/domain/IScript';
|
import { IScript } from '@/domain/IScript';
|
||||||
@@ -8,20 +8,67 @@ import { IRepository } from '@/infrastructure/Repository/IRepository';
|
|||||||
|
|
||||||
export class UserSelection implements IUserSelection {
|
export class UserSelection implements IUserSelection {
|
||||||
public readonly changed = new Signal<ReadonlyArray<SelectedScript>>();
|
public readonly changed = new Signal<ReadonlyArray<SelectedScript>>();
|
||||||
private readonly scripts: IRepository<string, SelectedScript> = new InMemoryRepository<string, SelectedScript>();
|
private readonly scripts: IRepository<string, SelectedScript>;
|
||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
private readonly app: IApplication,
|
private readonly app: IApplication,
|
||||||
/** Initially selected scripts */
|
selectedScripts: ReadonlyArray<SelectedScript>) {
|
||||||
selectedScripts: ReadonlyArray<IScript>) {
|
this.scripts = new InMemoryRepository<string, SelectedScript>();
|
||||||
if (selectedScripts && selectedScripts.length > 0) {
|
if (selectedScripts && selectedScripts.length > 0) {
|
||||||
for (const script of selectedScripts) {
|
for (const script of selectedScripts) {
|
||||||
const selected = new SelectedScript(script, false);
|
this.scripts.addItem(script);
|
||||||
this.scripts.addItem(selected);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public areAllSelected(category: ICategory): boolean {
|
||||||
|
if (this.selectedScripts.length === 0) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
const scripts = category.getAllScriptsRecursively();
|
||||||
|
if (this.selectedScripts.length < scripts.length) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return scripts.every((script) => this.selectedScripts.some((selected) => selected.id === script.id));
|
||||||
|
}
|
||||||
|
|
||||||
|
public isAnySelected(category: ICategory): boolean {
|
||||||
|
if (this.selectedScripts.length === 0) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return this.selectedScripts.some((s) => category.includes(s.script));
|
||||||
|
}
|
||||||
|
|
||||||
|
public removeAllInCategory(categoryId: number): void {
|
||||||
|
const category = this.app.findCategory(categoryId);
|
||||||
|
const scriptsToRemove = category.getAllScriptsRecursively()
|
||||||
|
.filter((script) => this.scripts.exists(script.id));
|
||||||
|
if (!scriptsToRemove.length) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
for (const script of scriptsToRemove) {
|
||||||
|
this.scripts.removeItem(script.id);
|
||||||
|
}
|
||||||
|
this.changed.notify(this.scripts.getItems());
|
||||||
|
}
|
||||||
|
|
||||||
|
public addOrUpdateAllInCategory(categoryId: number, revert: boolean = false): void {
|
||||||
|
const category = this.app.findCategory(categoryId);
|
||||||
|
const scriptsToAddOrUpdate = category.getAllScriptsRecursively()
|
||||||
|
.filter((script) =>
|
||||||
|
!this.scripts.exists(script.id)
|
||||||
|
|| this.scripts.getById(script.id).revert !== revert,
|
||||||
|
);
|
||||||
|
if (!scriptsToAddOrUpdate.length) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
for (const script of scriptsToAddOrUpdate) {
|
||||||
|
const selectedScript = new SelectedScript(script, revert);
|
||||||
|
this.scripts.addOrUpdateItem(selectedScript);
|
||||||
|
}
|
||||||
|
this.changed.notify(this.scripts.getItems());
|
||||||
|
}
|
||||||
|
|
||||||
public addSelectedScript(scriptId: string, revert: boolean): void {
|
public addSelectedScript(scriptId: string, revert: boolean): void {
|
||||||
const script = this.app.findScript(scriptId);
|
const script = this.app.findScript(scriptId);
|
||||||
if (!script) {
|
if (!script) {
|
||||||
@@ -44,8 +91,8 @@ export class UserSelection implements IUserSelection {
|
|||||||
this.changed.notify(this.scripts.getItems());
|
this.changed.notify(this.scripts.getItems());
|
||||||
}
|
}
|
||||||
|
|
||||||
public isSelected(script: IScript): boolean {
|
public isSelected(scriptId: string): boolean {
|
||||||
return this.scripts.exists(script.id);
|
return this.scripts.exists(scriptId);
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Get users scripts based on his/her selections */
|
/** Get users scripts based on his/her selections */
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
48
src/application/application.yaml.d.ts
vendored
48
src/application/application.yaml.d.ts
vendored
@@ -1,16 +1,7 @@
|
|||||||
declare module 'js-yaml-loader!*' {
|
declare module 'js-yaml-loader!*' {
|
||||||
export type CategoryOrScript = YamlCategory | YamlScript;
|
export interface ApplicationYaml {
|
||||||
export type DocumentationUrls = ReadonlyArray<string> | string;
|
actions: ReadonlyArray<YamlCategory>;
|
||||||
|
functions: ReadonlyArray<YamlFunction> | undefined;
|
||||||
export interface YamlDocumentable {
|
|
||||||
docs?: DocumentationUrls;
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface YamlScript extends YamlDocumentable {
|
|
||||||
name: string;
|
|
||||||
code: string;
|
|
||||||
revertCode: string;
|
|
||||||
recommend: boolean;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface YamlCategory extends YamlDocumentable {
|
export interface YamlCategory extends YamlDocumentable {
|
||||||
@@ -18,10 +9,37 @@ declare module 'js-yaml-loader!*' {
|
|||||||
category: string;
|
category: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface ApplicationYaml {
|
export type CategoryOrScript = YamlCategory | YamlScript;
|
||||||
|
export type DocumentationUrls = ReadonlyArray<string> | string;
|
||||||
|
|
||||||
|
export interface YamlDocumentable {
|
||||||
|
docs?: DocumentationUrls;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface YamlFunction {
|
||||||
name: string;
|
name: string;
|
||||||
repositoryUrl: string;
|
code: string;
|
||||||
actions: ReadonlyArray<YamlCategory>;
|
revertCode?: string;
|
||||||
|
parameters?: readonly string[];
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface FunctionCallParameters {
|
||||||
|
[index: string]: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface FunctionCall {
|
||||||
|
function: string;
|
||||||
|
parameters?: FunctionCallParameters;
|
||||||
|
}
|
||||||
|
|
||||||
|
export type ScriptFunctionCall = readonly FunctionCall[] | FunctionCall | undefined;
|
||||||
|
|
||||||
|
export interface YamlScript extends YamlDocumentable {
|
||||||
|
name: string;
|
||||||
|
code: string | undefined;
|
||||||
|
revertCode: string | undefined;
|
||||||
|
call: ScriptFunctionCall;
|
||||||
|
recommend: string | undefined;
|
||||||
}
|
}
|
||||||
|
|
||||||
const content: ApplicationYaml;
|
const content: ApplicationYaml;
|
||||||
|
|||||||
@@ -2,45 +2,51 @@ import { IEntity } from '../infrastructure/Entity/IEntity';
|
|||||||
import { ICategory } from './ICategory';
|
import { ICategory } from './ICategory';
|
||||||
import { IScript } from './IScript';
|
import { IScript } from './IScript';
|
||||||
import { IApplication } from './IApplication';
|
import { IApplication } from './IApplication';
|
||||||
|
import { IProjectInformation } from './IProjectInformation';
|
||||||
|
import { RecommendationLevel, RecommendationLevelNames, RecommendationLevels } from './RecommendationLevel';
|
||||||
|
|
||||||
export class Application implements IApplication {
|
export class Application implements IApplication {
|
||||||
public get totalScripts(): number { return this.flattened.allScripts.length; }
|
public get totalScripts(): number { return this.queryable.allScripts.length; }
|
||||||
public get totalCategories(): number { return this.flattened.allCategories.length; }
|
public get totalCategories(): number { return this.queryable.allCategories.length; }
|
||||||
|
|
||||||
private readonly flattened: IFlattenedApplication;
|
private readonly queryable: IQueryableApplication;
|
||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
public readonly name: string,
|
public readonly info: IProjectInformation,
|
||||||
public readonly repositoryUrl: string,
|
|
||||||
public readonly version: string,
|
|
||||||
public readonly actions: ReadonlyArray<ICategory>) {
|
public readonly actions: ReadonlyArray<ICategory>) {
|
||||||
if (!name) { throw Error('Application has no name'); }
|
if (!info) {
|
||||||
if (!repositoryUrl) { throw Error('Application has no repository url'); }
|
throw new Error('info is undefined');
|
||||||
if (!version) { throw Error('Version cannot be empty'); }
|
}
|
||||||
this.flattened = flatten(actions);
|
this.queryable = makeQueryable(actions);
|
||||||
ensureValid(this.flattened);
|
ensureValid(this.queryable);
|
||||||
ensureNoDuplicates(this.flattened.allCategories);
|
ensureNoDuplicates(this.queryable.allCategories);
|
||||||
ensureNoDuplicates(this.flattened.allScripts);
|
ensureNoDuplicates(this.queryable.allScripts);
|
||||||
}
|
}
|
||||||
|
|
||||||
public findCategory(categoryId: number): ICategory | undefined {
|
public findCategory(categoryId: number): ICategory | undefined {
|
||||||
return this.flattened.allCategories.find((category) => category.id === categoryId);
|
return this.queryable.allCategories.find((category) => category.id === categoryId);
|
||||||
}
|
}
|
||||||
|
|
||||||
public getRecommendedScripts(): readonly IScript[] {
|
public getScriptsByLevel(level: RecommendationLevel): readonly IScript[] {
|
||||||
return this.flattened.allScripts.filter((script) => script.isRecommended);
|
if (isNaN(level)) {
|
||||||
|
throw new Error('undefined level');
|
||||||
|
}
|
||||||
|
if (!(level in RecommendationLevel)) {
|
||||||
|
throw new Error(`invalid level: ${level}`);
|
||||||
|
}
|
||||||
|
return this.queryable.scriptsByLevel.get(level);
|
||||||
}
|
}
|
||||||
|
|
||||||
public findScript(scriptId: string): IScript | undefined {
|
public findScript(scriptId: string): IScript | undefined {
|
||||||
return this.flattened.allScripts.find((script) => script.id === scriptId);
|
return this.queryable.allScripts.find((script) => script.id === scriptId);
|
||||||
}
|
}
|
||||||
|
|
||||||
public getAllScripts(): IScript[] {
|
public getAllScripts(): IScript[] {
|
||||||
return this.flattened.allScripts;
|
return this.queryable.allScripts;
|
||||||
}
|
}
|
||||||
|
|
||||||
public getAllCategories(): ICategory[] {
|
public getAllCategories(): ICategory[] {
|
||||||
return this.flattened.allCategories;
|
return this.queryable.allCategories;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -62,55 +68,85 @@ function ensureNoDuplicates<TKey>(entities: ReadonlyArray<IEntity<TKey>>) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
interface IFlattenedApplication {
|
interface IQueryableApplication {
|
||||||
allCategories: ICategory[];
|
allCategories: ICategory[];
|
||||||
allScripts: IScript[];
|
allScripts: IScript[];
|
||||||
|
scriptsByLevel: Map<RecommendationLevel, readonly IScript[]>;
|
||||||
}
|
}
|
||||||
|
|
||||||
function ensureValid(application: IFlattenedApplication) {
|
function ensureValid(application: IQueryableApplication) {
|
||||||
if (!application.allCategories || application.allCategories.length === 0) {
|
ensureValidCategories(application.allCategories);
|
||||||
|
ensureValidScripts(application.allScripts);
|
||||||
|
}
|
||||||
|
|
||||||
|
function ensureValidCategories(allCategories: readonly ICategory[]) {
|
||||||
|
if (!allCategories || allCategories.length === 0) {
|
||||||
throw new Error('Application must consist of at least one category');
|
throw new Error('Application must consist of at least one category');
|
||||||
}
|
}
|
||||||
if (!application.allScripts || application.allScripts.length === 0) {
|
}
|
||||||
|
|
||||||
|
function ensureValidScripts(allScripts: readonly IScript[]) {
|
||||||
|
if (!allScripts || allScripts.length === 0) {
|
||||||
throw new Error('Application must consist of at least one script');
|
throw new Error('Application must consist of at least one script');
|
||||||
}
|
}
|
||||||
if (application.allScripts.filter((script) => script.isRecommended).length === 0) {
|
for (const level of RecommendationLevels) {
|
||||||
throw new Error('Application must consist of at least one recommended script');
|
if (allScripts.every((script) => script.level !== level)) {
|
||||||
|
throw new Error(`none of the scripts are recommended as ${RecommendationLevel[level]}`);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function flattenApplication(categories: ReadonlyArray<ICategory>): [ICategory[], IScript[]] {
|
||||||
|
const allCategories = new Array<ICategory>();
|
||||||
|
const allScripts = new Array<IScript>();
|
||||||
|
flattenCategories(categories, allCategories, allScripts);
|
||||||
|
return [
|
||||||
|
allCategories,
|
||||||
|
allScripts,
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
function flattenCategories(
|
function flattenCategories(
|
||||||
categories: ReadonlyArray<ICategory>,
|
categories: ReadonlyArray<ICategory>,
|
||||||
flattened: IFlattenedApplication): IFlattenedApplication {
|
allCategories: ICategory[],
|
||||||
|
allScripts: IScript[]): IQueryableApplication {
|
||||||
if (!categories || categories.length === 0) {
|
if (!categories || categories.length === 0) {
|
||||||
return flattened;
|
return;
|
||||||
}
|
}
|
||||||
for (const category of categories) {
|
for (const category of categories) {
|
||||||
flattened.allCategories.push(category);
|
allCategories.push(category);
|
||||||
flattened = flattenScripts(category.scripts, flattened);
|
flattenScripts(category.scripts, allScripts);
|
||||||
flattened = flattenCategories(category.subCategories, flattened);
|
flattenCategories(category.subCategories, allCategories, allScripts);
|
||||||
}
|
}
|
||||||
return flattened;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function flattenScripts(
|
function flattenScripts(
|
||||||
scripts: ReadonlyArray<IScript>,
|
scripts: ReadonlyArray<IScript>,
|
||||||
flattened: IFlattenedApplication): IFlattenedApplication {
|
allScripts: IScript[]): IScript[] {
|
||||||
if (!scripts) {
|
if (!scripts) {
|
||||||
return flattened;
|
return;
|
||||||
}
|
}
|
||||||
for (const script of scripts) {
|
for (const script of scripts) {
|
||||||
flattened.allScripts.push(script);
|
allScripts.push(script);
|
||||||
}
|
}
|
||||||
return flattened;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function flatten(
|
function makeQueryable(
|
||||||
categories: ReadonlyArray<ICategory>): IFlattenedApplication {
|
actions: ReadonlyArray<ICategory>): IQueryableApplication {
|
||||||
let flattened: IFlattenedApplication = {
|
const flattened = flattenApplication(actions);
|
||||||
allCategories: new Array<ICategory>(),
|
return {
|
||||||
allScripts: new Array<IScript>(),
|
allCategories: flattened[0],
|
||||||
|
allScripts: flattened[1],
|
||||||
|
scriptsByLevel: groupByLevel(flattened[1]),
|
||||||
};
|
};
|
||||||
flattened = flattenCategories(categories, flattened);
|
}
|
||||||
return flattened;
|
|
||||||
|
function groupByLevel(allScripts: readonly IScript[]): Map<RecommendationLevel, readonly IScript[]> {
|
||||||
|
const map = new Map<RecommendationLevel, readonly IScript[]>();
|
||||||
|
for (const levelName of RecommendationLevelNames) {
|
||||||
|
const level = RecommendationLevel[levelName];
|
||||||
|
const scripts = allScripts.filter((script) => script.level !== undefined && script.level <= level);
|
||||||
|
map.set(level, scripts);
|
||||||
|
}
|
||||||
|
return map;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,15 +3,7 @@ import { IScript } from './IScript';
|
|||||||
import { ICategory } from './ICategory';
|
import { ICategory } from './ICategory';
|
||||||
|
|
||||||
export class Category extends BaseEntity<number> implements ICategory {
|
export class Category extends BaseEntity<number> implements ICategory {
|
||||||
private static validate(category: ICategory) {
|
private allSubScripts: ReadonlyArray<IScript> = undefined;
|
||||||
if (!category.name) {
|
|
||||||
throw new Error('name is null or empty');
|
|
||||||
}
|
|
||||||
if ((!category.subCategories || category.subCategories.length === 0) &&
|
|
||||||
(!category.scripts || category.scripts.length === 0)) {
|
|
||||||
throw new Error('A category must have at least one sub-category or scripts');
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
id: number,
|
id: number,
|
||||||
@@ -20,6 +12,31 @@ export class Category extends BaseEntity<number> implements ICategory {
|
|||||||
public readonly subCategories?: ReadonlyArray<ICategory>,
|
public readonly subCategories?: ReadonlyArray<ICategory>,
|
||||||
public readonly scripts?: ReadonlyArray<IScript>) {
|
public readonly scripts?: ReadonlyArray<IScript>) {
|
||||||
super(id);
|
super(id);
|
||||||
Category.validate(this);
|
validateCategory(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
public includes(script: IScript): boolean {
|
||||||
|
return this.getAllScriptsRecursively().some((childScript) => childScript.id === script.id);
|
||||||
|
}
|
||||||
|
|
||||||
|
public getAllScriptsRecursively(): readonly IScript[] {
|
||||||
|
return this.allSubScripts || (this.allSubScripts = parseScriptsRecursively(this));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function parseScriptsRecursively(category: ICategory): ReadonlyArray<IScript> {
|
||||||
|
return [
|
||||||
|
...category.scripts,
|
||||||
|
...category.subCategories.flatMap((c) => c.getAllScriptsRecursively()),
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
function validateCategory(category: ICategory) {
|
||||||
|
if (!category.name) {
|
||||||
|
throw new Error('undefined or empty name');
|
||||||
|
}
|
||||||
|
if ((!category.subCategories || category.subCategories.length === 0) &&
|
||||||
|
(!category.scripts || category.scripts.length === 0)) {
|
||||||
|
throw new Error('A category must have at least one sub-category or script');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,15 +1,15 @@
|
|||||||
import { IScript } from '@/domain/IScript';
|
import { IScript } from '@/domain/IScript';
|
||||||
import { ICategory } from '@/domain/ICategory';
|
import { ICategory } from '@/domain/ICategory';
|
||||||
|
import { IProjectInformation } from './IProjectInformation';
|
||||||
|
import { RecommendationLevel } from './RecommendationLevel';
|
||||||
|
|
||||||
export interface IApplication {
|
export interface IApplication {
|
||||||
readonly name: string;
|
readonly info: IProjectInformation;
|
||||||
readonly repositoryUrl: string;
|
|
||||||
readonly version: string;
|
|
||||||
readonly totalScripts: number;
|
readonly totalScripts: number;
|
||||||
readonly totalCategories: number;
|
readonly totalCategories: number;
|
||||||
readonly actions: ReadonlyArray<ICategory>;
|
readonly actions: ReadonlyArray<ICategory>;
|
||||||
|
|
||||||
getRecommendedScripts(): ReadonlyArray<IScript>;
|
getScriptsByLevel(level: RecommendationLevel): ReadonlyArray<IScript>;
|
||||||
findCategory(categoryId: number): ICategory | undefined;
|
findCategory(categoryId: number): ICategory | undefined;
|
||||||
findScript(scriptId: string): IScript | undefined;
|
findScript(scriptId: string): IScript | undefined;
|
||||||
getAllScripts(): ReadonlyArray<IScript>;
|
getAllScripts(): ReadonlyArray<IScript>;
|
||||||
|
|||||||
@@ -7,6 +7,8 @@ export interface ICategory extends IEntity<number>, IDocumentable {
|
|||||||
readonly name: string;
|
readonly name: string;
|
||||||
readonly subCategories?: ReadonlyArray<ICategory>;
|
readonly subCategories?: ReadonlyArray<ICategory>;
|
||||||
readonly scripts?: ReadonlyArray<IScript>;
|
readonly scripts?: ReadonlyArray<IScript>;
|
||||||
|
includes(script: IScript): boolean;
|
||||||
|
getAllScriptsRecursively(): ReadonlyArray<IScript>;
|
||||||
}
|
}
|
||||||
|
|
||||||
export { IEntity } from '../infrastructure/Entity/IEntity';
|
export { IEntity } from '../infrastructure/Entity/IEntity';
|
||||||
|
|||||||
11
src/domain/IProjectInformation.ts
Normal file
11
src/domain/IProjectInformation.ts
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
import { OperatingSystem } from './OperatingSystem';
|
||||||
|
export interface IProjectInformation {
|
||||||
|
readonly name: string;
|
||||||
|
readonly version: string;
|
||||||
|
readonly repositoryUrl: string;
|
||||||
|
readonly homepage: string;
|
||||||
|
readonly feedbackUrl: string;
|
||||||
|
readonly releaseUrl: string;
|
||||||
|
readonly repositoryWebUrl: string;
|
||||||
|
getDownloadUrl(os: OperatingSystem): string;
|
||||||
|
}
|
||||||
@@ -1,11 +1,12 @@
|
|||||||
import { IEntity } from '../infrastructure/Entity/IEntity';
|
import { IEntity } from '../infrastructure/Entity/IEntity';
|
||||||
import { IDocumentable } from './IDocumentable';
|
import { IDocumentable } from './IDocumentable';
|
||||||
|
import { RecommendationLevel } from './RecommendationLevel';
|
||||||
|
import { IScriptCode } from './IScriptCode';
|
||||||
|
|
||||||
export interface IScript extends IEntity<string>, IDocumentable {
|
export interface IScript extends IEntity<string>, IDocumentable {
|
||||||
readonly name: string;
|
readonly name: string;
|
||||||
readonly isRecommended: boolean;
|
readonly level?: RecommendationLevel;
|
||||||
readonly documentationUrls: ReadonlyArray<string>;
|
readonly documentationUrls: ReadonlyArray<string>;
|
||||||
readonly code: string;
|
readonly code: IScriptCode;
|
||||||
readonly revertCode: string;
|
|
||||||
canRevert(): boolean;
|
canRevert(): boolean;
|
||||||
}
|
}
|
||||||
|
|||||||
4
src/domain/IScriptCode.ts
Normal file
4
src/domain/IScriptCode.ts
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
export interface IScriptCode {
|
||||||
|
readonly execute: string;
|
||||||
|
readonly revert: string;
|
||||||
|
}
|
||||||
55
src/domain/ProjectInformation.ts
Normal file
55
src/domain/ProjectInformation.ts
Normal file
@@ -0,0 +1,55 @@
|
|||||||
|
import { IProjectInformation } from './IProjectInformation';
|
||||||
|
import { OperatingSystem } from './OperatingSystem';
|
||||||
|
|
||||||
|
export class ProjectInformation implements IProjectInformation {
|
||||||
|
public readonly repositoryWebUrl: string;
|
||||||
|
constructor(
|
||||||
|
public readonly name: string,
|
||||||
|
public readonly version: string,
|
||||||
|
public readonly repositoryUrl: string,
|
||||||
|
public readonly homepage: string,
|
||||||
|
) {
|
||||||
|
if (!name) {
|
||||||
|
throw new Error('name is undefined');
|
||||||
|
}
|
||||||
|
if (!version || +version <= 0) {
|
||||||
|
throw new Error('version should be higher than zero');
|
||||||
|
}
|
||||||
|
if (!repositoryUrl) {
|
||||||
|
throw new Error('repositoryUrl is undefined');
|
||||||
|
}
|
||||||
|
if (!homepage) {
|
||||||
|
throw new Error('homepage is undefined');
|
||||||
|
}
|
||||||
|
this.repositoryWebUrl = getWebUrl(this.repositoryUrl);
|
||||||
|
}
|
||||||
|
public getDownloadUrl(os: OperatingSystem): string {
|
||||||
|
return `${this.repositoryWebUrl}/releases/download/${this.version}/${getFileName(os, this.version)}`;
|
||||||
|
}
|
||||||
|
public get feedbackUrl(): string {
|
||||||
|
return `${this.repositoryWebUrl}/issues`;
|
||||||
|
}
|
||||||
|
public get releaseUrl(): string {
|
||||||
|
return `${this.repositoryWebUrl}/releases/tag/${this.version}`;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function getWebUrl(gitUrl: string) {
|
||||||
|
if (gitUrl.endsWith('.git')) {
|
||||||
|
return gitUrl.substring(0, gitUrl.length - 4);
|
||||||
|
}
|
||||||
|
return gitUrl;
|
||||||
|
}
|
||||||
|
|
||||||
|
function getFileName(os: OperatingSystem, version: string): string {
|
||||||
|
switch (os) {
|
||||||
|
case OperatingSystem.Linux:
|
||||||
|
return `privacy.sexy-${version}.AppImage`;
|
||||||
|
case OperatingSystem.macOS:
|
||||||
|
return `privacy.sexy-${version}.dmg`;
|
||||||
|
case OperatingSystem.Windows:
|
||||||
|
return `privacy.sexy-Setup-${version}.exe`;
|
||||||
|
default:
|
||||||
|
throw new Error(`Unsupported os: ${OperatingSystem[os]}`);
|
||||||
|
}
|
||||||
|
}
|
||||||
11
src/domain/RecommendationLevel.ts
Normal file
11
src/domain/RecommendationLevel.ts
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
export enum RecommendationLevel {
|
||||||
|
Standard = 0,
|
||||||
|
Strict = 1,
|
||||||
|
}
|
||||||
|
|
||||||
|
export const RecommendationLevelNames = Object
|
||||||
|
.values(RecommendationLevel)
|
||||||
|
.filter((level) => typeof level === 'string') as string[];
|
||||||
|
|
||||||
|
export const RecommendationLevels = RecommendationLevelNames
|
||||||
|
.map((level) => RecommendationLevel[level]) as RecommendationLevel[];
|
||||||
@@ -1,57 +1,27 @@
|
|||||||
import { BaseEntity } from '@/infrastructure/Entity/BaseEntity';
|
import { BaseEntity } from '@/infrastructure/Entity/BaseEntity';
|
||||||
import { IScript } from './IScript';
|
import { IScript } from './IScript';
|
||||||
|
import { RecommendationLevel } from './RecommendationLevel';
|
||||||
|
import { IScriptCode } from './IScriptCode';
|
||||||
|
|
||||||
export class Script extends BaseEntity<string> implements IScript {
|
export class Script extends BaseEntity<string> implements IScript {
|
||||||
constructor(
|
constructor(
|
||||||
public readonly name: string,
|
public readonly name: string,
|
||||||
public readonly code: string,
|
public readonly code: IScriptCode,
|
||||||
public readonly revertCode: string,
|
|
||||||
public readonly documentationUrls: ReadonlyArray<string>,
|
public readonly documentationUrls: ReadonlyArray<string>,
|
||||||
public readonly isRecommended: boolean) {
|
public readonly level?: RecommendationLevel) {
|
||||||
super(name);
|
super(name);
|
||||||
validateCode(name, code);
|
if (!code) {
|
||||||
if (revertCode) {
|
throw new Error(`undefined code (script: ${name})`);
|
||||||
validateCode(name, revertCode);
|
|
||||||
if (code === revertCode) {
|
|
||||||
throw new Error(`${name}: Code itself and its reverting code cannot be the same`);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
validateLevel(level);
|
||||||
}
|
}
|
||||||
public canRevert(): boolean {
|
public canRevert(): boolean {
|
||||||
return Boolean(this.revertCode);
|
return Boolean(this.code.revert);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function validateCode(name: string, code: string): void {
|
function validateLevel(level?: RecommendationLevel) {
|
||||||
if (!code || code.length === 0) {
|
if (level !== undefined && !(level in RecommendationLevel)) {
|
||||||
throw new Error(`Code of ${name} is empty or null`);
|
throw new Error(`invalid level: ${level}`);
|
||||||
}
|
|
||||||
ensureCodeHasUniqueLines(name, code);
|
|
||||||
ensureNoEmptyLines(name, code);
|
|
||||||
}
|
|
||||||
|
|
||||||
function ensureNoEmptyLines(name: string, code: string): void {
|
|
||||||
if (code.split('\n').some((line) => line.trim().length === 0)) {
|
|
||||||
throw Error(`Script has empty lines "${name}"`);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function mayBeUniqueLine(codeLine: string): boolean {
|
|
||||||
const trimmed = codeLine.trim();
|
|
||||||
if (trimmed === ')' || trimmed === '(') { // "(" and ")" are used often in batch code
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
function ensureCodeHasUniqueLines(name: string, code: string): void {
|
|
||||||
const lines = code.split('\n')
|
|
||||||
.filter((line) => mayBeUniqueLine(line));
|
|
||||||
if (lines.length === 0) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
const duplicateLines = lines.filter((e, i, a) => a.indexOf(e) !== i);
|
|
||||||
if (duplicateLines.length !== 0) {
|
|
||||||
throw Error(`Duplicates detected in script "${name}":\n ${duplicateLines.join('\n')}`);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
57
src/domain/ScriptCode.ts
Normal file
57
src/domain/ScriptCode.ts
Normal file
@@ -0,0 +1,57 @@
|
|||||||
|
import { IScriptCode } from './IScriptCode';
|
||||||
|
|
||||||
|
export class ScriptCode implements IScriptCode {
|
||||||
|
constructor(
|
||||||
|
scriptName: string,
|
||||||
|
public readonly execute: string,
|
||||||
|
public readonly revert: string) {
|
||||||
|
if (!scriptName) {
|
||||||
|
throw new Error('script name is undefined');
|
||||||
|
}
|
||||||
|
validateCode(scriptName, execute);
|
||||||
|
if (revert) {
|
||||||
|
scriptName = `${scriptName} (revert)`;
|
||||||
|
validateCode(scriptName, revert);
|
||||||
|
if (execute === revert) {
|
||||||
|
throw new Error(`${scriptName}: Code itself and its reverting code cannot be the same`);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function validateCode(name: string, code: string): void {
|
||||||
|
if (!code || code.length === 0) {
|
||||||
|
throw new Error(`code of ${name} is empty or undefined`);
|
||||||
|
}
|
||||||
|
ensureCodeHasUniqueLines(name, code);
|
||||||
|
ensureNoEmptyLines(name, code);
|
||||||
|
}
|
||||||
|
|
||||||
|
function ensureNoEmptyLines(name: string, code: string): void {
|
||||||
|
if (code.split('\n').some((line) => line.trim().length === 0)) {
|
||||||
|
throw Error(`Script has empty lines "${name}"`);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function ensureCodeHasUniqueLines(name: string, code: string): void {
|
||||||
|
const lines = code.split('\n')
|
||||||
|
.filter((line) => mayBeUniqueLine(line));
|
||||||
|
if (lines.length === 0) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const duplicateLines = lines.filter((e, i, a) => a.indexOf(e) !== i);
|
||||||
|
if (duplicateLines.length !== 0) {
|
||||||
|
throw Error(`Duplicates detected in script "${name}":\n ${duplicateLines.join('\n')}`);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function mayBeUniqueLine(codeLine: string): boolean {
|
||||||
|
const trimmed = codeLine.trim();
|
||||||
|
if (trimmed === ')' || trimmed === '(') { // "(" and ")" are used often in batch code
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (codeLine.startsWith(':: ') || codeLine.startsWith('REM ')) { // Is comment?
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
@@ -3,6 +3,7 @@ import { IEntity } from '../Entity/IEntity';
|
|||||||
export interface IRepository<TKey, TEntity extends IEntity<TKey>> {
|
export interface IRepository<TKey, TEntity extends IEntity<TKey>> {
|
||||||
readonly length: number;
|
readonly length: number;
|
||||||
getItems(predicate?: (entity: TEntity) => boolean): TEntity[];
|
getItems(predicate?: (entity: TEntity) => boolean): TEntity[];
|
||||||
|
getById(id: TKey): TEntity | undefined;
|
||||||
addItem(item: TEntity): void;
|
addItem(item: TEntity): void;
|
||||||
addOrUpdateItem(item: TEntity): void;
|
addOrUpdateItem(item: TEntity): void;
|
||||||
removeItem(id: TKey): void;
|
removeItem(id: TKey): void;
|
||||||
|
|||||||
@@ -16,6 +16,14 @@ export class InMemoryRepository<TKey, TEntity extends IEntity<TKey>> implements
|
|||||||
return predicate ? this.items.filter(predicate) : this.items;
|
return predicate ? this.items.filter(predicate) : this.items;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public getById(id: TKey): TEntity | undefined {
|
||||||
|
const items = this.getItems((entity) => entity.id === id);
|
||||||
|
if (!items.length) {
|
||||||
|
return undefined;
|
||||||
|
}
|
||||||
|
return items[0];
|
||||||
|
}
|
||||||
|
|
||||||
public addItem(item: TEntity): void {
|
public addItem(item: TEntity): void {
|
||||||
if (!item) {
|
if (!item) {
|
||||||
throw new Error('item is null or undefined');
|
throw new Error('item is null or undefined');
|
||||||
|
|||||||
@@ -1,9 +1,18 @@
|
|||||||
import fileSaver from 'file-saver';
|
import fileSaver from 'file-saver';
|
||||||
|
|
||||||
|
export enum FileType {
|
||||||
|
BatchFile,
|
||||||
|
}
|
||||||
export class SaveFileDialog {
|
export class SaveFileDialog {
|
||||||
public static saveText(text: string, fileName: string): void {
|
public static saveFile(text: string, fileName: string, type: FileType): void {
|
||||||
this.saveBlob(text, 'text/plain;charset=utf-8', fileName);
|
const mimeType = this.mimeTypes.get(type);
|
||||||
|
this.saveBlob(text, mimeType, fileName);
|
||||||
}
|
}
|
||||||
|
private static readonly mimeTypes = new Map<FileType, string>([
|
||||||
|
// Some browsers (including firefox + IE) require right mime type
|
||||||
|
// otherwise they ignore extension and save the file as text.
|
||||||
|
[ FileType.BatchFile, 'application/bat' ], // https://en.wikipedia.org/wiki/Batch_file
|
||||||
|
]);
|
||||||
|
|
||||||
private static saveBlob(file: BlobPart, fileType: string, fileName: string): void {
|
private static saveBlob(file: BlobPart, fileType: string, fileName: string): void {
|
||||||
try {
|
try {
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
import Vue from 'vue';
|
import Vue from 'vue';
|
||||||
import App from './App.vue';
|
import App from './App.vue';
|
||||||
import { ApplicationBootstrapper } from './presentation/Bootstrapping/ApplicationBootstrapper';
|
import { ApplicationBootstrapper } from './presentation/Bootstrapping/ApplicationBootstrapper';
|
||||||
|
import 'core-js/fn/array/flat-map'; // Here until Vue 3 & CLI v4 https://github.com/vuejs/vue-cli/issues/3834
|
||||||
|
|
||||||
new ApplicationBootstrapper()
|
new ApplicationBootstrapper()
|
||||||
.bootstrap(Vue);
|
.bootstrap(Vue);
|
||||||
|
|||||||
@@ -6,7 +6,8 @@ import { FontAwesomeIcon } from '@fortawesome/vue-fontawesome';
|
|||||||
/** REGULAR ICONS (PREFIX: far) */
|
/** REGULAR ICONS (PREFIX: far) */
|
||||||
import { faFolderOpen, faFolder, faSmile } from '@fortawesome/free-regular-svg-icons';
|
import { faFolderOpen, faFolder, faSmile } from '@fortawesome/free-regular-svg-icons';
|
||||||
/** SOLID ICONS (PREFIX: fas (default)) */
|
/** SOLID ICONS (PREFIX: fas (default)) */
|
||||||
import { faTimes, faFileDownload, faCopy, faSearch, faInfoCircle, faUserSecret, faDesktop, faTag, faGlobe } from '@fortawesome/free-solid-svg-icons';
|
import { faTimes, faFileDownload, faCopy, faSearch, faInfoCircle, faUserSecret, faDesktop,
|
||||||
|
faTag, faGlobe, faSave, faBatteryFull, faBatteryHalf } from '@fortawesome/free-solid-svg-icons';
|
||||||
|
|
||||||
|
|
||||||
export class IconBootstrapper implements IVueBootstrapper {
|
export class IconBootstrapper implements IVueBootstrapper {
|
||||||
@@ -21,9 +22,10 @@ export class IconBootstrapper implements IVueBootstrapper {
|
|||||||
faFolderOpen,
|
faFolderOpen,
|
||||||
faFolder,
|
faFolder,
|
||||||
faTimes,
|
faTimes,
|
||||||
faFileDownload,
|
faFileDownload, faSave,
|
||||||
faCopy,
|
faCopy,
|
||||||
faSearch,
|
faSearch,
|
||||||
|
faBatteryFull, faBatteryHalf,
|
||||||
faInfoCircle);
|
faInfoCircle);
|
||||||
vue.component('font-awesome-icon', FontAwesomeIcon);
|
vue.component('font-awesome-icon', FontAwesomeIcon);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -8,10 +8,8 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { Component, Prop, Vue, Emit } from 'vue-property-decorator';
|
import { Component, Prop, Emit } from 'vue-property-decorator';
|
||||||
import { StatefulVue } from './StatefulVue';
|
import { StatefulVue } from './StatefulVue';
|
||||||
import { SaveFileDialog } from './../infrastructure/SaveFileDialog';
|
|
||||||
import { Clipboard } from './../infrastructure/Clipboard';
|
|
||||||
|
|
||||||
@Component
|
@Component
|
||||||
export default class IconButton extends StatefulVue {
|
export default class IconButton extends StatefulVue {
|
||||||
|
|||||||
@@ -16,7 +16,7 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { Component, Prop, Vue } from 'vue-property-decorator';
|
import { Component } from 'vue-property-decorator';
|
||||||
import CardListItem from './CardListItem.vue';
|
import CardListItem from './CardListItem.vue';
|
||||||
import { StatefulVue } from '@/presentation/StatefulVue';
|
import { StatefulVue } from '@/presentation/StatefulVue';
|
||||||
import { ICategory } from '@/domain/ICategory';
|
import { ICategory } from '@/domain/ICategory';
|
||||||
|
|||||||
@@ -8,9 +8,17 @@
|
|||||||
}"
|
}"
|
||||||
ref="cardElement">
|
ref="cardElement">
|
||||||
<div class="card__inner">
|
<div class="card__inner">
|
||||||
<span v-if="cardTitle && cardTitle.length > 0">{{cardTitle}}</span>
|
<span v-if="cardTitle && cardTitle.length > 0">
|
||||||
|
<span>{{cardTitle}}</span>
|
||||||
|
</span>
|
||||||
<span v-else>Oh no 😢</span>
|
<span v-else>Oh no 😢</span>
|
||||||
|
<!-- Expand icon -->
|
||||||
<font-awesome-icon :icon="['far', isExpanded ? 'folder-open' : 'folder']" class="card__inner__expand-icon" />
|
<font-awesome-icon :icon="['far', isExpanded ? 'folder-open' : 'folder']" class="card__inner__expand-icon" />
|
||||||
|
<!-- Indeterminate and full states -->
|
||||||
|
<div class="card__inner__state-icons">
|
||||||
|
<font-awesome-icon v-if="isAnyChildSelected && !areAllChildrenSelected" :icon="['fa', 'battery-half']" />
|
||||||
|
<font-awesome-icon v-if="areAllChildrenSelected" :icon="['fa', 'battery-full']" />
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="card__expander" v-on:click.stop>
|
<div class="card__expander" v-on:click.stop>
|
||||||
<div class="card__expander__content">
|
<div class="card__expander__content">
|
||||||
@@ -24,9 +32,11 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { Component, Prop, Vue, Watch, Emit } from 'vue-property-decorator';
|
import { Component, Prop, Watch, Emit } from 'vue-property-decorator';
|
||||||
import ScriptsTree from '@/presentation/Scripts/ScriptsTree/ScriptsTree.vue';
|
import ScriptsTree from '@/presentation/Scripts/ScriptsTree/ScriptsTree.vue';
|
||||||
import { StatefulVue } from '@/presentation/StatefulVue';
|
import { StatefulVue } from '@/presentation/StatefulVue';
|
||||||
|
import { ICategory } from '@/domain/ICategory';
|
||||||
|
import { IUserSelection } from '@/application/State/IApplicationState';
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
components: {
|
components: {
|
||||||
@@ -36,17 +46,21 @@ import { StatefulVue } from '@/presentation/StatefulVue';
|
|||||||
export default class CardListItem extends StatefulVue {
|
export default class CardListItem extends StatefulVue {
|
||||||
@Prop() public categoryId!: number;
|
@Prop() public categoryId!: number;
|
||||||
@Prop() public activeCategoryId!: number;
|
@Prop() public activeCategoryId!: number;
|
||||||
public cardTitle?: string = '';
|
public cardTitle = '';
|
||||||
public isExpanded: boolean = false;
|
public isExpanded = false;
|
||||||
|
public isAnyChildSelected = false;
|
||||||
|
public areAllChildrenSelected = false;
|
||||||
|
|
||||||
@Emit('selected')
|
@Emit('selected')
|
||||||
public onSelected(isExpanded: boolean) {
|
public onSelected(isExpanded: boolean) {
|
||||||
this.isExpanded = isExpanded;
|
this.isExpanded = isExpanded;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Watch('activeCategoryId')
|
@Watch('activeCategoryId')
|
||||||
public async onActiveCategoryChanged(value: |number) {
|
public async onActiveCategoryChanged(value: |number) {
|
||||||
this.isExpanded = value === this.categoryId;
|
this.isExpanded = value === this.categoryId;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Watch('isExpanded')
|
@Watch('isExpanded')
|
||||||
public async onExpansionChangedAsync(newValue: number, oldValue: number) {
|
public async onExpansionChangedAsync(newValue: number, oldValue: number) {
|
||||||
if (!oldValue && newValue) {
|
if (!oldValue && newValue) {
|
||||||
@@ -57,20 +71,23 @@ export default class CardListItem extends StatefulVue {
|
|||||||
}
|
}
|
||||||
|
|
||||||
public async mounted() {
|
public async mounted() {
|
||||||
this.cardTitle = this.categoryId ? await this.getCardTitleAsync(this.categoryId) : undefined;
|
const state = await this.getCurrentStateAsync();
|
||||||
|
state.selection.changed.on(() => {
|
||||||
|
this.updateStateAsync(this.categoryId);
|
||||||
|
});
|
||||||
|
this.updateStateAsync(this.categoryId);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Watch('categoryId')
|
@Watch('categoryId')
|
||||||
public async onCategoryIdChanged(value: |number) {
|
public async updateStateAsync(value: |number) {
|
||||||
this.cardTitle = value ? await this.getCardTitleAsync(value) : undefined;
|
const state = await this.getCurrentStateAsync();
|
||||||
}
|
const category = !value ? undefined : state.app.findCategory(this.categoryId);
|
||||||
|
this.cardTitle = category ? category.name : undefined;
|
||||||
private async getCardTitleAsync(categoryId: number): Promise<string | undefined> {
|
this.isAnyChildSelected = category ? state.selection.isAnySelected(category) : false;
|
||||||
const state = await this.getCurrentStateAsync();
|
this.areAllChildrenSelected = category ? state.selection.areAllSelected(category) : false;
|
||||||
const category = state.app.findCategory(this.categoryId);
|
|
||||||
return category ? category.name : undefined;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style scoped lang="scss">
|
<style scoped lang="scss">
|
||||||
@@ -93,7 +110,7 @@ $expanded-margin-top: 30px;
|
|||||||
@media screen and (max-width: $small-screen-width) { width: 90%; }
|
@media screen and (max-width: $small-screen-width) { width: 90%; }
|
||||||
|
|
||||||
&__inner {
|
&__inner {
|
||||||
padding: $card-padding;
|
padding: $card-padding $card-padding 0 $card-padding;
|
||||||
position: relative;
|
position: relative;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
background-color: $gray;
|
background-color: $gray;
|
||||||
@@ -115,13 +132,21 @@ $expanded-margin-top: 30px;
|
|||||||
&:after {
|
&:after {
|
||||||
transition: all 0.3s ease-in-out;
|
transition: all 0.3s ease-in-out;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
&__state-icons {
|
||||||
|
height: $card-padding;
|
||||||
|
margin-right: -$card-padding;
|
||||||
|
padding-right: 10px;
|
||||||
|
display: flex;
|
||||||
|
justify-content: flex-end;
|
||||||
|
}
|
||||||
|
|
||||||
&__expand-icon {
|
&__expand-icon {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
margin-top: .25em;
|
margin-top: .25em;
|
||||||
|
vertical-align: middle;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
&__expander {
|
&__expander {
|
||||||
transition: all 0.2s ease-in-out;
|
transition: all 0.2s ease-in-out;
|
||||||
position: relative;
|
position: relative;
|
||||||
|
|||||||
@@ -15,9 +15,8 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { Component, Prop, Vue } from 'vue-property-decorator';
|
import { Component } from 'vue-property-decorator';
|
||||||
import { StatefulVue } from '@/presentation/StatefulVue';
|
import { StatefulVue } from '@/presentation/StatefulVue';
|
||||||
import { IApplicationState } from '@/application/State/IApplicationState';
|
|
||||||
import { Grouping } from './Grouping';
|
import { Grouping } from './Grouping';
|
||||||
|
|
||||||
const DefaultGrouping = Grouping.Cards;
|
const DefaultGrouping = Grouping.Cards;
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
import { IApplication } from './../../../domain/IApplication';
|
import { IApplication } from './../../../domain/IApplication';
|
||||||
import { ICategory, IScript } from '@/domain/ICategory';
|
import { ICategory, IScript } from '@/domain/ICategory';
|
||||||
import { INode } from './SelectableTree/INode';
|
import { INode, NodeType } from './SelectableTree/Node/INode';
|
||||||
|
|
||||||
export function parseAllCategories(app: IApplication): INode[] | undefined {
|
export function parseAllCategories(app: IApplication): INode[] | undefined {
|
||||||
const nodes = new Array<INode>();
|
const nodes = new Array<INode>();
|
||||||
@@ -23,9 +23,15 @@ export function parseSingleCategory(categoryId: number, app: IApplication): INod
|
|||||||
export function getScriptNodeId(script: IScript): string {
|
export function getScriptNodeId(script: IScript): string {
|
||||||
return script.id;
|
return script.id;
|
||||||
}
|
}
|
||||||
|
export function getScriptId(nodeId: string): string {
|
||||||
|
return nodeId;
|
||||||
|
}
|
||||||
|
export function getCategoryId(nodeId: string): number {
|
||||||
|
return +nodeId;
|
||||||
|
}
|
||||||
|
|
||||||
export function getCategoryNodeId(category: ICategory): string {
|
export function getCategoryNodeId(category: ICategory): string {
|
||||||
return `Category${category.id}`;
|
return `${category.id}`;
|
||||||
}
|
}
|
||||||
|
|
||||||
function parseCategoryRecursively(
|
function parseCategoryRecursively(
|
||||||
@@ -64,16 +70,18 @@ function convertCategoryToNode(
|
|||||||
category: ICategory, children: readonly INode[]): INode {
|
category: ICategory, children: readonly INode[]): INode {
|
||||||
return {
|
return {
|
||||||
id: getCategoryNodeId(category),
|
id: getCategoryNodeId(category),
|
||||||
|
type: NodeType.Category,
|
||||||
text: category.name,
|
text: category.name,
|
||||||
children,
|
children,
|
||||||
documentationUrls: category.documentationUrls,
|
documentationUrls: category.documentationUrls,
|
||||||
isReversible: false,
|
isReversible: children && children.every((child) => child.isReversible),
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
function convertScriptToNode(script: IScript): INode {
|
function convertScriptToNode(script: IScript): INode {
|
||||||
return {
|
return {
|
||||||
id: getScriptNodeId(script),
|
id: getScriptNodeId(script),
|
||||||
|
type: NodeType.Script,
|
||||||
text: script.name,
|
text: script.name,
|
||||||
children: undefined,
|
children: undefined,
|
||||||
documentationUrls: script.documentationUrls,
|
documentationUrls: script.documentationUrls,
|
||||||
|
|||||||
@@ -7,7 +7,6 @@
|
|||||||
:filterPredicate="filterPredicate"
|
:filterPredicate="filterPredicate"
|
||||||
:filterText="filterText"
|
:filterText="filterText"
|
||||||
v-on:nodeSelected="toggleNodeSelectionAsync($event)"
|
v-on:nodeSelected="toggleNodeSelectionAsync($event)"
|
||||||
v-on:nodeRevertToggled="handleNodeRevertToggleAsync($event)"
|
|
||||||
>
|
>
|
||||||
</SelectableTree>
|
</SelectableTree>
|
||||||
</span>
|
</span>
|
||||||
@@ -16,18 +15,17 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { Component, Prop, Vue, Watch } from 'vue-property-decorator';
|
import { Component, Prop, Watch } from 'vue-property-decorator';
|
||||||
import { StatefulVue } from '@/presentation/StatefulVue';
|
import { StatefulVue } from '@/presentation/StatefulVue';
|
||||||
import { Category } from '@/domain/Category';
|
|
||||||
import { IRepository } from '@/infrastructure/Repository/IRepository';
|
|
||||||
import { IScript } from '@/domain/IScript';
|
import { IScript } from '@/domain/IScript';
|
||||||
import { ICategory } from '@/domain/ICategory';
|
import { ICategory } from '@/domain/ICategory';
|
||||||
import { IApplicationState, IUserSelection } from '@/application/State/IApplicationState';
|
import { IApplicationState } from '@/application/State/IApplicationState';
|
||||||
import { IFilterResult } from '@/application/State/Filter/IFilterResult';
|
import { IFilterResult } from '@/application/State/Filter/IFilterResult';
|
||||||
import { parseAllCategories, parseSingleCategory, getScriptNodeId, getCategoryNodeId } from './ScriptNodeParser';
|
import { parseAllCategories, parseSingleCategory, getScriptNodeId, getCategoryNodeId, getCategoryId, getScriptId } from './ScriptNodeParser';
|
||||||
import SelectableTree, { FilterPredicate } from './SelectableTree/SelectableTree.vue';
|
import SelectableTree from './SelectableTree/SelectableTree.vue';
|
||||||
import { INode } from './SelectableTree/INode';
|
import { INode, NodeType } from './SelectableTree/Node/INode';
|
||||||
import { SelectedScript } from '@/application/State/Selection/SelectedScript';
|
import { SelectedScript } from '@/application/State/Selection/SelectedScript';
|
||||||
|
import { INodeSelectedEvent } from './SelectableTree/INodeSelectedEvent';
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
components: {
|
components: {
|
||||||
@@ -51,17 +49,20 @@
|
|||||||
state.filter.filtered.on(this.handleFiltered);
|
state.filter.filtered.on(this.handleFiltered);
|
||||||
// Update initial state
|
// Update initial state
|
||||||
await this.initializeNodesAsync(this.categoryId);
|
await this.initializeNodesAsync(this.categoryId);
|
||||||
|
await this.initializeFilter(state.filter.currentFilter);
|
||||||
}
|
}
|
||||||
|
|
||||||
public async toggleNodeSelectionAsync(node: INode) {
|
public async toggleNodeSelectionAsync(event: INodeSelectedEvent) {
|
||||||
if (node.children != null && node.children.length > 0) {
|
|
||||||
return; // only interested in script nodes
|
|
||||||
}
|
|
||||||
const state = await this.getCurrentStateAsync();
|
const state = await this.getCurrentStateAsync();
|
||||||
if (!this.selectedNodeIds.some((id) => id === node.id)) {
|
switch (event.node.type) {
|
||||||
state.selection.addSelectedScript(node.id, false);
|
case NodeType.Category:
|
||||||
} else {
|
toggleCategoryNodeSelection(event, state);
|
||||||
state.selection.removeSelectedScript(node.id);
|
break;
|
||||||
|
case NodeType.Script:
|
||||||
|
toggleScriptNodeSelection(event, state);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
throw new Error(`Unknown node type: ${event.node.id}`);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -84,6 +85,14 @@
|
|||||||
(category: ICategory) => node.id === getCategoryNodeId(category));
|
(category: ICategory) => node.id === getCategoryNodeId(category));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private initializeFilter(currentFilter: IFilterResult | undefined) {
|
||||||
|
if (!currentFilter) {
|
||||||
|
this.handleFilterRemoved();
|
||||||
|
} else {
|
||||||
|
this.handleFiltered(currentFilter);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
private handleSelectionChanged(selectedScripts: ReadonlyArray<SelectedScript>): void {
|
private handleSelectionChanged(selectedScripts: ReadonlyArray<SelectedScript>): void {
|
||||||
this.selectedNodeIds = selectedScripts
|
this.selectedNodeIds = selectedScripts
|
||||||
.map((node) => node.id);
|
.map((node) => node.id);
|
||||||
@@ -99,6 +108,24 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function toggleCategoryNodeSelection(event: INodeSelectedEvent, state: IApplicationState): void {
|
||||||
|
const categoryId = getCategoryId(event.node.id);
|
||||||
|
if (event.isSelected) {
|
||||||
|
state.selection.addOrUpdateAllInCategory(categoryId, false);
|
||||||
|
} else {
|
||||||
|
state.selection.removeAllInCategory(categoryId);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
function toggleScriptNodeSelection(event: INodeSelectedEvent, state: IApplicationState): void {
|
||||||
|
const scriptId = getScriptId(event.node.id);
|
||||||
|
const actualToggleState = state.selection.isSelected(scriptId);
|
||||||
|
const targetToggleState = event.isSelected;
|
||||||
|
if (targetToggleState && !actualToggleState) {
|
||||||
|
state.selection.addSelectedScript(scriptId, false);
|
||||||
|
} else if (!targetToggleState && actualToggleState) {
|
||||||
|
state.selection.removeSelectedScript(scriptId);
|
||||||
|
}
|
||||||
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style scoped lang="scss">
|
<style scoped lang="scss">
|
||||||
|
|||||||
@@ -0,0 +1,6 @@
|
|||||||
|
import { INode } from './Node/INode';
|
||||||
|
|
||||||
|
export interface INodeSelectedEvent {
|
||||||
|
isSelected: boolean;
|
||||||
|
node: INode;
|
||||||
|
}
|
||||||
@@ -1,4 +1,3 @@
|
|||||||
// Two ways of typing other libraries: https://stackoverflow.com/a/53070501
|
|
||||||
|
|
||||||
declare module 'liquor-tree' {
|
declare module 'liquor-tree' {
|
||||||
import { PluginObject } from 'vue';
|
import { PluginObject } from 'vue';
|
||||||
@@ -10,33 +9,44 @@ declare module 'liquor-tree' {
|
|||||||
filter(query: string): void;
|
filter(query: string): void;
|
||||||
clearFilter(): void;
|
clearFilter(): void;
|
||||||
setModel(nodes: ReadonlyArray<ILiquorTreeNewNode>): void;
|
setModel(nodes: ReadonlyArray<ILiquorTreeNewNode>): void;
|
||||||
|
// getNodeById(id: string): ILiquorTreeExistingNode;
|
||||||
|
recurseDown(fn: (node: ILiquorTreeExistingNode) => void): void;
|
||||||
}
|
}
|
||||||
interface ICustomLiquorTreeData {
|
export interface ICustomLiquorTreeData {
|
||||||
|
type: number;
|
||||||
documentationUrls: ReadonlyArray<string>;
|
documentationUrls: ReadonlyArray<string>;
|
||||||
isReversible: boolean;
|
isReversible: boolean;
|
||||||
}
|
}
|
||||||
|
// https://github.com/amsik/liquor-tree/blob/master/src/lib/Node.js
|
||||||
|
export interface ILiquorTreeNodeState {
|
||||||
|
checked: boolean;
|
||||||
|
indeterminate: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface ILiquorTreeNode {
|
||||||
|
id: string;
|
||||||
|
data: ICustomLiquorTreeData;
|
||||||
|
children: ReadonlyArray<ILiquorTreeNode> | undefined;
|
||||||
|
}
|
||||||
/**
|
/**
|
||||||
* Returned from Node tree view events.
|
* Returned from Node tree view events.
|
||||||
* See constructor in https://github.com/amsik/liquor-tree/blob/master/src/lib/Node.js
|
* See constructor in https://github.com/amsik/liquor-tree/blob/master/src/lib/Node.js
|
||||||
*/
|
*/
|
||||||
export interface ILiquorTreeExistingNode {
|
export interface ILiquorTreeExistingNode extends ILiquorTreeNode {
|
||||||
id: string;
|
|
||||||
data: ILiquorTreeNodeData;
|
data: ILiquorTreeNodeData;
|
||||||
states: ILiquorTreeNodeState | undefined;
|
states: ILiquorTreeNodeState | undefined;
|
||||||
children: ReadonlyArray<ILiquorTreeExistingNode> | undefined;
|
children: ReadonlyArray<ILiquorTreeExistingNode> | undefined;
|
||||||
|
// expand(): void;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Sent to liquor tree to define of new nodes.
|
* Sent to liquor tree to define of new nodes.
|
||||||
* https://github.com/amsik/liquor-tree/blob/master/src/lib/Node.js
|
* https://github.com/amsik/liquor-tree/blob/master/src/lib/Node.js
|
||||||
*/
|
*/
|
||||||
export interface ILiquorTreeNewNode {
|
export interface ILiquorTreeNewNode extends ILiquorTreeNode {
|
||||||
id: string;
|
|
||||||
text: string;
|
text: string;
|
||||||
state: ILiquorTreeNodeState | undefined;
|
state: ILiquorTreeNodeState | undefined;
|
||||||
children: ReadonlyArray<ILiquorTreeNewNode> | undefined;
|
children: ReadonlyArray<ILiquorTreeNewNode> | undefined;
|
||||||
data: ICustomLiquorTreeData;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// https://amsik.github.io/liquor-tree/#Component-Options
|
// https://amsik.github.io/liquor-tree/#Component-Options
|
||||||
@@ -47,29 +57,16 @@ declare module 'liquor-tree' {
|
|||||||
autoCheckChildren: boolean;
|
autoCheckChildren: boolean;
|
||||||
parentSelect: boolean;
|
parentSelect: boolean;
|
||||||
keyboardNavigation: boolean;
|
keyboardNavigation: boolean;
|
||||||
deletion: (node: ILiquorTreeExistingNode) => void;
|
|
||||||
filter: ILiquorTreeFilter;
|
filter: ILiquorTreeFilter;
|
||||||
|
deletion(node: ILiquorTreeNode): boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
// https://github.com/amsik/liquor-tree/blob/master/src/lib/Node.js
|
export interface ILiquorTreeNodeData extends ICustomLiquorTreeData {
|
||||||
interface ILiquorTreeNodeState {
|
|
||||||
checked: boolean;
|
|
||||||
}
|
|
||||||
|
|
||||||
interface ILiquorTreeNodeData extends ICustomLiquorTreeData {
|
|
||||||
text: string;
|
text: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
// https://github.com/amsik/liquor-tree/blob/master/src/components/TreeRoot.vue
|
// https://github.com/amsik/liquor-tree/blob/master/src/components/TreeRoot.vue
|
||||||
interface ILiquorTreeOptions {
|
export interface ILiquorTreeFilter {
|
||||||
checkbox: boolean;
|
|
||||||
checkOnSelect: boolean;
|
|
||||||
filter: ILiquorTreeFilter;
|
|
||||||
deletion(node: ILiquorTreeNewNode): boolean;
|
|
||||||
}
|
|
||||||
|
|
||||||
// https://github.com/amsik/liquor-tree/blob/master/src/components/TreeRoot.vue
|
|
||||||
interface ILiquorTreeFilter {
|
|
||||||
emptyText: string;
|
emptyText: string;
|
||||||
matcher(query: string, node: ILiquorTreeExistingNode): boolean;
|
matcher(query: string, node: ILiquorTreeExistingNode): boolean;
|
||||||
}
|
}
|
||||||
@@ -0,0 +1,23 @@
|
|||||||
|
import { ILiquorTreeOptions, ILiquorTreeFilter, ILiquorTreeNode, ILiquorTreeExistingNode } from 'liquor-tree';
|
||||||
|
|
||||||
|
export class LiquorTreeOptions implements ILiquorTreeOptions {
|
||||||
|
public readonly multiple = true;
|
||||||
|
public readonly checkbox = true;
|
||||||
|
public readonly checkOnSelect = true;
|
||||||
|
/* For checkbox mode only. Children will have the same checked state as their parent.
|
||||||
|
⚠️ Setting this false, does not update indeterminate state of nodes.
|
||||||
|
This is false as it's handled manually to be able to batch select for performance + highlighting */
|
||||||
|
public readonly autoCheckChildren = false;
|
||||||
|
public readonly parentSelect = true;
|
||||||
|
public readonly keyboardNavigation = true;
|
||||||
|
public readonly filter = { // Wrap this in an arrow function as setting filter directly does not work JS APIs
|
||||||
|
emptyText: this.liquorTreeFilter.emptyText,
|
||||||
|
matcher: (query: string, node: ILiquorTreeExistingNode) => {
|
||||||
|
return this.liquorTreeFilter.matcher(query, node);
|
||||||
|
},
|
||||||
|
};
|
||||||
|
constructor(private readonly liquorTreeFilter: ILiquorTreeFilter) { }
|
||||||
|
public deletion(node: ILiquorTreeNode): boolean {
|
||||||
|
return false; // no op
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,17 @@
|
|||||||
|
import { ILiquorTreeFilter, ILiquorTreeExistingNode } from 'liquor-tree';
|
||||||
|
import { convertExistingToNode } from './NodeTranslator';
|
||||||
|
import { INode } from './../../Node/INode';
|
||||||
|
|
||||||
|
export type FilterPredicate = (node: INode) => boolean;
|
||||||
|
|
||||||
|
export class NodePredicateFilter implements ILiquorTreeFilter {
|
||||||
|
public emptyText = ''; // Does not matter as a custom mesage is shown
|
||||||
|
constructor(private readonly filterPredicate: FilterPredicate) {
|
||||||
|
if (!filterPredicate) {
|
||||||
|
throw new Error('filterPredicate is undefined');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
public matcher(query: string, node: ILiquorTreeExistingNode): boolean {
|
||||||
|
return this.filterPredicate(convertExistingToNode(node));
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,61 @@
|
|||||||
|
import { ILiquorTreeNode, ILiquorTreeNodeState } from 'liquor-tree';
|
||||||
|
import { NodeType } from './../../Node/INode';
|
||||||
|
|
||||||
|
export function getNewState(
|
||||||
|
node: ILiquorTreeNode,
|
||||||
|
selectedNodeIds: ReadonlyArray<string>): ILiquorTreeNodeState {
|
||||||
|
const checked = getNewCheckedState(node, selectedNodeIds);
|
||||||
|
const indeterminate = !checked && getNewIndeterminateState(node, selectedNodeIds);
|
||||||
|
return {
|
||||||
|
checked, indeterminate,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
function getNewIndeterminateState(
|
||||||
|
node: ILiquorTreeNode,
|
||||||
|
selectedNodeIds: ReadonlyArray<string>): boolean {
|
||||||
|
switch (node.data.type) {
|
||||||
|
case NodeType.Script:
|
||||||
|
return false;
|
||||||
|
case NodeType.Category:
|
||||||
|
return parseAllSubScriptIds(node).some((id) => selectedNodeIds.includes(id));
|
||||||
|
default:
|
||||||
|
throw new Error('Unknown node type');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function getNewCheckedState(
|
||||||
|
node: ILiquorTreeNode,
|
||||||
|
selectedNodeIds: ReadonlyArray<string>): boolean {
|
||||||
|
switch (node.data.type) {
|
||||||
|
case NodeType.Script:
|
||||||
|
return selectedNodeIds.some((id) => id === node.id);
|
||||||
|
case NodeType.Category:
|
||||||
|
return parseAllSubScriptIds(node).every((id) => selectedNodeIds.includes(id));
|
||||||
|
default:
|
||||||
|
throw new Error('Unknown node type');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function parseAllSubScriptIds(categoryNode: ILiquorTreeNode): ReadonlyArray<string> {
|
||||||
|
if (categoryNode.data.type !== NodeType.Category) {
|
||||||
|
throw new Error('Not a category node');
|
||||||
|
}
|
||||||
|
if (!categoryNode.children) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
return categoryNode
|
||||||
|
.children
|
||||||
|
.flatMap((child) => getNodeIds(child));
|
||||||
|
}
|
||||||
|
|
||||||
|
function getNodeIds(node: ILiquorTreeNode): ReadonlyArray<string> {
|
||||||
|
switch (node.data.type) {
|
||||||
|
case NodeType.Script:
|
||||||
|
return [ node.id ];
|
||||||
|
case NodeType.Category:
|
||||||
|
return parseAllSubScriptIds(node);
|
||||||
|
default:
|
||||||
|
throw new Error('Unknown node type');
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
import { ILiquorTreeNewNode, ILiquorTreeExistingNode } from 'liquor-tree';
|
import { ILiquorTreeNewNode, ILiquorTreeExistingNode } from 'liquor-tree';
|
||||||
import { INode } from './INode';
|
import { INode } from './../../Node/INode';
|
||||||
|
|
||||||
// Functions to translate INode to LiqourTree models and vice versa for anti-corruption
|
// Functions to translate INode to LiqourTree models and vice versa for anti-corruption
|
||||||
|
|
||||||
@@ -7,6 +7,7 @@ export function convertExistingToNode(liquorTreeNode: ILiquorTreeExistingNode):
|
|||||||
if (!liquorTreeNode) { throw new Error('liquorTreeNode is undefined'); }
|
if (!liquorTreeNode) { throw new Error('liquorTreeNode is undefined'); }
|
||||||
return {
|
return {
|
||||||
id: liquorTreeNode.id,
|
id: liquorTreeNode.id,
|
||||||
|
type: liquorTreeNode.data.type,
|
||||||
text: liquorTreeNode.data.text,
|
text: liquorTreeNode.data.text,
|
||||||
// selected: liquorTreeNode.states && liquorTreeNode.states.checked,
|
// selected: liquorTreeNode.states && liquorTreeNode.states.checked,
|
||||||
children: convertChildren(liquorTreeNode.children, convertExistingToNode),
|
children: convertChildren(liquorTreeNode.children, convertExistingToNode),
|
||||||
@@ -22,11 +23,13 @@ export function toNewLiquorTreeNode(node: INode): ILiquorTreeNewNode {
|
|||||||
text: node.text,
|
text: node.text,
|
||||||
state: {
|
state: {
|
||||||
checked: false,
|
checked: false,
|
||||||
|
indeterminate: false,
|
||||||
},
|
},
|
||||||
children: convertChildren(node.children, toNewLiquorTreeNode),
|
children: convertChildren(node.children, toNewLiquorTreeNode),
|
||||||
data: {
|
data: {
|
||||||
documentationUrls: node.documentationUrls,
|
documentationUrls: node.documentationUrls,
|
||||||
isReversible: node.isReversible,
|
isReversible: node.isReversible,
|
||||||
|
type: node.type,
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
@@ -1,7 +1,13 @@
|
|||||||
|
export enum NodeType {
|
||||||
|
Script,
|
||||||
|
Category,
|
||||||
|
}
|
||||||
|
|
||||||
export interface INode {
|
export interface INode {
|
||||||
readonly id: string;
|
readonly id: string;
|
||||||
readonly text: string;
|
readonly text: string;
|
||||||
readonly isReversible: boolean;
|
readonly isReversible: boolean;
|
||||||
readonly documentationUrls: ReadonlyArray<string>;
|
readonly documentationUrls: ReadonlyArray<string>;
|
||||||
readonly children?: ReadonlyArray<INode>;
|
readonly children?: ReadonlyArray<INode>;
|
||||||
|
readonly type: NodeType;
|
||||||
}
|
}
|
||||||
@@ -4,7 +4,7 @@
|
|||||||
<RevertToggle
|
<RevertToggle
|
||||||
class="item"
|
class="item"
|
||||||
v-if="data.isReversible"
|
v-if="data.isReversible"
|
||||||
:scriptId="data.id" />
|
:node="data" />
|
||||||
<DocumentationUrls
|
<DocumentationUrls
|
||||||
class="item"
|
class="item"
|
||||||
v-if="data.documentationUrls && data.documentationUrls.length > 0"
|
v-if="data.documentationUrls && data.documentationUrls.length > 0"
|
||||||
@@ -0,0 +1,151 @@
|
|||||||
|
<template>
|
||||||
|
<div class="checkbox-switch" >
|
||||||
|
<input type="checkbox" class="input-checkbox"
|
||||||
|
v-model="isReverted"
|
||||||
|
@change="onRevertToggledAsync()"
|
||||||
|
v-on:click.stop>
|
||||||
|
<div class="checkbox-animate">
|
||||||
|
<span class="checkbox-off">revert</span>
|
||||||
|
<span class="checkbox-on">revert</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
import { Component, Prop, Watch } from 'vue-property-decorator';
|
||||||
|
import { IReverter } from './Reverter/IReverter';
|
||||||
|
import { StatefulVue } from '@/presentation/StatefulVue';
|
||||||
|
import { INode } from './INode';
|
||||||
|
import { SelectedScript } from '@/application/State/Selection/SelectedScript';
|
||||||
|
import { getReverter } from './Reverter/ReverterFactory';
|
||||||
|
|
||||||
|
@Component
|
||||||
|
export default class RevertToggle extends StatefulVue {
|
||||||
|
@Prop() public node: INode;
|
||||||
|
public isReverted = false;
|
||||||
|
|
||||||
|
private handler: IReverter;
|
||||||
|
|
||||||
|
public async mounted() {
|
||||||
|
await this.onNodeChangedAsync(this.node);
|
||||||
|
const state = await this.getCurrentStateAsync();
|
||||||
|
this.updateState(state.selection.selectedScripts);
|
||||||
|
state.selection.changed.on((scripts) => this.updateState(scripts));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Watch('node') public async onNodeChangedAsync(node: INode) {
|
||||||
|
const state = await this.getCurrentStateAsync();
|
||||||
|
this.handler = getReverter(node, state.app);
|
||||||
|
}
|
||||||
|
|
||||||
|
public async onRevertToggledAsync() {
|
||||||
|
const state = await this.getCurrentStateAsync();
|
||||||
|
this.handler.selectWithRevertState(this.isReverted, state.selection);
|
||||||
|
}
|
||||||
|
|
||||||
|
private updateState(scripts: ReadonlyArray<SelectedScript>) {
|
||||||
|
this.isReverted = this.handler.getState(scripts);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
|
||||||
|
<style scoped lang="scss">
|
||||||
|
@import "@/presentation/styles/colors.scss";
|
||||||
|
$width: 85px;
|
||||||
|
$height: 30px;
|
||||||
|
// https://www.designlabthemes.com/css-toggle-switch/
|
||||||
|
.checkbox-switch {
|
||||||
|
cursor: pointer;
|
||||||
|
display: inline-block;
|
||||||
|
overflow: hidden;
|
||||||
|
position: relative;
|
||||||
|
width: $width;
|
||||||
|
height: $height;
|
||||||
|
-webkit-border-radius: $height;
|
||||||
|
border-radius: $height;
|
||||||
|
line-height: $height;
|
||||||
|
font-size: $height / 2;
|
||||||
|
display: inline-block;
|
||||||
|
|
||||||
|
input.input-checkbox {
|
||||||
|
position: absolute;
|
||||||
|
left: 0;
|
||||||
|
top: 0;
|
||||||
|
width: $width;
|
||||||
|
height: $height;
|
||||||
|
padding: 0;
|
||||||
|
margin: 0;
|
||||||
|
opacity: 0;
|
||||||
|
z-index: 2;
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
|
||||||
|
.checkbox-animate {
|
||||||
|
position: relative;
|
||||||
|
width: $width;
|
||||||
|
height: $height;
|
||||||
|
background-color: $gray;
|
||||||
|
-webkit-transition: background-color 0.25s ease-out 0s;
|
||||||
|
transition: background-color 0.25s ease-out 0s;
|
||||||
|
|
||||||
|
// Circle
|
||||||
|
&:before {
|
||||||
|
$circle-size: $height * 0.66;
|
||||||
|
|
||||||
|
content: "";
|
||||||
|
display: block;
|
||||||
|
position: absolute;
|
||||||
|
width: $circle-size;
|
||||||
|
height: $circle-size;
|
||||||
|
border-radius: $circle-size * 2;
|
||||||
|
-webkit-border-radius: $circle-size * 2;
|
||||||
|
background-color: $slate;
|
||||||
|
top: $height * 0.16;
|
||||||
|
left: $width * 0.05;
|
||||||
|
-webkit-transition: left 0.3s ease-out 0s;
|
||||||
|
transition: left 0.3s ease-out 0s;
|
||||||
|
z-index: 10;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
input.input-checkbox:checked {
|
||||||
|
+ .checkbox-animate {
|
||||||
|
background-color: $accent;
|
||||||
|
}
|
||||||
|
+ .checkbox-animate:before {
|
||||||
|
left: ($width - $width/3.5);
|
||||||
|
background-color: $light-gray;
|
||||||
|
}
|
||||||
|
+ .checkbox-animate .checkbox-off {
|
||||||
|
display: none;
|
||||||
|
opacity: 0;
|
||||||
|
}
|
||||||
|
+ .checkbox-animate .checkbox-on {
|
||||||
|
display: block;
|
||||||
|
opacity: 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.checkbox-off, .checkbox-on {
|
||||||
|
float: left;
|
||||||
|
color: $white;
|
||||||
|
font-weight: 700;
|
||||||
|
-webkit-transition: all 0.3s ease-out 0s;
|
||||||
|
transition: all 0.3s ease-out 0s;
|
||||||
|
}
|
||||||
|
|
||||||
|
.checkbox-off {
|
||||||
|
margin-left: $width / 3;
|
||||||
|
opacity: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.checkbox-on {
|
||||||
|
display: none;
|
||||||
|
float: right;
|
||||||
|
margin-right: $width / 3;
|
||||||
|
opacity: 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
||||||
@@ -0,0 +1,30 @@
|
|||||||
|
import { IReverter } from './IReverter';
|
||||||
|
import { getCategoryId } from '../../../ScriptNodeParser';
|
||||||
|
import { SelectedScript } from '@/application/State/Selection/SelectedScript';
|
||||||
|
import { IApplication } from '@/domain/IApplication';
|
||||||
|
import { ScriptReverter } from './ScriptReverter';
|
||||||
|
import { IUserSelection } from '@/application/State/Selection/IUserSelection';
|
||||||
|
|
||||||
|
export class CategoryReverter implements IReverter {
|
||||||
|
private readonly categoryId: number;
|
||||||
|
private readonly scriptReverters: ReadonlyArray<ScriptReverter>;
|
||||||
|
constructor(nodeId: string, app: IApplication) {
|
||||||
|
this.categoryId = getCategoryId(nodeId);
|
||||||
|
this.scriptReverters = getAllSubScriptReverters(this.categoryId, app);
|
||||||
|
}
|
||||||
|
public getState(selectedScripts: ReadonlyArray<SelectedScript>): boolean {
|
||||||
|
return this.scriptReverters.every((script) => script.getState(selectedScripts));
|
||||||
|
}
|
||||||
|
public selectWithRevertState(newState: boolean, selection: IUserSelection): void {
|
||||||
|
selection.addOrUpdateAllInCategory(this.categoryId, newState);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function getAllSubScriptReverters(categoryId: number, app: IApplication) {
|
||||||
|
const category = app.findCategory(categoryId);
|
||||||
|
if (!category) {
|
||||||
|
throw new Error(`Category with id "${categoryId}" does not exist`);
|
||||||
|
}
|
||||||
|
const scripts = category.getAllScriptsRecursively();
|
||||||
|
return scripts.map((script) => new ScriptReverter(script.id));
|
||||||
|
}
|
||||||
@@ -0,0 +1,7 @@
|
|||||||
|
import { SelectedScript } from '@/application/State/Selection/SelectedScript';
|
||||||
|
import { IUserSelection } from '@/application/State/IApplicationState';
|
||||||
|
|
||||||
|
export interface IReverter {
|
||||||
|
getState(selectedScripts: ReadonlyArray<SelectedScript>): boolean;
|
||||||
|
selectWithRevertState(newState: boolean, selection: IUserSelection): void;
|
||||||
|
}
|
||||||
@@ -0,0 +1,16 @@
|
|||||||
|
import { INode, NodeType } from '../INode';
|
||||||
|
import { IReverter } from './IReverter';
|
||||||
|
import { ScriptReverter } from './ScriptReverter';
|
||||||
|
import { IApplication } from '@/domain/IApplication';
|
||||||
|
import { CategoryReverter } from './CategoryReverter';
|
||||||
|
|
||||||
|
export function getReverter(node: INode, app: IApplication): IReverter {
|
||||||
|
switch (node.type) {
|
||||||
|
case NodeType.Category:
|
||||||
|
return new CategoryReverter(node.id, app);
|
||||||
|
case NodeType.Script:
|
||||||
|
return new ScriptReverter(node.id);
|
||||||
|
default:
|
||||||
|
throw new Error('Unknown script type');
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,21 @@
|
|||||||
|
import { IReverter } from './IReverter';
|
||||||
|
import { getScriptId } from '../../../ScriptNodeParser';
|
||||||
|
import { SelectedScript } from '@/application/State/Selection/SelectedScript';
|
||||||
|
import { IUserSelection } from '@/application/State/IApplicationState';
|
||||||
|
|
||||||
|
export class ScriptReverter implements IReverter {
|
||||||
|
private readonly scriptId: string;
|
||||||
|
constructor(nodeId: string) {
|
||||||
|
this.scriptId = getScriptId(nodeId);
|
||||||
|
}
|
||||||
|
public getState(selectedScripts: ReadonlyArray<SelectedScript>): boolean {
|
||||||
|
const selectedScript = selectedScripts.find((selected) => selected.id === this.scriptId);
|
||||||
|
if (!selectedScript) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return selectedScript.revert;
|
||||||
|
}
|
||||||
|
public selectWithRevertState(newState: boolean, selection: IUserSelection): void {
|
||||||
|
selection.addOrUpdateSelectedScript(this.scriptId, newState);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,141 +0,0 @@
|
|||||||
<template>
|
|
||||||
<div class="checkbox-switch" >
|
|
||||||
<input type="checkbox" class="input-checkbox"
|
|
||||||
v-model="isReverted"
|
|
||||||
@change="onRevertToggledAsync()" >
|
|
||||||
<div class="checkbox-animate">
|
|
||||||
<span class="checkbox-off">revert</span>
|
|
||||||
<span class="checkbox-on">revert</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</template>
|
|
||||||
|
|
||||||
|
|
||||||
<script lang="ts">
|
|
||||||
import { Component, Prop, Vue } from 'vue-property-decorator';
|
|
||||||
import { StatefulVue } from '@/presentation/StatefulVue';
|
|
||||||
import { INode } from './INode';
|
|
||||||
import { SelectedScript } from '@/application/State/Selection/SelectedScript';
|
|
||||||
|
|
||||||
@Component
|
|
||||||
export default class RevertToggle extends StatefulVue {
|
|
||||||
@Prop() public scriptId: string;
|
|
||||||
public isReverted = false;
|
|
||||||
public async mounted() {
|
|
||||||
const state = await this.getCurrentStateAsync();
|
|
||||||
state.selection.changed.on(this.handleSelectionChanged);
|
|
||||||
}
|
|
||||||
public async onRevertToggledAsync() {
|
|
||||||
const state = await this.getCurrentStateAsync();
|
|
||||||
state.selection.addOrUpdateSelectedScript(this.scriptId, this.isReverted);
|
|
||||||
}
|
|
||||||
private handleSelectionChanged(selectedScripts: ReadonlyArray<SelectedScript>): void {
|
|
||||||
const selectedScript = selectedScripts.find((script) => script.id === this.scriptId);
|
|
||||||
if (!selectedScript) {
|
|
||||||
this.isReverted = false;
|
|
||||||
} else {
|
|
||||||
this.isReverted = selectedScript.revert;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
</script>
|
|
||||||
|
|
||||||
|
|
||||||
<style scoped lang="scss">
|
|
||||||
@import "@/presentation/styles/colors.scss";
|
|
||||||
$width: 85px;
|
|
||||||
$height: 30px;
|
|
||||||
// https://www.designlabthemes.com/css-toggle-switch/
|
|
||||||
.checkbox-switch {
|
|
||||||
cursor: pointer;
|
|
||||||
display: inline-block;
|
|
||||||
overflow: hidden;
|
|
||||||
position: relative;
|
|
||||||
width: $width;
|
|
||||||
height: $height;
|
|
||||||
-webkit-border-radius: $height;
|
|
||||||
border-radius: $height;
|
|
||||||
line-height: $height;
|
|
||||||
font-size: $height / 2;
|
|
||||||
display: inline-block;
|
|
||||||
|
|
||||||
input.input-checkbox {
|
|
||||||
position: absolute;
|
|
||||||
left: 0;
|
|
||||||
top: 0;
|
|
||||||
width: $width;
|
|
||||||
height: $height;
|
|
||||||
padding: 0;
|
|
||||||
margin: 0;
|
|
||||||
opacity: 0;
|
|
||||||
z-index: 2;
|
|
||||||
cursor: pointer;
|
|
||||||
}
|
|
||||||
|
|
||||||
.checkbox-animate {
|
|
||||||
position: relative;
|
|
||||||
width: $width;
|
|
||||||
height: $height;
|
|
||||||
background-color: $gray;
|
|
||||||
-webkit-transition: background-color 0.25s ease-out 0s;
|
|
||||||
transition: background-color 0.25s ease-out 0s;
|
|
||||||
|
|
||||||
// Circle
|
|
||||||
&:before {
|
|
||||||
$circle-size: $height * 0.66;
|
|
||||||
|
|
||||||
content: "";
|
|
||||||
display: block;
|
|
||||||
position: absolute;
|
|
||||||
width: $circle-size;
|
|
||||||
height: $circle-size;
|
|
||||||
border-radius: $circle-size * 2;
|
|
||||||
-webkit-border-radius: $circle-size * 2;
|
|
||||||
background-color: $slate;
|
|
||||||
top: $height * 0.16;
|
|
||||||
left: $width * 0.05;
|
|
||||||
-webkit-transition: left 0.3s ease-out 0s;
|
|
||||||
transition: left 0.3s ease-out 0s;
|
|
||||||
z-index: 10;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
input.input-checkbox:checked {
|
|
||||||
+ .checkbox-animate {
|
|
||||||
background-color: $accent;
|
|
||||||
}
|
|
||||||
+ .checkbox-animate:before {
|
|
||||||
left: ($width - $width/3.5);
|
|
||||||
background-color: $light-gray;
|
|
||||||
}
|
|
||||||
+ .checkbox-animate .checkbox-off {
|
|
||||||
display: none;
|
|
||||||
opacity: 0;
|
|
||||||
}
|
|
||||||
+ .checkbox-animate .checkbox-on {
|
|
||||||
display: block;
|
|
||||||
opacity: 1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
.checkbox-off, .checkbox-on {
|
|
||||||
float: left;
|
|
||||||
color: $white;
|
|
||||||
font-weight: 700;
|
|
||||||
-webkit-transition: all 0.3s ease-out 0s;
|
|
||||||
transition: all 0.3s ease-out 0s;
|
|
||||||
}
|
|
||||||
|
|
||||||
.checkbox-off {
|
|
||||||
margin-left: $width / 3;
|
|
||||||
opacity: 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
.checkbox-on {
|
|
||||||
display: none;
|
|
||||||
float: right;
|
|
||||||
margin-right: $width / 3;
|
|
||||||
opacity: 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
</style>
|
|
||||||
@@ -7,7 +7,7 @@
|
|||||||
v-on:node:unchecked="nodeSelected($event)"
|
v-on:node:unchecked="nodeSelected($event)"
|
||||||
ref="treeElement"
|
ref="treeElement"
|
||||||
>
|
>
|
||||||
<span class="tree-text" slot-scope="{ node }">
|
<span class="tree-text" slot-scope="{ node }" >
|
||||||
<Node :data="convertExistingToNode(node)" />
|
<Node :data="convertExistingToNode(node)" />
|
||||||
</span>
|
</span>
|
||||||
</tree>
|
</tree>
|
||||||
@@ -17,28 +17,32 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { Component, Prop, Vue, Emit, Watch } from 'vue-property-decorator';
|
import { Component, Prop, Vue, Watch } from 'vue-property-decorator';
|
||||||
import LiquorTree, { ILiquorTreeNewNode, ILiquorTreeExistingNode, ILiquorTree, ILiquorTreeOptions } from 'liquor-tree';
|
import LiquorTree from 'liquor-tree';
|
||||||
import Node from './Node.vue';
|
import Node from './Node/Node.vue';
|
||||||
import { INode } from './INode';
|
import { INode } from './Node/INode';
|
||||||
import { convertExistingToNode, toNewLiquorTreeNode } from './NodeTranslator';
|
import { convertExistingToNode, toNewLiquorTreeNode } from './LiquorTree/NodeWrapper/NodeTranslator';
|
||||||
export type FilterPredicate = (node: INode) => boolean;
|
import { INodeSelectedEvent } from './/INodeSelectedEvent';
|
||||||
|
import { getNewState } from './LiquorTree/NodeWrapper/NodeStateUpdater';
|
||||||
|
import { LiquorTreeOptions } from './LiquorTree/LiquorTreeOptions';
|
||||||
|
import { FilterPredicate, NodePredicateFilter } from './LiquorTree/NodeWrapper/NodePredicateFilter';
|
||||||
|
import { ILiquorTreeNewNode, ILiquorTreeExistingNode, ILiquorTree, ILiquorTreeNode, ILiquorTreeNodeState } from 'liquor-tree';
|
||||||
|
|
||||||
/** Wrapper for Liquor Tree, reveals only abstracted INode for communication */
|
/** Wrapper for Liquor Tree, reveals only abstracted INode for communication */
|
||||||
@Component({
|
@Component({
|
||||||
components: {
|
components: {
|
||||||
LiquorTree,
|
LiquorTree,
|
||||||
Node,
|
Node,
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
export default class SelectableTree extends Vue {
|
export default class SelectableTree extends Vue { // Keep it stateless to make it easier to switch out
|
||||||
@Prop() public filterPredicate?: FilterPredicate;
|
@Prop() public filterPredicate?: FilterPredicate;
|
||||||
@Prop() public filterText?: string;
|
@Prop() public filterText?: string;
|
||||||
@Prop() public selectedNodeIds?: ReadonlyArray<string>;
|
@Prop() public selectedNodeIds?: ReadonlyArray<string>;
|
||||||
@Prop() public initialNodes?: ReadonlyArray<INode>;
|
@Prop() public initialNodes?: ReadonlyArray<INode>;
|
||||||
|
|
||||||
public initialLiquourTreeNodes?: ILiquorTreeNewNode[] = null;
|
public initialLiquourTreeNodes?: ILiquorTreeNewNode[] = null;
|
||||||
public liquorTreeOptions = this.getDefaults();
|
public liquorTreeOptions = new LiquorTreeOptions(new NodePredicateFilter((node) => this.filterPredicate(node)));
|
||||||
public convertExistingToNode = convertExistingToNode;
|
public convertExistingToNode = convertExistingToNode;
|
||||||
|
|
||||||
public mounted() {
|
public mounted() {
|
||||||
@@ -46,7 +50,7 @@
|
|||||||
const initialNodes = this.initialNodes.map((node) => toNewLiquorTreeNode(node));
|
const initialNodes = this.initialNodes.map((node) => toNewLiquorTreeNode(node));
|
||||||
if (this.selectedNodeIds) {
|
if (this.selectedNodeIds) {
|
||||||
recurseDown(initialNodes,
|
recurseDown(initialNodes,
|
||||||
(node) => node.state.checked = this.selectedNodeIds.includes(node.id));
|
(node) => node.state = updateState(node.state, node, this.selectedNodeIds));
|
||||||
}
|
}
|
||||||
this.initialLiquourTreeNodes = initialNodes;
|
this.initialLiquourTreeNodes = initialNodes;
|
||||||
} else {
|
} else {
|
||||||
@@ -58,7 +62,11 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
public nodeSelected(node: ILiquorTreeExistingNode) {
|
public nodeSelected(node: ILiquorTreeExistingNode) {
|
||||||
this.$emit('nodeSelected', convertExistingToNode(node));
|
const event: INodeSelectedEvent = {
|
||||||
|
node: convertExistingToNode(node),
|
||||||
|
isSelected: node.states.checked,
|
||||||
|
};
|
||||||
|
this.$emit('nodeSelected', event);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -73,23 +81,13 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Watch('selectedNodeIds')
|
@Watch('selectedNodeIds')
|
||||||
public setSelectedStatus(selectedNodeIds: ReadonlyArray<string>) {
|
public setSelectedStatusAsync(selectedNodeIds: ReadonlyArray<string>) {
|
||||||
if (!selectedNodeIds) {
|
if (!selectedNodeIds) {
|
||||||
throw new Error('Selected nodes are undefined');
|
throw new Error('SelectedrecurseDown nodes are undefined');
|
||||||
}
|
}
|
||||||
const newNodes = updateCheckedState(this.getLiquorTreeApi().model, selectedNodeIds);
|
this.getLiquorTreeApi().recurseDown(
|
||||||
this.getLiquorTreeApi().setModel(newNodes);
|
(node) => node.states = updateState(node.states, node, selectedNodeIds),
|
||||||
/* Alternative:
|
);
|
||||||
this.getLiquorTreeApi().recurseDown((node) => {
|
|
||||||
node.states.checked = selectedNodeIds.includes(node.id);
|
|
||||||
});
|
|
||||||
Problem: Does not check their parent if all children are checked, because it does not
|
|
||||||
trigger update on parent as we work with scripts not categories. */
|
|
||||||
/* Alternative:
|
|
||||||
this.getLiquorTreeApi().recurseDown((node) => {
|
|
||||||
if(selectedNodeIds.includes(node.id)) { node.select(); } else { node.unselect(); }
|
|
||||||
});
|
|
||||||
Problem: Emits nodeSelected() event again which will cause an infinite loop. */
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private getLiquorTreeApi(): ILiquorTree {
|
private getLiquorTreeApi(): ILiquorTree {
|
||||||
@@ -98,27 +96,13 @@
|
|||||||
}
|
}
|
||||||
return (this.$refs.treeElement as any).tree;
|
return (this.$refs.treeElement as any).tree;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
private getDefaults(): ILiquorTreeOptions {
|
function updateState(
|
||||||
return {
|
old: ILiquorTreeNodeState,
|
||||||
multiple: true,
|
node: ILiquorTreeNode,
|
||||||
checkbox: true,
|
selectedNodeIds: ReadonlyArray<string>): ILiquorTreeNodeState {
|
||||||
checkOnSelect: true,
|
return {...old, ...getNewState(node, selectedNodeIds)};
|
||||||
autoCheckChildren: true,
|
|
||||||
parentSelect: false,
|
|
||||||
keyboardNavigation: true,
|
|
||||||
deletion: (node) => !node.children || node.children.length === 0,
|
|
||||||
filter: {
|
|
||||||
matcher: (query: string, node: ILiquorTreeExistingNode) => {
|
|
||||||
if (!this.filterPredicate) {
|
|
||||||
throw new Error('Cannot filter as predicate is null');
|
|
||||||
}
|
|
||||||
return this.filterPredicate(convertExistingToNode(node));
|
|
||||||
},
|
|
||||||
emptyText: '🕵️Hmm.. Can not see one 🧐',
|
|
||||||
},
|
|
||||||
};
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function recurseDown(
|
function recurseDown(
|
||||||
@@ -131,27 +115,4 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function updateCheckedState(
|
|
||||||
oldNodes: ReadonlyArray<ILiquorTreeExistingNode>,
|
|
||||||
selectedNodeIds: ReadonlyArray<string>): ReadonlyArray<ILiquorTreeNewNode> {
|
|
||||||
const result = new Array<ILiquorTreeNewNode>();
|
|
||||||
for (const oldNode of oldNodes) {
|
|
||||||
const newState = oldNode.states;
|
|
||||||
newState.checked = selectedNodeIds.some((id) => id === oldNode.id);
|
|
||||||
const newNode: ILiquorTreeNewNode = {
|
|
||||||
id: oldNode.id,
|
|
||||||
text: oldNode.data.text,
|
|
||||||
data: {
|
|
||||||
documentationUrls: oldNode.data.documentationUrls,
|
|
||||||
isReversible: oldNode.data.isReversible,
|
|
||||||
},
|
|
||||||
children: oldNode.children == null ? [] :
|
|
||||||
updateCheckedState(oldNode.children, selectedNodeIds),
|
|
||||||
state: newState,
|
|
||||||
};
|
|
||||||
result.push(newNode);
|
|
||||||
}
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
@@ -6,7 +6,7 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { Component, Prop, Vue, Emit } from 'vue-property-decorator';
|
import { Component, Prop, Emit } from 'vue-property-decorator';
|
||||||
import { StatefulVue } from '@/presentation/StatefulVue';
|
import { StatefulVue } from '@/presentation/StatefulVue';
|
||||||
import { NonCollapsing } from '@/presentation/Scripts/Cards/NonCollapsingDirective';
|
import { NonCollapsing } from '@/presentation/Scripts/Cards/NonCollapsingDirective';
|
||||||
|
|
||||||
|
|||||||
@@ -5,45 +5,66 @@
|
|||||||
<div class="part">
|
<div class="part">
|
||||||
<SelectableOption
|
<SelectableOption
|
||||||
label="None"
|
label="None"
|
||||||
:enabled="isNoneSelected"
|
:enabled="this.currentSelection == SelectionState.None"
|
||||||
@click="selectNoneAsync()">
|
@click="selectAsync(SelectionState.None)"
|
||||||
</SelectableOption>
|
v-tooltip="'Deselect all selected scripts. Good start to dive deeper into tweaks and select only what you want.'"
|
||||||
|
/>
|
||||||
</div>
|
</div>
|
||||||
<div class="part"> | </div>
|
<div class="part"> | </div>
|
||||||
<div class="part">
|
<div class="part">
|
||||||
<SelectableOption
|
<SelectableOption
|
||||||
label="Recommended"
|
label="Standard"
|
||||||
:enabled="isRecommendedSelected"
|
:enabled="this.currentSelection == SelectionState.Standard"
|
||||||
@click="selectRecommendedAsync()" />
|
@click="selectAsync(SelectionState.Standard)"
|
||||||
|
v-tooltip="'🛡️ Balanced for privacy and functionality. OS and applications will function normally.'"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div class="part"> | </div>
|
||||||
|
<div class="part">
|
||||||
|
<SelectableOption
|
||||||
|
label="Strict"
|
||||||
|
:enabled="this.currentSelection == SelectionState.Strict"
|
||||||
|
@click="selectAsync(SelectionState.Strict)"
|
||||||
|
v-tooltip="'🚫 Stronger privacy, disables risky functions that may leak your data. Double check selected tweaks!'"
|
||||||
|
/>
|
||||||
</div>
|
</div>
|
||||||
<div class="part"> | </div>
|
<div class="part"> | </div>
|
||||||
<div class="part">
|
<div class="part">
|
||||||
<SelectableOption
|
<SelectableOption
|
||||||
label="All"
|
label="All"
|
||||||
:enabled="isAllSelected"
|
:enabled="this.currentSelection == SelectionState.All"
|
||||||
@click="selectAllAsync()" />
|
@click="selectAsync(SelectionState.All)"
|
||||||
|
v-tooltip="'🔒 Strongest privacy. Disables any functionality that may leak your data. ⚠️ Not recommended for inexperienced users'"
|
||||||
|
/>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { Component, Prop, Vue } from 'vue-property-decorator';
|
import { Component } from 'vue-property-decorator';
|
||||||
import { StatefulVue } from '@/presentation/StatefulVue';
|
import { StatefulVue } from '@/presentation/StatefulVue';
|
||||||
import SelectableOption from './SelectableOption.vue';
|
import SelectableOption from './SelectableOption.vue';
|
||||||
import { IApplicationState } from '@/application/State/IApplicationState';
|
import { IApplicationState } from '@/application/State/IApplicationState';
|
||||||
import { IScript } from '@/domain/IScript';
|
import { IScript } from '@/domain/IScript';
|
||||||
import { SelectedScript } from '../../../application/State/Selection/SelectedScript';
|
import { SelectedScript } from '@/application/State/Selection/SelectedScript';
|
||||||
|
import { RecommendationLevel } from '@/domain/RecommendationLevel';
|
||||||
|
|
||||||
|
enum SelectionState {
|
||||||
|
Standard,
|
||||||
|
Strict,
|
||||||
|
All,
|
||||||
|
None,
|
||||||
|
Custom,
|
||||||
|
}
|
||||||
@Component({
|
@Component({
|
||||||
components: {
|
components: {
|
||||||
SelectableOption,
|
SelectableOption,
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
export default class TheSelector extends StatefulVue {
|
export default class TheSelector extends StatefulVue {
|
||||||
public isAllSelected = false;
|
public SelectionState = SelectionState;
|
||||||
public isNoneSelected = false;
|
public currentSelection = SelectionState.None;
|
||||||
public isRecommendedSelected = false;
|
|
||||||
|
|
||||||
public async mounted() {
|
public async mounted() {
|
||||||
const state = await this.getCurrentStateAsync();
|
const state = await this.getCurrentStateAsync();
|
||||||
@@ -52,43 +73,73 @@ export default class TheSelector extends StatefulVue {
|
|||||||
this.updateSelections(state);
|
this.updateSelections(state);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
public async selectAsync(type: SelectionState): Promise<void> {
|
||||||
public async selectAllAsync(): Promise<void> {
|
if (this.currentSelection === type) {
|
||||||
if (this.isAllSelected) {
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
const state = await this.getCurrentStateAsync();
|
const state = await this.getCurrentStateAsync();
|
||||||
state.selection.selectAll();
|
selectType(state, type);
|
||||||
}
|
|
||||||
|
|
||||||
public async selectRecommendedAsync(): Promise<void> {
|
|
||||||
if (this.isRecommendedSelected) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
const state = await this.getCurrentStateAsync();
|
|
||||||
state.selection.selectOnly(state.app.getRecommendedScripts());
|
|
||||||
}
|
|
||||||
|
|
||||||
public async selectNoneAsync(): Promise<void> {
|
|
||||||
if (this.isNoneSelected) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
const state = await this.getCurrentStateAsync();
|
|
||||||
state.selection.deselectAll();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private updateSelections(state: IApplicationState) {
|
private updateSelections(state: IApplicationState) {
|
||||||
this.isNoneSelected = state.selection.totalSelected === 0;
|
this.currentSelection = getCurrentSelectionState(state);
|
||||||
this.isAllSelected = state.selection.totalSelected === state.app.totalScripts;
|
|
||||||
this.isRecommendedSelected = this.areAllRecommended(state.app.getRecommendedScripts(),
|
|
||||||
state.selection.selectedScripts);
|
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
private areAllRecommended(scripts: ReadonlyArray<IScript>, other: ReadonlyArray<SelectedScript>): boolean {
|
interface ITypeSelector {
|
||||||
other = other.filter((selected) => !(selected).revert);
|
isSelected: (state: IApplicationState) => boolean;
|
||||||
return (scripts.length === other.length) &&
|
select: (state: IApplicationState) => void;
|
||||||
scripts.every((script) => other.some((selected) => selected.id === script.id));
|
}
|
||||||
|
|
||||||
|
const selectors = new Map<SelectionState, ITypeSelector>([
|
||||||
|
[SelectionState.None, {
|
||||||
|
select: (state) => state.selection.deselectAll(),
|
||||||
|
isSelected: (state) => state.selection.totalSelected === 0,
|
||||||
|
}],
|
||||||
|
[SelectionState.Standard, {
|
||||||
|
select: (state) => state.selection.selectOnly(state.app.getScriptsByLevel(RecommendationLevel.Standard)),
|
||||||
|
isSelected: (state) => hasAllSelectedLevelOf(RecommendationLevel.Standard, state),
|
||||||
|
}],
|
||||||
|
[SelectionState.Strict, {
|
||||||
|
select: (state) => state.selection.selectOnly(state.app.getScriptsByLevel(RecommendationLevel.Strict)),
|
||||||
|
isSelected: (state) => hasAllSelectedLevelOf(RecommendationLevel.Strict, state),
|
||||||
|
}],
|
||||||
|
[SelectionState.All, {
|
||||||
|
select: (state) => state.selection.selectAll(),
|
||||||
|
isSelected: (state) => state.selection.totalSelected === state.app.totalScripts,
|
||||||
|
}],
|
||||||
|
]);
|
||||||
|
|
||||||
|
function selectType(state: IApplicationState, type: SelectionState) {
|
||||||
|
const selector = selectors.get(type);
|
||||||
|
selector.select(state);
|
||||||
|
}
|
||||||
|
|
||||||
|
function getCurrentSelectionState(state: IApplicationState): SelectionState {
|
||||||
|
for (const [type, selector] of Array.from(selectors.entries())) {
|
||||||
|
if (selector.isSelected(state)) {
|
||||||
|
return type;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
return SelectionState.Custom;
|
||||||
|
}
|
||||||
|
|
||||||
|
function hasAllSelectedLevelOf(level: RecommendationLevel, state: IApplicationState) {
|
||||||
|
const scripts = state.app.getScriptsByLevel(level);
|
||||||
|
const selectedScripts = state.selection.selectedScripts;
|
||||||
|
return areAllSelected(scripts, selectedScripts);
|
||||||
|
}
|
||||||
|
|
||||||
|
function areAllSelected(
|
||||||
|
expectedScripts: ReadonlyArray<IScript>,
|
||||||
|
selection: ReadonlyArray<SelectedScript>): boolean {
|
||||||
|
selection = selection.filter((selected) => !selected.revert);
|
||||||
|
if (expectedScripts.length < selection.length) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
const selectedScriptIds = selection.map((script) => script.id).sort();
|
||||||
|
const expectedScriptIds = expectedScripts.map((script) => script.id).sort();
|
||||||
|
return selectedScriptIds.every((id, index) => id === expectedScriptIds[index]);
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
|
|||||||
@@ -7,25 +7,37 @@
|
|||||||
v-show="!this.isSearching" />
|
v-show="!this.isSearching" />
|
||||||
</div>
|
</div>
|
||||||
<div class="scripts">
|
<div class="scripts">
|
||||||
<div v-if="!isSearching || searchHasMatches">
|
<div v-if="!isSearching">
|
||||||
<CardList v-if="this.showCards" />
|
<CardList v-if="currentGrouping === Grouping.Cards"/>
|
||||||
<div v-else-if="this.showList" class="tree">
|
<div class="tree" v-if="currentGrouping === Grouping.None">
|
||||||
<div v-if="this.isSearching" class="search-query">
|
<ScriptsTree />
|
||||||
Searching for "{{this.searchQuery | threeDotsTrim}}"</div>
|
</div>
|
||||||
<ScriptsTree />
|
</div>
|
||||||
|
<div v-else> <!-- Searching -->
|
||||||
|
<div class="search">
|
||||||
|
<div class="search__query">
|
||||||
|
<div>Searching for "{{this.searchQuery | threeDotsTrim}}"</div>
|
||||||
|
<div class="search__query__close-button">
|
||||||
|
<font-awesome-icon
|
||||||
|
:icon="['fas', 'times']"
|
||||||
|
v-on:click="clearSearchQueryAsync()"/>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
<div v-if="!searchHasMatches" class="search-no-matches">
|
||||||
|
<div>Sorry, no matches for "{{this.searchQuery | threeDotsTrim}}" 😞</div>
|
||||||
|
<div>Feel free to extend the scripts <a :href="repositoryUrl" target="_blank" class="child github" >here</a> ✨</div>
|
||||||
</div>
|
</div>
|
||||||
<div v-else class="search-no-matches">
|
</div>
|
||||||
Sorry, no matches for "{{this.searchQuery | threeDotsTrim}}" 😞
|
<div v-if="searchHasMatches" class="tree tree--searching">
|
||||||
Feel free to extend the scripts <a :href="repositoryUrl" target="_blank" class="child github" >here</a>.
|
<ScriptsTree />
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { Component, Prop, Vue, Emit } from 'vue-property-decorator';
|
import { Component } from 'vue-property-decorator';
|
||||||
import { Category } from '@/domain/Category';
|
|
||||||
import { StatefulVue } from '@/presentation/StatefulVue';
|
import { StatefulVue } from '@/presentation/StatefulVue';
|
||||||
import { Grouping } from './Grouping/Grouping';
|
import { Grouping } from './Grouping/Grouping';
|
||||||
import { IFilterResult } from '@/application/State/Filter/IFilterResult';
|
import { IFilterResult } from '@/application/State/Filter/IFilterResult';
|
||||||
@@ -34,7 +46,6 @@
|
|||||||
import ScriptsTree from '@/presentation/Scripts/ScriptsTree/ScriptsTree.vue';
|
import ScriptsTree from '@/presentation/Scripts/ScriptsTree/ScriptsTree.vue';
|
||||||
import CardList from '@/presentation/Scripts/Cards/CardList.vue';
|
import CardList from '@/presentation/Scripts/Cards/CardList.vue';
|
||||||
|
|
||||||
|
|
||||||
/** Shows content of single category or many categories */
|
/** Shows content of single category or many categories */
|
||||||
@Component({
|
@Component({
|
||||||
components: {
|
components: {
|
||||||
@@ -54,38 +65,33 @@
|
|||||||
},
|
},
|
||||||
})
|
})
|
||||||
export default class TheScripts extends StatefulVue {
|
export default class TheScripts extends StatefulVue {
|
||||||
public showCards = false;
|
|
||||||
public showList = false;
|
|
||||||
public repositoryUrl = '';
|
public repositoryUrl = '';
|
||||||
private searchQuery = '';
|
public Grouping = Grouping; // Make it accessible from view
|
||||||
private isSearching = false;
|
public currentGrouping = Grouping.Cards;
|
||||||
private searchHasMatches = false;
|
public searchQuery = '';
|
||||||
|
public isSearching = false;
|
||||||
private currentGrouping: Grouping;
|
public searchHasMatches = false;
|
||||||
|
|
||||||
public async mounted() {
|
public async mounted() {
|
||||||
const state = await this.getCurrentStateAsync();
|
const state = await this.getCurrentStateAsync();
|
||||||
this.repositoryUrl = state.app.repositoryUrl;
|
this.repositoryUrl = state.app.info.repositoryWebUrl;
|
||||||
state.filter.filterRemoved.on(() => {
|
state.filter.filterRemoved.on(() => {
|
||||||
this.isSearching = false;
|
this.isSearching = false;
|
||||||
this.updateGroups();
|
|
||||||
});
|
});
|
||||||
state.filter.filtered.on((result: IFilterResult) => {
|
state.filter.filtered.on((result: IFilterResult) => {
|
||||||
this.searchQuery = result.query;
|
this.searchQuery = result.query;
|
||||||
this.isSearching = true;
|
this.isSearching = true;
|
||||||
this.searchHasMatches = result.hasAnyMatches();
|
this.searchHasMatches = result.hasAnyMatches();
|
||||||
this.updateGroups();
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public async clearSearchQueryAsync() {
|
||||||
|
const state = await this.getCurrentStateAsync();
|
||||||
|
state.filter.removeFilter();
|
||||||
|
}
|
||||||
|
|
||||||
public onGroupingChanged(group: Grouping) {
|
public onGroupingChanged(group: Grouping) {
|
||||||
this.currentGrouping = group;
|
this.currentGrouping = group;
|
||||||
this.updateGroups();
|
|
||||||
}
|
|
||||||
|
|
||||||
private updateGroups(): void {
|
|
||||||
this.showCards = !this.isSearching && this.currentGrouping === Grouping.Cards;
|
|
||||||
this.showList = this.isSearching || this.currentGrouping === Grouping.None;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
@@ -95,26 +101,49 @@
|
|||||||
@import "@/presentation/styles/fonts.scss";
|
@import "@/presentation/styles/fonts.scss";
|
||||||
.scripts {
|
.scripts {
|
||||||
margin-top:10px;
|
margin-top:10px;
|
||||||
.search-no-matches {
|
|
||||||
word-break:break-word;
|
|
||||||
color: $white;
|
|
||||||
text-transform: uppercase;
|
|
||||||
color: $light-gray;
|
|
||||||
font-size: 1.5em;
|
|
||||||
background-color: $slate;
|
|
||||||
padding:5%;
|
|
||||||
text-align:center;
|
|
||||||
> a {
|
|
||||||
color: $gray;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
.tree {
|
.tree {
|
||||||
padding-left: 3%;
|
padding-left: 3%;
|
||||||
padding-top: 15px;
|
padding-top: 15px;
|
||||||
padding-bottom: 15px;
|
padding-bottom: 15px;
|
||||||
.search-query {
|
&--searching {
|
||||||
display: flex;
|
padding-top: 0px;
|
||||||
justify-content: center;
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.search {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
background-color: $slate;
|
||||||
|
&__query {
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
flex-direction: row;
|
||||||
|
align-items: center;
|
||||||
|
margin-top: 1em;
|
||||||
|
color: $gray;
|
||||||
|
&__close-button {
|
||||||
|
cursor: pointer;
|
||||||
|
font-size: 1.25em;
|
||||||
|
margin-left: 0.25rem;
|
||||||
|
&:hover {
|
||||||
|
opacity: 0.9;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
&-no-matches {
|
||||||
|
display:flex;
|
||||||
|
flex-direction: column;
|
||||||
|
word-break:break-word;
|
||||||
|
text-transform: uppercase;
|
||||||
|
color: $light-gray;
|
||||||
|
font-size: 1.5em;
|
||||||
|
padding:10px;
|
||||||
|
text-align:center;
|
||||||
|
> div {
|
||||||
|
padding-bottom:13px;
|
||||||
|
}
|
||||||
|
a {
|
||||||
color: $gray;
|
color: $gray;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,11 +3,13 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { Component, Prop, Watch, Vue } from 'vue-property-decorator';
|
import { Component, Prop } from 'vue-property-decorator';
|
||||||
import { StatefulVue } from './StatefulVue';
|
import { StatefulVue } from './StatefulVue';
|
||||||
import ace from 'ace-builds';
|
import ace from 'ace-builds';
|
||||||
import 'ace-builds/webpack-resolver';
|
import 'ace-builds/webpack-resolver';
|
||||||
import { CodeBuilder } from '../application/State/Code/CodeBuilder';
|
import { CodeBuilder } from '@/application/State/Code/Generation/CodeBuilder';
|
||||||
|
import { ICodeChangedEvent } from '@/application/State/Code/Event/ICodeChangedEvent';
|
||||||
|
import { IScript } from '@/domain/IScript';
|
||||||
|
|
||||||
const NothingChosenCode =
|
const NothingChosenCode =
|
||||||
new CodeBuilder()
|
new CodeBuilder()
|
||||||
@@ -15,32 +17,79 @@ const NothingChosenCode =
|
|||||||
.appendLine()
|
.appendLine()
|
||||||
.appendCommentLine('-- 🤔 How to use')
|
.appendCommentLine('-- 🤔 How to use')
|
||||||
.appendCommentLine(' 📙 Start by exploring different categories and choosing different tweaks.')
|
.appendCommentLine(' 📙 Start by exploring different categories and choosing different tweaks.')
|
||||||
.appendCommentLine(' 📙 You can select "Recommended" on the top to select "safer" tweaks. Always double check!')
|
.appendCommentLine(' 📙 On top left, you can apply predefined selections for privacy level you\'d like.')
|
||||||
.appendCommentLine(' 📙 After you choose any tweak, you can download & copy to execute your script.')
|
.appendCommentLine(' 📙 After you choose any tweak, you can download or copy to execute your script.')
|
||||||
|
.appendCommentLine(' 📙 Come back regularly to apply latest version for stronger privacy and security.')
|
||||||
.appendLine()
|
.appendLine()
|
||||||
.appendCommentLine('-- 🧐 Why privacy.sexy')
|
.appendCommentLine('-- 🧐 Why privacy.sexy')
|
||||||
.appendCommentLine(' ✔️ Rich tweak pool to harden security & privacy of the OS and other softwares on it.')
|
.appendCommentLine(' ✔️ Rich tweak pool to harden security & privacy of the OS and other software on it.')
|
||||||
.appendCommentLine(' ✔️ You don\'t need to run any compiled software on your system, just run the generated scripts.')
|
.appendCommentLine(' ✔️ No need to run any compiled software on your system, just run the generated scripts.')
|
||||||
.appendCommentLine(' ✔️ Have full visibility into what the tweaks do as you enable them.')
|
.appendCommentLine(' ✔️ Have full visibility into what the tweaks do as you enable them.')
|
||||||
.appendCommentLine(' ✔️ Free software, 100% transparency: both application & infrastructure code are open-sourced.')
|
.appendCommentLine(' ✔️ Open-source and free (both free as in beer and free as in speech).')
|
||||||
.toString();
|
.toString();
|
||||||
|
|
||||||
@Component
|
@Component
|
||||||
export default class TheCodeArea extends StatefulVue {
|
export default class TheCodeArea extends StatefulVue {
|
||||||
public readonly editorId = 'codeEditor';
|
public readonly editorId = 'codeEditor';
|
||||||
|
|
||||||
private editor!: ace.Ace.Editor;
|
private editor!: ace.Ace.Editor;
|
||||||
|
private currentMarkerId?: number;
|
||||||
|
|
||||||
@Prop() private theme!: string;
|
@Prop() private theme!: string;
|
||||||
|
|
||||||
public async mounted() {
|
public async mounted() {
|
||||||
this.editor = initializeEditor(this.theme, this.editorId);
|
this.editor = initializeEditor(this.theme, this.editorId);
|
||||||
const state = await this.getCurrentStateAsync();
|
const state = await this.getCurrentStateAsync();
|
||||||
this.updateCode(state.code.current);
|
this.editor.setValue(state.code.current || NothingChosenCode, 1);
|
||||||
state.code.changed.on((code) => this.updateCode(code));
|
state.code.changed.on((code) => this.updateCode(code));
|
||||||
}
|
}
|
||||||
|
|
||||||
private updateCode(code: string) {
|
private updateCode(event: ICodeChangedEvent) {
|
||||||
this.editor.setValue(code || NothingChosenCode, 1);
|
this.removeCurrentHighlighting();
|
||||||
|
if (event.isEmpty()) {
|
||||||
|
this.editor.setValue(NothingChosenCode, 1);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
this.editor.setValue(event.code, 1);
|
||||||
|
|
||||||
|
if (event.addedScripts && event.addedScripts.length) {
|
||||||
|
this.reactToChanges(event, event.addedScripts);
|
||||||
|
} else if (event.changedScripts && event.changedScripts.length) {
|
||||||
|
this.reactToChanges(event, event.changedScripts);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private reactToChanges(event: ICodeChangedEvent, scripts: ReadonlyArray<IScript>) {
|
||||||
|
const positions = scripts
|
||||||
|
.map((script) => event.getScriptPositionInCode(script));
|
||||||
|
const start = Math.min(
|
||||||
|
...positions.map((position) => position.startLine),
|
||||||
|
);
|
||||||
|
const end = Math.max(
|
||||||
|
...positions.map((position) => position.endLine),
|
||||||
|
);
|
||||||
|
this.scrollToLine(end + 2);
|
||||||
|
this.highlight(start, end);
|
||||||
|
}
|
||||||
|
|
||||||
|
private highlight(startRow: number, endRow: number) {
|
||||||
|
const AceRange = ace.require('ace/range').Range;
|
||||||
|
this.currentMarkerId = this.editor.session.addMarker(
|
||||||
|
new AceRange(startRow, 0, endRow, 0), 'code-area__highlight', 'fullLine',
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
private scrollToLine(row: number) {
|
||||||
|
const column = this.editor.session.getLine(row).length;
|
||||||
|
this.editor.gotoLine(row, column, true);
|
||||||
|
}
|
||||||
|
|
||||||
|
private removeCurrentHighlighting() {
|
||||||
|
if (!this.currentMarkerId) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
this.editor.session.removeMarker(this.currentMarkerId);
|
||||||
|
this.currentMarkerId = undefined;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -58,12 +107,17 @@ function initializeEditor(theme: string, editorId: string): ace.Ace.Editor {
|
|||||||
|
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style scoped lang="scss">
|
<style lang="scss">
|
||||||
|
@import "@/presentation/styles/colors.scss";
|
||||||
.code-area {
|
.code-area {
|
||||||
/* ----- Fill its parent div ------ */
|
|
||||||
width: 100%;
|
width: 100%;
|
||||||
/* height */
|
|
||||||
max-height: 1000px;
|
max-height: 1000px;
|
||||||
min-height: 200px;
|
min-height: 200px;
|
||||||
|
overflow: auto;
|
||||||
|
&__highlight {
|
||||||
|
background-color:$accent;
|
||||||
|
opacity: 0.2; // having procent fails in production (minified) build
|
||||||
|
position:absolute;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -1,25 +1,27 @@
|
|||||||
<template>
|
<template>
|
||||||
<div class="container" v-if="hasCode">
|
<div class="container" v-if="hasCode">
|
||||||
|
</IconButton>
|
||||||
|
<IconButton
|
||||||
|
:text="this.isDesktop ? 'Save' : 'Download'"
|
||||||
|
v-on:click="saveCodeAsync"
|
||||||
|
icon-prefix="fas"
|
||||||
|
:icon-name="this.isDesktop ? 'save' : 'file-download'">
|
||||||
|
</IconButton>
|
||||||
<IconButton
|
<IconButton
|
||||||
text="Copy"
|
text="Copy"
|
||||||
v-on:click="copyCodeAsync"
|
v-on:click="copyCodeAsync"
|
||||||
icon-prefix="fas" icon-name="copy">
|
icon-prefix="fas" icon-name="copy">
|
||||||
</IconButton>
|
</IconButton>
|
||||||
<IconButton
|
|
||||||
text="Download"
|
|
||||||
v-on:click="saveCodeAsync"
|
|
||||||
icon-prefix="fas" icon-name="file-download">
|
|
||||||
</IconButton>
|
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { Component, Prop, Vue } from 'vue-property-decorator';
|
import { Component } from 'vue-property-decorator';
|
||||||
import { StatefulVue } from './StatefulVue';
|
import { StatefulVue } from './StatefulVue';
|
||||||
import { SaveFileDialog } from './../infrastructure/SaveFileDialog';
|
import { SaveFileDialog, FileType } from '@/infrastructure/SaveFileDialog';
|
||||||
import { Clipboard } from './../infrastructure/Clipboard';
|
import { Clipboard } from '@/infrastructure/Clipboard';
|
||||||
import IconButton from './IconButton.vue';
|
import IconButton from './IconButton.vue';
|
||||||
|
import { Environment } from '@/application/Environment/Environment';
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
components: {
|
components: {
|
||||||
@@ -28,12 +30,14 @@ import IconButton from './IconButton.vue';
|
|||||||
})
|
})
|
||||||
export default class TheCodeButtons extends StatefulVue {
|
export default class TheCodeButtons extends StatefulVue {
|
||||||
public hasCode = false;
|
public hasCode = false;
|
||||||
|
public isDesktop = false;
|
||||||
|
|
||||||
public async mounted() {
|
public async mounted() {
|
||||||
const state = await this.getCurrentStateAsync();
|
const state = await this.getCurrentStateAsync();
|
||||||
|
this.isDesktop = Environment.CurrentEnvironment.isDesktop;
|
||||||
this.hasCode = state.code.current && state.code.current.length > 0;
|
this.hasCode = state.code.current && state.code.current.length > 0;
|
||||||
state.code.changed.on((code) => {
|
state.code.changed.on((code) => {
|
||||||
this.hasCode = code && code.length > 0;
|
this.hasCode = code && code.code.length > 0;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -44,7 +48,7 @@ export default class TheCodeButtons extends StatefulVue {
|
|||||||
|
|
||||||
public async saveCodeAsync() {
|
public async saveCodeAsync() {
|
||||||
const state = await this.getCurrentStateAsync();
|
const state = await this.getCurrentStateAsync();
|
||||||
SaveFileDialog.saveText(state.code.current, 'privacy-script.bat');
|
SaveFileDialog.saveFile(state.code.current, 'privacy-script.bat', FileType.BatchFile);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
@@ -15,9 +15,9 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { Component, Prop, Vue } from 'vue-property-decorator';
|
import { Component, Vue } from 'vue-property-decorator';
|
||||||
import { Environment } from '@/application/Environment/Environment';
|
import { Environment } from '@/application/Environment/Environment';
|
||||||
import { OperatingSystem } from '@/application/Environment/OperatingSystem';
|
import { OperatingSystem } from '@/domain/OperatingSystem';
|
||||||
import DownloadUrlListItem from './DownloadUrlListItem.vue';
|
import DownloadUrlListItem from './DownloadUrlListItem.vue';
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
|
|||||||
@@ -9,10 +9,10 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { Component, Prop, Vue, Watch } from 'vue-property-decorator';
|
import { Component, Prop, Watch } from 'vue-property-decorator';
|
||||||
import { StatefulVue } from '@/presentation/StatefulVue';
|
import { StatefulVue } from '@/presentation/StatefulVue';
|
||||||
import { Environment } from '@/application/Environment/Environment';
|
import { Environment } from '@/application/Environment/Environment';
|
||||||
import { OperatingSystem } from '@/application/Environment/OperatingSystem';
|
import { OperatingSystem } from '@/domain/OperatingSystem';
|
||||||
|
|
||||||
@Component
|
@Component
|
||||||
export default class DownloadUrlListItem extends StatefulVue {
|
export default class DownloadUrlListItem extends StatefulVue {
|
||||||
@@ -39,7 +39,7 @@ export default class DownloadUrlListItem extends StatefulVue {
|
|||||||
|
|
||||||
private async getDownloadUrlAsync(os: OperatingSystem): Promise<string> {
|
private async getDownloadUrlAsync(os: OperatingSystem): Promise<string> {
|
||||||
const state = await this.getCurrentStateAsync();
|
const state = await this.getCurrentStateAsync();
|
||||||
return `${state.app.repositoryUrl}/releases/download/${state.app.version}/${getFileName(os, state.app.version)}`;
|
return state.app.info.getDownloadUrl(os);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -62,18 +62,6 @@ function getOperatingSystemName(os: OperatingSystem): string {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function getFileName(os: OperatingSystem, version: string): string {
|
|
||||||
switch (os) {
|
|
||||||
case OperatingSystem.Linux:
|
|
||||||
return `privacy.sexy-${version}.AppImage`;
|
|
||||||
case OperatingSystem.macOS:
|
|
||||||
return `privacy.sexy-${version}.dmg`;
|
|
||||||
case OperatingSystem.Windows:
|
|
||||||
return `privacy.sexy-Setup-${version}.exe`;
|
|
||||||
default:
|
|
||||||
throw new Error(`Unsupported os: ${OperatingSystem[os]}`);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style scoped lang="scss">
|
<style scoped lang="scss">
|
||||||
|
|||||||
@@ -31,7 +31,7 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { Component, Prop, Vue } from 'vue-property-decorator';
|
import { Component } from 'vue-property-decorator';
|
||||||
import { StatefulVue } from '@/presentation/StatefulVue';
|
import { StatefulVue } from '@/presentation/StatefulVue';
|
||||||
import { Environment } from '@/application/Environment/Environment';
|
import { Environment } from '@/application/Environment/Environment';
|
||||||
|
|
||||||
@@ -48,8 +48,8 @@ export default class TheFooter extends StatefulVue {
|
|||||||
|
|
||||||
public async mounted() {
|
public async mounted() {
|
||||||
const state = await this.getCurrentStateAsync();
|
const state = await this.getCurrentStateAsync();
|
||||||
this.repositoryUrl = state.app.repositoryUrl;
|
this.repositoryUrl = state.app.info.repositoryWebUrl;
|
||||||
this.feedbackUrl = `${state.app.repositoryUrl}/issues`;
|
this.feedbackUrl = state.app.info.feedbackUrl;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
@@ -4,7 +4,7 @@
|
|||||||
<div class="footer__section">
|
<div class="footer__section">
|
||||||
<span v-if="isDesktop" class="footer__section__item">
|
<span v-if="isDesktop" class="footer__section__item">
|
||||||
<font-awesome-icon class="icon" :icon="['fas', 'globe']" />
|
<font-awesome-icon class="icon" :icon="['fas', 'globe']" />
|
||||||
<span>Online version at <a href="https://privacy.sexy" target="_blank">https://privacy.sexy</a></span>
|
<span>Online version at <a :href="homepageUrl" target="_blank">{{ homepageUrl }}</a></span>
|
||||||
</span>
|
</span>
|
||||||
<span v-else class="footer__section__item">
|
<span v-else class="footer__section__item">
|
||||||
<DownloadUrlList />
|
<DownloadUrlList />
|
||||||
@@ -47,12 +47,11 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { Component, Prop, Vue } from 'vue-property-decorator';
|
import { Component } from 'vue-property-decorator';
|
||||||
import { StatefulVue } from '@/presentation/StatefulVue';
|
import { StatefulVue } from '@/presentation/StatefulVue';
|
||||||
import { Environment } from '@/application/Environment/Environment';
|
import { Environment } from '@/application/Environment/Environment';
|
||||||
import PrivacyPolicy from './PrivacyPolicy.vue';
|
import PrivacyPolicy from './PrivacyPolicy.vue';
|
||||||
import DownloadUrlList from './DownloadUrlList.vue';
|
import DownloadUrlList from './DownloadUrlList.vue';
|
||||||
import { OperatingSystem } from '@/application/Environment/OperatingSystem';
|
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
components: {
|
components: {
|
||||||
@@ -67,6 +66,7 @@ export default class TheFooter extends StatefulVue {
|
|||||||
public repositoryUrl: string = '';
|
public repositoryUrl: string = '';
|
||||||
public releaseUrl: string = '';
|
public releaseUrl: string = '';
|
||||||
public feedbackUrl: string = '';
|
public feedbackUrl: string = '';
|
||||||
|
public homepageUrl: string = '';
|
||||||
|
|
||||||
constructor() {
|
constructor() {
|
||||||
super();
|
super();
|
||||||
@@ -75,12 +75,15 @@ export default class TheFooter extends StatefulVue {
|
|||||||
|
|
||||||
public async mounted() {
|
public async mounted() {
|
||||||
const state = await this.getCurrentStateAsync();
|
const state = await this.getCurrentStateAsync();
|
||||||
this.version = state.app.version;
|
const info = state.app.info;
|
||||||
this.repositoryUrl = state.app.repositoryUrl;
|
this.version = info.version;
|
||||||
this.releaseUrl = `${state.app.repositoryUrl}/releases/tag/${state.app.version}`;
|
this.homepageUrl = info.homepage;
|
||||||
this.feedbackUrl = `${state.app.repositoryUrl}/issues`;
|
this.repositoryUrl = info.repositoryWebUrl;
|
||||||
|
this.releaseUrl = info.releaseUrl;
|
||||||
|
this.feedbackUrl = info.feedbackUrl;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style scoped lang="scss">
|
<style scoped lang="scss">
|
||||||
|
|||||||
@@ -6,7 +6,7 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { Component, Prop, Vue } from 'vue-property-decorator';
|
import { Component } from 'vue-property-decorator';
|
||||||
import { StatefulVue } from './StatefulVue';
|
import { StatefulVue } from './StatefulVue';
|
||||||
|
|
||||||
@Component
|
@Component
|
||||||
@@ -16,7 +16,7 @@ export default class TheHeader extends StatefulVue {
|
|||||||
|
|
||||||
public async mounted() {
|
public async mounted() {
|
||||||
const state = await this.getCurrentStateAsync();
|
const state = await this.getCurrentStateAsync();
|
||||||
this.title = state.app.name;
|
this.title = state.app.info.name;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
<div class="search" v-non-collapsing>
|
<div class="search" v-non-collapsing>
|
||||||
<input type="search" class="searchTerm"
|
<input type="search" class="searchTerm"
|
||||||
:placeholder="searchPlaceHolder"
|
:placeholder="searchPlaceHolder"
|
||||||
@input="updateFilterAsync($event.target.value)" >
|
v-model="searchQuery" >
|
||||||
<div class="iconWrapper">
|
<div class="iconWrapper">
|
||||||
<font-awesome-icon :icon="['fas', 'search']" />
|
<font-awesome-icon :icon="['fas', 'search']" />
|
||||||
</div>
|
</div>
|
||||||
@@ -10,9 +10,10 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { Component, Prop, Vue, Watch } from 'vue-property-decorator';
|
import { Component, Watch } from 'vue-property-decorator';
|
||||||
import { StatefulVue } from './StatefulVue';
|
import { StatefulVue } from './StatefulVue';
|
||||||
import { NonCollapsing } from '@/presentation/Scripts/Cards/NonCollapsingDirective';
|
import { NonCollapsing } from '@/presentation/Scripts/Cards/NonCollapsingDirective';
|
||||||
|
import { IUserFilter } from '@/application/State/IApplicationState';
|
||||||
|
|
||||||
@Component( {
|
@Component( {
|
||||||
directives: { NonCollapsing },
|
directives: { NonCollapsing },
|
||||||
@@ -20,14 +21,16 @@ import { NonCollapsing } from '@/presentation/Scripts/Cards/NonCollapsingDirecti
|
|||||||
)
|
)
|
||||||
export default class TheSearchBar extends StatefulVue {
|
export default class TheSearchBar extends StatefulVue {
|
||||||
public searchPlaceHolder = 'Search';
|
public searchPlaceHolder = 'Search';
|
||||||
|
public searchQuery = '';
|
||||||
|
|
||||||
public async mounted() {
|
public async mounted() {
|
||||||
const state = await this.getCurrentStateAsync();
|
const state = await this.getCurrentStateAsync();
|
||||||
const totalScripts = state.app.totalScripts;
|
const totalScripts = state.app.totalScripts;
|
||||||
const totalCategories = state.app.totalCategories;
|
|
||||||
this.searchPlaceHolder = `Search in ${totalScripts} scripts`;
|
this.searchPlaceHolder = `Search in ${totalScripts} scripts`;
|
||||||
|
this.beginReacting(state.filter);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Watch('searchQuery')
|
||||||
public async updateFilterAsync(filter: |string) {
|
public async updateFilterAsync(filter: |string) {
|
||||||
const state = await this.getCurrentStateAsync();
|
const state = await this.getCurrentStateAsync();
|
||||||
if (!filter) {
|
if (!filter) {
|
||||||
@@ -37,6 +40,10 @@ export default class TheSearchBar extends StatefulVue {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private beginReacting(filter: IUserFilter) {
|
||||||
|
filter.filtered.on((result) => this.searchQuery = result.query);
|
||||||
|
filter.filterRemoved.on(() => this.searchQuery = '');
|
||||||
|
}
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
@@ -80,5 +87,4 @@ export default class TheSearchBar extends StatefulVue {
|
|||||||
font-size: 20px;
|
font-size: 20px;
|
||||||
padding:5px;
|
padding:5px;
|
||||||
}
|
}
|
||||||
|
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -2,42 +2,41 @@
|
|||||||
@import "@/presentation/styles/colors.scss";
|
@import "@/presentation/styles/colors.scss";
|
||||||
|
|
||||||
.tree {
|
.tree {
|
||||||
background-color: $slate;
|
background: $slate;
|
||||||
}
|
&-node {
|
||||||
|
white-space: normal !important;
|
||||||
.tree-node > .tree-content > .tree-anchor > span {
|
> .tree-content {
|
||||||
color: $white !important;
|
> .tree-anchor > span {
|
||||||
text-transform: uppercase;
|
color: $white;
|
||||||
color: $light-gray;
|
text-transform: uppercase;
|
||||||
font-size: 1.5em;
|
font-size: 1.5em;
|
||||||
}
|
}
|
||||||
|
&:hover {
|
||||||
.tree-node {
|
background: $dark-gray !important;
|
||||||
white-space: normal !important;
|
}
|
||||||
}
|
}
|
||||||
|
&.selected { // When using keyboard navigation it higlights current item and its child items
|
||||||
.tree-arrow.has-child {
|
background: $gray;
|
||||||
&.rtl:after, &:after {
|
.tree-text {
|
||||||
border-color: $white !important;
|
color: $black !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
&-checkbox {
|
||||||
|
&.checked {
|
||||||
|
background: $accent !important;
|
||||||
|
border-color: $accent !important;
|
||||||
|
}
|
||||||
|
&.indeterminate {
|
||||||
|
border-color: $gray !important;
|
||||||
|
}
|
||||||
|
background: $dark-slate !important;
|
||||||
|
}
|
||||||
|
&-arrow {
|
||||||
|
&.has-child {
|
||||||
|
&.rtl:after, &:after {
|
||||||
|
border-color: $white !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.tree-node.selected > .tree-content {
|
|
||||||
> .tree-anchor > span {
|
|
||||||
font-weight: bolder;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
.tree-content:hover {
|
|
||||||
background: $dark-gray !important;
|
|
||||||
}
|
|
||||||
|
|
||||||
.tree-checkbox {
|
|
||||||
&.checked {
|
|
||||||
background: $accent !important;
|
|
||||||
}
|
|
||||||
&.indeterminate {
|
|
||||||
border-color: $gray !important;
|
|
||||||
}
|
|
||||||
background: $dark-slate !important;
|
|
||||||
}
|
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
import { expect } from 'chai';
|
import { expect } from 'chai';
|
||||||
import { OperatingSystem } from '@/application/Environment/OperatingSystem';
|
import { OperatingSystem } from '@/domain/OperatingSystem';
|
||||||
import { BrowserOsDetector } from '@/application/Environment/BrowserOs/BrowserOsDetector';
|
import { BrowserOsDetector } from '@/application/Environment/BrowserOs/BrowserOsDetector';
|
||||||
import { BrowserOsTestCases } from './BrowserOsTestCases';
|
import { BrowserOsTestCases } from './BrowserOsTestCases';
|
||||||
|
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { OperatingSystem } from '@/application/Environment/OperatingSystem';
|
import { OperatingSystem } from '@/domain/OperatingSystem';
|
||||||
|
|
||||||
interface IBrowserOsTestCase {
|
interface IBrowserOsTestCase {
|
||||||
userAgent: string;
|
userAgent: string;
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { OperatingSystem } from '@/application/Environment/OperatingSystem';
|
import { OperatingSystem } from '@/domain/OperatingSystem';
|
||||||
|
|
||||||
interface IDesktopTestCase {
|
interface IDesktopTestCase {
|
||||||
processPlatform: string;
|
processPlatform: string;
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import { IBrowserOsDetector } from '@/application/Environment/BrowserOs/IBrowserOsDetector';
|
import { IBrowserOsDetector } from '@/application/Environment/BrowserOs/IBrowserOsDetector';
|
||||||
import { OperatingSystem } from '@/application/Environment/OperatingSystem';
|
import { OperatingSystem } from '@/domain/OperatingSystem';
|
||||||
import { DesktopOsTestCases } from './DesktopOsTestCases';
|
import { DesktopOsTestCases } from './DesktopOsTestCases';
|
||||||
import { Environment } from '@/application/Environment/Environment';
|
import { Environment } from '@/application/Environment/Environment';
|
||||||
import { expect } from 'chai';
|
import { expect } from 'chai';
|
||||||
|
|||||||
@@ -4,86 +4,95 @@ import { parseApplication } from '@/application/Parser/ApplicationParser';
|
|||||||
import 'mocha';
|
import 'mocha';
|
||||||
import { expect } from 'chai';
|
import { expect } from 'chai';
|
||||||
import { parseCategory } from '@/application/Parser/CategoryParser';
|
import { parseCategory } from '@/application/Parser/CategoryParser';
|
||||||
|
import { RecommendationLevel } from '@/domain/RecommendationLevel';
|
||||||
declare var process;
|
import { ScriptCompilerStub } from '../../stubs/ScriptCompilerStub';
|
||||||
|
|
||||||
describe('ApplicationParser', () => {
|
describe('ApplicationParser', () => {
|
||||||
describe('parseApplication', () => {
|
describe('parseApplication', () => {
|
||||||
it('can parse current application file', () => {
|
it('can parse current application file', () => {
|
||||||
expect(() => parseApplication(applicationFile)).to.not.throw();
|
// act
|
||||||
|
const act = () => parseApplication(applicationFile);
|
||||||
|
// assert
|
||||||
|
expect(act).to.not.throw();
|
||||||
});
|
});
|
||||||
it('throws when undefined', () => {
|
it('throws when undefined', () => {
|
||||||
expect(() => parseApplication(undefined)).to.throw('application is null or undefined');
|
// arrange
|
||||||
|
const expectedError = 'application is null or undefined';
|
||||||
|
// act
|
||||||
|
const act = () => parseApplication(undefined);
|
||||||
|
// assert
|
||||||
|
expect(act).to.throw(expectedError);
|
||||||
});
|
});
|
||||||
it('throws when undefined actions', () => {
|
it('throws when undefined actions', () => {
|
||||||
const sut: ApplicationYaml = {
|
// arrange
|
||||||
name: 'test',
|
const sut: ApplicationYaml = { actions: undefined, functions: undefined };
|
||||||
repositoryUrl: 'https://privacy.sexy',
|
const expectedError = 'application does not define any action';
|
||||||
actions: undefined,
|
// act
|
||||||
};
|
const act = () => parseApplication(sut);
|
||||||
expect(() => parseApplication(sut)).to.throw('application does not define any action');
|
// assert
|
||||||
|
expect(act).to.throw(expectedError);
|
||||||
});
|
});
|
||||||
it('throws when has no actions', () => {
|
it('throws when has no actions', () => {
|
||||||
const sut: ApplicationYaml = {
|
|
||||||
name: 'test',
|
|
||||||
repositoryUrl: 'https://privacy.sexy',
|
|
||||||
actions: [],
|
|
||||||
};
|
|
||||||
expect(() => parseApplication(sut)).to.throw('application does not define any action');
|
|
||||||
});
|
|
||||||
it('returns expected name', () => {
|
|
||||||
// arrange
|
// arrange
|
||||||
const expected = 'test-app-name';
|
const sut: ApplicationYaml = { actions: [], functions: undefined };
|
||||||
const sut: ApplicationYaml = {
|
const expectedError = 'application does not define any action';
|
||||||
name: expected,
|
|
||||||
repositoryUrl: 'https://privacy.sexy',
|
|
||||||
actions: [ getTestCategory() ],
|
|
||||||
};
|
|
||||||
// act
|
// act
|
||||||
const actual = parseApplication(sut).name;
|
const act = () => parseApplication(sut);
|
||||||
// assert
|
// assert
|
||||||
expect(actual).to.be.equal(actual);
|
expect(act).to.throw(expectedError);
|
||||||
});
|
});
|
||||||
it('returns expected repository url', () => {
|
describe('information', () => {
|
||||||
// arrange
|
it('returns expected repository version', () => {
|
||||||
const expected = 'https://privacy.sexy';
|
// arrange
|
||||||
const sut: ApplicationYaml = {
|
const expected = 'expected-version';
|
||||||
name: 'name',
|
const env = getProcessEnvironmentStub();
|
||||||
repositoryUrl: expected,
|
env.VUE_APP_VERSION = expected;
|
||||||
actions: [ getTestCategory() ],
|
const sut: ApplicationYaml = { actions: [ getTestCategory() ], functions: undefined };
|
||||||
};
|
// act
|
||||||
// act
|
const actual = parseApplication(sut, env).info.version;
|
||||||
const actual = parseApplication(sut).repositoryUrl;
|
// assert
|
||||||
// assert
|
expect(actual).to.be.equal(expected);
|
||||||
expect(actual).to.be.equal(actual);
|
});
|
||||||
});
|
it('returns expected repository url', () => {
|
||||||
it('returns expected repository version', () => {
|
// arrange
|
||||||
// arrange
|
const expected = 'https://expected-repository.url';
|
||||||
const expected = '1.0.0';
|
const env = getProcessEnvironmentStub();
|
||||||
process = {
|
env.VUE_APP_REPOSITORY_URL = expected;
|
||||||
env: {
|
const sut: ApplicationYaml = { actions: [ getTestCategory() ], functions: undefined };
|
||||||
VUE_APP_VERSION: expected,
|
// act
|
||||||
},
|
const actual = parseApplication(sut, env).info.repositoryUrl;
|
||||||
};
|
// assert
|
||||||
const sut: ApplicationYaml = {
|
expect(actual).to.be.equal(expected);
|
||||||
name: 'name',
|
});
|
||||||
repositoryUrl: 'https://privacy.sexy',
|
it('returns expected name', () => {
|
||||||
actions: [ getTestCategory() ],
|
// arrange
|
||||||
};
|
const expected = 'expected-app-name';
|
||||||
// act
|
const env = getProcessEnvironmentStub();
|
||||||
const actual = parseApplication(sut).version;
|
env.VUE_APP_NAME = expected;
|
||||||
// assert
|
const sut: ApplicationYaml = { actions: [ getTestCategory() ], functions: undefined };
|
||||||
expect(actual).to.be.equal(actual);
|
// act
|
||||||
|
const actual = parseApplication(sut, env).info.name;
|
||||||
|
// assert
|
||||||
|
expect(actual).to.be.equal(expected);
|
||||||
|
});
|
||||||
|
it('returns expected homepage url', () => {
|
||||||
|
// arrange
|
||||||
|
const expected = 'https://expected.sexy';
|
||||||
|
const env = getProcessEnvironmentStub();
|
||||||
|
env.VUE_APP_HOMEPAGE_URL = expected;
|
||||||
|
const sut: ApplicationYaml = { actions: [ getTestCategory() ], functions: undefined };
|
||||||
|
// act
|
||||||
|
const actual = parseApplication(sut, env).info.homepage;
|
||||||
|
// assert
|
||||||
|
expect(actual).to.be.equal(expected);
|
||||||
|
});
|
||||||
});
|
});
|
||||||
it('parses actions', () => {
|
it('parses actions', () => {
|
||||||
// arrange
|
// arrange
|
||||||
const actions = [ getTestCategory('test1'), getTestCategory('test2') ];
|
const actions = [ getTestCategory('test1'), getTestCategory('test2') ];
|
||||||
const expected = [ parseCategory(actions[0]), parseCategory(actions[1]) ];
|
const compiler = new ScriptCompilerStub();
|
||||||
const sut: ApplicationYaml = {
|
const expected = [ parseCategory(actions[0], compiler), parseCategory(actions[1], compiler) ];
|
||||||
name: 'name',
|
const sut: ApplicationYaml = { actions, functions: undefined };
|
||||||
repositoryUrl: 'https://privacy.sexy',
|
|
||||||
actions,
|
|
||||||
};
|
|
||||||
// act
|
// act
|
||||||
const actual = parseApplication(sut).actions;
|
const actual = parseApplication(sut).actions;
|
||||||
// assert
|
// assert
|
||||||
@@ -98,18 +107,31 @@ describe('ApplicationParser', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
function getTestCategory(scriptName = 'testScript'): YamlCategory {
|
function getTestCategory(scriptPrefix = 'testScript'): YamlCategory {
|
||||||
return {
|
return {
|
||||||
category: 'category name',
|
category: 'category name',
|
||||||
children: [ getTestScript(scriptName) ],
|
children: [
|
||||||
|
getTestScript(`${scriptPrefix}-standard`, RecommendationLevel.Standard),
|
||||||
|
getTestScript(`${scriptPrefix}-strict`, RecommendationLevel.Strict),
|
||||||
|
],
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
function getTestScript(scriptName: string): YamlScript {
|
function getTestScript(scriptName: string, level: RecommendationLevel = RecommendationLevel.Standard): YamlScript {
|
||||||
return {
|
return {
|
||||||
name: scriptName,
|
name: scriptName,
|
||||||
code: 'script code',
|
code: 'script code',
|
||||||
revertCode: 'revert code',
|
revertCode: 'revert code',
|
||||||
recommend: true,
|
recommend: RecommendationLevel[level].toLowerCase(),
|
||||||
|
call: undefined,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
function getProcessEnvironmentStub(): NodeJS.ProcessEnv {
|
||||||
|
return {
|
||||||
|
VUE_APP_VERSION: 'stub-version',
|
||||||
|
VUE_APP_NAME: 'stub-name',
|
||||||
|
VUE_APP_REPOSITORY_URL: 'stub-repository-url',
|
||||||
|
VUE_APP_HOMEPAGE_URL: 'stub-homepage-url',
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -4,87 +4,150 @@ import { parseCategory } from '@/application/Parser/CategoryParser';
|
|||||||
import { YamlCategory, CategoryOrScript, YamlScript } from 'js-yaml-loader!./application.yaml';
|
import { YamlCategory, CategoryOrScript, YamlScript } from 'js-yaml-loader!./application.yaml';
|
||||||
import { parseScript } from '@/application/Parser/ScriptParser';
|
import { parseScript } from '@/application/Parser/ScriptParser';
|
||||||
import { parseDocUrls } from '@/application/Parser/DocumentationParser';
|
import { parseDocUrls } from '@/application/Parser/DocumentationParser';
|
||||||
|
import { ScriptCompilerStub } from '../../stubs/ScriptCompilerStub';
|
||||||
|
import { YamlScriptStub } from '../../stubs/YamlScriptStub';
|
||||||
|
|
||||||
describe('CategoryParser', () => {
|
describe('CategoryParser', () => {
|
||||||
describe('parseCategory', () => {
|
describe('parseCategory', () => {
|
||||||
|
describe('invalid category', () => {
|
||||||
it('throws when undefined', () => {
|
it('throws when undefined', () => {
|
||||||
expect(() => parseCategory(undefined)).to.throw('category is null or undefined');
|
// arrange
|
||||||
|
const expectedMessage = 'category is null or undefined';
|
||||||
|
const category = undefined;
|
||||||
|
const compiler = new ScriptCompilerStub();
|
||||||
|
// act
|
||||||
|
const act = () => parseCategory(category, compiler);
|
||||||
|
// assert
|
||||||
|
expect(act).to.throw(expectedMessage);
|
||||||
|
});
|
||||||
|
it('throws when children are empty', () => {
|
||||||
|
// arrange
|
||||||
|
const expectedMessage = 'category has no children';
|
||||||
|
const category: YamlCategory = {
|
||||||
|
category: 'test',
|
||||||
|
children: [],
|
||||||
|
};
|
||||||
|
const compiler = new ScriptCompilerStub();
|
||||||
|
// act
|
||||||
|
const act = () => parseCategory(category, compiler);
|
||||||
|
// assert
|
||||||
|
expect(act).to.throw(expectedMessage);
|
||||||
|
});
|
||||||
|
it('throws when children are undefined', () => {
|
||||||
|
// arrange
|
||||||
|
const expectedMessage = 'category has no children';
|
||||||
|
const category: YamlCategory = {
|
||||||
|
category: 'test',
|
||||||
|
children: undefined,
|
||||||
|
};
|
||||||
|
const compiler = new ScriptCompilerStub();
|
||||||
|
// act
|
||||||
|
const act = () => parseCategory(category, compiler);
|
||||||
|
// assert
|
||||||
|
expect(act).to.throw(expectedMessage);
|
||||||
|
});
|
||||||
|
it('throws when name is empty or undefined', () => {
|
||||||
|
// arrange
|
||||||
|
const expectedMessage = 'category has no name';
|
||||||
|
const invalidNames = ['', undefined];
|
||||||
|
invalidNames.forEach((invalidName) => {
|
||||||
|
const category: YamlCategory = {
|
||||||
|
category: invalidName,
|
||||||
|
children: getTestChildren(),
|
||||||
|
};
|
||||||
|
const compiler = new ScriptCompilerStub();
|
||||||
|
// act
|
||||||
|
const act = () => parseCategory(category, compiler);
|
||||||
|
// assert
|
||||||
|
expect(act).to.throw(expectedMessage);
|
||||||
|
});
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
it('throws when compiler is undefined', () => {
|
||||||
it('throws when children is empty', () => {
|
// arrange
|
||||||
const category: YamlCategory = {
|
const expectedError = 'undefined compiler';
|
||||||
category: 'test',
|
const compiler = undefined;
|
||||||
children: [],
|
const category = getValidCategory();
|
||||||
};
|
// act
|
||||||
expect(() => parseCategory(category)).to.throw('category has no children');
|
const act = () => parseCategory(category, compiler);
|
||||||
|
// assert
|
||||||
|
expect(act).to.throw(expectedError);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('throws when children is undefined', () => {
|
|
||||||
const category: YamlCategory = {
|
|
||||||
category: 'test',
|
|
||||||
children: undefined,
|
|
||||||
};
|
|
||||||
expect(() => parseCategory(category)).to.throw('category has no children');
|
|
||||||
});
|
|
||||||
|
|
||||||
it('throws when name is empty', () => {
|
|
||||||
const category: YamlCategory = {
|
|
||||||
category: '',
|
|
||||||
children: getTestChildren(),
|
|
||||||
};
|
|
||||||
expect(() => parseCategory(category)).to.throw('category has no name');
|
|
||||||
});
|
|
||||||
|
|
||||||
it('throws when name is undefined', () => {
|
|
||||||
const category: YamlCategory = {
|
|
||||||
category: undefined,
|
|
||||||
children: getTestChildren(),
|
|
||||||
};
|
|
||||||
expect(() => parseCategory(category)).to.throw('category has no name');
|
|
||||||
});
|
|
||||||
|
|
||||||
it('returns expected docs', () => {
|
it('returns expected docs', () => {
|
||||||
// arrange
|
// arrange
|
||||||
const url = 'https://privacy.sexy';
|
const url = 'https://privacy.sexy';
|
||||||
const expected = parseDocUrls({ docs: url });
|
const expected = parseDocUrls({ docs: url });
|
||||||
|
const compiler = new ScriptCompilerStub();
|
||||||
const category: YamlCategory = {
|
const category: YamlCategory = {
|
||||||
category: 'category name',
|
category: 'category name',
|
||||||
children: getTestChildren(),
|
children: getTestChildren(),
|
||||||
docs: url,
|
docs: url,
|
||||||
};
|
};
|
||||||
// act
|
// act
|
||||||
const actual = parseCategory(category).documentationUrls;
|
const actual = parseCategory(category, compiler).documentationUrls;
|
||||||
// assert
|
// assert
|
||||||
expect(actual).to.deep.equal(expected);
|
expect(actual).to.deep.equal(expected);
|
||||||
});
|
});
|
||||||
|
describe('parses expected subscript', () => {
|
||||||
it('returns expected scripts', () => {
|
it('single script with code', () => {
|
||||||
// arrange
|
// arrange
|
||||||
const script = getTestScript();
|
const script = YamlScriptStub.createWithCode();
|
||||||
const expected = [ parseScript(script) ];
|
const compiler = new ScriptCompilerStub();
|
||||||
const category: YamlCategory = {
|
const expected = [ parseScript(script, compiler) ];
|
||||||
category: 'category name',
|
const category: YamlCategory = {
|
||||||
children: [ script ],
|
category: 'category name',
|
||||||
};
|
children: [ script ],
|
||||||
// act
|
};
|
||||||
const actual = parseCategory(category).scripts;
|
// act
|
||||||
// assert
|
const actual = parseCategory(category, compiler).scripts;
|
||||||
expect(actual).to.deep.equal(expected);
|
// assert
|
||||||
|
expect(actual).to.deep.equal(expected);
|
||||||
|
});
|
||||||
|
it('single script with function call', () => {
|
||||||
|
// arrange
|
||||||
|
const script = YamlScriptStub.createWithCall();
|
||||||
|
const compiler = new ScriptCompilerStub()
|
||||||
|
.withCompileAbility(script);
|
||||||
|
const expected = [ parseScript(script, compiler) ];
|
||||||
|
const category: YamlCategory = {
|
||||||
|
category: 'category name',
|
||||||
|
children: [ script ],
|
||||||
|
};
|
||||||
|
// act
|
||||||
|
const actual = parseCategory(category, compiler).scripts;
|
||||||
|
// assert
|
||||||
|
expect(actual).to.deep.equal(expected);
|
||||||
|
});
|
||||||
|
it('multiple scripts with function call and code', () => {
|
||||||
|
// arrange
|
||||||
|
const callableScript = YamlScriptStub.createWithCall();
|
||||||
|
const scripts = [ callableScript, YamlScriptStub.createWithCode() ];
|
||||||
|
const compiler = new ScriptCompilerStub()
|
||||||
|
.withCompileAbility(callableScript);
|
||||||
|
const expected = scripts.map((script) => parseScript(script, compiler));
|
||||||
|
const category: YamlCategory = {
|
||||||
|
category: 'category name',
|
||||||
|
children: scripts,
|
||||||
|
};
|
||||||
|
// act
|
||||||
|
const actual = parseCategory(category, compiler).scripts;
|
||||||
|
// assert
|
||||||
|
expect(actual).to.deep.equal(expected);
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
it('returns expected subcategories', () => {
|
it('returns expected subcategories', () => {
|
||||||
// arrange
|
// arrange
|
||||||
const expected: YamlCategory[] = [ {
|
const expected: YamlCategory[] = [ {
|
||||||
category: 'test category',
|
category: 'test category',
|
||||||
children: [ getTestScript() ],
|
children: [ YamlScriptStub.createWithCode() ],
|
||||||
}];
|
}];
|
||||||
const category: YamlCategory = {
|
const category: YamlCategory = {
|
||||||
category: 'category name',
|
category: 'category name',
|
||||||
children: expected,
|
children: expected,
|
||||||
};
|
};
|
||||||
|
const compiler = new ScriptCompilerStub();
|
||||||
// act
|
// act
|
||||||
const actual = parseCategory(category).subCategories;
|
const actual = parseCategory(category, compiler).subCategories;
|
||||||
// assert
|
// assert
|
||||||
expect(actual).to.have.lengthOf(1);
|
expect(actual).to.have.lengthOf(1);
|
||||||
expect(actual[0].name).to.equal(expected[0].category);
|
expect(actual[0].name).to.equal(expected[0].category);
|
||||||
@@ -93,17 +156,16 @@ describe('CategoryParser', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
function getTestChildren(): ReadonlyArray<CategoryOrScript> {
|
function getValidCategory(): YamlCategory {
|
||||||
return [
|
|
||||||
getTestScript(),
|
|
||||||
];
|
|
||||||
}
|
|
||||||
|
|
||||||
function getTestScript(): YamlScript {
|
|
||||||
return {
|
return {
|
||||||
name: 'script name',
|
category: 'category name',
|
||||||
code: 'script code',
|
children: getTestChildren(),
|
||||||
revertCode: 'revert code',
|
docs: undefined,
|
||||||
recommend: true,
|
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function getTestChildren(): ReadonlyArray<CategoryOrScript> {
|
||||||
|
return [
|
||||||
|
YamlScriptStub.createWithCode(),
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user