Compare commits
707 commits
2025-05-22
...
master
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
b17d879da8 | ||
|
|
6be9cec6e2 | ||
|
|
6d0a423dcf | ||
|
|
f72c82d0f9 | ||
|
|
da4ba222c0 | ||
|
|
cbfd286908 | ||
|
|
487bb84b6f | ||
|
|
9e03f82616 | ||
|
|
e674a39b87 | ||
|
|
1efc382c05 | ||
|
|
dc152e89f7 | ||
|
|
20cdcdb382 | ||
|
|
23da49ee5b | ||
|
|
c14a008080 | ||
|
|
0da2ac4087 | ||
|
|
29cc622ce3 | ||
|
|
86256602ff | ||
|
|
f37c418865 | ||
|
|
46d3b820db | ||
|
|
e0cbb7f06c | ||
|
|
f52dc6dda8 | ||
|
|
3fa377a11c | ||
|
|
c5372a9e92 | ||
|
|
6de55e9096 | ||
|
|
43c3bf5966 | ||
|
|
c4f4cece01 | ||
|
|
0d7047a728 | ||
|
|
7f30728f87 | ||
|
|
1d5d3f2d38 | ||
|
|
b3c89167c5 | ||
|
|
90ab663212 | ||
|
|
98c00c55ed | ||
|
|
d2164c3f08 | ||
|
|
8004d4f2d4 | ||
|
|
8751f0605d | ||
|
|
09d817770e | ||
|
|
81a2712b92 | ||
|
|
8dca14933c | ||
|
|
74102aa1ec | ||
|
|
a9003be30f | ||
|
|
6faa0d54e3 | ||
|
|
33e0f8699b | ||
|
|
03d54265fe | ||
|
|
491d1c9187 | ||
|
|
bddf9bd818 | ||
|
|
0549892092 | ||
|
|
10b9a65f17 | ||
|
|
5219cffa6b | ||
|
|
71790d8e10 | ||
|
|
fe31a49f86 | ||
|
|
55c84ca860 | ||
|
|
40b947c1e7 | ||
|
|
40cef0e436 | ||
|
|
9f1c225b7a | ||
|
|
af2f888293 | ||
|
|
7a5b2e9f0e | ||
|
|
021a9f8383 | ||
|
|
cba9ce2b2b | ||
|
|
bb1a5b33a1 | ||
|
|
6ac340026f | ||
|
|
059eeebe89 | ||
|
|
117ea543c5 | ||
|
|
989a5be23b | ||
|
|
f8ce5c2e39 | ||
|
|
20cd7ce73d | ||
|
|
aadee34238 | ||
|
|
7153f7d4c1 | ||
|
|
762e742be0 | ||
|
|
67f6ab66f0 | ||
|
|
7507103bb2 | ||
|
|
fe12f4cbb9 | ||
|
|
d18f3bce47 | ||
|
|
b66743c83c | ||
|
|
1a62f82aee | ||
|
|
5735a44a9a | ||
|
|
dbaf5f2e05 | ||
|
|
9c53dad4b8 | ||
|
|
7814204fe2 | ||
|
|
cdb171f201 | ||
|
|
48e21aad38 | ||
|
|
f223ff387e | ||
|
|
8845a75627 | ||
|
|
caf2bb9ded | ||
|
|
985936a917 | ||
|
|
2c51054e77 | ||
|
|
f7eeaeddcb | ||
|
|
6cace2a8e6 | ||
|
|
6b5f341e10 | ||
|
|
63143f9416 | ||
|
|
efe52b5412 | ||
|
|
0e014c0e5c | ||
|
|
511ccae738 | ||
|
|
0672603755 | ||
|
|
f5f326f65b | ||
|
|
c5702cc8b6 | ||
|
|
4f2f942121 | ||
|
|
a4c2b1411f | ||
|
|
43bee2316e | ||
|
|
19dbb17fb9 | ||
|
|
7c9fbe2be0 | ||
|
|
d30690236a | ||
|
|
ac2e995f15 | ||
|
|
dac611f0f1 | ||
|
|
45ab2602c6 | ||
|
|
5101cc3d74 | ||
|
|
9ac9a0c73a | ||
|
|
314a577807 | ||
|
|
c5ace60f26 | ||
|
|
8953ae3c67 | ||
|
|
b1fe4c85d3 | ||
|
|
a20f3c0fb4 | ||
|
|
9226bc9ddd | ||
|
|
501c4b96d4 | ||
|
|
6ab947418c | ||
|
|
6765831b92 | ||
|
|
98c4e829f8 | ||
|
|
ccb9901b28 | ||
|
|
58a8c7d3df | ||
|
|
655a8e52a1 | ||
|
|
db235ecae8 | ||
|
|
682ac4ed0c | ||
|
|
77978c7178 | ||
|
|
e918856fa4 | ||
|
|
87c4216e80 | ||
|
|
832f70496a | ||
|
|
f97864b72b | ||
|
|
338c56678a | ||
|
|
8cc65b8967 | ||
|
|
2e10b2f5d5 | ||
|
|
92fe406c22 | ||
|
|
d677e2bb70 | ||
|
|
e977f123ce | ||
|
|
f56b672307 | ||
|
|
29c1d7f3e4 | ||
|
|
36aba81b1b | ||
|
|
f9fb03b26b | ||
|
|
d2732ac742 | ||
|
|
9aa5702e14 | ||
|
|
2d412bfe52 | ||
|
|
ac06fb9d1c | ||
|
|
d7b31f2f9d | ||
|
|
635fae101d | ||
|
|
335022c4aa | ||
|
|
dbed6890da | ||
|
|
3ec9ae9772 | ||
|
|
a46ab5cd68 | ||
|
|
690a00aa6c | ||
|
|
ba786a0289 | ||
|
|
9dfac77ba2 | ||
|
|
f3370a4f52 | ||
|
|
7caa88bc58 | ||
|
|
34a5b8b9ce | ||
|
|
d8e3807ec5 | ||
|
|
42bd8164a0 | ||
|
|
abf6e72ad1 | ||
|
|
74cce5ccb2 | ||
|
|
dd053c76df | ||
|
|
5ef428b9d0 | ||
|
|
94ea574c76 | ||
|
|
70b41c2095 | ||
|
|
aa85a39d6a | ||
|
|
51c684251f | ||
|
|
fa2934373c | ||
|
|
414567f8b6 | ||
|
|
bc219191db | ||
|
|
652c8464a7 | ||
|
|
067fe9b534 | ||
|
|
38c85e6db1 | ||
|
|
c5cd7d8700 | ||
|
|
fc453c68a7 | ||
|
|
69c046cca4 | ||
|
|
dd8164611b | ||
|
|
b4a5c863d7 | ||
|
|
4a3d79d00b | ||
|
|
96f436b65e | ||
|
|
cf01dc770b | ||
|
|
ce652de272 | ||
|
|
9bb399606c | ||
|
|
8180d2e3b0 | ||
|
|
33d5721490 | ||
|
|
2b2a6db081 | ||
|
|
aa4592dc9e | ||
|
|
20ad9af989 | ||
|
|
9e2276a59f | ||
|
|
42cec10457 | ||
|
|
95d7e027fc | ||
|
|
8268311fab | ||
|
|
413b4b637b | ||
|
|
e79bbc67b9 | ||
|
|
15a1d5440b | ||
|
|
fd293444c5 | ||
|
|
852429f248 | ||
|
|
4606fcdbd5 | ||
|
|
c375cdbb1a | ||
|
|
f15b70e4ae | ||
|
|
2f10634ca2 | ||
|
|
dead993639 | ||
|
|
bd5cbb89d4 | ||
|
|
14f1925edc | ||
|
|
e39bbd2b31 | ||
|
|
04f06206b7 | ||
|
|
1bcea27a44 | ||
|
|
e7a3ad86eb | ||
|
|
b36ab66583 | ||
|
|
566c876bdc | ||
|
|
43acac5f5d | ||
|
|
846ecb7e8d | ||
|
|
2fba5dcd20 | ||
|
|
2828854d32 | ||
|
|
9794893b63 | ||
|
|
e57ee8e9c9 | ||
|
|
f978407a19 | ||
|
|
59ce70afc5 | ||
|
|
208ccc3a1a | ||
|
|
6f8f9f016a | ||
|
|
7ad2481e3d | ||
|
|
12c667afd7 | ||
|
|
2cb16c9fd0 | ||
|
|
9f00c6f955 | ||
|
|
a90997353b | ||
|
|
c6dc7eee64 | ||
|
|
0f2899b5c7 | ||
|
|
71cf3fabbf | ||
|
|
f0da3cff40 | ||
|
|
485e4d56aa | ||
|
|
99424e460b | ||
|
|
006abf79b1 | ||
|
|
5c3c3bfdba | ||
|
|
189f3a7bbc | ||
|
|
6ab558dd58 | ||
|
|
a7bb5254a3 | ||
|
|
ef87b54b43 | ||
|
|
88d0ebb12d | ||
|
|
bdb42bbbbd | ||
|
|
ac7ff3a0e9 | ||
|
|
1eb6027443 | ||
|
|
edc8691731 | ||
|
|
804a60f1ea | ||
|
|
a80a0531a6 | ||
|
|
24bc713ba8 | ||
|
|
4884640070 | ||
|
|
8d7535c039 | ||
|
|
32aa60bb14 | ||
|
|
3ada27eae1 | ||
|
|
a096a0e3bb | ||
|
|
d410078673 | ||
|
|
bf5891a910 | ||
|
|
c7249dfbd9 | ||
|
|
1b29e0bfa8 | ||
|
|
5cc5767c87 | ||
|
|
165c4ddd2a | ||
|
|
7b64970e97 | ||
|
|
5309dd17be | ||
|
|
364470b3c8 | ||
|
|
915da79cad | ||
|
|
630c71f123 | ||
|
|
0b4e7be596 | ||
|
|
303bd8b607 | ||
|
|
c02cf5e89e | ||
|
|
92f02fa4ee | ||
|
|
49e6cf95c4 | ||
|
|
8a126263a9 | ||
|
|
afdb385770 | ||
|
|
5b8897231d | ||
|
|
ffc55aff10 | ||
|
|
2b372c14e4 | ||
|
|
12b5525a2d | ||
|
|
3c48d92663 | ||
|
|
948ec9e042 | ||
|
|
5a274fdbed | ||
|
|
bfeb3a7ca9 | ||
|
|
d363ec5154 | ||
|
|
999733fc0f | ||
|
|
8d274c1924 | ||
|
|
2e1a0bec93 | ||
|
|
39ddaa0c35 | ||
|
|
d9b9c79112 | ||
|
|
485d5a8b48 | ||
|
|
f7e71a0868 | ||
|
|
af2995ba96 | ||
|
|
f7ffcc58fe | ||
|
|
792f077071 | ||
|
|
9c07c7a963 | ||
|
|
d579c82cb9 | ||
|
|
c7c7bf550a | ||
|
|
84483c56d7 | ||
|
|
1b71519ec6 | ||
|
|
154b9ace92 | ||
|
|
93f0715d02 | ||
|
|
57e6c91689 | ||
|
|
6213ccff48 | ||
|
|
c075deeb2d | ||
|
|
29f60c4a67 | ||
|
|
c553e15036 | ||
|
|
a4eef648bc | ||
|
|
47720ff286 | ||
|
|
289b139be9 | ||
|
|
21d60ec3f1 | ||
|
|
ed1115f4c0 | ||
|
|
cc5e2ab10a | ||
|
|
b19312be70 | ||
|
|
a0d1359860 | ||
|
|
52547bbfe8 | ||
|
|
9ab398f08d | ||
|
|
0deaa9d9b4 | ||
|
|
7c7755b61d | ||
|
|
6340c4a6b7 | ||
|
|
0a2fdb05ad | ||
|
|
b86853b65c | ||
|
|
192dac0396 | ||
|
|
85c9d8a9ff | ||
|
|
ee2699413c | ||
|
|
d50297bbe6 | ||
|
|
489ce416c3 | ||
|
|
731c487ccb | ||
|
|
2d5e8deb75 | ||
|
|
746f2af044 | ||
|
|
f16c552d97 | ||
|
|
72a85b58cf | ||
|
|
b88a98b09a | ||
|
|
4fbb9d9682 | ||
|
|
84aefda486 | ||
|
|
73cc0541f5 | ||
|
|
bcf3939fee | ||
|
|
2e6f1128bb | ||
|
|
bbd8671e6e | ||
|
|
84e6907fa9 | ||
|
|
93a4647b04 | ||
|
|
c1f93b37ab | ||
|
|
e7a1f86cbb | ||
|
|
987fe9c9e2 | ||
|
|
df9a8b2272 | ||
|
|
36d8280765 | ||
|
|
28c800dd37 | ||
|
|
9f90de2242 | ||
|
|
b2dd8eed3f | ||
|
|
0085015ebe | ||
|
|
db3bdb586b | ||
|
|
d722b2569c | ||
|
|
968be8a06f | ||
|
|
daa7db7ce3 | ||
|
|
cb2cf31cec | ||
|
|
ce4a3bf118 | ||
|
|
9d0bb0d51a | ||
|
|
296866a675 | ||
|
|
96c82a0377 | ||
|
|
ca3f6bba02 | ||
|
|
70f9982c29 | ||
|
|
521046fb09 | ||
|
|
421d6b334a | ||
|
|
e7af1bbec9 | ||
|
|
01e8e4d589 | ||
|
|
be17ee1902 | ||
|
|
e557ae0f2a | ||
|
|
e80f13b78e | ||
|
|
c12f4e9d2a | ||
|
|
a0f977e80c | ||
|
|
73a90bdf38 | ||
|
|
7f1d891e26 | ||
|
|
d6db21419c | ||
|
|
367507e054 | ||
|
|
715ee1d6fe | ||
|
|
ad06a81765 | ||
|
|
ebb02b27b2 | ||
|
|
d47dc35885 | ||
|
|
41aca8467a | ||
|
|
cd44392866 | ||
|
|
64bb5355ff | ||
|
|
1198db8891 | ||
|
|
9471adb4f7 | ||
|
|
b29909bdbe | ||
|
|
589e9a15a6 | ||
|
|
c218a66bcd | ||
|
|
8485bbe575 | ||
|
|
5d9d7d3aa5 | ||
|
|
ccdda39e78 | ||
|
|
2e2682aad4 | ||
|
|
a390c8ada7 | ||
|
|
da70344547 | ||
|
|
2b690f8c87 | ||
|
|
c8b419888a | ||
|
|
d3302d521f | ||
|
|
5c1bb27d5c | ||
|
|
dde36183ce | ||
|
|
7c7f2dd8d5 | ||
|
|
edb0a954e2 | ||
|
|
0a239712dd | ||
|
|
95c3434205 | ||
|
|
f0be6972cc | ||
|
|
a799cd097a | ||
|
|
b4e3f2cba9 | ||
|
|
658ae83157 | ||
|
|
d29e72ce72 | ||
|
|
30cc8ad6f9 | ||
|
|
f0ebd28148 | ||
|
|
364d0ca52b | ||
|
|
3ff2df2796 | ||
|
|
d57bec8ec6 | ||
|
|
2b64e65f45 | ||
|
|
eab4d435f8 | ||
|
|
de13c22552 | ||
|
|
8ee7163014 | ||
|
|
c5fde071e7 | ||
|
|
8abd04dab1 | ||
|
|
858361e6d3 | ||
|
|
9ece4bfd9b | ||
|
|
a1a3b02d3a | ||
|
|
bc2ae6c486 | ||
|
|
587a8bc524 | ||
|
|
122926c6cd | ||
|
|
bac6beeb50 | ||
|
|
c75a483ee6 | ||
|
|
1c5bfdbabe | ||
|
|
553952132f | ||
|
|
1931eb11a9 | ||
|
|
65aef396fb | ||
|
|
a21e45ed36 | ||
|
|
adee67115c | ||
|
|
aea468bc7f | ||
|
|
621c6a8d73 | ||
|
|
73591d5d0f | ||
|
|
846f16ddaa | ||
|
|
c46f6d1178 | ||
|
|
ab5d6db8a2 | ||
|
|
9957cb20e2 | ||
|
|
8788a7aada | ||
|
|
16392c28c5 | ||
|
|
a8ee0d7648 | ||
|
|
a405758222 | ||
|
|
537e29d937 | ||
|
|
9a3104c5ac | ||
|
|
722344967f | ||
|
|
73ce5e051c | ||
|
|
b8bbe141a0 | ||
|
|
3285596a93 | ||
|
|
73763b5ee6 | ||
|
|
27708d5964 | ||
|
|
827f22ed37 | ||
|
|
ace4063371 | ||
|
|
f62e29f5d5 | ||
|
|
5df00de246 | ||
|
|
28dfd62163 | ||
|
|
1c1599a9f4 | ||
|
|
6dff230e10 | ||
|
|
0f60824749 | ||
|
|
84e0732fb1 | ||
|
|
ae123587d7 | ||
|
|
2efcb48b7e | ||
|
|
3d9cae717d | ||
|
|
cc73a8cc85 | ||
|
|
648f028a63 | ||
|
|
840ee1379f | ||
|
|
3c85ca9cbc | ||
|
|
8e88749078 | ||
|
|
4c431e98a6 | ||
|
|
40cf3ced1a | ||
|
|
c9ccab8771 | ||
|
|
7d2700ca65 | ||
|
|
bfedc12fa8 | ||
|
|
c16267e60f | ||
|
|
0bd9b84931 | ||
|
|
e9a9475ed7 | ||
|
|
f00d415dd7 | ||
|
|
1e7ff3dbdf | ||
|
|
eb1c257484 | ||
|
|
4d652210dc | ||
|
|
9f2ac78609 | ||
|
|
484e8e64a6 | ||
|
|
e5d5dfa8d8 | ||
|
|
0ad31fea46 | ||
|
|
ec2d8f231d | ||
|
|
aeec56f800 | ||
|
|
7e6cad974f | ||
|
|
757e9f3415 | ||
|
|
6bc2293292 | ||
|
|
55aaca0e0d | ||
|
|
a8a3fca8c9 | ||
|
|
fb30515f72 | ||
|
|
9a39af6da0 | ||
|
|
6d75ce4b1c | ||
|
|
dbd1d30ca8 | ||
|
|
8f80996515 | ||
|
|
d206a70b8a | ||
|
|
bbec4d2c7e | ||
|
|
f24c36d6b1 | ||
|
|
adff828415 | ||
|
|
d914667238 | ||
|
|
1c209b3320 | ||
|
|
aa61032cdf | ||
|
|
3ae4a7d8a7 | ||
|
|
9fdecf21f2 | ||
|
|
e4d256790f | ||
|
|
d9f4faf4ec | ||
|
|
609a364971 | ||
|
|
2152ddd99b | ||
|
|
8caaf8515e | ||
|
|
ac822fa084 | ||
|
|
a265b865f6 | ||
|
|
8efc4f4817 | ||
|
|
817a3f979e | ||
|
|
8ebfc40de5 | ||
|
|
c42e953199 | ||
|
|
636aa72141 | ||
|
|
14e6e6eff4 | ||
|
|
474c1d0d89 | ||
|
|
b8983f27ab | ||
|
|
d9c65d4ae0 | ||
|
|
1ef07309d6 | ||
|
|
be1403c920 | ||
|
|
03e32f0a7c | ||
|
|
f4361d1b43 | ||
|
|
e1259e67d3 | ||
|
|
ca1b9bf75f | ||
|
|
3cff55b0bb | ||
|
|
c25b153185 | ||
|
|
9c58e6f90f | ||
|
|
cff16346ef | ||
|
|
30e6b52783 | ||
|
|
015570c833 | ||
|
|
7c31197b78 | ||
|
|
a69bfb8cb8 | ||
|
|
c5b361e94d | ||
|
|
201750c89f | ||
|
|
89a8d0f6b8 | ||
|
|
835e4af3e4 | ||
|
|
c33106eab4 | ||
|
|
bea8c3dbec | ||
|
|
b51d5d007b | ||
|
|
f8c4f774cf | ||
|
|
22c6756ce0 | ||
|
|
e318815025 | ||
|
|
0833f94502 | ||
|
|
ddbf5e1457 | ||
|
|
2a032f3116 | ||
|
|
5381562a5e | ||
|
|
f2ce5e9693 | ||
|
|
ed50fd98cd | ||
|
|
14991e1f9e | ||
|
|
5fa06746f1 | ||
|
|
d31b044529 | ||
|
|
754dd904d2 | ||
|
|
1503394662 | ||
|
|
436d69b710 | ||
|
|
891e7bf6e4 | ||
|
|
fad1280185 | ||
|
|
6187c7268f | ||
|
|
762ea47b8e | ||
|
|
23612ba6ec | ||
|
|
217646f031 | ||
|
|
91667d9ecd | ||
|
|
3501ee9a9d | ||
|
|
f0c3860032 | ||
|
|
17dcaf9afa | ||
|
|
f484c98152 | ||
|
|
46f68115b2 | ||
|
|
7ac22a6ce8 | ||
|
|
bed79ef89e | ||
|
|
54095b9a89 | ||
|
|
4b58060ab6 | ||
|
|
dbbb554735 | ||
|
|
9c3be1b851 | ||
|
|
190ab211e3 | ||
|
|
f4fbe90a72 | ||
|
|
a9cbd5a172 | ||
|
|
94ba1c83c6 | ||
|
|
9b3756e591 | ||
|
|
aff775f488 | ||
|
|
4de5274996 | ||
|
|
4e57868037 | ||
|
|
ab6b32b8ba | ||
|
|
46285a499e | ||
|
|
ce6cad5dfe | ||
|
|
d5ea86bc81 | ||
|
|
41ea424359 | ||
|
|
87b0259b97 | ||
|
|
2490e97ea0 | ||
|
|
eecfe9d387 | ||
|
|
9ca5ee52e7 | ||
|
|
fb23cc8c7a | ||
|
|
ff7ce39841 | ||
|
|
0f05d6bd74 | ||
|
|
93c15c8151 | ||
|
|
22c8268f02 | ||
|
|
216cd491cc | ||
|
|
5efc573783 | ||
|
|
bca0da6bd4 | ||
|
|
9601a1fa4e | ||
|
|
b8e545bfa4 | ||
|
|
5c16f0d027 | ||
|
|
1b4441baac | ||
|
|
0147a1d41f | ||
|
|
0f11fbe599 | ||
|
|
9c18e99fe2 | ||
|
|
6e0a7de9cc | ||
|
|
b141a65838 | ||
|
|
7f842bb1e8 | ||
|
|
bd65aae81e | ||
|
|
b8dedb568c | ||
|
|
ec94c29ed9 | ||
|
|
c77943d01c | ||
|
|
fc5fb956df | ||
|
|
2eba126ed7 | ||
|
|
da52d677c7 | ||
|
|
ab4373d025 | ||
|
|
5e88a0f0cc | ||
|
|
ba794c2b60 | ||
|
|
268559d8de | ||
|
|
473d147333 | ||
|
|
d5d9f9bedc | ||
|
|
f31d30bf84 | ||
|
|
03b216a6b4 | ||
|
|
3e6510b935 | ||
|
|
e87b35e0bb | ||
|
|
322fdb14de | ||
|
|
09381575a7 | ||
|
|
881243da6a | ||
|
|
851fad3e3f | ||
|
|
46d65f0b7e | ||
|
|
03bebbe4c2 | ||
|
|
1649f30389 | ||
|
|
38f76d449a | ||
|
|
f2cbdae829 | ||
|
|
3a42354efd | ||
|
|
fe7853a389 | ||
|
|
06738cae93 | ||
|
|
d6243a2dd2 | ||
|
|
04be0fe634 | ||
|
|
fd12a1f6be | ||
|
|
e10dd4ef42 | ||
|
|
62c02e3fce | ||
|
|
ae2c55c33b | ||
|
|
4a2a646943 | ||
|
|
ae47ee802b | ||
|
|
4fd2f1f974 | ||
|
|
b9f16e8cce | ||
|
|
70b4843bc4 | ||
|
|
95190c321c | ||
|
|
a9b3be33e0 | ||
|
|
e05dad4267 | ||
|
|
83b90d472f | ||
|
|
ee4ff6e732 | ||
|
|
2267d38352 | ||
|
|
4fbb47300e | ||
|
|
836e168a6c | ||
|
|
a9684f67cc | ||
|
|
686e90d0ed | ||
|
|
0b9b39fef7 | ||
|
|
67a3b03b07 | ||
|
|
388db4e995 | ||
|
|
a28a1aa601 | ||
|
|
ed82106359 | ||
|
|
c57b84cb17 | ||
|
|
2dfe9fcf45 | ||
|
|
51f978ac72 | ||
|
|
c216677e1e | ||
|
|
8a5d275136 | ||
|
|
a36b76ba15 | ||
|
|
f3913949b2 | ||
|
|
76fdbfaa2f | ||
|
|
6b44b9ae1e | ||
|
|
8615c4c3b0 | ||
|
|
f976bd3fff | ||
|
|
db55a2664f | ||
|
|
4f1b4b1283 | ||
|
|
208f8349a6 | ||
|
|
32e71b0386 | ||
|
|
1c687e7a45 | ||
|
|
53ed028663 | ||
|
|
2a4ebe1b3e | ||
|
|
df863355b7 | ||
|
|
66e44f3448 | ||
|
|
c1f12f52ae | ||
|
|
b69091a51a | ||
|
|
0fe30ebe49 | ||
|
|
34f5552c7d | ||
|
|
53e27ff4d3 | ||
|
|
f4569c513f | ||
|
|
90ce5f2c57 | ||
|
|
6f3a07b756 | ||
|
|
d42bfa88e1 | ||
|
|
61a6b32137 | ||
|
|
6cb4e203f1 | ||
|
|
2d27a721f8 | ||
|
|
867a8e855b | ||
|
|
d5dc70ccee | ||
|
|
18d9c1d609 | ||
|
|
c388cee1fe | ||
|
|
da2488f7d8 | ||
|
|
f059643187 | ||
|
|
30730fe632 | ||
|
|
39df168891 | ||
|
|
33946e61bb | ||
|
|
0b34d20716 | ||
|
|
8d0b36d2d4 | ||
|
|
7e08f7df67 | ||
|
|
a688a5fe72 | ||
|
|
fe57efb1a8 | ||
|
|
9af3fbc35f | ||
|
|
7495d2dc65 | ||
|
|
456da93465 | ||
|
|
e7a6126fbd | ||
|
|
aa41eb5da4 | ||
|
|
87767be4a6 | ||
|
|
1e0a356cd2 | ||
|
|
1b40c9e692 | ||
|
|
2cc7565841 | ||
|
|
cee67f4301 |
|
|
@ -1,8 +1,9 @@
|
||||||
FROM ubuntu:22.04
|
FROM debian:13
|
||||||
|
|
||||||
RUN apt-get update && \
|
RUN apt-get update && \
|
||||||
DEBIAN_FRONTEND=noninteractive apt-get install -y --no-install-recommends \
|
DEBIAN_FRONTEND=noninteractive apt-get install -y --no-install-recommends \
|
||||||
build-essential \
|
build-essential \
|
||||||
|
ca-certificates \
|
||||||
ccache \
|
ccache \
|
||||||
clang-format \
|
clang-format \
|
||||||
cmake \
|
cmake \
|
||||||
|
|
@ -15,14 +16,14 @@ RUN apt-get update && \
|
||||||
libprotobuf-dev \
|
libprotobuf-dev \
|
||||||
libqt6multimedia6 \
|
libqt6multimedia6 \
|
||||||
libqt6sql6-mysql \
|
libqt6sql6-mysql \
|
||||||
libqt6svg6-dev \
|
|
||||||
libqt6websockets6-dev \
|
|
||||||
ninja-build \
|
ninja-build \
|
||||||
protobuf-compiler \
|
protobuf-compiler \
|
||||||
qt6-image-formats-plugins \
|
qt6-image-formats-plugins \
|
||||||
qt6-l10n-tools \
|
qt6-l10n-tools \
|
||||||
qt6-multimedia-dev \
|
qt6-multimedia-dev \
|
||||||
|
qt6-svg-dev \
|
||||||
qt6-tools-dev \
|
qt6-tools-dev \
|
||||||
qt6-tools-dev-tools \
|
qt6-tools-dev-tools \
|
||||||
|
qt6-websockets-dev \
|
||||||
&& apt-get clean \
|
&& apt-get clean \
|
||||||
&& rm -rf /var/lib/apt/lists/*
|
&& rm -rf /var/lib/apt/lists/*
|
||||||
|
|
@ -1,4 +1,4 @@
|
||||||
FROM fedora:41
|
FROM fedora:43
|
||||||
|
|
||||||
RUN dnf install -y \
|
RUN dnf install -y \
|
||||||
ccache \
|
ccache \
|
||||||
|
|
@ -1,4 +1,4 @@
|
||||||
FROM fedora:42
|
FROM fedora:44
|
||||||
|
|
||||||
RUN dnf install -y \
|
RUN dnf install -y \
|
||||||
ccache \
|
ccache \
|
||||||
|
|
@ -1,4 +1,4 @@
|
||||||
FROM debian:11
|
FROM debian:12
|
||||||
|
|
||||||
RUN apt-get update && \
|
RUN apt-get update && \
|
||||||
DEBIAN_FRONTEND=noninteractive apt-get install -y --no-install-recommends \
|
DEBIAN_FRONTEND=noninteractive apt-get install -y --no-install-recommends \
|
||||||
|
|
@ -9,18 +9,13 @@ RUN apt-get update && \
|
||||||
file \
|
file \
|
||||||
g++ \
|
g++ \
|
||||||
git \
|
git \
|
||||||
liblzma-dev \
|
|
||||||
libmariadb-dev-compat \
|
libmariadb-dev-compat \
|
||||||
libprotobuf-dev \
|
libprotobuf-dev \
|
||||||
libqt5multimedia5-plugins \
|
libqt6sql6-mysql \
|
||||||
libqt5sql5-mysql \
|
|
||||||
libqt5svg5-dev \
|
|
||||||
libqt5websockets5-dev \
|
|
||||||
ninja-build \
|
ninja-build \
|
||||||
protobuf-compiler \
|
protobuf-compiler \
|
||||||
qt5-image-formats-plugins \
|
qt6-tools-dev \
|
||||||
qtmultimedia5-dev \
|
qt6-tools-dev-tools \
|
||||||
qttools5-dev \
|
qt6-websockets-dev \
|
||||||
qttools5-dev-tools \
|
|
||||||
&& apt-get clean \
|
&& apt-get clean \
|
||||||
&& rm -rf /var/lib/apt/lists/*
|
&& rm -rf /var/lib/apt/lists/*
|
||||||
29
.ci/Ubuntu26.04/Dockerfile
Normal file
|
|
@ -0,0 +1,29 @@
|
||||||
|
FROM ubuntu:26.04
|
||||||
|
|
||||||
|
RUN apt-get update && \
|
||||||
|
DEBIAN_FRONTEND=noninteractive apt-get install -y --no-install-recommends \
|
||||||
|
build-essential \
|
||||||
|
ca-certificates \
|
||||||
|
ccache \
|
||||||
|
clang-format \
|
||||||
|
cmake \
|
||||||
|
file \
|
||||||
|
g++ \
|
||||||
|
git \
|
||||||
|
libgl-dev \
|
||||||
|
liblzma-dev \
|
||||||
|
libmariadb-dev-compat \
|
||||||
|
libprotobuf-dev \
|
||||||
|
libqt6multimedia6 \
|
||||||
|
libqt6sql6-mysql \
|
||||||
|
ninja-build \
|
||||||
|
protobuf-compiler \
|
||||||
|
qt6-image-formats-plugins \
|
||||||
|
qt6-l10n-tools \
|
||||||
|
qt6-multimedia-dev \
|
||||||
|
qt6-svg-dev \
|
||||||
|
qt6-tools-dev \
|
||||||
|
qt6-tools-dev-tools \
|
||||||
|
qt6-websockets-dev \
|
||||||
|
&& apt-get clean \
|
||||||
|
&& rm -rf /var/lib/apt/lists/*
|
||||||
167
.ci/compile.sh
|
|
@ -10,8 +10,11 @@
|
||||||
# --test runs tests
|
# --test runs tests
|
||||||
# --debug or --release sets the build type ie CMAKE_BUILD_TYPE
|
# --debug or --release sets the build type ie CMAKE_BUILD_TYPE
|
||||||
# --ccache [<size>] uses ccache and shows stats, optionally provide size
|
# --ccache [<size>] uses ccache and shows stats, optionally provide size
|
||||||
|
# --evict-ccache <age> runs ccache eviction based on given age after build
|
||||||
# --dir <dir> sets the name of the build dir, default is "build"
|
# --dir <dir> sets the name of the build dir, default is "build"
|
||||||
# uses env: BUILDTYPE MAKE_INSTALL MAKE_PACKAGE PACKAGE_TYPE PACKAGE_SUFFIX MAKE_SERVER MAKE_TEST USE_CCACHE CCACHE_SIZE BUILD_DIR CMAKE_GENERATOR
|
# --cmake-generator <generator> sets CMAKE_GENERATOR as used by cmake
|
||||||
|
# --target-macos-version <version> sets the min os version - only used for macOS builds
|
||||||
|
# uses env: BUILDTYPE MAKE_INSTALL MAKE_PACKAGE PACKAGE_TYPE PACKAGE_SUFFIX MAKE_SERVER MAKE_NO_CLIENT MAKE_TEST USE_CCACHE CCACHE_SIZE CCACHE_EVICTION_AGE BUILD_DIR CMAKE_GENERATOR TARGET_MACOS_VERSION
|
||||||
# (correspond to args: --debug/--release --install --package <package type> --suffix <suffix> --server --test --ccache <ccache_size> --dir <dir>)
|
# (correspond to args: --debug/--release --install --package <package type> --suffix <suffix> --server --test --ccache <ccache_size> --dir <dir>)
|
||||||
# exitcode: 1 for failure, 3 for invalid arguments
|
# exitcode: 1 for failure, 3 for invalid arguments
|
||||||
|
|
||||||
|
|
@ -46,6 +49,10 @@ while [[ $# != 0 ]]; do
|
||||||
MAKE_SERVER=1
|
MAKE_SERVER=1
|
||||||
shift
|
shift
|
||||||
;;
|
;;
|
||||||
|
'--no-client')
|
||||||
|
MAKE_NO_CLIENT=1
|
||||||
|
shift
|
||||||
|
;;
|
||||||
'--test')
|
'--test')
|
||||||
MAKE_TEST=1
|
MAKE_TEST=1
|
||||||
shift
|
shift
|
||||||
|
|
@ -66,6 +73,19 @@ while [[ $# != 0 ]]; do
|
||||||
shift
|
shift
|
||||||
fi
|
fi
|
||||||
;;
|
;;
|
||||||
|
'--evict-ccache')
|
||||||
|
shift
|
||||||
|
if [[ $# == 0 ]]; then
|
||||||
|
echo "::error file=$0::--evict-ccache expects an argument"
|
||||||
|
exit 3
|
||||||
|
fi
|
||||||
|
CCACHE_EVICTION_AGE=$1
|
||||||
|
shift
|
||||||
|
;;
|
||||||
|
'--vcpkg')
|
||||||
|
USE_VCPKG=1
|
||||||
|
shift
|
||||||
|
;;
|
||||||
'--dir')
|
'--dir')
|
||||||
shift
|
shift
|
||||||
if [[ $# == 0 ]]; then
|
if [[ $# == 0 ]]; then
|
||||||
|
|
@ -75,6 +95,24 @@ while [[ $# != 0 ]]; do
|
||||||
BUILD_DIR="$1"
|
BUILD_DIR="$1"
|
||||||
shift
|
shift
|
||||||
;;
|
;;
|
||||||
|
'--cmake-generator')
|
||||||
|
shift
|
||||||
|
if [[ $# == 0 ]]; then
|
||||||
|
echo "::error file=$0::--cmake-generator expects an argument"
|
||||||
|
exit 3
|
||||||
|
fi
|
||||||
|
export CMAKE_GENERATOR=$1
|
||||||
|
shift
|
||||||
|
;;
|
||||||
|
'--target-macos-version')
|
||||||
|
shift
|
||||||
|
if [[ $# == 0 ]]; then
|
||||||
|
echo "::error file=$0::--target-macos-version expects an argument"
|
||||||
|
exit 3
|
||||||
|
fi
|
||||||
|
TARGET_MACOS_VERSION="$1"
|
||||||
|
shift
|
||||||
|
;;
|
||||||
*)
|
*)
|
||||||
echo "::error file=$0::unrecognized option: $1"
|
echo "::error file=$0::unrecognized option: $1"
|
||||||
exit 3
|
exit 3
|
||||||
|
|
@ -95,11 +133,17 @@ fi
|
||||||
mkdir -p "$BUILD_DIR"
|
mkdir -p "$BUILD_DIR"
|
||||||
cd "$BUILD_DIR"
|
cd "$BUILD_DIR"
|
||||||
|
|
||||||
|
# Set minimum CMake Version
|
||||||
|
export CMAKE_POLICY_VERSION_MINIMUM=3.10
|
||||||
|
|
||||||
# Add cmake flags
|
# Add cmake flags
|
||||||
flags=("-DCMAKE_BUILD_TYPE=$BUILDTYPE")
|
flags=("-DCMAKE_BUILD_TYPE=$BUILDTYPE")
|
||||||
if [[ $MAKE_SERVER ]]; then
|
if [[ $MAKE_SERVER ]]; then
|
||||||
flags+=("-DWITH_SERVER=1")
|
flags+=("-DWITH_SERVER=1")
|
||||||
fi
|
fi
|
||||||
|
if [[ $MAKE_NO_CLIENT ]]; then
|
||||||
|
flags+=("-DWITH_CLIENT=0" "-DWITH_ORACLE=0")
|
||||||
|
fi
|
||||||
if [[ $MAKE_TEST ]]; then
|
if [[ $MAKE_TEST ]]; then
|
||||||
flags+=("-DTEST=1")
|
flags+=("-DTEST=1")
|
||||||
fi
|
fi
|
||||||
|
|
@ -113,6 +157,9 @@ fi
|
||||||
if [[ $PACKAGE_TYPE ]]; then
|
if [[ $PACKAGE_TYPE ]]; then
|
||||||
flags+=("-DCPACK_GENERATOR=$PACKAGE_TYPE")
|
flags+=("-DCPACK_GENERATOR=$PACKAGE_TYPE")
|
||||||
fi
|
fi
|
||||||
|
if [[ $USE_VCPKG ]]; then
|
||||||
|
flags+=("-DUSE_VCPKG=1")
|
||||||
|
fi
|
||||||
|
|
||||||
# Add cmake --build flags
|
# Add cmake --build flags
|
||||||
buildflags=(--config "$BUILDTYPE")
|
buildflags=(--config "$BUILDTYPE")
|
||||||
|
|
@ -128,6 +175,88 @@ function ccachestatsverbose() {
|
||||||
}
|
}
|
||||||
|
|
||||||
# Compile
|
# Compile
|
||||||
|
if [[ $RUNNER_OS == macOS ]]; then
|
||||||
|
# QTDIR is needed for macOS since we actually only use the cached thin Qt binaries instead of the install-qt-action,
|
||||||
|
# which sets a few environment variables
|
||||||
|
if QTDIR=$(find "$GITHUB_WORKSPACE/Qt" -depth -maxdepth 2 -name macos -type d -print -quit); then
|
||||||
|
echo "found QTDIR at $QTDIR"
|
||||||
|
else
|
||||||
|
echo "could not find QTDIR!"
|
||||||
|
exit 2
|
||||||
|
fi
|
||||||
|
# the qtdir is located at Qt/[qtversion]/macos
|
||||||
|
# we use find to get the first subfolder with the name "macos"
|
||||||
|
# this works independent of the qt version as there should be only one version installed on the runner at a time
|
||||||
|
export QTDIR
|
||||||
|
|
||||||
|
if [[ $TARGET_MACOS_VERSION ]]; then
|
||||||
|
# CMAKE_OSX_DEPLOYMENT_TARGET is a vanilla cmake flag needed to compile to target macOS version
|
||||||
|
flags+=("-DCMAKE_OSX_DEPLOYMENT_TARGET=$TARGET_MACOS_VERSION")
|
||||||
|
|
||||||
|
# vcpkg dependencies need a vcpkg triplet file to compile to the target macOS version
|
||||||
|
# an easy way is to copy the x64-osx.cmake file and modify it
|
||||||
|
triplets_dir="/tmp/cmake/triplets"
|
||||||
|
triplet_version="custom-triplet"
|
||||||
|
triplet_file="$triplets_dir/$triplet_version.cmake"
|
||||||
|
arch=$(uname -m)
|
||||||
|
if [[ $arch == x86_64 ]]; then
|
||||||
|
arch="x64"
|
||||||
|
fi
|
||||||
|
mkdir -p "$triplets_dir"
|
||||||
|
triplet_source="../vcpkg/triplets/$arch-osx.cmake"
|
||||||
|
if [[ ! -f "$triplet_source" ]]; then
|
||||||
|
triplet_source="../vcpkg/triplets/community/$arch-osx.cmake"
|
||||||
|
fi
|
||||||
|
cp "$triplet_source" "$triplet_file"
|
||||||
|
echo "set(VCPKG_CMAKE_SYSTEM_VERSION $TARGET_MACOS_VERSION)" >>"$triplet_file"
|
||||||
|
echo "set(VCPKG_OSX_DEPLOYMENT_TARGET $TARGET_MACOS_VERSION)" >>"$triplet_file"
|
||||||
|
flags+=("-DVCPKG_OVERLAY_TRIPLETS=$triplets_dir")
|
||||||
|
flags+=("-DVCPKG_HOST_TRIPLET=$triplet_version")
|
||||||
|
flags+=("-DVCPKG_TARGET_TRIPLET=$triplet_version")
|
||||||
|
echo "::group::Generated triplet $triplet_file"
|
||||||
|
cat "$triplet_file"
|
||||||
|
echo "::endgroup::"
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo "::group::Signing Certificate"
|
||||||
|
if [[ -n "$MACOS_CERTIFICATE_NAME" ]]; then
|
||||||
|
echo "$MACOS_CERTIFICATE" | base64 --decode >"certificate.p12"
|
||||||
|
security create-keychain -p "$MACOS_CI_KEYCHAIN_PWD" build.keychain
|
||||||
|
security default-keychain -s build.keychain
|
||||||
|
security set-keychain-settings -t 3600 -l build.keychain
|
||||||
|
security unlock-keychain -p "$MACOS_CI_KEYCHAIN_PWD" build.keychain
|
||||||
|
security import certificate.p12 -k build.keychain -P "$MACOS_CERTIFICATE_PWD" -T /usr/bin/codesign
|
||||||
|
security set-key-partition-list -S apple-tool:,apple:,codesign: -s -k "$MACOS_CI_KEYCHAIN_PWD" build.keychain
|
||||||
|
echo "macOS signing certificate successfully imported and keychain configured."
|
||||||
|
else
|
||||||
|
echo "No signing certificate configured. Skipping set up of keychain in macOS environment."
|
||||||
|
fi
|
||||||
|
echo "::endgroup::"
|
||||||
|
|
||||||
|
if [[ $MAKE_PACKAGE ]]; then
|
||||||
|
# Workaround https://github.com/actions/runner-images/issues/7522
|
||||||
|
# have hdiutil repeat the command 10 times in hope of success
|
||||||
|
hdiutil_script="/tmp/hdiutil.sh"
|
||||||
|
# shellcheck disable=SC2016
|
||||||
|
echo '#!/bin/bash
|
||||||
|
i=0
|
||||||
|
while ! hdiutil "$@"; do
|
||||||
|
if (( ++i >= 10 )); then
|
||||||
|
echo "Error: hdiutil failed $i times!" >&2
|
||||||
|
break
|
||||||
|
fi
|
||||||
|
sleep 1
|
||||||
|
done' >"$hdiutil_script"
|
||||||
|
chmod +x "$hdiutil_script"
|
||||||
|
flags+=(-DCPACK_COMMAND_HDIUTIL="$hdiutil_script")
|
||||||
|
fi
|
||||||
|
|
||||||
|
elif [[ $RUNNER_OS == Windows ]]; then
|
||||||
|
# Enable MTT, see https://devblogs.microsoft.com/cppblog/improved-parallelism-in-msbuild/
|
||||||
|
# and https://devblogs.microsoft.com/cppblog/cpp-build-throughput-investigation-and-tune-up/#multitooltask-mtt
|
||||||
|
buildflags+=(-- -p:UseMultiToolTask=true -p:EnableClServerMode=true)
|
||||||
|
fi
|
||||||
|
|
||||||
if [[ $USE_CCACHE ]]; then
|
if [[ $USE_CCACHE ]]; then
|
||||||
echo "::group::Show ccache stats"
|
echo "::group::Show ccache stats"
|
||||||
ccachestatsverbose
|
ccachestatsverbose
|
||||||
|
|
@ -136,23 +265,39 @@ fi
|
||||||
|
|
||||||
echo "::group::Configure cmake"
|
echo "::group::Configure cmake"
|
||||||
cmake --version
|
cmake --version
|
||||||
|
echo "Running cmake with flags: ${flags[*]}"
|
||||||
cmake .. "${flags[@]}"
|
cmake .. "${flags[@]}"
|
||||||
echo "::endgroup::"
|
echo "::endgroup::"
|
||||||
|
|
||||||
echo "::group::Build project"
|
echo "::group::Build project"
|
||||||
if [[ $RUNNER_OS == Windows ]]; then
|
echo "Running cmake --build with flags: ${buildflags[*]}"
|
||||||
# Enable MTT, see https://devblogs.microsoft.com/cppblog/improved-parallelism-in-msbuild/
|
cmake --build . "${buildflags[@]}"
|
||||||
# and https://devblogs.microsoft.com/cppblog/cpp-build-throughput-investigation-and-tune-up/#multitooltask-mtt
|
|
||||||
cmake --build . "${buildflags[@]}" -- -p:UseMultiToolTask=true -p:EnableClServerMode=true
|
|
||||||
else
|
|
||||||
cmake --build . "${buildflags[@]}"
|
|
||||||
fi
|
|
||||||
echo "::endgroup::"
|
echo "::endgroup::"
|
||||||
|
|
||||||
if [[ $USE_CCACHE ]]; then
|
if [[ $USE_CCACHE ]]; then
|
||||||
|
if [[ $CCACHE_EVICTION_AGE ]]; then
|
||||||
|
echo "::group::evict ccache files older than $CCACHE_EVICTION_AGE"
|
||||||
|
ccache --evict-older-than "$CCACHE_EVICTION_AGE"
|
||||||
|
echo "::endgroup::"
|
||||||
|
fi
|
||||||
echo "::group::Show ccache stats again"
|
echo "::group::Show ccache stats again"
|
||||||
ccachestatsverbose
|
ccachestatsverbose
|
||||||
echo "::endgroup::"
|
echo "::endgroup::"
|
||||||
|
elif [[ $CCACHE_EVICTION_AGE ]]; then
|
||||||
|
echo "::error file=$0::ccache eviction is enabled while ccache is disabled!"
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [[ $RUNNER_OS == macOS ]]; then
|
||||||
|
echo "::group::Inspect Mach-O binaries"
|
||||||
|
for app in cockatrice oracle servatrice; do
|
||||||
|
binary="$GITHUB_WORKSPACE/build/$app/$app.app/Contents/MacOS/$app"
|
||||||
|
echo "Inspecting $app..."
|
||||||
|
vtool -show-build "$binary"
|
||||||
|
file "$binary"
|
||||||
|
lipo -info "$binary"
|
||||||
|
echo ""
|
||||||
|
done
|
||||||
|
echo "::endgroup::"
|
||||||
fi
|
fi
|
||||||
|
|
||||||
if [[ $MAKE_TEST ]]; then
|
if [[ $MAKE_TEST ]]; then
|
||||||
|
|
@ -169,12 +314,6 @@ fi
|
||||||
|
|
||||||
if [[ $MAKE_PACKAGE ]]; then
|
if [[ $MAKE_PACKAGE ]]; then
|
||||||
echo "::group::Create package"
|
echo "::group::Create package"
|
||||||
|
|
||||||
if [[ $RUNNER_OS == macOS ]]; then
|
|
||||||
# Workaround https://github.com/actions/runner-images/issues/7522
|
|
||||||
echo "killing XProtectBehaviorService"; sudo pkill -9 XProtect >/dev/null || true;
|
|
||||||
echo "waiting for XProtectBehaviorService kill"; while pgrep "XProtect"; do sleep 3; done;
|
|
||||||
fi
|
|
||||||
cmake --build . --target package --config "$BUILDTYPE"
|
cmake --build . --target package --config "$BUILDTYPE"
|
||||||
echo "::endgroup::"
|
echo "::endgroup::"
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -3,17 +3,28 @@
|
||||||
# This script is to be used by the ci environment from the project root directory, do not use it from somewhere else.
|
# This script is to be used by the ci environment from the project root directory, do not use it from somewhere else.
|
||||||
|
|
||||||
# Creates or loads docker images to use in compilation, creates RUN function to start compilation on the docker image.
|
# Creates or loads docker images to use in compilation, creates RUN function to start compilation on the docker image.
|
||||||
# <arg> sets the name of the docker image, these correspond to directories in .ci
|
#
|
||||||
|
# usage: source <script> <name> [--get] [--build] [--save] [--interactive] [--set-cache <location>]
|
||||||
|
# <name> sets the name of the docker image, these correspond to directories in .ci
|
||||||
# --get loads the image from a previously saved image cache, will build if no image is found
|
# --get loads the image from a previously saved image cache, will build if no image is found
|
||||||
# --build builds the image from the Dockerfile in .ci/$NAME
|
# --build builds the image from the Dockerfile in .ci/$NAME
|
||||||
# --save stores the image, if an image was loaded it will not be stored
|
# --save stores the image, if an image was loaded it will not be stored
|
||||||
# --interactive immediately starts the image interactively for debugging
|
# --interactive immediately starts the image interactively for debugging
|
||||||
# --set-cache <location> sets the location to cache the image or for ccache
|
# --set-cache <location> sets the location to cache the image or for ccache
|
||||||
|
#
|
||||||
# requires: docker
|
# requires: docker
|
||||||
# uses env: NAME CACHE BUILD GET SAVE INTERACTIVE
|
# uses env: NAME CACHE BUILD GET SAVE INTERACTIVE
|
||||||
# (correspond to args: <name> --set-cache <cache> --build --get --save --interactive)
|
# (correspond to args: <name> --set-cache <cache> --build --get --save --interactive)
|
||||||
# sets env: RUN CCACHE_DIR IMAGE_NAME RUN_ARGS RUN_OPTS BUILD_SCRIPT
|
# sets env: RUN CCACHE_DIR IMAGE_NAME RUN_ARGS RUN_OPTS BUILD_SCRIPT
|
||||||
# exitcode: 1 for failure, 2 for missing dockerfile, 3 for invalid arguments
|
# exitcode: 1 for failure, 2 for missing dockerfile, 3 for invalid arguments
|
||||||
|
#
|
||||||
|
# exported RUN function will run the BUILD_SCRIPT inside of the docker container.
|
||||||
|
# note that the docker container will not inherit any environment variables!
|
||||||
|
#
|
||||||
|
# usage: RUN [arguments for build script]
|
||||||
|
# roughly equivalent to `docker run $IMAGE_NAME bash $BUILD_SCRIPT $@`
|
||||||
|
# uses env: CCACHE_DIR IMAGE_NAME RUN_ARGS RUN_OPTS BUILD_SCRIPT
|
||||||
|
# exitcode: 3 for invalid arguments, returns the returncode of docker run
|
||||||
export BUILD_SCRIPT=".ci/compile.sh"
|
export BUILD_SCRIPT=".ci/compile.sh"
|
||||||
|
|
||||||
project_name="cockatrice"
|
project_name="cockatrice"
|
||||||
|
|
@ -41,12 +52,17 @@ while [[ $# != 0 ]]; do
|
||||||
shift
|
shift
|
||||||
;;
|
;;
|
||||||
'--set-cache')
|
'--set-cache')
|
||||||
CACHE=$2
|
shift
|
||||||
|
if [[ $# == 0 ]]; then
|
||||||
|
echo "--set-cache expects an argument" >&2
|
||||||
|
exit 3
|
||||||
|
fi
|
||||||
|
CACHE=$1
|
||||||
|
shift
|
||||||
if ! [[ -d $CACHE ]]; then
|
if ! [[ -d $CACHE ]]; then
|
||||||
echo "could not find cache path: $CACHE" >&2
|
echo "could not find cache path: $CACHE" >&2
|
||||||
return 3
|
return 3
|
||||||
fi
|
fi
|
||||||
shift 2
|
|
||||||
;;
|
;;
|
||||||
*)
|
*)
|
||||||
if [[ ${1:0:1} == - ]]; then
|
if [[ ${1:0:1} == - ]]; then
|
||||||
|
|
@ -137,18 +153,23 @@ if [[ $SAVE ]]; then
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# Set compile function, runs the compile script on the image, passes arguments to the script
|
# Set compile function, runs the compile script on the image, passes arguments to the script
|
||||||
|
# shellcheck disable=2120
|
||||||
function RUN ()
|
function RUN ()
|
||||||
{
|
{
|
||||||
echo "running image:"
|
echo "running image:"
|
||||||
if [[ $(docker images) =~ "$IMAGE_NAME" ]]; then
|
if [[ $(docker images) =~ $IMAGE_NAME ]]; then
|
||||||
local args=(--mount "type=bind,source=$PWD,target=/src")
|
local args=(--mount "type=bind,source=$PWD,target=/src")
|
||||||
args+=(--workdir "/src")
|
args+=(--workdir "/src")
|
||||||
args+=(--user "$(id -u):$(id -g)")
|
args+=(--user "$(id -u):$(id -g)")
|
||||||
if [[ $CCACHE_DIR ]]; then
|
if [[ $CCACHE_DIR ]]; then
|
||||||
args+=(--mount "type=bind,source=$CCACHE_DIR,target=/.ccache")
|
args+=(--mount "type=bind,source=$CCACHE_DIR,target=/.ccache")
|
||||||
args+=(--env "CCACHE_DIR=/.ccache")
|
args+=(--env "CCACHE_DIR=/.ccache")
|
||||||
args+=(--env "CMAKE_GENERATOR="Ninja"")
|
|
||||||
fi
|
fi
|
||||||
|
if [[ $GITHUB_OUTPUT ]]; then
|
||||||
|
args+=(--mount "type=bind,source=$GITHUB_OUTPUT,target=/gh_output")
|
||||||
|
args+=(--env "GITHUB_OUTPUT=/gh_output")
|
||||||
|
fi
|
||||||
|
# shellcheck disable=2086
|
||||||
docker run "${args[@]}" $RUN_ARGS "$IMAGE_NAME" bash "$BUILD_SCRIPT" $RUN_OPTS "$@"
|
docker run "${args[@]}" $RUN_ARGS "$IMAGE_NAME" bash "$BUILD_SCRIPT" $RUN_OPTS "$@"
|
||||||
return $?
|
return $?
|
||||||
else
|
else
|
||||||
|
|
@ -162,5 +183,6 @@ function RUN ()
|
||||||
if [[ $INTERACTIVE ]]; then
|
if [[ $INTERACTIVE ]]; then
|
||||||
export BUILD_SCRIPT="-i"
|
export BUILD_SCRIPT="-i"
|
||||||
export RUN_ARGS="$RUN_ARGS -it"
|
export RUN_ARGS="$RUN_ARGS -it"
|
||||||
|
# shellcheck disable=2119
|
||||||
RUN
|
RUN
|
||||||
fi
|
fi
|
||||||
|
|
|
||||||
|
|
@ -11,9 +11,9 @@ if ! git merge-base origin/master HEAD; then
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# Check formatting using format.sh
|
# Check formatting using format.sh
|
||||||
echo "Checking your code using clang-format/cmake-format..."
|
echo "Checking your code using format.sh..."
|
||||||
|
|
||||||
diff="$(./format.sh --diff --cmake --cf-version --branch origin/master)"
|
./format.sh --color-diff --cmake --shell --print-version --branch origin/master
|
||||||
err=$?
|
err=$?
|
||||||
|
|
||||||
case $err in
|
case $err in
|
||||||
|
|
@ -28,20 +28,10 @@ case $err in
|
||||||
*** Then commit and push those changes to this branch. ***
|
*** Then commit and push those changes to this branch. ***
|
||||||
*** Check our CONTRIBUTING.md file for more details. ***
|
*** Check our CONTRIBUTING.md file for more details. ***
|
||||||
*** ***
|
*** ***
|
||||||
*** Thank you ❤️ ***
|
*** Thank you ❤️ ***
|
||||||
*** ***
|
*** ***
|
||||||
***********************************************************
|
***********************************************************
|
||||||
|
|
||||||
Used version:
|
|
||||||
${diff%%
|
|
||||||
----------
|
|
||||||
*}
|
|
||||||
|
|
||||||
The following changes should be made:
|
|
||||||
${diff#*
|
|
||||||
----------
|
|
||||||
}
|
|
||||||
|
|
||||||
Exiting...
|
Exiting...
|
||||||
EOM
|
EOM
|
||||||
exit 2
|
exit 2
|
||||||
|
|
|
||||||
|
|
@ -2,7 +2,6 @@
|
||||||
# used by the ci to rename build artifacts
|
# used by the ci to rename build artifacts
|
||||||
# renames the file to [original name][SUFFIX].[original extension]
|
# renames the file to [original name][SUFFIX].[original extension]
|
||||||
# where SUFFIX is either available in the environment or as the first arg
|
# where SUFFIX is either available in the environment or as the first arg
|
||||||
# if MAKE_ZIP is set instead a zip is made
|
|
||||||
# expected to be run in the build directory unless BUILD_DIR is set
|
# expected to be run in the build directory unless BUILD_DIR is set
|
||||||
# adds output to GITHUB_OUTPUT
|
# adds output to GITHUB_OUTPUT
|
||||||
builddir="${BUILD_DIR:=.}"
|
builddir="${BUILD_DIR:=.}"
|
||||||
|
|
@ -22,8 +21,8 @@ set -e
|
||||||
|
|
||||||
# find file
|
# find file
|
||||||
found="$(find "$builddir" -maxdepth 1 -type f -name "$findrx" -print -quit)"
|
found="$(find "$builddir" -maxdepth 1 -type f -name "$findrx" -print -quit)"
|
||||||
path="${found%/*}" # remove all after last /
|
path="${found%/*}" # remove all including first "/" from right side
|
||||||
file="${found##*/}" # remove all before last /
|
file="${found##*/}" # remove all including last "/" from left side
|
||||||
if [[ ! $file ]]; then
|
if [[ ! $file ]]; then
|
||||||
echo "::error file=$0::could not find package"
|
echo "::error file=$0::could not find package"
|
||||||
exit 1
|
exit 1
|
||||||
|
|
@ -35,21 +34,16 @@ if ! cd "$path"; then
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# set filename
|
# set filename
|
||||||
name="${file%.*}" # remove all after last .
|
name="${file%.*}" # remove all including first "." from right side
|
||||||
new_name="$name$SUFFIX."
|
new_name="$name$SUFFIX"
|
||||||
if [[ $MAKE_ZIP ]]; then
|
extension="${file##*.}" # remove all including last "." from left side
|
||||||
filename="${new_name}zip"
|
filename="$new_name.$extension"
|
||||||
echo "creating zip '$filename' from '$file'"
|
echo "renaming '$file' to '$filename'"
|
||||||
zip "$filename" "$file"
|
mv "$file" "$filename"
|
||||||
else
|
|
||||||
extension="${file##*.}" # remove all before last .
|
|
||||||
filename="$new_name$extension"
|
|
||||||
echo "renaming '$file' to '$filename'"
|
|
||||||
mv "$file" "$filename"
|
|
||||||
fi
|
|
||||||
|
|
||||||
cd "$oldpwd"
|
cd "$oldpwd"
|
||||||
relative_path="$path/$filename"
|
relative_path="$path/$filename"
|
||||||
ls -l "$relative_path"
|
ls -l "$relative_path"
|
||||||
echo "path=$relative_path" >>"$GITHUB_OUTPUT"
|
echo "path=$relative_path" >>"$GITHUB_OUTPUT"
|
||||||
echo "name=$filename" >>"$GITHUB_OUTPUT"
|
echo "name=$new_name" >>"$GITHUB_OUTPUT"
|
||||||
|
echo "fullname=$filename" >>"$GITHUB_OUTPUT"
|
||||||
|
|
|
||||||
|
|
@ -89,6 +89,8 @@ else
|
||||||
echo "'$previous' to '$TAG' ($count commits)"
|
echo "'$previous' to '$TAG' ($count commits)"
|
||||||
# --> is the markdown comment escape sequence, emojis are way better
|
# --> is the markdown comment escape sequence, emojis are way better
|
||||||
generated_list="${generated_list//-->/→}"
|
generated_list="${generated_list//-->/→}"
|
||||||
|
# Escape & to preserve it from commit message into markdown output
|
||||||
|
generated_list="${generated_list//&/\\&}"
|
||||||
body="${body//--REPLACE-WITH-GENERATED-LIST--/$generated_list}"
|
body="${body//--REPLACE-WITH-GENERATED-LIST--/$generated_list}"
|
||||||
body="${body//--REPLACE-WITH-COMMIT-COUNT--/$count}"
|
body="${body//--REPLACE-WITH-COMMIT-COUNT--/$count}"
|
||||||
body="${body//--REPLACE-WITH-PREVIOUS-RELEASE-TAG--/$previous}"
|
body="${body//--REPLACE-WITH-PREVIOUS-RELEASE-TAG--/$previous}"
|
||||||
|
|
|
||||||
|
|
@ -10,7 +10,6 @@ Available pre-compiled binaries for installation:
|
||||||
|
|
||||||
<b>Windows</b>
|
<b>Windows</b>
|
||||||
• <kbd>Windows 10+</kbd>
|
• <kbd>Windows 10+</kbd>
|
||||||
• <kbd>Windows 7+</kbd>
|
|
||||||
|
|
||||||
<b>macOS</b>
|
<b>macOS</b>
|
||||||
• <kbd>macOS 15+</kbd> <sub><i>Sequoia</i></sub> <sub>Apple M</sub>
|
• <kbd>macOS 15+</kbd> <sub><i>Sequoia</i></sub> <sub>Apple M</sub>
|
||||||
|
|
@ -18,15 +17,17 @@ Available pre-compiled binaries for installation:
|
||||||
• <kbd>macOS 13+</kbd> <sub><i>Ventura</i></sub> <sub>Intel</sub>
|
• <kbd>macOS 13+</kbd> <sub><i>Ventura</i></sub> <sub>Intel</sub>
|
||||||
|
|
||||||
<b>Linux</b>
|
<b>Linux</b>
|
||||||
|
• <kbd>Ubuntu 26.04 LTS</kbd> <sub><i>Resolute Racoon</i></sub>
|
||||||
• <kbd>Ubuntu 24.04 LTS</kbd> <sub><i>Noble Numbat</i></sub>
|
• <kbd>Ubuntu 24.04 LTS</kbd> <sub><i>Noble Numbat</i></sub>
|
||||||
• <kbd>Ubuntu 22.04 LTS</kbd> <sub><i>Jammy Jellyfish</i></sub>
|
• <kbd>Debian 13</kbd> <sub><i>Trixie</i></sub>
|
||||||
• <kbd>Debian 12</kbd> <sub><i>Bookworm</i></sub>
|
• <kbd>Debian 12</kbd> <sub><i>Bookworm</i></sub>
|
||||||
• <kbd>Debian 11</kbd> <sub><i>Bullseye</i></sub>
|
• <kbd>Fedora 44</kbd>
|
||||||
• <kbd>Fedora 42</kbd>
|
• <kbd>Fedora 43</kbd>
|
||||||
• <kbd>Fedora 41</kbd>
|
|
||||||
|
|
||||||
<sub>We are also packaged in <kbd>Arch Linux</kbd>'s <a href="https://archlinux.org/packages/extra/x86_64/cockatrice">official extra repository</a>, courtesy of @FFY00.</sub>
|
<sub>We are also packaged in <kbd>Arch Linux</kbd>'s <a href="https://archlinux.org/packages/extra/x86_64/cockatrice">official extra repository</a>, courtesy of @FFY00.</sub>
|
||||||
<sub>General Linux support is available via a <kbd>flatpak</kbd> package at <a href="https://flathub.org/apps/io.github.Cockatrice.cockatrice">Flathub</a>!</sub>
|
<sub>General Linux support is available via a <kbd>flatpak</kbd> package at <a href="https://flathub.org/apps/io.github.Cockatrice.cockatrice">Flathub</a>!</sub>
|
||||||
|
|
||||||
|
<sub>We provide a <kbd>Docker</kbd> image for "Servatrice" in <a href="https://github.com/Cockatrice/Cockatrice/pkgs/container/servatrice">GHCR</a>. You can docker pull it or use our Docker Compose files!</sub>
|
||||||
</pre>
|
</pre>
|
||||||
|
|
||||||
|
|
||||||
|
|
@ -82,7 +83,6 @@ Remove empty headers when done.
|
||||||
### Under the Hood
|
### Under the Hood
|
||||||
### Oracle
|
### Oracle
|
||||||
### Servatrice
|
### Servatrice
|
||||||
### Webatrice
|
|
||||||
|
|
||||||
</details>
|
</details>
|
||||||
|
|
||||||
|
|
|
||||||
49
.ci/resolve_latest_aqt_qt_version.sh
Executable file
|
|
@ -0,0 +1,49 @@
|
||||||
|
#!/bin/bash
|
||||||
|
|
||||||
|
# This script is used to resolve the latest patch version of Qt using aqtinstall.
|
||||||
|
# It interprets wildcards to get the latest patch version. E.g. "6.6.*" -> "6.6.3".
|
||||||
|
|
||||||
|
# This script is meant to be used by the ci enironment.
|
||||||
|
# It uses the runner's GITHUB_OUTPUT env variable.
|
||||||
|
|
||||||
|
# Usage example: .ci/resolve_latest_aqt_qt_version.sh "6.6.*"
|
||||||
|
|
||||||
|
qt_spec=$1
|
||||||
|
if [[ ! $qt_spec ]]; then
|
||||||
|
echo "usage: $0 [version]"
|
||||||
|
exit 2
|
||||||
|
fi
|
||||||
|
|
||||||
|
# If version is already specific (no wildcard), use it as-is
|
||||||
|
if [[ $qt_spec != *"*" ]]; then
|
||||||
|
echo "version $qt_spec is already resolved"
|
||||||
|
echo "version=$qt_spec" >> "$GITHUB_OUTPUT"
|
||||||
|
exit 0
|
||||||
|
fi
|
||||||
|
|
||||||
|
if ! hash aqt; then
|
||||||
|
echo "aqt could not be found, has aqtinstall been installed?"
|
||||||
|
exit 2
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Resolve latest patch
|
||||||
|
if [[ $RUNNER_OS == macOS ]]; then
|
||||||
|
if ! qt_resolved=$(aqt list-qt mac desktop --spec "$qt_spec" --latest-version); then
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
elif [[ $RUNNER_OS == Windows ]]; then
|
||||||
|
if ! qt_resolved=$(aqt list-qt windows desktop --spec "$qt_spec" --latest-version); then
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
else
|
||||||
|
echo "aqt command for $RUNNER_OS not defined."
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo "resolved $qt_spec to $qt_resolved"
|
||||||
|
if [[ ! $qt_resolved ]]; then
|
||||||
|
echo "Error: Could not resolve Qt version for $qt_spec"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo "version=$qt_resolved" >> "$GITHUB_OUTPUT"
|
||||||
25
.ci/thin_macos_qtlib.sh
Executable file
|
|
@ -0,0 +1,25 @@
|
||||||
|
#!/bin/bash
|
||||||
|
|
||||||
|
# The macos binaries from aqt are fat (universal), so we thin them to the target architecture to reduce the size of
|
||||||
|
# the packages and caches using lipo.
|
||||||
|
|
||||||
|
# This script is meant to be used by the ci enironment on macos runners only.
|
||||||
|
# It uses the runner's GITHUB_WORKSPACE env variable.
|
||||||
|
arch=$(uname -m)
|
||||||
|
nproc=$(sysctl -n hw.ncpu)
|
||||||
|
|
||||||
|
function thin() {
|
||||||
|
local libfile=$1
|
||||||
|
if [[ $(file -b --mime-type "$libfile") == application/x-mach-binary* ]]; then
|
||||||
|
echo "Processing $libfile"
|
||||||
|
lipo "$libfile" -thin "$arch" -output "$libfile"
|
||||||
|
fi
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
export -f thin # export to allow use in xargs
|
||||||
|
export arch
|
||||||
|
set -eo pipefail
|
||||||
|
|
||||||
|
echo "::group::Thinning Qt libraries to $arch using $nproc cores"
|
||||||
|
find "$GITHUB_WORKSPACE/Qt" -type f -print0 | xargs -0 -n1 -P"$nproc" -I{} bash -c "thin '{}'"
|
||||||
|
echo "::endgroup::"
|
||||||
|
|
@ -10,5 +10,5 @@ Last changes are based on commit {{ .commit }}.
|
||||||
*This PR is automatically generated and updated by the workflow at `.github/workflows/translations-push.yml`. Review [action runs][2].*<br>
|
*This PR is automatically generated and updated by the workflow at `.github/workflows/translations-push.yml`. Review [action runs][2].*<br>
|
||||||
*After merging, all changes to the source language are available for translation at [Transifex][1] shortly.*
|
*After merging, all changes to the source language are available for translation at [Transifex][1] shortly.*
|
||||||
|
|
||||||
[1]: https://app.transifex.com/cockatrice/cockatrice/
|
[1]: https://explore.transifex.com/cockatrice/cockatrice/
|
||||||
[2]: https://github.com/Cockatrice/Cockatrice/actions/workflows/translations-push.yml?query=branch%3Amaster
|
[2]: https://github.com/Cockatrice/Cockatrice/actions/workflows/translations-push.yml?query=branch%3Amaster
|
||||||
|
|
|
||||||
|
|
@ -3,7 +3,9 @@ AccessModifierOffset: -4
|
||||||
ColumnLimit: 120
|
ColumnLimit: 120
|
||||||
---
|
---
|
||||||
Language: Cpp
|
Language: Cpp
|
||||||
BreakBeforeBraces: Custom
|
AllowAllParametersOfDeclarationOnNextLine: false
|
||||||
|
AllowShortFunctionsOnASingleLine: None
|
||||||
|
BinPackParameters: false
|
||||||
BraceWrapping:
|
BraceWrapping:
|
||||||
AfterClass: true
|
AfterClass: true
|
||||||
AfterControlStatement: false
|
AfterControlStatement: false
|
||||||
|
|
@ -18,13 +20,14 @@ BraceWrapping:
|
||||||
SplitEmptyFunction: true
|
SplitEmptyFunction: true
|
||||||
SplitEmptyRecord: true
|
SplitEmptyRecord: true
|
||||||
SplitEmptyNamespace: true
|
SplitEmptyNamespace: true
|
||||||
AllowShortFunctionsOnASingleLine: None
|
BreakBeforeBraces: Custom
|
||||||
BinPackParameters: false
|
|
||||||
AllowAllParametersOfDeclarationOnNextLine: false
|
|
||||||
IndentCaseLabels: true
|
|
||||||
PointerAlignment: Right
|
|
||||||
SortIncludes: true
|
|
||||||
IncludeBlocks: Regroup
|
IncludeBlocks: Regroup
|
||||||
|
IndentCaseLabels: true
|
||||||
|
InsertBraces: true
|
||||||
|
PointerAlignment: Right
|
||||||
|
RemoveSemicolon: true
|
||||||
|
SortIncludes: true
|
||||||
|
StatementAttributeLikeMacros: [emit]
|
||||||
---
|
---
|
||||||
Language: Proto
|
Language: Proto
|
||||||
AllowShortFunctionsOnASingleLine: None
|
AllowShortFunctionsOnASingleLine: None
|
||||||
|
|
|
||||||
18
.github/CONTRIBUTING.md
vendored
|
|
@ -1,3 +1,5 @@
|
||||||
|
<!--! @page contributing Contributing -->
|
||||||
|
|
||||||
[Introduction](#contributing-to-cockatrice) | [Code Style Guide](
|
[Introduction](#contributing-to-cockatrice) | [Code Style Guide](
|
||||||
#code-style-guide) | [Translations](#translations) | [Release Management](
|
#code-style-guide) | [Translations](#translations) | [Release Management](
|
||||||
#release-management)
|
#release-management)
|
||||||
|
|
@ -207,6 +209,16 @@ nowadays and clean it up for you.
|
||||||
Lines should be 120 characters or less. Please break up lines that are too long
|
Lines should be 120 characters or less. Please break up lines that are too long
|
||||||
into smaller parts, for example at spaces or after opening a brace.
|
into smaller parts, for example at spaces or after opening a brace.
|
||||||
|
|
||||||
|
### Documentation Comments ###
|
||||||
|
|
||||||
|
Use [Doxygen](https://www.doxygen.nl/) for code documentation:
|
||||||
|
|
||||||
|
- **Doc blocks**: Use `/** @brief Description */` (Javadoc-style), not `///`
|
||||||
|
- **Member comments**: Use trailing `///<` for inline member documentation
|
||||||
|
- **TODOs**: Use `//! \todo Description` (Qt-style), Doxygen collects them into a Todo List
|
||||||
|
(uses [Qt-style comments](https://www.doxygen.nl/manual/docblocks.html) with
|
||||||
|
Doxygen's [\todo command](https://www.doxygen.nl/manual/commands.html#cmdtodo))
|
||||||
|
|
||||||
### Memory Management ###
|
### Memory Management ###
|
||||||
|
|
||||||
New code should be written using references over pointers and stack allocation
|
New code should be written using references over pointers and stack allocation
|
||||||
|
|
@ -459,7 +471,11 @@ revoke the tag by doing the following:
|
||||||
git push --delete upstream $TAG_NAME
|
git push --delete upstream $TAG_NAME
|
||||||
git tag -d $TAG_NAME
|
git tag -d $TAG_NAME
|
||||||
```
|
```
|
||||||
You can also do this on GitHub, you'll also want to delete the false release.
|
You can also do this on GitHub.
|
||||||
|
|
||||||
|
> [!NOTE]
|
||||||
|
> If you want to push a new release to replace it immediately with the same
|
||||||
|
> name you have to delete the automatically created release first!
|
||||||
|
|
||||||
In the first lines of [CMakeLists.txt](
|
In the first lines of [CMakeLists.txt](
|
||||||
https://github.com/Cockatrice/Cockatrice/blob/master/CMakeLists.txt)
|
https://github.com/Cockatrice/Cockatrice/blob/master/CMakeLists.txt)
|
||||||
|
|
|
||||||
7
.github/ISSUE_TEMPLATE/config.yml
vendored
|
|
@ -1,9 +1,12 @@
|
||||||
blank_issues_enabled: false
|
blank_issues_enabled: false
|
||||||
contact_links:
|
contact_links:
|
||||||
- name: 💬 Discord Community (Get help with server issues, e.g. Login)
|
- name: 💬 Discord Community (Get help with server issues, e.g. Login)
|
||||||
url: https://discord.gg/3Z9yzmA
|
url: https://discord.com/invite/3Z9yzmA
|
||||||
about: Need help with using the client? Want to find some games? Try the Discord server!
|
about: Need help with using the client? Want to find some games? Try the Discord server!
|
||||||
- name: 🌐 Translations (Help improve the localization of the app)
|
- name: 🌐 Translations (Help improve the localization of the app)
|
||||||
url: https://www.transifex.com/cockatrice/cockatrice/
|
url: https://explore.transifex.com/cockatrice/cockatrice/
|
||||||
# it is not possible to add a link to the wiki to this description
|
# it is not possible to add a link to the wiki to this description
|
||||||
about: For more information and guidance check our Translation FAQ on our wiki!
|
about: For more information and guidance check our Translation FAQ on our wiki!
|
||||||
|
- name: 📖 Code Documentation
|
||||||
|
url: https://cockatrice.github.io/docs/
|
||||||
|
about: Helpful source focusing on developers, but there are also references for users!
|
||||||
|
|
|
||||||
34
.github/dependabot.yml
vendored
|
|
@ -2,19 +2,21 @@
|
||||||
|
|
||||||
version: 2
|
version: 2
|
||||||
updates:
|
updates:
|
||||||
# # Enable version updates for git submodules
|
# Enable version updates for git submodules
|
||||||
# Not yet possible to bump only on tags or releases, see:
|
# If SemVer is used, updates will happen to new releases only (not HEAD)
|
||||||
# https://github.com/dependabot/dependabot-core/issues/1639
|
# https://github.com/dependabot/dependabot-core/issues/1639
|
||||||
# https://github.com/dependabot/dependabot-core/issues/2192
|
# https://github.com/dependabot/dependabot-core/issues/2192
|
||||||
# Alternative: Action that updates submodule and can be manually run on demand (workflow_dispatch)
|
- package-ecosystem: "gitsubmodule"
|
||||||
# - package-ecosystem: "gitsubmodule"
|
# Look for `.gitmodules` in the `root` directory
|
||||||
# # Look for `.gitmodules` in the `root` directory
|
directory: "/"
|
||||||
# directory: "/"
|
ignore:
|
||||||
# # Check for updates once a month
|
# Ignore updates for vcpkg (Bump to latest tag not working (no SemVer used)
|
||||||
# schedule:
|
- dependency-name: "vcpkg"
|
||||||
# interval: "monthly"
|
# Check for updates once a month
|
||||||
# # Limit the amout of open PR's (default = 5, disabled = 0, security updates are not impacted)
|
schedule:
|
||||||
# open-pull-requests-limit: 1
|
interval: "monthly"
|
||||||
|
# Limit the amout of open PR's (default = 5, disabled = 0, security updates are not impacted)
|
||||||
|
open-pull-requests-limit: 2
|
||||||
|
|
||||||
# # Enable version updates for Docker
|
# # Enable version updates for Docker
|
||||||
# Not yet possible to bump from one LTS version to the next and skip others, see:
|
# Not yet possible to bump from one LTS version to the next and skip others, see:
|
||||||
|
|
@ -37,13 +39,3 @@ updates:
|
||||||
interval: "weekly"
|
interval: "weekly"
|
||||||
# Limit the amout of open PR's (default = 5, disabled = 0, security updates are not impacted)
|
# Limit the amout of open PR's (default = 5, disabled = 0, security updates are not impacted)
|
||||||
open-pull-requests-limit: 2
|
open-pull-requests-limit: 2
|
||||||
|
|
||||||
# # Enable version updates for npm
|
|
||||||
# - package-ecosystem: "npm"
|
|
||||||
# # Look for `package.json` and `lock` files in the `webclient` subdirectory
|
|
||||||
# directory: "/webclient"
|
|
||||||
# # Check the npm registry for updates once a week
|
|
||||||
# schedule:
|
|
||||||
# interval: "weekly"
|
|
||||||
# # Limit the amout of open PR's (default = 5, disabled = 0, security updates are not impacted)
|
|
||||||
# open-pull-requests-limit: 5
|
|
||||||
|
|
|
||||||
630
.github/workflows/desktop-build.yml
vendored
|
|
@ -1,39 +1,54 @@
|
||||||
name: Build Desktop
|
name: Build Desktop
|
||||||
|
|
||||||
|
permissions:
|
||||||
|
actions: write # needed to delete entries in GHA cache (update ccache)
|
||||||
|
attestations: write # needed to persist the attestation.
|
||||||
|
contents: write
|
||||||
|
id-token: write # needed for signing certificate in attestation
|
||||||
|
|
||||||
on:
|
on:
|
||||||
push:
|
push:
|
||||||
branches:
|
branches:
|
||||||
- master
|
- master
|
||||||
paths-ignore:
|
paths:
|
||||||
- '**.md'
|
- '*/**' # matches all files not in root
|
||||||
- 'webclient/**'
|
- '!**.md'
|
||||||
- '.github/workflows/web-*.yml'
|
- '!.github/**'
|
||||||
- '.github/workflows/translations-*.yml'
|
- '!.tx/**'
|
||||||
- '.github/workflows/docker-release.yml'
|
- '!doc/**'
|
||||||
|
- '.github/workflows/desktop-build.yml'
|
||||||
|
- 'CMakeLists.txt'
|
||||||
|
- 'vcpkg.json'
|
||||||
|
- 'vcpkg' # needed to match submodule bumps (gitlink)
|
||||||
tags:
|
tags:
|
||||||
- '*'
|
- '*'
|
||||||
pull_request:
|
pull_request:
|
||||||
paths-ignore:
|
paths:
|
||||||
- '**.md'
|
- '*/**' # matches all files not in root
|
||||||
- 'webclient/**'
|
- '!**.md'
|
||||||
- '.github/workflows/web-*.yml'
|
- '!.github/**'
|
||||||
- '.github/workflows/translations-*.yml'
|
- '!.tx/**'
|
||||||
|
- '!doc/**'
|
||||||
|
- '.github/workflows/desktop-build.yml'
|
||||||
|
- 'CMakeLists.txt'
|
||||||
|
- 'vcpkg.json'
|
||||||
|
- 'vcpkg' # needed to match submodule bumps (gitlink)
|
||||||
|
|
||||||
# Cancel earlier, unfinished runs of this workflow on the same branch (unless on master)
|
# Cancel earlier, unfinished runs of this workflow on the same branch (unless on release)
|
||||||
concurrency:
|
concurrency:
|
||||||
group: "${{ github.workflow }} @ ${{ github.ref_name }}"
|
group: "${{ github.workflow }} @ ${{ github.ref_name }}"
|
||||||
cancel-in-progress: ${{ github.ref_name != 'master' }}
|
cancel-in-progress: ${{ github.ref_type != 'tag' }}
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
configure:
|
configure:
|
||||||
name: Configure
|
name: Configure
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-slim
|
||||||
outputs:
|
outputs:
|
||||||
tag: ${{steps.configure.outputs.tag}}
|
tag: ${{ steps.configure.outputs.tag }}
|
||||||
sha: ${{steps.configure.outputs.sha}}
|
sha: ${{ steps.configure.outputs.sha }}
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- name: Configure
|
- name: "Configure"
|
||||||
id: configure
|
id: configure
|
||||||
shell: bash
|
shell: bash
|
||||||
run: |
|
run: |
|
||||||
|
|
@ -49,238 +64,423 @@ jobs:
|
||||||
fi
|
fi
|
||||||
echo "sha=$sha" >>"$GITHUB_OUTPUT"
|
echo "sha=$sha" >>"$GITHUB_OUTPUT"
|
||||||
|
|
||||||
- name: Checkout
|
- name: "Checkout"
|
||||||
if: steps.configure.outputs.tag != null
|
if: steps.configure.outputs.tag != null
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v6
|
||||||
with:
|
with:
|
||||||
fetch-depth: 0
|
fetch-depth: 0 # fetch all history for all branches and tags
|
||||||
|
|
||||||
- name: Prepare release parameters
|
- name: "Prepare release parameters"
|
||||||
id: prepare
|
id: prepare
|
||||||
if: steps.configure.outputs.tag != null
|
if: steps.configure.outputs.tag != null
|
||||||
shell: bash
|
shell: bash
|
||||||
env:
|
env:
|
||||||
TAG: ${{steps.configure.outputs.tag}}
|
TAG: ${{ steps.configure.outputs.tag }}
|
||||||
run: .ci/prep_release.sh
|
run: .ci/prep_release.sh
|
||||||
|
|
||||||
- name: Create release
|
- name: "Create release"
|
||||||
if: steps.configure.outputs.tag != null
|
if: steps.configure.outputs.tag != null
|
||||||
id: create_release
|
id: create_release
|
||||||
shell: bash
|
shell: bash
|
||||||
env:
|
env:
|
||||||
GH_TOKEN: ${{github.token}}
|
GH_TOKEN: ${{ github.token }}
|
||||||
tag_name: ${{steps.configure.outputs.tag}}
|
tag_name: ${{ steps.configure.outputs.tag }}
|
||||||
target: ${{steps.configure.outputs.sha}}
|
target: ${{ steps.configure.outputs.sha }}
|
||||||
release_name: ${{steps.prepare.outputs.title}}
|
release_name: ${{ steps.prepare.outputs.title }}
|
||||||
body_path: ${{steps.prepare.outputs.body_path}}
|
body_path: ${{ steps.prepare.outputs.body_path }}
|
||||||
prerelease: ${{steps.prepare.outputs.is_beta}}
|
prerelease: ${{ steps.prepare.outputs.is_beta }}
|
||||||
run: |
|
run: |
|
||||||
if [[ $prerelease == yes ]]; then
|
args=()
|
||||||
args="--prerelease"
|
[[ $prerelease == yes ]] && args+=(--prerelease)
|
||||||
fi
|
|
||||||
gh release create "$tag_name" --draft --verify-tag $args \
|
gh release create "$tag_name" --verify-tag --draft "${args[@]}" \
|
||||||
--target "$target" --title "$release_name" \
|
--target "$target" \
|
||||||
--notes-file "$body_path"
|
--title "$release_name" \
|
||||||
|
--notes-file "$body_path"
|
||||||
|
|
||||||
build-linux:
|
build-linux:
|
||||||
strategy:
|
strategy:
|
||||||
fail-fast: false
|
fail-fast: false
|
||||||
matrix:
|
matrix:
|
||||||
# These names correspond to the files in ".ci/$distro$version"
|
# The files in ".ci/$distro$version" correspond to the values given here
|
||||||
include:
|
include:
|
||||||
- distro: Arch
|
- distro: Arch
|
||||||
package: skip # We are packaged in Arch already
|
|
||||||
allow-failure: yes
|
allow-failure: yes
|
||||||
|
package: skip # We are packaged in Arch already
|
||||||
|
|
||||||
- distro: Debian
|
- distro: Servatrice_Debian
|
||||||
version: 11
|
version: 12
|
||||||
|
|
||||||
package: DEB
|
package: DEB
|
||||||
test: skip # Running tests on all distros is superfluous
|
server_only: yes
|
||||||
|
test: skip
|
||||||
|
|
||||||
- distro: Debian
|
- distro: Debian
|
||||||
version: 12
|
version: 12
|
||||||
|
|
||||||
|
package: DEB
|
||||||
|
test: skip # Running tests on all distros is superfluous
|
||||||
|
|
||||||
|
- distro: Debian
|
||||||
|
version: 13
|
||||||
|
|
||||||
package: DEB
|
package: DEB
|
||||||
|
|
||||||
- distro: Fedora
|
- distro: Fedora
|
||||||
version: 41
|
version: 43
|
||||||
|
|
||||||
package: RPM
|
package: RPM
|
||||||
test: skip # Running tests on all distros is superfluous
|
test: skip # Running tests on all distros is superfluous
|
||||||
|
|
||||||
- distro: Fedora
|
- distro: Fedora
|
||||||
version: 42
|
version: 44
|
||||||
|
|
||||||
package: RPM
|
package: RPM
|
||||||
|
|
||||||
- distro: Ubuntu
|
|
||||||
version: 22.04
|
|
||||||
package: DEB
|
|
||||||
test: skip # Running tests on all distros is superfluous
|
|
||||||
|
|
||||||
- distro: Ubuntu
|
- distro: Ubuntu
|
||||||
version: 24.04
|
version: 24.04
|
||||||
|
|
||||||
|
package: DEB
|
||||||
|
test: skip # Running tests on all distros is superfluous
|
||||||
|
|
||||||
|
- distro: Ubuntu
|
||||||
|
version: 26.04
|
||||||
|
|
||||||
package: DEB
|
package: DEB
|
||||||
|
|
||||||
name: ${{matrix.distro}} ${{matrix.version}}
|
name: ${{ matrix.distro }} ${{ matrix.version }}
|
||||||
needs: configure
|
needs: configure
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
continue-on-error: ${{matrix.allow-failure == 'yes'}}
|
continue-on-error: ${{ matrix.allow-failure == 'yes' }}
|
||||||
|
timeout-minutes: 70
|
||||||
env:
|
env:
|
||||||
NAME: ${{matrix.distro}}${{matrix.version}}
|
CACHE: ${{ github.workspace }}/.cache/${{ matrix.distro }}${{ matrix.version }} # directory for caching docker image and ccache
|
||||||
CACHE: /tmp/${{matrix.distro}}${{matrix.version}}-cache # ${{runner.temp}} does not work?
|
CCACHE_EVICTION_AGE: 7d
|
||||||
# Cache size over the entire repo is 10Gi:
|
CCACHE_SIZE: 550M # space of all repo is 10Gi: https://docs.github.com/en/actions/using-workflows/caching-dependencies-to-speed-up-workflows#usage-limits-and-eviction-policy
|
||||||
# https://docs.github.com/en/actions/using-workflows/caching-dependencies-to-speed-up-workflows#usage-limits-and-eviction-policy
|
|
||||||
CCACHE_SIZE: 500M
|
|
||||||
CMAKE_GENERATOR: 'Ninja'
|
CMAKE_GENERATOR: 'Ninja'
|
||||||
|
NAME: ${{ matrix.distro }}${{ matrix.version }}
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout
|
- name: "Checkout"
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v6
|
||||||
|
|
||||||
- name: Build ${{matrix.distro}} ${{matrix.version}} Docker image
|
- name: "Restore compiler cache (ccache)"
|
||||||
shell: bash
|
|
||||||
run: source .ci/docker.sh --build
|
|
||||||
|
|
||||||
- name: Restore compiler cache (ccache)
|
|
||||||
id: ccache_restore
|
id: ccache_restore
|
||||||
uses: actions/cache/restore@v4
|
uses: actions/cache/restore@v5
|
||||||
env:
|
env:
|
||||||
BRANCH_NAME: ${{ github.head_ref || github.ref_name }}
|
BRANCH_NAME: ${{ github.head_ref || github.ref_name }}
|
||||||
with:
|
with:
|
||||||
path: ${{env.CACHE}}
|
key: ccache-${{ matrix.distro }}${{ matrix.version }}-${{ env.BRANCH_NAME }}
|
||||||
key: ccache-${{matrix.distro}}${{matrix.version}}-${{env.BRANCH_NAME}}
|
path: ${{ env.CACHE }}
|
||||||
restore-keys: ccache-${{matrix.distro}}${{matrix.version}}-
|
restore-keys: ccache-${{ matrix.distro }}${{ matrix.version }}-
|
||||||
|
|
||||||
- name: Build debug and test
|
- name: "Build ${{ matrix.distro }} ${{ matrix.version }} Docker image"
|
||||||
|
shell: bash
|
||||||
|
run: source .ci/docker.sh --build
|
||||||
|
|
||||||
|
- name: "Build debug and test"
|
||||||
if: matrix.test != 'skip'
|
if: matrix.test != 'skip'
|
||||||
shell: bash
|
shell: bash
|
||||||
env:
|
|
||||||
CMAKE_GENERATOR: '${{env.CMAKE_GENERATOR}}'
|
|
||||||
run: |
|
run: |
|
||||||
source .ci/docker.sh
|
source .ci/docker.sh
|
||||||
RUN --server --debug --test --ccache "$CCACHE_SIZE"
|
RUN --server --debug --test --ccache "$CCACHE_SIZE" \
|
||||||
|
--cmake-generator "$CMAKE_GENERATOR"
|
||||||
|
|
||||||
- name: Build release package
|
- name: "Build release package"
|
||||||
id: build
|
id: build
|
||||||
if: matrix.package != 'skip'
|
if: matrix.package != 'skip'
|
||||||
shell: bash
|
shell: bash
|
||||||
env:
|
env:
|
||||||
BUILD_DIR: build
|
SUFFIX: '-${{ matrix.distro }}${{ matrix.version }}'
|
||||||
SUFFIX: '-${{matrix.distro}}${{matrix.version}}'
|
package: '${{ matrix.package }}'
|
||||||
type: '${{matrix.package}}'
|
server_only: '${{ matrix.server_only }}'
|
||||||
CMAKE_GENERATOR: '${{env.CMAKE_GENERATOR}}'
|
|
||||||
run: |
|
run: |
|
||||||
source .ci/docker.sh
|
source .ci/docker.sh
|
||||||
RUN --server --release --package "$type" --dir "$BUILD_DIR" \
|
args=()
|
||||||
--ccache "$CCACHE_SIZE"
|
[[ $server_only == yes ]] && args+=(--no-client)
|
||||||
.ci/name_build.sh
|
[[ $GITHUB_REF == "refs/heads/master" ]] && args+=(--evict-ccache "$CCACHE_EVICTION_AGE")
|
||||||
|
args+=(--ccache "$CCACHE_SIZE")
|
||||||
|
args+=(--cmake-generator "$CMAKE_GENERATOR")
|
||||||
|
args+=(--suffix "$SUFFIX")
|
||||||
|
|
||||||
|
RUN --server --release --package "$package" "${args[@]}"
|
||||||
|
|
||||||
- name: Save compiler cache (ccache)
|
# Delete used cache to emulate a ccache update. See https://github.com/actions/cache/issues/342
|
||||||
|
- name: "Delete remote compiler cache (ccache)"
|
||||||
|
if: github.ref == 'refs/heads/master' && steps.ccache_restore.outputs.cache-hit
|
||||||
|
continue-on-error: true
|
||||||
|
env:
|
||||||
|
GH_TOKEN: ${{ github.token }}
|
||||||
|
run: |
|
||||||
|
if gh cache delete --repo ${{ github.repository }} ${{ steps.ccache_restore.outputs.cache-primary-key }}; then
|
||||||
|
echo "Cache deleted successfully"
|
||||||
|
fi
|
||||||
|
|
||||||
|
- name: "Save updated compiler cache (ccache)"
|
||||||
if: github.ref == 'refs/heads/master'
|
if: github.ref == 'refs/heads/master'
|
||||||
uses: actions/cache/save@v4
|
uses: actions/cache/save@v5
|
||||||
with:
|
with:
|
||||||
path: ${{env.CACHE}}
|
|
||||||
key: ${{ steps.ccache_restore.outputs.cache-primary-key }}
|
key: ${{ steps.ccache_restore.outputs.cache-primary-key }}
|
||||||
|
path: ${{ env.CACHE }}
|
||||||
|
|
||||||
- name: Upload artifact
|
- name: "Upload artifact"
|
||||||
|
id: upload_artifact
|
||||||
if: matrix.package != 'skip'
|
if: matrix.package != 'skip'
|
||||||
uses: actions/upload-artifact@v4
|
uses: actions/upload-artifact@v7
|
||||||
with:
|
with:
|
||||||
name: ${{matrix.distro}}${{matrix.version}}-package
|
archive: false
|
||||||
path: ${{steps.build.outputs.path}}
|
|
||||||
if-no-files-found: error
|
if-no-files-found: error
|
||||||
|
path: ${{ steps.build.outputs.path }}
|
||||||
|
|
||||||
- name: Upload to release
|
- name: "Upload to release"
|
||||||
|
id: upload_release
|
||||||
if: matrix.package != 'skip' && needs.configure.outputs.tag != null
|
if: matrix.package != 'skip' && needs.configure.outputs.tag != null
|
||||||
shell: bash
|
shell: bash
|
||||||
env:
|
env:
|
||||||
GH_TOKEN: ${{github.token}}
|
asset_name: ${{ steps.build.outputs.fullname }}
|
||||||
tag_name: ${{needs.configure.outputs.tag}}
|
asset_path: ${{ steps.build.outputs.path }}
|
||||||
asset_path: ${{steps.build.outputs.path}}
|
GH_TOKEN: ${{ github.token }}
|
||||||
asset_name: ${{steps.build.outputs.name}}
|
tag_name: ${{ needs.configure.outputs.tag }}
|
||||||
run: gh release upload "$tag_name" "$asset_path#$asset_name"
|
run: gh release upload "$tag_name" "$asset_path#$asset_name"
|
||||||
|
|
||||||
build-macos:
|
- name: "Attest binary provenance"
|
||||||
|
id: attestation
|
||||||
|
if: steps.upload_release.outcome == 'success'
|
||||||
|
uses: actions/attest@v4
|
||||||
|
with:
|
||||||
|
show-summary: false
|
||||||
|
subject-path: ${{ steps.build.outputs.path }}
|
||||||
|
|
||||||
|
- name: "Verify binary attestation"
|
||||||
|
if: steps.attestation.outcome == 'success'
|
||||||
|
shell: bash
|
||||||
|
env:
|
||||||
|
GH_TOKEN: ${{ github.token }}
|
||||||
|
run: gh attestation verify "${{ steps.build.outputs.path }}" --repo Cockatrice/Cockatrice
|
||||||
|
|
||||||
|
build-vcpkg:
|
||||||
strategy:
|
strategy:
|
||||||
fail-fast: false
|
fail-fast: false
|
||||||
matrix:
|
matrix:
|
||||||
include:
|
include:
|
||||||
- target: 13
|
- os: macOS
|
||||||
|
target: 13
|
||||||
|
runner: macos-15-intel
|
||||||
|
|
||||||
|
ccache_eviction_age: 7d
|
||||||
|
cmake_generator: Ninja
|
||||||
|
make_package: 1
|
||||||
|
override_target: 13
|
||||||
|
package_suffix: "-macOS13_Intel"
|
||||||
|
qt_version: 6.11.0
|
||||||
|
qt_arch: clang_64
|
||||||
|
qt_modules: qtimageformats qtmultimedia qtwebsockets
|
||||||
soc: Intel
|
soc: Intel
|
||||||
os: macos-13
|
|
||||||
xcode: "14.3.1"
|
|
||||||
type: Release
|
type: Release
|
||||||
make_package: 1
|
use_ccache: 1
|
||||||
|
xcode: "16.4"
|
||||||
|
|
||||||
- target: 14
|
- os: macOS
|
||||||
|
target: 14
|
||||||
|
runner: macos-14
|
||||||
|
|
||||||
|
ccache_eviction_age: 7d
|
||||||
|
cmake_generator: Ninja
|
||||||
|
make_package: 1
|
||||||
|
package_suffix: "-macOS14"
|
||||||
|
qt_version: 6.11.0
|
||||||
|
qt_arch: clang_64
|
||||||
|
qt_modules: qtimageformats qtmultimedia qtwebsockets
|
||||||
soc: Apple
|
soc: Apple
|
||||||
os: macos-14
|
type: Release
|
||||||
|
use_ccache: 1
|
||||||
xcode: "15.4"
|
xcode: "15.4"
|
||||||
type: Release
|
|
||||||
make_package: 1
|
|
||||||
|
|
||||||
- target: 15
|
- os: macOS
|
||||||
soc: Apple
|
target: 15
|
||||||
os: macos-15
|
runner: macos-15
|
||||||
xcode: "16.2"
|
|
||||||
type: Release
|
ccache_eviction_age: 7d
|
||||||
|
cmake_generator: Ninja
|
||||||
make_package: 1
|
make_package: 1
|
||||||
|
package_suffix: "-macOS15"
|
||||||
- target: 15
|
qt_version: 6.11.0
|
||||||
|
qt_arch: clang_64
|
||||||
|
qt_modules: qtimageformats qtmultimedia qtwebsockets
|
||||||
|
soc: Apple
|
||||||
|
type: Release
|
||||||
|
use_ccache: 1
|
||||||
|
xcode: "16.4"
|
||||||
|
|
||||||
|
- os: macOS
|
||||||
|
target: 15
|
||||||
|
runner: macos-15
|
||||||
|
|
||||||
|
ccache_eviction_age: 7d
|
||||||
|
cmake_generator: Ninja
|
||||||
|
qt_version: 6.11.0
|
||||||
|
qt_arch: clang_64
|
||||||
|
qt_modules: qtimageformats qtmultimedia qtwebsockets
|
||||||
soc: Apple
|
soc: Apple
|
||||||
os: macos-15
|
|
||||||
xcode: "16.2"
|
|
||||||
type: Debug
|
type: Debug
|
||||||
|
use_ccache: 1
|
||||||
|
xcode: "16.4"
|
||||||
|
|
||||||
name: macOS ${{matrix.target}}${{ matrix.soc == 'Intel' && ' Intel' || '' }}${{ matrix.type == 'Debug' && ' Debug' || '' }}
|
- os: Windows
|
||||||
|
target: 10
|
||||||
|
runner: windows-2025
|
||||||
|
|
||||||
|
cmake_generator: "Visual Studio 17 2022"
|
||||||
|
cmake_generator_platform: x64
|
||||||
|
make_package: 1
|
||||||
|
package_suffix: "-Win10"
|
||||||
|
qt_version: 6.11.0
|
||||||
|
qt_arch: win64_msvc2022_64
|
||||||
|
qt_modules: qtimageformats qtmultimedia qtwebsockets
|
||||||
|
type: Release
|
||||||
|
|
||||||
|
name: ${{ matrix.os }} ${{ matrix.target }}${{ matrix.soc == 'Intel' && ' Intel' || '' }}${{ matrix.type == 'Debug' && ' Debug' || '' }}
|
||||||
needs: configure
|
needs: configure
|
||||||
runs-on: ${{matrix.os}}
|
runs-on: ${{ matrix.runner }}
|
||||||
continue-on-error: ${{matrix.allow-failure == 'yes'}}
|
timeout-minutes: 100
|
||||||
env:
|
env:
|
||||||
DEVELOPER_DIR:
|
CCACHE_DIR: ${{ github.workspace }}/.cache/
|
||||||
/Applications/Xcode_${{matrix.xcode}}.app/Contents/Developer
|
CCACHE_SIZE: 550M # space of all repo is 10Gi: https://docs.github.com/en/actions/using-workflows/caching-dependencies-to-speed-up-workflows#usage-limits-and-eviction-policy
|
||||||
CMAKE_GENERATOR: 'Ninja'
|
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout
|
- name: "Checkout"
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v6
|
||||||
|
with:
|
||||||
|
submodules: recursive
|
||||||
|
|
||||||
- name: Install dependencies using Homebrew
|
- name: "[Windows] Add msbuild to PATH"
|
||||||
shell: bash
|
if: matrix.os == 'Windows'
|
||||||
# CMake cannot find the MySQL connector
|
id: add-msbuild
|
||||||
# Neither of these works: mariadb-connector-c mysql-connector-c++
|
uses: microsoft/setup-msbuild@v3
|
||||||
|
with:
|
||||||
|
msbuild-architecture: x64
|
||||||
|
|
||||||
|
- name: "[macOS] Setup ccache"
|
||||||
|
if: matrix.os == 'macOS' && matrix.use_ccache == 1
|
||||||
|
run: brew install ccache
|
||||||
|
|
||||||
|
- name: "[macOS] Restore compiler cache (ccache)"
|
||||||
|
if: matrix.os == 'macOS' && matrix.use_ccache == 1
|
||||||
|
id: ccache_restore
|
||||||
|
uses: actions/cache/restore@v5
|
||||||
env:
|
env:
|
||||||
HOMEBREW_NO_INSTALLED_DEPENDENTS_CHECK: 1
|
BRANCH_NAME: ${{ github.head_ref || github.ref_name }}
|
||||||
run: |
|
with:
|
||||||
brew update
|
key: ccache-${{ matrix.runner }}-${{ matrix.soc }}-${{ matrix.type }}-${{ env.BRANCH_NAME }}
|
||||||
brew install protobuf qt --force-bottle
|
path: ${{ env.CCACHE_DIR }}
|
||||||
|
restore-keys: ccache-${{ matrix.runner }}-${{ matrix.soc }}-${{ matrix.type }}-
|
||||||
|
|
||||||
- name: Build & Sign on Xcode ${{matrix.xcode}}
|
- name: "Install aqtinstall"
|
||||||
|
run: pipx install aqtinstall
|
||||||
|
|
||||||
|
# Resolve given wildcard versions (e.g. Qt 6.6.*) to latest version via aqtinstall to avoid stale caches on new releases
|
||||||
|
- name: "Resolve latest Qt patch version"
|
||||||
|
id: resolve_qt_version
|
||||||
shell: bash
|
shell: bash
|
||||||
|
run: .ci/resolve_latest_aqt_qt_version.sh "${{ matrix.qt_version }}"
|
||||||
|
|
||||||
|
- name: "[macOS] Restore thin Qt ${{ steps.resolve_qt_version.outputs.version }} libraries"
|
||||||
|
if: matrix.os == 'macOS'
|
||||||
|
id: restore_qt
|
||||||
|
uses: actions/cache/restore@v5
|
||||||
|
with:
|
||||||
|
key: thin-qt-macos-${{ matrix.soc }}-${{ steps.resolve_qt_version.outputs.version }}
|
||||||
|
path: ${{ github.workspace }}/Qt
|
||||||
|
|
||||||
|
# Using jurplel/install-qt-action to install Qt without using brew
|
||||||
|
# Qt build using vcpkg either just fails or takes too long to build
|
||||||
|
- name: "[macOS] Install fat Qt ${{ steps.resolve_qt_version.outputs.version }}"
|
||||||
|
if: matrix.os == 'macOS' && steps.restore_qt.outputs.cache-hit != 'true'
|
||||||
|
uses: jurplel/install-qt-action@v4
|
||||||
|
with:
|
||||||
|
arch: ${{ matrix.qt_arch }}
|
||||||
|
cache: false
|
||||||
|
dir: ${{ github.workspace }}
|
||||||
|
modules: ${{ matrix.qt_modules }}
|
||||||
|
version: ${{ steps.resolve_qt_version.outputs.version }}
|
||||||
|
|
||||||
|
- name: "[macOS] Create thin Qt libraries"
|
||||||
|
if: matrix.os == 'macOS' && steps.restore_qt.outputs.cache-hit != 'true'
|
||||||
|
run: .ci/thin_macos_qtlib.sh
|
||||||
|
|
||||||
|
- name: "[macOS] Cache thin Qt libraries"
|
||||||
|
if: matrix.os == 'macOS' && steps.restore_qt.outputs.cache-hit != 'true'
|
||||||
|
uses: actions/cache/save@v5
|
||||||
|
with:
|
||||||
|
key: thin-qt-macos-${{ matrix.soc }}-${{ steps.resolve_qt_version.outputs.version }}
|
||||||
|
path: ${{ github.workspace }}/Qt
|
||||||
|
|
||||||
|
- name: "[Windows] Install Qt ${{ matrix.qt_version }}"
|
||||||
|
if: matrix.os == 'Windows'
|
||||||
|
uses: jurplel/install-qt-action@v4
|
||||||
|
with:
|
||||||
|
# Qt 6.11.0 only works with aqtinstall directly from git until aqtinstall 3.4 is released
|
||||||
|
aqtsource: git+https://github.com/miurahr/aqtinstall.git
|
||||||
|
arch: ${{ matrix.qt_arch }}
|
||||||
|
cache: true
|
||||||
|
modules: ${{ matrix.qt_modules }}
|
||||||
|
version: ${{ steps.resolve_qt_version.outputs.version }}
|
||||||
|
|
||||||
|
- name: "[Windows] Install NSIS"
|
||||||
|
if: matrix.os == 'Windows'
|
||||||
|
shell: bash
|
||||||
|
run: choco install nsis
|
||||||
|
|
||||||
|
- name: "Setup vcpkg cache"
|
||||||
|
id: vcpkg-cache
|
||||||
|
uses: TAServers/vcpkg-cache@v3
|
||||||
|
with:
|
||||||
|
token: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
|
||||||
|
# Uses environment variables, see compile.sh for more details
|
||||||
|
- name: "Build Cockatrice"
|
||||||
id: build
|
id: build
|
||||||
|
shell: bash
|
||||||
env:
|
env:
|
||||||
BUILDTYPE: '${{matrix.type}}'
|
BUILDTYPE: '${{ matrix.type }}'
|
||||||
MAKE_TEST: 1
|
CCACHE_EVICTION_AGE: ${{ matrix.ccache_eviction_age }}
|
||||||
MAKE_PACKAGE: '${{matrix.make_package}}'
|
CMAKE_GENERATOR: ${{ matrix.cmake_generator }}
|
||||||
PACKAGE_SUFFIX: '-macOS${{matrix.target}}_${{matrix.soc}}'
|
CMAKE_GENERATOR_PLATFORM: ${{ matrix.cmake_generator_platform }}
|
||||||
|
DEVELOPER_DIR: '/Applications/Xcode_${{ matrix.xcode }}.app/Contents/Developer'
|
||||||
MACOS_CERTIFICATE: ${{ secrets.PROD_MACOS_CERTIFICATE }}
|
MACOS_CERTIFICATE: ${{ secrets.PROD_MACOS_CERTIFICATE }}
|
||||||
MACOS_CERTIFICATE_PWD: ${{ secrets.PROD_MACOS_CERTIFICATE_PWD }}
|
|
||||||
MACOS_CERTIFICATE_NAME: ${{ secrets.PROD_MACOS_CERTIFICATE_NAME }}
|
MACOS_CERTIFICATE_NAME: ${{ secrets.PROD_MACOS_CERTIFICATE_NAME }}
|
||||||
|
MACOS_CERTIFICATE_PWD: ${{ secrets.PROD_MACOS_CERTIFICATE_PWD }}
|
||||||
MACOS_CI_KEYCHAIN_PWD: ${{ secrets.PROD_MACOS_CI_KEYCHAIN_PWD }}
|
MACOS_CI_KEYCHAIN_PWD: ${{ secrets.PROD_MACOS_CI_KEYCHAIN_PWD }}
|
||||||
CMAKE_GENERATOR: '${{env.CMAKE_GENERATOR}}'
|
MAKE_PACKAGE: '${{ matrix.make_package }}'
|
||||||
run: |
|
PACKAGE_SUFFIX: '${{ matrix.package_suffix }}'
|
||||||
if [[ -n "$MACOS_CERTIFICATE_NAME" ]]
|
TARGET_MACOS_VERSION: ${{ matrix.override_target }}
|
||||||
then
|
USE_CCACHE: ${{ matrix.use_ccache }}
|
||||||
echo $MACOS_CERTIFICATE | base64 --decode > certificate.p12
|
VCPKG_BINARY_SOURCES: 'clear;files,${{ steps.vcpkg-cache.outputs.path }},readwrite'
|
||||||
security create-keychain -p "$MACOS_CI_KEYCHAIN_PWD" build.keychain
|
VCPKG_DISABLE_METRICS: 1
|
||||||
security default-keychain -s build.keychain
|
run: .ci/compile.sh --server --test --vcpkg
|
||||||
security set-keychain-settings -t 3600 -l build.keychain
|
|
||||||
security unlock-keychain -p "$MACOS_CI_KEYCHAIN_PWD" build.keychain
|
|
||||||
security import certificate.p12 -k build.keychain -P "$MACOS_CERTIFICATE_PWD" -T /usr/bin/codesign
|
|
||||||
security set-key-partition-list -S apple-tool:,apple:,codesign: -s -k "$MACOS_CI_KEYCHAIN_PWD" build.keychain
|
|
||||||
fi
|
|
||||||
.ci/compile.sh --server
|
|
||||||
|
|
||||||
- name: Sign app bundle
|
# Delete used cache to emulate a ccache update. See https://github.com/actions/cache/issues/342
|
||||||
if: matrix.make_package
|
- name: "[macOS] Delete remote compiler cache (ccache)"
|
||||||
|
if: matrix.os == 'macOS' && matrix.use_ccache == 1 && github.ref == 'refs/heads/master' && steps.ccache_restore.outputs.cache-hit
|
||||||
|
continue-on-error: true
|
||||||
|
env:
|
||||||
|
GH_TOKEN: ${{ github.token }}
|
||||||
|
run: |
|
||||||
|
if gh cache delete --repo ${{ github.repository }} ${{ steps.ccache_restore.outputs.cache-primary-key }}; then
|
||||||
|
echo "Cache deleted successfully"
|
||||||
|
fi
|
||||||
|
|
||||||
|
- name: "[macOS] Save updated compiler cache (ccache)"
|
||||||
|
if: matrix.os == 'macOS' && matrix.use_ccache == 1 && github.ref == 'refs/heads/master'
|
||||||
|
uses: actions/cache/save@v5
|
||||||
|
with:
|
||||||
|
key: ${{ steps.ccache_restore.outputs.cache-primary-key }}
|
||||||
|
path: ${{ env.CCACHE_DIR }}
|
||||||
|
|
||||||
|
- name: "[macOS] Sign app bundle"
|
||||||
|
if: matrix.os == 'macOS' && matrix.make_package && needs.configure.outputs.tag != null
|
||||||
|
id: sign_macos
|
||||||
env:
|
env:
|
||||||
MACOS_CERTIFICATE_NAME: ${{ secrets.PROD_MACOS_CERTIFICATE_NAME }}
|
MACOS_CERTIFICATE_NAME: ${{ secrets.PROD_MACOS_CERTIFICATE_NAME }}
|
||||||
MACOS_CI_KEYCHAIN_PWD: ${{ secrets.PROD_MACOS_CI_KEYCHAIN_PWD }}
|
MACOS_CI_KEYCHAIN_PWD: ${{ secrets.PROD_MACOS_CI_KEYCHAIN_PWD }}
|
||||||
|
|
@ -288,143 +488,83 @@ jobs:
|
||||||
if [[ -n "$MACOS_CERTIFICATE_NAME" ]]
|
if [[ -n "$MACOS_CERTIFICATE_NAME" ]]
|
||||||
then
|
then
|
||||||
security unlock-keychain -p "$MACOS_CI_KEYCHAIN_PWD" build.keychain
|
security unlock-keychain -p "$MACOS_CI_KEYCHAIN_PWD" build.keychain
|
||||||
/usr/bin/codesign --sign="$MACOS_CERTIFICATE_NAME" --entitlements=".ci/macos.entitlements" --options=runtime --force --deep --timestamp --verbose ${{steps.build.outputs.path}}
|
/usr/bin/codesign --sign="$MACOS_CERTIFICATE_NAME" --entitlements=".ci/macos.entitlements" --options=runtime --force --deep --timestamp --verbose "${{ steps.build.outputs.path }}"
|
||||||
fi
|
fi
|
||||||
|
|
||||||
- name: Notarize app bundle
|
- name: "[macOS] Notarize app bundle"
|
||||||
if: matrix.make_package
|
if: matrix.os == 'macOS' && steps.sign_macos.outcome == 'success'
|
||||||
env:
|
env:
|
||||||
MACOS_NOTARIZATION_APPLE_ID: ${{ secrets.PROD_MACOS_NOTARIZATION_APPLE_ID }}
|
MACOS_NOTARIZATION_APPLE_ID: ${{ secrets.PROD_MACOS_NOTARIZATION_APPLE_ID }}
|
||||||
MACOS_NOTARIZATION_TEAM_ID: ${{ secrets.PROD_MACOS_NOTARIZATION_TEAM_ID }}
|
|
||||||
MACOS_NOTARIZATION_PWD: ${{ secrets.PROD_MACOS_NOTARIZATION_PWD }}
|
MACOS_NOTARIZATION_PWD: ${{ secrets.PROD_MACOS_NOTARIZATION_PWD }}
|
||||||
|
MACOS_NOTARIZATION_TEAM_ID: ${{ secrets.PROD_MACOS_NOTARIZATION_TEAM_ID }}
|
||||||
run: |
|
run: |
|
||||||
if [[ -n "$MACOS_NOTARIZATION_APPLE_ID" ]]
|
if [[ -n "$MACOS_NOTARIZATION_APPLE_ID" ]]
|
||||||
then
|
then
|
||||||
# Store the notarization credentials so that we can prevent a UI password dialog from blocking the CI
|
# Store the notarization credentials so that we can prevent a UI password dialog from blocking the CI
|
||||||
echo "Create keychain profile"
|
echo "Create keychain profile"
|
||||||
xcrun notarytool store-credentials "notarytool-profile" --apple-id "$MACOS_NOTARIZATION_APPLE_ID" --team-id "$MACOS_NOTARIZATION_TEAM_ID" --password "$MACOS_NOTARIZATION_PWD"
|
xcrun notarytool store-credentials "notarytool-profile" --apple-id "$MACOS_NOTARIZATION_APPLE_ID" --team-id "$MACOS_NOTARIZATION_TEAM_ID" --password "$MACOS_NOTARIZATION_PWD"
|
||||||
|
|
||||||
# We can't notarize an app bundle directly, but we need to compress it as an archive.
|
# We can't notarize an app bundle directly, but we need to compress it as an archive.
|
||||||
# Therefore, we create a zip file containing our app bundle, so that we can send it to the
|
# Therefore, we create a zip file containing our app bundle, so that we can send it to the
|
||||||
# notarization service
|
# notarization service
|
||||||
echo "Creating temp notarization archive"
|
echo "Creating temp notarization archive"
|
||||||
ditto -c -k --keepParent ${{steps.build.outputs.path}} "notarization.zip"
|
ditto -c -k --keepParent "${{ steps.build.outputs.path }}" "notarization.zip"
|
||||||
|
|
||||||
# Here we send the notarization request to the Apple's Notarization service, waiting for the result.
|
# Here we send the notarization request to the Apple's Notarization service, waiting for the result.
|
||||||
# This typically takes a few seconds inside a CI environment, but it might take more depending on the App
|
# This typically takes a few seconds inside a CI environment, but it might take more depending on the App
|
||||||
# characteristics. Visit the Notarization docs for more information and strategies on how to optimize it if
|
# characteristics. Visit the Notarization docs for more information and strategies on how to optimize it if
|
||||||
# you're curious
|
# you're curious
|
||||||
echo "Notarize app"
|
echo "Notarize app"
|
||||||
xcrun notarytool submit "notarization.zip" --keychain-profile "notarytool-profile" --wait
|
xcrun notarytool submit "notarization.zip" --keychain-profile "notarytool-profile" --wait
|
||||||
|
|
||||||
# Finally, we need to "attach the staple" to our executable, which will allow our app to be
|
# Finally, we need to "attach the staple" to our executable, which will allow our app to be
|
||||||
# validated by macOS even when an internet connection is not available.
|
# validated by macOS even when an internet connection is not available.
|
||||||
echo "Attach staple"
|
echo "Attach staple"
|
||||||
xcrun stapler staple ${{steps.build.outputs.path}}
|
xcrun stapler staple "${{ steps.build.outputs.path }}"
|
||||||
fi
|
fi
|
||||||
|
|
||||||
- name: Upload artifact
|
- name: "Upload artifact"
|
||||||
if: matrix.make_package
|
if: matrix.make_package
|
||||||
uses: actions/upload-artifact@v4
|
id: upload_artifact
|
||||||
|
uses: actions/upload-artifact@v7
|
||||||
with:
|
with:
|
||||||
name: macOS${{matrix.target}}${{ matrix.soc == 'Intel' && '_Intel' || '' }}${{ matrix.type == 'Debug' && '_Debug' || '' }}-package
|
archive: false
|
||||||
path: ${{steps.build.outputs.path}}
|
|
||||||
if-no-files-found: error
|
if-no-files-found: error
|
||||||
|
path: ${{ steps.build.outputs.path }}
|
||||||
|
|
||||||
- name: Upload to release
|
- name: "[Windows] Upload PDBs (Program Databases)"
|
||||||
if: matrix.package != 'skip' && needs.configure.outputs.tag != null
|
if: matrix.os == 'Windows' && github.ref_type != 'tag'
|
||||||
shell: bash
|
uses: actions/upload-artifact@v7
|
||||||
env:
|
|
||||||
GH_TOKEN: ${{github.token}}
|
|
||||||
tag_name: ${{needs.configure.outputs.tag}}
|
|
||||||
asset_path: ${{steps.build.outputs.path}}
|
|
||||||
asset_name: ${{steps.build.outputs.name}}
|
|
||||||
run: gh release upload "$tag_name" "$asset_path#$asset_name"
|
|
||||||
|
|
||||||
build-windows:
|
|
||||||
strategy:
|
|
||||||
fail-fast: false
|
|
||||||
matrix:
|
|
||||||
include:
|
|
||||||
- target: 7
|
|
||||||
qt_version: 5.15.*
|
|
||||||
qt_arch: msvc2019_64
|
|
||||||
|
|
||||||
- target: 10
|
|
||||||
qt_version: 6.6.*
|
|
||||||
qt_arch: msvc2019_64
|
|
||||||
qt_modules: "qtimageformats qtmultimedia qtwebsockets"
|
|
||||||
|
|
||||||
name: Windows ${{matrix.target}}
|
|
||||||
needs: configure
|
|
||||||
runs-on: windows-2022
|
|
||||||
env:
|
|
||||||
CMAKE_GENERATOR: 'Visual Studio 17 2022'
|
|
||||||
|
|
||||||
steps:
|
|
||||||
- name: Add msbuild to PATH
|
|
||||||
id: add-msbuild
|
|
||||||
uses: microsoft/setup-msbuild@v2
|
|
||||||
with:
|
with:
|
||||||
msbuild-architecture: x64
|
|
||||||
|
|
||||||
- name: Checkout
|
|
||||||
uses: actions/checkout@v4
|
|
||||||
with:
|
|
||||||
submodules: recursive
|
|
||||||
|
|
||||||
- name: Install Qt ${{matrix.qt_version}}
|
|
||||||
uses: jurplel/install-qt-action@v4
|
|
||||||
with:
|
|
||||||
cache: true
|
|
||||||
setup-python: true
|
|
||||||
version: ${{matrix.qt_version}}
|
|
||||||
arch: win64_${{matrix.qt_arch}}
|
|
||||||
tools: ${{matrix.qt_tools}}
|
|
||||||
modules: ${{matrix.qt_modules}}
|
|
||||||
|
|
||||||
- name: Setup vcpkg cache
|
|
||||||
id: vcpkg-cache
|
|
||||||
uses: TAServers/vcpkg-cache@v3
|
|
||||||
with:
|
|
||||||
token: ${{ secrets.GITHUB_TOKEN }}
|
|
||||||
|
|
||||||
- name: Build Cockatrice
|
|
||||||
id: build
|
|
||||||
shell: bash
|
|
||||||
env:
|
|
||||||
PACKAGE_SUFFIX: '-Win${{matrix.target}}'
|
|
||||||
CMAKE_GENERATOR: '${{env.CMAKE_GENERATOR}}'
|
|
||||||
CMAKE_GENERATOR_PLATFORM: 'x64'
|
|
||||||
QTDIR: '${{github.workspace}}\Qt\${{matrix.qt_version}}\win64_${{matrix.qt_arch}}'
|
|
||||||
VCPKG_DISABLE_METRICS: 1
|
|
||||||
VCPKG_BINARY_SOURCES: 'clear;files,${{ steps.vcpkg-cache.outputs.path }},readwrite'
|
|
||||||
# No need for --parallel flag, MTT is added in the compile script to let cmake/msbuild manage core count,
|
|
||||||
# project and process parallelism: https://devblogs.microsoft.com/cppblog/improved-parallelism-in-msbuild/
|
|
||||||
run: .ci/compile.sh --server --release --test --package
|
|
||||||
|
|
||||||
- name: Upload artifact
|
|
||||||
uses: actions/upload-artifact@v4
|
|
||||||
with:
|
|
||||||
name: Windows${{matrix.target}}-installer
|
|
||||||
path: ${{steps.build.outputs.path}}
|
|
||||||
if-no-files-found: error
|
if-no-files-found: error
|
||||||
|
name: ${{ steps.build.outputs.name }}-PDBs
|
||||||
- name: Upload pdb database
|
|
||||||
uses: actions/upload-artifact@v4
|
|
||||||
with:
|
|
||||||
name: Windows${{matrix.target}}-debug-pdbs
|
|
||||||
path: |
|
path: |
|
||||||
build/cockatrice/Release/*.pdb
|
build/cockatrice/Release/*.pdb
|
||||||
|
build/oracle/Release/*.pdb
|
||||||
build/servatrice/Release/*.pdb
|
build/servatrice/Release/*.pdb
|
||||||
if-no-files-found: error
|
|
||||||
|
|
||||||
- name: Upload to release
|
- name: "Upload to release"
|
||||||
if: matrix.package != 'skip' && needs.configure.outputs.tag != null
|
if: needs.configure.outputs.tag != null && matrix.make_package == '1'
|
||||||
|
id: upload_release
|
||||||
shell: bash
|
shell: bash
|
||||||
env:
|
env:
|
||||||
GH_TOKEN: ${{github.token}}
|
asset_name: ${{ steps.build.outputs.fullname }}
|
||||||
tag_name: ${{needs.configure.outputs.tag}}
|
asset_path: ${{ steps.build.outputs.path }}
|
||||||
asset_path: ${{steps.build.outputs.path}}
|
GH_TOKEN: ${{ github.token }}
|
||||||
asset_name: ${{steps.build.outputs.name}}
|
tag_name: ${{ needs.configure.outputs.tag }}
|
||||||
run: gh release upload "$tag_name" "$asset_path#$asset_name"
|
run: gh release upload "$tag_name" "$asset_path#$asset_name"
|
||||||
|
|
||||||
|
- name: "Attest binary provenance"
|
||||||
|
if: steps.upload_release.outcome == 'success'
|
||||||
|
id: attestation
|
||||||
|
uses: actions/attest@v4
|
||||||
|
with:
|
||||||
|
show-summary: false
|
||||||
|
subject-path: ${{ steps.build.outputs.path }}
|
||||||
|
|
||||||
|
- name: "Verify binary attestation"
|
||||||
|
if: steps.attestation.outcome == 'success'
|
||||||
|
shell: bash
|
||||||
|
env:
|
||||||
|
GH_TOKEN: ${{ github.token }}
|
||||||
|
run: gh attestation verify "${{ steps.build.outputs.path }}" --repo Cockatrice/Cockatrice
|
||||||
|
|
|
||||||
36
.github/workflows/desktop-lint.yml
vendored
|
|
@ -1,30 +1,40 @@
|
||||||
name: Code Style (C++)
|
name: Code Style (C++)
|
||||||
|
|
||||||
on:
|
on:
|
||||||
|
# Push trigger not needed for linting, we do not allow direct pushes to master
|
||||||
pull_request:
|
pull_request:
|
||||||
paths-ignore:
|
paths:
|
||||||
- '**.md'
|
- '*/**' # matches all files not in root
|
||||||
- 'webclient/**'
|
- '!**.md'
|
||||||
- '.github/workflows/web-*.yml'
|
- '!.ci/**'
|
||||||
- '.github/workflows/translations-*.yml'
|
- '!.github/**'
|
||||||
- '.github/workflows/docker-release.yml'
|
- '!.tx/**'
|
||||||
|
- '!doc/**'
|
||||||
|
- '.ci/lint_cpp.sh'
|
||||||
|
- '.github/workflows/desktop-lint.yml'
|
||||||
|
- '.clang-format'
|
||||||
|
- '.cmake-format.json'
|
||||||
|
- 'format.sh'
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
format:
|
format:
|
||||||
runs-on: ubuntu-22.04
|
runs-on: ubuntu-slim
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout
|
- name: "Checkout"
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v6
|
||||||
with:
|
with:
|
||||||
fetch-depth: 20 # should be enough to find merge base
|
fetch-depth: 20 # should be enough to find merge base
|
||||||
|
|
||||||
- name: Install dependencies
|
- name: "Install dependencies"
|
||||||
shell: bash
|
shell: bash
|
||||||
run: |
|
run: |
|
||||||
sudo apt-get update
|
sudo apt-get update
|
||||||
sudo apt-get install -y --no-install-recommends clang-format cmake-format
|
sudo apt-get install -y --no-install-recommends \
|
||||||
|
clang-format \
|
||||||
|
cmake-format \
|
||||||
|
shellcheck
|
||||||
|
|
||||||
- name: Check code formatting
|
- name: "Check code formatting"
|
||||||
shell: bash
|
shell: bash
|
||||||
run: ./.ci/lint_cpp.sh
|
run: ./.ci/lint_cpp.sh
|
||||||
|
|
|
||||||
70
.github/workflows/docker-release.yml
vendored
|
|
@ -1,9 +1,11 @@
|
||||||
name: Build Docker Image
|
name: Build Docker Image
|
||||||
|
|
||||||
|
permissions:
|
||||||
|
contents: read
|
||||||
|
packages: write
|
||||||
|
|
||||||
on:
|
on:
|
||||||
push:
|
push:
|
||||||
tags:
|
|
||||||
- '*Release*'
|
|
||||||
branches:
|
branches:
|
||||||
- master
|
- master
|
||||||
pull_request:
|
pull_request:
|
||||||
|
|
@ -11,61 +13,65 @@ on:
|
||||||
- master
|
- master
|
||||||
paths:
|
paths:
|
||||||
- '.github/workflows/docker-release.yml'
|
- '.github/workflows/docker-release.yml'
|
||||||
- 'CMakeLists.txt'
|
|
||||||
- 'Dockerfile'
|
- 'Dockerfile'
|
||||||
- 'servatrice/**'
|
release:
|
||||||
- 'common/**'
|
types:
|
||||||
- 'cmake/**'
|
- released # publishing of stable releases
|
||||||
- '!**.md'
|
|
||||||
|
# Cancel earlier, unfinished runs of this workflow on the same branch (unless on release)
|
||||||
|
concurrency:
|
||||||
|
group: "${{ github.workflow }} @ ${{ github.ref_name }}"
|
||||||
|
cancel-in-progress: ${{ github.event_name != 'release' }}
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
docker:
|
docker:
|
||||||
name: amd64 & arm64
|
name: amd64 & arm64
|
||||||
|
if: ${{ github.repository_owner == 'Cockatrice' }}
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
permissions:
|
|
||||||
contents: read
|
|
||||||
packages: write
|
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout
|
- name: "Checkout"
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v6
|
||||||
|
|
||||||
- name: Docker metadata
|
- name: "Docker metadata"
|
||||||
id: metadata
|
id: metadata
|
||||||
uses: docker/metadata-action@v5
|
uses: docker/metadata-action@v6
|
||||||
|
env:
|
||||||
|
DOCKER_METADATA_ANNOTATIONS_LEVELS: index # needed for GHCR
|
||||||
with:
|
with:
|
||||||
|
annotations: |
|
||||||
|
org.opencontainers.image.title=Servatrice
|
||||||
|
org.opencontainers.image.url=https://cockatrice.github.io/
|
||||||
|
org.opencontainers.image.description=Server for Cockatrice, a cross-platform virtual tabletop for multiplayer card games
|
||||||
images: |
|
images: |
|
||||||
ghcr.io/cockatrice/servatrice
|
ghcr.io/cockatrice/servatrice
|
||||||
labels: |
|
labels: |
|
||||||
org.opencontainers.image.title=Servatrice
|
org.opencontainers.image.title=Servatrice
|
||||||
org.opencontainers.image.url=https://cockatrice.github.io/
|
org.opencontainers.image.url=https://cockatrice.github.io/
|
||||||
org.opencontainers.image.description=Server for Cockatrice, a cross-platform virtual tabletop for multiplayer card games
|
org.opencontainers.image.description=Server for Cockatrice, a cross-platform virtual tabletop for multiplayer card games
|
||||||
annotations: |
|
|
||||||
org.opencontainers.image.title=Servatrice
|
|
||||||
org.opencontainers.image.url=https://cockatrice.github.io/
|
|
||||||
org.opencontainers.image.description=Server for Cockatrice, a cross-platform virtual tabletop for multiplayer card games
|
|
||||||
|
|
||||||
- name: Set up QEMU
|
- name: "Set up QEMU"
|
||||||
uses: docker/setup-qemu-action@v3
|
uses: docker/setup-qemu-action@v4
|
||||||
|
|
||||||
- name: Set up Docker buildx
|
- name: "Set up Docker buildx"
|
||||||
uses: docker/setup-buildx-action@v3
|
uses: docker/setup-buildx-action@v4
|
||||||
|
|
||||||
- name: Login to GitHub Container Registry
|
- name: "Login to GitHub Container Registry"
|
||||||
if: github.ref_type == 'tag'
|
if: contains(github.event.release.tag_name, 'Release') && github.event.release.target_commitish == 'master'
|
||||||
uses: docker/login-action@v3
|
uses: docker/login-action@v4
|
||||||
with:
|
with:
|
||||||
|
password: ${{ github.token }}
|
||||||
registry: ghcr.io
|
registry: ghcr.io
|
||||||
username: ${{ github.actor }}
|
username: ${{ github.actor }}
|
||||||
password: ${{ github.token }}
|
|
||||||
|
|
||||||
- name: Build and push Docker image
|
- name: "Build and push Docker image"
|
||||||
uses: docker/build-push-action@v6
|
uses: docker/build-push-action@v7
|
||||||
with:
|
with:
|
||||||
|
annotations: ${{ steps.metadata.outputs.annotations }}
|
||||||
|
cache-from: type=gha,scope=servatrice
|
||||||
|
cache-to: type=gha,mode=max,scope=servatrice
|
||||||
context: .
|
context: .
|
||||||
|
labels: ${{ steps.metadata.outputs.labels }}
|
||||||
platforms: linux/amd64,linux/arm64
|
platforms: linux/amd64,linux/arm64
|
||||||
push: ${{ github.ref_type == 'tag' }}
|
push: ${{ github.ref_type == 'tag' }}
|
||||||
tags: ${{ steps.metadata.outputs.tags }}
|
tags: ${{ steps.metadata.outputs.tags }}
|
||||||
labels: ${{ steps.metadata.outputs.labels }}
|
|
||||||
cache-from: type=gha
|
|
||||||
cache-to: type=gha,mode=max
|
|
||||||
|
|
|
||||||
63
.github/workflows/documentation-build.yml
vendored
Normal file
|
|
@ -0,0 +1,63 @@
|
||||||
|
name: Generate Docs
|
||||||
|
|
||||||
|
on:
|
||||||
|
pull_request:
|
||||||
|
paths:
|
||||||
|
- 'doc/doxygen/**'
|
||||||
|
- '.github/workflows/documentation-build.yml'
|
||||||
|
- 'Doxyfile'
|
||||||
|
release:
|
||||||
|
types:
|
||||||
|
- published # publishing of stable releases and pre-releases
|
||||||
|
workflow_dispatch:
|
||||||
|
|
||||||
|
env:
|
||||||
|
COCKATRICE_REF: ${{ github.ref_name }} # tag name if the commit is tagged, otherwise branch name
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
docs:
|
||||||
|
name: Doxygen
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: "Checkout code"
|
||||||
|
uses: actions/checkout@v6
|
||||||
|
with:
|
||||||
|
submodules: recursive
|
||||||
|
|
||||||
|
- name: "Install Graphviz"
|
||||||
|
run: |
|
||||||
|
sudo apt-get install -y graphviz
|
||||||
|
dot -V
|
||||||
|
|
||||||
|
- name: "Install Doxygen"
|
||||||
|
uses: ssciwr/doxygen-install@v2
|
||||||
|
with:
|
||||||
|
version: "1.16.1"
|
||||||
|
|
||||||
|
- name: "Update Doxygen Configuration"
|
||||||
|
run: |
|
||||||
|
git diff Doxyfile
|
||||||
|
doxygen -u Doxyfile
|
||||||
|
if git diff --quiet Doxyfile; then
|
||||||
|
echo "::notice::No config changes in Doxyfile detected."
|
||||||
|
else
|
||||||
|
echo "::error::Config changes in Doxyfile detected! Please update the file by running 'doxygen -u Doxyfile'."
|
||||||
|
echo ""
|
||||||
|
git diff --color=always Doxyfile
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
- name: "Generate Documentation"
|
||||||
|
if: always()
|
||||||
|
run: doxygen Doxyfile
|
||||||
|
|
||||||
|
- name: "Deploy to cockatrice.github.io"
|
||||||
|
if: github.event_name == 'release' || github.event_name == 'workflow_dispatch'
|
||||||
|
uses: peaceiris/actions-gh-pages@v4
|
||||||
|
with:
|
||||||
|
deploy_key: ${{ secrets.DOCS_DEPLOY_KEY }}
|
||||||
|
destination_dir: docs # docs will be available at https://cockatrice.github.io/docs/
|
||||||
|
external_repository: Cockatrice/cockatrice.github.io
|
||||||
|
publish_branch: master
|
||||||
|
publish_dir: ./docs/html
|
||||||
45
.github/workflows/translations-pull.yml
vendored
|
|
@ -1,13 +1,14 @@
|
||||||
name: Update Translations
|
name: Update Translations
|
||||||
|
|
||||||
on:
|
on:
|
||||||
workflow_dispatch:
|
|
||||||
schedule:
|
|
||||||
# runs in the middle of each month starting a quarter (UTC) = two weeks after new strings are built
|
|
||||||
- cron: '0 0 15 1,4,7,10 *'
|
|
||||||
pull_request:
|
pull_request:
|
||||||
paths:
|
paths:
|
||||||
|
- '.tx/**'
|
||||||
- '.github/workflows/translations-pull.yml'
|
- '.github/workflows/translations-pull.yml'
|
||||||
|
schedule:
|
||||||
|
# Runs in the middle of each month starting a quarter (UTC) = two weeks after new strings are built
|
||||||
|
- cron: '0 0 15 1,4,7,10 *'
|
||||||
|
workflow_dispatch:
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
translations:
|
translations:
|
||||||
|
|
@ -15,35 +16,29 @@ jobs:
|
||||||
if: github.event_name != 'schedule' || github.repository_owner == 'Cockatrice'
|
if: github.event_name != 'schedule' || github.repository_owner == 'Cockatrice'
|
||||||
|
|
||||||
name: Pull languages
|
name: Pull languages
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-slim
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout repo
|
- name: "Checkout repo"
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v6
|
||||||
|
|
||||||
- name: Pull translated strings from Transifex
|
- name: "Pull translated strings from Transifex"
|
||||||
uses: transifex/cli-action@v2
|
uses: transifex/cli-action@v2
|
||||||
with:
|
with:
|
||||||
# used config file: https://github.com/Cockatrice/Cockatrice/blob/master/.tx/config
|
# Used config file: https://github.com/Cockatrice/Cockatrice/blob/master/.tx/config
|
||||||
# https://github.com/transifex/cli#pulling-files-from-transifex
|
# Docs: https://github.com/transifex/cli#pulling-files-from-transifex
|
||||||
token: ${{ secrets.TX_TOKEN }}
|
|
||||||
args: pull --force --all
|
args: pull --force --all
|
||||||
|
token: ${{ secrets.TX_TOKEN }}
|
||||||
|
|
||||||
- name: Create pull request
|
- name: "Create pull request"
|
||||||
if: github.event_name != 'pull_request'
|
if: github.event_name != 'pull_request'
|
||||||
id: create_pr
|
id: create_pr
|
||||||
uses: peter-evans/create-pull-request@v7
|
uses: peter-evans/create-pull-request@v8
|
||||||
with:
|
with:
|
||||||
add-paths: |
|
add-paths: |
|
||||||
cockatrice/translations/*.ts
|
cockatrice/translations/*.ts
|
||||||
oracle/translations/*.ts
|
oracle/translations/*.ts
|
||||||
webclient/public/locales/*/translation.json
|
author: github-actions <github-actions@github.com> # owner of the commit
|
||||||
commit-message: Update translation files
|
|
||||||
# author is the owner of the commit
|
|
||||||
author: github-actions <github-actions@github.com>
|
|
||||||
branch: ci-update_translations
|
|
||||||
delete-branch: true
|
|
||||||
title: 'Update translations'
|
|
||||||
body: |
|
body: |
|
||||||
Pulled all translated strings from [Transifex][1].
|
Pulled all translated strings from [Transifex][1].
|
||||||
|
|
||||||
|
|
@ -51,14 +46,18 @@ jobs:
|
||||||
*This PR is automatically generated and updated by the workflow at `.github/workflows/translations-pull.yml`. Review [action runs][2].*<br>
|
*This PR is automatically generated and updated by the workflow at `.github/workflows/translations-pull.yml`. Review [action runs][2].*<br>
|
||||||
*After merging, all new languages and translations are available in the next build.*
|
*After merging, all new languages and translations are available in the next build.*
|
||||||
|
|
||||||
[1]: https://app.transifex.com/cockatrice/cockatrice/
|
[1]: https://explore.transifex.com/cockatrice/cockatrice/
|
||||||
[2]: https://github.com/Cockatrice/Cockatrice/actions/workflows/translations-pull.yml?query=branch%3Amaster
|
[2]: https://github.com/Cockatrice/Cockatrice/actions/workflows/translations-pull.yml?query=branch%3Amaster
|
||||||
|
branch: ci-update_translations
|
||||||
|
commit-message: Update translation files
|
||||||
|
delete-branch: true
|
||||||
|
draft: false
|
||||||
labels: |
|
labels: |
|
||||||
CI
|
CI
|
||||||
Translation
|
Translation
|
||||||
draft: false
|
title: 'Update translations'
|
||||||
|
|
||||||
- name: PR Status
|
- name: "PR Status"
|
||||||
if: github.event_name != 'pull_request'
|
if: github.event_name != 'pull_request'
|
||||||
shell: bash
|
shell: bash
|
||||||
env:
|
env:
|
||||||
|
|
|
||||||
54
.github/workflows/translations-push.yml
vendored
|
|
@ -1,13 +1,14 @@
|
||||||
name: Update Translation Source
|
name: Update Translation Source
|
||||||
|
|
||||||
on:
|
on:
|
||||||
workflow_dispatch:
|
|
||||||
schedule:
|
|
||||||
# runs at the start of each quarter (UTC)
|
|
||||||
- cron: '0 0 1 1,4,7,10 *'
|
|
||||||
pull_request:
|
pull_request:
|
||||||
paths:
|
paths:
|
||||||
|
- '.ci/update_translation_source_strings.sh'
|
||||||
- '.github/workflows/translations-push.yml'
|
- '.github/workflows/translations-push.yml'
|
||||||
|
schedule:
|
||||||
|
# Runs at the start of each quarter (UTC)
|
||||||
|
- cron: '0 0 1 1,4,7,10 *'
|
||||||
|
workflow_dispatch:
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
translations:
|
translations:
|
||||||
|
|
@ -15,37 +16,37 @@ jobs:
|
||||||
if: github.event_name != 'schedule' || github.repository_owner == 'Cockatrice'
|
if: github.event_name != 'schedule' || github.repository_owner == 'Cockatrice'
|
||||||
|
|
||||||
name: Push strings
|
name: Push strings
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-slim
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout repo
|
- name: "Checkout repo"
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v6
|
||||||
|
|
||||||
- name: Install lupdate
|
- name: "Install lupdate"
|
||||||
shell: bash
|
shell: bash
|
||||||
run: |
|
run: |
|
||||||
sudo apt-get update
|
sudo apt-get update
|
||||||
sudo apt-get install -y --no-install-recommends qttools5-dev-tools
|
sudo apt-get install -y --no-install-recommends qttools5-dev-tools
|
||||||
|
|
||||||
- name: Update Cockatrice translation source
|
- name: "Update Cockatrice translation source"
|
||||||
id: cockatrice
|
id: cockatrice
|
||||||
shell: bash
|
shell: bash
|
||||||
env:
|
run: |
|
||||||
FILE: 'cockatrice/cockatrice_en@source.ts'
|
FILE="cockatrice/cockatrice_en@source.ts"
|
||||||
DIRS: 'cockatrice/src common'
|
export DIRS="cockatrice/src $(find . -maxdepth 1 -type d -name 'libcockatrice_*')"
|
||||||
run: .ci/update_translation_source_strings.sh
|
FILE="$FILE" DIRS="$DIRS" .ci/update_translation_source_strings.sh
|
||||||
|
|
||||||
- name: Update Oracle translation source
|
- name: "Update Oracle translation source"
|
||||||
id: oracle
|
id: oracle
|
||||||
shell: bash
|
shell: bash
|
||||||
env:
|
env:
|
||||||
FILE: 'oracle/oracle_en@source.ts'
|
|
||||||
DIRS: 'oracle/src'
|
DIRS: 'oracle/src'
|
||||||
|
FILE: 'oracle/oracle_en@source.ts'
|
||||||
run: .ci/update_translation_source_strings.sh
|
run: .ci/update_translation_source_strings.sh
|
||||||
|
|
||||||
- name: Render template
|
- name: "Render template"
|
||||||
id: template
|
id: template
|
||||||
uses: chuhlomin/render-template@v1
|
uses: chuhlomin/render-template/binary@v1
|
||||||
with:
|
with:
|
||||||
template: .ci/update_translation_source_strings_template.md
|
template: .ci/update_translation_source_strings_template.md
|
||||||
vars: |
|
vars: |
|
||||||
|
|
@ -53,27 +54,26 @@ jobs:
|
||||||
oracle_output: ${{ steps.oracle.outputs.output }}
|
oracle_output: ${{ steps.oracle.outputs.output }}
|
||||||
commit: ${{ github.sha }}
|
commit: ${{ github.sha }}
|
||||||
|
|
||||||
- name: Create pull request
|
- name: "Create pull request"
|
||||||
if: github.event_name != 'pull_request'
|
if: github.event_name != 'pull_request'
|
||||||
id: create_pr
|
id: create_pr
|
||||||
uses: peter-evans/create-pull-request@v7
|
uses: peter-evans/create-pull-request@v8
|
||||||
with:
|
with:
|
||||||
add-paths: |
|
add-paths: |
|
||||||
cockatrice/cockatrice_en@source.ts
|
cockatrice/cockatrice_en@source.ts
|
||||||
oracle/oracle_en@source.ts
|
oracle/oracle_en@source.ts
|
||||||
commit-message: Update translation source strings
|
author: github-actions <github-actions@github.com> # owner of the commit
|
||||||
# author is the owner of the commit
|
|
||||||
author: github-actions <github-actions@github.com>
|
|
||||||
branch: ci-update_translation_source
|
|
||||||
delete-branch: true
|
|
||||||
title: 'Update source strings'
|
|
||||||
body: ${{ steps.template.outputs.result }}
|
body: ${{ steps.template.outputs.result }}
|
||||||
|
branch: ci-update_translation_source
|
||||||
|
commit-message: Update translation source strings
|
||||||
|
delete-branch: true
|
||||||
|
draft: false
|
||||||
labels: |
|
labels: |
|
||||||
CI
|
CI
|
||||||
Translation
|
Translation
|
||||||
draft: false
|
title: 'Update source strings'
|
||||||
|
|
||||||
- name: PR Status
|
- name: "PR Status"
|
||||||
if: github.event_name != 'pull_request'
|
if: github.event_name != 'pull_request'
|
||||||
shell: bash
|
shell: bash
|
||||||
env:
|
env:
|
||||||
|
|
|
||||||
52
.github/workflows/web-build.yml
vendored
|
|
@ -1,52 +0,0 @@
|
||||||
name: Build Web
|
|
||||||
|
|
||||||
on:
|
|
||||||
push:
|
|
||||||
branches:
|
|
||||||
- master
|
|
||||||
paths:
|
|
||||||
- '.github/workflows/web-*.yml'
|
|
||||||
- 'webclient/**'
|
|
||||||
- '!**.md'
|
|
||||||
pull_request:
|
|
||||||
paths:
|
|
||||||
- '.github/workflows/web-*.yml'
|
|
||||||
- 'webclient/**'
|
|
||||||
- '!**.md'
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
build-web:
|
|
||||||
name: React (Node ${{matrix.node_version}})
|
|
||||||
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
|
|
||||||
defaults:
|
|
||||||
run:
|
|
||||||
working-directory: webclient
|
|
||||||
|
|
||||||
strategy:
|
|
||||||
fail-fast: false
|
|
||||||
matrix:
|
|
||||||
node_version:
|
|
||||||
- 16
|
|
||||||
- lts/*
|
|
||||||
|
|
||||||
steps:
|
|
||||||
- name: Checkout
|
|
||||||
uses: actions/checkout@v4
|
|
||||||
|
|
||||||
- name: Setup Node.js
|
|
||||||
uses: actions/setup-node@v4
|
|
||||||
with:
|
|
||||||
node-version: ${{matrix.node_version}}
|
|
||||||
cache: 'npm'
|
|
||||||
cache-dependency-path: 'webclient/package-lock.json'
|
|
||||||
|
|
||||||
- name: Install dependencies
|
|
||||||
run: npm clean-install
|
|
||||||
|
|
||||||
- name: Build app
|
|
||||||
run: npm run build
|
|
||||||
|
|
||||||
- name: Test app
|
|
||||||
run: npm run test
|
|
||||||
32
.github/workflows/web-lint.yml
vendored
|
|
@ -1,32 +0,0 @@
|
||||||
name: Code Style (TypeScript)
|
|
||||||
|
|
||||||
on:
|
|
||||||
pull_request:
|
|
||||||
paths:
|
|
||||||
- '.github/workflows/web-*.yml'
|
|
||||||
- 'webclient/**'
|
|
||||||
- '!**.md'
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
ESLint:
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
|
|
||||||
defaults:
|
|
||||||
run:
|
|
||||||
working-directory: webclient
|
|
||||||
|
|
||||||
steps:
|
|
||||||
- name: Checkout
|
|
||||||
uses: actions/checkout@v4
|
|
||||||
|
|
||||||
- name: Setup Node.js
|
|
||||||
uses: actions/setup-node@v4
|
|
||||||
with:
|
|
||||||
cache: 'npm'
|
|
||||||
cache-dependency-path: 'webclient/package-lock.json'
|
|
||||||
|
|
||||||
- name: Install ESLint
|
|
||||||
run: npm clean-install --ignore-scripts
|
|
||||||
|
|
||||||
- name: Run ESLint
|
|
||||||
run: npm run lint
|
|
||||||
1
.gitignore
vendored
|
|
@ -14,3 +14,4 @@ compile_commands.json
|
||||||
.cache
|
.cache
|
||||||
.gdb_history
|
.gdb_history
|
||||||
cockatrice/resources/config/qtlogging.ini
|
cockatrice/resources/config/qtlogging.ini
|
||||||
|
docs/
|
||||||
|
|
|
||||||
3
.gitmodules
vendored
|
|
@ -1,3 +1,6 @@
|
||||||
[submodule "vcpkg"]
|
[submodule "vcpkg"]
|
||||||
path = vcpkg
|
path = vcpkg
|
||||||
url = https://github.com/microsoft/vcpkg.git
|
url = https://github.com/microsoft/vcpkg.git
|
||||||
|
[submodule "doxygen-awesome-css"]
|
||||||
|
path = doc/doxygen/theme
|
||||||
|
url = https://github.com/jothepro/doxygen-awesome-css.git
|
||||||
|
|
|
||||||
|
|
@ -1,7 +0,0 @@
|
||||||
#!/bin/sh
|
|
||||||
. "$(dirname "$0")/_/husky.sh"
|
|
||||||
|
|
||||||
cd webclient
|
|
||||||
npm run translate
|
|
||||||
|
|
||||||
git add src/i18n-default.json
|
|
||||||
|
|
@ -16,11 +16,3 @@ source_file = oracle/oracle_en@source.ts
|
||||||
file_filter = oracle/translations/oracle_<lang>.ts
|
file_filter = oracle/translations/oracle_<lang>.ts
|
||||||
type = QT
|
type = QT
|
||||||
minimum_perc = 10
|
minimum_perc = 10
|
||||||
|
|
||||||
[o:cockatrice:p:cockatrice:r:webclient-src-i18n-default-json--master]
|
|
||||||
resource_name = Webclient
|
|
||||||
source_lang = en
|
|
||||||
source_file = webclient/src/i18n-default.json
|
|
||||||
file_filter = webclient/public/locales/<lang>/translation.json
|
|
||||||
type = KEYVALUEJSON
|
|
||||||
minimum_perc = 10
|
|
||||||
|
|
|
||||||
|
|
@ -20,10 +20,10 @@ option(WITH_SERVER "build servatrice" OFF)
|
||||||
option(WITH_CLIENT "build cockatrice" ON)
|
option(WITH_CLIENT "build cockatrice" ON)
|
||||||
# Compile oracle
|
# Compile oracle
|
||||||
option(WITH_ORACLE "build oracle" ON)
|
option(WITH_ORACLE "build oracle" ON)
|
||||||
# Compile dbconverter
|
|
||||||
option(WITH_DBCONVERTER "build dbconverter" ON)
|
|
||||||
# Compile tests
|
# Compile tests
|
||||||
option(TEST "build tests" OFF)
|
option(TEST "build tests" OFF)
|
||||||
|
# Use vcpkg regardless of OS
|
||||||
|
option(USE_VCPKG "Use vcpkg regardless of OS" OFF)
|
||||||
|
|
||||||
# Default to "Release" build type
|
# Default to "Release" build type
|
||||||
# User-provided value for CMAKE_BUILD_TYPE must be checked before the PROJECT() call
|
# User-provided value for CMAKE_BUILD_TYPE must be checked before the PROJECT() call
|
||||||
|
|
@ -48,8 +48,8 @@ if(USE_CCACHE)
|
||||||
endif()
|
endif()
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
if(WIN32)
|
if(WIN32 OR USE_VCPKG)
|
||||||
# Use vcpkg toolchain on Windows
|
# Use vcpkg toolchain on Windows (and on macOS in CI)
|
||||||
set(CMAKE_TOOLCHAIN_FILE
|
set(CMAKE_TOOLCHAIN_FILE
|
||||||
${CMAKE_CURRENT_SOURCE_DIR}/vcpkg/scripts/buildsystems/vcpkg.cmake
|
${CMAKE_CURRENT_SOURCE_DIR}/vcpkg/scripts/buildsystems/vcpkg.cmake
|
||||||
CACHE STRING "Vcpkg toolchain file"
|
CACHE STRING "Vcpkg toolchain file"
|
||||||
|
|
@ -74,11 +74,11 @@ endif()
|
||||||
|
|
||||||
# A project name is needed for CPack
|
# A project name is needed for CPack
|
||||||
# Version can be overriden by git tags, see cmake/getversion.cmake
|
# Version can be overriden by git tags, see cmake/getversion.cmake
|
||||||
project("Cockatrice" VERSION 2.11.0)
|
project("Cockatrice" VERSION 3.1.0)
|
||||||
|
|
||||||
# Set release name if not provided via env/cmake var
|
# Set release name if not provided via env/cmake var
|
||||||
if(NOT DEFINED GIT_TAG_RELEASENAME)
|
if(NOT DEFINED GIT_TAG_RELEASENAME)
|
||||||
set(GIT_TAG_RELEASENAME "Omenpath")
|
set(GIT_TAG_RELEASENAME "Graduation Day")
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
# Use c++20 for all targets
|
# Use c++20 for all targets
|
||||||
|
|
@ -174,6 +174,7 @@ elseif(CMAKE_COMPILER_IS_GNUCXX)
|
||||||
-Wno-error=delete-non-virtual-dtor
|
-Wno-error=delete-non-virtual-dtor
|
||||||
-Wno-error=sign-compare
|
-Wno-error=sign-compare
|
||||||
-Wno-error=missing-declarations
|
-Wno-error=missing-declarations
|
||||||
|
-Wno-error=sfinae-incomplete # GCC 16+: Qt MOC + protobuf forward decls trigger this
|
||||||
)
|
)
|
||||||
|
|
||||||
foreach(FLAG ${ADDITIONAL_DEBUG_FLAGS})
|
foreach(FLAG ${ADDITIONAL_DEBUG_FLAGS})
|
||||||
|
|
@ -254,7 +255,9 @@ endif()
|
||||||
set(CPACK_PACKAGE_CONTACT "Zach Halpern <zach@cockatrice.us>")
|
set(CPACK_PACKAGE_CONTACT "Zach Halpern <zach@cockatrice.us>")
|
||||||
set(CPACK_PACKAGE_DESCRIPTION_SUMMARY "${PROJECT_NAME}")
|
set(CPACK_PACKAGE_DESCRIPTION_SUMMARY "${PROJECT_NAME}")
|
||||||
set(CPACK_PACKAGE_VENDOR "Cockatrice Development Team")
|
set(CPACK_PACKAGE_VENDOR "Cockatrice Development Team")
|
||||||
set(CPACK_PACKAGE_DESCRIPTION_FILE "${PROJECT_SOURCE_DIR}/README.md")
|
set(CPACK_PACKAGE_DESCRIPTION
|
||||||
|
"Cockatrice is an open-source, multiplatform application for playing tabletop card games over a network. The program's server design prevents users from manipulating the game for unfair advantage. The client also provides a single-player mode, which allows users to brew while offline."
|
||||||
|
)
|
||||||
set(CPACK_RESOURCE_FILE_LICENSE "${PROJECT_SOURCE_DIR}/LICENSE")
|
set(CPACK_RESOURCE_FILE_LICENSE "${PROJECT_SOURCE_DIR}/LICENSE")
|
||||||
set(CPACK_PACKAGE_VERSION_MAJOR "${PROJECT_VERSION_MAJOR}")
|
set(CPACK_PACKAGE_VERSION_MAJOR "${PROJECT_VERSION_MAJOR}")
|
||||||
set(CPACK_PACKAGE_VERSION_MINOR "${PROJECT_VERSION_MINOR}")
|
set(CPACK_PACKAGE_VERSION_MINOR "${PROJECT_VERSION_MINOR}")
|
||||||
|
|
@ -301,6 +304,7 @@ if(UNIX)
|
||||||
set(CPACK_DEBIAN_PACKAGE_HOMEPAGE "http://github.com/Cockatrice/Cockatrice")
|
set(CPACK_DEBIAN_PACKAGE_HOMEPAGE "http://github.com/Cockatrice/Cockatrice")
|
||||||
if(Qt6_FOUND)
|
if(Qt6_FOUND)
|
||||||
set(CPACK_DEBIAN_PACKAGE_DEPENDS "libqt6multimedia6, libqt6svg6, qt6-qpa-plugins, qt6-image-formats-plugins")
|
set(CPACK_DEBIAN_PACKAGE_DEPENDS "libqt6multimedia6, libqt6svg6, qt6-qpa-plugins, qt6-image-formats-plugins")
|
||||||
|
set(CPACK_DEBIAN_PACKAGE_RECOMMENDS "libqt6sql6-mysql") # for connecting servatrice to a mysql db
|
||||||
elseif(Qt5_FOUND)
|
elseif(Qt5_FOUND)
|
||||||
set(CPACK_DEBIAN_PACKAGE_DEPENDS "libqt5multimedia5-plugins, libqt5svg5")
|
set(CPACK_DEBIAN_PACKAGE_DEPENDS "libqt5multimedia5-plugins, libqt5svg5")
|
||||||
endif()
|
endif()
|
||||||
|
|
@ -325,7 +329,24 @@ endif()
|
||||||
|
|
||||||
include(CPack)
|
include(CPack)
|
||||||
|
|
||||||
add_subdirectory(common)
|
add_subdirectory(${CMAKE_SOURCE_DIR}/libcockatrice_interfaces ${CMAKE_BINARY_DIR}/libcockatrice_interfaces)
|
||||||
|
add_subdirectory(${CMAKE_SOURCE_DIR}/libcockatrice_protocol ${CMAKE_BINARY_DIR}/libcockatrice_protocol)
|
||||||
|
if(WITH_CLIENT
|
||||||
|
OR WITH_SERVER
|
||||||
|
OR WITH_ORACLE
|
||||||
|
)
|
||||||
|
add_subdirectory(${CMAKE_SOURCE_DIR}/libcockatrice_network ${CMAKE_BINARY_DIR}/libcockatrice_network)
|
||||||
|
endif()
|
||||||
|
add_subdirectory(${CMAKE_SOURCE_DIR}/libcockatrice_deck_list ${CMAKE_BINARY_DIR}/libcockatrice_deck_list)
|
||||||
|
add_subdirectory(${CMAKE_SOURCE_DIR}/libcockatrice_rng ${CMAKE_BINARY_DIR}/libcockatrice_rng)
|
||||||
|
add_subdirectory(${CMAKE_SOURCE_DIR}/libcockatrice_card ${CMAKE_BINARY_DIR}/libcockatrice_card)
|
||||||
|
add_subdirectory(${CMAKE_SOURCE_DIR}/libcockatrice_utility ${CMAKE_BINARY_DIR}/libcockatrice_utility)
|
||||||
|
if(WITH_ORACLE OR WITH_CLIENT)
|
||||||
|
add_subdirectory(${CMAKE_SOURCE_DIR}/libcockatrice_settings ${CMAKE_BINARY_DIR}/libcockatrice_settings)
|
||||||
|
add_subdirectory(${CMAKE_SOURCE_DIR}/libcockatrice_models ${CMAKE_BINARY_DIR}/libcockatrice_models)
|
||||||
|
add_subdirectory(${CMAKE_SOURCE_DIR}/libcockatrice_filters ${CMAKE_BINARY_DIR}/libcockatrice_filters)
|
||||||
|
endif()
|
||||||
|
|
||||||
if(WITH_SERVER)
|
if(WITH_SERVER)
|
||||||
add_subdirectory(servatrice)
|
add_subdirectory(servatrice)
|
||||||
set(CPACK_INSTALL_CMAKE_PROJECTS "Servatrice;Servatrice;ALL;/" ${CPACK_INSTALL_CMAKE_PROJECTS})
|
set(CPACK_INSTALL_CMAKE_PROJECTS "Servatrice;Servatrice;ALL;/" ${CPACK_INSTALL_CMAKE_PROJECTS})
|
||||||
|
|
@ -341,11 +362,6 @@ if(WITH_ORACLE)
|
||||||
set(CPACK_INSTALL_CMAKE_PROJECTS "Oracle;Oracle;ALL;/" ${CPACK_INSTALL_CMAKE_PROJECTS})
|
set(CPACK_INSTALL_CMAKE_PROJECTS "Oracle;Oracle;ALL;/" ${CPACK_INSTALL_CMAKE_PROJECTS})
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
if(WITH_DBCONVERTER)
|
|
||||||
add_subdirectory(dbconverter)
|
|
||||||
set(CPACK_INSTALL_CMAKE_PROJECTS "Dbconverter;Dbconverter;ALL;/" ${CPACK_INSTALL_CMAKE_PROJECTS})
|
|
||||||
endif()
|
|
||||||
|
|
||||||
if(TEST)
|
if(TEST)
|
||||||
include(CTest)
|
include(CTest)
|
||||||
add_subdirectory(tests)
|
add_subdirectory(tests)
|
||||||
|
|
|
||||||
33
Dockerfile
|
|
@ -1,8 +1,9 @@
|
||||||
FROM ubuntu:24.04
|
# -------- Build Stage --------
|
||||||
|
FROM ubuntu:26.04 AS build
|
||||||
|
|
||||||
ARG DEBIAN_FRONTEND=noninteractive
|
ARG DEBIAN_FRONTEND=noninteractive
|
||||||
|
|
||||||
RUN apt-get update && apt-get install -y\
|
RUN apt-get update && apt-get install -y --no-install-recommends \
|
||||||
build-essential \
|
build-essential \
|
||||||
cmake \
|
cmake \
|
||||||
file \
|
file \
|
||||||
|
|
@ -16,20 +17,28 @@ RUN apt-get update && apt-get install -y\
|
||||||
qt6-tools-dev \
|
qt6-tools-dev \
|
||||||
qt6-tools-dev-tools
|
qt6-tools-dev-tools
|
||||||
|
|
||||||
COPY ./CMakeLists.txt ./LICENSE ./README.md /home/servatrice/code/
|
WORKDIR /src
|
||||||
COPY ./cmake /home/servatrice/code/cmake
|
COPY . .
|
||||||
COPY ./common /home/servatrice/code/common
|
RUN mkdir build && cd build && \
|
||||||
COPY ./servatrice /home/servatrice/code/servatrice
|
cmake .. -DWITH_SERVER=1 -DWITH_CLIENT=0 -DWITH_ORACLE=0 && \
|
||||||
|
make -j$(nproc) && \
|
||||||
|
make install
|
||||||
|
|
||||||
WORKDIR /home/servatrice/code
|
|
||||||
|
|
||||||
WORKDIR build
|
# -------- Runtime Stage (clean) --------
|
||||||
RUN cmake .. -DWITH_SERVER=1 -DWITH_CLIENT=0 -DWITH_ORACLE=0 -DWITH_DBCONVERTER=0 &&\
|
FROM ubuntu:26.04
|
||||||
make &&\
|
|
||||||
make install
|
RUN apt-get update && apt-get install -y --no-install-recommends \
|
||||||
|
libprotobuf32t64 \
|
||||||
|
libqt6sql6-mysql \
|
||||||
|
libqt6websockets6 \
|
||||||
|
&& apt-get clean \
|
||||||
|
&& rm -rf /var/lib/apt/lists/*
|
||||||
|
|
||||||
|
# Only copy installed binaries, not source
|
||||||
|
COPY --from=build /usr/local /usr/local
|
||||||
|
|
||||||
WORKDIR /home/servatrice
|
WORKDIR /home/servatrice
|
||||||
|
|
||||||
EXPOSE 4748
|
EXPOSE 4748
|
||||||
|
|
||||||
ENTRYPOINT [ "servatrice", "--log-to-console" ]
|
ENTRYPOINT [ "servatrice", "--log-to-console" ]
|
||||||
|
|
|
||||||
9
LICENSE
|
|
@ -2,7 +2,7 @@
|
||||||
Version 2, June 1991
|
Version 2, June 1991
|
||||||
|
|
||||||
Copyright (C) 1989, 1991 Free Software Foundation, Inc.,
|
Copyright (C) 1989, 1991 Free Software Foundation, Inc.,
|
||||||
51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
<https://fsf.org/>
|
||||||
Everyone is permitted to copy and distribute verbatim copies
|
Everyone is permitted to copy and distribute verbatim copies
|
||||||
of this license document, but changing it is not allowed.
|
of this license document, but changing it is not allowed.
|
||||||
|
|
||||||
|
|
@ -304,8 +304,7 @@ the "copyright" line and a pointer to where the full notice is found.
|
||||||
GNU General Public License for more details.
|
GNU General Public License for more details.
|
||||||
|
|
||||||
You should have received a copy of the GNU General Public License along
|
You should have received a copy of the GNU General Public License along
|
||||||
with this program; if not, write to the Free Software Foundation, Inc.,
|
with this program; if not, see <https://www.gnu.org/licenses/>.
|
||||||
51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
||||||
|
|
||||||
Also add information on how to contact you by electronic and paper mail.
|
Also add information on how to contact you by electronic and paper mail.
|
||||||
|
|
||||||
|
|
@ -329,8 +328,8 @@ necessary. Here is a sample; alter the names:
|
||||||
Yoyodyne, Inc., hereby disclaims all copyright interest in the program
|
Yoyodyne, Inc., hereby disclaims all copyright interest in the program
|
||||||
`Gnomovision' (which makes passes at compilers) written by James Hacker.
|
`Gnomovision' (which makes passes at compilers) written by James Hacker.
|
||||||
|
|
||||||
<signature of Ty Coon>, 1 April 1989
|
<signature of Moe Ghoul>, 1 April 1989
|
||||||
Ty Coon, President of Vice
|
Moe Ghoul, President of Vice
|
||||||
|
|
||||||
This General Public License does not permit incorporating your program into
|
This General Public License does not permit incorporating your program into
|
||||||
proprietary programs. If your program is a subroutine library, you may
|
proprietary programs. If your program is a subroutine library, you may
|
||||||
|
|
|
||||||
65
README.md
|
|
@ -8,7 +8,7 @@
|
||||||
<a href="#related-repositories">Related</a> <b>|</b>
|
<a href="#related-repositories">Related</a> <b>|</b>
|
||||||
<a href="#community-resources-">Community</a> <b>|</b>
|
<a href="#community-resources-">Community</a> <b>|</b>
|
||||||
<a href="#contribute">Contribute</a> <b>|</b>
|
<a href="#contribute">Contribute</a> <b>|</b>
|
||||||
<a href="#build---">Build</a> <b>|</b>
|
<a href="#build--">Build</a> <b>|</b>
|
||||||
<a href="#run">Run</a>
|
<a href="#run">Run</a>
|
||||||
</p>
|
</p>
|
||||||
|
|
||||||
|
|
@ -25,7 +25,6 @@
|
||||||
|
|
||||||
Cockatrice is an open-source, multiplatform application for playing tabletop card games over a network. The program's server design prevents users from manipulating the game for unfair advantage. The client also provides a single-player mode, which allows users to brew while offline.<br><br>
|
Cockatrice is an open-source, multiplatform application for playing tabletop card games over a network. The program's server design prevents users from manipulating the game for unfair advantage. The client also provides a single-player mode, which allows users to brew while offline.<br><br>
|
||||||
This project uses <kbd>C++</kbd> and the <kbd>Qt</kbd> libraries.<br>
|
This project uses <kbd>C++</kbd> and the <kbd>Qt</kbd> libraries.<br>
|
||||||
First work on a webclient with <kbd>Typescript</kbd> was started as well.<br>
|
|
||||||
|
|
||||||
|
|
||||||
# Download [](https://tooomm.github.io/github-release-stats/?username=Cockatrice&repository=Cockatrice&search=0)
|
# Download [](https://tooomm.github.io/github-release-stats/?username=Cockatrice&repository=Cockatrice&search=0)
|
||||||
|
|
@ -44,24 +43,43 @@ Latest <kbd>beta</kbd> version:
|
||||||
|
|
||||||
# Related Repositories
|
# Related Repositories
|
||||||
|
|
||||||
- [Magic-Token](https://github.com/Cockatrice/Magic-Token): MtG token data to use in Cockatrice
|
- [Magic-Token](https://github.com/Cockatrice/Magic-Token): File with MtG token data for use in Cockatrice
|
||||||
- [Magic-Spoiler](https://github.com/Cockatrice/Magic-Spoiler): Script to generate MtG spoiler data from [MTGJSON](https://github.com/mtgjson/mtgjson) to use in Cockatrice
|
- [Magic-Spoiler](https://github.com/Cockatrice/Magic-Spoiler): Code to generate MtG spoiler data from [MTGJSON](https://github.com/mtgjson/mtgjson) for use in Cockatrice
|
||||||
- [cockatrice.github.io](https://github.com/Cockatrice/cockatrice.github.io): Code of the official webpage of the Cockatrice project
|
- [cockatrice.github.io](https://github.com/Cockatrice/cockatrice.github.io): Code of the official Cockatrice webpage
|
||||||
|
- [io.github.Cockatrice.cockatrice](https://github.com/flathub/io.github.Cockatrice.cockatrice): Configuration of our Linux `flatpak` package hosted at [Flathub](https://flathub.org/en/apps/io.github.Cockatrice.cockatrice)
|
||||||
|
- [Webatrice](https://github.com/Cockatrice/Webatrice): Web client for Cockatrice servers (TypeScript / React)
|
||||||
|
|
||||||
|
|
||||||
# Community Resources [](https://discord.gg/3Z9yzmA)
|
# Community Resources [](https://discord.gg/3Z9yzmA)
|
||||||
|
|
||||||
Join our [Discord community](https://discord.gg/3Z9yzmA) to connect with other projet contributors (`#dev` channel) or fellow users of the app. Come here to talk about the application, features, or just to hang out.
|
Join our [Discord community](https://discord.gg/3Z9yzmA) to connect with other project contributors (`#dev` channel) or fellow users of the app. Come here to talk about the application, features, or just to hang out.
|
||||||
- [Official Website](https://cockatrice.github.io)
|
- [Official Website](https://cockatrice.github.io)
|
||||||
- [Official Wiki](https://github.com/Cockatrice/Cockatrice/wiki)
|
- [Official Wiki](https://github.com/Cockatrice/Cockatrice/wiki)
|
||||||
- [Official Discord](https://discord.gg/3Z9yzmA)
|
- [Official Discord](https://discord.gg/3Z9yzmA)
|
||||||
- [reddit r/Cockatrice](https://reddit.com/r/cockatrice)
|
- [reddit r/Cockatrice](https://reddit.com/r/cockatrice)
|
||||||
|
|
||||||
>[!IMPORTANT]
|
> [!IMPORTANT]
|
||||||
>For support regarding specific servers, please contact that server's admin/mods and use their dedicated communication channels rather than contacting the team building the software.
|
> For support regarding specific servers, please contact that server's admin/mods and use their dedicated communication channels rather than contacting the team building the software.
|
||||||
|
|
||||||
|
|
||||||
# Contribute
|
# Contribute
|
||||||
|
<p>
|
||||||
|
<a href="#code">Code</a> <b>|</b>
|
||||||
|
<a href="#documentation-">Documentation</a> <b>|</b>
|
||||||
|
<a href="#translation-">Translation</a>
|
||||||
|
</p>
|
||||||
|
|
||||||
|
#### Repository Activity
|
||||||
|

|
||||||
|
|
||||||
|
<details>
|
||||||
|
<summary><b>Kudos to all our amazing contributors ❤️</b></summary>
|
||||||
|
<br>
|
||||||
|
<a href="https://github.com/Cockatrice/Cockatrice/graphs/contributors">
|
||||||
|
<img src="https://contrib.rocks/image?repo=Cockatrice/Cockatrice" />
|
||||||
|
</a><br>
|
||||||
|
<sub><i>Made with <a href="https://contrib.rocks">contrib.rocks</a></i></sub>
|
||||||
|
</details>
|
||||||
|
|
||||||
### Code
|
### Code
|
||||||
|
|
||||||
|
|
@ -75,25 +93,26 @@ This tag is used for issues that we are looking for somebody to pick up. Often t
|
||||||
For both tags, we're willing to provide help to contributors in showing them where and how they can make changes, as well as code reviews for submitted changes.<br>
|
For both tags, we're willing to provide help to contributors in showing them where and how they can make changes, as well as code reviews for submitted changes.<br>
|
||||||
We'll happily advice on how best to implement a feature, or we can show you where the codebase is doing something similar before you get too far along - put a note on an issue you want to discuss more on!
|
We'll happily advice on how best to implement a feature, or we can show you where the codebase is doing something similar before you get too far along - put a note on an issue you want to discuss more on!
|
||||||
|
|
||||||
|
You can also have a look at our `Todo List` in our [Code Documentation](https://cockatrice.github.io/docs) or search the repo for [`\todo` comments](https://github.com/search?q=repo%3ACockatrice%2FCockatrice%20%5Ctodo&type=code).
|
||||||
|
|
||||||
|
### Documentation [](https://github.com/Cockatrice/Cockatrice/actions/workflows/documentation-build.yml?query=event%3Apush)
|
||||||
|
|
||||||
|
There are various places where useful information for different needs are maintained:
|
||||||
|
- [Official Code Documentation](https://cockatrice.github.io/docs/)
|
||||||
|
- [Official Wiki](https://github.com/Cockatrice/Cockatrice/wiki) `Community supported`
|
||||||
|
- [Official Webpage](https://cockatrice.github.io/)
|
||||||
|
- [Official README](https://github.com/Cockatrice/Cockatrice/blob/master/README.md) `This file`
|
||||||
|
|
||||||
Cockatrice tries to use the [Google Developer Documentation Style Guide](https://developers.google.com/style/) to ensure consistent documentation. We encourage you to improve the documentation by suggesting edits based on this guide.
|
Cockatrice tries to use the [Google Developer Documentation Style Guide](https://developers.google.com/style/) to ensure consistent documentation. We encourage you to improve the documentation by suggesting edits based on this guide.
|
||||||
|
|
||||||
<details>
|
### Translation [](https://explore.transifex.com/cockatrice/cockatrice/)
|
||||||
<summary><b>Kudos to our amazing contributors ❤️</b></summary>
|
|
||||||
<br>
|
|
||||||
<a href="https://github.com/Cockatrice/Cockatrice/graphs/contributors">
|
|
||||||
<img src="https://contrib.rocks/image?repo=Cockatrice/Cockatrice" />
|
|
||||||
</a><br>
|
|
||||||
<sub><i>Made with <a href="https://contrib.rocks">contrib.rocks</a></i></sub>
|
|
||||||
</details>
|
|
||||||
|
|
||||||
### Translations [](https://transifex.com/cockatrice/cockatrice/)
|
Cockatrice uses Transifex to manage translations. You can help us bring <kbd>Cockatrice</kbd> and <kbd>Oracle</kbd> to your language and just adjust single wordings right from within your browser by visiting our [Transifex project page](https://explore.transifex.com/cockatrice/cockatrice/). The [Webatrice](https://github.com/seavor/Webatrice) web client manages its own translations in its repo.<br>
|
||||||
|
|
||||||
Cockatrice uses Transifex to manage translations. You can help us bring <kbd>Cockatrice</kbd>, <kbd>Oracle</kbd> and <kbd>Webatrice</kbd> to your language and just adjust single wordings right from within your browser by visiting our [Transifex project page](https://transifex.com/cockatrice/cockatrice/).<br>
|
Check out our [Translator FAQ](https://github.com/Cockatrice/Cockatrice/wiki/Translation-FAQ) for more information about getting involved, and join a group of hundreds of others!<br>
|
||||||
|
|
||||||
Check out our [Translator FAQ](https://github.com/Cockatrice/Cockatrice/wiki/Translation-FAQ) for more information about getting invovled, and join a group of hundreds of others!<br>
|
|
||||||
|
|
||||||
|
|
||||||
# Build [](https://github.com/Cockatrice/Cockatrice/actions/workflows/desktop-build.yml?query=branch%3Amaster+event%3Apush) [](https://github.com/Cockatrice/Cockatrice/actions/workflows/docker-release.yml?query=branch%3Amaster+event%3Apush) [](https://github.com/Cockatrice/Cockatrice/actions/workflows/web-build.yml?query=branch%3Amaster+event%3Apush)
|
# Build [](https://github.com/Cockatrice/Cockatrice/actions/workflows/desktop-build.yml?query=branch%3Amaster+event%3Apush) [](https://github.com/Cockatrice/Cockatrice/actions/workflows/docker-release.yml?query=branch%3Amaster+event%3Apush)
|
||||||
|
|
||||||
Dependencies: *(for minimum versions search our [CMake file](https://github.com/Cockatrice/Cockatrice/blob/master/CMakeLists.txt))*
|
Dependencies: *(for minimum versions search our [CMake file](https://github.com/Cockatrice/Cockatrice/blob/master/CMakeLists.txt))*
|
||||||
- [Qt](https://www.qt.io/developers/)
|
- [Qt](https://www.qt.io/developers/)
|
||||||
|
|
@ -124,8 +143,8 @@ You can then
|
||||||
make package
|
make package
|
||||||
```
|
```
|
||||||
|
|
||||||
>[!NOTE]
|
> [!NOTE]
|
||||||
>Detailed compiling instructions can be found in the Cockatrice wiki at [Compiling Cockatrice](https://github.com/Cockatrice/Cockatrice/wiki/Compiling-Cockatrice)
|
> Detailed compiling instructions can be found in the Cockatrice wiki at [Compiling Cockatrice](https://github.com/Cockatrice/Cockatrice/wiki/Compiling-Cockatrice)
|
||||||
|
|
||||||
<br>
|
<br>
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -42,7 +42,6 @@ tell disk image_name
|
||||||
set position of item "Cockatrice.app" to { 139, 214 }
|
set position of item "Cockatrice.app" to { 139, 214 }
|
||||||
set position of item "Oracle.app" to { 139, 414 }
|
set position of item "Oracle.app" to { 139, 414 }
|
||||||
set position of item "Servatrice.app" to { 139, 614 }
|
set position of item "Servatrice.app" to { 139, 614 }
|
||||||
set position of item "dbconverter.app" to { 1400, 1400 }
|
|
||||||
set position of item "Applications" to { 861, 414 }
|
set position of item "Applications" to { 861, 414 }
|
||||||
end tell
|
end tell
|
||||||
update without registering applications
|
update without registering applications
|
||||||
|
|
|
||||||
|
|
@ -1,12 +1,11 @@
|
||||||
# Find a compatible Qt version
|
# Find a compatible Qt version
|
||||||
# Inputs: WITH_SERVER, WITH_CLIENT, WITH_ORACLE, WITH_DBCONVERTER, FORCE_USE_QT5
|
# Inputs: WITH_SERVER, WITH_CLIENT, WITH_ORACLE, FORCE_USE_QT5
|
||||||
# Optional Input: QT6_DIR -- Hint as to where Qt6 lives on the system
|
# Optional Input: QT6_DIR -- Hint as to where Qt6 lives on the system
|
||||||
# Optional Input: QT5_DIR -- Hint as to where Qt5 lives on the system
|
# Optional Input: QT5_DIR -- Hint as to where Qt5 lives on the system
|
||||||
# Output: COCKATRICE_QT_VERSION_NAME -- Example values: Qt5, Qt6
|
# Output: COCKATRICE_QT_VERSION_NAME -- Example values: Qt5, Qt6
|
||||||
# Output: SERVATRICE_QT_MODULES
|
# Output: SERVATRICE_QT_MODULES
|
||||||
# Output: COCKATRICE_QT_MODULES
|
# Output: COCKATRICE_QT_MODULES
|
||||||
# Output: ORACLE_QT_MODULES
|
# Output: ORACLE_QT_MODULES
|
||||||
# Output: DBCONVERTER_QT_MODULES
|
|
||||||
# Output: TEST_QT_MODULES
|
# Output: TEST_QT_MODULES
|
||||||
|
|
||||||
set(REQUIRED_QT_COMPONENTS Core)
|
set(REQUIRED_QT_COMPONENTS Core)
|
||||||
|
|
@ -29,22 +28,21 @@ endif()
|
||||||
if(WITH_ORACLE)
|
if(WITH_ORACLE)
|
||||||
set(_ORACLE_NEEDED Concurrent Network Svg Widgets)
|
set(_ORACLE_NEEDED Concurrent Network Svg Widgets)
|
||||||
endif()
|
endif()
|
||||||
if(WITH_DBCONVERTER)
|
|
||||||
set(_DBCONVERTER_NEEDED Network Widgets)
|
|
||||||
endif()
|
|
||||||
if(TEST)
|
if(TEST)
|
||||||
set(_TEST_NEEDED Widgets)
|
# Union of Qt modules required across all test targets (independent of application targets).
|
||||||
|
# When adding a new test that needs additional Qt modules, add them here rather than in the test's CMakeLists.txt.
|
||||||
|
set(_TEST_NEEDED Concurrent Network Svg Widgets)
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
set(REQUIRED_QT_COMPONENTS ${REQUIRED_QT_COMPONENTS} ${_SERVATRICE_NEEDED} ${_COCKATRICE_NEEDED} ${_ORACLE_NEEDED}
|
set(REQUIRED_QT_COMPONENTS ${REQUIRED_QT_COMPONENTS} ${_SERVATRICE_NEEDED} ${_COCKATRICE_NEEDED} ${_ORACLE_NEEDED}
|
||||||
${_DBCONVERTER_NEEDED} ${_TEST_NEEDED}
|
${_TEST_NEEDED}
|
||||||
)
|
)
|
||||||
list(REMOVE_DUPLICATES REQUIRED_QT_COMPONENTS)
|
list(REMOVE_DUPLICATES REQUIRED_QT_COMPONENTS)
|
||||||
|
|
||||||
if(NOT FORCE_USE_QT5)
|
if(NOT FORCE_USE_QT5)
|
||||||
# Linguist is now a component in Qt6 instead of an external package
|
# Linguist is now a component in Qt6 instead of an external package
|
||||||
find_package(
|
find_package(
|
||||||
Qt6 6.2.3
|
Qt6 6.4.2
|
||||||
COMPONENTS ${REQUIRED_QT_COMPONENTS} Linguist
|
COMPONENTS ${REQUIRED_QT_COMPONENTS} Linguist
|
||||||
QUIET HINTS ${Qt6_DIR}
|
QUIET HINTS ${Qt6_DIR}
|
||||||
)
|
)
|
||||||
|
|
@ -63,7 +61,7 @@ if(Qt6_FOUND)
|
||||||
endif()
|
endif()
|
||||||
else()
|
else()
|
||||||
find_package(
|
find_package(
|
||||||
Qt5 5.8.0
|
Qt5 5.15.2
|
||||||
COMPONENTS ${REQUIRED_QT_COMPONENTS}
|
COMPONENTS ${REQUIRED_QT_COMPONENTS}
|
||||||
QUIET HINTS ${Qt5_DIR}
|
QUIET HINTS ${Qt5_DIR}
|
||||||
)
|
)
|
||||||
|
|
@ -112,7 +110,9 @@ message(DEBUG "QT_LIBRARY_DIR = ${QT_LIBRARY_DIR}")
|
||||||
string(REGEX REPLACE "([^;]+)" "${COCKATRICE_QT_VERSION_NAME}::\\1" SERVATRICE_QT_MODULES "${_SERVATRICE_NEEDED}")
|
string(REGEX REPLACE "([^;]+)" "${COCKATRICE_QT_VERSION_NAME}::\\1" SERVATRICE_QT_MODULES "${_SERVATRICE_NEEDED}")
|
||||||
string(REGEX REPLACE "([^;]+)" "${COCKATRICE_QT_VERSION_NAME}::\\1" COCKATRICE_QT_MODULES "${_COCKATRICE_NEEDED}")
|
string(REGEX REPLACE "([^;]+)" "${COCKATRICE_QT_VERSION_NAME}::\\1" COCKATRICE_QT_MODULES "${_COCKATRICE_NEEDED}")
|
||||||
string(REGEX REPLACE "([^;]+)" "${COCKATRICE_QT_VERSION_NAME}::\\1" ORACLE_QT_MODULES "${_ORACLE_NEEDED}")
|
string(REGEX REPLACE "([^;]+)" "${COCKATRICE_QT_VERSION_NAME}::\\1" ORACLE_QT_MODULES "${_ORACLE_NEEDED}")
|
||||||
string(REGEX REPLACE "([^;]+)" "${COCKATRICE_QT_VERSION_NAME}::\\1" DB_CONVERTER_QT_MODULES "${_DBCONVERTER_NEEDED}")
|
|
||||||
string(REGEX REPLACE "([^;]+)" "${COCKATRICE_QT_VERSION_NAME}::\\1" TEST_QT_MODULES "${_TEST_NEEDED}")
|
string(REGEX REPLACE "([^;]+)" "${COCKATRICE_QT_VERSION_NAME}::\\1" TEST_QT_MODULES "${_TEST_NEEDED}")
|
||||||
|
|
||||||
|
# Core-only export (useful for headless libs)
|
||||||
|
set(QT_CORE_MODULE "${COCKATRICE_QT_VERSION_NAME}::Core")
|
||||||
|
|
||||||
message(STATUS "Found Qt ${${COCKATRICE_QT_VERSION_NAME}_VERSION} at: ${${COCKATRICE_QT_VERSION_NAME}_DIR}")
|
message(STATUS "Found Qt ${${COCKATRICE_QT_VERSION_NAME}_VERSION} at: ${${COCKATRICE_QT_VERSION_NAME}_DIR}")
|
||||||
|
|
|
||||||
|
|
@ -11,6 +11,7 @@ SetCompressor LZMA
|
||||||
Var NormalDestDir
|
Var NormalDestDir
|
||||||
Var PortableDestDir
|
Var PortableDestDir
|
||||||
Var PortableMode
|
Var PortableMode
|
||||||
|
Var ReinstallMode
|
||||||
|
|
||||||
!include LogicLib.nsh
|
!include LogicLib.nsh
|
||||||
!include FileFunc.nsh
|
!include FileFunc.nsh
|
||||||
|
|
@ -26,14 +27,25 @@ Var PortableMode
|
||||||
!define MUI_WELCOMEPAGE_TEXT "This wizard will guide you through the installation of Cockatrice.$\r$\n$\r$\nClick Next to continue."
|
!define MUI_WELCOMEPAGE_TEXT "This wizard will guide you through the installation of Cockatrice.$\r$\n$\r$\nClick Next to continue."
|
||||||
!define MUI_FINISHPAGE_RUN "$INSTDIR/cockatrice.exe"
|
!define MUI_FINISHPAGE_RUN "$INSTDIR/cockatrice.exe"
|
||||||
!define MUI_FINISHPAGE_RUN_TEXT "Run 'Cockatrice' now"
|
!define MUI_FINISHPAGE_RUN_TEXT "Run 'Cockatrice' now"
|
||||||
|
!define MUI_ICON "${NSIS_SOURCE_PATH}\cockatrice\resources\appicon.ico"
|
||||||
|
|
||||||
|
!define MUI_PAGE_CUSTOMFUNCTION_PRE SkipIfReinstall
|
||||||
!insertmacro MUI_PAGE_WELCOME
|
!insertmacro MUI_PAGE_WELCOME
|
||||||
|
|
||||||
|
!define MUI_PAGE_CUSTOMFUNCTION_PRE SkipIfReinstall
|
||||||
!insertmacro MUI_PAGE_LICENSE "${NSIS_SOURCE_PATH}\LICENSE"
|
!insertmacro MUI_PAGE_LICENSE "${NSIS_SOURCE_PATH}\LICENSE"
|
||||||
|
|
||||||
Page Custom PortableModePageCreate PortableModePageLeave
|
Page Custom PortableModePageCreate PortableModePageLeave
|
||||||
!define MUI_PAGE_CUSTOMFUNCTION_PRE componentsPagePre
|
!define MUI_PAGE_CUSTOMFUNCTION_PRE componentsPagePre
|
||||||
!insertmacro MUI_PAGE_COMPONENTS
|
!insertmacro MUI_PAGE_COMPONENTS
|
||||||
|
|
||||||
|
!define MUI_PAGE_CUSTOMFUNCTION_PRE SkipIfReinstall
|
||||||
!insertmacro MUI_PAGE_DIRECTORY
|
!insertmacro MUI_PAGE_DIRECTORY
|
||||||
|
|
||||||
|
!define MUI_PAGE_CUSTOMFUNCTION_PRE SkipIfReinstall
|
||||||
!insertmacro MUI_PAGE_INSTFILES
|
!insertmacro MUI_PAGE_INSTFILES
|
||||||
|
|
||||||
|
!define MUI_PAGE_CUSTOMFUNCTION_PRE SkipIfReinstall
|
||||||
!insertmacro MUI_PAGE_FINISH
|
!insertmacro MUI_PAGE_FINISH
|
||||||
|
|
||||||
!insertmacro MUI_UNPAGE_CONFIRM
|
!insertmacro MUI_UNPAGE_CONFIRM
|
||||||
|
|
@ -72,6 +84,7 @@ ${IfNot} ${Errors}
|
||||||
MessageBox MB_ICONINFORMATION|MB_SETFOREGROUND "\
|
MessageBox MB_ICONINFORMATION|MB_SETFOREGROUND "\
|
||||||
/PORTABLE : Install in portable mode$\n\
|
/PORTABLE : Install in portable mode$\n\
|
||||||
/S : Silent install$\n\
|
/S : Silent install$\n\
|
||||||
|
/R : Silent upgrade$\n\
|
||||||
/D=%directory% : Specify destination directory$\n"
|
/D=%directory% : Specify destination directory$\n"
|
||||||
Quit
|
Quit
|
||||||
${EndIf}
|
${EndIf}
|
||||||
|
|
@ -89,6 +102,16 @@ ${Else}
|
||||||
${EndIf}
|
${EndIf}
|
||||||
${EndIf}
|
${EndIf}
|
||||||
|
|
||||||
|
ClearErrors
|
||||||
|
${GetOptions} $9 "/R" $8
|
||||||
|
${IfNot} ${Errors}
|
||||||
|
StrCpy $ReinstallMode 1
|
||||||
|
SetSilent silent
|
||||||
|
SetAutoClose true
|
||||||
|
${Else}
|
||||||
|
StrCpy $ReinstallMode 0
|
||||||
|
${EndIf}
|
||||||
|
|
||||||
${If} $InstDir == ""
|
${If} $InstDir == ""
|
||||||
; User did not use /D to specify a directory,
|
; User did not use /D to specify a directory,
|
||||||
; we need to set a default based on the install mode
|
; we need to set a default based on the install mode
|
||||||
|
|
@ -96,6 +119,22 @@ ${If} $InstDir == ""
|
||||||
${EndIf}
|
${EndIf}
|
||||||
Call SetModeDestinationFromInstdir
|
Call SetModeDestinationFromInstdir
|
||||||
|
|
||||||
|
; --- Detect portable install when using /R ---
|
||||||
|
${If} $ReinstallMode = 1
|
||||||
|
IfFileExists "$InstDir\portable.dat" 0 not_portable
|
||||||
|
StrCpy $PortableMode 1
|
||||||
|
Goto portable_done
|
||||||
|
|
||||||
|
not_portable:
|
||||||
|
StrCpy $PortableMode 0
|
||||||
|
|
||||||
|
portable_done:
|
||||||
|
${EndIf}
|
||||||
|
|
||||||
|
${If} $ReinstallMode = 1
|
||||||
|
Call AutoUninstallIfNeeded
|
||||||
|
${EndIf}
|
||||||
|
|
||||||
FunctionEnd
|
FunctionEnd
|
||||||
|
|
||||||
Function un.onInit
|
Function un.onInit
|
||||||
|
|
@ -125,8 +164,46 @@ ${Else}
|
||||||
${EndIf}
|
${EndIf}
|
||||||
FunctionEnd
|
FunctionEnd
|
||||||
|
|
||||||
|
Function SkipIfReinstall
|
||||||
|
${If} $ReinstallMode = 1
|
||||||
|
Abort
|
||||||
|
${EndIf}
|
||||||
|
FunctionEnd
|
||||||
|
|
||||||
|
Function AutoUninstallIfNeeded
|
||||||
|
|
||||||
|
SetShellVarContext all
|
||||||
|
|
||||||
|
; --- 32-bit uninstall ---
|
||||||
|
SetRegView 32
|
||||||
|
ReadRegStr $R0 HKLM "SOFTWARE\Microsoft\Windows\CurrentVersion\Uninstall\Cockatrice" "QuietUninstallString"
|
||||||
|
|
||||||
|
StrCmp $R0 "" done32
|
||||||
|
DetailPrint "Removing previous version (32-bit)..."
|
||||||
|
ExecWait '$R0'
|
||||||
|
|
||||||
|
done32:
|
||||||
|
|
||||||
|
; --- 64-bit uninstall ---
|
||||||
|
${If} ${RunningX64}
|
||||||
|
SetRegView 64
|
||||||
|
ReadRegStr $R0 HKLM "SOFTWARE\Microsoft\Windows\CurrentVersion\Uninstall\Cockatrice" "QuietUninstallString"
|
||||||
|
|
||||||
|
StrCmp $R0 "" done64
|
||||||
|
DetailPrint "Removing previous version (64-bit)..."
|
||||||
|
ExecWait '$R0'
|
||||||
|
|
||||||
|
done64:
|
||||||
|
${EndIf}
|
||||||
|
|
||||||
|
FunctionEnd
|
||||||
|
|
||||||
Function PortableModePageCreate
|
Function PortableModePageCreate
|
||||||
|
|
||||||
|
${If} $ReinstallMode = 1
|
||||||
|
Abort
|
||||||
|
${EndIf}
|
||||||
|
|
||||||
Call SetModeDestinationFromInstdir ; If the user clicks BACK on the directory page we will remember their mode specific directory
|
Call SetModeDestinationFromInstdir ; If the user clicks BACK on the directory page we will remember their mode specific directory
|
||||||
!insertmacro MUI_HEADER_TEXT "Install Mode" "Choose how you want to install Cockatrice."
|
!insertmacro MUI_HEADER_TEXT "Install Mode" "Choose how you want to install Cockatrice."
|
||||||
nsDialogs::Create 1018
|
nsDialogs::Create 1018
|
||||||
|
|
@ -158,6 +235,11 @@ ${EndIf}
|
||||||
FunctionEnd
|
FunctionEnd
|
||||||
|
|
||||||
Function componentsPagePre
|
Function componentsPagePre
|
||||||
|
|
||||||
|
${If} $ReinstallMode = 1
|
||||||
|
Return
|
||||||
|
${EndIf}
|
||||||
|
|
||||||
${If} $PortableMode = 0
|
${If} $PortableMode = 0
|
||||||
SetShellVarContext all
|
SetShellVarContext all
|
||||||
|
|
||||||
|
|
@ -167,8 +249,12 @@ ${If} $PortableMode = 0
|
||||||
ReadRegStr $R0 HKLM "SOFTWARE\Microsoft\Windows\CurrentVersion\Uninstall\Cockatrice" "UninstallString"
|
ReadRegStr $R0 HKLM "SOFTWARE\Microsoft\Windows\CurrentVersion\Uninstall\Cockatrice" "UninstallString"
|
||||||
StrCmp $R0 "" done32
|
StrCmp $R0 "" done32
|
||||||
|
|
||||||
MessageBox MB_OKCANCEL|MB_ICONEXCLAMATION "A previous version of Cockatrice must be uninstalled before installing the new one." IDOK uninst32
|
${If} $ReinstallMode = 0
|
||||||
Abort
|
MessageBox MB_OKCANCEL|MB_ICONEXCLAMATION "A previous version of Cockatrice must be uninstalled before installing the new one." IDOK uninst32
|
||||||
|
Abort
|
||||||
|
${Else}
|
||||||
|
Goto uninst32
|
||||||
|
${EndIf}
|
||||||
|
|
||||||
uninst32:
|
uninst32:
|
||||||
ClearErrors
|
ClearErrors
|
||||||
|
|
@ -183,8 +269,12 @@ ${If} $PortableMode = 0
|
||||||
ReadRegStr $R0 HKLM "SOFTWARE\Microsoft\Windows\CurrentVersion\Uninstall\Cockatrice" "UninstallString"
|
ReadRegStr $R0 HKLM "SOFTWARE\Microsoft\Windows\CurrentVersion\Uninstall\Cockatrice" "UninstallString"
|
||||||
StrCmp $R0 "" done64
|
StrCmp $R0 "" done64
|
||||||
|
|
||||||
MessageBox MB_OKCANCEL|MB_ICONEXCLAMATION "A previous version of Cockatrice must be uninstalled before installing the new one." IDOK uninst64
|
${If} $ReinstallMode = 0
|
||||||
Abort
|
MessageBox MB_OKCANCEL|MB_ICONEXCLAMATION "A previous version of Cockatrice must be uninstalled before installing the new one." IDOK uninst64
|
||||||
|
Abort
|
||||||
|
${Else}
|
||||||
|
Goto uninst64
|
||||||
|
${EndIf}
|
||||||
|
|
||||||
uninst64:
|
uninst64:
|
||||||
ClearErrors
|
ClearErrors
|
||||||
|
|
@ -213,7 +303,6 @@ ${AndIf} ${FileExists} "$INSTDIR\portable.dat"
|
||||||
Delete "$INSTDIR\uninstall.exe"
|
Delete "$INSTDIR\uninstall.exe"
|
||||||
Delete "$INSTDIR\cockatrice.exe"
|
Delete "$INSTDIR\cockatrice.exe"
|
||||||
Delete "$INSTDIR\oracle.exe"
|
Delete "$INSTDIR\oracle.exe"
|
||||||
Delete "$INSTDIR\dbconverter.exe"
|
|
||||||
Delete "$INSTDIR\servatrice.exe"
|
Delete "$INSTDIR\servatrice.exe"
|
||||||
Delete "$INSTDIR\Qt*.dll"
|
Delete "$INSTDIR\Qt*.dll"
|
||||||
Delete "$INSTDIR\libmysql.dll"
|
Delete "$INSTDIR\libmysql.dll"
|
||||||
|
|
@ -277,6 +366,12 @@ ${Else}
|
||||||
FileWrite $0 "PORTABLE"
|
FileWrite $0 "PORTABLE"
|
||||||
FileClose $0
|
FileClose $0
|
||||||
${EndIf}
|
${EndIf}
|
||||||
|
|
||||||
|
${If} $ReinstallMode = 1
|
||||||
|
IfFileExists "$INSTDIR\cockatrice.exe" 0 +2
|
||||||
|
Exec '"$INSTDIR\cockatrice.exe"'
|
||||||
|
${EndIf}
|
||||||
|
|
||||||
SectionEnd
|
SectionEnd
|
||||||
|
|
||||||
Section "Start menu item" SecStartMenu
|
Section "Start menu item" SecStartMenu
|
||||||
|
|
|
||||||
|
|
@ -3,7 +3,7 @@
|
||||||
string(LENGTH "$ENV{MACOS_CERTIFICATE_NAME}" MACOS_CERTIFICATE_NAME_LEN)
|
string(LENGTH "$ENV{MACOS_CERTIFICATE_NAME}" MACOS_CERTIFICATE_NAME_LEN)
|
||||||
|
|
||||||
if(APPLE AND MACOS_CERTIFICATE_NAME_LEN GREATER 0)
|
if(APPLE AND MACOS_CERTIFICATE_NAME_LEN GREATER 0)
|
||||||
set(APPLICATIONS "cockatrice" "servatrice" "oracle" "dbconverter")
|
set(APPLICATIONS "cockatrice" "servatrice" "oracle")
|
||||||
foreach(app_name IN LISTS APPLICATIONS)
|
foreach(app_name IN LISTS APPLICATIONS)
|
||||||
set(FULL_APP_PATH "${CPACK_TEMPORARY_INSTALL_DIRECTORY}/${app_name}.app")
|
set(FULL_APP_PATH "${CPACK_TEMPORARY_INSTALL_DIRECTORY}/${app_name}.app")
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -213,7 +213,8 @@ set(PROJECT_VERSION_FRIENDLY "${PROJECT_VERSION} (${GIT_COMMIT_DATE_FRIENDLY})")
|
||||||
# Format: <program name>[-ReleaseName]-MAJ.MIN.PATCH[-prerelease_label]
|
# Format: <program name>[-ReleaseName]-MAJ.MIN.PATCH[-prerelease_label]
|
||||||
set(PROJECT_VERSION_FILENAME "${PROJECT_NAME}")
|
set(PROJECT_VERSION_FILENAME "${PROJECT_NAME}")
|
||||||
if(PROJECT_VERSION_RELEASENAME)
|
if(PROJECT_VERSION_RELEASENAME)
|
||||||
set(PROJECT_VERSION_FILENAME "${PROJECT_VERSION_FILENAME}-${PROJECT_VERSION_RELEASENAME}")
|
string(REPLACE " " "-" PROJECT_VERSION_RELEASENAME_SAFE "${PROJECT_VERSION_RELEASENAME}")
|
||||||
|
set(PROJECT_VERSION_FILENAME "${PROJECT_VERSION_FILENAME}-${PROJECT_VERSION_RELEASENAME_SAFE}")
|
||||||
endif()
|
endif()
|
||||||
set(PROJECT_VERSION_FILENAME "${PROJECT_VERSION_FILENAME}-${PROJECT_VERSION}")
|
set(PROJECT_VERSION_FILENAME "${PROJECT_VERSION_FILENAME}-${PROJECT_VERSION}")
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -1,15 +1,16 @@
|
||||||
cmake_minimum_required(VERSION 3.2)
|
cmake_minimum_required(VERSION 3.10)
|
||||||
|
|
||||||
project(gtest-download LANGUAGES NONE)
|
project(gtest-download LANGUAGES NONE)
|
||||||
|
|
||||||
include(ExternalProject)
|
include(ExternalProject)
|
||||||
ExternalProject_Add(googletest
|
externalproject_add(
|
||||||
URL https://github.com/google/googletest/archive/release-1.11.0.zip
|
googletest
|
||||||
URL_HASH SHA1=9ffb7b5923f4a8fcdabf2f42c6540cce299f44c0
|
URL https://github.com/google/googletest/archive/refs/tags/v1.17.0.zip
|
||||||
|
URL_HASH SHA1=f638fa0e724760e2ba07ff8cfba32cd644e1ce28
|
||||||
SOURCE_DIR "${CMAKE_BINARY_DIR}/gtest-src"
|
SOURCE_DIR "${CMAKE_BINARY_DIR}/gtest-src"
|
||||||
BINARY_DIR "${CMAKE_BINARY_DIR}/gtest-build"
|
BINARY_DIR "${CMAKE_BINARY_DIR}/gtest-build"
|
||||||
CONFIGURE_COMMAND ""
|
CONFIGURE_COMMAND ""
|
||||||
BUILD_COMMAND ""
|
BUILD_COMMAND ""
|
||||||
INSTALL_COMMAND ""
|
INSTALL_COMMAND ""
|
||||||
TEST_COMMAND ""
|
TEST_COMMAND ""
|
||||||
)
|
)
|
||||||
|
|
|
||||||
|
|
@ -7,246 +7,348 @@ project(Cockatrice VERSION "${PROJECT_VERSION_MAJOR}.${PROJECT_VERSION_MINOR}.${
|
||||||
set(cockatrice_SOURCES
|
set(cockatrice_SOURCES
|
||||||
${VERSION_STRING_CPP}
|
${VERSION_STRING_CPP}
|
||||||
# sort by alphabetical order, so that there is no debate about where to add new sources to the list
|
# sort by alphabetical order, so that there is no debate about where to add new sources to the list
|
||||||
src/client/game_logic/abstract_client.cpp
|
src/client/network/connection_controller/remote_connection_controller.cpp
|
||||||
src/client/game_logic/key_signals.cpp
|
src/client/network/update/client/update_downloader.cpp
|
||||||
src/client/get_text_with_max.cpp
|
src/client/network/interfaces/deck_stats_interface.cpp
|
||||||
src/client/menus/deck_editor/deck_editor_menu.cpp
|
src/client/network/interfaces/tapped_out_interface.cpp
|
||||||
src/client/network/client_update_checker.cpp
|
src/client/network/parsers/deck_link_to_api_transformer.cpp
|
||||||
src/client/network/release_channel.cpp
|
src/client/network/update/client/client_update_checker.cpp
|
||||||
src/client/network/replay_timeline_widget.cpp
|
src/client/network/update/client/release_channel.cpp
|
||||||
src/client/network/sets_model.cpp
|
src/client/network/update/card_spoiler/spoiler_background_updater.cpp
|
||||||
src/client/network/spoiler_background_updater.cpp
|
|
||||||
src/client/sound_engine.cpp
|
src/client/sound_engine.cpp
|
||||||
src/client/tabs/abstract_tab_deck_editor.cpp
|
src/client/settings/cache_settings.cpp
|
||||||
src/client/tabs/api/edhrec/tab_edhrec.cpp
|
src/client/settings/card_counter_settings.cpp
|
||||||
src/client/tabs/api/edhrec/tab_edhrec_main.cpp
|
src/client/settings/shortcut_treeview.cpp
|
||||||
src/client/tabs/api/edhrec/display/commander/edhrec_commander_api_response_display_widget.cpp
|
src/client/settings/shortcuts_settings.cpp
|
||||||
src/client/tabs/api/edhrec/display/commander/edhrec_commander_api_response_navigation_widget.cpp
|
src/interface/deck_loader/card_node_function.cpp
|
||||||
src/client/tabs/api/edhrec/display/card_prices/edhrec_api_response_card_prices_display_widget.cpp
|
src/interface/deck_loader/deck_file_format.cpp
|
||||||
src/client/tabs/api/edhrec/display/cards/edhrec_api_response_card_details_display_widget.cpp
|
src/interface/deck_loader/deck_loader.cpp
|
||||||
src/client/tabs/api/edhrec/display/cards/edhrec_api_response_card_inclusion_display_widget.cpp
|
src/interface/deck_loader/loaded_deck.cpp
|
||||||
src/client/tabs/api/edhrec/display/cards/edhrec_api_response_card_list_display_widget.cpp
|
src/interface/widgets/dialogs/dlg_connect.cpp
|
||||||
src/client/tabs/api/edhrec/display/cards/edhrec_api_response_card_synergy_display_widget.cpp
|
src/interface/widgets/dialogs/dlg_convert_deck_to_cod_format.cpp
|
||||||
src/client/tabs/api/edhrec/display/commander/edhrec_api_response_commander_details_display_widget.cpp
|
src/interface/widgets/dialogs/dlg_create_game.cpp
|
||||||
src/client/tabs/api/edhrec/display/top_cards/edhrec_top_cards_api_response_display_widget.cpp
|
src/interface/widgets/dialogs/dlg_default_tags_editor.cpp
|
||||||
src/client/tabs/api/edhrec/display/top_commander/edhrec_top_commanders_api_response_display_widget.cpp
|
src/interface/widgets/dialogs/dlg_edit_avatar.cpp
|
||||||
src/client/tabs/api/edhrec/display/top_tags/edhrec_top_tags_api_response_display_widget.cpp
|
src/interface/widgets/dialogs/dlg_edit_password.cpp
|
||||||
src/client/tabs/api/edhrec/api_response/archidekt_links/edhrec_api_response_archidekt_links.cpp
|
src/interface/widgets/dialogs/dlg_edit_tokens.cpp
|
||||||
src/client/tabs/api/edhrec/api_response/commander/edhrec_commander_api_response_average_deck_statistics.cpp
|
src/interface/widgets/dialogs/dlg_edit_user.cpp
|
||||||
src/client/tabs/api/edhrec/api_response/cards/edhrec_api_response_card_details.cpp
|
src/interface/widgets/dialogs/dlg_filter_games.cpp
|
||||||
src/client/tabs/api/edhrec/api_response/cards/edhrec_api_response_card_list.cpp
|
src/interface/widgets/dialogs/dlg_forgot_password_challenge.cpp
|
||||||
src/client/tabs/api/edhrec/api_response/cards/edhrec_api_response_card_container.cpp
|
src/interface/widgets/dialogs/dlg_forgot_password_request.cpp
|
||||||
src/client/tabs/api/edhrec/api_response/card_prices/edhrec_api_response_card_prices.cpp
|
src/interface/widgets/dialogs/dlg_forgot_password_reset.cpp
|
||||||
src/client/tabs/api/edhrec/api_response/cards/edhrec_commander_api_response_commander_details.cpp
|
src/interface/widgets/dialogs/dlg_load_deck.cpp
|
||||||
src/client/tabs/api/edhrec/api_response/commander/edhrec_commander_api_response.cpp
|
src/interface/widgets/dialogs/dlg_load_deck_from_clipboard.cpp
|
||||||
src/client/tabs/api/edhrec/api_response/average_deck/edhrec_average_deck_api_response.cpp
|
src/interface/widgets/dialogs/dlg_load_deck_from_website.cpp
|
||||||
src/client/tabs/api/edhrec/api_response/average_deck/edhrec_deck_api_response.cpp
|
src/interface/widgets/dialogs/dlg_load_remote_deck.cpp
|
||||||
src/client/tabs/api/edhrec/api_response/top_cards/edhrec_top_cards_api_response.cpp
|
src/interface/widgets/dialogs/dlg_local_game_options.cpp
|
||||||
src/client/tabs/api/edhrec/api_response/top_commanders/edhrec_top_commanders_api_response.cpp
|
src/interface/widgets/dialogs/dlg_manage_sets.cpp
|
||||||
src/client/tabs/api/edhrec/api_response/top_tags/edhrec_top_tags_api_response.cpp
|
src/interface/widgets/dialogs/dlg_register.cpp
|
||||||
src/client/tabs/tab.cpp
|
src/interface/widgets/dialogs/dlg_select_set_for_cards.cpp
|
||||||
src/client/tabs/tab_account.cpp
|
src/interface/widgets/dialogs/dlg_settings.cpp
|
||||||
src/client/tabs/tab_admin.cpp
|
src/interface/widgets/dialogs/dlg_startup_card_check.cpp
|
||||||
src/client/tabs/tab_deck_editor.cpp
|
src/interface/widgets/dialogs/dlg_tip_of_the_day.cpp
|
||||||
src/client/tabs/tab_deck_storage.cpp
|
src/interface/widgets/dialogs/dlg_update.cpp
|
||||||
src/client/tabs/tab_game.cpp
|
src/interface/widgets/dialogs/dlg_view_log.cpp
|
||||||
src/client/tabs/tab_logs.cpp
|
src/interface/widgets/dialogs/override_printing_warning.cpp
|
||||||
src/client/tabs/tab_message.cpp
|
src/interface/widgets/dialogs/tip_of_the_day.cpp
|
||||||
src/client/tabs/tab_replays.cpp
|
src/filters/deck_filter_string.cpp
|
||||||
src/client/tabs/tab_room.cpp
|
src/filters/filter_builder.cpp
|
||||||
src/client/tabs/tab_server.cpp
|
src/filters/filter_tree_model.cpp
|
||||||
src/client/tabs/tab_supervisor.cpp
|
src/filters/syntax_help.cpp
|
||||||
src/client/tabs/tab_visual_database_display.cpp
|
src/game/abstract_game.cpp
|
||||||
src/client/tabs/visual_deck_editor/tab_deck_editor_visual.cpp
|
src/game/arrow_registry.cpp
|
||||||
src/client/tabs/visual_deck_editor/tab_deck_editor_visual_tab_widget.cpp
|
src/game_graphics/board/abstract_card_drag_item.cpp
|
||||||
src/client/tabs/visual_deck_storage/tab_deck_storage_visual.cpp
|
src/game_graphics/board/abstract_card_item.cpp
|
||||||
src/client/tapped_out_interface.cpp
|
src/game_graphics/board/abstract_counter.cpp
|
||||||
src/client/translate_counter_name.cpp
|
src/game/board/arrow_data.cpp
|
||||||
src/client/ui/layouts/flow_layout.cpp
|
src/game_graphics/board/arrow_item.cpp
|
||||||
src/client/ui/layouts/overlap_layout.cpp
|
src/game_graphics/board/arrow_target.cpp
|
||||||
src/client/ui/line_edit_completer.cpp
|
src/game_graphics/board/card_drag_item.cpp
|
||||||
src/client/ui/phases_toolbar.cpp
|
src/game_graphics/board/card_item.cpp
|
||||||
src/client/ui/picture_loader/picture_loader.cpp
|
|
||||||
src/client/ui/picture_loader/picture_loader_worker.cpp
|
|
||||||
src/client/ui/picture_loader/picture_to_load.cpp
|
|
||||||
src/client/ui/pixel_map_generator.cpp
|
|
||||||
src/client/ui/theme_manager.cpp
|
|
||||||
src/client/ui/tip_of_the_day.cpp
|
|
||||||
src/client/ui/widgets/cards/additional_info/color_identity_widget.cpp
|
|
||||||
src/client/ui/widgets/cards/additional_info/mana_cost_widget.cpp
|
|
||||||
src/client/ui/widgets/cards/additional_info/mana_symbol_widget.cpp
|
|
||||||
src/client/ui/widgets/cards/card_group_display_widgets/card_group_display_widget.cpp
|
|
||||||
src/client/ui/widgets/cards/card_group_display_widgets/flat_card_group_display_widget.cpp
|
|
||||||
src/client/ui/widgets/cards/card_group_display_widgets/overlapped_card_group_display_widget.cpp
|
|
||||||
src/client/ui/widgets/cards/card_info_display_widget.cpp
|
|
||||||
src/client/ui/widgets/cards/card_info_frame_widget.cpp
|
|
||||||
src/client/ui/widgets/cards/card_info_picture_enlarged_widget.cpp
|
|
||||||
src/client/ui/widgets/cards/card_info_picture_widget.cpp
|
|
||||||
src/client/ui/widgets/cards/card_info_picture_with_text_overlay_widget.cpp
|
|
||||||
src/client/ui/widgets/cards/card_info_text_widget.cpp
|
|
||||||
src/client/ui/widgets/cards/card_size_widget.cpp
|
|
||||||
src/client/ui/widgets/cards/deck_card_zone_display_widget.cpp
|
|
||||||
src/client/ui/widgets/cards/deck_preview_card_picture_widget.cpp
|
|
||||||
src/client/ui/widgets/deck_analytics/deck_analytics_widget.cpp
|
|
||||||
src/client/ui/widgets/deck_analytics/mana_base_widget.cpp
|
|
||||||
src/client/ui/widgets/deck_analytics/mana_curve_widget.cpp
|
|
||||||
src/client/ui/widgets/deck_analytics/mana_devotion_widget.cpp
|
|
||||||
src/client/ui/widgets/deck_editor/deck_editor_card_info_dock_widget.cpp
|
|
||||||
src/client/ui/widgets/deck_editor/deck_editor_database_display_widget.cpp
|
|
||||||
src/client/ui/widgets/deck_editor/deck_editor_deck_dock_widget.cpp
|
|
||||||
src/client/ui/widgets/deck_editor/deck_editor_filter_dock_widget.cpp
|
|
||||||
src/client/ui/widgets/deck_editor/deck_editor_printing_selector_dock_widget.cpp
|
|
||||||
src/client/ui/widgets/general/display/banner_widget.cpp
|
|
||||||
src/client/ui/widgets/general/display/bar_widget.cpp
|
|
||||||
src/client/ui/widgets/general/display/dynamic_font_size_label.cpp
|
|
||||||
src/client/ui/widgets/general/display/dynamic_font_size_push_button.cpp
|
|
||||||
src/client/ui/widgets/general/display/labeled_input.cpp
|
|
||||||
src/client/ui/widgets/general/display/percent_bar_widget.cpp
|
|
||||||
src/client/ui/widgets/general/display/shadow_background_label.cpp
|
|
||||||
src/client/ui/widgets/general/layout_containers/flow_widget.cpp
|
|
||||||
src/client/ui/widgets/general/layout_containers/overlap_control_widget.cpp
|
|
||||||
src/client/ui/widgets/general/layout_containers/overlap_widget.cpp
|
|
||||||
src/client/ui/widgets/printing_selector/all_zones_card_amount_widget.cpp
|
|
||||||
src/client/ui/widgets/printing_selector/card_amount_widget.cpp
|
|
||||||
src/client/ui/widgets/printing_selector/printing_selector.cpp
|
|
||||||
src/client/ui/widgets/printing_selector/printing_selector_card_display_widget.cpp
|
|
||||||
src/client/ui/widgets/printing_selector/printing_selector_card_overlay_widget.cpp
|
|
||||||
src/client/ui/widgets/printing_selector/printing_selector_card_search_widget.cpp
|
|
||||||
src/client/ui/widgets/printing_selector/printing_selector_card_selection_widget.cpp
|
|
||||||
src/client/ui/widgets/printing_selector/printing_selector_card_sorting_widget.cpp
|
|
||||||
src/client/ui/widgets/printing_selector/set_name_and_collectors_number_display_widget.cpp
|
|
||||||
src/client/ui/widgets/quick_settings/settings_button_widget.cpp
|
|
||||||
src/client/ui/widgets/quick_settings/settings_popup_widget.cpp
|
|
||||||
src/client/ui/widgets/visual_database_display/visual_database_display_widget.cpp
|
|
||||||
src/client/ui/widgets/visual_database_display/visual_database_display_color_filter_widget.cpp
|
|
||||||
src/client/ui/widgets/visual_database_display/visual_database_display_sub_type_filter_widget.cpp
|
|
||||||
src/client/ui/widgets/visual_database_display/visual_database_display_name_filter_widget.cpp
|
|
||||||
src/client/ui/widgets/visual_database_display/visual_database_display_main_type_filter_widget.cpp
|
|
||||||
src/client/ui/widgets/visual_database_display/visual_database_display_set_filter_widget.cpp
|
|
||||||
src/client/ui/widgets/visual_database_display/visual_database_display_filter_save_load_widget.cpp
|
|
||||||
src/client/ui/widgets/visual_database_display/visual_database_filter_display_widget.cpp
|
|
||||||
src/client/ui/widgets/visual_deck_editor/visual_deck_editor_widget.cpp
|
|
||||||
src/client/ui/widgets/visual_deck_editor/visual_deck_editor_sample_hand_widget.cpp
|
|
||||||
src/client/ui/widgets/visual_deck_storage/deck_preview/deck_preview_color_identity_filter_widget.cpp
|
|
||||||
src/client/ui/widgets/visual_deck_storage/deck_preview/deck_preview_deck_tags_display_widget.cpp
|
|
||||||
src/client/ui/widgets/visual_deck_storage/deck_preview/deck_preview_tag_addition_widget.cpp
|
|
||||||
src/client/ui/widgets/visual_deck_storage/deck_preview/deck_preview_tag_dialog.cpp
|
|
||||||
src/client/ui/widgets/visual_deck_storage/deck_preview/deck_preview_tag_display_widget.cpp
|
|
||||||
src/client/ui/widgets/visual_deck_storage/deck_preview/deck_preview_tag_item_widget.cpp
|
|
||||||
src/client/ui/widgets/visual_deck_storage/deck_preview/deck_preview_widget.cpp
|
|
||||||
src/client/ui/widgets/visual_deck_storage/visual_deck_storage_folder_display_widget.cpp
|
|
||||||
src/client/ui/widgets/visual_deck_storage/visual_deck_storage_quick_settings_widget.cpp
|
|
||||||
src/client/ui/widgets/visual_deck_storage/visual_deck_storage_search_widget.cpp
|
|
||||||
src/client/ui/widgets/visual_deck_storage/visual_deck_storage_sort_widget.cpp
|
|
||||||
src/client/ui/widgets/visual_deck_storage/visual_deck_storage_tag_filter_widget.cpp
|
|
||||||
src/client/ui/widgets/visual_deck_storage/visual_deck_storage_widget.cpp
|
|
||||||
src/client/ui/window_main.cpp
|
|
||||||
src/client/update_downloader.cpp
|
|
||||||
src/deck/custom_line_edit.cpp
|
|
||||||
src/deck/deck_list_model.cpp
|
|
||||||
src/deck/deck_loader.cpp
|
|
||||||
src/deck/deck_stats_interface.cpp
|
|
||||||
src/dialogs/dlg_connect.cpp
|
|
||||||
src/dialogs/dlg_convert_deck_to_cod_format.cpp
|
|
||||||
src/dialogs/dlg_create_game.cpp
|
|
||||||
src/dialogs/dlg_create_token.cpp
|
|
||||||
src/dialogs/dlg_edit_avatar.cpp
|
|
||||||
src/dialogs/dlg_edit_password.cpp
|
|
||||||
src/dialogs/dlg_edit_tokens.cpp
|
|
||||||
src/dialogs/dlg_edit_user.cpp
|
|
||||||
src/dialogs/dlg_filter_games.cpp
|
|
||||||
src/dialogs/dlg_forgot_password_challenge.cpp
|
|
||||||
src/dialogs/dlg_forgot_password_request.cpp
|
|
||||||
src/dialogs/dlg_forgot_password_reset.cpp
|
|
||||||
src/dialogs/dlg_load_deck.cpp
|
|
||||||
src/dialogs/dlg_load_deck_from_clipboard.cpp
|
|
||||||
src/dialogs/dlg_load_remote_deck.cpp
|
|
||||||
src/dialogs/dlg_manage_sets.cpp
|
|
||||||
src/dialogs/dlg_move_top_cards_until.cpp
|
|
||||||
src/dialogs/dlg_register.cpp
|
|
||||||
src/dialogs/dlg_roll_dice.cpp
|
|
||||||
src/dialogs/dlg_settings.cpp
|
|
||||||
src/dialogs/dlg_tip_of_the_day.cpp
|
|
||||||
src/dialogs/dlg_update.cpp
|
|
||||||
src/dialogs/dlg_view_log.cpp
|
|
||||||
src/game/board/abstract_card_drag_item.cpp
|
|
||||||
src/game/board/abstract_card_item.cpp
|
|
||||||
src/game/board/abstract_counter.cpp
|
|
||||||
src/game/board/abstract_graphics_item.cpp
|
|
||||||
src/game/board/arrow_item.cpp
|
|
||||||
src/game/board/arrow_target.cpp
|
|
||||||
src/game/board/card_drag_item.cpp
|
|
||||||
src/game/board/card_item.cpp
|
|
||||||
src/game/board/card_list.cpp
|
src/game/board/card_list.cpp
|
||||||
src/game/board/counter_general.cpp
|
src/game/board/card_state.cpp
|
||||||
src/game/cards/card_completer_proxy_model.cpp
|
src/game_graphics/board/counter_general.cpp
|
||||||
src/game/cards/card_database.cpp
|
src/game/board/counter_state.cpp
|
||||||
src/game/cards/card_database_manager.cpp
|
src/game_graphics/board/translate_counter_name.cpp
|
||||||
src/game/cards/card_database_model.cpp
|
src/game_graphics/deckview/deck_view.cpp
|
||||||
src/game/cards/card_database_parser/card_database_parser.cpp
|
src/game_graphics/deckview/deck_view_container.cpp
|
||||||
src/game/cards/card_database_parser/cockatrice_xml_3.cpp
|
src/game_graphics/deckview/tabbed_deck_view_container.cpp
|
||||||
src/game/cards/card_database_parser/cockatrice_xml_4.cpp
|
src/game_graphics/dialogs/dlg_create_token.cpp
|
||||||
src/game/cards/card_info.cpp
|
src/game_graphics/dialogs/dlg_move_top_cards_until.cpp
|
||||||
src/game/cards/card_search_model.cpp
|
src/game_graphics/dialogs/dlg_roll_dice.cpp
|
||||||
src/game/deckview/deck_view.cpp
|
src/game/game.cpp
|
||||||
src/game/deckview/deck_view_container.cpp
|
src/game/game_event_handler.cpp
|
||||||
src/game/filters/deck_filter_string.cpp
|
src/game/game_meta_info.cpp
|
||||||
src/game/filters/filter_builder.cpp
|
src/game_graphics/game_scene.cpp
|
||||||
src/game/filters/filter_card.cpp
|
src/game/game_state.cpp
|
||||||
src/game/filters/filter_string.cpp
|
src/game_graphics/game_view.cpp
|
||||||
src/game/filters/filter_tree.cpp
|
src/game_graphics/hand_counter.cpp
|
||||||
src/game/filters/filter_tree_model.cpp
|
src/game_graphics/log/message_log_widget.cpp
|
||||||
src/game/filters/syntax_help.cpp
|
|
||||||
src/game/game_scene.cpp
|
|
||||||
src/game/game_selector.cpp
|
|
||||||
src/game/game_view.cpp
|
|
||||||
src/game/games_model.cpp
|
|
||||||
src/game/hand_counter.cpp
|
|
||||||
src/game/phase.cpp
|
src/game/phase.cpp
|
||||||
src/game/player/player.cpp
|
src/game_graphics/phases_toolbar.cpp
|
||||||
src/game/player/player_list_widget.cpp
|
src/game_graphics/player/menu/card_menu.cpp
|
||||||
src/game/player/player_target.cpp
|
src/game_graphics/player/menu/custom_zone_menu.cpp
|
||||||
src/game/zones/card_zone.cpp
|
src/game_graphics/player/menu/grave_menu.cpp
|
||||||
src/game/zones/hand_zone.cpp
|
src/game_graphics/player/menu/hand_menu.cpp
|
||||||
src/game/zones/pile_zone.cpp
|
src/game_graphics/player/menu/library_menu.cpp
|
||||||
src/game/zones/select_zone.cpp
|
src/game_graphics/player/menu/move_menu.cpp
|
||||||
src/game/zones/stack_zone.cpp
|
src/game_graphics/player/menu/player_menu.cpp
|
||||||
src/game/zones/table_zone.cpp
|
src/game_graphics/player/menu/pt_menu.cpp
|
||||||
src/game/zones/view_zone.cpp
|
src/game_graphics/player/menu/rfg_menu.cpp
|
||||||
src/game/zones/view_zone_widget.cpp
|
src/game_graphics/player/menu/say_menu.cpp
|
||||||
|
src/game_graphics/player/menu/sideboard_menu.cpp
|
||||||
|
src/game_graphics/player/menu/utility_menu.cpp
|
||||||
|
src/game/player/player_actions.cpp
|
||||||
|
src/game_graphics/player/player_area.cpp
|
||||||
|
src/game_graphics/player/player_dialogs.cpp
|
||||||
|
src/game/player/player_event_handler.cpp
|
||||||
|
src/game_graphics/player/player_graphics_item.cpp
|
||||||
|
src/game/player/player_info.cpp
|
||||||
|
src/game_graphics/player/player_list_widget.cpp
|
||||||
|
src/game/player/player_logic.cpp
|
||||||
|
src/game/player/player_manager.cpp
|
||||||
|
src/game_graphics/player/player_target.cpp
|
||||||
|
src/game/replay.cpp
|
||||||
|
src/game/zones/card_zone_logic.cpp
|
||||||
|
src/game/zones/hand_zone_logic.cpp
|
||||||
|
src/game/zones/pile_zone_logic.cpp
|
||||||
|
src/game/zones/stack_zone_logic.cpp
|
||||||
|
src/game/zones/table_zone_logic.cpp
|
||||||
|
src/game/zones/view_zone_logic.cpp
|
||||||
|
src/game_graphics/zones/card_zone.cpp
|
||||||
|
src/game_graphics/zones/hand_zone.cpp
|
||||||
|
src/game_graphics/zones/pile_zone.cpp
|
||||||
|
src/game_graphics/zones/select_zone.cpp
|
||||||
|
src/game_graphics/zones/stack_zone.cpp
|
||||||
|
src/game_graphics/zones/table_zone.cpp
|
||||||
|
src/game_graphics/zones/view_zone.cpp
|
||||||
|
src/game_graphics/zones/view_zone_widget.cpp
|
||||||
|
src/game_graphics/board/abstract_graphics_item.cpp
|
||||||
|
src/interface/card_picture_loader/card_picture_loader.cpp
|
||||||
|
src/interface/card_picture_loader/card_picture_loader_local.cpp
|
||||||
|
src/interface/card_picture_loader/card_picture_loader_request_status_display_widget.cpp
|
||||||
|
src/interface/card_picture_loader/card_picture_loader_status_bar.cpp
|
||||||
|
src/interface/card_picture_loader/card_picture_loader_worker.cpp
|
||||||
|
src/interface/card_picture_loader/card_picture_loader_worker_work.cpp
|
||||||
|
src/interface/card_picture_loader/card_picture_to_load.cpp
|
||||||
|
src/interface/layouts/flow_layout.cpp
|
||||||
|
src/interface/layouts/overlap_layout.cpp
|
||||||
|
src/interface/widgets/utility/line_edit_completer.cpp
|
||||||
|
src/interface/pixel_map_generator.cpp
|
||||||
|
src/interface/theme_config.cpp
|
||||||
|
src/interface/theme_manager.cpp
|
||||||
|
src/interface/palette_editor/color_button.cpp
|
||||||
|
src/interface/palette_editor/palette_generator.cpp
|
||||||
|
src/interface/palette_editor/quick_setup_panel.cpp
|
||||||
|
src/interface/palette_editor/palette_grid_widget.cpp
|
||||||
|
src/interface/palette_editor/palette_editor_dialog.cpp
|
||||||
|
src/interface/widgets/cards/additional_info/color_identity_widget.cpp
|
||||||
|
src/interface/widgets/cards/additional_info/mana_cost_widget.cpp
|
||||||
|
src/interface/widgets/cards/additional_info/mana_symbol_widget.cpp
|
||||||
|
src/interface/widgets/cards/card_group_display_widgets/card_group_display_widget.cpp
|
||||||
|
src/interface/widgets/cards/card_group_display_widgets/flat_card_group_display_widget.cpp
|
||||||
|
src/interface/widgets/cards/card_group_display_widgets/overlapped_card_group_display_widget.cpp
|
||||||
|
src/interface/widgets/cards/card_info_display_widget.cpp
|
||||||
|
src/interface/widgets/cards/card_info_frame_widget.cpp
|
||||||
|
src/interface/widgets/cards/card_info_picture_art_crop_widget.cpp
|
||||||
|
src/interface/widgets/cards/card_info_picture_enlarged_widget.cpp
|
||||||
|
src/interface/widgets/cards/card_info_picture_widget.cpp
|
||||||
|
src/interface/widgets/cards/card_info_picture_with_text_overlay_widget.cpp
|
||||||
|
src/interface/widgets/cards/card_info_text_widget.cpp
|
||||||
|
src/interface/widgets/cards/card_size_widget.cpp
|
||||||
|
src/interface/widgets/cards/deck_card_zone_display_widget.cpp
|
||||||
|
src/interface/widgets/cards/deck_preview_card_picture_widget.cpp
|
||||||
|
src/interface/widgets/deck_analytics/abstract_analytics_panel_widget.cpp
|
||||||
|
src/interface/widgets/deck_analytics/add_analytics_panel_dialog.cpp
|
||||||
|
src/interface/widgets/deck_analytics/analytics_panel_widget_factory.cpp
|
||||||
|
src/interface/widgets/deck_analytics/analytics_panel_widget_registrar.cpp
|
||||||
|
src/interface/widgets/deck_analytics/deck_analytics_widget.cpp
|
||||||
|
src/interface/widgets/deck_analytics/deck_list_statistics_analyzer.cpp
|
||||||
|
src/interface/widgets/deck_analytics/resizable_panel.cpp
|
||||||
|
src/interface/widgets/deck_analytics/analyzer_modules/draw_probability/draw_probability_config.cpp
|
||||||
|
src/interface/widgets/deck_analytics/analyzer_modules/draw_probability/draw_probability_config_dialog.cpp
|
||||||
|
src/interface/widgets/deck_analytics/analyzer_modules/draw_probability/draw_probability_widget.cpp
|
||||||
|
src/interface/widgets/deck_analytics/analyzer_modules/mana_base/mana_base_config.cpp
|
||||||
|
src/interface/widgets/deck_analytics/analyzer_modules/mana_base/mana_base_config_dialog.cpp
|
||||||
|
src/interface/widgets/deck_analytics/analyzer_modules/mana_base/mana_base_widget.cpp
|
||||||
|
src/interface/widgets/deck_analytics/analyzer_modules/mana_curve/mana_curve_config.cpp
|
||||||
|
src/interface/widgets/deck_analytics/analyzer_modules/mana_curve/mana_curve_config_dialog.cpp
|
||||||
|
src/interface/widgets/deck_analytics/analyzer_modules/mana_curve/mana_curve_widget.cpp
|
||||||
|
src/interface/widgets/deck_analytics/analyzer_modules/mana_devotion/mana_devotion_config.cpp
|
||||||
|
src/interface/widgets/deck_analytics/analyzer_modules/mana_devotion/mana_devotion_config_dialog.cpp
|
||||||
|
src/interface/widgets/deck_analytics/analyzer_modules/mana_devotion/mana_devotion_widget.cpp
|
||||||
|
src/interface/widgets/deck_analytics/analyzer_modules/mana_distribution/mana_distribution_config.cpp
|
||||||
|
src/interface/widgets/deck_analytics/analyzer_modules/mana_distribution/mana_distribution_config_dialog.cpp
|
||||||
|
src/interface/widgets/deck_analytics/analyzer_modules/mana_distribution/mana_distribution_widget.cpp
|
||||||
|
src/interface/widgets/deck_analytics/analyzer_modules/mana_distribution/mana_distribution_single_display_widget.cpp
|
||||||
|
src/interface/widgets/deck_analytics/analyzer_modules/mana_curve/mana_curve_total_widget.cpp
|
||||||
|
src/interface/widgets/deck_analytics/analyzer_modules/mana_curve/mana_curve_category_widget.cpp
|
||||||
|
src/interface/widgets/deck_editor/card_database_view.cpp
|
||||||
|
src/interface/widgets/deck_editor/deck_list_history_manager_widget.cpp
|
||||||
|
src/interface/widgets/deck_editor/deck_editor_card_database_dock_widget.cpp
|
||||||
|
src/interface/widgets/deck_editor/deck_editor_card_info_dock_widget.cpp
|
||||||
|
src/interface/widgets/deck_editor/deck_editor_database_display_widget.cpp
|
||||||
|
src/interface/widgets/deck_editor/deck_editor_deck_dock_widget.cpp
|
||||||
|
src/interface/widgets/deck_editor/deck_editor_filter_dock_widget.cpp
|
||||||
|
src/interface/widgets/deck_editor/deck_editor_printing_selector_dock_widget.cpp
|
||||||
|
src/interface/widgets/deck_editor/deck_list_style_proxy.cpp
|
||||||
|
src/interface/widgets/deck_editor/deck_state_manager.cpp
|
||||||
|
src/interface/widgets/deck_editor/printing_disabled_info_widget.cpp
|
||||||
|
src/interface/widgets/general/background_sources.cpp
|
||||||
|
src/interface/widgets/general/display/background_plate_widget.cpp
|
||||||
|
src/interface/widgets/general/display/banner_widget.cpp
|
||||||
|
src/interface/widgets/general/display/dynamic_font_size_label.cpp
|
||||||
|
src/interface/widgets/general/display/dynamic_font_size_push_button.cpp
|
||||||
|
src/interface/widgets/general/display/labeled_input.cpp
|
||||||
|
src/interface/widgets/general/display/shadow_background_label.cpp
|
||||||
|
src/interface/widgets/general/display/charts/bars/bar_widget.cpp
|
||||||
|
src/interface/widgets/general/display/charts/bars/color_bar.cpp
|
||||||
|
src/interface/widgets/general/display/charts/bars/percent_bar_widget.cpp
|
||||||
|
src/interface/widgets/general/display/charts/bars/bar_chart_widget.cpp
|
||||||
|
src/interface/widgets/general/display/charts/bars/bar_chart_background_widget.cpp
|
||||||
|
src/interface/widgets/general/display/charts/bars/segmented_bar_widget.cpp
|
||||||
|
src/interface/widgets/general/display/charts/pies/color_pie.cpp
|
||||||
|
src/interface/widgets/general/home_styled_button.cpp
|
||||||
|
src/interface/widgets/general/home_widget.cpp
|
||||||
|
src/interface/widgets/general/layout_containers/flow_widget.cpp
|
||||||
|
src/interface/widgets/general/layout_containers/overlap_control_widget.cpp
|
||||||
|
src/interface/widgets/general/layout_containers/overlap_widget.cpp
|
||||||
|
src/interface/widgets/menus/deck_editor_menu.cpp
|
||||||
|
src/interface/widgets/printing_selector/all_zones_card_amount_widget.cpp
|
||||||
|
src/interface/widgets/printing_selector/card_amount_widget.cpp
|
||||||
|
src/interface/widgets/printing_selector/printing_selector.cpp
|
||||||
|
src/interface/widgets/printing_selector/printing_selector_card_display_widget.cpp
|
||||||
|
src/interface/widgets/printing_selector/printing_selector_card_overlay_widget.cpp
|
||||||
|
src/interface/widgets/printing_selector/printing_selector_placeholder_widget.cpp
|
||||||
|
src/interface/widgets/printing_selector/printing_selector_card_search_widget.cpp
|
||||||
|
src/interface/widgets/printing_selector/printing_selector_card_selection_widget.cpp
|
||||||
|
src/interface/widgets/printing_selector/printing_selector_card_sorting_widget.cpp
|
||||||
|
src/interface/widgets/printing_selector/set_name_and_collectors_number_display_widget.cpp
|
||||||
|
src/interface/widgets/quick_settings/settings_button_widget.cpp
|
||||||
|
src/interface/widgets/quick_settings/settings_popup_widget.cpp
|
||||||
|
src/interface/widgets/replay/replay_manager.cpp
|
||||||
|
src/interface/widgets/replay/replay_timeline_widget.cpp
|
||||||
|
src/interface/widgets/server/chat_view/chat_view.cpp
|
||||||
|
src/interface/widgets/server/game_filter_configs.cpp
|
||||||
|
src/interface/widgets/server/game_selector.cpp
|
||||||
|
src/interface/widgets/server/game_selector_quick_filter_toolbar.cpp
|
||||||
|
src/interface/widgets/server/games_model.cpp
|
||||||
|
src/interface/widgets/server/handle_public_servers.cpp
|
||||||
|
src/interface/widgets/server/remote/remote_decklist_tree_widget.cpp
|
||||||
|
src/interface/widgets/server/remote/remote_replay_list_tree_widget.cpp
|
||||||
|
src/interface/widgets/server/user/user_context_menu.cpp
|
||||||
|
src/interface/widgets/server/user/user_info_box.cpp
|
||||||
|
src/interface/widgets/server/user/user_info_connection.cpp
|
||||||
|
src/interface/widgets/server/user/user_list_manager.cpp
|
||||||
|
src/interface/widgets/server/user/user_list_widget.cpp
|
||||||
|
src/interface/widgets/settings_page/appearance_settings_page.cpp
|
||||||
|
src/interface/widgets/settings_page/deck_editor_settings_page.cpp
|
||||||
|
src/interface/widgets/settings_page/general_settings_page.cpp
|
||||||
|
src/interface/widgets/settings_page/messages_settings_page.cpp
|
||||||
|
src/interface/widgets/settings_page/shortcut_settings_page.cpp
|
||||||
|
src/interface/widgets/settings_page/sound_settings_page.cpp
|
||||||
|
src/interface/widgets/settings_page/storage_settings_page.cpp
|
||||||
|
src/interface/widgets/settings_page/user_interface_settings_page.cpp
|
||||||
|
src/interface/widgets/utility/custom_line_edit.cpp
|
||||||
|
src/interface/widgets/utility/get_text_with_max.cpp
|
||||||
|
src/interface/widgets/utility/sequence_edit.cpp
|
||||||
|
src/interface/widgets/utility/visibility_change_listener.cpp
|
||||||
|
src/interface/widgets/utility/visibility_change_listener.h
|
||||||
|
src/interface/widgets/visual_database_display/visual_database_display_color_filter_widget.cpp
|
||||||
|
src/interface/widgets/visual_database_display/visual_database_display_filter_save_load_widget.cpp
|
||||||
|
src/interface/widgets/visual_database_display/visual_database_display_filter_toolbar_widget.cpp
|
||||||
|
src/interface/widgets/visual_database_display/visual_database_display_format_legality_filter_widget.cpp
|
||||||
|
src/interface/widgets/visual_database_display/visual_database_display_main_type_filter_widget.cpp
|
||||||
|
src/interface/widgets/visual_database_display/visual_database_display_name_filter_widget.cpp
|
||||||
|
src/interface/widgets/visual_database_display/visual_database_display_set_filter_widget.cpp
|
||||||
|
src/interface/widgets/visual_database_display/visual_database_display_sub_type_filter_widget.cpp
|
||||||
|
src/interface/widgets/visual_database_display/visual_database_display_widget.cpp
|
||||||
|
src/interface/widgets/visual_database_display/visual_database_filter_display_widget.cpp
|
||||||
|
src/interface/widgets/visual_deck_editor/visual_deck_display_options_widget.cpp
|
||||||
|
src/interface/widgets/visual_deck_editor/visual_deck_editor_placeholder_widget.cpp
|
||||||
|
src/interface/widgets/visual_deck_editor/visual_deck_editor_sample_hand_widget.cpp
|
||||||
|
src/interface/widgets/visual_deck_editor/visual_deck_editor_widget.cpp
|
||||||
|
src/interface/widgets/visual_deck_storage/deck_preview/deck_preview_color_identity_filter_widget.cpp
|
||||||
|
src/interface/widgets/visual_deck_storage/deck_preview/deck_preview_deck_tags_display_widget.cpp
|
||||||
|
src/interface/widgets/visual_deck_storage/deck_preview/deck_preview_tag_addition_widget.cpp
|
||||||
|
src/interface/widgets/visual_deck_storage/deck_preview/deck_preview_tag_dialog.cpp
|
||||||
|
src/interface/widgets/visual_deck_storage/deck_preview/deck_preview_tag_display_widget.cpp
|
||||||
|
src/interface/widgets/visual_deck_storage/deck_preview/deck_preview_tag_item_widget.cpp
|
||||||
|
src/interface/widgets/visual_deck_storage/deck_preview/deck_preview_widget.cpp
|
||||||
|
src/interface/widgets/visual_deck_storage/visual_deck_storage_folder_display_widget.cpp
|
||||||
|
src/interface/widgets/visual_deck_storage/visual_deck_storage_quick_settings_widget.cpp
|
||||||
|
src/interface/widgets/visual_deck_storage/visual_deck_storage_search_widget.cpp
|
||||||
|
src/interface/widgets/visual_deck_storage/visual_deck_storage_sort_widget.cpp
|
||||||
|
src/interface/widgets/visual_deck_storage/visual_deck_storage_tag_filter_widget.cpp
|
||||||
|
src/interface/widgets/visual_deck_storage/visual_deck_storage_widget.cpp
|
||||||
|
src/interface/window_main.cpp
|
||||||
src/main.cpp
|
src/main.cpp
|
||||||
src/server/chat_view/chat_view.cpp
|
src/interface/widgets/tabs/abstract_tab_deck_editor.cpp
|
||||||
src/server/handle_public_servers.cpp
|
src/interface/widgets/tabs/api/archidekt/tab_archidekt.cpp
|
||||||
src/server/local_client.cpp
|
src/interface/widgets/tabs/api/archidekt/api_response/archidekt_deck_listing_api_response.cpp
|
||||||
src/server/local_server.cpp
|
src/interface/widgets/tabs/api/archidekt/api_response/archidekt_formats.h
|
||||||
src/server/local_server_interface.cpp
|
src/interface/widgets/tabs/api/archidekt/api_response/card/archidekt_api_response_card.cpp
|
||||||
src/server/message_log_widget.cpp
|
src/interface/widgets/tabs/api/archidekt/api_response/card/archidekt_api_response_card_entry.cpp
|
||||||
src/server/pending_command.cpp
|
src/interface/widgets/tabs/api/archidekt/api_response/card/archidekt_api_response_edition.cpp
|
||||||
src/server/remote/remote_client.cpp
|
src/interface/widgets/tabs/api/archidekt/api_response/deck/archidekt_api_response_deck.cpp
|
||||||
src/server/remote/remote_decklist_tree_widget.cpp
|
src/interface/widgets/tabs/api/archidekt/api_response/deck/archidekt_api_response_deck_category.cpp
|
||||||
src/server/remote/remote_replay_list_tree_widget.cpp
|
src/interface/widgets/tabs/api/archidekt/api_response/deck_listings/archidekt_api_response_deck_listing_container.cpp
|
||||||
src/server/user/user_context_menu.cpp
|
src/interface/widgets/tabs/api/archidekt/api_response/deck_listings/archidekt_api_response_deck_owner.cpp
|
||||||
src/server/user/user_info_box.cpp
|
src/interface/widgets/tabs/api/archidekt/display/archidekt_api_response_deck_display_widget.cpp
|
||||||
src/server/user/user_info_connection.cpp
|
src/interface/widgets/tabs/api/archidekt/display/archidekt_api_response_deck_entry_display_widget.cpp
|
||||||
src/server/user/user_list_manager.cpp
|
src/interface/widgets/tabs/api/archidekt/display/archidekt_api_response_deck_listings_display_widget.cpp
|
||||||
src/server/user/user_list_widget.cpp
|
src/interface/widgets/tabs/api/archidekt/display/archidekt_deck_preview_image_display_widget.cpp
|
||||||
src/settings/cache_settings.cpp
|
src/interface/widgets/tabs/api/edhrec/api_response/archidekt_links/edhrec_api_response_archidekt_links.cpp
|
||||||
src/settings/card_database_settings.cpp
|
src/interface/widgets/tabs/api/edhrec/api_response/average_deck/edhrec_average_deck_api_response.cpp
|
||||||
src/settings/card_override_settings.cpp
|
src/interface/widgets/tabs/api/edhrec/api_response/average_deck/edhrec_deck_api_response.cpp
|
||||||
src/settings/debug_settings.cpp
|
src/interface/widgets/tabs/api/edhrec/api_response/card_prices/edhrec_api_response_card_prices.cpp
|
||||||
src/settings/download_settings.cpp
|
src/interface/widgets/tabs/api/edhrec/api_response/cards/edhrec_api_response_card_container.cpp
|
||||||
src/settings/game_filters_settings.cpp
|
src/interface/widgets/tabs/api/edhrec/api_response/cards/edhrec_api_response_card_details.cpp
|
||||||
src/settings/layouts_settings.cpp
|
src/interface/widgets/tabs/api/edhrec/api_response/cards/edhrec_api_response_card_list.cpp
|
||||||
src/settings/message_settings.cpp
|
src/interface/widgets/tabs/api/edhrec/api_response/cards/edhrec_commander_api_response_commander_details.cpp
|
||||||
src/settings/recents_settings.cpp
|
src/interface/widgets/tabs/api/edhrec/api_response/commander/edhrec_commander_api_response.cpp
|
||||||
src/settings/servers_settings.cpp
|
src/interface/widgets/tabs/api/edhrec/api_response/commander/edhrec_commander_api_response_average_deck_statistics.cpp
|
||||||
src/settings/settings_manager.cpp
|
src/interface/widgets/tabs/api/edhrec/api_response/top_cards/edhrec_top_cards_api_response.cpp
|
||||||
src/settings/shortcut_treeview.cpp
|
src/interface/widgets/tabs/api/edhrec/api_response/top_commanders/edhrec_top_commanders_api_response.cpp
|
||||||
src/settings/shortcuts_settings.cpp
|
src/interface/widgets/tabs/api/edhrec/api_response/top_tags/edhrec_top_tags_api_response.cpp
|
||||||
src/utility/card_info_comparator.cpp
|
src/interface/widgets/tabs/api/edhrec/display/card_prices/edhrec_api_response_card_prices_display_widget.cpp
|
||||||
src/utility/levenshtein.cpp
|
src/interface/widgets/tabs/api/edhrec/display/cards/edhrec_api_response_card_details_display_widget.cpp
|
||||||
src/utility/logger.cpp
|
src/interface/widgets/tabs/api/edhrec/display/cards/edhrec_api_response_card_inclusion_display_widget.cpp
|
||||||
src/utility/sequence_edit.cpp
|
src/interface/widgets/tabs/api/edhrec/display/cards/edhrec_api_response_card_list_display_widget.cpp
|
||||||
|
src/interface/widgets/tabs/api/edhrec/display/cards/edhrec_api_response_card_synergy_display_widget.cpp
|
||||||
|
src/interface/widgets/tabs/api/edhrec/display/commander/edhrec_api_response_commander_details_display_widget.cpp
|
||||||
|
src/interface/widgets/tabs/api/edhrec/display/commander/edhrec_commander_api_response_display_widget.cpp
|
||||||
|
src/interface/widgets/tabs/api/edhrec/display/commander/edhrec_commander_api_response_navigation_widget.cpp
|
||||||
|
src/interface/widgets/tabs/api/edhrec/display/top_cards/edhrec_top_cards_api_response_display_widget.cpp
|
||||||
|
src/interface/widgets/tabs/api/edhrec/display/top_commander/edhrec_top_commanders_api_response_display_widget.cpp
|
||||||
|
src/interface/widgets/tabs/api/edhrec/display/top_tags/edhrec_top_tags_api_response_display_widget.cpp
|
||||||
|
src/interface/widgets/tabs/api/edhrec/tab_edhrec.cpp
|
||||||
|
src/interface/widgets/tabs/api/edhrec/tab_edhrec_main.cpp
|
||||||
|
src/interface/widgets/tabs/tab.cpp
|
||||||
|
src/interface/widgets/tabs/tab_account.cpp
|
||||||
|
src/interface/widgets/tabs/tab_admin.cpp
|
||||||
|
src/interface/widgets/tabs/tab_deck_editor.cpp
|
||||||
|
src/interface/widgets/tabs/tab_deck_storage.cpp
|
||||||
|
src/interface/widgets/tabs/tab_game.cpp
|
||||||
|
src/interface/widgets/tabs/tab_home.cpp
|
||||||
|
src/interface/widgets/tabs/tab_logs.cpp
|
||||||
|
src/interface/widgets/tabs/tab_message.cpp
|
||||||
|
src/interface/widgets/tabs/tab_replays.cpp
|
||||||
|
src/interface/widgets/tabs/tab_room.cpp
|
||||||
|
src/interface/widgets/tabs/tab_server.cpp
|
||||||
|
src/interface/widgets/tabs/tab_supervisor.cpp
|
||||||
|
src/interface/widgets/tabs/tab_visual_database_display.cpp
|
||||||
|
src/interface/widgets/tabs/visual_deck_editor/tab_deck_editor_visual.cpp
|
||||||
|
src/interface/widgets/tabs/visual_deck_editor/tab_deck_editor_visual_tab_widget.cpp
|
||||||
|
src/interface/widgets/tabs/visual_deck_storage/tab_deck_storage_visual.cpp
|
||||||
|
src/interface/key_signals.cpp
|
||||||
|
src/interface/logger.cpp
|
||||||
|
src/interface/widgets/tabs/api/edhrec/display/commander/edhrec_commander_api_response_bracket_navigation_widget.cpp
|
||||||
|
src/interface/widgets/tabs/api/edhrec/display/commander/edhrec_commander_api_response_bracket_navigation_widget.h
|
||||||
|
src/interface/widgets/tabs/api/edhrec/display/commander/edhrec_commander_api_response_budget_navigation_widget.cpp
|
||||||
|
src/interface/widgets/tabs/api/edhrec/display/commander/edhrec_commander_api_response_budget_navigation_widget.h
|
||||||
|
src/interface/widgets/utility/compact_push_button.cpp
|
||||||
|
src/interface/widgets/utility/compact_push_button.h
|
||||||
)
|
)
|
||||||
|
|
||||||
add_subdirectory(sounds)
|
add_subdirectory(sounds)
|
||||||
|
|
@ -258,15 +360,23 @@ configure_file(
|
||||||
set(cockatrice_RESOURCES cockatrice.qrc)
|
set(cockatrice_RESOURCES cockatrice.qrc)
|
||||||
|
|
||||||
if(UPDATE_TRANSLATIONS)
|
if(UPDATE_TRANSLATIONS)
|
||||||
|
# Cockatrice main sources
|
||||||
file(GLOB_RECURSE translate_cockatrice_SRCS ${CMAKE_SOURCE_DIR}/cockatrice/src/*.cpp
|
file(GLOB_RECURSE translate_cockatrice_SRCS ${CMAKE_SOURCE_DIR}/cockatrice/src/*.cpp
|
||||||
${CMAKE_SOURCE_DIR}/cockatrice/src/*.h
|
${CMAKE_SOURCE_DIR}/cockatrice/src/*.h
|
||||||
)
|
)
|
||||||
file(GLOB_RECURSE translate_common_SRCS ${CMAKE_SOURCE_DIR}/common/*.cpp ${CMAKE_SOURCE_DIR}/common/*.h)
|
|
||||||
set(translate_SRCS ${translate_cockatrice_SRCS} ${translate_common_SRCS})
|
# All libcockatrice_* libraries (recursively)
|
||||||
|
file(GLOB_RECURSE translate_lib_SRCS ${CMAKE_SOURCE_DIR}/libcockatrice_*/**/*.cpp
|
||||||
|
${CMAKE_SOURCE_DIR}/libcockatrice_*/**/*.h
|
||||||
|
)
|
||||||
|
|
||||||
|
# Combine all sources for translation
|
||||||
|
set(translate_SRCS ${translate_cockatrice_SRCS} ${translate_lib_SRCS})
|
||||||
|
|
||||||
set(cockatrice_TS "${CMAKE_CURRENT_SOURCE_DIR}/cockatrice_en@source.ts")
|
set(cockatrice_TS "${CMAKE_CURRENT_SOURCE_DIR}/cockatrice_en@source.ts")
|
||||||
else()
|
else()
|
||||||
file(GLOB cockatrice_TS "${CMAKE_CURRENT_SOURCE_DIR}/translations/*.ts")
|
file(GLOB cockatrice_TS "${CMAKE_CURRENT_SOURCE_DIR}/translations/*.ts")
|
||||||
endif(UPDATE_TRANSLATIONS)
|
endif()
|
||||||
|
|
||||||
if(WIN32)
|
if(WIN32)
|
||||||
set(cockatrice_SOURCES ${cockatrice_SOURCES} cockatrice.rc)
|
set(cockatrice_SOURCES ${cockatrice_SOURCES} cockatrice.rc)
|
||||||
|
|
@ -296,12 +406,6 @@ set(DESKTOPDIR
|
||||||
CACHE STRING "desktop file destination"
|
CACHE STRING "desktop file destination"
|
||||||
)
|
)
|
||||||
|
|
||||||
# Include directories
|
|
||||||
include_directories(../common)
|
|
||||||
include_directories(${PROTOBUF_INCLUDE_DIR})
|
|
||||||
include_directories(${CMAKE_BINARY_DIR}/common)
|
|
||||||
include_directories(${CMAKE_CURRENT_BINARY_DIR})
|
|
||||||
|
|
||||||
set(COCKATRICE_MAC_QM_INSTALL_DIR "cockatrice.app/Contents/Resources/translations")
|
set(COCKATRICE_MAC_QM_INSTALL_DIR "cockatrice.app/Contents/Resources/translations")
|
||||||
set(COCKATRICE_UNIX_QM_INSTALL_DIR "share/cockatrice/translations")
|
set(COCKATRICE_UNIX_QM_INSTALL_DIR "share/cockatrice/translations")
|
||||||
set(COCKATRICE_WIN32_QM_INSTALL_DIR "translations")
|
set(COCKATRICE_WIN32_QM_INSTALL_DIR "translations")
|
||||||
|
|
@ -341,9 +445,31 @@ elseif(Qt5_FOUND)
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
if(Qt5_FOUND)
|
if(Qt5_FOUND)
|
||||||
target_link_libraries(cockatrice cockatrice_common ${COCKATRICE_QT_MODULES})
|
target_link_libraries(
|
||||||
|
cockatrice
|
||||||
|
libcockatrice_card
|
||||||
|
libcockatrice_deck_list
|
||||||
|
libcockatrice_filters
|
||||||
|
libcockatrice_utility
|
||||||
|
libcockatrice_network
|
||||||
|
libcockatrice_models
|
||||||
|
libcockatrice_rng
|
||||||
|
libcockatrice_settings
|
||||||
|
${COCKATRICE_QT_MODULES}
|
||||||
|
)
|
||||||
else()
|
else()
|
||||||
target_link_libraries(cockatrice PUBLIC cockatrice_common ${COCKATRICE_QT_MODULES})
|
target_link_libraries(
|
||||||
|
cockatrice
|
||||||
|
PUBLIC libcockatrice_card
|
||||||
|
libcockatrice_deck_list
|
||||||
|
libcockatrice_filters
|
||||||
|
libcockatrice_utility
|
||||||
|
libcockatrice_network
|
||||||
|
libcockatrice_models
|
||||||
|
libcockatrice_rng
|
||||||
|
libcockatrice_settings
|
||||||
|
${COCKATRICE_QT_MODULES}
|
||||||
|
)
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
if(UNIX)
|
if(UNIX)
|
||||||
|
|
@ -437,6 +563,7 @@ if(WIN32)
|
||||||
DIRECTORY "${CMAKE_BINARY_DIR}/cockatrice/"
|
DIRECTORY "${CMAKE_BINARY_DIR}/cockatrice/"
|
||||||
DESTINATION ./
|
DESTINATION ./
|
||||||
FILES_MATCHING
|
FILES_MATCHING
|
||||||
|
PATTERN "CMakeFiles" EXCLUDE
|
||||||
PATTERN "*.ini"
|
PATTERN "*.ini"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
@ -462,6 +589,9 @@ if(WIN32)
|
||||||
PATTERN "styles/qopensslbackend.dll"
|
PATTERN "styles/qopensslbackend.dll"
|
||||||
PATTERN "styles/qschannelbackend.dll"
|
PATTERN "styles/qschannelbackend.dll"
|
||||||
PATTERN "styles/qwindowsvistastyle.dll"
|
PATTERN "styles/qwindowsvistastyle.dll"
|
||||||
|
PATTERN "styles/qwindows11style.dll"
|
||||||
|
PATTERN "styles/qmodernwindowsstyle.dll"
|
||||||
|
PATTERN "styles/qmodernwindowsstyled.dll"
|
||||||
PATTERN "tls/qcertonlybackend.dll"
|
PATTERN "tls/qcertonlybackend.dll"
|
||||||
PATTERN "tls/qopensslbackend.dll"
|
PATTERN "tls/qopensslbackend.dll"
|
||||||
PATTERN "tls/qschannelbackend.dll"
|
PATTERN "tls/qschannelbackend.dll"
|
||||||
|
|
|
||||||
|
|
@ -7,32 +7,46 @@
|
||||||
|
|
||||||
<file>resources/icons/arrow_bottom_green.svg</file>
|
<file>resources/icons/arrow_bottom_green.svg</file>
|
||||||
<file>resources/icons/arrow_down_green.svg</file>
|
<file>resources/icons/arrow_down_green.svg</file>
|
||||||
|
<file>resources/icons/arrow_history.svg</file>
|
||||||
<file>resources/icons/arrow_left_green.svg</file>
|
<file>resources/icons/arrow_left_green.svg</file>
|
||||||
|
<file>resources/icons/arrow_redo.svg</file>
|
||||||
<file>resources/icons/arrow_right_blue.svg</file>
|
<file>resources/icons/arrow_right_blue.svg</file>
|
||||||
<file>resources/icons/arrow_right_green.svg</file>
|
<file>resources/icons/arrow_right_green.svg</file>
|
||||||
<file>resources/icons/arrow_top_green.svg</file>
|
<file>resources/icons/arrow_top_green.svg</file>
|
||||||
<file>resources/icons/arrow_up_green.svg</file>
|
<file>resources/icons/arrow_up_green.svg</file>
|
||||||
|
<file>resources/icons/arrow_undo.svg</file>
|
||||||
|
<file>resources/icons/circle_half_stroke.svg</file>
|
||||||
<file>resources/icons/clearsearch.svg</file>
|
<file>resources/icons/clearsearch.svg</file>
|
||||||
<file>resources/icons/cogwheel.svg</file>
|
<file>resources/icons/cogwheel.svg</file>
|
||||||
<file>resources/icons/conceded.svg</file>
|
<file>resources/icons/conceded.svg</file>
|
||||||
<file>resources/icons/decrement.svg</file>
|
<file>resources/icons/decrement.svg</file>
|
||||||
<file>resources/icons/delete.svg</file>
|
<file>resources/icons/delete.svg</file>
|
||||||
|
<file>resources/icons/dragon.svg</file>
|
||||||
<file>resources/icons/dropdown_collapsed.svg</file>
|
<file>resources/icons/dropdown_collapsed.svg</file>
|
||||||
<file>resources/icons/dropdown_expanded.svg</file>
|
<file>resources/icons/dropdown_expanded.svg</file>
|
||||||
|
<file>resources/icons/filter.svg</file>
|
||||||
|
<file>resources/icons/floppy_disk.svg</file>
|
||||||
<file>resources/icons/forgot_password.svg</file>
|
<file>resources/icons/forgot_password.svg</file>
|
||||||
|
<file>resources/icons/gear.svg</file>
|
||||||
<file>resources/icons/increment.svg</file>
|
<file>resources/icons/increment.svg</file>
|
||||||
<file>resources/icons/info.svg</file>
|
<file>resources/icons/info.svg</file>
|
||||||
<file>resources/icons/lock.svg</file>
|
<file>resources/icons/lock.svg</file>
|
||||||
<file>resources/icons/not_ready_start.svg</file>
|
<file>resources/icons/not_ready_start.svg</file>
|
||||||
|
<file>resources/icons/pen_to_square.svg</file>
|
||||||
<file>resources/icons/pencil.svg</file>
|
<file>resources/icons/pencil.svg</file>
|
||||||
|
<file>resources/icons/pin.svg</file>
|
||||||
<file>resources/icons/player.svg</file>
|
<file>resources/icons/player.svg</file>
|
||||||
<file>resources/icons/ready_start.svg</file>
|
<file>resources/icons/ready_start.svg</file>
|
||||||
<file>resources/icons/reload.svg</file>
|
<file>resources/icons/reload.svg</file>
|
||||||
<file>resources/icons/remove_row.svg</file>
|
<file>resources/icons/remove_row.svg</file>
|
||||||
<file>resources/icons/rename.svg</file>
|
<file>resources/icons/rename.svg</file>
|
||||||
|
<file>resources/icons/scale_balanced.svg</file>
|
||||||
<file>resources/icons/scales.svg</file>
|
<file>resources/icons/scales.svg</file>
|
||||||
|
<file>resources/icons/scroll.svg</file>
|
||||||
<file>resources/icons/search.svg</file>
|
<file>resources/icons/search.svg</file>
|
||||||
<file>resources/icons/settings.svg</file>
|
<file>resources/icons/settings.svg</file>
|
||||||
|
<file>resources/icons/share.svg</file>
|
||||||
|
<file>resources/icons/sort_arrow_down.svg</file>
|
||||||
<file>resources/icons/spectator.svg</file>
|
<file>resources/icons/spectator.svg</file>
|
||||||
<file>resources/icons/swap.svg</file>
|
<file>resources/icons/swap.svg</file>
|
||||||
<file>resources/icons/sync.svg</file>
|
<file>resources/icons/sync.svg</file>
|
||||||
|
|
@ -41,16 +55,22 @@
|
||||||
<file>resources/icons/view.svg</file>
|
<file>resources/icons/view.svg</file>
|
||||||
|
|
||||||
<file>resources/icons/mana/B.svg</file>
|
<file>resources/icons/mana/B.svg</file>
|
||||||
|
<file>resources/icons/mana/C.svg</file>
|
||||||
<file>resources/icons/mana/G.svg</file>
|
<file>resources/icons/mana/G.svg</file>
|
||||||
<file>resources/icons/mana/R.svg</file>
|
<file>resources/icons/mana/R.svg</file>
|
||||||
<file>resources/icons/mana/U.svg</file>
|
<file>resources/icons/mana/U.svg</file>
|
||||||
<file>resources/icons/mana/W.svg</file>
|
<file>resources/icons/mana/W.svg</file>
|
||||||
|
|
||||||
|
<file>resources/backgrounds/home.png</file>
|
||||||
|
<file>resources/backgrounds/card_triplet.svg</file>
|
||||||
|
<file>resources/backgrounds/placeholder_printing_selector.svg</file>
|
||||||
|
|
||||||
<file>resources/config/general.svg</file>
|
<file>resources/config/general.svg</file>
|
||||||
<file>resources/config/appearance.svg</file>
|
<file>resources/config/appearance.svg</file>
|
||||||
<file>resources/config/interface.svg</file>
|
<file>resources/config/interface.svg</file>
|
||||||
<file>resources/config/messages.svg</file>
|
<file>resources/config/messages.svg</file>
|
||||||
<file>resources/config/deckeditor.svg</file>
|
<file>resources/config/deckeditor.svg</file>
|
||||||
|
<file>resources/config/storage.svg</file>
|
||||||
<file>resources/config/shorcuts.svg</file>
|
<file>resources/config/shorcuts.svg</file>
|
||||||
<file>resources/config/sound.svg</file>
|
<file>resources/config/sound.svg</file>
|
||||||
<file>resources/config/debug.ini</file>
|
<file>resources/config/debug.ini</file>
|
||||||
|
|
|
||||||
31
cockatrice/resources/backgrounds/card_triplet.svg
Normal file
|
|
@ -0,0 +1,31 @@
|
||||||
|
<?xml version="1.0" encoding="UTF-8" standalone="no"?>
|
||||||
|
<svg
|
||||||
|
version="1.1"
|
||||||
|
width="250"
|
||||||
|
id="svg13"
|
||||||
|
height="231.66667"
|
||||||
|
xmlns:svg="http://www.w3.org/2000/svg"
|
||||||
|
xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<defs
|
||||||
|
id="defs13" />
|
||||||
|
<g
|
||||||
|
transform="matrix(1.705559,0,0,1.705559,-18.310328,-4.2419088)"
|
||||||
|
id="g13">
|
||||||
|
<path
|
||||||
|
d="M 90.069854,3.479957 C 89.356513,1.2235709 86.980392,-0.01102897 84.723451,0.70218215 L 3.4767601,26.377781 C 1.2199188,27.090982 -0.01486587,29.46663 0.69839437,31.723116 L 33.512365,135.52112 c 0.713341,2.25639 3.089462,3.49099 5.346403,2.77777 l 81.246672,-25.6756 c 2.25684,-0.71319 3.49163,-3.08884 2.77837,-5.34533 L 90.074852,3.479957 Z"
|
||||||
|
style="display:none;fill:#c0c0c0;fill-opacity:1;stroke:#989898;stroke-width:1;stroke-opacity:1"
|
||||||
|
id="path1" />
|
||||||
|
<path
|
||||||
|
d="m 110.61293,7.4983294 c -0.36657,-2.337853 -2.53055,-3.9150142 -4.86886,-3.5484627 L 21.563382,17.14452 c -2.338314,0.366502 -3.915784,2.529976 -3.549207,4.867929 L 34.876507,129.55893 c 0.366577,2.33786 2.530549,3.91502 4.868863,3.54847 l 84.18069,-13.19466 c 2.33831,-0.3665 3.91578,-2.52997 3.5492,-4.86793 L 110.61093,7.4983294 Z"
|
||||||
|
style="fill:#c0c0c0;fill-opacity:1;stroke:#989898;stroke-width:1;stroke-opacity:1"
|
||||||
|
id="path4" />
|
||||||
|
<path
|
||||||
|
d="m 130.53623,15.555064 c 0,-2.366441 -1.89356,-4.259575 -4.26046,-4.259575 H 41.067426 c -2.366905,0 -4.260468,1.893134 -4.260468,4.259575 V 124.41102 c 0,2.36644 1.893563,4.25957 4.260468,4.25957 h 85.208344 c 2.3669,0 4.26046,-1.89313 4.26046,-4.25957 z"
|
||||||
|
style="fill:#c0c0c0;fill-opacity:1;stroke:#989898;stroke-width:1;stroke-opacity:1"
|
||||||
|
id="path7" />
|
||||||
|
<path
|
||||||
|
d="m 149.43988,26.480639 c 0.38018,-2.335754 -1.1846,-4.508374 -3.52082,-4.88852 L 61.817351,7.9076636 C 59.481136,7.5275576 57.308066,9.0920839 56.927894,11.427736 L 39.439773,118.87426 c -0.380182,2.33576 1.184602,4.50838 3.520816,4.88852 l 84.102711,13.68346 c 2.33622,0.38011 4.50929,-1.18442 4.88946,-3.52007 L 149.43688,26.479639 Z"
|
||||||
|
style="display:inline;fill:#c0c0c0;fill-opacity:1;stroke:#989898;stroke-width:1;stroke-opacity:1"
|
||||||
|
id="path10" />
|
||||||
|
</g>
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 2.1 KiB |
BIN
cockatrice/resources/backgrounds/home.png
Normal file
|
After Width: | Height: | Size: 12 MiB |
|
|
@ -0,0 +1,29 @@
|
||||||
|
<?xml version="1.0" encoding="UTF-8" standalone="no"?>
|
||||||
|
<!--!Font Awesome Free 7.1.0 by @fontawesome - https://fontawesome.com License - https://fontawesome.com/license/free Copyright 2026 Fonticons, Inc.-->
|
||||||
|
|
||||||
|
<svg
|
||||||
|
version="1.1"
|
||||||
|
width="172.65051"
|
||||||
|
id="svg13"
|
||||||
|
height="213.30714"
|
||||||
|
xml:space="preserve"
|
||||||
|
xmlns:svg="http://www.w3.org/2000/svg"
|
||||||
|
xmlns="http://www.w3.org/2000/svg"><defs
|
||||||
|
id="defs13" /><g
|
||||||
|
transform="matrix(1.705559,0,0,1.705559,-97.653345,-68.741256)"
|
||||||
|
id="g13"><path
|
||||||
|
d="m 151.48519,45.063813 c 0,-2.366441 -1.89356,-4.259575 -4.26046,-4.259575 H 62.016385 c -2.366905,0 -4.260468,1.893134 -4.260468,4.259575 V 153.91977 c 0,2.36644 1.893563,4.25957 4.260468,4.25957 h 85.208345 c 2.3669,0 4.26046,-1.89313 4.26046,-4.25957 z"
|
||||||
|
style="fill:#c0c0c0;fill-opacity:1;stroke:#989898;stroke-width:1;stroke-opacity:1"
|
||||||
|
id="path7" /><path
|
||||||
|
d="m 154.70135,48.441704 c 0,-2.366441 -1.89356,-4.259575 -4.26046,-4.259575 H 65.232545 c -2.366905,0 -4.260468,1.893134 -4.260468,4.259575 V 157.29767 c 0,2.36644 1.893563,4.25957 4.260468,4.25957 h 85.208345 c 2.3669,0 4.26046,-1.89313 4.26046,-4.25957 z"
|
||||||
|
style="fill:#c0c0c0;fill-opacity:1;stroke:#989898;stroke-width:1;stroke-opacity:1"
|
||||||
|
id="path7-5" /><path
|
||||||
|
d="m 157.98403,51.75453 c 0,-2.366441 -1.89356,-4.259575 -4.26046,-4.259575 H 68.515228 c -2.366905,0 -4.260468,1.893134 -4.260468,4.259575 v 108.85596 c 0,2.36644 1.893563,4.25957 4.260468,4.25957 h 85.208342 c 2.3669,0 4.26046,-1.89313 4.26046,-4.25957 z"
|
||||||
|
style="fill:#c0c0c0;fill-opacity:1;stroke:#989898;stroke-width:1;stroke-opacity:1"
|
||||||
|
id="path7-6" /></g><path
|
||||||
|
d="m 196.24576,207.42361 c 0,0.11213 0,0.22413 0,0.33621 -0.0498,4.54511 -4.18399,7.63329 -8.72909,7.63329 h -12.19086 c -3.29988,0 -5.97713,2.67727 -5.97713,5.97712 0,0.4234 0.0498,0.83433 0.12449,1.23279 0.26149,1.27014 0.80939,2.49046 1.34485,3.72325 0.75959,1.71843 1.50674,3.4244 1.50674,5.22998 0,3.95986 -2.68971,7.55859 -6.64956,7.72046 -0.43583,0.0128 -0.87166,0.025 -1.31995,0.025 -17.60761,0 -31.878,-14.27038 -31.878,-31.878 0,-17.60761 14.28284,-31.878 31.89046,-31.878 17.60762,0 31.87801,14.27039 31.87801,31.878 z m -47.81703,3.98475 c 0,-2.20407 -1.78067,-3.98475 -3.98473,-3.98475 -2.20407,0 -3.98477,1.78068 -3.98477,3.98475 0,2.20406 1.7807,3.98475 3.98477,3.98475 2.20406,0 3.98473,-1.78069 3.98473,-3.98475 z m 0,-11.95426 c 2.20407,0 3.98477,-1.78068 3.98477,-3.98475 0,-2.20407 -1.7807,-3.98475 -3.98477,-3.98475 -2.20405,0 -3.98473,1.78068 -3.98473,3.98475 0,2.20407 1.78068,3.98475 3.98473,3.98475 z m 19.92376,-11.95424 c 0,-2.20408 -1.78068,-3.98477 -3.98473,-3.98477 -2.20407,0 -3.98477,1.78069 -3.98477,3.98477 0,2.20405 1.7807,3.98474 3.98477,3.98474 2.20405,0 3.98473,-1.78069 3.98473,-3.98474 z m 11.95426,11.95424 c 2.20407,0 3.98475,-1.78068 3.98475,-3.98475 0,-2.20407 -1.78068,-3.98475 -3.98475,-3.98475 -2.20406,0 -3.98475,1.78068 -3.98475,3.98475 0,2.20407 1.78069,3.98475 3.98475,3.98475 z"
|
||||||
|
id="path1"
|
||||||
|
style="display:none;fill:#3b3b3b;fill-opacity:1;stroke:#000000;stroke-width:2.53798;stroke-dasharray:none;stroke-opacity:1" /><path
|
||||||
|
d="M 126.20915,54.574783 82.324247,83.8512 c -5.76807,3.845383 -9.435059,10.089163 -10.029703,16.90777 12.348823,2.53716 22.081191,12.26955 24.638191,24.63819 6.838435,-0.59465 13.062395,-4.26163 16.907775,-10.0297 l 29.2566,-43.904722 c 1.32804,-2.001984 2.04162,-4.340923 2.04162,-6.75915 0,-6.719506 -5.45092,-12.170428 -12.17043,-12.170428 -2.3984,0 -4.75718,0.713573 -6.75915,2.041623 z M 88.052677,131.81933 c 0,-12.26953 -9.930593,-22.20012 -22.200138,-22.20012 -12.269532,0 -22.200126,9.93059 -22.200126,22.20012 0,0.77305 0.03966,1.54609 0.118929,2.2993 0.356787,3.46877 -2.021792,7.21505 -5.510393,7.21505 h -0.951431 c -3.508409,0 -6.342895,2.83447 -6.342895,6.3429 0,3.5084 2.834486,6.34289 6.342895,6.34289 h 28.543021 c 12.269545,0 22.200138,-9.93059 22.200138,-22.20014 z"
|
||||||
|
id="path1-2"
|
||||||
|
style="fill:#989898;fill-opacity:1;stroke-width:0.198215" /></svg>
|
||||||
|
After Width: | Height: | Size: 3.9 KiB |
|
Before Width: | Height: | Size: 42 KiB After Width: | Height: | Size: 9.9 KiB |
|
|
@ -1,65 +1,74 @@
|
||||||
[Rules]
|
[Rules]
|
||||||
# The default log level is info
|
# The default log level is info
|
||||||
*.debug = false
|
*.debug = false
|
||||||
|
#*.info = true
|
||||||
|
#*.warning = true
|
||||||
|
#*.critical = true
|
||||||
|
#*.fatal = true
|
||||||
|
|
||||||
# Uncomment a rule to disable logging for that category,
|
# Uncomment a rule to see debug level logs for that category,
|
||||||
# or set .debug = true for that category to see debug level logs
|
# or set <category> = false to disable logging
|
||||||
|
|
||||||
# main = false
|
#main = true
|
||||||
# qt_translator = false
|
#qt_translator = true
|
||||||
# window_main.* = false
|
#window_main.* = true
|
||||||
# release_channel = false
|
#release_channel = true
|
||||||
# spoiler_background_updater = false
|
#spoiler_background_updater = true
|
||||||
# theme_manager = false
|
#theme_manager = true
|
||||||
# sound_engine = false
|
#sound_engine = true
|
||||||
# tapped_out_interface = false
|
#tapped_out_interface = true
|
||||||
|
|
||||||
# tab_game = false
|
#tab_game = true
|
||||||
# tab_message = false
|
#tab_message = true
|
||||||
# tab_supervisor = false
|
#tab_supervisor = true
|
||||||
|
|
||||||
# dlg_edit_avatar = false
|
#dlg_edit_avatar = true
|
||||||
# dlg_settings = false
|
#dlg_load_deck_from_website = true
|
||||||
# dlg_tip_of_the_day = false
|
#dlg_settings = true
|
||||||
# dlg_update = false
|
#dlg_tip_of_the_day = true
|
||||||
|
#dlg_update = true
|
||||||
|
|
||||||
# settings_cache = false
|
#general_settings_page = true
|
||||||
# servers_settings = false
|
|
||||||
# shortcuts_settings = false
|
|
||||||
|
|
||||||
# local_client = false
|
#settings_cache = true
|
||||||
# remote_client = false
|
#servers_settings = true
|
||||||
|
#shortcuts_settings = true
|
||||||
|
|
||||||
# player = false
|
#local_client = true
|
||||||
# game_scene = false
|
#remote_client = true
|
||||||
# game_scene.player_addition_removal = false
|
|
||||||
# card_zone = false
|
|
||||||
# view_zone = false
|
|
||||||
|
|
||||||
# user_info_connection = false
|
#player = true
|
||||||
|
#game_scene = true
|
||||||
|
#game_scene.player_addition_removal = true
|
||||||
|
#card_zone = true
|
||||||
|
#view_zone = true
|
||||||
|
|
||||||
# picture_loader = false
|
#game_event_handler = true
|
||||||
# picture_loader.worker = false
|
|
||||||
# picture_loader.card_back_cache_fail = false
|
|
||||||
# picture_loader.picture_to_load = false
|
|
||||||
# deck_loader = false
|
|
||||||
# card_database = false
|
|
||||||
# card_database.loading = false
|
|
||||||
# card_database.loading.success_or_failure = false
|
|
||||||
# cockatrice_xml.* = false
|
|
||||||
# cockatrice_xml.xml_3_parser = false
|
|
||||||
# cockatrice_xml.xml_4_parser = false
|
|
||||||
# card_info = false
|
|
||||||
# card_list = false
|
|
||||||
|
|
||||||
#flow_layout = false
|
#user_info_connection = true
|
||||||
#flow_widget = false
|
|
||||||
#flow_widget.size = false
|
|
||||||
|
|
||||||
# card_info_picture_widget = false
|
#card_picture_loader = true
|
||||||
|
#card_picture_loader.worker = true
|
||||||
|
#card_picture_loader.card_back_cache_fail = true
|
||||||
|
#card_picture_loader.picture_to_load = true
|
||||||
|
#deck_loader = true
|
||||||
|
#card_database = true
|
||||||
|
#card_database.loading = true
|
||||||
|
#card_database.loading.success_or_failure = true
|
||||||
|
#cockatrice_xml.* = true
|
||||||
|
#cockatrice_xml.xml_3_parser = true
|
||||||
|
#cockatrice_xml.xml_4_parser = true
|
||||||
|
#card_info = true
|
||||||
|
#card_list = true
|
||||||
|
|
||||||
# pixel_map_generator = false
|
#flow_layout = true
|
||||||
|
#flow_widget = true
|
||||||
|
#flow_widget.size = true
|
||||||
|
|
||||||
# deck_filter_string = false
|
#card_info_picture_widget = true
|
||||||
# filter_string = false
|
|
||||||
# syntax_help = false
|
#pixel_map_generator = true
|
||||||
|
|
||||||
|
#deck_filter_string = true
|
||||||
|
#filter_string = true
|
||||||
|
#syntax_help = true
|
||||||
|
|
|
||||||
799
cockatrice/resources/config/storage.svg
Normal file
|
|
@ -0,0 +1,799 @@
|
||||||
|
<?xml version="1.0" encoding="UTF-8" standalone="no"?>
|
||||||
|
<!-- Created with Inkscape (http://www.inkscape.org/) -->
|
||||||
|
|
||||||
|
<svg
|
||||||
|
xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape"
|
||||||
|
xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd"
|
||||||
|
xmlns:xlink="http://www.w3.org/1999/xlink"
|
||||||
|
xmlns:svg="http://www.w3.org/2000/svg"
|
||||||
|
xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
|
||||||
|
xmlns:cc="http://creativecommons.org/ns#"
|
||||||
|
xmlns:dc="http://purl.org/dc/elements/1.1/"
|
||||||
|
width="62.636364"
|
||||||
|
height="62.090908"
|
||||||
|
id="svg2"
|
||||||
|
sodipodi:version="0.32"
|
||||||
|
inkscape:version="1.4.3 (0d15f75042, 2025-12-25)"
|
||||||
|
sodipodi:docname="storage.svg"
|
||||||
|
inkscape:output_extension="org.inkscape.output.svg.inkscape"
|
||||||
|
version="1.0"
|
||||||
|
xml:space="preserve"
|
||||||
|
xmlns="http://www.w3.org/2000/svg"><defs
|
||||||
|
id="defs4"><linearGradient
|
||||||
|
id="linearGradient3169"><stop
|
||||||
|
style="stop-color:#0000ff;stop-opacity:1;"
|
||||||
|
offset="0"
|
||||||
|
id="stop3171" /><stop
|
||||||
|
style="stop-color:#000067;stop-opacity:1;"
|
||||||
|
offset="1"
|
||||||
|
id="stop3173" /></linearGradient><linearGradient
|
||||||
|
id="linearGradient4766"><stop
|
||||||
|
style="stop-color:#784421;stop-opacity:1;"
|
||||||
|
offset="0"
|
||||||
|
id="stop4768" /><stop
|
||||||
|
style="stop-color:#3d2210;stop-opacity:0;"
|
||||||
|
offset="1"
|
||||||
|
id="stop4770" /></linearGradient><linearGradient
|
||||||
|
id="linearGradient4758"><stop
|
||||||
|
style="stop-color:#a05a2c;stop-opacity:1;"
|
||||||
|
offset="0"
|
||||||
|
id="stop4760" /><stop
|
||||||
|
style="stop-color:#3d2210;stop-opacity:1;"
|
||||||
|
offset="1"
|
||||||
|
id="stop4762" /></linearGradient><inkscape:perspective
|
||||||
|
sodipodi:type="inkscape:persp3d"
|
||||||
|
inkscape:vp_x="0 : 526.18109 : 1"
|
||||||
|
inkscape:vp_y="0 : 1000 : 0"
|
||||||
|
inkscape:vp_z="744.09448 : 526.18109 : 1"
|
||||||
|
inkscape:persp3d-origin="372.04724 : 350.78739 : 1"
|
||||||
|
id="perspective10" /><inkscape:perspective
|
||||||
|
id="perspective2484"
|
||||||
|
inkscape:persp3d-origin="372.04724 : 350.78739 : 1"
|
||||||
|
inkscape:vp_z="744.09448 : 526.18109 : 1"
|
||||||
|
inkscape:vp_y="0 : 1000 : 0"
|
||||||
|
inkscape:vp_x="0 : 526.18109 : 1"
|
||||||
|
sodipodi:type="inkscape:persp3d" /><linearGradient
|
||||||
|
inkscape:collect="always"
|
||||||
|
xlink:href="#linearGradient4758"
|
||||||
|
id="linearGradient4764"
|
||||||
|
x1="466.09601"
|
||||||
|
y1="485.96021"
|
||||||
|
x2="715.14801"
|
||||||
|
y2="485.96021"
|
||||||
|
gradientUnits="userSpaceOnUse" /><linearGradient
|
||||||
|
inkscape:collect="always"
|
||||||
|
xlink:href="#linearGradient4766"
|
||||||
|
id="linearGradient4772"
|
||||||
|
x1="496.548"
|
||||||
|
y1="485.26816"
|
||||||
|
x2="683.31201"
|
||||||
|
y2="485.26816"
|
||||||
|
gradientUnits="userSpaceOnUse" /><radialGradient
|
||||||
|
inkscape:collect="always"
|
||||||
|
xlink:href="#linearGradient3169"
|
||||||
|
id="radialGradient3175"
|
||||||
|
cx="120.07376"
|
||||||
|
cy="56.138123"
|
||||||
|
fx="120.07376"
|
||||||
|
fy="56.138123"
|
||||||
|
r="82.790039"
|
||||||
|
gradientTransform="matrix(1,0,0,0.2116376,0,44.257186)"
|
||||||
|
gradientUnits="userSpaceOnUse" /><linearGradient
|
||||||
|
inkscape:collect="always"
|
||||||
|
xlink:href="#linearGradient6482"
|
||||||
|
id="linearGradient6488"
|
||||||
|
x1="32.18182"
|
||||||
|
y1="3.2835093"
|
||||||
|
x2="32.18182"
|
||||||
|
y2="13.02554"
|
||||||
|
gradientUnits="userSpaceOnUse"
|
||||||
|
gradientTransform="matrix(1.0281354,0,0,1.0429299,85.21874,131.0326)" /><linearGradient
|
||||||
|
id="linearGradient6482"><stop
|
||||||
|
style="stop-color:#ffffff;stop-opacity:1;"
|
||||||
|
offset="0"
|
||||||
|
id="stop6484" /><stop
|
||||||
|
style="stop-color:#00ff00;stop-opacity:0;"
|
||||||
|
offset="1"
|
||||||
|
id="stop6486" /></linearGradient><linearGradient
|
||||||
|
inkscape:collect="always"
|
||||||
|
xlink:href="#linearGradient6464"
|
||||||
|
id="linearGradient6470"
|
||||||
|
x1="32.090908"
|
||||||
|
y1="1.8181819"
|
||||||
|
x2="31.09091"
|
||||||
|
y2="62.909088"
|
||||||
|
gradientUnits="userSpaceOnUse"
|
||||||
|
gradientTransform="translate(0,-0.1818182)" /><linearGradient
|
||||||
|
id="linearGradient6464"><stop
|
||||||
|
style="stop-color:#0061ff;stop-opacity:1;"
|
||||||
|
offset="0"
|
||||||
|
id="stop6466" /><stop
|
||||||
|
style="stop-color:#001c4c;stop-opacity:1;"
|
||||||
|
offset="1"
|
||||||
|
id="stop6468" /></linearGradient><linearGradient
|
||||||
|
y2="62.909088"
|
||||||
|
x2="31.09091"
|
||||||
|
y1="1.8181819"
|
||||||
|
x1="32.090908"
|
||||||
|
gradientTransform="translate(86.2151,131.5372)"
|
||||||
|
gradientUnits="userSpaceOnUse"
|
||||||
|
id="linearGradient4477"
|
||||||
|
xlink:href="#linearGradient6464"
|
||||||
|
inkscape:collect="always" /><linearGradient
|
||||||
|
inkscape:collect="always"
|
||||||
|
id="linearGradient2916"><stop
|
||||||
|
style="stop-color:white;stop-opacity:1;"
|
||||||
|
offset="0"
|
||||||
|
id="stop2918" /><stop
|
||||||
|
style="stop-color:white;stop-opacity:0;"
|
||||||
|
offset="1"
|
||||||
|
id="stop2920" /></linearGradient><linearGradient
|
||||||
|
inkscape:collect="always"
|
||||||
|
id="linearGradient2902"><stop
|
||||||
|
style="stop-color:black;stop-opacity:1;"
|
||||||
|
offset="0"
|
||||||
|
id="stop2905" /><stop
|
||||||
|
style="stop-color:black;stop-opacity:0;"
|
||||||
|
offset="1"
|
||||||
|
id="stop2907" /></linearGradient><linearGradient
|
||||||
|
id="linearGradient2064"><stop
|
||||||
|
id="stop2066"
|
||||||
|
offset="0"
|
||||||
|
style="stop-color:white;stop-opacity:1;" /><stop
|
||||||
|
style="stop-color:#555753;stop-opacity:0.60000002;"
|
||||||
|
offset="0.5"
|
||||||
|
id="stop2070" /><stop
|
||||||
|
id="stop2068"
|
||||||
|
offset="1"
|
||||||
|
style="stop-color:#555753;stop-opacity:0;" /></linearGradient><linearGradient
|
||||||
|
id="linearGradient9641"><stop
|
||||||
|
style="stop-color:white;stop-opacity:1"
|
||||||
|
offset="0"
|
||||||
|
id="stop9643" /><stop
|
||||||
|
style="stop-color:#888a85;stop-opacity:1"
|
||||||
|
offset="1"
|
||||||
|
id="stop9645" /></linearGradient><linearGradient
|
||||||
|
id="linearGradient9633"><stop
|
||||||
|
style="stop-color:#eeeeec;stop-opacity:1"
|
||||||
|
offset="0"
|
||||||
|
id="stop9635" /><stop
|
||||||
|
style="stop-color:#888a85;stop-opacity:1"
|
||||||
|
offset="1"
|
||||||
|
id="stop9639" /></linearGradient><linearGradient
|
||||||
|
id="linearGradient9613"><stop
|
||||||
|
style="stop-color:white;stop-opacity:1"
|
||||||
|
offset="0"
|
||||||
|
id="stop9615" /><stop
|
||||||
|
id="stop9619"
|
||||||
|
offset="0.5"
|
||||||
|
style="stop-color:white;stop-opacity:1;" /><stop
|
||||||
|
style="stop-color:#cccfca;stop-opacity:1"
|
||||||
|
offset="1"
|
||||||
|
id="stop9617" /></linearGradient><linearGradient
|
||||||
|
id="linearGradient8710"><stop
|
||||||
|
style="stop-color:black;stop-opacity:1;"
|
||||||
|
offset="0"
|
||||||
|
id="stop8712" /><stop
|
||||||
|
style="stop-color:white;stop-opacity:1;"
|
||||||
|
offset="1"
|
||||||
|
id="stop8714" /></linearGradient><linearGradient
|
||||||
|
id="linearGradient8631"><stop
|
||||||
|
id="stop8633"
|
||||||
|
offset="0"
|
||||||
|
style="stop-color:#eeeeec;stop-opacity:1" /><stop
|
||||||
|
style="stop-color:#eeeeec;stop-opacity:1;"
|
||||||
|
offset="0.2"
|
||||||
|
id="stop8637" /><stop
|
||||||
|
id="stop8635"
|
||||||
|
offset="1"
|
||||||
|
style="stop-color:#babdb6;stop-opacity:1" /></linearGradient><linearGradient
|
||||||
|
id="linearGradient8625"><stop
|
||||||
|
id="stop8627"
|
||||||
|
offset="0"
|
||||||
|
style="stop-color:white;stop-opacity:1" /><stop
|
||||||
|
id="stop8629"
|
||||||
|
offset="1"
|
||||||
|
style="stop-color:#babdb6;stop-opacity:1" /></linearGradient><linearGradient
|
||||||
|
id="linearGradient8613"><stop
|
||||||
|
style="stop-color:#babdb6;stop-opacity:1"
|
||||||
|
offset="0"
|
||||||
|
id="stop8615" /><stop
|
||||||
|
style="stop-color:#2e3436;stop-opacity:1"
|
||||||
|
offset="1"
|
||||||
|
id="stop8617" /></linearGradient><linearGradient
|
||||||
|
id="linearGradient5740"><stop
|
||||||
|
style="stop-color:#d0d0cb;stop-opacity:1;"
|
||||||
|
offset="0"
|
||||||
|
id="stop5742" /><stop
|
||||||
|
style="stop-color:#babdb6;stop-opacity:1"
|
||||||
|
offset="1"
|
||||||
|
id="stop5744" /></linearGradient><linearGradient
|
||||||
|
id="linearGradient5690"><stop
|
||||||
|
style="stop-color:white;stop-opacity:1;"
|
||||||
|
offset="0"
|
||||||
|
id="stop5692" /><stop
|
||||||
|
style="stop-color:#888a85;stop-opacity:0.59848487"
|
||||||
|
offset="1"
|
||||||
|
id="stop5694" /></linearGradient><linearGradient
|
||||||
|
id="linearGradient2899"><stop
|
||||||
|
id="stop2901"
|
||||||
|
offset="0"
|
||||||
|
style="stop-color:#555753;stop-opacity:1" /><stop
|
||||||
|
id="stop2903"
|
||||||
|
offset="1"
|
||||||
|
style="stop-color:#2e3436;stop-opacity:1" /></linearGradient><linearGradient
|
||||||
|
id="linearGradient3468"><stop
|
||||||
|
style="stop-color:#fdfdfc;stop-opacity:1"
|
||||||
|
offset="0"
|
||||||
|
id="stop3470" /><stop
|
||||||
|
style="stop-color:white;stop-opacity:0.37121212"
|
||||||
|
offset="1"
|
||||||
|
id="stop3472" /></linearGradient><linearGradient
|
||||||
|
id="linearGradient2909"><stop
|
||||||
|
style="stop-color:white;stop-opacity:0;"
|
||||||
|
offset="0"
|
||||||
|
id="stop2911" /><stop
|
||||||
|
id="stop2917"
|
||||||
|
offset="0.5"
|
||||||
|
style="stop-color:white;stop-opacity:1;" /><stop
|
||||||
|
style="stop-color:white;stop-opacity:0;"
|
||||||
|
offset="1"
|
||||||
|
id="stop2913" /></linearGradient><linearGradient
|
||||||
|
id="linearGradient2839"><stop
|
||||||
|
style="stop-color:white;stop-opacity:0.25773194;"
|
||||||
|
offset="0"
|
||||||
|
id="stop2841" /><stop
|
||||||
|
id="stop2847"
|
||||||
|
offset="0.5472973"
|
||||||
|
style="stop-color:white;stop-opacity:1;" /><stop
|
||||||
|
style="stop-color:white;stop-opacity:0.24705882;"
|
||||||
|
offset="0.66243607"
|
||||||
|
id="stop2849" /><stop
|
||||||
|
id="stop2851"
|
||||||
|
offset="0.875"
|
||||||
|
style="stop-color:white;stop-opacity:0.83505154;" /><stop
|
||||||
|
style="stop-color:white;stop-opacity:0;"
|
||||||
|
offset="1"
|
||||||
|
id="stop2843" /></linearGradient><linearGradient
|
||||||
|
id="linearGradient2900"><stop
|
||||||
|
style="stop-color:black;stop-opacity:0;"
|
||||||
|
offset="0"
|
||||||
|
id="stop2902" /><stop
|
||||||
|
id="stop2908"
|
||||||
|
offset="0.5"
|
||||||
|
style="stop-color:black;stop-opacity:1;" /><stop
|
||||||
|
style="stop-color:black;stop-opacity:0;"
|
||||||
|
offset="1"
|
||||||
|
id="stop2904" /></linearGradient><linearGradient
|
||||||
|
inkscape:collect="always"
|
||||||
|
xlink:href="#linearGradient3468"
|
||||||
|
id="linearGradient3474"
|
||||||
|
x1="24.748737"
|
||||||
|
y1="35.354588"
|
||||||
|
x2="24.998737"
|
||||||
|
y2="14.997767"
|
||||||
|
gradientUnits="userSpaceOnUse"
|
||||||
|
gradientTransform="matrix(1,0,0,0.995556,0,-3.931113)" /><radialGradient
|
||||||
|
inkscape:collect="always"
|
||||||
|
xlink:href="#linearGradient2902"
|
||||||
|
id="radialGradient4700"
|
||||||
|
gradientUnits="userSpaceOnUse"
|
||||||
|
gradientTransform="matrix(4.095822,0,0,3.101282,-9.53921,-94.5433)"
|
||||||
|
cx="0"
|
||||||
|
cy="17"
|
||||||
|
fx="0"
|
||||||
|
fy="17"
|
||||||
|
r="2" /><radialGradient
|
||||||
|
inkscape:collect="always"
|
||||||
|
xlink:href="#linearGradient2902"
|
||||||
|
id="radialGradient4702"
|
||||||
|
gradientUnits="userSpaceOnUse"
|
||||||
|
gradientTransform="matrix(4.095822,0,0,3.101282,38.20996,-10.90025)"
|
||||||
|
cx="0"
|
||||||
|
cy="17"
|
||||||
|
fx="0"
|
||||||
|
fy="17"
|
||||||
|
r="2" /><linearGradient
|
||||||
|
inkscape:collect="always"
|
||||||
|
xlink:href="#linearGradient2900"
|
||||||
|
id="linearGradient4704"
|
||||||
|
gradientUnits="userSpaceOnUse"
|
||||||
|
gradientTransform="matrix(2.047911,0,0,2.067521,1.347566,6.673675)"
|
||||||
|
x1="9.8994951"
|
||||||
|
y1="20"
|
||||||
|
x2="9.8994951"
|
||||||
|
y2="13.979153" /><linearGradient
|
||||||
|
inkscape:collect="always"
|
||||||
|
xlink:href="#linearGradient2909"
|
||||||
|
id="linearGradient4711"
|
||||||
|
gradientUnits="userSpaceOnUse"
|
||||||
|
gradientTransform="matrix(1,0,0,1.42294,10.5,-14.95703)"
|
||||||
|
x1="15.335379"
|
||||||
|
y1="33.06237"
|
||||||
|
x2="20.329321"
|
||||||
|
y2="36.37693" /><linearGradient
|
||||||
|
inkscape:collect="always"
|
||||||
|
xlink:href="#linearGradient2909"
|
||||||
|
id="linearGradient4713"
|
||||||
|
gradientUnits="userSpaceOnUse"
|
||||||
|
gradientTransform="matrix(1,0,0,1.42294,-0.875,-15.04578)"
|
||||||
|
x1="15.335379"
|
||||||
|
y1="33.06237"
|
||||||
|
x2="20.329321"
|
||||||
|
y2="36.37693" /><linearGradient
|
||||||
|
inkscape:collect="always"
|
||||||
|
xlink:href="#linearGradient2909"
|
||||||
|
id="linearGradient4715"
|
||||||
|
gradientUnits="userSpaceOnUse"
|
||||||
|
gradientTransform="matrix(0.459833,0,-0.391165,1.370105,40.62503,-13.29892)"
|
||||||
|
x1="15.335379"
|
||||||
|
y1="33.06237"
|
||||||
|
x2="20.329321"
|
||||||
|
y2="36.37693" /><radialGradient
|
||||||
|
inkscape:collect="always"
|
||||||
|
xlink:href="#linearGradient5740"
|
||||||
|
id="radialGradient5748"
|
||||||
|
cx="25.251999"
|
||||||
|
cy="16.47991"
|
||||||
|
fx="25.251999"
|
||||||
|
fy="16.47991"
|
||||||
|
r="21.980215"
|
||||||
|
gradientTransform="matrix(1.032991,-0.596398,0.575121,0.99614,-12.23456,11.55448)"
|
||||||
|
gradientUnits="userSpaceOnUse" /><linearGradient
|
||||||
|
inkscape:collect="always"
|
||||||
|
xlink:href="#linearGradient2064"
|
||||||
|
id="linearGradient5790"
|
||||||
|
gradientUnits="userSpaceOnUse"
|
||||||
|
x1="18.048874"
|
||||||
|
y1="25.461344"
|
||||||
|
x2="22.211937"
|
||||||
|
y2="12.143078"
|
||||||
|
gradientTransform="matrix(0.940224,0,0,0.931632,1.331811,1.401537)" /><linearGradient
|
||||||
|
inkscape:collect="always"
|
||||||
|
xlink:href="#linearGradient8631"
|
||||||
|
id="linearGradient5865"
|
||||||
|
x1="24"
|
||||||
|
y1="36.638382"
|
||||||
|
x2="25.818018"
|
||||||
|
y2="6.8314762"
|
||||||
|
gradientUnits="userSpaceOnUse" /><linearGradient
|
||||||
|
inkscape:collect="always"
|
||||||
|
xlink:href="#linearGradient2839"
|
||||||
|
id="linearGradient7658"
|
||||||
|
gradientUnits="userSpaceOnUse"
|
||||||
|
x1="27.057796"
|
||||||
|
y1="12.669416"
|
||||||
|
x2="32.042896"
|
||||||
|
y2="31.219666" /><linearGradient
|
||||||
|
inkscape:collect="always"
|
||||||
|
xlink:href="#linearGradient5690"
|
||||||
|
id="linearGradient8603"
|
||||||
|
x1="20.304037"
|
||||||
|
y1="24.035707"
|
||||||
|
x2="18.498415"
|
||||||
|
y2="40.647167"
|
||||||
|
gradientUnits="userSpaceOnUse" /><radialGradient
|
||||||
|
inkscape:collect="always"
|
||||||
|
xlink:href="#linearGradient8613"
|
||||||
|
id="radialGradient8619"
|
||||||
|
cx="7.5177727"
|
||||||
|
cy="30.573555"
|
||||||
|
fx="7.5177727"
|
||||||
|
fy="30.573555"
|
||||||
|
r="0.53125"
|
||||||
|
gradientTransform="matrix(1.662477,0,0,1.61358,-4.989175,-18.65647)"
|
||||||
|
gradientUnits="userSpaceOnUse" /><radialGradient
|
||||||
|
inkscape:collect="always"
|
||||||
|
xlink:href="#linearGradient9613"
|
||||||
|
id="radialGradient8623"
|
||||||
|
gradientUnits="userSpaceOnUse"
|
||||||
|
gradientTransform="matrix(1.389748,0,0,1.348872,-2.91982,-10.63815)"
|
||||||
|
cx="7.5191436"
|
||||||
|
cy="30.304251"
|
||||||
|
fx="7.5191436"
|
||||||
|
fy="30.304251"
|
||||||
|
r="0.53125" /><radialGradient
|
||||||
|
inkscape:collect="always"
|
||||||
|
xlink:href="#linearGradient9633"
|
||||||
|
id="radialGradient8664"
|
||||||
|
gradientUnits="userSpaceOnUse"
|
||||||
|
gradientTransform="matrix(1.569487,0,0,1.523325,-4.288627,-15.92107)"
|
||||||
|
cx="7.5336008"
|
||||||
|
cy="30.307562"
|
||||||
|
fx="7.5336008"
|
||||||
|
fy="30.307562"
|
||||||
|
r="0.53125" /><radialGradient
|
||||||
|
inkscape:collect="always"
|
||||||
|
xlink:href="#linearGradient8613"
|
||||||
|
id="radialGradient8666"
|
||||||
|
gradientUnits="userSpaceOnUse"
|
||||||
|
gradientTransform="matrix(1.662477,0,0,1.61358,-4.989175,-18.65647)"
|
||||||
|
cx="7.5177727"
|
||||||
|
cy="30.573555"
|
||||||
|
fx="7.5177727"
|
||||||
|
fy="30.573555"
|
||||||
|
r="0.53125" /><radialGradient
|
||||||
|
inkscape:collect="always"
|
||||||
|
xlink:href="#linearGradient8625"
|
||||||
|
id="radialGradient8676"
|
||||||
|
gradientUnits="userSpaceOnUse"
|
||||||
|
gradientTransform="matrix(1.662477,0,0,1.61358,-4.989175,-18.65647)"
|
||||||
|
cx="7.4792061"
|
||||||
|
cy="30.36071"
|
||||||
|
fx="7.4792061"
|
||||||
|
fy="30.36071"
|
||||||
|
r="0.53125" /><radialGradient
|
||||||
|
inkscape:collect="always"
|
||||||
|
xlink:href="#linearGradient8613"
|
||||||
|
id="radialGradient8678"
|
||||||
|
gradientUnits="userSpaceOnUse"
|
||||||
|
gradientTransform="matrix(1.662477,0,0,1.61358,-4.989175,-18.65647)"
|
||||||
|
cx="7.5177727"
|
||||||
|
cy="30.573555"
|
||||||
|
fx="7.5177727"
|
||||||
|
fy="30.573555"
|
||||||
|
r="0.53125" /><radialGradient
|
||||||
|
inkscape:collect="always"
|
||||||
|
xlink:href="#linearGradient9641"
|
||||||
|
id="radialGradient8680"
|
||||||
|
gradientUnits="userSpaceOnUse"
|
||||||
|
gradientTransform="matrix(1.662477,0,0,1.61358,-4.989175,-18.65647)"
|
||||||
|
cx="7.4893188"
|
||||||
|
cy="30.337601"
|
||||||
|
fx="7.4893188"
|
||||||
|
fy="30.337601"
|
||||||
|
r="0.53125" /><radialGradient
|
||||||
|
inkscape:collect="always"
|
||||||
|
xlink:href="#linearGradient8613"
|
||||||
|
id="radialGradient8682"
|
||||||
|
gradientUnits="userSpaceOnUse"
|
||||||
|
gradientTransform="matrix(1.662477,0,0,1.61358,-4.989175,-18.65647)"
|
||||||
|
cx="7.5177727"
|
||||||
|
cy="30.573555"
|
||||||
|
fx="7.5177727"
|
||||||
|
fy="30.573555"
|
||||||
|
r="0.53125" /><linearGradient
|
||||||
|
inkscape:collect="always"
|
||||||
|
xlink:href="#linearGradient8710"
|
||||||
|
id="linearGradient8716"
|
||||||
|
x1="40.617188"
|
||||||
|
y1="30.554688"
|
||||||
|
x2="40.710938"
|
||||||
|
y2="30.359375"
|
||||||
|
gradientUnits="userSpaceOnUse" /><linearGradient
|
||||||
|
inkscape:collect="always"
|
||||||
|
xlink:href="#linearGradient8710"
|
||||||
|
id="linearGradient9605"
|
||||||
|
gradientUnits="userSpaceOnUse"
|
||||||
|
x1="40.617188"
|
||||||
|
y1="30.554688"
|
||||||
|
x2="40.710938"
|
||||||
|
y2="30.359375"
|
||||||
|
gradientTransform="matrix(0.602867,-0.797841,0.797841,0.602867,-41.12611,44.62773)" /><linearGradient
|
||||||
|
inkscape:collect="always"
|
||||||
|
xlink:href="#linearGradient8710"
|
||||||
|
id="linearGradient9649"
|
||||||
|
gradientUnits="userSpaceOnUse"
|
||||||
|
x1="40.617188"
|
||||||
|
y1="30.554688"
|
||||||
|
x2="40.710938"
|
||||||
|
y2="30.359375"
|
||||||
|
gradientTransform="rotate(-30.000012,-5.5813167,76.089146)" /><linearGradient
|
||||||
|
inkscape:collect="always"
|
||||||
|
xlink:href="#linearGradient8710"
|
||||||
|
id="linearGradient9654"
|
||||||
|
gradientUnits="userSpaceOnUse"
|
||||||
|
x1="40.617188"
|
||||||
|
y1="30.554688"
|
||||||
|
x2="40.710938"
|
||||||
|
y2="30.359375"
|
||||||
|
gradientTransform="matrix(0.707107,0.527555,-0.707107,0.527555,29.0058,-24.09196)" /><linearGradient
|
||||||
|
inkscape:collect="always"
|
||||||
|
xlink:href="#linearGradient2916"
|
||||||
|
id="linearGradient2973"
|
||||||
|
x1="12.5"
|
||||||
|
y1="43.1875"
|
||||||
|
x2="12.5"
|
||||||
|
y2="34.045513"
|
||||||
|
gradientUnits="userSpaceOnUse" /><linearGradient
|
||||||
|
inkscape:collect="always"
|
||||||
|
xlink:href="#linearGradient2899"
|
||||||
|
id="linearGradient5655"
|
||||||
|
gradientUnits="userSpaceOnUse"
|
||||||
|
x1="53.812813"
|
||||||
|
y1="43.573235"
|
||||||
|
x2="-2.8138931"
|
||||||
|
y2="35.500015"
|
||||||
|
gradientTransform="translate(0,50)" /><linearGradient
|
||||||
|
inkscape:collect="always"
|
||||||
|
xlink:href="#linearGradient2902"
|
||||||
|
id="linearGradient2992"
|
||||||
|
x1="21.9375"
|
||||||
|
y1="39"
|
||||||
|
x2="21.9375"
|
||||||
|
y2="37.995617"
|
||||||
|
gradientUnits="userSpaceOnUse" /><linearGradient
|
||||||
|
inkscape:collect="always"
|
||||||
|
xlink:href="#linearGradient2902"
|
||||||
|
id="linearGradient2910"
|
||||||
|
x1="22.101398"
|
||||||
|
y1="27.658131"
|
||||||
|
x2="22.971142"
|
||||||
|
y2="20.903238"
|
||||||
|
gradientUnits="userSpaceOnUse"
|
||||||
|
gradientTransform="translate(0,2)" /><linearGradient
|
||||||
|
inkscape:collect="always"
|
||||||
|
xlink:href="#linearGradient2916"
|
||||||
|
id="linearGradient2922"
|
||||||
|
x1="24.847851"
|
||||||
|
y1="28.908398"
|
||||||
|
x2="24.847851"
|
||||||
|
y2="25.757175"
|
||||||
|
gradientUnits="userSpaceOnUse"
|
||||||
|
gradientTransform="translate(0,2)" /></defs><sodipodi:namedview
|
||||||
|
id="base"
|
||||||
|
pagecolor="#ffffff"
|
||||||
|
bordercolor="#666666"
|
||||||
|
borderopacity="1.0"
|
||||||
|
inkscape:pageopacity="0.0"
|
||||||
|
inkscape:pageshadow="2"
|
||||||
|
inkscape:zoom="5.8830488"
|
||||||
|
inkscape:cx="-3.9945275"
|
||||||
|
inkscape:cy="-14.363301"
|
||||||
|
inkscape:document-units="px"
|
||||||
|
inkscape:current-layer="layer1"
|
||||||
|
showgrid="false"
|
||||||
|
inkscape:window-width="2560"
|
||||||
|
inkscape:window-height="1408"
|
||||||
|
inkscape:window-x="0"
|
||||||
|
inkscape:window-y="32"
|
||||||
|
inkscape:window-maximized="1"
|
||||||
|
fit-margin-top="0"
|
||||||
|
fit-margin-left="0"
|
||||||
|
fit-margin-right="0"
|
||||||
|
fit-margin-bottom="0"
|
||||||
|
inkscape:showpageshadow="0"
|
||||||
|
inkscape:pagecheckerboard="0"
|
||||||
|
inkscape:deskcolor="#505050" /><metadata
|
||||||
|
id="metadata7"><rdf:RDF><cc:Work
|
||||||
|
rdf:about=""><dc:format>image/svg+xml</dc:format><dc:type
|
||||||
|
rdf:resource="http://purl.org/dc/dcmitype/StillImage" /></cc:Work></rdf:RDF></metadata><g
|
||||||
|
inkscape:label="Ebene 1"
|
||||||
|
inkscape:groupmode="layer"
|
||||||
|
id="layer1"
|
||||||
|
transform="translate(-86.987816,-132.85536)"><rect
|
||||||
|
style="fill:url(#linearGradient4477);fill-opacity:1;fill-rule:nonzero;stroke:#000000;stroke-opacity:1"
|
||||||
|
id="rect6462"
|
||||||
|
width="61.636364"
|
||||||
|
height="61.090908"
|
||||||
|
x="87.487816"
|
||||||
|
y="133.35536"
|
||||||
|
ry="5.6363635" /><rect
|
||||||
|
style="fill:url(#linearGradient6488);fill-opacity:1;fill-rule:nonzero;stroke:none"
|
||||||
|
id="rect6472"
|
||||||
|
width="59.796619"
|
||||||
|
height="13.251164"
|
||||||
|
x="88.407707"
|
||||||
|
y="134.45705"
|
||||||
|
ry="4.7325583" /><g
|
||||||
|
inkscape:label="Livello 1"
|
||||||
|
id="layer1-3"
|
||||||
|
style="display:inline"
|
||||||
|
transform="matrix(1.1537183,0,0,1.1537183,91.003924,136.40297)"><g
|
||||||
|
id="g3519"
|
||||||
|
style="opacity:0.7"
|
||||||
|
transform="matrix(1.030831,0,0,1.151147,-0.73609,-12.57431)"
|
||||||
|
inkscape:export-filename="/home/lapo/Desktop/uhm.png"
|
||||||
|
inkscape:export-xdpi="90"
|
||||||
|
inkscape:export-ydpi="90"><rect
|
||||||
|
transform="scale(-1)"
|
||||||
|
y="-48.024086"
|
||||||
|
x="-9.5392103"
|
||||||
|
height="12.405126"
|
||||||
|
width="8.1916437"
|
||||||
|
id="rect2884"
|
||||||
|
style="opacity:1;fill:url(#radialGradient4700);fill-opacity:1;stroke:none;stroke-width:1;stroke-linecap:round;stroke-linejoin:round;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:0;stroke-opacity:1" /><rect
|
||||||
|
y="35.618961"
|
||||||
|
x="38.209965"
|
||||||
|
height="12.405126"
|
||||||
|
width="8.1916437"
|
||||||
|
id="rect2894"
|
||||||
|
style="opacity:1;fill:url(#radialGradient4702);fill-opacity:1;stroke:none;stroke-width:1;stroke-linecap:round;stroke-linejoin:round;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:0;stroke-opacity:1" /><rect
|
||||||
|
y="35.618961"
|
||||||
|
x="9.5392103"
|
||||||
|
height="12.405126"
|
||||||
|
width="28.670753"
|
||||||
|
id="rect2898"
|
||||||
|
style="opacity:1;fill:url(#linearGradient4704);fill-opacity:1;stroke:none;stroke-width:1;stroke-linecap:round;stroke-linejoin:round;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:0;stroke-opacity:1" /></g><g
|
||||||
|
id="g5672"
|
||||||
|
transform="translate(0,-48.99747)"><path
|
||||||
|
sodipodi:nodetypes="ccccccccccccc"
|
||||||
|
id="rect2010"
|
||||||
|
d="M 4.5182287,80.500013 H 43.481768 c 0.564099,0 1.018229,0.45413 1.018229,1.018229 v 2.963543 c 0,1.315584 -0.450231,3.018228 -2.455729,3.018228 L 40.5,87.5 v 1 h -33 v -1 l -1.8567713,1.3e-5 c -1.2712053,0 -2.1432282,-0.884627 -2.1432282,-2.255665 v -3.726106 c 0,-0.564099 0.4541297,-1.018229 1.0182282,-1.018229 z"
|
||||||
|
style="color:#000000;display:inline;overflow:visible;visibility:visible;fill:url(#linearGradient5655);fill-opacity:1;fill-rule:nonzero;stroke:#2e3436;stroke-width:1;stroke-linecap:square;stroke-linejoin:round;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:1.4;stroke-opacity:1;marker:none;marker-start:none;marker-mid:none;marker-end:none" /><path
|
||||||
|
transform="translate(0,50)"
|
||||||
|
style="color:#000000;display:inline;overflow:visible;visibility:visible;opacity:0.1;fill:none;fill-opacity:1;fill-rule:nonzero;stroke:url(#linearGradient2973);stroke-width:1;stroke-linecap:square;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dashoffset:1.4;stroke-opacity:1;marker:none;marker-start:none;marker-mid:none;marker-end:none"
|
||||||
|
d="m 4.59375,31.59375 v 3.729743 c 0,0.599619 0.3756505,1.104854 0.8863276,1.104854 H 42.426407 c 0.512469,0 0.979843,-0.507235 0.979843,-1.016466 V 31.59375 Z"
|
||||||
|
id="path2076"
|
||||||
|
sodipodi:nodetypes="ccccccc" /><g
|
||||||
|
transform="translate(0,50)"
|
||||||
|
style="opacity:0.5"
|
||||||
|
id="g4706"><path
|
||||||
|
style="opacity:0.109524;fill:url(#linearGradient4711);fill-opacity:1;fill-rule:evenodd;stroke:none;stroke-width:1px;stroke-linecap:butt;stroke-linejoin:miter;stroke-opacity:1"
|
||||||
|
d="m 26.144738,32.088747 c 0,0 -1.502602,5.533939 -3.226175,5.911253 0,0 6.231378,-0.125771 6.231378,-0.125771 1.387072,-0.317461 3.358758,-5.785482 3.358758,-5.785482 z"
|
||||||
|
id="path2907"
|
||||||
|
sodipodi:nodetypes="ccccc"
|
||||||
|
inkscape:export-filename="/home/lapo/Desktop/uhm.png"
|
||||||
|
inkscape:export-xdpi="90"
|
||||||
|
inkscape:export-ydpi="90" /><path
|
||||||
|
inkscape:export-ydpi="90"
|
||||||
|
inkscape:export-xdpi="90"
|
||||||
|
inkscape:export-filename="/home/lapo/Desktop/uhm.png"
|
||||||
|
sodipodi:nodetypes="ccccc"
|
||||||
|
id="path2892"
|
||||||
|
d="m 14.769738,32 c 0,0 -1.502602,5.533939 -3.226175,5.911253 0,0 6.231378,-0.125771 6.231378,-0.125771 C 19.162013,37.468021 21.133699,32 21.133699,32 Z"
|
||||||
|
style="opacity:0.109524;fill:url(#linearGradient4713);fill-opacity:1;fill-rule:evenodd;stroke:none;stroke-width:1px;stroke-linecap:butt;stroke-linejoin:miter;stroke-opacity:1" /><path
|
||||||
|
inkscape:export-ydpi="90"
|
||||||
|
inkscape:export-xdpi="90"
|
||||||
|
inkscape:export-filename="/home/lapo/Desktop/uhm.png"
|
||||||
|
sodipodi:nodetypes="ccccc"
|
||||||
|
id="path2896"
|
||||||
|
d="m 34.886139,32 c 0,0 -2.212224,5.328458 -3.108503,5.691761 0,0 2.899969,-0.121101 2.899969,-0.121101 C 35.402697,37.264987 37.8125,32 37.8125,32 Z"
|
||||||
|
style="opacity:0.109524;fill:url(#linearGradient4715);fill-opacity:1;fill-rule:evenodd;stroke:none;stroke-width:1px;stroke-linecap:butt;stroke-linejoin:miter;stroke-opacity:1" /></g></g><path
|
||||||
|
style="fill:url(#radialGradient5748);fill-opacity:1;stroke:#888a85;stroke-width:1;stroke-linecap:round;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dashoffset:0;stroke-opacity:1"
|
||||||
|
d="m 11.693127,10.498788 h 24.572566 c 1.68417,0 2.396517,0.117479 3.040019,2.385005 l 5.074491,17.881119 c 0.501024,1.765471 -1.355848,2.735101 -3.040018,2.735101 H 6.6186312 c -1.868408,0 -3.4893833,-1.181417 -3.0400182,-2.735101 L 8.8290448,12.611497 c 0.5683008,-1.964905 1.1799122,-2.112709 2.8640822,-2.112709 z"
|
||||||
|
id="rect1879"
|
||||||
|
sodipodi:nodetypes="cczzcczzc"
|
||||||
|
inkscape:export-filename="/home/lapo/Desktop/uhm.png"
|
||||||
|
inkscape:export-xdpi="90"
|
||||||
|
inkscape:export-ydpi="90" /><path
|
||||||
|
sodipodi:type="inkscape:offset"
|
||||||
|
inkscape:radius="-0.5"
|
||||||
|
inkscape:original="M 11.6875 10.5 C 10.00333 10.5 9.4120513 10.660095 8.84375 12.625 L 3.59375 30.75 C 3.1443849 32.303684 4.7565918 33.500002 6.625 33.5 L 41.34375 33.5 C 43.02792 33.5 44.876024 32.515471 44.375 30.75 L 39.3125 12.875 C 38.668998 10.607474 37.965419 10.5 36.28125 10.5 L 11.6875 10.5 z "
|
||||||
|
style="display:inline;opacity:0.462406;fill:url(#linearGradient7658);fill-opacity:1;stroke:none;stroke-width:1;stroke-linecap:round;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dashoffset:0;stroke-opacity:1"
|
||||||
|
id="path5806"
|
||||||
|
d="m 11.6875,11 c -0.826242,0 -1.28475,0.05742 -1.5625,0.242188 -0.2777497,0.184768 -0.5284825,0.580009 -0.8007812,1.521484 l -5.2500001,18.125 c -0.1708248,0.590628 0.021709,1.039316 0.4902344,1.4375 C 5.0329784,32.724356 5.7975106,33.000001 6.625,33 h 34.71875 c 0.744655,0 1.538941,-0.232575 2.03125,-0.609375 0.492309,-0.3768 0.719298,-0.799984 0.519531,-1.503906 l -5.0625,-17.875 C 38.52278,11.922001 38.224454,11.462814 37.910156,11.253906 37.595859,11.044998 37.112699,11 36.28125,11 Z" /><ellipse
|
||||||
|
style="color:#000000;display:inline;overflow:visible;visibility:visible;opacity:1;fill:url(#radialGradient8623);fill-opacity:1;fill-rule:nonzero;stroke:none;stroke-width:1;stroke-linecap:square;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:1.4;stroke-opacity:1;marker:none;marker-start:none;marker-mid:none;marker-end:none"
|
||||||
|
id="path8621"
|
||||||
|
transform="matrix(-2.628602,0,0,1.777765,27.79309,-23.77739)"
|
||||||
|
cx="7.625"
|
||||||
|
cy="30.578125"
|
||||||
|
rx="0.53125"
|
||||||
|
ry="0.515625" /><path
|
||||||
|
style="fill:none;fill-opacity:1;stroke:url(#linearGradient5790);stroke-width:1;stroke-linecap:round;stroke-linejoin:round;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:0;stroke-opacity:1"
|
||||||
|
d="m 16.110953,12.552805 c -0.573581,0 -1.02837,0.431821 -1.02837,0.989859 l -0.940223,3.230801 c -2.859962,1.276514 -4.6423552,3.099073 -4.6423552,5.123976 0,3.856957 6.4790242,6.987239 14.4853222,6.98724 8.006296,0 14.514705,-3.130284 14.514704,-6.98724 0,-2.039034 -1.835591,-3.875388 -4.730501,-5.153089 l -0.940224,-3.201688 c 0,-0.558038 -0.454788,-0.989859 -1.02837,-0.989859 z"
|
||||||
|
id="path2784"
|
||||||
|
sodipodi:nodetypes="cccssscccc" /><path
|
||||||
|
style="display:inline;fill:none;fill-opacity:1;stroke:url(#linearGradient3474);stroke-width:1;stroke-linecap:round;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dashoffset:0;stroke-opacity:1"
|
||||||
|
d="m 11.6875,11.500005 c -0.803124,0 -1.097168,0.07051 -1.21875,0.155556 -0.121582,0.08504 -0.357707,0.40212 -0.6875,1.306667 l -5.25,18.137786 c -0.1337204,0.366765 -0.054827,0.533865 0.3125,0.84 0.3673267,0.306136 1.066693,0.56 1.78125,0.560001 h 34.71875 c 0.639793,0 1.393345,-0.237954 1.78125,-0.52889 0.387905,-0.290935 0.488311,-0.382809 0.3125,-0.871111 L 38.375,13.242228 c -0.377206,-1.04766 -0.68208,-1.439297 -0.84375,-1.555556 -0.16167,-0.116259 -0.443711,-0.186667 -1.25,-0.186667 z"
|
||||||
|
id="path3394"
|
||||||
|
sodipodi:nodetypes="csccsccsccscc" /><g
|
||||||
|
id="g5657"
|
||||||
|
transform="translate(7,-1)"
|
||||||
|
style="opacity:0.302857"><rect
|
||||||
|
ry="0.74712253"
|
||||||
|
rx="0.75130093"
|
||||||
|
y="35.500008"
|
||||||
|
x="18.499996"
|
||||||
|
height="1.9999924"
|
||||||
|
width="14.000004"
|
||||||
|
id="rect5641"
|
||||||
|
style="color:#000000;display:inline;overflow:visible;visibility:visible;opacity:1;fill:#ffffff;fill-opacity:1;fill-rule:nonzero;stroke:#eeeeec;stroke-width:1;stroke-linecap:square;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:1.4;stroke-opacity:1;marker:none;marker-start:none;marker-mid:none;marker-end:none" /><rect
|
||||||
|
y="36"
|
||||||
|
x="19"
|
||||||
|
height="1"
|
||||||
|
width="1"
|
||||||
|
id="rect5645"
|
||||||
|
style="color:#000000;display:inline;overflow:visible;visibility:visible;opacity:1;fill:#000000;fill-opacity:1;fill-rule:nonzero;stroke:none;stroke-width:1;stroke-linecap:square;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:1.4;stroke-opacity:1;marker:none;marker-start:none;marker-mid:none;marker-end:none" /><rect
|
||||||
|
y="36"
|
||||||
|
x="22"
|
||||||
|
height="1"
|
||||||
|
width="1"
|
||||||
|
id="rect5647"
|
||||||
|
style="color:#000000;display:inline;overflow:visible;visibility:visible;opacity:1;fill:#000000;fill-opacity:1;fill-rule:nonzero;stroke:none;stroke-width:1;stroke-linecap:square;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:1.4;stroke-opacity:1;marker:none;marker-start:none;marker-mid:none;marker-end:none" /><rect
|
||||||
|
y="36"
|
||||||
|
x="24"
|
||||||
|
height="1"
|
||||||
|
width="1"
|
||||||
|
id="rect5649"
|
||||||
|
style="color:#000000;display:inline;overflow:visible;visibility:visible;opacity:1;fill:#000000;fill-opacity:1;fill-rule:nonzero;stroke:none;stroke-width:1;stroke-linecap:square;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:1.4;stroke-opacity:1;marker:none;marker-start:none;marker-mid:none;marker-end:none" /><rect
|
||||||
|
y="36"
|
||||||
|
x="26"
|
||||||
|
height="1"
|
||||||
|
width="1"
|
||||||
|
id="rect5651"
|
||||||
|
style="color:#000000;display:inline;overflow:visible;visibility:visible;opacity:1;fill:#000000;fill-opacity:1;fill-rule:nonzero;stroke:none;stroke-width:1;stroke-linecap:square;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:1.4;stroke-opacity:1;marker:none;marker-start:none;marker-mid:none;marker-end:none" /><rect
|
||||||
|
y="36"
|
||||||
|
x="29"
|
||||||
|
height="1"
|
||||||
|
width="2"
|
||||||
|
id="rect5653"
|
||||||
|
style="color:#000000;display:inline;overflow:visible;visibility:visible;opacity:1;fill:#000000;fill-opacity:1;fill-rule:nonzero;stroke:none;stroke-width:1;stroke-linecap:square;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:1.4;stroke-opacity:1;marker:none;marker-start:none;marker-mid:none;marker-end:none" /></g><path
|
||||||
|
sodipodi:type="inkscape:offset"
|
||||||
|
inkscape:radius="-0.44194174"
|
||||||
|
inkscape:original="M 16.125 12.5625 C 15.55142 12.5625 15.09375 12.973212 15.09375 13.53125 L 14.15625 16.78125 C 11.296288 18.057765 9.5 19.881347 9.5 21.90625 C 9.5 25.763206 15.993702 28.874999 24 28.875 C 32.006296 28.874999 38.500001 25.763206 38.5 21.90625 C 38.5 19.867215 36.67616 18.027701 33.78125 16.75 L 32.84375 13.53125 C 32.843748 12.973212 32.386082 12.5625 31.8125 12.5625 L 16.125 12.5625 z "
|
||||||
|
style="display:inline;fill:url(#linearGradient5865);fill-opacity:1;stroke:none;stroke-width:1;stroke-linecap:round;stroke-linejoin:round;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:0;stroke-opacity:1"
|
||||||
|
id="path5857"
|
||||||
|
d="m 16.125,13.003906 c -0.362612,0 -0.589844,0.210942 -0.589844,0.527344 a 0.44198593,0.44198593 0 0 1 -0.01758,0.123047 l -0.9375,3.25 a 0.44198593,0.44198593 0 0 1 -0.24414,0.28125 c -1.389903,0.620369 -2.504368,1.368471 -3.25586,2.177734 -0.751491,0.809263 -1.1386718,1.661199 -1.1386717,2.542969 0,1.680455 1.4530107,3.311153 3.9980467,4.533203 2.545037,1.22205 6.114654,1.99414 10.060547,1.994141 3.945892,-1e-6 7.51551,-0.772091 10.060547,-1.994141 2.545037,-1.22205 3.998047,-2.852748 3.998047,-4.533203 0,-0.887751 -0.391823,-1.747213 -1.154297,-2.5625 -0.762474,-0.815287 -1.893636,-1.568394 -3.300781,-2.189453 a 0.44198593,0.44198593 0 0 1 -0.246094,-0.28125 l -0.9375,-3.21875 a 0.44198593,0.44198593 0 0 1 -0.01758,-0.123047 c -10e-7,-0.316404 -0.22723,-0.527344 -0.589844,-0.527344 z" /><ellipse
|
||||||
|
style="color:#000000;display:inline;overflow:visible;visibility:visible;opacity:0.9;fill:#000000;fill-opacity:0.0530303;fill-rule:nonzero;stroke:url(#linearGradient8603);stroke-width:2.52015;stroke-linecap:square;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:1.4;stroke-opacity:1;marker:none;marker-start:none;marker-mid:none;marker-end:none"
|
||||||
|
id="path8595"
|
||||||
|
transform="matrix(0.449978,0,0,0.349909,16.36363,12.21469)"
|
||||||
|
cx="16.970562"
|
||||||
|
cy="25.107418"
|
||||||
|
rx="7.7781744"
|
||||||
|
ry="4.2868347" /><ellipse
|
||||||
|
style="color:#000000;display:inline;overflow:visible;visibility:visible;opacity:1;fill:url(#radialGradient8619);fill-opacity:1;fill-rule:nonzero;stroke:none;stroke-width:1;stroke-linecap:square;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:1.4;stroke-opacity:1;marker:none;marker-start:none;marker-mid:none;marker-end:none"
|
||||||
|
id="path8611"
|
||||||
|
transform="matrix(1.411772,0,0,0.969697,-3.014767,0.848485)"
|
||||||
|
cx="7.625"
|
||||||
|
cy="30.578125"
|
||||||
|
rx="0.53125"
|
||||||
|
ry="0.515625" /><ellipse
|
||||||
|
style="color:#000000;display:inline;overflow:visible;visibility:visible;opacity:1;fill:url(#radialGradient8664);fill-opacity:1;fill-rule:nonzero;stroke:none;stroke-width:0.462594;stroke-linecap:square;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:1.4;stroke-opacity:1;marker:none;marker-start:none;marker-mid:none;marker-end:none"
|
||||||
|
id="path8660"
|
||||||
|
transform="matrix(-2.628602,0,0,1.777765,60.79309,-23.77739)"
|
||||||
|
cx="7.625"
|
||||||
|
cy="30.578125"
|
||||||
|
rx="0.53125"
|
||||||
|
ry="0.515625" /><ellipse
|
||||||
|
style="color:#000000;display:inline;overflow:visible;visibility:visible;opacity:1;fill:url(#radialGradient8666);fill-opacity:1;fill-rule:nonzero;stroke:none;stroke-width:1;stroke-linecap:square;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:1.4;stroke-opacity:1;marker:none;marker-start:none;marker-mid:none;marker-end:none"
|
||||||
|
id="path8662"
|
||||||
|
transform="matrix(1.411772,0,0,0.969697,29.98523,0.848485)"
|
||||||
|
cx="7.625"
|
||||||
|
cy="30.578125"
|
||||||
|
rx="0.53125"
|
||||||
|
ry="0.515625" /><ellipse
|
||||||
|
style="color:#000000;display:inline;overflow:visible;visibility:visible;opacity:1;fill:url(#radialGradient8676);fill-opacity:1;fill-rule:nonzero;stroke:none;stroke-width:1;stroke-linecap:square;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:1.4;stroke-opacity:1;marker:none;marker-start:none;marker-mid:none;marker-end:none"
|
||||||
|
id="path8668"
|
||||||
|
transform="matrix(-2.628602,0,0,1.777765,31.79309,-40.77739)"
|
||||||
|
cx="7.625"
|
||||||
|
cy="30.578125"
|
||||||
|
rx="0.53125"
|
||||||
|
ry="0.515625" /><ellipse
|
||||||
|
style="color:#000000;display:inline;overflow:visible;visibility:visible;opacity:1;fill:url(#radialGradient8678);fill-opacity:1;fill-rule:nonzero;stroke:none;stroke-width:1;stroke-linecap:square;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:1.4;stroke-opacity:1;marker:none;marker-start:none;marker-mid:none;marker-end:none"
|
||||||
|
id="path8670"
|
||||||
|
transform="matrix(1.411772,0,0,0.969697,0.985233,-16.15152)"
|
||||||
|
cx="7.625"
|
||||||
|
cy="30.578125"
|
||||||
|
rx="0.53125"
|
||||||
|
ry="0.515625" /><ellipse
|
||||||
|
style="color:#000000;display:inline;overflow:visible;visibility:visible;opacity:1;fill:url(#radialGradient8680);fill-opacity:1;fill-rule:nonzero;stroke:none;stroke-width:1;stroke-linecap:square;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:1.4;stroke-opacity:1;marker:none;marker-start:none;marker-mid:none;marker-end:none"
|
||||||
|
id="path8672"
|
||||||
|
transform="matrix(-2.628602,0,0,1.777765,56.3029,-40.77739)"
|
||||||
|
cx="7.625"
|
||||||
|
cy="30.578125"
|
||||||
|
rx="0.53125"
|
||||||
|
ry="0.515625" /><ellipse
|
||||||
|
style="color:#000000;display:inline;overflow:visible;visibility:visible;opacity:1;fill:url(#radialGradient8682);fill-opacity:1;fill-rule:nonzero;stroke:none;stroke-width:1;stroke-linecap:square;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:1.4;stroke-opacity:1;marker:none;marker-start:none;marker-mid:none;marker-end:none"
|
||||||
|
id="path8674"
|
||||||
|
transform="matrix(1.411772,0,0,0.969697,25.49504,-16.15152)"
|
||||||
|
cx="7.625"
|
||||||
|
cy="30.578125"
|
||||||
|
rx="0.53125"
|
||||||
|
ry="0.515625" /><path
|
||||||
|
style="opacity:0.4;fill:none;fill-opacity:1;fill-rule:evenodd;stroke:url(#linearGradient8716);stroke-width:0.3;stroke-linecap:round;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1"
|
||||||
|
d="m 40.328109,30.261401 0.874999,0.430332"
|
||||||
|
id="path8700" /><path
|
||||||
|
style="display:inline;opacity:0.4;fill:none;fill-opacity:1;fill-rule:evenodd;stroke:url(#linearGradient9605);stroke-width:0.3;stroke-linecap:round;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1"
|
||||||
|
d="M 7.330186,30.695906 8.201031,30.257228"
|
||||||
|
id="path9603" /><path
|
||||||
|
style="display:inline;opacity:0.4;fill:none;fill-opacity:1;fill-rule:evenodd;stroke:url(#linearGradient9649);stroke-width:0.3;stroke-linecap:round;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1"
|
||||||
|
d="m 11.263531,13.446473 0.972937,-0.06482"
|
||||||
|
id="path9647" /><path
|
||||||
|
style="display:inline;opacity:0.4;fill:none;fill-opacity:1;fill-rule:evenodd;stroke:url(#linearGradient9654);stroke-width:0.3;stroke-linecap:round;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1"
|
||||||
|
d="m 36.124038,13.147874 0.314427,0.688634"
|
||||||
|
id="path9652" /><rect
|
||||||
|
style="color:#000000;display:inline;overflow:visible;visibility:visible;opacity:0.12;fill:url(#linearGradient2992);fill-opacity:1;fill-rule:nonzero;stroke:none;stroke-width:0.681836;stroke-linecap:butt;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:0;stroke-opacity:1;marker:none;marker-start:none;marker-mid:none;marker-end:none"
|
||||||
|
id="rect2984"
|
||||||
|
width="32.03125"
|
||||||
|
height="1"
|
||||||
|
x="8"
|
||||||
|
y="38" /><path
|
||||||
|
style="color:#000000;display:inline;overflow:visible;visibility:visible;opacity:0.12;fill:none;fill-opacity:1;fill-rule:nonzero;stroke:url(#linearGradient2910);stroke-width:1;stroke-linecap:round;stroke-linejoin:round;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:0;stroke-opacity:1;marker:none;marker-start:none;marker-mid:none;marker-end:none"
|
||||||
|
d="M 10.460155,15.082355 6.8513979,27.675762 C 8.2982685,28.375511 10.625,29.167061 10.429825,31.533131 H 37.299883 C 37.869398,29.640915 39.875,28.375 41.34614,28.25 L 37.498106,15.082355 32.350135,12.523347 H 14.318912 Z"
|
||||||
|
id="path1997"
|
||||||
|
sodipodi:nodetypes="ccccccccc" /><path
|
||||||
|
sodipodi:nodetypes="ccccc"
|
||||||
|
id="path2912"
|
||||||
|
d="m 7.9763979,27.050762 c 1.4468706,0.699749 3.1789321,1.433241 3.4256991,3.357369 H 36.857941 C 37.427456,28.515915 38.875,27.5 40.34614,27.375 Z"
|
||||||
|
style="color:#000000;display:inline;overflow:visible;visibility:visible;opacity:0.834286;fill:none;fill-opacity:1;fill-rule:nonzero;stroke:url(#linearGradient2922);stroke-width:1;stroke-linecap:round;stroke-linejoin:round;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:0;stroke-opacity:1;marker:none;marker-start:none;marker-mid:none;marker-end:none" /></g></g></svg>
|
||||||
|
After Width: | Height: | Size: 40 KiB |
|
|
@ -1,12 +1,38 @@
|
||||||
|
@page deck_search_syntax_help Deck Search Syntax Help
|
||||||
|
|
||||||
## Deck Search Syntax Help
|
## Deck Search Syntax Help
|
||||||
-----
|
|
||||||
The search bar recognizes a set of special commands.<br>
|
The search bar recognizes a set of special commands.<br>
|
||||||
In this list of examples below, each entry has an explanation and can be clicked to test the query. Note that all
|
In this list of examples below, each entry has an explanation and can be clicked to test the query. Note that all
|
||||||
searches are case insensitive.
|
searches are case insensitive.
|
||||||
|
|
||||||
<dl>
|
<dl>
|
||||||
<dt>Filename:</dt>
|
|
||||||
<dd>[red deck wins](#red deck wins) <small>(Any deck filename containing the words red, deck, and wins)</small></dd>
|
<dt>Display Name (The deck name, or the filename if the deck name isn't set):</dt>
|
||||||
<dd>["red deck wins"](#%22red deck wins%22) <small>(Any deck filename containing the exact phrase "red deck wins")</small></dd>
|
<dd>[red deck wins](#red deck wins) <small>(Any deck with a display name containing the words red, deck, and wins)</small></dd>
|
||||||
|
<dd>["red deck wins"](#%22red deck wins%22) <small>(Any deck with a display name containing the exact phrase "red deck wins")</small></dd>
|
||||||
|
|
||||||
|
<dt>Deck <u>N</u>ame:</dt>
|
||||||
|
<dd>[n:aggro](#n:aggro) <small>(Any deck with a name containing the word aggro)</small></dd>
|
||||||
|
<dd>[n:red n:deck n:wins](#n:red n:deck n:wins) <small>(Any deck with a name containing the words red, deck, and wins)</small></dd>
|
||||||
|
<dd>[n:"red deck wins"](#n:%22red deck wins%22) <small>(Any deck with a name containing the exact phrase "red deck wins")</small></dd>
|
||||||
|
|
||||||
|
<dt><u>F</u>ile <u>N</u>ame:</dt>
|
||||||
|
<dd>[fn:aggro](#fn:aggro) <small>(Any deck with a filename containing the word aggro)</small></dd>
|
||||||
|
<dd>[fn:red fn:deck fn:wins](#fn:red fn:deck fn:wins) <small>(Any deck with a filename containing the words red, deck, and wins)</small></dd>
|
||||||
|
<dd>[fn:"red deck wins"](#fn:%22red deck wins%22) <small>(Any deck with a filename containing the exact phrase "red deck wins")</small></dd>
|
||||||
|
|
||||||
|
<dt>Relative <u>P</u>ath (starting from the deck folder):</dt>
|
||||||
|
<dd>[p:aggro](#p:aggro) <small>(Any deck that has "aggro" somewhere in its relative path)</small></dd>
|
||||||
|
<dd>[p:edh/](#p:edh/) <small>(Any deck with "edh/" in its relative path, A.K.A. decks in the "edh" folder)</small></dd>
|
||||||
|
|
||||||
|
<dt><u>F</u>ormat:</dt>
|
||||||
|
<dd>[f:standard](#f:standard) <small>(Any deck with format set to standard)</small></dd>
|
||||||
|
|
||||||
|
<dt><u>C</u>omments:</dt>
|
||||||
|
<dd>[c:good](#c:good) <small>(Any deck with comments containing the word good)</small></dd>
|
||||||
|
<dd>[c:good c:deck](#c:good c:deck) <small>(Any deck with comments containing the words good and deck)</small></dd>
|
||||||
|
<dd>[c:"good deck"](#c:%22good deck%22) <small>(Any deck with comments containing the exact phrase "good deck")</small></dd>
|
||||||
|
|
||||||
<dt>Deck Contents (Uses [card search expressions](#cardSearchSyntaxHelp)):</dt>
|
<dt>Deck Contents (Uses [card search expressions](#cardSearchSyntaxHelp)):</dt>
|
||||||
<dd><a href="#[[plains]]">[[plains]]</a> <small>(Any deck that contains at least one card with "plains" in its name)</small></dd>
|
<dd><a href="#[[plains]]">[[plains]]</a> <small>(Any deck that contains at least one card with "plains" in its name)</small></dd>
|
||||||
|
|
@ -21,6 +47,6 @@ searches are case insensitive.
|
||||||
<dd>[t:aggro OR o:control](#t:aggro OR o:control) <small>(Any deck filename that contains either aggro or control)</small></dd>
|
<dd>[t:aggro OR o:control](#t:aggro OR o:control) <small>(Any deck filename that contains either aggro or control)</small></dd>
|
||||||
|
|
||||||
<dt>Grouping:</dt>
|
<dt>Grouping:</dt>
|
||||||
<dd><a href="#red -([[]]:100 or aggro)">red -([[]]:100 or aggro)</a> <small>(Any deck that has red in its filename but is not 100 cards or has aggro in its filename)</small></dd>
|
<dd><a href="#red -([[]]:100 OR aggro)">red -([[]]:100 OR aggro)</a> <small>(Any deck that has red in its filename but is not 100 cards or has aggro in its filename)</small></dd>
|
||||||
|
|
||||||
</dl>
|
</dl>
|
||||||
|
|
@ -1,8 +1,12 @@
|
||||||
|
@page search_syntax_help Search Syntax Help
|
||||||
|
|
||||||
## Search Syntax Help
|
## Search Syntax Help
|
||||||
-----
|
|
||||||
The search bar recognizes a set of special commands similar to some other card databases.<br>
|
The search bar recognizes a set of special commands similar to some other card databases.<br>
|
||||||
In this list of examples below, each entry has an explanation and can be clicked to test the query. Note that all searches are case insensitive.
|
In this list of examples below, each entry has an explanation and can be clicked to test the query. Note that all searches are case insensitive.
|
||||||
|
|
||||||
<dl>
|
<dl>
|
||||||
|
|
||||||
<dt>Name:</dt>
|
<dt>Name:</dt>
|
||||||
<dd>[birds of paradise](#birds of paradise) <small>(Any card name containing the words birds, of, and paradise)</small></dd>
|
<dd>[birds of paradise](#birds of paradise) <small>(Any card name containing the words birds, of, and paradise)</small></dd>
|
||||||
<dd>["birds of paradise"](#%22birds of paradise%22) <small>(Any card name containing the exact phrase "birds of paradise")</small></dd>
|
<dd>["birds of paradise"](#%22birds of paradise%22) <small>(Any card name containing the exact phrase "birds of paradise")</small></dd>
|
||||||
|
|
@ -47,15 +51,20 @@ In this list of examples below, each entry has an explanation and can be clicked
|
||||||
|
|
||||||
<dt><u>E</u>dition:</dt>
|
<dt><u>E</u>dition:</dt>
|
||||||
<dd>[set:lea](#set:lea) <small>(Cards that appear in Alpha, which has the set code LEA)</small></dd>
|
<dd>[set:lea](#set:lea) <small>(Cards that appear in Alpha, which has the set code LEA)</small></dd>
|
||||||
<dd>[e:lea or e:leb](#e:lea or e:leb) <small>(Cards that appear in Alpha or Beta)</small></dd>
|
<dd>[e:lea OR e:leb](#e:lea OR e:leb) <small>(Cards that appear in Alpha or Beta)</small></dd>
|
||||||
|
|
||||||
<dt>Negate:</dt>
|
<dt>Negate:</dt>
|
||||||
<dd>[c:wu -c:m](#c:wu -c:m) <small>(Any card that is white or blue, but not multicolored)</small></dd>
|
<dd>[c:wu -c:m](#c:wu -c:m) <small>(Any card that is white or blue, but not multicolored)</small></dd>
|
||||||
|
|
||||||
<dt>Branching:</dt>
|
<dt>Branching:</dt>
|
||||||
<dd>[t:sliver or o:changeling](#t:sliver or o:changeling) <small>(Any card that is either a sliver or has changeling)</small></dd>
|
<dd>[t:sliver OR o:changeling](#t:sliver OR o:changeling) <small>(Any card that is either a sliver or has changeling)</small></dd>
|
||||||
|
|
||||||
<dt>Grouping:</dt>
|
<dt>Grouping:</dt>
|
||||||
<dd><a href="#t:angel -(angel or c:w)">t:angel -(angel or c:w)</a> <small>(Any angel that doesn't have angel in its name and isn't white)</small></dd>
|
<dd><a href="#t:angel -(angel OR c:w)">t:angel -(angel OR c:w)</a> <small>(Any angel that doesn't have angel in its name and isn't white)</small></dd>
|
||||||
|
|
||||||
</dl>
|
<dt>Regular Expression:</dt>
|
||||||
|
<dd>[/^fell/](#/^fell/) <small>(Any card name that begins with "fell")</small></dd>
|
||||||
|
<dd>[o:/counter target .* spell/](#o:/counter target .* spell/) <small>(Any card text with "counter target *something* spell")</small></dd>
|
||||||
|
<dd>[o:/for each .* and\/or .*/](#o:/for each .* and\/or .*/) <small>(/'s can be escaped with a \)</small></dd>
|
||||||
|
|
||||||
|
</dl>
|
||||||
8
cockatrice/resources/icons/arrow_history.svg
Normal file
|
|
@ -0,0 +1,8 @@
|
||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<svg xmlns="http://www.w3.org/2000/svg" width="20" height="20" viewBox="0 0 20 20">
|
||||||
|
<title>
|
||||||
|
history
|
||||||
|
</title>
|
||||||
|
<path d="M9 6v5h.06l2.48 2.47 1.41-1.41L11 10.11V6H9z"/>
|
||||||
|
<path d="M10 1a9 9 0 0 0-7.85 13.35L.5 16H6v-5.5l-2.38 2.38A7 7 0 1 1 10 17v2a9 9 0 0 0 0-18z"/>
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 332 B |
40
cockatrice/resources/icons/arrow_redo.svg
Normal file
|
|
@ -0,0 +1,40 @@
|
||||||
|
<?xml version="1.0" encoding="windows-1252"?>
|
||||||
|
<!-- Generator: Adobe Illustrator 16.0.0, SVG Export Plug-In . SVG Version: 6.00 Build 0) -->
|
||||||
|
<!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN" "http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd">
|
||||||
|
<svg version="1.1" id="Capa_1" xmlns="http://www.w3.org/2000/svg" x="0px"
|
||||||
|
y="0px" width="485.212px" height="485.212px" viewBox="0 0 485.212 485.212"
|
||||||
|
style="enable-background:new 0 0 485.212 485.212;" xml:space="preserve">
|
||||||
|
<g>
|
||||||
|
<path d="M242.607,424.559c-75.252,0-136.468-61.209-136.468-136.465c0-75.252,61.216-136.466,136.468-136.466v90.978 l151.629-121.302L242.607,0v90.978c-108.687,0-197.117,88.432-197.117,197.117c0,108.691,88.43,197.118,197.117,197.118 c108.687,0,197.114-88.427,197.114-197.118h-60.645C379.077,363.35,317.859,424.559,242.607,424.559z"/>
|
||||||
|
</g>
|
||||||
|
<g>
|
||||||
|
</g>
|
||||||
|
<g>
|
||||||
|
</g>
|
||||||
|
<g>
|
||||||
|
</g>
|
||||||
|
<g>
|
||||||
|
</g>
|
||||||
|
<g>
|
||||||
|
</g>
|
||||||
|
<g>
|
||||||
|
</g>
|
||||||
|
<g>
|
||||||
|
</g>
|
||||||
|
<g>
|
||||||
|
</g>
|
||||||
|
<g>
|
||||||
|
</g>
|
||||||
|
<g>
|
||||||
|
</g>
|
||||||
|
<g>
|
||||||
|
</g>
|
||||||
|
<g>
|
||||||
|
</g>
|
||||||
|
<g>
|
||||||
|
</g>
|
||||||
|
<g>
|
||||||
|
</g>
|
||||||
|
<g>
|
||||||
|
</g>
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 1,018 B |
40
cockatrice/resources/icons/arrow_undo.svg
Normal file
|
|
@ -0,0 +1,40 @@
|
||||||
|
<?xml version="1.0" encoding="windows-1252"?>
|
||||||
|
<!-- Generator: Adobe Illustrator 16.0.0, SVG Export Plug-In . SVG Version: 6.00 Build 0) -->
|
||||||
|
<!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN" "http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd">
|
||||||
|
<svg version="1.1" id="Capa_1" xmlns="http://www.w3.org/2000/svg" x="0px"
|
||||||
|
y="0px" width="512px" height="512px" viewBox="0 0 512 512" style="enable-background:new 0 0 512 512;"
|
||||||
|
xml:space="preserve">
|
||||||
|
<g>
|
||||||
|
<path d="M256,448c79.406,0,144-64.594,144-144s-64.594-144-144-144v96L96,128L256,0v96c114.688,0,208,93.313,208,208 c0,114.688-93.312,208-208,208c-114.687,0-208-93.312-208-208h64C112,383.406,176.594,448,256,448z"/>
|
||||||
|
</g>
|
||||||
|
<g>
|
||||||
|
</g>
|
||||||
|
<g>
|
||||||
|
</g>
|
||||||
|
<g>
|
||||||
|
</g>
|
||||||
|
<g>
|
||||||
|
</g>
|
||||||
|
<g>
|
||||||
|
</g>
|
||||||
|
<g>
|
||||||
|
</g>
|
||||||
|
<g>
|
||||||
|
</g>
|
||||||
|
<g>
|
||||||
|
</g>
|
||||||
|
<g>
|
||||||
|
</g>
|
||||||
|
<g>
|
||||||
|
</g>
|
||||||
|
<g>
|
||||||
|
</g>
|
||||||
|
<g>
|
||||||
|
</g>
|
||||||
|
<g>
|
||||||
|
</g>
|
||||||
|
<g>
|
||||||
|
</g>
|
||||||
|
<g>
|
||||||
|
</g>
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 874 B |
1
cockatrice/resources/icons/circle_half_stroke.svg
Normal file
|
|
@ -0,0 +1 @@
|
||||||
|
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 640 640"><!--!Font Awesome Free 7.1.0 by @fontawesome - https://fontawesome.com License - https://fontawesome.com/license/free Copyright 2026 Fonticons, Inc.--><path d="M512 320C512 214 426 128 320 128L320 512C426 512 512 426 512 320zM64 320C64 178.6 178.6 64 320 64C461.4 64 576 178.6 576 320C576 461.4 461.4 576 320 576C178.6 576 64 461.4 64 320z"/></svg>
|
||||||
|
After Width: | Height: | Size: 410 B |
1
cockatrice/resources/icons/dragon.svg
Normal file
|
|
@ -0,0 +1 @@
|
||||||
|
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 640 640"><!--!Font Awesome Free 7.1.0 by @fontawesome - https://fontawesome.com License - https://fontawesome.com/license/free Copyright 2026 Fonticons, Inc.--><path d="M352 188.5L300.1 175.5C293.6 173.9 288.8 168.4 288.1 161.7C287.4 155 290.9 148.6 296.8 145.6L337.6 125.2L294.3 92.7C288.8 88.6 286.5 81.4 288.7 74.8C290.9 68.2 297.1 64 304 64L464 64C494.2 64 522.7 78.2 540.8 102.4L598.4 179.2C604.6 187.5 608 197.6 608 208C608 234.5 586.5 256 560 256L538.5 256C521.5 256 505.2 249.3 493.2 237.3L479.9 224L447.9 224L447.9 245.5C447.9 270.3 460.7 293.4 481.7 306.6L588.3 373.2C620.4 393.3 639.9 428.4 639.9 466.3C639.9 526.9 590.8 576.1 530.1 576.1L32.3 576C29 576 25.7 575.6 22.7 574.6C13.5 571.8 6 565 2.3 556C1 552.7 .1 549.1 0 545.3C-.2 541.6 .3 538 1.3 534.6C4.1 525.4 10.9 517.9 19.9 514.2C22.9 513 26.1 512.2 29.4 512L433.3 476C441.6 475.3 448 468.3 448 459.9C448 455.6 446.3 451.5 443.3 448.5L398.9 404.1C368.9 374.1 352 333.4 352 291L352 188.5zM512 136.3C512 136.2 512 136.1 512 136C512 135.9 512 135.8 512 135.7L512 136.3zM510.7 143.7L464.3 132.1C464.1 133.4 464 134.7 464 136C464 149.3 474.7 160 488 160C498.6 160 507.5 153.2 510.7 143.7zM130.9 180.5C147.2 166 171.3 164.3 189.4 176.4L320 263.4L320 290.9C320 323.7 328.4 355.7 344 383.9L112 383.9C105.3 383.9 99.3 379.7 97 373.5C94.7 367.3 96.5 360.2 101.6 355.8L171 296.3L18.4 319.8C11.4 320.9 4.5 317.2 1.5 310.8C-1.5 304.4 .1 296.8 5.4 292L130.9 180.5z"/></svg>
|
||||||
|
After Width: | Height: | Size: 1.4 KiB |
12
cockatrice/resources/icons/filter.svg
Normal file
|
|
@ -0,0 +1,12 @@
|
||||||
|
<?xml version="1.0" encoding="iso-8859-1"?>
|
||||||
|
<!-- Uploaded to: SVG Repo, www.svgrepo.com, Generator: SVG Repo Mixer Tools -->
|
||||||
|
<!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN" "http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd">
|
||||||
|
<svg fill="#000000" version="1.1" id="Capa_1" xmlns="http://www.w3.org/2000/svg"
|
||||||
|
width="800px" height="800px" viewBox="0 0 971.986 971.986"
|
||||||
|
xml:space="preserve">
|
||||||
|
<g>
|
||||||
|
<path d="M370.216,459.3c10.2,11.1,15.8,25.6,15.8,40.6v442c0,26.601,32.1,40.101,51.1,21.4l123.3-141.3
|
||||||
|
c16.5-19.8,25.6-29.601,25.6-49.2V500c0-15,5.7-29.5,15.8-40.601L955.615,75.5c26.5-28.8,6.101-75.5-33.1-75.5h-873
|
||||||
|
c-39.2,0-59.7,46.6-33.1,75.5L370.216,459.3z"/>
|
||||||
|
</g>
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 676 B |
1
cockatrice/resources/icons/floppy_disk.svg
Normal file
|
|
@ -0,0 +1 @@
|
||||||
|
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 640 640"><!--!Font Awesome Free 7.1.0 by @fontawesome - https://fontawesome.com License - https://fontawesome.com/license/free Copyright 2026 Fonticons, Inc.--><path d="M160 96C124.7 96 96 124.7 96 160L96 480C96 515.3 124.7 544 160 544L480 544C515.3 544 544 515.3 544 480L544 237.3C544 220.3 537.3 204 525.3 192L448 114.7C436 102.7 419.7 96 402.7 96L160 96zM192 192C192 174.3 206.3 160 224 160L384 160C401.7 160 416 174.3 416 192L416 256C416 273.7 401.7 288 384 288L224 288C206.3 288 192 273.7 192 256L192 192zM320 352C355.3 352 384 380.7 384 416C384 451.3 355.3 480 320 480C284.7 480 256 451.3 256 416C256 380.7 284.7 352 320 352z"/></svg>
|
||||||
|
After Width: | Height: | Size: 693 B |
1
cockatrice/resources/icons/gear.svg
Normal file
|
|
@ -0,0 +1 @@
|
||||||
|
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 640 640"><!--!Font Awesome Free 7.1.0 by @fontawesome - https://fontawesome.com License - https://fontawesome.com/license/free Copyright 2026 Fonticons, Inc.--><path d="M259.1 73.5C262.1 58.7 275.2 48 290.4 48L350.2 48C365.4 48 378.5 58.7 381.5 73.5L396 143.5C410.1 149.5 423.3 157.2 435.3 166.3L503.1 143.8C517.5 139 533.3 145 540.9 158.2L570.8 210C578.4 223.2 575.7 239.8 564.3 249.9L511 297.3C511.9 304.7 512.3 312.3 512.3 320C512.3 327.7 511.8 335.3 511 342.7L564.4 390.2C575.8 400.3 578.4 417 570.9 430.1L541 481.9C533.4 495 517.6 501.1 503.2 496.3L435.4 473.8C423.3 482.9 410.1 490.5 396.1 496.6L381.7 566.5C378.6 581.4 365.5 592 350.4 592L290.6 592C275.4 592 262.3 581.3 259.3 566.5L244.9 496.6C230.8 490.6 217.7 482.9 205.6 473.8L137.5 496.3C123.1 501.1 107.3 495.1 99.7 481.9L69.8 430.1C62.2 416.9 64.9 400.3 76.3 390.2L129.7 342.7C128.8 335.3 128.4 327.7 128.4 320C128.4 312.3 128.9 304.7 129.7 297.3L76.3 249.8C64.9 239.7 62.3 223 69.8 209.9L99.7 158.1C107.3 144.9 123.1 138.9 137.5 143.7L205.3 166.2C217.4 157.1 230.6 149.5 244.6 143.4L259.1 73.5zM320.3 400C364.5 399.8 400.2 363.9 400 319.7C399.8 275.5 363.9 239.8 319.7 240C275.5 240.2 239.8 276.1 240 320.3C240.2 364.5 276.1 400.2 320.3 400z"/></svg>
|
||||||
|
After Width: | Height: | Size: 1.2 KiB |
72
cockatrice/resources/icons/mana/C.svg
Normal file
|
|
@ -0,0 +1,72 @@
|
||||||
|
<?xml version="1.0" encoding="UTF-8" standalone="no"?>
|
||||||
|
<!-- Created with Inkscape (http://www.inkscape.org/) -->
|
||||||
|
|
||||||
|
<svg
|
||||||
|
width="169.33115mm"
|
||||||
|
height="169.59981mm"
|
||||||
|
viewBox="0 0 169.33115 169.59981"
|
||||||
|
version="1.1"
|
||||||
|
id="svg1"
|
||||||
|
inkscape:export-filename="C.svg"
|
||||||
|
inkscape:export-xdpi="96.000015"
|
||||||
|
inkscape:export-ydpi="96.000015"
|
||||||
|
inkscape:version="1.4.3 (0d15f75042, 2025-12-25)"
|
||||||
|
sodipodi:docname="colorless.svg"
|
||||||
|
xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape"
|
||||||
|
xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd"
|
||||||
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
|
xmlns:svg="http://www.w3.org/2000/svg">
|
||||||
|
<sodipodi:namedview
|
||||||
|
id="namedview1"
|
||||||
|
pagecolor="#ffffff"
|
||||||
|
bordercolor="#000000"
|
||||||
|
borderopacity="0.25"
|
||||||
|
inkscape:showpageshadow="2"
|
||||||
|
inkscape:pageopacity="0.0"
|
||||||
|
inkscape:pagecheckerboard="0"
|
||||||
|
inkscape:deskcolor="#d1d1d1"
|
||||||
|
inkscape:document-units="mm"
|
||||||
|
inkscape:zoom="0.71535494"
|
||||||
|
inkscape:cx="397.00572"
|
||||||
|
inkscape:cy="536.09751"
|
||||||
|
inkscape:window-width="1853"
|
||||||
|
inkscape:window-height="1011"
|
||||||
|
inkscape:window-x="0"
|
||||||
|
inkscape:window-y="0"
|
||||||
|
inkscape:window-maximized="1"
|
||||||
|
inkscape:current-layer="layer1">
|
||||||
|
<inkscape:page
|
||||||
|
x="0"
|
||||||
|
y="0"
|
||||||
|
width="169.33115"
|
||||||
|
height="169.59981"
|
||||||
|
id="page2"
|
||||||
|
margin="0"
|
||||||
|
bleed="0" />
|
||||||
|
</sodipodi:namedview>
|
||||||
|
<defs
|
||||||
|
id="defs1" />
|
||||||
|
<g
|
||||||
|
inkscape:label="Layer 1"
|
||||||
|
inkscape:groupmode="layer"
|
||||||
|
id="layer1"
|
||||||
|
transform="translate(-20.334425,-63.700102)">
|
||||||
|
<ellipse
|
||||||
|
style="fill:#cccccc;stroke:#000000;stroke-width:0.165333;stroke-linecap:round"
|
||||||
|
id="path1"
|
||||||
|
cx="-30.617094"
|
||||||
|
cy="179.22736"
|
||||||
|
transform="matrix(0.70654605,-0.70766707,0.70654608,0.70766704,0,0)"
|
||||||
|
rx="84.650612"
|
||||||
|
ry="84.65062" />
|
||||||
|
<rect
|
||||||
|
style="fill:none;stroke:#000000;stroke-width:14.5003;stroke-linecap:round;stroke-linejoin:bevel;stroke-dasharray:none"
|
||||||
|
id="rect1"
|
||||||
|
width="84.683701"
|
||||||
|
height="84.683769"
|
||||||
|
x="-221.60611"
|
||||||
|
y="-73.174698"
|
||||||
|
ry="0.084683768"
|
||||||
|
transform="matrix(-0.7073973,-0.70681615,0.7073973,-0.70681615,0,0)" />
|
||||||
|
</g>
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 2.2 KiB |
1
cockatrice/resources/icons/pen_to_square.svg
Normal file
|
|
@ -0,0 +1 @@
|
||||||
|
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 640 640"><!--!Font Awesome Free 7.1.0 by @fontawesome - https://fontawesome.com License - https://fontawesome.com/license/free Copyright 2026 Fonticons, Inc.--><path d="M535.6 85.7C513.7 63.8 478.3 63.8 456.4 85.7L432 110.1L529.9 208L554.3 183.6C576.2 161.7 576.2 126.3 554.3 104.4L535.6 85.7zM236.4 305.7C230.3 311.8 225.6 319.3 222.9 327.6L193.3 416.4C190.4 425 192.7 434.5 199.1 441C205.5 447.5 215 449.7 223.7 446.8L312.5 417.2C320.7 414.5 328.2 409.8 334.4 403.7L496 241.9L398.1 144L236.4 305.7zM160 128C107 128 64 171 64 224L64 480C64 533 107 576 160 576L416 576C469 576 512 533 512 480L512 384C512 366.3 497.7 352 480 352C462.3 352 448 366.3 448 384L448 480C448 497.7 433.7 512 416 512L160 512C142.3 512 128 497.7 128 480L128 224C128 206.3 142.3 192 160 192L256 192C273.7 192 288 177.7 288 160C288 142.3 273.7 128 256 128L160 128z"/></svg>
|
||||||
|
After Width: | Height: | Size: 899 B |
24
cockatrice/resources/icons/pin.svg
Normal file
|
|
@ -0,0 +1,24 @@
|
||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 100 100" width="64" height="64">
|
||||||
|
<g transform="matrix(0 1 -1 0 99.465813 0)" opacity="0.7">
|
||||||
|
<path fill="#000000" fill-rule="evenodd" clip-rule="evenodd"
|
||||||
|
stroke="#ffffff"
|
||||||
|
stroke-width="4"
|
||||||
|
stroke-linejoin="round"
|
||||||
|
stroke-linecap="round"
|
||||||
|
d="M65.5 62
|
||||||
|
L78 49
|
||||||
|
C73.5 44.5 69.5 42 63 44
|
||||||
|
L45 31 C47 25 46 22 41.5 18
|
||||||
|
L19 41.5
|
||||||
|
C23 45.5 25 46.5 31 45
|
||||||
|
L44 62.5
|
||||||
|
C42.3 69 45 73.5 49 78
|
||||||
|
L61.5 65.5
|
||||||
|
L84 87
|
||||||
|
L87 87
|
||||||
|
L87.5 86.5
|
||||||
|
L87.5 83.5 Z" />
|
||||||
|
</g>
|
||||||
|
</svg>
|
||||||
|
|
||||||
|
After Width: | Height: | Size: 736 B |
1
cockatrice/resources/icons/scale_balanced.svg
Normal file
|
|
@ -0,0 +1 @@
|
||||||
|
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 640 640"><!--!Font Awesome Free 7.1.0 by @fontawesome - https://fontawesome.com License - https://fontawesome.com/license/free Copyright 2026 Fonticons, Inc.--><path d="M384 96L512 96C529.7 96 544 110.3 544 128C544 145.7 529.7 160 512 160L398.4 160C393.2 185.8 375.5 207.1 352 217.3L352 512L512 512C529.7 512 544 526.3 544 544C544 561.7 529.7 576 512 576L128 576C110.3 576 96 561.7 96 544C96 526.3 110.3 512 128 512L288 512L288 217.3C264.5 207 246.8 185.7 241.6 160L128 160C110.3 160 96 145.7 96 128C96 110.3 110.3 96 128 96L256 96C270.6 76.6 293.8 64 320 64C346.2 64 369.4 76.6 384 96zM439.6 384L584.4 384L512 259.8L439.6 384zM512 480C449.1 480 396.8 446 386 401.1C383.4 390.1 387 378.8 392.7 369L487.9 205.8C492.9 197.2 502.1 192 512 192C521.9 192 531.1 197.3 536.1 205.8L631.3 369C637 378.8 640.6 390.1 638 401.1C627.2 445.9 574.9 480 512 480zM126.8 259.8L54.4 384L199.3 384L126.8 259.8zM.9 401.1C-1.7 390.1 1.9 378.8 7.6 369L102.8 205.8C107.8 197.2 117 192 126.9 192C136.8 192 146 197.3 151 205.8L246.2 369C251.9 378.8 255.5 390.1 252.9 401.1C242.1 445.9 189.8 480 126.9 480C64 480 11.7 446 .9 401.1z"/></svg>
|
||||||
|
After Width: | Height: | Size: 1.1 KiB |
1
cockatrice/resources/icons/scroll.svg
Normal file
|
|
@ -0,0 +1 @@
|
||||||
|
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 640 640"><!--!Font Awesome Free 7.1.0 by @fontawesome - https://fontawesome.com License - https://fontawesome.com/license/free Copyright 2026 Fonticons, Inc.--><path d="M32 176C32 134.5 63.6 100.4 104 96.4L104 96L384 96C437 96 480 139 480 192L480 368L304 368C264.2 368 232 400.2 232 440L232 500C232 524.3 212.3 544 188 544C163.7 544 144 524.3 144 500L144 272L80 272C53.5 272 32 250.5 32 224L32 176zM268.8 544C275.9 530.9 280 515.9 280 500L280 440C280 426.7 290.7 416 304 416L552 416C565.3 416 576 426.7 576 440L576 464C576 508.2 540.2 544 496 544L268.8 544zM112 144C94.3 144 80 158.3 80 176L80 224L144 224L144 176C144 158.3 129.7 144 112 144z"/></svg>
|
||||||
|
After Width: | Height: | Size: 704 B |
25
cockatrice/resources/icons/share.svg
Normal file
|
|
@ -0,0 +1,25 @@
|
||||||
|
<?xml version="1.0" encoding="UTF-8" standalone="no"?>
|
||||||
|
<!-- Uploaded to: SVG Repo, www.svgrepo.com, Generator: SVG Repo Mixer Tools -->
|
||||||
|
|
||||||
|
<svg height="800"
|
||||||
|
width="800"
|
||||||
|
version="1.1"
|
||||||
|
id="_x32_"
|
||||||
|
viewBox="0 0 512 512"
|
||||||
|
xml:space="preserve"
|
||||||
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
|
xmlns:svg="http://www.w3.org/2000/svg">
|
||||||
|
<defs id="defs1"/>
|
||||||
|
<style type="text/css"
|
||||||
|
id="style1">
|
||||||
|
.st0{fill:#64C0FF;stroke:black;stroke-width:3;stroke-miterlimit:4;stroke-opacity:1}
|
||||||
|
</style>
|
||||||
|
<g id="g1"
|
||||||
|
transform="matrix(0.87097097,0,0,1.0008579,38.609049,-0.21963163)"
|
||||||
|
style="stroke-width:3.42738;stroke-dasharray:none">
|
||||||
|
<path class="st0"
|
||||||
|
d="M 512,255.995 277.045,65.394 v 103.574 c -17.255,0 -36.408,0 -57.542,0 -208.59,0 -249.35,153.44 -201.394,266.128 9.586,-103.098 142.053,-100.701 237.358,-100.701 7.247,0 14.446,0 21.578,0 v 112.211 z"
|
||||||
|
id="path1"
|
||||||
|
style="stroke-width:20;stroke:#000000;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none"/>
|
||||||
|
</g>
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 1 KiB |
1
cockatrice/resources/icons/sort_arrow_down.svg
Normal file
|
|
@ -0,0 +1 @@
|
||||||
|
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 640 640"><!--!Font Awesome Free 7.1.0 by @fontawesome - https://fontawesome.com License - https://fontawesome.com/license/free Copyright 2026 Fonticons, Inc.--><path d="M278.6 438.6L182.6 534.6C170.1 547.1 149.8 547.1 137.3 534.6L41.3 438.6C28.8 426.1 28.8 405.8 41.3 393.3C53.8 380.8 74.1 380.8 86.6 393.3L128 434.7L128 128C128 110.3 142.3 96 160 96C177.7 96 192 110.3 192 128L192 434.7L233.4 393.3C245.9 380.8 266.2 380.8 278.7 393.3C291.2 405.8 291.2 426.1 278.7 438.6zM352 544C334.3 544 320 529.7 320 512C320 494.3 334.3 480 352 480L384 480C401.7 480 416 494.3 416 512C416 529.7 401.7 544 384 544L352 544zM352 416C334.3 416 320 401.7 320 384C320 366.3 334.3 352 352 352L448 352C465.7 352 480 366.3 480 384C480 401.7 465.7 416 448 416L352 416zM352 288C334.3 288 320 273.7 320 256C320 238.3 334.3 224 352 224L512 224C529.7 224 544 238.3 544 256C544 273.7 529.7 288 512 288L352 288zM352 160C334.3 160 320 145.7 320 128C320 110.3 334.3 96 352 96L576 96C593.7 96 608 110.3 608 128C608 145.7 593.7 160 576 160L352 160z"/></svg>
|
||||||
|
After Width: | Height: | Size: 1.1 KiB |
|
|
@ -350,11 +350,11 @@
|
||||||
style="fill-opacity:1;stroke:black;stroke-width:2.78220296;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none"
|
style="fill-opacity:1;stroke:black;stroke-width:2.78220296;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none"
|
||||||
d="M 49.84375 1.71875 C 36.719738 1.71875 26.0625 12.375988 26.0625 25.5 C 26.0625 32.977454 29.538325 39.612734 34.9375 43.96875 C 24.439951 49.943698 17.919149 62.196126 14.3125 75.65625 C 9.0380874 95.34065 30.224013 98.21875 49.84375 98.21875 C 69.463486 98.21875 90.549327 94.96715 85.375 75.65625 C 81.693381 61.916246 75.224585 49.827177 64.8125 43.9375 C 70.181573 39.580662 73.59375 32.953205 73.59375 25.5 C 73.59375 12.375988 62.967762 1.71875 49.84375 1.71875 z "
|
d="M 49.84375 1.71875 C 36.719738 1.71875 26.0625 12.375988 26.0625 25.5 C 26.0625 32.977454 29.538325 39.612734 34.9375 43.96875 C 24.439951 49.943698 17.919149 62.196126 14.3125 75.65625 C 9.0380874 95.34065 30.224013 98.21875 49.84375 98.21875 C 69.463486 98.21875 90.549327 94.96715 85.375 75.65625 C 81.693381 61.916246 75.224585 49.827177 64.8125 43.9375 C 70.181573 39.580662 73.59375 32.953205 73.59375 25.5 C 73.59375 12.375988 62.967762 1.71875 49.84375 1.71875 z "
|
||||||
transform="translate(0,952.36218)"
|
transform="translate(0,952.36218)"
|
||||||
id="right" />
|
id="left" />
|
||||||
<path
|
<path
|
||||||
style="opacity:1;fill-opacity:1;fill-rule:nonzero;stroke:#000000;stroke-width:1.73577702;stroke-linecap:butt;stroke-linejoin:round;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1"
|
style="opacity:1;fill-opacity:1;fill-rule:nonzero;stroke:#000000;stroke-width:1.73577702;stroke-linecap:butt;stroke-linejoin:round;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1"
|
||||||
d="m 51.28696,1001.834 0,-46.98372 1.434151,0.16768 c 5.155008,0.60274 9.462857,2.72154 12.938257,6.36366 4.74393,4.9715 6.87913,11.35611 6.16464,18.43328 -0.53702,5.31935 -3.09008,10.59498 -6.83833,14.13074 l -1.94072,1.83069 3.04083,2.20427 c 3.58084,2.5957 7.18975,6.4912 9.55296,10.3116 4.89572,7.9144 9.23593,21.4918 8.50487,26.6055 -0.81312,5.6877 -5.43872,9.6977 -13.62216,11.8093 -3.80822,0.9826 -7.68056,1.4713 -14.763321,1.8633 l -4.471177,0.2474 0,-46.9837 z"
|
d="m 51.28696,1001.834 0,-46.98372 1.434151,0.16768 c 5.155008,0.60274 9.462857,2.72154 12.938257,6.36366 4.74393,4.9715 6.87913,11.35611 6.16464,18.43328 -0.53702,5.31935 -3.09008,10.59498 -6.83833,14.13074 l -1.94072,1.83069 3.04083,2.20427 c 3.58084,2.5957 7.18975,6.4912 9.55296,10.3116 4.89572,7.9144 9.23593,21.4918 8.50487,26.6055 -0.81312,5.6877 -5.43872,9.6977 -13.62216,11.8093 -3.80822,0.9826 -7.68056,1.4713 -14.763321,1.8633 l -4.471177,0.2474 0,-46.9837 z"
|
||||||
id="left"
|
id="right"
|
||||||
inkscape:connector-curvature="0" />
|
inkscape:connector-curvature="0" />
|
||||||
<path
|
<path
|
||||||
style="display:inline;fill:url(#linearGradient3);fill-opacity:1;fill-rule:evenodd;stroke:#000000;stroke-width:3.77952756;stroke-dasharray:none;stroke-opacity:1;paint-order:stroke fill markers"
|
style="display:inline;fill:url(#linearGradient3);fill-opacity:1;fill-rule:evenodd;stroke:#000000;stroke-width:3.77952756;stroke-dasharray:none;stroke-opacity:1;paint-order:stroke fill markers"
|
||||||
|
|
|
||||||
|
Before Width: | Height: | Size: 13 KiB After Width: | Height: | Size: 13 KiB |
|
|
@ -321,11 +321,11 @@
|
||||||
style="fill-opacity:1;stroke:black;stroke-width:2.78220296;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none"
|
style="fill-opacity:1;stroke:black;stroke-width:2.78220296;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none"
|
||||||
d="M 49.84375 1.71875 C 36.719738 1.71875 26.0625 12.375988 26.0625 25.5 C 26.0625 32.977454 29.538325 39.612734 34.9375 43.96875 C 24.439951 49.943698 17.919149 62.196126 14.3125 75.65625 C 9.0380874 95.34065 30.224013 98.21875 49.84375 98.21875 C 69.463486 98.21875 90.549327 94.96715 85.375 75.65625 C 81.693381 61.916246 75.224585 49.827177 64.8125 43.9375 C 70.181573 39.580662 73.59375 32.953205 73.59375 25.5 C 73.59375 12.375988 62.967762 1.71875 49.84375 1.71875 z "
|
d="M 49.84375 1.71875 C 36.719738 1.71875 26.0625 12.375988 26.0625 25.5 C 26.0625 32.977454 29.538325 39.612734 34.9375 43.96875 C 24.439951 49.943698 17.919149 62.196126 14.3125 75.65625 C 9.0380874 95.34065 30.224013 98.21875 49.84375 98.21875 C 69.463486 98.21875 90.549327 94.96715 85.375 75.65625 C 81.693381 61.916246 75.224585 49.827177 64.8125 43.9375 C 70.181573 39.580662 73.59375 32.953205 73.59375 25.5 C 73.59375 12.375988 62.967762 1.71875 49.84375 1.71875 z "
|
||||||
transform="translate(0,952.36218)"
|
transform="translate(0,952.36218)"
|
||||||
id="right" />
|
id="left" />
|
||||||
<path
|
<path
|
||||||
style="opacity:1;fill-opacity:1;fill-rule:nonzero;stroke:#000000;stroke-width:1.73577702;stroke-linecap:butt;stroke-linejoin:round;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1"
|
style="opacity:1;fill-opacity:1;fill-rule:nonzero;stroke:#000000;stroke-width:1.73577702;stroke-linecap:butt;stroke-linejoin:round;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1"
|
||||||
d="m 51.28696,1001.834 0,-46.98372 1.434151,0.16768 c 5.155008,0.60274 9.462857,2.72154 12.938257,6.36366 4.74393,4.9715 6.87913,11.35611 6.16464,18.43328 -0.53702,5.31935 -3.09008,10.59498 -6.83833,14.13074 l -1.94072,1.83069 3.04083,2.20427 c 3.58084,2.5957 7.18975,6.4912 9.55296,10.3116 4.89572,7.9144 9.23593,21.4918 8.50487,26.6055 -0.81312,5.6877 -5.43872,9.6977 -13.62216,11.8093 -3.80822,0.9826 -7.68056,1.4713 -14.763321,1.8633 l -4.471177,0.2474 0,-46.9837 z"
|
d="m 51.28696,1001.834 0,-46.98372 1.434151,0.16768 c 5.155008,0.60274 9.462857,2.72154 12.938257,6.36366 4.74393,4.9715 6.87913,11.35611 6.16464,18.43328 -0.53702,5.31935 -3.09008,10.59498 -6.83833,14.13074 l -1.94072,1.83069 3.04083,2.20427 c 3.58084,2.5957 7.18975,6.4912 9.55296,10.3116 4.89572,7.9144 9.23593,21.4918 8.50487,26.6055 -0.81312,5.6877 -5.43872,9.6977 -13.62216,11.8093 -3.80822,0.9826 -7.68056,1.4713 -14.763321,1.8633 l -4.471177,0.2474 0,-46.9837 z"
|
||||||
id="left"
|
id="right"
|
||||||
inkscape:connector-curvature="0" />
|
inkscape:connector-curvature="0" />
|
||||||
<path
|
<path
|
||||||
d="m 46.656521,12.167234 18.055171,18.054184 a 6.6081919,6.6078288 0 0 1 -0.126303,9.352065 6.6804126,6.6800456 0 0 1 -8.233169,1.011048 l -7.944268,7.943843 6.463762,6.445343 a 6.9331851,6.9328042 0 0 1 5.741536,2.022073 l 28.057729,28.092294 a 6.9962797,6.9958953 0 0 1 -9.894222,9.893685 L 50.719018,66.907526 A 7.0595711,7.0591833 0 0 1 49.18433,59.270613 l -5.741527,-5.741238 -7.944298,7.943843 A 6.716523,6.7161541 0 0 1 25.134866,69.832263 L 7.079684,51.778091 a 6.716523,6.7161541 0 0 1 8.39566,-10.345064 L 36.31101,20.59853 a 6.716523,6.7161541 0 0 1 10.345612,-8.431329 z"
|
d="m 46.656521,12.167234 18.055171,18.054184 a 6.6081919,6.6078288 0 0 1 -0.126303,9.352065 6.6804126,6.6800456 0 0 1 -8.233169,1.011048 l -7.944268,7.943843 6.463762,6.445343 a 6.9331851,6.9328042 0 0 1 5.741536,2.022073 l 28.057729,28.092294 a 6.9962797,6.9958953 0 0 1 -9.894222,9.893685 L 50.719018,66.907526 A 7.0595711,7.0591833 0 0 1 49.18433,59.270613 l -5.741527,-5.741238 -7.944298,7.943843 A 6.716523,6.7161541 0 0 1 25.134866,69.832263 L 7.079684,51.778091 a 6.716523,6.7161541 0 0 1 8.39566,-10.345064 L 36.31101,20.59853 a 6.716523,6.7161541 0 0 1 10.345612,-8.431329 z"
|
||||||
|
|
|
||||||
|
Before Width: | Height: | Size: 12 KiB After Width: | Height: | Size: 12 KiB |
|
|
@ -340,11 +340,11 @@
|
||||||
style="fill-opacity:1;stroke:black;stroke-width:2.78220296;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none"
|
style="fill-opacity:1;stroke:black;stroke-width:2.78220296;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none"
|
||||||
d="M 49.84375 1.71875 C 36.719738 1.71875 26.0625 12.375988 26.0625 25.5 C 26.0625 32.977454 29.538325 39.612734 34.9375 43.96875 C 24.439951 49.943698 17.919149 62.196126 14.3125 75.65625 C 9.0380874 95.34065 30.224013 98.21875 49.84375 98.21875 C 69.463486 98.21875 90.549327 94.96715 85.375 75.65625 C 81.693381 61.916246 75.224585 49.827177 64.8125 43.9375 C 70.181573 39.580662 73.59375 32.953205 73.59375 25.5 C 73.59375 12.375988 62.967762 1.71875 49.84375 1.71875 z "
|
d="M 49.84375 1.71875 C 36.719738 1.71875 26.0625 12.375988 26.0625 25.5 C 26.0625 32.977454 29.538325 39.612734 34.9375 43.96875 C 24.439951 49.943698 17.919149 62.196126 14.3125 75.65625 C 9.0380874 95.34065 30.224013 98.21875 49.84375 98.21875 C 69.463486 98.21875 90.549327 94.96715 85.375 75.65625 C 81.693381 61.916246 75.224585 49.827177 64.8125 43.9375 C 70.181573 39.580662 73.59375 32.953205 73.59375 25.5 C 73.59375 12.375988 62.967762 1.71875 49.84375 1.71875 z "
|
||||||
transform="translate(0,952.36218)"
|
transform="translate(0,952.36218)"
|
||||||
id="right" />
|
id="left" />
|
||||||
<path
|
<path
|
||||||
style="opacity:1;fill-opacity:1;fill-rule:nonzero;stroke:#000000;stroke-width:1.73577702;stroke-linecap:butt;stroke-linejoin:round;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1"
|
style="opacity:1;fill-opacity:1;fill-rule:nonzero;stroke:#000000;stroke-width:1.73577702;stroke-linecap:butt;stroke-linejoin:round;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1"
|
||||||
d="m 51.28696,1001.834 0,-46.98372 1.434151,0.16768 c 5.155008,0.60274 9.462857,2.72154 12.938257,6.36366 4.74393,4.9715 6.87913,11.35611 6.16464,18.43328 -0.53702,5.31935 -3.09008,10.59498 -6.83833,14.13074 l -1.94072,1.83069 3.04083,2.20427 c 3.58084,2.5957 7.18975,6.4912 9.55296,10.3116 4.89572,7.9144 9.23593,21.4918 8.50487,26.6055 -0.81312,5.6877 -5.43872,9.6977 -13.62216,11.8093 -3.80822,0.9826 -7.68056,1.4713 -14.763321,1.8633 l -4.471177,0.2474 0,-46.9837 z"
|
d="m 51.28696,1001.834 0,-46.98372 1.434151,0.16768 c 5.155008,0.60274 9.462857,2.72154 12.938257,6.36366 4.74393,4.9715 6.87913,11.35611 6.16464,18.43328 -0.53702,5.31935 -3.09008,10.59498 -6.83833,14.13074 l -1.94072,1.83069 3.04083,2.20427 c 3.58084,2.5957 7.18975,6.4912 9.55296,10.3116 4.89572,7.9144 9.23593,21.4918 8.50487,26.6055 -0.81312,5.6877 -5.43872,9.6977 -13.62216,11.8093 -3.80822,0.9826 -7.68056,1.4713 -14.763321,1.8633 l -4.471177,0.2474 0,-46.9837 z"
|
||||||
id="left"
|
id="right"
|
||||||
inkscape:connector-curvature="0" />
|
inkscape:connector-curvature="0" />
|
||||||
<path
|
<path
|
||||||
sodipodi:type="star"
|
sodipodi:type="star"
|
||||||
|
|
@ -363,6 +363,6 @@
|
||||||
d="m 38.011063,984.77381 -10.143601,-5.23583 -10.063711,5.38779 1.845023,-11.2651 -8.233948,-7.90624 11.283888,-1.72639 4.974851,-10.27411 5.128803,10.19813 11.308575,1.55649 -8.114112,8.02918 z"
|
d="m 38.011063,984.77381 -10.143601,-5.23583 -10.063711,5.38779 1.845023,-11.2651 -8.233948,-7.90624 11.283888,-1.72639 4.974851,-10.27411 5.128803,10.19813 11.308575,1.55649 -8.114112,8.02918 z"
|
||||||
inkscape:transform-center-x="0.094945927"
|
inkscape:transform-center-x="0.094945927"
|
||||||
inkscape:transform-center-y="-3.9764964"
|
inkscape:transform-center-y="-3.9764964"
|
||||||
transform="matrix(2.3768784,0,0,2.4799382,-15.920285,-1400.1716)" />
|
transform="matrix(-2.3768784,0,0,2.4799382,115.920285,-1400.1716)" />
|
||||||
</g>
|
</g>
|
||||||
</svg>
|
</svg>
|
||||||
|
|
|
||||||
|
Before Width: | Height: | Size: 12 KiB After Width: | Height: | Size: 12 KiB |
|
|
@ -0,0 +1,587 @@
|
||||||
|
#include "remote_connection_controller.h"
|
||||||
|
|
||||||
|
#include "../../settings/cache_settings.h"
|
||||||
|
#include "../interface/widgets/dialogs/dlg_connect.h"
|
||||||
|
#include "../interface/widgets/dialogs/dlg_forgot_password_challenge.h"
|
||||||
|
#include "../interface/widgets/dialogs/dlg_forgot_password_request.h"
|
||||||
|
#include "../interface/widgets/dialogs/dlg_forgot_password_reset.h"
|
||||||
|
#include "../interface/widgets/dialogs/dlg_register.h"
|
||||||
|
#include "../interface/widgets/utility/get_text_with_max.h"
|
||||||
|
|
||||||
|
#include <QDateTime>
|
||||||
|
#include <QLineEdit>
|
||||||
|
#include <QMessageBox>
|
||||||
|
#include <QThread>
|
||||||
|
#include <libcockatrice/network/client/remote/remote_client.h>
|
||||||
|
#include <libcockatrice/protocol/pb/response.pb.h>
|
||||||
|
|
||||||
|
ConnectionController::ConnectionController(QWidget *dialogParent, QObject *parent)
|
||||||
|
: QObject(parent), dialogParent(dialogParent)
|
||||||
|
{
|
||||||
|
remoteClient = new RemoteClient(nullptr, &SettingsCache::instance());
|
||||||
|
|
||||||
|
clientThread = new QThread(this);
|
||||||
|
remoteClient->moveToThread(clientThread);
|
||||||
|
clientThread->start();
|
||||||
|
|
||||||
|
wireClientSignals();
|
||||||
|
}
|
||||||
|
|
||||||
|
ConnectionController::~ConnectionController()
|
||||||
|
{
|
||||||
|
remoteClient->deleteLater();
|
||||||
|
clientThread->wait();
|
||||||
|
}
|
||||||
|
|
||||||
|
void ConnectionController::wireClientSignals()
|
||||||
|
{
|
||||||
|
connect(remoteClient, &RemoteClient::connectionClosedEventReceived, this,
|
||||||
|
&ConnectionController::onConnectionClosedEvent);
|
||||||
|
|
||||||
|
connect(remoteClient, &RemoteClient::serverShutdownEventReceived, this,
|
||||||
|
&ConnectionController::onServerShutdownEvent);
|
||||||
|
|
||||||
|
connect(remoteClient, &RemoteClient::statusChanged, this, &ConnectionController::onStatusChanged);
|
||||||
|
|
||||||
|
connect(remoteClient, &RemoteClient::userInfoChanged, this, &ConnectionController::onUserInfoReceived,
|
||||||
|
Qt::BlockingQueuedConnection);
|
||||||
|
|
||||||
|
connect(remoteClient, &RemoteClient::loginError, this,
|
||||||
|
[this](Response::ResponseCode r, QString rs, quint32 et, QList<QString> mf) {
|
||||||
|
onLoginError(static_cast<int>(r), rs, et, mf);
|
||||||
|
});
|
||||||
|
|
||||||
|
connect(remoteClient, &RemoteClient::registerError, this,
|
||||||
|
[this](Response::ResponseCode r, QString rs, quint32 et) { onRegisterError(static_cast<int>(r), rs, et); });
|
||||||
|
|
||||||
|
connect(remoteClient, &RemoteClient::activateError, this, &ConnectionController::onActivateError);
|
||||||
|
connect(remoteClient, &RemoteClient::socketError, this, &ConnectionController::onSocketError);
|
||||||
|
connect(remoteClient, &RemoteClient::serverTimeout, this, &ConnectionController::onServerTimeout);
|
||||||
|
|
||||||
|
connect(remoteClient, &RemoteClient::protocolVersionMismatch, this,
|
||||||
|
&ConnectionController::onProtocolVersionMismatch);
|
||||||
|
|
||||||
|
connect(remoteClient, &RemoteClient::registerAccepted, this, &ConnectionController::onRegisterAccepted);
|
||||||
|
|
||||||
|
connect(remoteClient, &RemoteClient::registerAcceptedNeedsActivate, this,
|
||||||
|
&ConnectionController::onRegisterAcceptedNeedsActivate);
|
||||||
|
|
||||||
|
connect(remoteClient, &RemoteClient::activateAccepted, this, &ConnectionController::onActivateAccepted);
|
||||||
|
|
||||||
|
connect(remoteClient, &RemoteClient::notifyUserAboutUpdate, this, &ConnectionController::onNotifyUserAboutUpdate);
|
||||||
|
|
||||||
|
connect(remoteClient, &RemoteClient::sigForgotPasswordSuccess, this,
|
||||||
|
&ConnectionController::onForgotPasswordSuccess);
|
||||||
|
|
||||||
|
connect(remoteClient, &RemoteClient::sigForgotPasswordError, this, &ConnectionController::onForgotPasswordError);
|
||||||
|
|
||||||
|
connect(remoteClient, &RemoteClient::sigPromptForForgotPasswordReset, this,
|
||||||
|
&ConnectionController::onPromptForgotPasswordReset);
|
||||||
|
|
||||||
|
connect(remoteClient, &RemoteClient::sigPromptForForgotPasswordChallenge, this,
|
||||||
|
&ConnectionController::onPromptForgotPasswordChallenge);
|
||||||
|
}
|
||||||
|
|
||||||
|
void ConnectionController::connectToServer()
|
||||||
|
{
|
||||||
|
dlgConnect = new DlgConnect(dialogParent);
|
||||||
|
connect(dlgConnect, &DlgConnect::sigStartForgotPasswordRequest, this, &ConnectionController::forgotPasswordRequest);
|
||||||
|
|
||||||
|
if (dlgConnect->exec()) {
|
||||||
|
remoteClient->connectToServer(dlgConnect->getHost(), static_cast<unsigned int>(dlgConnect->getPort()),
|
||||||
|
dlgConnect->getPlayerName(), dlgConnect->getPassword());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void ConnectionController::connectToServerDirect(const QString &host,
|
||||||
|
unsigned int port,
|
||||||
|
const QString &playerName,
|
||||||
|
const QString &password)
|
||||||
|
{
|
||||||
|
remoteClient->connectToServer(host, port, playerName, password);
|
||||||
|
}
|
||||||
|
|
||||||
|
void ConnectionController::disconnectFromServer()
|
||||||
|
{
|
||||||
|
remoteClient->disconnectFromServer();
|
||||||
|
}
|
||||||
|
|
||||||
|
void ConnectionController::registerToServer()
|
||||||
|
{
|
||||||
|
DlgRegister dlg(dialogParent);
|
||||||
|
if (dlg.exec()) {
|
||||||
|
remoteClient->registerToServer(dlg.getHost(), static_cast<unsigned int>(dlg.getPort()), dlg.getPlayerName(),
|
||||||
|
dlg.getPassword(), dlg.getEmail(), dlg.getCountry(), dlg.getRealName());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void ConnectionController::forgotPasswordRequest()
|
||||||
|
{
|
||||||
|
DlgForgotPasswordRequest dlg(dialogParent);
|
||||||
|
if (dlg.exec()) {
|
||||||
|
remoteClient->requestForgotPasswordToServer(dlg.getHost(), static_cast<unsigned int>(dlg.getPort()),
|
||||||
|
dlg.getPlayerName());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void ConnectionController::onConnectionClosedEvent(const Event_ConnectionClosed &event)
|
||||||
|
{
|
||||||
|
remoteClient->disconnectFromServer();
|
||||||
|
|
||||||
|
QString reasonStr;
|
||||||
|
switch (event.reason()) {
|
||||||
|
case Event_ConnectionClosed::USER_LIMIT_REACHED: {
|
||||||
|
reasonStr = tr("The server has reached its maximum user capacity, please check back later.");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case Event_ConnectionClosed::TOO_MANY_CONNECTIONS: {
|
||||||
|
reasonStr = tr("There are too many concurrent connections from your address.");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case Event_ConnectionClosed::BANNED: {
|
||||||
|
reasonStr = tr("Banned by moderator");
|
||||||
|
if (event.has_end_time()) {
|
||||||
|
reasonStr.append(
|
||||||
|
"\n" + tr("Expected end time: %1").arg(QDateTime::fromSecsSinceEpoch(event.end_time()).toString()));
|
||||||
|
} else {
|
||||||
|
reasonStr.append("\n" + tr("This ban lasts indefinitely."));
|
||||||
|
}
|
||||||
|
if (event.has_reason_str()) {
|
||||||
|
reasonStr.append("\n\n" + QString::fromStdString(event.reason_str()));
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case Event_ConnectionClosed::SERVER_SHUTDOWN: {
|
||||||
|
reasonStr = tr("Scheduled server shutdown.");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case Event_ConnectionClosed::USERNAMEINVALID: {
|
||||||
|
reasonStr = tr("Invalid username.");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case Event_ConnectionClosed::LOGGEDINELSEWERE: {
|
||||||
|
reasonStr = tr("You have been logged out due to logging in at another location.");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
reasonStr = QString::fromStdString(event.reason_str());
|
||||||
|
}
|
||||||
|
|
||||||
|
QMessageBox::critical(dialogParent, tr("Connection closed"),
|
||||||
|
tr("The server has terminated your connection.\nReason: %1").arg(reasonStr));
|
||||||
|
}
|
||||||
|
|
||||||
|
void ConnectionController::onServerShutdownEvent(const Event_ServerShutdown &event)
|
||||||
|
{
|
||||||
|
serverShutdownMessageBox.setInformativeText(tr("The server is going to be restarted in %n minute(s).\nAll running "
|
||||||
|
"games will be lost.\nReason for shutdown: %1",
|
||||||
|
"", event.minutes())
|
||||||
|
.arg(QString::fromStdString(event.reason())));
|
||||||
|
serverShutdownMessageBox.setIconPixmap(QPixmap("theme:cockatrice").scaled(64, 64));
|
||||||
|
serverShutdownMessageBox.setText(tr("Scheduled server shutdown"));
|
||||||
|
serverShutdownMessageBox.setWindowModality(Qt::ApplicationModal);
|
||||||
|
serverShutdownMessageBox.setVisible(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
void ConnectionController::onStatusChanged(ClientStatus status)
|
||||||
|
{
|
||||||
|
// Update the window title first, then let MainWindow handle its own UI
|
||||||
|
// state via the forwarded signal
|
||||||
|
updateWindowTitle();
|
||||||
|
emit statusChanged(status);
|
||||||
|
|
||||||
|
// TabSupervisor::stop() needs calling on disconnect; start() is driven by
|
||||||
|
// onUserInfoReceived → tabSupervisorStartRequested.
|
||||||
|
if (status == StatusDisconnected) {
|
||||||
|
emit tabSupervisorStopRequested();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void ConnectionController::onUserInfoReceived(const ServerInfo_User &info)
|
||||||
|
{
|
||||||
|
emit tabSupervisorStartRequested(info);
|
||||||
|
}
|
||||||
|
|
||||||
|
void ConnectionController::onLoginError(int r,
|
||||||
|
QString reasonStr,
|
||||||
|
quint32 endTime,
|
||||||
|
const QList<QString> &missingFeatures)
|
||||||
|
{
|
||||||
|
switch (static_cast<Response::ResponseCode>(r)) {
|
||||||
|
case Response::RespClientUpdateRequired: {
|
||||||
|
QString formatted = "Missing Features: ";
|
||||||
|
for (int i = 0; i < missingFeatures.size(); ++i) {
|
||||||
|
formatted.append(QString("\n %1").arg(QChar(0x2022)) + " " + missingFeatures.value(i));
|
||||||
|
}
|
||||||
|
|
||||||
|
QMessageBox msgBox(dialogParent);
|
||||||
|
msgBox.setIcon(QMessageBox::Critical);
|
||||||
|
msgBox.setWindowTitle(tr("Failed Login"));
|
||||||
|
msgBox.setText(tr("Your client seems to be missing features this server requires for connection.") +
|
||||||
|
"\n\n" + tr("To update your client, go to 'Help -> Check for Client Updates'."));
|
||||||
|
msgBox.setDetailedText(formatted);
|
||||||
|
msgBox.exec();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
case Response::RespWrongPassword: {
|
||||||
|
QMessageBox::critical(dialogParent, tr("Error"),
|
||||||
|
tr("Incorrect username or password. "
|
||||||
|
"Please check your authentication information and try again."));
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
case Response::RespWouldOverwriteOldSession: {
|
||||||
|
QMessageBox::critical(dialogParent, tr("Error"),
|
||||||
|
tr("There is already an active session using this user name.\n"
|
||||||
|
"Please close that session first and re-login."));
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
case Response::RespUserIsBanned: {
|
||||||
|
QString bannedStr =
|
||||||
|
endTime ? tr("You are banned until %1.").arg(QDateTime::fromSecsSinceEpoch(endTime).toString())
|
||||||
|
: tr("You are banned indefinitely.");
|
||||||
|
if (!reasonStr.isEmpty()) {
|
||||||
|
bannedStr.append("\n\n" + reasonStr);
|
||||||
|
}
|
||||||
|
QMessageBox::critical(dialogParent, tr("Error"), bannedStr);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
case Response::RespUsernameInvalid: {
|
||||||
|
QMessageBox::critical(dialogParent, tr("Error"), extractInvalidUsernameMessage(reasonStr));
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
case Response::RespRegistrationRequired: {
|
||||||
|
if (QMessageBox::question(dialogParent, tr("Error"),
|
||||||
|
tr("This server requires user registration. Do you want to register now?"),
|
||||||
|
QMessageBox::Yes | QMessageBox::No) == QMessageBox::Yes) {
|
||||||
|
registerToServer();
|
||||||
|
}
|
||||||
|
return; // don't re-prompt connect
|
||||||
|
}
|
||||||
|
|
||||||
|
case Response::RespClientIdRequired: {
|
||||||
|
QMessageBox::critical(dialogParent, tr("Error"),
|
||||||
|
tr("This server requires client IDs. Your client is either failing to generate an "
|
||||||
|
"ID or you are running a modified client.\n"
|
||||||
|
"Please close and reopen your client to try again."));
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
case Response::RespContextError: {
|
||||||
|
QMessageBox::critical(dialogParent, tr("Error"),
|
||||||
|
tr("An internal error has occurred, please close and reopen Cockatrice before "
|
||||||
|
"trying again.\nIf the error persists, ensure you are running the latest "
|
||||||
|
"version of the software and if needed contact the software developers."));
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
case Response::RespAccountNotActivated: {
|
||||||
|
bool ok = false;
|
||||||
|
QString token =
|
||||||
|
getTextWithMax(dialogParent, tr("Account activation"),
|
||||||
|
tr("Your account has not been activated yet.\n"
|
||||||
|
"You need to provide the activation token received in the activation email."),
|
||||||
|
QLineEdit::Normal, QString(), &ok);
|
||||||
|
|
||||||
|
if (ok && !token.isEmpty()) {
|
||||||
|
remoteClient->activateToServer(token);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
remoteClient->disconnectFromServer();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
case Response::RespServerFull: {
|
||||||
|
QMessageBox::critical(dialogParent, tr("Server Full"),
|
||||||
|
tr("The server has reached its maximum user capacity, please check back later."));
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
default: {
|
||||||
|
QMessageBox::critical(dialogParent, tr("Error"),
|
||||||
|
tr("Unknown login error: %1").arg(r) +
|
||||||
|
tr("\nThis usually means that your client version is out of date, and the server "
|
||||||
|
"sent a reply your client doesn't understand."));
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Re-open the connect dialog after any handled error
|
||||||
|
connectToServer();
|
||||||
|
}
|
||||||
|
|
||||||
|
void ConnectionController::onRegisterError(int r, QString reasonStr, quint32 endTime)
|
||||||
|
{
|
||||||
|
switch (static_cast<Response::ResponseCode>(r)) {
|
||||||
|
case Response::RespRegistrationDisabled: {
|
||||||
|
QMessageBox::critical(dialogParent, tr("Registration denied"),
|
||||||
|
tr("Registration is currently disabled on this server"));
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case Response::RespUserAlreadyExists: {
|
||||||
|
QMessageBox::critical(dialogParent, tr("Registration denied"),
|
||||||
|
tr("There is already an existing account with the same user name."));
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case Response::RespEmailRequiredToRegister: {
|
||||||
|
QMessageBox::critical(dialogParent, tr("Registration denied"),
|
||||||
|
tr("It's mandatory to specify a valid email address when registering."));
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case Response::RespEmailBlackListed: {
|
||||||
|
if (reasonStr.isEmpty()) {
|
||||||
|
reasonStr =
|
||||||
|
"The email address provider used during registration has been blocked from use on this server.";
|
||||||
|
}
|
||||||
|
QMessageBox::critical(dialogParent, tr("Registration denied"), reasonStr);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case Response::RespTooManyRequests: {
|
||||||
|
QMessageBox::critical(dialogParent, tr("Registration denied"),
|
||||||
|
tr("It appears you are attempting to register a new account on this server yet you "
|
||||||
|
"already have an account registered with the email provided. This server "
|
||||||
|
"restricts the number of accounts a user can register per address. Please "
|
||||||
|
"contact the server operator for further assistance or to obtain your "
|
||||||
|
"credential information."));
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case Response::RespPasswordTooShort: {
|
||||||
|
QMessageBox::critical(dialogParent, tr("Registration denied"), tr("Password too short."));
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case Response::RespUserIsBanned: {
|
||||||
|
QString bannedStr =
|
||||||
|
endTime ? tr("You are banned until %1.").arg(QDateTime::fromSecsSinceEpoch(endTime).toString())
|
||||||
|
: tr("You are banned indefinitely.");
|
||||||
|
if (!reasonStr.isEmpty()) {
|
||||||
|
bannedStr.append("\n\n" + reasonStr);
|
||||||
|
}
|
||||||
|
QMessageBox::critical(dialogParent, tr("Error"), bannedStr);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case Response::RespUsernameInvalid: {
|
||||||
|
QMessageBox::critical(dialogParent, tr("Error"), extractInvalidUsernameMessage(reasonStr));
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case Response::RespRegistrationFailed: {
|
||||||
|
QMessageBox::critical(dialogParent, tr("Error"),
|
||||||
|
tr("Registration failed for a technical problem on the server."));
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case Response::RespNotConnected: {
|
||||||
|
QMessageBox::critical(dialogParent, tr("Error"), tr("The connection to the server has been lost."));
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
default: {
|
||||||
|
QMessageBox::critical(dialogParent, tr("Error"),
|
||||||
|
tr("Unknown registration error: %1").arg(r) +
|
||||||
|
tr("\nThis usually means that your client version is out of date, and the server "
|
||||||
|
"sent a reply your client doesn't understand."));
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
registerToServer();
|
||||||
|
}
|
||||||
|
|
||||||
|
void ConnectionController::onActivateError()
|
||||||
|
{
|
||||||
|
QMessageBox::critical(dialogParent, tr("Error"), tr("Account activation failed"));
|
||||||
|
remoteClient->disconnectFromServer();
|
||||||
|
connectToServer();
|
||||||
|
}
|
||||||
|
|
||||||
|
void ConnectionController::onSocketError(const QString &errorStr)
|
||||||
|
{
|
||||||
|
QMessageBox::critical(dialogParent, tr("Error"), tr("Socket error: %1").arg(errorStr));
|
||||||
|
connectToServer();
|
||||||
|
}
|
||||||
|
|
||||||
|
void ConnectionController::onServerTimeout()
|
||||||
|
{
|
||||||
|
QMessageBox::critical(dialogParent, tr("Error"), tr("Server timeout"));
|
||||||
|
connectToServer();
|
||||||
|
}
|
||||||
|
|
||||||
|
void ConnectionController::onProtocolVersionMismatch(int localVersion, int remoteVersion)
|
||||||
|
{
|
||||||
|
if (localVersion > remoteVersion) {
|
||||||
|
QMessageBox::critical(dialogParent, tr("Error"),
|
||||||
|
tr("You are trying to connect to an obsolete server. Please downgrade your Cockatrice "
|
||||||
|
"version or connect to a suitable server.\n"
|
||||||
|
"Local version is %1, remote version is %2.")
|
||||||
|
.arg(localVersion)
|
||||||
|
.arg(remoteVersion));
|
||||||
|
} else {
|
||||||
|
QMessageBox::critical(dialogParent, tr("Error"),
|
||||||
|
tr("Your Cockatrice client is obsolete. Please update your Cockatrice version.\n"
|
||||||
|
"Local version is %1, remote version is %2.")
|
||||||
|
.arg(localVersion)
|
||||||
|
.arg(remoteVersion));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void ConnectionController::onRegisterAccepted()
|
||||||
|
{
|
||||||
|
QMessageBox::information(dialogParent, tr("Success"), tr("Registration accepted.\nWill now login."));
|
||||||
|
}
|
||||||
|
|
||||||
|
void ConnectionController::onRegisterAcceptedNeedsActivate()
|
||||||
|
{
|
||||||
|
// Server will send activation email; nothing to display here.
|
||||||
|
}
|
||||||
|
|
||||||
|
void ConnectionController::onActivateAccepted()
|
||||||
|
{
|
||||||
|
QMessageBox::information(dialogParent, tr("Success"), tr("Account activation accepted.\nWill now login."));
|
||||||
|
}
|
||||||
|
|
||||||
|
void ConnectionController::onNotifyUserAboutUpdate()
|
||||||
|
{
|
||||||
|
QMessageBox::information(
|
||||||
|
dialogParent, tr("Information"),
|
||||||
|
tr("This server supports additional features that your client doesn't have.\n"
|
||||||
|
"This is most likely not a problem, but this message might mean there is a new version of "
|
||||||
|
"Cockatrice available or this server is running a custom or pre-release version.\n\n"
|
||||||
|
"To update your client, go to Help -> Check for Updates."));
|
||||||
|
}
|
||||||
|
|
||||||
|
void ConnectionController::onForgotPasswordSuccess()
|
||||||
|
{
|
||||||
|
QMessageBox::information(
|
||||||
|
dialogParent, tr("Reset Password"),
|
||||||
|
tr("Your password has been reset successfully, you can now log in using the new credentials."));
|
||||||
|
SettingsCache::instance().servers().setFPHostName("");
|
||||||
|
SettingsCache::instance().servers().setFPPort("");
|
||||||
|
SettingsCache::instance().servers().setFPPlayerName("");
|
||||||
|
}
|
||||||
|
|
||||||
|
void ConnectionController::onForgotPasswordError()
|
||||||
|
{
|
||||||
|
QMessageBox::warning(
|
||||||
|
dialogParent, tr("Reset Password"),
|
||||||
|
tr("Failed to reset user account password, please contact the server operator to reset your password."));
|
||||||
|
SettingsCache::instance().servers().setFPHostName("");
|
||||||
|
SettingsCache::instance().servers().setFPPort("");
|
||||||
|
SettingsCache::instance().servers().setFPPlayerName("");
|
||||||
|
}
|
||||||
|
|
||||||
|
void ConnectionController::onPromptForgotPasswordReset()
|
||||||
|
{
|
||||||
|
QMessageBox::information(dialogParent, tr("Reset Password"),
|
||||||
|
tr("Activation request received, please check your email for an activation token."));
|
||||||
|
DlgForgotPasswordReset dlg(dialogParent);
|
||||||
|
if (dlg.exec()) {
|
||||||
|
remoteClient->submitForgotPasswordResetToServer(dlg.getHost(), static_cast<unsigned int>(dlg.getPort()),
|
||||||
|
dlg.getPlayerName(), dlg.getToken(), dlg.getPassword());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void ConnectionController::onPromptForgotPasswordChallenge()
|
||||||
|
{
|
||||||
|
DlgForgotPasswordChallenge dlg(dialogParent);
|
||||||
|
if (dlg.exec()) {
|
||||||
|
remoteClient->submitForgotPasswordChallengeToServer(dlg.getHost(), static_cast<unsigned int>(dlg.getPort()),
|
||||||
|
dlg.getPlayerName(), dlg.getEmail());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void ConnectionController::updateWindowTitle()
|
||||||
|
{
|
||||||
|
const QString appName = QStringLiteral("Cockatrice");
|
||||||
|
QString title;
|
||||||
|
|
||||||
|
switch (remoteClient->getStatus()) {
|
||||||
|
case StatusConnecting: {
|
||||||
|
title = appName + " - " + tr("Connecting to %1...").arg(remoteClient->peerName());
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case StatusRegistering: {
|
||||||
|
title = appName + " - " +
|
||||||
|
tr("Registering to %1 as %2...").arg(remoteClient->peerName()).arg(remoteClient->getUserName());
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case StatusDisconnected: {
|
||||||
|
title = appName + " - " + tr("Disconnected");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case StatusLoggingIn: {
|
||||||
|
title = appName + " - " + tr("Connected, logging in at %1").arg(remoteClient->peerName());
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case StatusLoggedIn: {
|
||||||
|
title = remoteClient->getUserName() + "@" + remoteClient->peerName();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case StatusRequestingForgotPassword:
|
||||||
|
case StatusSubmitForgotPasswordChallenge:
|
||||||
|
case StatusSubmitForgotPasswordReset:
|
||||||
|
title = appName + " - " +
|
||||||
|
tr("Requesting forgotten password to %1 as %2...")
|
||||||
|
.arg(remoteClient->peerName())
|
||||||
|
.arg(remoteClient->getUserName());
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
title = appName;
|
||||||
|
}
|
||||||
|
|
||||||
|
emit windowTitleChanged(title);
|
||||||
|
}
|
||||||
|
|
||||||
|
// static
|
||||||
|
QString ConnectionController::extractInvalidUsernameMessage(QString &in)
|
||||||
|
{
|
||||||
|
QString out = tr("Invalid username.") + "<br/>";
|
||||||
|
QStringList rules = in.split(QChar('|'));
|
||||||
|
|
||||||
|
if (rules.size() == 7 || rules.size() == 9) {
|
||||||
|
out += tr("Your username must respect these rules:") + "<ul>";
|
||||||
|
|
||||||
|
out += "<li>" + tr("is %1 - %2 characters long").arg(rules.at(0)).arg(rules.at(1)) + "</li>";
|
||||||
|
out += "<li>" + tr("can %1 contain lowercase characters").arg((rules.at(2).toInt() > 0) ? "" : tr("NOT")) +
|
||||||
|
"</li>";
|
||||||
|
out += "<li>" + tr("can %1 contain uppercase characters").arg((rules.at(3).toInt() > 0) ? "" : tr("NOT")) +
|
||||||
|
"</li>";
|
||||||
|
out +=
|
||||||
|
"<li>" + tr("can %1 contain numeric characters").arg((rules.at(4).toInt() > 0) ? "" : tr("NOT")) + "</li>";
|
||||||
|
|
||||||
|
if (rules.at(6).size() > 0) {
|
||||||
|
out += "<li>" + tr("can contain the following punctuation: %1").arg(rules.at(6).toHtmlEscaped()) + "</li>";
|
||||||
|
}
|
||||||
|
|
||||||
|
out += "<li>" +
|
||||||
|
tr("first character can %1 be a punctuation mark").arg((rules.at(5).toInt() > 0) ? "" : tr("NOT")) +
|
||||||
|
"</li>";
|
||||||
|
|
||||||
|
if (rules.size() == 9) {
|
||||||
|
if (rules.at(7).size() > 0) {
|
||||||
|
QString words = rules.at(7).toHtmlEscaped();
|
||||||
|
if (words.startsWith("\n")) {
|
||||||
|
out += tr("no unacceptable language as specified by these server rules:",
|
||||||
|
"note that the following lines will not be translated");
|
||||||
|
for (QString &line : words.split("\n", Qt::SkipEmptyParts)) {
|
||||||
|
out += "<li>" + line + "</li>";
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
out += "<li>" + tr("can not contain any of the following words: %1").arg(words) + "</li>";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (rules.at(8).size() > 0) {
|
||||||
|
out += "<li>" +
|
||||||
|
tr("can not match any of the following expressions: %1").arg(rules.at(8).toHtmlEscaped()) +
|
||||||
|
"</li>";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
out += "</ul>";
|
||||||
|
} else {
|
||||||
|
out += tr("You may only use A-Z, a-z, 0-9, _, ., and - in your username.");
|
||||||
|
}
|
||||||
|
|
||||||
|
return out;
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,98 @@
|
||||||
|
#ifndef COCKATRICE_REMOTE_CONNECTION_CONTROLLER_H
|
||||||
|
#define COCKATRICE_REMOTE_CONNECTION_CONTROLLER_H
|
||||||
|
|
||||||
|
#include "abstract_client.h"
|
||||||
|
|
||||||
|
#include <QMessageBox>
|
||||||
|
#include <QObject>
|
||||||
|
#include <QString>
|
||||||
|
#include <QThread>
|
||||||
|
#include <libcockatrice/protocol/pb/event_connection_closed.pb.h>
|
||||||
|
#include <libcockatrice/protocol/pb/event_server_shutdown.pb.h>
|
||||||
|
|
||||||
|
class RemoteClient;
|
||||||
|
class ServerInfo_User;
|
||||||
|
class DlgConnect;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Owns the RemoteClient and its worker thread.
|
||||||
|
* Encapsulates all connection, authentication, and registration logic so that
|
||||||
|
* MainWindow only needs to react to high-level signals.
|
||||||
|
*/
|
||||||
|
class ConnectionController : public QObject
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
|
||||||
|
public:
|
||||||
|
explicit ConnectionController(QWidget *dialogParent, QObject *parent = nullptr);
|
||||||
|
~ConnectionController() override;
|
||||||
|
|
||||||
|
RemoteClient *client() const
|
||||||
|
{
|
||||||
|
return remoteClient;
|
||||||
|
}
|
||||||
|
|
||||||
|
void registerToServer();
|
||||||
|
void forgotPasswordRequest();
|
||||||
|
void connectToServer();
|
||||||
|
void
|
||||||
|
connectToServerDirect(const QString &host, unsigned int port, const QString &playerName, const QString &password);
|
||||||
|
void disconnectFromServer();
|
||||||
|
|
||||||
|
void refreshWindowTitle()
|
||||||
|
{
|
||||||
|
updateWindowTitle();
|
||||||
|
}
|
||||||
|
|
||||||
|
signals:
|
||||||
|
void windowTitleChanged(const QString &title);
|
||||||
|
|
||||||
|
void tabSupervisorStartRequested(const ServerInfo_User &info);
|
||||||
|
void tabSupervisorStopRequested();
|
||||||
|
|
||||||
|
// Passes the raw ClientStatus through so MainWindow can drive its own
|
||||||
|
// action enable/disable logic
|
||||||
|
void statusChanged(ClientStatus status);
|
||||||
|
|
||||||
|
private slots:
|
||||||
|
// Slots wired directly to RemoteClient signals
|
||||||
|
void onStatusChanged(ClientStatus status);
|
||||||
|
void onUserInfoReceived(const ServerInfo_User &info);
|
||||||
|
void onLoginError(int r, QString reasonStr, quint32 endTime, const QList<QString> &missingFeatures);
|
||||||
|
void onRegisterAccepted();
|
||||||
|
void onRegisterAcceptedNeedsActivate();
|
||||||
|
void onRegisterError(int r, QString reasonStr, quint32 endTime);
|
||||||
|
void onActivateAccepted();
|
||||||
|
void onActivateError();
|
||||||
|
void onProtocolVersionMismatch(int localVersion, int remoteVersion);
|
||||||
|
void onNotifyUserAboutUpdate();
|
||||||
|
void onConnectionClosedEvent(const Event_ConnectionClosed &event);
|
||||||
|
void onServerShutdownEvent(const Event_ServerShutdown &event);
|
||||||
|
void onSocketError(const QString &errorStr);
|
||||||
|
void onServerTimeout();
|
||||||
|
|
||||||
|
// Forgot-password flow
|
||||||
|
void onForgotPasswordSuccess();
|
||||||
|
void onForgotPasswordError();
|
||||||
|
void onPromptForgotPasswordReset();
|
||||||
|
void onPromptForgotPasswordChallenge();
|
||||||
|
|
||||||
|
private:
|
||||||
|
void wireClientSignals();
|
||||||
|
void updateWindowTitle();
|
||||||
|
|
||||||
|
/** Parse the server's pipe-delimited username-rule string into HTML. */
|
||||||
|
static QString extractInvalidUsernameMessage(QString &in);
|
||||||
|
|
||||||
|
RemoteClient *remoteClient{nullptr};
|
||||||
|
QThread *clientThread{nullptr};
|
||||||
|
QWidget *dialogParent{nullptr}; // used as parent for QMessageBox / dialog calls
|
||||||
|
|
||||||
|
// Persistent so it can be updated in-place by onServerShutdownEvent
|
||||||
|
QMessageBox serverShutdownMessageBox;
|
||||||
|
|
||||||
|
// Kept as a member so the forgot-password signal can be wired to it
|
||||||
|
DlgConnect *dlgConnect{nullptr};
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif // COCKATRICE_REMOTE_CONNECTION_CONTROLLER_H
|
||||||
|
|
@ -1,17 +1,17 @@
|
||||||
#include "deck_stats_interface.h"
|
#include "deck_stats_interface.h"
|
||||||
|
|
||||||
#include "decklist.h"
|
|
||||||
|
|
||||||
#include <QDesktopServices>
|
#include <QDesktopServices>
|
||||||
#include <QMessageBox>
|
#include <QMessageBox>
|
||||||
#include <QNetworkAccessManager>
|
|
||||||
#include <QNetworkReply>
|
#include <QNetworkReply>
|
||||||
#include <QNetworkRequest>
|
#include <QNetworkRequest>
|
||||||
#include <QRegularExpression>
|
#include <QRegularExpression>
|
||||||
#include <QUrlQuery>
|
#include <QUrlQuery>
|
||||||
|
#include <libcockatrice/card/database/card_database_manager.h>
|
||||||
|
#include <libcockatrice/deck_list/deck_list.h>
|
||||||
|
#include <libcockatrice/deck_list/tree/deck_list_card_node.h>
|
||||||
|
#include <version_string.h>
|
||||||
|
|
||||||
DeckStatsInterface::DeckStatsInterface(CardDatabase &_cardDatabase, QObject *parent)
|
DeckStatsInterface::DeckStatsInterface(QObject *parent) : QObject(parent)
|
||||||
: QObject(parent), cardDatabase(_cardDatabase)
|
|
||||||
{
|
{
|
||||||
manager = new QNetworkAccessManager(this);
|
manager = new QNetworkAccessManager(this);
|
||||||
connect(manager, &QNetworkAccessManager::finished, this, &DeckStatsInterface::queryFinished);
|
connect(manager, &QNetworkAccessManager::finished, this, &DeckStatsInterface::queryFinished);
|
||||||
|
|
@ -43,36 +43,37 @@ void DeckStatsInterface::queryFinished(QNetworkReply *reply)
|
||||||
deleteLater();
|
deleteLater();
|
||||||
}
|
}
|
||||||
|
|
||||||
void DeckStatsInterface::getAnalyzeRequestData(DeckList *deck, QByteArray *data)
|
void DeckStatsInterface::getAnalyzeRequestData(const DeckList &deck, QByteArray &data)
|
||||||
{
|
{
|
||||||
DeckList deckWithoutTokens;
|
DeckList deckWithoutTokens;
|
||||||
copyDeckWithoutTokens(*deck, deckWithoutTokens);
|
copyDeckWithoutTokens(deck, deckWithoutTokens);
|
||||||
|
|
||||||
QUrl params;
|
QUrl params;
|
||||||
QUrlQuery urlQuery;
|
QUrlQuery urlQuery;
|
||||||
urlQuery.addQueryItem("deck", deckWithoutTokens.writeToString_Plain());
|
urlQuery.addQueryItem("deck", deckWithoutTokens.writeToString_Plain());
|
||||||
urlQuery.addQueryItem("decktitle", deck->getName());
|
urlQuery.addQueryItem("decktitle", deck.getName());
|
||||||
params.setQuery(urlQuery);
|
params.setQuery(urlQuery);
|
||||||
data->append(params.query(QUrl::EncodeReserved).toUtf8());
|
data.append(params.query(QUrl::EncodeReserved).toUtf8());
|
||||||
}
|
}
|
||||||
|
|
||||||
void DeckStatsInterface::analyzeDeck(DeckList *deck)
|
void DeckStatsInterface::analyzeDeck(const DeckList &deck)
|
||||||
{
|
{
|
||||||
QByteArray data;
|
QByteArray data;
|
||||||
getAnalyzeRequestData(deck, &data);
|
getAnalyzeRequestData(deck, data);
|
||||||
|
|
||||||
QNetworkRequest request(QUrl("https://deckstats.net/index.php"));
|
QNetworkRequest request(QUrl("https://deckstats.net/index.php"));
|
||||||
request.setHeader(QNetworkRequest::ContentTypeHeader, "application/x-www-form-urlencoded");
|
request.setHeader(QNetworkRequest::ContentTypeHeader, "application/x-www-form-urlencoded");
|
||||||
|
request.setHeader(QNetworkRequest::UserAgentHeader, QString("Cockatrice %1").arg(VERSION_STRING));
|
||||||
|
|
||||||
manager->post(request, data);
|
manager->post(request, data);
|
||||||
}
|
}
|
||||||
|
|
||||||
void DeckStatsInterface::copyDeckWithoutTokens(DeckList &source, DeckList &destination)
|
void DeckStatsInterface::copyDeckWithoutTokens(const DeckList &source, DeckList &destination)
|
||||||
{
|
{
|
||||||
auto copyIfNotAToken = [this, &destination](const auto node, const auto card) {
|
auto copyIfNotAToken = [&destination](const auto node, const auto card) {
|
||||||
CardInfoPtr dbCard = cardDatabase.getCard(card->getName());
|
CardInfoPtr dbCard = CardDatabaseManager::query()->getCardInfo(card->getName());
|
||||||
if (dbCard && !dbCard->getIsToken()) {
|
if (dbCard && !dbCard->getIsToken()) {
|
||||||
DecklistCardNode *addedCard = destination.addCard(card->getName(), node->getName());
|
DecklistCardNode *addedCard = destination.addCard(card->getName(), node->getName(), -1);
|
||||||
addedCard->setNumber(card->getNumber());
|
addedCard->setNumber(card->getNumber());
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
@ -1,10 +1,13 @@
|
||||||
|
/**
|
||||||
|
* @file deck_stats_interface.h
|
||||||
|
* @ingroup ApiInterfaces
|
||||||
|
*/
|
||||||
|
//! \todo Document this file.
|
||||||
|
|
||||||
#ifndef DECKSTATS_INTERFACE_H
|
#ifndef DECKSTATS_INTERFACE_H
|
||||||
#define DECKSTATS_INTERFACE_H
|
#define DECKSTATS_INTERFACE_H
|
||||||
|
|
||||||
#include "../game/cards/card_database.h"
|
#include <libcockatrice/deck_list/deck_list.h>
|
||||||
#include "decklist.h"
|
|
||||||
|
|
||||||
#include <QObject>
|
|
||||||
|
|
||||||
class QByteArray;
|
class QByteArray;
|
||||||
class QNetworkAccessManager;
|
class QNetworkAccessManager;
|
||||||
|
|
@ -17,22 +20,20 @@ class DeckStatsInterface : public QObject
|
||||||
private:
|
private:
|
||||||
QNetworkAccessManager *manager;
|
QNetworkAccessManager *manager;
|
||||||
|
|
||||||
CardDatabase &cardDatabase;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Deckstats doesn't recognize token cards, and instead tries to find the
|
* Deckstats doesn't recognize token cards, and instead tries to find the
|
||||||
* closest non-token card instead. So we construct a new deck which has no
|
* closest non-token card instead. So we construct a new deck which has no
|
||||||
* tokens.
|
* tokens.
|
||||||
*/
|
*/
|
||||||
void copyDeckWithoutTokens(DeckList &source, DeckList &destination);
|
void copyDeckWithoutTokens(const DeckList &source, DeckList &destination);
|
||||||
|
|
||||||
private slots:
|
private slots:
|
||||||
void queryFinished(QNetworkReply *reply);
|
void queryFinished(QNetworkReply *reply);
|
||||||
void getAnalyzeRequestData(DeckList *deck, QByteArray *data);
|
void getAnalyzeRequestData(const DeckList &deck, QByteArray &data);
|
||||||
|
|
||||||
public:
|
public:
|
||||||
explicit DeckStatsInterface(CardDatabase &_cardDatabase, QObject *parent = nullptr);
|
explicit DeckStatsInterface(QObject *parent = nullptr);
|
||||||
void analyzeDeck(DeckList *deck);
|
void analyzeDeck(const DeckList &deck);
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
@ -1,17 +1,17 @@
|
||||||
#include "tapped_out_interface.h"
|
#include "tapped_out_interface.h"
|
||||||
|
|
||||||
#include "decklist.h"
|
|
||||||
|
|
||||||
#include <QDesktopServices>
|
#include <QDesktopServices>
|
||||||
#include <QMessageBox>
|
#include <QMessageBox>
|
||||||
#include <QNetworkAccessManager>
|
|
||||||
#include <QNetworkReply>
|
#include <QNetworkReply>
|
||||||
#include <QNetworkRequest>
|
#include <QNetworkRequest>
|
||||||
#include <QRegularExpression>
|
#include <QRegularExpression>
|
||||||
#include <QUrlQuery>
|
#include <QUrlQuery>
|
||||||
|
#include <libcockatrice/card/database/card_database_manager.h>
|
||||||
|
#include <libcockatrice/deck_list/deck_list.h>
|
||||||
|
#include <libcockatrice/deck_list/tree/deck_list_card_node.h>
|
||||||
|
#include <version_string.h>
|
||||||
|
|
||||||
TappedOutInterface::TappedOutInterface(CardDatabase &_cardDatabase, QObject *parent)
|
TappedOutInterface::TappedOutInterface(QObject *parent) : QObject(parent)
|
||||||
: QObject(parent), cardDatabase(_cardDatabase)
|
|
||||||
{
|
{
|
||||||
manager = new QNetworkAccessManager(this);
|
manager = new QNetworkAccessManager(this);
|
||||||
connect(manager, &QNetworkAccessManager::finished, this, &TappedOutInterface::queryFinished);
|
connect(manager, &QNetworkAccessManager::finished, this, &TappedOutInterface::queryFinished);
|
||||||
|
|
@ -67,43 +67,48 @@ void TappedOutInterface::queryFinished(QNetworkReply *reply)
|
||||||
deleteLater();
|
deleteLater();
|
||||||
}
|
}
|
||||||
|
|
||||||
void TappedOutInterface::getAnalyzeRequestData(DeckList *deck, QByteArray *data)
|
void TappedOutInterface::getAnalyzeRequestData(const DeckList &deck, QByteArray &data)
|
||||||
{
|
{
|
||||||
DeckList mainboard, sideboard;
|
DeckList mainboard, sideboard;
|
||||||
copyDeckSplitMainAndSide(*deck, mainboard, sideboard);
|
copyDeckSplitMainAndSide(deck, mainboard, sideboard);
|
||||||
|
|
||||||
QUrl params;
|
QUrl params;
|
||||||
QUrlQuery urlQuery;
|
QUrlQuery urlQuery;
|
||||||
urlQuery.addQueryItem("name", deck->getName());
|
urlQuery.addQueryItem("name", deck.getName());
|
||||||
urlQuery.addQueryItem("mainboard", mainboard.writeToString_Plain(false, true));
|
urlQuery.addQueryItem("mainboard", mainboard.writeToString_Plain(false, true));
|
||||||
urlQuery.addQueryItem("sideboard", sideboard.writeToString_Plain(false, true));
|
urlQuery.addQueryItem("sideboard", sideboard.writeToString_Plain(false, true));
|
||||||
params.setQuery(urlQuery);
|
params.setQuery(urlQuery);
|
||||||
data->append(params.query(QUrl::EncodeReserved).toUtf8());
|
data.append(params.query(QUrl::EncodeReserved).toUtf8());
|
||||||
}
|
}
|
||||||
|
|
||||||
void TappedOutInterface::analyzeDeck(DeckList *deck)
|
void TappedOutInterface::analyzeDeck(const DeckList &deck)
|
||||||
{
|
{
|
||||||
QByteArray data;
|
QByteArray data;
|
||||||
getAnalyzeRequestData(deck, &data);
|
getAnalyzeRequestData(deck, data);
|
||||||
|
|
||||||
QNetworkRequest request(QUrl("https://tappedout.net/mtg-decks/paste/"));
|
QNetworkRequest request(QUrl("https://tappedout.net/mtg-decks/paste/"));
|
||||||
request.setHeader(QNetworkRequest::ContentTypeHeader, "application/x-www-form-urlencoded");
|
request.setHeader(QNetworkRequest::ContentTypeHeader, "application/x-www-form-urlencoded");
|
||||||
|
request.setHeader(QNetworkRequest::UserAgentHeader, QString("Cockatrice %1").arg(VERSION_STRING));
|
||||||
|
// we interpret the redirect and open it in the browser instead, do not follow redirects
|
||||||
|
request.setAttribute(QNetworkRequest::RedirectPolicyAttribute, QNetworkRequest::ManualRedirectPolicy);
|
||||||
|
|
||||||
manager->post(request, data);
|
manager->post(request, data);
|
||||||
}
|
}
|
||||||
|
|
||||||
void TappedOutInterface::copyDeckSplitMainAndSide(DeckList &source, DeckList &mainboard, DeckList &sideboard)
|
void TappedOutInterface::copyDeckSplitMainAndSide(const DeckList &source, DeckList &mainboard, DeckList &sideboard)
|
||||||
{
|
{
|
||||||
auto copyMainOrSide = [this, &mainboard, &sideboard](const auto node, const auto card) {
|
auto copyMainOrSide = [&mainboard, &sideboard](const auto node, const auto card) {
|
||||||
CardInfoPtr dbCard = cardDatabase.getCard(card->getName());
|
CardInfoPtr dbCard = CardDatabaseManager::query()->getCardInfo(card->getName());
|
||||||
if (!dbCard || dbCard->getIsToken())
|
if (!dbCard || dbCard->getIsToken()) {
|
||||||
return;
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
DecklistCardNode *addedCard;
|
DecklistCardNode *addedCard;
|
||||||
if (node->getName() == DECK_ZONE_SIDE)
|
if (node->getName() == DECK_ZONE_SIDE) {
|
||||||
addedCard = sideboard.addCard(card->getName(), node->getName());
|
addedCard = sideboard.addCard(card->getName(), node->getName(), -1);
|
||||||
else
|
} else {
|
||||||
addedCard = mainboard.addCard(card->getName(), node->getName());
|
addedCard = mainboard.addCard(card->getName(), node->getName(), -1);
|
||||||
|
}
|
||||||
addedCard->setNumber(card->getNumber());
|
addedCard->setNumber(card->getNumber());
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
@ -1,9 +1,12 @@
|
||||||
|
/**
|
||||||
|
* @file tapped_out_interface.h
|
||||||
|
* @ingroup ApiInterfaces
|
||||||
|
*/
|
||||||
|
//! \todo Document this file.
|
||||||
|
|
||||||
#ifndef TAPPEDOUT_INTERFACE_H
|
#ifndef TAPPEDOUT_INTERFACE_H
|
||||||
#define TAPPEDOUT_INTERFACE_H
|
#define TAPPEDOUT_INTERFACE_H
|
||||||
|
|
||||||
#include "../game/cards/card_database.h"
|
|
||||||
#include "decklist.h"
|
|
||||||
|
|
||||||
#include <QLoggingCategory>
|
#include <QLoggingCategory>
|
||||||
#include <QObject>
|
#include <QObject>
|
||||||
|
|
||||||
|
|
@ -26,15 +29,14 @@ class TappedOutInterface : public QObject
|
||||||
private:
|
private:
|
||||||
QNetworkAccessManager *manager;
|
QNetworkAccessManager *manager;
|
||||||
|
|
||||||
CardDatabase &cardDatabase;
|
void copyDeckSplitMainAndSide(const DeckList &source, DeckList &mainboard, DeckList &sideboard);
|
||||||
void copyDeckSplitMainAndSide(DeckList &source, DeckList &mainboard, DeckList &sideboard);
|
|
||||||
private slots:
|
private slots:
|
||||||
void queryFinished(QNetworkReply *reply);
|
void queryFinished(QNetworkReply *reply);
|
||||||
void getAnalyzeRequestData(DeckList *deck, QByteArray *data);
|
void getAnalyzeRequestData(const DeckList &deck, QByteArray &data);
|
||||||
|
|
||||||
public:
|
public:
|
||||||
explicit TappedOutInterface(CardDatabase &_cardDatabase, QObject *parent = nullptr);
|
explicit TappedOutInterface(QObject *parent = nullptr);
|
||||||
void analyzeDeck(DeckList *deck);
|
void analyzeDeck(const DeckList &deck);
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
@ -0,0 +1,61 @@
|
||||||
|
#include "deck_link_to_api_transformer.h"
|
||||||
|
|
||||||
|
#include <QRegularExpression>
|
||||||
|
|
||||||
|
namespace DeckLinkToApiTransformer
|
||||||
|
{
|
||||||
|
|
||||||
|
static const QString TAPPEDOUT_BASE = "https://tappedout.net/mtg-decks/";
|
||||||
|
static const QString TAPPEDOUT_SUFFIX = "/?fmt=txt";
|
||||||
|
|
||||||
|
static const QString ARCHIDEKT_BASE = "https://archidekt.com/api/decks/";
|
||||||
|
static const QString ARCHIDEKT_SUFFIX = "/?format=json";
|
||||||
|
|
||||||
|
static const QString MOXFIELD_BASE = "https://api.moxfield.com/v2/decks/all/";
|
||||||
|
static const QString MOXFIELD_SUFFIX = "/";
|
||||||
|
|
||||||
|
static const QString DECKSTATS_SUFFIX = "?include_comments=1&export_mtgarena=1";
|
||||||
|
|
||||||
|
bool parseDeckUrl(const QString &url, ParsedDeckInfo &outInfo)
|
||||||
|
{
|
||||||
|
static QRegularExpression rxTappedOut("tappedout\\.net/(?:mtg-decks/)?([^/?#]+)");
|
||||||
|
static QRegularExpression rxArchidekt("archidekt\\.com/decks/(\\d+)");
|
||||||
|
static QRegularExpression rxMoxfield("moxfield\\.com/decks/([a-zA-Z0-9_-]+)");
|
||||||
|
static QRegularExpression rxDeckstats("deckstats\\.net/decks/(\\d+/[a-zA-Z0-9_-]+)");
|
||||||
|
|
||||||
|
QRegularExpressionMatch match;
|
||||||
|
|
||||||
|
if ((match = rxTappedOut.match(url)).hasMatch()) {
|
||||||
|
QString slug = match.captured(1);
|
||||||
|
outInfo = ParsedDeckInfo{.baseUrl = TAPPEDOUT_BASE,
|
||||||
|
.deckID = slug,
|
||||||
|
.fullUrl = TAPPEDOUT_BASE + slug + TAPPEDOUT_SUFFIX,
|
||||||
|
.provider = DeckProvider::TappedOut};
|
||||||
|
return true;
|
||||||
|
} else if ((match = rxArchidekt.match(url)).hasMatch()) {
|
||||||
|
QString deckID = match.captured(1);
|
||||||
|
outInfo = ParsedDeckInfo{.baseUrl = ARCHIDEKT_BASE,
|
||||||
|
.deckID = deckID,
|
||||||
|
.fullUrl = ARCHIDEKT_BASE + deckID + ARCHIDEKT_SUFFIX,
|
||||||
|
.provider = DeckProvider::Archidekt};
|
||||||
|
return true;
|
||||||
|
} else if ((match = rxMoxfield.match(url)).hasMatch()) {
|
||||||
|
QString deckID = match.captured(1);
|
||||||
|
outInfo = ParsedDeckInfo{.baseUrl = MOXFIELD_BASE,
|
||||||
|
.deckID = deckID,
|
||||||
|
.fullUrl = MOXFIELD_BASE + deckID + MOXFIELD_SUFFIX,
|
||||||
|
.provider = DeckProvider::Moxfield};
|
||||||
|
return true;
|
||||||
|
} else if ((match = rxDeckstats.match(url)).hasMatch()) {
|
||||||
|
QString deckPath = match.captured(1);
|
||||||
|
outInfo = ParsedDeckInfo{.baseUrl = "https://deckstats.net/decks/",
|
||||||
|
.deckID = deckPath,
|
||||||
|
.fullUrl = "https://deckstats.net/decks/" + deckPath + DECKSTATS_SUFFIX,
|
||||||
|
.provider = DeckProvider::Deckstats};
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace DeckLinkToApiTransformer
|
||||||
|
|
@ -0,0 +1,37 @@
|
||||||
|
/**
|
||||||
|
* @file deck_link_to_api_transformer.h
|
||||||
|
* @ingroup ApiInterfaces
|
||||||
|
*/
|
||||||
|
//! \todo Document this file.
|
||||||
|
|
||||||
|
#ifndef DECK_LINK_TO_API_TRANSFORMER_H
|
||||||
|
#define DECK_LINK_TO_API_TRANSFORMER_H
|
||||||
|
|
||||||
|
#include <QString>
|
||||||
|
|
||||||
|
enum class DeckProvider
|
||||||
|
{
|
||||||
|
TappedOut,
|
||||||
|
Archidekt,
|
||||||
|
Moxfield,
|
||||||
|
Deckstats,
|
||||||
|
Unknown
|
||||||
|
};
|
||||||
|
|
||||||
|
struct ParsedDeckInfo
|
||||||
|
{
|
||||||
|
QString baseUrl;
|
||||||
|
QString deckID;
|
||||||
|
QString fullUrl;
|
||||||
|
DeckProvider provider;
|
||||||
|
};
|
||||||
|
|
||||||
|
namespace DeckLinkToApiTransformer
|
||||||
|
{
|
||||||
|
|
||||||
|
// Returns true if the input URL is recognized and fills outInfo.
|
||||||
|
bool parseDeckUrl(const QString &url, ParsedDeckInfo &outInfo);
|
||||||
|
|
||||||
|
} // namespace DeckLinkToApiTransformer
|
||||||
|
|
||||||
|
#endif // DECK_LINK_TO_API_TRANSFORMER_H
|
||||||
|
|
@ -0,0 +1,121 @@
|
||||||
|
/**
|
||||||
|
* @file interface_json_deck_parser.h
|
||||||
|
* @ingroup ApiInterfaces
|
||||||
|
*/
|
||||||
|
//! \todo Document this file.
|
||||||
|
|
||||||
|
#ifndef INTERFACE_JSON_DECK_PARSER_H
|
||||||
|
#define INTERFACE_JSON_DECK_PARSER_H
|
||||||
|
|
||||||
|
#include "../../../interface/deck_loader/card_node_function.h"
|
||||||
|
|
||||||
|
#include <QJsonArray>
|
||||||
|
#include <QJsonObject>
|
||||||
|
#include <libcockatrice/card/import/card_name_normalizer.h>
|
||||||
|
#include <libcockatrice/deck_list/deck_list.h>
|
||||||
|
|
||||||
|
class IJsonDeckParser
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
virtual ~IJsonDeckParser() = default;
|
||||||
|
|
||||||
|
virtual DeckList parse(const QJsonObject &obj) = 0;
|
||||||
|
};
|
||||||
|
|
||||||
|
class ArchidektJsonParser : public IJsonDeckParser
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
DeckList parse(const QJsonObject &obj) override
|
||||||
|
{
|
||||||
|
DeckList deckList;
|
||||||
|
|
||||||
|
QString deckName = obj.value("name").toString();
|
||||||
|
QString deckDescription = obj.value("description").toString();
|
||||||
|
|
||||||
|
deckList.setName(deckName);
|
||||||
|
deckList.setComments(deckDescription);
|
||||||
|
|
||||||
|
QString outputText;
|
||||||
|
QTextStream outStream(&outputText);
|
||||||
|
|
||||||
|
for (auto entry : obj.value("cards").toArray()) {
|
||||||
|
auto quantity = entry.toObject().value("quantity").toInt();
|
||||||
|
|
||||||
|
auto card = entry.toObject().value("card").toObject();
|
||||||
|
auto oracleCard = card.value("oracleCard").toObject();
|
||||||
|
QString cardName = oracleCard.value("name").toString();
|
||||||
|
QString setName = card.value("edition").toObject().value("editioncode").toString().toUpper();
|
||||||
|
QString collectorNumber = card.value("collectorNumber").toString();
|
||||||
|
|
||||||
|
outStream << quantity << ' ' << cardName << " (" << setName << ") " << collectorNumber << '\n';
|
||||||
|
}
|
||||||
|
|
||||||
|
deckList.loadFromStream_Plain(outStream, false, CardNameNormalizer());
|
||||||
|
deckList.forEachCard(CardNodeFunction::ResolveProviderId());
|
||||||
|
|
||||||
|
return deckList;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
class MoxfieldJsonParser : public IJsonDeckParser
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
DeckList parse(const QJsonObject &obj) override
|
||||||
|
{
|
||||||
|
DeckList deckList;
|
||||||
|
|
||||||
|
QString deckName = obj.value("name").toString();
|
||||||
|
QString deckDescription = obj.value("description").toString();
|
||||||
|
|
||||||
|
deckList.setName(deckName);
|
||||||
|
deckList.setComments(deckDescription);
|
||||||
|
|
||||||
|
QString outputText;
|
||||||
|
QTextStream outStream(&outputText);
|
||||||
|
|
||||||
|
for (auto entry : obj.value("mainboard").toObject()) {
|
||||||
|
auto quantity = entry.toObject().value("quantity").toInt();
|
||||||
|
|
||||||
|
auto card = entry.toObject().value("card").toObject();
|
||||||
|
QString cardName = card.value("name").toString();
|
||||||
|
QString setName = card.value("set").toString().toUpper();
|
||||||
|
QString collectorNumber = card.value("cn").toString();
|
||||||
|
|
||||||
|
outStream << quantity << ' ' << cardName << " (" << setName << ") " << collectorNumber << '\n';
|
||||||
|
}
|
||||||
|
|
||||||
|
outStream << '\n';
|
||||||
|
|
||||||
|
for (auto entry : obj.value("sideboard").toObject()) {
|
||||||
|
auto quantity = entry.toObject().value("quantity").toInt();
|
||||||
|
|
||||||
|
auto card = entry.toObject().value("card").toObject();
|
||||||
|
QString cardName = card.value("name").toString();
|
||||||
|
QString setName = card.value("set").toString().toUpper();
|
||||||
|
QString collectorNumber = card.value("cn").toString();
|
||||||
|
|
||||||
|
outStream << quantity << ' ' << cardName << " (" << setName << ") " << collectorNumber << '\n';
|
||||||
|
}
|
||||||
|
|
||||||
|
deckList.loadFromStream_Plain(outStream, false, CardNameNormalizer());
|
||||||
|
deckList.forEachCard(CardNodeFunction::ResolveProviderId());
|
||||||
|
|
||||||
|
QJsonObject commandersObj = obj.value("commanders").toObject();
|
||||||
|
if (!commandersObj.isEmpty()) {
|
||||||
|
for (auto it = commandersObj.begin(); it != commandersObj.end(); ++it) {
|
||||||
|
QJsonObject cardData = it.value().toObject().value("card").toObject();
|
||||||
|
QString commanderName = cardData.value("name").toString();
|
||||||
|
QString setName = cardData.value("set").toString().toUpper();
|
||||||
|
QString collectorNumber = cardData.value("cn").toString();
|
||||||
|
QString providerId = cardData.value("scryfall_id").toString();
|
||||||
|
|
||||||
|
deckList.setBannerCard({commanderName, providerId});
|
||||||
|
deckList.addCard(commanderName, DECK_ZONE_MAIN, -1, setName, collectorNumber, providerId);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return deckList;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif // INTERFACE_JSON_DECK_PARSER_H
|
||||||
|
|
@ -1,21 +1,20 @@
|
||||||
#include "spoiler_background_updater.h"
|
#include "spoiler_background_updater.h"
|
||||||
|
|
||||||
#include "../../game/cards/card_database.h"
|
#include "../../../../interface/window_main.h"
|
||||||
#include "../../game/cards/card_database_manager.h"
|
#include "../../../../main.h"
|
||||||
#include "../../main.h"
|
#include "../../../settings/cache_settings.h"
|
||||||
#include "../../settings/cache_settings.h"
|
|
||||||
#include "../ui/window_main.h"
|
|
||||||
|
|
||||||
#include <QApplication>
|
|
||||||
#include <QCryptographicHash>
|
#include <QCryptographicHash>
|
||||||
#include <QDateTime>
|
#include <QDateTime>
|
||||||
#include <QDebug>
|
#include <QDebug>
|
||||||
#include <QFile>
|
#include <QFile>
|
||||||
#include <QLocale>
|
#include <QLocale>
|
||||||
#include <QMessageBox>
|
|
||||||
#include <QNetworkReply>
|
#include <QNetworkReply>
|
||||||
#include <QUrl>
|
#include <QUrl>
|
||||||
#include <QtConcurrent>
|
#include <QtConcurrent>
|
||||||
|
#include <libcockatrice/card/database/card_database.h>
|
||||||
|
#include <libcockatrice/card/database/card_database_manager.h>
|
||||||
|
#include <version_string.h>
|
||||||
|
|
||||||
#define SPOILERS_STATUS_URL "https://raw.githubusercontent.com/Cockatrice/Magic-Spoiler/files/SpoilerSeasonEnabled"
|
#define SPOILERS_STATUS_URL "https://raw.githubusercontent.com/Cockatrice/Magic-Spoiler/files/SpoilerSeasonEnabled"
|
||||||
#define SPOILERS_URL "https://raw.githubusercontent.com/Cockatrice/Magic-Spoiler/files/spoiler.xml"
|
#define SPOILERS_URL "https://raw.githubusercontent.com/Cockatrice/Magic-Spoiler/files/spoiler.xml"
|
||||||
|
|
@ -41,7 +40,9 @@ void SpoilerBackgroundUpdater::startSpoilerDownloadProcess(QString url, bool sav
|
||||||
void SpoilerBackgroundUpdater::downloadFromURL(QUrl url, bool saveResults)
|
void SpoilerBackgroundUpdater::downloadFromURL(QUrl url, bool saveResults)
|
||||||
{
|
{
|
||||||
auto *nam = new QNetworkAccessManager(this);
|
auto *nam = new QNetworkAccessManager(this);
|
||||||
QNetworkReply *reply = nam->get(QNetworkRequest(url));
|
auto request = QNetworkRequest(url);
|
||||||
|
request.setHeader(QNetworkRequest::UserAgentHeader, QString("Cockatrice %1").arg(VERSION_STRING));
|
||||||
|
QNetworkReply *reply = nam->get(request);
|
||||||
|
|
||||||
if (saveResults) {
|
if (saveResults) {
|
||||||
// This will write out to the file (used for spoiler.xml)
|
// This will write out to the file (used for spoiler.xml)
|
||||||
|
|
@ -1,3 +1,9 @@
|
||||||
|
/**
|
||||||
|
* @file spoiler_background_updater.h
|
||||||
|
* @ingroup Client
|
||||||
|
*/
|
||||||
|
//! \todo Document this file.
|
||||||
|
|
||||||
#ifndef COCKATRICE_SPOILER_DOWNLOADER_H
|
#ifndef COCKATRICE_SPOILER_DOWNLOADER_H
|
||||||
#define COCKATRICE_SPOILER_DOWNLOADER_H
|
#define COCKATRICE_SPOILER_DOWNLOADER_H
|
||||||
|
|
||||||
|
|
@ -16,7 +22,7 @@ public:
|
||||||
inline QString getCardUpdaterBinaryName()
|
inline QString getCardUpdaterBinaryName()
|
||||||
{
|
{
|
||||||
return "oracle";
|
return "oracle";
|
||||||
};
|
}
|
||||||
QByteArray getHash(const QString fileName);
|
QByteArray getHash(const QString fileName);
|
||||||
QByteArray getHash(QByteArray data);
|
QByteArray getHash(QByteArray data);
|
||||||
static bool deleteSpoilerFile();
|
static bool deleteSpoilerFile();
|
||||||
|
|
@ -1,6 +1,6 @@
|
||||||
#include "client_update_checker.h"
|
#include "client_update_checker.h"
|
||||||
|
|
||||||
#include "../../settings/cache_settings.h"
|
#include "../../../settings/cache_settings.h"
|
||||||
#include "release_channel.h"
|
#include "release_channel.h"
|
||||||
|
|
||||||
ClientUpdateChecker::ClientUpdateChecker(QObject *parent) : QObject(parent)
|
ClientUpdateChecker::ClientUpdateChecker(QObject *parent) : QObject(parent)
|
||||||
|
|
@ -1,3 +1,9 @@
|
||||||
|
/**
|
||||||
|
* @file client_update_checker.h
|
||||||
|
* @ingroup ClientUpdate
|
||||||
|
*/
|
||||||
|
//! \todo Document this file.
|
||||||
|
|
||||||
#ifndef CLIENT_UPDATE_CHECKER_H
|
#ifndef CLIENT_UPDATE_CHECKER_H
|
||||||
#define CLIENT_UPDATE_CHECKER_H
|
#define CLIENT_UPDATE_CHECKER_H
|
||||||
#include <QObject>
|
#include <QObject>
|
||||||
|
|
@ -129,8 +129,9 @@ void StableReleaseChannel::releaseListFinished()
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!lastRelease)
|
if (!lastRelease) {
|
||||||
lastRelease = new Release;
|
lastRelease = new Release;
|
||||||
|
}
|
||||||
|
|
||||||
lastRelease->setName(resultMap["name"].toString());
|
lastRelease->setName(resultMap["name"].toString());
|
||||||
lastRelease->setDescriptionUrl(resultMap["html_url"].toString());
|
lastRelease->setDescriptionUrl(resultMap["html_url"].toString());
|
||||||
|
|
@ -246,8 +247,9 @@ void BetaReleaseChannel::releaseListFinished()
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (lastRelease == nullptr)
|
if (lastRelease == nullptr) {
|
||||||
lastRelease = new Release;
|
lastRelease = new Release;
|
||||||
|
}
|
||||||
|
|
||||||
lastRelease->setCommitHash(resultMap["target_commitish"].toString());
|
lastRelease->setCommitHash(resultMap["target_commitish"].toString());
|
||||||
lastRelease->setPublishDate(resultMap["published_at"].toDate());
|
lastRelease->setPublishDate(resultMap["published_at"].toDate());
|
||||||
|
|
@ -1,3 +1,9 @@
|
||||||
|
/**
|
||||||
|
* @file release_channel.h
|
||||||
|
* @ingroup ClientUpdate
|
||||||
|
*/
|
||||||
|
//! \todo Document this file.
|
||||||
|
|
||||||
#ifndef RELEASECHANNEL_H
|
#ifndef RELEASECHANNEL_H
|
||||||
#define RELEASECHANNEL_H
|
#define RELEASECHANNEL_H
|
||||||
|
|
||||||
|
|
@ -51,27 +57,27 @@ protected:
|
||||||
}
|
}
|
||||||
|
|
||||||
public:
|
public:
|
||||||
QString getName() const
|
[[nodiscard]] QString getName() const
|
||||||
{
|
{
|
||||||
return name;
|
return name;
|
||||||
}
|
}
|
||||||
QString getDescriptionUrl() const
|
[[nodiscard]] QString getDescriptionUrl() const
|
||||||
{
|
{
|
||||||
return descriptionUrl;
|
return descriptionUrl;
|
||||||
}
|
}
|
||||||
QString getDownloadUrl() const
|
[[nodiscard]] QString getDownloadUrl() const
|
||||||
{
|
{
|
||||||
return downloadUrl;
|
return downloadUrl;
|
||||||
}
|
}
|
||||||
QString getCommitHash() const
|
[[nodiscard]] QString getCommitHash() const
|
||||||
{
|
{
|
||||||
return commitHash;
|
return commitHash;
|
||||||
}
|
}
|
||||||
QDate getPublishDate() const
|
[[nodiscard]] QDate getPublishDate() const
|
||||||
{
|
{
|
||||||
return publishDate;
|
return publishDate;
|
||||||
}
|
}
|
||||||
bool isCompatibleVersionFound() const
|
[[nodiscard]] bool isCompatibleVersionFound() const
|
||||||
{
|
{
|
||||||
return compatibleVersionFound;
|
return compatibleVersionFound;
|
||||||
}
|
}
|
||||||
|
|
@ -91,15 +97,15 @@ protected:
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
static bool downloadMatchesCurrentOS(const QString &fileName);
|
static bool downloadMatchesCurrentOS(const QString &fileName);
|
||||||
virtual QString getReleaseChannelUrl() const = 0;
|
[[nodiscard]] virtual QString getReleaseChannelUrl() const = 0;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
Release *getLastRelease()
|
Release *getLastRelease()
|
||||||
{
|
{
|
||||||
return lastRelease;
|
return lastRelease;
|
||||||
}
|
}
|
||||||
virtual QString getManualDownloadUrl() const = 0;
|
[[nodiscard]] virtual QString getManualDownloadUrl() const = 0;
|
||||||
virtual QString getName() const = 0;
|
[[nodiscard]] virtual QString getName() const = 0;
|
||||||
void checkForUpdates();
|
void checkForUpdates();
|
||||||
signals:
|
signals:
|
||||||
void finishedCheck(bool needToUpdate, bool isCompatible, Release *release);
|
void finishedCheck(bool needToUpdate, bool isCompatible, Release *release);
|
||||||
|
|
@ -116,12 +122,12 @@ public:
|
||||||
explicit StableReleaseChannel() = default;
|
explicit StableReleaseChannel() = default;
|
||||||
~StableReleaseChannel() override = default;
|
~StableReleaseChannel() override = default;
|
||||||
|
|
||||||
QString getManualDownloadUrl() const override;
|
[[nodiscard]] QString getManualDownloadUrl() const override;
|
||||||
|
|
||||||
QString getName() const override;
|
[[nodiscard]] QString getName() const override;
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
QString getReleaseChannelUrl() const override;
|
[[nodiscard]] QString getReleaseChannelUrl() const override;
|
||||||
protected slots:
|
protected slots:
|
||||||
|
|
||||||
void releaseListFinished() override;
|
void releaseListFinished() override;
|
||||||
|
|
@ -137,12 +143,12 @@ public:
|
||||||
BetaReleaseChannel() = default;
|
BetaReleaseChannel() = default;
|
||||||
~BetaReleaseChannel() override = default;
|
~BetaReleaseChannel() override = default;
|
||||||
|
|
||||||
QString getManualDownloadUrl() const override;
|
[[nodiscard]] QString getManualDownloadUrl() const override;
|
||||||
|
|
||||||
QString getName() const override;
|
[[nodiscard]] QString getName() const override;
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
QString getReleaseChannelUrl() const override;
|
[[nodiscard]] QString getReleaseChannelUrl() const override;
|
||||||
protected slots:
|
protected slots:
|
||||||
|
|
||||||
void releaseListFinished() override;
|
void releaseListFinished() override;
|
||||||
|
|
@ -1,6 +1,5 @@
|
||||||
#include "update_downloader.h"
|
#include "update_downloader.h"
|
||||||
|
|
||||||
#include <QDebug>
|
|
||||||
#include <QUrl>
|
#include <QUrl>
|
||||||
|
|
||||||
UpdateDownloader::UpdateDownloader(QObject *parent) : QObject(parent), response(nullptr)
|
UpdateDownloader::UpdateDownloader(QObject *parent) : QObject(parent), response(nullptr)
|
||||||
|
|
@ -11,8 +10,9 @@ UpdateDownloader::UpdateDownloader(QObject *parent) : QObject(parent), response(
|
||||||
void UpdateDownloader::beginDownload(QUrl downloadUrl)
|
void UpdateDownloader::beginDownload(QUrl downloadUrl)
|
||||||
{
|
{
|
||||||
// Save the original URL because we need it for the filename
|
// Save the original URL because we need it for the filename
|
||||||
if (originalUrl.isEmpty())
|
if (originalUrl.isEmpty()) {
|
||||||
originalUrl = downloadUrl;
|
originalUrl = downloadUrl;
|
||||||
|
}
|
||||||
|
|
||||||
response = netMan->get(QNetworkRequest(downloadUrl));
|
response = netMan->get(QNetworkRequest(downloadUrl));
|
||||||
connect(response, &QNetworkReply::finished, this, &UpdateDownloader::fileFinished);
|
connect(response, &QNetworkReply::finished, this, &UpdateDownloader::fileFinished);
|
||||||
|
|
@ -22,8 +22,9 @@ void UpdateDownloader::beginDownload(QUrl downloadUrl)
|
||||||
|
|
||||||
void UpdateDownloader::downloadError(QNetworkReply::NetworkError)
|
void UpdateDownloader::downloadError(QNetworkReply::NetworkError)
|
||||||
{
|
{
|
||||||
if (response == nullptr)
|
if (response == nullptr) {
|
||||||
return;
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
emit error(response->errorString().toUtf8());
|
emit error(response->errorString().toUtf8());
|
||||||
}
|
}
|
||||||
|
|
@ -1,13 +1,13 @@
|
||||||
//
|
/**
|
||||||
// Created by miguel on 28/12/15.
|
* @file update_downloader.h
|
||||||
//
|
* @ingroup ClientUpdate
|
||||||
|
*/
|
||||||
|
//! \todo Document this file.
|
||||||
|
|
||||||
#ifndef COCKATRICE_UPDATEDOWNLOADER_H
|
#ifndef COCKATRICE_UPDATEDOWNLOADER_H
|
||||||
#define COCKATRICE_UPDATEDOWNLOADER_H
|
#define COCKATRICE_UPDATEDOWNLOADER_H
|
||||||
|
|
||||||
#include <QDate>
|
|
||||||
#include <QObject>
|
#include <QObject>
|
||||||
#include <QUrl>
|
|
||||||
#include <QtNetwork>
|
#include <QtNetwork>
|
||||||
|
|
||||||
class UpdateDownloader : public QObject
|
class UpdateDownloader : public QObject
|
||||||
|
|
@ -1,8 +1,12 @@
|
||||||
#include "cache_settings.h"
|
#include "cache_settings.h"
|
||||||
|
|
||||||
#include "../client/network/release_channel.h"
|
#include "../../interface/card_picture_loader/card_picture_loader_cache_method.h"
|
||||||
#include "card_override_settings.h"
|
#include "../../interface/card_picture_loader/card_picture_loader_local_schemes.h"
|
||||||
|
#include "../network/update/client/release_channel.h"
|
||||||
|
#include "card_counter_settings.h"
|
||||||
|
#include "version_string.h"
|
||||||
|
|
||||||
|
#include <QAbstractListModel>
|
||||||
#include <QApplication>
|
#include <QApplication>
|
||||||
#include <QDebug>
|
#include <QDebug>
|
||||||
#include <QDir>
|
#include <QDir>
|
||||||
|
|
@ -10,16 +14,23 @@
|
||||||
#include <QGlobalStatic>
|
#include <QGlobalStatic>
|
||||||
#include <QSettings>
|
#include <QSettings>
|
||||||
#include <QStandardPaths>
|
#include <QStandardPaths>
|
||||||
|
#include <libcockatrice/settings/card_override_settings.h>
|
||||||
#include <utility>
|
#include <utility>
|
||||||
|
|
||||||
Q_GLOBAL_STATIC(SettingsCache, settingsCache);
|
Q_GLOBAL_STATIC(SettingsCache, settingsCache)
|
||||||
|
|
||||||
|
SettingsCache &SettingsCache::instance()
|
||||||
|
{
|
||||||
|
return *settingsCache; // returns a QT managed singleton reference
|
||||||
|
}
|
||||||
|
|
||||||
QString SettingsCache::getDataPath()
|
QString SettingsCache::getDataPath()
|
||||||
{
|
{
|
||||||
if (isPortableBuild)
|
if (isPortableBuild) {
|
||||||
return qApp->applicationDirPath() + "/data";
|
return qApp->applicationDirPath() + "/data";
|
||||||
else
|
} else {
|
||||||
return QStandardPaths::writableLocation(QStandardPaths::AppLocalDataLocation);
|
return QStandardPaths::writableLocation(QStandardPaths::AppLocalDataLocation);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
QString SettingsCache::getSettingsPath()
|
QString SettingsCache::getSettingsPath()
|
||||||
|
|
@ -29,10 +40,11 @@ QString SettingsCache::getSettingsPath()
|
||||||
|
|
||||||
QString SettingsCache::getCachePath() const
|
QString SettingsCache::getCachePath() const
|
||||||
{
|
{
|
||||||
if (isPortableBuild)
|
if (isPortableBuild) {
|
||||||
return qApp->applicationDirPath() + "/cache";
|
return qApp->applicationDirPath() + "/cache";
|
||||||
else
|
} else {
|
||||||
return QStandardPaths::writableLocation(QStandardPaths::CacheLocation);
|
return QStandardPaths::writableLocation(QStandardPaths::CacheLocation);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
QString SettingsCache::getNetworkCachePath() const
|
QString SettingsCache::getNetworkCachePath() const
|
||||||
|
|
@ -42,14 +54,17 @@ QString SettingsCache::getNetworkCachePath() const
|
||||||
|
|
||||||
void SettingsCache::translateLegacySettings()
|
void SettingsCache::translateLegacySettings()
|
||||||
{
|
{
|
||||||
if (isPortableBuild)
|
if (isPortableBuild) {
|
||||||
return;
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
// Layouts
|
// Layouts
|
||||||
QFile layoutFile(getSettingsPath() + "layouts/deckLayout.ini");
|
QFile layoutFile(getSettingsPath() + "layouts/deckLayout.ini");
|
||||||
if (layoutFile.exists())
|
if (layoutFile.exists()) {
|
||||||
if (layoutFile.copy(getSettingsPath() + "layouts.ini"))
|
if (layoutFile.copy(getSettingsPath() + "layouts.ini")) {
|
||||||
layoutFile.remove();
|
layoutFile.remove();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
QStringList usedKeys;
|
QStringList usedKeys;
|
||||||
QSettings legacySetting;
|
QSettings legacySetting;
|
||||||
|
|
@ -59,9 +74,9 @@ void SettingsCache::translateLegacySettings()
|
||||||
QStringList setsGroups = legacySetting.childGroups();
|
QStringList setsGroups = legacySetting.childGroups();
|
||||||
for (int i = 0; i < setsGroups.size(); i++) {
|
for (int i = 0; i < setsGroups.size(); i++) {
|
||||||
legacySetting.beginGroup(setsGroups.at(i));
|
legacySetting.beginGroup(setsGroups.at(i));
|
||||||
cardDatabase().setEnabled(setsGroups.at(i), legacySetting.value("enabled").toBool());
|
cardDatabase()->setEnabled(setsGroups.at(i), legacySetting.value("enabled").toBool());
|
||||||
cardDatabase().setIsKnown(setsGroups.at(i), legacySetting.value("isknown").toBool());
|
cardDatabase()->setIsKnown(setsGroups.at(i), legacySetting.value("isknown").toBool());
|
||||||
cardDatabase().setSortKey(setsGroups.at(i), legacySetting.value("sortkey").toUInt());
|
cardDatabase()->setSortKey(setsGroups.at(i), legacySetting.value("sortkey").toUInt());
|
||||||
legacySetting.endGroup();
|
legacySetting.endGroup();
|
||||||
}
|
}
|
||||||
QStringList setsKeys = legacySetting.allKeys();
|
QStringList setsKeys = legacySetting.allKeys();
|
||||||
|
|
@ -108,10 +123,11 @@ void SettingsCache::translateLegacySettings()
|
||||||
gameFilters().setHideIgnoredUserGames(legacySetting.value("hide_ignored_user_games").toBool());
|
gameFilters().setHideIgnoredUserGames(legacySetting.value("hide_ignored_user_games").toBool());
|
||||||
gameFilters().setMinPlayers(legacySetting.value("min_players").toInt());
|
gameFilters().setMinPlayers(legacySetting.value("min_players").toInt());
|
||||||
|
|
||||||
if (legacySetting.value("max_players").toInt() > 1)
|
if (legacySetting.value("max_players").toInt() > 1) {
|
||||||
gameFilters().setMaxPlayers(legacySetting.value("max_players").toInt());
|
gameFilters().setMaxPlayers(legacySetting.value("max_players").toInt());
|
||||||
else
|
} else {
|
||||||
gameFilters().setMaxPlayers(99); // This prevents a bug where no games will show if max was not set before
|
gameFilters().setMaxPlayers(99); // This prevents a bug where no games will show if max was not set before
|
||||||
|
}
|
||||||
|
|
||||||
QStringList allFilters = legacySetting.allKeys();
|
QStringList allFilters = legacySetting.allKeys();
|
||||||
for (int i = 0; i < allFilters.size(); ++i) {
|
for (int i = 0; i < allFilters.size(); ++i) {
|
||||||
|
|
@ -127,8 +143,9 @@ void SettingsCache::translateLegacySettings()
|
||||||
|
|
||||||
QStringList allLegacyKeys = legacySetting.allKeys();
|
QStringList allLegacyKeys = legacySetting.allKeys();
|
||||||
for (int i = 0; i < allLegacyKeys.size(); ++i) {
|
for (int i = 0; i < allLegacyKeys.size(); ++i) {
|
||||||
if (usedKeys.contains(allLegacyKeys.at(i)))
|
if (usedKeys.contains(allLegacyKeys.at(i))) {
|
||||||
continue;
|
continue;
|
||||||
|
}
|
||||||
settings->setValue(allLegacyKeys.at(i), legacySetting.value(allLegacyKeys.at(i)));
|
settings->setValue(allLegacyKeys.at(i), legacySetting.value(allLegacyKeys.at(i)));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
@ -139,8 +156,9 @@ QString SettingsCache::getSafeConfigPath(QString configEntry, QString defaultPat
|
||||||
// if the config settings is empty or refers to a not-existing folder,
|
// if the config settings is empty or refers to a not-existing folder,
|
||||||
// ensure that the defaut path exists and return it
|
// ensure that the defaut path exists and return it
|
||||||
if (tmp.isEmpty() || !QDir(tmp).exists()) {
|
if (tmp.isEmpty() || !QDir(tmp).exists()) {
|
||||||
if (!QDir().mkpath(defaultPath))
|
if (!QDir().mkpath(defaultPath)) {
|
||||||
qCInfo(SettingsCacheLog) << "[SettingsCache] Could not create folder:" << defaultPath;
|
qCInfo(SettingsCacheLog) << "[SettingsCache] Could not create folder:" << defaultPath;
|
||||||
|
}
|
||||||
tmp = defaultPath;
|
tmp = defaultPath;
|
||||||
}
|
}
|
||||||
return tmp;
|
return tmp;
|
||||||
|
|
@ -151,8 +169,9 @@ QString SettingsCache::getSafeConfigFilePath(QString configEntry, QString defaul
|
||||||
QString tmp = settings->value(configEntry).toString();
|
QString tmp = settings->value(configEntry).toString();
|
||||||
// if the config settings is empty or refers to a not-existing file,
|
// if the config settings is empty or refers to a not-existing file,
|
||||||
// return the default Path
|
// return the default Path
|
||||||
if (!QFile::exists(tmp) || tmp.isEmpty())
|
if (!QFile::exists(tmp) || tmp.isEmpty()) {
|
||||||
tmp = std::move(defaultPath);
|
tmp = std::move(defaultPath);
|
||||||
|
}
|
||||||
return tmp;
|
return tmp;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -160,8 +179,9 @@ SettingsCache::SettingsCache()
|
||||||
{
|
{
|
||||||
// first, figure out if we are running in portable mode
|
// first, figure out if we are running in portable mode
|
||||||
isPortableBuild = QFile::exists(qApp->applicationDirPath() + "/portable.dat");
|
isPortableBuild = QFile::exists(qApp->applicationDirPath() + "/portable.dat");
|
||||||
if (isPortableBuild)
|
if (isPortableBuild) {
|
||||||
qCInfo(SettingsCacheLog) << "Portable mode enabled";
|
qCInfo(SettingsCacheLog) << "Portable mode enabled";
|
||||||
|
}
|
||||||
|
|
||||||
// define a dummy context that will be used where needed
|
// define a dummy context that will be used where needed
|
||||||
QString dummy = QT_TRANSLATE_NOOP("i18n", "English");
|
QString dummy = QT_TRANSLATE_NOOP("i18n", "English");
|
||||||
|
|
@ -179,8 +199,11 @@ SettingsCache::SettingsCache()
|
||||||
cardOverrideSettings = new CardOverrideSettings(settingsPath, this);
|
cardOverrideSettings = new CardOverrideSettings(settingsPath, this);
|
||||||
debugSettings = new DebugSettings(settingsPath, this);
|
debugSettings = new DebugSettings(settingsPath, this);
|
||||||
|
|
||||||
if (!QFile(settingsPath + "global.ini").exists())
|
cardCounterSettings = new CardCounterSettings(settingsPath, this);
|
||||||
|
|
||||||
|
if (!QFile(settingsPath + "global.ini").exists()) {
|
||||||
translateLegacySettings();
|
translateLegacySettings();
|
||||||
|
}
|
||||||
|
|
||||||
// updates - don't reorder them or their index in the settings won't match
|
// updates - don't reorder them or their index in the settings won't match
|
||||||
// append channels one by one, or msvc will add them in the wrong order.
|
// append channels one by one, or msvc will add them in the wrong order.
|
||||||
|
|
@ -189,10 +212,30 @@ SettingsCache::SettingsCache()
|
||||||
|
|
||||||
mbDownloadSpoilers = settings->value("personal/downloadspoilers", false).toBool();
|
mbDownloadSpoilers = settings->value("personal/downloadspoilers", false).toBool();
|
||||||
|
|
||||||
checkUpdatesOnStartup = settings->value("personal/startupUpdateCheck", true).toBool();
|
if (settings->contains("personal/startupUpdateCheck")) {
|
||||||
|
checkUpdatesOnStartup = settings->value("personal/startupUpdateCheck", true).toBool();
|
||||||
|
} else if (QString(VERSION_STRING).contains("custom", Qt::CaseInsensitive)) {
|
||||||
|
checkUpdatesOnStartup = false; // do not run auto updater on custom version
|
||||||
|
} else {
|
||||||
|
checkUpdatesOnStartup = true; // default to run auto updater
|
||||||
|
}
|
||||||
|
startupCardUpdateCheckPromptForUpdate =
|
||||||
|
settings->value("personal/startupCardUpdateCheckPromptForUpdate", true).toBool();
|
||||||
|
startupCardUpdateCheckAlwaysUpdate = settings->value("personal/startupCardUpdateCheckAlwaysUpdate", false).toBool();
|
||||||
|
cardUpdateCheckInterval = settings->value("personal/cardUpdateCheckInterval", 7).toInt();
|
||||||
|
lastCardUpdateCheck = settings->value("personal/lastCardUpdateCheck", QDateTime::currentDateTime().date()).toDate();
|
||||||
|
alwaysEnableNewSets = settings->value("personal/alwaysEnableNewSets", false).toBool();
|
||||||
notifyAboutUpdates = settings->value("personal/updatenotification", true).toBool();
|
notifyAboutUpdates = settings->value("personal/updatenotification", true).toBool();
|
||||||
notifyAboutNewVersion = settings->value("personal/newversionnotification", true).toBool();
|
notifyAboutNewVersion = settings->value("personal/newversionnotification", true).toBool();
|
||||||
updateReleaseChannel = settings->value("personal/updatereleasechannel", 0).toInt();
|
|
||||||
|
if (settings->contains("personal/updatereleasechannel")) {
|
||||||
|
updateReleaseChannel = settings->value("personal/updatereleasechannel").toInt();
|
||||||
|
} else if (QString(VERSION_STRING).contains("beta", Qt::CaseInsensitive)) {
|
||||||
|
// default to beta if this is a beta release
|
||||||
|
updateReleaseChannel = 1;
|
||||||
|
} else {
|
||||||
|
updateReleaseChannel = 0; // stable
|
||||||
|
}
|
||||||
|
|
||||||
lang = settings->value("personal/lang").toString();
|
lang = settings->value("personal/lang").toString();
|
||||||
keepalive = settings->value("personal/keepalive", 3).toInt();
|
keepalive = settings->value("personal/keepalive", 3).toInt();
|
||||||
|
|
@ -208,6 +251,10 @@ SettingsCache::SettingsCache()
|
||||||
|
|
||||||
themeName = settings->value("theme/name").toString();
|
themeName = settings->value("theme/name").toString();
|
||||||
|
|
||||||
|
homeTabBackgroundSource = settings->value("home/background", "themed").toString();
|
||||||
|
homeTabBackgroundShuffleFrequency = settings->value("home/background/shuffleTimer", 0).toInt();
|
||||||
|
homeTabDisplayCardName = settings->value("home/background/displayCardName", true).toBool();
|
||||||
|
|
||||||
tabVisualDeckStorageOpen = settings->value("tabs/visualDeckStorage", true).toBool();
|
tabVisualDeckStorageOpen = settings->value("tabs/visualDeckStorage", true).toBool();
|
||||||
tabServerOpen = settings->value("tabs/server", true).toBool();
|
tabServerOpen = settings->value("tabs/server", true).toBool();
|
||||||
tabAccountOpen = settings->value("tabs/account", true).toBool();
|
tabAccountOpen = settings->value("tabs/account", true).toBool();
|
||||||
|
|
@ -223,29 +270,39 @@ SettingsCache::SettingsCache()
|
||||||
settings->setValue("personal/pixmapCacheSize", pixmapCacheSize);
|
settings->setValue("personal/pixmapCacheSize", pixmapCacheSize);
|
||||||
settings->setValue("personal/picturedownloadhq", false);
|
settings->setValue("personal/picturedownloadhq", false);
|
||||||
settings->setValue("revert/pixmapCacheSize", true);
|
settings->setValue("revert/pixmapCacheSize", true);
|
||||||
} else
|
} else {
|
||||||
pixmapCacheSize = settings->value("personal/pixmapCacheSize", PIXMAPCACHE_SIZE_DEFAULT).toInt();
|
pixmapCacheSize = settings->value("personal/pixmapCacheSize", PIXMAPCACHE_SIZE_DEFAULT).toInt();
|
||||||
|
}
|
||||||
// sanity check
|
// sanity check
|
||||||
if (pixmapCacheSize < PIXMAPCACHE_SIZE_MIN || pixmapCacheSize > PIXMAPCACHE_SIZE_MAX)
|
if (pixmapCacheSize < PIXMAPCACHE_SIZE_MIN || pixmapCacheSize > PIXMAPCACHE_SIZE_MAX) {
|
||||||
pixmapCacheSize = PIXMAPCACHE_SIZE_DEFAULT;
|
pixmapCacheSize = PIXMAPCACHE_SIZE_DEFAULT;
|
||||||
|
}
|
||||||
|
|
||||||
networkCacheSize = settings->value("personal/networkCacheSize", NETWORK_CACHE_SIZE_DEFAULT).toInt();
|
networkCacheSize = settings->value("personal/networkCacheSize", NETWORK_CACHE_SIZE_DEFAULT).toInt();
|
||||||
redirectCacheTtl = settings->value("personal/redirectCacheTtl", NETWORK_REDIRECT_CACHE_TTL_DEFAULT).toInt();
|
redirectCacheTtl = settings->value("personal/redirectCacheTtl", NETWORK_REDIRECT_CACHE_TTL_DEFAULT).toInt();
|
||||||
|
cardPictureLoaderCacheMethod =
|
||||||
|
settings
|
||||||
|
->value("personal/cardPictureLoaderCacheMethod",
|
||||||
|
static_cast<int>(CardPictureLoaderCacheMethod::CacheMethod::NETWORK_CACHE))
|
||||||
|
.toInt();
|
||||||
|
localCardImageStorageNamingScheme =
|
||||||
|
settings
|
||||||
|
->value("personal/localCardImageStorageNamingScheme",
|
||||||
|
static_cast<int>(CardPictureLoaderLocalSchemes::NamingScheme::Set_Folder_Name_Set_Collector))
|
||||||
|
.toInt();
|
||||||
|
|
||||||
picDownload = settings->value("personal/picturedownload", true).toBool();
|
picDownload = settings->value("personal/picturedownload", true).toBool();
|
||||||
|
showStatusBar = settings->value("personal/showStatusBar", false).toBool();
|
||||||
|
|
||||||
mainWindowGeometry = settings->value("interface/main_window_geometry").toByteArray();
|
|
||||||
tokenDialogGeometry = settings->value("interface/token_dialog_geometry").toByteArray();
|
|
||||||
setsDialogGeometry = settings->value("interface/sets_dialog_geometry").toByteArray();
|
|
||||||
notificationsEnabled = settings->value("interface/notificationsenabled", true).toBool();
|
notificationsEnabled = settings->value("interface/notificationsenabled", true).toBool();
|
||||||
spectatorNotificationsEnabled = settings->value("interface/specnotificationsenabled", false).toBool();
|
spectatorNotificationsEnabled = settings->value("interface/specnotificationsenabled", false).toBool();
|
||||||
buddyConnectNotificationsEnabled = settings->value("interface/buddyconnectnotificationsenabled", true).toBool();
|
buddyConnectNotificationsEnabled = settings->value("interface/buddyconnectnotificationsenabled", true).toBool();
|
||||||
doubleClickToPlay = settings->value("interface/doubleclicktoplay", true).toBool();
|
doubleClickToPlay = settings->value("interface/doubleclicktoplay", true).toBool();
|
||||||
clickPlaysAllSelected = settings->value("interface/clickPlaysAllSelected", true).toBool();
|
clickPlaysAllSelected = settings->value("interface/clickPlaysAllSelected", true).toBool();
|
||||||
playToStack = settings->value("interface/playtostack", true).toBool();
|
playToStack = settings->value("interface/playtostack", true).toBool();
|
||||||
|
doNotDeleteArrowsInSubPhases = settings->value("interface/doNotDeleteArrowsInSubPhases", true).toBool();
|
||||||
startingHandSize = settings->value("interface/startinghandsize", 7).toInt();
|
startingHandSize = settings->value("interface/startinghandsize", 7).toInt();
|
||||||
annotateTokens = settings->value("interface/annotatetokens", false).toBool();
|
annotateTokens = settings->value("interface/annotatetokens", false).toBool();
|
||||||
tabGameSplitterSizes = settings->value("interface/tabgame_splittersizes").toByteArray();
|
|
||||||
knownMissingFeatures = settings->value("interface/knownmissingfeatures", "").toString();
|
knownMissingFeatures = settings->value("interface/knownmissingfeatures", "").toString();
|
||||||
useTearOffMenus = settings->value("interface/usetearoffmenus", true).toBool();
|
useTearOffMenus = settings->value("interface/usetearoffmenus", true).toBool();
|
||||||
cardViewInitialRowsMax = settings->value("interface/cardViewInitialRowsMax", 14).toInt();
|
cardViewInitialRowsMax = settings->value("interface/cardViewInitialRowsMax", 14).toInt();
|
||||||
|
|
@ -253,7 +310,11 @@ SettingsCache::SettingsCache()
|
||||||
closeEmptyCardView = settings->value("interface/closeEmptyCardView", true).toBool();
|
closeEmptyCardView = settings->value("interface/closeEmptyCardView", true).toBool();
|
||||||
focusCardViewSearchBar = settings->value("interface/focusCardViewSearchBar", true).toBool();
|
focusCardViewSearchBar = settings->value("interface/focusCardViewSearchBar", true).toBool();
|
||||||
|
|
||||||
|
showDragSelectionCount = settings->value("interface/showlassoselectioncount", true).toBool();
|
||||||
|
showTotalSelectionCount = settings->value("interface/showpersistentselectioncount", true).toBool();
|
||||||
|
|
||||||
showShortcuts = settings->value("menu/showshortcuts", true).toBool();
|
showShortcuts = settings->value("menu/showshortcuts", true).toBool();
|
||||||
|
showGameSelectorFilterToolbar = settings->value("menu/showgameselectorfiltertoolbar", true).toBool();
|
||||||
displayCardNames = settings->value("cards/displaycardnames", true).toBool();
|
displayCardNames = settings->value("cards/displaycardnames", true).toBool();
|
||||||
roundCardCorners = settings->value("cards/roundcardcorners", true).toBool();
|
roundCardCorners = settings->value("cards/roundcardcorners", true).toBool();
|
||||||
overrideAllCardArtWithPersonalPreference =
|
overrideAllCardArtWithPersonalPreference =
|
||||||
|
|
@ -271,7 +332,10 @@ SettingsCache::SettingsCache()
|
||||||
visualDeckStorageSortingOrder = settings->value("interface/visualdeckstoragesortingorder", 0).toInt();
|
visualDeckStorageSortingOrder = settings->value("interface/visualdeckstoragesortingorder", 0).toInt();
|
||||||
visualDeckStorageShowFolders = settings->value("interface/visualdeckstorageshowfolders", true).toBool();
|
visualDeckStorageShowFolders = settings->value("interface/visualdeckstorageshowfolders", true).toBool();
|
||||||
visualDeckStorageShowTagFilter = settings->value("interface/visualdeckstorageshowtagfilter", true).toBool();
|
visualDeckStorageShowTagFilter = settings->value("interface/visualdeckstorageshowtagfilter", true).toBool();
|
||||||
|
visualDeckStorageDefaultTagsList =
|
||||||
|
settings->value("interface/visualdeckstoragedefaulttagslist", defaultTags).toStringList();
|
||||||
visualDeckStorageSearchFolderNames = settings->value("interface/visualdeckstoragesearchfoldernames", true).toBool();
|
visualDeckStorageSearchFolderNames = settings->value("interface/visualdeckstoragesearchfoldernames", true).toBool();
|
||||||
|
visualDeckStorageShowColorIdentity = settings->value("interface/visualdeckstorageshowcoloridentity", true).toBool();
|
||||||
visualDeckStorageShowBannerCardComboBox =
|
visualDeckStorageShowBannerCardComboBox =
|
||||||
settings->value("interface/visualdeckstorageshowbannercardcombobox", true).toBool();
|
settings->value("interface/visualdeckstorageshowbannercardcombobox", true).toBool();
|
||||||
visualDeckStorageShowTagsOnDeckPreviews =
|
visualDeckStorageShowTagsOnDeckPreviews =
|
||||||
|
|
@ -289,10 +353,15 @@ SettingsCache::SettingsCache()
|
||||||
settings->value("interface/visualdeckstorageselectionanimation", true).toBool();
|
settings->value("interface/visualdeckstorageselectionanimation", true).toBool();
|
||||||
defaultDeckEditorType = settings->value("interface/defaultDeckEditorType", 1).toInt();
|
defaultDeckEditorType = settings->value("interface/defaultDeckEditorType", 1).toInt();
|
||||||
visualDatabaseDisplayFilterToMostRecentSetsEnabled =
|
visualDatabaseDisplayFilterToMostRecentSetsEnabled =
|
||||||
settings->value("interface/visualdatabasedisplayfiltertomostrecentsetsenabled", true).toBool();
|
settings->value("interface/visualdatabasedisplayfiltertomostrecentsetsenabled", false).toBool();
|
||||||
visualDatabaseDisplayFilterToMostRecentSetsAmount =
|
visualDatabaseDisplayFilterToMostRecentSetsAmount =
|
||||||
settings->value("interface/visualdatabasedisplayfiltertomostrecentsetsamount", 10).toInt();
|
settings->value("interface/visualdatabasedisplayfiltertomostrecentsetsamount", 10).toInt();
|
||||||
visualDeckEditorSampleHandSize = settings->value("interface/visualdeckeditorsamplehandsize", 7).toInt();
|
visualDeckEditorSampleHandSize = settings->value("interface/visualdeckeditorsamplehandsize", 7).toInt();
|
||||||
|
visualDeckEditorCardSize = settings->value("interface/visualdeckeditorcardsize", 100).toInt();
|
||||||
|
visualDatabaseDisplayCardSize = settings->value("interface/visualdatabasedisplaycardsize", 100).toInt();
|
||||||
|
edhrecCardSize = settings->value("interface/edhreccardsize", 100).toInt();
|
||||||
|
archidektPreviewSize = settings->value("interface/archidektpreviewsize", 100).toInt();
|
||||||
|
|
||||||
horizontalHand = settings->value("hand/horizontal", true).toBool();
|
horizontalHand = settings->value("hand/horizontal", true).toBool();
|
||||||
invertVerticalCoordinate = settings->value("table/invert_vertical", false).toBool();
|
invertVerticalCoordinate = settings->value("table/invert_vertical", false).toBool();
|
||||||
minPlayersForMultiColumnLayout = settings->value("interface/min_players_multicolumn", 4).toInt();
|
minPlayersForMultiColumnLayout = settings->value("interface/min_players_multicolumn", 4).toInt();
|
||||||
|
|
@ -319,6 +388,7 @@ SettingsCache::SettingsCache()
|
||||||
|
|
||||||
ignoreUnregisteredUsers = settings->value("chat/ignore_unregistered", false).toBool();
|
ignoreUnregisteredUsers = settings->value("chat/ignore_unregistered", false).toBool();
|
||||||
ignoreUnregisteredUserMessages = settings->value("chat/ignore_unregistered_messages", false).toBool();
|
ignoreUnregisteredUserMessages = settings->value("chat/ignore_unregistered_messages", false).toBool();
|
||||||
|
ignoreNonBuddyUserMessages = settings->value("chat/ignore_nonbuddy_messages", false).toBool();
|
||||||
|
|
||||||
scaleCards = settings->value("cards/scaleCards", true).toBool();
|
scaleCards = settings->value("cards/scaleCards", true).toBool();
|
||||||
verticalCardOverlapPercent = settings->value("cards/verticalCardOverlapPercent", 33).toInt();
|
verticalCardOverlapPercent = settings->value("cards/verticalCardOverlapPercent", 33).toInt();
|
||||||
|
|
@ -343,7 +413,15 @@ SettingsCache::SettingsCache()
|
||||||
spectatorsCanSeeEverything = settings->value("game/spectatorscanseeeverything", false).toBool();
|
spectatorsCanSeeEverything = settings->value("game/spectatorscanseeeverything", false).toBool();
|
||||||
createGameAsSpectator = settings->value("game/creategameasspectator", false).toBool();
|
createGameAsSpectator = settings->value("game/creategameasspectator", false).toBool();
|
||||||
defaultStartingLifeTotal = settings->value("game/defaultstartinglifetotal", 20).toInt();
|
defaultStartingLifeTotal = settings->value("game/defaultstartinglifetotal", 20).toInt();
|
||||||
|
shareDecklistsOnLoad = settings->value("game/sharedecklistsonload", false).toBool();
|
||||||
rememberGameSettings = settings->value("game/remembergamesettings", true).toBool();
|
rememberGameSettings = settings->value("game/remembergamesettings", true).toBool();
|
||||||
|
|
||||||
|
// Local game settings use "localgameoptions/" prefix to keep them separate
|
||||||
|
// from server game settings which use "game/" prefix
|
||||||
|
localGameRememberSettings = settings->value("localgameoptions/remembersettings", false).toBool();
|
||||||
|
localGameMaxPlayers = settings->value("localgameoptions/maxplayers", 1).toInt();
|
||||||
|
localGameStartingLifeTotal = settings->value("localgameoptions/startinglifetotal", 20).toInt();
|
||||||
|
|
||||||
clientID = settings->value("personal/clientid", CLIENT_INFO_NOT_SET).toString();
|
clientID = settings->value("personal/clientid", CLIENT_INFO_NOT_SET).toString();
|
||||||
clientVersion = settings->value("personal/clientversion", CLIENT_INFO_NOT_SET).toString();
|
clientVersion = settings->value("personal/clientversion", CLIENT_INFO_NOT_SET).toString();
|
||||||
}
|
}
|
||||||
|
|
@ -537,6 +615,27 @@ void SettingsCache::setThemeName(const QString &_themeName)
|
||||||
emit themeChanged();
|
emit themeChanged();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void SettingsCache::setHomeTabBackgroundSource(const QString &_backgroundSource)
|
||||||
|
{
|
||||||
|
homeTabBackgroundSource = _backgroundSource;
|
||||||
|
settings->setValue("home/background", homeTabBackgroundSource);
|
||||||
|
emit homeTabBackgroundSourceChanged();
|
||||||
|
}
|
||||||
|
|
||||||
|
void SettingsCache::setHomeTabBackgroundShuffleFrequency(int _frequency)
|
||||||
|
{
|
||||||
|
homeTabBackgroundShuffleFrequency = _frequency;
|
||||||
|
settings->setValue("home/background/shuffleTimer", homeTabBackgroundShuffleFrequency);
|
||||||
|
emit homeTabBackgroundShuffleFrequencyChanged();
|
||||||
|
}
|
||||||
|
|
||||||
|
void SettingsCache::setHomeTabDisplayCardName(QT_STATE_CHANGED_T _displayCardName)
|
||||||
|
{
|
||||||
|
homeTabDisplayCardName = static_cast<bool>(_displayCardName);
|
||||||
|
settings->setValue("home/background/displayCardName", homeTabDisplayCardName);
|
||||||
|
emit homeTabDisplayCardNameChanged();
|
||||||
|
}
|
||||||
|
|
||||||
void SettingsCache::setTabVisualDeckStorageOpen(bool value)
|
void SettingsCache::setTabVisualDeckStorageOpen(bool value)
|
||||||
{
|
{
|
||||||
tabVisualDeckStorageOpen = value;
|
tabVisualDeckStorageOpen = value;
|
||||||
|
|
@ -586,6 +685,13 @@ void SettingsCache::setPicDownload(QT_STATE_CHANGED_T _picDownload)
|
||||||
emit picDownloadChanged();
|
emit picDownloadChanged();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void SettingsCache::setShowStatusBar(bool value)
|
||||||
|
{
|
||||||
|
showStatusBar = value;
|
||||||
|
settings->setValue("personal/showStatusBar", showStatusBar);
|
||||||
|
emit showStatusBarChanged(value);
|
||||||
|
}
|
||||||
|
|
||||||
void SettingsCache::setNotificationsEnabled(QT_STATE_CHANGED_T _notificationsEnabled)
|
void SettingsCache::setNotificationsEnabled(QT_STATE_CHANGED_T _notificationsEnabled)
|
||||||
{
|
{
|
||||||
notificationsEnabled = static_cast<bool>(_notificationsEnabled);
|
notificationsEnabled = static_cast<bool>(_notificationsEnabled);
|
||||||
|
|
@ -622,6 +728,12 @@ void SettingsCache::setPlayToStack(QT_STATE_CHANGED_T _playToStack)
|
||||||
settings->setValue("interface/playtostack", playToStack);
|
settings->setValue("interface/playtostack", playToStack);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void SettingsCache::setDoNotDeleteArrowsInSubPhases(QT_STATE_CHANGED_T _doNotDeleteArrowsInSubPhases)
|
||||||
|
{
|
||||||
|
doNotDeleteArrowsInSubPhases = static_cast<bool>(_doNotDeleteArrowsInSubPhases);
|
||||||
|
settings->setValue("interface/doNotDeleteArrowsInSubPhases", doNotDeleteArrowsInSubPhases);
|
||||||
|
}
|
||||||
|
|
||||||
void SettingsCache::setStartingHandSize(int _startingHandSize)
|
void SettingsCache::setStartingHandSize(int _startingHandSize)
|
||||||
{
|
{
|
||||||
startingHandSize = _startingHandSize;
|
startingHandSize = _startingHandSize;
|
||||||
|
|
@ -634,18 +746,19 @@ void SettingsCache::setAnnotateTokens(QT_STATE_CHANGED_T _annotateTokens)
|
||||||
settings->setValue("interface/annotatetokens", annotateTokens);
|
settings->setValue("interface/annotatetokens", annotateTokens);
|
||||||
}
|
}
|
||||||
|
|
||||||
void SettingsCache::setTabGameSplitterSizes(const QByteArray &_tabGameSplitterSizes)
|
|
||||||
{
|
|
||||||
tabGameSplitterSizes = _tabGameSplitterSizes;
|
|
||||||
settings->setValue("interface/tabgame_splittersizes", tabGameSplitterSizes);
|
|
||||||
}
|
|
||||||
|
|
||||||
void SettingsCache::setShowShortcuts(QT_STATE_CHANGED_T _showShortcuts)
|
void SettingsCache::setShowShortcuts(QT_STATE_CHANGED_T _showShortcuts)
|
||||||
{
|
{
|
||||||
showShortcuts = static_cast<bool>(_showShortcuts);
|
showShortcuts = static_cast<bool>(_showShortcuts);
|
||||||
settings->setValue("menu/showshortcuts", showShortcuts);
|
settings->setValue("menu/showshortcuts", showShortcuts);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void SettingsCache::setShowGameSelectorFilterToolbar(QT_STATE_CHANGED_T _showGameSelectorFilterToolbar)
|
||||||
|
{
|
||||||
|
showGameSelectorFilterToolbar = static_cast<bool>(_showGameSelectorFilterToolbar);
|
||||||
|
settings->setValue("menu/showgameselectorfiltertoolbar", showGameSelectorFilterToolbar);
|
||||||
|
emit showGameSelectorFilterToolbarChanged(showGameSelectorFilterToolbar);
|
||||||
|
}
|
||||||
|
|
||||||
void SettingsCache::setDisplayCardNames(QT_STATE_CHANGED_T _displayCardNames)
|
void SettingsCache::setDisplayCardNames(QT_STATE_CHANGED_T _displayCardNames)
|
||||||
{
|
{
|
||||||
displayCardNames = static_cast<bool>(_displayCardNames);
|
displayCardNames = static_cast<bool>(_displayCardNames);
|
||||||
|
|
@ -683,8 +796,9 @@ void SettingsCache::setPrintingSelectorCardSize(int _printingSelectorCardSize)
|
||||||
|
|
||||||
void SettingsCache::setIncludeRebalancedCards(bool _includeRebalancedCards)
|
void SettingsCache::setIncludeRebalancedCards(bool _includeRebalancedCards)
|
||||||
{
|
{
|
||||||
if (includeRebalancedCards == _includeRebalancedCards)
|
if (includeRebalancedCards == _includeRebalancedCards) {
|
||||||
return;
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
includeRebalancedCards = _includeRebalancedCards;
|
includeRebalancedCards = _includeRebalancedCards;
|
||||||
settings->setValue("cards/includerebalancedcards", includeRebalancedCards);
|
settings->setValue("cards/includerebalancedcards", includeRebalancedCards);
|
||||||
|
|
@ -731,12 +845,26 @@ void SettingsCache::setVisualDeckStorageShowTagFilter(QT_STATE_CHANGED_T _showTa
|
||||||
emit visualDeckStorageShowTagFilterChanged(visualDeckStorageShowTagFilter);
|
emit visualDeckStorageShowTagFilterChanged(visualDeckStorageShowTagFilter);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void SettingsCache::setVisualDeckStorageDefaultTagsList(QStringList _defaultTagsList)
|
||||||
|
{
|
||||||
|
visualDeckStorageDefaultTagsList = _defaultTagsList;
|
||||||
|
settings->setValue("interface/visualdeckstoragedefaulttagslist", visualDeckStorageDefaultTagsList);
|
||||||
|
emit visualDeckStorageDefaultTagsListChanged();
|
||||||
|
}
|
||||||
|
|
||||||
void SettingsCache::setVisualDeckStorageSearchFolderNames(QT_STATE_CHANGED_T value)
|
void SettingsCache::setVisualDeckStorageSearchFolderNames(QT_STATE_CHANGED_T value)
|
||||||
{
|
{
|
||||||
visualDeckStorageSearchFolderNames = value;
|
visualDeckStorageSearchFolderNames = value;
|
||||||
settings->setValue("interface/visualdeckstoragesearchfoldernames", visualDeckStorageSearchFolderNames);
|
settings->setValue("interface/visualdeckstoragesearchfoldernames", visualDeckStorageSearchFolderNames);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void SettingsCache::setVisualDeckStorageShowColorIdentity(QT_STATE_CHANGED_T value)
|
||||||
|
{
|
||||||
|
visualDeckStorageShowColorIdentity = value;
|
||||||
|
settings->setValue("interface/visualdeckstorageshowcoloridentity", visualDeckStorageShowColorIdentity);
|
||||||
|
emit visualDeckStorageShowColorIdentityChanged(visualDeckStorageShowColorIdentity);
|
||||||
|
}
|
||||||
|
|
||||||
void SettingsCache::setVisualDeckStorageShowBannerCardComboBox(QT_STATE_CHANGED_T _showBannerCardComboBox)
|
void SettingsCache::setVisualDeckStorageShowBannerCardComboBox(QT_STATE_CHANGED_T _showBannerCardComboBox)
|
||||||
{
|
{
|
||||||
visualDeckStorageShowBannerCardComboBox = _showBannerCardComboBox;
|
visualDeckStorageShowBannerCardComboBox = _showBannerCardComboBox;
|
||||||
|
|
@ -807,6 +935,34 @@ void SettingsCache::setVisualDeckStorageSelectionAnimation(QT_STATE_CHANGED_T va
|
||||||
emit visualDeckStorageSelectionAnimationChanged(visualDeckStorageSelectionAnimation);
|
emit visualDeckStorageSelectionAnimationChanged(visualDeckStorageSelectionAnimation);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void SettingsCache::setVisualDeckEditorCardSize(int _visualDeckEditorCardSize)
|
||||||
|
{
|
||||||
|
visualDeckEditorCardSize = _visualDeckEditorCardSize;
|
||||||
|
settings->setValue("interface/visualdeckeditorcardsize", visualDeckEditorCardSize);
|
||||||
|
emit visualDeckEditorCardSizeChanged();
|
||||||
|
}
|
||||||
|
|
||||||
|
void SettingsCache::setVisualDatabaseDisplayCardSize(int _visualDatabaseDisplayCardSize)
|
||||||
|
{
|
||||||
|
visualDatabaseDisplayCardSize = _visualDatabaseDisplayCardSize;
|
||||||
|
settings->setValue("interface/visualdatabasedisplaycardsize", visualDatabaseDisplayCardSize);
|
||||||
|
emit visualDatabaseDisplayCardSizeChanged();
|
||||||
|
}
|
||||||
|
|
||||||
|
void SettingsCache::setEDHRecCardSize(int _edhrecCardSize)
|
||||||
|
{
|
||||||
|
edhrecCardSize = _edhrecCardSize;
|
||||||
|
settings->setValue("interface/edhreccardsize", edhrecCardSize);
|
||||||
|
emit edhRecCardSizeChanged();
|
||||||
|
}
|
||||||
|
|
||||||
|
void SettingsCache::setArchidektPreviewCardSize(int _archidektPreviewCardSize)
|
||||||
|
{
|
||||||
|
archidektPreviewSize = _archidektPreviewCardSize;
|
||||||
|
settings->setValue("interface/archidektpreviewsize", archidektPreviewSize);
|
||||||
|
emit archidektPreviewSizeChanged();
|
||||||
|
}
|
||||||
|
|
||||||
void SettingsCache::setDefaultDeckEditorType(int value)
|
void SettingsCache::setDefaultDeckEditorType(int value)
|
||||||
{
|
{
|
||||||
defaultDeckEditorType = value;
|
defaultDeckEditorType = value;
|
||||||
|
|
@ -962,22 +1118,10 @@ void SettingsCache::setIgnoreUnregisteredUserMessages(QT_STATE_CHANGED_T _ignore
|
||||||
settings->setValue("chat/ignore_unregistered_messages", ignoreUnregisteredUserMessages);
|
settings->setValue("chat/ignore_unregistered_messages", ignoreUnregisteredUserMessages);
|
||||||
}
|
}
|
||||||
|
|
||||||
void SettingsCache::setMainWindowGeometry(const QByteArray &_mainWindowGeometry)
|
void SettingsCache::setIgnoreNonBuddyUserMessages(QT_STATE_CHANGED_T _ignoreNonBuddyUserMessages)
|
||||||
{
|
{
|
||||||
mainWindowGeometry = _mainWindowGeometry;
|
ignoreNonBuddyUserMessages = static_cast<bool>(_ignoreNonBuddyUserMessages);
|
||||||
settings->setValue("interface/main_window_geometry", mainWindowGeometry);
|
settings->setValue("chat/ignore_nonbuddy_messages", ignoreNonBuddyUserMessages);
|
||||||
}
|
|
||||||
|
|
||||||
void SettingsCache::setTokenDialogGeometry(const QByteArray &_tokenDialogGeometry)
|
|
||||||
{
|
|
||||||
tokenDialogGeometry = _tokenDialogGeometry;
|
|
||||||
settings->setValue("interface/token_dialog_geometry", tokenDialogGeometry);
|
|
||||||
}
|
|
||||||
|
|
||||||
void SettingsCache::setSetsDialogGeometry(const QByteArray &_setsDialogGeometry)
|
|
||||||
{
|
|
||||||
setsDialogGeometry = _setsDialogGeometry;
|
|
||||||
settings->setValue("interface/sets_dialog_geometry", setsDialogGeometry);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void SettingsCache::setPixmapCacheSize(const int _pixmapCacheSize)
|
void SettingsCache::setPixmapCacheSize(const int _pixmapCacheSize)
|
||||||
|
|
@ -987,6 +1131,13 @@ void SettingsCache::setPixmapCacheSize(const int _pixmapCacheSize)
|
||||||
emit pixmapCacheSizeChanged(pixmapCacheSize);
|
emit pixmapCacheSizeChanged(pixmapCacheSize);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void SettingsCache::setCardImageCacheMethod(const CardPictureLoaderCacheMethod::CacheMethod _cardImageCachingMethod)
|
||||||
|
{
|
||||||
|
cardPictureLoaderCacheMethod = static_cast<int>(_cardImageCachingMethod);
|
||||||
|
settings->setValue("personal/cardPictureLoaderCacheMethod", cardPictureLoaderCacheMethod);
|
||||||
|
emit cardPictureLoaderCacheMethodChanged(cardPictureLoaderCacheMethod);
|
||||||
|
}
|
||||||
|
|
||||||
void SettingsCache::setNetworkCacheSizeInMB(const int _networkCacheSize)
|
void SettingsCache::setNetworkCacheSizeInMB(const int _networkCacheSize)
|
||||||
{
|
{
|
||||||
networkCacheSize = _networkCacheSize;
|
networkCacheSize = _networkCacheSize;
|
||||||
|
|
@ -1001,6 +1152,14 @@ void SettingsCache::setNetworkRedirectCacheTtl(const int _redirectCacheTtl)
|
||||||
emit redirectCacheTtlChanged(redirectCacheTtl);
|
emit redirectCacheTtlChanged(redirectCacheTtl);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void SettingsCache::setLocalCardImageStorageNamingScheme(
|
||||||
|
const CardPictureLoaderLocalSchemes::NamingScheme _localCardImageStorageNamingScheme)
|
||||||
|
{
|
||||||
|
localCardImageStorageNamingScheme = static_cast<int>(_localCardImageStorageNamingScheme);
|
||||||
|
settings->setValue("personal/localCardImageStorageNamingScheme", localCardImageStorageNamingScheme);
|
||||||
|
emit localCardImageStorageNamingSchemeChanged(localCardImageStorageNamingScheme);
|
||||||
|
}
|
||||||
|
|
||||||
void SettingsCache::setClientID(const QString &_clientID)
|
void SettingsCache::setClientID(const QString &_clientID)
|
||||||
{
|
{
|
||||||
clientID = _clientID;
|
clientID = _clientID;
|
||||||
|
|
@ -1015,257 +1174,21 @@ void SettingsCache::setClientVersion(const QString &_clientVersion)
|
||||||
|
|
||||||
QStringList SettingsCache::getCountries() const
|
QStringList SettingsCache::getCountries() const
|
||||||
{
|
{
|
||||||
static QStringList countries = QStringList() << "ad"
|
static const QStringList countries = {
|
||||||
<< "ae"
|
"ad", "ae", "af", "ag", "ai", "al", "am", "ao", "aq", "ar", "as", "at", "au", "aw", "ax", "az", "ba", "bb",
|
||||||
<< "af"
|
"bd", "be", "bf", "bg", "bh", "bi", "bj", "bl", "bm", "bn", "bo", "bq", "br", "bs", "bt", "bv", "bw", "by",
|
||||||
<< "ag"
|
"bz", "ca", "cc", "cd", "cf", "cg", "ch", "ci", "ck", "cl", "cm", "cn", "co", "cr", "cu", "cv", "cw", "cx",
|
||||||
<< "ai"
|
"cy", "cz", "de", "dj", "dk", "dm", "do", "dz", "ec", "ee", "eg", "eh", "er", "es", "et", "eu", "fi", "fj",
|
||||||
<< "al"
|
"fk", "fm", "fo", "fr", "ga", "gb", "gd", "ge", "gf", "gg", "gh", "gi", "gl", "gm", "gn", "gp", "gq", "gr",
|
||||||
<< "am"
|
"gs", "gt", "gu", "gw", "gy", "hk", "hm", "hn", "hr", "ht", "hu", "id", "ie", "il", "im", "in", "io", "iq",
|
||||||
<< "ao"
|
"ir", "is", "it", "je", "jm", "jo", "jp", "ke", "kg", "kh", "ki", "km", "kn", "kp", "kr", "kw", "ky", "kz",
|
||||||
<< "aq"
|
"la", "lb", "lc", "li", "lk", "lr", "ls", "lt", "lu", "lv", "ly", "ma", "mc", "md", "me", "mf", "mg", "mh",
|
||||||
<< "ar"
|
"mk", "ml", "mm", "mn", "mo", "mp", "mq", "mr", "ms", "mt", "mu", "mv", "mw", "mx", "my", "mz", "na", "nc",
|
||||||
<< "as"
|
"ne", "nf", "ng", "ni", "nl", "no", "np", "nr", "nu", "nz", "om", "pa", "pe", "pf", "pg", "ph", "pk", "pl",
|
||||||
<< "at"
|
"pm", "pn", "pr", "ps", "pt", "pw", "py", "qa", "re", "ro", "rs", "ru", "rw", "sa", "sb", "sc", "sd", "se",
|
||||||
<< "au"
|
"sg", "sh", "si", "sj", "sk", "sl", "sm", "sn", "so", "sr", "ss", "st", "sv", "sx", "sy", "sz", "tc", "td",
|
||||||
<< "aw"
|
"tf", "tg", "th", "tj", "tk", "tl", "tm", "tn", "to", "tr", "tt", "tv", "tw", "tz", "ua", "ug", "um", "us",
|
||||||
<< "ax"
|
"uy", "uz", "va", "vc", "ve", "vg", "vi", "vn", "vu", "wf", "ws", "xk", "ye", "yt", "za", "zm", "zw"};
|
||||||
<< "az"
|
|
||||||
<< "ba"
|
|
||||||
<< "bb"
|
|
||||||
<< "bd"
|
|
||||||
<< "be"
|
|
||||||
<< "bf"
|
|
||||||
<< "bg"
|
|
||||||
<< "bh"
|
|
||||||
<< "bi"
|
|
||||||
<< "bj"
|
|
||||||
<< "bl"
|
|
||||||
<< "bm"
|
|
||||||
<< "bn"
|
|
||||||
<< "bo"
|
|
||||||
<< "bq"
|
|
||||||
<< "br"
|
|
||||||
<< "bs"
|
|
||||||
<< "bt"
|
|
||||||
<< "bv"
|
|
||||||
<< "bw"
|
|
||||||
<< "by"
|
|
||||||
<< "bz"
|
|
||||||
<< "ca"
|
|
||||||
<< "cc"
|
|
||||||
<< "cd"
|
|
||||||
<< "cf"
|
|
||||||
<< "cg"
|
|
||||||
<< "ch"
|
|
||||||
<< "ci"
|
|
||||||
<< "ck"
|
|
||||||
<< "cl"
|
|
||||||
<< "cm"
|
|
||||||
<< "cn"
|
|
||||||
<< "co"
|
|
||||||
<< "cr"
|
|
||||||
<< "cu"
|
|
||||||
<< "cv"
|
|
||||||
<< "cw"
|
|
||||||
<< "cx"
|
|
||||||
<< "cy"
|
|
||||||
<< "cz"
|
|
||||||
<< "de"
|
|
||||||
<< "dj"
|
|
||||||
<< "dk"
|
|
||||||
<< "dm"
|
|
||||||
<< "do"
|
|
||||||
<< "dz"
|
|
||||||
<< "ec"
|
|
||||||
<< "ee"
|
|
||||||
<< "eg"
|
|
||||||
<< "eh"
|
|
||||||
<< "er"
|
|
||||||
<< "es"
|
|
||||||
<< "et"
|
|
||||||
<< "eu"
|
|
||||||
<< "fi"
|
|
||||||
<< "fj"
|
|
||||||
<< "fk"
|
|
||||||
<< "fm"
|
|
||||||
<< "fo"
|
|
||||||
<< "fr"
|
|
||||||
<< "ga"
|
|
||||||
<< "gb"
|
|
||||||
<< "gd"
|
|
||||||
<< "ge"
|
|
||||||
<< "gf"
|
|
||||||
<< "gg"
|
|
||||||
<< "gh"
|
|
||||||
<< "gi"
|
|
||||||
<< "gl"
|
|
||||||
<< "gm"
|
|
||||||
<< "gn"
|
|
||||||
<< "gp"
|
|
||||||
<< "gq"
|
|
||||||
<< "gr"
|
|
||||||
<< "gs"
|
|
||||||
<< "gt"
|
|
||||||
<< "gu"
|
|
||||||
<< "gw"
|
|
||||||
<< "gy"
|
|
||||||
<< "hk"
|
|
||||||
<< "hm"
|
|
||||||
<< "hn"
|
|
||||||
<< "hr"
|
|
||||||
<< "ht"
|
|
||||||
<< "hu"
|
|
||||||
<< "id"
|
|
||||||
<< "ie"
|
|
||||||
<< "il"
|
|
||||||
<< "im"
|
|
||||||
<< "in"
|
|
||||||
<< "io"
|
|
||||||
<< "iq"
|
|
||||||
<< "ir"
|
|
||||||
<< "is"
|
|
||||||
<< "it"
|
|
||||||
<< "je"
|
|
||||||
<< "jm"
|
|
||||||
<< "jo"
|
|
||||||
<< "jp"
|
|
||||||
<< "ke"
|
|
||||||
<< "kg"
|
|
||||||
<< "kh"
|
|
||||||
<< "ki"
|
|
||||||
<< "km"
|
|
||||||
<< "kn"
|
|
||||||
<< "kp"
|
|
||||||
<< "kr"
|
|
||||||
<< "kw"
|
|
||||||
<< "ky"
|
|
||||||
<< "kz"
|
|
||||||
<< "la"
|
|
||||||
<< "lb"
|
|
||||||
<< "lc"
|
|
||||||
<< "li"
|
|
||||||
<< "lk"
|
|
||||||
<< "lr"
|
|
||||||
<< "ls"
|
|
||||||
<< "lt"
|
|
||||||
<< "lu"
|
|
||||||
<< "lv"
|
|
||||||
<< "ly"
|
|
||||||
<< "ma"
|
|
||||||
<< "mc"
|
|
||||||
<< "md"
|
|
||||||
<< "me"
|
|
||||||
<< "mf"
|
|
||||||
<< "mg"
|
|
||||||
<< "mh"
|
|
||||||
<< "mk"
|
|
||||||
<< "ml"
|
|
||||||
<< "mm"
|
|
||||||
<< "mn"
|
|
||||||
<< "mo"
|
|
||||||
<< "mp"
|
|
||||||
<< "mq"
|
|
||||||
<< "mr"
|
|
||||||
<< "ms"
|
|
||||||
<< "mt"
|
|
||||||
<< "mu"
|
|
||||||
<< "mv"
|
|
||||||
<< "mw"
|
|
||||||
<< "mx"
|
|
||||||
<< "my"
|
|
||||||
<< "mz"
|
|
||||||
<< "na"
|
|
||||||
<< "nc"
|
|
||||||
<< "ne"
|
|
||||||
<< "nf"
|
|
||||||
<< "ng"
|
|
||||||
<< "ni"
|
|
||||||
<< "nl"
|
|
||||||
<< "no"
|
|
||||||
<< "np"
|
|
||||||
<< "nr"
|
|
||||||
<< "nu"
|
|
||||||
<< "nz"
|
|
||||||
<< "om"
|
|
||||||
<< "pa"
|
|
||||||
<< "pe"
|
|
||||||
<< "pf"
|
|
||||||
<< "pg"
|
|
||||||
<< "ph"
|
|
||||||
<< "pk"
|
|
||||||
<< "pl"
|
|
||||||
<< "pm"
|
|
||||||
<< "pn"
|
|
||||||
<< "pr"
|
|
||||||
<< "ps"
|
|
||||||
<< "pt"
|
|
||||||
<< "pw"
|
|
||||||
<< "py"
|
|
||||||
<< "qa"
|
|
||||||
<< "re"
|
|
||||||
<< "ro"
|
|
||||||
<< "rs"
|
|
||||||
<< "ru"
|
|
||||||
<< "rw"
|
|
||||||
<< "sa"
|
|
||||||
<< "sb"
|
|
||||||
<< "sc"
|
|
||||||
<< "sd"
|
|
||||||
<< "se"
|
|
||||||
<< "sg"
|
|
||||||
<< "sh"
|
|
||||||
<< "si"
|
|
||||||
<< "sj"
|
|
||||||
<< "sk"
|
|
||||||
<< "sl"
|
|
||||||
<< "sm"
|
|
||||||
<< "sn"
|
|
||||||
<< "so"
|
|
||||||
<< "sr"
|
|
||||||
<< "ss"
|
|
||||||
<< "st"
|
|
||||||
<< "sv"
|
|
||||||
<< "sx"
|
|
||||||
<< "sy"
|
|
||||||
<< "sz"
|
|
||||||
<< "tc"
|
|
||||||
<< "td"
|
|
||||||
<< "tf"
|
|
||||||
<< "tg"
|
|
||||||
<< "th"
|
|
||||||
<< "tj"
|
|
||||||
<< "tk"
|
|
||||||
<< "tl"
|
|
||||||
<< "tm"
|
|
||||||
<< "tn"
|
|
||||||
<< "to"
|
|
||||||
<< "tr"
|
|
||||||
<< "tt"
|
|
||||||
<< "tv"
|
|
||||||
<< "tw"
|
|
||||||
<< "tz"
|
|
||||||
<< "ua"
|
|
||||||
<< "ug"
|
|
||||||
<< "um"
|
|
||||||
<< "us"
|
|
||||||
<< "uy"
|
|
||||||
<< "uz"
|
|
||||||
<< "va"
|
|
||||||
<< "vc"
|
|
||||||
<< "ve"
|
|
||||||
<< "vg"
|
|
||||||
<< "vi"
|
|
||||||
<< "vn"
|
|
||||||
<< "vu"
|
|
||||||
<< "wf"
|
|
||||||
<< "ws"
|
|
||||||
<< "xk"
|
|
||||||
<< "ye"
|
|
||||||
<< "yt"
|
|
||||||
<< "za"
|
|
||||||
<< "zm"
|
|
||||||
<< "zw";
|
|
||||||
|
|
||||||
return countries;
|
return countries;
|
||||||
}
|
}
|
||||||
|
|
@ -1334,7 +1257,13 @@ void SettingsCache::setDefaultStartingLifeTotal(const int _defaultStartingLifeTo
|
||||||
{
|
{
|
||||||
defaultStartingLifeTotal = _defaultStartingLifeTotal;
|
defaultStartingLifeTotal = _defaultStartingLifeTotal;
|
||||||
settings->setValue("game/defaultstartinglifetotal", defaultStartingLifeTotal);
|
settings->setValue("game/defaultstartinglifetotal", defaultStartingLifeTotal);
|
||||||
};
|
}
|
||||||
|
|
||||||
|
void SettingsCache::setShareDecklistsOnLoad(const bool _shareDecklistsOnLoad)
|
||||||
|
{
|
||||||
|
shareDecklistsOnLoad = _shareDecklistsOnLoad;
|
||||||
|
settings->setValue("game/sharedecklistsonload", shareDecklistsOnLoad);
|
||||||
|
}
|
||||||
|
|
||||||
void SettingsCache::setCheckUpdatesOnStartup(QT_STATE_CHANGED_T value)
|
void SettingsCache::setCheckUpdatesOnStartup(QT_STATE_CHANGED_T value)
|
||||||
{
|
{
|
||||||
|
|
@ -1342,12 +1271,60 @@ void SettingsCache::setCheckUpdatesOnStartup(QT_STATE_CHANGED_T value)
|
||||||
settings->setValue("personal/startupUpdateCheck", checkUpdatesOnStartup);
|
settings->setValue("personal/startupUpdateCheck", checkUpdatesOnStartup);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void SettingsCache::setStartupCardUpdateCheckPromptForUpdate(bool value)
|
||||||
|
{
|
||||||
|
startupCardUpdateCheckPromptForUpdate = value;
|
||||||
|
settings->setValue("personal/startupCardUpdateCheckPromptForUpdate", startupCardUpdateCheckPromptForUpdate);
|
||||||
|
}
|
||||||
|
|
||||||
|
void SettingsCache::setStartupCardUpdateCheckAlwaysUpdate(bool value)
|
||||||
|
{
|
||||||
|
startupCardUpdateCheckAlwaysUpdate = value;
|
||||||
|
settings->setValue("personal/startupCardUpdateCheckAlwaysUpdate", startupCardUpdateCheckAlwaysUpdate);
|
||||||
|
}
|
||||||
|
|
||||||
|
void SettingsCache::setCardUpdateCheckInterval(int value)
|
||||||
|
{
|
||||||
|
cardUpdateCheckInterval = value;
|
||||||
|
settings->setValue("personal/cardUpdateCheckInterval", cardUpdateCheckInterval);
|
||||||
|
}
|
||||||
|
|
||||||
|
void SettingsCache::setLastCardUpdateCheck(QDate value)
|
||||||
|
{
|
||||||
|
lastCardUpdateCheck = value;
|
||||||
|
settings->setValue("personal/lastCardUpdateCheck", lastCardUpdateCheck);
|
||||||
|
}
|
||||||
|
|
||||||
|
void SettingsCache::setAlwaysEnableNewSets(bool value)
|
||||||
|
{
|
||||||
|
alwaysEnableNewSets = value;
|
||||||
|
settings->setValue("personal/alwaysEnableNewSets", alwaysEnableNewSets);
|
||||||
|
}
|
||||||
|
|
||||||
void SettingsCache::setRememberGameSettings(const bool _rememberGameSettings)
|
void SettingsCache::setRememberGameSettings(const bool _rememberGameSettings)
|
||||||
{
|
{
|
||||||
rememberGameSettings = _rememberGameSettings;
|
rememberGameSettings = _rememberGameSettings;
|
||||||
settings->setValue("game/remembergamesettings", rememberGameSettings);
|
settings->setValue("game/remembergamesettings", rememberGameSettings);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void SettingsCache::setLocalGameRememberSettings(bool value)
|
||||||
|
{
|
||||||
|
localGameRememberSettings = value;
|
||||||
|
settings->setValue("localgameoptions/remembersettings", value);
|
||||||
|
}
|
||||||
|
|
||||||
|
void SettingsCache::setLocalGameMaxPlayers(int value)
|
||||||
|
{
|
||||||
|
localGameMaxPlayers = value;
|
||||||
|
settings->setValue("localgameoptions/maxplayers", value);
|
||||||
|
}
|
||||||
|
|
||||||
|
void SettingsCache::setLocalGameStartingLifeTotal(int value)
|
||||||
|
{
|
||||||
|
localGameStartingLifeTotal = value;
|
||||||
|
settings->setValue("localgameoptions/startinglifetotal", value);
|
||||||
|
}
|
||||||
|
|
||||||
void SettingsCache::setNotifyAboutUpdate(QT_STATE_CHANGED_T _notifyaboutupdate)
|
void SettingsCache::setNotifyAboutUpdate(QT_STATE_CHANGED_T _notifyaboutupdate)
|
||||||
{
|
{
|
||||||
notifyAboutUpdates = static_cast<bool>(_notifyaboutupdate);
|
notifyAboutUpdates = static_cast<bool>(_notifyaboutupdate);
|
||||||
|
|
@ -1381,14 +1358,27 @@ void SettingsCache::setMaxFontSize(int _max)
|
||||||
|
|
||||||
void SettingsCache::setRoundCardCorners(bool _roundCardCorners)
|
void SettingsCache::setRoundCardCorners(bool _roundCardCorners)
|
||||||
{
|
{
|
||||||
if (_roundCardCorners == roundCardCorners)
|
if (_roundCardCorners == roundCardCorners) {
|
||||||
return;
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
roundCardCorners = _roundCardCorners;
|
roundCardCorners = _roundCardCorners;
|
||||||
settings->setValue("cards/roundcardcorners", _roundCardCorners);
|
settings->setValue("cards/roundcardcorners", _roundCardCorners);
|
||||||
emit roundCardCornersChanged(roundCardCorners);
|
emit roundCardCornersChanged(roundCardCorners);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void SettingsCache::setShowDragSelectionCount(QT_STATE_CHANGED_T _showDragSelectionCount)
|
||||||
|
{
|
||||||
|
showDragSelectionCount = static_cast<bool>(_showDragSelectionCount);
|
||||||
|
settings->setValue("interface/showlassoselectioncount", showDragSelectionCount);
|
||||||
|
}
|
||||||
|
|
||||||
|
void SettingsCache::setShowTotalSelectionCount(QT_STATE_CHANGED_T _showTotalSelectionCount)
|
||||||
|
{
|
||||||
|
showTotalSelectionCount = static_cast<bool>(_showTotalSelectionCount);
|
||||||
|
settings->setValue("interface/showpersistentselectioncount", showTotalSelectionCount);
|
||||||
|
}
|
||||||
|
|
||||||
void SettingsCache::loadPaths()
|
void SettingsCache::loadPaths()
|
||||||
{
|
{
|
||||||
QString dataPath = getDataPath();
|
QString dataPath = getDataPath();
|
||||||
|
|
@ -1426,7 +1416,7 @@ void SettingsCache::resetPaths()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
SettingsCache &SettingsCache::instance()
|
CardCounterSettings &SettingsCache::cardCounters() const
|
||||||
{
|
{
|
||||||
return *settingsCache;
|
return *cardCounterSettings;
|
||||||
}
|
}
|
||||||
59
cockatrice/src/client/settings/card_counter_settings.cpp
Normal file
|
|
@ -0,0 +1,59 @@
|
||||||
|
#include "card_counter_settings.h"
|
||||||
|
|
||||||
|
#include <QColor>
|
||||||
|
#include <QSettings>
|
||||||
|
#include <QtMath>
|
||||||
|
|
||||||
|
CardCounterSettings::CardCounterSettings(const QString &settingsPath, QObject *parent)
|
||||||
|
: SettingsManager(settingsPath + "global.ini", "cards", "counters", parent)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
void CardCounterSettings::setColor(int counterId, const QColor &color)
|
||||||
|
{
|
||||||
|
QSettings settings = getSettings();
|
||||||
|
|
||||||
|
QString key = QString("cards/counters/%1/color").arg(counterId);
|
||||||
|
|
||||||
|
if (settings.value(key).value<QColor>() == color) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
settings.setValue(key, color);
|
||||||
|
emit colorChanged(counterId, color);
|
||||||
|
}
|
||||||
|
|
||||||
|
QColor CardCounterSettings::color(int counterId) const
|
||||||
|
{
|
||||||
|
QColor defaultColor;
|
||||||
|
|
||||||
|
if (counterId < 6) {
|
||||||
|
// Preserve legacy colors
|
||||||
|
defaultColor = QColor::fromHsv(counterId * 60, 150, 255);
|
||||||
|
} else {
|
||||||
|
// Future-proof support for more counters with pseudo-random colors
|
||||||
|
int h = (counterId * 37) % 360;
|
||||||
|
int s = 128 + 64 * qSin((counterId * 97) * 0.1); // 64-192
|
||||||
|
int v = 196 + 32 * qSin((counterId * 101) * 0.07); // 164-228
|
||||||
|
|
||||||
|
defaultColor = QColor::fromHsv(h, s, v);
|
||||||
|
}
|
||||||
|
|
||||||
|
return getSettings().value(QString("cards/counters/%1/color").arg(counterId), defaultColor).value<QColor>();
|
||||||
|
}
|
||||||
|
|
||||||
|
QString CardCounterSettings::displayName(int counterId) const
|
||||||
|
{
|
||||||
|
// Currently, card counters name are fixed to A, B, ..., Z, AA, AB, ...
|
||||||
|
|
||||||
|
auto nChars = 1 + counterId / 26;
|
||||||
|
QString str;
|
||||||
|
str.resize(nChars);
|
||||||
|
|
||||||
|
for (auto it = str.rbegin(); it != str.rend(); ++it) {
|
||||||
|
*it = QChar('A' + (counterId) % 26);
|
||||||
|
counterId /= 26;
|
||||||
|
}
|
||||||
|
|
||||||
|
return str;
|
||||||
|
}
|
||||||
33
cockatrice/src/client/settings/card_counter_settings.h
Normal file
|
|
@ -0,0 +1,33 @@
|
||||||
|
/**
|
||||||
|
* @file card_counter_settings.h
|
||||||
|
* @ingroup GameSettings
|
||||||
|
*/
|
||||||
|
//! \todo Document this file.
|
||||||
|
|
||||||
|
#ifndef CARD_COUNTER_SETTINGS_H
|
||||||
|
#define CARD_COUNTER_SETTINGS_H
|
||||||
|
|
||||||
|
#include <libcockatrice/settings/settings_manager.h>
|
||||||
|
|
||||||
|
class QSettings;
|
||||||
|
class QColor;
|
||||||
|
|
||||||
|
class CardCounterSettings : public SettingsManager
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
|
||||||
|
public:
|
||||||
|
CardCounterSettings(const QString &settingsPath, QObject *parent = nullptr);
|
||||||
|
|
||||||
|
[[nodiscard]] QColor color(int counterId) const;
|
||||||
|
|
||||||
|
[[nodiscard]] QString displayName(int counterId) const;
|
||||||
|
|
||||||
|
public slots:
|
||||||
|
void setColor(int counterId, const QColor &color);
|
||||||
|
|
||||||
|
signals:
|
||||||
|
void colorChanged(int counterId, const QColor &color);
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif // CARD_COUNTER_SETTINGS_H
|
||||||
|
|
@ -150,12 +150,7 @@ void ShortcutTreeView::currentChanged(const QModelIndex ¤t, const QModelIn
|
||||||
*/
|
*/
|
||||||
void ShortcutTreeView::updateSearchString(const QString &searchString)
|
void ShortcutTreeView::updateSearchString(const QString &searchString)
|
||||||
{
|
{
|
||||||
#if QT_VERSION > QT_VERSION_CHECK(5, 14, 0)
|
QStringList searchWords = searchString.split(" ", Qt::SkipEmptyParts);
|
||||||
const auto skipEmptyParts = Qt::SkipEmptyParts;
|
|
||||||
#else
|
|
||||||
const auto skipEmptyParts = QString::SkipEmptyParts;
|
|
||||||
#endif
|
|
||||||
QStringList searchWords = searchString.split(" ", skipEmptyParts);
|
|
||||||
|
|
||||||
auto escapeRegex = [](const QString &s) { return QRegularExpression::escape(s); };
|
auto escapeRegex = [](const QString &s) { return QRegularExpression::escape(s); };
|
||||||
std::transform(searchWords.begin(), searchWords.end(), searchWords.begin(), escapeRegex);
|
std::transform(searchWords.begin(), searchWords.end(), searchWords.begin(), escapeRegex);
|
||||||
|
|
@ -1,7 +1,12 @@
|
||||||
|
/**
|
||||||
|
* @file shortcut_treeview.h
|
||||||
|
* @ingroup CoreSettings
|
||||||
|
*/
|
||||||
|
//! \todo Document this file.
|
||||||
|
|
||||||
#ifndef SHORTCUT_TREEVIEW_H
|
#ifndef SHORTCUT_TREEVIEW_H
|
||||||
#define SHORTCUT_TREEVIEW_H
|
#define SHORTCUT_TREEVIEW_H
|
||||||
|
|
||||||
#include <QModelIndex>
|
|
||||||
#include <QSortFilterProxyModel>
|
#include <QSortFilterProxyModel>
|
||||||
#include <QStandardItemModel>
|
#include <QStandardItemModel>
|
||||||
#include <QTreeView>
|
#include <QTreeView>
|
||||||
|
|
@ -16,7 +21,7 @@ public:
|
||||||
explicit ShortcutFilterProxyModel(QObject *parent = nullptr);
|
explicit ShortcutFilterProxyModel(QObject *parent = nullptr);
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
bool filterAcceptsRow(int sourceRow, const QModelIndex &sourceParent) const override;
|
[[nodiscard]] bool filterAcceptsRow(int sourceRow, const QModelIndex &sourceParent) const override;
|
||||||
};
|
};
|
||||||
|
|
||||||
class ShortcutTreeView : public QTreeView
|
class ShortcutTreeView : public QTreeView
|
||||||
|
|
@ -64,8 +64,13 @@ ShortcutsSettings::ShortcutsSettings(const QString &settingsPath, QObject *paren
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// PR 5079 changes Textbox/unfocusTextBox to Player/unfocusTextBox and tab_game/aFocusChat to Player/aFocusChat.
|
/**
|
||||||
/// A migration is necessary to let players keep their already configured shortcuts.
|
* @brief Migrates legacy shortcut key names to current naming scheme.
|
||||||
|
*
|
||||||
|
* PR 5079 changed Textbox/unfocusTextBox to Player/unfocusTextBox and
|
||||||
|
* tab_game/aFocusChat to Player/aFocusChat. This migration allows players
|
||||||
|
* to keep their already configured shortcuts.
|
||||||
|
*/
|
||||||
void ShortcutsSettings::migrateShortcuts()
|
void ShortcutsSettings::migrateShortcuts()
|
||||||
{
|
{
|
||||||
if (QFile(settingsFilePath).exists()) {
|
if (QFile(settingsFilePath).exists()) {
|
||||||
|
|
@ -115,6 +120,13 @@ ShortcutKey ShortcutsSettings::getShortcut(const QString &name) const
|
||||||
return getDefaultShortcut(name);
|
return getDefaultShortcut(name);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Gets the first shortcut for the given action.
|
||||||
|
*
|
||||||
|
* NOTE: In most cases you should be using ShortcutsSettings::getShortcut instead,
|
||||||
|
* as that will return all shortcuts if there are multiple shortcuts.
|
||||||
|
* The only reason to use this method is if an object does not accept multiple shortcuts, such as with QButtons.
|
||||||
|
*/
|
||||||
QKeySequence ShortcutsSettings::getSingleShortcut(const QString &name) const
|
QKeySequence ShortcutsSettings::getSingleShortcut(const QString &name) const
|
||||||
{
|
{
|
||||||
return getShortcut(name).at(0);
|
return getShortcut(name).at(0);
|
||||||
|
|
@ -229,9 +241,7 @@ bool ShortcutsSettings::isValid(const QString &name, const QString &sequences) c
|
||||||
return findOverlaps(name, sequences).isEmpty();
|
return findOverlaps(name, sequences).isEmpty();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/** @brief Checks if the shortcut is a shortcut that is active in all windows. */
|
||||||
* Checks if the shortcut is a shortcut that is active in all windows
|
|
||||||
*/
|
|
||||||
static bool isAlwaysActiveShortcut(const QString &shortcutName)
|
static bool isAlwaysActiveShortcut(const QString &shortcutName)
|
||||||
{
|
{
|
||||||
return shortcutName.startsWith("MainWindow") || shortcutName.startsWith("Tabs");
|
return shortcutName.startsWith("MainWindow") || shortcutName.startsWith("Tabs");
|
||||||
|
|
@ -1,3 +1,9 @@
|
||||||
|
/**
|
||||||
|
* @file shortcuts_settings.h
|
||||||
|
* @ingroup CoreSettings
|
||||||
|
*/
|
||||||
|
//! \todo Document this file.
|
||||||
|
|
||||||
#ifndef SHORTCUTSSETTINGS_H
|
#ifndef SHORTCUTSSETTINGS_H
|
||||||
#define SHORTCUTSSETTINGS_H
|
#define SHORTCUTSSETTINGS_H
|
||||||
|
|
||||||
|
|
@ -27,6 +33,7 @@ public:
|
||||||
Move_bottom,
|
Move_bottom,
|
||||||
Gameplay,
|
Gameplay,
|
||||||
Drawing,
|
Drawing,
|
||||||
|
Hand,
|
||||||
Chat_room,
|
Chat_room,
|
||||||
Game_window,
|
Game_window,
|
||||||
Load_deck,
|
Load_deck,
|
||||||
|
|
@ -65,6 +72,8 @@ public:
|
||||||
return QApplication::translate("shortcutsTab", "Gameplay");
|
return QApplication::translate("shortcutsTab", "Gameplay");
|
||||||
case Drawing:
|
case Drawing:
|
||||||
return QApplication::translate("shortcutsTab", "Drawing");
|
return QApplication::translate("shortcutsTab", "Drawing");
|
||||||
|
case Hand:
|
||||||
|
return QApplication::translate("shortcutsTab", "Hand");
|
||||||
case Chat_room:
|
case Chat_room:
|
||||||
return QApplication::translate("shortcutsTab", "Chat Room");
|
return QApplication::translate("shortcutsTab", "Chat Room");
|
||||||
case Game_window:
|
case Game_window:
|
||||||
|
|
@ -90,15 +99,15 @@ public:
|
||||||
void setSequence(const QList &_sequence)
|
void setSequence(const QList &_sequence)
|
||||||
{
|
{
|
||||||
QList::operator=(_sequence);
|
QList::operator=(_sequence);
|
||||||
};
|
}
|
||||||
QString getName() const
|
[[nodiscard]] QString getName() const
|
||||||
{
|
{
|
||||||
return QApplication::translate("shortcutsTab", name.toUtf8().data());
|
return QApplication::translate("shortcutsTab", name.toUtf8().data());
|
||||||
};
|
}
|
||||||
QString getGroupName() const
|
[[nodiscard]] QString getGroupName() const
|
||||||
{
|
{
|
||||||
return ShortcutGroup::getGroupName(group);
|
return ShortcutGroup::getGroupName(group);
|
||||||
};
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
QString name;
|
QString name;
|
||||||
|
|
@ -111,24 +120,24 @@ class ShortcutsSettings : public QObject
|
||||||
public:
|
public:
|
||||||
explicit ShortcutsSettings(const QString &settingsFilePath, QObject *parent = nullptr);
|
explicit ShortcutsSettings(const QString &settingsFilePath, QObject *parent = nullptr);
|
||||||
|
|
||||||
ShortcutKey getDefaultShortcut(const QString &name) const;
|
[[nodiscard]] ShortcutKey getDefaultShortcut(const QString &name) const;
|
||||||
ShortcutKey getShortcut(const QString &name) const;
|
[[nodiscard]] ShortcutKey getShortcut(const QString &name) const;
|
||||||
QKeySequence getSingleShortcut(const QString &name) const;
|
[[nodiscard]] QKeySequence getSingleShortcut(const QString &name) const;
|
||||||
QString getDefaultShortcutString(const QString &name) const;
|
[[nodiscard]] QString getDefaultShortcutString(const QString &name) const;
|
||||||
QString getShortcutString(const QString &name) const;
|
[[nodiscard]] QString getShortcutString(const QString &name) const;
|
||||||
QString getShortcutFriendlyName(const QString &shortcutName) const;
|
[[nodiscard]] QString getShortcutFriendlyName(const QString &shortcutName) const;
|
||||||
QList<QString> getAllShortcutKeys() const
|
[[nodiscard]] QList<QString> getAllShortcutKeys() const
|
||||||
{
|
{
|
||||||
return shortCuts.keys();
|
return shortCuts.keys();
|
||||||
};
|
}
|
||||||
|
|
||||||
void setShortcuts(const QString &name, const QList<QKeySequence> &Sequence);
|
void setShortcuts(const QString &name, const QList<QKeySequence> &Sequence);
|
||||||
void setShortcuts(const QString &name, const QKeySequence &Sequence);
|
void setShortcuts(const QString &name, const QKeySequence &Sequence);
|
||||||
void setShortcuts(const QString &name, const QString &sequences);
|
void setShortcuts(const QString &name, const QString &sequences);
|
||||||
|
|
||||||
bool isKeyAllowed(const QString &name, const QString &sequences) const;
|
[[nodiscard]] bool isKeyAllowed(const QString &name, const QString &sequences) const;
|
||||||
bool isValid(const QString &name, const QString &sequences) const;
|
[[nodiscard]] bool isValid(const QString &name, const QString &sequences) const;
|
||||||
QStringList findOverlaps(const QString &name, const QString &sequences) const;
|
[[nodiscard]] QStringList findOverlaps(const QString &name, const QString &sequences) const;
|
||||||
|
|
||||||
void resetAllShortcuts();
|
void resetAllShortcuts();
|
||||||
void clearAllShortcuts();
|
void clearAllShortcuts();
|
||||||
|
|
@ -143,8 +152,8 @@ private:
|
||||||
QString settingsFilePath;
|
QString settingsFilePath;
|
||||||
QHash<QString, ShortcutKey> shortCuts;
|
QHash<QString, ShortcutKey> shortCuts;
|
||||||
|
|
||||||
QString stringifySequence(const QList<QKeySequence> &Sequence) const;
|
[[nodiscard]] QString stringifySequence(const QList<QKeySequence> &Sequence) const;
|
||||||
QList<QKeySequence> parseSequenceString(const QString &stringSequence) const;
|
[[nodiscard]] QList<QKeySequence> parseSequenceString(const QString &stringSequence) const;
|
||||||
|
|
||||||
const QHash<QString, ShortcutKey> defaultShortCuts = {
|
const QHash<QString, ShortcutKey> defaultShortCuts = {
|
||||||
{"MainWindow/aCheckCardUpdates", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Check for Card Updates..."),
|
{"MainWindow/aCheckCardUpdates", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Check for Card Updates..."),
|
||||||
|
|
@ -173,6 +182,9 @@ private:
|
||||||
{"MainWindow/aWatchReplay", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Watch Replay..."),
|
{"MainWindow/aWatchReplay", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Watch Replay..."),
|
||||||
parseSequenceString(""),
|
parseSequenceString(""),
|
||||||
ShortcutGroup::Main_Window)},
|
ShortcutGroup::Main_Window)},
|
||||||
|
{"MainWindow/aStatusBar", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Show Status Bar"),
|
||||||
|
parseSequenceString(""),
|
||||||
|
ShortcutGroup::Main_Window)},
|
||||||
{"TabDeckEditor/aAnalyzeDeck", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Analyze Deck (deckstats.net)"),
|
{"TabDeckEditor/aAnalyzeDeck", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Analyze Deck (deckstats.net)"),
|
||||||
parseSequenceString(""),
|
parseSequenceString(""),
|
||||||
ShortcutGroup::Deck_Editor)},
|
ShortcutGroup::Deck_Editor)},
|
||||||
|
|
@ -284,31 +296,58 @@ private:
|
||||||
{"DeckViewContainer/forceStartGameButton", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Force Start"),
|
{"DeckViewContainer/forceStartGameButton", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Force Start"),
|
||||||
parseSequenceString(""),
|
parseSequenceString(""),
|
||||||
ShortcutGroup::Game_Lobby)},
|
ShortcutGroup::Game_Lobby)},
|
||||||
{"Player/aCCGreen", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Add Green Counter"),
|
{"Player/aCCMagenta", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Add Card Counter (F)"),
|
||||||
|
parseSequenceString(""),
|
||||||
|
ShortcutGroup::Card_Counters)},
|
||||||
|
{"Player/aRCMagenta", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Remove Card Counter (F)"),
|
||||||
|
parseSequenceString(""),
|
||||||
|
ShortcutGroup::Card_Counters)},
|
||||||
|
{"Player/aSCMagenta", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Set Card Counters (F)..."),
|
||||||
|
parseSequenceString(""),
|
||||||
|
ShortcutGroup::Card_Counters)},
|
||||||
|
{"Player/aCCPurple", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Add Card Counter (E)"),
|
||||||
|
parseSequenceString(""),
|
||||||
|
ShortcutGroup::Card_Counters)},
|
||||||
|
{"Player/aRCPurple", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Remove Card Counter (E)"),
|
||||||
|
parseSequenceString(""),
|
||||||
|
ShortcutGroup::Card_Counters)},
|
||||||
|
{"Player/aSCPurple", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Set Card Counters (E)..."),
|
||||||
|
parseSequenceString(""),
|
||||||
|
ShortcutGroup::Card_Counters)},
|
||||||
|
{"Player/aCCCyan", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Add Card Counter(D)"),
|
||||||
|
parseSequenceString(""),
|
||||||
|
ShortcutGroup::Card_Counters)},
|
||||||
|
{"Player/aRCCyan", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Remove Card Counter (D)"),
|
||||||
|
parseSequenceString(""),
|
||||||
|
ShortcutGroup::Card_Counters)},
|
||||||
|
{"Player/aSCCyan", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Set Card Counters (D)..."),
|
||||||
|
parseSequenceString(""),
|
||||||
|
ShortcutGroup::Card_Counters)},
|
||||||
|
{"Player/aCCGreen", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Add Card Counter (C)"),
|
||||||
parseSequenceString("Ctrl+>"),
|
parseSequenceString("Ctrl+>"),
|
||||||
ShortcutGroup::Card_Counters)},
|
ShortcutGroup::Card_Counters)},
|
||||||
{"Player/aRCGreen", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Remove Green Counter"),
|
{"Player/aRCGreen", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Remove Card Counter (C)"),
|
||||||
parseSequenceString("Ctrl+<"),
|
parseSequenceString("Ctrl+<"),
|
||||||
ShortcutGroup::Card_Counters)},
|
ShortcutGroup::Card_Counters)},
|
||||||
{"Player/aSCGreen", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Set Green Counters..."),
|
{"Player/aSCGreen", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Set Card Counters (C)..."),
|
||||||
parseSequenceString("Ctrl+?"),
|
parseSequenceString("Ctrl+?"),
|
||||||
ShortcutGroup::Card_Counters)},
|
ShortcutGroup::Card_Counters)},
|
||||||
{"Player/aCCYellow", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Add Yellow Counter"),
|
{"Player/aCCYellow", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Add Card Counter (B)"),
|
||||||
parseSequenceString("Ctrl+."),
|
parseSequenceString("Ctrl+."),
|
||||||
ShortcutGroup::Card_Counters)},
|
ShortcutGroup::Card_Counters)},
|
||||||
{"Player/aRCYellow", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Remove Yellow Counter"),
|
{"Player/aRCYellow", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Remove Card Counter (B)"),
|
||||||
parseSequenceString("Ctrl+,"),
|
parseSequenceString("Ctrl+,"),
|
||||||
ShortcutGroup::Card_Counters)},
|
ShortcutGroup::Card_Counters)},
|
||||||
{"Player/aSCYellow", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Set Yellow Counters..."),
|
{"Player/aSCYellow", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Set Card Counters (B)..."),
|
||||||
parseSequenceString("Ctrl+/"),
|
parseSequenceString("Ctrl+/"),
|
||||||
ShortcutGroup::Card_Counters)},
|
ShortcutGroup::Card_Counters)},
|
||||||
{"Player/aCCRed", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Add Red Counter"),
|
{"Player/aCCRed", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Add Card Counter (A)"),
|
||||||
parseSequenceString("Alt+."),
|
parseSequenceString("Alt+."),
|
||||||
ShortcutGroup::Card_Counters)},
|
ShortcutGroup::Card_Counters)},
|
||||||
{"Player/aRCRed", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Remove Red Counter"),
|
{"Player/aRCRed", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Remove Card Counter (A)"),
|
||||||
parseSequenceString("Alt+,"),
|
parseSequenceString("Alt+,"),
|
||||||
ShortcutGroup::Card_Counters)},
|
ShortcutGroup::Card_Counters)},
|
||||||
{"Player/aSCRed", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Set Red Counters..."),
|
{"Player/aSCRed", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Set Card Counters (A)..."),
|
||||||
parseSequenceString("Alt+/"),
|
parseSequenceString("Alt+/"),
|
||||||
ShortcutGroup::Card_Counters)},
|
ShortcutGroup::Card_Counters)},
|
||||||
{"Player/aInc", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Add Life Counter"),
|
{"Player/aInc", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Add Life Counter"),
|
||||||
|
|
@ -383,6 +422,10 @@ private:
|
||||||
{"Player/aSetCounter_storm", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Set Other Counters..."),
|
{"Player/aSetCounter_storm", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Set Other Counters..."),
|
||||||
parseSequenceString("Ctrl+\\"),
|
parseSequenceString("Ctrl+\\"),
|
||||||
ShortcutGroup::Player_Counters)},
|
ShortcutGroup::Player_Counters)},
|
||||||
|
{"Player/aIncrementAllCardCounters",
|
||||||
|
ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Increment all card counters"),
|
||||||
|
parseSequenceString("Ctrl+Shift+A"),
|
||||||
|
ShortcutGroup::Playing_Area)},
|
||||||
{"Player/aIncP", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Add Power (+1/+0)"),
|
{"Player/aIncP", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Add Power (+1/+0)"),
|
||||||
parseSequenceString("Ctrl++;Ctrl+="),
|
parseSequenceString("Ctrl++;Ctrl+="),
|
||||||
ShortcutGroup::Power_Toughness)},
|
ShortcutGroup::Power_Toughness)},
|
||||||
|
|
@ -458,7 +501,7 @@ private:
|
||||||
{"Player/aUntapAll", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Untap All"),
|
{"Player/aUntapAll", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Untap All"),
|
||||||
parseSequenceString("Ctrl+U"),
|
parseSequenceString("Ctrl+U"),
|
||||||
ShortcutGroup::Playing_Area)},
|
ShortcutGroup::Playing_Area)},
|
||||||
{"Player/aDoesntUntap", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Toggle Untap"),
|
{"Player/aDoesntUntap", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Toggle Skip Untapping"),
|
||||||
parseSequenceString("Alt+U"),
|
parseSequenceString("Alt+U"),
|
||||||
ShortcutGroup::Playing_Area)},
|
ShortcutGroup::Playing_Area)},
|
||||||
{"Player/aFlip", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Turn Card Over"),
|
{"Player/aFlip", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Turn Card Over"),
|
||||||
|
|
@ -470,6 +513,9 @@ private:
|
||||||
{"Player/aPlay", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Play Card"),
|
{"Player/aPlay", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Play Card"),
|
||||||
parseSequenceString(""),
|
parseSequenceString(""),
|
||||||
ShortcutGroup::Playing_Area)},
|
ShortcutGroup::Playing_Area)},
|
||||||
|
{"Player/aPlayFacedown", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Play Card, Face Down"),
|
||||||
|
parseSequenceString(""),
|
||||||
|
ShortcutGroup::Playing_Area)},
|
||||||
{"Player/aAttach", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Attach Card..."),
|
{"Player/aAttach", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Attach Card..."),
|
||||||
parseSequenceString("Ctrl+Alt+A"),
|
parseSequenceString("Ctrl+Alt+A"),
|
||||||
ShortcutGroup::Playing_Area)},
|
ShortcutGroup::Playing_Area)},
|
||||||
|
|
@ -491,6 +537,9 @@ private:
|
||||||
{"Player/aSetAnnotation", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Set Annotation..."),
|
{"Player/aSetAnnotation", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Set Annotation..."),
|
||||||
parseSequenceString("Alt+N"),
|
parseSequenceString("Alt+N"),
|
||||||
ShortcutGroup::Playing_Area)},
|
ShortcutGroup::Playing_Area)},
|
||||||
|
{"Player/aReduceLifeByPower", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Reduce Life by Power"),
|
||||||
|
parseSequenceString("Ctrl+Shift+L"),
|
||||||
|
ShortcutGroup::Playing_Area)},
|
||||||
{"Player/aSelectAll", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Select All Cards in Zone"),
|
{"Player/aSelectAll", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Select All Cards in Zone"),
|
||||||
parseSequenceString("Ctrl+A"),
|
parseSequenceString("Ctrl+A"),
|
||||||
ShortcutGroup::Playing_Area)},
|
ShortcutGroup::Playing_Area)},
|
||||||
|
|
@ -500,6 +549,9 @@ private:
|
||||||
{"Player/aSelectColumn", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Select All Cards in Column"),
|
{"Player/aSelectColumn", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Select All Cards in Column"),
|
||||||
parseSequenceString("Ctrl+Shift+C"),
|
parseSequenceString("Ctrl+Shift+C"),
|
||||||
ShortcutGroup::Playing_Area)},
|
ShortcutGroup::Playing_Area)},
|
||||||
|
{"Player/aRevealToAll", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Reveal Selected Cards to All Players"),
|
||||||
|
parseSequenceString(""),
|
||||||
|
ShortcutGroup::Playing_Area)},
|
||||||
{"Player/aMoveToBottomLibrary", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Bottom of Library"),
|
{"Player/aMoveToBottomLibrary", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Bottom of Library"),
|
||||||
parseSequenceString("Ctrl+B"),
|
parseSequenceString("Ctrl+B"),
|
||||||
ShortcutGroup::Move_selected)},
|
ShortcutGroup::Move_selected)},
|
||||||
|
|
@ -514,12 +566,9 @@ private:
|
||||||
{"Player/aMoveToTopLibrary", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Top of Library"),
|
{"Player/aMoveToTopLibrary", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Top of Library"),
|
||||||
parseSequenceString(""),
|
parseSequenceString(""),
|
||||||
ShortcutGroup::Move_selected)},
|
ShortcutGroup::Move_selected)},
|
||||||
{"Player/aPlayFacedown", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Battlefield, Face Down"),
|
{"Player/aMoveToTable", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Battlefield"),
|
||||||
parseSequenceString(""),
|
parseSequenceString(""),
|
||||||
ShortcutGroup::Move_selected)},
|
ShortcutGroup::Move_selected)},
|
||||||
{"Player/aPlay", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Battlefield"),
|
|
||||||
parseSequenceString(""),
|
|
||||||
ShortcutGroup::Move_selected)},
|
|
||||||
{"Player/aViewHand",
|
{"Player/aViewHand",
|
||||||
ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Hand"), parseSequenceString(""), ShortcutGroup::View)},
|
ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Hand"), parseSequenceString(""), ShortcutGroup::View)},
|
||||||
{"Player/aViewGraveyard",
|
{"Player/aViewGraveyard",
|
||||||
|
|
@ -552,11 +601,19 @@ private:
|
||||||
{"Player/aMoveTopCardsToGraveyard", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Graveyard (Multiple)"),
|
{"Player/aMoveTopCardsToGraveyard", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Graveyard (Multiple)"),
|
||||||
parseSequenceString("Alt+M"),
|
parseSequenceString("Alt+M"),
|
||||||
ShortcutGroup::Move_top)},
|
ShortcutGroup::Move_top)},
|
||||||
|
{"Player/aMoveTopCardsToGraveyardFaceDown",
|
||||||
|
ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Graveyard (Multiple), Face Down"),
|
||||||
|
parseSequenceString(""),
|
||||||
|
ShortcutGroup::Move_top)},
|
||||||
{"Player/aMoveTopCardToExile",
|
{"Player/aMoveTopCardToExile",
|
||||||
ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Exile"), parseSequenceString(""), ShortcutGroup::Move_top)},
|
ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Exile"), parseSequenceString(""), ShortcutGroup::Move_top)},
|
||||||
{"Player/aMoveTopCardsToExile", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Exile (Multiple)"),
|
{"Player/aMoveTopCardsToExile", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Exile (Multiple)"),
|
||||||
parseSequenceString(""),
|
parseSequenceString(""),
|
||||||
ShortcutGroup::Move_top)},
|
ShortcutGroup::Move_top)},
|
||||||
|
{"Player/aMoveTopCardsToExileFaceDown",
|
||||||
|
ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Exile (Multiple), Face Down"),
|
||||||
|
parseSequenceString(""),
|
||||||
|
ShortcutGroup::Move_top)},
|
||||||
{"Player/aMoveTopCardsUntil", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Stack Until Found"),
|
{"Player/aMoveTopCardsUntil", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Stack Until Found"),
|
||||||
parseSequenceString("Ctrl+Shift+Y"),
|
parseSequenceString("Ctrl+Shift+Y"),
|
||||||
ShortcutGroup::Move_top)},
|
ShortcutGroup::Move_top)},
|
||||||
|
|
@ -574,11 +631,19 @@ private:
|
||||||
{"Player/aMoveBottomCardsToGrave", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Graveyard (Multiple)"),
|
{"Player/aMoveBottomCardsToGrave", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Graveyard (Multiple)"),
|
||||||
parseSequenceString(""),
|
parseSequenceString(""),
|
||||||
ShortcutGroup::Move_bottom)},
|
ShortcutGroup::Move_bottom)},
|
||||||
|
{"Player/aMoveBottomCardsToGraveFaceDown",
|
||||||
|
ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Graveyard (Multiple), Face Down"),
|
||||||
|
parseSequenceString(""),
|
||||||
|
ShortcutGroup::Move_bottom)},
|
||||||
{"Player/aMoveBottomCardToExile",
|
{"Player/aMoveBottomCardToExile",
|
||||||
ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Exile"), parseSequenceString(""), ShortcutGroup::Move_bottom)},
|
ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Exile"), parseSequenceString(""), ShortcutGroup::Move_bottom)},
|
||||||
{"Player/aMoveBottomCardsToExile", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Exile (Multiple)"),
|
{"Player/aMoveBottomCardsToExile", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Exile (Multiple)"),
|
||||||
parseSequenceString(""),
|
parseSequenceString(""),
|
||||||
ShortcutGroup::Move_bottom)},
|
ShortcutGroup::Move_bottom)},
|
||||||
|
{"Player/aMoveBottomCardsToExileFaceDown",
|
||||||
|
ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Exile (Multiple), Face Down"),
|
||||||
|
parseSequenceString(""),
|
||||||
|
ShortcutGroup::Move_bottom)},
|
||||||
{"Player/aMoveBottomCardToTop", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Top of Library"),
|
{"Player/aMoveBottomCardToTop", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Top of Library"),
|
||||||
parseSequenceString(""),
|
parseSequenceString(""),
|
||||||
ShortcutGroup::Move_bottom)},
|
ShortcutGroup::Move_bottom)},
|
||||||
|
|
@ -602,6 +667,9 @@ private:
|
||||||
{"Player/aRollDie", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Roll Dice..."),
|
{"Player/aRollDie", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Roll Dice..."),
|
||||||
parseSequenceString("Ctrl+I"),
|
parseSequenceString("Ctrl+I"),
|
||||||
ShortcutGroup::Gameplay)},
|
ShortcutGroup::Gameplay)},
|
||||||
|
{"Player/aFlipCoin", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Flip Coin"),
|
||||||
|
parseSequenceString("Ctrl+Shift+I"),
|
||||||
|
ShortcutGroup::Gameplay)},
|
||||||
{"Player/aShuffle", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Shuffle Library"),
|
{"Player/aShuffle", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Shuffle Library"),
|
||||||
parseSequenceString("Ctrl+S"),
|
parseSequenceString("Ctrl+S"),
|
||||||
ShortcutGroup::Gameplay)},
|
ShortcutGroup::Gameplay)},
|
||||||
|
|
@ -614,6 +682,12 @@ private:
|
||||||
{"Player/aMulligan", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Mulligan"),
|
{"Player/aMulligan", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Mulligan"),
|
||||||
parseSequenceString("Ctrl+M"),
|
parseSequenceString("Ctrl+M"),
|
||||||
ShortcutGroup::Drawing)},
|
ShortcutGroup::Drawing)},
|
||||||
|
{"Player/aMulliganSame", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Mulligan (Same hand size)"),
|
||||||
|
parseSequenceString("Ctrl+Shift+M"),
|
||||||
|
ShortcutGroup::Drawing)},
|
||||||
|
{"Player/aMulliganMinusOne", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Mulligan (Hand size - 1)"),
|
||||||
|
parseSequenceString("Ctrl+Shift+Alt+M"),
|
||||||
|
ShortcutGroup::Drawing)},
|
||||||
{"Player/aDrawCard", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Draw a Card"),
|
{"Player/aDrawCard", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Draw a Card"),
|
||||||
parseSequenceString("Ctrl+D"),
|
parseSequenceString("Ctrl+D"),
|
||||||
ShortcutGroup::Drawing)},
|
ShortcutGroup::Drawing)},
|
||||||
|
|
@ -629,6 +703,22 @@ private:
|
||||||
{"Player/aAlwaysLookAtTopCard", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Always Look At Top Card"),
|
{"Player/aAlwaysLookAtTopCard", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Always Look At Top Card"),
|
||||||
parseSequenceString("Ctrl+Shift+N"),
|
parseSequenceString("Ctrl+Shift+N"),
|
||||||
ShortcutGroup::Drawing)},
|
ShortcutGroup::Drawing)},
|
||||||
|
{"Player/aSortHandByName", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Sort Hand by Name"),
|
||||||
|
parseSequenceString(""),
|
||||||
|
ShortcutGroup::Hand)},
|
||||||
|
{"Player/aSortHandByType", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Sort Hand by Type"),
|
||||||
|
parseSequenceString("Ctrl+Shift+H"),
|
||||||
|
ShortcutGroup::Hand)},
|
||||||
|
{"Player/aSortHandByManaValue", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Sort Hand by Mana Value"),
|
||||||
|
parseSequenceString(""),
|
||||||
|
ShortcutGroup::Hand)},
|
||||||
|
{"Player/aRevealHandToAll", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Reveal Hand to All Players"),
|
||||||
|
parseSequenceString(""),
|
||||||
|
ShortcutGroup::Hand)},
|
||||||
|
{"Player/aRevealRandomHandCardToAll",
|
||||||
|
ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Reveal Random Card to All Players"),
|
||||||
|
parseSequenceString(""),
|
||||||
|
ShortcutGroup::Hand)},
|
||||||
{"Player/aRotateViewCW", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Rotate View Clockwise"),
|
{"Player/aRotateViewCW", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Rotate View Clockwise"),
|
||||||
parseSequenceString(""),
|
parseSequenceString(""),
|
||||||
ShortcutGroup::Gameplay)},
|
ShortcutGroup::Gameplay)},
|
||||||
|
|
@ -670,6 +760,8 @@ private:
|
||||||
ShortcutGroup::Replays)},
|
ShortcutGroup::Replays)},
|
||||||
{"Tabs/aTabDeckEditor",
|
{"Tabs/aTabDeckEditor",
|
||||||
ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Deck Editor"), parseSequenceString(""), ShortcutGroup::Tabs)},
|
ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Deck Editor"), parseSequenceString(""), ShortcutGroup::Tabs)},
|
||||||
|
{"Tabs/aTabHome",
|
||||||
|
ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Home"), parseSequenceString(""), ShortcutGroup::Tabs)},
|
||||||
{"Tabs/aTabVisualDeckStorage", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Visual Deck Storage"),
|
{"Tabs/aTabVisualDeckStorage", ShortcutKey(QT_TRANSLATE_NOOP("shortcutsTab", "Visual Deck Storage"),
|
||||||
parseSequenceString(""),
|
parseSequenceString(""),
|
||||||
ShortcutGroup::Tabs)},
|
ShortcutGroup::Tabs)},
|
||||||
|
|
@ -1,6 +1,6 @@
|
||||||
#include "sound_engine.h"
|
#include "sound_engine.h"
|
||||||
|
|
||||||
#include "../settings/cache_settings.h"
|
#include "settings/cache_settings.h"
|
||||||
|
|
||||||
#include <QDir>
|
#include <QDir>
|
||||||
#include <QMediaPlayer>
|
#include <QMediaPlayer>
|
||||||
|
|
@ -105,8 +105,9 @@ QStringMap &SoundEngine::getAvailableThemes()
|
||||||
dir.setPath(SettingsCache::instance().getDataPath() + "/sounds");
|
dir.setPath(SettingsCache::instance().getDataPath() + "/sounds");
|
||||||
|
|
||||||
for (const QString &themeName : dir.entryList(QDir::AllDirs | QDir::NoDotAndDotDot, QDir::Name)) {
|
for (const QString &themeName : dir.entryList(QDir::AllDirs | QDir::NoDotAndDotDot, QDir::Name)) {
|
||||||
if (!availableThemes.contains(themeName))
|
if (!availableThemes.contains(themeName)) {
|
||||||
availableThemes.insert(themeName, dir.absoluteFilePath(themeName));
|
availableThemes.insert(themeName, dir.absoluteFilePath(themeName));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// load themes from cockatrice system dir
|
// load themes from cockatrice system dir
|
||||||
|
|
@ -121,8 +122,9 @@ QStringMap &SoundEngine::getAvailableThemes()
|
||||||
);
|
);
|
||||||
|
|
||||||
for (const QString &themeName : dir.entryList(QDir::AllDirs | QDir::NoDotAndDotDot, QDir::Name)) {
|
for (const QString &themeName : dir.entryList(QDir::AllDirs | QDir::NoDotAndDotDot, QDir::Name)) {
|
||||||
if (!availableThemes.contains(themeName))
|
if (!availableThemes.contains(themeName)) {
|
||||||
availableThemes.insert(themeName, dir.absoluteFilePath(themeName));
|
availableThemes.insert(themeName, dir.absoluteFilePath(themeName));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return availableThemes;
|
return availableThemes;
|
||||||
|
|
|
||||||
|
|
@ -1,3 +1,9 @@
|
||||||
|
/**
|
||||||
|
* @file sound_engine.h
|
||||||
|
* @ingroup Core
|
||||||
|
*/
|
||||||
|
//! \todo Document this file.
|
||||||
|
|
||||||
#ifndef SOUNDENGINE_H
|
#ifndef SOUNDENGINE_H
|
||||||
#define SOUNDENGINE_H
|
#define SOUNDENGINE_H
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -1,581 +0,0 @@
|
||||||
#include "abstract_tab_deck_editor.h"
|
|
||||||
|
|
||||||
#include "../../client/game_logic/abstract_client.h"
|
|
||||||
#include "../../client/tapped_out_interface.h"
|
|
||||||
#include "../../client/ui/widgets/cards/card_info_frame_widget.h"
|
|
||||||
#include "../../deck/deck_stats_interface.h"
|
|
||||||
#include "../../dialogs/dlg_load_deck.h"
|
|
||||||
#include "../../dialogs/dlg_load_deck_from_clipboard.h"
|
|
||||||
#include "../../game/cards/card_database_manager.h"
|
|
||||||
#include "../../game/cards/card_database_model.h"
|
|
||||||
#include "../../server/pending_command.h"
|
|
||||||
#include "../../settings/cache_settings.h"
|
|
||||||
#include "../ui/picture_loader/picture_loader.h"
|
|
||||||
#include "../ui/pixel_map_generator.h"
|
|
||||||
#include "pb/command_deck_upload.pb.h"
|
|
||||||
#include "pb/response.pb.h"
|
|
||||||
#include "tab_supervisor.h"
|
|
||||||
#include "trice_limits.h"
|
|
||||||
|
|
||||||
#include <QAction>
|
|
||||||
#include <QApplication>
|
|
||||||
#include <QClipboard>
|
|
||||||
#include <QCloseEvent>
|
|
||||||
#include <QDebug>
|
|
||||||
#include <QDesktopServices>
|
|
||||||
#include <QDir>
|
|
||||||
#include <QFileDialog>
|
|
||||||
#include <QHeaderView>
|
|
||||||
#include <QLineEdit>
|
|
||||||
#include <QMenuBar>
|
|
||||||
#include <QMessageBox>
|
|
||||||
#include <QPrintPreviewDialog>
|
|
||||||
#include <QPrinter>
|
|
||||||
#include <QProcessEnvironment>
|
|
||||||
#include <QPushButton>
|
|
||||||
#include <QRegularExpression>
|
|
||||||
#include <QSplitter>
|
|
||||||
#include <QTextStream>
|
|
||||||
#include <QTreeView>
|
|
||||||
#include <QUrl>
|
|
||||||
|
|
||||||
AbstractTabDeckEditor::AbstractTabDeckEditor(TabSupervisor *_tabSupervisor) : Tab(_tabSupervisor)
|
|
||||||
{
|
|
||||||
setDockOptions(QMainWindow::AnimatedDocks | QMainWindow::AllowNestedDocks | QMainWindow::AllowTabbedDocks);
|
|
||||||
|
|
||||||
databaseDisplayDockWidget = new DeckEditorDatabaseDisplayWidget(this);
|
|
||||||
deckDockWidget = new DeckEditorDeckDockWidget(this);
|
|
||||||
cardInfoDockWidget = new DeckEditorCardInfoDockWidget(this);
|
|
||||||
filterDockWidget = new DeckEditorFilterDockWidget(this);
|
|
||||||
printingSelectorDockWidget = new DeckEditorPrintingSelectorDockWidget(this);
|
|
||||||
|
|
||||||
connect(deckDockWidget, &DeckEditorDeckDockWidget::deckChanged, this, &AbstractTabDeckEditor::onDeckChanged);
|
|
||||||
connect(deckDockWidget, &DeckEditorDeckDockWidget::deckModified, this, &AbstractTabDeckEditor::onDeckModified);
|
|
||||||
connect(deckDockWidget, &DeckEditorDeckDockWidget::cardChanged, this, &AbstractTabDeckEditor::updateCard);
|
|
||||||
connect(this, &AbstractTabDeckEditor::decrementCard, deckDockWidget, &DeckEditorDeckDockWidget::actDecrementCard);
|
|
||||||
connect(databaseDisplayDockWidget, &DeckEditorDatabaseDisplayWidget::cardChanged, this,
|
|
||||||
&AbstractTabDeckEditor::updateCard);
|
|
||||||
connect(databaseDisplayDockWidget, &DeckEditorDatabaseDisplayWidget::addCardToMainDeck, this,
|
|
||||||
&AbstractTabDeckEditor::actAddCard);
|
|
||||||
connect(databaseDisplayDockWidget, &DeckEditorDatabaseDisplayWidget::addCardToSideboard, this,
|
|
||||||
&AbstractTabDeckEditor::actAddCardToSideboard);
|
|
||||||
connect(databaseDisplayDockWidget, &DeckEditorDatabaseDisplayWidget::decrementCardFromMainDeck, this,
|
|
||||||
&AbstractTabDeckEditor::actDecrementCard);
|
|
||||||
connect(databaseDisplayDockWidget, &DeckEditorDatabaseDisplayWidget::decrementCardFromSideboard, this,
|
|
||||||
&AbstractTabDeckEditor::actDecrementCardFromSideboard);
|
|
||||||
|
|
||||||
connect(filterDockWidget, &DeckEditorFilterDockWidget::clearAllDatabaseFilters, databaseDisplayDockWidget,
|
|
||||||
&DeckEditorDatabaseDisplayWidget::clearAllDatabaseFilters);
|
|
||||||
|
|
||||||
connect(&SettingsCache::instance().shortcuts(), &ShortcutsSettings::shortCutChanged, this,
|
|
||||||
&AbstractTabDeckEditor::refreshShortcuts);
|
|
||||||
}
|
|
||||||
|
|
||||||
void AbstractTabDeckEditor::updateCard(CardInfoPtr _card)
|
|
||||||
{
|
|
||||||
cardInfoDockWidget->updateCard(_card);
|
|
||||||
printingSelectorDockWidget->printingSelector->setCard(_card, DECK_ZONE_MAIN);
|
|
||||||
}
|
|
||||||
|
|
||||||
void AbstractTabDeckEditor::onDeckChanged()
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
void AbstractTabDeckEditor::onDeckModified()
|
|
||||||
{
|
|
||||||
setModified(!isBlankNewDeck());
|
|
||||||
deckMenu->setSaveStatus(!isBlankNewDeck());
|
|
||||||
}
|
|
||||||
|
|
||||||
void AbstractTabDeckEditor::addCardHelper(const CardInfoPtr info, QString zoneName)
|
|
||||||
{
|
|
||||||
if (!info)
|
|
||||||
return;
|
|
||||||
if (info->getIsToken())
|
|
||||||
zoneName = DECK_ZONE_TOKENS;
|
|
||||||
|
|
||||||
QModelIndex newCardIndex = deckDockWidget->deckModel->addPreferredPrintingCard(info->getName(), zoneName, false);
|
|
||||||
// recursiveExpand(newCardIndex);
|
|
||||||
deckDockWidget->deckView->clearSelection();
|
|
||||||
deckDockWidget->deckView->setCurrentIndex(newCardIndex);
|
|
||||||
setModified(true);
|
|
||||||
databaseDisplayDockWidget->searchEdit->setSelection(0, databaseDisplayDockWidget->searchEdit->text().length());
|
|
||||||
}
|
|
||||||
|
|
||||||
void AbstractTabDeckEditor::actAddCard(CardInfoPtr info)
|
|
||||||
{
|
|
||||||
if (QApplication::keyboardModifiers() & Qt::ControlModifier)
|
|
||||||
actAddCardToSideboard(info);
|
|
||||||
else
|
|
||||||
addCardHelper(info, DECK_ZONE_MAIN);
|
|
||||||
deckMenu->setSaveStatus(true);
|
|
||||||
}
|
|
||||||
|
|
||||||
void AbstractTabDeckEditor::actAddCardToSideboard(CardInfoPtr info)
|
|
||||||
{
|
|
||||||
addCardHelper(info, DECK_ZONE_SIDE);
|
|
||||||
deckMenu->setSaveStatus(true);
|
|
||||||
}
|
|
||||||
|
|
||||||
void AbstractTabDeckEditor::actDecrementCard(CardInfoPtr info)
|
|
||||||
{
|
|
||||||
emit decrementCard(info, DECK_ZONE_MAIN);
|
|
||||||
}
|
|
||||||
|
|
||||||
void AbstractTabDeckEditor::actDecrementCardFromSideboard(CardInfoPtr info)
|
|
||||||
{
|
|
||||||
emit decrementCard(info, DECK_ZONE_SIDE);
|
|
||||||
}
|
|
||||||
|
|
||||||
void AbstractTabDeckEditor::actSwapCard(CardInfoPtr info, QString zoneName)
|
|
||||||
{
|
|
||||||
QString providerId = CardDatabaseManager::getInstance()->getSetInfoForCard(info).getProperty("uuid");
|
|
||||||
QString collectorNumber = CardDatabaseManager::getInstance()->getSetInfoForCard(info).getProperty("num");
|
|
||||||
|
|
||||||
QModelIndex foundCard = deckDockWidget->deckModel->findCard(info->getName(), zoneName, providerId, collectorNumber);
|
|
||||||
if (!foundCard.isValid()) {
|
|
||||||
foundCard = deckDockWidget->deckModel->findCard(info->getName(), zoneName);
|
|
||||||
}
|
|
||||||
|
|
||||||
deckDockWidget->swapCard(foundCard);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Opens the deck in this tab.
|
|
||||||
* @param deck The deck. Takes ownership of the object
|
|
||||||
*/
|
|
||||||
void AbstractTabDeckEditor::openDeck(DeckLoader *deck)
|
|
||||||
{
|
|
||||||
setDeck(deck);
|
|
||||||
|
|
||||||
if (!deck->getLastFileName().isEmpty()) {
|
|
||||||
SettingsCache::instance().recents().updateRecentlyOpenedDeckPaths(deck->getLastFileName());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Sets the currently active deck for this tab
|
|
||||||
* @param _deck The deck. Takes ownership of the object
|
|
||||||
*/
|
|
||||||
void AbstractTabDeckEditor::setDeck(DeckLoader *_deck)
|
|
||||||
{
|
|
||||||
deckDockWidget->setDeck(_deck);
|
|
||||||
PictureLoader::cacheCardPixmaps(
|
|
||||||
CardDatabaseManager::getInstance()->getCardsByNameAndProviderId(getDeckList()->getCardListWithProviderId()));
|
|
||||||
setModified(false);
|
|
||||||
|
|
||||||
// If they load a deck, make the deck list appear
|
|
||||||
aDeckDockVisible->setChecked(true);
|
|
||||||
deckDockWidget->setVisible(aDeckDockVisible->isChecked());
|
|
||||||
}
|
|
||||||
|
|
||||||
DeckLoader *AbstractTabDeckEditor::getDeckList() const
|
|
||||||
{
|
|
||||||
return deckDockWidget->getDeckList();
|
|
||||||
}
|
|
||||||
|
|
||||||
void AbstractTabDeckEditor::setModified(bool _modified)
|
|
||||||
{
|
|
||||||
modified = _modified;
|
|
||||||
emit tabTextChanged(this, getTabText());
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @brief Returns true if this tab is a blank newly opened tab, as if it was just created with the `New Deck` action.
|
|
||||||
*/
|
|
||||||
bool AbstractTabDeckEditor::isBlankNewDeck() const
|
|
||||||
{
|
|
||||||
DeckLoader *deck = getDeckList();
|
|
||||||
return !modified && deck->hasNotBeenLoaded();
|
|
||||||
}
|
|
||||||
|
|
||||||
void AbstractTabDeckEditor::actNewDeck()
|
|
||||||
{
|
|
||||||
auto deckOpenLocation = confirmOpen(false);
|
|
||||||
|
|
||||||
if (deckOpenLocation == CANCELLED) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (deckOpenLocation == NEW_TAB) {
|
|
||||||
emit openDeckEditor(nullptr);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
cleanDeckAndResetModified();
|
|
||||||
}
|
|
||||||
|
|
||||||
void AbstractTabDeckEditor::cleanDeckAndResetModified()
|
|
||||||
{
|
|
||||||
deckMenu->setSaveStatus(false);
|
|
||||||
deckDockWidget->cleanDeck();
|
|
||||||
setModified(false);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @brief Displays the save confirmation dialogue that is shown before loading a deck, if required. Takes into
|
|
||||||
* account the `openDeckInNewTab` settting.
|
|
||||||
*
|
|
||||||
* @param openInSameTabIfBlank Open the deck in the same tab instead of a new tab if the current tab is completely
|
|
||||||
* blank. Only relevant when the `openDeckInNewTab` setting is enabled.
|
|
||||||
*
|
|
||||||
* @returns An enum that indicates if and where to load the deck
|
|
||||||
*/
|
|
||||||
AbstractTabDeckEditor::DeckOpenLocation AbstractTabDeckEditor::confirmOpen(const bool openInSameTabIfBlank)
|
|
||||||
{
|
|
||||||
// handle `openDeckInNewTab` setting
|
|
||||||
if (SettingsCache::instance().getOpenDeckInNewTab()) {
|
|
||||||
if (openInSameTabIfBlank && isBlankNewDeck()) {
|
|
||||||
return SAME_TAB;
|
|
||||||
} else {
|
|
||||||
return NEW_TAB;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// early return if deck is unmodified
|
|
||||||
if (!modified) {
|
|
||||||
return SAME_TAB;
|
|
||||||
}
|
|
||||||
|
|
||||||
// do the save confirmation dialogue
|
|
||||||
tabSupervisor->setCurrentWidget(this);
|
|
||||||
|
|
||||||
QMessageBox *msgBox = createSaveConfirmationWindow();
|
|
||||||
QPushButton *newTabButton = msgBox->addButton(tr("Open in new tab"), QMessageBox::ApplyRole);
|
|
||||||
|
|
||||||
int ret = msgBox->exec();
|
|
||||||
|
|
||||||
// `exec()` returns an opaque value if a non-standard button was clicked.
|
|
||||||
// Directly check if newTabButton was clicked before switching over the standard buttons.
|
|
||||||
if (msgBox->clickedButton() == newTabButton) {
|
|
||||||
return NEW_TAB;
|
|
||||||
}
|
|
||||||
|
|
||||||
switch (ret) {
|
|
||||||
case QMessageBox::Save:
|
|
||||||
return actSaveDeck() ? SAME_TAB : CANCELLED;
|
|
||||||
case QMessageBox::Discard:
|
|
||||||
return SAME_TAB;
|
|
||||||
default:
|
|
||||||
return CANCELLED;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @brief Creates the base save confirmation dialogue box.
|
|
||||||
*
|
|
||||||
* @returns A QMessageBox that can be further modified
|
|
||||||
*/
|
|
||||||
QMessageBox *AbstractTabDeckEditor::createSaveConfirmationWindow()
|
|
||||||
{
|
|
||||||
QMessageBox *msgBox = new QMessageBox(this);
|
|
||||||
msgBox->setIcon(QMessageBox::Warning);
|
|
||||||
msgBox->setWindowTitle(tr("Are you sure?"));
|
|
||||||
msgBox->setText(tr("The decklist has been modified.\nDo you want to save the changes?"));
|
|
||||||
msgBox->setStandardButtons(QMessageBox::Save | QMessageBox::Discard | QMessageBox::Cancel);
|
|
||||||
return msgBox;
|
|
||||||
}
|
|
||||||
|
|
||||||
void AbstractTabDeckEditor::actLoadDeck()
|
|
||||||
{
|
|
||||||
auto deckOpenLocation = confirmOpen();
|
|
||||||
|
|
||||||
if (deckOpenLocation == CANCELLED) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
DlgLoadDeck dialog(this);
|
|
||||||
if (!dialog.exec())
|
|
||||||
return;
|
|
||||||
|
|
||||||
QString fileName = dialog.selectedFiles().at(0);
|
|
||||||
openDeckFromFile(fileName, deckOpenLocation);
|
|
||||||
deckDockWidget->updateBannerCardComboBox();
|
|
||||||
}
|
|
||||||
|
|
||||||
void AbstractTabDeckEditor::actOpenRecent(const QString &fileName)
|
|
||||||
{
|
|
||||||
auto deckOpenLocation = confirmOpen();
|
|
||||||
|
|
||||||
if (deckOpenLocation == CANCELLED) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
openDeckFromFile(fileName, deckOpenLocation);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Actually opens the deck from file
|
|
||||||
* @param fileName The path of the deck to open
|
|
||||||
* @param deckOpenLocation Which tab to open the deck
|
|
||||||
*/
|
|
||||||
void AbstractTabDeckEditor::openDeckFromFile(const QString &fileName, DeckOpenLocation deckOpenLocation)
|
|
||||||
{
|
|
||||||
DeckLoader::FileFormat fmt = DeckLoader::getFormatFromName(fileName);
|
|
||||||
|
|
||||||
auto *l = new DeckLoader;
|
|
||||||
if (l->loadFromFile(fileName, fmt, true)) {
|
|
||||||
if (deckOpenLocation == NEW_TAB) {
|
|
||||||
emit openDeckEditor(l);
|
|
||||||
l->deleteLater();
|
|
||||||
} else {
|
|
||||||
deckMenu->setSaveStatus(false);
|
|
||||||
openDeck(l);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
l->deleteLater();
|
|
||||||
QMessageBox::critical(this, tr("Error"), tr("Could not open deck at %1").arg(fileName));
|
|
||||||
}
|
|
||||||
deckMenu->setSaveStatus(true);
|
|
||||||
}
|
|
||||||
|
|
||||||
bool AbstractTabDeckEditor::actSaveDeck()
|
|
||||||
{
|
|
||||||
DeckLoader *const deck = getDeckList();
|
|
||||||
if (deck->getLastRemoteDeckId() != -1) {
|
|
||||||
QString deckString = deck->writeToString_Native();
|
|
||||||
if (deckString.length() > MAX_FILE_LENGTH) {
|
|
||||||
QMessageBox::critical(this, tr("Error"), tr("Could not save remote deck"));
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
Command_DeckUpload cmd;
|
|
||||||
cmd.set_deck_id(static_cast<google::protobuf::uint32>(deck->getLastRemoteDeckId()));
|
|
||||||
cmd.set_deck_list(deckString.toStdString());
|
|
||||||
|
|
||||||
PendingCommand *pend = AbstractClient::prepareSessionCommand(cmd);
|
|
||||||
connect(pend, &PendingCommand::finished, this, &AbstractTabDeckEditor::saveDeckRemoteFinished);
|
|
||||||
tabSupervisor->getClient()->sendCommand(pend);
|
|
||||||
|
|
||||||
return true;
|
|
||||||
} else if (deck->getLastFileName().isEmpty())
|
|
||||||
return actSaveDeckAs();
|
|
||||||
else if (deck->saveToFile(deck->getLastFileName(), deck->getLastFileFormat())) {
|
|
||||||
setModified(false);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
QMessageBox::critical(
|
|
||||||
this, tr("Error"),
|
|
||||||
tr("The deck could not be saved.\nPlease check that the directory is writable and try again."));
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool AbstractTabDeckEditor::actSaveDeckAs()
|
|
||||||
{
|
|
||||||
QFileDialog dialog(this, tr("Save deck"));
|
|
||||||
dialog.setDirectory(SettingsCache::instance().getDeckPath());
|
|
||||||
dialog.setAcceptMode(QFileDialog::AcceptSave);
|
|
||||||
dialog.setDefaultSuffix("cod");
|
|
||||||
dialog.setNameFilters(DeckLoader::FILE_NAME_FILTERS);
|
|
||||||
dialog.selectFile(getDeckList()->getName().trimmed());
|
|
||||||
|
|
||||||
if (!dialog.exec())
|
|
||||||
return false;
|
|
||||||
|
|
||||||
QString fileName = dialog.selectedFiles().at(0);
|
|
||||||
DeckLoader::FileFormat fmt = DeckLoader::getFormatFromName(fileName);
|
|
||||||
|
|
||||||
if (!getDeckList()->saveToFile(fileName, fmt)) {
|
|
||||||
QMessageBox::critical(
|
|
||||||
this, tr("Error"),
|
|
||||||
tr("The deck could not be saved.\nPlease check that the directory is writable and try again."));
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
setModified(false);
|
|
||||||
|
|
||||||
SettingsCache::instance().recents().updateRecentlyOpenedDeckPaths(fileName);
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
void AbstractTabDeckEditor::saveDeckRemoteFinished(const Response &response)
|
|
||||||
{
|
|
||||||
if (response.response_code() != Response::RespOk)
|
|
||||||
QMessageBox::critical(this, tr("Error"), tr("The deck could not be saved."));
|
|
||||||
else
|
|
||||||
setModified(false);
|
|
||||||
}
|
|
||||||
|
|
||||||
void AbstractTabDeckEditor::actLoadDeckFromClipboard()
|
|
||||||
{
|
|
||||||
auto deckOpenLocation = confirmOpen();
|
|
||||||
|
|
||||||
if (deckOpenLocation == CANCELLED) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
DlgLoadDeckFromClipboard dlg(this);
|
|
||||||
if (!dlg.exec())
|
|
||||||
return;
|
|
||||||
|
|
||||||
if (deckOpenLocation == NEW_TAB) {
|
|
||||||
emit openDeckEditor(dlg.getDeckList());
|
|
||||||
} else {
|
|
||||||
setDeck(dlg.getDeckList());
|
|
||||||
setModified(true);
|
|
||||||
}
|
|
||||||
|
|
||||||
deckMenu->setSaveStatus(true);
|
|
||||||
}
|
|
||||||
|
|
||||||
void AbstractTabDeckEditor::editDeckInClipboard(bool annotated)
|
|
||||||
{
|
|
||||||
DlgEditDeckInClipboard dlg(*getDeckList(), annotated, this);
|
|
||||||
if (!dlg.exec())
|
|
||||||
return;
|
|
||||||
|
|
||||||
setDeck(dlg.getDeckList());
|
|
||||||
setModified(true);
|
|
||||||
|
|
||||||
deckMenu->setSaveStatus(true);
|
|
||||||
}
|
|
||||||
|
|
||||||
void AbstractTabDeckEditor::actEditDeckInClipboard()
|
|
||||||
{
|
|
||||||
editDeckInClipboard(true);
|
|
||||||
}
|
|
||||||
|
|
||||||
void AbstractTabDeckEditor::actEditDeckInClipboardRaw()
|
|
||||||
{
|
|
||||||
editDeckInClipboard(false);
|
|
||||||
}
|
|
||||||
|
|
||||||
void AbstractTabDeckEditor::actSaveDeckToClipboard()
|
|
||||||
{
|
|
||||||
getDeckList()->saveToClipboard(true, true);
|
|
||||||
}
|
|
||||||
|
|
||||||
void AbstractTabDeckEditor::actSaveDeckToClipboardNoSetInfo()
|
|
||||||
{
|
|
||||||
getDeckList()->saveToClipboard(true, false);
|
|
||||||
}
|
|
||||||
|
|
||||||
void AbstractTabDeckEditor::actSaveDeckToClipboardRaw()
|
|
||||||
{
|
|
||||||
getDeckList()->saveToClipboard(false, true);
|
|
||||||
}
|
|
||||||
|
|
||||||
void AbstractTabDeckEditor::actSaveDeckToClipboardRawNoSetInfo()
|
|
||||||
{
|
|
||||||
getDeckList()->saveToClipboard(false, false);
|
|
||||||
}
|
|
||||||
|
|
||||||
void AbstractTabDeckEditor::actPrintDeck()
|
|
||||||
{
|
|
||||||
auto *dlg = new QPrintPreviewDialog(this);
|
|
||||||
connect(dlg, &QPrintPreviewDialog::paintRequested, deckDockWidget->deckModel, &DeckListModel::printDeckList);
|
|
||||||
dlg->exec();
|
|
||||||
}
|
|
||||||
|
|
||||||
void AbstractTabDeckEditor::exportToDecklistWebsite(DeckLoader::DecklistWebsite website)
|
|
||||||
{
|
|
||||||
// check if deck is not null
|
|
||||||
if (DeckLoader *const deck = getDeckList()) {
|
|
||||||
// Get the decklist url string from the deck loader class.
|
|
||||||
QString decklistUrlString = deck->exportDeckToDecklist(website);
|
|
||||||
// Check to make sure the string isn't empty.
|
|
||||||
if (QString::compare(decklistUrlString, "", Qt::CaseInsensitive) == 0) {
|
|
||||||
// Show an error if the deck is empty, and return.
|
|
||||||
QMessageBox::critical(this, tr("Error"), tr("There are no cards in your deck to be exported"));
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
// Encode the string recieved from the model to make sure all characters are encoded.
|
|
||||||
// first we put it into a qurl object
|
|
||||||
QUrl decklistUrl = QUrl(decklistUrlString);
|
|
||||||
// we get the correctly encoded url.
|
|
||||||
decklistUrlString = decklistUrl.toEncoded();
|
|
||||||
// We open the url in the user's default browser
|
|
||||||
QDesktopServices::openUrl(decklistUrlString);
|
|
||||||
} else {
|
|
||||||
// if there's no deck loader object, return an error
|
|
||||||
QMessageBox::critical(this, tr("Error"), tr("No deck was selected to be exported."));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Exports the deck to www.decklist.org (the old website)
|
|
||||||
*/
|
|
||||||
void AbstractTabDeckEditor::actExportDeckDecklist()
|
|
||||||
{
|
|
||||||
exportToDecklistWebsite(DeckLoader::DecklistOrg);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Exports the deck to www.decklist.xyz (the new website)
|
|
||||||
*/
|
|
||||||
void AbstractTabDeckEditor::actExportDeckDecklistXyz()
|
|
||||||
{
|
|
||||||
exportToDecklistWebsite(DeckLoader::DecklistXyz);
|
|
||||||
}
|
|
||||||
|
|
||||||
void AbstractTabDeckEditor::actAnalyzeDeckDeckstats()
|
|
||||||
{
|
|
||||||
auto *interface = new DeckStatsInterface(*databaseDisplayDockWidget->databaseModel->getDatabase(),
|
|
||||||
this); // it deletes itself when done
|
|
||||||
interface->analyzeDeck(getDeckList());
|
|
||||||
}
|
|
||||||
|
|
||||||
void AbstractTabDeckEditor::actAnalyzeDeckTappedout()
|
|
||||||
{
|
|
||||||
auto *interface = new TappedOutInterface(*databaseDisplayDockWidget->databaseModel->getDatabase(),
|
|
||||||
this); // it deletes itself when done
|
|
||||||
interface->analyzeDeck(getDeckList());
|
|
||||||
}
|
|
||||||
|
|
||||||
void AbstractTabDeckEditor::filterTreeChanged(FilterTree *filterTree)
|
|
||||||
{
|
|
||||||
databaseDisplayDockWidget->setFilterTree(filterTree);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Method uses to sync docks state with menu items state
|
|
||||||
bool AbstractTabDeckEditor::eventFilter(QObject *o, QEvent *e)
|
|
||||||
{
|
|
||||||
if (e->type() == QEvent::Close) {
|
|
||||||
if (o == cardInfoDockWidget) {
|
|
||||||
aCardInfoDockVisible->setChecked(false);
|
|
||||||
aCardInfoDockFloating->setEnabled(false);
|
|
||||||
} else if (o == deckDockWidget) {
|
|
||||||
aDeckDockVisible->setChecked(false);
|
|
||||||
aDeckDockFloating->setEnabled(false);
|
|
||||||
} else if (o == filterDockWidget) {
|
|
||||||
aFilterDockVisible->setChecked(false);
|
|
||||||
aFilterDockFloating->setEnabled(false);
|
|
||||||
} else if (o == printingSelectorDockWidget) {
|
|
||||||
aPrintingSelectorDockVisible->setChecked(false);
|
|
||||||
aPrintingSelectorDockFloating->setEnabled(false);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (o == this && e->type() == QEvent::Hide) {
|
|
||||||
LayoutsSettings &layouts = SettingsCache::instance().layouts();
|
|
||||||
layouts.setDeckEditorLayoutState(saveState());
|
|
||||||
layouts.setDeckEditorGeometry(saveGeometry());
|
|
||||||
layouts.setDeckEditorCardSize(cardInfoDockWidget->size());
|
|
||||||
layouts.setDeckEditorFilterSize(filterDockWidget->size());
|
|
||||||
layouts.setDeckEditorDeckSize(deckDockWidget->size());
|
|
||||||
layouts.setDeckEditorPrintingSelectorSize(printingSelectorDockWidget->size());
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool AbstractTabDeckEditor::confirmClose()
|
|
||||||
{
|
|
||||||
if (modified) {
|
|
||||||
tabSupervisor->setCurrentWidget(this);
|
|
||||||
int ret = createSaveConfirmationWindow()->exec();
|
|
||||||
if (ret == QMessageBox::Save)
|
|
||||||
return actSaveDeck();
|
|
||||||
else if (ret == QMessageBox::Cancel)
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
void AbstractTabDeckEditor::closeRequest(bool forced)
|
|
||||||
{
|
|
||||||
if (!forced && !confirmClose()) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
emit deckEditorClosing(this);
|
|
||||||
close();
|
|
||||||
}
|
|
||||||
|
|
@ -1,158 +0,0 @@
|
||||||
#ifndef TAB_GENERIC_DECK_EDITOR_H
|
|
||||||
#define TAB_GENERIC_DECK_EDITOR_H
|
|
||||||
|
|
||||||
#include "../../game/cards/card_info.h"
|
|
||||||
#include "../menus/deck_editor/deck_editor_menu.h"
|
|
||||||
#include "../ui/widgets/deck_editor/deck_editor_card_info_dock_widget.h"
|
|
||||||
#include "../ui/widgets/deck_editor/deck_editor_database_display_widget.h"
|
|
||||||
#include "../ui/widgets/deck_editor/deck_editor_deck_dock_widget.h"
|
|
||||||
#include "../ui/widgets/deck_editor/deck_editor_filter_dock_widget.h"
|
|
||||||
#include "../ui/widgets/deck_editor/deck_editor_printing_selector_dock_widget.h"
|
|
||||||
#include "../ui/widgets/visual_deck_storage/deck_preview/deck_preview_deck_tags_display_widget.h"
|
|
||||||
#include "tab.h"
|
|
||||||
|
|
||||||
class CardDatabaseModel;
|
|
||||||
class CardDatabaseDisplayModel;
|
|
||||||
|
|
||||||
class CardInfoFrameWidget;
|
|
||||||
class DeckLoader;
|
|
||||||
class DeckEditorMenu;
|
|
||||||
class DeckEditorCardInfoDockWidget;
|
|
||||||
class DeckEditorDatabaseDisplayWidget;
|
|
||||||
class DeckEditorDeckDockWidget;
|
|
||||||
class DeckEditorFilterDockWidget;
|
|
||||||
class DeckEditorPrintingSelectorDockWidget;
|
|
||||||
class DeckPreviewDeckTagsDisplayWidget;
|
|
||||||
class Response;
|
|
||||||
class FilterTreeModel;
|
|
||||||
class FilterBuilder;
|
|
||||||
|
|
||||||
class QTreeView;
|
|
||||||
class QTextEdit;
|
|
||||||
class QLabel;
|
|
||||||
class QComboBox;
|
|
||||||
class QGroupBox;
|
|
||||||
class QMessageBox;
|
|
||||||
class QHBoxLayout;
|
|
||||||
class QVBoxLayout;
|
|
||||||
class QPushButton;
|
|
||||||
class QDockWidget;
|
|
||||||
class QMenu;
|
|
||||||
class QAction;
|
|
||||||
|
|
||||||
class AbstractTabDeckEditor : public Tab
|
|
||||||
{
|
|
||||||
Q_OBJECT
|
|
||||||
|
|
||||||
friend class DeckEditorMenu;
|
|
||||||
|
|
||||||
public:
|
|
||||||
explicit AbstractTabDeckEditor(TabSupervisor *_tabSupervisor);
|
|
||||||
|
|
||||||
// UI and Navigation
|
|
||||||
virtual void createMenus() = 0;
|
|
||||||
[[nodiscard]] virtual QString getTabText() const override = 0;
|
|
||||||
bool confirmClose();
|
|
||||||
virtual void retranslateUi() override = 0;
|
|
||||||
|
|
||||||
// Deck Management
|
|
||||||
void openDeck(DeckLoader *deck);
|
|
||||||
DeckLoader *getDeckList() const;
|
|
||||||
void setModified(bool _windowModified);
|
|
||||||
|
|
||||||
// UI Elements
|
|
||||||
DeckEditorMenu *deckMenu;
|
|
||||||
DeckEditorDatabaseDisplayWidget *databaseDisplayDockWidget;
|
|
||||||
DeckEditorCardInfoDockWidget *cardInfoDockWidget;
|
|
||||||
DeckEditorDeckDockWidget *deckDockWidget;
|
|
||||||
DeckEditorFilterDockWidget *filterDockWidget;
|
|
||||||
DeckEditorPrintingSelectorDockWidget *printingSelectorDockWidget;
|
|
||||||
|
|
||||||
public slots:
|
|
||||||
virtual void onDeckChanged();
|
|
||||||
virtual void onDeckModified();
|
|
||||||
void updateCard(CardInfoPtr _card);
|
|
||||||
void actAddCard(CardInfoPtr info);
|
|
||||||
void actAddCardToSideboard(CardInfoPtr info);
|
|
||||||
void actDecrementCard(CardInfoPtr info);
|
|
||||||
void actDecrementCardFromSideboard(CardInfoPtr info);
|
|
||||||
void actOpenRecent(const QString &fileName);
|
|
||||||
void filterTreeChanged(FilterTree *filterTree);
|
|
||||||
void closeRequest(bool forced = false) override;
|
|
||||||
virtual void showPrintingSelector() = 0;
|
|
||||||
virtual void dockTopLevelChanged(bool topLevel) = 0;
|
|
||||||
|
|
||||||
signals:
|
|
||||||
void openDeckEditor(const DeckLoader *deckLoader);
|
|
||||||
void deckEditorClosing(AbstractTabDeckEditor *tab);
|
|
||||||
void decrementCard(CardInfoPtr card, QString zoneName);
|
|
||||||
|
|
||||||
protected slots:
|
|
||||||
// Deck Operations
|
|
||||||
virtual void actNewDeck();
|
|
||||||
void cleanDeckAndResetModified();
|
|
||||||
virtual void actLoadDeck();
|
|
||||||
bool actSaveDeck();
|
|
||||||
virtual bool actSaveDeckAs();
|
|
||||||
virtual void actLoadDeckFromClipboard();
|
|
||||||
void actEditDeckInClipboard();
|
|
||||||
void actEditDeckInClipboardRaw();
|
|
||||||
void actSaveDeckToClipboard();
|
|
||||||
void actSaveDeckToClipboardNoSetInfo();
|
|
||||||
void actSaveDeckToClipboardRaw();
|
|
||||||
void actSaveDeckToClipboardRawNoSetInfo();
|
|
||||||
void actPrintDeck();
|
|
||||||
void actExportDeckDecklist();
|
|
||||||
void actExportDeckDecklistXyz();
|
|
||||||
void actAnalyzeDeckDeckstats();
|
|
||||||
void actAnalyzeDeckTappedout();
|
|
||||||
|
|
||||||
// Remote Save
|
|
||||||
void saveDeckRemoteFinished(const Response &r);
|
|
||||||
|
|
||||||
// UI Layout Management
|
|
||||||
virtual void loadLayout() = 0;
|
|
||||||
virtual void restartLayout() = 0;
|
|
||||||
virtual void freeDocksSize() = 0;
|
|
||||||
virtual void refreshShortcuts() = 0;
|
|
||||||
|
|
||||||
bool eventFilter(QObject *o, QEvent *e) override;
|
|
||||||
virtual void dockVisibleTriggered() = 0;
|
|
||||||
virtual void dockFloatingTriggered() = 0;
|
|
||||||
|
|
||||||
private:
|
|
||||||
virtual void setDeck(DeckLoader *_deck);
|
|
||||||
void editDeckInClipboard(bool annotated);
|
|
||||||
void exportToDecklistWebsite(DeckLoader::DecklistWebsite website);
|
|
||||||
|
|
||||||
protected:
|
|
||||||
/**
|
|
||||||
* @brief Enum for selecting deck open location
|
|
||||||
*/
|
|
||||||
enum DeckOpenLocation
|
|
||||||
{
|
|
||||||
CANCELLED,
|
|
||||||
SAME_TAB,
|
|
||||||
NEW_TAB
|
|
||||||
};
|
|
||||||
|
|
||||||
DeckOpenLocation confirmOpen(bool openInSameTabIfBlank = true);
|
|
||||||
QMessageBox *createSaveConfirmationWindow();
|
|
||||||
bool isBlankNewDeck() const;
|
|
||||||
|
|
||||||
// Helper functions for card actions
|
|
||||||
void addCardHelper(CardInfoPtr info, QString zoneName);
|
|
||||||
void actSwapCard(CardInfoPtr info, QString zoneName);
|
|
||||||
virtual void openDeckFromFile(const QString &fileName, DeckOpenLocation deckOpenLocation);
|
|
||||||
|
|
||||||
// UI Menu Elements
|
|
||||||
QMenu *viewMenu, *cardInfoDockMenu, *deckDockMenu, *filterDockMenu, *printingSelectorDockMenu;
|
|
||||||
|
|
||||||
QAction *aResetLayout;
|
|
||||||
QAction *aCardInfoDockVisible, *aCardInfoDockFloating, *aDeckDockVisible, *aDeckDockFloating;
|
|
||||||
QAction *aFilterDockVisible, *aFilterDockFloating, *aPrintingSelectorDockVisible, *aPrintingSelectorDockFloating;
|
|
||||||
|
|
||||||
bool modified = false;
|
|
||||||
};
|
|
||||||
|
|
||||||
#endif // TAB_GENERIC_DECK_EDITOR_H
|
|
||||||
|
|
@ -1,174 +0,0 @@
|
||||||
#include "edhrec_commander_api_response_navigation_widget.h"
|
|
||||||
|
|
||||||
#include "../../tab_edhrec_main.h"
|
|
||||||
|
|
||||||
EdhrecCommanderApiResponseNavigationWidget::EdhrecCommanderApiResponseNavigationWidget(
|
|
||||||
QWidget *parent,
|
|
||||||
const EdhrecCommanderApiResponseCommanderDetails &_commanderDetails,
|
|
||||||
QString baseUrl)
|
|
||||||
: QWidget(parent), commanderDetails(_commanderDetails)
|
|
||||||
{
|
|
||||||
layout = new QGridLayout(this);
|
|
||||||
setLayout(layout);
|
|
||||||
|
|
||||||
gameChangerLabel = new QLabel(this);
|
|
||||||
budgetLabel = new QLabel(this);
|
|
||||||
|
|
||||||
comboPushButton = new QPushButton(this);
|
|
||||||
averageDeckPushButton = new QPushButton(this);
|
|
||||||
|
|
||||||
layout->addWidget(comboPushButton, 0, 0, 1, 1);
|
|
||||||
layout->addWidget(averageDeckPushButton, 0, 1, 1, 1);
|
|
||||||
|
|
||||||
layout->addWidget(gameChangerLabel, 1, 0, 1, 2);
|
|
||||||
|
|
||||||
for (int i = 0; i < gameChangerOptions.length(); i++) {
|
|
||||||
QString option = gameChangerOptions.at(i);
|
|
||||||
QString label = option.isEmpty() ? "All" : option.at(0).toUpper() + option.mid(1);
|
|
||||||
QPushButton *optionButton = new QPushButton(label, this);
|
|
||||||
gameChangerButtons[option] = optionButton;
|
|
||||||
layout->addWidget(optionButton, 2, i);
|
|
||||||
connect(optionButton, &QPushButton::clicked, this, [=, this]() {
|
|
||||||
selectedGameChanger = option;
|
|
||||||
updateOptionButtonSelection(gameChangerButtons, option);
|
|
||||||
actRequestCommanderNavigation();
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
layout->addWidget(budgetLabel, 3, 0, 1, 2);
|
|
||||||
|
|
||||||
for (int i = 0; i < budgetOptions.length(); i++) {
|
|
||||||
QString option = budgetOptions.at(i);
|
|
||||||
QString label = option.isEmpty() ? "Any" : option.at(0).toUpper() + option.mid(1);
|
|
||||||
QPushButton *btn = new QPushButton(label, this);
|
|
||||||
budgetButtons[option] = btn;
|
|
||||||
layout->addWidget(btn, 4, i);
|
|
||||||
connect(btn, &QPushButton::clicked, this, [=, this]() {
|
|
||||||
selectedBudget = option;
|
|
||||||
updateOptionButtonSelection(budgetButtons, option);
|
|
||||||
actRequestCommanderNavigation();
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
updateOptionButtonSelection(gameChangerButtons, "");
|
|
||||||
updateOptionButtonSelection(budgetButtons, "");
|
|
||||||
|
|
||||||
QWidget *currentParent = parentWidget();
|
|
||||||
TabEdhRecMain *parentTab = nullptr;
|
|
||||||
|
|
||||||
while (currentParent) {
|
|
||||||
if ((parentTab = qobject_cast<TabEdhRecMain *>(currentParent))) {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
currentParent = currentParent->parentWidget();
|
|
||||||
}
|
|
||||||
|
|
||||||
if (parentTab) {
|
|
||||||
connect(comboPushButton, &QPushButton::clicked, this,
|
|
||||||
&EdhrecCommanderApiResponseNavigationWidget::actRequestComboNavigation);
|
|
||||||
connect(averageDeckPushButton, &QPushButton::clicked, this,
|
|
||||||
&EdhrecCommanderApiResponseNavigationWidget::actRequestAverageDeckNavigation);
|
|
||||||
connect(this, &EdhrecCommanderApiResponseNavigationWidget::requestUrl, parentTab,
|
|
||||||
&TabEdhRecMain::actNavigatePage);
|
|
||||||
}
|
|
||||||
|
|
||||||
retranslateUi();
|
|
||||||
applyOptionsFromUrl(baseUrl);
|
|
||||||
}
|
|
||||||
|
|
||||||
void EdhrecCommanderApiResponseNavigationWidget::retranslateUi()
|
|
||||||
{
|
|
||||||
comboPushButton->setText(tr("Combos"));
|
|
||||||
averageDeckPushButton->setText(tr("Average Deck"));
|
|
||||||
gameChangerLabel->setText(tr("Game Changers"));
|
|
||||||
budgetLabel->setText(tr("Budget"));
|
|
||||||
}
|
|
||||||
|
|
||||||
void EdhrecCommanderApiResponseNavigationWidget::applyOptionsFromUrl(const QString &url)
|
|
||||||
{
|
|
||||||
QString cleanedUrl = url;
|
|
||||||
|
|
||||||
// Remove base and file extension
|
|
||||||
if (cleanedUrl.startsWith("https://json.edhrec.com/pages/")) {
|
|
||||||
cleanedUrl = cleanedUrl.mid(QString("https://json.edhrec.com/pages/").length());
|
|
||||||
}
|
|
||||||
if (cleanedUrl.endsWith(".json")) {
|
|
||||||
cleanedUrl.chop(5);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Expecting something like: "commanders/the-ur-dragon/core/expensive"
|
|
||||||
#if (QT_VERSION >= QT_VERSION_CHECK(5, 14, 0))
|
|
||||||
QStringList parts = cleanedUrl.split('/', Qt::SkipEmptyParts);
|
|
||||||
#else
|
|
||||||
QStringList parts = cleanedUrl.split('/', QString::SkipEmptyParts);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
if (parts.size() < 2) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
QString commanderName = parts[1];
|
|
||||||
QString gameChangerOpt, budgetOpt;
|
|
||||||
|
|
||||||
// Define valid sets
|
|
||||||
QSet<QString> validGameChangers = {"core", "upgraded", "optimized"};
|
|
||||||
QSet<QString> validBudgets = {"budget", "expensive"};
|
|
||||||
|
|
||||||
// Check remaining parts after commander
|
|
||||||
for (int i = 2; i < parts.size(); ++i) {
|
|
||||||
QString part = parts[i].toLower();
|
|
||||||
if (validGameChangers.contains(part)) {
|
|
||||||
gameChangerOpt = part;
|
|
||||||
} else if (validBudgets.contains(part)) {
|
|
||||||
budgetOpt = part;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Validate and apply
|
|
||||||
if (!gameChangerButtons.contains(gameChangerOpt)) {
|
|
||||||
gameChangerOpt.clear();
|
|
||||||
}
|
|
||||||
if (!budgetButtons.contains(budgetOpt)) {
|
|
||||||
budgetOpt.clear();
|
|
||||||
}
|
|
||||||
|
|
||||||
selectedGameChanger = gameChangerOpt;
|
|
||||||
selectedBudget = budgetOpt;
|
|
||||||
|
|
||||||
updateOptionButtonSelection(gameChangerButtons, selectedGameChanger);
|
|
||||||
updateOptionButtonSelection(budgetButtons, selectedBudget);
|
|
||||||
}
|
|
||||||
|
|
||||||
void EdhrecCommanderApiResponseNavigationWidget::updateOptionButtonSelection(QMap<QString, QPushButton *> &buttons,
|
|
||||||
const QString &selectedKey)
|
|
||||||
{
|
|
||||||
for (auto it = buttons.begin(); it != buttons.end(); ++it) {
|
|
||||||
it.value()->setStyleSheet(it.key() == selectedKey ? "background-color: lightblue; font-weight: bold;" : "");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
QString EdhrecCommanderApiResponseNavigationWidget::addNavigationOptionsToUrl(QString baseUrl)
|
|
||||||
{
|
|
||||||
if (!selectedGameChanger.isEmpty()) {
|
|
||||||
baseUrl += "/" + selectedGameChanger;
|
|
||||||
}
|
|
||||||
if (!selectedBudget.isEmpty()) {
|
|
||||||
baseUrl += "/" + selectedBudget;
|
|
||||||
}
|
|
||||||
return baseUrl;
|
|
||||||
}
|
|
||||||
|
|
||||||
void EdhrecCommanderApiResponseNavigationWidget::actRequestCommanderNavigation()
|
|
||||||
{
|
|
||||||
emit requestUrl(addNavigationOptionsToUrl("/commanders/" + commanderDetails.getSanitized()));
|
|
||||||
}
|
|
||||||
|
|
||||||
void EdhrecCommanderApiResponseNavigationWidget::actRequestComboNavigation()
|
|
||||||
{
|
|
||||||
emit requestUrl("/combos/" + commanderDetails.getSanitized());
|
|
||||||
}
|
|
||||||
|
|
||||||
void EdhrecCommanderApiResponseNavigationWidget::actRequestAverageDeckNavigation()
|
|
||||||
{
|
|
||||||
emit requestUrl(addNavigationOptionsToUrl("/average-decks/" + commanderDetails.getSanitized()));
|
|
||||||
}
|
|
||||||