Compare commits

..

84 Commits

Author SHA1 Message Date
Corentin Thomasset
d87091b23f chore(config): no index 2022-04-23 00:48:31 +02:00
Dante-DaCapo
b65b69dd97 fix: GitMemo displayed commands 2022-03-02 14:36:50 +01:00
Corentin Thomasset
5bd5d2c2f3 1.7.0 2020-07-23 23:25:52 +02:00
Corentin Thomasset
55dffa5635 feat: crontab generator
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-07-23 23:25:15 +02:00
Corentin Thomasset
baee5e6656 1.6.0 2020-07-22 13:05:46 +02:00
Corentin Thomasset
df5804c71d chore: changelog version
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-07-22 13:05:35 +02:00
dependabot[bot]
8e99a0a783 chore(deps): bump lodash from 4.17.15 to 4.17.19
Bumps [lodash](https://github.com/lodash/lodash) from 4.17.15 to 4.17.19.
- [Release notes](https://github.com/lodash/lodash/releases)
- [Commits](https://github.com/lodash/lodash/compare/4.17.15...4.17.19)

Signed-off-by: dependabot[bot] <support@github.com>
2020-07-22 13:04:27 +02:00
Corentin Thomasset
6e22b12494 feat: base64 generator
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-07-22 13:02:33 +02:00
Corentin THOMASSET
aaa154d0b8 chore: updated roadmap 2020-07-20 10:26:54 +02:00
Corentin Thomasset
37b28c8d21 docs: updated README.md
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-07-08 20:18:34 +02:00
Corentin Thomasset
06aac238b2 fix: array to bool in rules
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-07-08 20:03:22 +02:00
Corentin Thomasset
a282c526c1 feat: BIP39 generator
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-07-08 19:54:55 +02:00
Corentin Thomasset
19eafdbe37 doc: fixed link in CHANGELOG.md
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-07-08 19:54:33 +02:00
Corentin Thomasset
635552fbce 1.5.2 2020-07-08 13:37:39 +02:00
Corentin Thomasset
7f8404645d chore: version
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-07-08 13:37:35 +02:00
Corentin Thomasset
66c569f886 chore: pwa auto update
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-07-08 13:35:24 +02:00
Corentin Thomasset
f02a816eaa docs: updated README.md 2020-06-23 13:19:44 +02:00
Corentin Thomasset
f0d8a3ad26 feat: humans.txt
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-06-23 10:30:57 +02:00
Corentin Thomasset
f91a2a1343 Merge branch 'master' of github.com:CorentinTh/it-tools 2020-06-23 10:18:31 +02:00
Corentin Thomasset
845d106927 1.5.1 2020-06-23 10:16:45 +02:00
Corentin Thomasset
25bbf9b5f3 chore: CHANGELOG.md version
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-06-23 10:16:35 +02:00
Corentin Thomasset
853e24ca75 chore: updated CHANGELOG.md
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-06-23 10:15:58 +02:00
Corentin Thomasset
41fadc7334 1.5.0 2020-06-23 10:15:58 +02:00
Corentin Thomasset
19ab6d06c3 chore: updated version
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-06-23 10:15:58 +02:00
Corentin Thomasset
d92c46c60b feat: history mode
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-06-23 10:10:58 +02:00
Corentin Thomasset
ed176c7b8c 1.5.0 2020-06-23 09:40:58 +02:00
Corentin Thomasset
e152651a4b chore: updated version
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-06-23 09:40:50 +02:00
Corentin Thomasset
a20858dfb8 feat: added qr-code generator
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-06-23 00:01:14 +02:00
Corentin Thomasset
b75603f311 docs: create tool guideline
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-06-22 21:15:50 +02:00
Corentin Thomasset
d27c133a66 fix: CHANGELOG.md version
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-06-22 20:55:16 +02:00
Corentin Thomasset
bd64eb9dbe 1.4.0 2020-06-22 20:54:35 +02:00
Corentin Thomasset
ac54250e8a refactor: changed app title
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-06-22 20:53:53 +02:00
Corentin Thomasset
e0e7715ce6 feat: added canonical url
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-06-22 18:42:19 +02:00
Corentin Thomasset
aa3a424f10 docs: updated README.md
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-06-21 18:45:12 +02:00
Corentin Thomasset
03e073d4f0 feat: git memo
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-06-21 18:42:19 +02:00
Corentin Thomasset
cb0e3f91db feat: git memo
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-06-21 18:34:17 +02:00
Corentin Thomasset
c8c0dceb21 refactor: globalised code style
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-06-21 18:33:56 +02:00
Corentin Thomasset
ba5e07d063 refactor: home page layout
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-06-21 18:33:17 +02:00
Corentin Thomasset
9e94a985ca feat: ga monitor app version
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-06-21 18:32:44 +02:00
Corentin Thomasset
a14a7338f8 feat: added call to action
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-06-21 18:31:56 +02:00
Corentin Thomasset
b65472ce96 feat: condensed + colored sidenav
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-06-21 11:31:22 +02:00
Corentin Thomasset
87f269a792 Merge branch 'master' of github.com:CorentinTh/it-tools 2020-06-15 22:26:00 +02:00
Corentin Thomasset
27f957a098 fix: added version in CHANGELOG.md
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-06-15 22:22:37 +02:00
Corentin Thomasset
551738f559 1.3.0 2020-06-15 22:12:46 +02:00
Corentin Thomasset
8f7b7f628e docs: checked Lorem ipsum generator in roadmap
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-06-15 22:12:46 +02:00
Corentin Thomasset
46514cf71b feat: fixed icons width in sidebar
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-06-15 22:12:46 +02:00
Corentin Thomasset
78157b4cb5 feat: fixed icons width in sidebar
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-06-15 22:12:46 +02:00
Corentin Thomasset
bf4260f926 docs: checked markdown editor in roadmap
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-06-15 22:12:46 +02:00
Corentin Thomasset
9fc566e55a refactor: now using marked for changelog
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-06-15 22:12:46 +02:00
Corentin Thomasset
507d961cd8 feat: added MarkdownEditor.vue
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-06-15 22:12:46 +02:00
Corentin Thomasset
0631a7f750 chore: added stuff to roadmap
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-06-15 22:12:46 +02:00
Corentin Thomasset
99ffe08d01 chore: update CHANGELOG.md
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-06-15 22:12:46 +02:00
Corentin Thomasset
fcc1335f97 refactor: removed lazyLoad method
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-06-15 22:12:46 +02:00
Corentin THOMASSET
3988d33218 chore: updated changelog 2020-06-15 22:12:46 +02:00
Alexander Janke
ecb4b1bb7b Use vue-typecast
https://vuejs.org/v2/guide/forms.html#number
2020-06-15 22:12:46 +02:00
Corentin Thomasset
b63c4b5415 refactor: lazy-loading tools routes
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-06-15 22:12:46 +02:00
Corentin Thomasset
26bda68f41 fix: ordered contributors by contribution count
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-06-15 22:12:46 +02:00
Corentin Thomasset
f9f80b4e3a fix: replace next by version
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-06-15 22:12:46 +02:00
Corentin Thomasset
f50c5cb91b 1.3.0 2020-06-15 22:10:09 +02:00
Corentin Thomasset
da1da1d269 docs: checked Lorem ipsum generator in roadmap
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-06-15 22:10:08 +02:00
Corentin Thomasset
4129169e2b feat: fixed icons width in sidebar
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-06-15 22:10:08 +02:00
Corentin Thomasset
cffbfe3e39 feat: fixed icons width in sidebar
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-06-15 22:10:08 +02:00
Corentin Thomasset
583ce878d2 docs: checked markdown editor in roadmap
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-06-15 22:10:08 +02:00
Corentin Thomasset
c179f6a800 refactor: now using marked for changelog
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-06-15 22:10:08 +02:00
Corentin Thomasset
7944fd9f17 feat: added MarkdownEditor.vue
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-06-15 22:10:08 +02:00
Corentin Thomasset
84763e1d5f chore: added stuff to roadmap
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-06-15 22:10:08 +02:00
Corentin Thomasset
ab56a3b598 chore: update CHANGELOG.md
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-06-15 22:10:08 +02:00
Corentin Thomasset
b7d7454df5 refactor: removed lazyLoad method
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-06-15 22:10:08 +02:00
Corentin THOMASSET
b730c1d1b4 chore: updated changelog 2020-06-15 22:10:08 +02:00
Alexander Janke
f6da2a44e5 Use vue-typecast
https://vuejs.org/v2/guide/forms.html#number
2020-06-15 22:10:08 +02:00
Corentin Thomasset
89713c4986 refactor: lazy-loading tools routes
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-06-15 22:10:08 +02:00
Corentin Thomasset
3401fa4cf1 fix: ordered contributors by contribution count
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-06-15 22:10:08 +02:00
Corentin Thomasset
2a7abceba1 fix: replace next by version
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-06-15 22:10:08 +02:00
Corentin Thomasset
cd9a2318fd fix: added parseFloat in isInt
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-06-09 08:22:26 +02:00
Corentin Thomasset
0ebe60e21e 1.2.1 2020-06-09 08:22:26 +02:00
Corentin Thomasset
634147f0b3 chore: update CHANGELOG.md
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-06-09 08:22:26 +02:00
dotvirus
318691837e Use Number.isInteger 2020-06-09 08:22:26 +02:00
Corentin Thomasset
17f3e7de35 refactor: proper validation rules
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-06-09 08:22:26 +02:00
Corentin Thomasset
aa4fb5ffff fix: added quantity validation rules
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-06-09 08:22:26 +02:00
Corentin Thomasset
e51a37844a 1.2.0 2020-06-08 19:15:52 +02:00
Corentin Thomasset
6d8db0a5d5 chore: updated CHANGELOG.md
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-06-08 19:15:52 +02:00
Corentin Thomasset
84e727a43a feat: can generate multiple uuids
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-06-08 19:15:52 +02:00
Corentin Thomasset
6f7c399823 refactor: removed 404 page from tools roadmap
Signed-off-by: Corentin Thomasset <corentin.thomasset74@gmail.com>
2020-06-08 19:15:52 +02:00
dependabot[bot]
3777eb941d chore(deps): bump websocket-extensions from 0.1.3 to 0.1.4
Bumps [websocket-extensions](https://github.com/faye/websocket-extensions-node) from 0.1.3 to 0.1.4.
- [Release notes](https://github.com/faye/websocket-extensions-node/releases)
- [Changelog](https://github.com/faye/websocket-extensions-node/blob/master/CHANGELOG.md)
- [Commits](https://github.com/faye/websocket-extensions-node/compare/0.1.3...0.1.4)

Signed-off-by: dependabot[bot] <support@github.com>
2020-06-08 19:15:52 +02:00
31 changed files with 1408 additions and 120 deletions

View File

@@ -3,6 +3,39 @@ All notable changes to this project will be documented in this file.
The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/),
and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).
## Next
- [feat] [Crontab friendly generator](https://it-tools.tech/crontab-generator)
## 1.6.0
- [feat] [BIP39 generator](https://it-tools.tech/bip39-generator)
- [feat] [Base 64 converter](https://it-tools.tech/base64-string-converter)
## 1.5.2
- [feat] [humans.txt](https://it-tools.tech/humans.txt)
- [feat] pwa auto update on new changes
## 1.5.1
- [feat] switched back to history mode (no more '#' in url)
## 1.5.0
- [feat] added [qr-code generator](https://it-tools.tech/qrcode-generator)
## 1.4.0
- [ui] condensed + colored sidenav
- [feat] added [git memo](https://it-tools.tech/git-memo)
- [refactor] changed app title
## 1.3.0
- [fix] [GithubContributors] ordered contributors by contribution count
- [refactor] used vue-typecasting for number inputs
- [feat] lazy loading tools routes
- [feat] added [markdown editor](https://it-tools.tech/markdown-editor)
- [feat] added [lorem ipsum generator](https://it-tools.tech/lorem-ipsum-generator)
## 1.2.1
- [fix] [UuidGenerator] added quantity validation rules
- [refactor] better isInt checker
## 1.2.0
- [feat] [UuidGenerator] can generate multiple uuids
@@ -18,4 +51,4 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
- [fix] remove history move (incompatible with vercel.com)
## 1.0.0
- First release
- First release

View File

@@ -2,7 +2,7 @@
Aggregated set of useful tools that every developer may need once in a while. Available [here](https://it-tools.tech).
## Functionality/roadmap
## Functionalities roadmap
Here is an unordered list of the current functionalities, and some that may come.
- [x] Token generator
@@ -15,15 +15,23 @@ Here is an unordered list of the current functionalities, and some that may come
- [x] Url encoder
- [x] Base 64 generator
- [x] Text information
- [ ] Lorem ipsum text generator
- [x] Markdown editor
- [x] Lorem ipsum text generator
- [x] Git memo (cheat sheet)
- [x] QR code generator
- [x] Bip39 pass-phrase generator
- [x] Base 64 string converter
- [x] Crontab friendly generator
- [ ] CSS memo (cheat sheet)
- [ ] REGEX memo (cheat sheet) + tester?
- [ ] Minify/un-minify
- [ ] Image exif editor/remover
- [ ] QR code generator
- [ ] Bip39 pass-phrase generator
- [ ] Crontab friendly generator
- [ ] Image format converter?
- [ ] Image cropper
- [ ] Image resizer
- [ ] HTTP client (w/ axios)
- [ ] HTTP client (w/ axios + cors proxy)
- [ ] Math expression evaluator
- [ ] Math expression graph
You have an idea of a tool? Submit a feature request!
@@ -46,10 +54,33 @@ npm run lint
## Contribute
**Pull requests are welcome !** Feel free to contribute.
### Add a tool
To add a tool you just have to create a vue component in [src/routes/tools](./src/routes/tools), example:
```vue
<template>
<v-card class="single-card">
<v-card-title>My component</v-card-title>
<v-card-text>Lorem ipsum dolor sit amet, consectetur adipisicing elit.</v-card-text>
</v-card>
</template>
<script>
export default {
name: "My component"
}
</script>
<style scoped lang="less">
</style>
```
Then, update the file [router.js](./src/router.js) specifying info of the component.
Use [fontawesome 5](https://fontawesome.com/icons?d=gallery&m=free) for icons.
## Credits
Coded with ❤️ by [Corentin Thomasset](//corentin-thomasset.fr).
This project is continuously deployed using [vercel.com](https://vercel.com).
## License
This project is under the [MIT license](LICENSE).
This project is under the [MIT license](LICENSE).

78
package-lock.json generated
View File

@@ -1,6 +1,6 @@
{
"name": "it-tools",
"version": "1.2.0",
"version": "1.7.0",
"lockfileVersion": 1,
"requires": true,
"dependencies": {
@@ -2483,6 +2483,24 @@
"file-uri-to-path": "1.0.0"
}
},
"bip39": {
"version": "3.0.2",
"resolved": "https://registry.npmjs.org/bip39/-/bip39-3.0.2.tgz",
"integrity": "sha512-J4E1r2N0tUylTKt07ibXvhpT2c5pyAFgvuA5q1H9uDy6dEGpjV8jmymh3MTYJDLCNbIVClSB9FbND49I6N24MQ==",
"requires": {
"@types/node": "11.11.6",
"create-hash": "^1.1.0",
"pbkdf2": "^3.0.9",
"randombytes": "^2.0.1"
},
"dependencies": {
"@types/node": {
"version": "11.11.6",
"resolved": "https://registry.npmjs.org/@types/node/-/node-11.11.6.tgz",
"integrity": "sha512-Exw4yUWMBXM3X+8oqzJNRqZSwUAaS4+7NdvHqQuFi/d+synz++xmX3QIf+BFqneW8N31R8Ky+sikfZUXq07ggQ=="
}
}
},
"bluebird": {
"version": "3.7.2",
"resolved": "https://registry.npmjs.org/bluebird/-/bluebird-3.7.2.tgz",
@@ -3023,7 +3041,6 @@
"version": "1.0.4",
"resolved": "https://registry.npmjs.org/cipher-base/-/cipher-base-1.0.4.tgz",
"integrity": "sha512-Kkht5ye6ZGmwv40uUDZztayT2ThLQGfnj/T71N/XzeZeo3nf8foyW7zGTsPYkEya3m5f3cAypH+qe7YOrM1U2Q==",
"dev": true,
"requires": {
"inherits": "^2.0.1",
"safe-buffer": "^5.0.1"
@@ -3139,6 +3156,12 @@
"integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==",
"dev": true
},
"highlight.js": {
"version": "9.18.1",
"resolved": "https://registry.npmjs.org/highlight.js/-/highlight.js-9.18.1.tgz",
"integrity": "sha512-OrVKYz70LHsnCgmbXctv/bfuvntIKDz177h0Co37DQ5jamGZLVmoCVMtjMtNZY3X9DrCcKfklHPNeA0uPZhSJg==",
"dev": true
},
"supports-color": {
"version": "7.1.0",
"resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.1.0.tgz",
@@ -3676,7 +3699,6 @@
"version": "1.2.0",
"resolved": "https://registry.npmjs.org/create-hash/-/create-hash-1.2.0.tgz",
"integrity": "sha512-z00bCGNHDG8mHAkP7CtT1qVu+bFQUPjYq/4Iv3C3kWjTFV10zIjfSoeqXo9Asws8gwSHDGj/hl2u4OGIjapeCg==",
"dev": true,
"requires": {
"cipher-base": "^1.0.1",
"inherits": "^2.0.1",
@@ -3689,7 +3711,6 @@
"version": "1.1.7",
"resolved": "https://registry.npmjs.org/create-hmac/-/create-hmac-1.1.7.tgz",
"integrity": "sha512-MJG9liiZ+ogc4TzUwuvbER1JRdgvUFSB5+VR/g5h82fGaIRWMWddtKBHi7/sVhfjQZ6SehlyhvQYrcYkaUIpLg==",
"dev": true,
"requires": {
"cipher-base": "^1.0.3",
"create-hash": "^1.1.0",
@@ -3699,6 +3720,16 @@
"sha.js": "^2.4.8"
}
},
"cron-validator": {
"version": "1.1.1",
"resolved": "https://registry.npmjs.org/cron-validator/-/cron-validator-1.1.1.tgz",
"integrity": "sha512-vfZb05w/wezuwPZBDvdIBmJp2BvuJExHeyKRa5oBqD2ZDXR61hb3QgPc/3ZhBEQJlAy8Jlnn5XC/JCT3IDqxwg=="
},
"cronstrue": {
"version": "1.96.0",
"resolved": "https://registry.npmjs.org/cronstrue/-/cronstrue-1.96.0.tgz",
"integrity": "sha512-kNuuN0AXqspm+jlpwiecFSFBrG5dYBaL5EaW5MEjGAh4IFT/jbKwKCNhzWb8N5NulKQUHEDN7w98GjoAm6PmMg=="
},
"cross-spawn": {
"version": "6.0.5",
"resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-6.0.5.tgz",
@@ -4373,6 +4404,11 @@
"domelementtype": "1"
}
},
"dompurify": {
"version": "2.0.11",
"resolved": "https://registry.npmjs.org/dompurify/-/dompurify-2.0.11.tgz",
"integrity": "sha512-qVoGPjIW9IqxRij7klDQQ2j6nSe4UNWANBhZNLnsS7ScTtLb+3YdxkRY8brNTpkUiTtcXsCJO+jS0UCDfenLuA=="
},
"domutils": {
"version": "1.7.0",
"resolved": "https://registry.npmjs.org/domutils/-/domutils-1.7.0.tgz",
@@ -6214,7 +6250,6 @@
"version": "3.0.4",
"resolved": "https://registry.npmjs.org/hash-base/-/hash-base-3.0.4.tgz",
"integrity": "sha1-X8hoaEfs1zSZQDMZprCj8/auSRg=",
"dev": true,
"requires": {
"inherits": "^2.0.1",
"safe-buffer": "^5.0.1"
@@ -6253,12 +6288,6 @@
"integrity": "sha512-l9sfDFsuqtOqKDsQdqrMRk0U85RZc0RtOR9yPI7mRVOa4FsR/BVnZ0shmQRM96Ji99kYZP/7hn1cedc1+ApsTQ==",
"dev": true
},
"highlight.js": {
"version": "9.18.1",
"resolved": "https://registry.npmjs.org/highlight.js/-/highlight.js-9.18.1.tgz",
"integrity": "sha512-OrVKYz70LHsnCgmbXctv/bfuvntIKDz177h0Co37DQ5jamGZLVmoCVMtjMtNZY3X9DrCcKfklHPNeA0uPZhSJg==",
"dev": true
},
"hmac-drbg": {
"version": "1.0.1",
"resolved": "https://registry.npmjs.org/hmac-drbg/-/hmac-drbg-1.0.1.tgz",
@@ -6628,8 +6657,7 @@
"inherits": {
"version": "2.0.4",
"resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz",
"integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==",
"dev": true
"integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ=="
},
"inquirer": {
"version": "7.1.0",
@@ -7913,9 +7941,9 @@
}
},
"lodash": {
"version": "4.17.15",
"resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.15.tgz",
"integrity": "sha512-8xOcRHvCjnocdS5cpwXQXVzmmh5e5+saE2QGoeQmbKmRS6J3VQppPOIt0MnmE+4xlZoumy0GPG0D0MVIQbNA1A==",
"version": "4.17.19",
"resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.19.tgz",
"integrity": "sha512-JNvd8XER9GQX0v2qJgsaN/mzFCNA5BRe/j8JN9d+tWyGLSodKQHKFicdwNYzWwI3wjRnaKPsGj1XkBjx/F96DQ==",
"dev": true
},
"lodash._reinterpolate": {
@@ -8103,11 +8131,15 @@
"object-visit": "^1.0.0"
}
},
"marked": {
"version": "1.1.0",
"resolved": "https://registry.npmjs.org/marked/-/marked-1.1.0.tgz",
"integrity": "sha512-EkE7RW6KcXfMHy2PA7Jg0YJE1l8UPEZE8k45tylzmZM30/r1M1MUXWQfJlrSbsTeh7m/XTwHbWUENvAJZpp1YA=="
},
"md5.js": {
"version": "1.3.5",
"resolved": "https://registry.npmjs.org/md5.js/-/md5.js-1.3.5.tgz",
"integrity": "sha512-xitP+WxNPcTTOgnTJcrhM0xvdPepipPSf3I8EIpGKeFLjt3PlJLIDG3u8EX53ZIubkb+5U2+3rELYpEhHhzdkg==",
"dev": true,
"requires": {
"hash-base": "^3.0.0",
"inherits": "^2.0.1",
@@ -9084,7 +9116,6 @@
"version": "3.0.17",
"resolved": "https://registry.npmjs.org/pbkdf2/-/pbkdf2-3.0.17.tgz",
"integrity": "sha512-U/il5MsrZp7mGg3mSQfn742na2T+1/vHDCG5/iTI3X9MKUuYUZVLQhyRsg06mCgDBTd57TxzgZt7P+fYfjRLtA==",
"dev": true,
"requires": {
"create-hash": "^1.1.2",
"create-hmac": "^1.1.4",
@@ -9980,6 +10011,11 @@
"integrity": "sha1-fjL3W0E4EpHQRhHxvxQQmsAGUdc=",
"dev": true
},
"qrcode.vue": {
"version": "1.7.0",
"resolved": "https://registry.npmjs.org/qrcode.vue/-/qrcode.vue-1.7.0.tgz",
"integrity": "sha512-R7t6Y3fDDtcU7L4rtqwGUDP9xD64gJhIwpfjhRCTKmBoYF6SS49PIJHRJ048cse6OI7iwTwgyy2C46N9Ygoc6g=="
},
"qs": {
"version": "6.5.2",
"resolved": "https://registry.npmjs.org/qs/-/qs-6.5.2.tgz",
@@ -10018,7 +10054,6 @@
"version": "2.1.0",
"resolved": "https://registry.npmjs.org/randombytes/-/randombytes-2.1.0.tgz",
"integrity": "sha512-vYl3iOX+4CKUWuxGi9Ukhie6fsqXqS9FE2Zaic4tNFD2N2QQaXOMFbuKK4QmDHC0JO6B1Zp41J0LpT0oR68amQ==",
"dev": true,
"requires": {
"safe-buffer": "^5.1.0"
}
@@ -10451,7 +10486,6 @@
"version": "2.0.2",
"resolved": "https://registry.npmjs.org/ripemd160/-/ripemd160-2.0.2.tgz",
"integrity": "sha512-ii4iagi25WusVoiC4B4lq7pbXfAp3D9v5CwfkY33vffw2+pkDjY1D8GaN7spsxvCSx8dkPqOZCEZyfxcmJG2IA==",
"dev": true,
"requires": {
"hash-base": "^3.0.0",
"inherits": "^2.0.1"
@@ -10498,8 +10532,7 @@
"safe-buffer": {
"version": "5.1.2",
"resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz",
"integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==",
"dev": true
"integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g=="
},
"safe-regex": {
"version": "1.1.0",
@@ -10760,7 +10793,6 @@
"version": "2.4.11",
"resolved": "https://registry.npmjs.org/sha.js/-/sha.js-2.4.11.tgz",
"integrity": "sha512-QMEp5B7cftE7APOjk5Y6xgrbWu+WkLVQwk8JNjZ8nKRciZaByEW6MubieAiToS7+dwvrjGhH8jRXz3MVd0AYqQ==",
"dev": true,
"requires": {
"inherits": "^2.0.1",
"safe-buffer": "^5.0.1"

View File

@@ -1,7 +1,7 @@
{
"name": "it-tools",
"description": "",
"version": "1.2.0",
"version": "1.7.0",
"private": true,
"scripts": {
"serve": "vue-cli-service serve",
@@ -10,9 +10,15 @@
},
"dependencies": {
"axios": "^0.19.2",
"bip39": "^3.0.2",
"color-convert": "^2.0.1",
"color-name": "^1.1.4",
"core-js": "^3.6.4",
"cron-validator": "^1.1.1",
"cronstrue": "^1.96.0",
"dompurify": "^2.0.11",
"marked": "^1.1.0",
"qrcode.vue": "^1.7.0",
"register-service-worker": "^1.7.1",
"roboto-fontface": "*",
"vue": "^2.6.11",

5
public/humans.txt Normal file
View File

@@ -0,0 +1,5 @@
/* TEAM */
Developer: Corentin Thomasset
Site: https://github.com/CorentinTh
Twitter: @cthmsst

View File

@@ -5,7 +5,8 @@
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width,initial-scale=1.0">
<link rel="icon" href="<%= BASE_URL %>favicon.ico">
<title><%= htmlWebpackPlugin.options.title %></title>
<title>IT Tools - Set of handy developer tools</title>
<link rel="canonical" href="https://it-tools.tech">
<meta itemprop="name" content="IT-Tools">
<meta property="og:title" content="IT-Tools">
<meta name="twitter:title" content="IT-Tools">
@@ -16,6 +17,8 @@
<meta itemprop="image" content="/img/banner.png">
<meta property="og:image" content="/img/banner.png">
<meta name="twitter:image" content="/img/banner.png">
<link rel="author" href="humans.txt" />
<meta name="robots" content="noindex">
</head>
<body>
<noscript>

View File

@@ -1,2 +1,2 @@
User-agent: *
Disallow:
Disallow: *

View File

@@ -1,23 +1,24 @@
<template>
<v-app id="inspire">
<vue-headful
:title="currentRoute ? `${currentRoute.text} - IT-Tools` : 'IT-Tools'"
:description="currentRoute ? currentRoute.description: 'Aggregated set of useful tools that every developer may need once in a while.'"
:keywords="currentRoute ? currentRoute.keywords: null"
image="/img/banner.png"
:title="currentRoute ? `${currentRoute.text} - IT Tools` : 'IT Tools - Set of handy developer tools'"
:description="currentRoute ? currentRoute.description: 'Aggregated set of useful tools that every developer may need once in a while.'"
:keywords="currentRoute ? currentRoute.keywords: null"
image="/img/banner.png"
/>
<v-navigation-drawer v-model="drawer" app clipped>
<template v-slot:prepend>
<SearchBar class="hidden-sm-and-up"/>
</template>
<SearchBar class="hidden-sm-and-up" />
<v-list dense>
<v-list dense id="navigation-list">
<div v-for="section in items" :key="section.title">
<v-subheader class="mt-4 pl-4">{{section.title}}</v-subheader>
<v-list-item v-for="item in section.child" :key="item.text" :to="item.path">
<v-list-item-action>
<v-icon>{{ item.icon }}</v-icon>
<v-icon style="width: 1.25em">{{ item.icon }}</v-icon>
</v-list-item-action>
<v-list-item-content>
<v-list-item-title>
@@ -99,7 +100,7 @@
appVersion: 'v' + process.env.APPLICATION_VERSION,
drawer: null,
items: toolsComponents,
currentRoute:{}
currentRoute: {}
}),
mounted() {
this.setTitle()
@@ -107,14 +108,14 @@
created() {
this.$vuetify.theme.dark = true
},
methods:{
setTitle(){
methods: {
setTitle() {
const path = this.$router.currentRoute.path;
this.currentRoute = toolsComponents.map(p => p.child).flat().find(p => p.path === path)
}
},
watch:{
'$route'(){
watch: {
'$route'() {
this.setTitle()
}
}
@@ -125,6 +126,51 @@
html {
overflow-y: auto !important;
}
code{
background-color: rgba(0, 0, 0, 0.15) !important;
box-shadow: none !important;
color: #9a9a9a !important;
font-weight: normal !important;
}
.pretty-scrollbar{
&::-webkit-scrollbar {
width: 5px!important;
height: 5px !important;
}
/* Track */
&::-webkit-scrollbar-track {
opacity: 0 !important;
}
/* Handle */
&::-webkit-scrollbar-thumb {
background: rgba(241, 241, 241, 0.10) !important;
border-radius: 10px;
}
/* Handle on hover */
&::-webkit-scrollbar-thumb:hover {
background: rgba(241, 241, 241, 0.20)!important;
}
}
#navigation-list{
div:first-child .v-subheader{
margin-top: 0 !important;
}
.v-list-item__action{
margin: 8px 25px 8px 0;
.v-icon{
color: #4CAF50 !important;
}
}
}
.v-navigation-drawer__content{
.pretty-scrollbar;
}
.single-card {
width: 100%;

View File

@@ -4,7 +4,9 @@
<p class="text-justify">
Welcome to <strong>IT-Tools</strong>! This wonderful website, originally created with by
<a href="//corentin-thomasset.fr">Corentin Thomasset</a>, aggregate a set of useful tools
that every developer may need once in a while.
that every developer may need once in a while. And don't forget to add <strong>IT-Tools</strong> to your
shortcut bar (press <code>{{ isMacOS ? 'Cmd' : 'Ctrl' }} +
D</code>).
</p>
@@ -30,7 +32,10 @@
<script>
export default {
name: "Abstract"
name: "Abstract",
data: () => ({
isMacOS: navigator.platform.toUpperCase().indexOf('MAC') >= 0
})
}
</script>

View File

@@ -0,0 +1,58 @@
<template>
<v-text-field v-model="color" hide-details class="ma-0 pa-0" outlined :label="label" v-on:input="$emit('input', color)">
<template v-slot:append>
<v-menu v-model="menu" top nudge-bottom="101" nudge-left="16" :close-on-content-click="false">
<template v-slot:activator="{ on }">
<div :style="swatchStyle" v-on="on" />
</template>
<v-card>
<v-card-text class="pa-0">
<v-color-picker v-model="color" flat v-on:input="$emit('input', color)"/>
</v-card-text>
</v-card>
</v-menu>
</template>
</v-text-field>
</template>
<script>
// From: https://codepen.io/JamieCurnow/pen/KKPjraK
export default {
name: "ColorInput",
props:{
value: {
type: String,
default: '#FFFFFF'
},
label:String
},
data: () => ({
menu: false,
color:''
}),
mounted() {
this.color = this.value
},
computed: {
swatchStyle() {
const { color, menu } = this
return {
backgroundColor: color,
cursor: 'pointer',
height: '30px',
width: '30px',
borderRadius: menu ? '50%' : '4px',
transition: 'border-radius 200ms ease-in-out'
}
}
}
}
</script>
<style scoped lang="less">
::v-deep .v-input__append-inner{
margin-top: 13px;
}
</style>

View File

@@ -0,0 +1,43 @@
<template>
<div class="copyable-code-content" @click="copy($slots.default[0].text)">
<pre class="pretty-scrollbar"><slot></slot></pre>
<v-icon>far fa-copy</v-icon>
</div>
</template>
<script>
import {copyable} from "../mixins/copyable.mixin";
export default {
name: "CopyableCodeContent",
mixins: [copyable]
}
</script>
<style lang="less" scoped>
.copyable-code-content {
cursor: pointer;
background-color: rgba(0, 0, 0, 0.1);
border-radius: 4px;
padding: 8px 15px;
display: flex;
flex-direction: row;
align-items: center;
justify-content: center;
pre {
flex: 1;
overflow-x: auto;
}
.v-icon {
opacity: 0;
}
&:hover {
.v-icon {
opacity: 1;
}
}
}
</style>

View File

@@ -36,7 +36,7 @@
axios
.get(url)
.then(({data}) => {
this.contributors = data.sort((a, b) => a.contributions - b.contributions)
this.contributors = data.sort((a, b) => b.contributions - a.contributions)
this.loading = false
})
.catch(() => this.hasError = true)

View File

@@ -0,0 +1,73 @@
<template>
<div class="memo-viewer" v-bind:style="{ columns: `auto ${colWidth}` }">
<div class="section" v-for="(group,i) in memo" :key="i">
<h2>{{group.section}}</h2>
<div class="tip" v-for="(tips,i) in group.child" :key="i">
<v-card>
<v-card-text>
<template v-for="tip in (Array.isArray(tips) ? tips : [tips])">
<p :key="tip.text">{{tip.text}}</p>
<CopyableCodeContent class="code" :key="tip.code">{{tip.code}}</CopyableCodeContent>
</template>
</v-card-text>
</v-card>
</div>
</div>
</div>
</template>
<script>
import CopyableCodeContent from "./CopyableCodeContent";
export default {
name: "MemoViewer",
props: {
memo: Array,
colWidth: {
type: String,
default: '400px'
}
},
components: {
CopyableCodeContent
}
}
</script>
<style lang="less" scoped>
.memo-viewer {
column-gap: 30px;
column-rule: 1px solid #37373961;
column-fill: auto;
}
.section {
break-inside: avoid-column;
display: inline-block;
margin-bottom: 20px;
width: 100%;
h2 {
margin: 25px 0 15px;
display: inline-block;
}
.tip {
margin: 20px 0;
.v-card{
background-color: rgba(47, 46, 46, 0.44);
}
p {
margin-bottom: 10px;
&:not(:first-child){
margin-top: 16px;
}
}
}
}
</style>

View File

@@ -0,0 +1,10 @@
import {copyToClipboard} from "../utils/helpers";
export const copyable = {
methods: {
copy(text, toastText = 'Copied to clipboard !'){
copyToClipboard(text);
this.$toast.success(toastText)
}
}
}

View File

@@ -5,6 +5,12 @@ import router from "../router";
if(process.env.VUE_APP_GANALYTICS){
Vue.use(VueAnalytics, {
id: process.env.VUE_APP_GANALYTICS,
router
router,
set:[
{
field: 'dimension1',
value: process.env.APPLICATION_VERSION
}
]
})
}

View File

@@ -20,7 +20,8 @@ if (process.env.NODE_ENV === 'production') {
console.log('New content is downloading.')
},
updated () {
console.log('New content is available; please refresh.')
console.log('New content is available; hard refresh.');
window.location.reload(true);
},
offline () {
console.log('No internet connection found. App is running in offline mode.')

View File

@@ -1,20 +1,9 @@
import Vue from 'vue'
import VueRouter from 'vue-router'
import Home from './routes/Home.vue'
import TokenGenerator from "./routes/tools/TokenGenerator";
import Hash from "./routes/tools/Hash";
import DateConverter from "./routes/tools/DateConverter";
import UrlEncoder from "./routes/tools/UrlEncoder";
import FileToBase64 from "./routes/tools/FileToBase64";
import TextCypher from "./routes/tools/TextCypher";
import TextStats from "./routes/tools/TextStats";
import BaseConverter from "./routes/tools/BaseConverter";
import UuidGenerator from "./routes/tools/UuidGenerator";
import ColorConverter from "./routes/tools/ColorConverter";
Vue.use(VueRouter)
const toolsComponents = [
{
title: 'Crypto',
@@ -23,7 +12,7 @@ const toolsComponents = [
icon: 'fa-key',
text: 'Token generator',
path: '/token-generator',
component: TokenGenerator,
component: () => import('./routes/tools/TokenGenerator'),
keywords: ['token', 'random', 'string', 'alphanumeric'],
description: 'Generate random tokens.'
},
@@ -31,14 +20,14 @@ const toolsComponents = [
icon: 'fa-fingerprint',
text: 'Uuid generator',
path: '/uuid-generator',
component: UuidGenerator,
component: () => import('./routes/tools/UuidGenerator'),
keywords: ['token', 'v4', 'string', 'alphanumeric']
},
{
icon: 'fa-font',
text: 'Hash text',
path: '/hash',
component: Hash,
component: () => import('./routes/tools/Hash'),
keywords: ['md5', 'sha1', 'sha256', 'sha224', 'sha512', 'sha384', 'sha3', 'ripemd160', 'random']
},
@@ -46,9 +35,16 @@ const toolsComponents = [
icon: 'fa-lock',
text: 'Cypher/uncypher text',
path: '/cypher',
component: TextCypher,
component: () => import('./routes/tools/TextCypher'),
keywords: ['aes', 'tripledes', 'rabbit', 'rabbitlegacy', 'rc4']
},
{
icon: 'fa-file-word',
text: 'BIP39 passphrase generator',
path: '/bip39-generator',
component: () => import('./routes/tools/BIP39Generator'),
keywords: []
},
],
},
{
@@ -58,21 +54,21 @@ const toolsComponents = [
icon: 'fa-calendar',
text: 'Date/Time converter',
path: '/date-converter',
component: DateConverter,
component: () => import('./routes/tools/DateConverter'),
keywords: ['locale', 'format', 'iso 8601', 'utc', 'timestamp', 'unix', 'year', 'month', 'day', 'hours', 'minutes', 'seconds']
},
{
icon: 'fa-exchange-alt',
text: 'Base converter',
path: '/base-converter',
component: BaseConverter,
component: () => import('./routes/tools/BaseConverter'),
keywords: ['binary', 'hexadecimal', 'decimal']
},
{
icon: 'fa-palette',
text: 'Color picker/converter',
path: '/color-picker-converter',
component: ColorConverter,
component: () => import('./routes/tools/ColorConverter'),
keywords: ['rgb', 'rgba', 'hexadecimal', 'hsla', 'red', 'green', 'blue', 'alpha']
},
],
@@ -84,28 +80,79 @@ const toolsComponents = [
icon: 'fa-link',
text: 'URL encode/decode',
path: '/url-encoder',
component: UrlEncoder,
component: () => import('./routes/tools/UrlEncoder'),
keywords: ['%20']
},
{
icon: 'fa-file-export',
text: 'File to Base64',
path: '/file-to-base64',
component: FileToBase64
component: () => import('./routes/tools/FileToBase64')
},
{
icon: 'fa-file-alt',
text: 'Base64 string converter',
path: '/base64-string-converter',
component: () => import('./routes/tools/StringToBase64')
}
],
},
{
title: 'Text',
child: [
{
icon: 'fa-align-left',
text: 'Text stats',
path: '/text-stats',
component: () => import('./routes/tools/TextStats'),
keywords: ['word', 'count', 'size', 'bytes', 'length']
},
{
icon: 'fab fa-markdown',
text: 'Markdown editor',
path: '/markdown-editor',
component: () => import('./routes/tools/MarkdownEditor'),
keywords: ['text', 'html', 'markdown']
},
{
icon: 'fa-align-justify',
text: 'Lorem ipsum generator',
path: '/lorem-ipsum-generator',
component: () => import('./routes/tools/LoremIpsumGenerator'),
keywords: ['text', 'dolor', 'sit', 'placeholder', 'fill', 'dummy']
}
],
},
{
title: 'Memos',
child: [
{
text: 'Git memo',
path: '/git-memo',
icon: 'fa-code-branch',
component: () => import('./routes/tools/GitMemo'),
keywords: ['git', 'push', 'rebase', 'merge', 'tag', 'commit', 'checkout']
}
]
},
{
title: 'Miscellaneous',
child: [
{
icon: 'fa-align-left\n',
text: 'Text stats',
path: '/text-stats',
component: TextStats,
keywords: ['word', 'count', 'size', 'bytes', 'length']
text: 'Crontab generator',
path: '/crontab-generator',
icon: 'fa-clock',
component: () => import('./routes/tools/CrontabGenerator'),
keywords: ['Cron', 'Schedule']
},
],
{
text: 'QR Code generator',
path: '/qrcode-generator',
icon: 'fa-qrcode',
component: () => import('./routes/tools/QRCodeGenerator'),
keywords: []
}
]
}
];
@@ -131,6 +178,7 @@ const routes = [
const router = new VueRouter({
base: process.env.BASE_URL,
mode: 'history',
routes
});

View File

@@ -11,7 +11,7 @@
</v-col>
</v-row>
<v-row justify="center" >
<v-row justify="center">
<v-col cols="12" md="5" sm="12">
<v-card>
<v-card-title>Contributors</v-card-title>
@@ -22,12 +22,7 @@
<v-card>
<v-card-title>Changelog</v-card-title>
<v-card-text>
<div v-for="(section, i) in changelog" :key="i">
<h2>{{section.title}}</h2>
<ul>
<li v-for="(log, i) in section.logs" :key="i"> {{log}}</li>
</ul>
<br>
<div v-html="changelog" class="changelog">
</div>
</v-card-text>
</v-card>
@@ -39,6 +34,8 @@
import Abstract from "../components/Abstract";
import GithubContributors from "../components/GithubContributors";
import changelog from "../../CHANGELOG.md"
import marked from 'marked'
import DOMPurify from 'dompurify';
export default {
name: "About",
@@ -46,28 +43,21 @@
changelog: []
}),
mounted() {
this.changelog = ('##' + changelog.replace(/^(.*?)##/s, ''))
.split('\n')
.filter(v => v !== '')
.reduce((sections, v) => {
v = v.trim();
if(v.startsWith('##')){
sections.push({
title: v.replace(/^##/, '').trim(),
logs: []
})
}else {
sections.slice(-1)[0].logs.push(v.replace(/^-/, '').trim())
}
return sections
}, []);
console.log(this.changelog);
this.changelog = DOMPurify.sanitize(marked('##' + changelog.replace(/^(.*?)##/s, '')));
},
components: {
Abstract,
GithubContributors
},
}
</script>
</script>
<style scoped lang="less">
::v-deep {
.changelog {
h2 {
margin-top: 10px;
}
}
}
</style>

View File

@@ -9,9 +9,7 @@
</v-card-text>
</v-card>
</v-col>
</v-row>
<v-row justify="center" align="center">
<v-col cols="12" lg="8" md="12">
<v-col cols="12" lg="5" md="12">
<v-card class="card-auto">
<v-card-text>
<div class="card-wrapper ">
@@ -56,7 +54,7 @@
flex-wrap: wrap;
div {
flex: 0 1 20%;
flex: 0 1 33%;
@media only screen and (max-width: 800px) {
flex: 0 1 33%;

View File

@@ -0,0 +1,99 @@
<template>
<v-card class="single-card">
<v-card-title>BIP39 passphrase generator</v-card-title>
<v-card-text>
<v-select
outlined
label="Language"
@change="languageChanged"
:items="languageList"
v-model="language"
/>
<v-text-field
outlined
v-model="entropy"
label="Entropy"
append-icon="fa-clipboard"
@click:append="copy(entropy)"
:rules="rules.entropy"
ref="entropy"
/>
<v-text-field
outlined
v-model="passphrase"
label="Passphrase"
append-icon="fa-clipboard"
@click:append="copy(passphrase)"
:rules="rules.passphrase"
ref="passphrase"
/>
<div class="text-center">
<v-btn @click="refresh">refresh</v-btn>
</div>
</v-card-text>
</v-card>
</template>
<script>
import * as bip39 from "bip39";
import {copyable} from "../../mixins/copyable.mixin";
const shuffle = (str) => str.split('').sort(() => 0.5 - Math.random()).join('');
const getRandomBuffer = () => {
return Buffer.from(shuffle('0123456789abcdef'.repeat(16)).substring(0, 32), 'hex');
}
export default {
name: 'BIP39Generator',
mixins: [copyable],
data: () => ({
buffer: getRandomBuffer(),
languageList: Object.keys(bip39.wordlists).filter(k => !k.match(/[A-Z]{2}/)).map(k => ({text: k.split('_').map(k => k.charAt(0).toUpperCase() + k.slice(1)).join(' '), value:k})),
language: 'english',
rules: {
passphrase: [
v => (!!v && bip39.validateMnemonic(v)) || 'Invalid mnemonic.'
],
entropy: [
v => (!!v && !!v.match(/[0-9a-fA-F]{32}/)) || 'Invalid entropy.'
]
}
}),
methods:{
refresh(){
this.buffer = getRandomBuffer();
},
languageChanged(){
bip39.setDefaultWordlist(this.language);
this.passphrase = bip39.entropyToMnemonic(this.buffer)
}
},
computed: {
entropy: {
get() {
return this.buffer.toString('hex')
},
set(value) {
if(this.$refs.entropy.validate()) {
this.buffer = Buffer.from(value, 'hex')
}
}
},
passphrase: {
get() {
return bip39.entropyToMnemonic(this.buffer)
},
set(value) {
if(this.$refs.passphrase.validate()){
this.buffer = Buffer.from(bip39.mnemonicToEntropy(value), 'hex')
}
}
}
}
}
</script>
<style scoped>
</style>

View File

@@ -8,7 +8,7 @@
label="Input base"
outlined
type="number"
v-model="inputBase"
v-model.number="inputBase"
ref="inputBase"
hide-details="auto"
:rules="baseRules"
@@ -33,7 +33,7 @@
label="Output base"
outlined
type="number"
v-model="outputBase"
v-model.number="outputBase"
ref="outputBase"
:rules="baseRules"
/>

View File

@@ -0,0 +1,221 @@
<template>
<v-row justify="center" align="center">
<v-col cols="12" xl="5" lg="5" md="12">
<v-card>
<v-card-title>Crontab generator</v-card-title>
<v-card-text>
<div class="result">{{cronString}}</div>
<v-text-field
ref="cron"
class="cron-wrapper"
outlined
v-model="cron"
label="Cron"
append-icon="fa-copy"
@click:append="copy(cron)"
:rules="[isCronValid]"
hide-details="auto"
/>
<v-checkbox
hide-details
label="Verbose"
v-model="cronstrueConfig.verbose"
/>
<v-checkbox
hide-details
label="Use 24 hour time format"
v-model="cronstrueConfig.use24HourTimeFormat"
/>
<v-checkbox
hide-details
label="Day of the week start a index 0"
v-model="cronstrueConfig.dayOfWeekStartIndexZero"
/>
</v-card-text>
</v-card>
</v-col>
<v-col cols="12" xl="6" lg="7" md="12">
<v-card>
<v-card-title>Crontab helper</v-card-title>
<v-card-text>
<div class="text-center">
<pre>
day of week (0 - 6, sunday=0) OR sun,mon ...
| month (1 - 12) OR jan,feb,mar,apr ...
| | day of month (1 - 31)
| | | hour (0 - 23)
| | | | minute (0 - 59)
| | | | |
* * * * * command
</pre>
</div>
<br>
<v-simple-table dense>
<template v-slot:default>
<thead>
<tr>
<th class="text-left">Symbol</th>
<th class="text-left">Meaning</th>
<th class="text-left">Example</th>
<th class="text-left">Equivalent</th>
</tr>
</thead>
<tbody>
<tr>
<td>*</td>
<td>Any value</td>
<td>
<pre>* * * *</pre>
</td>
<td>Every minute</td>
</tr>
<tr>
<td>-</td>
<td>Range of values</td>
<td>
<pre>1-10 * * *</pre>
</td>
<td>Minutes 1 through 10</td>
</tr>
<tr>
<td>,</td>
<td>List of values</td>
<td>
<pre>1,10 * * *</pre>
</td>
<td>At minutes 1 and 10</td>
</tr>
<tr>
<td>/</td>
<td>Step values</td>
<td>
<pre>*/10 * * *</pre>
</td>
<td>Every 10 minutes</td>
</tr>
<tr>
<td>@yearly</td>
<td>Once every year at midnight of 1 January</td>
<td>
<pre>@yearly</pre>
</td>
<td>0 0 1 1 *</td>
</tr>
<tr>
<td>@annually</td>
<td>Same as @yearly</td>
<td>
<pre>@annually</pre>
</td>
<td>0 0 1 1 *</td>
</tr>
<tr>
<td>@monthly</td>
<td>Once a month at midnight on the first day</td>
<td>
<pre>@monthly</pre>
</td>
<td>0 0 1 * *</td>
</tr>
<tr>
<td>@weekly</td>
<td>Once a week at midnight on Sunday morning</td>
<td>
<pre>@weekly</pre>
</td>
<td>0 0 * * 0</td>
</tr>
<tr>
<td>@daily</td>
<td>Once a day at midnight</td>
<td>
<pre>@daily</pre>
</td>
<td>0 0 * * *</td>
</tr>
<tr>
<td>@midnight</td>
<td>Same as @daily</td>
<td>
<pre>@midnight</pre>
</td>
<td>0 0 * * *</td>
</tr>
<tr>
<td>@hourly</td>
<td>Once an hour at the beginning of the hour</td>
<td>
<pre>@hourly</pre>
</td>
<td>0 * * * *</td>
</tr>
<tr>
<td>@reboot</td>
<td>Run at startup</td>
<td></td>
<td></td>
</tr>
</tbody>
</template>
</v-simple-table>
</v-card-text>
</v-card>
</v-col>
</v-row>
</template>
<script>
import {copyable} from "../../mixins/copyable.mixin";
import cronstrue from 'cronstrue';
import * as CronValidator from 'cron-validator'
export default {
name: "CrontabGenerator",
mixins: [copyable],
data: () => ({
cron: '* * * * *',
cronstrueConfig: {
verbose: true,
dayOfWeekStartIndexZero: true,
use24HourTimeFormat: true,
throwExceptionOnParseError: true
}
}),
methods: {
isCronValid(v) {
return CronValidator.isValidCron(v, {allowBlankDay: true, alias: true});
}
},
computed: {
cronString() {
if (this.isCronValid(this.cron)) {
return cronstrue.toString(this.cron, this.cronstrueConfig)
} else {
return ' '
}
}
}
}
</script>
<style scoped lang="less">
::v-deep .cron-wrapper input {
text-align: center;
font-size: 22px;
font-family: Consolas, monospace;
}
.result {
text-align: center;
font-size: 18px;
margin-bottom: 22px;
}
.text-center{
pre{
display: inline-block;
text-align: left;
}
}
</style>

View File

@@ -0,0 +1,150 @@
<template>
<v-row>
<v-col cols="12" xl="12">
<v-card>
<v-card-title>Git Memo</v-card-title>
<v-card-text>
<MemoViewer :memo="tips"/>
</v-card-text>
</v-card>
</v-col>
</v-row>
</template>
<script>
import MemoViewer from "../../components/MemoViewer";
export default {
name: "GitMemo",
data: () => ({
tips: [
{
section: 'Get started',
child: [
{
text: 'Create a git repo',
code: 'git init'
},
{
text: 'Clone an existing repository',
code: 'git clone [repo url]'
},
{
text: 'Add current files to next commit',
code: 'git add .'
},
{
text: 'Commit tracked files changes',
code: 'git commit -am "[commit message]"'
},
{
text: 'List files that has changed',
code: 'git status'
},
{
text: 'List changes in tracked files',
code: 'git diff'
}
]
},
{
section: 'Basic configuration',
child: [
{
text: 'Set the name that will be associated to every operation',
code: 'git config --global user.name "[nom]"'
},
{
text: 'Set the email address that will be associated to every operation',
code: 'git config --global user.email "[email]"'
},
{
text: 'Tell git to always push tags',
code: 'git config --global push.followTags true'
}
]
},
{
section: 'I\'ve made a mistake',
child: [
{
text: 'Change last commit message',
code: 'git commit --amend'
},
{
text: 'Undo most recent commit and keep changes',
code: 'git reset HEAD~1'
},
{
text: 'Undo most recent commit and get rid of changes',
code: 'git reset HEAD~1 --hard'
},
{
text: 'Reset branch to remote state',
code: 'git fetch origin\ngit reset --hard origin/[branch-name]'
}
]
},
{
section: 'Setup SSH',
child: [
[
{
text: '1). Generate an SSH key.',
code: 'ssh-keygen -t rsa -b 4096 -C "[email]"'
},
{
text: '2). Start the ssh-agent in the background.',
code: 'eval "$(ssh-agent -s)"'
},
{
text: '3). Add your SSH private key to the ssh-agent.',
code: 'ssh-add ~/.ssh/id_rsa'
},
{
text: '4). Add your SSH public key to your git server (for github: Settings -> SSH and GPG keys)',
code: 'cat ~/.ssh/id_rsa.pub'
},
{
text: '5). (Optional) Testing your SSH connection',
code: 'ssh -T git@github.com'
},
]
]
},
{
section: 'Merge and rebase',
child: [
{
text: 'Merge a branch into the current',
code: 'git merge [branch]'
},
{
text: 'Abort merge (conflicts)',
code: 'git merge --abort'
},
{
text: 'Continue merge after resolving conflicts',
code: 'git merge --continue'
},
{
text: 'Rebase a branch into the current',
code: 'git rebase [branch]'
},
{
text: 'Continue rebase after resolving conflicts',
code: 'git rebase --continue'
},
]
},
]
}),
components: {
MemoViewer
}
}
</script>
<style lang="less" scoped>
</style>

View File

@@ -0,0 +1,94 @@
<template>
<v-row justify="center" align="center" class="lorem-ipsum-generator">
<v-col cols="12" xl="5" lg="6" md="12">
<v-card>
<v-card-title>Lorem ipsum generator</v-card-title>
<v-card-text>
<v-slider v-model="paragraphs" min="1" max="20" label="Paragraphs" thumb-label/>
<v-range-slider v-model="sentencePerParagraph" min="1" max="50" label="Sentences per paragraph"
thumb-label/>
<v-range-slider v-model="wordPerSentence" min="1" max="50" label="Words per sentence" thumb-label hide-details/>
<v-checkbox v-model="startWithLoremIpsum" label="Start with 'Lorem ipsum ...'" hide-details/>
<v-checkbox v-model="asHTML" label="As HTML" hide-details/>
</v-card-text>
</v-card>
</v-col>
<v-col cols="12" xl="5" lg="6" md="12">
<v-card>
<v-card-text>
<v-textarea outlined readonly hide-details="auto" v-model="loremIpsum" rows="15"
class="text-justify"></v-textarea>
<div class="text-center mt-4">
<v-btn depressed @click="copy()">Copy</v-btn>
</div>
</v-card-text>
</v-card>
</v-col>
</v-row>
</template>
<script>
import {copyToClipboard, randFromArray, randIntFromInterval} from "../../utils/helpers";
const vocabulary = ['a', 'ac', 'accumsan', 'ad', 'adipiscing', 'aenean', 'aliquam', 'aliquet', 'amet', 'ante', 'aptent', 'arcu', 'at', 'auctor', 'bibendum', 'blandit', 'class', 'commodo', 'condimentum', 'congue', 'consectetur', 'consequat', 'conubia', 'convallis', 'cras', 'cubilia', 'cum', 'curabitur', 'curae', 'dapibus', 'diam', 'dictum', 'dictumst', 'dignissim', 'dolor', 'donec', 'dui', 'duis', 'egestas', 'eget', 'eleifend', 'elementum', 'elit', 'enim', 'erat', 'eros', 'est', 'et', 'etiam', 'eu', 'euismod', 'facilisi', 'faucibus', 'felis', 'fermentum', 'feugiat', 'fringilla', 'fusce', 'gravida', 'habitant', 'habitasse', 'hac', 'hendrerit', 'himenaeos', 'iaculis', 'id', 'imperdiet', 'in', 'inceptos', 'integer', 'interdum', 'ipsum', 'justo', 'lacinia', 'lacus', 'laoreet', 'lectus', 'leo', 'ligula', 'litora', 'lobortis', 'lorem', 'luctus', 'maecenas', 'magna', 'magnis', 'malesuada', 'massa', 'mattis', 'mauris', 'metus', 'mi', 'molestie', 'mollis', 'montes', 'morbi', 'mus', 'nam', 'nascetur', 'natoque', 'nec', 'neque', 'netus', 'nisi', 'nisl', 'non', 'nostra', 'nulla', 'nullam', 'nunc', 'odio', 'orci', 'ornare', 'parturient', 'pellentesque', 'penatibus', 'per', 'pharetra', 'phasellus', 'placerat', 'platea', 'porta', 'porttitor', 'posuere', 'potenti', 'praesent', 'pretium', 'primis', 'proin', 'pulvinar', 'purus', 'quam', 'quis', 'quisque', 'rhoncus', 'ridiculus', 'risus', 'rutrum', 'sagittis', 'sapien', 'scelerisque', 'sed', 'sem', 'semper', 'senectus', 'sit', 'sociis', 'sociosqu', 'sodales', 'sollicitudin', 'suscipit', 'suspendisse', 'taciti', 'tellus', 'tempor', 'tempus', 'tincidunt', 'torquent', 'tortor', 'turpis', 'ullamcorper', 'ultrices', 'ultricies', 'urna', 'varius', 'vehicula', 'vel', 'velit', 'venenatis', 'vestibulum', 'vitae', 'vivamus', 'viverra', 'volutpat', 'vulputate'];
const firstSentence = 'Lorem ipsum dolor sit amet, consectetur adipiscing elit.';
const generateSentence = (length) => {
let sentence = Array.from({length}).map(() => randFromArray(vocabulary)).join(' ')
sentence = sentence.charAt(0).toUpperCase() + sentence.slice(1) + '.'
return sentence
}
export default {
name: "LoremIpsumGenerator",
data: () => ({
paragraphs: 1,
sentencePerParagraph: [3, 8],
wordPerSentence: [8, 15],
startWithLoremIpsum: true,
asHTML: false
}),
methods:{
copy(){
copyToClipboard(this.loremIpsum)
this.$toast.success('Copied to clipboard.')
}
},
computed: {
loremIpsum: function () {
const lorem = Array
.from({length: this.paragraphs})
.map(() => {
const length = randIntFromInterval(...this.sentencePerParagraph);
return Array.from({length}).map(() => {
const wordCount = randIntFromInterval(...this.wordPerSentence);
return generateSentence(wordCount);
})
});
if (this.startWithLoremIpsum) {
lorem[0][0] = firstSentence
}
let result;
if(this.asHTML){
result = `<p>${lorem.map(s => s.join(' ')).join('</p>\n\n<p>')}</p>`
}else{
result = lorem.map(s => s.join(' ')).join('\n\n')
}
return result;
}
}
}
</script>
<style scoped lang="less">
::v-deep {
.v-label{
min-width: 200px !important;
}
}
</style>

View File

@@ -0,0 +1,71 @@
<template>
<v-row justify="center" align="center">
<v-col cols="12" xl="5" lg="6" md="12">
<v-card>
<v-card-text>
<v-textarea v-model="markdown" auto-grow outlined label="Markdown editor"/>
<div class="text-center">
<v-btn @click="copy(markdown)">copy markdown</v-btn>
</div>
</v-card-text>
</v-card>
</v-col>
<v-col cols="12" xl="5" lg="6" md="12">
<v-card>
<v-card-text >
<div class="preview" v-html="html"></div>
<div class="text-center">
<v-divider />
<br>
<v-btn @click="copy(html)">copy html</v-btn>
</div>
</v-card-text>
</v-card>
</v-col>
</v-row>
</template>
<script>
// import {debounce} from "../../utils/helpers";
import marked from 'marked'
import DOMPurify from 'dompurify';
import {copyToClipboard} from "../../utils/helpers";
export default {
name: "MarkdownEditor",
data: () => ({
markdown: '# Hello, World!\nLorem ipsum **dolor** sit *amet*, consectetur adipisicing elit. A aspernatur commodi consequuntur distinctio dolore doloribus eaque earum est ipsum nobis numquam pariatur perspiciatis quasi quis, sed, sunt tempore tenetur, veniam!\n',
}),
methods: {
copy(text){
copyToClipboard(text)
this.$toast.success('Copied to clipboard.')
}
},
computed: {
html() {
return DOMPurify.sanitize(marked(this.markdown))
}
}
}
</script>
<style scoped lang="less">
::v-deep {
.preview {
padding: 20px;
h1{
margin-bottom: 15px;
}
pre {
width: 100%;
code {
width: 100% !important;
padding: 10px;
}
}
}
}
</style>

View File

@@ -0,0 +1,137 @@
<template>
<v-card class="single-card">
<v-card-title>QR-code generator</v-card-title>
<v-card-text>
<v-row justify="center" align="center">
<v-col cols="12" lg="6" sm="12">
<v-text-field
outlined
v-model="value"
label="Data"
:rules="rules.value"
/>
<v-slider v-model="size" min="100" max="1920" label="Size (preview will not change): " thumb-label/>
<v-select
outlined
v-model="level"
:items="levels"
label="Error resistance"
/>
<v-row>
<v-col cols="12" md="6" sm="12">
<ColorInput v-model="fgcolor" label="Foreground color"/>
</v-col>
<v-col cols="12" md="6" sm="12">
<ColorInput v-model="bgcolor" label="Background color"/>
</v-col>
</v-row>
</v-col>
<v-col cols="12" lg="6" sm="12" class="text-center">
<qrcode-vue
:value="input"
:size="size"
:level="level"
:background="bgcolor"
:foreground="fgcolor"
render-as="svg"
class-name="qrcode-wrapper"
/>
</v-col>
</v-row>
<div class="text-center mt-3 mb-sm-2">
<v-btn @click="download('png')" class="mr-1" color="primary">download as png</v-btn>
<v-btn @click="download('svg')" class="ml-1" color="primary">download as svg</v-btn>
</div>
</v-card-text>
</v-card>
</template>
<script>
import QrcodeVue from 'qrcode.vue'
import colors from "color-name";
import ColorInput from "../../components/ColorInput";
import {downloadBase64File} from "../../utils/helpers";
export default {
name: "QRCodeGenerator",
data: () => ({
value: 'https://it-tools.tech',
size: 300,
level: 'M',
bgcolor: '#ffffff',
fgcolor: '#000000',
levels: [
{text: 'Low', value: 'L'},
{text: 'Medium', value: 'M'},
{text: 'Quartile', value: 'Q'},
{text: 'High', value: 'H'}
],
rules: {
value: [
v => v.length > 0 || 'Value is needed'
],
color: [
v => {
v = v.trim()
const isFFFFFF = /^#(?:[0-9a-fA-F]{6})$/.test(v);
const isFFF = /^#(?:[0-9a-fA-F]{3})$/.test(v);
const isRGB = /^rgb\((\d+),\s*(\d+),\s*(\d+)\)$/.test(v);
const isHSL = /^hsl\((\d+),\s*(\d+)%,\s*(\d+)%\)$/.test(v);
const isKeyword = v in colors;
const isTransparent = v === 'transparent';
return isFFFFFF || isFFF || isKeyword || isTransparent || isRGB || isHSL || 'Incorrect color.'
}
]
}
}),
methods: {
download(type) {
const svgEl = this.$el.querySelector('.qrcode-wrapper svg');
const svgString = new XMLSerializer().serializeToString(svgEl);
const svgUrl = `data:image/svg+xml;base64,${btoa(svgString)}`;
if (type === 'png') {
const canvas = document.createElement("canvas");
canvas.width = this.size;
canvas.height = this.size;
const ctx = canvas.getContext("2d");
const image = new Image();
image.onload = function () {
ctx.drawImage(image, 0, 0);
const result = canvas.toDataURL();
downloadBase64File(result, 'qr-code');
};
image.src = svgUrl;
} else {
downloadBase64File(svgUrl, 'qr-code');
}
}
},
computed: {
input() {
return this.value
}
},
components: {
QrcodeVue,
ColorInput
}
}
</script>
<style scoped lang="less">
::v-deep .qrcode-wrapper {
& > * {
width: 300px !important;
height: 300px !important;
}
}
</style>

View File

@@ -0,0 +1,66 @@
<template>
<v-card class="single-card">
<v-card-title>Base64 string converter</v-card-title>
<v-card-text>
<v-textarea
outlined
v-model="clear"
label="Clear text"
></v-textarea>
<v-textarea
outlined
v-model="base64"
label="Base64 text"
:rules="rules.base64"
ref="base64"
></v-textarea>
<div class="text-center">
<v-btn class="mr-1" depressed @click="copy(clear)">Copy clear</v-btn>
<v-btn class="ml-1" depressed @click="copy(base64)">Copy base64</v-btn>
</div>
</v-card-text>
</v-card>
</template>
<script>
import {copyable} from "../../mixins/copyable.mixin";
export default {
name: "StringToBase64",
mixins: [copyable],
data() {
return {
clear: 'Lorem ipsum dolor sit amet.',
rules:{
base64: [
v => {
try{
return btoa(atob(v)) === v || 'Input is not base64.'
}catch (e) {
return 'Input is not base64.'
}
}
]
}
}
},
computed: {
base64: {
get(){
return btoa(this.clear)
},
set(value){
if(this.$refs.base64.validate()){
this.clear = atob(value)
}
}
}
}
}
</script>
<style scoped>
</style>

View File

@@ -3,8 +3,18 @@
<v-card-title>Uuid v4 generator</v-card-title>
<v-card-text>
<v-text-field outlined v-model="quantity" type="number" label="Quantity" dense class="quantity"/>
<v-textarea outlined v-model="token" class="centered-input" :rows="quantity <= 10 ? quantity : 10" readonly/>
<v-text-field
outlined
v-model.number="quantity"
ref="quantity"
type="number"
label="Quantity"
dense
class="quantity"
:rules="rules.quantity"
/>
<v-textarea outlined v-model="token" class="centered-input" :rows="quantity <= 10 ? quantity : 10"
readonly/>
<div class="text-center">
<v-btn @click="refreshBool = !refreshBool" depressed class="mr-4">Refresh</v-btn>
@@ -15,7 +25,7 @@
</template>
<script>
import {copyToClipboard} from "../../utils/helpers";
import {copyToClipboard, isInt} from "../../utils/helpers";
const noop = () => {
};
@@ -26,8 +36,19 @@
name: "UuidGenerator",
data: () => ({
refreshBool: true,
quantity: 1
quantity: 1,
rules: {
quantity: [
v => !!v || 'Quantity is required',
v => (v > 0 && v <= 50 ) || 'Quantity should be > 0 and <= 50',
v => isInt(v) || 'Quantity should be an integer'
]
},
isMounted:false
}),
mounted() {
this.isMounted = true;
},
methods: {
copyToken() {
copyToClipboard(this.token);
@@ -36,21 +57,25 @@
},
computed: {
token() {
if (this.refreshBool) noop(); // To force recomputation
if (this.isMounted && this.$refs.quantity.validate()) {
if (this.refreshBool) noop(); // To force recomputation
return Array.from({length: this.quantity}, generateUuid).join('\n');
return Array.from({length: this.quantity}, generateUuid).join('\n');
} else {
return '';
}
}
}
}
</script>
<style scoped lang="less">
.quantity{
.quantity {
width: 100px;
margin: auto;
text-align: center;
::v-deep input{
::v-deep input {
text-align: center;
}
}

View File

@@ -25,12 +25,36 @@ const formatBytes = (bytes, decimals = 2) => {
}
const isInt = (value) => {
return !isNaN(value) && ((x) => (x | 0) === x)(parseFloat(value))
return Number.isInteger(value);
}
const debounce = (callback, delay = 300) => {
let timer;
return function(...args) {
clearTimeout(timer);
timer = setTimeout(() => callback(...args), delay);
}
}
const randFromArray = (array) => array[Math.floor(Math.random() * array.length)];
const randIntFromInterval = (min, max) => Math.floor(Math.random() * (max - min) + min)
const downloadBase64File = (dataUrl, name = 'file') => {
const a = document.createElement("a");
a.href = dataUrl;
a.download = name;
a.click();
}
export {
copyToClipboard,
fileIsImage,
formatBytes,
isInt
isInt,
debounce,
randFromArray,
randIntFromInterval,
downloadBase64File
}

7
vercel.json Normal file
View File

@@ -0,0 +1,7 @@
{
"version": 2,
"routes": [
{ "handle": "filesystem" },
{ "src": "/.*", "dest": "/index.html" }
]
}

View File

@@ -19,5 +19,11 @@ module.exports = {
})
]
}
},
pwa: {
workboxOptions: {
skipWaiting: true,
clientsClaim: true,
}
}
}