Compare commits
714 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
e56d865416 | ||
|
|
efc779a8d7 | ||
|
|
7e6807509b | ||
|
|
a683866397 | ||
|
|
7a4f56d943 | ||
|
|
ec127519aa | ||
|
|
5ea3e2006d | ||
|
|
bd914bf42a | ||
|
|
e66c95fc76 | ||
|
|
9130302c93 | ||
|
|
11b979e408 | ||
|
|
44f13ef162 | ||
|
|
17253adcdf | ||
|
|
04dcc262ee | ||
|
|
e3bee43163 | ||
|
|
fb2bb5a4bc | ||
|
|
cf0a25fdfc | ||
|
|
e9436426b7 | ||
|
|
37ba95f8b3 | ||
|
|
deadcd5339 | ||
|
|
95d01f0ef5 | ||
|
|
02b8e34242 | ||
|
|
7177e0a576 | ||
|
|
b2c99fd0fd | ||
|
|
cc5910f997 | ||
|
|
cb6090c773 | ||
|
|
c0a649ffcb | ||
|
|
bebc1a22f3 | ||
|
|
b4151a256b | ||
|
|
3898602f0e | ||
|
|
0216425209 | ||
|
|
499671a637 | ||
|
|
ab82faf7ba | ||
|
|
fdf7173f14 | ||
|
|
4b11511bd1 | ||
|
|
53cbc0fb8b | ||
|
|
b209f3e50d | ||
|
|
5ec495987e | ||
|
|
66c9fd8431 | ||
|
|
576af67587 | ||
|
|
cad44cf379 | ||
|
|
ff07e99e39 | ||
|
|
18b137a1a6 | ||
|
|
2344fb910b | ||
|
|
085d309528 | ||
|
|
c8069dfee7 | ||
|
|
9c90f59c25 | ||
|
|
15cf8fe0be | ||
|
|
4258f53bc8 | ||
|
|
3640f1ba77 | ||
|
|
23bb409639 | ||
|
|
6364b9e06b | ||
|
|
5cefecced8 | ||
|
|
8b17972b11 | ||
|
|
e799187aaf | ||
|
|
c89123b18a | ||
|
|
03277c69af | ||
|
|
20b3bb8f13 | ||
|
|
a6ca84308e | ||
|
|
a71f3f55b0 | ||
|
|
24ac823f06 | ||
|
|
e407ba3ad3 | ||
|
|
7a049d64d8 | ||
|
|
8ec9d95cd2 | ||
|
|
91501d39ee | ||
|
|
5d70b4b95b | ||
|
|
c7bc6e9128 | ||
|
|
b911ef98bb | ||
|
|
f464868934 | ||
|
|
d89c33fbe8 | ||
|
|
f61f8a36f8 | ||
|
|
b9b34eea37 | ||
|
|
d948607039 | ||
|
|
5a135e108a | ||
|
|
ced11c88bc | ||
|
|
92a748a762 | ||
|
|
f4c9645d73 | ||
|
|
89ddbfc83d | ||
|
|
fa53b1be73 | ||
|
|
c7ec7db61a | ||
|
|
27cf4b0218 | ||
|
|
482858e36b | ||
|
|
50372ed9f3 | ||
|
|
d8e8765cec | ||
|
|
6ec28be541 | ||
|
|
6e5aec080a | ||
|
|
c7fb517e6c | ||
|
|
ca10ab0274 | ||
|
|
41e611d7db | ||
|
|
77a63a39ad | ||
|
|
0267ad574f | ||
|
|
8f2aae6065 | ||
|
|
a4edf827c8 | ||
|
|
f9947d2e53 | ||
|
|
0a6614d1b3 | ||
|
|
98cfb4d5ed | ||
|
|
18a780147e | ||
|
|
74878b5cb1 | ||
|
|
57d5628750 | ||
|
|
fc058476ed | ||
|
|
c24eb231ab | ||
|
|
28e44d33c4 | ||
|
|
f7dbe807e4 | ||
|
|
a76e99af86 | ||
|
|
13bb78ffa0 | ||
|
|
8c194398a9 | ||
|
|
21238e7376 | ||
|
|
ca30f49481 | ||
|
|
45da578756 | ||
|
|
1c80314cfd | ||
|
|
23d621e9c7 | ||
|
|
2884e4b847 | ||
|
|
0c7f6432d8 | ||
|
|
56bb23400a | ||
|
|
33a278ccb4 | ||
|
|
e79494b5cb | ||
|
|
3c3147d665 | ||
|
|
23f3f68767 | ||
|
|
6cfd1a1ef6 | ||
|
|
354c6e17c0 | ||
|
|
c31a710aa7 | ||
|
|
0304512b8a | ||
|
|
6bdd960623 | ||
|
|
80d0a1555a | ||
|
|
ee2cba89d5 | ||
|
|
84d80525fa | ||
|
|
4d9fab6cf2 | ||
|
|
77856a3ff7 | ||
|
|
eb99132b8c | ||
|
|
e1cdf80b17 | ||
|
|
9fbeeda1c6 | ||
|
|
1a8cd2e24c | ||
|
|
6c620182ea | ||
|
|
ff542a9946 | ||
|
|
6281826b92 | ||
|
|
eb94a4ee14 | ||
|
|
09ebb7a01c | ||
|
|
11d2ada6b3 | ||
|
|
ffbb6bb40b | ||
|
|
604f315a83 | ||
|
|
ba67c01dd0 | ||
|
|
fab3704dbd | ||
|
|
1436e78305 | ||
|
|
125677f787 | ||
|
|
63d5539427 | ||
|
|
4be101e9fc | ||
|
|
d0302cdbf3 | ||
|
|
c077f6e1e8 | ||
|
|
4ca9a26bdd | ||
|
|
6b713b283a | ||
|
|
9abbd8c158 | ||
|
|
4df0641d67 | ||
|
|
4b093812bc | ||
|
|
d7120af5cb | ||
|
|
e36a666542 | ||
|
|
b609b5909b | ||
|
|
cbd83b5e39 | ||
|
|
2ddd768393 | ||
|
|
c508f30d92 | ||
|
|
9b7750c1f3 | ||
|
|
a9b5949a7b | ||
|
|
f56247df7a | ||
|
|
0ef76aa66d | ||
|
|
5c98283932 | ||
|
|
ce6bd0fd6d | ||
|
|
aca1001f9a | ||
|
|
d5a58be440 | ||
|
|
b33dd7e91c | ||
|
|
30e13259b3 | ||
|
|
be2fa1dfad | ||
|
|
bc7b584001 | ||
|
|
690571eec0 | ||
|
|
0620c635e5 | ||
|
|
8e00bdcf30 | ||
|
|
0bd0602655 | ||
|
|
2bcbc3294b | ||
|
|
9aa5460688 | ||
|
|
e451cdd519 | ||
|
|
ab9cdf5610 | ||
|
|
c443eee15d | ||
|
|
9e136bad94 | ||
|
|
ccb97e34d5 | ||
|
|
917e2e40fb | ||
|
|
c5ed0afb4e | ||
|
|
b10fa9e5b2 | ||
|
|
210967cfef | ||
|
|
2a5118326a | ||
|
|
a6225962b0 | ||
|
|
01f4a9b23b | ||
|
|
3202d62a92 | ||
|
|
270dc351a1 | ||
|
|
12c746a4e4 | ||
|
|
a116a2a553 | ||
|
|
9555b10b97 | ||
|
|
ae6b4211b9 | ||
|
|
3fe7e4e8da | ||
|
|
ba805739b0 | ||
|
|
89d8aa99ac | ||
|
|
5b65c2e27e | ||
|
|
6f9b712230 | ||
|
|
ca3bfa2239 | ||
|
|
35a3a89d2d | ||
|
|
be8632d94b | ||
|
|
1bc4b16d38 | ||
|
|
863c159fbb | ||
|
|
a647efe3fd | ||
|
|
c79c23899f | ||
|
|
a04e432aea | ||
|
|
5c120e532f | ||
|
|
cfa95eef39 | ||
|
|
a327b9cfa8 | ||
|
|
c4ae6121b4 | ||
|
|
23d07600f7 | ||
|
|
418467e467 | ||
|
|
e60b3880ce | ||
|
|
92acf009b0 | ||
|
|
a141df434c | ||
|
|
32201dda72 | ||
|
|
13b0c8880d | ||
|
|
24e7d755eb | ||
|
|
e534a6b7fc | ||
|
|
277d75016c | ||
|
|
278a10cf61 | ||
|
|
8f283b2226 | ||
|
|
857b76980b | ||
|
|
fae7bfa8c5 | ||
|
|
e9e7d2e289 | ||
|
|
e3cbe118c5 | ||
|
|
9d05cfba36 | ||
|
|
b5966afe25 | ||
|
|
1cf91c0ce3 | ||
|
|
0f09a0b152 | ||
|
|
b434c37f01 | ||
|
|
4655bd5502 | ||
|
|
fcb74d4fba | ||
|
|
2bf7d06a91 | ||
|
|
ce68c2a3e9 | ||
|
|
e58194ac3c | ||
|
|
857f36e01a | ||
|
|
83c3a80766 | ||
|
|
736552f3b1 | ||
|
|
0b28f97d8f | ||
|
|
47b627d669 | ||
|
|
b051a296fc | ||
|
|
908a569413 | ||
|
|
31f83dfb40 | ||
|
|
022b16c8e6 | ||
|
|
d462d9c89b | ||
|
|
86360049c3 | ||
|
|
f524be6738 | ||
|
|
519d2c1714 | ||
|
|
c19fcb831f | ||
|
|
ff6268e616 | ||
|
|
b0f1ad63ee | ||
|
|
a7b017f307 | ||
|
|
29c3df8ba7 | ||
|
|
570226a051 | ||
|
|
96cfb51bd8 | ||
|
|
ba0c7907ca | ||
|
|
4c6ae83d4b | ||
|
|
e4231e012d | ||
|
|
9693de8885 | ||
|
|
d096a7ceb1 | ||
|
|
fd77dfd66d | ||
|
|
3f1389818c | ||
|
|
c920fa6740 | ||
|
|
b4088dcfbc | ||
|
|
b20f026c7b | ||
|
|
f0fa126e7b | ||
|
|
2999694f48 | ||
|
|
fcc60957c1 | ||
|
|
4781351184 | ||
|
|
d97a585eba | ||
|
|
2d65c9734c | ||
|
|
3b5a9c0a10 | ||
|
|
f6184004e5 | ||
|
|
d70dd05221 | ||
|
|
5602cb8920 | ||
|
|
4a250b4e27 | ||
|
|
37afb46957 | ||
|
|
9708d3183f | ||
|
|
433cb99eab | ||
|
|
241b86d704 | ||
|
|
3b0c66429c | ||
|
|
8f137eadf0 | ||
|
|
468a6c1dca | ||
|
|
f9597569ba | ||
|
|
93353ff900 | ||
|
|
ed7e77b2db | ||
|
|
2108aab523 | ||
|
|
75f01099ee | ||
|
|
26e1ad51ff | ||
|
|
c1a5482ad2 | ||
|
|
ed232248a2 | ||
|
|
e215324292 | ||
|
|
a8bc2f10e4 | ||
|
|
b64cbc223f | ||
|
|
de40df7f7b | ||
|
|
94896451ac | ||
|
|
0e35bb4852 | ||
|
|
cd30b1d9b3 | ||
|
|
8d66d43a98 | ||
|
|
8ca861360b | ||
|
|
410db16160 | ||
|
|
1c1df81341 | ||
|
|
66a1cc7b71 | ||
|
|
0543569315 | ||
|
|
e5a94bd140 | ||
|
|
adfebdb3f6 | ||
|
|
dbe06b0a93 | ||
|
|
e8305d1245 | ||
|
|
b2ea733b45 | ||
|
|
fc320fb9ee | ||
|
|
a3268eefbf | ||
|
|
5a77eebb34 | ||
|
|
68af32bc9f | ||
|
|
100e5042de | ||
|
|
b3e188a5f4 | ||
|
|
ae3bcac3b7 | ||
|
|
afed54b217 | ||
|
|
2d768990ec | ||
|
|
10c44d0a4e | ||
|
|
57b0e3e311 | ||
|
|
19e62786c9 | ||
|
|
e45e04a6dc | ||
|
|
97cac33773 | ||
|
|
a889571b98 | ||
|
|
cf1148e7c8 | ||
|
|
e2f74fd4f0 | ||
|
|
57bcea56b5 | ||
|
|
380a955e7a | ||
|
|
cf50f47459 | ||
|
|
eb69dce585 | ||
|
|
b3adf7b331 | ||
|
|
a8900bb50e | ||
|
|
c79f0b76ff | ||
|
|
ca24e64c42 | ||
|
|
d1a3752b2d | ||
|
|
00e70cbb6a | ||
|
|
7f2c2d9d6b | ||
|
|
959f756ff5 | ||
|
|
90b9ba71de | ||
|
|
26025a3018 | ||
|
|
cce35b4a6f | ||
|
|
09bdfd7ac1 | ||
|
|
ce2b73b737 | ||
|
|
61e9611bab | ||
|
|
83d1707b56 | ||
|
|
cbdf252fe6 | ||
|
|
55cd0c2bae | ||
|
|
6cc1cecf5a | ||
|
|
7a94ebcf2a | ||
|
|
84a126253a | ||
|
|
ad3ce3f9fc | ||
|
|
c74ee2150c | ||
|
|
665f0ee03e | ||
|
|
f8a4128c88 | ||
|
|
c931bbf107 | ||
|
|
b020336f8c | ||
|
|
299c2af59b | ||
|
|
37a73eda37 | ||
|
|
3938022002 | ||
|
|
450fc456be | ||
|
|
f7e0dc4818 | ||
|
|
b496f51afc | ||
|
|
c1163a078a | ||
|
|
90c243dec8 | ||
|
|
fab22a152c | ||
|
|
3905da1a62 | ||
|
|
0b02ada315 | ||
|
|
3dd1fc2efe | ||
|
|
c1066e94f2 | ||
|
|
3a3c23d2f1 | ||
|
|
1f439b4b0c | ||
|
|
2af750cfec | ||
|
|
42b3b19a57 | ||
|
|
496498b913 | ||
|
|
b7a7e7dcc0 | ||
|
|
3fcdfbd01a | ||
|
|
72fc673020 | ||
|
|
c3967c79c9 | ||
|
|
3dd07d2f4a | ||
|
|
036f233946 | ||
|
|
16c243d270 | ||
|
|
9776884a91 | ||
|
|
131f646024 | ||
|
|
1bf728359d | ||
|
|
0ee5c1cd86 | ||
|
|
8a83e0fc1a | ||
|
|
0be49002ba | ||
|
|
2b96c820fd | ||
|
|
9641a388e2 | ||
|
|
37fcd81e87 | ||
|
|
b589d8a1df | ||
|
|
d0a44906fe | ||
|
|
5b8ee48600 | ||
|
|
e5f6b7f66c | ||
|
|
941ebd39bf | ||
|
|
52b73b5a9d | ||
|
|
f5b7e6bb60 | ||
|
|
44965a42e4 | ||
|
|
d8aea69c0c | ||
|
|
aa110ec9a6 | ||
|
|
dbfe5196e7 | ||
|
|
13e911eaa1 | ||
|
|
cbbd5ffc62 | ||
|
|
cf40aec9d7 | ||
|
|
cdda421920 | ||
|
|
0db22f2cff | ||
|
|
7a22ca2841 | ||
|
|
4fc8e321af | ||
|
|
59fe8f444f | ||
|
|
8cc3a865e2 | ||
|
|
8a13482c38 | ||
|
|
d8d8552176 | ||
|
|
96475557ae | ||
|
|
bb4c498e5a | ||
|
|
3ba57c3473 | ||
|
|
094aa5a326 | ||
|
|
4b71b30418 | ||
|
|
d960703a5c | ||
|
|
6244551d69 | ||
|
|
3ed6be1d6d | ||
|
|
91c3cb93f0 | ||
|
|
8c369d9874 | ||
|
|
b5008f4ad3 | ||
|
|
deefab8a5b | ||
|
|
ff6dc98bbe | ||
|
|
8581213126 | ||
|
|
19d058a644 | ||
|
|
aa83510fbe | ||
|
|
5aa995b5ef | ||
|
|
1a9d701d32 | ||
|
|
9f4f107b6d | ||
|
|
90716ebaad | ||
|
|
496b4e77e8 | ||
|
|
6e9d833c77 | ||
|
|
48061d3cae | ||
|
|
292ce33e0f | ||
|
|
365fc6af20 | ||
|
|
97360b33f4 | ||
|
|
5012313dc4 | ||
|
|
95f10aeb06 | ||
|
|
8a965e56cd | ||
|
|
fb14a83b9a | ||
|
|
8cef351ed9 | ||
|
|
4261e6242c | ||
|
|
68bfd3017b | ||
|
|
ccd25c7e76 | ||
|
|
7df232c65d | ||
|
|
32fb79478c | ||
|
|
d950396f68 | ||
|
|
7d5f336e36 | ||
|
|
b5cba3abfd | ||
|
|
1d8630dd7d | ||
|
|
bfcce4f7ab | ||
|
|
622f0d9355 | ||
|
|
434256b0a5 | ||
|
|
e7a4ce2e19 | ||
|
|
23dc62dbd2 | ||
|
|
739e636a2c | ||
|
|
b020bf6816 | ||
|
|
47d61c4a0e | ||
|
|
714c5fd784 | ||
|
|
7f2a0a77c7 | ||
|
|
0bebe5008b | ||
|
|
ee24eb8d1b | ||
|
|
d396149521 | ||
|
|
a4962c5225 | ||
|
|
bece87c790 | ||
|
|
e21d92cf4d | ||
|
|
e6f9ce888f | ||
|
|
871b6548ab | ||
|
|
cb569be3b3 | ||
|
|
0e50fbf3f2 | ||
|
|
3adc599278 | ||
|
|
68593acf89 | ||
|
|
6007e511ba | ||
|
|
01fc7108d8 | ||
|
|
494e415bd3 | ||
|
|
c17b34c70d | ||
|
|
02ab945ab8 | ||
|
|
97e5ac4c1e | ||
|
|
458d4378f0 | ||
|
|
b715e0d34e | ||
|
|
cce4113b2d | ||
|
|
89f06506c1 | ||
|
|
cc26e61d7a | ||
|
|
4b6b1b43cd | ||
|
|
913ee34c8d | ||
|
|
4e77f24a06 | ||
|
|
fcf62ada8f | ||
|
|
86b7966027 | ||
|
|
43c9bc51dc | ||
|
|
1ff6a7e423 | ||
|
|
439d2c88ae | ||
|
|
37aced0ad3 | ||
|
|
a3a1e6d3a9 | ||
|
|
2c769cad0b | ||
|
|
b1ad8f917d | ||
|
|
221f7be21b | ||
|
|
bed34c4cad | ||
|
|
f8b0819fb3 | ||
|
|
25778a143c | ||
|
|
5b11875a5a | ||
|
|
6967093689 | ||
|
|
0cafa152bf | ||
|
|
697e288de2 | ||
|
|
aa28c3a45b | ||
|
|
815daccff1 | ||
|
|
f61aa6068b | ||
|
|
21fe0f9246 | ||
|
|
6db1175c95 | ||
|
|
00b78402eb | ||
|
|
30ddd91a34 | ||
|
|
1789fadbf6 | ||
|
|
86d5803b63 | ||
|
|
98ec2dcf44 | ||
|
|
117e2e2cb6 | ||
|
|
4b39e5fab6 | ||
|
|
7ea4a503f7 | ||
|
|
7cf619e620 | ||
|
|
7a65604cc8 | ||
|
|
d67402e6aa | ||
|
|
2191437f39 | ||
|
|
c6efa6ca2e | ||
|
|
5172ae781f | ||
|
|
343a94bf05 | ||
|
|
7cd96e379e | ||
|
|
ddf39461fb | ||
|
|
62608bea6c | ||
|
|
05356776c5 | ||
|
|
1956fccbe9 | ||
|
|
53ca2d0755 | ||
|
|
641a4b87c6 | ||
|
|
0559b43bde | ||
|
|
397220e682 | ||
|
|
636da8f95f | ||
|
|
94f2e880cd | ||
|
|
c03aa6300e | ||
|
|
54d0cae450 | ||
|
|
efd9280f96 | ||
|
|
a1ca9d396c | ||
|
|
c686bf6149 | ||
|
|
20b95fa063 | ||
|
|
9db7b1385d | ||
|
|
ec5f74b753 | ||
|
|
528cb6bf40 | ||
|
|
0936fc6550 | ||
|
|
5ee369f26a | ||
|
|
198dff7864 | ||
|
|
af483e93f3 | ||
|
|
0813f00ea1 | ||
|
|
06dcc875bd | ||
|
|
eab226f04d | ||
|
|
dd0ad495f7 | ||
|
|
4ff3517343 | ||
|
|
537e25af9f | ||
|
|
8a80e82b5d | ||
|
|
c14fb51eb8 | ||
|
|
9302b7c15c | ||
|
|
662283908c | ||
|
|
ed276ac79e | ||
|
|
84548187f5 | ||
|
|
532eeb7ab0 | ||
|
|
e78579581f | ||
|
|
027e8dc35e | ||
|
|
d83a852681 | ||
|
|
826d5705c9 | ||
|
|
fbf8655ea4 | ||
|
|
737c63da5e | ||
|
|
f0302ca7dc | ||
|
|
795464e737 | ||
|
|
2a1110c358 | ||
|
|
c54df7cbd1 | ||
|
|
a98f42012a | ||
|
|
8526c67e4d | ||
|
|
29051d4281 | ||
|
|
1d0063ed39 | ||
|
|
18d81f6daa | ||
|
|
17bc8fd9ed | ||
|
|
707e57e72d | ||
|
|
46e6b839b9 | ||
|
|
30fc96e4da | ||
|
|
f049706539 | ||
|
|
203e4bc0a4 | ||
|
|
d6cd197435 | ||
|
|
684b57d139 | ||
|
|
3418979f4d | ||
|
|
59530909d6 | ||
|
|
e0c1bb8da1 | ||
|
|
fd352f4ab1 | ||
|
|
7f40247b89 | ||
|
|
674d259df1 | ||
|
|
b20a2f74c4 | ||
|
|
5c748fbe54 | ||
|
|
8eb2a13efd | ||
|
|
3af18094f7 | ||
|
|
9e309c9592 | ||
|
|
c73286358c | ||
|
|
967dab21bc | ||
|
|
4e9d1d348e | ||
|
|
2dd67c229c | ||
|
|
7b3d60215b | ||
|
|
d0783f352d | ||
|
|
7d2fd514be | ||
|
|
2981dd857f | ||
|
|
664b2c3ced | ||
|
|
357fc37a79 | ||
|
|
5b075ead9e | ||
|
|
e5656896e7 | ||
|
|
238c589ad6 | ||
|
|
85988a2ff9 | ||
|
|
788f46cc55 | ||
|
|
4bafa32549 | ||
|
|
23e6f89eef | ||
|
|
e330b28e49 | ||
|
|
1984921883 | ||
|
|
68f8ef54ac | ||
|
|
d5493e6149 | ||
|
|
216b3969ed | ||
|
|
940b8b72f9 | ||
|
|
ba7be0293c | ||
|
|
5323ef68ca | ||
|
|
34f4eda1cf | ||
|
|
1b36325450 | ||
|
|
0f8168324f | ||
|
|
58ed9754a4 | ||
|
|
172ac78466 | ||
|
|
2eae473429 | ||
|
|
12686460c1 | ||
|
|
713875100b | ||
|
|
ba84bb82b4 | ||
|
|
3903750ba2 | ||
|
|
ed0b796c5a | ||
|
|
4a7c722673 | ||
|
|
714fdc751a | ||
|
|
e2effa1c5d | ||
|
|
5846af9453 | ||
|
|
ac4369ce3b | ||
|
|
13619a14dc | ||
|
|
b9f565f5fc | ||
|
|
4cb500277b | ||
|
|
ad2e2032e6 | ||
|
|
52c0a8e573 | ||
|
|
4dff6d834a | ||
|
|
2037ebaaba | ||
|
|
807c4688c2 | ||
|
|
b4ec2ccb90 | ||
|
|
ae89a004a6 | ||
|
|
95db857fb5 | ||
|
|
155f3945a6 | ||
|
|
0b2a933faa | ||
|
|
01ba051c43 | ||
|
|
1cd30e9057 | ||
|
|
a023c4fbac | ||
|
|
99c12595a9 | ||
|
|
3056e9e79c | ||
|
|
b822a02d41 | ||
|
|
d0b414b61d | ||
|
|
f056edded7 | ||
|
|
ea9b3cc7dd | ||
|
|
8fffc1cf9c | ||
|
|
539971b5a0 | ||
|
|
2a0d3b05d3 | ||
|
|
d1b51844e4 | ||
|
|
94b815b8ab | ||
|
|
1ab9961cfe | ||
|
|
e67ddf9430 | ||
|
|
a0f85a8e4b | ||
|
|
9674ca5e87 | ||
|
|
5834dc9b2e | ||
|
|
9d49c7d836 | ||
|
|
3ce7284fc5 | ||
|
|
bf6e33791d | ||
|
|
c9342d60d7 | ||
|
|
6a11f0dd8d | ||
|
|
3fa9d5ed10 | ||
|
|
83452b1e36 | ||
|
|
d88a908e29 | ||
|
|
9c05ddbd7d | ||
|
|
89ca9ad213 | ||
|
|
0310614392 | ||
|
|
460496339b | ||
|
|
1c8b451ec8 | ||
|
|
689794cab5 | ||
|
|
8743a60d02 | ||
|
|
2dc0a79263 | ||
|
|
387e912f6a | ||
|
|
c46f1ec590 | ||
|
|
3131e8c7e9 | ||
|
|
7d8b0f9941 | ||
|
|
d3313d6438 | ||
|
|
c611066791 | ||
|
|
1e456d6560 | ||
|
|
6a564f6693 | ||
|
|
45a3be7642 | ||
|
|
a25b6a9952 | ||
|
|
25063ed1f0 | ||
|
|
292dbf101b | ||
|
|
099d88ba7b | ||
|
|
992fcc73b8 | ||
|
|
9b3c9aaea2 | ||
|
|
41e4f87e06 | ||
|
|
6de48b4535 | ||
|
|
70b104c936 | ||
|
|
827f72f595 | ||
|
|
5faff6260e | ||
|
|
59e8a85c7e | ||
|
|
d3ec50825e | ||
|
|
2518564f92 | ||
|
|
81de9875d3 | ||
|
|
eae07c15a7 | ||
|
|
1fdffa8be5 |
6
.gitignore
vendored
@@ -9,14 +9,20 @@
|
|||||||
/desktop/mindustry-maps/
|
/desktop/mindustry-maps/
|
||||||
/desktop/gifexport/
|
/desktop/gifexport/
|
||||||
/core/lib/
|
/core/lib/
|
||||||
|
/core/assets-raw/sprites/generated/
|
||||||
|
/annotations/build/
|
||||||
/kryonet/build/
|
/kryonet/build/
|
||||||
|
/packer/build/
|
||||||
/server/build/
|
/server/build/
|
||||||
|
/annotations/build/
|
||||||
/android/assets/mindustry-maps/
|
/android/assets/mindustry-maps/
|
||||||
/android/assets/mindustry-saves/
|
/android/assets/mindustry-saves/
|
||||||
/core/assets/gifexport/
|
/core/assets/gifexport/
|
||||||
/core/assets/version.properties
|
/core/assets/version.properties
|
||||||
*.gif
|
*.gif
|
||||||
|
|
||||||
|
version.properties
|
||||||
|
|
||||||
.attach_*
|
.attach_*
|
||||||
## Java
|
## Java
|
||||||
|
|
||||||
|
|||||||
687
LICENSE
@@ -1,21 +1,674 @@
|
|||||||
MIT License
|
GNU GENERAL PUBLIC LICENSE
|
||||||
|
Version 3, 29 June 2007
|
||||||
|
|
||||||
Copyright (c) 2017 Anton Kramskoi
|
Copyright (C) 2007 Free Software Foundation, Inc. <http://fsf.org/>
|
||||||
|
Everyone is permitted to copy and distribute verbatim copies
|
||||||
|
of this license document, but changing it is not allowed.
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
Preamble
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
|
||||||
in the Software without restriction, including without limitation the rights
|
|
||||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
copies of the Software, and to permit persons to whom the Software is
|
|
||||||
furnished to do so, subject to the following conditions:
|
|
||||||
|
|
||||||
The above copyright notice and this permission notice shall be included in all
|
The GNU General Public License is a free, copyleft license for
|
||||||
copies or substantial portions of the Software.
|
software and other kinds of works.
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
The licenses for most software and other practical works are designed
|
||||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
to take away your freedom to share and change the works. By contrast,
|
||||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
the GNU General Public License is intended to guarantee your freedom to
|
||||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
share and change all versions of a program--to make sure it remains free
|
||||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
software for all its users. We, the Free Software Foundation, use the
|
||||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
GNU General Public License for most of our software; it applies also to
|
||||||
SOFTWARE.
|
any other work released this way by its authors. You can apply it to
|
||||||
|
your programs, too.
|
||||||
|
|
||||||
|
When we speak of free software, we are referring to freedom, not
|
||||||
|
price. Our General Public Licenses are designed to make sure that you
|
||||||
|
have the freedom to distribute copies of free software (and charge for
|
||||||
|
them if you wish), that you receive source code or can get it if you
|
||||||
|
want it, that you can change the software or use pieces of it in new
|
||||||
|
free programs, and that you know you can do these things.
|
||||||
|
|
||||||
|
To protect your rights, we need to prevent others from denying you
|
||||||
|
these rights or asking you to surrender the rights. Therefore, you have
|
||||||
|
certain responsibilities if you distribute copies of the software, or if
|
||||||
|
you modify it: responsibilities to respect the freedom of others.
|
||||||
|
|
||||||
|
For example, if you distribute copies of such a program, whether
|
||||||
|
gratis or for a fee, you must pass on to the recipients the same
|
||||||
|
freedoms that you received. You must make sure that they, too, receive
|
||||||
|
or can get the source code. And you must show them these terms so they
|
||||||
|
know their rights.
|
||||||
|
|
||||||
|
Developers that use the GNU GPL protect your rights with two steps:
|
||||||
|
(1) assert copyright on the software, and (2) offer you this License
|
||||||
|
giving you legal permission to copy, distribute and/or modify it.
|
||||||
|
|
||||||
|
For the developers' and authors' protection, the GPL clearly explains
|
||||||
|
that there is no warranty for this free software. For both users' and
|
||||||
|
authors' sake, the GPL requires that modified versions be marked as
|
||||||
|
changed, so that their problems will not be attributed erroneously to
|
||||||
|
authors of previous versions.
|
||||||
|
|
||||||
|
Some devices are designed to deny users access to install or run
|
||||||
|
modified versions of the software inside them, although the manufacturer
|
||||||
|
can do so. This is fundamentally incompatible with the aim of
|
||||||
|
protecting users' freedom to change the software. The systematic
|
||||||
|
pattern of such abuse occurs in the area of products for individuals to
|
||||||
|
use, which is precisely where it is most unacceptable. Therefore, we
|
||||||
|
have designed this version of the GPL to prohibit the practice for those
|
||||||
|
products. If such problems arise substantially in other domains, we
|
||||||
|
stand ready to extend this provision to those domains in future versions
|
||||||
|
of the GPL, as needed to protect the freedom of users.
|
||||||
|
|
||||||
|
Finally, every program is threatened constantly by software patents.
|
||||||
|
States should not allow patents to restrict development and use of
|
||||||
|
software on general-purpose computers, but in those that do, we wish to
|
||||||
|
avoid the special danger that patents applied to a free program could
|
||||||
|
make it effectively proprietary. To prevent this, the GPL assures that
|
||||||
|
patents cannot be used to render the program non-free.
|
||||||
|
|
||||||
|
The precise terms and conditions for copying, distribution and
|
||||||
|
modification follow.
|
||||||
|
|
||||||
|
TERMS AND CONDITIONS
|
||||||
|
|
||||||
|
0. Definitions.
|
||||||
|
|
||||||
|
"This License" refers to version 3 of the GNU General Public License.
|
||||||
|
|
||||||
|
"Copyright" also means copyright-like laws that apply to other kinds of
|
||||||
|
works, such as semiconductor masks.
|
||||||
|
|
||||||
|
"The Program" refers to any copyrightable work licensed under this
|
||||||
|
License. Each licensee is addressed as "you". "Licensees" and
|
||||||
|
"recipients" may be individuals or organizations.
|
||||||
|
|
||||||
|
To "modify" a work means to copy from or adapt all or part of the work
|
||||||
|
in a fashion requiring copyright permission, other than the making of an
|
||||||
|
exact copy. The resulting work is called a "modified version" of the
|
||||||
|
earlier work or a work "based on" the earlier work.
|
||||||
|
|
||||||
|
A "covered work" means either the unmodified Program or a work based
|
||||||
|
on the Program.
|
||||||
|
|
||||||
|
To "propagate" a work means to do anything with it that, without
|
||||||
|
permission, would make you directly or secondarily liable for
|
||||||
|
infringement under applicable copyright law, except executing it on a
|
||||||
|
computer or modifying a private copy. Propagation includes copying,
|
||||||
|
distribution (with or without modification), making available to the
|
||||||
|
public, and in some countries other activities as well.
|
||||||
|
|
||||||
|
To "convey" a work means any kind of propagation that enables other
|
||||||
|
parties to make or receive copies. Mere interaction with a user through
|
||||||
|
a computer network, with no transfer of a copy, is not conveying.
|
||||||
|
|
||||||
|
An interactive user interface displays "Appropriate Legal Notices"
|
||||||
|
to the extent that it includes a convenient and prominently visible
|
||||||
|
feature that (1) displays an appropriate copyright notice, and (2)
|
||||||
|
tells the user that there is no warranty for the work (except to the
|
||||||
|
extent that warranties are provided), that licensees may convey the
|
||||||
|
work under this License, and how to view a copy of this License. If
|
||||||
|
the interface presents a list of user commands or options, such as a
|
||||||
|
menu, a prominent item in the list meets this criterion.
|
||||||
|
|
||||||
|
1. Source Code.
|
||||||
|
|
||||||
|
The "source code" for a work means the preferred form of the work
|
||||||
|
for making modifications to it. "Object code" means any non-source
|
||||||
|
form of a work.
|
||||||
|
|
||||||
|
A "Standard Interface" means an interface that either is an official
|
||||||
|
standard defined by a recognized standards body, or, in the case of
|
||||||
|
interfaces specified for a particular programming language, one that
|
||||||
|
is widely used among developers working in that language.
|
||||||
|
|
||||||
|
The "System Libraries" of an executable work include anything, other
|
||||||
|
than the work as a whole, that (a) is included in the normal form of
|
||||||
|
packaging a Major Component, but which is not part of that Major
|
||||||
|
Component, and (b) serves only to enable use of the work with that
|
||||||
|
Major Component, or to implement a Standard Interface for which an
|
||||||
|
implementation is available to the public in source code form. A
|
||||||
|
"Major Component", in this context, means a major essential component
|
||||||
|
(kernel, window system, and so on) of the specific operating system
|
||||||
|
(if any) on which the executable work runs, or a compiler used to
|
||||||
|
produce the work, or an object code interpreter used to run it.
|
||||||
|
|
||||||
|
The "Corresponding Source" for a work in object code form means all
|
||||||
|
the source code needed to generate, install, and (for an executable
|
||||||
|
work) run the object code and to modify the work, including scripts to
|
||||||
|
control those activities. However, it does not include the work's
|
||||||
|
System Libraries, or general-purpose tools or generally available free
|
||||||
|
programs which are used unmodified in performing those activities but
|
||||||
|
which are not part of the work. For example, Corresponding Source
|
||||||
|
includes interface definition files associated with source files for
|
||||||
|
the work, and the source code for shared libraries and dynamically
|
||||||
|
linked subprograms that the work is specifically designed to require,
|
||||||
|
such as by intimate data communication or control flow between those
|
||||||
|
subprograms and other parts of the work.
|
||||||
|
|
||||||
|
The Corresponding Source need not include anything that users
|
||||||
|
can regenerate automatically from other parts of the Corresponding
|
||||||
|
Source.
|
||||||
|
|
||||||
|
The Corresponding Source for a work in source code form is that
|
||||||
|
same work.
|
||||||
|
|
||||||
|
2. Basic Permissions.
|
||||||
|
|
||||||
|
All rights granted under this License are granted for the term of
|
||||||
|
copyright on the Program, and are irrevocable provided the stated
|
||||||
|
conditions are met. This License explicitly affirms your unlimited
|
||||||
|
permission to run the unmodified Program. The output from running a
|
||||||
|
covered work is covered by this License only if the output, given its
|
||||||
|
content, constitutes a covered work. This License acknowledges your
|
||||||
|
rights of fair use or other equivalent, as provided by copyright law.
|
||||||
|
|
||||||
|
You may make, run and propagate covered works that you do not
|
||||||
|
convey, without conditions so long as your license otherwise remains
|
||||||
|
in force. You may convey covered works to others for the sole purpose
|
||||||
|
of having them make modifications exclusively for you, or provide you
|
||||||
|
with facilities for running those works, provided that you comply with
|
||||||
|
the terms of this License in conveying all material for which you do
|
||||||
|
not control copyright. Those thus making or running the covered works
|
||||||
|
for you must do so exclusively on your behalf, under your direction
|
||||||
|
and control, on terms that prohibit them from making any copies of
|
||||||
|
your copyrighted material outside their relationship with you.
|
||||||
|
|
||||||
|
Conveying under any other circumstances is permitted solely under
|
||||||
|
the conditions stated below. Sublicensing is not allowed; section 10
|
||||||
|
makes it unnecessary.
|
||||||
|
|
||||||
|
3. Protecting Users' Legal Rights From Anti-Circumvention Law.
|
||||||
|
|
||||||
|
No covered work shall be deemed part of an effective technological
|
||||||
|
measure under any applicable law fulfilling obligations under article
|
||||||
|
11 of the WIPO copyright treaty adopted on 20 December 1996, or
|
||||||
|
similar laws prohibiting or restricting circumvention of such
|
||||||
|
measures.
|
||||||
|
|
||||||
|
When you convey a covered work, you waive any legal power to forbid
|
||||||
|
circumvention of technological measures to the extent such circumvention
|
||||||
|
is effected by exercising rights under this License with respect to
|
||||||
|
the covered work, and you disclaim any intention to limit operation or
|
||||||
|
modification of the work as a means of enforcing, against the work's
|
||||||
|
users, your or third parties' legal rights to forbid circumvention of
|
||||||
|
technological measures.
|
||||||
|
|
||||||
|
4. Conveying Verbatim Copies.
|
||||||
|
|
||||||
|
You may convey verbatim copies of the Program's source code as you
|
||||||
|
receive it, in any medium, provided that you conspicuously and
|
||||||
|
appropriately publish on each copy an appropriate copyright notice;
|
||||||
|
keep intact all notices stating that this License and any
|
||||||
|
non-permissive terms added in accord with section 7 apply to the code;
|
||||||
|
keep intact all notices of the absence of any warranty; and give all
|
||||||
|
recipients a copy of this License along with the Program.
|
||||||
|
|
||||||
|
You may charge any price or no price for each copy that you convey,
|
||||||
|
and you may offer support or warranty protection for a fee.
|
||||||
|
|
||||||
|
5. Conveying Modified Source Versions.
|
||||||
|
|
||||||
|
You may convey a work based on the Program, or the modifications to
|
||||||
|
produce it from the Program, in the form of source code under the
|
||||||
|
terms of section 4, provided that you also meet all of these conditions:
|
||||||
|
|
||||||
|
a) The work must carry prominent notices stating that you modified
|
||||||
|
it, and giving a relevant date.
|
||||||
|
|
||||||
|
b) The work must carry prominent notices stating that it is
|
||||||
|
released under this License and any conditions added under section
|
||||||
|
7. This requirement modifies the requirement in section 4 to
|
||||||
|
"keep intact all notices".
|
||||||
|
|
||||||
|
c) You must license the entire work, as a whole, under this
|
||||||
|
License to anyone who comes into possession of a copy. This
|
||||||
|
License will therefore apply, along with any applicable section 7
|
||||||
|
additional terms, to the whole of the work, and all its parts,
|
||||||
|
regardless of how they are packaged. This License gives no
|
||||||
|
permission to license the work in any other way, but it does not
|
||||||
|
invalidate such permission if you have separately received it.
|
||||||
|
|
||||||
|
d) If the work has interactive user interfaces, each must display
|
||||||
|
Appropriate Legal Notices; however, if the Program has interactive
|
||||||
|
interfaces that do not display Appropriate Legal Notices, your
|
||||||
|
work need not make them do so.
|
||||||
|
|
||||||
|
A compilation of a covered work with other separate and independent
|
||||||
|
works, which are not by their nature extensions of the covered work,
|
||||||
|
and which are not combined with it such as to form a larger program,
|
||||||
|
in or on a volume of a storage or distribution medium, is called an
|
||||||
|
"aggregate" if the compilation and its resulting copyright are not
|
||||||
|
used to limit the access or legal rights of the compilation's users
|
||||||
|
beyond what the individual works permit. Inclusion of a covered work
|
||||||
|
in an aggregate does not cause this License to apply to the other
|
||||||
|
parts of the aggregate.
|
||||||
|
|
||||||
|
6. Conveying Non-Source Forms.
|
||||||
|
|
||||||
|
You may convey a covered work in object code form under the terms
|
||||||
|
of sections 4 and 5, provided that you also convey the
|
||||||
|
machine-readable Corresponding Source under the terms of this License,
|
||||||
|
in one of these ways:
|
||||||
|
|
||||||
|
a) Convey the object code in, or embodied in, a physical product
|
||||||
|
(including a physical distribution medium), accompanied by the
|
||||||
|
Corresponding Source fixed on a durable physical medium
|
||||||
|
customarily used for software interchange.
|
||||||
|
|
||||||
|
b) Convey the object code in, or embodied in, a physical product
|
||||||
|
(including a physical distribution medium), accompanied by a
|
||||||
|
written offer, valid for at least three years and valid for as
|
||||||
|
long as you offer spare parts or customer support for that product
|
||||||
|
model, to give anyone who possesses the object code either (1) a
|
||||||
|
copy of the Corresponding Source for all the software in the
|
||||||
|
product that is covered by this License, on a durable physical
|
||||||
|
medium customarily used for software interchange, for a price no
|
||||||
|
more than your reasonable cost of physically performing this
|
||||||
|
conveying of source, or (2) access to copy the
|
||||||
|
Corresponding Source from a network server at no charge.
|
||||||
|
|
||||||
|
c) Convey individual copies of the object code with a copy of the
|
||||||
|
written offer to provide the Corresponding Source. This
|
||||||
|
alternative is allowed only occasionally and noncommercially, and
|
||||||
|
only if you received the object code with such an offer, in accord
|
||||||
|
with subsection 6b.
|
||||||
|
|
||||||
|
d) Convey the object code by offering access from a designated
|
||||||
|
place (gratis or for a charge), and offer equivalent access to the
|
||||||
|
Corresponding Source in the same way through the same place at no
|
||||||
|
further charge. You need not require recipients to copy the
|
||||||
|
Corresponding Source along with the object code. If the place to
|
||||||
|
copy the object code is a network server, the Corresponding Source
|
||||||
|
may be on a different server (operated by you or a third party)
|
||||||
|
that supports equivalent copying facilities, provided you maintain
|
||||||
|
clear directions next to the object code saying where to find the
|
||||||
|
Corresponding Source. Regardless of what server hosts the
|
||||||
|
Corresponding Source, you remain obligated to ensure that it is
|
||||||
|
available for as long as needed to satisfy these requirements.
|
||||||
|
|
||||||
|
e) Convey the object code using peer-to-peer transmission, provided
|
||||||
|
you inform other peers where the object code and Corresponding
|
||||||
|
Source of the work are being offered to the general public at no
|
||||||
|
charge under subsection 6d.
|
||||||
|
|
||||||
|
A separable portion of the object code, whose source code is excluded
|
||||||
|
from the Corresponding Source as a System Library, need not be
|
||||||
|
included in conveying the object code work.
|
||||||
|
|
||||||
|
A "User Product" is either (1) a "consumer product", which means any
|
||||||
|
tangible personal property which is normally used for personal, family,
|
||||||
|
or household purposes, or (2) anything designed or sold for incorporation
|
||||||
|
into a dwelling. In determining whether a product is a consumer product,
|
||||||
|
doubtful cases shall be resolved in favor of coverage. For a particular
|
||||||
|
product received by a particular user, "normally used" refers to a
|
||||||
|
typical or common use of that class of product, regardless of the status
|
||||||
|
of the particular user or of the way in which the particular user
|
||||||
|
actually uses, or expects or is expected to use, the product. A product
|
||||||
|
is a consumer product regardless of whether the product has substantial
|
||||||
|
commercial, industrial or non-consumer uses, unless such uses represent
|
||||||
|
the only significant mode of use of the product.
|
||||||
|
|
||||||
|
"Installation Information" for a User Product means any methods,
|
||||||
|
procedures, authorization keys, or other information required to install
|
||||||
|
and execute modified versions of a covered work in that User Product from
|
||||||
|
a modified version of its Corresponding Source. The information must
|
||||||
|
suffice to ensure that the continued functioning of the modified object
|
||||||
|
code is in no case prevented or interfered with solely because
|
||||||
|
modification has been made.
|
||||||
|
|
||||||
|
If you convey an object code work under this section in, or with, or
|
||||||
|
specifically for use in, a User Product, and the conveying occurs as
|
||||||
|
part of a transaction in which the right of possession and use of the
|
||||||
|
User Product is transferred to the recipient in perpetuity or for a
|
||||||
|
fixed term (regardless of how the transaction is characterized), the
|
||||||
|
Corresponding Source conveyed under this section must be accompanied
|
||||||
|
by the Installation Information. But this requirement does not apply
|
||||||
|
if neither you nor any third party retains the ability to install
|
||||||
|
modified object code on the User Product (for example, the work has
|
||||||
|
been installed in ROM).
|
||||||
|
|
||||||
|
The requirement to provide Installation Information does not include a
|
||||||
|
requirement to continue to provide support service, warranty, or updates
|
||||||
|
for a work that has been modified or installed by the recipient, or for
|
||||||
|
the User Product in which it has been modified or installed. Access to a
|
||||||
|
network may be denied when the modification itself materially and
|
||||||
|
adversely affects the operation of the network or violates the rules and
|
||||||
|
protocols for communication across the network.
|
||||||
|
|
||||||
|
Corresponding Source conveyed, and Installation Information provided,
|
||||||
|
in accord with this section must be in a format that is publicly
|
||||||
|
documented (and with an implementation available to the public in
|
||||||
|
source code form), and must require no special password or key for
|
||||||
|
unpacking, reading or copying.
|
||||||
|
|
||||||
|
7. Additional Terms.
|
||||||
|
|
||||||
|
"Additional permissions" are terms that supplement the terms of this
|
||||||
|
License by making exceptions from one or more of its conditions.
|
||||||
|
Additional permissions that are applicable to the entire Program shall
|
||||||
|
be treated as though they were included in this License, to the extent
|
||||||
|
that they are valid under applicable law. If additional permissions
|
||||||
|
apply only to part of the Program, that part may be used separately
|
||||||
|
under those permissions, but the entire Program remains governed by
|
||||||
|
this License without regard to the additional permissions.
|
||||||
|
|
||||||
|
When you convey a copy of a covered work, you may at your option
|
||||||
|
remove any additional permissions from that copy, or from any part of
|
||||||
|
it. (Additional permissions may be written to require their own
|
||||||
|
removal in certain cases when you modify the work.) You may place
|
||||||
|
additional permissions on material, added by you to a covered work,
|
||||||
|
for which you have or can give appropriate copyright permission.
|
||||||
|
|
||||||
|
Notwithstanding any other provision of this License, for material you
|
||||||
|
add to a covered work, you may (if authorized by the copyright holders of
|
||||||
|
that material) supplement the terms of this License with terms:
|
||||||
|
|
||||||
|
a) Disclaiming warranty or limiting liability differently from the
|
||||||
|
terms of sections 15 and 16 of this License; or
|
||||||
|
|
||||||
|
b) Requiring preservation of specified reasonable legal notices or
|
||||||
|
author attributions in that material or in the Appropriate Legal
|
||||||
|
Notices displayed by works containing it; or
|
||||||
|
|
||||||
|
c) Prohibiting misrepresentation of the origin of that material, or
|
||||||
|
requiring that modified versions of such material be marked in
|
||||||
|
reasonable ways as different from the original version; or
|
||||||
|
|
||||||
|
d) Limiting the use for publicity purposes of names of licensors or
|
||||||
|
authors of the material; or
|
||||||
|
|
||||||
|
e) Declining to grant rights under trademark law for use of some
|
||||||
|
trade names, trademarks, or service marks; or
|
||||||
|
|
||||||
|
f) Requiring indemnification of licensors and authors of that
|
||||||
|
material by anyone who conveys the material (or modified versions of
|
||||||
|
it) with contractual assumptions of liability to the recipient, for
|
||||||
|
any liability that these contractual assumptions directly impose on
|
||||||
|
those licensors and authors.
|
||||||
|
|
||||||
|
All other non-permissive additional terms are considered "further
|
||||||
|
restrictions" within the meaning of section 10. If the Program as you
|
||||||
|
received it, or any part of it, contains a notice stating that it is
|
||||||
|
governed by this License along with a term that is a further
|
||||||
|
restriction, you may remove that term. If a license document contains
|
||||||
|
a further restriction but permits relicensing or conveying under this
|
||||||
|
License, you may add to a covered work material governed by the terms
|
||||||
|
of that license document, provided that the further restriction does
|
||||||
|
not survive such relicensing or conveying.
|
||||||
|
|
||||||
|
If you add terms to a covered work in accord with this section, you
|
||||||
|
must place, in the relevant source files, a statement of the
|
||||||
|
additional terms that apply to those files, or a notice indicating
|
||||||
|
where to find the applicable terms.
|
||||||
|
|
||||||
|
Additional terms, permissive or non-permissive, may be stated in the
|
||||||
|
form of a separately written license, or stated as exceptions;
|
||||||
|
the above requirements apply either way.
|
||||||
|
|
||||||
|
8. Termination.
|
||||||
|
|
||||||
|
You may not propagate or modify a covered work except as expressly
|
||||||
|
provided under this License. Any attempt otherwise to propagate or
|
||||||
|
modify it is void, and will automatically terminate your rights under
|
||||||
|
this License (including any patent licenses granted under the third
|
||||||
|
paragraph of section 11).
|
||||||
|
|
||||||
|
However, if you cease all violation of this License, then your
|
||||||
|
license from a particular copyright holder is reinstated (a)
|
||||||
|
provisionally, unless and until the copyright holder explicitly and
|
||||||
|
finally terminates your license, and (b) permanently, if the copyright
|
||||||
|
holder fails to notify you of the violation by some reasonable means
|
||||||
|
prior to 60 days after the cessation.
|
||||||
|
|
||||||
|
Moreover, your license from a particular copyright holder is
|
||||||
|
reinstated permanently if the copyright holder notifies you of the
|
||||||
|
violation by some reasonable means, this is the first time you have
|
||||||
|
received notice of violation of this License (for any work) from that
|
||||||
|
copyright holder, and you cure the violation prior to 30 days after
|
||||||
|
your receipt of the notice.
|
||||||
|
|
||||||
|
Termination of your rights under this section does not terminate the
|
||||||
|
licenses of parties who have received copies or rights from you under
|
||||||
|
this License. If your rights have been terminated and not permanently
|
||||||
|
reinstated, you do not qualify to receive new licenses for the same
|
||||||
|
material under section 10.
|
||||||
|
|
||||||
|
9. Acceptance Not Required for Having Copies.
|
||||||
|
|
||||||
|
You are not required to accept this License in order to receive or
|
||||||
|
run a copy of the Program. Ancillary propagation of a covered work
|
||||||
|
occurring solely as a consequence of using peer-to-peer transmission
|
||||||
|
to receive a copy likewise does not require acceptance. However,
|
||||||
|
nothing other than this License grants you permission to propagate or
|
||||||
|
modify any covered work. These actions infringe copyright if you do
|
||||||
|
not accept this License. Therefore, by modifying or propagating a
|
||||||
|
covered work, you indicate your acceptance of this License to do so.
|
||||||
|
|
||||||
|
10. Automatic Licensing of Downstream Recipients.
|
||||||
|
|
||||||
|
Each time you convey a covered work, the recipient automatically
|
||||||
|
receives a license from the original licensors, to run, modify and
|
||||||
|
propagate that work, subject to this License. You are not responsible
|
||||||
|
for enforcing compliance by third parties with this License.
|
||||||
|
|
||||||
|
An "entity transaction" is a transaction transferring control of an
|
||||||
|
organization, or substantially all assets of one, or subdividing an
|
||||||
|
organization, or merging organizations. If propagation of a covered
|
||||||
|
work results from an entity transaction, each party to that
|
||||||
|
transaction who receives a copy of the work also receives whatever
|
||||||
|
licenses to the work the party's predecessor in interest had or could
|
||||||
|
give under the previous paragraph, plus a right to possession of the
|
||||||
|
Corresponding Source of the work from the predecessor in interest, if
|
||||||
|
the predecessor has it or can get it with reasonable efforts.
|
||||||
|
|
||||||
|
You may not impose any further restrictions on the exercise of the
|
||||||
|
rights granted or affirmed under this License. For example, you may
|
||||||
|
not impose a license fee, royalty, or other charge for exercise of
|
||||||
|
rights granted under this License, and you may not initiate litigation
|
||||||
|
(including a cross-claim or counterclaim in a lawsuit) alleging that
|
||||||
|
any patent claim is infringed by making, using, selling, offering for
|
||||||
|
sale, or importing the Program or any portion of it.
|
||||||
|
|
||||||
|
11. Patents.
|
||||||
|
|
||||||
|
A "contributor" is a copyright holder who authorizes use under this
|
||||||
|
License of the Program or a work on which the Program is based. The
|
||||||
|
work thus licensed is called the contributor's "contributor version".
|
||||||
|
|
||||||
|
A contributor's "essential patent claims" are all patent claims
|
||||||
|
owned or controlled by the contributor, whether already acquired or
|
||||||
|
hereafter acquired, that would be infringed by some manner, permitted
|
||||||
|
by this License, of making, using, or selling its contributor version,
|
||||||
|
but do not include claims that would be infringed only as a
|
||||||
|
consequence of further modification of the contributor version. For
|
||||||
|
purposes of this definition, "control" includes the right to grant
|
||||||
|
patent sublicenses in a manner consistent with the requirements of
|
||||||
|
this License.
|
||||||
|
|
||||||
|
Each contributor grants you a non-exclusive, worldwide, royalty-free
|
||||||
|
patent license under the contributor's essential patent claims, to
|
||||||
|
make, use, sell, offer for sale, import and otherwise run, modify and
|
||||||
|
propagate the contents of its contributor version.
|
||||||
|
|
||||||
|
In the following three paragraphs, a "patent license" is any express
|
||||||
|
agreement or commitment, however denominated, not to enforce a patent
|
||||||
|
(such as an express permission to practice a patent or covenant not to
|
||||||
|
sue for patent infringement). To "grant" such a patent license to a
|
||||||
|
party means to make such an agreement or commitment not to enforce a
|
||||||
|
patent against the party.
|
||||||
|
|
||||||
|
If you convey a covered work, knowingly relying on a patent license,
|
||||||
|
and the Corresponding Source of the work is not available for anyone
|
||||||
|
to copy, free of charge and under the terms of this License, through a
|
||||||
|
publicly available network server or other readily accessible means,
|
||||||
|
then you must either (1) cause the Corresponding Source to be so
|
||||||
|
available, or (2) arrange to deprive yourself of the benefit of the
|
||||||
|
patent license for this particular work, or (3) arrange, in a manner
|
||||||
|
consistent with the requirements of this License, to extend the patent
|
||||||
|
license to downstream recipients. "Knowingly relying" means you have
|
||||||
|
actual knowledge that, but for the patent license, your conveying the
|
||||||
|
covered work in a country, or your recipient's use of the covered work
|
||||||
|
in a country, would infringe one or more identifiable patents in that
|
||||||
|
country that you have reason to believe are valid.
|
||||||
|
|
||||||
|
If, pursuant to or in connection with a single transaction or
|
||||||
|
arrangement, you convey, or propagate by procuring conveyance of, a
|
||||||
|
covered work, and grant a patent license to some of the parties
|
||||||
|
receiving the covered work authorizing them to use, propagate, modify
|
||||||
|
or convey a specific copy of the covered work, then the patent license
|
||||||
|
you grant is automatically extended to all recipients of the covered
|
||||||
|
work and works based on it.
|
||||||
|
|
||||||
|
A patent license is "discriminatory" if it does not include within
|
||||||
|
the scope of its coverage, prohibits the exercise of, or is
|
||||||
|
conditioned on the non-exercise of one or more of the rights that are
|
||||||
|
specifically granted under this License. You may not convey a covered
|
||||||
|
work if you are a party to an arrangement with a third party that is
|
||||||
|
in the business of distributing software, under which you make payment
|
||||||
|
to the third party based on the extent of your activity of conveying
|
||||||
|
the work, and under which the third party grants, to any of the
|
||||||
|
parties who would receive the covered work from you, a discriminatory
|
||||||
|
patent license (a) in connection with copies of the covered work
|
||||||
|
conveyed by you (or copies made from those copies), or (b) primarily
|
||||||
|
for and in connection with specific products or compilations that
|
||||||
|
contain the covered work, unless you entered into that arrangement,
|
||||||
|
or that patent license was granted, prior to 28 March 2007.
|
||||||
|
|
||||||
|
Nothing in this License shall be construed as excluding or limiting
|
||||||
|
any implied license or other defenses to infringement that may
|
||||||
|
otherwise be available to you under applicable patent law.
|
||||||
|
|
||||||
|
12. No Surrender of Others' Freedom.
|
||||||
|
|
||||||
|
If conditions are imposed on you (whether by court order, agreement or
|
||||||
|
otherwise) that contradict the conditions of this License, they do not
|
||||||
|
excuse you from the conditions of this License. If you cannot convey a
|
||||||
|
covered work so as to satisfy simultaneously your obligations under this
|
||||||
|
License and any other pertinent obligations, then as a consequence you may
|
||||||
|
not convey it at all. For example, if you agree to terms that obligate you
|
||||||
|
to collect a royalty for further conveying from those to whom you convey
|
||||||
|
the Program, the only way you could satisfy both those terms and this
|
||||||
|
License would be to refrain entirely from conveying the Program.
|
||||||
|
|
||||||
|
13. Use with the GNU Affero General Public License.
|
||||||
|
|
||||||
|
Notwithstanding any other provision of this License, you have
|
||||||
|
permission to link or combine any covered work with a work licensed
|
||||||
|
under version 3 of the GNU Affero General Public License into a single
|
||||||
|
combined work, and to convey the resulting work. The terms of this
|
||||||
|
License will continue to apply to the part which is the covered work,
|
||||||
|
but the special requirements of the GNU Affero General Public License,
|
||||||
|
section 13, concerning interaction through a network will apply to the
|
||||||
|
combination as such.
|
||||||
|
|
||||||
|
14. Revised Versions of this License.
|
||||||
|
|
||||||
|
The Free Software Foundation may publish revised and/or new versions of
|
||||||
|
the GNU General Public License from time to time. Such new versions will
|
||||||
|
be similar in spirit to the present version, but may differ in detail to
|
||||||
|
address new problems or concerns.
|
||||||
|
|
||||||
|
Each version is given a distinguishing version number. If the
|
||||||
|
Program specifies that a certain numbered version of the GNU General
|
||||||
|
Public License "or any later version" applies to it, you have the
|
||||||
|
option of following the terms and conditions either of that numbered
|
||||||
|
version or of any later version published by the Free Software
|
||||||
|
Foundation. If the Program does not specify a version number of the
|
||||||
|
GNU General Public License, you may choose any version ever published
|
||||||
|
by the Free Software Foundation.
|
||||||
|
|
||||||
|
If the Program specifies that a proxy can decide which future
|
||||||
|
versions of the GNU General Public License can be used, that proxy's
|
||||||
|
public statement of acceptance of a version permanently authorizes you
|
||||||
|
to choose that version for the Program.
|
||||||
|
|
||||||
|
Later license versions may give you additional or different
|
||||||
|
permissions. However, no additional obligations are imposed on any
|
||||||
|
author or copyright holder as a result of your choosing to follow a
|
||||||
|
later version.
|
||||||
|
|
||||||
|
15. Disclaimer of Warranty.
|
||||||
|
|
||||||
|
THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY
|
||||||
|
APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT
|
||||||
|
HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY
|
||||||
|
OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO,
|
||||||
|
THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
|
||||||
|
PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM
|
||||||
|
IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF
|
||||||
|
ALL NECESSARY SERVICING, REPAIR OR CORRECTION.
|
||||||
|
|
||||||
|
16. Limitation of Liability.
|
||||||
|
|
||||||
|
IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
|
||||||
|
WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS
|
||||||
|
THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY
|
||||||
|
GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE
|
||||||
|
USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF
|
||||||
|
DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD
|
||||||
|
PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS),
|
||||||
|
EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF
|
||||||
|
SUCH DAMAGES.
|
||||||
|
|
||||||
|
17. Interpretation of Sections 15 and 16.
|
||||||
|
|
||||||
|
If the disclaimer of warranty and limitation of liability provided
|
||||||
|
above cannot be given local legal effect according to their terms,
|
||||||
|
reviewing courts shall apply local law that most closely approximates
|
||||||
|
an absolute waiver of all civil liability in connection with the
|
||||||
|
Program, unless a warranty or assumption of liability accompanies a
|
||||||
|
copy of the Program in return for a fee.
|
||||||
|
|
||||||
|
END OF TERMS AND CONDITIONS
|
||||||
|
|
||||||
|
How to Apply These Terms to Your New Programs
|
||||||
|
|
||||||
|
If you develop a new program, and you want it to be of the greatest
|
||||||
|
possible use to the public, the best way to achieve this is to make it
|
||||||
|
free software which everyone can redistribute and change under these terms.
|
||||||
|
|
||||||
|
To do so, attach the following notices to the program. It is safest
|
||||||
|
to attach them to the start of each source file to most effectively
|
||||||
|
state the exclusion of warranty; and each file should have at least
|
||||||
|
the "copyright" line and a pointer to where the full notice is found.
|
||||||
|
|
||||||
|
<one line to give the program's name and a brief idea of what it does.>
|
||||||
|
Copyright (C) <year> <name of author>
|
||||||
|
|
||||||
|
This program is free software: you can redistribute it and/or modify
|
||||||
|
it under the terms of the GNU General Public License as published by
|
||||||
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
|
(at your option) any later version.
|
||||||
|
|
||||||
|
This program is distributed in the hope that it will be useful,
|
||||||
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
GNU General Public License for more details.
|
||||||
|
|
||||||
|
You should have received a copy of the GNU General Public License
|
||||||
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
Also add information on how to contact you by electronic and paper mail.
|
||||||
|
|
||||||
|
If the program does terminal interaction, make it output a short
|
||||||
|
notice like this when it starts in an interactive mode:
|
||||||
|
|
||||||
|
<program> Copyright (C) <year> <name of author>
|
||||||
|
This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'.
|
||||||
|
This is free software, and you are welcome to redistribute it
|
||||||
|
under certain conditions; type `show c' for details.
|
||||||
|
|
||||||
|
The hypothetical commands `show w' and `show c' should show the appropriate
|
||||||
|
parts of the General Public License. Of course, your program's commands
|
||||||
|
might be different; for a GUI interface, you would use an "about box".
|
||||||
|
|
||||||
|
You should also get your employer (if you work as a programmer) or school,
|
||||||
|
if any, to sign a "copyright disclaimer" for the program, if necessary.
|
||||||
|
For more information on this, and how to apply and follow the GNU GPL, see
|
||||||
|
<http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
The GNU General Public License does not permit incorporating your program
|
||||||
|
into proprietary programs. If your program is a subroutine library, you
|
||||||
|
may consider it more useful to permit linking proprietary applications with
|
||||||
|
the library. If this is what you want to do, use the GNU Lesser General
|
||||||
|
Public License instead of this License. But first, please read
|
||||||
|
<http://www.gnu.org/philosophy/why-not-lgpl.html>.
|
||||||
|
|||||||
@@ -38,6 +38,4 @@ After building, the output .JAR file should be in the output JAR file should be
|
|||||||
|
|
||||||
<a href="https://anuke.itch.io/mindustry"><img src="https://i.imgur.com/sk26hTV.png" width="auto" height="75"></a>
|
<a href="https://anuke.itch.io/mindustry"><img src="https://i.imgur.com/sk26hTV.png" width="auto" height="75"></a>
|
||||||
|
|
||||||
<a href="https://anuke.itch.io/mindustry"><img src="https://i.imgur.com/ZSPVQpn.gif" width="auto" height="75"></a>
|
|
||||||
|
|
||||||
<a href="https://play.google.com/store/apps/details?id=io.anuke.mindustry&hl=en"><img src="https://i.imgur.com/8dF6l81.png" width="auto" height="75"></a>
|
<a href="https://play.google.com/store/apps/details?id=io.anuke.mindustry&hl=en"><img src="https://i.imgur.com/8dF6l81.png" width="auto" height="75"></a>
|
||||||
|
|||||||
@@ -9,22 +9,58 @@
|
|||||||
<uses-permission android:name="android.permission.INTERNET" />
|
<uses-permission android:name="android.permission.INTERNET" />
|
||||||
|
|
||||||
<application
|
<application
|
||||||
android:allowBackup="true"
|
android:resizeableActivity="false"
|
||||||
android:icon="@mipmap/ic_launcher"
|
android:allowBackup="true"
|
||||||
android:roundIcon="@mipmap/ic_launcher_round"
|
android:icon="@mipmap/ic_launcher"
|
||||||
android:label="@string/app_name"
|
android:roundIcon="@mipmap/ic_launcher_round"
|
||||||
android:theme="@style/GdxTheme" >
|
android:isGame="true"
|
||||||
|
android:appCategory="game"
|
||||||
|
android:label="@string/app_name"
|
||||||
|
android:theme="@style/GdxTheme" android:fullBackupContent="@xml/backup_rules">
|
||||||
<activity
|
<activity
|
||||||
android:name="io.anuke.mindustry.AndroidLauncher"
|
android:name="io.anuke.mindustry.AndroidLauncher"
|
||||||
android:label="@string/app_name"
|
android:label="@string/app_name"
|
||||||
android:screenOrientation="sensorLandscape"
|
android:screenOrientation="sensor"
|
||||||
android:configChanges="keyboard|keyboardHidden|orientation|screenSize">
|
android:configChanges="keyboard|keyboardHidden|orientation|screenSize">
|
||||||
|
|
||||||
<intent-filter>
|
<intent-filter>
|
||||||
<action android:name="android.intent.action.MAIN" />
|
<action android:name="android.intent.action.MAIN" />
|
||||||
<category android:name="android.intent.category.LAUNCHER" />
|
<category android:name="android.intent.category.LAUNCHER" />
|
||||||
</intent-filter>
|
</intent-filter>
|
||||||
|
|
||||||
|
<intent-filter>
|
||||||
|
<action android:name="android.intent.action.VIEW"/>
|
||||||
|
<category android:name="android.intent.category.DEFAULT"/>
|
||||||
|
<category android:name="android.intent.category.BROWSABLE"/>
|
||||||
|
<data android:mimeType="application/octet-stream"/>
|
||||||
|
<data android:scheme="file" android:host="*" android:pathPattern=".*\\.mmap" />
|
||||||
|
</intent-filter>
|
||||||
|
|
||||||
|
<intent-filter>
|
||||||
|
<action android:name="android.intent.action.VIEW"/>
|
||||||
|
<category android:name="android.intent.category.DEFAULT"/>
|
||||||
|
<category android:name="android.intent.category.BROWSABLE"/>
|
||||||
|
<data android:mimeType="application/octet-stream"/>
|
||||||
|
<data android:scheme="file" android:host="*" android:pathPattern=".*\\.msav" />
|
||||||
|
</intent-filter>
|
||||||
|
|
||||||
|
|
||||||
|
<intent-filter android:icon="@mipmap/ic_launcher"
|
||||||
|
android:label="Mindustry Map"
|
||||||
|
android:priority="1">
|
||||||
|
<action android:name="android.intent.action.VIEW" />
|
||||||
|
<category android:name="android.intent.category.DEFAULT" />
|
||||||
|
<data android:scheme="file" android:host="*" android:pathPattern=".*\\.mmap" android:mimeType="*/*" />
|
||||||
|
</intent-filter>
|
||||||
|
|
||||||
|
<intent-filter>
|
||||||
|
<action android:name="android.intent.action.VIEW" />
|
||||||
|
<category android:name="android.intent.category.DEFAULT" />
|
||||||
|
<data android:scheme="file" android:host="*" android:pathPattern=".*\\.msav" android:mimeType="*/*" />
|
||||||
|
</intent-filter>
|
||||||
</activity>
|
</activity>
|
||||||
|
|
||||||
|
|
||||||
<activity android:name=".DonationsActivity"
|
<activity android:name=".DonationsActivity"
|
||||||
android:theme="@style/GdxTheme" />
|
android:theme="@style/GdxTheme" />
|
||||||
|
|
||||||
|
|||||||
@@ -1,4 +1,8 @@
|
|||||||
|
|
||||||
|
apply plugin: "com.android.application"
|
||||||
|
|
||||||
|
configurations { natives }
|
||||||
|
|
||||||
repositories {
|
repositories {
|
||||||
mavenCentral()
|
mavenCentral()
|
||||||
jcenter()
|
jcenter()
|
||||||
@@ -8,8 +12,20 @@ repositories {
|
|||||||
}
|
}
|
||||||
|
|
||||||
dependencies {
|
dependencies {
|
||||||
implementation 'com.android.support:support-v4:22.1.1'
|
implementation project(":core")
|
||||||
|
implementation project(":kryonet")
|
||||||
|
implementation 'com.android.support:support-v4:25.3.1'
|
||||||
implementation 'org.sufficientlysecure:donations:2.5'
|
implementation 'org.sufficientlysecure:donations:2.5'
|
||||||
|
implementation 'com.google.android.gms:play-services-auth:11.8.0'
|
||||||
|
|
||||||
|
implementation "com.badlogicgames.gdx:gdx-backend-android:$gdxVersion"
|
||||||
|
natives "com.badlogicgames.gdx:gdx-platform:$gdxVersion:natives-armeabi"
|
||||||
|
natives "com.badlogicgames.gdx:gdx-platform:$gdxVersion:natives-armeabi-v7a"
|
||||||
|
natives "com.badlogicgames.gdx:gdx-platform:$gdxVersion:natives-arm64-v8a"
|
||||||
|
natives "com.badlogicgames.gdx:gdx-platform:$gdxVersion:natives-x86"
|
||||||
|
natives "com.badlogicgames.gdx:gdx-platform:$gdxVersion:natives-x86_64"
|
||||||
|
implementation "com.badlogicgames.gdx:gdx-ai:$aiVersion"
|
||||||
|
implementation "com.badlogicgames.gdx:gdx-controllers-android:$gdxVersion"
|
||||||
}
|
}
|
||||||
|
|
||||||
task deploy(type: Copy){
|
task deploy(type: Copy){
|
||||||
@@ -21,8 +37,8 @@ task deploy(type: Copy){
|
|||||||
}
|
}
|
||||||
|
|
||||||
android {
|
android {
|
||||||
buildToolsVersion '26.0.2'
|
buildToolsVersion '27.0.3'
|
||||||
compileSdkVersion 26
|
compileSdkVersion 27
|
||||||
sourceSets {
|
sourceSets {
|
||||||
main {
|
main {
|
||||||
manifest.srcFile 'AndroidManifest.xml'
|
manifest.srcFile 'AndroidManifest.xml'
|
||||||
@@ -34,7 +50,7 @@ android {
|
|||||||
jniLibs.srcDirs = ['libs']
|
jniLibs.srcDirs = ['libs']
|
||||||
}
|
}
|
||||||
|
|
||||||
instrumentTest.setRoot('tests')
|
androidTest.setRoot('tests')
|
||||||
}
|
}
|
||||||
packagingOptions {
|
packagingOptions {
|
||||||
exclude 'META-INF/robovm/ios/robovm.xml'
|
exclude 'META-INF/robovm/ios/robovm.xml'
|
||||||
@@ -42,18 +58,26 @@ android {
|
|||||||
|
|
||||||
defaultConfig {
|
defaultConfig {
|
||||||
def vfile = file('../core/assets/version.properties')
|
def vfile = file('../core/assets/version.properties')
|
||||||
def props = new Properties()
|
|
||||||
props.load(new FileInputStream(vfile))
|
|
||||||
|
|
||||||
def code = props['androidBuildCode'].toInteger() + 1
|
def code = 0
|
||||||
props['androidBuildCode'] = code.toString()
|
def versionNameResult = "unknown"
|
||||||
props.store(vfile.newWriter(), "Autogenerated file. Do not modify.")
|
|
||||||
|
if(vfile.exists()){
|
||||||
|
def props = new Properties()
|
||||||
|
props.load(new FileInputStream(vfile))
|
||||||
|
|
||||||
|
code = (props['androidBuildCode'] == null ? 0 : props['androidBuildCode']).toInteger() + 1
|
||||||
|
|
||||||
|
props['androidBuildCode'] = code.toString()
|
||||||
|
props.store(vfile.newWriter(), "Autogenerated file. Do not modify.")
|
||||||
|
versionNameResult = "$versionNumber-$versionType-${props['build'].replace(" ", "-")}"
|
||||||
|
}
|
||||||
|
|
||||||
applicationId "io.anuke.mindustry"
|
applicationId "io.anuke.mindustry"
|
||||||
minSdkVersion 9
|
minSdkVersion 14
|
||||||
targetSdkVersion 26
|
targetSdkVersion 27
|
||||||
versionCode code
|
versionCode code
|
||||||
versionName "$versionNumber-$versionType-${props['build'].replace(" ", "-")}"
|
versionName versionNameResult
|
||||||
}
|
}
|
||||||
|
|
||||||
compileOptions {
|
compileOptions {
|
||||||
|
|||||||
|
Before Width: | Height: | Size: 2.1 KiB |
|
Before Width: | Height: | Size: 3.9 KiB |
|
Before Width: | Height: | Size: 1.6 KiB |
|
Before Width: | Height: | Size: 2.3 KiB |
|
Before Width: | Height: | Size: 3.0 KiB |
|
Before Width: | Height: | Size: 3.8 KiB |
|
Before Width: | Height: | Size: 2.1 KiB |
@@ -1,22 +0,0 @@
|
|||||||
<?xml version="1.0" encoding="utf-8"?>
|
|
||||||
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
|
|
||||||
android:id="@+id/layout_root"
|
|
||||||
android:layout_width="fill_parent"
|
|
||||||
android:layout_height="fill_parent"
|
|
||||||
android:orientation="vertical"
|
|
||||||
android:padding="10dp" >
|
|
||||||
|
|
||||||
<EditText
|
|
||||||
android:id="@+id/gdxDialogsEditTextInput"
|
|
||||||
android:layout_width="match_parent"
|
|
||||||
android:layout_height="wrap_content"
|
|
||||||
android:inputType="text"
|
|
||||||
android:maxLines="1"
|
|
||||||
android:maxLength="15"
|
|
||||||
>
|
|
||||||
|
|
||||||
<requestFocus />
|
|
||||||
|
|
||||||
</EditText>
|
|
||||||
|
|
||||||
</LinearLayout>
|
|
||||||
@@ -6,27 +6,6 @@
|
|||||||
android:orientation="vertical"
|
android:orientation="vertical"
|
||||||
android:padding="10dp" >
|
android:padding="10dp" >
|
||||||
|
|
||||||
<TextView
|
|
||||||
android:id="@+id/gdxDialogsEnterTitle"
|
|
||||||
android:layout_width="wrap_content"
|
|
||||||
android:layout_height="wrap_content"
|
|
||||||
|
|
||||||
android:textAppearance="?android:attr/textAppearanceLarge"
|
|
||||||
|
|
||||||
|
|
||||||
/>
|
|
||||||
|
|
||||||
<TextView
|
|
||||||
android:id="@+id/gdxDialogsEnterMessage"
|
|
||||||
android:layout_width="wrap_content"
|
|
||||||
android:layout_height="wrap_content"
|
|
||||||
|
|
||||||
android:textAppearance="?android:attr/textAppearanceMedium"
|
|
||||||
|
|
||||||
|
|
||||||
/>
|
|
||||||
|
|
||||||
|
|
||||||
<EditText
|
<EditText
|
||||||
android:id="@+id/gdxDialogsEditTextInput"
|
android:id="@+id/gdxDialogsEditTextInput"
|
||||||
android:layout_width="match_parent"
|
android:layout_width="match_parent"
|
||||||
|
|||||||
@@ -1,6 +1,5 @@
|
|||||||
<?xml version="1.0" encoding="utf-8"?>
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
<resources>
|
<resources>
|
||||||
|
|
||||||
<style name="GdxTheme" parent="android:Theme.Material.NoActionBar">
|
<style name="GdxTheme" parent="android:Theme.Material.NoActionBar">
|
||||||
<item name="android:windowBackground">@android:color/transparent</item>
|
<item name="android:windowBackground">@android:color/transparent</item>
|
||||||
<item name="android:colorBackgroundCacheHint">@null</item>
|
<item name="android:colorBackgroundCacheHint">@null</item>
|
||||||
@@ -9,11 +8,4 @@
|
|||||||
<item name="android:windowContentOverlay">@null</item>
|
<item name="android:windowContentOverlay">@null</item>
|
||||||
<item name="android:windowFullscreen">true</item>
|
<item name="android:windowFullscreen">true</item>
|
||||||
</style>
|
</style>
|
||||||
|
|
||||||
<style name="LightBackground" parent="android:Theme.Material.Light" >
|
|
||||||
<item name="android:colorBackground">@android:color/black</item>
|
|
||||||
<item name="android:textColorPrimary">@android:color/white</item>
|
|
||||||
<item name="android:textColorSecondary">@android:color/white</item>
|
|
||||||
<item name="android:textColorSecondaryInverse">@android:color/white</item>
|
|
||||||
</style>
|
|
||||||
</resources>
|
</resources>
|
||||||
4
android/res/xml/backup_rules.xml
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<full-backup-content>
|
||||||
|
|
||||||
|
</full-backup-content>
|
||||||
@@ -3,34 +3,56 @@ package io.anuke.mindustry;
|
|||||||
import android.Manifest;
|
import android.Manifest;
|
||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.content.Intent;
|
import android.content.Intent;
|
||||||
|
import android.content.pm.ActivityInfo;
|
||||||
import android.content.pm.PackageManager;
|
import android.content.pm.PackageManager;
|
||||||
|
import android.net.Uri;
|
||||||
import android.os.Build;
|
import android.os.Build;
|
||||||
import android.os.Bundle;
|
import android.os.Bundle;
|
||||||
import android.provider.Settings.Secure;
|
import android.provider.Settings.Secure;
|
||||||
import android.telephony.TelephonyManager;
|
import android.telephony.TelephonyManager;
|
||||||
|
import android.util.Log;
|
||||||
|
import com.badlogic.gdx.Gdx;
|
||||||
import com.badlogic.gdx.backends.android.AndroidApplication;
|
import com.badlogic.gdx.backends.android.AndroidApplication;
|
||||||
import com.badlogic.gdx.backends.android.AndroidApplicationConfiguration;
|
import com.badlogic.gdx.backends.android.AndroidApplicationConfiguration;
|
||||||
|
import com.badlogic.gdx.files.FileHandle;
|
||||||
import com.badlogic.gdx.utils.Base64Coder;
|
import com.badlogic.gdx.utils.Base64Coder;
|
||||||
|
import com.google.android.gms.common.GoogleApiAvailability;
|
||||||
|
import com.google.android.gms.common.GooglePlayServicesNotAvailableException;
|
||||||
|
import com.google.android.gms.common.GooglePlayServicesRepairableException;
|
||||||
|
import com.google.android.gms.security.ProviderInstaller;
|
||||||
import io.anuke.kryonet.DefaultThreadImpl;
|
import io.anuke.kryonet.DefaultThreadImpl;
|
||||||
import io.anuke.kryonet.KryoClient;
|
import io.anuke.kryonet.KryoClient;
|
||||||
import io.anuke.kryonet.KryoServer;
|
import io.anuke.kryonet.KryoServer;
|
||||||
|
import io.anuke.mindustry.core.Platform;
|
||||||
import io.anuke.mindustry.core.ThreadHandler.ThreadProvider;
|
import io.anuke.mindustry.core.ThreadHandler.ThreadProvider;
|
||||||
import io.anuke.mindustry.io.Platform;
|
import io.anuke.mindustry.io.SaveIO;
|
||||||
|
import io.anuke.mindustry.io.Saves.SaveSlot;
|
||||||
import io.anuke.mindustry.net.Net;
|
import io.anuke.mindustry.net.Net;
|
||||||
import io.anuke.ucore.core.Settings;
|
import io.anuke.mindustry.ui.dialogs.FileChooser;
|
||||||
|
import io.anuke.ucore.function.Consumer;
|
||||||
import io.anuke.ucore.scene.ui.TextField;
|
import io.anuke.ucore.scene.ui.TextField;
|
||||||
import io.anuke.ucore.scene.ui.layout.Unit;
|
import io.anuke.ucore.scene.ui.layout.Unit;
|
||||||
|
import io.anuke.ucore.util.Bundles;
|
||||||
|
import io.anuke.ucore.util.Strings;
|
||||||
|
|
||||||
|
import java.io.File;
|
||||||
|
import java.io.FileInputStream;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.InputStream;
|
||||||
import java.text.DateFormat;
|
import java.text.DateFormat;
|
||||||
import java.text.NumberFormat;
|
import java.text.NumberFormat;
|
||||||
import java.text.SimpleDateFormat;
|
import java.text.SimpleDateFormat;
|
||||||
|
import java.util.ArrayList;
|
||||||
import java.util.Date;
|
import java.util.Date;
|
||||||
import java.util.Locale;
|
import java.util.Locale;
|
||||||
import java.util.Random;
|
|
||||||
|
import static io.anuke.mindustry.Vars.*;
|
||||||
|
|
||||||
public class AndroidLauncher extends AndroidApplication{
|
public class AndroidLauncher extends AndroidApplication{
|
||||||
|
public static final int PERMISSION_REQUEST_CODE = 1;
|
||||||
|
|
||||||
boolean doubleScaleTablets = true;
|
boolean doubleScaleTablets = true;
|
||||||
int WRITE_REQUEST_CODE = 1;
|
FileChooser chooser;
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected void onCreate(Bundle savedInstanceState){
|
protected void onCreate(Bundle savedInstanceState){
|
||||||
@@ -72,26 +94,6 @@ public class AndroidLauncher extends AndroidApplication{
|
|||||||
showDonations();
|
showDonations();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
|
||||||
public void requestWritePerms() {
|
|
||||||
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
|
|
||||||
if(checkSelfPermission(Manifest.permission.WRITE_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED &&
|
|
||||||
checkSelfPermission(Manifest.permission.READ_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED) {
|
|
||||||
requestPermissions(new String[]{Manifest.permission.WRITE_EXTERNAL_STORAGE,
|
|
||||||
Manifest.permission.READ_EXTERNAL_STORAGE}, WRITE_REQUEST_CODE);
|
|
||||||
}else{
|
|
||||||
|
|
||||||
if (checkSelfPermission(Manifest.permission.WRITE_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED) {
|
|
||||||
requestPermissions(new String[]{Manifest.permission.WRITE_EXTERNAL_STORAGE}, WRITE_REQUEST_CODE);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (checkSelfPermission(Manifest.permission.READ_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED) {
|
|
||||||
requestPermissions(new String[]{Manifest.permission.READ_EXTERNAL_STORAGE}, WRITE_REQUEST_CODE);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public ThreadProvider getThreadProvider() {
|
public ThreadProvider getThreadProvider() {
|
||||||
return new DefaultThreadImpl();
|
return new DefaultThreadImpl();
|
||||||
@@ -103,7 +105,7 @@ public class AndroidLauncher extends AndroidApplication{
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public byte[] getUUID() {
|
public String getUUID() {
|
||||||
try {
|
try {
|
||||||
String s = Secure.getString(getContext().getContentResolver(),
|
String s = Secure.getString(getContext().getContentResolver(),
|
||||||
Secure.ANDROID_ID);
|
Secure.ANDROID_ID);
|
||||||
@@ -115,24 +117,69 @@ public class AndroidLauncher extends AndroidApplication{
|
|||||||
+ Character.digit(s.charAt(i + 1), 16));
|
+ Character.digit(s.charAt(i + 1), 16));
|
||||||
}
|
}
|
||||||
|
|
||||||
return data;
|
String result = new String(Base64Coder.encode(data));
|
||||||
}catch (Exception e){
|
|
||||||
Settings.defaults("uuid", "");
|
|
||||||
|
|
||||||
String uuid = Settings.getString("uuid");
|
if(result.equals("AAAAAAAAAOA=")) throw new RuntimeException("Bad UUID.");
|
||||||
if(uuid.isEmpty()){
|
|
||||||
byte[] result = new byte[8];
|
return result;
|
||||||
new Random().nextBytes(result);
|
}catch (Exception e){
|
||||||
uuid = new String(Base64Coder.encode(result));
|
return super.getUUID();
|
||||||
Settings.putString("uuid", uuid);
|
|
||||||
Settings.save();
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
return Base64Coder.decode(uuid);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void shareFile(FileHandle file){
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void showFileChooser(String text, String content, Consumer<FileHandle> cons, boolean open, String filetype) {
|
||||||
|
chooser = new FileChooser(text, file -> file.extension().equalsIgnoreCase(filetype), open, cons);
|
||||||
|
|
||||||
|
if(Build.VERSION.SDK_INT < Build.VERSION_CODES.M || (checkSelfPermission(Manifest.permission.WRITE_EXTERNAL_STORAGE) == PackageManager.PERMISSION_GRANTED &&
|
||||||
|
checkSelfPermission(Manifest.permission.READ_EXTERNAL_STORAGE) == PackageManager.PERMISSION_GRANTED)){
|
||||||
|
chooser.show();
|
||||||
|
chooser = null;
|
||||||
|
}else {
|
||||||
|
ArrayList<String> perms = new ArrayList<>();
|
||||||
|
|
||||||
|
if (checkSelfPermission(Manifest.permission.WRITE_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED) {
|
||||||
|
perms.add(Manifest.permission.WRITE_EXTERNAL_STORAGE);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (checkSelfPermission(Manifest.permission.READ_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED) {
|
||||||
|
perms.add(Manifest.permission.READ_EXTERNAL_STORAGE);
|
||||||
|
}
|
||||||
|
|
||||||
|
requestPermissions(perms.toArray(new String[perms.size()]), PERMISSION_REQUEST_CODE);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void beginForceLandscape() {
|
||||||
|
setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_LANDSCAPE);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void endForceLandscape() {
|
||||||
|
setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_SENSOR);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean canDonate(){
|
||||||
|
return true;
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
try {
|
||||||
|
ProviderInstaller.installIfNeeded(this);
|
||||||
|
} catch (GooglePlayServicesRepairableException e) {
|
||||||
|
GoogleApiAvailability apiAvailability = GoogleApiAvailability.getInstance();
|
||||||
|
apiAvailability.getErrorDialog(this, e.getConnectionStatusCode(), 0).show();
|
||||||
|
} catch (GooglePlayServicesNotAvailableException e) {
|
||||||
|
Log.e("SecurityException", "Google Play Services not available.");
|
||||||
|
}
|
||||||
|
|
||||||
if(doubleScaleTablets && isTablet(this.getContext())){
|
if(doubleScaleTablets && isTablet(this.getContext())){
|
||||||
Unit.dp.addition = 0.5f;
|
Unit.dp.addition = 0.5f;
|
||||||
}
|
}
|
||||||
@@ -143,6 +190,78 @@ public class AndroidLauncher extends AndroidApplication{
|
|||||||
Net.setServerProvider(new KryoServer());
|
Net.setServerProvider(new KryoServer());
|
||||||
|
|
||||||
initialize(new Mindustry(), config);
|
initialize(new Mindustry(), config);
|
||||||
|
|
||||||
|
checkFiles(getIntent());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onRequestPermissionsResult(int requestCode, String permissions[], int[] grantResults) {
|
||||||
|
if(requestCode == PERMISSION_REQUEST_CODE){
|
||||||
|
for(int i : grantResults){
|
||||||
|
if(i != PackageManager.PERMISSION_GRANTED) return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if(chooser != null){
|
||||||
|
chooser.show();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void checkFiles(Intent intent){
|
||||||
|
try {
|
||||||
|
Uri uri = intent.getData();
|
||||||
|
if (uri != null) {
|
||||||
|
File myFile = null;
|
||||||
|
String scheme = uri.getScheme();
|
||||||
|
if (scheme.equals("file")) {
|
||||||
|
String fileName = uri.getEncodedPath();
|
||||||
|
myFile = new File(fileName);
|
||||||
|
} else if (!scheme.equals("content")) {
|
||||||
|
//error
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
boolean save = uri.getPath().endsWith(saveExtension);
|
||||||
|
boolean map = uri.getPath().endsWith(mapExtension);
|
||||||
|
|
||||||
|
InputStream inStream;
|
||||||
|
if (myFile != null) inStream = new FileInputStream(myFile);
|
||||||
|
else inStream = getContentResolver().openInputStream(uri);
|
||||||
|
|
||||||
|
Gdx.app.postRunnable(() -> {
|
||||||
|
|
||||||
|
if(save){ //open save
|
||||||
|
System.out.println("Opening save.");
|
||||||
|
FileHandle file = Gdx.files.local("temp-save." + saveExtension);
|
||||||
|
file.write(inStream, false);
|
||||||
|
|
||||||
|
if(SaveIO.isSaveValid(file)){
|
||||||
|
try{
|
||||||
|
SaveSlot slot = control.getSaves().importSave(file);
|
||||||
|
ui.load.runLoadSave(slot);
|
||||||
|
}catch (IOException e){
|
||||||
|
ui.showError(Bundles.format("text.save.import.fail", Strings.parseException(e, false)));
|
||||||
|
}
|
||||||
|
}else{
|
||||||
|
ui.showError("$text.save.import.invalid");
|
||||||
|
}
|
||||||
|
|
||||||
|
}else if(map){ //open map
|
||||||
|
Gdx.app.postRunnable(() -> {
|
||||||
|
System.out.println("Opening map.");
|
||||||
|
if (!ui.editor.isShown()) {
|
||||||
|
ui.editor.show();
|
||||||
|
}
|
||||||
|
|
||||||
|
ui.editor.beginEditMap(inStream);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
}catch (IOException e){
|
||||||
|
e.printStackTrace();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private boolean isPackageInstalled(String packagename) {
|
private boolean isPackageInstalled(String packagename) {
|
||||||
|
|||||||
@@ -25,7 +25,6 @@ public class AndroidTextFieldDialog{
|
|||||||
public AndroidTextFieldDialog show() {
|
public AndroidTextFieldDialog show() {
|
||||||
|
|
||||||
activity.runOnUiThread(() -> {
|
activity.runOnUiThread(() -> {
|
||||||
Gdx.app.error("Android Dialogs", AndroidTextFieldDialog.class.getSimpleName() + " now shown.");
|
|
||||||
AlertDialog dialog = builder.create();
|
AlertDialog dialog = builder.create();
|
||||||
|
|
||||||
dialog.getWindow().setSoftInputMode(LayoutParams.SOFT_INPUT_STATE_VISIBLE);
|
dialog.getWindow().setSoftInputMode(LayoutParams.SOFT_INPUT_STATE_VISIBLE);
|
||||||
@@ -48,7 +47,7 @@ public class AndroidTextFieldDialog{
|
|||||||
|
|
||||||
alertDialogBuilder.setView(promptsView);
|
alertDialogBuilder.setView(promptsView);
|
||||||
|
|
||||||
userInput = (EditText) promptsView.findViewById(getResourceId("gdxDialogsEditTextInput", "id"));
|
userInput = promptsView.findViewById(getResourceId("gdxDialogsEditTextInput", "id"));
|
||||||
|
|
||||||
alertDialogBuilder.setCancelable(false);
|
alertDialogBuilder.setCancelable(false);
|
||||||
builder = alertDialogBuilder;
|
builder = alertDialogBuilder;
|
||||||
|
|||||||
@@ -26,7 +26,7 @@ public class TextFieldDialogListener extends ClickListener{
|
|||||||
}
|
}
|
||||||
|
|
||||||
public static void add(TextField field){
|
public static void add(TextField field){
|
||||||
add(field, 0, 15);
|
add(field, 0, 16);
|
||||||
}
|
}
|
||||||
|
|
||||||
//type - 0 is text, 1 is numbers, 2 is decimals
|
//type - 0 is text, 1 is numbers, 2 is decimals
|
||||||
|
|||||||
4
annotations/build.gradle
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
apply plugin: "java"
|
||||||
|
|
||||||
|
sourceCompatibility = 1.8
|
||||||
|
sourceSets.main.java.srcDirs = [ "src/" ]
|
||||||
88
annotations/src/io/anuke/annotations/Annotations.java
Normal file
@@ -0,0 +1,88 @@
|
|||||||
|
package io.anuke.annotations;
|
||||||
|
|
||||||
|
import java.lang.annotation.ElementType;
|
||||||
|
import java.lang.annotation.Retention;
|
||||||
|
import java.lang.annotation.RetentionPolicy;
|
||||||
|
import java.lang.annotation.Target;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Goal: To create a system to send events to the server from the client and vice versa, without creating a new packet type each time.<br>
|
||||||
|
* These events may optionally also trigger on the caller client/server as well.<br>
|
||||||
|
*/
|
||||||
|
public class Annotations {
|
||||||
|
|
||||||
|
/**Marks a method as invokable remotely across a server/client connection.*/
|
||||||
|
@Target(ElementType.METHOD)
|
||||||
|
@Retention(RetentionPolicy.CLASS)
|
||||||
|
public @interface Remote {
|
||||||
|
/**Specifies the locations from which this method can be invoked.*/
|
||||||
|
Loc targets() default Loc.server;
|
||||||
|
/**Specifies which methods are generated. Only affects server-to-client methods.*/
|
||||||
|
Variant variants() default Variant.all;
|
||||||
|
/**The local locations where this method is called locally, when invoked.*/
|
||||||
|
Loc called() default Loc.none;
|
||||||
|
/**Whether to forward this packet to all other clients upon recieval. Server only.*/
|
||||||
|
boolean forward() default false;
|
||||||
|
/**Whether the packet for this method is sent with UDP instead of TCP.
|
||||||
|
* UDP is faster, but is prone to packet loss and duplication.*/
|
||||||
|
boolean unreliable() default false;
|
||||||
|
/**The simple class name where this method is placed.*/
|
||||||
|
String in() default "Call";
|
||||||
|
}
|
||||||
|
|
||||||
|
/**Specifies that this method will be used to write classes of the type returned by {@link #value()}.<br>
|
||||||
|
* This method must return void and have two parameters, the first being of type {@link java.nio.ByteBuffer} and the second
|
||||||
|
* being the type returned by {@link #value()}.*/
|
||||||
|
@Target(ElementType.METHOD)
|
||||||
|
@Retention(RetentionPolicy.CLASS)
|
||||||
|
public @interface WriteClass {
|
||||||
|
Class<?> value();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**Specifies that this method will be used to read classes of the type returned by {@link #value()}. <br>
|
||||||
|
* This method must return the type returned by {@link #value()},
|
||||||
|
* and have one parameter, being of type {@link java.nio.ByteBuffer}.*/
|
||||||
|
@Target(ElementType.METHOD)
|
||||||
|
@Retention(RetentionPolicy.CLASS)
|
||||||
|
public @interface ReadClass {
|
||||||
|
Class<?> value();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**A set of two booleans, one specifying server and one specifying client.*/
|
||||||
|
public enum Loc {
|
||||||
|
/**Method can only be invoked on the client from the server.*/
|
||||||
|
server(true, false),
|
||||||
|
/**Method can only be invoked on the server from the client.*/
|
||||||
|
client(false, true),
|
||||||
|
/**Method can be invoked from anywhere*/
|
||||||
|
both(true, true),
|
||||||
|
/**Neither server no client.*/
|
||||||
|
none(false, false);
|
||||||
|
|
||||||
|
/**If true, this method can be invoked ON clients FROM servers.*/
|
||||||
|
public final boolean isServer;
|
||||||
|
/**If true, this method can be invoked ON servers FROM clients.*/
|
||||||
|
public final boolean isClient;
|
||||||
|
|
||||||
|
Loc(boolean server, boolean client){
|
||||||
|
this.isServer = server;
|
||||||
|
this.isClient = client;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public enum Variant {
|
||||||
|
/**Method can only be invoked targeting one player.*/
|
||||||
|
one(true, false),
|
||||||
|
/**Method can only be invoked targeting all players.*/
|
||||||
|
all(false, true),
|
||||||
|
/**Method targets both one player and all players.*/
|
||||||
|
both(true, true);
|
||||||
|
|
||||||
|
public final boolean isOne, isAll;
|
||||||
|
|
||||||
|
Variant(boolean isOne, boolean isAll){
|
||||||
|
this.isOne = isOne;
|
||||||
|
this.isAll = isAll;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
15
annotations/src/io/anuke/annotations/ClassEntry.java
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
package io.anuke.annotations;
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
|
|
||||||
|
/**Represents a class witha list method entries to include in it.*/
|
||||||
|
public class ClassEntry {
|
||||||
|
/**All methods in this generated class.*/
|
||||||
|
public final ArrayList<MethodEntry> methods = new ArrayList<>();
|
||||||
|
/**Simple class name.*/
|
||||||
|
public final String name;
|
||||||
|
|
||||||
|
public ClassEntry(String name) {
|
||||||
|
this.name = name;
|
||||||
|
}
|
||||||
|
}
|
||||||
86
annotations/src/io/anuke/annotations/IOFinder.java
Normal file
@@ -0,0 +1,86 @@
|
|||||||
|
package io.anuke.annotations;
|
||||||
|
|
||||||
|
import io.anuke.annotations.Annotations.ReadClass;
|
||||||
|
import io.anuke.annotations.Annotations.WriteClass;
|
||||||
|
|
||||||
|
import javax.annotation.processing.RoundEnvironment;
|
||||||
|
import javax.lang.model.element.Element;
|
||||||
|
import javax.lang.model.type.MirroredTypeException;
|
||||||
|
import javax.tools.Diagnostic.Kind;
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.Set;
|
||||||
|
|
||||||
|
/**This class finds reader and writer methods annotated by the {@link io.anuke.annotations.Annotations.WriteClass}
|
||||||
|
* and {@link io.anuke.annotations.Annotations.ReadClass} annotations.*/
|
||||||
|
public class IOFinder {
|
||||||
|
|
||||||
|
/**Finds all class serializers for all types and returns them. Logs errors when necessary.
|
||||||
|
* Maps fully qualified class names to their serializers.*/
|
||||||
|
public HashMap<String, ClassSerializer> findSerializers(RoundEnvironment env){
|
||||||
|
HashMap<String, ClassSerializer> result = new HashMap<>();
|
||||||
|
|
||||||
|
//get methods with the types
|
||||||
|
Set<? extends Element> writers = env.getElementsAnnotatedWith(WriteClass.class);
|
||||||
|
Set<? extends Element> readers = env.getElementsAnnotatedWith(ReadClass.class);
|
||||||
|
|
||||||
|
//look for writers first
|
||||||
|
for(Element writer : writers){
|
||||||
|
WriteClass writean = writer.getAnnotation(WriteClass.class);
|
||||||
|
String typeName = getValue(writean);
|
||||||
|
|
||||||
|
//make sure there's only one read method
|
||||||
|
if(readers.stream().filter(elem -> getValue(elem.getAnnotation(ReadClass.class)).equals(typeName)).count() > 1){
|
||||||
|
Utils.messager.printMessage(Kind.ERROR, "Multiple writer methods for type '" + typeName + "'", writer);
|
||||||
|
}
|
||||||
|
|
||||||
|
//make sure there's only one write method
|
||||||
|
long count = readers.stream().filter(elem -> getValue(elem.getAnnotation(ReadClass.class)).equals(typeName)).count();
|
||||||
|
if(count == 0){
|
||||||
|
Utils.messager.printMessage(Kind.ERROR, "Writer method does not have an accompanying reader: ", writer);
|
||||||
|
}else if(count > 1){
|
||||||
|
Utils.messager.printMessage(Kind.ERROR, "Writer method has multiple reader for type: ", writer);
|
||||||
|
}
|
||||||
|
|
||||||
|
Element reader = readers.stream().filter(elem -> getValue(elem.getAnnotation(ReadClass.class)).equals(typeName)).findFirst().get();
|
||||||
|
|
||||||
|
//add to result list
|
||||||
|
result.put(typeName, new ClassSerializer(Utils.getMethodName(reader), Utils.getMethodName(writer), typeName));
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
private String getValue(WriteClass write){
|
||||||
|
try {
|
||||||
|
Class<?> type = write.value();
|
||||||
|
return type.getName();
|
||||||
|
}catch (MirroredTypeException e){
|
||||||
|
return e.getTypeMirror().toString();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private String getValue(ReadClass read){
|
||||||
|
try {
|
||||||
|
Class<?> type = read.value();
|
||||||
|
return type.getName();
|
||||||
|
}catch (MirroredTypeException e){
|
||||||
|
return e.getTypeMirror().toString();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**Information about read/write methods for a specific class type.*/
|
||||||
|
public static class ClassSerializer{
|
||||||
|
/**Fully qualified method name of the reader.*/
|
||||||
|
public final String readMethod;
|
||||||
|
/**Fully qualified method name of the writer.*/
|
||||||
|
public final String writeMethod;
|
||||||
|
/**Fully qualified class type name.*/
|
||||||
|
public final String classType;
|
||||||
|
|
||||||
|
public ClassSerializer(String readMethod, String writeMethod, String classType) {
|
||||||
|
this.readMethod = readMethod;
|
||||||
|
this.writeMethod = writeMethod;
|
||||||
|
this.classType = classType;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
47
annotations/src/io/anuke/annotations/MethodEntry.java
Normal file
@@ -0,0 +1,47 @@
|
|||||||
|
package io.anuke.annotations;
|
||||||
|
|
||||||
|
import io.anuke.annotations.Annotations.Loc;
|
||||||
|
import io.anuke.annotations.Annotations.Variant;
|
||||||
|
|
||||||
|
import javax.lang.model.element.ExecutableElement;
|
||||||
|
|
||||||
|
/**Class that repesents a remote method to be constructed and put into a class.*/
|
||||||
|
public class MethodEntry {
|
||||||
|
/**Simple target class name.*/
|
||||||
|
public final String className;
|
||||||
|
/**Fully qualified target method to call.*/
|
||||||
|
public final String targetMethod;
|
||||||
|
/**Whether this method can be called on a client/server.*/
|
||||||
|
public final Loc where;
|
||||||
|
/**Whether an additional 'one' and 'all' method variant is generated. At least one of these must be true.
|
||||||
|
* Only applicable to client (server-invoked) methods.*/
|
||||||
|
public final Variant target;
|
||||||
|
/**Whether this method is called locally as well as remotely.*/
|
||||||
|
public final Loc local;
|
||||||
|
/**Whether this method is unreliable and uses UDP.*/
|
||||||
|
public final boolean unreliable;
|
||||||
|
/**Whether to forward this method call to all other clients when a client invokes it. Server only.*/
|
||||||
|
public final boolean forward;
|
||||||
|
/**Unique method ID.*/
|
||||||
|
public final int id;
|
||||||
|
/**The element method associated with this entry.*/
|
||||||
|
public final ExecutableElement element;
|
||||||
|
|
||||||
|
public MethodEntry(String className, String targetMethod, Loc where, Variant target,
|
||||||
|
Loc local, boolean unreliable, boolean forward, int id, ExecutableElement element) {
|
||||||
|
this.className = className;
|
||||||
|
this.forward = forward;
|
||||||
|
this.targetMethod = targetMethod;
|
||||||
|
this.where = where;
|
||||||
|
this.target = target;
|
||||||
|
this.local = local;
|
||||||
|
this.id = id;
|
||||||
|
this.element = element;
|
||||||
|
this.unreliable = unreliable;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int hashCode() {
|
||||||
|
return targetMethod.hashCode();
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,157 @@
|
|||||||
|
package io.anuke.annotations;
|
||||||
|
|
||||||
|
import com.squareup.javapoet.FieldSpec;
|
||||||
|
import com.squareup.javapoet.JavaFile;
|
||||||
|
import com.squareup.javapoet.TypeSpec;
|
||||||
|
import io.anuke.annotations.Annotations.Loc;
|
||||||
|
import io.anuke.annotations.Annotations.Remote;
|
||||||
|
import io.anuke.annotations.IOFinder.ClassSerializer;
|
||||||
|
|
||||||
|
import javax.annotation.processing.*;
|
||||||
|
import javax.lang.model.SourceVersion;
|
||||||
|
import javax.lang.model.element.Element;
|
||||||
|
import javax.lang.model.element.ExecutableElement;
|
||||||
|
import javax.lang.model.element.Modifier;
|
||||||
|
import javax.lang.model.element.TypeElement;
|
||||||
|
import javax.tools.Diagnostic.Kind;
|
||||||
|
import java.util.*;
|
||||||
|
import java.util.stream.Collectors;
|
||||||
|
|
||||||
|
|
||||||
|
/**The annotation processor for generating remote method call code.*/
|
||||||
|
@SupportedSourceVersion(SourceVersion.RELEASE_8)
|
||||||
|
@SupportedAnnotationTypes({
|
||||||
|
"io.anuke.annotations.Annotations.Remote",
|
||||||
|
"io.anuke.annotations.Annotations.WriteClass",
|
||||||
|
"io.anuke.annotations.Annotations.ReadClass",
|
||||||
|
})
|
||||||
|
public class RemoteMethodAnnotationProcessor extends AbstractProcessor {
|
||||||
|
/**Maximum size of each event packet.*/
|
||||||
|
public static final int maxPacketSize = 4096;
|
||||||
|
/**Name of the base package to put all the generated classes.*/
|
||||||
|
private static final String packageName = "io.anuke.mindustry.gen";
|
||||||
|
|
||||||
|
/**Name of class that handles reading and invoking packets on the server.*/
|
||||||
|
private static final String readServerName = "RemoteReadServer";
|
||||||
|
/**Name of class that handles reading and invoking packets on the client.*/
|
||||||
|
private static final String readClientName = "RemoteReadClient";
|
||||||
|
|
||||||
|
/**Processing round number.*/
|
||||||
|
private int round;
|
||||||
|
|
||||||
|
//class serializers
|
||||||
|
private HashMap<String, ClassSerializer> serializers;
|
||||||
|
//all elements with the Remote annotation
|
||||||
|
private Set<? extends Element> elements;
|
||||||
|
//map of all classes to generate by name
|
||||||
|
private HashMap<String, ClassEntry> classMap;
|
||||||
|
//list of all method entries
|
||||||
|
private ArrayList<MethodEntry> methods;
|
||||||
|
//list of all method entries
|
||||||
|
private ArrayList<ClassEntry> classes;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public synchronized void init(ProcessingEnvironment processingEnv) {
|
||||||
|
super.init(processingEnv);
|
||||||
|
//put all relevant utils into utils class
|
||||||
|
Utils.typeUtils = processingEnv.getTypeUtils();
|
||||||
|
Utils.elementUtils = processingEnv.getElementUtils();
|
||||||
|
Utils.filer = processingEnv.getFiler();
|
||||||
|
Utils.messager = processingEnv.getMessager();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean process(Set<? extends TypeElement> annotations, RoundEnvironment roundEnv) {
|
||||||
|
if(round > 1) return false; //only process 2 rounds
|
||||||
|
|
||||||
|
round ++;
|
||||||
|
|
||||||
|
try {
|
||||||
|
|
||||||
|
//round 1: find all annotations, generate *writers*
|
||||||
|
if(round == 1) {
|
||||||
|
//get serializers
|
||||||
|
serializers = new IOFinder().findSerializers(roundEnv);
|
||||||
|
|
||||||
|
//last method ID used
|
||||||
|
int lastMethodID = 0;
|
||||||
|
//find all elements with the Remote annotation
|
||||||
|
elements = roundEnv.getElementsAnnotatedWith(Remote.class);
|
||||||
|
//map of all classes to generate by name
|
||||||
|
classMap = new HashMap<>();
|
||||||
|
//list of all method entries
|
||||||
|
methods = new ArrayList<>();
|
||||||
|
//list of all method entries
|
||||||
|
classes = new ArrayList<>();
|
||||||
|
|
||||||
|
List<Element> orderedElements = new ArrayList<>(elements);
|
||||||
|
orderedElements.sort(Comparator.comparing(Object::toString));
|
||||||
|
|
||||||
|
//create methods
|
||||||
|
for (Element element : orderedElements) {
|
||||||
|
Remote annotation = element.getAnnotation(Remote.class);
|
||||||
|
|
||||||
|
//check for static
|
||||||
|
if (!element.getModifiers().contains(Modifier.STATIC) || !element.getModifiers().contains(Modifier.PUBLIC)) {
|
||||||
|
Utils.messager.printMessage(Kind.ERROR, "All @Remote methods must be public and static: ", element);
|
||||||
|
}
|
||||||
|
|
||||||
|
//can't generate none methods
|
||||||
|
if (annotation.targets() == Loc.none) {
|
||||||
|
Utils.messager.printMessage(Kind.ERROR, "A @Remote method's targets() cannot be equal to 'none':", element);
|
||||||
|
}
|
||||||
|
|
||||||
|
//get and create class entry if needed
|
||||||
|
if (!classMap.containsKey(annotation.in())) {
|
||||||
|
ClassEntry clas = new ClassEntry(annotation.in());
|
||||||
|
classMap.put(annotation.in(), clas);
|
||||||
|
classes.add(clas);
|
||||||
|
|
||||||
|
Utils.messager.printMessage(Kind.NOTE, "Generating class '" + clas.name + "'.");
|
||||||
|
}
|
||||||
|
|
||||||
|
ClassEntry entry = classMap.get(annotation.in());
|
||||||
|
|
||||||
|
//create and add entry
|
||||||
|
MethodEntry method = new MethodEntry(entry.name, Utils.getMethodName(element), annotation.targets(), annotation.variants(),
|
||||||
|
annotation.called(), annotation.unreliable(), annotation.forward(), lastMethodID++, (ExecutableElement) element);
|
||||||
|
|
||||||
|
entry.methods.add(method);
|
||||||
|
methods.add(method);
|
||||||
|
}
|
||||||
|
|
||||||
|
//create read/write generators
|
||||||
|
RemoteWriteGenerator writegen = new RemoteWriteGenerator(serializers);
|
||||||
|
|
||||||
|
//generate the methods to invoke (write)
|
||||||
|
writegen.generateFor(classes, packageName);
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}else if(round == 2) { //round 2: generate all *readers*
|
||||||
|
RemoteReadGenerator readgen = new RemoteReadGenerator(serializers);
|
||||||
|
|
||||||
|
//generate server readers
|
||||||
|
readgen.generateFor(methods.stream().filter(method -> method.where.isClient).collect(Collectors.toList()), readServerName, packageName, true);
|
||||||
|
//generate client readers
|
||||||
|
readgen.generateFor(methods.stream().filter(method -> method.where.isServer).collect(Collectors.toList()), readClientName, packageName, false);
|
||||||
|
|
||||||
|
//create class for storing unique method hash
|
||||||
|
TypeSpec.Builder hashBuilder = TypeSpec.classBuilder("MethodHash").addModifiers(Modifier.PUBLIC);
|
||||||
|
hashBuilder.addField(FieldSpec.builder(int.class, "HASH", Modifier.STATIC, Modifier.PUBLIC, Modifier.FINAL)
|
||||||
|
.initializer("$1L", Objects.hash(methods)).build());
|
||||||
|
|
||||||
|
//build and write resulting hash class
|
||||||
|
TypeSpec spec = hashBuilder.build();
|
||||||
|
JavaFile.builder(packageName, spec).build().writeTo(Utils.filer);
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
}catch (Exception e){
|
||||||
|
e.printStackTrace();
|
||||||
|
throw new RuntimeException(e);
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
143
annotations/src/io/anuke/annotations/RemoteReadGenerator.java
Normal file
@@ -0,0 +1,143 @@
|
|||||||
|
package io.anuke.annotations;
|
||||||
|
|
||||||
|
import com.squareup.javapoet.*;
|
||||||
|
import io.anuke.annotations.IOFinder.ClassSerializer;
|
||||||
|
|
||||||
|
import javax.lang.model.element.Modifier;
|
||||||
|
import javax.lang.model.element.TypeElement;
|
||||||
|
import javax.lang.model.element.VariableElement;
|
||||||
|
import javax.tools.Diagnostic.Kind;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.lang.reflect.Constructor;
|
||||||
|
import java.lang.reflect.InvocationTargetException;
|
||||||
|
import java.nio.ByteBuffer;
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
/**Generates code for reading remote invoke packets on the client and server.*/
|
||||||
|
public class RemoteReadGenerator {
|
||||||
|
private final HashMap<String, ClassSerializer> serializers;
|
||||||
|
|
||||||
|
/**Creates a read generator that uses the supplied serializer setup.*/
|
||||||
|
public RemoteReadGenerator(HashMap<String, ClassSerializer> serializers) {
|
||||||
|
this.serializers = serializers;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**Generates a class for reading remote invoke packets.
|
||||||
|
* @param entries List of methods to use/
|
||||||
|
* @param className Simple target class name.
|
||||||
|
* @param packageName Full target package name.
|
||||||
|
* @param needsPlayer Whether this read method requires a reference to the player sender.*/
|
||||||
|
public void generateFor(List<MethodEntry> entries, String className, String packageName, boolean needsPlayer)
|
||||||
|
throws IllegalAccessException, InvocationTargetException, InstantiationException, NoSuchMethodException, IOException {
|
||||||
|
|
||||||
|
TypeSpec.Builder classBuilder = TypeSpec.classBuilder(className).addModifiers(Modifier.PUBLIC);
|
||||||
|
|
||||||
|
//create main method builder
|
||||||
|
MethodSpec.Builder readMethod = MethodSpec.methodBuilder("readPacket")
|
||||||
|
.addModifiers(Modifier.PUBLIC, Modifier.STATIC)
|
||||||
|
.addParameter(ByteBuffer.class, "buffer") //buffer to read form
|
||||||
|
.addParameter(int.class, "id") //ID of method type to read
|
||||||
|
.returns(void.class);
|
||||||
|
|
||||||
|
if(needsPlayer){
|
||||||
|
//since the player type isn't loaded yet, creating a type def is necessary
|
||||||
|
//this requires reflection since the TypeName constructor is private for some reason
|
||||||
|
Constructor<TypeName> cons = TypeName.class.getDeclaredConstructor(String.class);
|
||||||
|
cons.setAccessible(true);
|
||||||
|
|
||||||
|
TypeName playerType = cons.newInstance("io.anuke.mindustry.entities.Player");
|
||||||
|
//add player parameter
|
||||||
|
readMethod.addParameter(playerType, "player");
|
||||||
|
}
|
||||||
|
|
||||||
|
CodeBlock.Builder readBlock = CodeBlock.builder(); //start building block of code inside read method
|
||||||
|
boolean started = false; //whether an if() statement has been written yet
|
||||||
|
|
||||||
|
for(MethodEntry entry : entries){
|
||||||
|
//write if check for this entry ID
|
||||||
|
if(!started){
|
||||||
|
started = true;
|
||||||
|
readBlock.beginControlFlow("if(id == " + entry.id + ")");
|
||||||
|
}else{
|
||||||
|
readBlock.nextControlFlow("else if(id == " + entry.id + ")");
|
||||||
|
}
|
||||||
|
|
||||||
|
readBlock.beginControlFlow("try");
|
||||||
|
|
||||||
|
//concatenated list of variable names for method invocation
|
||||||
|
StringBuilder varResult = new StringBuilder();
|
||||||
|
|
||||||
|
//go through each parameter
|
||||||
|
for(int i = 0; i < entry.element.getParameters().size(); i ++){
|
||||||
|
VariableElement var = entry.element.getParameters().get(i);
|
||||||
|
|
||||||
|
if(!needsPlayer || i != 0) { //if client, skip first parameter since it's always of type player and doesn't need to be read
|
||||||
|
//full type name of parameter
|
||||||
|
String typeName = var.asType().toString();
|
||||||
|
//name of parameter
|
||||||
|
String varName = var.getSimpleName().toString();
|
||||||
|
//captialized version of type name for reading primitives
|
||||||
|
String capName = typeName.equals("byte") ? "" : Character.toUpperCase(typeName.charAt(0)) + typeName.substring(1);
|
||||||
|
|
||||||
|
//write primitives automatically
|
||||||
|
if (Utils.isPrimitive(typeName)) {
|
||||||
|
if (typeName.equals("boolean")) {
|
||||||
|
readBlock.addStatement("boolean " + varName + " = buffer.get() == 1");
|
||||||
|
} else {
|
||||||
|
readBlock.addStatement(typeName + " " + varName + " = buffer.get" + capName + "()");
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
//else, try and find a serializer
|
||||||
|
ClassSerializer ser = serializers.get(typeName);
|
||||||
|
|
||||||
|
if (ser == null) { //make sure a serializer exists!
|
||||||
|
Utils.messager.printMessage(Kind.ERROR, "No @ReadClass method to read class type: '" + typeName + "'", var);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
//add statement for reading it
|
||||||
|
readBlock.addStatement(typeName + " " + varName + " = " + ser.readMethod + "(buffer)");
|
||||||
|
}
|
||||||
|
|
||||||
|
//append variable name to string builder
|
||||||
|
varResult.append(var.getSimpleName());
|
||||||
|
if(i != entry.element.getParameters().size() - 1) varResult.append(", ");
|
||||||
|
}else{
|
||||||
|
varResult.append("player");
|
||||||
|
if(i != entry.element.getParameters().size() - 1) varResult.append(", ");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//execute the relevant method before the forward
|
||||||
|
//if it throws a ValidateException, the method won't be forwarded
|
||||||
|
readBlock.addStatement("$N." + entry.element.getSimpleName() + "(" + varResult.toString() + ")", ((TypeElement) entry.element.getEnclosingElement()).getQualifiedName().toString());
|
||||||
|
|
||||||
|
//call forwarded method, don't forward on the client reader
|
||||||
|
if(entry.forward && entry.where.isServer && needsPlayer){
|
||||||
|
//call forwarded method
|
||||||
|
readBlock.addStatement(packageName + "." + entry.className + "." + entry.element.getSimpleName() +
|
||||||
|
"__forward(player.clientid" + (varResult.length() == 0 ? "" : ", ") + varResult.toString() + ")");
|
||||||
|
}
|
||||||
|
|
||||||
|
readBlock.nextControlFlow("catch (java.lang.Exception e)");
|
||||||
|
readBlock.addStatement("throw new java.lang.RuntimeException(\"Failed to to read remote method '"+entry.element.getSimpleName() +"'!\", e)");
|
||||||
|
readBlock.endControlFlow();
|
||||||
|
}
|
||||||
|
|
||||||
|
//end control flow if necessary
|
||||||
|
if(started){
|
||||||
|
readBlock.nextControlFlow("else");
|
||||||
|
readBlock.addStatement("throw new $1N(\"Invalid read method ID: \" + id + \"\")", RuntimeException.class.getName()); //handle invalid method IDs
|
||||||
|
readBlock.endControlFlow();
|
||||||
|
}
|
||||||
|
|
||||||
|
//add block and method to class
|
||||||
|
readMethod.addCode(readBlock.build());
|
||||||
|
classBuilder.addMethod(readMethod.build());
|
||||||
|
|
||||||
|
//build and write resulting class
|
||||||
|
TypeSpec spec = classBuilder.build();
|
||||||
|
JavaFile.builder(packageName, spec).build().writeTo(Utils.filer);
|
||||||
|
}
|
||||||
|
}
|
||||||
224
annotations/src/io/anuke/annotations/RemoteWriteGenerator.java
Normal file
@@ -0,0 +1,224 @@
|
|||||||
|
package io.anuke.annotations;
|
||||||
|
|
||||||
|
import com.squareup.javapoet.*;
|
||||||
|
import io.anuke.annotations.Annotations.Loc;
|
||||||
|
import io.anuke.annotations.IOFinder.ClassSerializer;
|
||||||
|
|
||||||
|
import javax.lang.model.element.ExecutableElement;
|
||||||
|
import javax.lang.model.element.Modifier;
|
||||||
|
import javax.lang.model.element.TypeElement;
|
||||||
|
import javax.lang.model.element.VariableElement;
|
||||||
|
import javax.tools.Diagnostic.Kind;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.nio.ByteBuffer;
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
/**Generates code for writing remote invoke packets on the client and server.*/
|
||||||
|
public class RemoteWriteGenerator {
|
||||||
|
private final HashMap<String, ClassSerializer> serializers;
|
||||||
|
|
||||||
|
/**Creates a write generator that uses the supplied serializer setup.*/
|
||||||
|
public RemoteWriteGenerator(HashMap<String, ClassSerializer> serializers) {
|
||||||
|
this.serializers = serializers;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**Generates all classes in this list.*/
|
||||||
|
public void generateFor(List<ClassEntry> entries, String packageName) throws IOException {
|
||||||
|
|
||||||
|
for(ClassEntry entry : entries){
|
||||||
|
//create builder
|
||||||
|
TypeSpec.Builder classBuilder = TypeSpec.classBuilder(entry.name).addModifiers(Modifier.PUBLIC);
|
||||||
|
|
||||||
|
//add temporary write buffer
|
||||||
|
classBuilder.addField(FieldSpec.builder(ByteBuffer.class, "TEMP_BUFFER", Modifier.STATIC, Modifier.PRIVATE, Modifier.FINAL)
|
||||||
|
.initializer("ByteBuffer.allocate($1L)", RemoteMethodAnnotationProcessor.maxPacketSize).build());
|
||||||
|
|
||||||
|
//go through each method entry in this class
|
||||||
|
for(MethodEntry methodEntry : entry.methods){
|
||||||
|
//write the 'send event to all players' variant: always happens for clients, but only happens if 'all' is enabled on the server method
|
||||||
|
if(methodEntry.where.isClient || methodEntry.target.isAll){
|
||||||
|
writeMethodVariant(classBuilder, methodEntry, true, false);
|
||||||
|
}
|
||||||
|
|
||||||
|
//write the 'send event to one player' variant, which is only applicable on the server
|
||||||
|
if(methodEntry.where.isServer && methodEntry.target.isOne){
|
||||||
|
writeMethodVariant(classBuilder, methodEntry, false, false);
|
||||||
|
}
|
||||||
|
|
||||||
|
//write the forwarded method version
|
||||||
|
if(methodEntry.where.isServer && methodEntry.forward){
|
||||||
|
writeMethodVariant(classBuilder, methodEntry, true, true);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//build and write resulting class
|
||||||
|
TypeSpec spec = classBuilder.build();
|
||||||
|
JavaFile.builder(packageName, spec).build().writeTo(Utils.filer);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**Creates a specific variant for a method entry.*/
|
||||||
|
private void writeMethodVariant(TypeSpec.Builder classBuilder, MethodEntry methodEntry, boolean toAll, boolean forwarded){
|
||||||
|
ExecutableElement elem = methodEntry.element;
|
||||||
|
|
||||||
|
//create builder
|
||||||
|
MethodSpec.Builder method = MethodSpec.methodBuilder(elem.getSimpleName().toString() + (forwarded ? "__forward" : "")) //add except suffix when forwarding
|
||||||
|
.addModifiers(Modifier.STATIC, Modifier.SYNCHRONIZED)
|
||||||
|
.returns(void.class);
|
||||||
|
|
||||||
|
//forwarded methods aren't intended for use, and are not public
|
||||||
|
if(!forwarded){
|
||||||
|
method.addModifiers(Modifier.PUBLIC);
|
||||||
|
}
|
||||||
|
|
||||||
|
//validate client methods to make sure
|
||||||
|
if(methodEntry.where.isClient){
|
||||||
|
if(elem.getParameters().isEmpty()){
|
||||||
|
Utils.messager.printMessage(Kind.ERROR, "Client invoke methods must have a first parameter of type Player.", elem);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if(!elem.getParameters().get(0).asType().toString().equals("io.anuke.mindustry.entities.Player")){
|
||||||
|
Utils.messager.printMessage(Kind.ERROR, "Client invoke methods should have a first parameter of type Player.", elem);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//if toAll is false, it's a 'send to one player' variant, so add the player as a parameter
|
||||||
|
if(!toAll){
|
||||||
|
method.addParameter(int.class, "playerClientID");
|
||||||
|
}
|
||||||
|
|
||||||
|
//add sender to ignore
|
||||||
|
if(forwarded){
|
||||||
|
method.addParameter(int.class, "exceptSenderID");
|
||||||
|
}
|
||||||
|
|
||||||
|
//call local method if applicable, shouldn't happen when forwarding method as that already happens by default
|
||||||
|
if(!forwarded && methodEntry.local != Loc.none){
|
||||||
|
//add in local checks
|
||||||
|
if(methodEntry.local != Loc.both){
|
||||||
|
method.beginControlFlow("if("+getCheckString(methodEntry.local) + " || !io.anuke.mindustry.net.Net.active())");
|
||||||
|
}
|
||||||
|
|
||||||
|
//concatenate parameters
|
||||||
|
int index = 0;
|
||||||
|
StringBuilder results = new StringBuilder();
|
||||||
|
for(VariableElement var : elem.getParameters()){
|
||||||
|
//special case: calling local-only methods uses the local player
|
||||||
|
if(index == 0 && methodEntry.where == Loc.client){
|
||||||
|
results.append("io.anuke.mindustry.Vars.players[0]");
|
||||||
|
}else {
|
||||||
|
results.append(var.getSimpleName());
|
||||||
|
}
|
||||||
|
if(index != elem.getParameters().size() - 1) results.append(", ");
|
||||||
|
index ++;
|
||||||
|
}
|
||||||
|
|
||||||
|
//add the statement to call it
|
||||||
|
method.addStatement("$N." + elem.getSimpleName() + "(" + results.toString() + ")",
|
||||||
|
((TypeElement)elem.getEnclosingElement()).getQualifiedName().toString());
|
||||||
|
|
||||||
|
if(methodEntry.local != Loc.both){
|
||||||
|
method.endControlFlow();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//start control flow to check if it's actually client/server so no netcode is called
|
||||||
|
method.beginControlFlow("if("+getCheckString(methodEntry.where)+")");
|
||||||
|
|
||||||
|
//add statement to create packet from pool
|
||||||
|
method.addStatement("$1N packet = $2N.obtain($1N.class)", "io.anuke.mindustry.net.Packets.InvokePacket", "com.badlogic.gdx.utils.Pools");
|
||||||
|
//assign buffer
|
||||||
|
method.addStatement("packet.writeBuffer = TEMP_BUFFER");
|
||||||
|
//assign method ID
|
||||||
|
method.addStatement("packet.type = (byte)" + methodEntry.id);
|
||||||
|
//rewind buffer
|
||||||
|
method.addStatement("TEMP_BUFFER.position(0)");
|
||||||
|
|
||||||
|
for(int i = 0; i < elem.getParameters().size(); i ++){
|
||||||
|
//first argument is skipped as it is always the player caller
|
||||||
|
if((!methodEntry.where.isServer/* || methodEntry.mode == Loc.both*/) && i == 0){
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
VariableElement var = elem.getParameters().get(i);
|
||||||
|
|
||||||
|
//add parameter to method
|
||||||
|
method.addParameter(TypeName.get(var.asType()), var.getSimpleName().toString());
|
||||||
|
|
||||||
|
//name of parameter
|
||||||
|
String varName = var.getSimpleName().toString();
|
||||||
|
//name of parameter type
|
||||||
|
String typeName = var.asType().toString();
|
||||||
|
//captialized version of type name for writing primitives
|
||||||
|
String capName = typeName.equals("byte") ? "" : Character.toUpperCase(typeName.charAt(0)) + typeName.substring(1);
|
||||||
|
//special case: method can be called from anywhere to anywhere
|
||||||
|
//thus, only write the player when the SERVER is writing data, since the client is the only one who reads it
|
||||||
|
boolean writePlayerSkipCheck = methodEntry.where == Loc.both && i == 0;
|
||||||
|
|
||||||
|
if(writePlayerSkipCheck){ //write begin check
|
||||||
|
method.beginControlFlow("if(io.anuke.mindustry.net.Net.server())");
|
||||||
|
}
|
||||||
|
|
||||||
|
if(Utils.isPrimitive(typeName)) { //check if it's a primitive, and if so write it
|
||||||
|
if(typeName.equals("boolean")){ //booleans are special
|
||||||
|
method.addStatement("TEMP_BUFFER.put(" + varName + " ? (byte)1 : 0)");
|
||||||
|
}else{
|
||||||
|
method.addStatement("TEMP_BUFFER.put" +
|
||||||
|
capName + "(" + varName + ")");
|
||||||
|
}
|
||||||
|
}else{
|
||||||
|
//else, try and find a serializer
|
||||||
|
ClassSerializer ser = serializers.get(typeName);
|
||||||
|
|
||||||
|
if(ser == null){ //make sure a serializer exists!
|
||||||
|
Utils.messager.printMessage(Kind.ERROR, "No @WriteClass method to write class type: '" + typeName + "'", var);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
//add statement for writing it
|
||||||
|
method.addStatement(ser.writeMethod + "(TEMP_BUFFER, " + varName +")");
|
||||||
|
}
|
||||||
|
|
||||||
|
if(writePlayerSkipCheck){ //write end check
|
||||||
|
method.endControlFlow();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//assign packet length
|
||||||
|
method.addStatement("packet.writeLength = TEMP_BUFFER.position()");
|
||||||
|
|
||||||
|
String sendString;
|
||||||
|
|
||||||
|
if(forwarded){ //forward packet
|
||||||
|
if(!methodEntry.local.isClient){ //if the client doesn't get it called locally, forward it back after validation
|
||||||
|
sendString = "send(";
|
||||||
|
}else {
|
||||||
|
sendString = "sendExcept(exceptSenderID, ";
|
||||||
|
}
|
||||||
|
}else if(toAll){ //send to all players / to server
|
||||||
|
sendString = "send(";
|
||||||
|
}else{ //send to specific client from server
|
||||||
|
sendString = "sendTo(playerClientID, ";
|
||||||
|
}
|
||||||
|
|
||||||
|
//send the actual packet
|
||||||
|
method.addStatement("io.anuke.mindustry.net.Net." + sendString + "packet, "+
|
||||||
|
(methodEntry.unreliable ? "io.anuke.mindustry.net.Net.SendMode.udp" : "io.anuke.mindustry.net.Net.SendMode.tcp")+")");
|
||||||
|
|
||||||
|
|
||||||
|
//end check for server/client
|
||||||
|
method.endControlFlow();
|
||||||
|
|
||||||
|
//add method to class, finally
|
||||||
|
classBuilder.addMethod(method.build());
|
||||||
|
}
|
||||||
|
|
||||||
|
private String getCheckString(Loc loc){
|
||||||
|
return loc.isClient && loc.isServer ? "io.anuke.mindustry.net.Net.server() || io.anuke.mindustry.net.Net.client()" :
|
||||||
|
loc.isClient ? "io.anuke.mindustry.net.Net.client()" :
|
||||||
|
loc.isServer ? "io.anuke.mindustry.net.Net.server()" : "false";
|
||||||
|
}
|
||||||
|
}
|
||||||
24
annotations/src/io/anuke/annotations/Utils.java
Normal file
@@ -0,0 +1,24 @@
|
|||||||
|
package io.anuke.annotations;
|
||||||
|
|
||||||
|
import javax.annotation.processing.Filer;
|
||||||
|
import javax.annotation.processing.Messager;
|
||||||
|
import javax.lang.model.element.Element;
|
||||||
|
import javax.lang.model.element.TypeElement;
|
||||||
|
import javax.lang.model.util.Elements;
|
||||||
|
import javax.lang.model.util.Types;
|
||||||
|
|
||||||
|
public class Utils {
|
||||||
|
public static Types typeUtils;
|
||||||
|
public static Elements elementUtils;
|
||||||
|
public static Filer filer;
|
||||||
|
public static Messager messager;
|
||||||
|
|
||||||
|
public static String getMethodName(Element element){
|
||||||
|
return ((TypeElement)element.getEnclosingElement()).getQualifiedName().toString() + "." + element.getSimpleName();
|
||||||
|
}
|
||||||
|
|
||||||
|
public static boolean isPrimitive(String type){
|
||||||
|
return type.equals("boolean") || type.equals("byte") || type.equals("short") || type.equals("int")
|
||||||
|
|| type.equals("long") || type.equals("float") || type.equals("double") || type.equals("char");
|
||||||
|
}
|
||||||
|
}
|
||||||
95
build.gradle
@@ -7,8 +7,9 @@ buildscript {
|
|||||||
jcenter()
|
jcenter()
|
||||||
}
|
}
|
||||||
dependencies {
|
dependencies {
|
||||||
|
classpath 'com.mobidevelop.robovm:robovm-gradle-plugin:2.3.0'
|
||||||
classpath 'de.richsource.gradle.plugins:gwt-gradle-plugin:0.6'
|
classpath 'de.richsource.gradle.plugins:gwt-gradle-plugin:0.6'
|
||||||
classpath 'com.android.tools.build:gradle:3.0.1'
|
classpath 'com.android.tools.build:gradle:3.1.0'
|
||||||
classpath "com.badlogicgames.gdx:gdx-tools:1.9.8"
|
classpath "com.badlogicgames.gdx:gdx-tools:1.9.8"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -20,12 +21,13 @@ allprojects {
|
|||||||
version = 'release'
|
version = 'release'
|
||||||
|
|
||||||
ext {
|
ext {
|
||||||
versionNumber = '3.4'
|
versionNumber = '4.0'
|
||||||
versionType = 'release'
|
versionType = 'alpha'
|
||||||
appName = 'Mindustry'
|
appName = 'Mindustry'
|
||||||
gdxVersion = '1.9.8'
|
gdxVersion = '1.9.8'
|
||||||
|
roboVMVersion = '2.3.0'
|
||||||
aiVersion = '1.8.1'
|
aiVersion = '1.8.1'
|
||||||
uCoreVersion = '21f04f8'
|
uCoreVersion = 'bc1c2e0848'
|
||||||
|
|
||||||
getVersionString = {
|
getVersionString = {
|
||||||
String buildVersion = getBuildVersion()
|
String buildVersion = getBuildVersion()
|
||||||
@@ -40,6 +42,29 @@ allprojects {
|
|||||||
getPackage = {
|
getPackage = {
|
||||||
return project.ext.mainClassName.substring(0, project.ext.mainClassName.indexOf("desktop") - 1)
|
return project.ext.mainClassName.substring(0, project.ext.mainClassName.indexOf("desktop") - 1)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
writeVersion = {
|
||||||
|
def pfile = new File('core/assets/version.properties')
|
||||||
|
def props = new Properties()
|
||||||
|
|
||||||
|
try{
|
||||||
|
pfile.createNewFile()
|
||||||
|
}catch (Exception e){}
|
||||||
|
|
||||||
|
if(pfile.exists()) {
|
||||||
|
|
||||||
|
props.load(new FileInputStream(pfile))
|
||||||
|
|
||||||
|
String code = getBuildVersion()
|
||||||
|
|
||||||
|
props["name"] = appName
|
||||||
|
props["version"] = versionType
|
||||||
|
props["code"] = versionNumber
|
||||||
|
props["build"] = code
|
||||||
|
|
||||||
|
props.store(pfile.newWriter(), "Autogenerated file. Do not modify.")
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
repositories {
|
repositories {
|
||||||
@@ -60,7 +85,8 @@ project(":desktop") {
|
|||||||
compile "com.badlogicgames.gdx:gdx-backend-lwjgl3:$gdxVersion"
|
compile "com.badlogicgames.gdx:gdx-backend-lwjgl3:$gdxVersion"
|
||||||
compile "com.badlogicgames.gdx:gdx-platform:$gdxVersion:natives-desktop"
|
compile "com.badlogicgames.gdx:gdx-platform:$gdxVersion:natives-desktop"
|
||||||
compile "com.badlogicgames.gdx:gdx-controllers-lwjgl3:$gdxVersion"
|
compile "com.badlogicgames.gdx:gdx-controllers-lwjgl3:$gdxVersion"
|
||||||
compile 'com.github.MinnDevelopment:java-discord-rpc:v1.3.2'
|
compile 'com.github.MinnDevelopment:java-discord-rpc:v2.0.0'
|
||||||
|
compile 'com.yuvimasory:orange-extensions:1.3.0'
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -71,6 +97,8 @@ project(":html") {
|
|||||||
|
|
||||||
dependencies {
|
dependencies {
|
||||||
compile project(":core")
|
compile project(":core")
|
||||||
|
compileOnly project(":annotations")
|
||||||
|
|
||||||
compile "com.badlogicgames.gdx:gdx-backend-gwt:$gdxVersion"
|
compile "com.badlogicgames.gdx:gdx-backend-gwt:$gdxVersion"
|
||||||
compile "com.badlogicgames.gdx:gdx:$gdxVersion:sources"
|
compile "com.badlogicgames.gdx:gdx:$gdxVersion:sources"
|
||||||
compile "com.badlogicgames.gdx:gdx-backend-gwt:$gdxVersion:sources"
|
compile "com.badlogicgames.gdx:gdx-backend-gwt:$gdxVersion:sources"
|
||||||
@@ -82,24 +110,25 @@ project(":html") {
|
|||||||
compile "com.sksamuel.gwt:gwt-websockets:1.0.4"
|
compile "com.sksamuel.gwt:gwt-websockets:1.0.4"
|
||||||
compile "com.sksamuel.gwt:gwt-websockets:1.0.4:sources"
|
compile "com.sksamuel.gwt:gwt-websockets:1.0.4:sources"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
compileJava.options.compilerArgs = [
|
||||||
|
"-processor", "io.anuke.annotations.RemoteMethodAnnotationProcessor"
|
||||||
|
]
|
||||||
}
|
}
|
||||||
|
|
||||||
project(":android") {
|
project(":ios") {
|
||||||
apply plugin: "android"
|
apply plugin: "java"
|
||||||
|
apply plugin: "robovm"
|
||||||
|
|
||||||
configurations { natives }
|
|
||||||
|
|
||||||
dependencies {
|
dependencies {
|
||||||
implementation project(":core")
|
compile project(":core")
|
||||||
implementation project(":kryonet")
|
implementation project(":kryonet")
|
||||||
implementation "com.badlogicgames.gdx:gdx-backend-android:$gdxVersion"
|
|
||||||
natives "com.badlogicgames.gdx:gdx-platform:$gdxVersion:natives-armeabi"
|
compile "com.mobidevelop.robovm:robovm-rt:$roboVMVersion"
|
||||||
natives "com.badlogicgames.gdx:gdx-platform:$gdxVersion:natives-armeabi-v7a"
|
compile "com.mobidevelop.robovm:robovm-cocoatouch:$roboVMVersion"
|
||||||
natives "com.badlogicgames.gdx:gdx-platform:$gdxVersion:natives-arm64-v8a"
|
compile "com.badlogicgames.gdx:gdx-backend-robovm:$gdxVersion"
|
||||||
natives "com.badlogicgames.gdx:gdx-platform:$gdxVersion:natives-x86"
|
compile "com.badlogicgames.gdx:gdx-platform:$gdxVersion:natives-ios"
|
||||||
natives "com.badlogicgames.gdx:gdx-platform:$gdxVersion:natives-x86_64"
|
|
||||||
implementation "com.badlogicgames.gdx:gdx-ai:$aiVersion"
|
|
||||||
implementation "com.badlogicgames.gdx:gdx-controllers-android:$gdxVersion"
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -107,7 +136,9 @@ project(":core") {
|
|||||||
apply plugin: "java"
|
apply plugin: "java"
|
||||||
|
|
||||||
dependencies {
|
dependencies {
|
||||||
boolean comp = System.properties["release"] == null || System.properties["release"].equals("false")
|
compileOnly project(":annotations")
|
||||||
|
|
||||||
|
boolean comp = System.properties["release"] == null || System.properties["release"] == "false"
|
||||||
|
|
||||||
if(!comp){
|
if(!comp){
|
||||||
println("NOTICE: Compiling release build.")
|
println("NOTICE: Compiling release build.")
|
||||||
@@ -115,25 +146,31 @@ project(":core") {
|
|||||||
println("Compiling DEBUG build.")
|
println("Compiling DEBUG build.")
|
||||||
}
|
}
|
||||||
|
|
||||||
if(new File('../uCore').exists() && comp){
|
if(new File(projectDir.parent, '../uCore').exists() && comp){
|
||||||
compile project(":uCore")
|
compile project(":uCore")
|
||||||
}else{
|
}else{
|
||||||
compile "com.github.anuken:ucore:$uCoreVersion"
|
compile "com.github.anuken:ucore:$uCoreVersion"
|
||||||
}
|
}
|
||||||
|
|
||||||
if(new File('../GDXGifRecorder').exists() && comp) {
|
if(new File(projectDir.parent, '../GDXGifRecorder').exists() && comp) {
|
||||||
compile project(":GDXGifRecorder")
|
compile project(":GDXGifRecorder")
|
||||||
}
|
}
|
||||||
|
|
||||||
compile "com.badlogicgames.gdx:gdx:$gdxVersion"
|
compile "com.badlogicgames.gdx:gdx:$gdxVersion"
|
||||||
compile "com.badlogicgames.gdx:gdx-ai:$aiVersion"
|
|
||||||
compile "com.badlogicgames.gdx:gdx-controllers:$gdxVersion"
|
compile "com.badlogicgames.gdx:gdx-controllers:$gdxVersion"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
compileJava.options.compilerArgs = [
|
||||||
|
"-processor", "io.anuke.annotations.RemoteMethodAnnotationProcessor"
|
||||||
|
]
|
||||||
}
|
}
|
||||||
|
|
||||||
project(":server") {
|
project(":server") {
|
||||||
apply plugin: "java"
|
apply plugin: "java"
|
||||||
|
|
||||||
dependencies {
|
dependencies {
|
||||||
|
compileOnly project(":annotations")
|
||||||
|
|
||||||
compile project(":core")
|
compile project(":core")
|
||||||
compile project(":kryonet")
|
compile project(":kryonet")
|
||||||
compile "com.badlogicgames.gdx:gdx-backend-headless:$gdxVersion"
|
compile "com.badlogicgames.gdx:gdx-backend-headless:$gdxVersion"
|
||||||
@@ -141,6 +178,22 @@ project(":server") {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
project(":packer") {
|
||||||
|
apply plugin: "java"
|
||||||
|
|
||||||
|
dependencies {
|
||||||
|
compile project(":core")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
project(":annotations") {
|
||||||
|
apply plugin: "java"
|
||||||
|
|
||||||
|
dependencies {
|
||||||
|
compile 'com.squareup:javapoet:1.11.0'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
project(":kryonet") {
|
project(":kryonet") {
|
||||||
apply plugin: "java"
|
apply plugin: "java"
|
||||||
|
|
||||||
|
|||||||
@@ -1,12 +0,0 @@
|
|||||||
#!/usr/bin/env bash
|
|
||||||
|
|
||||||
OLD_TRAVIS_BUILD_NUMBER=`expr $TRAVIS_BUILD_NUMBER - 7`
|
|
||||||
OLD_DESKFILE=$OLD_TRAVIS_BUILD_BUMBER"-desktop-bleeding-edge.jar"
|
|
||||||
OLD_FILE1="Bleeding-Edge-Build-"$OLD_TRAVIS_BUILD_NUMBER".md"
|
|
||||||
|
|
||||||
if [ -e $OLD_FILE1 ]; then
|
|
||||||
rm -f $OLD_FILE1
|
|
||||||
rm -f $OLD_DESKFILE
|
|
||||||
git add $OLD_FILE1
|
|
||||||
git add $OLD_DESKFILE
|
|
||||||
fi
|
|
||||||
|
Before Width: | Height: | Size: 50 KiB After Width: | Height: | Size: 59 KiB |
|
Before Width: | Height: | Size: 207 B |
|
Before Width: | Height: | Size: 208 B |
|
Before Width: | Height: | Size: 206 B |
|
Before Width: | Height: | Size: 315 B |
|
Before Width: | Height: | Size: 378 B |
|
Before Width: | Height: | Size: 182 B |
|
Before Width: | Height: | Size: 281 B |
|
Before Width: | Height: | Size: 514 B |
|
Before Width: | Height: | Size: 258 B |
|
Before Width: | Height: | Size: 237 B |
|
Before Width: | Height: | Size: 256 B |
|
Before Width: | Height: | Size: 253 B |
|
Before Width: | Height: | Size: 300 B |
|
Before Width: | Height: | Size: 274 B |
|
Before Width: | Height: | Size: 254 B |
|
Before Width: | Height: | Size: 290 B |
|
Before Width: | Height: | Size: 222 B |
|
Before Width: | Height: | Size: 195 B |
|
Before Width: | Height: | Size: 195 B |
|
Before Width: | Height: | Size: 246 B |
|
Before Width: | Height: | Size: 266 B |
|
After Width: | Height: | Size: 160 B |
|
After Width: | Height: | Size: 177 B |
|
After Width: | Height: | Size: 178 B |
BIN
core/assets-raw/sprites/blocks/distribution/bridge-conveyor.png
Normal file
|
After Width: | Height: | Size: 227 B |
BIN
core/assets-raw/sprites/blocks/distribution/conveyor.png
Normal file
|
After Width: | Height: | Size: 197 B |
|
Before Width: | Height: | Size: 257 B After Width: | Height: | Size: 257 B |
|
After Width: | Height: | Size: 682 B |
BIN
core/assets-raw/sprites/blocks/distribution/mass-driver.png
Normal file
|
After Width: | Height: | Size: 580 B |
BIN
core/assets-raw/sprites/blocks/distribution/multiplexer.png
Normal file
|
After Width: | Height: | Size: 296 B |
BIN
core/assets-raw/sprites/blocks/distribution/overflow-gate.png
Normal file
|
After Width: | Height: | Size: 233 B |
|
After Width: | Height: | Size: 179 B |
|
After Width: | Height: | Size: 189 B |
|
After Width: | Height: | Size: 172 B |
BIN
core/assets-raw/sprites/blocks/distribution/phase-conveyor.png
Normal file
|
After Width: | Height: | Size: 301 B |
|
Before Width: | Height: | Size: 226 B After Width: | Height: | Size: 226 B |
|
Before Width: | Height: | Size: 248 B After Width: | Height: | Size: 248 B |
BIN
core/assets-raw/sprites/blocks/distribution/splitter.png
Normal file
|
After Width: | Height: | Size: 234 B |
|
After Width: | Height: | Size: 210 B |
BIN
core/assets-raw/sprites/blocks/distribution/warp-gate-top.png
Normal file
|
After Width: | Height: | Size: 346 B |
BIN
core/assets-raw/sprites/blocks/distribution/warp-gate.png
Normal file
|
After Width: | Height: | Size: 492 B |
|
Before Width: | Height: | Size: 253 B |
|
Before Width: | Height: | Size: 428 B |
|
Before Width: | Height: | Size: 397 B |
|
Before Width: | Height: | Size: 254 B |
|
Before Width: | Height: | Size: 224 B |
|
Before Width: | Height: | Size: 308 B |
BIN
core/assets-raw/sprites/blocks/drills/blast-drill-rim.png
Normal file
|
After Width: | Height: | Size: 219 B |
BIN
core/assets-raw/sprites/blocks/drills/blast-drill-rotator.png
Normal file
|
After Width: | Height: | Size: 298 B |
BIN
core/assets-raw/sprites/blocks/drills/blast-drill-top.png
Normal file
|
After Width: | Height: | Size: 289 B |
BIN
core/assets-raw/sprites/blocks/drills/blast-drill.png
Normal file
|
After Width: | Height: | Size: 574 B |
BIN
core/assets-raw/sprites/blocks/drills/carbide-drill-rotator.png
Normal file
|
After Width: | Height: | Size: 174 B |
BIN
core/assets-raw/sprites/blocks/drills/carbide-drill-top.png
Normal file
|
After Width: | Height: | Size: 196 B |
BIN
core/assets-raw/sprites/blocks/drills/carbide-drill.png
Normal file
|
After Width: | Height: | Size: 233 B |
BIN
core/assets-raw/sprites/blocks/drills/laser-drill-rotator.png
Normal file
|
After Width: | Height: | Size: 261 B |
BIN
core/assets-raw/sprites/blocks/drills/laser-drill-top.png
Normal file
|
After Width: | Height: | Size: 286 B |
BIN
core/assets-raw/sprites/blocks/drills/laser-drill.png
Normal file
|
After Width: | Height: | Size: 327 B |
BIN
core/assets-raw/sprites/blocks/drills/oil-extractor-liquid.png
Normal file
|
After Width: | Height: | Size: 239 B |
BIN
core/assets-raw/sprites/blocks/drills/oil-extractor-rotator.png
Normal file
|
After Width: | Height: | Size: 341 B |
BIN
core/assets-raw/sprites/blocks/drills/oil-extractor-top.png
Normal file
|
After Width: | Height: | Size: 379 B |
BIN
core/assets-raw/sprites/blocks/drills/oil-extractor.png
Normal file
|
After Width: | Height: | Size: 473 B |
BIN
core/assets-raw/sprites/blocks/drills/plasma-drill-rim.png
Normal file
|
After Width: | Height: | Size: 461 B |
BIN
core/assets-raw/sprites/blocks/drills/plasma-drill-rotator.png
Normal file
|
After Width: | Height: | Size: 354 B |
BIN
core/assets-raw/sprites/blocks/drills/plasma-drill-top.png
Normal file
|
After Width: | Height: | Size: 296 B |
BIN
core/assets-raw/sprites/blocks/drills/plasma-drill.png
Normal file
|
After Width: | Height: | Size: 678 B |
BIN
core/assets-raw/sprites/blocks/drills/tungsten-drill-rotator.png
Normal file
|
After Width: | Height: | Size: 174 B |
BIN
core/assets-raw/sprites/blocks/drills/tungsten-drill-top.png
Normal file
|
After Width: | Height: | Size: 188 B |
BIN
core/assets-raw/sprites/blocks/drills/tungsten-drill.png
Normal file
|
After Width: | Height: | Size: 188 B |
BIN
core/assets-raw/sprites/blocks/drills/water-extractor-liquid.png
Normal file
|
After Width: | Height: | Size: 192 B |