Compare commits
965 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
f41edc2ea4 | ||
|
|
30ab4442cf | ||
|
|
90e556be6e | ||
|
|
e9ff6a93fb | ||
|
|
5879ad7ec4 | ||
|
|
731d186b01 | ||
|
|
47d98500cf | ||
|
|
2561fd92f9 | ||
|
|
9129cdef0d | ||
|
|
613ee337aa | ||
|
|
bd6595e40a | ||
|
|
d827e51ba4 | ||
|
|
77fd5c5a3f | ||
|
|
a2e5b73107 | ||
|
|
4e42f61faa | ||
|
|
3a04d14872 | ||
|
|
e18e912d11 | ||
|
|
15478751db | ||
|
|
d715a4299d | ||
|
|
09d450aef4 | ||
|
|
e5fd79e4c1 | ||
|
|
86fc6689fb | ||
|
|
d6ba15df55 | ||
|
|
86e9447dd0 | ||
|
|
29e919caad | ||
|
|
88e5149a9f | ||
|
|
33c0b24731 | ||
|
|
d4c443607e | ||
|
|
6422d7faca | ||
|
|
1468612a78 | ||
|
|
bbb0951525 | ||
|
|
47083b4a86 | ||
|
|
5a51e2748b | ||
|
|
93dbf25938 | ||
|
|
898c7f58c9 | ||
|
|
b7d503fd39 | ||
|
|
95da509b83 | ||
|
|
34c8f511d9 | ||
|
|
3f0d00c83c | ||
|
|
79433a5c83 | ||
|
|
abc1d2cf64 | ||
|
|
f7c4d5dfae | ||
|
|
2305b7072b | ||
|
|
643eff4a90 | ||
|
|
e2e06ca027 | ||
|
|
a7df290d58 | ||
|
|
123524d126 | ||
|
|
28b98b055a | ||
|
|
76743cd414 | ||
|
|
9fcc5e99d3 | ||
|
|
4a55d9ec45 | ||
|
|
7d690dc75d | ||
|
|
2ece6d5375 | ||
|
|
421929c147 | ||
|
|
755156c19a | ||
|
|
43deec1c9e | ||
|
|
e20517430e | ||
|
|
16940eca53 | ||
|
|
37d000ee44 | ||
|
|
0db724d40c | ||
|
|
1da7ae6de3 | ||
|
|
1b3478e0cb | ||
|
|
11bf45466d | ||
|
|
b36caeddd2 | ||
|
|
f606ebdd72 | ||
|
|
0f9ad1e028 | ||
|
|
fcdbfb07c3 | ||
|
|
06637c6c0f | ||
|
|
c28fd5d5c3 | ||
|
|
d450841001 | ||
|
|
5191892950 | ||
|
|
bfa531f1f4 | ||
|
|
6aa3f00fa3 | ||
|
|
f3d7d73acc | ||
|
|
880ea7c998 | ||
|
|
bc9bf12e11 | ||
|
|
02f901ddf1 | ||
|
|
cbb3f99dfa | ||
|
|
7bf97ef618 | ||
|
|
1d92cc8f3e | ||
|
|
9faf63d5d3 | ||
|
|
7dcbc32540 | ||
|
|
387555f3c0 | ||
|
|
10ebd00cde | ||
|
|
fda93e9dd5 | ||
|
|
23759b1aaa | ||
|
|
165bb820bc | ||
|
|
83e9c09690 | ||
|
|
b7476a4ea1 | ||
|
|
38b12759c6 | ||
|
|
c6d30f8f9f | ||
|
|
4f40510248 | ||
|
|
c733a41d10 | ||
|
|
81c31f4de3 | ||
|
|
ac1c118ae3 | ||
|
|
7d2dfc1cf2 | ||
|
|
0c2cf50362 | ||
|
|
a485be9c1e | ||
|
|
e2ccec9ba3 | ||
|
|
7d8fb77481 | ||
|
|
4c5ccadfc7 | ||
|
|
69d0e2e3cd | ||
|
|
733c24e6cd | ||
|
|
4155a1f586 | ||
|
|
34ce565bb8 | ||
|
|
d805c7c66d | ||
|
|
3291216bbb | ||
|
|
eb6754fe8a | ||
|
|
b4c9c0d51b | ||
|
|
8058433031 | ||
|
|
c877a2814e | ||
|
|
2865212937 | ||
|
|
ed7da32952 | ||
|
|
a0b53f2e0c | ||
|
|
714bdcbdbb | ||
|
|
a2fd884543 | ||
|
|
93c6c121f6 | ||
|
|
dab7eee4ad | ||
|
|
f90ae67864 | ||
|
|
ddb560abbf | ||
|
|
5de4210cfb | ||
|
|
daf730efd5 | ||
|
|
c61c9c085d | ||
|
|
e0dfc94363 | ||
|
|
9b451ecac3 | ||
|
|
26736d06c8 | ||
|
|
01d3c03fb2 | ||
|
|
b783bbe6d6 | ||
|
|
640c77dd56 | ||
|
|
b6ae8ad1cc | ||
|
|
c5e2485e0e | ||
|
|
de2ce99fdd | ||
|
|
0867b96302 | ||
|
|
78895a002f | ||
|
|
9f9e61289b | ||
|
|
69dea9c4c8 | ||
|
|
94ce790891 | ||
|
|
21ed97037a | ||
|
|
1ab7d13b96 | ||
|
|
47b3d514bb | ||
|
|
e653c3ad1d | ||
|
|
0e33e39759 | ||
|
|
d11ed993ee | ||
|
|
8d4a7f2d00 | ||
|
|
06528cdd67 | ||
|
|
0d71db9b24 | ||
|
|
05a5fdd539 | ||
|
|
b35f386e46 | ||
|
|
59a4f4abb4 | ||
|
|
7ae4a5cb7e | ||
|
|
2dc8319b50 | ||
|
|
eea4f6fe30 | ||
|
|
c44d40b2e5 | ||
|
|
88b2fcbcd1 | ||
|
|
f7f86d5cfa | ||
|
|
2b2d2a65b2 | ||
|
|
123123605c | ||
|
|
710cfa3d31 | ||
|
|
05c690dcd9 | ||
|
|
fa52c674ef | ||
|
|
8ce6dde45b | ||
|
|
bfd45b3a54 | ||
|
|
8ddc4172cd | ||
|
|
81eeaa3a20 | ||
|
|
47c00fabc1 | ||
|
|
3bbb7da6e7 | ||
|
|
c4d9a93e56 | ||
|
|
f92bd65407 | ||
|
|
c91856641c | ||
|
|
3479606f83 | ||
|
|
797835ad79 | ||
|
|
96b3cbe594 | ||
|
|
3ada99b201 | ||
|
|
bd6b127358 | ||
|
|
481062423e | ||
|
|
328dcea8c8 | ||
|
|
f8ed7d3a78 | ||
|
|
12d79a62da | ||
|
|
c7b2e4c8b7 | ||
|
|
91d652ed84 | ||
|
|
590385339c | ||
|
|
44bb1e5f0e | ||
|
|
a4a961666a | ||
|
|
7af718b834 | ||
|
|
8060a631b1 | ||
|
|
01096abede | ||
|
|
8110982412 | ||
|
|
4429959fed | ||
|
|
920e64404d | ||
|
|
87af1db2bc | ||
|
|
391971f80b | ||
|
|
5aab3ed97d | ||
|
|
e6f838375a | ||
|
|
907acb5ad2 | ||
|
|
b6fc24d6e0 | ||
|
|
1b399ef5be | ||
|
|
c3c597df4d | ||
|
|
25f529757e | ||
|
|
edf437f056 | ||
|
|
8c79f3d093 | ||
|
|
55217146ae | ||
|
|
456824c504 | ||
|
|
4f0f64884d | ||
|
|
1f11f1cf8b | ||
|
|
d81b781d1f | ||
|
|
053fc9dc65 | ||
|
|
11c33771c2 | ||
|
|
5fd23f0ba3 | ||
|
|
c209fe2adc | ||
|
|
940351acbe | ||
|
|
f7a4a4d3d1 | ||
|
|
e81b5ae658 | ||
|
|
588093519a | ||
|
|
3a83a7fe62 | ||
|
|
b5a9f30d34 | ||
|
|
25026a55f7 | ||
|
|
467231d4b7 | ||
|
|
3aeb4b09bf | ||
|
|
9eed41ecd8 | ||
|
|
25e0d61e01 | ||
|
|
080bb3626d | ||
|
|
c26a194694 | ||
|
|
a577689cd4 | ||
|
|
d3f6d6d092 | ||
|
|
87e6531c65 | ||
|
|
faa6adcb23 | ||
|
|
880a8433cb | ||
|
|
f319a24ad3 | ||
|
|
17462abae7 | ||
|
|
978526c4e6 | ||
|
|
2901d09d01 | ||
|
|
af64c4820f | ||
|
|
0d01354737 | ||
|
|
d04364a186 | ||
|
|
4a66c65034 | ||
|
|
4eebad6f5f | ||
|
|
281a89508b | ||
|
|
f38c50724d | ||
|
|
76d737340e | ||
|
|
6edf90ee97 | ||
|
|
1b1647892d | ||
|
|
7863875641 | ||
|
|
39edc4e382 | ||
|
|
3f26303d6e | ||
|
|
37fed9dde7 | ||
|
|
d5800fd92f | ||
|
|
a1070a7c26 | ||
|
|
8549a4e1de | ||
|
|
06d735fe89 | ||
|
|
bcd1f8ea0b | ||
|
|
e6d01fc7f5 | ||
|
|
4adef0a280 | ||
|
|
b8a94d8db5 | ||
|
|
1bc346d22c | ||
|
|
1d2a0a6f47 | ||
|
|
f4e0ca1677 | ||
|
|
b5d7e957cf | ||
|
|
2b26ea4ddd | ||
|
|
52d122583e | ||
|
|
8e0e4568fe | ||
|
|
a59274c371 | ||
|
|
d0e9f67a25 | ||
|
|
b2485f6133 | ||
|
|
953b553089 | ||
|
|
4db29780e0 | ||
|
|
69e6d8d6e2 | ||
|
|
5b8eef255d | ||
|
|
a77d2d67f6 | ||
|
|
8f7bde6e0a | ||
|
|
b7ad6ed815 | ||
|
|
99978de2ab | ||
|
|
37cefe6d61 | ||
|
|
6c016c5913 | ||
|
|
a684544982 | ||
|
|
006d35914a | ||
|
|
c85120054e | ||
|
|
3fe8f17f94 | ||
|
|
273254ff4e | ||
|
|
8cad4a51e4 | ||
|
|
371b95608d | ||
|
|
f13673bff1 | ||
|
|
2ac83ca857 | ||
|
|
f1b9665c53 | ||
|
|
bd0804d443 | ||
|
|
107b57849e | ||
|
|
7803f64630 | ||
|
|
94c3315bf0 | ||
|
|
7710efd4db | ||
|
|
c31c67f69e | ||
|
|
f50f08f40d | ||
|
|
d9f035ef44 | ||
|
|
6b4f0160d5 | ||
|
|
ec1529061b | ||
|
|
65db04d3f9 | ||
|
|
9b96a443ce | ||
|
|
2e0bcc158f | ||
|
|
91a2161776 | ||
|
|
7a894873d1 | ||
|
|
8b6abd75f8 | ||
|
|
4c8610a960 | ||
|
|
31e3abb041 | ||
|
|
43a7da75cd | ||
|
|
0b5ee00b58 | ||
|
|
f1725b991e | ||
|
|
fd02942f4d | ||
|
|
30ce474aab | ||
|
|
e6ea691385 | ||
|
|
2ceed980b4 | ||
|
|
fa0d3959b6 | ||
|
|
446b7327e4 | ||
|
|
27e3bc1c6d | ||
|
|
8956692e9b | ||
|
|
979989e166 | ||
|
|
5c368b66e4 | ||
|
|
7bcbe2f01d | ||
|
|
0bd0cf9e65 | ||
|
|
fc64e7af99 | ||
|
|
9946dfa568 | ||
|
|
92b7a9b798 | ||
|
|
738590b631 | ||
|
|
5c62cdcc8a | ||
|
|
107c4c4a76 | ||
|
|
fa08aecbc2 | ||
|
|
ca792e2c7f | ||
|
|
c54826de6a | ||
|
|
a79ff1fd38 | ||
|
|
3cb3424250 | ||
|
|
f4d656d8ef | ||
|
|
29aed42c67 | ||
|
|
fed9590a2c | ||
|
|
01aff3edf4 | ||
|
|
3fadf33113 | ||
|
|
b0570fcdc5 | ||
|
|
ea984897b9 | ||
|
|
2b74e3cd77 | ||
|
|
bf026f6d7d | ||
|
|
d3bd896c49 | ||
|
|
cb5cab493e | ||
|
|
af3a081a81 | ||
|
|
d2b8174476 | ||
|
|
558086655f | ||
|
|
4ea054084c | ||
|
|
ad134527d4 | ||
|
|
8962d6b4cf | ||
|
|
cab30f4822 | ||
|
|
a0b28d00e8 | ||
|
|
4c5f4890f6 | ||
|
|
c33c375fb2 | ||
|
|
3ef59ba3ba | ||
|
|
2f8bd4d281 | ||
|
|
fa7ddaa531 | ||
|
|
1bd0a5ca10 | ||
|
|
f8a3f4186c | ||
|
|
a98f76f6e3 | ||
|
|
b4c8cf61e7 | ||
|
|
4215695759 | ||
|
|
69d2bfa1ba | ||
|
|
17239ae5cf | ||
|
|
2ab8484794 | ||
|
|
4a0d1f09f6 | ||
|
|
5f2bc51bbc | ||
|
|
e8c97e1938 | ||
|
|
527a18a1e5 | ||
|
|
e0cddc9a7b | ||
|
|
a746c341ae | ||
|
|
300925260e | ||
|
|
0043c50f32 | ||
|
|
d01aaa0e1f | ||
|
|
e1906ff533 | ||
|
|
7836aadd71 | ||
|
|
cca5baa447 | ||
|
|
e5d8c5cb48 | ||
|
|
44e64b007b | ||
|
|
c54822686e | ||
|
|
1d5e322c39 | ||
|
|
8856611706 | ||
|
|
7037da7834 | ||
|
|
672a0a98e8 | ||
|
|
62b4e94b91 | ||
|
|
f48eaa0e04 | ||
|
|
e09333c948 | ||
|
|
117be556cf | ||
|
|
3e0d2c35cd | ||
|
|
cf0877c777 | ||
|
|
72f6733cfe | ||
|
|
3f258453da | ||
|
|
4d4589a1b8 | ||
|
|
73dcc40929 | ||
|
|
7b8a3b8088 | ||
|
|
0ec2b3f2cb | ||
|
|
5e241f43b0 | ||
|
|
05ba79d185 | ||
|
|
6bc7f12746 | ||
|
|
de688c6f7f | ||
|
|
e283488256 | ||
|
|
c499645797 | ||
|
|
e0dad876ec | ||
|
|
a6b200b44a | ||
|
|
adeac3e16b | ||
|
|
f15f55d20a | ||
|
|
f62bff78c9 | ||
|
|
42804bf56d | ||
|
|
8b6b66e766 | ||
|
|
cf4e221fdf | ||
|
|
67a7302595 | ||
|
|
3b53e8b69d | ||
|
|
d80d27a10c | ||
|
|
38b8c9536d | ||
|
|
c7428f2d55 | ||
|
|
d7060f3256 | ||
|
|
50f43d955a | ||
|
|
d1cabf3a4d | ||
|
|
c57a6048cf | ||
|
|
dda2debf36 | ||
|
|
863e72a256 | ||
|
|
8d435bc1b1 | ||
|
|
22b28f8cd4 | ||
|
|
c37b23eca9 | ||
|
|
b9f2035b7c | ||
|
|
f01f8b4fca | ||
|
|
5c840fe9ba | ||
|
|
86b8dd966e | ||
|
|
b53cd1cb2b | ||
|
|
65da42e1f3 | ||
|
|
2359ca8838 | ||
|
|
8e574ba33d | ||
|
|
e1e9858876 | ||
|
|
3b521eb57f | ||
|
|
66a4b3b68b | ||
|
|
ee260724ae | ||
|
|
645b7c2e7c | ||
|
|
953d057876 | ||
|
|
1d33eec01f | ||
|
|
114d20d97a | ||
|
|
257a9c73d1 | ||
|
|
01e8fa5407 | ||
|
|
fc3a12c7aa | ||
|
|
a454a7dafe | ||
|
|
c742712f6f | ||
|
|
110cb80e82 | ||
|
|
efdbb3e6c7 | ||
|
|
86c3ef089c | ||
|
|
bbf3cfd9bd | ||
|
|
1bb7296d53 | ||
|
|
b3dc796d73 | ||
|
|
a513680769 | ||
|
|
7b04f14c2a | ||
|
|
ded3c79d50 | ||
|
|
2795a051eb | ||
|
|
35c6f57a89 | ||
|
|
f7d554fea5 | ||
|
|
8f3cae7792 | ||
|
|
ae342db18d | ||
|
|
32638ce7d4 | ||
|
|
c2d25d7a28 | ||
|
|
90743d5b78 | ||
|
|
07d70e6321 | ||
|
|
57cf0d4836 | ||
|
|
9e1e8e576c | ||
|
|
07aae3b04a | ||
|
|
beee2bd39a | ||
|
|
850ee61614 | ||
|
|
ce4931ca06 | ||
|
|
070c5362c3 | ||
|
|
88f5eafb45 | ||
|
|
954f3fdef5 | ||
|
|
98edc7cc1b | ||
|
|
3a056a21ae | ||
|
|
82cef67c71 | ||
|
|
8bc23853fc | ||
|
|
92f64c4645 | ||
|
|
b4637a808b | ||
|
|
9cc5865a0d | ||
|
|
be3f728e3a | ||
|
|
b0c924f9c7 | ||
|
|
aa7bbfc26f | ||
|
|
5415fa3576 | ||
|
|
cf7d241b51 | ||
|
|
ab6c28166e | ||
|
|
d6d19ad3cf | ||
|
|
07ac11c6ab | ||
|
|
b09b1cf0e0 | ||
|
|
5244a15e70 | ||
|
|
4f43a92380 | ||
|
|
96399541b6 | ||
|
|
bd44c40fdf | ||
|
|
f7c386f480 | ||
|
|
06d322e6cf | ||
|
|
7228dacb91 | ||
|
|
1234027472 | ||
|
|
7004064567 | ||
|
|
c47f788094 | ||
|
|
c97efce5f6 | ||
|
|
87e4045a75 | ||
|
|
dbe84e64b9 | ||
|
|
902b0df117 | ||
|
|
7ded999868 | ||
|
|
5014eceaad | ||
|
|
db5258ea45 | ||
|
|
815c48bfcc | ||
|
|
d162be7e4a | ||
|
|
fda267bd77 | ||
|
|
4065ee315a | ||
|
|
41cb7231f1 | ||
|
|
596a23670f | ||
|
|
08d2e3a7b5 | ||
|
|
ba088d441a | ||
|
|
d95fe53d78 | ||
|
|
e6c2eb4380 | ||
|
|
8dce10bda8 | ||
|
|
0f4f0c5567 | ||
|
|
c74a890068 | ||
|
|
67edd707eb | ||
|
|
a7454566c8 | ||
|
|
3a71ea75ee | ||
|
|
bcb8ee44de | ||
|
|
e7a454e8ab | ||
|
|
2fb66be14d | ||
|
|
4ca4ac4285 | ||
|
|
25ae548fa6 | ||
|
|
123d6597a7 | ||
|
|
4114527daf | ||
|
|
c0decd2efe | ||
|
|
a5deca0508 | ||
|
|
b179025c7d | ||
|
|
626045549f | ||
|
|
9f91c32a0e | ||
|
|
f757e3a948 | ||
|
|
9b8de340fd | ||
|
|
a84bc69a26 | ||
|
|
625f91cc7e | ||
|
|
34d32a1e74 | ||
|
|
6bfc1a735c | ||
|
|
d2ab0aa751 | ||
|
|
7dcc329cf8 | ||
|
|
82793f6fd4 | ||
|
|
1e9218a90c | ||
|
|
a1144caa61 | ||
|
|
5ab7f2f4a8 | ||
|
|
4184edcd30 | ||
|
|
2cacf51d95 | ||
|
|
f64b8c2f26 | ||
|
|
104ebf746b | ||
|
|
0b37c524c7 | ||
|
|
9f603fc889 | ||
|
|
50c55e090e | ||
|
|
2951ad9e47 | ||
|
|
ed9f5f9aa5 | ||
|
|
4fdec752ca | ||
|
|
9079930081 | ||
|
|
97a4f73d3a | ||
|
|
f7af23d5cb | ||
|
|
634eed9599 | ||
|
|
b6344ff0f1 | ||
|
|
125569ff2a | ||
|
|
8d5b3ca476 | ||
|
|
3a8f405e2d | ||
|
|
bb98c4b7fc | ||
|
|
a0522ccce3 | ||
|
|
47d92c57ad | ||
|
|
0871034893 | ||
|
|
a8796c9c60 | ||
|
|
6b92bc5676 | ||
|
|
616b5fd19b | ||
|
|
5d6806fe7d | ||
|
|
b3ded74acf | ||
|
|
f73a19318a | ||
|
|
235967bd44 | ||
|
|
12583140ac | ||
|
|
1ed3172f9e | ||
|
|
fd270c18ac | ||
|
|
f409eeb7b0 | ||
|
|
db4dfa30df | ||
|
|
c027959454 | ||
|
|
76fdac00e6 | ||
|
|
7680c0e6ed | ||
|
|
9fada0e168 | ||
|
|
609e243866 | ||
|
|
20cf4cc080 | ||
|
|
392aa3cd4c | ||
|
|
96e7799988 | ||
|
|
8521efbe32 | ||
|
|
649db6688a | ||
|
|
0061ba43d8 | ||
|
|
e98c6d7ebc | ||
|
|
1033d2d168 | ||
|
|
60dafcf770 | ||
|
|
a8114818d1 | ||
|
|
4a0cabd62c | ||
|
|
b82cec20a7 | ||
|
|
cd2daf8752 | ||
|
|
7f1b54027b | ||
|
|
d1e5d499fb | ||
|
|
42e92ecbff | ||
|
|
01a6476f9c | ||
|
|
d238f4f3e8 | ||
|
|
8793f172e3 | ||
|
|
4f41d3b12f | ||
|
|
43d4ad7218 | ||
|
|
6babd83c3b | ||
|
|
7b5e2017ee | ||
|
|
80209ba155 | ||
|
|
4a88933eb2 | ||
|
|
1dbdd8c707 | ||
|
|
c90b2cb3ce | ||
|
|
7d33f25040 | ||
|
|
16a74dc589 | ||
|
|
826fcbe74d | ||
|
|
75577ca5ca | ||
|
|
7a739c9103 | ||
|
|
47ea1861da | ||
|
|
3af2ff5e02 | ||
|
|
a593c53116 | ||
|
|
6c92f02edd | ||
|
|
fb6244de69 | ||
|
|
9d30182c64 | ||
|
|
a59e8b5e2e | ||
|
|
9aa53ea118 | ||
|
|
19f786a0f1 | ||
|
|
9cb47c7d8a | ||
|
|
b51e5f0654 | ||
|
|
16b9bc5585 | ||
|
|
4703dd4d27 | ||
|
|
d79ba0620d | ||
|
|
c7863f758e | ||
|
|
30486ed486 | ||
|
|
f8c9557166 | ||
|
|
a3aa6ea44a | ||
|
|
14ea41b1d9 | ||
|
|
8039e6b09a | ||
|
|
1e8a1b7b76 | ||
|
|
32a1b97012 | ||
|
|
ed84031fbb | ||
|
|
a8cc7d9441 | ||
|
|
b0c9bca53a | ||
|
|
352e0ee84a | ||
|
|
0baeeb8dbb | ||
|
|
98be88f8ae | ||
|
|
76664f6732 | ||
|
|
6b560af913 | ||
|
|
7bf98c0c42 | ||
|
|
8cf93ec241 | ||
|
|
164e54c608 | ||
|
|
c691d6a8d5 | ||
|
|
bae4776747 | ||
|
|
f94f418947 | ||
|
|
b33345bc92 | ||
|
|
3c36226b71 | ||
|
|
420f6db56b | ||
|
|
127238b98d | ||
|
|
15f248cad2 | ||
|
|
2bd59d0c65 | ||
|
|
f8d6fd7133 | ||
|
|
518fe1f3cc | ||
|
|
67b0893f33 | ||
|
|
6e6495f948 | ||
|
|
06a787bedf | ||
|
|
0ed44e458e | ||
|
|
3a404d24a1 | ||
|
|
6d13aa448f | ||
|
|
0b68f8586c | ||
|
|
2b2d83d0a6 | ||
|
|
71c58db50e | ||
|
|
fda5d69071 | ||
|
|
bb45eb09f4 | ||
|
|
8dfdb678f9 | ||
|
|
bf1a8bd924 | ||
|
|
1a02ded4be | ||
|
|
aa95508482 | ||
|
|
22c7acd74e | ||
|
|
15c2f1aca6 | ||
|
|
ba26a3b1c1 | ||
|
|
6ac7303e16 | ||
|
|
8a93f69e03 | ||
|
|
1132d5f0a7 | ||
|
|
681689b7dd | ||
|
|
cce917681f | ||
|
|
0079abaa73 | ||
|
|
ab9b080fbd | ||
|
|
6dc34d5cae | ||
|
|
dfdb1907c7 | ||
|
|
97c67219e2 | ||
|
|
11ff1be321 | ||
|
|
4c48b1a563 | ||
|
|
a2a4b38cc4 | ||
|
|
d190f2bc80 | ||
|
|
35c24548f4 | ||
|
|
53e35355ac | ||
|
|
edcdbb510e | ||
|
|
1c1cea7a3b | ||
|
|
b9ca846835 | ||
|
|
9e667c7a86 | ||
|
|
68c910d107 | ||
|
|
8045f5689f | ||
|
|
5f4eebff65 | ||
|
|
e66aa7c802 | ||
|
|
e3667c0bbd | ||
|
|
8f26fd515b | ||
|
|
0f0150990a | ||
|
|
0db4cce3fe | ||
|
|
cd7906710a | ||
|
|
ece01d3803 | ||
|
|
88cd9d430f | ||
|
|
21e5e78876 | ||
|
|
f8696508fa | ||
|
|
febbaa82cc | ||
|
|
95cdcb37be | ||
|
|
ce6860dd86 | ||
|
|
87886aeec9 | ||
|
|
382f2d7da6 | ||
|
|
3c57b550df | ||
|
|
ae81e2aae5 | ||
|
|
6035b5d038 | ||
|
|
f56bf965ee | ||
|
|
bd65329641 | ||
|
|
fbbfa97d4b | ||
|
|
af4c76ea4d | ||
|
|
49ea4c8c4d | ||
|
|
cafb42321e | ||
|
|
0b0c77daac | ||
|
|
ca2977f3e3 | ||
|
|
ca1ef6540b | ||
|
|
282875d4f7 | ||
|
|
20a23cb8ba | ||
|
|
a5d0c78ba6 | ||
|
|
242abf9079 | ||
|
|
4965e032f8 | ||
|
|
a47395a7c7 | ||
|
|
f756dd3646 | ||
|
|
00cb04d355 | ||
|
|
5c68f07208 | ||
|
|
b08bd05909 | ||
|
|
e1f3a3fe27 | ||
|
|
d27409f949 | ||
|
|
38c6e2af9a | ||
|
|
0c4fabfa85 | ||
|
|
1b20031cd7 | ||
|
|
c698d6cae7 | ||
|
|
2ce0247063 | ||
|
|
a66827790f | ||
|
|
0bc25f0c01 | ||
|
|
950456ba99 | ||
|
|
e32cb6ce10 | ||
|
|
23af06525d | ||
|
|
044ec9e2c0 | ||
|
|
8a7e4d3770 | ||
|
|
463d23fd2c | ||
|
|
94362dbf07 | ||
|
|
3a94699c39 | ||
|
|
33d621d8cc | ||
|
|
576ba1ace6 | ||
|
|
42790d5321 | ||
|
|
38dd031c7f | ||
|
|
bd1c334722 | ||
|
|
b5f21bae0b | ||
|
|
d3ec9e76bd | ||
|
|
a9a1535a12 | ||
|
|
cb535e33e7 | ||
|
|
2541feebe7 | ||
|
|
5a65d378d2 | ||
|
|
3b83c75e5f | ||
|
|
204bbeb7f5 | ||
|
|
b18c22ec83 | ||
|
|
9dc282f1fc | ||
|
|
d95b1c53c5 | ||
|
|
ff665e0d94 | ||
|
|
e8908b23e3 | ||
|
|
36329a44b3 | ||
|
|
28483e669b | ||
|
|
e08f6c4039 | ||
|
|
04999bfb4e | ||
|
|
34134c8f08 | ||
|
|
de9310a452 | ||
|
|
7c06858d17 | ||
|
|
ebfa0710cb | ||
|
|
4d617c9321 | ||
|
|
92a23b185f | ||
|
|
27e9591630 | ||
|
|
90d927df0a | ||
|
|
ea046e7d7c | ||
|
|
eca218ac79 | ||
|
|
eb26616752 | ||
|
|
e29c0c1814 | ||
|
|
5080c84e3d | ||
|
|
bc05bce077 | ||
|
|
c284558886 | ||
|
|
22498953ba | ||
|
|
8a0a24d0e7 | ||
|
|
146f00018d | ||
|
|
0b90a0c57a | ||
|
|
81b34f61cf | ||
|
|
ab85d3f191 | ||
|
|
968415806d | ||
|
|
c9b802b7a1 | ||
|
|
f0d6cd5f8c | ||
|
|
e8b30b2282 | ||
|
|
b662cd160a | ||
|
|
f5e290beec | ||
|
|
99e7799dae | ||
|
|
e87485441e | ||
|
|
15a48e6723 | ||
|
|
7a83dda059 | ||
|
|
a287d9ca59 | ||
|
|
336deaab35 | ||
|
|
19fe56c3b3 | ||
|
|
ba06f39669 | ||
|
|
8219fbe163 | ||
|
|
6e00de04b0 | ||
|
|
90638a61d5 | ||
|
|
b833fe89ca | ||
|
|
1acb9e722e | ||
|
|
12b08baed9 | ||
|
|
bb22db0fd6 | ||
|
|
495071089a | ||
|
|
5e6dd2ad46 | ||
|
|
f593a34cf1 | ||
|
|
75e8140318 | ||
|
|
60f3f7e9de | ||
|
|
86e2590f4f | ||
|
|
1c385b33f1 | ||
|
|
3c97199a10 | ||
|
|
6db185b03e | ||
|
|
c8d957ef5f | ||
|
|
d250b766ce | ||
|
|
eb1f772e31 | ||
|
|
8350747af2 | ||
|
|
2bcbb38778 | ||
|
|
a7762d0c23 | ||
|
|
73b2f2d7cb | ||
|
|
2795a5e9c4 | ||
|
|
a2029da351 | ||
|
|
fbb2b12d85 | ||
|
|
ec5c560ca1 | ||
|
|
ec82a760ed | ||
|
|
21f047337b | ||
|
|
10fe87e1af | ||
|
|
ab96594746 | ||
|
|
b69b1b0485 | ||
|
|
176daf7c6c | ||
|
|
57dc25f9aa | ||
|
|
5e83bb6d75 | ||
|
|
494c965e8f | ||
|
|
ec45eac7b6 | ||
|
|
2e7f96099d | ||
|
|
c22c95a963 | ||
|
|
348fb4fcf0 | ||
|
|
49ad0f765f | ||
|
|
4c4f272d0e | ||
|
|
34753e7c80 | ||
|
|
0444c087e4 | ||
|
|
e372f750bd | ||
|
|
bc84346bf5 | ||
|
|
4cc0ab1eef | ||
|
|
4499bf05a7 | ||
|
|
cc69485c8a | ||
|
|
ad8343e051 | ||
|
|
0ffddcba16 | ||
|
|
3f5216c29b | ||
|
|
90fe3ad33c | ||
|
|
07a2da0278 | ||
|
|
8924a26c86 | ||
|
|
23ca647bcf | ||
|
|
ad969736ec | ||
|
|
4bfb00d664 | ||
|
|
620a85e417 | ||
|
|
69e5af7a31 | ||
|
|
317bca6c2e | ||
|
|
e5a5425f0b | ||
|
|
41d8ddbe23 | ||
|
|
bf1aecfa8b | ||
|
|
b537f763eb | ||
|
|
1ecd37622e | ||
|
|
b4420091e1 | ||
|
|
5b7a06a19e | ||
|
|
5fb64dece8 | ||
|
|
b90876f778 | ||
|
|
92eddb5bf0 | ||
|
|
2153e14745 | ||
|
|
3b72f4d949 | ||
|
|
34fc9c5621 | ||
|
|
6120679bf1 | ||
|
|
97f4fdca28 | ||
|
|
0ff2c8d2a9 | ||
|
|
6cb1be07ce | ||
|
|
179c48a28d | ||
|
|
6926bf940a | ||
|
|
9363eb04b7 | ||
|
|
8703207b56 | ||
|
|
b15f59cbf3 | ||
|
|
8ea076d500 | ||
|
|
625eb229c9 | ||
|
|
40f3deff4d | ||
|
|
9467e46962 | ||
|
|
22496e77a8 | ||
|
|
27fd91a351 | ||
|
|
2c370ec3d9 | ||
|
|
7c086fb9c7 | ||
|
|
09f4d17dfa | ||
|
|
803833808d | ||
|
|
614e0ce8ce | ||
|
|
3c51b99c0b | ||
|
|
5447b6cf93 | ||
|
|
cf40acf2ff | ||
|
|
fdc8d271d8 | ||
|
|
dddec5f86f | ||
|
|
c68554c26d | ||
|
|
c4556ee04a | ||
|
|
03b1debbec | ||
|
|
e85ebedb57 | ||
|
|
6cbb433aaa | ||
|
|
9df9421baf | ||
|
|
e623ae10b2 | ||
|
|
6d67dca928 | ||
|
|
2e98dcb6f6 | ||
|
|
a611c9bba9 | ||
|
|
0ab2d88fb3 | ||
|
|
d0cde330d7 | ||
|
|
28b0b6c90d | ||
|
|
20181467f3 | ||
|
|
fa6fbaf96a | ||
|
|
56b81f8027 | ||
|
|
05c67d7882 | ||
|
|
0921668ae8 | ||
|
|
1d0c20cb36 | ||
|
|
fc649dc263 | ||
|
|
8313e71d5d | ||
|
|
51801f2d1a | ||
|
|
f36691adb0 | ||
|
|
7e11da2e10 | ||
|
|
53345e1511 | ||
|
|
95a60496e6 | ||
|
|
dcb2aa5d8b | ||
|
|
6d3e436070 | ||
|
|
f5736a1389 | ||
|
|
9ee9dcd09a | ||
|
|
0e730afbaf | ||
|
|
38cda7aab9 | ||
|
|
558bf954fe | ||
|
|
0f6da5de63 | ||
|
|
8736c1cc49 | ||
|
|
77ce34f553 | ||
|
|
038d567a59 | ||
|
|
4e9f2d32cc | ||
|
|
a6d8012a8e | ||
|
|
4645ff5f58 | ||
|
|
8636314078 | ||
|
|
99972eec1b | ||
|
|
f8190395e3 | ||
|
|
58e425fd62 | ||
|
|
9a7d6bc4b9 | ||
|
|
e754c4832f | ||
|
|
93b5fcd526 | ||
|
|
6fba8c70ae | ||
|
|
fe4bf53cfc | ||
|
|
0884ba5ab3 | ||
|
|
3dbfdb359f | ||
|
|
db4388e322 | ||
|
|
a3540195ab | ||
|
|
f37618331d | ||
|
|
4dc1c58fbe | ||
|
|
2011d51d27 | ||
|
|
0ac5a78031 | ||
|
|
a05181a2e3 | ||
|
|
8f07453f38 | ||
|
|
6d64f7ae15 |
@ -49,11 +49,365 @@
|
|||||||
"contributions": [
|
"contributions": [
|
||||||
"code"
|
"code"
|
||||||
]
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"login": "Uriziel01",
|
||||||
|
"name": "Paweł Borecki",
|
||||||
|
"avatar_url": "https://avatars1.githubusercontent.com/u/568207?v=4",
|
||||||
|
"profile": "http://uriziel.pl",
|
||||||
|
"contributions": [
|
||||||
|
"code"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"login": "marcus-sa",
|
||||||
|
"name": "Marcus S. Abildskov",
|
||||||
|
"avatar_url": "https://avatars0.githubusercontent.com/u/8391194?v=4",
|
||||||
|
"profile": "http://marcus-sa.me",
|
||||||
|
"contributions": [
|
||||||
|
"test"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"login": "atilacamurca",
|
||||||
|
"name": "Átila Camurça Alves",
|
||||||
|
"avatar_url": "https://avatars3.githubusercontent.com/u/508624?v=4",
|
||||||
|
"profile": "http://www.mad3linux.org",
|
||||||
|
"contributions": [
|
||||||
|
"doc"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"login": "jameshibbard",
|
||||||
|
"name": "James Hibbard",
|
||||||
|
"avatar_url": "https://avatars2.githubusercontent.com/u/1940994?v=4",
|
||||||
|
"profile": "http://hibbard.eu",
|
||||||
|
"contributions": [
|
||||||
|
"code"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"login": "soonoo",
|
||||||
|
"name": "Soonwoo Hong",
|
||||||
|
"avatar_url": "https://avatars2.githubusercontent.com/u/5436405?v=4",
|
||||||
|
"profile": "https://github.com/soonoo",
|
||||||
|
"contributions": [
|
||||||
|
"code"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"login": "illBeRoy",
|
||||||
|
"name": "Roy Sommer",
|
||||||
|
"avatar_url": "https://avatars2.githubusercontent.com/u/6681893?v=4",
|
||||||
|
"profile": "https://github.com/illBeRoy",
|
||||||
|
"contributions": [
|
||||||
|
"code"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"login": "paulocoghi",
|
||||||
|
"name": "Paulo Coghi",
|
||||||
|
"avatar_url": "https://avatars1.githubusercontent.com/u/378397?v=4",
|
||||||
|
"profile": "https://github.com/paulocoghi",
|
||||||
|
"contributions": [
|
||||||
|
"ideas"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"login": "balthild",
|
||||||
|
"name": "Balthild Ires",
|
||||||
|
"avatar_url": "https://avatars2.githubusercontent.com/u/2662758?v=4",
|
||||||
|
"profile": "https://balthild.com",
|
||||||
|
"contributions": [
|
||||||
|
"code"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"login": "dimitarnestorov",
|
||||||
|
"name": "Dimitar Nestorov",
|
||||||
|
"avatar_url": "https://avatars0.githubusercontent.com/u/8790386?v=4",
|
||||||
|
"profile": "https://dimitarnestorov.com",
|
||||||
|
"contributions": [
|
||||||
|
"code"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"login": "msairio",
|
||||||
|
"name": "Mikko Sairio",
|
||||||
|
"avatar_url": "https://avatars3.githubusercontent.com/u/3404389?v=4",
|
||||||
|
"profile": "https://www.greatapes.fi",
|
||||||
|
"contributions": [
|
||||||
|
"code"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"login": "pepf",
|
||||||
|
"name": "Pepijn",
|
||||||
|
"avatar_url": "https://avatars1.githubusercontent.com/u/1265435?v=4",
|
||||||
|
"profile": "http://blog.pepf.nl",
|
||||||
|
"contributions": [
|
||||||
|
"code"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"login": "slidinghotdog",
|
||||||
|
"name": "slidinghotdog",
|
||||||
|
"avatar_url": "https://avatars3.githubusercontent.com/u/33790211?v=4",
|
||||||
|
"profile": "https://github.com/slidinghotdog",
|
||||||
|
"contributions": [
|
||||||
|
"code"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"login": "bundyo",
|
||||||
|
"name": "Bundyo (Kamen Bundev)",
|
||||||
|
"avatar_url": "https://avatars1.githubusercontent.com/u/98318?v=4",
|
||||||
|
"profile": "http://www.bundyo.org",
|
||||||
|
"contributions": [
|
||||||
|
"code"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"login": "feng8848",
|
||||||
|
"name": "feng8848",
|
||||||
|
"avatar_url": "https://avatars2.githubusercontent.com/u/40539968?v=4",
|
||||||
|
"profile": "https://github.com/feng8848",
|
||||||
|
"contributions": [
|
||||||
|
"code"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"login": "Ty3uK",
|
||||||
|
"name": "Maksim Karelov",
|
||||||
|
"avatar_url": "https://avatars3.githubusercontent.com/u/2384454?v=4",
|
||||||
|
"profile": "https://karelov.info",
|
||||||
|
"contributions": [
|
||||||
|
"code"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"login": "mspencer92",
|
||||||
|
"name": "mspencer92",
|
||||||
|
"avatar_url": "https://avatars2.githubusercontent.com/u/1910455?v=4",
|
||||||
|
"profile": "https://github.com/mspencer92",
|
||||||
|
"contributions": [
|
||||||
|
"code"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"login": "blncd2020",
|
||||||
|
"name": "blncd2020",
|
||||||
|
"avatar_url": "https://avatars1.githubusercontent.com/u/59541979?v=4",
|
||||||
|
"profile": "https://github.com/blncd2020",
|
||||||
|
"contributions": [
|
||||||
|
"doc"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"login": "gluaxspeed",
|
||||||
|
"name": "gluaxspeed",
|
||||||
|
"avatar_url": "https://avatars2.githubusercontent.com/u/16431709?v=4",
|
||||||
|
"profile": "https://github.com/gluaxspeed",
|
||||||
|
"contributions": [
|
||||||
|
"code"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"login": "Solant",
|
||||||
|
"name": "Solant",
|
||||||
|
"avatar_url": "https://avatars2.githubusercontent.com/u/5971578?v=4",
|
||||||
|
"profile": "https://github.com/Solant",
|
||||||
|
"contributions": [
|
||||||
|
"code"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"login": "shubhamzanwar",
|
||||||
|
"name": "Shubham Zanwar",
|
||||||
|
"avatar_url": "https://avatars0.githubusercontent.com/u/15626155?v=4",
|
||||||
|
"profile": "https://github.com/shubhamzanwar",
|
||||||
|
"contributions": [
|
||||||
|
"code"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"login": "Singha360",
|
||||||
|
"name": "Singha360",
|
||||||
|
"avatar_url": "https://avatars1.githubusercontent.com/u/35334787?v=4",
|
||||||
|
"profile": "https://github.com/Singha360",
|
||||||
|
"contributions": [
|
||||||
|
"code"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"login": "MihkelBaranov",
|
||||||
|
"name": "Mihkel",
|
||||||
|
"avatar_url": "https://avatars3.githubusercontent.com/u/3790782?v=4",
|
||||||
|
"profile": "http://wellenline.com",
|
||||||
|
"contributions": [
|
||||||
|
"code"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"login": "stevo2588",
|
||||||
|
"name": "Stephen A",
|
||||||
|
"avatar_url": "https://avatars1.githubusercontent.com/u/3278045?v=4",
|
||||||
|
"profile": "https://github.com/stevo2588",
|
||||||
|
"contributions": [
|
||||||
|
"code"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"login": "jardenliu",
|
||||||
|
"name": "流君酱",
|
||||||
|
"avatar_url": "https://avatars1.githubusercontent.com/u/15191056?v=4",
|
||||||
|
"profile": "http://liujunjiang.com",
|
||||||
|
"contributions": [
|
||||||
|
"code"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"login": "agg23",
|
||||||
|
"name": "Adam Gastineau",
|
||||||
|
"avatar_url": "https://avatars1.githubusercontent.com/u/238679?v=4",
|
||||||
|
"profile": "https://github.com/agg23",
|
||||||
|
"contributions": [
|
||||||
|
"code"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"login": "swittk",
|
||||||
|
"name": "swittk",
|
||||||
|
"avatar_url": "https://avatars2.githubusercontent.com/u/5000572?v=4",
|
||||||
|
"profile": "https://github.com/swittk",
|
||||||
|
"contributions": [
|
||||||
|
"code"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"login": "craftingmod",
|
||||||
|
"name": "craftingmod",
|
||||||
|
"avatar_url": "https://avatars2.githubusercontent.com/u/9389278?v=4",
|
||||||
|
"profile": "https://github.com/craftingmod",
|
||||||
|
"contributions": [
|
||||||
|
"code"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"login": "dooglio",
|
||||||
|
"name": "Doug Barbieri",
|
||||||
|
"avatar_url": "https://avatars1.githubusercontent.com/u/643129?v=4",
|
||||||
|
"profile": "http://www.m2osw.com",
|
||||||
|
"contributions": [
|
||||||
|
"code"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"login": "NeryHenrique",
|
||||||
|
"name": "HENRIQUE DE SOUZA NERY",
|
||||||
|
"avatar_url": "https://avatars0.githubusercontent.com/u/6879141?v=4",
|
||||||
|
"profile": "https://github.com/NeryHenrique",
|
||||||
|
"contributions": [
|
||||||
|
"code"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"login": "ruslang02",
|
||||||
|
"name": "Ruslan Garifullin",
|
||||||
|
"avatar_url": "https://avatars0.githubusercontent.com/u/25264730?v=4",
|
||||||
|
"profile": "https://ruslang.xyz",
|
||||||
|
"contributions": [
|
||||||
|
"code"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"login": "ran-j",
|
||||||
|
"name": "Ranieri",
|
||||||
|
"avatar_url": "https://avatars0.githubusercontent.com/u/17410205?v=4",
|
||||||
|
"profile": "https://github.com/ran-j",
|
||||||
|
"contributions": [
|
||||||
|
"code",
|
||||||
|
"doc"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"login": "NathanaelA",
|
||||||
|
"name": "Nathanael Anderson",
|
||||||
|
"avatar_url": "https://avatars3.githubusercontent.com/u/850871?v=4",
|
||||||
|
"profile": "https://master-technology.com",
|
||||||
|
"contributions": [
|
||||||
|
"code"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"login": "rocbear",
|
||||||
|
"name": "Ross",
|
||||||
|
"avatar_url": "https://avatars0.githubusercontent.com/u/4415071?v=4",
|
||||||
|
"profile": "https://ubiq.co.za",
|
||||||
|
"contributions": [
|
||||||
|
"code",
|
||||||
|
"doc"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"login": "4h7l",
|
||||||
|
"name": "4h7l",
|
||||||
|
"avatar_url": "https://avatars.githubusercontent.com/u/69183283?v=4",
|
||||||
|
"profile": "https://github.com/4h7l",
|
||||||
|
"contributions": [
|
||||||
|
"code"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"login": "Maks-s",
|
||||||
|
"name": "Maks",
|
||||||
|
"avatar_url": "https://avatars.githubusercontent.com/u/26678512?v=4",
|
||||||
|
"profile": "https://github.com/Maks-s",
|
||||||
|
"contributions": [
|
||||||
|
"code"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"login": "zhb124404",
|
||||||
|
"name": "zhb124404",
|
||||||
|
"avatar_url": "https://avatars.githubusercontent.com/u/16805041?v=4",
|
||||||
|
"profile": "https://github.com/zhb124404",
|
||||||
|
"contributions": [
|
||||||
|
"doc"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"login": "wkirby",
|
||||||
|
"name": "Wyatt Kirby",
|
||||||
|
"avatar_url": "https://avatars.githubusercontent.com/u/579688?v=4",
|
||||||
|
"profile": "http://www.apsis.io",
|
||||||
|
"contributions": [
|
||||||
|
"code"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"login": "stvkoch",
|
||||||
|
"name": "Steven Koch",
|
||||||
|
"avatar_url": "https://avatars.githubusercontent.com/u/14454?v=4",
|
||||||
|
"profile": "https://stvkoch.github.io.",
|
||||||
|
"contributions": [
|
||||||
|
"code"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"login": "sedwards2009",
|
||||||
|
"name": "Simon Edwards",
|
||||||
|
"avatar_url": "https://avatars.githubusercontent.com/u/6926644?v=4",
|
||||||
|
"profile": "https://github.com/sedwards2009",
|
||||||
|
"contributions": [
|
||||||
|
"code"
|
||||||
|
]
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"contributorsPerLine": 7,
|
"contributorsPerLine": 7,
|
||||||
"projectName": "nodegui",
|
"projectName": "nodegui",
|
||||||
"projectOwner": "nodegui",
|
"projectOwner": "nodegui",
|
||||||
"repoType": "github",
|
"repoType": "github",
|
||||||
"repoHost": "https://github.com"
|
"repoHost": "https://github.com",
|
||||||
|
"skipCi": true
|
||||||
}
|
}
|
||||||
|
|||||||
8
.clang-format
Normal file
8
.clang-format
Normal file
@ -0,0 +1,8 @@
|
|||||||
|
BasedOnStyle: Google
|
||||||
|
IndentWidth: 2
|
||||||
|
---
|
||||||
|
Language: Cpp
|
||||||
|
|
||||||
|
PointerAlignment: Left
|
||||||
|
AlwaysBreakAfterReturnType: None
|
||||||
|
SortIncludes: true
|
||||||
29
.eslintrc.js
Normal file
29
.eslintrc.js
Normal file
@ -0,0 +1,29 @@
|
|||||||
|
module.exports = {
|
||||||
|
extends: [
|
||||||
|
'plugin:prettier/recommended', // Enables eslint-plugin-prettier and eslint-config-prettier. This will display prettier errors as ESLint errors. Make sure this is always the last configuration in the extends array.
|
||||||
|
],
|
||||||
|
parserOptions: {
|
||||||
|
ecmaVersion: 2018, // Allows for the parsing of modern ECMAScript features
|
||||||
|
sourceType: 'module', // Allows for the use of imports
|
||||||
|
},
|
||||||
|
overrides: [
|
||||||
|
{
|
||||||
|
files: ['*.ts', '*.tsx'],
|
||||||
|
parser: '@typescript-eslint/parser', // Specifies the ESLint parser
|
||||||
|
extends: [
|
||||||
|
'plugin:@typescript-eslint/recommended', // Uses the recommended rules from the @typescript-eslint/eslint-plugin
|
||||||
|
'plugin:prettier/recommended', // Enables eslint-plugin-prettier and eslint-config-prettier. This will display prettier errors as ESLint errors. Make sure this is always the last configuration in the extends array.
|
||||||
|
],
|
||||||
|
parserOptions: {
|
||||||
|
ecmaVersion: 2018, // Allows for the parsing of modern ECMAScript features
|
||||||
|
sourceType: 'module', // Allows for the use of imports
|
||||||
|
},
|
||||||
|
rules: {
|
||||||
|
'@typescript-eslint/camelcase': 0,
|
||||||
|
'@typescript-eslint/no-var-requires': 0,
|
||||||
|
'@typescript-eslint/no-explicit-any': 0,
|
||||||
|
'@typescript-eslint/no-unused-vars': ['warn', { args: 'none' }],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
],
|
||||||
|
};
|
||||||
9
.github/FUNDING.yml
vendored
9
.github/FUNDING.yml
vendored
@ -1,9 +0,0 @@
|
|||||||
# These are supported funding model platforms
|
|
||||||
|
|
||||||
# github: [master-atul] # Replace with up to 4 GitHub Sponsors-enabled usernames e.g., [user1, user2]
|
|
||||||
patreon: # Replace with a single Patreon username
|
|
||||||
open_collective: # Replace with a single Open Collective username
|
|
||||||
tidelift: # Replace with a single Tidelift platform-name/package-name e.g., npm/babel
|
|
||||||
issuehunt: nodegui/nodegui
|
|
||||||
ko_fi: atul94 # Replace with a single Ko-fi username
|
|
||||||
custom: # Replace with a single custom sponsorship URL
|
|
||||||
32
.github/ISSUE_TEMPLATE/bug_report.md
vendored
Normal file
32
.github/ISSUE_TEMPLATE/bug_report.md
vendored
Normal file
@ -0,0 +1,32 @@
|
|||||||
|
---
|
||||||
|
name: Bug report
|
||||||
|
about: Create a report to help us improve
|
||||||
|
title: ''
|
||||||
|
labels: ''
|
||||||
|
assignees: ''
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
**Describe the bug**
|
||||||
|
A clear and concise description of what the bug is.
|
||||||
|
|
||||||
|
**To Reproduce**
|
||||||
|
Steps to reproduce the behavior:
|
||||||
|
1. Go to '...'
|
||||||
|
2. Click on '....'
|
||||||
|
3. Scroll down to '....'
|
||||||
|
4. See error
|
||||||
|
|
||||||
|
**Expected behavior**
|
||||||
|
A clear and concise description of what you expected to happen.
|
||||||
|
|
||||||
|
**Screenshots**
|
||||||
|
If applicable, add screenshots to help explain your problem.
|
||||||
|
|
||||||
|
**Desktop (please complete the following information):**
|
||||||
|
- OS: [e.g. Mac, Linux, Win]
|
||||||
|
- NodeGUI version
|
||||||
|
- OS Version
|
||||||
|
|
||||||
|
**Additional context**
|
||||||
|
Add any other context about the problem here.
|
||||||
20
.github/ISSUE_TEMPLATE/feature_request.md
vendored
Normal file
20
.github/ISSUE_TEMPLATE/feature_request.md
vendored
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
---
|
||||||
|
name: Feature request
|
||||||
|
about: Suggest an idea for this project
|
||||||
|
title: ''
|
||||||
|
labels: ''
|
||||||
|
assignees: ''
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
**Is your feature request related to a problem? Please describe.**
|
||||||
|
A clear and concise description of what the problem is. Ex. I'm always frustrated when [...]
|
||||||
|
|
||||||
|
**Describe the solution you'd like**
|
||||||
|
A clear and concise description of what you want to happen.
|
||||||
|
|
||||||
|
**Describe alternatives you've considered**
|
||||||
|
A clear and concise description of any alternative solutions or features you've considered.
|
||||||
|
|
||||||
|
**Additional context**
|
||||||
|
Add any other context or screenshots about the feature request here.
|
||||||
46
.github/workflows/docs.yml
vendored
Normal file
46
.github/workflows/docs.yml
vendored
Normal file
@ -0,0 +1,46 @@
|
|||||||
|
name: documentation
|
||||||
|
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
branches:
|
||||||
|
- master
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
gh-release:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v2
|
||||||
|
- uses: actions/setup-node@v2
|
||||||
|
with:
|
||||||
|
node-version: '16.x'
|
||||||
|
- name: Add key to allow access to repository
|
||||||
|
env:
|
||||||
|
SSH_AUTH_SOCK: /tmp/ssh_agent.sock
|
||||||
|
run: |
|
||||||
|
mkdir -p ~/.ssh
|
||||||
|
ssh-keyscan github.com >> ~/.ssh/known_hosts
|
||||||
|
echo "${{ secrets.GH_PAGES_DEPLOY }}" > ~/.ssh/id_rsa
|
||||||
|
chmod 600 ~/.ssh/id_rsa
|
||||||
|
cat <<EOT >> ~/.ssh/config
|
||||||
|
Host github.com
|
||||||
|
HostName github.com
|
||||||
|
IdentityFile ~/.ssh/id_rsa
|
||||||
|
EOT
|
||||||
|
- name: Release to GitHub Pages
|
||||||
|
env:
|
||||||
|
USE_SSH: true
|
||||||
|
GIT_USER: git
|
||||||
|
run: |
|
||||||
|
git config --global user.email "actions@gihub.com"
|
||||||
|
git config --global user.name "gh-actions"
|
||||||
|
yarn install --ignore-scripts
|
||||||
|
npm run docs
|
||||||
|
cd website
|
||||||
|
if [ -e yarn.lock ]; then
|
||||||
|
yarn install --frozen-lockfile
|
||||||
|
elif [ -e package-lock.json ]; then
|
||||||
|
npm ci
|
||||||
|
else
|
||||||
|
npm i
|
||||||
|
fi
|
||||||
|
yarn deploy
|
||||||
50
.github/workflows/latest.yml
vendored
Normal file
50
.github/workflows/latest.yml
vendored
Normal file
@ -0,0 +1,50 @@
|
|||||||
|
on:
|
||||||
|
push:
|
||||||
|
branches:
|
||||||
|
- master # Push events on master branch
|
||||||
|
jobs:
|
||||||
|
build:
|
||||||
|
strategy:
|
||||||
|
fail-fast: false
|
||||||
|
matrix:
|
||||||
|
os: [ubuntu-20.04]
|
||||||
|
runs-on: ${{ matrix.os }}
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v2
|
||||||
|
- uses: actions/setup-node@v2
|
||||||
|
with:
|
||||||
|
node-version: '16.x'
|
||||||
|
- name: Install ubuntu deps
|
||||||
|
if: contains(matrix.os, 'ubuntu-20.04')
|
||||||
|
run: sudo apt install mesa-common-dev libglu1-mesa-dev libegl1 libopengl-dev
|
||||||
|
- name: Install deps
|
||||||
|
run: npm install
|
||||||
|
- name: Build nodegui
|
||||||
|
run: npm run build
|
||||||
|
- name: Archive using npm pack
|
||||||
|
run: npm --no-git-tag-version version 0.0.0-latest-master && npm pack
|
||||||
|
- uses: actions/upload-artifact@v1
|
||||||
|
with:
|
||||||
|
name: latest-release
|
||||||
|
path: nodegui-nodegui-0.0.0-latest-master.tgz
|
||||||
|
- uses: nodegui/create-release@master
|
||||||
|
with:
|
||||||
|
token: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
code: v0.0.0-latest-master
|
||||||
|
name: Latest Master Release
|
||||||
|
body: >
|
||||||
|
Latest auto release corresponding to commit ${{github.sha}} 🔥.
|
||||||
|
To install do:
|
||||||
|
`npm install https://github.com/nodegui/nodegui/releases/download/v0.0.0-latest-master/nodegui-master.tgz`
|
||||||
|
or
|
||||||
|
`npm install http://master-release.nodegui.org`
|
||||||
|
assets: >
|
||||||
|
nodegui-nodegui-0.0.0-latest-master.tgz:nodegui-master.tgz:application/tar+gzip
|
||||||
|
recreate: true
|
||||||
|
- name: Repository Dispatch
|
||||||
|
uses: peter-evans/repository-dispatch@v1
|
||||||
|
with:
|
||||||
|
token: ${{ secrets.REPO_ACCESS_TOKEN }}
|
||||||
|
repository: nodegui/nodegui
|
||||||
|
event-type: on-demand-build
|
||||||
|
client-payload: '{"ref": "${{ github.ref }}","tag": "v0.0.0-latest-master"}'
|
||||||
97
.github/workflows/release.yml
vendored
Normal file
97
.github/workflows/release.yml
vendored
Normal file
@ -0,0 +1,97 @@
|
|||||||
|
name: Build and Deploy to npm
|
||||||
|
|
||||||
|
on:
|
||||||
|
release:
|
||||||
|
types: [published]
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
precompile:
|
||||||
|
if: contains(github.event.release.tag_name, 'v0.0.0-latest-master') == false
|
||||||
|
strategy:
|
||||||
|
matrix:
|
||||||
|
include:
|
||||||
|
- os: ubuntu-20.04
|
||||||
|
platform: linux
|
||||||
|
arch: x64
|
||||||
|
- os: windows-latest
|
||||||
|
platform: win32
|
||||||
|
arch: x64
|
||||||
|
- os: macos-latest
|
||||||
|
platform: darwin
|
||||||
|
arch: x64
|
||||||
|
runs-on: ${{ matrix.os }}
|
||||||
|
env:
|
||||||
|
ARCHIVE_FILENAME: nodegui-binary-${{github.event.release.tag_name}}-${{ matrix.platform }}-${{ matrix.arch }}.tar.gz
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v2
|
||||||
|
|
||||||
|
- uses: actions/setup-node@v2
|
||||||
|
with:
|
||||||
|
node-version: '16.x'
|
||||||
|
|
||||||
|
- name: Install ubuntu deps
|
||||||
|
if: contains(matrix.os, 'ubuntu-20.04')
|
||||||
|
run: sudo apt install mesa-common-dev libglu1-mesa-dev libegl1 libopengl-dev
|
||||||
|
|
||||||
|
- name: Install deps
|
||||||
|
run: npm install
|
||||||
|
|
||||||
|
- name: Build nodegui
|
||||||
|
run: npm run build
|
||||||
|
env:
|
||||||
|
CMAKE_BUILD_PARALLEL_LEVEL: 8
|
||||||
|
|
||||||
|
- name: Compress files
|
||||||
|
if: ${{!contains(matrix.os, 'windows-latest')}}
|
||||||
|
uses: a7ul/tar-action@v1.0.2
|
||||||
|
id: compress
|
||||||
|
with:
|
||||||
|
command: c
|
||||||
|
cwd: ./build/Release
|
||||||
|
files: |
|
||||||
|
./nodegui_core.node
|
||||||
|
outPath: ${{ env.ARCHIVE_FILENAME }}
|
||||||
|
|
||||||
|
- name: Compress files (Windows)
|
||||||
|
if: contains(matrix.os, 'windows-latest')
|
||||||
|
uses: a7ul/tar-action@v1.0.2
|
||||||
|
id: compress-windows
|
||||||
|
with:
|
||||||
|
command: c
|
||||||
|
cwd: ./build/Release
|
||||||
|
files: |
|
||||||
|
./nodegui_core.node
|
||||||
|
./nodegui_core.lib
|
||||||
|
./nodegui_core.exp
|
||||||
|
outPath: ${{ env.ARCHIVE_FILENAME }}
|
||||||
|
|
||||||
|
- uses: actions/upload-artifact@v1
|
||||||
|
with:
|
||||||
|
name: ${{ env.ARCHIVE_FILENAME }}
|
||||||
|
path: ${{ env.ARCHIVE_FILENAME }}
|
||||||
|
|
||||||
|
- name: Upload release binaries
|
||||||
|
uses: alexellis/upload-assets@0.2.2
|
||||||
|
env:
|
||||||
|
GITHUB_TOKEN: ${{ github.token }}
|
||||||
|
with:
|
||||||
|
asset_paths: '["${{ env.ARCHIVE_FILENAME }}"]'
|
||||||
|
|
||||||
|
publish-npm-package:
|
||||||
|
needs: precompile
|
||||||
|
if: contains(github.event.release.tag_name, 'v0.0.0-latest-master') == false
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v2
|
||||||
|
- uses: actions/setup-node@v2
|
||||||
|
with:
|
||||||
|
node-version: '16.x'
|
||||||
|
- name: Install ubuntu deps
|
||||||
|
run: sudo apt update
|
||||||
|
- run: sudo apt install mesa-common-dev libglu1-mesa-dev
|
||||||
|
- run: npm install
|
||||||
|
- run: echo '//registry.npmjs.org/:_authToken=${NPM_TOKEN}' >> .npmrc
|
||||||
|
- run: npm publish --access=public
|
||||||
|
env:
|
||||||
|
NPM_TOKEN: ${{ secrets.NPM_PUBLISH_TOKEN }}
|
||||||
|
CMAKE_BUILD_PARALLEL_LEVEL: 8
|
||||||
30
.github/workflows/test.yml
vendored
Normal file
30
.github/workflows/test.yml
vendored
Normal file
@ -0,0 +1,30 @@
|
|||||||
|
on:
|
||||||
|
pull_request:
|
||||||
|
types: [opened, synchronize, reopened]
|
||||||
|
jobs:
|
||||||
|
build:
|
||||||
|
strategy:
|
||||||
|
fail-fast: false
|
||||||
|
matrix:
|
||||||
|
os: [ubuntu-20.04, windows-latest, macos-latest]
|
||||||
|
runs-on: ${{ matrix.os }}
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v2
|
||||||
|
- uses: actions/setup-node@v2
|
||||||
|
with:
|
||||||
|
node-version: '16.x'
|
||||||
|
- name: Install ubuntu deps
|
||||||
|
if: contains(matrix.os, 'ubuntu-20.04')
|
||||||
|
run: sudo apt install mesa-common-dev libglu1-mesa-dev libegl1 libopengl-dev
|
||||||
|
- name: Install deps
|
||||||
|
run: npm install
|
||||||
|
- name: Build nodegui
|
||||||
|
run: npm run build
|
||||||
|
env:
|
||||||
|
CMAKE_BUILD_PARALLEL_LEVEL: 8
|
||||||
|
- name: Run tests
|
||||||
|
run: npm run test
|
||||||
|
- name: Run linters for cpp
|
||||||
|
run: npm run lint:cpp
|
||||||
|
- name: Run linters for ts
|
||||||
|
run: npm run lint:ts
|
||||||
6
.gitignore
vendored
6
.gitignore
vendored
@ -4,3 +4,9 @@ build
|
|||||||
dist
|
dist
|
||||||
.vscode
|
.vscode
|
||||||
.cache
|
.cache
|
||||||
|
coverage
|
||||||
|
.DS_Store
|
||||||
|
/.idea/
|
||||||
|
/prebuilds
|
||||||
|
miniqt
|
||||||
|
cmake-build-debug
|
||||||
|
|||||||
11
.npmignore
Normal file
11
.npmignore
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
*.test.*
|
||||||
|
build
|
||||||
|
website
|
||||||
|
.*
|
||||||
|
!dist
|
||||||
|
extras/*
|
||||||
|
!extras/legal
|
||||||
|
src/*
|
||||||
|
!src/cpp
|
||||||
|
prebuilds/*
|
||||||
|
miniqt/
|
||||||
7
.prettierrc.js
Normal file
7
.prettierrc.js
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
module.exports = {
|
||||||
|
semi: true,
|
||||||
|
trailingComma: 'all',
|
||||||
|
singleQuote: true,
|
||||||
|
printWidth: 120,
|
||||||
|
tabWidth: 4,
|
||||||
|
};
|
||||||
283
CMakeLists.txt
Normal file
283
CMakeLists.txt
Normal file
@ -0,0 +1,283 @@
|
|||||||
|
cmake_minimum_required(VERSION 3.1)
|
||||||
|
|
||||||
|
include(./config/common.cmake)
|
||||||
|
include(./config/qt.cmake)
|
||||||
|
include(./config/napi.cmake)
|
||||||
|
include(./config/qode.cmake)
|
||||||
|
|
||||||
|
# ---------------------------------------
|
||||||
|
|
||||||
|
set(CORE_WIDGETS_ADDON "nodegui_core")
|
||||||
|
|
||||||
|
project(${CORE_WIDGETS_ADDON})
|
||||||
|
|
||||||
|
set(CMAKE_BUILD_WITH_INSTALL_RPATH ON)
|
||||||
|
|
||||||
|
# Note: CMake+moc also use this list when finding files which `moc` applied.
|
||||||
|
|
||||||
|
add_library(${CORE_WIDGETS_ADDON} SHARED
|
||||||
|
"${CMAKE_JS_SRC}"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/main.cpp"
|
||||||
|
# core internals
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/Extras/Utils/nutils.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/core/Integration/integration.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/core/FlexLayout/flexutils.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/core/FlexLayout/flexlayout.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/core/FlexLayout/flexitem.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/core/YogaWidget/nodestyle.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/core/Events/eventsmap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/core/Events/eventwidget.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/core/YogaWidget/yogawidget.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/core/WrapperCache/wrappercache.h"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/core/WrapperCache/wrappercache.cpp"
|
||||||
|
# core deps
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/deps/yoga/log.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/deps/yoga/Utils.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/deps/yoga/YGConfig.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/deps/yoga/YGEnums.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/deps/yoga/YGLayout.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/deps/yoga/YGNode.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/deps/yoga/YGNodePrint.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/deps/yoga/YGStyle.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/deps/yoga/YGValue.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/deps/yoga/Yoga.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/deps/yoga/event/event.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/deps/yoga/internal/experiments.cpp"
|
||||||
|
# wrapped cpps
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtGui/QBrush/qbrush_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtGui/QPen/qpen_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtGui/QColor/qcolor_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtGui/QApplication/qapplication_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtGui/QClipboard/qclipboard_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtGui/QDrag/qdrag_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtGui/QEvent/QKeyEvent/qkeyevent_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtGui/QEvent/QMouseEvent/qmouseevent_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtGui/QEvent/QMoveEvent/qmoveevent_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtGui/QEvent/QWheelEvent/qwheelevent_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtGui/QEvent/QNativeGestureEvent/qnativegestureevent_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtGui/QEvent/QPaintEvent/qpaintevent_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtGui/QEvent/QTabletEvent/qtabletevent_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtGui/QEvent/QTimerEvent/qtimerevent_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtGui/QEvent/QDropEvent/qdropevent_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtGui/QEvent/QDragMoveEvent/qdragmoveevent_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtGui/QEvent/QDragLeaveEvent/qdragleaveevent_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtGui/QEvent/QResizeEvent/qresizeevent_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtGui/QEvent/QInputMethodEvent/qinputmethodevent_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtGui/QEvent/QInputMethodQueryEvent/qinputmethodqueryevent_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtGui/QFontDatabase/qfontdatabase_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtGui/QFontMetrics/qfontmetrics_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtGui/QFontMetricsF/qfontmetricsf_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtGui/QPicture/qpicture_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtGui/QPixmap/qpixmap_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtGui/QIcon/qicon_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtGui/QImage/qimage_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtGui/QFont/qfont_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtGui/QCursor/qcursor_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtGui/QKeySequence/qkeysequence_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtGui/QMovie/qmovie_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtGui/QPalette/qpalette_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtGui/QScreen/qscreen_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtGui/QStyle/qstyle_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtGui/QWindow/qwindow_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtCore/QAbstractItemModel/qabstractitemmodel_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtCore/QDate/qdate_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtCore/QDateTime/qdatetime_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtCore/QItemSelectionModel/qitemselectionmodel_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtCore/QModelIndex/qmodelindex_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtCore/QObject/qobject_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtCore/QVariant/qvariant_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtCore/QSize/qsize_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtCore/QSizeF/qsizef_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtCore/QSettings/qsettings_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtCore/QRect/qrect_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtCore/QRectF/qrectf_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtCore/QPoint/qpoint_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtCore/QPointF/qpointf_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtCore/QTime/qtime_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtCore/QUrl/qurl_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtCore/QMimeData/qmimedata_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QWidget/qwidget_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QDialog/qdialog_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QBoxLayout/qboxlayout_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QCalendarWidget/qcalendarwidget_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QColorDialog/qcolordialog_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QComboBox/qcombobox_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QDateEdit/qdateedit_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QDateTimeEdit/qdatetimeedit_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QDoubleSpinBox/qdoublespinbox_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QErrorMessage/qerrormessage_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QFileDialog/qfiledialog_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QFontDialog/qfontdialog_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QFrame/qframe_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QGraphicsBlurEffect/qgraphicsblureffect_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QGraphicsDropShadowEffect/qgraphicsdropshadoweffect_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QHeaderView/qheaderview_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QListView/qlistview_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QListWidget/qlistwidget_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QListWidgetItem/qlistwidgetitem_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QTableView/qtableview_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QTableWidget/qtablewidget_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QTableWidgetItem/qtablewidgetitem_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QPainter/qpainter_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QPainterPath/qpainterpath_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QGridLayout/qgridlayout_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QDial/qdial_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QLabel/qlabel_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QLayout/qlayout_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QLCDNumber/qlcdnumber_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QMainWindow/qmainwindow_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QPushButton/qpushbutton_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QToolButton/qtoolbutton_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QSpinBox/qspinbox_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QCheckBox/qcheckbox_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QProgressBar/qprogressbar_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QProgressDialog/qprogressdialog_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QRadioButton/qradiobutton_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QStackedWidget/qstackedwidget_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QTabBar/qtabbar_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QTabWidget/qtabwidget_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QLineEdit/qlineedit_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QPlainTextEdit/qplaintextedit_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QScrollArea/qscrollarea_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QScrollBar/qscrollbar_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QSystemTrayIcon/qsystemtrayicon_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QAction/qaction_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QShortcut/qshortcut_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QMenuBar/qmenubar_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QMenu/qmenu_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QGroupBox/qgroupbox_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QSlider/qslider_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QTimeEdit/qtimeedit_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QTreeWidget/qtreewidget_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QTreeWidgetItem/qtreewidgetitem_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QMessageBox/qmessagebox_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QInputDialog/qinputdialog_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QButtonGroup/qbuttongroup_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QStatusBar/qstatusbar_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/core/FlexLayout/flexlayout_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QTextBrowser/qtextbrowser_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QTextEdit/qtextedit_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QStandardItemModel/qstandarditemmodel_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QStandardItem/qstandarditem_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QSvgWidget/qsvgwidget_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QStyleFactory/qstylefactory_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QSplitter/qsplitter_wrap.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/QtWidgets/QAbstractItemDelegate/qabstractitemdelegate_wrap.cpp"
|
||||||
|
# Custom widgets (include them for automoc since they contain Q_OBJECT)
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtCore/QAbstractItemModel/nabstractitemmodel.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtCore/QItemSelectionModel/nitemselectionmodel.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtCore/QObject/nobject.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/core/FlexLayout/flexlayout.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtGui/QMovie/nmovie.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtGui/QApplication/napplication.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtGui/QDrag/ndrag.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QWidget/nwidget.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QDialog/ndialog.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QLabel/nlabel.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QCalendarWidget/ncalendarwidget.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QCheckBox/ncheckbox.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QColorDialog/ncolordialog.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QDateEdit/ndateedit.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QDateTimeEdit/ndatetimeedit.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QDial/ndial.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QDoubleSpinBox/ndoublespinbox.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QErrorMessage/nerrormessage.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QFileDialog/nfiledialog.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QFontDialog/nfontdialog.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QFrame/nframe.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QGraphicsBlurEffect/ngraphicsblureffect.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QGraphicsDropShadowEffect/ngraphicsdropshadoweffect.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QLCDNumber/nlcdnumber.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QListView/nlistview.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QListWidget/nlistwidget.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QTableView/ntableview.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QTableWidget/ntablewidget.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QLineEdit/nlineedit.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QMainWindow/nmainwindow.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QProgressBar/nprogressbar.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QProgressDialog/nprogressdialog.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QPushButton/npushbutton.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QMessageBox/nmessagebox.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QInputDialog/ninputdialog.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QToolButton/ntoolbutton.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QSpinBox/nspinbox.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QRadioButton/nradiobutton.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QPlainTextEdit/nplaintextedit.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QScrollArea/nscrollarea.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QScrollBar/nscrollbar.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QStackedWidget/nstackedwidget.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QTabBar/ntabbar.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QTabWidget/ntabwidget.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QSystemTrayIcon/nsystemtrayicon.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QAction/naction.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QShortcut/nshortcut.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QMenuBar/nmenubar.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QMenu/nmenu.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QLayout/nlayout.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QGridLayout/ngridlayout.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QBoxLayout/nboxlayout.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QComboBox/ncombobox.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QGroupBox/ngroupbox.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QSlider/nslider.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QTimeEdit/ntimeedit.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QTreeWidget/ntreewidget.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QButtonGroup/nbuttongroup.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QStatusBar/nstatusbar.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QStandardItemModel/nstandarditemmodel.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QTextBrowser/ntextbrowser.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QTextEdit/ntextedit.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QSvgWidget/nsvgwidget.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QHeaderView/nheaderview.hpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/QtWidgets/QSplitter/nsplitter.hpp"
|
||||||
|
|
||||||
|
# Test
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui/test/CacheTestQObject.h"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/test/CacheTestQObject.cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/lib/test/cachetestqobject_wrap.cpp"
|
||||||
|
)
|
||||||
|
|
||||||
|
AddCommonConfig(${CORE_WIDGETS_ADDON})
|
||||||
|
AddQtSupport(${CORE_WIDGETS_ADDON})
|
||||||
|
AddNapiSupport(${CORE_WIDGETS_ADDON})
|
||||||
|
AddQodeSupport(${CORE_WIDGETS_ADDON})
|
||||||
|
|
||||||
|
target_include_directories(${CORE_WIDGETS_ADDON} PRIVATE
|
||||||
|
"${CMAKE_JS_INC}"
|
||||||
|
"${PROJECT_SOURCE_DIR}"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/deps"
|
||||||
|
"${PROJECT_SOURCE_DIR}/src/cpp/include/nodegui"
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
if (WIN32)
|
||||||
|
add_definitions(/bigobj)
|
||||||
|
target_compile_definitions(${CORE_WIDGETS_ADDON} PRIVATE
|
||||||
|
ENABLE_DLL_EXPORT=1
|
||||||
|
)
|
||||||
|
target_compile_options(${CORE_WIDGETS_ADDON} PRIVATE "/MP")
|
||||||
|
endif()
|
||||||
|
|
||||||
|
if (UNIX AND NOT APPLE)
|
||||||
|
# Set the RPATH in the binary to a relative one to allow it to find our Qt
|
||||||
|
# libraries regardless of where the base installation directory is.
|
||||||
|
file(RELATIVE_PATH QT_LIBRARY_REL_PATH "${CMAKE_BINARY_DIR}/Release" "${QT_CMAKE_HOME_DIR}/../..")
|
||||||
|
set_target_properties(${CORE_WIDGETS_ADDON} PROPERTIES INSTALL_RPATH "\$ORIGIN/${QT_LIBRARY_REL_PATH}")
|
||||||
|
set_target_properties(${CORE_WIDGETS_ADDON} PROPERTIES BUILD_WITH_INSTALL_RPATH TRUE)
|
||||||
|
endif()
|
||||||
|
|
||||||
|
if (APPLE)
|
||||||
|
set(CMAKE_MACOSX_RPATH ON)
|
||||||
|
file(RELATIVE_PATH QT_LIBRARY_REL_PATH "${CMAKE_BINARY_DIR}/Release" "${QT_CMAKE_HOME_DIR}/../..")
|
||||||
|
set_target_properties(${CORE_WIDGETS_ADDON} PROPERTIES INSTALL_RPATH "@loader_path/${QT_LIBRARY_REL_PATH}")
|
||||||
|
endif()
|
||||||
|
|
||||||
|
target_link_libraries(${CORE_WIDGETS_ADDON} PRIVATE
|
||||||
|
"${CMAKE_JS_LIB}"
|
||||||
|
Qt6::Widgets
|
||||||
|
Qt6::Core
|
||||||
|
Qt6::Gui
|
||||||
|
Qt6::Svg
|
||||||
|
Qt6::SvgWidgets
|
||||||
|
)
|
||||||
323
README.md
323
README.md
@ -1,13 +1,21 @@
|
|||||||
# NodeGUI
|
# NodeGui
|
||||||
[](#contributors)
|
|
||||||
|
|
||||||
Build **performant**, **native** and **cross-platform** desktop applications with **JavaScript** + powerful **CSS like styling**.🚀
|
[](#contributors)
|
||||||
|
[](https://changelog.com/jsparty/96)
|
||||||
|
[](https://github.com/nodegui/nodegui/actions)
|
||||||
|
[](https://github.com/nodegui/nodegui/discussions)
|
||||||
|
|
||||||
NodeGUI is powered by **Qt5** 💚 which makes it CPU and memory efficient as compared to other chromium based solutions like electron.
|
Build **performant**, **native** and **cross-platform** desktop applications with **Node.js** and **CSS like styling**.🚀
|
||||||
|
|
||||||
If you are looking for React based version, check out: [React NodeGUI](https://github.com/nodegui/react-nodegui).
|
NodeGUI is powered by **Qt6** 💚 which makes it CPU and memory efficient as compared to other chromium based solutions like Electron.
|
||||||
|
|
||||||
Visit: https://nodegui.github.io/nodegui for docs.
|
<img src="https://upload.wikimedia.org/wikipedia/commons/thumb/a/a7/React-icon.svg/1024px-React-icon.svg.png" alt="" width="25"> If you are looking for **React** based version, check out: **[React NodeGUI](https://github.com/nodegui/react-nodegui)**.
|
||||||
|
|
||||||
|
<img src="https://vuejs.org/images/logo.png" alt="" width="25" /> If you are looking for **Vue** based version, check out: **[Vue NodeGUI](https://github.com/nodegui/vue-nodegui)**.
|
||||||
|
|
||||||
|
<img src="https://github.com/sveltejs/branding/raw/master/svelte-logo.png" alt="" width="25" /> If you are looking for **Svelte** based version, check out: **[Svelte NodeGUI](https://github.com/nodegui/svelte-nodegui)**
|
||||||
|
|
||||||
|
Visit https://nodegui.github.io/nodegui for docs.
|
||||||
|
|
||||||
<img alt="logo" src="https://github.com/nodegui/nodegui/raw/master/extras/logo/nodegui.png" height="200" />
|
<img alt="logo" src="https://github.com/nodegui/nodegui/raw/master/extras/logo/nodegui.png" height="200" />
|
||||||
|
|
||||||
@ -22,9 +30,7 @@ Visit: https://nodegui.github.io/nodegui for docs.
|
|||||||
<div style="display:inline; margin: 0 auto;"><img alt="kitchen" src="https://github.com/nodegui/nodegui/raw/master/extras/assets/kitchen.png" height="280" /><img alt="demo_mac" src="https://github.com/nodegui/examples/raw/master/react-nodegui/weather-app-widget/weather_widget_mac.png" height="280" /><img alt="demo_win" src="https://github.com/nodegui/examples/raw/master/react-nodegui/image-view/image_view_win.jpg" height="280" />
|
<div style="display:inline; margin: 0 auto;"><img alt="kitchen" src="https://github.com/nodegui/nodegui/raw/master/extras/assets/kitchen.png" height="280" /><img alt="demo_mac" src="https://github.com/nodegui/examples/raw/master/react-nodegui/weather-app-widget/weather_widget_mac.png" height="280" /><img alt="demo_win" src="https://github.com/nodegui/examples/raw/master/react-nodegui/image-view/image_view_win.jpg" height="280" />
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
**More screenshots?**
|
### Code Examples
|
||||||
|
|
||||||
### More Examples:
|
|
||||||
|
|
||||||
https://github.com/nodegui/examples
|
https://github.com/nodegui/examples
|
||||||
|
|
||||||
@ -32,46 +38,222 @@ https://github.com/nodegui/examples
|
|||||||
|
|
||||||
## Features
|
## Features
|
||||||
|
|
||||||
- 🧬 Cross platform. Should work on major Linux flavours, Windows and MacOS.
|
- 🧬 **Cross platform.** Works on major Linux flavours, Windows, and MacOS.
|
||||||
- 📉 Low CPU and memory footprint. Current CPU stays at 0% on idle and memory usage is under 20mb for a hello world program.
|
- 📉 **Low CPU and memory** footprint. Current CPU stays at 0% on idle and memory usage is under 20MB for a Hello World program.
|
||||||
- 💅 Styling with CSS (includes actual cascading). Also has full support for Flexbox layout (thanks to Yoga).
|
- 💅 **Styling with CSS** (includes actual cascading). Also has full support for Flexbox layout (thanks to Yoga).
|
||||||
- ✅ Complete Nodejs api support (Currently runs on Node v12.x - and is easily upgradable). Hence has access to all nodejs compatible npm modules.
|
- ✅ **Complete Nodejs API support** (Currently runs on Node v16.x - and is easily upgradable). Hence has access to all Nodejs compatible NPM modules.
|
||||||
- 🎪 Native widget event listener support. Supports all events available from Qt / NodeJs.
|
- 🎪 **Native widget event listener support.** Supports all events available from Qt / NodeJs.
|
||||||
- 💸 Can be used for Commercial applications.
|
- 💸 **Can be used for Commercial applications.**
|
||||||
- 🕵️♂️ Good Devtools support.
|
- 🕵️♂️ **Good Devtools support.**
|
||||||
- 📚 Good documentation and website.
|
- 📚 **Good documentation and website.**
|
||||||
- 🧙♂️ Good documentation for contributors.
|
- 🧙♂️ **Good documentation for contributors.**
|
||||||
- 🦹🏻♀️ Good support for dark mode (Thanks to QT).
|
- 🦹🏻♀️ **Good support for dark mode (Thanks to Qt).**
|
||||||
- 🏅First class Typescript support. (Works on regular JS projects too 😉).
|
- 🏅**First class Typescript support.** (Works on regular JS projects too 😉).
|
||||||
|
|
||||||
## Current focus:
|
|
||||||
|
|
||||||
- [ ] (Partial support is present) Easily exstensible for creating custom native widgets (like react native).
|
|
||||||
- [ ] (Partial) Should have a decent list of stylable native widgets.
|
|
||||||
- [ ] Easy build and packaging process.
|
|
||||||
|
|
||||||
## Getting Started
|
## Getting Started
|
||||||
|
|
||||||
- Check out [nodegui-starter](https://github.com/nodegui/nodegui-starter) to get up and running with your own React NodeGUI app!
|
- Check out [nodegui-starter](https://github.com/nodegui/nodegui-starter) to get up and running with your own React NodeGUI app!
|
||||||
- Read through the [docs](https://nodegui.github.io/nodegui).
|
- Read through the [docs](https://nodegui.github.io/nodegui).
|
||||||
|
- Checkout the examples: https://github.com/nodegui/examples .
|
||||||
|
- [Tutorial: Build a native Meme Search Desktop app with Javascript (NodeGui) and Giphy API](https://www.sitepoint.com/build-native-desktop-gif-searcher-app-using-nodegui/)
|
||||||
|
|
||||||
|
|
||||||
|
## Installation
|
||||||
|
|
||||||
|
To install latest stable release:
|
||||||
|
|
||||||
|
```
|
||||||
|
npm install @nodegui/nodegui
|
||||||
|
```
|
||||||
|
|
||||||
|
To install the latest version available on master branch:
|
||||||
|
|
||||||
|
```
|
||||||
|
npm install https://github.com/nodegui/nodegui/releases/download/v0.0.0-latest-master/nodegui-master.tgz
|
||||||
|
```
|
||||||
|
|
||||||
|
or a shorter version:
|
||||||
|
|
||||||
|
```
|
||||||
|
npm i http://master-release.nodegui.org
|
||||||
|
```
|
||||||
|
|
||||||
|
If the installation fails to download the Qt binaries, a mirror can be used by setting the following environment variable and running the install command again:
|
||||||
|
|
||||||
|
```sh
|
||||||
|
QT_LINK_MIRROR=<alternative domain> # eg. QT_LINK_MIRROR=https://qt-mirror.dannhauer.de
|
||||||
|
|
||||||
|
npm install @nodegui/nodegui
|
||||||
|
```
|
||||||
|
|
||||||
|
See [FAQs](https://github.com/nodegui/nodegui/tree/master/website/docs/faq.md#why-does-installation-fail-at-minimal-qt-setup) for more details.
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
## Community guides
|
||||||
|
|
||||||
|
- [Tutorial: Build a native Meme Search Desktop app with Javascript (NodeGui) and Giphy API](https://www.sitepoint.com/build-native-desktop-gif-searcher-app-using-nodegui/)
|
||||||
|
- https://blog.logrocket.com/electron-alternatives-exploring-nodegui-and-react-nodegui/ - Electron alternatives: Exploring NodeGUI and React NodeGUI by [Siegfried Grimbeek](https://blog.logrocket.com/author/siegfriedgrimbeek/).
|
||||||
|
- https://hibbard.eu/node-gui/ - Excellent guide from [James Hibbard](https://github.com/jameshibbard).
|
||||||
|
|
||||||
|
## Talks/Podcasts
|
||||||
|
|
||||||
|
- [NodeGui and React NodeGui at KarmaJS Nov 2019 meetup: https://www.youtube.com/watch?v=8jH5gaEEDv4](https://www.youtube.com/watch?v=8jH5gaEEDv4)
|
||||||
|
|
||||||
|
- <audio data-theme="night" data-src="https://changelog.com/jsparty/96/embed" src="https://cdn.changelog.com/uploads/jsparty/96/js-party-96.mp3" preload="none" class="changelog-episode" controls></audio><p><a href="https://changelog.com/jsparty/96">JS Party 96: Performant Node desktop apps with NodeGUI</a> – Listen on <a href="https://changelog.com/">Changelog.com</a></p>
|
||||||
|
|
||||||
## Docs for contributing
|
## Docs for contributing
|
||||||
|
|
||||||
Looking to contribute? If you wish to implement a new widget/add more features and need help understanding the codebase, you can start here: [Contributing developer docs](https://github.com/nodegui/nodegui/tree/master/docs/development).
|
```
|
||||||
|
It is easier than you think, try it
|
||||||
|
```
|
||||||
|
|
||||||
## Building
|
Looking to contribute? If you wish to implement a new widget/add more features and need help understanding the codebase, you can start here: [Contributing developer docs](https://github.com/nodegui/nodegui/tree/master/website/docs/development).
|
||||||
|
|
||||||
`npm run build [--qt_home_dir=/path/to/qt]`
|
Please read https://github.com/nodegui/.github/blob/master/CONTRIBUTING.md
|
||||||
|
|
||||||
### LICENSE
|
|
||||||
|
## Building from source
|
||||||
|
|
||||||
|
This section is useful for those wanting to work on NodeGui itself.
|
||||||
|
|
||||||
|
### Required tooling
|
||||||
|
|
||||||
|
NodeGui requires CMake and Compilation Tools as it is a wrapper for a native C++ widget toolkit QT.
|
||||||
|
Detailed instructions here: https://www.sitepoint.com/build-native-desktop-gif-searcher-app-using-nodegui/
|
||||||
|
|
||||||
|
TL;DR:
|
||||||
|
|
||||||
|
**MacOS**
|
||||||
|
|
||||||
|
```
|
||||||
|
brew install cmake
|
||||||
|
brew install make
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
|
**Windows**
|
||||||
|
|
||||||
|
https://cmake.org/download/
|
||||||
|
|
||||||
|
**Linux (Debian/Ubuntu)**
|
||||||
|
|
||||||
|
```
|
||||||
|
sudo apt-get install pkg-config build-essential
|
||||||
|
sudo apt-get install cmake make
|
||||||
|
sudo apt-get install mesa-common-dev libglu1-mesa-dev
|
||||||
|
```
|
||||||
|
|
||||||
|
**Linux (Fedora/RHEL/CentOS)**
|
||||||
|
|
||||||
|
```
|
||||||
|
sudo dnf groupinstall "Development Tools" "Development Libraries"
|
||||||
|
sudo dnf groupinstall "C Development Tools and Libraries"
|
||||||
|
sudo dnf install mesa-libGL mesa-libGL-devel
|
||||||
|
```
|
||||||
|
|
||||||
|
### Building
|
||||||
|
|
||||||
|
Once you have cloned this git repository, run this to build:
|
||||||
|
|
||||||
|
```
|
||||||
|
npm install
|
||||||
|
npm run build
|
||||||
|
```
|
||||||
|
|
||||||
|
The last step will takes some time to run as it builds the C++ binaries at the core of NodeGui.
|
||||||
|
|
||||||
|
|
||||||
|
### Using your own custom Qt installation (Optional)
|
||||||
|
|
||||||
|
**Compiling Qt from source**
|
||||||
|
|
||||||
|
You will need to download and install Qt from source since there are no binaries from Qt for M1 yet.
|
||||||
|
|
||||||
|
(https://www.reddit.com/r/QtFramework/comments/ll58wg/how_to_build_qt_creator_for_macos_arm64_a_guide/)
|
||||||
|
|
||||||
|
```
|
||||||
|
git clone git://code.qt.io/qt/qt5.git
|
||||||
|
cd qt5
|
||||||
|
git checkout 5.15
|
||||||
|
|
||||||
|
./init-repository --module-subset=essential -f
|
||||||
|
git submodule init qtsvg
|
||||||
|
git submodule update qtsvg
|
||||||
|
|
||||||
|
cd ..
|
||||||
|
mkdir qt5-5.15-macOS-release
|
||||||
|
cd qt5-5.15-macOS-release
|
||||||
|
|
||||||
|
../qt5/configure -release QMAKE_APPLE_DEVICE_ARCHS=arm64 -opensource -confirm-license -nomake examples -nomake tests -skip qt3d -skip webengine -skip qtactiveqt -skip qtcanvas3d -skip qtdeclarative -skip qtdatavis3d -skip qtdoc -skip qtgamepad -skip qtcharts -skip qtgraphicaleffects -skip qtlocation -skip qtpurchasing -skip qtquickcontrols -skip qtquickcontrols2 -skip qtremoteobjects -skip qtscxml -skip qtsensors -skip qtserialbus -skip qtserialport -skip qtspeech -skip qtvirtualkeyboard -skip qtscript
|
||||||
|
|
||||||
|
make -j15
|
||||||
|
|
||||||
|
make install
|
||||||
|
```
|
||||||
|
|
||||||
|
This should install Qt into something like this `/usr/local/Qt-5.15.3` (your directory can change. This will be displayed when running make)
|
||||||
|
|
||||||
|
**Pointing nodegui to use your custom Qt installation**
|
||||||
|
|
||||||
|
Now just set `export QT_INSTALL_DIR=<your qt path>` . In the above example it would look something like this `export QT_INSTALL_DIR=/usr/local/Qt-5.15.3`. Add this in your .zshrc or .bashrc so that you dont need to repeat this process again.
|
||||||
|
|
||||||
|
Now just `rm -rf node_modules` and do `npm install` again.
|
||||||
|
|
||||||
|
The logs should say something like `CustomQt detected at <your qt path>. Hence, skipping Mini Qt installation`.
|
||||||
|
|
||||||
|
|
||||||
|
### Running example programs
|
||||||
|
|
||||||
|
A number of small programs to show different parts of the Qt API are in `src/examples`. They can be run directly with:
|
||||||
|
|
||||||
|
```
|
||||||
|
npm run qode dist/examples/modelview_1_readonly.js
|
||||||
|
```
|
||||||
|
|
||||||
|
## Updating docs
|
||||||
|
|
||||||
|
`npm run docs`
|
||||||
|
|
||||||
|
then followed by:
|
||||||
|
|
||||||
|
`cd website && GIT_USER=<your_git_username> yarn deploy`
|
||||||
|
|
||||||
|
## Funding
|
||||||
|
|
||||||
|
NodeGui is an open source project and requires your support. If you like this project, please consider supporting my work by clicking on the Sponsor button on this Github repository or via Ko-Fi.
|
||||||
|
Alternatively, Issues on NodeGui can be funded by anyone via Issuehunt and the amount will be distributed to respective contributors.
|
||||||
|
|
||||||
|
<p>
|
||||||
|
<a href='https://ko-fi.com/E1E510AV9' target='_blank'><img height='36' style='border:0px;height:36px;' src='https://az743702.vo.msecnd.net/cdn/kofi4.png?v=2' border='0' alt='Buy Me a Coffee at ko-fi.com' /></a> <a href="https://issuehunt.io/r/nodegui/nodegui"><img alt="issuehunt" src="https://github.com/BoostIO/issuehunt-materials/raw/master/v1/issuehunt-button-v1.svg?sanitize=true" height="30px" /></a>
|
||||||
|
</p>
|
||||||
|
|
||||||
|
## Special Thanks
|
||||||
|
|
||||||
|
- [Logo: Thanks to Vishwas Shetty from the Noun Project.](https://github.com/nodegui/nodegui/blob/master/extras/legal/logo/thanks.md)
|
||||||
|
|
||||||
|
## Code of Conduct
|
||||||
|
|
||||||
|
https://github.com/nodegui/.github/blob/master/CODE_OF_CONDUCT.md
|
||||||
|
|
||||||
|
## License
|
||||||
|
|
||||||
MIT
|
MIT
|
||||||
|
|
||||||
## Note
|
## Backers 🚀
|
||||||
|
|
||||||
> Since we do not in any way modify the code of Qt and only link to it dynamically, I believe we are in compliance with the LGPL license requirements of QT. And hence this library can be licensed under its own License (for which we have chosen MIT License).
|
Thanks goes to these wonderful people.
|
||||||
> The links to QT source code and appropriate license notices are attached. We try our best to abide by the software licenses and any non compliance is not by will. If there is some discrepancy please let us know in the issues and we will try and fix it up.
|
|
||||||
> If you follow the recommended build steps and do not statically link QT libraries on your own you are safe to use this library for commerical purposes (provided you abide by MIT License).
|
<!-- prettier-ignore -->
|
||||||
|
<table>
|
||||||
|
<tr>
|
||||||
|
<td align="center"><a href="https://github.com/yazla"><img src="https://avatars1.githubusercontent.com/u/681281?s=460&v=4" width="100px;" alt="Yuriy Yazlovytskyy"/><br /><sub><b>Yuriy Yazlovytskyy</b></sub></a></td>
|
||||||
|
<td align="center"><a href="https://github.com/johnsusek"><img src="https://avatars1.githubusercontent.com/u/611996?s=460&v=4" width="100px;" alt="John Susek"/><br /><sub><b>John Susek</b></sub></a></td>
|
||||||
|
<td align="center"><a href="https://github.com/Spharax"><img src="https://avatars2.githubusercontent.com/u/2892381?s=460&v=4" width="100px;" alt="Marc Dijoux"/><br /><sub><b>Marc Dijoux</b></sub></a></td>
|
||||||
|
<td align="center"><a href="https://github.com/Qard"><img src="https://avatars2.githubusercontent.com/u/205482?s=460&v=4" width="100px;" alt="Stephen Belanger"/><br /><sub><b>Stephen Belanger</b></sub></a></td>
|
||||||
|
<td align="center"><a href="https://github.com/irustm"><img src="https://avatars3.githubusercontent.com/u/16316579?s=460&v=4" width="100px;" alt="Rustam"/><br /><sub><b>Rustam</b></sub></a></td>
|
||||||
|
</tr>
|
||||||
|
</table>
|
||||||
|
|
||||||
## Maintainers ✨
|
## Maintainers ✨
|
||||||
|
|
||||||
@ -81,26 +263,81 @@ People maintaining this project.
|
|||||||
<table>
|
<table>
|
||||||
<tr>
|
<tr>
|
||||||
<td align="center"><a href="https://blog.atulr.com"><img src="https://avatars2.githubusercontent.com/u/4029423?v=4" width="100px;" alt="Atul R"/><br /><sub><b>Atul R</b></sub></a></td>
|
<td align="center"><a href="https://blog.atulr.com"><img src="https://avatars2.githubusercontent.com/u/4029423?v=4" width="100px;" alt="Atul R"/><br /><sub><b>Atul R</b></sub></a></td>
|
||||||
|
<td align="center"><a href="https://github.com/sedwards2009"><img src="https://avatars.githubusercontent.com/u/6926644?v=4" width="100px;" alt="Simon Edwards"/><br /><sub><b>Simon Edwards</b></sub></a></td>
|
||||||
</tr>
|
</tr>
|
||||||
</table>
|
</table>
|
||||||
|
|
||||||
|
|
||||||
## Contributors ✨
|
## Contributors ✨
|
||||||
|
|
||||||
Thanks goes to these wonderful people ([emoji key](https://allcontributors.org/docs/en/emoji-key)):
|
Thanks goes to these wonderful people ([emoji key](https://allcontributors.org/docs/en/emoji-key)):
|
||||||
|
|
||||||
<!-- ALL-CONTRIBUTORS-LIST:START - Do not remove or modify this section -->
|
<!-- ALL-CONTRIBUTORS-LIST:START - Do not remove or modify this section -->
|
||||||
<!-- prettier-ignore -->
|
<!-- prettier-ignore-start -->
|
||||||
|
<!-- markdownlint-disable -->
|
||||||
<table>
|
<table>
|
||||||
<tr>
|
<tr>
|
||||||
<td align="center"><a href="https://lramage.gitlab.io"><img src="https://avatars1.githubusercontent.com/u/43783393?v=4" width="100px;" alt="Lucas Ramage"/><br /><sub><b>Lucas Ramage</b></sub></a><br /><a href="https://github.com/nodegui/nodegui/commits?author=oxr463" title="Documentation">📖</a></td>
|
<td align="center"><a href="https://lramage.gitlab.io"><img src="https://avatars1.githubusercontent.com/u/43783393?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Lucas Ramage</b></sub></a><br /><a href="https://github.com/nodegui/nodegui/commits?author=oxr463" title="Documentation">📖</a></td>
|
||||||
<td align="center"><a href="https://github.com/gamtiq"><img src="https://avatars3.githubusercontent.com/u/1177323?v=4" width="100px;" alt="Denis Sikuler"/><br /><sub><b>Denis Sikuler</b></sub></a><br /><a href="https://github.com/nodegui/nodegui/commits?author=gamtiq" title="Documentation">📖</a></td>
|
<td align="center"><a href="https://github.com/gamtiq"><img src="https://avatars3.githubusercontent.com/u/1177323?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Denis Sikuler</b></sub></a><br /><a href="https://github.com/nodegui/nodegui/commits?author=gamtiq" title="Documentation">📖</a></td>
|
||||||
<td align="center"><a href="https://twitter.com/nahueljo"><img src="https://avatars1.githubusercontent.com/u/1612488?v=4" width="100px;" alt="Nahuel José"/><br /><sub><b>Nahuel José</b></sub></a><br /><a href="#question-Naahuel" title="Answering Questions">💬</a></td>
|
<td align="center"><a href="https://twitter.com/nahueljo"><img src="https://avatars1.githubusercontent.com/u/1612488?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Nahuel José</b></sub></a><br /><a href="#question-Naahuel" title="Answering Questions">💬</a></td>
|
||||||
<td align="center"><a href="https://github.com/kakulgupta"><img src="https://avatars3.githubusercontent.com/u/10727047?v=4" width="100px;" alt="Kakul Gupta"/><br /><sub><b>Kakul Gupta</b></sub></a><br /><a href="https://github.com/nodegui/nodegui/commits?author=kakulgupta" title="Code">💻</a></td>
|
<td align="center"><a href="https://github.com/kakulgupta"><img src="https://avatars3.githubusercontent.com/u/10727047?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Kakul Gupta</b></sub></a><br /><a href="https://github.com/nodegui/nodegui/commits?author=kakulgupta" title="Code">💻</a></td>
|
||||||
<td align="center"><a href="http://rahulgaba.com"><img src="https://avatars3.githubusercontent.com/u/7898942?v=4" width="100px;" alt="Rahul Gaba"/><br /><sub><b>Rahul Gaba</b></sub></a><br /><a href="https://github.com/nodegui/nodegui/commits?author=rgabs" title="Code">💻</a></td>
|
<td align="center"><a href="http://rahulgaba.com"><img src="https://avatars3.githubusercontent.com/u/7898942?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Rahul Gaba</b></sub></a><br /><a href="https://github.com/nodegui/nodegui/commits?author=rgabs" title="Code">💻</a></td>
|
||||||
|
<td align="center"><a href="http://uriziel.pl"><img src="https://avatars1.githubusercontent.com/u/568207?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Paweł Borecki</b></sub></a><br /><a href="https://github.com/nodegui/nodegui/commits?author=Uriziel01" title="Code">💻</a></td>
|
||||||
|
<td align="center"><a href="http://marcus-sa.me"><img src="https://avatars0.githubusercontent.com/u/8391194?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Marcus S. Abildskov</b></sub></a><br /><a href="https://github.com/nodegui/nodegui/commits?author=marcus-sa" title="Tests">⚠️</a></td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td align="center"><a href="http://www.mad3linux.org"><img src="https://avatars3.githubusercontent.com/u/508624?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Átila Camurça Alves</b></sub></a><br /><a href="https://github.com/nodegui/nodegui/commits?author=atilacamurca" title="Documentation">📖</a></td>
|
||||||
|
<td align="center"><a href="http://hibbard.eu"><img src="https://avatars2.githubusercontent.com/u/1940994?v=4?s=100" width="100px;" alt=""/><br /><sub><b>James Hibbard</b></sub></a><br /><a href="https://github.com/nodegui/nodegui/commits?author=jameshibbard" title="Code">💻</a></td>
|
||||||
|
<td align="center"><a href="https://github.com/soonoo"><img src="https://avatars2.githubusercontent.com/u/5436405?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Soonwoo Hong</b></sub></a><br /><a href="https://github.com/nodegui/nodegui/commits?author=soonoo" title="Code">💻</a></td>
|
||||||
|
<td align="center"><a href="https://github.com/illBeRoy"><img src="https://avatars2.githubusercontent.com/u/6681893?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Roy Sommer</b></sub></a><br /><a href="https://github.com/nodegui/nodegui/commits?author=illBeRoy" title="Code">💻</a></td>
|
||||||
|
<td align="center"><a href="https://github.com/paulocoghi"><img src="https://avatars1.githubusercontent.com/u/378397?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Paulo Coghi</b></sub></a><br /><a href="#ideas-paulocoghi" title="Ideas, Planning, & Feedback">🤔</a></td>
|
||||||
|
<td align="center"><a href="https://balthild.com"><img src="https://avatars2.githubusercontent.com/u/2662758?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Balthild Ires</b></sub></a><br /><a href="https://github.com/nodegui/nodegui/commits?author=balthild" title="Code">💻</a></td>
|
||||||
|
<td align="center"><a href="https://dimitarnestorov.com"><img src="https://avatars0.githubusercontent.com/u/8790386?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Dimitar Nestorov</b></sub></a><br /><a href="https://github.com/nodegui/nodegui/commits?author=dimitarnestorov" title="Code">💻</a></td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td align="center"><a href="https://www.greatapes.fi"><img src="https://avatars3.githubusercontent.com/u/3404389?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Mikko Sairio</b></sub></a><br /><a href="https://github.com/nodegui/nodegui/commits?author=msairio" title="Code">💻</a></td>
|
||||||
|
<td align="center"><a href="http://blog.pepf.nl"><img src="https://avatars1.githubusercontent.com/u/1265435?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Pepijn</b></sub></a><br /><a href="https://github.com/nodegui/nodegui/commits?author=pepf" title="Code">💻</a></td>
|
||||||
|
<td align="center"><a href="https://github.com/slidinghotdog"><img src="https://avatars3.githubusercontent.com/u/33790211?v=4?s=100" width="100px;" alt=""/><br /><sub><b>slidinghotdog</b></sub></a><br /><a href="https://github.com/nodegui/nodegui/commits?author=slidinghotdog" title="Code">💻</a></td>
|
||||||
|
<td align="center"><a href="http://www.bundyo.org"><img src="https://avatars1.githubusercontent.com/u/98318?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Bundyo (Kamen Bundev)</b></sub></a><br /><a href="https://github.com/nodegui/nodegui/commits?author=bundyo" title="Code">💻</a></td>
|
||||||
|
<td align="center"><a href="https://github.com/feng8848"><img src="https://avatars2.githubusercontent.com/u/40539968?v=4?s=100" width="100px;" alt=""/><br /><sub><b>feng8848</b></sub></a><br /><a href="https://github.com/nodegui/nodegui/commits?author=feng8848" title="Code">💻</a></td>
|
||||||
|
<td align="center"><a href="https://karelov.info"><img src="https://avatars3.githubusercontent.com/u/2384454?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Maksim Karelov</b></sub></a><br /><a href="https://github.com/nodegui/nodegui/commits?author=Ty3uK" title="Code">💻</a></td>
|
||||||
|
<td align="center"><a href="https://github.com/mspencer92"><img src="https://avatars2.githubusercontent.com/u/1910455?v=4?s=100" width="100px;" alt=""/><br /><sub><b>mspencer92</b></sub></a><br /><a href="https://github.com/nodegui/nodegui/commits?author=mspencer92" title="Code">💻</a></td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td align="center"><a href="https://github.com/blncd2020"><img src="https://avatars1.githubusercontent.com/u/59541979?v=4?s=100" width="100px;" alt=""/><br /><sub><b>blncd2020</b></sub></a><br /><a href="https://github.com/nodegui/nodegui/commits?author=blncd2020" title="Documentation">📖</a></td>
|
||||||
|
<td align="center"><a href="https://github.com/gluaxspeed"><img src="https://avatars2.githubusercontent.com/u/16431709?v=4?s=100" width="100px;" alt=""/><br /><sub><b>gluaxspeed</b></sub></a><br /><a href="https://github.com/nodegui/nodegui/commits?author=gluaxspeed" title="Code">💻</a></td>
|
||||||
|
<td align="center"><a href="https://github.com/Solant"><img src="https://avatars2.githubusercontent.com/u/5971578?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Solant</b></sub></a><br /><a href="https://github.com/nodegui/nodegui/commits?author=Solant" title="Code">💻</a></td>
|
||||||
|
<td align="center"><a href="https://github.com/shubhamzanwar"><img src="https://avatars0.githubusercontent.com/u/15626155?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Shubham Zanwar</b></sub></a><br /><a href="https://github.com/nodegui/nodegui/commits?author=shubhamzanwar" title="Code">💻</a></td>
|
||||||
|
<td align="center"><a href="https://github.com/Singha360"><img src="https://avatars1.githubusercontent.com/u/35334787?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Singha360</b></sub></a><br /><a href="https://github.com/nodegui/nodegui/commits?author=Singha360" title="Code">💻</a></td>
|
||||||
|
<td align="center"><a href="http://wellenline.com"><img src="https://avatars3.githubusercontent.com/u/3790782?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Mihkel</b></sub></a><br /><a href="https://github.com/nodegui/nodegui/commits?author=MihkelBaranov" title="Code">💻</a></td>
|
||||||
|
<td align="center"><a href="https://github.com/stevo2588"><img src="https://avatars1.githubusercontent.com/u/3278045?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Stephen A</b></sub></a><br /><a href="https://github.com/nodegui/nodegui/commits?author=stevo2588" title="Code">💻</a></td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td align="center"><a href="http://liujunjiang.com"><img src="https://avatars1.githubusercontent.com/u/15191056?v=4?s=100" width="100px;" alt=""/><br /><sub><b>流君酱</b></sub></a><br /><a href="https://github.com/nodegui/nodegui/commits?author=jardenliu" title="Code">💻</a></td>
|
||||||
|
<td align="center"><a href="https://github.com/agg23"><img src="https://avatars1.githubusercontent.com/u/238679?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Adam Gastineau</b></sub></a><br /><a href="https://github.com/nodegui/nodegui/commits?author=agg23" title="Code">💻</a></td>
|
||||||
|
<td align="center"><a href="https://github.com/swittk"><img src="https://avatars2.githubusercontent.com/u/5000572?v=4?s=100" width="100px;" alt=""/><br /><sub><b>swittk</b></sub></a><br /><a href="https://github.com/nodegui/nodegui/commits?author=swittk" title="Code">💻</a></td>
|
||||||
|
<td align="center"><a href="https://github.com/craftingmod"><img src="https://avatars2.githubusercontent.com/u/9389278?v=4?s=100" width="100px;" alt=""/><br /><sub><b>craftingmod</b></sub></a><br /><a href="https://github.com/nodegui/nodegui/commits?author=craftingmod" title="Code">💻</a></td>
|
||||||
|
<td align="center"><a href="http://www.m2osw.com"><img src="https://avatars1.githubusercontent.com/u/643129?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Doug Barbieri</b></sub></a><br /><a href="https://github.com/nodegui/nodegui/commits?author=dooglio" title="Code">💻</a></td>
|
||||||
|
<td align="center"><a href="https://github.com/NeryHenrique"><img src="https://avatars0.githubusercontent.com/u/6879141?v=4?s=100" width="100px;" alt=""/><br /><sub><b>HENRIQUE DE SOUZA NERY</b></sub></a><br /><a href="https://github.com/nodegui/nodegui/commits?author=NeryHenrique" title="Code">💻</a></td>
|
||||||
|
<td align="center"><a href="https://ruslang.xyz"><img src="https://avatars0.githubusercontent.com/u/25264730?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Ruslan Garifullin</b></sub></a><br /><a href="https://github.com/nodegui/nodegui/commits?author=ruslang02" title="Code">💻</a></td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td align="center"><a href="https://github.com/ran-j"><img src="https://avatars0.githubusercontent.com/u/17410205?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Ranieri</b></sub></a><br /><a href="https://github.com/nodegui/nodegui/commits?author=ran-j" title="Code">💻</a> <a href="https://github.com/nodegui/nodegui/commits?author=ran-j" title="Documentation">📖</a></td>
|
||||||
|
<td align="center"><a href="https://master-technology.com"><img src="https://avatars3.githubusercontent.com/u/850871?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Nathanael Anderson</b></sub></a><br /><a href="https://github.com/nodegui/nodegui/commits?author=NathanaelA" title="Code">💻</a></td>
|
||||||
|
<td align="center"><a href="https://ubiq.co.za"><img src="https://avatars0.githubusercontent.com/u/4415071?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Ross</b></sub></a><br /><a href="https://github.com/nodegui/nodegui/commits?author=rocbear" title="Code">💻</a> <a href="https://github.com/nodegui/nodegui/commits?author=rocbear" title="Documentation">📖</a></td>
|
||||||
|
<td align="center"><a href="https://github.com/4h7l"><img src="https://avatars.githubusercontent.com/u/69183283?v=4?s=100" width="100px;" alt=""/><br /><sub><b>4h7l</b></sub></a><br /><a href="https://github.com/nodegui/nodegui/commits?author=4h7l" title="Code">💻</a></td>
|
||||||
|
<td align="center"><a href="https://github.com/Maks-s"><img src="https://avatars.githubusercontent.com/u/26678512?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Maks</b></sub></a><br /><a href="https://github.com/nodegui/nodegui/commits?author=Maks-s" title="Code">💻</a></td>
|
||||||
|
<td align="center"><a href="https://github.com/zhb124404"><img src="https://avatars.githubusercontent.com/u/16805041?v=4?s=100" width="100px;" alt=""/><br /><sub><b>zhb124404</b></sub></a><br /><a href="https://github.com/nodegui/nodegui/commits?author=zhb124404" title="Documentation">📖</a></td>
|
||||||
|
<td align="center"><a href="http://www.apsis.io"><img src="https://avatars.githubusercontent.com/u/579688?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Wyatt Kirby</b></sub></a><br /><a href="https://github.com/nodegui/nodegui/commits?author=wkirby" title="Code">💻</a></td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td align="center"><a href="https://stvkoch.github.io."><img src="https://avatars.githubusercontent.com/u/14454?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Steven Koch</b></sub></a><br /><a href="https://github.com/nodegui/nodegui/commits?author=stvkoch" title="Code">💻</a></td>
|
||||||
|
<td align="center"><a href="https://github.com/sedwards2009"><img src="https://avatars.githubusercontent.com/u/6926644?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Simon Edwards</b></sub></a><br /><a href="https://github.com/nodegui/nodegui/commits?author=sedwards2009" title="Code">💻</a></td>
|
||||||
</tr>
|
</tr>
|
||||||
</table>
|
</table>
|
||||||
|
|
||||||
|
<!-- markdownlint-restore -->
|
||||||
|
<!-- prettier-ignore-end -->
|
||||||
|
|
||||||
<!-- ALL-CONTRIBUTORS-LIST:END -->
|
<!-- ALL-CONTRIBUTORS-LIST:END -->
|
||||||
|
|
||||||
This project follows the [all-contributors](https://github.com/all-contributors/all-contributors) specification. Contributions of any kind welcome!
|
This project follows the [all-contributors](https://github.com/all-contributors/all-contributors) specification. Contributions of any kind welcome!
|
||||||
|
|||||||
26
binding.gyp
26
binding.gyp
@ -1,26 +0,0 @@
|
|||||||
{
|
|
||||||
"includes": [],
|
|
||||||
"targets": [
|
|
||||||
{
|
|
||||||
"target_name": "qtnode",
|
|
||||||
'include_dirs': ['.', './deps/'],
|
|
||||||
"includes": [
|
|
||||||
"./config/application.gypi",
|
|
||||||
"./config/moc.gypi",
|
|
||||||
"./config/common.gypi",
|
|
||||||
"./config/qt.gypi",
|
|
||||||
"./config/deps.gypi",
|
|
||||||
],
|
|
||||||
'dependencies': [
|
|
||||||
# './config/deps.gypi:nodeguidep',
|
|
||||||
],
|
|
||||||
'conditions':[
|
|
||||||
['OS=="mac"', {
|
|
||||||
'xcode_settings': {
|
|
||||||
'OTHER_LDFLAGS': ['-Wl,-rpath,@loader_path'],
|
|
||||||
},
|
|
||||||
}]
|
|
||||||
]
|
|
||||||
},
|
|
||||||
]
|
|
||||||
}
|
|
||||||
@ -1,30 +0,0 @@
|
|||||||
{
|
|
||||||
"sources": [
|
|
||||||
"../src/cpp/main.cpp",
|
|
||||||
# non wrapped
|
|
||||||
"../src/cpp/Extras/Utils/nutils.cpp",
|
|
||||||
"../src/cpp/core/FlexLayout/flexlayout.cpp",
|
|
||||||
"../src/cpp/core/FlexLayout/flexitem.cpp",
|
|
||||||
"../src/cpp/core/YogaWidget/nodestyle.cpp",
|
|
||||||
"../src/cpp/core/Events/eventsmap.cpp",
|
|
||||||
"../src/cpp/core/Events/eventwidget.cpp",
|
|
||||||
"../src/cpp/core/YogaWidget/yogawidget.cpp",
|
|
||||||
# wrapped cpps. Move non wrapped ones to shared gypi
|
|
||||||
"../src/cpp/QtGui/QApplication/qapplication_wrap.cpp",
|
|
||||||
"../src/cpp/QtGui/QEvent/QKeyEvent/qkeyevent_wrap.cpp",
|
|
||||||
"../src/cpp/QtGui/QPixmap/qpixmap_wrap.cpp",
|
|
||||||
"../src/cpp/QtGui/QIcon/qicon_wrap.cpp",
|
|
||||||
'../src/cpp/core/FlexLayout/flexlayout_wrap.cpp',
|
|
||||||
"../src/cpp/QtWidgets/QWidget/qwidget_wrap.cpp",
|
|
||||||
"../src/cpp/QtWidgets/QGridLayout/qgridlayout_wrap.cpp",
|
|
||||||
"../src/cpp/QtWidgets/QLabel/qlabel_wrap.cpp",
|
|
||||||
"../src/cpp/QtWidgets/QLayout/qlayout_wrap.cpp",
|
|
||||||
"../src/cpp/QtWidgets/QMainWindow/qmainwindow_wrap.cpp",
|
|
||||||
"../src/cpp/QtWidgets/QPushButton/qpushbutton_wrap.cpp",
|
|
||||||
"../src/cpp/QtWidgets/QCheckBox/qcheckbox_wrap.cpp",
|
|
||||||
"../src/cpp/QtWidgets/QProgressBar/qprogressbar_wrap.cpp",
|
|
||||||
"../src/cpp/QtWidgets/QRadioButton/qradiobutton_wrap.cpp",
|
|
||||||
"../src/cpp/QtWidgets/QLineEdit/qlineedit_wrap.cpp",
|
|
||||||
"../src/cpp/QtWidgets/QPlainTextEdit/qplaintextedit_wrap.cpp",
|
|
||||||
],
|
|
||||||
}
|
|
||||||
43
config/common.cmake
Normal file
43
config/common.cmake
Normal file
@ -0,0 +1,43 @@
|
|||||||
|
set(CMAKE_INCLUDE_CURRENT_DIR ON)
|
||||||
|
|
||||||
|
find_program(CCACHE_PROGRAM ccache)
|
||||||
|
if(CCACHE_PROGRAM)
|
||||||
|
set_property(GLOBAL PROPERTY RULE_LAUNCH_COMPILE "${CCACHE_PROGRAM}")
|
||||||
|
endif()
|
||||||
|
|
||||||
|
set(CMAKE_OSX_DEPLOYMENT_TARGET "10.13" CACHE STRING "Minimum OS X deployment version")
|
||||||
|
|
||||||
|
function(AddCommonConfig addonName)
|
||||||
|
target_compile_features(${addonName} PRIVATE
|
||||||
|
cxx_constexpr
|
||||||
|
cxx_inheriting_constructors
|
||||||
|
cxx_lambdas
|
||||||
|
cxx_auto_type
|
||||||
|
cxx_variadic_templates
|
||||||
|
cxx_variable_templates
|
||||||
|
cxx_std_17
|
||||||
|
)
|
||||||
|
|
||||||
|
if(napi_build_version)
|
||||||
|
target_compile_definitions(${addonName} PRIVATE
|
||||||
|
NAPI_VERSION=${napi_build_version}
|
||||||
|
)
|
||||||
|
endif()
|
||||||
|
|
||||||
|
if (WIN32)
|
||||||
|
target_compile_definitions(${addonName} PRIVATE
|
||||||
|
ENUM_BITFIELDS_NOT_SUPPORTED
|
||||||
|
)
|
||||||
|
endif()
|
||||||
|
endfunction(AddCommonConfig addonName)
|
||||||
|
|
||||||
|
function(GetModuleVersion moduleName packageJsonDir)
|
||||||
|
execute_process(COMMAND node -p "require('${packageJsonDir}/package.json').version"
|
||||||
|
WORKING_DIRECTORY ${CMAKE_SOURCE_DIR}
|
||||||
|
OUTPUT_VARIABLE packageJsonVersion
|
||||||
|
)
|
||||||
|
string(REPLACE "\n" "" packageJsonVersion "${packageJsonVersion}")
|
||||||
|
string(REPLACE "\"" "" packageJsonVersion "${packageJsonVersion}")
|
||||||
|
|
||||||
|
set("${moduleName}_VERSION" "${packageJsonVersion}" PARENT_SCOPE)
|
||||||
|
endfunction(GetModuleVersion moduleName packageJsonDir)
|
||||||
@ -1,47 +0,0 @@
|
|||||||
{
|
|
||||||
'cflags!': ['-fno-exceptions'],
|
|
||||||
'cflags_cc!': ['-fno-exceptions'],
|
|
||||||
'sources': [],
|
|
||||||
'includes': [],
|
|
||||||
'include_dirs': [
|
|
||||||
"<!@(node -p \"require('node-addon-api').include\")",
|
|
||||||
'..',
|
|
||||||
'../deps/',
|
|
||||||
],
|
|
||||||
'dependencies': [
|
|
||||||
"<!(node -p \"require('node-addon-api').gyp\")",
|
|
||||||
],
|
|
||||||
'defines': ['NAPI_CPP_EXCEPTIONS'],
|
|
||||||
'target_conditions': [
|
|
||||||
['OS=="mac"', {
|
|
||||||
'cflags+': ['-fvisibility=hidden'],
|
|
||||||
'xcode_settings': {
|
|
||||||
'GCC_SYMBOLS_PRIVATE_EXTERN': 'YES', # -fvisibility=hidden
|
|
||||||
'GCC_ENABLE_CPP_EXCEPTIONS': 'YES',
|
|
||||||
'CLANG_CXX_LIBRARY': 'libc++',
|
|
||||||
'MACOSX_DEPLOYMENT_TARGET': '10.7',
|
|
||||||
'OTHER_CFLAGS': ['-std=c++14'],
|
|
||||||
},
|
|
||||||
}],
|
|
||||||
['OS=="linux"', {
|
|
||||||
'cflags': [
|
|
||||||
'-fPIC'
|
|
||||||
],
|
|
||||||
}],
|
|
||||||
['OS=="win"', {
|
|
||||||
'msvs_settings': {
|
|
||||||
'VCCLCompilerTool': {
|
|
||||||
'ExceptionHandling': 1,
|
|
||||||
'AdditionalOptions': [
|
|
||||||
'/GR-',
|
|
||||||
'/MT',
|
|
||||||
'/EHsc'
|
|
||||||
]
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"defines": [
|
|
||||||
"_HAS_EXCEPTIONS=1",
|
|
||||||
],
|
|
||||||
}],
|
|
||||||
],
|
|
||||||
}
|
|
||||||
@ -1,33 +0,0 @@
|
|||||||
{
|
|
||||||
# "targets": [
|
|
||||||
# {
|
|
||||||
# "target_name": "nodeguidep",
|
|
||||||
# 'type': 'shared_library',
|
|
||||||
"include_dirs": ['../deps/'],
|
|
||||||
"sources": [
|
|
||||||
"../deps/yoga/log.cpp",
|
|
||||||
"../deps/yoga/Utils.cpp",
|
|
||||||
"../deps/yoga/YGConfig.cpp",
|
|
||||||
"../deps/yoga/YGEnums.cpp",
|
|
||||||
"../deps/yoga/YGLayout.cpp",
|
|
||||||
"../deps/yoga/YGNode.cpp",
|
|
||||||
"../deps/yoga/YGNodePrint.cpp",
|
|
||||||
"../deps/yoga/YGStyle.cpp",
|
|
||||||
"../deps/yoga/YGValue.cpp",
|
|
||||||
"../deps/yoga/Yoga.cpp",
|
|
||||||
"../deps/yoga/event/event.cpp",
|
|
||||||
"../deps/yoga/internal/experiments.cpp",
|
|
||||||
],
|
|
||||||
'defines': [
|
|
||||||
'SPDLOG_COMPILED_LIB'
|
|
||||||
],
|
|
||||||
'conditions': [
|
|
||||||
['OS=="win"', {
|
|
||||||
'defines': [
|
|
||||||
'ENUM_BITFIELDS_NOT_SUPPORTED', # Needed for yoga to work nicely on windows
|
|
||||||
],
|
|
||||||
}]
|
|
||||||
],
|
|
||||||
# }
|
|
||||||
# ]
|
|
||||||
}
|
|
||||||
@ -1,15 +0,0 @@
|
|||||||
# AUTOGENERATED FILE. DO NOT MODIFY . ALL CHANGES WILL BE LOST
|
|
||||||
# RUN: npm run automoc after updating moc.json
|
|
||||||
{
|
|
||||||
"sources": [
|
|
||||||
"../src/cpp/autogen/nwidget_moc.cpp",
|
|
||||||
"../src/cpp/autogen/nlabel_moc.cpp",
|
|
||||||
"../src/cpp/autogen/ncheckbox_moc.cpp",
|
|
||||||
"../src/cpp/autogen/nlineedit_moc.cpp",
|
|
||||||
"../src/cpp/autogen/nmainwindow_moc.cpp",
|
|
||||||
"../src/cpp/autogen/nprogressbar_moc.cpp",
|
|
||||||
"../src/cpp/autogen/npushbutton_moc.cpp",
|
|
||||||
"../src/cpp/autogen/nradiobutton_moc.cpp",
|
|
||||||
"../src/cpp/autogen/nplaintextedit_moc.cpp"
|
|
||||||
]
|
|
||||||
}
|
|
||||||
@ -1,14 +0,0 @@
|
|||||||
{
|
|
||||||
"include": "src/cpp/core/NodeWidget/nodewidget.h",
|
|
||||||
"headers": [
|
|
||||||
"src/cpp/QtWidgets/QWidget/nwidget.h",
|
|
||||||
"src/cpp/QtWidgets/QLabel/nlabel.h",
|
|
||||||
"src/cpp/QtWidgets/QCheckBox/ncheckbox.h",
|
|
||||||
"src/cpp/QtWidgets/QLineEdit/nlineedit.h",
|
|
||||||
"src/cpp/QtWidgets/QMainWindow/nmainwindow.h",
|
|
||||||
"src/cpp/QtWidgets/QProgressBar/nprogressbar.h",
|
|
||||||
"src/cpp/QtWidgets/QPushButton/npushbutton.h",
|
|
||||||
"src/cpp/QtWidgets/QRadioButton/nradiobutton.h",
|
|
||||||
"src/cpp/QtWidgets/QPlainTextEdit/nplaintextedit.h"
|
|
||||||
]
|
|
||||||
}
|
|
||||||
27
config/napi.cmake
Normal file
27
config/napi.cmake
Normal file
@ -0,0 +1,27 @@
|
|||||||
|
# NAPI stuff
|
||||||
|
|
||||||
|
function(AddNapiSupport addonName)
|
||||||
|
execute_process(COMMAND node -p "require('node-addon-api').include"
|
||||||
|
WORKING_DIRECTORY ${CMAKE_SOURCE_DIR}
|
||||||
|
OUTPUT_VARIABLE NODE_ADDON_API_DIR
|
||||||
|
)
|
||||||
|
string(REPLACE "\n" "" NODE_ADDON_API_DIR ${NODE_ADDON_API_DIR})
|
||||||
|
string(REPLACE "\"" "" NODE_ADDON_API_DIR ${NODE_ADDON_API_DIR})
|
||||||
|
|
||||||
|
set_target_properties(${addonName} PROPERTIES PREFIX "" SUFFIX ".node")
|
||||||
|
|
||||||
|
target_include_directories(${addonName} PRIVATE
|
||||||
|
"${NODE_ADDON_API_DIR}"
|
||||||
|
)
|
||||||
|
target_compile_definitions(${addonName} PRIVATE
|
||||||
|
NAPI_CPP_EXCEPTIONS
|
||||||
|
)
|
||||||
|
|
||||||
|
if (WIN32)
|
||||||
|
target_compile_definitions(${addonName} PRIVATE
|
||||||
|
_HAS_EXCEPTIONS=1
|
||||||
|
)
|
||||||
|
endif()
|
||||||
|
endfunction(AddNapiSupport addonName)
|
||||||
|
|
||||||
|
|
||||||
33
config/plugin.cmake
Normal file
33
config/plugin.cmake
Normal file
@ -0,0 +1,33 @@
|
|||||||
|
# This cmake file will be used in the plugins
|
||||||
|
set(PLUGIN_CMAKE_DIR "${CMAKE_CURRENT_LIST_DIR}")
|
||||||
|
set(NODEGUI_ROOT "${PLUGIN_CMAKE_DIR}/..")
|
||||||
|
set(NODEGUI_LIBRARY "${NODEGUI_ROOT}/build/Release/nodegui_core.node")
|
||||||
|
|
||||||
|
if(WIN32)
|
||||||
|
set(NODEGUI_LIBRARY "${NODEGUI_ROOT}\\build\\Release\\nodegui_core.lib")
|
||||||
|
endif()
|
||||||
|
|
||||||
|
include("${PLUGIN_CMAKE_DIR}/common.cmake")
|
||||||
|
include("${PLUGIN_CMAKE_DIR}/qt.cmake")
|
||||||
|
include("${PLUGIN_CMAKE_DIR}/napi.cmake")
|
||||||
|
|
||||||
|
macro(AddPluginConfig addonName)
|
||||||
|
AddCommonConfig(${addonName})
|
||||||
|
AddQtSupport(${addonName})
|
||||||
|
AddNapiSupport(${addonName})
|
||||||
|
|
||||||
|
target_link_libraries(${addonName} PRIVATE
|
||||||
|
"${NODEGUI_LIBRARY}"
|
||||||
|
)
|
||||||
|
|
||||||
|
target_include_directories(${addonName} PRIVATE
|
||||||
|
"${CMAKE_JS_INC}"
|
||||||
|
"${NODEGUI_ROOT}"
|
||||||
|
"${NODEGUI_ROOT}/src/cpp"
|
||||||
|
"${NODEGUI_ROOT}/src/cpp/include"
|
||||||
|
"${NODEGUI_ROOT}/src/cpp/include/deps"
|
||||||
|
"${NODEGUI_ROOT}/src/cpp/include/nodegui"
|
||||||
|
)
|
||||||
|
|
||||||
|
endmacro(AddPluginConfig addonName)
|
||||||
|
|
||||||
24
config/qode.cmake
Normal file
24
config/qode.cmake
Normal file
@ -0,0 +1,24 @@
|
|||||||
|
# Adds Qode shared binary support
|
||||||
|
|
||||||
|
set(QODE_CONFIG_FILE @nodegui/qode/src/config.js)
|
||||||
|
|
||||||
|
macro(AddQodeSupport addonName)
|
||||||
|
if (WIN32)
|
||||||
|
|
||||||
|
execute_process(COMMAND node -p "require('${QODE_CONFIG_FILE}').extractDir"
|
||||||
|
WORKING_DIRECTORY ${CMAKE_SOURCE_DIR}
|
||||||
|
OUTPUT_VARIABLE QODE_BINARY_DIR
|
||||||
|
)
|
||||||
|
|
||||||
|
string(REPLACE "\n" "" QODE_BINARY_DIR "${QODE_BINARY_DIR}")
|
||||||
|
string(REPLACE "\"" "" QODE_BINARY_DIR "${QODE_BINARY_DIR}")
|
||||||
|
|
||||||
|
message(STATUS "Using Qode installation for ${addonName} QODE_BINARY_DIR:${QODE_BINARY_DIR}")
|
||||||
|
|
||||||
|
target_link_libraries(${CORE_WIDGETS_ADDON} PRIVATE
|
||||||
|
"${QODE_BINARY_DIR}\\node.lib"
|
||||||
|
)
|
||||||
|
|
||||||
|
endif()
|
||||||
|
|
||||||
|
endmacro(AddQodeSupport addonName)
|
||||||
25
config/qt.cmake
Normal file
25
config/qt.cmake
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
# Adds Qt support
|
||||||
|
# make sure you include this at the top of whichever Cmakelist file you are going to use.
|
||||||
|
# Need for automatic moc.
|
||||||
|
|
||||||
|
set(CMAKE_AUTOMOC ON)
|
||||||
|
set(CMAKE_AUTORCC ON)
|
||||||
|
set(CMAKE_AUTOUIC ON)
|
||||||
|
|
||||||
|
set(QTCONFIG_FILE ${CMAKE_CURRENT_LIST_DIR}/qtConfig.js)
|
||||||
|
|
||||||
|
macro(AddQtSupport addonName)
|
||||||
|
execute_process(COMMAND node -p "require('${QTCONFIG_FILE}').qtCmakeDir"
|
||||||
|
WORKING_DIRECTORY ${CMAKE_SOURCE_DIR}
|
||||||
|
OUTPUT_VARIABLE QT_CMAKE_HOME_DIR
|
||||||
|
)
|
||||||
|
|
||||||
|
string(REPLACE "\n" "" QT_CMAKE_HOME_DIR "${QT_CMAKE_HOME_DIR}")
|
||||||
|
string(REPLACE "\"" "" QT_CMAKE_HOME_DIR "${QT_CMAKE_HOME_DIR}")
|
||||||
|
|
||||||
|
message(STATUS "Using Qt installation for ${addonName} QT_CMAKE_HOME_DIR:${QT_CMAKE_HOME_DIR}")
|
||||||
|
|
||||||
|
list(APPEND CMAKE_PREFIX_PATH "${QT_CMAKE_HOME_DIR}/../../..")
|
||||||
|
find_package(Qt6 REQUIRED COMPONENTS Widgets Gui Core Svg SvgWidgets)
|
||||||
|
|
||||||
|
endmacro(AddQtSupport addonName)
|
||||||
@ -1,64 +0,0 @@
|
|||||||
{
|
|
||||||
'variables': {
|
|
||||||
'qt_home_dir%': "<!(node -p \"require('@nodegui/qode').qtHome\")",
|
|
||||||
},
|
|
||||||
'target_conditions': [
|
|
||||||
['OS=="mac"', {
|
|
||||||
'actions': [
|
|
||||||
{
|
|
||||||
'action_name': 'symlink_qt_headers',
|
|
||||||
'inputs': [],
|
|
||||||
'outputs': [
|
|
||||||
'<(qt_home_dir)/include/QtCore',
|
|
||||||
'<(qt_home_dir)/include/QtGui',
|
|
||||||
'<(qt_home_dir)/include/QtWidgets',
|
|
||||||
],
|
|
||||||
'action': [
|
|
||||||
'sh', '-c',
|
|
||||||
'mkdir -p <(qt_home_dir)/include && \
|
|
||||||
ln -sfn <(qt_home_dir)/lib/QtCore.framework/Versions/5/Headers <(qt_home_dir)/include/QtCore && \
|
|
||||||
ln -sfn <(qt_home_dir)/lib/QtGui.framework/Versions/5/Headers <(qt_home_dir)/include/QtGui && \
|
|
||||||
ln -sfn <(qt_home_dir)/lib/QtWidgets.framework/Versions/5/Headers <(qt_home_dir)/include/QtWidgets'
|
|
||||||
],
|
|
||||||
}
|
|
||||||
],
|
|
||||||
'include_dirs': [
|
|
||||||
'<(qt_home_dir)/include',
|
|
||||||
'<(qt_home_dir)/lib/QtCore.framework/Versions/5/Headers',
|
|
||||||
'<(qt_home_dir)/lib/QtGui.framework/Versions/5/Headers',
|
|
||||||
'<(qt_home_dir)/lib/QtWidgets.framework/Versions/5/Headers',
|
|
||||||
],
|
|
||||||
'libraries': [
|
|
||||||
'<(qt_home_dir)/lib/QtCore.framework/Versions/5/QtCore',
|
|
||||||
'<(qt_home_dir)/lib/QtGui.framework/Versions/5/QtGui',
|
|
||||||
'<(qt_home_dir)/lib/QtWidgets.framework/Versions/5/QtWidgets',
|
|
||||||
],
|
|
||||||
}],
|
|
||||||
['OS=="linux"', {
|
|
||||||
'include_dirs': [
|
|
||||||
'<(qt_home_dir)/include',
|
|
||||||
'<(qt_home_dir)/include/QtCore',
|
|
||||||
'<(qt_home_dir)/include/QtGui',
|
|
||||||
'<(qt_home_dir)/include/QtWidgets',
|
|
||||||
],
|
|
||||||
'libraries': [
|
|
||||||
'<(qt_home_dir)/lib/libQt5Core.so',
|
|
||||||
'<(qt_home_dir)/lib/libQt5Gui.so',
|
|
||||||
'<(qt_home_dir)/lib/libQt5Widgets.so',
|
|
||||||
],
|
|
||||||
}],
|
|
||||||
['OS=="win"', {
|
|
||||||
'include_dirs': [
|
|
||||||
'<(qt_home_dir)\\include',
|
|
||||||
'<(qt_home_dir)\\include\\QtCore',
|
|
||||||
'<(qt_home_dir)\\include\\QtGui',
|
|
||||||
'<(qt_home_dir)\\include\\QtWidgets',
|
|
||||||
],
|
|
||||||
'libraries': [
|
|
||||||
'<(qt_home_dir)\\lib\\Qt5Core.lib',
|
|
||||||
'<(qt_home_dir)\\lib\\Qt5Gui.lib',
|
|
||||||
'<(qt_home_dir)\\lib\\Qt5Widgets.lib',
|
|
||||||
]
|
|
||||||
}],
|
|
||||||
],
|
|
||||||
}
|
|
||||||
102
config/qtConfig.js
Normal file
102
config/qtConfig.js
Normal file
@ -0,0 +1,102 @@
|
|||||||
|
const os = require('os');
|
||||||
|
const path = require('path');
|
||||||
|
const fs = require('fs');
|
||||||
|
|
||||||
|
const SETUP_DIR = path.resolve(__dirname, '..', 'miniqt');
|
||||||
|
const QT_VERSION = '6.4.1';
|
||||||
|
const MIRROR = Boolean(process.env.QT_LINK_MIRROR) ? process.env.QT_LINK_MIRROR : 'https://download.qt.io';
|
||||||
|
|
||||||
|
const checkIfExists = (fullPath) => {
|
||||||
|
return () => fs.existsSync(fullPath);
|
||||||
|
};
|
||||||
|
|
||||||
|
function getMiniQtConfig() {
|
||||||
|
switch (os.platform()) {
|
||||||
|
case 'darwin': {
|
||||||
|
const qtHome = path.resolve(SETUP_DIR, QT_VERSION, 'macos');
|
||||||
|
return {
|
||||||
|
qtHome,
|
||||||
|
artifacts: [
|
||||||
|
{
|
||||||
|
name: 'Qt Base',
|
||||||
|
link: `${MIRROR}/online/qtsdkrepository/mac_x64/desktop/qt6_641/qt.qt6.641.clang_64/6.4.1-0-202211101256qtbase-MacOS-MacOS_12-Clang-MacOS-MacOS_12-X86_64-ARM64.7z`,
|
||||||
|
skipSetup: checkIfExists(path.resolve(qtHome, 'plugins', 'platforms', 'libqcocoa.dylib')),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Qt Svg',
|
||||||
|
link: `${MIRROR}/online/qtsdkrepository/mac_x64/desktop/qt6_641/qt.qt6.641.clang_64/6.4.1-0-202211101256qtsvg-MacOS-MacOS_12-Clang-MacOS-MacOS_12-X86_64-ARM64.7z`,
|
||||||
|
skipSetup: checkIfExists(path.resolve(qtHome, 'lib', 'QtSvg.framework', 'QtSvg')),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Qt Tools',
|
||||||
|
link: `${MIRROR}/online/qtsdkrepository/mac_x64/desktop/qt6_641/qt.qt6.641.clang_64/6.4.1-0-202211101256qttools-MacOS-MacOS_12-Clang-MacOS-MacOS_12-X86_64-ARM64.7z`,
|
||||||
|
skipSetup: checkIfExists(path.resolve(qtHome, 'bin', 'macdeployqt')),
|
||||||
|
},
|
||||||
|
],
|
||||||
|
};
|
||||||
|
}
|
||||||
|
case 'win32': {
|
||||||
|
const qtHome = path.resolve(SETUP_DIR, QT_VERSION, 'msvc2019_64');
|
||||||
|
return {
|
||||||
|
qtHome,
|
||||||
|
artifacts: [
|
||||||
|
{
|
||||||
|
name: 'Qt Base',
|
||||||
|
link: `${MIRROR}/online/qtsdkrepository/windows_x86/desktop/qt6_641/qt.qt6.641.win64_msvc2019_64/6.4.1-0-202211101254qtbase-Windows-Windows_10_21H2-MSVC2019-Windows-Windows_10_21H2-X86_64.7z`,
|
||||||
|
skipSetup: checkIfExists(path.resolve(qtHome, 'bin', 'Qt6Core.dll')),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Qt SVG',
|
||||||
|
link: `${MIRROR}/online/qtsdkrepository/windows_x86/desktop/qt6_641/qt.qt6.641.win64_msvc2019_64/6.4.1-0-202211101254qtsvg-Windows-Windows_10_21H2-MSVC2019-Windows-Windows_10_21H2-X86_64.7z`,
|
||||||
|
skipSetup: checkIfExists(path.resolve(qtHome, 'bin', 'Qt6Svg.dll')),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Qt Tools',
|
||||||
|
link: `${MIRROR}/online/qtsdkrepository/windows_x86/desktop/qt6_641/qt.qt6.641.win64_msvc2019_64/6.4.1-0-202211101254qttools-Windows-Windows_10_21H2-MSVC2019-Windows-Windows_10_21H2-X86_64.7z`,
|
||||||
|
skipSetup: checkIfExists(path.resolve(qtHome, 'bin', 'windeployqt.exe')),
|
||||||
|
},
|
||||||
|
],
|
||||||
|
};
|
||||||
|
}
|
||||||
|
case 'linux': {
|
||||||
|
const qtHome = path.resolve(SETUP_DIR, QT_VERSION, 'gcc_64');
|
||||||
|
return {
|
||||||
|
qtHome,
|
||||||
|
artifacts: [
|
||||||
|
{
|
||||||
|
name: 'Qt Base',
|
||||||
|
link: `${MIRROR}/online/qtsdkrepository/linux_x64/desktop/qt6_641/qt.qt6.641.gcc_64/6.4.1-0-202211101305qtbase-Linux-RHEL_8_4-GCC-Linux-RHEL_8_4-X86_64.7z`,
|
||||||
|
skipSetup: checkIfExists(path.resolve(qtHome, 'bin', 'moc')),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Qt SVG',
|
||||||
|
link: `${MIRROR}/online/qtsdkrepository/linux_x64/desktop/qt6_641/qt.qt6.641.gcc_64/6.4.1-0-202211101305qtsvg-Linux-RHEL_8_4-GCC-Linux-RHEL_8_4-X86_64.7z`,
|
||||||
|
skipSetup: checkIfExists(path.resolve(qtHome, 'lib', 'libQt6Svg.so')),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Qt ICU',
|
||||||
|
link: `${MIRROR}/online/qtsdkrepository/linux_x64/desktop/qt6_641/qt.qt6.641.gcc_64/6.4.1-0-202211101305icu-linux-Rhel7.2-x64.7z`,
|
||||||
|
skipSetup: checkIfExists(path.resolve(qtHome, 'lib', 'libicuuc.so')),
|
||||||
|
},
|
||||||
|
],
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const miniQt = {
|
||||||
|
...getMiniQtConfig(),
|
||||||
|
version: QT_VERSION,
|
||||||
|
setupDir: SETUP_DIR,
|
||||||
|
};
|
||||||
|
|
||||||
|
const useCustomQt = Boolean(process.env.QT_INSTALL_DIR);
|
||||||
|
const qtHome = useCustomQt ? process.env.QT_INSTALL_DIR : miniQt.qtHome;
|
||||||
|
const qtCmakeDir = path.resolve(qtHome, 'lib', 'cmake', 'Qt6');
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
qtHome,
|
||||||
|
miniQt,
|
||||||
|
qtCmakeDir,
|
||||||
|
useCustomQt,
|
||||||
|
};
|
||||||
87
deps/spdlog/async.h
vendored
87
deps/spdlog/async.h
vendored
@ -1,87 +0,0 @@
|
|||||||
|
|
||||||
//
|
|
||||||
// Copyright(c) 2018 Gabi Melman.
|
|
||||||
// Distributed under the MIT License (http://opensource.org/licenses/MIT)
|
|
||||||
//
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
//
|
|
||||||
// Async logging using global thread pool
|
|
||||||
// All loggers created here share same global thread pool.
|
|
||||||
// Each log message is pushed to a queue along withe a shared pointer to the
|
|
||||||
// logger.
|
|
||||||
// If a logger deleted while having pending messages in the queue, it's actual
|
|
||||||
// destruction will defer
|
|
||||||
// until all its messages are processed by the thread pool.
|
|
||||||
// This is because each message in the queue holds a shared_ptr to the
|
|
||||||
// originating logger.
|
|
||||||
|
|
||||||
#include "spdlog/async_logger.h"
|
|
||||||
#include "spdlog/details/registry.h"
|
|
||||||
#include "spdlog/details/thread_pool.h"
|
|
||||||
|
|
||||||
#include <memory>
|
|
||||||
#include <mutex>
|
|
||||||
|
|
||||||
namespace spdlog {
|
|
||||||
|
|
||||||
namespace details {
|
|
||||||
static const size_t default_async_q_size = 8192;
|
|
||||||
}
|
|
||||||
|
|
||||||
// async logger factory - creates async loggers backed with thread pool.
|
|
||||||
// if a global thread pool doesn't already exist, create it with default queue
|
|
||||||
// size of 8192 items and single thread.
|
|
||||||
template<async_overflow_policy OverflowPolicy = async_overflow_policy::block>
|
|
||||||
struct async_factory_impl
|
|
||||||
{
|
|
||||||
template<typename Sink, typename... SinkArgs>
|
|
||||||
static std::shared_ptr<async_logger> create(std::string logger_name, SinkArgs &&... args)
|
|
||||||
{
|
|
||||||
auto ®istry_inst = details::registry::instance();
|
|
||||||
|
|
||||||
// create global thread pool if not already exists..
|
|
||||||
std::lock_guard<std::recursive_mutex> tp_lock(registry_inst.tp_mutex());
|
|
||||||
auto tp = registry_inst.get_tp();
|
|
||||||
if (tp == nullptr)
|
|
||||||
{
|
|
||||||
tp = std::make_shared<details::thread_pool>(details::default_async_q_size, 1);
|
|
||||||
registry_inst.set_tp(tp);
|
|
||||||
}
|
|
||||||
|
|
||||||
auto sink = std::make_shared<Sink>(std::forward<SinkArgs>(args)...);
|
|
||||||
auto new_logger = std::make_shared<async_logger>(std::move(logger_name), std::move(sink), std::move(tp), OverflowPolicy);
|
|
||||||
registry_inst.initialize_logger(new_logger);
|
|
||||||
return new_logger;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
using async_factory = async_factory_impl<async_overflow_policy::block>;
|
|
||||||
using async_factory_nonblock = async_factory_impl<async_overflow_policy::overrun_oldest>;
|
|
||||||
|
|
||||||
template<typename Sink, typename... SinkArgs>
|
|
||||||
inline std::shared_ptr<spdlog::logger> create_async(std::string logger_name, SinkArgs &&... sink_args)
|
|
||||||
{
|
|
||||||
return async_factory::create<Sink>(std::move(logger_name), std::forward<SinkArgs>(sink_args)...);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename Sink, typename... SinkArgs>
|
|
||||||
inline std::shared_ptr<spdlog::logger> create_async_nb(std::string logger_name, SinkArgs &&... sink_args)
|
|
||||||
{
|
|
||||||
return async_factory_nonblock::create<Sink>(std::move(logger_name), std::forward<SinkArgs>(sink_args)...);
|
|
||||||
}
|
|
||||||
|
|
||||||
// set global thread pool.
|
|
||||||
inline void init_thread_pool(size_t q_size, size_t thread_count)
|
|
||||||
{
|
|
||||||
auto tp = std::make_shared<details::thread_pool>(q_size, thread_count);
|
|
||||||
details::registry::instance().set_tp(std::move(tp));
|
|
||||||
}
|
|
||||||
|
|
||||||
// get the global thread pool.
|
|
||||||
inline std::shared_ptr<spdlog::details::thread_pool> thread_pool()
|
|
||||||
{
|
|
||||||
return details::registry::instance().get_tp();
|
|
||||||
}
|
|
||||||
} // namespace spdlog
|
|
||||||
73
deps/spdlog/async_logger.h
vendored
73
deps/spdlog/async_logger.h
vendored
@ -1,73 +0,0 @@
|
|||||||
//
|
|
||||||
// Copyright(c) 2015 Gabi Melman.
|
|
||||||
// Distributed under the MIT License (http://opensource.org/licenses/MIT)
|
|
||||||
//
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
// Very fast asynchronous logger (millions of logs per second on an average
|
|
||||||
// desktop)
|
|
||||||
// Uses pre allocated lockfree queue for maximum throughput even under large
|
|
||||||
// number of threads.
|
|
||||||
// Creates a single back thread to pop messages from the queue and log them.
|
|
||||||
//
|
|
||||||
// Upon each log write the logger:
|
|
||||||
// 1. Checks if its log level is enough to log the message
|
|
||||||
// 2. Push a new copy of the message to a queue (or block the caller until
|
|
||||||
// space is available in the queue)
|
|
||||||
// 3. will throw spdlog_ex upon log exceptions
|
|
||||||
// Upon destruction, logs all remaining messages in the queue before
|
|
||||||
// destructing..
|
|
||||||
|
|
||||||
#include "spdlog/common.h"
|
|
||||||
#include "spdlog/logger.h"
|
|
||||||
|
|
||||||
#include <chrono>
|
|
||||||
#include <memory>
|
|
||||||
#include <string>
|
|
||||||
|
|
||||||
namespace spdlog {
|
|
||||||
|
|
||||||
// Async overflow policy - block by default.
|
|
||||||
enum class async_overflow_policy
|
|
||||||
{
|
|
||||||
block, // Block until message can be enqueued
|
|
||||||
overrun_oldest // Discard oldest message in the queue if full when trying to
|
|
||||||
// add new item.
|
|
||||||
};
|
|
||||||
|
|
||||||
namespace details {
|
|
||||||
class thread_pool;
|
|
||||||
}
|
|
||||||
|
|
||||||
class async_logger final : public std::enable_shared_from_this<async_logger>, public logger
|
|
||||||
{
|
|
||||||
friend class details::thread_pool;
|
|
||||||
|
|
||||||
public:
|
|
||||||
template<typename It>
|
|
||||||
async_logger(std::string logger_name, It begin, It end, std::weak_ptr<details::thread_pool> tp,
|
|
||||||
async_overflow_policy overflow_policy = async_overflow_policy::block);
|
|
||||||
|
|
||||||
async_logger(std::string logger_name, sinks_init_list sinks_list, std::weak_ptr<details::thread_pool> tp,
|
|
||||||
async_overflow_policy overflow_policy = async_overflow_policy::block);
|
|
||||||
|
|
||||||
async_logger(std::string logger_name, sink_ptr single_sink, std::weak_ptr<details::thread_pool> tp,
|
|
||||||
async_overflow_policy overflow_policy = async_overflow_policy::block);
|
|
||||||
|
|
||||||
std::shared_ptr<logger> clone(std::string new_name) override;
|
|
||||||
|
|
||||||
protected:
|
|
||||||
void sink_it_(details::log_msg &msg) override;
|
|
||||||
void flush_() override;
|
|
||||||
|
|
||||||
void backend_log_(const details::log_msg &incoming_log_msg);
|
|
||||||
void backend_flush_();
|
|
||||||
|
|
||||||
private:
|
|
||||||
std::weak_ptr<details::thread_pool> thread_pool_;
|
|
||||||
async_overflow_policy overflow_policy_;
|
|
||||||
};
|
|
||||||
} // namespace spdlog
|
|
||||||
|
|
||||||
#include "details/async_logger_impl.h"
|
|
||||||
243
deps/spdlog/common.h
vendored
243
deps/spdlog/common.h
vendored
@ -1,243 +0,0 @@
|
|||||||
//
|
|
||||||
// Copyright(c) 2015 Gabi Melman.
|
|
||||||
// Distributed under the MIT License (http://opensource.org/licenses/MIT)
|
|
||||||
//
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include "spdlog/tweakme.h"
|
|
||||||
|
|
||||||
#include <atomic>
|
|
||||||
#include <chrono>
|
|
||||||
#include <functional>
|
|
||||||
#include <initializer_list>
|
|
||||||
#include <memory>
|
|
||||||
#include <stdexcept>
|
|
||||||
#include <string>
|
|
||||||
#include <cstring>
|
|
||||||
#include <type_traits>
|
|
||||||
#include <unordered_map>
|
|
||||||
|
|
||||||
#if defined(SPDLOG_WCHAR_FILENAMES) || defined(SPDLOG_WCHAR_TO_UTF8_SUPPORT)
|
|
||||||
#include <codecvt>
|
|
||||||
#include <locale>
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#include "spdlog/details/null_mutex.h"
|
|
||||||
|
|
||||||
#include "spdlog/fmt/fmt.h"
|
|
||||||
|
|
||||||
// visual studio upto 2013 does not support noexcept nor constexpr
|
|
||||||
#if defined(_MSC_VER) && (_MSC_VER < 1900)
|
|
||||||
#define SPDLOG_NOEXCEPT throw()
|
|
||||||
#define SPDLOG_CONSTEXPR
|
|
||||||
#else
|
|
||||||
#define SPDLOG_NOEXCEPT noexcept
|
|
||||||
#define SPDLOG_CONSTEXPR constexpr
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#if defined(__GNUC__) || defined(__clang__)
|
|
||||||
#define SPDLOG_DEPRECATED __attribute__((deprecated))
|
|
||||||
#elif defined(_MSC_VER)
|
|
||||||
#define SPDLOG_DEPRECATED __declspec(deprecated)
|
|
||||||
#else
|
|
||||||
#define SPDLOG_DEPRECATED
|
|
||||||
#endif
|
|
||||||
|
|
||||||
// disable thread local on msvc 2013
|
|
||||||
#ifndef SPDLOG_NO_TLS
|
|
||||||
#if (defined(_MSC_VER) && (_MSC_VER < 1900)) || defined(__cplusplus_winrt)
|
|
||||||
#define SPDLOG_NO_TLS 1
|
|
||||||
#endif
|
|
||||||
#endif
|
|
||||||
|
|
||||||
// Get the basename of __FILE__ (at compile time if possible)
|
|
||||||
#if FMT_HAS_FEATURE(__builtin_strrchr)
|
|
||||||
#define SPDLOG_STRRCHR(str, sep) __builtin_strrchr(str, sep)
|
|
||||||
#else
|
|
||||||
#define SPDLOG_STRRCHR(str, sep) strrchr(str, sep)
|
|
||||||
#endif //__builtin_strrchr not defined
|
|
||||||
|
|
||||||
#ifdef _WIN32
|
|
||||||
#define SPDLOG_FILE_BASENAME(file) SPDLOG_STRRCHR("\\" file, '\\') + 1
|
|
||||||
#else
|
|
||||||
#define SPDLOG_FILE_BASENAME(file) SPDLOG_STRRCHR("/" file, '/') + 1
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifndef SPDLOG_FUNCTION
|
|
||||||
#define SPDLOG_FUNCTION __FUNCTION__
|
|
||||||
#endif
|
|
||||||
|
|
||||||
namespace spdlog {
|
|
||||||
|
|
||||||
class formatter;
|
|
||||||
|
|
||||||
namespace sinks {
|
|
||||||
class sink;
|
|
||||||
}
|
|
||||||
|
|
||||||
using log_clock = std::chrono::system_clock;
|
|
||||||
using sink_ptr = std::shared_ptr<sinks::sink>;
|
|
||||||
using sinks_init_list = std::initializer_list<sink_ptr>;
|
|
||||||
using log_err_handler = std::function<void(const std::string &err_msg)>;
|
|
||||||
|
|
||||||
// string_view type - either std::string_view or fmt::string_view (pre c++17)
|
|
||||||
#if defined(FMT_USE_STD_STRING_VIEW)
|
|
||||||
using string_view_t = std::string_view;
|
|
||||||
#else
|
|
||||||
using string_view_t = fmt::string_view;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#if defined(SPDLOG_NO_ATOMIC_LEVELS)
|
|
||||||
using level_t = details::null_atomic_int;
|
|
||||||
#else
|
|
||||||
using level_t = std::atomic<int>;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#define SPDLOG_LEVEL_TRACE 0
|
|
||||||
#define SPDLOG_LEVEL_DEBUG 1
|
|
||||||
#define SPDLOG_LEVEL_INFO 2
|
|
||||||
#define SPDLOG_LEVEL_WARN 3
|
|
||||||
#define SPDLOG_LEVEL_ERROR 4
|
|
||||||
#define SPDLOG_LEVEL_CRITICAL 5
|
|
||||||
#define SPDLOG_LEVEL_OFF 6
|
|
||||||
|
|
||||||
#if !defined(SPDLOG_ACTIVE_LEVEL)
|
|
||||||
#define SPDLOG_ACTIVE_LEVEL SPDLOG_LEVEL_INFO
|
|
||||||
#endif
|
|
||||||
|
|
||||||
// Log level enum
|
|
||||||
namespace level {
|
|
||||||
enum level_enum
|
|
||||||
{
|
|
||||||
trace = SPDLOG_LEVEL_TRACE,
|
|
||||||
debug = SPDLOG_LEVEL_DEBUG,
|
|
||||||
info = SPDLOG_LEVEL_INFO,
|
|
||||||
warn = SPDLOG_LEVEL_WARN,
|
|
||||||
err = SPDLOG_LEVEL_ERROR,
|
|
||||||
critical = SPDLOG_LEVEL_CRITICAL,
|
|
||||||
off = SPDLOG_LEVEL_OFF,
|
|
||||||
};
|
|
||||||
|
|
||||||
#if !defined(SPDLOG_LEVEL_NAMES)
|
|
||||||
#define SPDLOG_LEVEL_NAMES \
|
|
||||||
{ \
|
|
||||||
"trace", "debug", "info", "warning", "error", "critical", "off" \
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
static string_view_t level_string_views[] SPDLOG_LEVEL_NAMES;
|
|
||||||
static const char *short_level_names[]{"T", "D", "I", "W", "E", "C", "O"};
|
|
||||||
|
|
||||||
inline string_view_t &to_string_view(spdlog::level::level_enum l) SPDLOG_NOEXCEPT
|
|
||||||
{
|
|
||||||
return level_string_views[l];
|
|
||||||
}
|
|
||||||
|
|
||||||
inline const char *to_short_c_str(spdlog::level::level_enum l) SPDLOG_NOEXCEPT
|
|
||||||
{
|
|
||||||
return short_level_names[l];
|
|
||||||
}
|
|
||||||
|
|
||||||
inline spdlog::level::level_enum from_str(const std::string &name) SPDLOG_NOEXCEPT
|
|
||||||
{
|
|
||||||
int level = 0;
|
|
||||||
for (const auto &level_str : level_string_views)
|
|
||||||
{
|
|
||||||
if (level_str == name)
|
|
||||||
{
|
|
||||||
return static_cast<level::level_enum>(level);
|
|
||||||
}
|
|
||||||
level++;
|
|
||||||
}
|
|
||||||
return level::off;
|
|
||||||
}
|
|
||||||
|
|
||||||
using level_hasher = std::hash<int>;
|
|
||||||
} // namespace level
|
|
||||||
|
|
||||||
//
|
|
||||||
// Pattern time - specific time getting to use for pattern_formatter.
|
|
||||||
// local time by default
|
|
||||||
//
|
|
||||||
enum class pattern_time_type
|
|
||||||
{
|
|
||||||
local, // log localtime
|
|
||||||
utc // log utc
|
|
||||||
};
|
|
||||||
|
|
||||||
//
|
|
||||||
// Log exception
|
|
||||||
//
|
|
||||||
class spdlog_ex : public std::exception
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
explicit spdlog_ex(std::string msg)
|
|
||||||
: msg_(std::move(msg))
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
spdlog_ex(const std::string &msg, int last_errno)
|
|
||||||
{
|
|
||||||
fmt::memory_buffer outbuf;
|
|
||||||
fmt::format_system_error(outbuf, last_errno, msg);
|
|
||||||
msg_ = fmt::to_string(outbuf);
|
|
||||||
}
|
|
||||||
|
|
||||||
const char *what() const SPDLOG_NOEXCEPT override
|
|
||||||
{
|
|
||||||
return msg_.c_str();
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
|
||||||
std::string msg_;
|
|
||||||
};
|
|
||||||
|
|
||||||
//
|
|
||||||
// wchar support for windows file names (SPDLOG_WCHAR_FILENAMES must be defined)
|
|
||||||
//
|
|
||||||
#if defined(_WIN32) && defined(SPDLOG_WCHAR_FILENAMES)
|
|
||||||
using filename_t = std::wstring;
|
|
||||||
#else
|
|
||||||
using filename_t = std::string;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
struct source_loc
|
|
||||||
{
|
|
||||||
SPDLOG_CONSTEXPR source_loc()
|
|
||||||
: filename{""}
|
|
||||||
, line{0}
|
|
||||||
, funcname{""}
|
|
||||||
{
|
|
||||||
}
|
|
||||||
SPDLOG_CONSTEXPR source_loc(const char *filename, int line, const char *funcname)
|
|
||||||
: filename{filename}
|
|
||||||
, line{static_cast<uint32_t>(line)}
|
|
||||||
, funcname{funcname}
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
SPDLOG_CONSTEXPR bool empty() const SPDLOG_NOEXCEPT
|
|
||||||
{
|
|
||||||
return line == 0;
|
|
||||||
}
|
|
||||||
const char *filename;
|
|
||||||
uint32_t line;
|
|
||||||
const char *funcname;
|
|
||||||
};
|
|
||||||
|
|
||||||
namespace details {
|
|
||||||
// make_unique support for pre c++14
|
|
||||||
|
|
||||||
#if __cplusplus >= 201402L // C++14 and beyond
|
|
||||||
using std::make_unique;
|
|
||||||
#else
|
|
||||||
template<typename T, typename... Args>
|
|
||||||
std::unique_ptr<T> make_unique(Args &&... args)
|
|
||||||
{
|
|
||||||
static_assert(!std::is_array<T>::value, "arrays not supported");
|
|
||||||
return std::unique_ptr<T>(new T(std::forward<Args>(args)...));
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
} // namespace details
|
|
||||||
} // namespace spdlog
|
|
||||||
110
deps/spdlog/details/async_logger_impl.h
vendored
110
deps/spdlog/details/async_logger_impl.h
vendored
@ -1,110 +0,0 @@
|
|||||||
//
|
|
||||||
// Copyright(c) 2015 Gabi Melman.
|
|
||||||
// Distributed under the MIT License (http://opensource.org/licenses/MIT)
|
|
||||||
//
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
// async logger implementation
|
|
||||||
// uses a thread pool to perform the actual logging
|
|
||||||
|
|
||||||
#include "spdlog/details/thread_pool.h"
|
|
||||||
|
|
||||||
#include <chrono>
|
|
||||||
#include <memory>
|
|
||||||
#include <string>
|
|
||||||
|
|
||||||
template<typename It>
|
|
||||||
inline spdlog::async_logger::async_logger(
|
|
||||||
std::string logger_name, It begin, It end, std::weak_ptr<details::thread_pool> tp, async_overflow_policy overflow_policy)
|
|
||||||
: logger(std::move(logger_name), begin, end)
|
|
||||||
, thread_pool_(std::move(tp))
|
|
||||||
, overflow_policy_(overflow_policy)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
inline spdlog::async_logger::async_logger(
|
|
||||||
std::string logger_name, sinks_init_list sinks_list, std::weak_ptr<details::thread_pool> tp, async_overflow_policy overflow_policy)
|
|
||||||
: async_logger(std::move(logger_name), sinks_list.begin(), sinks_list.end(), std::move(tp), overflow_policy)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
inline spdlog::async_logger::async_logger(
|
|
||||||
std::string logger_name, sink_ptr single_sink, std::weak_ptr<details::thread_pool> tp, async_overflow_policy overflow_policy)
|
|
||||||
: async_logger(std::move(logger_name), {std::move(single_sink)}, std::move(tp), overflow_policy)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
// send the log message to the thread pool
|
|
||||||
inline void spdlog::async_logger::sink_it_(details::log_msg &msg)
|
|
||||||
{
|
|
||||||
#if defined(SPDLOG_ENABLE_MESSAGE_COUNTER)
|
|
||||||
incr_msg_counter_(msg);
|
|
||||||
#endif
|
|
||||||
if (auto pool_ptr = thread_pool_.lock())
|
|
||||||
{
|
|
||||||
pool_ptr->post_log(shared_from_this(), msg, overflow_policy_);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
throw spdlog_ex("async log: thread pool doesn't exist anymore");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// send flush request to the thread pool
|
|
||||||
inline void spdlog::async_logger::flush_()
|
|
||||||
{
|
|
||||||
if (auto pool_ptr = thread_pool_.lock())
|
|
||||||
{
|
|
||||||
pool_ptr->post_flush(shared_from_this(), overflow_policy_);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
throw spdlog_ex("async flush: thread pool doesn't exist anymore");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
//
|
|
||||||
// backend functions - called from the thread pool to do the actual job
|
|
||||||
//
|
|
||||||
inline void spdlog::async_logger::backend_log_(const details::log_msg &incoming_log_msg)
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
for (auto &s : sinks_)
|
|
||||||
{
|
|
||||||
if (s->should_log(incoming_log_msg.level))
|
|
||||||
{
|
|
||||||
s->log(incoming_log_msg);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
SPDLOG_CATCH_AND_HANDLE
|
|
||||||
|
|
||||||
if (should_flush_(incoming_log_msg))
|
|
||||||
{
|
|
||||||
backend_flush_();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
inline void spdlog::async_logger::backend_flush_()
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
for (auto &sink : sinks_)
|
|
||||||
{
|
|
||||||
sink->flush();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
SPDLOG_CATCH_AND_HANDLE
|
|
||||||
}
|
|
||||||
|
|
||||||
inline std::shared_ptr<spdlog::logger> spdlog::async_logger::clone(std::string new_name)
|
|
||||||
{
|
|
||||||
auto cloned = std::make_shared<spdlog::async_logger>(std::move(new_name), sinks_.begin(), sinks_.end(), thread_pool_, overflow_policy_);
|
|
||||||
|
|
||||||
cloned->set_level(this->level());
|
|
||||||
cloned->flush_on(this->flush_level());
|
|
||||||
cloned->set_error_handler(this->error_handler());
|
|
||||||
return std::move(cloned);
|
|
||||||
}
|
|
||||||
72
deps/spdlog/details/circular_q.h
vendored
72
deps/spdlog/details/circular_q.h
vendored
@ -1,72 +0,0 @@
|
|||||||
//
|
|
||||||
// Copyright(c) 2018 Gabi Melman.
|
|
||||||
// Distributed under the MIT License (http://opensource.org/licenses/MIT)
|
|
||||||
//
|
|
||||||
|
|
||||||
// cirucal q view of std::vector.
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include <vector>
|
|
||||||
|
|
||||||
namespace spdlog {
|
|
||||||
namespace details {
|
|
||||||
template<typename T>
|
|
||||||
class circular_q
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
using item_type = T;
|
|
||||||
|
|
||||||
explicit circular_q(size_t max_items)
|
|
||||||
: max_items_(max_items + 1) // one item is reserved as marker for full q
|
|
||||||
, v_(max_items_)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
// push back, overrun (oldest) item if no room left
|
|
||||||
void push_back(T &&item)
|
|
||||||
{
|
|
||||||
v_[tail_] = std::move(item);
|
|
||||||
tail_ = (tail_ + 1) % max_items_;
|
|
||||||
|
|
||||||
if (tail_ == head_) // overrun last item if full
|
|
||||||
{
|
|
||||||
head_ = (head_ + 1) % max_items_;
|
|
||||||
++overrun_counter_;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Pop item from front.
|
|
||||||
// If there are no elements in the container, the behavior is undefined.
|
|
||||||
void pop_front(T &popped_item)
|
|
||||||
{
|
|
||||||
popped_item = std::move(v_[head_]);
|
|
||||||
head_ = (head_ + 1) % max_items_;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool empty()
|
|
||||||
{
|
|
||||||
return tail_ == head_;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool full()
|
|
||||||
{
|
|
||||||
// head is ahead of the tail by 1
|
|
||||||
return ((tail_ + 1) % max_items_) == head_;
|
|
||||||
}
|
|
||||||
|
|
||||||
size_t overrun_counter() const
|
|
||||||
{
|
|
||||||
return overrun_counter_;
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
|
||||||
size_t max_items_;
|
|
||||||
typename std::vector<T>::size_type head_ = 0;
|
|
||||||
typename std::vector<T>::size_type tail_ = 0;
|
|
||||||
|
|
||||||
std::vector<T> v_;
|
|
||||||
|
|
||||||
size_t overrun_counter_ = 0;
|
|
||||||
};
|
|
||||||
} // namespace details
|
|
||||||
} // namespace spdlog
|
|
||||||
74
deps/spdlog/details/console_globals.h
vendored
74
deps/spdlog/details/console_globals.h
vendored
@ -1,74 +0,0 @@
|
|||||||
#pragma once
|
|
||||||
//
|
|
||||||
// Copyright(c) 2018 Gabi Melman.
|
|
||||||
// Distributed under the MIT License (http://opensource.org/licenses/MIT)
|
|
||||||
//
|
|
||||||
|
|
||||||
#include "spdlog/details/null_mutex.h"
|
|
||||||
#include <cstdio>
|
|
||||||
#include <mutex>
|
|
||||||
|
|
||||||
#ifdef _WIN32
|
|
||||||
|
|
||||||
#ifndef NOMINMAX
|
|
||||||
#define NOMINMAX // prevent windows redefining min/max
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifndef WIN32_LEAN_AND_MEAN
|
|
||||||
#define WIN32_LEAN_AND_MEAN
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#include <windows.h>
|
|
||||||
#endif
|
|
||||||
|
|
||||||
namespace spdlog {
|
|
||||||
namespace details {
|
|
||||||
struct console_stdout
|
|
||||||
{
|
|
||||||
static std::FILE *stream()
|
|
||||||
{
|
|
||||||
return stdout;
|
|
||||||
}
|
|
||||||
#ifdef _WIN32
|
|
||||||
static HANDLE handle()
|
|
||||||
{
|
|
||||||
return ::GetStdHandle(STD_OUTPUT_HANDLE);
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
};
|
|
||||||
|
|
||||||
struct console_stderr
|
|
||||||
{
|
|
||||||
static std::FILE *stream()
|
|
||||||
{
|
|
||||||
return stderr;
|
|
||||||
}
|
|
||||||
#ifdef _WIN32
|
|
||||||
static HANDLE handle()
|
|
||||||
{
|
|
||||||
return ::GetStdHandle(STD_ERROR_HANDLE);
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
};
|
|
||||||
|
|
||||||
struct console_mutex
|
|
||||||
{
|
|
||||||
using mutex_t = std::mutex;
|
|
||||||
static mutex_t &mutex()
|
|
||||||
{
|
|
||||||
static mutex_t s_mutex;
|
|
||||||
return s_mutex;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
struct console_nullmutex
|
|
||||||
{
|
|
||||||
using mutex_t = null_mutex;
|
|
||||||
static mutex_t &mutex()
|
|
||||||
{
|
|
||||||
static mutex_t s_mutex;
|
|
||||||
return s_mutex;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
} // namespace details
|
|
||||||
} // namespace spdlog
|
|
||||||
152
deps/spdlog/details/file_helper.h
vendored
152
deps/spdlog/details/file_helper.h
vendored
@ -1,152 +0,0 @@
|
|||||||
//
|
|
||||||
// Copyright(c) 2015 Gabi Melman.
|
|
||||||
// Distributed under the MIT License (http://opensource.org/licenses/MIT)
|
|
||||||
//
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
// Helper class for file sinks.
|
|
||||||
// When failing to open a file, retry several times(5) with a delay interval(10 ms).
|
|
||||||
// Throw spdlog_ex exception on errors.
|
|
||||||
|
|
||||||
#include "spdlog/details/log_msg.h"
|
|
||||||
#include "spdlog/details/os.h"
|
|
||||||
|
|
||||||
#include <cerrno>
|
|
||||||
#include <chrono>
|
|
||||||
#include <cstdio>
|
|
||||||
#include <string>
|
|
||||||
#include <thread>
|
|
||||||
#include <tuple>
|
|
||||||
|
|
||||||
namespace spdlog {
|
|
||||||
namespace details {
|
|
||||||
|
|
||||||
class file_helper
|
|
||||||
{
|
|
||||||
|
|
||||||
public:
|
|
||||||
const int open_tries = 5;
|
|
||||||
const int open_interval = 10;
|
|
||||||
|
|
||||||
explicit file_helper() = default;
|
|
||||||
|
|
||||||
file_helper(const file_helper &) = delete;
|
|
||||||
file_helper &operator=(const file_helper &) = delete;
|
|
||||||
|
|
||||||
~file_helper()
|
|
||||||
{
|
|
||||||
close();
|
|
||||||
}
|
|
||||||
|
|
||||||
void open(const filename_t &fname, bool truncate = false)
|
|
||||||
{
|
|
||||||
close();
|
|
||||||
auto *mode = truncate ? SPDLOG_FILENAME_T("wb") : SPDLOG_FILENAME_T("ab");
|
|
||||||
_filename = fname;
|
|
||||||
for (int tries = 0; tries < open_tries; ++tries)
|
|
||||||
{
|
|
||||||
if (!os::fopen_s(&fd_, fname, mode))
|
|
||||||
{
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
details::os::sleep_for_millis(open_interval);
|
|
||||||
}
|
|
||||||
|
|
||||||
throw spdlog_ex("Failed opening file " + os::filename_to_str(_filename) + " for writing", errno);
|
|
||||||
}
|
|
||||||
|
|
||||||
void reopen(bool truncate)
|
|
||||||
{
|
|
||||||
if (_filename.empty())
|
|
||||||
{
|
|
||||||
throw spdlog_ex("Failed re opening file - was not opened before");
|
|
||||||
}
|
|
||||||
open(_filename, truncate);
|
|
||||||
}
|
|
||||||
|
|
||||||
void flush()
|
|
||||||
{
|
|
||||||
std::fflush(fd_);
|
|
||||||
}
|
|
||||||
|
|
||||||
void close()
|
|
||||||
{
|
|
||||||
if (fd_ != nullptr)
|
|
||||||
{
|
|
||||||
std::fclose(fd_);
|
|
||||||
fd_ = nullptr;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void write(const fmt::memory_buffer &buf)
|
|
||||||
{
|
|
||||||
size_t msg_size = buf.size();
|
|
||||||
auto data = buf.data();
|
|
||||||
if (std::fwrite(data, 1, msg_size, fd_) != msg_size)
|
|
||||||
{
|
|
||||||
throw spdlog_ex("Failed writing to file " + os::filename_to_str(_filename), errno);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
size_t size() const
|
|
||||||
{
|
|
||||||
if (fd_ == nullptr)
|
|
||||||
{
|
|
||||||
throw spdlog_ex("Cannot use size() on closed file " + os::filename_to_str(_filename));
|
|
||||||
}
|
|
||||||
return os::filesize(fd_);
|
|
||||||
}
|
|
||||||
|
|
||||||
const filename_t &filename() const
|
|
||||||
{
|
|
||||||
return _filename;
|
|
||||||
}
|
|
||||||
|
|
||||||
static bool file_exists(const filename_t &fname)
|
|
||||||
{
|
|
||||||
return os::file_exists(fname);
|
|
||||||
}
|
|
||||||
|
|
||||||
//
|
|
||||||
// return file path and its extension:
|
|
||||||
//
|
|
||||||
// "mylog.txt" => ("mylog", ".txt")
|
|
||||||
// "mylog" => ("mylog", "")
|
|
||||||
// "mylog." => ("mylog.", "")
|
|
||||||
// "/dir1/dir2/mylog.txt" => ("/dir1/dir2/mylog", ".txt")
|
|
||||||
//
|
|
||||||
// the starting dot in filenames is ignored (hidden files):
|
|
||||||
//
|
|
||||||
// ".mylog" => (".mylog". "")
|
|
||||||
// "my_folder/.mylog" => ("my_folder/.mylog", "")
|
|
||||||
// "my_folder/.mylog.txt" => ("my_folder/.mylog", ".txt")
|
|
||||||
static std::tuple<filename_t, filename_t> split_by_extension(const spdlog::filename_t &fname)
|
|
||||||
{
|
|
||||||
auto ext_index = fname.rfind('.');
|
|
||||||
|
|
||||||
// no valid extension found - return whole path and empty string as
|
|
||||||
// extension
|
|
||||||
if (ext_index == filename_t::npos || ext_index == 0 || ext_index == fname.size() - 1)
|
|
||||||
{
|
|
||||||
return std::make_tuple(fname, spdlog::filename_t());
|
|
||||||
}
|
|
||||||
|
|
||||||
// treat casese like "/etc/rc.d/somelogfile or "/abc/.hiddenfile"
|
|
||||||
auto folder_index = fname.rfind(details::os::folder_sep);
|
|
||||||
if (folder_index != filename_t::npos && folder_index >= ext_index - 1)
|
|
||||||
{
|
|
||||||
return std::make_tuple(fname, spdlog::filename_t());
|
|
||||||
}
|
|
||||||
|
|
||||||
// finally - return a valid base and extension tuple
|
|
||||||
return std::make_tuple(fname.substr(0, ext_index), fname.substr(ext_index));
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
|
||||||
std::FILE *fd_{nullptr};
|
|
||||||
filename_t _filename;
|
|
||||||
};
|
|
||||||
} // namespace details
|
|
||||||
} // namespace spdlog
|
|
||||||
122
deps/spdlog/details/fmt_helper.h
vendored
122
deps/spdlog/details/fmt_helper.h
vendored
@ -1,122 +0,0 @@
|
|||||||
//
|
|
||||||
// Created by gabi on 6/15/18.
|
|
||||||
//
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include <chrono>
|
|
||||||
#include <type_traits>
|
|
||||||
#include "spdlog/fmt/fmt.h"
|
|
||||||
|
|
||||||
// Some fmt helpers to efficiently format and pad ints and strings
|
|
||||||
namespace spdlog {
|
|
||||||
namespace details {
|
|
||||||
namespace fmt_helper {
|
|
||||||
|
|
||||||
template<size_t Buffer_Size>
|
|
||||||
inline spdlog::string_view_t to_string_view(const fmt::basic_memory_buffer<char, Buffer_Size> &buf) SPDLOG_NOEXCEPT
|
|
||||||
{
|
|
||||||
return spdlog::string_view_t(buf.data(), buf.size());
|
|
||||||
}
|
|
||||||
|
|
||||||
template<size_t Buffer_Size1, size_t Buffer_Size2>
|
|
||||||
inline void append_buf(const fmt::basic_memory_buffer<char, Buffer_Size1> &buf, fmt::basic_memory_buffer<char, Buffer_Size2> &dest)
|
|
||||||
{
|
|
||||||
auto *buf_ptr = buf.data();
|
|
||||||
dest.append(buf_ptr, buf_ptr + buf.size());
|
|
||||||
}
|
|
||||||
|
|
||||||
template<size_t Buffer_Size>
|
|
||||||
inline void append_string_view(spdlog::string_view_t view, fmt::basic_memory_buffer<char, Buffer_Size> &dest)
|
|
||||||
{
|
|
||||||
auto *buf_ptr = view.data();
|
|
||||||
if (buf_ptr != nullptr)
|
|
||||||
{
|
|
||||||
dest.append(buf_ptr, buf_ptr + view.size());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename T, size_t Buffer_Size>
|
|
||||||
inline void append_int(T n, fmt::basic_memory_buffer<char, Buffer_Size> &dest)
|
|
||||||
{
|
|
||||||
fmt::format_int i(n);
|
|
||||||
dest.append(i.data(), i.data() + i.size());
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename T>
|
|
||||||
inline unsigned count_digits(T n)
|
|
||||||
{
|
|
||||||
using count_type = typename std::conditional<(sizeof(T) > sizeof(uint32_t)), uint64_t, uint32_t>::type;
|
|
||||||
return static_cast<unsigned>(fmt::internal::count_digits(static_cast<count_type>(n)));
|
|
||||||
}
|
|
||||||
|
|
||||||
template<size_t Buffer_Size>
|
|
||||||
inline void pad2(int n, fmt::basic_memory_buffer<char, Buffer_Size> &dest)
|
|
||||||
{
|
|
||||||
if (n > 99)
|
|
||||||
{
|
|
||||||
append_int(n, dest);
|
|
||||||
}
|
|
||||||
else if (n > 9) // 10-99
|
|
||||||
{
|
|
||||||
dest.push_back(static_cast<char>('0' + n / 10));
|
|
||||||
dest.push_back(static_cast<char>('0' + n % 10));
|
|
||||||
}
|
|
||||||
else if (n >= 0) // 0-9
|
|
||||||
{
|
|
||||||
dest.push_back('0');
|
|
||||||
dest.push_back(static_cast<char>('0' + n));
|
|
||||||
}
|
|
||||||
else // negatives (unlikely, but just in case, let fmt deal with it)
|
|
||||||
{
|
|
||||||
fmt::format_to(dest, "{:02}", n);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename T, size_t Buffer_Size>
|
|
||||||
inline void pad_uint(T n, unsigned int width, fmt::basic_memory_buffer<char, Buffer_Size> &dest)
|
|
||||||
{
|
|
||||||
static_assert(std::is_unsigned<T>::value, "pad_uint must get unsigned T");
|
|
||||||
auto digits = count_digits(n);
|
|
||||||
if (width > digits)
|
|
||||||
{
|
|
||||||
const char *zeroes = "0000000000000000000";
|
|
||||||
dest.append(zeroes, zeroes + width - digits);
|
|
||||||
}
|
|
||||||
append_int(n, dest);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename T, size_t Buffer_Size>
|
|
||||||
inline void pad3(T n, fmt::basic_memory_buffer<char, Buffer_Size> &dest)
|
|
||||||
{
|
|
||||||
pad_uint(n, 3, dest);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename T, size_t Buffer_Size>
|
|
||||||
inline void pad6(T n, fmt::basic_memory_buffer<char, Buffer_Size> &dest)
|
|
||||||
{
|
|
||||||
pad_uint(n, 6, dest);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename T, size_t Buffer_Size>
|
|
||||||
inline void pad9(T n, fmt::basic_memory_buffer<char, Buffer_Size> &dest)
|
|
||||||
{
|
|
||||||
pad_uint(n, 9, dest);
|
|
||||||
}
|
|
||||||
|
|
||||||
// return fraction of a second of the given time_point.
|
|
||||||
// e.g.
|
|
||||||
// fraction<std::milliseconds>(tp) -> will return the millis part of the second
|
|
||||||
template<typename ToDuration>
|
|
||||||
inline ToDuration time_fraction(const log_clock::time_point &tp)
|
|
||||||
{
|
|
||||||
using std::chrono::duration_cast;
|
|
||||||
using std::chrono::seconds;
|
|
||||||
auto duration = tp.time_since_epoch();
|
|
||||||
auto secs = duration_cast<seconds>(duration);
|
|
||||||
return duration_cast<ToDuration>(duration) - duration_cast<ToDuration>(secs);
|
|
||||||
}
|
|
||||||
|
|
||||||
} // namespace fmt_helper
|
|
||||||
} // namespace details
|
|
||||||
} // namespace spdlog
|
|
||||||
55
deps/spdlog/details/log_msg.h
vendored
55
deps/spdlog/details/log_msg.h
vendored
@ -1,55 +0,0 @@
|
|||||||
//
|
|
||||||
// Copyright(c) 2015 Gabi Melman.
|
|
||||||
// Distributed under the MIT License (http://opensource.org/licenses/MIT)
|
|
||||||
//
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include "spdlog/common.h"
|
|
||||||
#include "spdlog/details/os.h"
|
|
||||||
|
|
||||||
#include <string>
|
|
||||||
#include <utility>
|
|
||||||
|
|
||||||
namespace spdlog {
|
|
||||||
namespace details {
|
|
||||||
struct log_msg
|
|
||||||
{
|
|
||||||
|
|
||||||
log_msg(source_loc loc, const std::string *loggers_name, level::level_enum lvl, string_view_t view)
|
|
||||||
: logger_name(loggers_name)
|
|
||||||
, level(lvl)
|
|
||||||
#ifndef SPDLOG_NO_DATETIME
|
|
||||||
, time(os::now())
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifndef SPDLOG_NO_THREAD_ID
|
|
||||||
, thread_id(os::thread_id())
|
|
||||||
#endif
|
|
||||||
, source(loc)
|
|
||||||
, payload(view)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
log_msg(const std::string *loggers_name, level::level_enum lvl, string_view_t view)
|
|
||||||
: log_msg(source_loc{}, loggers_name, lvl, view)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
log_msg(const log_msg &other) = default;
|
|
||||||
|
|
||||||
const std::string *logger_name{nullptr};
|
|
||||||
level::level_enum level{level::off};
|
|
||||||
log_clock::time_point time;
|
|
||||||
size_t thread_id{0};
|
|
||||||
size_t msg_id{0};
|
|
||||||
|
|
||||||
// wrapping the formatted text with color (updated by pattern_formatter).
|
|
||||||
mutable size_t color_range_start{0};
|
|
||||||
mutable size_t color_range_end{0};
|
|
||||||
|
|
||||||
source_loc source;
|
|
||||||
const string_view_t payload;
|
|
||||||
};
|
|
||||||
} // namespace details
|
|
||||||
} // namespace spdlog
|
|
||||||
441
deps/spdlog/details/logger_impl.h
vendored
441
deps/spdlog/details/logger_impl.h
vendored
@ -1,441 +0,0 @@
|
|||||||
//
|
|
||||||
// Copyright(c) 2015 Gabi Melman.
|
|
||||||
// Distributed under the MIT License (http://opensource.org/licenses/MIT)
|
|
||||||
//
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include "spdlog/details/fmt_helper.h"
|
|
||||||
|
|
||||||
#include <memory>
|
|
||||||
#include <string>
|
|
||||||
|
|
||||||
#define SPDLOG_CATCH_AND_HANDLE \
|
|
||||||
catch (const std::exception &ex) \
|
|
||||||
{ \
|
|
||||||
err_handler_(ex.what()); \
|
|
||||||
} \
|
|
||||||
catch (...) \
|
|
||||||
{ \
|
|
||||||
err_handler_("Unknown exception in logger"); \
|
|
||||||
}
|
|
||||||
|
|
||||||
// create logger with given name, sinks and the default pattern formatter
|
|
||||||
// all other ctors will call this one
|
|
||||||
template<typename It>
|
|
||||||
inline spdlog::logger::logger(std::string logger_name, It begin, It end)
|
|
||||||
: name_(std::move(logger_name))
|
|
||||||
, sinks_(begin, end)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
// ctor with sinks as init list
|
|
||||||
inline spdlog::logger::logger(std::string logger_name, sinks_init_list sinks_list)
|
|
||||||
: logger(std::move(logger_name), sinks_list.begin(), sinks_list.end())
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
// ctor with single sink
|
|
||||||
inline spdlog::logger::logger(std::string logger_name, spdlog::sink_ptr single_sink)
|
|
||||||
: logger(std::move(logger_name), {std::move(single_sink)})
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
inline spdlog::logger::~logger() = default;
|
|
||||||
|
|
||||||
inline void spdlog::logger::set_formatter(std::unique_ptr<spdlog::formatter> f)
|
|
||||||
{
|
|
||||||
for (auto &sink : sinks_)
|
|
||||||
{
|
|
||||||
sink->set_formatter(f->clone());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
inline void spdlog::logger::set_pattern(std::string pattern, pattern_time_type time_type)
|
|
||||||
{
|
|
||||||
auto new_formatter = details::make_unique<spdlog::pattern_formatter>(std::move(pattern), time_type);
|
|
||||||
set_formatter(std::move(new_formatter));
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename... Args>
|
|
||||||
inline void spdlog::logger::log(source_loc source, level::level_enum lvl, const char *fmt, const Args &... args)
|
|
||||||
{
|
|
||||||
if (!should_log(lvl))
|
|
||||||
{
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
try
|
|
||||||
{
|
|
||||||
using details::fmt_helper::to_string_view;
|
|
||||||
fmt::memory_buffer buf;
|
|
||||||
fmt::format_to(buf, fmt, args...);
|
|
||||||
details::log_msg log_msg(source, &name_, lvl, to_string_view(buf));
|
|
||||||
sink_it_(log_msg);
|
|
||||||
}
|
|
||||||
SPDLOG_CATCH_AND_HANDLE
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename... Args>
|
|
||||||
inline void spdlog::logger::log(level::level_enum lvl, const char *fmt, const Args &... args)
|
|
||||||
{
|
|
||||||
log(source_loc{}, lvl, fmt, args...);
|
|
||||||
}
|
|
||||||
|
|
||||||
inline void spdlog::logger::log(source_loc source, level::level_enum lvl, const char *msg)
|
|
||||||
{
|
|
||||||
if (!should_log(lvl))
|
|
||||||
{
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
try
|
|
||||||
{
|
|
||||||
details::log_msg log_msg(source, &name_, lvl, spdlog::string_view_t(msg));
|
|
||||||
sink_it_(log_msg);
|
|
||||||
}
|
|
||||||
SPDLOG_CATCH_AND_HANDLE
|
|
||||||
}
|
|
||||||
|
|
||||||
inline void spdlog::logger::log(level::level_enum lvl, const char *msg)
|
|
||||||
{
|
|
||||||
log(source_loc{}, lvl, msg);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<class T, typename std::enable_if<std::is_convertible<T, spdlog::string_view_t>::value, T>::type *>
|
|
||||||
inline void spdlog::logger::log(source_loc source, level::level_enum lvl, const T &msg)
|
|
||||||
{
|
|
||||||
if (!should_log(lvl))
|
|
||||||
{
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
try
|
|
||||||
{
|
|
||||||
details::log_msg log_msg(source, &name_, lvl, msg);
|
|
||||||
sink_it_(log_msg);
|
|
||||||
}
|
|
||||||
SPDLOG_CATCH_AND_HANDLE
|
|
||||||
}
|
|
||||||
|
|
||||||
template<class T, typename std::enable_if<std::is_convertible<T, spdlog::string_view_t>::value, T>::type *>
|
|
||||||
inline void spdlog::logger::log(level::level_enum lvl, const T &msg)
|
|
||||||
{
|
|
||||||
log(source_loc{}, lvl, msg);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<class T, typename std::enable_if<!std::is_convertible<T, spdlog::string_view_t>::value, T>::type *>
|
|
||||||
inline void spdlog::logger::log(source_loc source, level::level_enum lvl, const T &msg)
|
|
||||||
{
|
|
||||||
if (!should_log(lvl))
|
|
||||||
{
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
try
|
|
||||||
{
|
|
||||||
using details::fmt_helper::to_string_view;
|
|
||||||
fmt::memory_buffer buf;
|
|
||||||
fmt::format_to(buf, "{}", msg);
|
|
||||||
details::log_msg log_msg(source, &name_, lvl, to_string_view(buf));
|
|
||||||
sink_it_(log_msg);
|
|
||||||
}
|
|
||||||
SPDLOG_CATCH_AND_HANDLE
|
|
||||||
}
|
|
||||||
|
|
||||||
template<class T, typename std::enable_if<!std::is_convertible<T, spdlog::string_view_t>::value, T>::type *>
|
|
||||||
inline void spdlog::logger::log(level::level_enum lvl, const T &msg)
|
|
||||||
{
|
|
||||||
log(source_loc{}, lvl, msg);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename... Args>
|
|
||||||
inline void spdlog::logger::trace(const char *fmt, const Args &... args)
|
|
||||||
{
|
|
||||||
log(level::trace, fmt, args...);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename... Args>
|
|
||||||
inline void spdlog::logger::debug(const char *fmt, const Args &... args)
|
|
||||||
{
|
|
||||||
log(level::debug, fmt, args...);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename... Args>
|
|
||||||
inline void spdlog::logger::info(const char *fmt, const Args &... args)
|
|
||||||
{
|
|
||||||
log(level::info, fmt, args...);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename... Args>
|
|
||||||
inline void spdlog::logger::warn(const char *fmt, const Args &... args)
|
|
||||||
{
|
|
||||||
log(level::warn, fmt, args...);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename... Args>
|
|
||||||
inline void spdlog::logger::error(const char *fmt, const Args &... args)
|
|
||||||
{
|
|
||||||
log(level::err, fmt, args...);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename... Args>
|
|
||||||
inline void spdlog::logger::critical(const char *fmt, const Args &... args)
|
|
||||||
{
|
|
||||||
log(level::critical, fmt, args...);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename T>
|
|
||||||
inline void spdlog::logger::trace(const T &msg)
|
|
||||||
{
|
|
||||||
log(level::trace, msg);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename T>
|
|
||||||
inline void spdlog::logger::debug(const T &msg)
|
|
||||||
{
|
|
||||||
log(level::debug, msg);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename T>
|
|
||||||
inline void spdlog::logger::info(const T &msg)
|
|
||||||
{
|
|
||||||
log(level::info, msg);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename T>
|
|
||||||
inline void spdlog::logger::warn(const T &msg)
|
|
||||||
{
|
|
||||||
log(level::warn, msg);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename T>
|
|
||||||
inline void spdlog::logger::error(const T &msg)
|
|
||||||
{
|
|
||||||
log(level::err, msg);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename T>
|
|
||||||
inline void spdlog::logger::critical(const T &msg)
|
|
||||||
{
|
|
||||||
log(level::critical, msg);
|
|
||||||
}
|
|
||||||
|
|
||||||
#ifdef SPDLOG_WCHAR_TO_UTF8_SUPPORT
|
|
||||||
|
|
||||||
inline void wbuf_to_utf8buf(const fmt::wmemory_buffer &wbuf, fmt::memory_buffer &target)
|
|
||||||
{
|
|
||||||
int wbuf_size = static_cast<int>(wbuf.size());
|
|
||||||
if (wbuf_size == 0)
|
|
||||||
{
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
auto result_size = ::WideCharToMultiByte(CP_UTF8, 0, wbuf.data(), wbuf_size, NULL, 0, NULL, NULL);
|
|
||||||
|
|
||||||
if (result_size > 0)
|
|
||||||
{
|
|
||||||
target.resize(result_size);
|
|
||||||
::WideCharToMultiByte(CP_UTF8, 0, wbuf.data(), wbuf_size, &target.data()[0], result_size, NULL, NULL);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
throw spdlog::spdlog_ex(fmt::format("WideCharToMultiByte failed. Last error: {}", ::GetLastError()));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename... Args>
|
|
||||||
inline void spdlog::logger::log(source_loc source, level::level_enum lvl, const wchar_t *fmt, const Args &... args)
|
|
||||||
{
|
|
||||||
if (!should_log(lvl))
|
|
||||||
{
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
try
|
|
||||||
{
|
|
||||||
// format to wmemory_buffer and convert to utf8
|
|
||||||
using details::fmt_helper::to_string_view;
|
|
||||||
fmt::wmemory_buffer wbuf;
|
|
||||||
fmt::format_to(wbuf, fmt, args...);
|
|
||||||
fmt::memory_buffer buf;
|
|
||||||
wbuf_to_utf8buf(wbuf, buf);
|
|
||||||
details::log_msg log_msg(source, &name_, lvl, to_string_view(buf));
|
|
||||||
sink_it_(log_msg);
|
|
||||||
}
|
|
||||||
SPDLOG_CATCH_AND_HANDLE
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename... Args>
|
|
||||||
inline void spdlog::logger::log(level::level_enum lvl, const wchar_t *fmt, const Args &... args)
|
|
||||||
{
|
|
||||||
log(source_loc{}, lvl, fmt, args...);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename... Args>
|
|
||||||
inline void spdlog::logger::trace(const wchar_t *fmt, const Args &... args)
|
|
||||||
{
|
|
||||||
log(level::trace, fmt, args...);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename... Args>
|
|
||||||
inline void spdlog::logger::debug(const wchar_t *fmt, const Args &... args)
|
|
||||||
{
|
|
||||||
log(level::debug, fmt, args...);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename... Args>
|
|
||||||
inline void spdlog::logger::info(const wchar_t *fmt, const Args &... args)
|
|
||||||
{
|
|
||||||
log(level::info, fmt, args...);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename... Args>
|
|
||||||
inline void spdlog::logger::warn(const wchar_t *fmt, const Args &... args)
|
|
||||||
{
|
|
||||||
log(level::warn, fmt, args...);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename... Args>
|
|
||||||
inline void spdlog::logger::error(const wchar_t *fmt, const Args &... args)
|
|
||||||
{
|
|
||||||
log(level::err, fmt, args...);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename... Args>
|
|
||||||
inline void spdlog::logger::critical(const wchar_t *fmt, const Args &... args)
|
|
||||||
{
|
|
||||||
log(level::critical, fmt, args...);
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif // SPDLOG_WCHAR_TO_UTF8_SUPPORT
|
|
||||||
|
|
||||||
//
|
|
||||||
// name and level
|
|
||||||
//
|
|
||||||
inline const std::string &spdlog::logger::name() const
|
|
||||||
{
|
|
||||||
return name_;
|
|
||||||
}
|
|
||||||
|
|
||||||
inline void spdlog::logger::set_level(spdlog::level::level_enum log_level)
|
|
||||||
{
|
|
||||||
level_.store(log_level);
|
|
||||||
}
|
|
||||||
|
|
||||||
inline void spdlog::logger::set_error_handler(spdlog::log_err_handler err_handler)
|
|
||||||
{
|
|
||||||
err_handler_ = std::move(err_handler);
|
|
||||||
}
|
|
||||||
|
|
||||||
inline spdlog::log_err_handler spdlog::logger::error_handler() const
|
|
||||||
{
|
|
||||||
return err_handler_;
|
|
||||||
}
|
|
||||||
|
|
||||||
inline void spdlog::logger::flush()
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
flush_();
|
|
||||||
}
|
|
||||||
SPDLOG_CATCH_AND_HANDLE
|
|
||||||
}
|
|
||||||
|
|
||||||
inline void spdlog::logger::flush_on(level::level_enum log_level)
|
|
||||||
{
|
|
||||||
flush_level_.store(log_level);
|
|
||||||
}
|
|
||||||
|
|
||||||
inline spdlog::level::level_enum spdlog::logger::flush_level() const
|
|
||||||
{
|
|
||||||
return static_cast<spdlog::level::level_enum>(flush_level_.load(std::memory_order_relaxed));
|
|
||||||
}
|
|
||||||
|
|
||||||
inline bool spdlog::logger::should_flush_(const details::log_msg &msg)
|
|
||||||
{
|
|
||||||
auto flush_level = flush_level_.load(std::memory_order_relaxed);
|
|
||||||
return (msg.level >= flush_level) && (msg.level != level::off);
|
|
||||||
}
|
|
||||||
|
|
||||||
inline spdlog::level::level_enum spdlog::logger::default_level()
|
|
||||||
{
|
|
||||||
return static_cast<spdlog::level::level_enum>(SPDLOG_ACTIVE_LEVEL);
|
|
||||||
}
|
|
||||||
|
|
||||||
inline spdlog::level::level_enum spdlog::logger::level() const
|
|
||||||
{
|
|
||||||
return static_cast<spdlog::level::level_enum>(level_.load(std::memory_order_relaxed));
|
|
||||||
}
|
|
||||||
|
|
||||||
inline bool spdlog::logger::should_log(spdlog::level::level_enum msg_level) const
|
|
||||||
{
|
|
||||||
return msg_level >= level_.load(std::memory_order_relaxed);
|
|
||||||
}
|
|
||||||
|
|
||||||
//
|
|
||||||
// protected virtual called at end of each user log call (if enabled) by the
|
|
||||||
// line_logger
|
|
||||||
//
|
|
||||||
inline void spdlog::logger::sink_it_(details::log_msg &msg)
|
|
||||||
{
|
|
||||||
#if defined(SPDLOG_ENABLE_MESSAGE_COUNTER)
|
|
||||||
incr_msg_counter_(msg);
|
|
||||||
#endif
|
|
||||||
for (auto &sink : sinks_)
|
|
||||||
{
|
|
||||||
if (sink->should_log(msg.level))
|
|
||||||
{
|
|
||||||
sink->log(msg);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (should_flush_(msg))
|
|
||||||
{
|
|
||||||
flush_();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
inline void spdlog::logger::flush_()
|
|
||||||
{
|
|
||||||
for (auto &sink : sinks_)
|
|
||||||
{
|
|
||||||
sink->flush();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
inline void spdlog::logger::default_err_handler_(const std::string &msg)
|
|
||||||
{
|
|
||||||
auto now = time(nullptr);
|
|
||||||
if (now - last_err_time_ < 60)
|
|
||||||
{
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
last_err_time_ = now;
|
|
||||||
auto tm_time = details::os::localtime(now);
|
|
||||||
char date_buf[100];
|
|
||||||
std::strftime(date_buf, sizeof(date_buf), "%Y-%m-%d %H:%M:%S", &tm_time);
|
|
||||||
fmt::print(stderr, "[*** LOG ERROR ***] [{}] [{}] {}\n", date_buf, name(), msg);
|
|
||||||
}
|
|
||||||
|
|
||||||
inline void spdlog::logger::incr_msg_counter_(details::log_msg &msg)
|
|
||||||
{
|
|
||||||
msg.msg_id = msg_counter_.fetch_add(1, std::memory_order_relaxed);
|
|
||||||
}
|
|
||||||
|
|
||||||
inline const std::vector<spdlog::sink_ptr> &spdlog::logger::sinks() const
|
|
||||||
{
|
|
||||||
return sinks_;
|
|
||||||
}
|
|
||||||
|
|
||||||
inline std::vector<spdlog::sink_ptr> &spdlog::logger::sinks()
|
|
||||||
{
|
|
||||||
return sinks_;
|
|
||||||
}
|
|
||||||
|
|
||||||
inline std::shared_ptr<spdlog::logger> spdlog::logger::clone(std::string logger_name)
|
|
||||||
{
|
|
||||||
auto cloned = std::make_shared<spdlog::logger>(std::move(logger_name), sinks_.begin(), sinks_.end());
|
|
||||||
cloned->set_level(this->level());
|
|
||||||
cloned->flush_on(this->flush_level());
|
|
||||||
cloned->set_error_handler(this->error_handler());
|
|
||||||
return cloned;
|
|
||||||
}
|
|
||||||
121
deps/spdlog/details/mpmc_blocking_q.h
vendored
121
deps/spdlog/details/mpmc_blocking_q.h
vendored
@ -1,121 +0,0 @@
|
|||||||
#pragma once
|
|
||||||
|
|
||||||
//
|
|
||||||
// Copyright(c) 2018 Gabi Melman.
|
|
||||||
// Distributed under the MIT License (http://opensource.org/licenses/MIT)
|
|
||||||
//
|
|
||||||
|
|
||||||
// multi producer-multi consumer blocking queue.
|
|
||||||
// enqueue(..) - will block until room found to put the new message.
|
|
||||||
// enqueue_nowait(..) - will return immediately with false if no room left in
|
|
||||||
// the queue.
|
|
||||||
// dequeue_for(..) - will block until the queue is not empty or timeout have
|
|
||||||
// passed.
|
|
||||||
|
|
||||||
#include "spdlog/details/circular_q.h"
|
|
||||||
|
|
||||||
#include <condition_variable>
|
|
||||||
#include <mutex>
|
|
||||||
|
|
||||||
namespace spdlog {
|
|
||||||
namespace details {
|
|
||||||
|
|
||||||
template<typename T>
|
|
||||||
class mpmc_blocking_queue
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
using item_type = T;
|
|
||||||
explicit mpmc_blocking_queue(size_t max_items)
|
|
||||||
: q_(max_items)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
#ifndef __MINGW32__
|
|
||||||
// try to enqueue and block if no room left
|
|
||||||
void enqueue(T &&item)
|
|
||||||
{
|
|
||||||
{
|
|
||||||
std::unique_lock<std::mutex> lock(queue_mutex_);
|
|
||||||
pop_cv_.wait(lock, [this] { return !this->q_.full(); });
|
|
||||||
q_.push_back(std::move(item));
|
|
||||||
}
|
|
||||||
push_cv_.notify_one();
|
|
||||||
}
|
|
||||||
|
|
||||||
// enqueue immediately. overrun oldest message in the queue if no room left.
|
|
||||||
void enqueue_nowait(T &&item)
|
|
||||||
{
|
|
||||||
{
|
|
||||||
std::unique_lock<std::mutex> lock(queue_mutex_);
|
|
||||||
q_.push_back(std::move(item));
|
|
||||||
}
|
|
||||||
push_cv_.notify_one();
|
|
||||||
}
|
|
||||||
|
|
||||||
// try to dequeue item. if no item found. wait upto timeout and try again
|
|
||||||
// Return true, if succeeded dequeue item, false otherwise
|
|
||||||
bool dequeue_for(T &popped_item, std::chrono::milliseconds wait_duration)
|
|
||||||
{
|
|
||||||
{
|
|
||||||
std::unique_lock<std::mutex> lock(queue_mutex_);
|
|
||||||
if (!push_cv_.wait_for(lock, wait_duration, [this] { return !this->q_.empty(); }))
|
|
||||||
{
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
q_.pop_front(popped_item);
|
|
||||||
}
|
|
||||||
pop_cv_.notify_one();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
#else
|
|
||||||
// apparently mingw deadlocks if the mutex is released before cv.notify_one(),
|
|
||||||
// so release the mutex at the very end each function.
|
|
||||||
|
|
||||||
// try to enqueue and block if no room left
|
|
||||||
void enqueue(T &&item)
|
|
||||||
{
|
|
||||||
std::unique_lock<std::mutex> lock(queue_mutex_);
|
|
||||||
pop_cv_.wait(lock, [this] { return !this->q_.full(); });
|
|
||||||
q_.push_back(std::move(item));
|
|
||||||
push_cv_.notify_one();
|
|
||||||
}
|
|
||||||
|
|
||||||
// enqueue immediately. overrun oldest message in the queue if no room left.
|
|
||||||
void enqueue_nowait(T &&item)
|
|
||||||
{
|
|
||||||
std::unique_lock<std::mutex> lock(queue_mutex_);
|
|
||||||
q_.push_back(std::move(item));
|
|
||||||
push_cv_.notify_one();
|
|
||||||
}
|
|
||||||
|
|
||||||
// try to dequeue item. if no item found. wait upto timeout and try again
|
|
||||||
// Return true, if succeeded dequeue item, false otherwise
|
|
||||||
bool dequeue_for(T &popped_item, std::chrono::milliseconds wait_duration)
|
|
||||||
{
|
|
||||||
std::unique_lock<std::mutex> lock(queue_mutex_);
|
|
||||||
if (!push_cv_.wait_for(lock, wait_duration, [this] { return !this->q_.empty(); }))
|
|
||||||
{
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
q_.pop_front(popped_item);
|
|
||||||
pop_cv_.notify_one();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
|
||||||
|
|
||||||
size_t overrun_counter()
|
|
||||||
{
|
|
||||||
std::unique_lock<std::mutex> lock(queue_mutex_);
|
|
||||||
return q_.overrun_counter();
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
|
||||||
std::mutex queue_mutex_;
|
|
||||||
std::condition_variable push_cv_;
|
|
||||||
std::condition_variable pop_cv_;
|
|
||||||
spdlog::details::circular_q<T> q_;
|
|
||||||
};
|
|
||||||
} // namespace details
|
|
||||||
} // namespace spdlog
|
|
||||||
45
deps/spdlog/details/null_mutex.h
vendored
45
deps/spdlog/details/null_mutex.h
vendored
@ -1,45 +0,0 @@
|
|||||||
//
|
|
||||||
// Copyright(c) 2015 Gabi Melman.
|
|
||||||
// Distributed under the MIT License (http://opensource.org/licenses/MIT)
|
|
||||||
//
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include <atomic>
|
|
||||||
// null, no cost dummy "mutex" and dummy "atomic" int
|
|
||||||
|
|
||||||
namespace spdlog {
|
|
||||||
namespace details {
|
|
||||||
struct null_mutex
|
|
||||||
{
|
|
||||||
void lock() {}
|
|
||||||
void unlock() {}
|
|
||||||
bool try_lock()
|
|
||||||
{
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
struct null_atomic_int
|
|
||||||
{
|
|
||||||
int value;
|
|
||||||
null_atomic_int() = default;
|
|
||||||
|
|
||||||
explicit null_atomic_int(int val)
|
|
||||||
: value(val)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
int load(std::memory_order) const
|
|
||||||
{
|
|
||||||
return value;
|
|
||||||
}
|
|
||||||
|
|
||||||
void store(int val)
|
|
||||||
{
|
|
||||||
value = val;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
} // namespace details
|
|
||||||
} // namespace spdlog
|
|
||||||
421
deps/spdlog/details/os.h
vendored
421
deps/spdlog/details/os.h
vendored
@ -1,421 +0,0 @@
|
|||||||
//
|
|
||||||
// Copyright(c) 2015 Gabi Melman.
|
|
||||||
// Distributed under the MIT License (http://opensource.org/licenses/MIT)
|
|
||||||
//
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include "../common.h"
|
|
||||||
|
|
||||||
#include <algorithm>
|
|
||||||
#include <chrono>
|
|
||||||
#include <cstdio>
|
|
||||||
#include <cstdlib>
|
|
||||||
#include <cstring>
|
|
||||||
#include <ctime>
|
|
||||||
#include <functional>
|
|
||||||
#include <string>
|
|
||||||
#include <sys/stat.h>
|
|
||||||
#include <sys/types.h>
|
|
||||||
#include <thread>
|
|
||||||
|
|
||||||
#ifdef _WIN32
|
|
||||||
|
|
||||||
#ifndef NOMINMAX
|
|
||||||
#define NOMINMAX // prevent windows redefining min/max
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifndef WIN32_LEAN_AND_MEAN
|
|
||||||
#define WIN32_LEAN_AND_MEAN
|
|
||||||
#endif
|
|
||||||
#include <io.h> // _get_osfhandle and _isatty support
|
|
||||||
#include <process.h> // _get_pid support
|
|
||||||
#include <windows.h>
|
|
||||||
|
|
||||||
#ifdef __MINGW32__
|
|
||||||
#include <share.h>
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#else // unix
|
|
||||||
|
|
||||||
#include <fcntl.h>
|
|
||||||
#include <unistd.h>
|
|
||||||
|
|
||||||
#ifdef __linux__
|
|
||||||
#include <sys/syscall.h> //Use gettid() syscall under linux to get thread id
|
|
||||||
|
|
||||||
#elif __FreeBSD__
|
|
||||||
#include <sys/thr.h> //Use thr_self() syscall under FreeBSD to get thread id
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#endif // unix
|
|
||||||
|
|
||||||
#ifndef __has_feature // Clang - feature checking macros.
|
|
||||||
#define __has_feature(x) 0 // Compatibility with non-clang compilers.
|
|
||||||
#endif
|
|
||||||
|
|
||||||
namespace spdlog {
|
|
||||||
namespace details {
|
|
||||||
namespace os {
|
|
||||||
|
|
||||||
inline spdlog::log_clock::time_point now() SPDLOG_NOEXCEPT
|
|
||||||
{
|
|
||||||
|
|
||||||
#if defined __linux__ && defined SPDLOG_CLOCK_COARSE
|
|
||||||
timespec ts;
|
|
||||||
::clock_gettime(CLOCK_REALTIME_COARSE, &ts);
|
|
||||||
return std::chrono::time_point<log_clock, typename log_clock::duration>(
|
|
||||||
std::chrono::duration_cast<typename log_clock::duration>(std::chrono::seconds(ts.tv_sec) + std::chrono::nanoseconds(ts.tv_nsec)));
|
|
||||||
|
|
||||||
#else
|
|
||||||
return log_clock::now();
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
inline std::tm localtime(const std::time_t &time_tt) SPDLOG_NOEXCEPT
|
|
||||||
{
|
|
||||||
|
|
||||||
#ifdef _WIN32
|
|
||||||
std::tm tm;
|
|
||||||
localtime_s(&tm, &time_tt);
|
|
||||||
#else
|
|
||||||
std::tm tm;
|
|
||||||
localtime_r(&time_tt, &tm);
|
|
||||||
#endif
|
|
||||||
return tm;
|
|
||||||
}
|
|
||||||
|
|
||||||
inline std::tm localtime() SPDLOG_NOEXCEPT
|
|
||||||
{
|
|
||||||
std::time_t now_t = time(nullptr);
|
|
||||||
return localtime(now_t);
|
|
||||||
}
|
|
||||||
|
|
||||||
inline std::tm gmtime(const std::time_t &time_tt) SPDLOG_NOEXCEPT
|
|
||||||
{
|
|
||||||
|
|
||||||
#ifdef _WIN32
|
|
||||||
std::tm tm;
|
|
||||||
gmtime_s(&tm, &time_tt);
|
|
||||||
#else
|
|
||||||
std::tm tm;
|
|
||||||
gmtime_r(&time_tt, &tm);
|
|
||||||
#endif
|
|
||||||
return tm;
|
|
||||||
}
|
|
||||||
|
|
||||||
inline std::tm gmtime() SPDLOG_NOEXCEPT
|
|
||||||
{
|
|
||||||
std::time_t now_t = time(nullptr);
|
|
||||||
return gmtime(now_t);
|
|
||||||
}
|
|
||||||
|
|
||||||
// eol definition
|
|
||||||
#if !defined(SPDLOG_EOL)
|
|
||||||
#ifdef _WIN32
|
|
||||||
#define SPDLOG_EOL "\r\n"
|
|
||||||
#else
|
|
||||||
#define SPDLOG_EOL "\n"
|
|
||||||
#endif
|
|
||||||
#endif
|
|
||||||
|
|
||||||
SPDLOG_CONSTEXPR static const char *default_eol = SPDLOG_EOL;
|
|
||||||
|
|
||||||
// folder separator
|
|
||||||
#ifdef _WIN32
|
|
||||||
SPDLOG_CONSTEXPR static const char folder_sep = '\\';
|
|
||||||
#else
|
|
||||||
SPDLOG_CONSTEXPR static const char folder_sep = '/';
|
|
||||||
#endif
|
|
||||||
|
|
||||||
inline void prevent_child_fd(FILE *f)
|
|
||||||
{
|
|
||||||
|
|
||||||
#ifdef _WIN32
|
|
||||||
#if !defined(__cplusplus_winrt)
|
|
||||||
auto file_handle = (HANDLE)_get_osfhandle(_fileno(f));
|
|
||||||
if (!::SetHandleInformation(file_handle, HANDLE_FLAG_INHERIT, 0))
|
|
||||||
throw spdlog_ex("SetHandleInformation failed", errno);
|
|
||||||
#endif
|
|
||||||
#else
|
|
||||||
auto fd = fileno(f);
|
|
||||||
if (fcntl(fd, F_SETFD, FD_CLOEXEC) == -1)
|
|
||||||
{
|
|
||||||
throw spdlog_ex("fcntl with FD_CLOEXEC failed", errno);
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
// fopen_s on non windows for writing
|
|
||||||
inline bool fopen_s(FILE **fp, const filename_t &filename, const filename_t &mode)
|
|
||||||
{
|
|
||||||
#ifdef _WIN32
|
|
||||||
#ifdef SPDLOG_WCHAR_FILENAMES
|
|
||||||
*fp = _wfsopen((filename.c_str()), mode.c_str(), _SH_DENYNO);
|
|
||||||
#else
|
|
||||||
*fp = _fsopen((filename.c_str()), mode.c_str(), _SH_DENYNO);
|
|
||||||
#endif
|
|
||||||
#else // unix
|
|
||||||
*fp = fopen((filename.c_str()), mode.c_str());
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifdef SPDLOG_PREVENT_CHILD_FD
|
|
||||||
if (*fp != nullptr)
|
|
||||||
{
|
|
||||||
prevent_child_fd(*fp);
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
return *fp == nullptr;
|
|
||||||
}
|
|
||||||
|
|
||||||
inline int remove(const filename_t &filename) SPDLOG_NOEXCEPT
|
|
||||||
{
|
|
||||||
#if defined(_WIN32) && defined(SPDLOG_WCHAR_FILENAMES)
|
|
||||||
return _wremove(filename.c_str());
|
|
||||||
#else
|
|
||||||
return std::remove(filename.c_str());
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
inline int rename(const filename_t &filename1, const filename_t &filename2) SPDLOG_NOEXCEPT
|
|
||||||
{
|
|
||||||
#if defined(_WIN32) && defined(SPDLOG_WCHAR_FILENAMES)
|
|
||||||
return _wrename(filename1.c_str(), filename2.c_str());
|
|
||||||
#else
|
|
||||||
return std::rename(filename1.c_str(), filename2.c_str());
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
// Return if file exists
|
|
||||||
inline bool file_exists(const filename_t &filename) SPDLOG_NOEXCEPT
|
|
||||||
{
|
|
||||||
#ifdef _WIN32
|
|
||||||
#ifdef SPDLOG_WCHAR_FILENAMES
|
|
||||||
auto attribs = GetFileAttributesW(filename.c_str());
|
|
||||||
#else
|
|
||||||
auto attribs = GetFileAttributesA(filename.c_str());
|
|
||||||
#endif
|
|
||||||
return (attribs != INVALID_FILE_ATTRIBUTES && !(attribs & FILE_ATTRIBUTE_DIRECTORY));
|
|
||||||
#else // common linux/unix all have the stat system call
|
|
||||||
struct stat buffer;
|
|
||||||
return (stat(filename.c_str(), &buffer) == 0);
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
// Return file size according to open FILE* object
|
|
||||||
inline size_t filesize(FILE *f)
|
|
||||||
{
|
|
||||||
if (f == nullptr)
|
|
||||||
{
|
|
||||||
throw spdlog_ex("Failed getting file size. fd is null");
|
|
||||||
}
|
|
||||||
#if defined(_WIN32) && !defined(__CYGWIN__)
|
|
||||||
int fd = _fileno(f);
|
|
||||||
#if _WIN64 // 64 bits
|
|
||||||
__int64 ret = _filelengthi64(fd);
|
|
||||||
if (ret >= 0)
|
|
||||||
{
|
|
||||||
return static_cast<size_t>(ret);
|
|
||||||
}
|
|
||||||
|
|
||||||
#else // windows 32 bits
|
|
||||||
long ret = _filelength(fd);
|
|
||||||
if (ret >= 0)
|
|
||||||
{
|
|
||||||
return static_cast<size_t>(ret);
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#else // unix
|
|
||||||
int fd = fileno(f);
|
|
||||||
// 64 bits(but not in osx or cygwin, where fstat64 is deprecated)
|
|
||||||
#if !defined(__FreeBSD__) && !defined(__APPLE__) && (defined(__x86_64__) || defined(__ppc64__)) && !defined(__CYGWIN__)
|
|
||||||
struct stat64 st;
|
|
||||||
if (fstat64(fd, &st) == 0)
|
|
||||||
{
|
|
||||||
return static_cast<size_t>(st.st_size);
|
|
||||||
}
|
|
||||||
#else // unix 32 bits or cygwin
|
|
||||||
struct stat st;
|
|
||||||
|
|
||||||
if (fstat(fd, &st) == 0)
|
|
||||||
{
|
|
||||||
return static_cast<size_t>(st.st_size);
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
#endif
|
|
||||||
throw spdlog_ex("Failed getting file size from fd", errno);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Return utc offset in minutes or throw spdlog_ex on failure
|
|
||||||
inline int utc_minutes_offset(const std::tm &tm = details::os::localtime())
|
|
||||||
{
|
|
||||||
|
|
||||||
#ifdef _WIN32
|
|
||||||
#if _WIN32_WINNT < _WIN32_WINNT_WS08
|
|
||||||
TIME_ZONE_INFORMATION tzinfo;
|
|
||||||
auto rv = GetTimeZoneInformation(&tzinfo);
|
|
||||||
#else
|
|
||||||
DYNAMIC_TIME_ZONE_INFORMATION tzinfo;
|
|
||||||
auto rv = GetDynamicTimeZoneInformation(&tzinfo);
|
|
||||||
#endif
|
|
||||||
if (rv == TIME_ZONE_ID_INVALID)
|
|
||||||
throw spdlog::spdlog_ex("Failed getting timezone info. ", errno);
|
|
||||||
|
|
||||||
int offset = -tzinfo.Bias;
|
|
||||||
if (tm.tm_isdst)
|
|
||||||
{
|
|
||||||
offset -= tzinfo.DaylightBias;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
offset -= tzinfo.StandardBias;
|
|
||||||
}
|
|
||||||
return offset;
|
|
||||||
#else
|
|
||||||
|
|
||||||
#if defined(sun) || defined(__sun) || defined(_AIX)
|
|
||||||
// 'tm_gmtoff' field is BSD extension and it's missing on SunOS/Solaris
|
|
||||||
struct helper
|
|
||||||
{
|
|
||||||
static long int calculate_gmt_offset(const std::tm &localtm = details::os::localtime(), const std::tm &gmtm = details::os::gmtime())
|
|
||||||
{
|
|
||||||
int local_year = localtm.tm_year + (1900 - 1);
|
|
||||||
int gmt_year = gmtm.tm_year + (1900 - 1);
|
|
||||||
|
|
||||||
long int days = (
|
|
||||||
// difference in day of year
|
|
||||||
localtm.tm_yday -
|
|
||||||
gmtm.tm_yday
|
|
||||||
|
|
||||||
// + intervening leap days
|
|
||||||
+ ((local_year >> 2) - (gmt_year >> 2)) - (local_year / 100 - gmt_year / 100) +
|
|
||||||
((local_year / 100 >> 2) - (gmt_year / 100 >> 2))
|
|
||||||
|
|
||||||
// + difference in years * 365 */
|
|
||||||
+ (long int)(local_year - gmt_year) * 365);
|
|
||||||
|
|
||||||
long int hours = (24 * days) + (localtm.tm_hour - gmtm.tm_hour);
|
|
||||||
long int mins = (60 * hours) + (localtm.tm_min - gmtm.tm_min);
|
|
||||||
long int secs = (60 * mins) + (localtm.tm_sec - gmtm.tm_sec);
|
|
||||||
|
|
||||||
return secs;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
auto offset_seconds = helper::calculate_gmt_offset(tm);
|
|
||||||
#else
|
|
||||||
auto offset_seconds = tm.tm_gmtoff;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
return static_cast<int>(offset_seconds / 60);
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
// Return current thread id as size_t
|
|
||||||
// It exists because the std::this_thread::get_id() is much slower(especially
|
|
||||||
// under VS 2013)
|
|
||||||
inline size_t _thread_id() SPDLOG_NOEXCEPT
|
|
||||||
{
|
|
||||||
#ifdef _WIN32
|
|
||||||
return static_cast<size_t>(::GetCurrentThreadId());
|
|
||||||
#elif __linux__
|
|
||||||
#if defined(__ANDROID__) && defined(__ANDROID_API__) && (__ANDROID_API__ < 21)
|
|
||||||
#define SYS_gettid __NR_gettid
|
|
||||||
#endif
|
|
||||||
return static_cast<size_t>(syscall(SYS_gettid));
|
|
||||||
#elif __FreeBSD__
|
|
||||||
long tid;
|
|
||||||
thr_self(&tid);
|
|
||||||
return static_cast<size_t>(tid);
|
|
||||||
#elif __APPLE__
|
|
||||||
uint64_t tid;
|
|
||||||
pthread_threadid_np(nullptr, &tid);
|
|
||||||
return static_cast<size_t>(tid);
|
|
||||||
#else // Default to standard C++11 (other Unix)
|
|
||||||
return static_cast<size_t>(std::hash<std::thread::id>()(std::this_thread::get_id()));
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
// Return current thread id as size_t (from thread local storage)
|
|
||||||
inline size_t thread_id() SPDLOG_NOEXCEPT
|
|
||||||
{
|
|
||||||
#if defined(SPDLOG_NO_TLS)
|
|
||||||
return _thread_id();
|
|
||||||
#else // cache thread id in tls
|
|
||||||
static thread_local const size_t tid = _thread_id();
|
|
||||||
return tid;
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
// This is avoid msvc issue in sleep_for that happens if the clock changes.
|
|
||||||
// See https://github.com/gabime/spdlog/issues/609
|
|
||||||
inline void sleep_for_millis(int milliseconds) SPDLOG_NOEXCEPT
|
|
||||||
{
|
|
||||||
#if defined(_WIN32)
|
|
||||||
::Sleep(milliseconds);
|
|
||||||
#else
|
|
||||||
std::this_thread::sleep_for(std::chrono::milliseconds(milliseconds));
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
// wchar support for windows file names (SPDLOG_WCHAR_FILENAMES must be defined)
|
|
||||||
#if defined(_WIN32) && defined(SPDLOG_WCHAR_FILENAMES)
|
|
||||||
#define SPDLOG_FILENAME_T(s) L##s
|
|
||||||
inline std::string filename_to_str(const filename_t &filename)
|
|
||||||
{
|
|
||||||
std::wstring_convert<std::codecvt_utf8<wchar_t>, wchar_t> c;
|
|
||||||
return c.to_bytes(filename);
|
|
||||||
}
|
|
||||||
#else
|
|
||||||
#define SPDLOG_FILENAME_T(s) s
|
|
||||||
inline std::string filename_to_str(const filename_t &filename)
|
|
||||||
{
|
|
||||||
return filename;
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
inline int pid()
|
|
||||||
{
|
|
||||||
|
|
||||||
#ifdef _WIN32
|
|
||||||
return static_cast<int>(::GetCurrentProcessId());
|
|
||||||
#else
|
|
||||||
return static_cast<int>(::getpid());
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
// Determine if the terminal supports colors
|
|
||||||
// Source: https://github.com/agauniyal/rang/
|
|
||||||
inline bool is_color_terminal() SPDLOG_NOEXCEPT
|
|
||||||
{
|
|
||||||
#ifdef _WIN32
|
|
||||||
return true;
|
|
||||||
#else
|
|
||||||
static constexpr const char *Terms[] = {
|
|
||||||
"ansi", "color", "console", "cygwin", "gnome", "konsole", "kterm", "linux", "msys", "putty", "rxvt", "screen", "vt100", "xterm"};
|
|
||||||
|
|
||||||
const char *env_p = std::getenv("TERM");
|
|
||||||
if (env_p == nullptr)
|
|
||||||
{
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
static const bool result =
|
|
||||||
std::any_of(std::begin(Terms), std::end(Terms), [&](const char *term) { return std::strstr(env_p, term) != nullptr; });
|
|
||||||
return result;
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
// Detrmine if the terminal attached
|
|
||||||
// Source: https://github.com/agauniyal/rang/
|
|
||||||
inline bool in_terminal(FILE *file) SPDLOG_NOEXCEPT
|
|
||||||
{
|
|
||||||
|
|
||||||
#ifdef _WIN32
|
|
||||||
return _isatty(_fileno(file)) != 0;
|
|
||||||
#else
|
|
||||||
return isatty(fileno(file)) != 0;
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
} // namespace os
|
|
||||||
} // namespace details
|
|
||||||
} // namespace spdlog
|
|
||||||
1336
deps/spdlog/details/pattern_formatter.h
vendored
1336
deps/spdlog/details/pattern_formatter.h
vendored
File diff suppressed because it is too large
Load Diff
71
deps/spdlog/details/periodic_worker.h
vendored
71
deps/spdlog/details/periodic_worker.h
vendored
@ -1,71 +0,0 @@
|
|||||||
|
|
||||||
//
|
|
||||||
// Copyright(c) 2018 Gabi Melman.
|
|
||||||
// Distributed under the MIT License (http://opensource.org/licenses/MIT)
|
|
||||||
//
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
// periodic worker thread - periodically executes the given callback function.
|
|
||||||
//
|
|
||||||
// RAII over the owned thread:
|
|
||||||
// creates the thread on construction.
|
|
||||||
// stops and joins the thread on destruction (if the thread is executing a callback, wait for it to finish first).
|
|
||||||
|
|
||||||
#include <chrono>
|
|
||||||
#include <condition_variable>
|
|
||||||
#include <functional>
|
|
||||||
#include <mutex>
|
|
||||||
#include <thread>
|
|
||||||
namespace spdlog {
|
|
||||||
namespace details {
|
|
||||||
|
|
||||||
class periodic_worker
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
periodic_worker(const std::function<void()> &callback_fun, std::chrono::seconds interval)
|
|
||||||
{
|
|
||||||
active_ = (interval > std::chrono::seconds::zero());
|
|
||||||
if (!active_)
|
|
||||||
{
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
worker_thread_ = std::thread([this, callback_fun, interval]() {
|
|
||||||
for (;;)
|
|
||||||
{
|
|
||||||
std::unique_lock<std::mutex> lock(this->mutex_);
|
|
||||||
if (this->cv_.wait_for(lock, interval, [this] { return !this->active_; }))
|
|
||||||
{
|
|
||||||
return; // active_ == false, so exit this thread
|
|
||||||
}
|
|
||||||
callback_fun();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
periodic_worker(const periodic_worker &) = delete;
|
|
||||||
periodic_worker &operator=(const periodic_worker &) = delete;
|
|
||||||
|
|
||||||
// stop the worker thread and join it
|
|
||||||
~periodic_worker()
|
|
||||||
{
|
|
||||||
if (worker_thread_.joinable())
|
|
||||||
{
|
|
||||||
{
|
|
||||||
std::lock_guard<std::mutex> lock(mutex_);
|
|
||||||
active_ = false;
|
|
||||||
}
|
|
||||||
cv_.notify_one();
|
|
||||||
worker_thread_.join();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
|
||||||
bool active_;
|
|
||||||
std::thread worker_thread_;
|
|
||||||
std::mutex mutex_;
|
|
||||||
std::condition_variable cv_;
|
|
||||||
};
|
|
||||||
} // namespace details
|
|
||||||
} // namespace spdlog
|
|
||||||
285
deps/spdlog/details/registry.h
vendored
285
deps/spdlog/details/registry.h
vendored
@ -1,285 +0,0 @@
|
|||||||
//
|
|
||||||
// Copyright(c) 2015 Gabi Melman.
|
|
||||||
// Distributed under the MIT License (http://opensource.org/licenses/MIT)
|
|
||||||
//
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
// Loggers registy of unique name->logger pointer
|
|
||||||
// An attempt to create a logger with an already existing name will be ignored
|
|
||||||
// If user requests a non existing logger, nullptr will be returned
|
|
||||||
// This class is thread safe
|
|
||||||
|
|
||||||
#include "spdlog/common.h"
|
|
||||||
#include "spdlog/details/periodic_worker.h"
|
|
||||||
#include "spdlog/logger.h"
|
|
||||||
|
|
||||||
#ifndef SPDLOG_DISABLE_DEFAULT_LOGGER
|
|
||||||
// support for the default stdout color logger
|
|
||||||
#ifdef _WIN32
|
|
||||||
#include "spdlog/sinks/wincolor_sink.h"
|
|
||||||
#else
|
|
||||||
#include "spdlog/sinks/ansicolor_sink.h"
|
|
||||||
#endif
|
|
||||||
#endif // SPDLOG_DISABLE_DEFAULT_LOGGER
|
|
||||||
|
|
||||||
#include <chrono>
|
|
||||||
#include <functional>
|
|
||||||
#include <memory>
|
|
||||||
#include <string>
|
|
||||||
#include <unordered_map>
|
|
||||||
|
|
||||||
namespace spdlog {
|
|
||||||
namespace details {
|
|
||||||
class thread_pool;
|
|
||||||
|
|
||||||
class registry
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
registry(const registry &) = delete;
|
|
||||||
registry &operator=(const registry &) = delete;
|
|
||||||
|
|
||||||
void register_logger(std::shared_ptr<logger> new_logger)
|
|
||||||
{
|
|
||||||
std::lock_guard<std::mutex> lock(logger_map_mutex_);
|
|
||||||
register_logger_(std::move(new_logger));
|
|
||||||
}
|
|
||||||
|
|
||||||
void initialize_logger(std::shared_ptr<logger> new_logger)
|
|
||||||
{
|
|
||||||
std::lock_guard<std::mutex> lock(logger_map_mutex_);
|
|
||||||
new_logger->set_formatter(formatter_->clone());
|
|
||||||
|
|
||||||
if (err_handler_)
|
|
||||||
{
|
|
||||||
new_logger->set_error_handler(err_handler_);
|
|
||||||
}
|
|
||||||
|
|
||||||
new_logger->set_level(level_);
|
|
||||||
new_logger->flush_on(flush_level_);
|
|
||||||
|
|
||||||
if (automatic_registration_)
|
|
||||||
{
|
|
||||||
register_logger_(std::move(new_logger));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
std::shared_ptr<logger> get(const std::string &logger_name)
|
|
||||||
{
|
|
||||||
std::lock_guard<std::mutex> lock(logger_map_mutex_);
|
|
||||||
auto found = loggers_.find(logger_name);
|
|
||||||
return found == loggers_.end() ? nullptr : found->second;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::shared_ptr<logger> default_logger()
|
|
||||||
{
|
|
||||||
std::lock_guard<std::mutex> lock(logger_map_mutex_);
|
|
||||||
return default_logger_;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Return raw ptr to the default logger.
|
|
||||||
// To be used directly by the spdlog default api (e.g. spdlog::info)
|
|
||||||
// This make the default API faster, but cannot be used concurrently with set_default_logger().
|
|
||||||
// e.g do not call set_default_logger() from one thread while calling spdlog::info() from another.
|
|
||||||
logger *get_default_raw()
|
|
||||||
{
|
|
||||||
return default_logger_.get();
|
|
||||||
}
|
|
||||||
|
|
||||||
// set default logger.
|
|
||||||
// default logger is stored in default_logger_ (for faster retrieval) and in the loggers_ map.
|
|
||||||
void set_default_logger(std::shared_ptr<logger> new_default_logger)
|
|
||||||
{
|
|
||||||
std::lock_guard<std::mutex> lock(logger_map_mutex_);
|
|
||||||
// remove previous default logger from the map
|
|
||||||
if (default_logger_ != nullptr)
|
|
||||||
{
|
|
||||||
loggers_.erase(default_logger_->name());
|
|
||||||
}
|
|
||||||
if (new_default_logger != nullptr)
|
|
||||||
{
|
|
||||||
loggers_[new_default_logger->name()] = new_default_logger;
|
|
||||||
}
|
|
||||||
default_logger_ = std::move(new_default_logger);
|
|
||||||
}
|
|
||||||
|
|
||||||
void set_tp(std::shared_ptr<thread_pool> tp)
|
|
||||||
{
|
|
||||||
std::lock_guard<std::recursive_mutex> lock(tp_mutex_);
|
|
||||||
tp_ = std::move(tp);
|
|
||||||
}
|
|
||||||
|
|
||||||
std::shared_ptr<thread_pool> get_tp()
|
|
||||||
{
|
|
||||||
std::lock_guard<std::recursive_mutex> lock(tp_mutex_);
|
|
||||||
return tp_;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Set global formatter. Each sink in each logger will get a clone of this object
|
|
||||||
void set_formatter(std::unique_ptr<formatter> formatter)
|
|
||||||
{
|
|
||||||
std::lock_guard<std::mutex> lock(logger_map_mutex_);
|
|
||||||
formatter_ = std::move(formatter);
|
|
||||||
for (auto &l : loggers_)
|
|
||||||
{
|
|
||||||
l.second->set_formatter(formatter_->clone());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void set_level(level::level_enum log_level)
|
|
||||||
{
|
|
||||||
std::lock_guard<std::mutex> lock(logger_map_mutex_);
|
|
||||||
for (auto &l : loggers_)
|
|
||||||
{
|
|
||||||
l.second->set_level(log_level);
|
|
||||||
}
|
|
||||||
level_ = log_level;
|
|
||||||
}
|
|
||||||
|
|
||||||
void flush_on(level::level_enum log_level)
|
|
||||||
{
|
|
||||||
std::lock_guard<std::mutex> lock(logger_map_mutex_);
|
|
||||||
for (auto &l : loggers_)
|
|
||||||
{
|
|
||||||
l.second->flush_on(log_level);
|
|
||||||
}
|
|
||||||
flush_level_ = log_level;
|
|
||||||
}
|
|
||||||
|
|
||||||
void flush_every(std::chrono::seconds interval)
|
|
||||||
{
|
|
||||||
std::lock_guard<std::mutex> lock(flusher_mutex_);
|
|
||||||
std::function<void()> clbk = std::bind(®istry::flush_all, this);
|
|
||||||
periodic_flusher_ = details::make_unique<periodic_worker>(clbk, interval);
|
|
||||||
}
|
|
||||||
|
|
||||||
void set_error_handler(log_err_handler handler)
|
|
||||||
{
|
|
||||||
std::lock_guard<std::mutex> lock(logger_map_mutex_);
|
|
||||||
for (auto &l : loggers_)
|
|
||||||
{
|
|
||||||
l.second->set_error_handler(handler);
|
|
||||||
}
|
|
||||||
err_handler_ = handler;
|
|
||||||
}
|
|
||||||
|
|
||||||
void apply_all(const std::function<void(const std::shared_ptr<logger>)> &fun)
|
|
||||||
{
|
|
||||||
std::lock_guard<std::mutex> lock(logger_map_mutex_);
|
|
||||||
for (auto &l : loggers_)
|
|
||||||
{
|
|
||||||
fun(l.second);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void flush_all()
|
|
||||||
{
|
|
||||||
std::lock_guard<std::mutex> lock(logger_map_mutex_);
|
|
||||||
for (auto &l : loggers_)
|
|
||||||
{
|
|
||||||
l.second->flush();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void drop(const std::string &logger_name)
|
|
||||||
{
|
|
||||||
std::lock_guard<std::mutex> lock(logger_map_mutex_);
|
|
||||||
loggers_.erase(logger_name);
|
|
||||||
if (default_logger_ && default_logger_->name() == logger_name)
|
|
||||||
{
|
|
||||||
default_logger_.reset();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void drop_all()
|
|
||||||
{
|
|
||||||
std::lock_guard<std::mutex> lock(logger_map_mutex_);
|
|
||||||
loggers_.clear();
|
|
||||||
default_logger_.reset();
|
|
||||||
}
|
|
||||||
|
|
||||||
// clean all resources and threads started by the registry
|
|
||||||
void shutdown()
|
|
||||||
{
|
|
||||||
{
|
|
||||||
std::lock_guard<std::mutex> lock(flusher_mutex_);
|
|
||||||
periodic_flusher_.reset();
|
|
||||||
}
|
|
||||||
|
|
||||||
drop_all();
|
|
||||||
|
|
||||||
{
|
|
||||||
std::lock_guard<std::recursive_mutex> lock(tp_mutex_);
|
|
||||||
tp_.reset();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
std::recursive_mutex &tp_mutex()
|
|
||||||
{
|
|
||||||
return tp_mutex_;
|
|
||||||
}
|
|
||||||
|
|
||||||
void set_automatic_registration(bool automatic_regsistration)
|
|
||||||
{
|
|
||||||
std::lock_guard<std::mutex> lock(logger_map_mutex_);
|
|
||||||
automatic_registration_ = automatic_regsistration;
|
|
||||||
}
|
|
||||||
|
|
||||||
static registry &instance()
|
|
||||||
{
|
|
||||||
static registry s_instance;
|
|
||||||
return s_instance;
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
|
||||||
registry()
|
|
||||||
: formatter_(new pattern_formatter())
|
|
||||||
{
|
|
||||||
|
|
||||||
#ifndef SPDLOG_DISABLE_DEFAULT_LOGGER
|
|
||||||
// create default logger (ansicolor_stdout_sink_mt or wincolor_stdout_sink_mt in windows).
|
|
||||||
#ifdef _WIN32
|
|
||||||
auto color_sink = std::make_shared<sinks::wincolor_stdout_sink_mt>();
|
|
||||||
#else
|
|
||||||
auto color_sink = std::make_shared<sinks::ansicolor_stdout_sink_mt>();
|
|
||||||
#endif
|
|
||||||
|
|
||||||
const char *default_logger_name = "";
|
|
||||||
default_logger_ = std::make_shared<spdlog::logger>(default_logger_name, std::move(color_sink));
|
|
||||||
loggers_[default_logger_name] = default_logger_;
|
|
||||||
|
|
||||||
#endif // SPDLOG_DISABLE_DEFAULT_LOGGER
|
|
||||||
}
|
|
||||||
|
|
||||||
~registry() = default;
|
|
||||||
|
|
||||||
void throw_if_exists_(const std::string &logger_name)
|
|
||||||
{
|
|
||||||
if (loggers_.find(logger_name) != loggers_.end())
|
|
||||||
{
|
|
||||||
throw spdlog_ex("logger with name '" + logger_name + "' already exists");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void register_logger_(std::shared_ptr<logger> new_logger)
|
|
||||||
{
|
|
||||||
auto logger_name = new_logger->name();
|
|
||||||
throw_if_exists_(logger_name);
|
|
||||||
loggers_[logger_name] = std::move(new_logger);
|
|
||||||
}
|
|
||||||
|
|
||||||
std::mutex logger_map_mutex_, flusher_mutex_;
|
|
||||||
std::recursive_mutex tp_mutex_;
|
|
||||||
std::unordered_map<std::string, std::shared_ptr<logger>> loggers_;
|
|
||||||
std::unique_ptr<formatter> formatter_;
|
|
||||||
level::level_enum level_ = spdlog::logger::default_level();
|
|
||||||
level::level_enum flush_level_ = level::off;
|
|
||||||
log_err_handler err_handler_;
|
|
||||||
std::shared_ptr<thread_pool> tp_;
|
|
||||||
std::unique_ptr<periodic_worker> periodic_flusher_;
|
|
||||||
std::shared_ptr<logger> default_logger_;
|
|
||||||
bool automatic_registration_ = true;
|
|
||||||
};
|
|
||||||
|
|
||||||
} // namespace details
|
|
||||||
} // namespace spdlog
|
|
||||||
238
deps/spdlog/details/thread_pool.h
vendored
238
deps/spdlog/details/thread_pool.h
vendored
@ -1,238 +0,0 @@
|
|||||||
#pragma once
|
|
||||||
|
|
||||||
#include "spdlog/details/fmt_helper.h"
|
|
||||||
#include "spdlog/details/log_msg.h"
|
|
||||||
#include "spdlog/details/mpmc_blocking_q.h"
|
|
||||||
#include "spdlog/details/os.h"
|
|
||||||
|
|
||||||
#include <chrono>
|
|
||||||
#include <memory>
|
|
||||||
#include <thread>
|
|
||||||
#include <vector>
|
|
||||||
|
|
||||||
namespace spdlog {
|
|
||||||
namespace details {
|
|
||||||
|
|
||||||
using async_logger_ptr = std::shared_ptr<spdlog::async_logger>;
|
|
||||||
|
|
||||||
enum class async_msg_type
|
|
||||||
{
|
|
||||||
log,
|
|
||||||
flush,
|
|
||||||
terminate
|
|
||||||
};
|
|
||||||
|
|
||||||
// Async msg to move to/from the queue
|
|
||||||
// Movable only. should never be copied
|
|
||||||
struct async_msg
|
|
||||||
{
|
|
||||||
async_msg_type msg_type;
|
|
||||||
level::level_enum level;
|
|
||||||
log_clock::time_point time;
|
|
||||||
size_t thread_id;
|
|
||||||
fmt::basic_memory_buffer<char, 176> raw;
|
|
||||||
|
|
||||||
size_t msg_id;
|
|
||||||
source_loc source;
|
|
||||||
async_logger_ptr worker_ptr;
|
|
||||||
|
|
||||||
async_msg() = default;
|
|
||||||
~async_msg() = default;
|
|
||||||
|
|
||||||
// should only be moved in or out of the queue..
|
|
||||||
async_msg(const async_msg &) = delete;
|
|
||||||
|
|
||||||
// support for vs2013 move
|
|
||||||
#if defined(_MSC_VER) && _MSC_VER <= 1800
|
|
||||||
async_msg(async_msg &&other) SPDLOG_NOEXCEPT : msg_type(other.msg_type),
|
|
||||||
level(other.level),
|
|
||||||
time(other.time),
|
|
||||||
thread_id(other.thread_id),
|
|
||||||
raw(move(other.raw)),
|
|
||||||
msg_id(other.msg_id),
|
|
||||||
source(other.source),
|
|
||||||
worker_ptr(std::move(other.worker_ptr))
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
async_msg &operator=(async_msg &&other) SPDLOG_NOEXCEPT
|
|
||||||
{
|
|
||||||
msg_type = other.msg_type;
|
|
||||||
level = other.level;
|
|
||||||
time = other.time;
|
|
||||||
thread_id = other.thread_id;
|
|
||||||
raw = std::move(other.raw);
|
|
||||||
msg_id = other.msg_id;
|
|
||||||
source = other.source;
|
|
||||||
worker_ptr = std::move(other.worker_ptr);
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
#else // (_MSC_VER) && _MSC_VER <= 1800
|
|
||||||
async_msg(async_msg &&) = default;
|
|
||||||
async_msg &operator=(async_msg &&) = default;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
// construct from log_msg with given type
|
|
||||||
async_msg(async_logger_ptr &&worker, async_msg_type the_type, details::log_msg &m)
|
|
||||||
: msg_type(the_type)
|
|
||||||
, level(m.level)
|
|
||||||
, time(m.time)
|
|
||||||
, thread_id(m.thread_id)
|
|
||||||
, msg_id(m.msg_id)
|
|
||||||
, source(m.source)
|
|
||||||
, worker_ptr(std::move(worker))
|
|
||||||
{
|
|
||||||
fmt_helper::append_string_view(m.payload, raw);
|
|
||||||
}
|
|
||||||
|
|
||||||
async_msg(async_logger_ptr &&worker, async_msg_type the_type)
|
|
||||||
: msg_type(the_type)
|
|
||||||
, level(level::off)
|
|
||||||
, time()
|
|
||||||
, thread_id(0)
|
|
||||||
, msg_id(0)
|
|
||||||
, source()
|
|
||||||
, worker_ptr(std::move(worker))
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
explicit async_msg(async_msg_type the_type)
|
|
||||||
: async_msg(nullptr, the_type)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
// copy into log_msg
|
|
||||||
log_msg to_log_msg()
|
|
||||||
{
|
|
||||||
log_msg msg(&worker_ptr->name(), level, string_view_t(raw.data(), raw.size()));
|
|
||||||
msg.time = time;
|
|
||||||
msg.thread_id = thread_id;
|
|
||||||
msg.msg_id = msg_id;
|
|
||||||
msg.source = source;
|
|
||||||
msg.color_range_start = 0;
|
|
||||||
msg.color_range_end = 0;
|
|
||||||
return msg;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
class thread_pool
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
using item_type = async_msg;
|
|
||||||
using q_type = details::mpmc_blocking_queue<item_type>;
|
|
||||||
|
|
||||||
thread_pool(size_t q_max_items, size_t threads_n)
|
|
||||||
: q_(q_max_items)
|
|
||||||
{
|
|
||||||
// std::cout << "thread_pool() q_size_bytes: " << q_size_bytes <<
|
|
||||||
// "\tthreads_n: " << threads_n << std::endl;
|
|
||||||
if (threads_n == 0 || threads_n > 1000)
|
|
||||||
{
|
|
||||||
throw spdlog_ex("spdlog::thread_pool(): invalid threads_n param (valid "
|
|
||||||
"range is 1-1000)");
|
|
||||||
}
|
|
||||||
for (size_t i = 0; i < threads_n; i++)
|
|
||||||
{
|
|
||||||
threads_.emplace_back(&thread_pool::worker_loop_, this);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// message all threads to terminate gracefully join them
|
|
||||||
~thread_pool()
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
for (size_t i = 0; i < threads_.size(); i++)
|
|
||||||
{
|
|
||||||
post_async_msg_(async_msg(async_msg_type::terminate), async_overflow_policy::block);
|
|
||||||
}
|
|
||||||
|
|
||||||
for (auto &t : threads_)
|
|
||||||
{
|
|
||||||
t.join();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
catch (...)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
thread_pool(const thread_pool &) = delete;
|
|
||||||
thread_pool &operator=(thread_pool &&) = delete;
|
|
||||||
|
|
||||||
void post_log(async_logger_ptr &&worker_ptr, details::log_msg &msg, async_overflow_policy overflow_policy)
|
|
||||||
{
|
|
||||||
async_msg async_m(std::move(worker_ptr), async_msg_type::log, msg);
|
|
||||||
post_async_msg_(std::move(async_m), overflow_policy);
|
|
||||||
}
|
|
||||||
|
|
||||||
void post_flush(async_logger_ptr &&worker_ptr, async_overflow_policy overflow_policy)
|
|
||||||
{
|
|
||||||
post_async_msg_(async_msg(std::move(worker_ptr), async_msg_type::flush), overflow_policy);
|
|
||||||
}
|
|
||||||
|
|
||||||
size_t overrun_counter()
|
|
||||||
{
|
|
||||||
return q_.overrun_counter();
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
|
||||||
q_type q_;
|
|
||||||
|
|
||||||
std::vector<std::thread> threads_;
|
|
||||||
|
|
||||||
void post_async_msg_(async_msg &&new_msg, async_overflow_policy overflow_policy)
|
|
||||||
{
|
|
||||||
if (overflow_policy == async_overflow_policy::block)
|
|
||||||
{
|
|
||||||
q_.enqueue(std::move(new_msg));
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
q_.enqueue_nowait(std::move(new_msg));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void worker_loop_()
|
|
||||||
{
|
|
||||||
while (process_next_msg_()) {};
|
|
||||||
}
|
|
||||||
|
|
||||||
// process next message in the queue
|
|
||||||
// return true if this thread should still be active (while no terminate msg
|
|
||||||
// was received)
|
|
||||||
bool process_next_msg_()
|
|
||||||
{
|
|
||||||
async_msg incoming_async_msg;
|
|
||||||
bool dequeued = q_.dequeue_for(incoming_async_msg, std::chrono::seconds(10));
|
|
||||||
if (!dequeued)
|
|
||||||
{
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
switch (incoming_async_msg.msg_type)
|
|
||||||
{
|
|
||||||
case async_msg_type::log:
|
|
||||||
{
|
|
||||||
auto msg = incoming_async_msg.to_log_msg();
|
|
||||||
incoming_async_msg.worker_ptr->backend_log_(msg);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
case async_msg_type::flush:
|
|
||||||
{
|
|
||||||
incoming_async_msg.worker_ptr->backend_flush_();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
case async_msg_type::terminate:
|
|
||||||
{
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
assert(false && "Unexpected async_msg_type");
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
} // namespace details
|
|
||||||
} // namespace spdlog
|
|
||||||
172
deps/spdlog/fmt/bin_to_hex.h
vendored
172
deps/spdlog/fmt/bin_to_hex.h
vendored
@ -1,172 +0,0 @@
|
|||||||
//
|
|
||||||
// Copyright(c) 2015 Gabi Melman.
|
|
||||||
// Distributed under the MIT License (http://opensource.org/licenses/MIT)
|
|
||||||
//
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
//
|
|
||||||
// Support for logging binary data as hex
|
|
||||||
// format flags:
|
|
||||||
// {:X} - print in uppercase.
|
|
||||||
// {:s} - don't separate each byte with space.
|
|
||||||
// {:p} - don't print the position on each line start.
|
|
||||||
// {:n} - don't split the output to lines.
|
|
||||||
|
|
||||||
//
|
|
||||||
// Examples:
|
|
||||||
//
|
|
||||||
// std::vector<char> v(200, 0x0b);
|
|
||||||
// logger->info("Some buffer {}", spdlog::to_hex(v));
|
|
||||||
// char buf[128];
|
|
||||||
// logger->info("Some buffer {:X}", spdlog::to_hex(std::begin(buf), std::end(buf)));
|
|
||||||
|
|
||||||
namespace spdlog {
|
|
||||||
namespace details {
|
|
||||||
|
|
||||||
template<typename It>
|
|
||||||
class bytes_range
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
bytes_range(It range_begin, It range_end)
|
|
||||||
: begin_(range_begin)
|
|
||||||
, end_(range_end)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
It begin() const
|
|
||||||
{
|
|
||||||
return begin_;
|
|
||||||
}
|
|
||||||
It end() const
|
|
||||||
{
|
|
||||||
return end_;
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
|
||||||
It begin_, end_;
|
|
||||||
};
|
|
||||||
} // namespace details
|
|
||||||
|
|
||||||
// create a bytes_range that wraps the given container
|
|
||||||
template<typename Container>
|
|
||||||
inline details::bytes_range<typename Container::const_iterator> to_hex(const Container &container)
|
|
||||||
{
|
|
||||||
static_assert(sizeof(typename Container::value_type) == 1, "sizeof(Container::value_type) != 1");
|
|
||||||
using Iter = typename Container::const_iterator;
|
|
||||||
return details::bytes_range<Iter>(std::begin(container), std::end(container));
|
|
||||||
}
|
|
||||||
|
|
||||||
// create bytes_range from ranges
|
|
||||||
template<typename It>
|
|
||||||
inline details::bytes_range<It> to_hex(const It range_begin, const It range_end)
|
|
||||||
{
|
|
||||||
return details::bytes_range<It>(range_begin, range_end);
|
|
||||||
}
|
|
||||||
|
|
||||||
} // namespace spdlog
|
|
||||||
|
|
||||||
namespace fmt {
|
|
||||||
|
|
||||||
template<typename T>
|
|
||||||
struct formatter<spdlog::details::bytes_range<T>>
|
|
||||||
{
|
|
||||||
const std::size_t line_size = 100;
|
|
||||||
const char delimiter = ' ';
|
|
||||||
|
|
||||||
bool put_newlines = true;
|
|
||||||
bool put_delimiters = true;
|
|
||||||
bool use_uppercase = false;
|
|
||||||
bool put_positions = true; // position on start of each line
|
|
||||||
|
|
||||||
// parse the format string flags
|
|
||||||
template<typename ParseContext>
|
|
||||||
auto parse(ParseContext &ctx) -> decltype(ctx.begin())
|
|
||||||
{
|
|
||||||
auto it = ctx.begin();
|
|
||||||
while (*it && *it != '}')
|
|
||||||
{
|
|
||||||
switch (*it)
|
|
||||||
{
|
|
||||||
case 'X':
|
|
||||||
use_uppercase = true;
|
|
||||||
break;
|
|
||||||
case 's':
|
|
||||||
put_delimiters = false;
|
|
||||||
break;
|
|
||||||
case 'p':
|
|
||||||
put_positions = false;
|
|
||||||
break;
|
|
||||||
case 'n':
|
|
||||||
put_newlines = false;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
++it;
|
|
||||||
}
|
|
||||||
return it;
|
|
||||||
}
|
|
||||||
|
|
||||||
// format the given bytes range as hex
|
|
||||||
template<typename FormatContext, typename Container>
|
|
||||||
auto format(const spdlog::details::bytes_range<Container> &the_range, FormatContext &ctx) -> decltype(ctx.out())
|
|
||||||
{
|
|
||||||
SPDLOG_CONSTEXPR const char *hex_upper = "0123456789ABCDEF";
|
|
||||||
SPDLOG_CONSTEXPR const char *hex_lower = "0123456789abcdef";
|
|
||||||
const char *hex_chars = use_uppercase ? hex_upper : hex_lower;
|
|
||||||
|
|
||||||
std::size_t pos = 0;
|
|
||||||
std::size_t column = line_size;
|
|
||||||
auto inserter = ctx.begin();
|
|
||||||
|
|
||||||
for (auto &item : the_range)
|
|
||||||
{
|
|
||||||
auto ch = static_cast<unsigned char>(item);
|
|
||||||
pos++;
|
|
||||||
|
|
||||||
if (put_newlines && column >= line_size)
|
|
||||||
{
|
|
||||||
column = put_newline(inserter, pos);
|
|
||||||
|
|
||||||
// put first byte without delimiter in front of it
|
|
||||||
*inserter++ = hex_chars[(ch >> 4) & 0x0f];
|
|
||||||
*inserter++ = hex_chars[ch & 0x0f];
|
|
||||||
column += 2;
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (put_delimiters)
|
|
||||||
{
|
|
||||||
*inserter++ = delimiter;
|
|
||||||
++column;
|
|
||||||
}
|
|
||||||
|
|
||||||
*inserter++ = hex_chars[(ch >> 4) & 0x0f];
|
|
||||||
*inserter++ = hex_chars[ch & 0x0f];
|
|
||||||
column += 2;
|
|
||||||
}
|
|
||||||
return inserter;
|
|
||||||
}
|
|
||||||
|
|
||||||
// put newline(and position header)
|
|
||||||
// return the next column
|
|
||||||
template<typename It>
|
|
||||||
std::size_t put_newline(It inserter, std::size_t pos)
|
|
||||||
{
|
|
||||||
#ifdef _WIN32
|
|
||||||
*inserter++ = '\r';
|
|
||||||
#endif
|
|
||||||
*inserter++ = '\n';
|
|
||||||
|
|
||||||
if (put_positions)
|
|
||||||
{
|
|
||||||
fmt::format_to(inserter, "{:<04X}: ", pos - 1);
|
|
||||||
return 7;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
} // namespace fmt
|
|
||||||
23
deps/spdlog/fmt/bundled/LICENSE.rst
vendored
23
deps/spdlog/fmt/bundled/LICENSE.rst
vendored
@ -1,23 +0,0 @@
|
|||||||
Copyright (c) 2012 - 2016, Victor Zverovich
|
|
||||||
|
|
||||||
All rights reserved.
|
|
||||||
|
|
||||||
Redistribution and use in source and binary forms, with or without
|
|
||||||
modification, are permitted provided that the following conditions are met:
|
|
||||||
|
|
||||||
1. Redistributions of source code must retain the above copyright notice, this
|
|
||||||
list of conditions and the following disclaimer.
|
|
||||||
2. Redistributions in binary form must reproduce the above copyright notice,
|
|
||||||
this list of conditions and the following disclaimer in the documentation
|
|
||||||
and/or other materials provided with the distribution.
|
|
||||||
|
|
||||||
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
|
|
||||||
ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
|
||||||
WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
||||||
DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR
|
|
||||||
ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
||||||
(INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
|
||||||
LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
|
||||||
ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
||||||
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
|
||||||
SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
||||||
452
deps/spdlog/fmt/bundled/chrono.h
vendored
452
deps/spdlog/fmt/bundled/chrono.h
vendored
@ -1,452 +0,0 @@
|
|||||||
// Formatting library for C++ - chrono support
|
|
||||||
//
|
|
||||||
// Copyright (c) 2012 - present, Victor Zverovich
|
|
||||||
// All rights reserved.
|
|
||||||
//
|
|
||||||
// For the license information refer to format.h.
|
|
||||||
|
|
||||||
#ifndef FMT_CHRONO_H_
|
|
||||||
#define FMT_CHRONO_H_
|
|
||||||
|
|
||||||
#include "format.h"
|
|
||||||
#include "locale.h"
|
|
||||||
|
|
||||||
#include <chrono>
|
|
||||||
#include <ctime>
|
|
||||||
#include <locale>
|
|
||||||
#include <sstream>
|
|
||||||
|
|
||||||
FMT_BEGIN_NAMESPACE
|
|
||||||
|
|
||||||
namespace internal{
|
|
||||||
|
|
||||||
enum class numeric_system {
|
|
||||||
standard,
|
|
||||||
// Alternative numeric system, e.g. 十二 instead of 12 in ja_JP locale.
|
|
||||||
alternative
|
|
||||||
};
|
|
||||||
|
|
||||||
// Parses a put_time-like format string and invokes handler actions.
|
|
||||||
template <typename Char, typename Handler>
|
|
||||||
FMT_CONSTEXPR const Char *parse_chrono_format(
|
|
||||||
const Char *begin, const Char *end, Handler &&handler) {
|
|
||||||
auto ptr = begin;
|
|
||||||
while (ptr != end) {
|
|
||||||
auto c = *ptr;
|
|
||||||
if (c == '}') break;
|
|
||||||
if (c != '%') {
|
|
||||||
++ptr;
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
if (begin != ptr)
|
|
||||||
handler.on_text(begin, ptr);
|
|
||||||
++ptr; // consume '%'
|
|
||||||
if (ptr == end)
|
|
||||||
throw format_error("invalid format");
|
|
||||||
c = *ptr++;
|
|
||||||
switch (c) {
|
|
||||||
case '%':
|
|
||||||
handler.on_text(ptr - 1, ptr);
|
|
||||||
break;
|
|
||||||
case 'n': {
|
|
||||||
const char newline[] = "\n";
|
|
||||||
handler.on_text(newline, newline + 1);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case 't': {
|
|
||||||
const char tab[] = "\t";
|
|
||||||
handler.on_text(tab, tab + 1);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
// Day of the week:
|
|
||||||
case 'a':
|
|
||||||
handler.on_abbr_weekday();
|
|
||||||
break;
|
|
||||||
case 'A':
|
|
||||||
handler.on_full_weekday();
|
|
||||||
break;
|
|
||||||
case 'w':
|
|
||||||
handler.on_dec0_weekday(numeric_system::standard);
|
|
||||||
break;
|
|
||||||
case 'u':
|
|
||||||
handler.on_dec1_weekday(numeric_system::standard);
|
|
||||||
break;
|
|
||||||
// Month:
|
|
||||||
case 'b':
|
|
||||||
handler.on_abbr_month();
|
|
||||||
break;
|
|
||||||
case 'B':
|
|
||||||
handler.on_full_month();
|
|
||||||
break;
|
|
||||||
// Hour, minute, second:
|
|
||||||
case 'H':
|
|
||||||
handler.on_24_hour(numeric_system::standard);
|
|
||||||
break;
|
|
||||||
case 'I':
|
|
||||||
handler.on_12_hour(numeric_system::standard);
|
|
||||||
break;
|
|
||||||
case 'M':
|
|
||||||
handler.on_minute(numeric_system::standard);
|
|
||||||
break;
|
|
||||||
case 'S':
|
|
||||||
handler.on_second(numeric_system::standard);
|
|
||||||
break;
|
|
||||||
// Other:
|
|
||||||
case 'c':
|
|
||||||
handler.on_datetime(numeric_system::standard);
|
|
||||||
break;
|
|
||||||
case 'x':
|
|
||||||
handler.on_loc_date(numeric_system::standard);
|
|
||||||
break;
|
|
||||||
case 'X':
|
|
||||||
handler.on_loc_time(numeric_system::standard);
|
|
||||||
break;
|
|
||||||
case 'D':
|
|
||||||
handler.on_us_date();
|
|
||||||
break;
|
|
||||||
case 'F':
|
|
||||||
handler.on_iso_date();
|
|
||||||
break;
|
|
||||||
case 'r':
|
|
||||||
handler.on_12_hour_time();
|
|
||||||
break;
|
|
||||||
case 'R':
|
|
||||||
handler.on_24_hour_time();
|
|
||||||
break;
|
|
||||||
case 'T':
|
|
||||||
handler.on_iso_time();
|
|
||||||
break;
|
|
||||||
case 'p':
|
|
||||||
handler.on_am_pm();
|
|
||||||
break;
|
|
||||||
case 'z':
|
|
||||||
handler.on_utc_offset();
|
|
||||||
break;
|
|
||||||
case 'Z':
|
|
||||||
handler.on_tz_name();
|
|
||||||
break;
|
|
||||||
// Alternative representation:
|
|
||||||
case 'E': {
|
|
||||||
if (ptr == end)
|
|
||||||
throw format_error("invalid format");
|
|
||||||
c = *ptr++;
|
|
||||||
switch (c) {
|
|
||||||
case 'c':
|
|
||||||
handler.on_datetime(numeric_system::alternative);
|
|
||||||
break;
|
|
||||||
case 'x':
|
|
||||||
handler.on_loc_date(numeric_system::alternative);
|
|
||||||
break;
|
|
||||||
case 'X':
|
|
||||||
handler.on_loc_time(numeric_system::alternative);
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
throw format_error("invalid format");
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case 'O':
|
|
||||||
if (ptr == end)
|
|
||||||
throw format_error("invalid format");
|
|
||||||
c = *ptr++;
|
|
||||||
switch (c) {
|
|
||||||
case 'w':
|
|
||||||
handler.on_dec0_weekday(numeric_system::alternative);
|
|
||||||
break;
|
|
||||||
case 'u':
|
|
||||||
handler.on_dec1_weekday(numeric_system::alternative);
|
|
||||||
break;
|
|
||||||
case 'H':
|
|
||||||
handler.on_24_hour(numeric_system::alternative);
|
|
||||||
break;
|
|
||||||
case 'I':
|
|
||||||
handler.on_12_hour(numeric_system::alternative);
|
|
||||||
break;
|
|
||||||
case 'M':
|
|
||||||
handler.on_minute(numeric_system::alternative);
|
|
||||||
break;
|
|
||||||
case 'S':
|
|
||||||
handler.on_second(numeric_system::alternative);
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
throw format_error("invalid format");
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
throw format_error("invalid format");
|
|
||||||
}
|
|
||||||
begin = ptr;
|
|
||||||
}
|
|
||||||
if (begin != ptr)
|
|
||||||
handler.on_text(begin, ptr);
|
|
||||||
return ptr;
|
|
||||||
}
|
|
||||||
|
|
||||||
struct chrono_format_checker {
|
|
||||||
void report_no_date() { throw format_error("no date"); }
|
|
||||||
|
|
||||||
template <typename Char>
|
|
||||||
void on_text(const Char *, const Char *) {}
|
|
||||||
void on_abbr_weekday() { report_no_date(); }
|
|
||||||
void on_full_weekday() { report_no_date(); }
|
|
||||||
void on_dec0_weekday(numeric_system) { report_no_date(); }
|
|
||||||
void on_dec1_weekday(numeric_system) { report_no_date(); }
|
|
||||||
void on_abbr_month() { report_no_date(); }
|
|
||||||
void on_full_month() { report_no_date(); }
|
|
||||||
void on_24_hour(numeric_system) {}
|
|
||||||
void on_12_hour(numeric_system) {}
|
|
||||||
void on_minute(numeric_system) {}
|
|
||||||
void on_second(numeric_system) {}
|
|
||||||
void on_datetime(numeric_system) { report_no_date(); }
|
|
||||||
void on_loc_date(numeric_system) { report_no_date(); }
|
|
||||||
void on_loc_time(numeric_system) { report_no_date(); }
|
|
||||||
void on_us_date() { report_no_date(); }
|
|
||||||
void on_iso_date() { report_no_date(); }
|
|
||||||
void on_12_hour_time() {}
|
|
||||||
void on_24_hour_time() {}
|
|
||||||
void on_iso_time() {}
|
|
||||||
void on_am_pm() {}
|
|
||||||
void on_utc_offset() { report_no_date(); }
|
|
||||||
void on_tz_name() { report_no_date(); }
|
|
||||||
};
|
|
||||||
|
|
||||||
template <typename Int>
|
|
||||||
inline int to_int(Int value) {
|
|
||||||
FMT_ASSERT(value >= (std::numeric_limits<int>::min)() &&
|
|
||||||
value <= (std::numeric_limits<int>::max)(), "invalid value");
|
|
||||||
return static_cast<int>(value);
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename FormatContext, typename OutputIt>
|
|
||||||
struct chrono_formatter {
|
|
||||||
FormatContext &context;
|
|
||||||
OutputIt out;
|
|
||||||
std::chrono::seconds s;
|
|
||||||
std::chrono::milliseconds ms;
|
|
||||||
|
|
||||||
typedef typename FormatContext::char_type char_type;
|
|
||||||
|
|
||||||
explicit chrono_formatter(FormatContext &ctx, OutputIt o)
|
|
||||||
: context(ctx), out(o) {}
|
|
||||||
|
|
||||||
int hour() const { return to_int((s.count() / 3600) % 24); }
|
|
||||||
|
|
||||||
int hour12() const {
|
|
||||||
auto hour = to_int((s.count() / 3600) % 12);
|
|
||||||
return hour > 0 ? hour : 12;
|
|
||||||
}
|
|
||||||
|
|
||||||
int minute() const { return to_int((s.count() / 60) % 60); }
|
|
||||||
int second() const { return to_int(s.count() % 60); }
|
|
||||||
|
|
||||||
std::tm time() const {
|
|
||||||
auto time = std::tm();
|
|
||||||
time.tm_hour = hour();
|
|
||||||
time.tm_min = minute();
|
|
||||||
time.tm_sec = second();
|
|
||||||
return time;
|
|
||||||
}
|
|
||||||
|
|
||||||
void write(int value, int width) {
|
|
||||||
typedef typename int_traits<int>::main_type main_type;
|
|
||||||
main_type n = to_unsigned(value);
|
|
||||||
int num_digits = internal::count_digits(n);
|
|
||||||
if (width > num_digits)
|
|
||||||
out = std::fill_n(out, width - num_digits, '0');
|
|
||||||
out = format_decimal<char_type>(out, n, num_digits);
|
|
||||||
}
|
|
||||||
|
|
||||||
void format_localized(const tm &time, const char *format) {
|
|
||||||
auto locale = context.locale().template get<std::locale>();
|
|
||||||
auto &facet = std::use_facet<std::time_put<char_type>>(locale);
|
|
||||||
std::basic_ostringstream<char_type> os;
|
|
||||||
os.imbue(locale);
|
|
||||||
facet.put(os, os, ' ', &time, format, format + std::strlen(format));
|
|
||||||
auto str = os.str();
|
|
||||||
std::copy(str.begin(), str.end(), out);
|
|
||||||
}
|
|
||||||
|
|
||||||
void on_text(const char_type *begin, const char_type *end) {
|
|
||||||
std::copy(begin, end, out);
|
|
||||||
}
|
|
||||||
|
|
||||||
// These are not implemented because durations don't have date information.
|
|
||||||
void on_abbr_weekday() {}
|
|
||||||
void on_full_weekday() {}
|
|
||||||
void on_dec0_weekday(numeric_system) {}
|
|
||||||
void on_dec1_weekday(numeric_system) {}
|
|
||||||
void on_abbr_month() {}
|
|
||||||
void on_full_month() {}
|
|
||||||
void on_datetime(numeric_system) {}
|
|
||||||
void on_loc_date(numeric_system) {}
|
|
||||||
void on_loc_time(numeric_system) {}
|
|
||||||
void on_us_date() {}
|
|
||||||
void on_iso_date() {}
|
|
||||||
void on_utc_offset() {}
|
|
||||||
void on_tz_name() {}
|
|
||||||
|
|
||||||
void on_24_hour(numeric_system ns) {
|
|
||||||
if (ns == numeric_system::standard)
|
|
||||||
return write(hour(), 2);
|
|
||||||
auto time = tm();
|
|
||||||
time.tm_hour = hour();
|
|
||||||
format_localized(time, "%OH");
|
|
||||||
}
|
|
||||||
|
|
||||||
void on_12_hour(numeric_system ns) {
|
|
||||||
if (ns == numeric_system::standard)
|
|
||||||
return write(hour12(), 2);
|
|
||||||
auto time = tm();
|
|
||||||
time.tm_hour = hour();
|
|
||||||
format_localized(time, "%OI");
|
|
||||||
}
|
|
||||||
|
|
||||||
void on_minute(numeric_system ns) {
|
|
||||||
if (ns == numeric_system::standard)
|
|
||||||
return write(minute(), 2);
|
|
||||||
auto time = tm();
|
|
||||||
time.tm_min = minute();
|
|
||||||
format_localized(time, "%OM");
|
|
||||||
}
|
|
||||||
|
|
||||||
void on_second(numeric_system ns) {
|
|
||||||
if (ns == numeric_system::standard) {
|
|
||||||
write(second(), 2);
|
|
||||||
if (ms != std::chrono::milliseconds(0)) {
|
|
||||||
*out++ = '.';
|
|
||||||
write(to_int(ms.count()), 3);
|
|
||||||
}
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
auto time = tm();
|
|
||||||
time.tm_sec = second();
|
|
||||||
format_localized(time, "%OS");
|
|
||||||
}
|
|
||||||
|
|
||||||
void on_12_hour_time() { format_localized(time(), "%r"); }
|
|
||||||
|
|
||||||
void on_24_hour_time() {
|
|
||||||
write(hour(), 2);
|
|
||||||
*out++ = ':';
|
|
||||||
write(minute(), 2);
|
|
||||||
}
|
|
||||||
|
|
||||||
void on_iso_time() {
|
|
||||||
on_24_hour_time();
|
|
||||||
*out++ = ':';
|
|
||||||
write(second(), 2);
|
|
||||||
}
|
|
||||||
|
|
||||||
void on_am_pm() { format_localized(time(), "%p"); }
|
|
||||||
};
|
|
||||||
} // namespace internal
|
|
||||||
|
|
||||||
template <typename Period> FMT_CONSTEXPR const char *get_units() {
|
|
||||||
return FMT_NULL;
|
|
||||||
}
|
|
||||||
template <> FMT_CONSTEXPR const char *get_units<std::atto>() { return "as"; }
|
|
||||||
template <> FMT_CONSTEXPR const char *get_units<std::femto>() { return "fs"; }
|
|
||||||
template <> FMT_CONSTEXPR const char *get_units<std::pico>() { return "ps"; }
|
|
||||||
template <> FMT_CONSTEXPR const char *get_units<std::nano>() { return "ns"; }
|
|
||||||
template <> FMT_CONSTEXPR const char *get_units<std::micro>() { return "µs"; }
|
|
||||||
template <> FMT_CONSTEXPR const char *get_units<std::milli>() { return "ms"; }
|
|
||||||
template <> FMT_CONSTEXPR const char *get_units<std::centi>() { return "cs"; }
|
|
||||||
template <> FMT_CONSTEXPR const char *get_units<std::deci>() { return "ds"; }
|
|
||||||
template <> FMT_CONSTEXPR const char *get_units<std::ratio<1>>() { return "s"; }
|
|
||||||
template <> FMT_CONSTEXPR const char *get_units<std::deca>() { return "das"; }
|
|
||||||
template <> FMT_CONSTEXPR const char *get_units<std::hecto>() { return "hs"; }
|
|
||||||
template <> FMT_CONSTEXPR const char *get_units<std::kilo>() { return "ks"; }
|
|
||||||
template <> FMT_CONSTEXPR const char *get_units<std::mega>() { return "Ms"; }
|
|
||||||
template <> FMT_CONSTEXPR const char *get_units<std::giga>() { return "Gs"; }
|
|
||||||
template <> FMT_CONSTEXPR const char *get_units<std::tera>() { return "Ts"; }
|
|
||||||
template <> FMT_CONSTEXPR const char *get_units<std::peta>() { return "Ps"; }
|
|
||||||
template <> FMT_CONSTEXPR const char *get_units<std::exa>() { return "Es"; }
|
|
||||||
template <> FMT_CONSTEXPR const char *get_units<std::ratio<60>>() {
|
|
||||||
return "m";
|
|
||||||
}
|
|
||||||
template <> FMT_CONSTEXPR const char *get_units<std::ratio<3600>>() {
|
|
||||||
return "h";
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename Rep, typename Period, typename Char>
|
|
||||||
struct formatter<std::chrono::duration<Rep, Period>, Char> {
|
|
||||||
private:
|
|
||||||
align_spec spec;
|
|
||||||
internal::arg_ref<Char> width_ref;
|
|
||||||
mutable basic_string_view<Char> format_str;
|
|
||||||
typedef std::chrono::duration<Rep, Period> duration;
|
|
||||||
|
|
||||||
struct spec_handler {
|
|
||||||
formatter &f;
|
|
||||||
basic_parse_context<Char> &context;
|
|
||||||
|
|
||||||
typedef internal::arg_ref<Char> arg_ref_type;
|
|
||||||
|
|
||||||
template <typename Id>
|
|
||||||
FMT_CONSTEXPR arg_ref_type make_arg_ref(Id arg_id) {
|
|
||||||
context.check_arg_id(arg_id);
|
|
||||||
return arg_ref_type(arg_id);
|
|
||||||
}
|
|
||||||
|
|
||||||
FMT_CONSTEXPR arg_ref_type make_arg_ref(internal::auto_id) {
|
|
||||||
return arg_ref_type(context.next_arg_id());
|
|
||||||
}
|
|
||||||
|
|
||||||
void on_error(const char *msg) { throw format_error(msg); }
|
|
||||||
void on_fill(Char fill) { f.spec.fill_ = fill; }
|
|
||||||
void on_align(alignment align) { f.spec.align_ = align; }
|
|
||||||
void on_width(unsigned width) { f.spec.width_ = width; }
|
|
||||||
|
|
||||||
template <typename Id>
|
|
||||||
void on_dynamic_width(Id arg_id) {
|
|
||||||
f.width_ref = make_arg_ref(arg_id);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
public:
|
|
||||||
formatter() : spec() {}
|
|
||||||
|
|
||||||
FMT_CONSTEXPR auto parse(basic_parse_context<Char> &ctx)
|
|
||||||
-> decltype(ctx.begin()) {
|
|
||||||
auto begin = ctx.begin(), end = ctx.end();
|
|
||||||
if (begin == end) return begin;
|
|
||||||
spec_handler handler{*this, ctx};
|
|
||||||
begin = internal::parse_align(begin, end, handler);
|
|
||||||
if (begin == end) return begin;
|
|
||||||
begin = internal::parse_width(begin, end, handler);
|
|
||||||
end = parse_chrono_format(begin, end, internal::chrono_format_checker());
|
|
||||||
format_str = basic_string_view<Char>(&*begin, internal::to_unsigned(end - begin));
|
|
||||||
return end;
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename FormatContext>
|
|
||||||
auto format(const duration &d, FormatContext &ctx)
|
|
||||||
-> decltype(ctx.out()) {
|
|
||||||
auto begin = format_str.begin(), end = format_str.end();
|
|
||||||
memory_buffer buf;
|
|
||||||
typedef output_range<decltype(ctx.out()), Char> range;
|
|
||||||
basic_writer<range> w(range(ctx.out()));
|
|
||||||
if (begin == end || *begin == '}') {
|
|
||||||
if (const char *unit = get_units<Period>())
|
|
||||||
format_to(buf, "{}{}", d.count(), unit);
|
|
||||||
else if (Period::den == 1)
|
|
||||||
format_to(buf, "{}[{}]s", d.count(), Period::num);
|
|
||||||
else
|
|
||||||
format_to(buf, "{}[{}/{}]s", d.count(), Period::num, Period::den);
|
|
||||||
internal::handle_dynamic_spec<internal::width_checker>(
|
|
||||||
spec.width_, width_ref, ctx);
|
|
||||||
} else {
|
|
||||||
auto out = std::back_inserter(buf);
|
|
||||||
internal::chrono_formatter<FormatContext, decltype(out)> f(ctx, out);
|
|
||||||
f.s = std::chrono::duration_cast<std::chrono::seconds>(d);
|
|
||||||
f.ms = std::chrono::duration_cast<std::chrono::milliseconds>(d - f.s);
|
|
||||||
parse_chrono_format(begin, end, f);
|
|
||||||
}
|
|
||||||
w.write(buf.data(), buf.size(), spec);
|
|
||||||
return w.out();
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
FMT_END_NAMESPACE
|
|
||||||
|
|
||||||
#endif // FMT_CHRONO_H_
|
|
||||||
577
deps/spdlog/fmt/bundled/color.h
vendored
577
deps/spdlog/fmt/bundled/color.h
vendored
@ -1,577 +0,0 @@
|
|||||||
// Formatting library for C++ - color support
|
|
||||||
//
|
|
||||||
// Copyright (c) 2018 - present, Victor Zverovich and fmt contributors
|
|
||||||
// All rights reserved.
|
|
||||||
//
|
|
||||||
// For the license information refer to format.h.
|
|
||||||
|
|
||||||
#ifndef FMT_COLOR_H_
|
|
||||||
#define FMT_COLOR_H_
|
|
||||||
|
|
||||||
#include "format.h"
|
|
||||||
|
|
||||||
FMT_BEGIN_NAMESPACE
|
|
||||||
|
|
||||||
#ifdef FMT_DEPRECATED_COLORS
|
|
||||||
|
|
||||||
// color and (v)print_colored are deprecated.
|
|
||||||
enum color { black, red, green, yellow, blue, magenta, cyan, white };
|
|
||||||
FMT_API void vprint_colored(color c, string_view format, format_args args);
|
|
||||||
FMT_API void vprint_colored(color c, wstring_view format, wformat_args args);
|
|
||||||
template <typename... Args>
|
|
||||||
inline void print_colored(color c, string_view format_str,
|
|
||||||
const Args & ... args) {
|
|
||||||
vprint_colored(c, format_str, make_format_args(args...));
|
|
||||||
}
|
|
||||||
template <typename... Args>
|
|
||||||
inline void print_colored(color c, wstring_view format_str,
|
|
||||||
const Args & ... args) {
|
|
||||||
vprint_colored(c, format_str, make_format_args<wformat_context>(args...));
|
|
||||||
}
|
|
||||||
|
|
||||||
inline void vprint_colored(color c, string_view format, format_args args) {
|
|
||||||
char escape[] = "\x1b[30m";
|
|
||||||
escape[3] = static_cast<char>('0' + c);
|
|
||||||
std::fputs(escape, stdout);
|
|
||||||
vprint(format, args);
|
|
||||||
std::fputs(internal::data::RESET_COLOR, stdout);
|
|
||||||
}
|
|
||||||
|
|
||||||
inline void vprint_colored(color c, wstring_view format, wformat_args args) {
|
|
||||||
wchar_t escape[] = L"\x1b[30m";
|
|
||||||
escape[3] = static_cast<wchar_t>('0' + c);
|
|
||||||
std::fputws(escape, stdout);
|
|
||||||
vprint(format, args);
|
|
||||||
std::fputws(internal::data::WRESET_COLOR, stdout);
|
|
||||||
}
|
|
||||||
|
|
||||||
#else
|
|
||||||
|
|
||||||
enum class color : uint32_t {
|
|
||||||
alice_blue = 0xF0F8FF, // rgb(240,248,255)
|
|
||||||
antique_white = 0xFAEBD7, // rgb(250,235,215)
|
|
||||||
aqua = 0x00FFFF, // rgb(0,255,255)
|
|
||||||
aquamarine = 0x7FFFD4, // rgb(127,255,212)
|
|
||||||
azure = 0xF0FFFF, // rgb(240,255,255)
|
|
||||||
beige = 0xF5F5DC, // rgb(245,245,220)
|
|
||||||
bisque = 0xFFE4C4, // rgb(255,228,196)
|
|
||||||
black = 0x000000, // rgb(0,0,0)
|
|
||||||
blanched_almond = 0xFFEBCD, // rgb(255,235,205)
|
|
||||||
blue = 0x0000FF, // rgb(0,0,255)
|
|
||||||
blue_violet = 0x8A2BE2, // rgb(138,43,226)
|
|
||||||
brown = 0xA52A2A, // rgb(165,42,42)
|
|
||||||
burly_wood = 0xDEB887, // rgb(222,184,135)
|
|
||||||
cadet_blue = 0x5F9EA0, // rgb(95,158,160)
|
|
||||||
chartreuse = 0x7FFF00, // rgb(127,255,0)
|
|
||||||
chocolate = 0xD2691E, // rgb(210,105,30)
|
|
||||||
coral = 0xFF7F50, // rgb(255,127,80)
|
|
||||||
cornflower_blue = 0x6495ED, // rgb(100,149,237)
|
|
||||||
cornsilk = 0xFFF8DC, // rgb(255,248,220)
|
|
||||||
crimson = 0xDC143C, // rgb(220,20,60)
|
|
||||||
cyan = 0x00FFFF, // rgb(0,255,255)
|
|
||||||
dark_blue = 0x00008B, // rgb(0,0,139)
|
|
||||||
dark_cyan = 0x008B8B, // rgb(0,139,139)
|
|
||||||
dark_golden_rod = 0xB8860B, // rgb(184,134,11)
|
|
||||||
dark_gray = 0xA9A9A9, // rgb(169,169,169)
|
|
||||||
dark_green = 0x006400, // rgb(0,100,0)
|
|
||||||
dark_khaki = 0xBDB76B, // rgb(189,183,107)
|
|
||||||
dark_magenta = 0x8B008B, // rgb(139,0,139)
|
|
||||||
dark_olive_green = 0x556B2F, // rgb(85,107,47)
|
|
||||||
dark_orange = 0xFF8C00, // rgb(255,140,0)
|
|
||||||
dark_orchid = 0x9932CC, // rgb(153,50,204)
|
|
||||||
dark_red = 0x8B0000, // rgb(139,0,0)
|
|
||||||
dark_salmon = 0xE9967A, // rgb(233,150,122)
|
|
||||||
dark_sea_green = 0x8FBC8F, // rgb(143,188,143)
|
|
||||||
dark_slate_blue = 0x483D8B, // rgb(72,61,139)
|
|
||||||
dark_slate_gray = 0x2F4F4F, // rgb(47,79,79)
|
|
||||||
dark_turquoise = 0x00CED1, // rgb(0,206,209)
|
|
||||||
dark_violet = 0x9400D3, // rgb(148,0,211)
|
|
||||||
deep_pink = 0xFF1493, // rgb(255,20,147)
|
|
||||||
deep_sky_blue = 0x00BFFF, // rgb(0,191,255)
|
|
||||||
dim_gray = 0x696969, // rgb(105,105,105)
|
|
||||||
dodger_blue = 0x1E90FF, // rgb(30,144,255)
|
|
||||||
fire_brick = 0xB22222, // rgb(178,34,34)
|
|
||||||
floral_white = 0xFFFAF0, // rgb(255,250,240)
|
|
||||||
forest_green = 0x228B22, // rgb(34,139,34)
|
|
||||||
fuchsia = 0xFF00FF, // rgb(255,0,255)
|
|
||||||
gainsboro = 0xDCDCDC, // rgb(220,220,220)
|
|
||||||
ghost_white = 0xF8F8FF, // rgb(248,248,255)
|
|
||||||
gold = 0xFFD700, // rgb(255,215,0)
|
|
||||||
golden_rod = 0xDAA520, // rgb(218,165,32)
|
|
||||||
gray = 0x808080, // rgb(128,128,128)
|
|
||||||
green = 0x008000, // rgb(0,128,0)
|
|
||||||
green_yellow = 0xADFF2F, // rgb(173,255,47)
|
|
||||||
honey_dew = 0xF0FFF0, // rgb(240,255,240)
|
|
||||||
hot_pink = 0xFF69B4, // rgb(255,105,180)
|
|
||||||
indian_red = 0xCD5C5C, // rgb(205,92,92)
|
|
||||||
indigo = 0x4B0082, // rgb(75,0,130)
|
|
||||||
ivory = 0xFFFFF0, // rgb(255,255,240)
|
|
||||||
khaki = 0xF0E68C, // rgb(240,230,140)
|
|
||||||
lavender = 0xE6E6FA, // rgb(230,230,250)
|
|
||||||
lavender_blush = 0xFFF0F5, // rgb(255,240,245)
|
|
||||||
lawn_green = 0x7CFC00, // rgb(124,252,0)
|
|
||||||
lemon_chiffon = 0xFFFACD, // rgb(255,250,205)
|
|
||||||
light_blue = 0xADD8E6, // rgb(173,216,230)
|
|
||||||
light_coral = 0xF08080, // rgb(240,128,128)
|
|
||||||
light_cyan = 0xE0FFFF, // rgb(224,255,255)
|
|
||||||
light_golden_rod_yellow = 0xFAFAD2, // rgb(250,250,210)
|
|
||||||
light_gray = 0xD3D3D3, // rgb(211,211,211)
|
|
||||||
light_green = 0x90EE90, // rgb(144,238,144)
|
|
||||||
light_pink = 0xFFB6C1, // rgb(255,182,193)
|
|
||||||
light_salmon = 0xFFA07A, // rgb(255,160,122)
|
|
||||||
light_sea_green = 0x20B2AA, // rgb(32,178,170)
|
|
||||||
light_sky_blue = 0x87CEFA, // rgb(135,206,250)
|
|
||||||
light_slate_gray = 0x778899, // rgb(119,136,153)
|
|
||||||
light_steel_blue = 0xB0C4DE, // rgb(176,196,222)
|
|
||||||
light_yellow = 0xFFFFE0, // rgb(255,255,224)
|
|
||||||
lime = 0x00FF00, // rgb(0,255,0)
|
|
||||||
lime_green = 0x32CD32, // rgb(50,205,50)
|
|
||||||
linen = 0xFAF0E6, // rgb(250,240,230)
|
|
||||||
magenta = 0xFF00FF, // rgb(255,0,255)
|
|
||||||
maroon = 0x800000, // rgb(128,0,0)
|
|
||||||
medium_aquamarine = 0x66CDAA, // rgb(102,205,170)
|
|
||||||
medium_blue = 0x0000CD, // rgb(0,0,205)
|
|
||||||
medium_orchid = 0xBA55D3, // rgb(186,85,211)
|
|
||||||
medium_purple = 0x9370DB, // rgb(147,112,219)
|
|
||||||
medium_sea_green = 0x3CB371, // rgb(60,179,113)
|
|
||||||
medium_slate_blue = 0x7B68EE, // rgb(123,104,238)
|
|
||||||
medium_spring_green = 0x00FA9A, // rgb(0,250,154)
|
|
||||||
medium_turquoise = 0x48D1CC, // rgb(72,209,204)
|
|
||||||
medium_violet_red = 0xC71585, // rgb(199,21,133)
|
|
||||||
midnight_blue = 0x191970, // rgb(25,25,112)
|
|
||||||
mint_cream = 0xF5FFFA, // rgb(245,255,250)
|
|
||||||
misty_rose = 0xFFE4E1, // rgb(255,228,225)
|
|
||||||
moccasin = 0xFFE4B5, // rgb(255,228,181)
|
|
||||||
navajo_white = 0xFFDEAD, // rgb(255,222,173)
|
|
||||||
navy = 0x000080, // rgb(0,0,128)
|
|
||||||
old_lace = 0xFDF5E6, // rgb(253,245,230)
|
|
||||||
olive = 0x808000, // rgb(128,128,0)
|
|
||||||
olive_drab = 0x6B8E23, // rgb(107,142,35)
|
|
||||||
orange = 0xFFA500, // rgb(255,165,0)
|
|
||||||
orange_red = 0xFF4500, // rgb(255,69,0)
|
|
||||||
orchid = 0xDA70D6, // rgb(218,112,214)
|
|
||||||
pale_golden_rod = 0xEEE8AA, // rgb(238,232,170)
|
|
||||||
pale_green = 0x98FB98, // rgb(152,251,152)
|
|
||||||
pale_turquoise = 0xAFEEEE, // rgb(175,238,238)
|
|
||||||
pale_violet_red = 0xDB7093, // rgb(219,112,147)
|
|
||||||
papaya_whip = 0xFFEFD5, // rgb(255,239,213)
|
|
||||||
peach_puff = 0xFFDAB9, // rgb(255,218,185)
|
|
||||||
peru = 0xCD853F, // rgb(205,133,63)
|
|
||||||
pink = 0xFFC0CB, // rgb(255,192,203)
|
|
||||||
plum = 0xDDA0DD, // rgb(221,160,221)
|
|
||||||
powder_blue = 0xB0E0E6, // rgb(176,224,230)
|
|
||||||
purple = 0x800080, // rgb(128,0,128)
|
|
||||||
rebecca_purple = 0x663399, // rgb(102,51,153)
|
|
||||||
red = 0xFF0000, // rgb(255,0,0)
|
|
||||||
rosy_brown = 0xBC8F8F, // rgb(188,143,143)
|
|
||||||
royal_blue = 0x4169E1, // rgb(65,105,225)
|
|
||||||
saddle_brown = 0x8B4513, // rgb(139,69,19)
|
|
||||||
salmon = 0xFA8072, // rgb(250,128,114)
|
|
||||||
sandy_brown = 0xF4A460, // rgb(244,164,96)
|
|
||||||
sea_green = 0x2E8B57, // rgb(46,139,87)
|
|
||||||
sea_shell = 0xFFF5EE, // rgb(255,245,238)
|
|
||||||
sienna = 0xA0522D, // rgb(160,82,45)
|
|
||||||
silver = 0xC0C0C0, // rgb(192,192,192)
|
|
||||||
sky_blue = 0x87CEEB, // rgb(135,206,235)
|
|
||||||
slate_blue = 0x6A5ACD, // rgb(106,90,205)
|
|
||||||
slate_gray = 0x708090, // rgb(112,128,144)
|
|
||||||
snow = 0xFFFAFA, // rgb(255,250,250)
|
|
||||||
spring_green = 0x00FF7F, // rgb(0,255,127)
|
|
||||||
steel_blue = 0x4682B4, // rgb(70,130,180)
|
|
||||||
tan = 0xD2B48C, // rgb(210,180,140)
|
|
||||||
teal = 0x008080, // rgb(0,128,128)
|
|
||||||
thistle = 0xD8BFD8, // rgb(216,191,216)
|
|
||||||
tomato = 0xFF6347, // rgb(255,99,71)
|
|
||||||
turquoise = 0x40E0D0, // rgb(64,224,208)
|
|
||||||
violet = 0xEE82EE, // rgb(238,130,238)
|
|
||||||
wheat = 0xF5DEB3, // rgb(245,222,179)
|
|
||||||
white = 0xFFFFFF, // rgb(255,255,255)
|
|
||||||
white_smoke = 0xF5F5F5, // rgb(245,245,245)
|
|
||||||
yellow = 0xFFFF00, // rgb(255,255,0)
|
|
||||||
yellow_green = 0x9ACD32 // rgb(154,205,50)
|
|
||||||
}; // enum class color
|
|
||||||
|
|
||||||
enum class terminal_color : uint8_t {
|
|
||||||
black = 30,
|
|
||||||
red,
|
|
||||||
green,
|
|
||||||
yellow,
|
|
||||||
blue,
|
|
||||||
magenta,
|
|
||||||
cyan,
|
|
||||||
white,
|
|
||||||
bright_black = 90,
|
|
||||||
bright_red,
|
|
||||||
bright_green,
|
|
||||||
bright_yellow,
|
|
||||||
bright_blue,
|
|
||||||
bright_magenta,
|
|
||||||
bright_cyan,
|
|
||||||
bright_white
|
|
||||||
}; // enum class terminal_color
|
|
||||||
|
|
||||||
enum class emphasis : uint8_t {
|
|
||||||
bold = 1,
|
|
||||||
italic = 1 << 1,
|
|
||||||
underline = 1 << 2,
|
|
||||||
strikethrough = 1 << 3
|
|
||||||
}; // enum class emphasis
|
|
||||||
|
|
||||||
// rgb is a struct for red, green and blue colors.
|
|
||||||
// We use rgb as name because some editors will show it as color direct in the
|
|
||||||
// editor.
|
|
||||||
struct rgb {
|
|
||||||
FMT_CONSTEXPR_DECL rgb() : r(0), g(0), b(0) {}
|
|
||||||
FMT_CONSTEXPR_DECL rgb(uint8_t r_, uint8_t g_, uint8_t b_)
|
|
||||||
: r(r_), g(g_), b(b_) {}
|
|
||||||
FMT_CONSTEXPR_DECL rgb(uint32_t hex)
|
|
||||||
: r((hex >> 16) & 0xFF), g((hex >> 8) & 0xFF), b((hex) & 0xFF) {}
|
|
||||||
FMT_CONSTEXPR_DECL rgb(color hex)
|
|
||||||
: r((uint32_t(hex) >> 16) & 0xFF), g((uint32_t(hex) >> 8) & 0xFF),
|
|
||||||
b(uint32_t(hex) & 0xFF) {}
|
|
||||||
uint8_t r;
|
|
||||||
uint8_t g;
|
|
||||||
uint8_t b;
|
|
||||||
};
|
|
||||||
|
|
||||||
namespace internal {
|
|
||||||
|
|
||||||
// color is a struct of either a rgb color or a terminal color.
|
|
||||||
struct color_type {
|
|
||||||
FMT_CONSTEXPR color_type() FMT_NOEXCEPT
|
|
||||||
: is_rgb(), value{} {}
|
|
||||||
FMT_CONSTEXPR color_type(color rgb_color) FMT_NOEXCEPT
|
|
||||||
: is_rgb(true), value{} {
|
|
||||||
value.rgb_color = static_cast<uint32_t>(rgb_color);
|
|
||||||
}
|
|
||||||
FMT_CONSTEXPR color_type(rgb rgb_color) FMT_NOEXCEPT
|
|
||||||
: is_rgb(true), value{} {
|
|
||||||
value.rgb_color = (static_cast<uint32_t>(rgb_color.r) << 16)
|
|
||||||
| (static_cast<uint32_t>(rgb_color.g) << 8) | rgb_color.b;
|
|
||||||
}
|
|
||||||
FMT_CONSTEXPR color_type(terminal_color term_color) FMT_NOEXCEPT
|
|
||||||
: is_rgb(), value{} {
|
|
||||||
value.term_color = static_cast<uint8_t>(term_color);
|
|
||||||
}
|
|
||||||
bool is_rgb;
|
|
||||||
union color_union {
|
|
||||||
uint8_t term_color;
|
|
||||||
uint32_t rgb_color;
|
|
||||||
} value;
|
|
||||||
};
|
|
||||||
} // namespace internal
|
|
||||||
|
|
||||||
// Experimental text formatting support.
|
|
||||||
class text_style {
|
|
||||||
public:
|
|
||||||
FMT_CONSTEXPR text_style(emphasis em = emphasis()) FMT_NOEXCEPT
|
|
||||||
: set_foreground_color(), set_background_color(), ems(em) {}
|
|
||||||
|
|
||||||
FMT_CONSTEXPR text_style &operator|=(const text_style &rhs) {
|
|
||||||
if (!set_foreground_color) {
|
|
||||||
set_foreground_color = rhs.set_foreground_color;
|
|
||||||
foreground_color = rhs.foreground_color;
|
|
||||||
} else if (rhs.set_foreground_color) {
|
|
||||||
if (!foreground_color.is_rgb || !rhs.foreground_color.is_rgb)
|
|
||||||
throw format_error("can't OR a terminal color");
|
|
||||||
foreground_color.value.rgb_color |= rhs.foreground_color.value.rgb_color;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!set_background_color) {
|
|
||||||
set_background_color = rhs.set_background_color;
|
|
||||||
background_color = rhs.background_color;
|
|
||||||
} else if (rhs.set_background_color) {
|
|
||||||
if (!background_color.is_rgb || !rhs.background_color.is_rgb)
|
|
||||||
throw format_error("can't OR a terminal color");
|
|
||||||
background_color.value.rgb_color |= rhs.background_color.value.rgb_color;
|
|
||||||
}
|
|
||||||
|
|
||||||
ems = static_cast<emphasis>(static_cast<uint8_t>(ems) |
|
|
||||||
static_cast<uint8_t>(rhs.ems));
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
|
|
||||||
friend FMT_CONSTEXPR
|
|
||||||
text_style operator|(text_style lhs, const text_style &rhs) {
|
|
||||||
return lhs |= rhs;
|
|
||||||
}
|
|
||||||
|
|
||||||
FMT_CONSTEXPR text_style &operator&=(const text_style &rhs) {
|
|
||||||
if (!set_foreground_color) {
|
|
||||||
set_foreground_color = rhs.set_foreground_color;
|
|
||||||
foreground_color = rhs.foreground_color;
|
|
||||||
} else if (rhs.set_foreground_color) {
|
|
||||||
if (!foreground_color.is_rgb || !rhs.foreground_color.is_rgb)
|
|
||||||
throw format_error("can't AND a terminal color");
|
|
||||||
foreground_color.value.rgb_color &= rhs.foreground_color.value.rgb_color;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!set_background_color) {
|
|
||||||
set_background_color = rhs.set_background_color;
|
|
||||||
background_color = rhs.background_color;
|
|
||||||
} else if (rhs.set_background_color) {
|
|
||||||
if (!background_color.is_rgb || !rhs.background_color.is_rgb)
|
|
||||||
throw format_error("can't AND a terminal color");
|
|
||||||
background_color.value.rgb_color &= rhs.background_color.value.rgb_color;
|
|
||||||
}
|
|
||||||
|
|
||||||
ems = static_cast<emphasis>(static_cast<uint8_t>(ems) &
|
|
||||||
static_cast<uint8_t>(rhs.ems));
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
|
|
||||||
friend FMT_CONSTEXPR
|
|
||||||
text_style operator&(text_style lhs, const text_style &rhs) {
|
|
||||||
return lhs &= rhs;
|
|
||||||
}
|
|
||||||
|
|
||||||
FMT_CONSTEXPR bool has_foreground() const FMT_NOEXCEPT {
|
|
||||||
return set_foreground_color;
|
|
||||||
}
|
|
||||||
FMT_CONSTEXPR bool has_background() const FMT_NOEXCEPT {
|
|
||||||
return set_background_color;
|
|
||||||
}
|
|
||||||
FMT_CONSTEXPR bool has_emphasis() const FMT_NOEXCEPT {
|
|
||||||
return static_cast<uint8_t>(ems) != 0;
|
|
||||||
}
|
|
||||||
FMT_CONSTEXPR internal::color_type get_foreground() const FMT_NOEXCEPT {
|
|
||||||
assert(has_foreground() && "no foreground specified for this style");
|
|
||||||
return foreground_color;
|
|
||||||
}
|
|
||||||
FMT_CONSTEXPR internal::color_type get_background() const FMT_NOEXCEPT {
|
|
||||||
assert(has_background() && "no background specified for this style");
|
|
||||||
return background_color;
|
|
||||||
}
|
|
||||||
FMT_CONSTEXPR emphasis get_emphasis() const FMT_NOEXCEPT {
|
|
||||||
assert(has_emphasis() && "no emphasis specified for this style");
|
|
||||||
return ems;
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
|
||||||
FMT_CONSTEXPR text_style(bool is_foreground,
|
|
||||||
internal::color_type text_color) FMT_NOEXCEPT
|
|
||||||
: set_foreground_color(),
|
|
||||||
set_background_color(),
|
|
||||||
ems() {
|
|
||||||
if (is_foreground) {
|
|
||||||
foreground_color = text_color;
|
|
||||||
set_foreground_color = true;
|
|
||||||
} else {
|
|
||||||
background_color = text_color;
|
|
||||||
set_background_color = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
friend FMT_CONSTEXPR_DECL text_style fg(internal::color_type foreground)
|
|
||||||
FMT_NOEXCEPT;
|
|
||||||
friend FMT_CONSTEXPR_DECL text_style bg(internal::color_type background)
|
|
||||||
FMT_NOEXCEPT;
|
|
||||||
|
|
||||||
internal::color_type foreground_color;
|
|
||||||
internal::color_type background_color;
|
|
||||||
bool set_foreground_color;
|
|
||||||
bool set_background_color;
|
|
||||||
emphasis ems;
|
|
||||||
};
|
|
||||||
|
|
||||||
FMT_CONSTEXPR text_style fg(internal::color_type foreground) FMT_NOEXCEPT {
|
|
||||||
return text_style(/*is_foreground=*/true, foreground);
|
|
||||||
}
|
|
||||||
|
|
||||||
FMT_CONSTEXPR text_style bg(internal::color_type background) FMT_NOEXCEPT {
|
|
||||||
return text_style(/*is_foreground=*/false, background);
|
|
||||||
}
|
|
||||||
|
|
||||||
FMT_CONSTEXPR text_style operator|(emphasis lhs, emphasis rhs) FMT_NOEXCEPT {
|
|
||||||
return text_style(lhs) | rhs;
|
|
||||||
}
|
|
||||||
|
|
||||||
namespace internal {
|
|
||||||
|
|
||||||
template <typename Char>
|
|
||||||
struct ansi_color_escape {
|
|
||||||
FMT_CONSTEXPR ansi_color_escape(internal::color_type text_color,
|
|
||||||
const char * esc) FMT_NOEXCEPT {
|
|
||||||
// If we have a terminal color, we need to output another escape code
|
|
||||||
// sequence.
|
|
||||||
if (!text_color.is_rgb) {
|
|
||||||
bool is_background = esc == internal::data::BACKGROUND_COLOR;
|
|
||||||
uint32_t value = text_color.value.term_color;
|
|
||||||
// Background ASCII codes are the same as the foreground ones but with
|
|
||||||
// 10 more.
|
|
||||||
if (is_background)
|
|
||||||
value += 10u;
|
|
||||||
|
|
||||||
std::size_t index = 0;
|
|
||||||
buffer[index++] = static_cast<Char>('\x1b');
|
|
||||||
buffer[index++] = static_cast<Char>('[');
|
|
||||||
|
|
||||||
if (value >= 100u) {
|
|
||||||
buffer[index++] = static_cast<Char>('1');
|
|
||||||
value %= 100u;
|
|
||||||
}
|
|
||||||
buffer[index++] = static_cast<Char>('0' + value / 10u);
|
|
||||||
buffer[index++] = static_cast<Char>('0' + value % 10u);
|
|
||||||
|
|
||||||
buffer[index++] = static_cast<Char>('m');
|
|
||||||
buffer[index++] = static_cast<Char>('\0');
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
for (int i = 0; i < 7; i++) {
|
|
||||||
buffer[i] = static_cast<Char>(esc[i]);
|
|
||||||
}
|
|
||||||
rgb color(text_color.value.rgb_color);
|
|
||||||
to_esc(color.r, buffer + 7, ';');
|
|
||||||
to_esc(color.g, buffer + 11, ';');
|
|
||||||
to_esc(color.b, buffer + 15, 'm');
|
|
||||||
buffer[19] = static_cast<Char>(0);
|
|
||||||
}
|
|
||||||
FMT_CONSTEXPR ansi_color_escape(emphasis em) FMT_NOEXCEPT {
|
|
||||||
uint8_t em_codes[4] = {};
|
|
||||||
uint8_t em_bits = static_cast<uint8_t>(em);
|
|
||||||
if (em_bits & static_cast<uint8_t>(emphasis::bold))
|
|
||||||
em_codes[0] = 1;
|
|
||||||
if (em_bits & static_cast<uint8_t>(emphasis::italic))
|
|
||||||
em_codes[1] = 3;
|
|
||||||
if (em_bits & static_cast<uint8_t>(emphasis::underline))
|
|
||||||
em_codes[2] = 4;
|
|
||||||
if (em_bits & static_cast<uint8_t>(emphasis::strikethrough))
|
|
||||||
em_codes[3] = 9;
|
|
||||||
|
|
||||||
std::size_t index = 0;
|
|
||||||
for (int i = 0; i < 4; ++i) {
|
|
||||||
if (!em_codes[i])
|
|
||||||
continue;
|
|
||||||
buffer[index++] = static_cast<Char>('\x1b');
|
|
||||||
buffer[index++] = static_cast<Char>('[');
|
|
||||||
buffer[index++] = static_cast<Char>('0' + em_codes[i]);
|
|
||||||
buffer[index++] = static_cast<Char>('m');
|
|
||||||
}
|
|
||||||
buffer[index++] = static_cast<Char>(0);
|
|
||||||
}
|
|
||||||
FMT_CONSTEXPR operator const Char *() const FMT_NOEXCEPT { return buffer; }
|
|
||||||
|
|
||||||
private:
|
|
||||||
Char buffer[7u + 3u * 4u + 1u];
|
|
||||||
|
|
||||||
static FMT_CONSTEXPR void to_esc(uint8_t c, Char *out,
|
|
||||||
char delimiter) FMT_NOEXCEPT {
|
|
||||||
out[0] = static_cast<Char>('0' + c / 100);
|
|
||||||
out[1] = static_cast<Char>('0' + c / 10 % 10);
|
|
||||||
out[2] = static_cast<Char>('0' + c % 10);
|
|
||||||
out[3] = static_cast<Char>(delimiter);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
template <typename Char>
|
|
||||||
FMT_CONSTEXPR ansi_color_escape<Char>
|
|
||||||
make_foreground_color(internal::color_type foreground) FMT_NOEXCEPT {
|
|
||||||
return ansi_color_escape<Char>(foreground, internal::data::FOREGROUND_COLOR);
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename Char>
|
|
||||||
FMT_CONSTEXPR ansi_color_escape<Char>
|
|
||||||
make_background_color(internal::color_type background) FMT_NOEXCEPT {
|
|
||||||
return ansi_color_escape<Char>(background, internal::data::BACKGROUND_COLOR);
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename Char>
|
|
||||||
FMT_CONSTEXPR ansi_color_escape<Char>
|
|
||||||
make_emphasis(emphasis em) FMT_NOEXCEPT {
|
|
||||||
return ansi_color_escape<Char>(em);
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename Char>
|
|
||||||
inline void fputs(const Char *chars, FILE *stream) FMT_NOEXCEPT {
|
|
||||||
std::fputs(chars, stream);
|
|
||||||
}
|
|
||||||
|
|
||||||
template <>
|
|
||||||
inline void fputs<wchar_t>(const wchar_t *chars, FILE *stream) FMT_NOEXCEPT {
|
|
||||||
std::fputws(chars, stream);
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename Char>
|
|
||||||
inline void reset_color(FILE *stream) FMT_NOEXCEPT {
|
|
||||||
fputs(internal::data::RESET_COLOR, stream);
|
|
||||||
}
|
|
||||||
|
|
||||||
template <>
|
|
||||||
inline void reset_color<wchar_t>(FILE *stream) FMT_NOEXCEPT {
|
|
||||||
fputs(internal::data::WRESET_COLOR, stream);
|
|
||||||
}
|
|
||||||
|
|
||||||
// The following specialiazation disables using std::FILE as a character type,
|
|
||||||
// which is needed because or else
|
|
||||||
// fmt::print(stderr, fmt::emphasis::bold, "");
|
|
||||||
// would take stderr (a std::FILE *) as the format string.
|
|
||||||
template <>
|
|
||||||
struct is_string<std::FILE *> : std::false_type {};
|
|
||||||
template <>
|
|
||||||
struct is_string<const std::FILE *> : std::false_type {};
|
|
||||||
} // namespace internal
|
|
||||||
|
|
||||||
template <
|
|
||||||
typename S, typename Char = typename internal::char_t<S>::type>
|
|
||||||
void vprint(std::FILE *f, const text_style &ts, const S &format,
|
|
||||||
basic_format_args<typename buffer_context<Char>::type> args) {
|
|
||||||
bool has_style = false;
|
|
||||||
if (ts.has_emphasis()) {
|
|
||||||
has_style = true;
|
|
||||||
internal::fputs<Char>(
|
|
||||||
internal::make_emphasis<Char>(ts.get_emphasis()), f);
|
|
||||||
}
|
|
||||||
if (ts.has_foreground()) {
|
|
||||||
has_style = true;
|
|
||||||
internal::fputs<Char>(
|
|
||||||
internal::make_foreground_color<Char>(ts.get_foreground()), f);
|
|
||||||
}
|
|
||||||
if (ts.has_background()) {
|
|
||||||
has_style = true;
|
|
||||||
internal::fputs<Char>(
|
|
||||||
internal::make_background_color<Char>(ts.get_background()), f);
|
|
||||||
}
|
|
||||||
vprint(f, format, args);
|
|
||||||
if (has_style) {
|
|
||||||
internal::reset_color<Char>(f);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
Formats a string and prints it to the specified file stream using ANSI
|
|
||||||
escape sequences to specify text formatting.
|
|
||||||
Example:
|
|
||||||
fmt::print(fmt::emphasis::bold | fg(fmt::color::red),
|
|
||||||
"Elapsed time: {0:.2f} seconds", 1.23);
|
|
||||||
*/
|
|
||||||
template <typename String, typename... Args>
|
|
||||||
typename std::enable_if<internal::is_string<String>::value>::type print(
|
|
||||||
std::FILE *f, const text_style &ts, const String &format_str,
|
|
||||||
const Args &... args) {
|
|
||||||
internal::check_format_string<Args...>(format_str);
|
|
||||||
typedef typename internal::char_t<String>::type char_t;
|
|
||||||
typedef typename buffer_context<char_t>::type context_t;
|
|
||||||
format_arg_store<context_t, Args...> as{args...};
|
|
||||||
vprint(f, ts, format_str, basic_format_args<context_t>(as));
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
Formats a string and prints it to stdout using ANSI escape sequences to
|
|
||||||
specify text formatting.
|
|
||||||
Example:
|
|
||||||
fmt::print(fmt::emphasis::bold | fg(fmt::color::red),
|
|
||||||
"Elapsed time: {0:.2f} seconds", 1.23);
|
|
||||||
*/
|
|
||||||
template <typename String, typename... Args>
|
|
||||||
typename std::enable_if<internal::is_string<String>::value>::type print(
|
|
||||||
const text_style &ts, const String &format_str,
|
|
||||||
const Args &... args) {
|
|
||||||
return print(stdout, ts, format_str, args...);
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
|
||||||
|
|
||||||
FMT_END_NAMESPACE
|
|
||||||
|
|
||||||
#endif // FMT_COLOR_H_
|
|
||||||
1502
deps/spdlog/fmt/bundled/core.h
vendored
1502
deps/spdlog/fmt/bundled/core.h
vendored
File diff suppressed because it is too large
Load Diff
972
deps/spdlog/fmt/bundled/format-inl.h
vendored
972
deps/spdlog/fmt/bundled/format-inl.h
vendored
@ -1,972 +0,0 @@
|
|||||||
// Formatting library for C++
|
|
||||||
//
|
|
||||||
// Copyright (c) 2012 - 2016, Victor Zverovich
|
|
||||||
// All rights reserved.
|
|
||||||
//
|
|
||||||
// For the license information refer to format.h.
|
|
||||||
|
|
||||||
#ifndef FMT_FORMAT_INL_H_
|
|
||||||
#define FMT_FORMAT_INL_H_
|
|
||||||
|
|
||||||
#include "format.h"
|
|
||||||
|
|
||||||
#include <string.h>
|
|
||||||
|
|
||||||
#include <cctype>
|
|
||||||
#include <cerrno>
|
|
||||||
#include <climits>
|
|
||||||
#include <cmath>
|
|
||||||
#include <cstdarg>
|
|
||||||
#include <cstddef> // for std::ptrdiff_t
|
|
||||||
#include <cstring> // for std::memmove
|
|
||||||
#if !defined(FMT_STATIC_THOUSANDS_SEPARATOR)
|
|
||||||
# include <locale>
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#if FMT_USE_WINDOWS_H
|
|
||||||
# if !defined(FMT_HEADER_ONLY) && !defined(WIN32_LEAN_AND_MEAN)
|
|
||||||
# define WIN32_LEAN_AND_MEAN
|
|
||||||
# endif
|
|
||||||
# if defined(NOMINMAX) || defined(FMT_WIN_MINMAX)
|
|
||||||
# include <windows.h>
|
|
||||||
# else
|
|
||||||
# define NOMINMAX
|
|
||||||
# include <windows.h>
|
|
||||||
# undef NOMINMAX
|
|
||||||
# endif
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#if FMT_EXCEPTIONS
|
|
||||||
# define FMT_TRY try
|
|
||||||
# define FMT_CATCH(x) catch (x)
|
|
||||||
#else
|
|
||||||
# define FMT_TRY if (true)
|
|
||||||
# define FMT_CATCH(x) if (false)
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifdef _MSC_VER
|
|
||||||
# pragma warning(push)
|
|
||||||
# pragma warning(disable: 4127) // conditional expression is constant
|
|
||||||
# pragma warning(disable: 4702) // unreachable code
|
|
||||||
// Disable deprecation warning for strerror. The latter is not called but
|
|
||||||
// MSVC fails to detect it.
|
|
||||||
# pragma warning(disable: 4996)
|
|
||||||
#endif
|
|
||||||
|
|
||||||
// Dummy implementations of strerror_r and strerror_s called if corresponding
|
|
||||||
// system functions are not available.
|
|
||||||
inline fmt::internal::null<> strerror_r(int, char *, ...) {
|
|
||||||
return fmt::internal::null<>();
|
|
||||||
}
|
|
||||||
inline fmt::internal::null<> strerror_s(char *, std::size_t, ...) {
|
|
||||||
return fmt::internal::null<>();
|
|
||||||
}
|
|
||||||
|
|
||||||
FMT_BEGIN_NAMESPACE
|
|
||||||
|
|
||||||
namespace {
|
|
||||||
|
|
||||||
#ifndef _MSC_VER
|
|
||||||
# define FMT_SNPRINTF snprintf
|
|
||||||
#else // _MSC_VER
|
|
||||||
inline int fmt_snprintf(char *buffer, size_t size, const char *format, ...) {
|
|
||||||
va_list args;
|
|
||||||
va_start(args, format);
|
|
||||||
int result = vsnprintf_s(buffer, size, _TRUNCATE, format, args);
|
|
||||||
va_end(args);
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
# define FMT_SNPRINTF fmt_snprintf
|
|
||||||
#endif // _MSC_VER
|
|
||||||
|
|
||||||
#if defined(_WIN32) && defined(__MINGW32__) && !defined(__NO_ISOCEXT)
|
|
||||||
# define FMT_SWPRINTF snwprintf
|
|
||||||
#else
|
|
||||||
# define FMT_SWPRINTF swprintf
|
|
||||||
#endif // defined(_WIN32) && defined(__MINGW32__) && !defined(__NO_ISOCEXT)
|
|
||||||
|
|
||||||
typedef void (*FormatFunc)(internal::buffer &, int, string_view);
|
|
||||||
|
|
||||||
// Portable thread-safe version of strerror.
|
|
||||||
// Sets buffer to point to a string describing the error code.
|
|
||||||
// This can be either a pointer to a string stored in buffer,
|
|
||||||
// or a pointer to some static immutable string.
|
|
||||||
// Returns one of the following values:
|
|
||||||
// 0 - success
|
|
||||||
// ERANGE - buffer is not large enough to store the error message
|
|
||||||
// other - failure
|
|
||||||
// Buffer should be at least of size 1.
|
|
||||||
int safe_strerror(
|
|
||||||
int error_code, char *&buffer, std::size_t buffer_size) FMT_NOEXCEPT {
|
|
||||||
FMT_ASSERT(buffer != FMT_NULL && buffer_size != 0, "invalid buffer");
|
|
||||||
|
|
||||||
class dispatcher {
|
|
||||||
private:
|
|
||||||
int error_code_;
|
|
||||||
char *&buffer_;
|
|
||||||
std::size_t buffer_size_;
|
|
||||||
|
|
||||||
// A noop assignment operator to avoid bogus warnings.
|
|
||||||
void operator=(const dispatcher &) {}
|
|
||||||
|
|
||||||
// Handle the result of XSI-compliant version of strerror_r.
|
|
||||||
int handle(int result) {
|
|
||||||
// glibc versions before 2.13 return result in errno.
|
|
||||||
return result == -1 ? errno : result;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Handle the result of GNU-specific version of strerror_r.
|
|
||||||
int handle(char *message) {
|
|
||||||
// If the buffer is full then the message is probably truncated.
|
|
||||||
if (message == buffer_ && strlen(buffer_) == buffer_size_ - 1)
|
|
||||||
return ERANGE;
|
|
||||||
buffer_ = message;
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Handle the case when strerror_r is not available.
|
|
||||||
int handle(internal::null<>) {
|
|
||||||
return fallback(strerror_s(buffer_, buffer_size_, error_code_));
|
|
||||||
}
|
|
||||||
|
|
||||||
// Fallback to strerror_s when strerror_r is not available.
|
|
||||||
int fallback(int result) {
|
|
||||||
// If the buffer is full then the message is probably truncated.
|
|
||||||
return result == 0 && strlen(buffer_) == buffer_size_ - 1 ?
|
|
||||||
ERANGE : result;
|
|
||||||
}
|
|
||||||
|
|
||||||
#if !FMT_MSC_VER
|
|
||||||
// Fallback to strerror if strerror_r and strerror_s are not available.
|
|
||||||
int fallback(internal::null<>) {
|
|
||||||
errno = 0;
|
|
||||||
buffer_ = strerror(error_code_);
|
|
||||||
return errno;
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
public:
|
|
||||||
dispatcher(int err_code, char *&buf, std::size_t buf_size)
|
|
||||||
: error_code_(err_code), buffer_(buf), buffer_size_(buf_size) {}
|
|
||||||
|
|
||||||
int run() {
|
|
||||||
return handle(strerror_r(error_code_, buffer_, buffer_size_));
|
|
||||||
}
|
|
||||||
};
|
|
||||||
return dispatcher(error_code, buffer, buffer_size).run();
|
|
||||||
}
|
|
||||||
|
|
||||||
void format_error_code(internal::buffer &out, int error_code,
|
|
||||||
string_view message) FMT_NOEXCEPT {
|
|
||||||
// Report error code making sure that the output fits into
|
|
||||||
// inline_buffer_size to avoid dynamic memory allocation and potential
|
|
||||||
// bad_alloc.
|
|
||||||
out.resize(0);
|
|
||||||
static const char SEP[] = ": ";
|
|
||||||
static const char ERROR_STR[] = "error ";
|
|
||||||
// Subtract 2 to account for terminating null characters in SEP and ERROR_STR.
|
|
||||||
std::size_t error_code_size = sizeof(SEP) + sizeof(ERROR_STR) - 2;
|
|
||||||
typedef internal::int_traits<int>::main_type main_type;
|
|
||||||
main_type abs_value = static_cast<main_type>(error_code);
|
|
||||||
if (internal::is_negative(error_code)) {
|
|
||||||
abs_value = 0 - abs_value;
|
|
||||||
++error_code_size;
|
|
||||||
}
|
|
||||||
error_code_size += internal::to_unsigned(internal::count_digits(abs_value));
|
|
||||||
writer w(out);
|
|
||||||
if (message.size() <= inline_buffer_size - error_code_size) {
|
|
||||||
w.write(message);
|
|
||||||
w.write(SEP);
|
|
||||||
}
|
|
||||||
w.write(ERROR_STR);
|
|
||||||
w.write(error_code);
|
|
||||||
assert(out.size() <= inline_buffer_size);
|
|
||||||
}
|
|
||||||
|
|
||||||
void report_error(FormatFunc func, int error_code,
|
|
||||||
string_view message) FMT_NOEXCEPT {
|
|
||||||
memory_buffer full_message;
|
|
||||||
func(full_message, error_code, message);
|
|
||||||
// Use Writer::data instead of Writer::c_str to avoid potential memory
|
|
||||||
// allocation.
|
|
||||||
std::fwrite(full_message.data(), full_message.size(), 1, stderr);
|
|
||||||
std::fputc('\n', stderr);
|
|
||||||
}
|
|
||||||
} // namespace
|
|
||||||
|
|
||||||
FMT_FUNC size_t internal::count_code_points(basic_string_view<char8_t> s) {
|
|
||||||
const char8_t *data = s.data();
|
|
||||||
size_t num_code_points = 0;
|
|
||||||
for (size_t i = 0, size = s.size(); i != size; ++i) {
|
|
||||||
if ((data[i] & 0xc0) != 0x80)
|
|
||||||
++num_code_points;
|
|
||||||
}
|
|
||||||
return num_code_points;
|
|
||||||
}
|
|
||||||
|
|
||||||
#if !defined(FMT_STATIC_THOUSANDS_SEPARATOR)
|
|
||||||
namespace internal {
|
|
||||||
|
|
||||||
template <typename Locale>
|
|
||||||
locale_ref::locale_ref(const Locale &loc) : locale_(&loc) {
|
|
||||||
static_assert(std::is_same<Locale, std::locale>::value, "");
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename Locale>
|
|
||||||
Locale locale_ref::get() const {
|
|
||||||
static_assert(std::is_same<Locale, std::locale>::value, "");
|
|
||||||
return locale_ ? *static_cast<const std::locale*>(locale_) : std::locale();
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename Char>
|
|
||||||
FMT_FUNC Char thousands_sep_impl(locale_ref loc) {
|
|
||||||
return std::use_facet<std::numpunct<Char> >(
|
|
||||||
loc.get<std::locale>()).thousands_sep();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#else
|
|
||||||
template <typename Char>
|
|
||||||
FMT_FUNC Char internal::thousands_sep_impl(locale_ref) {
|
|
||||||
return FMT_STATIC_THOUSANDS_SEPARATOR;
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
FMT_FUNC void system_error::init(
|
|
||||||
int err_code, string_view format_str, format_args args) {
|
|
||||||
error_code_ = err_code;
|
|
||||||
memory_buffer buffer;
|
|
||||||
format_system_error(buffer, err_code, vformat(format_str, args));
|
|
||||||
std::runtime_error &base = *this;
|
|
||||||
base = std::runtime_error(to_string(buffer));
|
|
||||||
}
|
|
||||||
|
|
||||||
namespace internal {
|
|
||||||
template <typename T>
|
|
||||||
int char_traits<char>::format_float(
|
|
||||||
char *buf, std::size_t size, const char *format, int precision, T value) {
|
|
||||||
return precision < 0 ?
|
|
||||||
FMT_SNPRINTF(buf, size, format, value) :
|
|
||||||
FMT_SNPRINTF(buf, size, format, precision, value);
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename T>
|
|
||||||
int char_traits<wchar_t>::format_float(
|
|
||||||
wchar_t *buf, std::size_t size, const wchar_t *format, int precision,
|
|
||||||
T value) {
|
|
||||||
return precision < 0 ?
|
|
||||||
FMT_SWPRINTF(buf, size, format, value) :
|
|
||||||
FMT_SWPRINTF(buf, size, format, precision, value);
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename T>
|
|
||||||
const char basic_data<T>::DIGITS[] =
|
|
||||||
"0001020304050607080910111213141516171819"
|
|
||||||
"2021222324252627282930313233343536373839"
|
|
||||||
"4041424344454647484950515253545556575859"
|
|
||||||
"6061626364656667686970717273747576777879"
|
|
||||||
"8081828384858687888990919293949596979899";
|
|
||||||
|
|
||||||
#define FMT_POWERS_OF_10(factor) \
|
|
||||||
factor * 10, \
|
|
||||||
factor * 100, \
|
|
||||||
factor * 1000, \
|
|
||||||
factor * 10000, \
|
|
||||||
factor * 100000, \
|
|
||||||
factor * 1000000, \
|
|
||||||
factor * 10000000, \
|
|
||||||
factor * 100000000, \
|
|
||||||
factor * 1000000000
|
|
||||||
|
|
||||||
template <typename T>
|
|
||||||
const uint32_t basic_data<T>::POWERS_OF_10_32[] = {
|
|
||||||
1, FMT_POWERS_OF_10(1)
|
|
||||||
};
|
|
||||||
|
|
||||||
template <typename T>
|
|
||||||
const uint32_t basic_data<T>::ZERO_OR_POWERS_OF_10_32[] = {
|
|
||||||
0, FMT_POWERS_OF_10(1)
|
|
||||||
};
|
|
||||||
|
|
||||||
template <typename T>
|
|
||||||
const uint64_t basic_data<T>::ZERO_OR_POWERS_OF_10_64[] = {
|
|
||||||
0,
|
|
||||||
FMT_POWERS_OF_10(1),
|
|
||||||
FMT_POWERS_OF_10(1000000000ull),
|
|
||||||
10000000000000000000ull
|
|
||||||
};
|
|
||||||
|
|
||||||
// Normalized 64-bit significands of pow(10, k), for k = -348, -340, ..., 340.
|
|
||||||
// These are generated by support/compute-powers.py.
|
|
||||||
template <typename T>
|
|
||||||
const uint64_t basic_data<T>::POW10_SIGNIFICANDS[] = {
|
|
||||||
0xfa8fd5a0081c0288, 0xbaaee17fa23ebf76, 0x8b16fb203055ac76,
|
|
||||||
0xcf42894a5dce35ea, 0x9a6bb0aa55653b2d, 0xe61acf033d1a45df,
|
|
||||||
0xab70fe17c79ac6ca, 0xff77b1fcbebcdc4f, 0xbe5691ef416bd60c,
|
|
||||||
0x8dd01fad907ffc3c, 0xd3515c2831559a83, 0x9d71ac8fada6c9b5,
|
|
||||||
0xea9c227723ee8bcb, 0xaecc49914078536d, 0x823c12795db6ce57,
|
|
||||||
0xc21094364dfb5637, 0x9096ea6f3848984f, 0xd77485cb25823ac7,
|
|
||||||
0xa086cfcd97bf97f4, 0xef340a98172aace5, 0xb23867fb2a35b28e,
|
|
||||||
0x84c8d4dfd2c63f3b, 0xc5dd44271ad3cdba, 0x936b9fcebb25c996,
|
|
||||||
0xdbac6c247d62a584, 0xa3ab66580d5fdaf6, 0xf3e2f893dec3f126,
|
|
||||||
0xb5b5ada8aaff80b8, 0x87625f056c7c4a8b, 0xc9bcff6034c13053,
|
|
||||||
0x964e858c91ba2655, 0xdff9772470297ebd, 0xa6dfbd9fb8e5b88f,
|
|
||||||
0xf8a95fcf88747d94, 0xb94470938fa89bcf, 0x8a08f0f8bf0f156b,
|
|
||||||
0xcdb02555653131b6, 0x993fe2c6d07b7fac, 0xe45c10c42a2b3b06,
|
|
||||||
0xaa242499697392d3, 0xfd87b5f28300ca0e, 0xbce5086492111aeb,
|
|
||||||
0x8cbccc096f5088cc, 0xd1b71758e219652c, 0x9c40000000000000,
|
|
||||||
0xe8d4a51000000000, 0xad78ebc5ac620000, 0x813f3978f8940984,
|
|
||||||
0xc097ce7bc90715b3, 0x8f7e32ce7bea5c70, 0xd5d238a4abe98068,
|
|
||||||
0x9f4f2726179a2245, 0xed63a231d4c4fb27, 0xb0de65388cc8ada8,
|
|
||||||
0x83c7088e1aab65db, 0xc45d1df942711d9a, 0x924d692ca61be758,
|
|
||||||
0xda01ee641a708dea, 0xa26da3999aef774a, 0xf209787bb47d6b85,
|
|
||||||
0xb454e4a179dd1877, 0x865b86925b9bc5c2, 0xc83553c5c8965d3d,
|
|
||||||
0x952ab45cfa97a0b3, 0xde469fbd99a05fe3, 0xa59bc234db398c25,
|
|
||||||
0xf6c69a72a3989f5c, 0xb7dcbf5354e9bece, 0x88fcf317f22241e2,
|
|
||||||
0xcc20ce9bd35c78a5, 0x98165af37b2153df, 0xe2a0b5dc971f303a,
|
|
||||||
0xa8d9d1535ce3b396, 0xfb9b7cd9a4a7443c, 0xbb764c4ca7a44410,
|
|
||||||
0x8bab8eefb6409c1a, 0xd01fef10a657842c, 0x9b10a4e5e9913129,
|
|
||||||
0xe7109bfba19c0c9d, 0xac2820d9623bf429, 0x80444b5e7aa7cf85,
|
|
||||||
0xbf21e44003acdd2d, 0x8e679c2f5e44ff8f, 0xd433179d9c8cb841,
|
|
||||||
0x9e19db92b4e31ba9, 0xeb96bf6ebadf77d9, 0xaf87023b9bf0ee6b,
|
|
||||||
};
|
|
||||||
|
|
||||||
// Binary exponents of pow(10, k), for k = -348, -340, ..., 340, corresponding
|
|
||||||
// to significands above.
|
|
||||||
template <typename T>
|
|
||||||
const int16_t basic_data<T>::POW10_EXPONENTS[] = {
|
|
||||||
-1220, -1193, -1166, -1140, -1113, -1087, -1060, -1034, -1007, -980, -954,
|
|
||||||
-927, -901, -874, -847, -821, -794, -768, -741, -715, -688, -661,
|
|
||||||
-635, -608, -582, -555, -529, -502, -475, -449, -422, -396, -369,
|
|
||||||
-343, -316, -289, -263, -236, -210, -183, -157, -130, -103, -77,
|
|
||||||
-50, -24, 3, 30, 56, 83, 109, 136, 162, 189, 216,
|
|
||||||
242, 269, 295, 322, 348, 375, 402, 428, 455, 481, 508,
|
|
||||||
534, 561, 588, 614, 641, 667, 694, 720, 747, 774, 800,
|
|
||||||
827, 853, 880, 907, 933, 960, 986, 1013, 1039, 1066
|
|
||||||
};
|
|
||||||
|
|
||||||
template <typename T> const char basic_data<T>::FOREGROUND_COLOR[] = "\x1b[38;2;";
|
|
||||||
template <typename T> const char basic_data<T>::BACKGROUND_COLOR[] = "\x1b[48;2;";
|
|
||||||
template <typename T> const char basic_data<T>::RESET_COLOR[] = "\x1b[0m";
|
|
||||||
template <typename T> const wchar_t basic_data<T>::WRESET_COLOR[] = L"\x1b[0m";
|
|
||||||
|
|
||||||
// A handmade floating-point number f * pow(2, e).
|
|
||||||
class fp {
|
|
||||||
private:
|
|
||||||
typedef uint64_t significand_type;
|
|
||||||
|
|
||||||
// All sizes are in bits.
|
|
||||||
static FMT_CONSTEXPR_DECL const int char_size =
|
|
||||||
std::numeric_limits<unsigned char>::digits;
|
|
||||||
// Subtract 1 to account for an implicit most significant bit in the
|
|
||||||
// normalized form.
|
|
||||||
static FMT_CONSTEXPR_DECL const int double_significand_size =
|
|
||||||
std::numeric_limits<double>::digits - 1;
|
|
||||||
static FMT_CONSTEXPR_DECL const uint64_t implicit_bit =
|
|
||||||
1ull << double_significand_size;
|
|
||||||
|
|
||||||
public:
|
|
||||||
significand_type f;
|
|
||||||
int e;
|
|
||||||
|
|
||||||
static FMT_CONSTEXPR_DECL const int significand_size =
|
|
||||||
sizeof(significand_type) * char_size;
|
|
||||||
|
|
||||||
fp(): f(0), e(0) {}
|
|
||||||
fp(uint64_t f_val, int e_val): f(f_val), e(e_val) {}
|
|
||||||
|
|
||||||
// Constructs fp from an IEEE754 double. It is a template to prevent compile
|
|
||||||
// errors on platforms where double is not IEEE754.
|
|
||||||
template <typename Double>
|
|
||||||
explicit fp(Double d) {
|
|
||||||
// Assume double is in the format [sign][exponent][significand].
|
|
||||||
typedef std::numeric_limits<Double> limits;
|
|
||||||
const int double_size = static_cast<int>(sizeof(Double) * char_size);
|
|
||||||
const int exponent_size =
|
|
||||||
double_size - double_significand_size - 1; // -1 for sign
|
|
||||||
const uint64_t significand_mask = implicit_bit - 1;
|
|
||||||
const uint64_t exponent_mask = (~0ull >> 1) & ~significand_mask;
|
|
||||||
const int exponent_bias = (1 << exponent_size) - limits::max_exponent - 1;
|
|
||||||
auto u = bit_cast<uint64_t>(d);
|
|
||||||
auto biased_e = (u & exponent_mask) >> double_significand_size;
|
|
||||||
f = u & significand_mask;
|
|
||||||
if (biased_e != 0)
|
|
||||||
f += implicit_bit;
|
|
||||||
else
|
|
||||||
biased_e = 1; // Subnormals use biased exponent 1 (min exponent).
|
|
||||||
e = static_cast<int>(biased_e - exponent_bias - double_significand_size);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Normalizes the value converted from double and multiplied by (1 << SHIFT).
|
|
||||||
template <int SHIFT = 0>
|
|
||||||
void normalize() {
|
|
||||||
// Handle subnormals.
|
|
||||||
auto shifted_implicit_bit = implicit_bit << SHIFT;
|
|
||||||
while ((f & shifted_implicit_bit) == 0) {
|
|
||||||
f <<= 1;
|
|
||||||
--e;
|
|
||||||
}
|
|
||||||
// Subtract 1 to account for hidden bit.
|
|
||||||
auto offset = significand_size - double_significand_size - SHIFT - 1;
|
|
||||||
f <<= offset;
|
|
||||||
e -= offset;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Compute lower and upper boundaries (m^- and m^+ in the Grisu paper), where
|
|
||||||
// a boundary is a value half way between the number and its predecessor
|
|
||||||
// (lower) or successor (upper). The upper boundary is normalized and lower
|
|
||||||
// has the same exponent but may be not normalized.
|
|
||||||
void compute_boundaries(fp &lower, fp &upper) const {
|
|
||||||
lower = f == implicit_bit ?
|
|
||||||
fp((f << 2) - 1, e - 2) : fp((f << 1) - 1, e - 1);
|
|
||||||
upper = fp((f << 1) + 1, e - 1);
|
|
||||||
upper.normalize<1>(); // 1 is to account for the exponent shift above.
|
|
||||||
lower.f <<= lower.e - upper.e;
|
|
||||||
lower.e = upper.e;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
// Returns an fp number representing x - y. Result may not be normalized.
|
|
||||||
inline fp operator-(fp x, fp y) {
|
|
||||||
FMT_ASSERT(x.f >= y.f && x.e == y.e, "invalid operands");
|
|
||||||
return fp(x.f - y.f, x.e);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Computes an fp number r with r.f = x.f * y.f / pow(2, 64) rounded to nearest
|
|
||||||
// with half-up tie breaking, r.e = x.e + y.e + 64. Result may not be normalized.
|
|
||||||
FMT_API fp operator*(fp x, fp y);
|
|
||||||
|
|
||||||
// Returns cached power (of 10) c_k = c_k.f * pow(2, c_k.e) such that its
|
|
||||||
// (binary) exponent satisfies min_exponent <= c_k.e <= min_exponent + 3.
|
|
||||||
FMT_API fp get_cached_power(int min_exponent, int &pow10_exponent);
|
|
||||||
|
|
||||||
FMT_FUNC fp operator*(fp x, fp y) {
|
|
||||||
// Multiply 32-bit parts of significands.
|
|
||||||
uint64_t mask = (1ULL << 32) - 1;
|
|
||||||
uint64_t a = x.f >> 32, b = x.f & mask;
|
|
||||||
uint64_t c = y.f >> 32, d = y.f & mask;
|
|
||||||
uint64_t ac = a * c, bc = b * c, ad = a * d, bd = b * d;
|
|
||||||
// Compute mid 64-bit of result and round.
|
|
||||||
uint64_t mid = (bd >> 32) + (ad & mask) + (bc & mask) + (1U << 31);
|
|
||||||
return fp(ac + (ad >> 32) + (bc >> 32) + (mid >> 32), x.e + y.e + 64);
|
|
||||||
}
|
|
||||||
|
|
||||||
FMT_FUNC fp get_cached_power(int min_exponent, int &pow10_exponent) {
|
|
||||||
const double one_over_log2_10 = 0.30102999566398114; // 1 / log2(10)
|
|
||||||
int index = static_cast<int>(std::ceil(
|
|
||||||
(min_exponent + fp::significand_size - 1) * one_over_log2_10));
|
|
||||||
// Decimal exponent of the first (smallest) cached power of 10.
|
|
||||||
const int first_dec_exp = -348;
|
|
||||||
// Difference between 2 consecutive decimal exponents in cached powers of 10.
|
|
||||||
const int dec_exp_step = 8;
|
|
||||||
index = (index - first_dec_exp - 1) / dec_exp_step + 1;
|
|
||||||
pow10_exponent = first_dec_exp + index * dec_exp_step;
|
|
||||||
return fp(data::POW10_SIGNIFICANDS[index], data::POW10_EXPONENTS[index]);
|
|
||||||
}
|
|
||||||
|
|
||||||
FMT_FUNC bool grisu2_round(
|
|
||||||
char *buf, int &size, int max_digits, uint64_t delta,
|
|
||||||
uint64_t remainder, uint64_t exp, uint64_t diff, int &exp10) {
|
|
||||||
while (remainder < diff && delta - remainder >= exp &&
|
|
||||||
(remainder + exp < diff || diff - remainder > remainder + exp - diff)) {
|
|
||||||
--buf[size - 1];
|
|
||||||
remainder += exp;
|
|
||||||
}
|
|
||||||
if (size > max_digits) {
|
|
||||||
--size;
|
|
||||||
++exp10;
|
|
||||||
if (buf[size] >= '5')
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Generates output using Grisu2 digit-gen algorithm.
|
|
||||||
FMT_FUNC bool grisu2_gen_digits(
|
|
||||||
char *buf, int &size, uint32_t hi, uint64_t lo, int &exp,
|
|
||||||
uint64_t delta, const fp &one, const fp &diff, int max_digits) {
|
|
||||||
// Generate digits for the most significant part (hi).
|
|
||||||
while (exp > 0) {
|
|
||||||
uint32_t digit = 0;
|
|
||||||
// This optimization by miloyip reduces the number of integer divisions by
|
|
||||||
// one per iteration.
|
|
||||||
switch (exp) {
|
|
||||||
case 10: digit = hi / 1000000000; hi %= 1000000000; break;
|
|
||||||
case 9: digit = hi / 100000000; hi %= 100000000; break;
|
|
||||||
case 8: digit = hi / 10000000; hi %= 10000000; break;
|
|
||||||
case 7: digit = hi / 1000000; hi %= 1000000; break;
|
|
||||||
case 6: digit = hi / 100000; hi %= 100000; break;
|
|
||||||
case 5: digit = hi / 10000; hi %= 10000; break;
|
|
||||||
case 4: digit = hi / 1000; hi %= 1000; break;
|
|
||||||
case 3: digit = hi / 100; hi %= 100; break;
|
|
||||||
case 2: digit = hi / 10; hi %= 10; break;
|
|
||||||
case 1: digit = hi; hi = 0; break;
|
|
||||||
default:
|
|
||||||
FMT_ASSERT(false, "invalid number of digits");
|
|
||||||
}
|
|
||||||
if (digit != 0 || size != 0)
|
|
||||||
buf[size++] = static_cast<char>('0' + digit);
|
|
||||||
--exp;
|
|
||||||
uint64_t remainder = (static_cast<uint64_t>(hi) << -one.e) + lo;
|
|
||||||
if (remainder <= delta || size > max_digits) {
|
|
||||||
return grisu2_round(
|
|
||||||
buf, size, max_digits, delta, remainder,
|
|
||||||
static_cast<uint64_t>(data::POWERS_OF_10_32[exp]) << -one.e,
|
|
||||||
diff.f, exp);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// Generate digits for the least significant part (lo).
|
|
||||||
for (;;) {
|
|
||||||
lo *= 10;
|
|
||||||
delta *= 10;
|
|
||||||
char digit = static_cast<char>(lo >> -one.e);
|
|
||||||
if (digit != 0 || size != 0)
|
|
||||||
buf[size++] = static_cast<char>('0' + digit);
|
|
||||||
lo &= one.f - 1;
|
|
||||||
--exp;
|
|
||||||
if (lo < delta || size > max_digits) {
|
|
||||||
return grisu2_round(buf, size, max_digits, delta, lo, one.f,
|
|
||||||
diff.f * data::POWERS_OF_10_32[-exp], exp);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#if FMT_CLANG_VERSION
|
|
||||||
# define FMT_FALLTHROUGH [[clang::fallthrough]];
|
|
||||||
#elif FMT_GCC_VERSION >= 700
|
|
||||||
# define FMT_FALLTHROUGH [[gnu::fallthrough]];
|
|
||||||
#else
|
|
||||||
# define FMT_FALLTHROUGH
|
|
||||||
#endif
|
|
||||||
|
|
||||||
struct gen_digits_params {
|
|
||||||
int num_digits;
|
|
||||||
bool fixed;
|
|
||||||
bool upper;
|
|
||||||
bool trailing_zeros;
|
|
||||||
};
|
|
||||||
|
|
||||||
struct prettify_handler {
|
|
||||||
char *data;
|
|
||||||
ptrdiff_t size;
|
|
||||||
buffer &buf;
|
|
||||||
|
|
||||||
explicit prettify_handler(buffer &b, ptrdiff_t n)
|
|
||||||
: data(b.data()), size(n), buf(b) {}
|
|
||||||
~prettify_handler() {
|
|
||||||
assert(buf.size() >= to_unsigned(size));
|
|
||||||
buf.resize(to_unsigned(size));
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename F>
|
|
||||||
void insert(ptrdiff_t pos, ptrdiff_t n, F f) {
|
|
||||||
std::memmove(data + pos + n, data + pos, to_unsigned(size - pos));
|
|
||||||
f(data + pos);
|
|
||||||
size += n;
|
|
||||||
}
|
|
||||||
|
|
||||||
void insert(ptrdiff_t pos, char c) {
|
|
||||||
std::memmove(data + pos + 1, data + pos, to_unsigned(size - pos));
|
|
||||||
data[pos] = c;
|
|
||||||
++size;
|
|
||||||
}
|
|
||||||
|
|
||||||
void append(ptrdiff_t n, char c) {
|
|
||||||
std::uninitialized_fill_n(data + size, n, c);
|
|
||||||
size += n;
|
|
||||||
}
|
|
||||||
|
|
||||||
void append(char c) { data[size++] = c; }
|
|
||||||
|
|
||||||
void remove_trailing(char c) {
|
|
||||||
while (data[size - 1] == c) --size;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
// Writes the exponent exp in the form "[+-]d{2,3}" to buffer.
|
|
||||||
template <typename Handler>
|
|
||||||
FMT_FUNC void write_exponent(int exp, Handler &&h) {
|
|
||||||
FMT_ASSERT(-1000 < exp && exp < 1000, "exponent out of range");
|
|
||||||
if (exp < 0) {
|
|
||||||
h.append('-');
|
|
||||||
exp = -exp;
|
|
||||||
} else {
|
|
||||||
h.append('+');
|
|
||||||
}
|
|
||||||
if (exp >= 100) {
|
|
||||||
h.append(static_cast<char>('0' + exp / 100));
|
|
||||||
exp %= 100;
|
|
||||||
const char *d = data::DIGITS + exp * 2;
|
|
||||||
h.append(d[0]);
|
|
||||||
h.append(d[1]);
|
|
||||||
} else {
|
|
||||||
const char *d = data::DIGITS + exp * 2;
|
|
||||||
h.append(d[0]);
|
|
||||||
h.append(d[1]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
struct fill {
|
|
||||||
size_t n;
|
|
||||||
void operator()(char *buf) const {
|
|
||||||
buf[0] = '0';
|
|
||||||
buf[1] = '.';
|
|
||||||
std::uninitialized_fill_n(buf + 2, n, '0');
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
// The number is given as v = f * pow(10, exp), where f has size digits.
|
|
||||||
template <typename Handler>
|
|
||||||
FMT_FUNC void grisu2_prettify(const gen_digits_params ¶ms,
|
|
||||||
int size, int exp, Handler &&handler) {
|
|
||||||
if (!params.fixed) {
|
|
||||||
// Insert a decimal point after the first digit and add an exponent.
|
|
||||||
handler.insert(1, '.');
|
|
||||||
exp += size - 1;
|
|
||||||
if (size < params.num_digits)
|
|
||||||
handler.append(params.num_digits - size, '0');
|
|
||||||
handler.append(params.upper ? 'E' : 'e');
|
|
||||||
write_exponent(exp, handler);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
// pow(10, full_exp - 1) <= v <= pow(10, full_exp).
|
|
||||||
int full_exp = size + exp;
|
|
||||||
const int exp_threshold = 21;
|
|
||||||
if (size <= full_exp && full_exp <= exp_threshold) {
|
|
||||||
// 1234e7 -> 12340000000[.0+]
|
|
||||||
handler.append(full_exp - size, '0');
|
|
||||||
int num_zeros = params.num_digits - full_exp;
|
|
||||||
if (num_zeros > 0 && params.trailing_zeros) {
|
|
||||||
handler.append('.');
|
|
||||||
handler.append(num_zeros, '0');
|
|
||||||
}
|
|
||||||
} else if (full_exp > 0) {
|
|
||||||
// 1234e-2 -> 12.34[0+]
|
|
||||||
handler.insert(full_exp, '.');
|
|
||||||
if (!params.trailing_zeros) {
|
|
||||||
// Remove trailing zeros.
|
|
||||||
handler.remove_trailing('0');
|
|
||||||
} else if (params.num_digits > size) {
|
|
||||||
// Add trailing zeros.
|
|
||||||
ptrdiff_t num_zeros = params.num_digits - size;
|
|
||||||
handler.append(num_zeros, '0');
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
// 1234e-6 -> 0.001234
|
|
||||||
handler.insert(0, 2 - full_exp, fill{to_unsigned(-full_exp)});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
struct char_counter {
|
|
||||||
ptrdiff_t size;
|
|
||||||
|
|
||||||
template <typename F>
|
|
||||||
void insert(ptrdiff_t, ptrdiff_t n, F) { size += n; }
|
|
||||||
void insert(ptrdiff_t, char) { ++size; }
|
|
||||||
void append(ptrdiff_t n, char) { size += n; }
|
|
||||||
void append(char) { ++size; }
|
|
||||||
void remove_trailing(char) {}
|
|
||||||
};
|
|
||||||
|
|
||||||
// Converts format specifiers into parameters for digit generation and computes
|
|
||||||
// output buffer size for a number in the range [pow(10, exp - 1), pow(10, exp)
|
|
||||||
// or 0 if exp == 1.
|
|
||||||
FMT_FUNC gen_digits_params process_specs(const core_format_specs &specs,
|
|
||||||
int exp, buffer &buf) {
|
|
||||||
auto params = gen_digits_params();
|
|
||||||
int num_digits = specs.precision >= 0 ? specs.precision : 6;
|
|
||||||
switch (specs.type) {
|
|
||||||
case 'G':
|
|
||||||
params.upper = true;
|
|
||||||
FMT_FALLTHROUGH
|
|
||||||
case '\0': case 'g':
|
|
||||||
params.trailing_zeros = (specs.flags & HASH_FLAG) != 0;
|
|
||||||
if (-4 <= exp && exp < num_digits + 1) {
|
|
||||||
params.fixed = true;
|
|
||||||
if (!specs.type && params.trailing_zeros && exp >= 0)
|
|
||||||
num_digits = exp + 1;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case 'F':
|
|
||||||
params.upper = true;
|
|
||||||
FMT_FALLTHROUGH
|
|
||||||
case 'f': {
|
|
||||||
params.fixed = true;
|
|
||||||
params.trailing_zeros = true;
|
|
||||||
int adjusted_min_digits = num_digits + exp;
|
|
||||||
if (adjusted_min_digits > 0)
|
|
||||||
num_digits = adjusted_min_digits;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case 'E':
|
|
||||||
params.upper = true;
|
|
||||||
FMT_FALLTHROUGH
|
|
||||||
case 'e':
|
|
||||||
++num_digits;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
params.num_digits = num_digits;
|
|
||||||
char_counter counter{num_digits};
|
|
||||||
grisu2_prettify(params, params.num_digits, exp - num_digits, counter);
|
|
||||||
buf.resize(to_unsigned(counter.size));
|
|
||||||
return params;
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename Double>
|
|
||||||
FMT_FUNC typename std::enable_if<sizeof(Double) == sizeof(uint64_t), bool>::type
|
|
||||||
grisu2_format(Double value, buffer &buf, core_format_specs specs) {
|
|
||||||
FMT_ASSERT(value >= 0, "value is negative");
|
|
||||||
if (value == 0) {
|
|
||||||
gen_digits_params params = process_specs(specs, 1, buf);
|
|
||||||
const size_t size = 1;
|
|
||||||
buf[0] = '0';
|
|
||||||
grisu2_prettify(params, size, 0, prettify_handler(buf, size));
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
fp fp_value(value);
|
|
||||||
fp lower, upper; // w^- and w^+ in the Grisu paper.
|
|
||||||
fp_value.compute_boundaries(lower, upper);
|
|
||||||
|
|
||||||
// Find a cached power of 10 close to 1 / upper and use it to scale upper.
|
|
||||||
const int min_exp = -60; // alpha in Grisu.
|
|
||||||
int cached_exp = 0; // K in Grisu.
|
|
||||||
auto cached_pow = get_cached_power( // \tilde{c}_{-k} in Grisu.
|
|
||||||
min_exp - (upper.e + fp::significand_size), cached_exp);
|
|
||||||
cached_exp = -cached_exp;
|
|
||||||
upper = upper * cached_pow; // \tilde{M}^+ in Grisu.
|
|
||||||
--upper.f; // \tilde{M}^+ - 1 ulp -> M^+_{\downarrow}.
|
|
||||||
fp one(1ull << -upper.e, upper.e);
|
|
||||||
// hi (p1 in Grisu) contains the most significant digits of scaled_upper.
|
|
||||||
// hi = floor(upper / one).
|
|
||||||
uint32_t hi = static_cast<uint32_t>(upper.f >> -one.e);
|
|
||||||
int exp = count_digits(hi); // kappa in Grisu.
|
|
||||||
gen_digits_params params = process_specs(specs, cached_exp + exp, buf);
|
|
||||||
fp_value.normalize();
|
|
||||||
fp scaled_value = fp_value * cached_pow;
|
|
||||||
lower = lower * cached_pow; // \tilde{M}^- in Grisu.
|
|
||||||
++lower.f; // \tilde{M}^- + 1 ulp -> M^-_{\uparrow}.
|
|
||||||
uint64_t delta = upper.f - lower.f;
|
|
||||||
fp diff = upper - scaled_value; // wp_w in Grisu.
|
|
||||||
// lo (p2 in Grisu) contains the least significants digits of scaled_upper.
|
|
||||||
// lo = supper % one.
|
|
||||||
uint64_t lo = upper.f & (one.f - 1);
|
|
||||||
int size = 0;
|
|
||||||
if (!grisu2_gen_digits(buf.data(), size, hi, lo, exp, delta, one, diff,
|
|
||||||
params.num_digits)) {
|
|
||||||
buf.clear();
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
grisu2_prettify(params, size, cached_exp + exp, prettify_handler(buf, size));
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename Double>
|
|
||||||
void sprintf_format(Double value, internal::buffer &buf,
|
|
||||||
core_format_specs spec) {
|
|
||||||
// Buffer capacity must be non-zero, otherwise MSVC's vsnprintf_s will fail.
|
|
||||||
FMT_ASSERT(buf.capacity() != 0, "empty buffer");
|
|
||||||
|
|
||||||
// Build format string.
|
|
||||||
enum { MAX_FORMAT_SIZE = 10}; // longest format: %#-*.*Lg
|
|
||||||
char format[MAX_FORMAT_SIZE];
|
|
||||||
char *format_ptr = format;
|
|
||||||
*format_ptr++ = '%';
|
|
||||||
if (spec.has(HASH_FLAG))
|
|
||||||
*format_ptr++ = '#';
|
|
||||||
if (spec.precision >= 0) {
|
|
||||||
*format_ptr++ = '.';
|
|
||||||
*format_ptr++ = '*';
|
|
||||||
}
|
|
||||||
if (std::is_same<Double, long double>::value)
|
|
||||||
*format_ptr++ = 'L';
|
|
||||||
*format_ptr++ = spec.type;
|
|
||||||
*format_ptr = '\0';
|
|
||||||
|
|
||||||
// Format using snprintf.
|
|
||||||
char *start = FMT_NULL;
|
|
||||||
for (;;) {
|
|
||||||
std::size_t buffer_size = buf.capacity();
|
|
||||||
start = &buf[0];
|
|
||||||
int result = internal::char_traits<char>::format_float(
|
|
||||||
start, buffer_size, format, spec.precision, value);
|
|
||||||
if (result >= 0) {
|
|
||||||
unsigned n = internal::to_unsigned(result);
|
|
||||||
if (n < buf.capacity()) {
|
|
||||||
buf.resize(n);
|
|
||||||
break; // The buffer is large enough - continue with formatting.
|
|
||||||
}
|
|
||||||
buf.reserve(n + 1);
|
|
||||||
} else {
|
|
||||||
// If result is negative we ask to increase the capacity by at least 1,
|
|
||||||
// but as std::vector, the buffer grows exponentially.
|
|
||||||
buf.reserve(buf.capacity() + 1);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} // namespace internal
|
|
||||||
|
|
||||||
#if FMT_USE_WINDOWS_H
|
|
||||||
|
|
||||||
FMT_FUNC internal::utf8_to_utf16::utf8_to_utf16(string_view s) {
|
|
||||||
static const char ERROR_MSG[] = "cannot convert string from UTF-8 to UTF-16";
|
|
||||||
if (s.size() > INT_MAX)
|
|
||||||
FMT_THROW(windows_error(ERROR_INVALID_PARAMETER, ERROR_MSG));
|
|
||||||
int s_size = static_cast<int>(s.size());
|
|
||||||
if (s_size == 0) {
|
|
||||||
// MultiByteToWideChar does not support zero length, handle separately.
|
|
||||||
buffer_.resize(1);
|
|
||||||
buffer_[0] = 0;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
int length = MultiByteToWideChar(
|
|
||||||
CP_UTF8, MB_ERR_INVALID_CHARS, s.data(), s_size, FMT_NULL, 0);
|
|
||||||
if (length == 0)
|
|
||||||
FMT_THROW(windows_error(GetLastError(), ERROR_MSG));
|
|
||||||
buffer_.resize(length + 1);
|
|
||||||
length = MultiByteToWideChar(
|
|
||||||
CP_UTF8, MB_ERR_INVALID_CHARS, s.data(), s_size, &buffer_[0], length);
|
|
||||||
if (length == 0)
|
|
||||||
FMT_THROW(windows_error(GetLastError(), ERROR_MSG));
|
|
||||||
buffer_[length] = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
FMT_FUNC internal::utf16_to_utf8::utf16_to_utf8(wstring_view s) {
|
|
||||||
if (int error_code = convert(s)) {
|
|
||||||
FMT_THROW(windows_error(error_code,
|
|
||||||
"cannot convert string from UTF-16 to UTF-8"));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
FMT_FUNC int internal::utf16_to_utf8::convert(wstring_view s) {
|
|
||||||
if (s.size() > INT_MAX)
|
|
||||||
return ERROR_INVALID_PARAMETER;
|
|
||||||
int s_size = static_cast<int>(s.size());
|
|
||||||
if (s_size == 0) {
|
|
||||||
// WideCharToMultiByte does not support zero length, handle separately.
|
|
||||||
buffer_.resize(1);
|
|
||||||
buffer_[0] = 0;
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
int length = WideCharToMultiByte(
|
|
||||||
CP_UTF8, 0, s.data(), s_size, FMT_NULL, 0, FMT_NULL, FMT_NULL);
|
|
||||||
if (length == 0)
|
|
||||||
return GetLastError();
|
|
||||||
buffer_.resize(length + 1);
|
|
||||||
length = WideCharToMultiByte(
|
|
||||||
CP_UTF8, 0, s.data(), s_size, &buffer_[0], length, FMT_NULL, FMT_NULL);
|
|
||||||
if (length == 0)
|
|
||||||
return GetLastError();
|
|
||||||
buffer_[length] = 0;
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
FMT_FUNC void windows_error::init(
|
|
||||||
int err_code, string_view format_str, format_args args) {
|
|
||||||
error_code_ = err_code;
|
|
||||||
memory_buffer buffer;
|
|
||||||
internal::format_windows_error(buffer, err_code, vformat(format_str, args));
|
|
||||||
std::runtime_error &base = *this;
|
|
||||||
base = std::runtime_error(to_string(buffer));
|
|
||||||
}
|
|
||||||
|
|
||||||
FMT_FUNC void internal::format_windows_error(
|
|
||||||
internal::buffer &out, int error_code, string_view message) FMT_NOEXCEPT {
|
|
||||||
FMT_TRY {
|
|
||||||
wmemory_buffer buf;
|
|
||||||
buf.resize(inline_buffer_size);
|
|
||||||
for (;;) {
|
|
||||||
wchar_t *system_message = &buf[0];
|
|
||||||
int result = FormatMessageW(
|
|
||||||
FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_IGNORE_INSERTS,
|
|
||||||
FMT_NULL, error_code, MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT),
|
|
||||||
system_message, static_cast<uint32_t>(buf.size()), FMT_NULL);
|
|
||||||
if (result != 0) {
|
|
||||||
utf16_to_utf8 utf8_message;
|
|
||||||
if (utf8_message.convert(system_message) == ERROR_SUCCESS) {
|
|
||||||
writer w(out);
|
|
||||||
w.write(message);
|
|
||||||
w.write(": ");
|
|
||||||
w.write(utf8_message);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
if (GetLastError() != ERROR_INSUFFICIENT_BUFFER)
|
|
||||||
break; // Can't get error message, report error code instead.
|
|
||||||
buf.resize(buf.size() * 2);
|
|
||||||
}
|
|
||||||
} FMT_CATCH(...) {}
|
|
||||||
format_error_code(out, error_code, message);
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif // FMT_USE_WINDOWS_H
|
|
||||||
|
|
||||||
FMT_FUNC void format_system_error(
|
|
||||||
internal::buffer &out, int error_code, string_view message) FMT_NOEXCEPT {
|
|
||||||
FMT_TRY {
|
|
||||||
memory_buffer buf;
|
|
||||||
buf.resize(inline_buffer_size);
|
|
||||||
for (;;) {
|
|
||||||
char *system_message = &buf[0];
|
|
||||||
int result = safe_strerror(error_code, system_message, buf.size());
|
|
||||||
if (result == 0) {
|
|
||||||
writer w(out);
|
|
||||||
w.write(message);
|
|
||||||
w.write(": ");
|
|
||||||
w.write(system_message);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (result != ERANGE)
|
|
||||||
break; // Can't get error message, report error code instead.
|
|
||||||
buf.resize(buf.size() * 2);
|
|
||||||
}
|
|
||||||
} FMT_CATCH(...) {}
|
|
||||||
format_error_code(out, error_code, message);
|
|
||||||
}
|
|
||||||
|
|
||||||
FMT_FUNC void internal::error_handler::on_error(const char *message) {
|
|
||||||
FMT_THROW(format_error(message));
|
|
||||||
}
|
|
||||||
|
|
||||||
FMT_FUNC void report_system_error(
|
|
||||||
int error_code, fmt::string_view message) FMT_NOEXCEPT {
|
|
||||||
report_error(format_system_error, error_code, message);
|
|
||||||
}
|
|
||||||
|
|
||||||
#if FMT_USE_WINDOWS_H
|
|
||||||
FMT_FUNC void report_windows_error(
|
|
||||||
int error_code, fmt::string_view message) FMT_NOEXCEPT {
|
|
||||||
report_error(internal::format_windows_error, error_code, message);
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
FMT_FUNC void vprint(std::FILE *f, string_view format_str, format_args args) {
|
|
||||||
memory_buffer buffer;
|
|
||||||
internal::vformat_to(buffer, format_str,
|
|
||||||
basic_format_args<buffer_context<char>::type>(args));
|
|
||||||
std::fwrite(buffer.data(), 1, buffer.size(), f);
|
|
||||||
}
|
|
||||||
|
|
||||||
FMT_FUNC void vprint(std::FILE *f, wstring_view format_str, wformat_args args) {
|
|
||||||
wmemory_buffer buffer;
|
|
||||||
internal::vformat_to(buffer, format_str, args);
|
|
||||||
std::fwrite(buffer.data(), sizeof(wchar_t), buffer.size(), f);
|
|
||||||
}
|
|
||||||
|
|
||||||
FMT_FUNC void vprint(string_view format_str, format_args args) {
|
|
||||||
vprint(stdout, format_str, args);
|
|
||||||
}
|
|
||||||
|
|
||||||
FMT_FUNC void vprint(wstring_view format_str, wformat_args args) {
|
|
||||||
vprint(stdout, format_str, args);
|
|
||||||
}
|
|
||||||
|
|
||||||
FMT_END_NAMESPACE
|
|
||||||
|
|
||||||
#ifdef _MSC_VER
|
|
||||||
# pragma warning(pop)
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#endif // FMT_FORMAT_INL_H_
|
|
||||||
3555
deps/spdlog/fmt/bundled/format.h
vendored
3555
deps/spdlog/fmt/bundled/format.h
vendored
File diff suppressed because it is too large
Load Diff
77
deps/spdlog/fmt/bundled/locale.h
vendored
77
deps/spdlog/fmt/bundled/locale.h
vendored
@ -1,77 +0,0 @@
|
|||||||
// Formatting library for C++ - std::locale support
|
|
||||||
//
|
|
||||||
// Copyright (c) 2012 - present, Victor Zverovich
|
|
||||||
// All rights reserved.
|
|
||||||
//
|
|
||||||
// For the license information refer to format.h.
|
|
||||||
|
|
||||||
#ifndef FMT_LOCALE_H_
|
|
||||||
#define FMT_LOCALE_H_
|
|
||||||
|
|
||||||
#include "format.h"
|
|
||||||
#include <locale>
|
|
||||||
|
|
||||||
FMT_BEGIN_NAMESPACE
|
|
||||||
|
|
||||||
namespace internal {
|
|
||||||
template <typename Char>
|
|
||||||
typename buffer_context<Char>::type::iterator vformat_to(
|
|
||||||
const std::locale &loc, basic_buffer<Char> &buf,
|
|
||||||
basic_string_view<Char> format_str,
|
|
||||||
basic_format_args<typename buffer_context<Char>::type> args) {
|
|
||||||
typedef back_insert_range<basic_buffer<Char> > range;
|
|
||||||
return vformat_to<arg_formatter<range>>(
|
|
||||||
buf, to_string_view(format_str), args, internal::locale_ref(loc));
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename Char>
|
|
||||||
std::basic_string<Char> vformat(
|
|
||||||
const std::locale &loc, basic_string_view<Char> format_str,
|
|
||||||
basic_format_args<typename buffer_context<Char>::type> args) {
|
|
||||||
basic_memory_buffer<Char> buffer;
|
|
||||||
internal::vformat_to(loc, buffer, format_str, args);
|
|
||||||
return fmt::to_string(buffer);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename S, typename Char = FMT_CHAR(S)>
|
|
||||||
inline std::basic_string<Char> vformat(
|
|
||||||
const std::locale &loc, const S &format_str,
|
|
||||||
basic_format_args<typename buffer_context<Char>::type> args) {
|
|
||||||
return internal::vformat(loc, to_string_view(format_str), args);
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename S, typename... Args>
|
|
||||||
inline std::basic_string<FMT_CHAR(S)> format(
|
|
||||||
const std::locale &loc, const S &format_str, const Args &... args) {
|
|
||||||
return internal::vformat(
|
|
||||||
loc, to_string_view(format_str),
|
|
||||||
*internal::checked_args<S, Args...>(format_str, args...));
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename String, typename OutputIt, typename... Args>
|
|
||||||
inline typename std::enable_if<internal::is_output_iterator<OutputIt>::value,
|
|
||||||
OutputIt>::type
|
|
||||||
vformat_to(OutputIt out, const std::locale &loc, const String &format_str,
|
|
||||||
typename format_args_t<OutputIt, FMT_CHAR(String)>::type args) {
|
|
||||||
typedef output_range<OutputIt, FMT_CHAR(String)> range;
|
|
||||||
return vformat_to<arg_formatter<range>>(
|
|
||||||
range(out), to_string_view(format_str), args, internal::locale_ref(loc));
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename OutputIt, typename S, typename... Args>
|
|
||||||
inline typename std::enable_if<
|
|
||||||
internal::is_string<S>::value &&
|
|
||||||
internal::is_output_iterator<OutputIt>::value, OutputIt>::type
|
|
||||||
format_to(OutputIt out, const std::locale &loc, const S &format_str,
|
|
||||||
const Args &... args) {
|
|
||||||
internal::check_format_string<Args...>(format_str);
|
|
||||||
typedef typename format_context_t<OutputIt, FMT_CHAR(S)>::type context;
|
|
||||||
format_arg_store<context, Args...> as{args...};
|
|
||||||
return vformat_to(out, loc, to_string_view(format_str),
|
|
||||||
basic_format_args<context>(as));
|
|
||||||
}
|
|
||||||
|
|
||||||
FMT_END_NAMESPACE
|
|
||||||
|
|
||||||
#endif // FMT_LOCALE_H_
|
|
||||||
153
deps/spdlog/fmt/bundled/ostream.h
vendored
153
deps/spdlog/fmt/bundled/ostream.h
vendored
@ -1,153 +0,0 @@
|
|||||||
// Formatting library for C++ - std::ostream support
|
|
||||||
//
|
|
||||||
// Copyright (c) 2012 - present, Victor Zverovich
|
|
||||||
// All rights reserved.
|
|
||||||
//
|
|
||||||
// For the license information refer to format.h.
|
|
||||||
|
|
||||||
#ifndef FMT_OSTREAM_H_
|
|
||||||
#define FMT_OSTREAM_H_
|
|
||||||
|
|
||||||
#include "format.h"
|
|
||||||
#include <ostream>
|
|
||||||
|
|
||||||
FMT_BEGIN_NAMESPACE
|
|
||||||
namespace internal {
|
|
||||||
|
|
||||||
template <class Char>
|
|
||||||
class formatbuf : public std::basic_streambuf<Char> {
|
|
||||||
private:
|
|
||||||
typedef typename std::basic_streambuf<Char>::int_type int_type;
|
|
||||||
typedef typename std::basic_streambuf<Char>::traits_type traits_type;
|
|
||||||
|
|
||||||
basic_buffer<Char> &buffer_;
|
|
||||||
|
|
||||||
public:
|
|
||||||
formatbuf(basic_buffer<Char> &buffer) : buffer_(buffer) {}
|
|
||||||
|
|
||||||
protected:
|
|
||||||
// The put-area is actually always empty. This makes the implementation
|
|
||||||
// simpler and has the advantage that the streambuf and the buffer are always
|
|
||||||
// in sync and sputc never writes into uninitialized memory. The obvious
|
|
||||||
// disadvantage is that each call to sputc always results in a (virtual) call
|
|
||||||
// to overflow. There is no disadvantage here for sputn since this always
|
|
||||||
// results in a call to xsputn.
|
|
||||||
|
|
||||||
int_type overflow(int_type ch = traits_type::eof()) FMT_OVERRIDE {
|
|
||||||
if (!traits_type::eq_int_type(ch, traits_type::eof()))
|
|
||||||
buffer_.push_back(static_cast<Char>(ch));
|
|
||||||
return ch;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::streamsize xsputn(const Char *s, std::streamsize count) FMT_OVERRIDE {
|
|
||||||
buffer_.append(s, s + count);
|
|
||||||
return count;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
template <typename Char>
|
|
||||||
struct test_stream : std::basic_ostream<Char> {
|
|
||||||
private:
|
|
||||||
struct null;
|
|
||||||
// Hide all operator<< from std::basic_ostream<Char>.
|
|
||||||
void operator<<(null);
|
|
||||||
};
|
|
||||||
|
|
||||||
// Checks if T has a user-defined operator<< (e.g. not a member of std::ostream).
|
|
||||||
template <typename T, typename Char>
|
|
||||||
class is_streamable {
|
|
||||||
private:
|
|
||||||
template <typename U>
|
|
||||||
static decltype(
|
|
||||||
internal::declval<test_stream<Char>&>()
|
|
||||||
<< internal::declval<U>(), std::true_type()) test(int);
|
|
||||||
|
|
||||||
template <typename>
|
|
||||||
static std::false_type test(...);
|
|
||||||
|
|
||||||
typedef decltype(test<T>(0)) result;
|
|
||||||
|
|
||||||
public:
|
|
||||||
static const bool value = result::value;
|
|
||||||
};
|
|
||||||
|
|
||||||
// Write the content of buf to os.
|
|
||||||
template <typename Char>
|
|
||||||
void write(std::basic_ostream<Char> &os, basic_buffer<Char> &buf) {
|
|
||||||
const Char *data = buf.data();
|
|
||||||
typedef std::make_unsigned<std::streamsize>::type UnsignedStreamSize;
|
|
||||||
UnsignedStreamSize size = buf.size();
|
|
||||||
UnsignedStreamSize max_size =
|
|
||||||
internal::to_unsigned((std::numeric_limits<std::streamsize>::max)());
|
|
||||||
do {
|
|
||||||
UnsignedStreamSize n = size <= max_size ? size : max_size;
|
|
||||||
os.write(data, static_cast<std::streamsize>(n));
|
|
||||||
data += n;
|
|
||||||
size -= n;
|
|
||||||
} while (size != 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename Char, typename T>
|
|
||||||
void format_value(basic_buffer<Char> &buffer, const T &value) {
|
|
||||||
internal::formatbuf<Char> format_buf(buffer);
|
|
||||||
std::basic_ostream<Char> output(&format_buf);
|
|
||||||
output.exceptions(std::ios_base::failbit | std::ios_base::badbit);
|
|
||||||
output << value;
|
|
||||||
buffer.resize(buffer.size());
|
|
||||||
}
|
|
||||||
} // namespace internal
|
|
||||||
|
|
||||||
// Disable conversion to int if T has an overloaded operator<< which is a free
|
|
||||||
// function (not a member of std::ostream).
|
|
||||||
template <typename T, typename Char>
|
|
||||||
struct convert_to_int<T, Char, void> {
|
|
||||||
static const bool value =
|
|
||||||
convert_to_int<T, Char, int>::value &&
|
|
||||||
!internal::is_streamable<T, Char>::value;
|
|
||||||
};
|
|
||||||
|
|
||||||
// Formats an object of type T that has an overloaded ostream operator<<.
|
|
||||||
template <typename T, typename Char>
|
|
||||||
struct formatter<T, Char,
|
|
||||||
typename std::enable_if<
|
|
||||||
internal::is_streamable<T, Char>::value &&
|
|
||||||
!internal::format_type<
|
|
||||||
typename buffer_context<Char>::type, T>::value>::type>
|
|
||||||
: formatter<basic_string_view<Char>, Char> {
|
|
||||||
|
|
||||||
template <typename Context>
|
|
||||||
auto format(const T &value, Context &ctx) -> decltype(ctx.out()) {
|
|
||||||
basic_memory_buffer<Char> buffer;
|
|
||||||
internal::format_value(buffer, value);
|
|
||||||
basic_string_view<Char> str(buffer.data(), buffer.size());
|
|
||||||
return formatter<basic_string_view<Char>, Char>::format(str, ctx);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
template <typename Char>
|
|
||||||
inline void vprint(std::basic_ostream<Char> &os,
|
|
||||||
basic_string_view<Char> format_str,
|
|
||||||
basic_format_args<typename buffer_context<Char>::type> args) {
|
|
||||||
basic_memory_buffer<Char> buffer;
|
|
||||||
internal::vformat_to(buffer, format_str, args);
|
|
||||||
internal::write(os, buffer);
|
|
||||||
}
|
|
||||||
/**
|
|
||||||
\rst
|
|
||||||
Prints formatted data to the stream *os*.
|
|
||||||
|
|
||||||
**Example**::
|
|
||||||
|
|
||||||
fmt::print(cerr, "Don't {}!", "panic");
|
|
||||||
\endrst
|
|
||||||
*/
|
|
||||||
template <typename S, typename... Args>
|
|
||||||
inline typename std::enable_if<internal::is_string<S>::value>::type
|
|
||||||
print(std::basic_ostream<FMT_CHAR(S)> &os, const S &format_str,
|
|
||||||
const Args & ... args) {
|
|
||||||
internal::checked_args<S, Args...> ca(format_str, args...);
|
|
||||||
vprint(os, to_string_view(format_str), *ca);
|
|
||||||
}
|
|
||||||
FMT_END_NAMESPACE
|
|
||||||
|
|
||||||
#endif // FMT_OSTREAM_H_
|
|
||||||
324
deps/spdlog/fmt/bundled/posix.h
vendored
324
deps/spdlog/fmt/bundled/posix.h
vendored
@ -1,324 +0,0 @@
|
|||||||
// A C++ interface to POSIX functions.
|
|
||||||
//
|
|
||||||
// Copyright (c) 2012 - 2016, Victor Zverovich
|
|
||||||
// All rights reserved.
|
|
||||||
//
|
|
||||||
// For the license information refer to format.h.
|
|
||||||
|
|
||||||
#ifndef FMT_POSIX_H_
|
|
||||||
#define FMT_POSIX_H_
|
|
||||||
|
|
||||||
#if defined(__MINGW32__) || defined(__CYGWIN__)
|
|
||||||
// Workaround MinGW bug https://sourceforge.net/p/mingw/bugs/2024/.
|
|
||||||
# undef __STRICT_ANSI__
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#include <errno.h>
|
|
||||||
#include <fcntl.h> // for O_RDONLY
|
|
||||||
#include <locale.h> // for locale_t
|
|
||||||
#include <stdio.h>
|
|
||||||
#include <stdlib.h> // for strtod_l
|
|
||||||
|
|
||||||
#include <cstddef>
|
|
||||||
|
|
||||||
#if defined __APPLE__ || defined(__FreeBSD__)
|
|
||||||
# include <xlocale.h> // for LC_NUMERIC_MASK on OS X
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#include "format.h"
|
|
||||||
|
|
||||||
#ifndef FMT_POSIX
|
|
||||||
# if defined(_WIN32) && !defined(__MINGW32__)
|
|
||||||
// Fix warnings about deprecated symbols.
|
|
||||||
# define FMT_POSIX(call) _##call
|
|
||||||
# else
|
|
||||||
# define FMT_POSIX(call) call
|
|
||||||
# endif
|
|
||||||
#endif
|
|
||||||
|
|
||||||
// Calls to system functions are wrapped in FMT_SYSTEM for testability.
|
|
||||||
#ifdef FMT_SYSTEM
|
|
||||||
# define FMT_POSIX_CALL(call) FMT_SYSTEM(call)
|
|
||||||
#else
|
|
||||||
# define FMT_SYSTEM(call) call
|
|
||||||
# ifdef _WIN32
|
|
||||||
// Fix warnings about deprecated symbols.
|
|
||||||
# define FMT_POSIX_CALL(call) ::_##call
|
|
||||||
# else
|
|
||||||
# define FMT_POSIX_CALL(call) ::call
|
|
||||||
# endif
|
|
||||||
#endif
|
|
||||||
|
|
||||||
// Retries the expression while it evaluates to error_result and errno
|
|
||||||
// equals to EINTR.
|
|
||||||
#ifndef _WIN32
|
|
||||||
# define FMT_RETRY_VAL(result, expression, error_result) \
|
|
||||||
do { \
|
|
||||||
result = (expression); \
|
|
||||||
} while (result == error_result && errno == EINTR)
|
|
||||||
#else
|
|
||||||
# define FMT_RETRY_VAL(result, expression, error_result) result = (expression)
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#define FMT_RETRY(result, expression) FMT_RETRY_VAL(result, expression, -1)
|
|
||||||
|
|
||||||
FMT_BEGIN_NAMESPACE
|
|
||||||
|
|
||||||
/**
|
|
||||||
\rst
|
|
||||||
A reference to a null-terminated string. It can be constructed from a C
|
|
||||||
string or ``std::string``.
|
|
||||||
|
|
||||||
You can use one of the following typedefs for common character types:
|
|
||||||
|
|
||||||
+---------------+-----------------------------+
|
|
||||||
| Type | Definition |
|
|
||||||
+===============+=============================+
|
|
||||||
| cstring_view | basic_cstring_view<char> |
|
|
||||||
+---------------+-----------------------------+
|
|
||||||
| wcstring_view | basic_cstring_view<wchar_t> |
|
|
||||||
+---------------+-----------------------------+
|
|
||||||
|
|
||||||
This class is most useful as a parameter type to allow passing
|
|
||||||
different types of strings to a function, for example::
|
|
||||||
|
|
||||||
template <typename... Args>
|
|
||||||
std::string format(cstring_view format_str, const Args & ... args);
|
|
||||||
|
|
||||||
format("{}", 42);
|
|
||||||
format(std::string("{}"), 42);
|
|
||||||
\endrst
|
|
||||||
*/
|
|
||||||
template <typename Char>
|
|
||||||
class basic_cstring_view {
|
|
||||||
private:
|
|
||||||
const Char *data_;
|
|
||||||
|
|
||||||
public:
|
|
||||||
/** Constructs a string reference object from a C string. */
|
|
||||||
basic_cstring_view(const Char *s) : data_(s) {}
|
|
||||||
|
|
||||||
/**
|
|
||||||
\rst
|
|
||||||
Constructs a string reference from an ``std::string`` object.
|
|
||||||
\endrst
|
|
||||||
*/
|
|
||||||
basic_cstring_view(const std::basic_string<Char> &s) : data_(s.c_str()) {}
|
|
||||||
|
|
||||||
/** Returns the pointer to a C string. */
|
|
||||||
const Char *c_str() const { return data_; }
|
|
||||||
};
|
|
||||||
|
|
||||||
typedef basic_cstring_view<char> cstring_view;
|
|
||||||
typedef basic_cstring_view<wchar_t> wcstring_view;
|
|
||||||
|
|
||||||
// An error code.
|
|
||||||
class error_code {
|
|
||||||
private:
|
|
||||||
int value_;
|
|
||||||
|
|
||||||
public:
|
|
||||||
explicit error_code(int value = 0) FMT_NOEXCEPT : value_(value) {}
|
|
||||||
|
|
||||||
int get() const FMT_NOEXCEPT { return value_; }
|
|
||||||
};
|
|
||||||
|
|
||||||
// A buffered file.
|
|
||||||
class buffered_file {
|
|
||||||
private:
|
|
||||||
FILE *file_;
|
|
||||||
|
|
||||||
friend class file;
|
|
||||||
|
|
||||||
explicit buffered_file(FILE *f) : file_(f) {}
|
|
||||||
|
|
||||||
public:
|
|
||||||
// Constructs a buffered_file object which doesn't represent any file.
|
|
||||||
buffered_file() FMT_NOEXCEPT : file_(FMT_NULL) {}
|
|
||||||
|
|
||||||
// Destroys the object closing the file it represents if any.
|
|
||||||
FMT_API ~buffered_file() FMT_NOEXCEPT;
|
|
||||||
|
|
||||||
private:
|
|
||||||
buffered_file(const buffered_file &) = delete;
|
|
||||||
void operator=(const buffered_file &) = delete;
|
|
||||||
|
|
||||||
|
|
||||||
public:
|
|
||||||
buffered_file(buffered_file &&other) FMT_NOEXCEPT : file_(other.file_) {
|
|
||||||
other.file_ = FMT_NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
buffered_file& operator=(buffered_file &&other) {
|
|
||||||
close();
|
|
||||||
file_ = other.file_;
|
|
||||||
other.file_ = FMT_NULL;
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Opens a file.
|
|
||||||
FMT_API buffered_file(cstring_view filename, cstring_view mode);
|
|
||||||
|
|
||||||
// Closes the file.
|
|
||||||
FMT_API void close();
|
|
||||||
|
|
||||||
// Returns the pointer to a FILE object representing this file.
|
|
||||||
FILE *get() const FMT_NOEXCEPT { return file_; }
|
|
||||||
|
|
||||||
// We place parentheses around fileno to workaround a bug in some versions
|
|
||||||
// of MinGW that define fileno as a macro.
|
|
||||||
FMT_API int (fileno)() const;
|
|
||||||
|
|
||||||
void vprint(string_view format_str, format_args args) {
|
|
||||||
fmt::vprint(file_, format_str, args);
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename... Args>
|
|
||||||
inline void print(string_view format_str, const Args & ... args) {
|
|
||||||
vprint(format_str, make_format_args(args...));
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
// A file. Closed file is represented by a file object with descriptor -1.
|
|
||||||
// Methods that are not declared with FMT_NOEXCEPT may throw
|
|
||||||
// fmt::system_error in case of failure. Note that some errors such as
|
|
||||||
// closing the file multiple times will cause a crash on Windows rather
|
|
||||||
// than an exception. You can get standard behavior by overriding the
|
|
||||||
// invalid parameter handler with _set_invalid_parameter_handler.
|
|
||||||
class file {
|
|
||||||
private:
|
|
||||||
int fd_; // File descriptor.
|
|
||||||
|
|
||||||
// Constructs a file object with a given descriptor.
|
|
||||||
explicit file(int fd) : fd_(fd) {}
|
|
||||||
|
|
||||||
public:
|
|
||||||
// Possible values for the oflag argument to the constructor.
|
|
||||||
enum {
|
|
||||||
RDONLY = FMT_POSIX(O_RDONLY), // Open for reading only.
|
|
||||||
WRONLY = FMT_POSIX(O_WRONLY), // Open for writing only.
|
|
||||||
RDWR = FMT_POSIX(O_RDWR) // Open for reading and writing.
|
|
||||||
};
|
|
||||||
|
|
||||||
// Constructs a file object which doesn't represent any file.
|
|
||||||
file() FMT_NOEXCEPT : fd_(-1) {}
|
|
||||||
|
|
||||||
// Opens a file and constructs a file object representing this file.
|
|
||||||
FMT_API file(cstring_view path, int oflag);
|
|
||||||
|
|
||||||
private:
|
|
||||||
file(const file &) = delete;
|
|
||||||
void operator=(const file &) = delete;
|
|
||||||
|
|
||||||
public:
|
|
||||||
file(file &&other) FMT_NOEXCEPT : fd_(other.fd_) {
|
|
||||||
other.fd_ = -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
file& operator=(file &&other) {
|
|
||||||
close();
|
|
||||||
fd_ = other.fd_;
|
|
||||||
other.fd_ = -1;
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Destroys the object closing the file it represents if any.
|
|
||||||
FMT_API ~file() FMT_NOEXCEPT;
|
|
||||||
|
|
||||||
// Returns the file descriptor.
|
|
||||||
int descriptor() const FMT_NOEXCEPT { return fd_; }
|
|
||||||
|
|
||||||
// Closes the file.
|
|
||||||
FMT_API void close();
|
|
||||||
|
|
||||||
// Returns the file size. The size has signed type for consistency with
|
|
||||||
// stat::st_size.
|
|
||||||
FMT_API long long size() const;
|
|
||||||
|
|
||||||
// Attempts to read count bytes from the file into the specified buffer.
|
|
||||||
FMT_API std::size_t read(void *buffer, std::size_t count);
|
|
||||||
|
|
||||||
// Attempts to write count bytes from the specified buffer to the file.
|
|
||||||
FMT_API std::size_t write(const void *buffer, std::size_t count);
|
|
||||||
|
|
||||||
// Duplicates a file descriptor with the dup function and returns
|
|
||||||
// the duplicate as a file object.
|
|
||||||
FMT_API static file dup(int fd);
|
|
||||||
|
|
||||||
// Makes fd be the copy of this file descriptor, closing fd first if
|
|
||||||
// necessary.
|
|
||||||
FMT_API void dup2(int fd);
|
|
||||||
|
|
||||||
// Makes fd be the copy of this file descriptor, closing fd first if
|
|
||||||
// necessary.
|
|
||||||
FMT_API void dup2(int fd, error_code &ec) FMT_NOEXCEPT;
|
|
||||||
|
|
||||||
// Creates a pipe setting up read_end and write_end file objects for reading
|
|
||||||
// and writing respectively.
|
|
||||||
FMT_API static void pipe(file &read_end, file &write_end);
|
|
||||||
|
|
||||||
// Creates a buffered_file object associated with this file and detaches
|
|
||||||
// this file object from the file.
|
|
||||||
FMT_API buffered_file fdopen(const char *mode);
|
|
||||||
};
|
|
||||||
|
|
||||||
// Returns the memory page size.
|
|
||||||
long getpagesize();
|
|
||||||
|
|
||||||
#if (defined(LC_NUMERIC_MASK) || defined(_MSC_VER)) && \
|
|
||||||
!defined(__ANDROID__) && !defined(__CYGWIN__) && !defined(__OpenBSD__) && \
|
|
||||||
!defined(__NEWLIB_H__)
|
|
||||||
# define FMT_LOCALE
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifdef FMT_LOCALE
|
|
||||||
// A "C" numeric locale.
|
|
||||||
class Locale {
|
|
||||||
private:
|
|
||||||
# ifdef _MSC_VER
|
|
||||||
typedef _locale_t locale_t;
|
|
||||||
|
|
||||||
enum { LC_NUMERIC_MASK = LC_NUMERIC };
|
|
||||||
|
|
||||||
static locale_t newlocale(int category_mask, const char *locale, locale_t) {
|
|
||||||
return _create_locale(category_mask, locale);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void freelocale(locale_t locale) {
|
|
||||||
_free_locale(locale);
|
|
||||||
}
|
|
||||||
|
|
||||||
static double strtod_l(const char *nptr, char **endptr, _locale_t locale) {
|
|
||||||
return _strtod_l(nptr, endptr, locale);
|
|
||||||
}
|
|
||||||
# endif
|
|
||||||
|
|
||||||
locale_t locale_;
|
|
||||||
|
|
||||||
Locale(const Locale &) = delete;
|
|
||||||
void operator=(const Locale &) = delete;
|
|
||||||
|
|
||||||
public:
|
|
||||||
typedef locale_t Type;
|
|
||||||
|
|
||||||
Locale() : locale_(newlocale(LC_NUMERIC_MASK, "C", FMT_NULL)) {
|
|
||||||
if (!locale_)
|
|
||||||
FMT_THROW(system_error(errno, "cannot create locale"));
|
|
||||||
}
|
|
||||||
~Locale() { freelocale(locale_); }
|
|
||||||
|
|
||||||
Type get() const { return locale_; }
|
|
||||||
|
|
||||||
// Converts string to floating-point number and advances str past the end
|
|
||||||
// of the parsed input.
|
|
||||||
double strtod(const char *&str) const {
|
|
||||||
char *end = FMT_NULL;
|
|
||||||
double result = strtod_l(str, &end, locale_);
|
|
||||||
str = end;
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
#endif // FMT_LOCALE
|
|
||||||
FMT_END_NAMESPACE
|
|
||||||
|
|
||||||
#endif // FMT_POSIX_H_
|
|
||||||
855
deps/spdlog/fmt/bundled/printf.h
vendored
855
deps/spdlog/fmt/bundled/printf.h
vendored
@ -1,855 +0,0 @@
|
|||||||
// Formatting library for C++
|
|
||||||
//
|
|
||||||
// Copyright (c) 2012 - 2016, Victor Zverovich
|
|
||||||
// All rights reserved.
|
|
||||||
//
|
|
||||||
// For the license information refer to format.h.
|
|
||||||
|
|
||||||
#ifndef FMT_PRINTF_H_
|
|
||||||
#define FMT_PRINTF_H_
|
|
||||||
|
|
||||||
#include <algorithm> // std::fill_n
|
|
||||||
#include <limits> // std::numeric_limits
|
|
||||||
|
|
||||||
#include "ostream.h"
|
|
||||||
|
|
||||||
FMT_BEGIN_NAMESPACE
|
|
||||||
namespace internal {
|
|
||||||
|
|
||||||
// An iterator that produces a null terminator on *end. This simplifies parsing
|
|
||||||
// and allows comparing the performance of processing a null-terminated string
|
|
||||||
// vs string_view.
|
|
||||||
template <typename Char>
|
|
||||||
class null_terminating_iterator {
|
|
||||||
public:
|
|
||||||
typedef std::ptrdiff_t difference_type;
|
|
||||||
typedef Char value_type;
|
|
||||||
typedef const Char* pointer;
|
|
||||||
typedef const Char& reference;
|
|
||||||
typedef std::random_access_iterator_tag iterator_category;
|
|
||||||
|
|
||||||
null_terminating_iterator() : ptr_(0), end_(0) {}
|
|
||||||
|
|
||||||
FMT_CONSTEXPR null_terminating_iterator(const Char *ptr, const Char *end)
|
|
||||||
: ptr_(ptr), end_(end) {}
|
|
||||||
|
|
||||||
template <typename Range>
|
|
||||||
FMT_CONSTEXPR explicit null_terminating_iterator(const Range &r)
|
|
||||||
: ptr_(r.begin()), end_(r.end()) {}
|
|
||||||
|
|
||||||
FMT_CONSTEXPR null_terminating_iterator &operator=(const Char *ptr) {
|
|
||||||
assert(ptr <= end_);
|
|
||||||
ptr_ = ptr;
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
|
|
||||||
FMT_CONSTEXPR Char operator*() const {
|
|
||||||
return ptr_ != end_ ? *ptr_ : Char();
|
|
||||||
}
|
|
||||||
|
|
||||||
FMT_CONSTEXPR null_terminating_iterator operator++() {
|
|
||||||
++ptr_;
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
|
|
||||||
FMT_CONSTEXPR null_terminating_iterator operator++(int) {
|
|
||||||
null_terminating_iterator result(*this);
|
|
||||||
++ptr_;
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
FMT_CONSTEXPR null_terminating_iterator operator--() {
|
|
||||||
--ptr_;
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
|
|
||||||
FMT_CONSTEXPR null_terminating_iterator operator+(difference_type n) {
|
|
||||||
return null_terminating_iterator(ptr_ + n, end_);
|
|
||||||
}
|
|
||||||
|
|
||||||
FMT_CONSTEXPR null_terminating_iterator operator-(difference_type n) {
|
|
||||||
return null_terminating_iterator(ptr_ - n, end_);
|
|
||||||
}
|
|
||||||
|
|
||||||
FMT_CONSTEXPR null_terminating_iterator operator+=(difference_type n) {
|
|
||||||
ptr_ += n;
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
|
|
||||||
FMT_CONSTEXPR difference_type operator-(
|
|
||||||
null_terminating_iterator other) const {
|
|
||||||
return ptr_ - other.ptr_;
|
|
||||||
}
|
|
||||||
|
|
||||||
FMT_CONSTEXPR bool operator!=(null_terminating_iterator other) const {
|
|
||||||
return ptr_ != other.ptr_;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool operator>=(null_terminating_iterator other) const {
|
|
||||||
return ptr_ >= other.ptr_;
|
|
||||||
}
|
|
||||||
|
|
||||||
// This should be a friend specialization pointer_from<Char> but the latter
|
|
||||||
// doesn't compile by gcc 5.1 due to a compiler bug.
|
|
||||||
template <typename CharT>
|
|
||||||
friend FMT_CONSTEXPR_DECL const CharT *pointer_from(
|
|
||||||
null_terminating_iterator<CharT> it);
|
|
||||||
|
|
||||||
private:
|
|
||||||
const Char *ptr_;
|
|
||||||
const Char *end_;
|
|
||||||
};
|
|
||||||
|
|
||||||
template <typename T>
|
|
||||||
FMT_CONSTEXPR const T *pointer_from(const T *p) { return p; }
|
|
||||||
|
|
||||||
template <typename Char>
|
|
||||||
FMT_CONSTEXPR const Char *pointer_from(null_terminating_iterator<Char> it) {
|
|
||||||
return it.ptr_;
|
|
||||||
}
|
|
||||||
|
|
||||||
// DEPRECATED: Parses the input as an unsigned integer. This function assumes
|
|
||||||
// that the first character is a digit and presence of a non-digit character at
|
|
||||||
// the end.
|
|
||||||
// it: an iterator pointing to the beginning of the input range.
|
|
||||||
template <typename Iterator, typename ErrorHandler>
|
|
||||||
FMT_CONSTEXPR unsigned parse_nonnegative_int(Iterator &it, ErrorHandler &&eh) {
|
|
||||||
assert('0' <= *it && *it <= '9');
|
|
||||||
if (*it == '0') {
|
|
||||||
++it;
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
unsigned value = 0;
|
|
||||||
// Convert to unsigned to prevent a warning.
|
|
||||||
unsigned max_int = (std::numeric_limits<int>::max)();
|
|
||||||
unsigned big = max_int / 10;
|
|
||||||
do {
|
|
||||||
// Check for overflow.
|
|
||||||
if (value > big) {
|
|
||||||
value = max_int + 1;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
value = value * 10 + unsigned(*it - '0');
|
|
||||||
// Workaround for MSVC "setup_exception stack overflow" error:
|
|
||||||
auto next = it;
|
|
||||||
++next;
|
|
||||||
it = next;
|
|
||||||
} while ('0' <= *it && *it <= '9');
|
|
||||||
if (value > max_int)
|
|
||||||
eh.on_error("number is too big");
|
|
||||||
return value;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Checks if a value fits in int - used to avoid warnings about comparing
|
|
||||||
// signed and unsigned integers.
|
|
||||||
template <bool IsSigned>
|
|
||||||
struct int_checker {
|
|
||||||
template <typename T>
|
|
||||||
static bool fits_in_int(T value) {
|
|
||||||
unsigned max = std::numeric_limits<int>::max();
|
|
||||||
return value <= max;
|
|
||||||
}
|
|
||||||
static bool fits_in_int(bool) { return true; }
|
|
||||||
};
|
|
||||||
|
|
||||||
template <>
|
|
||||||
struct int_checker<true> {
|
|
||||||
template <typename T>
|
|
||||||
static bool fits_in_int(T value) {
|
|
||||||
return value >= std::numeric_limits<int>::min() &&
|
|
||||||
value <= std::numeric_limits<int>::max();
|
|
||||||
}
|
|
||||||
static bool fits_in_int(int) { return true; }
|
|
||||||
};
|
|
||||||
|
|
||||||
class printf_precision_handler: public function<int> {
|
|
||||||
public:
|
|
||||||
template <typename T>
|
|
||||||
typename std::enable_if<std::is_integral<T>::value, int>::type
|
|
||||||
operator()(T value) {
|
|
||||||
if (!int_checker<std::numeric_limits<T>::is_signed>::fits_in_int(value))
|
|
||||||
FMT_THROW(format_error("number is too big"));
|
|
||||||
return static_cast<int>(value);
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename T>
|
|
||||||
typename std::enable_if<!std::is_integral<T>::value, int>::type operator()(T) {
|
|
||||||
FMT_THROW(format_error("precision is not integer"));
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
// An argument visitor that returns true iff arg is a zero integer.
|
|
||||||
class is_zero_int: public function<bool> {
|
|
||||||
public:
|
|
||||||
template <typename T>
|
|
||||||
typename std::enable_if<std::is_integral<T>::value, bool>::type
|
|
||||||
operator()(T value) { return value == 0; }
|
|
||||||
|
|
||||||
template <typename T>
|
|
||||||
typename std::enable_if<!std::is_integral<T>::value, bool>::type
|
|
||||||
operator()(T) { return false; }
|
|
||||||
};
|
|
||||||
|
|
||||||
template <typename T>
|
|
||||||
struct make_unsigned_or_bool : std::make_unsigned<T> {};
|
|
||||||
|
|
||||||
template <>
|
|
||||||
struct make_unsigned_or_bool<bool> {
|
|
||||||
typedef bool type;
|
|
||||||
};
|
|
||||||
|
|
||||||
template <typename T, typename Context>
|
|
||||||
class arg_converter: public function<void> {
|
|
||||||
private:
|
|
||||||
typedef typename Context::char_type Char;
|
|
||||||
|
|
||||||
basic_format_arg<Context> &arg_;
|
|
||||||
typename Context::char_type type_;
|
|
||||||
|
|
||||||
public:
|
|
||||||
arg_converter(basic_format_arg<Context> &arg, Char type)
|
|
||||||
: arg_(arg), type_(type) {}
|
|
||||||
|
|
||||||
void operator()(bool value) {
|
|
||||||
if (type_ != 's')
|
|
||||||
operator()<bool>(value);
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename U>
|
|
||||||
typename std::enable_if<std::is_integral<U>::value>::type
|
|
||||||
operator()(U value) {
|
|
||||||
bool is_signed = type_ == 'd' || type_ == 'i';
|
|
||||||
typedef typename std::conditional<
|
|
||||||
std::is_same<T, void>::value, U, T>::type TargetType;
|
|
||||||
if (const_check(sizeof(TargetType) <= sizeof(int))) {
|
|
||||||
// Extra casts are used to silence warnings.
|
|
||||||
if (is_signed) {
|
|
||||||
arg_ = internal::make_arg<Context>(
|
|
||||||
static_cast<int>(static_cast<TargetType>(value)));
|
|
||||||
} else {
|
|
||||||
typedef typename make_unsigned_or_bool<TargetType>::type Unsigned;
|
|
||||||
arg_ = internal::make_arg<Context>(
|
|
||||||
static_cast<unsigned>(static_cast<Unsigned>(value)));
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
if (is_signed) {
|
|
||||||
// glibc's printf doesn't sign extend arguments of smaller types:
|
|
||||||
// std::printf("%lld", -42); // prints "4294967254"
|
|
||||||
// but we don't have to do the same because it's a UB.
|
|
||||||
arg_ = internal::make_arg<Context>(static_cast<long long>(value));
|
|
||||||
} else {
|
|
||||||
arg_ = internal::make_arg<Context>(
|
|
||||||
static_cast<typename make_unsigned_or_bool<U>::type>(value));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename U>
|
|
||||||
typename std::enable_if<!std::is_integral<U>::value>::type operator()(U) {
|
|
||||||
// No coversion needed for non-integral types.
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
// Converts an integer argument to T for printf, if T is an integral type.
|
|
||||||
// If T is void, the argument is converted to corresponding signed or unsigned
|
|
||||||
// type depending on the type specifier: 'd' and 'i' - signed, other -
|
|
||||||
// unsigned).
|
|
||||||
template <typename T, typename Context, typename Char>
|
|
||||||
void convert_arg(basic_format_arg<Context> &arg, Char type) {
|
|
||||||
visit_format_arg(arg_converter<T, Context>(arg, type), arg);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Converts an integer argument to char for printf.
|
|
||||||
template <typename Context>
|
|
||||||
class char_converter: public function<void> {
|
|
||||||
private:
|
|
||||||
basic_format_arg<Context> &arg_;
|
|
||||||
|
|
||||||
public:
|
|
||||||
explicit char_converter(basic_format_arg<Context> &arg) : arg_(arg) {}
|
|
||||||
|
|
||||||
template <typename T>
|
|
||||||
typename std::enable_if<std::is_integral<T>::value>::type
|
|
||||||
operator()(T value) {
|
|
||||||
typedef typename Context::char_type Char;
|
|
||||||
arg_ = internal::make_arg<Context>(static_cast<Char>(value));
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename T>
|
|
||||||
typename std::enable_if<!std::is_integral<T>::value>::type operator()(T) {
|
|
||||||
// No coversion needed for non-integral types.
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
// Checks if an argument is a valid printf width specifier and sets
|
|
||||||
// left alignment if it is negative.
|
|
||||||
template <typename Char>
|
|
||||||
class printf_width_handler: public function<unsigned> {
|
|
||||||
private:
|
|
||||||
typedef basic_format_specs<Char> format_specs;
|
|
||||||
|
|
||||||
format_specs &spec_;
|
|
||||||
|
|
||||||
public:
|
|
||||||
explicit printf_width_handler(format_specs &spec) : spec_(spec) {}
|
|
||||||
|
|
||||||
template <typename T>
|
|
||||||
typename std::enable_if<std::is_integral<T>::value, unsigned>::type
|
|
||||||
operator()(T value) {
|
|
||||||
typedef typename internal::int_traits<T>::main_type UnsignedType;
|
|
||||||
UnsignedType width = static_cast<UnsignedType>(value);
|
|
||||||
if (internal::is_negative(value)) {
|
|
||||||
spec_.align_ = ALIGN_LEFT;
|
|
||||||
width = 0 - width;
|
|
||||||
}
|
|
||||||
unsigned int_max = std::numeric_limits<int>::max();
|
|
||||||
if (width > int_max)
|
|
||||||
FMT_THROW(format_error("number is too big"));
|
|
||||||
return static_cast<unsigned>(width);
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename T>
|
|
||||||
typename std::enable_if<!std::is_integral<T>::value, unsigned>::type
|
|
||||||
operator()(T) {
|
|
||||||
FMT_THROW(format_error("width is not integer"));
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
template <typename Char, typename Context>
|
|
||||||
void printf(basic_buffer<Char> &buf, basic_string_view<Char> format,
|
|
||||||
basic_format_args<Context> args) {
|
|
||||||
Context(std::back_inserter(buf), format, args).format();
|
|
||||||
}
|
|
||||||
} // namespace internal
|
|
||||||
|
|
||||||
using internal::printf; // For printing into memory_buffer.
|
|
||||||
|
|
||||||
template <typename Range>
|
|
||||||
class printf_arg_formatter;
|
|
||||||
|
|
||||||
template <
|
|
||||||
typename OutputIt, typename Char,
|
|
||||||
typename ArgFormatter =
|
|
||||||
printf_arg_formatter<back_insert_range<internal::basic_buffer<Char>>>>
|
|
||||||
class basic_printf_context;
|
|
||||||
|
|
||||||
/**
|
|
||||||
\rst
|
|
||||||
The ``printf`` argument formatter.
|
|
||||||
\endrst
|
|
||||||
*/
|
|
||||||
template <typename Range>
|
|
||||||
class printf_arg_formatter:
|
|
||||||
public internal::function<
|
|
||||||
typename internal::arg_formatter_base<Range>::iterator>,
|
|
||||||
public internal::arg_formatter_base<Range> {
|
|
||||||
private:
|
|
||||||
typedef typename Range::value_type char_type;
|
|
||||||
typedef decltype(internal::declval<Range>().begin()) iterator;
|
|
||||||
typedef internal::arg_formatter_base<Range> base;
|
|
||||||
typedef basic_printf_context<iterator, char_type> context_type;
|
|
||||||
|
|
||||||
context_type &context_;
|
|
||||||
|
|
||||||
void write_null_pointer(char) {
|
|
||||||
this->spec()->type = 0;
|
|
||||||
this->write("(nil)");
|
|
||||||
}
|
|
||||||
|
|
||||||
void write_null_pointer(wchar_t) {
|
|
||||||
this->spec()->type = 0;
|
|
||||||
this->write(L"(nil)");
|
|
||||||
}
|
|
||||||
|
|
||||||
public:
|
|
||||||
typedef typename base::format_specs format_specs;
|
|
||||||
|
|
||||||
/**
|
|
||||||
\rst
|
|
||||||
Constructs an argument formatter object.
|
|
||||||
*buffer* is a reference to the output buffer and *spec* contains format
|
|
||||||
specifier information for standard argument types.
|
|
||||||
\endrst
|
|
||||||
*/
|
|
||||||
printf_arg_formatter(internal::basic_buffer<char_type> &buffer,
|
|
||||||
format_specs &spec, context_type &ctx)
|
|
||||||
: base(back_insert_range<internal::basic_buffer<char_type>>(buffer), &spec,
|
|
||||||
ctx.locale()),
|
|
||||||
context_(ctx) {}
|
|
||||||
|
|
||||||
template <typename T>
|
|
||||||
typename std::enable_if<std::is_integral<T>::value, iterator>::type
|
|
||||||
operator()(T value) {
|
|
||||||
// MSVC2013 fails to compile separate overloads for bool and char_type so
|
|
||||||
// use std::is_same instead.
|
|
||||||
if (std::is_same<T, bool>::value) {
|
|
||||||
format_specs &fmt_spec = *this->spec();
|
|
||||||
if (fmt_spec.type != 's')
|
|
||||||
return base::operator()(value ? 1 : 0);
|
|
||||||
fmt_spec.type = 0;
|
|
||||||
this->write(value != 0);
|
|
||||||
} else if (std::is_same<T, char_type>::value) {
|
|
||||||
format_specs &fmt_spec = *this->spec();
|
|
||||||
if (fmt_spec.type && fmt_spec.type != 'c')
|
|
||||||
return (*this)(static_cast<int>(value));
|
|
||||||
fmt_spec.flags = 0;
|
|
||||||
fmt_spec.align_ = ALIGN_RIGHT;
|
|
||||||
return base::operator()(value);
|
|
||||||
} else {
|
|
||||||
return base::operator()(value);
|
|
||||||
}
|
|
||||||
return this->out();
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename T>
|
|
||||||
typename std::enable_if<std::is_floating_point<T>::value, iterator>::type
|
|
||||||
operator()(T value) {
|
|
||||||
return base::operator()(value);
|
|
||||||
}
|
|
||||||
|
|
||||||
/** Formats a null-terminated C string. */
|
|
||||||
iterator operator()(const char *value) {
|
|
||||||
if (value)
|
|
||||||
base::operator()(value);
|
|
||||||
else if (this->spec()->type == 'p')
|
|
||||||
write_null_pointer(char_type());
|
|
||||||
else
|
|
||||||
this->write("(null)");
|
|
||||||
return this->out();
|
|
||||||
}
|
|
||||||
|
|
||||||
/** Formats a null-terminated wide C string. */
|
|
||||||
iterator operator()(const wchar_t *value) {
|
|
||||||
if (value)
|
|
||||||
base::operator()(value);
|
|
||||||
else if (this->spec()->type == 'p')
|
|
||||||
write_null_pointer(char_type());
|
|
||||||
else
|
|
||||||
this->write(L"(null)");
|
|
||||||
return this->out();
|
|
||||||
}
|
|
||||||
|
|
||||||
iterator operator()(basic_string_view<char_type> value) {
|
|
||||||
return base::operator()(value);
|
|
||||||
}
|
|
||||||
|
|
||||||
iterator operator()(monostate value) {
|
|
||||||
return base::operator()(value);
|
|
||||||
}
|
|
||||||
|
|
||||||
/** Formats a pointer. */
|
|
||||||
iterator operator()(const void *value) {
|
|
||||||
if (value)
|
|
||||||
return base::operator()(value);
|
|
||||||
this->spec()->type = 0;
|
|
||||||
write_null_pointer(char_type());
|
|
||||||
return this->out();
|
|
||||||
}
|
|
||||||
|
|
||||||
/** Formats an argument of a custom (user-defined) type. */
|
|
||||||
iterator operator()(typename basic_format_arg<context_type>::handle handle) {
|
|
||||||
handle.format(context_);
|
|
||||||
return this->out();
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
template <typename T>
|
|
||||||
struct printf_formatter {
|
|
||||||
template <typename ParseContext>
|
|
||||||
auto parse(ParseContext &ctx) -> decltype(ctx.begin()) { return ctx.begin(); }
|
|
||||||
|
|
||||||
template <typename FormatContext>
|
|
||||||
auto format(const T &value, FormatContext &ctx) -> decltype(ctx.out()) {
|
|
||||||
internal::format_value(internal::get_container(ctx.out()), value);
|
|
||||||
return ctx.out();
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
/** This template formats data and writes the output to a writer. */
|
|
||||||
template <typename OutputIt, typename Char, typename ArgFormatter>
|
|
||||||
class basic_printf_context :
|
|
||||||
// Inherit publicly as a workaround for the icc bug
|
|
||||||
// https://software.intel.com/en-us/forums/intel-c-compiler/topic/783476.
|
|
||||||
public internal::context_base<
|
|
||||||
OutputIt, basic_printf_context<OutputIt, Char, ArgFormatter>, Char> {
|
|
||||||
public:
|
|
||||||
/** The character type for the output. */
|
|
||||||
typedef Char char_type;
|
|
||||||
|
|
||||||
template <typename T>
|
|
||||||
struct formatter_type { typedef printf_formatter<T> type; };
|
|
||||||
|
|
||||||
private:
|
|
||||||
typedef internal::context_base<OutputIt, basic_printf_context, Char> base;
|
|
||||||
typedef typename base::format_arg format_arg;
|
|
||||||
typedef basic_format_specs<char_type> format_specs;
|
|
||||||
typedef internal::null_terminating_iterator<char_type> iterator;
|
|
||||||
|
|
||||||
void parse_flags(format_specs &spec, iterator &it);
|
|
||||||
|
|
||||||
// Returns the argument with specified index or, if arg_index is equal
|
|
||||||
// to the maximum unsigned value, the next argument.
|
|
||||||
format_arg get_arg(
|
|
||||||
iterator it,
|
|
||||||
unsigned arg_index = (std::numeric_limits<unsigned>::max)());
|
|
||||||
|
|
||||||
// Parses argument index, flags and width and returns the argument index.
|
|
||||||
unsigned parse_header(iterator &it, format_specs &spec);
|
|
||||||
|
|
||||||
public:
|
|
||||||
/**
|
|
||||||
\rst
|
|
||||||
Constructs a ``printf_context`` object. References to the arguments and
|
|
||||||
the writer are stored in the context object so make sure they have
|
|
||||||
appropriate lifetimes.
|
|
||||||
\endrst
|
|
||||||
*/
|
|
||||||
basic_printf_context(OutputIt out, basic_string_view<char_type> format_str,
|
|
||||||
basic_format_args<basic_printf_context> args)
|
|
||||||
: base(out, format_str, args) {}
|
|
||||||
|
|
||||||
using base::parse_context;
|
|
||||||
using base::out;
|
|
||||||
using base::advance_to;
|
|
||||||
|
|
||||||
/** Formats stored arguments and writes the output to the range. */
|
|
||||||
void format();
|
|
||||||
};
|
|
||||||
|
|
||||||
template <typename OutputIt, typename Char, typename AF>
|
|
||||||
void basic_printf_context<OutputIt, Char, AF>::parse_flags(
|
|
||||||
format_specs &spec, iterator &it) {
|
|
||||||
for (;;) {
|
|
||||||
switch (*it++) {
|
|
||||||
case '-':
|
|
||||||
spec.align_ = ALIGN_LEFT;
|
|
||||||
break;
|
|
||||||
case '+':
|
|
||||||
spec.flags |= SIGN_FLAG | PLUS_FLAG;
|
|
||||||
break;
|
|
||||||
case '0':
|
|
||||||
spec.fill_ = '0';
|
|
||||||
break;
|
|
||||||
case ' ':
|
|
||||||
spec.flags |= SIGN_FLAG;
|
|
||||||
break;
|
|
||||||
case '#':
|
|
||||||
spec.flags |= HASH_FLAG;
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
--it;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename OutputIt, typename Char, typename AF>
|
|
||||||
typename basic_printf_context<OutputIt, Char, AF>::format_arg
|
|
||||||
basic_printf_context<OutputIt, Char, AF>::get_arg(
|
|
||||||
iterator it, unsigned arg_index) {
|
|
||||||
(void)it;
|
|
||||||
if (arg_index == std::numeric_limits<unsigned>::max())
|
|
||||||
return this->do_get_arg(this->parse_context().next_arg_id());
|
|
||||||
return base::get_arg(arg_index - 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename OutputIt, typename Char, typename AF>
|
|
||||||
unsigned basic_printf_context<OutputIt, Char, AF>::parse_header(
|
|
||||||
iterator &it, format_specs &spec) {
|
|
||||||
unsigned arg_index = std::numeric_limits<unsigned>::max();
|
|
||||||
char_type c = *it;
|
|
||||||
if (c >= '0' && c <= '9') {
|
|
||||||
// Parse an argument index (if followed by '$') or a width possibly
|
|
||||||
// preceded with '0' flag(s).
|
|
||||||
internal::error_handler eh;
|
|
||||||
unsigned value = parse_nonnegative_int(it, eh);
|
|
||||||
if (*it == '$') { // value is an argument index
|
|
||||||
++it;
|
|
||||||
arg_index = value;
|
|
||||||
} else {
|
|
||||||
if (c == '0')
|
|
||||||
spec.fill_ = '0';
|
|
||||||
if (value != 0) {
|
|
||||||
// Nonzero value means that we parsed width and don't need to
|
|
||||||
// parse it or flags again, so return now.
|
|
||||||
spec.width_ = value;
|
|
||||||
return arg_index;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
parse_flags(spec, it);
|
|
||||||
// Parse width.
|
|
||||||
if (*it >= '0' && *it <= '9') {
|
|
||||||
internal::error_handler eh;
|
|
||||||
spec.width_ = parse_nonnegative_int(it, eh);
|
|
||||||
} else if (*it == '*') {
|
|
||||||
++it;
|
|
||||||
spec.width_ = visit_format_arg(
|
|
||||||
internal::printf_width_handler<char_type>(spec), get_arg(it));
|
|
||||||
}
|
|
||||||
return arg_index;
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename OutputIt, typename Char, typename AF>
|
|
||||||
void basic_printf_context<OutputIt, Char, AF>::format() {
|
|
||||||
auto &buffer = internal::get_container(this->out());
|
|
||||||
auto start = iterator(this->parse_context());
|
|
||||||
auto it = start;
|
|
||||||
using internal::pointer_from;
|
|
||||||
while (*it) {
|
|
||||||
char_type c = *it++;
|
|
||||||
if (c != '%') continue;
|
|
||||||
if (*it == c) {
|
|
||||||
buffer.append(pointer_from(start), pointer_from(it));
|
|
||||||
start = ++it;
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
buffer.append(pointer_from(start), pointer_from(it) - 1);
|
|
||||||
|
|
||||||
format_specs spec;
|
|
||||||
spec.align_ = ALIGN_RIGHT;
|
|
||||||
|
|
||||||
// Parse argument index, flags and width.
|
|
||||||
unsigned arg_index = parse_header(it, spec);
|
|
||||||
|
|
||||||
// Parse precision.
|
|
||||||
if (*it == '.') {
|
|
||||||
++it;
|
|
||||||
if ('0' <= *it && *it <= '9') {
|
|
||||||
internal::error_handler eh;
|
|
||||||
spec.precision = static_cast<int>(parse_nonnegative_int(it, eh));
|
|
||||||
} else if (*it == '*') {
|
|
||||||
++it;
|
|
||||||
spec.precision =
|
|
||||||
visit_format_arg(internal::printf_precision_handler(), get_arg(it));
|
|
||||||
} else {
|
|
||||||
spec.precision = 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
format_arg arg = get_arg(it, arg_index);
|
|
||||||
if (spec.has(HASH_FLAG) && visit_format_arg(internal::is_zero_int(), arg))
|
|
||||||
spec.flags = static_cast<uint_least8_t>(spec.flags & (~internal::to_unsigned<int>(HASH_FLAG)));
|
|
||||||
if (spec.fill_ == '0') {
|
|
||||||
if (arg.is_arithmetic())
|
|
||||||
spec.align_ = ALIGN_NUMERIC;
|
|
||||||
else
|
|
||||||
spec.fill_ = ' '; // Ignore '0' flag for non-numeric types.
|
|
||||||
}
|
|
||||||
|
|
||||||
// Parse length and convert the argument to the required type.
|
|
||||||
using internal::convert_arg;
|
|
||||||
switch (*it++) {
|
|
||||||
case 'h':
|
|
||||||
if (*it == 'h')
|
|
||||||
convert_arg<signed char>(arg, *++it);
|
|
||||||
else
|
|
||||||
convert_arg<short>(arg, *it);
|
|
||||||
break;
|
|
||||||
case 'l':
|
|
||||||
if (*it == 'l')
|
|
||||||
convert_arg<long long>(arg, *++it);
|
|
||||||
else
|
|
||||||
convert_arg<long>(arg, *it);
|
|
||||||
break;
|
|
||||||
case 'j':
|
|
||||||
convert_arg<intmax_t>(arg, *it);
|
|
||||||
break;
|
|
||||||
case 'z':
|
|
||||||
convert_arg<std::size_t>(arg, *it);
|
|
||||||
break;
|
|
||||||
case 't':
|
|
||||||
convert_arg<std::ptrdiff_t>(arg, *it);
|
|
||||||
break;
|
|
||||||
case 'L':
|
|
||||||
// printf produces garbage when 'L' is omitted for long double, no
|
|
||||||
// need to do the same.
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
--it;
|
|
||||||
convert_arg<void>(arg, *it);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Parse type.
|
|
||||||
if (!*it)
|
|
||||||
FMT_THROW(format_error("invalid format string"));
|
|
||||||
spec.type = static_cast<char>(*it++);
|
|
||||||
if (arg.is_integral()) {
|
|
||||||
// Normalize type.
|
|
||||||
switch (spec.type) {
|
|
||||||
case 'i': case 'u':
|
|
||||||
spec.type = 'd';
|
|
||||||
break;
|
|
||||||
case 'c':
|
|
||||||
// TODO: handle wchar_t better?
|
|
||||||
visit_format_arg(
|
|
||||||
internal::char_converter<basic_printf_context>(arg), arg);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
start = it;
|
|
||||||
|
|
||||||
// Format argument.
|
|
||||||
visit_format_arg(AF(buffer, spec, *this), arg);
|
|
||||||
}
|
|
||||||
buffer.append(pointer_from(start), pointer_from(it));
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename Buffer>
|
|
||||||
struct basic_printf_context_t {
|
|
||||||
typedef basic_printf_context<
|
|
||||||
std::back_insert_iterator<Buffer>, typename Buffer::value_type> type;
|
|
||||||
};
|
|
||||||
|
|
||||||
typedef basic_printf_context_t<internal::buffer>::type printf_context;
|
|
||||||
typedef basic_printf_context_t<internal::wbuffer>::type wprintf_context;
|
|
||||||
|
|
||||||
typedef basic_format_args<printf_context> printf_args;
|
|
||||||
typedef basic_format_args<wprintf_context> wprintf_args;
|
|
||||||
|
|
||||||
/**
|
|
||||||
\rst
|
|
||||||
Constructs an `~fmt::format_arg_store` object that contains references to
|
|
||||||
arguments and can be implicitly converted to `~fmt::printf_args`.
|
|
||||||
\endrst
|
|
||||||
*/
|
|
||||||
template<typename... Args>
|
|
||||||
inline format_arg_store<printf_context, Args...>
|
|
||||||
make_printf_args(const Args &... args) { return {args...}; }
|
|
||||||
|
|
||||||
/**
|
|
||||||
\rst
|
|
||||||
Constructs an `~fmt::format_arg_store` object that contains references to
|
|
||||||
arguments and can be implicitly converted to `~fmt::wprintf_args`.
|
|
||||||
\endrst
|
|
||||||
*/
|
|
||||||
template<typename... Args>
|
|
||||||
inline format_arg_store<wprintf_context, Args...>
|
|
||||||
make_wprintf_args(const Args &... args) { return {args...}; }
|
|
||||||
|
|
||||||
template <typename S, typename Char = FMT_CHAR(S)>
|
|
||||||
inline std::basic_string<Char>
|
|
||||||
vsprintf(const S &format,
|
|
||||||
basic_format_args<typename basic_printf_context_t<
|
|
||||||
internal::basic_buffer<Char>>::type> args) {
|
|
||||||
basic_memory_buffer<Char> buffer;
|
|
||||||
printf(buffer, to_string_view(format), args);
|
|
||||||
return to_string(buffer);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
\rst
|
|
||||||
Formats arguments and returns the result as a string.
|
|
||||||
|
|
||||||
**Example**::
|
|
||||||
|
|
||||||
std::string message = fmt::sprintf("The answer is %d", 42);
|
|
||||||
\endrst
|
|
||||||
*/
|
|
||||||
template <typename S, typename... Args>
|
|
||||||
inline FMT_ENABLE_IF_T(
|
|
||||||
internal::is_string<S>::value, std::basic_string<FMT_CHAR(S)>)
|
|
||||||
sprintf(const S &format, const Args & ... args) {
|
|
||||||
internal::check_format_string<Args...>(format);
|
|
||||||
typedef internal::basic_buffer<FMT_CHAR(S)> buffer;
|
|
||||||
typedef typename basic_printf_context_t<buffer>::type context;
|
|
||||||
format_arg_store<context, Args...> as{ args... };
|
|
||||||
return vsprintf(to_string_view(format),
|
|
||||||
basic_format_args<context>(as));
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename S, typename Char = FMT_CHAR(S)>
|
|
||||||
inline int vfprintf(std::FILE *f, const S &format,
|
|
||||||
basic_format_args<typename basic_printf_context_t<
|
|
||||||
internal::basic_buffer<Char>>::type> args) {
|
|
||||||
basic_memory_buffer<Char> buffer;
|
|
||||||
printf(buffer, to_string_view(format), args);
|
|
||||||
std::size_t size = buffer.size();
|
|
||||||
return std::fwrite(
|
|
||||||
buffer.data(), sizeof(Char), size, f) < size ? -1 : static_cast<int>(size);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
\rst
|
|
||||||
Prints formatted data to the file *f*.
|
|
||||||
|
|
||||||
**Example**::
|
|
||||||
|
|
||||||
fmt::fprintf(stderr, "Don't %s!", "panic");
|
|
||||||
\endrst
|
|
||||||
*/
|
|
||||||
template <typename S, typename... Args>
|
|
||||||
inline FMT_ENABLE_IF_T(internal::is_string<S>::value, int)
|
|
||||||
fprintf(std::FILE *f, const S &format, const Args & ... args) {
|
|
||||||
internal::check_format_string<Args...>(format);
|
|
||||||
typedef internal::basic_buffer<FMT_CHAR(S)> buffer;
|
|
||||||
typedef typename basic_printf_context_t<buffer>::type context;
|
|
||||||
format_arg_store<context, Args...> as{ args... };
|
|
||||||
return vfprintf(f, to_string_view(format),
|
|
||||||
basic_format_args<context>(as));
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename S, typename Char = FMT_CHAR(S)>
|
|
||||||
inline int vprintf(const S &format,
|
|
||||||
basic_format_args<typename basic_printf_context_t<
|
|
||||||
internal::basic_buffer<Char>>::type> args) {
|
|
||||||
return vfprintf(stdout, to_string_view(format), args);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
\rst
|
|
||||||
Prints formatted data to ``stdout``.
|
|
||||||
|
|
||||||
**Example**::
|
|
||||||
|
|
||||||
fmt::printf("Elapsed time: %.2f seconds", 1.23);
|
|
||||||
\endrst
|
|
||||||
*/
|
|
||||||
template <typename S, typename... Args>
|
|
||||||
inline FMT_ENABLE_IF_T(internal::is_string<S>::value, int)
|
|
||||||
printf(const S &format_str, const Args & ... args) {
|
|
||||||
internal::check_format_string<Args...>(format_str);
|
|
||||||
typedef internal::basic_buffer<FMT_CHAR(S)> buffer;
|
|
||||||
typedef typename basic_printf_context_t<buffer>::type context;
|
|
||||||
format_arg_store<context, Args...> as{ args... };
|
|
||||||
return vprintf(to_string_view(format_str),
|
|
||||||
basic_format_args<context>(as));
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename S, typename Char = FMT_CHAR(S)>
|
|
||||||
inline int vfprintf(std::basic_ostream<Char> &os,
|
|
||||||
const S &format,
|
|
||||||
basic_format_args<typename basic_printf_context_t<
|
|
||||||
internal::basic_buffer<Char>>::type> args) {
|
|
||||||
basic_memory_buffer<Char> buffer;
|
|
||||||
printf(buffer, to_string_view(format), args);
|
|
||||||
internal::write(os, buffer);
|
|
||||||
return static_cast<int>(buffer.size());
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
\rst
|
|
||||||
Prints formatted data to the stream *os*.
|
|
||||||
|
|
||||||
**Example**::
|
|
||||||
|
|
||||||
fmt::fprintf(cerr, "Don't %s!", "panic");
|
|
||||||
\endrst
|
|
||||||
*/
|
|
||||||
template <typename S, typename... Args>
|
|
||||||
inline FMT_ENABLE_IF_T(internal::is_string<S>::value, int)
|
|
||||||
fprintf(std::basic_ostream<FMT_CHAR(S)> &os,
|
|
||||||
const S &format_str, const Args & ... args) {
|
|
||||||
internal::check_format_string<Args...>(format_str);
|
|
||||||
typedef internal::basic_buffer<FMT_CHAR(S)> buffer;
|
|
||||||
typedef typename basic_printf_context_t<buffer>::type context;
|
|
||||||
format_arg_store<context, Args...> as{ args... };
|
|
||||||
return vfprintf(os, to_string_view(format_str),
|
|
||||||
basic_format_args<context>(as));
|
|
||||||
}
|
|
||||||
FMT_END_NAMESPACE
|
|
||||||
|
|
||||||
#endif // FMT_PRINTF_H_
|
|
||||||
308
deps/spdlog/fmt/bundled/ranges.h
vendored
308
deps/spdlog/fmt/bundled/ranges.h
vendored
@ -1,308 +0,0 @@
|
|||||||
// Formatting library for C++ - the core API
|
|
||||||
//
|
|
||||||
// Copyright (c) 2012 - present, Victor Zverovich
|
|
||||||
// All rights reserved.
|
|
||||||
//
|
|
||||||
// For the license information refer to format.h.
|
|
||||||
//
|
|
||||||
// Copyright (c) 2018 - present, Remotion (Igor Schulz)
|
|
||||||
// All Rights Reserved
|
|
||||||
// {fmt} support for ranges, containers and types tuple interface.
|
|
||||||
|
|
||||||
#ifndef FMT_RANGES_H_
|
|
||||||
#define FMT_RANGES_H_
|
|
||||||
|
|
||||||
#include "format.h"
|
|
||||||
#include <type_traits>
|
|
||||||
|
|
||||||
// output only up to N items from the range.
|
|
||||||
#ifndef FMT_RANGE_OUTPUT_LENGTH_LIMIT
|
|
||||||
# define FMT_RANGE_OUTPUT_LENGTH_LIMIT 256
|
|
||||||
#endif
|
|
||||||
|
|
||||||
FMT_BEGIN_NAMESPACE
|
|
||||||
|
|
||||||
template <typename Char>
|
|
||||||
struct formatting_base {
|
|
||||||
template <typename ParseContext>
|
|
||||||
FMT_CONSTEXPR auto parse(ParseContext &ctx) -> decltype(ctx.begin()) {
|
|
||||||
return ctx.begin();
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
template <typename Char, typename Enable = void>
|
|
||||||
struct formatting_range : formatting_base<Char> {
|
|
||||||
static FMT_CONSTEXPR_DECL const std::size_t range_length_limit =
|
|
||||||
FMT_RANGE_OUTPUT_LENGTH_LIMIT; // output only up to N items from the range.
|
|
||||||
Char prefix;
|
|
||||||
Char delimiter;
|
|
||||||
Char postfix;
|
|
||||||
formatting_range() : prefix('{'), delimiter(','), postfix('}') {}
|
|
||||||
static FMT_CONSTEXPR_DECL const bool add_delimiter_spaces = true;
|
|
||||||
static FMT_CONSTEXPR_DECL const bool add_prepostfix_space = false;
|
|
||||||
};
|
|
||||||
|
|
||||||
template <typename Char, typename Enable = void>
|
|
||||||
struct formatting_tuple : formatting_base<Char> {
|
|
||||||
Char prefix;
|
|
||||||
Char delimiter;
|
|
||||||
Char postfix;
|
|
||||||
formatting_tuple() : prefix('('), delimiter(','), postfix(')') {}
|
|
||||||
static FMT_CONSTEXPR_DECL const bool add_delimiter_spaces = true;
|
|
||||||
static FMT_CONSTEXPR_DECL const bool add_prepostfix_space = false;
|
|
||||||
};
|
|
||||||
|
|
||||||
namespace internal {
|
|
||||||
|
|
||||||
template <typename RangeT, typename OutputIterator>
|
|
||||||
void copy(const RangeT &range, OutputIterator out) {
|
|
||||||
for (auto it = range.begin(), end = range.end(); it != end; ++it)
|
|
||||||
*out++ = *it;
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename OutputIterator>
|
|
||||||
void copy(const char *str, OutputIterator out) {
|
|
||||||
const char *p_curr = str;
|
|
||||||
while (*p_curr) {
|
|
||||||
*out++ = *p_curr++;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename OutputIterator>
|
|
||||||
void copy(char ch, OutputIterator out) {
|
|
||||||
*out++ = ch;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Return true value if T has std::string interface, like std::string_view.
|
|
||||||
template <typename T>
|
|
||||||
class is_like_std_string {
|
|
||||||
template <typename U>
|
|
||||||
static auto check(U *p) ->
|
|
||||||
decltype(p->find('a'), p->length(), p->data(), int());
|
|
||||||
template <typename>
|
|
||||||
static void check(...);
|
|
||||||
|
|
||||||
public:
|
|
||||||
static FMT_CONSTEXPR_DECL const bool value =
|
|
||||||
!std::is_void<decltype(check<T>(FMT_NULL))>::value;
|
|
||||||
};
|
|
||||||
|
|
||||||
template <typename Char>
|
|
||||||
struct is_like_std_string<fmt::basic_string_view<Char>> : std::true_type {};
|
|
||||||
|
|
||||||
template <typename... Ts>
|
|
||||||
struct conditional_helper {};
|
|
||||||
|
|
||||||
template <typename T, typename _ = void>
|
|
||||||
struct is_range_ : std::false_type {};
|
|
||||||
|
|
||||||
#if !FMT_MSC_VER || FMT_MSC_VER > 1800
|
|
||||||
template <typename T>
|
|
||||||
struct is_range_<T, typename std::conditional<
|
|
||||||
false,
|
|
||||||
conditional_helper<decltype(internal::declval<T>().begin()),
|
|
||||||
decltype(internal::declval<T>().end())>,
|
|
||||||
void>::type> : std::true_type {};
|
|
||||||
#endif
|
|
||||||
|
|
||||||
/// tuple_size and tuple_element check.
|
|
||||||
template <typename T>
|
|
||||||
class is_tuple_like_ {
|
|
||||||
template <typename U>
|
|
||||||
static auto check(U *p) ->
|
|
||||||
decltype(std::tuple_size<U>::value,
|
|
||||||
internal::declval<typename std::tuple_element<0, U>::type>(), int());
|
|
||||||
template <typename>
|
|
||||||
static void check(...);
|
|
||||||
|
|
||||||
public:
|
|
||||||
static FMT_CONSTEXPR_DECL const bool value =
|
|
||||||
!std::is_void<decltype(check<T>(FMT_NULL))>::value;
|
|
||||||
};
|
|
||||||
|
|
||||||
// Check for integer_sequence
|
|
||||||
#if defined(__cpp_lib_integer_sequence) || FMT_MSC_VER >= 1900
|
|
||||||
template <typename T, T... N>
|
|
||||||
using integer_sequence = std::integer_sequence<T, N...>;
|
|
||||||
template <std::size_t... N>
|
|
||||||
using index_sequence = std::index_sequence<N...>;
|
|
||||||
template <std::size_t N>
|
|
||||||
using make_index_sequence = std::make_index_sequence<N>;
|
|
||||||
#else
|
|
||||||
template <typename T, T... N>
|
|
||||||
struct integer_sequence {
|
|
||||||
typedef T value_type;
|
|
||||||
|
|
||||||
static FMT_CONSTEXPR std::size_t size() {
|
|
||||||
return sizeof...(N);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
template <std::size_t... N>
|
|
||||||
using index_sequence = integer_sequence<std::size_t, N...>;
|
|
||||||
|
|
||||||
template <typename T, std::size_t N, T... Ns>
|
|
||||||
struct make_integer_sequence : make_integer_sequence<T, N - 1, N - 1, Ns...> {};
|
|
||||||
template <typename T, T... Ns>
|
|
||||||
struct make_integer_sequence<T, 0, Ns...> : integer_sequence<T, Ns...> {};
|
|
||||||
|
|
||||||
template <std::size_t N>
|
|
||||||
using make_index_sequence = make_integer_sequence<std::size_t, N>;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
template <class Tuple, class F, size_t... Is>
|
|
||||||
void for_each(index_sequence<Is...>, Tuple &&tup, F &&f) FMT_NOEXCEPT {
|
|
||||||
using std::get;
|
|
||||||
// using free function get<I>(T) now.
|
|
||||||
const int _[] = {0, ((void)f(get<Is>(tup)), 0)...};
|
|
||||||
(void)_; // blocks warnings
|
|
||||||
}
|
|
||||||
|
|
||||||
template <class T>
|
|
||||||
FMT_CONSTEXPR make_index_sequence<std::tuple_size<T>::value>
|
|
||||||
get_indexes(T const &) { return {}; }
|
|
||||||
|
|
||||||
template <class Tuple, class F>
|
|
||||||
void for_each(Tuple &&tup, F &&f) {
|
|
||||||
const auto indexes = get_indexes(tup);
|
|
||||||
for_each(indexes, std::forward<Tuple>(tup), std::forward<F>(f));
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename Arg>
|
|
||||||
FMT_CONSTEXPR const char* format_str_quoted(bool add_space, const Arg&,
|
|
||||||
typename std::enable_if<
|
|
||||||
!is_like_std_string<typename std::decay<Arg>::type>::value>::type* = nullptr) {
|
|
||||||
return add_space ? " {}" : "{}";
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename Arg>
|
|
||||||
FMT_CONSTEXPR const char* format_str_quoted(bool add_space, const Arg&,
|
|
||||||
typename std::enable_if<
|
|
||||||
is_like_std_string<typename std::decay<Arg>::type>::value>::type* = nullptr) {
|
|
||||||
return add_space ? " \"{}\"" : "\"{}\"";
|
|
||||||
}
|
|
||||||
|
|
||||||
FMT_CONSTEXPR const char* format_str_quoted(bool add_space, const char*) {
|
|
||||||
return add_space ? " \"{}\"" : "\"{}\"";
|
|
||||||
}
|
|
||||||
FMT_CONSTEXPR const wchar_t* format_str_quoted(bool add_space, const wchar_t*) {
|
|
||||||
return add_space ? L" \"{}\"" : L"\"{}\"";
|
|
||||||
}
|
|
||||||
|
|
||||||
FMT_CONSTEXPR const char* format_str_quoted(bool add_space, const char) {
|
|
||||||
return add_space ? " '{}'" : "'{}'";
|
|
||||||
}
|
|
||||||
FMT_CONSTEXPR const wchar_t* format_str_quoted(bool add_space, const wchar_t) {
|
|
||||||
return add_space ? L" '{}'" : L"'{}'";
|
|
||||||
}
|
|
||||||
|
|
||||||
} // namespace internal
|
|
||||||
|
|
||||||
template <typename T>
|
|
||||||
struct is_tuple_like {
|
|
||||||
static FMT_CONSTEXPR_DECL const bool value =
|
|
||||||
internal::is_tuple_like_<T>::value && !internal::is_range_<T>::value;
|
|
||||||
};
|
|
||||||
|
|
||||||
template <typename TupleT, typename Char>
|
|
||||||
struct formatter<TupleT, Char,
|
|
||||||
typename std::enable_if<fmt::is_tuple_like<TupleT>::value>::type> {
|
|
||||||
private:
|
|
||||||
// C++11 generic lambda for format()
|
|
||||||
template <typename FormatContext>
|
|
||||||
struct format_each {
|
|
||||||
template <typename T>
|
|
||||||
void operator()(const T& v) {
|
|
||||||
if (i > 0) {
|
|
||||||
if (formatting.add_prepostfix_space) {
|
|
||||||
*out++ = ' ';
|
|
||||||
}
|
|
||||||
internal::copy(formatting.delimiter, out);
|
|
||||||
}
|
|
||||||
format_to(out,
|
|
||||||
internal::format_str_quoted(
|
|
||||||
(formatting.add_delimiter_spaces && i > 0), v),
|
|
||||||
v);
|
|
||||||
++i;
|
|
||||||
}
|
|
||||||
|
|
||||||
formatting_tuple<Char>& formatting;
|
|
||||||
std::size_t& i;
|
|
||||||
typename std::add_lvalue_reference<decltype(std::declval<FormatContext>().out())>::type out;
|
|
||||||
};
|
|
||||||
|
|
||||||
public:
|
|
||||||
formatting_tuple<Char> formatting;
|
|
||||||
|
|
||||||
template <typename ParseContext>
|
|
||||||
FMT_CONSTEXPR auto parse(ParseContext &ctx) -> decltype(ctx.begin()) {
|
|
||||||
return formatting.parse(ctx);
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename FormatContext = format_context>
|
|
||||||
auto format(const TupleT &values, FormatContext &ctx) -> decltype(ctx.out()) {
|
|
||||||
auto out = ctx.out();
|
|
||||||
std::size_t i = 0;
|
|
||||||
internal::copy(formatting.prefix, out);
|
|
||||||
|
|
||||||
internal::for_each(values, format_each<FormatContext>{formatting, i, out});
|
|
||||||
if (formatting.add_prepostfix_space) {
|
|
||||||
*out++ = ' ';
|
|
||||||
}
|
|
||||||
internal::copy(formatting.postfix, out);
|
|
||||||
|
|
||||||
return ctx.out();
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
template <typename T>
|
|
||||||
struct is_range {
|
|
||||||
static FMT_CONSTEXPR_DECL const bool value =
|
|
||||||
internal::is_range_<T>::value && !internal::is_like_std_string<T>::value;
|
|
||||||
};
|
|
||||||
|
|
||||||
template <typename RangeT, typename Char>
|
|
||||||
struct formatter<RangeT, Char,
|
|
||||||
typename std::enable_if<fmt::is_range<RangeT>::value>::type> {
|
|
||||||
|
|
||||||
formatting_range<Char> formatting;
|
|
||||||
|
|
||||||
template <typename ParseContext>
|
|
||||||
FMT_CONSTEXPR auto parse(ParseContext &ctx) -> decltype(ctx.begin()) {
|
|
||||||
return formatting.parse(ctx);
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename FormatContext>
|
|
||||||
typename FormatContext::iterator format(
|
|
||||||
const RangeT &values, FormatContext &ctx) {
|
|
||||||
auto out = ctx.out();
|
|
||||||
internal::copy(formatting.prefix, out);
|
|
||||||
std::size_t i = 0;
|
|
||||||
for (auto it = values.begin(), end = values.end(); it != end; ++it) {
|
|
||||||
if (i > 0) {
|
|
||||||
if (formatting.add_prepostfix_space) {
|
|
||||||
*out++ = ' ';
|
|
||||||
}
|
|
||||||
internal::copy(formatting.delimiter, out);
|
|
||||||
}
|
|
||||||
format_to(out,
|
|
||||||
internal::format_str_quoted(
|
|
||||||
(formatting.add_delimiter_spaces && i > 0), *it),
|
|
||||||
*it);
|
|
||||||
if (++i > formatting.range_length_limit) {
|
|
||||||
format_to(out, " ... <other elements>");
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (formatting.add_prepostfix_space) {
|
|
||||||
*out++ = ' ';
|
|
||||||
}
|
|
||||||
internal::copy(formatting.postfix, out);
|
|
||||||
return ctx.out();
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
FMT_END_NAMESPACE
|
|
||||||
|
|
||||||
#endif // FMT_RANGES_H_
|
|
||||||
|
|
||||||
160
deps/spdlog/fmt/bundled/time.h
vendored
160
deps/spdlog/fmt/bundled/time.h
vendored
@ -1,160 +0,0 @@
|
|||||||
// Formatting library for C++ - time formatting
|
|
||||||
//
|
|
||||||
// Copyright (c) 2012 - present, Victor Zverovich
|
|
||||||
// All rights reserved.
|
|
||||||
//
|
|
||||||
// For the license information refer to format.h.
|
|
||||||
|
|
||||||
#ifndef FMT_TIME_H_
|
|
||||||
#define FMT_TIME_H_
|
|
||||||
|
|
||||||
#include "format.h"
|
|
||||||
#include <ctime>
|
|
||||||
#include <locale>
|
|
||||||
|
|
||||||
FMT_BEGIN_NAMESPACE
|
|
||||||
|
|
||||||
// Prevents expansion of a preceding token as a function-style macro.
|
|
||||||
// Usage: f FMT_NOMACRO()
|
|
||||||
#define FMT_NOMACRO
|
|
||||||
|
|
||||||
namespace internal{
|
|
||||||
inline null<> localtime_r FMT_NOMACRO(...) { return null<>(); }
|
|
||||||
inline null<> localtime_s(...) { return null<>(); }
|
|
||||||
inline null<> gmtime_r(...) { return null<>(); }
|
|
||||||
inline null<> gmtime_s(...) { return null<>(); }
|
|
||||||
} // namespace internal
|
|
||||||
|
|
||||||
// Thread-safe replacement for std::localtime
|
|
||||||
inline std::tm localtime(std::time_t time) {
|
|
||||||
struct dispatcher {
|
|
||||||
std::time_t time_;
|
|
||||||
std::tm tm_;
|
|
||||||
|
|
||||||
dispatcher(std::time_t t): time_(t) {}
|
|
||||||
|
|
||||||
bool run() {
|
|
||||||
using namespace fmt::internal;
|
|
||||||
return handle(localtime_r(&time_, &tm_));
|
|
||||||
}
|
|
||||||
|
|
||||||
bool handle(std::tm *tm) { return tm != FMT_NULL; }
|
|
||||||
|
|
||||||
bool handle(internal::null<>) {
|
|
||||||
using namespace fmt::internal;
|
|
||||||
return fallback(localtime_s(&tm_, &time_));
|
|
||||||
}
|
|
||||||
|
|
||||||
bool fallback(int res) { return res == 0; }
|
|
||||||
|
|
||||||
#if !FMT_MSC_VER
|
|
||||||
bool fallback(internal::null<>) {
|
|
||||||
using namespace fmt::internal;
|
|
||||||
std::tm *tm = std::localtime(&time_);
|
|
||||||
if (tm) tm_ = *tm;
|
|
||||||
return tm != FMT_NULL;
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
};
|
|
||||||
dispatcher lt(time);
|
|
||||||
// Too big time values may be unsupported.
|
|
||||||
if (!lt.run())
|
|
||||||
FMT_THROW(format_error("time_t value out of range"));
|
|
||||||
return lt.tm_;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Thread-safe replacement for std::gmtime
|
|
||||||
inline std::tm gmtime(std::time_t time) {
|
|
||||||
struct dispatcher {
|
|
||||||
std::time_t time_;
|
|
||||||
std::tm tm_;
|
|
||||||
|
|
||||||
dispatcher(std::time_t t): time_(t) {}
|
|
||||||
|
|
||||||
bool run() {
|
|
||||||
using namespace fmt::internal;
|
|
||||||
return handle(gmtime_r(&time_, &tm_));
|
|
||||||
}
|
|
||||||
|
|
||||||
bool handle(std::tm *tm) { return tm != FMT_NULL; }
|
|
||||||
|
|
||||||
bool handle(internal::null<>) {
|
|
||||||
using namespace fmt::internal;
|
|
||||||
return fallback(gmtime_s(&tm_, &time_));
|
|
||||||
}
|
|
||||||
|
|
||||||
bool fallback(int res) { return res == 0; }
|
|
||||||
|
|
||||||
#if !FMT_MSC_VER
|
|
||||||
bool fallback(internal::null<>) {
|
|
||||||
std::tm *tm = std::gmtime(&time_);
|
|
||||||
if (tm) tm_ = *tm;
|
|
||||||
return tm != FMT_NULL;
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
};
|
|
||||||
dispatcher gt(time);
|
|
||||||
// Too big time values may be unsupported.
|
|
||||||
if (!gt.run())
|
|
||||||
FMT_THROW(format_error("time_t value out of range"));
|
|
||||||
return gt.tm_;
|
|
||||||
}
|
|
||||||
|
|
||||||
namespace internal {
|
|
||||||
inline std::size_t strftime(char *str, std::size_t count, const char *format,
|
|
||||||
const std::tm *time) {
|
|
||||||
return std::strftime(str, count, format, time);
|
|
||||||
}
|
|
||||||
|
|
||||||
inline std::size_t strftime(wchar_t *str, std::size_t count,
|
|
||||||
const wchar_t *format, const std::tm *time) {
|
|
||||||
return std::wcsftime(str, count, format, time);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename Char>
|
|
||||||
struct formatter<std::tm, Char> {
|
|
||||||
template <typename ParseContext>
|
|
||||||
auto parse(ParseContext &ctx) -> decltype(ctx.begin()) {
|
|
||||||
auto it = ctx.begin();
|
|
||||||
if (it != ctx.end() && *it == ':')
|
|
||||||
++it;
|
|
||||||
auto end = it;
|
|
||||||
while (end != ctx.end() && *end != '}')
|
|
||||||
++end;
|
|
||||||
tm_format.reserve(internal::to_unsigned(end - it + 1));
|
|
||||||
tm_format.append(it, end);
|
|
||||||
tm_format.push_back('\0');
|
|
||||||
return end;
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename FormatContext>
|
|
||||||
auto format(const std::tm &tm, FormatContext &ctx) -> decltype(ctx.out()) {
|
|
||||||
basic_memory_buffer<Char> buf;
|
|
||||||
std::size_t start = buf.size();
|
|
||||||
for (;;) {
|
|
||||||
std::size_t size = buf.capacity() - start;
|
|
||||||
std::size_t count =
|
|
||||||
internal::strftime(&buf[start], size, &tm_format[0], &tm);
|
|
||||||
if (count != 0) {
|
|
||||||
buf.resize(start + count);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
if (size >= tm_format.size() * 256) {
|
|
||||||
// If the buffer is 256 times larger than the format string, assume
|
|
||||||
// that `strftime` gives an empty result. There doesn't seem to be a
|
|
||||||
// better way to distinguish the two cases:
|
|
||||||
// https://github.com/fmtlib/fmt/issues/367
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
const std::size_t MIN_GROWTH = 10;
|
|
||||||
buf.reserve(buf.capacity() + (size > MIN_GROWTH ? size : MIN_GROWTH));
|
|
||||||
}
|
|
||||||
return std::copy(buf.begin(), buf.end(), ctx.out());
|
|
||||||
}
|
|
||||||
|
|
||||||
basic_memory_buffer<Char> tm_format;
|
|
||||||
};
|
|
||||||
FMT_END_NAMESPACE
|
|
||||||
|
|
||||||
#endif // FMT_TIME_H_
|
|
||||||
25
deps/spdlog/fmt/fmt.h
vendored
25
deps/spdlog/fmt/fmt.h
vendored
@ -1,25 +0,0 @@
|
|||||||
//
|
|
||||||
// Copyright(c) 2016-2018 Gabi Melman.
|
|
||||||
// Distributed under the MIT License (http://opensource.org/licenses/MIT)
|
|
||||||
//
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
//
|
|
||||||
// Include a bundled header-only copy of fmtlib or an external one.
|
|
||||||
// By default spdlog include its own copy.
|
|
||||||
//
|
|
||||||
|
|
||||||
#if !defined(SPDLOG_FMT_EXTERNAL)
|
|
||||||
#ifndef FMT_HEADER_ONLY
|
|
||||||
#define FMT_HEADER_ONLY
|
|
||||||
#endif
|
|
||||||
#ifndef FMT_USE_WINDOWS_H
|
|
||||||
#define FMT_USE_WINDOWS_H 0
|
|
||||||
#endif
|
|
||||||
#include "bundled/core.h"
|
|
||||||
#include "bundled/format.h"
|
|
||||||
#else // external fmtlib
|
|
||||||
#include <fmt/core.h>
|
|
||||||
#include <fmt/format.h>
|
|
||||||
#endif
|
|
||||||
18
deps/spdlog/fmt/ostr.h
vendored
18
deps/spdlog/fmt/ostr.h
vendored
@ -1,18 +0,0 @@
|
|||||||
//
|
|
||||||
// Copyright(c) 2016 Gabi Melman.
|
|
||||||
// Distributed under the MIT License (http://opensource.org/licenses/MIT)
|
|
||||||
//
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
//
|
|
||||||
// include bundled or external copy of fmtlib's ostream support
|
|
||||||
//
|
|
||||||
#if !defined(SPDLOG_FMT_EXTERNAL)
|
|
||||||
#ifndef FMT_HEADER_ONLY
|
|
||||||
#define FMT_HEADER_ONLY
|
|
||||||
#endif
|
|
||||||
#include "bundled/ostream.h"
|
|
||||||
#include "fmt.h"
|
|
||||||
#else
|
|
||||||
#include <fmt/ostream.h>
|
|
||||||
#endif
|
|
||||||
20
deps/spdlog/formatter.h
vendored
20
deps/spdlog/formatter.h
vendored
@ -1,20 +0,0 @@
|
|||||||
//
|
|
||||||
// Copyright(c) 2015 Gabi Melman.
|
|
||||||
// Distributed under the MIT License (http://opensource.org/licenses/MIT)
|
|
||||||
//
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include "fmt/fmt.h"
|
|
||||||
#include "spdlog/details/log_msg.h"
|
|
||||||
|
|
||||||
namespace spdlog {
|
|
||||||
|
|
||||||
class formatter
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
virtual ~formatter() = default;
|
|
||||||
virtual void format(const details::log_msg &msg, fmt::memory_buffer &dest) = 0;
|
|
||||||
virtual std::unique_ptr<formatter> clone() const = 0;
|
|
||||||
};
|
|
||||||
} // namespace spdlog
|
|
||||||
188
deps/spdlog/logger.h
vendored
188
deps/spdlog/logger.h
vendored
@ -1,188 +0,0 @@
|
|||||||
//
|
|
||||||
// Copyright(c) 2015-2108 Gabi Melman.
|
|
||||||
// Distributed under the MIT License (http://opensource.org/licenses/MIT)
|
|
||||||
//
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
// Thread safe logger (except for set_pattern(..), set_formatter(..) and
|
|
||||||
// set_error_handler())
|
|
||||||
// Has name, log level, vector of std::shared sink pointers and formatter
|
|
||||||
// Upon each log write the logger:
|
|
||||||
// 1. Checks if its log level is enough to log the message and if yes:
|
|
||||||
// 2. Call the underlying sinks to do the job.
|
|
||||||
// 3. Each sink use its own private copy of a formatter to format the message
|
|
||||||
// and send to its destination.
|
|
||||||
//
|
|
||||||
// The use of private formatter per sink provides the opportunity to cache some
|
|
||||||
// formatted data,
|
|
||||||
// and support customize format per each sink.
|
|
||||||
|
|
||||||
#include "spdlog/common.h"
|
|
||||||
#include "spdlog/formatter.h"
|
|
||||||
#include "spdlog/sinks/sink.h"
|
|
||||||
|
|
||||||
#include <memory>
|
|
||||||
#include <string>
|
|
||||||
#include <vector>
|
|
||||||
|
|
||||||
namespace spdlog {
|
|
||||||
|
|
||||||
class logger
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
logger(std::string name, sink_ptr single_sink);
|
|
||||||
logger(std::string name, sinks_init_list sinks);
|
|
||||||
|
|
||||||
template<typename It>
|
|
||||||
logger(std::string name, It begin, It end);
|
|
||||||
|
|
||||||
virtual ~logger();
|
|
||||||
|
|
||||||
logger(const logger &) = delete;
|
|
||||||
logger &operator=(const logger &) = delete;
|
|
||||||
|
|
||||||
template<typename... Args>
|
|
||||||
void log(level::level_enum lvl, const char *fmt, const Args &... args);
|
|
||||||
|
|
||||||
template<typename... Args>
|
|
||||||
void log(source_loc loc, level::level_enum lvl, const char *fmt, const Args &... args);
|
|
||||||
|
|
||||||
void log(level::level_enum lvl, const char *msg);
|
|
||||||
|
|
||||||
void log(source_loc loc, level::level_enum lvl, const char *msg);
|
|
||||||
|
|
||||||
template<typename... Args>
|
|
||||||
void trace(const char *fmt, const Args &... args);
|
|
||||||
|
|
||||||
template<typename... Args>
|
|
||||||
void debug(const char *fmt, const Args &... args);
|
|
||||||
|
|
||||||
template<typename... Args>
|
|
||||||
void info(const char *fmt, const Args &... args);
|
|
||||||
|
|
||||||
template<typename... Args>
|
|
||||||
void warn(const char *fmt, const Args &... args);
|
|
||||||
|
|
||||||
template<typename... Args>
|
|
||||||
void error(const char *fmt, const Args &... args);
|
|
||||||
|
|
||||||
template<typename... Args>
|
|
||||||
void critical(const char *fmt, const Args &... args);
|
|
||||||
|
|
||||||
#ifdef SPDLOG_WCHAR_TO_UTF8_SUPPORT
|
|
||||||
#ifndef _WIN32
|
|
||||||
#error SPDLOG_WCHAR_TO_UTF8_SUPPORT only supported on windows
|
|
||||||
#else
|
|
||||||
template<typename... Args>
|
|
||||||
void log(level::level_enum lvl, const wchar_t *fmt, const Args &... args);
|
|
||||||
|
|
||||||
template<typename... Args>
|
|
||||||
void log(source_loc source, level::level_enum lvl, const wchar_t *fmt, const Args &... args);
|
|
||||||
|
|
||||||
template<typename... Args>
|
|
||||||
void trace(const wchar_t *fmt, const Args &... args);
|
|
||||||
|
|
||||||
template<typename... Args>
|
|
||||||
void debug(const wchar_t *fmt, const Args &... args);
|
|
||||||
|
|
||||||
template<typename... Args>
|
|
||||||
void info(const wchar_t *fmt, const Args &... args);
|
|
||||||
|
|
||||||
template<typename... Args>
|
|
||||||
void warn(const wchar_t *fmt, const Args &... args);
|
|
||||||
|
|
||||||
template<typename... Args>
|
|
||||||
void error(const wchar_t *fmt, const Args &... args);
|
|
||||||
|
|
||||||
template<typename... Args>
|
|
||||||
void critical(const wchar_t *fmt, const Args &... args);
|
|
||||||
#endif // _WIN32
|
|
||||||
#endif // SPDLOG_WCHAR_TO_UTF8_SUPPORT
|
|
||||||
|
|
||||||
// T can be statically converted to string_view
|
|
||||||
template<class T, typename std::enable_if<std::is_convertible<T, spdlog::string_view_t>::value, T>::type * = nullptr>
|
|
||||||
void log(level::level_enum lvl, const T &);
|
|
||||||
|
|
||||||
// T can be statically converted to string_view
|
|
||||||
template<class T, typename std::enable_if<std::is_convertible<T, spdlog::string_view_t>::value, T>::type * = nullptr>
|
|
||||||
void log(source_loc loc, level::level_enum lvl, const T &);
|
|
||||||
|
|
||||||
// T cannot be statically converted to string_view
|
|
||||||
template<class T, typename std::enable_if<!std::is_convertible<T, spdlog::string_view_t>::value, T>::type * = nullptr>
|
|
||||||
void log(level::level_enum lvl, const T &);
|
|
||||||
|
|
||||||
// T cannot be statically converted to string_view
|
|
||||||
template<class T, typename std::enable_if<!std::is_convertible<T, spdlog::string_view_t>::value, T>::type * = nullptr>
|
|
||||||
void log(source_loc loc, level::level_enum lvl, const T &);
|
|
||||||
|
|
||||||
template<typename T>
|
|
||||||
void trace(const T &msg);
|
|
||||||
|
|
||||||
template<typename T>
|
|
||||||
void debug(const T &msg);
|
|
||||||
|
|
||||||
template<typename T>
|
|
||||||
void info(const T &msg);
|
|
||||||
|
|
||||||
template<typename T>
|
|
||||||
void warn(const T &msg);
|
|
||||||
|
|
||||||
template<typename T>
|
|
||||||
void error(const T &msg);
|
|
||||||
|
|
||||||
template<typename T>
|
|
||||||
void critical(const T &msg);
|
|
||||||
|
|
||||||
bool should_log(level::level_enum msg_level) const;
|
|
||||||
void set_level(level::level_enum log_level);
|
|
||||||
|
|
||||||
static level::level_enum default_level();
|
|
||||||
level::level_enum level() const;
|
|
||||||
const std::string &name() const;
|
|
||||||
|
|
||||||
// set formatting for the sinks in this logger.
|
|
||||||
// each sink will get a seperate instance of the formatter object.
|
|
||||||
void set_formatter(std::unique_ptr<formatter> formatter);
|
|
||||||
void set_pattern(std::string pattern, pattern_time_type time_type = pattern_time_type::local);
|
|
||||||
|
|
||||||
// flush functions
|
|
||||||
void flush();
|
|
||||||
void flush_on(level::level_enum log_level);
|
|
||||||
level::level_enum flush_level() const;
|
|
||||||
|
|
||||||
// sinks
|
|
||||||
const std::vector<sink_ptr> &sinks() const;
|
|
||||||
std::vector<sink_ptr> &sinks();
|
|
||||||
|
|
||||||
// error handler
|
|
||||||
void set_error_handler(log_err_handler err_handler);
|
|
||||||
log_err_handler error_handler() const;
|
|
||||||
|
|
||||||
// create new logger with same sinks and configuration.
|
|
||||||
virtual std::shared_ptr<logger> clone(std::string logger_name);
|
|
||||||
|
|
||||||
protected:
|
|
||||||
virtual void sink_it_(details::log_msg &msg);
|
|
||||||
virtual void flush_();
|
|
||||||
|
|
||||||
bool should_flush_(const details::log_msg &msg);
|
|
||||||
|
|
||||||
// default error handler.
|
|
||||||
// print the error to stderr with the max rate of 1 message/minute.
|
|
||||||
void default_err_handler_(const std::string &msg);
|
|
||||||
|
|
||||||
// increment the message count (only if defined(SPDLOG_ENABLE_MESSAGE_COUNTER))
|
|
||||||
void incr_msg_counter_(details::log_msg &msg);
|
|
||||||
|
|
||||||
const std::string name_;
|
|
||||||
std::vector<sink_ptr> sinks_;
|
|
||||||
spdlog::level_t level_{spdlog::logger::default_level()};
|
|
||||||
spdlog::level_t flush_level_{level::off};
|
|
||||||
log_err_handler err_handler_{[this](const std::string &msg) { this->default_err_handler_(msg); }};
|
|
||||||
std::atomic<time_t> last_err_time_{0};
|
|
||||||
std::atomic<size_t> msg_counter_{1};
|
|
||||||
};
|
|
||||||
} // namespace spdlog
|
|
||||||
|
|
||||||
#include "details/logger_impl.h"
|
|
||||||
121
deps/spdlog/sinks/android_sink.h
vendored
121
deps/spdlog/sinks/android_sink.h
vendored
@ -1,121 +0,0 @@
|
|||||||
//
|
|
||||||
// Copyright(c) 2015 Gabi Melman.
|
|
||||||
// Distributed under the MIT License (http://opensource.org/licenses/MIT)
|
|
||||||
//
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#ifndef SPDLOG_H
|
|
||||||
#include "spdlog/spdlog.h"
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#include "spdlog/details/fmt_helper.h"
|
|
||||||
#include "spdlog/details/null_mutex.h"
|
|
||||||
#include "spdlog/details/os.h"
|
|
||||||
#include "spdlog/sinks/base_sink.h"
|
|
||||||
|
|
||||||
#include <android/log.h>
|
|
||||||
#include <chrono>
|
|
||||||
#include <mutex>
|
|
||||||
#include <string>
|
|
||||||
#include <thread>
|
|
||||||
|
|
||||||
#if !defined(SPDLOG_ANDROID_RETRIES)
|
|
||||||
#define SPDLOG_ANDROID_RETRIES 2
|
|
||||||
#endif
|
|
||||||
|
|
||||||
namespace spdlog {
|
|
||||||
namespace sinks {
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Android sink (logging using __android_log_write)
|
|
||||||
*/
|
|
||||||
template<typename Mutex>
|
|
||||||
class android_sink final : public base_sink<Mutex>
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
explicit android_sink(std::string tag = "spdlog", bool use_raw_msg = false)
|
|
||||||
: tag_(std::move(tag))
|
|
||||||
, use_raw_msg_(use_raw_msg)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
protected:
|
|
||||||
void sink_it_(const details::log_msg &msg) override
|
|
||||||
{
|
|
||||||
const android_LogPriority priority = convert_to_android_(msg.level);
|
|
||||||
fmt::memory_buffer formatted;
|
|
||||||
if (use_raw_msg_)
|
|
||||||
{
|
|
||||||
details::fmt_helper::append_string_view(msg.payload, formatted);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
sink::formatter_->format(msg, formatted);
|
|
||||||
}
|
|
||||||
formatted.push_back('\0');
|
|
||||||
const char *msg_output = formatted.data();
|
|
||||||
|
|
||||||
// See system/core/liblog/logger_write.c for explanation of return value
|
|
||||||
int ret = __android_log_write(priority, tag_.c_str(), msg_output);
|
|
||||||
int retry_count = 0;
|
|
||||||
while ((ret == -11 /*EAGAIN*/) && (retry_count < SPDLOG_ANDROID_RETRIES))
|
|
||||||
{
|
|
||||||
details::os::sleep_for_millis(5);
|
|
||||||
ret = __android_log_write(priority, tag_.c_str(), msg_output);
|
|
||||||
retry_count++;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (ret < 0)
|
|
||||||
{
|
|
||||||
throw spdlog_ex("__android_log_write() failed", ret);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void flush_() override {}
|
|
||||||
|
|
||||||
private:
|
|
||||||
static android_LogPriority convert_to_android_(spdlog::level::level_enum level)
|
|
||||||
{
|
|
||||||
switch (level)
|
|
||||||
{
|
|
||||||
case spdlog::level::trace:
|
|
||||||
return ANDROID_LOG_VERBOSE;
|
|
||||||
case spdlog::level::debug:
|
|
||||||
return ANDROID_LOG_DEBUG;
|
|
||||||
case spdlog::level::info:
|
|
||||||
return ANDROID_LOG_INFO;
|
|
||||||
case spdlog::level::warn:
|
|
||||||
return ANDROID_LOG_WARN;
|
|
||||||
case spdlog::level::err:
|
|
||||||
return ANDROID_LOG_ERROR;
|
|
||||||
case spdlog::level::critical:
|
|
||||||
return ANDROID_LOG_FATAL;
|
|
||||||
default:
|
|
||||||
return ANDROID_LOG_DEFAULT;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
std::string tag_;
|
|
||||||
bool use_raw_msg_;
|
|
||||||
};
|
|
||||||
|
|
||||||
using android_sink_mt = android_sink<std::mutex>;
|
|
||||||
using android_sink_st = android_sink<details::null_mutex>;
|
|
||||||
} // namespace sinks
|
|
||||||
|
|
||||||
// Create and register android syslog logger
|
|
||||||
|
|
||||||
template<typename Factory = default_factory>
|
|
||||||
inline std::shared_ptr<logger> android_logger_mt(const std::string &logger_name, const std::string &tag = "spdlog")
|
|
||||||
{
|
|
||||||
return Factory::template create<sinks::android_sink_mt>(logger_name, tag);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename Factory = default_factory>
|
|
||||||
inline std::shared_ptr<logger> android_logger_st(const std::string &logger_name, const std::string &tag = "spdlog")
|
|
||||||
{
|
|
||||||
return Factory::template create<sinks::android_sink_st>(logger_name, tag);
|
|
||||||
}
|
|
||||||
|
|
||||||
} // namespace spdlog
|
|
||||||
161
deps/spdlog/sinks/ansicolor_sink.h
vendored
161
deps/spdlog/sinks/ansicolor_sink.h
vendored
@ -1,161 +0,0 @@
|
|||||||
//
|
|
||||||
// Copyright(c) 2017 spdlog authors.
|
|
||||||
// Distributed under the MIT License (http://opensource.org/licenses/MIT)
|
|
||||||
//
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#ifndef SPDLOG_H
|
|
||||||
#include "spdlog/spdlog.h"
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#include "spdlog/details/console_globals.h"
|
|
||||||
#include "spdlog/details/null_mutex.h"
|
|
||||||
#include "spdlog/details/os.h"
|
|
||||||
#include "spdlog/sinks/sink.h"
|
|
||||||
|
|
||||||
#include <memory>
|
|
||||||
#include <mutex>
|
|
||||||
#include <string>
|
|
||||||
#include <unordered_map>
|
|
||||||
|
|
||||||
namespace spdlog {
|
|
||||||
namespace sinks {
|
|
||||||
|
|
||||||
/**
|
|
||||||
* This sink prefixes the output with an ANSI escape sequence color code
|
|
||||||
* depending on the severity
|
|
||||||
* of the message.
|
|
||||||
* If no color terminal detected, omit the escape codes.
|
|
||||||
*/
|
|
||||||
template<typename TargetStream, class ConsoleMutex>
|
|
||||||
class ansicolor_sink final : public sink
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
using mutex_t = typename ConsoleMutex::mutex_t;
|
|
||||||
ansicolor_sink()
|
|
||||||
: target_file_(TargetStream::stream())
|
|
||||||
, mutex_(ConsoleMutex::mutex())
|
|
||||||
|
|
||||||
{
|
|
||||||
should_do_colors_ = details::os::in_terminal(target_file_) && details::os::is_color_terminal();
|
|
||||||
colors_[level::trace] = white;
|
|
||||||
colors_[level::debug] = cyan;
|
|
||||||
colors_[level::info] = green;
|
|
||||||
colors_[level::warn] = yellow + bold;
|
|
||||||
colors_[level::err] = red + bold;
|
|
||||||
colors_[level::critical] = bold + on_red;
|
|
||||||
colors_[level::off] = reset;
|
|
||||||
}
|
|
||||||
|
|
||||||
~ansicolor_sink() override = default;
|
|
||||||
|
|
||||||
ansicolor_sink(const ansicolor_sink &other) = delete;
|
|
||||||
ansicolor_sink &operator=(const ansicolor_sink &other) = delete;
|
|
||||||
|
|
||||||
void set_color(level::level_enum color_level, const std::string &color)
|
|
||||||
{
|
|
||||||
std::lock_guard<mutex_t> lock(mutex_);
|
|
||||||
colors_[color_level] = color;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Formatting codes
|
|
||||||
const std::string reset = "\033[m";
|
|
||||||
const std::string bold = "\033[1m";
|
|
||||||
const std::string dark = "\033[2m";
|
|
||||||
const std::string underline = "\033[4m";
|
|
||||||
const std::string blink = "\033[5m";
|
|
||||||
const std::string reverse = "\033[7m";
|
|
||||||
const std::string concealed = "\033[8m";
|
|
||||||
const std::string clear_line = "\033[K";
|
|
||||||
|
|
||||||
// Foreground colors
|
|
||||||
const std::string black = "\033[30m";
|
|
||||||
const std::string red = "\033[31m";
|
|
||||||
const std::string green = "\033[32m";
|
|
||||||
const std::string yellow = "\033[33m";
|
|
||||||
const std::string blue = "\033[34m";
|
|
||||||
const std::string magenta = "\033[35m";
|
|
||||||
const std::string cyan = "\033[36m";
|
|
||||||
const std::string white = "\033[37m";
|
|
||||||
|
|
||||||
/// Background colors
|
|
||||||
const std::string on_black = "\033[40m";
|
|
||||||
const std::string on_red = "\033[41m";
|
|
||||||
const std::string on_green = "\033[42m";
|
|
||||||
const std::string on_yellow = "\033[43m";
|
|
||||||
const std::string on_blue = "\033[44m";
|
|
||||||
const std::string on_magenta = "\033[45m";
|
|
||||||
const std::string on_cyan = "\033[46m";
|
|
||||||
const std::string on_white = "\033[47m";
|
|
||||||
|
|
||||||
void log(const details::log_msg &msg) override
|
|
||||||
{
|
|
||||||
// Wrap the originally formatted message in color codes.
|
|
||||||
// If color is not supported in the terminal, log as is instead.
|
|
||||||
std::lock_guard<mutex_t> lock(mutex_);
|
|
||||||
|
|
||||||
fmt::memory_buffer formatted;
|
|
||||||
formatter_->format(msg, formatted);
|
|
||||||
if (should_do_colors_ && msg.color_range_end > msg.color_range_start)
|
|
||||||
{
|
|
||||||
// before color range
|
|
||||||
print_range_(formatted, 0, msg.color_range_start);
|
|
||||||
// in color range
|
|
||||||
print_ccode_(colors_[msg.level]);
|
|
||||||
print_range_(formatted, msg.color_range_start, msg.color_range_end);
|
|
||||||
print_ccode_(reset);
|
|
||||||
// after color range
|
|
||||||
print_range_(formatted, msg.color_range_end, formatted.size());
|
|
||||||
}
|
|
||||||
else // no color
|
|
||||||
{
|
|
||||||
print_range_(formatted, 0, formatted.size());
|
|
||||||
}
|
|
||||||
fflush(target_file_);
|
|
||||||
}
|
|
||||||
|
|
||||||
void flush() override
|
|
||||||
{
|
|
||||||
std::lock_guard<mutex_t> lock(mutex_);
|
|
||||||
fflush(target_file_);
|
|
||||||
}
|
|
||||||
|
|
||||||
void set_pattern(const std::string &pattern) final
|
|
||||||
{
|
|
||||||
std::lock_guard<mutex_t> lock(mutex_);
|
|
||||||
formatter_ = std::unique_ptr<spdlog::formatter>(new pattern_formatter(pattern));
|
|
||||||
}
|
|
||||||
|
|
||||||
void set_formatter(std::unique_ptr<spdlog::formatter> sink_formatter) override
|
|
||||||
{
|
|
||||||
std::lock_guard<mutex_t> lock(mutex_);
|
|
||||||
formatter_ = std::move(sink_formatter);
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
|
||||||
void print_ccode_(const std::string &color_code)
|
|
||||||
{
|
|
||||||
fwrite(color_code.data(), sizeof(char), color_code.size(), target_file_);
|
|
||||||
}
|
|
||||||
void print_range_(const fmt::memory_buffer &formatted, size_t start, size_t end)
|
|
||||||
{
|
|
||||||
fwrite(formatted.data() + start, sizeof(char), end - start, target_file_);
|
|
||||||
}
|
|
||||||
|
|
||||||
FILE *target_file_;
|
|
||||||
mutex_t &mutex_;
|
|
||||||
|
|
||||||
bool should_do_colors_;
|
|
||||||
std::unordered_map<level::level_enum, std::string, level::level_hasher> colors_;
|
|
||||||
};
|
|
||||||
|
|
||||||
using ansicolor_stdout_sink_mt = ansicolor_sink<details::console_stdout, details::console_mutex>;
|
|
||||||
using ansicolor_stdout_sink_st = ansicolor_sink<details::console_stdout, details::console_nullmutex>;
|
|
||||||
|
|
||||||
using ansicolor_stderr_sink_mt = ansicolor_sink<details::console_stderr, details::console_mutex>;
|
|
||||||
using ansicolor_stderr_sink_st = ansicolor_sink<details::console_stderr, details::console_nullmutex>;
|
|
||||||
|
|
||||||
} // namespace sinks
|
|
||||||
|
|
||||||
} // namespace spdlog
|
|
||||||
69
deps/spdlog/sinks/base_sink.h
vendored
69
deps/spdlog/sinks/base_sink.h
vendored
@ -1,69 +0,0 @@
|
|||||||
//
|
|
||||||
// Copyright(c) 2015 Gabi Melman.
|
|
||||||
// Distributed under the MIT License (http://opensource.org/licenses/MIT)
|
|
||||||
//
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
//
|
|
||||||
// base sink templated over a mutex (either dummy or real)
|
|
||||||
// concrete implementation should override the sink_it_() and flush_() methods.
|
|
||||||
// locking is taken care of in this class - no locking needed by the
|
|
||||||
// implementers..
|
|
||||||
//
|
|
||||||
|
|
||||||
#include "spdlog/common.h"
|
|
||||||
#include "spdlog/details/log_msg.h"
|
|
||||||
#include "spdlog/formatter.h"
|
|
||||||
#include "spdlog/sinks/sink.h"
|
|
||||||
|
|
||||||
namespace spdlog {
|
|
||||||
namespace sinks {
|
|
||||||
template<typename Mutex>
|
|
||||||
class base_sink : public sink
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
base_sink() = default;
|
|
||||||
base_sink(const base_sink &) = delete;
|
|
||||||
base_sink &operator=(const base_sink &) = delete;
|
|
||||||
|
|
||||||
void log(const details::log_msg &msg) final
|
|
||||||
{
|
|
||||||
std::lock_guard<Mutex> lock(mutex_);
|
|
||||||
sink_it_(msg);
|
|
||||||
}
|
|
||||||
|
|
||||||
void flush() final
|
|
||||||
{
|
|
||||||
std::lock_guard<Mutex> lock(mutex_);
|
|
||||||
flush_();
|
|
||||||
}
|
|
||||||
|
|
||||||
void set_pattern(const std::string &pattern) final
|
|
||||||
{
|
|
||||||
std::lock_guard<Mutex> lock(mutex_);
|
|
||||||
set_pattern_(pattern);
|
|
||||||
}
|
|
||||||
|
|
||||||
void set_formatter(std::unique_ptr<spdlog::formatter> sink_formatter) final
|
|
||||||
{
|
|
||||||
std::lock_guard<Mutex> lock(mutex_);
|
|
||||||
set_formatter_(std::move(sink_formatter));
|
|
||||||
}
|
|
||||||
|
|
||||||
protected:
|
|
||||||
virtual void sink_it_(const details::log_msg &msg) = 0;
|
|
||||||
virtual void flush_() = 0;
|
|
||||||
|
|
||||||
virtual void set_pattern_(const std::string &pattern)
|
|
||||||
{
|
|
||||||
set_formatter_(details::make_unique<spdlog::pattern_formatter>(pattern));
|
|
||||||
}
|
|
||||||
|
|
||||||
virtual void set_formatter_(std::unique_ptr<spdlog::formatter> sink_formatter)
|
|
||||||
{
|
|
||||||
formatter_ = std::move(sink_formatter);
|
|
||||||
}
|
|
||||||
Mutex mutex_;
|
|
||||||
};
|
|
||||||
} // namespace sinks
|
|
||||||
} // namespace spdlog
|
|
||||||
70
deps/spdlog/sinks/basic_file_sink.h
vendored
70
deps/spdlog/sinks/basic_file_sink.h
vendored
@ -1,70 +0,0 @@
|
|||||||
//
|
|
||||||
// Copyright(c) 2015-2018 Gabi Melman.
|
|
||||||
// Distributed under the MIT License (http://opensource.org/licenses/MIT)
|
|
||||||
//
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#ifndef SPDLOG_H
|
|
||||||
#include "spdlog/spdlog.h"
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#include "spdlog/details/file_helper.h"
|
|
||||||
#include "spdlog/details/null_mutex.h"
|
|
||||||
#include "spdlog/sinks/base_sink.h"
|
|
||||||
|
|
||||||
#include <mutex>
|
|
||||||
#include <string>
|
|
||||||
|
|
||||||
namespace spdlog {
|
|
||||||
namespace sinks {
|
|
||||||
/*
|
|
||||||
* Trivial file sink with single file as target
|
|
||||||
*/
|
|
||||||
template<typename Mutex>
|
|
||||||
class basic_file_sink final : public base_sink<Mutex>
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
explicit basic_file_sink(const filename_t &filename, bool truncate = false)
|
|
||||||
{
|
|
||||||
file_helper_.open(filename, truncate);
|
|
||||||
}
|
|
||||||
|
|
||||||
protected:
|
|
||||||
void sink_it_(const details::log_msg &msg) override
|
|
||||||
{
|
|
||||||
fmt::memory_buffer formatted;
|
|
||||||
sink::formatter_->format(msg, formatted);
|
|
||||||
file_helper_.write(formatted);
|
|
||||||
}
|
|
||||||
|
|
||||||
void flush_() override
|
|
||||||
{
|
|
||||||
file_helper_.flush();
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
|
||||||
details::file_helper file_helper_;
|
|
||||||
};
|
|
||||||
|
|
||||||
using basic_file_sink_mt = basic_file_sink<std::mutex>;
|
|
||||||
using basic_file_sink_st = basic_file_sink<details::null_mutex>;
|
|
||||||
|
|
||||||
} // namespace sinks
|
|
||||||
|
|
||||||
//
|
|
||||||
// factory functions
|
|
||||||
//
|
|
||||||
template<typename Factory = default_factory>
|
|
||||||
inline std::shared_ptr<logger> basic_logger_mt(const std::string &logger_name, const filename_t &filename, bool truncate = false)
|
|
||||||
{
|
|
||||||
return Factory::template create<sinks::basic_file_sink_mt>(logger_name, filename, truncate);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename Factory = default_factory>
|
|
||||||
inline std::shared_ptr<logger> basic_logger_st(const std::string &logger_name, const filename_t &filename, bool truncate = false)
|
|
||||||
{
|
|
||||||
return Factory::template create<sinks::basic_file_sink_st>(logger_name, filename, truncate);
|
|
||||||
}
|
|
||||||
|
|
||||||
} // namespace spdlog
|
|
||||||
136
deps/spdlog/sinks/daily_file_sink.h
vendored
136
deps/spdlog/sinks/daily_file_sink.h
vendored
@ -1,136 +0,0 @@
|
|||||||
//
|
|
||||||
// Copyright(c) 2015 Gabi Melman.
|
|
||||||
// Distributed under the MIT License (http://opensource.org/licenses/MIT)
|
|
||||||
//
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#ifndef SPDLOG_H
|
|
||||||
#include "spdlog/spdlog.h"
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#include "spdlog/details/file_helper.h"
|
|
||||||
#include "spdlog/details/null_mutex.h"
|
|
||||||
#include "spdlog/fmt/fmt.h"
|
|
||||||
#include "spdlog/sinks/base_sink.h"
|
|
||||||
|
|
||||||
#include <chrono>
|
|
||||||
#include <cstdio>
|
|
||||||
#include <ctime>
|
|
||||||
#include <mutex>
|
|
||||||
#include <string>
|
|
||||||
|
|
||||||
namespace spdlog {
|
|
||||||
namespace sinks {
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Generator of daily log file names in format basename.YYYY-MM-DD.ext
|
|
||||||
*/
|
|
||||||
struct daily_filename_calculator
|
|
||||||
{
|
|
||||||
// Create filename for the form basename.YYYY-MM-DD
|
|
||||||
static filename_t calc_filename(const filename_t &filename, const tm &now_tm)
|
|
||||||
{
|
|
||||||
filename_t basename, ext;
|
|
||||||
std::tie(basename, ext) = details::file_helper::split_by_extension(filename);
|
|
||||||
std::conditional<std::is_same<filename_t::value_type, char>::value, fmt::memory_buffer, fmt::wmemory_buffer>::type w;
|
|
||||||
fmt::format_to(
|
|
||||||
w, SPDLOG_FILENAME_T("{}_{:04d}-{:02d}-{:02d}{}"), basename, now_tm.tm_year + 1900, now_tm.tm_mon + 1, now_tm.tm_mday, ext);
|
|
||||||
return fmt::to_string(w);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Rotating file sink based on date. rotates at midnight
|
|
||||||
*/
|
|
||||||
template<typename Mutex, typename FileNameCalc = daily_filename_calculator>
|
|
||||||
class daily_file_sink final : public base_sink<Mutex>
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
// create daily file sink which rotates on given time
|
|
||||||
daily_file_sink(filename_t base_filename, int rotation_hour, int rotation_minute, bool truncate = false)
|
|
||||||
: base_filename_(std::move(base_filename))
|
|
||||||
, rotation_h_(rotation_hour)
|
|
||||||
, rotation_m_(rotation_minute)
|
|
||||||
, truncate_(truncate)
|
|
||||||
{
|
|
||||||
if (rotation_hour < 0 || rotation_hour > 23 || rotation_minute < 0 || rotation_minute > 59)
|
|
||||||
{
|
|
||||||
throw spdlog_ex("daily_file_sink: Invalid rotation time in ctor");
|
|
||||||
}
|
|
||||||
auto now = log_clock::now();
|
|
||||||
file_helper_.open(FileNameCalc::calc_filename(base_filename_, now_tm(now)), truncate_);
|
|
||||||
rotation_tp_ = next_rotation_tp_();
|
|
||||||
}
|
|
||||||
|
|
||||||
protected:
|
|
||||||
void sink_it_(const details::log_msg &msg) override
|
|
||||||
{
|
|
||||||
|
|
||||||
if (msg.time >= rotation_tp_)
|
|
||||||
{
|
|
||||||
file_helper_.open(FileNameCalc::calc_filename(base_filename_, now_tm(msg.time)), truncate_);
|
|
||||||
rotation_tp_ = next_rotation_tp_();
|
|
||||||
}
|
|
||||||
fmt::memory_buffer formatted;
|
|
||||||
sink::formatter_->format(msg, formatted);
|
|
||||||
file_helper_.write(formatted);
|
|
||||||
}
|
|
||||||
|
|
||||||
void flush_() override
|
|
||||||
{
|
|
||||||
file_helper_.flush();
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
|
||||||
tm now_tm(log_clock::time_point tp)
|
|
||||||
{
|
|
||||||
time_t tnow = log_clock::to_time_t(tp);
|
|
||||||
return spdlog::details::os::localtime(tnow);
|
|
||||||
}
|
|
||||||
|
|
||||||
log_clock::time_point next_rotation_tp_()
|
|
||||||
{
|
|
||||||
auto now = log_clock::now();
|
|
||||||
tm date = now_tm(now);
|
|
||||||
date.tm_hour = rotation_h_;
|
|
||||||
date.tm_min = rotation_m_;
|
|
||||||
date.tm_sec = 0;
|
|
||||||
auto rotation_time = log_clock::from_time_t(std::mktime(&date));
|
|
||||||
if (rotation_time > now)
|
|
||||||
{
|
|
||||||
return rotation_time;
|
|
||||||
}
|
|
||||||
return {rotation_time + std::chrono::hours(24)};
|
|
||||||
}
|
|
||||||
|
|
||||||
filename_t base_filename_;
|
|
||||||
int rotation_h_;
|
|
||||||
int rotation_m_;
|
|
||||||
log_clock::time_point rotation_tp_;
|
|
||||||
details::file_helper file_helper_;
|
|
||||||
bool truncate_;
|
|
||||||
};
|
|
||||||
|
|
||||||
using daily_file_sink_mt = daily_file_sink<std::mutex>;
|
|
||||||
using daily_file_sink_st = daily_file_sink<details::null_mutex>;
|
|
||||||
|
|
||||||
} // namespace sinks
|
|
||||||
|
|
||||||
//
|
|
||||||
// factory functions
|
|
||||||
//
|
|
||||||
template<typename Factory = default_factory>
|
|
||||||
inline std::shared_ptr<logger> daily_logger_mt(
|
|
||||||
const std::string &logger_name, const filename_t &filename, int hour = 0, int minute = 0, bool truncate = false)
|
|
||||||
{
|
|
||||||
return Factory::template create<sinks::daily_file_sink_mt>(logger_name, filename, hour, minute, truncate);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename Factory = default_factory>
|
|
||||||
inline std::shared_ptr<logger> daily_logger_st(
|
|
||||||
const std::string &logger_name, const filename_t &filename, int hour = 0, int minute = 0, bool truncate = false)
|
|
||||||
{
|
|
||||||
return Factory::template create<sinks::daily_file_sink_st>(logger_name, filename, hour, minute, truncate);
|
|
||||||
}
|
|
||||||
} // namespace spdlog
|
|
||||||
94
deps/spdlog/sinks/dist_sink.h
vendored
94
deps/spdlog/sinks/dist_sink.h
vendored
@ -1,94 +0,0 @@
|
|||||||
//
|
|
||||||
// Copyright (c) 2015 David Schury, Gabi Melman
|
|
||||||
// Distributed under the MIT License (http://opensource.org/licenses/MIT)
|
|
||||||
//
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#ifndef SPDLOG_H
|
|
||||||
#include "spdlog/spdlog.h"
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#include "base_sink.h"
|
|
||||||
#include "spdlog/details/log_msg.h"
|
|
||||||
#include "spdlog/details/null_mutex.h"
|
|
||||||
|
|
||||||
#include <algorithm>
|
|
||||||
#include <memory>
|
|
||||||
#include <mutex>
|
|
||||||
#include <vector>
|
|
||||||
|
|
||||||
// Distribution sink (mux). Stores a vector of sinks which get called when log
|
|
||||||
// is called
|
|
||||||
|
|
||||||
namespace spdlog {
|
|
||||||
namespace sinks {
|
|
||||||
|
|
||||||
template<typename Mutex>
|
|
||||||
class dist_sink : public base_sink<Mutex>
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
dist_sink() = default;
|
|
||||||
dist_sink(const dist_sink &) = delete;
|
|
||||||
dist_sink &operator=(const dist_sink &) = delete;
|
|
||||||
|
|
||||||
void add_sink(std::shared_ptr<sink> sink)
|
|
||||||
{
|
|
||||||
std::lock_guard<Mutex> lock(base_sink<Mutex>::mutex_);
|
|
||||||
sinks_.push_back(sink);
|
|
||||||
}
|
|
||||||
|
|
||||||
void remove_sink(std::shared_ptr<sink> sink)
|
|
||||||
{
|
|
||||||
std::lock_guard<Mutex> lock(base_sink<Mutex>::mutex_);
|
|
||||||
sinks_.erase(std::remove(sinks_.begin(), sinks_.end(), sink), sinks_.end());
|
|
||||||
}
|
|
||||||
|
|
||||||
void set_sinks(std::vector<std::shared_ptr<sink>> sinks)
|
|
||||||
{
|
|
||||||
std::lock_guard<Mutex> lock(base_sink<Mutex>::mutex_);
|
|
||||||
sinks_ = std::move(sinks);
|
|
||||||
}
|
|
||||||
|
|
||||||
protected:
|
|
||||||
void sink_it_(const details::log_msg &msg) override
|
|
||||||
{
|
|
||||||
|
|
||||||
for (auto &sink : sinks_)
|
|
||||||
{
|
|
||||||
if (sink->should_log(msg.level))
|
|
||||||
{
|
|
||||||
sink->log(msg);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void flush_() override
|
|
||||||
{
|
|
||||||
for (auto &sink : sinks_)
|
|
||||||
{
|
|
||||||
sink->flush();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void set_pattern_(const std::string &pattern) override
|
|
||||||
{
|
|
||||||
set_formatter_(details::make_unique<spdlog::pattern_formatter>(pattern));
|
|
||||||
}
|
|
||||||
|
|
||||||
void set_formatter_(std::unique_ptr<spdlog::formatter> sink_formatter) override
|
|
||||||
{
|
|
||||||
base_sink<Mutex>::formatter_ = std::move(sink_formatter);
|
|
||||||
for (auto &sink : sinks_)
|
|
||||||
{
|
|
||||||
sink->set_formatter(base_sink<Mutex>::formatter_->clone());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
std::vector<std::shared_ptr<sink>> sinks_;
|
|
||||||
};
|
|
||||||
|
|
||||||
using dist_sink_mt = dist_sink<std::mutex>;
|
|
||||||
using dist_sink_st = dist_sink<details::null_mutex>;
|
|
||||||
|
|
||||||
} // namespace sinks
|
|
||||||
} // namespace spdlog
|
|
||||||
54
deps/spdlog/sinks/msvc_sink.h
vendored
54
deps/spdlog/sinks/msvc_sink.h
vendored
@ -1,54 +0,0 @@
|
|||||||
//
|
|
||||||
// Copyright(c) 2016 Alexander Dalshov.
|
|
||||||
// Distributed under the MIT License (http://opensource.org/licenses/MIT)
|
|
||||||
//
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#ifndef SPDLOG_H
|
|
||||||
#include "spdlog/spdlog.h"
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#if defined(_WIN32)
|
|
||||||
|
|
||||||
#include "spdlog/details/null_mutex.h"
|
|
||||||
#include "spdlog/sinks/base_sink.h"
|
|
||||||
|
|
||||||
#include <winbase.h>
|
|
||||||
|
|
||||||
#include <mutex>
|
|
||||||
#include <string>
|
|
||||||
|
|
||||||
namespace spdlog {
|
|
||||||
namespace sinks {
|
|
||||||
/*
|
|
||||||
* MSVC sink (logging using OutputDebugStringA)
|
|
||||||
*/
|
|
||||||
template<typename Mutex>
|
|
||||||
class msvc_sink : public base_sink<Mutex>
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
explicit msvc_sink() {}
|
|
||||||
|
|
||||||
protected:
|
|
||||||
void sink_it_(const details::log_msg &msg) override
|
|
||||||
{
|
|
||||||
|
|
||||||
fmt::memory_buffer formatted;
|
|
||||||
sink::formatter_->format(msg, formatted);
|
|
||||||
OutputDebugStringA(fmt::to_string(formatted).c_str());
|
|
||||||
}
|
|
||||||
|
|
||||||
void flush_() override {}
|
|
||||||
};
|
|
||||||
|
|
||||||
using msvc_sink_mt = msvc_sink<std::mutex>;
|
|
||||||
using msvc_sink_st = msvc_sink<details::null_mutex>;
|
|
||||||
|
|
||||||
using windebug_sink_mt = msvc_sink_mt;
|
|
||||||
using windebug_sink_st = msvc_sink_st;
|
|
||||||
|
|
||||||
} // namespace sinks
|
|
||||||
} // namespace spdlog
|
|
||||||
|
|
||||||
#endif
|
|
||||||
49
deps/spdlog/sinks/null_sink.h
vendored
49
deps/spdlog/sinks/null_sink.h
vendored
@ -1,49 +0,0 @@
|
|||||||
//
|
|
||||||
// Copyright(c) 2015 Gabi Melman.
|
|
||||||
// Distributed under the MIT License (http://opensource.org/licenses/MIT)
|
|
||||||
//
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#ifndef SPDLOG_H
|
|
||||||
#include "spdlog/spdlog.h"
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#include "spdlog/details/null_mutex.h"
|
|
||||||
#include "spdlog/sinks/base_sink.h"
|
|
||||||
|
|
||||||
#include <mutex>
|
|
||||||
|
|
||||||
namespace spdlog {
|
|
||||||
namespace sinks {
|
|
||||||
|
|
||||||
template<typename Mutex>
|
|
||||||
class null_sink : public base_sink<Mutex>
|
|
||||||
{
|
|
||||||
protected:
|
|
||||||
void sink_it_(const details::log_msg &) override {}
|
|
||||||
void flush_() override {}
|
|
||||||
};
|
|
||||||
|
|
||||||
using null_sink_mt = null_sink<std::mutex>;
|
|
||||||
using null_sink_st = null_sink<details::null_mutex>;
|
|
||||||
|
|
||||||
} // namespace sinks
|
|
||||||
|
|
||||||
template<typename Factory = default_factory>
|
|
||||||
inline std::shared_ptr<logger> null_logger_mt(const std::string &logger_name)
|
|
||||||
{
|
|
||||||
auto null_logger = Factory::template create<sinks::null_sink_mt>(logger_name);
|
|
||||||
null_logger->set_level(level::off);
|
|
||||||
return null_logger;
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename Factory = default_factory>
|
|
||||||
inline std::shared_ptr<logger> null_logger_st(const std::string &logger_name)
|
|
||||||
{
|
|
||||||
auto null_logger = Factory::template create<sinks::null_sink_st>(logger_name);
|
|
||||||
null_logger->set_level(level::off);
|
|
||||||
return null_logger;
|
|
||||||
}
|
|
||||||
|
|
||||||
} // namespace spdlog
|
|
||||||
57
deps/spdlog/sinks/ostream_sink.h
vendored
57
deps/spdlog/sinks/ostream_sink.h
vendored
@ -1,57 +0,0 @@
|
|||||||
//
|
|
||||||
// Copyright(c) 2015 Gabi Melman.
|
|
||||||
// Distributed under the MIT License (http://opensource.org/licenses/MIT)
|
|
||||||
//
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#ifndef SPDLOG_H
|
|
||||||
#include "spdlog/spdlog.h"
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#include "spdlog/details/null_mutex.h"
|
|
||||||
#include "spdlog/sinks/base_sink.h"
|
|
||||||
|
|
||||||
#include <mutex>
|
|
||||||
#include <ostream>
|
|
||||||
|
|
||||||
namespace spdlog {
|
|
||||||
namespace sinks {
|
|
||||||
template<typename Mutex>
|
|
||||||
class ostream_sink final : public base_sink<Mutex>
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
explicit ostream_sink(std::ostream &os, bool force_flush = false)
|
|
||||||
: ostream_(os)
|
|
||||||
, force_flush_(force_flush)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
ostream_sink(const ostream_sink &) = delete;
|
|
||||||
ostream_sink &operator=(const ostream_sink &) = delete;
|
|
||||||
|
|
||||||
protected:
|
|
||||||
void sink_it_(const details::log_msg &msg) override
|
|
||||||
{
|
|
||||||
fmt::memory_buffer formatted;
|
|
||||||
sink::formatter_->format(msg, formatted);
|
|
||||||
ostream_.write(formatted.data(), static_cast<std::streamsize>(formatted.size()));
|
|
||||||
if (force_flush_)
|
|
||||||
{
|
|
||||||
ostream_.flush();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void flush_() override
|
|
||||||
{
|
|
||||||
ostream_.flush();
|
|
||||||
}
|
|
||||||
|
|
||||||
std::ostream &ostream_;
|
|
||||||
bool force_flush_;
|
|
||||||
};
|
|
||||||
|
|
||||||
using ostream_sink_mt = ostream_sink<std::mutex>;
|
|
||||||
using ostream_sink_st = ostream_sink<details::null_mutex>;
|
|
||||||
|
|
||||||
} // namespace sinks
|
|
||||||
} // namespace spdlog
|
|
||||||
155
deps/spdlog/sinks/rotating_file_sink.h
vendored
155
deps/spdlog/sinks/rotating_file_sink.h
vendored
@ -1,155 +0,0 @@
|
|||||||
//
|
|
||||||
// Copyright(c) 2015 Gabi Melman.
|
|
||||||
// Distributed under the MIT License (http://opensource.org/licenses/MIT)
|
|
||||||
//
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#ifndef SPDLOG_H
|
|
||||||
#include "spdlog/spdlog.h"
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#include "spdlog/details/file_helper.h"
|
|
||||||
#include "spdlog/details/null_mutex.h"
|
|
||||||
#include "spdlog/fmt/fmt.h"
|
|
||||||
#include "spdlog/sinks/base_sink.h"
|
|
||||||
|
|
||||||
#include <cerrno>
|
|
||||||
#include <chrono>
|
|
||||||
#include <ctime>
|
|
||||||
#include <mutex>
|
|
||||||
#include <string>
|
|
||||||
#include <tuple>
|
|
||||||
|
|
||||||
namespace spdlog {
|
|
||||||
namespace sinks {
|
|
||||||
|
|
||||||
//
|
|
||||||
// Rotating file sink based on size
|
|
||||||
//
|
|
||||||
template<typename Mutex>
|
|
||||||
class rotating_file_sink final : public base_sink<Mutex>
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
rotating_file_sink(filename_t base_filename, std::size_t max_size, std::size_t max_files)
|
|
||||||
: base_filename_(std::move(base_filename))
|
|
||||||
, max_size_(max_size)
|
|
||||||
, max_files_(max_files)
|
|
||||||
{
|
|
||||||
file_helper_.open(calc_filename(base_filename_, 0));
|
|
||||||
current_size_ = file_helper_.size(); // expensive. called only once
|
|
||||||
}
|
|
||||||
|
|
||||||
// calc filename according to index and file extension if exists.
|
|
||||||
// e.g. calc_filename("logs/mylog.txt, 3) => "logs/mylog.3.txt".
|
|
||||||
static filename_t calc_filename(const filename_t &filename, std::size_t index)
|
|
||||||
{
|
|
||||||
typename std::conditional<std::is_same<filename_t::value_type, char>::value, fmt::memory_buffer, fmt::wmemory_buffer>::type w;
|
|
||||||
if (index != 0u)
|
|
||||||
{
|
|
||||||
filename_t basename, ext;
|
|
||||||
std::tie(basename, ext) = details::file_helper::split_by_extension(filename);
|
|
||||||
fmt::format_to(w, SPDLOG_FILENAME_T("{}.{}{}"), basename, index, ext);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
fmt::format_to(w, SPDLOG_FILENAME_T("{}"), filename);
|
|
||||||
}
|
|
||||||
return fmt::to_string(w);
|
|
||||||
}
|
|
||||||
|
|
||||||
protected:
|
|
||||||
void sink_it_(const details::log_msg &msg) override
|
|
||||||
{
|
|
||||||
fmt::memory_buffer formatted;
|
|
||||||
sink::formatter_->format(msg, formatted);
|
|
||||||
current_size_ += formatted.size();
|
|
||||||
if (current_size_ > max_size_)
|
|
||||||
{
|
|
||||||
rotate_();
|
|
||||||
current_size_ = formatted.size();
|
|
||||||
}
|
|
||||||
file_helper_.write(formatted);
|
|
||||||
}
|
|
||||||
|
|
||||||
void flush_() override
|
|
||||||
{
|
|
||||||
file_helper_.flush();
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
|
||||||
// Rotate files:
|
|
||||||
// log.txt -> log.1.txt
|
|
||||||
// log.1.txt -> log.2.txt
|
|
||||||
// log.2.txt -> log.3.txt
|
|
||||||
// log.3.txt -> delete
|
|
||||||
void rotate_()
|
|
||||||
{
|
|
||||||
using details::os::filename_to_str;
|
|
||||||
file_helper_.close();
|
|
||||||
for (auto i = max_files_; i > 0; --i)
|
|
||||||
{
|
|
||||||
filename_t src = calc_filename(base_filename_, i - 1);
|
|
||||||
if (!details::file_helper::file_exists(src))
|
|
||||||
{
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
filename_t target = calc_filename(base_filename_, i);
|
|
||||||
|
|
||||||
if (!rename_file(src, target))
|
|
||||||
{
|
|
||||||
// if failed try again after a small delay.
|
|
||||||
// this is a workaround to a windows issue, where very high rotation
|
|
||||||
// rates can cause the rename to fail with permission denied (because of antivirus?).
|
|
||||||
details::os::sleep_for_millis(100);
|
|
||||||
if (!rename_file(src, target))
|
|
||||||
{
|
|
||||||
file_helper_.reopen(true); // truncate the log file anyway to prevent it to grow beyond its limit!
|
|
||||||
current_size_ = 0;
|
|
||||||
throw spdlog_ex(
|
|
||||||
"rotating_file_sink: failed renaming " + filename_to_str(src) + " to " + filename_to_str(target), errno);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
file_helper_.reopen(true);
|
|
||||||
}
|
|
||||||
|
|
||||||
// delete the target if exists, and rename the src file to target
|
|
||||||
// return true on success, false otherwise.
|
|
||||||
bool rename_file(const filename_t &src_filename, const filename_t &target_filename)
|
|
||||||
{
|
|
||||||
// try to delete the target file in case it already exists.
|
|
||||||
(void)details::os::remove(target_filename);
|
|
||||||
return details::os::rename(src_filename, target_filename) == 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
filename_t base_filename_;
|
|
||||||
std::size_t max_size_;
|
|
||||||
std::size_t max_files_;
|
|
||||||
std::size_t current_size_;
|
|
||||||
details::file_helper file_helper_;
|
|
||||||
};
|
|
||||||
|
|
||||||
using rotating_file_sink_mt = rotating_file_sink<std::mutex>;
|
|
||||||
using rotating_file_sink_st = rotating_file_sink<details::null_mutex>;
|
|
||||||
|
|
||||||
} // namespace sinks
|
|
||||||
|
|
||||||
//
|
|
||||||
// factory functions
|
|
||||||
//
|
|
||||||
|
|
||||||
template<typename Factory = default_factory>
|
|
||||||
inline std::shared_ptr<logger> rotating_logger_mt(
|
|
||||||
const std::string &logger_name, const filename_t &filename, size_t max_file_size, size_t max_files)
|
|
||||||
{
|
|
||||||
return Factory::template create<sinks::rotating_file_sink_mt>(logger_name, filename, max_file_size, max_files);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename Factory = default_factory>
|
|
||||||
inline std::shared_ptr<logger> rotating_logger_st(
|
|
||||||
const std::string &logger_name, const filename_t &filename, size_t max_file_size, size_t max_files)
|
|
||||||
{
|
|
||||||
return Factory::template create<sinks::rotating_file_sink_st>(logger_name, filename, max_file_size, max_files);
|
|
||||||
}
|
|
||||||
} // namespace spdlog
|
|
||||||
59
deps/spdlog/sinks/sink.h
vendored
59
deps/spdlog/sinks/sink.h
vendored
@ -1,59 +0,0 @@
|
|||||||
//
|
|
||||||
// Copyright(c) 2015 Gabi Melman.
|
|
||||||
// Distributed under the MIT License (http://opensource.org/licenses/MIT)
|
|
||||||
//
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include "spdlog/details/log_msg.h"
|
|
||||||
#include "spdlog/details/pattern_formatter.h"
|
|
||||||
#include "spdlog/formatter.h"
|
|
||||||
|
|
||||||
namespace spdlog {
|
|
||||||
namespace sinks {
|
|
||||||
class sink
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
sink()
|
|
||||||
: level_(level::trace)
|
|
||||||
, formatter_(new pattern_formatter())
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
explicit sink(std::unique_ptr<spdlog::pattern_formatter> formatter)
|
|
||||||
: level_(level::trace)
|
|
||||||
, formatter_(std::move(formatter))
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
virtual ~sink() = default;
|
|
||||||
virtual void log(const details::log_msg &msg) = 0;
|
|
||||||
virtual void flush() = 0;
|
|
||||||
virtual void set_pattern(const std::string &pattern) = 0;
|
|
||||||
virtual void set_formatter(std::unique_ptr<spdlog::formatter> sink_formatter) = 0;
|
|
||||||
|
|
||||||
bool should_log(level::level_enum msg_level) const
|
|
||||||
{
|
|
||||||
return msg_level >= level_.load(std::memory_order_relaxed);
|
|
||||||
}
|
|
||||||
|
|
||||||
void set_level(level::level_enum log_level)
|
|
||||||
{
|
|
||||||
level_.store(log_level);
|
|
||||||
}
|
|
||||||
|
|
||||||
level::level_enum level() const
|
|
||||||
{
|
|
||||||
return static_cast<spdlog::level::level_enum>(level_.load(std::memory_order_relaxed));
|
|
||||||
}
|
|
||||||
|
|
||||||
protected:
|
|
||||||
// sink log level - default is all
|
|
||||||
level_t level_;
|
|
||||||
|
|
||||||
// sink formatter - default is full format
|
|
||||||
std::unique_ptr<spdlog::formatter> formatter_;
|
|
||||||
};
|
|
||||||
|
|
||||||
} // namespace sinks
|
|
||||||
} // namespace spdlog
|
|
||||||
56
deps/spdlog/sinks/stdout_color_sinks.h
vendored
56
deps/spdlog/sinks/stdout_color_sinks.h
vendored
@ -1,56 +0,0 @@
|
|||||||
//
|
|
||||||
// Copyright(c) 2018 spdlog
|
|
||||||
// Distributed under the MIT License (http://opensource.org/licenses/MIT)
|
|
||||||
//
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#ifndef SPDLOG_H
|
|
||||||
#include "spdlog/spdlog.h"
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifdef _WIN32
|
|
||||||
#include "spdlog/sinks/wincolor_sink.h"
|
|
||||||
#else
|
|
||||||
#include "spdlog/sinks/ansicolor_sink.h"
|
|
||||||
#endif
|
|
||||||
|
|
||||||
namespace spdlog {
|
|
||||||
namespace sinks {
|
|
||||||
#ifdef _WIN32
|
|
||||||
using stdout_color_sink_mt = wincolor_stdout_sink_mt;
|
|
||||||
using stdout_color_sink_st = wincolor_stdout_sink_st;
|
|
||||||
using stderr_color_sink_mt = wincolor_stderr_sink_mt;
|
|
||||||
using stderr_color_sink_st = wincolor_stderr_sink_st;
|
|
||||||
#else
|
|
||||||
using stdout_color_sink_mt = ansicolor_stdout_sink_mt;
|
|
||||||
using stdout_color_sink_st = ansicolor_stdout_sink_st;
|
|
||||||
using stderr_color_sink_mt = ansicolor_stderr_sink_mt;
|
|
||||||
using stderr_color_sink_st = ansicolor_stderr_sink_st;
|
|
||||||
#endif
|
|
||||||
} // namespace sinks
|
|
||||||
|
|
||||||
template<typename Factory = default_factory>
|
|
||||||
inline std::shared_ptr<logger> stdout_color_mt(const std::string &logger_name)
|
|
||||||
{
|
|
||||||
return Factory::template create<sinks::stdout_color_sink_mt>(logger_name);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename Factory = default_factory>
|
|
||||||
inline std::shared_ptr<logger> stdout_color_st(const std::string &logger_name)
|
|
||||||
{
|
|
||||||
return Factory::template create<sinks::stdout_color_sink_st>(logger_name);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename Factory = default_factory>
|
|
||||||
inline std::shared_ptr<logger> stderr_color_mt(const std::string &logger_name)
|
|
||||||
{
|
|
||||||
return Factory::template create<sinks::stderr_color_sink_mt>(logger_name);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename Factory = default_factory>
|
|
||||||
inline std::shared_ptr<logger> stderr_color_st(const std::string &logger_name)
|
|
||||||
{
|
|
||||||
return Factory::template create<sinks::stderr_color_sink_mt>(logger_name);
|
|
||||||
}
|
|
||||||
} // namespace spdlog
|
|
||||||
102
deps/spdlog/sinks/stdout_sinks.h
vendored
102
deps/spdlog/sinks/stdout_sinks.h
vendored
@ -1,102 +0,0 @@
|
|||||||
//
|
|
||||||
// Copyright(c) 2015 Gabi Melman.
|
|
||||||
// Distributed under the MIT License (http://opensource.org/licenses/MIT)
|
|
||||||
//
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#ifndef SPDLOG_H
|
|
||||||
#include "spdlog/spdlog.h"
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#include "spdlog/details/console_globals.h"
|
|
||||||
#include "spdlog/details/null_mutex.h"
|
|
||||||
|
|
||||||
#include <cstdio>
|
|
||||||
#include <memory>
|
|
||||||
#include <mutex>
|
|
||||||
|
|
||||||
namespace spdlog {
|
|
||||||
|
|
||||||
namespace sinks {
|
|
||||||
|
|
||||||
template<typename TargetStream, typename ConsoleMutex>
|
|
||||||
class stdout_sink final : public sink
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
using mutex_t = typename ConsoleMutex::mutex_t;
|
|
||||||
stdout_sink()
|
|
||||||
: mutex_(ConsoleMutex::mutex())
|
|
||||||
, file_(TargetStream::stream())
|
|
||||||
{
|
|
||||||
}
|
|
||||||
~stdout_sink() override = default;
|
|
||||||
|
|
||||||
stdout_sink(const stdout_sink &other) = delete;
|
|
||||||
stdout_sink &operator=(const stdout_sink &other) = delete;
|
|
||||||
|
|
||||||
void log(const details::log_msg &msg) override
|
|
||||||
{
|
|
||||||
std::lock_guard<mutex_t> lock(mutex_);
|
|
||||||
fmt::memory_buffer formatted;
|
|
||||||
formatter_->format(msg, formatted);
|
|
||||||
fwrite(formatted.data(), sizeof(char), formatted.size(), file_);
|
|
||||||
fflush(TargetStream::stream());
|
|
||||||
}
|
|
||||||
|
|
||||||
void flush() override
|
|
||||||
{
|
|
||||||
std::lock_guard<mutex_t> lock(mutex_);
|
|
||||||
fflush(file_);
|
|
||||||
}
|
|
||||||
|
|
||||||
void set_pattern(const std::string &pattern) override
|
|
||||||
{
|
|
||||||
std::lock_guard<mutex_t> lock(mutex_);
|
|
||||||
formatter_ = std::unique_ptr<spdlog::formatter>(new pattern_formatter(pattern));
|
|
||||||
}
|
|
||||||
|
|
||||||
void set_formatter(std::unique_ptr<spdlog::formatter> sink_formatter) override
|
|
||||||
{
|
|
||||||
std::lock_guard<mutex_t> lock(mutex_);
|
|
||||||
formatter_ = std::move(sink_formatter);
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
|
||||||
mutex_t &mutex_;
|
|
||||||
FILE *file_;
|
|
||||||
};
|
|
||||||
|
|
||||||
using stdout_sink_mt = stdout_sink<details::console_stdout, details::console_mutex>;
|
|
||||||
using stdout_sink_st = stdout_sink<details::console_stdout, details::console_nullmutex>;
|
|
||||||
|
|
||||||
using stderr_sink_mt = stdout_sink<details::console_stderr, details::console_mutex>;
|
|
||||||
using stderr_sink_st = stdout_sink<details::console_stderr, details::console_nullmutex>;
|
|
||||||
|
|
||||||
} // namespace sinks
|
|
||||||
|
|
||||||
// factory methods
|
|
||||||
template<typename Factory = default_factory>
|
|
||||||
inline std::shared_ptr<logger> stdout_logger_mt(const std::string &logger_name)
|
|
||||||
{
|
|
||||||
return Factory::template create<sinks::stdout_sink_mt>(logger_name);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename Factory = default_factory>
|
|
||||||
inline std::shared_ptr<logger> stdout_logger_st(const std::string &logger_name)
|
|
||||||
{
|
|
||||||
return Factory::template create<sinks::stdout_sink_st>(logger_name);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename Factory = default_factory>
|
|
||||||
inline std::shared_ptr<logger> stderr_logger_mt(const std::string &logger_name)
|
|
||||||
{
|
|
||||||
return Factory::template create<sinks::stderr_sink_mt>(logger_name);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename Factory = default_factory>
|
|
||||||
inline std::shared_ptr<logger> stderr_logger_st(const std::string &logger_name)
|
|
||||||
{
|
|
||||||
return Factory::template create<sinks::stderr_sink_st>(logger_name);
|
|
||||||
}
|
|
||||||
} // namespace spdlog
|
|
||||||
94
deps/spdlog/sinks/syslog_sink.h
vendored
94
deps/spdlog/sinks/syslog_sink.h
vendored
@ -1,94 +0,0 @@
|
|||||||
//
|
|
||||||
// Copyright(c) 2015 Gabi Melman.
|
|
||||||
// Distributed under the MIT License (http://opensource.org/licenses/MIT)
|
|
||||||
//
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#ifndef SPDLOG_H
|
|
||||||
#include "spdlog/spdlog.h"
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#include "spdlog/sinks/base_sink.h"
|
|
||||||
|
|
||||||
#include <array>
|
|
||||||
#include <string>
|
|
||||||
#include <syslog.h>
|
|
||||||
|
|
||||||
namespace spdlog {
|
|
||||||
namespace sinks {
|
|
||||||
/**
|
|
||||||
* Sink that write to syslog using the `syscall()` library call.
|
|
||||||
*
|
|
||||||
* Locking is not needed, as `syslog()` itself is thread-safe.
|
|
||||||
*/
|
|
||||||
template<typename Mutex>
|
|
||||||
class syslog_sink : public base_sink<Mutex>
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
//
|
|
||||||
explicit syslog_sink(std::string ident = "", int syslog_option = 0, int syslog_facility = LOG_USER)
|
|
||||||
: ident_(std::move(ident))
|
|
||||||
{
|
|
||||||
priorities_[static_cast<size_t>(level::trace)] = LOG_DEBUG;
|
|
||||||
priorities_[static_cast<size_t>(level::debug)] = LOG_DEBUG;
|
|
||||||
priorities_[static_cast<size_t>(level::info)] = LOG_INFO;
|
|
||||||
priorities_[static_cast<size_t>(level::warn)] = LOG_WARNING;
|
|
||||||
priorities_[static_cast<size_t>(level::err)] = LOG_ERR;
|
|
||||||
priorities_[static_cast<size_t>(level::critical)] = LOG_CRIT;
|
|
||||||
priorities_[static_cast<size_t>(level::off)] = LOG_INFO;
|
|
||||||
|
|
||||||
// set ident to be program name if empty
|
|
||||||
::openlog(ident_.empty() ? nullptr : ident_.c_str(), syslog_option, syslog_facility);
|
|
||||||
}
|
|
||||||
|
|
||||||
~syslog_sink() override
|
|
||||||
{
|
|
||||||
::closelog();
|
|
||||||
}
|
|
||||||
|
|
||||||
syslog_sink(const syslog_sink &) = delete;
|
|
||||||
syslog_sink &operator=(const syslog_sink &) = delete;
|
|
||||||
|
|
||||||
protected:
|
|
||||||
void sink_it_(const details::log_msg &msg) override
|
|
||||||
{
|
|
||||||
::syslog(syslog_prio_from_level(msg), "%s", fmt::to_string(msg.payload).c_str());
|
|
||||||
}
|
|
||||||
|
|
||||||
void flush_() override {}
|
|
||||||
|
|
||||||
private:
|
|
||||||
std::array<int, 7> priorities_;
|
|
||||||
// must store the ident because the man says openlog might use the pointer as
|
|
||||||
// is and not a string copy
|
|
||||||
const std::string ident_;
|
|
||||||
|
|
||||||
//
|
|
||||||
// Simply maps spdlog's log level to syslog priority level.
|
|
||||||
//
|
|
||||||
int syslog_prio_from_level(const details::log_msg &msg) const
|
|
||||||
{
|
|
||||||
return priorities_[static_cast<size_t>(msg.level)];
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
using syslog_sink_mt = syslog_sink<std::mutex>;
|
|
||||||
using syslog_sink_st = syslog_sink<details::null_mutex>;
|
|
||||||
} // namespace sinks
|
|
||||||
|
|
||||||
// Create and register a syslog logger
|
|
||||||
template<typename Factory = default_factory>
|
|
||||||
inline std::shared_ptr<logger> syslog_logger_mt(
|
|
||||||
const std::string &logger_name, const std::string &syslog_ident = "", int syslog_option = 0, int syslog_facility = (1 << 3))
|
|
||||||
{
|
|
||||||
return Factory::template create<sinks::syslog_sink_mt>(logger_name, syslog_ident, syslog_option, syslog_facility);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename Factory = default_factory>
|
|
||||||
inline std::shared_ptr<logger> syslog_logger_st(
|
|
||||||
const std::string &logger_name, const std::string &syslog_ident = "", int syslog_option = 0, int syslog_facility = (1 << 3))
|
|
||||||
{
|
|
||||||
return Factory::template create<sinks::syslog_sink_st>(logger_name, syslog_ident, syslog_option, syslog_facility);
|
|
||||||
}
|
|
||||||
} // namespace spdlog
|
|
||||||
143
deps/spdlog/sinks/wincolor_sink.h
vendored
143
deps/spdlog/sinks/wincolor_sink.h
vendored
@ -1,143 +0,0 @@
|
|||||||
//
|
|
||||||
// Copyright(c) 2016 spdlog
|
|
||||||
// Distributed under the MIT License (http://opensource.org/licenses/MIT)
|
|
||||||
//
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#ifndef SPDLOG_H
|
|
||||||
#include "spdlog/spdlog.h"
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#include "spdlog/common.h"
|
|
||||||
#include "spdlog/details/console_globals.h"
|
|
||||||
#include "spdlog/details/null_mutex.h"
|
|
||||||
#include "spdlog/sinks/sink.h"
|
|
||||||
|
|
||||||
#include <memory>
|
|
||||||
#include <mutex>
|
|
||||||
#include <string>
|
|
||||||
#include <unordered_map>
|
|
||||||
#include <wincon.h>
|
|
||||||
|
|
||||||
namespace spdlog {
|
|
||||||
namespace sinks {
|
|
||||||
/*
|
|
||||||
* Windows color console sink. Uses WriteConsoleA to write to the console with
|
|
||||||
* colors
|
|
||||||
*/
|
|
||||||
template<typename OutHandle, typename ConsoleMutex>
|
|
||||||
class wincolor_sink : public sink
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
const WORD BOLD = FOREGROUND_INTENSITY;
|
|
||||||
const WORD RED = FOREGROUND_RED;
|
|
||||||
const WORD GREEN = FOREGROUND_GREEN;
|
|
||||||
const WORD CYAN = FOREGROUND_GREEN | FOREGROUND_BLUE;
|
|
||||||
const WORD WHITE = FOREGROUND_RED | FOREGROUND_GREEN | FOREGROUND_BLUE;
|
|
||||||
const WORD YELLOW = FOREGROUND_RED | FOREGROUND_GREEN;
|
|
||||||
|
|
||||||
wincolor_sink()
|
|
||||||
: out_handle_(OutHandle::handle())
|
|
||||||
, mutex_(ConsoleMutex::mutex())
|
|
||||||
{
|
|
||||||
colors_[level::trace] = WHITE;
|
|
||||||
colors_[level::debug] = CYAN;
|
|
||||||
colors_[level::info] = GREEN;
|
|
||||||
colors_[level::warn] = YELLOW | BOLD;
|
|
||||||
colors_[level::err] = RED | BOLD; // red bold
|
|
||||||
colors_[level::critical] = BACKGROUND_RED | WHITE | BOLD; // white bold on red background
|
|
||||||
colors_[level::off] = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
~wincolor_sink() override
|
|
||||||
{
|
|
||||||
this->flush();
|
|
||||||
}
|
|
||||||
|
|
||||||
wincolor_sink(const wincolor_sink &other) = delete;
|
|
||||||
wincolor_sink &operator=(const wincolor_sink &other) = delete;
|
|
||||||
|
|
||||||
// change the color for the given level
|
|
||||||
void set_color(level::level_enum level, WORD color)
|
|
||||||
{
|
|
||||||
std::lock_guard<mutex_t> lock(mutex_);
|
|
||||||
colors_[level] = color;
|
|
||||||
}
|
|
||||||
|
|
||||||
void log(const details::log_msg &msg) final override
|
|
||||||
{
|
|
||||||
std::lock_guard<mutex_t> lock(mutex_);
|
|
||||||
fmt::memory_buffer formatted;
|
|
||||||
formatter_->format(msg, formatted);
|
|
||||||
if (msg.color_range_end > msg.color_range_start)
|
|
||||||
{
|
|
||||||
// before color range
|
|
||||||
print_range_(formatted, 0, msg.color_range_start);
|
|
||||||
|
|
||||||
// in color range
|
|
||||||
auto orig_attribs = set_console_attribs(colors_[msg.level]);
|
|
||||||
print_range_(formatted, msg.color_range_start, msg.color_range_end);
|
|
||||||
::SetConsoleTextAttribute(out_handle_,
|
|
||||||
orig_attribs); // reset to orig colors
|
|
||||||
// after color range
|
|
||||||
print_range_(formatted, msg.color_range_end, formatted.size());
|
|
||||||
}
|
|
||||||
else // print without colors if color range is invalid
|
|
||||||
{
|
|
||||||
print_range_(formatted, 0, formatted.size());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void flush() final override
|
|
||||||
{
|
|
||||||
// windows console always flushed?
|
|
||||||
}
|
|
||||||
|
|
||||||
void set_pattern(const std::string &pattern) override final
|
|
||||||
{
|
|
||||||
std::lock_guard<mutex_t> lock(mutex_);
|
|
||||||
formatter_ = std::unique_ptr<spdlog::formatter>(new pattern_formatter(pattern));
|
|
||||||
}
|
|
||||||
|
|
||||||
void set_formatter(std::unique_ptr<spdlog::formatter> sink_formatter) override final
|
|
||||||
{
|
|
||||||
std::lock_guard<mutex_t> lock(mutex_);
|
|
||||||
formatter_ = std::move(sink_formatter);
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
|
||||||
using mutex_t = typename ConsoleMutex::mutex_t;
|
|
||||||
// set color and return the orig console attributes (for resetting later)
|
|
||||||
WORD set_console_attribs(WORD attribs)
|
|
||||||
{
|
|
||||||
CONSOLE_SCREEN_BUFFER_INFO orig_buffer_info;
|
|
||||||
::GetConsoleScreenBufferInfo(out_handle_, &orig_buffer_info);
|
|
||||||
WORD back_color = orig_buffer_info.wAttributes;
|
|
||||||
// retrieve the current background color
|
|
||||||
back_color &= static_cast<WORD>(~(FOREGROUND_RED | FOREGROUND_GREEN | FOREGROUND_BLUE | FOREGROUND_INTENSITY));
|
|
||||||
// keep the background color unchanged
|
|
||||||
::SetConsoleTextAttribute(out_handle_, attribs | back_color);
|
|
||||||
return orig_buffer_info.wAttributes; // return orig attribs
|
|
||||||
}
|
|
||||||
|
|
||||||
// print a range of formatted message to console
|
|
||||||
void print_range_(const fmt::memory_buffer &formatted, size_t start, size_t end)
|
|
||||||
{
|
|
||||||
auto size = static_cast<DWORD>(end - start);
|
|
||||||
::WriteConsoleA(out_handle_, formatted.data() + start, size, nullptr, nullptr);
|
|
||||||
}
|
|
||||||
|
|
||||||
HANDLE out_handle_;
|
|
||||||
mutex_t &mutex_;
|
|
||||||
std::unordered_map<level::level_enum, WORD, level::level_hasher> colors_;
|
|
||||||
};
|
|
||||||
|
|
||||||
using wincolor_stdout_sink_mt = wincolor_sink<details::console_stdout, details::console_mutex>;
|
|
||||||
using wincolor_stdout_sink_st = wincolor_sink<details::console_stdout, details::console_nullmutex>;
|
|
||||||
|
|
||||||
using wincolor_stderr_sink_mt = wincolor_sink<details::console_stderr, details::console_mutex>;
|
|
||||||
using wincolor_stderr_sink_st = wincolor_sink<details::console_stderr, details::console_nullmutex>;
|
|
||||||
|
|
||||||
} // namespace sinks
|
|
||||||
} // namespace spdlog
|
|
||||||
366
deps/spdlog/spdlog.h
vendored
366
deps/spdlog/spdlog.h
vendored
@ -1,366 +0,0 @@
|
|||||||
//
|
|
||||||
// Copyright(c) 2015-2018 Gabi Melman.
|
|
||||||
// Distributed under the MIT License (http://opensource.org/licenses/MIT)
|
|
||||||
//
|
|
||||||
// spdlog main header file.
|
|
||||||
// see example.cpp for usage example
|
|
||||||
|
|
||||||
#ifndef SPDLOG_H
|
|
||||||
#define SPDLOG_H
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include "spdlog/common.h"
|
|
||||||
#include "spdlog/details/registry.h"
|
|
||||||
#include "spdlog/logger.h"
|
|
||||||
#include "spdlog/version.h"
|
|
||||||
|
|
||||||
#include <chrono>
|
|
||||||
#include <functional>
|
|
||||||
#include <memory>
|
|
||||||
#include <string>
|
|
||||||
|
|
||||||
namespace spdlog {
|
|
||||||
|
|
||||||
// Default logger factory- creates synchronous loggers
|
|
||||||
struct synchronous_factory
|
|
||||||
{
|
|
||||||
template<typename Sink, typename... SinkArgs>
|
|
||||||
static std::shared_ptr<spdlog::logger> create(std::string logger_name, SinkArgs &&... args)
|
|
||||||
{
|
|
||||||
auto sink = std::make_shared<Sink>(std::forward<SinkArgs>(args)...);
|
|
||||||
auto new_logger = std::make_shared<logger>(std::move(logger_name), std::move(sink));
|
|
||||||
details::registry::instance().initialize_logger(new_logger);
|
|
||||||
return new_logger;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
using default_factory = synchronous_factory;
|
|
||||||
|
|
||||||
// Create and register a logger with a templated sink type
|
|
||||||
// The logger's level, formatter and flush level will be set according the
|
|
||||||
// global settings.
|
|
||||||
// Example:
|
|
||||||
// spdlog::create<daily_file_sink_st>("logger_name", "dailylog_filename", 11, 59);
|
|
||||||
template<typename Sink, typename... SinkArgs>
|
|
||||||
inline std::shared_ptr<spdlog::logger> create(std::string logger_name, SinkArgs &&... sink_args)
|
|
||||||
{
|
|
||||||
return default_factory::create<Sink>(std::move(logger_name), std::forward<SinkArgs>(sink_args)...);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Return an existing logger or nullptr if a logger with such name doesn't
|
|
||||||
// exist.
|
|
||||||
// example: spdlog::get("my_logger")->info("hello {}", "world");
|
|
||||||
inline std::shared_ptr<logger> get(const std::string &name)
|
|
||||||
{
|
|
||||||
return details::registry::instance().get(name);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Set global formatter. Each sink in each logger will get a clone of this object
|
|
||||||
inline void set_formatter(std::unique_ptr<spdlog::formatter> formatter)
|
|
||||||
{
|
|
||||||
details::registry::instance().set_formatter(std::move(formatter));
|
|
||||||
}
|
|
||||||
|
|
||||||
// Set global format string.
|
|
||||||
// example: spdlog::set_pattern("%Y-%m-%d %H:%M:%S.%e %l : %v");
|
|
||||||
inline void set_pattern(std::string pattern, pattern_time_type time_type = pattern_time_type::local)
|
|
||||||
{
|
|
||||||
set_formatter(std::unique_ptr<spdlog::formatter>(new pattern_formatter(std::move(pattern), time_type)));
|
|
||||||
}
|
|
||||||
|
|
||||||
// Set global logging level
|
|
||||||
inline void set_level(level::level_enum log_level)
|
|
||||||
{
|
|
||||||
details::registry::instance().set_level(log_level);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Set global flush level
|
|
||||||
inline void flush_on(level::level_enum log_level)
|
|
||||||
{
|
|
||||||
details::registry::instance().flush_on(log_level);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Start/Restart a periodic flusher thread
|
|
||||||
// Warning: Use only if all your loggers are thread safe!
|
|
||||||
inline void flush_every(std::chrono::seconds interval)
|
|
||||||
{
|
|
||||||
details::registry::instance().flush_every(interval);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Set global error handler
|
|
||||||
inline void set_error_handler(log_err_handler handler)
|
|
||||||
{
|
|
||||||
details::registry::instance().set_error_handler(std::move(handler));
|
|
||||||
}
|
|
||||||
|
|
||||||
// Register the given logger with the given name
|
|
||||||
inline void register_logger(std::shared_ptr<logger> logger)
|
|
||||||
{
|
|
||||||
details::registry::instance().register_logger(std::move(logger));
|
|
||||||
}
|
|
||||||
|
|
||||||
// Apply a user defined function on all registered loggers
|
|
||||||
// Example:
|
|
||||||
// spdlog::apply_all([&](std::shared_ptr<spdlog::logger> l) {l->flush();});
|
|
||||||
inline void apply_all(const std::function<void(std::shared_ptr<logger>)> &fun)
|
|
||||||
{
|
|
||||||
details::registry::instance().apply_all(fun);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Drop the reference to the given logger
|
|
||||||
inline void drop(const std::string &name)
|
|
||||||
{
|
|
||||||
details::registry::instance().drop(name);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Drop all references from the registry
|
|
||||||
inline void drop_all()
|
|
||||||
{
|
|
||||||
details::registry::instance().drop_all();
|
|
||||||
}
|
|
||||||
|
|
||||||
// stop any running threads started by spdlog and clean registry loggers
|
|
||||||
inline void shutdown()
|
|
||||||
{
|
|
||||||
details::registry::instance().shutdown();
|
|
||||||
}
|
|
||||||
|
|
||||||
// Automatic registration of loggers when using spdlog::create() or spdlog::create_async
|
|
||||||
inline void set_automatic_registration(bool automatic_registation)
|
|
||||||
{
|
|
||||||
details::registry::instance().set_automatic_registration(automatic_registation);
|
|
||||||
}
|
|
||||||
|
|
||||||
// API for using default logger (stdout_color_mt),
|
|
||||||
// e.g: spdlog::info("Message {}", 1);
|
|
||||||
//
|
|
||||||
// The default logger object can be accessed using the spdlog::default_logger():
|
|
||||||
// For example, to add another sink to it:
|
|
||||||
// spdlog::default_logger()->sinks()->push_back(some_sink);
|
|
||||||
//
|
|
||||||
// The default logger can replaced using spdlog::set_default_logger(new_logger).
|
|
||||||
// For example, to replace it with a file logger.
|
|
||||||
//
|
|
||||||
// IMPORTANT:
|
|
||||||
// The default API is thread safe (for _mt loggers), but:
|
|
||||||
// set_default_logger() *should not* be used concurrently with the default API.
|
|
||||||
// e.g do not call set_default_logger() from one thread while calling spdlog::info() from another.
|
|
||||||
|
|
||||||
inline std::shared_ptr<spdlog::logger> default_logger()
|
|
||||||
{
|
|
||||||
return details::registry::instance().default_logger();
|
|
||||||
}
|
|
||||||
|
|
||||||
inline spdlog::logger *default_logger_raw()
|
|
||||||
{
|
|
||||||
return details::registry::instance().get_default_raw();
|
|
||||||
}
|
|
||||||
|
|
||||||
inline void set_default_logger(std::shared_ptr<spdlog::logger> default_logger)
|
|
||||||
{
|
|
||||||
details::registry::instance().set_default_logger(std::move(default_logger));
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename... Args>
|
|
||||||
inline void log(source_loc source, level::level_enum lvl, const char *fmt, const Args &... args)
|
|
||||||
{
|
|
||||||
default_logger_raw()->log(source, lvl, fmt, args...);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename... Args>
|
|
||||||
inline void log(level::level_enum lvl, const char *fmt, const Args &... args)
|
|
||||||
{
|
|
||||||
default_logger_raw()->log(source_loc{}, lvl, fmt, args...);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename... Args>
|
|
||||||
inline void trace(const char *fmt, const Args &... args)
|
|
||||||
{
|
|
||||||
default_logger_raw()->trace(fmt, args...);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename... Args>
|
|
||||||
inline void debug(const char *fmt, const Args &... args)
|
|
||||||
{
|
|
||||||
default_logger_raw()->debug(fmt, args...);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename... Args>
|
|
||||||
inline void info(const char *fmt, const Args &... args)
|
|
||||||
{
|
|
||||||
default_logger_raw()->info(fmt, args...);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename... Args>
|
|
||||||
inline void warn(const char *fmt, const Args &... args)
|
|
||||||
{
|
|
||||||
default_logger_raw()->warn(fmt, args...);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename... Args>
|
|
||||||
inline void error(const char *fmt, const Args &... args)
|
|
||||||
{
|
|
||||||
default_logger_raw()->error(fmt, args...);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename... Args>
|
|
||||||
inline void critical(const char *fmt, const Args &... args)
|
|
||||||
{
|
|
||||||
default_logger_raw()->critical(fmt, args...);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename T>
|
|
||||||
inline void log(level::level_enum lvl, const T &msg)
|
|
||||||
{
|
|
||||||
default_logger_raw()->log(lvl, msg);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename T>
|
|
||||||
inline void trace(const T &msg)
|
|
||||||
{
|
|
||||||
default_logger_raw()->trace(msg);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename T>
|
|
||||||
inline void debug(const T &msg)
|
|
||||||
{
|
|
||||||
default_logger_raw()->debug(msg);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename T>
|
|
||||||
inline void info(const T &msg)
|
|
||||||
{
|
|
||||||
default_logger_raw()->info(msg);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename T>
|
|
||||||
inline void warn(const T &msg)
|
|
||||||
{
|
|
||||||
default_logger_raw()->warn(msg);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename T>
|
|
||||||
inline void error(const T &msg)
|
|
||||||
{
|
|
||||||
default_logger_raw()->error(msg);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename T>
|
|
||||||
inline void critical(const T &msg)
|
|
||||||
{
|
|
||||||
default_logger_raw()->critical(msg);
|
|
||||||
}
|
|
||||||
|
|
||||||
#ifdef SPDLOG_WCHAR_TO_UTF8_SUPPORT
|
|
||||||
template<typename... Args>
|
|
||||||
inline void log(level::level_enum lvl, const wchar_t *fmt, const Args &... args)
|
|
||||||
{
|
|
||||||
default_logger_raw()->log(lvl, fmt, args...);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename... Args>
|
|
||||||
inline void trace(const wchar_t *fmt, const Args &... args)
|
|
||||||
{
|
|
||||||
default_logger_raw()->trace(fmt, args...);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename... Args>
|
|
||||||
inline void debug(const wchar_t *fmt, const Args &... args)
|
|
||||||
{
|
|
||||||
default_logger_raw()->debug(fmt, args...);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename... Args>
|
|
||||||
inline void info(const wchar_t *fmt, const Args &... args)
|
|
||||||
{
|
|
||||||
default_logger_raw()->info(fmt, args...);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename... Args>
|
|
||||||
inline void warn(const wchar_t *fmt, const Args &... args)
|
|
||||||
{
|
|
||||||
default_logger_raw()->warn(fmt, args...);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename... Args>
|
|
||||||
inline void error(const wchar_t *fmt, const Args &... args)
|
|
||||||
{
|
|
||||||
default_logger_raw()->error(fmt, args...);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<typename... Args>
|
|
||||||
inline void critical(const wchar_t *fmt, const Args &... args)
|
|
||||||
{
|
|
||||||
default_logger_raw()->critical(fmt, args...);
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif // SPDLOG_WCHAR_TO_UTF8_SUPPORT
|
|
||||||
|
|
||||||
} // namespace spdlog
|
|
||||||
|
|
||||||
//
|
|
||||||
// enable/disable log calls at compile time according to global level.
|
|
||||||
//
|
|
||||||
// define SPDLOG_ACTIVE_LEVEL to one of those (before including spdlog.h):
|
|
||||||
// SPDLOG_LEVEL_TRACE,
|
|
||||||
// SPDLOG_LEVEL_DEBUG,
|
|
||||||
// SPDLOG_LEVEL_INFO,
|
|
||||||
// SPDLOG_LEVEL_WARN,
|
|
||||||
// SPDLOG_LEVEL_ERROR,
|
|
||||||
// SPDLOG_LEVEL_CRITICAL,
|
|
||||||
// SPDLOG_LEVEL_OFF
|
|
||||||
//
|
|
||||||
|
|
||||||
#define SPDLOG_LOGGER_CALL(logger, level, ...) \
|
|
||||||
if (logger->should_log(level)) \
|
|
||||||
logger->log(spdlog::source_loc{SPDLOG_FILE_BASENAME(__FILE__), __LINE__, SPDLOG_FUNCTION}, level, __VA_ARGS__)
|
|
||||||
|
|
||||||
#if SPDLOG_ACTIVE_LEVEL <= SPDLOG_LEVEL_TRACE
|
|
||||||
#define SPDLOG_LOGGER_TRACE(logger, ...) SPDLOG_LOGGER_CALL(logger, spdlog::level::trace, __VA_ARGS__)
|
|
||||||
#define SPDLOG_TRACE(...) SPDLOG_LOGGER_TRACE(spdlog::default_logger_raw(), __VA_ARGS__)
|
|
||||||
#else
|
|
||||||
#define SPDLOG_LOGGER_TRACE(logger, ...) (void)0
|
|
||||||
#define SPDLOG_TRACE(...) (void)0
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#if SPDLOG_ACTIVE_LEVEL <= SPDLOG_LEVEL_DEBUG
|
|
||||||
#define SPDLOG_LOGGER_DEBUG(logger, ...) SPDLOG_LOGGER_CALL(logger, spdlog::level::debug, __VA_ARGS__)
|
|
||||||
#define SPDLOG_DEBUG(...) SPDLOG_LOGGER_DEBUG(spdlog::default_logger_raw(), __VA_ARGS__)
|
|
||||||
#else
|
|
||||||
#define SPDLOG_LOGGER_DEBUG(logger, ...) (void)0
|
|
||||||
#define SPDLOG_DEBUG(...) (void)0
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#if SPDLOG_ACTIVE_LEVEL <= SPDLOG_LEVEL_INFO
|
|
||||||
#define SPDLOG_LOGGER_INFO(logger, ...) SPDLOG_LOGGER_CALL(logger, spdlog::level::info, __VA_ARGS__)
|
|
||||||
#define SPDLOG_INFO(...) SPDLOG_LOGGER_INFO(spdlog::default_logger_raw(), __VA_ARGS__)
|
|
||||||
#else
|
|
||||||
#define SPDLOG_LOGGER_INFO(logger, ...) (void)0
|
|
||||||
#define SPDLOG_INFO(...) (void)0
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#if SPDLOG_ACTIVE_LEVEL <= SPDLOG_LEVEL_WARN
|
|
||||||
#define SPDLOG_LOGGER_WARN(logger, ...) SPDLOG_LOGGER_CALL(logger, spdlog::level::warn, __VA_ARGS__)
|
|
||||||
#define SPDLOG_WARN(...) SPDLOG_LOGGER_WARN(spdlog::default_logger_raw(), __VA_ARGS__)
|
|
||||||
#else
|
|
||||||
#define SPDLOG_LOGGER_WARN(logger, ...) (void)0
|
|
||||||
#define SPDLOG_WARN(...) (void)0
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#if SPDLOG_ACTIVE_LEVEL <= SPDLOG_LEVEL_ERROR
|
|
||||||
#define SPDLOG_LOGGER_ERROR(logger, ...) SPDLOG_LOGGER_CALL(logger, spdlog::level::err, __VA_ARGS__)
|
|
||||||
#define SPDLOG_ERROR(...) SPDLOG_LOGGER_ERROR(spdlog::default_logger_raw(), __VA_ARGS__)
|
|
||||||
#else
|
|
||||||
#define SPDLOG_LOGGER_ERROR(logger, ...) (void)0
|
|
||||||
#define SPDLOG_ERROR(...) (void)0
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#if SPDLOG_ACTIVE_LEVEL <= SPDLOG_LEVEL_CRITICAL
|
|
||||||
#define SPDLOG_LOGGER_CRITICAL(logger, ...) SPDLOG_LOGGER_CALL(logger, spdlog::level::critical, __VA_ARGS__)
|
|
||||||
#define SPDLOG_CRITICAL(...) SPDLOG_LOGGER_CRITICAL(spdlog::default_logger_raw(), __VA_ARGS__)
|
|
||||||
#else
|
|
||||||
#define SPDLOG_LOGGER_CRITICAL(logger, ...) (void)0
|
|
||||||
#define SPDLOG_CRITICAL(...) (void)0
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#endif // SPDLOG_H
|
|
||||||
145
deps/spdlog/tweakme.h
vendored
145
deps/spdlog/tweakme.h
vendored
@ -1,145 +0,0 @@
|
|||||||
//
|
|
||||||
// Copyright(c) 2015 Gabi Melman.
|
|
||||||
// Distributed under the MIT License (http://opensource.org/licenses/MIT)
|
|
||||||
//
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
//
|
|
||||||
// Edit this file to squeeze more performance, and to customize supported
|
|
||||||
// features
|
|
||||||
//
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
// Under Linux, the much faster CLOCK_REALTIME_COARSE clock can be used.
|
|
||||||
// This clock is less accurate - can be off by dozens of millis - depending on
|
|
||||||
// the kernel HZ.
|
|
||||||
// Uncomment to use it instead of the regular clock.
|
|
||||||
//
|
|
||||||
// #define SPDLOG_CLOCK_COARSE
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
// Uncomment if date/time logging is not needed and never appear in the log
|
|
||||||
// pattern.
|
|
||||||
// This will prevent spdlog from querying the clock on each log call.
|
|
||||||
//
|
|
||||||
// WARNING: If the log pattern contains any date/time while this flag is on, the
|
|
||||||
// result is undefined.
|
|
||||||
// You must set new pattern(spdlog::set_pattern(..") without any
|
|
||||||
// date/time in it
|
|
||||||
//
|
|
||||||
// #define SPDLOG_NO_DATETIME
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
// Uncomment if thread id logging is not needed (i.e. no %t in the log pattern).
|
|
||||||
// This will prevent spdlog from querying the thread id on each log call.
|
|
||||||
//
|
|
||||||
// WARNING: If the log pattern contains thread id (i.e, %t) while this flag is
|
|
||||||
// on, the result is undefined.
|
|
||||||
//
|
|
||||||
// #define SPDLOG_NO_THREAD_ID
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
// Uncomment to prevent spdlog from using thread local storage.
|
|
||||||
//
|
|
||||||
// WARNING: if your program forks, UNCOMMENT this flag to prevent undefined
|
|
||||||
// thread ids in the children logs.
|
|
||||||
//
|
|
||||||
// #define SPDLOG_NO_TLS
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
// Uncomment if logger name logging is not needed.
|
|
||||||
// This will prevent spdlog from copying the logger name on each log call.
|
|
||||||
//
|
|
||||||
// #define SPDLOG_NO_NAME
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
// Uncomment to enable the SPDLOG_DEBUG/SPDLOG_TRACE macros.
|
|
||||||
//
|
|
||||||
// #define SPDLOG_DEBUG_ON
|
|
||||||
// #define SPDLOG_TRACE_ON
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
// Uncomment to avoid spdlog's usage of atomic log levels
|
|
||||||
// Use only if your code never modifies a logger's log levels concurrently by
|
|
||||||
// different threads.
|
|
||||||
//
|
|
||||||
// #define SPDLOG_NO_ATOMIC_LEVELS
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
// Uncomment to enable usage of wchar_t for file names on Windows.
|
|
||||||
//
|
|
||||||
// #define SPDLOG_WCHAR_FILENAMES
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
// Uncomment to override default eol ("\n" or "\r\n" under Linux/Windows)
|
|
||||||
//
|
|
||||||
// #define SPDLOG_EOL ";-)\n"
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
// Uncomment to use your own copy of the fmt library instead of spdlog's copy.
|
|
||||||
// In this case spdlog will try to include <fmt/format.h> so set your -I flag
|
|
||||||
// accordingly.
|
|
||||||
//
|
|
||||||
// #define SPDLOG_FMT_EXTERNAL
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
// Uncomment to enable wchar_t support (convert to utf8)
|
|
||||||
//
|
|
||||||
// #define SPDLOG_WCHAR_TO_UTF8_SUPPORT
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
// Uncomment to prevent child processes from inheriting log file descriptors
|
|
||||||
//
|
|
||||||
// #define SPDLOG_PREVENT_CHILD_FD
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
// Uncomment to enable message counting feature.
|
|
||||||
// Use the %i in the logger pattern to display log message sequence id.
|
|
||||||
//
|
|
||||||
// #define SPDLOG_ENABLE_MESSAGE_COUNTER
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
// Uncomment to customize level names (e.g. "MT TRACE")
|
|
||||||
//
|
|
||||||
// #define SPDLOG_LEVEL_NAMES { "MY TRACE", "MY DEBUG", "MY INFO", "MY WARNING",
|
|
||||||
// "MY ERROR", "MY CRITICAL", "OFF" }
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
// Uncomment to disable default logger creation.
|
|
||||||
// This might save some (very) small initialization time if no default logger is needed.
|
|
||||||
//
|
|
||||||
// #define SPDLOG_DISABLE_DEFAULT_LOGGER
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
// Uncomment and set to compile time level with zero cost (default is INFO).
|
|
||||||
// Macros like SPDLOG_DEBUG(..), SPDLOG_INFO(..) will expand to empty statements if not enabled
|
|
||||||
//
|
|
||||||
// #define SPDLOG_ACTIVE_LEVEL SPDLOG_LEVEL_INFO
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
// Uncomment (and change if desired) macro to use for function names.
|
|
||||||
// This is compiler dependent.
|
|
||||||
// __PRETTY_FUNCTION__ might be nicer in clang/gcc, and __FUNCTION__ in msvc.
|
|
||||||
// Defaults to __FUNCTION__ (should work on all compilers) if not defined.
|
|
||||||
//
|
|
||||||
// #define SPDLOG_FUNCTION __PRETTY_FUNCTION__
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
12
deps/spdlog/version.h
vendored
12
deps/spdlog/version.h
vendored
@ -1,12 +0,0 @@
|
|||||||
//
|
|
||||||
// Copyright(c) 2015 Gabi Melman.
|
|
||||||
// Distributed under the MIT License (http://opensource.org/licenses/MIT)
|
|
||||||
//
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#define SPDLOG_VER_MAJOR 1
|
|
||||||
#define SPDLOG_VER_MINOR 3
|
|
||||||
#define SPDLOG_VER_PATCH 1
|
|
||||||
|
|
||||||
#define SPDLOG_VERSION (SPDLOG_VER_MAJOR * 10000 + SPDLOG_VER_MINOR * 100 + SPDLOG_VER_PATCH)
|
|
||||||
183
deps/yoga/CompactValue.h
vendored
183
deps/yoga/CompactValue.h
vendored
@ -1,183 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) Facebook, Inc. and its affiliates.
|
|
||||||
*
|
|
||||||
* This source code is licensed under the MIT license found in the LICENSE
|
|
||||||
* file in the root directory of this source tree.
|
|
||||||
*/
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include "YGValue.h"
|
|
||||||
|
|
||||||
#include <cmath>
|
|
||||||
#include <cstdint>
|
|
||||||
#include <limits>
|
|
||||||
|
|
||||||
static_assert(
|
|
||||||
std::numeric_limits<float>::is_iec559,
|
|
||||||
"facebook::yoga::detail::CompactValue only works with IEEE754 floats");
|
|
||||||
|
|
||||||
#ifdef YOGA_COMPACT_VALUE_TEST
|
|
||||||
#define VISIBLE_FOR_TESTING public:
|
|
||||||
#else
|
|
||||||
#define VISIBLE_FOR_TESTING private:
|
|
||||||
#endif
|
|
||||||
|
|
||||||
namespace facebook {
|
|
||||||
namespace yoga {
|
|
||||||
namespace detail {
|
|
||||||
|
|
||||||
// This class stores YGValue in 32 bits.
|
|
||||||
// - The value does not matter for Undefined and Auto. NaNs are used for their
|
|
||||||
// representation.
|
|
||||||
// - To differentiate between Point and Percent, one exponent bit is used.
|
|
||||||
// Supported the range [0x40, 0xbf] (0xbf is inclusive for point, but
|
|
||||||
// exclusive for percent).
|
|
||||||
// - Value ranges:
|
|
||||||
// points: 1.08420217e-19f to 36893485948395847680
|
|
||||||
// 0x00000000 0x3fffffff
|
|
||||||
// percent: 1.08420217e-19f to 18446742974197923840
|
|
||||||
// 0x40000000 0x7f7fffff
|
|
||||||
// - Zero is supported, negative zero is not
|
|
||||||
// - values outside of the representable range are clamped
|
|
||||||
class CompactValue {
|
|
||||||
friend constexpr bool operator==(CompactValue, CompactValue) noexcept;
|
|
||||||
|
|
||||||
public:
|
|
||||||
static constexpr auto LOWER_BOUND = 1.08420217e-19f;
|
|
||||||
static constexpr auto UPPER_BOUND_POINT = 36893485948395847680.0f;
|
|
||||||
static constexpr auto UPPER_BOUND_PERCENT = 18446742974197923840.0f;
|
|
||||||
|
|
||||||
template <YGUnit Unit>
|
|
||||||
static CompactValue of(float value) noexcept {
|
|
||||||
if (value == 0.0f || (value < LOWER_BOUND && value > -LOWER_BOUND)) {
|
|
||||||
constexpr auto zero =
|
|
||||||
Unit == YGUnitPercent ? ZERO_BITS_PERCENT : ZERO_BITS_POINT;
|
|
||||||
return {Payload{zero}};
|
|
||||||
}
|
|
||||||
|
|
||||||
constexpr auto upperBound =
|
|
||||||
Unit == YGUnitPercent ? UPPER_BOUND_PERCENT : UPPER_BOUND_POINT;
|
|
||||||
if (value > upperBound || value < -upperBound) {
|
|
||||||
value = copysignf(upperBound, value);
|
|
||||||
}
|
|
||||||
|
|
||||||
uint32_t unitBit = Unit == YGUnitPercent ? PERCENT_BIT : 0;
|
|
||||||
auto data = Payload{value};
|
|
||||||
data.repr -= BIAS;
|
|
||||||
data.repr |= unitBit;
|
|
||||||
return {data};
|
|
||||||
}
|
|
||||||
|
|
||||||
template <YGUnit Unit>
|
|
||||||
static CompactValue ofMaybe(float value) noexcept {
|
|
||||||
return std::isnan(value) || std::isinf(value) ? ofUndefined()
|
|
||||||
: of<Unit>(value);
|
|
||||||
}
|
|
||||||
|
|
||||||
static constexpr CompactValue ofZero() noexcept {
|
|
||||||
return CompactValue{Payload{ZERO_BITS_POINT}};
|
|
||||||
}
|
|
||||||
|
|
||||||
static constexpr CompactValue ofUndefined() noexcept {
|
|
||||||
return CompactValue{};
|
|
||||||
}
|
|
||||||
|
|
||||||
static constexpr CompactValue ofAuto() noexcept {
|
|
||||||
return CompactValue{Payload{AUTO_BITS}};
|
|
||||||
}
|
|
||||||
|
|
||||||
constexpr CompactValue() noexcept
|
|
||||||
: payload_(std::numeric_limits<float>::quiet_NaN()) {}
|
|
||||||
|
|
||||||
CompactValue(const YGValue& x) noexcept : payload_(uint32_t{0}) {
|
|
||||||
switch (x.unit) {
|
|
||||||
case YGUnitUndefined:
|
|
||||||
*this = ofUndefined();
|
|
||||||
break;
|
|
||||||
case YGUnitAuto:
|
|
||||||
*this = ofAuto();
|
|
||||||
break;
|
|
||||||
case YGUnitPoint:
|
|
||||||
*this = of<YGUnitPoint>(x.value);
|
|
||||||
break;
|
|
||||||
case YGUnitPercent:
|
|
||||||
*this = of<YGUnitPercent>(x.value);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
operator YGValue() const noexcept {
|
|
||||||
switch (payload_.repr) {
|
|
||||||
case AUTO_BITS:
|
|
||||||
return YGValueAuto;
|
|
||||||
case ZERO_BITS_POINT:
|
|
||||||
return YGValue{0.0f, YGUnitPoint};
|
|
||||||
case ZERO_BITS_PERCENT:
|
|
||||||
return YGValue{0.0f, YGUnitPercent};
|
|
||||||
}
|
|
||||||
|
|
||||||
if (std::isnan(payload_.value)) {
|
|
||||||
return YGValueUndefined;
|
|
||||||
}
|
|
||||||
|
|
||||||
auto data = payload_;
|
|
||||||
data.repr &= ~PERCENT_BIT;
|
|
||||||
data.repr += BIAS;
|
|
||||||
|
|
||||||
return YGValue{data.value,
|
|
||||||
payload_.repr & 0x40000000 ? YGUnitPercent : YGUnitPoint};
|
|
||||||
}
|
|
||||||
|
|
||||||
bool isUndefined() const noexcept {
|
|
||||||
return (
|
|
||||||
payload_.repr != AUTO_BITS && payload_.repr != ZERO_BITS_POINT &&
|
|
||||||
payload_.repr != ZERO_BITS_PERCENT && std::isnan(payload_.value));
|
|
||||||
}
|
|
||||||
|
|
||||||
bool isAuto() const noexcept { return payload_.repr == AUTO_BITS; }
|
|
||||||
|
|
||||||
private:
|
|
||||||
union Payload {
|
|
||||||
float value;
|
|
||||||
uint32_t repr;
|
|
||||||
Payload() = delete;
|
|
||||||
constexpr Payload(uint32_t r) : repr(r) {}
|
|
||||||
constexpr Payload(float v) : value(v) {}
|
|
||||||
};
|
|
||||||
|
|
||||||
static constexpr uint32_t BIAS = 0x20000000;
|
|
||||||
static constexpr uint32_t PERCENT_BIT = 0x40000000;
|
|
||||||
|
|
||||||
// these are signaling NaNs with specific bit pattern as payload they will be
|
|
||||||
// silenced whenever going through an FPU operation on ARM + x86
|
|
||||||
static constexpr uint32_t AUTO_BITS = 0x7faaaaaa;
|
|
||||||
static constexpr uint32_t ZERO_BITS_POINT = 0x7f8f0f0f;
|
|
||||||
static constexpr uint32_t ZERO_BITS_PERCENT = 0x7f80f0f0;
|
|
||||||
|
|
||||||
constexpr CompactValue(Payload data) noexcept : payload_(data) {}
|
|
||||||
|
|
||||||
Payload payload_;
|
|
||||||
|
|
||||||
VISIBLE_FOR_TESTING uint32_t repr() { return payload_.repr; }
|
|
||||||
};
|
|
||||||
|
|
||||||
template <>
|
|
||||||
CompactValue CompactValue::of<YGUnitUndefined>(float) noexcept = delete;
|
|
||||||
template <>
|
|
||||||
CompactValue CompactValue::of<YGUnitAuto>(float) noexcept = delete;
|
|
||||||
template <>
|
|
||||||
CompactValue CompactValue::ofMaybe<YGUnitUndefined>(float) noexcept = delete;
|
|
||||||
template <>
|
|
||||||
CompactValue CompactValue::ofMaybe<YGUnitAuto>(float) noexcept = delete;
|
|
||||||
|
|
||||||
constexpr bool operator==(CompactValue a, CompactValue b) noexcept {
|
|
||||||
return a.payload_.repr == b.payload_.repr;
|
|
||||||
}
|
|
||||||
|
|
||||||
constexpr bool operator!=(CompactValue a, CompactValue b) noexcept {
|
|
||||||
return !(a == b);
|
|
||||||
}
|
|
||||||
|
|
||||||
} // namespace detail
|
|
||||||
} // namespace yoga
|
|
||||||
} // namespace facebook
|
|
||||||
66
deps/yoga/Utils.cpp
vendored
66
deps/yoga/Utils.cpp
vendored
@ -1,66 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) Facebook, Inc. and its affiliates.
|
|
||||||
*
|
|
||||||
* This source code is licensed under the MIT license found in the LICENSE
|
|
||||||
* file in the root directory of this source tree.
|
|
||||||
*/
|
|
||||||
#include "Utils.h"
|
|
||||||
|
|
||||||
using namespace facebook;
|
|
||||||
|
|
||||||
YGFlexDirection YGFlexDirectionCross(
|
|
||||||
const YGFlexDirection flexDirection,
|
|
||||||
const YGDirection direction) {
|
|
||||||
return YGFlexDirectionIsColumn(flexDirection)
|
|
||||||
? YGResolveFlexDirection(YGFlexDirectionRow, direction)
|
|
||||||
: YGFlexDirectionColumn;
|
|
||||||
}
|
|
||||||
|
|
||||||
float YGFloatMax(const float a, const float b) {
|
|
||||||
if (!yoga::isUndefined(a) && !yoga::isUndefined(b)) {
|
|
||||||
return fmaxf(a, b);
|
|
||||||
}
|
|
||||||
return yoga::isUndefined(a) ? b : a;
|
|
||||||
}
|
|
||||||
|
|
||||||
float YGFloatMin(const float a, const float b) {
|
|
||||||
if (!yoga::isUndefined(a) && !yoga::isUndefined(b)) {
|
|
||||||
return fminf(a, b);
|
|
||||||
}
|
|
||||||
|
|
||||||
return yoga::isUndefined(a) ? b : a;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool YGValueEqual(const YGValue& a, const YGValue& b) {
|
|
||||||
if (a.unit != b.unit) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (a.unit == YGUnitUndefined ||
|
|
||||||
(yoga::isUndefined(a.value) && yoga::isUndefined(b.value))) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
return fabs(a.value - b.value) < 0.0001f;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool YGFloatsEqual(const float a, const float b) {
|
|
||||||
if (!yoga::isUndefined(a) && !yoga::isUndefined(b)) {
|
|
||||||
return fabs(a - b) < 0.0001f;
|
|
||||||
}
|
|
||||||
return yoga::isUndefined(a) && yoga::isUndefined(b);
|
|
||||||
}
|
|
||||||
|
|
||||||
float YGFloatSanitize(const float val) {
|
|
||||||
return yoga::isUndefined(val) ? 0 : val;
|
|
||||||
}
|
|
||||||
|
|
||||||
YGFloatOptional YGFloatOptionalMax(YGFloatOptional op1, YGFloatOptional op2) {
|
|
||||||
if (op1 >= op2) {
|
|
||||||
return op1;
|
|
||||||
}
|
|
||||||
if (op2 > op1) {
|
|
||||||
return op2;
|
|
||||||
}
|
|
||||||
return op1.isUndefined() ? op2 : op1;
|
|
||||||
}
|
|
||||||
142
deps/yoga/Utils.h
vendored
142
deps/yoga/Utils.h
vendored
@ -1,142 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) Facebook, Inc. and its affiliates.
|
|
||||||
*
|
|
||||||
* This source code is licensed under the MIT license found in the LICENSE
|
|
||||||
* file in the root directory of this source tree.
|
|
||||||
*/
|
|
||||||
#pragma once
|
|
||||||
#include "YGNode.h"
|
|
||||||
#include "Yoga-internal.h"
|
|
||||||
#include "CompactValue.h"
|
|
||||||
|
|
||||||
// This struct is an helper model to hold the data for step 4 of flexbox algo,
|
|
||||||
// which is collecting the flex items in a line.
|
|
||||||
//
|
|
||||||
// - itemsOnLine: Number of items which can fit in a line considering the
|
|
||||||
// available Inner dimension, the flex items computed flexbasis and their
|
|
||||||
// margin. It may be different than the difference between start and end
|
|
||||||
// indicates because we skip over absolute-positioned items.
|
|
||||||
//
|
|
||||||
// - sizeConsumedOnCurrentLine: It is accumulation of the dimensions and margin
|
|
||||||
// of all the children on the current line. This will be used in order to
|
|
||||||
// either set the dimensions of the node if none already exist or to compute
|
|
||||||
// the remaining space left for the flexible children.
|
|
||||||
//
|
|
||||||
// - totalFlexGrowFactors: total flex grow factors of flex items which are to be
|
|
||||||
// layed in the current line
|
|
||||||
//
|
|
||||||
// - totalFlexShrinkFactors: total flex shrink factors of flex items which are
|
|
||||||
// to be layed in the current line
|
|
||||||
//
|
|
||||||
// - endOfLineIndex: Its the end index of the last flex item which was examined
|
|
||||||
// and it may or may not be part of the current line(as it may be absolutely
|
|
||||||
// positioned or including it may have caused to overshoot availableInnerDim)
|
|
||||||
//
|
|
||||||
// - relativeChildren: Maintain a vector of the child nodes that can shrink
|
|
||||||
// and/or grow.
|
|
||||||
|
|
||||||
struct YGCollectFlexItemsRowValues {
|
|
||||||
uint32_t itemsOnLine;
|
|
||||||
float sizeConsumedOnCurrentLine;
|
|
||||||
float totalFlexGrowFactors;
|
|
||||||
float totalFlexShrinkScaledFactors;
|
|
||||||
uint32_t endOfLineIndex;
|
|
||||||
std::vector<YGNodeRef> relativeChildren;
|
|
||||||
float remainingFreeSpace;
|
|
||||||
// The size of the mainDim for the row after considering size, padding, margin
|
|
||||||
// and border of flex items. This is used to calculate maxLineDim after going
|
|
||||||
// through all the rows to decide on the main axis size of owner.
|
|
||||||
float mainDim;
|
|
||||||
// The size of the crossDim for the row after considering size, padding,
|
|
||||||
// margin and border of flex items. Used for calculating containers crossSize.
|
|
||||||
float crossDim;
|
|
||||||
};
|
|
||||||
|
|
||||||
bool YGValueEqual(const YGValue& a, const YGValue& b);
|
|
||||||
inline bool YGValueEqual(
|
|
||||||
facebook::yoga::detail::CompactValue a,
|
|
||||||
facebook::yoga::detail::CompactValue b) {
|
|
||||||
return YGValueEqual((YGValue) a, (YGValue) b);
|
|
||||||
}
|
|
||||||
|
|
||||||
// This custom float equality function returns true if either absolute
|
|
||||||
// difference between two floats is less than 0.0001f or both are undefined.
|
|
||||||
bool YGFloatsEqual(const float a, const float b);
|
|
||||||
|
|
||||||
float YGFloatMax(const float a, const float b);
|
|
||||||
|
|
||||||
YGFloatOptional YGFloatOptionalMax(
|
|
||||||
const YGFloatOptional op1,
|
|
||||||
const YGFloatOptional op2);
|
|
||||||
|
|
||||||
float YGFloatMin(const float a, const float b);
|
|
||||||
|
|
||||||
// This custom float comparison function compares the array of float with
|
|
||||||
// YGFloatsEqual, as the default float comparison operator will not work(Look
|
|
||||||
// at the comments of YGFloatsEqual function).
|
|
||||||
template <std::size_t size>
|
|
||||||
bool YGFloatArrayEqual(
|
|
||||||
const std::array<float, size>& val1,
|
|
||||||
const std::array<float, size>& val2) {
|
|
||||||
bool areEqual = true;
|
|
||||||
for (std::size_t i = 0; i < size && areEqual; ++i) {
|
|
||||||
areEqual = YGFloatsEqual(val1[i], val2[i]);
|
|
||||||
}
|
|
||||||
return areEqual;
|
|
||||||
}
|
|
||||||
|
|
||||||
// This function returns 0 if YGFloatIsUndefined(val) is true and val otherwise
|
|
||||||
float YGFloatSanitize(const float val);
|
|
||||||
|
|
||||||
YGFlexDirection YGFlexDirectionCross(
|
|
||||||
const YGFlexDirection flexDirection,
|
|
||||||
const YGDirection direction);
|
|
||||||
|
|
||||||
inline bool YGFlexDirectionIsRow(const YGFlexDirection flexDirection) {
|
|
||||||
return flexDirection == YGFlexDirectionRow ||
|
|
||||||
flexDirection == YGFlexDirectionRowReverse;
|
|
||||||
}
|
|
||||||
|
|
||||||
inline YGFloatOptional YGResolveValue(
|
|
||||||
const YGValue value,
|
|
||||||
const float ownerSize) {
|
|
||||||
switch (value.unit) {
|
|
||||||
case YGUnitPoint:
|
|
||||||
return YGFloatOptional{value.value};
|
|
||||||
case YGUnitPercent:
|
|
||||||
return YGFloatOptional{value.value * ownerSize * 0.01f};
|
|
||||||
default:
|
|
||||||
return YGFloatOptional{};
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
inline YGFloatOptional YGResolveValue(
|
|
||||||
yoga::detail::CompactValue value,
|
|
||||||
float ownerSize) {
|
|
||||||
return YGResolveValue((YGValue) value, ownerSize);
|
|
||||||
}
|
|
||||||
|
|
||||||
inline bool YGFlexDirectionIsColumn(const YGFlexDirection flexDirection) {
|
|
||||||
return flexDirection == YGFlexDirectionColumn ||
|
|
||||||
flexDirection == YGFlexDirectionColumnReverse;
|
|
||||||
}
|
|
||||||
|
|
||||||
inline YGFlexDirection YGResolveFlexDirection(
|
|
||||||
const YGFlexDirection flexDirection,
|
|
||||||
const YGDirection direction) {
|
|
||||||
if (direction == YGDirectionRTL) {
|
|
||||||
if (flexDirection == YGFlexDirectionRow) {
|
|
||||||
return YGFlexDirectionRowReverse;
|
|
||||||
} else if (flexDirection == YGFlexDirectionRowReverse) {
|
|
||||||
return YGFlexDirectionRow;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return flexDirection;
|
|
||||||
}
|
|
||||||
|
|
||||||
inline YGFloatOptional YGResolveValueMargin(
|
|
||||||
yoga::detail::CompactValue value,
|
|
||||||
const float ownerSize) {
|
|
||||||
return value.isAuto() ? YGFloatOptional{0} : YGResolveValue(value, ownerSize);
|
|
||||||
}
|
|
||||||
43
deps/yoga/YGConfig.cpp
vendored
43
deps/yoga/YGConfig.cpp
vendored
@ -1,43 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) Facebook, Inc. and its affiliates.
|
|
||||||
*
|
|
||||||
* This source code is licensed under the MIT license found in the LICENSE
|
|
||||||
* file in the root directory of this source tree.
|
|
||||||
*/
|
|
||||||
#include "YGConfig.h"
|
|
||||||
|
|
||||||
YGConfig::YGConfig(YGLogger logger) : cloneNodeCallback_{nullptr} {
|
|
||||||
logger_.noContext = logger;
|
|
||||||
loggerUsesContext_ = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
void YGConfig::log(
|
|
||||||
YGConfig* config,
|
|
||||||
YGNode* node,
|
|
||||||
YGLogLevel logLevel,
|
|
||||||
void* logContext,
|
|
||||||
const char* format,
|
|
||||||
va_list args) {
|
|
||||||
if (loggerUsesContext_) {
|
|
||||||
logger_.withContext(config, node, logLevel, logContext, format, args);
|
|
||||||
} else {
|
|
||||||
logger_.noContext(config, node, logLevel, format, args);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
YGNodeRef YGConfig::cloneNode(
|
|
||||||
YGNodeRef node,
|
|
||||||
YGNodeRef owner,
|
|
||||||
int childIndex,
|
|
||||||
void* cloneContext) {
|
|
||||||
YGNodeRef clone = nullptr;
|
|
||||||
if (cloneNodeCallback_.noContext != nullptr) {
|
|
||||||
clone = cloneNodeUsesContext_
|
|
||||||
? cloneNodeCallback_.withContext(node, owner, childIndex, cloneContext)
|
|
||||||
: cloneNodeCallback_.noContext(node, owner, childIndex);
|
|
||||||
}
|
|
||||||
if (clone == nullptr) {
|
|
||||||
clone = YGNodeClone(node);
|
|
||||||
}
|
|
||||||
return clone;
|
|
||||||
}
|
|
||||||
75
deps/yoga/YGConfig.h
vendored
75
deps/yoga/YGConfig.h
vendored
@ -1,75 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) Facebook, Inc. and its affiliates.
|
|
||||||
*
|
|
||||||
* This source code is licensed under the MIT license found in the LICENSE
|
|
||||||
* file in the root directory of this source tree.
|
|
||||||
*/
|
|
||||||
#pragma once
|
|
||||||
#include "Yoga-internal.h"
|
|
||||||
#include "Yoga.h"
|
|
||||||
|
|
||||||
struct YGConfig {
|
|
||||||
using LogWithContextFn = int (*)(
|
|
||||||
YGConfigRef config,
|
|
||||||
YGNodeRef node,
|
|
||||||
YGLogLevel level,
|
|
||||||
void* context,
|
|
||||||
const char* format,
|
|
||||||
va_list args);
|
|
||||||
using CloneWithContextFn = YGNodeRef (*)(
|
|
||||||
YGNodeRef node,
|
|
||||||
YGNodeRef owner,
|
|
||||||
int childIndex,
|
|
||||||
void* cloneContext);
|
|
||||||
|
|
||||||
private:
|
|
||||||
union {
|
|
||||||
CloneWithContextFn withContext;
|
|
||||||
YGCloneNodeFunc noContext;
|
|
||||||
} cloneNodeCallback_;
|
|
||||||
union {
|
|
||||||
LogWithContextFn withContext;
|
|
||||||
YGLogger noContext;
|
|
||||||
} logger_;
|
|
||||||
bool cloneNodeUsesContext_;
|
|
||||||
bool loggerUsesContext_;
|
|
||||||
|
|
||||||
public:
|
|
||||||
bool useWebDefaults = false;
|
|
||||||
bool useLegacyStretchBehaviour = false;
|
|
||||||
bool shouldDiffLayoutWithoutLegacyStretchBehaviour = false;
|
|
||||||
bool printTree = false;
|
|
||||||
float pointScaleFactor = 1.0f;
|
|
||||||
std::array<bool, facebook::yoga::enums::count<YGExperimentalFeature>()>
|
|
||||||
experimentalFeatures = {};
|
|
||||||
void* context = nullptr;
|
|
||||||
|
|
||||||
YGConfig(YGLogger logger);
|
|
||||||
void log(YGConfig*, YGNode*, YGLogLevel, void*, const char*, va_list);
|
|
||||||
void setLogger(YGLogger logger) {
|
|
||||||
logger_.noContext = logger;
|
|
||||||
loggerUsesContext_ = false;
|
|
||||||
}
|
|
||||||
void setLogger(LogWithContextFn logger) {
|
|
||||||
logger_.withContext = logger;
|
|
||||||
loggerUsesContext_ = true;
|
|
||||||
}
|
|
||||||
void setLogger(std::nullptr_t) { setLogger(YGLogger{nullptr}); }
|
|
||||||
|
|
||||||
YGNodeRef cloneNode(
|
|
||||||
YGNodeRef node,
|
|
||||||
YGNodeRef owner,
|
|
||||||
int childIndex,
|
|
||||||
void* cloneContext);
|
|
||||||
void setCloneNodeCallback(YGCloneNodeFunc cloneNode) {
|
|
||||||
cloneNodeCallback_.noContext = cloneNode;
|
|
||||||
cloneNodeUsesContext_ = false;
|
|
||||||
}
|
|
||||||
void setCloneNodeCallback(CloneWithContextFn cloneNode) {
|
|
||||||
cloneNodeCallback_.withContext = cloneNode;
|
|
||||||
cloneNodeUsesContext_ = true;
|
|
||||||
}
|
|
||||||
void setCloneNodeCallback(std::nullptr_t) {
|
|
||||||
setCloneNodeCallback(YGCloneNodeFunc{nullptr});
|
|
||||||
}
|
|
||||||
};
|
|
||||||
225
deps/yoga/YGEnums.cpp
vendored
225
deps/yoga/YGEnums.cpp
vendored
@ -1,225 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) Facebook, Inc. and its affiliates.
|
|
||||||
*
|
|
||||||
* This source code is licensed under the MIT license found in the LICENSE
|
|
||||||
* file in the root directory of this source tree.
|
|
||||||
*/
|
|
||||||
#include "YGEnums.h"
|
|
||||||
|
|
||||||
const char* YGAlignToString(const YGAlign value) {
|
|
||||||
switch (value) {
|
|
||||||
case YGAlignAuto:
|
|
||||||
return "auto";
|
|
||||||
case YGAlignFlexStart:
|
|
||||||
return "flex-start";
|
|
||||||
case YGAlignCenter:
|
|
||||||
return "center";
|
|
||||||
case YGAlignFlexEnd:
|
|
||||||
return "flex-end";
|
|
||||||
case YGAlignStretch:
|
|
||||||
return "stretch";
|
|
||||||
case YGAlignBaseline:
|
|
||||||
return "baseline";
|
|
||||||
case YGAlignSpaceBetween:
|
|
||||||
return "space-between";
|
|
||||||
case YGAlignSpaceAround:
|
|
||||||
return "space-around";
|
|
||||||
}
|
|
||||||
return "unknown";
|
|
||||||
}
|
|
||||||
|
|
||||||
const char* YGDimensionToString(const YGDimension value) {
|
|
||||||
switch (value) {
|
|
||||||
case YGDimensionWidth:
|
|
||||||
return "width";
|
|
||||||
case YGDimensionHeight:
|
|
||||||
return "height";
|
|
||||||
}
|
|
||||||
return "unknown";
|
|
||||||
}
|
|
||||||
|
|
||||||
const char* YGDirectionToString(const YGDirection value) {
|
|
||||||
switch (value) {
|
|
||||||
case YGDirectionInherit:
|
|
||||||
return "inherit";
|
|
||||||
case YGDirectionLTR:
|
|
||||||
return "ltr";
|
|
||||||
case YGDirectionRTL:
|
|
||||||
return "rtl";
|
|
||||||
}
|
|
||||||
return "unknown";
|
|
||||||
}
|
|
||||||
|
|
||||||
const char* YGDisplayToString(const YGDisplay value) {
|
|
||||||
switch (value) {
|
|
||||||
case YGDisplayFlex:
|
|
||||||
return "flex";
|
|
||||||
case YGDisplayNone:
|
|
||||||
return "none";
|
|
||||||
}
|
|
||||||
return "unknown";
|
|
||||||
}
|
|
||||||
|
|
||||||
const char* YGEdgeToString(const YGEdge value) {
|
|
||||||
switch (value) {
|
|
||||||
case YGEdgeLeft:
|
|
||||||
return "left";
|
|
||||||
case YGEdgeTop:
|
|
||||||
return "top";
|
|
||||||
case YGEdgeRight:
|
|
||||||
return "right";
|
|
||||||
case YGEdgeBottom:
|
|
||||||
return "bottom";
|
|
||||||
case YGEdgeStart:
|
|
||||||
return "start";
|
|
||||||
case YGEdgeEnd:
|
|
||||||
return "end";
|
|
||||||
case YGEdgeHorizontal:
|
|
||||||
return "horizontal";
|
|
||||||
case YGEdgeVertical:
|
|
||||||
return "vertical";
|
|
||||||
case YGEdgeAll:
|
|
||||||
return "all";
|
|
||||||
}
|
|
||||||
return "unknown";
|
|
||||||
}
|
|
||||||
|
|
||||||
const char* YGExperimentalFeatureToString(const YGExperimentalFeature value) {
|
|
||||||
switch (value) {
|
|
||||||
case YGExperimentalFeatureWebFlexBasis:
|
|
||||||
return "web-flex-basis";
|
|
||||||
}
|
|
||||||
return "unknown";
|
|
||||||
}
|
|
||||||
|
|
||||||
const char* YGFlexDirectionToString(const YGFlexDirection value) {
|
|
||||||
switch (value) {
|
|
||||||
case YGFlexDirectionColumn:
|
|
||||||
return "column";
|
|
||||||
case YGFlexDirectionColumnReverse:
|
|
||||||
return "column-reverse";
|
|
||||||
case YGFlexDirectionRow:
|
|
||||||
return "row";
|
|
||||||
case YGFlexDirectionRowReverse:
|
|
||||||
return "row-reverse";
|
|
||||||
}
|
|
||||||
return "unknown";
|
|
||||||
}
|
|
||||||
|
|
||||||
const char* YGJustifyToString(const YGJustify value) {
|
|
||||||
switch (value) {
|
|
||||||
case YGJustifyFlexStart:
|
|
||||||
return "flex-start";
|
|
||||||
case YGJustifyCenter:
|
|
||||||
return "center";
|
|
||||||
case YGJustifyFlexEnd:
|
|
||||||
return "flex-end";
|
|
||||||
case YGJustifySpaceBetween:
|
|
||||||
return "space-between";
|
|
||||||
case YGJustifySpaceAround:
|
|
||||||
return "space-around";
|
|
||||||
case YGJustifySpaceEvenly:
|
|
||||||
return "space-evenly";
|
|
||||||
}
|
|
||||||
return "unknown";
|
|
||||||
}
|
|
||||||
|
|
||||||
const char* YGLogLevelToString(const YGLogLevel value) {
|
|
||||||
switch (value) {
|
|
||||||
case YGLogLevelError:
|
|
||||||
return "error";
|
|
||||||
case YGLogLevelWarn:
|
|
||||||
return "warn";
|
|
||||||
case YGLogLevelInfo:
|
|
||||||
return "info";
|
|
||||||
case YGLogLevelDebug:
|
|
||||||
return "debug";
|
|
||||||
case YGLogLevelVerbose:
|
|
||||||
return "verbose";
|
|
||||||
case YGLogLevelFatal:
|
|
||||||
return "fatal";
|
|
||||||
}
|
|
||||||
return "unknown";
|
|
||||||
}
|
|
||||||
|
|
||||||
const char* YGMeasureModeToString(const YGMeasureMode value) {
|
|
||||||
switch (value) {
|
|
||||||
case YGMeasureModeUndefined:
|
|
||||||
return "undefined";
|
|
||||||
case YGMeasureModeExactly:
|
|
||||||
return "exactly";
|
|
||||||
case YGMeasureModeAtMost:
|
|
||||||
return "at-most";
|
|
||||||
}
|
|
||||||
return "unknown";
|
|
||||||
}
|
|
||||||
|
|
||||||
const char* YGNodeTypeToString(const YGNodeType value) {
|
|
||||||
switch (value) {
|
|
||||||
case YGNodeTypeDefault:
|
|
||||||
return "default";
|
|
||||||
case YGNodeTypeText:
|
|
||||||
return "text";
|
|
||||||
}
|
|
||||||
return "unknown";
|
|
||||||
}
|
|
||||||
|
|
||||||
const char* YGOverflowToString(const YGOverflow value) {
|
|
||||||
switch (value) {
|
|
||||||
case YGOverflowVisible:
|
|
||||||
return "visible";
|
|
||||||
case YGOverflowHidden:
|
|
||||||
return "hidden";
|
|
||||||
case YGOverflowScroll:
|
|
||||||
return "scroll";
|
|
||||||
}
|
|
||||||
return "unknown";
|
|
||||||
}
|
|
||||||
|
|
||||||
const char* YGPositionTypeToString(const YGPositionType value) {
|
|
||||||
switch (value) {
|
|
||||||
case YGPositionTypeRelative:
|
|
||||||
return "relative";
|
|
||||||
case YGPositionTypeAbsolute:
|
|
||||||
return "absolute";
|
|
||||||
}
|
|
||||||
return "unknown";
|
|
||||||
}
|
|
||||||
|
|
||||||
const char* YGPrintOptionsToString(const YGPrintOptions value) {
|
|
||||||
switch (value) {
|
|
||||||
case YGPrintOptionsLayout:
|
|
||||||
return "layout";
|
|
||||||
case YGPrintOptionsStyle:
|
|
||||||
return "style";
|
|
||||||
case YGPrintOptionsChildren:
|
|
||||||
return "children";
|
|
||||||
}
|
|
||||||
return "unknown";
|
|
||||||
}
|
|
||||||
|
|
||||||
const char* YGUnitToString(const YGUnit value) {
|
|
||||||
switch (value) {
|
|
||||||
case YGUnitUndefined:
|
|
||||||
return "undefined";
|
|
||||||
case YGUnitPoint:
|
|
||||||
return "point";
|
|
||||||
case YGUnitPercent:
|
|
||||||
return "percent";
|
|
||||||
case YGUnitAuto:
|
|
||||||
return "auto";
|
|
||||||
}
|
|
||||||
return "unknown";
|
|
||||||
}
|
|
||||||
|
|
||||||
const char* YGWrapToString(const YGWrap value) {
|
|
||||||
switch (value) {
|
|
||||||
case YGWrapNoWrap:
|
|
||||||
return "no-wrap";
|
|
||||||
case YGWrapWrap:
|
|
||||||
return "wrap";
|
|
||||||
case YGWrapWrapReverse:
|
|
||||||
return "wrap-reverse";
|
|
||||||
}
|
|
||||||
return "unknown";
|
|
||||||
}
|
|
||||||
150
deps/yoga/YGEnums.h
vendored
150
deps/yoga/YGEnums.h
vendored
@ -1,150 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) Facebook, Inc. and its affiliates.
|
|
||||||
*
|
|
||||||
* This source code is licensed under the MIT license found in the LICENSE
|
|
||||||
* file in the root directory of this source tree.
|
|
||||||
*/
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include "YGMacros.h"
|
|
||||||
|
|
||||||
#ifdef __cplusplus
|
|
||||||
namespace facebook {
|
|
||||||
namespace yoga {
|
|
||||||
namespace enums {
|
|
||||||
|
|
||||||
template <typename T>
|
|
||||||
constexpr int count(); // can't use `= delete` due to a defect in clang < 3.9
|
|
||||||
|
|
||||||
namespace detail {
|
|
||||||
template <int... xs>
|
|
||||||
constexpr int n() {
|
|
||||||
return sizeof...(xs);
|
|
||||||
}
|
|
||||||
} // namespace detail
|
|
||||||
|
|
||||||
} // namespace enums
|
|
||||||
} // namespace yoga
|
|
||||||
} // namespace facebook
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#define YG_ENUM_DECL(NAME, ...) \
|
|
||||||
typedef YG_ENUM_BEGIN(NAME){__VA_ARGS__} YG_ENUM_END(NAME); \
|
|
||||||
WIN_EXPORT const char* NAME##ToString(NAME);
|
|
||||||
|
|
||||||
#ifdef __cplusplus
|
|
||||||
#define YG_ENUM_SEQ_DECL(NAME, ...) \
|
|
||||||
YG_ENUM_DECL(NAME, __VA_ARGS__) \
|
|
||||||
YG_EXTERN_C_END \
|
|
||||||
namespace facebook { \
|
|
||||||
namespace yoga { \
|
|
||||||
namespace enums { \
|
|
||||||
template <> \
|
|
||||||
constexpr int count<NAME>() { \
|
|
||||||
return detail::n<__VA_ARGS__>(); \
|
|
||||||
} \
|
|
||||||
} \
|
|
||||||
} \
|
|
||||||
} \
|
|
||||||
YG_EXTERN_C_BEGIN
|
|
||||||
#else
|
|
||||||
#define YG_ENUM_SEQ_DECL YG_ENUM_DECL
|
|
||||||
#endif
|
|
||||||
|
|
||||||
YG_EXTERN_C_BEGIN
|
|
||||||
|
|
||||||
YG_ENUM_SEQ_DECL(
|
|
||||||
YGAlign,
|
|
||||||
YGAlignAuto,
|
|
||||||
YGAlignFlexStart,
|
|
||||||
YGAlignCenter,
|
|
||||||
YGAlignFlexEnd,
|
|
||||||
YGAlignStretch,
|
|
||||||
YGAlignBaseline,
|
|
||||||
YGAlignSpaceBetween,
|
|
||||||
YGAlignSpaceAround);
|
|
||||||
|
|
||||||
YG_ENUM_SEQ_DECL(YGDimension, YGDimensionWidth, YGDimensionHeight)
|
|
||||||
|
|
||||||
YG_ENUM_SEQ_DECL(
|
|
||||||
YGDirection,
|
|
||||||
YGDirectionInherit,
|
|
||||||
YGDirectionLTR,
|
|
||||||
YGDirectionRTL)
|
|
||||||
|
|
||||||
YG_ENUM_SEQ_DECL(YGDisplay, YGDisplayFlex, YGDisplayNone)
|
|
||||||
|
|
||||||
YG_ENUM_SEQ_DECL(
|
|
||||||
YGEdge,
|
|
||||||
YGEdgeLeft,
|
|
||||||
YGEdgeTop,
|
|
||||||
YGEdgeRight,
|
|
||||||
YGEdgeBottom,
|
|
||||||
YGEdgeStart,
|
|
||||||
YGEdgeEnd,
|
|
||||||
YGEdgeHorizontal,
|
|
||||||
YGEdgeVertical,
|
|
||||||
YGEdgeAll)
|
|
||||||
|
|
||||||
YG_ENUM_SEQ_DECL(YGExperimentalFeature, YGExperimentalFeatureWebFlexBasis)
|
|
||||||
|
|
||||||
YG_ENUM_SEQ_DECL(
|
|
||||||
YGFlexDirection,
|
|
||||||
YGFlexDirectionColumn,
|
|
||||||
YGFlexDirectionColumnReverse,
|
|
||||||
YGFlexDirectionRow,
|
|
||||||
YGFlexDirectionRowReverse)
|
|
||||||
|
|
||||||
YG_ENUM_SEQ_DECL(
|
|
||||||
YGJustify,
|
|
||||||
YGJustifyFlexStart,
|
|
||||||
YGJustifyCenter,
|
|
||||||
YGJustifyFlexEnd,
|
|
||||||
YGJustifySpaceBetween,
|
|
||||||
YGJustifySpaceAround,
|
|
||||||
YGJustifySpaceEvenly)
|
|
||||||
|
|
||||||
YG_ENUM_SEQ_DECL(
|
|
||||||
YGLogLevel,
|
|
||||||
YGLogLevelError,
|
|
||||||
YGLogLevelWarn,
|
|
||||||
YGLogLevelInfo,
|
|
||||||
YGLogLevelDebug,
|
|
||||||
YGLogLevelVerbose,
|
|
||||||
YGLogLevelFatal)
|
|
||||||
|
|
||||||
YG_ENUM_SEQ_DECL(
|
|
||||||
YGMeasureMode,
|
|
||||||
YGMeasureModeUndefined,
|
|
||||||
YGMeasureModeExactly,
|
|
||||||
YGMeasureModeAtMost)
|
|
||||||
|
|
||||||
YG_ENUM_SEQ_DECL(YGNodeType, YGNodeTypeDefault, YGNodeTypeText)
|
|
||||||
|
|
||||||
YG_ENUM_SEQ_DECL(
|
|
||||||
YGOverflow,
|
|
||||||
YGOverflowVisible,
|
|
||||||
YGOverflowHidden,
|
|
||||||
YGOverflowScroll)
|
|
||||||
|
|
||||||
YG_ENUM_SEQ_DECL(YGPositionType, YGPositionTypeRelative, YGPositionTypeAbsolute)
|
|
||||||
|
|
||||||
YG_ENUM_DECL(
|
|
||||||
YGPrintOptions,
|
|
||||||
YGPrintOptionsLayout = 1,
|
|
||||||
YGPrintOptionsStyle = 2,
|
|
||||||
YGPrintOptionsChildren = 4)
|
|
||||||
|
|
||||||
YG_ENUM_SEQ_DECL(
|
|
||||||
YGUnit,
|
|
||||||
YGUnitUndefined,
|
|
||||||
YGUnitPoint,
|
|
||||||
YGUnitPercent,
|
|
||||||
YGUnitAuto)
|
|
||||||
|
|
||||||
YG_ENUM_SEQ_DECL(YGWrap, YGWrapNoWrap, YGWrapWrap, YGWrapWrapReverse)
|
|
||||||
|
|
||||||
YG_EXTERN_C_END
|
|
||||||
|
|
||||||
#undef YG_ENUM_DECL
|
|
||||||
#undef YG_ENUM_SEQ_DECL
|
|
||||||
69
deps/yoga/YGFloatOptional.h
vendored
69
deps/yoga/YGFloatOptional.h
vendored
@ -1,69 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) Facebook, Inc. and its affiliates.
|
|
||||||
*
|
|
||||||
* This source code is licensed under the MIT license found in the LICENSE
|
|
||||||
* file in the root directory of this source tree.
|
|
||||||
*/
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include <cmath>
|
|
||||||
#include <limits>
|
|
||||||
#include "Yoga-internal.h"
|
|
||||||
|
|
||||||
struct YGFloatOptional {
|
|
||||||
private:
|
|
||||||
float value_ = std::numeric_limits<float>::quiet_NaN();
|
|
||||||
|
|
||||||
public:
|
|
||||||
explicit constexpr YGFloatOptional(float value) : value_(value) {}
|
|
||||||
constexpr YGFloatOptional() = default;
|
|
||||||
|
|
||||||
// returns the wrapped value, or a value x with YGIsUndefined(x) == true
|
|
||||||
constexpr float unwrap() const { return value_; }
|
|
||||||
|
|
||||||
bool isUndefined() const { return std::isnan(value_); }
|
|
||||||
};
|
|
||||||
|
|
||||||
// operators take YGFloatOptional by value, as it is a 32bit value
|
|
||||||
|
|
||||||
inline bool operator==(YGFloatOptional lhs, YGFloatOptional rhs) {
|
|
||||||
return lhs.unwrap() == rhs.unwrap() ||
|
|
||||||
(lhs.isUndefined() && rhs.isUndefined());
|
|
||||||
}
|
|
||||||
inline bool operator!=(YGFloatOptional lhs, YGFloatOptional rhs) {
|
|
||||||
return !(lhs == rhs);
|
|
||||||
}
|
|
||||||
|
|
||||||
inline bool operator==(YGFloatOptional lhs, float rhs) {
|
|
||||||
return lhs == YGFloatOptional{rhs};
|
|
||||||
}
|
|
||||||
inline bool operator!=(YGFloatOptional lhs, float rhs) {
|
|
||||||
return !(lhs == rhs);
|
|
||||||
}
|
|
||||||
|
|
||||||
inline bool operator==(float lhs, YGFloatOptional rhs) {
|
|
||||||
return rhs == lhs;
|
|
||||||
}
|
|
||||||
inline bool operator!=(float lhs, YGFloatOptional rhs) {
|
|
||||||
return !(lhs == rhs);
|
|
||||||
}
|
|
||||||
|
|
||||||
inline YGFloatOptional operator+(YGFloatOptional lhs, YGFloatOptional rhs) {
|
|
||||||
return YGFloatOptional{lhs.unwrap() + rhs.unwrap()};
|
|
||||||
}
|
|
||||||
|
|
||||||
inline bool operator>(YGFloatOptional lhs, YGFloatOptional rhs) {
|
|
||||||
return lhs.unwrap() > rhs.unwrap();
|
|
||||||
}
|
|
||||||
|
|
||||||
inline bool operator<(YGFloatOptional lhs, YGFloatOptional rhs) {
|
|
||||||
return lhs.unwrap() < rhs.unwrap();
|
|
||||||
}
|
|
||||||
|
|
||||||
inline bool operator>=(YGFloatOptional lhs, YGFloatOptional rhs) {
|
|
||||||
return lhs > rhs || lhs == rhs;
|
|
||||||
}
|
|
||||||
|
|
||||||
inline bool operator<=(YGFloatOptional lhs, YGFloatOptional rhs) {
|
|
||||||
return lhs < rhs || lhs == rhs;
|
|
||||||
}
|
|
||||||
40
deps/yoga/YGLayout.cpp
vendored
40
deps/yoga/YGLayout.cpp
vendored
@ -1,40 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) Facebook, Inc. and its affiliates.
|
|
||||||
*
|
|
||||||
* This source code is licensed under the MIT license found in the LICENSE
|
|
||||||
* file in the root directory of this source tree.
|
|
||||||
*/
|
|
||||||
#include "YGLayout.h"
|
|
||||||
#include "Utils.h"
|
|
||||||
|
|
||||||
using namespace facebook;
|
|
||||||
|
|
||||||
bool YGLayout::operator==(YGLayout layout) const {
|
|
||||||
bool isEqual = YGFloatArrayEqual(position, layout.position) &&
|
|
||||||
YGFloatArrayEqual(dimensions, layout.dimensions) &&
|
|
||||||
YGFloatArrayEqual(margin, layout.margin) &&
|
|
||||||
YGFloatArrayEqual(border, layout.border) &&
|
|
||||||
YGFloatArrayEqual(padding, layout.padding) &&
|
|
||||||
direction == layout.direction && hadOverflow == layout.hadOverflow &&
|
|
||||||
lastOwnerDirection == layout.lastOwnerDirection &&
|
|
||||||
nextCachedMeasurementsIndex == layout.nextCachedMeasurementsIndex &&
|
|
||||||
cachedLayout == layout.cachedLayout &&
|
|
||||||
computedFlexBasis == layout.computedFlexBasis;
|
|
||||||
|
|
||||||
for (uint32_t i = 0; i < YG_MAX_CACHED_RESULT_COUNT && isEqual; ++i) {
|
|
||||||
isEqual = isEqual && cachedMeasurements[i] == layout.cachedMeasurements[i];
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!yoga::isUndefined(measuredDimensions[0]) ||
|
|
||||||
!yoga::isUndefined(layout.measuredDimensions[0])) {
|
|
||||||
isEqual =
|
|
||||||
isEqual && (measuredDimensions[0] == layout.measuredDimensions[0]);
|
|
||||||
}
|
|
||||||
if (!yoga::isUndefined(measuredDimensions[1]) ||
|
|
||||||
!yoga::isUndefined(layout.measuredDimensions[1])) {
|
|
||||||
isEqual =
|
|
||||||
isEqual && (measuredDimensions[1] == layout.measuredDimensions[1]);
|
|
||||||
}
|
|
||||||
|
|
||||||
return isEqual;
|
|
||||||
}
|
|
||||||
45
deps/yoga/YGLayout.h
vendored
45
deps/yoga/YGLayout.h
vendored
@ -1,45 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) Facebook, Inc. and its affiliates.
|
|
||||||
*
|
|
||||||
* This source code is licensed under the MIT license found in the LICENSE
|
|
||||||
* file in the root directory of this source tree.
|
|
||||||
*/
|
|
||||||
#pragma once
|
|
||||||
#include "YGFloatOptional.h"
|
|
||||||
#include "Yoga-internal.h"
|
|
||||||
|
|
||||||
struct YGLayout {
|
|
||||||
std::array<float, 4> position = {};
|
|
||||||
std::array<float, 2> dimensions = {{YGUndefined, YGUndefined}};
|
|
||||||
std::array<float, 4> margin = {};
|
|
||||||
std::array<float, 4> border = {};
|
|
||||||
std::array<float, 4> padding = {};
|
|
||||||
YGDirection direction : 2;
|
|
||||||
bool didUseLegacyFlag : 1;
|
|
||||||
bool doesLegacyStretchFlagAffectsLayout : 1;
|
|
||||||
bool hadOverflow : 1;
|
|
||||||
|
|
||||||
uint32_t computedFlexBasisGeneration = 0;
|
|
||||||
YGFloatOptional computedFlexBasis = {};
|
|
||||||
|
|
||||||
// Instead of recomputing the entire layout every single time, we cache some
|
|
||||||
// information to break early when nothing changed
|
|
||||||
uint32_t generationCount = 0;
|
|
||||||
YGDirection lastOwnerDirection = (YGDirection) -1;
|
|
||||||
|
|
||||||
uint32_t nextCachedMeasurementsIndex = 0;
|
|
||||||
std::array<YGCachedMeasurement, YG_MAX_CACHED_RESULT_COUNT>
|
|
||||||
cachedMeasurements = {};
|
|
||||||
std::array<float, 2> measuredDimensions = {{YGUndefined, YGUndefined}};
|
|
||||||
|
|
||||||
YGCachedMeasurement cachedLayout = YGCachedMeasurement();
|
|
||||||
|
|
||||||
YGLayout()
|
|
||||||
: direction(YGDirectionInherit),
|
|
||||||
didUseLegacyFlag(false),
|
|
||||||
doesLegacyStretchFlagAffectsLayout(false),
|
|
||||||
hadOverflow(false) {}
|
|
||||||
|
|
||||||
bool operator==(YGLayout layout) const;
|
|
||||||
bool operator!=(YGLayout layout) const { return !(*this == layout); }
|
|
||||||
};
|
|
||||||
44
deps/yoga/YGMacros.h
vendored
44
deps/yoga/YGMacros.h
vendored
@ -1,44 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) Facebook, Inc. and its affiliates.
|
|
||||||
*
|
|
||||||
* This source code is licensed under the MIT license found in the LICENSE
|
|
||||||
* file in the root directory of this source tree.
|
|
||||||
*/
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#ifdef __cplusplus
|
|
||||||
#define YG_EXTERN_C_BEGIN extern "C" {
|
|
||||||
#define YG_EXTERN_C_END }
|
|
||||||
#else
|
|
||||||
#define YG_EXTERN_C_BEGIN
|
|
||||||
#define YG_EXTERN_C_END
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifdef _WINDLL
|
|
||||||
#define WIN_EXPORT __declspec(dllexport)
|
|
||||||
#else
|
|
||||||
#define WIN_EXPORT
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifdef NS_ENUM
|
|
||||||
// Cannot use NSInteger as NSInteger has a different size than int (which is the
|
|
||||||
// default type of a enum). Therefor when linking the Yoga C library into obj-c
|
|
||||||
// the header is a missmatch for the Yoga ABI.
|
|
||||||
#define YG_ENUM_BEGIN(name) NS_ENUM(int, name)
|
|
||||||
#define YG_ENUM_END(name)
|
|
||||||
#else
|
|
||||||
#define YG_ENUM_BEGIN(name) enum name
|
|
||||||
#define YG_ENUM_END(name) name
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifdef __GNUC__
|
|
||||||
#define YG_DEPRECATED __attribute__((deprecated))
|
|
||||||
#elif defined(_MSC_VER)
|
|
||||||
#define YG_DEPRECATED __declspec(deprecated)
|
|
||||||
#elif __cplusplus >= 201402L
|
|
||||||
#if defined(__has_cpp_attribute)
|
|
||||||
#if __has_cpp_attribute(deprecated)
|
|
||||||
#define YG_DEPRECATED [[deprecated]]
|
|
||||||
#endif
|
|
||||||
#endif
|
|
||||||
#endif
|
|
||||||
590
deps/yoga/YGNode.cpp
vendored
590
deps/yoga/YGNode.cpp
vendored
@ -1,590 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) Facebook, Inc. and its affiliates.
|
|
||||||
*
|
|
||||||
* This source code is licensed under the MIT license found in the LICENSE
|
|
||||||
* file in the root directory of this source tree.
|
|
||||||
*/
|
|
||||||
#include "YGNode.h"
|
|
||||||
#include <algorithm>
|
|
||||||
#include <iostream>
|
|
||||||
#include "CompactValue.h"
|
|
||||||
#include "Utils.h"
|
|
||||||
|
|
||||||
using namespace facebook;
|
|
||||||
using facebook::yoga::detail::CompactValue;
|
|
||||||
|
|
||||||
YGNode::YGNode(YGNode&& node) {
|
|
||||||
context_ = node.context_;
|
|
||||||
hasNewLayout_ = node.hasNewLayout_;
|
|
||||||
isReferenceBaseline_ = node.isReferenceBaseline_;
|
|
||||||
isDirty_ = node.isDirty_;
|
|
||||||
nodeType_ = node.nodeType_;
|
|
||||||
measureUsesContext_ = node.measureUsesContext_;
|
|
||||||
baselineUsesContext_ = node.baselineUsesContext_;
|
|
||||||
printUsesContext_ = node.printUsesContext_;
|
|
||||||
useWebDefaults_ = node.useWebDefaults_;
|
|
||||||
measure_ = node.measure_;
|
|
||||||
baseline_ = node.baseline_;
|
|
||||||
print_ = node.print_;
|
|
||||||
dirtied_ = node.dirtied_;
|
|
||||||
style_ = node.style_;
|
|
||||||
layout_ = node.layout_;
|
|
||||||
lineIndex_ = node.lineIndex_;
|
|
||||||
owner_ = node.owner_;
|
|
||||||
children_ = std::move(node.children_);
|
|
||||||
config_ = node.config_;
|
|
||||||
resolvedDimensions_ = node.resolvedDimensions_;
|
|
||||||
for (auto c : children_) {
|
|
||||||
c->setOwner(c);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
YGNode::YGNode(const YGNode& node, YGConfigRef config) : YGNode{node} {
|
|
||||||
config_ = config;
|
|
||||||
if (config->useWebDefaults) {
|
|
||||||
useWebDefaults();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void YGNode::print(void* printContext) {
|
|
||||||
if (print_.noContext != nullptr) {
|
|
||||||
if (printUsesContext_) {
|
|
||||||
print_.withContext(this, printContext);
|
|
||||||
} else {
|
|
||||||
print_.noContext(this);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
YGFloatOptional YGNode::getLeadingPosition(
|
|
||||||
const YGFlexDirection axis,
|
|
||||||
const float axisSize) const {
|
|
||||||
if (YGFlexDirectionIsRow(axis)) {
|
|
||||||
auto leadingPosition = YGComputedEdgeValue(
|
|
||||||
style_.position(), YGEdgeStart, CompactValue::ofUndefined());
|
|
||||||
if (!leadingPosition.isUndefined()) {
|
|
||||||
return YGResolveValue(leadingPosition, axisSize);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
auto leadingPosition = YGComputedEdgeValue(
|
|
||||||
style_.position(), leading[axis], CompactValue::ofUndefined());
|
|
||||||
|
|
||||||
return leadingPosition.isUndefined()
|
|
||||||
? YGFloatOptional{0}
|
|
||||||
: YGResolveValue(leadingPosition, axisSize);
|
|
||||||
}
|
|
||||||
|
|
||||||
YGFloatOptional YGNode::getTrailingPosition(
|
|
||||||
const YGFlexDirection axis,
|
|
||||||
const float axisSize) const {
|
|
||||||
if (YGFlexDirectionIsRow(axis)) {
|
|
||||||
auto trailingPosition = YGComputedEdgeValue(
|
|
||||||
style_.position(), YGEdgeEnd, CompactValue::ofUndefined());
|
|
||||||
if (!trailingPosition.isUndefined()) {
|
|
||||||
return YGResolveValue(trailingPosition, axisSize);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
auto trailingPosition = YGComputedEdgeValue(
|
|
||||||
style_.position(), trailing[axis], CompactValue::ofUndefined());
|
|
||||||
|
|
||||||
return trailingPosition.isUndefined()
|
|
||||||
? YGFloatOptional{0}
|
|
||||||
: YGResolveValue(trailingPosition, axisSize);
|
|
||||||
}
|
|
||||||
|
|
||||||
bool YGNode::isLeadingPositionDefined(const YGFlexDirection axis) const {
|
|
||||||
return (YGFlexDirectionIsRow(axis) &&
|
|
||||||
!YGComputedEdgeValue(
|
|
||||||
style_.position(), YGEdgeStart, CompactValue::ofUndefined())
|
|
||||||
.isUndefined()) ||
|
|
||||||
!YGComputedEdgeValue(
|
|
||||||
style_.position(), leading[axis], CompactValue::ofUndefined())
|
|
||||||
.isUndefined();
|
|
||||||
}
|
|
||||||
|
|
||||||
bool YGNode::isTrailingPosDefined(const YGFlexDirection axis) const {
|
|
||||||
return (YGFlexDirectionIsRow(axis) &&
|
|
||||||
!YGComputedEdgeValue(
|
|
||||||
style_.position(), YGEdgeEnd, CompactValue::ofUndefined())
|
|
||||||
.isUndefined()) ||
|
|
||||||
!YGComputedEdgeValue(
|
|
||||||
style_.position(), trailing[axis], CompactValue::ofUndefined())
|
|
||||||
.isUndefined();
|
|
||||||
}
|
|
||||||
|
|
||||||
YGFloatOptional YGNode::getLeadingMargin(
|
|
||||||
const YGFlexDirection axis,
|
|
||||||
const float widthSize) const {
|
|
||||||
if (YGFlexDirectionIsRow(axis) &&
|
|
||||||
!style_.margin()[YGEdgeStart].isUndefined()) {
|
|
||||||
return YGResolveValueMargin(style_.margin()[YGEdgeStart], widthSize);
|
|
||||||
}
|
|
||||||
|
|
||||||
return YGResolveValueMargin(
|
|
||||||
YGComputedEdgeValue(
|
|
||||||
style_.margin(), leading[axis], CompactValue::ofZero()),
|
|
||||||
widthSize);
|
|
||||||
}
|
|
||||||
|
|
||||||
YGFloatOptional YGNode::getTrailingMargin(
|
|
||||||
const YGFlexDirection axis,
|
|
||||||
const float widthSize) const {
|
|
||||||
if (YGFlexDirectionIsRow(axis) && !style_.margin()[YGEdgeEnd].isUndefined()) {
|
|
||||||
return YGResolveValueMargin(style_.margin()[YGEdgeEnd], widthSize);
|
|
||||||
}
|
|
||||||
|
|
||||||
return YGResolveValueMargin(
|
|
||||||
YGComputedEdgeValue(
|
|
||||||
style_.margin(), trailing[axis], CompactValue::ofZero()),
|
|
||||||
widthSize);
|
|
||||||
}
|
|
||||||
|
|
||||||
YGFloatOptional YGNode::getMarginForAxis(
|
|
||||||
const YGFlexDirection axis,
|
|
||||||
const float widthSize) const {
|
|
||||||
return getLeadingMargin(axis, widthSize) + getTrailingMargin(axis, widthSize);
|
|
||||||
}
|
|
||||||
|
|
||||||
YGSize YGNode::measure(
|
|
||||||
float width,
|
|
||||||
YGMeasureMode widthMode,
|
|
||||||
float height,
|
|
||||||
YGMeasureMode heightMode,
|
|
||||||
void* layoutContext) {
|
|
||||||
|
|
||||||
return measureUsesContext_
|
|
||||||
? measure_.withContext(
|
|
||||||
this, width, widthMode, height, heightMode, layoutContext)
|
|
||||||
: measure_.noContext(this, width, widthMode, height, heightMode);
|
|
||||||
}
|
|
||||||
|
|
||||||
float YGNode::baseline(float width, float height, void* layoutContext) {
|
|
||||||
return baselineUsesContext_
|
|
||||||
? baseline_.withContext(this, width, height, layoutContext)
|
|
||||||
: baseline_.noContext(this, width, height);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Setters
|
|
||||||
|
|
||||||
void YGNode::setMeasureFunc(decltype(YGNode::measure_) measureFunc) {
|
|
||||||
if (measureFunc.noContext == nullptr) {
|
|
||||||
// TODO: t18095186 Move nodeType to opt-in function and mark appropriate
|
|
||||||
// places in Litho
|
|
||||||
nodeType_ = YGNodeTypeDefault;
|
|
||||||
} else {
|
|
||||||
YGAssertWithNode(
|
|
||||||
this,
|
|
||||||
children_.size() == 0,
|
|
||||||
"Cannot set measure function: Nodes with measure functions cannot have "
|
|
||||||
"children.");
|
|
||||||
// TODO: t18095186 Move nodeType to opt-in function and mark appropriate
|
|
||||||
// places in Litho
|
|
||||||
setNodeType(YGNodeTypeText);
|
|
||||||
}
|
|
||||||
|
|
||||||
measure_ = measureFunc;
|
|
||||||
}
|
|
||||||
|
|
||||||
void YGNode::setMeasureFunc(YGMeasureFunc measureFunc) {
|
|
||||||
measureUsesContext_ = false;
|
|
||||||
decltype(YGNode::measure_) m;
|
|
||||||
m.noContext = measureFunc;
|
|
||||||
setMeasureFunc(m);
|
|
||||||
}
|
|
||||||
|
|
||||||
void YGNode::setMeasureFunc(MeasureWithContextFn measureFunc) {
|
|
||||||
measureUsesContext_ = true;
|
|
||||||
decltype(YGNode::measure_) m;
|
|
||||||
m.withContext = measureFunc;
|
|
||||||
setMeasureFunc(m);
|
|
||||||
}
|
|
||||||
|
|
||||||
void YGNode::replaceChild(YGNodeRef child, uint32_t index) {
|
|
||||||
children_[index] = child;
|
|
||||||
}
|
|
||||||
|
|
||||||
void YGNode::replaceChild(YGNodeRef oldChild, YGNodeRef newChild) {
|
|
||||||
std::replace(children_.begin(), children_.end(), oldChild, newChild);
|
|
||||||
}
|
|
||||||
|
|
||||||
void YGNode::insertChild(YGNodeRef child, uint32_t index) {
|
|
||||||
children_.insert(children_.begin() + index, child);
|
|
||||||
}
|
|
||||||
|
|
||||||
void YGNode::setDirty(bool isDirty) {
|
|
||||||
if (isDirty == isDirty_) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
isDirty_ = isDirty;
|
|
||||||
if (isDirty && dirtied_) {
|
|
||||||
dirtied_(this);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
bool YGNode::removeChild(YGNodeRef child) {
|
|
||||||
std::vector<YGNodeRef>::iterator p =
|
|
||||||
std::find(children_.begin(), children_.end(), child);
|
|
||||||
if (p != children_.end()) {
|
|
||||||
children_.erase(p);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
void YGNode::removeChild(uint32_t index) {
|
|
||||||
children_.erase(children_.begin() + index);
|
|
||||||
}
|
|
||||||
|
|
||||||
void YGNode::setLayoutDirection(YGDirection direction) {
|
|
||||||
layout_.direction = direction;
|
|
||||||
}
|
|
||||||
|
|
||||||
void YGNode::setLayoutMargin(float margin, int index) {
|
|
||||||
layout_.margin[index] = margin;
|
|
||||||
}
|
|
||||||
|
|
||||||
void YGNode::setLayoutBorder(float border, int index) {
|
|
||||||
layout_.border[index] = border;
|
|
||||||
}
|
|
||||||
|
|
||||||
void YGNode::setLayoutPadding(float padding, int index) {
|
|
||||||
layout_.padding[index] = padding;
|
|
||||||
}
|
|
||||||
|
|
||||||
void YGNode::setLayoutLastOwnerDirection(YGDirection direction) {
|
|
||||||
layout_.lastOwnerDirection = direction;
|
|
||||||
}
|
|
||||||
|
|
||||||
void YGNode::setLayoutComputedFlexBasis(
|
|
||||||
const YGFloatOptional computedFlexBasis) {
|
|
||||||
layout_.computedFlexBasis = computedFlexBasis;
|
|
||||||
}
|
|
||||||
|
|
||||||
void YGNode::setLayoutPosition(float position, int index) {
|
|
||||||
layout_.position[index] = position;
|
|
||||||
}
|
|
||||||
|
|
||||||
void YGNode::setLayoutComputedFlexBasisGeneration(
|
|
||||||
uint32_t computedFlexBasisGeneration) {
|
|
||||||
layout_.computedFlexBasisGeneration = computedFlexBasisGeneration;
|
|
||||||
}
|
|
||||||
|
|
||||||
void YGNode::setLayoutMeasuredDimension(float measuredDimension, int index) {
|
|
||||||
layout_.measuredDimensions[index] = measuredDimension;
|
|
||||||
}
|
|
||||||
|
|
||||||
void YGNode::setLayoutHadOverflow(bool hadOverflow) {
|
|
||||||
layout_.hadOverflow = hadOverflow;
|
|
||||||
}
|
|
||||||
|
|
||||||
void YGNode::setLayoutDimension(float dimension, int index) {
|
|
||||||
layout_.dimensions[index] = dimension;
|
|
||||||
}
|
|
||||||
|
|
||||||
// If both left and right are defined, then use left. Otherwise return +left or
|
|
||||||
// -right depending on which is defined.
|
|
||||||
YGFloatOptional YGNode::relativePosition(
|
|
||||||
const YGFlexDirection axis,
|
|
||||||
const float axisSize) const {
|
|
||||||
if (isLeadingPositionDefined(axis)) {
|
|
||||||
return getLeadingPosition(axis, axisSize);
|
|
||||||
}
|
|
||||||
|
|
||||||
YGFloatOptional trailingPosition = getTrailingPosition(axis, axisSize);
|
|
||||||
if (!trailingPosition.isUndefined()) {
|
|
||||||
trailingPosition = YGFloatOptional{-1 * trailingPosition.unwrap()};
|
|
||||||
}
|
|
||||||
return trailingPosition;
|
|
||||||
}
|
|
||||||
|
|
||||||
void YGNode::setPosition(
|
|
||||||
const YGDirection direction,
|
|
||||||
const float mainSize,
|
|
||||||
const float crossSize,
|
|
||||||
const float ownerWidth) {
|
|
||||||
/* Root nodes should be always layouted as LTR, so we don't return negative
|
|
||||||
* values. */
|
|
||||||
const YGDirection directionRespectingRoot =
|
|
||||||
owner_ != nullptr ? direction : YGDirectionLTR;
|
|
||||||
const YGFlexDirection mainAxis =
|
|
||||||
YGResolveFlexDirection(style_.flexDirection(), directionRespectingRoot);
|
|
||||||
const YGFlexDirection crossAxis =
|
|
||||||
YGFlexDirectionCross(mainAxis, directionRespectingRoot);
|
|
||||||
|
|
||||||
const YGFloatOptional relativePositionMain =
|
|
||||||
relativePosition(mainAxis, mainSize);
|
|
||||||
const YGFloatOptional relativePositionCross =
|
|
||||||
relativePosition(crossAxis, crossSize);
|
|
||||||
|
|
||||||
setLayoutPosition(
|
|
||||||
(getLeadingMargin(mainAxis, ownerWidth) + relativePositionMain).unwrap(),
|
|
||||||
leading[mainAxis]);
|
|
||||||
setLayoutPosition(
|
|
||||||
(getTrailingMargin(mainAxis, ownerWidth) + relativePositionMain).unwrap(),
|
|
||||||
trailing[mainAxis]);
|
|
||||||
setLayoutPosition(
|
|
||||||
(getLeadingMargin(crossAxis, ownerWidth) + relativePositionCross)
|
|
||||||
.unwrap(),
|
|
||||||
leading[crossAxis]);
|
|
||||||
setLayoutPosition(
|
|
||||||
(getTrailingMargin(crossAxis, ownerWidth) + relativePositionCross)
|
|
||||||
.unwrap(),
|
|
||||||
trailing[crossAxis]);
|
|
||||||
}
|
|
||||||
|
|
||||||
YGValue YGNode::marginLeadingValue(const YGFlexDirection axis) const {
|
|
||||||
if (YGFlexDirectionIsRow(axis) &&
|
|
||||||
!style_.margin()[YGEdgeStart].isUndefined()) {
|
|
||||||
return style_.margin()[YGEdgeStart];
|
|
||||||
} else {
|
|
||||||
return style_.margin()[leading[axis]];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
YGValue YGNode::marginTrailingValue(const YGFlexDirection axis) const {
|
|
||||||
if (YGFlexDirectionIsRow(axis) && !style_.margin()[YGEdgeEnd].isUndefined()) {
|
|
||||||
return style_.margin()[YGEdgeEnd];
|
|
||||||
} else {
|
|
||||||
return style_.margin()[trailing[axis]];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
YGValue YGNode::resolveFlexBasisPtr() const {
|
|
||||||
YGValue flexBasis = style_.flexBasis();
|
|
||||||
if (flexBasis.unit != YGUnitAuto && flexBasis.unit != YGUnitUndefined) {
|
|
||||||
return flexBasis;
|
|
||||||
}
|
|
||||||
if (!style_.flex().isUndefined() && style_.flex().unwrap() > 0.0f) {
|
|
||||||
return useWebDefaults_ ? YGValueAuto : YGValueZero;
|
|
||||||
}
|
|
||||||
return YGValueAuto;
|
|
||||||
}
|
|
||||||
|
|
||||||
void YGNode::resolveDimension() {
|
|
||||||
using namespace yoga;
|
|
||||||
const YGStyle& style = getStyle();
|
|
||||||
for (auto dim : {YGDimensionWidth, YGDimensionHeight}) {
|
|
||||||
if (!style.maxDimensions()[dim].isUndefined() &&
|
|
||||||
YGValueEqual(style.maxDimensions()[dim], style.minDimensions()[dim])) {
|
|
||||||
resolvedDimensions_[dim] = style.maxDimensions()[dim];
|
|
||||||
} else {
|
|
||||||
resolvedDimensions_[dim] = style.dimensions()[dim];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
YGDirection YGNode::resolveDirection(const YGDirection ownerDirection) {
|
|
||||||
if (style_.direction() == YGDirectionInherit) {
|
|
||||||
return ownerDirection > YGDirectionInherit ? ownerDirection
|
|
||||||
: YGDirectionLTR;
|
|
||||||
} else {
|
|
||||||
return style_.direction();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void YGNode::clearChildren() {
|
|
||||||
children_.clear();
|
|
||||||
children_.shrink_to_fit();
|
|
||||||
}
|
|
||||||
|
|
||||||
// Other Methods
|
|
||||||
|
|
||||||
void YGNode::cloneChildrenIfNeeded(void* cloneContext) {
|
|
||||||
iterChildrenAfterCloningIfNeeded([](YGNodeRef, void*) {}, cloneContext);
|
|
||||||
}
|
|
||||||
|
|
||||||
void YGNode::markDirtyAndPropogate() {
|
|
||||||
if (!isDirty_) {
|
|
||||||
setDirty(true);
|
|
||||||
setLayoutComputedFlexBasis(YGFloatOptional());
|
|
||||||
if (owner_) {
|
|
||||||
owner_->markDirtyAndPropogate();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void YGNode::markDirtyAndPropogateDownwards() {
|
|
||||||
isDirty_ = true;
|
|
||||||
for_each(children_.begin(), children_.end(), [](YGNodeRef childNode) {
|
|
||||||
childNode->markDirtyAndPropogateDownwards();
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
float YGNode::resolveFlexGrow() const {
|
|
||||||
// Root nodes flexGrow should always be 0
|
|
||||||
if (owner_ == nullptr) {
|
|
||||||
return 0.0;
|
|
||||||
}
|
|
||||||
if (!style_.flexGrow().isUndefined()) {
|
|
||||||
return style_.flexGrow().unwrap();
|
|
||||||
}
|
|
||||||
if (!style_.flex().isUndefined() && style_.flex().unwrap() > 0.0f) {
|
|
||||||
return style_.flex().unwrap();
|
|
||||||
}
|
|
||||||
return kDefaultFlexGrow;
|
|
||||||
}
|
|
||||||
|
|
||||||
float YGNode::resolveFlexShrink() const {
|
|
||||||
if (owner_ == nullptr) {
|
|
||||||
return 0.0;
|
|
||||||
}
|
|
||||||
if (!style_.flexShrink().isUndefined()) {
|
|
||||||
return style_.flexShrink().unwrap();
|
|
||||||
}
|
|
||||||
if (!useWebDefaults_ && !style_.flex().isUndefined() &&
|
|
||||||
style_.flex().unwrap() < 0.0f) {
|
|
||||||
return -style_.flex().unwrap();
|
|
||||||
}
|
|
||||||
return useWebDefaults_ ? kWebDefaultFlexShrink : kDefaultFlexShrink;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool YGNode::isNodeFlexible() {
|
|
||||||
return (
|
|
||||||
(style_.positionType() == YGPositionTypeRelative) &&
|
|
||||||
(resolveFlexGrow() != 0 || resolveFlexShrink() != 0));
|
|
||||||
}
|
|
||||||
|
|
||||||
float YGNode::getLeadingBorder(const YGFlexDirection axis) const {
|
|
||||||
YGValue leadingBorder;
|
|
||||||
if (YGFlexDirectionIsRow(axis) &&
|
|
||||||
!style_.border()[YGEdgeStart].isUndefined()) {
|
|
||||||
leadingBorder = style_.border()[YGEdgeStart];
|
|
||||||
if (leadingBorder.value >= 0) {
|
|
||||||
return leadingBorder.value;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
leadingBorder = YGComputedEdgeValue(
|
|
||||||
style_.border(), leading[axis], CompactValue::ofZero());
|
|
||||||
return YGFloatMax(leadingBorder.value, 0.0f);
|
|
||||||
}
|
|
||||||
|
|
||||||
float YGNode::getTrailingBorder(const YGFlexDirection flexDirection) const {
|
|
||||||
YGValue trailingBorder;
|
|
||||||
if (YGFlexDirectionIsRow(flexDirection) &&
|
|
||||||
!style_.border()[YGEdgeEnd].isUndefined()) {
|
|
||||||
trailingBorder = style_.border()[YGEdgeEnd];
|
|
||||||
if (trailingBorder.value >= 0.0f) {
|
|
||||||
return trailingBorder.value;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
trailingBorder = YGComputedEdgeValue(
|
|
||||||
style_.border(), trailing[flexDirection], CompactValue::ofZero());
|
|
||||||
return YGFloatMax(trailingBorder.value, 0.0f);
|
|
||||||
}
|
|
||||||
|
|
||||||
YGFloatOptional YGNode::getLeadingPadding(
|
|
||||||
const YGFlexDirection axis,
|
|
||||||
const float widthSize) const {
|
|
||||||
const YGFloatOptional paddingEdgeStart =
|
|
||||||
YGResolveValue(style_.padding()[YGEdgeStart], widthSize);
|
|
||||||
if (YGFlexDirectionIsRow(axis) &&
|
|
||||||
!style_.padding()[YGEdgeStart].isUndefined() &&
|
|
||||||
!paddingEdgeStart.isUndefined() && paddingEdgeStart.unwrap() >= 0.0f) {
|
|
||||||
return paddingEdgeStart;
|
|
||||||
}
|
|
||||||
|
|
||||||
YGFloatOptional resolvedValue = YGResolveValue(
|
|
||||||
YGComputedEdgeValue(
|
|
||||||
style_.padding(), leading[axis], CompactValue::ofZero()),
|
|
||||||
widthSize);
|
|
||||||
return YGFloatOptionalMax(resolvedValue, YGFloatOptional(0.0f));
|
|
||||||
}
|
|
||||||
|
|
||||||
YGFloatOptional YGNode::getTrailingPadding(
|
|
||||||
const YGFlexDirection axis,
|
|
||||||
const float widthSize) const {
|
|
||||||
const YGFloatOptional paddingEdgeEnd =
|
|
||||||
YGResolveValue(style_.padding()[YGEdgeEnd], widthSize);
|
|
||||||
if (YGFlexDirectionIsRow(axis) && paddingEdgeEnd >= YGFloatOptional{0.0f}) {
|
|
||||||
return paddingEdgeEnd;
|
|
||||||
}
|
|
||||||
|
|
||||||
YGFloatOptional resolvedValue = YGResolveValue(
|
|
||||||
YGComputedEdgeValue(
|
|
||||||
style_.padding(), trailing[axis], CompactValue::ofZero()),
|
|
||||||
widthSize);
|
|
||||||
|
|
||||||
return YGFloatOptionalMax(resolvedValue, YGFloatOptional(0.0f));
|
|
||||||
}
|
|
||||||
|
|
||||||
YGFloatOptional YGNode::getLeadingPaddingAndBorder(
|
|
||||||
const YGFlexDirection axis,
|
|
||||||
const float widthSize) const {
|
|
||||||
return getLeadingPadding(axis, widthSize) +
|
|
||||||
YGFloatOptional(getLeadingBorder(axis));
|
|
||||||
}
|
|
||||||
|
|
||||||
YGFloatOptional YGNode::getTrailingPaddingAndBorder(
|
|
||||||
const YGFlexDirection axis,
|
|
||||||
const float widthSize) const {
|
|
||||||
return getTrailingPadding(axis, widthSize) +
|
|
||||||
YGFloatOptional(getTrailingBorder(axis));
|
|
||||||
}
|
|
||||||
|
|
||||||
bool YGNode::didUseLegacyFlag() {
|
|
||||||
bool didUseLegacyFlag = layout_.didUseLegacyFlag;
|
|
||||||
if (didUseLegacyFlag) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
for (const auto& child : children_) {
|
|
||||||
if (child->layout_.didUseLegacyFlag) {
|
|
||||||
didUseLegacyFlag = true;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return didUseLegacyFlag;
|
|
||||||
}
|
|
||||||
|
|
||||||
void YGNode::setLayoutDoesLegacyFlagAffectsLayout(
|
|
||||||
bool doesLegacyFlagAffectsLayout) {
|
|
||||||
layout_.doesLegacyStretchFlagAffectsLayout = doesLegacyFlagAffectsLayout;
|
|
||||||
}
|
|
||||||
|
|
||||||
void YGNode::setLayoutDidUseLegacyFlag(bool didUseLegacyFlag) {
|
|
||||||
layout_.didUseLegacyFlag = didUseLegacyFlag;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool YGNode::isLayoutTreeEqualToNode(const YGNode& node) const {
|
|
||||||
if (children_.size() != node.children_.size()) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
if (layout_ != node.layout_) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
if (children_.size() == 0) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool isLayoutTreeEqual = true;
|
|
||||||
YGNodeRef otherNodeChildren = nullptr;
|
|
||||||
for (std::vector<YGNodeRef>::size_type i = 0; i < children_.size(); ++i) {
|
|
||||||
otherNodeChildren = node.children_[i];
|
|
||||||
isLayoutTreeEqual =
|
|
||||||
children_[i]->isLayoutTreeEqualToNode(*otherNodeChildren);
|
|
||||||
if (!isLayoutTreeEqual) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return isLayoutTreeEqual;
|
|
||||||
}
|
|
||||||
|
|
||||||
void YGNode::reset() {
|
|
||||||
YGAssertWithNode(
|
|
||||||
this,
|
|
||||||
children_.size() == 0,
|
|
||||||
"Cannot reset a node which still has children attached");
|
|
||||||
YGAssertWithNode(
|
|
||||||
this, owner_ == nullptr, "Cannot reset a node still attached to a owner");
|
|
||||||
|
|
||||||
clearChildren();
|
|
||||||
|
|
||||||
auto webDefaults = useWebDefaults_;
|
|
||||||
*this = YGNode{getConfig()};
|
|
||||||
if (webDefaults) {
|
|
||||||
useWebDefaults();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
323
deps/yoga/YGNode.h
vendored
323
deps/yoga/YGNode.h
vendored
@ -1,323 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) Facebook, Inc. and its affiliates.
|
|
||||||
*
|
|
||||||
* This source code is licensed under the MIT license found in the LICENSE
|
|
||||||
* file in the root directory of this source tree.
|
|
||||||
*/
|
|
||||||
#pragma once
|
|
||||||
#include <cstdint>
|
|
||||||
#include <stdio.h>
|
|
||||||
#include "CompactValue.h"
|
|
||||||
#include "YGConfig.h"
|
|
||||||
#include "YGLayout.h"
|
|
||||||
#include "YGStyle.h"
|
|
||||||
#include "YGMacros.h"
|
|
||||||
#include "Yoga-internal.h"
|
|
||||||
|
|
||||||
YGConfigRef YGConfigGetDefault();
|
|
||||||
|
|
||||||
struct YGNode {
|
|
||||||
using MeasureWithContextFn =
|
|
||||||
YGSize (*)(YGNode*, float, YGMeasureMode, float, YGMeasureMode, void*);
|
|
||||||
using BaselineWithContextFn = float (*)(YGNode*, float, float, void*);
|
|
||||||
using PrintWithContextFn = void (*)(YGNode*, void*);
|
|
||||||
|
|
||||||
private:
|
|
||||||
void* context_ = nullptr;
|
|
||||||
bool hasNewLayout_ : 1;
|
|
||||||
bool isReferenceBaseline_ : 1;
|
|
||||||
bool isDirty_ : 1;
|
|
||||||
YGNodeType nodeType_ : 1;
|
|
||||||
bool measureUsesContext_ : 1;
|
|
||||||
bool baselineUsesContext_ : 1;
|
|
||||||
bool printUsesContext_ : 1;
|
|
||||||
bool useWebDefaults_ : 1;
|
|
||||||
uint8_t reserved_ = 0;
|
|
||||||
union {
|
|
||||||
YGMeasureFunc noContext;
|
|
||||||
MeasureWithContextFn withContext;
|
|
||||||
} measure_ = {nullptr};
|
|
||||||
union {
|
|
||||||
YGBaselineFunc noContext;
|
|
||||||
BaselineWithContextFn withContext;
|
|
||||||
} baseline_ = {nullptr};
|
|
||||||
union {
|
|
||||||
YGPrintFunc noContext;
|
|
||||||
PrintWithContextFn withContext;
|
|
||||||
} print_ = {nullptr};
|
|
||||||
YGDirtiedFunc dirtied_ = nullptr;
|
|
||||||
YGStyle style_ = {};
|
|
||||||
YGLayout layout_ = {};
|
|
||||||
uint32_t lineIndex_ = 0;
|
|
||||||
YGNodeRef owner_ = nullptr;
|
|
||||||
YGVector children_ = {};
|
|
||||||
YGConfigRef config_;
|
|
||||||
std::array<YGValue, 2> resolvedDimensions_ = {
|
|
||||||
{YGValueUndefined, YGValueUndefined}};
|
|
||||||
|
|
||||||
YGFloatOptional relativePosition(
|
|
||||||
const YGFlexDirection axis,
|
|
||||||
const float axisSize) const;
|
|
||||||
|
|
||||||
void setMeasureFunc(decltype(measure_));
|
|
||||||
void setBaselineFunc(decltype(baseline_));
|
|
||||||
|
|
||||||
void useWebDefaults() {
|
|
||||||
useWebDefaults_ = true;
|
|
||||||
style_.flexDirection() = YGFlexDirectionRow;
|
|
||||||
style_.alignContent() = YGAlignStretch;
|
|
||||||
}
|
|
||||||
|
|
||||||
// DANGER DANGER DANGER!
|
|
||||||
// If the the node assigned to has children, we'd either have to deallocate
|
|
||||||
// them (potentially incorrect) or ignore them (danger of leaks). Only ever
|
|
||||||
// use this after checking that there are no children.
|
|
||||||
// DO NOT CHANGE THE VISIBILITY OF THIS METHOD!
|
|
||||||
YGNode& operator=(YGNode&&) = default;
|
|
||||||
|
|
||||||
using CompactValue = facebook::yoga::detail::CompactValue;
|
|
||||||
|
|
||||||
public:
|
|
||||||
YGNode() : YGNode{YGConfigGetDefault()} {}
|
|
||||||
explicit YGNode(const YGConfigRef config)
|
|
||||||
: hasNewLayout_{true},
|
|
||||||
isReferenceBaseline_{false},
|
|
||||||
isDirty_{false},
|
|
||||||
nodeType_{YGNodeTypeDefault},
|
|
||||||
measureUsesContext_{false},
|
|
||||||
baselineUsesContext_{false},
|
|
||||||
printUsesContext_{false},
|
|
||||||
useWebDefaults_{config->useWebDefaults},
|
|
||||||
config_{config} {
|
|
||||||
if (useWebDefaults_) {
|
|
||||||
useWebDefaults();
|
|
||||||
}
|
|
||||||
};
|
|
||||||
~YGNode() = default; // cleanup of owner/children relationships in YGNodeFree
|
|
||||||
|
|
||||||
YGNode(YGNode&&);
|
|
||||||
|
|
||||||
// Does not expose true value semantics, as children are not cloned eagerly.
|
|
||||||
// Should we remove this?
|
|
||||||
YGNode(const YGNode& node) = default;
|
|
||||||
|
|
||||||
// for RB fabric
|
|
||||||
YGNode(const YGNode& node, YGConfigRef config);
|
|
||||||
|
|
||||||
// assignment means potential leaks of existing children, or alternatively
|
|
||||||
// freeing unowned memory, double free, or freeing stack memory.
|
|
||||||
YGNode& operator=(const YGNode&) = delete;
|
|
||||||
|
|
||||||
// Getters
|
|
||||||
void* getContext() const { return context_; }
|
|
||||||
|
|
||||||
uint8_t& reserved() { return reserved_; }
|
|
||||||
uint8_t reserved() const { return reserved_; }
|
|
||||||
|
|
||||||
void print(void*);
|
|
||||||
|
|
||||||
bool getHasNewLayout() const { return hasNewLayout_; }
|
|
||||||
|
|
||||||
YGNodeType getNodeType() const { return nodeType_; }
|
|
||||||
|
|
||||||
bool hasMeasureFunc() const noexcept { return measure_.noContext != nullptr; }
|
|
||||||
|
|
||||||
YGSize measure(float, YGMeasureMode, float, YGMeasureMode, void*);
|
|
||||||
|
|
||||||
bool hasBaselineFunc() const noexcept {
|
|
||||||
return baseline_.noContext != nullptr;
|
|
||||||
}
|
|
||||||
|
|
||||||
float baseline(float width, float height, void* layoutContext);
|
|
||||||
|
|
||||||
YGDirtiedFunc getDirtied() const { return dirtied_; }
|
|
||||||
|
|
||||||
// For Performance reasons passing as reference.
|
|
||||||
YGStyle& getStyle() { return style_; }
|
|
||||||
|
|
||||||
const YGStyle& getStyle() const { return style_; }
|
|
||||||
|
|
||||||
// For Performance reasons passing as reference.
|
|
||||||
YGLayout& getLayout() { return layout_; }
|
|
||||||
|
|
||||||
const YGLayout& getLayout() const { return layout_; }
|
|
||||||
|
|
||||||
uint32_t getLineIndex() const { return lineIndex_; }
|
|
||||||
|
|
||||||
bool isReferenceBaseline() { return isReferenceBaseline_; }
|
|
||||||
|
|
||||||
// returns the YGNodeRef that owns this YGNode. An owner is used to identify
|
|
||||||
// the YogaTree that a YGNode belongs to. This method will return the parent
|
|
||||||
// of the YGNode when a YGNode only belongs to one YogaTree or nullptr when
|
|
||||||
// the YGNode is shared between two or more YogaTrees.
|
|
||||||
YGNodeRef getOwner() const { return owner_; }
|
|
||||||
|
|
||||||
// Deprecated, use getOwner() instead.
|
|
||||||
YGNodeRef getParent() const { return getOwner(); }
|
|
||||||
|
|
||||||
const YGVector& getChildren() const { return children_; }
|
|
||||||
|
|
||||||
// Applies a callback to all children, after cloning them if they are not
|
|
||||||
// owned.
|
|
||||||
template <typename T>
|
|
||||||
void iterChildrenAfterCloningIfNeeded(T callback, void* cloneContext) {
|
|
||||||
int i = 0;
|
|
||||||
for (YGNodeRef& child : children_) {
|
|
||||||
if (child->getOwner() != this) {
|
|
||||||
child = config_->cloneNode(child, this, i, cloneContext);
|
|
||||||
child->setOwner(this);
|
|
||||||
}
|
|
||||||
i += 1;
|
|
||||||
|
|
||||||
callback(child, cloneContext);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
YGNodeRef getChild(uint32_t index) const { return children_.at(index); }
|
|
||||||
|
|
||||||
YGConfigRef getConfig() const { return config_; }
|
|
||||||
|
|
||||||
bool isDirty() const { return isDirty_; }
|
|
||||||
|
|
||||||
std::array<YGValue, 2> getResolvedDimensions() const {
|
|
||||||
return resolvedDimensions_;
|
|
||||||
}
|
|
||||||
|
|
||||||
YGValue getResolvedDimension(int index) const {
|
|
||||||
return resolvedDimensions_[index];
|
|
||||||
}
|
|
||||||
|
|
||||||
// Methods related to positions, margin, padding and border
|
|
||||||
YGFloatOptional getLeadingPosition(
|
|
||||||
const YGFlexDirection axis,
|
|
||||||
const float axisSize) const;
|
|
||||||
bool isLeadingPositionDefined(const YGFlexDirection axis) const;
|
|
||||||
bool isTrailingPosDefined(const YGFlexDirection axis) const;
|
|
||||||
YGFloatOptional getTrailingPosition(
|
|
||||||
const YGFlexDirection axis,
|
|
||||||
const float axisSize) const;
|
|
||||||
YGFloatOptional getLeadingMargin(
|
|
||||||
const YGFlexDirection axis,
|
|
||||||
const float widthSize) const;
|
|
||||||
YGFloatOptional getTrailingMargin(
|
|
||||||
const YGFlexDirection axis,
|
|
||||||
const float widthSize) const;
|
|
||||||
float getLeadingBorder(const YGFlexDirection flexDirection) const;
|
|
||||||
float getTrailingBorder(const YGFlexDirection flexDirection) const;
|
|
||||||
YGFloatOptional getLeadingPadding(
|
|
||||||
const YGFlexDirection axis,
|
|
||||||
const float widthSize) const;
|
|
||||||
YGFloatOptional getTrailingPadding(
|
|
||||||
const YGFlexDirection axis,
|
|
||||||
const float widthSize) const;
|
|
||||||
YGFloatOptional getLeadingPaddingAndBorder(
|
|
||||||
const YGFlexDirection axis,
|
|
||||||
const float widthSize) const;
|
|
||||||
YGFloatOptional getTrailingPaddingAndBorder(
|
|
||||||
const YGFlexDirection axis,
|
|
||||||
const float widthSize) const;
|
|
||||||
YGFloatOptional getMarginForAxis(
|
|
||||||
const YGFlexDirection axis,
|
|
||||||
const float widthSize) const;
|
|
||||||
// Setters
|
|
||||||
|
|
||||||
void setContext(void* context) { context_ = context; }
|
|
||||||
|
|
||||||
void setPrintFunc(YGPrintFunc printFunc) {
|
|
||||||
print_.noContext = printFunc;
|
|
||||||
printUsesContext_ = false;
|
|
||||||
}
|
|
||||||
void setPrintFunc(PrintWithContextFn printFunc) {
|
|
||||||
print_.withContext = printFunc;
|
|
||||||
printUsesContext_ = true;
|
|
||||||
}
|
|
||||||
void setPrintFunc(std::nullptr_t) { setPrintFunc(YGPrintFunc{nullptr}); }
|
|
||||||
|
|
||||||
void setHasNewLayout(bool hasNewLayout) { hasNewLayout_ = hasNewLayout; }
|
|
||||||
|
|
||||||
void setNodeType(YGNodeType nodeType) { nodeType_ = nodeType; }
|
|
||||||
|
|
||||||
void setMeasureFunc(YGMeasureFunc measureFunc);
|
|
||||||
void setMeasureFunc(MeasureWithContextFn);
|
|
||||||
void setMeasureFunc(std::nullptr_t) {
|
|
||||||
return setMeasureFunc(YGMeasureFunc{nullptr});
|
|
||||||
}
|
|
||||||
|
|
||||||
void setBaselineFunc(YGBaselineFunc baseLineFunc) {
|
|
||||||
baselineUsesContext_ = false;
|
|
||||||
baseline_.noContext = baseLineFunc;
|
|
||||||
}
|
|
||||||
void setBaselineFunc(BaselineWithContextFn baseLineFunc) {
|
|
||||||
baselineUsesContext_ = true;
|
|
||||||
baseline_.withContext = baseLineFunc;
|
|
||||||
}
|
|
||||||
void setBaselineFunc(std::nullptr_t) {
|
|
||||||
return setBaselineFunc(YGBaselineFunc{nullptr});
|
|
||||||
}
|
|
||||||
|
|
||||||
void setDirtiedFunc(YGDirtiedFunc dirtiedFunc) { dirtied_ = dirtiedFunc; }
|
|
||||||
|
|
||||||
void setStyle(const YGStyle& style) { style_ = style; }
|
|
||||||
|
|
||||||
void setLayout(const YGLayout& layout) { layout_ = layout; }
|
|
||||||
|
|
||||||
void setLineIndex(uint32_t lineIndex) { lineIndex_ = lineIndex; }
|
|
||||||
|
|
||||||
void setIsReferenceBaseline(bool isReferenceBaseline) {
|
|
||||||
isReferenceBaseline_ = isReferenceBaseline;
|
|
||||||
}
|
|
||||||
|
|
||||||
void setOwner(YGNodeRef owner) { owner_ = owner; }
|
|
||||||
|
|
||||||
void setChildren(const YGVector& children) { children_ = children; }
|
|
||||||
|
|
||||||
// TODO: rvalue override for setChildren
|
|
||||||
|
|
||||||
YG_DEPRECATED void setConfig(YGConfigRef config) { config_ = config; }
|
|
||||||
|
|
||||||
void setDirty(bool isDirty);
|
|
||||||
void setLayoutLastOwnerDirection(YGDirection direction);
|
|
||||||
void setLayoutComputedFlexBasis(const YGFloatOptional computedFlexBasis);
|
|
||||||
void setLayoutComputedFlexBasisGeneration(
|
|
||||||
uint32_t computedFlexBasisGeneration);
|
|
||||||
void setLayoutMeasuredDimension(float measuredDimension, int index);
|
|
||||||
void setLayoutHadOverflow(bool hadOverflow);
|
|
||||||
void setLayoutDimension(float dimension, int index);
|
|
||||||
void setLayoutDirection(YGDirection direction);
|
|
||||||
void setLayoutMargin(float margin, int index);
|
|
||||||
void setLayoutBorder(float border, int index);
|
|
||||||
void setLayoutPadding(float padding, int index);
|
|
||||||
void setLayoutPosition(float position, int index);
|
|
||||||
void setPosition(
|
|
||||||
const YGDirection direction,
|
|
||||||
const float mainSize,
|
|
||||||
const float crossSize,
|
|
||||||
const float ownerWidth);
|
|
||||||
void setLayoutDoesLegacyFlagAffectsLayout(bool doesLegacyFlagAffectsLayout);
|
|
||||||
void setLayoutDidUseLegacyFlag(bool didUseLegacyFlag);
|
|
||||||
void markDirtyAndPropogateDownwards();
|
|
||||||
|
|
||||||
// Other methods
|
|
||||||
YGValue marginLeadingValue(const YGFlexDirection axis) const;
|
|
||||||
YGValue marginTrailingValue(const YGFlexDirection axis) const;
|
|
||||||
YGValue resolveFlexBasisPtr() const;
|
|
||||||
void resolveDimension();
|
|
||||||
YGDirection resolveDirection(const YGDirection ownerDirection);
|
|
||||||
void clearChildren();
|
|
||||||
/// Replaces the occurrences of oldChild with newChild
|
|
||||||
void replaceChild(YGNodeRef oldChild, YGNodeRef newChild);
|
|
||||||
void replaceChild(YGNodeRef child, uint32_t index);
|
|
||||||
void insertChild(YGNodeRef child, uint32_t index);
|
|
||||||
/// Removes the first occurrence of child
|
|
||||||
bool removeChild(YGNodeRef child);
|
|
||||||
void removeChild(uint32_t index);
|
|
||||||
|
|
||||||
void cloneChildrenIfNeeded(void*);
|
|
||||||
void markDirtyAndPropogate();
|
|
||||||
float resolveFlexGrow() const;
|
|
||||||
float resolveFlexShrink() const;
|
|
||||||
bool isNodeFlexible();
|
|
||||||
bool didUseLegacyFlag();
|
|
||||||
bool isLayoutTreeEqualToNode(const YGNode& node) const;
|
|
||||||
void reset();
|
|
||||||
};
|
|
||||||
224
deps/yoga/YGNodePrint.cpp
vendored
224
deps/yoga/YGNodePrint.cpp
vendored
@ -1,224 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) Facebook, Inc. and its affiliates.
|
|
||||||
*
|
|
||||||
* This source code is licensed under the MIT license found in the LICENSE
|
|
||||||
* file in the root directory of this source tree.
|
|
||||||
*/
|
|
||||||
#ifdef DEBUG
|
|
||||||
#include "YGNodePrint.h"
|
|
||||||
#include <stdarg.h>
|
|
||||||
#include "YGEnums.h"
|
|
||||||
#include "YGNode.h"
|
|
||||||
#include "Yoga-internal.h"
|
|
||||||
#include "Utils.h"
|
|
||||||
|
|
||||||
namespace facebook {
|
|
||||||
namespace yoga {
|
|
||||||
typedef std::string string;
|
|
||||||
|
|
||||||
static void indent(string& base, uint32_t level) {
|
|
||||||
for (uint32_t i = 0; i < level; ++i) {
|
|
||||||
base.append(" ");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static bool areFourValuesEqual(const YGStyle::Edges& four) {
|
|
||||||
return YGValueEqual(four[0], four[1]) && YGValueEqual(four[0], four[2]) &&
|
|
||||||
YGValueEqual(four[0], four[3]);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void appendFormatedString(string& str, const char* fmt, ...) {
|
|
||||||
va_list args;
|
|
||||||
va_start(args, fmt);
|
|
||||||
va_list argsCopy;
|
|
||||||
va_copy(argsCopy, args);
|
|
||||||
std::vector<char> buf(1 + vsnprintf(NULL, 0, fmt, args));
|
|
||||||
va_end(args);
|
|
||||||
vsnprintf(buf.data(), buf.size(), fmt, argsCopy);
|
|
||||||
va_end(argsCopy);
|
|
||||||
string result = string(buf.begin(), buf.end() - 1);
|
|
||||||
str.append(result);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void appendFloatOptionalIfDefined(
|
|
||||||
string& base,
|
|
||||||
const string key,
|
|
||||||
const YGFloatOptional num) {
|
|
||||||
if (!num.isUndefined()) {
|
|
||||||
appendFormatedString(base, "%s: %g; ", key.c_str(), num.unwrap());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static void appendNumberIfNotUndefined(
|
|
||||||
string& base,
|
|
||||||
const string key,
|
|
||||||
const YGValue number) {
|
|
||||||
if (number.unit != YGUnitUndefined) {
|
|
||||||
if (number.unit == YGUnitAuto) {
|
|
||||||
base.append(key + ": auto; ");
|
|
||||||
} else {
|
|
||||||
string unit = number.unit == YGUnitPoint ? "px" : "%%";
|
|
||||||
appendFormatedString(
|
|
||||||
base, "%s: %g%s; ", key.c_str(), number.value, unit.c_str());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static void appendNumberIfNotAuto(
|
|
||||||
string& base,
|
|
||||||
const string& key,
|
|
||||||
const YGValue number) {
|
|
||||||
if (number.unit != YGUnitAuto) {
|
|
||||||
appendNumberIfNotUndefined(base, key, number);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static void appendNumberIfNotZero(
|
|
||||||
string& base,
|
|
||||||
const string& str,
|
|
||||||
const YGValue number) {
|
|
||||||
if (number.unit == YGUnitAuto) {
|
|
||||||
base.append(str + ": auto; ");
|
|
||||||
} else if (!YGFloatsEqual(number.value, 0)) {
|
|
||||||
appendNumberIfNotUndefined(base, str, number);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static void appendEdges(
|
|
||||||
string& base,
|
|
||||||
const string& key,
|
|
||||||
const YGStyle::Edges& edges) {
|
|
||||||
if (areFourValuesEqual(edges)) {
|
|
||||||
appendNumberIfNotZero(base, key, edges[YGEdgeLeft]);
|
|
||||||
} else {
|
|
||||||
for (int edge = YGEdgeLeft; edge != YGEdgeAll; ++edge) {
|
|
||||||
string str = key + "-" + YGEdgeToString(static_cast<YGEdge>(edge));
|
|
||||||
appendNumberIfNotZero(base, str, edges[edge]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static void appendEdgeIfNotUndefined(
|
|
||||||
string& base,
|
|
||||||
const string& str,
|
|
||||||
const YGStyle::Edges& edges,
|
|
||||||
const YGEdge edge) {
|
|
||||||
appendNumberIfNotUndefined(
|
|
||||||
base,
|
|
||||||
str,
|
|
||||||
YGComputedEdgeValue(edges, edge, detail::CompactValue::ofUndefined()));
|
|
||||||
}
|
|
||||||
|
|
||||||
void YGNodeToString(
|
|
||||||
std::string& str,
|
|
||||||
YGNodeRef node,
|
|
||||||
YGPrintOptions options,
|
|
||||||
uint32_t level) {
|
|
||||||
indent(str, level);
|
|
||||||
appendFormatedString(str, "<div ");
|
|
||||||
|
|
||||||
if (options & YGPrintOptionsLayout) {
|
|
||||||
appendFormatedString(str, "layout=\"");
|
|
||||||
appendFormatedString(
|
|
||||||
str, "width: %g; ", node->getLayout().dimensions[YGDimensionWidth]);
|
|
||||||
appendFormatedString(
|
|
||||||
str, "height: %g; ", node->getLayout().dimensions[YGDimensionHeight]);
|
|
||||||
appendFormatedString(
|
|
||||||
str, "top: %g; ", node->getLayout().position[YGEdgeTop]);
|
|
||||||
appendFormatedString(
|
|
||||||
str, "left: %g;", node->getLayout().position[YGEdgeLeft]);
|
|
||||||
appendFormatedString(str, "\" ");
|
|
||||||
}
|
|
||||||
|
|
||||||
if (options & YGPrintOptionsStyle) {
|
|
||||||
appendFormatedString(str, "style=\"");
|
|
||||||
const auto& style = node->getStyle();
|
|
||||||
if (style.flexDirection() != YGNode().getStyle().flexDirection()) {
|
|
||||||
appendFormatedString(
|
|
||||||
str,
|
|
||||||
"flex-direction: %s; ",
|
|
||||||
YGFlexDirectionToString(style.flexDirection()));
|
|
||||||
}
|
|
||||||
if (style.justifyContent() != YGNode().getStyle().justifyContent()) {
|
|
||||||
appendFormatedString(
|
|
||||||
str,
|
|
||||||
"justify-content: %s; ",
|
|
||||||
YGJustifyToString(style.justifyContent()));
|
|
||||||
}
|
|
||||||
if (style.alignItems() != YGNode().getStyle().alignItems()) {
|
|
||||||
appendFormatedString(
|
|
||||||
str, "align-items: %s; ", YGAlignToString(style.alignItems()));
|
|
||||||
}
|
|
||||||
if (style.alignContent() != YGNode().getStyle().alignContent()) {
|
|
||||||
appendFormatedString(
|
|
||||||
str, "align-content: %s; ", YGAlignToString(style.alignContent()));
|
|
||||||
}
|
|
||||||
if (style.alignSelf() != YGNode().getStyle().alignSelf()) {
|
|
||||||
appendFormatedString(
|
|
||||||
str, "align-self: %s; ", YGAlignToString(style.alignSelf()));
|
|
||||||
}
|
|
||||||
appendFloatOptionalIfDefined(str, "flex-grow", style.flexGrow());
|
|
||||||
appendFloatOptionalIfDefined(str, "flex-shrink", style.flexShrink());
|
|
||||||
appendNumberIfNotAuto(str, "flex-basis", style.flexBasis());
|
|
||||||
appendFloatOptionalIfDefined(str, "flex", style.flex());
|
|
||||||
|
|
||||||
if (style.flexWrap() != YGNode().getStyle().flexWrap()) {
|
|
||||||
appendFormatedString(
|
|
||||||
str, "flex-wrap: %s; ", YGWrapToString(style.flexWrap()));
|
|
||||||
}
|
|
||||||
|
|
||||||
if (style.overflow() != YGNode().getStyle().overflow()) {
|
|
||||||
appendFormatedString(
|
|
||||||
str, "overflow: %s; ", YGOverflowToString(style.overflow()));
|
|
||||||
}
|
|
||||||
|
|
||||||
if (style.display() != YGNode().getStyle().display()) {
|
|
||||||
appendFormatedString(
|
|
||||||
str, "display: %s; ", YGDisplayToString(style.display()));
|
|
||||||
}
|
|
||||||
appendEdges(str, "margin", style.margin());
|
|
||||||
appendEdges(str, "padding", style.padding());
|
|
||||||
appendEdges(str, "border", style.border());
|
|
||||||
|
|
||||||
appendNumberIfNotAuto(str, "width", style.dimensions()[YGDimensionWidth]);
|
|
||||||
appendNumberIfNotAuto(str, "height", style.dimensions()[YGDimensionHeight]);
|
|
||||||
appendNumberIfNotAuto(
|
|
||||||
str, "max-width", style.maxDimensions()[YGDimensionWidth]);
|
|
||||||
appendNumberIfNotAuto(
|
|
||||||
str, "max-height", style.maxDimensions()[YGDimensionHeight]);
|
|
||||||
appendNumberIfNotAuto(
|
|
||||||
str, "min-width", style.minDimensions()[YGDimensionWidth]);
|
|
||||||
appendNumberIfNotAuto(
|
|
||||||
str, "min-height", style.minDimensions()[YGDimensionHeight]);
|
|
||||||
|
|
||||||
if (style.positionType() != YGNode().getStyle().positionType()) {
|
|
||||||
appendFormatedString(
|
|
||||||
str, "position: %s; ", YGPositionTypeToString(style.positionType()));
|
|
||||||
}
|
|
||||||
|
|
||||||
appendEdgeIfNotUndefined(str, "left", style.position(), YGEdgeLeft);
|
|
||||||
appendEdgeIfNotUndefined(str, "right", style.position(), YGEdgeRight);
|
|
||||||
appendEdgeIfNotUndefined(str, "top", style.position(), YGEdgeTop);
|
|
||||||
appendEdgeIfNotUndefined(str, "bottom", style.position(), YGEdgeBottom);
|
|
||||||
appendFormatedString(str, "\" ");
|
|
||||||
|
|
||||||
if (node->hasMeasureFunc()) {
|
|
||||||
appendFormatedString(str, "has-custom-measure=\"true\"");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
appendFormatedString(str, ">");
|
|
||||||
|
|
||||||
const uint32_t childCount = static_cast<uint32_t>(node->getChildren().size());
|
|
||||||
if (options & YGPrintOptionsChildren && childCount > 0) {
|
|
||||||
for (uint32_t i = 0; i < childCount; i++) {
|
|
||||||
appendFormatedString(str, "\n");
|
|
||||||
YGNodeToString(str, YGNodeGetChild(node, i), options, level + 1);
|
|
||||||
}
|
|
||||||
appendFormatedString(str, "\n");
|
|
||||||
indent(str, level);
|
|
||||||
}
|
|
||||||
appendFormatedString(str, "</div>");
|
|
||||||
}
|
|
||||||
} // namespace yoga
|
|
||||||
} // namespace facebook
|
|
||||||
#endif
|
|
||||||
24
deps/yoga/YGNodePrint.h
vendored
24
deps/yoga/YGNodePrint.h
vendored
@ -1,24 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) Facebook, Inc. and its affiliates.
|
|
||||||
*
|
|
||||||
* This source code is licensed under the MIT license found in the LICENSE
|
|
||||||
* file in the root directory of this source tree.
|
|
||||||
*/
|
|
||||||
#ifdef DEBUG
|
|
||||||
#pragma once
|
|
||||||
#include <string>
|
|
||||||
|
|
||||||
#include "Yoga.h"
|
|
||||||
|
|
||||||
namespace facebook {
|
|
||||||
namespace yoga {
|
|
||||||
|
|
||||||
void YGNodeToString(
|
|
||||||
std::string& str,
|
|
||||||
YGNodeRef node,
|
|
||||||
YGPrintOptions options,
|
|
||||||
uint32_t level);
|
|
||||||
|
|
||||||
} // namespace yoga
|
|
||||||
} // namespace facebook
|
|
||||||
#endif
|
|
||||||
55
deps/yoga/YGStyle.cpp
vendored
55
deps/yoga/YGStyle.cpp
vendored
@ -1,55 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) Facebook, Inc. and its affiliates.
|
|
||||||
*
|
|
||||||
* This source code is licensed under the MIT license found in the LICENSE
|
|
||||||
* file in the root directory of this source tree.
|
|
||||||
*/
|
|
||||||
#include "YGStyle.h"
|
|
||||||
#include "Utils.h"
|
|
||||||
|
|
||||||
// Yoga specific properties, not compatible with flexbox specification
|
|
||||||
bool operator==(const YGStyle& lhs, const YGStyle& rhs) {
|
|
||||||
bool areNonFloatValuesEqual = lhs.direction() == rhs.direction() &&
|
|
||||||
lhs.flexDirection() == rhs.flexDirection() &&
|
|
||||||
lhs.justifyContent() == rhs.justifyContent() &&
|
|
||||||
lhs.alignContent() == rhs.alignContent() &&
|
|
||||||
lhs.alignItems() == rhs.alignItems() &&
|
|
||||||
lhs.alignSelf() == rhs.alignSelf() &&
|
|
||||||
lhs.positionType() == rhs.positionType() &&
|
|
||||||
lhs.flexWrap() == rhs.flexWrap() && lhs.overflow() == rhs.overflow() &&
|
|
||||||
lhs.display() == rhs.display() &&
|
|
||||||
YGValueEqual(lhs.flexBasis(), rhs.flexBasis()) &&
|
|
||||||
lhs.margin() == rhs.margin() && lhs.position() == rhs.position() &&
|
|
||||||
lhs.padding() == rhs.padding() && lhs.border() == rhs.border() &&
|
|
||||||
lhs.dimensions() == rhs.dimensions() &&
|
|
||||||
lhs.minDimensions() == rhs.minDimensions() &&
|
|
||||||
lhs.maxDimensions() == rhs.maxDimensions();
|
|
||||||
|
|
||||||
areNonFloatValuesEqual = areNonFloatValuesEqual &&
|
|
||||||
lhs.flex().isUndefined() == rhs.flex().isUndefined();
|
|
||||||
if (areNonFloatValuesEqual && !lhs.flex().isUndefined() &&
|
|
||||||
!rhs.flex().isUndefined()) {
|
|
||||||
areNonFloatValuesEqual = areNonFloatValuesEqual && lhs.flex() == rhs.flex();
|
|
||||||
}
|
|
||||||
|
|
||||||
areNonFloatValuesEqual = areNonFloatValuesEqual &&
|
|
||||||
lhs.flexGrow().isUndefined() == rhs.flexGrow().isUndefined();
|
|
||||||
if (areNonFloatValuesEqual && !lhs.flexGrow().isUndefined()) {
|
|
||||||
areNonFloatValuesEqual =
|
|
||||||
areNonFloatValuesEqual && lhs.flexGrow() == rhs.flexGrow();
|
|
||||||
}
|
|
||||||
|
|
||||||
areNonFloatValuesEqual = areNonFloatValuesEqual &&
|
|
||||||
lhs.flexShrink().isUndefined() == rhs.flexShrink().isUndefined();
|
|
||||||
if (areNonFloatValuesEqual && !rhs.flexShrink().isUndefined()) {
|
|
||||||
areNonFloatValuesEqual =
|
|
||||||
areNonFloatValuesEqual && lhs.flexShrink() == rhs.flexShrink();
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!(lhs.aspectRatio().isUndefined() && rhs.aspectRatio().isUndefined())) {
|
|
||||||
areNonFloatValuesEqual =
|
|
||||||
areNonFloatValuesEqual && lhs.aspectRatio() == rhs.aspectRatio();
|
|
||||||
}
|
|
||||||
|
|
||||||
return areNonFloatValuesEqual;
|
|
||||||
}
|
|
||||||
301
deps/yoga/YGStyle.h
vendored
301
deps/yoga/YGStyle.h
vendored
@ -1,301 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) Facebook, Inc. and its affiliates.
|
|
||||||
*
|
|
||||||
* This source code is licensed under the MIT license found in the LICENSE
|
|
||||||
* file in the root directory of this source tree.
|
|
||||||
*/
|
|
||||||
#pragma once
|
|
||||||
#include <algorithm>
|
|
||||||
#include <array>
|
|
||||||
#include <bitset>
|
|
||||||
#include <cstdint>
|
|
||||||
#include <type_traits>
|
|
||||||
#include "CompactValue.h"
|
|
||||||
#include "YGEnums.h"
|
|
||||||
#include "YGFloatOptional.h"
|
|
||||||
#include "Yoga-internal.h"
|
|
||||||
#include "Yoga.h"
|
|
||||||
|
|
||||||
#if !defined(ENUM_BITFIELDS_NOT_SUPPORTED)
|
|
||||||
#define BITFIELD_ENUM_SIZED(num) : num
|
|
||||||
#else
|
|
||||||
#define BITFIELD_ENUM_SIZED(num)
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#define BITFIELD_ACCESSORS(FIELD) \
|
|
||||||
decltype(FIELD##_) get_##FIELD() const { return FIELD##_; } \
|
|
||||||
void set_##FIELD(decltype(FIELD##_) x) { FIELD##_ = x; }
|
|
||||||
|
|
||||||
#define BITFIELD_REF(FIELD) \
|
|
||||||
BitfieldRef< \
|
|
||||||
decltype(FIELD##_), \
|
|
||||||
&YGStyle::get_##FIELD, \
|
|
||||||
&YGStyle::set_##FIELD, \
|
|
||||||
FIELD##Bit>
|
|
||||||
|
|
||||||
class YGStyle {
|
|
||||||
template <typename Enum>
|
|
||||||
using Values =
|
|
||||||
facebook::yoga::detail::Values<facebook::yoga::enums::count<Enum>()>;
|
|
||||||
using CompactValue = facebook::yoga::detail::CompactValue;
|
|
||||||
|
|
||||||
static constexpr uint64_t allBits(int fromBit, int toBit) {
|
|
||||||
return fromBit < toBit
|
|
||||||
? (uint64_t{1} << fromBit) | allBits(fromBit + 1, toBit)
|
|
||||||
: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
public:
|
|
||||||
using Dimensions = Values<YGDimension>;
|
|
||||||
using Edges = Values<YGEdge>;
|
|
||||||
|
|
||||||
template <typename T, T YGStyle::*Prop, int PropBit>
|
|
||||||
struct Ref {
|
|
||||||
YGStyle& style;
|
|
||||||
operator T() const { return style.*Prop; }
|
|
||||||
Ref<T, Prop, PropBit>& operator=(T value) {
|
|
||||||
style.*Prop = value;
|
|
||||||
style.assignedProps_.set(PropBit);
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
template <typename Idx, Values<Idx> YGStyle::*Prop, int PropBit>
|
|
||||||
struct IdxRef {
|
|
||||||
struct Ref {
|
|
||||||
YGStyle& style;
|
|
||||||
Idx idx;
|
|
||||||
operator CompactValue() const { return (style.*Prop)[idx]; }
|
|
||||||
operator YGValue() const { return (style.*Prop)[idx]; }
|
|
||||||
Ref& operator=(CompactValue value) {
|
|
||||||
(style.*Prop)[idx] = value;
|
|
||||||
style.assignedProps_.set(PropBit + idx);
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
YGStyle& style;
|
|
||||||
IdxRef<Idx, Prop, PropBit>& operator=(const Values<Idx>& values) {
|
|
||||||
style.*Prop = values;
|
|
||||||
style.assignedProps_ |=
|
|
||||||
allBits(PropBit, PropBit + facebook::yoga::enums::count<Idx>());
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
operator const Values<Idx>&() const { return style.*Prop; }
|
|
||||||
Ref operator[](Idx idx) { return {style, idx}; }
|
|
||||||
CompactValue operator[](Idx idx) const { return (style.*Prop)[idx]; }
|
|
||||||
};
|
|
||||||
|
|
||||||
template <
|
|
||||||
typename T,
|
|
||||||
T (YGStyle::*Get)() const,
|
|
||||||
void (YGStyle::*Set)(T),
|
|
||||||
int PropBit>
|
|
||||||
struct BitfieldRef {
|
|
||||||
YGStyle& style;
|
|
||||||
|
|
||||||
operator T() const { return (style.*Get)(); }
|
|
||||||
BitfieldRef<T, Get, Set, PropBit>& operator=(T x) {
|
|
||||||
(style.*Set)(x);
|
|
||||||
style.assignedProps_.set(PropBit);
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
#ifdef __clang__
|
|
||||||
#pragma clang diagnostic push
|
|
||||||
#pragma clang diagnostic ignored "-Wbitfield-constant-conversion"
|
|
||||||
#endif
|
|
||||||
|
|
||||||
YGStyle()
|
|
||||||
: direction_(YGDirectionInherit),
|
|
||||||
flexDirection_(YGFlexDirectionColumn),
|
|
||||||
justifyContent_(YGJustifyFlexStart),
|
|
||||||
alignContent_(YGAlignFlexStart),
|
|
||||||
alignItems_(YGAlignStretch),
|
|
||||||
alignSelf_(YGAlignAuto),
|
|
||||||
positionType_(YGPositionTypeRelative),
|
|
||||||
flexWrap_(YGWrapNoWrap),
|
|
||||||
overflow_(YGOverflowVisible),
|
|
||||||
display_(YGDisplayFlex) {}
|
|
||||||
|
|
||||||
#ifdef __clang__
|
|
||||||
#pragma clang diagnostic pop
|
|
||||||
#endif
|
|
||||||
|
|
||||||
~YGStyle() = default;
|
|
||||||
|
|
||||||
static constexpr int directionBit = 0;
|
|
||||||
static constexpr int flexDirectionBit = directionBit + 1;
|
|
||||||
static constexpr int justifyContentBit = flexDirectionBit + 1;
|
|
||||||
static constexpr int alignContentBit = justifyContentBit + 1;
|
|
||||||
static constexpr int alignItemsBit = alignContentBit + 1;
|
|
||||||
static constexpr int alignSelfBit = alignItemsBit + 1;
|
|
||||||
static constexpr int positionTypeBit = alignSelfBit + 1;
|
|
||||||
static constexpr int flexWrapBit = positionTypeBit + 1;
|
|
||||||
static constexpr int overflowBit = flexWrapBit + 1;
|
|
||||||
static constexpr int displayBit = overflowBit + 1;
|
|
||||||
static constexpr int flexBit = displayBit + 1;
|
|
||||||
static constexpr int flexGrowBit = flexBit + 1;
|
|
||||||
static constexpr int flexShrinkBit = flexGrowBit + 1;
|
|
||||||
static constexpr int flexBasisBit = flexShrinkBit + 1;
|
|
||||||
static constexpr int marginBit = flexBasisBit + 1;
|
|
||||||
static constexpr int positionBit =
|
|
||||||
marginBit + facebook::yoga::enums::count<YGEdge>();
|
|
||||||
static constexpr int paddingBit =
|
|
||||||
positionBit + facebook::yoga::enums::count<YGEdge>();
|
|
||||||
static constexpr int borderBit =
|
|
||||||
paddingBit + facebook::yoga::enums::count<YGEdge>();
|
|
||||||
static constexpr int dimensionsBit =
|
|
||||||
borderBit + facebook::yoga::enums::count<YGEdge>();
|
|
||||||
static constexpr int maxDimensionsBit =
|
|
||||||
dimensionsBit + facebook::yoga::enums::count<YGDimension>();
|
|
||||||
static constexpr int minDimensionsBit =
|
|
||||||
maxDimensionsBit + facebook::yoga::enums::count<YGDimension>();
|
|
||||||
static constexpr int aspectRatioBit =
|
|
||||||
minDimensionsBit + facebook::yoga::enums::count<YGDimension>();
|
|
||||||
|
|
||||||
static constexpr int numStyles = aspectRatioBit + 1;
|
|
||||||
|
|
||||||
private:
|
|
||||||
std::bitset<aspectRatioBit + 1> assignedProps_;
|
|
||||||
|
|
||||||
/* Some platforms don't support enum bitfields,
|
|
||||||
so please use BITFIELD_ENUM_SIZED(BITS_COUNT) */
|
|
||||||
YGDirection direction_ BITFIELD_ENUM_SIZED(2);
|
|
||||||
YGFlexDirection flexDirection_ BITFIELD_ENUM_SIZED(2);
|
|
||||||
YGJustify justifyContent_ BITFIELD_ENUM_SIZED(3);
|
|
||||||
YGAlign alignContent_ BITFIELD_ENUM_SIZED(3);
|
|
||||||
YGAlign alignItems_ BITFIELD_ENUM_SIZED(3);
|
|
||||||
YGAlign alignSelf_ BITFIELD_ENUM_SIZED(3);
|
|
||||||
YGPositionType positionType_ BITFIELD_ENUM_SIZED(1);
|
|
||||||
YGWrap flexWrap_ BITFIELD_ENUM_SIZED(2);
|
|
||||||
YGOverflow overflow_ BITFIELD_ENUM_SIZED(2);
|
|
||||||
YGDisplay display_ BITFIELD_ENUM_SIZED(1);
|
|
||||||
YGFloatOptional flex_ = {};
|
|
||||||
YGFloatOptional flexGrow_ = {};
|
|
||||||
YGFloatOptional flexShrink_ = {};
|
|
||||||
CompactValue flexBasis_ = CompactValue::ofAuto();
|
|
||||||
Edges margin_ = {};
|
|
||||||
Edges position_ = {};
|
|
||||||
Edges padding_ = {};
|
|
||||||
Edges border_ = {};
|
|
||||||
Dimensions dimensions_{CompactValue::ofAuto()};
|
|
||||||
Dimensions minDimensions_ = {};
|
|
||||||
Dimensions maxDimensions_ = {};
|
|
||||||
// Yoga specific properties, not compatible with flexbox specification
|
|
||||||
YGFloatOptional aspectRatio_ = {};
|
|
||||||
|
|
||||||
BITFIELD_ACCESSORS(direction)
|
|
||||||
BITFIELD_ACCESSORS(flexDirection)
|
|
||||||
BITFIELD_ACCESSORS(justifyContent)
|
|
||||||
BITFIELD_ACCESSORS(alignContent);
|
|
||||||
BITFIELD_ACCESSORS(alignItems);
|
|
||||||
BITFIELD_ACCESSORS(alignSelf);
|
|
||||||
BITFIELD_ACCESSORS(positionType);
|
|
||||||
BITFIELD_ACCESSORS(flexWrap);
|
|
||||||
BITFIELD_ACCESSORS(overflow);
|
|
||||||
BITFIELD_ACCESSORS(display);
|
|
||||||
|
|
||||||
public:
|
|
||||||
const decltype(assignedProps_)& assignedProps() const {
|
|
||||||
return assignedProps_;
|
|
||||||
}
|
|
||||||
|
|
||||||
// for library users needing a type
|
|
||||||
using ValueRepr = std::remove_reference<decltype(margin_[0])>::type;
|
|
||||||
|
|
||||||
YGDirection direction() const { return direction_; }
|
|
||||||
BITFIELD_REF(direction) direction() { return {*this}; }
|
|
||||||
|
|
||||||
YGFlexDirection flexDirection() const { return flexDirection_; }
|
|
||||||
BITFIELD_REF(flexDirection) flexDirection() { return {*this}; }
|
|
||||||
|
|
||||||
YGJustify justifyContent() const { return justifyContent_; }
|
|
||||||
BITFIELD_REF(justifyContent) justifyContent() { return {*this}; }
|
|
||||||
|
|
||||||
YGAlign alignContent() const { return alignContent_; }
|
|
||||||
BITFIELD_REF(alignContent) alignContent() { return {*this}; }
|
|
||||||
|
|
||||||
YGAlign alignItems() const { return alignItems_; }
|
|
||||||
BITFIELD_REF(alignItems) alignItems() { return {*this}; }
|
|
||||||
|
|
||||||
YGAlign alignSelf() const { return alignSelf_; }
|
|
||||||
BITFIELD_REF(alignSelf) alignSelf() { return {*this}; }
|
|
||||||
|
|
||||||
YGPositionType positionType() const { return positionType_; }
|
|
||||||
BITFIELD_REF(positionType) positionType() { return {*this}; }
|
|
||||||
|
|
||||||
YGWrap flexWrap() const { return flexWrap_; }
|
|
||||||
BITFIELD_REF(flexWrap) flexWrap() { return {*this}; }
|
|
||||||
|
|
||||||
YGOverflow overflow() const { return overflow_; }
|
|
||||||
BITFIELD_REF(overflow) overflow() { return {*this}; }
|
|
||||||
|
|
||||||
YGDisplay display() const { return display_; }
|
|
||||||
BITFIELD_REF(display) display() { return {*this}; }
|
|
||||||
|
|
||||||
YGFloatOptional flex() const { return flex_; }
|
|
||||||
Ref<YGFloatOptional, &YGStyle::flex_, flexBit> flex() { return {*this}; }
|
|
||||||
|
|
||||||
YGFloatOptional flexGrow() const { return flexGrow_; }
|
|
||||||
Ref<YGFloatOptional, &YGStyle::flexGrow_, flexGrowBit> flexGrow() {
|
|
||||||
return {*this};
|
|
||||||
}
|
|
||||||
|
|
||||||
YGFloatOptional flexShrink() const { return flexShrink_; }
|
|
||||||
Ref<YGFloatOptional, &YGStyle::flexShrink_, flexShrinkBit> flexShrink() {
|
|
||||||
return {*this};
|
|
||||||
}
|
|
||||||
|
|
||||||
CompactValue flexBasis() const { return flexBasis_; }
|
|
||||||
Ref<CompactValue, &YGStyle::flexBasis_, flexBasisBit> flexBasis() {
|
|
||||||
return {*this};
|
|
||||||
}
|
|
||||||
|
|
||||||
const Edges& margin() const { return margin_; }
|
|
||||||
IdxRef<YGEdge, &YGStyle::margin_, marginBit> margin() { return {*this}; }
|
|
||||||
|
|
||||||
const Edges& position() const { return position_; }
|
|
||||||
IdxRef<YGEdge, &YGStyle::position_, positionBit> position() {
|
|
||||||
return {*this};
|
|
||||||
}
|
|
||||||
|
|
||||||
const Edges& padding() const { return padding_; }
|
|
||||||
IdxRef<YGEdge, &YGStyle::padding_, paddingBit> padding() { return {*this}; }
|
|
||||||
|
|
||||||
const Edges& border() const { return border_; }
|
|
||||||
IdxRef<YGEdge, &YGStyle::border_, borderBit> border() { return {*this}; }
|
|
||||||
|
|
||||||
const Dimensions& dimensions() const { return dimensions_; }
|
|
||||||
IdxRef<YGDimension, &YGStyle::dimensions_, dimensionsBit> dimensions() {
|
|
||||||
return {*this};
|
|
||||||
}
|
|
||||||
|
|
||||||
const Dimensions& minDimensions() const { return minDimensions_; }
|
|
||||||
IdxRef<YGDimension, &YGStyle::minDimensions_, minDimensionsBit>
|
|
||||||
minDimensions() {
|
|
||||||
return {*this};
|
|
||||||
}
|
|
||||||
|
|
||||||
const Dimensions& maxDimensions() const { return maxDimensions_; }
|
|
||||||
IdxRef<YGDimension, &YGStyle::maxDimensions_, maxDimensionsBit>
|
|
||||||
maxDimensions() {
|
|
||||||
return {*this};
|
|
||||||
}
|
|
||||||
|
|
||||||
// Yoga specific properties, not compatible with flexbox specification
|
|
||||||
YGFloatOptional aspectRatio() const { return aspectRatio_; }
|
|
||||||
Ref<YGFloatOptional, &YGStyle::aspectRatio_, aspectRatioBit> aspectRatio() {
|
|
||||||
return {*this};
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
bool operator==(const YGStyle& lhs, const YGStyle& rhs);
|
|
||||||
inline bool operator!=(const YGStyle& lhs, const YGStyle& rhs) {
|
|
||||||
return !(lhs == rhs);
|
|
||||||
}
|
|
||||||
|
|
||||||
#undef BITFIELD_ENUM_SIZED
|
|
||||||
#undef BITFIELD_ACCESSORS
|
|
||||||
#undef BITFIELD_REF
|
|
||||||
11
deps/yoga/YGValue.cpp
vendored
11
deps/yoga/YGValue.cpp
vendored
@ -1,11 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) Facebook, Inc. and its affiliates.
|
|
||||||
*
|
|
||||||
* This source code is licensed under the MIT license found in the LICENSE
|
|
||||||
* file in the root directory of this source tree.
|
|
||||||
*/
|
|
||||||
#include "YGValue.h"
|
|
||||||
|
|
||||||
const YGValue YGValueZero = {0, YGUnitPoint};
|
|
||||||
const YGValue YGValueUndefined = {YGUndefined, YGUnitUndefined};
|
|
||||||
const YGValue YGValueAuto = {YGUndefined, YGUnitAuto};
|
|
||||||
83
deps/yoga/YGValue.h
vendored
83
deps/yoga/YGValue.h
vendored
@ -1,83 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) Facebook, Inc. and its affiliates.
|
|
||||||
*
|
|
||||||
* This source code is licensed under the MIT license found in the LICENSE
|
|
||||||
* file in the root directory of this source tree.
|
|
||||||
*/
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include <math.h>
|
|
||||||
#include "YGEnums.h"
|
|
||||||
#include "YGMacros.h"
|
|
||||||
|
|
||||||
YG_EXTERN_C_BEGIN
|
|
||||||
|
|
||||||
// Not defined in MSVC++
|
|
||||||
#ifndef NAN
|
|
||||||
static const uint32_t __nan = 0x7fc00000;
|
|
||||||
#define NAN (*(const float*) __nan)
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#define YGUndefined NAN
|
|
||||||
|
|
||||||
typedef struct YGValue {
|
|
||||||
float value;
|
|
||||||
YGUnit unit;
|
|
||||||
} YGValue;
|
|
||||||
|
|
||||||
extern const YGValue YGValueAuto;
|
|
||||||
extern const YGValue YGValueUndefined;
|
|
||||||
extern const YGValue YGValueZero;
|
|
||||||
|
|
||||||
YG_EXTERN_C_END
|
|
||||||
|
|
||||||
#ifdef __cplusplus
|
|
||||||
|
|
||||||
inline bool operator==(const YGValue& lhs, const YGValue& rhs) {
|
|
||||||
if (lhs.unit != rhs.unit) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
switch (lhs.unit) {
|
|
||||||
case YGUnitUndefined:
|
|
||||||
case YGUnitAuto:
|
|
||||||
return true;
|
|
||||||
case YGUnitPoint:
|
|
||||||
case YGUnitPercent:
|
|
||||||
return lhs.value == rhs.value;
|
|
||||||
}
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
inline bool operator!=(const YGValue& lhs, const YGValue& rhs) {
|
|
||||||
return !(lhs == rhs);
|
|
||||||
}
|
|
||||||
|
|
||||||
inline YGValue operator-(const YGValue& value) {
|
|
||||||
return {-value.value, value.unit};
|
|
||||||
}
|
|
||||||
|
|
||||||
namespace facebook {
|
|
||||||
namespace yoga {
|
|
||||||
namespace literals {
|
|
||||||
|
|
||||||
inline YGValue operator"" _pt(long double value) {
|
|
||||||
return YGValue{static_cast<float>(value), YGUnitPoint};
|
|
||||||
}
|
|
||||||
inline YGValue operator"" _pt(unsigned long long value) {
|
|
||||||
return operator"" _pt(static_cast<long double>(value));
|
|
||||||
}
|
|
||||||
|
|
||||||
inline YGValue operator"" _percent(long double value) {
|
|
||||||
return YGValue{static_cast<float>(value), YGUnitPercent};
|
|
||||||
}
|
|
||||||
inline YGValue operator"" _percent(unsigned long long value) {
|
|
||||||
return operator"" _percent(static_cast<long double>(value));
|
|
||||||
}
|
|
||||||
|
|
||||||
} // namespace literals
|
|
||||||
} // namespace yoga
|
|
||||||
} // namespace facebook
|
|
||||||
|
|
||||||
#endif
|
|
||||||
150
deps/yoga/Yoga-internal.h
vendored
150
deps/yoga/Yoga-internal.h
vendored
@ -1,150 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) Facebook, Inc. and its affiliates.
|
|
||||||
*
|
|
||||||
* This source code is licensed under the MIT license found in the LICENSE
|
|
||||||
* file in the root directory of this source tree.
|
|
||||||
*/
|
|
||||||
#pragma once
|
|
||||||
#include <algorithm>
|
|
||||||
#include <array>
|
|
||||||
#include <cmath>
|
|
||||||
#include <vector>
|
|
||||||
#include "CompactValue.h"
|
|
||||||
#include "Yoga.h"
|
|
||||||
|
|
||||||
using YGVector = std::vector<YGNodeRef>;
|
|
||||||
|
|
||||||
YG_EXTERN_C_BEGIN
|
|
||||||
|
|
||||||
void YGNodeCalculateLayoutWithContext(
|
|
||||||
YGNodeRef node,
|
|
||||||
float availableWidth,
|
|
||||||
float availableHeight,
|
|
||||||
YGDirection ownerDirection,
|
|
||||||
void* layoutContext);
|
|
||||||
|
|
||||||
YG_EXTERN_C_END
|
|
||||||
|
|
||||||
namespace facebook {
|
|
||||||
namespace yoga {
|
|
||||||
|
|
||||||
inline bool isUndefined(float value) {
|
|
||||||
return std::isnan(value);
|
|
||||||
}
|
|
||||||
|
|
||||||
} // namespace yoga
|
|
||||||
} // namespace facebook
|
|
||||||
|
|
||||||
using namespace facebook;
|
|
||||||
|
|
||||||
extern const std::array<YGEdge, 4> trailing;
|
|
||||||
extern const std::array<YGEdge, 4> leading;
|
|
||||||
extern const YGValue YGValueUndefined;
|
|
||||||
extern const YGValue YGValueAuto;
|
|
||||||
extern const YGValue YGValueZero;
|
|
||||||
|
|
||||||
struct YGCachedMeasurement {
|
|
||||||
float availableWidth;
|
|
||||||
float availableHeight;
|
|
||||||
YGMeasureMode widthMeasureMode;
|
|
||||||
YGMeasureMode heightMeasureMode;
|
|
||||||
|
|
||||||
float computedWidth;
|
|
||||||
float computedHeight;
|
|
||||||
|
|
||||||
YGCachedMeasurement()
|
|
||||||
: availableWidth(0),
|
|
||||||
availableHeight(0),
|
|
||||||
widthMeasureMode((YGMeasureMode) -1),
|
|
||||||
heightMeasureMode((YGMeasureMode) -1),
|
|
||||||
computedWidth(-1),
|
|
||||||
computedHeight(-1) {}
|
|
||||||
|
|
||||||
bool operator==(YGCachedMeasurement measurement) const {
|
|
||||||
bool isEqual = widthMeasureMode == measurement.widthMeasureMode &&
|
|
||||||
heightMeasureMode == measurement.heightMeasureMode;
|
|
||||||
|
|
||||||
if (!yoga::isUndefined(availableWidth) ||
|
|
||||||
!yoga::isUndefined(measurement.availableWidth)) {
|
|
||||||
isEqual = isEqual && availableWidth == measurement.availableWidth;
|
|
||||||
}
|
|
||||||
if (!yoga::isUndefined(availableHeight) ||
|
|
||||||
!yoga::isUndefined(measurement.availableHeight)) {
|
|
||||||
isEqual = isEqual && availableHeight == measurement.availableHeight;
|
|
||||||
}
|
|
||||||
if (!yoga::isUndefined(computedWidth) ||
|
|
||||||
!yoga::isUndefined(measurement.computedWidth)) {
|
|
||||||
isEqual = isEqual && computedWidth == measurement.computedWidth;
|
|
||||||
}
|
|
||||||
if (!yoga::isUndefined(computedHeight) ||
|
|
||||||
!yoga::isUndefined(measurement.computedHeight)) {
|
|
||||||
isEqual = isEqual && computedHeight == measurement.computedHeight;
|
|
||||||
}
|
|
||||||
|
|
||||||
return isEqual;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
// This value was chosen based on empirical data:
|
|
||||||
// 98% of analyzed layouts require less than 8 entries.
|
|
||||||
#define YG_MAX_CACHED_RESULT_COUNT 8
|
|
||||||
|
|
||||||
namespace facebook {
|
|
||||||
namespace yoga {
|
|
||||||
namespace detail {
|
|
||||||
|
|
||||||
template <size_t Size>
|
|
||||||
class Values {
|
|
||||||
private:
|
|
||||||
std::array<CompactValue, Size> values_;
|
|
||||||
|
|
||||||
public:
|
|
||||||
Values() = default;
|
|
||||||
explicit Values(const YGValue& defaultValue) noexcept {
|
|
||||||
values_.fill(defaultValue);
|
|
||||||
}
|
|
||||||
|
|
||||||
const CompactValue& operator[](size_t i) const noexcept { return values_[i]; }
|
|
||||||
CompactValue& operator[](size_t i) noexcept { return values_[i]; }
|
|
||||||
|
|
||||||
template <size_t I>
|
|
||||||
YGValue get() const noexcept {
|
|
||||||
return std::get<I>(values_);
|
|
||||||
}
|
|
||||||
|
|
||||||
template <size_t I>
|
|
||||||
void set(YGValue& value) noexcept {
|
|
||||||
std::get<I>(values_) = value;
|
|
||||||
}
|
|
||||||
|
|
||||||
template <size_t I>
|
|
||||||
void set(YGValue&& value) noexcept {
|
|
||||||
set<I>(value);
|
|
||||||
}
|
|
||||||
|
|
||||||
bool operator==(const Values& other) const noexcept {
|
|
||||||
for (size_t i = 0; i < Size; ++i) {
|
|
||||||
if (values_[i] != other.values_[i]) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
Values& operator=(const Values& other) = default;
|
|
||||||
};
|
|
||||||
|
|
||||||
} // namespace detail
|
|
||||||
} // namespace yoga
|
|
||||||
} // namespace facebook
|
|
||||||
|
|
||||||
static const float kDefaultFlexGrow = 0.0f;
|
|
||||||
static const float kDefaultFlexShrink = 0.0f;
|
|
||||||
static const float kWebDefaultFlexShrink = 1.0f;
|
|
||||||
|
|
||||||
extern bool YGFloatsEqual(const float a, const float b);
|
|
||||||
extern facebook::yoga::detail::CompactValue YGComputedEdgeValue(
|
|
||||||
const facebook::yoga::detail::Values<
|
|
||||||
facebook::yoga::enums::count<YGEdge>()>& edges,
|
|
||||||
YGEdge edge,
|
|
||||||
facebook::yoga::detail::CompactValue defaultValue);
|
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue
Block a user