Compare commits
804 Commits
2474-serve
...
2024-12-20
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
91d2485940 | ||
|
|
0d99b2bcf4 | ||
|
|
a54a424f84 | ||
|
|
3514699f5b | ||
|
|
d196988cab | ||
|
|
03aff83135 | ||
|
|
17e6bfaca6 | ||
|
|
90281262be | ||
|
|
5bbc118920 | ||
|
|
dde2f8b9ad | ||
|
|
d41aa30e10 | ||
|
|
231d0380a7 | ||
|
|
a5c509981b | ||
|
|
71b01e6110 | ||
|
|
c716f85962 | ||
|
|
245d51caea | ||
|
|
e588917f6c | ||
|
|
27e5d21b6b | ||
|
|
b894b75e6a | ||
|
|
116397cdb3 | ||
|
|
a6b5abf271 | ||
|
|
fd5a649246 | ||
|
|
e8e57989ba | ||
|
|
03db4ccce6 | ||
|
|
c9d5d5609c | ||
|
|
ac16206ddb | ||
|
|
5f8bcbd02d | ||
|
|
a0f74134bb | ||
|
|
0463a6fd70 | ||
|
|
a5de633c64 | ||
|
|
b2ad2acff3 | ||
|
|
4ee6ff73e0 | ||
|
|
628bdde939 | ||
|
|
e9b78c1c59 | ||
|
|
315cbc0925 | ||
|
|
69741d858c | ||
|
|
20d99a78b6 | ||
|
|
2d68393e07 | ||
|
|
8cb1470643 | ||
|
|
8d9b27bf47 | ||
|
|
0c5d9f1a7d | ||
|
|
a7d88c06c1 | ||
|
|
2735000fcf | ||
|
|
a39de270cd | ||
|
|
10f11213d3 | ||
|
|
3b49cbf73b | ||
|
|
e4cfe08113 | ||
|
|
fa02cb885c | ||
|
|
69b864fa02 | ||
|
|
b9ed9a6c0b | ||
|
|
5156495b47 | ||
|
|
b92047bc3f | ||
|
|
70559d32df | ||
|
|
bb84b75db9 | ||
|
|
f634177973 | ||
|
|
e33ff37c82 | ||
|
|
d2bc7f6ac0 | ||
|
|
5ef1ca06f5 | ||
|
|
1d8651bc00 | ||
|
|
17eabf2004 | ||
|
|
37bb1367db | ||
|
|
24b5dab456 | ||
|
|
c6bfc8b8ea | ||
|
|
f2b0fa164e | ||
|
|
0ca8bdb3a8 | ||
|
|
a8471f62bc | ||
|
|
5f1c03682f | ||
|
|
3255ed3ffb | ||
|
|
c51b54c0c5 | ||
|
|
a3f0807d47 | ||
|
|
27055944df | ||
|
|
7b1653034b | ||
|
|
39d8ca050f | ||
|
|
50274cb66d | ||
|
|
bd60a9fd2e | ||
|
|
83409c32c4 | ||
|
|
1bc92623dc | ||
|
|
f73196841a | ||
|
|
86a4b130ff | ||
|
|
f4e2f117c3 | ||
|
|
8ef92d26c5 | ||
|
|
c8336df49d | ||
|
|
c2fe3cda35 | ||
|
|
c54f47efbf | ||
|
|
3c40cc4b7d | ||
|
|
f0fb77bade | ||
|
|
e894e78346 | ||
|
|
dd04c610ec | ||
|
|
2e674efe50 | ||
|
|
4d394c31f9 | ||
|
|
11d58abbc3 | ||
|
|
5f4ad87a47 | ||
|
|
e43a21866c | ||
|
|
6652012f4c | ||
|
|
0c4e8ca290 | ||
|
|
230a2c5c62 | ||
|
|
590fb7f533 | ||
|
|
e8b88248f2 | ||
|
|
c6ba1b6a4e | ||
|
|
c4c52bd8c0 | ||
|
|
c633a792f5 | ||
|
|
8d5421d9da | ||
|
|
b041f4ace2 | ||
|
|
d26f96db9e | ||
|
|
fa999880ee | ||
|
|
d1e0f9dfc5 | ||
|
|
2d86938375 | ||
|
|
4865269a73 | ||
|
|
038ce3dcec | ||
|
|
43b997fe40 | ||
|
|
44e92f61ca | ||
|
|
b4bfa17cee | ||
|
|
500b694cc6 | ||
|
|
b998282304 | ||
|
|
b704216553 | ||
|
|
03ec02a749 | ||
|
|
248ea82573 | ||
|
|
bbe125beee | ||
|
|
95cd1c6f87 | ||
|
|
1c2107ae8f | ||
|
|
e826e17c6c | ||
|
|
b111f0921c | ||
|
|
090a48515c | ||
|
|
b8555d8c42 | ||
|
|
cf1f4f12a9 | ||
|
|
ef4413633a | ||
|
|
c5bb38e907 | ||
|
|
9f515fc804 | ||
|
|
245edcefdd | ||
|
|
153f73c308 | ||
|
|
315837b267 | ||
|
|
ea8da24215 | ||
|
|
1ab723ca64 | ||
|
|
f8bc6cf998 | ||
|
|
8687163cca | ||
|
|
e261e16d99 | ||
|
|
bdcd083eea | ||
|
|
c4bf9eb61c | ||
|
|
0994d10410 | ||
|
|
291c535edb | ||
|
|
f04702fdd1 | ||
|
|
b7fbc12ac0 | ||
|
|
e2ab8db958 | ||
|
|
34d70980e8 | ||
|
|
e45c4042fe | ||
|
|
ce8092318e | ||
|
|
c95cc1dd9d | ||
|
|
1f72877728 | ||
|
|
93b40343d9 | ||
|
|
ba10108207 | ||
|
|
c28f66d673 | ||
|
|
59f327f97a | ||
|
|
872c92a244 | ||
|
|
2303880b87 | ||
|
|
0e97cc1712 | ||
|
|
d550e42441 | ||
|
|
4279753030 | ||
|
|
2f6c018b7a | ||
|
|
be5d42baba | ||
|
|
f174614496 | ||
|
|
e8c7fba8b0 | ||
|
|
5c49283023 | ||
|
|
ad56b431a3 | ||
|
|
b0d8a33d5f | ||
|
|
1715bcb216 | ||
|
|
96caeaca72 | ||
|
|
c8723ae935 | ||
|
|
94e39c044c | ||
|
|
9776ea53c9 | ||
|
|
675d07dac0 | ||
|
|
1217820288 | ||
|
|
90e1a3cb76 | ||
|
|
7c1095ea50 | ||
|
|
203e916a07 | ||
|
|
7201e34b38 | ||
|
|
6d032c378f | ||
|
|
badd8952f5 | ||
|
|
7209eddb2d | ||
|
|
a7ffd43b29 | ||
|
|
724ba69483 | ||
|
|
1716801437 | ||
|
|
fa727524dc | ||
|
|
28f80e18a0 | ||
|
|
4acc8bfe80 | ||
|
|
9f86ed7887 | ||
|
|
cb18a55338 | ||
|
|
78a928464c | ||
|
|
07a8cd0a5f | ||
|
|
b73ef58567 | ||
|
|
519531f3a0 | ||
|
|
4b8e47d079 | ||
|
|
ed170f7e07 | ||
|
|
6bb559874c | ||
|
|
9cd68e25b3 | ||
|
|
72ac441598 | ||
|
|
f5fe56c85d | ||
|
|
d4fc1be2cc | ||
|
|
7b3617a273 | ||
|
|
3e8adae3de | ||
|
|
9943133d6d | ||
|
|
a5706a47af | ||
|
|
b3b911c64d | ||
|
|
05beb4fcaf | ||
|
|
324b50e381 | ||
|
|
186f4289e9 | ||
|
|
cb90a8356b | ||
|
|
e9c502ab32 | ||
|
|
f728520e97 | ||
|
|
c1b0d50237 | ||
|
|
b9cfc29059 | ||
|
|
6bf7c79891 | ||
|
|
2bd0e58354 | ||
|
|
ee674cb0cf | ||
|
|
dd1b354d48 | ||
|
|
d3e96f4a99 | ||
|
|
90e2eb3db9 | ||
|
|
102be6a350 | ||
|
|
be6152948c | ||
|
|
f14bf4b205 | ||
|
|
123ac2ca25 | ||
|
|
7216b976ec | ||
|
|
7fb698cfbf | ||
|
|
b0470ab678 | ||
|
|
0deb037035 | ||
|
|
064b362d60 | ||
|
|
6bbe228a84 | ||
|
|
a8ba8b6ab5 | ||
|
|
e850f6c2a5 | ||
|
|
e9eb7d6db1 | ||
|
|
56d21321d0 | ||
|
|
3888a74212 | ||
|
|
0035e29f9e | ||
|
|
90679d5669 | ||
|
|
ac5dc2578a | ||
|
|
671e6823be | ||
|
|
0d76662311 | ||
|
|
8dd59cf3cf | ||
|
|
332d25dc00 | ||
|
|
0fa81a77dc | ||
|
|
9a74d8f72d | ||
|
|
8c539351e3 | ||
|
|
e3552fc0ae | ||
|
|
ca308636c3 | ||
|
|
244cb847fb | ||
|
|
176c52daf2 | ||
|
|
ee3525ec64 | ||
|
|
adce921be7 | ||
|
|
20ceb1c284 | ||
|
|
db4364b8f8 | ||
|
|
48d6435e09 | ||
|
|
7c20e9ab34 | ||
|
|
cb52605928 | ||
|
|
bd3100dcda | ||
|
|
afb7c35cfd | ||
|
|
5b694a55d2 | ||
|
|
f750a4cd72 | ||
|
|
eddeaaf52a | ||
|
|
2b42bee424 | ||
|
|
b9706c0cc1 | ||
|
|
800b21b000 | ||
|
|
d1736a25bb | ||
|
|
f269e5fe58 | ||
|
|
45a5296013 | ||
|
|
5f0ab2a177 | ||
|
|
70ab02987a | ||
|
|
421da882d8 | ||
|
|
1a40102f71 | ||
|
|
1fbc10cd77 | ||
|
|
87462398d8 | ||
|
|
b33246b29f | ||
|
|
dbffe30f63 | ||
|
|
9ce450d0b0 | ||
|
|
f5f8acf1fd | ||
|
|
304ed3cd60 | ||
|
|
07248692ce | ||
|
|
cef99cba71 | ||
|
|
cab5f29b57 | ||
|
|
55a2f75d16 | ||
|
|
a416ee8f2b | ||
|
|
c14936c63c | ||
|
|
42e7a8b423 | ||
|
|
4558b1c7ef | ||
|
|
f444ba9665 | ||
|
|
787c8d740b | ||
|
|
ef38a8bb2b | ||
|
|
b5d35d346a | ||
|
|
9a7b15d19b | ||
|
|
ba35a11e82 | ||
|
|
00c9efe541 | ||
|
|
44d1ab348b | ||
|
|
f25e4785ae | ||
|
|
4c290aec57 | ||
|
|
42d1d66d9b | ||
|
|
06c25301a5 | ||
|
|
43dbb45cc6 | ||
|
|
9fb62de5cb | ||
|
|
b282df2e27 | ||
|
|
da8f57f397 | ||
|
|
e9f1992c7f | ||
|
|
2c94a6a64e | ||
|
|
6a5e0a8501 | ||
|
|
26d7fe2ff0 | ||
|
|
5854a635ca | ||
|
|
3d4858b840 | ||
|
|
dec2a252fa | ||
|
|
f619ef23fd | ||
|
|
72743e834e | ||
|
|
8e4ddf366c | ||
|
|
b99bd0176a | ||
|
|
a68b98b245 | ||
|
|
a69d6ff1b4 | ||
|
|
ec679e95fd | ||
|
|
3f78235a74 | ||
|
|
c8a2fd78b0 | ||
|
|
45cf08111a | ||
|
|
527ac36129 | ||
|
|
a7232513a7 | ||
|
|
90f187e885 | ||
|
|
235adbbdf1 | ||
|
|
40c88fe385 | ||
|
|
54b7943d17 | ||
|
|
2f100f2ba3 | ||
|
|
b5305aa5e4 | ||
|
|
a2624e36f3 | ||
|
|
540511befd | ||
|
|
014e73d569 | ||
|
|
e2c256db5b | ||
|
|
28aa473362 | ||
|
|
b79506fbcf | ||
|
|
afbd7252ac | ||
|
|
3e5b7cd392 | ||
|
|
77622095d5 | ||
|
|
8ee71300a2 | ||
|
|
d79971edbc | ||
|
|
273ebb22e4 | ||
|
|
6b86e4d463 | ||
|
|
a8e1dc3b18 | ||
|
|
a95b338c80 | ||
|
|
3e90f109a2 | ||
|
|
b02adccf87 | ||
|
|
accd5e4df7 | ||
|
|
d007196059 | ||
|
|
79501a4af7 | ||
|
|
64c6611ea5 | ||
|
|
a532a63403 | ||
|
|
c10c69d0a9 | ||
|
|
191d5a83a9 | ||
|
|
de69e2c41f | ||
|
|
6d200d17b7 | ||
|
|
4899b6cfef | ||
|
|
0ff59e6d1e | ||
|
|
00a2a8ab71 | ||
|
|
2c702d3579 | ||
|
|
b464fa8d99 | ||
|
|
2b330940e1 | ||
|
|
0d0337f091 | ||
|
|
533045445a | ||
|
|
21f7dd5eba | ||
|
|
f5b973e15c | ||
|
|
baaf261116 | ||
|
|
92ed53e13a | ||
|
|
2a54e9d7d1 | ||
|
|
9577ada171 | ||
|
|
217dc09c0f | ||
|
|
7108eb42c8 | ||
|
|
eb3ce1fd7e | ||
|
|
c88d44e16c | ||
|
|
ec2ad4c713 | ||
|
|
4c04b4ef5a | ||
|
|
88b861d632 | ||
|
|
408a13c937 | ||
|
|
7d0a255a49 | ||
|
|
252883f67e | ||
|
|
bf08a04cda | ||
|
|
6928a2bd98 | ||
|
|
bb16ae09ef | ||
|
|
81d031ca0f | ||
|
|
8203a2fdeb | ||
|
|
75f0d60dff | ||
|
|
992e28797f | ||
|
|
92f941a54c | ||
|
|
4c31527832 | ||
|
|
febe029ed4 | ||
|
|
1d780058c8 | ||
|
|
513fcb0908 | ||
|
|
4bb13677c8 | ||
|
|
a5baf4303c | ||
|
|
7aba404f2e | ||
|
|
1b7e8f3a16 | ||
|
|
5cf93ad61c | ||
|
|
5a52e085a7 | ||
|
|
5d31b70406 | ||
|
|
2885f93fdf | ||
|
|
d225f55e5a | ||
|
|
69edc73585 | ||
|
|
ead1143f2e | ||
|
|
2fc85e0c08 | ||
|
|
fcafcb340a | ||
|
|
ae9b8b8f34 | ||
|
|
994704d353 | ||
|
|
d61c604bf4 | ||
|
|
baaf22d0c4 | ||
|
|
368ff1793f | ||
|
|
3253ad64fd | ||
|
|
1e70989f38 | ||
|
|
f6634de18d | ||
|
|
7903cd520a | ||
|
|
26d1fcc944 | ||
|
|
59d4e64a8d | ||
|
|
1347d88ddb | ||
|
|
6981cca2ae | ||
|
|
4d6c9ede8c | ||
|
|
e845c95816 | ||
|
|
a9f2fc427b | ||
|
|
07e6aadbbe | ||
|
|
86881bbbc3 | ||
|
|
1f15445c69 | ||
|
|
811ee54c76 | ||
|
|
d1a40fd36e | ||
|
|
a3d3aaaca8 | ||
|
|
c5aaa0bc2e | ||
|
|
6dc9f004ce | ||
|
|
6ce346af4a | ||
|
|
37879c4255 | ||
|
|
0683d1aced | ||
|
|
73c5956ece | ||
|
|
7c27e955d5 | ||
|
|
6ef394000b | ||
|
|
755a09bd83 | ||
|
|
8b1daa21ef | ||
|
|
691bcb9338 | ||
|
|
911a303326 | ||
|
|
5652b56b45 | ||
|
|
26acfd5102 | ||
|
|
f789e02096 | ||
|
|
43eee6b32e | ||
|
|
45d86e7ab7 | ||
|
|
b0845837c2 | ||
|
|
a10226d096 | ||
|
|
5e3a524401 | ||
|
|
b095d9b82f | ||
|
|
4cb7240f9a | ||
|
|
a87c66885c | ||
|
|
ac300b0b6d | ||
|
|
013bb8269f | ||
|
|
7712862036 | ||
|
|
656e3230de | ||
|
|
915c14f6cf | ||
|
|
ed32e72dc1 | ||
|
|
d684a9c5fc | ||
|
|
6f360374cc | ||
|
|
bbbf3e2a65 | ||
|
|
b1ef8220ee | ||
|
|
ebebb9c4bb | ||
|
|
586f23cfa9 | ||
|
|
f75ff2a7c8 | ||
|
|
e9ba195d7d | ||
|
|
19333c53f6 | ||
|
|
36e5a399d5 | ||
|
|
dde0f568d9 | ||
|
|
3bc90003b3 | ||
|
|
689f65b38a | ||
|
|
154e1084ba | ||
|
|
6df5cece04 | ||
|
|
c8bb8b0aae | ||
|
|
1e995cd97c | ||
|
|
c25bf491e4 | ||
|
|
affc288144 | ||
|
|
051be37419 | ||
|
|
c0bd49cf13 | ||
|
|
2fe572c398 | ||
|
|
904e740460 | ||
|
|
00add2a527 | ||
|
|
bbe43d4246 | ||
|
|
0280fea3e6 | ||
|
|
a65ce8694c | ||
|
|
c9ddd042fc | ||
|
|
da9222929b | ||
|
|
b858e36183 | ||
|
|
ebe2c494aa | ||
|
|
fac7bfaa92 | ||
|
|
0d05f9097d | ||
|
|
8db9475804 | ||
|
|
294229622d | ||
|
|
5cf9023a21 | ||
|
|
66d24f086e | ||
|
|
19a7c4092c | ||
|
|
b27cb58727 | ||
|
|
472f401590 | ||
|
|
744099277a | ||
|
|
dff25a175b | ||
|
|
61f1141fe8 | ||
|
|
0a73162bcf | ||
|
|
5f32892e75 | ||
|
|
ae7437750b | ||
|
|
046a3649ed | ||
|
|
890810f5b9 | ||
|
|
8fb561b4c4 | ||
|
|
b9c4b496e4 | ||
|
|
ff6f28390a | ||
|
|
e034de9083 | ||
|
|
ecf57b4226 | ||
|
|
63fe34437a | ||
|
|
1062894397 | ||
|
|
5969656429 | ||
|
|
ad0f313c9d | ||
|
|
6c004155ff | ||
|
|
88a8ee09bd | ||
|
|
7d1f082b27 | ||
|
|
1c48656623 | ||
|
|
b940e17fe7 | ||
|
|
8e954b10e6 | ||
|
|
1b4543aa11 | ||
|
|
406c0b17ae | ||
|
|
09de56ac87 | ||
|
|
07ea2d4334 | ||
|
|
c5fac2ee35 | ||
|
|
a5b8245227 | ||
|
|
073349fd05 | ||
|
|
00ed5c370c | ||
|
|
8e1d7d12e0 | ||
|
|
1811f7305e | ||
|
|
06bfc0291a | ||
|
|
b722864caf | ||
|
|
18e27ef932 | ||
|
|
7e3a669af0 | ||
|
|
aa6a0313e9 | ||
|
|
9bbe2f36bc | ||
|
|
0c7830b53c | ||
|
|
2b0a9975be | ||
|
|
df0b867d9e | ||
|
|
d27f108cbd | ||
|
|
fd0076e920 | ||
|
|
db5f6e01c9 | ||
|
|
f3255180ec | ||
|
|
039d058770 | ||
|
|
401fdcaf7a | ||
|
|
1bfcca91be | ||
|
|
3edb862561 | ||
|
|
00c0162da3 | ||
|
|
6fa5f4f9a5 | ||
|
|
db528c6762 | ||
|
|
0c54cdf6bc | ||
|
|
b63145c0a1 | ||
|
|
1ddc9cc929 | ||
|
|
a0deb73df6 | ||
|
|
0457e65751 | ||
|
|
d5b36e8b8a | ||
|
|
f595a61d50 | ||
|
|
dbf6cd745e | ||
|
|
0ce813b826 | ||
|
|
77be6a120c | ||
|
|
37053cc909 | ||
|
|
34e951298f | ||
|
|
8845a23d5d | ||
|
|
38606bdb87 | ||
|
|
56a51c7834 | ||
|
|
402c09e028 | ||
|
|
9e702ec358 | ||
|
|
c047a8ae3c | ||
|
|
b4740ad395 | ||
|
|
99b0abe7fe | ||
|
|
b0239c11ab | ||
|
|
46cf50d468 | ||
|
|
9f9581c2be | ||
|
|
f3cf1f0dde | ||
|
|
2f62671d8a | ||
|
|
589fbcdcd5 | ||
|
|
51b24bb92c | ||
|
|
6e00db4ef6 | ||
|
|
0d842b5a35 | ||
|
|
8441cb7ba9 | ||
|
|
4aaedf64d2 | ||
|
|
45d838a0b3 | ||
|
|
ca5f1dd434 | ||
|
|
d07bf1211a | ||
|
|
0966a8e90f | ||
|
|
68074b0f74 | ||
|
|
f11f072e0a | ||
|
|
3064621a7e | ||
|
|
0405c82cb2 | ||
|
|
8e9d4e3a67 | ||
|
|
ef78fdf342 | ||
|
|
a49c4865bb | ||
|
|
1a94261490 | ||
|
|
e10446f5b8 | ||
|
|
2081639970 | ||
|
|
8cbc4c91f6 | ||
|
|
b8cd3af21f | ||
|
|
1e8464c1d4 | ||
|
|
5df069ab19 | ||
|
|
752ba7d905 | ||
|
|
9cf7621102 | ||
|
|
9330774632 | ||
|
|
e33f802ae8 | ||
|
|
48c6458766 | ||
|
|
35fe6f624c | ||
|
|
e2251fe06b | ||
|
|
a5511190a3 | ||
|
|
eba9c097f6 | ||
|
|
58d024d067 | ||
|
|
be7b172e55 | ||
|
|
a8b79fd020 | ||
|
|
14fcb2e5d7 | ||
|
|
02935be14f | ||
|
|
bec02b4952 | ||
|
|
79f590c99a | ||
|
|
ade3e81682 | ||
|
|
45cbdad5fb | ||
|
|
3536fa8a75 | ||
|
|
b0c7b9078d | ||
|
|
964207d04f | ||
|
|
feee9cc328 | ||
|
|
4a563a131b | ||
|
|
69f035f017 | ||
|
|
daa89a9fb4 | ||
|
|
44297dcd1c | ||
|
|
80f613a77a | ||
|
|
776aa5c0ff | ||
|
|
446f9be24d | ||
|
|
5d9d91262b | ||
|
|
fe63dfa762 | ||
|
|
a76a3e5db6 | ||
|
|
faecfd65fe | ||
|
|
0063493066 | ||
|
|
2a7268c088 | ||
|
|
7fa1936d0f | ||
|
|
0f0e0193c1 | ||
|
|
0337f58088 | ||
|
|
53c6440cac | ||
|
|
28f3229a0e | ||
|
|
3dff63069e | ||
|
|
1eea8e9a37 | ||
|
|
2de863f645 | ||
|
|
dc8603596d | ||
|
|
fd0620445c | ||
|
|
04274d2497 | ||
|
|
c69e77f330 | ||
|
|
1988e4558f | ||
|
|
9246c190fa | ||
|
|
d30691559a | ||
|
|
46fe0cd725 | ||
|
|
45d62b6880 | ||
|
|
db85ec48c7 | ||
|
|
1976c4caed | ||
|
|
2c3eab9b0c | ||
|
|
27b7ebe208 | ||
|
|
a135ad064a | ||
|
|
e8d5715f7a | ||
|
|
9cec0852bb | ||
|
|
ca618c6cc1 | ||
|
|
c5e0b08800 | ||
|
|
568a4973fa | ||
|
|
5508699e92 | ||
|
|
71dd6b8a30 | ||
|
|
18a07274d4 | ||
|
|
1eb766b9d8 | ||
|
|
e84409c0cf | ||
|
|
91dc8b3b08 | ||
|
|
63b4f9b2f0 | ||
|
|
7285f24a29 | ||
|
|
17efe8c003 | ||
|
|
1815094249 | ||
|
|
a80c756dcb | ||
|
|
361833e023 | ||
|
|
632e44b0b7 | ||
|
|
8dd1e39ea9 | ||
|
|
0f18fa9546 | ||
|
|
7bfefee073 | ||
|
|
0ff7472ce5 | ||
|
|
8fb0baa449 | ||
|
|
57c02dcd5a | ||
|
|
b072b540a2 | ||
|
|
cd431594e2 | ||
|
|
e4c98e2ab8 | ||
|
|
32b557b862 | ||
|
|
e690b45f27 | ||
|
|
eadcdc6f7c | ||
|
|
b187fb52e0 | ||
|
|
7e89933552 | ||
|
|
096a472ed0 | ||
|
|
365b0a31ed | ||
|
|
e8fd2ce2aa | ||
|
|
a3fc9b6ee5 | ||
|
|
9eebc590c1 | ||
|
|
f840dcbd66 | ||
|
|
8879fc2e39 | ||
|
|
cd29e2f252 | ||
|
|
bcf505c98b | ||
|
|
933f3e1392 | ||
|
|
ba0462b24f | ||
|
|
7e8a63cd62 | ||
|
|
03e109ef12 | ||
|
|
257f2eb34c | ||
|
|
53728598fe | ||
|
|
3b98eb77f5 | ||
|
|
b6df5a4ac3 | ||
|
|
f54165025e | ||
|
|
013137c418 | ||
|
|
faf02100a5 | ||
|
|
a8b34d51a5 | ||
|
|
2ef3e6fc93 | ||
|
|
a3a1e20074 | ||
|
|
ce54aa6813 | ||
|
|
5139039402 | ||
|
|
f4adf79ad9 | ||
|
|
f10f9ada3a | ||
|
|
965a6cdde7 | ||
|
|
62c0825874 | ||
|
|
4e918f0f5d | ||
|
|
3e3154a58c | ||
|
|
22e2e442f5 | ||
|
|
c0c4a6df50 | ||
|
|
36ba9c2d94 | ||
|
|
1288795de9 | ||
|
|
0380de9571 | ||
|
|
6ac3852995 | ||
|
|
1d8fb79e11 | ||
|
|
e084bd18a9 | ||
|
|
7c3cc527f6 | ||
|
|
218aec07f8 | ||
|
|
ada13f6578 | ||
|
|
8682367b52 | ||
|
|
a80f3b77da | ||
|
|
fd8bf66acd | ||
|
|
115ed78059 | ||
|
|
00ca69fced | ||
|
|
6c21855f98 | ||
|
|
3830c85ce6 | ||
|
|
0dc4cc7e03 | ||
|
|
8f5bfd1f87 | ||
|
|
95d6efcdbf | ||
|
|
63cf0ae764 | ||
|
|
f1e79707e8 | ||
|
|
9073cb53a8 | ||
|
|
d018070891 | ||
|
|
837924b819 | ||
|
|
00cfb1347a | ||
|
|
0ce2e61db9 | ||
|
|
1854e3440b | ||
|
|
a0260eb0b2 | ||
|
|
c874f201c3 | ||
|
|
7072f24103 | ||
|
|
eb4914d36f | ||
|
|
6d27631764 | ||
|
|
a522255baf | ||
|
|
7eb2e36740 | ||
|
|
2d8f01b2e9 | ||
|
|
6f95556632 | ||
|
|
0326f0d4c9 | ||
|
|
389f7fdc25 | ||
|
|
11b2942d09 | ||
|
|
52cc725de4 | ||
|
|
a304d4235d | ||
|
|
4d7024e066 | ||
|
|
4ce928eb41 | ||
|
|
b172172be1 | ||
|
|
5fd86954d6 | ||
|
|
55c4207182 | ||
|
|
440311c939 | ||
|
|
81059fa3d8 | ||
|
|
9f856acba0 | ||
|
|
69474bad22 | ||
|
|
fafdb65453 | ||
|
|
477faafb84 | ||
|
|
abe4f5ca60 | ||
|
|
a86aeb00c5 | ||
|
|
754b9bdc5a | ||
|
|
5a3f2fcff7 | ||
|
|
9411396b97 | ||
|
|
18ad3cf4a5 | ||
|
|
a7f19f7848 | ||
|
|
63839eb464 | ||
|
|
eb60fec8e2 | ||
|
|
4427ad1451 | ||
|
|
e77e439c4b | ||
|
|
45b16ba78d | ||
|
|
e68305d7bf | ||
|
|
c2150fd9bd | ||
|
|
ea8201af5c | ||
|
|
9d27b36704 | ||
|
|
d519f992d2 | ||
|
|
1b3da22dd7 | ||
|
|
317f29b321 | ||
|
|
c32f2190bd | ||
|
|
d49ec0088f | ||
|
|
5b197c072e | ||
|
|
294b4332cc | ||
|
|
6d6b7b4537 | ||
|
|
b3fd5857f8 | ||
|
|
78c18b7db6 | ||
|
|
099d524a42 | ||
|
|
612edae5f6 | ||
|
|
b065ada633 | ||
|
|
e4bac025dd | ||
|
|
923b4acdbc | ||
|
|
2703db75d0 | ||
|
|
d932581f4a | ||
|
|
6e3fd30fcb | ||
|
|
46b34d6515 | ||
|
|
073c531855 | ||
|
|
9118871afb | ||
|
|
a7d485d6f5 | ||
|
|
a719e3b306 |
107
.appveyor.yml
@@ -1,107 +0,0 @@
|
|||||||
version: build {build}
|
|
||||||
|
|
||||||
# Skipping commits affecting specific files (GitHub only).
|
|
||||||
# More details here: https://www.appveyor.com/docs/appveyor-yml and https://www.appveyor.com/docs/how-to/filtering-commits
|
|
||||||
skip_commits:
|
|
||||||
files:
|
|
||||||
- .ci/travis-*
|
|
||||||
- .github/
|
|
||||||
- .tx/
|
|
||||||
- webclient/
|
|
||||||
- .clang-format
|
|
||||||
- .*ignore
|
|
||||||
- .codacy.yml
|
|
||||||
- .gitlab-ci.yml
|
|
||||||
- .travis.yml
|
|
||||||
- '**/*.md'
|
|
||||||
- Dockerfile
|
|
||||||
- LICENSE
|
|
||||||
|
|
||||||
skip_branch_with_pr: true
|
|
||||||
|
|
||||||
clone_depth: 50 #same as travis, see https://www.appveyor.com/blog/2014/06/04/shallow-clone-for-git-repositories/
|
|
||||||
|
|
||||||
image: Visual Studio 2017
|
|
||||||
|
|
||||||
cache:
|
|
||||||
- c:\Tools\vcpkg\installed
|
|
||||||
|
|
||||||
environment:
|
|
||||||
matrix:
|
|
||||||
- target_arch: win64
|
|
||||||
qt_ver: 5.9\msvc2017_64
|
|
||||||
cmake_generator: Visual Studio 15 2017 Win64
|
|
||||||
cmake_toolset: v141,host=x64
|
|
||||||
vcpkg_arch: x64
|
|
||||||
|
|
||||||
- target_arch: win32
|
|
||||||
qt_ver: 5.9\msvc2015 # Qt doesn't provide a msvc2017_32
|
|
||||||
cmake_generator: Visual Studio 15 2017
|
|
||||||
cmake_toolset: v141
|
|
||||||
vcpkg_arch: x86
|
|
||||||
|
|
||||||
install:
|
|
||||||
- vcpkg remove --outdated --recurse
|
|
||||||
- vcpkg install openssl protobuf liblzma zlib --triplet %vcpkg_arch%-windows
|
|
||||||
|
|
||||||
services:
|
|
||||||
- mysql
|
|
||||||
|
|
||||||
build_script:
|
|
||||||
- ps: |
|
|
||||||
New-Item -ItemType directory -Path $env:APPVEYOR_BUILD_FOLDER\build
|
|
||||||
Set-Location -Path $env:APPVEYOR_BUILD_FOLDER\build
|
|
||||||
$vcpkgbindir = "C:\Tools\vcpkg\installed\$env:vcpkg_arch-windows\bin"
|
|
||||||
$mysqldll = "c:\Program Files\MySQL\MySQL Server 5.7\lib\libmysql.dll"
|
|
||||||
cmake --version
|
|
||||||
cmake .. -G "$env:cmake_generator" -T "$env:cmake_toolset" "-DCMAKE_TOOLCHAIN_FILE=c:/tools/vcpkg/scripts/buildsystems/vcpkg.cmake" "-DCMAKE_PREFIX_PATH=c:/Qt/$env:qt_ver;$vcpkgbindir" "-DWITH_SERVER=1" "-DMYSQLCLIENT_LIBRARIES=$mysqldll"
|
|
||||||
- msbuild PACKAGE.vcxproj /p:Configuration=Release
|
|
||||||
- ps: |
|
|
||||||
$exe = dir -name *.exe
|
|
||||||
$new_name = $exe.Replace(".exe", "-${env:target_arch}.exe")
|
|
||||||
Push-AppveyorArtifact $exe -FileName $new_name
|
|
||||||
$cmake_name = $exe.Replace(".exe", "-${env:target_arch}.cmake.txt")
|
|
||||||
Push-AppveyorArtifact CMakeCache.txt -FileName $cmake_name
|
|
||||||
$json = New-Object PSObject
|
|
||||||
(New-Object PSObject | Add-Member -PassThru NoteProperty bin $new_name | Add-Member -PassThru NoteProperty cmake $cmake_name | Add-Member -PassThru NoteProperty commit $env:APPVEYOR_REPO_COMMIT) | ConvertTo-JSON | Out-File -FilePath "latest-$env:target_arch" -Encoding ASCII
|
|
||||||
Push-AppveyorArtifact "latest-$env:target_arch"
|
|
||||||
$version = $matches['content']
|
|
||||||
|
|
||||||
test: off
|
|
||||||
|
|
||||||
|
|
||||||
# Builds for pull requests skip the deployment step altogether
|
|
||||||
deploy:
|
|
||||||
# Deploy configuration for "beta" releases
|
|
||||||
- provider: GitHub
|
|
||||||
auth_token:
|
|
||||||
secure: p+7wPVry2XEa6TBm9XH8IaQZbBmXQ/J2ldbGmcIxUZD3NkkPrSRRlmE7Of1CBBIO
|
|
||||||
tag: "$(APPVEYOR_REPO_TAG_NAME)"
|
|
||||||
release: "Cockatrice $(APPVEYOR_REPO_TAG_NAME)"
|
|
||||||
description: "Beta release of Cockatrice"
|
|
||||||
artifact: /.*\.exe/
|
|
||||||
force_update: true
|
|
||||||
draft: false
|
|
||||||
prerelease: true
|
|
||||||
on:
|
|
||||||
APPVEYOR_REPO_TAG: true
|
|
||||||
APPVEYOR_REPO_TAG_NAME: /([0-9]|[1-9][0-9])(\.([0-9]|[1-9][0-9])){2}-beta(\.([2-9]|[1-9][0-9]))?$/ # regex to match semver naming convention for beta pre-releases
|
|
||||||
|
|
||||||
# Deploy configuration for "stable" releases
|
|
||||||
- provider: GitHub
|
|
||||||
auth_token:
|
|
||||||
secure: p+7wPVry2XEa6TBm9XH8IaQZbBmXQ/J2ldbGmcIxUZD3NkkPrSRRlmE7Of1CBBIO
|
|
||||||
tag: "$(APPVEYOR_REPO_TAG_NAME)"
|
|
||||||
release: "Cockatrice $(APPVEYOR_REPO_TAG_NAME)"
|
|
||||||
artifact: /.*\.exe/
|
|
||||||
force_update: true
|
|
||||||
draft: false
|
|
||||||
prerelease: false
|
|
||||||
on:
|
|
||||||
APPVEYOR_REPO_TAG: true
|
|
||||||
APPVEYOR_REPO_TAG_NAME: /([0-9]|[1-9][0-9])(\.([0-9]|[1-9][0-9])){2}$/ # regex to match semver naming convention for stable full releases
|
|
||||||
|
|
||||||
|
|
||||||
# Announcements of build image updates: https://www.appveyor.com/updates/
|
|
||||||
# Official validator for ".appveyor.yml" config file: https://ci.appveyor.com/tools/validate-yaml
|
|
||||||
# AppVeyor config documentation: https://www.appveyor.com/docs/build-configuration/
|
|
||||||
18
.ci/Arch/Dockerfile
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
from archlinux:latest
|
||||||
|
|
||||||
|
RUN pacman --sync --refresh --sysupgrade --needed --noconfirm \
|
||||||
|
base-devel \
|
||||||
|
ccache \
|
||||||
|
cmake \
|
||||||
|
git \
|
||||||
|
gtest \
|
||||||
|
mariadb-libs \
|
||||||
|
protobuf \
|
||||||
|
qt6-base \
|
||||||
|
qt6-imageformats \
|
||||||
|
qt6-multimedia \
|
||||||
|
qt6-svg \
|
||||||
|
qt6-tools \
|
||||||
|
qt6-translations \
|
||||||
|
qt6-websockets \
|
||||||
|
&& pacman --sync --clean --clean --noconfirm
|
||||||
25
.ci/Debian11/Dockerfile
Normal file
@@ -0,0 +1,25 @@
|
|||||||
|
FROM debian:11
|
||||||
|
|
||||||
|
RUN apt-get update && \
|
||||||
|
DEBIAN_FRONTEND=noninteractive apt-get install -y --no-install-recommends \
|
||||||
|
build-essential \
|
||||||
|
ccache \
|
||||||
|
clang-format \
|
||||||
|
cmake \
|
||||||
|
file \
|
||||||
|
g++ \
|
||||||
|
git \
|
||||||
|
liblzma-dev \
|
||||||
|
libmariadb-dev-compat \
|
||||||
|
libprotobuf-dev \
|
||||||
|
libqt5multimedia5-plugins \
|
||||||
|
libqt5sql5-mysql \
|
||||||
|
libqt5svg5-dev \
|
||||||
|
libqt5websockets5-dev \
|
||||||
|
protobuf-compiler \
|
||||||
|
qt5-image-formats-plugins \
|
||||||
|
qtmultimedia5-dev \
|
||||||
|
qttools5-dev \
|
||||||
|
qttools5-dev-tools \
|
||||||
|
&& apt-get clean \
|
||||||
|
&& rm -rf /var/lib/apt/lists/*
|
||||||
27
.ci/Debian12/Dockerfile
Normal file
@@ -0,0 +1,27 @@
|
|||||||
|
FROM debian:12
|
||||||
|
|
||||||
|
RUN apt-get update && \
|
||||||
|
DEBIAN_FRONTEND=noninteractive apt-get install -y --no-install-recommends \
|
||||||
|
build-essential \
|
||||||
|
ccache \
|
||||||
|
clang-format \
|
||||||
|
cmake \
|
||||||
|
file \
|
||||||
|
g++ \
|
||||||
|
git \
|
||||||
|
libgl-dev \
|
||||||
|
liblzma-dev \
|
||||||
|
libmariadb-dev-compat \
|
||||||
|
libprotobuf-dev \
|
||||||
|
libqt6multimedia6 \
|
||||||
|
libqt6sql6-mysql \
|
||||||
|
qt6-svg-dev \
|
||||||
|
qt6-websockets-dev \
|
||||||
|
protobuf-compiler \
|
||||||
|
qt6-image-formats-plugins \
|
||||||
|
qt6-l10n-tools \
|
||||||
|
qt6-multimedia-dev \
|
||||||
|
qt6-tools-dev \
|
||||||
|
qt6-tools-dev-tools \
|
||||||
|
&& apt-get clean \
|
||||||
|
&& rm -rf /var/lib/apt/lists/*
|
||||||
@@ -1,19 +0,0 @@
|
|||||||
FROM fedora:29
|
|
||||||
|
|
||||||
RUN dnf install -y \
|
|
||||||
@development-tools \
|
|
||||||
ccache \
|
|
||||||
cmake \
|
|
||||||
desktop-file-utils \
|
|
||||||
file \
|
|
||||||
gcc-c++ \
|
|
||||||
hicolor-icon-theme \
|
|
||||||
libappstream-glib \
|
|
||||||
protobuf-devel \
|
|
||||||
qt5-{qttools,qtsvg,qtmultimedia,qtwebsockets}-devel-5.11.1-2.fc29 \
|
|
||||||
rpm-build \
|
|
||||||
sqlite-devel \
|
|
||||||
wget \
|
|
||||||
zlib-devel \
|
|
||||||
xz-devel \
|
|
||||||
&& dnf clean all
|
|
||||||
15
.ci/Fedora40/Dockerfile
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
FROM fedora:40
|
||||||
|
|
||||||
|
RUN dnf install -y \
|
||||||
|
ccache \
|
||||||
|
cmake \
|
||||||
|
gcc-c++ \
|
||||||
|
git \
|
||||||
|
mariadb-devel \
|
||||||
|
protobuf-devel \
|
||||||
|
qt6-{qttools,qtsvg,qtmultimedia,qtwebsockets}-devel \
|
||||||
|
qt6-qtimageformats \
|
||||||
|
rpm-build \
|
||||||
|
xz-devel \
|
||||||
|
zlib-devel \
|
||||||
|
&& dnf clean all
|
||||||
15
.ci/Fedora41/Dockerfile
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
FROM fedora:41
|
||||||
|
|
||||||
|
RUN dnf install -y \
|
||||||
|
ccache \
|
||||||
|
cmake \
|
||||||
|
gcc-c++ \
|
||||||
|
git \
|
||||||
|
mariadb-devel \
|
||||||
|
protobuf-devel \
|
||||||
|
qt6-{qttools,qtsvg,qtmultimedia,qtwebsockets}-devel \
|
||||||
|
qt6-qtimageformats \
|
||||||
|
rpm-build \
|
||||||
|
xz-devel \
|
||||||
|
zlib-devel \
|
||||||
|
&& dnf clean all
|
||||||
@@ -1,22 +1,26 @@
|
|||||||
FROM ubuntu:bionic
|
FROM ubuntu:20.04
|
||||||
|
|
||||||
RUN apt-get update && apt-get install -y --no-install-recommends \
|
RUN apt-get update && \
|
||||||
|
DEBIAN_FRONTEND=noninteractive apt-get install -y --no-install-recommends \
|
||||||
build-essential \
|
build-essential \
|
||||||
|
ccache \
|
||||||
clang-format \
|
clang-format \
|
||||||
|
cmake \
|
||||||
file \
|
file \
|
||||||
g++ \
|
g++ \
|
||||||
git \
|
git \
|
||||||
ccache \
|
|
||||||
cmake \
|
|
||||||
liblzma-dev \
|
liblzma-dev \
|
||||||
|
libmariadb-dev-compat \
|
||||||
libprotobuf-dev \
|
libprotobuf-dev \
|
||||||
libqt5multimedia5-plugins \
|
libqt5multimedia5-plugins \
|
||||||
libqt5svg5-dev \
|
|
||||||
libqt5sql5-mysql \
|
libqt5sql5-mysql \
|
||||||
|
libqt5svg5-dev \
|
||||||
libqt5websockets5-dev \
|
libqt5websockets5-dev \
|
||||||
protobuf-compiler \
|
protobuf-compiler \
|
||||||
qt5-default \
|
qt5-default \
|
||||||
|
qt5-image-formats-plugins \
|
||||||
|
qtmultimedia5-dev \
|
||||||
qttools5-dev \
|
qttools5-dev \
|
||||||
qttools5-dev-tools \
|
qttools5-dev-tools \
|
||||||
qtmultimedia5-dev \
|
&& apt-get clean \
|
||||||
&& rm -rf /var/lib/apt/lists/*
|
&& rm -rf /var/lib/apt/lists/*
|
||||||
27
.ci/Ubuntu22.04/Dockerfile
Normal file
@@ -0,0 +1,27 @@
|
|||||||
|
FROM ubuntu:22.04
|
||||||
|
|
||||||
|
RUN apt-get update && \
|
||||||
|
DEBIAN_FRONTEND=noninteractive apt-get install -y --no-install-recommends \
|
||||||
|
build-essential \
|
||||||
|
ccache \
|
||||||
|
clang-format \
|
||||||
|
cmake \
|
||||||
|
file \
|
||||||
|
g++ \
|
||||||
|
git \
|
||||||
|
libgl-dev \
|
||||||
|
liblzma-dev \
|
||||||
|
libmariadb-dev-compat \
|
||||||
|
libprotobuf-dev \
|
||||||
|
libqt6multimedia6 \
|
||||||
|
libqt6sql6-mysql \
|
||||||
|
libqt6svg6-dev \
|
||||||
|
libqt6websockets6-dev \
|
||||||
|
protobuf-compiler \
|
||||||
|
qt6-image-formats-plugins \
|
||||||
|
qt6-l10n-tools \
|
||||||
|
qt6-multimedia-dev \
|
||||||
|
qt6-tools-dev \
|
||||||
|
qt6-tools-dev-tools \
|
||||||
|
&& apt-get clean \
|
||||||
|
&& rm -rf /var/lib/apt/lists/*
|
||||||
27
.ci/Ubuntu24.04/Dockerfile
Normal file
@@ -0,0 +1,27 @@
|
|||||||
|
FROM ubuntu:24.04
|
||||||
|
|
||||||
|
RUN apt-get update && \
|
||||||
|
DEBIAN_FRONTEND=noninteractive apt-get install -y --no-install-recommends \
|
||||||
|
build-essential \
|
||||||
|
ccache \
|
||||||
|
clang-format \
|
||||||
|
cmake \
|
||||||
|
file \
|
||||||
|
g++ \
|
||||||
|
git \
|
||||||
|
libgl-dev \
|
||||||
|
liblzma-dev \
|
||||||
|
libmariadb-dev-compat \
|
||||||
|
libprotobuf-dev \
|
||||||
|
libqt6multimedia6 \
|
||||||
|
libqt6sql6-mysql \
|
||||||
|
qt6-svg-dev \
|
||||||
|
qt6-websockets-dev \
|
||||||
|
protobuf-compiler \
|
||||||
|
qt6-image-formats-plugins \
|
||||||
|
qt6-l10n-tools \
|
||||||
|
qt6-multimedia-dev \
|
||||||
|
qt6-tools-dev \
|
||||||
|
qt6-tools-dev-tools \
|
||||||
|
&& apt-get clean \
|
||||||
|
&& rm -rf /var/lib/apt/lists/*
|
||||||
207
.ci/compile.sh
Executable file
@@ -0,0 +1,207 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
|
||||||
|
# This script is to be used by the ci environment from the project root directory, do not use it from somewhere else.
|
||||||
|
|
||||||
|
# Compiles cockatrice inside of a ci environment
|
||||||
|
# --install runs make install
|
||||||
|
# --package [<package type>] runs make package, optionally force the type
|
||||||
|
# --suffix <suffix> renames package with this suffix, requires arg
|
||||||
|
# --server compiles servatrice
|
||||||
|
# --test runs tests
|
||||||
|
# --debug or --release sets the build type ie CMAKE_BUILD_TYPE
|
||||||
|
# --ccache [<size>] uses ccache and shows stats, optionally provide size
|
||||||
|
# --dir <dir> sets the name of the build dir, default is "build"
|
||||||
|
# --parallel <core count> sets how many cores cmake should build with in parallel
|
||||||
|
# uses env: BUILDTYPE MAKE_INSTALL MAKE_PACKAGE PACKAGE_TYPE PACKAGE_SUFFIX MAKE_SERVER MAKE_TEST USE_CCACHE CCACHE_SIZE BUILD_DIR PARALLEL_COUNT
|
||||||
|
# (correspond to args: --debug/--release --install --package <package type> --suffix <suffix> --server --test --ccache <ccache_size> --dir <dir> --parallel <core_count>)
|
||||||
|
# exitcode: 1 for failure, 3 for invalid arguments
|
||||||
|
|
||||||
|
# Read arguments
|
||||||
|
while [[ $# != 0 ]]; do
|
||||||
|
case "$1" in
|
||||||
|
'--')
|
||||||
|
shift
|
||||||
|
;;
|
||||||
|
'--install')
|
||||||
|
MAKE_INSTALL=1
|
||||||
|
shift
|
||||||
|
;;
|
||||||
|
'--package')
|
||||||
|
MAKE_PACKAGE=1
|
||||||
|
shift
|
||||||
|
if [[ $# != 0 && ${1:0:1} != - ]]; then
|
||||||
|
PACKAGE_TYPE="$1"
|
||||||
|
shift
|
||||||
|
fi
|
||||||
|
;;
|
||||||
|
'--suffix')
|
||||||
|
shift
|
||||||
|
if [[ $# == 0 ]]; then
|
||||||
|
echo "::error file=$0::--suffix expects an argument"
|
||||||
|
exit 3
|
||||||
|
fi
|
||||||
|
PACKAGE_SUFFIX="$1"
|
||||||
|
shift
|
||||||
|
;;
|
||||||
|
'--server')
|
||||||
|
MAKE_SERVER=1
|
||||||
|
shift
|
||||||
|
;;
|
||||||
|
'--test')
|
||||||
|
MAKE_TEST=1
|
||||||
|
shift
|
||||||
|
;;
|
||||||
|
'--debug')
|
||||||
|
BUILDTYPE="Debug"
|
||||||
|
shift
|
||||||
|
;;
|
||||||
|
'--release')
|
||||||
|
BUILDTYPE="Release"
|
||||||
|
shift
|
||||||
|
;;
|
||||||
|
'--ccache')
|
||||||
|
USE_CCACHE=1
|
||||||
|
shift
|
||||||
|
if [[ $# != 0 && ${1:0:1} != - ]]; then
|
||||||
|
CCACHE_SIZE="$1"
|
||||||
|
shift
|
||||||
|
fi
|
||||||
|
;;
|
||||||
|
'--dir')
|
||||||
|
shift
|
||||||
|
if [[ $# == 0 ]]; then
|
||||||
|
echo "::error file=$0::--dir expects an argument"
|
||||||
|
exit 3
|
||||||
|
fi
|
||||||
|
BUILD_DIR="$1"
|
||||||
|
shift
|
||||||
|
;;
|
||||||
|
'--parallel')
|
||||||
|
shift
|
||||||
|
if [[ $# == 0 ]]; then
|
||||||
|
echo "::error file=$0::--parallel expects an argument"
|
||||||
|
exit 3
|
||||||
|
fi
|
||||||
|
PARALLEL_COUNT="$1"
|
||||||
|
shift
|
||||||
|
;;
|
||||||
|
*)
|
||||||
|
echo "::error file=$0::unrecognized option: $1"
|
||||||
|
exit 3
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
done
|
||||||
|
|
||||||
|
set -e
|
||||||
|
|
||||||
|
# Setup
|
||||||
|
./servatrice/check_schema_version.sh
|
||||||
|
if [[ ! $BUILDTYPE ]]; then
|
||||||
|
BUILDTYPE=Release
|
||||||
|
fi
|
||||||
|
if [[ ! $BUILD_DIR ]]; then
|
||||||
|
BUILD_DIR="build"
|
||||||
|
fi
|
||||||
|
mkdir -p "$BUILD_DIR"
|
||||||
|
cd "$BUILD_DIR"
|
||||||
|
|
||||||
|
# Add cmake flags
|
||||||
|
flags=("-DCMAKE_BUILD_TYPE=$BUILDTYPE")
|
||||||
|
if [[ $MAKE_SERVER ]]; then
|
||||||
|
flags+=("-DWITH_SERVER=1")
|
||||||
|
fi
|
||||||
|
if [[ $MAKE_TEST ]]; then
|
||||||
|
flags+=("-DTEST=1")
|
||||||
|
fi
|
||||||
|
if [[ $USE_CCACHE ]]; then
|
||||||
|
flags+=("-DUSE_CCACHE=1")
|
||||||
|
if [[ $CCACHE_SIZE ]]; then
|
||||||
|
# note, this setting persists after running the script
|
||||||
|
ccache --max-size "$CCACHE_SIZE"
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
if [[ $PACKAGE_TYPE ]]; then
|
||||||
|
flags+=("-DCPACK_GENERATOR=$PACKAGE_TYPE")
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Add cmake --build flags
|
||||||
|
buildflags=(--config "$BUILDTYPE")
|
||||||
|
if [[ $PARALLEL_COUNT ]]; then
|
||||||
|
if [[ $(cmake --build /not_a_dir --parallel 2>&1 | head -1) =~ parallel ]]; then
|
||||||
|
# workaround for bionic having an old cmake
|
||||||
|
echo "this version of cmake does not support --parallel, using native build tool -j instead"
|
||||||
|
buildflags+=(-- -j "$PARALLEL_COUNT")
|
||||||
|
# note, no normal build flags should be added after this
|
||||||
|
else
|
||||||
|
buildflags+=(--parallel "$PARALLEL_COUNT")
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
function ccachestatsverbose() {
|
||||||
|
# note, verbose only works on newer ccache, discard the error
|
||||||
|
local got
|
||||||
|
if got="$(ccache --show-stats --verbose 2>/dev/null)"; then
|
||||||
|
echo "$got"
|
||||||
|
else
|
||||||
|
ccache --show-stats
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
|
# Compile
|
||||||
|
if [[ $USE_CCACHE ]]; then
|
||||||
|
echo "::group::Show ccache stats"
|
||||||
|
ccachestatsverbose
|
||||||
|
echo "::endgroup::"
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo "::group::Configure cmake"
|
||||||
|
cmake --version
|
||||||
|
cmake .. "${flags[@]}"
|
||||||
|
echo "::endgroup::"
|
||||||
|
|
||||||
|
echo "::group::Build project"
|
||||||
|
if [[ $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
|
||||||
|
cmake --build . "${buildflags[@]}" -- -p:UseMultiToolTask=true -p:EnableClServerMode=true
|
||||||
|
else
|
||||||
|
cmake --build . "${buildflags[@]}"
|
||||||
|
fi
|
||||||
|
echo "::endgroup::"
|
||||||
|
|
||||||
|
if [[ $USE_CCACHE ]]; then
|
||||||
|
echo "::group::Show ccache stats again"
|
||||||
|
ccachestatsverbose
|
||||||
|
echo "::endgroup::"
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [[ $MAKE_TEST ]]; then
|
||||||
|
echo "::group::Run tests"
|
||||||
|
ctest -C "$BUILDTYPE" --output-on-failure
|
||||||
|
echo "::endgroup::"
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [[ $MAKE_INSTALL ]]; then
|
||||||
|
echo "::group::Install"
|
||||||
|
cmake --build . --target install --config "$BUILDTYPE"
|
||||||
|
echo "::endgroup::"
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [[ $MAKE_PACKAGE ]]; then
|
||||||
|
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"
|
||||||
|
echo "::endgroup::"
|
||||||
|
|
||||||
|
if [[ $PACKAGE_SUFFIX ]]; then
|
||||||
|
echo "::group::Update package name"
|
||||||
|
cd ..
|
||||||
|
BUILD_DIR="$BUILD_DIR" .ci/name_build.sh "$PACKAGE_SUFFIX"
|
||||||
|
echo "::endgroup::"
|
||||||
|
fi
|
||||||
|
fi
|
||||||
165
.ci/docker.sh
Normal file
@@ -0,0 +1,165 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
|
||||||
|
# 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.
|
||||||
|
# <arg> 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
|
||||||
|
# --build builds the image from the Dockerfile in .ci/$NAME
|
||||||
|
# --save stores the image, if an image was loaded it will not be stored
|
||||||
|
# --interactive immediately starts the image interactively for debugging
|
||||||
|
# --set-cache <location> sets the location to cache the image or for ccache
|
||||||
|
# requires: docker
|
||||||
|
# uses env: NAME 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
|
||||||
|
# exitcode: 1 for failure, 2 for missing dockerfile, 3 for invalid arguments
|
||||||
|
export BUILD_SCRIPT=".ci/compile.sh"
|
||||||
|
|
||||||
|
project_name="cockatrice"
|
||||||
|
save_extension=".tar.gz"
|
||||||
|
image_cache="image"
|
||||||
|
ccache_cache=".ccache"
|
||||||
|
|
||||||
|
# Read arguments
|
||||||
|
while [[ $# != 0 ]]; do
|
||||||
|
case "$1" in
|
||||||
|
'--build')
|
||||||
|
BUILD=1
|
||||||
|
shift
|
||||||
|
;;
|
||||||
|
'--get')
|
||||||
|
GET=1
|
||||||
|
shift
|
||||||
|
;;
|
||||||
|
'--interactive')
|
||||||
|
INTERACTIVE=1
|
||||||
|
shift
|
||||||
|
;;
|
||||||
|
'--save')
|
||||||
|
SAVE=1
|
||||||
|
shift
|
||||||
|
;;
|
||||||
|
'--set-cache')
|
||||||
|
CACHE=$2
|
||||||
|
if ! [[ -d $CACHE ]]; then
|
||||||
|
echo "could not find cache path: $CACHE" >&2
|
||||||
|
return 3
|
||||||
|
fi
|
||||||
|
shift 2
|
||||||
|
;;
|
||||||
|
*)
|
||||||
|
if [[ ${1:0:1} == - ]]; then
|
||||||
|
echo "unrecognized option: $1"
|
||||||
|
return 3
|
||||||
|
fi
|
||||||
|
NAME="$1"
|
||||||
|
shift
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
done
|
||||||
|
|
||||||
|
# Setup
|
||||||
|
if ! [[ $NAME ]]; then
|
||||||
|
echo "no build name given" >&2
|
||||||
|
return 3
|
||||||
|
fi
|
||||||
|
|
||||||
|
export IMAGE_NAME="${project_name,,}_${NAME,,}" # lower case
|
||||||
|
|
||||||
|
docker_dir=".ci/$NAME"
|
||||||
|
if ! [[ -r $docker_dir/Dockerfile ]]; then
|
||||||
|
echo "could not find Dockerfile in $docker_dir" >&2
|
||||||
|
return 2 # even if the image is cached, we do not want to run if there is no way to build this image
|
||||||
|
fi
|
||||||
|
|
||||||
|
if ! [[ $CACHE ]]; then
|
||||||
|
echo "cache dir is not set!" >&2
|
||||||
|
CACHE="$(mktemp -d)"
|
||||||
|
echo "set cache dir to $CACHE" >&2
|
||||||
|
fi
|
||||||
|
if ! [[ -d $CACHE ]]; then
|
||||||
|
echo "could not find cache dir: $CACHE" >&2
|
||||||
|
mkdir -p "$CACHE"
|
||||||
|
unset GET # the dir is empty
|
||||||
|
fi
|
||||||
|
if [[ $GET || $SAVE ]]; then
|
||||||
|
img_dir="$CACHE/$image_cache"
|
||||||
|
img_save="$img_dir/$IMAGE_NAME$save_extension"
|
||||||
|
if ! [[ -d $img_dir ]]; then
|
||||||
|
echo "could not find image dir: $img_dir" >&2
|
||||||
|
mkdir -p "$img_dir"
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
export CCACHE_DIR="$CACHE/$ccache_cache"
|
||||||
|
if ! [[ -d $CCACHE_DIR ]]; then
|
||||||
|
echo "could not find ccache dir: $CCACHE_DIR" >&2
|
||||||
|
mkdir -p "$CCACHE_DIR"
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Get the docker image from previously stored save
|
||||||
|
if [[ $GET ]]; then
|
||||||
|
if [[ $img_save ]] && docker load --input "$img_save"; then
|
||||||
|
echo "loaded image"
|
||||||
|
docker images
|
||||||
|
unset BUILD # do not overwrite the loaded image with build
|
||||||
|
unset SAVE # do not overwrite the stored image with the same image
|
||||||
|
if [[ $(find "$CCACHE_DIR" -type f -print -quit) ]]; then # check contents of ccache
|
||||||
|
echo "setting ccache to readonly"
|
||||||
|
export RUN_ARGS="$RUN_ARGS -e CCACHE_READONLY=1 -e CCACHE_NOSTATS=1" # do not overwrite ccache
|
||||||
|
else
|
||||||
|
echo "ccache is empty: $(find "$CCACHE_DIR")" >&2
|
||||||
|
fi
|
||||||
|
else
|
||||||
|
echo "could not load cached image, building instead" >&2
|
||||||
|
BUILD=1
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Build the docker image from dockerfile
|
||||||
|
if [[ $BUILD ]]; then
|
||||||
|
if docker build --tag "$IMAGE_NAME" "$docker_dir"; then
|
||||||
|
echo "built image"
|
||||||
|
docker images
|
||||||
|
else
|
||||||
|
echo "could not build image $IMAGE_NAME" >&2
|
||||||
|
return 1
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Save docker image to cache (compressed)
|
||||||
|
if [[ $SAVE ]]; then
|
||||||
|
if [[ $img_save ]] && docker save --output "$img_save" "$IMAGE_NAME"; then
|
||||||
|
echo "saved image to: $img_save"
|
||||||
|
else
|
||||||
|
echo "could not save image $IMAGE_NAME" >&2
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Set compile function, runs the compile script on the image, passes arguments to the script
|
||||||
|
function RUN ()
|
||||||
|
{
|
||||||
|
echo "running image:"
|
||||||
|
if [[ $(docker images) =~ "$IMAGE_NAME" ]]; then
|
||||||
|
local args=(--mount "type=bind,source=$PWD,target=/src")
|
||||||
|
args+=(--workdir "/src")
|
||||||
|
args+=(--user "$(id -u):$(id -g)")
|
||||||
|
if [[ $CCACHE_DIR ]]; then
|
||||||
|
args+=(--mount "type=bind,source=$CCACHE_DIR,target=/.ccache")
|
||||||
|
args+=(--env "CCACHE_DIR=/.ccache")
|
||||||
|
fi
|
||||||
|
docker run "${args[@]}" $RUN_ARGS "$IMAGE_NAME" bash "$BUILD_SCRIPT" $RUN_OPTS "$@"
|
||||||
|
return $?
|
||||||
|
else
|
||||||
|
echo "could not find docker image: $IMAGE_NAME" >&2
|
||||||
|
return 3
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
|
# for debugging, start the docker image interactively instead of building
|
||||||
|
# starts immediately, does not require sourcing or RUN
|
||||||
|
if [[ $INTERACTIVE ]]; then
|
||||||
|
export BUILD_SCRIPT="-i"
|
||||||
|
export RUN_ARGS="$RUN_ARGS -it"
|
||||||
|
RUN
|
||||||
|
fi
|
||||||
69
.ci/lint_cpp.sh
Executable file
@@ -0,0 +1,69 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
|
||||||
|
# fetch master branch
|
||||||
|
git fetch origin master
|
||||||
|
|
||||||
|
# unshallow if needed
|
||||||
|
echo "Finding merge base"
|
||||||
|
if ! git merge-base origin/master HEAD; then
|
||||||
|
echo "Could not find merge base, unshallowing repo"
|
||||||
|
git fetch --unshallow
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Check formatting using format.sh
|
||||||
|
echo "Checking your code using clang-format/cmake-format..."
|
||||||
|
|
||||||
|
diff="$(./format.sh --diff --cmake --cf-version --branch origin/master)"
|
||||||
|
err=$?
|
||||||
|
|
||||||
|
case $err in
|
||||||
|
1)
|
||||||
|
cat <<EOM
|
||||||
|
|
||||||
|
***********************************************************
|
||||||
|
*** ***
|
||||||
|
*** Your code does not comply with our style guide. ***
|
||||||
|
*** ***
|
||||||
|
*** Please correct it or run the "format.sh" script. ***
|
||||||
|
*** Then commit and push those changes to this branch. ***
|
||||||
|
*** Check our CONTRIBUTING.md file for more details. ***
|
||||||
|
*** ***
|
||||||
|
*** Thank you ❤️ ***
|
||||||
|
*** ***
|
||||||
|
***********************************************************
|
||||||
|
|
||||||
|
Used version:
|
||||||
|
${diff%%
|
||||||
|
----------
|
||||||
|
*}
|
||||||
|
|
||||||
|
The following changes should be made:
|
||||||
|
${diff#*
|
||||||
|
----------
|
||||||
|
}
|
||||||
|
|
||||||
|
Exiting...
|
||||||
|
EOM
|
||||||
|
exit 2
|
||||||
|
;;
|
||||||
|
|
||||||
|
0)
|
||||||
|
cat <<EOM
|
||||||
|
|
||||||
|
***********************************************************
|
||||||
|
*** ***
|
||||||
|
*** Your code complies with our style guide! ***
|
||||||
|
*** ***
|
||||||
|
*** Awesome 👍 ***
|
||||||
|
*** ***
|
||||||
|
***********************************************************
|
||||||
|
|
||||||
|
Exiting...
|
||||||
|
EOM
|
||||||
|
exit 0
|
||||||
|
;;
|
||||||
|
|
||||||
|
*)
|
||||||
|
echo "Something went wrong in our formatting checks: format.sh returned $err" >&2
|
||||||
|
;;
|
||||||
|
esac
|
||||||
55
.ci/name_build.sh
Executable file
@@ -0,0 +1,55 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
# used by the ci to rename build artifacts
|
||||||
|
# renames the file to [original name][SUFFIX].[original extension]
|
||||||
|
# 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
|
||||||
|
# adds output to GITHUB_OUTPUT
|
||||||
|
builddir="${BUILD_DIR:=.}"
|
||||||
|
findrx="Cockatrice-*.*"
|
||||||
|
|
||||||
|
if [[ $1 ]]; then
|
||||||
|
SUFFIX="$1"
|
||||||
|
fi
|
||||||
|
|
||||||
|
# check env
|
||||||
|
if [[ ! $SUFFIX ]]; then
|
||||||
|
echo "::error file=$0::SUFFIX is missing"
|
||||||
|
exit 2
|
||||||
|
fi
|
||||||
|
|
||||||
|
set -e
|
||||||
|
|
||||||
|
# find file
|
||||||
|
found="$(find "$builddir" -maxdepth 1 -type f -name "$findrx" -print -quit)"
|
||||||
|
path="${found%/*}" # remove all after last /
|
||||||
|
file="${found##*/}" # remove all before last /
|
||||||
|
if [[ ! $file ]]; then
|
||||||
|
echo "::error file=$0::could not find package"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
oldpwd="$PWD"
|
||||||
|
if ! cd "$path"; then
|
||||||
|
echo "::error file=$0::could not get file path"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
# set filename
|
||||||
|
name="${file%.*}" # remove all after last .
|
||||||
|
new_name="$name$SUFFIX."
|
||||||
|
if [[ $MAKE_ZIP ]]; then
|
||||||
|
filename="${new_name}zip"
|
||||||
|
echo "creating zip '$filename' from '$file'"
|
||||||
|
zip "$filename" "$file"
|
||||||
|
else
|
||||||
|
extension="${file##*.}" # remove all before last .
|
||||||
|
filename="$new_name$extension"
|
||||||
|
echo "renaming '$file' to '$filename'"
|
||||||
|
mv "$file" "$filename"
|
||||||
|
fi
|
||||||
|
|
||||||
|
cd "$oldpwd"
|
||||||
|
relative_path="$path/$filename"
|
||||||
|
ls -l "$relative_path"
|
||||||
|
echo "path=$relative_path" >>"$GITHUB_OUTPUT"
|
||||||
|
echo "name=$filename" >>"$GITHUB_OUTPUT"
|
||||||
113
.ci/prep_release.sh
Executable file
@@ -0,0 +1,113 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
# sets the properties of ci releases
|
||||||
|
# this doesn't have to be 100% foolproof
|
||||||
|
# the releases are first made as drafts and will be vetted by a human
|
||||||
|
# it just has to provide a template
|
||||||
|
# this requires the repo to be unshallowed
|
||||||
|
# adds output to GITHUB_OUTPUT
|
||||||
|
template_path=".ci/release_template.md"
|
||||||
|
body_path="/tmp/release.md"
|
||||||
|
beta_regex='beta'
|
||||||
|
name_regex='set\(GIT_TAG_RELEASENAME "([[:print:]]+)")'
|
||||||
|
whitespace='^\s*$'
|
||||||
|
|
||||||
|
if [[ $1 ]]; then
|
||||||
|
TAG="$1"
|
||||||
|
fi
|
||||||
|
|
||||||
|
# check env
|
||||||
|
if [[ ! $TAG ]]; then
|
||||||
|
echo "::error file=$0::TAG is missing"
|
||||||
|
exit 2
|
||||||
|
fi
|
||||||
|
|
||||||
|
# create title
|
||||||
|
if [[ $TAG =~ $beta_regex ]]; then
|
||||||
|
echo "is_beta=yes" >>"$GITHUB_OUTPUT"
|
||||||
|
title="$TAG"
|
||||||
|
echo "creating beta release '$title'"
|
||||||
|
elif [[ ! $(cat CMakeLists.txt) =~ $name_regex ]]; then
|
||||||
|
echo "::error file=$0::could not find releasename in CMakeLists.txt"
|
||||||
|
exit 1
|
||||||
|
else
|
||||||
|
echo "is_beta=no" >>"$GITHUB_OUTPUT"
|
||||||
|
name="${BASH_REMATCH[1]}"
|
||||||
|
version="${TAG##*-}"
|
||||||
|
title="Cockatrice $version: $name"
|
||||||
|
no_beta=1
|
||||||
|
echo "friendly_name=$name" >>"$GITHUB_OUTPUT"
|
||||||
|
echo "creating full release '$title'"
|
||||||
|
fi
|
||||||
|
echo "title=$title" >>"$GITHUB_OUTPUT"
|
||||||
|
|
||||||
|
# add release notes template
|
||||||
|
if [[ $no_beta ]]; then
|
||||||
|
body="$(cat "$template_path")"
|
||||||
|
if [[ ! $body ]]; then
|
||||||
|
echo "::warning file=$0::could not find release template"
|
||||||
|
fi
|
||||||
|
body="${body//--REPLACE-WITH-RELEASE-TITLE--/$title}"
|
||||||
|
else
|
||||||
|
body="--REPLACE-WITH-COMMIT-COUNT-- commits have been included over the previous --REPLACE-WITH-PREVIOUS-RELEASE-TYPE--
|
||||||
|
|
||||||
|
<details>
|
||||||
|
<summary><b>show changes</b></summary>
|
||||||
|
|
||||||
|
--REPLACE-WITH-GENERATED-LIST--
|
||||||
|
</details>"
|
||||||
|
fi
|
||||||
|
|
||||||
|
# add git log to release notes
|
||||||
|
all_tags="
|
||||||
|
$(git tag)" # tags are ordered alphabetically
|
||||||
|
before="${all_tags%%
|
||||||
|
"$TAG"*}" # strip line with current tag an all lines after it
|
||||||
|
# note the extra newlines are needed to always have a last line
|
||||||
|
if [[ $all_tags == "$before" ]]; then
|
||||||
|
echo "::warning file=$0::could not find current tag"
|
||||||
|
else
|
||||||
|
while
|
||||||
|
previous="${before##*
|
||||||
|
}" # get the last line
|
||||||
|
# skip this tag if this is a full release and it's a beta or if empty
|
||||||
|
[[ $no_beta && $previous =~ $beta_regex || ! $previous ]]
|
||||||
|
do
|
||||||
|
beta_list+=" $previous" # add to list of skipped betas
|
||||||
|
next_before="${before%
|
||||||
|
*}" # strip the last line
|
||||||
|
if [[ $next_before == "$before" ]]; then
|
||||||
|
unset previous
|
||||||
|
break
|
||||||
|
fi
|
||||||
|
before="$next_before"
|
||||||
|
done
|
||||||
|
if [[ $previous ]]; then
|
||||||
|
if generated_list="$(git log "$previous..$TAG" --pretty="- %s")"; then
|
||||||
|
count="$(git rev-list --count "$previous..$TAG")"
|
||||||
|
[[ $previous =~ $beta_regex ]] && previousreleasetype="beta release" || previousreleasetype="full release"
|
||||||
|
echo "adding list of commits to release notes:"
|
||||||
|
echo "'$previous' to '$TAG' ($count commits)"
|
||||||
|
# --> is the markdown comment escape sequence, emojis are way better
|
||||||
|
generated_list="${generated_list//-->/→}"
|
||||||
|
body="${body//--REPLACE-WITH-GENERATED-LIST--/$generated_list}"
|
||||||
|
body="${body//--REPLACE-WITH-COMMIT-COUNT--/$count}"
|
||||||
|
body="${body//--REPLACE-WITH-PREVIOUS-RELEASE-TAG--/$previous}"
|
||||||
|
body="${body//--REPLACE-WITH-PREVIOUS-RELEASE-TYPE--/$previousreleasetype}"
|
||||||
|
if [[ $beta_list =~ $whitespace ]]; then
|
||||||
|
beta_list="-n there are no betas to delete!"
|
||||||
|
else
|
||||||
|
echo "the following betas should be deleted after publishing:"
|
||||||
|
echo "$beta_list"
|
||||||
|
fi
|
||||||
|
body="${body//--REPLACE-WITH-BETA-LIST--/$beta_list}"
|
||||||
|
else
|
||||||
|
echo "::warning file=$0::failed to produce git log"
|
||||||
|
fi
|
||||||
|
else
|
||||||
|
echo "::warning file=$0::could not find previous tag"
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
# write to file
|
||||||
|
echo "body_path=$body_path" >>"$GITHUB_OUTPUT"
|
||||||
|
echo "$body" >"$body_path"
|
||||||
96
.ci/release_template.md
Normal file
@@ -0,0 +1,96 @@
|
|||||||
|
<!-- this template comes from .ci/release_template.md -->
|
||||||
|
|
||||||
|
<!-- Don't forget to delete the previous betas after publishing this!
|
||||||
|
git push -d origin --REPLACE-WITH-BETA-LIST--
|
||||||
|
-->
|
||||||
|
|
||||||
|
<!-- This list of binaries should be updated every time the CI is changed to
|
||||||
|
include different targets -->
|
||||||
|
<pre>
|
||||||
|
<b>Pre-compiled binaries we serve:</b>
|
||||||
|
- <kbd>Windows 10+</kbd>
|
||||||
|
- <kbd>Windows 7+</kbd>
|
||||||
|
- <kbd>macOS 14+</kbd> ("Sonoma") / Apple M
|
||||||
|
- <kbd>macOS 13+</kbd> ("Ventura") / Intel
|
||||||
|
- <kbd>Ubuntu 24.04 LTS</kbd> ("Noble Numbat")
|
||||||
|
- <kbd>Ubuntu 22.04 LTS</kbd> ("Jammy Jellyfish")
|
||||||
|
- <kbd>Ubuntu 20.04 LTS</kbd> ("Focal Fossa")
|
||||||
|
- <kbd>Debian 12</kbd> ("Bookworm")
|
||||||
|
- <kbd>Debian 11</kbd> ("Bullseye")
|
||||||
|
- <kbd>Fedora 41</kbd>
|
||||||
|
- <kbd>Fedora 40</kbd>
|
||||||
|
|
||||||
|
<i>We are also packaged in <kbd>Arch Linux</kbd>'s official "extra" repository, courtesy of @FFY00</i>
|
||||||
|
<i>General Linux support is available via a <kbd>flatpak</kbd> package (Flathub)</i>
|
||||||
|
</pre>
|
||||||
|
|
||||||
|
|
||||||
|
## General Notes
|
||||||
|
|
||||||
|
We're pleased to announce the newest official release: <kbd>--REPLACE-WITH-RELEASE-TITLE--</kbd>
|
||||||
|
|
||||||
|
We hope you enjoy the changes made! All improvements with their corresponding tickets since the last version of Cockatrice are listed in the changelog below.
|
||||||
|
|
||||||
|
If you ever encounter a bug, have a suggestion or idea, or feel a need for a developer to look into something, please feel free to [open a ticket](https://github.com/Cockatrice/Cockatrice/issues). ([How to create a Ticket for Cockatrice](https://github.com/Cockatrice/Cockatrice/wiki/How-to-Create-a-GitHub-Ticket-Regarding-Cockatrice))
|
||||||
|
|
||||||
|
For basic information related to the app and getting started, please take a look at our official site: **https://cockatrice.github.io**
|
||||||
|
|
||||||
|
If you'd like to help and contribute to Cockatrice in any way, check out our [README](https://github.com/Cockatrice/Cockatrice#get-involved-).
|
||||||
|
We're always available to answer questions you may have on how the program works and how you can provide a meaningful contribution.
|
||||||
|
|
||||||
|
|
||||||
|
## Upgrading Cockatrice
|
||||||
|
<!-- this optional section puts a warning banner for problems with updating
|
||||||
|
> [!IMPORTANT]
|
||||||
|
> **With this release, we no longer provide a ready-to-install binary for:**
|
||||||
|
> --DEPRECATED-OS-HERE--
|
||||||
|
-->
|
||||||
|
|
||||||
|
Run the internal software updater: <kbd>Help → Check for Client Updates</kbd>
|
||||||
|
|
||||||
|
Don't forget to update your card database right after! (<kbd>Help → Check for Card Updates...</kbd>)
|
||||||
|
|
||||||
|
|
||||||
|
## Changelog
|
||||||
|
<!--
|
||||||
|
This list is generated and should be moved to their respective header and
|
||||||
|
possibly edited a little.
|
||||||
|
Append PR numbers of fixups to their main PR to keep the list coherent.
|
||||||
|
Put the quantity of remaining PR's below the highlights section.
|
||||||
|
Remove empty headers when done.
|
||||||
|
|
||||||
|
--REPLACE-WITH-GENERATED-LIST--
|
||||||
|
-->
|
||||||
|
|
||||||
|
<!-- Highlights of the release -->
|
||||||
|
### 🔖 Highlights:
|
||||||
|
### ✨ New Features:
|
||||||
|
### 🐛 Fixed Bugs / Resolved Issues:
|
||||||
|
|
||||||
|
<!-- Complete list of changes (foldable) -->
|
||||||
|
<details>
|
||||||
|
<summary>
|
||||||
|
<b>Show all changes</b> (--REPLACE-WITH-COMMIT-COUNT-- commits)
|
||||||
|
</summary>
|
||||||
|
|
||||||
|
### User Interface
|
||||||
|
### Under the Hood
|
||||||
|
### Oracle
|
||||||
|
### Servatrice
|
||||||
|
### Webatrice
|
||||||
|
|
||||||
|
</details>
|
||||||
|
|
||||||
|
|
||||||
|
## Translations
|
||||||
|
- **Thanks for over 300 people contributing to 20+ different languages up to now!**
|
||||||
|
- Without the help of the community we couldn't offer that great language support... keep up the good work!
|
||||||
|
- It's actually very easy to join and help for yourself - find out more here:
|
||||||
|
- [Help us Translate Cockatrice into your native language!](https://github.com/Cockatrice/Cockatrice/wiki/Translation-FAQ)
|
||||||
|
|
||||||
|
|
||||||
|
## Special Thanks
|
||||||
|
<!-- Personalise this a bit! -->
|
||||||
|
It's amazing that so many people contribute their time, knowledge, code, testing and more to the project.
|
||||||
|
We'd like to thank the entire Cockatrice community for their efforts! 🙏
|
||||||
|
<!-- We'd like to especially recognize @ZeldaZach, --ADD-CONTRIBUTORS-HERE-- for their help in preparing so many amazing new features for the user base. -->
|
||||||
@@ -1,149 +0,0 @@
|
|||||||
#!/bin/bash
|
|
||||||
|
|
||||||
# This script is to be used in .travis.yaml from the project root directory, do not use it from somewhere else.
|
|
||||||
|
|
||||||
# Read arguments
|
|
||||||
while [[ "$@" ]]; do
|
|
||||||
case "$1" in
|
|
||||||
'--format')
|
|
||||||
CHECK_FORMAT=1
|
|
||||||
shift
|
|
||||||
;;
|
|
||||||
'--install')
|
|
||||||
MAKE_INSTALL=1
|
|
||||||
shift
|
|
||||||
;;
|
|
||||||
'--package')
|
|
||||||
MAKE_PACKAGE=1
|
|
||||||
shift
|
|
||||||
if [[ $# != 0 && $1 != -* ]]; then
|
|
||||||
PACKAGE_NAME="$1"
|
|
||||||
shift
|
|
||||||
if [[ $# != 0 && $1 != -* ]]; then
|
|
||||||
PACKAGE_TYPE="$1"
|
|
||||||
shift
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
;;
|
|
||||||
'--server')
|
|
||||||
MAKE_SERVER=1
|
|
||||||
shift
|
|
||||||
;;
|
|
||||||
'--test')
|
|
||||||
MAKE_TEST=1
|
|
||||||
shift
|
|
||||||
;;
|
|
||||||
'--debug')
|
|
||||||
BUILDTYPE="Debug"
|
|
||||||
shift
|
|
||||||
;;
|
|
||||||
'--release')
|
|
||||||
BUILDTYPE="Release"
|
|
||||||
shift
|
|
||||||
;;
|
|
||||||
*)
|
|
||||||
if [[ $1 == -* ]]; then
|
|
||||||
echo "unrecognized option: $1"
|
|
||||||
exit 3
|
|
||||||
fi
|
|
||||||
BUILDTYPE="$1"
|
|
||||||
shift
|
|
||||||
;;
|
|
||||||
esac
|
|
||||||
done
|
|
||||||
|
|
||||||
# Check formatting using clang-format
|
|
||||||
if [[ $CHECK_FORMAT ]]; then
|
|
||||||
echo "Checking your code using clang-format..."
|
|
||||||
diff="$(./clangify.sh --diff --cf-version)"
|
|
||||||
err=$?
|
|
||||||
case $err in
|
|
||||||
1)
|
|
||||||
cat <<EOM
|
|
||||||
***********************************************************
|
|
||||||
*** ***
|
|
||||||
*** Your code does not comply with our styleguide. ***
|
|
||||||
*** ***
|
|
||||||
*** Please correct it or run the "clangify.sh" script. ***
|
|
||||||
*** Then commit and push those changes to this branch. ***
|
|
||||||
*** Check our CONTRIBUTING.md file for more details. ***
|
|
||||||
*** ***
|
|
||||||
*** Thank you ♥ ***
|
|
||||||
*** ***
|
|
||||||
***********************************************************
|
|
||||||
|
|
||||||
Used clang-format version:
|
|
||||||
${diff%%
|
|
||||||
*}
|
|
||||||
|
|
||||||
The following changes should be made:
|
|
||||||
${diff#*
|
|
||||||
}
|
|
||||||
|
|
||||||
Exiting...
|
|
||||||
EOM
|
|
||||||
exit 2
|
|
||||||
;;
|
|
||||||
0)
|
|
||||||
echo "Thank you for complying with our code standards."
|
|
||||||
;;
|
|
||||||
*)
|
|
||||||
echo "Something went wrong in our formatting checks: clangify returned $err" >&2
|
|
||||||
;;
|
|
||||||
esac
|
|
||||||
fi
|
|
||||||
|
|
||||||
set -e
|
|
||||||
|
|
||||||
# Setup
|
|
||||||
./servatrice/check_schema_version.sh
|
|
||||||
mkdir -p build
|
|
||||||
cd build
|
|
||||||
|
|
||||||
# Add cmake flags
|
|
||||||
if [[ $MAKE_SERVER ]]; then
|
|
||||||
flags+=" -DWITH_SERVER=1"
|
|
||||||
fi
|
|
||||||
if [[ $MAKE_TEST ]]; then
|
|
||||||
flags+=" -DTEST=1"
|
|
||||||
BUILDTYPE="Debug" # test requires buildtype Debug
|
|
||||||
fi
|
|
||||||
if [[ $BUILDTYPE ]]; then
|
|
||||||
flags+=" -DCMAKE_BUILD_TYPE=$BUILDTYPE"
|
|
||||||
fi
|
|
||||||
if [[ $PACKAGE_TYPE ]]; then
|
|
||||||
flags+=" -DCPACK_GENERATOR=$PACKAGE_TYPE"
|
|
||||||
fi
|
|
||||||
|
|
||||||
# Add qt install location when using brew
|
|
||||||
if [[ $(uname) == "Darwin" ]]; then
|
|
||||||
PATH="/usr/local/opt/ccache/bin:$PATH"
|
|
||||||
flags+=" -DCMAKE_PREFIX_PATH=/usr/local/opt/qt5/"
|
|
||||||
fi
|
|
||||||
|
|
||||||
# Compile
|
|
||||||
cmake --version
|
|
||||||
cmake .. $flags
|
|
||||||
make -j2
|
|
||||||
|
|
||||||
if [[ $MAKE_TEST ]]; then
|
|
||||||
make test
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [[ $MAKE_INSTALL ]]; then
|
|
||||||
make install
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [[ $MAKE_PACKAGE ]]; then
|
|
||||||
make package
|
|
||||||
if [[ $PACKAGE_NAME ]]; then
|
|
||||||
found=$(find . -maxdepth 1 -type f -name "Cockatrice-*.*" -print -quit)
|
|
||||||
path=${found%/*}
|
|
||||||
file=${found##*/}
|
|
||||||
if [[ ! $file ]]; then
|
|
||||||
echo "could not find package" >&2
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
mv "$path/$file" "$path/${file%.*}-$PACKAGE_NAME.${file##*.}"
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
58
.ci/update_translation_source_strings.sh
Executable file
@@ -0,0 +1,58 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
|
||||||
|
# ci script to update translation files
|
||||||
|
# usage:
|
||||||
|
# $0 cockatrice/cockatrice_en@source.ts cockatrice/src common
|
||||||
|
# or
|
||||||
|
# FILE="cockatrice/cockatrice_en@source.ts"
|
||||||
|
# DIRS="cockatrice/src common"
|
||||||
|
# $0
|
||||||
|
# note: directories can't contain spaces
|
||||||
|
|
||||||
|
# check parameters
|
||||||
|
if [[ ! $FILE ]]; then
|
||||||
|
FILE="$1"
|
||||||
|
shift
|
||||||
|
fi
|
||||||
|
if [[ ! $FILE ]]; then
|
||||||
|
echo "no output file selected" >&2
|
||||||
|
exit 2;
|
||||||
|
fi
|
||||||
|
if [[ ! $DIRS ]]; then
|
||||||
|
DIRS="$*"
|
||||||
|
fi
|
||||||
|
if [[ ! $DIRS ]]; then
|
||||||
|
echo "no source directories selected to translate" >&2
|
||||||
|
exit 2;
|
||||||
|
fi
|
||||||
|
if [[ ! -e $FILE ]]; then
|
||||||
|
echo "output file does not exist at: $FILE" >&2
|
||||||
|
exit 3;
|
||||||
|
fi
|
||||||
|
|
||||||
|
# print version
|
||||||
|
if ! lupdate -version; then
|
||||||
|
echo "failed to run lupdate" >&2
|
||||||
|
exit 4;
|
||||||
|
fi
|
||||||
|
|
||||||
|
# run lupdate, duplicating the output in stderr and saving it
|
||||||
|
# for convenience we ignore that $DIRS will be split on spaces
|
||||||
|
# shellcheck disable=SC2086
|
||||||
|
if ! got="$(lupdate $DIRS -ts "$FILE" | tee /dev/stderr)"; then
|
||||||
|
echo "failed to update $FILE with $DIRS" >&2
|
||||||
|
exit 4;
|
||||||
|
fi
|
||||||
|
|
||||||
|
# trim output
|
||||||
|
# the line we are interested in is:
|
||||||
|
# Found xxx source text(s) (x new and xxx already existing)
|
||||||
|
output="${got##* source text(s) (}" # get stuff in between brackets
|
||||||
|
output="${output%%)*}" # trim everything after first )
|
||||||
|
if [[ $output == "$got" ]]; then
|
||||||
|
echo "could not parse generated output" >&2
|
||||||
|
exit 4;
|
||||||
|
fi
|
||||||
|
|
||||||
|
# write output to ci environment file
|
||||||
|
echo "output=$output" >> "$GITHUB_OUTPUT"
|
||||||
14
.ci/update_translation_source_strings_template.md
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
Updated source strings for translations:
|
||||||
|
- {{ .cockatrice_output }} (Cockatrice)
|
||||||
|
- {{ .oracle_output }} (Oracle)
|
||||||
|
|
||||||
|
<br>
|
||||||
|
|
||||||
|
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>
|
||||||
|
*After merging, all changes to the source language are available for translation at [Transifex][1] shortly.*
|
||||||
|
|
||||||
|
[1]: https://app.transifex.com/cockatrice/cockatrice/
|
||||||
|
[2]: https://github.com/Cockatrice/Cockatrice/actions/workflows/translations-push.yml?query=branch%3Amaster
|
||||||
@@ -22,4 +22,10 @@ AllowShortFunctionsOnASingleLine: None
|
|||||||
BinPackParameters: false
|
BinPackParameters: false
|
||||||
AllowAllParametersOfDeclarationOnNextLine: false
|
AllowAllParametersOfDeclarationOnNextLine: false
|
||||||
IndentCaseLabels: true
|
IndentCaseLabels: true
|
||||||
PointerAlignment: Right
|
PointerAlignment: Right
|
||||||
|
SortIncludes: true
|
||||||
|
IncludeBlocks: Regroup
|
||||||
|
---
|
||||||
|
Language: Proto
|
||||||
|
AllowShortFunctionsOnASingleLine: None
|
||||||
|
SpacesInContainerLiterals: false
|
||||||
|
|||||||
109
.cmake-format.json
Normal file
@@ -0,0 +1,109 @@
|
|||||||
|
{
|
||||||
|
"format": {
|
||||||
|
"_help_line_width": [
|
||||||
|
"How wide to allow formatted cmake files"
|
||||||
|
],
|
||||||
|
"line_width": 120,
|
||||||
|
"_help_tab_size": [
|
||||||
|
"How many spaces to tab for indent"
|
||||||
|
],
|
||||||
|
"tab_size": 2,
|
||||||
|
"_help_max_subgroups_hwrap": [
|
||||||
|
"If an argument group contains more than this many sub-groups",
|
||||||
|
"(parg or kwarg groups) then force it to a vertical layout."
|
||||||
|
],
|
||||||
|
"max_subgroups_hwrap": 2,
|
||||||
|
"_help_max_pargs_hwrap": [
|
||||||
|
"If a positional argument group contains more than this many",
|
||||||
|
"arguments, then force it to a vertical layout."
|
||||||
|
],
|
||||||
|
"max_pargs_hwrap": 6,
|
||||||
|
"_help_max_rows_cmdline": [
|
||||||
|
"If a cmdline positional group consumes more than this many",
|
||||||
|
"lines without nesting, then invalidate the layout (and nest)"
|
||||||
|
],
|
||||||
|
"max_rows_cmdline": 5,
|
||||||
|
"_help_separate_ctrl_name_with_space": [
|
||||||
|
"If true, separate flow control names from their parentheses",
|
||||||
|
"with a space"
|
||||||
|
],
|
||||||
|
"separate_ctrl_name_with_space": false,
|
||||||
|
"_help_separate_fn_name_with_space": [
|
||||||
|
"If true, separate function names from parentheses with a",
|
||||||
|
"space"
|
||||||
|
],
|
||||||
|
"separate_fn_name_with_space": false,
|
||||||
|
"_help_dangle_parens": [
|
||||||
|
"If a statement is wrapped to more than one line, than dangle",
|
||||||
|
"the closing parenthesis on its own line."
|
||||||
|
],
|
||||||
|
"dangle_parens": true,
|
||||||
|
"_help_dangle_align": [
|
||||||
|
"If the trailing parenthesis must be 'dangled' on its on",
|
||||||
|
"line, then align it to this reference: `prefix`: the start",
|
||||||
|
"of the statement, `prefix-indent`: the start of the",
|
||||||
|
"statement, plus one indentation level, `child`: align to",
|
||||||
|
"the column of the arguments"
|
||||||
|
],
|
||||||
|
"dangle_align": "prefix",
|
||||||
|
"_help_min_prefix_chars": [
|
||||||
|
"If the statement spelling length (including space and",
|
||||||
|
"parenthesis) is smaller than this amount, then force reject",
|
||||||
|
"nested layouts."
|
||||||
|
],
|
||||||
|
"min_prefix_chars": 4,
|
||||||
|
"_help_max_prefix_chars": [
|
||||||
|
"If the statement spelling length (including space and",
|
||||||
|
"parenthesis) is larger than the tab width by more than this",
|
||||||
|
"amount, then force reject un-nested layouts."
|
||||||
|
],
|
||||||
|
"max_prefix_chars": 10,
|
||||||
|
"_help_max_lines_hwrap": [
|
||||||
|
"If a candidate layout is wrapped horizontally but it exceeds",
|
||||||
|
"this many lines, then reject the layout."
|
||||||
|
],
|
||||||
|
"max_lines_hwrap": 2,
|
||||||
|
"_help_line_ending": [
|
||||||
|
"What style line endings to use in the output."
|
||||||
|
],
|
||||||
|
"line_ending": "auto",
|
||||||
|
"_help_command_case": [
|
||||||
|
"Format command names consistently as 'lower' or 'upper' case"
|
||||||
|
],
|
||||||
|
"command_case": "lower",
|
||||||
|
"_help_keyword_case": [
|
||||||
|
"Format keywords consistently as 'lower' or 'upper' case"
|
||||||
|
],
|
||||||
|
"keyword_case": "upper",
|
||||||
|
"_help_always_wrap": [
|
||||||
|
"A list of command names which should always be wrapped"
|
||||||
|
],
|
||||||
|
"always_wrap": [],
|
||||||
|
"_help_enable_sort": [
|
||||||
|
"If true, the argument lists which are known to be sortable",
|
||||||
|
"will be sorted lexicographically"
|
||||||
|
],
|
||||||
|
"enable_sort": true,
|
||||||
|
"_help_autosort": [
|
||||||
|
"If true, the parsers may infer whether or not an argument",
|
||||||
|
"list is sortable (without annotation)."
|
||||||
|
],
|
||||||
|
"autosort": true,
|
||||||
|
"_help_require_valid_layout": [
|
||||||
|
"By default, if cmake-format cannot successfully fit",
|
||||||
|
"everything into the desired line-width it will apply the",
|
||||||
|
"last, most aggressive attempt that it made. If this flag is",
|
||||||
|
"True, however, cmake-format will print error, exit with non-",
|
||||||
|
"zero status code, and write-out nothing"
|
||||||
|
],
|
||||||
|
"require_valid_layout": false,
|
||||||
|
"_help_layout_passes": [
|
||||||
|
"A dictionary mapping layout nodes to a list of wrap",
|
||||||
|
"decisions. See the documentation for more information."
|
||||||
|
],
|
||||||
|
"layout_passes": {}
|
||||||
|
},
|
||||||
|
"markup": {
|
||||||
|
"enable_markup": false
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,7 +1,6 @@
|
|||||||
.git/
|
.git/
|
||||||
build/
|
build/
|
||||||
.github/
|
.github/
|
||||||
.travis/
|
|
||||||
.tx/
|
.tx/
|
||||||
cockatrice/
|
cockatrice/
|
||||||
doc/
|
doc/
|
||||||
|
|||||||
330
.github/CONTRIBUTING.md
vendored
@@ -1,4 +1,6 @@
|
|||||||
[Introduction](#contributing-to-cockatrice) | [Code Style Guide](#code-style-guide) | [Translations](#translations) | [Release Management](#release-management)
|
[Introduction](#contributing-to-cockatrice) | [Code Style Guide](
|
||||||
|
#code-style-guide) | [Translations](#translations) | [Release Management](
|
||||||
|
#release-management)
|
||||||
|
|
||||||
----
|
----
|
||||||
|
|
||||||
@@ -7,50 +9,89 @@
|
|||||||
# Contributing to Cockatrice #
|
# Contributing to Cockatrice #
|
||||||
First off, thanks for taking the time to contribute to our project! 🎉 ❤ ️✨
|
First off, thanks for taking the time to contribute to our project! 🎉 ❤ ️✨
|
||||||
|
|
||||||
The following is a set of guidelines for contributing to Cockatrice. These are mostly guidelines, not rules. Use your best judgment, and feel free to propose changes to this document in a pull request.
|
The following is a set of guidelines for contributing to Cockatrice. These are
|
||||||
|
mostly guidelines, not rules. Use your best judgment, and feel free to propose
|
||||||
|
changes to this document in a pull request.
|
||||||
|
|
||||||
|
|
||||||
# Recommended Setups #
|
# Recommended Setups #
|
||||||
|
|
||||||
For those developers who like the Linux or MacOS environment, many of our developers like working with a nifty program called [CLion](https://www.jetbrains.com/clion/). The program's a great asset and one of the best tools you'll find on these systems, but you're welcomed to use any IDE you most enjoy.
|
For those developers who like the Linux or MacOS environment, many of our
|
||||||
|
developers like working with a nifty program called [CLion](
|
||||||
|
https://www.jetbrains.com/clion/). The program's a great asset and one of the
|
||||||
|
best tools you'll find on these systems, but you're welcomed to use any IDE
|
||||||
|
you most enjoy.
|
||||||
|
|
||||||
Developers who like Windows development tend to find [Visual Studio](https://www.visualstudio.com/) the best tool for the job.
|
Developers who like Windows development tend to find [Visual Studio](
|
||||||
|
https://www.visualstudio.com/) the best tool for the job.
|
||||||
|
|
||||||
If you have any questions on IDEs, feel free to chat with us on [Gitter](https://gitter.im/Cockatrice/Cockatrice) and we would love to help answer your questions!
|
[](https://discord.gg/ZASRzKu)
|
||||||
|
[](https://gitter.im/Cockatrice/Cockatrice)
|
||||||
|
|
||||||
|
If you'd like to ask questions, get advice, or just want to say hi,
|
||||||
|
the Cockatrice Development Team uses [Discord](https://discord.gg/ZASRzKu)
|
||||||
|
for communications in the #dev channel. If you're not into Discord, we also
|
||||||
|
have a [Gitter](https://gitter.im/Cockatrice/Cockatrice) channel available,
|
||||||
|
albeit slightly less active.
|
||||||
|
|
||||||
|
|
||||||
# Code Style Guide #
|
# Code Style Guide #
|
||||||
|
|
||||||
### Formatting and continuous integration (ci) ###
|
### Formatting and continuous integration (CI) ###
|
||||||
|
|
||||||
We currently use Travis CI to check your code for formatting issues, if your pull request was rejected because of this it would show a message in the logs. Click on "Details" next to the failed Travis CI build and then click on the failed build (most likely the fastest one) to see the log.
|
We use a separate job on the CI to check your code for formatting issues. If
|
||||||
|
your pull request failed the test, you can check the output on the checks tab.
|
||||||
|
It's the first job called "linter", you can click the "Check code formatting"
|
||||||
|
step to see the output of the test.
|
||||||
|
|
||||||
The message will look somewhat similar to this:
|
The message will look like this:
|
||||||
```
|
```
|
||||||
************************************************************
|
***********************************************************
|
||||||
*** Your code does not meet our formatting guidelines. ***
|
*** ***
|
||||||
*** Please correct it then commit and push your changes. ***
|
*** Your code does not comply with our style guide. ***
|
||||||
*** See our CONTRIBUTING.md file for more information. ***
|
*** ***
|
||||||
************************************************************
|
*** Please correct it or run the "format.sh" script. ***
|
||||||
|
*** Then commit and push those changes to this branch. ***
|
||||||
|
*** Check our CONTRIBUTING.md file for more details. ***
|
||||||
|
*** ***
|
||||||
|
*** Thank you ❤️ ***
|
||||||
|
*** ***
|
||||||
|
***********************************************************
|
||||||
```
|
```
|
||||||
The CONTRIBUTING.md file mentioned is this file. Please read [this section](#Formatting) for full information on our formatting guidelines.
|
The CONTRIBUTING.md file mentioned in that message is the file you are
|
||||||
|
currently reading. Please see [this section](#formatting) below for full
|
||||||
|
information on our formatting guidelines.
|
||||||
|
|
||||||
### Compatibility ###
|
### Compatibility ###
|
||||||
|
|
||||||
Cockatrice is currently compiled on all platforms using <kbd>C++11</kbd>. You'll notice <kbd>C++03</kbd> code throughout the codebase. Please feel free to help convert it over!
|
Cockatrice is currently compiled on all platforms using <kbd>C++11</kbd>.
|
||||||
|
You'll notice <kbd>C++03</kbd> code throughout the codebase. Please feel free
|
||||||
|
to help convert it over!
|
||||||
|
|
||||||
For consistency, we use Qt data structures where possible. For example, `QString` over
|
For consistency, we use Qt data structures where possible. For example,
|
||||||
`std::string` and `QList` over `std::vector`.
|
`QString` over `std::string` and `QList` over `std::vector`.
|
||||||
|
|
||||||
|
Do not use old C style casts in new code, instead use a [`static_cast<>`](
|
||||||
|
https://en.cppreference.com/w/cpp/language/static_cast)
|
||||||
|
or other appropriate conversion.
|
||||||
|
|
||||||
### Formatting ###
|
### Formatting ###
|
||||||
|
|
||||||
The handy tool `clang-format` can format your code for you, it is available for almost any environment. A special `.clang-format` configuration file is included in the project and is used to format your code.
|
The handy tool `clang-format` can format your code for you, it is available for
|
||||||
|
almost any environment. A special `.clang-format` configuration file is
|
||||||
|
included in the project and is used to format your code.
|
||||||
|
|
||||||
We've also included a bash script, `clangify.sh`, that will use clang-format to format all files in one go. Use `./clangify.sh --help` to show a full help page.
|
We've also included a bash script, `format.sh`, that will use clang-format to
|
||||||
|
format all files in your pr in one go. Use `./format.sh --help` to show a full
|
||||||
|
help page.
|
||||||
|
|
||||||
To run clang-format on a single source file simply use the command `clang-format -i <filename>` to format it in place. (some systems install clang-format with a specific version number appended, `find /usr/bin -name clang-format*` should find it for you)
|
To run clang-format on a single source file simply use the command
|
||||||
|
`clang-format -i <filename>` to format it in place. (Some systems install
|
||||||
|
clang-format with a specific version number appended,
|
||||||
|
`find /usr/bin -name clang-format*` should find it for you)
|
||||||
|
|
||||||
See [the clang-format documentation](https://clang.llvm.org/docs/ClangFormat.html) for more information about the tool.
|
See [the clang-format documentation](
|
||||||
|
https://clang.llvm.org/docs/ClangFormat.html) for more information about the tool.
|
||||||
|
|
||||||
#### Header files ####
|
#### Header files ####
|
||||||
|
|
||||||
@@ -62,7 +103,8 @@ Use header guards in the form of `FILE_NAME_H`.
|
|||||||
Simple functions, such as getters, may be written inline in the header file,
|
Simple functions, such as getters, may be written inline in the header file,
|
||||||
but other functions should be written in the source file.
|
but other functions should be written in the source file.
|
||||||
|
|
||||||
Group library includes after project includes, and in alphabetic order. Like this:
|
Group project includes first, followed by library includes. All in alphabetic order.
|
||||||
|
Like this:
|
||||||
```c++
|
```c++
|
||||||
// Good
|
// Good
|
||||||
#include "card.h"
|
#include "card.h"
|
||||||
@@ -88,9 +130,15 @@ Group library includes after project includes, and in alphabetic order. Like thi
|
|||||||
Use `UpperCamelCase` for classes, structs, enums, etc. and `lowerCamelCase` for
|
Use `UpperCamelCase` for classes, structs, enums, etc. and `lowerCamelCase` for
|
||||||
function and variable names.
|
function and variable names.
|
||||||
|
|
||||||
Member variables aren't decorated in any way. Don't prefix or suffix with
|
Don't use [Hungarian Notation](
|
||||||
|
https://en.wikipedia.org/wiki/Hungarian_notation).
|
||||||
|
|
||||||
|
Member variables aren't decorated in any way. Don't prefix or suffix them with
|
||||||
underscores, etc.
|
underscores, etc.
|
||||||
|
|
||||||
|
Use a separate line for each declaration, don't use a single line like this
|
||||||
|
`int one = 1, two = 2` and instead split them into two lines.
|
||||||
|
|
||||||
For arguments to constructors which have the same names as member variables,
|
For arguments to constructors which have the same names as member variables,
|
||||||
prefix those arguments with underscores:
|
prefix those arguments with underscores:
|
||||||
```c++
|
```c++
|
||||||
@@ -115,7 +163,8 @@ If you find any usage of the old keywords, we encourage you to fix it.
|
|||||||
|
|
||||||
#### Braces ####
|
#### Braces ####
|
||||||
|
|
||||||
Braces should go on their own line except for control statements, the use of braces around single line statements is preferred.
|
Braces should go on their own line except for control statements, the use of
|
||||||
|
braces around single line statements is preferred.
|
||||||
See the following example:
|
See the following example:
|
||||||
```c++
|
```c++
|
||||||
int main()
|
int main()
|
||||||
@@ -136,17 +185,25 @@ int main()
|
|||||||
|
|
||||||
#### Indentation and Spacing ####
|
#### Indentation and Spacing ####
|
||||||
|
|
||||||
Always indent using 4 spaces, do not use tabs. Opening and closing braces should be on the same indentation layer, member access specifiers in classes or structs should not be indented.
|
Always indent using 4 spaces, do not use tabs. Opening and closing braces
|
||||||
|
should be on the same indentation layer, member access specifiers in classes or
|
||||||
|
structs should not be indented.
|
||||||
|
|
||||||
All operators and braces should be separated by spaces, do not add a space next to the inside of a brace.
|
All operators and braces should be separated by spaces, do not add a space next
|
||||||
|
to the inside of a brace.
|
||||||
|
|
||||||
If multiple lines of code that follow eachother have single line comments behind them, place all of them on the same indentation level. This indentation level should be equal to the longest line of code for each of these comments, without added spacing.
|
If multiple lines of code that follow eachother have single line comments
|
||||||
|
behind them, place all of them on the same indentation level. This indentation
|
||||||
|
level should be equal to the longest line of code for each of these comments,
|
||||||
|
without added spacing.
|
||||||
|
|
||||||
#### Lines ####
|
#### Lines ####
|
||||||
|
|
||||||
Do not have trailing whitespace in your lines. Most IDEs check for this nowadays and clean it up for you.
|
Do not leave trailing whitespace on any line. Most IDEs check for this
|
||||||
|
nowadays and clean it up for you.
|
||||||
|
|
||||||
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.
|
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.
|
||||||
|
|
||||||
### Memory Management ###
|
### Memory Management ###
|
||||||
|
|
||||||
@@ -182,53 +239,129 @@ as `QScopedPointer`, or, less preferably, `QSharedPointer`.
|
|||||||
The servatrice database's schema can be found at `servatrice/servatrice.sql`.
|
The servatrice database's schema can be found at `servatrice/servatrice.sql`.
|
||||||
Everytime the schema gets modified, some other steps are due:
|
Everytime the schema gets modified, some other steps are due:
|
||||||
1. Increment the value of `cockatrice_schema_version` in `servatrice.sql`;
|
1. Increment the value of `cockatrice_schema_version` in `servatrice.sql`;
|
||||||
2. Increment the value of `DATABASE_SCHEMA_VERSION` in `servatrice_database_interface.h` accordingly;
|
2. Increment the value of `DATABASE_SCHEMA_VERSION` in
|
||||||
3. Create a new migration file inside the `servatrice/migrations` directory named after the new schema version.
|
`servatrice_database_interface.h` accordingly;
|
||||||
4. Run the `servatrice/check_schema_version.sh` script to ensure everything is fine.
|
3. Create a new migration file inside the `servatrice/migrations` directory
|
||||||
|
named after the new schema version.
|
||||||
|
4. Run the `servatrice/check_schema_version.sh` script to ensure everything is
|
||||||
|
fine.
|
||||||
|
|
||||||
The migration file should include the sql statements needed to migrate the database schema and data from the previous to the new version, and an additional statement that updates `cockatrice_schema_version` to the correct value.
|
The migration file should include the sql statements needed to migrate the
|
||||||
|
database schema and data from the previous to the new version, and an
|
||||||
|
additional statement that updates `cockatrice_schema_version` to the correct
|
||||||
|
value.
|
||||||
|
|
||||||
Ensure that the migration produces the expected effects; e.g. if you add a new column, make sure the migration places it in the same order as servatrice.sql.
|
Ensure that the migration produces the expected effects; e.g. if you add a
|
||||||
|
new column, make sure the migration places it in the same order as
|
||||||
|
servatrice.sql.
|
||||||
|
|
||||||
### Protocol buffer ###
|
### Protocol buffer ###
|
||||||
|
|
||||||
Cockatrice and Servatrice exchange data using binary messages. The syntax of these messages is defined in the `proto` files in the `common/pb` folder. These files defines the way data contained in each message is serialized using Google's [protocol buffer](https://developers.google.com/protocol-buffers/).
|
Cockatrice and Servatrice exchange data using binary messages. The syntax of
|
||||||
Any change to the `proto` file should be taken with caution and tested intensively before being merged, becaus a change to the protocol could make new clients incompatible to the old server and vice versa.
|
these messages is defined in the `proto` files in the `common/pb` folder. These
|
||||||
|
files define the way data contained in each message is serialized using
|
||||||
|
Google's [protocol buffers](https://developers.google.com/protocol-buffers/).
|
||||||
|
Any change to the `proto` files should be taken with caution and tested
|
||||||
|
intensively before being merged, because a change to the protocol could make
|
||||||
|
new clients incompatible to the old server and vice versa.
|
||||||
|
|
||||||
You can find more information on how we use Protobuf on [our wiki!](https://github.com/Cockatrice/Cockatrice/wiki/Client-server-protocol)
|
You can find more information on how we use Protobuf on [our wiki!](
|
||||||
|
https://github.com/Cockatrice/Cockatrice/wiki/Client-server-protocol)
|
||||||
|
|
||||||
|
# Reviewing Pull Requests #
|
||||||
|
|
||||||
|
After you have finished your changes to the project you should put them on a
|
||||||
|
separate branch of your fork on GitHub and open a [pull request](
|
||||||
|
https://docs.github.com/en/free-pro-team@latest/desktop/contributing-and-collaborating-using-github-desktop/creating-an-issue-or-pull-request
|
||||||
|
).
|
||||||
|
Your code will then be automatically compiled by GitHub actions for Linux and
|
||||||
|
macOS, and by Appveyor for Windows. Additionally GitHub will perform a [Linting
|
||||||
|
check](#formatting-and-continuous-integration-ci). If any issues come up you
|
||||||
|
can check their status at the bottom of the pull request page, click on details
|
||||||
|
to go to the CI website and see the different build logs.
|
||||||
|
|
||||||
|
If your pull request passes our tests and has no merge conflicts, it will be
|
||||||
|
reviewed by our team members. You can then address any requested changes. When
|
||||||
|
all changes have been approved your pull request will be squashed into a single
|
||||||
|
commit and merged into the master branch by a team member. Your change will then
|
||||||
|
be included in the next release 👍
|
||||||
|
|
||||||
# Translations #
|
# Translations #
|
||||||
|
|
||||||
Basic workflow for translations:
|
Basic workflow for translations:
|
||||||
1. Developer adds a `tr("foo")` string in the code;
|
1. Developer adds a `tr("foo")` string in the code;
|
||||||
2. Every few days, a maintainer updates the `*_en.ts files` with the new strings;
|
2. CI updates the `*_en@source.ts files` regularly and creates a PR automatically;
|
||||||
3. Transifex picks up the new files from github every 24 hours;
|
3. Maintainer verifies and merges the change;
|
||||||
4. Translators translate the new untranslated strings on Transifex;
|
4. Transifex picks up the new files from GitHub automatically;
|
||||||
5. Before a release, a maintainer fetches the updated translations from Transifex.
|
5. Translators translate the new untranslated strings on Transifex;
|
||||||
|
6. Before a release, a maintainer fetches the updated translations from Transifex.
|
||||||
|
|
||||||
### Using Translations (for developers) ###
|
### Using Translations (for developers) ###
|
||||||
|
|
||||||
All the user-interface strings inside Cockatrice's source code must be written in english.
|
All user interface strings inside Cockatrice's source code must be written
|
||||||
Translations to other languages are managed using [Transifex](https://www.transifex.com/projects/p/cockatrice/).
|
in English (US).
|
||||||
|
Translations to other languages are managed using [Transifex](
|
||||||
|
https://www.transifex.com/projects/p/cockatrice/).
|
||||||
|
|
||||||
Adding a new string to translate is as easy as adding the string in the 'tr("")' function, the string will be picked up as translatable automatically and translated as needed.
|
Adding a new string to translate is as easy as adding the string in the
|
||||||
For example setting the text of this label in a way that the string "My name is:" can be translated:
|
`tr("")` function, the string will be picked up as translatable automatically
|
||||||
|
and translated as needed.
|
||||||
|
For example, setting the text of a label in the way that the string
|
||||||
|
`"My name is:"` can be translated:
|
||||||
```c++
|
```c++
|
||||||
nameLabel.setText(tr("My name is:"));
|
nameLabel.setText(tr("My name is:"));
|
||||||
```
|
```
|
||||||
|
|
||||||
If you're about to propose a change that adds or modifies any translatable string in the code, you don't need to take care of adding the new strings to the translation files.
|
To translate a string that would have plural forms you can add the amount to
|
||||||
Every few days, or when a lot of new strings have been added, someone from the development team will take care of extracting all the new strings and adding them to the english translation files and making them available to translators on Transifex.
|
the tr() call, also you can add an extra string as a hint for translators:
|
||||||
|
```c++
|
||||||
|
QString message = tr("Everyone draws %n cards", "pop up message", amount);
|
||||||
|
```
|
||||||
|
See [QT's wiki on translations](
|
||||||
|
https://doc.qt.io/qt-5/i18n-source-translation.html#handling-plurals)
|
||||||
|
|
||||||
|
If you're about to propose a change that adds or modifies any translatable
|
||||||
|
string in the code, you don't need to take care of adding the new strings to
|
||||||
|
the translation files.<br>
|
||||||
|
We have an automated process to update our language source files on a schedule
|
||||||
|
and provide the translators on Transifex with the new contents.<br>
|
||||||
|
Maintainers can also manually trigger this on demand.
|
||||||
|
|
||||||
### Maintaining Translations (for maintainers) ###
|
### Maintaining Translations (for maintainers) ###
|
||||||
|
|
||||||
When new translatable strings have been added to the code, a maintainer should
|
When new translatable strings have been added to the code, a maintainer has to
|
||||||
make them available to translators on Transifex. Every few days, or when a lot
|
make them available to translators on Transifex.
|
||||||
of new strings have been added, a maintainer should take care of extracting all
|
|
||||||
the new strings and add them to the english translation files.
|
|
||||||
|
|
||||||
To update the english translation files, re-run cmake enabling the appropriate
|
To help with that, we have an automated CI workflow, that regularly looks at the
|
||||||
|
code in the master branch, extracts all strings and updates dedicated source string
|
||||||
|
files with any changes. These updates are not commited right away, the CI creates a
|
||||||
|
PR for reviewing instead.<br>
|
||||||
|
After approval, our translation tool automatically picks the changes up and deploys
|
||||||
|
them to our translators. Be mindful when merging only a few changes!
|
||||||
|
|
||||||
|
Once a release is planned, or when a lot of strings have been added or changed, a
|
||||||
|
maintainer can manually trigger a CI run to extract all strings on demand.
|
||||||
|
|
||||||
|
<details>
|
||||||
|
<summary><b>Manually trigger CI run (Workflow Dispatch)</b></summary>
|
||||||
|
|
||||||
|
Maintainers can always request the CI to run on demand if it's required.
|
||||||
|
|
||||||
|
Go to the `Actions` tab and select our dedicated translation workflow:
|
||||||
|
https://github.com/Cockatrice/Cockatrice/actions/workflows/translations.yml
|
||||||
|
|
||||||
|
You see a "This workflow has a workflow_dispatch event trigger." hint at the top of
|
||||||
|
the list.<br>
|
||||||
|
Select `Run workflow` on the right and trigger a run from master branch.
|
||||||
|
|
||||||
|
The CI will now check for changed strings and create a PR if there are any updates.
|
||||||
|
|
||||||
|
</details>
|
||||||
|
|
||||||
|
<details>
|
||||||
|
<summary><b>Manually update source strings locally</b></summary>
|
||||||
|
|
||||||
|
To update the english source files for translation, re-run cmake enabling the appropriate
|
||||||
parameter and then run make:
|
parameter and then run make:
|
||||||
```sh
|
```sh
|
||||||
cd cockatrice/build
|
cd cockatrice/build
|
||||||
@@ -238,23 +371,26 @@ make
|
|||||||
If the parameter has been enabled correctly, when running "make" you should see
|
If the parameter has been enabled correctly, when running "make" you should see
|
||||||
a line similar to this one (the numbers may vary):
|
a line similar to this one (the numbers may vary):
|
||||||
```sh
|
```sh
|
||||||
[ 76%] Generating ../../cockatrice/translations/cockatrice_en.ts
|
[ 76%] Generating ../../cockatrice/translations/cockatrice_en@source.ts
|
||||||
Updating '../../cockatrice/translations/cockatrice_en.ts'...
|
Updating '../../cockatrice/translations/cockatrice_en@source.ts'...
|
||||||
Found 857 source text(s) (8 new and 849 already existing)
|
Found 857 source text(s) (8 new and 849 already existing)
|
||||||
```
|
```
|
||||||
You should then notice that the following files have uncommitted changes:
|
You should then notice that the following files have uncommitted changes:
|
||||||
|
|
||||||
cockatrice/translations/cockatrice_en.ts
|
cockatrice/translations/cockatrice_en@source.ts
|
||||||
oracle/translations/oracle_en.ts
|
oracle/translations/oracle_en@source.ts
|
||||||
|
|
||||||
It is recommended to disable the parameter afterwards using:
|
It is recommended to disable the parameter afterwards using:
|
||||||
```sh
|
```sh
|
||||||
cmake .. -DUPDATE_TRANSLATIONS=OFF
|
cmake .. -DUPDATE_TRANSLATIONS=OFF
|
||||||
```
|
```
|
||||||
Now you are ready to propose your change.
|
Now you are ready to commit your changes and open a PR.
|
||||||
|
|
||||||
Once your change gets merged, Transifex will pick up the modified files automatically (checked every 24 hours)
|
</details>
|
||||||
and update the interface where translators will be able to translate the new strings.
|
|
||||||
|
Once the changes get merged, Transifex will pick up the modified files
|
||||||
|
automatically (checked every few hours) and update their online editor where
|
||||||
|
translators will be able to translate the new strings right in the browser.
|
||||||
|
|
||||||
### Releasing Translations (for maintainers) ###
|
### Releasing Translations (for maintainers) ###
|
||||||
|
|
||||||
@@ -272,51 +408,85 @@ from Transifex to the source code and vice versa.
|
|||||||
|
|
||||||
### Adding Translations (for translators) ###
|
### Adding Translations (for translators) ###
|
||||||
|
|
||||||
As a translator you can help translate the new strings on [Transifex](https://www.transifex.com/projects/p/cockatrice/).
|
As a translator you can help translate the new strings on [Transifex](
|
||||||
Please have a look at the specific [FAQ for translators](https://github.com/Cockatrice/Cockatrice/wiki/Translation-FAQ).
|
https://www.transifex.com/projects/p/cockatrice/).
|
||||||
|
Please have a look at the specific [FAQ for translators](
|
||||||
|
https://github.com/Cockatrice/Cockatrice/wiki/Translation-FAQ).
|
||||||
|
|
||||||
|
|
||||||
# Release Management #
|
# Release Management #
|
||||||
|
|
||||||
### Publishing A New Beta Release ###
|
### Publishing A New Release ###
|
||||||
|
|
||||||
Travis and AppVeyor have been configured to upload files to GitHub Releases whenever a <kbd>tag</kbd> is pushed.<br>
|
We use [GitHub Releases](https://github.com/Cockatrice/Cockatrice/releases) to
|
||||||
Usually, tags are created through publishing a (pre-)release, but there's a way around that.
|
publish new stable versions and betas.
|
||||||
|
Whenever a git tag is pushed to the repository github will create a draft
|
||||||
|
release and upload binaries automatically.
|
||||||
|
|
||||||
To trigger Travis and AppVeyor, simply do the following:
|
To create a tag, simply do the following:
|
||||||
```bash
|
```bash
|
||||||
cd $COCKATRICE_REPO
|
|
||||||
git checkout master
|
git checkout master
|
||||||
git remote update -p
|
git remote update -p
|
||||||
git pull
|
git pull
|
||||||
git tag $TAG_NAME
|
git tag $TAG_NAME
|
||||||
git push upstream $TAG_NAME
|
git push $UPSTREAM $TAG_NAME
|
||||||
```
|
```
|
||||||
|
|
||||||
You should define the variables as such:
|
You should define the variables as such:
|
||||||
```
|
```
|
||||||
upstream - git@github.com:Cockatrice/Cockatrice.git
|
`$UPSTREAM` - the remote for git@github.com:Cockatrice/Cockatrice.git
|
||||||
$COCKATRICE_REPO - /Location/of/repository/cockatrice.git
|
`$TAG_NAME` should be formatted as:
|
||||||
`$TAG_NAME` should be:
|
|
||||||
- `YYYY-MM-DD-Release-MAJ.MIN.PATCH` for **stable releases**
|
- `YYYY-MM-DD-Release-MAJ.MIN.PATCH` for **stable releases**
|
||||||
- `YYYY-MM-DD-Development-MAJ.MIN.PATCH-beta.X` for **beta releases**<br>
|
- `YYYY-MM-DD-Development-MAJ.MIN.PATCH-beta.X` for **beta releases**<br>
|
||||||
With *MAJ.MIN.PATCH* being the NEXT release version!
|
With *MAJ.MIN.PATCH* being the NEXT release version!
|
||||||
```
|
```
|
||||||
|
|
||||||
This will cause a tagged release to be established on the GitHub repository, which will then lead to the upload of binaries. If you use this method, the tags (releases) that you create will be marked as a "Pre-release". The `/latest` URL will not be impacted (for stable release downloads) so that's good.
|
This will cause a tagged release to be established on the GitHub repository,
|
||||||
|
with the binaries being added to the release whenever they are ready.
|
||||||
|
The release is initially a draft, where the path notes can be edited and after
|
||||||
|
all is good and ready it can be published on GitHub.
|
||||||
|
If you use a SemVer tag including "beta", the release that will be created at
|
||||||
|
GitHub will be marked as a "Pre-release" automatically.
|
||||||
|
The target of the `.../latest` URL will not be changed in that case, it always
|
||||||
|
points to the latest stable release and not pre-releases/betas.
|
||||||
|
|
||||||
If you accidentally push a tag incorrectly (the tag is outdated, you didn't pull in the latest branch accidentally, you named the tag wrong, etc.) you can revoke the tag by doing the following:
|
If you accidentally push a tag incorrectly (the tag is outdated, you didn't
|
||||||
|
pull in the latest branch accidentally, you named the tag wrong, etc.) you can
|
||||||
|
revoke the tag by doing the following:
|
||||||
```bash
|
```bash
|
||||||
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 new release.
|
||||||
|
|
||||||
**NOTE:** Unfortunately, due to the method of how Travis and AppVeyor work, to publish a stable release you will need to make a copy of the release notes locally and then paste them into the GitHub GUI once the binaries have been uploaded by them. These CI services will automatically overwrite the name of the release (to "Cockatrice $TAG_NAME"), the status of the release (to "Pre-release"), and the release body (to "Beta build of Cockatrice").
|
In the first lines of [CMakeLists.txt](
|
||||||
|
https://github.com/Cockatrice/Cockatrice/blob/master/CMakeLists.txt)
|
||||||
**NOTE 2:** In the first lines of https://github.com/Cockatrice/Cockatrice/blob/master/CMakeLists.txt there's an hardcoded version number used when compiling custom (not tagged) versions. While on tagged versions these numbers are overridden by the version numbers coming from the tag title, it's good practice to keep them aligned with the real ones.
|
there's an hardcoded version number and pretty name used when compiling from
|
||||||
|
master or custom (not tagged) versions.
|
||||||
|
While on tagged versions these numbers are overridden by the version numbers
|
||||||
|
coming from the tag title, it's good practice to increment the ones at CMake
|
||||||
|
after every full release to stress that master is ahead of the last stable
|
||||||
|
release.
|
||||||
The preferred flow of operation is:
|
The preferred flow of operation is:
|
||||||
* just before a release, update the version number in CMakeLists.txt to "next release version";
|
* Just before a release, make sure the version number in CMakeLists.txt is set
|
||||||
* tag the release following the previously described syntax in order to get it built by CI;
|
to the same release version you are about to tag.
|
||||||
* wait for CI to upload the binaries, double check if everything is in order
|
* This is also the time to change the pretty name in CMakeLists.txt called
|
||||||
* after the release is complete, update the version number again to "next targeted beta version", typically increasing `PROJECT_VERSION_PATCH` by one.
|
GIT_TAG_RELEASENAME and commit and push these changes.
|
||||||
|
* Tag the release following the previously described syntax in order to get it
|
||||||
|
correctly built and deployed by CI.
|
||||||
|
* Wait for the configure step to create the release and update the patch
|
||||||
|
notes.
|
||||||
|
* Check on the github actions page for build progress which should be the top
|
||||||
|
listed [here](
|
||||||
|
https://github.com/Cockatrice/Cockatrice/actions?query=event%3Apush+-branch%3Amaster
|
||||||
|
).
|
||||||
|
* When the build has been completed you can verify all uploaded files on the
|
||||||
|
release are in order and hit the publish button.
|
||||||
|
* After the release is complete, update the CMake version number again to the
|
||||||
|
next targeted beta version, typically increasing `PROJECT_VERSION_PATCH` by
|
||||||
|
one.
|
||||||
|
|
||||||
**NOTE 3:** When releasing a new stable version, all the previous beta versions should be deleted. This is needed for Cockatrice to update users of the "beta" release channel to the latest version like other users.
|
When releasing a new stable version, all previous beta releases (and tags)
|
||||||
|
should be deleted. This is needed for Cockatrice to update users of the "Beta"
|
||||||
|
release channel correctly to the latest stable version as well.
|
||||||
|
This can be done the same way as revoking tags, mentioned above.
|
||||||
|
|||||||
12
.github/FUNDING.yml
vendored
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
# These are supported funding model platforms
|
||||||
|
|
||||||
|
github: [ZeldaZach]
|
||||||
|
patreon: mtgjson
|
||||||
|
open_collective: # Replace with a single Open Collective username
|
||||||
|
ko_fi: # Replace with a single Ko-fi username
|
||||||
|
tidelift: # Replace with a single Tidelift platform-name/package-name e.g., npm/babel
|
||||||
|
community_bridge: # Replace with a single Community Bridge project-name e.g., cloud-foundry
|
||||||
|
liberapay: # Replace with a single Liberapay username
|
||||||
|
issuehunt: # Replace with a single IssueHunt username
|
||||||
|
otechie: # Replace with a single Otechie username
|
||||||
|
custom: ["paypal.me/zachhalpern"]
|
||||||
14
.github/ISSUE_TEMPLATE.md
vendored
@@ -1,14 +0,0 @@
|
|||||||
<b>System Information:</b>
|
|
||||||
<!-- Go to "Help → View Debug Log" and copy all lines above the separation here! -->
|
|
||||||
|
|
||||||
<!-- If you can't install Cockatrice to access that information, make
|
|
||||||
sure to include your OS and the app version from the setup file here -->
|
|
||||||
__________________________________________________________________________________________
|
|
||||||
|
|
||||||
<!-- Explain your issue/request/suggestion in detail here! -->
|
|
||||||
|
|
||||||
<!-- This repository is ONLY about development of the Cockatrice app.
|
|
||||||
If you have any problems with a server (e.g. registering, connecting, ban...)
|
|
||||||
you have to contact that server's owner/admin.
|
|
||||||
Check this list of public servers with webpage links and contact details:
|
|
||||||
https://github.com/Cockatrice/Cockatrice/wiki/Public-Servers -->
|
|
||||||
33
.github/ISSUE_TEMPLATE/bug_report.md
vendored
Normal file
@@ -0,0 +1,33 @@
|
|||||||
|
---
|
||||||
|
name: "🐛 Bug Report"
|
||||||
|
about: Report an issue encountered while using Cockatrice
|
||||||
|
title: ''
|
||||||
|
labels: 'Bug'
|
||||||
|
assignees: ''
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
<!-- READ THIS BEFORE POSTING
|
||||||
|
Go to "Help → View Debug Log" in Cockatrice and copy all information at the
|
||||||
|
top (above the separation line) below "System Information" in this ticket!
|
||||||
|
If you can't start Cockatrice to access these details, make
|
||||||
|
sure to post your OS and the file name of the setup binary instead. -->
|
||||||
|
|
||||||
|
**System Information:**
|
||||||
|
|
||||||
|
|
||||||
|
_______________________________________________________________________________________
|
||||||
|
|
||||||
|
<!-- Explain your issue in detail here! Please attach screenshots if possible. -->
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
_______________________________________________________________________________________
|
||||||
|
|
||||||
|
<!-- Describe the sequence of actions needed to experience the bug -->
|
||||||
|
|
||||||
|
**Steps to reproduce:**
|
||||||
|
- Do A
|
||||||
|
- Do B
|
||||||
|
- Do C ...
|
||||||
9
.github/ISSUE_TEMPLATE/config.yml
vendored
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
blank_issues_enabled: false
|
||||||
|
contact_links:
|
||||||
|
- name: 💬 Discord Community (Get help with server issues, e.g. Login)
|
||||||
|
url: https://discord.gg/3Z9yzmA
|
||||||
|
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)
|
||||||
|
url: https://www.transifex.com/cockatrice/cockatrice/
|
||||||
|
# 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!
|
||||||
23
.github/ISSUE_TEMPLATE/feature_request.md
vendored
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
---
|
||||||
|
name: "💡 Feature Request"
|
||||||
|
about: Request a new feature
|
||||||
|
title: ''
|
||||||
|
labels: 'Feature Request'
|
||||||
|
assignees: ''
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
<!--
|
||||||
|
Please search the issue tracker for similar issues before posting!
|
||||||
|
If your request is related to another request (but not the same!) list it here
|
||||||
|
-->
|
||||||
|
**Similar Requests**
|
||||||
|
|
||||||
|
|
||||||
|
<!-- Describe your feature idea here in detail -->
|
||||||
|
**Description of New Feature**
|
||||||
|
|
||||||
|
|
||||||
|
<!-- If your feature requires some context, provide it here -->
|
||||||
|
**Additional Context**
|
||||||
|
|
||||||
49
.github/dependabot.yml
vendored
Normal file
@@ -0,0 +1,49 @@
|
|||||||
|
# Configuration options: https://docs.github.com/en/github/administering-a-repository/configuration-options-for-dependency-updates
|
||||||
|
|
||||||
|
version: 2
|
||||||
|
updates:
|
||||||
|
# # Enable version updates for git submodules
|
||||||
|
# Not yet possible to bump only on tags or releases, see:
|
||||||
|
# https://github.com/dependabot/dependabot-core/issues/1639
|
||||||
|
# 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"
|
||||||
|
# # Look for `.gitmodules` in the `root` directory
|
||||||
|
# directory: "/"
|
||||||
|
# # Check for updates once a month
|
||||||
|
# schedule:
|
||||||
|
# interval: "monthly"
|
||||||
|
# # Limit the amout of open PR's (default = 5, disabled = 0, security updates are not impacted)
|
||||||
|
# open-pull-requests-limit: 1
|
||||||
|
|
||||||
|
# # Enable version updates for Docker
|
||||||
|
# Not yet possible to bump from one LTS version to the next and skip others, see:
|
||||||
|
# https://github.com/dependabot/dependabot-core/issues/2247
|
||||||
|
# - package-ecosystem: "docker"
|
||||||
|
# # Look for a `Dockerfile` in the `root` directory
|
||||||
|
# directory: "/"
|
||||||
|
# # Check 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: 1
|
||||||
|
|
||||||
|
# Enable version updates for GitHub Actions
|
||||||
|
- package-ecosystem: "github-actions"
|
||||||
|
# Directory must be set to "/" to check for workflow files in .github/workflows
|
||||||
|
directory: "/"
|
||||||
|
# Check for updates to GitHub Actions 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: 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
|
||||||
358
.github/workflows/desktop-build.yml
vendored
Normal file
@@ -0,0 +1,358 @@
|
|||||||
|
name: Build Desktop
|
||||||
|
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
branches:
|
||||||
|
- master
|
||||||
|
paths-ignore:
|
||||||
|
- '**.md'
|
||||||
|
- 'webclient/**'
|
||||||
|
- '.github/workflows/web-*.yml'
|
||||||
|
- '.github/workflows/translations-*.yml'
|
||||||
|
tags:
|
||||||
|
- '*'
|
||||||
|
pull_request:
|
||||||
|
paths-ignore:
|
||||||
|
- '**.md'
|
||||||
|
- 'webclient/**'
|
||||||
|
- '.github/workflows/web-*.yml'
|
||||||
|
- '.github/workflows/translations-*.yml'
|
||||||
|
|
||||||
|
# Cancel earlier, unfinished runs of this workflow on the same branch (unless on master)
|
||||||
|
concurrency:
|
||||||
|
group: "${{ github.workflow }} @ ${{ github.ref_name }}"
|
||||||
|
cancel-in-progress: ${{ github.ref_name != 'master' }}
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
configure:
|
||||||
|
name: Configure
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
outputs:
|
||||||
|
tag: ${{steps.configure.outputs.tag}}
|
||||||
|
sha: ${{steps.configure.outputs.sha}}
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: Configure
|
||||||
|
id: configure
|
||||||
|
shell: bash
|
||||||
|
run: |
|
||||||
|
tag_regex='^refs/tags/'
|
||||||
|
if [[ $GITHUB_EVENT_NAME == pull-request ]]; then # pull request
|
||||||
|
sha="${{github.event.pull_request.head.sha}}"
|
||||||
|
elif [[ $GITHUB_REF =~ $tag_regex ]]; then # release
|
||||||
|
sha="$GITHUB_SHA"
|
||||||
|
tag="${GITHUB_REF/refs\/tags\//}"
|
||||||
|
echo "tag=$tag" >>"$GITHUB_OUTPUT"
|
||||||
|
else # push to branch
|
||||||
|
sha="$GITHUB_SHA"
|
||||||
|
fi
|
||||||
|
echo "sha=$sha" >>"$GITHUB_OUTPUT"
|
||||||
|
|
||||||
|
- name: Checkout
|
||||||
|
if: steps.configure.outputs.tag != null
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
with:
|
||||||
|
fetch-depth: 0
|
||||||
|
|
||||||
|
- name: Prepare release parameters
|
||||||
|
id: prepare
|
||||||
|
if: steps.configure.outputs.tag != null
|
||||||
|
shell: bash
|
||||||
|
env:
|
||||||
|
TAG: ${{steps.configure.outputs.tag}}
|
||||||
|
run: .ci/prep_release.sh
|
||||||
|
|
||||||
|
- name: Create release
|
||||||
|
if: steps.configure.outputs.tag != null
|
||||||
|
id: create_release
|
||||||
|
shell: bash
|
||||||
|
env:
|
||||||
|
GH_TOKEN: ${{github.token}}
|
||||||
|
tag_name: ${{steps.configure.outputs.tag}}
|
||||||
|
target: ${{steps.configure.outputs.sha}}
|
||||||
|
release_name: ${{steps.prepare.outputs.title}}
|
||||||
|
body_path: ${{steps.prepare.outputs.body_path}}
|
||||||
|
prerelease: ${{steps.prepare.outputs.is_beta}}
|
||||||
|
run: |
|
||||||
|
if [[ $prerelease == yes ]]; then
|
||||||
|
args="--prerelease"
|
||||||
|
fi
|
||||||
|
gh release create "$tag_name" --draft --verify-tag $args \
|
||||||
|
--target "$target" --title "$release_name" \
|
||||||
|
--notes-file "$body_path"
|
||||||
|
|
||||||
|
build-linux:
|
||||||
|
strategy:
|
||||||
|
fail-fast: false
|
||||||
|
matrix:
|
||||||
|
# These names correspond to the files in ".ci/$distro$version"
|
||||||
|
include:
|
||||||
|
- distro: Arch
|
||||||
|
package: skip # We are packaged in Arch already
|
||||||
|
allow-failure: yes
|
||||||
|
|
||||||
|
- distro: Debian
|
||||||
|
version: 11
|
||||||
|
package: DEB
|
||||||
|
test: skip # Running tests on all distros is superfluous
|
||||||
|
|
||||||
|
- distro: Debian
|
||||||
|
version: 12
|
||||||
|
package: DEB
|
||||||
|
|
||||||
|
- distro: Fedora
|
||||||
|
version: 40
|
||||||
|
package: RPM
|
||||||
|
test: skip # Running tests on all distros is superfluous
|
||||||
|
|
||||||
|
- distro: Fedora
|
||||||
|
version: 41
|
||||||
|
package: RPM
|
||||||
|
|
||||||
|
- distro: Ubuntu
|
||||||
|
version: 20.04
|
||||||
|
package: DEB
|
||||||
|
test: skip # Ubuntu 20.04 has a broken Qt for debug builds
|
||||||
|
|
||||||
|
- distro: Ubuntu
|
||||||
|
version: 22.04
|
||||||
|
package: DEB
|
||||||
|
test: skip # Running tests on all distros is superfluous
|
||||||
|
|
||||||
|
- distro: Ubuntu
|
||||||
|
version: 24.04
|
||||||
|
package: DEB
|
||||||
|
|
||||||
|
name: ${{matrix.distro}} ${{matrix.version}}
|
||||||
|
needs: configure
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
continue-on-error: ${{matrix.allow-failure == 'yes'}}
|
||||||
|
env:
|
||||||
|
NAME: ${{matrix.distro}}${{matrix.version}}
|
||||||
|
CACHE: /tmp/${{matrix.distro}}${{matrix.version}}-cache # ${{runner.temp}} does not work?
|
||||||
|
# Cache size over the entire repo is 10Gi:
|
||||||
|
# https://docs.github.com/en/actions/using-workflows/caching-dependencies-to-speed-up-workflows#usage-limits-and-eviction-policy
|
||||||
|
CCACHE_SIZE: 200M
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: Checkout
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
|
||||||
|
- name: Generate cache timestamp
|
||||||
|
id: cache_timestamp
|
||||||
|
shell: bash
|
||||||
|
run: echo "timestamp=$(date -u '+%Y%m%d%H%M%S')" >>"$GITHUB_OUTPUT"
|
||||||
|
|
||||||
|
- name: Restore cache
|
||||||
|
uses: actions/cache@v4
|
||||||
|
env:
|
||||||
|
timestamp: ${{steps.cache_timestamp.outputs.timestamp}}
|
||||||
|
with:
|
||||||
|
path: ${{env.CACHE}}
|
||||||
|
key: docker-${{matrix.distro}}${{matrix.version}}-cache-${{env.timestamp}}
|
||||||
|
restore-keys: |
|
||||||
|
docker-${{matrix.distro}}${{matrix.version}}-cache-
|
||||||
|
|
||||||
|
- 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'
|
||||||
|
shell: bash
|
||||||
|
run: |
|
||||||
|
source .ci/docker.sh
|
||||||
|
RUN --server --debug --test --ccache "$CCACHE_SIZE" --parallel 4
|
||||||
|
|
||||||
|
- name: Build release package
|
||||||
|
id: build
|
||||||
|
if: matrix.package != 'skip'
|
||||||
|
shell: bash
|
||||||
|
env:
|
||||||
|
BUILD_DIR: build
|
||||||
|
SUFFIX: '-${{matrix.distro}}${{matrix.version}}'
|
||||||
|
type: '${{matrix.package}}'
|
||||||
|
run: |
|
||||||
|
source .ci/docker.sh
|
||||||
|
RUN --server --release --package "$type" --dir "$BUILD_DIR" \
|
||||||
|
--ccache "$CCACHE_SIZE" --parallel 4
|
||||||
|
.ci/name_build.sh
|
||||||
|
|
||||||
|
- name: Upload artifact
|
||||||
|
if: matrix.package != 'skip'
|
||||||
|
uses: actions/upload-artifact@v4
|
||||||
|
with:
|
||||||
|
name: ${{matrix.distro}}${{matrix.version}}-package
|
||||||
|
path: ${{steps.build.outputs.path}}
|
||||||
|
if-no-files-found: error
|
||||||
|
|
||||||
|
- name: Upload to release
|
||||||
|
if: matrix.package != 'skip' && needs.configure.outputs.tag != null
|
||||||
|
shell: bash
|
||||||
|
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-macos:
|
||||||
|
strategy:
|
||||||
|
fail-fast: false
|
||||||
|
matrix:
|
||||||
|
include:
|
||||||
|
- target: 13
|
||||||
|
soc: Intel
|
||||||
|
os: macos-13
|
||||||
|
xcode: "14.3.1"
|
||||||
|
type: Release
|
||||||
|
core_count: 4
|
||||||
|
make_package: 1
|
||||||
|
|
||||||
|
- target: 14
|
||||||
|
soc: Apple
|
||||||
|
os: macos-14
|
||||||
|
xcode: "15.4"
|
||||||
|
type: Release
|
||||||
|
core_count: 3
|
||||||
|
make_package: 1
|
||||||
|
|
||||||
|
- target: 14
|
||||||
|
soc: Apple
|
||||||
|
os: macos-14
|
||||||
|
xcode: "15.4"
|
||||||
|
type: Debug
|
||||||
|
core_count: 3
|
||||||
|
|
||||||
|
name: macOS ${{matrix.target}}${{ matrix.soc == 'Intel' && ' Intel' || '' }}${{ matrix.type == 'Debug' && ' Debug' || '' }}
|
||||||
|
needs: configure
|
||||||
|
runs-on: ${{matrix.os}}
|
||||||
|
continue-on-error: ${{matrix.allow-failure == 'yes'}}
|
||||||
|
env:
|
||||||
|
DEVELOPER_DIR:
|
||||||
|
/Applications/Xcode_${{matrix.xcode}}.app/Contents/Developer
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: Checkout
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
|
||||||
|
- name: Install dependencies using Homebrew
|
||||||
|
shell: bash
|
||||||
|
# CMake cannot find the MySQL connector
|
||||||
|
# Neither of these works: mariadb-connector-c mysql-connector-c++
|
||||||
|
env:
|
||||||
|
HOMEBREW_NO_INSTALLED_DEPENDENTS_CHECK: 1
|
||||||
|
run: |
|
||||||
|
brew update
|
||||||
|
brew install protobuf qt --force-bottle
|
||||||
|
|
||||||
|
- name: Build on Xcode ${{matrix.xcode}}
|
||||||
|
shell: bash
|
||||||
|
id: build
|
||||||
|
env:
|
||||||
|
BUILDTYPE: '${{matrix.type}}'
|
||||||
|
MAKE_TEST: 1
|
||||||
|
MAKE_PACKAGE: '${{matrix.make_package}}'
|
||||||
|
PACKAGE_SUFFIX: '-macOS${{matrix.target}}_${{matrix.soc}}'
|
||||||
|
# macOS runner have 3 cores usually - only the macos-13 image has 4:
|
||||||
|
# https://docs.github.com/en/actions/using-github-hosted-runners/using-github-hosted-runners/about-github-hosted-runners#standard-github-hosted-runners-for-public-repositories
|
||||||
|
# https://github.com/actions/runner-images?tab=readme-ov-file#available-images
|
||||||
|
run: .ci/compile.sh --server --parallel ${{matrix.core_count}}
|
||||||
|
|
||||||
|
- name: Upload artifact
|
||||||
|
if: matrix.make_package
|
||||||
|
uses: actions/upload-artifact@v4
|
||||||
|
with:
|
||||||
|
name: macOS${{matrix.target}}${{ matrix.soc == 'Intel' && '_Intel' || '' }}${{ matrix.type == 'Debug' && '_Debug' || '' }}-dmg
|
||||||
|
path: ${{steps.build.outputs.path}}
|
||||||
|
if-no-files-found: error
|
||||||
|
|
||||||
|
- name: Upload to release
|
||||||
|
if: matrix.package != 'skip' && needs.configure.outputs.tag != null
|
||||||
|
shell: bash
|
||||||
|
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:
|
||||||
|
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: false
|
||||||
|
version: ${{matrix.qt_version}}
|
||||||
|
arch: win64_${{matrix.qt_arch}}
|
||||||
|
tools: ${{matrix.qt_tools}}
|
||||||
|
modules: ${{matrix.qt_modules}}
|
||||||
|
|
||||||
|
- name: Run vcpkg
|
||||||
|
uses: lukka/run-vcpkg@v11
|
||||||
|
with:
|
||||||
|
runVcpkgInstall: true
|
||||||
|
doNotCache: false
|
||||||
|
env:
|
||||||
|
VCPKG_DEFAULT_TRIPLET: 'x64-windows'
|
||||||
|
VCPKG_DISABLE_METRICS: 1
|
||||||
|
|
||||||
|
- 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}}'
|
||||||
|
# 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
|
||||||
|
|
||||||
|
- name: Upload to release
|
||||||
|
if: matrix.package != 'skip' && needs.configure.outputs.tag != null
|
||||||
|
shell: bash
|
||||||
|
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"
|
||||||
29
.github/workflows/desktop-lint.yml
vendored
Normal file
@@ -0,0 +1,29 @@
|
|||||||
|
name: Code Style (C++)
|
||||||
|
|
||||||
|
on:
|
||||||
|
pull_request:
|
||||||
|
paths-ignore:
|
||||||
|
- '**.md'
|
||||||
|
- 'webclient/**'
|
||||||
|
- '.github/workflows/web-*.yml'
|
||||||
|
- '.github/workflows/translations-*.yml'
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
format:
|
||||||
|
runs-on: ubuntu-22.04
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: Checkout
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
with:
|
||||||
|
fetch-depth: 20 # should be enough to find merge base
|
||||||
|
|
||||||
|
- name: Install dependencies
|
||||||
|
shell: bash
|
||||||
|
run: |
|
||||||
|
sudo apt-get update
|
||||||
|
sudo apt-get install -y --no-install-recommends clang-format cmake-format
|
||||||
|
|
||||||
|
- name: Check code formatting
|
||||||
|
shell: bash
|
||||||
|
run: ./.ci/lint_cpp.sh
|
||||||
72
.github/workflows/translations-pull.yml
vendored
Normal file
@@ -0,0 +1,72 @@
|
|||||||
|
name: Update Translations
|
||||||
|
|
||||||
|
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:
|
||||||
|
paths:
|
||||||
|
- '.github/workflows/translations-pull.yml'
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
translations:
|
||||||
|
# Do not run the scheduled workflow on forks
|
||||||
|
if: github.event_name != 'schedule' || github.repository_owner == 'Cockatrice'
|
||||||
|
|
||||||
|
name: Pull languages
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: Checkout repo
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
|
||||||
|
- name: Pull translated strings from Transifex
|
||||||
|
uses: transifex/cli-action@v2
|
||||||
|
with:
|
||||||
|
# used config file: https://github.com/Cockatrice/Cockatrice/blob/master/.tx/config
|
||||||
|
# https://github.com/transifex/cli#pulling-files-from-transifex
|
||||||
|
token: ${{ secrets.TX_TOKEN }}
|
||||||
|
args: pull --force --all
|
||||||
|
|
||||||
|
- name: Create pull request
|
||||||
|
if: github.event_name != 'pull_request'
|
||||||
|
id: create_pr
|
||||||
|
uses: peter-evans/create-pull-request@v7
|
||||||
|
with:
|
||||||
|
add-paths: |
|
||||||
|
cockatrice/translations/*.ts
|
||||||
|
oracle/translations/*.ts
|
||||||
|
webclient/public/locales/*/translation.json
|
||||||
|
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: |
|
||||||
|
Pulled all translated strings from [Transifex][1].
|
||||||
|
|
||||||
|
---
|
||||||
|
*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.*
|
||||||
|
|
||||||
|
[1]: https://app.transifex.com/cockatrice/cockatrice/
|
||||||
|
[2]: https://github.com/Cockatrice/Cockatrice/actions/workflows/translations-pull.yml?query=branch%3Amaster
|
||||||
|
labels: |
|
||||||
|
CI
|
||||||
|
Translation
|
||||||
|
draft: false
|
||||||
|
|
||||||
|
- name: PR Status
|
||||||
|
if: github.event_name != 'pull_request'
|
||||||
|
shell: bash
|
||||||
|
env:
|
||||||
|
STATUS: ${{ steps.create_pr.outputs.pull-request-operation }}
|
||||||
|
run: |
|
||||||
|
if [[ "$STATUS" == "none" ]]; then
|
||||||
|
echo "PR #${{ steps.create_pr.outputs.pull-request-number }} unchanged!" >> $GITHUB_STEP_SUMMARY
|
||||||
|
else
|
||||||
|
echo "PR #${{ steps.create_pr.outputs.pull-request-number }} $STATUS!" >> $GITHUB_STEP_SUMMARY
|
||||||
|
fi
|
||||||
|
echo "URL: ${{ steps.create_pr.outputs.pull-request-url }}" >> $GITHUB_STEP_SUMMARY
|
||||||
87
.github/workflows/translations-push.yml
vendored
Normal file
@@ -0,0 +1,87 @@
|
|||||||
|
name: Update Translation Source
|
||||||
|
|
||||||
|
on:
|
||||||
|
workflow_dispatch:
|
||||||
|
schedule:
|
||||||
|
# runs at the start of each quarter (UTC)
|
||||||
|
- cron: '0 0 1 1,4,7,10 *'
|
||||||
|
pull_request:
|
||||||
|
paths:
|
||||||
|
- '.github/workflows/translations-push.yml'
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
translations:
|
||||||
|
# Do not run the scheduled workflow on forks
|
||||||
|
if: github.event_name != 'schedule' || github.repository_owner == 'Cockatrice'
|
||||||
|
|
||||||
|
name: Push strings
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: Checkout repo
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
|
||||||
|
- name: Install lupdate
|
||||||
|
shell: bash
|
||||||
|
run: |
|
||||||
|
sudo apt-get update
|
||||||
|
sudo apt-get install -y --no-install-recommends qttools5-dev-tools
|
||||||
|
|
||||||
|
- name: Update Cockatrice translation source
|
||||||
|
id: cockatrice
|
||||||
|
shell: bash
|
||||||
|
env:
|
||||||
|
FILE: 'cockatrice/cockatrice_en@source.ts'
|
||||||
|
DIRS: 'cockatrice/src common'
|
||||||
|
run: .ci/update_translation_source_strings.sh
|
||||||
|
|
||||||
|
- name: Update Oracle translation source
|
||||||
|
id: oracle
|
||||||
|
shell: bash
|
||||||
|
env:
|
||||||
|
FILE: 'oracle/oracle_en@source.ts'
|
||||||
|
DIRS: 'oracle/src'
|
||||||
|
run: .ci/update_translation_source_strings.sh
|
||||||
|
|
||||||
|
- name: Render template
|
||||||
|
id: template
|
||||||
|
uses: chuhlomin/render-template@v1
|
||||||
|
with:
|
||||||
|
template: .ci/update_translation_source_strings_template.md
|
||||||
|
vars: |
|
||||||
|
cockatrice_output: ${{ steps.cockatrice.outputs.output }}
|
||||||
|
oracle_output: ${{ steps.oracle.outputs.output }}
|
||||||
|
commit: ${{ github.sha }}
|
||||||
|
|
||||||
|
- name: Create pull request
|
||||||
|
if: github.event_name != 'pull_request'
|
||||||
|
id: create_pr
|
||||||
|
uses: peter-evans/create-pull-request@v7
|
||||||
|
with:
|
||||||
|
add-paths: |
|
||||||
|
cockatrice/cockatrice_en@source.ts
|
||||||
|
oracle/oracle_en@source.ts
|
||||||
|
commit-message: Update translation source strings
|
||||||
|
# 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 }}
|
||||||
|
labels: |
|
||||||
|
CI
|
||||||
|
Translation
|
||||||
|
draft: false
|
||||||
|
|
||||||
|
- name: PR Status
|
||||||
|
if: github.event_name != 'pull_request'
|
||||||
|
shell: bash
|
||||||
|
env:
|
||||||
|
STATUS: ${{ steps.create_pr.outputs.pull-request-operation }}
|
||||||
|
run: |
|
||||||
|
if [[ "$STATUS" == "none" ]]; then
|
||||||
|
echo "PR #${{ steps.create_pr.outputs.pull-request-number }} unchanged!" >> $GITHUB_STEP_SUMMARY
|
||||||
|
else
|
||||||
|
echo "PR #${{ steps.create_pr.outputs.pull-request-number }} $STATUS!" >> $GITHUB_STEP_SUMMARY
|
||||||
|
fi
|
||||||
|
echo "URL: ${{ steps.create_pr.outputs.pull-request-url }}" >> $GITHUB_STEP_SUMMARY
|
||||||
52
.github/workflows/web-build.yml
vendored
Normal file
@@ -0,0 +1,52 @@
|
|||||||
|
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
Normal file
@@ -0,0 +1,32 @@
|
|||||||
|
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
|
||||||
7
.gitignore
vendored
@@ -6,5 +6,10 @@ mysql.cnf
|
|||||||
.DS_Store
|
.DS_Store
|
||||||
.idea/
|
.idea/
|
||||||
*.aps
|
*.aps
|
||||||
cmake-build-debug/
|
cmake-build-debug*
|
||||||
preferences
|
preferences
|
||||||
|
compile_commands.json
|
||||||
|
.vs/
|
||||||
|
.vscode/
|
||||||
|
.cache
|
||||||
|
.gdb_history
|
||||||
|
|||||||
139
.gitlab-ci.yml
@@ -1,139 +0,0 @@
|
|||||||
---
|
|
||||||
stages:
|
|
||||||
- build
|
|
||||||
|
|
||||||
.artifacts: &artifacts
|
|
||||||
artifacts:
|
|
||||||
paths:
|
|
||||||
- build/
|
|
||||||
|
|
||||||
.cache: &cache
|
|
||||||
cache:
|
|
||||||
key: "$CI_BUILD_NAME"
|
|
||||||
paths:
|
|
||||||
- cache/
|
|
||||||
|
|
||||||
.branches: &branches
|
|
||||||
only:
|
|
||||||
- master
|
|
||||||
|
|
||||||
.tags: &tags
|
|
||||||
tags:
|
|
||||||
- linux
|
|
||||||
- docker
|
|
||||||
|
|
||||||
|
|
||||||
#================================ DEBIAN-BASED ================================
|
|
||||||
|
|
||||||
.build_rc_package_deb: &build_rc_package_deb
|
|
||||||
stage: build
|
|
||||||
script:
|
|
||||||
- mkdir -p build
|
|
||||||
- cd build
|
|
||||||
- cmake .. -DWITH_SERVER=1 -DCMAKE_BUILD_TYPE=Release -DCPACK_GENERATOR=DEB
|
|
||||||
- make package -j2
|
|
||||||
|
|
||||||
.build_debug_package_deb: &build_debug_package_deb
|
|
||||||
stage: build
|
|
||||||
script:
|
|
||||||
- mkdir -p build
|
|
||||||
- cd build
|
|
||||||
- cmake .. -DWITH_SERVER=1 -DCMAKE_BUILD_TYPE=Debug -DCPACK_GENERATOR=DEB
|
|
||||||
- make package -j2
|
|
||||||
|
|
||||||
.deb-artifacts: &artifacts_deb
|
|
||||||
artifacts:
|
|
||||||
paths:
|
|
||||||
- build/*.deb
|
|
||||||
- build/CMakeFiles/*.log
|
|
||||||
when: always
|
|
||||||
|
|
||||||
#----------------------------------- UBUNTU -----------------------------------
|
|
||||||
|
|
||||||
.requirements_16xx: &install_requirements_16xx
|
|
||||||
before_script:
|
|
||||||
- apt-get -o dir::cache::archives="cache" update -qq
|
|
||||||
- apt-get -o dir::cache::archives="cache" install -y build-essential g++ cmake git
|
|
||||||
- apt-get -o dir::cache::archives="cache" install -y libprotobuf-dev protobuf-compiler
|
|
||||||
- apt-get -o dir::cache::archives="cache" install -y qt5-default qttools5-dev qttools5-dev-tools
|
|
||||||
- apt-get -o dir::cache::archives="cache" install -y qtmultimedia5-dev libqt5multimedia5-plugins
|
|
||||||
- apt-get -o dir::cache::archives="cache" install -y libqt5svg5-dev libqt5sql5-mysql
|
|
||||||
- apt-get -o dir::cache::archives="cache" install -y libqt5websockets5-dev
|
|
||||||
|
|
||||||
.requirements_17xx: &install_requirements_17xx
|
|
||||||
before_script:
|
|
||||||
- apt-get -o dir::cache::archives="cache" update -qq
|
|
||||||
- apt-get -o dir::cache::archives="cache" install -y build-essential g++ cmake git
|
|
||||||
- apt-get -o dir::cache::archives="cache" install -y libprotobuf-dev protobuf-compiler
|
|
||||||
- apt-get -o dir::cache::archives="cache" install -y qt5-default qttools5-dev qttools5-dev-tools
|
|
||||||
- apt-get -o dir::cache::archives="cache" install -y qtmultimedia5-dev libqt5multimedia5-plugins
|
|
||||||
- apt-get -o dir::cache::archives="cache" install -y libqt5svg5-dev libqt5sql5-mysql
|
|
||||||
- apt-get -o dir::cache::archives="cache" install -y libqt5websockets5-dev
|
|
||||||
|
|
||||||
.build_1604: &1604
|
|
||||||
image: ubuntu:16.04
|
|
||||||
<<: *tags
|
|
||||||
<<: *branches
|
|
||||||
<<: *install_requirements_16xx
|
|
||||||
<<: *artifacts_deb
|
|
||||||
<<: *cache
|
|
||||||
|
|
||||||
.build_1710: &1710
|
|
||||||
image: ubuntu:17.10
|
|
||||||
<<: *tags
|
|
||||||
<<: *branches
|
|
||||||
<<: *install_requirements_17xx
|
|
||||||
<<: *artifacts_deb
|
|
||||||
<<: *cache
|
|
||||||
|
|
||||||
build_rc_1604:
|
|
||||||
<<: *1604
|
|
||||||
<<: *build_rc_package_deb
|
|
||||||
when: always
|
|
||||||
|
|
||||||
build_debug_1604:
|
|
||||||
<<: *1604
|
|
||||||
<<: *build_debug_package_deb
|
|
||||||
when: always
|
|
||||||
|
|
||||||
build_rc_1710:
|
|
||||||
<<: *1710
|
|
||||||
<<: *build_rc_package_deb
|
|
||||||
when: always
|
|
||||||
|
|
||||||
build_debug_1710:
|
|
||||||
<<: *1710
|
|
||||||
<<: *build_debug_package_deb
|
|
||||||
when: always
|
|
||||||
allow_failure: true
|
|
||||||
|
|
||||||
#----------------------------------- DEBIAN -----------------------------------
|
|
||||||
|
|
||||||
.requirements_stretch: &install_requirements_stretch
|
|
||||||
before_script:
|
|
||||||
- apt-get -o dir::cache::archives="cache" update -qq
|
|
||||||
- apt-get -o dir::cache::archives="cache" install -y build-essential g++ cmake git
|
|
||||||
- apt-get -o dir::cache::archives="cache" install -y qt5-default qtbase5-dev-tools
|
|
||||||
- apt-get -o dir::cache::archives="cache" install -y qttools5-dev-tools qtmultimedia5-dev
|
|
||||||
- apt-get -o dir::cache::archives="cache" install -y libqt5svg5-dev libqt5websockets5-dev
|
|
||||||
- apt-get -o dir::cache::archives="cache" install -y libprotobuf-dev protobuf-compiler
|
|
||||||
|
|
||||||
.build_stretch: &stretch
|
|
||||||
image: debian:stretch
|
|
||||||
<<: *tags
|
|
||||||
<<: *branches
|
|
||||||
<<: *install_requirements_stretch
|
|
||||||
<<: *artifacts_deb
|
|
||||||
<<: *cache
|
|
||||||
|
|
||||||
build_rc_stretch:
|
|
||||||
<<: *stretch
|
|
||||||
<<: *build_rc_package_deb
|
|
||||||
when: always
|
|
||||||
|
|
||||||
build_debug_stretch:
|
|
||||||
<<: *stretch
|
|
||||||
<<: *build_debug_package_deb
|
|
||||||
when: always
|
|
||||||
allow_failure: true
|
|
||||||
|
|
||||||
3
.gitmodules
vendored
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
[submodule "vcpkg"]
|
||||||
|
path = vcpkg
|
||||||
|
url = https://github.com/microsoft/vcpkg.git
|
||||||
7
.husky/pre-commit
Executable file
@@ -0,0 +1,7 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
. "$(dirname "$0")/_/husky.sh"
|
||||||
|
|
||||||
|
cd webclient
|
||||||
|
npm run translate
|
||||||
|
|
||||||
|
git add src/i18n-default.json
|
||||||
163
.travis.yml
@@ -1,163 +0,0 @@
|
|||||||
language: cpp
|
|
||||||
compiler: gcc
|
|
||||||
|
|
||||||
matrix:
|
|
||||||
include:
|
|
||||||
#Ubuntu Xenial (Debug only)
|
|
||||||
- name: Ubuntu Xenial (Debug)
|
|
||||||
if: tag IS NOT present
|
|
||||||
os: linux
|
|
||||||
dist: xenial
|
|
||||||
group: stable
|
|
||||||
cache: ccache
|
|
||||||
addons:
|
|
||||||
apt:
|
|
||||||
packages:
|
|
||||||
- libprotobuf-dev
|
|
||||||
- protobuf-compiler
|
|
||||||
- liblzma-dev
|
|
||||||
- qt5-default
|
|
||||||
- qttools5-dev
|
|
||||||
- qttools5-dev-tools
|
|
||||||
- qtmultimedia5-dev
|
|
||||||
- libqt5multimedia5-plugins
|
|
||||||
- libqt5svg5-dev
|
|
||||||
- libqt5sql5-mysql
|
|
||||||
- libqt5websockets5-dev
|
|
||||||
script: bash ./.ci/travis-compile.sh --format --server --test --debug
|
|
||||||
|
|
||||||
#Ubuntu Bionic (on docker)
|
|
||||||
- name: Ubuntu Bionic (Debug)
|
|
||||||
if: tag IS NOT present
|
|
||||||
services: docker
|
|
||||||
env: NAME=UbuntuBionic
|
|
||||||
cache:
|
|
||||||
directories:
|
|
||||||
- $HOME/$NAME/
|
|
||||||
before_install: docker build -t "cockatrice_${NAME,,}" .ci/$NAME && mkdir -p $HOME/$NAME/.ccache
|
|
||||||
script: docker run --mount "type=bind,source=$(pwd),target=/src" -w="/src"
|
|
||||||
--mount "type=bind,source=$HOME/$NAME/.ccache,target=/.ccache" -e "CCACHE_DIR=/.ccache"
|
|
||||||
"cockatrice_${NAME,,}"
|
|
||||||
bash .ci/travis-compile.sh --server --debug
|
|
||||||
|
|
||||||
- name: Ubuntu Bionic (Release)
|
|
||||||
if: (branch = master AND NOT type = pull_request) OR tag IS present
|
|
||||||
services: docker
|
|
||||||
env: NAME=UbuntuBionic
|
|
||||||
cache:
|
|
||||||
directories:
|
|
||||||
- $HOME/$NAME/
|
|
||||||
before_install: docker build -t "cockatrice_${NAME,,}" .ci/$NAME && mkdir -p $HOME/$NAME/.ccache
|
|
||||||
script: docker run --mount "type=bind,source=$(pwd),target=/src" -w="/src"
|
|
||||||
--mount "type=bind,source=$HOME/$NAME/.ccache,target=/.ccache" -e "CCACHE_DIR=/.ccache"
|
|
||||||
"cockatrice_${NAME,,}"
|
|
||||||
bash .ci/travis-compile.sh --server --package "$NAME" --release
|
|
||||||
|
|
||||||
#Fedora 29 (on docker)
|
|
||||||
- name: Fedora 29 (Debug)
|
|
||||||
if: tag IS NOT present
|
|
||||||
services: docker
|
|
||||||
env: NAME=Fedora29
|
|
||||||
cache:
|
|
||||||
directories:
|
|
||||||
- $HOME/$NAME/
|
|
||||||
before_install: docker build -t "cockatrice_${NAME,,}" .ci/$NAME && mkdir -p $HOME/$NAME/.ccache
|
|
||||||
script: docker run --mount "type=bind,source=$(pwd),target=/src" -w="/src"
|
|
||||||
--mount "type=bind,source=$HOME/$NAME/.ccache,target=/.ccache" -e "CCACHE_DIR=/.ccache"
|
|
||||||
"cockatrice_${NAME,,}"
|
|
||||||
bash .ci/travis-compile.sh --server --debug
|
|
||||||
|
|
||||||
- name: Fedora 29 (Release)
|
|
||||||
if: (branch = master AND NOT type = pull_request) OR tag IS present
|
|
||||||
services: docker
|
|
||||||
env: NAME=Fedora29
|
|
||||||
cache:
|
|
||||||
directories:
|
|
||||||
- $HOME/$NAME/
|
|
||||||
before_install: docker build -t "cockatrice_${NAME,,}" .ci/$NAME && mkdir -p $HOME/$NAME/.ccache
|
|
||||||
script: docker run --mount "type=bind,source=$(pwd),target=/src" -w="/src"
|
|
||||||
--mount "type=bind,source=$HOME/$NAME/.ccache,target=/.ccache" -e "CCACHE_DIR=/.ccache"
|
|
||||||
"cockatrice_${NAME,,}"
|
|
||||||
bash .ci/travis-compile.sh --server --package "$NAME" "RPM" --release
|
|
||||||
|
|
||||||
#macOS
|
|
||||||
- name: macOS (Debug)
|
|
||||||
if: tag IS NOT present
|
|
||||||
os: osx
|
|
||||||
osx_image: xcode10.1
|
|
||||||
cache: ccache
|
|
||||||
addons:
|
|
||||||
homebrew:
|
|
||||||
packages:
|
|
||||||
- ccache
|
|
||||||
- protobuf
|
|
||||||
- qt
|
|
||||||
- xz
|
|
||||||
script: bash ./.ci/travis-compile.sh --server --install --debug
|
|
||||||
|
|
||||||
- name: macOS (Release)
|
|
||||||
if: (branch = master AND NOT type = pull_request) OR tag IS present
|
|
||||||
os: osx
|
|
||||||
osx_image: xcode9.2
|
|
||||||
cache: ccache
|
|
||||||
addons:
|
|
||||||
homebrew:
|
|
||||||
packages:
|
|
||||||
- ccache
|
|
||||||
- protobuf
|
|
||||||
- qt
|
|
||||||
- xz
|
|
||||||
update: true
|
|
||||||
script: bash ./.ci/travis-compile.sh --server --package "$TRAVIS_OS_NAME" --release
|
|
||||||
|
|
||||||
# Builds for pull requests skip the deployment step altogether
|
|
||||||
deploy:
|
|
||||||
# Deploy configuration for "beta" releases
|
|
||||||
- provider: releases
|
|
||||||
api_key:
|
|
||||||
secure: mLMF41q7xgOR1sjczsilEy7HQis2PkZCzhfOGbn/8FoOQnmmPOZjrsdhn06ZSl3SFsbfCLuClDYXAbFscQmdgjcGN5AmHV+JYfW650QEuQa/f4/lQFsVRtEqUA1O3FQ0OuRxdpCfJubZBdFVH8SbZ93GLC5zXJbkWQNq+xCX1fU=
|
|
||||||
skip_cleanup: true
|
|
||||||
name: "Cockatrice $TRAVIS_TAG"
|
|
||||||
body: "Beta release of Cockatrice"
|
|
||||||
file_glob: true
|
|
||||||
file: "build/Cockatrice-*"
|
|
||||||
overwrite: true
|
|
||||||
draft: false
|
|
||||||
prerelease: true
|
|
||||||
on:
|
|
||||||
tags: true
|
|
||||||
repo: Cockatrice/Cockatrice
|
|
||||||
condition: $TRAVIS_TAG =~ ([0-9]|[1-9][0-9])(\.([0-9]|[1-9][0-9])){2}-beta(\.([2-9]|[1-9][0-9]))?$ # regex to match semver naming convention for beta pre-releases
|
|
||||||
|
|
||||||
# Deploy configuration for "stable" releases
|
|
||||||
- provider: releases
|
|
||||||
api_key:
|
|
||||||
secure: mLMF41q7xgOR1sjczsilEy7HQis2PkZCzhfOGbn/8FoOQnmmPOZjrsdhn06ZSl3SFsbfCLuClDYXAbFscQmdgjcGN5AmHV+JYfW650QEuQa/f4/lQFsVRtEqUA1O3FQ0OuRxdpCfJubZBdFVH8SbZ93GLC5zXJbkWQNq+xCX1fU=
|
|
||||||
skip_cleanup: true
|
|
||||||
file_glob: true
|
|
||||||
file: "build/Cockatrice-*"
|
|
||||||
overwrite: true
|
|
||||||
draft: false
|
|
||||||
prerelease: false
|
|
||||||
on:
|
|
||||||
tags: true
|
|
||||||
repo: Cockatrice/Cockatrice
|
|
||||||
condition: $TRAVIS_TAG =~ ([0-9]|[1-9][0-9])(\.([0-9]|[1-9][0-9])){2}$ # regex to match semver naming convention for stable full releases
|
|
||||||
|
|
||||||
|
|
||||||
notifications:
|
|
||||||
email: false
|
|
||||||
webhooks:
|
|
||||||
urls:
|
|
||||||
- https://webhooks.gitter.im/e/d94969c3b01b22cbdcb7
|
|
||||||
on_success: change
|
|
||||||
on_failure: change
|
|
||||||
on_start: never
|
|
||||||
on_cancel: change
|
|
||||||
on_error: change
|
|
||||||
|
|
||||||
|
|
||||||
# Announcements of build image updates: https://docs.travis-ci.com/user/build-environment-updates/
|
|
||||||
# For precise versions of preinstalled tools on the VM, check “Build system information” in the build log!
|
|
||||||
# Official validator for ".travis.yml" config file: https://yaml.travis-ci.org
|
|
||||||
# Travis CI config documentation: https://docs.travis-ci.com/user/customizing-the-build
|
|
||||||
27
.tx/config
@@ -1,13 +1,26 @@
|
|||||||
[main]
|
[main]
|
||||||
host = https://www.transifex.com
|
host = https://app.transifex.com
|
||||||
|
|
||||||
[cockatrice.cockatrice]
|
[o:cockatrice:p:cockatrice:r:cockatrice-cockatrice-en-source-ts--master]
|
||||||
|
resource_name = Cockatrice
|
||||||
|
source_lang = en
|
||||||
|
source_file = cockatrice/cockatrice_en@source.ts
|
||||||
file_filter = cockatrice/translations/cockatrice_<lang>.ts
|
file_filter = cockatrice/translations/cockatrice_<lang>.ts
|
||||||
source_file = cockatrice/translations/cockatrice_en.ts
|
type = QT
|
||||||
source_lang = en
|
minimum_perc = 10
|
||||||
|
|
||||||
[cockatrice.oracle]
|
[o:cockatrice:p:cockatrice:r:oracle-oracle-en-source-ts--master]
|
||||||
|
resource_name = Oracle
|
||||||
|
source_lang = en
|
||||||
|
source_file = oracle/oracle_en@source.ts
|
||||||
file_filter = oracle/translations/oracle_<lang>.ts
|
file_filter = oracle/translations/oracle_<lang>.ts
|
||||||
source_file = oracle/translations/oracle_en.ts
|
type = QT
|
||||||
source_lang = en
|
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
|
||||||
|
|||||||
438
CMakeLists.txt
@@ -5,48 +5,96 @@
|
|||||||
# This file sets all the variables shared between the projects
|
# This file sets all the variables shared between the projects
|
||||||
# like the installation path, compilation flags etc..
|
# like the installation path, compilation flags etc..
|
||||||
|
|
||||||
# Cmake 3.1 is required to enable C++11 support correctly
|
# cmake 3.16 is required if using qt6
|
||||||
cmake_minimum_required(VERSION 3.1)
|
cmake_minimum_required(VERSION 3.10)
|
||||||
|
|
||||||
if(POLICY CMP0064)
|
# Early detect ccache
|
||||||
cmake_policy(SET CMP0064 NEW)
|
option(USE_CCACHE "Cache the build results with ccache" ON)
|
||||||
endif()
|
# Treat warnings as errors (Debug builds only)
|
||||||
|
option(WARNING_AS_ERROR "Treat warnings as errors in debug builds" ON)
|
||||||
if(POLICY CMP0071)
|
# Check for translation updates
|
||||||
cmake_policy(SET CMP0071 NEW)
|
option(UPDATE_TRANSLATIONS "Update translations on compile" OFF)
|
||||||
endif()
|
# Compile servatrice
|
||||||
|
option(WITH_SERVER "build servatrice" OFF)
|
||||||
|
# Compile cockatrice
|
||||||
|
option(WITH_CLIENT "build cockatrice" ON)
|
||||||
|
# Compile oracle
|
||||||
|
option(WITH_ORACLE "build oracle" ON)
|
||||||
|
# Compile dbconverter
|
||||||
|
option(WITH_DBCONVERTER "build dbconverter" ON)
|
||||||
|
# Compile tests
|
||||||
|
option(TEST "build tests" 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
|
||||||
IF(DEFINED CMAKE_BUILD_TYPE)
|
if(DEFINED CMAKE_BUILD_TYPE)
|
||||||
SET(CMAKE_BUILD_TYPE ${CMAKE_BUILD_TYPE} CACHE STRING "Type of build")
|
set(CMAKE_BUILD_TYPE
|
||||||
ELSE()
|
${CMAKE_BUILD_TYPE}
|
||||||
SET(CMAKE_BUILD_TYPE Release CACHE STRING "Type of build")
|
CACHE STRING "Type of build"
|
||||||
ENDIF()
|
)
|
||||||
|
else()
|
||||||
|
set(CMAKE_BUILD_TYPE
|
||||||
|
Release
|
||||||
|
CACHE STRING "Type of build"
|
||||||
|
)
|
||||||
|
endif()
|
||||||
|
|
||||||
# Early detect ccache
|
if(USE_CCACHE)
|
||||||
find_program(CCACHE_PROGRAM ccache)
|
find_program(CCACHE_PROGRAM ccache)
|
||||||
if(CCACHE_PROGRAM)
|
if(CCACHE_PROGRAM)
|
||||||
# Support Unix Makefiles and Ninja
|
# Support Unix Makefiles and Ninja
|
||||||
set_property(GLOBAL PROPERTY RULE_LAUNCH_COMPILE "${CCACHE_PROGRAM}")
|
set_property(GLOBAL PROPERTY RULE_LAUNCH_COMPILE "${CCACHE_PROGRAM}")
|
||||||
MESSAGE(STATUS "Found CCache ${CCACHE_PROGRAM}")
|
message(STATUS "Found CCache ${CCACHE_PROGRAM}")
|
||||||
|
endif()
|
||||||
|
endif()
|
||||||
|
|
||||||
|
if(WIN32)
|
||||||
|
# Use vcpkg toolchain on Windows
|
||||||
|
set(CMAKE_TOOLCHAIN_FILE
|
||||||
|
${CMAKE_CURRENT_SOURCE_DIR}/vcpkg/scripts/buildsystems/vcpkg.cmake
|
||||||
|
CACHE STRING "Vcpkg toolchain file"
|
||||||
|
)
|
||||||
|
# Qt path set by user or env var
|
||||||
|
if(QTDIR
|
||||||
|
OR DEFINED ENV{QTDIR}
|
||||||
|
OR DEFINED ENV{QTDIR32}
|
||||||
|
OR DEFINED ENV{QTDIR64}
|
||||||
|
)
|
||||||
|
|
||||||
|
else()
|
||||||
|
set(QTDIR
|
||||||
|
""
|
||||||
|
CACHE PATH "Path to Qt (e.g. C:/Qt/5.7/msvc2015_64)"
|
||||||
|
)
|
||||||
|
message(
|
||||||
|
WARNING "QTDIR variable is missing. Please set this variable to specify path to Qt (e.g. C:/Qt/5.7/msvc2015_64)"
|
||||||
|
)
|
||||||
|
endif()
|
||||||
endif()
|
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.6.3)
|
project("Cockatrice" VERSION 2.10.0)
|
||||||
|
|
||||||
# Use c++11 for all targets
|
# Set release name if not provided via env/cmake var
|
||||||
set(CMAKE_CXX_STANDARD 11 CACHE STRING "C++ ISO Standard")
|
if(NOT DEFINED GIT_TAG_RELEASENAME)
|
||||||
|
set(GIT_TAG_RELEASENAME "Rings of the Wild")
|
||||||
|
endif()
|
||||||
|
|
||||||
|
# Use c++20 for all targets
|
||||||
|
set(CMAKE_CXX_STANDARD
|
||||||
|
20
|
||||||
|
CACHE STRING "C++ ISO Standard"
|
||||||
|
)
|
||||||
set(CMAKE_CXX_STANDARD_REQUIRED True)
|
set(CMAKE_CXX_STANDARD_REQUIRED True)
|
||||||
|
|
||||||
# Set conventional loops
|
# Set conventional loops
|
||||||
set(CMAKE_ALLOW_LOOSE_LOOP_CONSTRUCTS true)
|
set(CMAKE_ALLOW_LOOSE_LOOP_CONSTRUCTS true)
|
||||||
|
|
||||||
# Search path for cmake modules
|
# Search path for cmake modules
|
||||||
SET(CMAKE_MODULE_PATH ${PROJECT_SOURCE_DIR}/cmake)
|
set(COCKATRICE_CMAKE_PATH "${PROJECT_SOURCE_DIR}/cmake")
|
||||||
|
list(INSERT CMAKE_MODULE_PATH 0 "${COCKATRICE_CMAKE_PATH}")
|
||||||
|
|
||||||
# Retrieve git version hash
|
|
||||||
include(getversion)
|
include(getversion)
|
||||||
|
|
||||||
# Create a header and a cpp file containing the version hash
|
# Create a header and a cpp file containing the version hash
|
||||||
@@ -54,126 +102,152 @@ include(createversionfile)
|
|||||||
|
|
||||||
# Define a proper install path
|
# Define a proper install path
|
||||||
if(UNIX)
|
if(UNIX)
|
||||||
if(APPLE)
|
if(APPLE)
|
||||||
# macOS
|
# macOS
|
||||||
# Due to the special bundle structure ignore
|
# Due to the special bundle structure ignore
|
||||||
# the prefix eventually set by the user.
|
# the prefix eventually set by the user.
|
||||||
set(CMAKE_INSTALL_PREFIX ${CMAKE_BINARY_DIR}/release)
|
|
||||||
|
|
||||||
# Force ccache usage if available
|
|
||||||
get_property(RULE_LAUNCH_COMPILE GLOBAL PROPERTY RULE_LAUNCH_COMPILE)
|
|
||||||
if(RULE_LAUNCH_COMPILE)
|
|
||||||
MESSAGE(STATUS "Force enabling CCache usage under macOS")
|
|
||||||
# Set up wrapper scripts
|
|
||||||
configure_file(${CMAKE_MODULE_PATH}/launch-c.in launch-c)
|
|
||||||
configure_file(${CMAKE_MODULE_PATH}/launch-cxx.in launch-cxx)
|
|
||||||
execute_process(COMMAND chmod a+rx
|
|
||||||
"${CMAKE_BINARY_DIR}/launch-c"
|
|
||||||
"${CMAKE_BINARY_DIR}/launch-cxx")
|
|
||||||
|
|
||||||
# Set Xcode project attributes to route compilation through our scripts
|
|
||||||
set(CMAKE_XCODE_ATTRIBUTE_CC "${CMAKE_BINARY_DIR}/launch-c")
|
|
||||||
set(CMAKE_XCODE_ATTRIBUTE_CXX "${CMAKE_BINARY_DIR}/launch-cxx")
|
|
||||||
set(CMAKE_XCODE_ATTRIBUTE_LD "${CMAKE_BINARY_DIR}/launch-c")
|
|
||||||
set(CMAKE_XCODE_ATTRIBUTE_LDPLUSPLUS "${CMAKE_BINARY_DIR}/launch-cxx")
|
|
||||||
endif()
|
|
||||||
else()
|
|
||||||
# Linux / BSD
|
|
||||||
if(CMAKE_INSTALL_PREFIX_INITIALIZED_TO_DEFAULT)
|
|
||||||
#fix package build
|
|
||||||
if(PREFIX)
|
|
||||||
set(CMAKE_INSTALL_PREFIX ${PREFIX})
|
|
||||||
else()
|
|
||||||
set(CMAKE_INSTALL_PREFIX /usr/local)
|
|
||||||
endif()
|
|
||||||
endif()
|
|
||||||
endif()
|
|
||||||
elseif(WIN32)
|
|
||||||
set(CMAKE_INSTALL_PREFIX ${CMAKE_BINARY_DIR}/release)
|
set(CMAKE_INSTALL_PREFIX ${CMAKE_BINARY_DIR}/release)
|
||||||
|
|
||||||
|
# Force ccache usage if available
|
||||||
|
get_property(RULE_LAUNCH_COMPILE GLOBAL PROPERTY RULE_LAUNCH_COMPILE)
|
||||||
|
if(RULE_LAUNCH_COMPILE)
|
||||||
|
message(STATUS "Force enabling CCache usage under macOS")
|
||||||
|
# Set up wrapper scripts
|
||||||
|
configure_file("${COCKATRICE_CMAKE_PATH}/launch-c.in" launch-c)
|
||||||
|
configure_file("${COCKATRICE_CMAKE_PATH}/launch-cxx.in" launch-cxx)
|
||||||
|
execute_process(COMMAND chmod a+rx "${CMAKE_BINARY_DIR}/launch-c" "${CMAKE_BINARY_DIR}/launch-cxx")
|
||||||
|
|
||||||
|
# Set Xcode project attributes to route compilation through our scripts
|
||||||
|
set(CMAKE_XCODE_ATTRIBUTE_CC "${CMAKE_BINARY_DIR}/launch-c")
|
||||||
|
set(CMAKE_XCODE_ATTRIBUTE_CXX "${CMAKE_BINARY_DIR}/launch-cxx")
|
||||||
|
set(CMAKE_XCODE_ATTRIBUTE_LD "${CMAKE_BINARY_DIR}/launch-c")
|
||||||
|
set(CMAKE_XCODE_ATTRIBUTE_LDPLUSPLUS "${CMAKE_BINARY_DIR}/launch-cxx")
|
||||||
|
endif()
|
||||||
|
else()
|
||||||
|
# Linux / BSD
|
||||||
|
if(CMAKE_INSTALL_PREFIX_INITIALIZED_TO_DEFAULT)
|
||||||
|
#fix package build
|
||||||
|
if(PREFIX)
|
||||||
|
set(CMAKE_INSTALL_PREFIX ${PREFIX})
|
||||||
|
else()
|
||||||
|
set(CMAKE_INSTALL_PREFIX /usr/local)
|
||||||
|
endif()
|
||||||
|
endif()
|
||||||
|
endif()
|
||||||
|
elseif(WIN32)
|
||||||
|
set(CMAKE_INSTALL_PREFIX ${CMAKE_BINARY_DIR}/rundir/${CMAKE_BUILD_TYPE})
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
# Treat warnings as errors (Debug builds only)
|
|
||||||
option(WARNING_AS_ERROR "Treat warnings as errors in debug builds" ON)
|
|
||||||
|
|
||||||
# Define proper compilation flags
|
# Define proper compilation flags
|
||||||
IF(MSVC)
|
if(MSVC)
|
||||||
# Visual Studio:
|
# Visual Studio: Disable Warning C4251, C++20 compatibility, Multi-threaded Builds, Warn Detection, Unwind Semantics
|
||||||
# Maximum optimization
|
set(CMAKE_CXX_FLAGS "/wd4251 /Zc:__cplusplus /std:c++20 /permissive- /W4 /MP /EHsc")
|
||||||
# Disable warning C4251
|
# Visual Studio: Maximum Optimization, Multi-threaded DLL
|
||||||
set(CMAKE_CXX_FLAGS_RELEASE "/Ox /MD /wd4251")
|
set(CMAKE_CXX_FLAGS_RELEASE "/Ox /MD")
|
||||||
# Generate complete debugging information
|
# Visual Studio: No Optimization, Multi-threaded Debug DLL, Debug Symbols
|
||||||
#set(CMAKE_CXX_FLAGS_DEBUG "/Zi")
|
set(CMAKE_CXX_FLAGS_DEBUG "/Od /MDd /Zi")
|
||||||
ELSEIF (CMAKE_COMPILER_IS_GNUCXX)
|
elseif(CMAKE_COMPILER_IS_GNUCXX)
|
||||||
# linux/gcc, bsd/gcc, windows/mingw
|
# linux/gcc, bsd/gcc, windows/mingw
|
||||||
include(CheckCXXCompilerFlag)
|
include(CheckCXXCompilerFlag)
|
||||||
|
|
||||||
set(CMAKE_CXX_FLAGS_RELEASE "-s -O2")
|
set(CMAKE_CXX_FLAGS_RELEASE "-s -O2")
|
||||||
if(WARNING_AS_ERROR)
|
if(WARNING_AS_ERROR)
|
||||||
set(CMAKE_CXX_FLAGS_DEBUG "-ggdb -O0 -Wall -Wextra -Werror")
|
set(CMAKE_CXX_FLAGS_DEBUG "-ggdb -O0 -Wall -Wextra -Werror")
|
||||||
else()
|
else()
|
||||||
set(CMAKE_CXX_FLAGS_DEBUG "-ggdb -O0 -Wall -Wextra")
|
set(CMAKE_CXX_FLAGS_DEBUG "-ggdb -O0 -Wall -Wextra")
|
||||||
|
endif()
|
||||||
|
|
||||||
|
if(APPLE)
|
||||||
|
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -std=gnu++20")
|
||||||
|
endif()
|
||||||
|
|
||||||
|
set(ADDITIONAL_DEBUG_FLAGS
|
||||||
|
-Wcast-align
|
||||||
|
-Wmissing-declarations
|
||||||
|
-Wno-long-long
|
||||||
|
-Wno-error=extra
|
||||||
|
-Wno-error=delete-non-virtual-dtor
|
||||||
|
-Wno-error=sign-compare
|
||||||
|
-Wno-error=missing-declarations
|
||||||
|
)
|
||||||
|
|
||||||
|
foreach(FLAG ${ADDITIONAL_DEBUG_FLAGS})
|
||||||
|
check_cxx_compiler_flag("${FLAG}" CXX_HAS_WARNING_${FLAG})
|
||||||
|
if(CXX_HAS_WARNING_${FLAG})
|
||||||
|
set(CMAKE_CXX_FLAGS_DEBUG "${CMAKE_CXX_FLAGS_DEBUG} ${FLAG}")
|
||||||
endif()
|
endif()
|
||||||
|
endforeach()
|
||||||
set(ADDITIONAL_DEBUG_FLAGS -Wcast-align -Wmissing-declarations -Wno-long-long -Wno-error=extra -Wno-error=delete-non-virtual-dtor -Wno-error=sign-compare -Wno-error=missing-declarations)
|
else()
|
||||||
|
# other: osx/llvm, bsd/llvm
|
||||||
FOREACH(FLAG ${ADDITIONAL_DEBUG_FLAGS})
|
set(CMAKE_CXX_FLAGS_RELEASE "-O2")
|
||||||
CHECK_CXX_COMPILER_FLAG("${FLAG}" CXX_HAS_WARNING_${FLAG})
|
if(WARNING_AS_ERROR)
|
||||||
IF(CXX_HAS_WARNING_${FLAG})
|
set(CMAKE_CXX_FLAGS_DEBUG "-g -O0 -Wall -Wextra -Werror -Wno-unused-parameter")
|
||||||
SET(CMAKE_CXX_FLAGS_DEBUG "${CMAKE_CXX_FLAGS_DEBUG} ${FLAG}")
|
else()
|
||||||
ENDIF()
|
set(CMAKE_CXX_FLAGS_DEBUG "-g -O0 -Wall -Wextra")
|
||||||
ENDFOREACH()
|
endif()
|
||||||
ELSE()
|
endif()
|
||||||
# other: osx/llvm, bsd/llvm
|
|
||||||
set(CMAKE_CXX_FLAGS_RELEASE "-O2")
|
|
||||||
set(CMAKE_CXX_FLAGS_DEBUG "-g -O0")
|
|
||||||
ENDIF()
|
|
||||||
|
|
||||||
# GNU systems need to define the Mersenne exponent for the RNG to compile w/o warning
|
# GNU systems need to define the Mersenne exponent for the RNG to compile w/o warning
|
||||||
IF(CMAKE_CXX_COMPILER_ID MATCHES "GNU|Clang")
|
if(CMAKE_CXX_COMPILER_ID MATCHES "GNU|Clang")
|
||||||
ADD_DEFINITIONS("-DSFMT_MEXP=19937")
|
add_definitions("-DSFMT_MEXP=19937")
|
||||||
ENDIF()
|
endif()
|
||||||
|
|
||||||
# Find Qt5
|
find_package(Threads REQUIRED)
|
||||||
OPTION(UPDATE_TRANSLATIONS "Update translations on compile" OFF)
|
|
||||||
MESSAGE(STATUS "UPDATE TRANSLATIONS: ${UPDATE_TRANSLATIONS}")
|
|
||||||
|
|
||||||
FIND_PACKAGE(Qt5Core 5.5.0 REQUIRED)
|
# Determine 32 or 64 bit build
|
||||||
|
if(CMAKE_SIZEOF_VOID_P EQUAL 8)
|
||||||
|
set(_lib_suffix 64)
|
||||||
|
else()
|
||||||
|
set(_lib_suffix 32)
|
||||||
|
endif()
|
||||||
|
|
||||||
IF(Qt5Core_FOUND)
|
if(DEFINED QTDIR${_lib_suffix})
|
||||||
MESSAGE(STATUS "Found Qt ${Qt5Core_VERSION_STRING}")
|
list(APPEND CMAKE_PREFIX_PATH "${QTDIR${_lib_suffix}}")
|
||||||
|
elseif(DEFINED QTDIR)
|
||||||
|
list(APPEND CMAKE_PREFIX_PATH "${QTDIR}")
|
||||||
|
elseif(DEFINED ENV{QTDIR${_lib_suffix}})
|
||||||
|
list(APPEND CMAKE_PREFIX_PATH "$ENV{QTDIR${_lib_suffix}}")
|
||||||
|
elseif(DEFINED ENV{QTDIR})
|
||||||
|
list(APPEND CMAKE_PREFIX_PATH "$ENV{QTDIR}")
|
||||||
|
endif()
|
||||||
|
|
||||||
# FIX: Qt was built with -reduce-relocations
|
message(STATUS "Update Translations: ${UPDATE_TRANSLATIONS}")
|
||||||
if (Qt5_POSITION_INDEPENDENT_CODE)
|
|
||||||
SET(CMAKE_POSITION_INDEPENDENT_CODE ON)
|
|
||||||
endif()
|
|
||||||
|
|
||||||
# guess plugins and libraries directory
|
include(FindQtRuntime)
|
||||||
set(QT_PLUGINS_DIR "${Qt5Core_DIR}/../../../plugins")
|
|
||||||
get_target_property(QT_LIBRARY_DIR Qt5::Core LOCATION)
|
|
||||||
get_filename_component(QT_LIBRARY_DIR ${QT_LIBRARY_DIR} PATH)
|
|
||||||
|
|
||||||
ELSE()
|
|
||||||
MESSAGE(FATAL_ERROR "No Qt5 found!")
|
|
||||||
ENDIF()
|
|
||||||
|
|
||||||
set(CMAKE_AUTOMOC TRUE)
|
set(CMAKE_AUTOMOC TRUE)
|
||||||
|
|
||||||
# Find other needed libraries
|
# Find other needed libraries
|
||||||
FIND_PACKAGE(Protobuf REQUIRED)
|
find_package(Protobuf CONFIG)
|
||||||
|
if(NOT Protobuf_FOUND)
|
||||||
|
find_package(Protobuf REQUIRED)
|
||||||
|
endif()
|
||||||
|
|
||||||
|
if(${Protobuf_VERSION} VERSION_LESS "3.21.0.0" AND NOT EXISTS "${Protobuf_PROTOC_EXECUTABLE}")
|
||||||
|
message(FATAL_ERROR "No protoc command found!")
|
||||||
|
endif()
|
||||||
|
|
||||||
#Find OpenSSL
|
#Find OpenSSL
|
||||||
IF(WIN32)
|
if(WIN32)
|
||||||
FIND_PACKAGE(Win32SslRuntime)
|
find_package(OpenSSL REQUIRED)
|
||||||
ENDIF()
|
if(OPENSSL_FOUND)
|
||||||
|
include_directories(${OPENSSL_INCLUDE_DIRS})
|
||||||
|
else()
|
||||||
|
message(
|
||||||
|
WARNING
|
||||||
|
"Could not find OpenSSL runtime libraries. They are not required for compiling, but needs to be available at runtime."
|
||||||
|
)
|
||||||
|
endif()
|
||||||
|
endif()
|
||||||
|
|
||||||
#Find VCredist
|
#Find VCredist
|
||||||
IF(MSVC)
|
if(MSVC)
|
||||||
FIND_PACKAGE(VCredistRuntime)
|
find_package(VCredistRuntime)
|
||||||
ENDIF()
|
endif()
|
||||||
|
|
||||||
# Package builder
|
# Package builder
|
||||||
set(CPACK_PACKAGE_CONTACT "Zach Halpern <zahalpern+github@gmail.com>")
|
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_FILE "${PROJECT_SOURCE_DIR}/README.md")
|
||||||
set(CPACK_RESOURCE_FILE_LICENSE "${PROJECT_SOURCE_DIR}/LICENSE")
|
set(CPACK_RESOURCE_FILE_LICENSE "${PROJECT_SOURCE_DIR}/LICENSE")
|
||||||
@@ -183,75 +257,95 @@ set(CPACK_PACKAGE_VERSION_PATCH "${PROJECT_VERSION_PATCH}")
|
|||||||
set(CPACK_PACKAGE_FILE_NAME "${PROJECT_VERSION_FILENAME}")
|
set(CPACK_PACKAGE_FILE_NAME "${PROJECT_VERSION_FILENAME}")
|
||||||
|
|
||||||
if(UNIX)
|
if(UNIX)
|
||||||
if(APPLE)
|
if(APPLE)
|
||||||
set(CPACK_GENERATOR DragNDrop ${CPACK_GENERATOR})
|
set(CPACK_GENERATOR DragNDrop ${CPACK_GENERATOR})
|
||||||
set(CPACK_GENERATOR "DragNDrop")
|
set(CPACK_GENERATOR "DragNDrop")
|
||||||
set(CPACK_DMG_FORMAT "UDBZ")
|
set(CPACK_DMG_FORMAT "UDBZ")
|
||||||
set(CPACK_DMG_VOLUME_NAME "${PROJECT_NAME}")
|
set(CPACK_DMG_VOLUME_NAME "${PROJECT_NAME}")
|
||||||
set(CPACK_SYSTEM_NAME "OSX")
|
set(CPACK_SYSTEM_NAME "OSX")
|
||||||
set(CPACK_PACKAGE_ICON "${CMAKE_CURRENT_SOURCE_DIR}/cockatrice/resources/appicon.icns")
|
set(CPACK_PACKAGE_ICON "${CMAKE_CURRENT_SOURCE_DIR}/cockatrice/resources/appicon.icns")
|
||||||
|
set(CPACK_DMG_DS_STORE_SETUP_SCRIPT "${CMAKE_CURRENT_SOURCE_DIR}/cmake/CMakeDMGSetup.script")
|
||||||
|
set(CPACK_DMG_BACKGROUND_IMAGE "${CMAKE_CURRENT_SOURCE_DIR}/cmake/dmgBackground.tif")
|
||||||
|
else()
|
||||||
|
# linux
|
||||||
|
if(CPACK_GENERATOR STREQUAL "RPM")
|
||||||
|
set(CPACK_RPM_PACKAGE_LICENSE "GPLv2")
|
||||||
|
set(CPACK_RPM_MAIN_COMPONENT "cockatrice")
|
||||||
|
if(Qt6_FOUND)
|
||||||
|
set(CPACK_RPM_PACKAGE_REQUIRES "protobuf, qt6-qttools, qt6-qtsvg, qt6-qtmultimedia, qt6-qtimageformats")
|
||||||
|
elseif(Qt5_FOUND)
|
||||||
|
set(CPACK_RPM_PACKAGE_REQUIRES "protobuf, qt5-qttools, qt5-qtsvg, qt5-qtmultimedia")
|
||||||
|
endif()
|
||||||
|
set(CPACK_RPM_PACKAGE_GROUP "Amusements/Games")
|
||||||
|
set(CPACK_RPM_PACKAGE_URL "http://github.com/Cockatrice/Cockatrice")
|
||||||
|
# stop directories from making package conflicts
|
||||||
|
set(CPACK_RPM_EXCLUDE_FROM_AUTO_FILELIST_ADDITION
|
||||||
|
/usr/share/applications
|
||||||
|
/usr/share/icons
|
||||||
|
/usr/share/icons/hicolor
|
||||||
|
/usr/share/icons/hicolor/48x48
|
||||||
|
/usr/share/icons/hicolor/48x48/apps
|
||||||
|
/usr/share/icons/hicolor/scalable
|
||||||
|
/usr/share/icons/hicolor/scalable/apps
|
||||||
|
)
|
||||||
else()
|
else()
|
||||||
# linux
|
set(CPACK_GENERATOR DEB)
|
||||||
IF(CPACK_GENERATOR STREQUAL "RPM")
|
set(CPACK_DEBIAN_PACKAGE_SHLIBDEPS ON)
|
||||||
set(CPACK_RPM_PACKAGE_LICENSE "GPLv2")
|
set(CPACK_DEBIAN_PACKAGE_SECTION "games")
|
||||||
set(CPACK_RPM_PACKAGE_REQUIRES "protobuf, qt5-qttools, qt5-qtsvg, qt5-qtmultimedia")
|
set(CPACK_DEBIAN_PACKAGE_HOMEPAGE "http://github.com/Cockatrice/Cockatrice")
|
||||||
set(CPACK_RPM_PACKAGE_GROUP "Amusements/Games")
|
if(Qt6_FOUND)
|
||||||
set(CPACK_RPM_PACKAGE_URL "http://github.com/Cockatrice/Cockatrice")
|
set(CPACK_DEBIAN_PACKAGE_DEPENDS "libqt6multimedia6, libqt6svg6, qt6-qpa-plugins, qt6-image-formats-plugins")
|
||||||
ELSE()
|
elseif(Qt5_FOUND)
|
||||||
set(CPACK_GENERATOR DEB)
|
set(CPACK_DEBIAN_PACKAGE_DEPENDS "libqt5multimedia5-plugins, libqt5svg5")
|
||||||
set(CPACK_DEBIAN_PACKAGE_SHLIBDEPS ON)
|
endif()
|
||||||
set(CPACK_DEBIAN_PACKAGE_SECTION "games")
|
|
||||||
set(CPACK_DEBIAN_PACKAGE_HOMEPAGE "http://github.com/Cockatrice/Cockatrice")
|
|
||||||
set(CPACK_DEBIAN_PACKAGE_DEPENDS "libqt5multimedia5-plugins, libqt5svg5")
|
|
||||||
ENDIF()
|
|
||||||
endif()
|
endif()
|
||||||
|
endif()
|
||||||
elseif(WIN32)
|
elseif(WIN32)
|
||||||
set(CPACK_GENERATOR NSIS ${CPACK_GENERATOR})
|
set(CPACK_GENERATOR NSIS ${CPACK_GENERATOR})
|
||||||
if("${CMAKE_GENERATOR}" MATCHES "(Win64|IA64)")
|
if("${CMAKE_GENERATOR_PLATFORM}" MATCHES "(x64)")
|
||||||
set(TRICE_IS_64_BIT 1)
|
set(TRICE_IS_64_BIT 1)
|
||||||
else()
|
else()
|
||||||
set(TRICE_IS_64_BIT 0)
|
set(TRICE_IS_64_BIT 0)
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
# Configure file with custom definitions for NSIS.
|
# Configure file with custom definitions for NSIS.
|
||||||
configure_file(
|
configure_file("${COCKATRICE_CMAKE_PATH}/NSIS.definitions.nsh.in" "${PROJECT_BINARY_DIR}/NSIS.definitions.nsh")
|
||||||
${CMAKE_MODULE_PATH}/NSIS.definitions.nsh.in
|
|
||||||
${PROJECT_BINARY_DIR}/NSIS.definitions.nsh
|
|
||||||
)
|
|
||||||
|
|
||||||
# include vcredist into the package; NSIS will take care of running it
|
# include vcredist into the package; NSIS will take care of running it
|
||||||
if(VCREDISTRUNTIME_FOUND)
|
if(VCREDISTRUNTIME_FOUND)
|
||||||
INSTALL(FILES "${VCREDISTRUNTIME_FILE}" DESTINATION ./)
|
install(FILES "${VCREDISTRUNTIME_FILE}" DESTINATION ./)
|
||||||
endif()
|
endif()
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
include(CPack)
|
include(CPack)
|
||||||
|
|
||||||
# Compile servatrice (default off)
|
|
||||||
option(WITH_SERVER "build servatrice" OFF)
|
|
||||||
add_subdirectory(common)
|
add_subdirectory(common)
|
||||||
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})
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
# Compile cockatrice (default on)
|
|
||||||
option(WITH_CLIENT "build cockatrice" ON)
|
|
||||||
if(WITH_CLIENT)
|
if(WITH_CLIENT)
|
||||||
add_subdirectory(cockatrice)
|
add_subdirectory(cockatrice)
|
||||||
SET(CPACK_INSTALL_CMAKE_PROJECTS "Cockatrice;Cockatrice;ALL;/" ${CPACK_INSTALL_CMAKE_PROJECTS})
|
set(CPACK_INSTALL_CMAKE_PROJECTS "Cockatrice;Cockatrice;ALL;/" ${CPACK_INSTALL_CMAKE_PROJECTS})
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
# Compile oracle (default on)
|
|
||||||
option(WITH_ORACLE "build oracle" ON)
|
|
||||||
if(WITH_ORACLE)
|
if(WITH_ORACLE)
|
||||||
add_subdirectory(oracle)
|
add_subdirectory(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()
|
||||||
|
|
||||||
# Compile tests (default off)
|
if(WITH_DBCONVERTER)
|
||||||
option(TEST "build tests" OFF)
|
add_subdirectory(dbconverter)
|
||||||
if(TEST)
|
set(CPACK_INSTALL_CMAKE_PROJECTS "Dbconverter;Dbconverter;ALL;/" ${CPACK_INSTALL_CMAKE_PROJECTS})
|
||||||
include(CTest)
|
endif()
|
||||||
add_subdirectory(tests)
|
|
||||||
|
if(TEST)
|
||||||
|
include(CTest)
|
||||||
|
add_subdirectory(tests)
|
||||||
|
endif()
|
||||||
|
|
||||||
|
if(Qt6_FOUND AND Qt6_VERSION_MINOR GREATER_EQUAL 3)
|
||||||
|
# Qt6.3+ requires project finalization to support translations
|
||||||
|
qt6_finalize_project()
|
||||||
endif()
|
endif()
|
||||||
|
|||||||
@@ -7,6 +7,7 @@ RUN apt-get update && apt-get install -y\
|
|||||||
git\
|
git\
|
||||||
libprotobuf-dev\
|
libprotobuf-dev\
|
||||||
libqt5sql5-mysql\
|
libqt5sql5-mysql\
|
||||||
|
libmysqlclient-dev\
|
||||||
libqt5websockets5-dev\
|
libqt5websockets5-dev\
|
||||||
protobuf-compiler\
|
protobuf-compiler\
|
||||||
qt5-default\
|
qt5-default\
|
||||||
@@ -18,7 +19,7 @@ COPY . /home/servatrice/code/
|
|||||||
WORKDIR /home/servatrice/code
|
WORKDIR /home/servatrice/code
|
||||||
|
|
||||||
WORKDIR build
|
WORKDIR build
|
||||||
RUN cmake .. -DWITH_SERVER=1 -DWITH_CLIENT=0 -DWITH_ORACLE=0 &&\
|
RUN cmake .. -DWITH_SERVER=1 -DWITH_CLIENT=0 -DWITH_ORACLE=0 -DWITH_DBCONVERTER=0 &&\
|
||||||
make &&\
|
make &&\
|
||||||
make install
|
make install
|
||||||
|
|
||||||
@@ -26,4 +27,5 @@ WORKDIR /home/servatrice
|
|||||||
|
|
||||||
EXPOSE 4747
|
EXPOSE 4747
|
||||||
|
|
||||||
CMD [ "servatrice", "--log-to-console" ]
|
ENTRYPOINT [ "servatrice", "--log-to-console" ]
|
||||||
|
|
||||||
|
|||||||
60
README.md
@@ -29,21 +29,21 @@ Cockatrice is an open-source, multiplatform program for playing tabletop card ga
|
|||||||
|
|
||||||
# Download [](https://tooomm.github.io/github-release-stats/?username=Cockatrice&repository=Cockatrice)
|
# Download [](https://tooomm.github.io/github-release-stats/?username=Cockatrice&repository=Cockatrice)
|
||||||
|
|
||||||
Downloads are available for full releases and the current beta version in development.<br>
|
Downloads are available for full releases and the current beta version in development. There is no strict release schedule for either of them.
|
||||||
Full releases are checkpoints featuring major feature or UI enhancements - we recommend to use those. There is no strict schedule for new full releases.
|
|
||||||
|
|
||||||
The beta release contains the most recently added features and bugfixes, but can be unstable. They are released as we feel need.
|
- Latest `stable` release: [](https://github.com/cockatrice/cockatrice/releases/latest) [](https://tooomm.github.io/github-release-stats/?username=Cockatrice&repository=Cockatrice)<br>
|
||||||
|
- Stable versions are checkpoints featuring major feature and UI enhancements.
|
||||||
|
- **Recommended for most users!**
|
||||||
|
|
||||||
- Latest `stable` release (**recommended**): [](https://github.com/cockatrice/cockatrice/releases/latest) [](https://tooomm.github.io/github-release-stats/?username=Cockatrice&repository=Cockatrice)<br>
|
- Latest `beta` release: [](https://github.com/cockatrice/cockatrice/releases) [](https://tooomm.github.io/github-release-stats/?username=Cockatrice&repository=Cockatrice)
|
||||||
|
- Beta versions include the most recently added features and bugfixes, but can be unstable.
|
||||||
- Latest `beta` release: [](https://github.com/cockatrice/cockatrice/releases) [](https://tooomm.github.io/github-release-stats/?username=Cockatrice&repository=Cockatrice)
|
|
||||||
- Beta versions may be unstable and contain bugs.
|
|
||||||
- To be a Cockatrice Beta Tester, use this version. Find more information [here](https://github.com/Cockatrice/Cockatrice/wiki/Release-Channels)!
|
- To be a Cockatrice Beta Tester, use this version. Find more information [here](https://github.com/Cockatrice/Cockatrice/wiki/Release-Channels)!
|
||||||
|
|
||||||
|
|
||||||
|
# Get Involved [](https://discord.gg/3Z9yzmA)
|
||||||
|
|
||||||
# Get Involved [](https://gitter.im/Cockatrice/Cockatrice)
|
Join our [Discord community](https://discord.gg/3Z9yzmA) to connect with the project, contributors or fellow users of the app. Come here to talk about the application, features, or just to hang out.<br>
|
||||||
|
For support regarding specific servers, please contact that server's admin or forum for support rather than asking here.<br>
|
||||||
[Chat](https://gitter.im/Cockatrice/Cockatrice) with the Cockatrice developers on Gitter. Come here to talk about the application, features, or just to hang out. For support regarding specific servers, please contact that server's admin or forum for support rather than asking here.<br>
|
|
||||||
|
|
||||||
To contribute code to the project, please review [the guidelines](https://github.com/Cockatrice/Cockatrice/blob/master/.github/CONTRIBUTING.md).
|
To contribute code to the project, please review [the guidelines](https://github.com/Cockatrice/Cockatrice/blob/master/.github/CONTRIBUTING.md).
|
||||||
We maintain two tags for contributors to find issues to work on:
|
We maintain two tags for contributors to find issues to work on:
|
||||||
@@ -52,8 +52,6 @@ We maintain two tags for contributors to find issues to work on:
|
|||||||
|
|
||||||
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 review for changes they submit.
|
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 review for changes they submit.
|
||||||
|
|
||||||
Read the long-term project **roadmap** to see planned edits and milestones [here](https://docs.google.com/document/d/1Ewe5uSaRE2nR2pNPMaGmP6gVZdqgFbBgwSscGqIr4W0/edit).
|
|
||||||
|
|
||||||
We try to be responsive to new issues. We'll provide advice on how best to implement a feature; alternately, we can show you where the codebase is doing something similar before you get too far along.
|
We try to be responsive to new issues. We'll provide advice on how best to implement a feature; alternately, we can show you where the codebase is doing something similar before you get too far along.
|
||||||
|
|
||||||
Cockatrice uses 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 uses 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.
|
||||||
@@ -67,29 +65,25 @@ Cockatrice uses the [Google Developer Documentation Style Guide](https://develop
|
|||||||
- [reddit r/Cockatrice](https://reddit.com/r/cockatrice)
|
- [reddit r/Cockatrice](https://reddit.com/r/cockatrice)
|
||||||
|
|
||||||
|
|
||||||
# Translations [](https://www.transifex.com/projects/p/cockatrice/)
|
# Translations [](https://transifex.com/cockatrice/cockatrice/)
|
||||||
|
|
||||||
Cockatrice uses Transifex for translations. You can help us bring Cockatrice and Oracle to your language or just edit single wordings right from within your browser by visiting our [Transifex project page](https://www.transifex.com/projects/p/cockatrice/).<br>
|
Cockatrice uses Transifex for translations. You can help us bring Cockatrice, Oracle and Webatrice to your language or just adjust single wordings right from within your browser by visiting our [Transifex project page](https://transifex.com/cockatrice/cockatrice/).<br>
|
||||||
|
|
||||||
| Cockatrice | Oracle |
|
|
||||||
|:-:|:-:|
|
|
||||||
| [](https://www.transifex.com/projects/p/cockatrice/) | [](https://www.transifex.com/projects/p/cockatrice/) |
|
|
||||||
|
|
||||||
Check out our [Translator FAQ](https://github.com/Cockatrice/Cockatrice/wiki/Translation-FAQ) for more information about contributing!<br>
|
Check out our [Translator FAQ](https://github.com/Cockatrice/Cockatrice/wiki/Translation-FAQ) for more information about contributing!<br>
|
||||||
|
|
||||||
|
|
||||||
# Build [](https://travis-ci.org/Cockatrice/Cockatrice) [](https://ci.appveyor.com/project/Daenyth/cockatrice/branch/master)
|
# Build [](https://github.com/Cockatrice/Cockatrice/actions/workflows/desktop-build.yml?query=branch%3Amaster+event%3Apush) [](https://github.com/Cockatrice/Cockatrice/actions/workflows/web-build.yml?query=branch%3Amaster+event%3Apush)
|
||||||
|
|
||||||
**Detailed compiling instructions are on the Cockatrice wiki under [Compiling Cockatrice](https://github.com/Cockatrice/Cockatrice/wiki/Compiling-Cockatrice)**
|
**Detailed compiling instructions can be found on the Cockatrice wiki under [Compiling Cockatrice](https://github.com/Cockatrice/Cockatrice/wiki/Compiling-Cockatrice)**
|
||||||
|
|
||||||
Dependencies: *(for minimum requirements search our [CMake file](https://github.com/Cockatrice/Cockatrice/blob/master/CMakeLists.txt))*
|
Dependencies: *(for minimum requirements 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/)
|
||||||
- [protobuf](https://github.com/google/protobuf)
|
- [protobuf](https://github.com/protocolbuffers/protobuf)
|
||||||
- [CMake](https://www.cmake.org/)
|
- [CMake](https://www.cmake.org/)
|
||||||
|
|
||||||
Oracle can optionally use zlib and xz to load compressed files:
|
Oracle can optionally use zlib and xz to load compressed files:
|
||||||
- [zlib](https://www.zlib.net/)
|
|
||||||
- [xz](https://tukaani.org/xz/)
|
- [xz](https://tukaani.org/xz/)
|
||||||
|
- [zlib](https://www.zlib.net/)
|
||||||
|
|
||||||
To compile:
|
To compile:
|
||||||
|
|
||||||
@@ -117,6 +111,7 @@ The following flags can be passed to `cmake`:
|
|||||||
- `-DWARNING_AS_ERROR=0` Whether to treat compilation warnings as errors in debug mode (default 1 = yes).
|
- `-DWARNING_AS_ERROR=0` Whether to treat compilation warnings as errors in debug mode (default 1 = yes).
|
||||||
- `-DUPDATE_TRANSLATIONS=1` Configure `make` to update the translation .ts files for new strings in the source code. Note: Running `make clean` will remove the .ts files (default 0 = no).
|
- `-DUPDATE_TRANSLATIONS=1` Configure `make` to update the translation .ts files for new strings in the source code. Note: Running `make clean` will remove the .ts files (default 0 = no).
|
||||||
- `-DTEST=1` Enable regression tests (default 0 = no). Note: needs googletest, will be downloaded on the fly if unavailable. To run tests: ```make test```.
|
- `-DTEST=1` Enable regression tests (default 0 = no). Note: needs googletest, will be downloaded on the fly if unavailable. To run tests: ```make test```.
|
||||||
|
- `-DFORCE_USE_QT5=1` Skip looking for Qt6 before trying to find Qt5
|
||||||
|
|
||||||
|
|
||||||
# Run
|
# Run
|
||||||
@@ -125,7 +120,6 @@ The following flags can be passed to `cmake`:
|
|||||||
`Oracle` fetches card data<br>
|
`Oracle` fetches card data<br>
|
||||||
`Servatrice` is the server<br>
|
`Servatrice` is the server<br>
|
||||||
|
|
||||||
|
|
||||||
**Servatrice Docker container**
|
**Servatrice Docker container**
|
||||||
|
|
||||||
You can run an instance of Servatrice (the Cockatrice server) using [Docker](https://www.docker.com/what-docker) and the Cockatrice Dockerfile.<br>
|
You can run an instance of Servatrice (the Cockatrice server) using [Docker](https://www.docker.com/what-docker) and the Cockatrice Dockerfile.<br>
|
||||||
@@ -141,6 +135,26 @@ to permit connections to the server.
|
|||||||
|
|
||||||
Find more information on how to use Servatrice with Docker in our [wiki](https://github.com/Cockatrice/Cockatrice/wiki/Setting-up-Servatrice#using-docker).
|
Find more information on how to use Servatrice with Docker in our [wiki](https://github.com/Cockatrice/Cockatrice/wiki/Setting-up-Servatrice#using-docker).
|
||||||
|
|
||||||
|
**Docker compose**
|
||||||
|
|
||||||
|
There is also a docker-compose file available which will configure and run both a MySQL server and Servatrice. The docker-compose setup scripts can be found in the `servatrice/docker` folder and vary only slightly from the default sql and server .ini files. The setup scripts can either be modified in place, or docker-compose can mount alternative files into the images, as you prefer.
|
||||||
|
|
||||||
|
To run Servatrice via docker-compose, first install docker-compose following the [install instructions](https://docs.docker.com/compose/install/). Once installed, run the following from the root of the repository:
|
||||||
|
```bash
|
||||||
|
docker-compose build # Build the Servatrice image using the same Dockerfile as above.
|
||||||
|
docker-compose up # Setup and run both the MySQL server and Servatrice.
|
||||||
|
```
|
||||||
|
|
||||||
|
>Note: Similar to the above Docker setup, this will expose TCP ports 4747 and 4748.
|
||||||
|
|
||||||
|
>Note: The first time running the docker-compose setup, the MySQL server will take a little time to run the initial setup scripts. Due to this, the Servatrice instance may fail the first few attempts to connect to the database. Servatrice is set to `restart: always` in the docker-compose.yml, which will allow it to continue attempting to start up. Once the MySQL scripts have completed, Servatrice should then connect automatically on the next attempt.
|
||||||
|
|
||||||
|
**Docker compose in Windows**
|
||||||
|
A out of box working docker-compose file has been added to help setup in Windows.
|
||||||
|
|
||||||
|
Docker in Windows requires additional steps in form of using Docker Desktop to allow resource sharing from the drive the volumes are mapped from, as well as potential workarounds needed to get file sharing working in Windows. This [StackOverflow discussion sheds some light on it](https://stackoverflow.com/questions/42203488/settings-to-windows-firewall-to-allow-docker-for-windows-to-share-drive)
|
||||||
|
|
||||||
|
|
||||||
# License [](https://github.com/Cockatrice/Cockatrice/blob/master/LICENSE)
|
# License [](https://github.com/Cockatrice/Cockatrice/blob/master/LICENSE)
|
||||||
|
|
||||||
Cockatrice is free software, licensed under the [GPLv2](https://github.com/Cockatrice/Cockatrice/blob/master/LICENSE).
|
Cockatrice is free software, licensed under the [GPLv2](https://github.com/Cockatrice/Cockatrice/blob/master/LICENSE).
|
||||||
|
|||||||
55
cmake/CMakeDMGSetup.script
Normal file
@@ -0,0 +1,55 @@
|
|||||||
|
on run argv
|
||||||
|
set image_name to item 1 of argv
|
||||||
|
|
||||||
|
tell application "Finder"
|
||||||
|
tell disk image_name
|
||||||
|
|
||||||
|
-- wait for the image to finish mounting
|
||||||
|
set open_attempts to 0
|
||||||
|
repeat while open_attempts < 4
|
||||||
|
try
|
||||||
|
open
|
||||||
|
delay 1
|
||||||
|
set open_attempts to 5
|
||||||
|
close
|
||||||
|
on error errStr number errorNumber
|
||||||
|
set open_attempts to open_attempts + 1
|
||||||
|
delay 10
|
||||||
|
end try
|
||||||
|
end repeat
|
||||||
|
delay 5
|
||||||
|
|
||||||
|
-- open the image the first time and save a DS_Store with just
|
||||||
|
-- background and icon setup
|
||||||
|
open
|
||||||
|
set current view of container window to icon view
|
||||||
|
set theViewOptions to the icon view options of container window
|
||||||
|
set background picture of theViewOptions to file ".background:background.tif"
|
||||||
|
set arrangement of theViewOptions to not arranged
|
||||||
|
set icon size of theViewOptions to 128
|
||||||
|
delay 5
|
||||||
|
close
|
||||||
|
|
||||||
|
-- next setup the position of the app and Applications symlink
|
||||||
|
-- plus hide all the window decoration
|
||||||
|
open
|
||||||
|
update without registering applications
|
||||||
|
tell container window
|
||||||
|
set sidebar width to 0
|
||||||
|
set statusbar visible to false
|
||||||
|
set toolbar visible to false
|
||||||
|
set the bounds to { 400, 100, 1400, 922 }
|
||||||
|
set position of item "Cockatrice.app" to { 139, 214 }
|
||||||
|
set position of item "Oracle.app" to { 139, 414 }
|
||||||
|
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 }
|
||||||
|
end tell
|
||||||
|
update without registering applications
|
||||||
|
delay 5
|
||||||
|
close
|
||||||
|
|
||||||
|
end tell
|
||||||
|
delay 1
|
||||||
|
end tell
|
||||||
|
end run
|
||||||
@@ -1,18 +1,18 @@
|
|||||||
# Find the LibExecinfo library - FreeBSD only
|
# Find the LibExecinfo library - FreeBSD only
|
||||||
|
|
||||||
FIND_PATH(LIBEXECINFO_INCLUDE_DIR execinfo.h)
|
find_path(LIBEXECINFO_INCLUDE_DIR execinfo.h)
|
||||||
FIND_LIBRARY(LIBEXECINFO_LIBRARY NAMES execinfo)
|
find_library(LIBEXECINFO_LIBRARY NAMES execinfo)
|
||||||
|
|
||||||
IF(LIBEXECINFO_INCLUDE_DIR AND LIBEXECINFO_LIBRARY)
|
if(LIBEXECINFO_INCLUDE_DIR AND LIBEXECINFO_LIBRARY)
|
||||||
SET(LIBEXECINFO_FOUND TRUE)
|
set(LIBEXECINFO_FOUND TRUE)
|
||||||
ENDIF()
|
endif()
|
||||||
|
|
||||||
IF(LIBEXECINFO_FOUND)
|
if(LIBEXECINFO_FOUND)
|
||||||
IF(NOT LIBEXECINFO_FIND_QUIETLY)
|
if(NOT LIBEXECINFO_FIND_QUIETLY)
|
||||||
MESSAGE(STATUS "Found LibExecinfo: ${EXECINFO_LIBRARY}")
|
message(STATUS "Found LibExecinfo: ${EXECINFO_LIBRARY}")
|
||||||
ENDIF()
|
endif()
|
||||||
ELSE()
|
else()
|
||||||
IF(LIBEXECINFO_FIND_REQUIRED)
|
if(LIBEXECINFO_FIND_REQUIRED)
|
||||||
MESSAGE(FATAL_ERROR "Could not find LibExecinfo")
|
message(FATAL_ERROR "Could not find LibExecinfo")
|
||||||
ENDIF()
|
endif()
|
||||||
ENDIF()
|
endif()
|
||||||
|
|||||||
117
cmake/FindQtRuntime.cmake
Normal file
@@ -0,0 +1,117 @@
|
|||||||
|
# Find a compatible Qt version
|
||||||
|
# Inputs: WITH_SERVER, WITH_CLIENT, WITH_ORACLE, WITH_DBCONVERTER, FORCE_USE_QT5
|
||||||
|
# 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
|
||||||
|
# Output: COCKATRICE_QT_VERSION_NAME -- Example values: Qt5, Qt6
|
||||||
|
# Output: SERVATRICE_QT_MODULES
|
||||||
|
# Output: COCKATRICE_QT_MODULES
|
||||||
|
# Output: ORACLE_QT_MODULES
|
||||||
|
# Output: DBCONVERTER_QT_MODULES
|
||||||
|
# Output: TEST_QT_MODULES
|
||||||
|
|
||||||
|
set(REQUIRED_QT_COMPONENTS Core)
|
||||||
|
if(WITH_SERVER)
|
||||||
|
set(_SERVATRICE_NEEDED Network Sql WebSockets)
|
||||||
|
endif()
|
||||||
|
if(WITH_CLIENT)
|
||||||
|
set(_COCKATRICE_NEEDED
|
||||||
|
Concurrent
|
||||||
|
Gui
|
||||||
|
Multimedia
|
||||||
|
Network
|
||||||
|
PrintSupport
|
||||||
|
Svg
|
||||||
|
WebSockets
|
||||||
|
Widgets
|
||||||
|
)
|
||||||
|
endif()
|
||||||
|
if(WITH_ORACLE)
|
||||||
|
set(_ORACLE_NEEDED Concurrent Network Svg Widgets)
|
||||||
|
endif()
|
||||||
|
if(WITH_DBCONVERTER)
|
||||||
|
set(_DBCONVERTER_NEEDED Network Widgets)
|
||||||
|
endif()
|
||||||
|
if(TEST)
|
||||||
|
set(_TEST_NEEDED Widgets)
|
||||||
|
endif()
|
||||||
|
|
||||||
|
set(REQUIRED_QT_COMPONENTS ${REQUIRED_QT_COMPONENTS} ${_SERVATRICE_NEEDED} ${_COCKATRICE_NEEDED} ${_ORACLE_NEEDED}
|
||||||
|
${_DBCONVERTER_NEEDED} ${_TEST_NEEDED}
|
||||||
|
)
|
||||||
|
list(REMOVE_DUPLICATES REQUIRED_QT_COMPONENTS)
|
||||||
|
|
||||||
|
if(NOT FORCE_USE_QT5)
|
||||||
|
# Linguist is now a component in Qt6 instead of an external package
|
||||||
|
find_package(
|
||||||
|
Qt6 6.2.3
|
||||||
|
COMPONENTS ${REQUIRED_QT_COMPONENTS} Linguist
|
||||||
|
QUIET HINTS ${Qt6_DIR}
|
||||||
|
)
|
||||||
|
endif()
|
||||||
|
if(Qt6_FOUND)
|
||||||
|
set(COCKATRICE_QT_VERSION_NAME Qt6)
|
||||||
|
|
||||||
|
list(FIND Qt6LinguistTools_TARGETS Qt6::lrelease QT6_LRELEASE_INDEX)
|
||||||
|
if(QT6_LRELEASE_INDEX EQUAL -1)
|
||||||
|
message(WARNING "Qt6 lrelease not found.")
|
||||||
|
endif()
|
||||||
|
|
||||||
|
list(FIND Qt6LinguistTools_TARGETS Qt6::lupdate QT6_LUPDATE_INDEX)
|
||||||
|
if(QT6_LUPDATE_INDEX EQUAL -1)
|
||||||
|
message(WARNING "Qt6 lupdate not found.")
|
||||||
|
endif()
|
||||||
|
else()
|
||||||
|
find_package(
|
||||||
|
Qt5 5.8.0
|
||||||
|
COMPONENTS ${REQUIRED_QT_COMPONENTS}
|
||||||
|
QUIET HINTS ${Qt5_DIR}
|
||||||
|
)
|
||||||
|
if(Qt5_FOUND)
|
||||||
|
set(COCKATRICE_QT_VERSION_NAME Qt5)
|
||||||
|
else()
|
||||||
|
message(FATAL_ERROR "No suitable version of Qt was found")
|
||||||
|
endif()
|
||||||
|
|
||||||
|
# Qt5 Linguist is in a separate package
|
||||||
|
find_package(Qt5LinguistTools QUIET)
|
||||||
|
if(Qt5LinguistTools_FOUND)
|
||||||
|
if(NOT Qt5_LRELEASE_EXECUTABLE)
|
||||||
|
message(WARNING "Qt5 lrelease not found.")
|
||||||
|
endif()
|
||||||
|
if(NOT Qt5_LUPDATE_EXECUTABLE)
|
||||||
|
message(WARNING "Qt5 lupdate not found.")
|
||||||
|
endif()
|
||||||
|
else()
|
||||||
|
message(WARNING "Linguist Tools not found, cannot handle translations")
|
||||||
|
endif()
|
||||||
|
endif()
|
||||||
|
|
||||||
|
if(Qt5_POSITION_INDEPENDENT_CODE OR Qt6_FOUND)
|
||||||
|
set(CMAKE_POSITION_INDEPENDENT_CODE ON)
|
||||||
|
endif()
|
||||||
|
|
||||||
|
# Establish Qt Plugins directory & Library directories
|
||||||
|
get_target_property(QT_LIBRARY_DIR ${COCKATRICE_QT_VERSION_NAME}::Core LOCATION)
|
||||||
|
get_filename_component(QT_LIBRARY_DIR ${QT_LIBRARY_DIR} DIRECTORY)
|
||||||
|
if(Qt6_FOUND)
|
||||||
|
get_filename_component(QT_PLUGINS_DIR "${Qt6Core_DIR}/../../../${QT6_INSTALL_PLUGINS}" ABSOLUTE)
|
||||||
|
get_filename_component(QT_LIBRARY_DIR "${QT_LIBRARY_DIR}/../../.." ABSOLUTE)
|
||||||
|
if(UNIX AND APPLE)
|
||||||
|
# Mac needs a bit more help finding all necessary components
|
||||||
|
list(APPEND QT_LIBRARY_DIR "/usr/local/lib")
|
||||||
|
endif()
|
||||||
|
elseif(Qt5_FOUND)
|
||||||
|
get_filename_component(QT_PLUGINS_DIR "${Qt5Core_DIR}/../../../plugins" ABSOLUTE)
|
||||||
|
get_filename_component(QT_LIBRARY_DIR "${QT_LIBRARY_DIR}/.." ABSOLUTE)
|
||||||
|
endif()
|
||||||
|
message(DEBUG "QT_PLUGINS_DIR = ${QT_PLUGINS_DIR}")
|
||||||
|
message(DEBUG "QT_LIBRARY_DIR = ${QT_LIBRARY_DIR}")
|
||||||
|
|
||||||
|
# Establish exports
|
||||||
|
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" 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}")
|
||||||
|
|
||||||
|
message(STATUS "Found Qt ${${COCKATRICE_QT_VERSION_NAME}_VERSION} at: ${${COCKATRICE_QT_VERSION_NAME}_DIR}")
|
||||||
@@ -1,36 +1,42 @@
|
|||||||
# Find the MS Visual Studio VC redistributable package
|
# Find the MS Visual Studio VC redistributable package
|
||||||
|
|
||||||
if (WIN32)
|
if(WIN32)
|
||||||
set(VCREDISTRUNTIME_FOUND "NO")
|
set(VCREDISTRUNTIME_FOUND "NO")
|
||||||
|
|
||||||
if(CMAKE_SIZEOF_VOID_P EQUAL 8) # 64-bit
|
if(CMAKE_SIZEOF_VOID_P EQUAL 8) # 64-bit
|
||||||
set(REDIST_ARCH x64)
|
set(REDIST_ARCH x64)
|
||||||
else()
|
else()
|
||||||
set(REDIST_ARCH x86)
|
set(REDIST_ARCH x86)
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
set(REDIST_FILE vcredist_${REDIST_ARCH}.exe)
|
# VS 2017 uses vcredist_ARCH.exe, VS 2022 uses vc_redist.ARCH.exe
|
||||||
|
set(REDIST_FILE_NAMES vcredist_${REDIST_ARCH}.exe vcredist.${REDIST_ARCH}.exe vc_redist.${REDIST_ARCH}.exe)
|
||||||
|
|
||||||
set(CMAKE_INSTALL_SYSTEM_RUNTIME_LIBS_SKIP TRUE)
|
set(CMAKE_INSTALL_SYSTEM_RUNTIME_LIBS_SKIP TRUE)
|
||||||
include(InstallRequiredSystemLibraries)
|
include(InstallRequiredSystemLibraries)
|
||||||
|
|
||||||
# Check if the list contains minimum one element, to get the path from
|
# Check if the list contains minimum one element, to get the path from
|
||||||
list(LENGTH CMAKE_INSTALL_SYSTEM_RUNTIME_LIBS libsCount)
|
list(LENGTH CMAKE_INSTALL_SYSTEM_RUNTIME_LIBS libsCount)
|
||||||
if (libsCount GREATER 0)
|
if(libsCount GREATER 0)
|
||||||
list(GET CMAKE_INSTALL_SYSTEM_RUNTIME_LIBS 0 _path)
|
list(GET CMAKE_INSTALL_SYSTEM_RUNTIME_LIBS 0 _path)
|
||||||
|
|
||||||
get_filename_component(_path ${_path} DIRECTORY)
|
get_filename_component(_path ${_path} DIRECTORY)
|
||||||
get_filename_component(_path ${_path}/../../ ABSOLUTE)
|
get_filename_component(_path ${_path}/../../ ABSOLUTE)
|
||||||
|
|
||||||
if (EXISTS "${_path}/${REDIST_FILE}") # VS 2017
|
foreach(redist_file ${REDIST_FILE_NAMES})
|
||||||
set(VCREDISTRUNTIME_FOUND "YES")
|
if(EXISTS "${_path}/${redist_file}")
|
||||||
set(VCREDISTRUNTIME_FILE ${_path}/${REDIST_FILE})
|
set(VCREDISTRUNTIME_FOUND "YES")
|
||||||
endif()
|
set(VCREDISTRUNTIME_FILE ${_path}/${redist_file})
|
||||||
endif()
|
break()
|
||||||
|
endif()
|
||||||
|
endforeach()
|
||||||
|
endif()
|
||||||
|
|
||||||
if(VCREDISTRUNTIME_FOUND)
|
if(VCREDISTRUNTIME_FOUND)
|
||||||
message(STATUS "Found VCredist ${VCREDISTRUNTIME_FILE}")
|
message(STATUS "Found VCredist ${VCREDISTRUNTIME_FILE}")
|
||||||
else()
|
else()
|
||||||
message(WARNING "Could not find VCredist package. It's not required for compiling, but needs to be available at runtime.")
|
message(
|
||||||
endif()
|
WARNING "Could not find VCredist package. It's not required for compiling, but needs to be available at runtime."
|
||||||
|
)
|
||||||
|
endif()
|
||||||
endif()
|
endif()
|
||||||
|
|||||||
@@ -1,71 +0,0 @@
|
|||||||
# Find the OpenSSL runtime libraries (.dll) for Windows that
|
|
||||||
# will be needed by Qt in order to access https urls.
|
|
||||||
|
|
||||||
if (WIN32)
|
|
||||||
# Get standard installation paths for OpenSSL under Windows
|
|
||||||
|
|
||||||
# http://www.slproweb.com/products/Win32OpenSSL.html
|
|
||||||
|
|
||||||
if( CMAKE_SIZEOF_VOID_P EQUAL 8 )
|
|
||||||
# target win64
|
|
||||||
set(_OPENSSL_ROOT_HINTS
|
|
||||||
${OPENSSL_ROOT_DIR}
|
|
||||||
"[HKEY_LOCAL_MACHINE\\SOFTWARE\\Microsoft\\Windows\\CurrentVersion\\Uninstall\\OpenSSL (64-bit)_is1;Inno Setup: App Path]"
|
|
||||||
ENV OPENSSL_ROOT_DIR
|
|
||||||
)
|
|
||||||
file(TO_CMAKE_PATH "$ENV{PROGRAMFILES}" _programfiles)
|
|
||||||
set(_OPENSSL_ROOT_PATHS
|
|
||||||
"C:/Tools/vcpkg/installed/x64-windows/bin"
|
|
||||||
"${_programfiles}/OpenSSL-Win64"
|
|
||||||
"C:/OpenSSL-Win64/"
|
|
||||||
)
|
|
||||||
unset(_programfiles)
|
|
||||||
else( CMAKE_SIZEOF_VOID_P EQUAL 8 )
|
|
||||||
# target win32
|
|
||||||
set(_OPENSSL_ROOT_HINTS
|
|
||||||
${OPENSSL_ROOT_DIR}
|
|
||||||
"[HKEY_LOCAL_MACHINE\\SOFTWARE\\Microsoft\\Windows\\CurrentVersion\\Uninstall\\OpenSSL (32-bit)_is1;Inno Setup: App Path]"
|
|
||||||
ENV OPENSSL_ROOT_DIR
|
|
||||||
)
|
|
||||||
file(TO_CMAKE_PATH "$ENV{PROGRAMFILES}" _programfiles)
|
|
||||||
set(_OPENSSL_ROOT_PATHS
|
|
||||||
"C:/Tools/vcpkg/installed/x86-windows/bin"
|
|
||||||
"${_programfiles}/OpenSSL"
|
|
||||||
"${_programfiles}/OpenSSL-Win32"
|
|
||||||
"C:/OpenSSL/"
|
|
||||||
"C:/OpenSSL-Win32/"
|
|
||||||
)
|
|
||||||
unset(_programfiles)
|
|
||||||
endif( CMAKE_SIZEOF_VOID_P EQUAL 8 )
|
|
||||||
|
|
||||||
else ()
|
|
||||||
set(_OPENSSL_ROOT_HINTS
|
|
||||||
${OPENSSL_ROOT_DIR}
|
|
||||||
ENV OPENSSL_ROOT_DIR
|
|
||||||
)
|
|
||||||
endif ()
|
|
||||||
|
|
||||||
set(_OPENSSL_ROOT_HINTS_AND_PATHS
|
|
||||||
HINTS ${_OPENSSL_ROOT_HINTS}
|
|
||||||
PATHS ${_OPENSSL_ROOT_PATHS}
|
|
||||||
)
|
|
||||||
|
|
||||||
# For OpenSSL < 1.1, they are named libeay32 and ssleay32 and even if the dll is 64bit, it's still suffixed as *32.dll
|
|
||||||
# For OpenSSL >= 1.1, they are named libcrypto and libssl with no suffix
|
|
||||||
FIND_FILE(WIN32SSLRUNTIME_LIBEAY NAMES libeay32.dll libcrypto.dll ${_OPENSSL_ROOT_HINTS_AND_PATHS})
|
|
||||||
FIND_FILE(WIN32SSLRUNTIME_SSLEAY NAMES ssleay32.dll libssl.dll ${_OPENSSL_ROOT_HINTS_AND_PATHS})
|
|
||||||
|
|
||||||
|
|
||||||
IF(WIN32SSLRUNTIME_LIBEAY AND WIN32SSLRUNTIME_SSLEAY)
|
|
||||||
SET(WIN32SSLRUNTIME_LIBRARIES "${WIN32SSLRUNTIME_LIBEAY}" "${WIN32SSLRUNTIME_SSLEAY}")
|
|
||||||
SET(WIN32SSLRUNTIME_FOUND "YES")
|
|
||||||
message(STATUS "Found OpenSSL ${WIN32SSLRUNTIME_LIBRARIES}")
|
|
||||||
ELSE()
|
|
||||||
SET(WIN32SSLRUNTIME_FOUND "NO")
|
|
||||||
message(WARNING "Could not find OpenSSL runtime libraries. They are not required for compiling, but needs to be available at runtime.")
|
|
||||||
ENDIF()
|
|
||||||
|
|
||||||
MARK_AS_ADVANCED(
|
|
||||||
WIN32SSLRUNTIME_LIBEAY
|
|
||||||
WIN32SSLRUNTIME_SSLEAY
|
|
||||||
)
|
|
||||||
@@ -34,7 +34,5 @@
|
|||||||
<string>${MACOSX_BUNDLE_COPYRIGHT}</string>
|
<string>${MACOSX_BUNDLE_COPYRIGHT}</string>
|
||||||
<key>NSHighResolutionCapable</key>
|
<key>NSHighResolutionCapable</key>
|
||||||
<true/>
|
<true/>
|
||||||
<key>NSRequiresAquaSystemAppearance</key>
|
|
||||||
<true/>
|
|
||||||
</dict>
|
</dict>
|
||||||
</plist>
|
</plist>
|
||||||
|
|||||||
@@ -213,6 +213,7 @@ ${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"
|
||||||
@@ -247,20 +248,20 @@ ${If} $PortableMode = 0
|
|||||||
WriteRegDWORD HKLM "Software\Microsoft\Windows\CurrentVersion\Uninstall\Cockatrice" "VersionMajor" "@CPACK_PACKAGE_VERSION_MAJOR@"
|
WriteRegDWORD HKLM "Software\Microsoft\Windows\CurrentVersion\Uninstall\Cockatrice" "VersionMajor" "@CPACK_PACKAGE_VERSION_MAJOR@"
|
||||||
WriteRegDWORD HKLM "Software\Microsoft\Windows\CurrentVersion\Uninstall\Cockatrice" "VersionMinor" "@CPACK_PACKAGE_VERSION_MINOR@"
|
WriteRegDWORD HKLM "Software\Microsoft\Windows\CurrentVersion\Uninstall\Cockatrice" "VersionMinor" "@CPACK_PACKAGE_VERSION_MINOR@"
|
||||||
|
|
||||||
IfFileExists "$INSTDIR\vcredist_x86.exe" VcRedist86Exists PastVcRedist86Check
|
IfFileExists "$INSTDIR\vc_redist.x86.exe" VcRedist86Exists PastVcRedist86Check
|
||||||
VcRedist86Exists:
|
VcRedist86Exists:
|
||||||
ExecWait '"$INSTDIR\vcredist_x86.exe" /passive /norestart'
|
ExecWait '"$INSTDIR\vc_redist.x86.exe" /passive /norestart'
|
||||||
DetailPrint "Sleep to ensure unlock of vc_redist file after installation..."
|
DetailPrint "Wait to ensure unlock of vc_redist file after installation..."
|
||||||
Sleep 3000
|
Sleep 3000
|
||||||
Delete "$INSTDIR\vcredist_x86.exe"
|
Delete "$INSTDIR\vc_redist.x86.exe"
|
||||||
PastVcRedist86Check:
|
PastVcRedist86Check:
|
||||||
|
|
||||||
IfFileExists "$INSTDIR\vcredist_x64.exe" VcRedist64Exists PastVcRedist64Check
|
IfFileExists "$INSTDIR\vc_redist.x64.exe" VcRedist64Exists PastVcRedist64Check
|
||||||
VcRedist64Exists:
|
VcRedist64Exists:
|
||||||
ExecWait '"$INSTDIR\vcredist_x64.exe" /passive /norestart'
|
ExecWait '"$INSTDIR\vc_redist.x64.exe" /passive /norestart'
|
||||||
DetailPrint "Sleep to ensure unlock of vc_redist file after installation..."
|
DetailPrint "Sleep to ensure unlock of vc_redist file after installation..."
|
||||||
Sleep 3000
|
Sleep 3000
|
||||||
Delete "$INSTDIR\vcredist_x64.exe"
|
Delete "$INSTDIR\vc_redist.x64.exe"
|
||||||
PastVcRedist64Check:
|
PastVcRedist64Check:
|
||||||
|
|
||||||
${Else}
|
${Else}
|
||||||
|
|||||||
@@ -1,21 +1,24 @@
|
|||||||
set(VERSION_STRING_CPP "${PROJECT_BINARY_DIR}/version_string.cpp")
|
set(VERSION_STRING_CPP "${PROJECT_BINARY_DIR}/version_string.cpp")
|
||||||
set(VERSION_STRING_H "${PROJECT_BINARY_DIR}/version_string.h")
|
set(VERSION_STRING_H "${PROJECT_BINARY_DIR}/version_string.h")
|
||||||
INCLUDE_DIRECTORIES(${PROJECT_BINARY_DIR})
|
include_directories(${PROJECT_BINARY_DIR})
|
||||||
|
|
||||||
set( hstring "extern const char *VERSION_STRING\;
|
set(hstring
|
||||||
|
"extern const char *VERSION_STRING\;
|
||||||
extern const char *VERSION_COMMIT\;
|
extern const char *VERSION_COMMIT\;
|
||||||
extern const char *VERSION_DATE\;\n" )
|
extern const char *VERSION_DATE\;\n"
|
||||||
set( cppstring "const char *VERSION_STRING = \"${PROJECT_VERSION_FRIENDLY}\"\;
|
)
|
||||||
|
set(cppstring
|
||||||
|
"const char *VERSION_STRING = \"${PROJECT_VERSION_FRIENDLY}\"\;
|
||||||
const char *VERSION_COMMIT = \"${GIT_COMMIT_ID}\"\;
|
const char *VERSION_COMMIT = \"${GIT_COMMIT_ID}\"\;
|
||||||
const char *VERSION_DATE = \"${GIT_COMMIT_DATE_FRIENDLY}\"\;\n")
|
const char *VERSION_DATE = \"${GIT_COMMIT_DATE_FRIENDLY}\"\;\n"
|
||||||
|
|
||||||
file(WRITE ${PROJECT_BINARY_DIR}/version_string.cpp.txt ${cppstring} )
|
|
||||||
file(WRITE ${PROJECT_BINARY_DIR}/version_string.h.txt ${hstring} )
|
|
||||||
|
|
||||||
execute_process(
|
|
||||||
COMMAND ${CMAKE_COMMAND} -E copy_if_different ${PROJECT_BINARY_DIR}/version_string.h.txt ${VERSION_STRING_H}
|
|
||||||
)
|
|
||||||
execute_process(
|
|
||||||
COMMAND ${CMAKE_COMMAND} -E copy_if_different ${PROJECT_BINARY_DIR}/version_string.cpp.txt ${VERSION_STRING_CPP}
|
|
||||||
)
|
)
|
||||||
|
|
||||||
|
file(WRITE ${PROJECT_BINARY_DIR}/version_string.cpp.txt ${cppstring})
|
||||||
|
file(WRITE ${PROJECT_BINARY_DIR}/version_string.h.txt ${hstring})
|
||||||
|
|
||||||
|
execute_process(
|
||||||
|
COMMAND ${CMAKE_COMMAND} -E copy_if_different ${PROJECT_BINARY_DIR}/version_string.h.txt ${VERSION_STRING_H}
|
||||||
|
)
|
||||||
|
execute_process(
|
||||||
|
COMMAND ${CMAKE_COMMAND} -E copy_if_different ${PROJECT_BINARY_DIR}/version_string.cpp.txt ${VERSION_STRING_CPP}
|
||||||
|
)
|
||||||
|
|||||||
BIN
cmake/dmgBackground.tif
Normal file
@@ -1,188 +1,187 @@
|
|||||||
# HELPER FUNCTIONS
|
# HELPER FUNCTIONS
|
||||||
|
|
||||||
function(get_commit_id)
|
function(get_commit_id)
|
||||||
# get last commit hash
|
# get last commit hash
|
||||||
execute_process(
|
execute_process(
|
||||||
COMMAND ${GIT_EXECUTABLE} log -1 --abbrev=7 --date=short "--pretty=%h"
|
COMMAND ${GIT_EXECUTABLE} log -1 --abbrev=7 --date=short "--pretty=%h"
|
||||||
WORKING_DIRECTORY ${PROJECT_SOURCE_DIR}
|
WORKING_DIRECTORY ${PROJECT_SOURCE_DIR}
|
||||||
RESULT_VARIABLE res_var
|
RESULT_VARIABLE res_var
|
||||||
OUTPUT_VARIABLE GIT_COM_ID
|
OUTPUT_VARIABLE GIT_COM_ID
|
||||||
)
|
)
|
||||||
if(NOT ${res_var} EQUAL 0)
|
if(NOT ${res_var} EQUAL 0)
|
||||||
message(WARNING "Git failed (not a repo, or no tags). Build will not contain git revision info.")
|
message(WARNING "Git failed (not a repo, or no tags). Build will not contain git revision info.")
|
||||||
return()
|
return()
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
string(REPLACE "\n" "" GIT_COM_ID "${GIT_COM_ID}")
|
string(REPLACE "\n" "" GIT_COM_ID "${GIT_COM_ID}")
|
||||||
set(GIT_COMMIT_ID "${GIT_COM_ID}" PARENT_SCOPE)
|
set(GIT_COMMIT_ID
|
||||||
set(PROJECT_VERSION_LABEL "custom(${GIT_COM_ID})" PARENT_SCOPE)
|
"${GIT_COM_ID}"
|
||||||
|
PARENT_SCOPE
|
||||||
|
)
|
||||||
|
set(PROJECT_VERSION_LABEL
|
||||||
|
"custom-${GIT_COM_ID}"
|
||||||
|
PARENT_SCOPE
|
||||||
|
)
|
||||||
endfunction()
|
endfunction()
|
||||||
|
|
||||||
function(get_commit_date)
|
function(get_commit_date)
|
||||||
# get last commit date
|
# get last commit date
|
||||||
execute_process(
|
execute_process(
|
||||||
COMMAND ${GIT_EXECUTABLE} log -1 --date=short "--pretty=%cd"
|
COMMAND ${GIT_EXECUTABLE} log -1 --date=short "--pretty=%cd"
|
||||||
WORKING_DIRECTORY ${PROJECT_SOURCE_DIR}
|
WORKING_DIRECTORY ${PROJECT_SOURCE_DIR}
|
||||||
RESULT_VARIABLE res_var
|
RESULT_VARIABLE res_var
|
||||||
OUTPUT_VARIABLE GIT_COM_DATE
|
OUTPUT_VARIABLE GIT_COM_DATE
|
||||||
)
|
)
|
||||||
if(NOT ${res_var} EQUAL 0)
|
if(NOT ${res_var} EQUAL 0)
|
||||||
message(WARNING "Git failed (not a repo, or no tags). Build will not contain git revision info.")
|
message(WARNING "Git failed (not a repo, or no tags). Build will not contain git revision info.")
|
||||||
return()
|
return()
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
string(REPLACE "\n" "" GIT_COM_DATE "${GIT_COM_DATE}")
|
string(REPLACE "\n" "" GIT_COM_DATE "${GIT_COM_DATE}")
|
||||||
set(GIT_COMMIT_DATE_FRIENDLY "${GIT_COM_DATE}" PARENT_SCOPE)
|
set(GIT_COMMIT_DATE_FRIENDLY
|
||||||
|
"${GIT_COM_DATE}"
|
||||||
|
PARENT_SCOPE
|
||||||
|
)
|
||||||
|
|
||||||
string(REPLACE "-" "" GIT_COM_DATE "${GIT_COM_DATE}")
|
string(REPLACE "-" "" GIT_COM_DATE "${GIT_COM_DATE}")
|
||||||
set(GIT_COMMIT_DATE "${GIT_COM_DATE}" PARENT_SCOPE)
|
set(GIT_COMMIT_DATE
|
||||||
endfunction()
|
"${GIT_COM_DATE}"
|
||||||
|
PARENT_SCOPE
|
||||||
function(clean_release_name name)
|
)
|
||||||
# "name": "Cockatrice: Thopter Pie Network, Revision 2"
|
|
||||||
|
|
||||||
# Remove all double quotes
|
|
||||||
STRING(REPLACE "\"" "" name "${name}")
|
|
||||||
# Remove json prefix "name: "
|
|
||||||
STRING(REPLACE " name: " "" name "${name}")
|
|
||||||
# Remove "cockatrice" name
|
|
||||||
STRING(REPLACE "Cockatrice" "" name "${name}")
|
|
||||||
# Remove all unwanted chars
|
|
||||||
STRING(REGEX REPLACE "[^A-Za-z0-9_ ]" "" name "${name}")
|
|
||||||
# Strip (trim) whitespaces
|
|
||||||
STRING(STRIP "${name}" name)
|
|
||||||
# Replace all spaces with underscores
|
|
||||||
STRING(REPLACE " " "_" name "${name}")
|
|
||||||
|
|
||||||
set(GIT_TAG_RELEASENAME "${name}" PARENT_SCOPE)
|
|
||||||
endfunction()
|
endfunction()
|
||||||
|
|
||||||
function(get_tag_name commit)
|
function(get_tag_name commit)
|
||||||
if(${commit} STREQUAL "unknown")
|
if(${commit} STREQUAL "unknown")
|
||||||
return()
|
return()
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
execute_process(
|
execute_process(
|
||||||
COMMAND ${GIT_EXECUTABLE} describe --exact-match --tags ${commit}
|
COMMAND ${GIT_EXECUTABLE} describe --exact-match --tags ${commit}
|
||||||
WORKING_DIRECTORY ${PROJECT_SOURCE_DIR}
|
WORKING_DIRECTORY ${PROJECT_SOURCE_DIR}
|
||||||
RESULT_VARIABLE res_var
|
RESULT_VARIABLE res_var
|
||||||
OUTPUT_VARIABLE GIT_TAG
|
OUTPUT_VARIABLE GIT_TAG
|
||||||
ERROR_VARIABLE GIT_TAG_ERR
|
ERROR_VARIABLE GIT_TAG_ERR
|
||||||
)
|
)
|
||||||
|
|
||||||
if((NOT ${res_var} EQUAL 0) OR (${GIT_TAG_ERR} MATCHES "fatal: no tag exactly matches.*"))
|
if((NOT ${res_var} EQUAL 0) OR (${GIT_TAG_ERR} MATCHES "fatal: no tag exactly matches.*"))
|
||||||
message(STATUS "Commit is not a release or prerelease (no git tag found)")
|
message(STATUS "Commit is not a release or prerelease (no git tag found)")
|
||||||
return()
|
return()
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
string(REPLACE "\n" "" GIT_TAG "${GIT_TAG}")
|
string(REPLACE "\n" "" GIT_TAG "${GIT_TAG}")
|
||||||
message(STATUS "Commit is a release or prerelease, git tag: ${GIT_TAG}")
|
message(STATUS "Commit is a release or prerelease, git tag: ${GIT_TAG}")
|
||||||
|
|
||||||
# Extract information from tag:
|
# Extract information from tag:
|
||||||
# YYYY-MM-DD-Release-MAJ.MIN.PATCH
|
# YYYY-MM-DD-Release-MAJ.MIN.PATCH
|
||||||
# YYYY-MM-DD-Development-MAJ.MIN.PATCH-beta.X
|
# YYYY-MM-DD-Development-MAJ.MIN.PATCH-beta.X
|
||||||
string(REPLACE "-" ";" GIT_TAG_EXPLODED "${GIT_TAG}")
|
string(REPLACE "-" ";" GIT_TAG_EXPLODED "${GIT_TAG}")
|
||||||
string(REPLACE "." ";" GIT_TAG_EXPLODED "${GIT_TAG_EXPLODED}")
|
string(REPLACE "." ";" GIT_TAG_EXPLODED "${GIT_TAG_EXPLODED}")
|
||||||
|
|
||||||
# Sanity checks: length
|
# Sanity checks: length
|
||||||
list(LENGTH GIT_TAG_EXPLODED GIT_TAG_LISTCOUNT)
|
list(LENGTH GIT_TAG_EXPLODED GIT_TAG_LISTCOUNT)
|
||||||
if(${GIT_TAG_LISTCOUNT} LESS 7 OR ${GIT_TAG_LISTCOUNT} GREATER 9)
|
if(${GIT_TAG_LISTCOUNT} LESS 7 OR ${GIT_TAG_LISTCOUNT} GREATER 9)
|
||||||
message(WARNING "Invalid tag format, got ${GIT_TAG_LISTCOUNT} tokens")
|
message(WARNING "Invalid tag format, got ${GIT_TAG_LISTCOUNT} tokens")
|
||||||
return()
|
return()
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
# Year
|
# Year
|
||||||
list(GET GIT_TAG_EXPLODED 0 GIT_TAG_YEAR)
|
list(GET GIT_TAG_EXPLODED 0 GIT_TAG_YEAR)
|
||||||
if(${GIT_TAG_YEAR} LESS 2017 OR ${GIT_TAG_LISTCOUNT} GREATER 2100)
|
if(${GIT_TAG_YEAR} LESS 2017 OR ${GIT_TAG_LISTCOUNT} GREATER 2100)
|
||||||
message(WARNING "Invalid tag year ${GIT_TAG_YEAR}")
|
message(WARNING "Invalid tag year ${GIT_TAG_YEAR}")
|
||||||
return()
|
return()
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
# Month
|
# Month
|
||||||
list(GET GIT_TAG_EXPLODED 1 GIT_TAG_MONTH)
|
list(GET GIT_TAG_EXPLODED 1 GIT_TAG_MONTH)
|
||||||
if(${GIT_TAG_MONTH} LESS 1 OR ${GIT_TAG_MONTH} GREATER 12)
|
if(${GIT_TAG_MONTH} LESS 1 OR ${GIT_TAG_MONTH} GREATER 12)
|
||||||
message(WARNING "Invalid tag month ${GIT_TAG_MONTH}")
|
message(WARNING "Invalid tag month ${GIT_TAG_MONTH}")
|
||||||
return()
|
return()
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
# Day
|
# Day
|
||||||
list(GET GIT_TAG_EXPLODED 2 GIT_TAG_DAY)
|
list(GET GIT_TAG_EXPLODED 2 GIT_TAG_DAY)
|
||||||
if(${GIT_TAG_DAY} LESS 1 OR ${GIT_TAG_DAY} GREATER 31)
|
if(${GIT_TAG_DAY} LESS 1 OR ${GIT_TAG_DAY} GREATER 31)
|
||||||
message(WARNING "Invalid tag day ${GIT_TAG_DAY}")
|
message(WARNING "Invalid tag day ${GIT_TAG_DAY}")
|
||||||
return()
|
return()
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
# Type
|
# Type
|
||||||
list(GET GIT_TAG_EXPLODED 3 GIT_TAG_TYPE)
|
list(GET GIT_TAG_EXPLODED 3 GIT_TAG_TYPE)
|
||||||
if(NOT(${GIT_TAG_TYPE} STREQUAL "Release" OR ${GIT_TAG_TYPE} STREQUAL "Development"))
|
if(NOT (${GIT_TAG_TYPE} STREQUAL "Release" OR ${GIT_TAG_TYPE} STREQUAL "Development"))
|
||||||
message(WARNING "Invalid tag type ${GIT_TAG_TYPE}")
|
message(WARNING "Invalid tag type ${GIT_TAG_TYPE}")
|
||||||
return()
|
return()
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
# Major
|
# Major
|
||||||
list(GET GIT_TAG_EXPLODED 4 GIT_TAG_MAJOR)
|
list(GET GIT_TAG_EXPLODED 4 GIT_TAG_MAJOR)
|
||||||
if(${GIT_TAG_MAJOR} LESS 0 OR ${GIT_TAG_MAJOR} GREATER 99)
|
if(${GIT_TAG_MAJOR} LESS 0 OR ${GIT_TAG_MAJOR} GREATER 99)
|
||||||
message(WARNING "Invalid tag major version ${GIT_TAG_MAJOR}")
|
message(WARNING "Invalid tag major version ${GIT_TAG_MAJOR}")
|
||||||
return()
|
return()
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
# Minor
|
# Minor
|
||||||
list(GET GIT_TAG_EXPLODED 5 GIT_TAG_MINOR)
|
list(GET GIT_TAG_EXPLODED 5 GIT_TAG_MINOR)
|
||||||
if(${GIT_TAG_MINOR} LESS 0 OR ${GIT_TAG_MINOR} GREATER 99)
|
if(${GIT_TAG_MINOR} LESS 0 OR ${GIT_TAG_MINOR} GREATER 99)
|
||||||
message(WARNING "Invalid tag minor version ${GIT_TAG_MINOR}")
|
message(WARNING "Invalid tag minor version ${GIT_TAG_MINOR}")
|
||||||
return()
|
return()
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
# Patch
|
# Patch
|
||||||
list(GET GIT_TAG_EXPLODED 6 GIT_TAG_PATCH)
|
list(GET GIT_TAG_EXPLODED 6 GIT_TAG_PATCH)
|
||||||
if(${GIT_TAG_PATCH} LESS 0 OR ${GIT_TAG_PATCH} GREATER 99)
|
if(${GIT_TAG_PATCH} LESS 0 OR ${GIT_TAG_PATCH} GREATER 99)
|
||||||
message(WARNING "Invalid tag patch version ${GIT_TAG_PATCH}")
|
message(WARNING "Invalid tag patch version ${GIT_TAG_PATCH}")
|
||||||
return()
|
return()
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
# Label
|
# Label
|
||||||
# 7 = Stable release
|
# 7 = Stable release
|
||||||
# 8 = Dev release, first beta so only "beta" attached
|
# 8 = Dev release, first beta so only "beta" attached
|
||||||
# 9 = Dev release, subsequent beta so "beta.N" attached (N>=2)
|
# 9 = Dev release, subsequent beta so "beta.N" attached (N>=2)
|
||||||
if(${GIT_TAG_LISTCOUNT} EQUAL 8)
|
if(${GIT_TAG_LISTCOUNT} EQUAL 8)
|
||||||
list(GET GIT_TAG_EXPLODED 7 GIT_TAG_LABEL)
|
list(GET GIT_TAG_EXPLODED 7 GIT_TAG_LABEL)
|
||||||
elseif(${GIT_TAG_LISTCOUNT} EQUAL 9)
|
elseif(${GIT_TAG_LISTCOUNT} EQUAL 9)
|
||||||
list(GET GIT_TAG_EXPLODED 7 GIT_TAG_LABEL)
|
list(GET GIT_TAG_EXPLODED 7 GIT_TAG_LABEL)
|
||||||
list(GET GIT_TAG_EXPLODED 8 GIT_TAG_LABEL_NUM)
|
list(GET GIT_TAG_EXPLODED 8 GIT_TAG_LABEL_NUM)
|
||||||
set(GIT_TAG_LABEL ${GIT_TAG_LABEL} ${GIT_TAG_LABEL_NUM})
|
set(GIT_TAG_LABEL ${GIT_TAG_LABEL} ${GIT_TAG_LABEL_NUM})
|
||||||
string(REPLACE ";" "." GIT_TAG_LABEL "${GIT_TAG_LABEL}")
|
string(REPLACE ";" "." GIT_TAG_LABEL "${GIT_TAG_LABEL}")
|
||||||
else()
|
else()
|
||||||
SET(GIT_TAG_LABEL "")
|
set(GIT_TAG_LABEL "")
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
# Override hardcoded version with the informations from the tag
|
# Override hardcoded version with the informations from the tag
|
||||||
set(PROJECT_VERSION_MAJOR ${GIT_TAG_MAJOR} PARENT_SCOPE)
|
set(PROJECT_VERSION_MAJOR
|
||||||
set(PROJECT_VERSION_MINOR ${GIT_TAG_MINOR} PARENT_SCOPE)
|
${GIT_TAG_MAJOR}
|
||||||
set(PROJECT_VERSION_PATCH ${GIT_TAG_PATCH} PARENT_SCOPE)
|
PARENT_SCOPE
|
||||||
set(PROJECT_VERSION_LABEL ${GIT_TAG_LABEL} PARENT_SCOPE)
|
)
|
||||||
|
set(PROJECT_VERSION_MINOR
|
||||||
|
${GIT_TAG_MINOR}
|
||||||
|
PARENT_SCOPE
|
||||||
|
)
|
||||||
|
set(PROJECT_VERSION_PATCH
|
||||||
|
${GIT_TAG_PATCH}
|
||||||
|
PARENT_SCOPE
|
||||||
|
)
|
||||||
|
set(PROJECT_VERSION_LABEL
|
||||||
|
${GIT_TAG_LABEL}
|
||||||
|
PARENT_SCOPE
|
||||||
|
)
|
||||||
|
|
||||||
if(${GIT_TAG_TYPE} STREQUAL "Development")
|
if(${GIT_TAG_TYPE} STREQUAL "Development")
|
||||||
set(PROJECT_VERSION_LABEL ${GIT_TAG_LABEL} PARENT_SCOPE)
|
set(PROJECT_VERSION_LABEL
|
||||||
elseif(${GIT_TAG_TYPE} STREQUAL "Release")
|
${GIT_TAG_LABEL}
|
||||||
set(PROJECT_VERSION_LABEL "" PARENT_SCOPE)
|
PARENT_SCOPE
|
||||||
|
)
|
||||||
# get version name from github
|
elseif(${GIT_TAG_TYPE} STREQUAL "Release")
|
||||||
set(GIT_TAG_TEMP_FILE "${PROJECT_BINARY_DIR}/tag_informations.txt")
|
set(PROJECT_VERSION_LABEL
|
||||||
set(GIT_TAG_TEMP_URL "https://api.github.com/repos/Cockatrice/Cockatrice/releases/tags/${GIT_TAG}")
|
""
|
||||||
message(STATUS "Fetching tag informations from ${GIT_TAG_TEMP_URL}")
|
PARENT_SCOPE
|
||||||
file(REMOVE "${GIT_TAG_TEMP_FILE}")
|
)
|
||||||
file(DOWNLOAD "${GIT_TAG_TEMP_URL}" "${GIT_TAG_TEMP_FILE}" STATUS status LOG log INACTIVITY_TIMEOUT 30 TIMEOUT 300 SHOW_PROGRESS)
|
# set release name from env var
|
||||||
list(GET status 0 err)
|
set(PROJECT_VERSION_RELEASENAME
|
||||||
list(GET status 1 msg)
|
"${GIT_TAG_RELEASENAME}"
|
||||||
if(err)
|
PARENT_SCOPE
|
||||||
message(WARNING "Download failed with error ${msg}: ${log}")
|
)
|
||||||
return()
|
endif()
|
||||||
endif()
|
|
||||||
file(STRINGS "${GIT_TAG_TEMP_FILE}" GIT_TAG_RAW_RELEASENAME REGEX "\"name\": \"" LIMIT_COUNT 1)
|
|
||||||
|
|
||||||
clean_release_name("${GIT_TAG_RAW_RELEASENAME}")
|
|
||||||
set(PROJECT_VERSION_RELEASENAME "${GIT_TAG_RELEASENAME}" PARENT_SCOPE)
|
|
||||||
endif()
|
|
||||||
|
|
||||||
endfunction()
|
endfunction()
|
||||||
|
|
||||||
@@ -190,23 +189,23 @@ endfunction()
|
|||||||
|
|
||||||
# fallback defaults
|
# fallback defaults
|
||||||
set(GIT_COMMIT_ID "unknown")
|
set(GIT_COMMIT_ID "unknown")
|
||||||
set(GIT_COMMIT_DATE "unknown")
|
set(GIT_COMMIT_DATE "")
|
||||||
set(GIT_COMMIT_DATE_FRIENDLY "unknown")
|
set(GIT_COMMIT_DATE_FRIENDLY "")
|
||||||
set(PROJECT_VERSION_LABEL "custom(unknown)")
|
set(PROJECT_VERSION_LABEL "")
|
||||||
set(PROJECT_VERSION_RELEASENAME "")
|
set(PROJECT_VERSION_RELEASENAME "")
|
||||||
|
|
||||||
find_package(Git)
|
find_package(Git)
|
||||||
if(GIT_FOUND)
|
if(GIT_FOUND)
|
||||||
get_commit_id()
|
get_commit_id()
|
||||||
get_commit_date()
|
get_commit_date()
|
||||||
get_tag_name(${GIT_COMMIT_ID})
|
get_tag_name(${GIT_COMMIT_ID})
|
||||||
else()
|
else()
|
||||||
message( WARNING "Git not found. Build will not contain git revision info." )
|
message(WARNING "Git not found. Build will not contain git revision info.")
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
set(PROJECT_VERSION "${PROJECT_VERSION_MAJOR}.${PROJECT_VERSION_MINOR}.${PROJECT_VERSION_PATCH}")
|
set(PROJECT_VERSION "${PROJECT_VERSION_MAJOR}.${PROJECT_VERSION_MINOR}.${PROJECT_VERSION_PATCH}")
|
||||||
if(PROJECT_VERSION_LABEL)
|
if(PROJECT_VERSION_LABEL)
|
||||||
set(PROJECT_VERSION "${PROJECT_VERSION}-${PROJECT_VERSION_LABEL}")
|
set(PROJECT_VERSION "${PROJECT_VERSION}-${PROJECT_VERSION_LABEL}")
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
set(PROJECT_VERSION_FRIENDLY "${PROJECT_VERSION} (${GIT_COMMIT_DATE_FRIENDLY})")
|
set(PROJECT_VERSION_FRIENDLY "${PROJECT_VERSION} (${GIT_COMMIT_DATE_FRIENDLY})")
|
||||||
@@ -214,7 +213,7 @@ 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}")
|
set(PROJECT_VERSION_FILENAME "${PROJECT_VERSION_FILENAME}-${PROJECT_VERSION_RELEASENAME}")
|
||||||
endif()
|
endif()
|
||||||
set(PROJECT_VERSION_FILENAME "${PROJECT_VERSION_FILENAME}-${PROJECT_VERSION}")
|
set(PROJECT_VERSION_FILENAME "${PROJECT_VERSION_FILENAME}-${PROJECT_VERSION}")
|
||||||
|
|
||||||
|
|||||||
@@ -4,12 +4,12 @@ project(gtest-download LANGUAGES NONE)
|
|||||||
|
|
||||||
include(ExternalProject)
|
include(ExternalProject)
|
||||||
ExternalProject_Add(googletest
|
ExternalProject_Add(googletest
|
||||||
URL https://github.com/google/googletest/archive/release-1.7.0.zip
|
URL https://github.com/google/googletest/archive/release-1.11.0.zip
|
||||||
URL_HASH SHA1=f89bc9f55477df2fde082481e2d709bfafdb057b
|
URL_HASH SHA1=9ffb7b5923f4a8fcdabf2f42c6540cce299f44c0
|
||||||
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 ""
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -2,282 +2,413 @@
|
|||||||
#
|
#
|
||||||
# provides the cockatrice binary
|
# provides the cockatrice binary
|
||||||
|
|
||||||
PROJECT(Cockatrice VERSION "${PROJECT_VERSION_MAJOR}.${PROJECT_VERSION_MINOR}.${PROJECT_VERSION_PATCH}")
|
project(Cockatrice VERSION "${PROJECT_VERSION_MAJOR}.${PROJECT_VERSION_MINOR}.${PROJECT_VERSION_PATCH}")
|
||||||
|
|
||||||
SET(cockatrice_SOURCES
|
set(cockatrice_SOURCES
|
||||||
src/abstractcounter.cpp
|
src/game/cards/abstract_card_drag_item.cpp
|
||||||
src/counter_general.cpp
|
src/game/cards/abstract_card_item.cpp
|
||||||
src/dlg_creategame.cpp
|
src/client/game_logic/abstract_client.cpp
|
||||||
src/dlg_filter_games.cpp
|
src/game/board/abstract_counter.cpp
|
||||||
src/dlg_connect.cpp
|
src/game/board/abstract_graphics_item.cpp
|
||||||
src/dlg_create_token.cpp
|
src/game/board/arrow_item.cpp
|
||||||
src/dlg_edit_avatar.cpp
|
src/game/board/arrow_target.cpp
|
||||||
src/dlg_edit_password.cpp
|
src/game/cards/card_database.cpp
|
||||||
src/dlg_edit_tokens.cpp
|
src/game/cards/card_database_manager.cpp
|
||||||
src/dlg_edit_user.cpp
|
src/game/cards/card_database_model.cpp
|
||||||
src/dlg_forgotpasswordrequest.cpp
|
src/game/cards/card_database_parser/card_database_parser.cpp
|
||||||
src/dlg_forgotpasswordreset.cpp
|
src/game/cards/card_database_parser/cockatrice_xml_3.cpp
|
||||||
src/dlg_forgotpasswordchallenge.cpp
|
src/game/cards/card_database_parser/cockatrice_xml_4.cpp
|
||||||
src/dlg_register.cpp
|
src/game/cards/card_drag_item.cpp
|
||||||
src/dlg_tip_of_the_day.cpp
|
src/game/filters/filter_card.cpp
|
||||||
src/tip_of_the_day.cpp
|
src/client/ui/widgets/cards/card_info_frame_widget.cpp
|
||||||
src/dlg_update.cpp
|
src/client/ui/widgets/cards/card_info_picture_widget.cpp
|
||||||
src/dlg_viewlog.cpp
|
src/client/ui/widgets/cards/card_info_text_widget.cpp
|
||||||
src/abstractclient.cpp
|
src/client/ui/widgets/cards/card_info_display_widget.cpp
|
||||||
src/remoteclient.cpp
|
src/client/ui/widgets/cards/card_size_widget.cpp
|
||||||
|
src/game/cards/card_item.cpp
|
||||||
|
src/game/cards/card_list.cpp
|
||||||
|
src/game/zones/card_zone.cpp
|
||||||
|
src/server/chat_view/chat_view.cpp
|
||||||
|
src/game/board/counter_general.cpp
|
||||||
|
src/deck/custom_line_edit.cpp
|
||||||
|
src/deck/deck_loader.cpp
|
||||||
|
src/deck/deck_list_model.cpp
|
||||||
|
src/deck/deck_stats_interface.cpp
|
||||||
|
src/deck/deck_view.cpp
|
||||||
|
src/dialogs/dlg_connect.cpp
|
||||||
|
src/dialogs/dlg_create_token.cpp
|
||||||
|
src/dialogs/dlg_create_game.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_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/filters/filter_string.cpp
|
||||||
|
src/game/filters/filter_builder.cpp
|
||||||
|
src/game/filters/filter_tree.cpp
|
||||||
|
src/game/filters/filter_tree_model.cpp
|
||||||
|
src/client/ui/layouts/flow_layout.cpp
|
||||||
|
src/client/ui/layouts/horizontal_flow_layout.cpp
|
||||||
|
src/client/ui/layouts/vertical_flow_layout.cpp
|
||||||
|
src/client/ui/widgets/general/layout_containers/flow_widget.cpp
|
||||||
|
src/game/game_scene.cpp
|
||||||
|
src/game/game_selector.cpp
|
||||||
|
src/game/games_model.cpp
|
||||||
|
src/game/game_view.cpp
|
||||||
|
src/client/get_text_with_max.cpp
|
||||||
|
src/game/hand_counter.cpp
|
||||||
|
src/server/handle_public_servers.cpp
|
||||||
|
src/game/zones/hand_zone.cpp
|
||||||
|
src/client/game_logic/key_signals.cpp
|
||||||
|
src/client/ui/line_edit_completer.cpp
|
||||||
|
src/server/local_client.cpp
|
||||||
|
src/server/local_server.cpp
|
||||||
|
src/server/local_server_interface.cpp
|
||||||
|
src/utility/logger.cpp
|
||||||
|
src/client/ui/widgets/cards/card_info_picture_enlarged_widget.cpp
|
||||||
|
src/client/ui/widgets/cards/card_info_picture_with_text_overlay_widget.cpp
|
||||||
|
src/client/ui/widgets/general/display/labeled_input.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/shadow_background_label.cpp
|
||||||
src/main.cpp
|
src/main.cpp
|
||||||
src/window_main.cpp
|
src/server/message_log_widget.cpp
|
||||||
src/gamesmodel.cpp
|
src/client/ui/layouts/overlap_layout.cpp
|
||||||
src/player.cpp
|
src/client/ui/widgets/general/layout_containers/overlap_widget.cpp
|
||||||
src/playertarget.cpp
|
src/client/ui/widgets/general/layout_containers/overlap_control_widget.cpp
|
||||||
src/cardzone.cpp
|
src/server/pending_command.cpp
|
||||||
src/selectzone.cpp
|
src/game/phase.cpp
|
||||||
src/cardlist.cpp
|
src/client/ui/phases_toolbar.cpp
|
||||||
src/abstractcarditem.cpp
|
src/client/ui/picture_loader.cpp
|
||||||
src/carditem.cpp
|
src/game/zones/pile_zone.cpp
|
||||||
src/tablezone.cpp
|
src/client/ui/pixel_map_generator.cpp
|
||||||
src/handzone.cpp
|
src/game/player/player.cpp
|
||||||
src/handcounter.cpp
|
src/game/player/player_list_widget.cpp
|
||||||
src/carddatabase.cpp
|
src/game/player/player_target.cpp
|
||||||
src/keysignals.cpp
|
src/client/ui/widgets/printing_selector/all_zones_card_amount_widget.cpp
|
||||||
src/gameview.cpp
|
src/client/ui/widgets/printing_selector/card_amount_widget.cpp
|
||||||
src/gameselector.cpp
|
src/client/ui/widgets/printing_selector/printing_selector.cpp
|
||||||
src/decklistmodel.cpp
|
src/client/ui/widgets/printing_selector/printing_selector_card_display_widget.cpp
|
||||||
src/deck_loader.cpp
|
src/client/ui/widgets/printing_selector/printing_selector_card_overlay_widget.cpp
|
||||||
src/dlg_load_deck_from_clipboard.cpp
|
src/client/ui/widgets/printing_selector/printing_selector_card_search_widget.cpp
|
||||||
src/dlg_load_remote_deck.cpp
|
src/client/ui/widgets/printing_selector/printing_selector_card_selection_widget.cpp
|
||||||
src/cardinfowidget.cpp
|
src/client/ui/widgets/printing_selector/printing_selector_card_sorting_widget.cpp
|
||||||
src/cardframe.cpp
|
src/client/ui/widgets/printing_selector/printing_selector_view_options_toolbar_widget.cpp
|
||||||
src/cardinfopicture.cpp
|
src/client/ui/widgets/printing_selector/printing_selector_view_options_widget.cpp
|
||||||
src/cardinfotext.cpp
|
src/client/ui/widgets/printing_selector/set_name_and_collectors_number_display_widget.cpp
|
||||||
src/filterbuilder.cpp
|
src/client/network/release_channel.cpp
|
||||||
src/cardfilter.cpp
|
src/server/remote/remote_client.cpp
|
||||||
src/filtertreemodel.cpp
|
src/server/remote/remote_decklist_tree_widget.cpp
|
||||||
src/filtertree.cpp
|
src/server/remote/remote_replay_list_tree_widget.cpp
|
||||||
src/messagelogwidget.cpp
|
src/client/network/replay_timeline_widget.cpp
|
||||||
src/zoneviewzone.cpp
|
src/game/zones/select_zone.cpp
|
||||||
src/zoneviewwidget.cpp
|
src/utility/sequence_edit.cpp
|
||||||
src/pilezone.cpp
|
src/client/network/sets_model.cpp
|
||||||
src/stackzone.cpp
|
src/settings/card_database_settings.cpp
|
||||||
src/carddragitem.cpp
|
src/settings/download_settings.cpp
|
||||||
src/carddatabasemodel.cpp
|
src/settings/game_filters_settings.cpp
|
||||||
src/setsmodel.cpp
|
src/settings/layouts_settings.cpp
|
||||||
src/window_sets.cpp
|
src/settings/message_settings.cpp
|
||||||
src/abstractgraphicsitem.cpp
|
src/settings/servers_settings.cpp
|
||||||
src/abstractcarddragitem.cpp
|
src/settings/settings_manager.cpp
|
||||||
src/dlg_settings.cpp
|
src/settings/cache_settings.cpp
|
||||||
src/phasestoolbar.cpp
|
src/settings/shortcuts_settings.cpp
|
||||||
src/gamescene.cpp
|
src/client/sound_engine.cpp
|
||||||
src/arrowitem.cpp
|
src/client/network/spoiler_background_updater.cpp
|
||||||
src/arrowtarget.cpp
|
src/game/zones/stack_zone.cpp
|
||||||
src/tab.cpp
|
src/client/tabs/tab.cpp
|
||||||
src/tab_server.cpp
|
src/client/tabs/tab_account.cpp
|
||||||
src/tab_room.cpp
|
src/client/tabs/tab_admin.cpp
|
||||||
src/tab_message.cpp
|
src/client/tabs/tab_deck_editor.cpp
|
||||||
src/tab_game.cpp
|
src/client/tabs/tab_deck_storage.cpp
|
||||||
src/tab_deck_storage.cpp
|
src/client/tabs/tab_game.cpp
|
||||||
src/tab_replays.cpp
|
src/client/tabs/tab_logs.cpp
|
||||||
src/tab_supervisor.cpp
|
src/client/tabs/tab_message.cpp
|
||||||
src/tab_admin.cpp
|
src/client/tabs/tab_replays.cpp
|
||||||
src/tab_userlists.cpp
|
src/client/tabs/tab_room.cpp
|
||||||
src/tab_deck_editor.cpp
|
src/client/tabs/tab_server.cpp
|
||||||
src/tab_logs.cpp
|
src/client/tabs/tab_supervisor.cpp
|
||||||
src/replay_timeline_widget.cpp
|
src/game/zones/table_zone.cpp
|
||||||
src/deckstats_interface.cpp
|
src/client/tapped_out_interface.cpp
|
||||||
src/tappedout_interface.cpp
|
src/client/ui/theme_manager.cpp
|
||||||
src/chatview/chatview.cpp
|
src/client/ui/tip_of_the_day.cpp
|
||||||
src/chatview/userlistProxy.h
|
src/client/translate_counter_name.cpp
|
||||||
src/userlist.cpp
|
src/client/update_downloader.cpp
|
||||||
src/userinfobox.cpp
|
src/server/user/user_context_menu.cpp
|
||||||
src/user_context_menu.cpp
|
src/server/user/user_info_connection.cpp
|
||||||
src/remotedecklist_treewidget.cpp
|
src/server/user/user_info_box.cpp
|
||||||
src/remotereplaylist_treewidget.cpp
|
src/server/user/user_list.cpp
|
||||||
src/deckview.cpp
|
src/client/ui/window_main.cpp
|
||||||
src/playerlistwidget.cpp
|
src/game/zones/view_zone_widget.cpp
|
||||||
src/pixmapgenerator.cpp
|
src/game/zones/view_zone.cpp
|
||||||
src/settingscache.cpp
|
|
||||||
src/thememanager.cpp
|
|
||||||
src/localserver.cpp
|
|
||||||
src/localserverinterface.cpp
|
|
||||||
src/localclient.cpp
|
|
||||||
src/soundengine.cpp
|
|
||||||
src/pending_command.cpp
|
|
||||||
src/pictureloader.cpp
|
|
||||||
src/shortcutssettings.cpp
|
|
||||||
src/sequenceEdit/sequenceedit.cpp
|
|
||||||
src/sequenceEdit/shortcutstab.cpp
|
|
||||||
src/lineeditcompleter.cpp
|
|
||||||
src/settings/settingsmanager.cpp
|
|
||||||
src/settings/carddatabasesettings.cpp
|
|
||||||
src/settings/serverssettings.cpp
|
|
||||||
src/settings/messagesettings.cpp
|
|
||||||
src/settings/gamefilterssettings.cpp
|
|
||||||
src/settings/layoutssettings.cpp
|
|
||||||
src/settings/downloadsettings.cpp
|
|
||||||
src/update_downloader.cpp
|
|
||||||
src/logger.cpp
|
|
||||||
src/releasechannel.cpp
|
|
||||||
src/userconnection_information.cpp
|
|
||||||
src/spoilerbackgroundupdater.cpp
|
|
||||||
src/handle_public_servers.cpp
|
|
||||||
src/carddbparser/carddatabaseparser.cpp
|
|
||||||
src/carddbparser/cockatricexml3.cpp
|
|
||||||
src/carddbparser/cockatricexml4.cpp
|
|
||||||
${VERSION_STRING_CPP}
|
${VERSION_STRING_CPP}
|
||||||
)
|
)
|
||||||
|
|
||||||
add_subdirectory(sounds)
|
add_subdirectory(sounds)
|
||||||
add_subdirectory(themes)
|
add_subdirectory(themes)
|
||||||
|
|
||||||
set(cockatrice_RESOURCES cockatrice.qrc)
|
set(cockatrice_RESOURCES cockatrice.qrc)
|
||||||
|
|
||||||
IF(UPDATE_TRANSLATIONS)
|
if(UPDATE_TRANSLATIONS)
|
||||||
FILE(GLOB_RECURSE translate_cockatrice_SRCS ${CMAKE_SOURCE_DIR}/cockatrice/src/*.cpp ${CMAKE_SOURCE_DIR}/cockatrice/src/*.h)
|
file(GLOB_RECURSE translate_cockatrice_SRCS ${CMAKE_SOURCE_DIR}/cockatrice/src/*.cpp
|
||||||
FILE(GLOB_RECURSE translate_common_SRCS ${CMAKE_SOURCE_DIR}/common/*.cpp ${CMAKE_SOURCE_DIR}/common/*.h)
|
${CMAKE_SOURCE_DIR}/cockatrice/src/*.h
|
||||||
SET(translate_SRCS ${translate_cockatrice_SRCS} ${translate_common_SRCS})
|
)
|
||||||
SET(cockatrice_TS "${CMAKE_CURRENT_SOURCE_DIR}/translations/cockatrice_en.ts")
|
file(GLOB_RECURSE translate_common_SRCS ${CMAKE_SOURCE_DIR}/common/*.cpp ${CMAKE_SOURCE_DIR}/common/*.h)
|
||||||
ELSE()
|
set(translate_SRCS ${translate_cockatrice_SRCS} ${translate_common_SRCS})
|
||||||
FILE(GLOB cockatrice_TS "${CMAKE_CURRENT_SOURCE_DIR}/translations/*.ts")
|
set(cockatrice_TS "${CMAKE_CURRENT_SOURCE_DIR}/cockatrice_en@source.ts")
|
||||||
ENDIF(UPDATE_TRANSLATIONS)
|
else()
|
||||||
|
file(GLOB cockatrice_TS "${CMAKE_CURRENT_SOURCE_DIR}/translations/*.ts")
|
||||||
|
endif(UPDATE_TRANSLATIONS)
|
||||||
|
|
||||||
if(WIN32)
|
if(WIN32)
|
||||||
set(cockatrice_SOURCES ${cockatrice_SOURCES} cockatrice.rc)
|
set(cockatrice_SOURCES ${cockatrice_SOURCES} cockatrice.rc)
|
||||||
endif(WIN32)
|
endif(WIN32)
|
||||||
|
|
||||||
if(APPLE)
|
if(APPLE)
|
||||||
set(MACOSX_BUNDLE_ICON_FILE appicon.icns)
|
set(MACOSX_BUNDLE_ICON_FILE appicon.icns)
|
||||||
set_source_files_properties(${CMAKE_CURRENT_SOURCE_DIR}/resources/appicon.icns PROPERTIES MACOSX_PACKAGE_LOCATION Resources)
|
set_source_files_properties(
|
||||||
set(cockatrice_SOURCES ${cockatrice_SOURCES} ${CMAKE_CURRENT_SOURCE_DIR}/resources/appicon.icns)
|
${CMAKE_CURRENT_SOURCE_DIR}/resources/appicon.icns PROPERTIES MACOSX_PACKAGE_LOCATION Resources
|
||||||
ENDIF(APPLE)
|
)
|
||||||
|
set(cockatrice_SOURCES ${cockatrice_SOURCES} ${CMAKE_CURRENT_SOURCE_DIR}/resources/appicon.icns)
|
||||||
|
endif(APPLE)
|
||||||
|
|
||||||
# Qt5
|
if(Qt6_FOUND)
|
||||||
find_package(Qt5 COMPONENTS Concurrent Multimedia Network PrintSupport Svg WebSockets Widgets REQUIRED)
|
qt6_add_resources(cockatrice_RESOURCES_RCC ${cockatrice_RESOURCES})
|
||||||
set(COCKATRICE_QT_MODULES Qt5::Concurrent Qt5::Multimedia Qt5::Network Qt5::PrintSupport Qt5::Svg Qt5::Widgets Qt5::WebSockets)
|
elseif(Qt5_FOUND)
|
||||||
|
qt5_add_resources(cockatrice_RESOURCES_RCC ${cockatrice_RESOURCES})
|
||||||
# Qt5LinguistTools
|
|
||||||
find_package(Qt5LinguistTools)
|
|
||||||
if(Qt5LinguistTools_FOUND)
|
|
||||||
list(APPEND COCKATRICE_LIBS Qt5::LinguistTools)
|
|
||||||
|
|
||||||
if(NOT Qt5_LRELEASE_EXECUTABLE)
|
|
||||||
MESSAGE(WARNING "Qt's lrelease not found.")
|
|
||||||
endif()
|
|
||||||
|
|
||||||
if(UPDATE_TRANSLATIONS)
|
|
||||||
if(NOT Qt5_LUPDATE_EXECUTABLE)
|
|
||||||
MESSAGE(WARNING "Qt's lupdate not found.")
|
|
||||||
endif()
|
|
||||||
QT5_CREATE_TRANSLATION(cockatrice_QM ${translate_SRCS} ${cockatrice_TS})
|
|
||||||
else()
|
|
||||||
QT5_ADD_TRANSLATION(cockatrice_QM ${cockatrice_TS})
|
|
||||||
endif()
|
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
QT5_ADD_RESOURCES(cockatrice_RESOURCES_RCC ${cockatrice_RESOURCES})
|
|
||||||
|
|
||||||
# Declare path variables
|
# Declare path variables
|
||||||
set(ICONDIR share/icons CACHE STRING "icon dir")
|
set(ICONDIR
|
||||||
set(DESKTOPDIR share/applications CACHE STRING "desktop file destination")
|
share/icons
|
||||||
|
CACHE STRING "icon dir"
|
||||||
|
)
|
||||||
|
set(DESKTOPDIR
|
||||||
|
share/applications
|
||||||
|
CACHE STRING "desktop file destination"
|
||||||
|
)
|
||||||
|
|
||||||
# Include directories
|
# Include directories
|
||||||
INCLUDE_DIRECTORIES(../common)
|
include_directories(../common)
|
||||||
INCLUDE_DIRECTORIES(${PROTOBUF_INCLUDE_DIR})
|
include_directories(${PROTOBUF_INCLUDE_DIR})
|
||||||
INCLUDE_DIRECTORIES(${CMAKE_BINARY_DIR}/common)
|
include_directories(${CMAKE_BINARY_DIR}/common)
|
||||||
INCLUDE_DIRECTORIES(${CMAKE_CURRENT_BINARY_DIR})
|
include_directories(${CMAKE_CURRENT_BINARY_DIR})
|
||||||
|
|
||||||
# Build cockatrice binary and link it
|
set(COCKATRICE_MAC_QM_INSTALL_DIR "cockatrice.app/Contents/Resources/translations")
|
||||||
ADD_EXECUTABLE(cockatrice WIN32 MACOSX_BUNDLE ${cockatrice_SOURCES} ${cockatrice_QM} ${cockatrice_RESOURCES_RCC} ${cockatrice_MOC_SRCS})
|
set(COCKATRICE_UNIX_QM_INSTALL_DIR "share/cockatrice/translations")
|
||||||
|
set(COCKATRICE_WIN32_QM_INSTALL_DIR "translations")
|
||||||
|
|
||||||
TARGET_LINK_LIBRARIES(cockatrice cockatrice_common ${COCKATRICE_QT_MODULES})
|
if(Qt6_FOUND)
|
||||||
|
qt6_add_executable(
|
||||||
|
cockatrice
|
||||||
|
WIN32
|
||||||
|
MACOSX_BUNDLE
|
||||||
|
${cockatrice_SOURCES}
|
||||||
|
${cockatrice_RESOURCES_RCC}
|
||||||
|
${cockatrice_MOC_SRCS}
|
||||||
|
MANUAL_FINALIZATION
|
||||||
|
)
|
||||||
|
elseif(Qt5_FOUND)
|
||||||
|
# Qt5 Translations need to be linked at executable creation time
|
||||||
|
if(Qt5LinguistTools_FOUND)
|
||||||
|
if(UPDATE_TRANSLATIONS)
|
||||||
|
qt5_create_translation(cockatrice_QM ${translate_SRCS} ${cockatrice_TS})
|
||||||
|
else()
|
||||||
|
qt5_add_translation(cockatrice_QM ${cockatrice_TS})
|
||||||
|
endif()
|
||||||
|
endif()
|
||||||
|
add_executable(
|
||||||
|
cockatrice WIN32 MACOSX_BUNDLE ${cockatrice_MOC_SRCS} ${cockatrice_QM} ${cockatrice_RESOURCES_RCC}
|
||||||
|
${cockatrice_SOURCES}
|
||||||
|
)
|
||||||
|
if(UNIX)
|
||||||
|
if(APPLE)
|
||||||
|
install(FILES ${cockatrice_QM} DESTINATION ${COCKATRICE_MAC_QM_INSTALL_DIR})
|
||||||
|
else()
|
||||||
|
install(FILES ${cockatrice_QM} DESTINATION ${COCKATRICE_UNIX_QM_INSTALL_DIR})
|
||||||
|
endif()
|
||||||
|
elseif(WIN32)
|
||||||
|
install(FILES ${cockatrice_QM} DESTINATION ${COCKATRICE_WIN32_QM_INSTALL_DIR})
|
||||||
|
endif()
|
||||||
|
endif()
|
||||||
|
|
||||||
|
if(Qt5_FOUND)
|
||||||
|
target_link_libraries(cockatrice cockatrice_common ${COCKATRICE_QT_MODULES})
|
||||||
|
else()
|
||||||
|
target_link_libraries(cockatrice PUBLIC cockatrice_common ${COCKATRICE_QT_MODULES})
|
||||||
|
endif()
|
||||||
|
|
||||||
if(UNIX)
|
if(UNIX)
|
||||||
if(APPLE)
|
if(APPLE)
|
||||||
set(MACOSX_BUNDLE_INFO_STRING "${PROJECT_NAME}")
|
set(MACOSX_BUNDLE_INFO_STRING "${PROJECT_NAME}")
|
||||||
set(MACOSX_BUNDLE_GUI_IDENTIFIER "com.cockatrice.${PROJECT_NAME}")
|
set(MACOSX_BUNDLE_GUI_IDENTIFIER "com.cockatrice.${PROJECT_NAME}")
|
||||||
set(MACOSX_BUNDLE_LONG_VERSION_STRING "${PROJECT_NAME}-${PROJECT_VERSION}")
|
set(MACOSX_BUNDLE_LONG_VERSION_STRING "${PROJECT_NAME}-${PROJECT_VERSION}")
|
||||||
set(MACOSX_BUNDLE_BUNDLE_NAME "${PROJECT_NAME}")
|
set(MACOSX_BUNDLE_BUNDLE_NAME "${PROJECT_NAME}")
|
||||||
set(MACOSX_BUNDLE_SHORT_VERSION_STRING ${PROJECT_VERSION})
|
set(MACOSX_BUNDLE_SHORT_VERSION_STRING ${PROJECT_VERSION})
|
||||||
set(MACOSX_BUNDLE_BUNDLE_VERSION ${PROJECT_VERSION})
|
set(MACOSX_BUNDLE_BUNDLE_VERSION ${PROJECT_VERSION})
|
||||||
|
|
||||||
set_target_properties(cockatrice PROPERTIES MACOSX_BUNDLE_INFO_PLIST ${CMAKE_SOURCE_DIR}/cmake/Info.plist)
|
set_target_properties(cockatrice PROPERTIES MACOSX_BUNDLE_INFO_PLIST ${CMAKE_SOURCE_DIR}/cmake/Info.plist)
|
||||||
|
|
||||||
INSTALL(TARGETS cockatrice BUNDLE DESTINATION ./)
|
install(TARGETS cockatrice BUNDLE DESTINATION ./)
|
||||||
INSTALL(FILES ${cockatrice_QM} DESTINATION ./cockatrice.app/Contents/Resources/translations)
|
else()
|
||||||
else()
|
# Assume linux
|
||||||
# Assume linux
|
install(TARGETS cockatrice RUNTIME DESTINATION bin/)
|
||||||
INSTALL(TARGETS cockatrice RUNTIME DESTINATION bin/)
|
install(FILES ${CMAKE_CURRENT_SOURCE_DIR}/resources/cockatrice.png DESTINATION ${ICONDIR}/hicolor/48x48/apps)
|
||||||
INSTALL(FILES ${CMAKE_CURRENT_SOURCE_DIR}/resources/cockatrice.png DESTINATION ${ICONDIR}/hicolor/48x48/apps)
|
install(FILES ${CMAKE_CURRENT_SOURCE_DIR}/resources/cockatrice.svg DESTINATION ${ICONDIR}/hicolor/scalable/apps)
|
||||||
INSTALL(FILES ${CMAKE_CURRENT_SOURCE_DIR}/resources/cockatrice.svg DESTINATION ${ICONDIR}/hicolor/scalable/apps)
|
install(FILES ${CMAKE_CURRENT_SOURCE_DIR}/cockatrice.desktop DESTINATION ${DESKTOPDIR})
|
||||||
INSTALL(FILES ${CMAKE_CURRENT_SOURCE_DIR}/cockatrice.desktop DESTINATION ${DESKTOPDIR})
|
endif()
|
||||||
INSTALL(FILES ${cockatrice_QM} DESTINATION share/cockatrice/translations)
|
|
||||||
endif()
|
|
||||||
elseif(WIN32)
|
elseif(WIN32)
|
||||||
INSTALL(TARGETS cockatrice RUNTIME DESTINATION ./)
|
install(TARGETS cockatrice RUNTIME DESTINATION ./)
|
||||||
INSTALL(FILES ${cockatrice_QM} DESTINATION ./translations)
|
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
if(APPLE)
|
if(APPLE)
|
||||||
# these needs to be relative to CMAKE_INSTALL_PREFIX
|
# these needs to be relative to CMAKE_INSTALL_PREFIX
|
||||||
set(plugin_dest_dir cockatrice.app/Contents/Plugins)
|
set(plugin_dest_dir cockatrice.app/Contents/Plugins)
|
||||||
set(qtconf_dest_dir cockatrice.app/Contents/Resources)
|
set(qtconf_dest_dir cockatrice.app/Contents/Resources)
|
||||||
get_filename_component(QT_LIBRARY_DIR "${QT_LIBRARY_DIR}/.." ABSOLUTE)
|
|
||||||
|
|
||||||
# qt5 plugins: audio, iconengines, imageformats, platforms, printsupport
|
# Qt plugins: audio (Qt5), iconengines, imageformats, platforms, printsupport (Qt5), styles, tls (Qt6)
|
||||||
install(DIRECTORY "${QT_PLUGINS_DIR}/" DESTINATION ${plugin_dest_dir} COMPONENT Runtime
|
install(
|
||||||
FILES_MATCHING
|
DIRECTORY "${QT_PLUGINS_DIR}/"
|
||||||
PATTERN "*.dSYM" EXCLUDE
|
DESTINATION ${plugin_dest_dir}
|
||||||
PATTERN "*_debug.dylib" EXCLUDE
|
COMPONENT Runtime
|
||||||
PATTERN "audio/*.dylib"
|
FILES_MATCHING
|
||||||
PATTERN "iconengines/*.dylib"
|
PATTERN "*.dSYM" EXCLUDE
|
||||||
PATTERN "imageformats/*.dylib"
|
PATTERN "*_debug.dylib" EXCLUDE
|
||||||
PATTERN "platforms/*.dylib"
|
PATTERN "audio/*.dylib"
|
||||||
PATTERN "printsupport/*.dylib"
|
PATTERN "iconengines/*.dylib"
|
||||||
PATTERN "styles/*.dylib"
|
PATTERN "imageformats/*.dylib"
|
||||||
)
|
PATTERN "platforms/*.dylib"
|
||||||
|
PATTERN "printsupport/*.dylib"
|
||||||
|
PATTERN "styles/*.dylib"
|
||||||
|
PATTERN "tls/*.dylib"
|
||||||
|
)
|
||||||
|
|
||||||
install(CODE "
|
install(
|
||||||
|
CODE "
|
||||||
file(WRITE \"\${CMAKE_INSTALL_PREFIX}/${qtconf_dest_dir}/qt.conf\" \"[Paths]
|
file(WRITE \"\${CMAKE_INSTALL_PREFIX}/${qtconf_dest_dir}/qt.conf\" \"[Paths]
|
||||||
Plugins = Plugins
|
Plugins = Plugins
|
||||||
Translations = Resources/translations
|
Translations = Resources/translations
|
||||||
Data = Resources\")
|
Data = Resources\")
|
||||||
" COMPONENT Runtime)
|
"
|
||||||
|
COMPONENT Runtime
|
||||||
|
)
|
||||||
|
|
||||||
install(CODE "
|
install(
|
||||||
|
CODE "
|
||||||
file(GLOB_RECURSE QTPLUGINS
|
file(GLOB_RECURSE QTPLUGINS
|
||||||
\"\${CMAKE_INSTALL_PREFIX}/${plugin_dest_dir}/*.dylib\")
|
\"\${CMAKE_INSTALL_PREFIX}/${plugin_dest_dir}/*.dylib\")
|
||||||
set(BU_CHMOD_BUNDLE_ITEMS ON)
|
set(BU_CHMOD_BUNDLE_ITEMS ON)
|
||||||
include(BundleUtilities)
|
include(BundleUtilities)
|
||||||
fixup_bundle(\"\${CMAKE_INSTALL_PREFIX}/cockatrice.app\" \"\${QTPLUGINS}\" \"${QT_LIBRARY_DIR}\")
|
fixup_bundle(\"\${CMAKE_INSTALL_PREFIX}/cockatrice.app\" \"\${QTPLUGINS}\" \"${QT_LIBRARY_DIR}\")
|
||||||
" COMPONENT Runtime)
|
"
|
||||||
|
COMPONENT Runtime
|
||||||
|
)
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
if(WIN32)
|
if(WIN32)
|
||||||
# these needs to be relative to CMAKE_INSTALL_PREFIX
|
# these needs to be relative to CMAKE_INSTALL_PREFIX
|
||||||
set(plugin_dest_dir Plugins)
|
set(plugin_dest_dir Plugins)
|
||||||
set(qtconf_dest_dir .)
|
set(qtconf_dest_dir .)
|
||||||
|
|
||||||
install(DIRECTORY "${CMAKE_BINARY_DIR}/${PROJECT_NAME}/${CMAKE_BUILD_TYPE}/" DESTINATION ./ FILES_MATCHING PATTERN "*.dll")
|
install(
|
||||||
|
DIRECTORY "${CMAKE_BINARY_DIR}/${PROJECT_NAME}/${CMAKE_BUILD_TYPE}/"
|
||||||
|
DESTINATION ./
|
||||||
|
FILES_MATCHING
|
||||||
|
PATTERN "*.dll"
|
||||||
|
)
|
||||||
|
|
||||||
# qt5 plugins: audio, iconengines, imageformats, platforms, printsupport
|
# Qt plugins: audio (Qt5), iconengines, imageformats, platforms, printsupport (Qt5), styles, tls (Qt6)
|
||||||
install(DIRECTORY "${QT_PLUGINS_DIR}/" DESTINATION ${plugin_dest_dir} COMPONENT Runtime
|
install(
|
||||||
FILES_MATCHING REGEX "(audio|iconengines|imageformats|platforms|printsupport)/.*[^d]\\.dll")
|
DIRECTORY "${QT_PLUGINS_DIR}/"
|
||||||
|
DESTINATION ${plugin_dest_dir}
|
||||||
|
COMPONENT Runtime
|
||||||
|
FILES_MATCHING
|
||||||
|
PATTERN "audio/qtaudio_wasapi.dll"
|
||||||
|
PATTERN "audio/qtaudio_windows.dll"
|
||||||
|
PATTERN "iconengines/qsvgicon.dll"
|
||||||
|
PATTERN "imageformats/*.dll"
|
||||||
|
PATTERN "mediaservice/dsengine.dll"
|
||||||
|
PATTERN "mediaservice/wmfengine.dll"
|
||||||
|
PATTERN "platforms/qdirect2d.dll"
|
||||||
|
PATTERN "platforms/qminimal.dll"
|
||||||
|
PATTERN "platforms/qoffscreen.dll"
|
||||||
|
PATTERN "platforms/qwindows.dll"
|
||||||
|
PATTERN "printsupport/windowsprintersupport.dll"
|
||||||
|
PATTERN "styles/qcertonlybackend.dll"
|
||||||
|
PATTERN "styles/qopensslbackend.dll"
|
||||||
|
PATTERN "styles/qschannelbackend.dll"
|
||||||
|
PATTERN "styles/qwindowsvistastyle.dll"
|
||||||
|
PATTERN "tls/qcertonlybackend.dll"
|
||||||
|
PATTERN "tls/qopensslbackend.dll"
|
||||||
|
PATTERN "tls/qschannelbackend.dll"
|
||||||
|
)
|
||||||
|
|
||||||
install(CODE "
|
install(
|
||||||
|
CODE "
|
||||||
file(WRITE \"\${CMAKE_INSTALL_PREFIX}/${qtconf_dest_dir}/qt.conf\" \"[Paths]
|
file(WRITE \"\${CMAKE_INSTALL_PREFIX}/${qtconf_dest_dir}/qt.conf\" \"[Paths]
|
||||||
Plugins = Plugins
|
Plugins = Plugins
|
||||||
Translations = Resources/translations
|
Translations = Resources/translations
|
||||||
Data = Resources\")
|
Data = Resources\")
|
||||||
" COMPONENT Runtime)
|
"
|
||||||
|
COMPONENT Runtime
|
||||||
|
)
|
||||||
|
|
||||||
install(CODE "
|
install(
|
||||||
|
CODE "
|
||||||
file(GLOB_RECURSE QTPLUGINS
|
file(GLOB_RECURSE QTPLUGINS
|
||||||
\"\${CMAKE_INSTALL_PREFIX}/${plugin_dest_dir}/*.dll\")
|
\"\${CMAKE_INSTALL_PREFIX}/${plugin_dest_dir}/*.dll\")
|
||||||
set(BU_CHMOD_BUNDLE_ITEMS ON)
|
set(BU_CHMOD_BUNDLE_ITEMS ON)
|
||||||
include(BundleUtilities)
|
include(BundleUtilities)
|
||||||
fixup_bundle(\"\${CMAKE_INSTALL_PREFIX}/Cockatrice.exe\" \"\${QTPLUGINS}\" \"${QT_LIBRARY_DIR}\")
|
fixup_bundle(\"\${CMAKE_INSTALL_PREFIX}/Cockatrice.exe\" \"\${QTPLUGINS}\" \"${QT_LIBRARY_DIR}\")
|
||||||
" COMPONENT Runtime)
|
"
|
||||||
|
COMPONENT Runtime
|
||||||
|
)
|
||||||
|
|
||||||
if(WIN32SSLRUNTIME_FOUND)
|
if(OPENSSL_FOUND)
|
||||||
install(FILES ${WIN32SSLRUNTIME_LIBRARIES} DESTINATION ./)
|
install(FILES ${OPENSSL_INCLUDE_DIRS} DESTINATION ./)
|
||||||
endif()
|
endif()
|
||||||
|
endif()
|
||||||
|
|
||||||
|
if(Qt6_FOUND AND Qt6LinguistTools_FOUND)
|
||||||
|
#Qt6 Translations happen after the executable is built up
|
||||||
|
if(UPDATE_TRANSLATIONS)
|
||||||
|
qt6_add_translations(
|
||||||
|
cockatrice
|
||||||
|
TS_FILES
|
||||||
|
${cockatrice_TS}
|
||||||
|
SOURCES
|
||||||
|
${translate_SRCS}
|
||||||
|
QM_FILES_OUTPUT_VARIABLE
|
||||||
|
cockatrice_QM
|
||||||
|
)
|
||||||
|
else()
|
||||||
|
qt6_add_translations(cockatrice TS_FILES ${cockatrice_TS} QM_FILES_OUTPUT_VARIABLE cockatrice_QM)
|
||||||
|
endif()
|
||||||
|
|
||||||
|
if(UNIX)
|
||||||
|
if(APPLE)
|
||||||
|
install(FILES ${cockatrice_QM} DESTINATION ${COCKATRICE_MAC_QM_INSTALL_DIR})
|
||||||
|
else()
|
||||||
|
install(FILES ${cockatrice_QM} DESTINATION ${COCKATRICE_UNIX_QM_INSTALL_DIR})
|
||||||
|
endif()
|
||||||
|
elseif(WIN32)
|
||||||
|
install(FILES ${cockatrice_QM} DESTINATION ${COCKATRICE_WIN32_QM_INSTALL_DIR})
|
||||||
|
endif()
|
||||||
|
endif()
|
||||||
|
|
||||||
|
if(Qt6_FOUND)
|
||||||
|
qt6_finalize_target(cockatrice)
|
||||||
endif()
|
endif()
|
||||||
|
|||||||
@@ -18,12 +18,14 @@
|
|||||||
<file>resources/icons/delete.svg</file>
|
<file>resources/icons/delete.svg</file>
|
||||||
<file>resources/icons/forgot_password.svg</file>
|
<file>resources/icons/forgot_password.svg</file>
|
||||||
<file>resources/icons/increment.svg</file>
|
<file>resources/icons/increment.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/pencil.svg</file>
|
<file>resources/icons/pencil.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/remove_row.svg</file>
|
<file>resources/icons/remove_row.svg</file>
|
||||||
|
<file>resources/icons/scales.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/spectator.svg</file>
|
<file>resources/icons/spectator.svg</file>
|
||||||
@@ -124,6 +126,7 @@
|
|||||||
<file>resources/countries/er.svg</file>
|
<file>resources/countries/er.svg</file>
|
||||||
<file>resources/countries/es.svg</file>
|
<file>resources/countries/es.svg</file>
|
||||||
<file>resources/countries/et.svg</file>
|
<file>resources/countries/et.svg</file>
|
||||||
|
<file>resources/countries/eu.svg</file>
|
||||||
<file>resources/countries/fi.svg</file>
|
<file>resources/countries/fi.svg</file>
|
||||||
<file>resources/countries/fj.svg</file>
|
<file>resources/countries/fj.svg</file>
|
||||||
<file>resources/countries/fk.svg</file>
|
<file>resources/countries/fk.svg</file>
|
||||||
@@ -299,16 +302,13 @@
|
|||||||
<file>resources/countries/vu.svg</file>
|
<file>resources/countries/vu.svg</file>
|
||||||
<file>resources/countries/wf.svg</file>
|
<file>resources/countries/wf.svg</file>
|
||||||
<file>resources/countries/ws.svg</file>
|
<file>resources/countries/ws.svg</file>
|
||||||
|
<file>resources/countries/xk.svg</file>
|
||||||
<file>resources/countries/ye.svg</file>
|
<file>resources/countries/ye.svg</file>
|
||||||
<file>resources/countries/yt.svg</file>
|
<file>resources/countries/yt.svg</file>
|
||||||
<file>resources/countries/za.svg</file>
|
<file>resources/countries/za.svg</file>
|
||||||
<file>resources/countries/zm.svg</file>
|
<file>resources/countries/zm.svg</file>
|
||||||
<file>resources/countries/zw.svg</file>
|
<file>resources/countries/zw.svg</file>
|
||||||
|
|
||||||
<file>resources/genders/male.svg</file>
|
|
||||||
<file>resources/genders/female.svg</file>
|
|
||||||
<file>resources/genders/unknown.svg</file>
|
|
||||||
|
|
||||||
<file>resources/phases/untap.svg</file>
|
<file>resources/phases/untap.svg</file>
|
||||||
<file>resources/phases/upkeep.svg</file>
|
<file>resources/phases/upkeep.svg</file>
|
||||||
<file>resources/phases/draw.svg</file>
|
<file>resources/phases/draw.svg</file>
|
||||||
@@ -345,19 +345,22 @@
|
|||||||
<file>resources/userlevels/admin_vip_buddy.svg</file>
|
<file>resources/userlevels/admin_vip_buddy.svg</file>
|
||||||
|
|
||||||
<!-- ADD TIP OF THE DAY IMAGES HERE -->
|
<!-- ADD TIP OF THE DAY IMAGES HERE -->
|
||||||
<file>resources/tips/tips_of_the_day.xml</file>
|
|
||||||
<file>resources/tips/images/accounts_tab.png</file>
|
<file>resources/tips/images/accounts_tab.png</file>
|
||||||
<file>resources/tips/images/arrows.png</file>
|
<file>resources/tips/images/arrows.png</file>
|
||||||
<file>resources/tips/images/cockatrice_register.png</file>
|
<file>resources/tips/images/cockatrice_register.png</file>
|
||||||
<file>resources/tips/images/cockatrice_wiki.png</file>
|
<file>resources/tips/images/cockatrice_wiki.png</file>
|
||||||
<file>resources/tips/images/coin_flip.png</file>
|
<file>resources/tips/images/coin_flip.png</file>
|
||||||
<file>resources/tips/images/counter_expression.png</file>
|
<file>resources/tips/images/counter_expression.png</file>
|
||||||
|
<file>resources/tips/images/discord.png</file>
|
||||||
<file>resources/tips/images/face_down.png</file>
|
<file>resources/tips/images/face_down.png</file>
|
||||||
<file>resources/tips/images/filter_games.png</file>
|
<file>resources/tips/images/filter_games.png</file>
|
||||||
<file>resources/tips/images/github_logo.png</file>
|
<file>resources/tips/images/github_logo.png</file>
|
||||||
<file>resources/tips/images/gitter.png</file>
|
<file>resources/tips/images/setpt.png</file>
|
||||||
|
<file>resources/tips/images/shortcuts.png</file>
|
||||||
<file>resources/tips/images/themes.png</file>
|
<file>resources/tips/images/themes.png</file>
|
||||||
<file>resources/tips/images/tip_of_the_day.png</file>
|
<file>resources/tips/images/tip_of_the_day.png</file>
|
||||||
|
<file>resources/tips/tips_of_the_day.xml</file>
|
||||||
|
|
||||||
|
<file>resources/help/search.md</file>
|
||||||
</qresource>
|
</qresource>
|
||||||
</RCC>
|
</RCC>
|
||||||
|
|||||||
92
cockatrice/resources/countries/xk.svg
Normal file
|
After Width: | Height: | Size: 8.3 KiB |
@@ -1,33 +0,0 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8" standalone="no"?>
|
|
||||||
<!-- Created with Inkscape (http://www.inkscape.org/) -->
|
|
||||||
<svg
|
|
||||||
xmlns:svg="http://www.w3.org/2000/svg"
|
|
||||||
xmlns="http://www.w3.org/2000/svg"
|
|
||||||
version="1.0"
|
|
||||||
width="75"
|
|
||||||
height="75"
|
|
||||||
id="svg34864">
|
|
||||||
<defs
|
|
||||||
id="defs34866" />
|
|
||||||
<g
|
|
||||||
transform="translate(-348.7552,-478.0905)"
|
|
||||||
id="layer1">
|
|
||||||
<g
|
|
||||||
transform="matrix(1.071197,0,0,1.075147,-13.30677,-36.99488)"
|
|
||||||
id="g3773">
|
|
||||||
<path
|
|
||||||
d="M 176 33 A 11 11 0 1 1 154,33 A 11 11 0 1 1 176 33 z"
|
|
||||||
transform="matrix(1.540096,0,0,1.5384,118.8893,454.0543)"
|
|
||||||
style="color:black;fill:none;fill-opacity:1;fill-rule:evenodd;stroke:black;stroke-width:3;stroke-linecap:butt;stroke-linejoin:miter;marker:none;marker-start:none;marker-mid:none;marker-end:none;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:0;stroke-opacity:1;visibility:visible;display:inline;overflow:visible"
|
|
||||||
id="path3939" />
|
|
||||||
<path
|
|
||||||
d="M 373.00525,521.74399 L 373.00525,543.28159"
|
|
||||||
style="fill:none;fill-opacity:0.75;fill-rule:evenodd;stroke:black;stroke-width:4.61774349;stroke-linecap:butt;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1"
|
|
||||||
id="path3941" />
|
|
||||||
<path
|
|
||||||
d="M 363.76467,534.05119 L 382.24582,534.05119"
|
|
||||||
style="fill:none;fill-opacity:0.75;fill-rule:evenodd;stroke:black;stroke-width:4.61774349;stroke-linecap:butt;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1"
|
|
||||||
id="path4816" />
|
|
||||||
</g>
|
|
||||||
</g>
|
|
||||||
</svg>
|
|
||||||
|
Before Width: | Height: | Size: 1.5 KiB |
@@ -1,12 +0,0 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8" standalone="no"?>
|
|
||||||
<!-- Created with Inkscape (http://www.inkscape.org/) -->
|
|
||||||
<svg xmlns:svg="http://www.w3.org/2000/svg" xmlns="http://www.w3.org/2000/svg" version="1.0" width="75" height="75" id="svg34864">
|
|
||||||
<defs id="defs34866"/>
|
|
||||||
<g transform="translate(-348.755, -478.091)" id="layer1">
|
|
||||||
<g transform="matrix(1.94812, 0, 0, 1.93731, -342.43, -460.01)" id="g1872">
|
|
||||||
<path d="M 387.95009,489.60348 L 378.66214,498.89143" style="opacity: 1; color: black; fill: none; fill-opacity: 0.75; fill-rule: evenodd; stroke: black; stroke-width: 3; stroke-linecap: butt; stroke-linejoin: miter; marker: none; stroke-miterlimit: 4; stroke-dasharray: none; stroke-dashoffset: 0pt; stroke-opacity: 1; visibility: visible; display: inline; overflow: visible;" id="path26867"/>
|
|
||||||
<path d="M 49.396475 36.70454 A 15.623922 16.319134 0 1 1 18.14863,36.70454 A 15.623922 16.319134 0 1 1 49.396475 36.70454 z" transform="matrix(0.48802, 0.48802, -0.467594, 0.467594, 371.609, 473.136)" style="opacity: 1; color: black; fill: none; fill-opacity: 1; fill-rule: evenodd; stroke: black; stroke-width: 4.44072; stroke-linecap: butt; stroke-linejoin: miter; marker: none; stroke-miterlimit: 4; stroke-dasharray: none; stroke-dashoffset: 0pt; stroke-opacity: 1; visibility: visible; display: inline; overflow: visible;" id="path26871"/>
|
|
||||||
<path d="M 379.92823,489.70212 C 387.842,489.70212 387.842,489.70212 387.842,489.70212 L 387.842,497.61589" style="fill: none; fill-rule: evenodd; stroke: black; stroke-width: 3; stroke-linecap: butt; stroke-linejoin: miter; stroke-miterlimit: 4; stroke-dasharray: none; stroke-opacity: 1;" id="path27759"/>
|
|
||||||
</g>
|
|
||||||
</g>
|
|
||||||
</svg>
|
|
||||||
|
Before Width: | Height: | Size: 1.7 KiB |
@@ -1,10 +0,0 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8" standalone="no"?>
|
|
||||||
<!-- Intersexual/Transgendered symbol by Gregory Maxwell. Copyright 2005. GFDL-1.2 only -->
|
|
||||||
<svg
|
|
||||||
xmlns="http://www.w3.org/2000/svg"
|
|
||||||
version="1.1"
|
|
||||||
width="210"
|
|
||||||
height="280">
|
|
||||||
<path d="M 188,38 v 20 l 14,-8 v -42 h -42 l -8,14 h 20 l -37,37 a 79,79 0 1,0 -59,141 v 22 h -27 v 23 h 27 v 27 h 23 v -27 h 27 v -23 h -27 v -22 a 79,79 0 0,0 52,-125 zm -100,27 a 57,57 0 1,1 0,114 a 57,57 0 1,1 0,-114 z"/>
|
|
||||||
<!-- 88,65 // -63,-10 -->
|
|
||||||
</svg>
|
|
||||||
|
Before Width: | Height: | Size: 500 B |
61
cockatrice/resources/help/search.md
Normal file
@@ -0,0 +1,61 @@
|
|||||||
|
## Search Syntax Help
|
||||||
|
-----
|
||||||
|
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.
|
||||||
|
<dl>
|
||||||
|
<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"](#%22birds of paradise%22) <small>(Any card name containing the exact phrase "birds of paradise")</small></dd>
|
||||||
|
|
||||||
|
<dt>Rules Text (<u>O</u>racle):</dt>
|
||||||
|
<dd>[o:flying](#o:flying) <small>(Any card text that has the word flying)</small></dd>
|
||||||
|
<dd>[o:"first strike"](#o:%22first strike%22) <small>(Any card text that has the exact phrase "first strike")</small></dd>
|
||||||
|
<dd>[o:"{T}" o:"add one mana of any color"](#o:%22{T}%22 o:%22add one mana of any color%22) <small>(Any card text that has a tap symbol and the phrase "add one mana of any color")</small></dd>
|
||||||
|
|
||||||
|
<dt><u>T</u>ypes:</dt>
|
||||||
|
<dd>[t:angel](#t:angel) <small>(Any card with the type angel)</small></dd>
|
||||||
|
<dd>[t:angel t:legendary](#t:angel t:legendary) <small>(Any angel that's also legendary)</small></dd>
|
||||||
|
<dd>[t:basic](#t:basic) <small>(Any card with the type basic)</small></dd>
|
||||||
|
<dd>[t:arcane t:instant](#t:arcane t:instant) <small>(Any card with the types arcane and instant)</small></dd>
|
||||||
|
|
||||||
|
<dt><u>C</u>olors:</dt>
|
||||||
|
<dd>[c:w](#c:w) <small>(Any card that is white)</small></dd>
|
||||||
|
<dd>[c:wu](#c:wu) <small>(Any card that is white or blue)</small></dd>
|
||||||
|
<dd>[c:wum](#c:wum) <small>(Any card that is white or blue, and multicolored)</small></dd>
|
||||||
|
<!--
|
||||||
|
<dd>[c!w](#c!w) <small>(Cards that are only white)</small></dd>
|
||||||
|
<dd>[c!wu](#c!wu) <small>(Cards that are only white or blue, or both)</small></dd>
|
||||||
|
<dd>[c!wum](#c!wum) <small>(Cards that are only white and blue, and multicolored)</small></dd>
|
||||||
|
<dd>[c=wubrg](#c%3Dwubrg) <small>(Cards that are all five colors)</small></dd>
|
||||||
|
-->
|
||||||
|
<dd>[c:c](#c:c) <small>(Any colorless card)</small></dd>
|
||||||
|
<dd>[ci:w](#ci:w) <small>(Any card that has white in it's color identity)</small></dd>
|
||||||
|
|
||||||
|
<dt><u>Pow</u>er, <u>Tou</u>ghness, <u>M</u>ana <u>V</u>alue:</dt>
|
||||||
|
<dd>[tou:1](#tou:1) <small>(Any card with a toughness of 1)</small></dd>
|
||||||
|
<dd>[pow>=8](#pow>=8) <small>(Any card with a power greater than or equal to 8)</small></dd>
|
||||||
|
<dd>[mv=7](#mv=7) <small>(Any card with a mana value equal to 7)</small></dd>
|
||||||
|
|
||||||
|
<dt><u>R</u>arity:</dt>
|
||||||
|
<dd>[r:mythic](#r:mythic) <small>(Any card that has the mythic-rare rarity)</small></dd>
|
||||||
|
|
||||||
|
<dt><u>F</u>ormat:</dt>
|
||||||
|
<dd>[f:standard](#f:standard) <small>(Any card that can be played in standard)</small></dd>
|
||||||
|
<dd>[banned:modern](#banned:modern) <small>(Any card that is banned in modern)</small></dd>
|
||||||
|
<dd>[restricted:vintage](#restricted:vintage) <small>(Any card that is restricted in vintage)</small></dd>
|
||||||
|
<dd>[legal:pauper](#legal:pauper) <small>(Any card that is legal in pauper)</small></dd>
|
||||||
|
|
||||||
|
<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>[e:lea or e:leb](#e:lea or e:leb) <small>(Cards that appear in Alpha or Beta)</small></dd>
|
||||||
|
|
||||||
|
<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>
|
||||||
|
|
||||||
|
<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>
|
||||||
|
|
||||||
|
<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>
|
||||||
|
|
||||||
|
</dl>
|
||||||
43
cockatrice/resources/icons/info.svg
Normal file
@@ -0,0 +1,43 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8" standalone="no"?>
|
||||||
|
<!-- Created with Inkscape (http://www.inkscape.org/) -->
|
||||||
|
|
||||||
|
<svg
|
||||||
|
xmlns:dc="http://purl.org/dc/elements/1.1/"
|
||||||
|
xmlns:cc="http://creativecommons.org/ns#"
|
||||||
|
xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
|
||||||
|
xmlns:svg="http://www.w3.org/2000/svg"
|
||||||
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
|
xmlns:xlink="http://www.w3.org/1999/xlink"
|
||||||
|
xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd"
|
||||||
|
xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape"
|
||||||
|
width="100"
|
||||||
|
height="100"
|
||||||
|
id="svg2858"
|
||||||
|
version="1.1"
|
||||||
|
inkscape:version="0.48.5 r10040"
|
||||||
|
sodipodi:docname="info.svg">
|
||||||
|
<g
|
||||||
|
inkscape:groupmode="layer"
|
||||||
|
id="layer2"
|
||||||
|
inkscape:label="Triangle"
|
||||||
|
style="display:inline">
|
||||||
|
<path
|
||||||
|
sodipodi:type="arc"
|
||||||
|
style="fill:#aaaaaa;fill-opacity:1;fill-rule:nonzero;stroke:none;display:inline"
|
||||||
|
id="path3758-1"
|
||||||
|
sodipodi:cx="53.900002"
|
||||||
|
sodipodi:cy="78.5"
|
||||||
|
sodipodi:rx="46.5"
|
||||||
|
sodipodi:ry="46.5"
|
||||||
|
d="M 100.4,78.5 A 46.5,46.5 0 1 1 7.4000015,78.5 46.5,46.5 0 1 1 100.4,78.5 z"
|
||||||
|
transform="matrix(1.05866,0,0,1.05866,-7.0617752,-32.704809)" />
|
||||||
|
<g
|
||||||
|
style="font-size:133.49534607px;font-style:normal;font-variant:normal;font-weight:normal;font-stretch:normal;line-height:125%;letter-spacing:0px;word-spacing:0px;fill:#ffffff;fill-opacity:1;stroke:#ffffff;stroke-width:1.06498075;stroke-opacity:1;font-family:'Magic:the Gathering';-inkscape-font-specification:'Magic:the Gathering'"
|
||||||
|
id="text3759">
|
||||||
|
<path
|
||||||
|
d="m 46.854005,16.085692 c 0.977733,-2.411704 1.683885,-4.193378 2.118456,-5.345028 0.369355,-1.1514901 0.771318,-1.9988718 1.205891,-2.5421482 0.456264,-0.4344718 1.347101,-0.825571 2.672514,-1.1732989 1.23846,-0.2823765 2.705082,-0.4670623 4.399871,-0.5540578 0.369346,8.46e-5 0.727854,8.46e-5 1.075524,0 1.694737,8.46e-5 2.672485,0.5758696 2.933248,1.7273567 -3e-5,0.1956326 -3e-5,0.3803183 0,0.5540579 -3e-5,1.3254743 -0.662726,2.9224633 -1.98809,4.7909703 -1.520969,2.238035 -3.43301,4.24785 -5.736128,6.029453 -2.32489,1.760018 -4.551983,2.781222 -6.681286,3.063614 -1.260224,-0.543125 -1.977239,-1.075454 -2.151048,-1.59699 -1.3e-5,-0.08684 -1.3e-5,-0.184616 0,-0.293325 -1.3e-5,-0.434484 0.228128,-1.010269 0.684424,-1.727357 0.521453,-0.890765 1.010327,-1.868513 1.466624,-2.933247 z M 31.372977,37.856906 c 2.650785,-0.716962 6.159814,-1.651255 10.527099,-2.802881 4.280354,-1.173242 7.398284,-1.803346 9.3538,-1.890315 0.08689,5.8e-5 0.173803,5.8e-5 0.260733,0 1.06464,5.8e-5 1.82511,0.358566 2.281415,1.075524 0.260711,0.630161 0.391077,1.520998 0.391099,2.672514 -2.2e-5,0.260787 -2.2e-5,0.532384 0,0.814791 -0.173844,1.586178 -0.260755,2.781204 -0.260733,3.58508 l 0,39.533656 c 1.238459,1e-5 2.976678,-0.673549 5.214662,-2.020681 1.673009,-0.977736 2.868034,-1.46661 3.58508,-1.466624 0.173791,1.4e-5 0.304158,0.04347 0.3911,0.130367 0.543162,0.369385 0.814759,0.771348 0.814791,1.20589 -3.2e-5,1.151583 -1.868617,2.66166 -5.605762,4.530238 -1.238507,0.717023 -3.063637,1.922912 -5.475395,3.617671 -2.49871,1.781679 -4.769259,3.204846 -6.811652,4.269505 -0.347658,0.347644 -0.923443,0.662696 -1.727357,0.945157 -0.260745,0.08691 -0.532342,0.130366 -0.814791,0.130367 -0.521477,-10e-7 -1.010351,-0.271598 -1.466623,-0.814791 C 41.856611,91.111641 41.7697,81.594892 41.76971,62.8221 c -1e-5,-1.955468 -1e-5,-4.008739 0,-6.15982 -1e-5,-2.129283 -0.08692,-4.128235 -0.260734,-5.996861 -0.260742,-1.781634 -0.706161,-3.204801 -1.336257,-4.269504 -0.717023,-1.347075 -1.60786,-2.368279 -2.672514,-3.063614 -1.151576,-0.630057 -2.357465,-1.260161 -3.617672,-1.890315 -1.325394,-0.521416 -2.161911,-1.053746 -2.509556,-1.59699 -0.195549,-0.173771 -0.293324,-0.434504 -0.293324,-0.7822 -0.08691,-0.369319 0.01086,-0.771282 0.293324,-1.20589 z"
|
||||||
|
style=""
|
||||||
|
id="path2993" />
|
||||||
|
</g>
|
||||||
|
</g>
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 3.8 KiB |
6
cockatrice/resources/icons/scales.svg
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="52pt" height="52pt" viewBox="0 0 52 52" version="1.1">
|
||||||
|
<g id="surface1">
|
||||||
|
<path style=" stroke:none;fill-rule:nonzero;fill:#000000;fill-opacity:1;" d="M 25.605469 0.125 C 25.605469 0.125 23.265625 3.140625 23.265625 4.433594 C 23.265625 5.316406 23.765625 6.070313 24.5 6.464844 C 23.898438 6.65625 23.375 6.992188 22.960938 7.449219 C 22.011719 6.640625 20.628906 5.910156 18.710938 5.910156 C 16.640625 5.910156 14.863281 6.753906 13.355469 7.507813 C 12.601563 7.886719 11.925781 8.25 11.324219 8.496094 C 10.964844 8.09375 10.425781 7.878906 9.847656 7.878906 C 8.75 7.878906 7.878906 8.75 7.878906 9.847656 C 7.878906 10.949219 8.75 11.816406 9.847656 11.816406 C 10.757813 11.816406 11.519531 11.234375 11.757813 10.402344 C 12.578125 10.078125 13.386719 9.648438 14.21875 9.234375 C 15.671875 8.507813 17.179688 7.878906 18.710938 7.878906 C 20.496094 7.878906 21.519531 8.679688 22.097656 9.355469 C 22.074219 9.519531 22.035156 9.6875 22.035156 9.847656 C 22.035156 11.816406 23.636719 13.417969 25.605469 13.417969 C 27.574219 13.417969 29.175781 11.816406 29.175781 9.847656 C 29.175781 9.6875 29.136719 9.519531 29.113281 9.355469 C 29.691406 8.679688 30.714844 7.878906 32.5 7.878906 C 34.03125 7.878906 35.539063 8.507813 36.992188 9.234375 C 37.824219 9.648438 38.632813 10.078125 39.457031 10.402344 C 39.695313 11.234375 40.457031 11.816406 41.363281 11.816406 C 42.464844 11.816406 43.332031 10.949219 43.332031 9.847656 C 43.332031 8.75 42.464844 7.878906 41.363281 7.878906 C 40.785156 7.878906 40.246094 8.09375 39.886719 8.496094 C 39.285156 8.25 38.609375 7.886719 37.855469 7.507813 C 36.347656 6.753906 34.570313 5.910156 32.5 5.910156 C 30.585938 5.910156 29.199219 6.640625 28.253906 7.449219 C 27.835938 6.992188 27.3125 6.65625 26.714844 6.464844 C 27.445313 6.070313 27.945313 5.316406 27.945313 4.433594 C 27.945313 3.140625 25.605469 0.125 25.605469 0.125 Z M 9.539063 13.234375 C 9.25 13.332031 9.023438 13.558594 8.925781 13.847656 L 2.277344 31.515625 L 0 31.515625 C 1.445313 35.578125 5.285156 38.53125 9.847656 38.53125 C 14.410156 38.53125 18.25 35.578125 19.695313 31.515625 L 17.417969 31.515625 L 10.773438 13.847656 C 10.601563 13.402344 10.132813 13.140625 9.664063 13.234375 C 9.625 13.234375 9.578125 13.234375 9.539063 13.234375 Z M 41.054688 13.234375 C 40.761719 13.332031 40.539063 13.558594 40.441406 13.847656 L 33.792969 31.515625 L 31.515625 31.515625 C 32.960938 35.578125 36.800781 38.53125 41.363281 38.53125 C 45.925781 38.53125 49.765625 35.578125 51.210938 31.515625 L 48.933594 31.515625 L 42.285156 13.847656 C 42.117188 13.402344 41.648438 13.140625 41.179688 13.234375 C 41.140625 13.234375 41.09375 13.234375 41.054688 13.234375 Z M 21.914063 15.757813 C 21.914063 17.082031 22.589844 18.242188 23.636719 18.898438 L 23.636719 41.980469 C 21.429688 42.773438 19.796875 44.816406 19.695313 47.273438 L 15.757813 47.273438 C 14.671875 47.273438 13.789063 48.15625 13.789063 49.242188 L 13.789063 51.210938 L 37.425781 51.210938 L 37.425781 49.242188 C 37.425781 48.15625 36.539063 47.273438 35.453125 47.273438 L 31.515625 47.273438 C 31.414063 44.816406 29.785156 42.773438 27.574219 41.980469 L 27.574219 18.898438 C 28.621094 18.242188 29.300781 17.082031 29.300781 15.757813 Z M 9.847656 17.050781 L 15.328125 31.515625 L 4.371094 31.515625 Z M 41.363281 17.050781 L 46.84375 31.515625 L 35.886719 31.515625 Z "/>
|
||||||
|
</g>
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 3.4 KiB |
BIN
cockatrice/resources/tips/images/discord.png
Normal file
|
After Width: | Height: | Size: 30 KiB |
|
Before Width: | Height: | Size: 5.9 KiB |
BIN
cockatrice/resources/tips/images/setpt.png
Normal file
|
After Width: | Height: | Size: 18 KiB |
BIN
cockatrice/resources/tips/images/shortcuts.png
Normal file
|
After Width: | Height: | Size: 21 KiB |
@@ -7,9 +7,9 @@
|
|||||||
</tip>
|
</tip>
|
||||||
<tip>
|
<tip>
|
||||||
<title>Suggesting New Tips</title>
|
<title>Suggesting New Tips</title>
|
||||||
<text>You can suggest new Tips of the Day by reaching out to the development team on <a href="https://gitter.im/cockatrice/cockatrice">Gitter</a>!</text>
|
<text>You can suggest new Tips of the Day by reaching out to the development team on <a href="https://discord.gg/3Z9yzmA">Discord</a>!</text>
|
||||||
<image>gitter.png</image>
|
<image>discord.png</image>
|
||||||
<date>2018-03-01</date>
|
<date>2023-10-18</date>
|
||||||
</tip>
|
</tip>
|
||||||
<tip>
|
<tip>
|
||||||
<title>Reporting Bugs</title>
|
<title>Reporting Bugs</title>
|
||||||
@@ -63,6 +63,7 @@
|
|||||||
<br>Change Life: CTRL + L
|
<br>Change Life: CTRL + L
|
||||||
<br>All shortcuts can be customized via Settings->Shortcuts!
|
<br>All shortcuts can be customized via Settings->Shortcuts!
|
||||||
</text>
|
</text>
|
||||||
|
<image>shortcuts.png</image>
|
||||||
<date>2018-03-01</date>
|
<date>2018-03-01</date>
|
||||||
</tip>
|
</tip>
|
||||||
<tip>
|
<tip>
|
||||||
@@ -89,4 +90,10 @@
|
|||||||
<image>counter_expression.png</image>
|
<image>counter_expression.png</image>
|
||||||
<date>2019-02-02</date>
|
<date>2019-02-02</date>
|
||||||
</tip>
|
</tip>
|
||||||
</tips>
|
<tip>
|
||||||
|
<title>Power and Toughness</title>
|
||||||
|
<text>You can add and subtract to a creature's stats.<br>With a card selected, set the power and toughness ( default: ctrl + p ) and enter +3/-1 to increase power by three while decreasing toughness by one.<br>You can also reset it to the original value ( default: ctrl + alt + 0 ).</text>
|
||||||
|
<image>setpt.png</image>
|
||||||
|
<date>2019-03-02</date>
|
||||||
|
</tip>
|
||||||
|
</tips>
|
||||||
|
|||||||
@@ -2,18 +2,15 @@
|
|||||||
#
|
#
|
||||||
# add sounds subfolders
|
# add sounds subfolders
|
||||||
|
|
||||||
SET(defsounds
|
set(defsounds Default Legacy)
|
||||||
Default
|
|
||||||
Legacy
|
|
||||||
)
|
|
||||||
|
|
||||||
if(UNIX)
|
if(UNIX)
|
||||||
if(APPLE)
|
if(APPLE)
|
||||||
INSTALL(DIRECTORY ${defsounds} DESTINATION Cockatrice.app/Contents/Resources/sounds/)
|
install(DIRECTORY ${defsounds} DESTINATION Cockatrice.app/Contents/Resources/sounds/)
|
||||||
else()
|
else()
|
||||||
# Assume linux
|
# Assume linux
|
||||||
INSTALL(DIRECTORY ${defsounds} DESTINATION share/cockatrice/sounds/)
|
install(DIRECTORY ${defsounds} DESTINATION share/cockatrice/sounds/)
|
||||||
endif()
|
endif()
|
||||||
elseif(WIN32)
|
elseif(WIN32)
|
||||||
INSTALL(DIRECTORY ${defsounds} DESTINATION sounds/)
|
install(DIRECTORY ${defsounds} DESTINATION sounds/)
|
||||||
endif()
|
endif()
|
||||||
|
|||||||
@@ -1,60 +0,0 @@
|
|||||||
#include "cardinfopicture.h"
|
|
||||||
|
|
||||||
#include <QPainter>
|
|
||||||
#include <QStyle>
|
|
||||||
#include <QWidget>
|
|
||||||
|
|
||||||
#include "carditem.h"
|
|
||||||
#include "main.h"
|
|
||||||
#include "pictureloader.h"
|
|
||||||
|
|
||||||
CardInfoPicture::CardInfoPicture(QWidget *parent) : QWidget(parent), info(nullptr), pixmapDirty(true)
|
|
||||||
{
|
|
||||||
setMinimumHeight(100);
|
|
||||||
}
|
|
||||||
|
|
||||||
void CardInfoPicture::setCard(CardInfoPtr card)
|
|
||||||
{
|
|
||||||
if (info) {
|
|
||||||
disconnect(info.data(), nullptr, this, nullptr);
|
|
||||||
}
|
|
||||||
|
|
||||||
info = card;
|
|
||||||
|
|
||||||
if (info) {
|
|
||||||
connect(info.data(), SIGNAL(pixmapUpdated()), this, SLOT(updatePixmap()));
|
|
||||||
}
|
|
||||||
|
|
||||||
updatePixmap();
|
|
||||||
}
|
|
||||||
|
|
||||||
void CardInfoPicture::resizeEvent(QResizeEvent *)
|
|
||||||
{
|
|
||||||
updatePixmap();
|
|
||||||
}
|
|
||||||
|
|
||||||
void CardInfoPicture::updatePixmap()
|
|
||||||
{
|
|
||||||
pixmapDirty = true;
|
|
||||||
update();
|
|
||||||
}
|
|
||||||
|
|
||||||
void CardInfoPicture::loadPixmap()
|
|
||||||
{
|
|
||||||
if (info)
|
|
||||||
PictureLoader::getPixmap(resizedPixmap, info, size());
|
|
||||||
else
|
|
||||||
PictureLoader::getCardBackPixmap(resizedPixmap, size());
|
|
||||||
}
|
|
||||||
|
|
||||||
void CardInfoPicture::paintEvent(QPaintEvent *)
|
|
||||||
{
|
|
||||||
if (width() == 0 || height() == 0)
|
|
||||||
return;
|
|
||||||
|
|
||||||
if (pixmapDirty)
|
|
||||||
loadPixmap();
|
|
||||||
|
|
||||||
QPainter painter(this);
|
|
||||||
style()->drawItemPixmap(&painter, rect(), Qt::AlignHCenter, resizedPixmap);
|
|
||||||
}
|
|
||||||
@@ -1,31 +0,0 @@
|
|||||||
#ifndef CARDINFOPICTURE_H
|
|
||||||
#define CARDINFOPICTURE_H
|
|
||||||
|
|
||||||
#include <QWidget>
|
|
||||||
|
|
||||||
#include "carddatabase.h"
|
|
||||||
|
|
||||||
class AbstractCardItem;
|
|
||||||
|
|
||||||
class CardInfoPicture : public QWidget
|
|
||||||
{
|
|
||||||
Q_OBJECT
|
|
||||||
|
|
||||||
private:
|
|
||||||
CardInfoPtr info;
|
|
||||||
QPixmap resizedPixmap;
|
|
||||||
bool pixmapDirty;
|
|
||||||
|
|
||||||
public:
|
|
||||||
CardInfoPicture(QWidget *parent = 0);
|
|
||||||
|
|
||||||
protected:
|
|
||||||
void resizeEvent(QResizeEvent *event);
|
|
||||||
void paintEvent(QPaintEvent *);
|
|
||||||
void loadPixmap();
|
|
||||||
public slots:
|
|
||||||
void setCard(CardInfoPtr card);
|
|
||||||
void updatePixmap();
|
|
||||||
};
|
|
||||||
|
|
||||||
#endif
|
|
||||||
@@ -1,60 +0,0 @@
|
|||||||
#include "cardlist.h"
|
|
||||||
#include "carddatabase.h"
|
|
||||||
#include "carditem.h"
|
|
||||||
|
|
||||||
CardList::CardList(bool _contentsKnown) : QList<CardItem *>(), contentsKnown(_contentsKnown)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
CardItem *CardList::findCard(const int id, const bool remove, int *position)
|
|
||||||
{
|
|
||||||
if (!contentsKnown) {
|
|
||||||
if (empty())
|
|
||||||
return 0;
|
|
||||||
CardItem *temp = at(0);
|
|
||||||
if (remove)
|
|
||||||
removeAt(0);
|
|
||||||
if (position)
|
|
||||||
*position = id;
|
|
||||||
return temp;
|
|
||||||
} else
|
|
||||||
for (int i = 0; i < size(); i++) {
|
|
||||||
CardItem *temp = at(i);
|
|
||||||
if (temp->getId() == id) {
|
|
||||||
if (remove)
|
|
||||||
removeAt(i);
|
|
||||||
if (position)
|
|
||||||
*position = i;
|
|
||||||
return temp;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
class CardList::compareFunctor
|
|
||||||
{
|
|
||||||
private:
|
|
||||||
int flags;
|
|
||||||
|
|
||||||
public:
|
|
||||||
explicit compareFunctor(int _flags) : flags(_flags)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
inline bool operator()(CardItem *a, CardItem *b) const
|
|
||||||
{
|
|
||||||
if (flags & SortByType) {
|
|
||||||
QString t1 = a->getInfo() ? a->getInfo()->getMainCardType() : QString();
|
|
||||||
QString t2 = b->getInfo() ? b->getInfo()->getMainCardType() : QString();
|
|
||||||
if ((t1 == t2) && (flags & SortByName))
|
|
||||||
return a->getName() < b->getName();
|
|
||||||
return t1 < t2;
|
|
||||||
} else
|
|
||||||
return a->getName() < b->getName();
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
void CardList::sort(int flags)
|
|
||||||
{
|
|
||||||
compareFunctor cf(flags);
|
|
||||||
qSort(begin(), end(), cf);
|
|
||||||
}
|
|
||||||
@@ -1,31 +0,0 @@
|
|||||||
#ifndef CARDLIST_H
|
|
||||||
#define CARDLIST_H
|
|
||||||
|
|
||||||
#include <QList>
|
|
||||||
|
|
||||||
class CardItem;
|
|
||||||
|
|
||||||
class CardList : public QList<CardItem *>
|
|
||||||
{
|
|
||||||
private:
|
|
||||||
class compareFunctor;
|
|
||||||
|
|
||||||
protected:
|
|
||||||
bool contentsKnown;
|
|
||||||
|
|
||||||
public:
|
|
||||||
enum SortFlags
|
|
||||||
{
|
|
||||||
SortByName = 1,
|
|
||||||
SortByType = 2
|
|
||||||
};
|
|
||||||
CardList(bool _contentsKnown);
|
|
||||||
CardItem *findCard(const int id, const bool remove, int *position = NULL);
|
|
||||||
bool getContentsKnown() const
|
|
||||||
{
|
|
||||||
return contentsKnown;
|
|
||||||
}
|
|
||||||
void sort(int flags = SortByName);
|
|
||||||
};
|
|
||||||
|
|
||||||
#endif
|
|
||||||
@@ -1,6 +1,6 @@
|
|||||||
#include "abstractclient.h"
|
#include "abstract_client.h"
|
||||||
|
|
||||||
#include "client_metatypes.h"
|
#include "../../server/pending_command.h"
|
||||||
#include "featureset.h"
|
#include "featureset.h"
|
||||||
#include "get_pb_extension.h"
|
#include "get_pb_extension.h"
|
||||||
#include "pb/commands.pb.h"
|
#include "pb/commands.pb.h"
|
||||||
@@ -18,10 +18,11 @@
|
|||||||
#include "pb/event_user_left.pb.h"
|
#include "pb/event_user_left.pb.h"
|
||||||
#include "pb/event_user_message.pb.h"
|
#include "pb/event_user_message.pb.h"
|
||||||
#include "pb/server_message.pb.h"
|
#include "pb/server_message.pb.h"
|
||||||
#include "pending_command.h"
|
|
||||||
#include <google/protobuf/descriptor.h>
|
#include <google/protobuf/descriptor.h>
|
||||||
|
|
||||||
AbstractClient::AbstractClient(QObject *parent) : QObject(parent), nextCmdId(0), status(StatusDisconnected)
|
AbstractClient::AbstractClient(QObject *parent)
|
||||||
|
: QObject(parent), nextCmdId(0), status(StatusDisconnected), serverSupportsPasswordHash(false)
|
||||||
{
|
{
|
||||||
qRegisterMetaType<QVariant>("QVariant");
|
qRegisterMetaType<QVariant>("QVariant");
|
||||||
qRegisterMetaType<CommandContainer>("CommandContainer");
|
qRegisterMetaType<CommandContainer>("CommandContainer");
|
||||||
@@ -46,6 +47,7 @@ AbstractClient::AbstractClient(QObject *parent) : QObject(parent), nextCmdId(0),
|
|||||||
qRegisterMetaType<QList<ServerInfo_User>>("QList<ServerInfo_User>");
|
qRegisterMetaType<QList<ServerInfo_User>>("QList<ServerInfo_User>");
|
||||||
qRegisterMetaType<Event_ReplayAdded>("Event_ReplayAdded");
|
qRegisterMetaType<Event_ReplayAdded>("Event_ReplayAdded");
|
||||||
qRegisterMetaType<QList<QString>>("missingFeatures");
|
qRegisterMetaType<QList<QString>>("missingFeatures");
|
||||||
|
qRegisterMetaType<PendingCommand *>("pendingCommand");
|
||||||
|
|
||||||
FeatureSet features;
|
FeatureSet features;
|
||||||
features.initalizeFeatureList(clientFeatures);
|
features.initalizeFeatureList(clientFeatures);
|
||||||
@@ -3,6 +3,7 @@
|
|||||||
|
|
||||||
#include "pb/response.pb.h"
|
#include "pb/response.pb.h"
|
||||||
#include "pb/serverinfo_user.pb.h"
|
#include "pb/serverinfo_user.pb.h"
|
||||||
|
|
||||||
#include <QMutex>
|
#include <QMutex>
|
||||||
#include <QObject>
|
#include <QObject>
|
||||||
#include <QVariant>
|
#include <QVariant>
|
||||||
@@ -39,6 +40,7 @@ enum ClientStatus
|
|||||||
StatusRequestingForgotPassword,
|
StatusRequestingForgotPassword,
|
||||||
StatusSubmitForgotPasswordReset,
|
StatusSubmitForgotPasswordReset,
|
||||||
StatusSubmitForgotPasswordChallenge,
|
StatusSubmitForgotPasswordChallenge,
|
||||||
|
StatusGettingPasswordSalt,
|
||||||
};
|
};
|
||||||
|
|
||||||
class AbstractClient : public QObject
|
class AbstractClient : public QObject
|
||||||
@@ -86,7 +88,7 @@ protected slots:
|
|||||||
protected:
|
protected:
|
||||||
QMap<int, PendingCommand *> pendingCommands;
|
QMap<int, PendingCommand *> pendingCommands;
|
||||||
QString userName, password, email, country, realName, token;
|
QString userName, password, email, country, realName, token;
|
||||||
int gender;
|
bool serverSupportsPasswordHash;
|
||||||
void setStatus(ClientStatus _status);
|
void setStatus(ClientStatus _status);
|
||||||
int getNewCmdId()
|
int getNewCmdId()
|
||||||
{
|
{
|
||||||
@@ -95,7 +97,7 @@ protected:
|
|||||||
virtual void sendCommandContainer(const CommandContainer &cont) = 0;
|
virtual void sendCommandContainer(const CommandContainer &cont) = 0;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
AbstractClient(QObject *parent = 0);
|
AbstractClient(QObject *parent = nullptr);
|
||||||
~AbstractClient();
|
~AbstractClient();
|
||||||
|
|
||||||
ClientStatus getStatus() const
|
ClientStatus getStatus() const
|
||||||
@@ -106,7 +108,11 @@ public:
|
|||||||
void sendCommand(const CommandContainer &cont);
|
void sendCommand(const CommandContainer &cont);
|
||||||
void sendCommand(PendingCommand *pend);
|
void sendCommand(PendingCommand *pend);
|
||||||
|
|
||||||
const QString getUserName()
|
bool getServerSupportsPasswordHash() const
|
||||||
|
{
|
||||||
|
return serverSupportsPasswordHash;
|
||||||
|
}
|
||||||
|
const QString &getUserName() const
|
||||||
{
|
{
|
||||||
return userName;
|
return userName;
|
||||||
}
|
}
|
||||||
@@ -1,4 +1,5 @@
|
|||||||
#include "keysignals.h"
|
#include "key_signals.h"
|
||||||
|
|
||||||
#include <QKeyEvent>
|
#include <QKeyEvent>
|
||||||
|
|
||||||
bool KeySignals::eventFilter(QObject * /*object*/, QEvent *event)
|
bool KeySignals::eventFilter(QObject * /*object*/, QEvent *event)
|
||||||
@@ -59,6 +60,11 @@ bool KeySignals::eventFilter(QObject * /*object*/, QEvent *event)
|
|||||||
if (kevent->modifiers() & Qt::ShiftModifier)
|
if (kevent->modifiers() & Qt::ShiftModifier)
|
||||||
emit onShiftS();
|
emit onShiftS();
|
||||||
|
|
||||||
|
break;
|
||||||
|
case Qt::Key_C:
|
||||||
|
if (kevent->modifiers() & Qt::ControlModifier)
|
||||||
|
emit onCtrlC();
|
||||||
|
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
return false;
|
return false;
|
||||||
@@ -20,6 +20,7 @@ signals:
|
|||||||
void onCtrlAltLBracket();
|
void onCtrlAltLBracket();
|
||||||
void onCtrlAltRBracket();
|
void onCtrlAltRBracket();
|
||||||
void onShiftS();
|
void onShiftS();
|
||||||
|
void onCtrlC();
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
virtual bool eventFilter(QObject *, QEvent *event);
|
virtual bool eventFilter(QObject *, QEvent *event);
|
||||||
32
cockatrice/src/client/get_text_with_max.cpp
Normal file
@@ -0,0 +1,32 @@
|
|||||||
|
#include "get_text_with_max.h"
|
||||||
|
|
||||||
|
QString getTextWithMax(QWidget *parent,
|
||||||
|
const QString &title,
|
||||||
|
const QString &label,
|
||||||
|
QLineEdit::EchoMode mode,
|
||||||
|
const QString &text,
|
||||||
|
bool *ok,
|
||||||
|
int max,
|
||||||
|
Qt::WindowFlags flags,
|
||||||
|
Qt::InputMethodHints inputMethodHints)
|
||||||
|
{
|
||||||
|
auto *dialog = new QInputDialog(parent, flags);
|
||||||
|
dialog->setWindowTitle(title);
|
||||||
|
dialog->setLabelText(label);
|
||||||
|
dialog->setTextValue(text);
|
||||||
|
dialog->setTextEchoMode(mode);
|
||||||
|
dialog->setInputMethodHints(inputMethodHints);
|
||||||
|
|
||||||
|
// find the qlineedit that this dialog holds, there should be only one
|
||||||
|
dialog->findChild<QLineEdit *>()->setMaxLength(max);
|
||||||
|
|
||||||
|
const int ret = dialog->exec();
|
||||||
|
if (ok != nullptr) {
|
||||||
|
*ok = !!ret;
|
||||||
|
}
|
||||||
|
if (ret) {
|
||||||
|
return dialog->textValue();
|
||||||
|
} else {
|
||||||
|
return QString();
|
||||||
|
}
|
||||||
|
}
|
||||||
23
cockatrice/src/client/get_text_with_max.h
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
// custom QInputDialog::getText implementation that allows configuration of the max length
|
||||||
|
#ifndef GETTEXTWITHMAX_H
|
||||||
|
#define GETTEXTWITHMAX_H
|
||||||
|
|
||||||
|
#include "trice_limits.h"
|
||||||
|
|
||||||
|
#include <QInputDialog>
|
||||||
|
|
||||||
|
QString getTextWithMax(QWidget *parent,
|
||||||
|
const QString &title,
|
||||||
|
const QString &label,
|
||||||
|
QLineEdit::EchoMode echo = QLineEdit::Normal,
|
||||||
|
const QString &text = QString(),
|
||||||
|
bool *ok = nullptr,
|
||||||
|
int max = MAX_NAME_LENGTH,
|
||||||
|
Qt::WindowFlags flags = Qt::WindowFlags(),
|
||||||
|
Qt::InputMethodHints inputMethodHints = Qt::ImhNone);
|
||||||
|
static inline QString getTextWithMax(QWidget *parent, const QString &title, const QString &label, int max)
|
||||||
|
{
|
||||||
|
return getTextWithMax(parent, title, label, QLineEdit::Normal, QString(), nullptr, max);
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif // GETTEXTWITHMAX_H
|
||||||
@@ -1,4 +1,5 @@
|
|||||||
#include "releasechannel.h"
|
#include "release_channel.h"
|
||||||
|
|
||||||
#include "version_string.h"
|
#include "version_string.h"
|
||||||
|
|
||||||
#include <QJsonArray>
|
#include <QJsonArray>
|
||||||
@@ -6,6 +7,9 @@
|
|||||||
#include <QJsonObject>
|
#include <QJsonObject>
|
||||||
#include <QMessageBox>
|
#include <QMessageBox>
|
||||||
#include <QNetworkReply>
|
#include <QNetworkReply>
|
||||||
|
#include <QRegularExpression>
|
||||||
|
#include <QSysInfo>
|
||||||
|
#include <QtGlobal>
|
||||||
|
|
||||||
#define STABLERELEASE_URL "https://api.github.com/repos/Cockatrice/Cockatrice/releases/latest"
|
#define STABLERELEASE_URL "https://api.github.com/repos/Cockatrice/Cockatrice/releases/latest"
|
||||||
#define STABLEMANUALDOWNLOAD_URL "https://github.com/Cockatrice/Cockatrice/releases/latest"
|
#define STABLEMANUALDOWNLOAD_URL "https://github.com/Cockatrice/Cockatrice/releases/latest"
|
||||||
@@ -19,10 +23,9 @@
|
|||||||
|
|
||||||
int ReleaseChannel::sharedIndex = 0;
|
int ReleaseChannel::sharedIndex = 0;
|
||||||
|
|
||||||
ReleaseChannel::ReleaseChannel() : response(nullptr), lastRelease(nullptr)
|
ReleaseChannel::ReleaseChannel() : netMan(new QNetworkAccessManager(this)), response(nullptr), lastRelease(nullptr)
|
||||||
{
|
{
|
||||||
index = sharedIndex++;
|
index = sharedIndex++;
|
||||||
netMan = new QNetworkAccessManager(this);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ReleaseChannel::~ReleaseChannel()
|
ReleaseChannel::~ReleaseChannel()
|
||||||
@@ -38,41 +41,45 @@ void ReleaseChannel::checkForUpdates()
|
|||||||
connect(response, SIGNAL(finished()), this, SLOT(releaseListFinished()));
|
connect(response, SIGNAL(finished()), this, SLOT(releaseListFinished()));
|
||||||
}
|
}
|
||||||
|
|
||||||
#if defined(Q_OS_OSX)
|
// Different release channel checking functions for different operating systems
|
||||||
|
#if defined(Q_OS_MACOS)
|
||||||
bool ReleaseChannel::downloadMatchesCurrentOS(const QString &fileName)
|
bool ReleaseChannel::downloadMatchesCurrentOS(const QString &fileName)
|
||||||
{
|
{
|
||||||
return fileName.endsWith(".dmg");
|
static QRegularExpression version_regex("macOS-(\\d+)\\.(\\d+)");
|
||||||
}
|
auto match = version_regex.match(fileName);
|
||||||
|
if (!match.hasMatch()) {
|
||||||
#elif defined(Q_OS_WIN)
|
|
||||||
|
|
||||||
#include <QSysInfo>
|
|
||||||
|
|
||||||
bool ReleaseChannel::downloadMatchesCurrentOS(const QString &fileName)
|
|
||||||
{
|
|
||||||
QString wordSize = QSysInfo::buildAbi().split('-')[2];
|
|
||||||
QString arch;
|
|
||||||
|
|
||||||
if (wordSize == "llp64") {
|
|
||||||
arch = "win64";
|
|
||||||
} else if (wordSize == "ilp32") {
|
|
||||||
arch = "win32";
|
|
||||||
} else {
|
|
||||||
qWarning() << "Error checking for upgrade version: wordSize is" << wordSize;
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto exeName = arch + ".exe";
|
// older(smaller) releases are compatible with a newer or the same system version
|
||||||
return (fileName.endsWith(exeName));
|
int sys_maj = QSysInfo::productVersion().split(".")[0].toInt();
|
||||||
|
int sys_min = QSysInfo::productVersion().split(".")[1].toInt();
|
||||||
|
int rel_maj = match.captured(1).toInt();
|
||||||
|
int rel_min = match.captured(2).toInt();
|
||||||
|
return rel_maj < sys_maj || (rel_maj == sys_maj && rel_min <= sys_min);
|
||||||
|
}
|
||||||
|
#elif defined(Q_OS_WIN)
|
||||||
|
bool ReleaseChannel::downloadMatchesCurrentOS(const QString &fileName)
|
||||||
|
{
|
||||||
|
#if Q_PROCESSOR_WORDSIZE == 4
|
||||||
|
return fileName.contains("32bit");
|
||||||
|
#elif Q_PROCESSOR_WORDSIZE == 8
|
||||||
|
const QString &version = QSysInfo::productVersion();
|
||||||
|
if (version.startsWith("7") || version.startsWith("8")) {
|
||||||
|
return fileName.contains("Win7");
|
||||||
|
} else {
|
||||||
|
return fileName.contains("Win10");
|
||||||
|
}
|
||||||
|
#else
|
||||||
|
return false;
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
#else
|
#else
|
||||||
|
|
||||||
bool ReleaseChannel::downloadMatchesCurrentOS(const QString &)
|
bool ReleaseChannel::downloadMatchesCurrentOS(const QString &)
|
||||||
{
|
{
|
||||||
// If the OS doesn't fit one of the above #defines, then it will never match
|
// If the OS doesn't fit one of the above #defines, then it will never match
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
QString StableReleaseChannel::getManualDownloadUrl() const
|
QString StableReleaseChannel::getManualDownloadUrl() const
|
||||||
@@ -276,18 +283,21 @@ void BetaReleaseChannel::fileListFinished()
|
|||||||
bool needToUpdate = (QString::compare(shortHash, myHash, Qt::CaseInsensitive) != 0);
|
bool needToUpdate = (QString::compare(shortHash, myHash, Qt::CaseInsensitive) != 0);
|
||||||
bool compatibleVersion = false;
|
bool compatibleVersion = false;
|
||||||
|
|
||||||
foreach (QVariant file, resultList) {
|
QStringList resultUrlList{};
|
||||||
|
for (QVariant file : resultList) {
|
||||||
QVariantMap map = file.toMap();
|
QVariantMap map = file.toMap();
|
||||||
|
resultUrlList << map["browser_download_url"].toString();
|
||||||
|
}
|
||||||
|
|
||||||
QString url = map["browser_download_url"].toString();
|
resultUrlList.sort();
|
||||||
|
// iterate in reverse so the first item is the latest os version
|
||||||
if (!downloadMatchesCurrentOS(url))
|
for (auto url = resultUrlList.rbegin(); url < resultUrlList.rend(); ++url) {
|
||||||
continue;
|
if (downloadMatchesCurrentOS(*url)) {
|
||||||
|
compatibleVersion = true;
|
||||||
compatibleVersion = true;
|
lastRelease->setDownloadUrl(*url);
|
||||||
lastRelease->setDownloadUrl(url);
|
qDebug() << "Found compatible version url=" << *url;
|
||||||
qDebug() << "Found compatible version url=" << url;
|
break;
|
||||||
break;
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
emit finishedCheck(needToUpdate, compatibleVersion, lastRelease);
|
emit finishedCheck(needToUpdate, compatibleVersion, lastRelease);
|
||||||
@@ -78,7 +78,7 @@ class ReleaseChannel : public QObject
|
|||||||
{
|
{
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
public:
|
public:
|
||||||
ReleaseChannel();
|
explicit ReleaseChannel();
|
||||||
~ReleaseChannel() override;
|
~ReleaseChannel() override;
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
@@ -117,7 +117,7 @@ class StableReleaseChannel : public ReleaseChannel
|
|||||||
{
|
{
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
public:
|
public:
|
||||||
StableReleaseChannel() = default;
|
explicit StableReleaseChannel() = default;
|
||||||
~StableReleaseChannel() override = default;
|
~StableReleaseChannel() override = default;
|
||||||
|
|
||||||
QString getManualDownloadUrl() const override;
|
QString getManualDownloadUrl() const override;
|
||||||
193
cockatrice/src/client/network/replay_timeline_widget.cpp
Normal file
@@ -0,0 +1,193 @@
|
|||||||
|
#include "replay_timeline_widget.h"
|
||||||
|
|
||||||
|
#include <QPainter>
|
||||||
|
#include <QPainterPath>
|
||||||
|
#include <QPalette>
|
||||||
|
#include <QTimer>
|
||||||
|
|
||||||
|
ReplayTimelineWidget::ReplayTimelineWidget(QWidget *parent)
|
||||||
|
: QWidget(parent), maxBinValue(1), maxTime(1), timeScaleFactor(1.0), currentVisualTime(0), currentProcessedTime(0),
|
||||||
|
currentEvent(0)
|
||||||
|
{
|
||||||
|
replayTimer = new QTimer(this);
|
||||||
|
connect(replayTimer, SIGNAL(timeout()), this, SLOT(replayTimerTimeout()));
|
||||||
|
|
||||||
|
rewindBufferingTimer = new QTimer(this);
|
||||||
|
rewindBufferingTimer->setSingleShot(true);
|
||||||
|
connect(rewindBufferingTimer, &QTimer::timeout, this, &ReplayTimelineWidget::processRewind);
|
||||||
|
}
|
||||||
|
|
||||||
|
void ReplayTimelineWidget::setTimeline(const QList<int> &_replayTimeline)
|
||||||
|
{
|
||||||
|
replayTimeline = _replayTimeline;
|
||||||
|
histogram.clear();
|
||||||
|
int binEndTime = BIN_LENGTH - 1;
|
||||||
|
int binValue = 0;
|
||||||
|
for (int i : replayTimeline) {
|
||||||
|
if (i > binEndTime) {
|
||||||
|
histogram.append(binValue);
|
||||||
|
if (binValue > maxBinValue)
|
||||||
|
maxBinValue = binValue;
|
||||||
|
while (i > binEndTime + BIN_LENGTH) {
|
||||||
|
histogram.append(0);
|
||||||
|
binEndTime += BIN_LENGTH;
|
||||||
|
}
|
||||||
|
binValue = 1;
|
||||||
|
binEndTime += BIN_LENGTH;
|
||||||
|
} else
|
||||||
|
++binValue;
|
||||||
|
}
|
||||||
|
histogram.append(binValue);
|
||||||
|
if (!replayTimeline.isEmpty())
|
||||||
|
maxTime = replayTimeline.last();
|
||||||
|
|
||||||
|
update();
|
||||||
|
}
|
||||||
|
|
||||||
|
void ReplayTimelineWidget::paintEvent(QPaintEvent * /* event */)
|
||||||
|
{
|
||||||
|
QPainter painter(this);
|
||||||
|
painter.drawRect(0, 0, width() - 1, height() - 1);
|
||||||
|
|
||||||
|
qreal binWidth = (qreal)width() / histogram.size();
|
||||||
|
QPainterPath path;
|
||||||
|
path.moveTo(0, height() - 1);
|
||||||
|
for (int i = 0; i < histogram.size(); ++i)
|
||||||
|
path.lineTo(qRound(i * binWidth), (height() - 1) * (1.0 - (qreal)histogram[i] / maxBinValue));
|
||||||
|
path.lineTo(width() - 1, height() - 1);
|
||||||
|
path.lineTo(0, height() - 1);
|
||||||
|
painter.fillPath(path, Qt::black);
|
||||||
|
|
||||||
|
const QColor barColor = QColor::fromHsv(120, 255, 255, 100);
|
||||||
|
quint64 w = (quint64)(width() - 1) * (quint64)currentVisualTime / maxTime;
|
||||||
|
painter.fillRect(0, 0, static_cast<int>(w), height() - 1, barColor);
|
||||||
|
}
|
||||||
|
|
||||||
|
void ReplayTimelineWidget::mousePressEvent(QMouseEvent *event)
|
||||||
|
{
|
||||||
|
#if (QT_VERSION >= QT_VERSION_CHECK(6, 0, 0))
|
||||||
|
int newTime = static_cast<int>((qint64)maxTime * (qint64)event->position().x() / width());
|
||||||
|
#else
|
||||||
|
int newTime = static_cast<int>((qint64)maxTime * (qint64)event->x() / width());
|
||||||
|
#endif
|
||||||
|
// don't buffer rewinds from clicks, since clicks usually don't happen fast enough to require buffering
|
||||||
|
skipToTime(newTime, false);
|
||||||
|
}
|
||||||
|
|
||||||
|
void ReplayTimelineWidget::skipToTime(int newTime, bool doRewindBuffering)
|
||||||
|
{
|
||||||
|
// check boundary conditions
|
||||||
|
if (newTime < 0) {
|
||||||
|
newTime = 0;
|
||||||
|
}
|
||||||
|
if (newTime > maxTime) {
|
||||||
|
newTime = maxTime;
|
||||||
|
}
|
||||||
|
|
||||||
|
newTime -= newTime % TIMER_INTERVAL_MS; // Time should always be a multiple of the interval
|
||||||
|
|
||||||
|
const bool isBackwardsSkip = newTime < currentProcessedTime;
|
||||||
|
currentVisualTime = newTime;
|
||||||
|
|
||||||
|
if (isBackwardsSkip) {
|
||||||
|
handleBackwardsSkip(doRewindBuffering);
|
||||||
|
} else {
|
||||||
|
processNewEvents(FORWARD_SKIP);
|
||||||
|
}
|
||||||
|
|
||||||
|
update();
|
||||||
|
}
|
||||||
|
|
||||||
|
/// @param doRewindBuffering When true, if multiple backward skips are made in quick succession, only a single rewind
|
||||||
|
/// is processed at the end. When false, the backwards skip will always cause an immediate rewind
|
||||||
|
void ReplayTimelineWidget::handleBackwardsSkip(bool doRewindBuffering)
|
||||||
|
{
|
||||||
|
if (doRewindBuffering) {
|
||||||
|
// We use a one-shot timer to implement the rewind buffering.
|
||||||
|
// The rewind only happens once the timer runs out.
|
||||||
|
// If another backwards skip happens, the timer will just get reset instead of rewinding.
|
||||||
|
rewindBufferingTimer->stop();
|
||||||
|
rewindBufferingTimer->start(SettingsCache::instance().getRewindBufferingMs());
|
||||||
|
} else {
|
||||||
|
// otherwise, process the rewind immediately
|
||||||
|
processRewind();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void ReplayTimelineWidget::processRewind()
|
||||||
|
{
|
||||||
|
// stop any queued-up rewinds
|
||||||
|
rewindBufferingTimer->stop();
|
||||||
|
|
||||||
|
// process the rewind
|
||||||
|
currentEvent = 0;
|
||||||
|
emit rewound();
|
||||||
|
processNewEvents(BACKWARD_SKIP);
|
||||||
|
}
|
||||||
|
|
||||||
|
QSize ReplayTimelineWidget::sizeHint() const
|
||||||
|
{
|
||||||
|
return {-1, 50};
|
||||||
|
}
|
||||||
|
|
||||||
|
QSize ReplayTimelineWidget::minimumSizeHint() const
|
||||||
|
{
|
||||||
|
return {400, 50};
|
||||||
|
}
|
||||||
|
|
||||||
|
void ReplayTimelineWidget::replayTimerTimeout()
|
||||||
|
{
|
||||||
|
currentVisualTime += TIMER_INTERVAL_MS;
|
||||||
|
|
||||||
|
processNewEvents(NORMAL_PLAYBACK);
|
||||||
|
|
||||||
|
if (!(currentVisualTime % 1000))
|
||||||
|
update();
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Processes all unprocessed events up to the current time.
|
||||||
|
void ReplayTimelineWidget::processNewEvents(PlaybackMode playbackMode)
|
||||||
|
{
|
||||||
|
currentProcessedTime = currentVisualTime;
|
||||||
|
|
||||||
|
while ((currentEvent < replayTimeline.size()) && (replayTimeline[currentEvent] < currentProcessedTime)) {
|
||||||
|
Player::EventProcessingOptions options;
|
||||||
|
|
||||||
|
// backwards skip => always skip reveal windows
|
||||||
|
// forwards skip => skip reveal windows that don't happen within a big skip of the target
|
||||||
|
if (playbackMode == BACKWARD_SKIP || currentProcessedTime - replayTimeline[currentEvent] > BIG_SKIP_MS)
|
||||||
|
options |= Player::EventProcessingOption::SKIP_REVEAL_WINDOW;
|
||||||
|
|
||||||
|
// backwards skip => always skip tap animation
|
||||||
|
if (playbackMode == BACKWARD_SKIP)
|
||||||
|
options |= Player::EventProcessingOption::SKIP_TAP_ANIMATION;
|
||||||
|
|
||||||
|
emit processNextEvent(options);
|
||||||
|
++currentEvent;
|
||||||
|
}
|
||||||
|
if (currentEvent == replayTimeline.size()) {
|
||||||
|
emit replayFinished();
|
||||||
|
replayTimer->stop();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void ReplayTimelineWidget::setTimeScaleFactor(qreal _timeScaleFactor)
|
||||||
|
{
|
||||||
|
timeScaleFactor = _timeScaleFactor;
|
||||||
|
replayTimer->setInterval(static_cast<int>(TIMER_INTERVAL_MS / timeScaleFactor));
|
||||||
|
}
|
||||||
|
|
||||||
|
void ReplayTimelineWidget::startReplay()
|
||||||
|
{
|
||||||
|
replayTimer->start(static_cast<int>(TIMER_INTERVAL_MS / timeScaleFactor));
|
||||||
|
}
|
||||||
|
|
||||||
|
void ReplayTimelineWidget::stopReplay()
|
||||||
|
{
|
||||||
|
replayTimer->stop();
|
||||||
|
}
|
||||||
|
|
||||||
|
void ReplayTimelineWidget::skipByAmount(int amount)
|
||||||
|
{
|
||||||
|
skipToTime(currentVisualTime + amount, amount < 0);
|
||||||
|
}
|
||||||
74
cockatrice/src/client/network/replay_timeline_widget.h
Normal file
@@ -0,0 +1,74 @@
|
|||||||
|
#ifndef REPLAY_TIMELINE_WIDGET
|
||||||
|
#define REPLAY_TIMELINE_WIDGET
|
||||||
|
|
||||||
|
#include "../../game/player/player.h"
|
||||||
|
|
||||||
|
#include <QList>
|
||||||
|
#include <QMouseEvent>
|
||||||
|
#include <QWidget>
|
||||||
|
|
||||||
|
class QPaintEvent;
|
||||||
|
class QTimer;
|
||||||
|
|
||||||
|
class ReplayTimelineWidget : public QWidget
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
signals:
|
||||||
|
void processNextEvent(Player::EventProcessingOptions options);
|
||||||
|
void replayFinished();
|
||||||
|
void rewound();
|
||||||
|
|
||||||
|
private:
|
||||||
|
enum PlaybackMode
|
||||||
|
{
|
||||||
|
NORMAL_PLAYBACK,
|
||||||
|
FORWARD_SKIP,
|
||||||
|
BACKWARD_SKIP
|
||||||
|
};
|
||||||
|
|
||||||
|
static constexpr int TIMER_INTERVAL_MS = 200;
|
||||||
|
static constexpr int BIN_LENGTH = 5000;
|
||||||
|
|
||||||
|
QTimer *replayTimer;
|
||||||
|
QTimer *rewindBufferingTimer;
|
||||||
|
QList<int> replayTimeline;
|
||||||
|
QList<int> histogram;
|
||||||
|
int maxBinValue, maxTime;
|
||||||
|
qreal timeScaleFactor;
|
||||||
|
int currentVisualTime; // time currently displayed by the timeline
|
||||||
|
int currentProcessedTime; // time that events are currently processed up to. Could differ from visual time due to
|
||||||
|
// rewind buffering
|
||||||
|
int currentEvent;
|
||||||
|
|
||||||
|
void skipToTime(int newTime, bool doRewindBuffering);
|
||||||
|
void handleBackwardsSkip(bool doRewindBuffering);
|
||||||
|
void processRewind();
|
||||||
|
void processNewEvents(PlaybackMode playbackMode);
|
||||||
|
private slots:
|
||||||
|
void replayTimerTimeout();
|
||||||
|
|
||||||
|
public:
|
||||||
|
static constexpr int SMALL_SKIP_MS = 1000;
|
||||||
|
static constexpr int BIG_SKIP_MS = 10000;
|
||||||
|
static constexpr qreal FAST_FORWARD_SCALE_FACTOR = 10.0;
|
||||||
|
|
||||||
|
explicit ReplayTimelineWidget(QWidget *parent = nullptr);
|
||||||
|
void setTimeline(const QList<int> &_replayTimeline);
|
||||||
|
QSize sizeHint() const override;
|
||||||
|
QSize minimumSizeHint() const override;
|
||||||
|
void setTimeScaleFactor(qreal _timeScaleFactor);
|
||||||
|
int getCurrentEvent() const
|
||||||
|
{
|
||||||
|
return currentEvent;
|
||||||
|
}
|
||||||
|
public slots:
|
||||||
|
void startReplay();
|
||||||
|
void stopReplay();
|
||||||
|
void skipByAmount(int amount); // use a negative amount to skip backwards
|
||||||
|
|
||||||
|
protected:
|
||||||
|
void paintEvent(QPaintEvent *event) override;
|
||||||
|
void mousePressEvent(QMouseEvent *event) override;
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif
|
||||||
@@ -1,4 +1,6 @@
|
|||||||
#include "setsmodel.h"
|
#include "sets_model.h"
|
||||||
|
|
||||||
|
#include <QSortFilterProxyModel>
|
||||||
|
|
||||||
SetsModel::SetsModel(CardDatabase *_db, QObject *parent) : QAbstractTableModel(parent), sets(_db->getSetList())
|
SetsModel::SetsModel(CardDatabase *_db, QObject *parent) : QAbstractTableModel(parent), sets(_db->getSetList())
|
||||||
{
|
{
|
||||||
@@ -96,7 +98,7 @@ QVariant SetsModel::headerData(int section, Qt::Orientation orientation, int rol
|
|||||||
Qt::ItemFlags SetsModel::flags(const QModelIndex &index) const
|
Qt::ItemFlags SetsModel::flags(const QModelIndex &index) const
|
||||||
{
|
{
|
||||||
if (!index.isValid())
|
if (!index.isValid())
|
||||||
return 0;
|
return Qt::NoItemFlags;
|
||||||
|
|
||||||
Qt::ItemFlags flags = QAbstractTableModel::flags(index) | Qt::ItemIsDragEnabled | Qt::ItemIsDropEnabled;
|
Qt::ItemFlags flags = QAbstractTableModel::flags(index) | Qt::ItemIsDragEnabled | Qt::ItemIsDropEnabled;
|
||||||
|
|
||||||
@@ -193,14 +195,21 @@ void SetsModel::swapRows(int oldRow, int newRow)
|
|||||||
emit dataChanged(index(0, 0), index(rowCount() - 1, columnCount() - 1));
|
emit dataChanged(index(0, 0), index(rowCount() - 1, columnCount() - 1));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void SetsModel::restoreOriginalOrder()
|
||||||
|
{
|
||||||
|
int numRows = rowCount();
|
||||||
|
sets.defaultSort();
|
||||||
|
emit dataChanged(index(0, 0), index(numRows - 1, columnCount() - 1));
|
||||||
|
}
|
||||||
|
|
||||||
void SetsModel::sort(int column, Qt::SortOrder order)
|
void SetsModel::sort(int column, Qt::SortOrder order)
|
||||||
{
|
{
|
||||||
QMap<QString, CardSetPtr> setMap;
|
QMultiMap<QString, CardSetPtr> setMap;
|
||||||
int numRows = rowCount();
|
int numRows = rowCount();
|
||||||
int row;
|
int row;
|
||||||
|
|
||||||
for (row = 0; row < numRows; ++row)
|
for (row = 0; row < numRows; ++row)
|
||||||
setMap.insertMulti(index(row, column).data(SetsModel::SortRole).toString(), sets.at(row));
|
setMap.insert(index(row, column).data(SetsModel::SortRole).toString(), sets.at(row));
|
||||||
|
|
||||||
QList<CardSetPtr> tmp = setMap.values();
|
QList<CardSetPtr> tmp = setMap.values();
|
||||||
sets.clear();
|
sets.clear();
|
||||||
@@ -274,9 +283,15 @@ bool SetsDisplayModel::filterAcceptsRow(int sourceRow, const QModelIndex &source
|
|||||||
auto nameIndex = sourceModel()->index(sourceRow, SetsModel::LongNameCol, sourceParent);
|
auto nameIndex = sourceModel()->index(sourceRow, SetsModel::LongNameCol, sourceParent);
|
||||||
auto shortNameIndex = sourceModel()->index(sourceRow, SetsModel::ShortNameCol, sourceParent);
|
auto shortNameIndex = sourceModel()->index(sourceRow, SetsModel::ShortNameCol, sourceParent);
|
||||||
|
|
||||||
return (sourceModel()->data(typeIndex).toString().contains(filterRegExp()) ||
|
#if (QT_VERSION >= QT_VERSION_CHECK(5, 12, 0))
|
||||||
sourceModel()->data(nameIndex).toString().contains(filterRegExp()) ||
|
const auto filter = filterRegularExpression();
|
||||||
sourceModel()->data(shortNameIndex).toString().contains(filterRegExp()));
|
#else
|
||||||
|
const auto filter = filterRegExp();
|
||||||
|
#endif
|
||||||
|
|
||||||
|
return (sourceModel()->data(typeIndex).toString().contains(filter) ||
|
||||||
|
sourceModel()->data(nameIndex).toString().contains(filter) ||
|
||||||
|
sourceModel()->data(shortNameIndex).toString().contains(filter));
|
||||||
}
|
}
|
||||||
|
|
||||||
bool SetsDisplayModel::lessThan(const QModelIndex &left, const QModelIndex &right) const
|
bool SetsDisplayModel::lessThan(const QModelIndex &left, const QModelIndex &right) const
|
||||||
@@ -1,7 +1,8 @@
|
|||||||
#ifndef SETSMODEL_H
|
#ifndef SETSMODEL_H
|
||||||
#define SETSMODEL_H
|
#define SETSMODEL_H
|
||||||
|
|
||||||
#include "carddatabase.h"
|
#include "../../game/cards/card_database.h"
|
||||||
|
|
||||||
#include <QAbstractTableModel>
|
#include <QAbstractTableModel>
|
||||||
#include <QMimeData>
|
#include <QMimeData>
|
||||||
#include <QSet>
|
#include <QSet>
|
||||||
@@ -48,14 +49,15 @@ public:
|
|||||||
LongNameCol,
|
LongNameCol,
|
||||||
ShortNameCol,
|
ShortNameCol,
|
||||||
SetTypeCol,
|
SetTypeCol,
|
||||||
ReleaseDateCol
|
ReleaseDateCol,
|
||||||
|
PriorityCol
|
||||||
};
|
};
|
||||||
enum Role
|
enum Role
|
||||||
{
|
{
|
||||||
SortRole = Qt::UserRole
|
SortRole = Qt::UserRole
|
||||||
};
|
};
|
||||||
|
|
||||||
SetsModel(CardDatabase *_db, QObject *parent = 0);
|
SetsModel(CardDatabase *_db, QObject *parent = nullptr);
|
||||||
~SetsModel();
|
~SetsModel();
|
||||||
int rowCount(const QModelIndex &parent = QModelIndex()) const;
|
int rowCount(const QModelIndex &parent = QModelIndex()) const;
|
||||||
int columnCount(const QModelIndex &parent = QModelIndex()) const
|
int columnCount(const QModelIndex &parent = QModelIndex()) const
|
||||||
@@ -79,6 +81,7 @@ public:
|
|||||||
void sort(int column, Qt::SortOrder order = Qt::AscendingOrder);
|
void sort(int column, Qt::SortOrder order = Qt::AscendingOrder);
|
||||||
void save(CardDatabase *db);
|
void save(CardDatabase *db);
|
||||||
void restore(CardDatabase *db);
|
void restore(CardDatabase *db);
|
||||||
|
void restoreOriginalOrder();
|
||||||
};
|
};
|
||||||
|
|
||||||
class SetsDisplayModel : public QSortFilterProxyModel
|
class SetsDisplayModel : public QSortFilterProxyModel
|
||||||
@@ -1,3 +1,11 @@
|
|||||||
|
#include "spoiler_background_updater.h"
|
||||||
|
|
||||||
|
#include "../../game/cards/card_database.h"
|
||||||
|
#include "../../game/cards/card_database_manager.h"
|
||||||
|
#include "../../main.h"
|
||||||
|
#include "../../settings/cache_settings.h"
|
||||||
|
#include "../ui/window_main.h"
|
||||||
|
|
||||||
#include <QApplication>
|
#include <QApplication>
|
||||||
#include <QCryptographicHash>
|
#include <QCryptographicHash>
|
||||||
#include <QDateTime>
|
#include <QDateTime>
|
||||||
@@ -9,24 +17,18 @@
|
|||||||
#include <QUrl>
|
#include <QUrl>
|
||||||
#include <QtConcurrent>
|
#include <QtConcurrent>
|
||||||
|
|
||||||
#include "carddatabase.h"
|
|
||||||
#include "main.h"
|
|
||||||
#include "settingscache.h"
|
|
||||||
#include "spoilerbackgroundupdater.h"
|
|
||||||
#include "window_main.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"
|
||||||
|
|
||||||
SpoilerBackgroundUpdater::SpoilerBackgroundUpdater(QObject *apParent) : QObject(apParent), cardUpdateProcess(nullptr)
|
SpoilerBackgroundUpdater::SpoilerBackgroundUpdater(QObject *apParent) : QObject(apParent), cardUpdateProcess(nullptr)
|
||||||
{
|
{
|
||||||
isSpoilerDownloadEnabled = settingsCache->getDownloadSpoilersStatus();
|
isSpoilerDownloadEnabled = SettingsCache::instance().getDownloadSpoilersStatus();
|
||||||
if (isSpoilerDownloadEnabled) {
|
if (isSpoilerDownloadEnabled) {
|
||||||
// Start the process of checking if we're in spoiler season
|
// Start the process of checking if we're in spoiler season
|
||||||
// File exists means we're in spoiler season
|
// File exists means we're in spoiler season
|
||||||
// We will load the database before attempting to download spoilers, incase they fail
|
|
||||||
QtConcurrent::run(db, &CardDatabase::loadCardDatabases);
|
|
||||||
startSpoilerDownloadProcess(SPOILERS_STATUS_URL, false);
|
startSpoilerDownloadProcess(SPOILERS_STATUS_URL, false);
|
||||||
|
} else {
|
||||||
|
qDebug() << "Spoilers Disabled";
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -72,7 +74,7 @@ void SpoilerBackgroundUpdater::actDownloadFinishedSpoilersFile()
|
|||||||
|
|
||||||
bool SpoilerBackgroundUpdater::deleteSpoilerFile()
|
bool SpoilerBackgroundUpdater::deleteSpoilerFile()
|
||||||
{
|
{
|
||||||
QString fileName = settingsCache->getSpoilerCardDatabasePath();
|
QString fileName = SettingsCache::instance().getSpoilerCardDatabasePath();
|
||||||
QFileInfo fi(fileName);
|
QFileInfo fi(fileName);
|
||||||
QDir fileDir(fi.path());
|
QDir fileDir(fi.path());
|
||||||
QFile file(fileName);
|
QFile file(fileName);
|
||||||
@@ -113,7 +115,7 @@ void SpoilerBackgroundUpdater::actCheckIfSpoilerSeasonEnabled()
|
|||||||
emit spoilerCheckerDone();
|
emit spoilerCheckerDone();
|
||||||
} else {
|
} else {
|
||||||
if (trayIcon) {
|
if (trayIcon) {
|
||||||
trayIcon->showMessage(tr("Spoilers download failed"), tr("Error") + " " + errorCode);
|
trayIcon->showMessage(tr("Spoilers download failed"), tr("Error") + " " + (short)errorCode);
|
||||||
}
|
}
|
||||||
|
|
||||||
qDebug() << "Spoiler download failed with reason" << errorCode;
|
qDebug() << "Spoiler download failed with reason" << errorCode;
|
||||||
@@ -123,7 +125,7 @@ void SpoilerBackgroundUpdater::actCheckIfSpoilerSeasonEnabled()
|
|||||||
|
|
||||||
bool SpoilerBackgroundUpdater::saveDownloadedFile(QByteArray data)
|
bool SpoilerBackgroundUpdater::saveDownloadedFile(QByteArray data)
|
||||||
{
|
{
|
||||||
QString fileName = settingsCache->getSpoilerCardDatabasePath();
|
QString fileName = SettingsCache::instance().getSpoilerCardDatabasePath();
|
||||||
QFileInfo fi(fileName);
|
QFileInfo fi(fileName);
|
||||||
QDir fileDir(fi.path());
|
QDir fileDir(fi.path());
|
||||||
|
|
||||||
@@ -158,7 +160,7 @@ bool SpoilerBackgroundUpdater::saveDownloadedFile(QByteArray data)
|
|||||||
|
|
||||||
// Data written, so reload the card database
|
// Data written, so reload the card database
|
||||||
qDebug() << "Spoiler Service Data Written";
|
qDebug() << "Spoiler Service Data Written";
|
||||||
QtConcurrent::run(db, &CardDatabase::loadCardDatabases);
|
const auto reloadOk = QtConcurrent::run([] { CardDatabaseManager::getInstance()->loadCardDatabases(); });
|
||||||
|
|
||||||
// If the user has notifications enabled, let them know
|
// If the user has notifications enabled, let them know
|
||||||
// when the database was last updated
|
// when the database was last updated
|
||||||
@@ -166,12 +168,16 @@ bool SpoilerBackgroundUpdater::saveDownloadedFile(QByteArray data)
|
|||||||
QList<QByteArray> lines = data.split('\n');
|
QList<QByteArray> lines = data.split('\n');
|
||||||
|
|
||||||
foreach (QByteArray line, lines) {
|
foreach (QByteArray line, lines) {
|
||||||
if (line.contains("created:")) {
|
if (line.contains("Created At:")) {
|
||||||
QString timeStamp = QString(line).replace("created:", "").trimmed();
|
QString timeStamp = QString(line).replace("Created At:", "").trimmed();
|
||||||
timeStamp.chop(6); // Remove " (UTC)"
|
timeStamp.chop(6); // Remove " (UTC)"
|
||||||
|
|
||||||
auto utcTime = QLocale().toDateTime(timeStamp, "ddd, MMM dd yyyy, hh:mm:ss");
|
auto utcTime = QLocale().toDateTime(timeStamp, "ddd, MMM dd yyyy, hh:mm:ss");
|
||||||
|
#if QT_VERSION >= QT_VERSION_CHECK(6, 7, 0)
|
||||||
|
utcTime.setTimeZone(QTimeZone::UTC);
|
||||||
|
#else
|
||||||
utcTime.setTimeSpec(Qt::UTC);
|
utcTime.setTimeSpec(Qt::UTC);
|
||||||
|
#endif
|
||||||
|
|
||||||
QString localTime = utcTime.toLocalTime().toString("MMM d, hh:mm");
|
QString localTime = utcTime.toLocalTime().toString("MMM d, hh:mm");
|
||||||
|
|
||||||
158
cockatrice/src/client/sound_engine.cpp
Normal file
@@ -0,0 +1,158 @@
|
|||||||
|
#include "sound_engine.h"
|
||||||
|
|
||||||
|
#include "../settings/cache_settings.h"
|
||||||
|
|
||||||
|
#include <QDir>
|
||||||
|
#include <QMediaPlayer>
|
||||||
|
|
||||||
|
#if (QT_VERSION >= QT_VERSION_CHECK(6, 0, 0))
|
||||||
|
#include <QAudioOutput>
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#define DEFAULT_THEME_NAME "Default"
|
||||||
|
#define TEST_SOUND_FILENAME "player_join"
|
||||||
|
|
||||||
|
SoundEngine::SoundEngine(QObject *parent) : QObject(parent), player(nullptr)
|
||||||
|
{
|
||||||
|
ensureThemeDirectoryExists();
|
||||||
|
connect(&SettingsCache::instance(), SIGNAL(soundThemeChanged()), this, SLOT(themeChangedSlot()));
|
||||||
|
connect(&SettingsCache::instance(), SIGNAL(soundEnabledChanged()), this, SLOT(soundEnabledChanged()));
|
||||||
|
|
||||||
|
soundEnabledChanged();
|
||||||
|
themeChangedSlot();
|
||||||
|
}
|
||||||
|
|
||||||
|
SoundEngine::~SoundEngine()
|
||||||
|
{
|
||||||
|
if (player) {
|
||||||
|
player->deleteLater();
|
||||||
|
player = nullptr;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void SoundEngine::soundEnabledChanged()
|
||||||
|
{
|
||||||
|
if (SettingsCache::instance().getSoundEnabled()) {
|
||||||
|
qDebug() << "SoundEngine: enabling sound with" << audioData.size() << "sounds";
|
||||||
|
if (!player) {
|
||||||
|
player = new QMediaPlayer;
|
||||||
|
#if (QT_VERSION >= QT_VERSION_CHECK(6, 0, 0))
|
||||||
|
auto qAudioOutput = new QAudioOutput;
|
||||||
|
player->setAudioOutput(qAudioOutput);
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
qDebug() << "SoundEngine: disabling sound";
|
||||||
|
if (player) {
|
||||||
|
player->stop();
|
||||||
|
player->deleteLater();
|
||||||
|
player = nullptr;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void SoundEngine::playSound(const QString &fileName)
|
||||||
|
{
|
||||||
|
if (!player) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!audioData.contains(fileName)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
player->stop();
|
||||||
|
int volumeSliderValue = SettingsCache::instance().getMasterVolume();
|
||||||
|
#if (QT_VERSION >= QT_VERSION_CHECK(6, 0, 0))
|
||||||
|
player->audioOutput()->setVolume(qreal(volumeSliderValue) / 100);
|
||||||
|
player->setSource(QUrl::fromLocalFile(audioData[fileName]));
|
||||||
|
#else
|
||||||
|
player->setVolume(volumeSliderValue);
|
||||||
|
player->setMedia(QUrl::fromLocalFile(audioData[fileName]));
|
||||||
|
#endif
|
||||||
|
player->play();
|
||||||
|
}
|
||||||
|
|
||||||
|
void SoundEngine::testSound()
|
||||||
|
{
|
||||||
|
playSound(TEST_SOUND_FILENAME);
|
||||||
|
}
|
||||||
|
|
||||||
|
void SoundEngine::ensureThemeDirectoryExists()
|
||||||
|
{
|
||||||
|
if (SettingsCache::instance().getSoundThemeName().isEmpty() ||
|
||||||
|
!getAvailableThemes().contains(SettingsCache::instance().getSoundThemeName())) {
|
||||||
|
qDebug() << "Sounds theme name not set, setting default value";
|
||||||
|
SettingsCache::instance().setSoundThemeName(DEFAULT_THEME_NAME);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
QStringMap &SoundEngine::getAvailableThemes()
|
||||||
|
{
|
||||||
|
QDir dir;
|
||||||
|
availableThemes.clear();
|
||||||
|
|
||||||
|
// load themes from user profile dir
|
||||||
|
|
||||||
|
dir.setPath(SettingsCache::instance().getDataPath() + "/sounds");
|
||||||
|
|
||||||
|
for (const QString &themeName : dir.entryList(QDir::AllDirs | QDir::NoDotAndDotDot, QDir::Name)) {
|
||||||
|
if (!availableThemes.contains(themeName))
|
||||||
|
availableThemes.insert(themeName, dir.absoluteFilePath(themeName));
|
||||||
|
}
|
||||||
|
|
||||||
|
// load themes from cockatrice system dir
|
||||||
|
dir.setPath(qApp->applicationDirPath() +
|
||||||
|
#ifdef Q_OS_MAC
|
||||||
|
"/../Resources/sounds"
|
||||||
|
#elif defined(Q_OS_WIN)
|
||||||
|
"/sounds"
|
||||||
|
#else // linux
|
||||||
|
"/../share/cockatrice/sounds"
|
||||||
|
#endif
|
||||||
|
);
|
||||||
|
|
||||||
|
for (const QString &themeName : dir.entryList(QDir::AllDirs | QDir::NoDotAndDotDot, QDir::Name)) {
|
||||||
|
if (!availableThemes.contains(themeName))
|
||||||
|
availableThemes.insert(themeName, dir.absoluteFilePath(themeName));
|
||||||
|
}
|
||||||
|
|
||||||
|
return availableThemes;
|
||||||
|
}
|
||||||
|
|
||||||
|
void SoundEngine::themeChangedSlot()
|
||||||
|
{
|
||||||
|
QString themeName = SettingsCache::instance().getSoundThemeName();
|
||||||
|
qDebug() << "Sound theme changed:" << themeName;
|
||||||
|
|
||||||
|
QDir dir = getAvailableThemes().value(themeName);
|
||||||
|
|
||||||
|
audioData.clear();
|
||||||
|
|
||||||
|
static const QStringList extensions = {".wav", ".mp3", ".ogg"};
|
||||||
|
static const QStringList fileNames = {
|
||||||
|
// Phases
|
||||||
|
"untap_step", "upkeep_step", "draw_step", "main_1", "start_combat", "attack_step", "block_step", "damage_step",
|
||||||
|
"end_combat", "main_2", "end_step",
|
||||||
|
// Game Actions
|
||||||
|
"draw_card", "play_card", "tap_card", "untap_card", "shuffle", "roll_dice", "life_change",
|
||||||
|
// Player
|
||||||
|
"player_join", "player_leave", "player_disconnect", "player_reconnect", "player_concede",
|
||||||
|
// Spectator
|
||||||
|
"spectator_join", "spectator_leave",
|
||||||
|
// Buddy
|
||||||
|
"buddy_join", "buddy_leave",
|
||||||
|
// Chat & UI
|
||||||
|
"chat_mention", "all_mention", "private_message"};
|
||||||
|
|
||||||
|
for (const QString &extension : extensions) {
|
||||||
|
for (const QString &name : fileNames) {
|
||||||
|
QFile file(dir.filePath(name + extension));
|
||||||
|
if (file.exists()) {
|
||||||
|
audioData.insert(name, file.fileName());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
soundEnabledChanged();
|
||||||
|
}
|
||||||
@@ -1,8 +1,8 @@
|
|||||||
#ifndef SOUNDENGINE_H
|
#ifndef SOUNDENGINE_H
|
||||||
#define SOUNDENGINE_H
|
#define SOUNDENGINE_H
|
||||||
|
|
||||||
#include <QDir>
|
|
||||||
#include <QMap>
|
#include <QMap>
|
||||||
|
#include <QMediaPlayer>
|
||||||
#include <QObject>
|
#include <QObject>
|
||||||
#include <QString>
|
#include <QString>
|
||||||
|
|
||||||
@@ -15,16 +15,15 @@ class SoundEngine : public QObject
|
|||||||
{
|
{
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
public:
|
public:
|
||||||
SoundEngine(QObject *parent = 0);
|
explicit SoundEngine(QObject *parent = nullptr);
|
||||||
~SoundEngine();
|
~SoundEngine() override;
|
||||||
void playSound(QString fileName);
|
void playSound(const QString &fileName);
|
||||||
QStringMap &getAvailableThemes();
|
QStringMap &getAvailableThemes();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
QMap<QString, QByteArray> audioData;
|
|
||||||
QBuffer *inputBuffer;
|
|
||||||
QAudioOutput *player;
|
|
||||||
QStringMap availableThemes;
|
QStringMap availableThemes;
|
||||||
|
QMap<QString, QString> audioData;
|
||||||
|
QMediaPlayer *player;
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
void ensureThemeDirectoryExists();
|
void ensureThemeDirectoryExists();
|
||||||
@@ -1,8 +1,11 @@
|
|||||||
#include "tab.h"
|
#include "tab.h"
|
||||||
#include "cardinfowidget.h"
|
|
||||||
|
#include "../ui/widgets/cards/card_info_display_widget.h"
|
||||||
|
|
||||||
#include <QApplication>
|
#include <QApplication>
|
||||||
#include <QDebug>
|
#include <QDebug>
|
||||||
#include <QDesktopWidget>
|
#include <QScreen>
|
||||||
|
|
||||||
Tab::Tab(TabSupervisor *_tabSupervisor, QWidget *parent)
|
Tab::Tab(TabSupervisor *_tabSupervisor, QWidget *parent)
|
||||||
: QMainWindow(parent), tabSupervisor(_tabSupervisor), contentsChanged(false), infoPopup(0)
|
: QMainWindow(parent), tabSupervisor(_tabSupervisor), contentsChanged(false), infoPopup(0)
|
||||||
{
|
{
|
||||||
@@ -15,10 +18,11 @@ void Tab::showCardInfoPopup(const QPoint &pos, const QString &cardName)
|
|||||||
infoPopup->deleteLater();
|
infoPopup->deleteLater();
|
||||||
}
|
}
|
||||||
currentCardName = cardName;
|
currentCardName = cardName;
|
||||||
infoPopup = new CardInfoWidget(
|
infoPopup = new CardInfoDisplayWidget(
|
||||||
cardName, 0, Qt::Widget | Qt::FramelessWindowHint | Qt::X11BypassWindowManagerHint | Qt::WindowStaysOnTopHint);
|
cardName, 0, Qt::Widget | Qt::FramelessWindowHint | Qt::X11BypassWindowManagerHint | Qt::WindowStaysOnTopHint);
|
||||||
infoPopup->setAttribute(Qt::WA_TransparentForMouseEvents);
|
infoPopup->setAttribute(Qt::WA_TransparentForMouseEvents);
|
||||||
QRect screenRect = qApp->desktop()->screenGeometry(this);
|
|
||||||
|
auto screenRect = qApp->primaryScreen()->geometry();
|
||||||
infoPopup->move(qMax(screenRect.left(), qMin(pos.x() - infoPopup->width() / 2,
|
infoPopup->move(qMax(screenRect.left(), qMin(pos.x() - infoPopup->width() / 2,
|
||||||
screenRect.left() + screenRect.width() - infoPopup->width())),
|
screenRect.left() + screenRect.width() - infoPopup->width())),
|
||||||
qMax(screenRect.top(), qMin(pos.y() - infoPopup->height() / 2,
|
qMax(screenRect.top(), qMin(pos.y() - infoPopup->height() / 2,
|
||||||
@@ -5,7 +5,7 @@
|
|||||||
|
|
||||||
class QMenu;
|
class QMenu;
|
||||||
class TabSupervisor;
|
class TabSupervisor;
|
||||||
class CardInfoWidget;
|
class CardInfoDisplayWidget;
|
||||||
|
|
||||||
class Tab : public QMainWindow
|
class Tab : public QMainWindow
|
||||||
{
|
{
|
||||||
@@ -27,11 +27,11 @@ protected slots:
|
|||||||
private:
|
private:
|
||||||
QString currentCardName;
|
QString currentCardName;
|
||||||
bool contentsChanged;
|
bool contentsChanged;
|
||||||
CardInfoWidget *infoPopup;
|
CardInfoDisplayWidget *infoPopup;
|
||||||
QList<QMenu *> tabMenus;
|
QList<QMenu *> tabMenus;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
Tab(TabSupervisor *_tabSupervisor, QWidget *parent = 0);
|
Tab(TabSupervisor *_tabSupervisor, QWidget *parent = nullptr);
|
||||||
const QList<QMenu *> &getTabMenus() const
|
const QList<QMenu *> &getTabMenus() const
|
||||||
{
|
{
|
||||||
return tabMenus;
|
return tabMenus;
|
||||||
@@ -1,20 +1,22 @@
|
|||||||
#include "tab_userlists.h"
|
#include "tab_account.h"
|
||||||
#include "abstractclient.h"
|
|
||||||
#include "soundengine.h"
|
|
||||||
#include "userinfobox.h"
|
|
||||||
#include "userlist.h"
|
|
||||||
#include <QHBoxLayout>
|
|
||||||
#include <QLineEdit>
|
|
||||||
#include <QPushButton>
|
|
||||||
#include <QVBoxLayout>
|
|
||||||
|
|
||||||
|
#include "../../deck/custom_line_edit.h"
|
||||||
|
#include "../../server/pending_command.h"
|
||||||
|
#include "../../server/user/user_info_box.h"
|
||||||
|
#include "../../server/user/user_list.h"
|
||||||
|
#include "../game_logic/abstract_client.h"
|
||||||
|
#include "../sound_engine.h"
|
||||||
#include "pb/event_add_to_list.pb.h"
|
#include "pb/event_add_to_list.pb.h"
|
||||||
#include "pb/event_remove_from_list.pb.h"
|
#include "pb/event_remove_from_list.pb.h"
|
||||||
#include "pb/event_user_joined.pb.h"
|
#include "pb/event_user_joined.pb.h"
|
||||||
#include "pb/event_user_left.pb.h"
|
#include "pb/event_user_left.pb.h"
|
||||||
#include "pb/response_list_users.pb.h"
|
#include "pb/response_list_users.pb.h"
|
||||||
#include "pb/session_commands.pb.h"
|
#include "pb/session_commands.pb.h"
|
||||||
#include "pending_command.h"
|
#include "trice_limits.h"
|
||||||
|
|
||||||
|
#include <QHBoxLayout>
|
||||||
|
#include <QPushButton>
|
||||||
|
#include <QVBoxLayout>
|
||||||
|
|
||||||
TabUserLists::TabUserLists(TabSupervisor *_tabSupervisor,
|
TabUserLists::TabUserLists(TabSupervisor *_tabSupervisor,
|
||||||
AbstractClient *_client,
|
AbstractClient *_client,
|
||||||
@@ -58,7 +60,8 @@ TabUserLists::TabUserLists(TabSupervisor *_tabSupervisor,
|
|||||||
vbox->addWidget(allUsersList);
|
vbox->addWidget(allUsersList);
|
||||||
|
|
||||||
QHBoxLayout *addToBuddyList = new QHBoxLayout;
|
QHBoxLayout *addToBuddyList = new QHBoxLayout;
|
||||||
addBuddyEdit = new QLineEdit;
|
addBuddyEdit = new LineEditUnfocusable;
|
||||||
|
addBuddyEdit->setMaxLength(MAX_NAME_LENGTH);
|
||||||
addBuddyEdit->setPlaceholderText(tr("Add to Buddy List"));
|
addBuddyEdit->setPlaceholderText(tr("Add to Buddy List"));
|
||||||
connect(addBuddyEdit, SIGNAL(returnPressed()), this, SLOT(addToBuddyList()));
|
connect(addBuddyEdit, SIGNAL(returnPressed()), this, SLOT(addToBuddyList()));
|
||||||
QPushButton *addBuddyButton = new QPushButton("Add");
|
QPushButton *addBuddyButton = new QPushButton("Add");
|
||||||
@@ -67,7 +70,8 @@ TabUserLists::TabUserLists(TabSupervisor *_tabSupervisor,
|
|||||||
addToBuddyList->addWidget(addBuddyButton);
|
addToBuddyList->addWidget(addBuddyButton);
|
||||||
|
|
||||||
QHBoxLayout *addToIgnoreList = new QHBoxLayout;
|
QHBoxLayout *addToIgnoreList = new QHBoxLayout;
|
||||||
addIgnoreEdit = new QLineEdit;
|
addIgnoreEdit = new LineEditUnfocusable;
|
||||||
|
addIgnoreEdit->setMaxLength(MAX_NAME_LENGTH);
|
||||||
addIgnoreEdit->setPlaceholderText(tr("Add to Ignore List"));
|
addIgnoreEdit->setPlaceholderText(tr("Add to Ignore List"));
|
||||||
connect(addIgnoreEdit, SIGNAL(returnPressed()), this, SLOT(addToIgnoreList()));
|
connect(addIgnoreEdit, SIGNAL(returnPressed()), this, SLOT(addToIgnoreList()));
|
||||||
QPushButton *addIgnoreButton = new QPushButton("Add");
|
QPushButton *addIgnoreButton = new QPushButton("Add");
|
||||||
@@ -1,13 +1,13 @@
|
|||||||
#ifndef TAB_USERLISTS_H
|
#ifndef TAB_ACCOUNT_H
|
||||||
#define TAB_USERLISTS_H
|
#define TAB_ACCOUNT_H
|
||||||
|
|
||||||
#include "pb/serverinfo_user.pb.h"
|
#include "pb/serverinfo_user.pb.h"
|
||||||
#include "tab.h"
|
#include "tab.h"
|
||||||
#include <QLineEdit>
|
|
||||||
|
|
||||||
class AbstractClient;
|
class AbstractClient;
|
||||||
class UserList;
|
class UserList;
|
||||||
class UserInfoBox;
|
class UserInfoBox;
|
||||||
|
class LineEditUnfocusable;
|
||||||
|
|
||||||
class Event_ListRooms;
|
class Event_ListRooms;
|
||||||
class Event_UserJoined;
|
class Event_UserJoined;
|
||||||
@@ -41,15 +41,15 @@ private:
|
|||||||
UserList *buddyList;
|
UserList *buddyList;
|
||||||
UserList *ignoreList;
|
UserList *ignoreList;
|
||||||
UserInfoBox *userInfoBox;
|
UserInfoBox *userInfoBox;
|
||||||
QLineEdit *addBuddyEdit;
|
LineEditUnfocusable *addBuddyEdit;
|
||||||
QLineEdit *addIgnoreEdit;
|
LineEditUnfocusable *addIgnoreEdit;
|
||||||
void addToList(const std::string &listName, const QString &userName);
|
void addToList(const std::string &listName, const QString &userName);
|
||||||
|
|
||||||
public:
|
public:
|
||||||
TabUserLists(TabSupervisor *_tabSupervisor,
|
TabUserLists(TabSupervisor *_tabSupervisor,
|
||||||
AbstractClient *_client,
|
AbstractClient *_client,
|
||||||
const ServerInfo_User &userInfo,
|
const ServerInfo_User &userInfo,
|
||||||
QWidget *parent = 0);
|
QWidget *parent = nullptr);
|
||||||
void retranslateUi();
|
void retranslateUi();
|
||||||
QString getTabText() const
|
QString getTabText() const
|
||||||
{
|
{
|
||||||
@@ -1,5 +1,9 @@
|
|||||||
#include "tab_admin.h"
|
#include "tab_admin.h"
|
||||||
#include "abstractclient.h"
|
|
||||||
|
#include "../game_logic/abstract_client.h"
|
||||||
|
#include "pb/admin_commands.pb.h"
|
||||||
|
#include "trice_limits.h"
|
||||||
|
|
||||||
#include <QDialogButtonBox>
|
#include <QDialogButtonBox>
|
||||||
#include <QGridLayout>
|
#include <QGridLayout>
|
||||||
#include <QGroupBox>
|
#include <QGroupBox>
|
||||||
@@ -11,12 +15,11 @@
|
|||||||
#include <QSpinBox>
|
#include <QSpinBox>
|
||||||
#include <QVBoxLayout>
|
#include <QVBoxLayout>
|
||||||
|
|
||||||
#include "pb/admin_commands.pb.h"
|
|
||||||
|
|
||||||
ShutdownDialog::ShutdownDialog(QWidget *parent) : QDialog(parent)
|
ShutdownDialog::ShutdownDialog(QWidget *parent) : QDialog(parent)
|
||||||
{
|
{
|
||||||
QLabel *reasonLabel = new QLabel(tr("&Reason for shutdown:"));
|
QLabel *reasonLabel = new QLabel(tr("&Reason for shutdown:"));
|
||||||
reasonEdit = new QLineEdit;
|
reasonEdit = new QLineEdit;
|
||||||
|
reasonEdit->setMaxLength(MAX_TEXT_LENGTH);
|
||||||
reasonLabel->setBuddy(reasonEdit);
|
reasonLabel->setBuddy(reasonEdit);
|
||||||
QLabel *minutesLabel = new QLabel(tr("&Time until shutdown (minutes):"));
|
QLabel *minutesLabel = new QLabel(tr("&Time until shutdown (minutes):"));
|
||||||
minutesEdit = new QSpinBox;
|
minutesEdit = new QSpinBox;
|
||||||
@@ -2,6 +2,7 @@
|
|||||||
#define TAB_ADMIN_H
|
#define TAB_ADMIN_H
|
||||||
|
|
||||||
#include "tab.h"
|
#include "tab.h"
|
||||||
|
|
||||||
#include <QDialog>
|
#include <QDialog>
|
||||||
|
|
||||||
class AbstractClient;
|
class AbstractClient;
|
||||||
@@ -19,7 +20,7 @@ private:
|
|||||||
QSpinBox *minutesEdit;
|
QSpinBox *minutesEdit;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
ShutdownDialog(QWidget *parent = 0);
|
ShutdownDialog(QWidget *parent = nullptr);
|
||||||
QString getReason() const;
|
QString getReason() const;
|
||||||
int getMinutes() const;
|
int getMinutes() const;
|
||||||
};
|
};
|
||||||
@@ -45,7 +46,7 @@ private slots:
|
|||||||
void actLock();
|
void actLock();
|
||||||
|
|
||||||
public:
|
public:
|
||||||
TabAdmin(TabSupervisor *_tabSupervisor, AbstractClient *_client, bool _fullAdmin, QWidget *parent = 0);
|
TabAdmin(TabSupervisor *_tabSupervisor, AbstractClient *_client, bool _fullAdmin, QWidget *parent = nullptr);
|
||||||
void retranslateUi();
|
void retranslateUi();
|
||||||
QString getTabText() const
|
QString getTabText() const
|
||||||
{
|
{
|
||||||
@@ -1,20 +1,21 @@
|
|||||||
#ifndef WINDOW_DECKEDITOR_H
|
#ifndef WINDOW_DECKEDITOR_H
|
||||||
#define WINDOW_DECKEDITOR_H
|
#define WINDOW_DECKEDITOR_H
|
||||||
|
|
||||||
#include "keysignals.h"
|
#include "../../deck/custom_line_edit.h"
|
||||||
|
#include "../../game/cards/card_database.h"
|
||||||
|
#include "../game_logic/key_signals.h"
|
||||||
|
#include "../ui/widgets/printing_selector/printing_selector.h"
|
||||||
#include "tab.h"
|
#include "tab.h"
|
||||||
|
|
||||||
#include <QAbstractItemModel>
|
#include <QAbstractItemModel>
|
||||||
#include <QDir>
|
#include <QDir>
|
||||||
#include <QLineEdit>
|
|
||||||
|
|
||||||
#include "carddatabase.h"
|
|
||||||
|
|
||||||
class CardDatabaseModel;
|
class CardDatabaseModel;
|
||||||
class CardDatabaseDisplayModel;
|
class CardDatabaseDisplayModel;
|
||||||
class DeckListModel;
|
class DeckListModel;
|
||||||
class QTreeView;
|
class QTreeView;
|
||||||
class QTableView;
|
|
||||||
class CardFrame;
|
class CardInfoFrameWidget;
|
||||||
class QTextEdit;
|
class QTextEdit;
|
||||||
class QLabel;
|
class QLabel;
|
||||||
class DeckLoader;
|
class DeckLoader;
|
||||||
@@ -22,21 +23,22 @@ class Response;
|
|||||||
class FilterTreeModel;
|
class FilterTreeModel;
|
||||||
class FilterBuilder;
|
class FilterBuilder;
|
||||||
class QGroupBox;
|
class QGroupBox;
|
||||||
|
class QMessageBox;
|
||||||
class QHBoxLayout;
|
class QHBoxLayout;
|
||||||
class QVBoxLayout;
|
class QVBoxLayout;
|
||||||
class QPushButton;
|
class QPushButton;
|
||||||
class QDockWidget;
|
class QDockWidget;
|
||||||
|
|
||||||
class SearchLineEdit : public QLineEdit
|
class SearchLineEdit : public LineEditUnfocusable
|
||||||
{
|
{
|
||||||
private:
|
private:
|
||||||
QTreeView *treeView;
|
QTreeView *treeView;
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
void keyPressEvent(QKeyEvent *event);
|
void keyPressEvent(QKeyEvent *event) override;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
SearchLineEdit() : QLineEdit(), treeView(0)
|
SearchLineEdit() : LineEditUnfocusable(), treeView(nullptr)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
void setTreeView(QTreeView *_treeView)
|
void setTreeView(QTreeView *_treeView)
|
||||||
@@ -54,8 +56,11 @@ private slots:
|
|||||||
void updateHash();
|
void updateHash();
|
||||||
void updateCardInfoLeft(const QModelIndex ¤t, const QModelIndex &previous);
|
void updateCardInfoLeft(const QModelIndex ¤t, const QModelIndex &previous);
|
||||||
void updateCardInfoRight(const QModelIndex ¤t, const QModelIndex &previous);
|
void updateCardInfoRight(const QModelIndex ¤t, const QModelIndex &previous);
|
||||||
|
void updatePrintingSelectorDatabase(const QModelIndex ¤t, const QModelIndex &previous);
|
||||||
|
void updatePrintingSelectorDeckView(const QModelIndex ¤t, const QModelIndex &previous);
|
||||||
void updateSearch(const QString &search);
|
void updateSearch(const QString &search);
|
||||||
void databaseCustomMenu(QPoint point);
|
void databaseCustomMenu(QPoint point);
|
||||||
|
void decklistCustomMenu(QPoint point);
|
||||||
|
|
||||||
void actNewDeck();
|
void actNewDeck();
|
||||||
void actLoadDeck();
|
void actLoadDeck();
|
||||||
@@ -80,6 +85,7 @@ private slots:
|
|||||||
void actDecrement();
|
void actDecrement();
|
||||||
void actDecrementCard();
|
void actDecrementCard();
|
||||||
void actDecrementCardFromSideboard();
|
void actDecrementCardFromSideboard();
|
||||||
|
void copyDatabaseCellContents();
|
||||||
|
|
||||||
void saveDeckRemoteFinished(const Response &r);
|
void saveDeckRemoteFinished(const Response &r);
|
||||||
void filterViewCustomContextMenu(const QPoint &point);
|
void filterViewCustomContextMenu(const QPoint &point);
|
||||||
@@ -90,14 +96,29 @@ private slots:
|
|||||||
void freeDocksSize();
|
void freeDocksSize();
|
||||||
void refreshShortcuts();
|
void refreshShortcuts();
|
||||||
|
|
||||||
bool eventFilter(QObject *o, QEvent *e);
|
bool eventFilter(QObject *o, QEvent *e) override;
|
||||||
void dockVisibleTriggered();
|
void dockVisibleTriggered();
|
||||||
void dockFloatingTriggered();
|
void dockFloatingTriggered();
|
||||||
void dockTopLevelChanged(bool topLevel);
|
void dockTopLevelChanged(bool topLevel);
|
||||||
void saveDbHeaderState();
|
void saveDbHeaderState();
|
||||||
void setSaveStatus(bool newStatus);
|
void setSaveStatus(bool newStatus);
|
||||||
|
void showSearchSyntaxHelp();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
/**
|
||||||
|
* @brief Which tab to open the new deck in
|
||||||
|
*/
|
||||||
|
enum DeckOpenLocation
|
||||||
|
{
|
||||||
|
CANCELLED,
|
||||||
|
SAME_TAB,
|
||||||
|
NEW_TAB
|
||||||
|
};
|
||||||
|
|
||||||
|
DeckOpenLocation confirmOpen(const bool openInSameTabIfBlank = true);
|
||||||
|
QMessageBox *createSaveConfirmationWindow();
|
||||||
|
|
||||||
|
bool isBlankNewDeck() const;
|
||||||
CardInfoPtr currentCardInfo() const;
|
CardInfoPtr currentCardInfo() const;
|
||||||
void addCardHelper(QString zoneName);
|
void addCardHelper(QString zoneName);
|
||||||
void offsetCountAtIndex(const QModelIndex &idx, int offset);
|
void offsetCountAtIndex(const QModelIndex &idx, int offset);
|
||||||
@@ -111,23 +132,24 @@ private:
|
|||||||
|
|
||||||
QTreeView *deckView;
|
QTreeView *deckView;
|
||||||
KeySignals deckViewKeySignals;
|
KeySignals deckViewKeySignals;
|
||||||
CardFrame *cardInfo;
|
CardInfoFrameWidget *cardInfo;
|
||||||
|
PrintingSelector *printingSelector;
|
||||||
SearchLineEdit *searchEdit;
|
SearchLineEdit *searchEdit;
|
||||||
KeySignals searchKeySignals;
|
KeySignals searchKeySignals;
|
||||||
|
|
||||||
QLabel *nameLabel;
|
QLabel *nameLabel;
|
||||||
QLineEdit *nameEdit;
|
LineEditUnfocusable *nameEdit;
|
||||||
QLabel *commentsLabel;
|
QLabel *commentsLabel;
|
||||||
QTextEdit *commentsEdit;
|
QTextEdit *commentsEdit;
|
||||||
QLabel *hashLabel1;
|
QLabel *hashLabel1;
|
||||||
QLabel *hashLabel;
|
LineEditUnfocusable *hashLabel;
|
||||||
FilterTreeModel *filterModel;
|
FilterTreeModel *filterModel;
|
||||||
QTreeView *filterView;
|
QTreeView *filterView;
|
||||||
KeySignals filterViewKeySignals;
|
KeySignals filterViewKeySignals;
|
||||||
QWidget *filterBox;
|
QWidget *filterBox;
|
||||||
|
|
||||||
QMenu *deckMenu, *viewMenu, *cardInfoDockMenu, *deckDockMenu, *filterDockMenu, *analyzeDeckMenu,
|
QMenu *deckMenu, *viewMenu, *cardInfoDockMenu, *deckDockMenu, *filterDockMenu, *printingSelectorDockMenu,
|
||||||
*saveDeckToClipboardMenu;
|
*analyzeDeckMenu, *saveDeckToClipboardMenu;
|
||||||
QAction *aNewDeck, *aLoadDeck, *aSaveDeck, *aSaveDeckAs, *aLoadDeckFromClipboard, *aSaveDeckToClipboard,
|
QAction *aNewDeck, *aLoadDeck, *aSaveDeck, *aSaveDeckAs, *aLoadDeckFromClipboard, *aSaveDeckToClipboard,
|
||||||
*aSaveDeckToClipboardRaw, *aPrintDeck, *aExportDeckDecklist, *aAnalyzeDeckDeckstats, *aAnalyzeDeckTappedout,
|
*aSaveDeckToClipboardRaw, *aPrintDeck, *aExportDeckDecklist, *aAnalyzeDeckDeckstats, *aAnalyzeDeckTappedout,
|
||||||
*aClose;
|
*aClose;
|
||||||
@@ -135,7 +157,7 @@ private:
|
|||||||
QAction *aAddCard, *aAddCardToSideboard, *aRemoveCard, *aIncrement, *aDecrement;
|
QAction *aAddCard, *aAddCardToSideboard, *aRemoveCard, *aIncrement, *aDecrement;
|
||||||
QAction *aResetLayout;
|
QAction *aResetLayout;
|
||||||
QAction *aCardInfoDockVisible, *aCardInfoDockFloating, *aDeckDockVisible, *aDeckDockFloating, *aFilterDockVisible,
|
QAction *aCardInfoDockVisible, *aCardInfoDockFloating, *aDeckDockVisible, *aDeckDockFloating, *aFilterDockVisible,
|
||||||
*aFilterDockFloating;
|
*aFilterDockFloating, *aPrintingSelectorDockVisible, *aPrintingSelectorDockFloating;
|
||||||
|
|
||||||
bool modified;
|
bool modified;
|
||||||
QVBoxLayout *centralFrame;
|
QVBoxLayout *centralFrame;
|
||||||
@@ -143,25 +165,30 @@ private:
|
|||||||
QDockWidget *cardInfoDock;
|
QDockWidget *cardInfoDock;
|
||||||
QDockWidget *deckDock;
|
QDockWidget *deckDock;
|
||||||
QDockWidget *filterDock;
|
QDockWidget *filterDock;
|
||||||
|
QDockWidget *printingSelectorDock;
|
||||||
QWidget *centralWidget;
|
QWidget *centralWidget;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
TabDeckEditor(TabSupervisor *_tabSupervisor, QWidget *parent = 0);
|
explicit TabDeckEditor(TabSupervisor *_tabSupervisor, QWidget *parent = nullptr);
|
||||||
~TabDeckEditor();
|
~TabDeckEditor() override;
|
||||||
void retranslateUi();
|
void retranslateUi() override;
|
||||||
QString getTabText() const;
|
QString getTabText() const override;
|
||||||
void setDeck(DeckLoader *_deckLoader);
|
void setDeck(DeckLoader *_deckLoader);
|
||||||
void setModified(bool _windowModified);
|
void setModified(bool _windowModified);
|
||||||
bool confirmClose();
|
bool confirmClose();
|
||||||
void createDeckDock();
|
void createDeckDock();
|
||||||
void createCardInfoDock();
|
void createCardInfoDock();
|
||||||
void createFiltersDock();
|
void createFiltersDock();
|
||||||
|
void createPrintingSelectorDock();
|
||||||
void createMenus();
|
void createMenus();
|
||||||
void createCentralFrame();
|
void createCentralFrame();
|
||||||
|
void updateCardInfo(CardInfoPtr _card);
|
||||||
|
|
||||||
public slots:
|
public slots:
|
||||||
void closeRequest();
|
void closeRequest() override;
|
||||||
|
void showPrintingSelector();
|
||||||
signals:
|
signals:
|
||||||
|
void openDeckEditor(const DeckLoader *deckLoader);
|
||||||
void deckEditorClosing(TabDeckEditor *tab);
|
void deckEditorClosing(TabDeckEditor *tab);
|
||||||
};
|
};
|
||||||
|
|
||||||