Compare commits
545 Commits
v1.0.6
...
serverProf
Author | SHA1 | Date | |
---|---|---|---|
|
95b12f109a | ||
|
f40ab912eb | ||
|
700301ffbb | ||
|
d92894697b | ||
|
47569d2ffa | ||
|
d3b18bbcd2 | ||
|
eeedc531a3 | ||
|
59a2b834a6 | ||
|
1cfeaf77c1 | ||
|
1757d17661 | ||
|
9485d2457a | ||
|
3fd2fc1d61 | ||
|
49bc6b8fd6 | ||
|
c165725297 | ||
|
29fbe3701a | ||
|
0b7c0e9587 | ||
|
d88524e8cf | ||
|
d6efd99849 | ||
|
fe6be987fd | ||
|
d688075c0a | ||
|
c752be45b2 | ||
|
07c1f5eed1 | ||
|
4df01b1e62 | ||
|
ebe10d3fad | ||
|
eca4af829f | ||
|
60458cdf1f | ||
|
714d87241c | ||
|
f628aa7a70 | ||
|
0f0282551d | ||
|
0335e1ca59 | ||
|
817f0f7473 | ||
|
98a03c8862 | ||
|
72ce7a5ad1 | ||
|
e699ea63c7 | ||
|
97e1e9eb7a | ||
|
4c4036546a | ||
|
d582e61ec3 | ||
|
ccd2ce8baf | ||
|
ede507e80c | ||
|
ffdf63563b | ||
|
55b755b2df | ||
|
ca439e9e9a | ||
|
e02fcce3dc | ||
|
3e732646e5 | ||
|
d5b3b51050 | ||
|
725fb27e54 | ||
|
243381fc91 | ||
|
54cbdfdad0 | ||
|
fe80b8cc85 | ||
|
742f5cf556 | ||
|
3b3da90c44 | ||
|
c79e065d09 | ||
|
3b8b43c7e0 | ||
|
6e7996659f | ||
|
abdf4ebb05 | ||
|
fa124d8877 | ||
|
135da2a5f3 | ||
|
c96a1a9998 | ||
|
8b6c8bc0ec | ||
|
1a62249da6 | ||
|
21318850b1 | ||
|
885ad134b3 | ||
|
3e7d4e2623 | ||
|
d3691f74c4 | ||
|
268f3a1840 | ||
|
d6c43986fd | ||
|
bb7deeb09c | ||
|
0407be9847 | ||
|
645749b5ae | ||
|
2e002107a6 | ||
|
cc07518a34 | ||
|
ea64b33e24 | ||
|
1a92d3ff8d | ||
|
45bb1af011 | ||
|
39ad88f433 | ||
|
8cf4d2a2c0 | ||
|
fe5e041db8 | ||
|
d18681c197 | ||
|
c024db1bc4 | ||
|
d8a0db8bee | ||
|
f62efa5aa7 | ||
|
1d77ab0ade | ||
|
9268cf3ffb | ||
|
208371c471 | ||
|
c69c6f8cb7 | ||
|
f2c6fcaa3b | ||
|
abf62f28db | ||
|
8620a1d86d | ||
|
198b35ffdc | ||
|
b4d0d95731 | ||
|
f785aa1473 | ||
|
d56e6560e5 | ||
|
a7e74ee4d5 | ||
|
1340f023a3 | ||
|
2bf0c324d7 | ||
|
f621cdb50b | ||
|
9717001783 | ||
|
065ab75627 | ||
|
8aea72c1be | ||
|
bea7a1711e | ||
|
e52ae62441 | ||
|
7cd1d4c60f | ||
|
2a318e390e | ||
|
7c7723bfb1 | ||
|
2db0e71e5b | ||
|
cde8074f44 | ||
|
8b1630bc99 | ||
|
bf34b2ae43 | ||
|
cb5f23d9b5 | ||
|
cd2cbfa0ef | ||
|
232e340fab | ||
|
8027daa2b0 | ||
|
0f7b9f588e | ||
|
93482ac2a5 | ||
|
994c3b3c92 | ||
|
c696c186e8 | ||
|
30d5e2108f | ||
|
1eabd1b701 | ||
|
1cbf2b43e1 | ||
|
4c197d5d51 | ||
|
f89027f46a | ||
|
07a0ebb1d2 | ||
|
f09b44b0d5 | ||
|
b607eebcb7 | ||
|
0936ca2985 | ||
|
13bde79ec8 | ||
|
b592defaaf | ||
|
73354973a3 | ||
|
e12c0e546c | ||
|
088a8bd1b6 | ||
|
51adb26d01 | ||
|
cb980a1cad | ||
|
69b10c1f07 | ||
|
8e9ba7c7ee | ||
|
12e3c9234d | ||
|
1d8dcef394 | ||
|
4fe2845234 | ||
|
5e71ed286e | ||
|
5edbd2391d | ||
|
8472c3823e | ||
|
2103e52115 | ||
|
afbfb641e8 | ||
|
d7ac418e05 | ||
|
214c101740 | ||
|
5a0e501829 | ||
|
92113da7c0 | ||
|
96f30a5359 | ||
|
ceb1f15188 | ||
|
626eb3613e | ||
|
3020fcc9bb | ||
|
bc0de3926c | ||
|
9820b79dfe | ||
|
ab811470fc | ||
|
e4162e7bd5 | ||
|
7e8397a4da | ||
|
555cf64080 | ||
|
2039e10fd5 | ||
|
e8d90d2b45 | ||
|
55af40ee74 | ||
|
a1fabcdf0a | ||
|
eaeb60308e | ||
|
662c0227eb | ||
|
543fdf4943 | ||
|
1225383723 | ||
|
07a9adbce2 | ||
|
42d8211871 | ||
|
ab3e993274 | ||
|
386dfe363a | ||
|
a4191c9f6c | ||
|
f1349a2787 | ||
|
3680c26f72 | ||
|
683c92f904 | ||
|
3410ed024f | ||
|
dbad10984a | ||
|
55543d8640 | ||
|
263fbc377e | ||
|
c9c0ab5aca | ||
|
7b2bf08b8f | ||
|
43011825af | ||
|
4abcea61f8 | ||
|
cba810cab5 | ||
|
5938c7d67c | ||
|
99d8b8b75f | ||
|
503d49d295 | ||
|
137b79d95b | ||
|
3c02d6e1b4 | ||
|
a2a33ca62d | ||
|
d8cd557fb2 | ||
|
7568bbaed0 | ||
|
9023d45d9e | ||
|
bee70390a9 | ||
|
3e3d05fc26 | ||
|
6300198a54 | ||
|
458c7ed4c5 | ||
|
d888a0a291 | ||
|
a94787a9f3 | ||
|
368d2bcdbb | ||
|
bc46bfa467 | ||
|
dab48288a8 | ||
|
9aef97c771 | ||
|
9d62dec6b9 | ||
|
6bf6583e7d | ||
|
5219fb700f | ||
|
184c03b28e | ||
|
ec091a7959 | ||
|
89a6c575c9 | ||
|
60325c6aa5 | ||
|
c2a1c4cbf6 | ||
|
1d6b78f6c6 | ||
|
341151a718 | ||
|
f6fd7cf37a | ||
|
d53476a32a | ||
|
fc943b7778 | ||
|
3f2bcd2cab | ||
|
235000cf41 | ||
|
263884cbd8 | ||
|
bb83c0b672 | ||
|
2815509c00 | ||
|
53ff2532f4 | ||
|
64b38348d4 | ||
|
9c1b3a9afd | ||
|
caf77a3d7f | ||
|
7a27de8927 | ||
|
1bc0678422 | ||
|
cd53cf38fe | ||
|
f13f9e80a9 | ||
|
c062f9bdeb | ||
|
f2ef96a420 | ||
|
16365d3ea1 | ||
|
1ec28a345b | ||
|
2fdc00b11e | ||
|
3da112680d | ||
|
1d93162036 | ||
|
7dcd32e838 | ||
|
ade31f993b | ||
|
3c7496ac6d | ||
|
63387a48ee | ||
|
3bb68467bb | ||
|
2b337eace1 | ||
|
5c5b009c41 | ||
|
0c54b1fa1d | ||
|
393f76749a | ||
|
1fe7f3c297 | ||
|
622e8dc3e0 | ||
|
a8b6aea26f | ||
|
e50c2fafa5 | ||
|
6e3cafce42 | ||
|
4d0a064425 | ||
|
d1ad6c47a7 | ||
|
d5c35055f3 | ||
|
cb385d1b28 | ||
|
195f1a032f | ||
|
dfda9e7f84 | ||
|
0d5e2d0696 | ||
|
2834bed518 | ||
|
0b2c3c834a | ||
|
3a54a24c70 | ||
|
244d10dc9e | ||
|
c25bc0ff4b | ||
|
22334663cf | ||
|
8813f81bde | ||
|
84371ed456 | ||
|
8f61119b99 | ||
|
0a89d09727 | ||
|
474932161f | ||
|
bd95a25f4c | ||
|
6a57ecc22b | ||
|
0d665b7e0b | ||
|
d94b28fb8e | ||
|
bc1d8694d4 | ||
|
7bc1362cbd | ||
|
4dce836ff7 | ||
|
9f534c0685 | ||
|
c62d05e1b3 | ||
|
6a1cb133cd | ||
|
c6196dff81 | ||
|
acdb390302 | ||
|
08d88b326d | ||
|
a73858d131 | ||
|
b0caa6f4db | ||
|
168d4b4cd9 | ||
|
06cee75a56 | ||
|
d589d22a0b | ||
|
4c13521a30 | ||
|
043381963b | ||
|
5b485806ea | ||
|
29c994648b | ||
|
a35b417194 | ||
|
070aa343ef | ||
|
b95c5c6619 | ||
|
bf795c49df | ||
|
a2e03084b0 | ||
|
ec72b4c91d | ||
|
d70d7c7b49 | ||
|
e7d0fc258d | ||
|
7b13b9a53e | ||
|
1b2cb52dac | ||
|
0fe0fecba2 | ||
|
c1fca76f94 | ||
|
0cc3901e4e | ||
|
f8ace5b53a | ||
|
c0954a1844 | ||
|
6548163d3e | ||
|
24f161d6e9 | ||
|
7d00b6a842 | ||
|
5f5d4b8961 | ||
|
5be86f9bd1 | ||
|
dfc3f05834 | ||
|
63fc354d48 | ||
|
c6f0c84935 | ||
|
a8d017811d | ||
|
8dd70f5d1a | ||
|
8be6c6e3ce | ||
|
7e96b5dcfb | ||
|
99a7d78e9b | ||
|
e70d00d008 | ||
|
c0ac6a4b86 | ||
|
29749e93c7 | ||
|
993c6be219 | ||
|
e2e1cf2bfd | ||
|
59e3c2c609 | ||
|
43d7ca4c30 | ||
|
5305447f44 | ||
|
76e74b3e40 | ||
|
e767da4b08 | ||
|
e4f3f57a28 | ||
|
72f6dd84ee | ||
|
9c929a4d98 | ||
|
dac9cad873 | ||
|
6fd5c7874f | ||
|
a56dfe269c | ||
|
7d55a81bac | ||
|
ce64631310 | ||
|
1caaa78490 | ||
|
d35654b887 | ||
|
ca5d24385f | ||
|
cb3bd4b881 | ||
|
ff3589d157 | ||
|
7a98f1dfcb | ||
|
9e6d3459e3 | ||
|
ea30ca418f | ||
|
1f7ec93a24 | ||
|
336c7bdd5e | ||
|
88ad4f1b05 | ||
|
f75f887861 | ||
|
96f640da67 | ||
|
e8809fc57b | ||
|
ca91ef4e39 | ||
|
db7fc3769b | ||
|
6c719f5ee9 | ||
|
c6fd8cae16 | ||
|
1adbf9e41a | ||
|
aee6bed48c | ||
|
c8817e805f | ||
|
c6f0d0763c | ||
|
3bd3012aa9 | ||
|
694a693a8e | ||
|
ed827c2d81 | ||
|
71849cac9a | ||
|
e34da54271 | ||
|
cfe41ef656 | ||
|
4d836524c1 | ||
|
edc96387f5 | ||
|
358eb6ad8e | ||
|
c997cb4958 | ||
|
83dab24fb9 | ||
|
8a305d2d11 | ||
|
7eb12f0fb7 | ||
|
0a3dc5c6e8 | ||
|
b21516d44e | ||
|
65f7cf9503 | ||
|
40a7aa5079 | ||
|
c4a3d25d37 | ||
|
613fa9a57b | ||
|
08822dd190 | ||
|
bfa20f2634 | ||
|
840da146b9 | ||
|
acc874c34f | ||
|
0dee968e98 | ||
|
09e919f0c6 | ||
|
eaf1af75bd | ||
|
7c514e4b1d | ||
|
1432baa28b | ||
|
f1f61195c3 | ||
|
8fefa2b716 | ||
|
2a0c30b66d | ||
|
97f8d4d515 | ||
|
2672dea8e3 | ||
|
63f5b0a663 | ||
|
e40ebacc5b | ||
|
e261c93563 | ||
|
df7357b357 | ||
|
2e6c5eacf7 | ||
|
c9fd404012 | ||
|
814302e272 | ||
|
72ba83924c | ||
|
9d742094cb | ||
|
38f3aac98d | ||
|
12ffb9d642 | ||
|
99391a4f0e | ||
|
6492908a62 | ||
|
676bc612d9 | ||
|
d8a5e43034 | ||
|
8ad710abca | ||
|
368cb7bc6b | ||
|
4aa7a052d0 | ||
|
f088f17a0a | ||
|
a55c758b0e | ||
|
f092f434fe | ||
|
2e6dfaa879 | ||
|
96dc2e12d0 | ||
|
d931790ed0 | ||
|
6b26c12bfa | ||
|
5bb08bdb64 | ||
|
405be7ef13 | ||
|
a7e2fb48ba | ||
|
ae80749dd8 | ||
|
8c47b7080d | ||
|
8378638ee4 | ||
|
7c563471f6 | ||
|
29382d2781 | ||
|
6226672ee8 | ||
|
5b5ee82f27 | ||
|
62f74f5917 | ||
|
265c7a18a7 | ||
|
462f191051 | ||
|
6960a439c9 | ||
|
4dff1c5bd5 | ||
|
2c8ebdce7d | ||
|
dae7cb67ef | ||
|
081b01b667 | ||
|
5340ea7ba0 | ||
|
84a649a671 | ||
|
efd9927696 | ||
|
c86a34a15d | ||
|
ff16513f21 | ||
|
906c265aea | ||
|
708c16176b | ||
|
035d1e24b2 | ||
|
48e9b1be7a | ||
|
6acdaf207d | ||
|
9d41b360c9 | ||
|
12cbd73e7f | ||
|
420b068094 | ||
|
ee943c4284 | ||
|
337b3709d6 | ||
|
eb318c678f | ||
|
081df6beb7 | ||
|
ab911b48b5 | ||
|
8cb3491086 | ||
|
ee794d140f | ||
|
a00542b61b | ||
|
041a13c9d3 | ||
|
24aa90bd9c | ||
|
c574f53417 | ||
|
92b84a9e94 | ||
|
bbf3c74cb2 | ||
|
93cb51a975 | ||
|
0b4ae729a3 | ||
|
b90392576e | ||
|
e143260891 | ||
|
644c5c4faa | ||
|
8d8cedd72c | ||
|
082ac62eda | ||
|
7923a790e6 | ||
|
1368c25824 | ||
|
d0b3678ad6 | ||
|
cae8b1a93b | ||
|
a1c1fec8cb | ||
|
55a66dbb39 | ||
|
a2f0c912f0 | ||
|
e29bbf73aa | ||
|
0ba3e9f469 | ||
|
6f200e9218 | ||
|
586b26d2d4 | ||
|
d482d33d6f | ||
|
37c2a8a5de | ||
|
265547213c | ||
|
87e46f5a5a | ||
|
e36f4e5b0a | ||
|
4aff11421f | ||
|
ea642d9e90 | ||
|
17c3496542 | ||
|
0fb79b763d | ||
|
5873bde6a6 | ||
|
0b79387800 | ||
|
6b493bc7d9 | ||
|
de53bc7991 | ||
|
4c5a56a8a5 | ||
|
ed873ef9de | ||
|
d8a553feb0 | ||
|
4717612090 | ||
|
5d1283bd85 | ||
|
3b945b87b8 | ||
|
19c762f9c1 | ||
|
990adf7527 | ||
|
983414d024 | ||
|
d5c05d857f | ||
|
bff6788546 | ||
|
253183a16a | ||
|
0fb3901a18 | ||
|
1b199ec5d8 | ||
|
40395d562a | ||
|
7322c3af04 | ||
|
36c27f1111 | ||
|
95db6c32a3 | ||
|
bed5e98bb0 | ||
|
a5392e5c53 | ||
|
abbd298b31 | ||
|
e219aaa062 | ||
|
cab72e1be6 | ||
|
92372bde1d | ||
|
6747276a87 | ||
|
03915b7533 | ||
|
5e2ec368ad | ||
|
ab8c93fbac | ||
|
d6a3edefd9 | ||
|
727297ec4e | ||
|
eccc4b0be1 | ||
|
8465140bc4 | ||
|
e6ccb751a0 | ||
|
dfc7a15083 | ||
|
37003edae9 | ||
|
faa90eccd3 | ||
|
c91b0df607 | ||
|
f56d99e133 | ||
|
c690662802 | ||
|
4918d699d5 | ||
|
5ec517875e | ||
|
cf56ad985b | ||
|
c09d1558f7 | ||
|
eb190b660e | ||
|
d6f9068695 | ||
|
cb507babaa | ||
|
235d114193 | ||
|
9aba70dcb1 | ||
|
0b61d29c31 | ||
|
335a13a38a | ||
|
128ee41252 | ||
|
ccca41a168 | ||
|
af4c7d8a90 | ||
|
77c691651e | ||
|
e14ec96e21 | ||
|
ff1f337699 | ||
|
3ca87848e5 |
@ -4,7 +4,7 @@
|
|||||||
"ignorePatterns": ["dist", "browser"],
|
"ignorePatterns": ["dist", "browser"],
|
||||||
"plugins": [
|
"plugins": [
|
||||||
"@typescript-eslint",
|
"@typescript-eslint",
|
||||||
"header",
|
"simple-header",
|
||||||
"simple-import-sort",
|
"simple-import-sort",
|
||||||
"unused-imports",
|
"unused-imports",
|
||||||
"path-alias"
|
"path-alias"
|
||||||
@ -26,35 +26,12 @@
|
|||||||
// Since it's only been a month and Vencord has already been stolen
|
// Since it's only been a month and Vencord has already been stolen
|
||||||
// by random skids who rebranded it to "AlphaCord" and erased all license
|
// by random skids who rebranded it to "AlphaCord" and erased all license
|
||||||
// information
|
// information
|
||||||
"header/header": [
|
"simple-header/header": [
|
||||||
2,
|
"error",
|
||||||
"block",
|
{
|
||||||
[
|
"files": ["scripts/header-new.txt", "scripts/header-old.txt"],
|
||||||
{
|
"templates": { "author": [".*", "Vendicated and contributors"] }
|
||||||
"pattern": "!?",
|
}
|
||||||
"template": " "
|
|
||||||
},
|
|
||||||
" * Vencord, a modification for Discord's desktop app",
|
|
||||||
{
|
|
||||||
"pattern": " \\* Copyright \\(c\\) \\d{4}",
|
|
||||||
"template": " * Copyright (c) 2023 Vendicated and contributors"
|
|
||||||
},
|
|
||||||
" *",
|
|
||||||
" * This program is free software: you can redistribute it and/or modify",
|
|
||||||
" * it under the terms of the GNU General Public License as published by",
|
|
||||||
" * the Free Software Foundation, either version 3 of the License, or",
|
|
||||||
" * (at your option) any later version.",
|
|
||||||
" *",
|
|
||||||
" * This program is distributed in the hope that it will be useful,",
|
|
||||||
" * but WITHOUT ANY WARRANTY; without even the implied warranty of",
|
|
||||||
" * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the",
|
|
||||||
" * GNU General Public License for more details.",
|
|
||||||
" *",
|
|
||||||
" * You should have received a copy of the GNU General Public License",
|
|
||||||
" * along with this program. If not, see <https://www.gnu.org/licenses/>.",
|
|
||||||
""
|
|
||||||
],
|
|
||||||
2
|
|
||||||
],
|
],
|
||||||
"quotes": ["error", "double", { "avoidEscape": true }],
|
"quotes": ["error", "double", { "avoidEscape": true }],
|
||||||
"jsx-quotes": ["error", "prefer-double"],
|
"jsx-quotes": ["error", "prefer-double"],
|
||||||
@ -62,7 +39,7 @@
|
|||||||
"indent": ["error", 4, { "SwitchCase": 1 }],
|
"indent": ["error", 4, { "SwitchCase": 1 }],
|
||||||
"arrow-parens": ["error", "as-needed"],
|
"arrow-parens": ["error", "as-needed"],
|
||||||
"eol-last": ["error", "always"],
|
"eol-last": ["error", "always"],
|
||||||
"func-call-spacing": ["error", "never"],
|
"@typescript-eslint/func-call-spacing": ["error", "never"],
|
||||||
"no-multi-spaces": "error",
|
"no-multi-spaces": "error",
|
||||||
"no-trailing-spaces": "error",
|
"no-trailing-spaces": "error",
|
||||||
"no-whitespace-before-property": "error",
|
"no-whitespace-before-property": "error",
|
||||||
|
22
.github/ISSUE_TEMPLATE/blank.yml
vendored
22
.github/ISSUE_TEMPLATE/blank.yml
vendored
@ -1,22 +0,0 @@
|
|||||||
name: Blank Template
|
|
||||||
description: Use this only if your issue does not fit into another template. **DO NOT ASK FOR SUPPORT OR REQUEST PLUGINS**
|
|
||||||
labels: []
|
|
||||||
|
|
||||||
body:
|
|
||||||
- type: textarea
|
|
||||||
id: info-sec
|
|
||||||
attributes:
|
|
||||||
label: Tell us all about it.
|
|
||||||
description: Go nuts, let us know what you're wanting to bring attention to.
|
|
||||||
placeholder: ...
|
|
||||||
validations:
|
|
||||||
required: true
|
|
||||||
|
|
||||||
- type: checkboxes
|
|
||||||
id: agreement-check
|
|
||||||
attributes:
|
|
||||||
label: Request Agreement
|
|
||||||
description: DO NOT USE THIS TEMPLATE FOR SUPPORT OR PLUGIN REQUESTS!!! For Support, **join our Discord**. For plugin requests, **use discussions**
|
|
||||||
options:
|
|
||||||
- label: This is not a support or plugin request
|
|
||||||
required: true
|
|
2
.github/ISSUE_TEMPLATE/config.yml
vendored
2
.github/ISSUE_TEMPLATE/config.yml
vendored
@ -1,4 +1,4 @@
|
|||||||
blank_issues_enabled: false
|
blank_issues_enabled: true
|
||||||
contact_links:
|
contact_links:
|
||||||
- name: Vencord Support Server
|
- name: Vencord Support Server
|
||||||
url: https://discord.gg/D9uwnFnqmd
|
url: https://discord.gg/D9uwnFnqmd
|
||||||
|
43
.github/workflows/build.yml
vendored
43
.github/workflows/build.yml
vendored
@ -37,39 +37,44 @@ jobs:
|
|||||||
- name: Build
|
- name: Build
|
||||||
run: pnpm build --standalone
|
run: pnpm build --standalone
|
||||||
|
|
||||||
|
- name: Generate plugin list
|
||||||
|
run: pnpm generatePluginJson dist/plugins.json
|
||||||
|
|
||||||
- name: Clean up obsolete files
|
- name: Clean up obsolete files
|
||||||
run: |
|
run: |
|
||||||
rm -rf dist/extension* Vencord.user.css
|
rm -rf dist/*-unpacked Vencord.user.css vencordDesktopRenderer.css vencordDesktopRenderer.css.map
|
||||||
|
|
||||||
- name: Get some values needed for the release
|
- name: Get some values needed for the release
|
||||||
id: release_values
|
id: release_values
|
||||||
run: |
|
run: |
|
||||||
echo "release_tag=$(git rev-parse --short HEAD)" >> $GITHUB_ENV
|
echo "release_tag=$(git rev-parse --short HEAD)" >> $GITHUB_ENV
|
||||||
|
|
||||||
- name: Upload DevBuild as release
|
- name: Upload DevBuild as release
|
||||||
|
if: github.repository == 'Vendicated/Vencord'
|
||||||
run: |
|
run: |
|
||||||
gh release upload devbuild --clobber dist/*
|
gh release upload devbuild --clobber dist/*
|
||||||
gh release edit devbuild --title "DevBuild $RELEASE_TAG"
|
gh release edit devbuild --title "DevBuild $RELEASE_TAG"
|
||||||
env:
|
env:
|
||||||
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||||
RELEASE_TAG: ${{ env.release_tag }}
|
RELEASE_TAG: ${{ env.release_tag }}
|
||||||
|
|
||||||
- name: Upload DevBuild to builds repo
|
- name: Upload DevBuild to builds repo
|
||||||
|
if: github.repository == 'Vendicated/Vencord'
|
||||||
run: |
|
run: |
|
||||||
git config --global user.name "$USERNAME"
|
git config --global user.name "$USERNAME"
|
||||||
git config --global user.email actions@github.com
|
git config --global user.email actions@github.com
|
||||||
|
|
||||||
git clone https://$USERNAME:$API_TOKEN@github.com/$GH_REPO.git upload
|
git clone https://$USERNAME:$API_TOKEN@github.com/$GH_REPO.git upload
|
||||||
cd upload
|
cd upload
|
||||||
|
|
||||||
GLOBIGNORE=.git:.gitignore:README.md:LICENSE
|
GLOBIGNORE=.git:.gitignore:README.md:LICENSE
|
||||||
rm -rf *
|
rm -rf *
|
||||||
cp -r ../dist/* .
|
cp -r ../dist/* .
|
||||||
|
|
||||||
git add -A
|
git add -A
|
||||||
git commit -m "Builds for https://github.com/$GITHUB_REPOSITORY/commit/$GITHUB_SHA"
|
git commit -m "Builds for https://github.com/$GITHUB_REPOSITORY/commit/$GITHUB_SHA"
|
||||||
git push --force https://$USERNAME:$API_TOKEN@github.com/$GH_REPO.git
|
git push --force https://$USERNAME:$API_TOKEN@github.com/$GH_REPO.git
|
||||||
env:
|
env:
|
||||||
API_TOKEN: ${{ secrets.BUILDS_TOKEN }}
|
API_TOKEN: ${{ secrets.BUILDS_TOKEN }}
|
||||||
GH_REPO: Vencord/builds
|
GH_REPO: Vencord/builds
|
||||||
USERNAME: GitHub-Actions
|
USERNAME: GitHub-Actions
|
||||||
|
22
.github/workflows/codeberg-mirror.yml
vendored
Normal file
22
.github/workflows/codeberg-mirror.yml
vendored
Normal file
@ -0,0 +1,22 @@
|
|||||||
|
name: Sync to Codeberg
|
||||||
|
concurrency:
|
||||||
|
group: ${{ github.ref }}
|
||||||
|
cancel-in-progress: true
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
workflow_dispatch:
|
||||||
|
schedule:
|
||||||
|
- cron: "0 */6 * * *"
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
codeberg:
|
||||||
|
if: github.repository == 'Vendicated/Vencord'
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v3
|
||||||
|
with:
|
||||||
|
fetch-depth: 0
|
||||||
|
- uses: pixta-dev/repository-mirroring-action@674e65a7d483ca28dafaacba0d07351bdcc8bd75 # v1.1.1
|
||||||
|
with:
|
||||||
|
target_repo_url: "git@codeberg.org:Ven/cord.git"
|
||||||
|
ssh_private_key: ${{ secrets.CODEBERG_SSH_PRIVATE_KEY }}
|
48
.github/workflows/publish.yml
vendored
48
.github/workflows/publish.yml
vendored
@ -6,6 +6,7 @@ on:
|
|||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
Publish:
|
Publish:
|
||||||
|
if: github.repository == 'Vendicated/Vencord'
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
@ -13,11 +14,11 @@ jobs:
|
|||||||
|
|
||||||
- name: check that tag matches package.json version
|
- name: check that tag matches package.json version
|
||||||
run: |
|
run: |
|
||||||
pkg_version="v$(jq -r .version < package.json)"
|
pkg_version="v$(jq -r .version < package.json)"
|
||||||
if [[ "${{ github.ref_name }}" != "$pkg_version" ]]; then
|
if [[ "${{ github.ref_name }}" != "$pkg_version" ]]; then
|
||||||
echo "Tag ${{ github.ref_name }} does not match package.json version $pkg_version" >&2
|
echo "Tag ${{ github.ref_name }} does not match package.json version $pkg_version" >&2
|
||||||
exit 1
|
exit 1
|
||||||
fi
|
fi
|
||||||
|
|
||||||
- uses: pnpm/action-setup@v2 # Install pnpm using packageManager key in package.json
|
- uses: pnpm/action-setup@v2 # Install pnpm using packageManager key in package.json
|
||||||
|
|
||||||
@ -35,27 +36,26 @@ jobs:
|
|||||||
|
|
||||||
- name: Publish extension
|
- name: Publish extension
|
||||||
run: |
|
run: |
|
||||||
cd dist/extension-unpacked
|
# Do not fail so that even if chrome fails, firefox gets a shot. But also store exit code to fail workflow later
|
||||||
|
EXIT_CODE=0
|
||||||
|
|
||||||
# Do not fail so that even if chrome fails, firefox gets a shot. But also store exit code to fail workflow later
|
# Chrome
|
||||||
EXIT_CODE=0
|
cd dist/chromium-unpacked
|
||||||
|
pnpx chrome-webstore-upload-cli@2.1.0 upload --auto-publish || EXIT_CODE=$?
|
||||||
|
|
||||||
# Chrome
|
# Firefox
|
||||||
pnpx chrome-webstore-upload-cli@2.1.0 upload --auto-publish || EXIT_CODE=$?
|
cd ../firefox-unpacked
|
||||||
|
npm i -g web-ext@7.4.0 web-ext-submit@7.4.0
|
||||||
|
web-ext-submit || EXIT_CODE=$?
|
||||||
|
|
||||||
# Firefox
|
exit $EXIT_CODE
|
||||||
npm i -g web-ext@7.4.0 web-ext-submit@7.4.0
|
|
||||||
web-ext-submit || EXIT_CODE=$?
|
|
||||||
|
|
||||||
exit $EXIT_CODE
|
|
||||||
env:
|
env:
|
||||||
# Chrome
|
# Chrome
|
||||||
EXTENSION_ID: ${{ secrets.CHROME_EXTENSION_ID }}
|
EXTENSION_ID: ${{ secrets.CHROME_EXTENSION_ID }}
|
||||||
CLIENT_ID: ${{ secrets.CHROME_CLIENT_ID }}
|
CLIENT_ID: ${{ secrets.CHROME_CLIENT_ID }}
|
||||||
CLIENT_SECRET: ${{ secrets.CHROME_CLIENT_SECRET }}
|
CLIENT_SECRET: ${{ secrets.CHROME_CLIENT_SECRET }}
|
||||||
REFRESH_TOKEN: ${{ secrets.CHROME_REFRESH_TOKEN }}
|
REFRESH_TOKEN: ${{ secrets.CHROME_REFRESH_TOKEN }}
|
||||||
|
|
||||||
# Firefox
|
|
||||||
WEB_EXT_API_KEY: ${{ secrets.WEBEXT_USER }}
|
|
||||||
WEB_EXT_API_SECRET: ${{ secrets.WEBEXT_SECRET }}
|
|
||||||
|
|
||||||
|
# Firefox
|
||||||
|
WEB_EXT_API_KEY: ${{ secrets.WEBEXT_USER }}
|
||||||
|
WEB_EXT_API_SECRET: ${{ secrets.WEBEXT_SECRET }}
|
||||||
|
37
.github/workflows/reportBrokenPlugins.yml
vendored
37
.github/workflows/reportBrokenPlugins.yml
vendored
@ -2,11 +2,12 @@ name: Test Patches
|
|||||||
on:
|
on:
|
||||||
workflow_dispatch:
|
workflow_dispatch:
|
||||||
schedule:
|
schedule:
|
||||||
# Every day at midnight
|
# Every day at midnight
|
||||||
- cron: 0 0 * * *
|
- cron: 0 0 * * *
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
TestPlugins:
|
TestPlugins:
|
||||||
|
if: github.repository == 'Vendicated/Vencord'
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
@ -22,10 +23,10 @@ jobs:
|
|||||||
|
|
||||||
- name: Install dependencies
|
- name: Install dependencies
|
||||||
run: |
|
run: |
|
||||||
pnpm install --frozen-lockfile
|
pnpm install --frozen-lockfile
|
||||||
pnpm add puppeteer
|
pnpm add puppeteer
|
||||||
|
|
||||||
sudo apt-get install -y chromium-browser
|
sudo apt-get install -y chromium-browser
|
||||||
|
|
||||||
- name: Build web
|
- name: Build web
|
||||||
run: pnpm buildWeb --standalone
|
run: pnpm buildWeb --standalone
|
||||||
@ -33,25 +34,25 @@ jobs:
|
|||||||
- name: Create Report
|
- name: Create Report
|
||||||
timeout-minutes: 10
|
timeout-minutes: 10
|
||||||
run: |
|
run: |
|
||||||
export PATH="$PWD/node_modules/.bin:$PATH"
|
export PATH="$PWD/node_modules/.bin:$PATH"
|
||||||
export CHROMIUM_BIN=$(which chromium-browser)
|
export CHROMIUM_BIN=$(which chromium-browser)
|
||||||
|
|
||||||
esbuild test/generateReport.ts > dist/report.mjs
|
esbuild scripts/generateReport.ts > dist/report.mjs
|
||||||
node dist/report.mjs >> $GITHUB_STEP_SUMMARY
|
node dist/report.mjs >> $GITHUB_STEP_SUMMARY
|
||||||
env:
|
env:
|
||||||
DISCORD_TOKEN: ${{ secrets.DISCORD_TOKEN }}
|
DISCORD_TOKEN: ${{ secrets.DISCORD_TOKEN }}
|
||||||
DISCORD_WEBHOOK: ${{ secrets.DISCORD_WEBHOOK }}
|
DISCORD_WEBHOOK: ${{ secrets.DISCORD_WEBHOOK }}
|
||||||
|
|
||||||
- name: Create Report (Canary)
|
- name: Create Report (Canary)
|
||||||
timeout-minutes: 10
|
timeout-minutes: 10
|
||||||
if: success() || failure() # even run if previous one failed
|
if: success() || failure() # even run if previous one failed
|
||||||
run: |
|
run: |
|
||||||
export PATH="$PWD/node_modules/.bin:$PATH"
|
export PATH="$PWD/node_modules/.bin:$PATH"
|
||||||
export CHROMIUM_BIN=$(which chromium-browser)
|
export CHROMIUM_BIN=$(which chromium-browser)
|
||||||
export USE_CANARY=true
|
export USE_CANARY=true
|
||||||
|
|
||||||
esbuild test/generateReport.ts > dist/report.mjs
|
esbuild scripts/generateReport.ts > dist/report.mjs
|
||||||
node dist/report.mjs >> $GITHUB_STEP_SUMMARY
|
node dist/report.mjs >> $GITHUB_STEP_SUMMARY
|
||||||
env:
|
env:
|
||||||
DISCORD_TOKEN: ${{ secrets.DISCORD_TOKEN }}
|
DISCORD_TOKEN: ${{ secrets.DISCORD_TOKEN }}
|
||||||
DISCORD_WEBHOOK: ${{ secrets.DISCORD_WEBHOOK }}
|
DISCORD_WEBHOOK: ${{ secrets.DISCORD_WEBHOOK }}
|
||||||
|
9
.github/workflows/test.yml
vendored
9
.github/workflows/test.yml
vendored
@ -15,7 +15,7 @@ jobs:
|
|||||||
- uses: pnpm/action-setup@v2 # Install pnpm using packageManager key in package.json
|
- uses: pnpm/action-setup@v2 # Install pnpm using packageManager key in package.json
|
||||||
|
|
||||||
- name: Use Node.js 18
|
- name: Use Node.js 18
|
||||||
uses: actions/setup-node@v2
|
uses: actions/setup-node@v3
|
||||||
with:
|
with:
|
||||||
node-version: 18
|
node-version: 18
|
||||||
cache: "pnpm"
|
cache: "pnpm"
|
||||||
@ -26,5 +26,8 @@ jobs:
|
|||||||
- name: Lint & Test if desktop version compiles
|
- name: Lint & Test if desktop version compiles
|
||||||
run: pnpm test
|
run: pnpm test
|
||||||
|
|
||||||
- name: Lint & Test if web version compiles
|
- name: Test if web version compiles
|
||||||
run: pnpm testWeb
|
run: pnpm buildWeb
|
||||||
|
|
||||||
|
- name: Test if plugin structure is valid
|
||||||
|
run: pnpm generatePluginJson
|
||||||
|
9
.vscode/settings.json
vendored
9
.vscode/settings.json
vendored
@ -12,5 +12,12 @@
|
|||||||
"javascript.format.semicolons": "insert",
|
"javascript.format.semicolons": "insert",
|
||||||
"typescript.format.semicolons": "insert",
|
"typescript.format.semicolons": "insert",
|
||||||
"typescript.preferences.quoteStyle": "double",
|
"typescript.preferences.quoteStyle": "double",
|
||||||
"javascript.preferences.quoteStyle": "double"
|
"javascript.preferences.quoteStyle": "double",
|
||||||
|
|
||||||
|
"gitlens.remotes": [
|
||||||
|
{
|
||||||
|
"domain": "codeberg.org",
|
||||||
|
"type": "Gitea"
|
||||||
|
}
|
||||||
|
]
|
||||||
}
|
}
|
||||||
|
25
.vscode/tasks.json
vendored
Normal file
25
.vscode/tasks.json
vendored
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
{
|
||||||
|
// See https://go.microsoft.com/fwlink/?LinkId=733558
|
||||||
|
// for the documentation about the tasks.json format
|
||||||
|
"version": "2.0.0",
|
||||||
|
"tasks": [
|
||||||
|
{
|
||||||
|
"label": "Build",
|
||||||
|
"type": "shell",
|
||||||
|
"command": "pnpm build",
|
||||||
|
"group": {
|
||||||
|
"kind": "build",
|
||||||
|
"isDefault": true
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"label": "Watch",
|
||||||
|
"type": "shell",
|
||||||
|
"command": "pnpm watch",
|
||||||
|
"problemMatcher": [],
|
||||||
|
"group": {
|
||||||
|
"kind": "build"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
20
CODE_OF_CONDUCT.md
Normal file
20
CODE_OF_CONDUCT.md
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
# Code of Conduct
|
||||||
|
|
||||||
|
Our community is welcoming to everyone, regardless of their characteristics.
|
||||||
|
|
||||||
|
As such, we expect you to treat everyone with respect and contribute to an open and welcoming community.
|
||||||
|
|
||||||
|
DO
|
||||||
|
- have empathy and be nice to others
|
||||||
|
- be respectful of differing opinions, even if you disagree
|
||||||
|
- give and accept constructive criticism
|
||||||
|
|
||||||
|
DON'T
|
||||||
|
- use offensive or derogatory language
|
||||||
|
- troll or spam
|
||||||
|
- personally attack or harass others
|
||||||
|
|
||||||
|
Repetitive violations of these guidelines might get your access to the repository restricted.
|
||||||
|
|
||||||
|
|
||||||
|
If you feel like a user is violating these guidelines or feel treated unfairly, please refrain from publicly challenging them and instead contact a Moderator on our Discord server or send an email to vendicated+conduct@riseup.net!
|
68
README.md
68
README.md
@ -1,41 +1,79 @@
|
|||||||
# Vencord
|
# Vencord
|
||||||
|
|
||||||
|
[![Codeberg Mirror](https://img.shields.io/static/v1?style=for-the-badge&label=Codeberg%20Mirror&message=codeberg.org/Ven/cord&color=2185D0&logo=data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAEAAAABACAYAAACqaXHeAAAABmJLR0QA/wD/AP+gvaeTAAAKbUlEQVR4nNVae3AV5RX/nW/3Pva+b24e5HHzIICQKGoiYiW8NFBFgohaa6ctglpbFSujSGurzUinohWsOij/gGX6R2fqOK0d1FYTEZXaTrWCBbEikJCEyCvkeXNvkrunf+zdkJDkPnex/c3cmd29+53v/M6e73znnF2Cydj4Tntldzi6qrN/qKqzf2jy6b7BnL4B1dI7oMp9AyoRAIdVsNMqhlxWMZjtspzyK/Jhr036OMsm//bh2vzPzNSPzBD6xFutd7R0Dq758ky4orkjYuc05RCAkixbeEq2/UCJ1/LczxcX/c5IPfU5DMHmxpbCpu7o1k/b+xc1n43YjJI7EqV+W2RmvuPt0oDjB2vn5bQbITNjAzzdeKK8qTO0bU9T77zucNQUjzofHrvENWWu3aUBZfW6+ZOOZiIrbYXrmUXo9daX3v6i667O/iGRiRLpwqtIvKDc+0efJ3hb/UIaSkdGWgZ4sqGt9r2m3lc/P9HvSWe80ZiRp3TPL/UsX1+bvyvVsSkb4NE3WjbuPNj5SM8Fcvdk4bAKrqvwv7DxhuCPUxmXNIn6XSy3nWr6R8OhrqrU1btwqJ3m/bgwu/SqZJdEUgbYsuuka09b9/4Pm3tLMlPvwuAbpe6m+RcplfdcURBKdG9CA2zZddLV2Nx1+JO2vlxj1LswqCpynlxc6SxLZIS40bueWfy9vXvv/xt5APhXa1/u7v+EPqvfxXK8++IaoO2Vpn9+cLS33FjVLhw+bOotOX7q6N/i3TOhAX7y+rHN/+sBLxm8fah71k93tjw/0f/jGuDJxtZrdh7setA8tS4sdn7eef+v3mmfP95/Ywxw6x9Yev9I35/6Iubv83WVfl5a6Uu3VkoavZEo7TnS/Vo98xi+Yy6UKC3bDp7sd5ut1OWFDjyzNMib6oq5Oug0ezp8dqLfG3r92Nbzr48ywNONJ8obDnV/z2xlAk4ZW1aUqhaJIAvCb5YVqwFn3GBtCBoO9dz5TOPxUbnMKAM0dYa2d5lc2AgCNi8r5klui3aBgWynjE11QZbI3FV3NjQkjnYNbB+lj36wubGlcE9T71xTNQDw0Px8nlvmHl73GmfCrKCL19Tkmh4P9jT1LHz2vVP5+vmwAZq71a1m1/PXTPXwD68eS5KIEVUZd1yZwwumeEw1Qld/lJrPhF7Sz4cNsO+rUK2ZExd6rfj10iCPZ2GJCCoAZuCJxQUc9FvNVAX72kPX6ccC0Hp4zR0Ru1kT2mTCSzeXqn5l/EAniMAqoDLDYZWwqa5EVSzmhaKmsxHbLxvbbgdiBmjpHFxj2mwANlxXxBdPUib8nwgQgqAyEFUZxT4L1i/MN3UpHDsTWQvEDHDoTLjCrIluuyzAt8zMSkhGFhp5hrYUFk3z8IqZftOMcKRj4GIAEM80tFccM8n9Z+Qq+MXigqRIWCQCMzQvYIbKwH1X53FFnjkr88iZsLKpoXWa6BiIrjbDzF67hK23lKp2Obm1LAstPEZVjTwDkAio/2ZQ9dolw/VjAB0DfKfoCg9WGy2cADy1NMhBX2rR3CIRGICq8rAhAg4Jj9UWsDBhg+4MR6vF2VC0zGjB99fk8eJp3pQdyyrRMHF9KURVxswCB6+alWO4o3b2RyeLU32D2UYKnVPm5gfm5qWlrF0Wo4hzbCmoDNw0089XlboNNcLpvsFc0RtRDXuNle+x4Lkbi9PO6WWJIBFGEY+qjGjswtq5eVzosRilLnoiUavoH1INiTCyIDy/vETNcmRW1dl0L4gRVxmx3YFhlwnrry1QrZIxASE0yJIIDaiGSHt8UQFXF2Ve1zusYgzxkXGhyGvFvePUE+mgfyAqhGqAqKWVPv5udbYhSjmtkpYWq6OJqzFjqCpjTpmbl1Rk3klSGRBWmTISNC3Hjo1LgoYFJ0GA1aIVR+cTVxlQoS2Pb18a4PLszMKXzSJYuCySmq4Al03CiytKVYfBhYvLKk1IXE+XLRLhwZp81WlNf26HTFHhd0jhdAYTgKduCPLkgPHfQjitYkLiAIEZBDBlu2R6aF7euCV2Mgg45bDw2qWOdAavnp3D109PPdlJBvpTnYg4kVY3MDMuylVw62WJi63x4LHLZ0TAIR9OdWBVodPUclUQwWmT4hLXfgCIUDfDi6oiR8rzBJzyl8LnkD9KZVCOU8aLN5eoshnJ+Qh4bFJC4gztmEjgrtk5anaKnWWfXfpIuBTLjmSpSILw/E0laq7LuGxsIngVCYmIa96hLRG3TaZ1C/KTfjAEQLFIO8TPFk7aH/RZI8kMWrdgEs8udqXLKSUoMkEW4ETEQTRsoHyPlVZfmVw+Uuy3hR9bVHBQAMD0XPu/Ew24dqqH777K/La1DiKCxyYlRRzQymgG4+oyDxZOTdxZnp5r3wvEWmJ5btuL8W4uzbJh87LitLebdOFVpKSJx4IlwIzbL81CcYLO8iSX/IImGQCYae6Wg/2tXQNjNnW7LPDKyilqZd7ETU2zEBlifNTSS4i9PNFIx44x4jh2nZlBsUr0dN8QP/6XVhEaHJvnlfhtkXd/NF0BUextKRFXFznfGk+JDdcX8tdBHtDa6YpFsB4I9ac88omf8wbEgqa2XAIOme6bM35foqrQ+QZIKwGG80ifVbrXZZNGDfhOVYBvviS9JMMoaP3AEcQpPnHdOxiMGXkKbrx4dGfZY5c4T8H9+vmwAeqXFLXOKXW9r59fWuDA44sKv1byAOBzyCkTH+kdS2f4MLPgXJI0p9T17vrFxcf181GVxEUB+0qfIqt+RcKWFSWGNR4ygd4RTpW4HiCJgFWzstmnSPA7ZLU827pypPwxDB/687GXl1X6Vs6bbGz/LRN80hZCT+yLFZ0cgHED4egACeiXm89GsP9EePuzy4rvGil7jAGYmQDsBjDHUBYZ4GhHBMfORigd4rpnyIS9u6d4rqgnGrUtjCmmSYuOqwB0GcwjbWh9xviurpNnxnDA1IspMPe6bOL755MHJvhKjIgOA7jbJD4pw22Thj+kSIW47h2KRaydVezeP57sCdspRPQqgGeNJJIuBAE+ReJUiOv32mXaXjPZs21C2QnmXgdghyEsMoRfkVMiDgCywF/by9z3xJMb1wCxeHAPgDczZpAh/Iq+HSYmDjCsstgThmf5t4ii8eQm7CgS0SCA5QBezoRApnBaBSyCEhIHCLJEb4ZUd+2SqZSwzE+qpUpEQ9CC4qb01M8cRIQsh8zxiKsMtsn08nvlnrpkyAPj5AGJwMw3AtgGwJ/q2ExxvHsQB74KxfKBMblAyGmTHq4pc4/5GjQeUm6qE9FrAK4E8H6ie41GlkN/jTk6F5Ak2ueUpNmpkgfSMAAAENERAAsB3AHgZDoy0oFdFnBYpXPEBfU4beLRD6Z4qmumug+kIzPjaoeZfQDWAHgAQFam8hLh4MkwWjsHemyS2OF08IYrCjynzZ4zKTCzi5nXMvOnzBw16bevIxR95JOj7DNKb1PqXWa+HMDtAGoBXII0lxq0N2OfAmgA8Hsi2muMhudgesHPzNkA5gKoADADwFRoS8UHQO+x9wLoBNAB4AsAnwM4AOADIjLVxf8L9kdXUOE0IskAAAAASUVORK5CYII=)](https://codeberg.org/Ven/cord)
|
||||||
|
|
||||||
The cutest Discord client mod
|
The cutest Discord client mod
|
||||||
|
|
||||||
|
![image](https://github.com/Vendicated/Vencord/assets/45497981/706722b1-32de-4d99-bee9-93993b504334)
|
||||||
|
|
||||||
## Features
|
## Features
|
||||||
|
|
||||||
- Super easy to install (one click installer)
|
- Super easy to install (Download Installer, open, click install button, done)
|
||||||
- 90+ plugins built in: [See a list](https://gist.github.com/Vendicated/8696cde7b92548064a3ae92ead84d033)
|
- 100+ plugins built in: [See a list](https://vencord.dev/plugins)
|
||||||
- Some highlights: SpotifyControls, Experiments, NoTrack, MessageLogger, QuickReply, Free Emotes/Stickers, CustomCommands, ShowHiddenChannels, PronounDB
|
- Some highlights: SpotifyControls, MessageLogger, Experiments, GameActivityToggle, Translate, NoTrack, QuickReply, Free Emotes/Stickers, PermissionsViewer, CustomCommands, ShowHiddenChannels, PronounDB
|
||||||
|
- Fairly lightweight despite the many inbuilt plugins
|
||||||
- Excellent Browser Support: Run Vencord in your Browser via extension or UserScript
|
- Excellent Browser Support: Run Vencord in your Browser via extension or UserScript
|
||||||
|
- Works on any Discord branch: Stable, Canary or PTB all work (though for the best experience I recommend stable!)
|
||||||
- Custom CSS and Themes: Inbuilt css editor with support to import any css files (including BetterDiscord themes)
|
- Custom CSS and Themes: Inbuilt css editor with support to import any css files (including BetterDiscord themes)
|
||||||
- Works in all Electron versions (Confirmed working on versions 13-23)
|
- Privacy friendly, blocks Discord analytics & crash reporting out of the box and has no telemetry
|
||||||
- Maintained very actively, broken plugins are usually fixed within 12 hours
|
- Maintained very actively, broken plugins are usually fixed within 12 hours
|
||||||
|
- Settings sync: Keep your plugins and their settings synchronised between devices / apps (optional)
|
||||||
|
|
||||||
|
|
||||||
## Installing / Uninstalling
|
## Installing / Uninstalling
|
||||||
|
|
||||||
[![Download and run the Installer ](https://img.shields.io/github/v/release/Vencord/Installer?label=Download%20Vencord%20Installer&style=for-the-badge)](https://github.com/Vencord/Installer#usage)
|
Click the below button to install Vencord to the Discord Desktop app
|
||||||
|
|
||||||
|
[![Download and run the Installer](https://img.shields.io/github/v/release/Vencord/Installer?label=Download%20Vencord%20Installer&style=for-the-badge)](https://github.com/Vencord/Installer#vencord-installer)
|
||||||
|
|
||||||
## Installing on Browser
|
## Installing on Browser
|
||||||
|
|
||||||
[![Get it on the Firefox Webstore](https://blog.mozilla.org/addons/files/2015/11/get-the-addon.png)](https://addons.mozilla.org/en-GB/firefox/addon/vencord-web/) [![Get it on the Chrome Webstore](https://storage.googleapis.com/web-dev-uploads/image/WlD8wC6g8khYWPJUsQceQkhXSlv1/UV4C4ybeBTsZt43U4xis.png)](https://chrome.google.com/webstore/detail/vencord-web/cbghhgpcnddeihccjmnadmkaejncjndb)
|
[![Get it on the Firefox Webstore](https://blog.mozilla.org/addons/files/2015/11/get-the-addon.png)](https://addons.mozilla.org/en-GB/firefox/addon/vencord-web/) [![Get it on the Chrome Webstore](https://storage.googleapis.com/web-dev-uploads/image/WlD8wC6g8khYWPJUsQceQkhXSlv1/UV4C4ybeBTsZt43U4xis.png)](https://chrome.google.com/webstore/detail/vencord-web/cbghhgpcnddeihccjmnadmkaejncjndb)
|
||||||
|
|
||||||
Or use the [UserScript](https://raw.githubusercontent.com/Vencord/builds/main/Vencord.user.js) - Please note that QuickCSS and plugins making use of external resources will not work with the UserScript.
|
Or use the [UserScript](https://raw.githubusercontent.com/Vencord/builds/main/Vencord.user.js) - Please note that the CSS Editor, Themes loaded from remote sources and co. will not work in the UserScript. Use the extension if you need any of those
|
||||||
|
|
||||||
## Building from Source
|
<details>
|
||||||
|
<summary>Alternative Downloads</summary>
|
||||||
|
|
||||||
See the docs folder
|
## Vencord Desktop
|
||||||
|
|
||||||
## Contributing
|
> **Warning**
|
||||||
|
> This is an alternative app. It currently doesn't support keybinds and possibly some more features. If you just want to install to the normal Discord Desktop app, scroll up
|
||||||
|
|
||||||
See [CONTRIBUTING.md](CONTRIBUTING.md) and [Megu's Plugin Guide!](docs/2_PLUGINS.md)
|
As an alternative to the Discord Desktop app, Vencord also has its own standalone Desktop app that is snappier and lighter than Discord's official Desktop app
|
||||||
|
|
||||||
[contribute]: CONTRIBUTING.md
|
[![Download Vencord Desktop](https://img.shields.io/github/v/release/Vencord/Desktop?label=Download%20Vencord%20Desktop&style=for-the-badge)](https://github.com/Vencord/Desktop#vencord-desktop)
|
||||||
|
|
||||||
[contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute] [contribute]
|
</details>
|
||||||
|
|
||||||
## Join
|
## Join our Support/Community Server
|
||||||
|
|
||||||
[join]: https://discord.gg/D9uwnFnqmd
|
https://discord.gg/D9uwnFnqmd
|
||||||
|
|
||||||
[join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join] [join]
|
## Star History
|
||||||
|
|
||||||
|
<a href="https://star-history.com/#Vendicated/Vencord&Timeline">
|
||||||
|
<picture>
|
||||||
|
<source media="(prefers-color-scheme: dark)" srcset="https://api.star-history.com/svg?repos=Vendicated/Vencord&type=Timeline&theme=dark" />
|
||||||
|
<source media="(prefers-color-scheme: light)" srcset="https://api.star-history.com/svg?repos=Vendicated/Vencord&type=Timeline" />
|
||||||
|
<img alt="Star History Chart" src="https://api.star-history.com/svg?repos=Vendicated/Vencord&type=Timeline" />
|
||||||
|
</picture>
|
||||||
|
</a>
|
||||||
|
|
||||||
|
## Disclaimer
|
||||||
|
|
||||||
|
Discord is trademark of Discord Inc. and solely mentioned for the sake of descriptivity.
|
||||||
|
Mention of it does not imply any affiliation with or endorsement by Discord Inc.
|
||||||
|
|
||||||
|
<details>
|
||||||
|
<summary>Using Vencord violates Discord's terms of service</summary>
|
||||||
|
|
||||||
|
Client modifications are against Discord’s Terms of Service.
|
||||||
|
|
||||||
|
However, Discord is pretty indifferent about them and there are no known cases of users getting banned for using client mods! So you should generally be fine as long as you don’t use any plugins that implement abusive behaviour. But no worries, all inbuilt plugins are safe to use!
|
||||||
|
|
||||||
|
Regardless, if your account is very important to you and it getting disabled would be a disaster for you, you should probably not use any client mods (not exclusive to Vencord), just to be safe
|
||||||
|
|
||||||
|
Additionally, make sure not to post screenshots with Vencord in a server where you might get banned for it
|
||||||
|
|
||||||
|
</details>
|
||||||
|
@ -16,20 +16,6 @@
|
|||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
function fetchOptions(url) {
|
|
||||||
return new Promise((resolve, reject) => {
|
|
||||||
const opt = {
|
|
||||||
method: "OPTIONS",
|
|
||||||
url: url,
|
|
||||||
};
|
|
||||||
opt.onload = resp => resolve(resp.responseHeaders);
|
|
||||||
opt.ontimeout = () => reject("fetch timeout");
|
|
||||||
opt.onerror = () => reject("fetch error");
|
|
||||||
opt.onabort = () => reject("fetch abort");
|
|
||||||
GM_xmlhttpRequest(opt);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
function parseHeaders(headers) {
|
function parseHeaders(headers) {
|
||||||
if (!headers)
|
if (!headers)
|
||||||
return {};
|
return {};
|
||||||
@ -52,19 +38,6 @@ function parseHeaders(headers) {
|
|||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
// returns true if CORS permits request
|
|
||||||
async function checkCors(url, method) {
|
|
||||||
const headers = parseHeaders(await fetchOptions(url));
|
|
||||||
|
|
||||||
const origin = headers["access-control-allow-origin"];
|
|
||||||
if (origin !== "*" && origin !== window.location.origin) return false;
|
|
||||||
|
|
||||||
const methods = headers["access-control-allow-methods"]?.split(/,\s/g);
|
|
||||||
if (methods && !methods.includes(method)) return false;
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
function blobTo(to, blob) {
|
function blobTo(to, blob) {
|
||||||
if (to === "arrayBuffer" && blob.arrayBuffer) return blob.arrayBuffer();
|
if (to === "arrayBuffer" && blob.arrayBuffer) return blob.arrayBuffer();
|
||||||
return new Promise((resolve, reject) => {
|
return new Promise((resolve, reject) => {
|
||||||
@ -78,30 +51,25 @@ function blobTo(to, blob) {
|
|||||||
|
|
||||||
function GM_fetch(url, opt) {
|
function GM_fetch(url, opt) {
|
||||||
return new Promise((resolve, reject) => {
|
return new Promise((resolve, reject) => {
|
||||||
checkCors(url, opt?.method || "GET")
|
// https://www.tampermonkey.net/documentation.php?ext=dhdg#GM_xmlhttpRequest
|
||||||
.then(can => {
|
const options = opt || {};
|
||||||
if (can) {
|
options.url = url;
|
||||||
// https://www.tampermonkey.net/documentation.php?ext=dhdg#GM_xmlhttpRequest
|
options.data = options.body;
|
||||||
const options = opt || {};
|
options.responseType = "blob";
|
||||||
options.url = url;
|
options.onload = resp => {
|
||||||
options.data = options.body;
|
var blob = resp.response;
|
||||||
options.responseType = "blob";
|
resp.blob = () => Promise.resolve(blob);
|
||||||
options.onload = resp => {
|
resp.arrayBuffer = () => blobTo("arrayBuffer", blob);
|
||||||
var blob = resp.response;
|
resp.text = () => blobTo("text", blob);
|
||||||
resp.blob = () => Promise.resolve(blob);
|
resp.json = async () => JSON.parse(await blobTo("text", blob));
|
||||||
resp.arrayBuffer = () => blobTo("arrayBuffer", blob);
|
resp.headers = new Headers(parseHeaders(resp.responseHeaders));
|
||||||
resp.text = () => blobTo("text", blob);
|
resp.ok = resp.status >= 200 && resp.status < 300;
|
||||||
resp.json = async () => JSON.parse(await blobTo("text", blob));
|
resolve(resp);
|
||||||
resolve(resp);
|
};
|
||||||
};
|
options.ontimeout = () => reject("fetch timeout");
|
||||||
options.ontimeout = () => reject("fetch timeout");
|
options.onerror = () => reject("fetch error");
|
||||||
options.onerror = () => reject("fetch error");
|
options.onabort = () => reject("fetch abort");
|
||||||
options.onabort = () => reject("fetch abort");
|
GM_xmlhttpRequest(options);
|
||||||
GM_xmlhttpRequest(options);
|
|
||||||
} else {
|
|
||||||
reject("CORS issue");
|
|
||||||
}
|
|
||||||
});
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
export const fetch = GM_fetch;
|
export const fetch = GM_fetch;
|
||||||
|
@ -16,51 +16,86 @@
|
|||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
/// <reference path="../src/modules.d.ts" />
|
||||||
|
/// <reference path="../src/globals.d.ts" />
|
||||||
|
|
||||||
|
import monacoHtml from "~fileContent/../src/components/monacoWin.html";
|
||||||
import * as DataStore from "../src/api/DataStore";
|
import * as DataStore from "../src/api/DataStore";
|
||||||
import IpcEvents from "../src/utils/IpcEvents";
|
import { debounce } from "../src/utils";
|
||||||
|
import { getTheme, Theme } from "../src/utils/discord";
|
||||||
|
import { getThemeInfo } from "../src/main/themes";
|
||||||
|
|
||||||
// Discord deletes this so need to store in variable
|
// Discord deletes this so need to store in variable
|
||||||
const { localStorage } = window;
|
const { localStorage } = window;
|
||||||
|
|
||||||
// listeners for ipc.on
|
// listeners for ipc.on
|
||||||
const listeners = {} as Record<string, Set<Function>>;
|
const cssListeners = new Set<(css: string) => void>();
|
||||||
|
const NOOP = () => { };
|
||||||
|
const NOOP_ASYNC = async () => { };
|
||||||
|
|
||||||
const handlers = {
|
const setCssDebounced = debounce((css: string) => VencordNative.quickCss.set(css));
|
||||||
[IpcEvents.GET_REPO]: () => "https://github.com/Vendicated/Vencord", // shrug
|
|
||||||
[IpcEvents.GET_SETTINGS_DIR]: () => "LocalStorage",
|
|
||||||
|
|
||||||
[IpcEvents.GET_QUICK_CSS]: () => DataStore.get("VencordQuickCss").then(s => s ?? ""),
|
const themeStore = DataStore.createStore("VencordThemes", "VencordThemeData");
|
||||||
[IpcEvents.SET_QUICK_CSS]: (css: string) => {
|
|
||||||
DataStore.set("VencordQuickCss", css);
|
|
||||||
listeners[IpcEvents.QUICK_CSS_UPDATE]?.forEach(l => l(null, css));
|
|
||||||
},
|
|
||||||
|
|
||||||
[IpcEvents.GET_SETTINGS]: () => localStorage.getItem("VencordSettings") || "{}",
|
|
||||||
[IpcEvents.SET_SETTINGS]: (s: string) => localStorage.setItem("VencordSettings", s),
|
|
||||||
|
|
||||||
[IpcEvents.GET_UPDATES]: () => ({ ok: true, value: [] }),
|
|
||||||
|
|
||||||
[IpcEvents.OPEN_EXTERNAL]: (url: string) => open(url, "_blank"),
|
|
||||||
};
|
|
||||||
|
|
||||||
function onEvent(event: string, ...args: any[]) {
|
|
||||||
const handler = handlers[event];
|
|
||||||
if (!handler) throw new Error(`Event ${event} not implemented.`);
|
|
||||||
return handler(...args);
|
|
||||||
}
|
|
||||||
|
|
||||||
// probably should make this less cursed at some point
|
// probably should make this less cursed at some point
|
||||||
window.VencordNative = {
|
window.VencordNative = {
|
||||||
getVersions: () => ({}),
|
themes: {
|
||||||
ipc: {
|
uploadTheme: (fileName: string, fileData: string) => DataStore.set(fileName, fileData, themeStore),
|
||||||
send: (event: string, ...args: any[]) => void onEvent(event, ...args),
|
deleteTheme: (fileName: string) => DataStore.del(fileName, themeStore),
|
||||||
sendSync: onEvent,
|
getThemesDir: async () => "",
|
||||||
on(event: string, listener: () => {}) {
|
getThemesList: () => DataStore.entries(themeStore).then(entries =>
|
||||||
(listeners[event] ??= new Set()).add(listener);
|
entries.map(([name, css]) => getThemeInfo(css, name.toString()))
|
||||||
},
|
),
|
||||||
off(event: string, listener: () => {}) {
|
getThemeData: (fileName: string) => DataStore.get(fileName, themeStore)
|
||||||
return listeners[event]?.delete(listener);
|
|
||||||
},
|
|
||||||
invoke: (event: string, ...args: any[]) => Promise.resolve(onEvent(event, ...args))
|
|
||||||
},
|
},
|
||||||
|
|
||||||
|
native: {
|
||||||
|
getVersions: () => ({}),
|
||||||
|
openExternal: async (url) => void open(url, "_blank")
|
||||||
|
},
|
||||||
|
|
||||||
|
updater: {
|
||||||
|
getRepo: async () => ({ ok: true, value: "https://github.com/Vendicated/Vencord" }),
|
||||||
|
getUpdates: async () => ({ ok: true, value: [] }),
|
||||||
|
update: async () => ({ ok: true, value: false }),
|
||||||
|
rebuild: async () => ({ ok: true, value: true }),
|
||||||
|
},
|
||||||
|
|
||||||
|
quickCss: {
|
||||||
|
get: () => DataStore.get("VencordQuickCss").then(s => s ?? ""),
|
||||||
|
set: async (css: string) => {
|
||||||
|
await DataStore.set("VencordQuickCss", css);
|
||||||
|
cssListeners.forEach(l => l(css));
|
||||||
|
},
|
||||||
|
addChangeListener(cb) {
|
||||||
|
cssListeners.add(cb);
|
||||||
|
},
|
||||||
|
addThemeChangeListener: NOOP,
|
||||||
|
openFile: NOOP_ASYNC,
|
||||||
|
async openEditor() {
|
||||||
|
const features = `popup,width=${Math.min(window.innerWidth, 1000)},height=${Math.min(window.innerHeight, 1000)}`;
|
||||||
|
const win = open("about:blank", "VencordQuickCss", features);
|
||||||
|
if (!win) {
|
||||||
|
alert("Failed to open QuickCSS popup. Make sure to allow popups!");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
win.setCss = setCssDebounced;
|
||||||
|
win.getCurrentCss = () => VencordNative.quickCss.get();
|
||||||
|
win.getTheme = () =>
|
||||||
|
getTheme() === Theme.Light
|
||||||
|
? "vs-light"
|
||||||
|
: "vs-dark";
|
||||||
|
|
||||||
|
win.document.write(monacoHtml);
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
|
settings: {
|
||||||
|
get: () => localStorage.getItem("VencordSettings") || "{}",
|
||||||
|
set: async (s: string) => localStorage.setItem("VencordSettings", s),
|
||||||
|
getSettingsDir: async () => "LocalStorage"
|
||||||
|
},
|
||||||
|
|
||||||
|
pluginHelpers: {} as any,
|
||||||
};
|
};
|
||||||
|
32
browser/background.js
Normal file
32
browser/background.js
Normal file
@ -0,0 +1,32 @@
|
|||||||
|
/**
|
||||||
|
* @template T
|
||||||
|
* @param {T[]} arr
|
||||||
|
* @param {(v: T) => boolean} predicate
|
||||||
|
*/
|
||||||
|
function removeFirst(arr, predicate) {
|
||||||
|
const idx = arr.findIndex(predicate);
|
||||||
|
if (idx !== -1) arr.splice(idx, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
chrome.webRequest.onHeadersReceived.addListener(
|
||||||
|
({ responseHeaders, type, url }) => {
|
||||||
|
if (!responseHeaders) return;
|
||||||
|
|
||||||
|
if (type === "main_frame") {
|
||||||
|
// In main frame requests, the CSP needs to be removed to enable fetching of custom css
|
||||||
|
// as desired by the user
|
||||||
|
removeFirst(responseHeaders, h => h.name.toLowerCase() === "content-security-policy");
|
||||||
|
} else if (type === "stylesheet" && url.startsWith("https://raw.githubusercontent.com/")) {
|
||||||
|
// Most users will load css from GitHub, but GitHub doesn't set the correct content type,
|
||||||
|
// so we fix it here
|
||||||
|
removeFirst(responseHeaders, h => h.name.toLowerCase() === "content-type");
|
||||||
|
responseHeaders.push({
|
||||||
|
name: "Content-Type",
|
||||||
|
value: "text/css"
|
||||||
|
});
|
||||||
|
}
|
||||||
|
return { responseHeaders };
|
||||||
|
},
|
||||||
|
{ urls: ["https://raw.githubusercontent.com/*", "*://*.discord.com/*"], types: ["main_frame", "stylesheet"] },
|
||||||
|
["blocking", "responseHeaders"]
|
||||||
|
);
|
BIN
browser/icon.png
BIN
browser/icon.png
Binary file not shown.
Before Width: | Height: | Size: 21 KiB After Width: | Height: | Size: 1.1 KiB |
@ -21,7 +21,8 @@
|
|||||||
{
|
{
|
||||||
"run_at": "document_start",
|
"run_at": "document_start",
|
||||||
"matches": ["*://*.discord.com/*"],
|
"matches": ["*://*.discord.com/*"],
|
||||||
"js": ["content.js"]
|
"js": ["content.js"],
|
||||||
|
"all_frames": true
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
|
|
||||||
@ -40,12 +41,5 @@
|
|||||||
"path": "modifyResponseHeaders.json"
|
"path": "modifyResponseHeaders.json"
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
},
|
|
||||||
|
|
||||||
"browser_specific_settings": {
|
|
||||||
"gecko": {
|
|
||||||
"id": "vencord-firefox@vendicated.dev",
|
|
||||||
"strict_min_version": "109.0"
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
41
browser/manifestv2.json
Normal file
41
browser/manifestv2.json
Normal file
@ -0,0 +1,41 @@
|
|||||||
|
{
|
||||||
|
"manifest_version": 2,
|
||||||
|
"minimum_chrome_version": "91",
|
||||||
|
|
||||||
|
"name": "Vencord Web",
|
||||||
|
"description": "The cutest Discord mod now in your browser",
|
||||||
|
"author": "Vendicated",
|
||||||
|
"homepage_url": "https://github.com/Vendicated/Vencord",
|
||||||
|
"icons": {
|
||||||
|
"128": "icon.png"
|
||||||
|
},
|
||||||
|
|
||||||
|
"permissions": [
|
||||||
|
"webRequest",
|
||||||
|
"webRequestBlocking",
|
||||||
|
"*://*.discord.com/*",
|
||||||
|
"https://raw.githubusercontent.com/*"
|
||||||
|
],
|
||||||
|
|
||||||
|
"content_scripts": [
|
||||||
|
{
|
||||||
|
"run_at": "document_start",
|
||||||
|
"matches": ["*://*.discord.com/*"],
|
||||||
|
"js": ["content.js"],
|
||||||
|
"all_frames": true
|
||||||
|
}
|
||||||
|
],
|
||||||
|
|
||||||
|
"background": {
|
||||||
|
"scripts": ["background.js"]
|
||||||
|
},
|
||||||
|
|
||||||
|
"web_accessible_resources": ["dist/Vencord.js", "dist/Vencord.css"],
|
||||||
|
|
||||||
|
"browser_specific_settings": {
|
||||||
|
"gecko": {
|
||||||
|
"id": "vencord-firefox@vendicated.dev",
|
||||||
|
"strict_min_version": "91.0"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -15,7 +15,7 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"condition": {
|
"condition": {
|
||||||
"resourceTypes": ["main_frame"]
|
"resourceTypes": ["main_frame", "sub_frame"]
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
@ -1,5 +1,6 @@
|
|||||||
> **Warning**
|
> [!WARNING]
|
||||||
> These instructions are only for advanced users. If you're not a Developer, you should use our [graphical installer](https://github.com/Vendicated/VencordInstaller#usage) instead.
|
> These instructions are only for advanced users. If you're not a Developer, you should use our [graphical installer](https://github.com/Vendicated/VencordInstaller#usage) instead.
|
||||||
|
> No support will be provided for installing in this fashion. If you cannot figure it out, you should just stick to a regular install.
|
||||||
|
|
||||||
# Installation Guide
|
# Installation Guide
|
||||||
|
|
||||||
@ -13,12 +14,6 @@ Welcome to Megu's Installation Guide! In this file, you will learn about how to
|
|||||||
- [Installing Vencord](#installing-vencord)
|
- [Installing Vencord](#installing-vencord)
|
||||||
- [Updating Vencord](#updating-vencord)
|
- [Updating Vencord](#updating-vencord)
|
||||||
- [Uninstalling Vencord](#uninstalling-vencord)
|
- [Uninstalling Vencord](#uninstalling-vencord)
|
||||||
- [Manually Installing Vencord](#manually-installing-vencord)
|
|
||||||
- [On Windows](#on-windows)
|
|
||||||
- [On Linux](#on-linux)
|
|
||||||
- [On MacOS](#on-macos)
|
|
||||||
- [Manual Patching](#manual-patching)
|
|
||||||
- [Manually Uninstalling Vencord](#manually-uninstalling-vencord)
|
|
||||||
|
|
||||||
## Dependencies
|
## Dependencies
|
||||||
|
|
||||||
@ -27,16 +22,16 @@ Welcome to Megu's Installation Guide! In this file, you will learn about how to
|
|||||||
|
|
||||||
## Installing Vencord
|
## Installing Vencord
|
||||||
|
|
||||||
> :exclamation: If this doesn't work, see [Manually Installing Vencord](#manually-installing-vencord)
|
|
||||||
|
|
||||||
Install `pnpm`:
|
Install `pnpm`:
|
||||||
|
|
||||||
> :exclamation: this may need to be run as admin depending on your system, and you may need to close and reopen your terminal.
|
> :exclamation: This next command may need to be run as admin/root depending on your system, and you may need to close and reopen your terminal for pnpm to be in your PATH.
|
||||||
|
|
||||||
```shell
|
```shell
|
||||||
npm i -g pnpm
|
npm i -g pnpm
|
||||||
```
|
```
|
||||||
|
|
||||||
|
> :exclamation: **IMPORTANT** Make sure you aren't using an admin/root terminal from here onwards. It **will** mess up your Discord/Vencord instance and you **will** most likely have to reinstall.
|
||||||
|
|
||||||
Clone Vencord:
|
Clone Vencord:
|
||||||
|
|
||||||
```shell
|
```shell
|
||||||
@ -101,102 +96,4 @@ Simply run:
|
|||||||
pnpm uninject
|
pnpm uninject
|
||||||
```
|
```
|
||||||
|
|
||||||
The above command may ask you to also run:
|
|
||||||
|
|
||||||
```shell
|
|
||||||
pnpm install --frozen-lockfile
|
|
||||||
pnpm uninject
|
|
||||||
```
|
|
||||||
|
|
||||||
## Manually Installing Vencord
|
|
||||||
|
|
||||||
- [Windows](#on-windows)
|
|
||||||
- [Linux](#on-linux)
|
|
||||||
- [MacOS](#on-macos)
|
|
||||||
|
|
||||||
### On Windows
|
|
||||||
|
|
||||||
Press Win+R and enter: `%LocalAppData%` and hit enter. In this page, find the page (Discord, DiscordPTB, DiscordCanary, etc) that you want to patch.
|
|
||||||
|
|
||||||
Now follow the instructions at [Manual Patching](#manual-patching)
|
|
||||||
|
|
||||||
### On Linux
|
|
||||||
|
|
||||||
The Discord folder is usually in one of the following paths:
|
|
||||||
|
|
||||||
- /usr/share
|
|
||||||
- /usr/lib64
|
|
||||||
- /opt
|
|
||||||
- /home/$USER/.local/share
|
|
||||||
|
|
||||||
If you use flatpak, it will usually be in one of the following paths:
|
|
||||||
|
|
||||||
- /var/lib/flatpak/app/com.discordapp.Discord/current/active/files
|
|
||||||
- /home/$USER/.local/share/flatpak/app/com.discordapp.Discord/current/active/files
|
|
||||||
|
|
||||||
You will need to give flatpak access to vencord with one of the following commands:
|
|
||||||
|
|
||||||
> :exclamation: If not on stable, replace `com.discordapp.Discord` with your branch name, e.g., `com.discordapp.DiscordCanary`
|
|
||||||
|
|
||||||
> :exclamation: Replace `/path/to/vencord/` with the path to your vencord folder (NOT the dist folder)
|
|
||||||
|
|
||||||
If Discord flatpak install is in /home/:
|
|
||||||
|
|
||||||
```shell
|
|
||||||
flatpak override --user com.discordapp.Discord --filesystem="/path/to/vencord/"
|
|
||||||
```
|
|
||||||
|
|
||||||
If Discord flatpak install not in /home/:
|
|
||||||
|
|
||||||
```shell
|
|
||||||
sudo flatpak override com.discordapp.Discord --filesystem="/path/to/vencord"
|
|
||||||
```
|
|
||||||
|
|
||||||
Now follow the instructions at [Manual Patching](#manual-patching)
|
|
||||||
|
|
||||||
### On MacOS
|
|
||||||
|
|
||||||
Open finder and go to your Applications folder. Right-Click on the Discord application you want to patch, and view contents.
|
|
||||||
|
|
||||||
Go to the `Contents/Resources` folder.
|
|
||||||
|
|
||||||
Now follow the instructions at [Manual Patching](#manual-patching)
|
|
||||||
|
|
||||||
### Manual Patching
|
|
||||||
|
|
||||||
> :exclamation: If using Flatpak on linux, go to the folder that contains the `app.asar` file, and skip to where we create the `app` folder below.
|
|
||||||
|
|
||||||
> :exclamation: On Linux/MacOS, there's a chance there won't be an `app-<number>` folder, but there probably is a `resources` folder, so keep reading :)
|
|
||||||
|
|
||||||
Inside there, look for the `app-<number>` folders. If you have multiple, use the highest number. If that doesn't work, do it for the rest of the `app-<number>` folders.
|
|
||||||
|
|
||||||
Inside there, go to the `resources` folder. There should be a file called `app.asar`. If there isn't, look at a different `app-<number>` folder instead.
|
|
||||||
|
|
||||||
Make a new folder in `resources` called `app`. In here, we will make two files:
|
|
||||||
|
|
||||||
`package.json` and `index.js`
|
|
||||||
|
|
||||||
In `index.js`:
|
|
||||||
|
|
||||||
> :exclamation: Replace the path in the first line with the path to `patcher.js` in your vencord dist folder.
|
|
||||||
> On Windows, you can get this by shift-rightclicking the patcher.js file and selecting "copy as path"
|
|
||||||
|
|
||||||
```js
|
|
||||||
require("C:/Users/<your user>/path/to/vencord/dist/patcher.js");
|
|
||||||
```
|
|
||||||
|
|
||||||
And in `package.json`:
|
|
||||||
|
|
||||||
```json
|
|
||||||
{ "name": "discord", "main": "index.js" }
|
|
||||||
```
|
|
||||||
|
|
||||||
Finally, fully close & reopen your Discord client and check to see that `Vencord` appears in settings!
|
|
||||||
|
|
||||||
### Manually Uninstalling Vencord
|
|
||||||
|
|
||||||
> :exclamation: Do not delete `app.asar` - Only delete the `app` folder we created.
|
|
||||||
|
|
||||||
Use the instructions above to find the `app` folder, and delete it. Then Close & Reopen Discord.
|
|
||||||
|
|
||||||
If you need more help, ask in the support channel in our [Discord Server](https://discord.gg/D9uwnFnqmd).
|
If you need more help, ask in the support channel in our [Discord Server](https://discord.gg/D9uwnFnqmd).
|
||||||
|
57
package.json
57
package.json
@ -1,9 +1,8 @@
|
|||||||
{
|
{
|
||||||
"name": "vencord",
|
"name": "vencord",
|
||||||
"private": "true",
|
"private": "true",
|
||||||
"version": "1.0.6",
|
"version": "1.4.5",
|
||||||
"description": "The cutest Discord client mod",
|
"description": "The cutest Discord client mod",
|
||||||
"keywords": [],
|
|
||||||
"homepage": "https://github.com/Vendicated/Vencord#readme",
|
"homepage": "https://github.com/Vendicated/Vencord#readme",
|
||||||
"bugs": {
|
"bugs": {
|
||||||
"url": "https://github.com/Vendicated/Vencord/issues"
|
"url": "https://github.com/Vendicated/Vencord/issues"
|
||||||
@ -12,7 +11,7 @@
|
|||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "git+https://github.com/Vendicated/Vencord.git"
|
"url": "git+https://github.com/Vendicated/Vencord.git"
|
||||||
},
|
},
|
||||||
"license": "GPL-3.0",
|
"license": "GPL-3.0-or-later",
|
||||||
"author": "Vendicated",
|
"author": "Vendicated",
|
||||||
"directories": {
|
"directories": {
|
||||||
"doc": "docs"
|
"doc": "docs"
|
||||||
@ -20,53 +19,58 @@
|
|||||||
"scripts": {
|
"scripts": {
|
||||||
"build": "node scripts/build/build.mjs",
|
"build": "node scripts/build/build.mjs",
|
||||||
"buildWeb": "node --require=./scripts/suppressExperimentalWarnings.js scripts/build/buildWeb.mjs",
|
"buildWeb": "node --require=./scripts/suppressExperimentalWarnings.js scripts/build/buildWeb.mjs",
|
||||||
|
"generatePluginJson": "tsx scripts/generatePluginList.ts",
|
||||||
"inject": "node scripts/runInstaller.mjs",
|
"inject": "node scripts/runInstaller.mjs",
|
||||||
"lint": "eslint . --ext .js,.jsx,.ts,.tsx",
|
"lint": "eslint . --ext .js,.jsx,.ts,.tsx --ignore-pattern src/userplugins",
|
||||||
"lint-styles": "stylelint \"src/**/*.css\"",
|
"lint-styles": "stylelint \"src/**/*.css\" --ignore-pattern src/userplugins",
|
||||||
"lint:fix": "pnpm lint --fix",
|
"lint:fix": "pnpm lint --fix",
|
||||||
"test": "pnpm build && pnpm lint && pnpm lint-styles && pnpm testTsc",
|
"test": "pnpm build && pnpm lint && pnpm lint-styles && pnpm testTsc && pnpm generatePluginJson",
|
||||||
"testWeb": "pnpm lint && pnpm buildWeb && pnpm testTsc",
|
"testWeb": "pnpm lint && pnpm buildWeb && pnpm testTsc",
|
||||||
"testTsc": "tsc --noEmit",
|
"testTsc": "tsc --noEmit",
|
||||||
"uninject": "node scripts/runInstaller.mjs",
|
"uninject": "node scripts/runInstaller.mjs",
|
||||||
"watch": "node scripts/build/build.mjs --watch"
|
"watch": "node scripts/build/build.mjs --watch"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
|
"@sapphi-red/web-noise-suppressor": "0.3.3",
|
||||||
"@vap/core": "0.0.12",
|
"@vap/core": "0.0.12",
|
||||||
"@vap/shiki": "0.10.3",
|
"@vap/shiki": "0.10.5",
|
||||||
"fflate": "^0.7.4"
|
"eslint-plugin-simple-header": "^1.0.2",
|
||||||
|
"fflate": "^0.7.4",
|
||||||
|
"nanoid": "^4.0.2",
|
||||||
|
"virtual-merge": "^1.0.1"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@types/diff": "^5.0.2",
|
"@types/diff": "^5.0.3",
|
||||||
"@types/lodash": "^4.14.191",
|
"@types/lodash": "^4.14.194",
|
||||||
"@types/node": "^18.11.18",
|
"@types/node": "^18.16.3",
|
||||||
"@types/react": "^18.0.27",
|
"@types/react": "^18.2.0",
|
||||||
"@types/react-dom": "^18.0.10",
|
"@types/react-dom": "^18.2.1",
|
||||||
"@types/yazl": "^2.4.2",
|
"@types/yazl": "^2.4.2",
|
||||||
"@typescript-eslint/eslint-plugin": "^5.49.0",
|
"@typescript-eslint/eslint-plugin": "^5.59.1",
|
||||||
"@typescript-eslint/parser": "^5.49.0",
|
"@typescript-eslint/parser": "^5.59.1",
|
||||||
"diff": "^5.1.0",
|
"diff": "^5.1.0",
|
||||||
"discord-types": "^1.3.26",
|
"discord-types": "^1.3.26",
|
||||||
"esbuild": "^0.15.18",
|
"esbuild": "^0.15.18",
|
||||||
"eslint": "^8.28.0",
|
"eslint": "^8.46.0",
|
||||||
"eslint-import-resolver-alias": "^1.1.2",
|
"eslint-import-resolver-alias": "^1.1.2",
|
||||||
"eslint-plugin-header": "^3.1.1",
|
|
||||||
"eslint-plugin-path-alias": "^1.0.0",
|
"eslint-plugin-path-alias": "^1.0.0",
|
||||||
"eslint-plugin-simple-import-sort": "^8.0.0",
|
"eslint-plugin-simple-import-sort": "^10.0.0",
|
||||||
"eslint-plugin-unused-imports": "^2.0.0",
|
"eslint-plugin-unused-imports": "^2.0.0",
|
||||||
"highlight.js": "10.6.0",
|
"highlight.js": "10.6.0",
|
||||||
"moment": "^2.29.4",
|
"moment": "^2.29.4",
|
||||||
"puppeteer-core": "^19.6.0",
|
"puppeteer-core": "^19.11.1",
|
||||||
"standalone-electron-types": "^1.0.0",
|
"standalone-electron-types": "^1.0.0",
|
||||||
"stylelint": "^14.16.1",
|
"stylelint": "^15.6.0",
|
||||||
"stylelint-config-standard": "^29.0.0",
|
"stylelint-config-standard": "^33.0.0",
|
||||||
"type-fest": "^3.5.3",
|
"tsx": "^3.12.7",
|
||||||
"typescript": "^4.9.4"
|
"type-fest": "^3.9.0",
|
||||||
|
"typescript": "^5.0.4"
|
||||||
},
|
},
|
||||||
"packageManager": "pnpm@7.13.4",
|
"packageManager": "pnpm@8.1.1",
|
||||||
"pnpm": {
|
"pnpm": {
|
||||||
"patchedDependencies": {
|
"patchedDependencies": {
|
||||||
"eslint-plugin-path-alias@1.0.0": "patches/eslint-plugin-path-alias@1.0.0.patch",
|
"eslint-plugin-path-alias@1.0.0": "patches/eslint-plugin-path-alias@1.0.0.patch",
|
||||||
"eslint@8.28.0": "patches/eslint@8.28.0.patch"
|
"eslint@8.46.0": "patches/eslint@8.46.0.patch"
|
||||||
},
|
},
|
||||||
"peerDependencyRules": {
|
"peerDependencyRules": {
|
||||||
"ignoreMissing": [
|
"ignoreMissing": [
|
||||||
@ -89,6 +93,7 @@
|
|||||||
"sourceDir": "./dist/extension-v2-unpacked"
|
"sourceDir": "./dist/extension-v2-unpacked"
|
||||||
},
|
},
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": ">=18"
|
"node": ">=18",
|
||||||
|
"pnpm": ">=8"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,8 +1,8 @@
|
|||||||
diff --git a/lib/rules/no-useless-escape.js b/lib/rules/no-useless-escape.js
|
diff --git a/lib/rules/no-useless-escape.js b/lib/rules/no-useless-escape.js
|
||||||
index 2046a148a17fd1d5f3a4bbc9f45f7700259d11fa..f4898c6b57355a4fd72c43a9f32bf1a36a6ccf4a 100644
|
index 0e0f6f09f2c35f3276173c08f832cde9f2cf56a0..7dc22851715f3574d935f513c1b5e35552985711 100644
|
||||||
--- a/lib/rules/no-useless-escape.js
|
--- a/lib/rules/no-useless-escape.js
|
||||||
+++ b/lib/rules/no-useless-escape.js
|
+++ b/lib/rules/no-useless-escape.js
|
||||||
@@ -97,12 +97,30 @@ module.exports = {
|
@@ -65,13 +65,31 @@ module.exports = {
|
||||||
escapeBackslash: "Replace the `\\` with `\\\\` to include the actual backslash character."
|
escapeBackslash: "Replace the `\\` with `\\\\` to include the actual backslash character."
|
||||||
},
|
},
|
||||||
|
|
||||||
@ -25,21 +25,25 @@ index 2046a148a17fd1d5f3a4bbc9f45f7700259d11fa..f4898c6b57355a4fd72c43a9f32bf1a3
|
|||||||
|
|
||||||
create(context) {
|
create(context) {
|
||||||
+ const options = context.options[0] || {};
|
+ const options = context.options[0] || {};
|
||||||
+ const { extra, extraCharClass } = options || ''
|
+ const { extra, extraCharClass } = options;
|
||||||
const sourceCode = context.getSourceCode();
|
const sourceCode = context.sourceCode;
|
||||||
|
const parser = new RegExpParser();
|
||||||
|
|
||||||
+ const NON_CHARCLASS_ESCAPES = union(REGEX_NON_CHARCLASS_ESCAPES, new Set(extra))
|
+ const NON_CHARCLASS_ESCAPES = union(REGEX_NON_CHARCLASS_ESCAPES, new Set(extra));
|
||||||
+ const CHARCLASS_ESCAPES = union(REGEX_GENERAL_ESCAPES, new Set(extraCharClass))
|
+ const CHARCLASS_ESCAPES = union(REGEX_GENERAL_ESCAPES, new Set(extraCharClass));
|
||||||
+
|
+
|
||||||
/**
|
/**
|
||||||
* Reports a node
|
* Reports a node
|
||||||
* @param {ASTNode} node The node to report
|
* @param {ASTNode} node The node to report
|
||||||
@@ -238,7 +256,7 @@ module.exports = {
|
@@ -200,9 +218,9 @@ module.exports = {
|
||||||
.filter(charInfo => charInfo.escaped)
|
let allowedEscapes;
|
||||||
|
|
||||||
// Filter out characters that are valid to escape, based on their position in the regular expression.
|
if (characterClassStack.length) {
|
||||||
- .filter(charInfo => !(charInfo.inCharClass ? REGEX_GENERAL_ESCAPES : REGEX_NON_CHARCLASS_ESCAPES).has(charInfo.text))
|
- allowedEscapes = unicodeSets ? REGEX_CLASSSET_CHARACTER_ESCAPES : REGEX_GENERAL_ESCAPES;
|
||||||
+ .filter(charInfo => !(charInfo.inCharClass ? CHARCLASS_ESCAPES : NON_CHARCLASS_ESCAPES).has(charInfo.text))
|
+ allowedEscapes = unicodeSets ? REGEX_CLASSSET_CHARACTER_ESCAPES : CHARCLASS_ESCAPES;
|
||||||
|
} else {
|
||||||
// Report all the remaining characters.
|
- allowedEscapes = REGEX_NON_CHARCLASS_ESCAPES;
|
||||||
.forEach(charInfo => report(node, charInfo.index, charInfo.text));
|
+ allowedEscapes = NON_CHARCLASS_ESCAPES;
|
||||||
|
}
|
||||||
|
if (allowedEscapes.has(escapedChar)) {
|
||||||
|
return;
|
1858
pnpm-lock.yaml
generated
1858
pnpm-lock.yaml
generated
File diff suppressed because it is too large
Load Diff
@ -19,11 +19,14 @@
|
|||||||
|
|
||||||
import esbuild from "esbuild";
|
import esbuild from "esbuild";
|
||||||
|
|
||||||
import { commonOpts, globPlugins, isStandalone, watch } from "./common.mjs";
|
import { BUILD_TIMESTAMP, commonOpts, globPlugins, isStandalone, updaterDisabled, VERSION, watch } from "./common.mjs";
|
||||||
|
|
||||||
const defines = {
|
const defines = {
|
||||||
IS_STANDALONE: isStandalone,
|
IS_STANDALONE: isStandalone,
|
||||||
IS_DEV: JSON.stringify(watch)
|
IS_DEV: JSON.stringify(watch),
|
||||||
|
IS_UPDATER_DISABLED: updaterDisabled,
|
||||||
|
VERSION: JSON.stringify(VERSION),
|
||||||
|
BUILD_TIMESTAMP,
|
||||||
};
|
};
|
||||||
if (defines.IS_STANDALONE === "false")
|
if (defines.IS_STANDALONE === "false")
|
||||||
// If this is a local build (not standalone), optimise
|
// If this is a local build (not standalone), optimise
|
||||||
@ -38,8 +41,6 @@ const nodeCommonOpts = {
|
|||||||
format: "cjs",
|
format: "cjs",
|
||||||
platform: "node",
|
platform: "node",
|
||||||
target: ["esnext"],
|
target: ["esnext"],
|
||||||
minify: true,
|
|
||||||
bundle: true,
|
|
||||||
external: ["electron", ...commonOpts.external],
|
external: ["electron", ...commonOpts.external],
|
||||||
define: defines,
|
define: defines,
|
||||||
};
|
};
|
||||||
@ -48,19 +49,18 @@ const sourceMapFooter = s => watch ? "" : `//# sourceMappingURL=vencord://${s}.j
|
|||||||
const sourcemap = watch ? "inline" : "external";
|
const sourcemap = watch ? "inline" : "external";
|
||||||
|
|
||||||
await Promise.all([
|
await Promise.all([
|
||||||
|
// Discord Desktop main & renderer & preload
|
||||||
esbuild.build({
|
esbuild.build({
|
||||||
...nodeCommonOpts,
|
...nodeCommonOpts,
|
||||||
entryPoints: ["src/preload.ts"],
|
entryPoints: ["src/main/index.ts"],
|
||||||
outfile: "dist/preload.js",
|
|
||||||
footer: { js: "//# sourceURL=VencordPreload\n" + sourceMapFooter("preload") },
|
|
||||||
sourcemap,
|
|
||||||
}),
|
|
||||||
esbuild.build({
|
|
||||||
...nodeCommonOpts,
|
|
||||||
entryPoints: ["src/patcher.ts"],
|
|
||||||
outfile: "dist/patcher.js",
|
outfile: "dist/patcher.js",
|
||||||
footer: { js: "//# sourceURL=VencordPatcher\n" + sourceMapFooter("patcher") },
|
footer: { js: "//# sourceURL=VencordPatcher\n" + sourceMapFooter("patcher") },
|
||||||
sourcemap,
|
sourcemap,
|
||||||
|
define: {
|
||||||
|
...defines,
|
||||||
|
IS_DISCORD_DESKTOP: true,
|
||||||
|
IS_VESKTOP: false
|
||||||
|
}
|
||||||
}),
|
}),
|
||||||
esbuild.build({
|
esbuild.build({
|
||||||
...commonOpts,
|
...commonOpts,
|
||||||
@ -72,12 +72,72 @@ await Promise.all([
|
|||||||
globalName: "Vencord",
|
globalName: "Vencord",
|
||||||
sourcemap,
|
sourcemap,
|
||||||
plugins: [
|
plugins: [
|
||||||
globPlugins,
|
globPlugins("discordDesktop"),
|
||||||
...commonOpts.plugins
|
...commonOpts.plugins
|
||||||
],
|
],
|
||||||
define: {
|
define: {
|
||||||
...defines,
|
...defines,
|
||||||
IS_WEB: false
|
IS_WEB: false,
|
||||||
|
IS_DISCORD_DESKTOP: true,
|
||||||
|
IS_VESKTOP: false
|
||||||
|
}
|
||||||
|
}),
|
||||||
|
esbuild.build({
|
||||||
|
...nodeCommonOpts,
|
||||||
|
entryPoints: ["src/preload.ts"],
|
||||||
|
outfile: "dist/preload.js",
|
||||||
|
footer: { js: "//# sourceURL=VencordPreload\n" + sourceMapFooter("preload") },
|
||||||
|
sourcemap,
|
||||||
|
define: {
|
||||||
|
...defines,
|
||||||
|
IS_DISCORD_DESKTOP: true,
|
||||||
|
IS_VESKTOP: false
|
||||||
|
}
|
||||||
|
}),
|
||||||
|
|
||||||
|
// Vencord Desktop main & renderer & preload
|
||||||
|
esbuild.build({
|
||||||
|
...nodeCommonOpts,
|
||||||
|
entryPoints: ["src/main/index.ts"],
|
||||||
|
outfile: "dist/vencordDesktopMain.js",
|
||||||
|
footer: { js: "//# sourceURL=VencordDesktopMain\n" + sourceMapFooter("vencordDesktopMain") },
|
||||||
|
sourcemap,
|
||||||
|
define: {
|
||||||
|
...defines,
|
||||||
|
IS_DISCORD_DESKTOP: false,
|
||||||
|
IS_VESKTOP: true
|
||||||
|
}
|
||||||
|
}),
|
||||||
|
esbuild.build({
|
||||||
|
...commonOpts,
|
||||||
|
entryPoints: ["src/Vencord.ts"],
|
||||||
|
outfile: "dist/vencordDesktopRenderer.js",
|
||||||
|
format: "iife",
|
||||||
|
target: ["esnext"],
|
||||||
|
footer: { js: "//# sourceURL=VencordDesktopRenderer\n" + sourceMapFooter("vencordDesktopRenderer") },
|
||||||
|
globalName: "Vencord",
|
||||||
|
sourcemap,
|
||||||
|
plugins: [
|
||||||
|
globPlugins("vencordDesktop"),
|
||||||
|
...commonOpts.plugins
|
||||||
|
],
|
||||||
|
define: {
|
||||||
|
...defines,
|
||||||
|
IS_WEB: false,
|
||||||
|
IS_DISCORD_DESKTOP: false,
|
||||||
|
IS_VESKTOP: true
|
||||||
|
}
|
||||||
|
}),
|
||||||
|
esbuild.build({
|
||||||
|
...nodeCommonOpts,
|
||||||
|
entryPoints: ["src/preload.ts"],
|
||||||
|
outfile: "dist/vencordDesktopPreload.js",
|
||||||
|
footer: { js: "//# sourceURL=VencordPreload\n" + sourceMapFooter("vencordDesktopPreload") },
|
||||||
|
sourcemap,
|
||||||
|
define: {
|
||||||
|
...defines,
|
||||||
|
IS_DISCORD_DESKTOP: false,
|
||||||
|
IS_VESKTOP: true
|
||||||
}
|
}
|
||||||
}),
|
}),
|
||||||
]).catch(err => {
|
]).catch(err => {
|
||||||
|
@ -24,9 +24,7 @@ import { readFileSync } from "fs";
|
|||||||
import { appendFile, mkdir, readFile, rm, writeFile } from "fs/promises";
|
import { appendFile, mkdir, readFile, rm, writeFile } from "fs/promises";
|
||||||
import { join } from "path";
|
import { join } from "path";
|
||||||
|
|
||||||
// wtf is this assert syntax
|
import { BUILD_TIMESTAMP, commonOpts, globPlugins, VERSION, watch } from "./common.mjs";
|
||||||
import PackageJSON from "../../package.json" assert { type: "json" };
|
|
||||||
import { commonOpts, globPlugins, watch } from "./common.mjs";
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @type {esbuild.BuildOptions}
|
* @type {esbuild.BuildOptions}
|
||||||
@ -36,16 +34,21 @@ const commonOptions = {
|
|||||||
entryPoints: ["browser/Vencord.ts"],
|
entryPoints: ["browser/Vencord.ts"],
|
||||||
globalName: "Vencord",
|
globalName: "Vencord",
|
||||||
format: "iife",
|
format: "iife",
|
||||||
external: ["plugins", "git-hash"],
|
external: ["plugins", "git-hash", "/assets/*"],
|
||||||
plugins: [
|
plugins: [
|
||||||
globPlugins,
|
globPlugins("web"),
|
||||||
...commonOpts.plugins,
|
...commonOpts.plugins,
|
||||||
],
|
],
|
||||||
target: ["esnext"],
|
target: ["esnext"],
|
||||||
define: {
|
define: {
|
||||||
IS_WEB: "true",
|
IS_WEB: "true",
|
||||||
IS_STANDALONE: "true",
|
IS_STANDALONE: "true",
|
||||||
IS_DEV: JSON.stringify(watch)
|
IS_DEV: JSON.stringify(watch),
|
||||||
|
IS_DISCORD_DESKTOP: "false",
|
||||||
|
IS_VESKTOP: "false",
|
||||||
|
IS_UPDATER_DISABLED: "true",
|
||||||
|
VERSION: JSON.stringify(VERSION),
|
||||||
|
BUILD_TIMESTAMP,
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -65,7 +68,7 @@ await Promise.all(
|
|||||||
},
|
},
|
||||||
outfile: "dist/Vencord.user.js",
|
outfile: "dist/Vencord.user.js",
|
||||||
banner: {
|
banner: {
|
||||||
js: readFileSync("browser/userscript.meta.js", "utf-8").replace("%version%", `${PackageJSON.version}.${new Date().getTime()}`)
|
js: readFileSync("browser/userscript.meta.js", "utf-8").replace("%version%", `${VERSION}.${new Date().getTime()}`)
|
||||||
},
|
},
|
||||||
footer: {
|
footer: {
|
||||||
// UserScripts get wrapped in an iife, so define Vencord prop on window that returns our local
|
// UserScripts get wrapped in an iife, so define Vencord prop on window that returns our local
|
||||||
@ -86,7 +89,7 @@ async function buildPluginZip(target, files, shouldZip) {
|
|||||||
let content = await readFile(join("browser", f));
|
let content = await readFile(join("browser", f));
|
||||||
if (f.startsWith("manifest")) {
|
if (f.startsWith("manifest")) {
|
||||||
const json = JSON.parse(content.toString("utf-8"));
|
const json = JSON.parse(content.toString("utf-8"));
|
||||||
json.version = PackageJSON.version;
|
json.version = VERSION;
|
||||||
content = new TextEncoder().encode(JSON.stringify(json));
|
content = new TextEncoder().encode(JSON.stringify(json));
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -140,6 +143,7 @@ const appendCssRuntime = readFile("dist/Vencord.user.css", "utf-8").then(content
|
|||||||
await Promise.all([
|
await Promise.all([
|
||||||
appendCssRuntime,
|
appendCssRuntime,
|
||||||
buildPluginZip("extension.zip", ["modifyResponseHeaders.json", "content.js", "manifest.json", "icon.png"], true),
|
buildPluginZip("extension.zip", ["modifyResponseHeaders.json", "content.js", "manifest.json", "icon.png"], true),
|
||||||
buildPluginZip("extension-unpacked", ["modifyResponseHeaders.json", "content.js", "manifest.json", "icon.png"], false),
|
buildPluginZip("chromium-unpacked", ["modifyResponseHeaders.json", "content.js", "manifest.json", "icon.png"], false),
|
||||||
|
buildPluginZip("firefox-unpacked", ["background.js", "content.js", "manifestv2.json", "icon.png"], false),
|
||||||
]);
|
]);
|
||||||
|
|
||||||
|
@ -16,23 +16,37 @@
|
|||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
import "../suppressExperimentalWarnings.js";
|
||||||
|
import "../checkNodeVersion.js";
|
||||||
|
|
||||||
import { exec, execSync } from "child_process";
|
import { exec, execSync } from "child_process";
|
||||||
import { existsSync, readFileSync } from "fs";
|
import { existsSync, readFileSync } from "fs";
|
||||||
import { readdir, readFile } from "fs/promises";
|
import { readdir, readFile } from "fs/promises";
|
||||||
import { join, relative } from "path";
|
import { join, relative } from "path";
|
||||||
import { promisify } from "util";
|
import { promisify } from "util";
|
||||||
|
|
||||||
|
// wtf is this assert syntax
|
||||||
|
import PackageJSON from "../../package.json" assert { type: "json" };
|
||||||
|
import { getPluginTarget } from "../utils.mjs";
|
||||||
|
|
||||||
|
export const VERSION = PackageJSON.version;
|
||||||
|
// https://reproducible-builds.org/docs/source-date-epoch/
|
||||||
|
export const BUILD_TIMESTAMP = Number(process.env.SOURCE_DATE_EPOCH) || Date.now();
|
||||||
export const watch = process.argv.includes("--watch");
|
export const watch = process.argv.includes("--watch");
|
||||||
export const isStandalone = JSON.stringify(process.argv.includes("--standalone"));
|
export const isStandalone = JSON.stringify(process.argv.includes("--standalone"));
|
||||||
export const gitHash = execSync("git rev-parse --short HEAD", { encoding: "utf-8" }).trim();
|
export const updaterDisabled = JSON.stringify(process.argv.includes("--disable-updater"));
|
||||||
|
export const gitHash = process.env.VENCORD_HASH || execSync("git rev-parse --short HEAD", { encoding: "utf-8" }).trim();
|
||||||
export const banner = {
|
export const banner = {
|
||||||
js: `
|
js: `
|
||||||
// Vencord ${gitHash}
|
// Vencord ${gitHash}
|
||||||
// Standalone: ${isStandalone}
|
// Standalone: ${isStandalone}
|
||||||
// Platform: ${isStandalone === "false" ? process.platform : "Universal"}
|
// Platform: ${isStandalone === "false" ? process.platform : "Universal"}
|
||||||
|
// Updater disabled: ${updaterDisabled}
|
||||||
`.trim()
|
`.trim()
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const isWeb = process.argv.slice(0, 2).some(f => f.endsWith("buildWeb.mjs"));
|
||||||
|
|
||||||
// https://github.com/evanw/esbuild/issues/619#issuecomment-751995294
|
// https://github.com/evanw/esbuild/issues/619#issuecomment-751995294
|
||||||
/**
|
/**
|
||||||
* @type {import("esbuild").Plugin}
|
* @type {import("esbuild").Plugin}
|
||||||
@ -46,9 +60,9 @@ export const makeAllPackagesExternalPlugin = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @type {import("esbuild").Plugin}
|
* @type {(kind: "web" | "discordDesktop" | "vencordDesktop") => import("esbuild").Plugin}
|
||||||
*/
|
*/
|
||||||
export const globPlugins = {
|
export const globPlugins = kind => ({
|
||||||
name: "glob-plugins",
|
name: "glob-plugins",
|
||||||
setup: build => {
|
setup: build => {
|
||||||
const filter = /^~plugins$/;
|
const filter = /^~plugins$/;
|
||||||
@ -60,7 +74,7 @@ export const globPlugins = {
|
|||||||
});
|
});
|
||||||
|
|
||||||
build.onLoad({ filter, namespace: "import-plugins" }, async () => {
|
build.onLoad({ filter, namespace: "import-plugins" }, async () => {
|
||||||
const pluginDirs = ["plugins", "userplugins"];
|
const pluginDirs = ["plugins/_api", "plugins/_core", "plugins", "userplugins"];
|
||||||
let code = "";
|
let code = "";
|
||||||
let plugins = "\n";
|
let plugins = "\n";
|
||||||
let i = 0;
|
let i = 0;
|
||||||
@ -68,10 +82,18 @@ export const globPlugins = {
|
|||||||
if (!existsSync(`./src/${dir}`)) continue;
|
if (!existsSync(`./src/${dir}`)) continue;
|
||||||
const files = await readdir(`./src/${dir}`);
|
const files = await readdir(`./src/${dir}`);
|
||||||
for (const file of files) {
|
for (const file of files) {
|
||||||
if (file.startsWith(".")) continue;
|
if (file.startsWith("_") || file.startsWith(".")) continue;
|
||||||
if (file === "index.ts") {
|
if (file === "index.ts") continue;
|
||||||
continue;
|
|
||||||
|
const target = getPluginTarget(file);
|
||||||
|
if (target) {
|
||||||
|
if (target === "dev" && !watch) continue;
|
||||||
|
if (target === "web" && kind === "discordDesktop") continue;
|
||||||
|
if (target === "desktop" && kind === "web") continue;
|
||||||
|
if (target === "discordDesktop" && kind !== "discordDesktop") continue;
|
||||||
|
if (target === "vencordDesktop" && kind !== "vencordDesktop") continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
const mod = `p${i}`;
|
const mod = `p${i}`;
|
||||||
code += `import ${mod} from "./${dir}/${file.replace(/\.tsx?$/, "")}";\n`;
|
code += `import ${mod} from "./${dir}/${file.replace(/\.tsx?$/, "")}";\n`;
|
||||||
plugins += `[${mod}.name]:${mod},\n`;
|
plugins += `[${mod}.name]:${mod},\n`;
|
||||||
@ -85,7 +107,7 @@ export const globPlugins = {
|
|||||||
};
|
};
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
};
|
});
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @type {import("esbuild").Plugin}
|
* @type {import("esbuild").Plugin}
|
||||||
@ -114,11 +136,14 @@ export const gitRemotePlugin = {
|
|||||||
namespace: "git-remote", path: args.path
|
namespace: "git-remote", path: args.path
|
||||||
}));
|
}));
|
||||||
build.onLoad({ filter, namespace: "git-remote" }, async () => {
|
build.onLoad({ filter, namespace: "git-remote" }, async () => {
|
||||||
const res = await promisify(exec)("git remote get-url origin", { encoding: "utf-8" });
|
let remote = process.env.VENCORD_REMOTE;
|
||||||
const remote = res.stdout.trim()
|
if (!remote) {
|
||||||
.replace("https://github.com/", "")
|
const res = await promisify(exec)("git remote get-url origin", { encoding: "utf-8" });
|
||||||
.replace("git@github.com:", "")
|
remote = res.stdout.trim()
|
||||||
.replace(/.git$/, "");
|
.replace("https://github.com/", "")
|
||||||
|
.replace("git@github.com:", "")
|
||||||
|
.replace(/.git$/, "");
|
||||||
|
}
|
||||||
|
|
||||||
return { contents: `export default "${remote}"` };
|
return { contents: `export default "${remote}"` };
|
||||||
});
|
});
|
||||||
@ -185,7 +210,7 @@ export const commonOpts = {
|
|||||||
legalComments: "linked",
|
legalComments: "linked",
|
||||||
banner,
|
banner,
|
||||||
plugins: [fileIncludePlugin, gitHashPlugin, gitRemotePlugin, stylePlugin],
|
plugins: [fileIncludePlugin, gitHashPlugin, gitRemotePlugin, stylePlugin],
|
||||||
external: ["~plugins", "~git-hash", "~git-remote"],
|
external: ["~plugins", "~git-hash", "~git-remote", "/assets/*"],
|
||||||
inject: ["./scripts/build/inject/react.mjs"],
|
inject: ["./scripts/build/inject/react.mjs"],
|
||||||
jsxFactory: "VencordCreateElement",
|
jsxFactory: "VencordCreateElement",
|
||||||
jsxFragment: "VencordFragment",
|
jsxFragment: "VencordFragment",
|
||||||
|
@ -1,62 +0,0 @@
|
|||||||
/*
|
|
||||||
* Vencord, a modification for Discord's desktop app
|
|
||||||
* Copyright (c) 2022 Vendicated and contributors
|
|
||||||
*
|
|
||||||
* This program is free software: you can redistribute it and/or modify
|
|
||||||
* it under the terms of the GNU General Public License as published by
|
|
||||||
* the Free Software Foundation, either version 3 of the License, or
|
|
||||||
* (at your option) any later version.
|
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
||||||
* GNU General Public License for more details.
|
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU General Public License
|
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
||||||
*/
|
|
||||||
|
|
||||||
// A script to automatically generate a list of all plugins.
|
|
||||||
// Just copy paste the entire file into a running Vencord install and it will prompt you
|
|
||||||
// to save the file
|
|
||||||
|
|
||||||
// eslint-disable-next-line spaced-comment
|
|
||||||
/// <reference types="../src/modules"/>
|
|
||||||
|
|
||||||
(() => {
|
|
||||||
/**
|
|
||||||
* @type {typeof import("~plugins").default}
|
|
||||||
*/
|
|
||||||
const Plugins = Vencord.Plugins.plugins;
|
|
||||||
|
|
||||||
const header = `
|
|
||||||
<!-- This file is auto generated, do not edit -->
|
|
||||||
|
|
||||||
# Vencord Plugins
|
|
||||||
`;
|
|
||||||
|
|
||||||
let tableOfContents = "\n\n";
|
|
||||||
|
|
||||||
let list = "\n\n";
|
|
||||||
|
|
||||||
for (const p of Object.values(Plugins).sort((a, b) => a.name.localeCompare(b.name))) {
|
|
||||||
tableOfContents += `- [${p.name}](#${p.name.replaceAll(" ", "-")})\n`;
|
|
||||||
|
|
||||||
list += `## ${p.name}
|
|
||||||
|
|
||||||
${p.description}
|
|
||||||
|
|
||||||
**Authors**: ${p.authors.map(a => a.name).join(", ")}
|
|
||||||
`;
|
|
||||||
|
|
||||||
if (p.commands?.length) {
|
|
||||||
list += "\n\n#### Commands\n";
|
|
||||||
for (const cmd of p.commands) {
|
|
||||||
list += `${cmd.name} - ${cmd.description}\n\n`;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
list += "\n\n";
|
|
||||||
}
|
|
||||||
|
|
||||||
copy(header + tableOfContents + list);
|
|
||||||
})();
|
|
212
scripts/generatePluginList.ts
Normal file
212
scripts/generatePluginList.ts
Normal file
@ -0,0 +1,212 @@
|
|||||||
|
/*
|
||||||
|
* Vencord, a modification for Discord's desktop app
|
||||||
|
* Copyright (c) 2023 Vendicated and contributors
|
||||||
|
*
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { Dirent, readdirSync, readFileSync, writeFileSync } from "fs";
|
||||||
|
import { access, readFile } from "fs/promises";
|
||||||
|
import { join } from "path";
|
||||||
|
import { BigIntLiteral, createSourceFile, Identifier, isArrayLiteralExpression, isCallExpression, isExportAssignment, isIdentifier, isObjectLiteralExpression, isPropertyAccessExpression, isPropertyAssignment, isSatisfiesExpression, isStringLiteral, isVariableStatement, NamedDeclaration, NodeArray, ObjectLiteralExpression, ScriptTarget, StringLiteral, SyntaxKind } from "typescript";
|
||||||
|
|
||||||
|
import { getPluginTarget } from "./utils.mjs";
|
||||||
|
|
||||||
|
interface Dev {
|
||||||
|
name: string;
|
||||||
|
id: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface PluginData {
|
||||||
|
name: string;
|
||||||
|
description: string;
|
||||||
|
tags: string[];
|
||||||
|
authors: Dev[];
|
||||||
|
dependencies: string[];
|
||||||
|
hasPatches: boolean;
|
||||||
|
hasCommands: boolean;
|
||||||
|
required: boolean;
|
||||||
|
enabledByDefault: boolean;
|
||||||
|
target: "discordDesktop" | "vencordDesktop" | "web" | "dev";
|
||||||
|
}
|
||||||
|
|
||||||
|
const devs = {} as Record<string, Dev>;
|
||||||
|
|
||||||
|
function getName(node: NamedDeclaration) {
|
||||||
|
return node.name && isIdentifier(node.name) ? node.name.text : undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
function hasName(node: NamedDeclaration, name: string) {
|
||||||
|
return getName(node) === name;
|
||||||
|
}
|
||||||
|
|
||||||
|
function getObjectProp(node: ObjectLiteralExpression, name: string) {
|
||||||
|
const prop = node.properties.find(p => hasName(p, name));
|
||||||
|
if (prop && isPropertyAssignment(prop)) return prop.initializer;
|
||||||
|
return prop;
|
||||||
|
}
|
||||||
|
|
||||||
|
function parseDevs() {
|
||||||
|
const file = createSourceFile("constants.ts", readFileSync("src/utils/constants.ts", "utf8"), ScriptTarget.Latest);
|
||||||
|
|
||||||
|
for (const child of file.getChildAt(0).getChildren()) {
|
||||||
|
if (!isVariableStatement(child)) continue;
|
||||||
|
|
||||||
|
const devsDeclaration = child.declarationList.declarations.find(d => hasName(d, "Devs"));
|
||||||
|
if (!devsDeclaration?.initializer || !isCallExpression(devsDeclaration.initializer)) continue;
|
||||||
|
|
||||||
|
const value = devsDeclaration.initializer.arguments[0];
|
||||||
|
|
||||||
|
if (!isSatisfiesExpression(value) || !isObjectLiteralExpression(value.expression)) throw new Error("Failed to parse devs: not an object literal");
|
||||||
|
|
||||||
|
for (const prop of value.expression.properties) {
|
||||||
|
const name = (prop.name as Identifier).text;
|
||||||
|
const value = isPropertyAssignment(prop) ? prop.initializer : prop;
|
||||||
|
|
||||||
|
if (!isObjectLiteralExpression(value)) throw new Error(`Failed to parse devs: ${name} is not an object literal`);
|
||||||
|
|
||||||
|
devs[name] = {
|
||||||
|
name: (getObjectProp(value, "name") as StringLiteral).text,
|
||||||
|
id: (getObjectProp(value, "id") as BigIntLiteral).text.slice(0, -1)
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
throw new Error("Could not find Devs constant");
|
||||||
|
}
|
||||||
|
|
||||||
|
async function parseFile(fileName: string) {
|
||||||
|
const file = createSourceFile(fileName, await readFile(fileName, "utf8"), ScriptTarget.Latest);
|
||||||
|
|
||||||
|
const fail = (reason: string) => {
|
||||||
|
return new Error(`Invalid plugin ${fileName}, because ${reason}`);
|
||||||
|
};
|
||||||
|
|
||||||
|
for (const node of file.getChildAt(0).getChildren()) {
|
||||||
|
if (!isExportAssignment(node) || !isCallExpression(node.expression)) continue;
|
||||||
|
|
||||||
|
const call = node.expression;
|
||||||
|
if (!isIdentifier(call.expression) || call.expression.text !== "definePlugin") continue;
|
||||||
|
|
||||||
|
const pluginObj = node.expression.arguments[0];
|
||||||
|
if (!isObjectLiteralExpression(pluginObj)) throw fail("no object literal passed to definePlugin");
|
||||||
|
|
||||||
|
const data = {
|
||||||
|
hasPatches: false,
|
||||||
|
hasCommands: false,
|
||||||
|
enabledByDefault: false,
|
||||||
|
required: false,
|
||||||
|
tags: [] as string[]
|
||||||
|
} as PluginData;
|
||||||
|
|
||||||
|
for (const prop of pluginObj.properties) {
|
||||||
|
const key = getName(prop);
|
||||||
|
const value = isPropertyAssignment(prop) ? prop.initializer : prop;
|
||||||
|
|
||||||
|
switch (key) {
|
||||||
|
case "name":
|
||||||
|
case "description":
|
||||||
|
if (!isStringLiteral(value)) throw fail(`${key} is not a string literal`);
|
||||||
|
data[key] = value.text;
|
||||||
|
break;
|
||||||
|
case "patches":
|
||||||
|
data.hasPatches = true;
|
||||||
|
break;
|
||||||
|
case "commands":
|
||||||
|
data.hasCommands = true;
|
||||||
|
break;
|
||||||
|
case "authors":
|
||||||
|
if (!isArrayLiteralExpression(value)) throw fail("authors is not an array literal");
|
||||||
|
data.authors = value.elements.map(e => {
|
||||||
|
if (!isPropertyAccessExpression(e)) throw fail("authors array contains non-property access expressions");
|
||||||
|
const d = devs[getName(e)!];
|
||||||
|
if (!d) throw fail(`couldn't look up author ${getName(e)}`);
|
||||||
|
return d;
|
||||||
|
});
|
||||||
|
break;
|
||||||
|
case "tags":
|
||||||
|
if (!isArrayLiteralExpression(value)) throw fail("tags is not an array literal");
|
||||||
|
data.tags = value.elements.map(e => {
|
||||||
|
if (!isStringLiteral(e)) throw fail("tags array contains non-string literals");
|
||||||
|
return e.text;
|
||||||
|
});
|
||||||
|
break;
|
||||||
|
case "dependencies":
|
||||||
|
if (!isArrayLiteralExpression(value)) throw fail("dependencies is not an array literal");
|
||||||
|
const { elements } = value;
|
||||||
|
if (elements.some(e => !isStringLiteral(e))) throw fail("dependencies array contains non-string elements");
|
||||||
|
data.dependencies = (elements as NodeArray<StringLiteral>).map(e => e.text);
|
||||||
|
break;
|
||||||
|
case "required":
|
||||||
|
case "enabledByDefault":
|
||||||
|
data[key] = value.kind === SyntaxKind.TrueKeyword;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!data.name || !data.description || !data.authors) throw fail("name, description or authors are missing");
|
||||||
|
|
||||||
|
const target = getPluginTarget(fileName);
|
||||||
|
if (target) {
|
||||||
|
if (!["web", "discordDesktop", "vencordDesktop", "desktop", "dev"].includes(target)) throw fail(`invalid target ${target}`);
|
||||||
|
data.target = target as any;
|
||||||
|
}
|
||||||
|
|
||||||
|
return data;
|
||||||
|
}
|
||||||
|
|
||||||
|
throw fail("no default export called 'definePlugin' found");
|
||||||
|
}
|
||||||
|
|
||||||
|
async function getEntryPoint(dir: string, dirent: Dirent) {
|
||||||
|
const base = join(dir, dirent.name);
|
||||||
|
if (!dirent.isDirectory()) return base;
|
||||||
|
|
||||||
|
for (const name of ["index.ts", "index.tsx"]) {
|
||||||
|
const full = join(base, name);
|
||||||
|
try {
|
||||||
|
await access(full);
|
||||||
|
return full;
|
||||||
|
} catch { }
|
||||||
|
}
|
||||||
|
|
||||||
|
throw new Error(`${dirent.name}: Couldn't find entry point`);
|
||||||
|
}
|
||||||
|
|
||||||
|
function isPluginFile({ name }: { name: string; }) {
|
||||||
|
if (name === "index.ts") return false;
|
||||||
|
return !name.startsWith("_") && !name.startsWith(".");
|
||||||
|
}
|
||||||
|
|
||||||
|
(async () => {
|
||||||
|
parseDevs();
|
||||||
|
|
||||||
|
const plugins = ["src/plugins", "src/plugins/_core"].flatMap(dir =>
|
||||||
|
readdirSync(dir, { withFileTypes: true })
|
||||||
|
.filter(isPluginFile)
|
||||||
|
.map(async dirent =>
|
||||||
|
parseFile(await getEntryPoint(dir, dirent))
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
const data = JSON.stringify(await Promise.all(plugins));
|
||||||
|
|
||||||
|
if (process.argv.length > 2) {
|
||||||
|
writeFileSync(process.argv[2], data);
|
||||||
|
} else {
|
||||||
|
console.log(data);
|
||||||
|
}
|
||||||
|
})();
|
@ -130,7 +130,7 @@ async function printReport() {
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
title: "Discord Errors",
|
title: "Discord Errors",
|
||||||
description: toCodeBlock(report.otherErrors.join("\n")),
|
description: report.otherErrors.length ? toCodeBlock(report.otherErrors.join("\n")) : "None",
|
||||||
color: report.otherErrors.length ? 0xff0000 : 0x00ff00
|
color: report.otherErrors.length ? 0xff0000 : 0x00ff00
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
@ -186,8 +186,18 @@ page.on("console", async e => {
|
|||||||
} else if (isDebug) {
|
} else if (isDebug) {
|
||||||
console.error(e.text());
|
console.error(e.text());
|
||||||
} else if (level === "error") {
|
} else if (level === "error") {
|
||||||
const text = e.text();
|
const text = await Promise.all(
|
||||||
if (!text.startsWith("Failed to load resource: the server responded with a status of")) {
|
e.args().map(async a => {
|
||||||
|
try {
|
||||||
|
return await maybeGetError(a) || await a.jsonValue();
|
||||||
|
} catch (e) {
|
||||||
|
return a.toString();
|
||||||
|
}
|
||||||
|
})
|
||||||
|
).then(a => a.join(" ").trim());
|
||||||
|
|
||||||
|
|
||||||
|
if (text.length && !text.startsWith("Failed to load resource: the server responded with a status of")) {
|
||||||
console.error("Got unexpected error", text);
|
console.error("Got unexpected error", text);
|
||||||
report.otherErrors.push(text);
|
report.otherErrors.push(text);
|
||||||
}
|
}
|
||||||
@ -224,7 +234,7 @@ function runTime(token: string) {
|
|||||||
// Needs native server to run
|
// Needs native server to run
|
||||||
if (p.name === "WebRichPresence (arRPC)") return;
|
if (p.name === "WebRichPresence (arRPC)") return;
|
||||||
|
|
||||||
p.required = true;
|
Vencord.Settings.plugins[p.name].enabled = true;
|
||||||
p.patches?.forEach(patch => {
|
p.patches?.forEach(patch => {
|
||||||
patch.plugin = p.name;
|
patch.plugin = p.name;
|
||||||
delete patch.predicate;
|
delete patch.predicate;
|
||||||
@ -253,12 +263,12 @@ function runTime(token: string) {
|
|||||||
for (const id in ids) {
|
for (const id in ids) {
|
||||||
const isWasm = await fetch(wreq.p + wreq.u(id))
|
const isWasm = await fetch(wreq.p + wreq.u(id))
|
||||||
.then(r => r.text())
|
.then(r => r.text())
|
||||||
.then(t => t.includes(".module.wasm"));
|
.then(t => t.includes(".module.wasm") || !t.includes("(this.webpackChunkdiscord_app=this.webpackChunkdiscord_app||[]).push"));
|
||||||
|
|
||||||
if (!isWasm)
|
if (!isWasm)
|
||||||
await wreq.e(id as any);
|
await wreq.e(id as any);
|
||||||
|
|
||||||
await new Promise(r => setTimeout(r, 100));
|
await new Promise(r => setTimeout(r, 150));
|
||||||
}
|
}
|
||||||
console.error("[PUP_DEBUG]", "Finished loading chunks!");
|
console.error("[PUP_DEBUG]", "Finished loading chunks!");
|
||||||
|
|
3
scripts/header-new.txt
Normal file
3
scripts/header-new.txt
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
Vencord, a Discord client mod
|
||||||
|
Copyright (c) {year} {author}
|
||||||
|
SPDX-License-Identifier: GPL-3.0-or-later
|
17
scripts/header-old.txt
Normal file
17
scripts/header-old.txt
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
/*
|
||||||
|
* Vencord, a modification for Discord's desktop app
|
||||||
|
* Copyright (c) {year} {author}
|
||||||
|
*
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
*/
|
30
scripts/utils.mjs
Normal file
30
scripts/utils.mjs
Normal file
@ -0,0 +1,30 @@
|
|||||||
|
/*
|
||||||
|
* Vencord, a modification for Discord's desktop app
|
||||||
|
* Copyright (c) 2023 Vendicated and contributors
|
||||||
|
*
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {string} filePath
|
||||||
|
* @returns {string | null}
|
||||||
|
*/
|
||||||
|
export function getPluginTarget(filePath) {
|
||||||
|
const pathParts = filePath.split(/[/\\]/);
|
||||||
|
if (/^index\.tsx?$/.test(pathParts.at(-1))) pathParts.pop();
|
||||||
|
|
||||||
|
const identifier = pathParts.at(-1).replace(/\.tsx?$/, "");
|
||||||
|
const identiferBits = identifier.split(".");
|
||||||
|
return identiferBits.length === 1 ? null : identiferBits.at(-1);
|
||||||
|
}
|
101
src/Vencord.ts
101
src/Vencord.ts
@ -27,19 +27,61 @@ export { PlainSettings, Settings };
|
|||||||
import "./utils/quickCss";
|
import "./utils/quickCss";
|
||||||
import "./webpack/patchWebpack";
|
import "./webpack/patchWebpack";
|
||||||
|
|
||||||
import { popNotice, showNotice } from "./api/Notices";
|
import { get as dsGet } from "./api/DataStore";
|
||||||
import { PlainSettings, Settings } from "./api/settings";
|
import { showNotification } from "./api/Notifications";
|
||||||
|
import { PlainSettings, Settings } from "./api/Settings";
|
||||||
import { patches, PMLogger, startAllPlugins } from "./plugins";
|
import { patches, PMLogger, startAllPlugins } from "./plugins";
|
||||||
import { checkForUpdates, rebuild, update, UpdateLogger } from "./utils/updater";
|
import { localStorage } from "./utils/localStorage";
|
||||||
|
import { relaunch } from "./utils/native";
|
||||||
|
import { getCloudSettings, putCloudSettings } from "./utils/settingsSync";
|
||||||
|
import { checkForUpdates, update, UpdateLogger } from "./utils/updater";
|
||||||
import { onceReady } from "./webpack";
|
import { onceReady } from "./webpack";
|
||||||
import { SettingsRouter } from "./webpack/common";
|
import { SettingsRouter } from "./webpack/common";
|
||||||
|
|
||||||
export let Components: any;
|
async function syncSettings() {
|
||||||
|
// pre-check for local shared settings
|
||||||
|
if (
|
||||||
|
Settings.cloud.authenticated &&
|
||||||
|
await dsGet("Vencord_cloudSecret") === null // this has been enabled due to local settings share or some other bug
|
||||||
|
) {
|
||||||
|
// show a notification letting them know and tell them how to fix it
|
||||||
|
showNotification({
|
||||||
|
title: "Cloud Integrations",
|
||||||
|
body: "We've noticed you have cloud integrations enabled in another client! Due to limitations, you will " +
|
||||||
|
"need to re-authenticate to continue using them. Click here to go to the settings page to do so!",
|
||||||
|
color: "var(--yellow-360)",
|
||||||
|
onClick: () => SettingsRouter.open("VencordCloud")
|
||||||
|
});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (
|
||||||
|
Settings.cloud.settingsSync && // if it's enabled
|
||||||
|
Settings.cloud.authenticated // if cloud integrations are enabled
|
||||||
|
) {
|
||||||
|
if (localStorage.Vencord_settingsDirty) {
|
||||||
|
await putCloudSettings();
|
||||||
|
delete localStorage.Vencord_settingsDirty;
|
||||||
|
} else if (await getCloudSettings(false)) { // if we synchronized something (false means no sync)
|
||||||
|
// we show a notification here instead of allowing getCloudSettings() to show one to declutter the amount of
|
||||||
|
// potential notifications that might occur. getCloudSettings() will always send a notification regardless if
|
||||||
|
// there was an error to notify the user, but besides that we only want to show one notification instead of all
|
||||||
|
// of the possible ones it has (such as when your settings are newer).
|
||||||
|
showNotification({
|
||||||
|
title: "Cloud Settings",
|
||||||
|
body: "Your settings have been updated! Click here to restart to fully apply changes!",
|
||||||
|
color: "var(--green-360)",
|
||||||
|
onClick: relaunch
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
async function init() {
|
async function init() {
|
||||||
await onceReady;
|
await onceReady;
|
||||||
startAllPlugins();
|
startAllPlugins();
|
||||||
Components = await import("./components");
|
|
||||||
|
syncSettings();
|
||||||
|
|
||||||
if (!IS_WEB) {
|
if (!IS_WEB) {
|
||||||
try {
|
try {
|
||||||
@ -48,33 +90,27 @@ async function init() {
|
|||||||
|
|
||||||
if (Settings.autoUpdate) {
|
if (Settings.autoUpdate) {
|
||||||
await update();
|
await update();
|
||||||
const needsFullRestart = await rebuild();
|
if (Settings.autoUpdateNotification)
|
||||||
setTimeout(() => {
|
setTimeout(() => showNotification({
|
||||||
showNotice(
|
title: "Vencord has been updated!",
|
||||||
"Vencord has been updated!",
|
body: "Click here to restart",
|
||||||
"Restart",
|
permanent: true,
|
||||||
() => {
|
noPersist: true,
|
||||||
if (needsFullRestart)
|
onClick: relaunch
|
||||||
window.DiscordNative.app.relaunch();
|
}), 10_000);
|
||||||
else
|
|
||||||
location.reload();
|
|
||||||
}
|
|
||||||
);
|
|
||||||
}, 10_000);
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (Settings.notifyAboutUpdates)
|
if (Settings.notifyAboutUpdates)
|
||||||
setTimeout(() => {
|
setTimeout(() => showNotification({
|
||||||
showNotice(
|
title: "A Vencord update is available!",
|
||||||
"A Vencord update is available!",
|
body: "Click here to view the update",
|
||||||
"View Update",
|
permanent: true,
|
||||||
() => {
|
noPersist: true,
|
||||||
popNotice();
|
onClick() {
|
||||||
SettingsRouter.open("VencordUpdater");
|
SettingsRouter.open("VencordUpdater");
|
||||||
}
|
}
|
||||||
);
|
}), 10_000);
|
||||||
}, 10_000);
|
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
UpdateLogger.error("Failed to check for updates", err);
|
UpdateLogger.error("Failed to check for updates", err);
|
||||||
}
|
}
|
||||||
@ -95,3 +131,12 @@ async function init() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
init();
|
init();
|
||||||
|
|
||||||
|
if (IS_DISCORD_DESKTOP && Settings.winNativeTitleBar && navigator.platform.toLowerCase().startsWith("win")) {
|
||||||
|
document.addEventListener("DOMContentLoaded", () => {
|
||||||
|
document.head.append(Object.assign(document.createElement("style"), {
|
||||||
|
id: "vencord-native-titlebar-style",
|
||||||
|
textContent: "[class*=titleBar-]{display: none!important}"
|
||||||
|
}));
|
||||||
|
}, { once: true });
|
||||||
|
}
|
||||||
|
@ -1,49 +1,71 @@
|
|||||||
/*
|
/*
|
||||||
* Vencord, a modification for Discord's desktop app
|
* Vencord, a Discord client mod
|
||||||
* Copyright (c) 2022
|
* Copyright (c) 2023 Vendicated and contributors
|
||||||
*
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
* This program is free software: you can redistribute it and/or modify
|
*/
|
||||||
* it under the terms of the GNU General Public License as published by
|
|
||||||
* the Free Software Foundation, either version 3 of the License, or
|
|
||||||
* (at your option) any later version.
|
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
||||||
* GNU General Public License for more details.
|
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU General Public License
|
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
||||||
*/
|
|
||||||
|
|
||||||
import IPC_EVENTS from "@utils/IpcEvents";
|
import { IpcEvents } from "@utils/IpcEvents";
|
||||||
import { IpcRenderer, ipcRenderer } from "electron";
|
import { IpcRes } from "@utils/types";
|
||||||
|
import { ipcRenderer } from "electron";
|
||||||
|
import type { UserThemeHeader } from "main/themes";
|
||||||
|
|
||||||
function assertEventAllowed(event: string) {
|
function invoke<T = any>(event: IpcEvents, ...args: any[]) {
|
||||||
if (!(event in IPC_EVENTS)) throw new Error(`Event ${event} not allowed.`);
|
return ipcRenderer.invoke(event, ...args) as Promise<T>;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function sendSync<T = any>(event: IpcEvents, ...args: any[]) {
|
||||||
|
return ipcRenderer.sendSync(event, ...args) as T;
|
||||||
|
}
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
getVersions: () => process.versions,
|
themes: {
|
||||||
ipc: {
|
uploadTheme: (fileName: string, fileData: string) => invoke<void>(IpcEvents.UPLOAD_THEME, fileName, fileData),
|
||||||
send(event: string, ...args: any[]) {
|
deleteTheme: (fileName: string) => invoke<void>(IpcEvents.DELETE_THEME, fileName),
|
||||||
assertEventAllowed(event);
|
getThemesDir: () => invoke<string>(IpcEvents.GET_THEMES_DIR),
|
||||||
ipcRenderer.send(event, ...args);
|
getThemesList: () => invoke<Array<UserThemeHeader>>(IpcEvents.GET_THEMES_LIST),
|
||||||
|
getThemeData: (fileName: string) => invoke<string | undefined>(IpcEvents.GET_THEME_DATA, fileName)
|
||||||
|
},
|
||||||
|
|
||||||
|
updater: {
|
||||||
|
getUpdates: () => invoke<IpcRes<Record<"hash" | "author" | "message", string>[]>>(IpcEvents.GET_UPDATES),
|
||||||
|
update: () => invoke<IpcRes<boolean>>(IpcEvents.UPDATE),
|
||||||
|
rebuild: () => invoke<IpcRes<boolean>>(IpcEvents.BUILD),
|
||||||
|
getRepo: () => invoke<IpcRes<string>>(IpcEvents.GET_REPO),
|
||||||
|
},
|
||||||
|
|
||||||
|
settings: {
|
||||||
|
get: () => sendSync<string>(IpcEvents.GET_SETTINGS),
|
||||||
|
set: (settings: string) => invoke<void>(IpcEvents.SET_SETTINGS, settings),
|
||||||
|
getSettingsDir: () => invoke<string>(IpcEvents.GET_SETTINGS_DIR),
|
||||||
|
},
|
||||||
|
|
||||||
|
quickCss: {
|
||||||
|
get: () => invoke<string>(IpcEvents.GET_QUICK_CSS),
|
||||||
|
set: (css: string) => invoke<void>(IpcEvents.SET_QUICK_CSS, css),
|
||||||
|
|
||||||
|
addChangeListener(cb: (newCss: string) => void) {
|
||||||
|
ipcRenderer.on(IpcEvents.QUICK_CSS_UPDATE, (_, css) => cb(css));
|
||||||
},
|
},
|
||||||
sendSync<T = any>(event: string, ...args: any[]): T {
|
|
||||||
assertEventAllowed(event);
|
addThemeChangeListener(cb: () => void) {
|
||||||
return ipcRenderer.sendSync(event, ...args);
|
ipcRenderer.on(IpcEvents.THEME_UPDATE, () => cb());
|
||||||
},
|
},
|
||||||
on(event: string, listener: Parameters<IpcRenderer["on"]>[1]) {
|
|
||||||
assertEventAllowed(event);
|
openFile: () => invoke<void>(IpcEvents.OPEN_QUICKCSS),
|
||||||
ipcRenderer.on(event, listener);
|
openEditor: () => invoke<void>(IpcEvents.OPEN_MONACO_EDITOR),
|
||||||
|
},
|
||||||
|
|
||||||
|
native: {
|
||||||
|
getVersions: () => process.versions as Partial<NodeJS.ProcessVersions>,
|
||||||
|
openExternal: (url: string) => invoke<void>(IpcEvents.OPEN_EXTERNAL, url)
|
||||||
|
},
|
||||||
|
|
||||||
|
pluginHelpers: {
|
||||||
|
OpenInApp: {
|
||||||
|
resolveRedirect: (url: string) => invoke<string>(IpcEvents.OPEN_IN_APP__RESOLVE_REDIRECT, url),
|
||||||
},
|
},
|
||||||
off(event: string, listener: Parameters<IpcRenderer["off"]>[1]) {
|
VoiceMessages: {
|
||||||
assertEventAllowed(event);
|
readRecording: (path: string) => invoke<Uint8Array | null>(IpcEvents.VOICE_MESSAGES_READ_RECORDING, path),
|
||||||
ipcRenderer.off(event, listener);
|
|
||||||
},
|
|
||||||
invoke<T = any>(event: string, ...args: any[]): Promise<T> {
|
|
||||||
assertEventAllowed(event);
|
|
||||||
return ipcRenderer.invoke(event, ...args);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
@ -22,18 +22,19 @@ import { ComponentType, HTMLProps } from "react";
|
|||||||
|
|
||||||
import Plugins from "~plugins";
|
import Plugins from "~plugins";
|
||||||
|
|
||||||
export enum BadgePosition {
|
export const enum BadgePosition {
|
||||||
START,
|
START,
|
||||||
END
|
END
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface ProfileBadge {
|
export interface ProfileBadge {
|
||||||
/** The tooltip to show on hover. Required for image badges */
|
/** The tooltip to show on hover. Required for image badges */
|
||||||
tooltip?: string;
|
description?: string;
|
||||||
/** Custom component for the badge (tooltip not included) */
|
/** Custom component for the badge (tooltip not included) */
|
||||||
component?: ComponentType<ProfileBadge & BadgeUserArgs>;
|
component?: ComponentType<ProfileBadge & BadgeUserArgs>;
|
||||||
/** The custom image to use */
|
/** The custom image to use */
|
||||||
image?: string;
|
image?: string;
|
||||||
|
link?: string;
|
||||||
/** Action to perform when you click the badge */
|
/** Action to perform when you click the badge */
|
||||||
onClick?(): void;
|
onClick?(): void;
|
||||||
/** Should the user display this badge? */
|
/** Should the user display this badge? */
|
||||||
@ -69,17 +70,19 @@ export function removeBadge(badge: ProfileBadge) {
|
|||||||
* Inject badges into the profile badges array.
|
* Inject badges into the profile badges array.
|
||||||
* You probably don't need to use this.
|
* You probably don't need to use this.
|
||||||
*/
|
*/
|
||||||
export function inject(badgeArray: ProfileBadge[], args: BadgeUserArgs) {
|
export function _getBadges(args: BadgeUserArgs) {
|
||||||
|
const badges = [] as ProfileBadge[];
|
||||||
for (const badge of Badges) {
|
for (const badge of Badges) {
|
||||||
if (!badge.shouldShow || badge.shouldShow(args)) {
|
if (!badge.shouldShow || badge.shouldShow(args)) {
|
||||||
badge.position === BadgePosition.START
|
badge.position === BadgePosition.START
|
||||||
? badgeArray.unshift({ ...badge, ...args })
|
? badges.unshift({ ...badge, ...args })
|
||||||
: badgeArray.push({ ...badge, ...args });
|
: badges.push({ ...badge, ...args });
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
(Plugins.BadgeAPI as any).addDonorBadge(badgeArray, args.user.id);
|
const donorBadges = (Plugins.BadgeAPI as unknown as typeof import("../plugins/_api/badges").default).getDonorBadges(args.user.id);
|
||||||
|
if (donorBadges) badges.unshift(...donorBadges);
|
||||||
|
|
||||||
return badgeArray;
|
return badges;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface BadgeUserArgs {
|
export interface BadgeUserArgs {
|
||||||
|
@ -16,7 +16,7 @@
|
|||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import { makeCodeblock } from "@utils/misc";
|
import { makeCodeblock } from "@utils/text";
|
||||||
|
|
||||||
import { sendBotMessage } from "./commandHelpers";
|
import { sendBotMessage } from "./commandHelpers";
|
||||||
import { ApplicationCommandInputType, ApplicationCommandOptionType, ApplicationCommandType, Argument, Command, CommandContext, Option } from "./types";
|
import { ApplicationCommandInputType, ApplicationCommandOptionType, ApplicationCommandType, Argument, Command, CommandContext, Option } from "./types";
|
||||||
@ -111,6 +111,7 @@ function registerSubCommands(cmd: Command, plugin: string) {
|
|||||||
...o,
|
...o,
|
||||||
type: ApplicationCommandType.CHAT_INPUT,
|
type: ApplicationCommandType.CHAT_INPUT,
|
||||||
name: `${cmd.name} ${o.name}`,
|
name: `${cmd.name} ${o.name}`,
|
||||||
|
id: `${o.name}-${cmd.id}`,
|
||||||
displayName: `${cmd.name} ${o.name}`,
|
displayName: `${cmd.name} ${o.name}`,
|
||||||
subCommandPath: [{
|
subCommandPath: [{
|
||||||
name: o.name,
|
name: o.name,
|
||||||
|
@ -24,7 +24,7 @@ export interface CommandContext {
|
|||||||
guild?: Guild;
|
guild?: Guild;
|
||||||
}
|
}
|
||||||
|
|
||||||
export enum ApplicationCommandOptionType {
|
export const enum ApplicationCommandOptionType {
|
||||||
SUB_COMMAND = 1,
|
SUB_COMMAND = 1,
|
||||||
SUB_COMMAND_GROUP = 2,
|
SUB_COMMAND_GROUP = 2,
|
||||||
STRING = 3,
|
STRING = 3,
|
||||||
@ -38,7 +38,7 @@ export enum ApplicationCommandOptionType {
|
|||||||
ATTACHMENT = 11,
|
ATTACHMENT = 11,
|
||||||
}
|
}
|
||||||
|
|
||||||
export enum ApplicationCommandInputType {
|
export const enum ApplicationCommandInputType {
|
||||||
BUILT_IN = 0,
|
BUILT_IN = 0,
|
||||||
BUILT_IN_TEXT = 1,
|
BUILT_IN_TEXT = 1,
|
||||||
BUILT_IN_INTEGRATION = 2,
|
BUILT_IN_INTEGRATION = 2,
|
||||||
@ -64,7 +64,7 @@ export interface ChoicesOption {
|
|||||||
displayName?: string;
|
displayName?: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
export enum ApplicationCommandType {
|
export const enum ApplicationCommandType {
|
||||||
CHAT_INPUT = 1,
|
CHAT_INPUT = 1,
|
||||||
USER = 2,
|
USER = 2,
|
||||||
MESSAGE = 3,
|
MESSAGE = 3,
|
||||||
|
158
src/api/ContextMenu.ts
Normal file
158
src/api/ContextMenu.ts
Normal file
@ -0,0 +1,158 @@
|
|||||||
|
/*
|
||||||
|
* Vencord, a modification for Discord's desktop app
|
||||||
|
* Copyright (c) 2023 Vendicated and contributors
|
||||||
|
*
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { Logger } from "@utils/Logger";
|
||||||
|
import type { ReactElement } from "react";
|
||||||
|
|
||||||
|
type ContextMenuPatchCallbackReturn = (() => void) | void;
|
||||||
|
/**
|
||||||
|
* @param children The rendered context menu elements
|
||||||
|
* @param args Any arguments passed into making the context menu, like the guild, channel, user or message for example
|
||||||
|
* @returns A callback which is only ran once used to modify the context menu elements (Use to avoid duplicates)
|
||||||
|
*/
|
||||||
|
export type NavContextMenuPatchCallback = (children: Array<ReactElement | null>, ...args: Array<any>) => ContextMenuPatchCallbackReturn;
|
||||||
|
/**
|
||||||
|
* @param navId The navId of the context menu being patched
|
||||||
|
* @param children The rendered context menu elements
|
||||||
|
* @param args Any arguments passed into making the context menu, like the guild, channel, user or message for example
|
||||||
|
* @returns A callback which is only ran once used to modify the context menu elements (Use to avoid duplicates)
|
||||||
|
*/
|
||||||
|
export type GlobalContextMenuPatchCallback = (navId: string, children: Array<ReactElement | null>, ...args: Array<any>) => ContextMenuPatchCallbackReturn;
|
||||||
|
|
||||||
|
const ContextMenuLogger = new Logger("ContextMenu");
|
||||||
|
|
||||||
|
export const navPatches = new Map<string, Set<NavContextMenuPatchCallback>>();
|
||||||
|
export const globalPatches = new Set<GlobalContextMenuPatchCallback>();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Add a context menu patch
|
||||||
|
* @param navId The navId(s) for the context menu(s) to patch
|
||||||
|
* @param patch The patch to be applied
|
||||||
|
*/
|
||||||
|
export function addContextMenuPatch(navId: string | Array<string>, patch: NavContextMenuPatchCallback) {
|
||||||
|
if (!Array.isArray(navId)) navId = [navId];
|
||||||
|
for (const id of navId) {
|
||||||
|
let contextMenuPatches = navPatches.get(id);
|
||||||
|
if (!contextMenuPatches) {
|
||||||
|
contextMenuPatches = new Set();
|
||||||
|
navPatches.set(id, contextMenuPatches);
|
||||||
|
}
|
||||||
|
|
||||||
|
contextMenuPatches.add(patch);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Add a global context menu patch that fires the patch for all context menus
|
||||||
|
* @param patch The patch to be applied
|
||||||
|
*/
|
||||||
|
export function addGlobalContextMenuPatch(patch: GlobalContextMenuPatchCallback) {
|
||||||
|
globalPatches.add(patch);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Remove a context menu patch
|
||||||
|
* @param navId The navId(s) for the context menu(s) to remove the patch
|
||||||
|
* @param patch The patch to be removed
|
||||||
|
* @returns Wheter the patch was sucessfully removed from the context menu(s)
|
||||||
|
*/
|
||||||
|
export function removeContextMenuPatch<T extends string | Array<string>>(navId: T, patch: NavContextMenuPatchCallback): T extends string ? boolean : Array<boolean> {
|
||||||
|
const navIds = Array.isArray(navId) ? navId : [navId as string];
|
||||||
|
|
||||||
|
const results = navIds.map(id => navPatches.get(id)?.delete(patch) ?? false);
|
||||||
|
|
||||||
|
return (Array.isArray(navId) ? results : results[0]) as T extends string ? boolean : Array<boolean>;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Remove a global context menu patch
|
||||||
|
* @param patch The patch to be removed
|
||||||
|
* @returns Wheter the patch was sucessfully removed
|
||||||
|
*/
|
||||||
|
export function removeGlobalContextMenuPatch(patch: GlobalContextMenuPatchCallback): boolean {
|
||||||
|
return globalPatches.delete(patch);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A helper function for finding the children array of a group nested inside a context menu based on the id(s) of its children
|
||||||
|
* @param id The id of the child. If an array is specified, all ids will be tried
|
||||||
|
* @param children The context menu children
|
||||||
|
*/
|
||||||
|
export function findGroupChildrenByChildId(id: string | string[], children: Array<ReactElement | null>, _itemsArray?: Array<ReactElement | null>): Array<ReactElement | null> | null {
|
||||||
|
for (const child of children) {
|
||||||
|
if (child == null) continue;
|
||||||
|
|
||||||
|
if (
|
||||||
|
(Array.isArray(id) && id.some(id => child.props?.id === id))
|
||||||
|
|| child.props?.id === id
|
||||||
|
) return _itemsArray ?? null;
|
||||||
|
|
||||||
|
let nextChildren = child.props?.children;
|
||||||
|
if (nextChildren) {
|
||||||
|
if (!Array.isArray(nextChildren)) {
|
||||||
|
nextChildren = [nextChildren];
|
||||||
|
child.props.children = nextChildren;
|
||||||
|
}
|
||||||
|
|
||||||
|
const found = findGroupChildrenByChildId(id, nextChildren, nextChildren);
|
||||||
|
if (found !== null) return found;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface ContextMenuProps {
|
||||||
|
contextMenuApiArguments?: Array<any>;
|
||||||
|
navId: string;
|
||||||
|
children: Array<ReactElement | null>;
|
||||||
|
"aria-label": string;
|
||||||
|
onSelect: (() => void) | undefined;
|
||||||
|
onClose: (callback: (...args: Array<any>) => any) => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
const patchedMenus = new WeakSet();
|
||||||
|
|
||||||
|
export function _patchContextMenu(props: ContextMenuProps) {
|
||||||
|
props.contextMenuApiArguments ??= [];
|
||||||
|
const contextMenuPatches = navPatches.get(props.navId);
|
||||||
|
|
||||||
|
if (!Array.isArray(props.children)) props.children = [props.children];
|
||||||
|
|
||||||
|
if (contextMenuPatches) {
|
||||||
|
for (const patch of contextMenuPatches) {
|
||||||
|
try {
|
||||||
|
const callback = patch(props.children, ...props.contextMenuApiArguments);
|
||||||
|
if (!patchedMenus.has(props)) callback?.();
|
||||||
|
} catch (err) {
|
||||||
|
ContextMenuLogger.error(`Patch for ${props.navId} errored,`, err);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for (const patch of globalPatches) {
|
||||||
|
try {
|
||||||
|
const callback = patch(props.navId, props.children, ...props.contextMenuApiArguments);
|
||||||
|
if (!patchedMenus.has(props)) callback?.();
|
||||||
|
} catch (err) {
|
||||||
|
ContextMenuLogger.error("Global patch errored,", err);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
patchedMenus.add(props);
|
||||||
|
}
|
@ -1,4 +1,4 @@
|
|||||||
/* eslint-disable header/header */
|
/* eslint-disable simple-header/header */
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* idb-keyval v6.2.0
|
* idb-keyval v6.2.0
|
||||||
|
@ -16,41 +16,74 @@
|
|||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import Logger from "@utils/Logger";
|
import { Logger } from "@utils/Logger";
|
||||||
import { MessageStore } from "@webpack/common";
|
import { MessageStore } from "@webpack/common";
|
||||||
|
import { CustomEmoji } from "@webpack/types";
|
||||||
import type { Channel, Message } from "discord-types/general";
|
import type { Channel, Message } from "discord-types/general";
|
||||||
|
import type { Promisable } from "type-fest";
|
||||||
|
|
||||||
const MessageEventsLogger = new Logger("MessageEvents", "#e5c890");
|
const MessageEventsLogger = new Logger("MessageEvents", "#e5c890");
|
||||||
|
|
||||||
export interface Emoji {
|
|
||||||
require_colons: boolean,
|
|
||||||
originalName: string,
|
|
||||||
animated: boolean;
|
|
||||||
guildId: string,
|
|
||||||
name: string,
|
|
||||||
url: string,
|
|
||||||
id: string,
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface MessageObject {
|
export interface MessageObject {
|
||||||
content: string,
|
content: string,
|
||||||
validNonShortcutEmojis: Emoji[];
|
validNonShortcutEmojis: CustomEmoji[];
|
||||||
|
invalidEmojis: any[];
|
||||||
|
tts: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface Upload {
|
||||||
|
classification: string;
|
||||||
|
currentSize: number;
|
||||||
|
description: string | null;
|
||||||
|
filename: string;
|
||||||
|
id: string;
|
||||||
|
isImage: boolean;
|
||||||
|
isVideo: boolean;
|
||||||
|
item: {
|
||||||
|
file: File;
|
||||||
|
platform: number;
|
||||||
|
};
|
||||||
|
loaded: number;
|
||||||
|
mimeType: string;
|
||||||
|
preCompressionSize: number;
|
||||||
|
responseUrl: string;
|
||||||
|
sensitive: boolean;
|
||||||
|
showLargeMessageDialog: boolean;
|
||||||
|
spoiler: boolean;
|
||||||
|
status: "NOT_STARTED" | "STARTED" | "UPLOADING" | "ERROR" | "COMPLETED" | "CANCELLED";
|
||||||
|
uniqueId: string;
|
||||||
|
uploadedFilename: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface MessageReplyOptions {
|
||||||
|
messageReference: Message["messageReference"];
|
||||||
|
allowedMentions?: {
|
||||||
|
parse: Array<string>;
|
||||||
|
repliedUser: boolean;
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface MessageExtra {
|
export interface MessageExtra {
|
||||||
stickerIds?: string[];
|
stickers?: string[];
|
||||||
|
uploads?: Upload[];
|
||||||
|
replyOptions: MessageReplyOptions;
|
||||||
|
content: string;
|
||||||
|
channel: Channel;
|
||||||
|
type?: any;
|
||||||
|
openWarningPopout: (props: any) => any;
|
||||||
}
|
}
|
||||||
|
|
||||||
export type SendListener = (channelId: string, messageObj: MessageObject, extra: MessageExtra) => void | { cancel: boolean; };
|
export type SendListener = (channelId: string, messageObj: MessageObject, extra: MessageExtra) => Promisable<void | { cancel: boolean; }>;
|
||||||
export type EditListener = (channelId: string, messageId: string, messageObj: MessageObject) => void;
|
export type EditListener = (channelId: string, messageId: string, messageObj: MessageObject) => Promisable<void>;
|
||||||
|
|
||||||
const sendListeners = new Set<SendListener>();
|
const sendListeners = new Set<SendListener>();
|
||||||
const editListeners = new Set<EditListener>();
|
const editListeners = new Set<EditListener>();
|
||||||
|
|
||||||
export function _handlePreSend(channelId: string, messageObj: MessageObject, extra: MessageExtra) {
|
export async function _handlePreSend(channelId: string, messageObj: MessageObject, extra: MessageExtra, replyOptions: MessageReplyOptions) {
|
||||||
|
extra.replyOptions = replyOptions;
|
||||||
for (const listener of sendListeners) {
|
for (const listener of sendListeners) {
|
||||||
try {
|
try {
|
||||||
const result = listener(channelId, messageObj, extra);
|
const result = await listener(channelId, messageObj, extra);
|
||||||
if (result && result.cancel === true) {
|
if (result && result.cancel === true) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
@ -61,10 +94,10 @@ export function _handlePreSend(channelId: string, messageObj: MessageObject, ext
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
export function _handlePreEdit(channelId: string, messageId: string, messageObj: MessageObject) {
|
export async function _handlePreEdit(channelId: string, messageId: string, messageObj: MessageObject) {
|
||||||
for (const listener of editListeners) {
|
for (const listener of editListeners) {
|
||||||
try {
|
try {
|
||||||
listener(channelId, messageId, messageObj);
|
await listener(channelId, messageId, messageObj);
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
MessageEventsLogger.error("MessageEditHandler: Listener encountered an unknown error\n", e);
|
MessageEventsLogger.error("MessageEditHandler: Listener encountered an unknown error\n", e);
|
||||||
}
|
}
|
||||||
|
@ -16,7 +16,7 @@
|
|||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import Logger from "@utils/Logger";
|
import { Logger } from "@utils/Logger";
|
||||||
import { Channel, Message } from "discord-types/general";
|
import { Channel, Message } from "discord-types/general";
|
||||||
import type { MouseEventHandler } from "react";
|
import type { MouseEventHandler } from "react";
|
||||||
|
|
||||||
|
@ -18,9 +18,10 @@
|
|||||||
|
|
||||||
import "./styles.css";
|
import "./styles.css";
|
||||||
|
|
||||||
import { useSettings } from "@api/settings";
|
import { useSettings } from "@api/Settings";
|
||||||
import ErrorBoundary from "@components/ErrorBoundary";
|
import ErrorBoundary from "@components/ErrorBoundary";
|
||||||
import { Forms, React, useEffect, useMemo, useState, useStateFromStores, WindowStore } from "@webpack/common";
|
import { classes } from "@utils/misc";
|
||||||
|
import { React, useEffect, useMemo, useState, useStateFromStores, WindowStore } from "@webpack/common";
|
||||||
|
|
||||||
import { NotificationData } from "./Notifications";
|
import { NotificationData } from "./Notifications";
|
||||||
|
|
||||||
@ -32,8 +33,11 @@ export default ErrorBoundary.wrap(function NotificationComponent({
|
|||||||
icon,
|
icon,
|
||||||
onClick,
|
onClick,
|
||||||
onClose,
|
onClose,
|
||||||
image
|
image,
|
||||||
}: NotificationData) {
|
permanent,
|
||||||
|
className,
|
||||||
|
dismissOnClick
|
||||||
|
}: NotificationData & { className?: string; }) {
|
||||||
const { timeout, position } = useSettings(["notifications.timeout", "notifications.position"]).notifications;
|
const { timeout, position } = useSettings(["notifications.timeout", "notifications.position"]).notifications;
|
||||||
const hasFocus = useStateFromStores([WindowStore], () => WindowStore.isFocused());
|
const hasFocus = useStateFromStores([WindowStore], () => WindowStore.isFocused());
|
||||||
|
|
||||||
@ -43,7 +47,7 @@ export default ErrorBoundary.wrap(function NotificationComponent({
|
|||||||
const start = useMemo(() => Date.now(), [timeout, isHover, hasFocus]);
|
const start = useMemo(() => Date.now(), [timeout, isHover, hasFocus]);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (isHover || !hasFocus || timeout === 0) return void setElapsed(0);
|
if (isHover || !hasFocus || timeout === 0 || permanent) return void setElapsed(0);
|
||||||
|
|
||||||
const intervalId = setInterval(() => {
|
const intervalId = setInterval(() => {
|
||||||
const elapsed = Date.now() - start;
|
const elapsed = Date.now() - start;
|
||||||
@ -60,9 +64,13 @@ export default ErrorBoundary.wrap(function NotificationComponent({
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<button
|
<button
|
||||||
className="vc-notification-root"
|
className={classes("vc-notification-root", className)}
|
||||||
style={position === "bottom-right" ? { bottom: "1rem" } : { top: "3rem" }}
|
style={position === "bottom-right" ? { bottom: "1rem" } : { top: "3rem" }}
|
||||||
onClick={onClick}
|
onClick={() => {
|
||||||
|
onClick?.();
|
||||||
|
if (dismissOnClick !== false)
|
||||||
|
onClose!();
|
||||||
|
}}
|
||||||
onContextMenu={e => {
|
onContextMenu={e => {
|
||||||
e.preventDefault();
|
e.preventDefault();
|
||||||
e.stopPropagation();
|
e.stopPropagation();
|
||||||
@ -74,14 +82,35 @@ export default ErrorBoundary.wrap(function NotificationComponent({
|
|||||||
<div className="vc-notification">
|
<div className="vc-notification">
|
||||||
{icon && <img className="vc-notification-icon" src={icon} alt="" />}
|
{icon && <img className="vc-notification-icon" src={icon} alt="" />}
|
||||||
<div className="vc-notification-content">
|
<div className="vc-notification-content">
|
||||||
<Forms.FormTitle tag="h2">{title}</Forms.FormTitle>
|
<div className="vc-notification-header">
|
||||||
|
<h2 className="vc-notification-title">{title}</h2>
|
||||||
|
<button
|
||||||
|
className="vc-notification-close-btn"
|
||||||
|
onClick={e => {
|
||||||
|
e.preventDefault();
|
||||||
|
e.stopPropagation();
|
||||||
|
onClose!();
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<svg
|
||||||
|
width="24"
|
||||||
|
height="24"
|
||||||
|
viewBox="0 0 24 24"
|
||||||
|
role="img"
|
||||||
|
aria-labelledby="vc-notification-dismiss-title"
|
||||||
|
>
|
||||||
|
<title id="vc-notification-dismiss-title">Dismiss Notification</title>
|
||||||
|
<path fill="currentColor" d="M18.4 4L12 10.4L5.6 4L4 5.6L10.4 12L4 18.4L5.6 20L12 13.6L18.4 20L20 18.4L13.6 12L20 5.6L18.4 4Z" />
|
||||||
|
</svg>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
<div>
|
<div>
|
||||||
{richBody ?? <p className="vc-notification-p">{body}</p>}
|
{richBody ?? <p className="vc-notification-p">{body}</p>}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
{image && <img className="vc-notification-img" src={image} alt="" />}
|
{image && <img className="vc-notification-img" src={image} alt="" />}
|
||||||
{timeout !== 0 && (
|
{timeout !== 0 && !permanent && (
|
||||||
<div
|
<div
|
||||||
className="vc-notification-progressbar"
|
className="vc-notification-progressbar"
|
||||||
style={{ width: `${(1 - timeoutProgress) * 100}%`, backgroundColor: color || "var(--brand-experiment)" }}
|
style={{ width: `${(1 - timeoutProgress) * 100}%`, backgroundColor: color || "var(--brand-experiment)" }}
|
||||||
@ -89,4 +118,6 @@ export default ErrorBoundary.wrap(function NotificationComponent({
|
|||||||
)}
|
)}
|
||||||
</button>
|
</button>
|
||||||
);
|
);
|
||||||
|
}, {
|
||||||
|
onError: ({ props }) => props.onClose!()
|
||||||
});
|
});
|
||||||
|
@ -16,13 +16,14 @@
|
|||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import { Settings } from "@api/settings";
|
import { Settings } from "@api/Settings";
|
||||||
import { Queue } from "@utils/Queue";
|
import { Queue } from "@utils/Queue";
|
||||||
import { ReactDOM } from "@webpack/common";
|
import { ReactDOM } from "@webpack/common";
|
||||||
import type { ReactNode } from "react";
|
import type { ReactNode } from "react";
|
||||||
import type { Root } from "react-dom/client";
|
import type { Root } from "react-dom/client";
|
||||||
|
|
||||||
import NotificationComponent from "./NotificationComponent";
|
import NotificationComponent from "./NotificationComponent";
|
||||||
|
import { persistNotification } from "./notificationLog";
|
||||||
|
|
||||||
const NotificationQueue = new Queue();
|
const NotificationQueue = new Queue();
|
||||||
|
|
||||||
@ -54,6 +55,12 @@ export interface NotificationData {
|
|||||||
onClick?(): void;
|
onClick?(): void;
|
||||||
onClose?(): void;
|
onClose?(): void;
|
||||||
color?: string;
|
color?: string;
|
||||||
|
/** Whether this notification should not have a timeout */
|
||||||
|
permanent?: boolean;
|
||||||
|
/** Whether this notification should not be persisted in the Notification Log */
|
||||||
|
noPersist?: boolean;
|
||||||
|
/** Whether this notification should be dismissed when clicked (defaults to true) */
|
||||||
|
dismissOnClick?: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
function _showNotification(notification: NotificationData, id: number) {
|
function _showNotification(notification: NotificationData, id: number) {
|
||||||
@ -70,6 +77,8 @@ function _showNotification(notification: NotificationData, id: number) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function shouldBeNative() {
|
function shouldBeNative() {
|
||||||
|
if (typeof Notification === "undefined") return false;
|
||||||
|
|
||||||
const { useNative } = Settings.notifications;
|
const { useNative } = Settings.notifications;
|
||||||
if (useNative === "always") return true;
|
if (useNative === "always") return true;
|
||||||
if (useNative === "not-focused") return !document.hasFocus();
|
if (useNative === "not-focused") return !document.hasFocus();
|
||||||
@ -84,6 +93,8 @@ export async function requestPermission() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export async function showNotification(data: NotificationData) {
|
export async function showNotification(data: NotificationData) {
|
||||||
|
persistNotification(data);
|
||||||
|
|
||||||
if (shouldBeNative() && await requestPermission()) {
|
if (shouldBeNative() && await requestPermission()) {
|
||||||
const { title, body, icon, image, onClick = null, onClose = null } = data;
|
const { title, body, icon, image, onClick = null, onClose = null } = data;
|
||||||
const n = new Notification(title, {
|
const n = new Notification(title, {
|
||||||
|
203
src/api/Notifications/notificationLog.tsx
Normal file
203
src/api/Notifications/notificationLog.tsx
Normal file
@ -0,0 +1,203 @@
|
|||||||
|
/*
|
||||||
|
* Vencord, a modification for Discord's desktop app
|
||||||
|
* Copyright (c) 2023 Vendicated and contributors
|
||||||
|
*
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import * as DataStore from "@api/DataStore";
|
||||||
|
import { Settings } from "@api/Settings";
|
||||||
|
import { classNameFactory } from "@api/Styles";
|
||||||
|
import { closeModal, ModalCloseButton, ModalContent, ModalFooter, ModalHeader, ModalProps, ModalRoot, ModalSize, openModal } from "@utils/modal";
|
||||||
|
import { useAwaiter } from "@utils/react";
|
||||||
|
import { Alerts, Button, Forms, moment, React, Text, Timestamp, useEffect, useReducer, useState } from "@webpack/common";
|
||||||
|
import { nanoid } from "nanoid";
|
||||||
|
import type { DispatchWithoutAction } from "react";
|
||||||
|
|
||||||
|
import NotificationComponent from "./NotificationComponent";
|
||||||
|
import type { NotificationData } from "./Notifications";
|
||||||
|
|
||||||
|
interface PersistentNotificationData extends Pick<NotificationData, "title" | "body" | "image" | "icon" | "color"> {
|
||||||
|
timestamp: number;
|
||||||
|
id: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
const KEY = "notification-log";
|
||||||
|
|
||||||
|
const getLog = async () => {
|
||||||
|
const log = await DataStore.get(KEY) as PersistentNotificationData[] | undefined;
|
||||||
|
return log ?? [];
|
||||||
|
};
|
||||||
|
|
||||||
|
const cl = classNameFactory("vc-notification-log-");
|
||||||
|
const signals = new Set<DispatchWithoutAction>();
|
||||||
|
|
||||||
|
export async function persistNotification(notification: NotificationData) {
|
||||||
|
if (notification.noPersist) return;
|
||||||
|
|
||||||
|
const limit = Settings.notifications.logLimit;
|
||||||
|
if (limit === 0) return;
|
||||||
|
|
||||||
|
await DataStore.update(KEY, (old: PersistentNotificationData[] | undefined) => {
|
||||||
|
const log = old ?? [];
|
||||||
|
|
||||||
|
// Omit stuff we don't need
|
||||||
|
const {
|
||||||
|
onClick, onClose, richBody, permanent, noPersist, dismissOnClick,
|
||||||
|
...pureNotification
|
||||||
|
} = notification;
|
||||||
|
|
||||||
|
log.unshift({
|
||||||
|
...pureNotification,
|
||||||
|
timestamp: Date.now(),
|
||||||
|
id: nanoid()
|
||||||
|
});
|
||||||
|
|
||||||
|
if (log.length > limit && limit !== 200)
|
||||||
|
log.length = limit;
|
||||||
|
|
||||||
|
return log;
|
||||||
|
});
|
||||||
|
|
||||||
|
signals.forEach(x => x());
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function deleteNotification(timestamp: number) {
|
||||||
|
const log = await getLog();
|
||||||
|
const index = log.findIndex(x => x.timestamp === timestamp);
|
||||||
|
if (index === -1) return;
|
||||||
|
|
||||||
|
log.splice(index, 1);
|
||||||
|
await DataStore.set(KEY, log);
|
||||||
|
signals.forEach(x => x());
|
||||||
|
}
|
||||||
|
|
||||||
|
export function useLogs() {
|
||||||
|
const [signal, setSignal] = useReducer(x => x + 1, 0);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
signals.add(setSignal);
|
||||||
|
return () => void signals.delete(setSignal);
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const [log, _, pending] = useAwaiter(getLog, {
|
||||||
|
fallbackValue: [],
|
||||||
|
deps: [signal]
|
||||||
|
});
|
||||||
|
|
||||||
|
return [log, pending] as const;
|
||||||
|
}
|
||||||
|
|
||||||
|
function NotificationEntry({ data }: { data: PersistentNotificationData; }) {
|
||||||
|
const [removing, setRemoving] = useState(false);
|
||||||
|
const ref = React.useRef<HTMLDivElement>(null);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const div = ref.current!;
|
||||||
|
|
||||||
|
const setHeight = () => {
|
||||||
|
if (div.clientHeight === 0) return requestAnimationFrame(setHeight);
|
||||||
|
div.style.height = `${div.clientHeight}px`;
|
||||||
|
};
|
||||||
|
|
||||||
|
setHeight();
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className={cl("wrapper", { removing })} ref={ref}>
|
||||||
|
<NotificationComponent
|
||||||
|
{...data}
|
||||||
|
permanent={true}
|
||||||
|
dismissOnClick={false}
|
||||||
|
onClose={() => {
|
||||||
|
if (removing) return;
|
||||||
|
setRemoving(true);
|
||||||
|
|
||||||
|
setTimeout(() => deleteNotification(data.timestamp), 200);
|
||||||
|
}}
|
||||||
|
richBody={
|
||||||
|
<div className={cl("body")}>
|
||||||
|
{data.body}
|
||||||
|
<Timestamp timestamp={moment(data.timestamp)} className={cl("timestamp")} />
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
export function NotificationLog({ log, pending }: { log: PersistentNotificationData[], pending: boolean; }) {
|
||||||
|
if (!log.length && !pending)
|
||||||
|
return (
|
||||||
|
<div className={cl("container")}>
|
||||||
|
<div className={cl("empty")} />
|
||||||
|
<Forms.FormText style={{ textAlign: "center" }}>
|
||||||
|
No notifications yet
|
||||||
|
</Forms.FormText>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className={cl("container")}>
|
||||||
|
{log.map(n => <NotificationEntry data={n} key={n.id} />)}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
function LogModal({ modalProps, close }: { modalProps: ModalProps; close(): void; }) {
|
||||||
|
const [log, pending] = useLogs();
|
||||||
|
|
||||||
|
return (
|
||||||
|
<ModalRoot {...modalProps} size={ModalSize.LARGE}>
|
||||||
|
<ModalHeader>
|
||||||
|
<Text variant="heading-lg/semibold" style={{ flexGrow: 1 }}>Notification Log</Text>
|
||||||
|
<ModalCloseButton onClick={close} />
|
||||||
|
</ModalHeader>
|
||||||
|
|
||||||
|
<ModalContent>
|
||||||
|
<NotificationLog log={log} pending={pending} />
|
||||||
|
</ModalContent>
|
||||||
|
|
||||||
|
<ModalFooter>
|
||||||
|
<Button
|
||||||
|
disabled={log.length === 0}
|
||||||
|
onClick={() => {
|
||||||
|
Alerts.show({
|
||||||
|
title: "Are you sure?",
|
||||||
|
body: `This will permanently remove ${log.length} notification${log.length === 1 ? "" : "s"}. This action cannot be undone.`,
|
||||||
|
async onConfirm() {
|
||||||
|
await DataStore.set(KEY, []);
|
||||||
|
signals.forEach(x => x());
|
||||||
|
},
|
||||||
|
confirmText: "Do it!",
|
||||||
|
confirmColor: "vc-notification-log-danger-btn",
|
||||||
|
cancelText: "Nevermind"
|
||||||
|
});
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
Clear Notification Log
|
||||||
|
</Button>
|
||||||
|
</ModalFooter>
|
||||||
|
</ModalRoot>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
export function openNotificationLogModal() {
|
||||||
|
const key = openModal(modalProps => (
|
||||||
|
<LogModal
|
||||||
|
modalProps={modalProps}
|
||||||
|
close={() => closeModal(key)}
|
||||||
|
/>
|
||||||
|
));
|
||||||
|
}
|
@ -3,16 +3,20 @@
|
|||||||
all: unset;
|
all: unset;
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
width: 25vw;
|
|
||||||
min-height: 10vh;
|
|
||||||
color: var(--text-normal);
|
color: var(--text-normal);
|
||||||
background-color: var(--background-secondary-alt);
|
background-color: var(--background-secondary-alt);
|
||||||
position: absolute;
|
|
||||||
z-index: 2147483647;
|
|
||||||
right: 1rem;
|
|
||||||
border-radius: 6px;
|
border-radius: 6px;
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.vc-notification-root:not(.vc-notification-log-wrapper > .vc-notification-root) {
|
||||||
|
position: absolute;
|
||||||
|
z-index: 2147483647;
|
||||||
|
right: 1rem;
|
||||||
|
width: 25vw;
|
||||||
|
min-height: 10vh;
|
||||||
}
|
}
|
||||||
|
|
||||||
.vc-notification {
|
.vc-notification {
|
||||||
@ -22,17 +26,42 @@
|
|||||||
gap: 1.25rem;
|
gap: 1.25rem;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.vc-notification-content {
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.vc-notification-header {
|
||||||
|
display: flex;
|
||||||
|
justify-content: space-between;
|
||||||
|
}
|
||||||
|
|
||||||
|
.vc-notification-title {
|
||||||
|
color: var(--header-primary);
|
||||||
|
font-size: 1rem;
|
||||||
|
font-weight: 600;
|
||||||
|
line-height: 1.25rem;
|
||||||
|
text-transform: uppercase;
|
||||||
|
}
|
||||||
|
|
||||||
|
.vc-notification-close-btn {
|
||||||
|
all: unset;
|
||||||
|
cursor: pointer;
|
||||||
|
color: var(--interactive-normal);
|
||||||
|
opacity: 0.5;
|
||||||
|
transition: opacity 0.2s ease-in-out, color 0.2s ease-in-out;
|
||||||
|
}
|
||||||
|
|
||||||
|
.vc-notification-close-btn:hover {
|
||||||
|
color: var(--interactive-hover);
|
||||||
|
opacity: 1;
|
||||||
|
}
|
||||||
|
|
||||||
.vc-notification-icon {
|
.vc-notification-icon {
|
||||||
height: 4rem;
|
height: 4rem;
|
||||||
width: 4rem;
|
width: 4rem;
|
||||||
border-radius: 6px;
|
border-radius: 6px;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Discord adding 3km margin to generic tags */
|
|
||||||
.vc-notification h2 {
|
|
||||||
margin: unset;
|
|
||||||
}
|
|
||||||
|
|
||||||
.vc-notification-progressbar {
|
.vc-notification-progressbar {
|
||||||
height: 0.25rem;
|
height: 0.25rem;
|
||||||
border-radius: 5px;
|
border-radius: 5px;
|
||||||
@ -47,3 +76,47 @@
|
|||||||
.vc-notification-img {
|
.vc-notification-img {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.vc-notification-log-empty {
|
||||||
|
height: 218px;
|
||||||
|
background: url("/assets/b36de980b174d7b798c89f35c116e5c6.svg") center no-repeat;
|
||||||
|
margin-bottom: 40px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.vc-notification-log-container {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
padding: 1em;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
.vc-notification-log-wrapper {
|
||||||
|
transition: 200ms ease;
|
||||||
|
transition-property: height, opacity;
|
||||||
|
}
|
||||||
|
|
||||||
|
.vc-notification-log-wrapper:not(:last-child) {
|
||||||
|
margin-bottom: 1em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.vc-notification-log-removing {
|
||||||
|
height: 0 !important;
|
||||||
|
opacity: 0;
|
||||||
|
margin-bottom: 1em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.vc-notification-log-body {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
}
|
||||||
|
|
||||||
|
.vc-notification-log-timestamp {
|
||||||
|
margin-left: auto;
|
||||||
|
font-size: 0.8em;
|
||||||
|
font-weight: lighter;
|
||||||
|
}
|
||||||
|
|
||||||
|
.vc-notification-log-danger-btn {
|
||||||
|
color: var(--white-500);
|
||||||
|
background-color: var(--button-danger-background);
|
||||||
|
}
|
||||||
|
@ -16,11 +16,11 @@
|
|||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import Logger from "@utils/Logger";
|
import { Logger } from "@utils/Logger";
|
||||||
|
|
||||||
const logger = new Logger("ServerListAPI");
|
const logger = new Logger("ServerListAPI");
|
||||||
|
|
||||||
export enum ServerListRenderPosition {
|
export const enum ServerListRenderPosition {
|
||||||
Above,
|
Above,
|
||||||
In,
|
In,
|
||||||
}
|
}
|
||||||
|
@ -16,9 +16,11 @@
|
|||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import IpcEvents from "@utils/IpcEvents";
|
import { debounce } from "@utils/debounce";
|
||||||
import Logger from "@utils/Logger";
|
import { localStorage } from "@utils/localStorage";
|
||||||
|
import { Logger } from "@utils/Logger";
|
||||||
import { mergeDefaults } from "@utils/misc";
|
import { mergeDefaults } from "@utils/misc";
|
||||||
|
import { putCloudSettings } from "@utils/settingsSync";
|
||||||
import { DefinedSettings, OptionType, SettingsChecks, SettingsDefinition } from "@utils/types";
|
import { DefinedSettings, OptionType, SettingsChecks, SettingsDefinition } from "@utils/types";
|
||||||
import { React } from "@webpack/common";
|
import { React } from "@webpack/common";
|
||||||
|
|
||||||
@ -28,12 +30,17 @@ const logger = new Logger("Settings");
|
|||||||
export interface Settings {
|
export interface Settings {
|
||||||
notifyAboutUpdates: boolean;
|
notifyAboutUpdates: boolean;
|
||||||
autoUpdate: boolean;
|
autoUpdate: boolean;
|
||||||
|
autoUpdateNotification: boolean,
|
||||||
useQuickCss: boolean;
|
useQuickCss: boolean;
|
||||||
enableReactDevtools: boolean;
|
enableReactDevtools: boolean;
|
||||||
themeLinks: string[];
|
themeLinks: string[];
|
||||||
|
enabledThemes: string[];
|
||||||
frameless: boolean;
|
frameless: boolean;
|
||||||
transparent: boolean;
|
transparent: boolean;
|
||||||
winCtrlQ: boolean;
|
winCtrlQ: boolean;
|
||||||
|
macosTranslucency: boolean;
|
||||||
|
disableMinSize: boolean;
|
||||||
|
winNativeTitleBar: boolean;
|
||||||
plugins: {
|
plugins: {
|
||||||
[plugin: string]: {
|
[plugin: string]: {
|
||||||
enabled: boolean;
|
enabled: boolean;
|
||||||
@ -45,36 +52,64 @@ export interface Settings {
|
|||||||
timeout: number;
|
timeout: number;
|
||||||
position: "top-right" | "bottom-right";
|
position: "top-right" | "bottom-right";
|
||||||
useNative: "always" | "never" | "not-focused";
|
useNative: "always" | "never" | "not-focused";
|
||||||
|
logLimit: number;
|
||||||
|
};
|
||||||
|
|
||||||
|
cloud: {
|
||||||
|
authenticated: boolean;
|
||||||
|
url: string;
|
||||||
|
settingsSync: boolean;
|
||||||
|
settingsSyncVersion: number;
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
const DefaultSettings: Settings = {
|
const DefaultSettings: Settings = {
|
||||||
notifyAboutUpdates: true,
|
notifyAboutUpdates: true,
|
||||||
autoUpdate: false,
|
autoUpdate: false,
|
||||||
|
autoUpdateNotification: true,
|
||||||
useQuickCss: true,
|
useQuickCss: true,
|
||||||
themeLinks: [],
|
themeLinks: [],
|
||||||
|
enabledThemes: [],
|
||||||
enableReactDevtools: false,
|
enableReactDevtools: false,
|
||||||
frameless: false,
|
frameless: false,
|
||||||
transparent: false,
|
transparent: false,
|
||||||
winCtrlQ: false,
|
winCtrlQ: false,
|
||||||
|
macosTranslucency: false,
|
||||||
|
disableMinSize: false,
|
||||||
|
winNativeTitleBar: false,
|
||||||
plugins: {},
|
plugins: {},
|
||||||
|
|
||||||
notifications: {
|
notifications: {
|
||||||
timeout: 5000,
|
timeout: 5000,
|
||||||
position: "bottom-right",
|
position: "bottom-right",
|
||||||
useNative: "not-focused"
|
useNative: "not-focused",
|
||||||
|
logLimit: 50
|
||||||
|
},
|
||||||
|
|
||||||
|
cloud: {
|
||||||
|
authenticated: false,
|
||||||
|
url: "https://api.vencord.dev/",
|
||||||
|
settingsSync: false,
|
||||||
|
settingsSyncVersion: 0
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
try {
|
try {
|
||||||
var settings = JSON.parse(VencordNative.ipc.sendSync(IpcEvents.GET_SETTINGS)) as Settings;
|
var settings = JSON.parse(VencordNative.settings.get()) as Settings;
|
||||||
mergeDefaults(settings, DefaultSettings);
|
mergeDefaults(settings, DefaultSettings);
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
var settings = mergeDefaults({} as Settings, DefaultSettings);
|
var settings = mergeDefaults({} as Settings, DefaultSettings);
|
||||||
logger.error("An error occurred while loading the settings. Corrupt settings file?\n", err);
|
logger.error("An error occurred while loading the settings. Corrupt settings file?\n", err);
|
||||||
}
|
}
|
||||||
|
|
||||||
type SubscriptionCallback = ((newValue: any, path: string) => void) & { _path?: string; };
|
const saveSettingsOnFrequentAction = debounce(async () => {
|
||||||
|
if (Settings.cloud.settingsSync && Settings.cloud.authenticated) {
|
||||||
|
await putCloudSettings();
|
||||||
|
delete localStorage.Vencord_settingsDirty;
|
||||||
|
}
|
||||||
|
}, 60_000);
|
||||||
|
|
||||||
|
type SubscriptionCallback = ((newValue: any, path: string) => void) & { _paths?: Array<string>; };
|
||||||
const subscriptions = new Set<SubscriptionCallback>();
|
const subscriptions = new Set<SubscriptionCallback>();
|
||||||
|
|
||||||
const proxyCache = {} as Record<string, any>;
|
const proxyCache = {} as Record<string, any>;
|
||||||
@ -90,7 +125,7 @@ function makeProxy(settings: any, root = settings, path = ""): Settings {
|
|||||||
// Return empty for plugins with no settings
|
// Return empty for plugins with no settings
|
||||||
if (path === "plugins" && p in plugins)
|
if (path === "plugins" && p in plugins)
|
||||||
return target[p] = makeProxy({
|
return target[p] = makeProxy({
|
||||||
enabled: plugins[p].required ?? false
|
enabled: plugins[p].required ?? plugins[p].enabledByDefault ?? false
|
||||||
}, root, `plugins.${p}`);
|
}, root, `plugins.${p}`);
|
||||||
|
|
||||||
// Since the property is not set, check if this is a plugin's setting and if so, try to resolve
|
// Since the property is not set, check if this is a plugin's setting and if so, try to resolve
|
||||||
@ -129,13 +164,17 @@ function makeProxy(settings: any, root = settings, path = ""): Settings {
|
|||||||
target[p] = v;
|
target[p] = v;
|
||||||
// Call any listeners that are listening to a setting of this path
|
// Call any listeners that are listening to a setting of this path
|
||||||
const setPath = `${path}${path && "."}${p}`;
|
const setPath = `${path}${path && "."}${p}`;
|
||||||
|
delete proxyCache[setPath];
|
||||||
for (const subscription of subscriptions) {
|
for (const subscription of subscriptions) {
|
||||||
if (!subscription._path || subscription._path === setPath) {
|
if (!subscription._paths || subscription._paths.includes(setPath)) {
|
||||||
subscription(v, setPath);
|
subscription(v, setPath);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// And don't forget to persist the settings!
|
// And don't forget to persist the settings!
|
||||||
VencordNative.ipc.invoke(IpcEvents.SET_SETTINGS, JSON.stringify(root, null, 4));
|
PlainSettings.cloud.settingsSyncVersion = Date.now();
|
||||||
|
localStorage.Vencord_settingsDirty = true;
|
||||||
|
saveSettingsOnFrequentAction();
|
||||||
|
VencordNative.settings.set(JSON.stringify(root, null, 4));
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
@ -165,11 +204,11 @@ export const Settings = makeProxy(settings);
|
|||||||
* @returns Settings
|
* @returns Settings
|
||||||
*/
|
*/
|
||||||
// TODO: Representing paths as essentially "string[].join('.')" wont allow dots in paths, change to "paths?: string[][]" later
|
// TODO: Representing paths as essentially "string[].join('.')" wont allow dots in paths, change to "paths?: string[][]" later
|
||||||
export function useSettings(paths?: string[]) {
|
export function useSettings(paths?: UseSettings<Settings>[]) {
|
||||||
const [, forceUpdate] = React.useReducer(() => ({}), {});
|
const [, forceUpdate] = React.useReducer(() => ({}), {});
|
||||||
|
|
||||||
const onUpdate: SubscriptionCallback = paths
|
const onUpdate: SubscriptionCallback = paths
|
||||||
? (value, path) => paths.includes(path) && forceUpdate()
|
? (value, path) => paths.includes(path as UseSettings<Settings>) && forceUpdate()
|
||||||
: forceUpdate;
|
: forceUpdate;
|
||||||
|
|
||||||
React.useEffect(() => {
|
React.useEffect(() => {
|
||||||
@ -198,7 +237,7 @@ type ResolvePropDeep<T, P> = P extends "" ? T :
|
|||||||
export function addSettingsListener<Path extends keyof Settings>(path: Path, onUpdate: (newValue: Settings[Path], path: Path) => void): void;
|
export function addSettingsListener<Path extends keyof Settings>(path: Path, onUpdate: (newValue: Settings[Path], path: Path) => void): void;
|
||||||
export function addSettingsListener<Path extends string>(path: Path, onUpdate: (newValue: Path extends "" ? any : ResolvePropDeep<Settings, Path>, path: Path extends "" ? string : Path) => void): void;
|
export function addSettingsListener<Path extends string>(path: Path, onUpdate: (newValue: Path extends "" ? any : ResolvePropDeep<Settings, Path>, path: Path extends "" ? string : Path) => void): void;
|
||||||
export function addSettingsListener(path: string, onUpdate: (newValue: any, path: string) => void) {
|
export function addSettingsListener(path: string, onUpdate: (newValue: any, path: string) => void) {
|
||||||
(onUpdate as SubscriptionCallback)._path = path;
|
((onUpdate as SubscriptionCallback)._paths ??= []).push(path);
|
||||||
subscriptions.add(onUpdate);
|
subscriptions.add(onUpdate);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -211,27 +250,45 @@ export function migratePluginSettings(name: string, ...oldNames: string[]) {
|
|||||||
logger.info(`Migrating settings from old name ${oldName} to ${name}`);
|
logger.info(`Migrating settings from old name ${oldName} to ${name}`);
|
||||||
plugins[name] = plugins[oldName];
|
plugins[name] = plugins[oldName];
|
||||||
delete plugins[oldName];
|
delete plugins[oldName];
|
||||||
VencordNative.ipc.invoke(
|
VencordNative.settings.set(JSON.stringify(settings, null, 4));
|
||||||
IpcEvents.SET_SETTINGS,
|
|
||||||
JSON.stringify(settings, null, 4)
|
|
||||||
);
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export function definePluginSettings<D extends SettingsDefinition, C extends SettingsChecks<D>>(def: D, checks?: C) {
|
export function definePluginSettings<
|
||||||
const definedSettings: DefinedSettings<D> = {
|
Def extends SettingsDefinition,
|
||||||
|
Checks extends SettingsChecks<Def>,
|
||||||
|
PrivateSettings extends object = {}
|
||||||
|
>(def: Def, checks?: Checks) {
|
||||||
|
const definedSettings: DefinedSettings<Def, Checks, PrivateSettings> = {
|
||||||
get store() {
|
get store() {
|
||||||
if (!definedSettings.pluginName) throw new Error("Cannot access settings before plugin is initialized");
|
if (!definedSettings.pluginName) throw new Error("Cannot access settings before plugin is initialized");
|
||||||
return Settings.plugins[definedSettings.pluginName] as any;
|
return Settings.plugins[definedSettings.pluginName] as any;
|
||||||
},
|
},
|
||||||
use: settings => useSettings(
|
use: settings => useSettings(
|
||||||
settings?.map(name => `plugins.${definedSettings.pluginName}.${name}`)
|
settings?.map(name => `plugins.${definedSettings.pluginName}.${name}`) as UseSettings<Settings>[]
|
||||||
).plugins[definedSettings.pluginName] as any,
|
).plugins[definedSettings.pluginName] as any,
|
||||||
def,
|
def,
|
||||||
checks: checks ?? {},
|
checks: checks ?? {} as any,
|
||||||
pluginName: "",
|
pluginName: "",
|
||||||
|
|
||||||
|
withPrivateSettings<T extends object>() {
|
||||||
|
return this as DefinedSettings<Def, Checks, T>;
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
return definedSettings;
|
return definedSettings;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type UseSettings<T extends object> = ResolveUseSettings<T>[keyof T];
|
||||||
|
|
||||||
|
type ResolveUseSettings<T extends object> = {
|
||||||
|
[Key in keyof T]:
|
||||||
|
Key extends string
|
||||||
|
? T[Key] extends Record<string, unknown>
|
||||||
|
// @ts-ignore "Type instantiation is excessively deep and possibly infinite"
|
||||||
|
? UseSettings<T[Key]> extends string ? `${Key}.${UseSettings<T[Key]>}` : never
|
||||||
|
: Key
|
||||||
|
: never;
|
||||||
|
};
|
69
src/api/SettingsStore.ts
Normal file
69
src/api/SettingsStore.ts
Normal file
@ -0,0 +1,69 @@
|
|||||||
|
/*
|
||||||
|
* Vencord, a modification for Discord's desktop app
|
||||||
|
* Copyright (c) 2023 Vendicated and contributors
|
||||||
|
*
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { proxyLazy } from "@utils/lazy";
|
||||||
|
import { Logger } from "@utils/Logger";
|
||||||
|
import { findModuleId, wreq } from "@webpack";
|
||||||
|
|
||||||
|
import { Settings } from "./Settings";
|
||||||
|
|
||||||
|
interface Setting<T> {
|
||||||
|
/**
|
||||||
|
* Get the setting value
|
||||||
|
*/
|
||||||
|
getSetting(): T;
|
||||||
|
/**
|
||||||
|
* Update the setting value
|
||||||
|
* @param value The new value
|
||||||
|
*/
|
||||||
|
updateSetting(value: T | ((old: T) => T)): Promise<void>;
|
||||||
|
/**
|
||||||
|
* React hook for automatically updating components when the setting is updated
|
||||||
|
*/
|
||||||
|
useSetting(): T;
|
||||||
|
settingsStoreApiGroup: string;
|
||||||
|
settingsStoreApiName: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
const SettingsStores: Array<Setting<any>> | undefined = proxyLazy(() => {
|
||||||
|
const modId = findModuleId('"textAndImages","renderSpoilers"');
|
||||||
|
if (modId == null) return new Logger("SettingsStoreAPI").error("Didn't find stores module.");
|
||||||
|
|
||||||
|
const mod = wreq(modId);
|
||||||
|
if (mod == null) return;
|
||||||
|
|
||||||
|
return Object.values(mod).filter((s: any) => s?.settingsStoreApiGroup) as any;
|
||||||
|
});
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the store for a setting
|
||||||
|
* @param group The setting group
|
||||||
|
* @param name The name of the setting
|
||||||
|
*/
|
||||||
|
export function getSettingStore<T = any>(group: string, name: string): Setting<T> | undefined {
|
||||||
|
if (!Settings.plugins.SettingsStoreAPI.enabled) throw new Error("Cannot use SettingsStoreAPI without setting as dependency.");
|
||||||
|
|
||||||
|
return SettingsStores?.find(s => s?.settingsStoreApiGroup === group && s?.settingsStoreApiName === name);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* getSettingStore but lazy
|
||||||
|
*/
|
||||||
|
export function getSettingStoreLazy<T = any>(group: string, name: string) {
|
||||||
|
return proxyLazy(() => getSettingStore<T>(group, name));
|
||||||
|
}
|
@ -141,7 +141,7 @@ export const compileStyle = (style: Style) => {
|
|||||||
*/
|
*/
|
||||||
export const classNameToSelector = (name: string, prefix = "") => name.split(" ").map(n => `.${prefix}${n}`).join("");
|
export const classNameToSelector = (name: string, prefix = "") => name.split(" ").map(n => `.${prefix}${n}`).join("");
|
||||||
|
|
||||||
type ClassNameFactoryArg = string | string[] | Record<string, unknown>;
|
type ClassNameFactoryArg = string | string[] | Record<string, unknown> | false | null | undefined | 0 | "";
|
||||||
/**
|
/**
|
||||||
* @param prefix The prefix to add to each class, defaults to `""`
|
* @param prefix The prefix to add to each class, defaults to `""`
|
||||||
* @returns A classname generator function
|
* @returns A classname generator function
|
||||||
@ -154,9 +154,9 @@ type ClassNameFactoryArg = string | string[] | Record<string, unknown>;
|
|||||||
export const classNameFactory = (prefix: string = "") => (...args: ClassNameFactoryArg[]) => {
|
export const classNameFactory = (prefix: string = "") => (...args: ClassNameFactoryArg[]) => {
|
||||||
const classNames = new Set<string>();
|
const classNames = new Set<string>();
|
||||||
for (const arg of args) {
|
for (const arg of args) {
|
||||||
if (typeof arg === "string") classNames.add(arg);
|
if (arg && typeof arg === "string") classNames.add(arg);
|
||||||
else if (Array.isArray(arg)) arg.forEach(name => classNames.add(name));
|
else if (Array.isArray(arg)) arg.forEach(name => classNames.add(name));
|
||||||
else if (typeof arg === "object") Object.entries(arg).forEach(([name, value]) => value && classNames.add(name));
|
else if (arg && typeof arg === "object") Object.entries(arg).forEach(([name, value]) => value && classNames.add(name));
|
||||||
}
|
}
|
||||||
return Array.from(classNames, name => prefix + name).join(" ");
|
return Array.from(classNames, name => prefix + name).join(" ");
|
||||||
};
|
};
|
||||||
|
@ -18,6 +18,7 @@
|
|||||||
|
|
||||||
import * as $Badges from "./Badges";
|
import * as $Badges from "./Badges";
|
||||||
import * as $Commands from "./Commands";
|
import * as $Commands from "./Commands";
|
||||||
|
import * as $ContextMenu from "./ContextMenu";
|
||||||
import * as $DataStore from "./DataStore";
|
import * as $DataStore from "./DataStore";
|
||||||
import * as $MemberListDecorators from "./MemberListDecorators";
|
import * as $MemberListDecorators from "./MemberListDecorators";
|
||||||
import * as $MessageAccessories from "./MessageAccessories";
|
import * as $MessageAccessories from "./MessageAccessories";
|
||||||
@ -27,6 +28,8 @@ import * as $MessagePopover from "./MessagePopover";
|
|||||||
import * as $Notices from "./Notices";
|
import * as $Notices from "./Notices";
|
||||||
import * as $Notifications from "./Notifications";
|
import * as $Notifications from "./Notifications";
|
||||||
import * as $ServerList from "./ServerList";
|
import * as $ServerList from "./ServerList";
|
||||||
|
import * as $Settings from "./Settings";
|
||||||
|
import * as $SettingsStore from "./SettingsStore";
|
||||||
import * as $Styles from "./Styles";
|
import * as $Styles from "./Styles";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -84,6 +87,14 @@ export const MessageDecorations = $MessageDecorations;
|
|||||||
* An API allowing you to add components to member list users, in both DM's and servers
|
* An API allowing you to add components to member list users, in both DM's and servers
|
||||||
*/
|
*/
|
||||||
export const MemberListDecorators = $MemberListDecorators;
|
export const MemberListDecorators = $MemberListDecorators;
|
||||||
|
/**
|
||||||
|
* An API allowing you to persist data
|
||||||
|
*/
|
||||||
|
export const Settings = $Settings;
|
||||||
|
/**
|
||||||
|
* An API allowing you to read, manipulate and automatically update components based on Discord settings
|
||||||
|
*/
|
||||||
|
export const SettingsStore = $SettingsStore;
|
||||||
/**
|
/**
|
||||||
* An API allowing you to dynamically load styles
|
* An API allowing you to dynamically load styles
|
||||||
* a
|
* a
|
||||||
@ -93,3 +104,8 @@ export const Styles = $Styles;
|
|||||||
* An API allowing you to display notifications
|
* An API allowing you to display notifications
|
||||||
*/
|
*/
|
||||||
export const Notifications = $Notifications;
|
export const Notifications = $Notifications;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* An api allowing you to patch and add/remove items to/from context menus
|
||||||
|
*/
|
||||||
|
export const ContextMenu = $ContextMenu;
|
||||||
|
@ -16,7 +16,6 @@
|
|||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import IpcEvents from "@utils/IpcEvents";
|
|
||||||
import { Button } from "@webpack/common";
|
import { Button } from "@webpack/common";
|
||||||
|
|
||||||
import { Heart } from "./Heart";
|
import { Heart } from "./Heart";
|
||||||
@ -27,9 +26,7 @@ export default function DonateButton(props: any) {
|
|||||||
{...props}
|
{...props}
|
||||||
look={Button.Looks.LINK}
|
look={Button.Looks.LINK}
|
||||||
color={Button.Colors.TRANSPARENT}
|
color={Button.Colors.TRANSPARENT}
|
||||||
onClick={() =>
|
onClick={() => VencordNative.native.openExternal("https://github.com/sponsors/Vendicated")}
|
||||||
VencordNative.ipc.invoke(IpcEvents.OPEN_EXTERNAL, "https://github.com/sponsors/Vendicated")
|
|
||||||
}
|
|
||||||
>
|
>
|
||||||
<Heart />
|
<Heart />
|
||||||
Donate
|
Donate
|
||||||
|
@ -16,21 +16,25 @@
|
|||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import Logger from "@utils/Logger";
|
import { Logger } from "@utils/Logger";
|
||||||
import { LazyComponent } from "@utils/misc";
|
import { Margins } from "@utils/margins";
|
||||||
import { Margins, React } from "@webpack/common";
|
import { LazyComponent } from "@utils/react";
|
||||||
|
import { React } from "@webpack/common";
|
||||||
|
|
||||||
import { ErrorCard } from "./ErrorCard";
|
import { ErrorCard } from "./ErrorCard";
|
||||||
|
|
||||||
interface Props {
|
interface Props<T = any> {
|
||||||
/** Render nothing if an error occurs */
|
/** Render nothing if an error occurs */
|
||||||
noop?: boolean;
|
noop?: boolean;
|
||||||
/** Fallback component to render if an error occurs */
|
/** Fallback component to render if an error occurs */
|
||||||
fallback?: React.ComponentType<React.PropsWithChildren<{ error: any; message: string; stack: string; }>>;
|
fallback?: React.ComponentType<React.PropsWithChildren<{ error: any; message: string; stack: string; }>>;
|
||||||
/** called when an error occurs */
|
/** called when an error occurs. The props property is only available if using .wrap */
|
||||||
onError?(error: Error, errorInfo: React.ErrorInfo): void;
|
onError?(data: { error: Error, errorInfo: React.ErrorInfo, props: T; }): void;
|
||||||
/** Custom error message */
|
/** Custom error message */
|
||||||
message?: string;
|
message?: string;
|
||||||
|
|
||||||
|
/** The props passed to the wrapped component. Only used by wrap */
|
||||||
|
wrappedProps?: T;
|
||||||
}
|
}
|
||||||
|
|
||||||
const color = "#e78284";
|
const color = "#e78284";
|
||||||
@ -65,7 +69,7 @@ const ErrorBoundary = LazyComponent(() => {
|
|||||||
}
|
}
|
||||||
|
|
||||||
componentDidCatch(error: Error, errorInfo: React.ErrorInfo) {
|
componentDidCatch(error: Error, errorInfo: React.ErrorInfo) {
|
||||||
this.props.onError?.(error, errorInfo);
|
this.props.onError?.({ error, errorInfo, props: this.props.wrappedProps });
|
||||||
logger.error("A component threw an Error\n", error);
|
logger.error("A component threw an Error\n", error);
|
||||||
logger.error("Component Stack", errorInfo.componentStack);
|
logger.error("Component Stack", errorInfo.componentStack);
|
||||||
}
|
}
|
||||||
@ -84,15 +88,13 @@ const ErrorBoundary = LazyComponent(() => {
|
|||||||
const msg = this.props.message || "An error occurred while rendering this Component. More info can be found below and in your console.";
|
const msg = this.props.message || "An error occurred while rendering this Component. More info can be found below and in your console.";
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<ErrorCard style={{
|
<ErrorCard style={{ overflow: "hidden" }}>
|
||||||
overflow: "hidden",
|
|
||||||
}}>
|
|
||||||
<h1>Oh no!</h1>
|
<h1>Oh no!</h1>
|
||||||
<p>{msg}</p>
|
<p>{msg}</p>
|
||||||
<code>
|
<code>
|
||||||
{this.state.message}
|
{this.state.message}
|
||||||
{!!this.state.stack && (
|
{!!this.state.stack && (
|
||||||
<pre className={Margins.marginTop8}>
|
<pre className={Margins.top8}>
|
||||||
{this.state.stack}
|
{this.state.stack}
|
||||||
</pre>
|
</pre>
|
||||||
)}
|
)}
|
||||||
@ -103,11 +105,11 @@ const ErrorBoundary = LazyComponent(() => {
|
|||||||
};
|
};
|
||||||
}) as
|
}) as
|
||||||
React.ComponentType<React.PropsWithChildren<Props>> & {
|
React.ComponentType<React.PropsWithChildren<Props>> & {
|
||||||
wrap<T extends object = any>(Component: React.ComponentType<T>, errorBoundaryProps?: Props): React.ComponentType<T>;
|
wrap<T extends object = any>(Component: React.ComponentType<T>, errorBoundaryProps?: Omit<Props<T>, "wrappedProps">): React.FunctionComponent<T>;
|
||||||
};
|
};
|
||||||
|
|
||||||
ErrorBoundary.wrap = (Component, errorBoundaryProps) => props => (
|
ErrorBoundary.wrap = (Component, errorBoundaryProps) => props => (
|
||||||
<ErrorBoundary {...errorBoundaryProps}>
|
<ErrorBoundary {...errorBoundaryProps} wrappedProps={props}>
|
||||||
<Component {...props} />
|
<Component {...props} />
|
||||||
</ErrorBoundary>
|
</ErrorBoundary>
|
||||||
);
|
);
|
||||||
|
7
src/components/ErrorCard.css
Normal file
7
src/components/ErrorCard.css
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
.vc-error-card {
|
||||||
|
padding: 2em;
|
||||||
|
background-color: #e7828430;
|
||||||
|
border: 1px solid #e78284;
|
||||||
|
border-radius: 5px;
|
||||||
|
color: var(--text-normal, white);
|
||||||
|
}
|
@ -16,24 +16,15 @@
|
|||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import { Card } from "@webpack/common";
|
import "./ErrorCard.css";
|
||||||
|
|
||||||
interface Props {
|
import { classes } from "@utils/misc";
|
||||||
style?: React.CSSProperties;
|
import type { HTMLProps } from "react";
|
||||||
className?: string;
|
|
||||||
}
|
export function ErrorCard(props: React.PropsWithChildren<HTMLProps<HTMLDivElement>>) {
|
||||||
export function ErrorCard(props: React.PropsWithChildren<Props>) {
|
|
||||||
return (
|
return (
|
||||||
<Card className={props.className} style={
|
<div {...props} className={classes(props.className, "vc-error-card")}>
|
||||||
{
|
|
||||||
padding: "2em",
|
|
||||||
backgroundColor: "#e7828430",
|
|
||||||
borderColor: "#e78284",
|
|
||||||
color: "var(--text-normal)",
|
|
||||||
...props.style
|
|
||||||
}
|
|
||||||
}>
|
|
||||||
{props.children}
|
{props.children}
|
||||||
</Card>
|
</div>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
12
src/components/ExpandableHeader.css
Normal file
12
src/components/ExpandableHeader.css
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
.vc-expandableheader-center-flex {
|
||||||
|
display: flex;
|
||||||
|
justify-items: center;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.vc-expandableheader-btn {
|
||||||
|
all: unset;
|
||||||
|
cursor: pointer;
|
||||||
|
width: 24px;
|
||||||
|
height: 24px;
|
||||||
|
}
|
108
src/components/ExpandableHeader.tsx
Normal file
108
src/components/ExpandableHeader.tsx
Normal file
@ -0,0 +1,108 @@
|
|||||||
|
/*
|
||||||
|
* Vencord, a modification for Discord's desktop app
|
||||||
|
* Copyright (c) 2023 Vendicated and contributors
|
||||||
|
*
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { classNameFactory } from "@api/Styles";
|
||||||
|
import { Text, Tooltip, useState } from "@webpack/common";
|
||||||
|
export const cl = classNameFactory("vc-expandableheader-");
|
||||||
|
import "./ExpandableHeader.css";
|
||||||
|
|
||||||
|
export interface ExpandableHeaderProps {
|
||||||
|
onMoreClick?: () => void;
|
||||||
|
moreTooltipText?: string;
|
||||||
|
onDropDownClick?: (state: boolean) => void;
|
||||||
|
defaultState?: boolean;
|
||||||
|
headerText: string;
|
||||||
|
children: React.ReactNode;
|
||||||
|
buttons?: React.ReactNode[];
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function ExpandableHeader({ children, onMoreClick, buttons, moreTooltipText, defaultState = false, onDropDownClick, headerText }: ExpandableHeaderProps) {
|
||||||
|
const [showContent, setShowContent] = useState(defaultState);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<div style={{
|
||||||
|
display: "flex",
|
||||||
|
justifyContent: "space-between",
|
||||||
|
alignItems: "center",
|
||||||
|
marginBottom: "8px"
|
||||||
|
}}>
|
||||||
|
<Text
|
||||||
|
tag="h2"
|
||||||
|
variant="eyebrow"
|
||||||
|
style={{
|
||||||
|
color: "var(--header-primary)",
|
||||||
|
display: "inline"
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{headerText}
|
||||||
|
</Text>
|
||||||
|
|
||||||
|
<div className={cl("center-flex")}>
|
||||||
|
{
|
||||||
|
buttons ?? null
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
onMoreClick && // only show more button if callback is provided
|
||||||
|
<Tooltip text={moreTooltipText}>
|
||||||
|
{tooltipProps => (
|
||||||
|
<button
|
||||||
|
{...tooltipProps}
|
||||||
|
className={cl("btn")}
|
||||||
|
onClick={onMoreClick}>
|
||||||
|
<svg
|
||||||
|
width="24"
|
||||||
|
height="24"
|
||||||
|
viewBox="0 0 24 24"
|
||||||
|
>
|
||||||
|
<path fill="var(--text-normal)" d="M7 12.001C7 10.8964 6.10457 10.001 5 10.001C3.89543 10.001 3 10.8964 3 12.001C3 13.1055 3.89543 14.001 5 14.001C6.10457 14.001 7 13.1055 7 12.001ZM14 12.001C14 10.8964 13.1046 10.001 12 10.001C10.8954 10.001 10 10.8964 10 12.001C10 13.1055 10.8954 14.001 12 14.001C13.1046 14.001 14 13.1055 14 12.001ZM19 10.001C20.1046 10.001 21 10.8964 21 12.001C21 13.1055 20.1046 14.001 19 14.001C17.8954 14.001 17 13.1055 17 12.001C17 10.8964 17.8954 10.001 19 10.001Z" />
|
||||||
|
</svg>
|
||||||
|
</button>
|
||||||
|
)}
|
||||||
|
</Tooltip>
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
<Tooltip text={showContent ? "Hide " + headerText : "Show " + headerText}>
|
||||||
|
{tooltipProps => (
|
||||||
|
<button
|
||||||
|
{...tooltipProps}
|
||||||
|
className={cl("btn")}
|
||||||
|
onClick={() => {
|
||||||
|
setShowContent(v => !v);
|
||||||
|
onDropDownClick?.(showContent);
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<svg
|
||||||
|
width="24"
|
||||||
|
height="24"
|
||||||
|
viewBox="0 0 24 24"
|
||||||
|
transform={showContent ? "scale(1 -1)" : "scale(1 1)"}
|
||||||
|
>
|
||||||
|
<path fill="var(--text-normal)" d="M16.59 8.59003L12 13.17L7.41 8.59003L6 10L12 16L18 10L16.59 8.59003Z" />
|
||||||
|
</svg>
|
||||||
|
</button>
|
||||||
|
)}
|
||||||
|
</Tooltip>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{showContent && children}
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
205
src/components/Icons.tsx
Normal file
205
src/components/Icons.tsx
Normal file
@ -0,0 +1,205 @@
|
|||||||
|
/*
|
||||||
|
* Vencord, a modification for Discord's desktop app
|
||||||
|
* Copyright (c) 2023 Vendicated and contributors
|
||||||
|
*
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import "./iconStyles.css";
|
||||||
|
|
||||||
|
import { classes } from "@utils/misc";
|
||||||
|
import { i18n } from "@webpack/common";
|
||||||
|
import type { PropsWithChildren, SVGProps } from "react";
|
||||||
|
|
||||||
|
interface BaseIconProps extends IconProps {
|
||||||
|
viewBox: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface IconProps extends SVGProps<SVGSVGElement> {
|
||||||
|
className?: string;
|
||||||
|
height?: number;
|
||||||
|
width?: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
function Icon({ height = 24, width = 24, className, children, viewBox, ...svgProps }: PropsWithChildren<BaseIconProps>) {
|
||||||
|
return (
|
||||||
|
<svg
|
||||||
|
className={classes(className, "vc-icon")}
|
||||||
|
role="img"
|
||||||
|
width={width}
|
||||||
|
height={height}
|
||||||
|
viewBox={viewBox}
|
||||||
|
{...svgProps}
|
||||||
|
>
|
||||||
|
{children}
|
||||||
|
</svg>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Discord's link icon, as seen in the Message context menu "Copy Message Link" option
|
||||||
|
*/
|
||||||
|
export function LinkIcon({ height = 24, width = 24, className }: IconProps) {
|
||||||
|
return (
|
||||||
|
<Icon
|
||||||
|
height={height}
|
||||||
|
width={width}
|
||||||
|
className={classes(className, "vc-link-icon")}
|
||||||
|
viewBox="0 0 24 24"
|
||||||
|
>
|
||||||
|
<g fill="none" fill-rule="evenodd">
|
||||||
|
<path fill="currentColor" d="M10.59 13.41c.41.39.41 1.03 0 1.42-.39.39-1.03.39-1.42 0a5.003 5.003 0 0 1 0-7.07l3.54-3.54a5.003 5.003 0 0 1 7.07 0 5.003 5.003 0 0 1 0 7.07l-1.49 1.49c.01-.82-.12-1.64-.4-2.42l.47-.48a2.982 2.982 0 0 0 0-4.24 2.982 2.982 0 0 0-4.24 0l-3.53 3.53a2.982 2.982 0 0 0 0 4.24zm2.82-4.24c.39-.39 1.03-.39 1.42 0a5.003 5.003 0 0 1 0 7.07l-3.54 3.54a5.003 5.003 0 0 1-7.07 0 5.003 5.003 0 0 1 0-7.07l1.49-1.49c-.01.82.12 1.64.4 2.43l-.47.47a2.982 2.982 0 0 0 0 4.24 2.982 2.982 0 0 0 4.24 0l3.53-3.53a2.982 2.982 0 0 0 0-4.24.973.973 0 0 1 0-1.42z" />
|
||||||
|
<rect width={width} height={height} />
|
||||||
|
</g>
|
||||||
|
</Icon>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Discord's copy icon, as seen in the user popout right of the username when clicking
|
||||||
|
* your own username in the bottom left user panel
|
||||||
|
*/
|
||||||
|
export function CopyIcon(props: IconProps) {
|
||||||
|
return (
|
||||||
|
<Icon
|
||||||
|
{...props}
|
||||||
|
className={classes(props.className, "vc-copy-icon")}
|
||||||
|
viewBox="0 0 24 24"
|
||||||
|
>
|
||||||
|
<g fill="currentColor">
|
||||||
|
<path d="M16 1H4c-1.1 0-2 .9-2 2v14h2V3h12V1z" />
|
||||||
|
<path d="M15 5H8c-1.1 0-1.99.9-1.99 2L6 21c0 1.1.89 2 1.99 2H19c1.1 0 2-.9 2-2V11l-6-6zM8 21V7h6v5h5v9H8z" />
|
||||||
|
</g>
|
||||||
|
</Icon>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Discord's open external icon, as seen in the user profile connections
|
||||||
|
*/
|
||||||
|
export function OpenExternalIcon(props: IconProps) {
|
||||||
|
return (
|
||||||
|
<Icon
|
||||||
|
{...props}
|
||||||
|
className={classes(props.className, "vc-open-external-icon")}
|
||||||
|
viewBox="0 0 24 24"
|
||||||
|
>
|
||||||
|
<polygon
|
||||||
|
fill="currentColor"
|
||||||
|
fill-rule="nonzero"
|
||||||
|
points="13 20 11 20 11 8 5.5 13.5 4.08 12.08 12 4.16 19.92 12.08 18.5 13.5 13 8"
|
||||||
|
/>
|
||||||
|
</Icon>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
export function ImageIcon(props: IconProps) {
|
||||||
|
return (
|
||||||
|
<Icon
|
||||||
|
{...props}
|
||||||
|
className={classes(props.className, "vc-image-icon")}
|
||||||
|
viewBox="0 0 24 24"
|
||||||
|
>
|
||||||
|
<path fill="currentColor" d="M21,19V5c0,-1.1 -0.9,-2 -2,-2H5c-1.1,0 -2,0.9 -2,2v14c0,1.1 0.9,2 2,2h14c1.1,0 2,-0.9 2,-2zM8.5,13.5l2.5,3.01L14.5,12l4.5,6H5l3.5,-4.5z" />
|
||||||
|
</Icon>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
export function InfoIcon(props: IconProps) {
|
||||||
|
return (
|
||||||
|
<Icon
|
||||||
|
{...props}
|
||||||
|
className={classes(props.className, "vc-info-icon")}
|
||||||
|
viewBox="0 0 12 12"
|
||||||
|
>
|
||||||
|
<path fill="currentColor" d="M6 1C3.243 1 1 3.244 1 6c0 2.758 2.243 5 5 5s5-2.242 5-5c0-2.756-2.243-5-5-5zm0 2.376a.625.625 0 110 1.25.625.625 0 010-1.25zM7.5 8.5h-3v-1h1V6H5V5h1a.5.5 0 01.5.5v2h1v1z" />
|
||||||
|
</Icon>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
export function OwnerCrownIcon(props: IconProps) {
|
||||||
|
return (
|
||||||
|
<Icon
|
||||||
|
aria-label={i18n.Messages.GUILD_OWNER}
|
||||||
|
{...props}
|
||||||
|
className={classes(props.className, "vc-owner-crown-icon")}
|
||||||
|
role="img"
|
||||||
|
viewBox="0 0 16 16"
|
||||||
|
>
|
||||||
|
<path
|
||||||
|
fill="currentColor"
|
||||||
|
fill-rule="evenodd"
|
||||||
|
clip-rule="evenodd"
|
||||||
|
d="M13.6572 5.42868C13.8879 5.29002 14.1806 5.30402 14.3973 5.46468C14.6133 5.62602 14.7119 5.90068 14.6473 6.16202L13.3139 11.4954C13.2393 11.7927 12.9726 12.0007 12.6666 12.0007H3.33325C3.02725 12.0007 2.76058 11.792 2.68592 11.4954L1.35258 6.16202C1.28792 5.90068 1.38658 5.62602 1.60258 5.46468C1.81992 5.30468 2.11192 5.29068 2.34325 5.42868L5.13192 7.10202L7.44592 3.63068C7.46173 3.60697 7.48377 3.5913 7.50588 3.57559C7.5192 3.56612 7.53255 3.55663 7.54458 3.54535L6.90258 2.90268C6.77325 2.77335 6.77325 2.56068 6.90258 2.43135L7.76458 1.56935C7.89392 1.44002 8.10658 1.44002 8.23592 1.56935L9.09792 2.43135C9.22725 2.56068 9.22725 2.77335 9.09792 2.90268L8.45592 3.54535C8.46794 3.55686 8.48154 3.56651 8.49516 3.57618C8.51703 3.5917 8.53897 3.60727 8.55458 3.63068L10.8686 7.10202L13.6572 5.42868ZM2.66667 12.6673H13.3333V14.0007H2.66667V12.6673Z"
|
||||||
|
/>
|
||||||
|
</Icon>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Discord's screenshare icon, as seen in the connection panel
|
||||||
|
*/
|
||||||
|
export function ScreenshareIcon(props: IconProps) {
|
||||||
|
return (
|
||||||
|
<Icon
|
||||||
|
{...props}
|
||||||
|
className={classes(props.className, "vc-screenshare-icon")}
|
||||||
|
viewBox="0 0 24 24"
|
||||||
|
>
|
||||||
|
<path
|
||||||
|
fill="currentColor"
|
||||||
|
fill-rule="evenodd"
|
||||||
|
clip-rule="evenodd"
|
||||||
|
d="M2 4.5C2 3.397 2.897 2.5 4 2.5H20C21.103 2.5 22 3.397 22 4.5V15.5C22 16.604 21.103 17.5 20 17.5H13V19.5H17V21.5H7V19.5H11V17.5H4C2.897 17.5 2 16.604 2 15.5V4.5ZM13.2 14.3375V11.6C9.864 11.6 7.668 12.6625 6 15C6.672 11.6625 8.532 8.3375 13.2 7.6625V5L18 9.6625L13.2 14.3375Z"
|
||||||
|
/>
|
||||||
|
</Icon>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
export function ImageVisible(props: IconProps) {
|
||||||
|
return (
|
||||||
|
<Icon
|
||||||
|
{...props}
|
||||||
|
className={classes(props.className, "vc-image-visible")}
|
||||||
|
viewBox="0 0 24 24"
|
||||||
|
>
|
||||||
|
<path fill="currentColor" d="M5 21q-.825 0-1.413-.587Q3 19.825 3 19V5q0-.825.587-1.413Q4.175 3 5 3h14q.825 0 1.413.587Q21 4.175 21 5v14q0 .825-.587 1.413Q19.825 21 19 21Zm0-2h14V5H5v14Zm1-2h12l-3.75-5-3 4L9 13Zm-1 2V5v14Z" />
|
||||||
|
</Icon>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
export function ImageInvisible(props: IconProps) {
|
||||||
|
return (
|
||||||
|
<Icon
|
||||||
|
{...props}
|
||||||
|
className={classes(props.className, "vc-image-invisible")}
|
||||||
|
viewBox="0 0 24 24"
|
||||||
|
>
|
||||||
|
<path fill="currentColor" d="m21 18.15-2-2V5H7.85l-2-2H19q.825 0 1.413.587Q21 4.175 21 5Zm-1.2 4.45L18.2 21H5q-.825 0-1.413-.587Q3 19.825 3 19V5.8L1.4 4.2l1.4-1.4 18.4 18.4ZM6 17l3-4 2.25 3 .825-1.1L5 7.825V19h11.175l-2-2Zm7.425-6.425ZM10.6 13.4Z" />
|
||||||
|
</Icon>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
export function Microphone(props: IconProps) {
|
||||||
|
return (
|
||||||
|
<Icon
|
||||||
|
{...props}
|
||||||
|
className={classes(props.className, "vc-microphone")}
|
||||||
|
viewBox="0 0 24 24"
|
||||||
|
>
|
||||||
|
<path fill-rule="evenodd" clip-rule="evenodd" d="M14.99 11C14.99 12.66 13.66 14 12 14C10.34 14 9 12.66 9 11V5C9 3.34 10.34 2 12 2C13.66 2 15 3.34 15 5L14.99 11ZM12 16.1C14.76 16.1 17.3 14 17.3 11H19C19 14.42 16.28 17.24 13 17.72V21H11V17.72C7.72 17.23 5 14.41 5 11H6.7C6.7 14 9.24 16.1 12 16.1ZM12 4C11.2 4 11 4.66667 11 5V11C11 11.3333 11.2 12 12 12C12.8 12 13 11.3333 13 11V5C13 4.66667 12.8 4 12 4Z" fill="currentColor" />
|
||||||
|
<path fill-rule="evenodd" clip-rule="evenodd" d="M14.99 11C14.99 12.66 13.66 14 12 14C10.34 14 9 12.66 9 11V5C9 3.34 10.34 2 12 2C13.66 2 15 3.34 15 5L14.99 11ZM12 16.1C14.76 16.1 17.3 14 17.3 11H19C19 14.42 16.28 17.24 13 17.72V22H11V17.72C7.72 17.23 5 14.41 5 11H6.7C6.7 14 9.24 16.1 12 16.1Z" fill="currentColor" />
|
||||||
|
</Icon >
|
||||||
|
);
|
||||||
|
}
|
@ -1,51 +0,0 @@
|
|||||||
/*
|
|
||||||
* Vencord, a modification for Discord's desktop app
|
|
||||||
* Copyright (c) 2022 Vendicated and contributors
|
|
||||||
*
|
|
||||||
* This program is free software: you can redistribute it and/or modify
|
|
||||||
* it under the terms of the GNU General Public License as published by
|
|
||||||
* the Free Software Foundation, either version 3 of the License, or
|
|
||||||
* (at your option) any later version.
|
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
||||||
* GNU General Public License for more details.
|
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU General Public License
|
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
||||||
*/
|
|
||||||
|
|
||||||
import { debounce } from "@utils/debounce";
|
|
||||||
import IpcEvents from "@utils/IpcEvents";
|
|
||||||
import { Queue } from "@utils/Queue";
|
|
||||||
import { find } from "@webpack";
|
|
||||||
|
|
||||||
import monacoHtml from "~fileContent/monacoWin.html";
|
|
||||||
|
|
||||||
const queue = new Queue();
|
|
||||||
const setCss = debounce((css: string) => {
|
|
||||||
queue.push(() => VencordNative.ipc.invoke(IpcEvents.SET_QUICK_CSS, css));
|
|
||||||
});
|
|
||||||
|
|
||||||
export async function launchMonacoEditor() {
|
|
||||||
const features = `popup,width=${Math.min(window.innerWidth, 1000)},height=${Math.min(window.innerHeight, 1000)}`;
|
|
||||||
const win = open("about:blank", "VencordQuickCss", features);
|
|
||||||
if (!win) {
|
|
||||||
alert("Failed to open QuickCSS popup. Make sure to allow popups!");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
win.setCss = setCss;
|
|
||||||
win.getCurrentCss = () => VencordNative.ipc.invoke(IpcEvents.GET_QUICK_CSS);
|
|
||||||
win.getTheme = () =>
|
|
||||||
find(m =>
|
|
||||||
m.ProtoClass?.typeName.endsWith("PreloadedUserSettings")
|
|
||||||
)?.getCurrentValue()?.appearance?.theme === 2
|
|
||||||
? "vs-light"
|
|
||||||
: "vs-dark";
|
|
||||||
|
|
||||||
win.document.write(monacoHtml);
|
|
||||||
|
|
||||||
window.__VENCORD_MONACO_WIN__ = new WeakRef(win);
|
|
||||||
}
|
|
@ -17,12 +17,15 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import { generateId } from "@api/Commands";
|
import { generateId } from "@api/Commands";
|
||||||
import { useSettings } from "@api/settings";
|
import { useSettings } from "@api/Settings";
|
||||||
|
import { disableStyle, enableStyle } from "@api/Styles";
|
||||||
import ErrorBoundary from "@components/ErrorBoundary";
|
import ErrorBoundary from "@components/ErrorBoundary";
|
||||||
import { Flex } from "@components/Flex";
|
import { Flex } from "@components/Flex";
|
||||||
import { LazyComponent } from "@utils/misc";
|
import { proxyLazy } from "@utils/lazy";
|
||||||
|
import { Margins } from "@utils/margins";
|
||||||
|
import { classes } from "@utils/misc";
|
||||||
import { ModalCloseButton, ModalContent, ModalFooter, ModalHeader, ModalProps, ModalRoot, ModalSize } from "@utils/modal";
|
import { ModalCloseButton, ModalContent, ModalFooter, ModalHeader, ModalProps, ModalRoot, ModalSize } from "@utils/modal";
|
||||||
import { proxyLazy } from "@utils/proxyLazy";
|
import { LazyComponent } from "@utils/react";
|
||||||
import { OptionType, Plugin } from "@utils/types";
|
import { OptionType, Plugin } from "@utils/types";
|
||||||
import { findByCode, findByPropsLazy } from "@webpack";
|
import { findByCode, findByPropsLazy } from "@webpack";
|
||||||
import { Button, FluxDispatcher, Forms, React, Text, Tooltip, UserStore, UserUtils } from "@webpack/common";
|
import { Button, FluxDispatcher, Forms, React, Text, Tooltip, UserStore, UserUtils } from "@webpack/common";
|
||||||
@ -38,6 +41,7 @@ import {
|
|||||||
SettingSliderComponent,
|
SettingSliderComponent,
|
||||||
SettingTextComponent
|
SettingTextComponent
|
||||||
} from "./components";
|
} from "./components";
|
||||||
|
import hideBotTagStyle from "./userPopoutHideBotTag.css?managed";
|
||||||
|
|
||||||
const UserSummaryItem = LazyComponent(() => findByCode("defaultRenderUser", "showDefaultAvatarsForNullUsers"));
|
const UserSummaryItem = LazyComponent(() => findByCode("defaultRenderUser", "showDefaultAvatarsForNullUsers"));
|
||||||
const AvatarStyles = findByPropsLazy("moreUsers", "emptyUser", "avatarContainer", "clickableAvatar");
|
const AvatarStyles = findByPropsLazy("moreUsers", "emptyUser", "avatarContainer", "clickableAvatar");
|
||||||
@ -48,11 +52,12 @@ interface PluginModalProps extends ModalProps {
|
|||||||
onRestartNeeded(): void;
|
onRestartNeeded(): void;
|
||||||
}
|
}
|
||||||
|
|
||||||
/** To stop discord making unwanted requests... */
|
function makeDummyUser(user: { username: string; id?: string; avatar?: string; }) {
|
||||||
function makeDummyUser(user: { name: string, id: BigInt; }) {
|
|
||||||
const newUser = new UserRecord({
|
const newUser = new UserRecord({
|
||||||
username: user.name,
|
username: user.username,
|
||||||
id: generateId(),
|
id: user.id ?? generateId(),
|
||||||
|
avatar: user.avatar,
|
||||||
|
/** To stop discord making unwanted requests... */
|
||||||
bot: true,
|
bot: true,
|
||||||
});
|
});
|
||||||
FluxDispatcher.dispatch({
|
FluxDispatcher.dispatch({
|
||||||
@ -87,14 +92,27 @@ export default function PluginModal({ plugin, onRestartNeeded, onClose, transiti
|
|||||||
const hasSettings = Boolean(pluginSettings && plugin.options);
|
const hasSettings = Boolean(pluginSettings && plugin.options);
|
||||||
|
|
||||||
React.useEffect(() => {
|
React.useEffect(() => {
|
||||||
|
enableStyle(hideBotTagStyle);
|
||||||
|
|
||||||
|
let originalUser: User;
|
||||||
(async () => {
|
(async () => {
|
||||||
for (const user of plugin.authors.slice(0, 6)) {
|
for (const user of plugin.authors.slice(0, 6)) {
|
||||||
const author = user.id
|
const author = user.id
|
||||||
? await UserUtils.fetchUser(`${user.id}`).catch(() => makeDummyUser(user))
|
? await UserUtils.fetchUser(`${user.id}`)
|
||||||
: makeDummyUser(user);
|
// only show name & pfp and no actions so users cannot harass plugin devs for support (send dms, add as friend, etc)
|
||||||
|
.then(u => (originalUser = u, makeDummyUser(u)))
|
||||||
|
.catch(() => makeDummyUser({ username: user.name }))
|
||||||
|
: makeDummyUser({ username: user.name });
|
||||||
|
|
||||||
setAuthors(a => [...a, author]);
|
setAuthors(a => [...a, author]);
|
||||||
}
|
}
|
||||||
})();
|
})();
|
||||||
|
|
||||||
|
return () => {
|
||||||
|
disableStyle(hideBotTagStyle);
|
||||||
|
if (originalUser)
|
||||||
|
FluxDispatcher.dispatch({ type: "USER_UPDATE", user: originalUser });
|
||||||
|
};
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
async function saveAndClose() {
|
async function saveAndClose() {
|
||||||
@ -127,6 +145,8 @@ export default function PluginModal({ plugin, onRestartNeeded, onClose, transiti
|
|||||||
return <Forms.FormText>There are no settings for this plugin.</Forms.FormText>;
|
return <Forms.FormText>There are no settings for this plugin.</Forms.FormText>;
|
||||||
} else {
|
} else {
|
||||||
const options = Object.entries(plugin.options).map(([key, setting]) => {
|
const options = Object.entries(plugin.options).map(([key, setting]) => {
|
||||||
|
if (setting.hidden) return null;
|
||||||
|
|
||||||
function onChange(newValue: any) {
|
function onChange(newValue: any) {
|
||||||
setTempSettings(s => ({ ...s, [key]: newValue }));
|
setTempSettings(s => ({ ...s, [key]: newValue }));
|
||||||
}
|
}
|
||||||
@ -149,7 +169,7 @@ export default function PluginModal({ plugin, onRestartNeeded, onClose, transiti
|
|||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
|
||||||
return <Flex flexDirection="column" style={{ gap: 12 }}>{options}</Flex>;
|
return <Flex flexDirection="column" style={{ gap: 12, marginBottom: 16 }}>{options}</Flex>;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -174,12 +194,12 @@ export default function PluginModal({ plugin, onRestartNeeded, onClose, transiti
|
|||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<ModalRoot transitionState={transitionState} size={ModalSize.MEDIUM}>
|
<ModalRoot transitionState={transitionState} size={ModalSize.MEDIUM} className="vc-text-selectable">
|
||||||
<ModalHeader separator={false}>
|
<ModalHeader separator={false}>
|
||||||
<Text variant="heading-lg/semibold" style={{ flexGrow: 1 }}>{plugin.name}</Text>
|
<Text variant="heading-lg/semibold" style={{ flexGrow: 1 }}>{plugin.name}</Text>
|
||||||
<ModalCloseButton onClick={onClose} />
|
<ModalCloseButton onClick={onClose} />
|
||||||
</ModalHeader>
|
</ModalHeader>
|
||||||
<ModalContent style={{ marginBottom: 8, marginTop: 8 }}>
|
<ModalContent>
|
||||||
<Forms.FormSection>
|
<Forms.FormSection>
|
||||||
<Forms.FormTitle tag="h3">About {plugin.name}</Forms.FormTitle>
|
<Forms.FormTitle tag="h3">About {plugin.name}</Forms.FormTitle>
|
||||||
<Forms.FormText>{plugin.description}</Forms.FormText>
|
<Forms.FormText>{plugin.description}</Forms.FormText>
|
||||||
@ -198,7 +218,7 @@ export default function PluginModal({ plugin, onRestartNeeded, onClose, transiti
|
|||||||
</div>
|
</div>
|
||||||
</Forms.FormSection>
|
</Forms.FormSection>
|
||||||
{!!plugin.settingsAboutComponent && (
|
{!!plugin.settingsAboutComponent && (
|
||||||
<div style={{ marginBottom: 8 }}>
|
<div className={classes(Margins.bottom8, "vc-text-selectable")}>
|
||||||
<Forms.FormSection>
|
<Forms.FormSection>
|
||||||
<ErrorBoundary message="An error occurred while rendering this plugin's custom InfoComponent">
|
<ErrorBoundary message="An error occurred while rendering this plugin's custom InfoComponent">
|
||||||
<plugin.settingsAboutComponent tempSettings={tempSettings} />
|
<plugin.settingsAboutComponent tempSettings={tempSettings} />
|
||||||
@ -206,7 +226,7 @@ export default function PluginModal({ plugin, onRestartNeeded, onClose, transiti
|
|||||||
</Forms.FormSection>
|
</Forms.FormSection>
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
<Forms.FormSection>
|
<Forms.FormSection className={Margins.bottom16}>
|
||||||
<Forms.FormTitle tag="h3">Settings</Forms.FormTitle>
|
<Forms.FormTitle tag="h3">Settings</Forms.FormTitle>
|
||||||
{renderSettings()}
|
{renderSettings()}
|
||||||
</Forms.FormSection>
|
</Forms.FormSection>
|
||||||
|
@ -16,8 +16,9 @@
|
|||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
import { wordsFromCamel, wordsToTitle } from "@utils/text";
|
||||||
import { PluginOptionBoolean } from "@utils/types";
|
import { PluginOptionBoolean } from "@utils/types";
|
||||||
import { Forms, React, Select } from "@webpack/common";
|
import { Forms, React, Switch } from "@webpack/common";
|
||||||
|
|
||||||
import { ISettingElementProps } from ".";
|
import { ISettingElementProps } from ".";
|
||||||
|
|
||||||
@ -31,11 +32,6 @@ export function SettingBooleanComponent({ option, pluginSettings, definedSetting
|
|||||||
onError(error !== null);
|
onError(error !== null);
|
||||||
}, [error]);
|
}, [error]);
|
||||||
|
|
||||||
const options = [
|
|
||||||
{ label: "Enabled", value: true, default: def === true },
|
|
||||||
{ label: "Disabled", value: false, default: typeof def === "undefined" || def === false },
|
|
||||||
];
|
|
||||||
|
|
||||||
function handleChange(newValue: boolean): void {
|
function handleChange(newValue: boolean): void {
|
||||||
const isValid = option.isValid?.call(definedSettings, newValue) ?? true;
|
const isValid = option.isValid?.call(definedSettings, newValue) ?? true;
|
||||||
if (typeof isValid === "string") setError(isValid);
|
if (typeof isValid === "string") setError(isValid);
|
||||||
@ -49,18 +45,17 @@ export function SettingBooleanComponent({ option, pluginSettings, definedSetting
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<Forms.FormSection>
|
<Forms.FormSection>
|
||||||
<Forms.FormTitle>{option.description}</Forms.FormTitle>
|
<Switch
|
||||||
<Select
|
value={state}
|
||||||
isDisabled={option.disabled?.call(definedSettings) ?? false}
|
onChange={handleChange}
|
||||||
options={options}
|
note={option.description}
|
||||||
placeholder={option.placeholder ?? "Select an option"}
|
disabled={option.disabled?.call(definedSettings) ?? false}
|
||||||
maxVisibleItems={5}
|
|
||||||
closeOnSelect={true}
|
|
||||||
select={handleChange}
|
|
||||||
isSelected={v => v === state}
|
|
||||||
serialize={v => String(v)}
|
|
||||||
{...option.componentProps}
|
{...option.componentProps}
|
||||||
/>
|
hideBorder
|
||||||
|
style={{ marginBottom: "0.5em" }}
|
||||||
|
>
|
||||||
|
{wordsToTitle(wordsFromCamel(id))}
|
||||||
|
</Switch>
|
||||||
{error && <Forms.FormText style={{ color: "var(--text-danger)" }}>{error}</Forms.FormText>}
|
{error && <Forms.FormText style={{ color: "var(--text-danger)" }}>{error}</Forms.FormText>}
|
||||||
</Forms.FormSection>
|
</Forms.FormSection>
|
||||||
);
|
);
|
||||||
|
@ -38,9 +38,12 @@ export function SettingNumericComponent({ option, pluginSettings, definedSetting
|
|||||||
|
|
||||||
function handleChange(newValue) {
|
function handleChange(newValue) {
|
||||||
const isValid = option.isValid?.call(definedSettings, newValue) ?? true;
|
const isValid = option.isValid?.call(definedSettings, newValue) ?? true;
|
||||||
|
|
||||||
|
setError(null);
|
||||||
if (typeof isValid === "string") setError(isValid);
|
if (typeof isValid === "string") setError(isValid);
|
||||||
else if (!isValid) setError("Invalid input provided.");
|
else if (!isValid) setError("Invalid input provided.");
|
||||||
else if (option.type === OptionType.NUMBER && BigInt(newValue) >= MAX_SAFE_NUMBER) {
|
|
||||||
|
if (option.type === OptionType.NUMBER && BigInt(newValue) >= MAX_SAFE_NUMBER) {
|
||||||
setState(`${Number.MAX_SAFE_INTEGER}`);
|
setState(`${Number.MAX_SAFE_INTEGER}`);
|
||||||
onChange(serialize(newValue));
|
onChange(serialize(newValue));
|
||||||
} else {
|
} else {
|
||||||
|
@ -36,6 +36,7 @@ export function SettingSelectComponent({ option, pluginSettings, definedSettings
|
|||||||
if (typeof isValid === "string") setError(isValid);
|
if (typeof isValid === "string") setError(isValid);
|
||||||
else if (!isValid) setError("Invalid input provided.");
|
else if (!isValid) setError("Invalid input provided.");
|
||||||
else {
|
else {
|
||||||
|
setError(null);
|
||||||
setState(newValue);
|
setState(newValue);
|
||||||
onChange(newValue);
|
onChange(newValue);
|
||||||
}
|
}
|
||||||
|
@ -33,10 +33,10 @@ export function SettingTextComponent({ option, pluginSettings, definedSettings,
|
|||||||
const isValid = option.isValid?.call(definedSettings, newValue) ?? true;
|
const isValid = option.isValid?.call(definedSettings, newValue) ?? true;
|
||||||
if (typeof isValid === "string") setError(isValid);
|
if (typeof isValid === "string") setError(isValid);
|
||||||
else if (!isValid) setError("Invalid input provided.");
|
else if (!isValid) setError("Invalid input provided.");
|
||||||
else {
|
else setError(null);
|
||||||
setState(newValue);
|
|
||||||
onChange(newValue);
|
setState(newValue);
|
||||||
}
|
onChange(newValue);
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
@ -20,21 +20,20 @@ import "./styles.css";
|
|||||||
|
|
||||||
import * as DataStore from "@api/DataStore";
|
import * as DataStore from "@api/DataStore";
|
||||||
import { showNotice } from "@api/Notices";
|
import { showNotice } from "@api/Notices";
|
||||||
import { useSettings } from "@api/settings";
|
import { Settings, useSettings } from "@api/Settings";
|
||||||
import { classNameFactory } from "@api/Styles";
|
import { classNameFactory } from "@api/Styles";
|
||||||
import ErrorBoundary from "@components/ErrorBoundary";
|
|
||||||
import { Flex } from "@components/Flex";
|
|
||||||
import { handleComponentFailed } from "@components/handleComponentFailed";
|
|
||||||
import { Badge } from "@components/PluginSettings/components";
|
|
||||||
import PluginModal from "@components/PluginSettings/PluginModal";
|
import PluginModal from "@components/PluginSettings/PluginModal";
|
||||||
import { Switch } from "@components/Switch";
|
import { AddonCard } from "@components/VencordSettings/AddonCard";
|
||||||
|
import { SettingsTab } from "@components/VencordSettings/shared";
|
||||||
import { ChangeList } from "@utils/ChangeList";
|
import { ChangeList } from "@utils/ChangeList";
|
||||||
import Logger from "@utils/Logger";
|
import { Logger } from "@utils/Logger";
|
||||||
import { classes, LazyComponent, useAwaiter } from "@utils/misc";
|
import { Margins } from "@utils/margins";
|
||||||
|
import { classes } from "@utils/misc";
|
||||||
import { openModalLazy } from "@utils/modal";
|
import { openModalLazy } from "@utils/modal";
|
||||||
|
import { LazyComponent, useAwaiter } from "@utils/react";
|
||||||
import { Plugin } from "@utils/types";
|
import { Plugin } from "@utils/types";
|
||||||
import { findByCode, findByPropsLazy } from "@webpack";
|
import { findByCode, findByPropsLazy } from "@webpack";
|
||||||
import { Alerts, Button, Card, Forms, Margins, Parser, React, Select, Text, TextInput, Toasts, Tooltip } from "@webpack/common";
|
import { Alerts, Button, Card, Forms, Parser, React, Select, Text, TextInput, Toasts, Tooltip } from "@webpack/common";
|
||||||
|
|
||||||
import Plugins from "~plugins";
|
import Plugins from "~plugins";
|
||||||
|
|
||||||
@ -45,6 +44,7 @@ const cl = classNameFactory("vc-plugins-");
|
|||||||
const logger = new Logger("PluginSettings", "#a6d189");
|
const logger = new Logger("PluginSettings", "#a6d189");
|
||||||
|
|
||||||
const InputStyles = findByPropsLazy("inputDefault", "inputWrapper");
|
const InputStyles = findByPropsLazy("inputDefault", "inputWrapper");
|
||||||
|
const ButtonClasses = findByPropsLazy("button", "disabled", "enabled");
|
||||||
|
|
||||||
const CogWheel = LazyComponent(() => findByCode("18.564C15.797 19.099 14.932 19.498 14 19.738V22H10V19.738C9.069"));
|
const CogWheel = LazyComponent(() => findByCode("18.564C15.797 19.099 14.932 19.498 14 19.738V22H10V19.738C9.069"));
|
||||||
const InfoIcon = LazyComponent(() => findByCode("4.4408921e-16 C4.4771525,-1.77635684e-15 4.4408921e-16"));
|
const InfoIcon = LazyComponent(() => findByCode("4.4408921e-16 C4.4771525,-1.77635684e-15 4.4408921e-16"));
|
||||||
@ -92,7 +92,7 @@ interface PluginCardProps extends React.HTMLProps<HTMLDivElement> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function PluginCard({ plugin, disabled, onRestartNeeded, onMouseEnter, onMouseLeave, isNew }: PluginCardProps) {
|
function PluginCard({ plugin, disabled, onRestartNeeded, onMouseEnter, onMouseLeave, isNew }: PluginCardProps) {
|
||||||
const settings = useSettings([`plugins.${plugin.name}`]).plugins[plugin.name];
|
const settings = Settings.plugins[plugin.name];
|
||||||
|
|
||||||
const isEnabled = () => settings.enabled ?? false;
|
const isEnabled = () => settings.enabled ?? false;
|
||||||
|
|
||||||
@ -123,7 +123,7 @@ function PluginCard({ plugin, disabled, onRestartNeeded, onMouseEnter, onMouseLe
|
|||||||
}
|
}
|
||||||
|
|
||||||
// if the plugin has patches, dont use stopPlugin/startPlugin. Wait for restart to apply changes.
|
// if the plugin has patches, dont use stopPlugin/startPlugin. Wait for restart to apply changes.
|
||||||
if (plugin.patches) {
|
if (plugin.patches?.length) {
|
||||||
settings.enabled = !wasEnabled;
|
settings.enabled = !wasEnabled;
|
||||||
onRestartNeeded(plugin.name);
|
onRestartNeeded(plugin.name);
|
||||||
return;
|
return;
|
||||||
@ -136,11 +136,13 @@ function PluginCard({ plugin, disabled, onRestartNeeded, onMouseEnter, onMouseLe
|
|||||||
}
|
}
|
||||||
|
|
||||||
const result = wasEnabled ? stopPlugin(plugin) : startPlugin(plugin);
|
const result = wasEnabled ? stopPlugin(plugin) : startPlugin(plugin);
|
||||||
const action = wasEnabled ? "stop" : "start";
|
|
||||||
|
|
||||||
if (!result) {
|
if (!result) {
|
||||||
logger.error(`Failed to ${action} plugin ${plugin.name}`);
|
settings.enabled = false;
|
||||||
showErrorToast(`Failed to ${action} plugin: ${plugin.name}`);
|
|
||||||
|
const msg = `Error while ${wasEnabled ? "stopping" : "starting"} plugin ${plugin.name}`;
|
||||||
|
logger.error(msg);
|
||||||
|
showErrorToast(msg);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -148,34 +150,34 @@ function PluginCard({ plugin, disabled, onRestartNeeded, onMouseEnter, onMouseLe
|
|||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Flex className={cl("card", { "card-disabled": disabled })} flexDirection="column" onMouseEnter={onMouseEnter} onMouseLeave={onMouseLeave}>
|
<AddonCard
|
||||||
<div className={cl("card-header")}>
|
name={plugin.name}
|
||||||
<Text variant="text-md/bold" className={cl("name")}>
|
description={plugin.description}
|
||||||
{plugin.name}{isNew && <Badge text="NEW" color="#ED4245" />}
|
isNew={isNew}
|
||||||
</Text>
|
enabled={isEnabled()}
|
||||||
<button role="switch" onClick={() => openModal()} className={classes("button-12Fmur", cl("info-button"))}>
|
setEnabled={toggleEnabled}
|
||||||
|
disabled={disabled}
|
||||||
|
onMouseEnter={onMouseEnter}
|
||||||
|
onMouseLeave={onMouseLeave}
|
||||||
|
infoButton={
|
||||||
|
<button role="switch" onClick={() => openModal()} className={classes(ButtonClasses.button, cl("info-button"))}>
|
||||||
{plugin.options
|
{plugin.options
|
||||||
? <CogWheel />
|
? <CogWheel />
|
||||||
: <InfoIcon width="24" height="24" />}
|
: <InfoIcon width="24" height="24" />}
|
||||||
</button>
|
</button>
|
||||||
<Switch
|
}
|
||||||
checked={isEnabled()}
|
/>
|
||||||
onChange={toggleEnabled}
|
|
||||||
disabled={disabled}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
<Text className={cl("note")} variant="text-sm/normal">{plugin.description}</Text>
|
|
||||||
</Flex >
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
enum SearchStatus {
|
const enum SearchStatus {
|
||||||
ALL,
|
ALL,
|
||||||
ENABLED,
|
ENABLED,
|
||||||
DISABLED
|
DISABLED,
|
||||||
|
NEW
|
||||||
}
|
}
|
||||||
|
|
||||||
export default ErrorBoundary.wrap(function PluginSettings() {
|
export default function PluginSettings() {
|
||||||
const settings = useSettings();
|
const settings = useSettings();
|
||||||
const changes = React.useMemo(() => new ChangeList<string>(), []);
|
const changes = React.useMemo(() => new ChangeList<string>(), []);
|
||||||
|
|
||||||
@ -225,10 +227,14 @@ export default ErrorBoundary.wrap(function PluginSettings() {
|
|||||||
const enabled = settings.plugins[plugin.name]?.enabled;
|
const enabled = settings.plugins[plugin.name]?.enabled;
|
||||||
if (enabled && searchValue.status === SearchStatus.DISABLED) return false;
|
if (enabled && searchValue.status === SearchStatus.DISABLED) return false;
|
||||||
if (!enabled && searchValue.status === SearchStatus.ENABLED) return false;
|
if (!enabled && searchValue.status === SearchStatus.ENABLED) return false;
|
||||||
|
if (searchValue.status === SearchStatus.NEW && !newPlugins?.includes(plugin.name)) return false;
|
||||||
if (!searchValue.value.length) return true;
|
if (!searchValue.value.length) return true;
|
||||||
|
|
||||||
|
const v = searchValue.value.toLowerCase();
|
||||||
return (
|
return (
|
||||||
plugin.name.toLowerCase().includes(searchValue.value.toLowerCase()) ||
|
plugin.name.toLowerCase().includes(v) ||
|
||||||
plugin.description.toLowerCase().includes(searchValue.value.toLowerCase())
|
plugin.description.toLowerCase().includes(v) ||
|
||||||
|
plugin.tags?.some(t => t.toLowerCase().includes(v))
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -256,6 +262,9 @@ export default ErrorBoundary.wrap(function PluginSettings() {
|
|||||||
requiredPlugins = [];
|
requiredPlugins = [];
|
||||||
|
|
||||||
for (const p of sortedPlugins) {
|
for (const p of sortedPlugins) {
|
||||||
|
if (!p.options && p.name.endsWith("API") && searchValue.value !== "API")
|
||||||
|
continue;
|
||||||
|
|
||||||
if (!pluginFilter(p)) continue;
|
if (!pluginFilter(p)) continue;
|
||||||
|
|
||||||
const isRequired = p.required || depMap[p.name]?.some(d => settings.plugins[d].enabled);
|
const isRequired = p.required || depMap[p.name]?.some(d => settings.plugins[d].enabled);
|
||||||
@ -296,22 +305,23 @@ export default ErrorBoundary.wrap(function PluginSettings() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Forms.FormSection className={Margins.marginTop16}>
|
<SettingsTab title="Plugins">
|
||||||
<ReloadRequiredCard required={changes.hasChanges} />
|
<ReloadRequiredCard required={changes.hasChanges} />
|
||||||
|
|
||||||
<Forms.FormTitle tag="h5" className={classes(Margins.marginTop20, Margins.marginBottom8)}>
|
<Forms.FormTitle tag="h5" className={classes(Margins.top20, Margins.bottom8)}>
|
||||||
Filters
|
Filters
|
||||||
</Forms.FormTitle>
|
</Forms.FormTitle>
|
||||||
|
|
||||||
<div className={cl("filter-controls")}>
|
<div className={cl("filter-controls")}>
|
||||||
<TextInput autoFocus value={searchValue.value} placeholder="Search for a plugin..." onChange={onSearch} className={Margins.marginBottom20} />
|
<TextInput autoFocus value={searchValue.value} placeholder="Search for a plugin..." onChange={onSearch} className={Margins.bottom20} />
|
||||||
<div className={InputStyles.inputWrapper}>
|
<div className={InputStyles.inputWrapper}>
|
||||||
<Select
|
<Select
|
||||||
className={InputStyles.inputDefault}
|
className={InputStyles.inputDefault}
|
||||||
options={[
|
options={[
|
||||||
{ label: "Show All", value: SearchStatus.ALL, default: true },
|
{ label: "Show All", value: SearchStatus.ALL, default: true },
|
||||||
{ label: "Show Enabled", value: SearchStatus.ENABLED },
|
{ label: "Show Enabled", value: SearchStatus.ENABLED },
|
||||||
{ label: "Show Disabled", value: SearchStatus.DISABLED }
|
{ label: "Show Disabled", value: SearchStatus.DISABLED },
|
||||||
|
{ label: "Show New", value: SearchStatus.NEW }
|
||||||
]}
|
]}
|
||||||
serialize={String}
|
serialize={String}
|
||||||
select={onStatusChange}
|
select={onStatusChange}
|
||||||
@ -321,26 +331,23 @@ export default ErrorBoundary.wrap(function PluginSettings() {
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<Forms.FormTitle className={Margins.marginTop20}>Plugins</Forms.FormTitle>
|
<Forms.FormTitle className={Margins.top20}>Plugins</Forms.FormTitle>
|
||||||
|
|
||||||
<div className={cl("grid")}>
|
<div className={cl("grid")}>
|
||||||
{plugins}
|
{plugins}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<Forms.FormDivider className={Margins.marginTop20} />
|
<Forms.FormDivider className={Margins.top20} />
|
||||||
|
|
||||||
<Forms.FormTitle tag="h5" className={classes(Margins.marginTop20, Margins.marginBottom8)}>
|
<Forms.FormTitle tag="h5" className={classes(Margins.top20, Margins.bottom8)}>
|
||||||
Required Plugins
|
Required Plugins
|
||||||
</Forms.FormTitle>
|
</Forms.FormTitle>
|
||||||
<div className={cl("grid")}>
|
<div className={cl("grid")}>
|
||||||
{requiredPlugins}
|
{requiredPlugins}
|
||||||
</div>
|
</div>
|
||||||
</Forms.FormSection >
|
</SettingsTab >
|
||||||
);
|
);
|
||||||
}, {
|
}
|
||||||
message: "Failed to render the Plugin Settings. If this persists, try using the installer to reinstall!",
|
|
||||||
onError: handleComponentFailed,
|
|
||||||
});
|
|
||||||
|
|
||||||
function makeDependencyList(deps: string[]) {
|
function makeDependencyList(deps: string[]) {
|
||||||
return (
|
return (
|
||||||
|
@ -23,38 +23,6 @@
|
|||||||
grid-template-columns: repeat(auto-fill, minmax(250px, 1fr));
|
grid-template-columns: repeat(auto-fill, minmax(250px, 1fr));
|
||||||
}
|
}
|
||||||
|
|
||||||
.vc-plugins-card {
|
|
||||||
background-color: var(--background-secondary-alt);
|
|
||||||
color: var(--interactive-active);
|
|
||||||
border-radius: 8px;
|
|
||||||
display: block;
|
|
||||||
height: 100%;
|
|
||||||
padding: 12px;
|
|
||||||
width: 100%;
|
|
||||||
transition: 0.1s ease-out;
|
|
||||||
transition-property: box-shadow, transform, background, opacity;
|
|
||||||
}
|
|
||||||
|
|
||||||
.vc-plugins-card-disabled {
|
|
||||||
opacity: 0.6;
|
|
||||||
}
|
|
||||||
|
|
||||||
.vc-plugins-card:hover {
|
|
||||||
background-color: var(--background-tertiary);
|
|
||||||
transform: translateY(-1px);
|
|
||||||
box-shadow: var(--elevation-high);
|
|
||||||
}
|
|
||||||
|
|
||||||
.vc-plugins-card-header {
|
|
||||||
margin-top: auto;
|
|
||||||
display: flex;
|
|
||||||
width: 100%;
|
|
||||||
justify-content: flex-end;
|
|
||||||
height: 1.5rem;
|
|
||||||
align-items: center;
|
|
||||||
gap: 8px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.vc-plugins-info-button {
|
.vc-plugins-info-button {
|
||||||
height: 24px;
|
height: 24px;
|
||||||
width: 24px;
|
width: 24px;
|
||||||
@ -86,27 +54,6 @@
|
|||||||
text-align: center;
|
text-align: center;
|
||||||
}
|
}
|
||||||
|
|
||||||
.vc-plugins-note {
|
|
||||||
height: 36px;
|
|
||||||
overflow: hidden;
|
|
||||||
text-overflow: ellipsis;
|
|
||||||
display: -webkit-box;
|
|
||||||
-webkit-line-clamp: 2;
|
|
||||||
line-clamp: 2;
|
|
||||||
-webkit-box-orient: vertical;
|
|
||||||
/* stylelint-disable-next-line property-no-unknown */
|
|
||||||
box-orient: vertical;
|
|
||||||
}
|
|
||||||
|
|
||||||
.vc-plugins-name {
|
|
||||||
display: flex;
|
|
||||||
width: 100%;
|
|
||||||
align-items: center;
|
|
||||||
flex-grow: 1;
|
|
||||||
gap: 8px;
|
|
||||||
cursor: "default";
|
|
||||||
}
|
|
||||||
|
|
||||||
.vc-plugins-dep-name {
|
.vc-plugins-dep-name {
|
||||||
margin: 0 auto;
|
margin: 0 auto;
|
||||||
}
|
}
|
||||||
|
3
src/components/PluginSettings/userPopoutHideBotTag.css
Normal file
3
src/components/PluginSettings/userPopoutHideBotTag.css
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
[class|="userPopoutOuter"] [class*="botTag"] {
|
||||||
|
display: none;
|
||||||
|
}
|
@ -18,6 +18,7 @@
|
|||||||
|
|
||||||
import "./Switch.css";
|
import "./Switch.css";
|
||||||
|
|
||||||
|
import { classes } from "@utils/misc";
|
||||||
import { findByPropsLazy } from "@webpack";
|
import { findByPropsLazy } from "@webpack";
|
||||||
|
|
||||||
interface SwitchProps {
|
interface SwitchProps {
|
||||||
@ -33,7 +34,7 @@ const SwitchClasses = findByPropsLazy("slider", "input", "container");
|
|||||||
export function Switch({ checked, onChange, disabled }: SwitchProps) {
|
export function Switch({ checked, onChange, disabled }: SwitchProps) {
|
||||||
return (
|
return (
|
||||||
<div>
|
<div>
|
||||||
<div className={`${SwitchClasses.container} default-colors`} style={{
|
<div className={classes(SwitchClasses.container, "default-colors", checked ? SwitchClasses.checked : void 0)} style={{
|
||||||
backgroundColor: checked ? SWITCH_ON : SWITCH_OFF,
|
backgroundColor: checked ? SWITCH_ON : SWITCH_OFF,
|
||||||
opacity: disabled ? 0.3 : 1
|
opacity: disabled ? 0.3 : 1
|
||||||
}}>
|
}}>
|
||||||
|
77
src/components/VencordSettings/AddonCard.tsx
Normal file
77
src/components/VencordSettings/AddonCard.tsx
Normal file
@ -0,0 +1,77 @@
|
|||||||
|
/*
|
||||||
|
* Vencord, a modification for Discord's desktop app
|
||||||
|
* Copyright (c) 2023 Vendicated and contributors
|
||||||
|
*
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import "./addonCard.css";
|
||||||
|
|
||||||
|
import { classNameFactory } from "@api/Styles";
|
||||||
|
import { Badge } from "@components/Badge";
|
||||||
|
import { Switch } from "@components/Switch";
|
||||||
|
import { Text } from "@webpack/common";
|
||||||
|
import type { MouseEventHandler, ReactNode } from "react";
|
||||||
|
|
||||||
|
const cl = classNameFactory("vc-addon-");
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
name: ReactNode;
|
||||||
|
description: ReactNode;
|
||||||
|
enabled: boolean;
|
||||||
|
setEnabled: (enabled: boolean) => void;
|
||||||
|
disabled?: boolean;
|
||||||
|
isNew?: boolean;
|
||||||
|
onMouseEnter?: MouseEventHandler<HTMLDivElement>;
|
||||||
|
onMouseLeave?: MouseEventHandler<HTMLDivElement>;
|
||||||
|
|
||||||
|
infoButton?: ReactNode;
|
||||||
|
footer?: ReactNode;
|
||||||
|
author?: ReactNode;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function AddonCard({ disabled, isNew, name, infoButton, footer, author, enabled, setEnabled, description, onMouseEnter, onMouseLeave }: Props) {
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
className={cl("card", { "card-disabled": disabled })}
|
||||||
|
onMouseEnter={onMouseEnter}
|
||||||
|
onMouseLeave={onMouseLeave}
|
||||||
|
>
|
||||||
|
<div className={cl("header")}>
|
||||||
|
<div className={cl("name-author")}>
|
||||||
|
<Text variant="text-md/bold" className={cl("name")}>
|
||||||
|
{name}{isNew && <Badge text="NEW" color="#ED4245" />}
|
||||||
|
</Text>
|
||||||
|
{!!author && (
|
||||||
|
<Text variant="text-md/normal" className={cl("author")}>
|
||||||
|
{author}
|
||||||
|
</Text>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{infoButton}
|
||||||
|
|
||||||
|
<Switch
|
||||||
|
checked={enabled}
|
||||||
|
onChange={setEnabled}
|
||||||
|
disabled={disabled}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<Text className={cl("note")} variant="text-sm/normal">{description}</Text>
|
||||||
|
|
||||||
|
{footer}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
@ -16,30 +16,33 @@
|
|||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import ErrorBoundary from "@components/ErrorBoundary";
|
|
||||||
import { Flex } from "@components/Flex";
|
import { Flex } from "@components/Flex";
|
||||||
|
import { Margins } from "@utils/margins";
|
||||||
import { classes } from "@utils/misc";
|
import { classes } from "@utils/misc";
|
||||||
import { downloadSettingsBackup, uploadSettingsBackup } from "@utils/settingsSync";
|
import { downloadSettingsBackup, uploadSettingsBackup } from "@utils/settingsSync";
|
||||||
import { Button, Card, Forms, Margins, Text } from "@webpack/common";
|
import { Button, Card, Text } from "@webpack/common";
|
||||||
|
|
||||||
|
import { SettingsTab, wrapTab } from "./shared";
|
||||||
|
|
||||||
function BackupRestoreTab() {
|
function BackupRestoreTab() {
|
||||||
return (
|
return (
|
||||||
<Forms.FormSection title="Settings Sync" className={Margins.marginTop16}>
|
<SettingsTab title="Backup & Restore">
|
||||||
<Card className={classes("vc-settings-card", "vc-backup-restore-card")}>
|
<Card className={classes("vc-settings-card", "vc-backup-restore-card")}>
|
||||||
<Flex flexDirection="column">
|
<Flex flexDirection="column">
|
||||||
<strong>Warning</strong>
|
<strong>Warning</strong>
|
||||||
<span>Importing a settings file will overwrite your current settings.</span>
|
<span>Importing a settings file will overwrite your current settings.</span>
|
||||||
</Flex>
|
</Flex>
|
||||||
</Card>
|
</Card>
|
||||||
<Text variant="text-md/normal" className={Margins.marginBottom8}>
|
<Text variant="text-md/normal" className={Margins.bottom8}>
|
||||||
You can import and export your Vencord settings as a JSON file.
|
You can import and export your Vencord settings as a JSON file.
|
||||||
This allows you to easily transfer your settings to another device,
|
This allows you to easily transfer your settings to another device,
|
||||||
or recover your settings after reinstalling Vencord or Discord.
|
or recover your settings after reinstalling Vencord or Discord.
|
||||||
</Text>
|
</Text>
|
||||||
<Text variant="text-md/normal" className={Margins.marginBottom8}>
|
<Text variant="text-md/normal" className={Margins.bottom8}>
|
||||||
Settings Export contains:
|
Settings Export contains:
|
||||||
<ul>
|
<ul>
|
||||||
<li>— Custom QuickCSS</li>
|
<li>— Custom QuickCSS</li>
|
||||||
|
<li>— Theme Links</li>
|
||||||
<li>— Plugin Settings</li>
|
<li>— Plugin Settings</li>
|
||||||
</ul>
|
</ul>
|
||||||
</Text>
|
</Text>
|
||||||
@ -57,8 +60,8 @@ function BackupRestoreTab() {
|
|||||||
Export Settings
|
Export Settings
|
||||||
</Button>
|
</Button>
|
||||||
</Flex>
|
</Flex>
|
||||||
</Forms.FormSection>
|
</SettingsTab>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
export default ErrorBoundary.wrap(BackupRestoreTab);
|
export default wrapTab(BackupRestoreTab, "Backup & Restore");
|
165
src/components/VencordSettings/CloudTab.tsx
Normal file
165
src/components/VencordSettings/CloudTab.tsx
Normal file
@ -0,0 +1,165 @@
|
|||||||
|
/*
|
||||||
|
* Vencord, a modification for Discord's desktop app
|
||||||
|
* Copyright (c) 2023 Vendicated and contributors
|
||||||
|
*
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { showNotification } from "@api/Notifications";
|
||||||
|
import { Settings, useSettings } from "@api/Settings";
|
||||||
|
import { CheckedTextInput } from "@components/CheckedTextInput";
|
||||||
|
import { Link } from "@components/Link";
|
||||||
|
import { authorizeCloud, cloudLogger, deauthorizeCloud, getCloudAuth, getCloudUrl } from "@utils/cloud";
|
||||||
|
import { Margins } from "@utils/margins";
|
||||||
|
import { deleteCloudSettings, getCloudSettings, putCloudSettings } from "@utils/settingsSync";
|
||||||
|
import { Alerts, Button, Forms, Switch, Tooltip } from "@webpack/common";
|
||||||
|
|
||||||
|
import { SettingsTab, wrapTab } from "./shared";
|
||||||
|
|
||||||
|
function validateUrl(url: string) {
|
||||||
|
try {
|
||||||
|
new URL(url);
|
||||||
|
return true;
|
||||||
|
} catch {
|
||||||
|
return "Invalid URL";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function eraseAllData() {
|
||||||
|
const res = await fetch(new URL("/v1/", getCloudUrl()), {
|
||||||
|
method: "DELETE",
|
||||||
|
headers: new Headers({
|
||||||
|
Authorization: await getCloudAuth()
|
||||||
|
})
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!res.ok) {
|
||||||
|
cloudLogger.error(`Failed to erase data, API returned ${res.status}`);
|
||||||
|
showNotification({
|
||||||
|
title: "Cloud Integrations",
|
||||||
|
body: `Could not erase all data (API returned ${res.status}), please contact support.`,
|
||||||
|
color: "var(--red-360)"
|
||||||
|
});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
Settings.cloud.authenticated = false;
|
||||||
|
await deauthorizeCloud();
|
||||||
|
|
||||||
|
showNotification({
|
||||||
|
title: "Cloud Integrations",
|
||||||
|
body: "Successfully erased all data.",
|
||||||
|
color: "var(--green-360)"
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function SettingsSyncSection() {
|
||||||
|
const { cloud } = useSettings(["cloud.authenticated", "cloud.settingsSync"]);
|
||||||
|
const sectionEnabled = cloud.authenticated && cloud.settingsSync;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Forms.FormSection title="Settings Sync" className={Margins.top16}>
|
||||||
|
<Forms.FormText variant="text-md/normal" className={Margins.bottom20}>
|
||||||
|
Synchronize your settings to the cloud. This allows easy synchronization across multiple devices with
|
||||||
|
minimal effort.
|
||||||
|
</Forms.FormText>
|
||||||
|
<Switch
|
||||||
|
key="cloud-sync"
|
||||||
|
disabled={!cloud.authenticated}
|
||||||
|
value={cloud.settingsSync}
|
||||||
|
onChange={v => { cloud.settingsSync = v; }}
|
||||||
|
>
|
||||||
|
Settings Sync
|
||||||
|
</Switch>
|
||||||
|
<div className="vc-cloud-settings-sync-grid">
|
||||||
|
<Button
|
||||||
|
size={Button.Sizes.SMALL}
|
||||||
|
disabled={!sectionEnabled}
|
||||||
|
onClick={() => putCloudSettings(true)}
|
||||||
|
>Sync to Cloud</Button>
|
||||||
|
<Tooltip text="This will overwrite your local settings with the ones on the cloud. Use wisely!">
|
||||||
|
{({ onMouseLeave, onMouseEnter }) => (
|
||||||
|
<Button
|
||||||
|
onMouseLeave={onMouseLeave}
|
||||||
|
onMouseEnter={onMouseEnter}
|
||||||
|
size={Button.Sizes.SMALL}
|
||||||
|
color={Button.Colors.RED}
|
||||||
|
disabled={!sectionEnabled}
|
||||||
|
onClick={() => getCloudSettings(true, true)}
|
||||||
|
>Sync from Cloud</Button>
|
||||||
|
)}
|
||||||
|
</Tooltip>
|
||||||
|
<Button
|
||||||
|
size={Button.Sizes.SMALL}
|
||||||
|
color={Button.Colors.RED}
|
||||||
|
disabled={!sectionEnabled}
|
||||||
|
onClick={() => deleteCloudSettings()}
|
||||||
|
>Delete Cloud Settings</Button>
|
||||||
|
</div>
|
||||||
|
</Forms.FormSection>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
function CloudTab() {
|
||||||
|
const settings = useSettings(["cloud.authenticated", "cloud.url"]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<SettingsTab title="Vencord Cloud">
|
||||||
|
<Forms.FormSection title="Cloud Settings" className={Margins.top16}>
|
||||||
|
<Forms.FormText variant="text-md/normal" className={Margins.bottom20}>
|
||||||
|
Vencord comes with a cloud integration that adds goodies like settings sync across devices.
|
||||||
|
It <Link href="https://vencord.dev/cloud/privacy">respects your privacy</Link>, and
|
||||||
|
the <Link href="https://github.com/Vencord/Backend">source code</Link> is AGPL 3.0 licensed so you
|
||||||
|
can host it yourself.
|
||||||
|
</Forms.FormText>
|
||||||
|
<Switch
|
||||||
|
key="backend"
|
||||||
|
value={settings.cloud.authenticated}
|
||||||
|
onChange={v => { v && authorizeCloud(); if (!v) settings.cloud.authenticated = v; }}
|
||||||
|
note="This will request authorization if you have not yet set up cloud integrations."
|
||||||
|
>
|
||||||
|
Enable Cloud Integrations
|
||||||
|
</Switch>
|
||||||
|
<Forms.FormTitle tag="h5">Backend URL</Forms.FormTitle>
|
||||||
|
<Forms.FormText className={Margins.bottom8}>
|
||||||
|
Which backend to use when using cloud integrations.
|
||||||
|
</Forms.FormText>
|
||||||
|
<CheckedTextInput
|
||||||
|
key="backendUrl"
|
||||||
|
value={settings.cloud.url}
|
||||||
|
onChange={v => { settings.cloud.url = v; settings.cloud.authenticated = false; deauthorizeCloud(); }}
|
||||||
|
validate={validateUrl}
|
||||||
|
/>
|
||||||
|
<Button
|
||||||
|
className={Margins.top8}
|
||||||
|
size={Button.Sizes.MEDIUM}
|
||||||
|
color={Button.Colors.RED}
|
||||||
|
disabled={!settings.cloud.authenticated}
|
||||||
|
onClick={() => Alerts.show({
|
||||||
|
title: "Are you sure?",
|
||||||
|
body: "Once your data is erased, we cannot recover it. There's no going back!",
|
||||||
|
onConfirm: eraseAllData,
|
||||||
|
confirmText: "Erase it!",
|
||||||
|
confirmColor: "vc-cloud-erase-data-danger-btn",
|
||||||
|
cancelText: "Nevermind"
|
||||||
|
})}
|
||||||
|
>Erase All Data</Button>
|
||||||
|
<Forms.FormDivider className={Margins.top16} />
|
||||||
|
</Forms.FormSection >
|
||||||
|
<SettingsSyncSection />
|
||||||
|
</SettingsTab>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
export default wrapTab(CloudTab, "Cloud");
|
@ -16,14 +16,16 @@
|
|||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
import { CheckedTextInput } from "@components/CheckedTextInput";
|
||||||
import { debounce } from "@utils/debounce";
|
import { debounce } from "@utils/debounce";
|
||||||
import { makeCodeblock } from "@utils/misc";
|
import { Margins } from "@utils/margins";
|
||||||
import { canonicalizeMatch, canonicalizeReplace, ReplaceFn } from "@utils/patches";
|
import { canonicalizeMatch, canonicalizeReplace } from "@utils/patches";
|
||||||
|
import { makeCodeblock } from "@utils/text";
|
||||||
|
import { ReplaceFn } from "@utils/types";
|
||||||
import { search } from "@webpack";
|
import { search } from "@webpack";
|
||||||
import { Button, Clipboard, Forms, Margins, Parser, React, Switch, Text, TextInput } from "@webpack/common";
|
import { Button, Clipboard, Forms, Parser, React, Switch, TextInput } from "@webpack/common";
|
||||||
|
|
||||||
import { CheckedTextInput } from "./CheckedTextInput";
|
import { SettingsTab, wrapTab } from "./shared";
|
||||||
import ErrorBoundary from "./ErrorBoundary";
|
|
||||||
|
|
||||||
// Do not include diff in non dev builds (side effects import)
|
// Do not include diff in non dev builds (side effects import)
|
||||||
if (IS_DEV) {
|
if (IS_DEV) {
|
||||||
@ -128,7 +130,7 @@ function ReplacementComponent({ module, match, replacement, setReplacementError
|
|||||||
)}
|
)}
|
||||||
|
|
||||||
{!!diff?.length && (
|
{!!diff?.length && (
|
||||||
<Button className={Margins.marginTop20} onClick={() => {
|
<Button className={Margins.top20} onClick={() => {
|
||||||
try {
|
try {
|
||||||
Function(patchedCode.replace(/^function\(/, "function patchedModule("));
|
Function(patchedCode.replace(/^function\(/, "function patchedModule("));
|
||||||
setCompileResult([true, "Compiled successfully"]);
|
setCompileResult([true, "Compiled successfully"]);
|
||||||
@ -184,9 +186,10 @@ function ReplacementInput({ replacement, setReplacement, replacementError }) {
|
|||||||
error={error ?? replacementError}
|
error={error ?? replacementError}
|
||||||
/>
|
/>
|
||||||
{!isFunc && (
|
{!isFunc && (
|
||||||
<>
|
<div className="vc-text-selectable">
|
||||||
<Forms.FormTitle>Cheat Sheet</Forms.FormTitle>
|
<Forms.FormTitle>Cheat Sheet</Forms.FormTitle>
|
||||||
{Object.entries({
|
{Object.entries({
|
||||||
|
"\\i": "Special regex escape sequence that matches identifiers (varnames, classnames, etc.)",
|
||||||
"$$": "Insert a $",
|
"$$": "Insert a $",
|
||||||
"$&": "Insert the entire match",
|
"$&": "Insert the entire match",
|
||||||
"$`\u200b": "Insert the substring before the match",
|
"$`\u200b": "Insert the substring before the match",
|
||||||
@ -198,11 +201,11 @@ function ReplacementInput({ replacement, setReplacement, replacementError }) {
|
|||||||
{Parser.parse("`" + placeholder + "`")}: {desc}
|
{Parser.parse("`" + placeholder + "`")}: {desc}
|
||||||
</Forms.FormText>
|
</Forms.FormText>
|
||||||
))}
|
))}
|
||||||
</>
|
</div>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
<Switch
|
<Switch
|
||||||
className={Margins.marginTop8}
|
className={Margins.top8}
|
||||||
value={isFunc}
|
value={isFunc}
|
||||||
onChange={setIsFunc}
|
onChange={setIsFunc}
|
||||||
note="'replacement' will be evaled if this is toggled"
|
note="'replacement' will be evaled if this is toggled"
|
||||||
@ -255,8 +258,7 @@ function PatchHelper() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Forms.FormSection>
|
<SettingsTab title="Patch Helper">
|
||||||
<Text variant="heading-md/normal" tag="h2" className={Margins.marginBottom8}>Patch Helper</Text>
|
|
||||||
<Forms.FormTitle>find</Forms.FormTitle>
|
<Forms.FormTitle>find</Forms.FormTitle>
|
||||||
<TextInput
|
<TextInput
|
||||||
type="text"
|
type="text"
|
||||||
@ -296,13 +298,13 @@ function PatchHelper() {
|
|||||||
|
|
||||||
{!!(find && match && replacement) && (
|
{!!(find && match && replacement) && (
|
||||||
<>
|
<>
|
||||||
<Forms.FormTitle className={Margins.marginTop20}>Code</Forms.FormTitle>
|
<Forms.FormTitle className={Margins.top20}>Code</Forms.FormTitle>
|
||||||
<div style={{ userSelect: "text" }}>{Parser.parse(makeCodeblock(code, "ts"))}</div>
|
<div style={{ userSelect: "text" }}>{Parser.parse(makeCodeblock(code, "ts"))}</div>
|
||||||
<Button onClick={() => Clipboard.copy(code)}>Copy to Clipboard</Button>
|
<Button onClick={() => Clipboard.copy(code)}>Copy to Clipboard</Button>
|
||||||
</>
|
</>
|
||||||
)}
|
)}
|
||||||
</Forms.FormSection>
|
</SettingsTab>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
export default IS_DEV ? ErrorBoundary.wrap(PatchHelper) : null;
|
export default IS_DEV ? wrapTab(PatchHelper, "PatchHelper") : null;
|
@ -16,7 +16,8 @@
|
|||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import ErrorBoundary from "@components/ErrorBoundary";
|
|
||||||
import PluginSettings from "@components/PluginSettings";
|
import PluginSettings from "@components/PluginSettings";
|
||||||
|
|
||||||
export default ErrorBoundary.wrap(PluginSettings);
|
import { wrapTab } from "./shared";
|
||||||
|
|
||||||
|
export default wrapTab(PluginSettings, "Plugins");
|
||||||
|
@ -16,15 +16,36 @@
|
|||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import { useSettings } from "@api/settings";
|
import { useSettings } from "@api/Settings";
|
||||||
import ErrorBoundary from "@components/ErrorBoundary";
|
import { classNameFactory } from "@api/Styles";
|
||||||
|
import { Flex } from "@components/Flex";
|
||||||
import { Link } from "@components/Link";
|
import { Link } from "@components/Link";
|
||||||
import { useAwaiter } from "@utils/misc";
|
import { Margins } from "@utils/margins";
|
||||||
import { findLazy } from "@webpack";
|
import { classes } from "@utils/misc";
|
||||||
import { Card, Forms, Margins, React, TextArea } from "@webpack/common";
|
import { showItemInFolder } from "@utils/native";
|
||||||
|
import { useAwaiter } from "@utils/react";
|
||||||
|
import { findByCodeLazy, findByPropsLazy, findLazy } from "@webpack";
|
||||||
|
import { Button, Card, FluxDispatcher, Forms, React, showToast, TabBar, TextArea, useEffect, useRef, useState } from "@webpack/common";
|
||||||
|
import { UserThemeHeader } from "main/themes";
|
||||||
|
import type { ComponentType, Ref, SyntheticEvent } from "react";
|
||||||
|
|
||||||
|
import { AddonCard } from "./AddonCard";
|
||||||
|
import { SettingsTab, wrapTab } from "./shared";
|
||||||
|
|
||||||
|
type FileInput = ComponentType<{
|
||||||
|
ref: Ref<HTMLInputElement>;
|
||||||
|
onChange: (e: SyntheticEvent<HTMLInputElement>) => void;
|
||||||
|
multiple?: boolean;
|
||||||
|
filters?: { name?: string; extensions: string[]; }[];
|
||||||
|
}>;
|
||||||
|
|
||||||
|
const InviteActions = findByPropsLazy("resolveInvite");
|
||||||
|
const TrashIcon = findByCodeLazy("M5 6.99902V18.999C5 20.101 5.897 20.999");
|
||||||
|
const FileInput: FileInput = findByCodeLazy("activateUploadDialogue=");
|
||||||
const TextAreaProps = findLazy(m => typeof m.textarea === "string");
|
const TextAreaProps = findLazy(m => typeof m.textarea === "string");
|
||||||
|
|
||||||
|
const cl = classNameFactory("vc-settings-theme-");
|
||||||
|
|
||||||
function Validator({ link }: { link: string; }) {
|
function Validator({ link }: { link: string; }) {
|
||||||
const [res, err, pending] = useAwaiter(() => fetch(link).then(res => {
|
const [res, err, pending] = useAwaiter(() => fetch(link).then(res => {
|
||||||
if (res.status > 300) throw `${res.status} ${res.statusText}`;
|
if (res.status > 300) throw `${res.status} ${res.statusText}`;
|
||||||
@ -51,7 +72,7 @@ function Validators({ themeLinks }: { themeLinks: string[]; }) {
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<Forms.FormTitle className={Margins.marginTop20} tag="h5">Validator</Forms.FormTitle>
|
<Forms.FormTitle className={Margins.top20} tag="h5">Validator</Forms.FormTitle>
|
||||||
<Forms.FormText>This section will tell you whether your themes can successfully be loaded</Forms.FormText>
|
<Forms.FormText>This section will tell you whether your themes can successfully be loaded</Forms.FormText>
|
||||||
<div>
|
<div>
|
||||||
{themeLinks.map(link => (
|
{themeLinks.map(link => (
|
||||||
@ -73,10 +94,191 @@ function Validators({ themeLinks }: { themeLinks: string[]; }) {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
export default ErrorBoundary.wrap(function () {
|
interface ThemeCardProps {
|
||||||
const settings = useSettings();
|
theme: UserThemeHeader;
|
||||||
const [themeText, setThemeText] = React.useState(settings.themeLinks.join("\n"));
|
enabled: boolean;
|
||||||
|
onChange: (enabled: boolean) => void;
|
||||||
|
onDelete: () => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
function ThemeCard({ theme, enabled, onChange, onDelete }: ThemeCardProps) {
|
||||||
|
return (
|
||||||
|
<AddonCard
|
||||||
|
name={theme.name}
|
||||||
|
description={theme.description}
|
||||||
|
author={theme.author}
|
||||||
|
enabled={enabled}
|
||||||
|
setEnabled={onChange}
|
||||||
|
infoButton={
|
||||||
|
IS_WEB && (
|
||||||
|
<div style={{ cursor: "pointer", color: "var(--status-danger" }} onClick={onDelete}>
|
||||||
|
<TrashIcon />
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
footer={
|
||||||
|
<Flex flexDirection="row" style={{ gap: "0.2em" }}>
|
||||||
|
{!!theme.website && <Link href={theme.website}>Website</Link>}
|
||||||
|
{!!(theme.website && theme.invite) && " • "}
|
||||||
|
{!!theme.invite && (
|
||||||
|
<Link
|
||||||
|
href={`https://discord.gg/${theme.invite}`}
|
||||||
|
onClick={async e => {
|
||||||
|
e.preventDefault();
|
||||||
|
const { invite } = await InviteActions.resolveInvite(theme.invite, "Desktop Modal");
|
||||||
|
if (!invite) return showToast("Invalid or expired invite");
|
||||||
|
|
||||||
|
FluxDispatcher.dispatch({
|
||||||
|
type: "INVITE_MODAL_OPEN",
|
||||||
|
invite,
|
||||||
|
code: theme.invite,
|
||||||
|
context: "APP"
|
||||||
|
});
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
Discord Server
|
||||||
|
</Link>
|
||||||
|
)}
|
||||||
|
</Flex>
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
enum ThemeTab {
|
||||||
|
LOCAL,
|
||||||
|
ONLINE
|
||||||
|
}
|
||||||
|
|
||||||
|
function ThemesTab() {
|
||||||
|
const settings = useSettings(["themeLinks", "enabledThemes"]);
|
||||||
|
|
||||||
|
const fileInputRef = useRef<HTMLInputElement>(null);
|
||||||
|
const [currentTab, setCurrentTab] = useState(ThemeTab.LOCAL);
|
||||||
|
const [themeText, setThemeText] = useState(settings.themeLinks.join("\n"));
|
||||||
|
const [userThemes, setUserThemes] = useState<UserThemeHeader[] | null>(null);
|
||||||
|
const [themeDir, , themeDirPending] = useAwaiter(VencordNative.themes.getThemesDir);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
refreshLocalThemes();
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
async function refreshLocalThemes() {
|
||||||
|
const themes = await VencordNative.themes.getThemesList();
|
||||||
|
setUserThemes(themes);
|
||||||
|
}
|
||||||
|
|
||||||
|
// When a local theme is enabled/disabled, update the settings
|
||||||
|
function onLocalThemeChange(fileName: string, value: boolean) {
|
||||||
|
if (value) {
|
||||||
|
if (settings.enabledThemes.includes(fileName)) return;
|
||||||
|
settings.enabledThemes = [...settings.enabledThemes, fileName];
|
||||||
|
} else {
|
||||||
|
settings.enabledThemes = settings.enabledThemes.filter(f => f !== fileName);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function onFileUpload(e: SyntheticEvent<HTMLInputElement>) {
|
||||||
|
e.stopPropagation();
|
||||||
|
e.preventDefault();
|
||||||
|
if (!e.currentTarget?.files?.length) return;
|
||||||
|
const { files } = e.currentTarget;
|
||||||
|
|
||||||
|
const uploads = Array.from(files, file => {
|
||||||
|
const { name } = file;
|
||||||
|
if (!name.endsWith(".css")) return;
|
||||||
|
|
||||||
|
return new Promise<void>((resolve, reject) => {
|
||||||
|
const reader = new FileReader();
|
||||||
|
reader.onload = () => {
|
||||||
|
VencordNative.themes.uploadTheme(name, reader.result as string)
|
||||||
|
.then(resolve)
|
||||||
|
.catch(reject);
|
||||||
|
};
|
||||||
|
reader.readAsText(file);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
await Promise.all(uploads);
|
||||||
|
refreshLocalThemes();
|
||||||
|
}
|
||||||
|
|
||||||
|
function renderLocalThemes() {
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<Card className="vc-settings-card">
|
||||||
|
<Forms.FormTitle tag="h5">Find Themes:</Forms.FormTitle>
|
||||||
|
<div style={{ marginBottom: ".5em", display: "flex", flexDirection: "column" }}>
|
||||||
|
<Link style={{ marginRight: ".5em" }} href="https://betterdiscord.app/themes">
|
||||||
|
BetterDiscord Themes
|
||||||
|
</Link>
|
||||||
|
<Link href="https://github.com/search?q=discord+theme">GitHub</Link>
|
||||||
|
</div>
|
||||||
|
<Forms.FormText>If using the BD site, click on "Download" and place the downloaded .theme.css file into your themes folder.</Forms.FormText>
|
||||||
|
</Card>
|
||||||
|
|
||||||
|
<Forms.FormSection title="Local Themes">
|
||||||
|
<Card className="vc-settings-quick-actions-card">
|
||||||
|
<>
|
||||||
|
{IS_WEB ?
|
||||||
|
(
|
||||||
|
<Button
|
||||||
|
size={Button.Sizes.SMALL}
|
||||||
|
disabled={themeDirPending}
|
||||||
|
>
|
||||||
|
Upload Theme
|
||||||
|
<FileInput
|
||||||
|
ref={fileInputRef}
|
||||||
|
onChange={onFileUpload}
|
||||||
|
multiple={true}
|
||||||
|
filters={[{ extensions: ["css"] }]}
|
||||||
|
/>
|
||||||
|
</Button>
|
||||||
|
) : (
|
||||||
|
<Button
|
||||||
|
onClick={() => showItemInFolder(themeDir!)}
|
||||||
|
size={Button.Sizes.SMALL}
|
||||||
|
disabled={themeDirPending}
|
||||||
|
>
|
||||||
|
Open Themes Folder
|
||||||
|
</Button>
|
||||||
|
)}
|
||||||
|
<Button
|
||||||
|
onClick={refreshLocalThemes}
|
||||||
|
size={Button.Sizes.SMALL}
|
||||||
|
>
|
||||||
|
Load missing Themes
|
||||||
|
</Button>
|
||||||
|
<Button
|
||||||
|
onClick={() => VencordNative.quickCss.openEditor()}
|
||||||
|
size={Button.Sizes.SMALL}
|
||||||
|
>
|
||||||
|
Edit QuickCSS
|
||||||
|
</Button>
|
||||||
|
</>
|
||||||
|
</Card>
|
||||||
|
|
||||||
|
<div className={cl("grid")}>
|
||||||
|
{userThemes?.map(theme => (
|
||||||
|
<ThemeCard
|
||||||
|
key={theme.fileName}
|
||||||
|
enabled={settings.enabledThemes.includes(theme.fileName)}
|
||||||
|
onChange={enabled => onLocalThemeChange(theme.fileName, enabled)}
|
||||||
|
onDelete={async () => {
|
||||||
|
onLocalThemeChange(theme.fileName, false);
|
||||||
|
await VencordNative.themes.deleteTheme(theme.fileName);
|
||||||
|
refreshLocalThemes();
|
||||||
|
}}
|
||||||
|
theme={theme}
|
||||||
|
/>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
</Forms.FormSection>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// When the user leaves the online theme textbox, update the settings
|
||||||
function onBlur() {
|
function onBlur() {
|
||||||
settings.themeLinks = [...new Set(
|
settings.themeLinks = [...new Set(
|
||||||
themeText
|
themeText
|
||||||
@ -87,46 +289,58 @@ export default ErrorBoundary.wrap(function () {
|
|||||||
)];
|
)];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function renderOnlineThemes() {
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<Card className="vc-settings-card vc-text-selectable">
|
||||||
|
<Forms.FormTitle tag="h5">Paste links to css files here</Forms.FormTitle>
|
||||||
|
<Forms.FormText>One link per line</Forms.FormText>
|
||||||
|
<Forms.FormText>Make sure to use direct links to files (raw or github.io)!</Forms.FormText>
|
||||||
|
</Card>
|
||||||
|
|
||||||
|
<Forms.FormSection title="Online Themes" tag="h5">
|
||||||
|
<TextArea
|
||||||
|
value={themeText}
|
||||||
|
onChange={setThemeText}
|
||||||
|
className={classes(TextAreaProps.textarea, "vc-settings-theme-links")}
|
||||||
|
placeholder="Theme Links"
|
||||||
|
spellCheck={false}
|
||||||
|
onBlur={onBlur}
|
||||||
|
rows={10}
|
||||||
|
/>
|
||||||
|
<Validators themeLinks={settings.themeLinks} />
|
||||||
|
</Forms.FormSection>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<SettingsTab title="Themes">
|
||||||
<Card className="vc-settings-card">
|
<TabBar
|
||||||
<Forms.FormTitle tag="h5">Paste links to .css / .theme.css files here</Forms.FormTitle>
|
type="top"
|
||||||
<Forms.FormText>One link per line</Forms.FormText>
|
look="brand"
|
||||||
<Forms.FormText>Make sure to use the raw links or github.io links!</Forms.FormText>
|
className="vc-settings-tab-bar"
|
||||||
<Forms.FormDivider className={Margins.marginTop8 + " " + Margins.marginBottom8} />
|
selectedItem={currentTab}
|
||||||
<Forms.FormTitle tag="h5">Find Themes:</Forms.FormTitle>
|
onItemSelect={setCurrentTab}
|
||||||
<div style={{ marginBottom: ".5em" }}>
|
>
|
||||||
<Link style={{ marginRight: ".5em" }} href="https://betterdiscord.app/themes">
|
<TabBar.Item
|
||||||
BetterDiscord Themes
|
className="vc-settings-tab-bar-item"
|
||||||
</Link>
|
id={ThemeTab.LOCAL}
|
||||||
<Link href="https://github.com/search?q=discord+theme">GitHub</Link>
|
>
|
||||||
</div>
|
Local Themes
|
||||||
<Forms.FormText>If using the BD site, click on "Source" somewhere below the Download button</Forms.FormText>
|
</TabBar.Item>
|
||||||
<Forms.FormText>In the GitHub repository of your theme, find X.theme.css / X.css, click on it, then click the "Raw" button</Forms.FormText>
|
<TabBar.Item
|
||||||
<Forms.FormText>
|
className="vc-settings-tab-bar-item"
|
||||||
If the theme has configuration that requires you to edit the file:
|
id={ThemeTab.ONLINE}
|
||||||
<ul>
|
>
|
||||||
<li>• Make a <Link href="https://github.com/signup">GitHub</Link> account</li>
|
Online Themes
|
||||||
<li>• Click the fork button on the top right</li>
|
</TabBar.Item>
|
||||||
<li>• Edit the file</li>
|
</TabBar>
|
||||||
<li>• Use the link to your own repository instead</li>
|
|
||||||
</ul>
|
{currentTab === ThemeTab.LOCAL && renderLocalThemes()}
|
||||||
</Forms.FormText>
|
{currentTab === ThemeTab.ONLINE && renderOnlineThemes()}
|
||||||
</Card>
|
</SettingsTab>
|
||||||
<Forms.FormTitle tag="h5">Themes</Forms.FormTitle>
|
|
||||||
<TextArea
|
|
||||||
style={{
|
|
||||||
padding: ".5em",
|
|
||||||
border: "1px solid var(--background-modifier-accent)"
|
|
||||||
}}
|
|
||||||
value={themeText}
|
|
||||||
onChange={e => setThemeText(e.currentTarget.value)}
|
|
||||||
className={TextAreaProps.textarea}
|
|
||||||
placeholder="Theme Links"
|
|
||||||
spellCheck={false}
|
|
||||||
onBlur={onBlur}
|
|
||||||
/>
|
|
||||||
<Validators themeLinks={settings.themeLinks} />
|
|
||||||
</>
|
|
||||||
);
|
);
|
||||||
});
|
}
|
||||||
|
|
||||||
|
export default wrapTab(ThemesTab, "Themes");
|
||||||
|
@ -16,18 +16,21 @@
|
|||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import { useSettings } from "@api/settings";
|
import { useSettings } from "@api/Settings";
|
||||||
import ErrorBoundary from "@components/ErrorBoundary";
|
|
||||||
import { ErrorCard } from "@components/ErrorCard";
|
import { ErrorCard } from "@components/ErrorCard";
|
||||||
import { Flex } from "@components/Flex";
|
import { Flex } from "@components/Flex";
|
||||||
import { handleComponentFailed } from "@components/handleComponentFailed";
|
|
||||||
import { Link } from "@components/Link";
|
import { Link } from "@components/Link";
|
||||||
import { classes, useAwaiter } from "@utils/misc";
|
import { Margins } from "@utils/margins";
|
||||||
import { changes, checkForUpdates, getRepo, isNewer, rebuild, update, updateError, UpdateLogger } from "@utils/updater";
|
import { classes } from "@utils/misc";
|
||||||
import { Alerts, Button, Card, Forms, Margins, Parser, React, Switch, Toasts } from "@webpack/common";
|
import { relaunch } from "@utils/native";
|
||||||
|
import { useAwaiter } from "@utils/react";
|
||||||
|
import { changes, checkForUpdates, getRepo, isNewer, update, updateError, UpdateLogger } from "@utils/updater";
|
||||||
|
import { Alerts, Button, Card, Forms, Parser, React, Switch, Toasts } from "@webpack/common";
|
||||||
|
|
||||||
import gitHash from "~git-hash";
|
import gitHash from "~git-hash";
|
||||||
|
|
||||||
|
import { SettingsTab, wrapTab } from "./shared";
|
||||||
|
|
||||||
function withDispatcher(dispatcher: React.Dispatch<React.SetStateAction<boolean>>, action: () => any) {
|
function withDispatcher(dispatcher: React.Dispatch<React.SetStateAction<boolean>>, action: () => any) {
|
||||||
return async () => {
|
return async () => {
|
||||||
dispatcher(true);
|
dispatcher(true);
|
||||||
@ -109,21 +112,20 @@ function Updatable(props: CommonProps) {
|
|||||||
</ErrorCard>
|
</ErrorCard>
|
||||||
</>
|
</>
|
||||||
) : (
|
) : (
|
||||||
<Forms.FormText className={Margins.marginBottom8}>
|
<Forms.FormText className={Margins.bottom8}>
|
||||||
{isOutdated ? `There are ${updates.length} Updates` : "Up to Date!"}
|
{isOutdated ? `There are ${updates.length} Updates` : "Up to Date!"}
|
||||||
</Forms.FormText>
|
</Forms.FormText>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
{isOutdated && <Changes updates={updates} {...props} />}
|
{isOutdated && <Changes updates={updates} {...props} />}
|
||||||
|
|
||||||
<Flex className={classes(Margins.marginBottom8, Margins.marginTop8)}>
|
<Flex className={classes(Margins.bottom8, Margins.top8)}>
|
||||||
{isOutdated && <Button
|
{isOutdated && <Button
|
||||||
size={Button.Sizes.SMALL}
|
size={Button.Sizes.SMALL}
|
||||||
disabled={isUpdating || isChecking}
|
disabled={isUpdating || isChecking}
|
||||||
onClick={withDispatcher(setIsUpdating, async () => {
|
onClick={withDispatcher(setIsUpdating, async () => {
|
||||||
if (await update()) {
|
if (await update()) {
|
||||||
setUpdates([]);
|
setUpdates([]);
|
||||||
const needFullRestart = await rebuild();
|
|
||||||
await new Promise<void>(r => {
|
await new Promise<void>(r => {
|
||||||
Alerts.show({
|
Alerts.show({
|
||||||
title: "Update Success!",
|
title: "Update Success!",
|
||||||
@ -131,10 +133,7 @@ function Updatable(props: CommonProps) {
|
|||||||
confirmText: "Restart",
|
confirmText: "Restart",
|
||||||
cancelText: "Not now!",
|
cancelText: "Not now!",
|
||||||
onConfirm() {
|
onConfirm() {
|
||||||
if (needFullRestart)
|
relaunch();
|
||||||
window.DiscordNative.app.relaunch();
|
|
||||||
else
|
|
||||||
location.reload();
|
|
||||||
r();
|
r();
|
||||||
},
|
},
|
||||||
onCancel: r
|
onCancel: r
|
||||||
@ -175,7 +174,7 @@ function Updatable(props: CommonProps) {
|
|||||||
function Newer(props: CommonProps) {
|
function Newer(props: CommonProps) {
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<Forms.FormText className={Margins.marginBottom8}>
|
<Forms.FormText className={Margins.bottom8}>
|
||||||
Your local copy has more recent commits. Please stash or reset them.
|
Your local copy has more recent commits. Please stash or reset them.
|
||||||
</Forms.FormText>
|
</Forms.FormText>
|
||||||
<Changes {...props} updates={changes} />
|
<Changes {...props} updates={changes} />
|
||||||
@ -184,7 +183,7 @@ function Newer(props: CommonProps) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function Updater() {
|
function Updater() {
|
||||||
const settings = useSettings(["notifyAboutUpdates", "autoUpdate"]);
|
const settings = useSettings(["notifyAboutUpdates", "autoUpdate", "autoUpdateNotification"]);
|
||||||
|
|
||||||
const [repo, err, repoPending] = useAwaiter(getRepo, { fallbackValue: "Loading..." });
|
const [repo, err, repoPending] = useAwaiter(getRepo, { fallbackValue: "Loading..." });
|
||||||
|
|
||||||
@ -199,12 +198,12 @@ function Updater() {
|
|||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Forms.FormSection className={Margins.marginTop16}>
|
<SettingsTab title="Vencord Updater">
|
||||||
<Forms.FormTitle tag="h5">Updater Settings</Forms.FormTitle>
|
<Forms.FormTitle tag="h5">Updater Settings</Forms.FormTitle>
|
||||||
<Switch
|
<Switch
|
||||||
value={settings.notifyAboutUpdates}
|
value={settings.notifyAboutUpdates}
|
||||||
onChange={(v: boolean) => settings.notifyAboutUpdates = v}
|
onChange={(v: boolean) => settings.notifyAboutUpdates = v}
|
||||||
note="Shows a toast on startup"
|
note="Shows a notification on startup"
|
||||||
disabled={settings.autoUpdate}
|
disabled={settings.autoUpdate}
|
||||||
>
|
>
|
||||||
Get notified about new updates
|
Get notified about new updates
|
||||||
@ -216,25 +215,38 @@ function Updater() {
|
|||||||
>
|
>
|
||||||
Automatically update
|
Automatically update
|
||||||
</Switch>
|
</Switch>
|
||||||
|
<Switch
|
||||||
|
value={settings.autoUpdateNotification}
|
||||||
|
onChange={(v: boolean) => settings.autoUpdateNotification = v}
|
||||||
|
note="Shows a notification when Vencord automatically updates"
|
||||||
|
disabled={!settings.autoUpdate}
|
||||||
|
>
|
||||||
|
Get notified when an automatic update completes
|
||||||
|
</Switch>
|
||||||
|
|
||||||
<Forms.FormTitle tag="h5">Repo</Forms.FormTitle>
|
<Forms.FormTitle tag="h5">Repo</Forms.FormTitle>
|
||||||
|
|
||||||
<Forms.FormText>{repoPending ? repo : err ? "Failed to retrieve - check console" : (
|
<Forms.FormText className="vc-text-selectable">
|
||||||
<Link href={repo}>
|
{repoPending
|
||||||
{repo.split("/").slice(-2).join("/")}
|
? repo
|
||||||
</Link>
|
: err
|
||||||
)} (<HashLink hash={gitHash} repo={repo} disabled={repoPending} />)</Forms.FormText>
|
? "Failed to retrieve - check console"
|
||||||
|
: (
|
||||||
|
<Link href={repo}>
|
||||||
|
{repo.split("/").slice(-2).join("/")}
|
||||||
|
</Link>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
{" "}(<HashLink hash={gitHash} repo={repo} disabled={repoPending} />)
|
||||||
|
</Forms.FormText>
|
||||||
|
|
||||||
<Forms.FormDivider className={Margins.marginTop8 + " " + Margins.marginBottom8} />
|
<Forms.FormDivider className={Margins.top8 + " " + Margins.bottom8} />
|
||||||
|
|
||||||
<Forms.FormTitle tag="h5">Updates</Forms.FormTitle>
|
<Forms.FormTitle tag="h5">Updates</Forms.FormTitle>
|
||||||
|
|
||||||
{isNewer ? <Newer {...commonProps} /> : <Updatable {...commonProps} />}
|
{isNewer ? <Newer {...commonProps} /> : <Updatable {...commonProps} />}
|
||||||
</Forms.FormSection >
|
</SettingsTab>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
export default IS_WEB ? null : ErrorBoundary.wrap(Updater, {
|
export default IS_UPDATER_DISABLED ? null : wrapTab(Updater, "Updater");
|
||||||
message: "Failed to render the Updater. If this persists, try using the installer to reinstall!",
|
|
||||||
onError: handleComponentFailed,
|
|
||||||
});
|
|
@ -16,17 +16,19 @@
|
|||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
import { openNotificationLogModal } from "@api/Notifications/notificationLog";
|
||||||
import { useSettings } from "@api/settings";
|
import { Settings, useSettings } from "@api/Settings";
|
||||||
import { classNameFactory } from "@api/Styles";
|
import { classNameFactory } from "@api/Styles";
|
||||||
import DonateButton from "@components/DonateButton";
|
import DonateButton from "@components/DonateButton";
|
||||||
import ErrorBoundary from "@components/ErrorBoundary";
|
|
||||||
import { ErrorCard } from "@components/ErrorCard";
|
import { ErrorCard } from "@components/ErrorCard";
|
||||||
import IpcEvents from "@utils/IpcEvents";
|
|
||||||
import { Margins } from "@utils/margins";
|
import { Margins } from "@utils/margins";
|
||||||
import { identity, useAwaiter } from "@utils/misc";
|
import { identity } from "@utils/misc";
|
||||||
|
import { relaunch, showItemInFolder } from "@utils/native";
|
||||||
|
import { useAwaiter } from "@utils/react";
|
||||||
import { Button, Card, Forms, React, Select, Slider, Switch } from "@webpack/common";
|
import { Button, Card, Forms, React, Select, Slider, Switch } from "@webpack/common";
|
||||||
|
|
||||||
|
import { SettingsTab, wrapTab } from "./shared";
|
||||||
|
|
||||||
const cl = classNameFactory("vc-settings-");
|
const cl = classNameFactory("vc-settings-");
|
||||||
|
|
||||||
const DEFAULT_DONATE_IMAGE = "https://cdn.discordapp.com/emojis/1026533090627174460.png";
|
const DEFAULT_DONATE_IMAGE = "https://cdn.discordapp.com/emojis/1026533090627174460.png";
|
||||||
@ -37,15 +39,15 @@ type KeysOfType<Object, Type> = {
|
|||||||
}[keyof Object];
|
}[keyof Object];
|
||||||
|
|
||||||
function VencordSettings() {
|
function VencordSettings() {
|
||||||
const [settingsDir, , settingsDirPending] = useAwaiter(() => VencordNative.ipc.invoke<string>(IpcEvents.GET_SETTINGS_DIR), {
|
const [settingsDir, , settingsDirPending] = useAwaiter(VencordNative.settings.getSettingsDir, {
|
||||||
fallbackValue: "Loading..."
|
fallbackValue: "Loading..."
|
||||||
});
|
});
|
||||||
const settings = useSettings();
|
const settings = useSettings();
|
||||||
const notifSettings = settings.notifications;
|
|
||||||
|
|
||||||
const donateImage = React.useMemo(() => Math.random() > 0.5 ? DEFAULT_DONATE_IMAGE : SHIGGY_DONATE_IMAGE, []);
|
const donateImage = React.useMemo(() => Math.random() > 0.5 ? DEFAULT_DONATE_IMAGE : SHIGGY_DONATE_IMAGE, []);
|
||||||
|
|
||||||
const isWindows = navigator.platform.toLowerCase().startsWith("win");
|
const isWindows = navigator.platform.toLowerCase().startsWith("win");
|
||||||
|
const isMac = navigator.platform.toLowerCase().startsWith("mac");
|
||||||
|
|
||||||
const Switches: Array<false | {
|
const Switches: Array<false | {
|
||||||
key: KeysOfType<typeof settings, boolean>;
|
key: KeysOfType<typeof settings, boolean>;
|
||||||
@ -63,12 +65,16 @@ function VencordSettings() {
|
|||||||
title: "Enable React Developer Tools",
|
title: "Enable React Developer Tools",
|
||||||
note: "Requires a full restart"
|
note: "Requires a full restart"
|
||||||
},
|
},
|
||||||
!IS_WEB && !isWindows && {
|
!IS_WEB && (!IS_DISCORD_DESKTOP || !isWindows ? {
|
||||||
key: "frameless",
|
key: "frameless",
|
||||||
title: "Disable the window frame",
|
title: "Disable the window frame",
|
||||||
note: "Requires a full restart"
|
note: "Requires a full restart"
|
||||||
},
|
} : {
|
||||||
!IS_WEB && {
|
key: "winNativeTitleBar",
|
||||||
|
title: "Use Windows' native title bar instead of Discord's custom one",
|
||||||
|
note: "Requires a full restart"
|
||||||
|
}),
|
||||||
|
!IS_WEB && false /* This causes electron to freeze / white screen for some people */ && {
|
||||||
key: "transparent",
|
key: "transparent",
|
||||||
title: "Enable window transparency",
|
title: "Enable window transparency",
|
||||||
note: "Requires a full restart"
|
note: "Requires a full restart"
|
||||||
@ -77,48 +83,53 @@ function VencordSettings() {
|
|||||||
key: "winCtrlQ",
|
key: "winCtrlQ",
|
||||||
title: "Register Ctrl+Q as shortcut to close Discord (Alternative to Alt+F4)",
|
title: "Register Ctrl+Q as shortcut to close Discord (Alternative to Alt+F4)",
|
||||||
note: "Requires a full restart"
|
note: "Requires a full restart"
|
||||||
|
},
|
||||||
|
IS_DISCORD_DESKTOP && {
|
||||||
|
key: "disableMinSize",
|
||||||
|
title: "Disable minimum window size",
|
||||||
|
note: "Requires a full restart"
|
||||||
|
},
|
||||||
|
IS_DISCORD_DESKTOP && isMac && {
|
||||||
|
key: "macosTranslucency",
|
||||||
|
title: "Enable translucent window",
|
||||||
|
note: "Requires a full restart"
|
||||||
}
|
}
|
||||||
];
|
];
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<React.Fragment>
|
<SettingsTab title="Vencord Settings">
|
||||||
<DonateCard image={donateImage} />
|
<DonateCard image={donateImage} />
|
||||||
<Forms.FormSection title="Quick Actions">
|
<Forms.FormSection title="Quick Actions">
|
||||||
<Card className={cl("quick-actions-card")}>
|
<Card className={cl("quick-actions-card")}>
|
||||||
{IS_WEB ? (
|
<React.Fragment>
|
||||||
|
{!IS_WEB && (
|
||||||
|
<Button
|
||||||
|
onClick={relaunch}
|
||||||
|
size={Button.Sizes.SMALL}>
|
||||||
|
Restart Client
|
||||||
|
</Button>
|
||||||
|
)}
|
||||||
<Button
|
<Button
|
||||||
onClick={() => require("../Monaco").launchMonacoEditor()}
|
onClick={() => VencordNative.quickCss.openEditor()}
|
||||||
size={Button.Sizes.SMALL}
|
size={Button.Sizes.SMALL}
|
||||||
disabled={settingsDir === "Loading..."}>
|
disabled={settingsDir === "Loading..."}>
|
||||||
Open QuickCSS File
|
Open QuickCSS File
|
||||||
</Button>
|
</Button>
|
||||||
) : (
|
{!IS_WEB && (
|
||||||
<React.Fragment>
|
|
||||||
<Button
|
<Button
|
||||||
onClick={() => window.DiscordNative.app.relaunch()}
|
onClick={() => showItemInFolder(settingsDir)}
|
||||||
size={Button.Sizes.SMALL}>
|
|
||||||
Restart Client
|
|
||||||
</Button>
|
|
||||||
<Button
|
|
||||||
onClick={() => VencordNative.ipc.invoke(IpcEvents.OPEN_MONACO_EDITOR)}
|
|
||||||
size={Button.Sizes.SMALL}
|
|
||||||
disabled={settingsDir === "Loading..."}>
|
|
||||||
Open QuickCSS File
|
|
||||||
</Button>
|
|
||||||
<Button
|
|
||||||
onClick={() => window.DiscordNative.fileManager.showItemInFolder(settingsDir)}
|
|
||||||
size={Button.Sizes.SMALL}
|
size={Button.Sizes.SMALL}
|
||||||
disabled={settingsDirPending}>
|
disabled={settingsDirPending}>
|
||||||
Open Settings Folder
|
Open Settings Folder
|
||||||
</Button>
|
</Button>
|
||||||
<Button
|
)}
|
||||||
onClick={() => VencordNative.ipc.invoke(IpcEvents.OPEN_EXTERNAL, "https://github.com/Vendicated/Vencord")}
|
<Button
|
||||||
size={Button.Sizes.SMALL}
|
onClick={() => VencordNative.native.openExternal("https://github.com/Vendicated/Vencord")}
|
||||||
disabled={settingsDirPending}>
|
size={Button.Sizes.SMALL}
|
||||||
Open in GitHub
|
disabled={settingsDirPending}>
|
||||||
</Button>
|
Open in GitHub
|
||||||
</React.Fragment>
|
</Button>
|
||||||
)}
|
</React.Fragment>
|
||||||
</Card>
|
</Card>
|
||||||
</Forms.FormSection>
|
</Forms.FormSection>
|
||||||
|
|
||||||
@ -141,8 +152,16 @@ function VencordSettings() {
|
|||||||
</Forms.FormSection>
|
</Forms.FormSection>
|
||||||
|
|
||||||
|
|
||||||
|
{typeof Notification !== "undefined" && <NotificationSection settings={settings.notifications} />}
|
||||||
|
</SettingsTab>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
function NotificationSection({ settings }: { settings: typeof Settings["notifications"]; }) {
|
||||||
|
return (
|
||||||
|
<>
|
||||||
<Forms.FormTitle tag="h5">Notification Style</Forms.FormTitle>
|
<Forms.FormTitle tag="h5">Notification Style</Forms.FormTitle>
|
||||||
{notifSettings.useNative !== "never" && Notification.permission === "denied" && (
|
{settings.useNative !== "never" && Notification?.permission === "denied" && (
|
||||||
<ErrorCard style={{ padding: "1em" }} className={Margins.bottom8}>
|
<ErrorCard style={{ padding: "1em" }} className={Margins.bottom8}>
|
||||||
<Forms.FormTitle tag="h5">Desktop Notification Permission denied</Forms.FormTitle>
|
<Forms.FormTitle tag="h5">Desktop Notification Permission denied</Forms.FormTitle>
|
||||||
<Forms.FormText>You have denied Notification Permissions. Thus, Desktop notifications will not work!</Forms.FormText>
|
<Forms.FormText>You have denied Notification Permissions. Thus, Desktop notifications will not work!</Forms.FormText>
|
||||||
@ -161,44 +180,66 @@ function VencordSettings() {
|
|||||||
{ label: "Only use Desktop notifications when Discord is not focused", value: "not-focused", default: true },
|
{ label: "Only use Desktop notifications when Discord is not focused", value: "not-focused", default: true },
|
||||||
{ label: "Always use Desktop notifications", value: "always" },
|
{ label: "Always use Desktop notifications", value: "always" },
|
||||||
{ label: "Always use Vencord notifications", value: "never" },
|
{ label: "Always use Vencord notifications", value: "never" },
|
||||||
]satisfies Array<{ value: typeof settings["notifications"]["useNative"]; } & Record<string, any>>}
|
] satisfies Array<{ value: typeof settings["useNative"]; } & Record<string, any>>}
|
||||||
closeOnSelect={true}
|
closeOnSelect={true}
|
||||||
select={v => notifSettings.useNative = v}
|
select={v => settings.useNative = v}
|
||||||
isSelected={v => v === notifSettings.useNative}
|
isSelected={v => v === settings.useNative}
|
||||||
serialize={identity}
|
serialize={identity}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<Forms.FormTitle tag="h5" className={Margins.top16 + " " + Margins.bottom8}>Notification Position</Forms.FormTitle>
|
<Forms.FormTitle tag="h5" className={Margins.top16 + " " + Margins.bottom8}>Notification Position</Forms.FormTitle>
|
||||||
<Select
|
<Select
|
||||||
isDisabled={notifSettings.useNative === "always"}
|
isDisabled={settings.useNative === "always"}
|
||||||
placeholder="Notification Position"
|
placeholder="Notification Position"
|
||||||
options={[
|
options={[
|
||||||
{ label: "Bottom Right", value: "bottom-right", default: true },
|
{ label: "Bottom Right", value: "bottom-right", default: true },
|
||||||
{ label: "Top Right", value: "top-right" },
|
{ label: "Top Right", value: "top-right" },
|
||||||
]satisfies Array<{ value: typeof settings["notifications"]["position"]; } & Record<string, any>>}
|
] satisfies Array<{ value: typeof settings["position"]; } & Record<string, any>>}
|
||||||
select={v => notifSettings.position = v}
|
select={v => settings.position = v}
|
||||||
isSelected={v => v === notifSettings.position}
|
isSelected={v => v === settings.position}
|
||||||
serialize={identity}
|
serialize={identity}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<Forms.FormTitle tag="h5" className={Margins.top16 + " " + Margins.bottom8}>Notification Timeout</Forms.FormTitle>
|
<Forms.FormTitle tag="h5" className={Margins.top16 + " " + Margins.bottom8}>Notification Timeout</Forms.FormTitle>
|
||||||
<Forms.FormText className={Margins.bottom16}>Set to 0s to never automatically time out</Forms.FormText>
|
<Forms.FormText className={Margins.bottom16}>Set to 0s to never automatically time out</Forms.FormText>
|
||||||
<Slider
|
<Slider
|
||||||
disabled={notifSettings.useNative === "always"}
|
disabled={settings.useNative === "always"}
|
||||||
markers={[0, 1000, 2500, 5000, 10_000, 20_000]}
|
markers={[0, 1000, 2500, 5000, 10_000, 20_000]}
|
||||||
minValue={0}
|
minValue={0}
|
||||||
maxValue={20_000}
|
maxValue={20_000}
|
||||||
initialValue={notifSettings.timeout}
|
initialValue={settings.timeout}
|
||||||
onValueChange={v => notifSettings.timeout = v}
|
onValueChange={v => settings.timeout = v}
|
||||||
onValueRender={v => (v / 1000).toFixed(2) + "s"}
|
onValueRender={v => (v / 1000).toFixed(2) + "s"}
|
||||||
onMarkerRender={v => (v / 1000) + "s"}
|
onMarkerRender={v => (v / 1000) + "s"}
|
||||||
stickToMarkers={false}
|
stickToMarkers={false}
|
||||||
/>
|
/>
|
||||||
</React.Fragment>
|
|
||||||
|
<Forms.FormTitle tag="h5" className={Margins.top16 + " " + Margins.bottom8}>Notification Log Limit</Forms.FormTitle>
|
||||||
|
<Forms.FormText className={Margins.bottom16}>
|
||||||
|
The amount of notifications to save in the log until old ones are removed.
|
||||||
|
Set to <code>0</code> to disable Notification log and <code>∞</code> to never automatically remove old Notifications
|
||||||
|
</Forms.FormText>
|
||||||
|
<Slider
|
||||||
|
markers={[0, 25, 50, 75, 100, 200]}
|
||||||
|
minValue={0}
|
||||||
|
maxValue={200}
|
||||||
|
stickToMarkers={true}
|
||||||
|
initialValue={settings.logLimit}
|
||||||
|
onValueChange={v => settings.logLimit = v}
|
||||||
|
onValueRender={v => v === 200 ? "∞" : v}
|
||||||
|
onMarkerRender={v => v === 200 ? "∞" : v}
|
||||||
|
/>
|
||||||
|
|
||||||
|
<Button
|
||||||
|
onClick={openNotificationLogModal}
|
||||||
|
disabled={settings.logLimit === 0}
|
||||||
|
>
|
||||||
|
Open Notification Log
|
||||||
|
</Button>
|
||||||
|
</>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
interface DonateCardProps {
|
interface DonateCardProps {
|
||||||
image: string;
|
image: string;
|
||||||
}
|
}
|
||||||
@ -222,4 +263,4 @@ function DonateCard({ image }: DonateCardProps) {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
export default ErrorBoundary.wrap(VencordSettings);
|
export default wrapTab(VencordSettings, "Vencord Settings");
|
||||||
|
63
src/components/VencordSettings/addonCard.css
Normal file
63
src/components/VencordSettings/addonCard.css
Normal file
@ -0,0 +1,63 @@
|
|||||||
|
.vc-addon-card {
|
||||||
|
background-color: var(--background-secondary-alt);
|
||||||
|
color: var(--interactive-active);
|
||||||
|
border-radius: 8px;
|
||||||
|
display: block;
|
||||||
|
height: 100%;
|
||||||
|
padding: 12px;
|
||||||
|
width: 100%;
|
||||||
|
transition: 0.1s ease-out;
|
||||||
|
transition-property: box-shadow, transform, background, opacity;
|
||||||
|
}
|
||||||
|
|
||||||
|
.vc-addon-card-disabled {
|
||||||
|
opacity: 0.6;
|
||||||
|
}
|
||||||
|
|
||||||
|
.vc-addon-card:hover {
|
||||||
|
background-color: var(--background-tertiary);
|
||||||
|
transform: translateY(-1px);
|
||||||
|
box-shadow: var(--elevation-high);
|
||||||
|
}
|
||||||
|
|
||||||
|
.vc-addon-header {
|
||||||
|
margin-top: auto;
|
||||||
|
display: flex;
|
||||||
|
width: 100%;
|
||||||
|
justify-content: flex-end;
|
||||||
|
align-items: center;
|
||||||
|
gap: 8px;
|
||||||
|
margin-bottom: 0.5em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.vc-addon-note {
|
||||||
|
height: 36px;
|
||||||
|
overflow: hidden;
|
||||||
|
text-overflow: ellipsis;
|
||||||
|
display: -webkit-box;
|
||||||
|
-webkit-line-clamp: 2;
|
||||||
|
line-clamp: 2;
|
||||||
|
-webkit-box-orient: vertical;
|
||||||
|
/* stylelint-disable-next-line property-no-unknown */
|
||||||
|
box-orient: vertical;
|
||||||
|
}
|
||||||
|
|
||||||
|
.vc-addon-name-author {
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.vc-addon-name {
|
||||||
|
display: flex;
|
||||||
|
width: 100%;
|
||||||
|
align-items: center;
|
||||||
|
flex-grow: 1;
|
||||||
|
gap: 8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.vc-addon-author {
|
||||||
|
font-size: 0.8em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.vc-addon-author::before {
|
||||||
|
content: "by ";
|
||||||
|
}
|
@ -1,89 +0,0 @@
|
|||||||
/*
|
|
||||||
* Vencord, a modification for Discord's desktop app
|
|
||||||
* Copyright (c) 2022 Vendicated and contributors
|
|
||||||
*
|
|
||||||
* This program is free software: you can redistribute it and/or modify
|
|
||||||
* it under the terms of the GNU General Public License as published by
|
|
||||||
* the Free Software Foundation, either version 3 of the License, or
|
|
||||||
* (at your option) any later version.
|
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
||||||
* GNU General Public License for more details.
|
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU General Public License
|
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
||||||
*/
|
|
||||||
|
|
||||||
import "./settingsStyles.css";
|
|
||||||
|
|
||||||
import { classNameFactory } from "@api/Styles";
|
|
||||||
import ErrorBoundary from "@components/ErrorBoundary";
|
|
||||||
import { findByCodeLazy } from "@webpack";
|
|
||||||
import { Forms, SettingsRouter, Text } from "@webpack/common";
|
|
||||||
|
|
||||||
import BackupRestoreTab from "./BackupRestoreTab";
|
|
||||||
import PluginsTab from "./PluginsTab";
|
|
||||||
import ThemesTab from "./ThemesTab";
|
|
||||||
import Updater from "./Updater";
|
|
||||||
import VencordSettings from "./VencordTab";
|
|
||||||
|
|
||||||
const cl = classNameFactory("vc-settings-");
|
|
||||||
|
|
||||||
const TabBar = findByCodeLazy('[role="tab"][aria-disabled="false"]');
|
|
||||||
|
|
||||||
interface SettingsProps {
|
|
||||||
tab: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
interface SettingsTab {
|
|
||||||
name: string;
|
|
||||||
component?: React.ComponentType;
|
|
||||||
}
|
|
||||||
|
|
||||||
const SettingsTabs: Record<string, SettingsTab> = {
|
|
||||||
VencordSettings: { name: "Vencord", component: () => <VencordSettings /> },
|
|
||||||
VencordPlugins: { name: "Plugins", component: () => <PluginsTab /> },
|
|
||||||
VencordThemes: { name: "Themes", component: () => <ThemesTab /> },
|
|
||||||
VencordUpdater: { name: "Updater" }, // Only show updater if IS_WEB is false
|
|
||||||
VencordSettingsSync: { name: "Backup & Restore", component: () => <BackupRestoreTab /> },
|
|
||||||
};
|
|
||||||
|
|
||||||
if (!IS_WEB) SettingsTabs.VencordUpdater.component = () => Updater && <Updater />;
|
|
||||||
|
|
||||||
function Settings(props: SettingsProps) {
|
|
||||||
const { tab = "VencordSettings" } = props;
|
|
||||||
|
|
||||||
const CurrentTab = SettingsTabs[tab]?.component;
|
|
||||||
|
|
||||||
return <Forms.FormSection>
|
|
||||||
<Text variant="heading-md/normal" tag="h2">Vencord Settings</Text>
|
|
||||||
|
|
||||||
<TabBar
|
|
||||||
type={TabBar.Types.TOP}
|
|
||||||
look={TabBar.Looks.BRAND}
|
|
||||||
className={cl("tab-bar")}
|
|
||||||
selectedItem={tab}
|
|
||||||
onItemSelect={SettingsRouter.open}
|
|
||||||
>
|
|
||||||
{Object.entries(SettingsTabs).map(([key, { name, component }]) => {
|
|
||||||
if (!component) return null;
|
|
||||||
return <TabBar.Item
|
|
||||||
id={key}
|
|
||||||
className={cl("tab-bar-item")}
|
|
||||||
key={key}>
|
|
||||||
{name}
|
|
||||||
</TabBar.Item>;
|
|
||||||
})}
|
|
||||||
</TabBar>
|
|
||||||
<Forms.FormDivider />
|
|
||||||
{CurrentTab && <CurrentTab />}
|
|
||||||
</Forms.FormSection >;
|
|
||||||
}
|
|
||||||
|
|
||||||
export default function (props: SettingsProps) {
|
|
||||||
return <ErrorBoundary>
|
|
||||||
<Settings tab={props.tab} />
|
|
||||||
</ErrorBoundary>;
|
|
||||||
}
|
|
@ -1,6 +1,6 @@
|
|||||||
.vc-settings-tab-bar {
|
.vc-settings-tab-bar {
|
||||||
margin-top: 20px;
|
margin-top: 20px;
|
||||||
margin-bottom: -2px;
|
margin-bottom: 10px;
|
||||||
border-bottom: 2px solid var(--background-modifier-accent);
|
border-bottom: 2px solid var(--background-modifier-accent);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -15,7 +15,7 @@
|
|||||||
display: flex;
|
display: flex;
|
||||||
gap: 1em;
|
gap: 1em;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
justify-content: space-between;
|
justify-content: space-evenly;
|
||||||
flex-grow: 1;
|
flex-grow: 1;
|
||||||
flex-flow: row wrap;
|
flex-flow: row wrap;
|
||||||
margin-bottom: 1em;
|
margin-bottom: 1em;
|
||||||
@ -29,12 +29,39 @@
|
|||||||
.vc-settings-card {
|
.vc-settings-card {
|
||||||
padding: 1em;
|
padding: 1em;
|
||||||
margin-bottom: 1em;
|
margin-bottom: 1em;
|
||||||
margin-top: 1em;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.vc-backup-restore-card {
|
.vc-backup-restore-card {
|
||||||
background-color: var(--info-warning-background);
|
background-color: var(--info-warning-background);
|
||||||
border-color: var(--info-warning-foreground);
|
border-color: var(--info-warning-foreground);
|
||||||
color: var(--info-warning-text);
|
color: var(--info-warning-text);
|
||||||
margin-top: 0;
|
}
|
||||||
|
|
||||||
|
.vc-settings-theme-links {
|
||||||
|
/* Needed to fix bad themes that hide certain textarea elements for whatever eldritch reason */
|
||||||
|
display: inline-block !important;
|
||||||
|
color: var(--text-normal) !important;
|
||||||
|
padding: 0.5em;
|
||||||
|
border: 1px solid var(--background-modifier-accent);
|
||||||
|
max-height: unset;
|
||||||
|
}
|
||||||
|
|
||||||
|
.vc-cloud-settings-sync-grid {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: repeat(3, 1fr);
|
||||||
|
grid-gap: 1em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.vc-cloud-erase-data-danger-btn {
|
||||||
|
color: var(--white-500);
|
||||||
|
background-color: var(--button-danger-background);
|
||||||
|
}
|
||||||
|
|
||||||
|
.vc-text-selectable,
|
||||||
|
.vc-text-selectable :where([class*="text" i], [class*="title" i]) {
|
||||||
|
/* make text selectable, silly discord makes the entirety of settings not selectable */
|
||||||
|
user-select: text;
|
||||||
|
|
||||||
|
/* discord also sets cursor: default which prevents the cursor from showing as text */
|
||||||
|
cursor: initial;
|
||||||
}
|
}
|
||||||
|
52
src/components/VencordSettings/shared.tsx
Normal file
52
src/components/VencordSettings/shared.tsx
Normal file
@ -0,0 +1,52 @@
|
|||||||
|
/*
|
||||||
|
* Vencord, a modification for Discord's desktop app
|
||||||
|
* Copyright (c) 2023 Vendicated and contributors
|
||||||
|
*
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import "./settingsStyles.css";
|
||||||
|
import "./themesStyles.css";
|
||||||
|
|
||||||
|
import ErrorBoundary from "@components/ErrorBoundary";
|
||||||
|
import { handleComponentFailed } from "@components/handleComponentFailed";
|
||||||
|
import { Margins } from "@utils/margins";
|
||||||
|
import { onlyOnce } from "@utils/onlyOnce";
|
||||||
|
import { Forms, Text } from "@webpack/common";
|
||||||
|
import type { ComponentType, PropsWithChildren } from "react";
|
||||||
|
|
||||||
|
export function SettingsTab({ title, children }: PropsWithChildren<{ title: string; }>) {
|
||||||
|
return (
|
||||||
|
<Forms.FormSection>
|
||||||
|
<Text
|
||||||
|
variant="heading-lg/semibold"
|
||||||
|
tag="h2"
|
||||||
|
className={Margins.bottom16}
|
||||||
|
>
|
||||||
|
{title}
|
||||||
|
</Text>
|
||||||
|
|
||||||
|
{children}
|
||||||
|
</Forms.FormSection>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
const onError = onlyOnce(handleComponentFailed);
|
||||||
|
|
||||||
|
export function wrapTab(component: ComponentType, tab: string) {
|
||||||
|
return ErrorBoundary.wrap(component, {
|
||||||
|
message: `Failed to render the ${tab} tab. If this issue persists, try using the installer to reinstall!`,
|
||||||
|
onError,
|
||||||
|
});
|
||||||
|
}
|
29
src/components/VencordSettings/themesStyles.css
Normal file
29
src/components/VencordSettings/themesStyles.css
Normal file
@ -0,0 +1,29 @@
|
|||||||
|
.vc-settings-theme-grid {
|
||||||
|
display: grid;
|
||||||
|
grid-gap: 16px;
|
||||||
|
grid-template-columns: repeat(auto-fill, minmax(250px, 1fr));
|
||||||
|
}
|
||||||
|
|
||||||
|
.vc-settings-theme-card {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
background-color: var(--background-secondary-alt);
|
||||||
|
color: var(--interactive-active);
|
||||||
|
border-radius: 8px;
|
||||||
|
padding: 1em;
|
||||||
|
width: 100%;
|
||||||
|
transition: 0.1s ease-out;
|
||||||
|
transition-property: box-shadow, transform, background, opacity;
|
||||||
|
}
|
||||||
|
|
||||||
|
.vc-settings-theme-card-text {
|
||||||
|
text-overflow: ellipsis;
|
||||||
|
height: 1.2em;
|
||||||
|
margin-bottom: 2px;
|
||||||
|
white-space: nowrap;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
.vc-settings-theme-author::before {
|
||||||
|
content: "by ";
|
||||||
|
}
|
@ -16,29 +16,12 @@
|
|||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import { isOutdated, rebuild, update } from "@utils/updater";
|
import { maybePromptToUpdate } from "@utils/updater";
|
||||||
|
|
||||||
export async function handleComponentFailed() {
|
export function handleComponentFailed() {
|
||||||
if (isOutdated) {
|
maybePromptToUpdate(
|
||||||
setImmediate(async () => {
|
"Uh Oh! Failed to render this Page." +
|
||||||
const wantsUpdate = confirm(
|
" However, there is an update available that might fix it." +
|
||||||
"Uh Oh! Failed to render this Page." +
|
" Would you like to update and restart now?"
|
||||||
" However, there is an update available that might fix it." +
|
);
|
||||||
" Would you like to update and restart now?"
|
|
||||||
);
|
|
||||||
if (wantsUpdate) {
|
|
||||||
try {
|
|
||||||
await update();
|
|
||||||
await rebuild();
|
|
||||||
if (IS_WEB)
|
|
||||||
location.reload();
|
|
||||||
else
|
|
||||||
DiscordNative.app.relaunch();
|
|
||||||
} catch (e) {
|
|
||||||
console.error(e);
|
|
||||||
alert("That also failed :( Try updating or reinstalling with the installer!");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
7
src/components/iconStyles.css
Normal file
7
src/components/iconStyles.css
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
.vc-open-external-icon {
|
||||||
|
transform: rotate(45deg);
|
||||||
|
}
|
||||||
|
|
||||||
|
.vc-owner-crown-icon {
|
||||||
|
color: var(--text-warning);
|
||||||
|
}
|
@ -1,52 +1,66 @@
|
|||||||
<!doctype html>
|
<!DOCTYPE html>
|
||||||
<html lang="en">
|
<html lang="en">
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8" />
|
||||||
|
<title>Vencord QuickCSS Editor</title>
|
||||||
|
<link
|
||||||
|
rel="stylesheet"
|
||||||
|
href="https://cdnjs.cloudflare.com/ajax/libs/monaco-editor/0.40.0/min/vs/editor/editor.main.min.css"
|
||||||
|
integrity="sha512-MOoQ02h80hklccfLrXFYkCzG+WVjORflOp9Zp8dltiaRP+35LYnO4LKOklR64oMGfGgJDLO8WJpkM1o5gZXYZQ=="
|
||||||
|
crossorigin="anonymous"
|
||||||
|
referrerpolicy="no-referrer"
|
||||||
|
/>
|
||||||
|
<style>
|
||||||
|
html,
|
||||||
|
body,
|
||||||
|
#container {
|
||||||
|
position: absolute;
|
||||||
|
left: 0;
|
||||||
|
top: 0;
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
margin: 0;
|
||||||
|
padding: 0;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
|
||||||
<head>
|
<body>
|
||||||
<meta charset="utf-8">
|
<div id="container"></div>
|
||||||
<title>QuickCss Editor</title>
|
<script
|
||||||
<link rel="stylesheet" data-name="vs/editor/editor.main"
|
src="https://cdnjs.cloudflare.com/ajax/libs/monaco-editor/0.40.0/min/vs/loader.min.js"
|
||||||
href="https://cdnjs.cloudflare.com/ajax/libs/monaco-editor/0.34.0/min/vs/editor/editor.main.min.css">
|
integrity="sha512-QzMpXeCPciAHP4wbYlV2PYgrQcaEkDQUjzkPU4xnjyVSD9T36/udamxtNBqb4qK4/bMQMPZ8ayrBe9hrGdBFjQ=="
|
||||||
<style>
|
crossorigin="anonymous"
|
||||||
html,
|
referrerpolicy="no-referrer"
|
||||||
body,
|
></script>
|
||||||
#container {
|
|
||||||
position: absolute;
|
|
||||||
left: 0;
|
|
||||||
top: 0;
|
|
||||||
width: 100%;
|
|
||||||
height: 100%;
|
|
||||||
margin: 0;
|
|
||||||
padding: 0;
|
|
||||||
overflow: hidden;
|
|
||||||
}
|
|
||||||
</style>
|
|
||||||
</head>
|
|
||||||
|
|
||||||
<body>
|
<script>
|
||||||
<div id="container"></div>
|
require.config({
|
||||||
<script src="https://cdnjs.cloudflare.com/ajax/libs/monaco-editor/0.34.0/min/vs/loader.min.js"></script>
|
paths: {
|
||||||
|
vs: "https://cdnjs.cloudflare.com/ajax/libs/monaco-editor/0.40.0/min/vs",
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
<script>
|
require(["vs/editor/editor.main"], () => {
|
||||||
require.config({ paths: { 'vs': 'https://cdnjs.cloudflare.com/ajax/libs/monaco-editor/0.34.0/min/vs' } });
|
getCurrentCss().then((css) => {
|
||||||
require(["vs/editor/editor.main"], () => {
|
var editor = monaco.editor.create(
|
||||||
getCurrentCss().then(css => {
|
document.getElementById("container"),
|
||||||
var editor = monaco.editor.create(document.getElementById('container'), {
|
{
|
||||||
value: css,
|
value: css,
|
||||||
language: 'css',
|
language: "css",
|
||||||
theme: getTheme(),
|
theme: getTheme(),
|
||||||
});
|
}
|
||||||
editor.onDidChangeModelContent(() =>
|
);
|
||||||
setCss(editor.getValue())
|
editor.onDidChangeModelContent(() =>
|
||||||
);
|
setCss(editor.getValue())
|
||||||
window.addEventListener("resize", () => {
|
);
|
||||||
// make monaco re-layout
|
window.addEventListener("resize", () => {
|
||||||
editor.layout();
|
// make monaco re-layout
|
||||||
|
editor.layout();
|
||||||
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
</script>
|
||||||
|
</body>
|
||||||
|
|
||||||
</script>
|
|
||||||
</body>
|
|
||||||
|
|
||||||
</html>
|
</html>
|
||||||
|
@ -16,7 +16,7 @@
|
|||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import Logger from "@utils/Logger";
|
import { Logger } from "@utils/Logger";
|
||||||
|
|
||||||
if (IS_DEV) {
|
if (IS_DEV) {
|
||||||
var traces = {} as Record<string, [number, any[]]>;
|
var traces = {} as Record<string, [number, any[]]>;
|
||||||
|
10
src/globals.d.ts
vendored
10
src/globals.d.ts
vendored
@ -35,6 +35,11 @@ declare global {
|
|||||||
export var IS_WEB: boolean;
|
export var IS_WEB: boolean;
|
||||||
export var IS_DEV: boolean;
|
export var IS_DEV: boolean;
|
||||||
export var IS_STANDALONE: boolean;
|
export var IS_STANDALONE: boolean;
|
||||||
|
export var IS_UPDATER_DISABLED: boolean;
|
||||||
|
export var IS_DISCORD_DESKTOP: boolean;
|
||||||
|
export var IS_VESKTOP: boolean;
|
||||||
|
export var VERSION: string;
|
||||||
|
export var BUILD_TIMESTAMP: number;
|
||||||
|
|
||||||
export var VencordNative: typeof import("./VencordNative").default;
|
export var VencordNative: typeof import("./VencordNative").default;
|
||||||
export var Vencord: typeof import("./Vencord");
|
export var Vencord: typeof import("./Vencord");
|
||||||
@ -51,10 +56,11 @@ declare global {
|
|||||||
* Only available when running in Electron, undefined on web.
|
* Only available when running in Electron, undefined on web.
|
||||||
* Thus, avoid using this or only use it inside an {@link IS_WEB} guard.
|
* Thus, avoid using this or only use it inside an {@link IS_WEB} guard.
|
||||||
*
|
*
|
||||||
* If you really must use it, mark your plugin as Desktop App only via
|
* If you really must use it, mark your plugin as Desktop App only by naming it Foo.desktop.ts(x)
|
||||||
* `target: "DESKTOP"`
|
|
||||||
*/
|
*/
|
||||||
export var DiscordNative: any;
|
export var DiscordNative: any;
|
||||||
|
export var Vesktop: any;
|
||||||
|
export var VesktopNative: any;
|
||||||
|
|
||||||
interface Window {
|
interface Window {
|
||||||
webpackChunkdiscord_app: {
|
webpackChunkdiscord_app: {
|
||||||
|
121
src/main/index.ts
Normal file
121
src/main/index.ts
Normal file
@ -0,0 +1,121 @@
|
|||||||
|
/*
|
||||||
|
* Vencord, a modification for Discord's desktop app
|
||||||
|
* Copyright (c) 2023 Vendicated and contributors
|
||||||
|
*
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { app, protocol, session } from "electron";
|
||||||
|
import { join } from "path";
|
||||||
|
|
||||||
|
import { ensureSafePath, getSettings } from "./ipcMain";
|
||||||
|
import { IS_VANILLA, THEMES_DIR } from "./utils/constants";
|
||||||
|
import { installExt } from "./utils/extensions";
|
||||||
|
|
||||||
|
if (IS_VESKTOP || !IS_VANILLA) {
|
||||||
|
app.whenReady().then(() => {
|
||||||
|
// Source Maps! Maybe there's a better way but since the renderer is executed
|
||||||
|
// from a string I don't think any other form of sourcemaps would work
|
||||||
|
protocol.registerFileProtocol("vencord", ({ url: unsafeUrl }, cb) => {
|
||||||
|
let url = unsafeUrl.slice("vencord://".length);
|
||||||
|
if (url.endsWith("/")) url = url.slice(0, -1);
|
||||||
|
if (url.startsWith("/themes/")) {
|
||||||
|
const theme = url.slice("/themes/".length);
|
||||||
|
const safeUrl = ensureSafePath(THEMES_DIR, theme);
|
||||||
|
if (!safeUrl) {
|
||||||
|
cb({ statusCode: 403 });
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
cb(safeUrl.replace(/\?v=\d+$/, ""));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
switch (url) {
|
||||||
|
case "renderer.js.map":
|
||||||
|
case "vencordDesktopRenderer.js.map":
|
||||||
|
case "preload.js.map":
|
||||||
|
case "vencordDesktopPreload.js.map":
|
||||||
|
case "patcher.js.map":
|
||||||
|
case "vencordDesktopMain.js.map":
|
||||||
|
cb(join(__dirname, url));
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
cb({ statusCode: 403 });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
try {
|
||||||
|
if (getSettings().enableReactDevtools)
|
||||||
|
installExt("fmkadmapgofadopljbjfkapdkoienihi")
|
||||||
|
.then(() => console.info("[Vencord] Installed React Developer Tools"))
|
||||||
|
.catch(err => console.error("[Vencord] Failed to install React Developer Tools", err));
|
||||||
|
} catch { }
|
||||||
|
|
||||||
|
|
||||||
|
// Remove CSP
|
||||||
|
type PolicyResult = Record<string, string[]>;
|
||||||
|
|
||||||
|
const parsePolicy = (policy: string): PolicyResult => {
|
||||||
|
const result: PolicyResult = {};
|
||||||
|
policy.split(";").forEach(directive => {
|
||||||
|
const [directiveKey, ...directiveValue] = directive.trim().split(/\s+/g);
|
||||||
|
if (directiveKey && !Object.prototype.hasOwnProperty.call(result, directiveKey)) {
|
||||||
|
result[directiveKey] = directiveValue;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
return result;
|
||||||
|
};
|
||||||
|
const stringifyPolicy = (policy: PolicyResult): string =>
|
||||||
|
Object.entries(policy)
|
||||||
|
.filter(([, values]) => values?.length)
|
||||||
|
.map(directive => directive.flat().join(" "))
|
||||||
|
.join("; ");
|
||||||
|
|
||||||
|
function patchCsp(headers: Record<string, string[]>, header: string) {
|
||||||
|
if (header in headers) {
|
||||||
|
const csp = parsePolicy(headers[header][0]);
|
||||||
|
|
||||||
|
for (const directive of ["style-src", "connect-src", "img-src", "font-src", "media-src", "worker-src"]) {
|
||||||
|
csp[directive] = ["*", "blob:", "data:", "vencord:", "'unsafe-inline'"];
|
||||||
|
}
|
||||||
|
// TODO: Restrict this to only imported packages with fixed version.
|
||||||
|
// Perhaps auto generate with esbuild
|
||||||
|
csp["script-src"] ??= [];
|
||||||
|
csp["script-src"].push("'unsafe-eval'", "https://unpkg.com", "https://cdnjs.cloudflare.com");
|
||||||
|
headers[header] = [stringifyPolicy(csp)];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
session.defaultSession.webRequest.onHeadersReceived(({ responseHeaders, resourceType }, cb) => {
|
||||||
|
if (responseHeaders) {
|
||||||
|
if (resourceType === "mainFrame")
|
||||||
|
patchCsp(responseHeaders, "content-security-policy");
|
||||||
|
|
||||||
|
// Fix hosts that don't properly set the css content type, such as
|
||||||
|
// raw.githubusercontent.com
|
||||||
|
if (resourceType === "stylesheet")
|
||||||
|
responseHeaders["content-type"] = ["text/css"];
|
||||||
|
}
|
||||||
|
cb({ cancel: false, responseHeaders });
|
||||||
|
});
|
||||||
|
|
||||||
|
// assign a noop to onHeadersReceived to prevent other mods from adding their own incompatible ones.
|
||||||
|
// For instance, OpenAsar adds their own that doesn't fix content-type for stylesheets which makes it
|
||||||
|
// impossible to load css from github raw despite our fix above
|
||||||
|
session.defaultSession.webRequest.onHeadersReceived = () => { };
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
if (IS_DISCORD_DESKTOP) {
|
||||||
|
require("./patcher");
|
||||||
|
}
|
@ -17,25 +17,60 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import "./updater";
|
import "./updater";
|
||||||
|
import "./ipcPlugins";
|
||||||
|
|
||||||
import { debounce } from "@utils/debounce";
|
import { debounce } from "@utils/debounce";
|
||||||
import IpcEvents from "@utils/IpcEvents";
|
import { IpcEvents } from "@utils/IpcEvents";
|
||||||
import { Queue } from "@utils/Queue";
|
import { Queue } from "@utils/Queue";
|
||||||
import { BrowserWindow, ipcMain, shell } from "electron";
|
import { BrowserWindow, ipcMain, shell } from "electron";
|
||||||
import { mkdirSync, readFileSync, watch } from "fs";
|
import { mkdirSync, readFileSync, watch } from "fs";
|
||||||
import { open, readFile, writeFile } from "fs/promises";
|
import { open, readdir, readFile, writeFile } from "fs/promises";
|
||||||
import { join } from "path";
|
import { join, normalize } from "path";
|
||||||
|
|
||||||
import monacoHtml from "~fileContent/../components/monacoWin.html;base64";
|
import monacoHtml from "~fileContent/../components/monacoWin.html;base64";
|
||||||
|
|
||||||
import { ALLOWED_PROTOCOLS, QUICKCSS_PATH, SETTINGS_DIR, SETTINGS_FILE } from "./constants";
|
import { getThemeInfo, stripBOM, UserThemeHeader } from "./themes";
|
||||||
|
import { ALLOWED_PROTOCOLS, QUICKCSS_PATH, SETTINGS_DIR, SETTINGS_FILE, THEMES_DIR } from "./utils/constants";
|
||||||
|
import { makeLinksOpenExternally } from "./utils/externalLinks";
|
||||||
|
|
||||||
mkdirSync(SETTINGS_DIR, { recursive: true });
|
mkdirSync(SETTINGS_DIR, { recursive: true });
|
||||||
|
mkdirSync(THEMES_DIR, { recursive: true });
|
||||||
|
|
||||||
|
export function ensureSafePath(basePath: string, path: string) {
|
||||||
|
const normalizedBasePath = normalize(basePath);
|
||||||
|
const newPath = join(basePath, path);
|
||||||
|
const normalizedPath = normalize(newPath);
|
||||||
|
return normalizedPath.startsWith(normalizedBasePath) ? normalizedPath : null;
|
||||||
|
}
|
||||||
|
|
||||||
function readCss() {
|
function readCss() {
|
||||||
return readFile(QUICKCSS_PATH, "utf-8").catch(() => "");
|
return readFile(QUICKCSS_PATH, "utf-8").catch(() => "");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async function listThemes(): Promise<UserThemeHeader[]> {
|
||||||
|
const files = await readdir(THEMES_DIR).catch(() => []);
|
||||||
|
|
||||||
|
const themeInfo: UserThemeHeader[] = [];
|
||||||
|
|
||||||
|
for (const fileName of files) {
|
||||||
|
if (!fileName.endsWith(".css")) continue;
|
||||||
|
|
||||||
|
const data = await getThemeData(fileName).then(stripBOM).catch(() => null);
|
||||||
|
if (data == null) continue;
|
||||||
|
|
||||||
|
themeInfo.push(getThemeInfo(data, fileName));
|
||||||
|
}
|
||||||
|
|
||||||
|
return themeInfo;
|
||||||
|
}
|
||||||
|
|
||||||
|
function getThemeData(fileName: string) {
|
||||||
|
fileName = fileName.replace(/\?v=\d+$/, "");
|
||||||
|
const safePath = ensureSafePath(THEMES_DIR, fileName);
|
||||||
|
if (!safePath) return Promise.reject(`Unsafe path ${fileName}`);
|
||||||
|
return readFile(safePath, "utf-8");
|
||||||
|
}
|
||||||
|
|
||||||
export function readSettings() {
|
export function readSettings() {
|
||||||
try {
|
try {
|
||||||
return readFileSync(SETTINGS_FILE, "utf-8");
|
return readFileSync(SETTINGS_FILE, "utf-8");
|
||||||
@ -44,6 +79,14 @@ export function readSettings() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function getSettings(): typeof import("@api/Settings").Settings {
|
||||||
|
try {
|
||||||
|
return JSON.parse(readSettings());
|
||||||
|
} catch {
|
||||||
|
return {} as any;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
ipcMain.handle(IpcEvents.OPEN_QUICKCSS, () => shell.openPath(QUICKCSS_PATH));
|
ipcMain.handle(IpcEvents.OPEN_QUICKCSS, () => shell.openPath(QUICKCSS_PATH));
|
||||||
|
|
||||||
ipcMain.handle(IpcEvents.OPEN_EXTERNAL, (_, url) => {
|
ipcMain.handle(IpcEvents.OPEN_EXTERNAL, (_, url) => {
|
||||||
@ -66,6 +109,10 @@ ipcMain.handle(IpcEvents.SET_QUICK_CSS, (_, css) =>
|
|||||||
cssWriteQueue.push(() => writeFile(QUICKCSS_PATH, css))
|
cssWriteQueue.push(() => writeFile(QUICKCSS_PATH, css))
|
||||||
);
|
);
|
||||||
|
|
||||||
|
ipcMain.handle(IpcEvents.GET_THEMES_DIR, () => THEMES_DIR);
|
||||||
|
ipcMain.handle(IpcEvents.GET_THEMES_LIST, () => listThemes());
|
||||||
|
ipcMain.handle(IpcEvents.GET_THEME_DATA, (_, fileName) => getThemeData(fileName));
|
||||||
|
|
||||||
ipcMain.handle(IpcEvents.GET_SETTINGS_DIR, () => SETTINGS_DIR);
|
ipcMain.handle(IpcEvents.GET_SETTINGS_DIR, () => SETTINGS_DIR);
|
||||||
ipcMain.on(IpcEvents.GET_SETTINGS, e => e.returnValue = readSettings());
|
ipcMain.on(IpcEvents.GET_SETTINGS, e => e.returnValue = readSettings());
|
||||||
|
|
||||||
@ -81,18 +128,26 @@ export function initIpc(mainWindow: BrowserWindow) {
|
|||||||
mainWindow.webContents.postMessage(IpcEvents.QUICK_CSS_UPDATE, await readCss());
|
mainWindow.webContents.postMessage(IpcEvents.QUICK_CSS_UPDATE, await readCss());
|
||||||
}, 50));
|
}, 50));
|
||||||
});
|
});
|
||||||
|
|
||||||
|
watch(THEMES_DIR, { persistent: false }, debounce(() => {
|
||||||
|
mainWindow.webContents.postMessage(IpcEvents.THEME_UPDATE, void 0);
|
||||||
|
}));
|
||||||
}
|
}
|
||||||
|
|
||||||
ipcMain.handle(IpcEvents.OPEN_MONACO_EDITOR, async () => {
|
ipcMain.handle(IpcEvents.OPEN_MONACO_EDITOR, async () => {
|
||||||
const win = new BrowserWindow({
|
const win = new BrowserWindow({
|
||||||
title: "QuickCss Editor",
|
title: "Vencord QuickCSS Editor",
|
||||||
autoHideMenuBar: true,
|
autoHideMenuBar: true,
|
||||||
darkTheme: true,
|
darkTheme: true,
|
||||||
webPreferences: {
|
webPreferences: {
|
||||||
preload: join(__dirname, "preload.js"),
|
preload: join(__dirname, IS_DISCORD_DESKTOP ? "preload.js" : "vencordDesktopPreload.js"),
|
||||||
contextIsolation: true,
|
contextIsolation: true,
|
||||||
nodeIntegration: false
|
nodeIntegration: false,
|
||||||
|
sandbox: false
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
makeLinksOpenExternally(win);
|
||||||
|
|
||||||
await win.loadURL(`data:text/html;base64,${monacoHtml}`);
|
await win.loadURL(`data:text/html;base64,${monacoHtml}`);
|
||||||
});
|
});
|
89
src/main/ipcPlugins.ts
Normal file
89
src/main/ipcPlugins.ts
Normal file
@ -0,0 +1,89 @@
|
|||||||
|
/*
|
||||||
|
* Vencord, a modification for Discord's desktop app
|
||||||
|
* Copyright (c) 2023 Vendicated and contributors
|
||||||
|
*
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { IpcEvents } from "@utils/IpcEvents";
|
||||||
|
import { app, ipcMain } from "electron";
|
||||||
|
import { readFile } from "fs/promises";
|
||||||
|
import { request } from "https";
|
||||||
|
import { basename, normalize } from "path";
|
||||||
|
|
||||||
|
import { getSettings } from "./ipcMain";
|
||||||
|
|
||||||
|
// FixSpotifyEmbeds
|
||||||
|
app.on("browser-window-created", (_, win) => {
|
||||||
|
win.webContents.on("frame-created", (_, { frame }) => {
|
||||||
|
frame.once("dom-ready", () => {
|
||||||
|
if (frame.url.startsWith("https://open.spotify.com/embed/")) {
|
||||||
|
const settings = getSettings().plugins?.FixSpotifyEmbeds;
|
||||||
|
if (!settings?.enabled) return;
|
||||||
|
|
||||||
|
frame.executeJavaScript(`
|
||||||
|
const original = Audio.prototype.play;
|
||||||
|
Audio.prototype.play = function() {
|
||||||
|
this.volume = ${(settings.volume / 100) || 0.1};
|
||||||
|
return original.apply(this, arguments);
|
||||||
|
}
|
||||||
|
`);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
// #region OpenInApp
|
||||||
|
// These links don't support CORS, so this has to be native
|
||||||
|
const validRedirectUrls = /^https:\/\/(spotify\.link|s\.team)\/.+$/;
|
||||||
|
|
||||||
|
function getRedirect(url: string) {
|
||||||
|
return new Promise<string>((resolve, reject) => {
|
||||||
|
const req = request(new URL(url), { method: "HEAD" }, res => {
|
||||||
|
resolve(
|
||||||
|
res.headers.location
|
||||||
|
? getRedirect(res.headers.location)
|
||||||
|
: url
|
||||||
|
);
|
||||||
|
});
|
||||||
|
req.on("error", reject);
|
||||||
|
req.end();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
ipcMain.handle(IpcEvents.OPEN_IN_APP__RESOLVE_REDIRECT, async (_, url: string) => {
|
||||||
|
if (!validRedirectUrls.test(url)) return url;
|
||||||
|
|
||||||
|
return getRedirect(url);
|
||||||
|
});
|
||||||
|
// #endregion
|
||||||
|
|
||||||
|
|
||||||
|
// #region VoiceMessages
|
||||||
|
ipcMain.handle(IpcEvents.VOICE_MESSAGES_READ_RECORDING, async (_, filePath: string) => {
|
||||||
|
filePath = normalize(filePath);
|
||||||
|
const filename = basename(filePath);
|
||||||
|
const discordBaseDirWithTrailingSlash = normalize(app.getPath("userData") + "/");
|
||||||
|
console.log(filename, discordBaseDirWithTrailingSlash, filePath);
|
||||||
|
if (filename !== "recording.ogg" || !filePath.startsWith(discordBaseDirWithTrailingSlash)) return null;
|
||||||
|
|
||||||
|
try {
|
||||||
|
const buf = await readFile(filePath);
|
||||||
|
return new Uint8Array(buf.buffer);
|
||||||
|
} catch {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// #endregion
|
@ -16,22 +16,10 @@
|
|||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import { app, autoUpdater } from "electron";
|
import { app } from "electron";
|
||||||
import { existsSync, mkdirSync, readdirSync, renameSync, statSync, writeFileSync } from "fs";
|
import { existsSync, mkdirSync, readdirSync, renameSync, statSync, writeFileSync } from "fs";
|
||||||
import { basename, dirname, join } from "path";
|
import { basename, dirname, join } from "path";
|
||||||
|
|
||||||
const { setAppUserModelId } = app;
|
|
||||||
|
|
||||||
// Apparently requiring Discords updater too early leads into issues,
|
|
||||||
// copied this workaround from powerCord
|
|
||||||
app.setAppUserModelId = function (id: string) {
|
|
||||||
app.setAppUserModelId = setAppUserModelId;
|
|
||||||
|
|
||||||
setAppUserModelId.call(this, id);
|
|
||||||
|
|
||||||
patchUpdater();
|
|
||||||
};
|
|
||||||
|
|
||||||
function isNewer($new: string, old: string) {
|
function isNewer($new: string, old: string) {
|
||||||
const newParts = $new.slice(4).split(".").map(Number);
|
const newParts = $new.slice(4).split(".").map(Number);
|
||||||
const oldParts = old.slice(4).split(".").map(Number);
|
const oldParts = old.slice(4).split(".").map(Number);
|
||||||
@ -77,23 +65,6 @@ function patchLatest() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Windows Host Updates install to a new folder app-{HOST_VERSION}, so we
|
// Try to patch latest on before-quit
|
||||||
// need to reinject
|
// Discord's Win32 updater will call app.quit() on restart and open new version on will-quit
|
||||||
function patchUpdater() {
|
app.on("before-quit", patchLatest);
|
||||||
try {
|
|
||||||
const autoStartScript = join(require.main!.filename, "..", "autoStart", "win32.js");
|
|
||||||
const { update } = require(autoStartScript);
|
|
||||||
|
|
||||||
require.cache[autoStartScript]!.exports.update = function () {
|
|
||||||
update.apply(this, arguments);
|
|
||||||
patchLatest();
|
|
||||||
};
|
|
||||||
} catch {
|
|
||||||
// OpenAsar uses electrons autoUpdater on Windows
|
|
||||||
const { quitAndInstall } = autoUpdater;
|
|
||||||
autoUpdater.quitAndInstall = function () {
|
|
||||||
patchLatest();
|
|
||||||
quitAndInstall.call(this);
|
|
||||||
};
|
|
||||||
}
|
|
||||||
}
|
|
@ -20,9 +20,8 @@ import { onceDefined } from "@utils/onceDefined";
|
|||||||
import electron, { app, BrowserWindowConstructorOptions, Menu } from "electron";
|
import electron, { app, BrowserWindowConstructorOptions, Menu } from "electron";
|
||||||
import { dirname, join } from "path";
|
import { dirname, join } from "path";
|
||||||
|
|
||||||
import { initIpc } from "./ipcMain";
|
import { getSettings, initIpc } from "./ipcMain";
|
||||||
import { installExt } from "./ipcMain/extensions";
|
import { IS_VANILLA } from "./utils/constants";
|
||||||
import { readSettings } from "./ipcMain/index";
|
|
||||||
|
|
||||||
console.log("[Vencord] Starting up...");
|
console.log("[Vencord] Starting up...");
|
||||||
|
|
||||||
@ -41,11 +40,8 @@ require.main!.filename = join(asarPath, discordPkg.main);
|
|||||||
// @ts-ignore Untyped method? Dies from cringe
|
// @ts-ignore Untyped method? Dies from cringe
|
||||||
app.setAppPath(asarPath);
|
app.setAppPath(asarPath);
|
||||||
|
|
||||||
if (!process.argv.includes("--vanilla")) {
|
if (!IS_VANILLA) {
|
||||||
let settings: typeof import("@api/settings").Settings = {} as any;
|
const settings = getSettings();
|
||||||
try {
|
|
||||||
settings = JSON.parse(readSettings());
|
|
||||||
} catch { }
|
|
||||||
|
|
||||||
// Repatch after host updates on Windows
|
// Repatch after host updates on Windows
|
||||||
if (process.platform === "win32") {
|
if (process.platform === "win32") {
|
||||||
@ -75,16 +71,25 @@ if (!process.argv.includes("--vanilla")) {
|
|||||||
constructor(options: BrowserWindowConstructorOptions) {
|
constructor(options: BrowserWindowConstructorOptions) {
|
||||||
if (options?.webPreferences?.preload && options.title) {
|
if (options?.webPreferences?.preload && options.title) {
|
||||||
const original = options.webPreferences.preload;
|
const original = options.webPreferences.preload;
|
||||||
options.webPreferences.preload = join(__dirname, "preload.js");
|
options.webPreferences.preload = join(__dirname, IS_DISCORD_DESKTOP ? "preload.js" : "vencordDesktopPreload.js");
|
||||||
options.webPreferences.sandbox = false;
|
options.webPreferences.sandbox = false;
|
||||||
if (settings.frameless) {
|
if (settings.frameless) {
|
||||||
options.frame = false;
|
options.frame = false;
|
||||||
|
} else if (process.platform === "win32" && settings.winNativeTitleBar) {
|
||||||
|
delete options.frame;
|
||||||
}
|
}
|
||||||
if (settings.transparent) {
|
|
||||||
|
// This causes electron to freeze / white screen for some people
|
||||||
|
if ((settings as any).transparentUNSAFE_USE_AT_OWN_RISK) {
|
||||||
options.transparent = true;
|
options.transparent = true;
|
||||||
options.backgroundColor = "#00000000";
|
options.backgroundColor = "#00000000";
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (settings.macosTranslucency && process.platform === "darwin") {
|
||||||
|
options.backgroundColor = "#00000000";
|
||||||
|
options.vibrancy = "sidebar";
|
||||||
|
}
|
||||||
|
|
||||||
process.env.DISCORD_PRELOAD = original;
|
process.env.DISCORD_PRELOAD = original;
|
||||||
|
|
||||||
super(options);
|
super(options);
|
||||||
@ -106,85 +111,19 @@ if (!process.argv.includes("--vanilla")) {
|
|||||||
BrowserWindow
|
BrowserWindow
|
||||||
};
|
};
|
||||||
|
|
||||||
// Patch appSettings to force enable devtools
|
// Patch appSettings to force enable devtools and optionally disable min size
|
||||||
onceDefined(global, "appSettings", s =>
|
onceDefined(global, "appSettings", s => {
|
||||||
s.set("DANGEROUS_ENABLE_DEVTOOLS_ONLY_ENABLE_IF_YOU_KNOW_WHAT_YOURE_DOING", true)
|
s.set("DANGEROUS_ENABLE_DEVTOOLS_ONLY_ENABLE_IF_YOU_KNOW_WHAT_YOURE_DOING", true);
|
||||||
);
|
if (settings.disableMinSize) {
|
||||||
|
s.set("MIN_WIDTH", 0);
|
||||||
|
s.set("MIN_HEIGHT", 0);
|
||||||
|
} else {
|
||||||
|
s.set("MIN_WIDTH", 940);
|
||||||
|
s.set("MIN_HEIGHT", 500);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
process.env.DATA_DIR = join(app.getPath("userData"), "..", "Vencord");
|
process.env.DATA_DIR = join(app.getPath("userData"), "..", "Vencord");
|
||||||
|
|
||||||
electron.app.whenReady().then(() => {
|
|
||||||
// Source Maps! Maybe there's a better way but since the renderer is executed
|
|
||||||
// from a string I don't think any other form of sourcemaps would work
|
|
||||||
electron.protocol.registerFileProtocol("vencord", ({ url: unsafeUrl }, cb) => {
|
|
||||||
let url = unsafeUrl.slice("vencord://".length);
|
|
||||||
if (url.endsWith("/")) url = url.slice(0, -1);
|
|
||||||
switch (url) {
|
|
||||||
case "renderer.js.map":
|
|
||||||
case "preload.js.map":
|
|
||||||
case "patcher.js.map": // doubt
|
|
||||||
cb(join(__dirname, url));
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
cb({ statusCode: 403 });
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
try {
|
|
||||||
if (settings?.enableReactDevtools)
|
|
||||||
installExt("fmkadmapgofadopljbjfkapdkoienihi")
|
|
||||||
.then(() => console.info("[Vencord] Installed React Developer Tools"))
|
|
||||||
.catch(err => console.error("[Vencord] Failed to install React Developer Tools", err));
|
|
||||||
} catch { }
|
|
||||||
|
|
||||||
|
|
||||||
// Remove CSP
|
|
||||||
type PolicyResult = Record<string, string[]>;
|
|
||||||
|
|
||||||
const parsePolicy = (policy: string): PolicyResult => {
|
|
||||||
const result: PolicyResult = {};
|
|
||||||
policy.split(";").forEach(directive => {
|
|
||||||
const [directiveKey, ...directiveValue] = directive.trim().split(/\s+/g);
|
|
||||||
if (directiveKey && !Object.prototype.hasOwnProperty.call(result, directiveKey)) {
|
|
||||||
result[directiveKey] = directiveValue;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
return result;
|
|
||||||
};
|
|
||||||
const stringifyPolicy = (policy: PolicyResult): string =>
|
|
||||||
Object.entries(policy)
|
|
||||||
.filter(([, values]) => values?.length)
|
|
||||||
.map(directive => directive.flat().join(" "))
|
|
||||||
.join("; ");
|
|
||||||
|
|
||||||
function patchCsp(headers: Record<string, string[]>, header: string) {
|
|
||||||
if (header in headers) {
|
|
||||||
const csp = parsePolicy(headers[header][0]);
|
|
||||||
|
|
||||||
for (const directive of ["style-src", "connect-src", "img-src", "font-src", "media-src", "worker-src"]) {
|
|
||||||
csp[directive] = ["*", "blob:", "data:", "'unsafe-inline'"];
|
|
||||||
}
|
|
||||||
// TODO: Restrict this to only imported packages with fixed version.
|
|
||||||
// Perhaps auto generate with esbuild
|
|
||||||
csp["script-src"] ??= [];
|
|
||||||
csp["script-src"].push("'unsafe-eval'", "https://unpkg.com", "https://cdnjs.cloudflare.com");
|
|
||||||
headers[header] = [stringifyPolicy(csp)];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
electron.session.defaultSession.webRequest.onHeadersReceived(({ responseHeaders, resourceType }, cb) => {
|
|
||||||
if (responseHeaders) {
|
|
||||||
if (resourceType === "mainFrame")
|
|
||||||
patchCsp(responseHeaders, "content-security-policy");
|
|
||||||
|
|
||||||
// Fix hosts that don't properly set the css content type, such as
|
|
||||||
// raw.githubusercontent.com
|
|
||||||
if (resourceType === "stylesheet")
|
|
||||||
responseHeaders["content-type"] = ["text/css"];
|
|
||||||
}
|
|
||||||
cb({ cancel: false, responseHeaders });
|
|
||||||
});
|
|
||||||
});
|
|
||||||
} else {
|
} else {
|
||||||
console.log("[Vencord] Running in vanilla mode. Not loading Vencord");
|
console.log("[Vencord] Running in vanilla mode. Not loading Vencord");
|
||||||
}
|
}
|
177
src/main/themes/LICENSE
Normal file
177
src/main/themes/LICENSE
Normal file
@ -0,0 +1,177 @@
|
|||||||
|
|
||||||
|
Apache License
|
||||||
|
Version 2.0, January 2004
|
||||||
|
http://www.apache.org/licenses/
|
||||||
|
|
||||||
|
TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION
|
||||||
|
|
||||||
|
1. Definitions.
|
||||||
|
|
||||||
|
"License" shall mean the terms and conditions for use, reproduction,
|
||||||
|
and distribution as defined by Sections 1 through 9 of this document.
|
||||||
|
|
||||||
|
"Licensor" shall mean the copyright owner or entity authorized by
|
||||||
|
the copyright owner that is granting the License.
|
||||||
|
|
||||||
|
"Legal Entity" shall mean the union of the acting entity and all
|
||||||
|
other entities that control, are controlled by, or are under common
|
||||||
|
control with that entity. For the purposes of this definition,
|
||||||
|
"control" means (i) the power, direct or indirect, to cause the
|
||||||
|
direction or management of such entity, whether by contract or
|
||||||
|
otherwise, or (ii) ownership of fifty percent (50%) or more of the
|
||||||
|
outstanding shares, or (iii) beneficial ownership of such entity.
|
||||||
|
|
||||||
|
"You" (or "Your") shall mean an individual or Legal Entity
|
||||||
|
exercising permissions granted by this License.
|
||||||
|
|
||||||
|
"Source" form shall mean the preferred form for making modifications,
|
||||||
|
including but not limited to software source code, documentation
|
||||||
|
source, and configuration files.
|
||||||
|
|
||||||
|
"Object" form shall mean any form resulting from mechanical
|
||||||
|
transformation or translation of a Source form, including but
|
||||||
|
not limited to compiled object code, generated documentation,
|
||||||
|
and conversions to other media types.
|
||||||
|
|
||||||
|
"Work" shall mean the work of authorship, whether in Source or
|
||||||
|
Object form, made available under the License, as indicated by a
|
||||||
|
copyright notice that is included in or attached to the work
|
||||||
|
(an example is provided in the Appendix below).
|
||||||
|
|
||||||
|
"Derivative Works" shall mean any work, whether in Source or Object
|
||||||
|
form, that is based on (or derived from) the Work and for which the
|
||||||
|
editorial revisions, annotations, elaborations, or other modifications
|
||||||
|
represent, as a whole, an original work of authorship. For the purposes
|
||||||
|
of this License, Derivative Works shall not include works that remain
|
||||||
|
separable from, or merely link (or bind by name) to the interfaces of,
|
||||||
|
the Work and Derivative Works thereof.
|
||||||
|
|
||||||
|
"Contribution" shall mean any work of authorship, including
|
||||||
|
the original version of the Work and any modifications or additions
|
||||||
|
to that Work or Derivative Works thereof, that is intentionally
|
||||||
|
submitted to Licensor for inclusion in the Work by the copyright owner
|
||||||
|
or by an individual or Legal Entity authorized to submit on behalf of
|
||||||
|
the copyright owner. For the purposes of this definition, "submitted"
|
||||||
|
means any form of electronic, verbal, or written communication sent
|
||||||
|
to the Licensor or its representatives, including but not limited to
|
||||||
|
communication on electronic mailing lists, source code control systems,
|
||||||
|
and issue tracking systems that are managed by, or on behalf of, the
|
||||||
|
Licensor for the purpose of discussing and improving the Work, but
|
||||||
|
excluding communication that is conspicuously marked or otherwise
|
||||||
|
designated in writing by the copyright owner as "Not a Contribution."
|
||||||
|
|
||||||
|
"Contributor" shall mean Licensor and any individual or Legal Entity
|
||||||
|
on behalf of whom a Contribution has been received by Licensor and
|
||||||
|
subsequently incorporated within the Work.
|
||||||
|
|
||||||
|
2. Grant of Copyright License. Subject to the terms and conditions of
|
||||||
|
this License, each Contributor hereby grants to You a perpetual,
|
||||||
|
worldwide, non-exclusive, no-charge, royalty-free, irrevocable
|
||||||
|
copyright license to reproduce, prepare Derivative Works of,
|
||||||
|
publicly display, publicly perform, sublicense, and distribute the
|
||||||
|
Work and such Derivative Works in Source or Object form.
|
||||||
|
|
||||||
|
3. Grant of Patent License. Subject to the terms and conditions of
|
||||||
|
this License, each Contributor hereby grants to You a perpetual,
|
||||||
|
worldwide, non-exclusive, no-charge, royalty-free, irrevocable
|
||||||
|
(except as stated in this section) patent license to make, have made,
|
||||||
|
use, offer to sell, sell, import, and otherwise transfer the Work,
|
||||||
|
where such license applies only to those patent claims licensable
|
||||||
|
by such Contributor that are necessarily infringed by their
|
||||||
|
Contribution(s) alone or by combination of their Contribution(s)
|
||||||
|
with the Work to which such Contribution(s) was submitted. If You
|
||||||
|
institute patent litigation against any entity (including a
|
||||||
|
cross-claim or counterclaim in a lawsuit) alleging that the Work
|
||||||
|
or a Contribution incorporated within the Work constitutes direct
|
||||||
|
or contributory patent infringement, then any patent licenses
|
||||||
|
granted to You under this License for that Work shall terminate
|
||||||
|
as of the date such litigation is filed.
|
||||||
|
|
||||||
|
4. Redistribution. You may reproduce and distribute copies of the
|
||||||
|
Work or Derivative Works thereof in any medium, with or without
|
||||||
|
modifications, and in Source or Object form, provided that You
|
||||||
|
meet the following conditions:
|
||||||
|
|
||||||
|
(a) You must give any other recipients of the Work or
|
||||||
|
Derivative Works a copy of this License; and
|
||||||
|
|
||||||
|
(b) You must cause any modified files to carry prominent notices
|
||||||
|
stating that You changed the files; and
|
||||||
|
|
||||||
|
(c) You must retain, in the Source form of any Derivative Works
|
||||||
|
that You distribute, all copyright, patent, trademark, and
|
||||||
|
attribution notices from the Source form of the Work,
|
||||||
|
excluding those notices that do not pertain to any part of
|
||||||
|
the Derivative Works; and
|
||||||
|
|
||||||
|
(d) If the Work includes a "NOTICE" text file as part of its
|
||||||
|
distribution, then any Derivative Works that You distribute must
|
||||||
|
include a readable copy of the attribution notices contained
|
||||||
|
within such NOTICE file, excluding those notices that do not
|
||||||
|
pertain to any part of the Derivative Works, in at least one
|
||||||
|
of the following places: within a NOTICE text file distributed
|
||||||
|
as part of the Derivative Works; within the Source form or
|
||||||
|
documentation, if provided along with the Derivative Works; or,
|
||||||
|
within a display generated by the Derivative Works, if and
|
||||||
|
wherever such third-party notices normally appear. The contents
|
||||||
|
of the NOTICE file are for informational purposes only and
|
||||||
|
do not modify the License. You may add Your own attribution
|
||||||
|
notices within Derivative Works that You distribute, alongside
|
||||||
|
or as an addendum to the NOTICE text from the Work, provided
|
||||||
|
that such additional attribution notices cannot be construed
|
||||||
|
as modifying the License.
|
||||||
|
|
||||||
|
You may add Your own copyright statement to Your modifications and
|
||||||
|
may provide additional or different license terms and conditions
|
||||||
|
for use, reproduction, or distribution of Your modifications, or
|
||||||
|
for any such Derivative Works as a whole, provided Your use,
|
||||||
|
reproduction, and distribution of the Work otherwise complies with
|
||||||
|
the conditions stated in this License.
|
||||||
|
|
||||||
|
5. Submission of Contributions. Unless You explicitly state otherwise,
|
||||||
|
any Contribution intentionally submitted for inclusion in the Work
|
||||||
|
by You to the Licensor shall be under the terms and conditions of
|
||||||
|
this License, without any additional terms or conditions.
|
||||||
|
Notwithstanding the above, nothing herein shall supersede or modify
|
||||||
|
the terms of any separate license agreement you may have executed
|
||||||
|
with Licensor regarding such Contributions.
|
||||||
|
|
||||||
|
6. Trademarks. This License does not grant permission to use the trade
|
||||||
|
names, trademarks, service marks, or product names of the Licensor,
|
||||||
|
except as required for reasonable and customary use in describing the
|
||||||
|
origin of the Work and reproducing the content of the NOTICE file.
|
||||||
|
|
||||||
|
7. Disclaimer of Warranty. Unless required by applicable law or
|
||||||
|
agreed to in writing, Licensor provides the Work (and each
|
||||||
|
Contributor provides its Contributions) on an "AS IS" BASIS,
|
||||||
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
|
||||||
|
implied, including, without limitation, any warranties or conditions
|
||||||
|
of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A
|
||||||
|
PARTICULAR PURPOSE. You are solely responsible for determining the
|
||||||
|
appropriateness of using or redistributing the Work and assume any
|
||||||
|
risks associated with Your exercise of permissions under this License.
|
||||||
|
|
||||||
|
8. Limitation of Liability. In no event and under no legal theory,
|
||||||
|
whether in tort (including negligence), contract, or otherwise,
|
||||||
|
unless required by applicable law (such as deliberate and grossly
|
||||||
|
negligent acts) or agreed to in writing, shall any Contributor be
|
||||||
|
liable to You for damages, including any direct, indirect, special,
|
||||||
|
incidental, or consequential damages of any character arising as a
|
||||||
|
result of this License or out of the use or inability to use the
|
||||||
|
Work (including but not limited to damages for loss of goodwill,
|
||||||
|
work stoppage, computer failure or malfunction, or any and all
|
||||||
|
other commercial damages or losses), even if such Contributor
|
||||||
|
has been advised of the possibility of such damages.
|
||||||
|
|
||||||
|
9. Accepting Warranty or Additional Liability. While redistributing
|
||||||
|
the Work or Derivative Works thereof, You may choose to offer,
|
||||||
|
and charge a fee for, acceptance of support, warranty, indemnity,
|
||||||
|
or other liability obligations and/or rights consistent with this
|
||||||
|
License. However, in accepting such obligations, You may act only
|
||||||
|
on Your own behalf and on Your sole responsibility, not on behalf
|
||||||
|
of any other Contributor, and only if You agree to indemnify,
|
||||||
|
defend, and hold each Contributor harmless for any liability
|
||||||
|
incurred by, or claims asserted against, such Contributor by reason
|
||||||
|
of your accepting any such warranty or additional liability.
|
||||||
|
|
||||||
|
END OF TERMS AND CONDITIONS
|
81
src/main/themes/index.ts
Normal file
81
src/main/themes/index.ts
Normal file
@ -0,0 +1,81 @@
|
|||||||
|
/* eslint-disable simple-header/header */
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* BetterDiscord addon meta parser
|
||||||
|
* Copyright 2023 BetterDiscord contributors
|
||||||
|
* Copyright 2023 Vendicated and Vencord contributors
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
const splitRegex = /[^\S\r\n]*?\r?(?:\r\n|\n)[^\S\r\n]*?\*[^\S\r\n]?/;
|
||||||
|
const escapedAtRegex = /^\\@/;
|
||||||
|
|
||||||
|
export interface UserThemeHeader {
|
||||||
|
fileName: string;
|
||||||
|
name: string;
|
||||||
|
author: string;
|
||||||
|
description: string;
|
||||||
|
version?: string;
|
||||||
|
license?: string;
|
||||||
|
source?: string;
|
||||||
|
website?: string;
|
||||||
|
invite?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
function makeHeader(fileName: string, opts: Partial<UserThemeHeader> = {}): UserThemeHeader {
|
||||||
|
return {
|
||||||
|
fileName,
|
||||||
|
name: opts.name ?? fileName.replace(/\.css$/i, ""),
|
||||||
|
author: opts.author ?? "Unknown Author",
|
||||||
|
description: opts.description ?? "A Discord Theme.",
|
||||||
|
version: opts.version,
|
||||||
|
license: opts.license,
|
||||||
|
source: opts.source,
|
||||||
|
website: opts.website,
|
||||||
|
invite: opts.invite
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export function stripBOM(fileContent: string) {
|
||||||
|
if (fileContent.charCodeAt(0) === 0xFEFF) {
|
||||||
|
fileContent = fileContent.slice(1);
|
||||||
|
}
|
||||||
|
return fileContent;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function getThemeInfo(css: string, fileName: string): UserThemeHeader {
|
||||||
|
if (!css) return makeHeader(fileName);
|
||||||
|
|
||||||
|
const block = css.split("/**", 2)?.[1]?.split("*/", 1)?.[0];
|
||||||
|
if (!block) return makeHeader(fileName);
|
||||||
|
|
||||||
|
const header: Partial<UserThemeHeader> = {};
|
||||||
|
let field = "";
|
||||||
|
let accum = "";
|
||||||
|
for (const line of block.split(splitRegex)) {
|
||||||
|
if (line.length === 0) continue;
|
||||||
|
if (line.charAt(0) === "@" && line.charAt(1) !== " ") {
|
||||||
|
header[field] = accum.trim();
|
||||||
|
const l = line.indexOf(" ");
|
||||||
|
field = line.substring(1, l);
|
||||||
|
accum = line.substring(l + 1);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
accum += " " + line.replace("\\n", "\n").replace(escapedAtRegex, "@");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
header[field] = accum.trim();
|
||||||
|
delete header[""];
|
||||||
|
return makeHeader(fileName, header);
|
||||||
|
}
|
@ -16,28 +16,12 @@
|
|||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import { createHash } from "crypto";
|
export const VENCORD_FILES = [
|
||||||
import { createReadStream } from "fs";
|
IS_DISCORD_DESKTOP ? "patcher.js" : "vencordDesktopMain.js",
|
||||||
import { join } from "path";
|
IS_DISCORD_DESKTOP ? "preload.js" : "vencordDesktopPreload.js",
|
||||||
|
IS_DISCORD_DESKTOP ? "renderer.js" : "vencordDesktopRenderer.js",
|
||||||
export async function calculateHashes() {
|
IS_DISCORD_DESKTOP ? "renderer.css" : "vencordDesktopRenderer.css",
|
||||||
const hashes = {} as Record<string, string>;
|
];
|
||||||
|
|
||||||
await Promise.all(
|
|
||||||
["patcher.js", "preload.js", "renderer.js", "renderer.css"].map(file => new Promise<void>(r => {
|
|
||||||
const fis = createReadStream(join(__dirname, file));
|
|
||||||
const hash = createHash("sha1", { encoding: "hex" });
|
|
||||||
fis.once("end", () => {
|
|
||||||
hash.end();
|
|
||||||
hashes[file] = hash.read();
|
|
||||||
r();
|
|
||||||
});
|
|
||||||
fis.pipe(hash);
|
|
||||||
}))
|
|
||||||
);
|
|
||||||
|
|
||||||
return hashes;
|
|
||||||
}
|
|
||||||
|
|
||||||
export function serializeErrors(func: (...args: any[]) => any) {
|
export function serializeErrors(func: (...args: any[]) => any) {
|
||||||
return async function () {
|
return async function () {
|
@ -16,13 +16,13 @@
|
|||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import IpcEvents from "@utils/IpcEvents";
|
import { IpcEvents } from "@utils/IpcEvents";
|
||||||
import { execFile as cpExecFile } from "child_process";
|
import { execFile as cpExecFile } from "child_process";
|
||||||
import { ipcMain } from "electron";
|
import { ipcMain } from "electron";
|
||||||
import { join } from "path";
|
import { join } from "path";
|
||||||
import { promisify } from "util";
|
import { promisify } from "util";
|
||||||
|
|
||||||
import { calculateHashes, serializeErrors } from "./common";
|
import { serializeErrors } from "./common";
|
||||||
|
|
||||||
const VENCORD_SRC_DIR = join(__dirname, "..");
|
const VENCORD_SRC_DIR = join(__dirname, "..");
|
||||||
|
|
||||||
@ -76,7 +76,6 @@ async function build() {
|
|||||||
return !res.stderr.includes("Build failed");
|
return !res.stderr.includes("Build failed");
|
||||||
}
|
}
|
||||||
|
|
||||||
ipcMain.handle(IpcEvents.GET_HASHES, serializeErrors(calculateHashes));
|
|
||||||
ipcMain.handle(IpcEvents.GET_REPO, serializeErrors(getRepo));
|
ipcMain.handle(IpcEvents.GET_REPO, serializeErrors(getRepo));
|
||||||
ipcMain.handle(IpcEvents.GET_UPDATES, serializeErrors(calculateGitChanges));
|
ipcMain.handle(IpcEvents.GET_UPDATES, serializeErrors(calculateGitChanges));
|
||||||
ipcMain.handle(IpcEvents.UPDATE, serializeErrors(pull));
|
ipcMain.handle(IpcEvents.UPDATE, serializeErrors(pull));
|
@ -17,7 +17,7 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import { VENCORD_USER_AGENT } from "@utils/constants";
|
import { VENCORD_USER_AGENT } from "@utils/constants";
|
||||||
import IpcEvents from "@utils/IpcEvents";
|
import { IpcEvents } from "@utils/IpcEvents";
|
||||||
import { ipcMain } from "electron";
|
import { ipcMain } from "electron";
|
||||||
import { writeFile } from "fs/promises";
|
import { writeFile } from "fs/promises";
|
||||||
import { join } from "path";
|
import { join } from "path";
|
||||||
@ -25,8 +25,8 @@ import { join } from "path";
|
|||||||
import gitHash from "~git-hash";
|
import gitHash from "~git-hash";
|
||||||
import gitRemote from "~git-remote";
|
import gitRemote from "~git-remote";
|
||||||
|
|
||||||
import { get } from "../simpleGet";
|
import { get } from "../utils/simpleGet";
|
||||||
import { calculateHashes, serializeErrors } from "./common";
|
import { serializeErrors, VENCORD_FILES } from "./common";
|
||||||
|
|
||||||
const API_BASE = `https://api.github.com/repos/${gitRemote}`;
|
const API_BASE = `https://api.github.com/repos/${gitRemote}`;
|
||||||
let PendingUpdates = [] as [string, string][];
|
let PendingUpdates = [] as [string, string][];
|
||||||
@ -66,7 +66,7 @@ async function fetchUpdates() {
|
|||||||
return false;
|
return false;
|
||||||
|
|
||||||
data.assets.forEach(({ name, browser_download_url }) => {
|
data.assets.forEach(({ name, browser_download_url }) => {
|
||||||
if (["patcher.js", "preload.js", "renderer.js", "renderer.css"].some(s => name.startsWith(s))) {
|
if (VENCORD_FILES.some(s => name.startsWith(s))) {
|
||||||
PendingUpdates.push([name, browser_download_url]);
|
PendingUpdates.push([name, browser_download_url]);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
@ -75,13 +75,15 @@ async function fetchUpdates() {
|
|||||||
|
|
||||||
async function applyUpdates() {
|
async function applyUpdates() {
|
||||||
await Promise.all(PendingUpdates.map(
|
await Promise.all(PendingUpdates.map(
|
||||||
async ([name, data]) => writeFile(join(__dirname, name), await get(data)))
|
async ([name, data]) => writeFile(
|
||||||
);
|
join(__dirname, name),
|
||||||
|
await get(data)
|
||||||
|
)
|
||||||
|
));
|
||||||
PendingUpdates = [];
|
PendingUpdates = [];
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
ipcMain.handle(IpcEvents.GET_HASHES, serializeErrors(calculateHashes));
|
|
||||||
ipcMain.handle(IpcEvents.GET_REPO, serializeErrors(() => `https://github.com/${gitRemote}`));
|
ipcMain.handle(IpcEvents.GET_REPO, serializeErrors(() => `https://github.com/${gitRemote}`));
|
||||||
ipcMain.handle(IpcEvents.GET_UPDATES, serializeErrors(calculateGitChanges));
|
ipcMain.handle(IpcEvents.GET_UPDATES, serializeErrors(calculateGitChanges));
|
||||||
ipcMain.handle(IpcEvents.UPDATE, serializeErrors(fetchUpdates));
|
ipcMain.handle(IpcEvents.UPDATE, serializeErrors(fetchUpdates));
|
@ -16,4 +16,5 @@
|
|||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import(IS_STANDALONE ? "./http" : "./git");
|
if (!IS_UPDATER_DISABLED)
|
||||||
|
import(IS_STANDALONE ? "./http" : "./git");
|
@ -25,11 +25,15 @@ export const DATA_DIR = process.env.VENCORD_USER_DATA_DIR ?? (
|
|||||||
: join(app.getPath("userData"), "..", "Vencord")
|
: join(app.getPath("userData"), "..", "Vencord")
|
||||||
);
|
);
|
||||||
export const SETTINGS_DIR = join(DATA_DIR, "settings");
|
export const SETTINGS_DIR = join(DATA_DIR, "settings");
|
||||||
|
export const THEMES_DIR = join(DATA_DIR, "themes");
|
||||||
export const QUICKCSS_PATH = join(SETTINGS_DIR, "quickCss.css");
|
export const QUICKCSS_PATH = join(SETTINGS_DIR, "quickCss.css");
|
||||||
export const SETTINGS_FILE = join(SETTINGS_DIR, "settings.json");
|
export const SETTINGS_FILE = join(SETTINGS_DIR, "settings.json");
|
||||||
export const ALLOWED_PROTOCOLS = [
|
export const ALLOWED_PROTOCOLS = [
|
||||||
"https:",
|
"https:",
|
||||||
"http:",
|
"http:",
|
||||||
"steam:",
|
"steam:",
|
||||||
"spotify:"
|
"spotify:",
|
||||||
|
"com.epicgames.launcher:",
|
||||||
];
|
];
|
||||||
|
|
||||||
|
export const IS_VANILLA = /* @__PURE__ */ process.argv.includes("--vanilla");
|
@ -1,4 +1,4 @@
|
|||||||
/* eslint-disable header/header */
|
/* eslint-disable simple-header/header */
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* crxToZip
|
* crxToZip
|
48
src/main/utils/externalLinks.ts
Normal file
48
src/main/utils/externalLinks.ts
Normal file
@ -0,0 +1,48 @@
|
|||||||
|
/*
|
||||||
|
* Vencord, a modification for Discord's desktop app
|
||||||
|
* Copyright (c) 2023 Vendicated and contributors
|
||||||
|
*
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { type BrowserWindow, shell } from "electron";
|
||||||
|
|
||||||
|
export function makeLinksOpenExternally(win: BrowserWindow) {
|
||||||
|
win.webContents.setWindowOpenHandler(({ url }) => {
|
||||||
|
switch (url) {
|
||||||
|
case "about:blank":
|
||||||
|
case "https://discord.com/popout":
|
||||||
|
case "https://ptb.discord.com/popout":
|
||||||
|
case "https://canary.discord.com/popout":
|
||||||
|
return { action: "allow" };
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
var { protocol } = new URL(url);
|
||||||
|
} catch {
|
||||||
|
return { action: "deny" };
|
||||||
|
}
|
||||||
|
|
||||||
|
switch (protocol) {
|
||||||
|
case "http:":
|
||||||
|
case "https:":
|
||||||
|
case "mailto:":
|
||||||
|
case "steam:":
|
||||||
|
case "spotify:":
|
||||||
|
shell.openExternal(url);
|
||||||
|
}
|
||||||
|
|
||||||
|
return { action: "deny" };
|
||||||
|
});
|
||||||
|
}
|
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user