Compare commits
819 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
fc07919a75 | ||
|
|
d7661edf67 | ||
|
|
1c91693294 | ||
|
|
3fc3cbc6d5 | ||
|
|
11b5895e69 | ||
|
|
b4c614edbf | ||
|
|
deddaed04a | ||
|
|
a155e503f4 | ||
|
|
c0663bc19f | ||
|
|
7ffa51b83d | ||
|
|
6d397b9988 | ||
|
|
571a7fa26b | ||
|
|
b3b5a38c3a | ||
|
|
a4b115f89b | ||
|
|
3d672bb145 | ||
|
|
15b656b753 | ||
|
|
f5785db163 | ||
|
|
1ff1bf3292 | ||
|
|
b062febea0 | ||
|
|
35301dc79e | ||
|
|
30c336726b | ||
|
|
3821680817 | ||
|
|
74242ea24f | ||
|
|
73c7024e11 | ||
|
|
bbdec92dc6 | ||
|
|
b4c289101c | ||
|
|
722d0797f6 | ||
|
|
deb27d5b55 | ||
|
|
9cc4ffdf6e | ||
|
|
c76d63d1d9 | ||
|
|
21bb4f33ad | ||
|
|
2897b881d6 | ||
|
|
e9b89d0929 | ||
|
|
e801254b2e | ||
|
|
6f49918ee9 | ||
|
|
0ae7c18f1f | ||
|
|
c9c3fb396a | ||
|
|
91db4f1934 | ||
|
|
9bbed6e95c | ||
|
|
748bc27b25 | ||
|
|
f42708e8bc | ||
|
|
160e4570a2 | ||
|
|
6272965143 | ||
|
|
1e9613bf7f | ||
|
|
a061dac298 | ||
|
|
914dee8571 | ||
|
|
47e2905edf | ||
|
|
ee6675aee0 | ||
|
|
62e0d9fe64 | ||
|
|
a174c142c1 | ||
|
|
95afc5fdec | ||
|
|
04d69f66c0 | ||
|
|
0347141edd | ||
|
|
1d6baae6e0 | ||
|
|
fb0f48f08a | ||
|
|
76e5e03d31 | ||
|
|
4cab743634 | ||
|
|
7c10ec97b7 | ||
|
|
8718a16889 | ||
|
|
75a0b924c3 | ||
|
|
83a90e0c05 | ||
|
|
e5eadd1315 | ||
|
|
f0142d90d4 | ||
|
|
cabd6e6e9d | ||
|
|
4804a05736 | ||
|
|
c258e5a3af | ||
|
|
e8492940a5 | ||
|
|
4bccb1ab47 | ||
|
|
b6d219e232 | ||
|
|
23ee17594d | ||
|
|
d9d39d8e25 | ||
|
|
224d5d46c1 | ||
|
|
09e0059930 | ||
|
|
0ddff2b087 | ||
|
|
2e6a29eacc | ||
|
|
ad2fb82aa9 | ||
|
|
a3f91c11e8 | ||
|
|
a50c978ce3 | ||
|
|
27d6f881cd | ||
|
|
944cd020af | ||
|
|
bbe5d64b99 | ||
|
|
f7b36ac4c7 | ||
|
|
1d555ca17e | ||
|
|
f91b6c3468 | ||
|
|
abe6eeb350 | ||
|
|
64f90a7912 | ||
|
|
5733966843 | ||
|
|
eb1344fcec | ||
|
|
c5fb29f00e | ||
|
|
63135b9c54 | ||
|
|
f7c666584e | ||
|
|
6834324de2 | ||
|
|
3a0d59e66f | ||
|
|
ffd2e2188a | ||
|
|
0e8d5bdc5d | ||
|
|
14806f6614 | ||
|
|
00ece83b9d | ||
|
|
8197ae2a2d | ||
|
|
5fe658bb16 | ||
|
|
617179f0c6 | ||
|
|
95ac85f642 | ||
|
|
ca8877ad47 | ||
|
|
6d8f31048c | ||
|
|
ac859eb576 | ||
|
|
2dfa171b6a | ||
|
|
6f72e3e2ea | ||
|
|
c2b73ae963 | ||
|
|
6c50077409 | ||
|
|
06746449a1 | ||
|
|
da7b8edf51 | ||
|
|
6e29b41f23 | ||
|
|
912bb069af | ||
|
|
6b2d7a67d8 | ||
|
|
dbb8ec0290 | ||
|
|
60b32760f2 | ||
|
|
73a40bcb49 | ||
|
|
0e258a5a32 | ||
|
|
7ca65c81d8 | ||
|
|
2ad1a60e59 | ||
|
|
cf17ff4478 | ||
|
|
fffc3aac68 | ||
|
|
295ac3c458 | ||
|
|
b6693a71f9 | ||
|
|
5b21118a8c | ||
|
|
0235b19801 | ||
|
|
7426399aa2 | ||
|
|
6861b0d668 | ||
|
|
c30fb0d38c | ||
|
|
e45521c6c0 | ||
|
|
d11fe8d4fc | ||
|
|
8e83e63e3d | ||
|
|
c8fd0ac4b3 | ||
|
|
d78419eb33 | ||
|
|
e44a3d013d | ||
|
|
aeb0d05017 | ||
|
|
62702e4b3d | ||
|
|
5146cdfa2f | ||
|
|
246e7018c3 | ||
|
|
24286dbe9d | ||
|
|
2f442dfbe1 | ||
|
|
b2fb01ee9c | ||
|
|
45e0a9a4ef | ||
|
|
675b7b4bf4 | ||
|
|
be7a35443e | ||
|
|
9918615fcd | ||
|
|
4b72ef77c1 | ||
|
|
e427c61fbb | ||
|
|
c78e8937d5 | ||
|
|
5d63f7a93d | ||
|
|
46551ada7f | ||
|
|
83f17a37b1 | ||
|
|
9e23e0a623 | ||
|
|
4a53d4f043 | ||
|
|
0fd61945c7 | ||
|
|
e3624eed30 | ||
|
|
1923c8be0c | ||
|
|
a3e37fbfe2 | ||
|
|
318c3bb422 | ||
|
|
db96e24028 | ||
|
|
cc2cce66df | ||
|
|
7afeabd288 | ||
|
|
eb43a812d6 | ||
|
|
12cbcdb6f4 | ||
|
|
a2e35e30dc | ||
|
|
b6ff7e6b47 | ||
|
|
6dba40c6bb | ||
|
|
896da46422 | ||
|
|
7c07f5eb2a | ||
|
|
298cff37de | ||
|
|
8add802fe9 | ||
|
|
6d080a0d59 | ||
|
|
f0ae1051ee | ||
|
|
1ad0ffeaa6 | ||
|
|
45b44d012d | ||
|
|
cb6cbb9ee5 | ||
|
|
013a0d9cb0 | ||
|
|
8abb3ea225 | ||
|
|
74a5233b5d | ||
|
|
cd695a4522 | ||
|
|
eec4b2cc0e | ||
|
|
616875d0d2 | ||
|
|
eb749bd76f | ||
|
|
7d1e404efd | ||
|
|
4a01a578d1 | ||
|
|
393a2e0ea0 | ||
|
|
8e21decb8d | ||
|
|
22bfbab832 | ||
|
|
28d0cc8795 | ||
|
|
51f2383a07 | ||
|
|
a6426c6eba | ||
|
|
0631da6c8e | ||
|
|
3960239cfe | ||
|
|
b16e48af6f | ||
|
|
4d3950565a | ||
|
|
8d24bc50c9 | ||
|
|
458efe9168 | ||
|
|
ae940a8681 | ||
|
|
d76b223b7c | ||
|
|
7243d5f7cb | ||
|
|
387f74f66a | ||
|
|
61ffb7d97a | ||
|
|
a553693f34 | ||
|
|
cdcfd64ec4 | ||
|
|
7f4302ba20 | ||
|
|
0d94f2d376 | ||
|
|
c1eb20a20b | ||
|
|
634db67685 | ||
|
|
86648258a3 | ||
|
|
287f0a3739 | ||
|
|
8f42df209a | ||
|
|
390fd64a74 | ||
|
|
48ee471def | ||
|
|
e862afcb78 | ||
|
|
872da05ff6 | ||
|
|
d09e2daf00 | ||
|
|
e2d5b44b0a | ||
|
|
52d42ef2a8 | ||
|
|
b6abc34461 | ||
|
|
ae6a50626d | ||
|
|
8f9dbfee2c | ||
|
|
c71ccda967 | ||
|
|
0f284f2e1d | ||
|
|
45ac80dd8c | ||
|
|
09cdba4b3d | ||
|
|
d94ee8863c | ||
|
|
cc8ced6430 | ||
|
|
cdb144e6a6 | ||
|
|
58fb8643c6 | ||
|
|
b81cfa8acb | ||
|
|
938ee20f1f | ||
|
|
d02dc54c1b | ||
|
|
8680a54ae4 | ||
|
|
30001be0ee | ||
|
|
4614cb5ec1 | ||
|
|
e390f3a399 | ||
|
|
e22d463d11 | ||
|
|
5877aa822c | ||
|
|
a2915913bb | ||
|
|
a8b40747b2 | ||
|
|
b63030d302 | ||
|
|
4457e416db | ||
|
|
c7c1a8ede1 | ||
|
|
fd1e9d7fb0 | ||
|
|
59a6d2317b | ||
|
|
531c171542 | ||
|
|
0c743452b0 | ||
|
|
4893d3da1f | ||
|
|
5cdc24bfb0 | ||
|
|
523cdca449 | ||
|
|
09537d73da | ||
|
|
99c0c0b3b1 | ||
|
|
45bc353341 | ||
|
|
b2fc41a911 | ||
|
|
ebefe8b6d2 | ||
|
|
9e3508134b | ||
|
|
3166ca2127 | ||
|
|
204508478f | ||
|
|
51ce674c97 | ||
|
|
6fcde9fe5f | ||
|
|
c688e3431d | ||
|
|
230376c7f3 | ||
|
|
91d0a2cd0c | ||
|
|
1ffd010e4a | ||
|
|
ae366e141f | ||
|
|
eb869c6e97 | ||
|
|
b559f7cd16 | ||
|
|
375d32f08e | ||
|
|
89d6b7cec0 | ||
|
|
fdf96e8342 | ||
|
|
c5d73d7b09 | ||
|
|
34b4cc2d9e | ||
|
|
40ed17e609 | ||
|
|
673d76af45 | ||
|
|
c13fb6f9d5 | ||
|
|
424b1e7d1f | ||
|
|
e80c9c4101 | ||
|
|
f75c3633e9 | ||
|
|
d535cd8276 | ||
|
|
c13f6ab2af | ||
|
|
d2a30bfb78 | ||
|
|
e5b17092b4 | ||
|
|
466cb4c78b | ||
|
|
9722e1ddc4 | ||
|
|
e3a9b0b943 | ||
|
|
d668a8aee2 | ||
|
|
75dc0aafcf | ||
|
|
41b94233c6 | ||
|
|
52cff01b9c | ||
|
|
372fc6603d | ||
|
|
9221797c9d | ||
|
|
7501131558 | ||
|
|
bba7be99a3 | ||
|
|
abb95534d0 | ||
|
|
0b7b4c12e4 | ||
|
|
4056f00169 | ||
|
|
10303cb9c0 | ||
|
|
e3587d369e | ||
|
|
2a048502cc | ||
|
|
cac01fbde9 | ||
|
|
2d2ce191ec | ||
|
|
6380cb8458 | ||
|
|
48e0e6af71 | ||
|
|
b756bccc7a | ||
|
|
ae8eed7a67 | ||
|
|
ae4882bec3 | ||
|
|
1379741859 | ||
|
|
b45c7ace78 | ||
|
|
3843d9f55b | ||
|
|
f3422cb2fe | ||
|
|
73692797dc | ||
|
|
31dd910289 | ||
|
|
da51580299 | ||
|
|
9db148b1ec | ||
|
|
091b1b8209 | ||
|
|
900a4071ed | ||
|
|
df4b2f7938 | ||
|
|
803f44d9bc | ||
|
|
4b67aee020 | ||
|
|
71dc47d6c9 | ||
|
|
9cb305fb0d | ||
|
|
1672130486 | ||
|
|
8e135e3e79 | ||
|
|
aa1a4f6ea5 | ||
|
|
ba34cf5258 | ||
|
|
647089d2a8 | ||
|
|
5211508eb4 | ||
|
|
e6f05ae465 | ||
|
|
a9193f9b76 | ||
|
|
e373a17cd3 | ||
|
|
ebab9b7e4a | ||
|
|
6de0399807 | ||
|
|
5cc7bb80f6 | ||
|
|
6e0bd9b3f8 | ||
|
|
d1943f72d3 | ||
|
|
81709873bd | ||
|
|
f6bb7cd8ed | ||
|
|
d1c19e431c | ||
|
|
9ab2dde8ab | ||
|
|
71559bb125 | ||
|
|
ae4c95262c | ||
|
|
dbbf07c9c1 | ||
|
|
ab8cd619d5 | ||
|
|
c30ad2819d | ||
|
|
311e91797a | ||
|
|
52be4df355 | ||
|
|
0364e81050 | ||
|
|
2d49b71bc9 | ||
|
|
a18a4bdf31 | ||
|
|
d3acbe8288 | ||
|
|
22afcbcbb6 | ||
|
|
495c6bc21c | ||
|
|
cfd1fe59f3 | ||
|
|
c46cf5bc84 | ||
|
|
a8742557bd | ||
|
|
3bea58cf56 | ||
|
|
5aaa5231b8 | ||
|
|
82d10dd7cb | ||
|
|
841065fb91 | ||
|
|
c7146ef138 | ||
|
|
08f32d1de6 | ||
|
|
4c3f86aa4d | ||
|
|
445cf22333 | ||
|
|
4919de647a | ||
|
|
77fae99528 | ||
|
|
10bd4b40b9 | ||
|
|
9d8e6bb253 | ||
|
|
49c71ecfad | ||
|
|
d18834b415 | ||
|
|
e52cda193e | ||
|
|
0d5edd35ea | ||
|
|
f3f51c3d9d | ||
|
|
1431916183 | ||
|
|
ae48008803 | ||
|
|
5664813931 | ||
|
|
3948af80b8 | ||
|
|
343d6f9aff | ||
|
|
51b379502f | ||
|
|
82142751de | ||
|
|
4e5755f218 | ||
|
|
2f4f29517e | ||
|
|
e8a073651f | ||
|
|
d23da7a612 | ||
|
|
1ae4938da3 | ||
|
|
b5e27d4d69 | ||
|
|
d74fc8d1ce | ||
|
|
c5d9a41967 | ||
|
|
7e3042c4f4 | ||
|
|
1291dadbd6 | ||
|
|
f645099a39 | ||
|
|
8f32038891 | ||
|
|
a6fb1dcdf9 | ||
|
|
f01b469f3f | ||
|
|
47884919c8 | ||
|
|
653a35496d | ||
|
|
9b14df6b27 | ||
|
|
09813f3c5a | ||
|
|
7d76b7e4c2 | ||
|
|
47b5dea95c | ||
|
|
b0df63f1e8 | ||
|
|
3c4209b676 | ||
|
|
d51b9a8e0e | ||
|
|
3b56125bd2 | ||
|
|
6e69e39007 | ||
|
|
97208cb329 | ||
|
|
414a4a97fb | ||
|
|
2a809f96c4 | ||
|
|
17799b03f3 | ||
|
|
efc55fc648 | ||
|
|
22bd41211a | ||
|
|
4cb0171bd0 | ||
|
|
ce6c5af72c | ||
|
|
171575256c | ||
|
|
d3664dcf78 | ||
|
|
0788981783 | ||
|
|
aeee7c02d5 | ||
|
|
6e49a50461 | ||
|
|
fbc99827ab | ||
|
|
3b134320c4 | ||
|
|
c315e5016d | ||
|
|
7aebd6ed57 | ||
|
|
1a8ca06495 | ||
|
|
a27ae7e4a6 | ||
|
|
dd0ab897aa | ||
|
|
00ad92fb6d | ||
|
|
f84128ecbd | ||
|
|
ba89e0f4e3 | ||
|
|
832d36a4d4 | ||
|
|
0b437b3bfb | ||
|
|
7b50ec4630 | ||
|
|
4ddb13abac | ||
|
|
d2ac081177 | ||
|
|
9692fbf15b | ||
|
|
be966488e8 | ||
|
|
0ce613264f | ||
|
|
34634d776e | ||
|
|
673ded3819 | ||
|
|
b9a94ad3ae | ||
|
|
3f80b330cc | ||
|
|
01632d538c | ||
|
|
b0a9b1cd09 | ||
|
|
1f772081fd | ||
|
|
4ae54dbaad | ||
|
|
e47f4ff731 | ||
|
|
465369d79e | ||
|
|
15ddf6ff20 | ||
|
|
16a753bd95 | ||
|
|
c15103636c | ||
|
|
8a5f82ee7d | ||
|
|
5ec33ec821 | ||
|
|
ab7d383cf1 | ||
|
|
184e9a5c93 | ||
|
|
c4f5363cde | ||
|
|
002882cebf | ||
|
|
1cb3ec0c7b | ||
|
|
6bf325c6f6 | ||
|
|
09c7ff9e8b | ||
|
|
c2a94b61bf | ||
|
|
1db16232de | ||
|
|
3da681a6b1 | ||
|
|
a79b3e7852 | ||
|
|
b1099e6974 | ||
|
|
4f6e06131c | ||
|
|
19f69e9e6c | ||
|
|
01481da773 | ||
|
|
3e8f7e1cf1 | ||
|
|
5da69646f2 | ||
|
|
3cac01583b | ||
|
|
d16a26605e | ||
|
|
a4f692c788 | ||
|
|
9f01206c57 | ||
|
|
d34fc551ed | ||
|
|
7aa5f0d258 | ||
|
|
276a34bb66 | ||
|
|
0d820eda12 | ||
|
|
1991c1b677 | ||
|
|
459404e3f0 | ||
|
|
badc623a3c | ||
|
|
8e39f92cb7 | ||
|
|
3ff4885973 | ||
|
|
f9d45f7657 | ||
|
|
789ff9df5c | ||
|
|
a2064ed16b | ||
|
|
b3d06c0868 | ||
|
|
db1a6b3e38 | ||
|
|
8390237cc4 | ||
|
|
9967eae7bb | ||
|
|
b3be7d1c6f | ||
|
|
33ccb5dbb2 | ||
|
|
5b90c0d695 | ||
|
|
472a660239 | ||
|
|
ab67536d9a | ||
|
|
ee85fb3aec | ||
|
|
214b6f4358 | ||
|
|
af0d092054 | ||
|
|
b07903c3e9 | ||
|
|
ffa4c6bf09 | ||
|
|
b4125fa56c | ||
|
|
2c72302087 | ||
|
|
0fa52bc64f | ||
|
|
f55a80b15a | ||
|
|
0735483321 | ||
|
|
53fc2c7c21 | ||
|
|
f22133c3c5 | ||
|
|
2d5a6d6583 | ||
|
|
dc4adf2836 | ||
|
|
dd6e254e4f | ||
|
|
4c028c1659 | ||
|
|
d332a6777a | ||
|
|
378251f229 | ||
|
|
b6b9b903ed | ||
|
|
143d68cfd5 | ||
|
|
a31eac1426 | ||
|
|
797196f7fc | ||
|
|
c9d0bc81dd | ||
|
|
b5448ff607 | ||
|
|
5ebd363d5d | ||
|
|
1d439e673e | ||
|
|
0b7b7656b2 | ||
|
|
eb270df835 | ||
|
|
ff73dd2183 | ||
|
|
e043a03eb6 | ||
|
|
3cb4e8e373 | ||
|
|
7e6de528b4 | ||
|
|
df901c30ef | ||
|
|
13856b33ec | ||
|
|
a3a1c6f4c8 | ||
|
|
638998a861 | ||
|
|
6e2ec89a05 | ||
|
|
af67de8aa6 | ||
|
|
425dac478e | ||
|
|
b15c4ecd10 | ||
|
|
d7f88cf3a4 | ||
|
|
5b7fbcd9b8 | ||
|
|
7af64b0782 | ||
|
|
b84c70e811 | ||
|
|
f5b245c72d | ||
|
|
4307183817 | ||
|
|
aeb32783d6 | ||
|
|
3927b3bf27 | ||
|
|
da9d2f9417 | ||
|
|
1cec48e8f8 | ||
|
|
f1105393da | ||
|
|
dc7047e3c2 | ||
|
|
08b2945623 | ||
|
|
cbcc223150 | ||
|
|
9830f21e4a | ||
|
|
5f49567bf7 | ||
|
|
6154ae7342 | ||
|
|
978f3a3682 | ||
|
|
1f0961d574 | ||
|
|
a101252701 | ||
|
|
c96c29b1e3 | ||
|
|
3b0fc180ff | ||
|
|
9b8bfdf33c | ||
|
|
4328831fcd | ||
|
|
e5b3df41e9 | ||
|
|
cf5259e218 | ||
|
|
f24b6a520c | ||
|
|
4140163ab2 | ||
|
|
7afde0e93f | ||
|
|
d27a571882 | ||
|
|
1c70e3be25 | ||
|
|
1819f64467 | ||
|
|
9852e588c1 | ||
|
|
71a1ea481b | ||
|
|
9e32f0d778 | ||
|
|
4478174dc2 | ||
|
|
07553476d4 | ||
|
|
1773283456 | ||
|
|
221ab51d90 | ||
|
|
43e0dd922b | ||
|
|
b3262652c3 | ||
|
|
0cfa2b8c20 | ||
|
|
c4600b2187 | ||
|
|
89b15e3b57 | ||
|
|
07d8c3e770 | ||
|
|
c5ae64070e | ||
|
|
2cb7c52147 | ||
|
|
c045b16213 | ||
|
|
b29387d409 | ||
|
|
614a09db1d | ||
|
|
bcc5450aef | ||
|
|
b0420a0566 | ||
|
|
08d627bc9f | ||
|
|
5b5f048faf | ||
|
|
212ebae043 | ||
|
|
a0c99df6b2 | ||
|
|
b0fabd7897 | ||
|
|
938811f24f | ||
|
|
374d6e4d0f | ||
|
|
5f54cdee0f | ||
|
|
c2bd452391 | ||
|
|
d2bfc73b91 | ||
|
|
42c62206c8 | ||
|
|
4208b512f8 | ||
|
|
3a3d5cd487 | ||
|
|
e143a6d126 | ||
|
|
a99a19aa60 | ||
|
|
aba7d2b6a4 | ||
|
|
2fc6f835a8 | ||
|
|
27ce0e6d60 | ||
|
|
05c70f2adb | ||
|
|
904097b7b1 | ||
|
|
4e003c12a6 | ||
|
|
24f2cfb29f | ||
|
|
3f2b683fca | ||
|
|
83cee26d7d | ||
|
|
c7df0c9b28 | ||
|
|
26a8bfaac4 | ||
|
|
fc8a3be7d9 | ||
|
|
b195c4033d | ||
|
|
ab2732c55d | ||
|
|
7fd5c058be | ||
|
|
b22e8b4702 | ||
|
|
e2c2af3447 | ||
|
|
4cd738ecb5 | ||
|
|
e838470c26 | ||
|
|
68e0bc40e1 | ||
|
|
9fc8bcdf62 | ||
|
|
227b14a0b6 | ||
|
|
abcd184d5d | ||
|
|
3fd9f4b0df | ||
|
|
12ff3e963b | ||
|
|
558e392234 | ||
|
|
e747c2e263 | ||
|
|
4d78b30e8c | ||
|
|
9a1520d5e3 | ||
|
|
04f3543424 | ||
|
|
eae287be37 | ||
|
|
a0e698b058 | ||
|
|
cd2932adea | ||
|
|
286b270592 | ||
|
|
d1ebef2cc5 | ||
|
|
deb567fde7 | ||
|
|
26586fb9ef | ||
|
|
5f71a558b9 | ||
|
|
08882639e0 | ||
|
|
837ae2932f | ||
|
|
b51cc21140 | ||
|
|
f1115ba706 | ||
|
|
dc36aee7ff | ||
|
|
740f9581e6 | ||
|
|
de0e3a7bc9 | ||
|
|
95e894fb4d | ||
|
|
faacb0d69f | ||
|
|
28b139e1b8 | ||
|
|
39b1e0676b | ||
|
|
c05fc5bd36 | ||
|
|
bbae9157fb | ||
|
|
fc5c63a869 | ||
|
|
e2e068f8e5 | ||
|
|
43a068e30e | ||
|
|
2beb4b8c6f | ||
|
|
ad4230f7fa | ||
|
|
00369a2cfe | ||
|
|
248e487dd5 | ||
|
|
3a3dc02a66 | ||
|
|
f49c47c20d | ||
|
|
417ef1a7a4 | ||
|
|
8e8889483f | ||
|
|
b5818a9b62 | ||
|
|
e709f676dc | ||
|
|
bb611bf655 | ||
|
|
24e8ca67c5 | ||
|
|
0695941a77 | ||
|
|
7c71efd1b3 | ||
|
|
5099aff5c3 | ||
|
|
d3463250a9 | ||
|
|
3b58c02db0 | ||
|
|
64f0313d3c | ||
|
|
d3958e757b | ||
|
|
651020388d | ||
|
|
72ede666d4 | ||
|
|
a68c249d4e | ||
|
|
56caab4461 | ||
|
|
13b60351a6 | ||
|
|
ad49d38e46 | ||
|
|
082c9097e4 | ||
|
|
2fbdb29ebc | ||
|
|
ef34dce4dc | ||
|
|
577b7d8ec8 | ||
|
|
4ce099294c | ||
|
|
72bff7fa35 | ||
|
|
8c36b8803f | ||
|
|
c69993bd57 | ||
|
|
ea3e4982d2 | ||
|
|
9a5f6c7b9b | ||
|
|
033d56a4b3 | ||
|
|
dfa684cccc | ||
|
|
b4e0f43dfc | ||
|
|
783cf7f1b0 | ||
|
|
92d6fc3fad | ||
|
|
436cdea4fd | ||
|
|
6b144b3b1f | ||
|
|
7820082eb8 | ||
|
|
e732f921a3 | ||
|
|
8b7c5d8585 | ||
|
|
9e959b189c | ||
|
|
5c7a4cdc22 | ||
|
|
756e619e07 | ||
|
|
eea6194b90 | ||
|
|
a09c1fa154 | ||
|
|
38c742328c | ||
|
|
1d5db1446d | ||
|
|
3f5f3d143f | ||
|
|
d297a7198a | ||
|
|
e5bd99dee4 | ||
|
|
2720e13e88 | ||
|
|
0e9c1789ff | ||
|
|
281cb10f84 | ||
|
|
69a92ffe72 | ||
|
|
6447f159e5 | ||
|
|
94430883ad | ||
|
|
046512eb3d | ||
|
|
4479d97e90 | ||
|
|
bf5fea8951 | ||
|
|
07282e3de6 | ||
|
|
c2f90a20df | ||
|
|
481d2d699e | ||
|
|
cf9a7e6ed3 | ||
|
|
c35235371a | ||
|
|
5d5723ad58 | ||
|
|
6c77294a86 | ||
|
|
823f65f1ca | ||
|
|
0c378c1642 | ||
|
|
bbb4162867 | ||
|
|
5dbdde3f2b | ||
|
|
2521954bd9 | ||
|
|
5f1002894e | ||
|
|
0489b312a3 | ||
|
|
732be5a34f | ||
|
|
1f45c78ebb | ||
|
|
8b86c79bf9 | ||
|
|
710ed81067 | ||
|
|
24ac0e7b9b | ||
|
|
15b2bfbb29 | ||
|
|
b7494eb381 | ||
|
|
27ac590250 | ||
|
|
972076edab | ||
|
|
bfa9a1eb8a | ||
|
|
b0966f14e6 | ||
|
|
37cf0c2fb6 | ||
|
|
4eb11c32b0 | ||
|
|
25457bc09a | ||
|
|
d5cfb5f733 | ||
|
|
79ba6e628e | ||
|
|
ef73add05a | ||
|
|
ec3d11fb27 | ||
|
|
3fbc7031b5 | ||
|
|
40beb5e428 | ||
|
|
0f608c8ef0 | ||
|
|
8509cb4743 | ||
|
|
f4429e8c4a | ||
|
|
e9e0829cdc | ||
|
|
93f0230423 | ||
|
|
f26a0df4a4 | ||
|
|
c7d4624282 | ||
|
|
b03eee2a22 | ||
|
|
7d4d72e706 | ||
|
|
e3c367984b | ||
|
|
0ebfa10d32 | ||
|
|
16d9a077f0 | ||
|
|
a9d8bbad42 | ||
|
|
d78bb94af3 | ||
|
|
b139c0a824 | ||
|
|
5b0b924d34 | ||
|
|
f75acf820c | ||
|
|
43a47f33ac | ||
|
|
fcea3a0877 | ||
|
|
a950ec3bd5 | ||
|
|
e35501ff0a | ||
|
|
4bfad9dad8 | ||
|
|
c5c7a07c12 | ||
|
|
7e22e0e552 | ||
|
|
84ec4bdc79 | ||
|
|
2bcad9b637 | ||
|
|
c8d5f03070 | ||
|
|
168e101a5a | ||
|
|
b4bc7333d9 | ||
|
|
e8b58c940e | ||
|
|
ec7202e3f6 | ||
|
|
9a740f7962 | ||
|
|
9210fdee0d | ||
|
|
d7661f0964 | ||
|
|
139e148912 | ||
|
|
1b8dedb4ed | ||
|
|
5d6b0fa329 | ||
|
|
f35bbd89c9 | ||
|
|
538a9e42f4 | ||
|
|
623147dea7 | ||
|
|
dfecd0cd12 | ||
|
|
fe3af3a676 | ||
|
|
25f60331ed | ||
|
|
d4860a3426 | ||
|
|
e7e77ed86b | ||
|
|
dc80459c59 | ||
|
|
2f2de59234 | ||
|
|
7bccc21878 | ||
|
|
40f9dafa44 | ||
|
|
355d436d29 | ||
|
|
079b684388 | ||
|
|
fd11f46d30 | ||
|
|
cb7099199a | ||
|
|
8566d91e89 | ||
|
|
f44ce49ea7 | ||
|
|
6ef69f6b32 | ||
|
|
f5983d5f10 | ||
|
|
54cce5e089 | ||
|
|
4e4e596a1e | ||
|
|
727a1f5ad1 | ||
|
|
85fa86625b | ||
|
|
2c91877f83 | ||
|
|
7d1fac44e9 | ||
|
|
2e34abfc0d | ||
|
|
81ba63e247 | ||
|
|
8b11a65522 | ||
|
|
7190ad1d15 | ||
|
|
1c9bae5df5 | ||
|
|
cc7fd73916 |
@@ -80,7 +80,6 @@ commands:
|
|||||||
libpulse-devel
|
libpulse-devel
|
||||||
gstreamer-devel
|
gstreamer-devel
|
||||||
gstreamer-plugins-base-devel
|
gstreamer-plugins-base-devel
|
||||||
libxine-devel
|
|
||||||
vlc-devel
|
vlc-devel
|
||||||
libQt5Core-devel
|
libQt5Core-devel
|
||||||
libQt5Gui-devel
|
libQt5Gui-devel
|
||||||
@@ -96,9 +95,7 @@ commands:
|
|||||||
libqt5-linguist-devel
|
libqt5-linguist-devel
|
||||||
libcdio-devel
|
libcdio-devel
|
||||||
libgpod-devel
|
libgpod-devel
|
||||||
libplist-devel
|
|
||||||
libmtp-devel
|
libmtp-devel
|
||||||
libusbmuxd-devel
|
|
||||||
libchromaprint-devel
|
libchromaprint-devel
|
||||||
desktop-file-utils
|
desktop-file-utils
|
||||||
update-desktop-files
|
update-desktop-files
|
||||||
@@ -142,7 +139,6 @@ commands:
|
|||||||
pulseaudio-libs-devel
|
pulseaudio-libs-devel
|
||||||
libnotify-devel
|
libnotify-devel
|
||||||
gnutls-devel
|
gnutls-devel
|
||||||
qt5-devel
|
|
||||||
qt5-qtbase-devel
|
qt5-qtbase-devel
|
||||||
qt5-qtx11extras-devel
|
qt5-qtx11extras-devel
|
||||||
qt5-qttools-devel
|
qt5-qttools-devel
|
||||||
@@ -151,8 +147,6 @@ commands:
|
|||||||
taglib-devel
|
taglib-devel
|
||||||
libcdio-devel
|
libcdio-devel
|
||||||
libgpod-devel
|
libgpod-devel
|
||||||
libplist-devel
|
|
||||||
libusbmuxd-devel
|
|
||||||
libmtp-devel
|
libmtp-devel
|
||||||
libchromaprint-devel
|
libchromaprint-devel
|
||||||
fftw-devel
|
fftw-devel
|
||||||
@@ -161,69 +155,75 @@ commands:
|
|||||||
hicolor-icon-theme
|
hicolor-icon-theme
|
||||||
|
|
||||||
|
|
||||||
install_mageia_dependencies:
|
install_centos_dependencies:
|
||||||
description: Install Mageia dependencies
|
description: Install CentOS dependencies
|
||||||
steps:
|
steps:
|
||||||
- run:
|
- run:
|
||||||
name: Add extra media
|
name: Install epel-release
|
||||||
command: urpmi.addmedia --distrib http://www.mirrorservice.org/sites/mageia.org/pub/mageia/distrib/7/x86_64
|
command: dnf install -y epel-release
|
||||||
|
- run:
|
||||||
|
name: Install epel-release-latest-8.noarch.rpm
|
||||||
|
command: dnf install -y https://dl.fedoraproject.org/pub/epel/epel-release-latest-8.noarch.rpm
|
||||||
|
- run:
|
||||||
|
name: Install config-manager
|
||||||
|
command: dnf install -y 'dnf-command(config-manager)'
|
||||||
|
- run:
|
||||||
|
name: PowerTools
|
||||||
|
command: dnf config-manager --set-enabled PowerTools
|
||||||
|
- run:
|
||||||
|
name: DNF Clean All
|
||||||
|
command: dnf clean all
|
||||||
- run:
|
- run:
|
||||||
name: Update packages
|
name: Update packages
|
||||||
command: urpmi.update -a
|
command: dnf update -y
|
||||||
- run:
|
- run:
|
||||||
name: Configure auto update
|
name: Install CentOS dependencies
|
||||||
command: urpmi --auto-update
|
|
||||||
- run:
|
|
||||||
name: Install dependencies
|
|
||||||
command: >
|
command: >
|
||||||
urpmi --force
|
dnf install -y
|
||||||
urpmi-debuginfo-install
|
|
||||||
git
|
|
||||||
tar
|
|
||||||
rpmdevtools
|
|
||||||
make
|
|
||||||
cmake
|
|
||||||
glibc
|
glibc
|
||||||
binutils
|
|
||||||
gcc-c++
|
gcc-c++
|
||||||
|
make
|
||||||
|
libtool
|
||||||
|
cmake3
|
||||||
|
rpmdevtools
|
||||||
|
redhat-lsb-core
|
||||||
|
git
|
||||||
man
|
man
|
||||||
|
tar
|
||||||
gettext
|
gettext
|
||||||
notification-daemon
|
boost-devel
|
||||||
|
fuse-devel
|
||||||
dbus-devel
|
dbus-devel
|
||||||
libgnutls-devel
|
libnotify-devel
|
||||||
lib64boost-devel
|
gnutls-devel
|
||||||
lib64protobuf-devel
|
sqlite-devel
|
||||||
|
protobuf-devel
|
||||||
protobuf-compiler
|
protobuf-compiler
|
||||||
lib64sqlite3-devel
|
alsa-lib-devel
|
||||||
lib64alsa2-devel
|
pulseaudio-libs-devel
|
||||||
lib64pulseaudio-devel
|
qt5-devel
|
||||||
lib64notify-devel
|
qt5-qtbase-devel
|
||||||
lib64qt5core-devel
|
qt5-qtx11extras-devel
|
||||||
lib64qt5gui-devel
|
qt5-qttools-devel
|
||||||
lib64qt5widgets-devel
|
fftw-devel
|
||||||
lib64qt5network-devel
|
libchromaprint-devel
|
||||||
lib64qt5concurrent-devel
|
libcdio-devel
|
||||||
lib64qt5sql-devel
|
libgpod-devel
|
||||||
lib64qt5dbus-devel
|
libmtp-devel
|
||||||
lib64qt5x11extras-devel
|
libjpeg-devel
|
||||||
lib64qt5help-devel
|
cairo-devel
|
||||||
libqt5test-devel
|
dbus-x11
|
||||||
lib64gstreamer1.0-devel
|
xorg-x11-server-Xvfb
|
||||||
lib64gstreamer-plugins-base1.0-devel
|
xorg-x11-xauth
|
||||||
lib64cdio-devel
|
vim-common
|
||||||
lib64gpod-devel
|
|
||||||
lib64plist-devel
|
|
||||||
lib64usbmuxd-devel
|
|
||||||
lib64mtp-devel
|
|
||||||
lib64raw1394-devel
|
|
||||||
lib64chromaprint-devel
|
|
||||||
libfftw-devel
|
|
||||||
desktop-file-utils
|
desktop-file-utils
|
||||||
appstream-util
|
libappstream-glib
|
||||||
libappstream-glib8
|
appstream-data
|
||||||
hicolor-icon-theme
|
hicolor-icon-theme
|
||||||
qt5ct
|
python3-pip
|
||||||
lib64mesaegl1
|
python3-devel
|
||||||
|
gstreamer1-devel
|
||||||
|
gstreamer1-plugins-base-devel
|
||||||
|
|
||||||
|
|
||||||
install_debian_dependencies:
|
install_debian_dependencies:
|
||||||
@@ -234,6 +234,7 @@ commands:
|
|||||||
command: >
|
command: >
|
||||||
apt-get update && apt-get install -y
|
apt-get update && apt-get install -y
|
||||||
build-essential
|
build-essential
|
||||||
|
dh-make
|
||||||
ssh
|
ssh
|
||||||
git
|
git
|
||||||
make
|
make
|
||||||
@@ -267,9 +268,6 @@ commands:
|
|||||||
libcdio-dev
|
libcdio-dev
|
||||||
libmtp-dev
|
libmtp-dev
|
||||||
libgpod-dev
|
libgpod-dev
|
||||||
libimobiledevice-dev
|
|
||||||
libplist-dev
|
|
||||||
libusbmuxd-dev
|
|
||||||
|
|
||||||
|
|
||||||
install_ubuntu_dependencies:
|
install_ubuntu_dependencies:
|
||||||
@@ -285,6 +283,7 @@ commands:
|
|||||||
command: >
|
command: >
|
||||||
apt-get update && apt-get install -y
|
apt-get update && apt-get install -y
|
||||||
build-essential
|
build-essential
|
||||||
|
dh-make
|
||||||
ssh
|
ssh
|
||||||
git
|
git
|
||||||
make
|
make
|
||||||
@@ -321,9 +320,6 @@ commands:
|
|||||||
libcdio-dev
|
libcdio-dev
|
||||||
libmtp-dev
|
libmtp-dev
|
||||||
libgpod-dev
|
libgpod-dev
|
||||||
libimobiledevice-dev
|
|
||||||
libplist-dev
|
|
||||||
libusbmuxd-dev
|
|
||||||
|
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
@@ -338,24 +334,6 @@ jobs:
|
|||||||
- build_source
|
- build_source
|
||||||
|
|
||||||
|
|
||||||
build_opensuse_tumbleweed:
|
|
||||||
docker:
|
|
||||||
- image: opensuse/tumbleweed
|
|
||||||
environment:
|
|
||||||
RPM_BUILD_NCPUS: "2"
|
|
||||||
steps:
|
|
||||||
- run:
|
|
||||||
name: Update packages
|
|
||||||
command: zypper --non-interactive --gpg-auto-import-keys ref
|
|
||||||
- run:
|
|
||||||
name: Upgrade packages
|
|
||||||
command: zypper --non-interactive --gpg-auto-import-keys dup
|
|
||||||
- install_opensuse_dependencies
|
|
||||||
- checkout
|
|
||||||
- cmake
|
|
||||||
- build_source
|
|
||||||
- build_rpm
|
|
||||||
|
|
||||||
build_opensuse_lp151:
|
build_opensuse_lp151:
|
||||||
docker:
|
docker:
|
||||||
- image: opensuse/leap:15.1
|
- image: opensuse/leap:15.1
|
||||||
@@ -381,18 +359,6 @@ jobs:
|
|||||||
- build_rpm
|
- build_rpm
|
||||||
|
|
||||||
|
|
||||||
build_fedora_30:
|
|
||||||
docker:
|
|
||||||
- image: fedora:30
|
|
||||||
environment:
|
|
||||||
RPM_BUILD_NCPUS: "2"
|
|
||||||
steps:
|
|
||||||
- install_fedora_dependencies
|
|
||||||
- checkout
|
|
||||||
- cmake
|
|
||||||
- build_source
|
|
||||||
- build_rpm
|
|
||||||
|
|
||||||
build_fedora_31:
|
build_fedora_31:
|
||||||
docker:
|
docker:
|
||||||
- image: fedora:31
|
- image: fedora:31
|
||||||
@@ -405,28 +371,31 @@ jobs:
|
|||||||
- build_source
|
- build_source
|
||||||
- build_rpm
|
- build_rpm
|
||||||
|
|
||||||
|
build_fedora_32:
|
||||||
build_mageia_7:
|
|
||||||
docker:
|
docker:
|
||||||
- image: mageia:7
|
- image: fedora:32
|
||||||
environment:
|
environment:
|
||||||
RPM_BUILD_NCPUS: "2"
|
RPM_BUILD_NCPUS: "2"
|
||||||
steps:
|
steps:
|
||||||
- install_mageia_dependencies
|
- install_fedora_dependencies
|
||||||
- checkout
|
- checkout
|
||||||
- cmake
|
- cmake
|
||||||
- build_source
|
- build_source
|
||||||
- build_rpm
|
- build_rpm
|
||||||
|
|
||||||
|
|
||||||
build_debian_stretch:
|
build_centos_8:
|
||||||
docker:
|
docker:
|
||||||
- image: debian:stretch
|
- image: centos:8
|
||||||
|
environment:
|
||||||
|
RPM_BUILD_NCPUS: "2"
|
||||||
steps:
|
steps:
|
||||||
- install_debian_dependencies
|
- install_centos_dependencies
|
||||||
- checkout
|
- checkout
|
||||||
- cmake
|
- cmake
|
||||||
- build_deb
|
- build_source
|
||||||
|
- build_rpm
|
||||||
|
|
||||||
|
|
||||||
build_debian_buster:
|
build_debian_buster:
|
||||||
docker:
|
docker:
|
||||||
@@ -450,16 +419,6 @@ jobs:
|
|||||||
build_ubuntu_bionic:
|
build_ubuntu_bionic:
|
||||||
docker:
|
docker:
|
||||||
- image: ubuntu:bionic
|
- image: ubuntu:bionic
|
||||||
|
|
||||||
steps:
|
|
||||||
- install_ubuntu_dependencies
|
|
||||||
- checkout
|
|
||||||
- cmake
|
|
||||||
- build_deb
|
|
||||||
|
|
||||||
build_ubuntu_eoan:
|
|
||||||
docker:
|
|
||||||
- image: ubuntu:eoan
|
|
||||||
steps:
|
steps:
|
||||||
- install_ubuntu_dependencies
|
- install_ubuntu_dependencies
|
||||||
- checkout
|
- checkout
|
||||||
@@ -475,6 +434,15 @@ jobs:
|
|||||||
- cmake
|
- cmake
|
||||||
- build_deb
|
- build_deb
|
||||||
|
|
||||||
|
build_ubuntu_groovy:
|
||||||
|
docker:
|
||||||
|
- image: ubuntu:groovy
|
||||||
|
steps:
|
||||||
|
- install_ubuntu_dependencies
|
||||||
|
- checkout
|
||||||
|
- cmake
|
||||||
|
- build_deb
|
||||||
|
|
||||||
workflows:
|
workflows:
|
||||||
version: 2
|
version: 2
|
||||||
build_all:
|
build_all:
|
||||||
@@ -486,10 +454,6 @@ workflows:
|
|||||||
only: /.*/
|
only: /.*/
|
||||||
|
|
||||||
|
|
||||||
- build_opensuse_tumbleweed:
|
|
||||||
filters:
|
|
||||||
tags:
|
|
||||||
only: /.*/
|
|
||||||
- build_opensuse_lp151:
|
- build_opensuse_lp151:
|
||||||
filters:
|
filters:
|
||||||
tags:
|
tags:
|
||||||
@@ -500,26 +464,22 @@ workflows:
|
|||||||
only: /.*/
|
only: /.*/
|
||||||
|
|
||||||
|
|
||||||
- build_fedora_30:
|
|
||||||
filters:
|
|
||||||
tags:
|
|
||||||
only: /.*/
|
|
||||||
- build_fedora_31:
|
- build_fedora_31:
|
||||||
filters:
|
filters:
|
||||||
tags:
|
tags:
|
||||||
only: /.*/
|
only: /.*/
|
||||||
|
- build_fedora_32:
|
||||||
|
|
||||||
- build_mageia_7:
|
|
||||||
filters:
|
filters:
|
||||||
tags:
|
tags:
|
||||||
only: /.*/
|
only: /.*/
|
||||||
|
|
||||||
|
|
||||||
- build_debian_stretch:
|
- build_centos_8:
|
||||||
filters:
|
filters:
|
||||||
tags:
|
tags:
|
||||||
only: /.*/
|
only: /.*/
|
||||||
|
|
||||||
|
|
||||||
- build_debian_buster:
|
- build_debian_buster:
|
||||||
filters:
|
filters:
|
||||||
tags:
|
tags:
|
||||||
@@ -530,10 +490,6 @@ workflows:
|
|||||||
only: /.*/
|
only: /.*/
|
||||||
|
|
||||||
|
|
||||||
- build_ubuntu_eoan:
|
|
||||||
filters:
|
|
||||||
tags:
|
|
||||||
only: /.*/
|
|
||||||
- build_ubuntu_bionic:
|
- build_ubuntu_bionic:
|
||||||
filters:
|
filters:
|
||||||
tags:
|
tags:
|
||||||
@@ -542,3 +498,7 @@ workflows:
|
|||||||
filters:
|
filters:
|
||||||
tags:
|
tags:
|
||||||
only: /.*/
|
only: /.*/
|
||||||
|
- build_ubuntu_groovy:
|
||||||
|
filters:
|
||||||
|
tags:
|
||||||
|
only: /.*/
|
||||||
|
|||||||
105
.clang-format
Normal file
105
.clang-format
Normal file
@@ -0,0 +1,105 @@
|
|||||||
|
BasedOnStyle: Chromium
|
||||||
|
Language: Cpp
|
||||||
|
Standard: Cpp11
|
||||||
|
AccessModifierOffset: -1
|
||||||
|
AlignAfterOpenBracket: false
|
||||||
|
AlignConsecutiveAssignments: false
|
||||||
|
AlignConsecutiveDeclarations: false
|
||||||
|
AlignConsecutiveMacros: true
|
||||||
|
AlignEscapedNewlines: true
|
||||||
|
AlignOperands: false
|
||||||
|
AlignTrailingComments: true
|
||||||
|
AllowAllArgumentsOnNextLine: false
|
||||||
|
AllowAllConstructorInitializersOnNextLine: true
|
||||||
|
AllowAllParametersOfDeclarationOnNextLine: false
|
||||||
|
AllowShortBlocksOnASingleLine: false
|
||||||
|
AllowShortCaseLabelsOnASingleLine: true
|
||||||
|
AllowShortFunctionsOnASingleLine: true
|
||||||
|
AllowShortIfStatementsOnASingleLine: true
|
||||||
|
AllowShortLambdasOnASingleLine: true
|
||||||
|
AllowShortLoopsOnASingleLine: true
|
||||||
|
AlwaysBreakAfterReturnType: None
|
||||||
|
AlwaysBreakBeforeMultilineStrings: false
|
||||||
|
AlwaysBreakTemplateDeclarations: No
|
||||||
|
BinPackArguments: true
|
||||||
|
BinPackParameters: true
|
||||||
|
BreakBeforeBraces: false
|
||||||
|
BreakBeforeBinaryOperators: false
|
||||||
|
BreakBeforeBraces: Stroustrup
|
||||||
|
BreakBeforeTernaryOperators: false
|
||||||
|
BreakConstructorInitializers: BeforeColon
|
||||||
|
BreakInheritanceList: BeforeColon
|
||||||
|
BreakStringLiterals: false
|
||||||
|
ColumnLimit: 0
|
||||||
|
CompactNamespaces: false
|
||||||
|
ConstructorInitializerAllOnOneLineOrOnePerLine: false
|
||||||
|
ConstructorInitializerIndentWidth: 4
|
||||||
|
ContinuationIndentWidth: 2
|
||||||
|
Cpp11BracedListStyle: false
|
||||||
|
DeriveLineEnding: true
|
||||||
|
DerivePointerAlignment: true
|
||||||
|
DisableFormat: false
|
||||||
|
ExperimentalAutoDetectBinPacking: true
|
||||||
|
FixNamespaceComments: true
|
||||||
|
IncludeBlocks: Preserve
|
||||||
|
IndentCaseLabels: true
|
||||||
|
IndentGotoLabels: true
|
||||||
|
IndentPPDirectives: AfterHash
|
||||||
|
IndentWidth: 2
|
||||||
|
IndentWrappedFunctionNames: false
|
||||||
|
KeepEmptyLinesAtTheStartOfBlocks: true
|
||||||
|
MaxEmptyLinesToKeep: 100
|
||||||
|
NamespaceIndentation: None
|
||||||
|
ObjCBlockIndentWidth: 2
|
||||||
|
ObjCSpaceAfterProperty: false
|
||||||
|
ObjCSpaceBeforeProtocolList: false
|
||||||
|
PenaltyBreakAssignment: 0
|
||||||
|
PenaltyBreakBeforeFirstCallParameter: 0
|
||||||
|
PenaltyBreakComment: 0
|
||||||
|
PenaltyBreakFirstLessLess: 0
|
||||||
|
PenaltyBreakString: 0
|
||||||
|
PenaltyBreakTemplateDeclaration: 0
|
||||||
|
PenaltyExcessCharacter: 0
|
||||||
|
PenaltyReturnTypeOnItsOwnLine: 0
|
||||||
|
PointerAlignment: Right
|
||||||
|
ReflowComments: false
|
||||||
|
SortIncludes: false
|
||||||
|
SortUsingDeclarations: false
|
||||||
|
SpaceAfterCStyleCast: false
|
||||||
|
SpaceAfterLogicalNot: false
|
||||||
|
SpaceAfterTemplateKeyword: false
|
||||||
|
SpaceBeforeAssignmentOperators: true
|
||||||
|
SpaceBeforeCpp11BracedList: true
|
||||||
|
SpaceBeforeCtorInitializerColon: true
|
||||||
|
SpaceBeforeInheritanceColon: true
|
||||||
|
SpaceBeforeParens: ControlStatements
|
||||||
|
SpaceBeforeRangeBasedForLoopColon: true
|
||||||
|
SpaceBeforeSquareBrackets: false
|
||||||
|
SpaceInEmptyBlock: false
|
||||||
|
SpaceInEmptyParentheses: false
|
||||||
|
SpacesBeforeTrailingComments: 2
|
||||||
|
SpacesInAngles: false
|
||||||
|
SpacesInCStyleCastParentheses: false
|
||||||
|
SpacesInConditionalStatement: false
|
||||||
|
SpacesInContainerLiterals: true
|
||||||
|
SpacesInParentheses: false
|
||||||
|
SpacesInSquareBrackets: false
|
||||||
|
UseCRLF: false
|
||||||
|
UseTab: Never
|
||||||
|
BreakBeforeBraces: Custom
|
||||||
|
BraceWrapping:
|
||||||
|
AfterFunction: false
|
||||||
|
AfterCaseLabel: false
|
||||||
|
AfterStruct: false
|
||||||
|
AfterClass: false
|
||||||
|
AfterEnum: false
|
||||||
|
AfterUnion: false
|
||||||
|
AfterControlStatement: Never
|
||||||
|
AfterNamespace: false
|
||||||
|
AfterObjCDeclaration: false
|
||||||
|
AfterExternBlock: false
|
||||||
|
BeforeCatch: false
|
||||||
|
BeforeElse: true
|
||||||
|
SplitEmptyFunction: false
|
||||||
|
SplitEmptyRecord: false
|
||||||
|
SplitEmptyNamespace: false
|
||||||
@@ -7,7 +7,12 @@ assignees: ''
|
|||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
For general technical questions and help with technical issues please use the forum on https://forum.strawberrymusicplayer.org/
|
For technical issues, questions and feature suggestions/requests please use the forum on https://forum.strawberrymusicplayer.org/
|
||||||
|
|
||||||
|
Check the Changelog to see if the issue is already fixed:
|
||||||
|
https://github.com/strawberrymusicplayer/strawberry/blob/master/Changelog
|
||||||
|
|
||||||
|
If it's fixed, try the latest development build from: https://builds.strawberrymusicplayer.org/
|
||||||
|
|
||||||
**Describe the bug**
|
**Describe the bug**
|
||||||
A clear and concise description of what the bug is.
|
A clear and concise description of what the bug is.
|
||||||
20
.github/ISSUE_TEMPLATE/feature_request.md
vendored
20
.github/ISSUE_TEMPLATE/feature_request.md
vendored
@@ -1,20 +0,0 @@
|
|||||||
---
|
|
||||||
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.
|
|
||||||
1256
.github/workflows/ccpp.yml
vendored
1256
.github/workflows/ccpp.yml
vendored
File diff suppressed because it is too large
Load Diff
1
.gitignore
vendored
1
.gitignore
vendored
@@ -104,7 +104,6 @@ Thumbs.db
|
|||||||
|
|
||||||
# Stuff in dist
|
# Stuff in dist
|
||||||
maketarball.sh
|
maketarball.sh
|
||||||
create-dmg.sh
|
|
||||||
changelog
|
changelog
|
||||||
PKGBUILD
|
PKGBUILD
|
||||||
|
|
||||||
|
|||||||
56
.travis.yml
56
.travis.yml
@@ -2,8 +2,6 @@ sudo: required
|
|||||||
language: C++
|
language: C++
|
||||||
os:
|
os:
|
||||||
- osx
|
- osx
|
||||||
services:
|
|
||||||
- docker
|
|
||||||
compiler:
|
compiler:
|
||||||
- gcc
|
- gcc
|
||||||
|
|
||||||
@@ -12,42 +10,36 @@ before_install:
|
|||||||
echo $DEPLOY_KEY_ENC | base64 --decode | openssl aes-256-cbc -K $encrypted_83a41ac424a6_key -iv $encrypted_83a41ac424a6_iv -out ~/.ssh/id_rsa -d ;
|
echo $DEPLOY_KEY_ENC | base64 --decode | openssl aes-256-cbc -K $encrypted_83a41ac424a6_key -iv $encrypted_83a41ac424a6_iv -out ~/.ssh/id_rsa -d ;
|
||||||
chmod 600 ~/.ssh/id_rsa ;
|
chmod 600 ~/.ssh/id_rsa ;
|
||||||
fi
|
fi
|
||||||
- if [[ "$TRAVIS_OS_NAME" == "linux" ]]; then
|
- git fetch --unshallow
|
||||||
docker build -f Dockerfile -t strawberry-build . || travis_terminate 1;
|
- git pull
|
||||||
docker run --name build -itd strawberry-build /bin/bash || travis_terminate 1;
|
- brew update
|
||||||
docker exec build git clone https://github.com/strawberrymusicplayer/strawberry;
|
- travis_wait 120 brew upgrade || echo "Failed"
|
||||||
fi
|
- travis_wait 120 brew upgrade || echo "Failed"
|
||||||
- if [[ "$TRAVIS_OS_NAME" == "osx" ]]; then
|
- brew install glib pkgconfig libffi protobuf protobuf-c qt gettext gnutls fftw sqlite chromaprint zlib taglib
|
||||||
git fetch --unshallow || travis_terminate 1;
|
- brew install gstreamer gst-plugins-base gst-plugins-good gst-plugins-bad gst-plugins-ugly gst-libav
|
||||||
git pull || travis_terminate 1;
|
- brew install libcdio libmtp
|
||||||
brew unlink python@2 || travis_terminate 1;
|
- brew install create-dmg
|
||||||
brew install glib pkgconfig libffi protobuf protobuf-c qt gettext gnutls fftw sqlite chromaprint zlib taglib;
|
- brew cask install sparkle
|
||||||
brew install gstreamer gst-plugins-base gst-plugins-good gst-plugins-bad gst-plugins-ugly gst-libav;
|
- sudo ln -s /usr/local/Caskroom/sparkle/$(ls /usr/local/Caskroom/sparkle | head -n1)/Sparkle.framework /Library/Frameworks/Sparkle.framework
|
||||||
brew install libcdio libmtp libimobiledevice libplist;
|
- sudo ln -s /usr/local/Caskroom/sparkle/$(ls /usr/local/Caskroom/sparkle | head -n1)/Sparkle.framework.dSYM /Library/Frameworks/Sparkle.framework.dSYM
|
||||||
brew install create-dmg;
|
- export Qt5_DIR=/usr/local/opt/qt5/lib/cmake
|
||||||
export Qt5_DIR=/usr/local/opt/qt5/lib/cmake;
|
- export Qt5LinguistTools_DIR=/usr/local/opt/qt5/lib/cmake/Qt5LinguistTools
|
||||||
export Qt5LinguistTools_DIR=/usr/local/opt/qt5/lib/cmake/Qt5LinguistTools;
|
- ls /usr/local/lib/gstreamer-1.0
|
||||||
export PATH="/usr/local/opt/gettext/bin:$PATH";
|
|
||||||
export PKG_CONFIG_PATH="/usr/local/opt/libffi/lib/pkgconfig/:/usr/local/opt/zlib/lib/pkgconfig:$PKG_CONFIG_PATH";
|
|
||||||
ls /usr/local/lib/gstreamer-1.0;
|
|
||||||
fi
|
|
||||||
before_script:
|
before_script:
|
||||||
- if [[ "$TRAVIS_OS_NAME" == "linux" ]]; then docker exec build cmake -Hstrawberry -Bbuild ; fi
|
- mkdir build
|
||||||
- if [[ "$TRAVIS_OS_NAME" == "osx" ]]; then mkdir build; cd build; cmake .. -DUSE_BUNDLE=ON -DUSE_SYSTEM_TAGLIB=ON ; fi
|
- cd build
|
||||||
|
- cmake .. -DUSE_BUNDLE=ON
|
||||||
script:
|
script:
|
||||||
- if [[ "$TRAVIS_OS_NAME" == "linux" ]]; then docker exec build make -C build -j8 ; fi
|
- make -j8
|
||||||
- if [[ "$TRAVIS_OS_NAME" == "osx" ]]; then
|
- make install
|
||||||
make -j8 || travis_terminate 1;
|
- make dmg
|
||||||
make install || travis_terminate 1;
|
|
||||||
make dmg;
|
|
||||||
fi
|
|
||||||
after_success:
|
after_success:
|
||||||
- if [[ "$TRAVIS_OS_NAME" == "osx" ]]; then ls -lh strawberry*.dmg; fi
|
- ls -lh strawberry*.dmg
|
||||||
- if [[ "$TRAVIS_OS_NAME" == "osx" ]] && [[ "$CC_FOR_BUILD" == "gcc" ]] && [ -f ~/.ssh/id_rsa ]; then
|
- if [[ "$TRAVIS_OS_NAME" == "osx" ]] && [[ "$CC_FOR_BUILD" == "gcc" ]] && [ -f ~/.ssh/id_rsa ]; then
|
||||||
if [[ "$TRAVIS_BRANCH" == "master" ]]; then
|
if [[ "$TRAVIS_BRANCH" == "master" ]]; then
|
||||||
rsync -e "ssh -o StrictHostKeyChecking=no" -va strawberry*.dmg travis@echoes.jkvinge.net:/home/travis/builds/macos;
|
rsync -e "ssh -o StrictHostKeyChecking=no" -va strawberry*.dmg travis@echoes.jkvinge.net:/home/travis/builds/macos/highsierra/;
|
||||||
elif [[ "$TRAVIS_BRANCH" == "macos" ]]; then
|
elif [[ "$TRAVIS_BRANCH" == "macos" ]]; then
|
||||||
rsync -e "ssh -o StrictHostKeyChecking=no" -va strawberry*.dmg travis@echoes.jkvinge.net:/home/travis/builds/macos;
|
rsync -e "ssh -o StrictHostKeyChecking=no" -va strawberry*.dmg travis@echoes.jkvinge.net:/home/travis/builds/macos/highsierra/;
|
||||||
fi
|
fi
|
||||||
fi
|
fi
|
||||||
|
|
||||||
|
|||||||
33
3rdparty/singleapplication/CMakeLists.txt
vendored
33
3rdparty/singleapplication/CMakeLists.txt
vendored
@@ -10,16 +10,37 @@ endif()
|
|||||||
|
|
||||||
set(SINGLEAPP-SOURCES singleapplication.cpp singleapplication_p.cpp)
|
set(SINGLEAPP-SOURCES singleapplication.cpp singleapplication_p.cpp)
|
||||||
set(SINGLEAPP-MOC-HEADERS singleapplication.h singleapplication_p.h)
|
set(SINGLEAPP-MOC-HEADERS singleapplication.h singleapplication_p.h)
|
||||||
QT5_WRAP_CPP(SINGLEAPP-SOURCES-MOC ${SINGLEAPP-MOC-HEADERS})
|
if(BUILD_WITH_QT6)
|
||||||
|
qt6_wrap_cpp(SINGLEAPP-SOURCES-MOC ${SINGLEAPP-MOC-HEADERS})
|
||||||
|
else()
|
||||||
|
qt5_wrap_cpp(SINGLEAPP-SOURCES-MOC ${SINGLEAPP-MOC-HEADERS})
|
||||||
|
endif()
|
||||||
add_library(singleapplication STATIC ${SINGLEAPP-SOURCES} ${SINGLEAPP-SOURCES-MOC})
|
add_library(singleapplication STATIC ${SINGLEAPP-SOURCES} ${SINGLEAPP-SOURCES-MOC})
|
||||||
target_link_libraries(singleapplication Qt5::Core Qt5::Widgets Qt5::Network)
|
target_include_directories(singleapplication PRIVATE
|
||||||
|
${CMAKE_CURRENT_SOURCE_DIR}
|
||||||
|
${CMAKE_CURRENT_BINARY_DIR}
|
||||||
|
)
|
||||||
|
target_link_libraries(singleapplication PRIVATE
|
||||||
|
${QtCore_LIBRARIES}
|
||||||
|
${QtWidgets_LIBRARIES}
|
||||||
|
${QtNetwork_LIBRARIES}
|
||||||
|
)
|
||||||
|
|
||||||
set(SINGLECOREAPP-SOURCES singlecoreapplication.cpp singlecoreapplication_p.cpp)
|
set(SINGLECOREAPP-SOURCES singlecoreapplication.cpp singlecoreapplication_p.cpp)
|
||||||
set(SINGLECOREAPP-MOC-HEADERS singlecoreapplication.h singlecoreapplication_p.h)
|
set(SINGLECOREAPP-MOC-HEADERS singlecoreapplication.h singlecoreapplication_p.h)
|
||||||
QT5_WRAP_CPP(SINGLECOREAPP-SOURCES-MOC ${SINGLECOREAPP-MOC-HEADERS})
|
if(BUILD_WITH_QT6)
|
||||||
|
qt6_wrap_cpp(SINGLECOREAPP-SOURCES-MOC ${SINGLECOREAPP-MOC-HEADERS})
|
||||||
|
else()
|
||||||
|
qt5_wrap_cpp(SINGLECOREAPP-SOURCES-MOC ${SINGLECOREAPP-MOC-HEADERS})
|
||||||
|
endif()
|
||||||
add_library(singlecoreapplication STATIC ${SINGLECOREAPP-SOURCES} ${SINGLECOREAPP-SOURCES-MOC})
|
add_library(singlecoreapplication STATIC ${SINGLECOREAPP-SOURCES} ${SINGLECOREAPP-SOURCES-MOC})
|
||||||
target_link_libraries(singlecoreapplication Qt5::Core Qt5::Widgets Qt5::Network)
|
target_include_directories(singlecoreapplication PRIVATE
|
||||||
|
${CMAKE_CURRENT_SOURCE_DIR}
|
||||||
|
${CMAKE_CURRENT_BINARY_DIR}
|
||||||
|
)
|
||||||
|
target_link_libraries(singlecoreapplication PRIVATE
|
||||||
|
${QtCore_LIBRARIES}
|
||||||
|
${QtNetwork_LIBRARIES}
|
||||||
|
)
|
||||||
|
|
||||||
configure_file(config.h.in "${CMAKE_CURRENT_BINARY_DIR}/config.h")
|
configure_file(config.h.in "${CMAKE_CURRENT_BINARY_DIR}/config.h")
|
||||||
|
|
||||||
include_directories(${CMAKE_CURRENT_BINARY_DIR})
|
|
||||||
|
|||||||
180
3rdparty/singleapplication/singleapplication.cpp
vendored
180
3rdparty/singleapplication/singleapplication.cpp
vendored
@@ -1,6 +1,6 @@
|
|||||||
// The MIT License (MIT)
|
// The MIT License (MIT)
|
||||||
//
|
//
|
||||||
// Copyright (c) Itay Grudev 2015 - 2018
|
// Copyright (c) Itay Grudev 2015 - 2020
|
||||||
//
|
//
|
||||||
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
// of this software and associated documentation files (the "Software"), to deal
|
// of this software and associated documentation files (the "Software"), to deal
|
||||||
@@ -31,16 +31,15 @@
|
|||||||
//
|
//
|
||||||
//
|
//
|
||||||
|
|
||||||
#include <stdlib.h>
|
#include <cstdlib>
|
||||||
#include <limits>
|
#include <limits>
|
||||||
|
|
||||||
#include <QtGlobal>
|
#include <QtGlobal>
|
||||||
#include <QCoreApplication>
|
#include <QApplication>
|
||||||
#include <QThread>
|
#include <QThread>
|
||||||
#include <QSharedMemory>
|
#include <QSharedMemory>
|
||||||
#include <QLocalSocket>
|
#include <QLocalSocket>
|
||||||
#include <QByteArray>
|
#include <QByteArray>
|
||||||
#include <QDateTime>
|
|
||||||
#include <QElapsedTimer>
|
#include <QElapsedTimer>
|
||||||
#include <QtDebug>
|
#include <QtDebug>
|
||||||
|
|
||||||
@@ -48,92 +47,119 @@
|
|||||||
#include "singleapplication_p.h"
|
#include "singleapplication_p.h"
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @brief Constructor.
|
* @brief Constructor. Checks and fires up LocalServer or closes the program if another instance already exists
|
||||||
* Checks and fires up LocalServer or closes the program if another instance already exists
|
|
||||||
* @param argc
|
* @param argc
|
||||||
* @param argv
|
* @param argv
|
||||||
* @param {bool} allowSecondaryInstances
|
* @param allowSecondary Whether to enable secondary instance support
|
||||||
|
* @param options Optional flags to toggle specific behaviour
|
||||||
|
* @param timeout Maximum time blocking functions are allowed during app load
|
||||||
*/
|
*/
|
||||||
SingleApplication::SingleApplication(int &argc, char *argv[], bool allowSecondary, Options options, int timeout)
|
SingleApplication::SingleApplication(int &argc, char *argv[], bool allowSecondary, Options options, int timeout)
|
||||||
: app_t(argc, argv), d_ptr(new SingleApplicationPrivate(this)) {
|
: app_t(argc, argv),
|
||||||
|
d_ptr(new SingleApplicationPrivate(this)) {
|
||||||
|
|
||||||
Q_D(SingleApplication);
|
Q_D(SingleApplication);
|
||||||
|
|
||||||
// Store the current mode of the program
|
// Store the current mode of the program
|
||||||
d->options = options;
|
d->options_ = options;
|
||||||
|
|
||||||
// Generating an application ID used for identifying the shared memory block and QLocalServer
|
// Generating an application ID used for identifying the shared memory block and QLocalServer
|
||||||
d->genBlockServerName();
|
d->genBlockServerName();
|
||||||
|
|
||||||
|
// To mitigate QSharedMemory issues with large amount of processes attempting to attach at the same time
|
||||||
|
d->randomSleep();
|
||||||
|
|
||||||
#ifdef Q_OS_UNIX
|
#ifdef Q_OS_UNIX
|
||||||
// By explicitly attaching it and then deleting it we make sure that the
|
// By explicitly attaching it and then deleting it we make sure that the memory is deleted even after the process has crashed on Unix.
|
||||||
// memory is deleted even after the process has crashed on Unix.
|
d->memory_ = new QSharedMemory(d->blockServerName_);
|
||||||
d->memory = new QSharedMemory(d->blockServerName);
|
d->memory_->attach();
|
||||||
d->memory->attach();
|
delete d->memory_;
|
||||||
delete d->memory;
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
// Guarantee thread safe behaviour with a shared memory block.
|
// Guarantee thread safe behaviour with a shared memory block.
|
||||||
d->memory = new QSharedMemory(d->blockServerName);
|
d->memory_ = new QSharedMemory(d->blockServerName_);
|
||||||
|
|
||||||
// Create a shared memory block
|
// Create a shared memory block
|
||||||
if (d->memory->create(sizeof(InstancesInfo))) {
|
if (d->memory_->create(sizeof(InstancesInfo))) {
|
||||||
// Initialize the shared memory block
|
// Initialize the shared memory block
|
||||||
d->memory->lock();
|
if (!d->memory_->lock()) {
|
||||||
|
qCritical() << "SingleApplication: Unable to lock memory block after create.";
|
||||||
|
abortSafely();
|
||||||
|
}
|
||||||
d->initializeMemoryBlock();
|
d->initializeMemoryBlock();
|
||||||
d->memory->unlock();
|
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
// Attempt to attach to the memory segment
|
if (d->memory_->error() == QSharedMemory::AlreadyExists) {
|
||||||
if (! d->memory->attach()) {
|
// Attempt to attach to the memory segment
|
||||||
qCritical() << "SingleApplication: Unable to attach to shared memory block.";
|
if (!d->memory_->attach()) {
|
||||||
qCritical() << d->memory->errorString();
|
qCritical() << "SingleApplication: Unable to attach to shared memory block.";
|
||||||
delete d;
|
abortSafely();
|
||||||
::exit(EXIT_FAILURE);
|
}
|
||||||
|
if (!d->memory_->lock()) {
|
||||||
|
qCritical() << "SingleApplication: Unable to lock memory block after attach.";
|
||||||
|
abortSafely();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
qCritical() << "SingleApplication: Unable to create block.";
|
||||||
|
abortSafely();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
InstancesInfo* inst = static_cast<InstancesInfo*>(d->memory->data());
|
InstancesInfo *inst = static_cast<InstancesInfo*>(d->memory_->data());
|
||||||
QElapsedTimer time;
|
QElapsedTimer time;
|
||||||
time.start();
|
time.start();
|
||||||
|
|
||||||
// Make sure the shared memory block is initialised and in consistent state
|
// Make sure the shared memory block is initialised and in consistent state
|
||||||
while (true) {
|
forever {
|
||||||
d->memory->lock();
|
// If the shared memory block's checksum is valid continue
|
||||||
|
|
||||||
if (d->blockChecksum() == inst->checksum) break;
|
if (d->blockChecksum() == inst->checksum) break;
|
||||||
|
|
||||||
|
// If more than 5s have elapsed, assume the primary instance crashed and assume it's position
|
||||||
if (time.elapsed() > 5000) {
|
if (time.elapsed() > 5000) {
|
||||||
qWarning() << "SingleApplication: Shared memory block has been in an inconsistent state from more than 5s. Assuming primary instance failure.";
|
qWarning() << "SingleApplication: Shared memory block has been in an inconsistent state from more than 5s. Assuming primary instance failure.";
|
||||||
d->initializeMemoryBlock();
|
d->initializeMemoryBlock();
|
||||||
}
|
}
|
||||||
|
|
||||||
d->memory->unlock();
|
// Otherwise wait for a random period and try again.
|
||||||
|
// The random sleep here limits the probability of a collision between two racing apps and allows the app to initialise faster
|
||||||
// Random sleep here limits the probability of a collision between two racing apps
|
if (!d->memory_->unlock()) {
|
||||||
qsrand(QDateTime::currentMSecsSinceEpoch() % std::numeric_limits<uint>::max());
|
qDebug() << "SingleApplication: Unable to unlock memory for random wait.";
|
||||||
QThread::sleep(8 + static_cast <unsigned long>(static_cast <float>(qrand()) / RAND_MAX * 10));
|
qDebug() << d->memory_->errorString();
|
||||||
|
}
|
||||||
|
d->randomSleep();
|
||||||
|
if (!d->memory_->lock()) {
|
||||||
|
qCritical() << "SingleApplication: Unable to lock memory after random wait.";
|
||||||
|
abortSafely();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (inst->primary == false) {
|
if (inst->primary == false) {
|
||||||
d->startPrimary();
|
d->startPrimary();
|
||||||
d->memory->unlock();
|
if (!d->memory_->unlock()) {
|
||||||
|
qDebug() << "SingleApplication: Unable to unlock memory after primary start.";
|
||||||
|
qDebug() << d->memory_->errorString();
|
||||||
|
}
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check if another instance can be started
|
// Check if another instance can be started
|
||||||
if (allowSecondary) {
|
if (allowSecondary) {
|
||||||
inst->secondary += 1;
|
|
||||||
inst->checksum = d->blockChecksum();
|
|
||||||
d->instanceNumber = inst->secondary;
|
|
||||||
d->startSecondary();
|
d->startSecondary();
|
||||||
if (d->options & Mode::SecondaryNotification) {
|
if (d->options_ & Mode::SecondaryNotification) {
|
||||||
d->connectToPrimary(timeout, SingleApplicationPrivate::SecondaryInstance);
|
d->connectToPrimary(timeout, SingleApplicationPrivate::SecondaryInstance);
|
||||||
}
|
}
|
||||||
d->memory->unlock();
|
if (!d->memory_->unlock()) {
|
||||||
|
qDebug() << "SingleApplication: Unable to unlock memory after secondary start.";
|
||||||
|
qDebug() << d->memory_->errorString();
|
||||||
|
}
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
d->memory->unlock();
|
if (!d->memory_->unlock()) {
|
||||||
|
qDebug() << "SingleApplication: Unable to unlock memory at end of execution.";
|
||||||
|
qDebug() << d->memory_->errorString();
|
||||||
|
}
|
||||||
|
|
||||||
d->connectToPrimary(timeout, SingleApplicationPrivate::NewInstance);
|
d->connectToPrimary(timeout, SingleApplicationPrivate::NewInstance);
|
||||||
|
|
||||||
@@ -143,34 +169,73 @@ SingleApplication::SingleApplication(int &argc, char *argv[], bool allowSecondar
|
|||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* @brief Destructor
|
|
||||||
*/
|
|
||||||
SingleApplication::~SingleApplication() {
|
SingleApplication::~SingleApplication() {
|
||||||
Q_D(SingleApplication);
|
Q_D(SingleApplication);
|
||||||
delete d;
|
delete d;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Checks if the current application instance is primary.
|
||||||
|
* @return Returns true if the instance is primary, false otherwise.
|
||||||
|
*/
|
||||||
bool SingleApplication::isPrimary() {
|
bool SingleApplication::isPrimary() {
|
||||||
Q_D(SingleApplication);
|
Q_D(SingleApplication);
|
||||||
return d->server != nullptr;
|
return d->server_ != nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Checks if the current application instance is secondary.
|
||||||
|
* @return Returns true if the instance is secondary, false otherwise.
|
||||||
|
*/
|
||||||
bool SingleApplication::isSecondary() {
|
bool SingleApplication::isSecondary() {
|
||||||
Q_D(SingleApplication);
|
Q_D(SingleApplication);
|
||||||
return d->server == nullptr;
|
return d->server_ == nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Allows you to identify an instance by returning unique consecutive instance ids.
|
||||||
|
* It is reset when the first (primary) instance of your app starts and only incremented afterwards.
|
||||||
|
* @return Returns a unique instance id.
|
||||||
|
*/
|
||||||
quint32 SingleApplication::instanceId() {
|
quint32 SingleApplication::instanceId() {
|
||||||
Q_D(SingleApplication);
|
Q_D(SingleApplication);
|
||||||
return d->instanceNumber;
|
return d->instanceNumber_;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the OS PID (Process Identifier) of the process running the primary instance.
|
||||||
|
* Especially useful when SingleApplication is coupled with OS. specific APIs.
|
||||||
|
* @return Returns the primary instance PID.
|
||||||
|
*/
|
||||||
qint64 SingleApplication::primaryPid() {
|
qint64 SingleApplication::primaryPid() {
|
||||||
Q_D(SingleApplication);
|
Q_D(SingleApplication);
|
||||||
return d->primaryPid();
|
return d->primaryPid();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the username the primary instance is running as.
|
||||||
|
* @return Returns the username the primary instance is running as.
|
||||||
|
*/
|
||||||
|
QString SingleApplication::primaryUser() {
|
||||||
|
Q_D(SingleApplication);
|
||||||
|
return d->primaryUser();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the username the current instance is running as.
|
||||||
|
* @return Returns the username the current instance is running as.
|
||||||
|
*/
|
||||||
|
QString SingleApplication::currentUser() {
|
||||||
|
Q_D(SingleApplication);
|
||||||
|
return d->getUsername();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sends message to the Primary Instance.
|
||||||
|
* @param message The message to send.
|
||||||
|
* @param timeout the maximum timeout in milliseconds for blocking functions.
|
||||||
|
* @return true if the message was sent successfuly, false otherwise.
|
||||||
|
*/
|
||||||
bool SingleApplication::sendMessage(QByteArray message, int timeout) {
|
bool SingleApplication::sendMessage(QByteArray message, int timeout) {
|
||||||
|
|
||||||
Q_D(SingleApplication);
|
Q_D(SingleApplication);
|
||||||
@@ -179,11 +244,26 @@ bool SingleApplication::sendMessage(QByteArray message, int timeout) {
|
|||||||
if (isPrimary()) return false;
|
if (isPrimary()) return false;
|
||||||
|
|
||||||
// Make sure the socket is connected
|
// Make sure the socket is connected
|
||||||
d->connectToPrimary(timeout, SingleApplicationPrivate::Reconnect);
|
if (!d->connectToPrimary(timeout, SingleApplicationPrivate::Reconnect))
|
||||||
|
return false;
|
||||||
|
|
||||||
d->socket->write(message);
|
d->socket_->write(message);
|
||||||
bool dataWritten = d->socket->waitForBytesWritten(timeout);
|
const bool dataWritten = d->socket_->waitForBytesWritten(timeout);
|
||||||
d->socket->flush();
|
d->socket_->flush();
|
||||||
return dataWritten;
|
return dataWritten;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Cleans up the shared memory block and exits with a failure.
|
||||||
|
* This function halts program execution.
|
||||||
|
*/
|
||||||
|
void SingleApplication::abortSafely() {
|
||||||
|
|
||||||
|
Q_D(SingleApplication);
|
||||||
|
|
||||||
|
qCritical() << "SingleApplication: " << d->memory_->error() << d->memory_->errorString();
|
||||||
|
delete d;
|
||||||
|
::exit(EXIT_FAILURE);
|
||||||
|
|
||||||
|
}
|
||||||
|
|||||||
43
3rdparty/singleapplication/singleapplication.h
vendored
43
3rdparty/singleapplication/singleapplication.h
vendored
@@ -1,6 +1,6 @@
|
|||||||
// The MIT License (MIT)
|
// The MIT License (MIT)
|
||||||
//
|
//
|
||||||
// Copyright (c) Itay Grudev 2015 - 2018
|
// Copyright (c) Itay Grudev 2015 - 2020
|
||||||
//
|
//
|
||||||
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
// of this software and associated documentation files (the "Software"), to deal
|
// of this software and associated documentation files (the "Software"), to deal
|
||||||
@@ -42,16 +42,15 @@
|
|||||||
class SingleApplicationPrivate;
|
class SingleApplicationPrivate;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @brief The SingleApplication class handles multipe instances of the same
|
* @brief The SingleApplication class handles multipe instances of the same Application
|
||||||
* Application
|
* @see QApplication
|
||||||
* @see QCoreApplication
|
|
||||||
*/
|
*/
|
||||||
class SingleApplication : public QApplication {
|
class SingleApplication : public QApplication {
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
|
|
||||||
typedef QApplication app_t;
|
typedef QApplication app_t;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
/**
|
/**
|
||||||
* @brief Mode of operation of SingleApplication.
|
* @brief Mode of operation of SingleApplication.
|
||||||
* Whether the block should be user-wide or system-wide and whether the
|
* Whether the block should be user-wide or system-wide and whether the
|
||||||
@@ -63,11 +62,11 @@ public:
|
|||||||
* @enum
|
* @enum
|
||||||
*/
|
*/
|
||||||
enum Mode {
|
enum Mode {
|
||||||
User = 1 << 0,
|
User = 1 << 0,
|
||||||
System = 1 << 1,
|
System = 1 << 1,
|
||||||
SecondaryNotification = 1 << 2,
|
SecondaryNotification = 1 << 2,
|
||||||
ExcludeAppVersion = 1 << 3,
|
ExcludeAppVersion = 1 << 3,
|
||||||
ExcludeAppPath = 1 << 4
|
ExcludeAppPath = 1 << 4
|
||||||
};
|
};
|
||||||
Q_DECLARE_FLAGS(Options, Mode)
|
Q_DECLARE_FLAGS(Options, Mode)
|
||||||
|
|
||||||
@@ -91,8 +90,8 @@ public:
|
|||||||
* Usually 4*timeout would be the worst case (fail) scenario.
|
* Usually 4*timeout would be the worst case (fail) scenario.
|
||||||
* @see See the corresponding QAPPLICATION_CLASS constructor for reference
|
* @see See the corresponding QAPPLICATION_CLASS constructor for reference
|
||||||
*/
|
*/
|
||||||
explicit SingleApplication( int &argc, char *argv[], bool allowSecondary = false, Options options = Mode::User, int timeout = 1000 );
|
explicit SingleApplication(int &argc, char *argv[], bool allowSecondary = false, Options options = Mode::User, int timeout = 1000);
|
||||||
~SingleApplication();
|
~SingleApplication() override;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @brief Returns if the instance is the primary instance
|
* @brief Returns if the instance is the primary instance
|
||||||
@@ -118,6 +117,18 @@ public:
|
|||||||
*/
|
*/
|
||||||
qint64 primaryPid();
|
qint64 primaryPid();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief Returns the username of the user running the primary instance
|
||||||
|
* @returns {QString}
|
||||||
|
*/
|
||||||
|
QString primaryUser();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief Returns the username of the current user
|
||||||
|
* @returns {QString}
|
||||||
|
*/
|
||||||
|
QString currentUser();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @brief Sends a message to the primary instance. Returns true on success.
|
* @brief Sends a message to the primary instance. Returns true on success.
|
||||||
* @param {int} timeout - Timeout for connecting
|
* @param {int} timeout - Timeout for connecting
|
||||||
@@ -125,18 +136,18 @@ public:
|
|||||||
* @note sendMessage() will return false if invoked from the primary
|
* @note sendMessage() will return false if invoked from the primary
|
||||||
* instance.
|
* instance.
|
||||||
*/
|
*/
|
||||||
bool sendMessage( QByteArray message, int timeout = 1000 );
|
bool sendMessage(QByteArray message, int timeout = 1000);
|
||||||
|
|
||||||
signals:
|
signals:
|
||||||
void instanceStarted();
|
void instanceStarted();
|
||||||
void receivedMessage( quint32 instanceId, QByteArray message );
|
void receivedMessage(quint32 instanceId, QByteArray message);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
SingleApplicationPrivate *d_ptr;
|
SingleApplicationPrivate *d_ptr;
|
||||||
Q_DECLARE_PRIVATE(SingleApplication)
|
Q_DECLARE_PRIVATE(SingleApplication)
|
||||||
|
void abortSafely();
|
||||||
};
|
};
|
||||||
|
|
||||||
Q_DECLARE_OPERATORS_FOR_FLAGS(SingleApplication::Options)
|
Q_DECLARE_OPERATORS_FOR_FLAGS(SingleApplication::Options)
|
||||||
|
|
||||||
#endif // SINGLEAPPLICATION_H
|
#endif // SINGLEAPPLICATION_H
|
||||||
|
|||||||
361
3rdparty/singleapplication/singleapplication_p.cpp
vendored
361
3rdparty/singleapplication/singleapplication_p.cpp
vendored
@@ -1,6 +1,6 @@
|
|||||||
// The MIT License (MIT)
|
// The MIT License (MIT)
|
||||||
//
|
//
|
||||||
// Copyright (c) Itay Grudev 2015 - 2018
|
// Copyright (c) Itay Grudev 2015 - 2020
|
||||||
//
|
//
|
||||||
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
// of this software and associated documentation files (the "Software"), to deal
|
// of this software and associated documentation files (the "Software"), to deal
|
||||||
@@ -44,6 +44,8 @@
|
|||||||
# include <pwd.h>
|
# include <pwd.h>
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#include <QObject>
|
||||||
|
#include <QThread>
|
||||||
#include <QIODevice>
|
#include <QIODevice>
|
||||||
#include <QSharedMemory>
|
#include <QSharedMemory>
|
||||||
#include <QByteArray>
|
#include <QByteArray>
|
||||||
@@ -52,6 +54,12 @@
|
|||||||
#include <QLocalServer>
|
#include <QLocalServer>
|
||||||
#include <QLocalSocket>
|
#include <QLocalSocket>
|
||||||
#include <QDir>
|
#include <QDir>
|
||||||
|
#include <QElapsedTimer>
|
||||||
|
#if QT_VERSION >= QT_VERSION_CHECK(5, 10, 0)
|
||||||
|
# include <QRandomGenerator>
|
||||||
|
#else
|
||||||
|
# include <QDateTime>
|
||||||
|
#endif
|
||||||
|
|
||||||
#include "singleapplication.h"
|
#include "singleapplication.h"
|
||||||
#include "singleapplication_p.h"
|
#include "singleapplication_p.h"
|
||||||
@@ -61,33 +69,73 @@
|
|||||||
# include <lmcons.h>
|
# include <lmcons.h>
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
SingleApplicationPrivate::SingleApplicationPrivate(SingleApplication *_q_ptr)
|
SingleApplicationPrivate::SingleApplicationPrivate(SingleApplication *ptr)
|
||||||
: q_ptr(_q_ptr),
|
: q_ptr(ptr),
|
||||||
memory(nullptr),
|
memory_(nullptr),
|
||||||
socket(nullptr),
|
socket_(nullptr),
|
||||||
server(nullptr),
|
server_(nullptr),
|
||||||
instanceNumber(-1)
|
instanceNumber_(-1) {}
|
||||||
{}
|
|
||||||
|
|
||||||
SingleApplicationPrivate::~SingleApplicationPrivate() {
|
SingleApplicationPrivate::~SingleApplicationPrivate() {
|
||||||
|
|
||||||
if (socket != nullptr) {
|
if (socket_ != nullptr) {
|
||||||
socket->close();
|
socket_->close();
|
||||||
delete socket;
|
delete socket_;
|
||||||
|
socket_ = nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
memory->lock();
|
if (memory_ != nullptr) {
|
||||||
InstancesInfo* inst = static_cast<InstancesInfo*>(memory->data());
|
memory_->lock();
|
||||||
if (server != nullptr) {
|
InstancesInfo *inst = static_cast<InstancesInfo*>(memory_->data());
|
||||||
server->close();
|
if (server_ != nullptr) {
|
||||||
delete server;
|
server_->close();
|
||||||
inst->primary = false;
|
delete server_;
|
||||||
inst->primaryPid = -1;
|
inst->primary = false;
|
||||||
inst->checksum = blockChecksum();
|
inst->primaryPid = -1;
|
||||||
}
|
inst->primaryUser[0] = '\0';
|
||||||
memory->unlock();
|
inst->checksum = blockChecksum();
|
||||||
|
}
|
||||||
|
memory_->unlock();
|
||||||
|
|
||||||
delete memory;
|
delete memory_;
|
||||||
|
memory_ = nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
QString SingleApplicationPrivate::getUsername() {
|
||||||
|
|
||||||
|
#ifdef Q_OS_UNIX
|
||||||
|
QString username;
|
||||||
|
#if defined(HAVE_GETEUID) && defined(HAVE_GETPWUID)
|
||||||
|
struct passwd *pw = getpwuid(geteuid());
|
||||||
|
if (pw) {
|
||||||
|
username = QString::fromLocal8Bit(pw->pw_name);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
if (username.isEmpty()) {
|
||||||
|
#if QT_VERSION >= QT_VERSION_CHECK(5, 10, 0)
|
||||||
|
username = qEnvironmentVariable("USER");
|
||||||
|
#else
|
||||||
|
username = QString::fromLocal8Bit(qgetenv("USER"));
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
return username;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef Q_OS_WIN
|
||||||
|
wchar_t username[UNLEN + 1];
|
||||||
|
// Specifies size of the buffer on input
|
||||||
|
DWORD usernameLength = UNLEN + 1;
|
||||||
|
if (GetUserNameW(username, &usernameLength)) {
|
||||||
|
return QString::fromWCharArray(username);
|
||||||
|
}
|
||||||
|
#if QT_VERSION >= QT_VERSION_CHECK(5, 10, 0)
|
||||||
|
return qEnvironmentVariable("USERNAME");
|
||||||
|
#else
|
||||||
|
return QString::fromLocal8Bit(qgetenv("USERNAME"));
|
||||||
|
#endif
|
||||||
|
#endif
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -99,11 +147,11 @@ void SingleApplicationPrivate::genBlockServerName() {
|
|||||||
appData.addData(SingleApplication::app_t::organizationName().toUtf8());
|
appData.addData(SingleApplication::app_t::organizationName().toUtf8());
|
||||||
appData.addData(SingleApplication::app_t::organizationDomain().toUtf8());
|
appData.addData(SingleApplication::app_t::organizationDomain().toUtf8());
|
||||||
|
|
||||||
if (!(options & SingleApplication::Mode::ExcludeAppVersion)) {
|
if (!(options_ & SingleApplication::Mode::ExcludeAppVersion)) {
|
||||||
appData.addData(SingleApplication::app_t::applicationVersion().toUtf8());
|
appData.addData(SingleApplication::app_t::applicationVersion().toUtf8());
|
||||||
}
|
}
|
||||||
|
|
||||||
if (! (options & SingleApplication::Mode::ExcludeAppPath)) {
|
if (!(options_ & SingleApplication::Mode::ExcludeAppPath)) {
|
||||||
#ifdef Q_OS_WIN
|
#ifdef Q_OS_WIN
|
||||||
appData.addData(SingleApplication::app_t::applicationFilePath().toLower().toUtf8());
|
appData.addData(SingleApplication::app_t::applicationFilePath().toLower().toUtf8());
|
||||||
#else
|
#else
|
||||||
@@ -112,42 +160,22 @@ void SingleApplicationPrivate::genBlockServerName() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// User level block requires a user specific data in the hash
|
// User level block requires a user specific data in the hash
|
||||||
if (options & SingleApplication::Mode::User) {
|
if (options_ & SingleApplication::Mode::User) {
|
||||||
#ifdef Q_OS_UNIX
|
appData.addData(getUsername().toUtf8());
|
||||||
QByteArray username;
|
|
||||||
#if defined(HAVE_GETEUID) && defined(HAVE_GETPWUID)
|
|
||||||
struct passwd *pw = getpwuid(geteuid());
|
|
||||||
if (pw) {
|
|
||||||
username = pw->pw_name;
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
if (username.isEmpty()) username = qgetenv("USER");
|
|
||||||
appData.addData(username);
|
|
||||||
#endif
|
|
||||||
#ifdef Q_OS_WIN
|
|
||||||
wchar_t username [ UNLEN + 1 ];
|
|
||||||
// Specifies size of the buffer on input
|
|
||||||
DWORD usernameLength = UNLEN + 1;
|
|
||||||
if (GetUserNameW(username, &usernameLength)) {
|
|
||||||
appData.addData(QString::fromWCharArray(username).toUtf8());
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
appData.addData(qgetenv("USERNAME"));
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Replace the backslash in RFC 2045 Base64 [a-zA-Z0-9+/=] to comply with server naming requirements.
|
// Replace the backslash in RFC 2045 Base64 [a-zA-Z0-9+/=] to comply with server naming requirements.
|
||||||
blockServerName = appData.result().toBase64().replace("/", "_");
|
blockServerName_ = appData.result().toBase64().replace("/", "_");
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void SingleApplicationPrivate::initializeMemoryBlock() {
|
void SingleApplicationPrivate::initializeMemoryBlock() {
|
||||||
|
|
||||||
InstancesInfo* inst = static_cast<InstancesInfo*>(memory->data());
|
InstancesInfo *inst = static_cast<InstancesInfo*>(memory_->data());
|
||||||
inst->primary = false;
|
inst->primary = false;
|
||||||
inst->secondary = 0;
|
inst->secondary = 0;
|
||||||
inst->primaryPid = -1;
|
inst->primaryPid = -1;
|
||||||
|
inst->primaryUser[0] = '\0';
|
||||||
inst->checksum = blockChecksum();
|
inst->checksum = blockChecksum();
|
||||||
|
|
||||||
}
|
}
|
||||||
@@ -156,133 +184,161 @@ void SingleApplicationPrivate::startPrimary() {
|
|||||||
|
|
||||||
Q_Q(SingleApplication);
|
Q_Q(SingleApplication);
|
||||||
|
|
||||||
// Successful creation means that no main process exists
|
|
||||||
// So we start a QLocalServer to listen for connections
|
|
||||||
QLocalServer::removeServer(blockServerName);
|
|
||||||
server = new QLocalServer();
|
|
||||||
|
|
||||||
// Restrict access to the socket according to the
|
|
||||||
// SingleApplication::Mode::User flag on User level or no restrictions
|
|
||||||
if (options & SingleApplication::Mode::User) {
|
|
||||||
server->setSocketOptions(QLocalServer::UserAccessOption);
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
server->setSocketOptions(QLocalServer::WorldAccessOption);
|
|
||||||
}
|
|
||||||
|
|
||||||
server->listen(blockServerName);
|
|
||||||
QObject::connect(server, &QLocalServer::newConnection, this, &SingleApplicationPrivate::slotConnectionEstablished);
|
|
||||||
|
|
||||||
// Reset the number of connections
|
// Reset the number of connections
|
||||||
InstancesInfo* inst = static_cast <InstancesInfo*>(memory->data());
|
InstancesInfo *inst = static_cast<InstancesInfo*>(memory_->data());
|
||||||
|
|
||||||
inst->primary = true;
|
inst->primary = true;
|
||||||
inst->primaryPid = q->applicationPid();
|
inst->primaryPid = q->applicationPid();
|
||||||
|
qstrncpy(inst->primaryUser, getUsername().toUtf8().data(), sizeof(inst->primaryUser));
|
||||||
inst->checksum = blockChecksum();
|
inst->checksum = blockChecksum();
|
||||||
|
instanceNumber_ = 0;
|
||||||
|
// Successful creation means that no main process exists
|
||||||
|
// So we start a QLocalServer to listen for connections
|
||||||
|
QLocalServer::removeServer(blockServerName_);
|
||||||
|
server_ = new QLocalServer();
|
||||||
|
|
||||||
instanceNumber = 0;
|
// Restrict access to the socket according to the SingleApplication::Mode::User flag on User level or no restrictions
|
||||||
|
if (options_ & SingleApplication::Mode::User) {
|
||||||
|
server_->setSocketOptions(QLocalServer::UserAccessOption);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
server_->setSocketOptions(QLocalServer::WorldAccessOption);
|
||||||
|
}
|
||||||
|
|
||||||
|
server_->listen(blockServerName_);
|
||||||
|
QObject::connect(server_, &QLocalServer::newConnection, this, &SingleApplicationPrivate::slotConnectionEstablished);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void SingleApplicationPrivate::startSecondary() {}
|
void SingleApplicationPrivate::startSecondary() {
|
||||||
|
|
||||||
void SingleApplicationPrivate::connectToPrimary(const int msecs, const ConnectionType connectionType) {
|
InstancesInfo *inst = static_cast<InstancesInfo*>(memory_->data());
|
||||||
|
|
||||||
|
inst->secondary += 1;
|
||||||
|
inst->checksum = blockChecksum();
|
||||||
|
instanceNumber_ = inst->secondary;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
bool SingleApplicationPrivate::connectToPrimary(int timeout, ConnectionType connectionType) {
|
||||||
|
|
||||||
|
QElapsedTimer time;
|
||||||
|
time.start();
|
||||||
|
|
||||||
// Connect to the Local Server of the Primary Instance if not already connected.
|
// Connect to the Local Server of the Primary Instance if not already connected.
|
||||||
if (socket == nullptr) {
|
if (socket_ == nullptr) {
|
||||||
socket = new QLocalSocket();
|
socket_ = new QLocalSocket();
|
||||||
}
|
}
|
||||||
|
|
||||||
// If already connected - we are done;
|
if (socket_->state() == QLocalSocket::ConnectedState) return true;
|
||||||
if (socket->state() == QLocalSocket::ConnectedState)
|
|
||||||
return;
|
|
||||||
|
|
||||||
// If not connect
|
if (socket_->state() != QLocalSocket::ConnectedState) {
|
||||||
if (socket->state() == QLocalSocket::UnconnectedState ||
|
|
||||||
socket->state() == QLocalSocket::ClosingState) {
|
|
||||||
socket->connectToServer(blockServerName);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Wait for being connected
|
forever {
|
||||||
if (socket->state() == QLocalSocket::ConnectingState) {
|
randomSleep();
|
||||||
socket->waitForConnected(msecs);
|
|
||||||
|
if (socket_->state() != QLocalSocket::ConnectingState)
|
||||||
|
socket_->connectToServer(blockServerName_);
|
||||||
|
|
||||||
|
if (socket_->state() == QLocalSocket::ConnectingState) {
|
||||||
|
socket_->waitForConnected(timeout - time.elapsed());
|
||||||
|
}
|
||||||
|
|
||||||
|
// If connected break out of the loop
|
||||||
|
if (socket_->state() == QLocalSocket::ConnectedState) break;
|
||||||
|
|
||||||
|
// If elapsed time since start is longer than the method timeout return
|
||||||
|
if (time.elapsed() >= timeout) return false;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Initialisation message according to the SingleApplication protocol
|
// Initialisation message according to the SingleApplication protocol
|
||||||
if (socket->state() == QLocalSocket::ConnectedState) {
|
QByteArray initMsg;
|
||||||
// Notify the parent that a new instance had been started;
|
QDataStream writeStream(&initMsg, QIODevice::WriteOnly);
|
||||||
QByteArray initMsg;
|
writeStream.setVersion(QDataStream::Qt_5_8);
|
||||||
QDataStream writeStream(&initMsg, QIODevice::WriteOnly);
|
|
||||||
|
|
||||||
writeStream.setVersion(QDataStream::Qt_5_6);
|
writeStream << blockServerName_.toLatin1();
|
||||||
|
writeStream << static_cast<quint8>(connectionType);
|
||||||
|
writeStream << instanceNumber_;
|
||||||
|
|
||||||
writeStream << blockServerName.toLatin1();
|
#if QT_VERSION >= QT_VERSION_CHECK(6, 0, 0)
|
||||||
writeStream << static_cast<quint8>(connectionType);
|
quint16 checksum = qChecksum(QByteArray(initMsg, static_cast<quint32>(initMsg.length())));
|
||||||
writeStream << instanceNumber;
|
#else
|
||||||
quint16 checksum = qChecksum(initMsg.constData(), static_cast<quint32>(initMsg.length()));
|
quint16 checksum = qChecksum(initMsg.constData(), static_cast<quint32>(initMsg.length()));
|
||||||
writeStream << checksum;
|
#endif
|
||||||
|
|
||||||
// The header indicates the message length that follows
|
writeStream << checksum;
|
||||||
QByteArray header;
|
|
||||||
QDataStream headerStream(&header, QIODevice::WriteOnly);
|
|
||||||
|
|
||||||
headerStream.setVersion(QDataStream::Qt_5_6);
|
// The header indicates the message length that follows
|
||||||
|
QByteArray header;
|
||||||
|
QDataStream headerStream(&header, QIODevice::WriteOnly);
|
||||||
|
headerStream.setVersion(QDataStream::Qt_5_8);
|
||||||
|
headerStream << static_cast<quint64>(initMsg.length());
|
||||||
|
|
||||||
headerStream << static_cast <quint64>(initMsg.length());
|
socket_->write(header);
|
||||||
|
socket_->write(initMsg);
|
||||||
|
bool result = socket_->waitForBytesWritten(timeout - time.elapsed());
|
||||||
|
socket_->flush();
|
||||||
|
|
||||||
socket->write(header);
|
return result;
|
||||||
socket->write(initMsg);
|
|
||||||
socket->flush();
|
|
||||||
socket->waitForBytesWritten(msecs);
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
quint16 SingleApplicationPrivate::blockChecksum() {
|
quint16 SingleApplicationPrivate::blockChecksum() {
|
||||||
|
|
||||||
return qChecksum(static_cast <const char *>(memory->data()), offsetof(InstancesInfo, checksum));
|
#if QT_VERSION >= QT_VERSION_CHECK(6, 0, 0)
|
||||||
|
quint16 checksum = qChecksum(QByteArray(static_cast<const char*>(memory_->constData()), offsetof(InstancesInfo, checksum)));
|
||||||
|
#else
|
||||||
|
quint16 checksum = qChecksum(static_cast<const char*>(memory_->constData()), offsetof(InstancesInfo, checksum));
|
||||||
|
#endif
|
||||||
|
|
||||||
|
return checksum;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
qint64 SingleApplicationPrivate::primaryPid() {
|
qint64 SingleApplicationPrivate::primaryPid() {
|
||||||
|
|
||||||
qint64 pid;
|
memory_->lock();
|
||||||
|
InstancesInfo *inst = static_cast<InstancesInfo*>(memory_->data());
|
||||||
memory->lock();
|
qint64 pid = inst->primaryPid;
|
||||||
InstancesInfo* inst = static_cast<InstancesInfo*>(memory->data());
|
memory_->unlock();
|
||||||
pid = inst->primaryPid;
|
|
||||||
memory->unlock();
|
|
||||||
|
|
||||||
return pid;
|
return pid;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
QString SingleApplicationPrivate::primaryUser() {
|
||||||
|
|
||||||
|
memory_->lock();
|
||||||
|
InstancesInfo *inst = static_cast<InstancesInfo*>(memory_->data());
|
||||||
|
QByteArray username = inst->primaryUser;
|
||||||
|
memory_->unlock();
|
||||||
|
|
||||||
|
return QString::fromUtf8(username);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @brief Executed when a connection has been made to the LocalServer
|
* @brief Executed when a connection has been made to the LocalServer
|
||||||
*/
|
*/
|
||||||
void SingleApplicationPrivate::slotConnectionEstablished() {
|
void SingleApplicationPrivate::slotConnectionEstablished() {
|
||||||
|
|
||||||
QLocalSocket *nextConnSocket = server->nextPendingConnection();
|
QLocalSocket *nextConnSocket = server_->nextPendingConnection();
|
||||||
connectionMap.insert(nextConnSocket, ConnectionInfo());
|
connectionMap_.insert(nextConnSocket, ConnectionInfo());
|
||||||
|
|
||||||
QObject::connect(nextConnSocket, &QLocalSocket::aboutToClose,
|
QObject::connect(nextConnSocket, &QLocalSocket::aboutToClose, [nextConnSocket, this]() {
|
||||||
[nextConnSocket, this]() {
|
auto &info = connectionMap_[nextConnSocket];
|
||||||
auto &info = connectionMap[nextConnSocket];
|
Q_EMIT this->slotClientConnectionClosed(nextConnSocket, info.instanceId);
|
||||||
Q_EMIT this->slotClientConnectionClosed(nextConnSocket, info.instanceId);
|
});
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
QObject::connect(nextConnSocket, &QLocalSocket::disconnected,
|
QObject::connect(nextConnSocket, &QLocalSocket::disconnected, [nextConnSocket, this]() {
|
||||||
[nextConnSocket, this](){
|
connectionMap_.remove(nextConnSocket);
|
||||||
connectionMap.remove(nextConnSocket);
|
nextConnSocket->deleteLater();
|
||||||
nextConnSocket->deleteLater();
|
});
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
QObject::connect(nextConnSocket, &QLocalSocket::readyRead,
|
QObject::connect(nextConnSocket, &QLocalSocket::readyRead, [nextConnSocket, this]() {
|
||||||
[nextConnSocket, this]() {
|
auto &info = connectionMap_[nextConnSocket];
|
||||||
auto &info = connectionMap[nextConnSocket];
|
switch (info.stage) {
|
||||||
switch(info.stage) {
|
|
||||||
case StageHeader:
|
case StageHeader:
|
||||||
readInitMessageHeader(nextConnSocket);
|
readInitMessageHeader(nextConnSocket);
|
||||||
break;
|
break;
|
||||||
@@ -294,34 +350,32 @@ void SingleApplicationPrivate::slotConnectionEstablished() {
|
|||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
break;
|
break;
|
||||||
};
|
};
|
||||||
}
|
});
|
||||||
);
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void SingleApplicationPrivate::readInitMessageHeader(QLocalSocket *sock) {
|
void SingleApplicationPrivate::readInitMessageHeader(QLocalSocket *sock) {
|
||||||
|
|
||||||
if (!connectionMap.contains(sock)) {
|
if (!connectionMap_.contains(sock)) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (sock->bytesAvailable() < (qint64) sizeof(quint64)) {
|
if (sock->bytesAvailable() < static_cast<qint64>(sizeof(quint64))) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
QDataStream headerStream(sock);
|
QDataStream headerStream(sock);
|
||||||
|
headerStream.setVersion(QDataStream::Qt_5_8);
|
||||||
headerStream.setVersion(QDataStream::Qt_5_6);
|
|
||||||
|
|
||||||
// Read the header to know the message length
|
// Read the header to know the message length
|
||||||
quint64 msgLen = 0;
|
quint64 msgLen = 0;
|
||||||
headerStream >> msgLen;
|
headerStream >> msgLen;
|
||||||
ConnectionInfo &info = connectionMap[sock];
|
ConnectionInfo &info = connectionMap_[sock];
|
||||||
info.stage = StageBody;
|
info.stage = StageBody;
|
||||||
info.msgLen = msgLen;
|
info.msgLen = msgLen;
|
||||||
|
|
||||||
if (sock->bytesAvailable() >= (qint64) msgLen) {
|
if (sock->bytesAvailable() >= static_cast<qint64>(msgLen)) {
|
||||||
readInitMessageBody(sock);
|
readInitMessageBody(sock);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -331,20 +385,19 @@ void SingleApplicationPrivate::readInitMessageBody(QLocalSocket *sock) {
|
|||||||
|
|
||||||
Q_Q(SingleApplication);
|
Q_Q(SingleApplication);
|
||||||
|
|
||||||
if (!connectionMap.contains(sock)) {
|
if (!connectionMap_.contains(sock)) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
ConnectionInfo &info = connectionMap[sock];
|
ConnectionInfo &info = connectionMap_[sock];
|
||||||
if (sock->bytesAvailable() < (qint64)info.msgLen) {
|
if (sock->bytesAvailable() < static_cast<qint64>(info.msgLen)) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Read the message body
|
// Read the message body
|
||||||
QByteArray msgBytes = sock->read(info.msgLen);
|
QByteArray msgBytes = sock->read(info.msgLen);
|
||||||
QDataStream readStream(msgBytes);
|
QDataStream readStream(msgBytes);
|
||||||
|
readStream.setVersion(QDataStream::Qt_5_8);
|
||||||
readStream.setVersion(QDataStream::Qt_5_6);
|
|
||||||
|
|
||||||
// server name
|
// server name
|
||||||
QByteArray latin1Name;
|
QByteArray latin1Name;
|
||||||
@@ -354,7 +407,7 @@ void SingleApplicationPrivate::readInitMessageBody(QLocalSocket *sock) {
|
|||||||
ConnectionType connectionType = InvalidConnection;
|
ConnectionType connectionType = InvalidConnection;
|
||||||
quint8 connTypeVal = InvalidConnection;
|
quint8 connTypeVal = InvalidConnection;
|
||||||
readStream >> connTypeVal;
|
readStream >> connTypeVal;
|
||||||
connectionType = static_cast <ConnectionType>(connTypeVal);
|
connectionType = static_cast<ConnectionType>(connTypeVal);
|
||||||
|
|
||||||
// instance id
|
// instance id
|
||||||
quint32 instanceId = 0;
|
quint32 instanceId = 0;
|
||||||
@@ -364,9 +417,13 @@ void SingleApplicationPrivate::readInitMessageBody(QLocalSocket *sock) {
|
|||||||
quint16 msgChecksum = 0;
|
quint16 msgChecksum = 0;
|
||||||
readStream >> msgChecksum;
|
readStream >> msgChecksum;
|
||||||
|
|
||||||
|
#if QT_VERSION >= QT_VERSION_CHECK(6, 0, 0)
|
||||||
|
const quint16 actualChecksum = qChecksum(QByteArray(msgBytes, static_cast<quint32>(msgBytes.length() - sizeof(quint16))));
|
||||||
|
#else
|
||||||
const quint16 actualChecksum = qChecksum(msgBytes.constData(), static_cast<quint32>(msgBytes.length() - sizeof(quint16)));
|
const quint16 actualChecksum = qChecksum(msgBytes.constData(), static_cast<quint32>(msgBytes.length() - sizeof(quint16)));
|
||||||
|
#endif
|
||||||
|
|
||||||
bool isValid = readStream.status() == QDataStream::Ok && QLatin1String(latin1Name) == blockServerName && msgChecksum == actualChecksum;
|
bool isValid = readStream.status() == QDataStream::Ok && QLatin1String(latin1Name) == blockServerName_ && msgChecksum == actualChecksum;
|
||||||
|
|
||||||
if (!isValid) {
|
if (!isValid) {
|
||||||
sock->close();
|
sock->close();
|
||||||
@@ -376,7 +433,7 @@ void SingleApplicationPrivate::readInitMessageBody(QLocalSocket *sock) {
|
|||||||
info.instanceId = instanceId;
|
info.instanceId = instanceId;
|
||||||
info.stage = StageConnected;
|
info.stage = StageConnected;
|
||||||
|
|
||||||
if (connectionType == NewInstance || (connectionType == SecondaryInstance && options & SingleApplication::Mode::SecondaryNotification)) {
|
if (connectionType == NewInstance || (connectionType == SecondaryInstance && options_ & SingleApplication::Mode::SecondaryNotification)) {
|
||||||
Q_EMIT q->instanceStarted();
|
Q_EMIT q->instanceStarted();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -395,7 +452,19 @@ void SingleApplicationPrivate::slotDataAvailable(QLocalSocket *dataSocket, const
|
|||||||
|
|
||||||
void SingleApplicationPrivate::slotClientConnectionClosed(QLocalSocket *closedSocket, const quint32 instanceId) {
|
void SingleApplicationPrivate::slotClientConnectionClosed(QLocalSocket *closedSocket, const quint32 instanceId) {
|
||||||
|
|
||||||
if (closedSocket->bytesAvailable() > 0)
|
if (closedSocket->bytesAvailable() > 0) {
|
||||||
Q_EMIT slotDataAvailable(closedSocket, instanceId);
|
Q_EMIT slotDataAvailable(closedSocket, instanceId);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
void SingleApplicationPrivate::randomSleep() {
|
||||||
|
|
||||||
|
#if QT_VERSION >= QT_VERSION_CHECK(5, 10, 0)
|
||||||
|
QThread::msleep(QRandomGenerator::global()->bounded(8u, 18u));
|
||||||
|
#else
|
||||||
|
qsrand(QDateTime::currentMSecsSinceEpoch() % std::numeric_limits<uint>::max());
|
||||||
|
QThread::msleep(8 + static_cast<unsigned long>(static_cast<float>(qrand()) / RAND_MAX * 10));
|
||||||
|
#endif
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
28
3rdparty/singleapplication/singleapplication_p.h
vendored
28
3rdparty/singleapplication/singleapplication_p.h
vendored
@@ -1,6 +1,6 @@
|
|||||||
// The MIT License (MIT)
|
// The MIT License (MIT)
|
||||||
//
|
//
|
||||||
// Copyright (c) Itay Grudev 2015 - 2016
|
// Copyright (c) Itay Grudev 2015 - 2020
|
||||||
//
|
//
|
||||||
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
// of this software and associated documentation files (the "Software"), to deal
|
// of this software and associated documentation files (the "Software"), to deal
|
||||||
@@ -36,6 +36,7 @@
|
|||||||
|
|
||||||
#include <QtGlobal>
|
#include <QtGlobal>
|
||||||
#include <QObject>
|
#include <QObject>
|
||||||
|
#include <QString>
|
||||||
#include <QMap>
|
#include <QMap>
|
||||||
|
|
||||||
#include "singleapplication.h"
|
#include "singleapplication.h"
|
||||||
@@ -48,6 +49,7 @@ struct InstancesInfo {
|
|||||||
bool primary;
|
bool primary;
|
||||||
quint32 secondary;
|
quint32 secondary;
|
||||||
qint64 primaryPid;
|
qint64 primaryPid;
|
||||||
|
char primaryUser[128];
|
||||||
quint16 checksum;
|
quint16 checksum;
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -60,6 +62,7 @@ struct ConnectionInfo {
|
|||||||
|
|
||||||
class SingleApplicationPrivate : public QObject {
|
class SingleApplicationPrivate : public QObject {
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
|
|
||||||
public:
|
public:
|
||||||
enum ConnectionType : quint8 {
|
enum ConnectionType : quint8 {
|
||||||
InvalidConnection = 0,
|
InvalidConnection = 0,
|
||||||
@@ -74,27 +77,30 @@ class SingleApplicationPrivate : public QObject {
|
|||||||
};
|
};
|
||||||
Q_DECLARE_PUBLIC(SingleApplication)
|
Q_DECLARE_PUBLIC(SingleApplication)
|
||||||
|
|
||||||
SingleApplicationPrivate(SingleApplication *_q_ptr);
|
explicit SingleApplicationPrivate(SingleApplication *ptr);
|
||||||
~SingleApplicationPrivate();
|
~SingleApplicationPrivate() override;
|
||||||
|
|
||||||
|
QString getUsername();
|
||||||
void genBlockServerName();
|
void genBlockServerName();
|
||||||
void initializeMemoryBlock();
|
void initializeMemoryBlock();
|
||||||
void startPrimary();
|
void startPrimary();
|
||||||
void startSecondary();
|
void startSecondary();
|
||||||
void connectToPrimary(const int msecs, const ConnectionType connectionType);
|
bool connectToPrimary(const int msecs, const ConnectionType connectionType);
|
||||||
quint16 blockChecksum();
|
quint16 blockChecksum();
|
||||||
qint64 primaryPid();
|
qint64 primaryPid();
|
||||||
|
QString primaryUser();
|
||||||
void readInitMessageHeader(QLocalSocket *socket);
|
void readInitMessageHeader(QLocalSocket *socket);
|
||||||
void readInitMessageBody(QLocalSocket *socket);
|
void readInitMessageBody(QLocalSocket *socket);
|
||||||
|
void randomSleep();
|
||||||
|
|
||||||
SingleApplication *q_ptr;
|
SingleApplication *q_ptr;
|
||||||
QSharedMemory *memory;
|
QSharedMemory *memory_;
|
||||||
QLocalSocket *socket;
|
QLocalSocket *socket_;
|
||||||
QLocalServer *server;
|
QLocalServer *server_;
|
||||||
quint32 instanceNumber;
|
quint32 instanceNumber_;
|
||||||
QString blockServerName;
|
QString blockServerName_;
|
||||||
SingleApplication::Options options;
|
SingleApplication::Options options_;
|
||||||
QMap<QLocalSocket*, ConnectionInfo> connectionMap;
|
QMap<QLocalSocket*, ConnectionInfo> connectionMap_;
|
||||||
|
|
||||||
public slots:
|
public slots:
|
||||||
void slotConnectionEstablished();
|
void slotConnectionEstablished();
|
||||||
|
|||||||
185
3rdparty/singleapplication/singlecoreapplication.cpp
vendored
185
3rdparty/singleapplication/singlecoreapplication.cpp
vendored
@@ -1,6 +1,6 @@
|
|||||||
// The MIT License (MIT)
|
// The MIT License (MIT)
|
||||||
//
|
//
|
||||||
// Copyright (c) Itay Grudev 2015 - 2018
|
// Copyright (c) Itay Grudev 2015 - 2020
|
||||||
//
|
//
|
||||||
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
// of this software and associated documentation files (the "Software"), to deal
|
// of this software and associated documentation files (the "Software"), to deal
|
||||||
@@ -31,7 +31,7 @@
|
|||||||
//
|
//
|
||||||
//
|
//
|
||||||
|
|
||||||
#include <stdlib.h>
|
#include <cstdlib>
|
||||||
#include <limits>
|
#include <limits>
|
||||||
|
|
||||||
#include <QtGlobal>
|
#include <QtGlobal>
|
||||||
@@ -40,7 +40,6 @@
|
|||||||
#include <QSharedMemory>
|
#include <QSharedMemory>
|
||||||
#include <QLocalSocket>
|
#include <QLocalSocket>
|
||||||
#include <QByteArray>
|
#include <QByteArray>
|
||||||
#include <QDateTime>
|
|
||||||
#include <QElapsedTimer>
|
#include <QElapsedTimer>
|
||||||
#include <QtDebug>
|
#include <QtDebug>
|
||||||
|
|
||||||
@@ -48,93 +47,119 @@
|
|||||||
#include "singlecoreapplication_p.h"
|
#include "singlecoreapplication_p.h"
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @brief Constructor. Checks and fires up LocalServer or closes the program
|
* @brief Constructor. Checks and fires up LocalServer or closes the program if another instance already exists
|
||||||
* if another instance already exists
|
|
||||||
* @param argc
|
* @param argc
|
||||||
* @param argv
|
* @param argv
|
||||||
* @param {bool} allowSecondaryInstances
|
* @param allowSecondary Whether to enable secondary instance support
|
||||||
|
* @param options Optional flags to toggle specific behaviour
|
||||||
|
* @param timeout Maximum time blocking functions are allowed during app load
|
||||||
*/
|
*/
|
||||||
SingleCoreApplication::SingleCoreApplication(int &argc, char *argv[], bool allowSecondary, Options options, int timeout)
|
SingleCoreApplication::SingleCoreApplication(int &argc, char *argv[], bool allowSecondary, Options options, int timeout)
|
||||||
: app_t(argc, argv), d_ptr(new SingleCoreApplicationPrivate(this)) {
|
: app_t(argc, argv),
|
||||||
|
d_ptr(new SingleCoreApplicationPrivate(this)) {
|
||||||
|
|
||||||
Q_D(SingleCoreApplication);
|
Q_D(SingleCoreApplication);
|
||||||
|
|
||||||
// Store the current mode of the program
|
// Store the current mode of the program
|
||||||
d->options = options;
|
d->options_ = options;
|
||||||
|
|
||||||
// Generating an application ID used for identifying the shared memory
|
// Generating an application ID used for identifying the shared memory block and QLocalServer
|
||||||
// block and QLocalServer
|
|
||||||
d->genBlockServerName();
|
d->genBlockServerName();
|
||||||
|
|
||||||
|
// To mitigate QSharedMemory issues with large amount of processes attempting to attach at the same time
|
||||||
|
d->randomSleep();
|
||||||
|
|
||||||
#ifdef Q_OS_UNIX
|
#ifdef Q_OS_UNIX
|
||||||
// By explicitly attaching it and then deleting it we make sure that the
|
// By explicitly attaching it and then deleting it we make sure that the memory is deleted even after the process has crashed on Unix.
|
||||||
// memory is deleted even after the process has crashed on Unix.
|
d->memory_ = new QSharedMemory(d->blockServerName_);
|
||||||
d->memory = new QSharedMemory(d->blockServerName);
|
d->memory_->attach();
|
||||||
d->memory->attach();
|
delete d->memory_;
|
||||||
delete d->memory;
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
// Guarantee thread safe behaviour with a shared memory block.
|
// Guarantee thread safe behaviour with a shared memory block.
|
||||||
d->memory = new QSharedMemory(d->blockServerName);
|
d->memory_ = new QSharedMemory(d->blockServerName_);
|
||||||
|
|
||||||
// Create a shared memory block
|
// Create a shared memory block
|
||||||
if (d->memory->create(sizeof(InstancesInfo))) {
|
if (d->memory_->create(sizeof(InstancesInfo))) {
|
||||||
// Initialize the shared memory block
|
// Initialize the shared memory block
|
||||||
d->memory->lock();
|
if (!d->memory_->lock()) {
|
||||||
|
qCritical() << "SingleCoreApplication: Unable to lock memory block after create.";
|
||||||
|
abortSafely();
|
||||||
|
}
|
||||||
d->initializeMemoryBlock();
|
d->initializeMemoryBlock();
|
||||||
d->memory->unlock();
|
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
// Attempt to attach to the memory segment
|
if (d->memory_->error() == QSharedMemory::AlreadyExists) {
|
||||||
if (!d->memory->attach()) {
|
// Attempt to attach to the memory segment
|
||||||
qCritical() << "SingleCoreApplication: Unable to attach to shared memory block.";
|
if (!d->memory_->attach()) {
|
||||||
qCritical() << d->memory->errorString();
|
qCritical() << "SingleCoreApplication: Unable to attach to shared memory block.";
|
||||||
delete d;
|
abortSafely();
|
||||||
::exit(EXIT_FAILURE);
|
}
|
||||||
|
if (!d->memory_->lock()) {
|
||||||
|
qCritical() << "SingleCoreApplication: Unable to lock memory block after attach.";
|
||||||
|
abortSafely();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
qCritical() << "SingleCoreApplication: Unable to create block.";
|
||||||
|
abortSafely();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
InstancesInfo* inst = static_cast<InstancesInfo*>(d->memory->data());
|
InstancesInfo *inst = static_cast<InstancesInfo*>(d->memory_->data());
|
||||||
QElapsedTimer time;
|
QElapsedTimer time;
|
||||||
time.start();
|
time.start();
|
||||||
|
|
||||||
// Make sure the shared memory block is initialised and in consistent state
|
// Make sure the shared memory block is initialised and in consistent state
|
||||||
while (true) {
|
forever {
|
||||||
d->memory->lock();
|
// If the shared memory block's checksum is valid continue
|
||||||
|
if (d->blockChecksum() == inst->checksum) break;
|
||||||
if(d->blockChecksum() == inst->checksum) break;
|
|
||||||
|
|
||||||
|
// If more than 5s have elapsed, assume the primary instance crashed and assume it's position
|
||||||
if (time.elapsed() > 5000) {
|
if (time.elapsed() > 5000) {
|
||||||
qWarning() << "SingleCoreApplication: Shared memory block has been in an inconsistent state from more than 5s. Assuming primary instance failure.";
|
qWarning() << "SingleCoreApplication: Shared memory block has been in an inconsistent state from more than 5s. Assuming primary instance failure.";
|
||||||
d->initializeMemoryBlock();
|
d->initializeMemoryBlock();
|
||||||
}
|
}
|
||||||
|
|
||||||
d->memory->unlock();
|
// Otherwise wait for a random period and try again.
|
||||||
|
// The random sleep here limits the probability of a collision between two racing apps and allows the app to initialise faster
|
||||||
// Random sleep here limits the probability of a collision between two racing apps
|
if (!d->memory_->unlock()) {
|
||||||
qsrand(QDateTime::currentMSecsSinceEpoch() % std::numeric_limits<uint>::max());
|
qDebug() << "SingleCoreApplication: Unable to unlock memory for random wait.";
|
||||||
QThread::sleep(8 + static_cast <unsigned long>(static_cast <float>(qrand()) / RAND_MAX * 10));
|
qDebug() << d->memory_->errorString();
|
||||||
|
}
|
||||||
|
d->randomSleep();
|
||||||
|
if (!d->memory_->lock()) {
|
||||||
|
qCritical() << "SingleCoreApplication: Unable to lock memory after random wait.";
|
||||||
|
abortSafely();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (inst->primary == false) {
|
if (inst->primary == false) {
|
||||||
d->startPrimary();
|
d->startPrimary();
|
||||||
d->memory->unlock();
|
if (!d->memory_->unlock()) {
|
||||||
|
qDebug() << "SingleCoreApplication: Unable to unlock memory after primary start.";
|
||||||
|
qDebug() << d->memory_->errorString();
|
||||||
|
}
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check if another instance can be started
|
// Check if another instance can be started
|
||||||
if (allowSecondary) {
|
if (allowSecondary) {
|
||||||
inst->secondary += 1;
|
|
||||||
inst->checksum = d->blockChecksum();
|
|
||||||
d->instanceNumber = inst->secondary;
|
|
||||||
d->startSecondary();
|
d->startSecondary();
|
||||||
if(d->options & Mode::SecondaryNotification) {
|
if (d->options_ & Mode::SecondaryNotification) {
|
||||||
d->connectToPrimary(timeout, SingleCoreApplicationPrivate::SecondaryInstance);
|
d->connectToPrimary(timeout, SingleCoreApplicationPrivate::SecondaryInstance);
|
||||||
}
|
}
|
||||||
d->memory->unlock();
|
if (!d->memory_->unlock()) {
|
||||||
|
qDebug() << "SingleCoreApplication: Unable to unlock memory after secondary start.";
|
||||||
|
qDebug() << d->memory_->errorString();
|
||||||
|
}
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
d->memory->unlock();
|
if (!d->memory_->unlock()) {
|
||||||
|
qDebug() << "SingleCoreApplication: Unable to unlock memory at end of execution.";
|
||||||
|
qDebug() << d->memory_->errorString();
|
||||||
|
}
|
||||||
|
|
||||||
d->connectToPrimary(timeout, SingleCoreApplicationPrivate::NewInstance);
|
d->connectToPrimary(timeout, SingleCoreApplicationPrivate::NewInstance);
|
||||||
|
|
||||||
@@ -144,47 +169,101 @@ SingleCoreApplication::SingleCoreApplication(int &argc, char *argv[], bool allow
|
|||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* @brief Destructor
|
|
||||||
*/
|
|
||||||
SingleCoreApplication::~SingleCoreApplication() {
|
SingleCoreApplication::~SingleCoreApplication() {
|
||||||
Q_D(SingleCoreApplication);
|
Q_D(SingleCoreApplication);
|
||||||
delete d;
|
delete d;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Checks if the current application instance is primary.
|
||||||
|
* @return Returns true if the instance is primary, false otherwise.
|
||||||
|
*/
|
||||||
bool SingleCoreApplication::isPrimary() {
|
bool SingleCoreApplication::isPrimary() {
|
||||||
Q_D(SingleCoreApplication);
|
Q_D(SingleCoreApplication);
|
||||||
return d->server != nullptr;
|
return d->server_ != nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Checks if the current application instance is secondary.
|
||||||
|
* @return Returns true if the instance is secondary, false otherwise.
|
||||||
|
*/
|
||||||
bool SingleCoreApplication::isSecondary() {
|
bool SingleCoreApplication::isSecondary() {
|
||||||
Q_D(SingleCoreApplication);
|
Q_D(SingleCoreApplication);
|
||||||
return d->server == nullptr;
|
return d->server_ == nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Allows you to identify an instance by returning unique consecutive instance ids.
|
||||||
|
* It is reset when the first (primary) instance of your app starts and only incremented afterwards.
|
||||||
|
* @return Returns a unique instance id.
|
||||||
|
*/
|
||||||
quint32 SingleCoreApplication::instanceId() {
|
quint32 SingleCoreApplication::instanceId() {
|
||||||
Q_D(SingleCoreApplication);
|
Q_D(SingleCoreApplication);
|
||||||
return d->instanceNumber;
|
return d->instanceNumber_;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the OS PID (Process Identifier) of the process running the primary instance.
|
||||||
|
* Especially useful when SingleCoreApplication is coupled with OS. specific APIs.
|
||||||
|
* @return Returns the primary instance PID.
|
||||||
|
*/
|
||||||
qint64 SingleCoreApplication::primaryPid() {
|
qint64 SingleCoreApplication::primaryPid() {
|
||||||
Q_D(SingleCoreApplication);
|
Q_D(SingleCoreApplication);
|
||||||
return d->primaryPid();
|
return d->primaryPid();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the username the primary instance is running as.
|
||||||
|
* @return Returns the username the primary instance is running as.
|
||||||
|
*/
|
||||||
|
QString SingleCoreApplication::primaryUser() {
|
||||||
|
Q_D(SingleCoreApplication);
|
||||||
|
return d->primaryUser();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the username the current instance is running as.
|
||||||
|
* @return Returns the username the current instance is running as.
|
||||||
|
*/
|
||||||
|
QString SingleCoreApplication::currentUser() {
|
||||||
|
Q_D(SingleCoreApplication);
|
||||||
|
return d->getUsername();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sends message to the Primary Instance.
|
||||||
|
* @param message The message to send.
|
||||||
|
* @param timeout the maximum timeout in milliseconds for blocking functions.
|
||||||
|
* @return true if the message was sent successfuly, false otherwise.
|
||||||
|
*/
|
||||||
bool SingleCoreApplication::sendMessage(QByteArray message, int timeout) {
|
bool SingleCoreApplication::sendMessage(QByteArray message, int timeout) {
|
||||||
|
|
||||||
Q_D(SingleCoreApplication);
|
Q_D(SingleCoreApplication);
|
||||||
|
|
||||||
// Nobody to connect to
|
// Nobody to connect to
|
||||||
if(isPrimary()) return false;
|
if (isPrimary()) return false;
|
||||||
|
|
||||||
// Make sure the socket is connected
|
// Make sure the socket is connected
|
||||||
d->connectToPrimary(timeout, SingleCoreApplicationPrivate::Reconnect);
|
if (!d->connectToPrimary(timeout, SingleCoreApplicationPrivate::Reconnect))
|
||||||
|
return false;
|
||||||
|
|
||||||
d->socket->write(message);
|
d->socket_->write(message);
|
||||||
bool dataWritten = d->socket->waitForBytesWritten(timeout);
|
const bool dataWritten = d->socket_->waitForBytesWritten(timeout);
|
||||||
d->socket->flush();
|
d->socket_->flush();
|
||||||
return dataWritten;
|
return dataWritten;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Cleans up the shared memory block and exits with a failure.
|
||||||
|
* This function halts program execution.
|
||||||
|
*/
|
||||||
|
void SingleCoreApplication::abortSafely() {
|
||||||
|
|
||||||
|
Q_D(SingleCoreApplication);
|
||||||
|
|
||||||
|
qCritical() << "SingleCoreApplication: " << d->memory_->error() << d->memory_->errorString();
|
||||||
|
delete d;
|
||||||
|
::exit(EXIT_FAILURE);
|
||||||
|
|
||||||
|
}
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
// The MIT License (MIT)
|
// The MIT License (MIT)
|
||||||
//
|
//
|
||||||
// Copyright (c) Itay Grudev 2015 - 2018
|
// Copyright (c) Itay Grudev 2015 - 2020
|
||||||
//
|
//
|
||||||
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
// of this software and associated documentation files (the "Software"), to deal
|
// of this software and associated documentation files (the "Software"), to deal
|
||||||
@@ -42,8 +42,7 @@
|
|||||||
class SingleCoreApplicationPrivate;
|
class SingleCoreApplicationPrivate;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @brief The SingleCoreApplication class handles multiple instances of the same
|
* @brief The SingleCoreApplication class handles multipe instances of the same Application
|
||||||
* Application
|
|
||||||
* @see QCoreApplication
|
* @see QCoreApplication
|
||||||
*/
|
*/
|
||||||
class SingleCoreApplication : public QCoreApplication {
|
class SingleCoreApplication : public QCoreApplication {
|
||||||
@@ -51,7 +50,7 @@ class SingleCoreApplication : public QCoreApplication {
|
|||||||
|
|
||||||
typedef QCoreApplication app_t;
|
typedef QCoreApplication app_t;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
/**
|
/**
|
||||||
* @brief Mode of operation of SingleCoreApplication.
|
* @brief Mode of operation of SingleCoreApplication.
|
||||||
* Whether the block should be user-wide or system-wide and whether the
|
* Whether the block should be user-wide or system-wide and whether the
|
||||||
@@ -63,11 +62,11 @@ public:
|
|||||||
* @enum
|
* @enum
|
||||||
*/
|
*/
|
||||||
enum Mode {
|
enum Mode {
|
||||||
User = 1 << 0,
|
User = 1 << 0,
|
||||||
System = 1 << 1,
|
System = 1 << 1,
|
||||||
SecondaryNotification = 1 << 2,
|
SecondaryNotification = 1 << 2,
|
||||||
ExcludeAppVersion = 1 << 3,
|
ExcludeAppVersion = 1 << 3,
|
||||||
ExcludeAppPath = 1 << 4
|
ExcludeAppPath = 1 << 4
|
||||||
};
|
};
|
||||||
Q_DECLARE_FLAGS(Options, Mode)
|
Q_DECLARE_FLAGS(Options, Mode)
|
||||||
|
|
||||||
@@ -89,10 +88,9 @@ public:
|
|||||||
* operations. It does not guarantee that the SingleCoreApplication
|
* operations. It does not guarantee that the SingleCoreApplication
|
||||||
* initialisation will be completed in given time, though is a good hint.
|
* initialisation will be completed in given time, though is a good hint.
|
||||||
* Usually 4*timeout would be the worst case (fail) scenario.
|
* Usually 4*timeout would be the worst case (fail) scenario.
|
||||||
* @see See the corresponding QAPPLICATION_CLASS constructor for reference
|
|
||||||
*/
|
*/
|
||||||
explicit SingleCoreApplication( int &argc, char *argv[], bool allowSecondary = false, Options options = Mode::User, int timeout = 1000 );
|
explicit SingleCoreApplication(int &argc, char *argv[], const bool allowSecondary = false, const Options options = Mode::User, const int timeout = 1000);
|
||||||
~SingleCoreApplication();
|
~SingleCoreApplication() override;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @brief Returns if the instance is the primary instance
|
* @brief Returns if the instance is the primary instance
|
||||||
@@ -118,6 +116,18 @@ public:
|
|||||||
*/
|
*/
|
||||||
qint64 primaryPid();
|
qint64 primaryPid();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief Returns the username of the user running the primary instance
|
||||||
|
* @returns {QString}
|
||||||
|
*/
|
||||||
|
QString primaryUser();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief Returns the username of the current user
|
||||||
|
* @returns {QString}
|
||||||
|
*/
|
||||||
|
QString currentUser();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @brief Sends a message to the primary instance. Returns true on success.
|
* @brief Sends a message to the primary instance. Returns true on success.
|
||||||
* @param {int} timeout - Timeout for connecting
|
* @param {int} timeout - Timeout for connecting
|
||||||
@@ -125,17 +135,18 @@ public:
|
|||||||
* @note sendMessage() will return false if invoked from the primary
|
* @note sendMessage() will return false if invoked from the primary
|
||||||
* instance.
|
* instance.
|
||||||
*/
|
*/
|
||||||
bool sendMessage( QByteArray message, int timeout = 1000 );
|
bool sendMessage(QByteArray message, int timeout = 1000);
|
||||||
|
|
||||||
signals:
|
signals:
|
||||||
void instanceStarted();
|
void instanceStarted();
|
||||||
void receivedMessage( quint32 instanceId, QByteArray message );
|
void receivedMessage(quint32 instanceId, QByteArray message);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
SingleCoreApplicationPrivate *d_ptr;
|
SingleCoreApplicationPrivate *d_ptr;
|
||||||
Q_DECLARE_PRIVATE(SingleCoreApplication)
|
Q_DECLARE_PRIVATE(SingleCoreApplication)
|
||||||
|
void abortSafely();
|
||||||
};
|
};
|
||||||
|
|
||||||
Q_DECLARE_OPERATORS_FOR_FLAGS(SingleCoreApplication::Options)
|
Q_DECLARE_OPERATORS_FOR_FLAGS(SingleCoreApplication::Options)
|
||||||
|
|
||||||
#endif // SINGLECOREAPPLICATION_H
|
#endif // SINGLECOREAPPLICATION_H
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
// The MIT License (MIT)
|
// The MIT License (MIT)
|
||||||
//
|
//
|
||||||
// Copyright (c) Itay Grudev 2015 - 2018
|
// Copyright (c) Itay Grudev 2015 - 2020
|
||||||
//
|
//
|
||||||
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
// of this software and associated documentation files (the "Software"), to deal
|
// of this software and associated documentation files (the "Software"), to deal
|
||||||
@@ -44,6 +44,8 @@
|
|||||||
# include <pwd.h>
|
# include <pwd.h>
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#include <QObject>
|
||||||
|
#include <QThread>
|
||||||
#include <QIODevice>
|
#include <QIODevice>
|
||||||
#include <QSharedMemory>
|
#include <QSharedMemory>
|
||||||
#include <QByteArray>
|
#include <QByteArray>
|
||||||
@@ -52,6 +54,12 @@
|
|||||||
#include <QLocalServer>
|
#include <QLocalServer>
|
||||||
#include <QLocalSocket>
|
#include <QLocalSocket>
|
||||||
#include <QDir>
|
#include <QDir>
|
||||||
|
#include <QElapsedTimer>
|
||||||
|
#if QT_VERSION >= QT_VERSION_CHECK(5, 10, 0)
|
||||||
|
# include <QRandomGenerator>
|
||||||
|
#else
|
||||||
|
# include <QDateTime>
|
||||||
|
#endif
|
||||||
|
|
||||||
#include "singlecoreapplication.h"
|
#include "singlecoreapplication.h"
|
||||||
#include "singlecoreapplication_p.h"
|
#include "singlecoreapplication_p.h"
|
||||||
@@ -61,33 +69,73 @@
|
|||||||
# include <lmcons.h>
|
# include <lmcons.h>
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
SingleCoreApplicationPrivate::SingleCoreApplicationPrivate(SingleCoreApplication *_q_ptr)
|
SingleCoreApplicationPrivate::SingleCoreApplicationPrivate(SingleCoreApplication *ptr)
|
||||||
: q_ptr(_q_ptr),
|
: q_ptr(ptr),
|
||||||
memory(nullptr),
|
memory_(nullptr),
|
||||||
socket(nullptr),
|
socket_(nullptr),
|
||||||
server(nullptr),
|
server_(nullptr),
|
||||||
instanceNumber(-1)
|
instanceNumber_(-1) {}
|
||||||
{}
|
|
||||||
|
|
||||||
SingleCoreApplicationPrivate::~SingleCoreApplicationPrivate() {
|
SingleCoreApplicationPrivate::~SingleCoreApplicationPrivate() {
|
||||||
|
|
||||||
if (socket != nullptr) {
|
if (socket_ != nullptr) {
|
||||||
socket->close();
|
socket_->close();
|
||||||
delete socket;
|
delete socket_;
|
||||||
|
socket_ = nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
memory->lock();
|
if (memory_ != nullptr) {
|
||||||
InstancesInfo* inst = static_cast<InstancesInfo*>(memory->data());
|
memory_->lock();
|
||||||
if (server != nullptr) {
|
InstancesInfo *inst = static_cast<InstancesInfo*>(memory_->data());
|
||||||
server->close();
|
if (server_ != nullptr) {
|
||||||
delete server;
|
server_->close();
|
||||||
inst->primary = false;
|
delete server_;
|
||||||
inst->primaryPid = -1;
|
inst->primary = false;
|
||||||
inst->checksum = blockChecksum();
|
inst->primaryPid = -1;
|
||||||
}
|
inst->primaryUser[0] = '\0';
|
||||||
memory->unlock();
|
inst->checksum = blockChecksum();
|
||||||
|
}
|
||||||
|
memory_->unlock();
|
||||||
|
|
||||||
delete memory;
|
delete memory_;
|
||||||
|
memory_ = nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
QString SingleCoreApplicationPrivate::getUsername() {
|
||||||
|
|
||||||
|
#ifdef Q_OS_UNIX
|
||||||
|
QString username;
|
||||||
|
#if defined(HAVE_GETEUID) && defined(HAVE_GETPWUID)
|
||||||
|
struct passwd *pw = getpwuid(geteuid());
|
||||||
|
if (pw) {
|
||||||
|
username = QString::fromLocal8Bit(pw->pw_name);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
if (username.isEmpty()) {
|
||||||
|
#if QT_VERSION >= QT_VERSION_CHECK(5, 10, 0)
|
||||||
|
username = qEnvironmentVariable("USER");
|
||||||
|
#else
|
||||||
|
username = QString::fromLocal8Bit(qgetenv("USER"));
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
return username;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef Q_OS_WIN
|
||||||
|
wchar_t username[UNLEN + 1];
|
||||||
|
// Specifies size of the buffer on input
|
||||||
|
DWORD usernameLength = UNLEN + 1;
|
||||||
|
if (GetUserNameW(username, &usernameLength)) {
|
||||||
|
return QString::fromWCharArray(username);
|
||||||
|
}
|
||||||
|
#if QT_VERSION >= QT_VERSION_CHECK(5, 10, 0)
|
||||||
|
return qEnvironmentVariable("USERNAME");
|
||||||
|
#else
|
||||||
|
return QString::fromLocal8Bit(qgetenv("USERNAME"));
|
||||||
|
#endif
|
||||||
|
#endif
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -99,11 +147,11 @@ void SingleCoreApplicationPrivate::genBlockServerName() {
|
|||||||
appData.addData(SingleCoreApplication::app_t::organizationName().toUtf8());
|
appData.addData(SingleCoreApplication::app_t::organizationName().toUtf8());
|
||||||
appData.addData(SingleCoreApplication::app_t::organizationDomain().toUtf8());
|
appData.addData(SingleCoreApplication::app_t::organizationDomain().toUtf8());
|
||||||
|
|
||||||
if (!(options & SingleCoreApplication::Mode::ExcludeAppVersion)) {
|
if (!(options_ & SingleCoreApplication::Mode::ExcludeAppVersion)) {
|
||||||
appData.addData(SingleCoreApplication::app_t::applicationVersion().toUtf8());
|
appData.addData(SingleCoreApplication::app_t::applicationVersion().toUtf8());
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!(options & SingleCoreApplication::Mode::ExcludeAppPath)) {
|
if (!(options_ & SingleCoreApplication::Mode::ExcludeAppPath)) {
|
||||||
#ifdef Q_OS_WIN
|
#ifdef Q_OS_WIN
|
||||||
appData.addData(SingleCoreApplication::app_t::applicationFilePath().toLower().toUtf8());
|
appData.addData(SingleCoreApplication::app_t::applicationFilePath().toLower().toUtf8());
|
||||||
#else
|
#else
|
||||||
@@ -112,42 +160,22 @@ void SingleCoreApplicationPrivate::genBlockServerName() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// User level block requires a user specific data in the hash
|
// User level block requires a user specific data in the hash
|
||||||
if (options & SingleCoreApplication::Mode::User) {
|
if (options_ & SingleCoreApplication::Mode::User) {
|
||||||
#ifdef Q_OS_UNIX
|
appData.addData(getUsername().toUtf8());
|
||||||
QByteArray username;
|
|
||||||
#if defined(HAVE_GETEUID) && defined(HAVE_GETPWUID)
|
|
||||||
struct passwd *pw = getpwuid(geteuid());
|
|
||||||
if (pw) {
|
|
||||||
username = pw->pw_name;
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
if (username.isEmpty()) username = qgetenv("USER");
|
|
||||||
appData.addData(username);
|
|
||||||
#endif
|
|
||||||
#ifdef Q_OS_WIN
|
|
||||||
wchar_t username [ UNLEN + 1 ];
|
|
||||||
// Specifies size of the buffer on input
|
|
||||||
DWORD usernameLength = UNLEN + 1;
|
|
||||||
if (GetUserNameW(username, &usernameLength)) {
|
|
||||||
appData.addData(QString::fromWCharArray(username).toUtf8());
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
appData.addData(qgetenv("USERNAME"));
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Replace the backslash in RFC 2045 Base64 [a-zA-Z0-9+/=] to comply with server naming requirements.
|
// Replace the backslash in RFC 2045 Base64 [a-zA-Z0-9+/=] to comply with server naming requirements.
|
||||||
blockServerName = appData.result().toBase64().replace("/", "_");
|
blockServerName_ = appData.result().toBase64().replace("/", "_");
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void SingleCoreApplicationPrivate::initializeMemoryBlock() {
|
void SingleCoreApplicationPrivate::initializeMemoryBlock() {
|
||||||
|
|
||||||
InstancesInfo* inst = static_cast<InstancesInfo*>(memory->data());
|
InstancesInfo *inst = static_cast<InstancesInfo*>(memory_->data());
|
||||||
inst->primary = false;
|
inst->primary = false;
|
||||||
inst->secondary = 0;
|
inst->secondary = 0;
|
||||||
inst->primaryPid = -1;
|
inst->primaryPid = -1;
|
||||||
|
inst->primaryUser[0] = '\0';
|
||||||
inst->checksum = blockChecksum();
|
inst->checksum = blockChecksum();
|
||||||
|
|
||||||
}
|
}
|
||||||
@@ -156,133 +184,161 @@ void SingleCoreApplicationPrivate::startPrimary() {
|
|||||||
|
|
||||||
Q_Q(SingleCoreApplication);
|
Q_Q(SingleCoreApplication);
|
||||||
|
|
||||||
// Successful creation means that no main process exists
|
|
||||||
// So we start a QLocalServer to listen for connections
|
|
||||||
QLocalServer::removeServer(blockServerName);
|
|
||||||
server = new QLocalServer();
|
|
||||||
|
|
||||||
// Restrict access to the socket according to the
|
|
||||||
// SingleCoreApplication::Mode::User flag on User level or no restrictions
|
|
||||||
if (options & SingleCoreApplication::Mode::User) {
|
|
||||||
server->setSocketOptions(QLocalServer::UserAccessOption);
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
server->setSocketOptions(QLocalServer::WorldAccessOption);
|
|
||||||
}
|
|
||||||
|
|
||||||
server->listen(blockServerName);
|
|
||||||
QObject::connect(server, &QLocalServer::newConnection, this, &SingleCoreApplicationPrivate::slotConnectionEstablished);
|
|
||||||
|
|
||||||
// Reset the number of connections
|
// Reset the number of connections
|
||||||
InstancesInfo* inst = static_cast <InstancesInfo*>(memory->data());
|
InstancesInfo *inst = static_cast<InstancesInfo*>(memory_->data());
|
||||||
|
|
||||||
inst->primary = true;
|
inst->primary = true;
|
||||||
inst->primaryPid = q->applicationPid();
|
inst->primaryPid = q->applicationPid();
|
||||||
|
qstrncpy(inst->primaryUser, getUsername().toUtf8().data(), sizeof(inst->primaryUser));
|
||||||
inst->checksum = blockChecksum();
|
inst->checksum = blockChecksum();
|
||||||
|
instanceNumber_ = 0;
|
||||||
|
// Successful creation means that no main process exists
|
||||||
|
// So we start a QLocalServer to listen for connections
|
||||||
|
QLocalServer::removeServer(blockServerName_);
|
||||||
|
server_ = new QLocalServer();
|
||||||
|
|
||||||
instanceNumber = 0;
|
// Restrict access to the socket according to the SingleCoreApplication::Mode::User flag on User level or no restrictions
|
||||||
|
if (options_ & SingleCoreApplication::Mode::User) {
|
||||||
|
server_->setSocketOptions(QLocalServer::UserAccessOption);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
server_->setSocketOptions(QLocalServer::WorldAccessOption);
|
||||||
|
}
|
||||||
|
|
||||||
|
server_->listen(blockServerName_);
|
||||||
|
QObject::connect(server_, &QLocalServer::newConnection, this, &SingleCoreApplicationPrivate::slotConnectionEstablished);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void SingleCoreApplicationPrivate::startSecondary() {}
|
void SingleCoreApplicationPrivate::startSecondary() {
|
||||||
|
|
||||||
void SingleCoreApplicationPrivate::connectToPrimary(const int msecs, const ConnectionType connectionType) {
|
InstancesInfo *inst = static_cast<InstancesInfo*>(memory_->data());
|
||||||
|
|
||||||
|
inst->secondary += 1;
|
||||||
|
inst->checksum = blockChecksum();
|
||||||
|
instanceNumber_ = inst->secondary;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
bool SingleCoreApplicationPrivate::connectToPrimary(int timeout, ConnectionType connectionType) {
|
||||||
|
|
||||||
|
QElapsedTimer time;
|
||||||
|
time.start();
|
||||||
|
|
||||||
// Connect to the Local Server of the Primary Instance if not already connected.
|
// Connect to the Local Server of the Primary Instance if not already connected.
|
||||||
if (socket == nullptr) {
|
if (socket_ == nullptr) {
|
||||||
socket = new QLocalSocket();
|
socket_ = new QLocalSocket();
|
||||||
}
|
}
|
||||||
|
|
||||||
// If already connected - we are done;
|
if (socket_->state() == QLocalSocket::ConnectedState) return true;
|
||||||
if (socket->state() == QLocalSocket::ConnectedState)
|
|
||||||
return;
|
|
||||||
|
|
||||||
// If not connect
|
if (socket_->state() != QLocalSocket::ConnectedState) {
|
||||||
if (socket->state() == QLocalSocket::UnconnectedState ||
|
|
||||||
socket->state() == QLocalSocket::ClosingState) {
|
|
||||||
socket->connectToServer(blockServerName);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Wait for being connected
|
forever {
|
||||||
if (socket->state() == QLocalSocket::ConnectingState) {
|
randomSleep();
|
||||||
socket->waitForConnected(msecs);
|
|
||||||
|
if (socket_->state() != QLocalSocket::ConnectingState)
|
||||||
|
socket_->connectToServer(blockServerName_);
|
||||||
|
|
||||||
|
if (socket_->state() == QLocalSocket::ConnectingState) {
|
||||||
|
socket_->waitForConnected(timeout - time.elapsed());
|
||||||
|
}
|
||||||
|
|
||||||
|
// If connected break out of the loop
|
||||||
|
if (socket_->state() == QLocalSocket::ConnectedState) break;
|
||||||
|
|
||||||
|
// If elapsed time since start is longer than the method timeout return
|
||||||
|
if (time.elapsed() >= timeout) return false;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Initialisation message according to the SingleCoreApplication protocol
|
// Initialisation message according to the SingleCoreApplication protocol
|
||||||
if (socket->state() == QLocalSocket::ConnectedState) {
|
QByteArray initMsg;
|
||||||
// Notify the parent that a new instance had been started;
|
QDataStream writeStream(&initMsg, QIODevice::WriteOnly);
|
||||||
QByteArray initMsg;
|
writeStream.setVersion(QDataStream::Qt_5_8);
|
||||||
QDataStream writeStream(&initMsg, QIODevice::WriteOnly);
|
|
||||||
|
|
||||||
writeStream.setVersion(QDataStream::Qt_5_6);
|
writeStream << blockServerName_.toLatin1();
|
||||||
|
writeStream << static_cast<quint8>(connectionType);
|
||||||
|
writeStream << instanceNumber_;
|
||||||
|
|
||||||
writeStream << blockServerName.toLatin1();
|
#if QT_VERSION >= QT_VERSION_CHECK(6, 0, 0)
|
||||||
writeStream << static_cast<quint8>(connectionType);
|
quint16 checksum = qChecksum(QByteArray(initMsg, static_cast<quint32>(initMsg.length())));
|
||||||
writeStream << instanceNumber;
|
#else
|
||||||
quint16 checksum = qChecksum(initMsg.constData(), static_cast<quint32>(initMsg.length()));
|
quint16 checksum = qChecksum(initMsg.constData(), static_cast<quint32>(initMsg.length()));
|
||||||
writeStream << checksum;
|
#endif
|
||||||
|
|
||||||
// The header indicates the message length that follows
|
writeStream << checksum;
|
||||||
QByteArray header;
|
|
||||||
QDataStream headerStream(&header, QIODevice::WriteOnly);
|
|
||||||
|
|
||||||
headerStream.setVersion(QDataStream::Qt_5_6);
|
// The header indicates the message length that follows
|
||||||
|
QByteArray header;
|
||||||
|
QDataStream headerStream(&header, QIODevice::WriteOnly);
|
||||||
|
headerStream.setVersion(QDataStream::Qt_5_8);
|
||||||
|
headerStream << static_cast<quint64>(initMsg.length());
|
||||||
|
|
||||||
headerStream << static_cast <quint64>(initMsg.length());
|
socket_->write(header);
|
||||||
|
socket_->write(initMsg);
|
||||||
|
bool result = socket_->waitForBytesWritten(timeout - time.elapsed());
|
||||||
|
socket_->flush();
|
||||||
|
|
||||||
socket->write(header);
|
return result;
|
||||||
socket->write(initMsg);
|
|
||||||
socket->flush();
|
|
||||||
socket->waitForBytesWritten(msecs);
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
quint16 SingleCoreApplicationPrivate::blockChecksum() {
|
quint16 SingleCoreApplicationPrivate::blockChecksum() {
|
||||||
|
|
||||||
return qChecksum(static_cast <const char*> (memory->data()), offsetof(InstancesInfo, checksum));
|
#if QT_VERSION >= QT_VERSION_CHECK(6, 0, 0)
|
||||||
|
quint16 checksum = qChecksum(QByteArray(static_cast<const char*>(memory_->constData()), offsetof(InstancesInfo, checksum)));
|
||||||
|
#else
|
||||||
|
quint16 checksum = qChecksum(static_cast<const char*>(memory_->constData()), offsetof(InstancesInfo, checksum));
|
||||||
|
#endif
|
||||||
|
|
||||||
|
return checksum;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
qint64 SingleCoreApplicationPrivate::primaryPid() {
|
qint64 SingleCoreApplicationPrivate::primaryPid() {
|
||||||
|
|
||||||
qint64 pid;
|
memory_->lock();
|
||||||
|
InstancesInfo *inst = static_cast<InstancesInfo*>(memory_->data());
|
||||||
memory->lock();
|
qint64 pid = inst->primaryPid;
|
||||||
InstancesInfo* inst = static_cast<InstancesInfo*>(memory->data());
|
memory_->unlock();
|
||||||
pid = inst->primaryPid;
|
|
||||||
memory->unlock();
|
|
||||||
|
|
||||||
return pid;
|
return pid;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
QString SingleCoreApplicationPrivate::primaryUser() {
|
||||||
|
|
||||||
|
memory_->lock();
|
||||||
|
InstancesInfo *inst = static_cast<InstancesInfo*>(memory_->data());
|
||||||
|
QByteArray username = inst->primaryUser;
|
||||||
|
memory_->unlock();
|
||||||
|
|
||||||
|
return QString::fromUtf8(username);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @brief Executed when a connection has been made to the LocalServer
|
* @brief Executed when a connection has been made to the LocalServer
|
||||||
*/
|
*/
|
||||||
void SingleCoreApplicationPrivate::slotConnectionEstablished() {
|
void SingleCoreApplicationPrivate::slotConnectionEstablished() {
|
||||||
|
|
||||||
QLocalSocket *nextConnSocket = server->nextPendingConnection();
|
QLocalSocket *nextConnSocket = server_->nextPendingConnection();
|
||||||
connectionMap.insert(nextConnSocket, ConnectionInfo());
|
connectionMap_.insert(nextConnSocket, ConnectionInfo());
|
||||||
|
|
||||||
QObject::connect(nextConnSocket, &QLocalSocket::aboutToClose,
|
QObject::connect(nextConnSocket, &QLocalSocket::aboutToClose, [nextConnSocket, this]() {
|
||||||
[nextConnSocket, this]() {
|
auto &info = connectionMap_[nextConnSocket];
|
||||||
auto &info = connectionMap[nextConnSocket];
|
Q_EMIT this->slotClientConnectionClosed(nextConnSocket, info.instanceId);
|
||||||
Q_EMIT this->slotClientConnectionClosed(nextConnSocket, info.instanceId);
|
});
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
QObject::connect(nextConnSocket, &QLocalSocket::disconnected,
|
QObject::connect(nextConnSocket, &QLocalSocket::disconnected, [nextConnSocket, this]() {
|
||||||
[nextConnSocket, this](){
|
connectionMap_.remove(nextConnSocket);
|
||||||
connectionMap.remove(nextConnSocket);
|
nextConnSocket->deleteLater();
|
||||||
nextConnSocket->deleteLater();
|
});
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
QObject::connect(nextConnSocket, &QLocalSocket::readyRead,
|
QObject::connect(nextConnSocket, &QLocalSocket::readyRead, [nextConnSocket, this]() {
|
||||||
[nextConnSocket, this]() {
|
auto &info = connectionMap_[nextConnSocket];
|
||||||
auto &info = connectionMap[nextConnSocket];
|
switch (info.stage) {
|
||||||
switch(info.stage) {
|
|
||||||
case StageHeader:
|
case StageHeader:
|
||||||
readInitMessageHeader(nextConnSocket);
|
readInitMessageHeader(nextConnSocket);
|
||||||
break;
|
break;
|
||||||
@@ -294,34 +350,32 @@ void SingleCoreApplicationPrivate::slotConnectionEstablished() {
|
|||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
break;
|
break;
|
||||||
};
|
};
|
||||||
}
|
});
|
||||||
);
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void SingleCoreApplicationPrivate::readInitMessageHeader(QLocalSocket *sock) {
|
void SingleCoreApplicationPrivate::readInitMessageHeader(QLocalSocket *sock) {
|
||||||
|
|
||||||
if (!connectionMap.contains(sock)) {
|
if (!connectionMap_.contains(sock)) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (sock->bytesAvailable() < (qint64)sizeof(quint64)) {
|
if (sock->bytesAvailable() < static_cast<qint64>(sizeof(quint64))) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
QDataStream headerStream(sock);
|
QDataStream headerStream(sock);
|
||||||
|
headerStream.setVersion(QDataStream::Qt_5_8);
|
||||||
headerStream.setVersion(QDataStream::Qt_5_6);
|
|
||||||
|
|
||||||
// Read the header to know the message length
|
// Read the header to know the message length
|
||||||
quint64 msgLen = 0;
|
quint64 msgLen = 0;
|
||||||
headerStream >> msgLen;
|
headerStream >> msgLen;
|
||||||
ConnectionInfo &info = connectionMap[sock];
|
ConnectionInfo &info = connectionMap_[sock];
|
||||||
info.stage = StageBody;
|
info.stage = StageBody;
|
||||||
info.msgLen = msgLen;
|
info.msgLen = msgLen;
|
||||||
|
|
||||||
if (sock->bytesAvailable() >= (qint64) msgLen) {
|
if (sock->bytesAvailable() >= static_cast<qint64>(msgLen)) {
|
||||||
readInitMessageBody(sock);
|
readInitMessageBody(sock);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -331,20 +385,19 @@ void SingleCoreApplicationPrivate::readInitMessageBody(QLocalSocket *sock) {
|
|||||||
|
|
||||||
Q_Q(SingleCoreApplication);
|
Q_Q(SingleCoreApplication);
|
||||||
|
|
||||||
if (!connectionMap.contains(sock)) {
|
if (!connectionMap_.contains(sock)) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
ConnectionInfo &info = connectionMap[sock];
|
ConnectionInfo &info = connectionMap_[sock];
|
||||||
if (sock->bytesAvailable() < (qint64)info.msgLen) {
|
if (sock->bytesAvailable() < static_cast<qint64>(info.msgLen)) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Read the message body
|
// Read the message body
|
||||||
QByteArray msgBytes = sock->read(info.msgLen);
|
QByteArray msgBytes = sock->read(info.msgLen);
|
||||||
QDataStream readStream(msgBytes);
|
QDataStream readStream(msgBytes);
|
||||||
|
readStream.setVersion(QDataStream::Qt_5_8);
|
||||||
readStream.setVersion(QDataStream::Qt_5_6);
|
|
||||||
|
|
||||||
// server name
|
// server name
|
||||||
QByteArray latin1Name;
|
QByteArray latin1Name;
|
||||||
@@ -354,7 +407,7 @@ void SingleCoreApplicationPrivate::readInitMessageBody(QLocalSocket *sock) {
|
|||||||
ConnectionType connectionType = InvalidConnection;
|
ConnectionType connectionType = InvalidConnection;
|
||||||
quint8 connTypeVal = InvalidConnection;
|
quint8 connTypeVal = InvalidConnection;
|
||||||
readStream >> connTypeVal;
|
readStream >> connTypeVal;
|
||||||
connectionType = static_cast <ConnectionType>(connTypeVal);
|
connectionType = static_cast<ConnectionType>(connTypeVal);
|
||||||
|
|
||||||
// instance id
|
// instance id
|
||||||
quint32 instanceId = 0;
|
quint32 instanceId = 0;
|
||||||
@@ -364,9 +417,13 @@ void SingleCoreApplicationPrivate::readInitMessageBody(QLocalSocket *sock) {
|
|||||||
quint16 msgChecksum = 0;
|
quint16 msgChecksum = 0;
|
||||||
readStream >> msgChecksum;
|
readStream >> msgChecksum;
|
||||||
|
|
||||||
|
#if QT_VERSION >= QT_VERSION_CHECK(6, 0, 0)
|
||||||
|
const quint16 actualChecksum = qChecksum(QByteArray(msgBytes, static_cast<quint32>(msgBytes.length() - sizeof(quint16))));
|
||||||
|
#else
|
||||||
const quint16 actualChecksum = qChecksum(msgBytes.constData(), static_cast<quint32>(msgBytes.length() - sizeof(quint16)));
|
const quint16 actualChecksum = qChecksum(msgBytes.constData(), static_cast<quint32>(msgBytes.length() - sizeof(quint16)));
|
||||||
|
#endif
|
||||||
|
|
||||||
bool isValid = readStream.status() == QDataStream::Ok && QLatin1String(latin1Name) == blockServerName && msgChecksum == actualChecksum;
|
bool isValid = readStream.status() == QDataStream::Ok && QLatin1String(latin1Name) == blockServerName_ && msgChecksum == actualChecksum;
|
||||||
|
|
||||||
if (!isValid) {
|
if (!isValid) {
|
||||||
sock->close();
|
sock->close();
|
||||||
@@ -376,7 +433,7 @@ void SingleCoreApplicationPrivate::readInitMessageBody(QLocalSocket *sock) {
|
|||||||
info.instanceId = instanceId;
|
info.instanceId = instanceId;
|
||||||
info.stage = StageConnected;
|
info.stage = StageConnected;
|
||||||
|
|
||||||
if (connectionType == NewInstance || (connectionType == SecondaryInstance && options & SingleCoreApplication::Mode::SecondaryNotification)) {
|
if (connectionType == NewInstance || (connectionType == SecondaryInstance && options_ & SingleCoreApplication::Mode::SecondaryNotification)) {
|
||||||
Q_EMIT q->instanceStarted();
|
Q_EMIT q->instanceStarted();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -395,7 +452,19 @@ void SingleCoreApplicationPrivate::slotDataAvailable(QLocalSocket *dataSocket, c
|
|||||||
|
|
||||||
void SingleCoreApplicationPrivate::slotClientConnectionClosed(QLocalSocket *closedSocket, const quint32 instanceId) {
|
void SingleCoreApplicationPrivate::slotClientConnectionClosed(QLocalSocket *closedSocket, const quint32 instanceId) {
|
||||||
|
|
||||||
if (closedSocket->bytesAvailable() > 0)
|
if (closedSocket->bytesAvailable() > 0) {
|
||||||
Q_EMIT slotDataAvailable(closedSocket, instanceId);
|
Q_EMIT slotDataAvailable(closedSocket, instanceId);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
void SingleCoreApplicationPrivate::randomSleep() {
|
||||||
|
|
||||||
|
#if QT_VERSION >= QT_VERSION_CHECK(5, 10, 0)
|
||||||
|
QThread::msleep(QRandomGenerator::global()->bounded(8u, 18u));
|
||||||
|
#else
|
||||||
|
qsrand(QDateTime::currentMSecsSinceEpoch() % std::numeric_limits<uint>::max());
|
||||||
|
QThread::msleep(8 + static_cast<unsigned long>(static_cast<float>(qrand()) / RAND_MAX * 10));
|
||||||
|
#endif
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
// The MIT License (MIT)
|
// The MIT License (MIT)
|
||||||
//
|
//
|
||||||
// Copyright (c) Itay Grudev 2015 - 2016
|
// Copyright (c) Itay Grudev 2015 - 2020
|
||||||
//
|
//
|
||||||
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
// of this software and associated documentation files (the "Software"), to deal
|
// of this software and associated documentation files (the "Software"), to deal
|
||||||
@@ -36,6 +36,7 @@
|
|||||||
|
|
||||||
#include <QtGlobal>
|
#include <QtGlobal>
|
||||||
#include <QObject>
|
#include <QObject>
|
||||||
|
#include <QString>
|
||||||
#include <QMap>
|
#include <QMap>
|
||||||
|
|
||||||
#include "singlecoreapplication.h"
|
#include "singlecoreapplication.h"
|
||||||
@@ -48,6 +49,7 @@ struct InstancesInfo {
|
|||||||
bool primary;
|
bool primary;
|
||||||
quint32 secondary;
|
quint32 secondary;
|
||||||
qint64 primaryPid;
|
qint64 primaryPid;
|
||||||
|
char primaryUser[128];
|
||||||
quint16 checksum;
|
quint16 checksum;
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -60,6 +62,7 @@ struct ConnectionInfo {
|
|||||||
|
|
||||||
class SingleCoreApplicationPrivate : public QObject {
|
class SingleCoreApplicationPrivate : public QObject {
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
|
|
||||||
public:
|
public:
|
||||||
enum ConnectionType : quint8 {
|
enum ConnectionType : quint8 {
|
||||||
InvalidConnection = 0,
|
InvalidConnection = 0,
|
||||||
@@ -74,27 +77,30 @@ class SingleCoreApplicationPrivate : public QObject {
|
|||||||
};
|
};
|
||||||
Q_DECLARE_PUBLIC(SingleCoreApplication)
|
Q_DECLARE_PUBLIC(SingleCoreApplication)
|
||||||
|
|
||||||
SingleCoreApplicationPrivate(SingleCoreApplication *_q_ptr);
|
explicit SingleCoreApplicationPrivate(SingleCoreApplication *ptr);
|
||||||
~SingleCoreApplicationPrivate();
|
~SingleCoreApplicationPrivate() override;
|
||||||
|
|
||||||
|
QString getUsername();
|
||||||
void genBlockServerName();
|
void genBlockServerName();
|
||||||
void initializeMemoryBlock();
|
void initializeMemoryBlock();
|
||||||
void startPrimary();
|
void startPrimary();
|
||||||
void startSecondary();
|
void startSecondary();
|
||||||
void connectToPrimary(const int msecs, const ConnectionType connectionType);
|
bool connectToPrimary(const int msecs, const ConnectionType connectionType);
|
||||||
quint16 blockChecksum();
|
quint16 blockChecksum();
|
||||||
qint64 primaryPid();
|
qint64 primaryPid();
|
||||||
|
QString primaryUser();
|
||||||
void readInitMessageHeader(QLocalSocket *socket);
|
void readInitMessageHeader(QLocalSocket *socket);
|
||||||
void readInitMessageBody(QLocalSocket *socket);
|
void readInitMessageBody(QLocalSocket *socket);
|
||||||
|
void randomSleep();
|
||||||
|
|
||||||
SingleCoreApplication *q_ptr;
|
SingleCoreApplication *q_ptr;
|
||||||
QSharedMemory *memory;
|
QSharedMemory *memory_;
|
||||||
QLocalSocket *socket;
|
QLocalSocket *socket_;
|
||||||
QLocalServer *server;
|
QLocalServer *server_;
|
||||||
quint32 instanceNumber;
|
quint32 instanceNumber_;
|
||||||
QString blockServerName;
|
QString blockServerName_;
|
||||||
SingleCoreApplication::Options options;
|
SingleCoreApplication::Options options_;
|
||||||
QMap<QLocalSocket*, ConnectionInfo> connectionMap;
|
QMap<QLocalSocket*, ConnectionInfo> connectionMap_;
|
||||||
|
|
||||||
public slots:
|
public slots:
|
||||||
void slotConnectionEstablished();
|
void slotConnectionEstablished();
|
||||||
|
|||||||
136
3rdparty/taglib/CMakeLists.txt
vendored
136
3rdparty/taglib/CMakeLists.txt
vendored
@@ -8,70 +8,24 @@ math(EXPR TAGLIB_SOVERSION_MAJOR "${TAGLIB_SOVERSION_CURRENT} - ${TAGLIB_SOVERSI
|
|||||||
math(EXPR TAGLIB_SOVERSION_MINOR "${TAGLIB_SOVERSION_AGE}")
|
math(EXPR TAGLIB_SOVERSION_MINOR "${TAGLIB_SOVERSION_AGE}")
|
||||||
math(EXPR TAGLIB_SOVERSION_PATCH "${TAGLIB_SOVERSION_REVISION}")
|
math(EXPR TAGLIB_SOVERSION_PATCH "${TAGLIB_SOVERSION_REVISION}")
|
||||||
|
|
||||||
include(TestBigEndian)
|
|
||||||
test_big_endian(IS_BIG_ENDIAN)
|
|
||||||
|
|
||||||
if(NOT IS_BIG_ENDIAN)
|
|
||||||
add_definitions(-DSYSTEM_BYTEORDER=1)
|
|
||||||
else()
|
|
||||||
add_definitions(-DSYSTEM_BYTEORDER=2)
|
|
||||||
endif()
|
|
||||||
|
|
||||||
include(ConfigureChecks.cmake)
|
include(ConfigureChecks.cmake)
|
||||||
configure_file(config.h.cmake "${CMAKE_CURRENT_BINARY_DIR}/config.h")
|
set(TESTS_DIR "${CMAKE_SOURCE_DIR}/tests/taglib/")
|
||||||
configure_file(taglib_config.h.cmake ${CMAKE_CURRENT_BINARY_DIR}/taglib_config.h)
|
configure_file(taglib-config.h.cmake "${CMAKE_CURRENT_BINARY_DIR}/taglib-config.h")
|
||||||
|
|
||||||
add_definitions(-DHAVE_CONFIG_H)
|
add_definitions(-DHAVE_CONFIG_H)
|
||||||
add_definitions(-DTAGLIB_STATIC)
|
add_definitions(-DTAGLIB_STATIC)
|
||||||
|
|
||||||
set(CMAKE_INCLUDE_CURRENT_DIR ON)
|
|
||||||
include_directories(
|
|
||||||
${CMAKE_CURRENT_SOURCE_DIR}/toolkit
|
|
||||||
${CMAKE_CURRENT_SOURCE_DIR}/asf
|
|
||||||
${CMAKE_CURRENT_SOURCE_DIR}/mpeg
|
|
||||||
${CMAKE_CURRENT_SOURCE_DIR}/ogg
|
|
||||||
${CMAKE_CURRENT_SOURCE_DIR}/ogg/flac
|
|
||||||
${CMAKE_CURRENT_SOURCE_DIR}/flac
|
|
||||||
${CMAKE_CURRENT_SOURCE_DIR}/mpc
|
|
||||||
${CMAKE_CURRENT_SOURCE_DIR}/mp4
|
|
||||||
${CMAKE_CURRENT_SOURCE_DIR}/ogg/vorbis
|
|
||||||
${CMAKE_CURRENT_SOURCE_DIR}/ogg/speex
|
|
||||||
${CMAKE_CURRENT_SOURCE_DIR}/ogg/opus
|
|
||||||
${CMAKE_CURRENT_SOURCE_DIR}/mpeg/id3v2
|
|
||||||
${CMAKE_CURRENT_SOURCE_DIR}/mpeg/id3v2/frames
|
|
||||||
${CMAKE_CURRENT_SOURCE_DIR}/mpeg/id3v1
|
|
||||||
${CMAKE_CURRENT_SOURCE_DIR}/ape
|
|
||||||
${CMAKE_CURRENT_SOURCE_DIR}/wavpack
|
|
||||||
${CMAKE_CURRENT_SOURCE_DIR}/trueaudio
|
|
||||||
${CMAKE_CURRENT_SOURCE_DIR}/riff
|
|
||||||
${CMAKE_CURRENT_SOURCE_DIR}/riff/aiff
|
|
||||||
${CMAKE_CURRENT_SOURCE_DIR}/riff/wav
|
|
||||||
${CMAKE_CURRENT_SOURCE_DIR}/mod
|
|
||||||
${CMAKE_CURRENT_SOURCE_DIR}/s3m
|
|
||||||
${CMAKE_CURRENT_SOURCE_DIR}/it
|
|
||||||
${CMAKE_CURRENT_SOURCE_DIR}/xm
|
|
||||||
${CMAKE_CURRENT_SOURCE_DIR}/dsf
|
|
||||||
${CMAKE_CURRENT_SOURCE_DIR}/dsdiff
|
|
||||||
${CMAKE_SOURCE_DIR}/3rdparty
|
|
||||||
)
|
|
||||||
|
|
||||||
if(ZLIB_FOUND)
|
|
||||||
include_directories(${ZLIB_INCLUDE_DIR})
|
|
||||||
elseif(HAVE_ZLIB_SOURCE)
|
|
||||||
include_directories(${ZLIB_SOURCE})
|
|
||||||
endif()
|
|
||||||
|
|
||||||
set(tag_HDRS
|
set(tag_HDRS
|
||||||
tag.h
|
tag.h
|
||||||
fileref.h
|
fileref.h
|
||||||
audioproperties.h
|
audioproperties.h
|
||||||
taglib_export.h
|
taglib_export.h
|
||||||
${CMAKE_CURRENT_BINARY_DIR}/taglib_config.h
|
|
||||||
toolkit/taglib.h
|
toolkit/taglib.h
|
||||||
toolkit/tstring.h
|
toolkit/tstring.h
|
||||||
toolkit/tlist.h
|
toolkit/tlist.h
|
||||||
toolkit/tlist.tcc
|
toolkit/tlist.tcc
|
||||||
toolkit/tstringlist.h
|
toolkit/tstringlist.h
|
||||||
|
toolkit/tstringhandler.h
|
||||||
toolkit/tbytevector.h
|
toolkit/tbytevector.h
|
||||||
toolkit/tbytevectorlist.h
|
toolkit/tbytevectorlist.h
|
||||||
toolkit/tbytevectorstream.h
|
toolkit/tbytevectorstream.h
|
||||||
@@ -80,6 +34,8 @@ set(tag_HDRS
|
|||||||
toolkit/tfilestream.h
|
toolkit/tfilestream.h
|
||||||
toolkit/tmap.h
|
toolkit/tmap.h
|
||||||
toolkit/tmap.tcc
|
toolkit/tmap.tcc
|
||||||
|
toolkit/tpicture.h
|
||||||
|
toolkit/tpicturemap.h
|
||||||
toolkit/tpropertymap.h
|
toolkit/tpropertymap.h
|
||||||
toolkit/trefcounter.h
|
toolkit/trefcounter.h
|
||||||
toolkit/tdebuglistener.h
|
toolkit/tdebuglistener.h
|
||||||
@@ -339,8 +295,10 @@ set(dsdiff_SRCS
|
|||||||
)
|
)
|
||||||
|
|
||||||
set(toolkit_SRCS
|
set(toolkit_SRCS
|
||||||
|
toolkit/taglib.cpp
|
||||||
toolkit/tstring.cpp
|
toolkit/tstring.cpp
|
||||||
toolkit/tstringlist.cpp
|
toolkit/tstringlist.cpp
|
||||||
|
toolkit/tstringhandler.cpp
|
||||||
toolkit/tbytevector.cpp
|
toolkit/tbytevector.cpp
|
||||||
toolkit/tbytevectorlist.cpp
|
toolkit/tbytevectorlist.cpp
|
||||||
toolkit/tbytevectorstream.cpp
|
toolkit/tbytevectorstream.cpp
|
||||||
@@ -348,28 +306,40 @@ set(toolkit_SRCS
|
|||||||
toolkit/tfile.cpp
|
toolkit/tfile.cpp
|
||||||
toolkit/tfilestream.cpp
|
toolkit/tfilestream.cpp
|
||||||
toolkit/tdebug.cpp
|
toolkit/tdebug.cpp
|
||||||
|
toolkit/tpicture.cpp
|
||||||
|
toolkit/tpicturemap.cpp
|
||||||
toolkit/tpropertymap.cpp
|
toolkit/tpropertymap.cpp
|
||||||
toolkit/trefcounter.cpp
|
toolkit/trefcounter.cpp
|
||||||
toolkit/tdebuglistener.cpp
|
toolkit/tdebuglistener.cpp
|
||||||
toolkit/tzlib.cpp
|
toolkit/tzlib.cpp
|
||||||
)
|
)
|
||||||
|
|
||||||
if(HAVE_ZLIB_SOURCE)
|
|
||||||
set(zlib_SRCS
|
|
||||||
${ZLIB_SOURCE}/adler32.c
|
|
||||||
${ZLIB_SOURCE}/crc32.c
|
|
||||||
${ZLIB_SOURCE}/inffast.c
|
|
||||||
${ZLIB_SOURCE}/inflate.c
|
|
||||||
${ZLIB_SOURCE}/inftrees.c
|
|
||||||
${ZLIB_SOURCE}/zutil.c
|
|
||||||
)
|
|
||||||
endif()
|
|
||||||
|
|
||||||
set(tag_LIB_SRCS
|
set(tag_LIB_SRCS
|
||||||
${mpeg_SRCS} ${id3v1_SRCS} ${id3v2_SRCS} ${frames_SRCS} ${ogg_SRCS}
|
${mpeg_SRCS}
|
||||||
${vorbis_SRCS} ${oggflacs_SRCS} ${mpc_SRCS} ${ape_SRCS} ${toolkit_SRCS} ${flacs_SRCS}
|
${id3v1_SRCS}
|
||||||
${wavpack_SRCS} ${speex_SRCS} ${trueaudio_SRCS} ${riff_SRCS} ${aiff_SRCS} ${wav_SRCS}
|
${id3v2_SRCS}
|
||||||
${asf_SRCS} ${mp4_SRCS} ${mod_SRCS} ${s3m_SRCS} ${it_SRCS} ${xm_SRCS} ${opus_SRCS} ${dsf_SRCS} ${dsdiff_SRCS}
|
${frames_SRCS}
|
||||||
|
${ogg_SRCS}
|
||||||
|
${vorbis_SRCS}
|
||||||
|
${oggflacs_SRCS}
|
||||||
|
${mpc_SRCS}
|
||||||
|
${ape_SRCS}
|
||||||
|
${toolkit_SRCS}
|
||||||
|
${flacs_SRCS}
|
||||||
|
${wavpack_SRCS}
|
||||||
|
${speex_SRCS}
|
||||||
|
${trueaudio_SRCS}
|
||||||
|
${riff_SRCS}
|
||||||
|
${aiff_SRCS} ${wav_SRCS}
|
||||||
|
${asf_SRCS}
|
||||||
|
${mp4_SRCS}
|
||||||
|
${mod_SRCS}
|
||||||
|
${s3m_SRCS}
|
||||||
|
${it_SRCS}
|
||||||
|
${xm_SRCS}
|
||||||
|
${opus_SRCS}
|
||||||
|
${dsf_SRCS}
|
||||||
|
${dsdiff_SRCS}
|
||||||
${zlib_SRCS}
|
${zlib_SRCS}
|
||||||
tag.cpp
|
tag.cpp
|
||||||
tagunion.cpp
|
tagunion.cpp
|
||||||
@@ -380,9 +350,40 @@ set(tag_LIB_SRCS
|
|||||||
|
|
||||||
add_library(tag STATIC ${tag_LIB_SRCS} ${tag_HDRS})
|
add_library(tag STATIC ${tag_LIB_SRCS} ${tag_HDRS})
|
||||||
|
|
||||||
if(HAVE_ZLIB AND NOT HAVE_ZLIB_SOURCE)
|
target_include_directories(tag PRIVATE
|
||||||
target_link_libraries(tag ${ZLIB_LIBRARIES})
|
${ZLIB_INCLUDE_DIR}
|
||||||
endif()
|
${CMAKE_CURRENT_BINARY_DIR}
|
||||||
|
${CMAKE_CURRENT_SOURCE_DIR}
|
||||||
|
${CMAKE_CURRENT_SOURCE_DIR}/toolkit
|
||||||
|
${CMAKE_CURRENT_SOURCE_DIR}/asf
|
||||||
|
${CMAKE_CURRENT_SOURCE_DIR}/mpeg
|
||||||
|
${CMAKE_CURRENT_SOURCE_DIR}/ogg
|
||||||
|
${CMAKE_CURRENT_SOURCE_DIR}/ogg/flac
|
||||||
|
${CMAKE_CURRENT_SOURCE_DIR}/flac
|
||||||
|
${CMAKE_CURRENT_SOURCE_DIR}/mpc
|
||||||
|
${CMAKE_CURRENT_SOURCE_DIR}/mp4
|
||||||
|
${CMAKE_CURRENT_SOURCE_DIR}/ogg/vorbis
|
||||||
|
${CMAKE_CURRENT_SOURCE_DIR}/ogg/speex
|
||||||
|
${CMAKE_CURRENT_SOURCE_DIR}/ogg/opus
|
||||||
|
${CMAKE_CURRENT_SOURCE_DIR}/mpeg/id3v2
|
||||||
|
${CMAKE_CURRENT_SOURCE_DIR}/mpeg/id3v2/frames
|
||||||
|
${CMAKE_CURRENT_SOURCE_DIR}/mpeg/id3v1
|
||||||
|
${CMAKE_CURRENT_SOURCE_DIR}/ape
|
||||||
|
${CMAKE_CURRENT_SOURCE_DIR}/wavpack
|
||||||
|
${CMAKE_CURRENT_SOURCE_DIR}/trueaudio
|
||||||
|
${CMAKE_CURRENT_SOURCE_DIR}/riff
|
||||||
|
${CMAKE_CURRENT_SOURCE_DIR}/riff/aiff
|
||||||
|
${CMAKE_CURRENT_SOURCE_DIR}/riff/wav
|
||||||
|
${CMAKE_CURRENT_SOURCE_DIR}/mod
|
||||||
|
${CMAKE_CURRENT_SOURCE_DIR}/s3m
|
||||||
|
${CMAKE_CURRENT_SOURCE_DIR}/it
|
||||||
|
${CMAKE_CURRENT_SOURCE_DIR}/xm
|
||||||
|
${CMAKE_CURRENT_SOURCE_DIR}/dsf
|
||||||
|
${CMAKE_CURRENT_SOURCE_DIR}/dsdiff
|
||||||
|
${CMAKE_SOURCE_DIR}/3rdparty
|
||||||
|
)
|
||||||
|
|
||||||
|
target_link_libraries(tag PRIVATE ${ZLIB_LIBRARIES})
|
||||||
|
|
||||||
set_target_properties(tag PROPERTIES
|
set_target_properties(tag PROPERTIES
|
||||||
VERSION ${TAGLIB_SOVERSION_MAJOR}.${TAGLIB_SOVERSION_MINOR}.${TAGLIB_SOVERSION_PATCH}
|
VERSION ${TAGLIB_SOVERSION_MAJOR}.${TAGLIB_SOVERSION_MINOR}.${TAGLIB_SOVERSION_PATCH}
|
||||||
@@ -399,4 +400,3 @@ foreach(header ${tag_HDRS})
|
|||||||
COPYONLY
|
COPYONLY
|
||||||
)
|
)
|
||||||
endforeach()
|
endforeach()
|
||||||
|
|
||||||
|
|||||||
15
3rdparty/taglib/ConfigureChecks.cmake
vendored
15
3rdparty/taglib/ConfigureChecks.cmake
vendored
@@ -1,36 +1,37 @@
|
|||||||
include(CheckLibraryExists)
|
include(CheckLibraryExists)
|
||||||
include(CheckTypeSize)
|
include(CheckTypeSize)
|
||||||
|
include(CheckCXXCompilerFlag)
|
||||||
include(CheckCXXSourceCompiles)
|
include(CheckCXXSourceCompiles)
|
||||||
|
|
||||||
# Check if the size of numeric types are suitable.
|
# Check if the size of numeric types are suitable.
|
||||||
|
|
||||||
check_type_size("short" SIZEOF_SHORT)
|
check_type_size("short" SIZEOF_SHORT)
|
||||||
if(NOT ${SIZEOF_SHORT} EQUAL 2)
|
if(NOT SIZEOF_SHORT EQUAL 2)
|
||||||
message(FATAL_ERROR "TagLib requires that short is 16-bit wide.")
|
message(FATAL_ERROR "TagLib requires that short is 16-bit wide.")
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
check_type_size("int" SIZEOF_INT)
|
check_type_size("int" SIZEOF_INT)
|
||||||
if(NOT ${SIZEOF_INT} EQUAL 4)
|
if(NOT SIZEOF_INT EQUAL 4)
|
||||||
message(FATAL_ERROR "TagLib requires that int is 32-bit wide.")
|
message(FATAL_ERROR "TagLib requires that int is 32-bit wide.")
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
check_type_size("long long" SIZEOF_LONGLONG)
|
check_type_size("long long" SIZEOF_LONGLONG)
|
||||||
if(NOT ${SIZEOF_LONGLONG} EQUAL 8)
|
if(NOT SIZEOF_LONGLONG EQUAL 8)
|
||||||
message(FATAL_ERROR "TagLib requires that long long is 64-bit wide.")
|
message(FATAL_ERROR "TagLib requires that long long is 64-bit wide.")
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
check_type_size("wchar_t" SIZEOF_WCHAR_T)
|
check_type_size("wchar_t" SIZEOF_WCHAR_T)
|
||||||
if(${SIZEOF_WCHAR_T} LESS 2)
|
if(SIZEOF_WCHAR_T LESS 2)
|
||||||
message(FATAL_ERROR "TagLib requires that wchar_t is sufficient to store a UTF-16 char.")
|
message(FATAL_ERROR "TagLib requires that wchar_t is sufficient to store a UTF-16 char.")
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
check_type_size("float" SIZEOF_FLOAT)
|
check_type_size("float" SIZEOF_FLOAT)
|
||||||
if(NOT ${SIZEOF_FLOAT} EQUAL 4)
|
if(NOT SIZEOF_FLOAT EQUAL 4)
|
||||||
message(FATAL_ERROR "TagLib requires that float is 32-bit wide.")
|
message(FATAL_ERROR "TagLib requires that float is 32-bit wide.")
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
check_type_size("double" SIZEOF_DOUBLE)
|
check_type_size("double" SIZEOF_DOUBLE)
|
||||||
if(NOT ${SIZEOF_DOUBLE} EQUAL 8)
|
if(NOT SIZEOF_DOUBLE EQUAL 8)
|
||||||
message(FATAL_ERROR "TagLib requires that double is 64-bit wide.")
|
message(FATAL_ERROR "TagLib requires that double is 64-bit wide.")
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
@@ -211,5 +212,5 @@ endif()
|
|||||||
|
|
||||||
# Detect WinRT mode
|
# Detect WinRT mode
|
||||||
if(CMAKE_SYSTEM_NAME STREQUAL "WindowsStore")
|
if(CMAKE_SYSTEM_NAME STREQUAL "WindowsStore")
|
||||||
set(PLATFORM WINRT 1)
|
set(PLATFORM WINRT 1)
|
||||||
endif()
|
endif()
|
||||||
|
|||||||
196
3rdparty/taglib/ape/apefile.cpp
vendored
196
3rdparty/taglib/ape/apefile.cpp
vendored
@@ -31,14 +31,16 @@
|
|||||||
* http://www.mozilla.org/MPL/ *
|
* http://www.mozilla.org/MPL/ *
|
||||||
***************************************************************************/
|
***************************************************************************/
|
||||||
|
|
||||||
#include <tbytevector.h>
|
#include <memory>
|
||||||
#include <tstring.h>
|
|
||||||
#include <tdebug.h>
|
#include "tbytevector.h"
|
||||||
#include <tagunion.h>
|
#include "tstring.h"
|
||||||
#include <id3v1tag.h>
|
#include "tdebug.h"
|
||||||
#include <id3v2header.h>
|
#include "tagunion.h"
|
||||||
#include <tpropertymap.h>
|
#include "id3v1tag.h"
|
||||||
#include <tagutils.h>
|
#include "id3v2header.h"
|
||||||
|
#include "tpropertymap.h"
|
||||||
|
#include "tagutils.h"
|
||||||
|
|
||||||
#include "apefile.h"
|
#include "apefile.h"
|
||||||
#include "apetag.h"
|
#include "apetag.h"
|
||||||
@@ -46,125 +48,98 @@
|
|||||||
|
|
||||||
using namespace Strawberry_TagLib::TagLib;
|
using namespace Strawberry_TagLib::TagLib;
|
||||||
|
|
||||||
namespace
|
namespace {
|
||||||
{
|
enum { ApeAPEIndex = 0,
|
||||||
enum { ApeAPEIndex = 0, ApeID3v1Index = 1 };
|
ApeID3v1Index = 1 };
|
||||||
}
|
}
|
||||||
|
|
||||||
class APE::File::FilePrivate
|
class APE::File::FilePrivate {
|
||||||
{
|
public:
|
||||||
public:
|
FilePrivate() : APELocation(-1),
|
||||||
FilePrivate() :
|
APESize(0),
|
||||||
APELocation(-1),
|
ID3v1Location(-1),
|
||||||
APESize(0),
|
ID3v2Location(-1),
|
||||||
ID3v1Location(-1),
|
ID3v2Size(0) {}
|
||||||
ID3v2Header(0),
|
|
||||||
ID3v2Location(-1),
|
|
||||||
ID3v2Size(0),
|
|
||||||
properties(0) {}
|
|
||||||
|
|
||||||
~FilePrivate()
|
long long APELocation;
|
||||||
{
|
long long APESize;
|
||||||
delete ID3v2Header;
|
|
||||||
delete properties;
|
|
||||||
}
|
|
||||||
|
|
||||||
long APELocation;
|
long long ID3v1Location;
|
||||||
long APESize;
|
|
||||||
|
|
||||||
long ID3v1Location;
|
std::unique_ptr<ID3v2::Header> ID3v2Header;
|
||||||
|
long long ID3v2Location;
|
||||||
|
long long ID3v2Size;
|
||||||
|
|
||||||
ID3v2::Header *ID3v2Header;
|
DoubleTagUnion tag;
|
||||||
long ID3v2Location;
|
|
||||||
long ID3v2Size;
|
|
||||||
|
|
||||||
TagUnion tag;
|
std::unique_ptr<AudioProperties> properties;
|
||||||
|
|
||||||
Properties *properties;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
// static members
|
// static members
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
bool APE::File::isSupported(IOStream*)
|
bool APE::File::isSupported(IOStream *stream) {
|
||||||
{
|
|
||||||
// An APE file has an ID "MAC " somewhere. An ID3v2 tag may precede.
|
// An APE file has an ID "MAC " somewhere. An ID3v2 tag may precede.
|
||||||
|
|
||||||
// FIXME:
|
const ByteVector buffer = Utils::readHeader(stream, bufferSize(), true);
|
||||||
//const ByteVector buffer = Utils::readHeader(stream, bufferSize(), true);
|
return (buffer.find("MAC ") == 0);
|
||||||
//return (buffer.find("MAC ") >= 0);
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
// public members
|
// public members
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
APE::File::File(FileName file, bool readProperties, Properties::ReadStyle) :
|
APE::File::File(FileName file, bool readProperties, AudioProperties::ReadStyle) : Strawberry_TagLib::TagLib::File(file), d(new FilePrivate()) {
|
||||||
Strawberry_TagLib::TagLib::File(file),
|
|
||||||
d(new FilePrivate())
|
if (isOpen())
|
||||||
{
|
|
||||||
if(isOpen())
|
|
||||||
read(readProperties);
|
read(readProperties);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
APE::File::File(IOStream *stream, bool readProperties, Properties::ReadStyle) :
|
APE::File::File(IOStream *stream, bool readProperties, AudioProperties::ReadStyle) : Strawberry_TagLib::TagLib::File(stream), d(new FilePrivate()) {
|
||||||
Strawberry_TagLib::TagLib::File(stream),
|
|
||||||
d(new FilePrivate())
|
if (isOpen())
|
||||||
{
|
|
||||||
if(isOpen())
|
|
||||||
read(readProperties);
|
read(readProperties);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
APE::File::~File()
|
APE::File::~File() {
|
||||||
{
|
|
||||||
delete d;
|
delete d;
|
||||||
}
|
}
|
||||||
|
|
||||||
Strawberry_TagLib::TagLib::Tag *APE::File::tag() const
|
Strawberry_TagLib::TagLib::Tag *APE::File::tag() const {
|
||||||
{
|
|
||||||
return &d->tag;
|
return &d->tag;
|
||||||
}
|
}
|
||||||
|
|
||||||
PropertyMap APE::File::properties() const
|
PropertyMap APE::File::setProperties(const PropertyMap &properties) {
|
||||||
{
|
|
||||||
return d->tag.properties();
|
|
||||||
}
|
|
||||||
|
|
||||||
void APE::File::removeUnsupportedProperties(const StringList &properties)
|
if (ID3v1Tag())
|
||||||
{
|
|
||||||
d->tag.removeUnsupportedProperties(properties);
|
|
||||||
}
|
|
||||||
|
|
||||||
PropertyMap APE::File::setProperties(const PropertyMap &properties)
|
|
||||||
{
|
|
||||||
if(ID3v1Tag())
|
|
||||||
ID3v1Tag()->setProperties(properties);
|
ID3v1Tag()->setProperties(properties);
|
||||||
|
|
||||||
return APETag(true)->setProperties(properties);
|
return APETag(true)->setProperties(properties);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
APE::Properties *APE::File::audioProperties() const
|
APE::AudioProperties *APE::File::audioProperties() const {
|
||||||
{
|
return d->properties.get();
|
||||||
return d->properties;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool APE::File::save()
|
bool APE::File::save() {
|
||||||
{
|
|
||||||
if(readOnly()) {
|
if (readOnly()) {
|
||||||
debug("APE::File::save() -- File is read only.");
|
debug("APE::File::save() -- File is read only.");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Update ID3v1 tag
|
// Update ID3v1 tag
|
||||||
|
|
||||||
if(ID3v1Tag() && !ID3v1Tag()->isEmpty()) {
|
if (ID3v1Tag() && !ID3v1Tag()->isEmpty()) {
|
||||||
|
|
||||||
// ID3v1 tag is not empty. Update the old one or create a new one.
|
// ID3v1 tag is not empty. Update the old one or create a new one.
|
||||||
|
|
||||||
if(d->ID3v1Location >= 0) {
|
if (d->ID3v1Location >= 0) {
|
||||||
seek(d->ID3v1Location);
|
seek(d->ID3v1Location);
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
@@ -178,7 +153,7 @@ bool APE::File::save()
|
|||||||
|
|
||||||
// ID3v1 tag is empty. Remove the old one.
|
// ID3v1 tag is empty. Remove the old one.
|
||||||
|
|
||||||
if(d->ID3v1Location >= 0) {
|
if (d->ID3v1Location >= 0) {
|
||||||
truncate(d->ID3v1Location);
|
truncate(d->ID3v1Location);
|
||||||
d->ID3v1Location = -1;
|
d->ID3v1Location = -1;
|
||||||
}
|
}
|
||||||
@@ -186,12 +161,12 @@ bool APE::File::save()
|
|||||||
|
|
||||||
// Update APE tag
|
// Update APE tag
|
||||||
|
|
||||||
if(APETag() && !APETag()->isEmpty()) {
|
if (APETag() && !APETag()->isEmpty()) {
|
||||||
|
|
||||||
// APE tag is not empty. Update the old one or create a new one.
|
// APE tag is not empty. Update the old one or create a new one.
|
||||||
|
|
||||||
if(d->APELocation < 0) {
|
if (d->APELocation < 0) {
|
||||||
if(d->ID3v1Location >= 0)
|
if (d->ID3v1Location >= 0)
|
||||||
d->APELocation = d->ID3v1Location;
|
d->APELocation = d->ID3v1Location;
|
||||||
else
|
else
|
||||||
d->APELocation = length();
|
d->APELocation = length();
|
||||||
@@ -200,7 +175,7 @@ bool APE::File::save()
|
|||||||
const ByteVector data = APETag()->render();
|
const ByteVector data = APETag()->render();
|
||||||
insert(data, d->APELocation, d->APESize);
|
insert(data, d->APELocation, d->APESize);
|
||||||
|
|
||||||
if(d->ID3v1Location >= 0)
|
if (d->ID3v1Location >= 0)
|
||||||
d->ID3v1Location += (static_cast<long>(data.size()) - d->APESize);
|
d->ID3v1Location += (static_cast<long>(data.size()) - d->APESize);
|
||||||
|
|
||||||
d->APESize = data.size();
|
d->APESize = data.size();
|
||||||
@@ -209,10 +184,10 @@ bool APE::File::save()
|
|||||||
|
|
||||||
// APE tag is empty. Remove the old one.
|
// APE tag is empty. Remove the old one.
|
||||||
|
|
||||||
if(d->APELocation >= 0) {
|
if (d->APELocation >= 0) {
|
||||||
removeBlock(d->APELocation, d->APESize);
|
removeBlock(d->APELocation, d->APESize);
|
||||||
|
|
||||||
if(d->ID3v1Location >= 0)
|
if (d->ID3v1Location >= 0)
|
||||||
d->ID3v1Location -= d->APESize;
|
d->ID3v1Location -= d->APESize;
|
||||||
|
|
||||||
d->APELocation = -1;
|
d->APELocation = -1;
|
||||||
@@ -221,37 +196,35 @@ bool APE::File::save()
|
|||||||
}
|
}
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ID3v1::Tag *APE::File::ID3v1Tag(bool create)
|
ID3v1::Tag *APE::File::ID3v1Tag(bool create) {
|
||||||
{
|
|
||||||
return d->tag.access<ID3v1::Tag>(ApeID3v1Index, create);
|
return d->tag.access<ID3v1::Tag>(ApeID3v1Index, create);
|
||||||
}
|
}
|
||||||
|
|
||||||
APE::Tag *APE::File::APETag(bool create)
|
APE::Tag *APE::File::APETag(bool create) {
|
||||||
{
|
|
||||||
return d->tag.access<APE::Tag>(ApeAPEIndex, create);
|
return d->tag.access<APE::Tag>(ApeAPEIndex, create);
|
||||||
}
|
}
|
||||||
|
|
||||||
void APE::File::strip(int tags)
|
void APE::File::strip(int tags) {
|
||||||
{
|
|
||||||
if(tags & ID3v1)
|
|
||||||
d->tag.set(ApeID3v1Index, 0);
|
|
||||||
|
|
||||||
if(tags & APE)
|
if (tags & ID3v1)
|
||||||
d->tag.set(ApeAPEIndex, 0);
|
d->tag.set(ApeID3v1Index, nullptr);
|
||||||
|
|
||||||
if(!ID3v1Tag())
|
if (tags & APE)
|
||||||
|
d->tag.set(ApeAPEIndex, nullptr);
|
||||||
|
|
||||||
|
if (!ID3v1Tag())
|
||||||
APETag(true);
|
APETag(true);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool APE::File::hasAPETag() const
|
bool APE::File::hasAPETag() const {
|
||||||
{
|
|
||||||
return (d->APELocation >= 0);
|
return (d->APELocation >= 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool APE::File::hasID3v1Tag() const
|
bool APE::File::hasID3v1Tag() const {
|
||||||
{
|
|
||||||
return (d->ID3v1Location >= 0);
|
return (d->ID3v1Location >= 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -259,15 +232,15 @@ bool APE::File::hasID3v1Tag() const
|
|||||||
// private members
|
// private members
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
void APE::File::read(bool readProperties)
|
void APE::File::read(bool readProperties) {
|
||||||
{
|
|
||||||
// Look for an ID3v2 tag
|
// Look for an ID3v2 tag
|
||||||
|
|
||||||
d->ID3v2Location = Utils::findID3v2(this);
|
d->ID3v2Location = Utils::findID3v2(this);
|
||||||
|
|
||||||
if(d->ID3v2Location >= 0) {
|
if (d->ID3v2Location >= 0) {
|
||||||
seek(d->ID3v2Location);
|
seek(d->ID3v2Location);
|
||||||
d->ID3v2Header = new ID3v2::Header(readBlock(ID3v2::Header::size()));
|
d->ID3v2Header.reset(new ID3v2::Header(readBlock(ID3v2::Header::size())));
|
||||||
d->ID3v2Size = d->ID3v2Header->completeTagSize();
|
d->ID3v2Size = d->ID3v2Header->completeTagSize();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -275,36 +248,36 @@ void APE::File::read(bool readProperties)
|
|||||||
|
|
||||||
d->ID3v1Location = Utils::findID3v1(this);
|
d->ID3v1Location = Utils::findID3v1(this);
|
||||||
|
|
||||||
if(d->ID3v1Location >= 0)
|
if (d->ID3v1Location >= 0)
|
||||||
d->tag.set(ApeID3v1Index, new ID3v1::Tag(this, d->ID3v1Location));
|
d->tag.set(ApeID3v1Index, new ID3v1::Tag(this, d->ID3v1Location));
|
||||||
|
|
||||||
// Look for an APE tag
|
// Look for an APE tag
|
||||||
|
|
||||||
d->APELocation = Utils::findAPE(this, d->ID3v1Location);
|
d->APELocation = Utils::findAPE(this, d->ID3v1Location);
|
||||||
|
|
||||||
if(d->APELocation >= 0) {
|
if (d->APELocation >= 0) {
|
||||||
d->tag.set(ApeAPEIndex, new APE::Tag(this, d->APELocation));
|
d->tag.set(ApeAPEIndex, new APE::Tag(this, d->APELocation));
|
||||||
d->APESize = APETag()->footer()->completeTagSize();
|
d->APESize = APETag()->footer()->completeTagSize();
|
||||||
d->APELocation = d->APELocation + APE::Footer::size() - d->APESize;
|
d->APELocation = d->APELocation + APE::Footer::size() - d->APESize;
|
||||||
}
|
}
|
||||||
|
|
||||||
if(d->ID3v1Location < 0)
|
if (d->ID3v1Location < 0)
|
||||||
APETag(true);
|
APETag(true);
|
||||||
|
|
||||||
// Look for APE audio properties
|
// Look for APE audio properties
|
||||||
|
|
||||||
if(readProperties) {
|
if (readProperties) {
|
||||||
|
|
||||||
long streamLength;
|
long long streamLength;
|
||||||
|
|
||||||
if(d->APELocation >= 0)
|
if (d->APELocation >= 0)
|
||||||
streamLength = d->APELocation;
|
streamLength = d->APELocation;
|
||||||
else if(d->ID3v1Location >= 0)
|
else if (d->ID3v1Location >= 0)
|
||||||
streamLength = d->ID3v1Location;
|
streamLength = d->ID3v1Location;
|
||||||
else
|
else
|
||||||
streamLength = length();
|
streamLength = length();
|
||||||
|
|
||||||
if(d->ID3v2Location >= 0) {
|
if (d->ID3v2Location >= 0) {
|
||||||
seek(d->ID3v2Location + d->ID3v2Size);
|
seek(d->ID3v2Location + d->ID3v2Size);
|
||||||
streamLength -= (d->ID3v2Location + d->ID3v2Size);
|
streamLength -= (d->ID3v2Location + d->ID3v2Size);
|
||||||
}
|
}
|
||||||
@@ -312,6 +285,7 @@ void APE::File::read(bool readProperties)
|
|||||||
seek(0);
|
seek(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
d->properties = new Properties(this, streamLength);
|
d->properties.reset(new AudioProperties(this, streamLength));
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
319
3rdparty/taglib/ape/apefile.h
vendored
319
3rdparty/taglib/ape/apefile.h
vendored
@@ -41,197 +41,174 @@
|
|||||||
namespace Strawberry_TagLib {
|
namespace Strawberry_TagLib {
|
||||||
namespace TagLib {
|
namespace TagLib {
|
||||||
|
|
||||||
class Tag;
|
class Tag;
|
||||||
|
|
||||||
namespace ID3v1 { class Tag; }
|
namespace ID3v1 {
|
||||||
namespace APE { class Tag; }
|
class Tag;
|
||||||
|
}
|
||||||
|
namespace APE {
|
||||||
|
class Tag;
|
||||||
|
}
|
||||||
|
|
||||||
//! An implementation of APE metadata
|
//! An implementation of APE metadata
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* This is implementation of APE metadata.
|
||||||
|
*
|
||||||
|
* This supports ID3v1 and APE (v1 and v2) style comments as well as reading stream properties from the file.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace APE {
|
||||||
|
|
||||||
|
//! An implementation of TagLib::File with APE specific methods
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* This implements and provides an interface for APE files to the
|
||||||
|
* TagLib::Tag and TagLib::AudioProperties interfaces by way of implementing
|
||||||
|
* the abstract TagLib::File API as well as providing some additional information specific to APE files.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
class TAGLIB_EXPORT File : public Strawberry_TagLib::TagLib::File {
|
||||||
|
public:
|
||||||
|
/*!
|
||||||
|
* This set of flags is used for various operations and is suitable for being OR-ed together.
|
||||||
|
*/
|
||||||
|
enum TagTypes {
|
||||||
|
//! Empty set. Matches no tag types.
|
||||||
|
NoTags = 0x0000,
|
||||||
|
//! Matches ID3v1 tags.
|
||||||
|
ID3v1 = 0x0001,
|
||||||
|
//! Matches APE tags.
|
||||||
|
APE = 0x0002,
|
||||||
|
//! Matches all tag types.
|
||||||
|
AllTags = 0xffff
|
||||||
|
};
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* This is implementation of APE metadata.
|
* Constructs an APE file from \a file.
|
||||||
|
* If \a readProperties is true the file's audio properties will also be read.
|
||||||
*
|
*
|
||||||
* This supports ID3v1 and APE (v1 and v2) style comments as well as reading stream
|
* \note In the current implementation, \a propertiesStyle is ignored.
|
||||||
* properties from the file.
|
|
||||||
*/
|
*/
|
||||||
|
explicit File(FileName file, bool readProperties = true, AudioProperties::ReadStyle propertiesStyle = AudioProperties::Average);
|
||||||
|
|
||||||
namespace APE {
|
/*!
|
||||||
|
* Constructs an APE file from \a stream.
|
||||||
|
* If \a readProperties is true the file's audio properties will also be read.
|
||||||
|
*
|
||||||
|
* \note TagLib will *not* take ownership of the stream, the caller is responsible for deleting it after the File object.
|
||||||
|
*
|
||||||
|
* \note In the current implementation, \a propertiesStyle is ignored.
|
||||||
|
*/
|
||||||
|
explicit File(IOStream *stream, bool readProperties = true, AudioProperties::ReadStyle propertiesStyle = AudioProperties::Average);
|
||||||
|
|
||||||
//! An implementation of TagLib::File with APE specific methods
|
/*!
|
||||||
|
* Destroys this instance of the File.
|
||||||
|
*/
|
||||||
|
~File() override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* This implements and provides an interface for APE files to the
|
* Returns the Tag for this file. This will be an APE tag, an ID3v1 tag or a combination of the two.
|
||||||
* TagLib::Tag and TagLib::AudioProperties interfaces by way of implementing
|
*/
|
||||||
* the abstract TagLib::File API as well as providing some additional
|
Strawberry_TagLib::TagLib::Tag *tag() const override;
|
||||||
* information specific to APE files.
|
|
||||||
*/
|
|
||||||
|
|
||||||
class TAGLIB_EXPORT File : public Strawberry_TagLib::TagLib::File
|
/*!
|
||||||
{
|
* Implements the unified property interface -- import function.
|
||||||
public:
|
* Creates an APEv2 tag if necessary.
|
||||||
/*!
|
* A potentially existing ID3v1 tag will be updated as well.
|
||||||
* This set of flags is used for various operations and is suitable for
|
*/
|
||||||
* being OR-ed together.
|
PropertyMap setProperties(const PropertyMap&) override;
|
||||||
*/
|
|
||||||
enum TagTypes {
|
|
||||||
//! Empty set. Matches no tag types.
|
|
||||||
NoTags = 0x0000,
|
|
||||||
//! Matches ID3v1 tags.
|
|
||||||
ID3v1 = 0x0001,
|
|
||||||
//! Matches APE tags.
|
|
||||||
APE = 0x0002,
|
|
||||||
//! Matches all tag types.
|
|
||||||
AllTags = 0xffff
|
|
||||||
};
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Constructs an APE file from \a file. If \a readProperties is true the
|
* Returns the APE::AudioProperties for this file.
|
||||||
* file's audio properties will also be read.
|
* If no audio properties were read then this will return a null pointer.
|
||||||
*
|
*/
|
||||||
* \note In the current implementation, \a propertiesStyle is ignored.
|
AudioProperties *audioProperties() const override;
|
||||||
*/
|
|
||||||
File(FileName file, bool readProperties = true,
|
|
||||||
Properties::ReadStyle propertiesStyle = Properties::Average);
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Constructs an APE file from \a stream. If \a readProperties is true the
|
* Saves the file.
|
||||||
* file's audio properties will also be read.
|
*
|
||||||
*
|
* \note According to the official Monkey's Audio SDK, an APE file
|
||||||
* \note TagLib will *not* take ownership of the stream, the caller is
|
* can only have either ID3V1 or APE tags, so a parameter is used here.
|
||||||
* responsible for deleting it after the File object.
|
*/
|
||||||
*
|
bool save() override;
|
||||||
* \note In the current implementation, \a propertiesStyle is ignored.
|
|
||||||
*/
|
|
||||||
File(IOStream *stream, bool readProperties = true,
|
|
||||||
Properties::ReadStyle propertiesStyle = Properties::Average);
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Destroys this instance of the File.
|
* Returns a pointer to the ID3v1 tag of the file.
|
||||||
*/
|
*
|
||||||
virtual ~File();
|
* If \a create is false (the default) this may return a null pointer if there is no valid ID3v1 tag.
|
||||||
|
* If \a create is true it will create an ID3v1 tag if one does not exist and returns a valid pointer.
|
||||||
|
*
|
||||||
|
* \note This may return a valid pointer regardless of whether or not the file on disk has an ID3v1 tag.
|
||||||
|
* Use hasID3v1Tag() to check if the file on disk actually has an ID3v1 tag.
|
||||||
|
*
|
||||||
|
* \note The Tag <b>is still</b> owned by the MPEG::File and should not be deleted by the user.
|
||||||
|
* It will be deleted when the file (object) is destroyed.
|
||||||
|
*
|
||||||
|
* \see hasID3v1Tag()
|
||||||
|
*/
|
||||||
|
ID3v1::Tag *ID3v1Tag(bool create = false);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the Tag for this file. This will be an APE tag, an ID3v1 tag
|
* Returns a pointer to the APE tag of the file.
|
||||||
* or a combination of the two.
|
*
|
||||||
*/
|
* If \a create is false (the default) this may return a null pointer if there is no valid APE tag.
|
||||||
virtual Strawberry_TagLib::TagLib::Tag *tag() const;
|
* If \a create is true it will create an APE tag if one does not exist and returns a valid pointer.
|
||||||
|
*
|
||||||
|
* \note This may return a valid pointer regardless of whether or not the file on disk has an APE tag.
|
||||||
|
* Use hasAPETag() to check if the file on disk actually has an APE tag.
|
||||||
|
*
|
||||||
|
* \note The Tag <b>is still</b> owned by the MPEG::File and should not be deleted by the user.
|
||||||
|
* It will be deleted when the file (object) is destroyed.
|
||||||
|
*
|
||||||
|
* \see hasAPETag()
|
||||||
|
*/
|
||||||
|
APE::Tag *APETag(bool create = false);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Implements the unified property interface -- export function.
|
* This will remove the tags that match the OR-ed together TagTypes from the file.
|
||||||
* If the file contains both an APE and an ID3v1 tag, only APE
|
* By default it removes all tags.
|
||||||
* will be converted to the PropertyMap.
|
*
|
||||||
*/
|
* \note This will also invalidate pointers to the tags as their memory will be freed.
|
||||||
PropertyMap properties() const;
|
* \note In order to make the removal permanent save() still needs to be called
|
||||||
|
*/
|
||||||
|
void strip(int tags = AllTags);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Removes unsupported properties. Forwards to the actual Tag's
|
* Returns whether or not the file on disk actually has an APE tag.
|
||||||
* removeUnsupportedProperties() function.
|
*
|
||||||
*/
|
* \see APETag()
|
||||||
void removeUnsupportedProperties(const StringList &properties);
|
*/
|
||||||
|
bool hasAPETag() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Implements the unified property interface -- import function.
|
* Returns whether or not the file on disk actually has an ID3v1 tag.
|
||||||
* Creates an APEv2 tag if necessary. A potentially existing ID3v1
|
*
|
||||||
* tag will be updated as well.
|
* \see ID3v1Tag()
|
||||||
*/
|
*/
|
||||||
PropertyMap setProperties(const PropertyMap &);
|
bool hasID3v1Tag() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the APE::Properties for this file. If no audio properties
|
* Returns whether or not the given \a stream can be opened as an APE file.
|
||||||
* were read then this will return a null pointer.
|
*
|
||||||
*/
|
* \note This method is designed to do a quick check.
|
||||||
virtual Properties *audioProperties() const;
|
* The result may not necessarily be correct.
|
||||||
|
*/
|
||||||
|
static bool isSupported(IOStream *stream);
|
||||||
|
|
||||||
/*!
|
private:
|
||||||
* Saves the file.
|
File(const File &);
|
||||||
*
|
File &operator=(const File &);
|
||||||
* \note According to the official Monkey's Audio SDK, an APE file
|
|
||||||
* can only have either ID3V1 or APE tags, so a parameter is used here.
|
|
||||||
*/
|
|
||||||
virtual bool save();
|
|
||||||
|
|
||||||
/*!
|
void read(bool readProperties);
|
||||||
* Returns a pointer to the ID3v1 tag of the file.
|
|
||||||
*
|
|
||||||
* If \a create is false (the default) this may return a null pointer
|
|
||||||
* if there is no valid ID3v1 tag. If \a create is true it will create
|
|
||||||
* an ID3v1 tag if one does not exist and returns a valid pointer.
|
|
||||||
*
|
|
||||||
* \note This may return a valid pointer regardless of whether or not the
|
|
||||||
* file on disk has an ID3v1 tag. Use hasID3v1Tag() to check if the file
|
|
||||||
* on disk actually has an ID3v1 tag.
|
|
||||||
*
|
|
||||||
* \note The Tag <b>is still</b> owned by the MPEG::File and should not be
|
|
||||||
* deleted by the user. It will be deleted when the file (object) is
|
|
||||||
* destroyed.
|
|
||||||
*
|
|
||||||
* \see hasID3v1Tag()
|
|
||||||
*/
|
|
||||||
ID3v1::Tag *ID3v1Tag(bool create = false);
|
|
||||||
|
|
||||||
/*!
|
class FilePrivate;
|
||||||
* Returns a pointer to the APE tag of the file.
|
FilePrivate *d;
|
||||||
*
|
};
|
||||||
* If \a create is false (the default) this may return a null pointer
|
} // namespace APE
|
||||||
* if there is no valid APE tag. If \a create is true it will create
|
} // namespace TagLib
|
||||||
* an APE tag if one does not exist and returns a valid pointer.
|
} // namespace Strawberry_TagLib
|
||||||
*
|
|
||||||
* \note This may return a valid pointer regardless of whether or not the
|
|
||||||
* file on disk has an APE tag. Use hasAPETag() to check if the file
|
|
||||||
* on disk actually has an APE tag.
|
|
||||||
*
|
|
||||||
* \note The Tag <b>is still</b> owned by the MPEG::File and should not be
|
|
||||||
* deleted by the user. It will be deleted when the file (object) is
|
|
||||||
* destroyed.
|
|
||||||
*
|
|
||||||
* \see hasAPETag()
|
|
||||||
*/
|
|
||||||
APE::Tag *APETag(bool create = false);
|
|
||||||
|
|
||||||
/*!
|
|
||||||
* This will remove the tags that match the OR-ed together TagTypes from the
|
|
||||||
* file. By default it removes all tags.
|
|
||||||
*
|
|
||||||
* \note This will also invalidate pointers to the tags
|
|
||||||
* as their memory will be freed.
|
|
||||||
* \note In order to make the removal permanent save() still needs to be called
|
|
||||||
*/
|
|
||||||
void strip(int tags = AllTags);
|
|
||||||
|
|
||||||
/*!
|
|
||||||
* Returns whether or not the file on disk actually has an APE tag.
|
|
||||||
*
|
|
||||||
* \see APETag()
|
|
||||||
*/
|
|
||||||
bool hasAPETag() const;
|
|
||||||
|
|
||||||
/*!
|
|
||||||
* Returns whether or not the file on disk actually has an ID3v1 tag.
|
|
||||||
*
|
|
||||||
* \see ID3v1Tag()
|
|
||||||
*/
|
|
||||||
bool hasID3v1Tag() const;
|
|
||||||
|
|
||||||
/*!
|
|
||||||
* Returns whether or not the given \a stream can be opened as an APE
|
|
||||||
* file.
|
|
||||||
*
|
|
||||||
* \note This method is designed to do a quick check. The result may
|
|
||||||
* not necessarily be correct.
|
|
||||||
*/
|
|
||||||
static bool isSupported(IOStream *stream);
|
|
||||||
|
|
||||||
private:
|
|
||||||
File(const File &);
|
|
||||||
File &operator=(const File &);
|
|
||||||
|
|
||||||
void read(bool readProperties);
|
|
||||||
|
|
||||||
class FilePrivate;
|
|
||||||
FilePrivate *d;
|
|
||||||
};
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
115
3rdparty/taglib/ape/apefooter.cpp
vendored
115
3rdparty/taglib/ape/apefooter.cpp
vendored
@@ -27,24 +27,22 @@
|
|||||||
#include <iostream>
|
#include <iostream>
|
||||||
#include <bitset>
|
#include <bitset>
|
||||||
|
|
||||||
#include <tstring.h>
|
#include "tstring.h"
|
||||||
#include <tdebug.h>
|
#include "tdebug.h"
|
||||||
|
|
||||||
#include "apefooter.h"
|
#include "apefooter.h"
|
||||||
|
|
||||||
using namespace Strawberry_TagLib::TagLib;
|
using namespace Strawberry_TagLib::TagLib;
|
||||||
using namespace APE;
|
using namespace APE;
|
||||||
|
|
||||||
class APE::Footer::FooterPrivate
|
class APE::Footer::FooterPrivate {
|
||||||
{
|
public:
|
||||||
public:
|
FooterPrivate() : version(0),
|
||||||
FooterPrivate() :
|
footerPresent(true),
|
||||||
version(0),
|
headerPresent(false),
|
||||||
footerPresent(true),
|
isHeader(false),
|
||||||
headerPresent(false),
|
itemCount(0),
|
||||||
isHeader(false),
|
tagSize(0) {}
|
||||||
itemCount(0),
|
|
||||||
tagSize(0) {}
|
|
||||||
|
|
||||||
unsigned int version;
|
unsigned int version;
|
||||||
|
|
||||||
@@ -61,13 +59,11 @@ public:
|
|||||||
// static members
|
// static members
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
unsigned int APE::Footer::size()
|
unsigned int APE::Footer::size() {
|
||||||
{
|
|
||||||
return 32;
|
return 32;
|
||||||
}
|
}
|
||||||
|
|
||||||
ByteVector APE::Footer::fileIdentifier()
|
ByteVector APE::Footer::fileIdentifier() {
|
||||||
{
|
|
||||||
return ByteVector("APETAGEX");
|
return ByteVector("APETAGEX");
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -75,119 +71,103 @@ ByteVector APE::Footer::fileIdentifier()
|
|||||||
// public members
|
// public members
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
APE::Footer::Footer() :
|
APE::Footer::Footer() : d(new FooterPrivate()) {
|
||||||
d(new FooterPrivate())
|
|
||||||
{
|
|
||||||
}
|
}
|
||||||
|
|
||||||
APE::Footer::Footer(const ByteVector &data) :
|
APE::Footer::Footer(const ByteVector &data) : d(new FooterPrivate()) {
|
||||||
d(new FooterPrivate())
|
|
||||||
{
|
|
||||||
parse(data);
|
parse(data);
|
||||||
}
|
}
|
||||||
|
|
||||||
APE::Footer::~Footer()
|
APE::Footer::~Footer() {
|
||||||
{
|
|
||||||
delete d;
|
delete d;
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned int APE::Footer::version() const
|
unsigned int APE::Footer::version() const {
|
||||||
{
|
|
||||||
return d->version;
|
return d->version;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool APE::Footer::headerPresent() const
|
bool APE::Footer::headerPresent() const {
|
||||||
{
|
|
||||||
return d->headerPresent;
|
return d->headerPresent;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool APE::Footer::footerPresent() const
|
bool APE::Footer::footerPresent() const {
|
||||||
{
|
|
||||||
return d->footerPresent;
|
return d->footerPresent;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool APE::Footer::isHeader() const
|
bool APE::Footer::isHeader() const {
|
||||||
{
|
|
||||||
return d->isHeader;
|
return d->isHeader;
|
||||||
}
|
}
|
||||||
|
|
||||||
void APE::Footer::setHeaderPresent(bool b) const
|
void APE::Footer::setHeaderPresent(bool b) const {
|
||||||
{
|
|
||||||
d->headerPresent = b;
|
d->headerPresent = b;
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned int APE::Footer::itemCount() const
|
unsigned int APE::Footer::itemCount() const {
|
||||||
{
|
|
||||||
return d->itemCount;
|
return d->itemCount;
|
||||||
}
|
}
|
||||||
|
|
||||||
void APE::Footer::setItemCount(unsigned int s)
|
void APE::Footer::setItemCount(unsigned int s) {
|
||||||
{
|
|
||||||
d->itemCount = s;
|
d->itemCount = s;
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned int APE::Footer::tagSize() const
|
unsigned int APE::Footer::tagSize() const {
|
||||||
{
|
|
||||||
return d->tagSize;
|
return d->tagSize;
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned int APE::Footer::completeTagSize() const
|
unsigned int APE::Footer::completeTagSize() const {
|
||||||
{
|
if (d->headerPresent)
|
||||||
if(d->headerPresent)
|
|
||||||
return d->tagSize + size();
|
return d->tagSize + size();
|
||||||
else
|
else
|
||||||
return d->tagSize;
|
return d->tagSize;
|
||||||
}
|
}
|
||||||
|
|
||||||
void APE::Footer::setTagSize(unsigned int s)
|
void APE::Footer::setTagSize(unsigned int s) {
|
||||||
{
|
|
||||||
d->tagSize = s;
|
d->tagSize = s;
|
||||||
}
|
}
|
||||||
|
|
||||||
void APE::Footer::setData(const ByteVector &data)
|
void APE::Footer::setData(const ByteVector &data) {
|
||||||
{
|
|
||||||
parse(data);
|
parse(data);
|
||||||
}
|
}
|
||||||
|
|
||||||
ByteVector APE::Footer::renderFooter() const
|
ByteVector APE::Footer::renderFooter() const {
|
||||||
{
|
|
||||||
return render(false);
|
return render(false);
|
||||||
}
|
}
|
||||||
|
|
||||||
ByteVector APE::Footer::renderHeader() const
|
ByteVector APE::Footer::renderHeader() const {
|
||||||
{
|
|
||||||
if(!d->headerPresent)
|
if (!d->headerPresent)
|
||||||
return ByteVector();
|
return ByteVector();
|
||||||
else
|
else
|
||||||
return render(true);
|
return render(true);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
// protected members
|
// protected members
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
void APE::Footer::parse(const ByteVector &data)
|
void APE::Footer::parse(const ByteVector &data) {
|
||||||
{
|
|
||||||
if(data.size() < size())
|
if (data.size() < size())
|
||||||
return;
|
return;
|
||||||
|
|
||||||
// The first eight bytes, data[0..7], are the File Identifier, "APETAGEX".
|
// The first eight bytes, data[0..7], are the File Identifier, "APETAGEX".
|
||||||
|
|
||||||
// Read the version number
|
// Read the version number
|
||||||
|
|
||||||
d->version = data.toUInt(8, false);
|
d->version = data.toUInt32LE(8);
|
||||||
|
|
||||||
// Read the tag size
|
// Read the tag size
|
||||||
|
|
||||||
d->tagSize = data.toUInt(12, false);
|
d->tagSize = data.toUInt32LE(12);
|
||||||
|
|
||||||
// Read the item count
|
// Read the item count
|
||||||
|
|
||||||
d->itemCount = data.toUInt(16, false);
|
d->itemCount = data.toUInt32LE(16);
|
||||||
|
|
||||||
// Read the flags
|
// Read the flags
|
||||||
|
|
||||||
std::bitset<32> flags(TAGLIB_CONSTRUCT_BITSET(data.toUInt(20, false)));
|
std::bitset<32> flags(TAGLIB_CONSTRUCT_BITSET(data.toUInt32LE(20)));
|
||||||
|
|
||||||
d->headerPresent = flags[31];
|
d->headerPresent = flags[31];
|
||||||
d->footerPresent = !flags[30];
|
d->footerPresent = !flags[30];
|
||||||
@@ -195,8 +175,8 @@ void APE::Footer::parse(const ByteVector &data)
|
|||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ByteVector APE::Footer::render(bool isHeader) const
|
ByteVector APE::Footer::render(bool isHeader) const {
|
||||||
{
|
|
||||||
ByteVector v;
|
ByteVector v;
|
||||||
|
|
||||||
// add the file identifier -- "APETAGEX"
|
// add the file identifier -- "APETAGEX"
|
||||||
@@ -206,29 +186,30 @@ ByteVector APE::Footer::render(bool isHeader) const
|
|||||||
// add the version number -- we always render a 2.000 tag regardless of what
|
// add the version number -- we always render a 2.000 tag regardless of what
|
||||||
// the tag originally was.
|
// the tag originally was.
|
||||||
|
|
||||||
v.append(ByteVector::fromUInt(2000, false));
|
v.append(ByteVector::fromUInt32LE(2000));
|
||||||
|
|
||||||
// add the tag size
|
// add the tag size
|
||||||
|
|
||||||
v.append(ByteVector::fromUInt(d->tagSize, false));
|
v.append(ByteVector::fromUInt32LE(d->tagSize));
|
||||||
|
|
||||||
// add the item count
|
// add the item count
|
||||||
|
|
||||||
v.append(ByteVector::fromUInt(d->itemCount, false));
|
v.append(ByteVector::fromUInt32LE(d->itemCount));
|
||||||
|
|
||||||
// render and add the flags
|
// render and add the flags
|
||||||
|
|
||||||
std::bitset<32> flags;
|
std::bitset<32> flags;
|
||||||
|
|
||||||
flags[31] = d->headerPresent;
|
flags[31] = d->headerPresent;
|
||||||
flags[30] = false; // footer is always present
|
flags[30] = false; // footer is always present
|
||||||
flags[29] = isHeader;
|
flags[29] = isHeader;
|
||||||
|
|
||||||
v.append(ByteVector::fromUInt(flags.to_ulong(), false));
|
v.append(ByteVector::fromUInt32LE(flags.to_ulong()));
|
||||||
|
|
||||||
// add the reserved 64bit
|
// add the reserved 64bit
|
||||||
|
|
||||||
v.append(ByteVector::fromLongLong(0));
|
v.append(ByteVector::fromUInt64BE(0));
|
||||||
|
|
||||||
return v;
|
return v;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
228
3rdparty/taglib/ape/apefooter.h
vendored
228
3rdparty/taglib/ape/apefooter.h
vendored
@@ -32,144 +32,144 @@
|
|||||||
namespace Strawberry_TagLib {
|
namespace Strawberry_TagLib {
|
||||||
namespace TagLib {
|
namespace TagLib {
|
||||||
|
|
||||||
namespace APE {
|
namespace APE {
|
||||||
|
|
||||||
//! An implementation of APE footers
|
//! An implementation of APE footers
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* This class implements APE footers (and headers). It attempts to follow, both
|
* This class implements APE footers (and headers).
|
||||||
* semantically and programmatically, the structure specified in
|
* It attempts to follow, both semantically and programmatically,
|
||||||
* the APE v2.0 standard. The API is based on the properties of APE footer and
|
* the structure specified in the APE v2.0 standard.
|
||||||
* headers specified there.
|
* The API is based on the properties of APE footer and headers specified there.
|
||||||
*/
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
class TAGLIB_EXPORT Footer
|
class TAGLIB_EXPORT Footer {
|
||||||
{
|
public:
|
||||||
public:
|
/*!
|
||||||
/*!
|
* Constructs an empty APE footer.
|
||||||
* Constructs an empty APE footer.
|
*/
|
||||||
*/
|
explicit Footer();
|
||||||
Footer();
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Constructs an APE footer based on \a data. parse() is called
|
* Constructs an APE footer based on \a data. parse() is called immediately.
|
||||||
* immediately.
|
*/
|
||||||
*/
|
explicit Footer(const ByteVector &data);
|
||||||
Footer(const ByteVector &data);
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Destroys the footer.
|
* Destroys the footer.
|
||||||
*/
|
*/
|
||||||
virtual ~Footer();
|
virtual ~Footer();
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the version number. (Note: This is the 1000 or 2000.)
|
* Returns the version number. (Note: This is the 1000 or 2000.)
|
||||||
*/
|
*/
|
||||||
unsigned int version() const;
|
unsigned int version() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns true if a header is present in the tag.
|
* Returns true if a header is present in the tag.
|
||||||
*/
|
*/
|
||||||
bool headerPresent() const;
|
bool headerPresent() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns true if a footer is present in the tag.
|
* Returns true if a footer is present in the tag.
|
||||||
*/
|
*/
|
||||||
bool footerPresent() const;
|
bool footerPresent() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns true this is actually the header.
|
* Returns true this is actually the header.
|
||||||
*/
|
*/
|
||||||
bool isHeader() const;
|
bool isHeader() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Sets whether the header should be rendered or not
|
* Sets whether the header should be rendered or not
|
||||||
*/
|
*/
|
||||||
void setHeaderPresent(bool b) const;
|
void setHeaderPresent(bool b) const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the number of items in the tag.
|
* Returns the number of items in the tag.
|
||||||
*/
|
*/
|
||||||
unsigned int itemCount() const;
|
unsigned int itemCount() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Set the item count to \a s.
|
* Set the item count to \a s.
|
||||||
* \see itemCount()
|
* \see itemCount()
|
||||||
*/
|
*/
|
||||||
void setItemCount(unsigned int s);
|
void setItemCount(unsigned int s);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the tag size in bytes. This is the size of the frame content and footer.
|
* Returns the tag size in bytes.
|
||||||
* The size of the \e entire tag will be this plus the header size, if present.
|
* This is the size of the frame content and footer.
|
||||||
*
|
* The size of the \e entire tag will be this plus the header size, if present.
|
||||||
* \see completeTagSize()
|
*
|
||||||
*/
|
* \see completeTagSize()
|
||||||
unsigned int tagSize() const;
|
*/
|
||||||
|
unsigned int tagSize() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the tag size, including if present, the header
|
* Returns the tag size, including if present, the header
|
||||||
* size.
|
* size.
|
||||||
*
|
*
|
||||||
* \see tagSize()
|
* \see tagSize()
|
||||||
*/
|
*/
|
||||||
unsigned int completeTagSize() const;
|
unsigned int completeTagSize() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Set the tag size to \a s.
|
* Set the tag size to \a s.
|
||||||
* \see tagSize()
|
* \see tagSize()
|
||||||
*/
|
*/
|
||||||
void setTagSize(unsigned int s);
|
void setTagSize(unsigned int s);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the size of the footer. Presently this is always 32 bytes.
|
* Returns the size of the footer. Presently this is always 32 bytes.
|
||||||
*/
|
*/
|
||||||
static unsigned int size();
|
static unsigned int size();
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the string used to identify an APE tag inside of a file.
|
* Returns the string used to identify an APE tag inside of a file.
|
||||||
* Presently this is always "APETAGEX".
|
* Presently this is always "APETAGEX".
|
||||||
*/
|
*/
|
||||||
static ByteVector fileIdentifier();
|
static ByteVector fileIdentifier();
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Sets the data that will be used as the footer. 32 bytes,
|
* Sets the data that will be used as the footer. 32 bytes,
|
||||||
* starting from \a data will be used.
|
* starting from \a data will be used.
|
||||||
*/
|
*/
|
||||||
void setData(const ByteVector &data);
|
void setData(const ByteVector &data);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Renders the footer back to binary format.
|
* Renders the footer back to binary format.
|
||||||
*/
|
*/
|
||||||
ByteVector renderFooter() const;
|
ByteVector renderFooter() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Renders the header corresponding to the footer. If headerPresent is
|
* Renders the header corresponding to the footer.
|
||||||
* set to false, it returns an empty ByteVector.
|
* If headerPresent is set to false, it returns an empty ByteVector.
|
||||||
*/
|
*/
|
||||||
ByteVector renderHeader() const;
|
ByteVector renderHeader() const;
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
/*!
|
/*!
|
||||||
* Called by setData() to parse the footer data. It makes this information
|
* Called by setData() to parse the footer data.
|
||||||
* available through the public API.
|
* It makes this information available through the public API.
|
||||||
*/
|
*/
|
||||||
void parse(const ByteVector &data);
|
void parse(const ByteVector &data);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Called by renderFooter and renderHeader
|
* Called by renderFooter and renderHeader
|
||||||
*/
|
*/
|
||||||
ByteVector render(bool isHeader) const;
|
ByteVector render(bool isHeader) const;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
Footer(const Footer &);
|
Footer(const Footer &);
|
||||||
Footer &operator=(const Footer &);
|
Footer &operator=(const Footer &);
|
||||||
|
|
||||||
class FooterPrivate;
|
class FooterPrivate;
|
||||||
FooterPrivate *d;
|
FooterPrivate *d;
|
||||||
};
|
};
|
||||||
|
|
||||||
}
|
} // namespace APE
|
||||||
}
|
} // namespace TagLib
|
||||||
}
|
} // namespace Strawberry_TagLib
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
256
3rdparty/taglib/ape/apeitem.cpp
vendored
256
3rdparty/taglib/ape/apeitem.cpp
vendored
@@ -23,20 +23,18 @@
|
|||||||
* http://www.mozilla.org/MPL/ *
|
* http://www.mozilla.org/MPL/ *
|
||||||
***************************************************************************/
|
***************************************************************************/
|
||||||
|
|
||||||
#include <tbytevectorlist.h>
|
#include <memory>
|
||||||
#include <tdebug.h>
|
|
||||||
|
#include "tbytevectorlist.h"
|
||||||
|
#include "tdebug.h"
|
||||||
|
|
||||||
#include "apeitem.h"
|
#include "apeitem.h"
|
||||||
|
|
||||||
using namespace Strawberry_TagLib::TagLib;
|
using namespace Strawberry_TagLib::TagLib;
|
||||||
using namespace APE;
|
using namespace APE;
|
||||||
|
|
||||||
class APE::Item::ItemPrivate
|
struct ItemData {
|
||||||
{
|
ItemData() : type(Item::Text), readOnly(false) {}
|
||||||
public:
|
|
||||||
ItemPrivate() :
|
|
||||||
type(Text),
|
|
||||||
readOnly(false) {}
|
|
||||||
|
|
||||||
Item::ItemTypes type;
|
Item::ItemTypes type;
|
||||||
String key;
|
String key;
|
||||||
@@ -45,257 +43,229 @@ public:
|
|||||||
bool readOnly;
|
bool readOnly;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
class APE::Item::ItemPrivate {
|
||||||
|
public:
|
||||||
|
ItemPrivate() : data(new ItemData()) {}
|
||||||
|
|
||||||
|
std::shared_ptr<ItemData> data;
|
||||||
|
};
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
// public members
|
// public members
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
APE::Item::Item() :
|
APE::Item::Item() : d(new ItemPrivate()) {}
|
||||||
d(new ItemPrivate())
|
|
||||||
{
|
APE::Item::Item(const String &key, const String &value) : d(new ItemPrivate()) {
|
||||||
|
d->data->key = key;
|
||||||
|
d->data->text.append(value);
|
||||||
}
|
}
|
||||||
|
|
||||||
APE::Item::Item(const String &key, const String &value) :
|
APE::Item::Item(const String &key, const StringList &values) : d(new ItemPrivate()) {
|
||||||
d(new ItemPrivate())
|
d->data->key = key;
|
||||||
{
|
d->data->text = values;
|
||||||
d->key = key;
|
|
||||||
d->text.append(value);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
APE::Item::Item(const String &key, const StringList &values) :
|
APE::Item::Item(const String &key, const ByteVector &value, bool binary) : d(new ItemPrivate()) {
|
||||||
d(new ItemPrivate())
|
d->data->key = key;
|
||||||
{
|
if (binary) {
|
||||||
d->key = key;
|
d->data->type = Binary;
|
||||||
d->text = values;
|
d->data->value = value;
|
||||||
}
|
|
||||||
|
|
||||||
APE::Item::Item(const String &key, const ByteVector &value, bool binary) :
|
|
||||||
d(new ItemPrivate())
|
|
||||||
{
|
|
||||||
d->key = key;
|
|
||||||
if(binary) {
|
|
||||||
d->type = Binary;
|
|
||||||
d->value = value;
|
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
d->text.append(value);
|
d->data->text.append(value);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
APE::Item::Item(const Item &item) :
|
APE::Item::Item(const Item &item) : d(new ItemPrivate(*item.d)) {}
|
||||||
d(new ItemPrivate(*item.d))
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
APE::Item::~Item()
|
APE::Item::~Item() {
|
||||||
{
|
|
||||||
delete d;
|
delete d;
|
||||||
}
|
}
|
||||||
|
|
||||||
Item &APE::Item::operator=(const Item &item)
|
Item &APE::Item::operator=(const Item &item) {
|
||||||
{
|
|
||||||
Item(item).swap(*this);
|
Item(item).swap(*this);
|
||||||
return *this;
|
return *this;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void APE::Item::swap(Item &item)
|
void APE::Item::swap(Item &item) {
|
||||||
{
|
|
||||||
using std::swap;
|
using std::swap;
|
||||||
|
|
||||||
swap(d, item.d);
|
swap(d, item.d);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void APE::Item::setReadOnly(bool readOnly)
|
void APE::Item::setReadOnly(bool readOnly) {
|
||||||
{
|
d->data->readOnly = readOnly;
|
||||||
d->readOnly = readOnly;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool APE::Item::isReadOnly() const
|
bool APE::Item::isReadOnly() const {
|
||||||
{
|
return d->data->readOnly;
|
||||||
return d->readOnly;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void APE::Item::setType(APE::Item::ItemTypes val)
|
void APE::Item::setType(APE::Item::ItemTypes val) {
|
||||||
{
|
d->data->type = val;
|
||||||
d->type = val;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
APE::Item::ItemTypes APE::Item::type() const
|
APE::Item::ItemTypes APE::Item::type() const {
|
||||||
{
|
return d->data->type;
|
||||||
return d->type;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
String APE::Item::key() const
|
String APE::Item::key() const {
|
||||||
{
|
return d->data->key;
|
||||||
return d->key;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ByteVector APE::Item::binaryData() const
|
ByteVector APE::Item::binaryData() const {
|
||||||
{
|
return d->data->value;
|
||||||
return d->value;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void APE::Item::setBinaryData(const ByteVector &value)
|
void APE::Item::setBinaryData(const ByteVector &value) {
|
||||||
{
|
d->data->type = Binary;
|
||||||
d->type = Binary;
|
d->data->value = value;
|
||||||
d->value = value;
|
d->data->text.clear();
|
||||||
d->text.clear();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ByteVector APE::Item::value() const
|
void APE::Item::setKey(const String &key) {
|
||||||
{
|
d->data->key = key;
|
||||||
// This seems incorrect as it won't be actually rendering the value to keep it
|
|
||||||
// up to date.
|
|
||||||
|
|
||||||
return d->value;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void APE::Item::setKey(const String &key)
|
void APE::Item::setValue(const String &value) {
|
||||||
{
|
d->data->type = Text;
|
||||||
d->key = key;
|
d->data->text = value;
|
||||||
|
d->data->value.clear();
|
||||||
}
|
}
|
||||||
|
|
||||||
void APE::Item::setValue(const String &value)
|
void APE::Item::setValues(const StringList &value) {
|
||||||
{
|
d->data->type = Text;
|
||||||
d->type = Text;
|
d->data->text = value;
|
||||||
d->text = value;
|
d->data->value.clear();
|
||||||
d->value.clear();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void APE::Item::setValues(const StringList &value)
|
void APE::Item::appendValue(const String &value) {
|
||||||
{
|
d->data->type = Text;
|
||||||
d->type = Text;
|
d->data->text.append(value);
|
||||||
d->text = value;
|
d->data->value.clear();
|
||||||
d->value.clear();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void APE::Item::appendValue(const String &value)
|
void APE::Item::appendValues(const StringList &values) {
|
||||||
{
|
d->data->type = Text;
|
||||||
d->type = Text;
|
d->data->text.append(values);
|
||||||
d->text.append(value);
|
d->data->value.clear();
|
||||||
d->value.clear();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void APE::Item::appendValues(const StringList &values)
|
int APE::Item::size() const {
|
||||||
{
|
size_t result = 8 + d->data->key.size() + 1;
|
||||||
d->type = Text;
|
switch (d->data->type) {
|
||||||
d->text.append(values);
|
|
||||||
d->value.clear();
|
|
||||||
}
|
|
||||||
|
|
||||||
int APE::Item::size() const
|
|
||||||
{
|
|
||||||
int result = 8 + d->key.size() + 1;
|
|
||||||
switch(d->type) {
|
|
||||||
case Text:
|
case Text:
|
||||||
if(!d->text.isEmpty()) {
|
if (!d->data->text.isEmpty()) {
|
||||||
StringList::ConstIterator it = d->text.begin();
|
StringList::ConstIterator it = d->data->text.begin();
|
||||||
|
|
||||||
result += it->data(String::UTF8).size();
|
result += it->data(String::UTF8).size();
|
||||||
it++;
|
it++;
|
||||||
for(; it != d->text.end(); ++it)
|
for (; it != d->data->text.end(); ++it)
|
||||||
result += 1 + it->data(String::UTF8).size();
|
result += 1 + it->data(String::UTF8).size();
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case Binary:
|
case Binary:
|
||||||
case Locator:
|
case Locator:
|
||||||
result += d->value.size();
|
result += d->data->value.size();
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
return result;
|
return result;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
StringList APE::Item::toStringList() const
|
StringList APE::Item::values() const {
|
||||||
{
|
return d->data->text;
|
||||||
return d->text;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
StringList APE::Item::values() const
|
String APE::Item::toString() const {
|
||||||
{
|
if (d->data->type == Text && !isEmpty())
|
||||||
return d->text;
|
return d->data->text.front();
|
||||||
}
|
|
||||||
|
|
||||||
String APE::Item::toString() const
|
|
||||||
{
|
|
||||||
if(d->type == Text && !isEmpty())
|
|
||||||
return d->text.front();
|
|
||||||
else
|
else
|
||||||
return String();
|
return String();
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool APE::Item::isEmpty() const
|
bool APE::Item::isEmpty() const {
|
||||||
{
|
switch (d->data->type) {
|
||||||
switch(d->type) {
|
|
||||||
case Text:
|
case Text:
|
||||||
if(d->text.isEmpty())
|
if (d->data->text.isEmpty())
|
||||||
return true;
|
return true;
|
||||||
if(d->text.size() == 1 && d->text.front().isEmpty())
|
if (d->data->text.size() == 1 && d->data->text.front().isEmpty())
|
||||||
return true;
|
return true;
|
||||||
return false;
|
return false;
|
||||||
case Binary:
|
case Binary:
|
||||||
case Locator:
|
case Locator:
|
||||||
return d->value.isEmpty();
|
return d->data->value.isEmpty();
|
||||||
default:
|
default:
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void APE::Item::parse(const ByteVector &data)
|
void APE::Item::parse(const ByteVector &data) {
|
||||||
{
|
|
||||||
// 11 bytes is the minimum size for an APE item
|
// 11 bytes is the minimum size for an APE item
|
||||||
|
|
||||||
if(data.size() < 11) {
|
if (data.size() < 11) {
|
||||||
debug("APE::Item::parse() -- no data in item");
|
debug("APE::Item::parse() -- no data in item");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
const unsigned int valueLength = data.toUInt(0, false);
|
const unsigned int valueLength = data.toUInt32LE(0);
|
||||||
const unsigned int flags = data.toUInt(4, false);
|
const unsigned int flags = data.toUInt32LE(4);
|
||||||
|
|
||||||
// An item key can contain ASCII characters from 0x20 up to 0x7E, not UTF-8.
|
// An item key can contain ASCII characters from 0x20 up to 0x7E, not UTF-8.
|
||||||
// We assume that the validity of the given key has been checked.
|
// We assume that the validity of the given key has been checked.
|
||||||
|
|
||||||
d->key = String(&data[8], String::Latin1);
|
d->data->key = String(&data[8], String::Latin1);
|
||||||
|
|
||||||
const ByteVector value = data.mid(8 + d->key.size() + 1, valueLength);
|
const ByteVector value = data.mid(8 + d->data->key.size() + 1, valueLength);
|
||||||
|
|
||||||
setReadOnly(flags & 1);
|
setReadOnly(flags & 1);
|
||||||
setType(ItemTypes((flags >> 1) & 3));
|
setType(ItemTypes((flags >> 1) & 3));
|
||||||
|
|
||||||
if(Text == d->type)
|
if (Text == d->data->type)
|
||||||
d->text = StringList(ByteVectorList::split(value, '\0'), String::UTF8);
|
d->data->text = StringList(ByteVectorList::split(value, '\0'), String::UTF8);
|
||||||
else
|
else
|
||||||
d->value = value;
|
d->data->value = value;
|
||||||
}
|
}
|
||||||
|
|
||||||
ByteVector APE::Item::render() const
|
ByteVector APE::Item::render() const {
|
||||||
{
|
|
||||||
ByteVector data;
|
ByteVector data;
|
||||||
unsigned int flags = ((d->readOnly) ? 1 : 0) | (d->type << 1);
|
unsigned int flags = ((d->data->readOnly) ? 1 : 0) | (d->data->type << 1);
|
||||||
ByteVector value;
|
ByteVector value;
|
||||||
|
|
||||||
if(isEmpty())
|
if (isEmpty())
|
||||||
return data;
|
return data;
|
||||||
|
|
||||||
if(d->type == Text) {
|
if (d->data->type == Text) {
|
||||||
StringList::ConstIterator it = d->text.begin();
|
StringList::ConstIterator it = d->data->text.begin();
|
||||||
|
|
||||||
value.append(it->data(String::UTF8));
|
value.append(it->data(String::UTF8));
|
||||||
it++;
|
it++;
|
||||||
for(; it != d->text.end(); ++it) {
|
for (; it != d->data->text.end(); ++it) {
|
||||||
value.append('\0');
|
value.append('\0');
|
||||||
value.append(it->data(String::UTF8));
|
value.append(it->data(String::UTF8));
|
||||||
}
|
}
|
||||||
d->value = value;
|
d->data->value = value;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
value.append(d->value);
|
value.append(d->data->value);
|
||||||
|
|
||||||
data.append(ByteVector::fromUInt(value.size(), false));
|
data.append(ByteVector::fromUInt32LE(value.size()));
|
||||||
data.append(ByteVector::fromUInt(flags, false));
|
data.append(ByteVector::fromUInt32LE(flags));
|
||||||
data.append(d->key.data(String::Latin1));
|
data.append(d->data->key.data(String::Latin1));
|
||||||
data.append(ByteVector('\0'));
|
data.append(ByteVector('\0'));
|
||||||
data.append(value);
|
data.append(value);
|
||||||
|
|
||||||
return data;
|
return data;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
305
3rdparty/taglib/ape/apeitem.h
vendored
305
3rdparty/taglib/ape/apeitem.h
vendored
@@ -32,195 +32,176 @@
|
|||||||
|
|
||||||
namespace Strawberry_TagLib {
|
namespace Strawberry_TagLib {
|
||||||
namespace TagLib {
|
namespace TagLib {
|
||||||
|
namespace APE {
|
||||||
|
|
||||||
namespace APE {
|
//! An implementation of APE-items
|
||||||
|
|
||||||
//! An implementation of APE-items
|
/*!
|
||||||
|
* This class provides the features of items in the APEv2 standard.
|
||||||
|
*/
|
||||||
|
class TAGLIB_EXPORT Item {
|
||||||
|
public:
|
||||||
|
/*!
|
||||||
|
* Enum of types an Item can have. The value of 3 is reserved.
|
||||||
|
*/
|
||||||
|
enum ItemTypes {
|
||||||
|
//! Item contains text information coded in UTF-8
|
||||||
|
Text = 0,
|
||||||
|
//! Item contains binary information
|
||||||
|
Binary = 1,
|
||||||
|
//! Item is a locator of external stored information
|
||||||
|
Locator = 2
|
||||||
|
};
|
||||||
|
/*!
|
||||||
|
* Constructs an empty item.
|
||||||
|
*/
|
||||||
|
explicit Item();
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* This class provides the features of items in the APEv2 standard.
|
* Constructs a text item with \a key and \a values.
|
||||||
*/
|
*/
|
||||||
class TAGLIB_EXPORT Item
|
explicit Item(const String &key, const String &values);
|
||||||
{
|
|
||||||
public:
|
|
||||||
/*!
|
|
||||||
* Enum of types an Item can have. The value of 3 is reserved.
|
|
||||||
*/
|
|
||||||
enum ItemTypes {
|
|
||||||
//! Item contains text information coded in UTF-8
|
|
||||||
Text = 0,
|
|
||||||
//! Item contains binary information
|
|
||||||
Binary = 1,
|
|
||||||
//! Item is a locator of external stored information
|
|
||||||
Locator = 2
|
|
||||||
};
|
|
||||||
/*!
|
|
||||||
* Constructs an empty item.
|
|
||||||
*/
|
|
||||||
Item();
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Constructs a text item with \a key and \a value.
|
* Constructs a text item with \a key and \a values.
|
||||||
*/
|
*/
|
||||||
// BIC: Remove this, StringList has a constructor from a single string
|
explicit Item(const String &key, const StringList &values);
|
||||||
Item(const String &key, const String &value);
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Constructs a text item with \a key and \a values.
|
* Constructs an item with \a key and \a value.
|
||||||
*/
|
* If \a binary is true a Binary item will be created, otherwise \a value will be interpreted as text
|
||||||
Item(const String &key, const StringList &values);
|
*/
|
||||||
|
explicit Item(const String &key, const ByteVector &value, bool binary);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Constructs an item with \a key and \a value.
|
* Construct an item as a copy of \a item.
|
||||||
* If \a binary is true a Binary item will be created, otherwise \a value will be interpreted as text
|
*/
|
||||||
*/
|
Item(const Item &item);
|
||||||
Item(const String &key, const ByteVector &value, bool binary);
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Construct an item as a copy of \a item.
|
* Destroys the item.
|
||||||
*/
|
*/
|
||||||
Item(const Item &item);
|
virtual ~Item();
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Destroys the item.
|
* Copies the contents of \a item into this item.
|
||||||
*/
|
*/
|
||||||
virtual ~Item();
|
Item &operator=(const Item &item);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Copies the contents of \a item into this item.
|
* Exchanges the content of this item by the content of \a item.
|
||||||
*/
|
*/
|
||||||
Item &operator=(const Item &item);
|
void swap(Item &item);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Exchanges the content of this item by the content of \a item.
|
* Returns the key.
|
||||||
*/
|
*/
|
||||||
void swap(Item &item);
|
String key() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the key.
|
* Returns the binary value.
|
||||||
*/
|
* If the item type is not \a Binary, always returns an empty ByteVector.
|
||||||
String key() const;
|
*/
|
||||||
|
ByteVector binaryData() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the binary value.
|
* Set the binary value to \a value
|
||||||
* If the item type is not \a Binary, always returns an empty ByteVector.
|
* The item's type will also be set to \a Binary
|
||||||
*/
|
*/
|
||||||
ByteVector binaryData() const;
|
void setBinaryData(const ByteVector &value);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Set the binary value to \a value
|
* Sets the key for the item to \a key.
|
||||||
* The item's type will also be set to \a Binary
|
*/
|
||||||
*/
|
void setKey(const String &key);
|
||||||
void setBinaryData(const ByteVector &value);
|
|
||||||
|
|
||||||
#ifndef DO_NOT_DOCUMENT
|
/*!
|
||||||
/* Remove in next binary incompatible release */
|
* Sets the text value of the item to \a value and clears any previous contents.
|
||||||
ByteVector value() const;
|
*
|
||||||
#endif
|
* \see toString()
|
||||||
|
*/
|
||||||
|
void setValue(const String &value);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Sets the key for the item to \a key.
|
* Sets the text value of the item to the list of values in \a value and clears any previous contents.
|
||||||
*/
|
*
|
||||||
void setKey(const String &key);
|
* \see toStringList()
|
||||||
|
*/
|
||||||
|
void setValues(const StringList &values);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Sets the text value of the item to \a value and clears any previous contents.
|
* Appends \a value to create (or extend) the current list of text values.
|
||||||
*
|
*
|
||||||
* \see toString()
|
* \see toString()
|
||||||
*/
|
*/
|
||||||
void setValue(const String &value);
|
void appendValue(const String &value);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Sets the text value of the item to the list of values in \a value and clears
|
* Appends \a values to extend the current list of text values.
|
||||||
* any previous contents.
|
*
|
||||||
*
|
* \see toStringList()
|
||||||
* \see toStringList()
|
*/
|
||||||
*/
|
void appendValues(const StringList &values);
|
||||||
void setValues(const StringList &values);
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Appends \a value to create (or extend) the current list of text values.
|
* Returns the size of the full item.
|
||||||
*
|
*/
|
||||||
* \see toString()
|
int size() const;
|
||||||
*/
|
|
||||||
void appendValue(const String &value);
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Appends \a values to extend the current list of text values.
|
* Returns the value as a single string. In case of multiple strings, the first is returned.
|
||||||
*
|
* If the data type is not \a Text, always returns an empty String.
|
||||||
* \see toStringList()
|
*/
|
||||||
*/
|
String toString() const;
|
||||||
void appendValues(const StringList &values);
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the size of the full item.
|
* Returns the list of text values. If the data type is not \a Text, always returns an empty StringList.
|
||||||
*/
|
*/
|
||||||
int size() const;
|
StringList values() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the value as a single string. In case of multiple strings,
|
* Render the item to a ByteVector.
|
||||||
* the first is returned. If the data type is not \a Text, always returns
|
*/
|
||||||
* an empty String.
|
ByteVector render() const;
|
||||||
*/
|
|
||||||
String toString() const;
|
|
||||||
|
|
||||||
#ifndef DO_NOT_DOCUMENT
|
/*!
|
||||||
/* Remove in next binary incompatible release */
|
* Parse the item from the ByteVector \a data.
|
||||||
StringList toStringList() const;
|
*/
|
||||||
#endif
|
void parse(const ByteVector &data);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the list of text values. If the data type is not \a Text, always
|
* Set the item to read-only.
|
||||||
* returns an empty StringList.
|
*/
|
||||||
*/
|
void setReadOnly(bool readOnly);
|
||||||
StringList values() const;
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Render the item to a ByteVector.
|
* Return true if the item is read-only.
|
||||||
*/
|
*/
|
||||||
ByteVector render() const;
|
bool isReadOnly() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Parse the item from the ByteVector \a data.
|
* Sets the type of the item to \a type.
|
||||||
*/
|
*
|
||||||
void parse(const ByteVector& data);
|
* \see ItemTypes
|
||||||
|
*/
|
||||||
|
void setType(ItemTypes type);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Set the item to read-only.
|
* Returns the type of the item.
|
||||||
*/
|
*/
|
||||||
void setReadOnly(bool readOnly);
|
ItemTypes type() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Return true if the item is read-only.
|
* Returns if the item has any real content.
|
||||||
*/
|
*/
|
||||||
bool isReadOnly() const;
|
bool isEmpty() const;
|
||||||
|
|
||||||
/*!
|
private:
|
||||||
* Sets the type of the item to \a type.
|
class ItemPrivate;
|
||||||
*
|
ItemPrivate *d;
|
||||||
* \see ItemTypes
|
};
|
||||||
*/
|
} // namespace APE
|
||||||
void setType(ItemTypes type);
|
} // namespace TagLib
|
||||||
|
} // namespace Strawberry_TagLib
|
||||||
/*!
|
|
||||||
* Returns the type of the item.
|
|
||||||
*/
|
|
||||||
ItemTypes type() const;
|
|
||||||
|
|
||||||
/*!
|
|
||||||
* Returns if the item has any real content.
|
|
||||||
*/
|
|
||||||
bool isEmpty() const;
|
|
||||||
|
|
||||||
private:
|
|
||||||
class ItemPrivate;
|
|
||||||
ItemPrivate *d;
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
165
3rdparty/taglib/ape/apeproperties.cpp
vendored
165
3rdparty/taglib/ape/apeproperties.cpp
vendored
@@ -27,9 +27,9 @@
|
|||||||
* http://www.mozilla.org/MPL/ *
|
* http://www.mozilla.org/MPL/ *
|
||||||
***************************************************************************/
|
***************************************************************************/
|
||||||
|
|
||||||
#include <tstring.h>
|
#include "tstring.h"
|
||||||
#include <tdebug.h>
|
#include "tdebug.h"
|
||||||
#include <bitset>
|
|
||||||
#include "id3v2tag.h"
|
#include "id3v2tag.h"
|
||||||
#include "apeproperties.h"
|
#include "apeproperties.h"
|
||||||
#include "apefile.h"
|
#include "apefile.h"
|
||||||
@@ -38,17 +38,15 @@
|
|||||||
|
|
||||||
using namespace Strawberry_TagLib::TagLib;
|
using namespace Strawberry_TagLib::TagLib;
|
||||||
|
|
||||||
class APE::Properties::PropertiesPrivate
|
class APE::AudioProperties::AudioPropertiesPrivate {
|
||||||
{
|
public:
|
||||||
public:
|
explicit AudioPropertiesPrivate() : length(0),
|
||||||
PropertiesPrivate() :
|
bitrate(0),
|
||||||
length(0),
|
sampleRate(0),
|
||||||
bitrate(0),
|
channels(0),
|
||||||
sampleRate(0),
|
version(0),
|
||||||
channels(0),
|
bitsPerSample(0),
|
||||||
version(0),
|
sampleFrames(0) {}
|
||||||
bitsPerSample(0),
|
|
||||||
sampleFrames(0) {}
|
|
||||||
|
|
||||||
int length;
|
int length;
|
||||||
int bitrate;
|
int bitrate;
|
||||||
@@ -63,67 +61,43 @@ public:
|
|||||||
// public members
|
// public members
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
APE::Properties::Properties(File *, ReadStyle style) :
|
APE::AudioProperties::AudioProperties(File *file, long long streamLength, ReadStyle) : Strawberry_TagLib::TagLib::AudioProperties(), d(new AudioPropertiesPrivate()) {
|
||||||
AudioProperties(style),
|
|
||||||
d(new PropertiesPrivate())
|
|
||||||
{
|
|
||||||
debug("APE::Properties::Properties() -- This constructor is no longer used.");
|
|
||||||
}
|
|
||||||
|
|
||||||
APE::Properties::Properties(File *file, long streamLength, ReadStyle style) :
|
|
||||||
AudioProperties(style),
|
|
||||||
d(new PropertiesPrivate())
|
|
||||||
{
|
|
||||||
read(file, streamLength);
|
read(file, streamLength);
|
||||||
}
|
}
|
||||||
|
|
||||||
APE::Properties::~Properties()
|
APE::AudioProperties::~AudioProperties() {
|
||||||
{
|
|
||||||
delete d;
|
delete d;
|
||||||
}
|
}
|
||||||
|
|
||||||
int APE::Properties::length() const
|
int APE::AudioProperties::lengthInSeconds() const {
|
||||||
{
|
|
||||||
return lengthInSeconds();
|
|
||||||
}
|
|
||||||
|
|
||||||
int APE::Properties::lengthInSeconds() const
|
|
||||||
{
|
|
||||||
return d->length / 1000;
|
return d->length / 1000;
|
||||||
}
|
}
|
||||||
|
|
||||||
int APE::Properties::lengthInMilliseconds() const
|
int APE::AudioProperties::lengthInMilliseconds() const {
|
||||||
{
|
|
||||||
return d->length;
|
return d->length;
|
||||||
}
|
}
|
||||||
|
|
||||||
int APE::Properties::bitrate() const
|
int APE::AudioProperties::bitrate() const {
|
||||||
{
|
|
||||||
return d->bitrate;
|
return d->bitrate;
|
||||||
}
|
}
|
||||||
|
|
||||||
int APE::Properties::sampleRate() const
|
int APE::AudioProperties::sampleRate() const {
|
||||||
{
|
|
||||||
return d->sampleRate;
|
return d->sampleRate;
|
||||||
}
|
}
|
||||||
|
|
||||||
int APE::Properties::channels() const
|
int APE::AudioProperties::channels() const {
|
||||||
{
|
|
||||||
return d->channels;
|
return d->channels;
|
||||||
}
|
}
|
||||||
|
|
||||||
int APE::Properties::version() const
|
int APE::AudioProperties::version() const {
|
||||||
{
|
|
||||||
return d->version;
|
return d->version;
|
||||||
}
|
}
|
||||||
|
|
||||||
int APE::Properties::bitsPerSample() const
|
int APE::AudioProperties::bitsPerSample() const {
|
||||||
{
|
|
||||||
return d->bitsPerSample;
|
return d->bitsPerSample;
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned int APE::Properties::sampleFrames() const
|
unsigned int APE::AudioProperties::sampleFrames() const {
|
||||||
{
|
|
||||||
return d->sampleFrames;
|
return d->sampleFrames;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -131,122 +105,123 @@ unsigned int APE::Properties::sampleFrames() const
|
|||||||
// private members
|
// private members
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
namespace
|
namespace {
|
||||||
{
|
int headerVersion(const ByteVector &header) {
|
||||||
int headerVersion(const ByteVector &header)
|
if (header.size() < 6 || !header.startsWith("MAC "))
|
||||||
{
|
return -1;
|
||||||
if(header.size() < 6 || !header.startsWith("MAC "))
|
|
||||||
return -1;
|
|
||||||
|
|
||||||
return header.toUShort(4, false);
|
return header.toUInt16LE(4);
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
} // namespace
|
||||||
|
|
||||||
|
void APE::AudioProperties::read(File *file, long long streamLength) {
|
||||||
|
|
||||||
void APE::Properties::read(File *file, long streamLength)
|
|
||||||
{
|
|
||||||
// First, we assume that the file pointer is set at the first descriptor.
|
// First, we assume that the file pointer is set at the first descriptor.
|
||||||
long offset = file->tell();
|
long long offset = file->tell();
|
||||||
int version = headerVersion(file->readBlock(6));
|
int version = headerVersion(file->readBlock(6));
|
||||||
|
|
||||||
// Next, we look for the descriptor.
|
// Next, we look for the descriptor.
|
||||||
if(version < 0) {
|
if (version < 0) {
|
||||||
offset = file->find("MAC ", offset);
|
offset = file->find("MAC ", offset);
|
||||||
file->seek(offset);
|
file->seek(offset);
|
||||||
version = headerVersion(file->readBlock(6));
|
version = headerVersion(file->readBlock(6));
|
||||||
}
|
}
|
||||||
|
|
||||||
if(version < 0) {
|
if (version < 0) {
|
||||||
debug("APE::Properties::read() -- APE descriptor not found");
|
debug("APE::AudioProperties::read() -- APE descriptor not found");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
d->version = version;
|
d->version = version;
|
||||||
|
|
||||||
if(d->version >= 3980)
|
if (d->version >= 3980)
|
||||||
analyzeCurrent(file);
|
analyzeCurrent(file);
|
||||||
else
|
else
|
||||||
analyzeOld(file);
|
analyzeOld(file);
|
||||||
|
|
||||||
if(d->sampleFrames > 0 && d->sampleRate > 0) {
|
if (d->sampleFrames > 0 && d->sampleRate > 0) {
|
||||||
const double length = d->sampleFrames * 1000.0 / d->sampleRate;
|
const double length = d->sampleFrames * 1000.0 / d->sampleRate;
|
||||||
d->length = static_cast<int>(length + 0.5);
|
d->length = static_cast<int>(length + 0.5);
|
||||||
d->bitrate = static_cast<int>(streamLength * 8.0 / length + 0.5);
|
d->bitrate = static_cast<int>(streamLength * 8.0 / length + 0.5);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void APE::Properties::analyzeCurrent(File *file)
|
void APE::AudioProperties::analyzeCurrent(File *file) {
|
||||||
{
|
|
||||||
// Read the descriptor
|
// Read the descriptor
|
||||||
file->seek(2, File::Current);
|
file->seek(2, File::Current);
|
||||||
const ByteVector descriptor = file->readBlock(44);
|
const ByteVector descriptor = file->readBlock(44);
|
||||||
if(descriptor.size() < 44) {
|
if (descriptor.size() < 44) {
|
||||||
debug("APE::Properties::analyzeCurrent() -- descriptor is too short.");
|
debug("APE::AudioProperties::analyzeCurrent() -- descriptor is too short.");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
const unsigned int descriptorBytes = descriptor.toUInt(0, false);
|
const unsigned int descriptorBytes = descriptor.toUInt32LE(0);
|
||||||
|
|
||||||
if((descriptorBytes - 52) > 0)
|
if ((descriptorBytes - 52) > 0)
|
||||||
file->seek(descriptorBytes - 52, File::Current);
|
file->seek(descriptorBytes - 52, File::Current);
|
||||||
|
|
||||||
// Read the header
|
// Read the header
|
||||||
const ByteVector header = file->readBlock(24);
|
const ByteVector header = file->readBlock(24);
|
||||||
if(header.size() < 24) {
|
if (header.size() < 24) {
|
||||||
debug("APE::Properties::analyzeCurrent() -- MAC header is too short.");
|
debug("APE::AudioProperties::analyzeCurrent() -- MAC header is too short.");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Get the APE info
|
// Get the APE info
|
||||||
d->channels = header.toShort(18, false);
|
d->channels = header.toUInt16LE(18);
|
||||||
d->sampleRate = header.toUInt(20, false);
|
d->sampleRate = header.toUInt32LE(20);
|
||||||
d->bitsPerSample = header.toShort(16, false);
|
d->bitsPerSample = header.toUInt16LE(16);
|
||||||
|
|
||||||
const unsigned int totalFrames = header.toUInt(12, false);
|
const unsigned int totalFrames = header.toUInt32LE(12);
|
||||||
if(totalFrames == 0)
|
if (totalFrames == 0)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
const unsigned int blocksPerFrame = header.toUInt(4, false);
|
const unsigned int blocksPerFrame = header.toUInt32LE(4);
|
||||||
const unsigned int finalFrameBlocks = header.toUInt(8, false);
|
const unsigned int finalFrameBlocks = header.toUInt32LE(8);
|
||||||
d->sampleFrames = (totalFrames - 1) * blocksPerFrame + finalFrameBlocks;
|
d->sampleFrames = (totalFrames - 1) * blocksPerFrame + finalFrameBlocks;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void APE::Properties::analyzeOld(File *file)
|
void APE::AudioProperties::analyzeOld(File *file) {
|
||||||
{
|
|
||||||
const ByteVector header = file->readBlock(26);
|
const ByteVector header = file->readBlock(26);
|
||||||
if(header.size() < 26) {
|
if (header.size() < 26) {
|
||||||
debug("APE::Properties::analyzeOld() -- MAC header is too short.");
|
debug("APE::AudioProperties::analyzeOld() -- MAC header is too short.");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
const unsigned int totalFrames = header.toUInt(18, false);
|
const unsigned int totalFrames = header.toUInt32LE(18);
|
||||||
|
|
||||||
// Fail on 0 length APE files (catches non-finalized APE files)
|
// Fail on 0 length APE files (catches non-finalized APE files)
|
||||||
if(totalFrames == 0)
|
if (totalFrames == 0)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
const short compressionLevel = header.toShort(0, false);
|
const short compressionLevel = header.toUInt32LE(0);
|
||||||
unsigned int blocksPerFrame;
|
unsigned int blocksPerFrame;
|
||||||
if(d->version >= 3950)
|
if (d->version >= 3950)
|
||||||
blocksPerFrame = 73728 * 4;
|
blocksPerFrame = 73728 * 4;
|
||||||
else if(d->version >= 3900 || (d->version >= 3800 && compressionLevel == 4000))
|
else if (d->version >= 3900 || (d->version >= 3800 && compressionLevel == 4000))
|
||||||
blocksPerFrame = 73728;
|
blocksPerFrame = 73728;
|
||||||
else
|
else
|
||||||
blocksPerFrame = 9216;
|
blocksPerFrame = 9216;
|
||||||
|
|
||||||
// Get the APE info
|
// Get the APE info
|
||||||
d->channels = header.toShort(4, false);
|
d->channels = header.toUInt16LE(4);
|
||||||
d->sampleRate = header.toUInt(6, false);
|
d->sampleRate = header.toUInt32LE(6);
|
||||||
|
|
||||||
const unsigned int finalFrameBlocks = header.toUInt(22, false);
|
const unsigned int finalFrameBlocks = header.toUInt32LE(22);
|
||||||
d->sampleFrames = (totalFrames - 1) * blocksPerFrame + finalFrameBlocks;
|
d->sampleFrames = (totalFrames - 1) * blocksPerFrame + finalFrameBlocks;
|
||||||
|
|
||||||
// Get the bit depth from the RIFF-fmt chunk.
|
// Get the bit depth from the RIFF-fmt chunk.
|
||||||
file->seek(16, File::Current);
|
file->seek(16, File::Current);
|
||||||
const ByteVector fmt = file->readBlock(28);
|
const ByteVector fmt = file->readBlock(28);
|
||||||
if(fmt.size() < 28 || !fmt.startsWith("WAVEfmt ")) {
|
if (fmt.size() < 28 || !fmt.startsWith("WAVEfmt ")) {
|
||||||
debug("APE::Properties::analyzeOld() -- fmt header is too short.");
|
debug("APE::AudioProperties::analyzeOld() -- fmt header is too short.");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
d->bitsPerSample = fmt.toShort(26, false);
|
d->bitsPerSample = fmt.toUInt16LE(26);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
151
3rdparty/taglib/ape/apeproperties.h
vendored
151
3rdparty/taglib/ape/apeproperties.h
vendored
@@ -35,111 +35,84 @@
|
|||||||
|
|
||||||
namespace Strawberry_TagLib {
|
namespace Strawberry_TagLib {
|
||||||
namespace TagLib {
|
namespace TagLib {
|
||||||
|
namespace APE {
|
||||||
|
|
||||||
namespace APE {
|
class File;
|
||||||
|
|
||||||
class File;
|
//! An implementation of audio property reading for APE
|
||||||
|
|
||||||
//! An implementation of audio property reading for APE
|
/*!
|
||||||
|
* This reads the data from an APE stream found in the AudioProperties API.
|
||||||
|
*/
|
||||||
|
|
||||||
/*!
|
class TAGLIB_EXPORT AudioProperties : public Strawberry_TagLib::TagLib::AudioProperties {
|
||||||
* This reads the data from an APE stream found in the AudioProperties
|
public:
|
||||||
* API.
|
|
||||||
*/
|
|
||||||
|
|
||||||
class TAGLIB_EXPORT Properties : public AudioProperties
|
/*!
|
||||||
{
|
* Create an instance of APE::AudioProperties with the data read from the APE::File \a file.
|
||||||
public:
|
*/
|
||||||
/*!
|
explicit AudioProperties(File *file, long long streamLength, ReadStyle style = Average);
|
||||||
* Create an instance of APE::Properties with the data read from the
|
|
||||||
* APE::File \a file.
|
|
||||||
*
|
|
||||||
* \deprecated
|
|
||||||
*/
|
|
||||||
TAGLIB_DEPRECATED Properties(File *file, ReadStyle style = Average);
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Create an instance of APE::Properties with the data read from the
|
* Destroys this APE::AudioProperties instance.
|
||||||
* APE::File \a file.
|
*/
|
||||||
*/
|
~AudioProperties() override;
|
||||||
Properties(File *file, long streamLength, ReadStyle style = Average);
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Destroys this APE::Properties instance.
|
* Returns the length of the file in seconds. The length is rounded down to the nearest whole second.
|
||||||
*/
|
*
|
||||||
virtual ~Properties();
|
* \see lengthInMilliseconds()
|
||||||
|
*/
|
||||||
|
int lengthInSeconds() const override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the length of the file in seconds. The length is rounded down to
|
* Returns the length of the file in milliseconds.
|
||||||
* the nearest whole second.
|
*
|
||||||
*
|
* \see lengthInSeconds()
|
||||||
* \note This method is just an alias of lengthInSeconds().
|
*/
|
||||||
*
|
int lengthInMilliseconds() const override;
|
||||||
* \deprecated
|
|
||||||
*/
|
|
||||||
TAGLIB_DEPRECATED virtual int length() const;
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the length of the file in seconds. The length is rounded down to
|
* Returns the average bit rate of the file in kb/s.
|
||||||
* the nearest whole second.
|
*/
|
||||||
*
|
int bitrate() const override;
|
||||||
* \see lengthInMilliseconds()
|
|
||||||
*/
|
|
||||||
// BIC: make virtual
|
|
||||||
int lengthInSeconds() const;
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the length of the file in milliseconds.
|
* Returns the sample rate in Hz.
|
||||||
*
|
*/
|
||||||
* \see lengthInSeconds()
|
int sampleRate() const override;
|
||||||
*/
|
|
||||||
// BIC: make virtual
|
|
||||||
int lengthInMilliseconds() const;
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the average bit rate of the file in kb/s.
|
* Returns the number of audio channels.
|
||||||
*/
|
*/
|
||||||
virtual int bitrate() const;
|
int channels() const override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the sample rate in Hz.
|
* Returns the number of bits per audio sample.
|
||||||
*/
|
*/
|
||||||
virtual int sampleRate() const;
|
int bitsPerSample() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the number of audio channels.
|
* Returns the total number of audio samples in file.
|
||||||
*/
|
*/
|
||||||
virtual int channels() const;
|
unsigned int sampleFrames() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the number of bits per audio sample.
|
* Returns APE version.
|
||||||
*/
|
*/
|
||||||
int bitsPerSample() const;
|
int version() const;
|
||||||
|
|
||||||
/*!
|
private:
|
||||||
* Returns the total number of audio samples in file.
|
void read(File *file, long long streamLength);
|
||||||
*/
|
|
||||||
unsigned int sampleFrames() const;
|
|
||||||
|
|
||||||
/*!
|
void analyzeCurrent(File *file);
|
||||||
* Returns APE version.
|
void analyzeOld(File *file);
|
||||||
*/
|
|
||||||
int version() const;
|
|
||||||
|
|
||||||
private:
|
class AudioPropertiesPrivate;
|
||||||
Properties(const Properties &);
|
AudioPropertiesPrivate *d;
|
||||||
Properties &operator=(const Properties &);
|
};
|
||||||
|
} // namespace APE
|
||||||
void read(File *file, long streamLength);
|
} // namespace TagLib
|
||||||
|
} // namespace Strawberry_TagLib
|
||||||
void analyzeCurrent(File *file);
|
|
||||||
void analyzeOld(File *file);
|
|
||||||
|
|
||||||
class PropertiesPrivate;
|
|
||||||
PropertiesPrivate *d;
|
|
||||||
};
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
406
3rdparty/taglib/ape/apetag.cpp
vendored
406
3rdparty/taglib/ape/apetag.cpp
vendored
@@ -28,15 +28,16 @@
|
|||||||
// it considers specializations with and without class types
|
// it considers specializations with and without class types
|
||||||
// to be different; this define forces Map to use only the
|
// to be different; this define forces Map to use only the
|
||||||
// specialization with the class keyword.
|
// specialization with the class keyword.
|
||||||
#define WANT_CLASS_INSTANTIATION_OF_MAP (1)
|
# define WANT_CLASS_INSTANTIATION_OF_MAP (1)
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#include <tfile.h>
|
#include "tfile.h"
|
||||||
#include <tstring.h>
|
#include "tstring.h"
|
||||||
#include <tmap.h>
|
#include "tmap.h"
|
||||||
#include <tpropertymap.h>
|
#include "tpicturemap.h"
|
||||||
#include <tdebug.h>
|
#include "tpropertymap.h"
|
||||||
#include <tutils.h>
|
#include "tdebug.h"
|
||||||
|
#include "tutils.h"
|
||||||
|
|
||||||
#include "apetag.h"
|
#include "apetag.h"
|
||||||
#include "apefooter.h"
|
#include "apefooter.h"
|
||||||
@@ -45,42 +46,38 @@
|
|||||||
using namespace Strawberry_TagLib::TagLib;
|
using namespace Strawberry_TagLib::TagLib;
|
||||||
using namespace APE;
|
using namespace APE;
|
||||||
|
|
||||||
namespace
|
namespace {
|
||||||
{
|
const unsigned int MinKeyLength = 2;
|
||||||
const unsigned int MinKeyLength = 2;
|
const unsigned int MaxKeyLength = 255;
|
||||||
const unsigned int MaxKeyLength = 255;
|
|
||||||
|
|
||||||
bool isKeyValid(const ByteVector &key)
|
bool isKeyValid(const ByteVector &key) {
|
||||||
{
|
|
||||||
const char *invalidKeys[] = { "ID3", "TAG", "OGGS", "MP+", 0 };
|
|
||||||
|
|
||||||
// only allow printable ASCII including space (32..126)
|
const char *invalidKeys[] = { "ID3", "TAG", "OGGS", "MP+", nullptr };
|
||||||
|
|
||||||
for(ByteVector::ConstIterator it = key.begin(); it != key.end(); ++it) {
|
// only allow printable ASCII including space (32..126)
|
||||||
const int c = static_cast<unsigned char>(*it);
|
|
||||||
if(c < 32 || c > 126)
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
const String upperKey = String(key).upper();
|
for (ByteVector::ConstIterator it = key.begin(); it != key.end(); ++it) {
|
||||||
for(size_t i = 0; invalidKeys[i] != 0; ++i) {
|
const int c = static_cast<unsigned char>(*it);
|
||||||
if(upperKey == invalidKeys[i])
|
if (c < 32 || c > 126)
|
||||||
return false;
|
return false;
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
class APE::Tag::TagPrivate
|
const String upperKey = String(key).upper();
|
||||||
{
|
for (size_t i = 0; invalidKeys[i] != nullptr; ++i) {
|
||||||
public:
|
if (upperKey == invalidKeys[i])
|
||||||
TagPrivate() :
|
return false;
|
||||||
file(0),
|
}
|
||||||
footerLocation(0) {}
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
} // namespace
|
||||||
|
|
||||||
|
class APE::Tag::TagPrivate {
|
||||||
|
public:
|
||||||
|
TagPrivate() : file(nullptr), footerLocation(0) {}
|
||||||
|
|
||||||
File *file;
|
File *file;
|
||||||
long footerLocation;
|
long long footerLocation;
|
||||||
|
|
||||||
Footer footer;
|
Footer footer;
|
||||||
ItemListMap itemListMap;
|
ItemListMap itemListMap;
|
||||||
@@ -90,172 +87,257 @@ public:
|
|||||||
// public methods
|
// public methods
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
APE::Tag::Tag() :
|
APE::Tag::Tag() : d(new TagPrivate()) {}
|
||||||
Strawberry_TagLib::TagLib::Tag(),
|
|
||||||
d(new TagPrivate())
|
APE::Tag::Tag(Strawberry_TagLib::TagLib::File *file, long long footerLocation) : d(new TagPrivate()) {
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
APE::Tag::Tag(Strawberry_TagLib::TagLib::File *file, long footerLocation) :
|
|
||||||
Strawberry_TagLib::TagLib::Tag(),
|
|
||||||
d(new TagPrivate())
|
|
||||||
{
|
|
||||||
d->file = file;
|
d->file = file;
|
||||||
d->footerLocation = footerLocation;
|
d->footerLocation = footerLocation;
|
||||||
|
|
||||||
read();
|
read();
|
||||||
}
|
}
|
||||||
|
|
||||||
APE::Tag::~Tag()
|
APE::Tag::~Tag() {
|
||||||
{
|
|
||||||
delete d;
|
delete d;
|
||||||
}
|
}
|
||||||
|
|
||||||
ByteVector APE::Tag::fileIdentifier()
|
ByteVector APE::Tag::fileIdentifier() {
|
||||||
{
|
|
||||||
return ByteVector::fromCString("APETAGEX");
|
return ByteVector::fromCString("APETAGEX");
|
||||||
}
|
}
|
||||||
|
|
||||||
String APE::Tag::title() const
|
String APE::Tag::title() const {
|
||||||
{
|
|
||||||
if(d->itemListMap["TITLE"].isEmpty())
|
if (d->itemListMap["TITLE"].isEmpty())
|
||||||
return String();
|
return String();
|
||||||
return d->itemListMap["TITLE"].values().toString();
|
return d->itemListMap["TITLE"].values().toString();
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
String APE::Tag::artist() const
|
String APE::Tag::artist() const {
|
||||||
{
|
|
||||||
if(d->itemListMap["ARTIST"].isEmpty())
|
if (d->itemListMap["ARTIST"].isEmpty())
|
||||||
return String();
|
return String();
|
||||||
return d->itemListMap["ARTIST"].values().toString();
|
return d->itemListMap["ARTIST"].values().toString();
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
String APE::Tag::album() const
|
String APE::Tag::album() const {
|
||||||
{
|
|
||||||
if(d->itemListMap["ALBUM"].isEmpty())
|
if (d->itemListMap["ALBUM"].isEmpty())
|
||||||
return String();
|
return String();
|
||||||
return d->itemListMap["ALBUM"].values().toString();
|
return d->itemListMap["ALBUM"].values().toString();
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
String APE::Tag::comment() const
|
String APE::Tag::comment() const {
|
||||||
{
|
|
||||||
if(d->itemListMap["COMMENT"].isEmpty())
|
if (d->itemListMap["COMMENT"].isEmpty())
|
||||||
return String();
|
return String();
|
||||||
return d->itemListMap["COMMENT"].values().toString();
|
return d->itemListMap["COMMENT"].values().toString();
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
String APE::Tag::genre() const
|
String APE::Tag::genre() const {
|
||||||
{
|
|
||||||
if(d->itemListMap["GENRE"].isEmpty())
|
if (d->itemListMap["GENRE"].isEmpty())
|
||||||
return String();
|
return String();
|
||||||
return d->itemListMap["GENRE"].values().toString();
|
return d->itemListMap["GENRE"].values().toString();
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned int APE::Tag::year() const
|
unsigned int APE::Tag::year() const {
|
||||||
{
|
|
||||||
if(d->itemListMap["YEAR"].isEmpty())
|
if (d->itemListMap["YEAR"].isEmpty())
|
||||||
return 0;
|
return 0;
|
||||||
return d->itemListMap["YEAR"].toString().toInt();
|
return d->itemListMap["YEAR"].toString().toInt();
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned int APE::Tag::track() const
|
unsigned int APE::Tag::track() const {
|
||||||
{
|
|
||||||
if(d->itemListMap["TRACK"].isEmpty())
|
if (d->itemListMap["TRACK"].isEmpty())
|
||||||
return 0;
|
return 0;
|
||||||
return d->itemListMap["TRACK"].toString().toInt();
|
return d->itemListMap["TRACK"].toString().toInt();
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void APE::Tag::setTitle(const String &s)
|
Strawberry_TagLib::TagLib::PictureMap APE::Tag::pictures() const {
|
||||||
{
|
|
||||||
|
PictureMap map;
|
||||||
|
if (d->itemListMap.contains(FRONT_COVER)) {
|
||||||
|
Item front = d->itemListMap[FRONT_COVER];
|
||||||
|
if (Item::Binary == front.type()) {
|
||||||
|
ByteVector picture = front.binaryData();
|
||||||
|
const size_t index = picture.find('\0');
|
||||||
|
if (index < picture.size()) {
|
||||||
|
ByteVector desc = picture.mid(0, index + 1);
|
||||||
|
String mime = "image/jpeg";
|
||||||
|
ByteVector data = picture.mid(index + 1);
|
||||||
|
Picture p(data, Picture::FrontCover, mime, desc);
|
||||||
|
map.insert(p);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (d->itemListMap.contains(BACK_COVER)) {
|
||||||
|
Item back = d->itemListMap[BACK_COVER];
|
||||||
|
if (Item::Binary == back.type()) {
|
||||||
|
ByteVector picture = back.binaryData();
|
||||||
|
const size_t index = picture.find('\0');
|
||||||
|
if (index < picture.size()) {
|
||||||
|
ByteVector desc = picture.mid(0, index + 1);
|
||||||
|
String mime = "image/jpeg";
|
||||||
|
ByteVector data = picture.mid(index + 1);
|
||||||
|
Picture p(data, Picture::BackCover, mime, desc);
|
||||||
|
map.insert(p);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return PictureMap(map);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
void APE::Tag::setTitle(const String &s) {
|
||||||
addValue("TITLE", s, true);
|
addValue("TITLE", s, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
void APE::Tag::setArtist(const String &s)
|
void APE::Tag::setArtist(const String &s) {
|
||||||
{
|
|
||||||
addValue("ARTIST", s, true);
|
addValue("ARTIST", s, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
void APE::Tag::setAlbum(const String &s)
|
void APE::Tag::setAlbum(const String &s) {
|
||||||
{
|
|
||||||
addValue("ALBUM", s, true);
|
addValue("ALBUM", s, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
void APE::Tag::setComment(const String &s)
|
void APE::Tag::setComment(const String &s) {
|
||||||
{
|
|
||||||
addValue("COMMENT", s, true);
|
addValue("COMMENT", s, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
void APE::Tag::setGenre(const String &s)
|
void APE::Tag::setGenre(const String &s) {
|
||||||
{
|
|
||||||
addValue("GENRE", s, true);
|
addValue("GENRE", s, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
void APE::Tag::setYear(unsigned int i)
|
void APE::Tag::setYear(unsigned int i) {
|
||||||
{
|
|
||||||
if(i == 0)
|
if (i == 0)
|
||||||
removeItem("YEAR");
|
removeItem("YEAR");
|
||||||
else
|
else
|
||||||
addValue("YEAR", String::number(i), true);
|
addValue("YEAR", String::number(i), true);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void APE::Tag::setTrack(unsigned int i)
|
void APE::Tag::setTrack(unsigned int i) {
|
||||||
{
|
|
||||||
if(i == 0)
|
if (i == 0)
|
||||||
removeItem("TRACK");
|
removeItem("TRACK");
|
||||||
else
|
else
|
||||||
addValue("TRACK", String::number(i), true);
|
addValue("TRACK", String::number(i), true);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
namespace
|
void APE::Tag::setPictures(const PictureMap &l) {
|
||||||
{
|
|
||||||
// conversions of tag keys between what we use in PropertyMap and what's usual
|
removeItem(FRONT_COVER);
|
||||||
// for APE tags
|
removeItem(BACK_COVER);
|
||||||
// usual, APE
|
|
||||||
const char *keyConversions[][2] = {{"TRACKNUMBER", "TRACK" },
|
for (PictureMap::ConstIterator pictureMapIt = l.begin(); pictureMapIt != l.end(); ++pictureMapIt) {
|
||||||
{"DATE", "YEAR" },
|
Picture::Type type = pictureMapIt->first;
|
||||||
{"ALBUMARTIST", "ALBUM ARTIST"},
|
if (Picture::FrontCover != type && Picture::BackCover != type) {
|
||||||
{"DISCNUMBER", "DISC" },
|
std::cout << "APE: Trying to add a picture with wrong type" << std::endl;
|
||||||
{"REMIXER", "MIXARTIST" }};
|
continue;
|
||||||
const size_t keyConversionsSize = sizeof(keyConversions) / sizeof(keyConversions[0]);
|
}
|
||||||
|
|
||||||
|
const char *id;
|
||||||
|
switch (type) {
|
||||||
|
case Picture::FrontCover:
|
||||||
|
id = FRONT_COVER;
|
||||||
|
break;
|
||||||
|
case Picture::BackCover:
|
||||||
|
id = BACK_COVER;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
id = FRONT_COVER;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
PictureList list = pictureMapIt->second;
|
||||||
|
for (PictureList::ConstIterator pictureListIt = list.begin(); pictureListIt != list.end(); ++pictureListIt) {
|
||||||
|
Picture picture = *pictureListIt;
|
||||||
|
if (d->itemListMap.contains(id)) {
|
||||||
|
std::cout << "APE: Already added a picture of type "
|
||||||
|
<< id
|
||||||
|
<< " '"
|
||||||
|
<< picture.description()
|
||||||
|
<< "' "
|
||||||
|
<< "and next are being ignored"
|
||||||
|
<< std::endl;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
ByteVector data = picture.description().data(String::Latin1).append('\0').append(picture.data());
|
||||||
|
|
||||||
|
Item item;
|
||||||
|
item.setKey(id);
|
||||||
|
item.setType(Item::Binary);
|
||||||
|
item.setBinaryData(data);
|
||||||
|
setItem(item.key(), item);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
PropertyMap APE::Tag::properties() const
|
namespace {
|
||||||
{
|
// conversions of tag keys between what we use in PropertyMap and what's usual
|
||||||
|
// for APE tags
|
||||||
|
// usual, APE
|
||||||
|
const char *keyConversions[][2] = { { "TRACKNUMBER", "TRACK" },
|
||||||
|
{ "DATE", "YEAR" },
|
||||||
|
{ "ALBUMARTIST", "ALBUM ARTIST" },
|
||||||
|
{ "DISCNUMBER", "DISC" },
|
||||||
|
{ "REMIXER", "MIXARTIST" } };
|
||||||
|
const size_t keyConversionsSize = sizeof(keyConversions) / sizeof(keyConversions[0]);
|
||||||
|
} // namespace
|
||||||
|
|
||||||
|
PropertyMap APE::Tag::properties() const {
|
||||||
|
|
||||||
PropertyMap properties;
|
PropertyMap properties;
|
||||||
ItemListMap::ConstIterator it = itemListMap().begin();
|
ItemListMap::ConstIterator it = itemListMap().begin();
|
||||||
for(; it != itemListMap().end(); ++it) {
|
for (; it != itemListMap().end(); ++it) {
|
||||||
String tagName = it->first.upper();
|
String tagName = it->first.upper();
|
||||||
// if the item is Binary or Locator, or if the key is an invalid string,
|
// if the item is Binary or Locator, or if the key is an invalid string,
|
||||||
// add to unsupportedData
|
// add to unsupportedData
|
||||||
if(it->second.type() != Item::Text || tagName.isEmpty()) {
|
if (it->second.type() != Item::Text || tagName.isEmpty()) {
|
||||||
properties.unsupportedData().append(it->first);
|
properties.unsupportedData().append(it->first);
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
// Some tags need to be handled specially
|
// Some tags need to be handled specially
|
||||||
for(size_t i = 0; i < keyConversionsSize; ++i) {
|
for (size_t i = 0; i < keyConversionsSize; ++i) {
|
||||||
if(tagName == keyConversions[i][1])
|
if (tagName == keyConversions[i][1])
|
||||||
tagName = keyConversions[i][0];
|
tagName = keyConversions[i][0];
|
||||||
}
|
}
|
||||||
properties[tagName].append(it->second.toStringList());
|
properties[tagName].append(it->second.values());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return properties;
|
return properties;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void APE::Tag::removeUnsupportedProperties(const StringList &properties)
|
void APE::Tag::removeUnsupportedProperties(const StringList &properties) {
|
||||||
{
|
|
||||||
StringList::ConstIterator it = properties.begin();
|
StringList::ConstIterator it = properties.begin();
|
||||||
for(; it != properties.end(); ++it)
|
for (; it != properties.end(); ++it)
|
||||||
removeItem(*it);
|
removeItem(*it);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
PropertyMap APE::Tag::setProperties(const PropertyMap &origProps)
|
PropertyMap APE::Tag::setProperties(const PropertyMap &origProps) {
|
||||||
{
|
|
||||||
PropertyMap properties(origProps); // make a local copy that can be modified
|
PropertyMap properties(origProps); // make a local copy that can be modified
|
||||||
|
|
||||||
// see comment in properties()
|
// see comment in properties()
|
||||||
for(size_t i = 0; i < keyConversionsSize; ++i)
|
for (size_t i = 0; i < keyConversionsSize; ++i)
|
||||||
if(properties.contains(keyConversions[i][0])) {
|
if (properties.contains(keyConversions[i][0])) {
|
||||||
properties.insert(keyConversions[i][1], properties[keyConversions[i][0]]);
|
properties.insert(keyConversions[i][1], properties[keyConversions[i][0]]);
|
||||||
properties.erase(keyConversions[i][0]);
|
properties.erase(keyConversions[i][0]);
|
||||||
}
|
}
|
||||||
@@ -263,31 +345,31 @@ PropertyMap APE::Tag::setProperties(const PropertyMap &origProps)
|
|||||||
// first check if tags need to be removed completely
|
// first check if tags need to be removed completely
|
||||||
StringList toRemove;
|
StringList toRemove;
|
||||||
ItemListMap::ConstIterator remIt = itemListMap().begin();
|
ItemListMap::ConstIterator remIt = itemListMap().begin();
|
||||||
for(; remIt != itemListMap().end(); ++remIt) {
|
for (; remIt != itemListMap().end(); ++remIt) {
|
||||||
String key = remIt->first.upper();
|
String key = remIt->first.upper();
|
||||||
// only remove if a) key is valid, b) type is text, c) key not contained in new properties
|
// only remove if a) key is valid, b) type is text, c) key not contained in new properties
|
||||||
if(!key.isEmpty() && remIt->second.type() == APE::Item::Text && !properties.contains(key))
|
if (!key.isEmpty() && remIt->second.type() == APE::Item::Text && !properties.contains(key))
|
||||||
toRemove.append(remIt->first);
|
toRemove.append(remIt->first);
|
||||||
}
|
}
|
||||||
|
|
||||||
for(StringList::ConstIterator removeIt = toRemove.begin(); removeIt != toRemove.end(); removeIt++)
|
for (StringList::ConstIterator removeIt = toRemove.begin(); removeIt != toRemove.end(); removeIt++)
|
||||||
removeItem(*removeIt);
|
removeItem(*removeIt);
|
||||||
|
|
||||||
// now sync in the "forward direction"
|
// now sync in the "forward direction"
|
||||||
PropertyMap::ConstIterator it = properties.begin();
|
PropertyMap::ConstIterator it = properties.begin();
|
||||||
PropertyMap invalid;
|
PropertyMap invalid;
|
||||||
for(; it != properties.end(); ++it) {
|
for (; it != properties.end(); ++it) {
|
||||||
const String &tagName = it->first;
|
const String &tagName = it->first;
|
||||||
if(!checkKey(tagName))
|
if (!checkKey(tagName))
|
||||||
invalid.insert(it->first, it->second);
|
invalid.insert(it->first, it->second);
|
||||||
else if(!(itemListMap().contains(tagName)) || !(itemListMap()[tagName].values() == it->second)) {
|
else if (!(itemListMap().contains(tagName)) || !(itemListMap()[tagName].values() == it->second)) {
|
||||||
if(it->second.isEmpty())
|
if (it->second.isEmpty())
|
||||||
removeItem(tagName);
|
removeItem(tagName);
|
||||||
else {
|
else {
|
||||||
StringList::ConstIterator valueIt = it->second.begin();
|
StringList::ConstIterator valueIt = it->second.begin();
|
||||||
addValue(tagName, *valueIt, true);
|
addValue(tagName, *valueIt, true);
|
||||||
++valueIt;
|
++valueIt;
|
||||||
for(; valueIt != it->second.end(); ++valueIt)
|
for (; valueIt != it->second.end(); ++valueIt)
|
||||||
addValue(tagName, *valueIt, false);
|
addValue(tagName, *valueIt, false);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -295,35 +377,33 @@ PropertyMap APE::Tag::setProperties(const PropertyMap &origProps)
|
|||||||
return invalid;
|
return invalid;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool APE::Tag::checkKey(const String &key)
|
bool APE::Tag::checkKey(const String &key) {
|
||||||
{
|
|
||||||
if(key.size() < MinKeyLength || key.size() > MaxKeyLength)
|
if (key.size() < MinKeyLength || key.size() > MaxKeyLength)
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
return isKeyValid(key.data(String::UTF8));
|
return isKeyValid(key.data(String::UTF8));
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
APE::Footer *APE::Tag::footer() const
|
APE::Footer *APE::Tag::footer() const {
|
||||||
{
|
|
||||||
return &d->footer;
|
return &d->footer;
|
||||||
}
|
}
|
||||||
|
|
||||||
const APE::ItemListMap& APE::Tag::itemListMap() const
|
const APE::ItemListMap &APE::Tag::itemListMap() const {
|
||||||
{
|
|
||||||
return d->itemListMap;
|
return d->itemListMap;
|
||||||
}
|
}
|
||||||
|
|
||||||
void APE::Tag::removeItem(const String &key)
|
void APE::Tag::removeItem(const String &key) {
|
||||||
{
|
|
||||||
d->itemListMap.erase(key.upper());
|
d->itemListMap.erase(key.upper());
|
||||||
}
|
}
|
||||||
|
|
||||||
void APE::Tag::addValue(const String &key, const String &value, bool replace)
|
void APE::Tag::addValue(const String &key, const String &value, bool replace) {
|
||||||
{
|
|
||||||
if(replace)
|
if (replace)
|
||||||
removeItem(key);
|
removeItem(key);
|
||||||
|
|
||||||
if(value.isEmpty())
|
if (value.isEmpty())
|
||||||
return;
|
return;
|
||||||
|
|
||||||
// Text items may contain more than one value.
|
// Text items may contain more than one value.
|
||||||
@@ -331,34 +411,36 @@ void APE::Tag::addValue(const String &key, const String &value, bool replace)
|
|||||||
|
|
||||||
ItemListMap::Iterator it = d->itemListMap.find(key.upper());
|
ItemListMap::Iterator it = d->itemListMap.find(key.upper());
|
||||||
|
|
||||||
if(it != d->itemListMap.end() && it->second.type() == Item::Text)
|
if (it != d->itemListMap.end() && it->second.type() == Item::Text)
|
||||||
it->second.appendValue(value);
|
it->second.appendValue(value);
|
||||||
else
|
else
|
||||||
setItem(key, Item(key, value));
|
setItem(key, Item(key, value));
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void APE::Tag::setData(const String &key, const ByteVector &value)
|
void APE::Tag::setData(const String &key, const ByteVector &value) {
|
||||||
{
|
|
||||||
removeItem(key);
|
removeItem(key);
|
||||||
|
|
||||||
if(value.isEmpty())
|
if (value.isEmpty())
|
||||||
return;
|
return;
|
||||||
|
|
||||||
setItem(key, Item(key, value, true));
|
setItem(key, Item(key, value, true));
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void APE::Tag::setItem(const String &key, const Item &item)
|
void APE::Tag::setItem(const String &key, const Item &item) {
|
||||||
{
|
|
||||||
if(!checkKey(key)) {
|
if (!checkKey(key)) {
|
||||||
debug("APE::Tag::setItem() - Couldn't set an item due to an invalid key.");
|
debug("APE::Tag::setItem() - Couldn't set an item due to an invalid key.");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
d->itemListMap[key.upper()] = item;
|
d->itemListMap[key.upper()] = item;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool APE::Tag::isEmpty() const
|
bool APE::Tag::isEmpty() const {
|
||||||
{
|
|
||||||
return d->itemListMap.isEmpty();
|
return d->itemListMap.isEmpty();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -366,28 +448,29 @@ bool APE::Tag::isEmpty() const
|
|||||||
// protected methods
|
// protected methods
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
void APE::Tag::read()
|
void APE::Tag::read() {
|
||||||
{
|
|
||||||
if(d->file && d->file->isValid()) {
|
if (d->file && d->file->isValid()) {
|
||||||
|
|
||||||
d->file->seek(d->footerLocation);
|
d->file->seek(d->footerLocation);
|
||||||
d->footer.setData(d->file->readBlock(Footer::size()));
|
d->footer.setData(d->file->readBlock(Footer::size()));
|
||||||
|
|
||||||
if(d->footer.tagSize() <= Footer::size() ||
|
if (d->footer.tagSize() <= Footer::size() ||
|
||||||
d->footer.tagSize() > static_cast<unsigned long>(d->file->length()))
|
d->footer.tagSize() > static_cast<unsigned long>(d->file->length()))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
d->file->seek(d->footerLocation + Footer::size() - d->footer.tagSize());
|
d->file->seek(d->footerLocation + Footer::size() - d->footer.tagSize());
|
||||||
parse(d->file->readBlock(d->footer.tagSize() - Footer::size()));
|
parse(d->file->readBlock(d->footer.tagSize() - Footer::size()));
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ByteVector APE::Tag::render() const
|
ByteVector APE::Tag::render() const {
|
||||||
{
|
|
||||||
ByteVector data;
|
ByteVector data;
|
||||||
unsigned int itemCount = 0;
|
unsigned int itemCount = 0;
|
||||||
|
|
||||||
for(ItemListMap::ConstIterator it = d->itemListMap.begin(); it != d->itemListMap.end(); ++it) {
|
for (ItemListMap::ConstIterator it = d->itemListMap.begin(); it != d->itemListMap.end(); ++it) {
|
||||||
data.append(it->second.render());
|
data.append(it->second.render());
|
||||||
itemCount++;
|
itemCount++;
|
||||||
}
|
}
|
||||||
@@ -397,32 +480,30 @@ ByteVector APE::Tag::render() const
|
|||||||
d->footer.setHeaderPresent(true);
|
d->footer.setHeaderPresent(true);
|
||||||
|
|
||||||
return d->footer.renderHeader() + data + d->footer.renderFooter();
|
return d->footer.renderHeader() + data + d->footer.renderFooter();
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void APE::Tag::parse(const ByteVector &data)
|
void APE::Tag::parse(const ByteVector &data) {
|
||||||
{
|
|
||||||
// 11 bytes is the minimum size for an APE item
|
// 11 bytes is the minimum size for an APE item
|
||||||
|
|
||||||
if(data.size() < 11)
|
if (data.size() < 11)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
unsigned int pos = 0;
|
size_t pos = 0;
|
||||||
|
|
||||||
for(unsigned int i = 0; i < d->footer.itemCount() && pos <= data.size() - 11; i++) {
|
for (unsigned int i = 0; i < d->footer.itemCount() && pos <= data.size() - 11; i++) {
|
||||||
|
|
||||||
const int nullPos = data.find('\0', pos + 8);
|
const size_t nullPos = data.find('\0', pos + 8);
|
||||||
if(nullPos < 0) {
|
if (nullPos == ByteVector::npos()) {
|
||||||
debug("APE::Tag::parse() - Couldn't find a key/value separator. Stopped parsing.");
|
debug("APE::Tag::parse() - Couldn't find a key/value separator. Stopped parsing.");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
const unsigned int keyLength = nullPos - pos - 8;
|
const size_t keyLength = nullPos - pos - 8;
|
||||||
const unsigned int valLegnth = data.toUInt(pos, false);
|
const size_t valLegnth = data.toUInt32LE(pos);
|
||||||
|
|
||||||
if(keyLength >= MinKeyLength
|
if (keyLength >= MinKeyLength && keyLength <= MaxKeyLength && isKeyValid(data.mid(pos + 8, keyLength))) {
|
||||||
&& keyLength <= MaxKeyLength
|
|
||||||
&& isKeyValid(data.mid(pos + 8, keyLength)))
|
|
||||||
{
|
|
||||||
APE::Item item;
|
APE::Item item;
|
||||||
item.parse(data.mid(pos));
|
item.parse(data.mid(pos));
|
||||||
|
|
||||||
@@ -434,4 +515,5 @@ void APE::Tag::parse(const ByteVector &data)
|
|||||||
|
|
||||||
pos += keyLength + valLegnth + 9;
|
pos += keyLength + valLegnth + 9;
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
284
3rdparty/taglib/ape/apetag.h
vendored
284
3rdparty/taglib/ape/apetag.h
vendored
@@ -34,177 +34,183 @@
|
|||||||
|
|
||||||
#include "apeitem.h"
|
#include "apeitem.h"
|
||||||
|
|
||||||
|
#define FRONT_COVER "COVER ART (FRONT)"
|
||||||
|
#define BACK_COVER "COVER ART (BACK)"
|
||||||
|
|
||||||
namespace Strawberry_TagLib {
|
namespace Strawberry_TagLib {
|
||||||
namespace TagLib {
|
namespace TagLib {
|
||||||
|
|
||||||
class File;
|
class File;
|
||||||
|
|
||||||
//! An implementation of the APE tagging format
|
//! An implementation of the APE tagging format
|
||||||
|
|
||||||
namespace APE {
|
namespace APE {
|
||||||
|
|
||||||
class Footer;
|
class Footer;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* A mapping between a list of item names, or keys, and the associated item.
|
* A mapping between a list of item names, or keys, and the associated item.
|
||||||
*
|
*
|
||||||
* \see APE::Tag::itemListMap()
|
* \see APE::Tag::itemListMap()
|
||||||
*/
|
*/
|
||||||
typedef Map<const String, Item> ItemListMap;
|
typedef Map<String, Item> ItemListMap;
|
||||||
|
|
||||||
|
|
||||||
//! An APE tag implementation
|
//! An APE tag implementation
|
||||||
|
|
||||||
class TAGLIB_EXPORT Tag : public Strawberry_TagLib::TagLib::Tag
|
class TAGLIB_EXPORT Tag : public Strawberry_TagLib::TagLib::Tag {
|
||||||
{
|
public:
|
||||||
public:
|
/*!
|
||||||
/*!
|
* Create an APE tag with default values.
|
||||||
* Create an APE tag with default values.
|
*/
|
||||||
*/
|
explicit Tag();
|
||||||
Tag();
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Create an APE tag and parse the data in \a file with APE footer at
|
* Create an APE tag and parse the data in \a file with APE footer at
|
||||||
* \a tagOffset.
|
* \a tagOffset.
|
||||||
*/
|
*/
|
||||||
Tag(Strawberry_TagLib::TagLib::File *file, long footerLocation);
|
explicit Tag(Strawberry_TagLib::TagLib::File *file, long long footerLocation);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Destroys this Tag instance.
|
* Destroys this Tag instance.
|
||||||
*/
|
*/
|
||||||
virtual ~Tag();
|
~Tag() override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Renders the in memory values to a ByteVector suitable for writing to
|
* Renders the in memory values to a ByteVector suitable for writing to the file.
|
||||||
* the file.
|
*/
|
||||||
*/
|
ByteVector render() const;
|
||||||
ByteVector render() const;
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the string "APETAGEX" suitable for usage in locating the tag in a
|
* Returns the string "APETAGEX" suitable for usage in locating the tag in a file.
|
||||||
* file.
|
*/
|
||||||
*/
|
static ByteVector fileIdentifier();
|
||||||
static ByteVector fileIdentifier();
|
|
||||||
|
|
||||||
// Reimplementations.
|
// Reimplementations.
|
||||||
|
|
||||||
virtual String title() const;
|
String title() const override;
|
||||||
virtual String artist() const;
|
String artist() const override;
|
||||||
virtual String album() const;
|
String album() const override;
|
||||||
virtual String comment() const;
|
String comment() const override;
|
||||||
virtual String genre() const;
|
String genre() const override;
|
||||||
virtual unsigned int year() const;
|
unsigned int year() const override;
|
||||||
virtual unsigned int track() const;
|
unsigned int track() const override;
|
||||||
|
|
||||||
virtual void setTitle(const String &s);
|
/**
|
||||||
virtual void setArtist(const String &s);
|
* @brief pictures
|
||||||
virtual void setAlbum(const String &s);
|
* According to :
|
||||||
virtual void setComment(const String &s);
|
* http://www.hydrogenaud.io/forums/index.php?showtopic=40603&st=50&p=504669&#entry504669
|
||||||
virtual void setGenre(const String &s);
|
* http://git.videolan.org/?p=vlc.git;a=blob;f=modules/meta_engine/taglib.cpp
|
||||||
virtual void setYear(unsigned int i);
|
* @return
|
||||||
virtual void setTrack(unsigned int i);
|
*/
|
||||||
|
PictureMap pictures() const override;
|
||||||
|
|
||||||
/*!
|
void setTitle(const String &s) override;
|
||||||
* Implements the unified tag dictionary interface -- export function.
|
void setArtist(const String &s) override;
|
||||||
* APE tags are perfectly compatible with the dictionary interface because they
|
void setAlbum(const String &s) override;
|
||||||
* support both arbitrary tag names and multiple values. Currently only
|
void setComment(const String &s) override;
|
||||||
* APE items of type *Text* are handled by the dictionary interface; all *Binary*
|
void setGenre(const String &s) override;
|
||||||
* and *Locator* items will be put into the unsupportedData list and can be
|
void setYear(unsigned int i) override;
|
||||||
* deleted on request using removeUnsupportedProperties(). The same happens
|
void setTrack(unsigned int i) override;
|
||||||
* to Text items if their key is invalid for PropertyMap (which should actually
|
void setPictures(const PictureMap &l) override;
|
||||||
* never happen).
|
|
||||||
*
|
|
||||||
* The only conversion done by this export function is to rename the APE tags
|
|
||||||
* TRACK to TRACKNUMBER, YEAR to DATE, and ALBUM ARTIST to ALBUMARTIST, respectively,
|
|
||||||
* in order to be compliant with the names used in other formats.
|
|
||||||
*/
|
|
||||||
PropertyMap properties() const;
|
|
||||||
|
|
||||||
void removeUnsupportedProperties(const StringList &properties);
|
/*!
|
||||||
|
* Implements the unified tag dictionary interface -- export function.
|
||||||
|
* APE tags are perfectly compatible with the dictionary interface because they
|
||||||
|
* support both arbitrary tag names and multiple values.
|
||||||
|
* Currently only APE items of type *Text* are handled by the dictionary interface; all *Binary*
|
||||||
|
* and *Locator* items will be put into the unsupportedData list and can be
|
||||||
|
* deleted on request using removeUnsupportedProperties().
|
||||||
|
* The same happens to Text items if their key is invalid for PropertyMap (which should actually never happen).
|
||||||
|
*
|
||||||
|
* The only conversion done by this export function is to rename the APE tags
|
||||||
|
* TRACK to TRACKNUMBER, YEAR to DATE, and ALBUM ARTIST to ALBUMARTIST,
|
||||||
|
* respectively, in order to be compliant with the names used in other formats.
|
||||||
|
*/
|
||||||
|
PropertyMap properties() const override;
|
||||||
|
|
||||||
/*!
|
void removeUnsupportedProperties(const StringList &properties) override;
|
||||||
* Implements the unified tag dictionary interface -- import function. The same
|
|
||||||
* comments as for the export function apply; additionally note that the APE tag
|
|
||||||
* specification requires keys to have between 2 and 16 printable ASCII characters
|
|
||||||
* with the exception of the fixed strings "ID3", "TAG", "OGGS", and "MP+".
|
|
||||||
*/
|
|
||||||
PropertyMap setProperties(const PropertyMap &);
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Check if the given String is a valid APE tag key.
|
* Implements the unified tag dictionary interface -- import function.
|
||||||
*/
|
* The same comments as for the export function apply; additionally note that the APE tag
|
||||||
static bool checkKey(const String&);
|
* specification requires keys to have between 2 and 16 printable ASCII characters
|
||||||
|
* with the exception of the fixed strings "ID3", "TAG", "OGGS", and "MP+".
|
||||||
|
*/
|
||||||
|
PropertyMap setProperties(const PropertyMap &) override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns a pointer to the tag's footer.
|
* Check if the given String is a valid APE tag key.
|
||||||
*/
|
*/
|
||||||
Footer *footer() const;
|
static bool checkKey(const String &);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns a reference to the item list map. This is an ItemListMap of
|
* Returns a pointer to the tag's footer.
|
||||||
* all of the items in the tag.
|
*/
|
||||||
*
|
Footer *footer() const;
|
||||||
* This is the most powerful structure for accessing the items of the tag.
|
|
||||||
*
|
|
||||||
* APE tags are case-insensitive, all keys in this map have been converted
|
|
||||||
* to upper case.
|
|
||||||
*
|
|
||||||
* \warning You should not modify this data structure directly, instead
|
|
||||||
* use setItem() and removeItem().
|
|
||||||
*/
|
|
||||||
const ItemListMap &itemListMap() const;
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Removes the \a key item from the tag
|
* Returns a reference to the item list map.
|
||||||
*/
|
* This is an ItemListMap of all of the items in the tag.
|
||||||
void removeItem(const String &key);
|
*
|
||||||
|
* This is the most powerful structure for accessing the items of the tag.
|
||||||
|
*
|
||||||
|
* APE tags are case-insensitive, all keys in this map have been converted
|
||||||
|
* to upper case.
|
||||||
|
*
|
||||||
|
* \warning You should not modify this data structure directly, instead
|
||||||
|
* use setItem() and removeItem().
|
||||||
|
*/
|
||||||
|
const ItemListMap &itemListMap() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Adds to the text item specified by \a key the data \a value. If \a replace
|
* Removes the \a key item from the tag
|
||||||
* is true, then all of the other values on the same key will be removed
|
*/
|
||||||
* first. If a binary item exists for \a key it will be removed first.
|
void removeItem(const String &key);
|
||||||
*/
|
|
||||||
void addValue(const String &key, const String &value, bool replace = true);
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Set the binary data for the key specified by \a item to \a value
|
* Adds to the text item specified by \a key the data \a value.
|
||||||
* This will convert the item to type \a Binary if it isn't already and
|
* If \a replace is true, then all of the other values on the same key will be removed first.
|
||||||
* all of the other values on the same key will be removed.
|
* If a binary item exists for \a key it will be removed first.
|
||||||
*/
|
*/
|
||||||
void setData(const String &key, const ByteVector &value);
|
void addValue(const String &key, const String &value, bool replace = true);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Sets the \a key item to the value of \a item. If an item with the \a key is already
|
* Set the binary data for the key specified by \a item to \a value
|
||||||
* present, it will be replaced.
|
* This will convert the item to type \a Binary if it isn't already and all of the other values on the same key will be removed.
|
||||||
*/
|
*/
|
||||||
void setItem(const String &key, const Item &item);
|
void setData(const String &key, const ByteVector &value);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns true if the tag does not contain any data.
|
* Sets the \a key item to the value of \a item. If an item with the \a key is already present, it will be replaced.
|
||||||
*/
|
*/
|
||||||
bool isEmpty() const;
|
void setItem(const String &key, const Item &item);
|
||||||
|
|
||||||
protected:
|
/*!
|
||||||
|
* Returns true if the tag does not contain any data.
|
||||||
|
*/
|
||||||
|
bool isEmpty() const override;
|
||||||
|
|
||||||
/*!
|
protected:
|
||||||
* Reads from the file specified in the constructor.
|
/*!
|
||||||
*/
|
* Reads from the file specified in the constructor.
|
||||||
void read();
|
*/
|
||||||
|
void read();
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Parses the body of the tag in \a data.
|
* Parses the body of the tag in \a data.
|
||||||
*/
|
*/
|
||||||
void parse(const ByteVector &data);
|
void parse(const ByteVector &data);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
Tag(const Tag &);
|
Tag(const Tag &);
|
||||||
Tag &operator=(const Tag &);
|
Tag &operator=(const Tag &);
|
||||||
|
|
||||||
class TagPrivate;
|
class TagPrivate;
|
||||||
TagPrivate *d;
|
TagPrivate *d;
|
||||||
};
|
};
|
||||||
}
|
} // namespace APE
|
||||||
}
|
} // namespace TagLib
|
||||||
}
|
} // namespace Strawberry_TagLib
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
391
3rdparty/taglib/asf/asfattribute.cpp
vendored
391
3rdparty/taglib/asf/asfattribute.cpp
vendored
@@ -23,9 +23,10 @@
|
|||||||
* http://www.mozilla.org/MPL/ *
|
* http://www.mozilla.org/MPL/ *
|
||||||
***************************************************************************/
|
***************************************************************************/
|
||||||
|
|
||||||
#include <taglib.h>
|
#include <memory>
|
||||||
#include <tdebug.h>
|
|
||||||
#include <trefcounter.h>
|
#include "taglib.h"
|
||||||
|
#include "tdebug.h"
|
||||||
|
|
||||||
#include "asfattribute.h"
|
#include "asfattribute.h"
|
||||||
#include "asffile.h"
|
#include "asffile.h"
|
||||||
@@ -33,15 +34,11 @@
|
|||||||
|
|
||||||
using namespace Strawberry_TagLib::TagLib;
|
using namespace Strawberry_TagLib::TagLib;
|
||||||
|
|
||||||
class ASF::Attribute::AttributePrivate : public RefCounter
|
namespace {
|
||||||
{
|
struct AttributeData {
|
||||||
public:
|
explicit AttributeData() : numericValue(0), stream(0), language(0) {}
|
||||||
AttributePrivate() :
|
|
||||||
pictureValue(ASF::Picture::fromInvalid()),
|
ASF::Attribute::AttributeTypes type;
|
||||||
numericValue(0),
|
|
||||||
stream(0),
|
|
||||||
language(0) {}
|
|
||||||
AttributeTypes type;
|
|
||||||
String stringValue;
|
String stringValue;
|
||||||
ByteVector byteVectorValue;
|
ByteVector byteVectorValue;
|
||||||
ASF::Picture pictureValue;
|
ASF::Picture pictureValue;
|
||||||
@@ -49,305 +46,283 @@ public:
|
|||||||
int stream;
|
int stream;
|
||||||
int language;
|
int language;
|
||||||
};
|
};
|
||||||
|
} // namespace
|
||||||
|
|
||||||
|
class ASF::Attribute::AttributePrivate {
|
||||||
|
public:
|
||||||
|
AttributePrivate() : data(new AttributeData()) {
|
||||||
|
data->pictureValue = ASF::Picture::fromInvalid();
|
||||||
|
}
|
||||||
|
|
||||||
|
std::shared_ptr<AttributeData> data;
|
||||||
|
};
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
// public members
|
// public members
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
ASF::Attribute::Attribute() :
|
ASF::Attribute::Attribute() : d(new AttributePrivate()) {
|
||||||
d(new AttributePrivate())
|
d->data->type = UnicodeType;
|
||||||
{
|
|
||||||
d->type = UnicodeType;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ASF::Attribute::Attribute(const ASF::Attribute &other) :
|
ASF::Attribute::Attribute(const ASF::Attribute &other) : d(new AttributePrivate(*other.d)) {}
|
||||||
d(other.d)
|
|
||||||
{
|
ASF::Attribute::Attribute(const String &value) : d(new AttributePrivate()) {
|
||||||
d->ref();
|
d->data->type = UnicodeType;
|
||||||
|
d->data->stringValue = value;
|
||||||
}
|
}
|
||||||
|
|
||||||
ASF::Attribute::Attribute(const String &value) :
|
ASF::Attribute::Attribute(const ByteVector &value) : d(new AttributePrivate()) {
|
||||||
d(new AttributePrivate())
|
d->data->type = BytesType;
|
||||||
{
|
d->data->byteVectorValue = value;
|
||||||
d->type = UnicodeType;
|
|
||||||
d->stringValue = value;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ASF::Attribute::Attribute(const ByteVector &value) :
|
ASF::Attribute::Attribute(const ASF::Picture &value) : d(new AttributePrivate()) {
|
||||||
d(new AttributePrivate())
|
d->data->type = BytesType;
|
||||||
{
|
d->data->pictureValue = value;
|
||||||
d->type = BytesType;
|
|
||||||
d->byteVectorValue = value;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ASF::Attribute::Attribute(const ASF::Picture &value) :
|
ASF::Attribute::Attribute(unsigned int value) : d(new AttributePrivate()) {
|
||||||
d(new AttributePrivate())
|
d->data->type = DWordType;
|
||||||
{
|
d->data->numericValue = value;
|
||||||
d->type = BytesType;
|
|
||||||
d->pictureValue = value;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ASF::Attribute::Attribute(unsigned int value) :
|
ASF::Attribute::Attribute(unsigned long long value) : d(new AttributePrivate()) {
|
||||||
d(new AttributePrivate())
|
d->data->type = QWordType;
|
||||||
{
|
d->data->numericValue = value;
|
||||||
d->type = DWordType;
|
|
||||||
d->numericValue = value;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ASF::Attribute::Attribute(unsigned long long value) :
|
ASF::Attribute::Attribute(unsigned short value) : d(new AttributePrivate()) {
|
||||||
d(new AttributePrivate())
|
d->data->type = WordType;
|
||||||
{
|
d->data->numericValue = value;
|
||||||
d->type = QWordType;
|
|
||||||
d->numericValue = value;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ASF::Attribute::Attribute(unsigned short value) :
|
ASF::Attribute::Attribute(bool value) : d(new AttributePrivate()) {
|
||||||
d(new AttributePrivate())
|
d->data->type = BoolType;
|
||||||
{
|
d->data->numericValue = value;
|
||||||
d->type = WordType;
|
|
||||||
d->numericValue = value;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ASF::Attribute::Attribute(bool value) :
|
ASF::Attribute &ASF::Attribute::operator=(const ASF::Attribute &other) {
|
||||||
d(new AttributePrivate())
|
|
||||||
{
|
|
||||||
d->type = BoolType;
|
|
||||||
d->numericValue = value;
|
|
||||||
}
|
|
||||||
|
|
||||||
ASF::Attribute &ASF::Attribute::operator=(const ASF::Attribute &other)
|
|
||||||
{
|
|
||||||
Attribute(other).swap(*this);
|
Attribute(other).swap(*this);
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
|
|
||||||
void ASF::Attribute::swap(Attribute &other)
|
void ASF::Attribute::swap(Attribute &other) {
|
||||||
{
|
|
||||||
using std::swap;
|
using std::swap;
|
||||||
|
|
||||||
swap(d, other.d);
|
swap(d, other.d);
|
||||||
}
|
}
|
||||||
|
|
||||||
ASF::Attribute::~Attribute()
|
ASF::Attribute::~Attribute() {
|
||||||
{
|
delete d;
|
||||||
if(d->deref())
|
|
||||||
delete d;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ASF::Attribute::AttributeTypes ASF::Attribute::type() const
|
ASF::Attribute::AttributeTypes ASF::Attribute::type() const {
|
||||||
{
|
return d->data->type;
|
||||||
return d->type;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
String ASF::Attribute::toString() const
|
String ASF::Attribute::toString() const {
|
||||||
{
|
return d->data->stringValue;
|
||||||
return d->stringValue;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ByteVector ASF::Attribute::toByteVector() const
|
ByteVector ASF::Attribute::toByteVector() const {
|
||||||
{
|
if (d->data->pictureValue.isValid())
|
||||||
if(d->pictureValue.isValid())
|
return d->data->pictureValue.render();
|
||||||
return d->pictureValue.render();
|
|
||||||
return d->byteVectorValue;
|
return d->data->byteVectorValue;
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned short ASF::Attribute::toBool() const
|
unsigned short ASF::Attribute::toBool() const {
|
||||||
{
|
return d->data->numericValue ? 1 : 0;
|
||||||
return d->numericValue ? 1 : 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned short ASF::Attribute::toUShort() const
|
unsigned short ASF::Attribute::toUShort() const {
|
||||||
{
|
return static_cast<unsigned short>(d->data->numericValue);
|
||||||
return static_cast<unsigned short>(d->numericValue);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned int ASF::Attribute::toUInt() const
|
unsigned int ASF::Attribute::toUInt() const {
|
||||||
{
|
return static_cast<unsigned int>(d->data->numericValue);
|
||||||
return static_cast<unsigned int>(d->numericValue);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned long long ASF::Attribute::toULongLong() const
|
unsigned long long ASF::Attribute::toULongLong() const {
|
||||||
{
|
return static_cast<unsigned long long>(d->data->numericValue);
|
||||||
return static_cast<unsigned long long>(d->numericValue);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ASF::Picture ASF::Attribute::toPicture() const
|
ASF::Picture ASF::Attribute::toPicture() const {
|
||||||
{
|
return d->data->pictureValue;
|
||||||
return d->pictureValue;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
String ASF::Attribute::parse(ASF::File &f, int kind)
|
String ASF::Attribute::parse(ASF::File &f, int kind) {
|
||||||
{
|
|
||||||
unsigned int size, nameLength;
|
unsigned int size, nameLength;
|
||||||
String name;
|
String name;
|
||||||
d->pictureValue = Picture::fromInvalid();
|
d->data->pictureValue = Picture::fromInvalid();
|
||||||
// extended content descriptor
|
// extended content descriptor
|
||||||
if(kind == 0) {
|
if (kind == 0) {
|
||||||
nameLength = readWORD(&f);
|
nameLength = readWORD(&f);
|
||||||
name = readString(&f, nameLength);
|
name = readString(&f, nameLength);
|
||||||
d->type = ASF::Attribute::AttributeTypes(readWORD(&f));
|
d->data->type = ASF::Attribute::AttributeTypes(readWORD(&f));
|
||||||
size = readWORD(&f);
|
size = readWORD(&f);
|
||||||
}
|
}
|
||||||
// metadata & metadata library
|
// metadata & metadata library
|
||||||
else {
|
else {
|
||||||
int temp = readWORD(&f);
|
int temp = readWORD(&f);
|
||||||
// metadata library
|
// metadata library
|
||||||
if(kind == 2) {
|
if (kind == 2) {
|
||||||
d->language = temp;
|
d->data->language = temp;
|
||||||
}
|
}
|
||||||
d->stream = readWORD(&f);
|
d->data->stream = readWORD(&f);
|
||||||
nameLength = readWORD(&f);
|
nameLength = readWORD(&f);
|
||||||
d->type = ASF::Attribute::AttributeTypes(readWORD(&f));
|
d->data->type = ASF::Attribute::AttributeTypes(readWORD(&f));
|
||||||
size = readDWORD(&f);
|
size = readDWORD(&f);
|
||||||
name = readString(&f, nameLength);
|
name = readString(&f, nameLength);
|
||||||
}
|
}
|
||||||
|
|
||||||
if(kind != 2 && size > 65535) {
|
if (kind != 2 && size > 65535) {
|
||||||
debug("ASF::Attribute::parse() -- Value larger than 64kB");
|
debug("ASF::Attribute::parse() -- Value larger than 64kB");
|
||||||
}
|
}
|
||||||
|
|
||||||
switch(d->type) {
|
switch (d->data->type) {
|
||||||
case WordType:
|
case WordType:
|
||||||
d->numericValue = readWORD(&f);
|
d->data->numericValue = readWORD(&f);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case BoolType:
|
case BoolType:
|
||||||
if(kind == 0) {
|
if (kind == 0) {
|
||||||
d->numericValue = (readDWORD(&f) != 0);
|
d->data->numericValue = (readDWORD(&f) != 0);
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
d->numericValue = (readWORD(&f) != 0);
|
d->data->numericValue = (readWORD(&f) != 0);
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case DWordType:
|
case DWordType:
|
||||||
d->numericValue = readDWORD(&f);
|
d->data->numericValue = readDWORD(&f);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case QWordType:
|
case QWordType:
|
||||||
d->numericValue = readQWORD(&f);
|
d->data->numericValue = readQWORD(&f);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case UnicodeType:
|
case UnicodeType:
|
||||||
d->stringValue = readString(&f, size);
|
d->data->stringValue = readString(&f, size);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case BytesType:
|
case BytesType:
|
||||||
case GuidType:
|
case GuidType:
|
||||||
d->byteVectorValue = f.readBlock(size);
|
d->data->byteVectorValue = f.readBlock(size);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
if(d->type == BytesType && name == "WM/Picture") {
|
if (d->data->type == BytesType && name == "WM/Picture") {
|
||||||
d->pictureValue.parse(d->byteVectorValue);
|
d->data->pictureValue.parse(d->data->byteVectorValue);
|
||||||
if(d->pictureValue.isValid()) {
|
if (d->data->pictureValue.isValid()) {
|
||||||
d->byteVectorValue.clear();
|
d->data->byteVectorValue.clear();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return name;
|
return name;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int ASF::Attribute::dataSize() const
|
int ASF::Attribute::dataSize() const {
|
||||||
{
|
|
||||||
switch (d->type) {
|
switch (d->data->type) {
|
||||||
case WordType:
|
case WordType:
|
||||||
return 2;
|
return 2;
|
||||||
case BoolType:
|
case BoolType:
|
||||||
return 4;
|
return 4;
|
||||||
case DWordType:
|
case DWordType:
|
||||||
return 4;
|
return 4;
|
||||||
case QWordType:
|
case QWordType:
|
||||||
return 5;
|
return 5;
|
||||||
case UnicodeType:
|
case UnicodeType:
|
||||||
return d->stringValue.size() * 2 + 2;
|
return static_cast<int>(d->data->stringValue.size() * 2 + 2);
|
||||||
case BytesType:
|
case BytesType:
|
||||||
if(d->pictureValue.isValid())
|
if (d->data->pictureValue.isValid())
|
||||||
return d->pictureValue.dataSize();
|
return d->data->pictureValue.dataSize();
|
||||||
break;
|
break;
|
||||||
case GuidType:
|
case GuidType:
|
||||||
return d->byteVectorValue.size();
|
return static_cast<int>(d->data->byteVectorValue.size());
|
||||||
}
|
}
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ByteVector ASF::Attribute::render(const String &name, int kind) const
|
ByteVector ASF::Attribute::render(const String &name, int kind) const {
|
||||||
{
|
|
||||||
ByteVector data;
|
ByteVector data;
|
||||||
|
|
||||||
switch (d->type) {
|
switch (d->data->type) {
|
||||||
case WordType:
|
case WordType:
|
||||||
data.append(ByteVector::fromShort(toUShort(), false));
|
data.append(ByteVector::fromUInt16LE(toUShort()));
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case BoolType:
|
case BoolType:
|
||||||
if(kind == 0) {
|
if (kind == 0) {
|
||||||
data.append(ByteVector::fromUInt(toBool(), false));
|
data.append(ByteVector::fromUInt32LE(toBool()));
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
data.append(ByteVector::fromShort(toBool(), false));
|
data.append(ByteVector::fromUInt16LE(toBool()));
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case DWordType:
|
case DWordType:
|
||||||
data.append(ByteVector::fromUInt(toUInt(), false));
|
data.append(ByteVector::fromUInt32LE(toUInt()));
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case QWordType:
|
case QWordType:
|
||||||
data.append(ByteVector::fromLongLong(toULongLong(), false));
|
data.append(ByteVector::fromUInt64LE(toULongLong()));
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case UnicodeType:
|
case UnicodeType:
|
||||||
data.append(renderString(d->stringValue));
|
data.append(renderString(d->data->stringValue));
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case BytesType:
|
case BytesType:
|
||||||
if(d->pictureValue.isValid()) {
|
if (d->data->pictureValue.isValid()) {
|
||||||
data.append(d->pictureValue.render());
|
data.append(d->data->pictureValue.render());
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case GuidType:
|
||||||
|
data.append(d->data->byteVectorValue);
|
||||||
break;
|
break;
|
||||||
}
|
|
||||||
break;
|
|
||||||
case GuidType:
|
|
||||||
data.append(d->byteVectorValue);
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if(kind == 0) {
|
if (kind == 0) {
|
||||||
data = renderString(name, true) +
|
data = renderString(name, true) +
|
||||||
ByteVector::fromShort((int)d->type, false) +
|
ByteVector::fromUInt16LE((int)d->data->type) +
|
||||||
ByteVector::fromShort(data.size(), false) +
|
ByteVector::fromUInt16LE(data.size()) +
|
||||||
data;
|
data;
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
ByteVector nameData = renderString(name);
|
ByteVector nameData = renderString(name);
|
||||||
data = ByteVector::fromShort(kind == 2 ? d->language : 0, false) +
|
data = ByteVector::fromUInt16LE(kind == 2 ? d->data->language : 0) +
|
||||||
ByteVector::fromShort(d->stream, false) +
|
ByteVector::fromUInt16LE(d->data->stream) +
|
||||||
ByteVector::fromShort(nameData.size(), false) +
|
ByteVector::fromUInt16LE(nameData.size()) +
|
||||||
ByteVector::fromShort((int)d->type, false) +
|
ByteVector::fromUInt16LE(static_cast<int>(d->data->type)) +
|
||||||
ByteVector::fromUInt(data.size(), false) +
|
ByteVector::fromUInt32LE(data.size()) +
|
||||||
nameData +
|
nameData +
|
||||||
data;
|
data;
|
||||||
}
|
}
|
||||||
|
|
||||||
return data;
|
return data;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int ASF::Attribute::language() const
|
int ASF::Attribute::language() const {
|
||||||
{
|
return d->data->language;
|
||||||
return d->language;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void ASF::Attribute::setLanguage(int value)
|
void ASF::Attribute::setLanguage(int value) {
|
||||||
{
|
d->data->language = value;
|
||||||
d->language = value;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int ASF::Attribute::stream() const
|
int ASF::Attribute::stream() const {
|
||||||
{
|
return d->data->stream;
|
||||||
return d->stream;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void ASF::Attribute::setStream(int value)
|
void ASF::Attribute::setStream(int value) {
|
||||||
{
|
d->data->stream = value;
|
||||||
d->stream = value;
|
|
||||||
}
|
}
|
||||||
|
|||||||
278
3rdparty/taglib/asf/asfattribute.h
vendored
278
3rdparty/taglib/asf/asfattribute.h
vendored
@@ -32,179 +32,169 @@
|
|||||||
#include "asfpicture.h"
|
#include "asfpicture.h"
|
||||||
|
|
||||||
namespace Strawberry_TagLib {
|
namespace Strawberry_TagLib {
|
||||||
namespace TagLib
|
namespace TagLib {
|
||||||
{
|
namespace ASF {
|
||||||
|
|
||||||
namespace ASF
|
class File;
|
||||||
{
|
class Picture;
|
||||||
|
|
||||||
class File;
|
class TAGLIB_EXPORT Attribute {
|
||||||
class Picture;
|
public:
|
||||||
|
/*!
|
||||||
|
* Enum of types an Attribute can have.
|
||||||
|
*/
|
||||||
|
enum AttributeTypes {
|
||||||
|
UnicodeType = 0,
|
||||||
|
BytesType = 1,
|
||||||
|
BoolType = 2,
|
||||||
|
DWordType = 3,
|
||||||
|
QWordType = 4,
|
||||||
|
WordType = 5,
|
||||||
|
GuidType = 6
|
||||||
|
};
|
||||||
|
|
||||||
class TAGLIB_EXPORT Attribute
|
/*!
|
||||||
{
|
* Constructs an empty attribute.
|
||||||
public:
|
*/
|
||||||
|
Attribute();
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Enum of types an Attribute can have.
|
* Constructs an attribute with \a key and a UnicodeType \a value.
|
||||||
*/
|
*/
|
||||||
enum AttributeTypes {
|
Attribute(const String &value);
|
||||||
UnicodeType = 0,
|
|
||||||
BytesType = 1,
|
|
||||||
BoolType = 2,
|
|
||||||
DWordType = 3,
|
|
||||||
QWordType = 4,
|
|
||||||
WordType = 5,
|
|
||||||
GuidType = 6
|
|
||||||
};
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Constructs an empty attribute.
|
* Constructs an attribute with \a key and a BytesType \a value.
|
||||||
*/
|
*/
|
||||||
Attribute();
|
Attribute(const ByteVector &value);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Constructs an attribute with \a key and a UnicodeType \a value.
|
* Constructs an attribute with \a key and a Picture \a value.
|
||||||
*/
|
*
|
||||||
Attribute(const String &value);
|
* This attribute is compatible with the ID3 frame, APIC. The ID3 specification for the APIC frame stipulates that,
|
||||||
|
* while there may be any number of APIC frames associated with a file,
|
||||||
|
* only one may be of type 1 and only one may be of type 2.
|
||||||
|
*
|
||||||
|
* The specification also states that the description of the picture can be no longer than 64 characters, but can be empty.
|
||||||
|
* WM/Picture attributes added with TagLib::ASF are not automatically validated to conform to ID3 specifications.
|
||||||
|
* You must add code in your application to perform validations if you want to maintain complete compatibility with ID3.
|
||||||
|
*/
|
||||||
|
Attribute(const Picture &value);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Constructs an attribute with \a key and a BytesType \a value.
|
* Constructs an attribute with \a key and a DWordType \a value.
|
||||||
*/
|
*/
|
||||||
Attribute(const ByteVector &value);
|
Attribute(unsigned int value);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Constructs an attribute with \a key and a Picture \a value.
|
* Constructs an attribute with \a key and a QWordType \a value.
|
||||||
*
|
*/
|
||||||
* This attribute is compatible with the ID3 frame, APIC. The ID3 specification for the APIC frame stipulates that,
|
Attribute(unsigned long long value);
|
||||||
* while there may be any number of APIC frames associated with a file,
|
|
||||||
* only one may be of type 1 and only one may be of type 2.
|
|
||||||
*
|
|
||||||
* The specification also states that the description of the picture can be no longer than 64 characters, but can be empty.
|
|
||||||
* WM/Picture attributes added with TagLib::ASF are not automatically validated to conform to ID3 specifications.
|
|
||||||
* You must add code in your application to perform validations if you want to maintain complete compatibility with ID3.
|
|
||||||
*/
|
|
||||||
Attribute(const Picture &value);
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Constructs an attribute with \a key and a DWordType \a value.
|
* Constructs an attribute with \a key and a WordType \a value.
|
||||||
*/
|
*/
|
||||||
Attribute(unsigned int value);
|
Attribute(unsigned short value);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Constructs an attribute with \a key and a QWordType \a value.
|
* Constructs an attribute with \a key and a BoolType \a value.
|
||||||
*/
|
*/
|
||||||
Attribute(unsigned long long value);
|
Attribute(bool value);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Constructs an attribute with \a key and a WordType \a value.
|
* Construct an attribute as a copy of \a other.
|
||||||
*/
|
*/
|
||||||
Attribute(unsigned short value);
|
Attribute(const Attribute &other);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Constructs an attribute with \a key and a BoolType \a value.
|
* Copies the contents of \a other into this item.
|
||||||
*/
|
*/
|
||||||
Attribute(bool value);
|
Attribute &operator=(const Attribute &other);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Construct an attribute as a copy of \a other.
|
* Exchanges the content of the Attribute by the content of \a other.
|
||||||
*/
|
*/
|
||||||
Attribute(const Attribute &item);
|
void swap(Attribute &other);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Copies the contents of \a other into this item.
|
* Destroys the attribute.
|
||||||
*/
|
*/
|
||||||
Attribute &operator=(const Attribute &other);
|
virtual ~Attribute();
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Exchanges the content of the Attribute by the content of \a other.
|
* Returns type of the value.
|
||||||
*/
|
*/
|
||||||
void swap(Attribute &other);
|
AttributeTypes type() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Destroys the attribute.
|
* Returns the BoolType \a value.
|
||||||
*/
|
*/
|
||||||
virtual ~Attribute();
|
unsigned short toBool() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns type of the value.
|
* Returns the WordType \a value.
|
||||||
*/
|
*/
|
||||||
AttributeTypes type() const;
|
unsigned short toUShort() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the BoolType \a value.
|
* Returns the DWordType \a value.
|
||||||
*/
|
*/
|
||||||
unsigned short toBool() const;
|
unsigned int toUInt() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the WordType \a value.
|
* Returns the QWordType \a value.
|
||||||
*/
|
*/
|
||||||
unsigned short toUShort() const;
|
unsigned long long toULongLong() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the DWordType \a value.
|
* Returns the UnicodeType \a value.
|
||||||
*/
|
*/
|
||||||
unsigned int toUInt() const;
|
String toString() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the QWordType \a value.
|
* Returns the BytesType \a value.
|
||||||
*/
|
*/
|
||||||
unsigned long long toULongLong() const;
|
ByteVector toByteVector() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the UnicodeType \a value.
|
* Returns the Picture \a value.
|
||||||
*/
|
*/
|
||||||
String toString() const;
|
Picture toPicture() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the BytesType \a value.
|
* Returns the language number, or 0 is no stream number was set.
|
||||||
*/
|
*/
|
||||||
ByteVector toByteVector() const;
|
int language() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the Picture \a value.
|
* Sets the language number.
|
||||||
*/
|
*/
|
||||||
Picture toPicture() const;
|
void setLanguage(int value);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the language number, or 0 is no stream number was set.
|
* Returns the stream number, or 0 is no stream number was set.
|
||||||
*/
|
*/
|
||||||
int language() const;
|
int stream() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Sets the language number.
|
* Sets the stream number.
|
||||||
*/
|
*/
|
||||||
void setLanguage(int value);
|
void setStream(int value);
|
||||||
|
|
||||||
/*!
|
//! Returns the size of the stored data
|
||||||
* Returns the stream number, or 0 is no stream number was set.
|
int dataSize() const;
|
||||||
*/
|
|
||||||
int stream() const;
|
|
||||||
|
|
||||||
/*!
|
private:
|
||||||
* Sets the stream number.
|
friend class File;
|
||||||
*/
|
|
||||||
void setStream(int value);
|
|
||||||
|
|
||||||
#ifndef DO_NOT_DOCUMENT
|
String parse(ASF::File &file, int kind = 0);
|
||||||
/* THIS IS PRIVATE, DON'T TOUCH IT! */
|
ByteVector render(const String &name, int kind = 0) const;
|
||||||
String parse(ASF::File &file, int kind = 0);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
//! Returns the size of the stored data
|
class AttributePrivate;
|
||||||
int dataSize() const;
|
AttributePrivate *d;
|
||||||
|
};
|
||||||
private:
|
} // namespace ASF
|
||||||
friend class File;
|
} // namespace TagLib
|
||||||
|
} // namespace Strawberry_TagLib
|
||||||
ByteVector render(const String &name, int kind = 0) const;
|
|
||||||
|
|
||||||
class AttributePrivate;
|
|
||||||
AttributePrivate *d;
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
556
3rdparty/taglib/asf/asffile.cpp
vendored
556
3rdparty/taglib/asf/asffile.cpp
vendored
@@ -23,11 +23,13 @@
|
|||||||
* http://www.mozilla.org/MPL/ *
|
* http://www.mozilla.org/MPL/ *
|
||||||
***************************************************************************/
|
***************************************************************************/
|
||||||
|
|
||||||
#include <tdebug.h>
|
#include <memory>
|
||||||
#include <tbytevectorlist.h>
|
|
||||||
#include <tpropertymap.h>
|
#include "tdebug.h"
|
||||||
#include <tstring.h>
|
#include "tbytevectorlist.h"
|
||||||
#include <tagutils.h>
|
#include "tpropertymap.h"
|
||||||
|
#include "tstring.h"
|
||||||
|
#include "tagutils.h"
|
||||||
|
|
||||||
#include "asffile.h"
|
#include "asffile.h"
|
||||||
#include "asftag.h"
|
#include "asftag.h"
|
||||||
@@ -36,9 +38,8 @@
|
|||||||
|
|
||||||
using namespace Strawberry_TagLib::TagLib;
|
using namespace Strawberry_TagLib::TagLib;
|
||||||
|
|
||||||
class ASF::File::FilePrivate
|
class ASF::File::FilePrivate {
|
||||||
{
|
public:
|
||||||
public:
|
|
||||||
class BaseObject;
|
class BaseObject;
|
||||||
class UnknownObject;
|
class UnknownObject;
|
||||||
class FilePropertiesObject;
|
class FilePropertiesObject;
|
||||||
@@ -50,58 +51,42 @@ public:
|
|||||||
class MetadataObject;
|
class MetadataObject;
|
||||||
class MetadataLibraryObject;
|
class MetadataLibraryObject;
|
||||||
|
|
||||||
FilePrivate():
|
typedef List<std::shared_ptr<BaseObject>> ObjectList;
|
||||||
headerSize(0),
|
typedef ObjectList::ConstIterator ObjectConstIterator;
|
||||||
tag(0),
|
|
||||||
properties(0),
|
|
||||||
contentDescriptionObject(0),
|
|
||||||
extendedContentDescriptionObject(0),
|
|
||||||
headerExtensionObject(0),
|
|
||||||
metadataObject(0),
|
|
||||||
metadataLibraryObject(0)
|
|
||||||
{
|
|
||||||
objects.setAutoDelete(true);
|
|
||||||
}
|
|
||||||
|
|
||||||
~FilePrivate()
|
FilePrivate() : headerSize(0) {}
|
||||||
{
|
|
||||||
delete tag;
|
|
||||||
delete properties;
|
|
||||||
}
|
|
||||||
|
|
||||||
unsigned long long headerSize;
|
unsigned long long headerSize;
|
||||||
|
|
||||||
ASF::Tag *tag;
|
std::unique_ptr<ASF::Tag> tag;
|
||||||
ASF::Properties *properties;
|
std::unique_ptr<ASF::AudioProperties> properties;
|
||||||
|
|
||||||
List<BaseObject *> objects;
|
ObjectList objects;
|
||||||
|
|
||||||
ContentDescriptionObject *contentDescriptionObject;
|
std::shared_ptr<ContentDescriptionObject> contentDescriptionObject;
|
||||||
ExtendedContentDescriptionObject *extendedContentDescriptionObject;
|
std::shared_ptr<ExtendedContentDescriptionObject> extendedContentDescriptionObject;
|
||||||
HeaderExtensionObject *headerExtensionObject;
|
std::shared_ptr<HeaderExtensionObject> headerExtensionObject;
|
||||||
MetadataObject *metadataObject;
|
std::shared_ptr<MetadataObject> metadataObject;
|
||||||
MetadataLibraryObject *metadataLibraryObject;
|
std::shared_ptr<MetadataLibraryObject> metadataLibraryObject;
|
||||||
};
|
};
|
||||||
|
|
||||||
namespace
|
namespace {
|
||||||
{
|
const ByteVector headerGuid("\x30\x26\xB2\x75\x8E\x66\xCF\x11\xA6\xD9\x00\xAA\x00\x62\xCE\x6C", 16);
|
||||||
const ByteVector headerGuid("\x30\x26\xB2\x75\x8E\x66\xCF\x11\xA6\xD9\x00\xAA\x00\x62\xCE\x6C", 16);
|
const ByteVector filePropertiesGuid("\xA1\xDC\xAB\x8C\x47\xA9\xCF\x11\x8E\xE4\x00\xC0\x0C\x20\x53\x65", 16);
|
||||||
const ByteVector filePropertiesGuid("\xA1\xDC\xAB\x8C\x47\xA9\xCF\x11\x8E\xE4\x00\xC0\x0C\x20\x53\x65", 16);
|
const ByteVector streamPropertiesGuid("\x91\x07\xDC\xB7\xB7\xA9\xCF\x11\x8E\xE6\x00\xC0\x0C\x20\x53\x65", 16);
|
||||||
const ByteVector streamPropertiesGuid("\x91\x07\xDC\xB7\xB7\xA9\xCF\x11\x8E\xE6\x00\xC0\x0C\x20\x53\x65", 16);
|
const ByteVector contentDescriptionGuid("\x33\x26\xB2\x75\x8E\x66\xCF\x11\xA6\xD9\x00\xAA\x00\x62\xCE\x6C", 16);
|
||||||
const ByteVector contentDescriptionGuid("\x33\x26\xB2\x75\x8E\x66\xCF\x11\xA6\xD9\x00\xAA\x00\x62\xCE\x6C", 16);
|
const ByteVector extendedContentDescriptionGuid("\x40\xA4\xD0\xD2\x07\xE3\xD2\x11\x97\xF0\x00\xA0\xC9\x5E\xA8\x50", 16);
|
||||||
const ByteVector extendedContentDescriptionGuid("\x40\xA4\xD0\xD2\x07\xE3\xD2\x11\x97\xF0\x00\xA0\xC9\x5E\xA8\x50", 16);
|
const ByteVector headerExtensionGuid("\xb5\x03\xbf_.\xa9\xcf\x11\x8e\xe3\x00\xc0\x0c Se", 16);
|
||||||
const ByteVector headerExtensionGuid("\xb5\x03\xbf_.\xa9\xcf\x11\x8e\xe3\x00\xc0\x0c Se", 16);
|
const ByteVector metadataGuid("\xEA\xCB\xF8\xC5\xAF[wH\204g\xAA\214D\xFAL\xCA", 16);
|
||||||
const ByteVector metadataGuid("\xEA\xCB\xF8\xC5\xAF[wH\204g\xAA\214D\xFAL\xCA", 16);
|
const ByteVector metadataLibraryGuid("\224\034#D\230\224\321I\241A\x1d\x13NEpT", 16);
|
||||||
const ByteVector metadataLibraryGuid("\224\034#D\230\224\321I\241A\x1d\x13NEpT", 16);
|
const ByteVector codecListGuid("\x40\x52\xd1\x86\x1d\x31\xd0\x11\xa3\xa4\x00\xa0\xc9\x03\x48\xf6", 16);
|
||||||
const ByteVector codecListGuid("\x40\x52\xd1\x86\x1d\x31\xd0\x11\xa3\xa4\x00\xa0\xc9\x03\x48\xf6", 16);
|
const ByteVector contentEncryptionGuid("\xFB\xB3\x11\x22\x23\xBD\xD2\x11\xB4\xB7\x00\xA0\xC9\x55\xFC\x6E", 16);
|
||||||
const ByteVector contentEncryptionGuid("\xFB\xB3\x11\x22\x23\xBD\xD2\x11\xB4\xB7\x00\xA0\xC9\x55\xFC\x6E", 16);
|
const ByteVector extendedContentEncryptionGuid("\x14\xE6\x8A\x29\x22\x26 \x17\x4C\xB9\x35\xDA\xE0\x7E\xE9\x28\x9C", 16);
|
||||||
const ByteVector extendedContentEncryptionGuid("\x14\xE6\x8A\x29\x22\x26 \x17\x4C\xB9\x35\xDA\xE0\x7E\xE9\x28\x9C", 16);
|
const ByteVector advancedContentEncryptionGuid("\xB6\x9B\x07\x7A\xA4\xDA\x12\x4E\xA5\xCA\x91\xD3\x8D\xC1\x1A\x8D", 16);
|
||||||
const ByteVector advancedContentEncryptionGuid("\xB6\x9B\x07\x7A\xA4\xDA\x12\x4E\xA5\xCA\x91\xD3\x8D\xC1\x1A\x8D", 16);
|
} // namespace
|
||||||
}
|
|
||||||
|
|
||||||
class ASF::File::FilePrivate::BaseObject
|
class ASF::File::FilePrivate::BaseObject {
|
||||||
{
|
public:
|
||||||
public:
|
|
||||||
ByteVector data;
|
ByteVector data;
|
||||||
virtual ~BaseObject() {}
|
virtual ~BaseObject() {}
|
||||||
virtual ByteVector guid() const = 0;
|
virtual ByteVector guid() const = 0;
|
||||||
@@ -109,355 +94,346 @@ public:
|
|||||||
virtual ByteVector render(ASF::File *file);
|
virtual ByteVector render(ASF::File *file);
|
||||||
};
|
};
|
||||||
|
|
||||||
class ASF::File::FilePrivate::UnknownObject : public ASF::File::FilePrivate::BaseObject
|
class ASF::File::FilePrivate::UnknownObject : public ASF::File::FilePrivate::BaseObject {
|
||||||
{
|
|
||||||
ByteVector myGuid;
|
ByteVector myGuid;
|
||||||
public:
|
|
||||||
|
public:
|
||||||
explicit UnknownObject(const ByteVector &guid);
|
explicit UnknownObject(const ByteVector &guid);
|
||||||
ByteVector guid() const;
|
ByteVector guid() const override;
|
||||||
};
|
};
|
||||||
|
|
||||||
class ASF::File::FilePrivate::FilePropertiesObject : public ASF::File::FilePrivate::BaseObject
|
class ASF::File::FilePrivate::FilePropertiesObject : public ASF::File::FilePrivate::BaseObject {
|
||||||
{
|
public:
|
||||||
public:
|
ByteVector guid() const override;
|
||||||
ByteVector guid() const;
|
void parse(ASF::File *file, unsigned int size) override;
|
||||||
void parse(ASF::File *file, unsigned int size);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
class ASF::File::FilePrivate::StreamPropertiesObject : public ASF::File::FilePrivate::BaseObject
|
class ASF::File::FilePrivate::StreamPropertiesObject : public ASF::File::FilePrivate::BaseObject {
|
||||||
{
|
public:
|
||||||
public:
|
ByteVector guid() const override;
|
||||||
ByteVector guid() const;
|
void parse(ASF::File *file, unsigned int size) override;
|
||||||
void parse(ASF::File *file, unsigned int size);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
class ASF::File::FilePrivate::ContentDescriptionObject : public ASF::File::FilePrivate::BaseObject
|
class ASF::File::FilePrivate::ContentDescriptionObject : public ASF::File::FilePrivate::BaseObject {
|
||||||
{
|
public:
|
||||||
public:
|
ByteVector guid() const override;
|
||||||
ByteVector guid() const;
|
void parse(ASF::File *file, unsigned int size) override;
|
||||||
void parse(ASF::File *file, unsigned int size);
|
ByteVector render(ASF::File *file) override;
|
||||||
ByteVector render(ASF::File *file);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
class ASF::File::FilePrivate::ExtendedContentDescriptionObject : public ASF::File::FilePrivate::BaseObject
|
class ASF::File::FilePrivate::ExtendedContentDescriptionObject : public ASF::File::FilePrivate::BaseObject {
|
||||||
{
|
public:
|
||||||
public:
|
|
||||||
ByteVectorList attributeData;
|
ByteVectorList attributeData;
|
||||||
ByteVector guid() const;
|
ByteVector guid() const override;
|
||||||
void parse(ASF::File *file, unsigned int size);
|
void parse(ASF::File *file, unsigned int size) override;
|
||||||
ByteVector render(ASF::File *file);
|
ByteVector render(ASF::File *file) override;
|
||||||
};
|
};
|
||||||
|
|
||||||
class ASF::File::FilePrivate::MetadataObject : public ASF::File::FilePrivate::BaseObject
|
class ASF::File::FilePrivate::MetadataObject : public ASF::File::FilePrivate::BaseObject {
|
||||||
{
|
public:
|
||||||
public:
|
|
||||||
ByteVectorList attributeData;
|
ByteVectorList attributeData;
|
||||||
ByteVector guid() const;
|
ByteVector guid() const override;
|
||||||
void parse(ASF::File *file, unsigned int size);
|
void parse(ASF::File *file, unsigned int size) override;
|
||||||
ByteVector render(ASF::File *file);
|
ByteVector render(ASF::File *file) override;
|
||||||
};
|
};
|
||||||
|
|
||||||
class ASF::File::FilePrivate::MetadataLibraryObject : public ASF::File::FilePrivate::BaseObject
|
class ASF::File::FilePrivate::MetadataLibraryObject : public ASF::File::FilePrivate::BaseObject {
|
||||||
{
|
public:
|
||||||
public:
|
|
||||||
ByteVectorList attributeData;
|
ByteVectorList attributeData;
|
||||||
ByteVector guid() const;
|
ByteVector guid() const override;
|
||||||
void parse(ASF::File *file, unsigned int size);
|
void parse(ASF::File *file, unsigned int size) override;
|
||||||
ByteVector render(ASF::File *file);
|
ByteVector render(ASF::File *file) override;
|
||||||
};
|
};
|
||||||
|
|
||||||
class ASF::File::FilePrivate::HeaderExtensionObject : public ASF::File::FilePrivate::BaseObject
|
class ASF::File::FilePrivate::HeaderExtensionObject : public ASF::File::FilePrivate::BaseObject {
|
||||||
{
|
public:
|
||||||
public:
|
ObjectList objects;
|
||||||
List<ASF::File::FilePrivate::BaseObject *> objects;
|
|
||||||
HeaderExtensionObject();
|
HeaderExtensionObject();
|
||||||
ByteVector guid() const;
|
ByteVector guid() const override;
|
||||||
void parse(ASF::File *file, unsigned int size);
|
void parse(ASF::File *file, unsigned int size) override;
|
||||||
ByteVector render(ASF::File *file);
|
ByteVector render(ASF::File *file) override;
|
||||||
};
|
};
|
||||||
|
|
||||||
class ASF::File::FilePrivate::CodecListObject : public ASF::File::FilePrivate::BaseObject
|
class ASF::File::FilePrivate::CodecListObject : public ASF::File::FilePrivate::BaseObject {
|
||||||
{
|
public:
|
||||||
public:
|
ByteVector guid() const override;
|
||||||
ByteVector guid() const;
|
void parse(ASF::File *file, unsigned int size) override;
|
||||||
void parse(ASF::File *file, unsigned int size);
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
enum CodecType
|
enum CodecType {
|
||||||
{
|
Video = 0x0001,
|
||||||
Video = 0x0001,
|
Audio = 0x0002,
|
||||||
Audio = 0x0002,
|
|
||||||
Unknown = 0xFFFF
|
Unknown = 0xFFFF
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
void ASF::File::FilePrivate::BaseObject::parse(ASF::File *file, unsigned int size)
|
void ASF::File::FilePrivate::BaseObject::parse(ASF::File *file, unsigned int size) {
|
||||||
{
|
|
||||||
data.clear();
|
data.clear();
|
||||||
if(size > 24 && size <= (unsigned int)(file->length()))
|
if (size > 24 && static_cast<long long>(size) <= file->length())
|
||||||
data = file->readBlock(size - 24);
|
data = file->readBlock(size - 24);
|
||||||
else
|
else
|
||||||
data = ByteVector();
|
data = ByteVector();
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ByteVector ASF::File::FilePrivate::BaseObject::render(ASF::File * /*file*/)
|
ByteVector ASF::File::FilePrivate::BaseObject::render(ASF::File * /*file*/) {
|
||||||
{
|
return guid() + ByteVector::fromUInt64LE(data.size() + 24) + data;
|
||||||
return guid() + ByteVector::fromLongLong(data.size() + 24, false) + data;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ASF::File::FilePrivate::UnknownObject::UnknownObject(const ByteVector &guid) : myGuid(guid)
|
ASF::File::FilePrivate::UnknownObject::UnknownObject(const ByteVector &guid) : myGuid(guid) {}
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
ByteVector ASF::File::FilePrivate::UnknownObject::guid() const
|
ByteVector ASF::File::FilePrivate::UnknownObject::guid() const {
|
||||||
{
|
|
||||||
return myGuid;
|
return myGuid;
|
||||||
}
|
}
|
||||||
|
|
||||||
ByteVector ASF::File::FilePrivate::FilePropertiesObject::guid() const
|
ByteVector ASF::File::FilePrivate::FilePropertiesObject::guid() const {
|
||||||
{
|
|
||||||
return filePropertiesGuid;
|
return filePropertiesGuid;
|
||||||
}
|
}
|
||||||
|
|
||||||
void ASF::File::FilePrivate::FilePropertiesObject::parse(ASF::File *file, unsigned int size)
|
void ASF::File::FilePrivate::FilePropertiesObject::parse(ASF::File *file, unsigned int size) {
|
||||||
{
|
|
||||||
BaseObject::parse(file, size);
|
BaseObject::parse(file, size);
|
||||||
if(data.size() < 64) {
|
if (data.size() < 64) {
|
||||||
debug("ASF::File::FilePrivate::FilePropertiesObject::parse() -- data is too short.");
|
debug("ASF::File::FilePrivate::FilePropertiesObject::parse() -- data is too short.");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
const long long duration = data.toLongLong(40, false);
|
const long long duration = data.toInt64LE(40);
|
||||||
const long long preroll = data.toLongLong(56, false);
|
const long long preroll = data.toInt64LE(56);
|
||||||
file->d->properties->setLengthInMilliseconds(static_cast<int>(duration / 10000.0 - preroll + 0.5));
|
file->d->properties->setLengthInMilliseconds(static_cast<int>(duration / 10000.0 - preroll + 0.5));
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ByteVector ASF::File::FilePrivate::StreamPropertiesObject::guid() const
|
ByteVector ASF::File::FilePrivate::StreamPropertiesObject::guid() const {
|
||||||
{
|
|
||||||
return streamPropertiesGuid;
|
return streamPropertiesGuid;
|
||||||
}
|
}
|
||||||
|
|
||||||
void ASF::File::FilePrivate::StreamPropertiesObject::parse(ASF::File *file, unsigned int size)
|
void ASF::File::FilePrivate::StreamPropertiesObject::parse(ASF::File *file, unsigned int size) {
|
||||||
{
|
|
||||||
BaseObject::parse(file, size);
|
BaseObject::parse(file, size);
|
||||||
if(data.size() < 70) {
|
if (data.size() < 70) {
|
||||||
debug("ASF::File::FilePrivate::StreamPropertiesObject::parse() -- data is too short.");
|
debug("ASF::File::FilePrivate::StreamPropertiesObject::parse() -- data is too short.");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
file->d->properties->setCodec(data.toUShort(54, false));
|
file->d->properties->setCodec(data.toUInt16LE(54));
|
||||||
file->d->properties->setChannels(data.toUShort(56, false));
|
file->d->properties->setChannels(data.toUInt16LE(56));
|
||||||
file->d->properties->setSampleRate(data.toUInt(58, false));
|
file->d->properties->setSampleRate(data.toUInt32LE(58));
|
||||||
file->d->properties->setBitrate(static_cast<int>(data.toUInt(62, false) * 8.0 / 1000.0 + 0.5));
|
file->d->properties->setBitrate(static_cast<int>(data.toUInt32LE(62) * 8.0 / 1000.0 + 0.5));
|
||||||
file->d->properties->setBitsPerSample(data.toUShort(68, false));
|
file->d->properties->setBitsPerSample(data.toUInt16LE(68));
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ByteVector ASF::File::FilePrivate::ContentDescriptionObject::guid() const
|
ByteVector ASF::File::FilePrivate::ContentDescriptionObject::guid() const {
|
||||||
{
|
|
||||||
return contentDescriptionGuid;
|
return contentDescriptionGuid;
|
||||||
}
|
}
|
||||||
|
|
||||||
void ASF::File::FilePrivate::ContentDescriptionObject::parse(ASF::File *file, unsigned int /*size*/)
|
void ASF::File::FilePrivate::ContentDescriptionObject::parse(ASF::File *file, unsigned int /*size*/) {
|
||||||
{
|
|
||||||
const int titleLength = readWORD(file);
|
const int titleLength = readWORD(file);
|
||||||
const int artistLength = readWORD(file);
|
const int artistLength = readWORD(file);
|
||||||
const int copyrightLength = readWORD(file);
|
const int copyrightLength = readWORD(file);
|
||||||
const int commentLength = readWORD(file);
|
const int commentLength = readWORD(file);
|
||||||
const int ratingLength = readWORD(file);
|
const int ratingLength = readWORD(file);
|
||||||
file->d->tag->setTitle(readString(file,titleLength));
|
file->d->tag->setTitle(readString(file, titleLength));
|
||||||
file->d->tag->setArtist(readString(file,artistLength));
|
file->d->tag->setArtist(readString(file, artistLength));
|
||||||
file->d->tag->setCopyright(readString(file,copyrightLength));
|
file->d->tag->setCopyright(readString(file, copyrightLength));
|
||||||
file->d->tag->setComment(readString(file,commentLength));
|
file->d->tag->setComment(readString(file, commentLength));
|
||||||
file->d->tag->setRating(readString(file,ratingLength));
|
file->d->tag->setRating(readString(file, ratingLength));
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ByteVector ASF::File::FilePrivate::ContentDescriptionObject::render(ASF::File *file)
|
ByteVector ASF::File::FilePrivate::ContentDescriptionObject::render(ASF::File *file) {
|
||||||
{
|
|
||||||
const ByteVector v1 = renderString(file->d->tag->title());
|
const ByteVector v1 = renderString(file->d->tag->title());
|
||||||
const ByteVector v2 = renderString(file->d->tag->artist());
|
const ByteVector v2 = renderString(file->d->tag->artist());
|
||||||
const ByteVector v3 = renderString(file->d->tag->copyright());
|
const ByteVector v3 = renderString(file->d->tag->copyright());
|
||||||
const ByteVector v4 = renderString(file->d->tag->comment());
|
const ByteVector v4 = renderString(file->d->tag->comment());
|
||||||
const ByteVector v5 = renderString(file->d->tag->rating());
|
const ByteVector v5 = renderString(file->d->tag->rating());
|
||||||
data.clear();
|
data.clear();
|
||||||
data.append(ByteVector::fromShort(v1.size(), false));
|
data.append(ByteVector::fromUInt16LE(v1.size()));
|
||||||
data.append(ByteVector::fromShort(v2.size(), false));
|
data.append(ByteVector::fromUInt16LE(v2.size()));
|
||||||
data.append(ByteVector::fromShort(v3.size(), false));
|
data.append(ByteVector::fromUInt16LE(v3.size()));
|
||||||
data.append(ByteVector::fromShort(v4.size(), false));
|
data.append(ByteVector::fromUInt16LE(v4.size()));
|
||||||
data.append(ByteVector::fromShort(v5.size(), false));
|
data.append(ByteVector::fromUInt16LE(v5.size()));
|
||||||
data.append(v1);
|
data.append(v1);
|
||||||
data.append(v2);
|
data.append(v2);
|
||||||
data.append(v3);
|
data.append(v3);
|
||||||
data.append(v4);
|
data.append(v4);
|
||||||
data.append(v5);
|
data.append(v5);
|
||||||
return BaseObject::render(file);
|
return BaseObject::render(file);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ByteVector ASF::File::FilePrivate::ExtendedContentDescriptionObject::guid() const
|
ByteVector ASF::File::FilePrivate::ExtendedContentDescriptionObject::guid() const {
|
||||||
{
|
|
||||||
return extendedContentDescriptionGuid;
|
return extendedContentDescriptionGuid;
|
||||||
}
|
}
|
||||||
|
|
||||||
void ASF::File::FilePrivate::ExtendedContentDescriptionObject::parse(ASF::File *file, unsigned int /*size*/)
|
void ASF::File::FilePrivate::ExtendedContentDescriptionObject::parse(ASF::File *file, unsigned int /*size*/) {
|
||||||
{
|
|
||||||
int count = readWORD(file);
|
int count = readWORD(file);
|
||||||
while(count--) {
|
while (count--) {
|
||||||
ASF::Attribute attribute;
|
ASF::Attribute attribute;
|
||||||
String name = attribute.parse(*file);
|
String name = attribute.parse(*file);
|
||||||
file->d->tag->addAttribute(name, attribute);
|
file->d->tag->addAttribute(name, attribute);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ByteVector ASF::File::FilePrivate::ExtendedContentDescriptionObject::render(ASF::File *file)
|
ByteVector ASF::File::FilePrivate::ExtendedContentDescriptionObject::render(ASF::File *file) {
|
||||||
{
|
|
||||||
data.clear();
|
data.clear();
|
||||||
data.append(ByteVector::fromShort(attributeData.size(), false));
|
data.append(ByteVector::fromUInt16LE(attributeData.size()));
|
||||||
data.append(attributeData.toByteVector(""));
|
data.append(attributeData.toByteVector(""));
|
||||||
return BaseObject::render(file);
|
return BaseObject::render(file);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ByteVector ASF::File::FilePrivate::MetadataObject::guid() const
|
ByteVector ASF::File::FilePrivate::MetadataObject::guid() const {
|
||||||
{
|
|
||||||
return metadataGuid;
|
return metadataGuid;
|
||||||
}
|
}
|
||||||
|
|
||||||
void ASF::File::FilePrivate::MetadataObject::parse(ASF::File *file, unsigned int /*size*/)
|
void ASF::File::FilePrivate::MetadataObject::parse(ASF::File *file, unsigned int /*size*/) {
|
||||||
{
|
|
||||||
int count = readWORD(file);
|
int count = readWORD(file);
|
||||||
while(count--) {
|
while (count--) {
|
||||||
ASF::Attribute attribute;
|
ASF::Attribute attribute;
|
||||||
String name = attribute.parse(*file, 1);
|
String name = attribute.parse(*file, 1);
|
||||||
file->d->tag->addAttribute(name, attribute);
|
file->d->tag->addAttribute(name, attribute);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ByteVector ASF::File::FilePrivate::MetadataObject::render(ASF::File *file)
|
ByteVector ASF::File::FilePrivate::MetadataObject::render(ASF::File *file) {
|
||||||
{
|
|
||||||
data.clear();
|
data.clear();
|
||||||
data.append(ByteVector::fromShort(attributeData.size(), false));
|
data.append(ByteVector::fromUInt16LE(attributeData.size()));
|
||||||
data.append(attributeData.toByteVector(""));
|
data.append(attributeData.toByteVector(""));
|
||||||
return BaseObject::render(file);
|
return BaseObject::render(file);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ByteVector ASF::File::FilePrivate::MetadataLibraryObject::guid() const
|
ByteVector ASF::File::FilePrivate::MetadataLibraryObject::guid() const {
|
||||||
{
|
|
||||||
return metadataLibraryGuid;
|
return metadataLibraryGuid;
|
||||||
}
|
}
|
||||||
|
|
||||||
void ASF::File::FilePrivate::MetadataLibraryObject::parse(ASF::File *file, unsigned int /*size*/)
|
void ASF::File::FilePrivate::MetadataLibraryObject::parse(ASF::File *file, unsigned int /*size*/) {
|
||||||
{
|
|
||||||
int count = readWORD(file);
|
int count = readWORD(file);
|
||||||
while(count--) {
|
while (count--) {
|
||||||
ASF::Attribute attribute;
|
ASF::Attribute attribute;
|
||||||
String name = attribute.parse(*file, 2);
|
String name = attribute.parse(*file, 2);
|
||||||
file->d->tag->addAttribute(name, attribute);
|
file->d->tag->addAttribute(name, attribute);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ByteVector ASF::File::FilePrivate::MetadataLibraryObject::render(ASF::File *file)
|
ByteVector ASF::File::FilePrivate::MetadataLibraryObject::render(ASF::File *file) {
|
||||||
{
|
|
||||||
data.clear();
|
data.clear();
|
||||||
data.append(ByteVector::fromShort(attributeData.size(), false));
|
data.append(ByteVector::fromUInt16LE(attributeData.size()));
|
||||||
data.append(attributeData.toByteVector(""));
|
data.append(attributeData.toByteVector(""));
|
||||||
return BaseObject::render(file);
|
return BaseObject::render(file);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ASF::File::FilePrivate::HeaderExtensionObject::HeaderExtensionObject()
|
ASF::File::FilePrivate::HeaderExtensionObject::HeaderExtensionObject() {
|
||||||
{
|
|
||||||
objects.setAutoDelete(true);
|
objects.setAutoDelete(true);
|
||||||
}
|
}
|
||||||
|
|
||||||
ByteVector ASF::File::FilePrivate::HeaderExtensionObject::guid() const
|
ByteVector ASF::File::FilePrivate::HeaderExtensionObject::guid() const {
|
||||||
{
|
|
||||||
return headerExtensionGuid;
|
return headerExtensionGuid;
|
||||||
}
|
}
|
||||||
|
|
||||||
void ASF::File::FilePrivate::HeaderExtensionObject::parse(ASF::File *file, unsigned int /*size*/)
|
void ASF::File::FilePrivate::HeaderExtensionObject::parse(ASF::File *file, unsigned int /*size*/) {
|
||||||
{
|
|
||||||
file->seek(18, File::Current);
|
file->seek(18, File::Current);
|
||||||
long long dataSize = readDWORD(file);
|
long long dataSize = readDWORD(file);
|
||||||
long long dataPos = 0;
|
long long dataPos = 0;
|
||||||
while(dataPos < dataSize) {
|
while (dataPos < dataSize) {
|
||||||
ByteVector guid = file->readBlock(16);
|
ByteVector guid = file->readBlock(16);
|
||||||
if(guid.size() != 16) {
|
if (guid.size() != 16) {
|
||||||
file->setValid(false);
|
file->setValid(false);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
bool ok;
|
bool ok;
|
||||||
long long size = readQWORD(file, &ok);
|
long long size = readQWORD(file, &ok);
|
||||||
if(!ok) {
|
if (!ok) {
|
||||||
file->setValid(false);
|
file->setValid(false);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
BaseObject *obj;
|
std::shared_ptr<BaseObject> obj;
|
||||||
if(guid == metadataGuid) {
|
if (guid == metadataGuid) {
|
||||||
file->d->metadataObject = new MetadataObject();
|
file->d->metadataObject.reset(new MetadataObject());
|
||||||
obj = file->d->metadataObject;
|
obj = file->d->metadataObject;
|
||||||
}
|
}
|
||||||
else if(guid == metadataLibraryGuid) {
|
else if (guid == metadataLibraryGuid) {
|
||||||
file->d->metadataLibraryObject = new MetadataLibraryObject();
|
file->d->metadataLibraryObject.reset(new MetadataLibraryObject());
|
||||||
obj = file->d->metadataLibraryObject;
|
obj = file->d->metadataLibraryObject;
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
obj = new UnknownObject(guid);
|
obj.reset(new UnknownObject(guid));
|
||||||
}
|
}
|
||||||
obj->parse(file, (unsigned int)size);
|
obj->parse(file, static_cast<unsigned int>(size));
|
||||||
objects.append(obj);
|
objects.append(obj);
|
||||||
dataPos += size;
|
dataPos += size;
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ByteVector ASF::File::FilePrivate::HeaderExtensionObject::render(ASF::File *file)
|
ByteVector ASF::File::FilePrivate::HeaderExtensionObject::render(ASF::File *file) {
|
||||||
{
|
|
||||||
data.clear();
|
data.clear();
|
||||||
for(List<BaseObject *>::ConstIterator it = objects.begin(); it != objects.end(); ++it) {
|
for (ObjectConstIterator it = objects.begin(); it != objects.end(); ++it) {
|
||||||
data.append((*it)->render(file));
|
data.append((*it)->render(file));
|
||||||
}
|
}
|
||||||
data = ByteVector("\x11\xD2\xD3\xAB\xBA\xA9\xcf\x11\x8E\xE6\x00\xC0\x0C\x20\x53\x65\x06\x00", 18) + ByteVector::fromUInt(data.size(), false) + data;
|
data = ByteVector("\x11\xD2\xD3\xAB\xBA\xA9\xcf\x11\x8E\xE6\x00\xC0\x0C\x20\x53\x65\x06\x00", 18) + ByteVector::fromUInt32LE(data.size()) + data;
|
||||||
return BaseObject::render(file);
|
return BaseObject::render(file);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ByteVector ASF::File::FilePrivate::CodecListObject::guid() const
|
ByteVector ASF::File::FilePrivate::CodecListObject::guid() const {
|
||||||
{
|
|
||||||
return codecListGuid;
|
return codecListGuid;
|
||||||
}
|
}
|
||||||
|
|
||||||
void ASF::File::FilePrivate::CodecListObject::parse(ASF::File *file, unsigned int size)
|
void ASF::File::FilePrivate::CodecListObject::parse(ASF::File *file, unsigned int size) {
|
||||||
{
|
|
||||||
BaseObject::parse(file, size);
|
BaseObject::parse(file, size);
|
||||||
if(data.size() <= 20) {
|
if (data.size() <= 20) {
|
||||||
debug("ASF::File::FilePrivate::CodecListObject::parse() -- data is too short.");
|
debug("ASF::File::FilePrivate::CodecListObject::parse() -- data is too short.");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned int pos = 16;
|
unsigned int pos = 16;
|
||||||
|
|
||||||
const int count = data.toUInt(pos, false);
|
const int count = data.toUInt32LE(pos);
|
||||||
pos += 4;
|
pos += 4;
|
||||||
|
|
||||||
for(int i = 0; i < count; ++i) {
|
for (int i = 0; i < count; ++i) {
|
||||||
|
|
||||||
if(pos >= data.size())
|
if (pos >= data.size())
|
||||||
break;
|
break;
|
||||||
|
|
||||||
const CodecType type = static_cast<CodecType>(data.toUShort(pos, false));
|
const CodecType type = static_cast<CodecType>(data.toUInt16LE(pos));
|
||||||
pos += 2;
|
pos += 2;
|
||||||
|
|
||||||
int nameLength = data.toUShort(pos, false);
|
int nameLength = data.toUInt16LE(pos);
|
||||||
pos += 2;
|
pos += 2;
|
||||||
|
|
||||||
const unsigned int namePos = pos;
|
const unsigned int namePos = pos;
|
||||||
pos += nameLength * 2;
|
pos += nameLength * 2;
|
||||||
|
|
||||||
const int descLength = data.toUShort(pos, false);
|
const int descLength = data.toUInt16LE(pos);
|
||||||
pos += 2;
|
pos += 2;
|
||||||
|
|
||||||
const unsigned int descPos = pos;
|
const unsigned int descPos = pos;
|
||||||
pos += descLength * 2;
|
pos += descLength * 2;
|
||||||
|
|
||||||
const int infoLength = data.toUShort(pos, false);
|
const int infoLength = data.toUInt16LE(pos);
|
||||||
pos += 2 + infoLength * 2;
|
pos += 2 + infoLength * 2;
|
||||||
|
|
||||||
if(type == CodecListObject::Audio) {
|
if (type == CodecListObject::Audio) {
|
||||||
// First audio codec found.
|
// First audio codec found.
|
||||||
|
|
||||||
const String name(data.mid(namePos, nameLength * 2), String::UTF16LE);
|
const String name(data.mid(namePos, nameLength * 2), String::UTF16LE);
|
||||||
@@ -469,100 +445,78 @@ void ASF::File::FilePrivate::CodecListObject::parse(ASF::File *file, unsigned in
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
// static members
|
// static members
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
bool ASF::File::isSupported(IOStream *stream)
|
bool ASF::File::isSupported(IOStream *stream) {
|
||||||
{
|
|
||||||
// An ASF file has to start with the designated GUID.
|
// An ASF file has to start with the designated GUID.
|
||||||
|
|
||||||
const ByteVector id = Utils::readHeader(stream, 16, false);
|
const ByteVector id = Utils::readHeader(stream, 16, false);
|
||||||
return (id == headerGuid);
|
return (id == headerGuid);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
// public members
|
// public members
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
ASF::File::File(FileName file, bool, Properties::ReadStyle) :
|
ASF::File::File(FileName file, bool, AudioProperties::ReadStyle) : Strawberry_TagLib::TagLib::File(file), d(new FilePrivate()) {
|
||||||
Strawberry_TagLib::TagLib::File(file),
|
if (isOpen())
|
||||||
d(new FilePrivate())
|
|
||||||
{
|
|
||||||
if(isOpen())
|
|
||||||
read();
|
read();
|
||||||
}
|
}
|
||||||
|
|
||||||
ASF::File::File(IOStream *stream, bool, Properties::ReadStyle) :
|
ASF::File::File(IOStream *stream, bool, AudioProperties::ReadStyle) : Strawberry_TagLib::TagLib::File(stream), d(new FilePrivate()) {
|
||||||
Strawberry_TagLib::TagLib::File(stream),
|
if (isOpen())
|
||||||
d(new FilePrivate())
|
|
||||||
{
|
|
||||||
if(isOpen())
|
|
||||||
read();
|
read();
|
||||||
}
|
}
|
||||||
|
|
||||||
ASF::File::~File()
|
ASF::File::~File() {
|
||||||
{
|
|
||||||
delete d;
|
delete d;
|
||||||
}
|
}
|
||||||
|
|
||||||
ASF::Tag *ASF::File::tag() const
|
ASF::Tag *ASF::File::tag() const {
|
||||||
{
|
return d->tag.get();
|
||||||
return d->tag;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
PropertyMap ASF::File::properties() const
|
ASF::AudioProperties *ASF::File::audioProperties() const {
|
||||||
{
|
return d->properties.get();
|
||||||
return d->tag->properties();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void ASF::File::removeUnsupportedProperties(const StringList &properties)
|
bool ASF::File::save() {
|
||||||
{
|
|
||||||
d->tag->removeUnsupportedProperties(properties);
|
|
||||||
}
|
|
||||||
|
|
||||||
PropertyMap ASF::File::setProperties(const PropertyMap &properties)
|
if (readOnly()) {
|
||||||
{
|
|
||||||
return d->tag->setProperties(properties);
|
|
||||||
}
|
|
||||||
|
|
||||||
ASF::Properties *ASF::File::audioProperties() const
|
|
||||||
{
|
|
||||||
return d->properties;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool ASF::File::save()
|
|
||||||
{
|
|
||||||
if(readOnly()) {
|
|
||||||
debug("ASF::File::save() -- File is read only.");
|
debug("ASF::File::save() -- File is read only.");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if(!isValid()) {
|
if (!isValid()) {
|
||||||
debug("ASF::File::save() -- Trying to save invalid file.");
|
debug("ASF::File::save() -- Trying to save invalid file.");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if(!d->contentDescriptionObject) {
|
if (!d->contentDescriptionObject) {
|
||||||
d->contentDescriptionObject = new FilePrivate::ContentDescriptionObject();
|
d->contentDescriptionObject.reset(new FilePrivate::ContentDescriptionObject());
|
||||||
d->objects.append(d->contentDescriptionObject);
|
d->objects.append(d->contentDescriptionObject);
|
||||||
}
|
}
|
||||||
if(!d->extendedContentDescriptionObject) {
|
if (!d->extendedContentDescriptionObject) {
|
||||||
d->extendedContentDescriptionObject = new FilePrivate::ExtendedContentDescriptionObject();
|
d->extendedContentDescriptionObject.reset(new FilePrivate::ExtendedContentDescriptionObject());
|
||||||
d->objects.append(d->extendedContentDescriptionObject);
|
d->objects.append(d->extendedContentDescriptionObject);
|
||||||
}
|
}
|
||||||
if(!d->headerExtensionObject) {
|
if (!d->headerExtensionObject) {
|
||||||
d->headerExtensionObject = new FilePrivate::HeaderExtensionObject();
|
d->headerExtensionObject.reset(new FilePrivate::HeaderExtensionObject());
|
||||||
d->objects.append(d->headerExtensionObject);
|
d->objects.append(d->headerExtensionObject);
|
||||||
}
|
}
|
||||||
if(!d->metadataObject) {
|
if (!d->metadataObject) {
|
||||||
d->metadataObject = new FilePrivate::MetadataObject();
|
d->metadataObject.reset(new FilePrivate::MetadataObject());
|
||||||
d->headerExtensionObject->objects.append(d->metadataObject);
|
d->headerExtensionObject->objects.append(d->metadataObject);
|
||||||
}
|
}
|
||||||
if(!d->metadataLibraryObject) {
|
if (!d->metadataLibraryObject) {
|
||||||
d->metadataLibraryObject = new FilePrivate::MetadataLibraryObject();
|
d->metadataLibraryObject.reset(new FilePrivate::MetadataLibraryObject());
|
||||||
d->headerExtensionObject->objects.append(d->metadataLibraryObject);
|
d->headerExtensionObject->objects.append(d->metadataLibraryObject);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -572,7 +526,7 @@ bool ASF::File::save()
|
|||||||
|
|
||||||
const AttributeListMap allAttributes = d->tag->attributeListMap();
|
const AttributeListMap allAttributes = d->tag->attributeListMap();
|
||||||
|
|
||||||
for(AttributeListMap::ConstIterator it = allAttributes.begin(); it != allAttributes.end(); ++it) {
|
for (AttributeListMap::ConstIterator it = allAttributes.begin(); it != allAttributes.end(); ++it) {
|
||||||
|
|
||||||
const String &name = it->first;
|
const String &name = it->first;
|
||||||
const AttributeList &attributes = it->second;
|
const AttributeList &attributes = it->second;
|
||||||
@@ -580,17 +534,17 @@ bool ASF::File::save()
|
|||||||
bool inExtendedContentDescriptionObject = false;
|
bool inExtendedContentDescriptionObject = false;
|
||||||
bool inMetadataObject = false;
|
bool inMetadataObject = false;
|
||||||
|
|
||||||
for(AttributeList::ConstIterator jt = attributes.begin(); jt != attributes.end(); ++jt) {
|
for (AttributeList::ConstIterator jt = attributes.begin(); jt != attributes.end(); ++jt) {
|
||||||
|
|
||||||
const Attribute &attribute = *jt;
|
const Attribute &attribute = *jt;
|
||||||
const bool largeValue = (attribute.dataSize() > 65535);
|
const bool largeValue = (attribute.dataSize() > 65535);
|
||||||
const bool guid = (attribute.type() == Attribute::GuidType);
|
const bool guid = (attribute.type() == Attribute::GuidType);
|
||||||
|
|
||||||
if(!inExtendedContentDescriptionObject && !guid && !largeValue && attribute.language() == 0 && attribute.stream() == 0) {
|
if (!inExtendedContentDescriptionObject && !guid && !largeValue && attribute.language() == 0 && attribute.stream() == 0) {
|
||||||
d->extendedContentDescriptionObject->attributeData.append(attribute.render(name));
|
d->extendedContentDescriptionObject->attributeData.append(attribute.render(name));
|
||||||
inExtendedContentDescriptionObject = true;
|
inExtendedContentDescriptionObject = true;
|
||||||
}
|
}
|
||||||
else if(!inMetadataObject && !guid && !largeValue && attribute.language() == 0 && attribute.stream() != 0) {
|
else if (!inMetadataObject && !guid && !largeValue && attribute.language() == 0 && attribute.stream() != 0) {
|
||||||
d->metadataObject->attributeData.append(attribute.render(name, 1));
|
d->metadataObject->attributeData.append(attribute.render(name, 1));
|
||||||
inMetadataObject = true;
|
inMetadataObject = true;
|
||||||
}
|
}
|
||||||
@@ -601,105 +555,107 @@ bool ASF::File::save()
|
|||||||
}
|
}
|
||||||
|
|
||||||
ByteVector data;
|
ByteVector data;
|
||||||
for(List<FilePrivate::BaseObject *>::ConstIterator it = d->objects.begin(); it != d->objects.end(); ++it) {
|
for (FilePrivate::ObjectConstIterator it = d->objects.begin(); it != d->objects.end(); ++it) {
|
||||||
data.append((*it)->render(this));
|
data.append((*it)->render(this));
|
||||||
}
|
}
|
||||||
|
|
||||||
seek(16);
|
seek(16);
|
||||||
writeBlock(ByteVector::fromLongLong(data.size() + 30, false));
|
writeBlock(ByteVector::fromUInt64LE(data.size() + 30));
|
||||||
writeBlock(ByteVector::fromUInt(d->objects.size(), false));
|
writeBlock(ByteVector::fromUInt32LE(d->objects.size()));
|
||||||
writeBlock(ByteVector("\x01\x02", 2));
|
writeBlock(ByteVector("\x01\x02", 2));
|
||||||
|
|
||||||
insert(data, 30, static_cast<unsigned long>(d->headerSize - 30));
|
insert(data, 30, static_cast<size_t>(d->headerSize - 30));
|
||||||
|
|
||||||
d->headerSize = data.size() + 30;
|
d->headerSize = data.size() + 30;
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
// private members
|
// private members
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
void ASF::File::read()
|
void ASF::File::read() {
|
||||||
{
|
|
||||||
if(!isValid())
|
if (!isValid())
|
||||||
return;
|
return;
|
||||||
|
|
||||||
if(readBlock(16) != headerGuid) {
|
if (readBlock(16) != headerGuid) {
|
||||||
debug("ASF::File::read(): Not an ASF file.");
|
debug("ASF::File::read(): Not an ASF file.");
|
||||||
setValid(false);
|
setValid(false);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
d->tag = new ASF::Tag();
|
d->tag.reset(new ASF::Tag());
|
||||||
d->properties = new ASF::Properties();
|
d->properties.reset(new ASF::AudioProperties());
|
||||||
|
|
||||||
bool ok;
|
bool ok;
|
||||||
d->headerSize = readQWORD(this, &ok);
|
d->headerSize = readQWORD(this, &ok);
|
||||||
if(!ok) {
|
if (!ok) {
|
||||||
setValid(false);
|
setValid(false);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
int numObjects = readDWORD(this, &ok);
|
int numObjects = readDWORD(this, &ok);
|
||||||
if(!ok) {
|
if (!ok) {
|
||||||
setValid(false);
|
setValid(false);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
seek(2, Current);
|
seek(2, Current);
|
||||||
|
|
||||||
FilePrivate::FilePropertiesObject *filePropertiesObject = 0;
|
std::shared_ptr<FilePrivate::FilePropertiesObject> filePropertiesObject;
|
||||||
FilePrivate::StreamPropertiesObject *streamPropertiesObject = 0;
|
std::shared_ptr<FilePrivate::StreamPropertiesObject> streamPropertiesObject;
|
||||||
for(int i = 0; i < numObjects; i++) {
|
for (int i = 0; i < numObjects; i++) {
|
||||||
const ByteVector guid = readBlock(16);
|
const ByteVector guid = readBlock(16);
|
||||||
if(guid.size() != 16) {
|
if (guid.size() != 16) {
|
||||||
setValid(false);
|
setValid(false);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
long size = (long)readQWORD(this, &ok);
|
long size = static_cast<long>(readQWORD(this, &ok));
|
||||||
if(!ok) {
|
if (!ok) {
|
||||||
setValid(false);
|
setValid(false);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
FilePrivate::BaseObject *obj;
|
std::shared_ptr<FilePrivate::BaseObject> obj;
|
||||||
if(guid == filePropertiesGuid) {
|
if (guid == filePropertiesGuid) {
|
||||||
filePropertiesObject = new FilePrivate::FilePropertiesObject();
|
filePropertiesObject.reset(new FilePrivate::FilePropertiesObject());
|
||||||
obj = filePropertiesObject;
|
obj = filePropertiesObject;
|
||||||
}
|
}
|
||||||
else if(guid == streamPropertiesGuid) {
|
else if (guid == streamPropertiesGuid) {
|
||||||
streamPropertiesObject = new FilePrivate::StreamPropertiesObject();
|
streamPropertiesObject.reset(new FilePrivate::StreamPropertiesObject());
|
||||||
obj = streamPropertiesObject;
|
obj = streamPropertiesObject;
|
||||||
}
|
}
|
||||||
else if(guid == contentDescriptionGuid) {
|
else if (guid == contentDescriptionGuid) {
|
||||||
d->contentDescriptionObject = new FilePrivate::ContentDescriptionObject();
|
d->contentDescriptionObject.reset(new FilePrivate::ContentDescriptionObject());
|
||||||
obj = d->contentDescriptionObject;
|
obj = d->contentDescriptionObject;
|
||||||
}
|
}
|
||||||
else if(guid == extendedContentDescriptionGuid) {
|
else if (guid == extendedContentDescriptionGuid) {
|
||||||
d->extendedContentDescriptionObject = new FilePrivate::ExtendedContentDescriptionObject();
|
d->extendedContentDescriptionObject.reset(new FilePrivate::ExtendedContentDescriptionObject());
|
||||||
obj = d->extendedContentDescriptionObject;
|
obj = d->extendedContentDescriptionObject;
|
||||||
}
|
}
|
||||||
else if(guid == headerExtensionGuid) {
|
else if (guid == headerExtensionGuid) {
|
||||||
d->headerExtensionObject = new FilePrivate::HeaderExtensionObject();
|
d->headerExtensionObject.reset(new FilePrivate::HeaderExtensionObject());
|
||||||
obj = d->headerExtensionObject;
|
obj = d->headerExtensionObject;
|
||||||
}
|
}
|
||||||
else if(guid == codecListGuid) {
|
else if (guid == codecListGuid) {
|
||||||
obj = new FilePrivate::CodecListObject();
|
obj.reset(new FilePrivate::CodecListObject());
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
if(guid == contentEncryptionGuid ||
|
if (guid == contentEncryptionGuid ||
|
||||||
guid == extendedContentEncryptionGuid ||
|
guid == extendedContentEncryptionGuid ||
|
||||||
guid == advancedContentEncryptionGuid) {
|
guid == advancedContentEncryptionGuid) {
|
||||||
d->properties->setEncrypted(true);
|
d->properties->setEncrypted(true);
|
||||||
}
|
}
|
||||||
obj = new FilePrivate::UnknownObject(guid);
|
obj.reset(new FilePrivate::UnknownObject(guid));
|
||||||
}
|
}
|
||||||
obj->parse(this, size);
|
obj->parse(this, size);
|
||||||
d->objects.append(obj);
|
d->objects.append(obj);
|
||||||
}
|
}
|
||||||
|
|
||||||
if(!filePropertiesObject || !streamPropertiesObject) {
|
if (!filePropertiesObject || !streamPropertiesObject) {
|
||||||
debug("ASF::File::read(): Missing mandatory header objects.");
|
debug("ASF::File::read(): Missing mandatory header objects.");
|
||||||
setValid(false);
|
setValid(false);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
158
3rdparty/taglib/asf/asffile.h
vendored
158
3rdparty/taglib/asf/asffile.h
vendored
@@ -35,106 +35,84 @@
|
|||||||
namespace Strawberry_TagLib {
|
namespace Strawberry_TagLib {
|
||||||
namespace TagLib {
|
namespace TagLib {
|
||||||
|
|
||||||
//! An implementation of ASF (WMA) metadata
|
//! An implementation of ASF (WMA) metadata
|
||||||
namespace ASF {
|
namespace ASF {
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* This implements and provides an interface for ASF files to the
|
* This implements and provides an interface for ASF files to the
|
||||||
* TagLib::Tag and TagLib::AudioProperties interfaces by way of implementing
|
* TagLib::Tag and TagLib::AudioProperties interfaces by way of implementing
|
||||||
* the abstract TagLib::File API as well as providing some additional
|
* the abstract TagLib::File API as well as providing some additional
|
||||||
* information specific to ASF files.
|
* information specific to ASF files.
|
||||||
*/
|
*/
|
||||||
class TAGLIB_EXPORT File : public Strawberry_TagLib::TagLib::File
|
class TAGLIB_EXPORT File : public Strawberry_TagLib::TagLib::File {
|
||||||
{
|
public:
|
||||||
public:
|
/*!
|
||||||
|
* Constructs an ASF file from \a file.
|
||||||
|
*
|
||||||
|
* \note In the current implementation, both \a readProperties and
|
||||||
|
* \a propertiesStyle are ignored. The audio properties are always
|
||||||
|
* read.
|
||||||
|
*/
|
||||||
|
explicit File(FileName file, bool readProperties = true, AudioProperties::ReadStyle propertiesStyle = AudioProperties::Average);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Constructs an ASF file from \a file.
|
* Constructs an ASF file from \a stream.
|
||||||
*
|
*
|
||||||
* \note In the current implementation, both \a readProperties and
|
* \note In the current implementation, both \a readProperties and
|
||||||
* \a propertiesStyle are ignored. The audio properties are always
|
* \a propertiesStyle are ignored. The audio properties are always
|
||||||
* read.
|
* read.
|
||||||
*/
|
*
|
||||||
File(FileName file, bool readProperties = true,
|
* \note TagLib will *not* take ownership of the stream, the caller is
|
||||||
Properties::ReadStyle propertiesStyle = Properties::Average);
|
* responsible for deleting it after the File object.
|
||||||
|
*/
|
||||||
|
explicit File(IOStream *stream, bool readProperties = true, AudioProperties::ReadStyle propertiesStyle = AudioProperties::Average);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Constructs an ASF file from \a stream.
|
* Destroys this instance of the File.
|
||||||
*
|
*/
|
||||||
* \note In the current implementation, both \a readProperties and
|
~File() override;
|
||||||
* \a propertiesStyle are ignored. The audio properties are always
|
|
||||||
* read.
|
|
||||||
*
|
|
||||||
* \note TagLib will *not* take ownership of the stream, the caller is
|
|
||||||
* responsible for deleting it after the File object.
|
|
||||||
*/
|
|
||||||
File(IOStream *stream, bool readProperties = true,
|
|
||||||
Properties::ReadStyle propertiesStyle = Properties::Average);
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Destroys this instance of the File.
|
* Returns a pointer to the ASF tag of the file.
|
||||||
*/
|
*
|
||||||
virtual ~File();
|
* ASF::Tag implements the tag interface, so this serves as the
|
||||||
|
* reimplementation of TagLib::File::tag().
|
||||||
|
*
|
||||||
|
* \note The Tag <b>is still</b> owned by the ASF::File and should not be
|
||||||
|
* deleted by the user. It will be deleted when the file (object) is
|
||||||
|
* destroyed.
|
||||||
|
*/
|
||||||
|
Tag *tag() const override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns a pointer to the ASF tag of the file.
|
* Returns the ASF audio properties for this file.
|
||||||
*
|
*/
|
||||||
* ASF::Tag implements the tag interface, so this serves as the
|
AudioProperties *audioProperties() const override;
|
||||||
* reimplementation of TagLib::File::tag().
|
|
||||||
*
|
|
||||||
* \note The Tag <b>is still</b> owned by the ASF::File and should not be
|
|
||||||
* deleted by the user. It will be deleted when the file (object) is
|
|
||||||
* destroyed.
|
|
||||||
*/
|
|
||||||
virtual Tag *tag() const;
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Implements the unified property interface -- export function.
|
* Save the file.
|
||||||
*/
|
*
|
||||||
PropertyMap properties() const;
|
* This returns true if the save was successful.
|
||||||
|
*/
|
||||||
|
bool save() override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Removes unsupported properties. Forwards to the actual Tag's
|
* Returns whether or not the given \a stream can be opened as an ASF file.
|
||||||
* removeUnsupportedProperties() function.
|
*
|
||||||
*/
|
* \note This method is designed to do a quick check. The result may not necessarily be correct.
|
||||||
void removeUnsupportedProperties(const StringList &properties);
|
*/
|
||||||
|
static bool isSupported(IOStream *stream);
|
||||||
|
|
||||||
/*!
|
private:
|
||||||
* Implements the unified property interface -- import function.
|
void read();
|
||||||
*/
|
|
||||||
PropertyMap setProperties(const PropertyMap &);
|
|
||||||
|
|
||||||
/*!
|
class FilePrivate;
|
||||||
* Returns the ASF audio properties for this file.
|
FilePrivate *d;
|
||||||
*/
|
};
|
||||||
virtual Properties *audioProperties() const;
|
|
||||||
|
|
||||||
/*!
|
} // namespace ASF
|
||||||
* Save the file.
|
|
||||||
*
|
|
||||||
* This returns true if the save was successful.
|
|
||||||
*/
|
|
||||||
virtual bool save();
|
|
||||||
|
|
||||||
/*!
|
} // namespace TagLib
|
||||||
* Returns whether or not the given \a stream can be opened as an ASF
|
} // namespace Strawberry_TagLib
|
||||||
* file.
|
|
||||||
*
|
|
||||||
* \note This method is designed to do a quick check. The result may
|
|
||||||
* not necessarily be correct.
|
|
||||||
*/
|
|
||||||
static bool isSupported(IOStream *stream);
|
|
||||||
|
|
||||||
private:
|
|
||||||
void read();
|
|
||||||
|
|
||||||
class FilePrivate;
|
|
||||||
FilePrivate *d;
|
|
||||||
};
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
159
3rdparty/taglib/asf/asfpicture.cpp
vendored
159
3rdparty/taglib/asf/asfpicture.cpp
vendored
@@ -23,9 +23,10 @@
|
|||||||
* http://www.mozilla.org/MPL/ *
|
* http://www.mozilla.org/MPL/ *
|
||||||
***************************************************************************/
|
***************************************************************************/
|
||||||
|
|
||||||
#include <taglib.h>
|
#include <memory>
|
||||||
#include <tdebug.h>
|
|
||||||
#include <trefcounter.h>
|
#include "taglib.h"
|
||||||
|
#include "tdebug.h"
|
||||||
|
|
||||||
#include "asfattribute.h"
|
#include "asfattribute.h"
|
||||||
#include "asffile.h"
|
#include "asffile.h"
|
||||||
@@ -34,150 +35,134 @@
|
|||||||
|
|
||||||
using namespace Strawberry_TagLib::TagLib;
|
using namespace Strawberry_TagLib::TagLib;
|
||||||
|
|
||||||
class ASF::Picture::PicturePrivate : public RefCounter
|
namespace {
|
||||||
{
|
struct PictureData {
|
||||||
public:
|
|
||||||
bool valid;
|
bool valid;
|
||||||
Type type;
|
ASF::Picture::Type type;
|
||||||
String mimeType;
|
String mimeType;
|
||||||
String description;
|
String description;
|
||||||
ByteVector picture;
|
ByteVector picture;
|
||||||
};
|
};
|
||||||
|
} // namespace
|
||||||
|
|
||||||
|
class ASF::Picture::PicturePrivate {
|
||||||
|
public:
|
||||||
|
explicit PicturePrivate() : data(new PictureData()) {}
|
||||||
|
|
||||||
|
std::shared_ptr<PictureData> data;
|
||||||
|
};
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
// Picture class members
|
// Picture class members
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
ASF::Picture::Picture() :
|
ASF::Picture::Picture() : d(new PicturePrivate()) {
|
||||||
d(new PicturePrivate())
|
d->data->valid = true;
|
||||||
{
|
|
||||||
d->valid = true;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ASF::Picture::Picture(const Picture& other) :
|
ASF::Picture::Picture(const Picture& other) : d(new PicturePrivate(*other.d)) {}
|
||||||
d(other.d)
|
|
||||||
{
|
ASF::Picture::~Picture() {
|
||||||
d->ref();
|
delete d;
|
||||||
}
|
}
|
||||||
|
|
||||||
ASF::Picture::~Picture()
|
bool ASF::Picture::isValid() const {
|
||||||
{
|
return d->data->valid;
|
||||||
if(d->deref())
|
|
||||||
delete d;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool ASF::Picture::isValid() const
|
String ASF::Picture::mimeType() const {
|
||||||
{
|
return d->data->mimeType;
|
||||||
return d->valid;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
String ASF::Picture::mimeType() const
|
void ASF::Picture::setMimeType(const String& value) {
|
||||||
{
|
d->data->mimeType = value;
|
||||||
return d->mimeType;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void ASF::Picture::setMimeType(const String &value)
|
ASF::Picture::Type ASF::Picture::type() const {
|
||||||
{
|
return d->data->type;
|
||||||
d->mimeType = value;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ASF::Picture::Type ASF::Picture::type() const
|
void ASF::Picture::setType(const ASF::Picture::Type &t) {
|
||||||
{
|
d->data->type = t;
|
||||||
return d->type;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void ASF::Picture::setType(const ASF::Picture::Type& t)
|
String ASF::Picture::description() const {
|
||||||
{
|
return d->data->description;
|
||||||
d->type = t;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
String ASF::Picture::description() const
|
void ASF::Picture::setDescription(const String& desc) {
|
||||||
{
|
d->data->description = desc;
|
||||||
return d->description;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void ASF::Picture::setDescription(const String &desc)
|
ByteVector ASF::Picture::picture() const {
|
||||||
{
|
return d->data->picture;
|
||||||
d->description = desc;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ByteVector ASF::Picture::picture() const
|
void ASF::Picture::setPicture(const ByteVector &p) {
|
||||||
{
|
d->data->picture = p;
|
||||||
return d->picture;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void ASF::Picture::setPicture(const ByteVector &p)
|
int ASF::Picture::dataSize() const {
|
||||||
{
|
return 9 + (d->data->mimeType.length() + d->data->description.length()) * 2 + d->data->picture.size();
|
||||||
d->picture = p;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int ASF::Picture::dataSize() const
|
ASF::Picture& ASF::Picture::operator=(const ASF::Picture& other) {
|
||||||
{
|
|
||||||
return
|
|
||||||
9 + (d->mimeType.length() + d->description.length()) * 2 +
|
|
||||||
d->picture.size();
|
|
||||||
}
|
|
||||||
|
|
||||||
ASF::Picture& ASF::Picture::operator=(const ASF::Picture& other)
|
|
||||||
{
|
|
||||||
Picture(other).swap(*this);
|
Picture(other).swap(*this);
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
|
|
||||||
void ASF::Picture::swap(Picture &other)
|
void ASF::Picture::swap(Picture& other) {
|
||||||
{
|
|
||||||
using std::swap;
|
using std::swap;
|
||||||
|
|
||||||
swap(d, other.d);
|
swap(d, other.d);
|
||||||
}
|
}
|
||||||
|
|
||||||
ByteVector ASF::Picture::render() const
|
ByteVector ASF::Picture::render() const {
|
||||||
{
|
|
||||||
if(!isValid())
|
if (!isValid())
|
||||||
return ByteVector();
|
return ByteVector();
|
||||||
|
|
||||||
return
|
return ByteVector(static_cast<char>(d->data->type)) + ByteVector::fromUInt32LE(d->data->picture.size()) + renderString(d->data->mimeType) + renderString(d->data->description) + d->data->picture;
|
||||||
ByteVector((char)d->type) +
|
|
||||||
ByteVector::fromUInt(d->picture.size(), false) +
|
|
||||||
renderString(d->mimeType) +
|
|
||||||
renderString(d->description) +
|
|
||||||
d->picture;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void ASF::Picture::parse(const ByteVector& bytes)
|
void ASF::Picture::parse(const ByteVector &bytes) {
|
||||||
{
|
|
||||||
d->valid = false;
|
d->data->valid = false;
|
||||||
if(bytes.size() < 9)
|
if (bytes.size() < 9)
|
||||||
return;
|
return;
|
||||||
int pos = 0;
|
size_t pos = 0;
|
||||||
d->type = (Type)bytes[0]; ++pos;
|
d->data->type = static_cast<Type>(bytes[0]);
|
||||||
const unsigned int dataLen = bytes.toUInt(pos, false); pos+=4;
|
++pos;
|
||||||
|
const unsigned int dataLen = bytes.toUInt32LE(pos);
|
||||||
|
pos += 4;
|
||||||
|
|
||||||
const ByteVector nullStringTerminator(2, 0);
|
const ByteVector nullStringTerminator(2, 0);
|
||||||
|
|
||||||
int endPos = bytes.find(nullStringTerminator, pos, 2);
|
size_t endPos = bytes.find(nullStringTerminator, pos, 2);
|
||||||
if(endPos < 0)
|
if (endPos == ByteVector::npos())
|
||||||
return;
|
return;
|
||||||
d->mimeType = String(bytes.mid(pos, endPos - pos), String::UTF16LE);
|
d->data->mimeType = String(bytes.mid(pos, endPos - pos), String::UTF16LE);
|
||||||
pos = endPos+2;
|
pos = endPos + 2;
|
||||||
|
|
||||||
endPos = bytes.find(nullStringTerminator, pos, 2);
|
endPos = bytes.find(nullStringTerminator, pos, 2);
|
||||||
if(endPos < 0)
|
if (endPos == ByteVector::npos())
|
||||||
return;
|
return;
|
||||||
d->description = String(bytes.mid(pos, endPos - pos), String::UTF16LE);
|
d->data->description = String(bytes.mid(pos, endPos - pos), String::UTF16LE);
|
||||||
pos = endPos+2;
|
pos = endPos + 2;
|
||||||
|
|
||||||
if(dataLen + pos != bytes.size())
|
if (dataLen + pos != bytes.size())
|
||||||
return;
|
return;
|
||||||
|
|
||||||
d->picture = bytes.mid(pos, dataLen);
|
d->data->picture = bytes.mid(pos, dataLen);
|
||||||
d->valid = true;
|
d->data->valid = true;
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ASF::Picture ASF::Picture::fromInvalid()
|
ASF::Picture ASF::Picture::fromInvalid() {
|
||||||
{
|
|
||||||
Picture ret;
|
Picture ret;
|
||||||
ret.d->valid = false;
|
ret.d->data->valid = false;
|
||||||
return ret;
|
return ret;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
330
3rdparty/taglib/asf/asfpicture.h
vendored
330
3rdparty/taglib/asf/asfpicture.h
vendored
@@ -32,193 +32,187 @@
|
|||||||
#include "attachedpictureframe.h"
|
#include "attachedpictureframe.h"
|
||||||
|
|
||||||
namespace Strawberry_TagLib {
|
namespace Strawberry_TagLib {
|
||||||
namespace TagLib
|
namespace TagLib {
|
||||||
{
|
namespace ASF {
|
||||||
namespace ASF
|
class Attribute;
|
||||||
{
|
|
||||||
|
|
||||||
//! An ASF attached picture interface implementation
|
//! An ASF attached picture interface implementation
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* This is an implementation of ASF attached pictures interface. Pictures may be
|
* This is an implementation of ASF attached pictures interface.
|
||||||
* included in attributes, one per WM/Picture attribute (but there may be multiple WM/Picture
|
* Pictures may be included in attributes, one per WM/Picture attribute (but there may be multiple WM/Picture attribute in a single tag).
|
||||||
* attribute in a single tag). These pictures are usually in either JPEG or
|
* These pictures are usually in either JPEG or PNG format.
|
||||||
* PNG format.
|
* \see Attribute::toPicture()
|
||||||
* \see Attribute::toPicture()
|
* \see Attribute::Attribute(const Picture& picture)
|
||||||
* \see Attribute::Attribute(const Picture& picture)
|
*/
|
||||||
*/
|
class TAGLIB_EXPORT Picture {
|
||||||
class TAGLIB_EXPORT Picture {
|
public:
|
||||||
public:
|
/*!
|
||||||
|
* This describes the function or content of the picture.
|
||||||
|
*/
|
||||||
|
enum Type {
|
||||||
|
//! A type not enumerated below
|
||||||
|
Other = 0x00,
|
||||||
|
//! 32x32 PNG image that should be used as the file icon
|
||||||
|
FileIcon = 0x01,
|
||||||
|
//! File icon of a different size or format
|
||||||
|
OtherFileIcon = 0x02,
|
||||||
|
//! Front cover image of the album
|
||||||
|
FrontCover = 0x03,
|
||||||
|
//! Back cover image of the album
|
||||||
|
BackCover = 0x04,
|
||||||
|
//! Inside leaflet page of the album
|
||||||
|
LeafletPage = 0x05,
|
||||||
|
//! Image from the album itself
|
||||||
|
Media = 0x06,
|
||||||
|
//! Picture of the lead artist or soloist
|
||||||
|
LeadArtist = 0x07,
|
||||||
|
//! Picture of the artist or performer
|
||||||
|
Artist = 0x08,
|
||||||
|
//! Picture of the conductor
|
||||||
|
Conductor = 0x09,
|
||||||
|
//! Picture of the band or orchestra
|
||||||
|
Band = 0x0A,
|
||||||
|
//! Picture of the composer
|
||||||
|
Composer = 0x0B,
|
||||||
|
//! Picture of the lyricist or text writer
|
||||||
|
Lyricist = 0x0C,
|
||||||
|
//! Picture of the recording location or studio
|
||||||
|
RecordingLocation = 0x0D,
|
||||||
|
//! Picture of the artists during recording
|
||||||
|
DuringRecording = 0x0E,
|
||||||
|
//! Picture of the artists during performance
|
||||||
|
DuringPerformance = 0x0F,
|
||||||
|
//! Picture from a movie or video related to the track
|
||||||
|
MovieScreenCapture = 0x10,
|
||||||
|
//! Picture of a large, coloured fish
|
||||||
|
ColouredFish = 0x11,
|
||||||
|
//! Illustration related to the track
|
||||||
|
Illustration = 0x12,
|
||||||
|
//! Logo of the band or performer
|
||||||
|
BandLogo = 0x13,
|
||||||
|
//! Logo of the publisher (record company)
|
||||||
|
PublisherLogo = 0x14
|
||||||
|
};
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* This describes the function or content of the picture.
|
* Constructs an empty picture.
|
||||||
*/
|
*/
|
||||||
enum Type {
|
explicit Picture();
|
||||||
//! A type not enumerated below
|
|
||||||
Other = 0x00,
|
|
||||||
//! 32x32 PNG image that should be used as the file icon
|
|
||||||
FileIcon = 0x01,
|
|
||||||
//! File icon of a different size or format
|
|
||||||
OtherFileIcon = 0x02,
|
|
||||||
//! Front cover image of the album
|
|
||||||
FrontCover = 0x03,
|
|
||||||
//! Back cover image of the album
|
|
||||||
BackCover = 0x04,
|
|
||||||
//! Inside leaflet page of the album
|
|
||||||
LeafletPage = 0x05,
|
|
||||||
//! Image from the album itself
|
|
||||||
Media = 0x06,
|
|
||||||
//! Picture of the lead artist or soloist
|
|
||||||
LeadArtist = 0x07,
|
|
||||||
//! Picture of the artist or performer
|
|
||||||
Artist = 0x08,
|
|
||||||
//! Picture of the conductor
|
|
||||||
Conductor = 0x09,
|
|
||||||
//! Picture of the band or orchestra
|
|
||||||
Band = 0x0A,
|
|
||||||
//! Picture of the composer
|
|
||||||
Composer = 0x0B,
|
|
||||||
//! Picture of the lyricist or text writer
|
|
||||||
Lyricist = 0x0C,
|
|
||||||
//! Picture of the recording location or studio
|
|
||||||
RecordingLocation = 0x0D,
|
|
||||||
//! Picture of the artists during recording
|
|
||||||
DuringRecording = 0x0E,
|
|
||||||
//! Picture of the artists during performance
|
|
||||||
DuringPerformance = 0x0F,
|
|
||||||
//! Picture from a movie or video related to the track
|
|
||||||
MovieScreenCapture = 0x10,
|
|
||||||
//! Picture of a large, coloured fish
|
|
||||||
ColouredFish = 0x11,
|
|
||||||
//! Illustration related to the track
|
|
||||||
Illustration = 0x12,
|
|
||||||
//! Logo of the band or performer
|
|
||||||
BandLogo = 0x13,
|
|
||||||
//! Logo of the publisher (record company)
|
|
||||||
PublisherLogo = 0x14
|
|
||||||
};
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Constructs an empty picture.
|
* Construct an picture as a copy of \a other.
|
||||||
*/
|
*/
|
||||||
Picture();
|
Picture(const Picture& other);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Construct an picture as a copy of \a other.
|
* Destroys the picture.
|
||||||
*/
|
*/
|
||||||
Picture(const Picture& other);
|
virtual ~Picture();
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Destroys the picture.
|
* Copies the contents of \a other into this picture.
|
||||||
*/
|
*/
|
||||||
virtual ~Picture();
|
Picture& operator=(const Picture& other);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Copies the contents of \a other into this picture.
|
* Exchanges the content of the Picture by the content of \a other.
|
||||||
*/
|
*/
|
||||||
Picture& operator=(const Picture& other);
|
void swap(Picture& other);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Exchanges the content of the Picture by the content of \a other.
|
* Returns true if Picture stores valid picture
|
||||||
*/
|
*/
|
||||||
void swap(Picture &other);
|
bool isValid() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns true if Picture stores valid picture
|
* Returns the mime type of the image. This should in most cases be "image/png" or "image/jpeg".
|
||||||
*/
|
* \see setMimeType(const String &)
|
||||||
bool isValid() const;
|
* \see picture()
|
||||||
|
* \see setPicture(const ByteArray&)
|
||||||
|
*/
|
||||||
|
String mimeType() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the mime type of the image. This should in most cases be
|
* Sets the mime type of the image. This should in most cases be "image/png" or "image/jpeg".
|
||||||
* "image/png" or "image/jpeg".
|
* \see setMimeType(const String &)
|
||||||
* \see setMimeType(const String &)
|
* \see picture()
|
||||||
* \see picture()
|
* \see setPicture(const ByteArray&)
|
||||||
* \see setPicture(const ByteArray&)
|
*/
|
||||||
*/
|
void setMimeType(const String& value);
|
||||||
String mimeType() const;
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Sets the mime type of the image. This should in most cases be
|
* Returns the type of the image.
|
||||||
* "image/png" or "image/jpeg".
|
*
|
||||||
* \see setMimeType(const String &)
|
* \see Type
|
||||||
* \see picture()
|
* \see setType()
|
||||||
* \see setPicture(const ByteArray&)
|
*/
|
||||||
*/
|
Type type() const;
|
||||||
void setMimeType(const String &value);
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the type of the image.
|
* Sets the type for the image.
|
||||||
*
|
*
|
||||||
* \see Type
|
* \see Type
|
||||||
* \see setType()
|
* \see type()
|
||||||
*/
|
*/
|
||||||
Type type() const;
|
void setType(const ASF::Picture::Type& t);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Sets the type for the image.
|
* Returns a text description of the image.
|
||||||
*
|
*
|
||||||
* \see Type
|
* \see setDescription()
|
||||||
* \see type()
|
*/
|
||||||
*/
|
String description() const;
|
||||||
void setType(const ASF::Picture::Type& t);
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns a text description of the image.
|
* Sets a textual description of the image to \a desc.
|
||||||
*
|
*
|
||||||
* \see setDescription()
|
* \see description()
|
||||||
*/
|
*/
|
||||||
String description() const;
|
void setDescription(const String& desc);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Sets a textual description of the image to \a desc.
|
* Returns the image data as a ByteVector.
|
||||||
*
|
*
|
||||||
* \see description()
|
* \note ByteVector has a data() method that returns a const char * which should make it easy to export this data to external programs.
|
||||||
*/
|
*
|
||||||
void setDescription(const String &desc);
|
* \see setPicture()
|
||||||
|
* \see mimeType()
|
||||||
|
*/
|
||||||
|
ByteVector picture() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the image data as a ByteVector.
|
* Sets the image data to \a p.
|
||||||
*
|
* \a p should be of the type specified in this frame's mime-type specification.
|
||||||
* \note ByteVector has a data() method that returns a const char * which
|
*
|
||||||
* should make it easy to export this data to external programs.
|
* \see picture()
|
||||||
*
|
* \see mimeType()
|
||||||
* \see setPicture()
|
* \see setMimeType()
|
||||||
* \see mimeType()
|
*/
|
||||||
*/
|
void setPicture(const ByteVector& p);
|
||||||
ByteVector picture() const;
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Sets the image data to \a p. \a p should be of the type specified in
|
* Returns picture as binary raw data \a value
|
||||||
* this frame's mime-type specification.
|
*/
|
||||||
*
|
ByteVector render() const;
|
||||||
* \see picture()
|
|
||||||
* \see mimeType()
|
|
||||||
* \see setMimeType()
|
|
||||||
*/
|
|
||||||
void setPicture(const ByteVector &p);
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns picture as binary raw data \a value
|
* Returns picture as binary raw data \a value
|
||||||
*/
|
*/
|
||||||
ByteVector render() const;
|
int dataSize() const;
|
||||||
|
|
||||||
/*!
|
private:
|
||||||
* Returns picture as binary raw data \a value
|
friend class Attribute;
|
||||||
*/
|
|
||||||
int dataSize() const;
|
|
||||||
|
|
||||||
#ifndef DO_NOT_DOCUMENT
|
void parse(const ByteVector&);
|
||||||
/* THIS IS PRIVATE, DON'T TOUCH IT! */
|
static Picture fromInvalid();
|
||||||
void parse(const ByteVector& );
|
|
||||||
static Picture fromInvalid();
|
|
||||||
#endif
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
class PicturePrivate;
|
class PicturePrivate;
|
||||||
PicturePrivate *d;
|
PicturePrivate* d;
|
||||||
};
|
};
|
||||||
}
|
} // namespace ASF
|
||||||
}
|
} // namespace TagLib
|
||||||
}
|
} // namespace Strawberry_TagLib
|
||||||
|
|
||||||
#endif // ASFPICTURE_H
|
#endif // ASFPICTURE_H
|
||||||
|
|||||||
136
3rdparty/taglib/asf/asfproperties.cpp
vendored
136
3rdparty/taglib/asf/asfproperties.cpp
vendored
@@ -23,30 +23,28 @@
|
|||||||
* http://www.mozilla.org/MPL/ *
|
* http://www.mozilla.org/MPL/ *
|
||||||
***************************************************************************/
|
***************************************************************************/
|
||||||
|
|
||||||
#include <tdebug.h>
|
#include "tdebug.h"
|
||||||
#include <tstring.h>
|
#include "tstring.h"
|
||||||
#include "asfproperties.h"
|
#include "asfproperties.h"
|
||||||
|
|
||||||
using namespace Strawberry_TagLib::TagLib;
|
using namespace Strawberry_TagLib::TagLib;
|
||||||
|
|
||||||
class ASF::Properties::PropertiesPrivate
|
class ASF::AudioProperties::AudioPropertiesPrivate {
|
||||||
{
|
public:
|
||||||
public:
|
explicit AudioPropertiesPrivate() : length(0),
|
||||||
PropertiesPrivate() :
|
bitrate(0),
|
||||||
length(0),
|
sampleRate(0),
|
||||||
bitrate(0),
|
channels(0),
|
||||||
sampleRate(0),
|
bitsPerSample(0),
|
||||||
channels(0),
|
codec(ASF::AudioProperties::Unknown),
|
||||||
bitsPerSample(0),
|
encrypted(false) {}
|
||||||
codec(ASF::Properties::Unknown),
|
|
||||||
encrypted(false) {}
|
|
||||||
|
|
||||||
int length;
|
int length;
|
||||||
int bitrate;
|
int bitrate;
|
||||||
int sampleRate;
|
int sampleRate;
|
||||||
int channels;
|
int channels;
|
||||||
int bitsPerSample;
|
int bitsPerSample;
|
||||||
ASF::Properties::Codec codec;
|
ASF::AudioProperties::Codec codec;
|
||||||
String codecName;
|
String codecName;
|
||||||
String codecDescription;
|
String codecDescription;
|
||||||
bool encrypted;
|
bool encrypted;
|
||||||
@@ -56,69 +54,50 @@ public:
|
|||||||
// public members
|
// public members
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
ASF::Properties::Properties() :
|
ASF::AudioProperties::AudioProperties() : Strawberry_TagLib::TagLib::AudioProperties(), d(new AudioPropertiesPrivate()) {
|
||||||
AudioProperties(AudioProperties::Average),
|
|
||||||
d(new PropertiesPrivate())
|
|
||||||
{
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ASF::Properties::~Properties()
|
ASF::AudioProperties::~AudioProperties() {
|
||||||
{
|
|
||||||
delete d;
|
delete d;
|
||||||
}
|
}
|
||||||
|
|
||||||
int ASF::Properties::length() const
|
int ASF::AudioProperties::lengthInSeconds() const {
|
||||||
{
|
|
||||||
return lengthInSeconds();
|
|
||||||
}
|
|
||||||
|
|
||||||
int ASF::Properties::lengthInSeconds() const
|
|
||||||
{
|
|
||||||
return d->length / 1000;
|
return d->length / 1000;
|
||||||
}
|
}
|
||||||
|
|
||||||
int ASF::Properties::lengthInMilliseconds() const
|
int ASF::AudioProperties::lengthInMilliseconds() const {
|
||||||
{
|
|
||||||
return d->length;
|
return d->length;
|
||||||
}
|
}
|
||||||
|
|
||||||
int ASF::Properties::bitrate() const
|
int ASF::AudioProperties::bitrate() const {
|
||||||
{
|
|
||||||
return d->bitrate;
|
return d->bitrate;
|
||||||
}
|
}
|
||||||
|
|
||||||
int ASF::Properties::sampleRate() const
|
int ASF::AudioProperties::sampleRate() const {
|
||||||
{
|
|
||||||
return d->sampleRate;
|
return d->sampleRate;
|
||||||
}
|
}
|
||||||
|
|
||||||
int ASF::Properties::channels() const
|
int ASF::AudioProperties::channels() const {
|
||||||
{
|
|
||||||
return d->channels;
|
return d->channels;
|
||||||
}
|
}
|
||||||
|
|
||||||
int ASF::Properties::bitsPerSample() const
|
int ASF::AudioProperties::bitsPerSample() const {
|
||||||
{
|
|
||||||
return d->bitsPerSample;
|
return d->bitsPerSample;
|
||||||
}
|
}
|
||||||
|
|
||||||
ASF::Properties::Codec ASF::Properties::codec() const
|
ASF::AudioProperties::Codec ASF::AudioProperties::codec() const {
|
||||||
{
|
|
||||||
return d->codec;
|
return d->codec;
|
||||||
}
|
}
|
||||||
|
|
||||||
String ASF::Properties::codecName() const
|
String ASF::AudioProperties::codecName() const {
|
||||||
{
|
|
||||||
return d->codecName;
|
return d->codecName;
|
||||||
}
|
}
|
||||||
|
|
||||||
String ASF::Properties::codecDescription() const
|
String ASF::AudioProperties::codecDescription() const {
|
||||||
{
|
|
||||||
return d->codecDescription;
|
return d->codecDescription;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool ASF::Properties::isEncrypted() const
|
bool ASF::AudioProperties::isEncrypted() const {
|
||||||
{
|
|
||||||
return d->encrypted;
|
return d->encrypted;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -126,69 +105,56 @@ bool ASF::Properties::isEncrypted() const
|
|||||||
// private members
|
// private members
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
void ASF::Properties::setLength(int /*length*/)
|
void ASF::AudioProperties::setLengthInMilliseconds(int value) {
|
||||||
{
|
|
||||||
debug("ASF::Properties::setLength() -- This method is deprecated. Do not use.");
|
|
||||||
}
|
|
||||||
|
|
||||||
void ASF::Properties::setLengthInMilliseconds(int value)
|
|
||||||
{
|
|
||||||
d->length = value;
|
d->length = value;
|
||||||
}
|
}
|
||||||
|
|
||||||
void ASF::Properties::setBitrate(int value)
|
void ASF::AudioProperties::setBitrate(int value) {
|
||||||
{
|
|
||||||
d->bitrate = value;
|
d->bitrate = value;
|
||||||
}
|
}
|
||||||
|
|
||||||
void ASF::Properties::setSampleRate(int value)
|
void ASF::AudioProperties::setSampleRate(int value) {
|
||||||
{
|
|
||||||
d->sampleRate = value;
|
d->sampleRate = value;
|
||||||
}
|
}
|
||||||
|
|
||||||
void ASF::Properties::setChannels(int value)
|
void ASF::AudioProperties::setChannels(int value) {
|
||||||
{
|
|
||||||
d->channels = value;
|
d->channels = value;
|
||||||
}
|
}
|
||||||
|
|
||||||
void ASF::Properties::setBitsPerSample(int value)
|
void ASF::AudioProperties::setBitsPerSample(int value) {
|
||||||
{
|
|
||||||
d->bitsPerSample = value;
|
d->bitsPerSample = value;
|
||||||
}
|
}
|
||||||
|
|
||||||
void ASF::Properties::setCodec(int value)
|
void ASF::AudioProperties::setCodec(int value) {
|
||||||
{
|
|
||||||
switch(value)
|
switch (value) {
|
||||||
{
|
case 0x0160:
|
||||||
case 0x0160:
|
d->codec = WMA1;
|
||||||
d->codec = WMA1;
|
break;
|
||||||
break;
|
case 0x0161:
|
||||||
case 0x0161:
|
d->codec = WMA2;
|
||||||
d->codec = WMA2;
|
break;
|
||||||
break;
|
case 0x0162:
|
||||||
case 0x0162:
|
d->codec = WMA9Pro;
|
||||||
d->codec = WMA9Pro;
|
break;
|
||||||
break;
|
case 0x0163:
|
||||||
case 0x0163:
|
d->codec = WMA9Lossless;
|
||||||
d->codec = WMA9Lossless;
|
break;
|
||||||
break;
|
default:
|
||||||
default:
|
d->codec = Unknown;
|
||||||
d->codec = Unknown;
|
break;
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void ASF::Properties::setCodecName(const String &value)
|
void ASF::AudioProperties::setCodecName(const String &value) {
|
||||||
{
|
|
||||||
d->codecName = value;
|
d->codecName = value;
|
||||||
}
|
}
|
||||||
|
|
||||||
void ASF::Properties::setCodecDescription(const String &value)
|
void ASF::AudioProperties::setCodecDescription(const String &value) {
|
||||||
{
|
|
||||||
d->codecDescription = value;
|
d->codecDescription = value;
|
||||||
}
|
}
|
||||||
|
|
||||||
void ASF::Properties::setEncrypted(bool value)
|
void ASF::AudioProperties::setEncrypted(bool value) {
|
||||||
{
|
|
||||||
d->encrypted = value;
|
d->encrypted = value;
|
||||||
}
|
}
|
||||||
|
|||||||
240
3rdparty/taglib/asf/asfproperties.h
vendored
240
3rdparty/taglib/asf/asfproperties.h
vendored
@@ -32,157 +32,137 @@
|
|||||||
|
|
||||||
namespace Strawberry_TagLib {
|
namespace Strawberry_TagLib {
|
||||||
namespace TagLib {
|
namespace TagLib {
|
||||||
|
namespace ASF {
|
||||||
|
|
||||||
namespace ASF {
|
//! An implementation of ASF audio properties
|
||||||
|
class TAGLIB_EXPORT AudioProperties : public Strawberry_TagLib::TagLib::AudioProperties {
|
||||||
|
friend class File;
|
||||||
|
|
||||||
//! An implementation of ASF audio properties
|
public:
|
||||||
class TAGLIB_EXPORT Properties : public AudioProperties
|
/*!
|
||||||
{
|
* Audio codec types can be used in ASF file.
|
||||||
public:
|
*/
|
||||||
|
enum Codec {
|
||||||
|
/*!
|
||||||
|
* Couldn't detect the codec.
|
||||||
|
*/
|
||||||
|
Unknown = 0,
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Audio codec types can be used in ASF file.
|
* Windows Media Audio 1
|
||||||
*/
|
*/
|
||||||
enum Codec
|
WMA1,
|
||||||
{
|
|
||||||
/*!
|
|
||||||
* Couldn't detect the codec.
|
|
||||||
*/
|
|
||||||
Unknown = 0,
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Windows Media Audio 1
|
* Windows Media Audio 2 or above
|
||||||
*/
|
*/
|
||||||
WMA1,
|
WMA2,
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Windows Media Audio 2 or above
|
* Windows Media Audio 9 Professional
|
||||||
*/
|
*/
|
||||||
WMA2,
|
WMA9Pro,
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Windows Media Audio 9 Professional
|
* Windows Media Audio 9 Lossless
|
||||||
*/
|
*/
|
||||||
WMA9Pro,
|
WMA9Lossless,
|
||||||
|
};
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Windows Media Audio 9 Lossless
|
* Creates an instance of ASF::Properties.
|
||||||
*/
|
*/
|
||||||
WMA9Lossless,
|
explicit AudioProperties();
|
||||||
};
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Creates an instance of ASF::Properties.
|
* Destroys this ASF::AudioProperties instance.
|
||||||
*/
|
*/
|
||||||
Properties();
|
~AudioProperties() override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Destroys this ASF::Properties instance.
|
* Returns the length of the file in seconds. The length is rounded down to
|
||||||
*/
|
* the nearest whole second.
|
||||||
virtual ~Properties();
|
*
|
||||||
|
* \see lengthInMilliseconds()
|
||||||
|
*/
|
||||||
|
int lengthInSeconds() const override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the length of the file in seconds. The length is rounded down to
|
* Returns the length of the file in milliseconds.
|
||||||
* the nearest whole second.
|
*
|
||||||
*
|
* \see lengthInSeconds()
|
||||||
* \note This method is just an alias of lengthInSeconds().
|
*/
|
||||||
*
|
int lengthInMilliseconds() const override;
|
||||||
* \deprecated
|
|
||||||
*/
|
|
||||||
virtual int length() const;
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the length of the file in seconds. The length is rounded down to
|
* Returns the average bit rate of the file in kb/s.
|
||||||
* the nearest whole second.
|
*/
|
||||||
*
|
int bitrate() const override;
|
||||||
* \see lengthInMilliseconds()
|
|
||||||
*/
|
|
||||||
// BIC: make virtual
|
|
||||||
int lengthInSeconds() const;
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the length of the file in milliseconds.
|
* Returns the sample rate in Hz.
|
||||||
*
|
*/
|
||||||
* \see lengthInSeconds()
|
int sampleRate() const override;
|
||||||
*/
|
|
||||||
// BIC: make virtual
|
|
||||||
int lengthInMilliseconds() const;
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the average bit rate of the file in kb/s.
|
* Returns the number of audio channels.
|
||||||
*/
|
*/
|
||||||
virtual int bitrate() const;
|
int channels() const override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the sample rate in Hz.
|
* Returns the number of bits per audio sample.
|
||||||
*/
|
*/
|
||||||
virtual int sampleRate() const;
|
int bitsPerSample() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the number of audio channels.
|
* Returns the codec used in the file.
|
||||||
*/
|
*
|
||||||
virtual int channels() const;
|
* \see codecName()
|
||||||
|
* \see codecDescription()
|
||||||
|
*/
|
||||||
|
Codec codec() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the number of bits per audio sample.
|
* Returns the concrete codec name, for example "Windows Media Audio 9.1" used in the file if available, otherwise an empty string.
|
||||||
*/
|
*
|
||||||
int bitsPerSample() const;
|
* \see codec()
|
||||||
|
* \see codecDescription()
|
||||||
|
*/
|
||||||
|
String codecName() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the codec used in the file.
|
* Returns the codec description, typically contains the encoder settings,
|
||||||
*
|
* for example "VBR Quality 50, 44kHz, stereo 1-pass VBR" if available, otherwise an empty string.
|
||||||
* \see codecName()
|
*
|
||||||
* \see codecDescription()
|
* \see codec()
|
||||||
*/
|
* \see codecName()
|
||||||
Codec codec() const;
|
*/
|
||||||
|
String codecDescription() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the concrete codec name, for example "Windows Media Audio 9.1"
|
* Returns whether or not the file is encrypted.
|
||||||
* used in the file if available, otherwise an empty string.
|
*/
|
||||||
*
|
bool isEncrypted() const;
|
||||||
* \see codec()
|
|
||||||
* \see codecDescription()
|
|
||||||
*/
|
|
||||||
String codecName() const;
|
|
||||||
|
|
||||||
/*!
|
private:
|
||||||
* Returns the codec description, typically contains the encoder settings,
|
void setLengthInMilliseconds(int value);
|
||||||
* for example "VBR Quality 50, 44kHz, stereo 1-pass VBR" if available,
|
void setBitrate(int value);
|
||||||
* otherwise an empty string.
|
void setSampleRate(int value);
|
||||||
*
|
void setChannels(int value);
|
||||||
* \see codec()
|
void setBitsPerSample(int value);
|
||||||
* \see codecName()
|
void setCodec(int value);
|
||||||
*/
|
void setCodecName(const String &value);
|
||||||
String codecDescription() const;
|
void setCodecDescription(const String &value);
|
||||||
|
void setEncrypted(bool value);
|
||||||
|
|
||||||
/*!
|
private:
|
||||||
* Returns whether or not the file is encrypted.
|
class AudioPropertiesPrivate;
|
||||||
*/
|
AudioPropertiesPrivate *d;
|
||||||
bool isEncrypted() const;
|
};
|
||||||
|
|
||||||
#ifndef DO_NOT_DOCUMENT
|
} // namespace ASF
|
||||||
// deprecated
|
|
||||||
void setLength(int value);
|
|
||||||
|
|
||||||
void setLengthInMilliseconds(int value);
|
} // namespace TagLib
|
||||||
void setBitrate(int value);
|
} // namespace Strawberry_TagLib
|
||||||
void setSampleRate(int value);
|
|
||||||
void setChannels(int value);
|
|
||||||
void setBitsPerSample(int value);
|
|
||||||
void setCodec(int value);
|
|
||||||
void setCodecName(const String &value);
|
|
||||||
void setCodecDescription(const String &value);
|
|
||||||
void setEncrypted(bool value);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
private:
|
|
||||||
class PropertiesPrivate;
|
|
||||||
PropertiesPrivate *d;
|
|
||||||
};
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
466
3rdparty/taglib/asf/asftag.cpp
vendored
466
3rdparty/taglib/asf/asftag.cpp
vendored
@@ -23,14 +23,14 @@
|
|||||||
* http://www.mozilla.org/MPL/ *
|
* http://www.mozilla.org/MPL/ *
|
||||||
***************************************************************************/
|
***************************************************************************/
|
||||||
|
|
||||||
#include <tpropertymap.h>
|
#include "tpicturemap.h"
|
||||||
|
#include "tpropertymap.h"
|
||||||
#include "asftag.h"
|
#include "asftag.h"
|
||||||
|
|
||||||
using namespace Strawberry_TagLib::TagLib;
|
using namespace Strawberry_TagLib::TagLib;
|
||||||
|
|
||||||
class ASF::Tag::TagPrivate
|
class ASF::Tag::TagPrivate {
|
||||||
{
|
public:
|
||||||
public:
|
|
||||||
String title;
|
String title;
|
||||||
String artist;
|
String artist;
|
||||||
String copyright;
|
String copyright;
|
||||||
@@ -39,258 +39,399 @@ public:
|
|||||||
AttributeListMap attributeListMap;
|
AttributeListMap attributeListMap;
|
||||||
};
|
};
|
||||||
|
|
||||||
ASF::Tag::Tag() :
|
ASF::Tag::Tag() : d(new TagPrivate()) {}
|
||||||
Strawberry_TagLib::TagLib::Tag(),
|
|
||||||
d(new TagPrivate())
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
ASF::Tag::~Tag()
|
ASF::Tag::~Tag() {
|
||||||
{
|
|
||||||
delete d;
|
delete d;
|
||||||
}
|
}
|
||||||
|
|
||||||
String ASF::Tag::title() const
|
String ASF::Tag::title() const {
|
||||||
{
|
|
||||||
return d->title;
|
return d->title;
|
||||||
}
|
}
|
||||||
|
|
||||||
String ASF::Tag::artist() const
|
String ASF::Tag::artist() const {
|
||||||
{
|
|
||||||
return d->artist;
|
return d->artist;
|
||||||
}
|
}
|
||||||
|
|
||||||
String ASF::Tag::album() const
|
String ASF::Tag::album() const {
|
||||||
{
|
|
||||||
if(d->attributeListMap.contains("WM/AlbumTitle"))
|
if (d->attributeListMap.contains("WM/AlbumTitle"))
|
||||||
return d->attributeListMap["WM/AlbumTitle"][0].toString();
|
return d->attributeListMap["WM/AlbumTitle"][0].toString();
|
||||||
return String();
|
return String();
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
String ASF::Tag::copyright() const
|
String ASF::Tag::copyright() const {
|
||||||
{
|
|
||||||
return d->copyright;
|
return d->copyright;
|
||||||
}
|
}
|
||||||
|
|
||||||
String ASF::Tag::comment() const
|
String ASF::Tag::comment() const {
|
||||||
{
|
|
||||||
return d->comment;
|
return d->comment;
|
||||||
}
|
}
|
||||||
|
|
||||||
String ASF::Tag::rating() const
|
String ASF::Tag::rating() const {
|
||||||
{
|
|
||||||
return d->rating;
|
return d->rating;
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned int ASF::Tag::year() const
|
unsigned int ASF::Tag::year() const {
|
||||||
{
|
|
||||||
if(d->attributeListMap.contains("WM/Year"))
|
if (d->attributeListMap.contains("WM/Year"))
|
||||||
return d->attributeListMap["WM/Year"][0].toString().toInt();
|
return d->attributeListMap["WM/Year"][0].toString().toInt();
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned int ASF::Tag::track() const
|
unsigned int ASF::Tag::track() const {
|
||||||
{
|
|
||||||
if(d->attributeListMap.contains("WM/TrackNumber")) {
|
if (d->attributeListMap.contains("WM/TrackNumber")) {
|
||||||
const ASF::Attribute attr = d->attributeListMap["WM/TrackNumber"][0];
|
const ASF::Attribute attr = d->attributeListMap["WM/TrackNumber"][0];
|
||||||
if(attr.type() == ASF::Attribute::DWordType)
|
if (attr.type() == ASF::Attribute::DWordType)
|
||||||
return attr.toUInt();
|
return attr.toUInt();
|
||||||
else
|
else
|
||||||
return attr.toString().toInt();
|
return attr.toString().toInt();
|
||||||
}
|
}
|
||||||
if(d->attributeListMap.contains("WM/Track"))
|
if (d->attributeListMap.contains("WM/Track"))
|
||||||
return d->attributeListMap["WM/Track"][0].toUInt();
|
return d->attributeListMap["WM/Track"][0].toUInt();
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
String ASF::Tag::genre() const
|
String ASF::Tag::genre() const {
|
||||||
{
|
|
||||||
if(d->attributeListMap.contains("WM/Genre"))
|
if (d->attributeListMap.contains("WM/Genre"))
|
||||||
return d->attributeListMap["WM/Genre"][0].toString();
|
return d->attributeListMap["WM/Genre"][0].toString();
|
||||||
return String();
|
return String();
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void ASF::Tag::setTitle(const String &value)
|
PictureMap ASF::Tag::pictures() const {
|
||||||
{
|
|
||||||
|
PictureMap map;
|
||||||
|
if (d->attributeListMap.contains("WM/Picture")) {
|
||||||
|
AttributeList list = d->attributeListMap["WM/Picture"];
|
||||||
|
for (AttributeList::ConstIterator it = list.begin(); it != list.end(); ++it) {
|
||||||
|
ASF::Picture asfPicture = (*it).toPicture();
|
||||||
|
TagLib::Picture::Type type;
|
||||||
|
switch (asfPicture.type()) {
|
||||||
|
case ASF::Picture::FileIcon:
|
||||||
|
type = TagLib::Picture::FileIcon;
|
||||||
|
break;
|
||||||
|
case ASF::Picture::OtherFileIcon:
|
||||||
|
type = TagLib::Picture::OtherFileIcon;
|
||||||
|
break;
|
||||||
|
case ASF::Picture::FrontCover:
|
||||||
|
type = TagLib::Picture::FrontCover;
|
||||||
|
break;
|
||||||
|
case ASF::Picture::BackCover:
|
||||||
|
type = TagLib::Picture::BackCover;
|
||||||
|
break;
|
||||||
|
case ASF::Picture::LeafletPage:
|
||||||
|
type = TagLib::Picture::LeafletPage;
|
||||||
|
break;
|
||||||
|
case ASF::Picture::Media:
|
||||||
|
type = TagLib::Picture::Media;
|
||||||
|
break;
|
||||||
|
case ASF::Picture::LeadArtist:
|
||||||
|
type = TagLib::Picture::LeadArtist;
|
||||||
|
break;
|
||||||
|
case ASF::Picture::Artist:
|
||||||
|
type = TagLib::Picture::Artist;
|
||||||
|
break;
|
||||||
|
case ASF::Picture::Conductor:
|
||||||
|
type = TagLib::Picture::Conductor;
|
||||||
|
break;
|
||||||
|
case ASF::Picture::Band:
|
||||||
|
type = TagLib::Picture::Band;
|
||||||
|
break;
|
||||||
|
case ASF::Picture::Composer:
|
||||||
|
type = TagLib::Picture::Composer;
|
||||||
|
break;
|
||||||
|
case ASF::Picture::Lyricist:
|
||||||
|
type = TagLib::Picture::Lyricist;
|
||||||
|
break;
|
||||||
|
case ASF::Picture::RecordingLocation:
|
||||||
|
type = TagLib::Picture::RecordingLocation;
|
||||||
|
break;
|
||||||
|
case ASF::Picture::DuringRecording:
|
||||||
|
type = TagLib::Picture::DuringRecording;
|
||||||
|
break;
|
||||||
|
case ASF::Picture::DuringPerformance:
|
||||||
|
type = TagLib::Picture::DuringPerformance;
|
||||||
|
break;
|
||||||
|
case ASF::Picture::MovieScreenCapture:
|
||||||
|
type = TagLib::Picture::MovieScreenCapture;
|
||||||
|
break;
|
||||||
|
case ASF::Picture::ColouredFish:
|
||||||
|
type = TagLib::Picture::ColouredFish;
|
||||||
|
break;
|
||||||
|
case ASF::Picture::Illustration:
|
||||||
|
type = TagLib::Picture::Illustration;
|
||||||
|
break;
|
||||||
|
case ASF::Picture::BandLogo:
|
||||||
|
type = TagLib::Picture::BandLogo;
|
||||||
|
break;
|
||||||
|
case ASF::Picture::PublisherLogo:
|
||||||
|
type = TagLib::Picture::PublisherLogo;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
type = TagLib::Picture::Other;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
TagLib::Picture picture(asfPicture.picture(), type, asfPicture.mimeType(), asfPicture.description());
|
||||||
|
map.insert(picture);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return PictureMap(map);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
void ASF::Tag::setTitle(const String &value) {
|
||||||
d->title = value;
|
d->title = value;
|
||||||
}
|
}
|
||||||
|
|
||||||
void ASF::Tag::setArtist(const String &value)
|
void ASF::Tag::setArtist(const String &value) {
|
||||||
{
|
|
||||||
d->artist = value;
|
d->artist = value;
|
||||||
}
|
}
|
||||||
|
|
||||||
void ASF::Tag::setCopyright(const String &value)
|
void ASF::Tag::setCopyright(const String &value) {
|
||||||
{
|
|
||||||
d->copyright = value;
|
d->copyright = value;
|
||||||
}
|
}
|
||||||
|
|
||||||
void ASF::Tag::setComment(const String &value)
|
void ASF::Tag::setComment(const String &value) {
|
||||||
{
|
|
||||||
d->comment = value;
|
d->comment = value;
|
||||||
}
|
}
|
||||||
|
|
||||||
void ASF::Tag::setRating(const String &value)
|
void ASF::Tag::setRating(const String &value) {
|
||||||
{
|
|
||||||
d->rating = value;
|
d->rating = value;
|
||||||
}
|
}
|
||||||
|
|
||||||
void ASF::Tag::setAlbum(const String &value)
|
void ASF::Tag::setAlbum(const String &value) {
|
||||||
{
|
|
||||||
setAttribute("WM/AlbumTitle", value);
|
setAttribute("WM/AlbumTitle", value);
|
||||||
}
|
}
|
||||||
|
|
||||||
void ASF::Tag::setGenre(const String &value)
|
void ASF::Tag::setGenre(const String &value) {
|
||||||
{
|
|
||||||
setAttribute("WM/Genre", value);
|
setAttribute("WM/Genre", value);
|
||||||
}
|
}
|
||||||
|
|
||||||
void ASF::Tag::setYear(unsigned int value)
|
void ASF::Tag::setYear(unsigned int value) {
|
||||||
{
|
|
||||||
setAttribute("WM/Year", String::number(value));
|
setAttribute("WM/Year", String::number(value));
|
||||||
}
|
}
|
||||||
|
|
||||||
void ASF::Tag::setTrack(unsigned int value)
|
void ASF::Tag::setTrack(unsigned int value) {
|
||||||
{
|
|
||||||
setAttribute("WM/TrackNumber", String::number(value));
|
setAttribute("WM/TrackNumber", String::number(value));
|
||||||
}
|
}
|
||||||
|
|
||||||
ASF::AttributeListMap& ASF::Tag::attributeListMap()
|
void ASF::Tag::setPictures(const PictureMap &l) {
|
||||||
{
|
|
||||||
|
removeItem("WM/Picture");
|
||||||
|
for (PictureMap::ConstIterator pictureMapIt = l.begin(); pictureMapIt != l.end(); ++pictureMapIt) {
|
||||||
|
PictureList list = pictureMapIt->second;
|
||||||
|
for (PictureList::ConstIterator pictureListIt = list.begin(); pictureListIt != list.end(); ++pictureListIt) {
|
||||||
|
const TagLib::Picture picture = (*pictureListIt);
|
||||||
|
ASF::Picture asfPicture;
|
||||||
|
asfPicture.setPicture(picture.data());
|
||||||
|
asfPicture.setMimeType(picture.mime());
|
||||||
|
asfPicture.setDescription(picture.description());
|
||||||
|
switch (picture.type()) {
|
||||||
|
case TagLib::Picture::Other:
|
||||||
|
asfPicture.setType(ASF::Picture::Other);
|
||||||
|
break;
|
||||||
|
case TagLib::Picture::FileIcon:
|
||||||
|
asfPicture.setType(ASF::Picture::FileIcon);
|
||||||
|
break;
|
||||||
|
case TagLib::Picture::OtherFileIcon:
|
||||||
|
asfPicture.setType(ASF::Picture::OtherFileIcon);
|
||||||
|
break;
|
||||||
|
case TagLib::Picture::FrontCover:
|
||||||
|
asfPicture.setType(ASF::Picture::FrontCover);
|
||||||
|
break;
|
||||||
|
case TagLib::Picture::BackCover:
|
||||||
|
asfPicture.setType(ASF::Picture::BackCover);
|
||||||
|
break;
|
||||||
|
case TagLib::Picture::LeafletPage:
|
||||||
|
asfPicture.setType(ASF::Picture::LeafletPage);
|
||||||
|
break;
|
||||||
|
case TagLib::Picture::Media:
|
||||||
|
asfPicture.setType(ASF::Picture::Media);
|
||||||
|
break;
|
||||||
|
case TagLib::Picture::LeadArtist:
|
||||||
|
asfPicture.setType(ASF::Picture::LeadArtist);
|
||||||
|
break;
|
||||||
|
case TagLib::Picture::Artist:
|
||||||
|
asfPicture.setType(ASF::Picture::Artist);
|
||||||
|
break;
|
||||||
|
case TagLib::Picture::Conductor:
|
||||||
|
asfPicture.setType(ASF::Picture::Conductor);
|
||||||
|
break;
|
||||||
|
case TagLib::Picture::Band:
|
||||||
|
asfPicture.setType(ASF::Picture::Band);
|
||||||
|
break;
|
||||||
|
case TagLib::Picture::Composer:
|
||||||
|
asfPicture.setType(ASF::Picture::Composer);
|
||||||
|
break;
|
||||||
|
case TagLib::Picture::Lyricist:
|
||||||
|
asfPicture.setType(ASF::Picture::Lyricist);
|
||||||
|
break;
|
||||||
|
case TagLib::Picture::RecordingLocation:
|
||||||
|
asfPicture.setType(ASF::Picture::RecordingLocation);
|
||||||
|
break;
|
||||||
|
case TagLib::Picture::DuringRecording:
|
||||||
|
asfPicture.setType(ASF::Picture::DuringRecording);
|
||||||
|
break;
|
||||||
|
case TagLib::Picture::DuringPerformance:
|
||||||
|
asfPicture.setType(ASF::Picture::DuringPerformance);
|
||||||
|
break;
|
||||||
|
case TagLib::Picture::MovieScreenCapture:
|
||||||
|
asfPicture.setType(ASF::Picture::MovieScreenCapture);
|
||||||
|
break;
|
||||||
|
case TagLib::Picture::ColouredFish:
|
||||||
|
asfPicture.setType(ASF::Picture::ColouredFish);
|
||||||
|
break;
|
||||||
|
case TagLib::Picture::Illustration:
|
||||||
|
asfPicture.setType(ASF::Picture::Illustration);
|
||||||
|
break;
|
||||||
|
case TagLib::Picture::BandLogo:
|
||||||
|
asfPicture.setType(ASF::Picture::BandLogo);
|
||||||
|
break;
|
||||||
|
case TagLib::Picture::PublisherLogo:
|
||||||
|
asfPicture.setType(ASF::Picture::PublisherLogo);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
addAttribute("WM/Picture", Attribute(asfPicture));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
const ASF::AttributeListMap &ASF::Tag::attributeListMap() const {
|
||||||
return d->attributeListMap;
|
return d->attributeListMap;
|
||||||
}
|
}
|
||||||
|
|
||||||
const ASF::AttributeListMap &ASF::Tag::attributeListMap() const
|
bool ASF::Tag::contains(const String &key) const {
|
||||||
{
|
|
||||||
return d->attributeListMap;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool ASF::Tag::contains(const String &key) const
|
|
||||||
{
|
|
||||||
return d->attributeListMap.contains(key);
|
return d->attributeListMap.contains(key);
|
||||||
}
|
}
|
||||||
|
|
||||||
void ASF::Tag::removeItem(const String &key)
|
void ASF::Tag::removeItem(const String &key) {
|
||||||
{
|
|
||||||
d->attributeListMap.erase(key);
|
d->attributeListMap.erase(key);
|
||||||
}
|
}
|
||||||
|
|
||||||
ASF::AttributeList ASF::Tag::attribute(const String &name) const
|
ASF::AttributeList ASF::Tag::attribute(const String &name) const {
|
||||||
{
|
|
||||||
return d->attributeListMap[name];
|
return d->attributeListMap[name];
|
||||||
}
|
}
|
||||||
|
|
||||||
void ASF::Tag::setAttribute(const String &name, const Attribute &attribute)
|
void ASF::Tag::setAttribute(const String &name, const Attribute &attribute) {
|
||||||
{
|
|
||||||
AttributeList value;
|
AttributeList value;
|
||||||
value.append(attribute);
|
value.append(attribute);
|
||||||
d->attributeListMap.insert(name, value);
|
d->attributeListMap.insert(name, value);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void ASF::Tag::setAttribute(const String &name, const AttributeList &values)
|
void ASF::Tag::setAttribute(const String &name, const AttributeList &values) {
|
||||||
{
|
|
||||||
d->attributeListMap.insert(name, values);
|
d->attributeListMap.insert(name, values);
|
||||||
}
|
}
|
||||||
|
|
||||||
void ASF::Tag::addAttribute(const String &name, const Attribute &attribute)
|
void ASF::Tag::addAttribute(const String &name, const Attribute &attribute) {
|
||||||
{
|
|
||||||
if(d->attributeListMap.contains(name)) {
|
if (d->attributeListMap.contains(name)) {
|
||||||
d->attributeListMap[name].append(attribute);
|
d->attributeListMap[name].append(attribute);
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
setAttribute(name, attribute);
|
setAttribute(name, attribute);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool ASF::Tag::isEmpty() const
|
bool ASF::Tag::isEmpty() const {
|
||||||
{
|
|
||||||
return Strawberry_TagLib::TagLib::Tag::isEmpty() &&
|
return Strawberry_TagLib::TagLib::Tag::isEmpty() &&
|
||||||
copyright().isEmpty() &&
|
copyright().isEmpty() &&
|
||||||
rating().isEmpty() &&
|
rating().isEmpty() &&
|
||||||
d->attributeListMap.isEmpty();
|
d->attributeListMap.isEmpty();
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
namespace
|
namespace {
|
||||||
{
|
const char *keyTranslation[][2] = {
|
||||||
const char *keyTranslation[][2] = {
|
{ "WM/AlbumTitle", "ALBUM" },
|
||||||
{ "WM/AlbumTitle", "ALBUM" },
|
{ "WM/AlbumArtist", "ALBUMARTIST" },
|
||||||
{ "WM/AlbumArtist", "ALBUMARTIST" },
|
{ "WM/Composer", "COMPOSER" },
|
||||||
{ "WM/Composer", "COMPOSER" },
|
{ "WM/Writer", "WRITER" },
|
||||||
{ "WM/Writer", "WRITER" },
|
{ "WM/Conductor", "CONDUCTOR" },
|
||||||
{ "WM/Conductor", "CONDUCTOR" },
|
{ "WM/ModifiedBy", "REMIXER" },
|
||||||
{ "WM/ModifiedBy", "REMIXER" },
|
{ "WM/Year", "DATE" },
|
||||||
{ "WM/Year", "DATE" },
|
{ "WM/OriginalReleaseYear", "ORIGINALDATE" },
|
||||||
{ "WM/OriginalReleaseYear", "ORIGINALDATE" },
|
{ "WM/Producer", "PRODUCER" },
|
||||||
{ "WM/Producer", "PRODUCER" },
|
{ "WM/ContentGroupDescription", "GROUPING" },
|
||||||
{ "WM/ContentGroupDescription", "GROUPING" },
|
{ "WM/SubTitle", "SUBTITLE" },
|
||||||
{ "WM/SubTitle", "SUBTITLE" },
|
{ "WM/SetSubTitle", "DISCSUBTITLE" },
|
||||||
{ "WM/SetSubTitle", "DISCSUBTITLE" },
|
{ "WM/TrackNumber", "TRACKNUMBER" },
|
||||||
{ "WM/TrackNumber", "TRACKNUMBER" },
|
{ "WM/PartOfSet", "DISCNUMBER" },
|
||||||
{ "WM/PartOfSet", "DISCNUMBER" },
|
{ "WM/Genre", "GENRE" },
|
||||||
{ "WM/Genre", "GENRE" },
|
{ "WM/BeatsPerMinute", "BPM" },
|
||||||
{ "WM/BeatsPerMinute", "BPM" },
|
{ "WM/Mood", "MOOD" },
|
||||||
{ "WM/Mood", "MOOD" },
|
{ "WM/ISRC", "ISRC" },
|
||||||
{ "WM/ISRC", "ISRC" },
|
{ "WM/Lyrics", "LYRICS" },
|
||||||
{ "WM/Lyrics", "LYRICS" },
|
{ "WM/Media", "MEDIA" },
|
||||||
{ "WM/Media", "MEDIA" },
|
{ "WM/Publisher", "LABEL" },
|
||||||
{ "WM/Publisher", "LABEL" },
|
{ "WM/CatalogNo", "CATALOGNUMBER" },
|
||||||
{ "WM/CatalogNo", "CATALOGNUMBER" },
|
{ "WM/Barcode", "BARCODE" },
|
||||||
{ "WM/Barcode", "BARCODE" },
|
{ "WM/EncodedBy", "ENCODEDBY" },
|
||||||
{ "WM/EncodedBy", "ENCODEDBY" },
|
{ "WM/AlbumSortOrder", "ALBUMSORT" },
|
||||||
{ "WM/AlbumSortOrder", "ALBUMSORT" },
|
{ "WM/AlbumArtistSortOrder", "ALBUMARTISTSORT" },
|
||||||
{ "WM/AlbumArtistSortOrder", "ALBUMARTISTSORT" },
|
{ "WM/ArtistSortOrder", "ARTISTSORT" },
|
||||||
{ "WM/ArtistSortOrder", "ARTISTSORT" },
|
{ "WM/TitleSortOrder", "TITLESORT" },
|
||||||
{ "WM/TitleSortOrder", "TITLESORT" },
|
{ "WM/Script", "SCRIPT" },
|
||||||
{ "WM/Script", "SCRIPT" },
|
{ "WM/Language", "LANGUAGE" },
|
||||||
{ "WM/Language", "LANGUAGE" },
|
{ "MusicBrainz/Track Id", "MUSICBRAINZ_TRACKID" },
|
||||||
{ "MusicBrainz/Track Id", "MUSICBRAINZ_TRACKID" },
|
{ "MusicBrainz/Artist Id", "MUSICBRAINZ_ARTISTID" },
|
||||||
{ "MusicBrainz/Artist Id", "MUSICBRAINZ_ARTISTID" },
|
{ "MusicBrainz/Album Id", "MUSICBRAINZ_ALBUMID" },
|
||||||
{ "MusicBrainz/Album Id", "MUSICBRAINZ_ALBUMID" },
|
{ "MusicBrainz/Album Artist Id", "MUSICBRAINZ_ALBUMARTISTID" },
|
||||||
{ "MusicBrainz/Album Artist Id", "MUSICBRAINZ_ALBUMARTISTID" },
|
{ "MusicBrainz/Release Group Id", "MUSICBRAINZ_RELEASEGROUPID" },
|
||||||
{ "MusicBrainz/Release Group Id", "MUSICBRAINZ_RELEASEGROUPID" },
|
{ "MusicBrainz/Work Id", "MUSICBRAINZ_WORKID" },
|
||||||
{ "MusicBrainz/Work Id", "MUSICBRAINZ_WORKID" },
|
{ "MusicIP/PUID", "MUSICIP_PUID" },
|
||||||
{ "MusicIP/PUID", "MUSICIP_PUID" },
|
{ "Acoustid/Id", "ACOUSTID_ID" },
|
||||||
{ "Acoustid/Id", "ACOUSTID_ID" },
|
{ "Acoustid/Fingerprint", "ACOUSTID_FINGERPRINT" },
|
||||||
{ "Acoustid/Fingerprint", "ACOUSTID_FINGERPRINT" },
|
};
|
||||||
};
|
const size_t keyTranslationSize = sizeof(keyTranslation) / sizeof(keyTranslation[0]);
|
||||||
const size_t keyTranslationSize = sizeof(keyTranslation) / sizeof(keyTranslation[0]);
|
|
||||||
|
|
||||||
String translateKey(const String &key)
|
String translateKey(const String &key) {
|
||||||
{
|
|
||||||
for(size_t i = 0; i < keyTranslationSize; ++i) {
|
|
||||||
if(key == keyTranslation[i][0])
|
|
||||||
return keyTranslation[i][1];
|
|
||||||
}
|
|
||||||
|
|
||||||
return String();
|
for (size_t i = 0; i < keyTranslationSize; ++i) {
|
||||||
|
if (key == keyTranslation[i][0])
|
||||||
|
return keyTranslation[i][1];
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
PropertyMap ASF::Tag::properties() const
|
return String();
|
||||||
{
|
|
||||||
|
}
|
||||||
|
} // namespace
|
||||||
|
|
||||||
|
PropertyMap ASF::Tag::properties() const {
|
||||||
|
|
||||||
PropertyMap props;
|
PropertyMap props;
|
||||||
|
|
||||||
if(!d->title.isEmpty()) {
|
if (!d->title.isEmpty()) {
|
||||||
props["TITLE"] = d->title;
|
props["TITLE"] = d->title;
|
||||||
}
|
}
|
||||||
if(!d->artist.isEmpty()) {
|
if (!d->artist.isEmpty()) {
|
||||||
props["ARTIST"] = d->artist;
|
props["ARTIST"] = d->artist;
|
||||||
}
|
}
|
||||||
if(!d->copyright.isEmpty()) {
|
if (!d->copyright.isEmpty()) {
|
||||||
props["COPYRIGHT"] = d->copyright;
|
props["COPYRIGHT"] = d->copyright;
|
||||||
}
|
}
|
||||||
if(!d->comment.isEmpty()) {
|
if (!d->comment.isEmpty()) {
|
||||||
props["COMMENT"] = d->comment;
|
props["COMMENT"] = d->comment;
|
||||||
}
|
}
|
||||||
|
|
||||||
ASF::AttributeListMap::ConstIterator it = d->attributeListMap.begin();
|
ASF::AttributeListMap::ConstIterator it = d->attributeListMap.begin();
|
||||||
for(; it != d->attributeListMap.end(); ++it) {
|
for (; it != d->attributeListMap.end(); ++it) {
|
||||||
const String key = translateKey(it->first);
|
const String key = translateKey(it->first);
|
||||||
if(!key.isEmpty()) {
|
if (!key.isEmpty()) {
|
||||||
AttributeList::ConstIterator it2 = it->second.begin();
|
AttributeList::ConstIterator it2 = it->second.begin();
|
||||||
for(; it2 != it->second.end(); ++it2) {
|
for (; it2 != it->second.end(); ++it2) {
|
||||||
if(key == "TRACKNUMBER") {
|
if (key == "TRACKNUMBER") {
|
||||||
if(it2->type() == ASF::Attribute::DWordType)
|
if (it2->type() == ASF::Attribute::DWordType)
|
||||||
props.insert(key, String::number(it2->toUInt()));
|
props.insert(key, String::number(it2->toUInt()));
|
||||||
else
|
else
|
||||||
props.insert(key, it2->toString());
|
props.insert(key, it2->toString());
|
||||||
@@ -305,39 +446,41 @@ PropertyMap ASF::Tag::properties() const
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
return props;
|
return props;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void ASF::Tag::removeUnsupportedProperties(const StringList &props)
|
void ASF::Tag::removeUnsupportedProperties(const StringList &props) {
|
||||||
{
|
|
||||||
StringList::ConstIterator it = props.begin();
|
StringList::ConstIterator it = props.begin();
|
||||||
for(; it != props.end(); ++it)
|
for (; it != props.end(); ++it)
|
||||||
d->attributeListMap.erase(*it);
|
d->attributeListMap.erase(*it);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
PropertyMap ASF::Tag::setProperties(const PropertyMap &props)
|
PropertyMap ASF::Tag::setProperties(const PropertyMap &props) {
|
||||||
{
|
|
||||||
static Map<String, String> reverseKeyMap;
|
static Map<String, String> reverseKeyMap;
|
||||||
if(reverseKeyMap.isEmpty()) {
|
if (reverseKeyMap.isEmpty()) {
|
||||||
int numKeys = sizeof(keyTranslation) / sizeof(keyTranslation[0]);
|
int numKeys = sizeof(keyTranslation) / sizeof(keyTranslation[0]);
|
||||||
for(int i = 0; i < numKeys; i++) {
|
for (int i = 0; i < numKeys; i++) {
|
||||||
reverseKeyMap[keyTranslation[i][1]] = keyTranslation[i][0];
|
reverseKeyMap[keyTranslation[i][1]] = keyTranslation[i][0];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
PropertyMap origProps = properties();
|
PropertyMap origProps = properties();
|
||||||
PropertyMap::ConstIterator it = origProps.begin();
|
PropertyMap::ConstIterator it = origProps.begin();
|
||||||
for(; it != origProps.end(); ++it) {
|
for (; it != origProps.end(); ++it) {
|
||||||
if(!props.contains(it->first) || props[it->first].isEmpty()) {
|
if (!props.contains(it->first) || props[it->first].isEmpty()) {
|
||||||
if(it->first == "TITLE") {
|
if (it->first == "TITLE") {
|
||||||
d->title.clear();
|
d->title.clear();
|
||||||
}
|
}
|
||||||
else if(it->first == "ARTIST") {
|
else if (it->first == "ARTIST") {
|
||||||
d->artist.clear();
|
d->artist.clear();
|
||||||
}
|
}
|
||||||
else if(it->first == "COMMENT") {
|
else if (it->first == "COMMENT") {
|
||||||
d->comment.clear();
|
d->comment.clear();
|
||||||
}
|
}
|
||||||
else if(it->first == "COPYRIGHT") {
|
else if (it->first == "COPYRIGHT") {
|
||||||
d->copyright.clear();
|
d->copyright.clear();
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
@@ -348,25 +491,25 @@ PropertyMap ASF::Tag::setProperties(const PropertyMap &props)
|
|||||||
|
|
||||||
PropertyMap ignoredProps;
|
PropertyMap ignoredProps;
|
||||||
it = props.begin();
|
it = props.begin();
|
||||||
for(; it != props.end(); ++it) {
|
for (; it != props.end(); ++it) {
|
||||||
if(reverseKeyMap.contains(it->first)) {
|
if (reverseKeyMap.contains(it->first)) {
|
||||||
String name = reverseKeyMap[it->first];
|
String name = reverseKeyMap[it->first];
|
||||||
removeItem(name);
|
removeItem(name);
|
||||||
StringList::ConstIterator it2 = it->second.begin();
|
StringList::ConstIterator it2 = it->second.begin();
|
||||||
for(; it2 != it->second.end(); ++it2) {
|
for (; it2 != it->second.end(); ++it2) {
|
||||||
addAttribute(name, *it2);
|
addAttribute(name, *it2);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else if(it->first == "TITLE") {
|
else if (it->first == "TITLE") {
|
||||||
d->title = it->second.toString();
|
d->title = it->second.toString();
|
||||||
}
|
}
|
||||||
else if(it->first == "ARTIST") {
|
else if (it->first == "ARTIST") {
|
||||||
d->artist = it->second.toString();
|
d->artist = it->second.toString();
|
||||||
}
|
}
|
||||||
else if(it->first == "COMMENT") {
|
else if (it->first == "COMMENT") {
|
||||||
d->comment = it->second.toString();
|
d->comment = it->second.toString();
|
||||||
}
|
}
|
||||||
else if(it->first == "COPYRIGHT") {
|
else if (it->first == "COPYRIGHT") {
|
||||||
d->copyright = it->second.toString();
|
d->copyright = it->second.toString();
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
@@ -375,4 +518,5 @@ PropertyMap ASF::Tag::setProperties(const PropertyMap &props)
|
|||||||
}
|
}
|
||||||
|
|
||||||
return ignoredProps;
|
return ignoredProps;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
265
3rdparty/taglib/asf/asftag.h
vendored
265
3rdparty/taglib/asf/asftag.h
vendored
@@ -35,178 +35,167 @@
|
|||||||
namespace Strawberry_TagLib {
|
namespace Strawberry_TagLib {
|
||||||
namespace TagLib {
|
namespace TagLib {
|
||||||
|
|
||||||
namespace ASF {
|
namespace ASF {
|
||||||
|
|
||||||
typedef List<Attribute> AttributeList;
|
typedef List<Attribute> AttributeList;
|
||||||
typedef Map<String, AttributeList> AttributeListMap;
|
typedef Map<String, AttributeList> AttributeListMap;
|
||||||
|
|
||||||
class TAGLIB_EXPORT Tag : public Strawberry_TagLib::TagLib::Tag {
|
class TAGLIB_EXPORT Tag : public Strawberry_TagLib::TagLib::Tag {
|
||||||
|
|
||||||
friend class File;
|
friend class File;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
|
explicit Tag();
|
||||||
|
|
||||||
Tag();
|
~Tag() override;
|
||||||
|
|
||||||
virtual ~Tag();
|
/*!
|
||||||
|
* Returns the track name.
|
||||||
|
*/
|
||||||
|
String title() const override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the track name.
|
* Returns the artist name.
|
||||||
*/
|
*/
|
||||||
virtual String title() const;
|
String artist() const override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the artist name.
|
* Returns the album name; if no album name is present in the tag String::null will be returned.
|
||||||
*/
|
*/
|
||||||
virtual String artist() const;
|
String album() const override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the album name; if no album name is present in the tag
|
* Returns the track comment.
|
||||||
* String::null will be returned.
|
*/
|
||||||
*/
|
String comment() const override;
|
||||||
virtual String album() const;
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the track comment.
|
* Returns the genre name; if no genre is present in the tag String::null will be returned.
|
||||||
*/
|
*/
|
||||||
virtual String comment() const;
|
String genre() const override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the genre name; if no genre is present in the tag String::null
|
* Returns the rating.
|
||||||
* will be returned.
|
*/
|
||||||
*/
|
virtual String rating() const;
|
||||||
virtual String genre() const;
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the rating.
|
* Returns the genre name; if no genre is present in the tag String::null will be returned.
|
||||||
*/
|
*/
|
||||||
virtual String rating() const;
|
virtual String copyright() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the genre name; if no genre is present in the tag String::null
|
* Returns the year; if there is no year set, this will return 0.
|
||||||
* will be returned.
|
*/
|
||||||
*/
|
unsigned int year() const override;
|
||||||
virtual String copyright() const;
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the year; if there is no year set, this will return 0.
|
* Returns the track number; if there is no track number set, this will return 0.
|
||||||
*/
|
*/
|
||||||
virtual unsigned int year() const;
|
unsigned int track() const override;
|
||||||
|
|
||||||
/*!
|
PictureMap pictures() const override;
|
||||||
* Returns the track number; if there is no track number set, this will
|
|
||||||
* return 0.
|
|
||||||
*/
|
|
||||||
virtual unsigned int track() const;
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Sets the title to \a s.
|
* Sets the title to \a s.
|
||||||
*/
|
*/
|
||||||
virtual void setTitle(const String &s);
|
void setTitle(const String &value) override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Sets the artist to \a s.
|
* Sets the artist to \a s.
|
||||||
*/
|
*/
|
||||||
virtual void setArtist(const String &s);
|
void setArtist(const String &value) override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Sets the album to \a s. If \a s is String::null then this value will be
|
* Sets the album to \a s. If \a s is String::null then this value will be cleared.
|
||||||
* cleared.
|
*/
|
||||||
*/
|
void setAlbum(const String &value) override;
|
||||||
virtual void setAlbum(const String &s);
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Sets the comment to \a s.
|
* Sets the comment to \a s.
|
||||||
*/
|
*/
|
||||||
virtual void setComment(const String &s);
|
void setComment(const String &value) override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Sets the rating to \a s.
|
* Sets the rating to \a s.
|
||||||
*/
|
*/
|
||||||
virtual void setRating(const String &s);
|
virtual void setRating(const String &value);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Sets the copyright to \a s.
|
* Sets the copyright to \a s.
|
||||||
*/
|
*/
|
||||||
virtual void setCopyright(const String &s);
|
virtual void setCopyright(const String &value);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Sets the genre to \a s.
|
* Sets the genre to \a s.
|
||||||
*/
|
*/
|
||||||
virtual void setGenre(const String &s);
|
void setGenre(const String &value) override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Sets the year to \a i. If \a s is 0 then this value will be cleared.
|
* Sets the year to \a i. If \a s is 0 then this value will be cleared.
|
||||||
*/
|
*/
|
||||||
virtual void setYear(unsigned int i);
|
void setYear(unsigned int value) override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Sets the track to \a i. If \a s is 0 then this value will be cleared.
|
* Sets the track to \a i. If \a s is 0 then this value will be cleared.
|
||||||
*/
|
*/
|
||||||
virtual void setTrack(unsigned int i);
|
void setTrack(unsigned int value) override;
|
||||||
|
|
||||||
/*!
|
void setPictures(const PictureMap&) override;
|
||||||
* Returns true if the tag does not contain any data. This should be
|
|
||||||
* reimplemented in subclasses that provide more than the basic tagging
|
|
||||||
* abilities in this class.
|
|
||||||
*/
|
|
||||||
virtual bool isEmpty() const;
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* \deprecated
|
* Returns true if the tag does not contain any data.
|
||||||
*/
|
* This should be reimplemented in subclasses that provide more than the basic tagging abilities in this class.
|
||||||
AttributeListMap &attributeListMap();
|
*/
|
||||||
|
bool isEmpty() const override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns a reference to the item list map. This is an AttributeListMap of
|
* Returns a reference to the item list map. This is an AttributeListMap of all of the items in the tag.
|
||||||
* all of the items in the tag.
|
*/
|
||||||
*/
|
const AttributeListMap &attributeListMap() const;
|
||||||
// BIC: return by value
|
|
||||||
const AttributeListMap &attributeListMap() const;
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* \return True if a value for \a attribute is currently set.
|
* \return True if a value for \a attribute is currently set.
|
||||||
*/
|
*/
|
||||||
bool contains(const String &name) const;
|
bool contains(const String &key) const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Removes the \a key attribute from the tag
|
* Removes the \a key attribute from the tag
|
||||||
*/
|
*/
|
||||||
void removeItem(const String &name);
|
void removeItem(const String &key);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* \return The list of values for the key \a name, or an empty list if no
|
* \return The list of values for the key \a name, or an empty list if no values have been set.
|
||||||
* values have been set.
|
*/
|
||||||
*/
|
AttributeList attribute(const String &name) const;
|
||||||
AttributeList attribute(const String &name) const;
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Sets the \a key attribute to the value of \a attribute. If an attribute
|
* Sets the \a key attribute to the value of \a attribute. If an attribute with the \a key is already present, it will be replaced.
|
||||||
* with the \a key is already present, it will be replaced.
|
*/
|
||||||
*/
|
void setAttribute(const String &name, const Attribute &attribute);
|
||||||
void setAttribute(const String &name, const Attribute &attribute);
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Sets multiple \a values to the key \a name.
|
* Sets multiple \a values to the key \a name.
|
||||||
*/
|
*/
|
||||||
void setAttribute(const String &name, const AttributeList &values);
|
void setAttribute(const String &name, const AttributeList &values);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Sets the \a key attribute to the value of \a attribute. If an attribute
|
* Sets the \a key attribute to the value of \a attribute. If an attribute
|
||||||
* with the \a key is already present, it will be added to the list.
|
* with the \a key is already present, it will be added to the list.
|
||||||
*/
|
*/
|
||||||
void addAttribute(const String &name, const Attribute &attribute);
|
void addAttribute(const String &name, const Attribute &attribute);
|
||||||
|
|
||||||
PropertyMap properties() const;
|
PropertyMap properties() const override;
|
||||||
void removeUnsupportedProperties(const StringList& properties);
|
void removeUnsupportedProperties(const StringList &props) override;
|
||||||
PropertyMap setProperties(const PropertyMap &properties);
|
PropertyMap setProperties(const PropertyMap &props) override;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
class TagPrivate;
|
||||||
|
TagPrivate *d;
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace ASF
|
||||||
|
} // namespace TagLib
|
||||||
|
} // namespace Strawberry_TagLib
|
||||||
|
|
||||||
class TagPrivate;
|
|
||||||
TagPrivate *d;
|
|
||||||
};
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
124
3rdparty/taglib/asf/asfutils.h
vendored
124
3rdparty/taglib/asf/asfutils.h
vendored
@@ -31,76 +31,70 @@
|
|||||||
#ifndef DO_NOT_DOCUMENT // tell Doxygen not to document this header
|
#ifndef DO_NOT_DOCUMENT // tell Doxygen not to document this header
|
||||||
|
|
||||||
namespace Strawberry_TagLib {
|
namespace Strawberry_TagLib {
|
||||||
namespace TagLib
|
namespace TagLib {
|
||||||
{
|
namespace ASF {
|
||||||
namespace ASF
|
namespace {
|
||||||
{
|
|
||||||
namespace
|
|
||||||
{
|
|
||||||
|
|
||||||
inline unsigned short readWORD(File *file, bool *ok = 0)
|
inline unsigned short readWORD(File *file, bool *ok = nullptr) {
|
||||||
{
|
const ByteVector v = file->readBlock(2);
|
||||||
const ByteVector v = file->readBlock(2);
|
if (v.size() != 2) {
|
||||||
if(v.size() != 2) {
|
if (ok) *ok = false;
|
||||||
if(ok) *ok = false;
|
return 0;
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
if(ok) *ok = true;
|
|
||||||
return v.toUShort(false);
|
|
||||||
}
|
|
||||||
|
|
||||||
inline unsigned int readDWORD(File *file, bool *ok = 0)
|
|
||||||
{
|
|
||||||
const ByteVector v = file->readBlock(4);
|
|
||||||
if(v.size() != 4) {
|
|
||||||
if(ok) *ok = false;
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
if(ok) *ok = true;
|
|
||||||
return v.toUInt(false);
|
|
||||||
}
|
|
||||||
|
|
||||||
inline long long readQWORD(File *file, bool *ok = 0)
|
|
||||||
{
|
|
||||||
const ByteVector v = file->readBlock(8);
|
|
||||||
if(v.size() != 8) {
|
|
||||||
if(ok) *ok = false;
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
if(ok) *ok = true;
|
|
||||||
return v.toLongLong(false);
|
|
||||||
}
|
|
||||||
|
|
||||||
inline String readString(File *file, int length)
|
|
||||||
{
|
|
||||||
ByteVector data = file->readBlock(length);
|
|
||||||
unsigned int size = data.size();
|
|
||||||
while (size >= 2) {
|
|
||||||
if(data[size - 1] != '\0' || data[size - 2] != '\0') {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
size -= 2;
|
|
||||||
}
|
|
||||||
if(size != data.size()) {
|
|
||||||
data.resize(size);
|
|
||||||
}
|
|
||||||
return String(data, String::UTF16LE);
|
|
||||||
}
|
|
||||||
|
|
||||||
inline ByteVector renderString(const String &str, bool includeLength = false)
|
|
||||||
{
|
|
||||||
ByteVector data = str.data(String::UTF16LE) + ByteVector::fromShort(0, false);
|
|
||||||
if(includeLength) {
|
|
||||||
data = ByteVector::fromShort(data.size(), false) + data;
|
|
||||||
}
|
|
||||||
return data;
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
if (ok) *ok = true;
|
||||||
|
return v.toUInt16LE(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
inline unsigned int readDWORD(File *file, bool *ok = nullptr) {
|
||||||
|
const ByteVector v = file->readBlock(4);
|
||||||
|
if (v.size() != 4) {
|
||||||
|
if (ok) *ok = false;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
if (ok) *ok = true;
|
||||||
|
return v.toUInt32LE(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
inline long long readQWORD(File *file, bool *ok = nullptr) {
|
||||||
|
const ByteVector v = file->readBlock(8);
|
||||||
|
if (v.size() != 8) {
|
||||||
|
if (ok) *ok = false;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
if (ok) *ok = true;
|
||||||
|
return v.toInt64LE(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
inline String readString(File *file, int length) {
|
||||||
|
|
||||||
|
ByteVector data = file->readBlock(length);
|
||||||
|
size_t size = data.size();
|
||||||
|
while (size >= 2) {
|
||||||
|
if (data[size - 1] != '\0' || data[size - 2] != '\0') {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
size -= 2;
|
||||||
|
}
|
||||||
|
if (size != data.size()) {
|
||||||
|
data.resize(size);
|
||||||
|
}
|
||||||
|
return String(data, String::UTF16LE);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
inline ByteVector renderString(const String &str, bool includeLength = false) {
|
||||||
|
ByteVector data = str.data(String::UTF16LE) + ByteVector::fromUInt16LE(0);
|
||||||
|
if (includeLength) {
|
||||||
|
data = ByteVector::fromUInt16LE(data.size()) + data;
|
||||||
|
}
|
||||||
|
return data;
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace
|
||||||
|
} // namespace ASF
|
||||||
|
} // namespace TagLib
|
||||||
|
} // namespace Strawberry_TagLib
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
84
3rdparty/taglib/audioproperties.cpp
vendored
84
3rdparty/taglib/audioproperties.cpp
vendored
@@ -23,95 +23,31 @@
|
|||||||
* http://www.mozilla.org/MPL/ *
|
* http://www.mozilla.org/MPL/ *
|
||||||
***************************************************************************/
|
***************************************************************************/
|
||||||
|
|
||||||
#include <tbytevector.h>
|
#include "tstringlist.h"
|
||||||
|
|
||||||
#include "aiffproperties.h"
|
|
||||||
#include "apeproperties.h"
|
|
||||||
#include "asfproperties.h"
|
|
||||||
#include "flacproperties.h"
|
|
||||||
#include "mp4properties.h"
|
|
||||||
#include "mpcproperties.h"
|
|
||||||
#include "mpegproperties.h"
|
|
||||||
#include "opusproperties.h"
|
|
||||||
#include "speexproperties.h"
|
|
||||||
#include "trueaudioproperties.h"
|
|
||||||
#include "vorbisproperties.h"
|
|
||||||
#include "wavproperties.h"
|
|
||||||
#include "wavpackproperties.h"
|
|
||||||
#include "dsfproperties.h"
|
|
||||||
#include "dsdiffproperties.h"
|
|
||||||
|
|
||||||
#include "audioproperties.h"
|
#include "audioproperties.h"
|
||||||
|
|
||||||
using namespace Strawberry_TagLib::TagLib;
|
using namespace Strawberry_TagLib::TagLib;
|
||||||
|
|
||||||
// This macro is a workaround for the fact that we can't add virtual functions.
|
|
||||||
// Should be true virtual functions in taglib2.
|
|
||||||
|
|
||||||
#define VIRTUAL_FUNCTION_WORKAROUND(function_name, default_value) \
|
|
||||||
if(dynamic_cast<const APE::Properties*>(this)) \
|
|
||||||
return dynamic_cast<const APE::Properties*>(this)->function_name(); \
|
|
||||||
else if(dynamic_cast<const ASF::Properties*>(this)) \
|
|
||||||
return dynamic_cast<const ASF::Properties*>(this)->function_name(); \
|
|
||||||
else if(dynamic_cast<const FLAC::Properties*>(this)) \
|
|
||||||
return dynamic_cast<const FLAC::Properties*>(this)->function_name(); \
|
|
||||||
else if(dynamic_cast<const MP4::Properties*>(this)) \
|
|
||||||
return dynamic_cast<const MP4::Properties*>(this)->function_name(); \
|
|
||||||
else if(dynamic_cast<const MPC::Properties*>(this)) \
|
|
||||||
return dynamic_cast<const MPC::Properties*>(this)->function_name(); \
|
|
||||||
else if(dynamic_cast<const MPEG::Properties*>(this)) \
|
|
||||||
return dynamic_cast<const MPEG::Properties*>(this)->function_name(); \
|
|
||||||
else if(dynamic_cast<const Ogg::Opus::Properties*>(this)) \
|
|
||||||
return dynamic_cast<const Ogg::Opus::Properties*>(this)->function_name(); \
|
|
||||||
else if(dynamic_cast<const Ogg::Speex::Properties*>(this)) \
|
|
||||||
return dynamic_cast<const Ogg::Speex::Properties*>(this)->function_name(); \
|
|
||||||
else if(dynamic_cast<const TrueAudio::Properties*>(this)) \
|
|
||||||
return dynamic_cast<const TrueAudio::Properties*>(this)->function_name(); \
|
|
||||||
else if(dynamic_cast<const RIFF::AIFF::Properties*>(this)) \
|
|
||||||
return dynamic_cast<const RIFF::AIFF::Properties*>(this)->function_name(); \
|
|
||||||
else if(dynamic_cast<const RIFF::WAV::Properties*>(this)) \
|
|
||||||
return dynamic_cast<const RIFF::WAV::Properties*>(this)->function_name(); \
|
|
||||||
else if(dynamic_cast<const Vorbis::Properties*>(this)) \
|
|
||||||
return dynamic_cast<const Vorbis::Properties*>(this)->function_name(); \
|
|
||||||
else if(dynamic_cast<const WavPack::Properties*>(this)) \
|
|
||||||
return dynamic_cast<const WavPack::Properties*>(this)->function_name(); \
|
|
||||||
else if(dynamic_cast<const DSF::Properties*>(this)) \
|
|
||||||
return dynamic_cast<const DSF::Properties*>(this)->function_name(); \
|
|
||||||
else if(dynamic_cast<const DSDIFF::Properties*>(this)) \
|
|
||||||
return dynamic_cast<const DSDIFF::Properties*>(this)->function_name(); \
|
|
||||||
else \
|
|
||||||
return (default_value);
|
|
||||||
|
|
||||||
class AudioProperties::AudioPropertiesPrivate
|
|
||||||
{
|
|
||||||
|
|
||||||
};
|
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
// public methods
|
// public methods
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
AudioProperties::~AudioProperties()
|
AudioProperties::~AudioProperties() {}
|
||||||
{
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
int AudioProperties::lengthInSeconds() const
|
String AudioProperties::toString() const {
|
||||||
{
|
|
||||||
VIRTUAL_FUNCTION_WORKAROUND(lengthInSeconds, 0)
|
StringList desc;
|
||||||
}
|
desc.append("Audio");
|
||||||
|
desc.append(String::number(lengthInSeconds()) + " seconds");
|
||||||
|
desc.append(String::number(bitrate()) + " kbps");
|
||||||
|
return desc.toString(", ");
|
||||||
|
|
||||||
int AudioProperties::lengthInMilliseconds() const
|
|
||||||
{
|
|
||||||
VIRTUAL_FUNCTION_WORKAROUND(lengthInMilliseconds, 0)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
// protected methods
|
// protected methods
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
AudioProperties::AudioProperties(ReadStyle) :
|
AudioProperties::AudioProperties() : d(nullptr) {}
|
||||||
d(0)
|
|
||||||
{
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|||||||
167
3rdparty/taglib/audioproperties.h
vendored
167
3rdparty/taglib/audioproperties.h
vendored
@@ -27,103 +27,94 @@
|
|||||||
#define TAGLIB_AUDIOPROPERTIES_H
|
#define TAGLIB_AUDIOPROPERTIES_H
|
||||||
|
|
||||||
#include "taglib_export.h"
|
#include "taglib_export.h"
|
||||||
|
#include "tstring.h"
|
||||||
|
|
||||||
namespace Strawberry_TagLib {
|
namespace Strawberry_TagLib {
|
||||||
namespace TagLib {
|
namespace TagLib {
|
||||||
|
|
||||||
//! A simple, abstract interface to common audio properties
|
//! A simple, abstract interface to common audio properties
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* The values here are common to most audio formats. For more specific, codec
|
* The values here are common to most audio formats.
|
||||||
* dependent values, please see see the subclasses APIs. This is meant to
|
* For more specific, codec dependent values, please see see the subclasses APIs.
|
||||||
* compliment the TagLib::File and TagLib::Tag APIs in providing a simple
|
* This is meant to compliment the TagLib::File and TagLib::Tag APIs in providing a simple
|
||||||
* interface that is sufficient for most applications.
|
* interface that is sufficient for most applications.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
class TAGLIB_EXPORT AudioProperties
|
class TAGLIB_EXPORT AudioProperties {
|
||||||
{
|
public:
|
||||||
public:
|
/*!
|
||||||
|
* Reading audio properties from a file can sometimes be very time consuming
|
||||||
/*!
|
* and for the most accurate results can often involve reading the entire file.
|
||||||
* Reading audio properties from a file can sometimes be very time consuming
|
* Because in many situations speed is critical or the accuracy of the values
|
||||||
* and for the most accurate results can often involve reading the entire
|
* is not particularly important this allows the level of desired accuracy to be set.
|
||||||
* file. Because in many situations speed is critical or the accuracy of the
|
*/
|
||||||
* values is not particularly important this allows the level of desired
|
enum ReadStyle {
|
||||||
* accuracy to be set.
|
//! Read as little of the file as possible
|
||||||
*/
|
Fast,
|
||||||
enum ReadStyle {
|
//! Read more of the file and make better values guesses
|
||||||
//! Read as little of the file as possible
|
Average,
|
||||||
Fast,
|
//! Read as much of the file as needed to report accurate values
|
||||||
//! Read more of the file and make better values guesses
|
Accurate
|
||||||
Average,
|
|
||||||
//! Read as much of the file as needed to report accurate values
|
|
||||||
Accurate
|
|
||||||
};
|
|
||||||
|
|
||||||
/*!
|
|
||||||
* Destroys this AudioProperties instance.
|
|
||||||
*/
|
|
||||||
virtual ~AudioProperties();
|
|
||||||
|
|
||||||
/*!
|
|
||||||
* Returns the length of the file in seconds.
|
|
||||||
*/
|
|
||||||
virtual int length() const = 0;
|
|
||||||
|
|
||||||
/*!
|
|
||||||
* Returns the length of the file in seconds. The length is rounded down to
|
|
||||||
* the nearest whole second.
|
|
||||||
*
|
|
||||||
* \see lengthInMilliseconds()
|
|
||||||
*/
|
|
||||||
// BIC: make virtual
|
|
||||||
int lengthInSeconds() const;
|
|
||||||
|
|
||||||
/*!
|
|
||||||
* Returns the length of the file in milliseconds.
|
|
||||||
*
|
|
||||||
* \see lengthInSeconds()
|
|
||||||
*/
|
|
||||||
// BIC: make virtual
|
|
||||||
int lengthInMilliseconds() const;
|
|
||||||
|
|
||||||
/*!
|
|
||||||
* Returns the most appropriate bit rate for the file in kb/s. For constant
|
|
||||||
* bitrate formats this is simply the bitrate of the file. For variable
|
|
||||||
* bitrate formats this is either the average or nominal bitrate.
|
|
||||||
*/
|
|
||||||
virtual int bitrate() const = 0;
|
|
||||||
|
|
||||||
/*!
|
|
||||||
* Returns the sample rate in Hz.
|
|
||||||
*/
|
|
||||||
virtual int sampleRate() const = 0;
|
|
||||||
|
|
||||||
/*!
|
|
||||||
* Returns the number of audio channels.
|
|
||||||
*/
|
|
||||||
virtual int channels() const = 0;
|
|
||||||
|
|
||||||
protected:
|
|
||||||
|
|
||||||
/*!
|
|
||||||
* Construct an audio properties instance. This is protected as this class
|
|
||||||
* should not be instantiated directly, but should be instantiated via its
|
|
||||||
* subclasses and can be fetched from the FileRef or File APIs.
|
|
||||||
*
|
|
||||||
* \see ReadStyle
|
|
||||||
*/
|
|
||||||
AudioProperties(ReadStyle style);
|
|
||||||
|
|
||||||
private:
|
|
||||||
AudioProperties(const AudioProperties &);
|
|
||||||
AudioProperties &operator=(const AudioProperties &);
|
|
||||||
|
|
||||||
class AudioPropertiesPrivate;
|
|
||||||
AudioPropertiesPrivate *d;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
}
|
/*!
|
||||||
}
|
* Destroys this AudioProperties instance.
|
||||||
|
*/
|
||||||
|
virtual ~AudioProperties();
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* Returns the length of the file in seconds. The length is rounded down to the nearest whole second.
|
||||||
|
*
|
||||||
|
* \see lengthInMilliseconds()
|
||||||
|
*/
|
||||||
|
virtual int lengthInSeconds() const = 0;
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* Returns the length of the file in milliseconds.
|
||||||
|
*
|
||||||
|
* \see lengthInSeconds()
|
||||||
|
*/
|
||||||
|
virtual int lengthInMilliseconds() const = 0;
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* Returns the most appropriate bit rate for the file in kb/s. For constant bitrate formats this is simply the bitrate of the file.
|
||||||
|
* For variable bitrate formats this is either the average or nominal bitrate.
|
||||||
|
*/
|
||||||
|
virtual int bitrate() const = 0;
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* Returns the sample rate in Hz.
|
||||||
|
*/
|
||||||
|
virtual int sampleRate() const = 0;
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* Returns the number of audio channels.
|
||||||
|
*/
|
||||||
|
virtual int channels() const = 0;
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* Returns description of the audio file.
|
||||||
|
*/
|
||||||
|
virtual String toString() const;
|
||||||
|
|
||||||
|
protected:
|
||||||
|
/*!
|
||||||
|
* Construct an audio properties instance.
|
||||||
|
* This is protected as this class should not be instantiated directly,
|
||||||
|
* but should be instantiated via its subclasses and can be fetched from the FileRef or File APIs.
|
||||||
|
*/
|
||||||
|
explicit AudioProperties();
|
||||||
|
|
||||||
|
private:
|
||||||
|
AudioProperties(const AudioProperties&);
|
||||||
|
AudioProperties &operator=(const AudioProperties&);
|
||||||
|
|
||||||
|
class AudioPropertiesPrivate;
|
||||||
|
AudioPropertiesPrivate *d;
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace TagLib
|
||||||
|
} // namespace Strawberry_TagLib
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
105
3rdparty/taglib/dsdiff/dsdiffdiintag.cpp
vendored
105
3rdparty/taglib/dsdiff/dsdiffdiintag.cpp
vendored
@@ -26,137 +26,122 @@
|
|||||||
#include "dsdiffdiintag.h"
|
#include "dsdiffdiintag.h"
|
||||||
#include "tstringlist.h"
|
#include "tstringlist.h"
|
||||||
#include "tpropertymap.h"
|
#include "tpropertymap.h"
|
||||||
|
#include "tpicturemap.h"
|
||||||
|
|
||||||
using namespace Strawberry_TagLib::TagLib;
|
using namespace Strawberry_TagLib::TagLib;
|
||||||
using namespace DSDIFF::DIIN;
|
using namespace DSDIFF::DIIN;
|
||||||
|
|
||||||
class DSDIFF::DIIN::Tag::TagPrivate
|
class DSDIFF::DIIN::Tag::TagPrivate {
|
||||||
{
|
public:
|
||||||
public:
|
TagPrivate() {
|
||||||
TagPrivate()
|
|
||||||
{
|
|
||||||
}
|
}
|
||||||
|
|
||||||
String title;
|
String title;
|
||||||
String artist;
|
String artist;
|
||||||
};
|
};
|
||||||
|
|
||||||
DSDIFF::DIIN::Tag::Tag() : Strawberry_TagLib::TagLib::Tag()
|
DSDIFF::DIIN::Tag::Tag() {
|
||||||
{
|
|
||||||
d = new TagPrivate;
|
d = new TagPrivate;
|
||||||
}
|
}
|
||||||
|
|
||||||
DSDIFF::DIIN::Tag::~Tag()
|
DSDIFF::DIIN::Tag::~Tag() {
|
||||||
{
|
|
||||||
delete d;
|
delete d;
|
||||||
}
|
}
|
||||||
|
|
||||||
String DSDIFF::DIIN::Tag::title() const
|
String DSDIFF::DIIN::Tag::title() const {
|
||||||
{
|
|
||||||
return d->title;
|
return d->title;
|
||||||
}
|
}
|
||||||
|
|
||||||
String DSDIFF::DIIN::Tag::artist() const
|
String DSDIFF::DIIN::Tag::artist() const {
|
||||||
{
|
|
||||||
return d->artist;
|
return d->artist;
|
||||||
}
|
}
|
||||||
|
|
||||||
String DSDIFF::DIIN::Tag::album() const
|
String DSDIFF::DIIN::Tag::album() const {
|
||||||
{
|
|
||||||
return String();
|
return String();
|
||||||
}
|
}
|
||||||
|
|
||||||
String DSDIFF::DIIN::Tag::comment() const
|
String DSDIFF::DIIN::Tag::comment() const {
|
||||||
{
|
|
||||||
return String();
|
return String();
|
||||||
}
|
}
|
||||||
|
|
||||||
String DSDIFF::DIIN::Tag::genre() const
|
String DSDIFF::DIIN::Tag::genre() const {
|
||||||
{
|
|
||||||
return String();
|
return String();
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned int DSDIFF::DIIN::Tag::year() const
|
unsigned int DSDIFF::DIIN::Tag::year() const {
|
||||||
{
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned int DSDIFF::DIIN::Tag::track() const
|
unsigned int DSDIFF::DIIN::Tag::track() const {
|
||||||
{
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
void DSDIFF::DIIN::Tag::setTitle(const String &title)
|
PictureMap DSDIFF::DIIN::Tag::pictures() const {
|
||||||
{
|
return PictureMap();
|
||||||
if(title.isNull() || title.isEmpty())
|
|
||||||
d->title = String();
|
|
||||||
else
|
|
||||||
d->title = title;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void DSDIFF::DIIN::Tag::setArtist(const String &artist)
|
void DSDIFF::DIIN::Tag::setTitle(const String &title) {
|
||||||
{
|
|
||||||
if(artist.isNull() || artist.isEmpty())
|
d->title = title;
|
||||||
d->artist = String();
|
|
||||||
else
|
|
||||||
d->artist = artist;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void DSDIFF::DIIN::Tag::setAlbum(const String &)
|
void DSDIFF::DIIN::Tag::setArtist(const String &artist) {
|
||||||
{
|
|
||||||
|
d->artist = artist;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void DSDIFF::DIIN::Tag::setComment(const String &)
|
void DSDIFF::DIIN::Tag::setAlbum(const String &) {}
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
void DSDIFF::DIIN::Tag::setGenre(const String &)
|
void DSDIFF::DIIN::Tag::setComment(const String &) {}
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
void DSDIFF::DIIN::Tag::setYear(unsigned int)
|
void DSDIFF::DIIN::Tag::setGenre(const String &) {}
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
void DSDIFF::DIIN::Tag::setTrack(unsigned int)
|
void DSDIFF::DIIN::Tag::setYear(unsigned int) {}
|
||||||
{
|
|
||||||
}
|
void DSDIFF::DIIN::Tag::setTrack(unsigned int) {}
|
||||||
|
|
||||||
|
void DSDIFF::DIIN::Tag::setPictures(const PictureMap&) {}
|
||||||
|
|
||||||
|
PropertyMap DSDIFF::DIIN::Tag::properties() const {
|
||||||
|
|
||||||
PropertyMap DSDIFF::DIIN::Tag::properties() const
|
|
||||||
{
|
|
||||||
PropertyMap properties;
|
PropertyMap properties;
|
||||||
properties["TITLE"] = d->title;
|
properties["TITLE"] = d->title;
|
||||||
properties["ARTIST"] = d->artist;
|
properties["ARTIST"] = d->artist;
|
||||||
return properties;
|
return properties;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
PropertyMap DSDIFF::DIIN::Tag::setProperties(const PropertyMap &origProps)
|
PropertyMap DSDIFF::DIIN::Tag::setProperties(const PropertyMap &origProps) {
|
||||||
{
|
|
||||||
PropertyMap properties(origProps);
|
PropertyMap properties(origProps);
|
||||||
properties.removeEmpty();
|
properties.removeEmpty();
|
||||||
StringList oneValueSet;
|
StringList oneValueSet;
|
||||||
|
|
||||||
if(properties.contains("TITLE")) {
|
if (properties.contains("TITLE")) {
|
||||||
d->title = properties["TITLE"].front();
|
d->title = properties["TITLE"].front();
|
||||||
oneValueSet.append("TITLE");
|
oneValueSet.append("TITLE");
|
||||||
} else
|
}
|
||||||
|
else
|
||||||
d->title = String();
|
d->title = String();
|
||||||
|
|
||||||
if(properties.contains("ARTIST")) {
|
if (properties.contains("ARTIST")) {
|
||||||
d->artist = properties["ARTIST"].front();
|
d->artist = properties["ARTIST"].front();
|
||||||
oneValueSet.append("ARTIST");
|
oneValueSet.append("ARTIST");
|
||||||
} else
|
}
|
||||||
|
else
|
||||||
d->artist = String();
|
d->artist = String();
|
||||||
|
|
||||||
// for each tag that has been set above, remove the first entry in the corresponding
|
// for each tag that has been set above, remove the first entry in the corresponding
|
||||||
// value list. The others will be returned as unsupported by this format.
|
// value list. The others will be returned as unsupported by this format.
|
||||||
for(StringList::Iterator it = oneValueSet.begin(); it != oneValueSet.end(); ++it) {
|
for (StringList::Iterator it = oneValueSet.begin(); it != oneValueSet.end(); ++it) {
|
||||||
if(properties[*it].size() == 1)
|
if (properties[*it].size() == 1)
|
||||||
properties.erase(*it);
|
properties.erase(*it);
|
||||||
else
|
else
|
||||||
properties[*it].erase(properties[*it].begin());
|
properties[*it].erase(properties[*it].begin());
|
||||||
}
|
}
|
||||||
|
|
||||||
return properties;
|
return properties;
|
||||||
}
|
|
||||||
|
|
||||||
|
}
|
||||||
|
|||||||
196
3rdparty/taglib/dsdiff/dsdiffdiintag.h
vendored
196
3rdparty/taglib/dsdiff/dsdiffdiintag.h
vendored
@@ -30,123 +30,125 @@
|
|||||||
|
|
||||||
namespace Strawberry_TagLib {
|
namespace Strawberry_TagLib {
|
||||||
namespace TagLib {
|
namespace TagLib {
|
||||||
|
namespace DSDIFF {
|
||||||
|
namespace DIIN {
|
||||||
|
|
||||||
namespace DSDIFF {
|
/*!
|
||||||
|
* Tags from the Edited Master Chunk Info
|
||||||
|
*
|
||||||
|
* Only Title and Artist tags are supported
|
||||||
|
*/
|
||||||
|
|
||||||
namespace DIIN {
|
class TAGLIB_EXPORT Tag : public Strawberry_TagLib::TagLib::Tag {
|
||||||
|
public:
|
||||||
|
explicit Tag();
|
||||||
|
~Tag() override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Tags from the Edited Master Chunk Info
|
* Returns the track name; if no track name is present in the tag String() will be returned.
|
||||||
*
|
*/
|
||||||
* Only Title and Artist tags are supported
|
String title() const override;
|
||||||
*/
|
|
||||||
class TAGLIB_EXPORT Tag : public Strawberry_TagLib::TagLib::Tag
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
Tag();
|
|
||||||
virtual ~Tag();
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the track name; if no track name is present in the tag
|
* Returns the artist name; if no artist name is present in the tag String() will be returned.
|
||||||
* String() will be returned.
|
*/
|
||||||
*/
|
String artist() const override;
|
||||||
String title() const;
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the artist name; if no artist name is present in the tag
|
* Not supported. Therefore always returns String().
|
||||||
* String() will be returned.
|
*/
|
||||||
*/
|
String album() const override;
|
||||||
String artist() const;
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Not supported. Therefore always returns String().
|
* Not supported. Therefore always returns String().
|
||||||
*/
|
*/
|
||||||
String album() const;
|
String comment() const override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Not supported. Therefore always returns String().
|
* Not supported. Therefore always returns String().
|
||||||
*/
|
*/
|
||||||
String comment() const;
|
String genre() const override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Not supported. Therefore always returns String().
|
* Not supported. Therefore always returns 0.
|
||||||
*/
|
*/
|
||||||
String genre() const;
|
unsigned int year() const override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Not supported. Therefore always returns 0.
|
* Not supported. Therefore always returns 0.
|
||||||
*/
|
*/
|
||||||
unsigned int year() const;
|
unsigned int track() const override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Not supported. Therefore always returns 0.
|
* Not supported. Therefore always returns an empty list.
|
||||||
*/
|
*/
|
||||||
unsigned int track() const;
|
PictureMap pictures() const override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Sets the title to \a title. If \a title is String() then this
|
* Sets the title to \a title. If \a title is String() then this value will be cleared.
|
||||||
* value will be cleared.
|
*/
|
||||||
*/
|
void setTitle(const String &title) override;
|
||||||
void setTitle(const String &title);
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Sets the artist to \a artist. If \a artist is String() then this
|
* Sets the artist to \a artist. If \a artist is String() then this value will be cleared.
|
||||||
* value will be cleared.
|
*/
|
||||||
*/
|
void setArtist(const String &artist) override;
|
||||||
void setArtist(const String &artist);
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Not supported and therefore ignored.
|
* Not supported and therefore ignored.
|
||||||
*/
|
*/
|
||||||
void setAlbum(const String &album);
|
void setAlbum(const String &album) override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Not supported and therefore ignored.
|
* Not supported and therefore ignored.
|
||||||
*/
|
*/
|
||||||
void setComment(const String &comment);
|
void setComment(const String &comment) override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Not supported and therefore ignored.
|
* Not supported and therefore ignored.
|
||||||
*/
|
*/
|
||||||
void setGenre(const String &genre);
|
void setGenre(const String &genre) override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Not supported and therefore ignored.
|
* Not supported and therefore ignored.
|
||||||
*/
|
*/
|
||||||
void setYear(unsigned int year);
|
void setYear(unsigned int year) override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Not supported and therefore ignored.
|
* Not supported and therefore ignored.
|
||||||
*/
|
*/
|
||||||
void setTrack(unsigned int track);
|
void setTrack(unsigned int track) override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Implements the unified property interface -- export function.
|
* Not supported and therefore ignored.
|
||||||
* Since the DIIN tag is very limited, the exported map is as well.
|
*/
|
||||||
*/
|
void setPictures(const PictureMap&) override;
|
||||||
PropertyMap properties() const;
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Implements the unified property interface -- import function.
|
* Implements the unified property interface -- export function.
|
||||||
* Because of the limitations of the DIIN file tag, any tags besides
|
* Since the DIIN tag is very limited, the exported map is as well.
|
||||||
* TITLE and ARTIST, will be
|
*/
|
||||||
* returned. Additionally, if the map contains tags with multiple values,
|
PropertyMap properties() const override;
|
||||||
* all but the first will be contained in the returned map of unsupported
|
|
||||||
* properties.
|
|
||||||
*/
|
|
||||||
PropertyMap setProperties(const PropertyMap &);
|
|
||||||
|
|
||||||
private:
|
/*!
|
||||||
Tag(const Tag &);
|
* Implements the unified property interface -- import function.
|
||||||
Tag &operator=(const Tag &);
|
* Because of the limitations of the DIIN file tag, any tags besides
|
||||||
|
* TITLE and ARTIST, will be returned.
|
||||||
|
* Additionally, if the map contains tags with multiple values,
|
||||||
|
* all but the first will be contained in the returned map of unsupported properties.
|
||||||
|
*/
|
||||||
|
PropertyMap setProperties(const PropertyMap &) override;
|
||||||
|
|
||||||
class TagPrivate;
|
private:
|
||||||
TagPrivate *d;
|
Tag(const Tag &);
|
||||||
};
|
Tag &operator=(const Tag &);
|
||||||
}
|
|
||||||
}
|
class TagPrivate;
|
||||||
}
|
TagPrivate *d;
|
||||||
}
|
};
|
||||||
|
} // namespace DIIN
|
||||||
|
} // namespace DSDIFF
|
||||||
|
} // namespace TagLib
|
||||||
|
} // namespace Strawberry_TagLib
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|||||||
662
3rdparty/taglib/dsdiff/dsdifffile.cpp
vendored
662
3rdparty/taglib/dsdiff/dsdifffile.cpp
vendored
File diff suppressed because it is too large
Load Diff
417
3rdparty/taglib/dsdiff/dsdifffile.h
vendored
417
3rdparty/taglib/dsdiff/dsdifffile.h
vendored
@@ -34,255 +34,232 @@
|
|||||||
namespace Strawberry_TagLib {
|
namespace Strawberry_TagLib {
|
||||||
namespace TagLib {
|
namespace TagLib {
|
||||||
|
|
||||||
//! An implementation of DSDIFF metadata
|
//! An implementation of DSDIFF metadata
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* This is implementation of DSDIFF metadata.
|
||||||
|
*
|
||||||
|
* This supports an ID3v2 tag as well as reading stream from the ID3 RIFF
|
||||||
|
* chunk as well as properties from the file.
|
||||||
|
* Description of the DSDIFF format is available at http://dsd-guide.com/sites/default/files/white-papers/DSDIFF_1.5_Spec.pdf
|
||||||
|
* DSDIFF standard does not explicitly specify the ID3V2 chunk
|
||||||
|
* It can be found at the root level, but also sometimes inside the PROP chunk.
|
||||||
|
* In addition, title and artist info are stored as part of the standard
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace DSDIFF {
|
||||||
|
|
||||||
|
//! An implementation of TagLib::File with DSDIFF specific methods
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* This implements and provides an interface for DSDIFF files to the
|
||||||
|
* TagLib::Tag and TagLib::AudioProperties interfaces by way of implementing the
|
||||||
|
* abstract TagLib::File API as well as providing some additional information specific to DSDIFF files.
|
||||||
|
*/
|
||||||
|
|
||||||
|
class TAGLIB_EXPORT File : public Strawberry_TagLib::TagLib::File {
|
||||||
|
public:
|
||||||
|
/*!
|
||||||
|
* This set of flags is used for various operations and is suitable for
|
||||||
|
* being OR-ed together.
|
||||||
|
*/
|
||||||
|
enum TagTypes {
|
||||||
|
//! Empty set. Matches no tag types.
|
||||||
|
NoTags = 0x0000,
|
||||||
|
//! Matches DIIN tags.
|
||||||
|
DIIN = 0x0002,
|
||||||
|
//! Matches ID3v1 tags.
|
||||||
|
ID3v2 = 0x0002,
|
||||||
|
//! Matches all tag types.
|
||||||
|
AllTags = 0xffff
|
||||||
|
};
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* This is implementation of DSDIFF metadata.
|
* Constructs an DSDIFF file from \a file.
|
||||||
|
* If \a readProperties is true the file's audio properties will also be read.
|
||||||
*
|
*
|
||||||
* This supports an ID3v2 tag as well as reading stream from the ID3 RIFF
|
* \note In the current implementation, \a propertiesStyle is ignored.
|
||||||
* chunk as well as properties from the file.
|
|
||||||
* Description of the DSDIFF format is available
|
|
||||||
* at http://dsd-guide.com/sites/default/files/white-papers/DSDIFF_1.5_Spec.pdf
|
|
||||||
* DSDIFF standard does not explicitly specify the ID3V2 chunk
|
|
||||||
* It can be found at the root level, but also sometimes inside the PROP chunk
|
|
||||||
* In addition, title and artist info are stored as part of the standard
|
|
||||||
*/
|
*/
|
||||||
|
explicit File(FileName file, bool readProperties = true, AudioProperties::ReadStyle propertiesStyle = AudioProperties::Average);
|
||||||
|
|
||||||
namespace DSDIFF {
|
/*!
|
||||||
|
* Constructs an DSDIFF file from \a stream.
|
||||||
|
* If \a readProperties is true the file's audio properties will also be read.
|
||||||
|
*
|
||||||
|
* \note TagLib will *not* take ownership of the stream, the caller is responsible for deleting it after the File object.
|
||||||
|
*
|
||||||
|
* \note In the current implementation, \a propertiesStyle is ignored.
|
||||||
|
*/
|
||||||
|
explicit File(IOStream *stream, bool readProperties = true, AudioProperties::ReadStyle propertiesStyle = AudioProperties::Average);
|
||||||
|
|
||||||
//! An implementation of TagLib::File with DSDIFF specific methods
|
/*!
|
||||||
|
* Destroys this instance of the File.
|
||||||
|
*/
|
||||||
|
~File() override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* This implements and provides an interface for DSDIFF files to the
|
* Returns a pointer to a tag that is the union of the ID3v2 and DIIN tags.
|
||||||
* TagLib::Tag and TagLib::AudioProperties interfaces by way of implementing
|
* The ID3v2 tag is given priority in reading the information -- if requested information exists in both the ID3v2 tag and the ID3v1 tag,
|
||||||
* the abstract TagLib::File API as well as providing some additional
|
* the information from the ID3v2 tag will be returned.
|
||||||
* information specific to DSDIFF files.
|
*
|
||||||
*/
|
* If you would like more granular control over the content of the tags, with the concession of generality, use the tag-type specific calls.
|
||||||
|
*
|
||||||
|
* \note As this tag is not implemented as an ID3v2 tag or a DIIN tag,
|
||||||
|
* but a union of the two this pointer may not be cast to the specific tag types.
|
||||||
|
*
|
||||||
|
* \see ID3v2Tag()
|
||||||
|
* \see DIINTag()
|
||||||
|
*/
|
||||||
|
Tag *tag() const override;
|
||||||
|
|
||||||
class TAGLIB_EXPORT File : public Strawberry_TagLib::TagLib::File
|
/*!
|
||||||
{
|
* Returns the ID3V2 Tag for this file.
|
||||||
public:
|
*
|
||||||
|
* \note This always returns a valid pointer regardless of whether or not the file on disk has an ID3v2 tag.
|
||||||
|
* Use hasID3v2Tag() to check if the file on disk actually has an ID3v2 tag.
|
||||||
|
*
|
||||||
|
* \see hasID3v2Tag()
|
||||||
|
*/
|
||||||
|
ID3v2::Tag *ID3v2Tag(bool create = false) const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* This set of flags is used for various operations and is suitable for
|
* Returns the DSDIFF DIIN Tag for this file
|
||||||
* being OR-ed together.
|
*
|
||||||
*/
|
*/
|
||||||
enum TagTypes {
|
DSDIFF::DIIN::Tag *DIINTag(bool create = false) const;
|
||||||
//! Empty set. Matches no tag types.
|
|
||||||
NoTags = 0x0000,
|
|
||||||
//! Matches DIIN tags.
|
|
||||||
DIIN = 0x0002,
|
|
||||||
//! Matches ID3v1 tags.
|
|
||||||
ID3v2 = 0x0002,
|
|
||||||
//! Matches all tag types.
|
|
||||||
AllTags = 0xffff
|
|
||||||
};
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Constructs an DSDIFF file from \a file. If \a readProperties is true
|
* Implements the unified property interface -- export function.
|
||||||
* the file's audio properties will also be read.
|
* This method forwards to ID3v2::Tag::properties().
|
||||||
*
|
*/
|
||||||
* \note In the current implementation, \a propertiesStyle is ignored.
|
PropertyMap properties() const override;
|
||||||
*/
|
|
||||||
File(FileName file, bool readProperties = true,
|
|
||||||
Properties::ReadStyle propertiesStyle = Properties::Average);
|
|
||||||
|
|
||||||
/*!
|
void removeUnsupportedProperties(const StringList &properties) override;
|
||||||
* Constructs an DSDIFF file from \a stream. If \a readProperties is true
|
|
||||||
* the file's audio properties will also be read.
|
|
||||||
*
|
|
||||||
* \note TagLib will *not* take ownership of the stream, the caller is
|
|
||||||
* responsible for deleting it after the File object.
|
|
||||||
*
|
|
||||||
* \note In the current implementation, \a propertiesStyle is ignored.
|
|
||||||
*/
|
|
||||||
File(IOStream *stream, bool readProperties = true,
|
|
||||||
Properties::ReadStyle propertiesStyle = Properties::Average);
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Destroys this instance of the File.
|
* Implements the unified property interface -- import function.
|
||||||
*/
|
* This method forwards to ID3v2::Tag::setProperties().
|
||||||
virtual ~File();
|
*/
|
||||||
|
PropertyMap setProperties(const PropertyMap &properties) override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns a pointer to a tag that is the union of the ID3v2 and DIIN
|
* Returns the AIFF::AudioProperties for this file.
|
||||||
* tags. The ID3v2 tag is given priority in reading the information -- if
|
* If no audio properties were read then this will return a null pointer.
|
||||||
* requested information exists in both the ID3v2 tag and the ID3v1 tag,
|
*/
|
||||||
* the information from the ID3v2 tag will be returned.
|
AudioProperties *audioProperties() const override;
|
||||||
*
|
|
||||||
* If you would like more granular control over the content of the tags,
|
|
||||||
* with the concession of generality, use the tag-type specific calls.
|
|
||||||
*
|
|
||||||
* \note As this tag is not implemented as an ID3v2 tag or a DIIN tag,
|
|
||||||
* but a union of the two this pointer may not be cast to the specific
|
|
||||||
* tag types.
|
|
||||||
*
|
|
||||||
* \see ID3v2Tag()
|
|
||||||
* \see DIINTag()
|
|
||||||
*/
|
|
||||||
virtual Tag *tag() const;
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the ID3V2 Tag for this file.
|
* Save the file. If at least one tag -- ID3v1 or DIIN -- exists this will duplicate its content into the other tag.
|
||||||
*
|
* This returns true if saving was successful.
|
||||||
* \note This always returns a valid pointer regardless of whether or not
|
*
|
||||||
* the file on disk has an ID3v2 tag. Use hasID3v2Tag() to check if the
|
* If neither exists or if both tags are empty, this will strip the tags from the file.
|
||||||
* file on disk actually has an ID3v2 tag.
|
*
|
||||||
*
|
* This is the same as calling save(AllTags);
|
||||||
* \see hasID3v2Tag()
|
*
|
||||||
*/
|
* If you would like more granular control over the content of the tags,
|
||||||
ID3v2::Tag *ID3v2Tag(bool create = false) const;
|
* with the concession of generality, use paramaterized save call below.
|
||||||
|
*
|
||||||
|
* \see save(int tags)
|
||||||
|
*/
|
||||||
|
bool save() override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the DSDIFF DIIN Tag for this file
|
* Save the file. If \a strip is specified,
|
||||||
*
|
* it is possible to choose if tags not specified in \a tags should be stripped from the file or retained.
|
||||||
*/
|
* With \a version, it is possible to specify whether ID3v2.4 or ID3v2.3 should be used.
|
||||||
DSDIFF::DIIN::Tag *DIINTag(bool create = false) const;
|
*/
|
||||||
|
bool save(TagTypes tags, StripTags strip = StripOthers, ID3v2::Version version = ID3v2::v4);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Implements the unified property interface -- export function.
|
* This will strip the tags that match the OR-ed together TagTypes from the file.
|
||||||
* This method forwards to ID3v2::Tag::properties().
|
* By default it strips all tags. It returns true if the tags are successfully stripped.
|
||||||
*/
|
*
|
||||||
PropertyMap properties() const;
|
* \note This will update the file immediately.
|
||||||
|
*/
|
||||||
|
void strip(TagTypes tags = AllTags);
|
||||||
|
|
||||||
void removeUnsupportedProperties(const StringList &properties);
|
/*!
|
||||||
|
* Returns whether or not the file on disk actually has an ID3v2 tag.
|
||||||
|
*
|
||||||
|
* \see ID3v2Tag()
|
||||||
|
*/
|
||||||
|
bool hasID3v2Tag() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Implements the unified property interface -- import function.
|
* Returns whether or not the file on disk actually has the DSDIFF title and artist tags.
|
||||||
* This method forwards to ID3v2::Tag::setProperties().
|
*
|
||||||
*/
|
* \see DIINTag()
|
||||||
PropertyMap setProperties(const PropertyMap &);
|
*/
|
||||||
|
bool hasDIINTag() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the AIFF::Properties for this file. If no audio properties
|
* Returns whether or not the given \a stream can be opened as a DSDIFF file.
|
||||||
* were read then this will return a null pointer.
|
*
|
||||||
*/
|
* \note This method is designed to do a quick check. The result may not necessarily be correct.
|
||||||
virtual Properties *audioProperties() const;
|
*/
|
||||||
|
static bool isSupported(IOStream *stream);
|
||||||
|
|
||||||
/*!
|
protected:
|
||||||
* Save the file. If at least one tag -- ID3v1 or DIIN -- exists this
|
enum Endianness {
|
||||||
* will duplicate its content into the other tag. This returns true
|
BigEndian,
|
||||||
* if saving was successful.
|
LittleEndian
|
||||||
*
|
};
|
||||||
* If neither exists or if both tags are empty, this will strip the tags
|
|
||||||
* from the file.
|
|
||||||
*
|
|
||||||
* This is the same as calling save(AllTags);
|
|
||||||
*
|
|
||||||
* If you would like more granular control over the content of the tags,
|
|
||||||
* with the concession of generality, use paramaterized save call below.
|
|
||||||
*
|
|
||||||
* \see save(int tags)
|
|
||||||
*/
|
|
||||||
virtual bool save();
|
|
||||||
|
|
||||||
/*!
|
File(FileName file, Endianness endianness);
|
||||||
* Save the file. If \a strip is specified, it is possible to choose if
|
File(IOStream *stream, Endianness endianness);
|
||||||
* tags not specified in \a tags should be stripped from the file or
|
|
||||||
* retained. With \a version, it is possible to specify whether ID3v2.4
|
|
||||||
* or ID3v2.3 should be used.
|
|
||||||
*/
|
|
||||||
bool save(TagTypes tags, StripTags strip = StripOthers, ID3v2::Version version = ID3v2::v4);
|
|
||||||
|
|
||||||
/*!
|
private:
|
||||||
* This will strip the tags that match the OR-ed together TagTypes from the
|
File(const File &);
|
||||||
* file. By default it strips all tags. It returns true if the tags are
|
File &operator=(const File &);
|
||||||
* successfully stripped.
|
|
||||||
*
|
|
||||||
* \note This will update the file immediately.
|
|
||||||
*/
|
|
||||||
void strip(TagTypes tags = AllTags);
|
|
||||||
|
|
||||||
/*!
|
void removeRootChunk(const ByteVector &id);
|
||||||
* Returns whether or not the file on disk actually has an ID3v2 tag.
|
void removeRootChunk(unsigned int i);
|
||||||
*
|
void removeChildChunk(unsigned int i, unsigned int childChunkNum);
|
||||||
* \see ID3v2Tag()
|
|
||||||
*/
|
|
||||||
bool hasID3v2Tag() const;
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns whether or not the file on disk actually has the DSDIFF
|
* Sets the data for the the specified chunk at root level to \a data.
|
||||||
* title and artist tags.
|
*
|
||||||
*
|
* \warning This will update the file immediately.
|
||||||
* \see DIINTag()
|
*/
|
||||||
*/
|
void setRootChunkData(unsigned int i, const ByteVector &data);
|
||||||
bool hasDIINTag() const;
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns whether or not the given \a stream can be opened as a DSDIFF
|
* Sets the data for the root-level chunk \a name to \a data.
|
||||||
* file.
|
* If a root-level chunk with the given name already exists it will be overwritten, otherwise it will be created after the existing chunks.
|
||||||
*
|
*
|
||||||
* \note This method is designed to do a quick check. The result may
|
* \warning This will update the file immediately.
|
||||||
* not necessarily be correct.
|
*/
|
||||||
*/
|
void setRootChunkData(const ByteVector &name, const ByteVector &data);
|
||||||
static bool isSupported(IOStream *stream);
|
|
||||||
|
|
||||||
protected:
|
/*!
|
||||||
enum Endianness { BigEndian, LittleEndian };
|
* Sets the data for the the specified child chunk to \a data.
|
||||||
|
*
|
||||||
|
* If data is null, then remove the chunk
|
||||||
|
*
|
||||||
|
* \warning This will update the file immediately.
|
||||||
|
*/
|
||||||
|
void setChildChunkData(unsigned int i, const ByteVector &data, unsigned int childChunkNum);
|
||||||
|
|
||||||
File(FileName file, Endianness endianness);
|
/*!
|
||||||
File(IOStream *stream, Endianness endianness);
|
* Sets the data for the child chunk \a name to \a data.
|
||||||
|
* If a chunk with the given name already exists it will be overwritten, otherwise it will be created after the existing chunks inside child chunk.
|
||||||
|
*
|
||||||
|
* If data is null, then remove the chunks with \a name name
|
||||||
|
*
|
||||||
|
* \warning This will update the file immediately.
|
||||||
|
*/
|
||||||
|
void setChildChunkData(const ByteVector &name, const ByteVector &data, unsigned int childChunkNum);
|
||||||
|
|
||||||
private:
|
void updateRootChunksStructure(unsigned int startingChunk);
|
||||||
File(const File &);
|
|
||||||
File &operator=(const File &);
|
|
||||||
|
|
||||||
void removeRootChunk(const ByteVector &id);
|
void read(bool readProperties, AudioProperties::ReadStyle propertiesStyle);
|
||||||
void removeRootChunk(unsigned int chunk);
|
void writeChunk(const ByteVector &name, const ByteVector &data, unsigned long long offset, unsigned long replace = 0, unsigned int leadingPadding = 0);
|
||||||
void removeChildChunk(unsigned int i, unsigned int chunk);
|
|
||||||
|
|
||||||
/*!
|
class FilePrivate;
|
||||||
* Sets the data for the the specified chunk at root level to \a data.
|
FilePrivate *d;
|
||||||
*
|
};
|
||||||
* \warning This will update the file immediately.
|
} // namespace DSDIFF
|
||||||
*/
|
} // namespace TagLib
|
||||||
void setRootChunkData(unsigned int i, const ByteVector &data);
|
} // namespace Strawberry_TagLib
|
||||||
|
|
||||||
/*!
|
|
||||||
* Sets the data for the root-level chunk \a name to \a data.
|
|
||||||
* If a root-level chunk with the given name already exists
|
|
||||||
* it will be overwritten, otherwise it will be
|
|
||||||
* created after the existing chunks.
|
|
||||||
*
|
|
||||||
* \warning This will update the file immediately.
|
|
||||||
*/
|
|
||||||
void setRootChunkData(const ByteVector &name, const ByteVector &data);
|
|
||||||
|
|
||||||
/*!
|
|
||||||
* Sets the data for the the specified child chunk to \a data.
|
|
||||||
*
|
|
||||||
* If data is null, then remove the chunk
|
|
||||||
*
|
|
||||||
* \warning This will update the file immediately.
|
|
||||||
*/
|
|
||||||
void setChildChunkData(unsigned int i, const ByteVector &data,
|
|
||||||
unsigned int childChunkNum);
|
|
||||||
|
|
||||||
/*!
|
|
||||||
* Sets the data for the child chunk \a name to \a data. If a chunk with
|
|
||||||
* the given name already exists it will be overwritten, otherwise it will
|
|
||||||
* be created after the existing chunks inside child chunk.
|
|
||||||
*
|
|
||||||
* If data is null, then remove the chunks with \a name name
|
|
||||||
*
|
|
||||||
* \warning This will update the file immediately.
|
|
||||||
*/
|
|
||||||
void setChildChunkData(const ByteVector &name, const ByteVector &data,
|
|
||||||
unsigned int childChunkNum);
|
|
||||||
|
|
||||||
void updateRootChunksStructure(unsigned int startingChunk);
|
|
||||||
|
|
||||||
void read(bool readProperties, Properties::ReadStyle propertiesStyle);
|
|
||||||
void writeChunk(const ByteVector &name, const ByteVector &data,
|
|
||||||
unsigned long long offset, unsigned long replace = 0,
|
|
||||||
unsigned int leadingPadding = 0);
|
|
||||||
|
|
||||||
class FilePrivate;
|
|
||||||
FilePrivate *d;
|
|
||||||
};
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|||||||
72
3rdparty/taglib/dsdiff/dsdiffproperties.cpp
vendored
72
3rdparty/taglib/dsdiff/dsdiffproperties.cpp
vendored
@@ -23,24 +23,21 @@
|
|||||||
* http://www.mozilla.org/MPL/ *
|
* http://www.mozilla.org/MPL/ *
|
||||||
***************************************************************************/
|
***************************************************************************/
|
||||||
|
|
||||||
#include <tstring.h>
|
#include "tstring.h"
|
||||||
#include <tdebug.h>
|
#include "tdebug.h"
|
||||||
|
|
||||||
#include "dsdiffproperties.h"
|
#include "dsdiffproperties.h"
|
||||||
|
|
||||||
using namespace Strawberry_TagLib::TagLib;
|
using namespace Strawberry_TagLib::TagLib;
|
||||||
|
|
||||||
class DSDIFF::Properties::PropertiesPrivate
|
class DSDIFF::AudioProperties::AudioPropertiesPrivate {
|
||||||
{
|
public:
|
||||||
public:
|
explicit AudioPropertiesPrivate() : length(0),
|
||||||
PropertiesPrivate() :
|
bitrate(0),
|
||||||
length(0),
|
sampleRate(0),
|
||||||
bitrate(0),
|
channels(0),
|
||||||
sampleRate(0),
|
sampleWidth(0),
|
||||||
channels(0),
|
sampleCount(0) {
|
||||||
sampleWidth(0),
|
|
||||||
sampleCount(0)
|
|
||||||
{
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int length;
|
int length;
|
||||||
@@ -55,66 +52,45 @@ public:
|
|||||||
// public members
|
// public members
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
DSDIFF::Properties::Properties(const unsigned int sampleRate,
|
DSDIFF::AudioProperties::AudioProperties(const unsigned int sampleRate, const unsigned short channels, const unsigned long long samplesCount, const int bitrate, ReadStyle) : Strawberry_TagLib::TagLib::AudioProperties(), d(new AudioPropertiesPrivate) {
|
||||||
const unsigned short channels,
|
|
||||||
const unsigned long long samplesCount,
|
|
||||||
const int bitrate,
|
|
||||||
ReadStyle style) : AudioProperties(style)
|
|
||||||
{
|
|
||||||
d = new PropertiesPrivate;
|
|
||||||
|
|
||||||
d->channels = channels;
|
d->channels = channels;
|
||||||
d->sampleCount = samplesCount;
|
d->sampleCount = samplesCount;
|
||||||
d->sampleWidth = 1;
|
d->sampleWidth = 1;
|
||||||
d->sampleRate = sampleRate;
|
d->sampleRate = sampleRate;
|
||||||
d->bitrate = bitrate;
|
d->bitrate = bitrate;
|
||||||
d->length = d->sampleRate > 0
|
d->length = d->sampleRate > 0 ? static_cast<int>((d->sampleCount * 1000.0) / d->sampleRate + 0.5) : 0;
|
||||||
? static_cast<int>((d->sampleCount * 1000.0) / d->sampleRate + 0.5)
|
|
||||||
: 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
DSDIFF::Properties::~Properties()
|
DSDIFF::AudioProperties::~AudioProperties() {
|
||||||
{
|
|
||||||
delete d;
|
delete d;
|
||||||
}
|
}
|
||||||
|
|
||||||
int DSDIFF::Properties::length() const
|
int DSDIFF::AudioProperties::lengthInSeconds() const {
|
||||||
{
|
|
||||||
return lengthInSeconds();
|
|
||||||
}
|
|
||||||
|
|
||||||
int DSDIFF::Properties::lengthInSeconds() const
|
|
||||||
{
|
|
||||||
return d->length / 1000;
|
return d->length / 1000;
|
||||||
}
|
}
|
||||||
|
|
||||||
int DSDIFF::Properties::lengthInMilliseconds() const
|
int DSDIFF::AudioProperties::lengthInMilliseconds() const {
|
||||||
{
|
|
||||||
return d->length;
|
return d->length;
|
||||||
}
|
}
|
||||||
|
|
||||||
int DSDIFF::Properties::bitrate() const
|
int DSDIFF::AudioProperties::bitrate() const {
|
||||||
{
|
|
||||||
return d->bitrate;
|
return d->bitrate;
|
||||||
}
|
}
|
||||||
|
|
||||||
int DSDIFF::Properties::sampleRate() const
|
int DSDIFF::AudioProperties::sampleRate() const {
|
||||||
{
|
|
||||||
return d->sampleRate;
|
return d->sampleRate;
|
||||||
}
|
}
|
||||||
|
|
||||||
int DSDIFF::Properties::channels() const
|
int DSDIFF::AudioProperties::channels() const {
|
||||||
{
|
|
||||||
return d->channels;
|
return d->channels;
|
||||||
}
|
}
|
||||||
|
|
||||||
int DSDIFF::Properties::bitsPerSample() const
|
int DSDIFF::AudioProperties::bitsPerSample() const {
|
||||||
{
|
|
||||||
return d->sampleWidth;
|
return d->sampleWidth;
|
||||||
}
|
}
|
||||||
|
|
||||||
long long DSDIFF::Properties::sampleCount() const
|
long long DSDIFF::AudioProperties::sampleCount() const {
|
||||||
{
|
|
||||||
return d->sampleCount;
|
return d->sampleCount;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
73
3rdparty/taglib/dsdiff/dsdiffproperties.h
vendored
73
3rdparty/taglib/dsdiff/dsdiffproperties.h
vendored
@@ -30,56 +30,49 @@
|
|||||||
|
|
||||||
namespace Strawberry_TagLib {
|
namespace Strawberry_TagLib {
|
||||||
namespace TagLib {
|
namespace TagLib {
|
||||||
|
namespace DSDIFF {
|
||||||
|
|
||||||
namespace DSDIFF {
|
class File;
|
||||||
|
|
||||||
class File;
|
//! An implementation of audio property reading for DSDIFF
|
||||||
|
|
||||||
//! An implementation of audio property reading for DSDIFF
|
/*!
|
||||||
|
* This reads the data from an DSDIFF stream found in the AudioProperties API.
|
||||||
|
*/
|
||||||
|
|
||||||
/*!
|
class TAGLIB_EXPORT AudioProperties : public Strawberry_TagLib::TagLib::AudioProperties {
|
||||||
* This reads the data from an DSDIFF stream found in the AudioProperties
|
public:
|
||||||
* API.
|
/*!
|
||||||
*/
|
* Create an instance of DSDIFF::AudioProperties with the data read from the ByteVector \a data.
|
||||||
|
*/
|
||||||
|
explicit AudioProperties(const unsigned int sampleRate, const unsigned short channels, const unsigned long long samplesCount, const int bitrate, ReadStyle);
|
||||||
|
|
||||||
class TAGLIB_EXPORT Properties : public AudioProperties
|
/*!
|
||||||
{
|
* Destroys this DSDIFF::AudioProperties instance.
|
||||||
public:
|
*/
|
||||||
/*!
|
~AudioProperties() override;
|
||||||
* Create an instance of DSDIFF::Properties with the data read from the
|
|
||||||
* ByteVector \a data.
|
|
||||||
*/
|
|
||||||
Properties(const unsigned int sampleRate, const unsigned short channels,
|
|
||||||
const unsigned long long samplesCount, const int bitrate,
|
|
||||||
ReadStyle style);
|
|
||||||
|
|
||||||
/*!
|
// Reimplementations.
|
||||||
* Destroys this DSDIFF::Properties instance.
|
|
||||||
*/
|
|
||||||
virtual ~Properties();
|
|
||||||
|
|
||||||
// Reimplementations.
|
int lengthInSeconds() const override;
|
||||||
|
int lengthInMilliseconds() const override;
|
||||||
|
int bitrate() const override;
|
||||||
|
int sampleRate() const override;
|
||||||
|
int channels() const override;
|
||||||
|
|
||||||
virtual int length() const;
|
int bitsPerSample() const;
|
||||||
virtual int lengthInSeconds() const;
|
long long sampleCount() const;
|
||||||
virtual int lengthInMilliseconds() const;
|
|
||||||
virtual int bitrate() const;
|
|
||||||
virtual int sampleRate() const;
|
|
||||||
virtual int channels() const;
|
|
||||||
|
|
||||||
int bitsPerSample() const;
|
private:
|
||||||
long long sampleCount() const;
|
AudioProperties(const AudioProperties &);
|
||||||
|
AudioProperties &operator=(const AudioProperties &);
|
||||||
|
|
||||||
private:
|
class AudioPropertiesPrivate;
|
||||||
Properties(const Properties &);
|
AudioPropertiesPrivate *d;
|
||||||
Properties &operator=(const Properties &);
|
};
|
||||||
|
|
||||||
class PropertiesPrivate;
|
} // namespace DSDIFF
|
||||||
PropertiesPrivate *d;
|
} // namespace TagLib
|
||||||
};
|
} // namespace Strawberry_TagLib
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|||||||
149
3rdparty/taglib/dsf/dsffile.cpp
vendored
149
3rdparty/taglib/dsf/dsffile.cpp
vendored
@@ -23,12 +23,14 @@
|
|||||||
* http://www.mozilla.org/MPL/ *
|
* http://www.mozilla.org/MPL/ *
|
||||||
***************************************************************************/
|
***************************************************************************/
|
||||||
|
|
||||||
#include <tbytevector.h>
|
#include <memory>
|
||||||
#include <tdebug.h>
|
|
||||||
#include <id3v2tag.h>
|
#include "tbytevector.h"
|
||||||
#include <tstringlist.h>
|
#include "tdebug.h"
|
||||||
#include <tpropertymap.h>
|
#include "id3v2tag.h"
|
||||||
#include <tagutils.h>
|
#include "tstringlist.h"
|
||||||
|
#include "tpropertymap.h"
|
||||||
|
#include "tagutils.h"
|
||||||
|
|
||||||
#include "dsffile.h"
|
#include "dsffile.h"
|
||||||
|
|
||||||
@@ -36,38 +38,28 @@ using namespace Strawberry_TagLib::TagLib;
|
|||||||
|
|
||||||
// The DSF specification is located at http://dsd-guide.com/sites/default/files/white-papers/DSFFileFormatSpec_E.pdf
|
// The DSF specification is located at http://dsd-guide.com/sites/default/files/white-papers/DSFFileFormatSpec_E.pdf
|
||||||
|
|
||||||
class DSF::File::FilePrivate
|
class DSF::File::FilePrivate {
|
||||||
{
|
public:
|
||||||
public:
|
FilePrivate() : fileSize(0),
|
||||||
FilePrivate() :
|
metadataOffset(0) {}
|
||||||
fileSize(0),
|
|
||||||
metadataOffset(0),
|
|
||||||
properties(nullptr),
|
|
||||||
tag(nullptr)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
~FilePrivate()
|
|
||||||
{
|
|
||||||
delete properties;
|
|
||||||
delete tag;
|
|
||||||
}
|
|
||||||
|
|
||||||
long long fileSize;
|
long long fileSize;
|
||||||
long long metadataOffset;
|
long long metadataOffset;
|
||||||
Properties *properties;
|
|
||||||
ID3v2::Tag *tag;
|
std::unique_ptr<AudioProperties> properties;
|
||||||
|
std::unique_ptr<ID3v2::Tag> tag;
|
||||||
};
|
};
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
// static members
|
// static members
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
bool DSF::File::isSupported(IOStream *stream)
|
bool DSF::File::isSupported(IOStream *stream) {
|
||||||
{
|
|
||||||
// A DSF file has to start with "DSD "
|
// A DSF file has to start with "DSD "
|
||||||
const ByteVector id = Utils::readHeader(stream, 4, false);
|
const ByteVector id = Utils::readHeader(stream, 4, false);
|
||||||
return id.startsWith("DSD ");
|
return id.startsWith("DSD ");
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
@@ -75,74 +67,66 @@ bool DSF::File::isSupported(IOStream *stream)
|
|||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
DSF::File::File(FileName file, bool readProperties,
|
DSF::File::File(FileName file, bool readProperties,
|
||||||
Properties::ReadStyle propertiesStyle) :
|
AudioProperties::ReadStyle propertiesStyle) : Strawberry_TagLib::TagLib::File(file), d(new FilePrivate()) {
|
||||||
Strawberry_TagLib::TagLib::File(file),
|
|
||||||
d(new FilePrivate())
|
if (isOpen())
|
||||||
{
|
|
||||||
if(isOpen())
|
|
||||||
read(readProperties, propertiesStyle);
|
read(readProperties, propertiesStyle);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
DSF::File::File(IOStream *stream, bool readProperties,
|
DSF::File::File(IOStream *stream, bool readProperties, AudioProperties::ReadStyle propertiesStyle) : Strawberry_TagLib::TagLib::File(stream), d(new FilePrivate()) {
|
||||||
Properties::ReadStyle propertiesStyle) :
|
|
||||||
Strawberry_TagLib::TagLib::File(stream),
|
if (isOpen())
|
||||||
d(new FilePrivate())
|
|
||||||
{
|
|
||||||
if(isOpen())
|
|
||||||
read(readProperties, propertiesStyle);
|
read(readProperties, propertiesStyle);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
DSF::File::~File()
|
DSF::File::~File() {
|
||||||
{
|
|
||||||
delete d;
|
delete d;
|
||||||
}
|
}
|
||||||
|
|
||||||
ID3v2::Tag *DSF::File::tag() const
|
ID3v2::Tag *DSF::File::tag() const {
|
||||||
{
|
return d->tag.get();
|
||||||
return d->tag;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
PropertyMap DSF::File::properties() const
|
DSF::AudioProperties *DSF::File::audioProperties() const {
|
||||||
{
|
return d->properties.get();
|
||||||
|
}
|
||||||
|
|
||||||
|
PropertyMap DSF::File::properties() const {
|
||||||
return d->tag->properties();
|
return d->tag->properties();
|
||||||
}
|
}
|
||||||
|
|
||||||
PropertyMap DSF::File::setProperties(const PropertyMap &properties)
|
PropertyMap DSF::File::setProperties(const PropertyMap &properties) {
|
||||||
{
|
|
||||||
return d->tag->setProperties(properties);
|
return d->tag->setProperties(properties);
|
||||||
}
|
}
|
||||||
|
|
||||||
DSF::Properties *DSF::File::audioProperties() const
|
bool DSF::File::save() {
|
||||||
{
|
|
||||||
return d->properties;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool DSF::File::save()
|
if (readOnly()) {
|
||||||
{
|
|
||||||
if(readOnly()) {
|
|
||||||
debug("DSF::File::save() -- File is read only.");
|
debug("DSF::File::save() -- File is read only.");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if(!isValid()) {
|
if (!isValid()) {
|
||||||
debug("DSF::File::save() -- Trying to save invalid file.");
|
debug("DSF::File::save() -- Trying to save invalid file.");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Three things must be updated: the file size, the tag data, and the metadata offset
|
// Three things must be updated: the file size, the tag data, and the metadata offset
|
||||||
|
|
||||||
if(d->tag->isEmpty()) {
|
if (d->tag->isEmpty()) {
|
||||||
long long newFileSize = d->metadataOffset ? d->metadataOffset : d->fileSize;
|
long long newFileSize = d->metadataOffset ? d->metadataOffset : d->fileSize;
|
||||||
|
|
||||||
// Update the file size
|
// Update the file size
|
||||||
if(d->fileSize != newFileSize) {
|
if (d->fileSize != newFileSize) {
|
||||||
insert(ByteVector::fromLongLong(newFileSize, false), 12, 8);
|
insert(ByteVector::fromUInt64LE(newFileSize), 12, 8);
|
||||||
d->fileSize = newFileSize;
|
d->fileSize = newFileSize;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Update the metadata offset to 0 since there is no longer a tag
|
// Update the metadata offset to 0 since there is no longer a tag
|
||||||
if(d->metadataOffset) {
|
if (d->metadataOffset) {
|
||||||
insert(ByteVector::fromLongLong(0ULL, false), 20, 8);
|
insert(ByteVector::fromUInt64LE(0ULL), 20, 8);
|
||||||
d->metadataOffset = 0;
|
d->metadataOffset = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -157,14 +141,14 @@ bool DSF::File::save()
|
|||||||
long long oldTagSize = d->fileSize - newMetadataOffset;
|
long long oldTagSize = d->fileSize - newMetadataOffset;
|
||||||
|
|
||||||
// Update the file size
|
// Update the file size
|
||||||
if(d->fileSize != newFileSize) {
|
if (d->fileSize != newFileSize) {
|
||||||
insert(ByteVector::fromLongLong(newFileSize, false), 12, 8);
|
insert(ByteVector::fromUInt64LE(newFileSize), 12, 8);
|
||||||
d->fileSize = newFileSize;
|
d->fileSize = newFileSize;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Update the metadata offset
|
// Update the metadata offset
|
||||||
if(d->metadataOffset != newMetadataOffset) {
|
if (d->metadataOffset != newMetadataOffset) {
|
||||||
insert(ByteVector::fromLongLong(newMetadataOffset, false), 20, 8);
|
insert(ByteVector::fromUInt64LE(newMetadataOffset), 20, 8);
|
||||||
d->metadataOffset = newMetadataOffset;
|
d->metadataOffset = newMetadataOffset;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -173,6 +157,7 @@ bool DSF::File::save()
|
|||||||
}
|
}
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
@@ -180,41 +165,41 @@ bool DSF::File::save()
|
|||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
|
|
||||||
void DSF::File::read(bool, Properties::ReadStyle propertiesStyle)
|
void DSF::File::read(bool, AudioProperties::ReadStyle propertiesStyle) {
|
||||||
{
|
|
||||||
// A DSF file consists of four chunks: DSD chunk, format chunk, data chunk, and metadata chunk
|
// A DSF file consists of four chunks: DSD chunk, format chunk, data chunk, and metadata chunk
|
||||||
// The file format is not chunked in the sense of a RIFF File, though
|
// The file format is not chunked in the sense of a RIFF File, though
|
||||||
|
|
||||||
// DSD chunk
|
// DSD chunk
|
||||||
ByteVector chunkName = readBlock(4);
|
ByteVector chunkName = readBlock(4);
|
||||||
if(chunkName != "DSD ") {
|
if (chunkName != "DSD ") {
|
||||||
debug("DSF::File::read() -- Not a DSF file.");
|
debug("DSF::File::read() -- Not a DSF file.");
|
||||||
setValid(false);
|
setValid(false);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
long long chunkSize = readBlock(8).toLongLong(false);
|
long long chunkSize = readBlock(8).toInt64LE(0);
|
||||||
|
|
||||||
// Integrity check
|
// Integrity check
|
||||||
if(28 != chunkSize) {
|
if (28 != chunkSize) {
|
||||||
debug("DSF::File::read() -- File is corrupted, wrong chunk size");
|
debug("DSF::File::read() -- File is corrupted, wrong chunk size");
|
||||||
setValid(false);
|
setValid(false);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
d->fileSize = readBlock(8).toLongLong(false);
|
d->fileSize = readBlock(8).toInt64LE(0);
|
||||||
|
|
||||||
// File is malformed or corrupted
|
// File is malformed or corrupted
|
||||||
if(d->fileSize != length()) {
|
if (d->fileSize != length()) {
|
||||||
debug("DSF::File::read() -- File is corrupted wrong length");
|
debug("DSF::File::read() -- File is corrupted wrong length");
|
||||||
setValid(false);
|
setValid(false);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
d->metadataOffset = readBlock(8).toLongLong(false);
|
d->metadataOffset = readBlock(8).toInt64LE(0);
|
||||||
|
|
||||||
// File is malformed or corrupted
|
// File is malformed or corrupted
|
||||||
if(d->metadataOffset > d->fileSize) {
|
if (d->metadataOffset > d->fileSize) {
|
||||||
debug("DSF::File::read() -- Invalid metadata offset.");
|
debug("DSF::File::read() -- Invalid metadata offset.");
|
||||||
setValid(false);
|
setValid(false);
|
||||||
return;
|
return;
|
||||||
@@ -222,22 +207,22 @@ void DSF::File::read(bool, Properties::ReadStyle propertiesStyle)
|
|||||||
|
|
||||||
// Format chunk
|
// Format chunk
|
||||||
chunkName = readBlock(4);
|
chunkName = readBlock(4);
|
||||||
if(chunkName != "fmt ") {
|
if (chunkName != "fmt ") {
|
||||||
debug("DSF::File::read() -- Missing 'fmt ' chunk.");
|
debug("DSF::File::read() -- Missing 'fmt ' chunk.");
|
||||||
setValid(false);
|
setValid(false);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
chunkSize = readBlock(8).toLongLong(false);
|
chunkSize = readBlock(8).toInt64LE(0);
|
||||||
|
|
||||||
d->properties = new Properties(readBlock(chunkSize), propertiesStyle);
|
d->properties.reset(new AudioProperties(readBlock(chunkSize), propertiesStyle));
|
||||||
|
|
||||||
// Skip the data chunk
|
// Skip the data chunk
|
||||||
|
|
||||||
// A metadata offset of 0 indicates the absence of an ID3v2 tag
|
// A metadata offset of 0 indicates the absence of an ID3v2 tag
|
||||||
if(0 == d->metadataOffset)
|
if (0 == d->metadataOffset)
|
||||||
d->tag = new ID3v2::Tag();
|
d->tag.reset(new ID3v2::Tag());
|
||||||
else
|
else
|
||||||
d->tag = new ID3v2::Tag(this, d->metadataOffset);
|
d->tag.reset(new ID3v2::Tag(this, d->metadataOffset));
|
||||||
}
|
|
||||||
|
|
||||||
|
}
|
||||||
|
|||||||
155
3rdparty/taglib/dsf/dsffile.h
vendored
155
3rdparty/taglib/dsf/dsffile.h
vendored
@@ -33,98 +33,93 @@
|
|||||||
namespace Strawberry_TagLib {
|
namespace Strawberry_TagLib {
|
||||||
namespace TagLib {
|
namespace TagLib {
|
||||||
|
|
||||||
//! An implementation of DSF metadata
|
//! An implementation of DSF metadata
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* This is implementation of DSF metadata.
|
||||||
|
*
|
||||||
|
* This supports an ID3v2 tag as well as properties from the file.
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace DSF {
|
||||||
|
|
||||||
|
//! An implementation of Strawberry_TagLib::TagLib::File with DSF specific methods
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* This implements and provides an interface for DSF files to the
|
||||||
|
* Strawberry_TagLib::TagLib::Tag and Strawberry_TagLib::TagLib::AudioProperties interfaces by way of implementing
|
||||||
|
* the abstract Strawberry_TagLib::TagLib::File API as well as providing some additional information specific to DSF files.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
class TAGLIB_EXPORT File : public Strawberry_TagLib::TagLib::File {
|
||||||
|
public:
|
||||||
|
/*!
|
||||||
|
* Constructs an DSF file from \a file.
|
||||||
|
* If \a readProperties is true the file's audio properties will also be read using \a propertiesStyle.
|
||||||
|
* If false, \a propertiesStyle is ignored.
|
||||||
|
*/
|
||||||
|
explicit File(FileName file, bool readProperties = true, AudioProperties::ReadStyle propertiesStyle = AudioProperties::Average);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* This is implementation of DSF metadata.
|
* Constructs an DSF file from \a file.
|
||||||
*
|
* If \a readProperties is true the file's audio properties will also be read using \a propertiesStyle.
|
||||||
* This supports an ID3v2 tag as well as properties from the file.
|
* If false, \a propertiesStyle is ignored.
|
||||||
*/
|
*/
|
||||||
|
explicit File(IOStream *stream, bool readProperties = true, AudioProperties::ReadStyle propertiesStyle = AudioProperties::Average);
|
||||||
|
|
||||||
namespace DSF {
|
/*!
|
||||||
|
* Destroys this instance of the File.
|
||||||
|
*/
|
||||||
|
~File() override;
|
||||||
|
|
||||||
//! An implementation of Strawberry_TagLib::TagLib::File with DSF specific methods
|
/*!
|
||||||
|
* Returns the Tag for this file.
|
||||||
|
*/
|
||||||
|
ID3v2::Tag *tag() const override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* This implements and provides an interface for DSF files to the
|
* Implements the unified property interface -- export function.
|
||||||
* Strawberry_TagLib::TagLib::Tag and Strawberry_TagLib::TagLib::AudioProperties interfaces by way of implementing
|
* This method forwards to ID3v2::Tag::properties().
|
||||||
* the abstract Strawberry_TagLib::TagLib::File API as well as providing some additional
|
*/
|
||||||
* information specific to DSF files.
|
PropertyMap properties() const override;
|
||||||
*/
|
|
||||||
|
|
||||||
class TAGLIB_EXPORT File : public Strawberry_TagLib::TagLib::File
|
/*!
|
||||||
{
|
* Implements the unified property interface -- import function.
|
||||||
public:
|
* This method forwards to ID3v2::Tag::setProperties().
|
||||||
/*!
|
*/
|
||||||
* Constructs an DSF file from \a file. If \a readProperties is true the
|
PropertyMap setProperties(const PropertyMap &) override;
|
||||||
* file's audio properties will also be read using \a propertiesStyle. If
|
|
||||||
* false, \a propertiesStyle is ignored.
|
|
||||||
*/
|
|
||||||
File(FileName file, bool readProperties = true,
|
|
||||||
Properties::ReadStyle propertiesStyle = Properties::Average);
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Constructs an DSF file from \a file. If \a readProperties is true the
|
* Returns the DSF::AudioProperties for this file.
|
||||||
* file's audio properties will also be read using \a propertiesStyle. If
|
* If no audio properties were read then this will return a null pointer.
|
||||||
* false, \a propertiesStyle is ignored.
|
*/
|
||||||
*/
|
AudioProperties *audioProperties() const override;
|
||||||
File(IOStream *stream, bool readProperties = true,
|
|
||||||
Properties::ReadStyle propertiesStyle = Properties::Average);
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Destroys this instance of the File.
|
* Saves the file.
|
||||||
*/
|
*/
|
||||||
virtual ~File();
|
bool save() override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the Tag for this file.
|
* Returns whether or not the given \a stream can be opened as a DSF file.
|
||||||
*/
|
*
|
||||||
ID3v2::Tag *tag() const;
|
* \note This method is designed to do a quick check.
|
||||||
|
* The result may not necessarily be correct.
|
||||||
|
*/
|
||||||
|
static bool isSupported(IOStream *stream);
|
||||||
|
|
||||||
/*!
|
private:
|
||||||
* Implements the unified property interface -- export function.
|
File(const File &);
|
||||||
* This method forwards to ID3v2::Tag::properties().
|
File &operator=(const File &);
|
||||||
*/
|
|
||||||
PropertyMap properties() const;
|
|
||||||
|
|
||||||
/*!
|
void read(bool readProperties, AudioProperties::ReadStyle propertiesStyle);
|
||||||
* Implements the unified property interface -- import function.
|
|
||||||
* This method forwards to ID3v2::Tag::setProperties().
|
|
||||||
*/
|
|
||||||
PropertyMap setProperties(const PropertyMap &);
|
|
||||||
|
|
||||||
/*!
|
class FilePrivate;
|
||||||
* Returns the DSF::AudioProperties for this file. If no audio properties
|
FilePrivate *d;
|
||||||
* were read then this will return a null pointer.
|
};
|
||||||
*/
|
} // namespace DSF
|
||||||
virtual Properties *audioProperties() const;
|
} // namespace TagLib
|
||||||
|
} // namespace Strawberry_TagLib
|
||||||
/*!
|
|
||||||
* Saves the file.
|
|
||||||
*/
|
|
||||||
virtual bool save();
|
|
||||||
|
|
||||||
/*!
|
|
||||||
* Returns whether or not the given \a stream can be opened as a DSF
|
|
||||||
* file.
|
|
||||||
*
|
|
||||||
* \note This method is designed to do a quick check. The result may
|
|
||||||
* not necessarily be correct.
|
|
||||||
*/
|
|
||||||
static bool isSupported(IOStream *stream);
|
|
||||||
|
|
||||||
private:
|
|
||||||
File(const File &);
|
|
||||||
File &operator=(const File &);
|
|
||||||
|
|
||||||
void read(bool readProperties, Properties::ReadStyle propertiesStyle);
|
|
||||||
|
|
||||||
class FilePrivate;
|
|
||||||
FilePrivate *d;
|
|
||||||
};
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|||||||
106
3rdparty/taglib/dsf/dsfproperties.cpp
vendored
106
3rdparty/taglib/dsf/dsfproperties.cpp
vendored
@@ -23,28 +23,25 @@
|
|||||||
* http://www.mozilla.org/MPL/ *
|
* http://www.mozilla.org/MPL/ *
|
||||||
***************************************************************************/
|
***************************************************************************/
|
||||||
|
|
||||||
#include <tstring.h>
|
#include "tstring.h"
|
||||||
#include <tdebug.h>
|
#include "tdebug.h"
|
||||||
|
|
||||||
#include "dsfproperties.h"
|
#include "dsfproperties.h"
|
||||||
|
|
||||||
using namespace Strawberry_TagLib::TagLib;
|
using namespace Strawberry_TagLib::TagLib;
|
||||||
|
|
||||||
class DSF::Properties::PropertiesPrivate
|
class DSF::AudioProperties::AudioPropertiesPrivate {
|
||||||
{
|
public:
|
||||||
public:
|
explicit AudioPropertiesPrivate() : formatVersion(0),
|
||||||
PropertiesPrivate() :
|
formatID(0),
|
||||||
formatVersion(0),
|
channelType(0),
|
||||||
formatID(0),
|
channelNum(0),
|
||||||
channelType(0),
|
samplingFrequency(0),
|
||||||
channelNum(0),
|
bitsPerSample(0),
|
||||||
samplingFrequency(0),
|
sampleCount(0),
|
||||||
bitsPerSample(0),
|
blockSizePerChannel(0),
|
||||||
sampleCount(0),
|
bitrate(0),
|
||||||
blockSizePerChannel(0),
|
length(0) {
|
||||||
bitrate(0),
|
|
||||||
length(0)
|
|
||||||
{
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Nomenclature is from DSF file format specification
|
// Nomenclature is from DSF file format specification
|
||||||
@@ -58,83 +55,64 @@ public:
|
|||||||
unsigned int blockSizePerChannel;
|
unsigned int blockSizePerChannel;
|
||||||
|
|
||||||
// Computed
|
// Computed
|
||||||
unsigned int bitrate;
|
int bitrate;
|
||||||
unsigned int length;
|
int length;
|
||||||
};
|
};
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
// public members
|
// public members
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
DSF::Properties::Properties(const ByteVector &data, ReadStyle style) : Strawberry_TagLib::TagLib::AudioProperties(style)
|
DSF::AudioProperties::AudioProperties(const ByteVector &data, ReadStyle) : Strawberry_TagLib::TagLib::AudioProperties(), d(new AudioPropertiesPrivate()) {
|
||||||
{
|
|
||||||
d = new PropertiesPrivate;
|
|
||||||
read(data);
|
read(data);
|
||||||
}
|
}
|
||||||
|
|
||||||
DSF::Properties::~Properties()
|
DSF::AudioProperties::~AudioProperties() {
|
||||||
{
|
|
||||||
delete d;
|
delete d;
|
||||||
}
|
}
|
||||||
|
|
||||||
int DSF::Properties::length() const
|
int DSF::AudioProperties::lengthInSeconds() const {
|
||||||
{
|
|
||||||
return lengthInSeconds();
|
|
||||||
}
|
|
||||||
|
|
||||||
int DSF::Properties::lengthInSeconds() const
|
|
||||||
{
|
|
||||||
return d->length / 1000;
|
return d->length / 1000;
|
||||||
}
|
}
|
||||||
|
|
||||||
int DSF::Properties::lengthInMilliseconds() const
|
int DSF::AudioProperties::lengthInMilliseconds() const {
|
||||||
{
|
|
||||||
return d->length;
|
return d->length;
|
||||||
}
|
}
|
||||||
|
|
||||||
int DSF::Properties::bitrate() const
|
int DSF::AudioProperties::bitrate() const {
|
||||||
{
|
|
||||||
return d->bitrate;
|
return d->bitrate;
|
||||||
}
|
}
|
||||||
|
|
||||||
int DSF::Properties::sampleRate() const
|
int DSF::AudioProperties::sampleRate() const {
|
||||||
{
|
|
||||||
return d->samplingFrequency;
|
return d->samplingFrequency;
|
||||||
}
|
}
|
||||||
|
|
||||||
int DSF::Properties::channels() const
|
int DSF::AudioProperties::channels() const {
|
||||||
{
|
|
||||||
return d->channelNum;
|
return d->channelNum;
|
||||||
}
|
}
|
||||||
|
|
||||||
// DSF specific
|
// DSF specific
|
||||||
int DSF::Properties::formatVersion() const
|
int DSF::AudioProperties::formatVersion() const {
|
||||||
{
|
|
||||||
return d->formatVersion;
|
return d->formatVersion;
|
||||||
}
|
}
|
||||||
|
|
||||||
int DSF::Properties::formatID() const
|
int DSF::AudioProperties::formatID() const {
|
||||||
{
|
|
||||||
return d->formatID;
|
return d->formatID;
|
||||||
}
|
}
|
||||||
|
|
||||||
int DSF::Properties::channelType() const
|
int DSF::AudioProperties::channelType() const {
|
||||||
{
|
|
||||||
return d->channelType;
|
return d->channelType;
|
||||||
}
|
}
|
||||||
|
|
||||||
int DSF::Properties::bitsPerSample() const
|
int DSF::AudioProperties::bitsPerSample() const {
|
||||||
{
|
|
||||||
return d->bitsPerSample;
|
return d->bitsPerSample;
|
||||||
}
|
}
|
||||||
|
|
||||||
long long DSF::Properties::sampleCount() const
|
long long DSF::AudioProperties::sampleCount() const {
|
||||||
{
|
|
||||||
return d->sampleCount;
|
return d->sampleCount;
|
||||||
}
|
}
|
||||||
|
|
||||||
int DSF::Properties::blockSizePerChannel() const
|
int DSF::AudioProperties::blockSizePerChannel() const {
|
||||||
{
|
|
||||||
return d->blockSizePerChannel;
|
return d->blockSizePerChannel;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -142,20 +120,16 @@ int DSF::Properties::blockSizePerChannel() const
|
|||||||
// private members
|
// private members
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
void DSF::Properties::read(const ByteVector &data)
|
void DSF::AudioProperties::read(const ByteVector &data) {
|
||||||
{
|
d->formatVersion = data.toUInt32LE(0);
|
||||||
d->formatVersion = data.toUInt(0U,false);
|
d->formatID = data.toUInt32LE(4);
|
||||||
d->formatID = data.toUInt(4U,false);
|
d->channelType = data.toUInt32LE(8);
|
||||||
d->channelType = data.toUInt(8U,false);
|
d->channelNum = data.toUInt32LE(12);
|
||||||
d->channelNum = data.toUInt(12U,false);
|
d->samplingFrequency = data.toUInt32LE(16);
|
||||||
d->samplingFrequency = data.toUInt(16U,false);
|
d->bitsPerSample = data.toUInt32LE(20);
|
||||||
d->bitsPerSample = data.toUInt(20U,false);
|
d->sampleCount = data.toInt64LE(24);
|
||||||
d->sampleCount = data.toLongLong(24U,false);
|
d->blockSizePerChannel = data.toUInt32LE(32);
|
||||||
d->blockSizePerChannel = data.toUInt(32U,false);
|
|
||||||
|
|
||||||
d->bitrate
|
d->bitrate = static_cast<int>((d->samplingFrequency * d->bitsPerSample * d->channelNum) / 1000.0 + 0.5);
|
||||||
= static_cast<unsigned int>((d->samplingFrequency * d->bitsPerSample * d->channelNum) / 1000.0 + 0.5);
|
d->length = d->samplingFrequency > 0 ? static_cast<int>(d->sampleCount * 1000.0 / d->samplingFrequency + 0.5) : 0;
|
||||||
d->length
|
|
||||||
= d->samplingFrequency > 0 ? static_cast<unsigned int>(d->sampleCount * 1000.0 / d->samplingFrequency + 0.5) : 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
96
3rdparty/taglib/dsf/dsfproperties.h
vendored
96
3rdparty/taglib/dsf/dsfproperties.h
vendored
@@ -30,65 +30,65 @@
|
|||||||
|
|
||||||
namespace Strawberry_TagLib {
|
namespace Strawberry_TagLib {
|
||||||
namespace TagLib {
|
namespace TagLib {
|
||||||
|
namespace DSF {
|
||||||
|
|
||||||
namespace DSF {
|
class File;
|
||||||
|
|
||||||
class File;
|
//! An implementation of audio property reading for DSF
|
||||||
|
|
||||||
//! An implementation of audio property reading for DSF
|
/*!
|
||||||
|
* This reads the data from a DSF stream found in the AudioProperties API.
|
||||||
|
*/
|
||||||
|
|
||||||
/*!
|
class TAGLIB_EXPORT AudioProperties : public Strawberry_TagLib::TagLib::AudioProperties {
|
||||||
* This reads the data from a DSF stream found in the AudioProperties
|
public:
|
||||||
* API.
|
/*!
|
||||||
*/
|
* Create an instance of DSF::AudioProperties with the data read from the ByteVector \a data.
|
||||||
|
*/
|
||||||
|
explicit AudioProperties(const ByteVector &data, ReadStyle);
|
||||||
|
|
||||||
class TAGLIB_EXPORT Properties : public Strawberry_TagLib::TagLib::AudioProperties
|
/*!
|
||||||
{
|
* Destroys this DSF::AudioProperties instance.
|
||||||
public:
|
*/
|
||||||
/*!
|
~AudioProperties() override;
|
||||||
* Create an instance of DSF::AudioProperties with the data read from the
|
|
||||||
* ByteVector \a data.
|
|
||||||
*/
|
|
||||||
Properties(const ByteVector &data, ReadStyle style);
|
|
||||||
|
|
||||||
/*!
|
// Reimplementations.
|
||||||
* Destroys this DSF::AudioProperties instance.
|
|
||||||
*/
|
|
||||||
virtual ~Properties();
|
|
||||||
|
|
||||||
// Reimplementations.
|
int lengthInSeconds() const override;
|
||||||
|
int lengthInMilliseconds() const override;
|
||||||
|
int bitrate() const override;
|
||||||
|
int sampleRate() const override;
|
||||||
|
int channels() const override;
|
||||||
|
|
||||||
virtual int length() const;
|
int formatVersion() const;
|
||||||
virtual int lengthInSeconds() const;
|
int formatID() const;
|
||||||
virtual int lengthInMilliseconds() const;
|
|
||||||
virtual int bitrate() const;
|
|
||||||
virtual int sampleRate() const;
|
|
||||||
virtual int channels() const;
|
|
||||||
|
|
||||||
int formatVersion() const;
|
/*!
|
||||||
int formatID() const;
|
* Channel type values:
|
||||||
|
* 1 = mono,
|
||||||
|
* 2 = stereo,
|
||||||
|
* 3 = 3 channels,
|
||||||
|
* 4 = quad,
|
||||||
|
* 5 = 4 channels,
|
||||||
|
* 6 = 5 channels,
|
||||||
|
* 7 = 5.1 channels
|
||||||
|
*/
|
||||||
|
int channelType() const;
|
||||||
|
int bitsPerSample() const;
|
||||||
|
long long sampleCount() const;
|
||||||
|
int blockSizePerChannel() const;
|
||||||
|
|
||||||
/*!
|
private:
|
||||||
* Channel type values: 1 = mono, 2 = stereo, 3 = 3 channels,
|
AudioProperties(const AudioProperties&);
|
||||||
* 4 = quad, 5 = 4 channels, 6 = 5 channels, 7 = 5.1 channels
|
AudioProperties &operator=(const AudioProperties&);
|
||||||
*/
|
|
||||||
int channelType() const;
|
|
||||||
int bitsPerSample() const;
|
|
||||||
long long sampleCount() const;
|
|
||||||
int blockSizePerChannel() const;
|
|
||||||
|
|
||||||
private:
|
void read(const ByteVector &data);
|
||||||
Properties(const Properties &);
|
|
||||||
Properties &operator=(const Properties &);
|
|
||||||
|
|
||||||
void read(const ByteVector &data);
|
class AudioPropertiesPrivate;
|
||||||
|
AudioPropertiesPrivate *d;
|
||||||
class PropertiesPrivate;
|
};
|
||||||
PropertiesPrivate *d;
|
} // namespace DSF
|
||||||
};
|
} // namespace TagLib
|
||||||
}
|
} // namespace Strawberry_TagLib
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|||||||
549
3rdparty/taglib/fileref.cpp
vendored
549
3rdparty/taglib/fileref.cpp
vendored
@@ -27,11 +27,12 @@
|
|||||||
* http://www.mozilla.org/MPL/ *
|
* http://www.mozilla.org/MPL/ *
|
||||||
***************************************************************************/
|
***************************************************************************/
|
||||||
|
|
||||||
#include <tfile.h>
|
#include <memory>
|
||||||
#include <tfilestream.h>
|
|
||||||
#include <tstring.h>
|
#include "tfile.h"
|
||||||
#include <tdebug.h>
|
#include "tfilestream.h"
|
||||||
#include <trefcounter.h>
|
#include "tstring.h"
|
||||||
|
#include "tdebug.h"
|
||||||
|
|
||||||
#include "fileref.h"
|
#include "fileref.h"
|
||||||
#include "asffile.h"
|
#include "asffile.h"
|
||||||
@@ -57,320 +58,259 @@
|
|||||||
|
|
||||||
using namespace Strawberry_TagLib::TagLib;
|
using namespace Strawberry_TagLib::TagLib;
|
||||||
|
|
||||||
namespace
|
namespace {
|
||||||
{
|
typedef List<const FileRef::FileTypeResolver*> ResolverList;
|
||||||
typedef List<const FileRef::FileTypeResolver *> ResolverList;
|
ResolverList fileTypeResolvers;
|
||||||
ResolverList fileTypeResolvers;
|
|
||||||
|
|
||||||
// Detect the file type by user-defined resolvers.
|
// Detect the file type by user-defined resolvers.
|
||||||
|
|
||||||
File *detectByResolvers(FileName fileName, bool readAudioProperties,
|
File *detectByResolvers(FileName fileName, bool readAudioProperties, AudioProperties::ReadStyle audioPropertiesStyle) {
|
||||||
AudioProperties::ReadStyle audioPropertiesStyle)
|
|
||||||
{
|
|
||||||
ResolverList::ConstIterator it = fileTypeResolvers.begin();
|
|
||||||
for(; it != fileTypeResolvers.end(); ++it) {
|
|
||||||
File *file = (*it)->createFile(fileName, readAudioProperties, audioPropertiesStyle);
|
|
||||||
if(file)
|
|
||||||
return file;
|
|
||||||
}
|
|
||||||
|
|
||||||
return nullptr;
|
ResolverList::ConstIterator it = fileTypeResolvers.begin();
|
||||||
}
|
for (; it != fileTypeResolvers.end(); ++it) {
|
||||||
|
File *file = (*it)->createFile(fileName, readAudioProperties, audioPropertiesStyle);
|
||||||
// Detect the file type based on the file extension.
|
if (file)
|
||||||
|
|
||||||
File* detectByExtension(IOStream *stream, bool readAudioProperties,
|
|
||||||
AudioProperties::ReadStyle audioPropertiesStyle)
|
|
||||||
{
|
|
||||||
#ifdef _WIN32
|
|
||||||
const String s = stream->name().toString();
|
|
||||||
#else
|
|
||||||
const String s(stream->name());
|
|
||||||
#endif
|
|
||||||
|
|
||||||
String ext;
|
|
||||||
const int pos = s.rfind(".");
|
|
||||||
if(pos != -1)
|
|
||||||
ext = s.substr(pos + 1).upper();
|
|
||||||
|
|
||||||
// If this list is updated, the method defaultFileExtensions() should also be
|
|
||||||
// updated. However at some point that list should be created at the same time
|
|
||||||
// that a default file type resolver is created.
|
|
||||||
|
|
||||||
if(ext.isEmpty())
|
|
||||||
return nullptr;
|
|
||||||
|
|
||||||
// .oga can be any audio in the Ogg container. So leave it to content-based detection.
|
|
||||||
|
|
||||||
if(ext == "MP3")
|
|
||||||
return new MPEG::File(stream, ID3v2::FrameFactory::instance(), readAudioProperties, audioPropertiesStyle);
|
|
||||||
if(ext == "OGG")
|
|
||||||
return new Ogg::Vorbis::File(stream, readAudioProperties, audioPropertiesStyle);
|
|
||||||
if(ext == "FLAC")
|
|
||||||
return new FLAC::File(stream, ID3v2::FrameFactory::instance(), readAudioProperties, audioPropertiesStyle);
|
|
||||||
if(ext == "MPC")
|
|
||||||
return new MPC::File(stream, readAudioProperties, audioPropertiesStyle);
|
|
||||||
if(ext == "WV")
|
|
||||||
return new WavPack::File(stream, readAudioProperties, audioPropertiesStyle);
|
|
||||||
if(ext == "SPX")
|
|
||||||
return new Ogg::Speex::File(stream, readAudioProperties, audioPropertiesStyle);
|
|
||||||
if(ext == "OPUS")
|
|
||||||
return new Ogg::Opus::File(stream, readAudioProperties, audioPropertiesStyle);
|
|
||||||
if(ext == "TTA")
|
|
||||||
return new TrueAudio::File(stream, readAudioProperties, audioPropertiesStyle);
|
|
||||||
if(ext == "M4A" || ext == "M4R" || ext == "M4B" || ext == "M4P" || ext == "MP4" || ext == "3G2" || ext == "M4V")
|
|
||||||
return new MP4::File(stream, readAudioProperties, audioPropertiesStyle);
|
|
||||||
if(ext == "WMA" || ext == "ASF")
|
|
||||||
return new ASF::File(stream, readAudioProperties, audioPropertiesStyle);
|
|
||||||
if(ext == "AIF" || ext == "AIFF" || ext == "AFC" || ext == "AIFC")
|
|
||||||
return new RIFF::AIFF::File(stream, readAudioProperties, audioPropertiesStyle);
|
|
||||||
if(ext == "WAV")
|
|
||||||
return new RIFF::WAV::File(stream, readAudioProperties, audioPropertiesStyle);
|
|
||||||
if(ext == "APE")
|
|
||||||
return new APE::File(stream, readAudioProperties, audioPropertiesStyle);
|
|
||||||
// module, nst and wow are possible but uncommon extensions
|
|
||||||
if(ext == "MOD" || ext == "MODULE" || ext == "NST" || ext == "WOW")
|
|
||||||
return new Mod::File(stream, readAudioProperties, audioPropertiesStyle);
|
|
||||||
if(ext == "S3M")
|
|
||||||
return new S3M::File(stream, readAudioProperties, audioPropertiesStyle);
|
|
||||||
if(ext == "IT")
|
|
||||||
return new IT::File(stream, readAudioProperties, audioPropertiesStyle);
|
|
||||||
if(ext == "XM")
|
|
||||||
return new XM::File(stream, readAudioProperties, audioPropertiesStyle);
|
|
||||||
if(ext == "DFF" || ext == "DSDIFF")
|
|
||||||
return new DSDIFF::File(stream, readAudioProperties, audioPropertiesStyle);
|
|
||||||
if(ext == "DSF")
|
|
||||||
return new DSF::File(stream, readAudioProperties, audioPropertiesStyle);
|
|
||||||
|
|
||||||
return nullptr;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Detect the file type based on the actual content of the stream.
|
|
||||||
|
|
||||||
File *detectByContent(IOStream *stream, bool readAudioProperties,
|
|
||||||
AudioProperties::ReadStyle audioPropertiesStyle)
|
|
||||||
{
|
|
||||||
File *file = 0;
|
|
||||||
|
|
||||||
if(MPEG::File::isSupported(stream))
|
|
||||||
file = new MPEG::File(stream, ID3v2::FrameFactory::instance(), readAudioProperties, audioPropertiesStyle);
|
|
||||||
else if(Ogg::Vorbis::File::isSupported(stream))
|
|
||||||
file = new Ogg::Vorbis::File(stream, readAudioProperties, audioPropertiesStyle);
|
|
||||||
else if(Ogg::FLAC::File::isSupported(stream))
|
|
||||||
file = new Ogg::FLAC::File(stream, readAudioProperties, audioPropertiesStyle);
|
|
||||||
else if(FLAC::File::isSupported(stream))
|
|
||||||
file = new FLAC::File(stream, ID3v2::FrameFactory::instance(), readAudioProperties, audioPropertiesStyle);
|
|
||||||
else if(MPC::File::isSupported(stream))
|
|
||||||
file = new MPC::File(stream, readAudioProperties, audioPropertiesStyle);
|
|
||||||
else if(WavPack::File::isSupported(stream))
|
|
||||||
file = new WavPack::File(stream, readAudioProperties, audioPropertiesStyle);
|
|
||||||
else if(Ogg::Speex::File::isSupported(stream))
|
|
||||||
file = new Ogg::Speex::File(stream, readAudioProperties, audioPropertiesStyle);
|
|
||||||
else if(Ogg::Opus::File::isSupported(stream))
|
|
||||||
file = new Ogg::Opus::File(stream, readAudioProperties, audioPropertiesStyle);
|
|
||||||
else if(TrueAudio::File::isSupported(stream))
|
|
||||||
file = new TrueAudio::File(stream, readAudioProperties, audioPropertiesStyle);
|
|
||||||
else if(MP4::File::isSupported(stream))
|
|
||||||
file = new MP4::File(stream, readAudioProperties, audioPropertiesStyle);
|
|
||||||
else if(ASF::File::isSupported(stream))
|
|
||||||
file = new ASF::File(stream, readAudioProperties, audioPropertiesStyle);
|
|
||||||
else if(RIFF::AIFF::File::isSupported(stream))
|
|
||||||
file = new RIFF::AIFF::File(stream, readAudioProperties, audioPropertiesStyle);
|
|
||||||
else if(RIFF::WAV::File::isSupported(stream))
|
|
||||||
file = new RIFF::WAV::File(stream, readAudioProperties, audioPropertiesStyle);
|
|
||||||
else if(APE::File::isSupported(stream))
|
|
||||||
file = new APE::File(stream, readAudioProperties, audioPropertiesStyle);
|
|
||||||
else if(DSDIFF::File::isSupported(stream))
|
|
||||||
file = new DSDIFF::File(stream, readAudioProperties, audioPropertiesStyle);
|
|
||||||
else if(DSF::File::isSupported(stream))
|
|
||||||
file = new DSF::File(stream, readAudioProperties, audioPropertiesStyle);
|
|
||||||
|
|
||||||
// isSupported() only does a quick check, so double check the file here.
|
|
||||||
|
|
||||||
if(file) {
|
|
||||||
if(file->isValid())
|
|
||||||
return file;
|
|
||||||
else
|
|
||||||
delete file;
|
|
||||||
}
|
|
||||||
|
|
||||||
return nullptr;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Internal function that supports FileRef::create().
|
|
||||||
// This looks redundant, but necessary in order not to change the previous
|
|
||||||
// behavior of FileRef::create().
|
|
||||||
|
|
||||||
File* createInternal(FileName fileName, bool readAudioProperties,
|
|
||||||
AudioProperties::ReadStyle audioPropertiesStyle)
|
|
||||||
{
|
|
||||||
File *file = detectByResolvers(fileName, readAudioProperties, audioPropertiesStyle);
|
|
||||||
if(file)
|
|
||||||
return file;
|
return file;
|
||||||
|
|
||||||
#ifdef _WIN32
|
|
||||||
const String s = fileName.toString();
|
|
||||||
#else
|
|
||||||
const String s(fileName);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
String ext;
|
|
||||||
const int pos = s.rfind(".");
|
|
||||||
if(pos != -1)
|
|
||||||
ext = s.substr(pos + 1).upper();
|
|
||||||
|
|
||||||
if(ext.isEmpty())
|
|
||||||
return nullptr;
|
|
||||||
|
|
||||||
if(ext == "MP3")
|
|
||||||
return new MPEG::File(fileName, ID3v2::FrameFactory::instance(), readAudioProperties, audioPropertiesStyle);
|
|
||||||
if(ext == "OGG")
|
|
||||||
return new Ogg::Vorbis::File(fileName, readAudioProperties, audioPropertiesStyle);
|
|
||||||
if(ext == "OGA") {
|
|
||||||
/* .oga can be any audio in the Ogg container. First try FLAC, then Vorbis. */
|
|
||||||
File *file_flac = new Ogg::FLAC::File(fileName, readAudioProperties, audioPropertiesStyle);
|
|
||||||
if (file_flac->isValid())
|
|
||||||
return file_flac;
|
|
||||||
delete file_flac;
|
|
||||||
return new Ogg::Vorbis::File(fileName, readAudioProperties, audioPropertiesStyle);
|
|
||||||
}
|
|
||||||
if(ext == "FLAC")
|
|
||||||
return new FLAC::File(fileName, ID3v2::FrameFactory::instance(), readAudioProperties, audioPropertiesStyle);
|
|
||||||
if(ext == "MPC")
|
|
||||||
return new MPC::File(fileName, readAudioProperties, audioPropertiesStyle);
|
|
||||||
if(ext == "WV")
|
|
||||||
return new WavPack::File(fileName, readAudioProperties, audioPropertiesStyle);
|
|
||||||
if(ext == "SPX")
|
|
||||||
return new Ogg::Speex::File(fileName, readAudioProperties, audioPropertiesStyle);
|
|
||||||
if(ext == "OPUS")
|
|
||||||
return new Ogg::Opus::File(fileName, readAudioProperties, audioPropertiesStyle);
|
|
||||||
if(ext == "TTA")
|
|
||||||
return new TrueAudio::File(fileName, readAudioProperties, audioPropertiesStyle);
|
|
||||||
if(ext == "M4A" || ext == "M4R" || ext == "M4B" || ext == "M4P" || ext == "MP4" || ext == "3G2" || ext == "M4V")
|
|
||||||
return new MP4::File(fileName, readAudioProperties, audioPropertiesStyle);
|
|
||||||
if(ext == "WMA" || ext == "ASF")
|
|
||||||
return new ASF::File(fileName, readAudioProperties, audioPropertiesStyle);
|
|
||||||
if(ext == "AIF" || ext == "AIFF" || ext == "AFC" || ext == "AIFC")
|
|
||||||
return new RIFF::AIFF::File(fileName, readAudioProperties, audioPropertiesStyle);
|
|
||||||
if(ext == "WAV")
|
|
||||||
return new RIFF::WAV::File(fileName, readAudioProperties, audioPropertiesStyle);
|
|
||||||
if(ext == "APE")
|
|
||||||
return new APE::File(fileName, readAudioProperties, audioPropertiesStyle);
|
|
||||||
// module, nst and wow are possible but uncommon extensions
|
|
||||||
if(ext == "MOD" || ext == "MODULE" || ext == "NST" || ext == "WOW")
|
|
||||||
return new Mod::File(fileName, readAudioProperties, audioPropertiesStyle);
|
|
||||||
if(ext == "S3M")
|
|
||||||
return new S3M::File(fileName, readAudioProperties, audioPropertiesStyle);
|
|
||||||
if(ext == "IT")
|
|
||||||
return new IT::File(fileName, readAudioProperties, audioPropertiesStyle);
|
|
||||||
if(ext == "XM")
|
|
||||||
return new XM::File(fileName, readAudioProperties, audioPropertiesStyle);
|
|
||||||
if(ext == "DFF" || ext == "DSDIFF")
|
|
||||||
return new DSDIFF::File(fileName, readAudioProperties, audioPropertiesStyle);
|
|
||||||
if(ext == "DSF")
|
|
||||||
return new DSF::File(fileName, readAudioProperties, audioPropertiesStyle);
|
|
||||||
|
|
||||||
return nullptr;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return nullptr;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
class FileRef::FileRefPrivate : public RefCounter
|
// Detect the file type based on the file extension.
|
||||||
{
|
|
||||||
public:
|
|
||||||
FileRefPrivate() :
|
|
||||||
RefCounter(),
|
|
||||||
file(0),
|
|
||||||
stream(0) {}
|
|
||||||
|
|
||||||
~FileRefPrivate() {
|
File *detectByExtension(IOStream *stream, bool readAudioProperties, AudioProperties::ReadStyle audioPropertiesStyle) {
|
||||||
|
|
||||||
|
#ifdef _WIN32
|
||||||
|
const String s(stream->name().wstr());
|
||||||
|
#else
|
||||||
|
const String s(stream->name());
|
||||||
|
#endif
|
||||||
|
|
||||||
|
String ext;
|
||||||
|
const size_t pos = s.rfind(".");
|
||||||
|
if (pos != String::npos())
|
||||||
|
ext = s.substr(pos + 1).upper();
|
||||||
|
|
||||||
|
// If this list is updated, the method defaultFileExtensions() should also be
|
||||||
|
// updated. However at some point that list should be created at the same time
|
||||||
|
// that a default file type resolver is created.
|
||||||
|
|
||||||
|
if (ext.isEmpty())
|
||||||
|
return nullptr;
|
||||||
|
|
||||||
|
// .oga can be any audio in the Ogg container. So leave it to content-based detection.
|
||||||
|
|
||||||
|
if (ext == "MP3")
|
||||||
|
return new MPEG::File(stream, ID3v2::FrameFactory::instance(), readAudioProperties, audioPropertiesStyle);
|
||||||
|
if (ext == "OGG")
|
||||||
|
return new Ogg::Vorbis::File(stream, readAudioProperties, audioPropertiesStyle);
|
||||||
|
if (ext == "FLAC")
|
||||||
|
return new FLAC::File(stream, readAudioProperties, audioPropertiesStyle);
|
||||||
|
if (ext == "MPC")
|
||||||
|
return new MPC::File(stream, readAudioProperties, audioPropertiesStyle);
|
||||||
|
if (ext == "WV")
|
||||||
|
return new WavPack::File(stream, readAudioProperties, audioPropertiesStyle);
|
||||||
|
if (ext == "SPX")
|
||||||
|
return new Ogg::Speex::File(stream, readAudioProperties, audioPropertiesStyle);
|
||||||
|
if (ext == "OPUS")
|
||||||
|
return new Ogg::Opus::File(stream, readAudioProperties, audioPropertiesStyle);
|
||||||
|
if (ext == "TTA")
|
||||||
|
return new TrueAudio::File(stream, readAudioProperties, audioPropertiesStyle);
|
||||||
|
if (ext == "M4A" || ext == "M4R" || ext == "M4B" || ext == "M4P" || ext == "MP4" || ext == "3G2" || ext == "M4V")
|
||||||
|
return new MP4::File(stream, readAudioProperties, audioPropertiesStyle);
|
||||||
|
if (ext == "WMA" || ext == "ASF")
|
||||||
|
return new ASF::File(stream, readAudioProperties, audioPropertiesStyle);
|
||||||
|
if (ext == "AIF" || ext == "AIFF" || ext == "AFC" || ext == "AIFC")
|
||||||
|
return new RIFF::AIFF::File(stream, readAudioProperties, audioPropertiesStyle);
|
||||||
|
if (ext == "WAV")
|
||||||
|
return new RIFF::WAV::File(stream, readAudioProperties, audioPropertiesStyle);
|
||||||
|
if (ext == "APE")
|
||||||
|
return new APE::File(stream, readAudioProperties, audioPropertiesStyle);
|
||||||
|
// module, nst and wow are possible but uncommon extensions
|
||||||
|
if (ext == "MOD" || ext == "MODULE" || ext == "NST" || ext == "WOW")
|
||||||
|
return new Mod::File(stream, readAudioProperties, audioPropertiesStyle);
|
||||||
|
if (ext == "S3M")
|
||||||
|
return new S3M::File(stream, readAudioProperties, audioPropertiesStyle);
|
||||||
|
if (ext == "IT")
|
||||||
|
return new IT::File(stream, readAudioProperties, audioPropertiesStyle);
|
||||||
|
if (ext == "XM")
|
||||||
|
return new XM::File(stream, readAudioProperties, audioPropertiesStyle);
|
||||||
|
if (ext == "DFF" || ext == "DSDIFF")
|
||||||
|
return new DSDIFF::File(stream, readAudioProperties, audioPropertiesStyle);
|
||||||
|
if (ext == "DSF")
|
||||||
|
return new DSF::File(stream, readAudioProperties, audioPropertiesStyle);
|
||||||
|
|
||||||
|
return nullptr;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
// Detect the file type based on the actual content of the stream.
|
||||||
|
|
||||||
|
File *detectByContent(IOStream *stream, bool readAudioProperties, AudioProperties::ReadStyle audioPropertiesStyle) {
|
||||||
|
|
||||||
|
File *file = nullptr;
|
||||||
|
|
||||||
|
if (MPEG::File::isSupported(stream))
|
||||||
|
file = new MPEG::File(stream, ID3v2::FrameFactory::instance(), readAudioProperties, audioPropertiesStyle);
|
||||||
|
else if (Ogg::Vorbis::File::isSupported(stream))
|
||||||
|
file = new Ogg::Vorbis::File(stream, readAudioProperties, audioPropertiesStyle);
|
||||||
|
else if (Ogg::FLAC::File::isSupported(stream))
|
||||||
|
file = new Ogg::FLAC::File(stream, readAudioProperties, audioPropertiesStyle);
|
||||||
|
else if (FLAC::File::isSupported(stream))
|
||||||
|
file = new FLAC::File(stream, readAudioProperties, audioPropertiesStyle);
|
||||||
|
else if (MPC::File::isSupported(stream))
|
||||||
|
file = new MPC::File(stream, readAudioProperties, audioPropertiesStyle);
|
||||||
|
else if (WavPack::File::isSupported(stream))
|
||||||
|
file = new WavPack::File(stream, readAudioProperties, audioPropertiesStyle);
|
||||||
|
else if (Ogg::Speex::File::isSupported(stream))
|
||||||
|
file = new Ogg::Speex::File(stream, readAudioProperties, audioPropertiesStyle);
|
||||||
|
else if (Ogg::Opus::File::isSupported(stream))
|
||||||
|
file = new Ogg::Opus::File(stream, readAudioProperties, audioPropertiesStyle);
|
||||||
|
else if (TrueAudio::File::isSupported(stream))
|
||||||
|
file = new TrueAudio::File(stream, readAudioProperties, audioPropertiesStyle);
|
||||||
|
else if (MP4::File::isSupported(stream))
|
||||||
|
file = new MP4::File(stream, readAudioProperties, audioPropertiesStyle);
|
||||||
|
else if (ASF::File::isSupported(stream))
|
||||||
|
file = new ASF::File(stream, readAudioProperties, audioPropertiesStyle);
|
||||||
|
else if (RIFF::AIFF::File::isSupported(stream))
|
||||||
|
file = new RIFF::AIFF::File(stream, readAudioProperties, audioPropertiesStyle);
|
||||||
|
else if (RIFF::WAV::File::isSupported(stream))
|
||||||
|
file = new RIFF::WAV::File(stream, readAudioProperties, audioPropertiesStyle);
|
||||||
|
else if (APE::File::isSupported(stream))
|
||||||
|
file = new APE::File(stream, readAudioProperties, audioPropertiesStyle);
|
||||||
|
else if (DSDIFF::File::isSupported(stream))
|
||||||
|
file = new DSDIFF::File(stream, readAudioProperties, audioPropertiesStyle);
|
||||||
|
else if (DSF::File::isSupported(stream))
|
||||||
|
file = new DSF::File(stream, readAudioProperties, audioPropertiesStyle);
|
||||||
|
|
||||||
|
// isSupported() only does a quick check, so double check the file here.
|
||||||
|
|
||||||
|
if (file) {
|
||||||
|
if (file->isValid())
|
||||||
|
return file;
|
||||||
|
else
|
||||||
|
delete file;
|
||||||
|
}
|
||||||
|
|
||||||
|
return nullptr;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
struct FileRefData {
|
||||||
|
FileRefData() : file(nullptr), stream(nullptr) {}
|
||||||
|
|
||||||
|
~FileRefData() {
|
||||||
delete file;
|
delete file;
|
||||||
delete stream;
|
delete stream;
|
||||||
}
|
}
|
||||||
|
|
||||||
File *file;
|
File *file;
|
||||||
IOStream *stream;
|
IOStream *stream;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
} // namespace
|
||||||
|
|
||||||
|
class FileRef::FileRefPrivate {
|
||||||
|
public:
|
||||||
|
FileRefPrivate() : data(new FileRefData()) {}
|
||||||
|
|
||||||
|
std::shared_ptr<FileRefData> data;
|
||||||
|
};
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
// public members
|
// public members
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
FileRef::FileRef() :
|
FileRef::FileRef() : d(new FileRefPrivate()) {}
|
||||||
d(new FileRefPrivate())
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
FileRef::FileRef(FileName fileName, bool readAudioProperties,
|
FileRef::FileRef(FileName fileName, bool readAudioProperties, AudioProperties::ReadStyle audioPropertiesStyle) : d(new FileRefPrivate()) {
|
||||||
AudioProperties::ReadStyle audioPropertiesStyle) :
|
|
||||||
d(new FileRefPrivate())
|
|
||||||
{
|
|
||||||
parse(fileName, readAudioProperties, audioPropertiesStyle);
|
parse(fileName, readAudioProperties, audioPropertiesStyle);
|
||||||
}
|
}
|
||||||
|
|
||||||
FileRef::FileRef(IOStream* stream, bool readAudioProperties, AudioProperties::ReadStyle audioPropertiesStyle) :
|
FileRef::FileRef(IOStream *stream, bool readAudioProperties, AudioProperties::ReadStyle audioPropertiesStyle) : d(new FileRefPrivate()) {
|
||||||
d(new FileRefPrivate())
|
|
||||||
{
|
|
||||||
parse(stream, readAudioProperties, audioPropertiesStyle);
|
parse(stream, readAudioProperties, audioPropertiesStyle);
|
||||||
}
|
}
|
||||||
|
|
||||||
FileRef::FileRef(File *file) :
|
FileRef::FileRef(File *file) : d(new FileRefPrivate()) {
|
||||||
d(new FileRefPrivate())
|
d->data->file = file;
|
||||||
{
|
|
||||||
d->file = file;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
FileRef::FileRef(const FileRef &ref) :
|
FileRef::FileRef(const FileRef &ref) : d(new FileRefPrivate(*ref.d)) {}
|
||||||
d(ref.d)
|
|
||||||
{
|
FileRef::~FileRef() {
|
||||||
d->ref();
|
delete d;
|
||||||
}
|
}
|
||||||
|
|
||||||
FileRef::~FileRef()
|
Tag *FileRef::tag() const {
|
||||||
{
|
|
||||||
if(d->deref())
|
|
||||||
delete d;
|
|
||||||
}
|
|
||||||
|
|
||||||
Tag *FileRef::tag() const
|
if (isNull()) {
|
||||||
{
|
|
||||||
if(isNull()) {
|
|
||||||
debug("FileRef::tag() - Called without a valid file.");
|
debug("FileRef::tag() - Called without a valid file.");
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
return d->file->tag();
|
return d->data->file->tag();
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
AudioProperties *FileRef::audioProperties() const
|
PropertyMap FileRef::properties() const {
|
||||||
{
|
if (isNull()) {
|
||||||
if(isNull()) {
|
debug("FileRef::properties() - Called without a valid file.");
|
||||||
|
return PropertyMap();
|
||||||
|
}
|
||||||
|
|
||||||
|
return d->data->file->properties();
|
||||||
|
}
|
||||||
|
|
||||||
|
void FileRef::removeUnsupportedProperties(const StringList &properties) {
|
||||||
|
if (isNull()) {
|
||||||
|
debug("FileRef::removeUnsupportedProperties() - Called without a valid file.");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
d->data->file->removeUnsupportedProperties(properties);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
PropertyMap FileRef::setProperties(const PropertyMap &properties) {
|
||||||
|
if (isNull()) {
|
||||||
|
debug("FileRef::setProperties() - Called without a valid file.");
|
||||||
|
return PropertyMap();
|
||||||
|
}
|
||||||
|
|
||||||
|
return d->data->file->setProperties(properties);
|
||||||
|
}
|
||||||
|
|
||||||
|
AudioProperties *FileRef::audioProperties() const {
|
||||||
|
|
||||||
|
if (isNull()) {
|
||||||
debug("FileRef::audioProperties() - Called without a valid file.");
|
debug("FileRef::audioProperties() - Called without a valid file.");
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
return d->file->audioProperties();
|
return d->data->file->audioProperties();
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
File *FileRef::file() const
|
File *FileRef::file() const {
|
||||||
{
|
return d->data->file;
|
||||||
return d->file;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool FileRef::save()
|
bool FileRef::save() {
|
||||||
{
|
|
||||||
if(isNull()) {
|
if (isNull()) {
|
||||||
debug("FileRef::save() - Called without a valid file.");
|
debug("FileRef::save() - Called without a valid file.");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
return d->file->save();
|
return d->data->file->save();
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const FileRef::FileTypeResolver *FileRef::addFileTypeResolver(const FileRef::FileTypeResolver *resolver) // static
|
const FileRef::FileTypeResolver *FileRef::addFileTypeResolver(const FileRef::FileTypeResolver *resolver) { // static
|
||||||
{
|
|
||||||
fileTypeResolvers.prepend(resolver);
|
fileTypeResolvers.prepend(resolver);
|
||||||
return resolver;
|
return resolver;
|
||||||
}
|
}
|
||||||
|
|
||||||
StringList FileRef::defaultFileExtensions()
|
StringList FileRef::defaultFileExtensions() {
|
||||||
{
|
|
||||||
StringList l;
|
StringList l;
|
||||||
|
|
||||||
l.append("ogg");
|
l.append("ogg");
|
||||||
@@ -395,97 +335,90 @@ StringList FileRef::defaultFileExtensions()
|
|||||||
l.append("wav");
|
l.append("wav");
|
||||||
l.append("ape");
|
l.append("ape");
|
||||||
l.append("mod");
|
l.append("mod");
|
||||||
l.append("module"); // alias for "mod"
|
l.append("module"); // alias for "mod"
|
||||||
l.append("nst"); // alias for "mod"
|
l.append("nst"); // alias for "mod"
|
||||||
l.append("wow"); // alias for "mod"
|
l.append("wow"); // alias for "mod"
|
||||||
l.append("s3m");
|
l.append("s3m");
|
||||||
l.append("it");
|
l.append("it");
|
||||||
l.append("xm");
|
l.append("xm");
|
||||||
l.append("dsf");
|
l.append("dsf");
|
||||||
l.append("dff");
|
l.append("dff");
|
||||||
l.append("dsdiff"); // alias for "dff"
|
l.append("dsdiff"); // alias for "dff"
|
||||||
|
|
||||||
return l;
|
return l;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool FileRef::isNull() const
|
bool FileRef::isValid() const {
|
||||||
{
|
return (d->data->file && d->data->file->isValid());
|
||||||
return (!d->file || !d->file->isValid());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
FileRef &FileRef::operator=(const FileRef &ref)
|
bool FileRef::isNull() const {
|
||||||
{
|
return (!d->data->file || !d->data->file->isValid());
|
||||||
|
}
|
||||||
|
|
||||||
|
FileRef &FileRef::operator=(const FileRef &ref) {
|
||||||
FileRef(ref).swap(*this);
|
FileRef(ref).swap(*this);
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
|
|
||||||
void FileRef::swap(FileRef &ref)
|
void FileRef::swap(FileRef &ref) {
|
||||||
{
|
|
||||||
using std::swap;
|
using std::swap;
|
||||||
|
|
||||||
swap(d, ref.d);
|
swap(d, ref.d);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool FileRef::operator==(const FileRef &ref) const
|
bool FileRef::operator==(const FileRef &ref) const {
|
||||||
{
|
return (ref.d->data == d->data);
|
||||||
return (ref.d->file == d->file);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool FileRef::operator!=(const FileRef &ref) const
|
bool FileRef::operator!=(const FileRef &ref) const {
|
||||||
{
|
return (ref.d->data != d->data);
|
||||||
return (ref.d->file != d->file);
|
|
||||||
}
|
|
||||||
|
|
||||||
File *FileRef::create(FileName fileName, bool readAudioProperties,
|
|
||||||
AudioProperties::ReadStyle audioPropertiesStyle) // static
|
|
||||||
{
|
|
||||||
return createInternal(fileName, readAudioProperties, audioPropertiesStyle);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
// private members
|
// private members
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
void FileRef::parse(FileName fileName, bool readAudioProperties,
|
void FileRef::parse(FileName fileName, bool readAudioProperties, AudioProperties::ReadStyle audioPropertiesStyle) {
|
||||||
AudioProperties::ReadStyle audioPropertiesStyle)
|
|
||||||
{
|
|
||||||
// Try user-defined resolvers.
|
// Try user-defined resolvers.
|
||||||
|
|
||||||
d->file = detectByResolvers(fileName, readAudioProperties, audioPropertiesStyle);
|
d->data->file = detectByResolvers(fileName, readAudioProperties, audioPropertiesStyle);
|
||||||
if(d->file)
|
if (d->data->file)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
// Try to resolve file types based on the file extension.
|
// Try to resolve file types based on the file extension.
|
||||||
|
|
||||||
d->stream = new FileStream(fileName);
|
d->data->stream = new FileStream(fileName);
|
||||||
d->file = detectByExtension(d->stream, readAudioProperties, audioPropertiesStyle);
|
d->data->file = detectByExtension(d->data->stream, readAudioProperties, audioPropertiesStyle);
|
||||||
if(d->file)
|
if (d->data->file)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
// At last, try to resolve file types based on the actual content.
|
// At last, try to resolve file types based on the actual content.
|
||||||
|
|
||||||
d->file = detectByContent(d->stream, readAudioProperties, audioPropertiesStyle);
|
d->data->file = detectByContent(d->data->stream, readAudioProperties, audioPropertiesStyle);
|
||||||
if(d->file)
|
if (d->data->file)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
// Stream have to be closed here if failed to resolve file types.
|
// Stream have to be closed here if failed to resolve file types.
|
||||||
|
|
||||||
delete d->stream;
|
delete d->data->stream;
|
||||||
d->stream = 0;
|
d->data->stream = nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
void FileRef::parse(IOStream *stream, bool readAudioProperties,
|
void FileRef::parse(IOStream *stream, bool readAudioProperties, AudioProperties::ReadStyle audioPropertiesStyle) {
|
||||||
AudioProperties::ReadStyle audioPropertiesStyle)
|
|
||||||
{
|
|
||||||
// User-defined resolvers won't work with a stream.
|
// User-defined resolvers won't work with a stream.
|
||||||
|
|
||||||
// Try to resolve file types based on the file extension.
|
// Try to resolve file types based on the file extension.
|
||||||
|
|
||||||
d->file = detectByExtension(stream, readAudioProperties, audioPropertiesStyle);
|
d->data->file = detectByExtension(stream, readAudioProperties, audioPropertiesStyle);
|
||||||
if(d->file)
|
if (d->data->file)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
// At last, try to resolve file types based on the actual content of the file.
|
// At last, try to resolve file types based on the actual content of the file.
|
||||||
|
|
||||||
d->file = detectByContent(stream, readAudioProperties, audioPropertiesStyle);
|
d->data->file = detectByContent(stream, readAudioProperties, audioPropertiesStyle);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
414
3rdparty/taglib/fileref.h
vendored
414
3rdparty/taglib/fileref.h
vendored
@@ -30,42 +30,37 @@
|
|||||||
#include "tstringlist.h"
|
#include "tstringlist.h"
|
||||||
|
|
||||||
#include "taglib_export.h"
|
#include "taglib_export.h"
|
||||||
|
#include "tpropertymap.h"
|
||||||
#include "audioproperties.h"
|
#include "audioproperties.h"
|
||||||
|
|
||||||
namespace Strawberry_TagLib {
|
namespace Strawberry_TagLib {
|
||||||
namespace TagLib {
|
namespace TagLib {
|
||||||
|
|
||||||
class Tag;
|
class Tag;
|
||||||
|
|
||||||
//! This class provides a simple abstraction for creating and handling files
|
//! This class provides a simple abstraction for creating and handling files
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* FileRef exists to provide a minimal, generic and value-based wrapper around
|
* FileRef exists to provide a minimal, generic and value-based wrapper around a File.
|
||||||
* a File. It is lightweight and implicitly shared, and as such suitable for
|
* It is lightweight and implicitly shared, and as such suitable for pass-by-value use.
|
||||||
* pass-by-value use. This hides some of the uglier details of TagLib::File
|
* This hides some of the uglier details of TagLib::File and the non-generic portions of the concrete file implementations.
|
||||||
* and the non-generic portions of the concrete file implementations.
|
|
||||||
*
|
*
|
||||||
* This class is useful in a "simple usage" situation where it is desirable
|
* This class is useful in a "simple usage" situation where it is desirable
|
||||||
* to be able to get and set some of the tag information that is similar
|
* to be able to get and set some of the tag information that is similar across file types.
|
||||||
* across file types.
|
|
||||||
*
|
*
|
||||||
* Also note that it is probably a good idea to plug this into your mime
|
* Also note that it is probably a good idea to plug this into your mime
|
||||||
* type system rather than using the constructor that accepts a file name using
|
* type system rather than using the constructor that accepts a file name using the FileTypeResolver.
|
||||||
* the FileTypeResolver.
|
|
||||||
*
|
*
|
||||||
* \see FileTypeResolver
|
* \see FileTypeResolver
|
||||||
* \see addFileTypeResolver()
|
* \see addFileTypeResolver()
|
||||||
*/
|
*/
|
||||||
|
|
||||||
class TAGLIB_EXPORT FileRef
|
class TAGLIB_EXPORT FileRef {
|
||||||
{
|
public:
|
||||||
public:
|
|
||||||
|
|
||||||
//! A class for pluggable file type resolution.
|
//! A class for pluggable file type resolution.
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* This class is used to add extend TagLib's very basic file name based file
|
* This class is used to add extend TagLib's very basic file name based file type resolution.
|
||||||
* type resolution.
|
|
||||||
*
|
*
|
||||||
* This can be accomplished with:
|
* This can be accomplished with:
|
||||||
*
|
*
|
||||||
@@ -85,205 +80,204 @@ namespace TagLib {
|
|||||||
*
|
*
|
||||||
* \endcode
|
* \endcode
|
||||||
*
|
*
|
||||||
* Naturally a less contrived example would be slightly more complex. This
|
* Naturally a less contrived example would be slightly more complex.
|
||||||
* can be used to plug in mime-type detection systems or to add new file types
|
* This can be used to plug in mime-type detection systems or to add new file types to TagLib.
|
||||||
* to TagLib.
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
class TAGLIB_EXPORT FileTypeResolver
|
class TAGLIB_EXPORT FileTypeResolver {
|
||||||
{
|
public:
|
||||||
public:
|
virtual ~FileTypeResolver() {}
|
||||||
virtual ~FileTypeResolver();
|
|
||||||
/*!
|
|
||||||
* This method must be overridden to provide an additional file type
|
|
||||||
* resolver. If the resolver is able to determine the file type it should
|
|
||||||
* return a valid File object; if not it should return 0.
|
|
||||||
*
|
|
||||||
* \note The created file is then owned by the FileRef and should not be
|
|
||||||
* deleted. Deletion will happen automatically when the FileRef passes
|
|
||||||
* out of scope.
|
|
||||||
*/
|
|
||||||
virtual File *createFile(FileName fileName,
|
|
||||||
bool readAudioProperties = true,
|
|
||||||
AudioProperties::ReadStyle
|
|
||||||
audioPropertiesStyle = AudioProperties::Average) const = 0;
|
|
||||||
};
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Creates a null FileRef.
|
* This method must be overridden to provide an additional file type resolver.
|
||||||
*/
|
* If the resolver is able to determine the file type it should return a valid File object; if not it should return 0.
|
||||||
FileRef();
|
|
||||||
|
|
||||||
/*!
|
|
||||||
* Create a FileRef from \a fileName. If \a readAudioProperties is true then
|
|
||||||
* the audio properties will be read using \a audioPropertiesStyle. If
|
|
||||||
* \a readAudioProperties is false then \a audioPropertiesStyle will be
|
|
||||||
* ignored.
|
|
||||||
*
|
*
|
||||||
* Also see the note in the class documentation about why you may not want to
|
* \note The created file is then owned by the FileRef and should not be deleted.
|
||||||
* use this method in your application.
|
* Deletion will happen automatically when the FileRef passes out of scope.
|
||||||
*/
|
*/
|
||||||
explicit FileRef(FileName fileName,
|
virtual File *createFile(FileName fileName, bool readAudioProperties = true, AudioProperties::ReadStyle audioPropertiesStyle = AudioProperties::Average) const = 0;
|
||||||
bool readAudioProperties = true,
|
|
||||||
AudioProperties::ReadStyle
|
|
||||||
audioPropertiesStyle = AudioProperties::Average);
|
|
||||||
|
|
||||||
/*!
|
|
||||||
* Construct a FileRef from an opened \a IOStream. If \a readAudioProperties
|
|
||||||
* is true then the audio properties will be read using \a audioPropertiesStyle.
|
|
||||||
* If \a readAudioProperties is false then \a audioPropertiesStyle will be
|
|
||||||
* ignored.
|
|
||||||
*
|
|
||||||
* Also see the note in the class documentation about why you may not want to
|
|
||||||
* use this method in your application.
|
|
||||||
*
|
|
||||||
* \note TagLib will *not* take ownership of the stream, the caller is
|
|
||||||
* responsible for deleting it after the File object.
|
|
||||||
*/
|
|
||||||
explicit FileRef(IOStream* stream,
|
|
||||||
bool readAudioProperties = true,
|
|
||||||
AudioProperties::ReadStyle
|
|
||||||
audioPropertiesStyle = AudioProperties::Average);
|
|
||||||
|
|
||||||
/*!
|
|
||||||
* Construct a FileRef using \a file. The FileRef now takes ownership of the
|
|
||||||
* pointer and will delete the File when it passes out of scope.
|
|
||||||
*/
|
|
||||||
explicit FileRef(File *file);
|
|
||||||
|
|
||||||
/*!
|
|
||||||
* Make a copy of \a ref.
|
|
||||||
*/
|
|
||||||
FileRef(const FileRef &ref);
|
|
||||||
|
|
||||||
/*!
|
|
||||||
* Destroys this FileRef instance.
|
|
||||||
*/
|
|
||||||
virtual ~FileRef();
|
|
||||||
|
|
||||||
/*!
|
|
||||||
* Returns a pointer to represented file's tag.
|
|
||||||
*
|
|
||||||
* \warning This pointer will become invalid when this FileRef and all
|
|
||||||
* copies pass out of scope.
|
|
||||||
*
|
|
||||||
* \warning Do not cast it to any subclasses of \class Tag.
|
|
||||||
* Use tag returning methods of appropriate subclasses of \class File instead.
|
|
||||||
*
|
|
||||||
* \see File::tag()
|
|
||||||
*/
|
|
||||||
Tag *tag() const;
|
|
||||||
|
|
||||||
/*!
|
|
||||||
* Returns the audio properties for this FileRef. If no audio properties
|
|
||||||
* were read then this will returns a null pointer.
|
|
||||||
*/
|
|
||||||
AudioProperties *audioProperties() const;
|
|
||||||
|
|
||||||
/*!
|
|
||||||
* Returns a pointer to the file represented by this handler class.
|
|
||||||
*
|
|
||||||
* As a general rule this call should be avoided since if you need to work
|
|
||||||
* with file objects directly, you are probably better served instantiating
|
|
||||||
* the File subclasses (i.e. MPEG::File) manually and working with their APIs.
|
|
||||||
*
|
|
||||||
* This <i>handle</i> exists to provide a minimal, generic and value-based
|
|
||||||
* wrapper around a File. Accessing the file directly generally indicates
|
|
||||||
* a moving away from this simplicity (and into things beyond the scope of
|
|
||||||
* FileRef).
|
|
||||||
*
|
|
||||||
* \warning This pointer will become invalid when this FileRef and all
|
|
||||||
* copies pass out of scope.
|
|
||||||
*/
|
|
||||||
File *file() const;
|
|
||||||
|
|
||||||
/*!
|
|
||||||
* Saves the file. Returns true on success.
|
|
||||||
*/
|
|
||||||
bool save();
|
|
||||||
|
|
||||||
/*!
|
|
||||||
* Adds a FileTypeResolver to the list of those used by TagLib. Each
|
|
||||||
* additional FileTypeResolver is added to the front of a list of resolvers
|
|
||||||
* that are tried. If the FileTypeResolver returns zero the next resolver
|
|
||||||
* is tried.
|
|
||||||
*
|
|
||||||
* Returns a pointer to the added resolver (the same one that's passed in --
|
|
||||||
* this is mostly so that static initializers have something to use for
|
|
||||||
* assignment).
|
|
||||||
*
|
|
||||||
* \see FileTypeResolver
|
|
||||||
*/
|
|
||||||
static const FileTypeResolver *addFileTypeResolver(const FileTypeResolver *resolver);
|
|
||||||
|
|
||||||
/*!
|
|
||||||
* As is mentioned elsewhere in this class's documentation, the default file
|
|
||||||
* type resolution code provided by TagLib only works by comparing file
|
|
||||||
* extensions.
|
|
||||||
*
|
|
||||||
* This method returns the list of file extensions that are used by default.
|
|
||||||
*
|
|
||||||
* The extensions are all returned in lowercase, though the comparison used
|
|
||||||
* by TagLib for resolution is case-insensitive.
|
|
||||||
*
|
|
||||||
* \note This does not account for any additional file type resolvers that
|
|
||||||
* are plugged in. Also note that this is not intended to replace a proper
|
|
||||||
* mime-type resolution system, but is just here for reference.
|
|
||||||
*
|
|
||||||
* \see FileTypeResolver
|
|
||||||
*/
|
|
||||||
static StringList defaultFileExtensions();
|
|
||||||
|
|
||||||
/*!
|
|
||||||
* Returns true if the file (and as such other pointers) are null.
|
|
||||||
*/
|
|
||||||
bool isNull() const;
|
|
||||||
|
|
||||||
/*!
|
|
||||||
* Assign the file pointed to by \a ref to this FileRef.
|
|
||||||
*/
|
|
||||||
FileRef &operator=(const FileRef &ref);
|
|
||||||
|
|
||||||
/*!
|
|
||||||
* Exchanges the content of the FileRef by the content of \a ref.
|
|
||||||
*/
|
|
||||||
void swap(FileRef &ref);
|
|
||||||
|
|
||||||
/*!
|
|
||||||
* Returns true if this FileRef and \a ref point to the same File object.
|
|
||||||
*/
|
|
||||||
bool operator==(const FileRef &ref) const;
|
|
||||||
|
|
||||||
/*!
|
|
||||||
* Returns true if this FileRef and \a ref do not point to the same File
|
|
||||||
* object.
|
|
||||||
*/
|
|
||||||
bool operator!=(const FileRef &ref) const;
|
|
||||||
|
|
||||||
/*!
|
|
||||||
* A simple implementation of file type guessing. If \a readAudioProperties
|
|
||||||
* is true then the audio properties will be read using
|
|
||||||
* \a audioPropertiesStyle. If \a readAudioProperties is false then
|
|
||||||
* \a audioPropertiesStyle will be ignored.
|
|
||||||
*
|
|
||||||
* \note You generally shouldn't use this method, but instead the constructor
|
|
||||||
* directly.
|
|
||||||
*
|
|
||||||
* \deprecated
|
|
||||||
*/
|
|
||||||
static File *create(FileName fileName,
|
|
||||||
bool readAudioProperties = true,
|
|
||||||
AudioProperties::ReadStyle audioPropertiesStyle = AudioProperties::Average);
|
|
||||||
|
|
||||||
private:
|
|
||||||
void parse(FileName fileName, bool readAudioProperties, AudioProperties::ReadStyle audioPropertiesStyle);
|
|
||||||
void parse(IOStream *stream, bool readAudioProperties, AudioProperties::ReadStyle audioPropertiesStyle);
|
|
||||||
|
|
||||||
class FileRefPrivate;
|
|
||||||
FileRefPrivate *d;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
}
|
/*!
|
||||||
} // namespace Strawberry_TagLib::TagLib
|
* Creates a null FileRef.
|
||||||
|
*/
|
||||||
|
explicit FileRef();
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* Create a FileRef from \a fileName.
|
||||||
|
* If \a readAudioProperties is true then the audio properties will be read using \a audioPropertiesStyle.
|
||||||
|
* If \a readAudioProperties is false then \a audioPropertiesStyle will be ignored.
|
||||||
|
*
|
||||||
|
* Also see the note in the class documentation about why you may not want to
|
||||||
|
* use this method in your application.
|
||||||
|
*/
|
||||||
|
explicit FileRef(FileName fileName, bool readAudioProperties = true, AudioProperties::ReadStyle audioPropertiesStyle = AudioProperties::Average);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* Construct a FileRef from an opened \a IOStream.
|
||||||
|
* If \a readAudioProperties is true then the audio properties will be read using \a audioPropertiesStyle.
|
||||||
|
* If \a readAudioProperties is false then \a audioPropertiesStyle will be ignored.
|
||||||
|
*
|
||||||
|
* Also see the note in the class documentation about why you may not want to use this method in your application.
|
||||||
|
*
|
||||||
|
* \note TagLib will *not* take ownership of the stream, the caller is responsible for deleting it after the File object.
|
||||||
|
*/
|
||||||
|
explicit FileRef(IOStream *stream, bool readAudioProperties = true, AudioProperties::ReadStyle audioPropertiesStyle = AudioProperties::Average);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* Construct a FileRef using \a file.
|
||||||
|
* The FileRef now takes ownership of the pointer and will delete the File when it passes out of scope.
|
||||||
|
*/
|
||||||
|
explicit FileRef(File *file);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* Make a copy of \a ref.
|
||||||
|
*/
|
||||||
|
FileRef(const FileRef &ref);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* Destroys this FileRef instance.
|
||||||
|
*/
|
||||||
|
virtual ~FileRef();
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* Returns a pointer to represented file's tag.
|
||||||
|
*
|
||||||
|
* \warning This pointer will become invalid when this FileRef and all
|
||||||
|
* copies pass out of scope.
|
||||||
|
*
|
||||||
|
* \warning Do not cast it to any subclasses of \class Tag.
|
||||||
|
* Use tag returning methods of appropriate subclasses of \class File instead.
|
||||||
|
*
|
||||||
|
* \see File::tag()
|
||||||
|
*/
|
||||||
|
Tag *tag() const;
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* Exports the tags of the file as dictionary mapping (human readable) tag names (uppercase Strings) to StringLists of tag values.
|
||||||
|
* Calls the according specialization in the File subclasses.
|
||||||
|
* For each metadata object of the file that could not be parsed into the PropertyMap format,
|
||||||
|
* the returend map's unsupportedData() list will contain one entry identifying that object (e.g. the frame type for ID3v2 tags).
|
||||||
|
* Use removeUnsupportedProperties() to remove (a subset of) them.
|
||||||
|
* For files that contain more than one tag (e.g. an MP3 with both an ID3v1 and an ID3v2 tag) only the most "modern" one will be exported (ID3v2 in this case).
|
||||||
|
*/
|
||||||
|
PropertyMap properties() const;
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* Removes unsupported properties, or a subset of them, from the file's metadata.
|
||||||
|
* The parameter \a properties must contain only entries from properties().unsupportedData().
|
||||||
|
*/
|
||||||
|
void removeUnsupportedProperties(const StringList &properties);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* Sets the tags of this File to those specified in \a properties.
|
||||||
|
* Calls the according specialization method in the subclasses of File to do the translation into the format-specific details.
|
||||||
|
* If some value(s) could not be written imported to the specific metadata format,
|
||||||
|
* the returned PropertyMap will contain those value(s). Otherwise it will be empty, indicating that no problems occured.
|
||||||
|
* With file types that support several tag formats (for instance, MP3 files can have ID3v1, ID3v2, and APEv2 tags),
|
||||||
|
* this function will create the most appropriate one (ID3v2 for MP3 files). Older formats will be updated as well,
|
||||||
|
* if they exist, but won't be taken into account for the return value of this function.
|
||||||
|
* See the documentation of the subclass implementations for detailed descriptions.
|
||||||
|
*/
|
||||||
|
PropertyMap setProperties(const PropertyMap &properties);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* Returns the audio properties for this FileRef.
|
||||||
|
* If no audio properties were read then this will returns a null pointer.
|
||||||
|
*/
|
||||||
|
AudioProperties *audioProperties() const;
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* Returns a pointer to the file represented by this handler class.
|
||||||
|
*
|
||||||
|
* As a general rule this call should be avoided since if you need to work
|
||||||
|
* with file objects directly, you are probably better served instantiating
|
||||||
|
* the File subclasses (i.e. MPEG::File) manually and working with their APIs.
|
||||||
|
*
|
||||||
|
* This <i>handle</i> exists to provide a minimal, generic and value-based
|
||||||
|
* wrapper around a File. Accessing the file directly generally indicates
|
||||||
|
* a moving away from this simplicity (and into things beyond the scope of
|
||||||
|
* FileRef).
|
||||||
|
*
|
||||||
|
* \warning This pointer will become invalid when this FileRef and all
|
||||||
|
* copies pass out of scope.
|
||||||
|
*/
|
||||||
|
File *file() const;
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* Saves the file. Returns true on success.
|
||||||
|
*/
|
||||||
|
bool save();
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* Adds a FileTypeResolver to the list of those used by TagLib.
|
||||||
|
* Each additional FileTypeResolver is added to the front of a list of resolvers that are tried.
|
||||||
|
* If the FileTypeResolver returns zero the next resolver is tried.
|
||||||
|
*
|
||||||
|
* Returns a pointer to the added resolver (the same one that's passed in --
|
||||||
|
* this is mostly so that static initializers have something to use for assignment).
|
||||||
|
*
|
||||||
|
* \see FileTypeResolver
|
||||||
|
*/
|
||||||
|
static const FileTypeResolver *addFileTypeResolver(const FileTypeResolver *resolver);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* As is mentioned elsewhere in this class's documentation, the default file
|
||||||
|
* type resolution code provided by TagLib only works by comparing file extensions.
|
||||||
|
*
|
||||||
|
* This method returns the list of file extensions that are used by default.
|
||||||
|
*
|
||||||
|
* The extensions are all returned in lowercase, though the comparison used
|
||||||
|
* by TagLib for resolution is case-insensitive.
|
||||||
|
*
|
||||||
|
* \note This does not account for any additional file type resolvers that
|
||||||
|
* are plugged in. Also note that this is not intended to replace a proper
|
||||||
|
* mime-type resolution system, but is just here for reference.
|
||||||
|
*
|
||||||
|
* \see FileTypeResolver
|
||||||
|
*/
|
||||||
|
static StringList defaultFileExtensions();
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* Returns true if the file is open and readable.
|
||||||
|
*
|
||||||
|
* \note Just a negative of isNull().
|
||||||
|
*/
|
||||||
|
bool isValid() const;
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* Returns true if the file (and as such other pointers) are null.
|
||||||
|
*/
|
||||||
|
bool isNull() const;
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* Assign the file pointed to by \a ref to this FileRef.
|
||||||
|
*/
|
||||||
|
FileRef &operator=(const FileRef &ref);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* Exchanges the content of the FileRef by the content of \a ref.
|
||||||
|
*/
|
||||||
|
void swap(FileRef &ref);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* Returns true if this FileRef and \a ref point to the same File object.
|
||||||
|
*/
|
||||||
|
bool operator==(const FileRef &ref) const;
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* Returns true if this FileRef and \a ref do not point to the same File object.
|
||||||
|
*/
|
||||||
|
bool operator!=(const FileRef &ref) const;
|
||||||
|
|
||||||
|
private:
|
||||||
|
void parse(FileName fileName, bool readAudioProperties, AudioProperties::ReadStyle audioPropertiesStyle);
|
||||||
|
void parse(IOStream *stream, bool readAudioProperties, AudioProperties::ReadStyle audioPropertiesStyle);
|
||||||
|
|
||||||
|
class FileRefPrivate;
|
||||||
|
FileRefPrivate *d;
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace TagLib
|
||||||
|
} // namespace Strawberry_TagLib
|
||||||
|
|
||||||
#endif // TAGLIB_FILEREF_H
|
#endif // TAGLIB_FILEREF_H
|
||||||
|
|||||||
370
3rdparty/taglib/flac/flacfile.cpp
vendored
370
3rdparty/taglib/flac/flacfile.cpp
vendored
@@ -23,18 +23,20 @@
|
|||||||
* http://www.mozilla.org/MPL/ *
|
* http://www.mozilla.org/MPL/ *
|
||||||
***************************************************************************/
|
***************************************************************************/
|
||||||
|
|
||||||
#include <tbytevector.h>
|
#include <memory>
|
||||||
#include <tstring.h>
|
|
||||||
#include <tlist.h>
|
|
||||||
#include <tdebug.h>
|
|
||||||
#include <tagunion.h>
|
|
||||||
#include <tpropertymap.h>
|
|
||||||
#include <tagutils.h>
|
|
||||||
|
|
||||||
#include <id3v2header.h>
|
#include "tbytevector.h"
|
||||||
#include <id3v2tag.h>
|
#include "tstring.h"
|
||||||
#include <id3v1tag.h>
|
#include "tlist.h"
|
||||||
#include <xiphcomment.h>
|
#include "tdebug.h"
|
||||||
|
#include "tagunion.h"
|
||||||
|
#include "tpropertymap.h"
|
||||||
|
#include "tagutils.h"
|
||||||
|
|
||||||
|
#include "id3v2header.h"
|
||||||
|
#include "id3v2tag.h"
|
||||||
|
#include "id3v1tag.h"
|
||||||
|
#include "xiphcomment.h"
|
||||||
|
|
||||||
#include "flacpicture.h"
|
#include "flacpicture.h"
|
||||||
#include "flacfile.h"
|
#include "flacfile.h"
|
||||||
@@ -43,55 +45,63 @@
|
|||||||
|
|
||||||
using namespace Strawberry_TagLib::TagLib;
|
using namespace Strawberry_TagLib::TagLib;
|
||||||
|
|
||||||
namespace
|
namespace {
|
||||||
{
|
typedef List<std::shared_ptr<FLAC::MetadataBlock>> BlockList;
|
||||||
typedef List<FLAC::MetadataBlock *> BlockList;
|
typedef BlockList::Iterator BlockIterator;
|
||||||
typedef BlockList::Iterator BlockIterator;
|
typedef BlockList::Iterator BlockConstIterator;
|
||||||
typedef BlockList::Iterator BlockConstIterator;
|
|
||||||
|
|
||||||
enum { FlacXiphIndex = 0, FlacID3v2Index = 1, FlacID3v1Index = 2 };
|
enum { FlacXiphIndex = 0,
|
||||||
|
FlacID3v2Index = 1,
|
||||||
|
FlacID3v1Index = 2 };
|
||||||
|
|
||||||
const long MinPaddingLength = 4096;
|
const long long MinPaddingLength = 4096;
|
||||||
const long MaxPaddingLegnth = 1024 * 1024;
|
const long long MaxPaddingLegnth = 1024 * 1024;
|
||||||
|
|
||||||
const char LastBlockFlag = '\x80';
|
const char LastBlockFlag = '\x80';
|
||||||
|
} // namespace
|
||||||
|
|
||||||
|
namespace Strawberry_TagLib {
|
||||||
|
namespace TagLib {
|
||||||
|
namespace FLAC {
|
||||||
|
// Enables BlockList::find() to take raw pointers.
|
||||||
|
|
||||||
|
bool operator==(std::shared_ptr<MetadataBlock> lhs, MetadataBlock *rhs);
|
||||||
|
bool operator==(std::shared_ptr<MetadataBlock> lhs, MetadataBlock *rhs) {
|
||||||
|
return lhs.get() == rhs;
|
||||||
}
|
}
|
||||||
|
} // namespace FLAC
|
||||||
|
} // namespace TagLib
|
||||||
|
} // namespace Strawberry_TagLib
|
||||||
|
|
||||||
class FLAC::File::FilePrivate
|
class FLAC::File::FilePrivate {
|
||||||
{
|
public:
|
||||||
public:
|
explicit FilePrivate(const ID3v2::FrameFactory *frameFactory) : ID3v2FrameFactory(ID3v2::FrameFactory::instance()),
|
||||||
explicit FilePrivate(const ID3v2::FrameFactory *frameFactory = ID3v2::FrameFactory::instance()) :
|
ID3v2Location(-1),
|
||||||
ID3v2FrameFactory(frameFactory),
|
ID3v2OriginalSize(0),
|
||||||
ID3v2Location(-1),
|
ID3v1Location(-1),
|
||||||
ID3v2OriginalSize(0),
|
flacStart(0),
|
||||||
ID3v1Location(-1),
|
streamStart(0),
|
||||||
properties(0),
|
scanned(false) {
|
||||||
flacStart(0),
|
|
||||||
streamStart(0),
|
if (frameFactory)
|
||||||
scanned(false)
|
ID3v2FrameFactory = frameFactory;
|
||||||
{
|
|
||||||
blocks.setAutoDelete(true);
|
|
||||||
}
|
|
||||||
|
|
||||||
~FilePrivate()
|
|
||||||
{
|
|
||||||
delete properties;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const ID3v2::FrameFactory *ID3v2FrameFactory;
|
const ID3v2::FrameFactory *ID3v2FrameFactory;
|
||||||
long ID3v2Location;
|
long long ID3v2Location;
|
||||||
long ID3v2OriginalSize;
|
long long ID3v2OriginalSize;
|
||||||
|
|
||||||
long ID3v1Location;
|
long long ID3v1Location;
|
||||||
|
|
||||||
TagUnion tag;
|
TripleTagUnion tag;
|
||||||
|
|
||||||
Properties *properties;
|
std::unique_ptr<AudioProperties> properties;
|
||||||
ByteVector xiphCommentData;
|
ByteVector xiphCommentData;
|
||||||
BlockList blocks;
|
BlockList blocks;
|
||||||
|
|
||||||
long flacStart;
|
long long flacStart;
|
||||||
long streamStart;
|
long long streamStart;
|
||||||
bool scanned;
|
bool scanned;
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -99,111 +109,85 @@ public:
|
|||||||
// static members
|
// static members
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
bool FLAC::File::isSupported(IOStream *stream)
|
bool FLAC::File::isSupported(IOStream *stream) {
|
||||||
{
|
|
||||||
// A FLAC file has an ID "fLaC" somewhere. An ID3v2 tag may precede.
|
// A FLAC file has an ID "fLaC" somewhere. An ID3v2 tag may precede.
|
||||||
|
|
||||||
const ByteVector buffer = Utils::readHeader(stream, bufferSize(), true);
|
const ByteVector buffer = Utils::readHeader(stream, bufferSize(), true);
|
||||||
return (buffer.find("fLaC") >= 0);
|
return (buffer.find("fLaC") != ByteVector::npos());
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
// public members
|
// public members
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
FLAC::File::File(FileName file, bool readProperties, Properties::ReadStyle) :
|
FLAC::File::File(FileName file, bool readProperties, AudioProperties::ReadStyle, ID3v2::FrameFactory *frameFactory) : Strawberry_TagLib::TagLib::File(file), d(new FilePrivate(frameFactory)) {
|
||||||
Strawberry_TagLib::TagLib::File(file),
|
|
||||||
d(new FilePrivate())
|
if (isOpen())
|
||||||
{
|
|
||||||
if(isOpen())
|
|
||||||
read(readProperties);
|
read(readProperties);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
FLAC::File::File(FileName file, ID3v2::FrameFactory *frameFactory,
|
FLAC::File::File(IOStream *stream, bool readProperties, AudioProperties::ReadStyle, ID3v2::FrameFactory *frameFactory) : Strawberry_TagLib::TagLib::File(stream), d(new FilePrivate(frameFactory)) {
|
||||||
bool readProperties, Properties::ReadStyle) :
|
|
||||||
Strawberry_TagLib::TagLib::File(file),
|
if (isOpen())
|
||||||
d(new FilePrivate(frameFactory))
|
|
||||||
{
|
|
||||||
if(isOpen())
|
|
||||||
read(readProperties);
|
read(readProperties);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
FLAC::File::File(IOStream *stream, ID3v2::FrameFactory *frameFactory,
|
FLAC::File::~File() {
|
||||||
bool readProperties, Properties::ReadStyle) :
|
|
||||||
Strawberry_TagLib::TagLib::File(stream),
|
|
||||||
d(new FilePrivate(frameFactory))
|
|
||||||
{
|
|
||||||
if(isOpen())
|
|
||||||
read(readProperties);
|
|
||||||
}
|
|
||||||
|
|
||||||
FLAC::File::~File()
|
|
||||||
{
|
|
||||||
delete d;
|
delete d;
|
||||||
}
|
}
|
||||||
|
|
||||||
Strawberry_TagLib::TagLib::Tag *FLAC::File::tag() const
|
Strawberry_TagLib::TagLib::Tag *FLAC::File::tag() const {
|
||||||
{
|
|
||||||
return &d->tag;
|
return &d->tag;
|
||||||
}
|
}
|
||||||
|
|
||||||
PropertyMap FLAC::File::properties() const
|
PropertyMap FLAC::File::setProperties(const PropertyMap &properties) {
|
||||||
{
|
|
||||||
return d->tag.properties();
|
|
||||||
}
|
|
||||||
|
|
||||||
void FLAC::File::removeUnsupportedProperties(const StringList &unsupported)
|
|
||||||
{
|
|
||||||
d->tag.removeUnsupportedProperties(unsupported);
|
|
||||||
}
|
|
||||||
|
|
||||||
PropertyMap FLAC::File::setProperties(const PropertyMap &properties)
|
|
||||||
{
|
|
||||||
return xiphComment(true)->setProperties(properties);
|
return xiphComment(true)->setProperties(properties);
|
||||||
}
|
}
|
||||||
|
|
||||||
FLAC::Properties *FLAC::File::audioProperties() const
|
FLAC::AudioProperties *FLAC::File::audioProperties() const {
|
||||||
{
|
return d->properties.get();
|
||||||
return d->properties;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool FLAC::File::save()
|
bool FLAC::File::save() {
|
||||||
{
|
|
||||||
if(readOnly()) {
|
if (readOnly()) {
|
||||||
debug("FLAC::File::save() - Cannot save to a read only file.");
|
debug("FLAC::File::save() - Cannot save to a read only file.");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if(!isValid()) {
|
if (!isValid()) {
|
||||||
debug("FLAC::File::save() -- Trying to save invalid file.");
|
debug("FLAC::File::save() -- Trying to save invalid file.");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Create new vorbis comments
|
// Create new vorbis comments
|
||||||
if(!hasXiphComment())
|
if (!hasXiphComment())
|
||||||
Tag::duplicate(&d->tag, xiphComment(true), false);
|
Tag::duplicate(&d->tag, xiphComment(true), false);
|
||||||
|
|
||||||
d->xiphCommentData = xiphComment()->render(false);
|
d->xiphCommentData = xiphComment()->render(false);
|
||||||
|
|
||||||
// Replace metadata blocks
|
// Replace metadata blocks
|
||||||
|
|
||||||
for(BlockIterator it = d->blocks.begin(); it != d->blocks.end(); ++it) {
|
for (BlockIterator it = d->blocks.begin(); it != d->blocks.end(); ++it) {
|
||||||
if((*it)->code() == MetadataBlock::VorbisComment) {
|
if ((*it)->code() == MetadataBlock::VorbisComment) {
|
||||||
// Set the new Vorbis Comment block
|
// Set the new Vorbis Comment block
|
||||||
delete *it;
|
|
||||||
d->blocks.erase(it);
|
d->blocks.erase(it);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
d->blocks.append(new UnknownMetadataBlock(MetadataBlock::VorbisComment, d->xiphCommentData));
|
d->blocks.append(std::shared_ptr<MetadataBlock>(new UnknownMetadataBlock(MetadataBlock::VorbisComment, d->xiphCommentData)));
|
||||||
|
|
||||||
// Render data for the metadata blocks
|
// Render data for the metadata blocks
|
||||||
|
|
||||||
ByteVector data;
|
ByteVector data;
|
||||||
for(BlockConstIterator it = d->blocks.begin(); it != d->blocks.end(); ++it) {
|
for (BlockConstIterator it = d->blocks.begin(); it != d->blocks.end(); ++it) {
|
||||||
ByteVector blockData = (*it)->render();
|
ByteVector blockData = (*it)->render();
|
||||||
ByteVector blockHeader = ByteVector::fromUInt(blockData.size());
|
ByteVector blockHeader = ByteVector::fromUInt32BE(blockData.size());
|
||||||
blockHeader[0] = (*it)->code();
|
blockHeader[0] = (*it)->code();
|
||||||
data.append(blockHeader);
|
data.append(blockHeader);
|
||||||
data.append(blockData);
|
data.append(blockData);
|
||||||
@@ -211,27 +195,27 @@ bool FLAC::File::save()
|
|||||||
|
|
||||||
// Compute the amount of padding, and append that to data.
|
// Compute the amount of padding, and append that to data.
|
||||||
|
|
||||||
long originalLength = d->streamStart - d->flacStart;
|
long long originalLength = d->streamStart - d->flacStart;
|
||||||
long paddingLength = originalLength - data.size() - 4;
|
long long paddingLength = originalLength - data.size() - 4;
|
||||||
|
|
||||||
if(paddingLength <= 0) {
|
if (paddingLength <= 0) {
|
||||||
paddingLength = MinPaddingLength;
|
paddingLength = MinPaddingLength;
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
// Padding won't increase beyond 1% of the file size or 1MB.
|
// Padding won't increase beyond 1% of the file size or 1MB.
|
||||||
|
|
||||||
long threshold = length() / 100;
|
long long threshold = length() / 100;
|
||||||
threshold = std::max(threshold, MinPaddingLength);
|
threshold = std::max(threshold, MinPaddingLength);
|
||||||
threshold = std::min(threshold, MaxPaddingLegnth);
|
threshold = std::min(threshold, MaxPaddingLegnth);
|
||||||
|
|
||||||
if(paddingLength > threshold)
|
if (paddingLength > threshold)
|
||||||
paddingLength = MinPaddingLength;
|
paddingLength = MinPaddingLength;
|
||||||
}
|
}
|
||||||
|
|
||||||
ByteVector paddingHeader = ByteVector::fromUInt(paddingLength);
|
ByteVector paddingHeader = ByteVector::fromUInt32BE(paddingLength);
|
||||||
paddingHeader[0] = static_cast<char>(MetadataBlock::Padding | LastBlockFlag);
|
paddingHeader[0] = static_cast<char>(MetadataBlock::Padding | LastBlockFlag);
|
||||||
data.append(paddingHeader);
|
data.append(paddingHeader);
|
||||||
data.resize(static_cast<unsigned int>(data.size() + paddingLength));
|
data.resize(static_cast<size_t>(data.size() + paddingLength));
|
||||||
|
|
||||||
// Write the data to the file
|
// Write the data to the file
|
||||||
|
|
||||||
@@ -239,25 +223,25 @@ bool FLAC::File::save()
|
|||||||
|
|
||||||
d->streamStart += (static_cast<long>(data.size()) - originalLength);
|
d->streamStart += (static_cast<long>(data.size()) - originalLength);
|
||||||
|
|
||||||
if(d->ID3v1Location >= 0)
|
if (d->ID3v1Location >= 0)
|
||||||
d->ID3v1Location += (static_cast<long>(data.size()) - originalLength);
|
d->ID3v1Location += (static_cast<long>(data.size()) - originalLength);
|
||||||
|
|
||||||
// Update ID3 tags
|
// Update ID3 tags
|
||||||
|
|
||||||
if(ID3v2Tag() && !ID3v2Tag()->isEmpty()) {
|
if (ID3v2Tag() && !ID3v2Tag()->isEmpty()) {
|
||||||
|
|
||||||
// ID3v2 tag is not empty. Update the old one or create a new one.
|
// ID3v2 tag is not empty. Update the old one or create a new one.
|
||||||
|
|
||||||
if(d->ID3v2Location < 0)
|
if (d->ID3v2Location < 0)
|
||||||
d->ID3v2Location = 0;
|
d->ID3v2Location = 0;
|
||||||
|
|
||||||
data = ID3v2Tag()->render();
|
data = ID3v2Tag()->render();
|
||||||
insert(data, d->ID3v2Location, d->ID3v2OriginalSize);
|
insert(data, d->ID3v2Location, d->ID3v2OriginalSize);
|
||||||
|
|
||||||
d->flacStart += (static_cast<long>(data.size()) - d->ID3v2OriginalSize);
|
d->flacStart += (static_cast<long>(data.size()) - d->ID3v2OriginalSize);
|
||||||
d->streamStart += (static_cast<long>(data.size()) - d->ID3v2OriginalSize);
|
d->streamStart += (static_cast<long>(data.size()) - d->ID3v2OriginalSize);
|
||||||
|
|
||||||
if(d->ID3v1Location >= 0)
|
if (d->ID3v1Location >= 0)
|
||||||
d->ID3v1Location += (static_cast<long>(data.size()) - d->ID3v2OriginalSize);
|
d->ID3v1Location += (static_cast<long>(data.size()) - d->ID3v2OriginalSize);
|
||||||
|
|
||||||
d->ID3v2OriginalSize = data.size();
|
d->ID3v2OriginalSize = data.size();
|
||||||
@@ -266,13 +250,13 @@ bool FLAC::File::save()
|
|||||||
|
|
||||||
// ID3v2 tag is empty. Remove the old one.
|
// ID3v2 tag is empty. Remove the old one.
|
||||||
|
|
||||||
if(d->ID3v2Location >= 0) {
|
if (d->ID3v2Location >= 0) {
|
||||||
removeBlock(d->ID3v2Location, d->ID3v2OriginalSize);
|
removeBlock(d->ID3v2Location, d->ID3v2OriginalSize);
|
||||||
|
|
||||||
d->flacStart -= d->ID3v2OriginalSize;
|
d->flacStart -= d->ID3v2OriginalSize;
|
||||||
d->streamStart -= d->ID3v2OriginalSize;
|
d->streamStart -= d->ID3v2OriginalSize;
|
||||||
|
|
||||||
if(d->ID3v1Location >= 0)
|
if (d->ID3v1Location >= 0)
|
||||||
d->ID3v1Location -= d->ID3v2OriginalSize;
|
d->ID3v1Location -= d->ID3v2OriginalSize;
|
||||||
|
|
||||||
d->ID3v2Location = -1;
|
d->ID3v2Location = -1;
|
||||||
@@ -280,11 +264,11 @@ bool FLAC::File::save()
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if(ID3v1Tag() && !ID3v1Tag()->isEmpty()) {
|
if (ID3v1Tag() && !ID3v1Tag()->isEmpty()) {
|
||||||
|
|
||||||
// ID3v1 tag is not empty. Update the old one or create a new one.
|
// ID3v1 tag is not empty. Update the old one or create a new one.
|
||||||
|
|
||||||
if(d->ID3v1Location >= 0) {
|
if (d->ID3v1Location >= 0) {
|
||||||
seek(d->ID3v1Location);
|
seek(d->ID3v1Location);
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
@@ -298,113 +282,90 @@ bool FLAC::File::save()
|
|||||||
|
|
||||||
// ID3v1 tag is empty. Remove the old one.
|
// ID3v1 tag is empty. Remove the old one.
|
||||||
|
|
||||||
if(d->ID3v1Location >= 0) {
|
if (d->ID3v1Location >= 0) {
|
||||||
truncate(d->ID3v1Location);
|
truncate(d->ID3v1Location);
|
||||||
d->ID3v1Location = -1;
|
d->ID3v1Location = -1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ID3v2::Tag *FLAC::File::ID3v2Tag(bool create)
|
ID3v2::Tag *FLAC::File::ID3v2Tag(bool create) {
|
||||||
{
|
|
||||||
return d->tag.access<ID3v2::Tag>(FlacID3v2Index, create);
|
return d->tag.access<ID3v2::Tag>(FlacID3v2Index, create);
|
||||||
}
|
}
|
||||||
|
|
||||||
ID3v1::Tag *FLAC::File::ID3v1Tag(bool create)
|
ID3v1::Tag *FLAC::File::ID3v1Tag(bool create) {
|
||||||
{
|
|
||||||
return d->tag.access<ID3v1::Tag>(FlacID3v1Index, create);
|
return d->tag.access<ID3v1::Tag>(FlacID3v1Index, create);
|
||||||
}
|
}
|
||||||
|
|
||||||
Ogg::XiphComment *FLAC::File::xiphComment(bool create)
|
Ogg::XiphComment *FLAC::File::xiphComment(bool create) {
|
||||||
{
|
|
||||||
return d->tag.access<Ogg::XiphComment>(FlacXiphIndex, create);
|
return d->tag.access<Ogg::XiphComment>(FlacXiphIndex, create);
|
||||||
}
|
}
|
||||||
|
|
||||||
void FLAC::File::setID3v2FrameFactory(const ID3v2::FrameFactory *factory)
|
List<FLAC::Picture *> FLAC::File::pictureList() {
|
||||||
{
|
|
||||||
d->ID3v2FrameFactory = factory;
|
|
||||||
}
|
|
||||||
|
|
||||||
ByteVector FLAC::File::streamInfoData()
|
|
||||||
{
|
|
||||||
debug("FLAC::File::streamInfoData() -- This function is obsolete. Returning an empty ByteVector.");
|
|
||||||
return ByteVector();
|
|
||||||
}
|
|
||||||
|
|
||||||
long FLAC::File::streamLength()
|
|
||||||
{
|
|
||||||
debug("FLAC::File::streamLength() -- This function is obsolete. Returning zero.");
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
List<FLAC::Picture *> FLAC::File::pictureList()
|
|
||||||
{
|
|
||||||
List<Picture *> pictures;
|
List<Picture *> pictures;
|
||||||
for(BlockConstIterator it = d->blocks.begin(); it != d->blocks.end(); ++it) {
|
for (BlockConstIterator it = d->blocks.begin(); it != d->blocks.end(); ++it) {
|
||||||
Picture *picture = dynamic_cast<Picture *>(*it);
|
Picture *picture = dynamic_cast<Picture *>(it->get());
|
||||||
if(picture) {
|
if (picture) {
|
||||||
pictures.append(picture);
|
pictures.append(picture);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return pictures;
|
return pictures;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void FLAC::File::addPicture(Picture *picture)
|
void FLAC::File::addPicture(Picture *picture) {
|
||||||
{
|
d->blocks.append(std::shared_ptr<Picture>(picture));
|
||||||
d->blocks.append(picture);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void FLAC::File::removePicture(Picture *picture, bool del)
|
void FLAC::File::removePicture(Picture *picture) {
|
||||||
{
|
|
||||||
BlockIterator it = d->blocks.find(picture);
|
BlockIterator it = d->blocks.find(picture);
|
||||||
if(it != d->blocks.end())
|
if (it != d->blocks.end())
|
||||||
d->blocks.erase(it);
|
d->blocks.erase(it);
|
||||||
|
|
||||||
if(del)
|
|
||||||
delete picture;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void FLAC::File::removePictures()
|
void FLAC::File::removePictures() {
|
||||||
{
|
|
||||||
for(BlockIterator it = d->blocks.begin(); it != d->blocks.end(); ) {
|
for (BlockIterator it = d->blocks.begin(); it != d->blocks.end();) {
|
||||||
if(dynamic_cast<Picture *>(*it)) {
|
if (dynamic_cast<Picture *>(it->get())) {
|
||||||
delete *it;
|
|
||||||
it = d->blocks.erase(it);
|
it = d->blocks.erase(it);
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
++it;
|
++it;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void FLAC::File::strip(int tags)
|
void FLAC::File::strip(int tags) {
|
||||||
{
|
|
||||||
if(tags & ID3v1)
|
|
||||||
d->tag.set(FlacID3v1Index, 0);
|
|
||||||
|
|
||||||
if(tags & ID3v2)
|
if (tags & ID3v1)
|
||||||
d->tag.set(FlacID3v2Index, 0);
|
d->tag.set(FlacID3v1Index, nullptr);
|
||||||
|
|
||||||
if(tags & XiphComment) {
|
if (tags & ID3v2)
|
||||||
|
d->tag.set(FlacID3v2Index, nullptr);
|
||||||
|
|
||||||
|
if (tags & XiphComment) {
|
||||||
xiphComment()->removeAllFields();
|
xiphComment()->removeAllFields();
|
||||||
xiphComment()->removeAllPictures();
|
xiphComment()->removeAllPictures();
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool FLAC::File::hasXiphComment() const
|
bool FLAC::File::hasXiphComment() const {
|
||||||
{
|
|
||||||
return !d->xiphCommentData.isEmpty();
|
return !d->xiphCommentData.isEmpty();
|
||||||
}
|
}
|
||||||
|
|
||||||
bool FLAC::File::hasID3v1Tag() const
|
bool FLAC::File::hasID3v1Tag() const {
|
||||||
{
|
|
||||||
return (d->ID3v1Location >= 0);
|
return (d->ID3v1Location >= 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool FLAC::File::hasID3v2Tag() const
|
bool FLAC::File::hasID3v2Tag() const {
|
||||||
{
|
|
||||||
return (d->ID3v2Location >= 0);
|
return (d->ID3v2Location >= 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -412,13 +373,13 @@ bool FLAC::File::hasID3v2Tag() const
|
|||||||
// private members
|
// private members
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
void FLAC::File::read(bool readProperties)
|
void FLAC::File::read(bool readProperties) {
|
||||||
{
|
|
||||||
// Look for an ID3v2 tag
|
// Look for an ID3v2 tag
|
||||||
|
|
||||||
d->ID3v2Location = Utils::findID3v2(this);
|
d->ID3v2Location = Utils::findID3v2(this);
|
||||||
|
|
||||||
if(d->ID3v2Location >= 0) {
|
if (d->ID3v2Location >= 0) {
|
||||||
d->tag.set(FlacID3v2Index, new ID3v2::Tag(this, d->ID3v2Location, d->ID3v2FrameFactory));
|
d->tag.set(FlacID3v2Index, new ID3v2::Tag(this, d->ID3v2Location, d->ID3v2FrameFactory));
|
||||||
d->ID3v2OriginalSize = ID3v2Tag()->header()->completeTagSize();
|
d->ID3v2OriginalSize = ID3v2Tag()->header()->completeTagSize();
|
||||||
}
|
}
|
||||||
@@ -427,56 +388,57 @@ void FLAC::File::read(bool readProperties)
|
|||||||
|
|
||||||
d->ID3v1Location = Utils::findID3v1(this);
|
d->ID3v1Location = Utils::findID3v1(this);
|
||||||
|
|
||||||
if(d->ID3v1Location >= 0)
|
if (d->ID3v1Location >= 0)
|
||||||
d->tag.set(FlacID3v1Index, new ID3v1::Tag(this, d->ID3v1Location));
|
d->tag.set(FlacID3v1Index, new ID3v1::Tag(this, d->ID3v1Location));
|
||||||
|
|
||||||
// Look for FLAC metadata, including vorbis comments
|
// Look for FLAC metadata, including vorbis comments
|
||||||
|
|
||||||
scan();
|
scan();
|
||||||
|
|
||||||
if(!isValid())
|
if (!isValid())
|
||||||
return;
|
return;
|
||||||
|
|
||||||
if(!d->xiphCommentData.isEmpty())
|
if (!d->xiphCommentData.isEmpty())
|
||||||
d->tag.set(FlacXiphIndex, new Ogg::XiphComment(d->xiphCommentData));
|
d->tag.set(FlacXiphIndex, new Ogg::XiphComment(d->xiphCommentData));
|
||||||
else
|
else
|
||||||
d->tag.set(FlacXiphIndex, new Ogg::XiphComment());
|
d->tag.set(FlacXiphIndex, new Ogg::XiphComment());
|
||||||
|
|
||||||
if(readProperties) {
|
if (readProperties) {
|
||||||
|
|
||||||
// First block should be the stream_info metadata
|
// First block should be the stream_info metadata
|
||||||
|
|
||||||
const ByteVector infoData = d->blocks.front()->render();
|
const ByteVector infoData = d->blocks.front()->render();
|
||||||
|
|
||||||
long streamLength;
|
long long streamLength;
|
||||||
|
|
||||||
if(d->ID3v1Location >= 0)
|
if (d->ID3v1Location >= 0)
|
||||||
streamLength = d->ID3v1Location - d->streamStart;
|
streamLength = d->ID3v1Location - d->streamStart;
|
||||||
else
|
else
|
||||||
streamLength = length() - d->streamStart;
|
streamLength = length() - d->streamStart;
|
||||||
|
|
||||||
d->properties = new Properties(infoData, streamLength);
|
d->properties.reset(new AudioProperties(infoData, streamLength));
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void FLAC::File::scan()
|
void FLAC::File::scan() {
|
||||||
{
|
|
||||||
// Scan the metadata pages
|
// Scan the metadata pages
|
||||||
|
|
||||||
if(d->scanned)
|
if (d->scanned)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
if(!isValid())
|
if (!isValid())
|
||||||
return;
|
return;
|
||||||
|
|
||||||
long nextBlockOffset;
|
long long nextBlockOffset;
|
||||||
|
|
||||||
if(d->ID3v2Location >= 0)
|
if (d->ID3v2Location >= 0)
|
||||||
nextBlockOffset = find("fLaC", d->ID3v2Location + d->ID3v2OriginalSize);
|
nextBlockOffset = find("fLaC", d->ID3v2Location + d->ID3v2OriginalSize);
|
||||||
else
|
else
|
||||||
nextBlockOffset = find("fLaC");
|
nextBlockOffset = find("fLaC");
|
||||||
|
|
||||||
if(nextBlockOffset < 0) {
|
if (nextBlockOffset < 0) {
|
||||||
debug("FLAC::File::scan() -- FLAC stream not found");
|
debug("FLAC::File::scan() -- FLAC stream not found");
|
||||||
setValid(false);
|
setValid(false);
|
||||||
return;
|
return;
|
||||||
@@ -485,7 +447,7 @@ void FLAC::File::scan()
|
|||||||
nextBlockOffset += 4;
|
nextBlockOffset += 4;
|
||||||
d->flacStart = nextBlockOffset;
|
d->flacStart = nextBlockOffset;
|
||||||
|
|
||||||
while(true) {
|
while (true) {
|
||||||
|
|
||||||
seek(nextBlockOffset);
|
seek(nextBlockOffset);
|
||||||
const ByteVector header = readBlock(4);
|
const ByteVector header = readBlock(4);
|
||||||
@@ -504,66 +466,63 @@ void FLAC::File::scan()
|
|||||||
|
|
||||||
const char blockType = header[0] & ~LastBlockFlag;
|
const char blockType = header[0] & ~LastBlockFlag;
|
||||||
const bool isLastBlock = (header[0] & LastBlockFlag) != 0;
|
const bool isLastBlock = (header[0] & LastBlockFlag) != 0;
|
||||||
const unsigned int blockLength = header.toUInt(1U, 3U);
|
const size_t blockLength = header.toUInt24BE(1);
|
||||||
|
|
||||||
// First block should be the stream_info metadata
|
// First block should be the stream_info metadata
|
||||||
|
|
||||||
if(d->blocks.isEmpty() && blockType != MetadataBlock::StreamInfo) {
|
if (d->blocks.isEmpty() && blockType != MetadataBlock::StreamInfo) {
|
||||||
debug("FLAC::File::scan() -- First block should be the stream_info metadata");
|
debug("FLAC::File::scan() -- First block should be the stream_info metadata");
|
||||||
setValid(false);
|
setValid(false);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if(blockLength == 0
|
if (blockLength == 0 && blockType != MetadataBlock::Padding && blockType != MetadataBlock::SeekTable) {
|
||||||
&& blockType != MetadataBlock::Padding && blockType != MetadataBlock::SeekTable)
|
|
||||||
{
|
|
||||||
debug("FLAC::File::scan() -- Zero-sized metadata block found");
|
debug("FLAC::File::scan() -- Zero-sized metadata block found");
|
||||||
setValid(false);
|
setValid(false);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
const ByteVector data = readBlock(blockLength);
|
const ByteVector data = readBlock(blockLength);
|
||||||
if(data.size() != blockLength) {
|
if (data.size() != blockLength) {
|
||||||
debug("FLAC::File::scan() -- Failed to read a metadata block");
|
debug("FLAC::File::scan() -- Failed to read a metadata block");
|
||||||
setValid(false);
|
setValid(false);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
MetadataBlock *block = 0;
|
std::shared_ptr<MetadataBlock> block;
|
||||||
|
|
||||||
// Found the vorbis-comment
|
// Found the vorbis-comment
|
||||||
if(blockType == MetadataBlock::VorbisComment) {
|
if (blockType == MetadataBlock::VorbisComment) {
|
||||||
if(d->xiphCommentData.isEmpty()) {
|
if (d->xiphCommentData.isEmpty()) {
|
||||||
d->xiphCommentData = data;
|
d->xiphCommentData = data;
|
||||||
block = new UnknownMetadataBlock(MetadataBlock::VorbisComment, data);
|
block.reset(new UnknownMetadataBlock(MetadataBlock::VorbisComment, data));
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
debug("FLAC::File::scan() -- multiple Vorbis Comment blocks found, discarding");
|
debug("FLAC::File::scan() -- multiple Vorbis Comment blocks found, discarding");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else if(blockType == MetadataBlock::Picture) {
|
else if (blockType == MetadataBlock::Picture) {
|
||||||
FLAC::Picture *picture = new FLAC::Picture();
|
std::shared_ptr<FLAC::Picture> picture(new FLAC::Picture());
|
||||||
if(picture->parse(data)) {
|
if (picture->parse(data)) {
|
||||||
block = picture;
|
block = picture;
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
debug("FLAC::File::scan() -- invalid picture found, discarding");
|
debug("FLAC::File::scan() -- invalid picture found, discarding");
|
||||||
delete picture;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else if(blockType == MetadataBlock::Padding) {
|
else if (blockType == MetadataBlock::Padding) {
|
||||||
// Skip all padding blocks.
|
// Skip all padding blocks.
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
block = new UnknownMetadataBlock(blockType, data);
|
block.reset(new UnknownMetadataBlock(blockType, data));
|
||||||
}
|
}
|
||||||
|
|
||||||
if(block)
|
if (block)
|
||||||
d->blocks.append(block);
|
d->blocks.append(block);
|
||||||
|
|
||||||
nextBlockOffset += blockLength + 4;
|
nextBlockOffset += blockLength + 4;
|
||||||
|
|
||||||
if(isLastBlock)
|
if (isLastBlock)
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -572,4 +531,5 @@ void FLAC::File::scan()
|
|||||||
d->streamStart = nextBlockOffset;
|
d->streamStart = nextBlockOffset;
|
||||||
|
|
||||||
d->scanned = true;
|
d->scanned = true;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
489
3rdparty/taglib/flac/flacfile.h
vendored
489
3rdparty/taglib/flac/flacfile.h
vendored
@@ -26,6 +26,8 @@
|
|||||||
#ifndef TAGLIB_FLACFILE_H
|
#ifndef TAGLIB_FLACFILE_H
|
||||||
#define TAGLIB_FLACFILE_H
|
#define TAGLIB_FLACFILE_H
|
||||||
|
|
||||||
|
#include <memory>
|
||||||
|
|
||||||
#include "taglib_export.h"
|
#include "taglib_export.h"
|
||||||
#include "tfile.h"
|
#include "tfile.h"
|
||||||
#include "tlist.h"
|
#include "tlist.h"
|
||||||
@@ -37,309 +39,240 @@
|
|||||||
namespace Strawberry_TagLib {
|
namespace Strawberry_TagLib {
|
||||||
namespace TagLib {
|
namespace TagLib {
|
||||||
|
|
||||||
class Tag;
|
class Tag;
|
||||||
namespace ID3v2 { class FrameFactory; class Tag; }
|
namespace ID3v2 {
|
||||||
namespace ID3v1 { class Tag; }
|
class FrameFactory;
|
||||||
namespace Ogg { class XiphComment; }
|
class Tag;
|
||||||
|
} // namespace ID3v2
|
||||||
|
namespace ID3v1 {
|
||||||
|
class Tag;
|
||||||
|
}
|
||||||
|
namespace Ogg {
|
||||||
|
class XiphComment;
|
||||||
|
}
|
||||||
|
|
||||||
//! An implementation of FLAC metadata
|
//! An implementation of FLAC metadata
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* This is implementation of FLAC metadata for non-Ogg FLAC files. At some
|
||||||
|
* point when Ogg / FLAC is more common there will be a similar implementation
|
||||||
|
* under the Ogg hierarchy.
|
||||||
|
*
|
||||||
|
* This supports ID3v1, ID3v2 and Xiph style comments as well as reading stream
|
||||||
|
* properties from the file.
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace FLAC {
|
||||||
|
|
||||||
|
//! An implementation of TagLib::File with FLAC specific methods
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* This implements and provides an interface for FLAC files to the TagLib::Tag and TagLib::AudioProperties interfaces
|
||||||
|
* by way of implementing the abstract TagLib::File API as well as providing some additional information specific to FLAC files.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
class TAGLIB_EXPORT File : public Strawberry_TagLib::TagLib::File {
|
||||||
|
public:
|
||||||
|
/*!
|
||||||
|
* This set of flags is used for various operations and is suitable for
|
||||||
|
* being OR-ed together.
|
||||||
|
*/
|
||||||
|
enum TagTypes {
|
||||||
|
//! Empty set. Matches no tag types.
|
||||||
|
NoTags = 0x0000,
|
||||||
|
//! Matches Vorbis comments.
|
||||||
|
XiphComment = 0x0001,
|
||||||
|
//! Matches ID3v1 tags.
|
||||||
|
ID3v1 = 0x0002,
|
||||||
|
//! Matches ID3v2 tags.
|
||||||
|
ID3v2 = 0x0004,
|
||||||
|
//! Matches all tag types.
|
||||||
|
AllTags = 0xffff
|
||||||
|
};
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* This is implementation of FLAC metadata for non-Ogg FLAC files. At some
|
* Constructs an FLAC file from \a file.
|
||||||
* point when Ogg / FLAC is more common there will be a similar implementation
|
* If \a readProperties is true the file's audio properties will also be read.
|
||||||
* under the Ogg hierarchy.
|
|
||||||
*
|
*
|
||||||
* This supports ID3v1, ID3v2 and Xiph style comments as well as reading stream
|
* If this file contains and ID3v2 tag the frames will be created using \a frameFactory.
|
||||||
* properties from the file.
|
*
|
||||||
|
* \note In the current implementation, \a propertiesStyle is ignored.
|
||||||
*/
|
*/
|
||||||
|
explicit File(FileName file, bool readProperties = true, AudioProperties::ReadStyle propertiesStyle = AudioProperties::Average, ID3v2::FrameFactory *frameFactory = nullptr);
|
||||||
|
|
||||||
namespace FLAC {
|
/*!
|
||||||
|
* Constructs a FLAC file from \a stream. If \a readProperties is true the file's audio properties will also be read.
|
||||||
|
*
|
||||||
|
* \note TagLib will *not* take ownership of the stream, the caller is responsible for deleting it after the File object.
|
||||||
|
*
|
||||||
|
* If this file contains and ID3v2 tag the frames will be created using \a frameFactory.
|
||||||
|
*
|
||||||
|
* \note In the current implementation, \a propertiesStyle is ignored.
|
||||||
|
*/
|
||||||
|
// BIC: merge with the above constructor
|
||||||
|
explicit File(IOStream *stream, bool readProperties = true, AudioProperties::ReadStyle propertiesStyle = AudioProperties::Average, ID3v2::FrameFactory *frameFactory = nullptr);
|
||||||
|
|
||||||
//! An implementation of TagLib::File with FLAC specific methods
|
/*!
|
||||||
|
* Destroys this instance of the File.
|
||||||
|
*/
|
||||||
|
~File() override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* This implements and provides an interface for FLAC files to the
|
* Returns the Tag for this file. This will be a union of XiphComment, ID3v1 and ID3v2 tags.
|
||||||
* TagLib::Tag and TagLib::AudioProperties interfaces by way of implementing
|
*
|
||||||
* the abstract TagLib::File API as well as providing some additional
|
* \see ID3v2Tag()
|
||||||
* information specific to FLAC files.
|
* \see ID3v1Tag()
|
||||||
*/
|
* \see XiphComment()
|
||||||
|
*/
|
||||||
|
Strawberry_TagLib::TagLib::Tag *tag() const override;
|
||||||
|
|
||||||
class TAGLIB_EXPORT File : public Strawberry_TagLib::TagLib::File
|
/*!
|
||||||
{
|
* Implements the unified property interface -- import function.
|
||||||
public:
|
* This always creates a Xiph comment, if none exists. The return value relates to the Xiph comment only.
|
||||||
/*!
|
* Ignores any changes to ID3v1 or ID3v2 comments since they are not allowed in the FLAC specification.
|
||||||
* This set of flags is used for various operations and is suitable for
|
*/
|
||||||
* being OR-ed together.
|
PropertyMap setProperties(const PropertyMap &) override;
|
||||||
*/
|
|
||||||
enum TagTypes {
|
|
||||||
//! Empty set. Matches no tag types.
|
|
||||||
NoTags = 0x0000,
|
|
||||||
//! Matches Vorbis comments.
|
|
||||||
XiphComment = 0x0001,
|
|
||||||
//! Matches ID3v1 tags.
|
|
||||||
ID3v1 = 0x0002,
|
|
||||||
//! Matches ID3v2 tags.
|
|
||||||
ID3v2 = 0x0004,
|
|
||||||
//! Matches all tag types.
|
|
||||||
AllTags = 0xffff
|
|
||||||
};
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Constructs a FLAC file from \a file. If \a readProperties is true the
|
* Returns the FLAC::AudioProperties for this file. If no audio properties were read then this will return a null pointer.
|
||||||
* file's audio properties will also be read.
|
*/
|
||||||
*
|
AudioProperties *audioProperties() const override;
|
||||||
* \note In the current implementation, \a propertiesStyle is ignored.
|
|
||||||
*
|
|
||||||
* \deprecated This constructor will be dropped in favor of the one below
|
|
||||||
* in a future version.
|
|
||||||
*/
|
|
||||||
File(FileName file, bool readProperties = true,
|
|
||||||
Properties::ReadStyle propertiesStyle = Properties::Average);
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Constructs an FLAC file from \a file. If \a readProperties is true the
|
* Save the file. This will primarily save the XiphComment, but will also keep any old ID3-tags up to date.
|
||||||
* file's audio properties will also be read.
|
* If the file has no XiphComment, one will be constructed from the ID3-tags.
|
||||||
*
|
*
|
||||||
* If this file contains and ID3v2 tag the frames will be created using
|
* This returns true if the save was successful.
|
||||||
* \a frameFactory.
|
*/
|
||||||
*
|
bool save() override;
|
||||||
* \note In the current implementation, \a propertiesStyle is ignored.
|
|
||||||
*/
|
|
||||||
// BIC: merge with the above constructor
|
|
||||||
File(FileName file, ID3v2::FrameFactory *frameFactory,
|
|
||||||
bool readProperties = true,
|
|
||||||
Properties::ReadStyle propertiesStyle = Properties::Average);
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Constructs a FLAC file from \a stream. If \a readProperties is true the
|
* Returns a pointer to the ID3v2 tag of the file.
|
||||||
* file's audio properties will also be read.
|
*
|
||||||
*
|
* If \a create is false (the default) this returns a null pointer
|
||||||
* \note TagLib will *not* take ownership of the stream, the caller is
|
* if there is no valid ID3v2 tag.
|
||||||
* responsible for deleting it after the File object.
|
* If \a create is true it will create an ID3v2 tag if one does not exist and returns a valid pointer.
|
||||||
*
|
*
|
||||||
* If this file contains and ID3v2 tag the frames will be created using
|
* \note This may return a valid pointer regardless of whether or not the file on disk has an ID3v2 tag.
|
||||||
* \a frameFactory.
|
* Use hasID3v2Tag() to check if the file on disk actually has an ID3v2 tag.
|
||||||
*
|
*
|
||||||
* \note In the current implementation, \a propertiesStyle is ignored.
|
* \note The Tag <b>is still</b> owned by the MPEG::File and should not be
|
||||||
*/
|
* deleted by the user. It will be deleted when the file (object) is destroyed.
|
||||||
// BIC: merge with the above constructor
|
*
|
||||||
File(IOStream *stream, ID3v2::FrameFactory *frameFactory,
|
* \see hasID3v2Tag()
|
||||||
bool readProperties = true,
|
*/
|
||||||
Properties::ReadStyle propertiesStyle = Properties::Average);
|
ID3v2::Tag *ID3v2Tag(bool create = false);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Destroys this instance of the File.
|
* Returns a pointer to the ID3v1 tag of the file.
|
||||||
*/
|
*
|
||||||
virtual ~File();
|
* If \a create is false (the default) this returns a null pointer if there is no valid APE tag.
|
||||||
|
* If \a create is true it will create an APE tag if one does not exist and returns a valid pointer.
|
||||||
|
*
|
||||||
|
* \note This may return a valid pointer regardless of whether or not the file on disk has an ID3v1 tag.
|
||||||
|
* Use hasID3v1Tag() to check if the file on disk actually has an ID3v1 tag.
|
||||||
|
*
|
||||||
|
* \note The Tag <b>is still</b> owned by the MPEG::File and should not be deleted by the user.
|
||||||
|
* It will be deleted when the file (object) is destroyed.
|
||||||
|
*
|
||||||
|
* \see hasID3v1Tag()
|
||||||
|
*/
|
||||||
|
ID3v1::Tag *ID3v1Tag(bool create = false);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the Tag for this file. This will be a union of XiphComment,
|
* Returns a pointer to the XiphComment for the file.
|
||||||
* ID3v1 and ID3v2 tags.
|
*
|
||||||
*
|
* If \a create is false (the default) this returns a null pointer if there is no valid XiphComment.
|
||||||
* \see ID3v2Tag()
|
* If \a create is true it will create a XiphComment if one does not exist and returns a valid pointer.
|
||||||
* \see ID3v1Tag()
|
*
|
||||||
* \see XiphComment()
|
* \note This may return a valid pointer regardless of whether or not the file on disk has a XiphComment.
|
||||||
*/
|
* Use hasXiphComment() to check if the file on disk actually has a XiphComment.
|
||||||
virtual Strawberry_TagLib::TagLib::Tag *tag() const;
|
*
|
||||||
|
* \note The Tag <b>is still</b> owned by the FLAC::File and should not be deleted by the user.
|
||||||
|
* It will be deleted when the file (object) is destroyed.
|
||||||
|
*
|
||||||
|
* \see hasXiphComment()
|
||||||
|
*/
|
||||||
|
Ogg::XiphComment *xiphComment(bool create = false);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Implements the unified property interface -- export function.
|
* Returns a list of pictures attached to the FLAC file.
|
||||||
* If the file contains more than one tag (e.g. XiphComment and ID3v1),
|
*/
|
||||||
* only the first one (in the order XiphComment, ID3v2, ID3v1) will be
|
List<Picture *> pictureList();
|
||||||
* converted to the PropertyMap.
|
|
||||||
*/
|
|
||||||
PropertyMap properties() const;
|
|
||||||
|
|
||||||
void removeUnsupportedProperties(const StringList &);
|
/*!
|
||||||
|
* Removes an attached picture. The picture's memory will be freed.
|
||||||
|
*/
|
||||||
|
void removePicture(Picture *picture);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Implements the unified property interface -- import function.
|
* Remove all attached images.
|
||||||
* This always creates a Xiph comment, if none exists. The return value
|
*/
|
||||||
* relates to the Xiph comment only.
|
void removePictures();
|
||||||
* Ignores any changes to ID3v1 or ID3v2 comments since they are not allowed
|
|
||||||
* in the FLAC specification.
|
|
||||||
*/
|
|
||||||
PropertyMap setProperties(const PropertyMap &);
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the FLAC::Properties for this file. If no audio properties
|
* Add a new picture to the file.
|
||||||
* were read then this will return a null pointer.
|
* The file takes ownership of the picture and will handle freeing its memory.
|
||||||
*/
|
*
|
||||||
virtual Properties *audioProperties() const;
|
* \note The file will be saved only after calling save().
|
||||||
|
*/
|
||||||
|
void addPicture(Picture *picture);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Save the file. This will primarily save the XiphComment, but
|
* This will remove the tags that match the OR-ed together TagTypes from the file.
|
||||||
* will also keep any old ID3-tags up to date. If the file
|
* By default it removes all tags.
|
||||||
* has no XiphComment, one will be constructed from the ID3-tags.
|
*
|
||||||
*
|
* \warning This will also invalidate pointers to the tags as their memory will be freed.
|
||||||
* This returns true if the save was successful.
|
*
|
||||||
*/
|
* \note In order to make the removal permanent save() still needs to be called.
|
||||||
virtual bool save();
|
*
|
||||||
|
* \note This won't remove the Vorbis comment block completely.
|
||||||
|
* The vendor ID will be preserved.
|
||||||
|
*/
|
||||||
|
void strip(int tags = AllTags);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns a pointer to the ID3v2 tag of the file.
|
* Returns whether or not the file on disk actually has a XiphComment.
|
||||||
*
|
*
|
||||||
* If \a create is false (the default) this returns a null pointer
|
* \see xiphComment()
|
||||||
* if there is no valid ID3v2 tag. If \a create is true it will create
|
*/
|
||||||
* an ID3v2 tag if one does not exist and returns a valid pointer.
|
bool hasXiphComment() const;
|
||||||
*
|
|
||||||
* \note This may return a valid pointer regardless of whether or not the
|
|
||||||
* file on disk has an ID3v2 tag. Use hasID3v2Tag() to check if the file
|
|
||||||
* on disk actually has an ID3v2 tag.
|
|
||||||
*
|
|
||||||
* \note The Tag <b>is still</b> owned by the MPEG::File and should not be
|
|
||||||
* deleted by the user. It will be deleted when the file (object) is
|
|
||||||
* destroyed.
|
|
||||||
*
|
|
||||||
* \see hasID3v2Tag()
|
|
||||||
*/
|
|
||||||
ID3v2::Tag *ID3v2Tag(bool create = false);
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns a pointer to the ID3v1 tag of the file.
|
* Returns whether or not the file on disk actually has an ID3v1 tag.
|
||||||
*
|
*
|
||||||
* If \a create is false (the default) this returns a null pointer
|
* \see ID3v1Tag()
|
||||||
* if there is no valid APE tag. If \a create is true it will create
|
*/
|
||||||
* an APE tag if one does not exist and returns a valid pointer.
|
bool hasID3v1Tag() const;
|
||||||
*
|
|
||||||
* \note This may return a valid pointer regardless of whether or not the
|
|
||||||
* file on disk has an ID3v1 tag. Use hasID3v1Tag() to check if the file
|
|
||||||
* on disk actually has an ID3v1 tag.
|
|
||||||
*
|
|
||||||
* \note The Tag <b>is still</b> owned by the MPEG::File and should not be
|
|
||||||
* deleted by the user. It will be deleted when the file (object) is
|
|
||||||
* destroyed.
|
|
||||||
*
|
|
||||||
* \see hasID3v1Tag()
|
|
||||||
*/
|
|
||||||
ID3v1::Tag *ID3v1Tag(bool create = false);
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns a pointer to the XiphComment for the file.
|
* Returns whether or not the file on disk actually has an ID3v2 tag.
|
||||||
*
|
*
|
||||||
* If \a create is false (the default) this returns a null pointer
|
* \see ID3v2Tag()
|
||||||
* if there is no valid XiphComment. If \a create is true it will create
|
*/
|
||||||
* a XiphComment if one does not exist and returns a valid pointer.
|
bool hasID3v2Tag() const;
|
||||||
*
|
|
||||||
* \note This may return a valid pointer regardless of whether or not the
|
|
||||||
* file on disk has a XiphComment. Use hasXiphComment() to check if the
|
|
||||||
* file on disk actually has a XiphComment.
|
|
||||||
*
|
|
||||||
* \note The Tag <b>is still</b> owned by the FLAC::File and should not be
|
|
||||||
* deleted by the user. It will be deleted when the file (object) is
|
|
||||||
* destroyed.
|
|
||||||
*
|
|
||||||
* \see hasXiphComment()
|
|
||||||
*/
|
|
||||||
Ogg::XiphComment *xiphComment(bool create = false);
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Set the ID3v2::FrameFactory to something other than the default. This
|
* Returns whether or not the given \a stream can be opened as a FLAC file.
|
||||||
* can be used to specify the way that ID3v2 frames will be interpreted
|
*
|
||||||
* when
|
* \note This method is designed to do a quick check.
|
||||||
*
|
* The result may not necessarily be correct.
|
||||||
* \see ID3v2FrameFactory
|
*/
|
||||||
* \deprecated This value should be passed in via the constructor
|
static bool isSupported(IOStream *stream);
|
||||||
*/
|
|
||||||
TAGLIB_DEPRECATED void setID3v2FrameFactory(const ID3v2::FrameFactory *factory);
|
|
||||||
|
|
||||||
/*!
|
private:
|
||||||
* Returns the block of data used by FLAC::Properties for parsing the
|
File(const File&);
|
||||||
* stream properties.
|
File &operator=(const File&);
|
||||||
*
|
|
||||||
* \deprecated Always returns an empty vector.
|
|
||||||
*/
|
|
||||||
TAGLIB_DEPRECATED ByteVector streamInfoData(); // BIC: remove
|
|
||||||
|
|
||||||
/*!
|
void read(bool readProperties);
|
||||||
* Returns the length of the audio-stream, used by FLAC::Properties for
|
void scan();
|
||||||
* calculating the bitrate.
|
|
||||||
*
|
|
||||||
* \deprecated Always returns zero.
|
|
||||||
*/
|
|
||||||
TAGLIB_DEPRECATED long streamLength(); // BIC: remove
|
|
||||||
|
|
||||||
/*!
|
class FilePrivate;
|
||||||
* Returns a list of pictures attached to the FLAC file.
|
FilePrivate *d;
|
||||||
*/
|
};
|
||||||
List<Picture *> pictureList();
|
} // namespace FLAC
|
||||||
|
} // namespace TagLib
|
||||||
/*!
|
} // namespace Strawberry_TagLib
|
||||||
* Removes an attached picture. If \a del is true the picture's memory
|
|
||||||
* will be freed; if it is false, it must be deleted by the user.
|
|
||||||
*/
|
|
||||||
void removePicture(Picture *picture, bool del = true);
|
|
||||||
|
|
||||||
/*!
|
|
||||||
* Remove all attached images.
|
|
||||||
*/
|
|
||||||
void removePictures();
|
|
||||||
|
|
||||||
/*!
|
|
||||||
* Add a new picture to the file. The file takes ownership of the
|
|
||||||
* picture and will handle freeing its memory.
|
|
||||||
*
|
|
||||||
* \note The file will be saved only after calling save().
|
|
||||||
*/
|
|
||||||
void addPicture(Picture *picture);
|
|
||||||
|
|
||||||
/*!
|
|
||||||
* This will remove the tags that match the OR-ed together TagTypes from
|
|
||||||
* the file. By default it removes all tags.
|
|
||||||
*
|
|
||||||
* \warning This will also invalidate pointers to the tags as their memory
|
|
||||||
* will be freed.
|
|
||||||
*
|
|
||||||
* \note In order to make the removal permanent save() still needs to be
|
|
||||||
* called.
|
|
||||||
*
|
|
||||||
* \note This won't remove the Vorbis comment block completely. The
|
|
||||||
* vendor ID will be preserved.
|
|
||||||
*/
|
|
||||||
void strip(int tags = AllTags);
|
|
||||||
|
|
||||||
/*!
|
|
||||||
* Returns whether or not the file on disk actually has a XiphComment.
|
|
||||||
*
|
|
||||||
* \see xiphComment()
|
|
||||||
*/
|
|
||||||
bool hasXiphComment() const;
|
|
||||||
|
|
||||||
/*!
|
|
||||||
* Returns whether or not the file on disk actually has an ID3v1 tag.
|
|
||||||
*
|
|
||||||
* \see ID3v1Tag()
|
|
||||||
*/
|
|
||||||
bool hasID3v1Tag() const;
|
|
||||||
|
|
||||||
/*!
|
|
||||||
* Returns whether or not the file on disk actually has an ID3v2 tag.
|
|
||||||
*
|
|
||||||
* \see ID3v2Tag()
|
|
||||||
*/
|
|
||||||
bool hasID3v2Tag() const;
|
|
||||||
|
|
||||||
/*!
|
|
||||||
* Returns whether or not the given \a stream can be opened as a FLAC
|
|
||||||
* file.
|
|
||||||
*
|
|
||||||
* \note This method is designed to do a quick check. The result may
|
|
||||||
* not necessarily be correct.
|
|
||||||
*/
|
|
||||||
static bool isSupported(IOStream *stream);
|
|
||||||
|
|
||||||
private:
|
|
||||||
File(const File &);
|
|
||||||
File &operator=(const File &);
|
|
||||||
|
|
||||||
void read(bool readProperties);
|
|
||||||
void scan();
|
|
||||||
|
|
||||||
class FilePrivate;
|
|
||||||
FilePrivate *d;
|
|
||||||
};
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
20
3rdparty/taglib/flac/flacmetadatablock.cpp
vendored
20
3rdparty/taglib/flac/flacmetadatablock.cpp
vendored
@@ -23,25 +23,17 @@
|
|||||||
* http://www.mozilla.org/MPL/ *
|
* http://www.mozilla.org/MPL/ *
|
||||||
***************************************************************************/
|
***************************************************************************/
|
||||||
|
|
||||||
#include <taglib.h>
|
#include "taglib.h"
|
||||||
#include <tdebug.h>
|
#include "tdebug.h"
|
||||||
#include "flacmetadatablock.h"
|
#include "flacmetadatablock.h"
|
||||||
|
|
||||||
using namespace Strawberry_TagLib::TagLib;
|
using namespace Strawberry_TagLib::TagLib;
|
||||||
|
|
||||||
class FLAC::MetadataBlock::MetadataBlockPrivate
|
class FLAC::MetadataBlock::MetadataBlockPrivate {
|
||||||
{
|
public:
|
||||||
public:
|
|
||||||
MetadataBlockPrivate() {}
|
MetadataBlockPrivate() {}
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
FLAC::MetadataBlock::MetadataBlock()
|
FLAC::MetadataBlock::MetadataBlock() : d(nullptr) {}
|
||||||
{
|
|
||||||
d = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
FLAC::MetadataBlock::~MetadataBlock()
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
|
FLAC::MetadataBlock::~MetadataBlock() {}
|
||||||
|
|||||||
65
3rdparty/taglib/flac/flacmetadatablock.h
vendored
65
3rdparty/taglib/flac/flacmetadatablock.h
vendored
@@ -32,46 +32,43 @@
|
|||||||
|
|
||||||
namespace Strawberry_TagLib {
|
namespace Strawberry_TagLib {
|
||||||
namespace TagLib {
|
namespace TagLib {
|
||||||
|
namespace FLAC {
|
||||||
|
|
||||||
namespace FLAC {
|
class TAGLIB_EXPORT MetadataBlock {
|
||||||
|
public:
|
||||||
|
explicit MetadataBlock();
|
||||||
|
virtual ~MetadataBlock();
|
||||||
|
|
||||||
class TAGLIB_EXPORT MetadataBlock
|
enum BlockType {
|
||||||
{
|
StreamInfo = 0,
|
||||||
public:
|
Padding,
|
||||||
MetadataBlock();
|
Application,
|
||||||
virtual ~MetadataBlock();
|
SeekTable,
|
||||||
|
VorbisComment,
|
||||||
|
CueSheet,
|
||||||
|
Picture
|
||||||
|
};
|
||||||
|
|
||||||
enum BlockType {
|
/*!
|
||||||
StreamInfo = 0,
|
* Returns the FLAC metadata block type.
|
||||||
Padding,
|
*/
|
||||||
Application,
|
virtual int code() const = 0;
|
||||||
SeekTable,
|
|
||||||
VorbisComment,
|
|
||||||
CueSheet,
|
|
||||||
Picture
|
|
||||||
};
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the FLAC metadata block type.
|
* Render the content of the block.
|
||||||
*/
|
*/
|
||||||
virtual int code() const = 0;
|
virtual ByteVector render() const = 0;
|
||||||
|
|
||||||
/*!
|
private:
|
||||||
* Render the content of the block.
|
MetadataBlock(const MetadataBlock &item);
|
||||||
*/
|
MetadataBlock &operator=(const MetadataBlock &item);
|
||||||
virtual ByteVector render() const = 0;
|
|
||||||
|
|
||||||
private:
|
class MetadataBlockPrivate;
|
||||||
MetadataBlock(const MetadataBlock &item);
|
MetadataBlockPrivate *d;
|
||||||
MetadataBlock &operator=(const MetadataBlock &item);
|
};
|
||||||
|
|
||||||
class MetadataBlockPrivate;
|
} // namespace FLAC
|
||||||
MetadataBlockPrivate *d;
|
} // namespace TagLib
|
||||||
};
|
} // namespace Strawberry_TagLib
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
137
3rdparty/taglib/flac/flacpicture.cpp
vendored
137
3rdparty/taglib/flac/flacpicture.cpp
vendored
@@ -23,22 +23,20 @@
|
|||||||
* http://www.mozilla.org/MPL/ *
|
* http://www.mozilla.org/MPL/ *
|
||||||
***************************************************************************/
|
***************************************************************************/
|
||||||
|
|
||||||
#include <taglib.h>
|
#include "taglib.h"
|
||||||
#include <tdebug.h>
|
#include "tdebug.h"
|
||||||
#include "flacpicture.h"
|
#include "flacpicture.h"
|
||||||
|
|
||||||
using namespace Strawberry_TagLib::TagLib;
|
using namespace Strawberry_TagLib::TagLib;
|
||||||
|
|
||||||
class FLAC::Picture::PicturePrivate
|
class FLAC::Picture::PicturePrivate {
|
||||||
{
|
public:
|
||||||
public:
|
explicit PicturePrivate() : type(FLAC::Picture::Other),
|
||||||
PicturePrivate() :
|
width(0),
|
||||||
type(FLAC::Picture::Other),
|
height(0),
|
||||||
width(0),
|
colorDepth(0),
|
||||||
height(0),
|
numColors(0) {
|
||||||
colorDepth(0),
|
}
|
||||||
numColors(0)
|
|
||||||
{}
|
|
||||||
|
|
||||||
Type type;
|
Type type;
|
||||||
String mimeType;
|
String mimeType;
|
||||||
@@ -50,168 +48,147 @@ public:
|
|||||||
ByteVector data;
|
ByteVector data;
|
||||||
};
|
};
|
||||||
|
|
||||||
FLAC::Picture::Picture() :
|
FLAC::Picture::Picture() : d(new PicturePrivate()) {
|
||||||
d(new PicturePrivate())
|
|
||||||
{
|
|
||||||
}
|
}
|
||||||
|
|
||||||
FLAC::Picture::Picture(const ByteVector &data) :
|
FLAC::Picture::Picture(const ByteVector &data) : d(new PicturePrivate()) {
|
||||||
d(new PicturePrivate())
|
|
||||||
{
|
|
||||||
parse(data);
|
parse(data);
|
||||||
}
|
}
|
||||||
|
|
||||||
FLAC::Picture::~Picture()
|
FLAC::Picture::~Picture() {
|
||||||
{
|
|
||||||
delete d;
|
delete d;
|
||||||
}
|
}
|
||||||
|
|
||||||
int FLAC::Picture::code() const
|
int FLAC::Picture::code() const {
|
||||||
{
|
|
||||||
return FLAC::MetadataBlock::Picture;
|
return FLAC::MetadataBlock::Picture;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool FLAC::Picture::parse(const ByteVector &data)
|
bool FLAC::Picture::parse(const ByteVector &data) {
|
||||||
{
|
|
||||||
if(data.size() < 32) {
|
if (data.size() < 32) {
|
||||||
debug("A picture block must contain at least 5 bytes.");
|
debug("A picture block must contain at least 5 bytes.");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned int pos = 0;
|
size_t pos = 0;
|
||||||
d->type = FLAC::Picture::Type(data.toUInt(pos));
|
d->type = FLAC::Picture::Type(data.toUInt32BE(pos));
|
||||||
pos += 4;
|
pos += 4;
|
||||||
unsigned int mimeTypeLength = data.toUInt(pos);
|
const unsigned int mimeTypeLength = data.toUInt32BE(pos);
|
||||||
pos += 4;
|
pos += 4;
|
||||||
if(pos + mimeTypeLength + 24 > data.size()) {
|
if (pos + mimeTypeLength + 24 > data.size()) {
|
||||||
debug("Invalid picture block.");
|
debug("Invalid picture block.");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
d->mimeType = String(data.mid(pos, mimeTypeLength), String::UTF8);
|
d->mimeType = String(data.mid(pos, mimeTypeLength), String::UTF8);
|
||||||
pos += mimeTypeLength;
|
pos += mimeTypeLength;
|
||||||
unsigned int descriptionLength = data.toUInt(pos);
|
const unsigned int descriptionLength = data.toUInt32BE(pos);
|
||||||
pos += 4;
|
pos += 4;
|
||||||
if(pos + descriptionLength + 20 > data.size()) {
|
if (pos + descriptionLength + 20 > data.size()) {
|
||||||
debug("Invalid picture block.");
|
debug("Invalid picture block.");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
d->description = String(data.mid(pos, descriptionLength), String::UTF8);
|
d->description = String(data.mid(pos, descriptionLength), String::UTF8);
|
||||||
pos += descriptionLength;
|
pos += descriptionLength;
|
||||||
d->width = data.toUInt(pos);
|
d->width = data.toUInt32BE(pos);
|
||||||
pos += 4;
|
pos += 4;
|
||||||
d->height = data.toUInt(pos);
|
d->height = data.toUInt32BE(pos);
|
||||||
pos += 4;
|
pos += 4;
|
||||||
d->colorDepth = data.toUInt(pos);
|
d->colorDepth = data.toUInt32BE(pos);
|
||||||
pos += 4;
|
pos += 4;
|
||||||
d->numColors = data.toUInt(pos);
|
d->numColors = data.toUInt32BE(pos);
|
||||||
pos += 4;
|
pos += 4;
|
||||||
unsigned int dataLength = data.toUInt(pos);
|
const unsigned int dataLength = data.toUInt32BE(pos);
|
||||||
pos += 4;
|
pos += 4;
|
||||||
if(pos + dataLength > data.size()) {
|
if (pos + dataLength > data.size()) {
|
||||||
debug("Invalid picture block.");
|
debug("Invalid picture block.");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
d->data = data.mid(pos, dataLength);
|
d->data = data.mid(pos, dataLength);
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ByteVector FLAC::Picture::render() const
|
ByteVector FLAC::Picture::render() const {
|
||||||
{
|
|
||||||
ByteVector result;
|
ByteVector result;
|
||||||
result.append(ByteVector::fromUInt(d->type));
|
result.append(ByteVector::fromUInt32BE(d->type));
|
||||||
ByteVector mimeTypeData = d->mimeType.data(String::UTF8);
|
ByteVector mimeTypeData = d->mimeType.data(String::UTF8);
|
||||||
result.append(ByteVector::fromUInt(mimeTypeData.size()));
|
result.append(ByteVector::fromUInt32BE(mimeTypeData.size()));
|
||||||
result.append(mimeTypeData);
|
result.append(mimeTypeData);
|
||||||
ByteVector descriptionData = d->description.data(String::UTF8);
|
ByteVector descriptionData = d->description.data(String::UTF8);
|
||||||
result.append(ByteVector::fromUInt(descriptionData.size()));
|
result.append(ByteVector::fromUInt32BE(descriptionData.size()));
|
||||||
result.append(descriptionData);
|
result.append(descriptionData);
|
||||||
result.append(ByteVector::fromUInt(d->width));
|
result.append(ByteVector::fromUInt32BE(d->width));
|
||||||
result.append(ByteVector::fromUInt(d->height));
|
result.append(ByteVector::fromUInt32BE(d->height));
|
||||||
result.append(ByteVector::fromUInt(d->colorDepth));
|
result.append(ByteVector::fromUInt32BE(d->colorDepth));
|
||||||
result.append(ByteVector::fromUInt(d->numColors));
|
result.append(ByteVector::fromUInt32BE(d->numColors));
|
||||||
result.append(ByteVector::fromUInt(d->data.size()));
|
result.append(ByteVector::fromUInt32BE(d->data.size()));
|
||||||
result.append(d->data);
|
result.append(d->data);
|
||||||
return result;
|
return result;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
FLAC::Picture::Type FLAC::Picture::type() const
|
FLAC::Picture::Type FLAC::Picture::type() const {
|
||||||
{
|
|
||||||
return d->type;
|
return d->type;
|
||||||
}
|
}
|
||||||
|
|
||||||
void FLAC::Picture::setType(FLAC::Picture::Type type)
|
void FLAC::Picture::setType(FLAC::Picture::Type type) {
|
||||||
{
|
|
||||||
d->type = type;
|
d->type = type;
|
||||||
}
|
}
|
||||||
|
|
||||||
String FLAC::Picture::mimeType() const
|
String FLAC::Picture::mimeType() const {
|
||||||
{
|
|
||||||
return d->mimeType;
|
return d->mimeType;
|
||||||
}
|
}
|
||||||
|
|
||||||
void FLAC::Picture::setMimeType(const String &mimeType)
|
void FLAC::Picture::setMimeType(const String &mimeType) {
|
||||||
{
|
|
||||||
d->mimeType = mimeType;
|
d->mimeType = mimeType;
|
||||||
}
|
}
|
||||||
|
|
||||||
String FLAC::Picture::description() const
|
String FLAC::Picture::description() const {
|
||||||
{
|
|
||||||
return d->description;
|
return d->description;
|
||||||
}
|
}
|
||||||
|
|
||||||
void FLAC::Picture::setDescription(const String &description)
|
void FLAC::Picture::setDescription(const String &description) {
|
||||||
{
|
|
||||||
d->description = description;
|
d->description = description;
|
||||||
}
|
}
|
||||||
|
|
||||||
int FLAC::Picture::width() const
|
int FLAC::Picture::width() const {
|
||||||
{
|
|
||||||
return d->width;
|
return d->width;
|
||||||
}
|
}
|
||||||
|
|
||||||
void FLAC::Picture::setWidth(int width)
|
void FLAC::Picture::setWidth(int width) {
|
||||||
{
|
|
||||||
d->width = width;
|
d->width = width;
|
||||||
}
|
}
|
||||||
|
|
||||||
int FLAC::Picture::height() const
|
int FLAC::Picture::height() const {
|
||||||
{
|
|
||||||
return d->height;
|
return d->height;
|
||||||
}
|
}
|
||||||
|
|
||||||
void FLAC::Picture::setHeight(int height)
|
void FLAC::Picture::setHeight(int height) {
|
||||||
{
|
|
||||||
d->height = height;
|
d->height = height;
|
||||||
}
|
}
|
||||||
|
|
||||||
int FLAC::Picture::colorDepth() const
|
int FLAC::Picture::colorDepth() const {
|
||||||
{
|
|
||||||
return d->colorDepth;
|
return d->colorDepth;
|
||||||
}
|
}
|
||||||
|
|
||||||
void FLAC::Picture::setColorDepth(int colorDepth)
|
void FLAC::Picture::setColorDepth(int colorDepth) {
|
||||||
{
|
|
||||||
d->colorDepth = colorDepth;
|
d->colorDepth = colorDepth;
|
||||||
}
|
}
|
||||||
|
|
||||||
int FLAC::Picture::numColors() const
|
int FLAC::Picture::numColors() const {
|
||||||
{
|
|
||||||
return d->numColors;
|
return d->numColors;
|
||||||
}
|
}
|
||||||
|
|
||||||
void FLAC::Picture::setNumColors(int numColors)
|
void FLAC::Picture::setNumColors(int numColors) {
|
||||||
{
|
|
||||||
d->numColors = numColors;
|
d->numColors = numColors;
|
||||||
}
|
}
|
||||||
|
|
||||||
ByteVector FLAC::Picture::data() const
|
ByteVector FLAC::Picture::data() const {
|
||||||
{
|
|
||||||
return d->data;
|
return d->data;
|
||||||
}
|
}
|
||||||
|
|
||||||
void FLAC::Picture::setData(const ByteVector &data)
|
void FLAC::Picture::setData(const ByteVector &data) {
|
||||||
{
|
|
||||||
d->data = data;
|
d->data = data;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
283
3rdparty/taglib/flac/flacpicture.h
vendored
283
3rdparty/taglib/flac/flacpicture.h
vendored
@@ -35,176 +35,169 @@
|
|||||||
namespace Strawberry_TagLib {
|
namespace Strawberry_TagLib {
|
||||||
namespace TagLib {
|
namespace TagLib {
|
||||||
|
|
||||||
namespace FLAC {
|
namespace FLAC {
|
||||||
|
|
||||||
class TAGLIB_EXPORT Picture : public MetadataBlock
|
class TAGLIB_EXPORT Picture : public MetadataBlock {
|
||||||
{
|
public:
|
||||||
public:
|
/*!
|
||||||
|
* This describes the function or content of the picture.
|
||||||
|
*/
|
||||||
|
enum Type {
|
||||||
|
//! A type not enumerated below
|
||||||
|
Other = 0x00,
|
||||||
|
//! 32x32 PNG image that should be used as the file icon
|
||||||
|
FileIcon = 0x01,
|
||||||
|
//! File icon of a different size or format
|
||||||
|
OtherFileIcon = 0x02,
|
||||||
|
//! Front cover image of the album
|
||||||
|
FrontCover = 0x03,
|
||||||
|
//! Back cover image of the album
|
||||||
|
BackCover = 0x04,
|
||||||
|
//! Inside leaflet page of the album
|
||||||
|
LeafletPage = 0x05,
|
||||||
|
//! Image from the album itself
|
||||||
|
Media = 0x06,
|
||||||
|
//! Picture of the lead artist or soloist
|
||||||
|
LeadArtist = 0x07,
|
||||||
|
//! Picture of the artist or performer
|
||||||
|
Artist = 0x08,
|
||||||
|
//! Picture of the conductor
|
||||||
|
Conductor = 0x09,
|
||||||
|
//! Picture of the band or orchestra
|
||||||
|
Band = 0x0A,
|
||||||
|
//! Picture of the composer
|
||||||
|
Composer = 0x0B,
|
||||||
|
//! Picture of the lyricist or text writer
|
||||||
|
Lyricist = 0x0C,
|
||||||
|
//! Picture of the recording location or studio
|
||||||
|
RecordingLocation = 0x0D,
|
||||||
|
//! Picture of the artists during recording
|
||||||
|
DuringRecording = 0x0E,
|
||||||
|
//! Picture of the artists during performance
|
||||||
|
DuringPerformance = 0x0F,
|
||||||
|
//! Picture from a movie or video related to the track
|
||||||
|
MovieScreenCapture = 0x10,
|
||||||
|
//! Picture of a large, coloured fish
|
||||||
|
ColouredFish = 0x11,
|
||||||
|
//! Illustration related to the track
|
||||||
|
Illustration = 0x12,
|
||||||
|
//! Logo of the band or performer
|
||||||
|
BandLogo = 0x13,
|
||||||
|
//! Logo of the publisher (record company)
|
||||||
|
PublisherLogo = 0x14
|
||||||
|
};
|
||||||
|
|
||||||
/*!
|
explicit Picture();
|
||||||
* This describes the function or content of the picture.
|
explicit Picture(const ByteVector &data);
|
||||||
*/
|
~Picture() override;
|
||||||
enum Type {
|
|
||||||
//! A type not enumerated below
|
|
||||||
Other = 0x00,
|
|
||||||
//! 32x32 PNG image that should be used as the file icon
|
|
||||||
FileIcon = 0x01,
|
|
||||||
//! File icon of a different size or format
|
|
||||||
OtherFileIcon = 0x02,
|
|
||||||
//! Front cover image of the album
|
|
||||||
FrontCover = 0x03,
|
|
||||||
//! Back cover image of the album
|
|
||||||
BackCover = 0x04,
|
|
||||||
//! Inside leaflet page of the album
|
|
||||||
LeafletPage = 0x05,
|
|
||||||
//! Image from the album itself
|
|
||||||
Media = 0x06,
|
|
||||||
//! Picture of the lead artist or soloist
|
|
||||||
LeadArtist = 0x07,
|
|
||||||
//! Picture of the artist or performer
|
|
||||||
Artist = 0x08,
|
|
||||||
//! Picture of the conductor
|
|
||||||
Conductor = 0x09,
|
|
||||||
//! Picture of the band or orchestra
|
|
||||||
Band = 0x0A,
|
|
||||||
//! Picture of the composer
|
|
||||||
Composer = 0x0B,
|
|
||||||
//! Picture of the lyricist or text writer
|
|
||||||
Lyricist = 0x0C,
|
|
||||||
//! Picture of the recording location or studio
|
|
||||||
RecordingLocation = 0x0D,
|
|
||||||
//! Picture of the artists during recording
|
|
||||||
DuringRecording = 0x0E,
|
|
||||||
//! Picture of the artists during performance
|
|
||||||
DuringPerformance = 0x0F,
|
|
||||||
//! Picture from a movie or video related to the track
|
|
||||||
MovieScreenCapture = 0x10,
|
|
||||||
//! Picture of a large, coloured fish
|
|
||||||
ColouredFish = 0x11,
|
|
||||||
//! Illustration related to the track
|
|
||||||
Illustration = 0x12,
|
|
||||||
//! Logo of the band or performer
|
|
||||||
BandLogo = 0x13,
|
|
||||||
//! Logo of the publisher (record company)
|
|
||||||
PublisherLogo = 0x14
|
|
||||||
};
|
|
||||||
|
|
||||||
Picture();
|
/*!
|
||||||
Picture(const ByteVector &data);
|
* Returns the type of the image.
|
||||||
~Picture();
|
*/
|
||||||
|
Type type() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the type of the image.
|
* Sets the type of the image.
|
||||||
*/
|
*/
|
||||||
Type type() const;
|
void setType(Type type);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Sets the type of the image.
|
* Returns the mime type of the image. This should in most cases be "image/png" or "image/jpeg".
|
||||||
*/
|
*/
|
||||||
void setType(Type type);
|
String mimeType() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the mime type of the image. This should in most cases be
|
* Sets the mime type of the image. This should in most cases be "image/png" or "image/jpeg".
|
||||||
* "image/png" or "image/jpeg".
|
*/
|
||||||
*/
|
void setMimeType(const String &m);
|
||||||
String mimeType() const;
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Sets the mime type of the image. This should in most cases be
|
* Returns a text description of the image.
|
||||||
* "image/png" or "image/jpeg".
|
*/
|
||||||
*/
|
|
||||||
void setMimeType(const String &m);
|
|
||||||
|
|
||||||
/*!
|
String description() const;
|
||||||
* Returns a text description of the image.
|
|
||||||
*/
|
|
||||||
|
|
||||||
String description() const;
|
/*!
|
||||||
|
* Sets a textual description of the image to \a desc.
|
||||||
|
*/
|
||||||
|
|
||||||
/*!
|
void setDescription(const String &desc);
|
||||||
* Sets a textual description of the image to \a desc.
|
|
||||||
*/
|
|
||||||
|
|
||||||
void setDescription(const String &desc);
|
/*!
|
||||||
|
* Returns the width of the image.
|
||||||
|
*/
|
||||||
|
int width() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the width of the image.
|
* Sets the width of the image.
|
||||||
*/
|
*/
|
||||||
int width() const;
|
void setWidth(int w);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Sets the width of the image.
|
* Returns the height of the image.
|
||||||
*/
|
*/
|
||||||
void setWidth(int w);
|
int height() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the height of the image.
|
* Sets the height of the image.
|
||||||
*/
|
*/
|
||||||
int height() const;
|
void setHeight(int h);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Sets the height of the image.
|
* Returns the color depth (in bits-per-pixel) of the image.
|
||||||
*/
|
*/
|
||||||
void setHeight(int h);
|
int colorDepth() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the color depth (in bits-per-pixel) of the image.
|
* Sets the color depth (in bits-per-pixel) of the image.
|
||||||
*/
|
*/
|
||||||
int colorDepth() const;
|
void setColorDepth(int depth);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Sets the color depth (in bits-per-pixel) of the image.
|
* Returns the number of colors used on the image..
|
||||||
*/
|
*/
|
||||||
void setColorDepth(int depth);
|
int numColors() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the number of colors used on the image..
|
* Sets the number of colors used on the image (for indexed images).
|
||||||
*/
|
*/
|
||||||
int numColors() const;
|
void setNumColors(int numColors);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Sets the number of colors used on the image (for indexed images).
|
* Returns the image data.
|
||||||
*/
|
*/
|
||||||
void setNumColors(int numColors);
|
ByteVector data() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the image data.
|
* Sets the image data.
|
||||||
*/
|
*/
|
||||||
ByteVector data() const;
|
void setData(const ByteVector &data);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Sets the image data.
|
* Returns the FLAC metadata block type.
|
||||||
*/
|
*/
|
||||||
void setData(const ByteVector &data);
|
int code() const override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the FLAC metadata block type.
|
* Render the content to the FLAC picture block format.
|
||||||
*/
|
*/
|
||||||
int code() const;
|
ByteVector render() const override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Render the content to the FLAC picture block format.
|
* Parse the picture data in the FLAC picture block format.
|
||||||
*/
|
*/
|
||||||
ByteVector render() const;
|
bool parse(const ByteVector &rawData);
|
||||||
|
|
||||||
/*!
|
private:
|
||||||
* Parse the picture data in the FLAC picture block format.
|
Picture(const Picture &item);
|
||||||
*/
|
Picture &operator=(const Picture &item);
|
||||||
bool parse(const ByteVector &rawData);
|
|
||||||
|
|
||||||
private:
|
class PicturePrivate;
|
||||||
Picture(const Picture &item);
|
PicturePrivate *d;
|
||||||
Picture &operator=(const Picture &item);
|
};
|
||||||
|
|
||||||
class PicturePrivate;
|
} // namespace FLAC
|
||||||
PicturePrivate *d;
|
} // namespace TagLib
|
||||||
};
|
} // namespace Strawberry_TagLib
|
||||||
|
|
||||||
typedef List<Picture> PictureList;
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
96
3rdparty/taglib/flac/flacproperties.cpp
vendored
96
3rdparty/taglib/flac/flacproperties.cpp
vendored
@@ -23,24 +23,22 @@
|
|||||||
* http://www.mozilla.org/MPL/ *
|
* http://www.mozilla.org/MPL/ *
|
||||||
***************************************************************************/
|
***************************************************************************/
|
||||||
|
|
||||||
#include <tstring.h>
|
#include "tstring.h"
|
||||||
#include <tdebug.h>
|
#include "tdebug.h"
|
||||||
|
|
||||||
#include "flacproperties.h"
|
#include "flacproperties.h"
|
||||||
#include "flacfile.h"
|
#include "flacfile.h"
|
||||||
|
|
||||||
using namespace Strawberry_TagLib::TagLib;
|
using namespace Strawberry_TagLib::TagLib;
|
||||||
|
|
||||||
class FLAC::Properties::PropertiesPrivate
|
class FLAC::AudioProperties::AudioPropertiesPrivate {
|
||||||
{
|
public:
|
||||||
public:
|
explicit AudioPropertiesPrivate() : length(0),
|
||||||
PropertiesPrivate() :
|
bitrate(0),
|
||||||
length(0),
|
sampleRate(0),
|
||||||
bitrate(0),
|
bitsPerSample(0),
|
||||||
sampleRate(0),
|
channels(0),
|
||||||
bitsPerSample(0),
|
sampleFrames(0) {}
|
||||||
channels(0),
|
|
||||||
sampleFrames(0) {}
|
|
||||||
|
|
||||||
int length;
|
int length;
|
||||||
int bitrate;
|
int bitrate;
|
||||||
@@ -55,72 +53,43 @@ public:
|
|||||||
// public members
|
// public members
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
FLAC::Properties::Properties(ByteVector data, long streamLength, ReadStyle style) :
|
FLAC::AudioProperties::AudioProperties(const ByteVector &data, long long streamLength, ReadStyle) : Strawberry_TagLib::TagLib::AudioProperties(), d(new AudioPropertiesPrivate()) {
|
||||||
AudioProperties(style),
|
|
||||||
d(new PropertiesPrivate())
|
|
||||||
{
|
|
||||||
read(data, streamLength);
|
read(data, streamLength);
|
||||||
}
|
}
|
||||||
|
|
||||||
FLAC::Properties::Properties(File *, ReadStyle style) :
|
FLAC::AudioProperties::~AudioProperties() {
|
||||||
AudioProperties(style),
|
|
||||||
d(new PropertiesPrivate())
|
|
||||||
{
|
|
||||||
debug("FLAC::Properties::Properties() - This constructor is no longer used.");
|
|
||||||
}
|
|
||||||
|
|
||||||
FLAC::Properties::~Properties()
|
|
||||||
{
|
|
||||||
delete d;
|
delete d;
|
||||||
}
|
}
|
||||||
|
|
||||||
int FLAC::Properties::length() const
|
int FLAC::AudioProperties::lengthInSeconds() const {
|
||||||
{
|
|
||||||
return lengthInSeconds();
|
|
||||||
}
|
|
||||||
|
|
||||||
int FLAC::Properties::lengthInSeconds() const
|
|
||||||
{
|
|
||||||
return d->length / 1000;
|
return d->length / 1000;
|
||||||
}
|
}
|
||||||
|
|
||||||
int FLAC::Properties::lengthInMilliseconds() const
|
int FLAC::AudioProperties::lengthInMilliseconds() const {
|
||||||
{
|
|
||||||
return d->length;
|
return d->length;
|
||||||
}
|
}
|
||||||
|
|
||||||
int FLAC::Properties::bitrate() const
|
int FLAC::AudioProperties::bitrate() const {
|
||||||
{
|
|
||||||
return d->bitrate;
|
return d->bitrate;
|
||||||
}
|
}
|
||||||
|
|
||||||
int FLAC::Properties::sampleRate() const
|
int FLAC::AudioProperties::sampleRate() const {
|
||||||
{
|
|
||||||
return d->sampleRate;
|
return d->sampleRate;
|
||||||
}
|
}
|
||||||
|
|
||||||
int FLAC::Properties::bitsPerSample() const
|
int FLAC::AudioProperties::bitsPerSample() const {
|
||||||
{
|
|
||||||
return d->bitsPerSample;
|
return d->bitsPerSample;
|
||||||
}
|
}
|
||||||
|
|
||||||
int FLAC::Properties::sampleWidth() const
|
int FLAC::AudioProperties::channels() const {
|
||||||
{
|
|
||||||
return bitsPerSample();
|
|
||||||
}
|
|
||||||
|
|
||||||
int FLAC::Properties::channels() const
|
|
||||||
{
|
|
||||||
return d->channels;
|
return d->channels;
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned long long FLAC::Properties::sampleFrames() const
|
unsigned long long FLAC::AudioProperties::sampleFrames() const {
|
||||||
{
|
|
||||||
return d->sampleFrames;
|
return d->sampleFrames;
|
||||||
}
|
}
|
||||||
|
|
||||||
ByteVector FLAC::Properties::signature() const
|
ByteVector FLAC::AudioProperties::signature() const {
|
||||||
{
|
|
||||||
return d->signature;
|
return d->signature;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -128,14 +97,14 @@ ByteVector FLAC::Properties::signature() const
|
|||||||
// private members
|
// private members
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
void FLAC::Properties::read(const ByteVector &data, long streamLength)
|
void FLAC::AudioProperties::read(const ByteVector &data, long long streamLength) {
|
||||||
{
|
|
||||||
if(data.size() < 18) {
|
if (data.size() < 18) {
|
||||||
debug("FLAC::Properties::read() - FLAC properties must contain at least 18 bytes.");
|
debug("FLAC::AudioProperties::read() - FLAC properties must contain at least 18 bytes.");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned int pos = 0;
|
size_t pos = 0;
|
||||||
|
|
||||||
// Minimum block size (in samples)
|
// Minimum block size (in samples)
|
||||||
pos += 2;
|
pos += 2;
|
||||||
@@ -149,28 +118,29 @@ void FLAC::Properties::read(const ByteVector &data, long streamLength)
|
|||||||
// Maximum frame size (in bytes)
|
// Maximum frame size (in bytes)
|
||||||
pos += 3;
|
pos += 3;
|
||||||
|
|
||||||
const unsigned int flags = data.toUInt(pos, true);
|
const unsigned int flags = data.toUInt32BE(pos);
|
||||||
pos += 4;
|
pos += 4;
|
||||||
|
|
||||||
d->sampleRate = flags >> 12;
|
d->sampleRate = flags >> 12;
|
||||||
d->channels = ((flags >> 9) & 7) + 1;
|
d->channels = ((flags >> 9) & 7) + 1;
|
||||||
d->bitsPerSample = ((flags >> 4) & 31) + 1;
|
d->bitsPerSample = ((flags >> 4) & 31) + 1;
|
||||||
|
|
||||||
// The last 4 bits are the most significant 4 bits for the 36 bit
|
// The last 4 bits are the most significant 4 bits for the 36 bit
|
||||||
// stream length in samples. (Audio files measured in days)
|
// stream length in samples. (Audio files measured in days)
|
||||||
|
|
||||||
const unsigned long long hi = flags & 0xf;
|
const unsigned long long hi = flags & 0xf;
|
||||||
const unsigned long long lo = data.toUInt(pos, true);
|
const unsigned long long lo = data.toUInt32BE(pos);
|
||||||
pos += 4;
|
pos += 4;
|
||||||
|
|
||||||
d->sampleFrames = (hi << 32) | lo;
|
d->sampleFrames = (hi << 32) | lo;
|
||||||
|
|
||||||
if(d->sampleFrames > 0 && d->sampleRate > 0) {
|
if (d->sampleFrames > 0 && d->sampleRate > 0) {
|
||||||
const double length = d->sampleFrames * 1000.0 / d->sampleRate;
|
const double length = d->sampleFrames * 1000.0 / d->sampleRate;
|
||||||
d->length = static_cast<int>(length + 0.5);
|
d->length = static_cast<int>(length + 0.5);
|
||||||
d->bitrate = static_cast<int>(streamLength * 8.0 / length + 0.5);
|
d->bitrate = static_cast<int>(streamLength * 8.0 / length + 0.5);
|
||||||
}
|
}
|
||||||
|
|
||||||
if(data.size() >= pos + 16)
|
if (data.size() >= pos + 16)
|
||||||
d->signature = data.mid(pos, 16);
|
d->signature = data.mid(pos, 16);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
159
3rdparty/taglib/flac/flacproperties.h
vendored
159
3rdparty/taglib/flac/flacproperties.h
vendored
@@ -32,119 +32,80 @@
|
|||||||
namespace Strawberry_TagLib {
|
namespace Strawberry_TagLib {
|
||||||
namespace TagLib {
|
namespace TagLib {
|
||||||
|
|
||||||
namespace FLAC {
|
namespace FLAC {
|
||||||
|
|
||||||
class File;
|
class File;
|
||||||
|
|
||||||
//! An implementation of audio property reading for FLAC
|
//! An implementation of audio property reading for FLAC
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* This reads the data from an FLAC stream found in the AudioProperties
|
* This reads the data from an FLAC stream found in the AudioProperties API.
|
||||||
* API.
|
*/
|
||||||
*/
|
|
||||||
|
|
||||||
class TAGLIB_EXPORT Properties : public AudioProperties
|
class TAGLIB_EXPORT AudioProperties : public Strawberry_TagLib::TagLib::AudioProperties {
|
||||||
{
|
public:
|
||||||
public:
|
/*!
|
||||||
/*!
|
* Create an instance of FLAC::AudioProperties with the data read from the ByteVector \a data.
|
||||||
* Create an instance of FLAC::Properties with the data read from the
|
*/
|
||||||
* ByteVector \a data.
|
explicit AudioProperties(const ByteVector &data, long long streamLength, ReadStyle style = Average);
|
||||||
*/
|
|
||||||
// BIC: switch to const reference
|
|
||||||
Properties(ByteVector data, long streamLength, ReadStyle style = Average);
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Create an instance of FLAC::Properties with the data read from the
|
* Destroys this FLAC::AudioProperties instance.
|
||||||
* FLAC::File \a file.
|
*/
|
||||||
*/
|
~AudioProperties() override;
|
||||||
// BIC: remove
|
|
||||||
Properties(File *file, ReadStyle style = Average);
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Destroys this FLAC::Properties instance.
|
* Returns the length of the file in seconds. The length is rounded down to the nearest whole second.
|
||||||
*/
|
*
|
||||||
virtual ~Properties();
|
* \see lengthInMilliseconds()
|
||||||
|
*/
|
||||||
|
int lengthInSeconds() const override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the length of the file in seconds. The length is rounded down to
|
* Returns the length of the file in milliseconds.
|
||||||
* the nearest whole second.
|
*
|
||||||
*
|
* \see lengthInSeconds()
|
||||||
* \note This method is just an alias of lengthInSeconds().
|
*/
|
||||||
*
|
int lengthInMilliseconds() const override;
|
||||||
* \deprecated
|
|
||||||
*/
|
|
||||||
TAGLIB_DEPRECATED virtual int length() const;
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the length of the file in seconds. The length is rounded down to
|
* Returns the average bit rate of the file in kb/s.
|
||||||
* the nearest whole second.
|
*/
|
||||||
*
|
int bitrate() const override;
|
||||||
* \see lengthInMilliseconds()
|
|
||||||
*/
|
|
||||||
// BIC: make virtual
|
|
||||||
int lengthInSeconds() const;
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the length of the file in milliseconds.
|
* Returns the sample rate in Hz.
|
||||||
*
|
*/
|
||||||
* \see lengthInSeconds()
|
int sampleRate() const override;
|
||||||
*/
|
|
||||||
// BIC: make virtual
|
|
||||||
int lengthInMilliseconds() const;
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the average bit rate of the file in kb/s.
|
* Returns the number of audio channels.
|
||||||
*/
|
*/
|
||||||
virtual int bitrate() const;
|
int channels() const override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the sample rate in Hz.
|
* Returns the number of bits per audio sample as read from the FLAC identification header.
|
||||||
*/
|
*/
|
||||||
virtual int sampleRate() const;
|
int bitsPerSample() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the number of audio channels.
|
* Return the number of sample frames.
|
||||||
*/
|
*/
|
||||||
virtual int channels() const;
|
unsigned long long sampleFrames() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the number of bits per audio sample as read from the FLAC
|
* Returns the MD5 signature of the uncompressed audio stream as read from the stream info header.
|
||||||
* identification header.
|
*/
|
||||||
*/
|
ByteVector signature() const;
|
||||||
int bitsPerSample() const;
|
|
||||||
|
|
||||||
/*!
|
private:
|
||||||
* Returns the sample width as read from the FLAC identification
|
void read(const ByteVector &data, long long streamLength);
|
||||||
* header.
|
|
||||||
*
|
|
||||||
* \note This method is just an alias of bitsPerSample().
|
|
||||||
*
|
|
||||||
* \deprecated
|
|
||||||
*/
|
|
||||||
TAGLIB_DEPRECATED int sampleWidth() const;
|
|
||||||
|
|
||||||
/*!
|
class AudioPropertiesPrivate;
|
||||||
* Return the number of sample frames.
|
AudioPropertiesPrivate *d;
|
||||||
*/
|
};
|
||||||
unsigned long long sampleFrames() const;
|
} // namespace FLAC
|
||||||
|
} // namespace TagLib
|
||||||
/*!
|
} // namespace Strawberry_TagLib
|
||||||
* Returns the MD5 signature of the uncompressed audio stream as read
|
|
||||||
* from the stream info header.
|
|
||||||
*/
|
|
||||||
ByteVector signature() const;
|
|
||||||
|
|
||||||
private:
|
|
||||||
Properties(const Properties &);
|
|
||||||
Properties &operator=(const Properties &);
|
|
||||||
|
|
||||||
void read(const ByteVector &data, long streamLength);
|
|
||||||
|
|
||||||
class PropertiesPrivate;
|
|
||||||
PropertiesPrivate *d;
|
|
||||||
};
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
@@ -23,56 +23,46 @@
|
|||||||
* http://www.mozilla.org/MPL/ *
|
* http://www.mozilla.org/MPL/ *
|
||||||
***************************************************************************/
|
***************************************************************************/
|
||||||
|
|
||||||
#include <taglib.h>
|
#include "taglib.h"
|
||||||
#include <tdebug.h>
|
#include "tdebug.h"
|
||||||
#include <tstring.h>
|
#include "tstring.h"
|
||||||
#include "flacunknownmetadatablock.h"
|
#include "flacunknownmetadatablock.h"
|
||||||
|
|
||||||
using namespace Strawberry_TagLib::TagLib;
|
using namespace Strawberry_TagLib::TagLib;
|
||||||
|
|
||||||
class FLAC::UnknownMetadataBlock::UnknownMetadataBlockPrivate
|
class FLAC::UnknownMetadataBlock::UnknownMetadataBlockPrivate {
|
||||||
{
|
public:
|
||||||
public:
|
explicit UnknownMetadataBlockPrivate() : code(0) {}
|
||||||
UnknownMetadataBlockPrivate() : code(0) {}
|
|
||||||
|
|
||||||
int code;
|
int code;
|
||||||
ByteVector data;
|
ByteVector data;
|
||||||
};
|
};
|
||||||
|
|
||||||
FLAC::UnknownMetadataBlock::UnknownMetadataBlock(int code, const ByteVector &data) :
|
FLAC::UnknownMetadataBlock::UnknownMetadataBlock(int code, const ByteVector &data) : d(new UnknownMetadataBlockPrivate()) {
|
||||||
d(new UnknownMetadataBlockPrivate())
|
|
||||||
{
|
|
||||||
d->code = code;
|
d->code = code;
|
||||||
d->data = data;
|
d->data = data;
|
||||||
}
|
}
|
||||||
|
|
||||||
FLAC::UnknownMetadataBlock::~UnknownMetadataBlock()
|
FLAC::UnknownMetadataBlock::~UnknownMetadataBlock() {
|
||||||
{
|
|
||||||
delete d;
|
delete d;
|
||||||
}
|
}
|
||||||
|
|
||||||
int FLAC::UnknownMetadataBlock::code() const
|
int FLAC::UnknownMetadataBlock::code() const {
|
||||||
{
|
|
||||||
return d->code;
|
return d->code;
|
||||||
}
|
}
|
||||||
|
|
||||||
void FLAC::UnknownMetadataBlock::setCode(int code)
|
void FLAC::UnknownMetadataBlock::setCode(int code) {
|
||||||
{
|
|
||||||
d->code = code;
|
d->code = code;
|
||||||
}
|
}
|
||||||
|
|
||||||
ByteVector FLAC::UnknownMetadataBlock::data() const
|
ByteVector FLAC::UnknownMetadataBlock::data() const {
|
||||||
{
|
|
||||||
return d->data;
|
return d->data;
|
||||||
}
|
}
|
||||||
|
|
||||||
void FLAC::UnknownMetadataBlock::setData(const ByteVector &data)
|
void FLAC::UnknownMetadataBlock::setData(const ByteVector &data) {
|
||||||
{
|
|
||||||
d->data = data;
|
d->data = data;
|
||||||
}
|
}
|
||||||
|
|
||||||
ByteVector FLAC::UnknownMetadataBlock::render() const
|
ByteVector FLAC::UnknownMetadataBlock::render() const {
|
||||||
{
|
|
||||||
return d->data;
|
return d->data;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
71
3rdparty/taglib/flac/flacunknownmetadatablock.h
vendored
71
3rdparty/taglib/flac/flacunknownmetadatablock.h
vendored
@@ -33,51 +33,48 @@
|
|||||||
|
|
||||||
namespace Strawberry_TagLib {
|
namespace Strawberry_TagLib {
|
||||||
namespace TagLib {
|
namespace TagLib {
|
||||||
|
namespace FLAC {
|
||||||
|
|
||||||
namespace FLAC {
|
class TAGLIB_EXPORT UnknownMetadataBlock : public MetadataBlock {
|
||||||
|
public:
|
||||||
|
explicit UnknownMetadataBlock(int code, const ByteVector &data);
|
||||||
|
~UnknownMetadataBlock() override;
|
||||||
|
|
||||||
class TAGLIB_EXPORT UnknownMetadataBlock : public MetadataBlock
|
/*!
|
||||||
{
|
* Returns the FLAC metadata block type.
|
||||||
public:
|
*/
|
||||||
UnknownMetadataBlock(int blockType, const ByteVector &data);
|
int code() const override;
|
||||||
~UnknownMetadataBlock();
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the FLAC metadata block type.
|
* Sets the FLAC metadata block type.
|
||||||
*/
|
*/
|
||||||
int code() const;
|
void setCode(int code);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Sets the FLAC metadata block type.
|
* Returns the FLAC metadata block type.
|
||||||
*/
|
*/
|
||||||
void setCode(int code);
|
ByteVector data() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the FLAC metadata block type.
|
* Sets the FLAC metadata block type.
|
||||||
*/
|
*/
|
||||||
ByteVector data() const;
|
void setData(const ByteVector &data);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Sets the FLAC metadata block type.
|
* Render the content of the block.
|
||||||
*/
|
*/
|
||||||
void setData(const ByteVector &data);
|
ByteVector render() const override;
|
||||||
|
|
||||||
/*!
|
private:
|
||||||
* Render the content of the block.
|
explicit UnknownMetadataBlock(const MetadataBlock &item);
|
||||||
*/
|
UnknownMetadataBlock &operator=(const MetadataBlock &item);
|
||||||
ByteVector render() const;
|
|
||||||
|
|
||||||
private:
|
class UnknownMetadataBlockPrivate;
|
||||||
UnknownMetadataBlock(const MetadataBlock &item);
|
UnknownMetadataBlockPrivate *d;
|
||||||
UnknownMetadataBlock &operator=(const MetadataBlock &item);
|
};
|
||||||
|
|
||||||
class UnknownMetadataBlockPrivate;
|
} // namespace FLAC
|
||||||
UnknownMetadataBlockPrivate *d;
|
} // namespace TagLib
|
||||||
};
|
} // namespace Strawberry_TagLib
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
141
3rdparty/taglib/it/itfile.cpp
vendored
141
3rdparty/taglib/it/itfile.cpp
vendored
@@ -33,65 +33,43 @@
|
|||||||
using namespace Strawberry_TagLib::TagLib;
|
using namespace Strawberry_TagLib::TagLib;
|
||||||
using namespace IT;
|
using namespace IT;
|
||||||
|
|
||||||
class IT::File::FilePrivate
|
class IT::File::FilePrivate {
|
||||||
{
|
public:
|
||||||
public:
|
explicit FilePrivate(AudioProperties::ReadStyle propertiesStyle) : properties(propertiesStyle) {}
|
||||||
explicit FilePrivate(AudioProperties::ReadStyle propertiesStyle)
|
|
||||||
: tag(), properties(propertiesStyle)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
Mod::Tag tag;
|
Mod::Tag tag;
|
||||||
IT::Properties properties;
|
IT::AudioProperties properties;
|
||||||
};
|
};
|
||||||
|
|
||||||
IT::File::File(FileName file, bool readProperties,
|
IT::File::File(FileName file, bool readProperties, AudioProperties::ReadStyle propertiesStyle) : Mod::FileBase(file), d(new FilePrivate(propertiesStyle)) {
|
||||||
AudioProperties::ReadStyle propertiesStyle) :
|
|
||||||
Mod::FileBase(file),
|
if (isOpen())
|
||||||
d(new FilePrivate(propertiesStyle))
|
|
||||||
{
|
|
||||||
if(isOpen())
|
|
||||||
read(readProperties);
|
read(readProperties);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
IT::File::File(IOStream *stream, bool readProperties,
|
IT::File::File(IOStream *stream, bool readProperties, AudioProperties::ReadStyle propertiesStyle) : Mod::FileBase(stream), d(new FilePrivate(propertiesStyle)) {
|
||||||
AudioProperties::ReadStyle propertiesStyle) :
|
|
||||||
Mod::FileBase(stream),
|
if (isOpen())
|
||||||
d(new FilePrivate(propertiesStyle))
|
|
||||||
{
|
|
||||||
if(isOpen())
|
|
||||||
read(readProperties);
|
read(readProperties);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
IT::File::~File()
|
IT::File::~File() {
|
||||||
{
|
|
||||||
delete d;
|
delete d;
|
||||||
}
|
}
|
||||||
|
|
||||||
Mod::Tag *IT::File::tag() const
|
Mod::Tag *IT::File::tag() const {
|
||||||
{
|
|
||||||
return &d->tag;
|
return &d->tag;
|
||||||
}
|
}
|
||||||
|
|
||||||
PropertyMap IT::File::properties() const
|
IT::AudioProperties *IT::File::audioProperties() const {
|
||||||
{
|
|
||||||
return d->tag.properties();
|
|
||||||
}
|
|
||||||
|
|
||||||
PropertyMap IT::File::setProperties(const PropertyMap &properties)
|
|
||||||
{
|
|
||||||
return d->tag.setProperties(properties);
|
|
||||||
}
|
|
||||||
|
|
||||||
IT::Properties *IT::File::audioProperties() const
|
|
||||||
{
|
|
||||||
return &d->properties;
|
return &d->properties;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool IT::File::save()
|
bool IT::File::save() {
|
||||||
{
|
|
||||||
if(readOnly())
|
if (readOnly()) {
|
||||||
{
|
|
||||||
debug("IT::File::save() - Cannot save to a read only file.");
|
debug("IT::File::save() - Cannot save to a read only file.");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@@ -105,37 +83,37 @@ bool IT::File::save()
|
|||||||
unsigned short instrumentCount = 0;
|
unsigned short instrumentCount = 0;
|
||||||
unsigned short sampleCount = 0;
|
unsigned short sampleCount = 0;
|
||||||
|
|
||||||
if(!readU16L(length) || !readU16L(instrumentCount) || !readU16L(sampleCount))
|
if (!readU16L(length) || !readU16L(instrumentCount) || !readU16L(sampleCount))
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
seek(15, Current);
|
seek(15, Current);
|
||||||
|
|
||||||
// write comment as instrument and sample names:
|
// write comment as instrument and sample names:
|
||||||
StringList lines = d->tag.comment().split("\n");
|
StringList lines = d->tag.comment().split("\n");
|
||||||
for(unsigned short i = 0; i < instrumentCount; ++ i) {
|
for (unsigned short i = 0; i < instrumentCount; ++i) {
|
||||||
seek(192L + length + ((long)i << 2));
|
seek(192L + length + (static_cast<long>(i) << 2));
|
||||||
unsigned long instrumentOffset = 0;
|
unsigned int instrumentOffset = 0;
|
||||||
if(!readU32L(instrumentOffset))
|
if (!readU32L(instrumentOffset))
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
seek(instrumentOffset + 32);
|
seek(instrumentOffset + 32);
|
||||||
|
|
||||||
if(i < lines.size())
|
if (i < lines.size())
|
||||||
writeString(lines[i], 25);
|
writeString(lines[i], 25);
|
||||||
else
|
else
|
||||||
writeString(String(), 25);
|
writeString(String(), 25);
|
||||||
writeByte(0);
|
writeByte(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
for(unsigned short i = 0; i < sampleCount; ++ i) {
|
for (unsigned short i = 0; i < sampleCount; ++i) {
|
||||||
seek(192L + length + ((long)instrumentCount << 2) + ((long)i << 2));
|
seek(192L + length + (static_cast<long>(instrumentCount) << 2) + (static_cast<long>(i) << 2));
|
||||||
unsigned long sampleOffset = 0;
|
unsigned int sampleOffset = 0;
|
||||||
if(!readU32L(sampleOffset))
|
if (!readU32L(sampleOffset))
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
seek(sampleOffset + 20);
|
seek(sampleOffset + 20);
|
||||||
|
|
||||||
if((unsigned int)(i + instrumentCount) < lines.size())
|
if (static_cast<unsigned int>(i + instrumentCount) < lines.size())
|
||||||
writeString(lines[i + instrumentCount], 25);
|
writeString(lines[i + instrumentCount], 25);
|
||||||
else
|
else
|
||||||
writeString(String(), 25);
|
writeString(String(), 25);
|
||||||
@@ -144,41 +122,40 @@ bool IT::File::save()
|
|||||||
|
|
||||||
// write rest as message:
|
// write rest as message:
|
||||||
StringList messageLines;
|
StringList messageLines;
|
||||||
for(unsigned int i = instrumentCount + sampleCount; i < lines.size(); ++ i)
|
for (unsigned int i = instrumentCount + sampleCount; i < lines.size(); ++i)
|
||||||
messageLines.append(lines[i]);
|
messageLines.append(lines[i]);
|
||||||
ByteVector message = messageLines.toString("\r").data(String::Latin1);
|
ByteVector message = messageLines.toString("\r").data(String::Latin1);
|
||||||
|
|
||||||
// it's actually not really stated if the message needs a
|
// it's actually not really stated if the message needs a
|
||||||
// terminating NUL but it does not hurt to add one:
|
// terminating NUL but it does not hurt to add one:
|
||||||
if(message.size() > 7999)
|
if (message.size() > 7999)
|
||||||
message.resize(7999);
|
message.resize(7999);
|
||||||
message.append((char)0);
|
message.append(static_cast<char>(0));
|
||||||
|
|
||||||
unsigned short special = 0;
|
unsigned short special = 0;
|
||||||
unsigned short messageLength = 0;
|
unsigned short messageLength = 0;
|
||||||
unsigned long messageOffset = 0;
|
unsigned int messageOffset = 0;
|
||||||
|
|
||||||
seek(46);
|
seek(46);
|
||||||
if(!readU16L(special))
|
if (!readU16L(special))
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
unsigned long fileSize = File::length();
|
unsigned int fileSize = File::length();
|
||||||
if(special & Properties::MessageAttached) {
|
if (special & AudioProperties::MessageAttached) {
|
||||||
seek(54);
|
seek(54);
|
||||||
if(!readU16L(messageLength) || !readU32L(messageOffset))
|
if (!readU16L(messageLength) || !readU32L(messageOffset))
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
if(messageLength == 0)
|
if (messageLength == 0)
|
||||||
messageOffset = fileSize;
|
messageOffset = fileSize;
|
||||||
}
|
}
|
||||||
else
|
else {
|
||||||
{
|
|
||||||
messageOffset = fileSize;
|
messageOffset = fileSize;
|
||||||
seek(46);
|
seek(46);
|
||||||
writeU16L(special | 0x1);
|
writeU16L(special | 0x1);
|
||||||
}
|
}
|
||||||
|
|
||||||
if(messageOffset + messageLength >= fileSize) {
|
if (messageOffset + messageLength >= fileSize) {
|
||||||
// append new message
|
// append new message
|
||||||
seek(54);
|
seek(54);
|
||||||
writeU16L(message.size());
|
writeU16L(message.size());
|
||||||
@@ -199,9 +176,8 @@ bool IT::File::save()
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
void IT::File::read(bool)
|
void IT::File::read(bool) {
|
||||||
{
|
if (!isOpen())
|
||||||
if(!isOpen())
|
|
||||||
return;
|
return;
|
||||||
|
|
||||||
seek(0);
|
seek(0);
|
||||||
@@ -233,14 +209,14 @@ void IT::File::read(bool)
|
|||||||
// sample/instrument names are abused as comments so
|
// sample/instrument names are abused as comments so
|
||||||
// I just add all together.
|
// I just add all together.
|
||||||
String message;
|
String message;
|
||||||
if(special & Properties::MessageAttached) {
|
if (special & AudioProperties::MessageAttached) {
|
||||||
READ_U16L_AS(messageLength);
|
READ_U16L_AS(messageLength);
|
||||||
READ_U32L_AS(messageOffset);
|
READ_U32L_AS(messageOffset);
|
||||||
seek(messageOffset);
|
seek(messageOffset);
|
||||||
ByteVector messageBytes = readBlock(messageLength);
|
ByteVector messageBytes = readBlock(messageLength);
|
||||||
READ_ASSERT(messageBytes.size() == messageLength);
|
READ_ASSERT(messageBytes.size() == messageLength);
|
||||||
int index = messageBytes.find((char) 0);
|
const size_t index = messageBytes.find(static_cast<char>(0));
|
||||||
if(index > -1)
|
if (index != ByteVector::npos())
|
||||||
messageBytes.resize(index, 0);
|
messageBytes.resize(index, 0);
|
||||||
messageBytes.replace('\r', '\n');
|
messageBytes.replace('\r', '\n');
|
||||||
message = messageBytes;
|
message = messageBytes;
|
||||||
@@ -249,26 +225,26 @@ void IT::File::read(bool)
|
|||||||
seek(64);
|
seek(64);
|
||||||
|
|
||||||
ByteVector pannings = readBlock(64);
|
ByteVector pannings = readBlock(64);
|
||||||
ByteVector volumes = readBlock(64);
|
ByteVector volumes = readBlock(64);
|
||||||
READ_ASSERT(pannings.size() == 64 && volumes.size() == 64);
|
READ_ASSERT(pannings.size() == 64 && volumes.size() == 64);
|
||||||
int channels = 0;
|
int channels = 0;
|
||||||
for(int i = 0; i < 64; ++ i) {
|
for (int i = 0; i < 64; ++i) {
|
||||||
// Strictly speaking an IT file has always 64 channels, but
|
// Strictly speaking an IT file has always 64 channels, but
|
||||||
// I don't count disabled and muted channels.
|
// I don't count disabled and muted channels.
|
||||||
// But this always gives 64 channels for all my files anyway.
|
// But this always gives 64 channels for all my files anyway.
|
||||||
// Strangely VLC does report other values. I wonder how VLC
|
// Strangely VLC does report other values. I wonder how VLC
|
||||||
// gets it's values.
|
// gets it's values.
|
||||||
if((unsigned char) pannings[i] < 128 && volumes[i] > 0)
|
if (static_cast<unsigned char>(pannings[i]) < 128 && volumes[i] > 0)
|
||||||
++channels;
|
++channels;
|
||||||
}
|
}
|
||||||
d->properties.setChannels(channels);
|
d->properties.setChannels(channels);
|
||||||
|
|
||||||
// real length might be shorter because of skips and terminator
|
// real length might be shorter because of skips and terminator
|
||||||
unsigned short realLength = 0;
|
unsigned short realLength = 0;
|
||||||
for(unsigned short i = 0; i < length; ++ i) {
|
for (unsigned short i = 0; i < length; ++i) {
|
||||||
READ_BYTE_AS(order);
|
READ_BYTE_AS(order);
|
||||||
if(order == 255) break;
|
if (order == 255) break;
|
||||||
if(order != 254) ++ realLength;
|
if (order != 254) ++realLength;
|
||||||
}
|
}
|
||||||
d->properties.setLengthInPatterns(realLength);
|
d->properties.setLengthInPatterns(realLength);
|
||||||
|
|
||||||
@@ -279,8 +255,8 @@ void IT::File::read(bool)
|
|||||||
// Currently I just discard anything after a nil, but
|
// Currently I just discard anything after a nil, but
|
||||||
// e.g. VLC seems to interpret a nil as a space. I
|
// e.g. VLC seems to interpret a nil as a space. I
|
||||||
// don't know what is the proper behaviour.
|
// don't know what is the proper behaviour.
|
||||||
for(unsigned short i = 0; i < instrumentCount; ++ i) {
|
for (unsigned short i = 0; i < instrumentCount; ++i) {
|
||||||
seek(192L + length + ((long)i << 2));
|
seek(192L + length + (static_cast<long>(i) << 2));
|
||||||
READ_U32L_AS(instrumentOffset);
|
READ_U32L_AS(instrumentOffset);
|
||||||
seek(instrumentOffset);
|
seek(instrumentOffset);
|
||||||
|
|
||||||
@@ -295,8 +271,8 @@ void IT::File::read(bool)
|
|||||||
comment.append(instrumentName);
|
comment.append(instrumentName);
|
||||||
}
|
}
|
||||||
|
|
||||||
for(unsigned short i = 0; i < sampleCount; ++ i) {
|
for (unsigned short i = 0; i < sampleCount; ++i) {
|
||||||
seek(192L + length + ((long)instrumentCount << 2) + ((long)i << 2));
|
seek(192L + length + (static_cast<long>(instrumentCount) << 2) + (static_cast<long>(i) << 2));
|
||||||
READ_U32L_AS(sampleOffset);
|
READ_U32L_AS(sampleOffset);
|
||||||
|
|
||||||
seek(sampleOffset);
|
seek(sampleOffset);
|
||||||
@@ -328,8 +304,9 @@ void IT::File::read(bool)
|
|||||||
comment.append(sampleName);
|
comment.append(sampleName);
|
||||||
}
|
}
|
||||||
|
|
||||||
if(message.size() > 0)
|
if (message.size() > 0)
|
||||||
comment.append(message);
|
comment.append(message);
|
||||||
d->tag.setComment(comment.toString("\n"));
|
d->tag.setComment(comment.toString("\n"));
|
||||||
d->tag.setTrackerName("Impulse Tracker");
|
d->tag.setTrackerName("Impulse Tracker");
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
112
3rdparty/taglib/it/itfile.h
vendored
112
3rdparty/taglib/it/itfile.h
vendored
@@ -31,81 +31,61 @@
|
|||||||
|
|
||||||
namespace Strawberry_TagLib {
|
namespace Strawberry_TagLib {
|
||||||
namespace TagLib {
|
namespace TagLib {
|
||||||
|
namespace IT {
|
||||||
|
|
||||||
namespace IT {
|
class TAGLIB_EXPORT File : public Mod::FileBase {
|
||||||
|
public:
|
||||||
|
/*!
|
||||||
|
* Constructs a Impulse Tracker file from \a file.
|
||||||
|
*
|
||||||
|
* \note In the current implementation, both \a readProperties and \a propertiesStyle are ignored.
|
||||||
|
* The audio properties are always read.
|
||||||
|
*/
|
||||||
|
explicit File(FileName file, bool readProperties = true, AudioProperties::ReadStyle propertiesStyle = AudioProperties::Average);
|
||||||
|
|
||||||
class TAGLIB_EXPORT File : public Mod::FileBase {
|
/*!
|
||||||
public:
|
* Constructs a Impulse Tracker file from \a stream.
|
||||||
/*!
|
*
|
||||||
* Constructs a Impulse Tracker file from \a file.
|
* \note In the current implementation, both \a readProperties and \a propertiesStyle are ignored.
|
||||||
*
|
* The audio properties are always read.
|
||||||
* \note In the current implementation, both \a readProperties and
|
*
|
||||||
* \a propertiesStyle are ignored. The audio properties are always
|
* \note TagLib will *not* take ownership of the stream, the caller is responsible for deleting it after the File object.
|
||||||
* read.
|
*/
|
||||||
*/
|
explicit File(IOStream *stream, bool readProperties = true, AudioProperties::ReadStyle propertiesStyle = AudioProperties::Average);
|
||||||
File(FileName file, bool readProperties = true,
|
|
||||||
AudioProperties::ReadStyle propertiesStyle =
|
|
||||||
AudioProperties::Average);
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Constructs a Impulse Tracker file from \a stream.
|
* Destroys this instance of the File.
|
||||||
*
|
*/
|
||||||
* \note In the current implementation, both \a readProperties and
|
~File() override;
|
||||||
* \a propertiesStyle are ignored. The audio properties are always
|
|
||||||
* read.
|
|
||||||
*
|
|
||||||
* \note TagLib will *not* take ownership of the stream, the caller is
|
|
||||||
* responsible for deleting it after the File object.
|
|
||||||
*/
|
|
||||||
File(IOStream *stream, bool readProperties = true,
|
|
||||||
AudioProperties::ReadStyle propertiesStyle =
|
|
||||||
AudioProperties::Average);
|
|
||||||
|
|
||||||
/*!
|
Mod::Tag *tag() const override;
|
||||||
* Destroys this instance of the File.
|
|
||||||
*/
|
|
||||||
virtual ~File();
|
|
||||||
|
|
||||||
Mod::Tag *tag() const;
|
/*!
|
||||||
|
* Returns the IT::AudioProperties for this file. If no audio properties
|
||||||
|
* were read then this will return a null pointer.
|
||||||
|
*/
|
||||||
|
IT::AudioProperties *audioProperties() const override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Forwards to Mod::Tag::properties().
|
* Save the file.
|
||||||
* BIC: will be removed once File::toDict() is made virtual
|
* This is the same as calling save(AllTags);
|
||||||
*/
|
*
|
||||||
PropertyMap properties() const;
|
* \note Saving Impulse Tracker tags is not supported.
|
||||||
|
*/
|
||||||
|
bool save() override;
|
||||||
|
|
||||||
/*!
|
private:
|
||||||
* Forwards to Mod::Tag::setProperties().
|
File(const File&);
|
||||||
* BIC: will be removed once File::setProperties() is made virtual
|
File &operator=(const File&);
|
||||||
*/
|
|
||||||
PropertyMap setProperties(const PropertyMap &);
|
|
||||||
|
|
||||||
/*!
|
void read(bool readProperties);
|
||||||
* Returns the IT::Properties for this file. If no audio properties
|
|
||||||
* were read then this will return a null pointer.
|
|
||||||
*/
|
|
||||||
IT::Properties *audioProperties() const;
|
|
||||||
|
|
||||||
/*!
|
class FilePrivate;
|
||||||
* Save the file.
|
FilePrivate *d;
|
||||||
* This is the same as calling save(AllTags);
|
};
|
||||||
*
|
|
||||||
* \note Saving Impulse Tracker tags is not supported.
|
|
||||||
*/
|
|
||||||
bool save();
|
|
||||||
|
|
||||||
|
} // namespace IT
|
||||||
private:
|
} // namespace TagLib
|
||||||
File(const File &);
|
} // namespace Strawberry_TagLib
|
||||||
File &operator=(const File &);
|
|
||||||
|
|
||||||
void read(bool readProperties);
|
|
||||||
|
|
||||||
class FilePrivate;
|
|
||||||
FilePrivate *d;
|
|
||||||
};
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
174
3rdparty/taglib/it/itproperties.cpp
vendored
174
3rdparty/taglib/it/itproperties.cpp
vendored
@@ -29,29 +29,26 @@
|
|||||||
using namespace Strawberry_TagLib::TagLib;
|
using namespace Strawberry_TagLib::TagLib;
|
||||||
using namespace IT;
|
using namespace IT;
|
||||||
|
|
||||||
class IT::Properties::PropertiesPrivate
|
class IT::AudioProperties::AudioPropertiesPrivate {
|
||||||
{
|
public:
|
||||||
public:
|
explicit AudioPropertiesPrivate() : channels(0),
|
||||||
PropertiesPrivate() :
|
lengthInPatterns(0),
|
||||||
channels(0),
|
instrumentCount(0),
|
||||||
lengthInPatterns(0),
|
sampleCount(0),
|
||||||
instrumentCount(0),
|
patternCount(0),
|
||||||
sampleCount(0),
|
version(0),
|
||||||
patternCount(0),
|
compatibleVersion(0),
|
||||||
version(0),
|
flags(0),
|
||||||
compatibleVersion(0),
|
special(0),
|
||||||
flags(0),
|
globalVolume(0),
|
||||||
special(0),
|
mixVolume(0),
|
||||||
globalVolume(0),
|
tempo(0),
|
||||||
mixVolume(0),
|
bpmSpeed(0),
|
||||||
tempo(0),
|
panningSeparation(0),
|
||||||
bpmSpeed(0),
|
pitchWheelDepth(0) {
|
||||||
panningSeparation(0),
|
|
||||||
pitchWheelDepth(0)
|
|
||||||
{
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int channels;
|
int channels;
|
||||||
unsigned short lengthInPatterns;
|
unsigned short lengthInPatterns;
|
||||||
unsigned short instrumentCount;
|
unsigned short instrumentCount;
|
||||||
unsigned short sampleCount;
|
unsigned short sampleCount;
|
||||||
@@ -60,201 +57,160 @@ public:
|
|||||||
unsigned short compatibleVersion;
|
unsigned short compatibleVersion;
|
||||||
unsigned short flags;
|
unsigned short flags;
|
||||||
unsigned short special;
|
unsigned short special;
|
||||||
unsigned char globalVolume;
|
unsigned char globalVolume;
|
||||||
unsigned char mixVolume;
|
unsigned char mixVolume;
|
||||||
unsigned char tempo;
|
unsigned char tempo;
|
||||||
unsigned char bpmSpeed;
|
unsigned char bpmSpeed;
|
||||||
unsigned char panningSeparation;
|
unsigned char panningSeparation;
|
||||||
unsigned char pitchWheelDepth;
|
unsigned char pitchWheelDepth;
|
||||||
};
|
};
|
||||||
|
|
||||||
IT::Properties::Properties(AudioProperties::ReadStyle propertiesStyle) :
|
IT::AudioProperties::AudioProperties(AudioProperties::ReadStyle) : Strawberry_TagLib::TagLib::AudioProperties(), d(new AudioPropertiesPrivate()) {}
|
||||||
AudioProperties(propertiesStyle),
|
|
||||||
d(new PropertiesPrivate())
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
IT::Properties::~Properties()
|
IT::AudioProperties::~AudioProperties() {
|
||||||
{
|
|
||||||
delete d;
|
delete d;
|
||||||
}
|
}
|
||||||
|
|
||||||
int IT::Properties::length() const
|
int IT::AudioProperties::lengthInSeconds() const {
|
||||||
{
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
int IT::Properties::lengthInSeconds() const
|
int IT::AudioProperties::lengthInMilliseconds() const {
|
||||||
{
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
int IT::Properties::lengthInMilliseconds() const
|
int IT::AudioProperties::bitrate() const {
|
||||||
{
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
int IT::Properties::bitrate() const
|
int IT::AudioProperties::sampleRate() const {
|
||||||
{
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
int IT::Properties::sampleRate() const
|
int IT::AudioProperties::channels() const {
|
||||||
{
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
int IT::Properties::channels() const
|
|
||||||
{
|
|
||||||
return d->channels;
|
return d->channels;
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned short IT::Properties::lengthInPatterns() const
|
unsigned short IT::AudioProperties::lengthInPatterns() const {
|
||||||
{
|
|
||||||
return d->lengthInPatterns;
|
return d->lengthInPatterns;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool IT::Properties::stereo() const
|
bool IT::AudioProperties::stereo() const {
|
||||||
{
|
|
||||||
return d->flags & Stereo;
|
return d->flags & Stereo;
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned short IT::Properties::instrumentCount() const
|
unsigned short IT::AudioProperties::instrumentCount() const {
|
||||||
{
|
|
||||||
return d->instrumentCount;
|
return d->instrumentCount;
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned short IT::Properties::sampleCount() const
|
unsigned short IT::AudioProperties::sampleCount() const {
|
||||||
{
|
|
||||||
return d->sampleCount;
|
return d->sampleCount;
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned short IT::Properties::patternCount() const
|
unsigned short IT::AudioProperties::patternCount() const {
|
||||||
{
|
|
||||||
return d->patternCount;
|
return d->patternCount;
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned short IT::Properties::version() const
|
unsigned short IT::AudioProperties::version() const {
|
||||||
{
|
|
||||||
return d->version;
|
return d->version;
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned short IT::Properties::compatibleVersion() const
|
unsigned short IT::AudioProperties::compatibleVersion() const {
|
||||||
{
|
|
||||||
return d->compatibleVersion;
|
return d->compatibleVersion;
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned short IT::Properties::flags() const
|
unsigned short IT::AudioProperties::flags() const {
|
||||||
{
|
|
||||||
return d->flags;
|
return d->flags;
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned short IT::Properties::special() const
|
unsigned short IT::AudioProperties::special() const {
|
||||||
{
|
|
||||||
return d->special;
|
return d->special;
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned char IT::Properties::globalVolume() const
|
unsigned char IT::AudioProperties::globalVolume() const {
|
||||||
{
|
|
||||||
return d->globalVolume;
|
return d->globalVolume;
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned char IT::Properties::mixVolume() const
|
unsigned char IT::AudioProperties::mixVolume() const {
|
||||||
{
|
|
||||||
return d->mixVolume;
|
return d->mixVolume;
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned char IT::Properties::tempo() const
|
unsigned char IT::AudioProperties::tempo() const {
|
||||||
{
|
|
||||||
return d->tempo;
|
return d->tempo;
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned char IT::Properties::bpmSpeed() const
|
unsigned char IT::AudioProperties::bpmSpeed() const {
|
||||||
{
|
|
||||||
return d->bpmSpeed;
|
return d->bpmSpeed;
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned char IT::Properties::panningSeparation() const
|
unsigned char IT::AudioProperties::panningSeparation() const {
|
||||||
{
|
|
||||||
return d->panningSeparation;
|
return d->panningSeparation;
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned char IT::Properties::pitchWheelDepth() const
|
unsigned char IT::AudioProperties::pitchWheelDepth() const {
|
||||||
{
|
|
||||||
return d->pitchWheelDepth;
|
return d->pitchWheelDepth;
|
||||||
}
|
}
|
||||||
|
|
||||||
void IT::Properties::setChannels(int channels)
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
{
|
// private members
|
||||||
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
|
void IT::AudioProperties::setChannels(int channels) {
|
||||||
d->channels = channels;
|
d->channels = channels;
|
||||||
}
|
}
|
||||||
|
|
||||||
void IT::Properties::setLengthInPatterns(unsigned short lengthInPatterns)
|
void IT::AudioProperties::setLengthInPatterns(unsigned short lengthInPatterns) {
|
||||||
{
|
|
||||||
d->lengthInPatterns = lengthInPatterns;
|
d->lengthInPatterns = lengthInPatterns;
|
||||||
}
|
}
|
||||||
|
|
||||||
void IT::Properties::setInstrumentCount(unsigned short instrumentCount)
|
void IT::AudioProperties::setInstrumentCount(unsigned short instrumentCount) {
|
||||||
{
|
|
||||||
d->instrumentCount = instrumentCount;
|
d->instrumentCount = instrumentCount;
|
||||||
}
|
}
|
||||||
|
|
||||||
void IT::Properties::setSampleCount(unsigned short sampleCount)
|
void IT::AudioProperties::setSampleCount(unsigned short sampleCount) {
|
||||||
{
|
|
||||||
d->sampleCount = sampleCount;
|
d->sampleCount = sampleCount;
|
||||||
}
|
}
|
||||||
|
|
||||||
void IT::Properties::setPatternCount(unsigned short patternCount)
|
void IT::AudioProperties::setPatternCount(unsigned short patternCount) {
|
||||||
{
|
|
||||||
d->patternCount = patternCount;
|
d->patternCount = patternCount;
|
||||||
}
|
}
|
||||||
|
|
||||||
void IT::Properties::setFlags(unsigned short flags)
|
void IT::AudioProperties::setFlags(unsigned short flags) {
|
||||||
{
|
|
||||||
d->flags = flags;
|
d->flags = flags;
|
||||||
}
|
}
|
||||||
|
|
||||||
void IT::Properties::setSpecial(unsigned short special)
|
void IT::AudioProperties::setSpecial(unsigned short special) {
|
||||||
{
|
|
||||||
d->special = special;
|
d->special = special;
|
||||||
}
|
}
|
||||||
|
|
||||||
void IT::Properties::setCompatibleVersion(unsigned short compatibleVersion)
|
void IT::AudioProperties::setCompatibleVersion(unsigned short compatibleVersion) {
|
||||||
{
|
|
||||||
d->compatibleVersion = compatibleVersion;
|
d->compatibleVersion = compatibleVersion;
|
||||||
}
|
}
|
||||||
|
|
||||||
void IT::Properties::setVersion(unsigned short version)
|
void IT::AudioProperties::setVersion(unsigned short version) {
|
||||||
{
|
|
||||||
d->version = version;
|
d->version = version;
|
||||||
}
|
}
|
||||||
|
|
||||||
void IT::Properties::setGlobalVolume(unsigned char globalVolume)
|
void IT::AudioProperties::setGlobalVolume(unsigned char globalVolume) {
|
||||||
{
|
|
||||||
d->globalVolume = globalVolume;
|
d->globalVolume = globalVolume;
|
||||||
}
|
}
|
||||||
|
|
||||||
void IT::Properties::setMixVolume(unsigned char mixVolume)
|
void IT::AudioProperties::setMixVolume(unsigned char mixVolume) {
|
||||||
{
|
|
||||||
d->mixVolume = mixVolume;
|
d->mixVolume = mixVolume;
|
||||||
}
|
}
|
||||||
|
|
||||||
void IT::Properties::setTempo(unsigned char tempo)
|
void IT::AudioProperties::setTempo(unsigned char tempo) {
|
||||||
{
|
|
||||||
d->tempo = tempo;
|
d->tempo = tempo;
|
||||||
}
|
}
|
||||||
|
|
||||||
void IT::Properties::setBpmSpeed(unsigned char bpmSpeed)
|
void IT::AudioProperties::setBpmSpeed(unsigned char bpmSpeed) {
|
||||||
{
|
|
||||||
d->bpmSpeed = bpmSpeed;
|
d->bpmSpeed = bpmSpeed;
|
||||||
}
|
}
|
||||||
|
|
||||||
void IT::Properties::setPanningSeparation(unsigned char panningSeparation)
|
void IT::AudioProperties::setPanningSeparation(unsigned char panningSeparation) {
|
||||||
{
|
|
||||||
d->panningSeparation = panningSeparation;
|
d->panningSeparation = panningSeparation;
|
||||||
}
|
}
|
||||||
|
|
||||||
void IT::Properties::setPitchWheelDepth(unsigned char pitchWheelDepth)
|
void IT::AudioProperties::setPitchWheelDepth(unsigned char pitchWheelDepth) {
|
||||||
{
|
|
||||||
d->pitchWheelDepth = pitchWheelDepth;
|
d->pitchWheelDepth = pitchWheelDepth;
|
||||||
}
|
}
|
||||||
|
|||||||
132
3rdparty/taglib/it/itproperties.h
vendored
132
3rdparty/taglib/it/itproperties.h
vendored
@@ -31,79 +31,77 @@
|
|||||||
|
|
||||||
namespace Strawberry_TagLib {
|
namespace Strawberry_TagLib {
|
||||||
namespace TagLib {
|
namespace TagLib {
|
||||||
namespace IT {
|
namespace IT {
|
||||||
class TAGLIB_EXPORT Properties : public AudioProperties {
|
class TAGLIB_EXPORT AudioProperties : public Strawberry_TagLib::TagLib::AudioProperties {
|
||||||
friend class File;
|
friend class File;
|
||||||
public:
|
|
||||||
/*! Flag bits. */
|
|
||||||
enum {
|
|
||||||
Stereo = 1,
|
|
||||||
Vol0MixOptimizations = 2,
|
|
||||||
UseInstruments = 4,
|
|
||||||
LinearSlides = 8,
|
|
||||||
OldEffects = 16,
|
|
||||||
LinkEffects = 32,
|
|
||||||
UseMidiPitchController = 64,
|
|
||||||
RequestEmbeddedMidiConf = 128
|
|
||||||
};
|
|
||||||
|
|
||||||
/*! Special bits. */
|
public:
|
||||||
enum {
|
/*! Flag bits. */
|
||||||
MessageAttached = 1,
|
enum {
|
||||||
MidiConfEmbedded = 8
|
Stereo = 1,
|
||||||
};
|
Vol0MixOptimizations = 2,
|
||||||
|
UseInstruments = 4,
|
||||||
|
LinearSlides = 8,
|
||||||
|
OldEffects = 16,
|
||||||
|
LinkEffects = 32,
|
||||||
|
UseMidiPitchController = 64,
|
||||||
|
RequestEmbeddedMidiConf = 128
|
||||||
|
};
|
||||||
|
|
||||||
Properties(AudioProperties::ReadStyle propertiesStyle);
|
/*! Special bits. */
|
||||||
virtual ~Properties();
|
enum {
|
||||||
|
MessageAttached = 1,
|
||||||
|
MidiConfEmbedded = 8
|
||||||
|
};
|
||||||
|
|
||||||
int length() const;
|
explicit AudioProperties(AudioProperties::ReadStyle);
|
||||||
int lengthInSeconds() const;
|
~AudioProperties() override;
|
||||||
int lengthInMilliseconds() const;
|
|
||||||
int bitrate() const;
|
|
||||||
int sampleRate() const;
|
|
||||||
int channels() const;
|
|
||||||
|
|
||||||
unsigned short lengthInPatterns() const;
|
int lengthInSeconds() const override;
|
||||||
bool stereo() const;
|
int lengthInMilliseconds() const override;
|
||||||
unsigned short instrumentCount() const;
|
int bitrate() const override;
|
||||||
unsigned short sampleCount() const;
|
int sampleRate() const override;
|
||||||
unsigned short patternCount() const;
|
int channels() const override;
|
||||||
unsigned short version() const;
|
|
||||||
unsigned short compatibleVersion() const;
|
|
||||||
unsigned short flags() const;
|
|
||||||
unsigned short special() const;
|
|
||||||
unsigned char globalVolume() const;
|
|
||||||
unsigned char mixVolume() const;
|
|
||||||
unsigned char tempo() const;
|
|
||||||
unsigned char bpmSpeed() const;
|
|
||||||
unsigned char panningSeparation() const;
|
|
||||||
unsigned char pitchWheelDepth() const;
|
|
||||||
|
|
||||||
void setChannels(int channels);
|
unsigned short lengthInPatterns() const;
|
||||||
void setLengthInPatterns(unsigned short lengthInPatterns);
|
bool stereo() const;
|
||||||
void setInstrumentCount(unsigned short instrumentCount);
|
unsigned short instrumentCount() const;
|
||||||
void setSampleCount (unsigned short sampleCount);
|
unsigned short sampleCount() const;
|
||||||
void setPatternCount(unsigned short patternCount);
|
unsigned short patternCount() const;
|
||||||
void setVersion (unsigned short version);
|
unsigned short version() const;
|
||||||
void setCompatibleVersion(unsigned short compatibleVersion);
|
unsigned short compatibleVersion() const;
|
||||||
void setFlags (unsigned short flags);
|
unsigned short flags() const;
|
||||||
void setSpecial (unsigned short special);
|
unsigned short special() const;
|
||||||
void setGlobalVolume(unsigned char globalVolume);
|
unsigned char globalVolume() const;
|
||||||
void setMixVolume (unsigned char mixVolume);
|
unsigned char mixVolume() const;
|
||||||
void setTempo (unsigned char tempo);
|
unsigned char tempo() const;
|
||||||
void setBpmSpeed (unsigned char bpmSpeed);
|
unsigned char bpmSpeed() const;
|
||||||
void setPanningSeparation(unsigned char panningSeparation);
|
unsigned char panningSeparation() const;
|
||||||
void setPitchWheelDepth (unsigned char pitchWheelDepth);
|
unsigned char pitchWheelDepth() const;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
Properties(const Properties&);
|
void setChannels(int channels);
|
||||||
Properties &operator=(const Properties&);
|
void setLengthInPatterns(unsigned short lengthInPatterns);
|
||||||
|
void setInstrumentCount(unsigned short instrumentCount);
|
||||||
|
void setSampleCount(unsigned short sampleCount);
|
||||||
|
void setPatternCount(unsigned short patternCount);
|
||||||
|
void setVersion(unsigned short version);
|
||||||
|
void setCompatibleVersion(unsigned short compatibleVersion);
|
||||||
|
void setFlags(unsigned short flags);
|
||||||
|
void setSpecial(unsigned short special);
|
||||||
|
void setGlobalVolume(unsigned char globalVolume);
|
||||||
|
void setMixVolume(unsigned char mixVolume);
|
||||||
|
void setTempo(unsigned char tempo);
|
||||||
|
void setBpmSpeed(unsigned char bpmSpeed);
|
||||||
|
void setPanningSeparation(unsigned char panningSeparation);
|
||||||
|
void setPitchWheelDepth(unsigned char pitchWheelDepth);
|
||||||
|
|
||||||
class PropertiesPrivate;
|
private:
|
||||||
PropertiesPrivate *d;
|
class AudioPropertiesPrivate;
|
||||||
};
|
AudioPropertiesPrivate *d;
|
||||||
}
|
};
|
||||||
}
|
} // namespace IT
|
||||||
}
|
} // namespace TagLib
|
||||||
|
} // namespace Strawberry_TagLib
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
93
3rdparty/taglib/mod/modfile.cpp
vendored
93
3rdparty/taglib/mod/modfile.cpp
vendored
@@ -33,115 +33,95 @@
|
|||||||
using namespace Strawberry_TagLib::TagLib;
|
using namespace Strawberry_TagLib::TagLib;
|
||||||
using namespace Mod;
|
using namespace Mod;
|
||||||
|
|
||||||
class Mod::File::FilePrivate
|
class Mod::File::FilePrivate {
|
||||||
{
|
public:
|
||||||
public:
|
explicit FilePrivate(AudioProperties::ReadStyle propertiesStyle) : properties(propertiesStyle) {}
|
||||||
explicit FilePrivate(AudioProperties::ReadStyle propertiesStyle)
|
|
||||||
: properties(propertiesStyle)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
Mod::Tag tag;
|
Mod::Tag tag;
|
||||||
Mod::Properties properties;
|
Mod::AudioProperties properties;
|
||||||
};
|
};
|
||||||
|
|
||||||
Mod::File::File(FileName file, bool readProperties,
|
Mod::File::File(FileName file, bool readProperties, AudioProperties::ReadStyle propertiesStyle) : Mod::FileBase(file), d(new FilePrivate(propertiesStyle)) {
|
||||||
AudioProperties::ReadStyle propertiesStyle) :
|
|
||||||
Mod::FileBase(file),
|
if (isOpen())
|
||||||
d(new FilePrivate(propertiesStyle))
|
|
||||||
{
|
|
||||||
if(isOpen())
|
|
||||||
read(readProperties);
|
read(readProperties);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Mod::File::File(IOStream *stream, bool readProperties,
|
Mod::File::File(IOStream *stream, bool readProperties, AudioProperties::ReadStyle propertiesStyle) : Mod::FileBase(stream), d(new FilePrivate(propertiesStyle)) {
|
||||||
AudioProperties::ReadStyle propertiesStyle) :
|
|
||||||
Mod::FileBase(stream),
|
if (isOpen())
|
||||||
d(new FilePrivate(propertiesStyle))
|
|
||||||
{
|
|
||||||
if(isOpen())
|
|
||||||
read(readProperties);
|
read(readProperties);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Mod::File::~File()
|
Mod::File::~File() {
|
||||||
{
|
|
||||||
delete d;
|
delete d;
|
||||||
}
|
}
|
||||||
|
|
||||||
Mod::Tag *Mod::File::tag() const
|
Mod::Tag *Mod::File::tag() const {
|
||||||
{
|
|
||||||
return &d->tag;
|
return &d->tag;
|
||||||
}
|
}
|
||||||
|
|
||||||
Mod::Properties *Mod::File::audioProperties() const
|
Mod::AudioProperties *Mod::File::audioProperties() const {
|
||||||
{
|
|
||||||
return &d->properties;
|
return &d->properties;
|
||||||
}
|
}
|
||||||
|
|
||||||
PropertyMap Mod::File::properties() const
|
bool Mod::File::save() {
|
||||||
{
|
|
||||||
return d->tag.properties();
|
|
||||||
}
|
|
||||||
|
|
||||||
PropertyMap Mod::File::setProperties(const PropertyMap &properties)
|
if (readOnly()) {
|
||||||
{
|
|
||||||
return d->tag.setProperties(properties);
|
|
||||||
}
|
|
||||||
|
|
||||||
bool Mod::File::save()
|
|
||||||
{
|
|
||||||
if(readOnly()) {
|
|
||||||
debug("Mod::File::save() - Cannot save to a read only file.");
|
debug("Mod::File::save() - Cannot save to a read only file.");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
seek(0);
|
seek(0);
|
||||||
writeString(d->tag.title(), 20);
|
writeString(d->tag.title(), 20);
|
||||||
StringList lines = d->tag.comment().split("\n");
|
StringList lines = d->tag.comment().split("\n");
|
||||||
unsigned int n = std::min(lines.size(), d->properties.instrumentCount());
|
size_t n = std::min<size_t>(lines.size(), d->properties.instrumentCount());
|
||||||
for(unsigned int i = 0; i < n; ++ i) {
|
for (size_t i = 0; i < n; ++i) {
|
||||||
writeString(lines[i], 22);
|
writeString(lines[i], 22);
|
||||||
seek(8, Current);
|
seek(8, Current);
|
||||||
}
|
}
|
||||||
|
|
||||||
for(unsigned int i = n; i < d->properties.instrumentCount(); ++ i) {
|
for (size_t i = n; i < d->properties.instrumentCount(); ++i) {
|
||||||
writeString(String(), 22);
|
writeString(String(), 22);
|
||||||
seek(8, Current);
|
seek(8, Current);
|
||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void Mod::File::read(bool)
|
void Mod::File::read(bool) {
|
||||||
{
|
|
||||||
if(!isOpen())
|
if (!isOpen())
|
||||||
return;
|
return;
|
||||||
|
|
||||||
seek(1080);
|
seek(1080);
|
||||||
ByteVector modId = readBlock(4);
|
ByteVector modId = readBlock(4);
|
||||||
READ_ASSERT(modId.size() == 4);
|
READ_ASSERT(modId.size() == 4);
|
||||||
|
|
||||||
int channels = 4;
|
int channels = 4;
|
||||||
unsigned int instruments = 31;
|
unsigned int instruments = 31;
|
||||||
if(modId == "M.K." || modId == "M!K!" || modId == "M&K!" || modId == "N.T.") {
|
if (modId == "M.K." || modId == "M!K!" || modId == "M&K!" || modId == "N.T.") {
|
||||||
d->tag.setTrackerName("ProTracker");
|
d->tag.setTrackerName("ProTracker");
|
||||||
channels = 4;
|
channels = 4;
|
||||||
}
|
}
|
||||||
else if(modId.startsWith("FLT") || modId.startsWith("TDZ")) {
|
else if (modId.startsWith("FLT") || modId.startsWith("TDZ")) {
|
||||||
d->tag.setTrackerName("StarTrekker");
|
d->tag.setTrackerName("StarTrekker");
|
||||||
char digit = modId[3];
|
char digit = modId[3];
|
||||||
READ_ASSERT(digit >= '0' && digit <= '9');
|
READ_ASSERT(digit >= '0' && digit <= '9');
|
||||||
channels = digit - '0';
|
channels = digit - '0';
|
||||||
}
|
}
|
||||||
else if(modId.endsWith("CHN")) {
|
else if (modId.endsWith("CHN")) {
|
||||||
d->tag.setTrackerName("StarTrekker");
|
d->tag.setTrackerName("StarTrekker");
|
||||||
char digit = modId[0];
|
char digit = modId[0];
|
||||||
READ_ASSERT(digit >= '0' && digit <= '9');
|
READ_ASSERT(digit >= '0' && digit <= '9');
|
||||||
channels = digit - '0';
|
channels = digit - '0';
|
||||||
}
|
}
|
||||||
else if(modId == "CD81" || modId == "OKTA") {
|
else if (modId == "CD81" || modId == "OKTA") {
|
||||||
d->tag.setTrackerName("Atari Oktalyzer");
|
d->tag.setTrackerName("Atari Oktalyzer");
|
||||||
channels = 8;
|
channels = 8;
|
||||||
}
|
}
|
||||||
else if(modId.endsWith("CH") || modId.endsWith("CN")) {
|
else if (modId.endsWith("CH") || modId.endsWith("CN")) {
|
||||||
d->tag.setTrackerName("TakeTracker");
|
d->tag.setTrackerName("TakeTracker");
|
||||||
char digit = modId[0];
|
char digit = modId[0];
|
||||||
READ_ASSERT(digit >= '0' && digit <= '9');
|
READ_ASSERT(digit >= '0' && digit <= '9');
|
||||||
@@ -153,8 +133,8 @@ void Mod::File::read(bool)
|
|||||||
else {
|
else {
|
||||||
// Not sure if this is correct. I'd need a file
|
// Not sure if this is correct. I'd need a file
|
||||||
// created with NoiseTracker to check this.
|
// created with NoiseTracker to check this.
|
||||||
d->tag.setTrackerName("NoiseTracker"); // probably
|
d->tag.setTrackerName("NoiseTracker"); // probably
|
||||||
channels = 4;
|
channels = 4;
|
||||||
instruments = 15;
|
instruments = 15;
|
||||||
}
|
}
|
||||||
d->properties.setChannels(channels);
|
d->properties.setChannels(channels);
|
||||||
@@ -164,7 +144,7 @@ void Mod::File::read(bool)
|
|||||||
READ_STRING(d->tag.setTitle, 20);
|
READ_STRING(d->tag.setTitle, 20);
|
||||||
|
|
||||||
StringList comment;
|
StringList comment;
|
||||||
for(unsigned int i = 0; i < instruments; ++ i) {
|
for (unsigned int i = 0; i < instruments; ++i) {
|
||||||
READ_STRING_AS(instrumentName, 22);
|
READ_STRING_AS(instrumentName, 22);
|
||||||
// value in words, * 2 (<< 1) for bytes:
|
// value in words, * 2 (<< 1) for bytes:
|
||||||
READ_U16B_AS(sampleLength);
|
READ_U16B_AS(sampleLength);
|
||||||
@@ -172,10 +152,10 @@ void Mod::File::read(bool)
|
|||||||
READ_BYTE_AS(fineTuneByte);
|
READ_BYTE_AS(fineTuneByte);
|
||||||
int fineTune = fineTuneByte & 0xF;
|
int fineTune = fineTuneByte & 0xF;
|
||||||
// > 7 means negative value
|
// > 7 means negative value
|
||||||
if(fineTune > 7) fineTune -= 16;
|
if (fineTune > 7) fineTune -= 16;
|
||||||
|
|
||||||
READ_BYTE_AS(volume);
|
READ_BYTE_AS(volume);
|
||||||
if(volume > 64) volume = 64;
|
if (volume > 64) volume = 64;
|
||||||
// volume in decibels: 20 * log10(volume / 64)
|
// volume in decibels: 20 * log10(volume / 64)
|
||||||
|
|
||||||
// value in words, * 2 (<< 1) for bytes:
|
// value in words, * 2 (<< 1) for bytes:
|
||||||
@@ -189,4 +169,5 @@ void Mod::File::read(bool)
|
|||||||
READ_BYTE(d->properties.setLengthInPatterns);
|
READ_BYTE(d->properties.setLengthInPatterns);
|
||||||
|
|
||||||
d->tag.setComment(comment.toString("\n"));
|
d->tag.setComment(comment.toString("\n"));
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
113
3rdparty/taglib/mod/modfile.h
vendored
113
3rdparty/taglib/mod/modfile.h
vendored
@@ -35,82 +35,61 @@
|
|||||||
|
|
||||||
namespace Strawberry_TagLib {
|
namespace Strawberry_TagLib {
|
||||||
namespace TagLib {
|
namespace TagLib {
|
||||||
|
namespace Mod {
|
||||||
|
|
||||||
namespace Mod {
|
class TAGLIB_EXPORT File : public Strawberry_TagLib::TagLib::Mod::FileBase {
|
||||||
|
public:
|
||||||
|
/*!
|
||||||
|
* Constructs a Protracker file from \a file.
|
||||||
|
*
|
||||||
|
* \note In the current implementation, both \a readProperties and \a propertiesStyle are ignored.
|
||||||
|
* The audio properties are always read.
|
||||||
|
*/
|
||||||
|
explicit File(FileName file, bool readProperties = true, AudioProperties::ReadStyle propertiesStyle = AudioProperties::Average);
|
||||||
|
|
||||||
class TAGLIB_EXPORT File : public Strawberry_TagLib::TagLib::Mod::FileBase
|
/*!
|
||||||
{
|
* Constructs a Protracker file from \a stream.
|
||||||
public:
|
*
|
||||||
/*!
|
* \note In the current implementation, both \a readProperties and \a propertiesStyle are ignored.
|
||||||
* Constructs a Protracker file from \a file.
|
* The audio properties are always read.
|
||||||
*
|
*
|
||||||
* \note In the current implementation, both \a readProperties and
|
* \note TagLib will *not* take ownership of the stream, the caller is
|
||||||
* \a propertiesStyle are ignored. The audio properties are always
|
* responsible for deleting it after the File object.
|
||||||
* read.
|
*/
|
||||||
*/
|
explicit File(IOStream *stream, bool readProperties = true, AudioProperties::ReadStyle propertiesStyle = AudioProperties::Average);
|
||||||
File(FileName file, bool readProperties = true,
|
|
||||||
AudioProperties::ReadStyle propertiesStyle =
|
|
||||||
AudioProperties::Average);
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Constructs a Protracker file from \a stream.
|
* Destroys this instance of the File.
|
||||||
*
|
*/
|
||||||
* \note In the current implementation, both \a readProperties and
|
~File() override;
|
||||||
* \a propertiesStyle are ignored. The audio properties are always
|
|
||||||
* read.
|
|
||||||
*
|
|
||||||
* \note TagLib will *not* take ownership of the stream, the caller is
|
|
||||||
* responsible for deleting it after the File object.
|
|
||||||
*/
|
|
||||||
File(IOStream *stream, bool readProperties = true,
|
|
||||||
AudioProperties::ReadStyle propertiesStyle =
|
|
||||||
AudioProperties::Average);
|
|
||||||
|
|
||||||
/*!
|
Mod::Tag *tag() const override;
|
||||||
* Destroys this instance of the File.
|
|
||||||
*/
|
|
||||||
virtual ~File();
|
|
||||||
|
|
||||||
Mod::Tag *tag() const;
|
/*!
|
||||||
|
* Returns the Mod::AudioProperties for this file. If no audio properties were read then this will return a null pointer.
|
||||||
|
*/
|
||||||
|
Mod::AudioProperties *audioProperties() const override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Implements the unified property interface -- export function.
|
* Save the file.
|
||||||
* Forwards to Mod::Tag::properties().
|
* This is the same as calling save(AllTags);
|
||||||
*/
|
*
|
||||||
PropertyMap properties() const;
|
* \note Saving Protracker tags is not supported.
|
||||||
|
*/
|
||||||
|
bool save() override;
|
||||||
|
|
||||||
/*!
|
private:
|
||||||
* Implements the unified property interface -- import function.
|
File(const File &);
|
||||||
* Forwards to Mod::Tag::setProperties().
|
File &operator=(const File &);
|
||||||
*/
|
|
||||||
PropertyMap setProperties(const PropertyMap &);
|
|
||||||
/*!
|
|
||||||
* Returns the Mod::Properties for this file. If no audio properties
|
|
||||||
* were read then this will return a null pointer.
|
|
||||||
*/
|
|
||||||
Mod::Properties *audioProperties() const;
|
|
||||||
|
|
||||||
/*!
|
void read(bool readProperties);
|
||||||
* Save the file.
|
|
||||||
* This is the same as calling save(AllTags);
|
|
||||||
*
|
|
||||||
* \note Saving Protracker tags is not supported.
|
|
||||||
*/
|
|
||||||
bool save();
|
|
||||||
|
|
||||||
private:
|
class FilePrivate;
|
||||||
File(const File &);
|
FilePrivate *d;
|
||||||
File &operator=(const File &);
|
};
|
||||||
|
|
||||||
void read(bool readProperties);
|
} // namespace Mod
|
||||||
|
} // namespace TagLib
|
||||||
class FilePrivate;
|
} // namespace Strawberry_TagLib
|
||||||
FilePrivate *d;
|
|
||||||
};
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
87
3rdparty/taglib/mod/modfilebase.cpp
vendored
87
3rdparty/taglib/mod/modfilebase.cpp
vendored
@@ -30,96 +30,93 @@
|
|||||||
using namespace Strawberry_TagLib::TagLib;
|
using namespace Strawberry_TagLib::TagLib;
|
||||||
using namespace Mod;
|
using namespace Mod;
|
||||||
|
|
||||||
Mod::FileBase::FileBase(FileName file) : Strawberry_TagLib::TagLib::File(file)
|
Mod::FileBase::FileBase(FileName file) : Strawberry_TagLib::TagLib::File(file) {}
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
Mod::FileBase::FileBase(IOStream *stream) : Strawberry_TagLib::TagLib::File(stream)
|
Mod::FileBase::FileBase(IOStream *stream) : Strawberry_TagLib::TagLib::File(stream) {}
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
void Mod::FileBase::writeString(const String &s, unsigned long size, char padding)
|
void Mod::FileBase::writeString(const String &s, unsigned int size, char padding) {
|
||||||
{
|
|
||||||
ByteVector data(s.data(String::Latin1));
|
ByteVector data(s.data(String::Latin1));
|
||||||
data.resize(size, padding);
|
data.resize(size, padding);
|
||||||
writeBlock(data);
|
writeBlock(data);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Mod::FileBase::readString(String &s, unsigned long size)
|
bool Mod::FileBase::readString(String &s, unsigned int size) {
|
||||||
{
|
|
||||||
ByteVector data(readBlock(size));
|
ByteVector data(readBlock(size));
|
||||||
if(data.size() < size) return false;
|
if (data.size() < size) return false;
|
||||||
int index = data.find((char) 0);
|
const size_t index = data.find(static_cast<char>(0));
|
||||||
if(index > -1)
|
if (index != ByteVector::npos()) {
|
||||||
{
|
|
||||||
data.resize(index);
|
data.resize(index);
|
||||||
}
|
}
|
||||||
data.replace('\xff', ' ');
|
data.replace('\xff', ' ');
|
||||||
|
|
||||||
s = data;
|
s = data;
|
||||||
return true;
|
return true;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void Mod::FileBase::writeByte(unsigned char _byte)
|
void Mod::FileBase::writeByte(unsigned char _byte) {
|
||||||
{
|
|
||||||
ByteVector data(1, _byte);
|
ByteVector data(1, _byte);
|
||||||
writeBlock(data);
|
writeBlock(data);
|
||||||
}
|
}
|
||||||
|
|
||||||
void Mod::FileBase::writeU16L(unsigned short number)
|
void Mod::FileBase::writeU16L(unsigned short number) {
|
||||||
{
|
writeBlock(ByteVector::fromUInt16LE(number));
|
||||||
writeBlock(ByteVector::fromShort(number, false));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void Mod::FileBase::writeU32L(unsigned long number)
|
void Mod::FileBase::writeU32L(unsigned int number) {
|
||||||
{
|
writeBlock(ByteVector::fromUInt32LE(number));
|
||||||
writeBlock(ByteVector::fromUInt(number, false));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void Mod::FileBase::writeU16B(unsigned short number)
|
void Mod::FileBase::writeU16B(unsigned short number) {
|
||||||
{
|
writeBlock(ByteVector::fromUInt16BE(number));
|
||||||
writeBlock(ByteVector::fromShort(number, true));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void Mod::FileBase::writeU32B(unsigned long number)
|
void Mod::FileBase::writeU32B(unsigned int number) {
|
||||||
{
|
writeBlock(ByteVector::fromUInt32BE(number));
|
||||||
writeBlock(ByteVector::fromUInt(number, true));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Mod::FileBase::readByte(unsigned char &_byte)
|
bool Mod::FileBase::readByte(unsigned char &_byte) {
|
||||||
{
|
|
||||||
ByteVector data(readBlock(1));
|
ByteVector data(readBlock(1));
|
||||||
if(data.size() < 1) return false;
|
if (data.size() < 1) return false;
|
||||||
_byte = data[0];
|
_byte = data[0];
|
||||||
return true;
|
return true;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Mod::FileBase::readU16L(unsigned short &number)
|
bool Mod::FileBase::readU16L(unsigned short &number) {
|
||||||
{
|
|
||||||
ByteVector data(readBlock(2));
|
ByteVector data(readBlock(2));
|
||||||
if(data.size() < 2) return false;
|
if (data.size() < 2) return false;
|
||||||
number = data.toUShort(false);
|
number = data.toUInt16LE(0);
|
||||||
return true;
|
return true;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Mod::FileBase::readU32L(unsigned long &number) {
|
bool Mod::FileBase::readU32L(unsigned int &number) {
|
||||||
|
|
||||||
ByteVector data(readBlock(4));
|
ByteVector data(readBlock(4));
|
||||||
if(data.size() < 4) return false;
|
if (data.size() < 4) return false;
|
||||||
number = data.toUInt(false);
|
number = data.toUInt32LE(0);
|
||||||
return true;
|
return true;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Mod::FileBase::readU16B(unsigned short &number)
|
bool Mod::FileBase::readU16B(unsigned short &number) {
|
||||||
{
|
|
||||||
ByteVector data(readBlock(2));
|
ByteVector data(readBlock(2));
|
||||||
if(data.size() < 2) return false;
|
if (data.size() < 2) return false;
|
||||||
number = data.toUShort(true);
|
number = data.toUInt16BE(0);
|
||||||
return true;
|
return true;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Mod::FileBase::readU32B(unsigned long &number) {
|
bool Mod::FileBase::readU32B(unsigned int &number) {
|
||||||
|
|
||||||
ByteVector data(readBlock(4));
|
ByteVector data(readBlock(4));
|
||||||
if(data.size() < 4) return false;
|
if (data.size() < 4) return false;
|
||||||
number = data.toUInt(true);
|
number = data.toUInt32BE(0);
|
||||||
return true;
|
return true;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
45
3rdparty/taglib/mod/modfilebase.h
vendored
45
3rdparty/taglib/mod/modfilebase.h
vendored
@@ -36,33 +36,30 @@
|
|||||||
|
|
||||||
namespace Strawberry_TagLib {
|
namespace Strawberry_TagLib {
|
||||||
namespace TagLib {
|
namespace TagLib {
|
||||||
|
namespace Mod {
|
||||||
|
|
||||||
namespace Mod {
|
class TAGLIB_EXPORT FileBase : public Strawberry_TagLib::TagLib::File {
|
||||||
|
protected:
|
||||||
|
explicit FileBase(FileName file);
|
||||||
|
explicit FileBase(IOStream *stream);
|
||||||
|
|
||||||
class TAGLIB_EXPORT FileBase : public Strawberry_TagLib::TagLib::File
|
void writeString(const String &s, unsigned int size, char padding = 0);
|
||||||
{
|
void writeByte(unsigned char byte);
|
||||||
protected:
|
void writeU16L(unsigned short number);
|
||||||
FileBase(FileName file);
|
void writeU32L(unsigned int number);
|
||||||
FileBase(IOStream *stream);
|
void writeU16B(unsigned short number);
|
||||||
|
void writeU32B(unsigned int number);
|
||||||
|
|
||||||
void writeString(const String &s, unsigned long size, char padding = 0);
|
bool readString(String &s, unsigned int size);
|
||||||
void writeByte(unsigned char byte);
|
bool readByte(unsigned char &_byte);
|
||||||
void writeU16L(unsigned short number);
|
bool readU16L(unsigned short &number);
|
||||||
void writeU32L(unsigned long number);
|
bool readU32L(unsigned int &number);
|
||||||
void writeU16B(unsigned short number);
|
bool readU16B(unsigned short &number);
|
||||||
void writeU32B(unsigned long number);
|
bool readU32B(unsigned int &number);
|
||||||
|
};
|
||||||
|
|
||||||
bool readString(String &s, unsigned long size);
|
} // namespace Mod
|
||||||
bool readByte(unsigned char &byte);
|
} // namespace TagLib
|
||||||
bool readU16L(unsigned short &number);
|
} // namespace Strawberry_TagLib
|
||||||
bool readU32L(unsigned long &number);
|
|
||||||
bool readU16B(unsigned short &number);
|
|
||||||
bool readU32B(unsigned long &number);
|
|
||||||
};
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
53
3rdparty/taglib/mod/modfileprivate.h
vendored
53
3rdparty/taglib/mod/modfileprivate.h
vendored
@@ -24,44 +24,43 @@
|
|||||||
|
|
||||||
// some helper-macros only used internally by (s3m|it|xm)file.cpp
|
// some helper-macros only used internally by (s3m|it|xm)file.cpp
|
||||||
#define READ_ASSERT(cond) \
|
#define READ_ASSERT(cond) \
|
||||||
if(!(cond)) \
|
if (!(cond)) { \
|
||||||
{ \
|
setValid(false); \
|
||||||
setValid(false); \
|
return; \
|
||||||
return; \
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#define READ(setter,type,read) \
|
#define READ(setter, type, read) \
|
||||||
{ \
|
{ \
|
||||||
type number; \
|
type number; \
|
||||||
READ_ASSERT(read(number)); \
|
READ_ASSERT(read(number)); \
|
||||||
setter(number); \
|
setter(number); \
|
||||||
}
|
}
|
||||||
|
|
||||||
#define READ_BYTE(setter) READ(setter,unsigned char,readByte)
|
#define READ_BYTE(setter) READ(setter, unsigned char, readByte)
|
||||||
#define READ_U16L(setter) READ(setter,unsigned short,readU16L)
|
#define READ_U16L(setter) READ(setter, unsigned short, readU16L)
|
||||||
#define READ_U32L(setter) READ(setter,unsigned long,readU32L)
|
#define READ_U32L(setter) READ(setter, unsigned int, readU32L)
|
||||||
#define READ_U16B(setter) READ(setter,unsigned short,readU16B)
|
#define READ_U16B(setter) READ(setter, unsigned short, readU16B)
|
||||||
#define READ_U32B(setter) READ(setter,unsigned long,readU32B)
|
#define READ_U32B(setter) READ(setter, unsigned int, readU32B)
|
||||||
|
|
||||||
#define READ_STRING(setter,size) \
|
#define READ_STRING(setter, size) \
|
||||||
{ \
|
{ \
|
||||||
String s; \
|
String s; \
|
||||||
READ_ASSERT(readString(s, size)); \
|
READ_ASSERT(readString(s, size)); \
|
||||||
setter(s); \
|
setter(s); \
|
||||||
}
|
}
|
||||||
|
|
||||||
#define READ_AS(type,name,read) \
|
#define READ_AS(type, name, read) \
|
||||||
type name = 0; \
|
type name = 0; \
|
||||||
READ_ASSERT(read(name));
|
READ_ASSERT(read(name));
|
||||||
|
|
||||||
#define READ_BYTE_AS(name) READ_AS(unsigned char,name,readByte)
|
#define READ_BYTE_AS(name) READ_AS(unsigned char, name, readByte)
|
||||||
#define READ_U16L_AS(name) READ_AS(unsigned short,name,readU16L)
|
#define READ_U16L_AS(name) READ_AS(unsigned short, name, readU16L)
|
||||||
#define READ_U32L_AS(name) READ_AS(unsigned long,name,readU32L)
|
#define READ_U32L_AS(name) READ_AS(unsigned int, name, readU32L)
|
||||||
#define READ_U16B_AS(name) READ_AS(unsigned short,name,readU16B)
|
#define READ_U16B_AS(name) READ_AS(unsigned short, name, readU16B)
|
||||||
#define READ_U32B_AS(name) READ_AS(unsigned long,name,readU32B)
|
#define READ_U32B_AS(name) READ_AS(unsigned int, name, readU32B)
|
||||||
|
|
||||||
#define READ_STRING_AS(name,size) \
|
#define READ_STRING_AS(name, size) \
|
||||||
String name; \
|
String name; \
|
||||||
READ_ASSERT(readString(name, size));
|
READ_ASSERT(readString(name, size));
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
67
3rdparty/taglib/mod/modproperties.cpp
vendored
67
3rdparty/taglib/mod/modproperties.cpp
vendored
@@ -29,83 +29,66 @@
|
|||||||
using namespace Strawberry_TagLib::TagLib;
|
using namespace Strawberry_TagLib::TagLib;
|
||||||
using namespace Mod;
|
using namespace Mod;
|
||||||
|
|
||||||
class Mod::Properties::PropertiesPrivate
|
class Mod::AudioProperties::AudioPropertiesPrivate {
|
||||||
{
|
public:
|
||||||
public:
|
explicit AudioPropertiesPrivate() : channels(0), instrumentCount(0), lengthInPatterns(0) {}
|
||||||
PropertiesPrivate() :
|
|
||||||
channels(0),
|
|
||||||
instrumentCount(0),
|
|
||||||
lengthInPatterns(0)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
int channels;
|
int channels;
|
||||||
unsigned int instrumentCount;
|
unsigned int instrumentCount;
|
||||||
unsigned char lengthInPatterns;
|
unsigned char lengthInPatterns;
|
||||||
};
|
};
|
||||||
|
|
||||||
Mod::Properties::Properties(AudioProperties::ReadStyle propertiesStyle) :
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
AudioProperties(propertiesStyle),
|
// public members
|
||||||
d(new PropertiesPrivate())
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
{
|
|
||||||
|
Mod::AudioProperties::AudioProperties(AudioProperties::ReadStyle) : Strawberry_TagLib::TagLib::AudioProperties(), d(new AudioPropertiesPrivate()) {
|
||||||
}
|
}
|
||||||
|
|
||||||
Mod::Properties::~Properties()
|
Mod::AudioProperties::~AudioProperties() {
|
||||||
{
|
|
||||||
delete d;
|
delete d;
|
||||||
}
|
}
|
||||||
|
|
||||||
int Mod::Properties::length() const
|
int Mod::AudioProperties::lengthInSeconds() const {
|
||||||
{
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
int Mod::Properties::lengthInSeconds() const
|
int Mod::AudioProperties::lengthInMilliseconds() const {
|
||||||
{
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
int Mod::Properties::lengthInMilliseconds() const
|
int Mod::AudioProperties::bitrate() const {
|
||||||
{
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
int Mod::Properties::bitrate() const
|
int Mod::AudioProperties::sampleRate() const {
|
||||||
{
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
int Mod::Properties::sampleRate() const
|
int Mod::AudioProperties::channels() const {
|
||||||
{
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
int Mod::Properties::channels() const
|
|
||||||
{
|
|
||||||
return d->channels;
|
return d->channels;
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned int Mod::Properties::instrumentCount() const
|
unsigned int Mod::AudioProperties::instrumentCount() const {
|
||||||
{
|
|
||||||
return d->instrumentCount;
|
return d->instrumentCount;
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned char Mod::Properties::lengthInPatterns() const
|
unsigned char Mod::AudioProperties::lengthInPatterns() const {
|
||||||
{
|
|
||||||
return d->lengthInPatterns;
|
return d->lengthInPatterns;
|
||||||
}
|
}
|
||||||
|
|
||||||
void Mod::Properties::setChannels(int channels)
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
{
|
// private members
|
||||||
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
|
void Mod::AudioProperties::setChannels(int channels) {
|
||||||
d->channels = channels;
|
d->channels = channels;
|
||||||
}
|
}
|
||||||
|
|
||||||
void Mod::Properties::setInstrumentCount(unsigned int instrumentCount)
|
void Mod::AudioProperties::setInstrumentCount(unsigned int instrumentCount) {
|
||||||
{
|
|
||||||
d->instrumentCount = instrumentCount;
|
d->instrumentCount = instrumentCount;
|
||||||
}
|
}
|
||||||
|
|
||||||
void Mod::Properties::setLengthInPatterns(unsigned char lengthInPatterns)
|
void Mod::AudioProperties::setLengthInPatterns(unsigned char lengthInPatterns) {
|
||||||
{
|
|
||||||
d->lengthInPatterns = lengthInPatterns;
|
d->lengthInPatterns = lengthInPatterns;
|
||||||
}
|
}
|
||||||
|
|||||||
53
3rdparty/taglib/mod/modproperties.h
vendored
53
3rdparty/taglib/mod/modproperties.h
vendored
@@ -31,43 +31,36 @@
|
|||||||
|
|
||||||
namespace Strawberry_TagLib {
|
namespace Strawberry_TagLib {
|
||||||
namespace TagLib {
|
namespace TagLib {
|
||||||
|
namespace Mod {
|
||||||
|
|
||||||
namespace Mod {
|
class TAGLIB_EXPORT AudioProperties : public Strawberry_TagLib::TagLib::AudioProperties {
|
||||||
|
friend class File;
|
||||||
|
|
||||||
class TAGLIB_EXPORT Properties : public AudioProperties
|
public:
|
||||||
{
|
explicit AudioProperties(AudioProperties::ReadStyle propertiesStyle);
|
||||||
public:
|
~AudioProperties() override;
|
||||||
Properties(AudioProperties::ReadStyle propertiesStyle);
|
|
||||||
virtual ~Properties();
|
|
||||||
|
|
||||||
int length() const;
|
int lengthInSeconds() const override;
|
||||||
int lengthInSeconds() const;
|
int lengthInMilliseconds() const override;
|
||||||
int lengthInMilliseconds() const;
|
int bitrate() const override;
|
||||||
int bitrate() const;
|
int sampleRate() const override;
|
||||||
int sampleRate() const;
|
int channels() const override;
|
||||||
int channels() const;
|
|
||||||
|
|
||||||
unsigned int instrumentCount() const;
|
unsigned int instrumentCount() const;
|
||||||
unsigned char lengthInPatterns() const;
|
unsigned char lengthInPatterns() const;
|
||||||
|
|
||||||
void setChannels(int channels);
|
private:
|
||||||
|
void setChannels(int channels);
|
||||||
|
|
||||||
void setInstrumentCount(unsigned int sampleCount);
|
void setInstrumentCount(unsigned int instrumentCount);
|
||||||
void setLengthInPatterns(unsigned char lengthInPatterns);
|
void setLengthInPatterns(unsigned char lengthInPatterns);
|
||||||
|
|
||||||
private:
|
class AudioPropertiesPrivate;
|
||||||
friend class File;
|
AudioPropertiesPrivate *d;
|
||||||
|
};
|
||||||
|
|
||||||
Properties(const Properties&);
|
} // namespace Mod
|
||||||
Properties &operator=(const Properties&);
|
} // namespace TagLib
|
||||||
|
} // namespace Strawberry_TagLib
|
||||||
class PropertiesPrivate;
|
|
||||||
PropertiesPrivate *d;
|
|
||||||
};
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
106
3rdparty/taglib/mod/modtag.cpp
vendored
106
3rdparty/taglib/mod/modtag.cpp
vendored
@@ -27,148 +27,136 @@
|
|||||||
#include "modtag.h"
|
#include "modtag.h"
|
||||||
#include "tstringlist.h"
|
#include "tstringlist.h"
|
||||||
#include "tpropertymap.h"
|
#include "tpropertymap.h"
|
||||||
|
#include "tpicturemap.h"
|
||||||
|
|
||||||
using namespace Strawberry_TagLib::TagLib;
|
using namespace Strawberry_TagLib::TagLib;
|
||||||
using namespace Mod;
|
using namespace Mod;
|
||||||
|
|
||||||
class Mod::Tag::TagPrivate
|
class Mod::Tag::TagPrivate {
|
||||||
{
|
public:
|
||||||
public:
|
explicit TagPrivate() {}
|
||||||
TagPrivate()
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
String title;
|
String title;
|
||||||
String comment;
|
String comment;
|
||||||
String trackerName;
|
String trackerName;
|
||||||
};
|
};
|
||||||
|
|
||||||
Mod::Tag::Tag() :
|
Mod::Tag::Tag() : d(new TagPrivate()) {}
|
||||||
Strawberry_TagLib::TagLib::Tag(),
|
|
||||||
d(new TagPrivate())
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
Mod::Tag::~Tag()
|
Mod::Tag::~Tag() {
|
||||||
{
|
|
||||||
delete d;
|
delete d;
|
||||||
}
|
}
|
||||||
|
|
||||||
String Mod::Tag::title() const
|
String Mod::Tag::title() const {
|
||||||
{
|
|
||||||
return d->title;
|
return d->title;
|
||||||
}
|
}
|
||||||
|
|
||||||
String Mod::Tag::artist() const
|
String Mod::Tag::artist() const {
|
||||||
{
|
|
||||||
return String();
|
return String();
|
||||||
}
|
}
|
||||||
|
|
||||||
String Mod::Tag::album() const
|
String Mod::Tag::album() const {
|
||||||
{
|
|
||||||
return String();
|
return String();
|
||||||
}
|
}
|
||||||
|
|
||||||
String Mod::Tag::comment() const
|
String Mod::Tag::comment() const {
|
||||||
{
|
|
||||||
return d->comment;
|
return d->comment;
|
||||||
}
|
}
|
||||||
|
|
||||||
String Mod::Tag::genre() const
|
String Mod::Tag::genre() const {
|
||||||
{
|
|
||||||
return String();
|
return String();
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned int Mod::Tag::year() const
|
unsigned int Mod::Tag::year() const {
|
||||||
{
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned int Mod::Tag::track() const
|
unsigned int Mod::Tag::track() const {
|
||||||
{
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
String Mod::Tag::trackerName() const
|
Strawberry_TagLib::TagLib::PictureMap Mod::Tag::pictures() const {
|
||||||
{
|
return PictureMap();
|
||||||
|
}
|
||||||
|
|
||||||
|
String Mod::Tag::trackerName() const {
|
||||||
return d->trackerName;
|
return d->trackerName;
|
||||||
}
|
}
|
||||||
|
|
||||||
void Mod::Tag::setTitle(const String &title)
|
void Mod::Tag::setTitle(const String &title) {
|
||||||
{
|
|
||||||
d->title = title;
|
d->title = title;
|
||||||
}
|
}
|
||||||
|
|
||||||
void Mod::Tag::setArtist(const String &)
|
void Mod::Tag::setArtist(const String &) {
|
||||||
{
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void Mod::Tag::setAlbum(const String &)
|
void Mod::Tag::setAlbum(const String &) {
|
||||||
{
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void Mod::Tag::setComment(const String &comment)
|
void Mod::Tag::setComment(const String &comment) {
|
||||||
{
|
|
||||||
d->comment = comment;
|
d->comment = comment;
|
||||||
}
|
}
|
||||||
|
|
||||||
void Mod::Tag::setGenre(const String &)
|
void Mod::Tag::setGenre(const String &) {
|
||||||
{
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void Mod::Tag::setYear(unsigned int)
|
void Mod::Tag::setYear(unsigned int) {
|
||||||
{
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void Mod::Tag::setTrack(unsigned int)
|
void Mod::Tag::setTrack(unsigned int) {
|
||||||
{
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void Mod::Tag::setTrackerName(const String &trackerName)
|
void Mod::Tag::setPictures(const PictureMap&) {}
|
||||||
{
|
|
||||||
|
void Mod::Tag::setTrackerName(const String &trackerName) {
|
||||||
d->trackerName = trackerName;
|
d->trackerName = trackerName;
|
||||||
}
|
}
|
||||||
|
|
||||||
PropertyMap Mod::Tag::properties() const
|
PropertyMap Mod::Tag::properties() const {
|
||||||
{
|
|
||||||
PropertyMap properties;
|
PropertyMap properties;
|
||||||
properties["TITLE"] = d->title;
|
properties["TITLE"] = d->title;
|
||||||
properties["COMMENT"] = d->comment;
|
properties["COMMENT"] = d->comment;
|
||||||
if(!(d->trackerName.isEmpty()))
|
if (!(d->trackerName.isEmpty()))
|
||||||
properties["TRACKERNAME"] = d->trackerName;
|
properties["TRACKERNAME"] = d->trackerName;
|
||||||
return properties;
|
return properties;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
PropertyMap Mod::Tag::setProperties(const PropertyMap &origProps)
|
PropertyMap Mod::Tag::setProperties(const PropertyMap &origProps) {
|
||||||
{
|
|
||||||
PropertyMap properties(origProps);
|
PropertyMap properties(origProps);
|
||||||
properties.removeEmpty();
|
properties.removeEmpty();
|
||||||
StringList oneValueSet;
|
StringList oneValueSet;
|
||||||
if(properties.contains("TITLE")) {
|
if (properties.contains("TITLE")) {
|
||||||
d->title = properties["TITLE"].front();
|
d->title = properties["TITLE"].front();
|
||||||
oneValueSet.append("TITLE");
|
oneValueSet.append("TITLE");
|
||||||
} else
|
}
|
||||||
|
else
|
||||||
d->title.clear();
|
d->title.clear();
|
||||||
|
|
||||||
if(properties.contains("COMMENT")) {
|
if (properties.contains("COMMENT")) {
|
||||||
d->comment = properties["COMMENT"].front();
|
d->comment = properties["COMMENT"].front();
|
||||||
oneValueSet.append("COMMENT");
|
oneValueSet.append("COMMENT");
|
||||||
} else
|
}
|
||||||
|
else
|
||||||
d->comment.clear();
|
d->comment.clear();
|
||||||
|
|
||||||
if(properties.contains("TRACKERNAME")) {
|
if (properties.contains("TRACKERNAME")) {
|
||||||
d->trackerName = properties["TRACKERNAME"].front();
|
d->trackerName = properties["TRACKERNAME"].front();
|
||||||
oneValueSet.append("TRACKERNAME");
|
oneValueSet.append("TRACKERNAME");
|
||||||
} else
|
}
|
||||||
|
else
|
||||||
d->trackerName.clear();
|
d->trackerName.clear();
|
||||||
|
|
||||||
// for each tag that has been set above, remove the first entry in the corresponding
|
// for each tag that has been set above, remove the first entry in the corresponding
|
||||||
// value list. The others will be returned as unsupported by this format.
|
// value list. The others will be returned as unsupported by this format.
|
||||||
for(StringList::ConstIterator it = oneValueSet.begin(); it != oneValueSet.end(); ++it) {
|
for (StringList::ConstIterator it = oneValueSet.begin(); it != oneValueSet.end(); ++it) {
|
||||||
if(properties[*it].size() == 1)
|
if (properties[*it].size() == 1)
|
||||||
properties.erase(*it);
|
properties.erase(*it);
|
||||||
else
|
else
|
||||||
properties[*it].erase( properties[*it].begin() );
|
properties[*it].erase(properties[*it].begin());
|
||||||
}
|
}
|
||||||
return properties;
|
return properties;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
266
3rdparty/taglib/mod/modtag.h
vendored
266
3rdparty/taglib/mod/modtag.h
vendored
@@ -30,167 +30,157 @@
|
|||||||
|
|
||||||
namespace Strawberry_TagLib {
|
namespace Strawberry_TagLib {
|
||||||
namespace TagLib {
|
namespace TagLib {
|
||||||
|
namespace Mod {
|
||||||
|
|
||||||
namespace Mod {
|
/*!
|
||||||
|
* Tags for module files (Mod, S3M, IT, XM).
|
||||||
|
*
|
||||||
|
* Note that only the \a title is supported as such by most module file formats.
|
||||||
|
* Except for XM files the \a trackerName is derived from the file format or the flavour of the file format.
|
||||||
|
* For XM files it is stored in the file.
|
||||||
|
*
|
||||||
|
* The \a comment tag is not strictly supported by module files,
|
||||||
|
* but it is common practice to abuse instrument/sample/pattern names as multiline comments.
|
||||||
|
* TagLib does so as well.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
class TAGLIB_EXPORT Tag : public Strawberry_TagLib::TagLib::Tag {
|
||||||
|
public:
|
||||||
|
explicit Tag();
|
||||||
|
~Tag() override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Tags for module files (Mod, S3M, IT, XM).
|
* Returns the track name; if no track name is present in the tag String::null will be returned.
|
||||||
*
|
*/
|
||||||
* Note that only the \a title is supported as such by most
|
String title() const override;
|
||||||
* module file formats. Except for XM files the \a trackerName
|
|
||||||
* is derived from the file format or the flavour of the file
|
|
||||||
* format. For XM files it is stored in the file.
|
|
||||||
*
|
|
||||||
* The \a comment tag is not strictly supported by module files,
|
|
||||||
* but it is common practice to abuse instrument/sample/pattern
|
|
||||||
* names as multiline comments. TagLib does so as well.
|
|
||||||
*/
|
|
||||||
class TAGLIB_EXPORT Tag : public Strawberry_TagLib::TagLib::Tag
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
Tag();
|
|
||||||
virtual ~Tag();
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the track name; if no track name is present in the tag
|
* Not supported by module files. Therefore always returns String::null.
|
||||||
* String::null will be returned.
|
*/
|
||||||
*/
|
String artist() const override;
|
||||||
virtual String title() const;
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Not supported by module files. Therefore always returns String::null.
|
* Not supported by module files. Therefore always returns String::null.
|
||||||
*/
|
*/
|
||||||
virtual String artist() const;
|
String album() const override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Not supported by module files. Therefore always returns String::null.
|
* Returns the track comment derived from the instrument/sample/pattern
|
||||||
*/
|
* names; if no comment is present in the tag String::null will be returned.
|
||||||
virtual String album() const;
|
*/
|
||||||
|
String comment() const override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the track comment derived from the instrument/sample/pattern
|
* Not supported by module files. Therefore always returns String::null.
|
||||||
* names; if no comment is present in the tag String::null will be
|
*/
|
||||||
* returned.
|
String genre() const override;
|
||||||
*/
|
|
||||||
virtual String comment() const;
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Not supported by module files. Therefore always returns String::null.
|
* Not supported by module files. Therefore always returns 0.
|
||||||
*/
|
*/
|
||||||
virtual String genre() const;
|
unsigned int year() const override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Not supported by module files. Therefore always returns 0.
|
* Not supported by module files. Therefore always returns 0.
|
||||||
*/
|
*/
|
||||||
virtual unsigned int year() const;
|
unsigned int track() const override;
|
||||||
|
|
||||||
/*!
|
PictureMap pictures() const override;
|
||||||
* Not supported by module files. Therefore always returns 0.
|
|
||||||
*/
|
|
||||||
virtual unsigned int track() const;
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the name of the tracker used to create/edit the module file.
|
* Returns the name of the tracker used to create/edit the module file.
|
||||||
* Only XM files store this tag to the file as such, for other formats
|
* Only XM files store this tag to the file as such, for other formats
|
||||||
* (Mod, S3M, IT) this is derived from the file type or the flavour of
|
* (Mod, S3M, IT) this is derived from the file type or the flavour of the file type.
|
||||||
* the file type. Therefore only XM files might have an empty
|
* Therefore only XM files might have an empty (String::null) tracker name.
|
||||||
* (String::null) tracker name.
|
*/
|
||||||
*/
|
String trackerName() const;
|
||||||
String trackerName() const;
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Sets the title to \a title. If \a title is String::null then this
|
* Sets the title to \a title.
|
||||||
* value will be cleared.
|
* If \a title is String::null then this value will be cleared.
|
||||||
*
|
*
|
||||||
* The length limits per file type are (1 character = 1 byte):
|
* The length limits per file type are (1 character = 1 byte):
|
||||||
* Mod 20 characters, S3M 27 characters, IT 25 characters and XM 20
|
* Mod 20 characters, S3M 27 characters, IT 25 characters and XM 20 characters.
|
||||||
* characters.
|
*/
|
||||||
*/
|
void setTitle(const String &title) override;
|
||||||
virtual void setTitle(const String &title);
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Not supported by module files and therefore ignored.
|
* Not supported by module files and therefore ignored.
|
||||||
*/
|
*/
|
||||||
virtual void setArtist(const String &artist);
|
void setArtist(const String &artist) override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Not supported by module files and therefore ignored.
|
* Not supported by module files and therefore ignored.
|
||||||
*/
|
*/
|
||||||
virtual void setAlbum(const String &album);
|
void setAlbum(const String &album) override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Sets the comment to \a comment. If \a comment is String::null then
|
* Sets the comment to \a comment.
|
||||||
* this value will be cleared.
|
* If \a comment is String::null then this value will be cleared.
|
||||||
*
|
*
|
||||||
* Note that module file formats don't actually support a comment tag.
|
* Note that module file formats don't actually support a comment tag.
|
||||||
* Instead the names of instruments/patterns/samples are abused as
|
* Instead the names of instruments/patterns/samples are abused as a multiline comment.
|
||||||
* a multiline comment. Because of this the number of lines in a
|
* Because of this the number of lines in a module file is fixed to the number of instruments/patterns/samples.
|
||||||
* module file is fixed to the number of instruments/patterns/samples.
|
*
|
||||||
*
|
* Also note that the instrument/pattern/sample name length is limited an thus the line length in comments are limited.
|
||||||
* Also note that the instrument/pattern/sample name length is limited
|
* Too big comments will be truncated.
|
||||||
* an thus the line length in comments are limited. Too big comments
|
*
|
||||||
* will be truncated.
|
* The line length limits per file type are (1 character = 1 byte):
|
||||||
*
|
* Mod 22 characters, S3M 27 characters, IT 25 characters and XM 22 characters.
|
||||||
* The line length limits per file type are (1 character = 1 byte):
|
*/
|
||||||
* Mod 22 characters, S3M 27 characters, IT 25 characters and XM 22
|
void setComment(const String &comment) override;
|
||||||
* characters.
|
|
||||||
*/
|
|
||||||
virtual void setComment(const String &comment);
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Not supported by module files and therefore ignored.
|
* Not supported by module files and therefore ignored.
|
||||||
*/
|
*/
|
||||||
virtual void setGenre(const String &genre);
|
void setGenre(const String &genre) override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Not supported by module files and therefore ignored.
|
* Not supported by module files and therefore ignored.
|
||||||
*/
|
*/
|
||||||
virtual void setYear(unsigned int year);
|
void setYear(unsigned int year) override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Not supported by module files and therefore ignored.
|
* Not supported by module files and therefore ignored.
|
||||||
*/
|
*/
|
||||||
virtual void setTrack(unsigned int track);
|
void setTrack(unsigned int track) override;
|
||||||
|
|
||||||
/*!
|
void setPictures(const PictureMap &l) override;
|
||||||
* Sets the tracker name to \a trackerName. If \a trackerName is
|
|
||||||
* String::null then this value will be cleared.
|
|
||||||
*
|
|
||||||
* Note that only XM files support this tag. Setting the
|
|
||||||
* tracker name for other module file formats will be ignored.
|
|
||||||
*
|
|
||||||
* The length of this tag is limited to 20 characters (1 character
|
|
||||||
* = 1 byte).
|
|
||||||
*/
|
|
||||||
void setTrackerName(const String &trackerName);
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Implements the unified property interface -- export function.
|
* Sets the tracker name to \a trackerName.
|
||||||
* Since the module tag is very limited, the exported map is as well.
|
* If \a trackerName is String::null then this value will be cleared.
|
||||||
*/
|
*
|
||||||
PropertyMap properties() const;
|
* Note that only XM files support this tag.
|
||||||
|
* Setting the tracker name for other module file formats will be ignored.
|
||||||
|
*
|
||||||
|
* The length of this tag is limited to 20 characters (1 character = 1 byte).
|
||||||
|
*/
|
||||||
|
void setTrackerName(const String &trackerName);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Implements the unified property interface -- import function.
|
* Implements the unified property interface -- export function.
|
||||||
* Because of the limitations of the module file tag, any tags besides
|
* Since the module tag is very limited, the exported map is as well.
|
||||||
* COMMENT, TITLE and, if it is an XM file, TRACKERNAME, will be
|
*/
|
||||||
* returned. Additionally, if the map contains tags with multiple values,
|
PropertyMap properties() const override;
|
||||||
* all but the first will be contained in the returned map of unsupported
|
|
||||||
* properties.
|
|
||||||
*/
|
|
||||||
PropertyMap setProperties(const PropertyMap &);
|
|
||||||
|
|
||||||
private:
|
/*!
|
||||||
Tag(const Tag &);
|
* Implements the unified property interface -- import function.
|
||||||
Tag &operator=(const Tag &);
|
* Because of the limitations of the module file tag, any tags besides COMMENT, TITLE and, if it is an XM file, TRACKERNAME, will be returned.
|
||||||
|
* Additionally, if the map contains tags with multiple values, all but the first will be contained in the returned map of unsupported properties.
|
||||||
|
*/
|
||||||
|
PropertyMap setProperties(const PropertyMap &) override;
|
||||||
|
|
||||||
class TagPrivate;
|
private:
|
||||||
TagPrivate *d;
|
Tag(const Tag&);
|
||||||
};
|
Tag &operator=(const Tag&);
|
||||||
|
|
||||||
}
|
class TagPrivate;
|
||||||
|
TagPrivate *d;
|
||||||
|
};
|
||||||
|
|
||||||
}
|
} // namespace Mod
|
||||||
}
|
} // namespace TagLib
|
||||||
|
} // namespace Strawberry_TagLib
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
124
3rdparty/taglib/mp4/mp4atom.cpp
vendored
124
3rdparty/taglib/mp4/mp4atom.cpp
vendored
@@ -25,25 +25,21 @@
|
|||||||
|
|
||||||
#include <climits>
|
#include <climits>
|
||||||
|
|
||||||
#include <tdebug.h>
|
#include "tdebug.h"
|
||||||
#include <tstring.h>
|
#include "tstring.h"
|
||||||
#include "mp4atom.h"
|
#include "mp4atom.h"
|
||||||
|
|
||||||
using namespace Strawberry_TagLib::TagLib;
|
using namespace Strawberry_TagLib::TagLib;
|
||||||
|
|
||||||
const char *MP4::Atom::containers[11] = {
|
const char *MP4::Atom::containers[11] = { "moov", "udta", "mdia", "meta", "ilst", "stbl", "minf", "moof", "traf", "trak", "stsd" };
|
||||||
"moov", "udta", "mdia", "meta", "ilst",
|
|
||||||
"stbl", "minf", "moof", "traf", "trak",
|
MP4::Atom::Atom(File *file) {
|
||||||
"stsd"
|
|
||||||
};
|
|
||||||
|
|
||||||
MP4::Atom::Atom(File *file)
|
|
||||||
{
|
|
||||||
children.setAutoDelete(true);
|
children.setAutoDelete(true);
|
||||||
|
|
||||||
offset = file->tell();
|
offset = file->tell();
|
||||||
ByteVector header = file->readBlock(8);
|
ByteVector header = file->readBlock(8);
|
||||||
if(header.size() != 8) {
|
if (header.size() != 8) {
|
||||||
// The atom header must be 8 bytes long, otherwise there is either
|
// The atom header must be 8 bytes long, otherwise there is either
|
||||||
// trailing garbage or the file is truncated
|
// trailing garbage or the file is truncated
|
||||||
debug("MP4: Couldn't read 8 bytes of data for atom header");
|
debug("MP4: Couldn't read 8 bytes of data for atom header");
|
||||||
@@ -52,28 +48,18 @@ MP4::Atom::Atom(File *file)
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
length = header.toUInt();
|
length = header.toUInt32BE(0);
|
||||||
|
|
||||||
if(length == 0) {
|
if (length == 0) {
|
||||||
// The last atom which extends to the end of the file.
|
// The last atom which extends to the end of the file.
|
||||||
length = file->length() - offset;
|
length = file->length() - offset;
|
||||||
}
|
}
|
||||||
else if(length == 1) {
|
else if (length == 1) {
|
||||||
// The atom has a 64-bit length.
|
// The atom has a 64-bit length.
|
||||||
const long long longLength = file->readBlock(8).toLongLong();
|
length = file->readBlock(8).toInt64BE(0);
|
||||||
if(longLength <= LONG_MAX) {
|
|
||||||
// The actual length fits in long. That's always the case if long is 64-bit.
|
|
||||||
length = static_cast<long>(longLength);
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
debug("MP4: 64-bit atoms are not supported");
|
|
||||||
length = 0;
|
|
||||||
file->seek(0, File::End);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if(length < 8) {
|
if (length < 8) {
|
||||||
debug("MP4: Invalid atom size");
|
debug("MP4: Invalid atom size");
|
||||||
length = 0;
|
length = 0;
|
||||||
file->seek(0, File::End);
|
file->seek(0, File::End);
|
||||||
@@ -82,18 +68,18 @@ MP4::Atom::Atom(File *file)
|
|||||||
|
|
||||||
name = header.mid(4, 4);
|
name = header.mid(4, 4);
|
||||||
|
|
||||||
for(int i = 0; i < numContainers; i++) {
|
for (int i = 0; i < numContainers; i++) {
|
||||||
if(name == containers[i]) {
|
if (name == containers[i]) {
|
||||||
if(name == "meta") {
|
if (name == "meta") {
|
||||||
file->seek(4, File::Current);
|
file->seek(4, File::Current);
|
||||||
}
|
}
|
||||||
else if(name == "stsd") {
|
else if (name == "stsd") {
|
||||||
file->seek(8, File::Current);
|
file->seek(8, File::Current);
|
||||||
}
|
}
|
||||||
while(file->tell() < offset + length) {
|
while (file->tell() < offset + length) {
|
||||||
MP4::Atom *child = new MP4::Atom(file);
|
MP4::Atom *child = new MP4::Atom(file);
|
||||||
children.append(child);
|
children.append(child);
|
||||||
if(child->length == 0)
|
if (child->length == 0)
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
return;
|
return;
|
||||||
@@ -101,97 +87,95 @@ MP4::Atom::Atom(File *file)
|
|||||||
}
|
}
|
||||||
|
|
||||||
file->seek(offset + length);
|
file->seek(offset + length);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
MP4::Atom::~Atom()
|
MP4::Atom::~Atom() {}
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
MP4::Atom *
|
MP4::Atom *MP4::Atom::find(const char *name1, const char *name2, const char *name3, const char *name4) {
|
||||||
MP4::Atom::find(const char *name1, const char *name2, const char *name3, const char *name4)
|
|
||||||
{
|
if (!name1) {
|
||||||
if(name1 == 0) {
|
|
||||||
return this;
|
return this;
|
||||||
}
|
}
|
||||||
for(AtomList::ConstIterator it = children.begin(); it != children.end(); ++it) {
|
for (AtomList::ConstIterator it = children.begin(); it != children.end(); ++it) {
|
||||||
if((*it)->name == name1) {
|
if ((*it)->name == name1) {
|
||||||
return (*it)->find(name2, name3, name4);
|
return (*it)->find(name2, name3, name4);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return nullptr;
|
return nullptr;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
MP4::AtomList
|
MP4::AtomList MP4::Atom::findall(const char *_name, bool recursive) {
|
||||||
MP4::Atom::findall(const char *_name, bool recursive)
|
|
||||||
{
|
|
||||||
MP4::AtomList result;
|
MP4::AtomList result;
|
||||||
for(AtomList::ConstIterator it = children.begin(); it != children.end(); ++it) {
|
for (AtomList::ConstIterator it = children.begin(); it != children.end(); ++it) {
|
||||||
if((*it)->name == _name) {
|
if ((*it)->name == _name) {
|
||||||
result.append(*it);
|
result.append(*it);
|
||||||
}
|
}
|
||||||
if(recursive) {
|
if (recursive) {
|
||||||
result.append((*it)->findall(_name, recursive));
|
result.append((*it)->findall(_name, recursive));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return result;
|
return result;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool
|
bool MP4::Atom::path(MP4::AtomList &path, const char *name1, const char *name2, const char *name3) {
|
||||||
MP4::Atom::path(MP4::AtomList &path, const char *name1, const char *name2, const char *name3)
|
|
||||||
{
|
|
||||||
path.append(this);
|
path.append(this);
|
||||||
if(name1 == 0) {
|
if (!name1) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
for(AtomList::ConstIterator it = children.begin(); it != children.end(); ++it) {
|
for (AtomList::ConstIterator it = children.begin(); it != children.end(); ++it) {
|
||||||
if((*it)->name == name1) {
|
if ((*it)->name == name1) {
|
||||||
return (*it)->path(path, name2, name3);
|
return (*it)->path(path, name2, name3);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
MP4::Atoms::Atoms(File *file)
|
MP4::Atoms::Atoms(File *file) {
|
||||||
{
|
|
||||||
atoms.setAutoDelete(true);
|
atoms.setAutoDelete(true);
|
||||||
|
|
||||||
file->seek(0, File::End);
|
file->seek(0, File::End);
|
||||||
long end = file->tell();
|
long long end = file->tell();
|
||||||
file->seek(0);
|
file->seek(0);
|
||||||
while(file->tell() + 8 <= end) {
|
while (file->tell() + 8 <= end) {
|
||||||
MP4::Atom *atom = new MP4::Atom(file);
|
MP4::Atom *atom = new MP4::Atom(file);
|
||||||
atoms.append(atom);
|
atoms.append(atom);
|
||||||
if (atom->length == 0)
|
if (atom->length == 0)
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
MP4::Atoms::~Atoms()
|
MP4::Atoms::~Atoms() {}
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
MP4::Atom *
|
MP4::Atom *MP4::Atoms::find(const char *name1, const char *name2, const char *name3, const char *name4) {
|
||||||
MP4::Atoms::find(const char *name1, const char *name2, const char *name3, const char *name4)
|
|
||||||
{
|
for (AtomList::ConstIterator it = atoms.begin(); it != atoms.end(); ++it) {
|
||||||
for(AtomList::ConstIterator it = atoms.begin(); it != atoms.end(); ++it) {
|
if ((*it)->name == name1) {
|
||||||
if((*it)->name == name1) {
|
|
||||||
return (*it)->find(name2, name3, name4);
|
return (*it)->find(name2, name3, name4);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return nullptr;
|
return nullptr;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
MP4::AtomList
|
MP4::AtomList MP4::Atoms::path(const char *name1, const char *name2, const char *name3, const char *name4) {
|
||||||
MP4::Atoms::path(const char *name1, const char *name2, const char *name3, const char *name4)
|
|
||||||
{
|
|
||||||
MP4::AtomList path;
|
MP4::AtomList path;
|
||||||
for(AtomList::ConstIterator it = atoms.begin(); it != atoms.end(); ++it) {
|
for (AtomList::ConstIterator it = atoms.begin(); it != atoms.end(); ++it) {
|
||||||
if((*it)->name == name1) {
|
if ((*it)->name == name1) {
|
||||||
if(!(*it)->path(path, name2, name3, name4)) {
|
if (!(*it)->path(path, name2, name3, name4)) {
|
||||||
path.clear();
|
path.clear();
|
||||||
}
|
}
|
||||||
return path;
|
return path;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return path;
|
return path;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
136
3rdparty/taglib/mp4/mp4atom.h
vendored
136
3rdparty/taglib/mp4/mp4atom.h
vendored
@@ -27,87 +27,85 @@
|
|||||||
|
|
||||||
#ifndef DO_NOT_DOCUMENT
|
#ifndef DO_NOT_DOCUMENT
|
||||||
|
|
||||||
#ifndef TAGLIB_MP4ATOM_H
|
# ifndef TAGLIB_MP4ATOM_H
|
||||||
#define TAGLIB_MP4ATOM_H
|
# define TAGLIB_MP4ATOM_H
|
||||||
|
|
||||||
#include "tfile.h"
|
# include "tfile.h"
|
||||||
#include "tlist.h"
|
# include "tlist.h"
|
||||||
|
|
||||||
namespace Strawberry_TagLib {
|
namespace Strawberry_TagLib {
|
||||||
namespace TagLib {
|
namespace TagLib {
|
||||||
|
namespace MP4 {
|
||||||
|
|
||||||
namespace MP4 {
|
class Atom;
|
||||||
|
typedef Strawberry_TagLib::TagLib::List<Atom *> AtomList;
|
||||||
|
|
||||||
class Atom;
|
enum AtomDataType {
|
||||||
typedef Strawberry_TagLib::TagLib::List<Atom *> AtomList;
|
TypeImplicit = 0, // for use with tags for which no type needs to be indicated because only one type is allowed
|
||||||
|
TypeUTF8 = 1, // without any count or null terminator
|
||||||
|
TypeUTF16 = 2, // also known as UTF-16BE
|
||||||
|
TypeSJIS = 3, // deprecated unless it is needed for special Japanese characters
|
||||||
|
TypeHTML = 6, // the HTML file header specifies which HTML version
|
||||||
|
TypeXML = 7, // the XML header must identify the DTD or schemas
|
||||||
|
TypeUUID = 8, // also known as GUID; stored as 16 bytes in binary (valid as an ID)
|
||||||
|
TypeISRC = 9, // stored as UTF-8 text (valid as an ID)
|
||||||
|
TypeMI3P = 10, // stored as UTF-8 text (valid as an ID)
|
||||||
|
TypeGIF = 12, // (deprecated) a GIF image
|
||||||
|
TypeJPEG = 13, // a JPEG image
|
||||||
|
TypePNG = 14, // a PNG image
|
||||||
|
TypeURL = 15, // absolute, in UTF-8 characters
|
||||||
|
TypeDuration = 16, // in milliseconds, 32-bit integer
|
||||||
|
TypeDateTime = 17, // in UTC, counting seconds since midnight, January 1, 1904; 32 or 64-bits
|
||||||
|
TypeGenred = 18, // a list of enumerated values
|
||||||
|
TypeInteger = 21, // a signed big-endian integer with length one of { 1,2,3,4,8 } bytes
|
||||||
|
TypeRIAAPA = 24, // RIAA parental advisory; { -1=no, 1=yes, 0=unspecified }, 8-bit integer
|
||||||
|
TypeUPC = 25, // Universal Product Code, in text UTF-8 format (valid as an ID)
|
||||||
|
TypeBMP = 27, // Windows bitmap image
|
||||||
|
TypeUndefined = 255 // undefined
|
||||||
|
};
|
||||||
|
|
||||||
enum AtomDataType
|
struct AtomData {
|
||||||
{
|
explicit AtomData(AtomDataType _type, ByteVector _data) : type(_type), locale(0), data(_data) {}
|
||||||
TypeImplicit = 0, // for use with tags for which no type needs to be indicated because only one type is allowed
|
AtomDataType type;
|
||||||
TypeUTF8 = 1, // without any count or null terminator
|
int locale;
|
||||||
TypeUTF16 = 2, // also known as UTF-16BE
|
ByteVector data;
|
||||||
TypeSJIS = 3, // deprecated unless it is needed for special Japanese characters
|
};
|
||||||
TypeHTML = 6, // the HTML file header specifies which HTML version
|
|
||||||
TypeXML = 7, // the XML header must identify the DTD or schemas
|
|
||||||
TypeUUID = 8, // also known as GUID; stored as 16 bytes in binary (valid as an ID)
|
|
||||||
TypeISRC = 9, // stored as UTF-8 text (valid as an ID)
|
|
||||||
TypeMI3P = 10, // stored as UTF-8 text (valid as an ID)
|
|
||||||
TypeGIF = 12, // (deprecated) a GIF image
|
|
||||||
TypeJPEG = 13, // a JPEG image
|
|
||||||
TypePNG = 14, // a PNG image
|
|
||||||
TypeURL = 15, // absolute, in UTF-8 characters
|
|
||||||
TypeDuration = 16, // in milliseconds, 32-bit integer
|
|
||||||
TypeDateTime = 17, // in UTC, counting seconds since midnight, January 1, 1904; 32 or 64-bits
|
|
||||||
TypeGenred = 18, // a list of enumerated values
|
|
||||||
TypeInteger = 21, // a signed big-endian integer with length one of { 1,2,3,4,8 } bytes
|
|
||||||
TypeRIAAPA = 24, // RIAA parental advisory; { -1=no, 1=yes, 0=unspecified }, 8-bit integer
|
|
||||||
TypeUPC = 25, // Universal Product Code, in text UTF-8 format (valid as an ID)
|
|
||||||
TypeBMP = 27, // Windows bitmap image
|
|
||||||
TypeUndefined = 255 // undefined
|
|
||||||
};
|
|
||||||
|
|
||||||
struct AtomData {
|
typedef Strawberry_TagLib::TagLib::List<AtomData> AtomDataList;
|
||||||
AtomData(AtomDataType _type, ByteVector _data) : type(_type), locale(0), data(_data) {}
|
|
||||||
AtomDataType type;
|
|
||||||
int locale;
|
|
||||||
ByteVector data;
|
|
||||||
};
|
|
||||||
|
|
||||||
typedef Strawberry_TagLib::TagLib::List<AtomData> AtomDataList;
|
class Atom {
|
||||||
|
public:
|
||||||
|
explicit Atom(File *file);
|
||||||
|
~Atom();
|
||||||
|
|
||||||
class Atom
|
Atom *find(const char *name1, const char *name2 = nullptr, const char *name3 = nullptr, const char *name4 = nullptr);
|
||||||
{
|
bool path(AtomList &path, const char *name1, const char *name2 = nullptr, const char *name3 = nullptr);
|
||||||
public:
|
AtomList findall(const char *name, bool recursive = false);
|
||||||
Atom(File *file);
|
long long offset;
|
||||||
~Atom();
|
long long length;
|
||||||
Atom *find(const char *name1, const char *name2 = 0, const char *name3 = 0, const char *name4 = 0);
|
Strawberry_TagLib::TagLib::ByteVector name;
|
||||||
bool path(AtomList &path, const char *name1, const char *name2 = 0, const char *name3 = 0);
|
AtomList children;
|
||||||
AtomList findall(const char *name, bool recursive = false);
|
|
||||||
long offset;
|
|
||||||
long length;
|
|
||||||
Strawberry_TagLib::TagLib::ByteVector name;
|
|
||||||
AtomList children;
|
|
||||||
private:
|
|
||||||
static const int numContainers = 11;
|
|
||||||
static const char *containers[11];
|
|
||||||
};
|
|
||||||
|
|
||||||
//! Root-level atoms
|
private:
|
||||||
class Atoms
|
static const int numContainers = 11;
|
||||||
{
|
static const char *containers[11];
|
||||||
public:
|
};
|
||||||
Atoms(File *file);
|
|
||||||
~Atoms();
|
|
||||||
Atom *find(const char *name1, const char *name2 = 0, const char *name3 = 0, const char *name4 = 0);
|
|
||||||
AtomList path(const char *name1, const char *name2 = 0, const char *name3 = 0, const char *name4 = 0);
|
|
||||||
AtomList atoms;
|
|
||||||
};
|
|
||||||
|
|
||||||
}
|
//! Root-level atoms
|
||||||
|
class Atoms {
|
||||||
|
public:
|
||||||
|
explicit Atoms(File *file);
|
||||||
|
~Atoms();
|
||||||
|
|
||||||
}
|
Atom *find(const char *name1, const char *name2 = nullptr, const char *name3 = nullptr, const char *name4 = nullptr);
|
||||||
}
|
AtomList path(const char *name1, const char *name2 = nullptr, const char *name3 = nullptr, const char *name4 = nullptr);
|
||||||
|
AtomList atoms;
|
||||||
#endif
|
};
|
||||||
|
|
||||||
|
} // namespace MP4
|
||||||
|
} // namespace TagLib
|
||||||
|
} // namespace Strawberry_TagLib
|
||||||
|
|
||||||
|
# endif
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
66
3rdparty/taglib/mp4/mp4coverart.cpp
vendored
66
3rdparty/taglib/mp4/mp4coverart.cpp
vendored
@@ -23,71 +23,63 @@
|
|||||||
* http://www.mozilla.org/MPL/ *
|
* http://www.mozilla.org/MPL/ *
|
||||||
***************************************************************************/
|
***************************************************************************/
|
||||||
|
|
||||||
#include <taglib.h>
|
#include <memory>
|
||||||
#include <tdebug.h>
|
|
||||||
#include "trefcounter.h"
|
#include "taglib.h"
|
||||||
|
#include "tdebug.h"
|
||||||
#include "mp4coverart.h"
|
#include "mp4coverart.h"
|
||||||
|
|
||||||
using namespace Strawberry_TagLib::TagLib;
|
using namespace Strawberry_TagLib::TagLib;
|
||||||
|
|
||||||
class MP4::CoverArt::CoverArtPrivate : public RefCounter
|
namespace {
|
||||||
{
|
|
||||||
public:
|
|
||||||
CoverArtPrivate() :
|
|
||||||
RefCounter(),
|
|
||||||
format(MP4::CoverArt::JPEG) {}
|
|
||||||
|
|
||||||
Format format;
|
struct CoverArtData {
|
||||||
|
MP4::CoverArt::Format format;
|
||||||
ByteVector data;
|
ByteVector data;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
} // namespace
|
||||||
|
|
||||||
|
class MP4::CoverArt::CoverArtPrivate {
|
||||||
|
public:
|
||||||
|
explicit CoverArtPrivate(Format f, const ByteVector &v) : data(new CoverArtData()) {
|
||||||
|
data->format = f;
|
||||||
|
data->data = v;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::shared_ptr<CoverArtData> data;
|
||||||
|
};
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
// public members
|
// public members
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
MP4::CoverArt::CoverArt(Format format, const ByteVector &data) :
|
MP4::CoverArt::CoverArt(Format format, const ByteVector &data) : d(new CoverArtPrivate(format, data)) {}
|
||||||
d(new CoverArtPrivate())
|
|
||||||
{
|
|
||||||
d->format = format;
|
|
||||||
d->data = data;
|
|
||||||
}
|
|
||||||
|
|
||||||
MP4::CoverArt::CoverArt(const CoverArt &item) :
|
MP4::CoverArt::CoverArt(const CoverArt &item) : d(new CoverArtPrivate(*item.d)) {}
|
||||||
d(item.d)
|
|
||||||
{
|
|
||||||
d->ref();
|
|
||||||
}
|
|
||||||
|
|
||||||
MP4::CoverArt &
|
MP4::CoverArt &
|
||||||
MP4::CoverArt::operator=(const CoverArt &item)
|
MP4::CoverArt::operator=(const CoverArt &item) {
|
||||||
{
|
|
||||||
CoverArt(item).swap(*this);
|
CoverArt(item).swap(*this);
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void MP4::CoverArt::swap(CoverArt &item) {
|
||||||
MP4::CoverArt::swap(CoverArt &item)
|
|
||||||
{
|
|
||||||
using std::swap;
|
using std::swap;
|
||||||
|
|
||||||
swap(d, item.d);
|
swap(d, item.d);
|
||||||
}
|
}
|
||||||
|
|
||||||
MP4::CoverArt::~CoverArt()
|
MP4::CoverArt::~CoverArt() {
|
||||||
{
|
delete d;
|
||||||
if(d->deref()) {
|
|
||||||
delete d;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
MP4::CoverArt::Format
|
MP4::CoverArt::Format
|
||||||
MP4::CoverArt::format() const
|
MP4::CoverArt::format() const {
|
||||||
{
|
return d->data->format;
|
||||||
return d->format;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ByteVector
|
ByteVector
|
||||||
MP4::CoverArt::data() const
|
MP4::CoverArt::data() const {
|
||||||
{
|
return d->data->data;
|
||||||
return d->data;
|
|
||||||
}
|
}
|
||||||
|
|||||||
75
3rdparty/taglib/mp4/mp4coverart.h
vendored
75
3rdparty/taglib/mp4/mp4coverart.h
vendored
@@ -33,54 +33,51 @@
|
|||||||
|
|
||||||
namespace Strawberry_TagLib {
|
namespace Strawberry_TagLib {
|
||||||
namespace TagLib {
|
namespace TagLib {
|
||||||
|
namespace MP4 {
|
||||||
|
|
||||||
namespace MP4 {
|
class TAGLIB_EXPORT CoverArt {
|
||||||
|
public:
|
||||||
|
/*!
|
||||||
|
* This describes the image type.
|
||||||
|
*/
|
||||||
|
enum Format {
|
||||||
|
JPEG = TypeJPEG,
|
||||||
|
PNG = TypePNG,
|
||||||
|
BMP = TypeBMP,
|
||||||
|
GIF = TypeGIF,
|
||||||
|
Unknown = TypeImplicit,
|
||||||
|
};
|
||||||
|
|
||||||
class TAGLIB_EXPORT CoverArt
|
explicit CoverArt(Format format, const ByteVector &data);
|
||||||
{
|
~CoverArt();
|
||||||
public:
|
|
||||||
/*!
|
|
||||||
* This describes the image type.
|
|
||||||
*/
|
|
||||||
enum Format {
|
|
||||||
JPEG = TypeJPEG,
|
|
||||||
PNG = TypePNG,
|
|
||||||
BMP = TypeBMP,
|
|
||||||
GIF = TypeGIF,
|
|
||||||
Unknown = TypeImplicit,
|
|
||||||
};
|
|
||||||
|
|
||||||
CoverArt(Format format, const ByteVector &data);
|
CoverArt(const CoverArt &item);
|
||||||
~CoverArt();
|
|
||||||
|
|
||||||
CoverArt(const CoverArt &item);
|
/*!
|
||||||
|
* Copies the contents of \a item into this CoverArt.
|
||||||
|
*/
|
||||||
|
CoverArt &operator=(const CoverArt &item);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Copies the contents of \a item into this CoverArt.
|
* Exchanges the content of the CoverArt by the content of \a item.
|
||||||
*/
|
*/
|
||||||
CoverArt &operator=(const CoverArt &item);
|
void swap(CoverArt &item);
|
||||||
|
|
||||||
/*!
|
//! Format of the image
|
||||||
* Exchanges the content of the CoverArt by the content of \a item.
|
Format format() const;
|
||||||
*/
|
|
||||||
void swap(CoverArt &item);
|
|
||||||
|
|
||||||
//! Format of the image
|
//! The image data
|
||||||
Format format() const;
|
ByteVector data() const;
|
||||||
|
|
||||||
//! The image data
|
private:
|
||||||
ByteVector data() const;
|
class CoverArtPrivate;
|
||||||
|
CoverArtPrivate *d;
|
||||||
|
};
|
||||||
|
|
||||||
private:
|
typedef List<CoverArt> CoverArtList;
|
||||||
class CoverArtPrivate;
|
|
||||||
CoverArtPrivate *d;
|
|
||||||
};
|
|
||||||
|
|
||||||
typedef List<CoverArt> CoverArtList;
|
} // namespace MP4
|
||||||
|
} // namespace TagLib
|
||||||
}
|
} // namespace Strawberry_TagLib
|
||||||
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
130
3rdparty/taglib/mp4/mp4file.cpp
vendored
130
3rdparty/taglib/mp4/mp4file.cpp
vendored
@@ -23,10 +23,10 @@
|
|||||||
* http://www.mozilla.org/MPL/ *
|
* http://www.mozilla.org/MPL/ *
|
||||||
***************************************************************************/
|
***************************************************************************/
|
||||||
|
|
||||||
#include <tdebug.h>
|
#include "tdebug.h"
|
||||||
#include <tstring.h>
|
#include "tstring.h"
|
||||||
#include <tpropertymap.h>
|
#include "tpropertymap.h"
|
||||||
#include <tagutils.h>
|
#include "tagutils.h"
|
||||||
|
|
||||||
#include "mp4atom.h"
|
#include "mp4atom.h"
|
||||||
#include "mp4tag.h"
|
#include "mp4tag.h"
|
||||||
@@ -34,149 +34,123 @@
|
|||||||
|
|
||||||
using namespace Strawberry_TagLib::TagLib;
|
using namespace Strawberry_TagLib::TagLib;
|
||||||
|
|
||||||
namespace
|
namespace {
|
||||||
{
|
bool checkValid(const MP4::AtomList &list) {
|
||||||
bool checkValid(const MP4::AtomList &list)
|
|
||||||
{
|
|
||||||
for(MP4::AtomList::ConstIterator it = list.begin(); it != list.end(); ++it) {
|
|
||||||
|
|
||||||
if((*it)->length == 0)
|
for (MP4::AtomList::ConstIterator it = list.begin(); it != list.end(); ++it) {
|
||||||
return false;
|
|
||||||
|
|
||||||
if(!checkValid((*it)->children))
|
if ((*it)->length == 0)
|
||||||
return false;
|
return false;
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
if (!checkValid((*it)->children))
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
} // namespace
|
||||||
|
|
||||||
class MP4::File::FilePrivate
|
class MP4::File::FilePrivate {
|
||||||
{
|
public:
|
||||||
public:
|
explicit FilePrivate() : tag(nullptr), atoms(nullptr), properties(nullptr) {}
|
||||||
FilePrivate() :
|
|
||||||
tag(0),
|
|
||||||
atoms(0),
|
|
||||||
properties(0) {}
|
|
||||||
|
|
||||||
~FilePrivate()
|
~FilePrivate() {
|
||||||
{
|
|
||||||
delete atoms;
|
delete atoms;
|
||||||
delete tag;
|
delete tag;
|
||||||
delete properties;
|
delete properties;
|
||||||
}
|
}
|
||||||
|
|
||||||
MP4::Tag *tag;
|
MP4::Tag *tag;
|
||||||
MP4::Atoms *atoms;
|
MP4::Atoms *atoms;
|
||||||
MP4::Properties *properties;
|
MP4::AudioProperties *properties;
|
||||||
};
|
};
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
// static members
|
// static members
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
bool MP4::File::isSupported(IOStream *stream)
|
bool MP4::File::isSupported(IOStream *stream) {
|
||||||
{
|
|
||||||
// An MP4 file has to have an "ftyp" box first.
|
// An MP4 file has to have an "ftyp" box first.
|
||||||
|
|
||||||
const ByteVector id = Utils::readHeader(stream, 8, false);
|
const ByteVector id = Utils::readHeader(stream, 8, false);
|
||||||
return id.containsAt("ftyp", 4);
|
return id.containsAt("ftyp", 4);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
// public members
|
// public members
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
MP4::File::File(FileName file, bool readProperties, AudioProperties::ReadStyle) :
|
MP4::File::File(FileName file, bool readProperties, AudioProperties::ReadStyle) : Strawberry_TagLib::TagLib::File(file), d(new FilePrivate()) {
|
||||||
Strawberry_TagLib::TagLib::File(file),
|
|
||||||
d(new FilePrivate())
|
if (isOpen())
|
||||||
{
|
|
||||||
if(isOpen())
|
|
||||||
read(readProperties);
|
read(readProperties);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
MP4::File::File(IOStream *stream, bool readProperties, AudioProperties::ReadStyle) :
|
MP4::File::File(IOStream *stream, bool readProperties, AudioProperties::ReadStyle) : Strawberry_TagLib::TagLib::File(stream), d(new FilePrivate()) {
|
||||||
Strawberry_TagLib::TagLib::File(stream),
|
|
||||||
d(new FilePrivate())
|
if (isOpen())
|
||||||
{
|
|
||||||
if(isOpen())
|
|
||||||
read(readProperties);
|
read(readProperties);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
MP4::File::~File()
|
MP4::File::~File() {
|
||||||
{
|
|
||||||
delete d;
|
delete d;
|
||||||
}
|
}
|
||||||
|
|
||||||
MP4::Tag *
|
MP4::Tag *
|
||||||
MP4::File::tag() const
|
MP4::File::tag() const {
|
||||||
{
|
|
||||||
return d->tag;
|
return d->tag;
|
||||||
}
|
}
|
||||||
|
|
||||||
PropertyMap MP4::File::properties() const
|
MP4::AudioProperties *
|
||||||
{
|
MP4::File::audioProperties() const {
|
||||||
return d->tag->properties();
|
|
||||||
}
|
|
||||||
|
|
||||||
void MP4::File::removeUnsupportedProperties(const StringList &properties)
|
|
||||||
{
|
|
||||||
d->tag->removeUnsupportedProperties(properties);
|
|
||||||
}
|
|
||||||
|
|
||||||
PropertyMap MP4::File::setProperties(const PropertyMap &properties)
|
|
||||||
{
|
|
||||||
return d->tag->setProperties(properties);
|
|
||||||
}
|
|
||||||
|
|
||||||
MP4::Properties *
|
|
||||||
MP4::File::audioProperties() const
|
|
||||||
{
|
|
||||||
return d->properties;
|
return d->properties;
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void MP4::File::read(bool readProperties) {
|
||||||
MP4::File::read(bool readProperties)
|
|
||||||
{
|
if (!isValid())
|
||||||
if(!isValid())
|
|
||||||
return;
|
return;
|
||||||
|
|
||||||
d->atoms = new Atoms(this);
|
d->atoms = new Atoms(this);
|
||||||
if(!checkValid(d->atoms->atoms)) {
|
if (!checkValid(d->atoms->atoms)) {
|
||||||
setValid(false);
|
setValid(false);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// must have a moov atom, otherwise consider it invalid
|
// must have a moov atom, otherwise consider it invalid
|
||||||
if(!d->atoms->find("moov")) {
|
if (!d->atoms->find("moov")) {
|
||||||
setValid(false);
|
setValid(false);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
d->tag = new Tag(this, d->atoms);
|
d->tag = new Tag(this, d->atoms);
|
||||||
if(readProperties) {
|
if (readProperties) {
|
||||||
d->properties = new Properties(this, d->atoms);
|
d->properties = new AudioProperties(this, d->atoms);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool
|
bool MP4::File::save() {
|
||||||
MP4::File::save()
|
|
||||||
{
|
if (readOnly()) {
|
||||||
if(readOnly()) {
|
|
||||||
debug("MP4::File::save() -- File is read only.");
|
debug("MP4::File::save() -- File is read only.");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if(!isValid()) {
|
if (!isValid()) {
|
||||||
debug("MP4::File::save() -- Trying to save invalid file.");
|
debug("MP4::File::save() -- Trying to save invalid file.");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
return d->tag->save();
|
return d->tag->save();
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool
|
bool MP4::File::hasMP4Tag() const {
|
||||||
MP4::File::hasMP4Tag() const
|
return (d->atoms->find("moov", "udta", "meta", "ilst") != nullptr);
|
||||||
{
|
|
||||||
return (d->atoms->find("moov", "udta", "meta", "ilst") != 0);
|
|
||||||
}
|
}
|
||||||
|
|||||||
163
3rdparty/taglib/mp4/mp4file.h
vendored
163
3rdparty/taglib/mp4/mp4file.h
vendored
@@ -35,111 +35,86 @@
|
|||||||
namespace Strawberry_TagLib {
|
namespace Strawberry_TagLib {
|
||||||
namespace TagLib {
|
namespace TagLib {
|
||||||
|
|
||||||
//! An implementation of MP4 (AAC, ALAC, ...) metadata
|
//! An implementation of MP4 (AAC, ALAC, ...) metadata
|
||||||
namespace MP4 {
|
namespace MP4 {
|
||||||
|
|
||||||
class Atoms;
|
class Atoms;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* This implements and provides an interface for MP4 files to the
|
* This implements and provides an interface for MP4 files to the
|
||||||
* TagLib::Tag and TagLib::AudioProperties interfaces by way of implementing
|
* TagLib::Tag and TagLib::AudioProperties interfaces by way of implementing
|
||||||
* the abstract TagLib::File API as well as providing some additional
|
* the abstract TagLib::File API as well as providing some additional
|
||||||
* information specific to MP4 files.
|
* information specific to MP4 files.
|
||||||
*/
|
*/
|
||||||
class TAGLIB_EXPORT File : public Strawberry_TagLib::TagLib::File
|
class TAGLIB_EXPORT File : public Strawberry_TagLib::TagLib::File {
|
||||||
{
|
public:
|
||||||
public:
|
/*!
|
||||||
/*!
|
* Constructs an MP4 file from \a file.
|
||||||
* Constructs an MP4 file from \a file. If \a readProperties is true the
|
* If \a readProperties is true the file's audio properties will also be read.
|
||||||
* file's audio properties will also be read.
|
*
|
||||||
*
|
* \note In the current implementation, \a propertiesStyle is ignored.
|
||||||
* \note In the current implementation, \a propertiesStyle is ignored.
|
*/
|
||||||
*/
|
explicit File(FileName file, bool readProperties = true, AudioProperties::ReadStyle audioPropertiesStyle = AudioProperties::Average);
|
||||||
File(FileName file, bool readProperties = true,
|
|
||||||
Properties::ReadStyle audioPropertiesStyle = Properties::Average);
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Constructs an MP4 file from \a stream. If \a readProperties is true the
|
* Constructs an MP4 file from \a stream.
|
||||||
* file's audio properties will also be read.
|
* If \a readProperties is true the file's audio properties will also be read.
|
||||||
*
|
*
|
||||||
* \note TagLib will *not* take ownership of the stream, the caller is
|
* \note TagLib will *not* take ownership of the stream, the caller is
|
||||||
* responsible for deleting it after the File object.
|
* responsible for deleting it after the File object.
|
||||||
*
|
*
|
||||||
* \note In the current implementation, \a propertiesStyle is ignored.
|
* \note In the current implementation, \a propertiesStyle is ignored.
|
||||||
*/
|
*/
|
||||||
File(IOStream *stream, bool readProperties = true,
|
explicit File(IOStream *stream, bool readProperties = true, AudioProperties::ReadStyle audioPropertiesStyle = AudioProperties::Average);
|
||||||
Properties::ReadStyle audioPropertiesStyle = Properties::Average);
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Destroys this instance of the File.
|
* Destroys this instance of the File.
|
||||||
*/
|
*/
|
||||||
virtual ~File();
|
~File() override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns a pointer to the MP4 tag of the file.
|
* Returns a pointer to the MP4 tag of the file.
|
||||||
*
|
*
|
||||||
* MP4::Tag implements the tag interface, so this serves as the
|
* MP4::Tag implements the tag interface, so this serves as the reimplementation of TagLib::File::tag().
|
||||||
* reimplementation of TagLib::File::tag().
|
*
|
||||||
*
|
* \note The Tag <b>is still</b> owned by the MP4::File and should not be deleted by the user.
|
||||||
* \note The Tag <b>is still</b> owned by the MP4::File and should not be
|
* It will be deleted when the file (object) is destroyed.
|
||||||
* deleted by the user. It will be deleted when the file (object) is
|
*/
|
||||||
* destroyed.
|
Tag *tag() const override;
|
||||||
*/
|
|
||||||
Tag *tag() const;
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Implements the unified property interface -- export function.
|
* Returns the MP4 audio properties for this file.
|
||||||
*/
|
*/
|
||||||
PropertyMap properties() const;
|
AudioProperties *audioProperties() const override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Removes unsupported properties. Forwards to the actual Tag's
|
* Save the file.
|
||||||
* removeUnsupportedProperties() function.
|
*
|
||||||
*/
|
* This returns true if the save was successful.
|
||||||
void removeUnsupportedProperties(const StringList &properties);
|
*/
|
||||||
|
bool save() override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Implements the unified property interface -- import function.
|
* Returns whether or not the file on disk actually has an MP4 tag, or the file has a Metadata Item List (ilst) atom.
|
||||||
*/
|
*/
|
||||||
PropertyMap setProperties(const PropertyMap &);
|
bool hasMP4Tag() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the MP4 audio properties for this file.
|
* Returns whether or not the given \a stream can be opened as an ASF file.
|
||||||
*/
|
*
|
||||||
Properties *audioProperties() const;
|
* \note This method is designed to do a quick check. The result may not necessarily be correct.
|
||||||
|
*/
|
||||||
|
static bool isSupported(IOStream *stream);
|
||||||
|
|
||||||
/*!
|
private:
|
||||||
* Save the file.
|
void read(bool readProperties);
|
||||||
*
|
|
||||||
* This returns true if the save was successful.
|
|
||||||
*/
|
|
||||||
bool save();
|
|
||||||
|
|
||||||
/*!
|
class FilePrivate;
|
||||||
* Returns whether or not the file on disk actually has an MP4 tag, or the
|
FilePrivate *d;
|
||||||
* file has a Metadata Item List (ilst) atom.
|
};
|
||||||
*/
|
|
||||||
bool hasMP4Tag() const;
|
|
||||||
|
|
||||||
/*!
|
} // namespace MP4
|
||||||
* Returns whether or not the given \a stream can be opened as an ASF
|
} // namespace TagLib
|
||||||
* file.
|
} // namespace Strawberry_TagLib
|
||||||
*
|
|
||||||
* \note This method is designed to do a quick check. The result may
|
|
||||||
* not necessarily be correct.
|
|
||||||
*/
|
|
||||||
static bool isSupported(IOStream *stream);
|
|
||||||
|
|
||||||
private:
|
|
||||||
void read(bool readProperties);
|
|
||||||
|
|
||||||
class FilePrivate;
|
|
||||||
FilePrivate *d;
|
|
||||||
};
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
231
3rdparty/taglib/mp4/mp4item.cpp
vendored
231
3rdparty/taglib/mp4/mp4item.cpp
vendored
@@ -23,27 +23,25 @@
|
|||||||
* http://www.mozilla.org/MPL/ *
|
* http://www.mozilla.org/MPL/ *
|
||||||
***************************************************************************/
|
***************************************************************************/
|
||||||
|
|
||||||
#include <taglib.h>
|
#include <memory>
|
||||||
#include <tdebug.h>
|
|
||||||
#include "trefcounter.h"
|
#include "taglib.h"
|
||||||
|
#include "tdebug.h"
|
||||||
#include "mp4item.h"
|
#include "mp4item.h"
|
||||||
|
#include "tutils.h"
|
||||||
|
|
||||||
using namespace Strawberry_TagLib::TagLib;
|
using namespace Strawberry_TagLib::TagLib;
|
||||||
|
|
||||||
class MP4::Item::ItemPrivate : public RefCounter
|
namespace {
|
||||||
{
|
struct ItemData {
|
||||||
public:
|
|
||||||
ItemPrivate() :
|
|
||||||
RefCounter(),
|
|
||||||
valid(true),
|
|
||||||
atomDataType(TypeUndefined) {}
|
|
||||||
|
|
||||||
bool valid;
|
bool valid;
|
||||||
AtomDataType atomDataType;
|
MP4::AtomDataType atomDataType;
|
||||||
|
MP4::Item::ItemType type;
|
||||||
union {
|
union {
|
||||||
bool m_bool;
|
bool m_bool;
|
||||||
int m_int;
|
int m_int;
|
||||||
IntPair m_intPair;
|
MP4::Item::IntPair m_intPair;
|
||||||
unsigned char m_byte;
|
unsigned char m_byte;
|
||||||
unsigned int m_uint;
|
unsigned int m_uint;
|
||||||
long long m_longlong;
|
long long m_longlong;
|
||||||
@@ -52,161 +50,166 @@ public:
|
|||||||
ByteVectorList m_byteVectorList;
|
ByteVectorList m_byteVectorList;
|
||||||
MP4::CoverArtList m_coverArtList;
|
MP4::CoverArtList m_coverArtList;
|
||||||
};
|
};
|
||||||
|
} // namespace
|
||||||
|
|
||||||
MP4::Item::Item() :
|
class MP4::Item::ItemPrivate {
|
||||||
d(new ItemPrivate())
|
public:
|
||||||
{
|
explicit ItemPrivate() : data(new ItemData()) {
|
||||||
d->valid = false;
|
data->valid = true;
|
||||||
|
data->atomDataType = MP4::TypeUndefined;
|
||||||
|
data->type = MP4::Item::TypeUndefined_;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::shared_ptr<ItemData> data;
|
||||||
|
};
|
||||||
|
|
||||||
|
MP4::Item::Item() : d(new ItemPrivate()) {
|
||||||
|
d->data->valid = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
MP4::Item::Item(const Item &item) :
|
MP4::Item::Item(const Item &item) : d(new ItemPrivate(*item.d)) {}
|
||||||
d(item.d)
|
|
||||||
{
|
|
||||||
d->ref();
|
|
||||||
}
|
|
||||||
|
|
||||||
MP4::Item &
|
MP4::Item &MP4::Item::operator=(const Item &item) {
|
||||||
MP4::Item::operator=(const Item &item)
|
|
||||||
{
|
|
||||||
Item(item).swap(*this);
|
Item(item).swap(*this);
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void MP4::Item::swap(Item &item) {
|
||||||
MP4::Item::swap(Item &item)
|
|
||||||
{
|
|
||||||
using std::swap;
|
using std::swap;
|
||||||
|
|
||||||
swap(d, item.d);
|
swap(d, item.d);
|
||||||
}
|
}
|
||||||
|
|
||||||
MP4::Item::~Item()
|
MP4::Item::~Item() {
|
||||||
{
|
delete d;
|
||||||
if(d->deref())
|
|
||||||
delete d;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
MP4::Item::Item(bool value) :
|
MP4::Item::Item(bool value) : d(new ItemPrivate()) {
|
||||||
d(new ItemPrivate())
|
d->data->m_bool = value;
|
||||||
{
|
d->data->type = TypeBool;
|
||||||
d->m_bool = value;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
MP4::Item::Item(int value) :
|
MP4::Item::Item(int value) : d(new ItemPrivate()) {
|
||||||
d(new ItemPrivate())
|
d->data->m_int = value;
|
||||||
{
|
d->data->type = TypeInt;
|
||||||
d->m_int = value;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
MP4::Item::Item(unsigned char value) :
|
MP4::Item::Item(unsigned char value) : d(new ItemPrivate()) {
|
||||||
d(new ItemPrivate())
|
d->data->m_byte = value;
|
||||||
{
|
d->data->type = TypeByte;
|
||||||
d->m_byte = value;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
MP4::Item::Item(unsigned int value) :
|
MP4::Item::Item(unsigned int value) : d(new ItemPrivate()) {
|
||||||
d(new ItemPrivate())
|
d->data->m_uint = value;
|
||||||
{
|
d->data->type = TypeUInt;
|
||||||
d->m_uint = value;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
MP4::Item::Item(long long value) :
|
MP4::Item::Item(long long value) : d(new ItemPrivate()) {
|
||||||
d(new ItemPrivate())
|
d->data->m_longlong = value;
|
||||||
{
|
d->data->type = TypeLongLong;
|
||||||
d->m_longlong = value;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
MP4::Item::Item(int value1, int value2) :
|
MP4::Item::Item(int value1, int value2) : d(new ItemPrivate()) {
|
||||||
d(new ItemPrivate())
|
d->data->m_intPair.first = value1;
|
||||||
{
|
d->data->m_intPair.second = value2;
|
||||||
d->m_intPair.first = value1;
|
d->data->type = TypeIntPair;
|
||||||
d->m_intPair.second = value2;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
MP4::Item::Item(const ByteVectorList &value) :
|
MP4::Item::Item(const ByteVectorList &value) : d(new ItemPrivate()) {
|
||||||
d(new ItemPrivate())
|
d->data->m_byteVectorList = value;
|
||||||
{
|
d->data->type = TypeByteVectorList;
|
||||||
d->m_byteVectorList = value;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
MP4::Item::Item(const StringList &value) :
|
MP4::Item::Item(const StringList &value) : d(new ItemPrivate()) {
|
||||||
d(new ItemPrivate())
|
d->data->m_stringList = value;
|
||||||
{
|
d->data->type = TypeStringList;
|
||||||
d->m_stringList = value;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
MP4::Item::Item(const MP4::CoverArtList &value) :
|
MP4::Item::Item(const MP4::CoverArtList &value) : d(new ItemPrivate()) {
|
||||||
d(new ItemPrivate())
|
d->data->m_coverArtList = value;
|
||||||
{
|
d->data->type = TypeCoverArtList;
|
||||||
d->m_coverArtList = value;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void MP4::Item::setAtomDataType(MP4::AtomDataType type)
|
void MP4::Item::setAtomDataType(MP4::AtomDataType type) {
|
||||||
{
|
d->data->atomDataType = type;
|
||||||
d->atomDataType = type;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
MP4::AtomDataType MP4::Item::atomDataType() const
|
MP4::AtomDataType MP4::Item::atomDataType() const {
|
||||||
{
|
return d->data->atomDataType;
|
||||||
return d->atomDataType;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool
|
bool MP4::Item::toBool() const {
|
||||||
MP4::Item::toBool() const
|
return d->data->m_bool;
|
||||||
{
|
|
||||||
return d->m_bool;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int
|
int MP4::Item::toInt() const {
|
||||||
MP4::Item::toInt() const
|
return d->data->m_int;
|
||||||
{
|
|
||||||
return d->m_int;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned char
|
unsigned char MP4::Item::toByte() const {
|
||||||
MP4::Item::toByte() const
|
return d->data->m_byte;
|
||||||
{
|
|
||||||
return d->m_byte;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned int
|
unsigned int MP4::Item::toUInt() const {
|
||||||
MP4::Item::toUInt() const
|
return d->data->m_uint;
|
||||||
{
|
|
||||||
return d->m_uint;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
long long
|
long long MP4::Item::toLongLong() const {
|
||||||
MP4::Item::toLongLong() const
|
return d->data->m_longlong;
|
||||||
{
|
|
||||||
return d->m_longlong;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
MP4::Item::IntPair
|
MP4::Item::IntPair MP4::Item::toIntPair() const {
|
||||||
MP4::Item::toIntPair() const
|
return d->data->m_intPair;
|
||||||
{
|
|
||||||
return d->m_intPair;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
StringList
|
StringList MP4::Item::toStringList() const {
|
||||||
MP4::Item::toStringList() const
|
return d->data->m_stringList;
|
||||||
{
|
|
||||||
return d->m_stringList;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ByteVectorList
|
ByteVectorList MP4::Item::toByteVectorList() const {
|
||||||
MP4::Item::toByteVectorList() const
|
return d->data->m_byteVectorList;
|
||||||
{
|
|
||||||
return d->m_byteVectorList;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
MP4::CoverArtList
|
MP4::CoverArtList MP4::Item::toCoverArtList() const {
|
||||||
MP4::Item::toCoverArtList() const
|
return d->data->m_coverArtList;
|
||||||
{
|
|
||||||
return d->m_coverArtList;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool
|
bool MP4::Item::isValid() const {
|
||||||
MP4::Item::isValid() const
|
return d->data->valid;
|
||||||
{
|
}
|
||||||
return d->valid;
|
|
||||||
|
String MP4::Item::toString() const {
|
||||||
|
|
||||||
|
StringList desc;
|
||||||
|
switch (d->data->type) {
|
||||||
|
case TypeBool:
|
||||||
|
return d->data->m_bool ? "true" : "false";
|
||||||
|
case TypeInt:
|
||||||
|
return Utils::formatString("%d", d->data->m_int);
|
||||||
|
case TypeIntPair:
|
||||||
|
return Utils::formatString("%d/%d", d->data->m_intPair.first, d->data->m_intPair.second);
|
||||||
|
case TypeByte:
|
||||||
|
return Utils::formatString("%d", d->data->m_byte);
|
||||||
|
case TypeUInt:
|
||||||
|
return Utils::formatString("%u", d->data->m_uint);
|
||||||
|
case TypeLongLong:
|
||||||
|
return Utils::formatString("%lld", d->data->m_longlong);
|
||||||
|
case TypeStringList:
|
||||||
|
return d->data->m_stringList.toString(" / ");
|
||||||
|
case TypeByteVectorList:
|
||||||
|
for (size_t i = 0; i < d->data->m_byteVectorList.size(); i++) {
|
||||||
|
desc.append(Utils::formatString(
|
||||||
|
"[%d bytes of data]", static_cast<int>(d->data->m_byteVectorList[i].size())));
|
||||||
|
}
|
||||||
|
return desc.toString(", ");
|
||||||
|
case TypeCoverArtList:
|
||||||
|
for (size_t i = 0; i < d->data->m_coverArtList.size(); i++) {
|
||||||
|
desc.append(Utils::formatString("[%d bytes of data]", static_cast<int>(d->data->m_coverArtList[i].data().size())));
|
||||||
|
}
|
||||||
|
return desc.toString(", ");
|
||||||
|
case TypeUndefined_:
|
||||||
|
return "[unknown]";
|
||||||
|
}
|
||||||
|
return String();
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
106
3rdparty/taglib/mp4/mp4item.h
vendored
106
3rdparty/taglib/mp4/mp4item.h
vendored
@@ -32,64 +32,78 @@
|
|||||||
|
|
||||||
namespace Strawberry_TagLib {
|
namespace Strawberry_TagLib {
|
||||||
namespace TagLib {
|
namespace TagLib {
|
||||||
|
namespace MP4 {
|
||||||
|
|
||||||
namespace MP4 {
|
class TAGLIB_EXPORT Item {
|
||||||
|
public:
|
||||||
|
struct IntPair {
|
||||||
|
int first, second;
|
||||||
|
};
|
||||||
|
|
||||||
class TAGLIB_EXPORT Item
|
enum ItemType {
|
||||||
{
|
TypeUndefined_ = 0,
|
||||||
public:
|
TypeBool,
|
||||||
struct IntPair {
|
TypeInt,
|
||||||
int first, second;
|
TypeIntPair,
|
||||||
};
|
TypeByte,
|
||||||
|
TypeUInt,
|
||||||
|
TypeLongLong,
|
||||||
|
TypeStringList,
|
||||||
|
TypeByteVectorList,
|
||||||
|
TypeCoverArtList,
|
||||||
|
};
|
||||||
|
|
||||||
Item();
|
explicit Item();
|
||||||
Item(const Item &item);
|
Item(const Item &item);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Copies the contents of \a item into this Item.
|
* Copies the contents of \a item into this Item.
|
||||||
*/
|
*/
|
||||||
Item &operator=(const Item &item);
|
Item &operator=(const Item &item);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Exchanges the content of the Item by the content of \a item.
|
* Exchanges the content of the Item by the content of \a item.
|
||||||
*/
|
*/
|
||||||
void swap(Item &item);
|
void swap(Item &item);
|
||||||
|
|
||||||
~Item();
|
~Item();
|
||||||
|
|
||||||
Item(int value);
|
Item(int value);
|
||||||
Item(unsigned char value);
|
Item(unsigned char value);
|
||||||
Item(unsigned int value);
|
Item(unsigned int value);
|
||||||
Item(long long value);
|
Item(long long value);
|
||||||
Item(bool value);
|
Item(bool value);
|
||||||
Item(int first, int second);
|
Item(int first, int second);
|
||||||
Item(const StringList &value);
|
Item(const StringList &value);
|
||||||
Item(const ByteVectorList &value);
|
Item(const ByteVectorList &value);
|
||||||
Item(const CoverArtList &value);
|
Item(const CoverArtList &value);
|
||||||
|
|
||||||
void setAtomDataType(AtomDataType type);
|
void setAtomDataType(AtomDataType type);
|
||||||
AtomDataType atomDataType() const;
|
AtomDataType atomDataType() const;
|
||||||
|
|
||||||
int toInt() const;
|
int toInt() const;
|
||||||
unsigned char toByte() const;
|
unsigned char toByte() const;
|
||||||
unsigned int toUInt() const;
|
unsigned int toUInt() const;
|
||||||
long long toLongLong() const;
|
long long toLongLong() const;
|
||||||
bool toBool() const;
|
bool toBool() const;
|
||||||
IntPair toIntPair() const;
|
IntPair toIntPair() const;
|
||||||
StringList toStringList() const;
|
StringList toStringList() const;
|
||||||
ByteVectorList toByteVectorList() const;
|
ByteVectorList toByteVectorList() const;
|
||||||
CoverArtList toCoverArtList() const;
|
CoverArtList toCoverArtList() const;
|
||||||
|
|
||||||
bool isValid() const;
|
ItemType type() const;
|
||||||
|
|
||||||
private:
|
bool isValid() const;
|
||||||
class ItemPrivate;
|
|
||||||
ItemPrivate *d;
|
|
||||||
};
|
|
||||||
|
|
||||||
}
|
String toString() const;
|
||||||
|
|
||||||
}
|
private:
|
||||||
}
|
class ItemPrivate;
|
||||||
|
ItemPrivate *d;
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace MP4
|
||||||
|
} // namespace TagLib
|
||||||
|
} // namespace Strawberry_TagLib
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
165
3rdparty/taglib/mp4/mp4properties.cpp
vendored
165
3rdparty/taglib/mp4/mp4properties.cpp
vendored
@@ -23,25 +23,23 @@
|
|||||||
* http://www.mozilla.org/MPL/ *
|
* http://www.mozilla.org/MPL/ *
|
||||||
***************************************************************************/
|
***************************************************************************/
|
||||||
|
|
||||||
#include <tdebug.h>
|
#include "tdebug.h"
|
||||||
#include <tstring.h>
|
#include "tstring.h"
|
||||||
#include "mp4file.h"
|
#include "mp4file.h"
|
||||||
#include "mp4atom.h"
|
#include "mp4atom.h"
|
||||||
#include "mp4properties.h"
|
#include "mp4properties.h"
|
||||||
|
|
||||||
using namespace Strawberry_TagLib::TagLib;
|
using namespace Strawberry_TagLib::TagLib;
|
||||||
|
|
||||||
class MP4::Properties::PropertiesPrivate
|
class MP4::AudioProperties::AudioPropertiesPrivate {
|
||||||
{
|
public:
|
||||||
public:
|
explicit AudioPropertiesPrivate() : length(0),
|
||||||
PropertiesPrivate() :
|
bitrate(0),
|
||||||
length(0),
|
sampleRate(0),
|
||||||
bitrate(0),
|
channels(0),
|
||||||
sampleRate(0),
|
bitsPerSample(0),
|
||||||
channels(0),
|
encrypted(false),
|
||||||
bitsPerSample(0),
|
codec(MP4::AudioProperties::Unknown) {}
|
||||||
encrypted(false),
|
|
||||||
codec(MP4::Properties::Unknown) {}
|
|
||||||
|
|
||||||
int length;
|
int length;
|
||||||
int bitrate;
|
int bitrate;
|
||||||
@@ -56,110 +54,104 @@ public:
|
|||||||
// public members
|
// public members
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
MP4::Properties::Properties(File *file, MP4::Atoms *atoms, ReadStyle style) :
|
MP4::AudioProperties::AudioProperties(File *file, MP4::Atoms *atoms, ReadStyle) : Strawberry_TagLib::TagLib::AudioProperties(), d(new AudioPropertiesPrivate()) {
|
||||||
AudioProperties(style),
|
|
||||||
d(new PropertiesPrivate())
|
|
||||||
{
|
|
||||||
read(file, atoms);
|
read(file, atoms);
|
||||||
}
|
}
|
||||||
|
|
||||||
MP4::Properties::~Properties()
|
MP4::AudioProperties::~AudioProperties() {
|
||||||
{
|
|
||||||
delete d;
|
delete d;
|
||||||
}
|
}
|
||||||
|
|
||||||
int
|
int MP4::AudioProperties::channels() const {
|
||||||
MP4::Properties::channels() const
|
|
||||||
{
|
|
||||||
return d->channels;
|
return d->channels;
|
||||||
}
|
}
|
||||||
|
|
||||||
int
|
int MP4::AudioProperties::sampleRate() const {
|
||||||
MP4::Properties::sampleRate() const
|
|
||||||
{
|
|
||||||
return d->sampleRate;
|
return d->sampleRate;
|
||||||
}
|
}
|
||||||
|
|
||||||
int
|
int MP4::AudioProperties::lengthInSeconds() const {
|
||||||
MP4::Properties::length() const
|
|
||||||
{
|
|
||||||
return lengthInSeconds();
|
|
||||||
}
|
|
||||||
|
|
||||||
int
|
|
||||||
MP4::Properties::lengthInSeconds() const
|
|
||||||
{
|
|
||||||
return d->length / 1000;
|
return d->length / 1000;
|
||||||
}
|
}
|
||||||
|
|
||||||
int
|
int MP4::AudioProperties::lengthInMilliseconds() const {
|
||||||
MP4::Properties::lengthInMilliseconds() const
|
|
||||||
{
|
|
||||||
return d->length;
|
return d->length;
|
||||||
}
|
}
|
||||||
|
|
||||||
int
|
int MP4::AudioProperties::bitrate() const {
|
||||||
MP4::Properties::bitrate() const
|
|
||||||
{
|
|
||||||
return d->bitrate;
|
return d->bitrate;
|
||||||
}
|
}
|
||||||
|
|
||||||
int
|
int MP4::AudioProperties::bitsPerSample() const {
|
||||||
MP4::Properties::bitsPerSample() const
|
|
||||||
{
|
|
||||||
return d->bitsPerSample;
|
return d->bitsPerSample;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool
|
bool MP4::AudioProperties::isEncrypted() const {
|
||||||
MP4::Properties::isEncrypted() const
|
|
||||||
{
|
|
||||||
return d->encrypted;
|
return d->encrypted;
|
||||||
}
|
}
|
||||||
|
|
||||||
MP4::Properties::Codec
|
MP4::AudioProperties::Codec
|
||||||
MP4::Properties::codec() const
|
MP4::AudioProperties::codec() const {
|
||||||
{
|
|
||||||
return d->codec;
|
return d->codec;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
String MP4::AudioProperties::toString() const {
|
||||||
|
|
||||||
|
String format;
|
||||||
|
if (d->codec == AAC) {
|
||||||
|
format = "AAC";
|
||||||
|
}
|
||||||
|
else if (d->codec == ALAC) {
|
||||||
|
format = "ALAC";
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
format = "Unknown";
|
||||||
|
}
|
||||||
|
StringList desc;
|
||||||
|
desc.append("MPEG-4 audio (" + format + ")");
|
||||||
|
desc.append(String::number(lengthInSeconds()) + " seconds");
|
||||||
|
desc.append(String::number(bitrate()) + " kbps");
|
||||||
|
return desc.toString(", ");
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
// private members
|
// private members
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
void
|
void MP4::AudioProperties::read(File *file, Atoms *atoms) {
|
||||||
MP4::Properties::read(File *file, Atoms *atoms)
|
|
||||||
{
|
|
||||||
MP4::Atom *moov = atoms->find("moov");
|
MP4::Atom *moov = atoms->find("moov");
|
||||||
if(!moov) {
|
if (!moov) {
|
||||||
debug("MP4: Atom 'moov' not found");
|
debug("MP4: Atom 'moov' not found");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
MP4::Atom *trak = 0;
|
MP4::Atom *trak = nullptr;
|
||||||
ByteVector data;
|
ByteVector data;
|
||||||
|
|
||||||
const MP4::AtomList trakList = moov->findall("trak");
|
const MP4::AtomList trakList = moov->findall("trak");
|
||||||
for(MP4::AtomList::ConstIterator it = trakList.begin(); it != trakList.end(); ++it) {
|
for (MP4::AtomList::ConstIterator it = trakList.begin(); it != trakList.end(); ++it) {
|
||||||
trak = *it;
|
trak = *it;
|
||||||
MP4::Atom *hdlr = trak->find("mdia", "hdlr");
|
MP4::Atom *hdlr = trak->find("mdia", "hdlr");
|
||||||
if(!hdlr) {
|
if (!hdlr) {
|
||||||
debug("MP4: Atom 'trak.mdia.hdlr' not found");
|
debug("MP4: Atom 'trak.mdia.hdlr' not found");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
file->seek(hdlr->offset);
|
file->seek(hdlr->offset);
|
||||||
data = file->readBlock(hdlr->length);
|
data = file->readBlock(hdlr->length);
|
||||||
if(data.containsAt("soun", 16)) {
|
if (data.containsAt("soun", 16)) {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
trak = 0;
|
trak = nullptr;
|
||||||
}
|
}
|
||||||
if(!trak) {
|
if (!trak) {
|
||||||
debug("MP4: No audio tracks");
|
debug("MP4: No audio tracks");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
MP4::Atom *mdhd = trak->find("mdia", "mdhd");
|
MP4::Atom *mdhd = trak->find("mdia", "mdhd");
|
||||||
if(!mdhd) {
|
if (!mdhd) {
|
||||||
debug("MP4: Atom 'trak.mdia.mdhd' not found");
|
debug("MP4: Atom 'trak.mdia.mdhd' not found");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@@ -170,65 +162,66 @@ MP4::Properties::read(File *file, Atoms *atoms)
|
|||||||
const unsigned int version = data[8];
|
const unsigned int version = data[8];
|
||||||
long long unit;
|
long long unit;
|
||||||
long long length;
|
long long length;
|
||||||
if(version == 1) {
|
if (version == 1) {
|
||||||
if(data.size() < 36 + 8) {
|
if (data.size() < 36 + 8) {
|
||||||
debug("MP4: Atom 'trak.mdia.mdhd' is smaller than expected");
|
debug("MP4: Atom 'trak.mdia.mdhd' is smaller than expected");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
unit = data.toUInt(28U);
|
unit = data.toUInt32BE(28);
|
||||||
length = data.toLongLong(32U);
|
length = data.toInt64BE(32);
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
if(data.size() < 24 + 8) {
|
if (data.size() < 24 + 8) {
|
||||||
debug("MP4: Atom 'trak.mdia.mdhd' is smaller than expected");
|
debug("MP4: Atom 'trak.mdia.mdhd' is smaller than expected");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
unit = data.toUInt(20U);
|
unit = data.toUInt32BE(20);
|
||||||
length = data.toUInt(24U);
|
length = data.toUInt32BE(24);
|
||||||
}
|
}
|
||||||
if(unit > 0 && length > 0)
|
if (unit > 0 && length > 0)
|
||||||
d->length = static_cast<int>(length * 1000.0 / unit + 0.5);
|
d->length = static_cast<int>(length * 1000.0 / unit + 0.5);
|
||||||
|
|
||||||
MP4::Atom *atom = trak->find("mdia", "minf", "stbl", "stsd");
|
MP4::Atom *atom = trak->find("mdia", "minf", "stbl", "stsd");
|
||||||
if(!atom) {
|
if (!atom) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
file->seek(atom->offset);
|
file->seek(atom->offset);
|
||||||
data = file->readBlock(atom->length);
|
data = file->readBlock(atom->length);
|
||||||
if(data.containsAt("mp4a", 20)) {
|
if (data.containsAt("mp4a", 20)) {
|
||||||
d->codec = AAC;
|
d->codec = AAC;
|
||||||
d->channels = data.toShort(40U);
|
d->channels = data.toUInt16BE(40);
|
||||||
d->bitsPerSample = data.toShort(42U);
|
d->bitsPerSample = data.toUInt16BE(42);
|
||||||
d->sampleRate = data.toUInt(46U);
|
d->sampleRate = data.toUInt32BE(46);
|
||||||
if(data.containsAt("esds", 56) && data[64] == 0x03) {
|
if (data.containsAt("esds", 56) && data[64] == 0x03) {
|
||||||
unsigned int pos = 65;
|
unsigned int pos = 65;
|
||||||
if(data.containsAt("\x80\x80\x80", pos)) {
|
if (data.containsAt("\x80\x80\x80", pos)) {
|
||||||
pos += 3;
|
pos += 3;
|
||||||
}
|
}
|
||||||
pos += 4;
|
pos += 4;
|
||||||
if(data[pos] == 0x04) {
|
if (data[pos] == 0x04) {
|
||||||
pos += 1;
|
pos += 1;
|
||||||
if(data.containsAt("\x80\x80\x80", pos)) {
|
if (data.containsAt("\x80\x80\x80", pos)) {
|
||||||
pos += 3;
|
pos += 3;
|
||||||
}
|
}
|
||||||
pos += 10;
|
pos += 10;
|
||||||
d->bitrate = static_cast<int>((data.toUInt(pos) + 500) / 1000.0 + 0.5);
|
d->bitrate = static_cast<int>((data.toUInt32BE(pos) + 500) / 1000.0 + 0.5);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else if(data.containsAt("alac", 20)) {
|
else if (data.containsAt("alac", 20)) {
|
||||||
if(atom->length == 88 && data.containsAt("alac", 56)) {
|
if (atom->length == 88 && data.containsAt("alac", 56)) {
|
||||||
d->codec = ALAC;
|
d->codec = ALAC;
|
||||||
d->bitsPerSample = data.at(69);
|
d->bitsPerSample = data.at(69);
|
||||||
d->channels = data.at(73);
|
d->channels = data.at(73);
|
||||||
d->bitrate = static_cast<int>(data.toUInt(80U) / 1000.0 + 0.5);
|
d->bitrate = static_cast<int>(data.toUInt32BE(80) / 1000.0 + 0.5);
|
||||||
d->sampleRate = data.toUInt(84U);
|
d->sampleRate = data.toUInt32BE(84);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
MP4::Atom *drms = atom->find("drms");
|
MP4::Atom *drms = atom->find("drms");
|
||||||
if(drms) {
|
if (drms) {
|
||||||
d->encrypted = true;
|
d->encrypted = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
130
3rdparty/taglib/mp4/mp4properties.h
vendored
130
3rdparty/taglib/mp4/mp4properties.h
vendored
@@ -31,92 +31,78 @@
|
|||||||
|
|
||||||
namespace Strawberry_TagLib {
|
namespace Strawberry_TagLib {
|
||||||
namespace TagLib {
|
namespace TagLib {
|
||||||
|
namespace MP4 {
|
||||||
|
|
||||||
namespace MP4 {
|
class Atoms;
|
||||||
|
class File;
|
||||||
|
|
||||||
class Atoms;
|
//! An implementation of MP4 audio properties
|
||||||
class File;
|
class TAGLIB_EXPORT AudioProperties : public Strawberry_TagLib::TagLib::AudioProperties {
|
||||||
|
public:
|
||||||
|
enum Codec {
|
||||||
|
Unknown = 0,
|
||||||
|
AAC,
|
||||||
|
ALAC
|
||||||
|
};
|
||||||
|
|
||||||
//! An implementation of MP4 audio properties
|
explicit AudioProperties(File *file, Atoms *atoms, ReadStyle style = Average);
|
||||||
class TAGLIB_EXPORT Properties : public AudioProperties
|
~AudioProperties() override;
|
||||||
{
|
|
||||||
public:
|
|
||||||
enum Codec {
|
|
||||||
Unknown = 0,
|
|
||||||
AAC,
|
|
||||||
ALAC
|
|
||||||
};
|
|
||||||
|
|
||||||
Properties(File *file, Atoms *atoms, ReadStyle style = Average);
|
/*!
|
||||||
virtual ~Properties();
|
* Returns the length of the file in seconds. The length is rounded down to the nearest whole second.
|
||||||
|
*
|
||||||
|
* \see lengthInMilliseconds()
|
||||||
|
*/
|
||||||
|
int lengthInSeconds() const override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the length of the file in seconds. The length is rounded down to
|
* Returns the length of the file in milliseconds.
|
||||||
* the nearest whole second.
|
*
|
||||||
*
|
* \see lengthInSeconds()
|
||||||
* \note This method is just an alias of lengthInSeconds().
|
*/
|
||||||
*
|
int lengthInMilliseconds() const override;
|
||||||
* \deprecated
|
|
||||||
*/
|
|
||||||
TAGLIB_DEPRECATED virtual int length() const;
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the length of the file in seconds. The length is rounded down to
|
* Returns the average bit rate of the file in kb/s.
|
||||||
* the nearest whole second.
|
*/
|
||||||
*
|
int bitrate() const override;
|
||||||
* \see lengthInMilliseconds()
|
|
||||||
*/
|
|
||||||
// BIC: make virtual
|
|
||||||
int lengthInSeconds() const;
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the length of the file in milliseconds.
|
* Returns the sample rate in Hz.
|
||||||
*
|
*/
|
||||||
* \see lengthInSeconds()
|
int sampleRate() const override;
|
||||||
*/
|
|
||||||
// BIC: make virtual
|
|
||||||
int lengthInMilliseconds() const;
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the average bit rate of the file in kb/s.
|
* Returns the number of audio channels.
|
||||||
*/
|
*/
|
||||||
virtual int bitrate() const;
|
int channels() const override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the sample rate in Hz.
|
* Returns the number of bits per audio sample.
|
||||||
*/
|
*/
|
||||||
virtual int sampleRate() const;
|
virtual int bitsPerSample() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the number of audio channels.
|
* Returns whether or not the file is encrypted.
|
||||||
*/
|
*/
|
||||||
virtual int channels() const;
|
bool isEncrypted() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the number of bits per audio sample.
|
* Returns the codec used in the file.
|
||||||
*/
|
*/
|
||||||
virtual int bitsPerSample() const;
|
Codec codec() const;
|
||||||
|
|
||||||
/*!
|
String toString() const override;
|
||||||
* Returns whether or not the file is encrypted.
|
|
||||||
*/
|
|
||||||
bool isEncrypted() const;
|
|
||||||
|
|
||||||
/*!
|
private:
|
||||||
* Returns the codec used in the file.
|
void read(File *file, Atoms *atoms);
|
||||||
*/
|
|
||||||
Codec codec() const;
|
|
||||||
|
|
||||||
private:
|
class AudioPropertiesPrivate;
|
||||||
void read(File *file, Atoms *atoms);
|
AudioPropertiesPrivate *d;
|
||||||
|
};
|
||||||
|
|
||||||
class PropertiesPrivate;
|
} // namespace MP4
|
||||||
PropertiesPrivate *d;
|
} // namespace TagLib
|
||||||
};
|
} // namespace Strawberry_TagLib
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
872
3rdparty/taglib/mp4/mp4tag.cpp
vendored
872
3rdparty/taglib/mp4/mp4tag.cpp
vendored
File diff suppressed because it is too large
Load Diff
187
3rdparty/taglib/mp4/mp4tag.h
vendored
187
3rdparty/taglib/mp4/mp4tag.h
vendored
@@ -37,123 +37,110 @@
|
|||||||
|
|
||||||
namespace Strawberry_TagLib {
|
namespace Strawberry_TagLib {
|
||||||
namespace TagLib {
|
namespace TagLib {
|
||||||
|
namespace MP4 {
|
||||||
|
|
||||||
namespace MP4 {
|
typedef Strawberry_TagLib::TagLib::Map<String, Item> ItemMap;
|
||||||
|
typedef Strawberry_TagLib::TagLib::Map<String, Item> ItemListMap;
|
||||||
|
|
||||||
/*!
|
class TAGLIB_EXPORT Tag : public Strawberry_TagLib::TagLib::Tag {
|
||||||
* \deprecated
|
public:
|
||||||
*/
|
explicit Tag();
|
||||||
TAGLIB_DEPRECATED typedef Strawberry_TagLib::TagLib::Map<String, Item> ItemListMap;
|
explicit Tag(Strawberry_TagLib::TagLib::File *file, Atoms *atoms);
|
||||||
typedef Strawberry_TagLib::TagLib::Map<String, Item> ItemMap;
|
~Tag() override;
|
||||||
|
bool save();
|
||||||
|
|
||||||
class TAGLIB_EXPORT Tag: public Strawberry_TagLib::TagLib::Tag
|
String title() const override;
|
||||||
{
|
String artist() const override;
|
||||||
public:
|
String album() const override;
|
||||||
Tag();
|
String comment() const override;
|
||||||
Tag(Strawberry_TagLib::TagLib::File *file, Atoms *atoms);
|
String genre() const override;
|
||||||
virtual ~Tag();
|
unsigned int year() const override;
|
||||||
bool save();
|
unsigned int track() const override;
|
||||||
|
PictureMap pictures() const override;
|
||||||
|
|
||||||
virtual String title() const;
|
void setTitle(const String &value) override;
|
||||||
virtual String artist() const;
|
void setArtist(const String &value) override;
|
||||||
virtual String album() const;
|
void setAlbum(const String &value) override;
|
||||||
virtual String comment() const;
|
void setComment(const String &value) override;
|
||||||
virtual String genre() const;
|
void setGenre(const String &value) override;
|
||||||
virtual unsigned int year() const;
|
void setYear(unsigned int value) override;
|
||||||
virtual unsigned int track() const;
|
void setTrack(unsigned int value) override;
|
||||||
|
void setPictures(const PictureMap &l) override;
|
||||||
|
|
||||||
virtual void setTitle(const String &value);
|
bool isEmpty() const override;
|
||||||
virtual void setArtist(const String &value);
|
/*!
|
||||||
virtual void setAlbum(const String &value);
|
* Returns a string-keyed map of the MP4::Items for this tag.
|
||||||
virtual void setComment(const String &value);
|
*/
|
||||||
virtual void setGenre(const String &value);
|
const ItemMap &itemMap() const;
|
||||||
virtual void setYear(unsigned int value);
|
|
||||||
virtual void setTrack(unsigned int value);
|
|
||||||
|
|
||||||
virtual bool isEmpty() const;
|
/*!
|
||||||
|
* \return The item, if any, corresponding to \a key.
|
||||||
|
*/
|
||||||
|
Item item(const String &key) const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* \deprecated Use the item() and setItem() API instead
|
* Sets the value of \a key to \a value, overwriting any previous value.
|
||||||
*/
|
*/
|
||||||
TAGLIB_DEPRECATED ItemMap &itemListMap();
|
void setItem(const String &key, const Item &value);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns a string-keyed map of the MP4::Items for this tag.
|
* Removes the entry with \a key from the tag, or does nothing if it does not exist.
|
||||||
*/
|
*/
|
||||||
const ItemMap &itemMap() const;
|
void removeItem(const String &key);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* \return The item, if any, corresponding to \a key.
|
* \return True if the tag contains an entry for \a key.
|
||||||
*/
|
*/
|
||||||
Item item(const String &key) const;
|
bool contains(const String &key) const;
|
||||||
|
|
||||||
/*!
|
String toString() const override;
|
||||||
* Sets the value of \a key to \a value, overwriting any previous value.
|
|
||||||
*/
|
|
||||||
void setItem(const String &key, const Item &value);
|
|
||||||
|
|
||||||
/*!
|
PropertyMap properties() const override;
|
||||||
* Removes the entry with \a key from the tag, or does nothing if it does
|
void removeUnsupportedProperties(const StringList &props) override;
|
||||||
* not exist.
|
PropertyMap setProperties(const PropertyMap &props) override;
|
||||||
*/
|
|
||||||
void removeItem(const String &key);
|
|
||||||
|
|
||||||
/*!
|
private:
|
||||||
* \return True if the tag contains an entry for \a key.
|
AtomDataList parseData2(const Atom *atom, int expectedFlags = -1, bool freeForm = false);
|
||||||
*/
|
ByteVectorList parseData(const Atom *atom, int expectedFlags = -1, bool freeForm = false);
|
||||||
bool contains(const String &key) const;
|
void parseText(const Atom *atom, int expectedFlags = 1);
|
||||||
|
void parseFreeForm(const Atom *atom);
|
||||||
|
void parseInt(const Atom *atom);
|
||||||
|
void parseByte(const Atom *atom);
|
||||||
|
void parseUInt(const Atom *atom);
|
||||||
|
void parseLongLong(const Atom *atom);
|
||||||
|
void parseGnre(const Atom *atom);
|
||||||
|
void parseIntPair(const Atom *atom);
|
||||||
|
void parseBool(const Atom *atom);
|
||||||
|
void parseCovr(const Atom *atom);
|
||||||
|
|
||||||
PropertyMap properties() const;
|
ByteVector padIlst(const ByteVector &data, int length = -1) const;
|
||||||
void removeUnsupportedProperties(const StringList& properties);
|
ByteVector renderAtom(const ByteVector &name, const ByteVector &data) const;
|
||||||
PropertyMap setProperties(const PropertyMap &properties);
|
ByteVector renderData(const ByteVector &name, int flags, const ByteVectorList &data) const;
|
||||||
|
ByteVector renderText(const ByteVector &name, const Item &item, int flags = TypeUTF8) const;
|
||||||
|
ByteVector renderFreeForm(const String &name, const Item &item) const;
|
||||||
|
ByteVector renderBool(const ByteVector &name, const Item &item) const;
|
||||||
|
ByteVector renderInt(const ByteVector &name, const Item &item) const;
|
||||||
|
ByteVector renderByte(const ByteVector &name, const Item &item) const;
|
||||||
|
ByteVector renderUInt(const ByteVector &name, const Item &item) const;
|
||||||
|
ByteVector renderLongLong(const ByteVector &name, const Item &item) const;
|
||||||
|
ByteVector renderIntPair(const ByteVector &name, const Item &item) const;
|
||||||
|
ByteVector renderIntPairNoTrailing(const ByteVector &name, const Item &item) const;
|
||||||
|
ByteVector renderCovr(const ByteVector &name, const Item &item) const;
|
||||||
|
|
||||||
private:
|
void updateParents(const AtomList &path, long delta, int ignore = 0);
|
||||||
AtomDataList parseData2(const Atom *atom, int expectedFlags = -1,
|
void updateOffsets(long delta, long long offset);
|
||||||
bool freeForm = false);
|
|
||||||
ByteVectorList parseData(const Atom *atom, int expectedFlags = -1,
|
|
||||||
bool freeForm = false);
|
|
||||||
void parseText(const Atom *atom, int expectedFlags = 1);
|
|
||||||
void parseFreeForm(const Atom *atom);
|
|
||||||
void parseInt(const Atom *atom);
|
|
||||||
void parseByte(const Atom *atom);
|
|
||||||
void parseUInt(const Atom *atom);
|
|
||||||
void parseLongLong(const Atom *atom);
|
|
||||||
void parseGnre(const Atom *atom);
|
|
||||||
void parseIntPair(const Atom *atom);
|
|
||||||
void parseBool(const Atom *atom);
|
|
||||||
void parseCovr(const Atom *atom);
|
|
||||||
|
|
||||||
ByteVector padIlst(const ByteVector &data, int length = -1) const;
|
void saveNew(ByteVector data);
|
||||||
ByteVector renderAtom(const ByteVector &name, const ByteVector &data) const;
|
void saveExisting(ByteVector data, const AtomList &path);
|
||||||
ByteVector renderData(const ByteVector &name, int flags,
|
|
||||||
const ByteVectorList &data) const;
|
|
||||||
ByteVector renderText(const ByteVector &name, const Item &item,
|
|
||||||
int flags = TypeUTF8) const;
|
|
||||||
ByteVector renderFreeForm(const String &name, const Item &item) const;
|
|
||||||
ByteVector renderBool(const ByteVector &name, const Item &item) const;
|
|
||||||
ByteVector renderInt(const ByteVector &name, const Item &item) const;
|
|
||||||
ByteVector renderByte(const ByteVector &name, const Item &item) const;
|
|
||||||
ByteVector renderUInt(const ByteVector &name, const Item &item) const;
|
|
||||||
ByteVector renderLongLong(const ByteVector &name, const Item &item) const;
|
|
||||||
ByteVector renderIntPair(const ByteVector &name, const Item &item) const;
|
|
||||||
ByteVector renderIntPairNoTrailing(const ByteVector &name, const Item &item) const;
|
|
||||||
ByteVector renderCovr(const ByteVector &name, const Item &item) const;
|
|
||||||
|
|
||||||
void updateParents(const AtomList &path, long delta, int ignore = 0);
|
void addItem(const String &name, const Item &value);
|
||||||
void updateOffsets(long delta, long offset);
|
|
||||||
|
|
||||||
void saveNew(ByteVector data);
|
class TagPrivate;
|
||||||
void saveExisting(ByteVector data, const AtomList &path);
|
TagPrivate *d;
|
||||||
|
};
|
||||||
|
|
||||||
void addItem(const String &name, const Item &value);
|
} // namespace MP4
|
||||||
|
} // namespace TagLib
|
||||||
class TagPrivate;
|
} // namespace Strawberry_TagLib
|
||||||
TagPrivate *d;
|
|
||||||
};
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
204
3rdparty/taglib/mpc/mpcfile.cpp
vendored
204
3rdparty/taglib/mpc/mpcfile.cpp
vendored
@@ -23,12 +23,14 @@
|
|||||||
* http://www.mozilla.org/MPL/ *
|
* http://www.mozilla.org/MPL/ *
|
||||||
***************************************************************************/
|
***************************************************************************/
|
||||||
|
|
||||||
#include <tbytevector.h>
|
#include <memory>
|
||||||
#include <tstring.h>
|
|
||||||
#include <tagunion.h>
|
#include "tbytevector.h"
|
||||||
#include <tdebug.h>
|
#include "tstring.h"
|
||||||
#include <tpropertymap.h>
|
#include "tagunion.h"
|
||||||
#include <tagutils.h>
|
#include "tdebug.h"
|
||||||
|
#include "tpropertymap.h"
|
||||||
|
#include "tagutils.h"
|
||||||
|
|
||||||
#include "mpcfile.h"
|
#include "mpcfile.h"
|
||||||
#include "id3v1tag.h"
|
#include "id3v1tag.h"
|
||||||
@@ -38,125 +40,100 @@
|
|||||||
|
|
||||||
using namespace Strawberry_TagLib::TagLib;
|
using namespace Strawberry_TagLib::TagLib;
|
||||||
|
|
||||||
namespace
|
namespace {
|
||||||
{
|
enum { MPCAPEIndex = 0,
|
||||||
enum { MPCAPEIndex = 0, MPCID3v1Index = 1 };
|
MPCID3v1Index = 1 };
|
||||||
}
|
}
|
||||||
|
|
||||||
class MPC::File::FilePrivate
|
class MPC::File::FilePrivate {
|
||||||
{
|
public:
|
||||||
public:
|
explicit FilePrivate() : APELocation(-1),
|
||||||
FilePrivate() :
|
APESize(0),
|
||||||
APELocation(-1),
|
ID3v1Location(-1),
|
||||||
APESize(0),
|
ID3v2Location(-1),
|
||||||
ID3v1Location(-1),
|
ID3v2Size(0) {}
|
||||||
ID3v2Header(0),
|
|
||||||
ID3v2Location(-1),
|
|
||||||
ID3v2Size(0),
|
|
||||||
properties(0) {}
|
|
||||||
|
|
||||||
~FilePrivate()
|
long long APELocation;
|
||||||
{
|
long long APESize;
|
||||||
delete ID3v2Header;
|
|
||||||
delete properties;
|
|
||||||
}
|
|
||||||
|
|
||||||
long APELocation;
|
long long ID3v1Location;
|
||||||
long APESize;
|
|
||||||
|
|
||||||
long ID3v1Location;
|
std::unique_ptr<ID3v2::Header> ID3v2Header;
|
||||||
|
long long ID3v2Location;
|
||||||
|
long long ID3v2Size;
|
||||||
|
|
||||||
ID3v2::Header *ID3v2Header;
|
DoubleTagUnion tag;
|
||||||
long ID3v2Location;
|
|
||||||
long ID3v2Size;
|
|
||||||
|
|
||||||
TagUnion tag;
|
std::unique_ptr<AudioProperties> properties;
|
||||||
|
|
||||||
Properties *properties;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
// static members
|
// static members
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
bool MPC::File::isSupported(IOStream *stream)
|
bool MPC::File::isSupported(IOStream *stream) {
|
||||||
{
|
|
||||||
// A newer MPC file has to start with "MPCK" or "MP+", but older files don't
|
// A newer MPC file has to start with "MPCK" or "MP+", but older files don't
|
||||||
// have keys to do a quick check.
|
// have keys to do a quick check.
|
||||||
|
|
||||||
const ByteVector id = Utils::readHeader(stream, 4, false);
|
const ByteVector id = Utils::readHeader(stream, 4, false);
|
||||||
return (id == "MPCK" || id.startsWith("MP+"));
|
return (id == "MPCK" || id.startsWith("MP+"));
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
// public members
|
// public members
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
MPC::File::File(FileName file, bool readProperties, Properties::ReadStyle) :
|
MPC::File::File(FileName file, bool readProperties, AudioProperties::ReadStyle) : Strawberry_TagLib::TagLib::File(file), d(new FilePrivate()) {
|
||||||
Strawberry_TagLib::TagLib::File(file),
|
|
||||||
d(new FilePrivate())
|
if (isOpen())
|
||||||
{
|
|
||||||
if(isOpen())
|
|
||||||
read(readProperties);
|
read(readProperties);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
MPC::File::File(IOStream *stream, bool readProperties, Properties::ReadStyle) :
|
MPC::File::File(IOStream *stream, bool readProperties, AudioProperties::ReadStyle) : Strawberry_TagLib::TagLib::File(stream), d(new FilePrivate()) {
|
||||||
Strawberry_TagLib::TagLib::File(stream),
|
|
||||||
d(new FilePrivate())
|
if (isOpen())
|
||||||
{
|
|
||||||
if(isOpen())
|
|
||||||
read(readProperties);
|
read(readProperties);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
MPC::File::~File()
|
MPC::File::~File() {
|
||||||
{
|
|
||||||
delete d;
|
delete d;
|
||||||
}
|
}
|
||||||
|
|
||||||
Strawberry_TagLib::TagLib::Tag *MPC::File::tag() const
|
Strawberry_TagLib::TagLib::Tag *MPC::File::tag() const {
|
||||||
{
|
|
||||||
return &d->tag;
|
return &d->tag;
|
||||||
}
|
}
|
||||||
|
|
||||||
PropertyMap MPC::File::properties() const
|
PropertyMap MPC::File::setProperties(const PropertyMap &properties) {
|
||||||
{
|
if (ID3v1Tag())
|
||||||
return d->tag.properties();
|
|
||||||
}
|
|
||||||
|
|
||||||
void MPC::File::removeUnsupportedProperties(const StringList &properties)
|
|
||||||
{
|
|
||||||
d->tag.removeUnsupportedProperties(properties);
|
|
||||||
}
|
|
||||||
|
|
||||||
PropertyMap MPC::File::setProperties(const PropertyMap &properties)
|
|
||||||
{
|
|
||||||
if(ID3v1Tag())
|
|
||||||
ID3v1Tag()->setProperties(properties);
|
ID3v1Tag()->setProperties(properties);
|
||||||
|
|
||||||
return APETag(true)->setProperties(properties);
|
return APETag(true)->setProperties(properties);
|
||||||
}
|
}
|
||||||
|
|
||||||
MPC::Properties *MPC::File::audioProperties() const
|
MPC::AudioProperties *MPC::File::audioProperties() const {
|
||||||
{
|
return d->properties.get();
|
||||||
return d->properties;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool MPC::File::save()
|
bool MPC::File::save() {
|
||||||
{
|
|
||||||
if(readOnly()) {
|
if (readOnly()) {
|
||||||
debug("MPC::File::save() -- File is read only.");
|
debug("MPC::File::save() -- File is read only.");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Possibly strip ID3v2 tag
|
// Possibly strip ID3v2 tag
|
||||||
|
|
||||||
if(!d->ID3v2Header && d->ID3v2Location >= 0) {
|
if (!d->ID3v2Header && d->ID3v2Location >= 0) {
|
||||||
removeBlock(d->ID3v2Location, d->ID3v2Size);
|
removeBlock(d->ID3v2Location, d->ID3v2Size);
|
||||||
|
|
||||||
if(d->APELocation >= 0)
|
if (d->APELocation >= 0)
|
||||||
d->APELocation -= d->ID3v2Size;
|
d->APELocation -= d->ID3v2Size;
|
||||||
|
|
||||||
if(d->ID3v1Location >= 0)
|
if (d->ID3v1Location >= 0)
|
||||||
d->ID3v1Location -= d->ID3v2Size;
|
d->ID3v1Location -= d->ID3v2Size;
|
||||||
|
|
||||||
d->ID3v2Location = -1;
|
d->ID3v2Location = -1;
|
||||||
@@ -165,11 +142,11 @@ bool MPC::File::save()
|
|||||||
|
|
||||||
// Update ID3v1 tag
|
// Update ID3v1 tag
|
||||||
|
|
||||||
if(ID3v1Tag() && !ID3v1Tag()->isEmpty()) {
|
if (ID3v1Tag() && !ID3v1Tag()->isEmpty()) {
|
||||||
|
|
||||||
// ID3v1 tag is not empty. Update the old one or create a new one.
|
// ID3v1 tag is not empty. Update the old one or create a new one.
|
||||||
|
|
||||||
if(d->ID3v1Location >= 0) {
|
if (d->ID3v1Location >= 0) {
|
||||||
seek(d->ID3v1Location);
|
seek(d->ID3v1Location);
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
@@ -183,7 +160,7 @@ bool MPC::File::save()
|
|||||||
|
|
||||||
// ID3v1 tag is empty. Remove the old one.
|
// ID3v1 tag is empty. Remove the old one.
|
||||||
|
|
||||||
if(d->ID3v1Location >= 0) {
|
if (d->ID3v1Location >= 0) {
|
||||||
truncate(d->ID3v1Location);
|
truncate(d->ID3v1Location);
|
||||||
d->ID3v1Location = -1;
|
d->ID3v1Location = -1;
|
||||||
}
|
}
|
||||||
@@ -191,12 +168,12 @@ bool MPC::File::save()
|
|||||||
|
|
||||||
// Update APE tag
|
// Update APE tag
|
||||||
|
|
||||||
if(APETag() && !APETag()->isEmpty()) {
|
if (APETag() && !APETag()->isEmpty()) {
|
||||||
|
|
||||||
// APE tag is not empty. Update the old one or create a new one.
|
// APE tag is not empty. Update the old one or create a new one.
|
||||||
|
|
||||||
if(d->APELocation < 0) {
|
if (d->APELocation < 0) {
|
||||||
if(d->ID3v1Location >= 0)
|
if (d->ID3v1Location >= 0)
|
||||||
d->APELocation = d->ID3v1Location;
|
d->APELocation = d->ID3v1Location;
|
||||||
else
|
else
|
||||||
d->APELocation = length();
|
d->APELocation = length();
|
||||||
@@ -205,7 +182,7 @@ bool MPC::File::save()
|
|||||||
const ByteVector data = APETag()->render();
|
const ByteVector data = APETag()->render();
|
||||||
insert(data, d->APELocation, d->APESize);
|
insert(data, d->APELocation, d->APESize);
|
||||||
|
|
||||||
if(d->ID3v1Location >= 0)
|
if (d->ID3v1Location >= 0)
|
||||||
d->ID3v1Location += (static_cast<long>(data.size()) - d->APESize);
|
d->ID3v1Location += (static_cast<long>(data.size()) - d->APESize);
|
||||||
|
|
||||||
d->APESize = data.size();
|
d->APESize = data.size();
|
||||||
@@ -214,10 +191,10 @@ bool MPC::File::save()
|
|||||||
|
|
||||||
// APE tag is empty. Remove the old one.
|
// APE tag is empty. Remove the old one.
|
||||||
|
|
||||||
if(d->APELocation >= 0) {
|
if (d->APELocation >= 0) {
|
||||||
removeBlock(d->APELocation, d->APESize);
|
removeBlock(d->APELocation, d->APESize);
|
||||||
|
|
||||||
if(d->ID3v1Location >= 0)
|
if (d->ID3v1Location >= 0)
|
||||||
d->ID3v1Location -= d->APESize;
|
d->ID3v1Location -= d->APESize;
|
||||||
|
|
||||||
d->APELocation = -1;
|
d->APELocation = -1;
|
||||||
@@ -226,47 +203,37 @@ bool MPC::File::save()
|
|||||||
}
|
}
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ID3v1::Tag *MPC::File::ID3v1Tag(bool create)
|
ID3v1::Tag *MPC::File::ID3v1Tag(bool create) {
|
||||||
{
|
|
||||||
return d->tag.access<ID3v1::Tag>(MPCID3v1Index, create);
|
return d->tag.access<ID3v1::Tag>(MPCID3v1Index, create);
|
||||||
}
|
}
|
||||||
|
|
||||||
APE::Tag *MPC::File::APETag(bool create)
|
APE::Tag *MPC::File::APETag(bool create) {
|
||||||
{
|
|
||||||
return d->tag.access<APE::Tag>(MPCAPEIndex, create);
|
return d->tag.access<APE::Tag>(MPCAPEIndex, create);
|
||||||
}
|
}
|
||||||
|
|
||||||
void MPC::File::strip(int tags)
|
void MPC::File::strip(int tags) {
|
||||||
{
|
|
||||||
if(tags & ID3v1)
|
|
||||||
d->tag.set(MPCID3v1Index, 0);
|
|
||||||
|
|
||||||
if(tags & APE)
|
if (tags & ID3v1)
|
||||||
d->tag.set(MPCAPEIndex, 0);
|
d->tag.set(MPCID3v1Index, nullptr);
|
||||||
|
|
||||||
if(!ID3v1Tag())
|
if (tags & APE)
|
||||||
|
d->tag.set(MPCAPEIndex, nullptr);
|
||||||
|
|
||||||
|
if (!ID3v1Tag())
|
||||||
APETag(true);
|
APETag(true);
|
||||||
|
|
||||||
if(tags & ID3v2) {
|
if (tags & ID3v2)
|
||||||
delete d->ID3v2Header;
|
d->ID3v2Header.reset();
|
||||||
d->ID3v2Header = 0;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void MPC::File::remove(int tags)
|
bool MPC::File::hasID3v1Tag() const {
|
||||||
{
|
|
||||||
strip(tags);
|
|
||||||
}
|
|
||||||
|
|
||||||
bool MPC::File::hasID3v1Tag() const
|
|
||||||
{
|
|
||||||
return (d->ID3v1Location >= 0);
|
return (d->ID3v1Location >= 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool MPC::File::hasAPETag() const
|
bool MPC::File::hasAPETag() const {
|
||||||
{
|
|
||||||
return (d->APELocation >= 0);
|
return (d->APELocation >= 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -274,15 +241,15 @@ bool MPC::File::hasAPETag() const
|
|||||||
// private members
|
// private members
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
void MPC::File::read(bool readProperties)
|
void MPC::File::read(bool readProperties) {
|
||||||
{
|
|
||||||
// Look for an ID3v2 tag
|
// Look for an ID3v2 tag
|
||||||
|
|
||||||
d->ID3v2Location = Utils::findID3v2(this);
|
d->ID3v2Location = Utils::findID3v2(this);
|
||||||
|
|
||||||
if(d->ID3v2Location >= 0) {
|
if (d->ID3v2Location >= 0) {
|
||||||
seek(d->ID3v2Location);
|
seek(d->ID3v2Location);
|
||||||
d->ID3v2Header = new ID3v2::Header(readBlock(ID3v2::Header::size()));
|
d->ID3v2Header.reset(new ID3v2::Header(readBlock(ID3v2::Header::size())));
|
||||||
d->ID3v2Size = d->ID3v2Header->completeTagSize();
|
d->ID3v2Size = d->ID3v2Header->completeTagSize();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -290,36 +257,36 @@ void MPC::File::read(bool readProperties)
|
|||||||
|
|
||||||
d->ID3v1Location = Utils::findID3v1(this);
|
d->ID3v1Location = Utils::findID3v1(this);
|
||||||
|
|
||||||
if(d->ID3v1Location >= 0)
|
if (d->ID3v1Location >= 0)
|
||||||
d->tag.set(MPCID3v1Index, new ID3v1::Tag(this, d->ID3v1Location));
|
d->tag.set(MPCID3v1Index, new ID3v1::Tag(this, d->ID3v1Location));
|
||||||
|
|
||||||
// Look for an APE tag
|
// Look for an APE tag
|
||||||
|
|
||||||
d->APELocation = Utils::findAPE(this, d->ID3v1Location);
|
d->APELocation = Utils::findAPE(this, d->ID3v1Location);
|
||||||
|
|
||||||
if(d->APELocation >= 0) {
|
if (d->APELocation >= 0) {
|
||||||
d->tag.set(MPCAPEIndex, new APE::Tag(this, d->APELocation));
|
d->tag.set(MPCAPEIndex, new APE::Tag(this, d->APELocation));
|
||||||
d->APESize = APETag()->footer()->completeTagSize();
|
d->APESize = APETag()->footer()->completeTagSize();
|
||||||
d->APELocation = d->APELocation + APE::Footer::size() - d->APESize;
|
d->APELocation = d->APELocation + APE::Footer::size() - d->APESize;
|
||||||
}
|
}
|
||||||
|
|
||||||
if(d->ID3v1Location < 0)
|
if (d->ID3v1Location < 0)
|
||||||
APETag(true);
|
APETag(true);
|
||||||
|
|
||||||
// Look for MPC metadata
|
// Look for MPC metadata
|
||||||
|
|
||||||
if(readProperties) {
|
if (readProperties) {
|
||||||
|
|
||||||
long streamLength;
|
long long streamLength;
|
||||||
|
|
||||||
if(d->APELocation >= 0)
|
if (d->APELocation >= 0)
|
||||||
streamLength = d->APELocation;
|
streamLength = d->APELocation;
|
||||||
else if(d->ID3v1Location >= 0)
|
else if (d->ID3v1Location >= 0)
|
||||||
streamLength = d->ID3v1Location;
|
streamLength = d->ID3v1Location;
|
||||||
else
|
else
|
||||||
streamLength = length();
|
streamLength = length();
|
||||||
|
|
||||||
if(d->ID3v2Location >= 0) {
|
if (d->ID3v2Location >= 0) {
|
||||||
seek(d->ID3v2Location + d->ID3v2Size);
|
seek(d->ID3v2Location + d->ID3v2Size);
|
||||||
streamLength -= (d->ID3v2Location + d->ID3v2Size);
|
streamLength -= (d->ID3v2Location + d->ID3v2Size);
|
||||||
}
|
}
|
||||||
@@ -327,6 +294,7 @@ void MPC::File::read(bool readProperties)
|
|||||||
seek(0);
|
seek(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
d->properties = new Properties(this, streamLength);
|
d->properties.reset(new AudioProperties(this, streamLength));
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
328
3rdparty/taglib/mpc/mpcfile.h
vendored
328
3rdparty/taglib/mpc/mpcfile.h
vendored
@@ -37,204 +37,176 @@
|
|||||||
namespace Strawberry_TagLib {
|
namespace Strawberry_TagLib {
|
||||||
namespace TagLib {
|
namespace TagLib {
|
||||||
|
|
||||||
class Tag;
|
class Tag;
|
||||||
|
|
||||||
namespace ID3v1 { class Tag; }
|
namespace ID3v1 {
|
||||||
namespace APE { class Tag; }
|
class Tag;
|
||||||
|
}
|
||||||
|
namespace APE {
|
||||||
|
class Tag;
|
||||||
|
}
|
||||||
|
|
||||||
//! An implementation of MPC metadata
|
//! An implementation of MPC metadata
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* This is implementation of MPC metadata.
|
||||||
|
*
|
||||||
|
* This supports ID3v1 and APE (v1 and v2) style comments as well as reading stream properties from the file.
|
||||||
|
* ID3v2 tags are invalid in MPC-files, but will be skipped and ignored.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace MPC {
|
||||||
|
|
||||||
|
//! An implementation of TagLib::File with MPC specific methods
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* This implements and provides an interface for MPC files to the TagLib::Tag and TagLib::AudioProperties interfaces by way of implementing the abstract TagLib::File API as well as providing some additional information specific to MPC files.
|
||||||
|
* The only invalid tag combination supported is an ID3v1 tag after an APE tag.
|
||||||
|
*/
|
||||||
|
|
||||||
|
class TAGLIB_EXPORT File : public Strawberry_TagLib::TagLib::File {
|
||||||
|
public:
|
||||||
|
/*!
|
||||||
|
* This set of flags is used for various operations and is suitable for being OR-ed together.
|
||||||
|
*/
|
||||||
|
enum TagTypes {
|
||||||
|
//! Empty set. Matches no tag types.
|
||||||
|
NoTags = 0x0000,
|
||||||
|
//! Matches ID3v1 tags.
|
||||||
|
ID3v1 = 0x0001,
|
||||||
|
//! Matches ID3v2 tags.
|
||||||
|
ID3v2 = 0x0002,
|
||||||
|
//! Matches APE tags.
|
||||||
|
APE = 0x0004,
|
||||||
|
//! Matches all tag types.
|
||||||
|
AllTags = 0xffff
|
||||||
|
};
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* This is implementation of MPC metadata.
|
* Constructs an MPC file from \a file.
|
||||||
|
* If \a readProperties is true the file's audio properties will also be read.
|
||||||
*
|
*
|
||||||
* This supports ID3v1 and APE (v1 and v2) style comments as well as reading stream
|
* \note In the current implementation, \a propertiesStyle is ignored.
|
||||||
* properties from the file. ID3v2 tags are invalid in MPC-files, but will be skipped
|
|
||||||
* and ignored.
|
|
||||||
*/
|
*/
|
||||||
|
explicit File(FileName file, bool readProperties = true, AudioProperties::ReadStyle propertiesStyle = AudioProperties::Average);
|
||||||
|
|
||||||
namespace MPC {
|
/*!
|
||||||
|
* Constructs an MPC file from \a stream.
|
||||||
|
* If \a readProperties is true the file's audio properties will also be read.
|
||||||
|
*
|
||||||
|
* \note TagLib will *not* take ownership of the stream, the caller is responsible for deleting it after the File object.
|
||||||
|
*
|
||||||
|
* \note In the current implementation, \a propertiesStyle is ignored.
|
||||||
|
*/
|
||||||
|
explicit File(IOStream *stream, bool readProperties = true, AudioProperties::ReadStyle propertiesStyle = AudioProperties::Average);
|
||||||
|
|
||||||
//! An implementation of TagLib::File with MPC specific methods
|
/*!
|
||||||
|
* Destroys this instance of the File.
|
||||||
|
*/
|
||||||
|
~File() override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* This implements and provides an interface for MPC files to the
|
* Returns the Tag for this file.
|
||||||
* TagLib::Tag and TagLib::AudioProperties interfaces by way of implementing
|
* This will be an APE tag, an ID3v1 tag or a combination of the two.
|
||||||
* the abstract TagLib::File API as well as providing some additional
|
*/
|
||||||
* information specific to MPC files.
|
Strawberry_TagLib::TagLib::Tag *tag() const override;
|
||||||
* The only invalid tag combination supported is an ID3v1 tag after an APE tag.
|
|
||||||
*/
|
|
||||||
|
|
||||||
class TAGLIB_EXPORT File : public Strawberry_TagLib::TagLib::File
|
/*!
|
||||||
{
|
* Implements the unified property interface -- import function.
|
||||||
public:
|
* Affects only the APEv2 tag which will be created if necessary.
|
||||||
/*!
|
* If an ID3v1 tag exists, it will be updated as well.
|
||||||
* This set of flags is used for various operations and is suitable for
|
*/
|
||||||
* being OR-ed together.
|
PropertyMap setProperties(const PropertyMap &) override;
|
||||||
*/
|
|
||||||
enum TagTypes {
|
|
||||||
//! Empty set. Matches no tag types.
|
|
||||||
NoTags = 0x0000,
|
|
||||||
//! Matches ID3v1 tags.
|
|
||||||
ID3v1 = 0x0001,
|
|
||||||
//! Matches ID3v2 tags.
|
|
||||||
ID3v2 = 0x0002,
|
|
||||||
//! Matches APE tags.
|
|
||||||
APE = 0x0004,
|
|
||||||
//! Matches all tag types.
|
|
||||||
AllTags = 0xffff
|
|
||||||
};
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Constructs an MPC file from \a file. If \a readProperties is true the
|
* Returns the MPC::AudioProperties for this file.
|
||||||
* file's audio properties will also be read.
|
* If no audio properties were read then this will return a null pointer.
|
||||||
*
|
*/
|
||||||
* \note In the current implementation, \a propertiesStyle is ignored.
|
AudioProperties *audioProperties() const override;
|
||||||
*/
|
|
||||||
File(FileName file, bool readProperties = true,
|
|
||||||
Properties::ReadStyle propertiesStyle = Properties::Average);
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Constructs an MPC file from \a stream. If \a readProperties is true the
|
* Saves the file.
|
||||||
* file's audio properties will also be read.
|
*
|
||||||
*
|
* This returns true if the save was successful.
|
||||||
* \note TagLib will *not* take ownership of the stream, the caller is
|
*/
|
||||||
* responsible for deleting it after the File object.
|
bool save() override;
|
||||||
*
|
|
||||||
* \note In the current implementation, \a propertiesStyle is ignored.
|
|
||||||
*/
|
|
||||||
File(IOStream *stream, bool readProperties = true,
|
|
||||||
Properties::ReadStyle propertiesStyle = Properties::Average);
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Destroys this instance of the File.
|
* Returns a pointer to the ID3v1 tag of the file.
|
||||||
*/
|
*
|
||||||
virtual ~File();
|
* If \a create is false (the default) this returns a null pointer if there is no valid APE tag.
|
||||||
|
* If \a create is true it will create an APE tag if one does not exist and returns a valid pointer.
|
||||||
|
*
|
||||||
|
* \note This may return a valid pointer regardless of whether or not the file on disk has an ID3v1 tag.
|
||||||
|
* Use hasID3v1Tag() to check if the file on disk actually has an ID3v1 tag.
|
||||||
|
*
|
||||||
|
* \note The Tag <b>is still</b> owned by the MPEG::File and should not be deleted by the user.
|
||||||
|
* It will be deleted when the file (object) is destroyed.
|
||||||
|
*
|
||||||
|
* \see hasID3v1Tag()
|
||||||
|
*/
|
||||||
|
ID3v1::Tag *ID3v1Tag(bool create = false);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the Tag for this file. This will be an APE tag, an ID3v1 tag
|
* Returns a pointer to the APE tag of the file.
|
||||||
* or a combination of the two.
|
*
|
||||||
*/
|
* If \a create is false (the default) this may return a null pointer
|
||||||
virtual Strawberry_TagLib::TagLib::Tag *tag() const;
|
* if there is no valid APE tag.
|
||||||
|
* If \a create is true it will create an APE tag if one does not exist and returns a valid pointer.
|
||||||
|
* If there already be an ID3v1 tag, the new APE tag will be placed before it.
|
||||||
|
*
|
||||||
|
* \note This may return a valid pointer regardless of whether or not the file on disk has an APE tag.
|
||||||
|
* Use hasAPETag() to check if the file on disk actually has an APE tag.
|
||||||
|
*
|
||||||
|
* \note The Tag <b>is still</b> owned by the MPEG::File and should not be deleted by the user.
|
||||||
|
* It will be deleted when the file (object) is destroyed.
|
||||||
|
*
|
||||||
|
* \see hasAPETag()
|
||||||
|
*/
|
||||||
|
APE::Tag *APETag(bool create = false);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Implements the unified property interface -- export function.
|
* This will remove the tags that match the OR-ed together TagTypes from the file. By default it removes all tags.
|
||||||
* If the file contains both an APE and an ID3v1 tag, only the APE
|
*
|
||||||
* tag will be converted to the PropertyMap.
|
* \warning This will also invalidate pointers to the tags as their memory will be freed.
|
||||||
*/
|
*
|
||||||
PropertyMap properties() const;
|
* \note In order to make the removal permanent save() still needs to be called.
|
||||||
|
*/
|
||||||
|
void strip(int tags = AllTags);
|
||||||
|
|
||||||
void removeUnsupportedProperties(const StringList &properties);
|
/*!
|
||||||
|
* Returns whether or not the file on disk actually has an ID3v1 tag.
|
||||||
|
*
|
||||||
|
* \see ID3v1Tag()
|
||||||
|
*/
|
||||||
|
bool hasID3v1Tag() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Implements the unified property interface -- import function.
|
* Returns whether or not the file on disk actually has an APE tag.
|
||||||
* Affects only the APEv2 tag which will be created if necessary.
|
*
|
||||||
* If an ID3v1 tag exists, it will be updated as well.
|
* \see APETag()
|
||||||
*/
|
*/
|
||||||
PropertyMap setProperties(const PropertyMap &);
|
bool hasAPETag() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the MPC::Properties for this file. If no audio properties
|
* Returns whether or not the given \a stream can be opened as an MPC file.
|
||||||
* were read then this will return a null pointer.
|
*
|
||||||
*/
|
* \note This method is designed to do a quick check. The result may not necessarily be correct.
|
||||||
virtual Properties *audioProperties() const;
|
*/
|
||||||
|
static bool isSupported(IOStream *stream);
|
||||||
|
|
||||||
/*!
|
private:
|
||||||
* Saves the file.
|
File(const File&);
|
||||||
*
|
File &operator=(const File&);
|
||||||
* This returns true if the save was successful.
|
|
||||||
*/
|
|
||||||
virtual bool save();
|
|
||||||
|
|
||||||
/*!
|
void read(bool readProperties);
|
||||||
* Returns a pointer to the ID3v1 tag of the file.
|
|
||||||
*
|
|
||||||
* If \a create is false (the default) this returns a null pointer
|
|
||||||
* if there is no valid APE tag. If \a create is true it will create
|
|
||||||
* an APE tag if one does not exist and returns a valid pointer.
|
|
||||||
*
|
|
||||||
* \note This may return a valid pointer regardless of whether or not the
|
|
||||||
* file on disk has an ID3v1 tag. Use hasID3v1Tag() to check if the file
|
|
||||||
* on disk actually has an ID3v1 tag.
|
|
||||||
*
|
|
||||||
* \note The Tag <b>is still</b> owned by the MPEG::File and should not be
|
|
||||||
* deleted by the user. It will be deleted when the file (object) is
|
|
||||||
* destroyed.
|
|
||||||
*
|
|
||||||
* \see hasID3v1Tag()
|
|
||||||
*/
|
|
||||||
ID3v1::Tag *ID3v1Tag(bool create = false);
|
|
||||||
|
|
||||||
/*!
|
class FilePrivate;
|
||||||
* Returns a pointer to the APE tag of the file.
|
FilePrivate *d;
|
||||||
*
|
};
|
||||||
* If \a create is false (the default) this may return a null pointer
|
} // namespace MPC
|
||||||
* if there is no valid APE tag. If \a create is true it will create
|
} // namespace TagLib
|
||||||
* an APE tag if one does not exist and returns a valid pointer. If
|
} // namespace Strawberry_TagLib
|
||||||
* there already be an ID3v1 tag, the new APE tag will be placed before it.
|
|
||||||
*
|
|
||||||
* \note This may return a valid pointer regardless of whether or not the
|
|
||||||
* file on disk has an APE tag. Use hasAPETag() to check if the file
|
|
||||||
* on disk actually has an APE tag.
|
|
||||||
*
|
|
||||||
* \note The Tag <b>is still</b> owned by the MPEG::File and should not be
|
|
||||||
* deleted by the user. It will be deleted when the file (object) is
|
|
||||||
* destroyed.
|
|
||||||
*
|
|
||||||
* \see hasAPETag()
|
|
||||||
*/
|
|
||||||
APE::Tag *APETag(bool create = false);
|
|
||||||
|
|
||||||
/*!
|
|
||||||
* This will remove the tags that match the OR-ed together TagTypes from the
|
|
||||||
* file. By default it removes all tags.
|
|
||||||
*
|
|
||||||
* \warning This will also invalidate pointers to the tags
|
|
||||||
* as their memory will be freed.
|
|
||||||
*
|
|
||||||
* \note In order to make the removal permanent save() still needs to be called.
|
|
||||||
*/
|
|
||||||
void strip(int tags = AllTags);
|
|
||||||
|
|
||||||
/*!
|
|
||||||
* \deprecated
|
|
||||||
* \see strip
|
|
||||||
*/
|
|
||||||
TAGLIB_DEPRECATED void remove(int tags = AllTags);
|
|
||||||
|
|
||||||
/*!
|
|
||||||
* Returns whether or not the file on disk actually has an ID3v1 tag.
|
|
||||||
*
|
|
||||||
* \see ID3v1Tag()
|
|
||||||
*/
|
|
||||||
bool hasID3v1Tag() const;
|
|
||||||
|
|
||||||
/*!
|
|
||||||
* Returns whether or not the file on disk actually has an APE tag.
|
|
||||||
*
|
|
||||||
* \see APETag()
|
|
||||||
*/
|
|
||||||
bool hasAPETag() const;
|
|
||||||
|
|
||||||
/*!
|
|
||||||
* Returns whether or not the given \a stream can be opened as an MPC
|
|
||||||
* file.
|
|
||||||
*
|
|
||||||
* \note This method is designed to do a quick check. The result may
|
|
||||||
* not necessarily be correct.
|
|
||||||
*/
|
|
||||||
static bool isSupported(IOStream *stream);
|
|
||||||
|
|
||||||
private:
|
|
||||||
File(const File &);
|
|
||||||
File &operator=(const File &);
|
|
||||||
|
|
||||||
void read(bool readProperties);
|
|
||||||
|
|
||||||
class FilePrivate;
|
|
||||||
FilePrivate *d;
|
|
||||||
};
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
301
3rdparty/taglib/mpc/mpcproperties.cpp
vendored
301
3rdparty/taglib/mpc/mpcproperties.cpp
vendored
@@ -23,31 +23,35 @@
|
|||||||
* http://www.mozilla.org/MPL/ *
|
* http://www.mozilla.org/MPL/ *
|
||||||
***************************************************************************/
|
***************************************************************************/
|
||||||
|
|
||||||
#include <tstring.h>
|
|
||||||
#include <tdebug.h>
|
|
||||||
#include <bitset>
|
#include <bitset>
|
||||||
#include <math.h>
|
#include <cmath>
|
||||||
|
|
||||||
|
#include "tstring.h"
|
||||||
|
#include "tdebug.h"
|
||||||
|
|
||||||
#include "mpcproperties.h"
|
#include "mpcproperties.h"
|
||||||
#include "mpcfile.h"
|
#include "mpcfile.h"
|
||||||
|
|
||||||
using namespace Strawberry_TagLib::TagLib;
|
using namespace Strawberry_TagLib::TagLib;
|
||||||
|
|
||||||
class MPC::Properties::PropertiesPrivate
|
namespace {
|
||||||
{
|
const unsigned int HeaderSize = 56;
|
||||||
public:
|
}
|
||||||
PropertiesPrivate() :
|
|
||||||
version(0),
|
class MPC::AudioProperties::AudioPropertiesPrivate {
|
||||||
length(0),
|
public:
|
||||||
bitrate(0),
|
explicit AudioPropertiesPrivate() : version(0),
|
||||||
sampleRate(0),
|
length(0),
|
||||||
channels(0),
|
bitrate(0),
|
||||||
totalFrames(0),
|
sampleRate(0),
|
||||||
sampleFrames(0),
|
channels(0),
|
||||||
trackGain(0),
|
totalFrames(0),
|
||||||
trackPeak(0),
|
sampleFrames(0),
|
||||||
albumGain(0),
|
trackGain(0),
|
||||||
albumPeak(0) {}
|
trackPeak(0),
|
||||||
|
albumGain(0),
|
||||||
|
albumPeak(0) {}
|
||||||
|
|
||||||
int version;
|
int version;
|
||||||
int length;
|
int length;
|
||||||
@@ -56,105 +60,79 @@ public:
|
|||||||
int channels;
|
int channels;
|
||||||
unsigned int totalFrames;
|
unsigned int totalFrames;
|
||||||
unsigned int sampleFrames;
|
unsigned int sampleFrames;
|
||||||
int trackGain;
|
unsigned int trackGain;
|
||||||
int trackPeak;
|
unsigned int trackPeak;
|
||||||
int albumGain;
|
unsigned int albumGain;
|
||||||
int albumPeak;
|
unsigned int albumPeak;
|
||||||
};
|
};
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
// public members
|
// public members
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
MPC::Properties::Properties(const ByteVector &data, long streamLength, ReadStyle style) :
|
MPC::AudioProperties::AudioProperties(File *file, long long streamLength, ReadStyle) : Strawberry_TagLib::TagLib::AudioProperties(), d(new AudioPropertiesPrivate()) {
|
||||||
AudioProperties(style),
|
|
||||||
d(new PropertiesPrivate())
|
|
||||||
{
|
|
||||||
readSV7(data, streamLength);
|
|
||||||
}
|
|
||||||
|
|
||||||
MPC::Properties::Properties(File *file, long streamLength, ReadStyle style) :
|
|
||||||
AudioProperties(style),
|
|
||||||
d(new PropertiesPrivate())
|
|
||||||
{
|
|
||||||
ByteVector magic = file->readBlock(4);
|
ByteVector magic = file->readBlock(4);
|
||||||
if(magic == "MPCK") {
|
if (magic == "MPCK") {
|
||||||
// Musepack version 8
|
// Musepack version 8
|
||||||
readSV8(file, streamLength);
|
readSV8(file, streamLength);
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
// Musepack version 7 or older, fixed size header
|
// Musepack version 7 or older, fixed size header
|
||||||
readSV7(magic + file->readBlock(MPC::HeaderSize - 4), streamLength);
|
readSV7(magic + file->readBlock(HeaderSize - 4), streamLength);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
MPC::Properties::~Properties()
|
MPC::AudioProperties::~AudioProperties() {
|
||||||
{
|
|
||||||
delete d;
|
delete d;
|
||||||
}
|
}
|
||||||
|
|
||||||
int MPC::Properties::length() const
|
int MPC::AudioProperties::lengthInSeconds() const {
|
||||||
{
|
|
||||||
return lengthInSeconds();
|
|
||||||
}
|
|
||||||
|
|
||||||
int MPC::Properties::lengthInSeconds() const
|
|
||||||
{
|
|
||||||
return d->length / 1000;
|
return d->length / 1000;
|
||||||
}
|
}
|
||||||
|
|
||||||
int MPC::Properties::lengthInMilliseconds() const
|
int MPC::AudioProperties::lengthInMilliseconds() const {
|
||||||
{
|
|
||||||
return d->length;
|
return d->length;
|
||||||
}
|
}
|
||||||
|
|
||||||
int MPC::Properties::bitrate() const
|
int MPC::AudioProperties::bitrate() const {
|
||||||
{
|
|
||||||
return d->bitrate;
|
return d->bitrate;
|
||||||
}
|
}
|
||||||
|
|
||||||
int MPC::Properties::sampleRate() const
|
int MPC::AudioProperties::sampleRate() const {
|
||||||
{
|
|
||||||
return d->sampleRate;
|
return d->sampleRate;
|
||||||
}
|
}
|
||||||
|
|
||||||
int MPC::Properties::channels() const
|
int MPC::AudioProperties::channels() const {
|
||||||
{
|
|
||||||
return d->channels;
|
return d->channels;
|
||||||
}
|
}
|
||||||
|
|
||||||
int MPC::Properties::mpcVersion() const
|
int MPC::AudioProperties::mpcVersion() const {
|
||||||
{
|
|
||||||
return d->version;
|
return d->version;
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned int MPC::Properties::totalFrames() const
|
unsigned int MPC::AudioProperties::totalFrames() const {
|
||||||
{
|
|
||||||
return d->totalFrames;
|
return d->totalFrames;
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned int MPC::Properties::sampleFrames() const
|
unsigned int MPC::AudioProperties::sampleFrames() const {
|
||||||
{
|
|
||||||
return d->sampleFrames;
|
return d->sampleFrames;
|
||||||
}
|
}
|
||||||
|
|
||||||
int MPC::Properties::trackGain() const
|
int MPC::AudioProperties::trackGain() const {
|
||||||
{
|
|
||||||
return d->trackGain;
|
return d->trackGain;
|
||||||
}
|
}
|
||||||
|
|
||||||
int MPC::Properties::trackPeak() const
|
int MPC::AudioProperties::trackPeak() const {
|
||||||
{
|
|
||||||
return d->trackPeak;
|
return d->trackPeak;
|
||||||
}
|
}
|
||||||
|
|
||||||
int MPC::Properties::albumGain() const
|
int MPC::AudioProperties::albumGain() const {
|
||||||
{
|
|
||||||
return d->albumGain;
|
return d->albumGain;
|
||||||
}
|
}
|
||||||
|
|
||||||
int MPC::Properties::albumPeak() const
|
int MPC::AudioProperties::albumPeak() const {
|
||||||
{
|
|
||||||
return d->albumPeak;
|
return d->albumPeak;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -162,106 +140,111 @@ int MPC::Properties::albumPeak() const
|
|||||||
// private members
|
// private members
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
namespace
|
namespace {
|
||||||
{
|
unsigned long readSize(File *file, size_t &sizeLength, bool &eof) {
|
||||||
unsigned long readSize(File *file, unsigned int &sizeLength, bool &eof)
|
|
||||||
{
|
|
||||||
sizeLength = 0;
|
|
||||||
eof = false;
|
|
||||||
|
|
||||||
unsigned char tmp;
|
sizeLength = 0;
|
||||||
unsigned long size = 0;
|
eof = false;
|
||||||
|
|
||||||
do {
|
unsigned char tmp;
|
||||||
const ByteVector b = file->readBlock(1);
|
unsigned long size = 0;
|
||||||
if(b.isEmpty()) {
|
|
||||||
eof = true;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
tmp = b[0];
|
do {
|
||||||
size = (size << 7) | (tmp & 0x7F);
|
const ByteVector b = file->readBlock(1);
|
||||||
sizeLength++;
|
if (b.isEmpty()) {
|
||||||
} while((tmp & 0x80));
|
eof = true;
|
||||||
return size;
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
tmp = b[0];
|
||||||
|
size = (size << 7) | (tmp & 0x7F);
|
||||||
|
sizeLength++;
|
||||||
}
|
}
|
||||||
|
while ((tmp & 0x80));
|
||||||
|
|
||||||
unsigned long readSize(const ByteVector &data, unsigned int &pos)
|
return size;
|
||||||
{
|
|
||||||
unsigned char tmp;
|
|
||||||
unsigned long size = 0;
|
|
||||||
|
|
||||||
do {
|
|
||||||
tmp = data[pos++];
|
|
||||||
size = (size << 7) | (tmp & 0x7F);
|
|
||||||
} while((tmp & 0x80) && (pos < data.size()));
|
|
||||||
return size;
|
|
||||||
}
|
|
||||||
|
|
||||||
// This array looks weird, but the same as original MusePack code found at:
|
|
||||||
// https://www.musepack.net/index.php?pg=src
|
|
||||||
const unsigned short sftable [8] = { 44100, 48000, 37800, 32000, 0, 0, 0, 0 };
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void MPC::Properties::readSV8(File *file, long streamLength)
|
unsigned long readSize(const ByteVector &data, size_t &pos) {
|
||||||
{
|
|
||||||
|
unsigned char tmp;
|
||||||
|
unsigned long size = 0;
|
||||||
|
|
||||||
|
do {
|
||||||
|
tmp = data[pos++];
|
||||||
|
size = (size << 7) | (tmp & 0x7F);
|
||||||
|
}
|
||||||
|
while ((tmp & 0x80) && (pos < data.size()));
|
||||||
|
|
||||||
|
return size;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
// This array looks weird, but the same as original MusePack code found at:
|
||||||
|
// https://www.musepack.net/index.php?pg=src
|
||||||
|
const unsigned short sftable[8] = { 44100, 48000, 37800, 32000, 0, 0, 0, 0 };
|
||||||
|
} // namespace
|
||||||
|
|
||||||
|
void MPC::AudioProperties::readSV8(File *file, long long streamLength) {
|
||||||
|
|
||||||
bool readSH = false, readRG = false;
|
bool readSH = false, readRG = false;
|
||||||
|
|
||||||
while(!readSH && !readRG) {
|
while (!readSH && !readRG) {
|
||||||
const ByteVector packetType = file->readBlock(2);
|
const ByteVector packetType = file->readBlock(2);
|
||||||
|
|
||||||
unsigned int packetSizeLength;
|
size_t packetSizeLength;
|
||||||
bool eof;
|
bool eof;
|
||||||
const unsigned long packetSize = readSize(file, packetSizeLength, eof);
|
const size_t packetSize = readSize(file, packetSizeLength, eof);
|
||||||
if(eof) {
|
if (eof) {
|
||||||
debug("MPC::Properties::readSV8() - Reached to EOF.");
|
debug("MPC::AudioProperties::readSV8() - Reached to EOF.");
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
const unsigned long dataSize = packetSize - 2 - packetSizeLength;
|
const size_t dataSize = packetSize - 2 - packetSizeLength;
|
||||||
|
|
||||||
const ByteVector data = file->readBlock(dataSize);
|
const ByteVector data = file->readBlock(dataSize);
|
||||||
if(data.size() != dataSize) {
|
if (data.size() != dataSize) {
|
||||||
debug("MPC::Properties::readSV8() - dataSize doesn't match the actual data size.");
|
debug("MPC::AudioProperties::readSV8() - dataSize doesn't match the actual data size.");
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
if(packetType == "SH") {
|
if (packetType == "SH") {
|
||||||
// Stream Header
|
// Stream Header
|
||||||
// http://trac.musepack.net/wiki/SV8Specification#StreamHeaderPacket
|
// http://trac.musepack.net/wiki/SV8Specification#StreamHeaderPacket
|
||||||
|
|
||||||
if(dataSize <= 5) {
|
if (dataSize <= 5) {
|
||||||
debug("MPC::Properties::readSV8() - \"SH\" packet is too short to parse.");
|
debug("MPC::AudioProperties::readSV8() - \"SH\" packet is too short to parse.");
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
readSH = true;
|
readSH = true;
|
||||||
|
|
||||||
unsigned int pos = 4;
|
size_t pos = 4;
|
||||||
d->version = data[pos];
|
d->version = data[pos];
|
||||||
pos += 1;
|
pos += 1;
|
||||||
d->sampleFrames = readSize(data, pos);
|
d->sampleFrames = readSize(data, pos);
|
||||||
if(pos > dataSize - 3) {
|
if (pos > dataSize - 3) {
|
||||||
debug("MPC::Properties::readSV8() - \"SH\" packet is corrupt.");
|
debug("MPC::AudioProperties::readSV8() - \"SH\" packet is corrupt.");
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
const unsigned long begSilence = readSize(data, pos);
|
const unsigned long begSilence = readSize(data, pos);
|
||||||
if(pos > dataSize - 2) {
|
if (pos > dataSize - 2) {
|
||||||
debug("MPC::Properties::readSV8() - \"SH\" packet is corrupt.");
|
debug("MPC::AudioProperties::readSV8() - \"SH\" packet is corrupt.");
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
const unsigned short flags = data.toUShort(pos, true);
|
const unsigned short flags = data.toUInt16BE(pos);
|
||||||
pos += 2;
|
pos += 2;
|
||||||
|
|
||||||
d->sampleRate = sftable[(flags >> 13) & 0x07];
|
d->sampleRate = sftable[(flags >> 13) & 0x07];
|
||||||
d->channels = ((flags >> 4) & 0x0F) + 1;
|
d->channels = ((flags >> 4) & 0x0F) + 1;
|
||||||
|
|
||||||
const unsigned int frameCount = d->sampleFrames - begSilence;
|
const unsigned int frameCount = d->sampleFrames - begSilence;
|
||||||
if(frameCount > 0 && d->sampleRate > 0) {
|
if (frameCount > 0 && d->sampleRate > 0) {
|
||||||
const double length = frameCount * 1000.0 / d->sampleRate;
|
const double length = frameCount * 1000.0 / d->sampleRate;
|
||||||
d->length = static_cast<int>(length + 0.5);
|
d->length = static_cast<int>(length + 0.5);
|
||||||
d->bitrate = static_cast<int>(streamLength * 8.0 / length + 0.5);
|
d->bitrate = static_cast<int>(streamLength * 8.0 / length + 0.5);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -269,23 +252,23 @@ void MPC::Properties::readSV8(File *file, long streamLength)
|
|||||||
// Replay Gain
|
// Replay Gain
|
||||||
// http://trac.musepack.net/wiki/SV8Specification#ReplaygainPacket
|
// http://trac.musepack.net/wiki/SV8Specification#ReplaygainPacket
|
||||||
|
|
||||||
if(dataSize <= 9) {
|
if (dataSize <= 9) {
|
||||||
debug("MPC::Properties::readSV8() - \"RG\" packet is too short to parse.");
|
debug("MPC::AudioProperties::readSV8() - \"RG\" packet is too short to parse.");
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
readRG = true;
|
readRG = true;
|
||||||
|
|
||||||
const int replayGainVersion = data[0];
|
const int replayGainVersion = data[0];
|
||||||
if(replayGainVersion == 1) {
|
if (replayGainVersion == 1) {
|
||||||
d->trackGain = data.toShort(1, true);
|
d->trackGain = data.toUInt16BE(1);
|
||||||
d->trackPeak = data.toShort(3, true);
|
d->trackPeak = data.toUInt16BE(3);
|
||||||
d->albumGain = data.toShort(5, true);
|
d->albumGain = data.toUInt16BE(5);
|
||||||
d->albumPeak = data.toShort(7, true);
|
d->albumPeak = data.toUInt16BE(7);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
else if(packetType == "SE") {
|
else if (packetType == "SE") {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -293,49 +276,50 @@ void MPC::Properties::readSV8(File *file, long streamLength)
|
|||||||
file->seek(dataSize, File::Current);
|
file->seek(dataSize, File::Current);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void MPC::Properties::readSV7(const ByteVector &data, long streamLength)
|
void MPC::AudioProperties::readSV7(const ByteVector &data, long long streamLength) {
|
||||||
{
|
|
||||||
if(data.startsWith("MP+")) {
|
if (data.startsWith("MP+")) {
|
||||||
d->version = data[3] & 15;
|
d->version = data[3] & 15;
|
||||||
if(d->version < 7)
|
if (d->version < 7)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
d->totalFrames = data.toUInt(4, false);
|
d->totalFrames = data.toUInt32LE(4);
|
||||||
|
|
||||||
const unsigned int flags = data.toUInt(8, false);
|
const unsigned int flags = data.toUInt32LE(8);
|
||||||
d->sampleRate = sftable[(flags >> 16) & 0x03];
|
d->sampleRate = sftable[(flags >> 16) & 0x03];
|
||||||
d->channels = 2;
|
d->channels = 2;
|
||||||
|
|
||||||
const unsigned int gapless = data.toUInt(5, false);
|
const unsigned int gapless = data.toUInt32LE(5);
|
||||||
|
|
||||||
d->trackGain = data.toShort(14, false);
|
d->trackGain = data.toUInt16LE(14);
|
||||||
d->trackPeak = data.toUShort(12, false);
|
d->trackPeak = data.toUInt16LE(12);
|
||||||
d->albumGain = data.toShort(18, false);
|
d->albumGain = data.toUInt16LE(18);
|
||||||
d->albumPeak = data.toUShort(16, false);
|
d->albumPeak = data.toUInt16LE(16);
|
||||||
|
|
||||||
// convert gain info
|
// convert gain info
|
||||||
if(d->trackGain != 0) {
|
if (d->trackGain != 0) {
|
||||||
int tmp = (int)((64.82 - (short)d->trackGain / 100.) * 256. + .5);
|
int tmp = static_cast<int>((64.82 - static_cast<short>(d->trackGain) / 100.) * 256. + .5);
|
||||||
if(tmp >= (1 << 16) || tmp < 0) tmp = 0;
|
if (tmp >= (1 << 16) || tmp < 0) tmp = 0;
|
||||||
d->trackGain = tmp;
|
d->trackGain = tmp;
|
||||||
}
|
}
|
||||||
|
|
||||||
if(d->albumGain != 0) {
|
if (d->albumGain != 0) {
|
||||||
int tmp = (int)((64.82 - d->albumGain / 100.) * 256. + .5);
|
int tmp = static_cast<int>((64.82 - d->albumGain / 100.) * 256. + .5);
|
||||||
if(tmp >= (1 << 16) || tmp < 0) tmp = 0;
|
if (tmp >= (1 << 16) || tmp < 0) tmp = 0;
|
||||||
d->albumGain = tmp;
|
d->albumGain = tmp;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (d->trackPeak != 0)
|
if (d->trackPeak != 0)
|
||||||
d->trackPeak = (int)(log10((double)d->trackPeak) * 20 * 256 + .5);
|
d->trackPeak = static_cast<int>(log10(static_cast<double>(d->trackPeak)) * 20 * 256 + .5);
|
||||||
|
|
||||||
if (d->albumPeak != 0)
|
if (d->albumPeak != 0)
|
||||||
d->albumPeak = (int)(log10((double)d->albumPeak) * 20 * 256 + .5);
|
d->albumPeak = static_cast<int>(log10(static_cast<double>(d->albumPeak)) * 20 * 256 + .5);
|
||||||
|
|
||||||
bool trueGapless = (gapless >> 31) & 0x0001;
|
bool trueGapless = (gapless >> 31) & 0x0001;
|
||||||
if(trueGapless) {
|
if (trueGapless) {
|
||||||
unsigned int lastFrameSamples = (gapless >> 20) & 0x07FF;
|
unsigned int lastFrameSamples = (gapless >> 20) & 0x07FF;
|
||||||
d->sampleFrames = d->totalFrames * 1152 - lastFrameSamples;
|
d->sampleFrames = d->totalFrames * 1152 - lastFrameSamples;
|
||||||
}
|
}
|
||||||
@@ -343,26 +327,27 @@ void MPC::Properties::readSV7(const ByteVector &data, long streamLength)
|
|||||||
d->sampleFrames = d->totalFrames * 1152 - 576;
|
d->sampleFrames = d->totalFrames * 1152 - 576;
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
const unsigned int headerData = data.toUInt(0, false);
|
const unsigned int headerData = data.toUInt32LE(0);
|
||||||
|
|
||||||
d->bitrate = (headerData >> 23) & 0x01ff;
|
d->bitrate = (headerData >> 23) & 0x01ff;
|
||||||
d->version = (headerData >> 11) & 0x03ff;
|
d->version = (headerData >> 11) & 0x03ff;
|
||||||
d->sampleRate = 44100;
|
d->sampleRate = 44100;
|
||||||
d->channels = 2;
|
d->channels = 2;
|
||||||
|
|
||||||
if(d->version >= 5)
|
if (d->version >= 5)
|
||||||
d->totalFrames = data.toUInt(4, false);
|
d->totalFrames = data.toUInt32LE(4);
|
||||||
else
|
else
|
||||||
d->totalFrames = data.toUShort(6, false);
|
d->totalFrames = data.toUInt16LE(6);
|
||||||
|
|
||||||
d->sampleFrames = d->totalFrames * 1152 - 576;
|
d->sampleFrames = d->totalFrames * 1152 - 576;
|
||||||
}
|
}
|
||||||
|
|
||||||
if(d->sampleFrames > 0 && d->sampleRate > 0) {
|
if (d->sampleFrames > 0 && d->sampleRate > 0) {
|
||||||
const double length = d->sampleFrames * 1000.0 / d->sampleRate;
|
const double length = d->sampleFrames * 1000.0 / d->sampleRate;
|
||||||
d->length = static_cast<int>(length + 0.5);
|
d->length = static_cast<int>(length + 0.5);
|
||||||
|
|
||||||
if(d->bitrate == 0)
|
if (d->bitrate == 0)
|
||||||
d->bitrate = static_cast<int>(streamLength * 8.0 / length + 0.5);
|
d->bitrate = static_cast<int>(streamLength * 8.0 / length + 0.5);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
184
3rdparty/taglib/mpc/mpcproperties.h
vendored
184
3rdparty/taglib/mpc/mpcproperties.h
vendored
@@ -32,129 +32,101 @@
|
|||||||
namespace Strawberry_TagLib {
|
namespace Strawberry_TagLib {
|
||||||
namespace TagLib {
|
namespace TagLib {
|
||||||
|
|
||||||
namespace MPC {
|
namespace MPC {
|
||||||
|
|
||||||
class File;
|
class File;
|
||||||
|
|
||||||
static const unsigned int HeaderSize = 8 * 7;
|
//! An implementation of audio property reading for MPC
|
||||||
|
|
||||||
//! An implementation of audio property reading for MPC
|
/*!
|
||||||
|
* This reads the data from an MPC stream found in the AudioProperties API.
|
||||||
|
*/
|
||||||
|
|
||||||
/*!
|
class TAGLIB_EXPORT AudioProperties : public Strawberry_TagLib::TagLib::AudioProperties {
|
||||||
* This reads the data from an MPC stream found in the AudioProperties
|
public:
|
||||||
* API.
|
/*!
|
||||||
*/
|
* Create an instance of MPC::AudioProperties with the data read directly from a MPC::File.
|
||||||
|
*/
|
||||||
|
explicit AudioProperties(File *file, long long streamLength, ReadStyle style = Average);
|
||||||
|
|
||||||
class TAGLIB_EXPORT Properties : public AudioProperties
|
/*!
|
||||||
{
|
* Destroys this MPC::AudioProperties instance.
|
||||||
public:
|
*/
|
||||||
/*!
|
~AudioProperties() override;
|
||||||
* Create an instance of MPC::Properties with the data read from the
|
|
||||||
* ByteVector \a data.
|
|
||||||
*
|
|
||||||
* This constructor is deprecated. It only works for MPC version up to 7.
|
|
||||||
*/
|
|
||||||
Properties(const ByteVector &data, long streamLength, ReadStyle style = Average);
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Create an instance of MPC::Properties with the data read directly
|
* Returns the length of the file in seconds.
|
||||||
* from a MPC::File.
|
* The length is rounded down to the nearest whole second.
|
||||||
*/
|
*
|
||||||
Properties(File *file, long streamLength, ReadStyle style = Average);
|
* \see lengthInMilliseconds()
|
||||||
|
*/
|
||||||
|
int lengthInSeconds() const override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Destroys this MPC::Properties instance.
|
* Returns the length of the file in milliseconds.
|
||||||
*/
|
*
|
||||||
virtual ~Properties();
|
* \see lengthInSeconds()
|
||||||
|
*/
|
||||||
|
int lengthInMilliseconds() const override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the length of the file in seconds. The length is rounded down to
|
* Returns the average bit rate of the file in kb/s.
|
||||||
* the nearest whole second.
|
*/
|
||||||
*
|
int bitrate() const override;
|
||||||
* \note This method is just an alias of lengthInSeconds().
|
|
||||||
*
|
|
||||||
* \deprecated
|
|
||||||
*/
|
|
||||||
TAGLIB_DEPRECATED virtual int length() const;
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the length of the file in seconds. The length is rounded down to
|
* Returns the sample rate in Hz.
|
||||||
* the nearest whole second.
|
*/
|
||||||
*
|
int sampleRate() const override;
|
||||||
* \see lengthInMilliseconds()
|
|
||||||
*/
|
|
||||||
// BIC: make virtual
|
|
||||||
int lengthInSeconds() const;
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the length of the file in milliseconds.
|
* Returns the number of audio channels.
|
||||||
*
|
*/
|
||||||
* \see lengthInSeconds()
|
int channels() const override;
|
||||||
*/
|
|
||||||
// BIC: make virtual
|
|
||||||
int lengthInMilliseconds() const;
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the average bit rate of the file in kb/s.
|
* Returns the version of the bitstream (SV4-SV8)
|
||||||
*/
|
*/
|
||||||
virtual int bitrate() const;
|
int mpcVersion() const;
|
||||||
|
|
||||||
/*!
|
unsigned int totalFrames() const;
|
||||||
* Returns the sample rate in Hz.
|
unsigned int sampleFrames() const;
|
||||||
*/
|
|
||||||
virtual int sampleRate() const;
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the number of audio channels.
|
* Returns the track gain as an integer value,
|
||||||
*/
|
* to convert to dB: trackGain in dB = 64.82 - (trackGain / 256)
|
||||||
virtual int channels() const;
|
*/
|
||||||
|
int trackGain() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the version of the bitstream (SV4-SV8)
|
* Returns the track peak as an integer value,
|
||||||
*/
|
* to convert to dB: trackPeak in dB = trackPeak / 256
|
||||||
int mpcVersion() const;
|
* to convert to floating [-1..1]: trackPeak = 10^(trackPeak / 256 / 20)/32768
|
||||||
|
*/
|
||||||
|
int trackPeak() const;
|
||||||
|
|
||||||
unsigned int totalFrames() const;
|
/*!
|
||||||
unsigned int sampleFrames() const;
|
* Returns the album gain as an integer value,
|
||||||
|
* to convert to dB: albumGain in dB = 64.82 - (albumGain / 256)
|
||||||
|
*/
|
||||||
|
int albumGain() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the track gain as an integer value,
|
* Returns the album peak as an integer value,
|
||||||
* to convert to dB: trackGain in dB = 64.82 - (trackGain / 256)
|
* to convert to dB: albumPeak in dB = albumPeak / 256
|
||||||
*/
|
* to convert to floating [-1..1]: albumPeak = 10^(albumPeak / 256 / 20)/32768
|
||||||
int trackGain() const;
|
*/
|
||||||
|
int albumPeak() const;
|
||||||
|
|
||||||
/*!
|
private:
|
||||||
* Returns the track peak as an integer value,
|
void readSV7(const ByteVector &data, long long streamLength);
|
||||||
* to convert to dB: trackPeak in dB = trackPeak / 256
|
void readSV8(File *file, long long streamLength);
|
||||||
* to convert to floating [-1..1]: trackPeak = 10^(trackPeak / 256 / 20)/32768
|
|
||||||
*/
|
|
||||||
int trackPeak() const;
|
|
||||||
|
|
||||||
/*!
|
class AudioPropertiesPrivate;
|
||||||
* Returns the album gain as an integer value,
|
AudioPropertiesPrivate *d;
|
||||||
* to convert to dB: albumGain in dB = 64.82 - (albumGain / 256)
|
};
|
||||||
*/
|
} // namespace MPC
|
||||||
int albumGain() const;
|
} // namespace TagLib
|
||||||
|
} // namespace Strawberry_TagLib
|
||||||
/*!
|
|
||||||
* Returns the album peak as an integer value,
|
|
||||||
* to convert to dB: albumPeak in dB = albumPeak / 256
|
|
||||||
* to convert to floating [-1..1]: albumPeak = 10^(albumPeak / 256 / 20)/32768
|
|
||||||
*/
|
|
||||||
int albumPeak() const;
|
|
||||||
|
|
||||||
private:
|
|
||||||
Properties(const Properties &);
|
|
||||||
Properties &operator=(const Properties &);
|
|
||||||
|
|
||||||
void readSV7(const ByteVector &data, long streamLength);
|
|
||||||
void readSV8(File *file, long streamLength);
|
|
||||||
|
|
||||||
class PropertiesPrivate;
|
|
||||||
PropertiesPrivate *d;
|
|
||||||
};
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
427
3rdparty/taglib/mpeg/id3v1/id3v1genres.cpp
vendored
427
3rdparty/taglib/mpeg/id3v1/id3v1genres.cpp
vendored
@@ -27,239 +27,242 @@
|
|||||||
|
|
||||||
using namespace Strawberry_TagLib::TagLib;
|
using namespace Strawberry_TagLib::TagLib;
|
||||||
|
|
||||||
namespace
|
namespace {
|
||||||
{
|
const wchar_t *genres[] = {
|
||||||
const wchar_t *genres[] = {
|
L"Blues",
|
||||||
L"Blues",
|
L"Classic Rock",
|
||||||
L"Classic Rock",
|
L"Country",
|
||||||
L"Country",
|
L"Dance",
|
||||||
L"Dance",
|
L"Disco",
|
||||||
L"Disco",
|
L"Funk",
|
||||||
L"Funk",
|
L"Grunge",
|
||||||
L"Grunge",
|
L"Hip-Hop",
|
||||||
L"Hip-Hop",
|
L"Jazz",
|
||||||
L"Jazz",
|
L"Metal",
|
||||||
L"Metal",
|
L"New Age",
|
||||||
L"New Age",
|
L"Oldies",
|
||||||
L"Oldies",
|
L"Other",
|
||||||
L"Other",
|
L"Pop",
|
||||||
L"Pop",
|
L"R&B",
|
||||||
L"R&B",
|
L"Rap",
|
||||||
L"Rap",
|
L"Reggae",
|
||||||
L"Reggae",
|
L"Rock",
|
||||||
L"Rock",
|
L"Techno",
|
||||||
L"Techno",
|
L"Industrial",
|
||||||
L"Industrial",
|
L"Alternative",
|
||||||
L"Alternative",
|
L"Ska",
|
||||||
L"Ska",
|
L"Death Metal",
|
||||||
L"Death Metal",
|
L"Pranks",
|
||||||
L"Pranks",
|
L"Soundtrack",
|
||||||
L"Soundtrack",
|
L"Euro-Techno",
|
||||||
L"Euro-Techno",
|
L"Ambient",
|
||||||
L"Ambient",
|
L"Trip-Hop",
|
||||||
L"Trip-Hop",
|
L"Vocal",
|
||||||
L"Vocal",
|
L"Jazz+Funk",
|
||||||
L"Jazz+Funk",
|
L"Fusion",
|
||||||
L"Fusion",
|
L"Trance",
|
||||||
L"Trance",
|
L"Classical",
|
||||||
L"Classical",
|
L"Instrumental",
|
||||||
L"Instrumental",
|
L"Acid",
|
||||||
L"Acid",
|
L"House",
|
||||||
L"House",
|
L"Game",
|
||||||
L"Game",
|
L"Sound Clip",
|
||||||
L"Sound Clip",
|
L"Gospel",
|
||||||
L"Gospel",
|
L"Noise",
|
||||||
L"Noise",
|
L"Alternative Rock",
|
||||||
L"Alternative Rock",
|
L"Bass",
|
||||||
L"Bass",
|
L"Soul",
|
||||||
L"Soul",
|
L"Punk",
|
||||||
L"Punk",
|
L"Space",
|
||||||
L"Space",
|
L"Meditative",
|
||||||
L"Meditative",
|
L"Instrumental Pop",
|
||||||
L"Instrumental Pop",
|
L"Instrumental Rock",
|
||||||
L"Instrumental Rock",
|
L"Ethnic",
|
||||||
L"Ethnic",
|
L"Gothic",
|
||||||
L"Gothic",
|
L"Darkwave",
|
||||||
L"Darkwave",
|
L"Techno-Industrial",
|
||||||
L"Techno-Industrial",
|
L"Electronic",
|
||||||
L"Electronic",
|
L"Pop-Folk",
|
||||||
L"Pop-Folk",
|
L"Eurodance",
|
||||||
L"Eurodance",
|
L"Dream",
|
||||||
L"Dream",
|
L"Southern Rock",
|
||||||
L"Southern Rock",
|
L"Comedy",
|
||||||
L"Comedy",
|
L"Cult",
|
||||||
L"Cult",
|
L"Gangsta",
|
||||||
L"Gangsta",
|
L"Top 40",
|
||||||
L"Top 40",
|
L"Christian Rap",
|
||||||
L"Christian Rap",
|
L"Pop/Funk",
|
||||||
L"Pop/Funk",
|
L"Jungle",
|
||||||
L"Jungle",
|
L"Native American",
|
||||||
L"Native American",
|
L"Cabaret",
|
||||||
L"Cabaret",
|
L"New Wave",
|
||||||
L"New Wave",
|
L"Psychedelic",
|
||||||
L"Psychedelic",
|
L"Rave",
|
||||||
L"Rave",
|
L"Showtunes",
|
||||||
L"Showtunes",
|
L"Trailer",
|
||||||
L"Trailer",
|
L"Lo-Fi",
|
||||||
L"Lo-Fi",
|
L"Tribal",
|
||||||
L"Tribal",
|
L"Acid Punk",
|
||||||
L"Acid Punk",
|
L"Acid Jazz",
|
||||||
L"Acid Jazz",
|
L"Polka",
|
||||||
L"Polka",
|
L"Retro",
|
||||||
L"Retro",
|
L"Musical",
|
||||||
L"Musical",
|
L"Rock & Roll",
|
||||||
L"Rock & Roll",
|
L"Hard Rock",
|
||||||
L"Hard Rock",
|
L"Folk",
|
||||||
L"Folk",
|
L"Folk/Rock",
|
||||||
L"Folk/Rock",
|
L"National Folk",
|
||||||
L"National Folk",
|
L"Swing",
|
||||||
L"Swing",
|
L"Fusion",
|
||||||
L"Fusion",
|
L"Bebob",
|
||||||
L"Bebob",
|
L"Latin",
|
||||||
L"Latin",
|
L"Revival",
|
||||||
L"Revival",
|
L"Celtic",
|
||||||
L"Celtic",
|
L"Bluegrass",
|
||||||
L"Bluegrass",
|
L"Avantgarde",
|
||||||
L"Avantgarde",
|
L"Gothic Rock",
|
||||||
L"Gothic Rock",
|
L"Progressive Rock",
|
||||||
L"Progressive Rock",
|
L"Psychedelic Rock",
|
||||||
L"Psychedelic Rock",
|
L"Symphonic Rock",
|
||||||
L"Symphonic Rock",
|
L"Slow Rock",
|
||||||
L"Slow Rock",
|
L"Big Band",
|
||||||
L"Big Band",
|
L"Chorus",
|
||||||
L"Chorus",
|
L"Easy Listening",
|
||||||
L"Easy Listening",
|
L"Acoustic",
|
||||||
L"Acoustic",
|
L"Humour",
|
||||||
L"Humour",
|
L"Speech",
|
||||||
L"Speech",
|
L"Chanson",
|
||||||
L"Chanson",
|
L"Opera",
|
||||||
L"Opera",
|
L"Chamber Music",
|
||||||
L"Chamber Music",
|
L"Sonata",
|
||||||
L"Sonata",
|
L"Symphony",
|
||||||
L"Symphony",
|
L"Booty Bass",
|
||||||
L"Booty Bass",
|
L"Primus",
|
||||||
L"Primus",
|
L"Porn Groove",
|
||||||
L"Porn Groove",
|
L"Satire",
|
||||||
L"Satire",
|
L"Slow Jam",
|
||||||
L"Slow Jam",
|
L"Club",
|
||||||
L"Club",
|
L"Tango",
|
||||||
L"Tango",
|
L"Samba",
|
||||||
L"Samba",
|
L"Folklore",
|
||||||
L"Folklore",
|
L"Ballad",
|
||||||
L"Ballad",
|
L"Power Ballad",
|
||||||
L"Power Ballad",
|
L"Rhythmic Soul",
|
||||||
L"Rhythmic Soul",
|
L"Freestyle",
|
||||||
L"Freestyle",
|
L"Duet",
|
||||||
L"Duet",
|
L"Punk Rock",
|
||||||
L"Punk Rock",
|
L"Drum Solo",
|
||||||
L"Drum Solo",
|
L"A Cappella",
|
||||||
L"A Cappella",
|
L"Euro-House",
|
||||||
L"Euro-House",
|
L"Dance Hall",
|
||||||
L"Dance Hall",
|
L"Goa",
|
||||||
L"Goa",
|
L"Drum & Bass",
|
||||||
L"Drum & Bass",
|
L"Club-House",
|
||||||
L"Club-House",
|
L"Hardcore",
|
||||||
L"Hardcore",
|
L"Terror",
|
||||||
L"Terror",
|
L"Indie",
|
||||||
L"Indie",
|
L"BritPop",
|
||||||
L"BritPop",
|
L"Negerpunk",
|
||||||
L"Negerpunk",
|
L"Polsk Punk",
|
||||||
L"Polsk Punk",
|
L"Beat",
|
||||||
L"Beat",
|
L"Christian Gangsta Rap",
|
||||||
L"Christian Gangsta Rap",
|
L"Heavy Metal",
|
||||||
L"Heavy Metal",
|
L"Black Metal",
|
||||||
L"Black Metal",
|
L"Crossover",
|
||||||
L"Crossover",
|
L"Contemporary Christian",
|
||||||
L"Contemporary Christian",
|
L"Christian Rock",
|
||||||
L"Christian Rock",
|
L"Merengue",
|
||||||
L"Merengue",
|
L"Salsa",
|
||||||
L"Salsa",
|
L"Thrash Metal",
|
||||||
L"Thrash Metal",
|
L"Anime",
|
||||||
L"Anime",
|
L"Jpop",
|
||||||
L"Jpop",
|
L"Synthpop",
|
||||||
L"Synthpop",
|
L"Abstract",
|
||||||
L"Abstract",
|
L"Art Rock",
|
||||||
L"Art Rock",
|
L"Baroque",
|
||||||
L"Baroque",
|
L"Bhangra",
|
||||||
L"Bhangra",
|
L"Big Beat",
|
||||||
L"Big Beat",
|
L"Breakbeat",
|
||||||
L"Breakbeat",
|
L"Chillout",
|
||||||
L"Chillout",
|
L"Downtempo",
|
||||||
L"Downtempo",
|
L"Dub",
|
||||||
L"Dub",
|
L"EBM",
|
||||||
L"EBM",
|
L"Eclectic",
|
||||||
L"Eclectic",
|
L"Electro",
|
||||||
L"Electro",
|
L"Electroclash",
|
||||||
L"Electroclash",
|
L"Emo",
|
||||||
L"Emo",
|
L"Experimental",
|
||||||
L"Experimental",
|
L"Garage",
|
||||||
L"Garage",
|
L"Global",
|
||||||
L"Global",
|
L"IDM",
|
||||||
L"IDM",
|
L"Illbient",
|
||||||
L"Illbient",
|
L"Industro-Goth",
|
||||||
L"Industro-Goth",
|
L"Jam Band",
|
||||||
L"Jam Band",
|
L"Krautrock",
|
||||||
L"Krautrock",
|
L"Leftfield",
|
||||||
L"Leftfield",
|
L"Lounge",
|
||||||
L"Lounge",
|
L"Math Rock",
|
||||||
L"Math Rock",
|
L"New Romantic",
|
||||||
L"New Romantic",
|
L"Nu-Breakz",
|
||||||
L"Nu-Breakz",
|
L"Post-Punk",
|
||||||
L"Post-Punk",
|
L"Post-Rock",
|
||||||
L"Post-Rock",
|
L"Psytrance",
|
||||||
L"Psytrance",
|
L"Shoegaze",
|
||||||
L"Shoegaze",
|
L"Space Rock",
|
||||||
L"Space Rock",
|
L"Trop Rock",
|
||||||
L"Trop Rock",
|
L"World Music",
|
||||||
L"World Music",
|
L"Neoclassical",
|
||||||
L"Neoclassical",
|
L"Audiobook",
|
||||||
L"Audiobook",
|
L"Audio Theatre",
|
||||||
L"Audio Theatre",
|
L"Neue Deutsche Welle",
|
||||||
L"Neue Deutsche Welle",
|
L"Podcast",
|
||||||
L"Podcast",
|
L"Indie Rock",
|
||||||
L"Indie Rock",
|
L"G-Funk",
|
||||||
L"G-Funk",
|
L"Dubstep",
|
||||||
L"Dubstep",
|
L"Garage Rock",
|
||||||
L"Garage Rock",
|
L"Psybient"
|
||||||
L"Psybient"
|
};
|
||||||
};
|
const int genresSize = sizeof(genres) / sizeof(genres[0]);
|
||||||
const int genresSize = sizeof(genres) / sizeof(genres[0]);
|
} // namespace
|
||||||
}
|
|
||||||
|
StringList ID3v1::genreList() {
|
||||||
|
|
||||||
StringList ID3v1::genreList()
|
|
||||||
{
|
|
||||||
StringList l;
|
StringList l;
|
||||||
for(int i = 0; i < genresSize; i++) {
|
for (int i = 0; i < genresSize; i++) {
|
||||||
l.append(genres[i]);
|
l.append(genres[i]);
|
||||||
}
|
}
|
||||||
|
|
||||||
return l;
|
return l;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ID3v1::GenreMap ID3v1::genreMap()
|
ID3v1::GenreMap ID3v1::genreMap() {
|
||||||
{
|
|
||||||
GenreMap m;
|
GenreMap m;
|
||||||
for(int i = 0; i < genresSize; i++) {
|
for (int i = 0; i < genresSize; i++) {
|
||||||
m.insert(genres[i], i);
|
m.insert(genres[i], i);
|
||||||
}
|
}
|
||||||
|
|
||||||
return m;
|
return m;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
String ID3v1::genre(int i)
|
String ID3v1::genre(int i) {
|
||||||
{
|
|
||||||
if(i >= 0 && i < genresSize)
|
if (i >= 0 && i < genresSize)
|
||||||
return String(genres[i]); // always make a copy
|
return String(genres[i]); // always make a copy
|
||||||
else
|
else
|
||||||
return String();
|
return String();
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int ID3v1::genreIndex(const String &name)
|
int ID3v1::genreIndex(const String &name) {
|
||||||
{
|
|
||||||
for(int i = 0; i < genresSize; ++i) {
|
for (int i = 0; i < genresSize; ++i) {
|
||||||
if(name == genres[i])
|
if (name == genres[i])
|
||||||
return i;
|
return i;
|
||||||
}
|
}
|
||||||
|
|
||||||
return 255;
|
return 255;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
50
3rdparty/taglib/mpeg/id3v1/id3v1genres.h
vendored
50
3rdparty/taglib/mpeg/id3v1/id3v1genres.h
vendored
@@ -32,37 +32,33 @@
|
|||||||
|
|
||||||
namespace Strawberry_TagLib {
|
namespace Strawberry_TagLib {
|
||||||
namespace TagLib {
|
namespace TagLib {
|
||||||
namespace ID3v1 {
|
namespace ID3v1 {
|
||||||
|
|
||||||
typedef Map<String, int> GenreMap;
|
typedef Map<String, int> GenreMap;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the list of canonical ID3v1 genre names in the order that they
|
* Returns the list of canonical ID3v1 genre names in the order that they are listed in the standard.
|
||||||
* are listed in the standard.
|
*/
|
||||||
*/
|
StringList TAGLIB_EXPORT genreList();
|
||||||
StringList TAGLIB_EXPORT genreList();
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* A "reverse mapping" that goes from the canonical ID3v1 genre name to the
|
* A "reverse mapping" that goes from the canonical ID3v1 genre name to the respective genre number. genreMap()["Rock"] ==
|
||||||
* respective genre number. genreMap()["Rock"] ==
|
*/
|
||||||
*/
|
GenreMap TAGLIB_EXPORT genreMap();
|
||||||
GenreMap TAGLIB_EXPORT genreMap();
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the name of the genre at \a index in the ID3v1 genre list. If
|
* Returns the name of the genre at \a index in the ID3v1 genre list.
|
||||||
* \a index is out of range -- less than zero or greater than 191 -- a null
|
* If \a index is out of range -- less than zero or greater than 191 -- a null string will be returned.
|
||||||
* string will be returned.
|
*/
|
||||||
*/
|
String TAGLIB_EXPORT genre(int index);
|
||||||
String TAGLIB_EXPORT genre(int index);
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the genre index for the (case sensitive) genre \a name. If the
|
* Returns the genre index for the (case sensitive) genre \a name.
|
||||||
* genre is not in the list 255 (which signifies an unknown genre in ID3v1)
|
* If the genre is not in the list 255 (which signifies an unknown genre in ID3v1) will be returned.
|
||||||
* will be returned.
|
*/
|
||||||
*/
|
int TAGLIB_EXPORT genreIndex(const String &name);
|
||||||
int TAGLIB_EXPORT genreIndex(const String &name);
|
} // namespace ID3v1
|
||||||
}
|
} // namespace TagLib
|
||||||
}
|
} // namespace Strawberry_TagLib
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
167
3rdparty/taglib/mpeg/id3v1/id3v1tag.cpp
vendored
167
3rdparty/taglib/mpeg/id3v1/id3v1tag.cpp
vendored
@@ -23,8 +23,9 @@
|
|||||||
* http://www.mozilla.org/MPL/ *
|
* http://www.mozilla.org/MPL/ *
|
||||||
***************************************************************************/
|
***************************************************************************/
|
||||||
|
|
||||||
#include <tdebug.h>
|
#include "tdebug.h"
|
||||||
#include <tfile.h>
|
#include "tfile.h"
|
||||||
|
#include "tpicturemap.h"
|
||||||
|
|
||||||
#include "id3v1tag.h"
|
#include "id3v1tag.h"
|
||||||
#include "id3v1genres.h"
|
#include "id3v1genres.h"
|
||||||
@@ -32,23 +33,36 @@
|
|||||||
using namespace Strawberry_TagLib::TagLib;
|
using namespace Strawberry_TagLib::TagLib;
|
||||||
using namespace ID3v1;
|
using namespace ID3v1;
|
||||||
|
|
||||||
namespace
|
namespace {
|
||||||
{
|
class DefaultStringHandler : public Strawberry_TagLib::TagLib::StringHandler {
|
||||||
const ID3v1::StringHandler defaultStringHandler;
|
public:
|
||||||
const ID3v1::StringHandler *stringHandler = &defaultStringHandler;
|
explicit DefaultStringHandler() : Strawberry_TagLib::TagLib::StringHandler() {}
|
||||||
}
|
|
||||||
|
|
||||||
class ID3v1::Tag::TagPrivate
|
String parse(const ByteVector &data) const override {
|
||||||
{
|
return String(data, String::Latin1).stripWhiteSpace();
|
||||||
public:
|
}
|
||||||
TagPrivate() :
|
|
||||||
file(0),
|
ByteVector render(const String &s) const override {
|
||||||
tagOffset(0),
|
if (s.isLatin1())
|
||||||
track(0),
|
return s.data(String::Latin1);
|
||||||
genre(255) {}
|
else
|
||||||
|
return ByteVector();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const DefaultStringHandler defaultStringHandler;
|
||||||
|
const Strawberry_TagLib::TagLib::StringHandler *stringHandler = &defaultStringHandler;
|
||||||
|
} // namespace
|
||||||
|
|
||||||
|
class ID3v1::Tag::TagPrivate {
|
||||||
|
public:
|
||||||
|
explicit TagPrivate() : file(nullptr),
|
||||||
|
tagOffset(0),
|
||||||
|
track(0),
|
||||||
|
genre(255) {}
|
||||||
|
|
||||||
File *file;
|
File *file;
|
||||||
long tagOffset;
|
long long tagOffset;
|
||||||
|
|
||||||
String title;
|
String title;
|
||||||
String artist;
|
String artist;
|
||||||
@@ -59,54 +73,26 @@ public:
|
|||||||
unsigned char genre;
|
unsigned char genre;
|
||||||
};
|
};
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
|
||||||
// StringHandler implementation
|
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
StringHandler::StringHandler()
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
String ID3v1::StringHandler::parse(const ByteVector &data) const
|
|
||||||
{
|
|
||||||
return String(data, String::Latin1).stripWhiteSpace();
|
|
||||||
}
|
|
||||||
|
|
||||||
ByteVector ID3v1::StringHandler::render(const String &s) const
|
|
||||||
{
|
|
||||||
if(s.isLatin1())
|
|
||||||
return s.data(String::Latin1);
|
|
||||||
else
|
|
||||||
return ByteVector();
|
|
||||||
}
|
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
// public methods
|
// public methods
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
ID3v1::Tag::Tag() :
|
ID3v1::Tag::Tag() : d(new TagPrivate()) {}
|
||||||
Strawberry_TagLib::TagLib::Tag(),
|
|
||||||
d(new TagPrivate())
|
ID3v1::Tag::Tag(File *file, long long tagOffset) : d(new TagPrivate()) {
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
ID3v1::Tag::Tag(File *file, long tagOffset) :
|
|
||||||
Strawberry_TagLib::TagLib::Tag(),
|
|
||||||
d(new TagPrivate())
|
|
||||||
{
|
|
||||||
d->file = file;
|
d->file = file;
|
||||||
d->tagOffset = tagOffset;
|
d->tagOffset = tagOffset;
|
||||||
|
|
||||||
read();
|
read();
|
||||||
}
|
}
|
||||||
|
|
||||||
ID3v1::Tag::~Tag()
|
ID3v1::Tag::~Tag() {
|
||||||
{
|
|
||||||
delete d;
|
delete d;
|
||||||
}
|
}
|
||||||
|
|
||||||
ByteVector ID3v1::Tag::render() const
|
ByteVector ID3v1::Tag::render() const {
|
||||||
{
|
|
||||||
ByteVector data;
|
ByteVector data;
|
||||||
|
|
||||||
data.append(fileIdentifier());
|
data.append(fileIdentifier());
|
||||||
@@ -120,96 +106,85 @@ ByteVector ID3v1::Tag::render() const
|
|||||||
data.append(char(d->genre));
|
data.append(char(d->genre));
|
||||||
|
|
||||||
return data;
|
return data;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ByteVector ID3v1::Tag::fileIdentifier()
|
ByteVector ID3v1::Tag::fileIdentifier() {
|
||||||
{
|
|
||||||
return ByteVector::fromCString("TAG");
|
return ByteVector::fromCString("TAG");
|
||||||
}
|
}
|
||||||
|
|
||||||
String ID3v1::Tag::title() const
|
String ID3v1::Tag::title() const {
|
||||||
{
|
|
||||||
return d->title;
|
return d->title;
|
||||||
}
|
}
|
||||||
|
|
||||||
String ID3v1::Tag::artist() const
|
String ID3v1::Tag::artist() const {
|
||||||
{
|
|
||||||
return d->artist;
|
return d->artist;
|
||||||
}
|
}
|
||||||
|
|
||||||
String ID3v1::Tag::album() const
|
String ID3v1::Tag::album() const {
|
||||||
{
|
|
||||||
return d->album;
|
return d->album;
|
||||||
}
|
}
|
||||||
|
|
||||||
String ID3v1::Tag::comment() const
|
String ID3v1::Tag::comment() const {
|
||||||
{
|
|
||||||
return d->comment;
|
return d->comment;
|
||||||
}
|
}
|
||||||
|
|
||||||
String ID3v1::Tag::genre() const
|
String ID3v1::Tag::genre() const {
|
||||||
{
|
|
||||||
return ID3v1::genre(d->genre);
|
return ID3v1::genre(d->genre);
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned int ID3v1::Tag::year() const
|
unsigned int ID3v1::Tag::year() const {
|
||||||
{
|
|
||||||
return d->year.toInt();
|
return d->year.toInt();
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned int ID3v1::Tag::track() const
|
unsigned int ID3v1::Tag::track() const {
|
||||||
{
|
|
||||||
return d->track;
|
return d->track;
|
||||||
}
|
}
|
||||||
|
|
||||||
void ID3v1::Tag::setTitle(const String &s)
|
Strawberry_TagLib::TagLib::PictureMap ID3v1::Tag::pictures() const {
|
||||||
{
|
return PictureMap();
|
||||||
|
}
|
||||||
|
|
||||||
|
void ID3v1::Tag::setTitle(const String &s) {
|
||||||
d->title = s;
|
d->title = s;
|
||||||
}
|
}
|
||||||
|
|
||||||
void ID3v1::Tag::setArtist(const String &s)
|
void ID3v1::Tag::setArtist(const String &s) {
|
||||||
{
|
|
||||||
d->artist = s;
|
d->artist = s;
|
||||||
}
|
}
|
||||||
|
|
||||||
void ID3v1::Tag::setAlbum(const String &s)
|
void ID3v1::Tag::setAlbum(const String &s) {
|
||||||
{
|
|
||||||
d->album = s;
|
d->album = s;
|
||||||
}
|
}
|
||||||
|
|
||||||
void ID3v1::Tag::setComment(const String &s)
|
void ID3v1::Tag::setComment(const String &s) {
|
||||||
{
|
|
||||||
d->comment = s;
|
d->comment = s;
|
||||||
}
|
}
|
||||||
|
|
||||||
void ID3v1::Tag::setGenre(const String &s)
|
void ID3v1::Tag::setGenre(const String &s) {
|
||||||
{
|
|
||||||
d->genre = ID3v1::genreIndex(s);
|
d->genre = ID3v1::genreIndex(s);
|
||||||
}
|
}
|
||||||
|
|
||||||
void ID3v1::Tag::setYear(unsigned int i)
|
void ID3v1::Tag::setYear(unsigned int i) {
|
||||||
{
|
|
||||||
d->year = i > 0 ? String::number(i) : String();
|
d->year = i > 0 ? String::number(i) : String();
|
||||||
}
|
}
|
||||||
|
|
||||||
void ID3v1::Tag::setTrack(unsigned int i)
|
void ID3v1::Tag::setTrack(unsigned int i) {
|
||||||
{
|
|
||||||
d->track = i < 256 ? i : 0;
|
d->track = i < 256 ? i : 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned int ID3v1::Tag::genreNumber() const
|
void ID3v1::Tag::setPictures(const PictureMap&) {}
|
||||||
{
|
|
||||||
|
unsigned int ID3v1::Tag::genreNumber() const {
|
||||||
return d->genre;
|
return d->genre;
|
||||||
}
|
}
|
||||||
|
|
||||||
void ID3v1::Tag::setGenreNumber(unsigned int i)
|
void ID3v1::Tag::setGenreNumber(unsigned int i) {
|
||||||
{
|
|
||||||
d->genre = i < 256 ? i : 255;
|
d->genre = i < 256 ? i : 255;
|
||||||
}
|
}
|
||||||
|
|
||||||
void ID3v1::Tag::setStringHandler(const StringHandler *handler)
|
void ID3v1::Tag::setStringHandler(const Strawberry_TagLib::TagLib::StringHandler *handler) {
|
||||||
{
|
if (handler)
|
||||||
if(handler)
|
|
||||||
stringHandler = handler;
|
stringHandler = handler;
|
||||||
else
|
else
|
||||||
stringHandler = &defaultStringHandler;
|
stringHandler = &defaultStringHandler;
|
||||||
@@ -219,23 +194,24 @@ void ID3v1::Tag::setStringHandler(const StringHandler *handler)
|
|||||||
// protected methods
|
// protected methods
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
void ID3v1::Tag::read()
|
void ID3v1::Tag::read() {
|
||||||
{
|
|
||||||
if(d->file && d->file->isValid()) {
|
if (d->file && d->file->isValid()) {
|
||||||
d->file->seek(d->tagOffset);
|
d->file->seek(d->tagOffset);
|
||||||
// read the tag -- always 128 bytes
|
// read the tag -- always 128 bytes
|
||||||
const ByteVector data = d->file->readBlock(128);
|
const ByteVector data = d->file->readBlock(128);
|
||||||
|
|
||||||
// some initial sanity checking
|
// some initial sanity checking
|
||||||
if(data.size() == 128 && data.startsWith("TAG"))
|
if (data.size() == 128 && data.startsWith("TAG"))
|
||||||
parse(data);
|
parse(data);
|
||||||
else
|
else
|
||||||
debug("ID3v1 tag is not valid or could not be read at the specified offset.");
|
debug("ID3v1 tag is not valid or could not be read at the specified offset.");
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void ID3v1::Tag::parse(const ByteVector &data)
|
void ID3v1::Tag::parse(const ByteVector &data) {
|
||||||
{
|
|
||||||
int offset = 3;
|
int offset = 3;
|
||||||
|
|
||||||
d->title = stringHandler->parse(data.mid(offset, 30));
|
d->title = stringHandler->parse(data.mid(offset, 30));
|
||||||
@@ -255,11 +231,11 @@ void ID3v1::Tag::parse(const ByteVector &data)
|
|||||||
// indicate the end of a C-String, specifically the comment string, a value of
|
// indicate the end of a C-String, specifically the comment string, a value of
|
||||||
// zero must be assumed to be just that.
|
// zero must be assumed to be just that.
|
||||||
|
|
||||||
if(data[offset + 28] == 0 && data[offset + 29] != 0) {
|
if (data[offset + 28] == 0 && data[offset + 29] != 0) {
|
||||||
// ID3v1.1 detected
|
// ID3v1.1 detected
|
||||||
|
|
||||||
d->comment = stringHandler->parse(data.mid(offset, 28));
|
d->comment = stringHandler->parse(data.mid(offset, 28));
|
||||||
d->track = static_cast<unsigned char>(data[offset + 29]);
|
d->track = static_cast<unsigned char>(data[offset + 29]);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
d->comment = data.mid(offset, 30);
|
d->comment = data.mid(offset, 30);
|
||||||
@@ -267,4 +243,5 @@ void ID3v1::Tag::parse(const ByteVector &data)
|
|||||||
offset += 30;
|
offset += 30;
|
||||||
|
|
||||||
d->genre = static_cast<unsigned char>(data[offset]);
|
d->genre = static_cast<unsigned char>(data[offset]);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
246
3rdparty/taglib/mpeg/id3v1/id3v1tag.h
vendored
246
3rdparty/taglib/mpeg/id3v1/id3v1tag.h
vendored
@@ -28,177 +28,127 @@
|
|||||||
|
|
||||||
#include "tag.h"
|
#include "tag.h"
|
||||||
#include "tbytevector.h"
|
#include "tbytevector.h"
|
||||||
|
#include "tstringhandler.h"
|
||||||
#include "taglib_export.h"
|
#include "taglib_export.h"
|
||||||
|
|
||||||
namespace Strawberry_TagLib {
|
namespace Strawberry_TagLib {
|
||||||
namespace TagLib {
|
namespace TagLib {
|
||||||
|
|
||||||
class File;
|
class File;
|
||||||
|
|
||||||
//! An ID3v1 implementation
|
//! An ID3v1 implementation
|
||||||
|
|
||||||
namespace ID3v1 {
|
namespace ID3v1 {
|
||||||
|
|
||||||
//! A abstraction for the string to data encoding in ID3v1 tags.
|
//! The main class in the ID3v1 implementation
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* ID3v1 should in theory always contain ISO-8859-1 (Latin1) data. In
|
* This is an implementation of the ID3v1 format.
|
||||||
* practice it does not. TagLib by default only supports ISO-8859-1 data
|
* ID3v1 is both the simplest and most common of tag formats but is rather limited.
|
||||||
* in ID3v1 tags.
|
* Because of its pervasiveness and the way that applications have been written around the
|
||||||
*
|
* fields that it provides, the generic TagLib::Tag API is a mirror of what is provided by ID3v1.
|
||||||
* However by subclassing this class and reimplementing parse() and render()
|
*
|
||||||
* and setting your reimplementation as the default with
|
* ID3v1 tags should generally only contain Latin1 information.
|
||||||
* ID3v1::Tag::setStringHandler() you can define how you would like these
|
* However because many applications do not follow this rule there is now support for overriding
|
||||||
* transformations to be done.
|
* the ID3v1 string handling using the ID3v1::StringHandler class.
|
||||||
*
|
* Please see the documentation for that class for more information.
|
||||||
* \warning It is advisable <b>not</b> to write non-ISO-8859-1 data to ID3v1
|
*
|
||||||
* tags. Please consider disabling the writing of ID3v1 tags in the case
|
* \see StringHandler
|
||||||
* that the data is not ISO-8859-1.
|
*
|
||||||
*
|
* \note Most fields are truncated to a maximum of 28-30 bytes.
|
||||||
* \see ID3v1::Tag::setStringHandler()
|
* The truncation happens automatically when the tag is rendered.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
class TAGLIB_EXPORT StringHandler
|
class TAGLIB_EXPORT Tag : public Strawberry_TagLib::TagLib::Tag {
|
||||||
{
|
public:
|
||||||
TAGLIB_IGNORE_MISSING_DESTRUCTOR
|
/*!
|
||||||
public:
|
* Create an ID3v1 tag with default values.
|
||||||
// BIC: Add virtual destructor.
|
*/
|
||||||
StringHandler();
|
explicit Tag();
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Decode a string from \a data. The default implementation assumes that
|
* Create an ID3v1 tag and parse the data in \a file starting at \a tagOffset.
|
||||||
* \a data is an ISO-8859-1 (Latin1) character array.
|
*/
|
||||||
*/
|
explicit Tag(File *file, long long tagOffset);
|
||||||
virtual String parse(const ByteVector &data) const;
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Encode a ByteVector with the data from \a s. The default implementation
|
* Destroys this Tag instance.
|
||||||
* assumes that \a s is an ISO-8859-1 (Latin1) string. If the string is
|
*/
|
||||||
* does not conform to ISO-8859-1, no value is written.
|
~Tag() override;
|
||||||
*
|
|
||||||
* \warning It is recommended that you <b>not</b> override this method, but
|
|
||||||
* instead do not write an ID3v1 tag in the case that the data is not
|
|
||||||
* ISO-8859-1.
|
|
||||||
*/
|
|
||||||
virtual ByteVector render(const String &s) const;
|
|
||||||
};
|
|
||||||
|
|
||||||
//! The main class in the ID3v1 implementation
|
/*!
|
||||||
|
* Renders the in memory values to a ByteVector suitable for writing to the file.
|
||||||
|
*/
|
||||||
|
ByteVector render() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* This is an implementation of the ID3v1 format. ID3v1 is both the simplest
|
* Returns the string "TAG" suitable for usage in locating the tag in a file.
|
||||||
* and most common of tag formats but is rather limited. Because of its
|
*/
|
||||||
* pervasiveness and the way that applications have been written around the
|
static ByteVector fileIdentifier();
|
||||||
* fields that it provides, the generic TagLib::Tag API is a mirror of what is
|
|
||||||
* provided by ID3v1.
|
|
||||||
*
|
|
||||||
* ID3v1 tags should generally only contain Latin1 information. However because
|
|
||||||
* many applications do not follow this rule there is now support for overriding
|
|
||||||
* the ID3v1 string handling using the ID3v1::StringHandler class. Please see
|
|
||||||
* the documentation for that class for more information.
|
|
||||||
*
|
|
||||||
* \see StringHandler
|
|
||||||
*
|
|
||||||
* \note Most fields are truncated to a maximum of 28-30 bytes. The
|
|
||||||
* truncation happens automatically when the tag is rendered.
|
|
||||||
*/
|
|
||||||
|
|
||||||
class TAGLIB_EXPORT Tag : public Strawberry_TagLib::TagLib::Tag
|
// Reimplementations.
|
||||||
{
|
|
||||||
public:
|
|
||||||
/*!
|
|
||||||
* Create an ID3v1 tag with default values.
|
|
||||||
*/
|
|
||||||
Tag();
|
|
||||||
|
|
||||||
/*!
|
String title() const override;
|
||||||
* Create an ID3v1 tag and parse the data in \a file starting at
|
String artist() const override;
|
||||||
* \a tagOffset.
|
String album() const override;
|
||||||
*/
|
String comment() const override;
|
||||||
Tag(File *file, long tagOffset);
|
String genre() const override;
|
||||||
|
unsigned int year() const override;
|
||||||
|
unsigned int track() const override;
|
||||||
|
PictureMap pictures() const override;
|
||||||
|
|
||||||
/*!
|
void setTitle(const String &s) override;
|
||||||
* Destroys this Tag instance.
|
void setArtist(const String &s) override;
|
||||||
*/
|
void setAlbum(const String &s) override;
|
||||||
virtual ~Tag();
|
void setComment(const String &s) override;
|
||||||
|
void setGenre(const String &s) override;
|
||||||
|
void setYear(unsigned int i) override;
|
||||||
|
void setTrack(unsigned int i) override;
|
||||||
|
void setPictures(const PictureMap&) override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Renders the in memory values to a ByteVector suitable for writing to
|
* Returns the genre in number.
|
||||||
* the file.
|
*
|
||||||
*/
|
* \note Normally 255 indicates that this tag contains no genre.
|
||||||
ByteVector render() const;
|
*/
|
||||||
|
unsigned int genreNumber() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the string "TAG" suitable for usage in locating the tag in a
|
* Sets the genre in number to \a i.
|
||||||
* file.
|
*
|
||||||
*/
|
* \note Valid value is from 0 up to 255. Normally 255 indicates that this tag contains no genre.
|
||||||
static ByteVector fileIdentifier();
|
*/
|
||||||
|
void setGenreNumber(unsigned int i);
|
||||||
|
|
||||||
// Reimplementations.
|
/*!
|
||||||
|
* Sets the string handler that decides how the ID3v1 data will be converted to and from binary data.
|
||||||
|
* If the parameter \a handler is null, the previous handler is released and default ISO-8859-1 handler is restored.
|
||||||
|
*
|
||||||
|
* \note The caller is responsible for deleting the previous handler as needed after it is released.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
static void setStringHandler(const Strawberry_TagLib::TagLib::StringHandler *handler);
|
||||||
|
|
||||||
virtual String title() const;
|
protected:
|
||||||
virtual String artist() const;
|
/*!
|
||||||
virtual String album() const;
|
* Reads from the file specified in the constructor.
|
||||||
virtual String comment() const;
|
*/
|
||||||
virtual String genre() const;
|
void read();
|
||||||
virtual unsigned int year() const;
|
/*!
|
||||||
virtual unsigned int track() const;
|
* Pareses the body of the tag in \a data.
|
||||||
|
*/
|
||||||
|
void parse(const ByteVector &data);
|
||||||
|
|
||||||
virtual void setTitle(const String &s);
|
private:
|
||||||
virtual void setArtist(const String &s);
|
Tag(const Tag&);
|
||||||
virtual void setAlbum(const String &s);
|
Tag &operator=(const Tag&);
|
||||||
virtual void setComment(const String &s);
|
|
||||||
virtual void setGenre(const String &s);
|
|
||||||
virtual void setYear(unsigned int i);
|
|
||||||
virtual void setTrack(unsigned int i);
|
|
||||||
|
|
||||||
/*!
|
class TagPrivate;
|
||||||
* Returns the genre in number.
|
TagPrivate *d;
|
||||||
*
|
};
|
||||||
* \note Normally 255 indicates that this tag contains no genre.
|
|
||||||
*/
|
|
||||||
unsigned int genreNumber() const;
|
|
||||||
|
|
||||||
/*!
|
} // namespace ID3v1
|
||||||
* Sets the genre in number to \a i.
|
} // namespace TagLib
|
||||||
*
|
} // namespace Strawberry_TagLib
|
||||||
* \note Valid value is from 0 up to 255. Normally 255 indicates that
|
|
||||||
* this tag contains no genre.
|
|
||||||
*/
|
|
||||||
void setGenreNumber(unsigned int i);
|
|
||||||
|
|
||||||
/*!
|
|
||||||
* Sets the string handler that decides how the ID3v1 data will be
|
|
||||||
* converted to and from binary data.
|
|
||||||
* If the parameter \a handler is null, the previous handler is
|
|
||||||
* released and default ISO-8859-1 handler is restored.
|
|
||||||
*
|
|
||||||
* \note The caller is responsible for deleting the previous handler
|
|
||||||
* as needed after it is released.
|
|
||||||
*
|
|
||||||
* \see StringHandler
|
|
||||||
*/
|
|
||||||
static void setStringHandler(const StringHandler *handler);
|
|
||||||
|
|
||||||
protected:
|
|
||||||
/*!
|
|
||||||
* Reads from the file specified in the constructor.
|
|
||||||
*/
|
|
||||||
void read();
|
|
||||||
/*!
|
|
||||||
* Pareses the body of the tag in \a data.
|
|
||||||
*/
|
|
||||||
void parse(const ByteVector &data);
|
|
||||||
|
|
||||||
private:
|
|
||||||
Tag(const Tag &);
|
|
||||||
Tag &operator=(const Tag &);
|
|
||||||
|
|
||||||
class TagPrivate;
|
|
||||||
TagPrivate *d;
|
|
||||||
};
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
@@ -25,17 +25,15 @@
|
|||||||
|
|
||||||
#include "attachedpictureframe.h"
|
#include "attachedpictureframe.h"
|
||||||
|
|
||||||
#include <tstringlist.h>
|
#include "tstringlist.h"
|
||||||
#include <tdebug.h>
|
#include "tdebug.h"
|
||||||
|
|
||||||
using namespace Strawberry_TagLib::TagLib;
|
using namespace Strawberry_TagLib::TagLib;
|
||||||
using namespace ID3v2;
|
using namespace ID3v2;
|
||||||
|
|
||||||
class AttachedPictureFrame::AttachedPictureFramePrivate
|
class AttachedPictureFrame::AttachedPictureFramePrivate {
|
||||||
{
|
public:
|
||||||
public:
|
explicit AttachedPictureFramePrivate() : textEncoding(String::Latin1), type(AttachedPictureFrame::Other) {}
|
||||||
AttachedPictureFramePrivate() : textEncoding(String::Latin1),
|
|
||||||
type(AttachedPictureFrame::Other) {}
|
|
||||||
|
|
||||||
String::Type textEncoding;
|
String::Type textEncoding;
|
||||||
String mimeType;
|
String mimeType;
|
||||||
@@ -48,77 +46,58 @@ public:
|
|||||||
// public members
|
// public members
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
AttachedPictureFrame::AttachedPictureFrame() :
|
AttachedPictureFrame::AttachedPictureFrame() : Frame("APIC"), d(new AttachedPictureFramePrivate()) {}
|
||||||
Frame("APIC"),
|
|
||||||
d(new AttachedPictureFramePrivate())
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
AttachedPictureFrame::AttachedPictureFrame(const ByteVector &data) :
|
AttachedPictureFrame::AttachedPictureFrame(const ByteVector &data) : Frame(data), d(new AttachedPictureFramePrivate()) {
|
||||||
Frame(data),
|
|
||||||
d(new AttachedPictureFramePrivate())
|
|
||||||
{
|
|
||||||
setData(data);
|
setData(data);
|
||||||
}
|
}
|
||||||
|
|
||||||
AttachedPictureFrame::~AttachedPictureFrame()
|
AttachedPictureFrame::~AttachedPictureFrame() {
|
||||||
{
|
|
||||||
delete d;
|
delete d;
|
||||||
}
|
}
|
||||||
|
|
||||||
String AttachedPictureFrame::toString() const
|
String AttachedPictureFrame::toString() const {
|
||||||
{
|
|
||||||
String s = "[" + d->mimeType + "]";
|
String s = "[" + d->mimeType + "]";
|
||||||
return d->description.isEmpty() ? s : d->description + " " + s;
|
return d->description.isEmpty() ? s : d->description + " " + s;
|
||||||
}
|
}
|
||||||
|
|
||||||
String::Type AttachedPictureFrame::textEncoding() const
|
String::Type AttachedPictureFrame::textEncoding() const {
|
||||||
{
|
|
||||||
return d->textEncoding;
|
return d->textEncoding;
|
||||||
}
|
}
|
||||||
|
|
||||||
void AttachedPictureFrame::setTextEncoding(String::Type t)
|
void AttachedPictureFrame::setTextEncoding(String::Type t) {
|
||||||
{
|
|
||||||
d->textEncoding = t;
|
d->textEncoding = t;
|
||||||
}
|
}
|
||||||
|
|
||||||
String AttachedPictureFrame::mimeType() const
|
String AttachedPictureFrame::mimeType() const {
|
||||||
{
|
|
||||||
return d->mimeType;
|
return d->mimeType;
|
||||||
}
|
}
|
||||||
|
|
||||||
void AttachedPictureFrame::setMimeType(const String &m)
|
void AttachedPictureFrame::setMimeType(const String &m) {
|
||||||
{
|
|
||||||
d->mimeType = m;
|
d->mimeType = m;
|
||||||
}
|
}
|
||||||
|
|
||||||
AttachedPictureFrame::Type AttachedPictureFrame::type() const
|
AttachedPictureFrame::Type AttachedPictureFrame::type() const {
|
||||||
{
|
|
||||||
return d->type;
|
return d->type;
|
||||||
}
|
}
|
||||||
|
|
||||||
void AttachedPictureFrame::setType(Type t)
|
void AttachedPictureFrame::setType(Type t) {
|
||||||
{
|
|
||||||
d->type = t;
|
d->type = t;
|
||||||
}
|
}
|
||||||
|
|
||||||
String AttachedPictureFrame::description() const
|
String AttachedPictureFrame::description() const {
|
||||||
{
|
|
||||||
return d->description;
|
return d->description;
|
||||||
}
|
}
|
||||||
|
|
||||||
void AttachedPictureFrame::setDescription(const String &desc)
|
void AttachedPictureFrame::setDescription(const String &desc) {
|
||||||
{
|
|
||||||
d->description = desc;
|
d->description = desc;
|
||||||
}
|
}
|
||||||
|
|
||||||
ByteVector AttachedPictureFrame::picture() const
|
ByteVector AttachedPictureFrame::picture() const {
|
||||||
{
|
|
||||||
return d->data;
|
return d->data;
|
||||||
}
|
}
|
||||||
|
|
||||||
void AttachedPictureFrame::setPicture(const ByteVector &p)
|
void AttachedPictureFrame::setPicture(const ByteVector &p) {
|
||||||
{
|
|
||||||
d->data = p;
|
d->data = p;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -126,32 +105,33 @@ void AttachedPictureFrame::setPicture(const ByteVector &p)
|
|||||||
// protected members
|
// protected members
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
void AttachedPictureFrame::parseFields(const ByteVector &data)
|
void AttachedPictureFrame::parseFields(const ByteVector &data) {
|
||||||
{
|
|
||||||
if(data.size() < 5) {
|
if (data.size() < 5) {
|
||||||
debug("A picture frame must contain at least 5 bytes.");
|
debug("A picture frame must contain at least 5 bytes.");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
d->textEncoding = String::Type(data[0]);
|
d->textEncoding = String::Type(data[0]);
|
||||||
|
|
||||||
int pos = 1;
|
size_t pos = 1;
|
||||||
|
|
||||||
d->mimeType = readStringField(data, String::Latin1, &pos);
|
d->mimeType = readStringField(data, String::Latin1, pos);
|
||||||
/* Now we need at least two more bytes available */
|
/* Now we need at least two more bytes available */
|
||||||
if(static_cast<unsigned int>(pos) + 1 >= data.size()) {
|
if (pos + 1 >= data.size()) {
|
||||||
debug("Truncated picture frame.");
|
debug("Truncated picture frame.");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
d->type = (Strawberry_TagLib::TagLib::ID3v2::AttachedPictureFrame::Type)data[pos++];
|
d->type = static_cast<ID3v2::AttachedPictureFrame::Type>(data[pos++]);
|
||||||
d->description = readStringField(data, d->textEncoding, &pos);
|
d->description = readStringField(data, d->textEncoding, pos);
|
||||||
|
|
||||||
d->data = data.mid(pos);
|
d->data = data.mid(pos);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ByteVector AttachedPictureFrame::renderFields() const
|
ByteVector AttachedPictureFrame::renderFields() const {
|
||||||
{
|
|
||||||
ByteVector data;
|
ByteVector data;
|
||||||
|
|
||||||
String::Type encoding = checkTextEncoding(d->description, d->textEncoding);
|
String::Type encoding = checkTextEncoding(d->description, d->textEncoding);
|
||||||
@@ -165,16 +145,14 @@ ByteVector AttachedPictureFrame::renderFields() const
|
|||||||
data.append(d->data);
|
data.append(d->data);
|
||||||
|
|
||||||
return data;
|
return data;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
// private members
|
// private members
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
AttachedPictureFrame::AttachedPictureFrame(const ByteVector &data, Header *h) :
|
AttachedPictureFrame::AttachedPictureFrame(const ByteVector &data, Header *h) : Frame(h), d(new AttachedPictureFramePrivate()) {
|
||||||
Frame(h),
|
|
||||||
d(new AttachedPictureFramePrivate())
|
|
||||||
{
|
|
||||||
parseFields(fieldData(data));
|
parseFields(fieldData(data));
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -182,37 +160,40 @@ AttachedPictureFrame::AttachedPictureFrame(const ByteVector &data, Header *h) :
|
|||||||
// support for ID3v2.2 PIC frames
|
// support for ID3v2.2 PIC frames
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
void AttachedPictureFrameV22::parseFields(const ByteVector &data)
|
void AttachedPictureFrameV22::parseFields(const ByteVector &data) {
|
||||||
{
|
|
||||||
if(data.size() < 5) {
|
if (data.size() < 5) {
|
||||||
debug("A picture frame must contain at least 5 bytes.");
|
debug("A picture frame must contain at least 5 bytes.");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
d->textEncoding = String::Type(data[0]);
|
d->textEncoding = String::Type(data[0]);
|
||||||
|
|
||||||
int pos = 1;
|
size_t pos = 1;
|
||||||
|
|
||||||
String fixedString = String(data.mid(pos, 3), String::Latin1);
|
String fixedString = String(data.mid(pos, 3), String::Latin1);
|
||||||
pos += 3;
|
pos += 3;
|
||||||
// convert fixed string image type to mime string
|
// convert fixed string image type to mime string
|
||||||
if (fixedString.upper() == "JPG") {
|
if (fixedString.upper() == "JPG") {
|
||||||
d->mimeType = "image/jpeg";
|
d->mimeType = "image/jpeg";
|
||||||
} else if (fixedString.upper() == "PNG") {
|
}
|
||||||
|
else if (fixedString.upper() == "PNG") {
|
||||||
d->mimeType = "image/png";
|
d->mimeType = "image/png";
|
||||||
} else {
|
}
|
||||||
|
else {
|
||||||
debug("probably unsupported image type");
|
debug("probably unsupported image type");
|
||||||
d->mimeType = "image/" + fixedString;
|
d->mimeType = "image/" + fixedString;
|
||||||
}
|
}
|
||||||
|
|
||||||
d->type = (Strawberry_TagLib::TagLib::ID3v2::AttachedPictureFrame::Type)data[pos++];
|
d->type = static_cast<ID3v2::AttachedPictureFrame::Type>(data[pos++]);
|
||||||
d->description = readStringField(data, d->textEncoding, &pos);
|
d->description = readStringField(data, d->textEncoding, pos);
|
||||||
|
|
||||||
d->data = data.mid(pos);
|
d->data = data.mid(pos);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
AttachedPictureFrameV22::AttachedPictureFrameV22(const ByteVector &data, Header *h)
|
AttachedPictureFrameV22::AttachedPictureFrameV22(const ByteVector &data, Header *h) {
|
||||||
{
|
|
||||||
// set v2.2 header to make fieldData work correctly
|
// set v2.2 header to make fieldData work correctly
|
||||||
setHeader(h, true);
|
setHeader(h, true);
|
||||||
|
|
||||||
@@ -222,4 +203,5 @@ AttachedPictureFrameV22::AttachedPictureFrameV22(const ByteVector &data, Header
|
|||||||
Frame::Header *newHeader = new Frame::Header("APIC");
|
Frame::Header *newHeader = new Frame::Header("APIC");
|
||||||
newHeader->setFrameSize(h->frameSize());
|
newHeader->setFrameSize(h->frameSize());
|
||||||
setHeader(newHeader, true);
|
setHeader(newHeader, true);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -32,201 +32,195 @@
|
|||||||
|
|
||||||
namespace Strawberry_TagLib {
|
namespace Strawberry_TagLib {
|
||||||
namespace TagLib {
|
namespace TagLib {
|
||||||
|
namespace ID3v2 {
|
||||||
|
|
||||||
namespace ID3v2 {
|
//! An ID3v2 attached picture frame implementation
|
||||||
|
|
||||||
//! An ID3v2 attached picture frame implementation
|
/*!
|
||||||
|
* This is an implementation of ID3v2 attached pictures.
|
||||||
|
* Pictures may be included in tags, one per APIC frame (but there may be multiple APIC frames in a single tag).
|
||||||
|
* These pictures are usually in either JPEG or PNG format.
|
||||||
|
*/
|
||||||
|
|
||||||
/*!
|
class TAGLIB_EXPORT AttachedPictureFrame : public Frame {
|
||||||
* This is an implementation of ID3v2 attached pictures. Pictures may be
|
friend class FrameFactory;
|
||||||
* included in tags, one per APIC frame (but there may be multiple APIC
|
|
||||||
* frames in a single tag). These pictures are usually in either JPEG or
|
|
||||||
* PNG format.
|
|
||||||
*/
|
|
||||||
|
|
||||||
class TAGLIB_EXPORT AttachedPictureFrame : public Frame
|
public:
|
||||||
{
|
/*!
|
||||||
friend class FrameFactory;
|
* This describes the function or content of the picture.
|
||||||
|
*/
|
||||||
|
enum Type {
|
||||||
|
//! A type not enumerated below
|
||||||
|
Other = 0x00,
|
||||||
|
//! 32x32 PNG image that should be used as the file icon
|
||||||
|
FileIcon = 0x01,
|
||||||
|
//! File icon of a different size or format
|
||||||
|
OtherFileIcon = 0x02,
|
||||||
|
//! Front cover image of the album
|
||||||
|
FrontCover = 0x03,
|
||||||
|
//! Back cover image of the album
|
||||||
|
BackCover = 0x04,
|
||||||
|
//! Inside leaflet page of the album
|
||||||
|
LeafletPage = 0x05,
|
||||||
|
//! Image from the album itself
|
||||||
|
Media = 0x06,
|
||||||
|
//! Picture of the lead artist or soloist
|
||||||
|
LeadArtist = 0x07,
|
||||||
|
//! Picture of the artist or performer
|
||||||
|
Artist = 0x08,
|
||||||
|
//! Picture of the conductor
|
||||||
|
Conductor = 0x09,
|
||||||
|
//! Picture of the band or orchestra
|
||||||
|
Band = 0x0A,
|
||||||
|
//! Picture of the composer
|
||||||
|
Composer = 0x0B,
|
||||||
|
//! Picture of the lyricist or text writer
|
||||||
|
Lyricist = 0x0C,
|
||||||
|
//! Picture of the recording location or studio
|
||||||
|
RecordingLocation = 0x0D,
|
||||||
|
//! Picture of the artists during recording
|
||||||
|
DuringRecording = 0x0E,
|
||||||
|
//! Picture of the artists during performance
|
||||||
|
DuringPerformance = 0x0F,
|
||||||
|
//! Picture from a movie or video related to the track
|
||||||
|
MovieScreenCapture = 0x10,
|
||||||
|
//! Picture of a large, coloured fish
|
||||||
|
ColouredFish = 0x11,
|
||||||
|
//! Illustration related to the track
|
||||||
|
Illustration = 0x12,
|
||||||
|
//! Logo of the band or performer
|
||||||
|
BandLogo = 0x13,
|
||||||
|
//! Logo of the publisher (record company)
|
||||||
|
PublisherLogo = 0x14
|
||||||
|
};
|
||||||
|
|
||||||
public:
|
/*!
|
||||||
|
* Constructs an empty picture frame.
|
||||||
|
* The description, content and text encoding should be set manually.
|
||||||
|
*/
|
||||||
|
explicit AttachedPictureFrame();
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* This describes the function or content of the picture.
|
* Constructs an AttachedPicture frame based on \a data.
|
||||||
*/
|
*/
|
||||||
enum Type {
|
explicit AttachedPictureFrame(const ByteVector &data);
|
||||||
//! A type not enumerated below
|
|
||||||
Other = 0x00,
|
|
||||||
//! 32x32 PNG image that should be used as the file icon
|
|
||||||
FileIcon = 0x01,
|
|
||||||
//! File icon of a different size or format
|
|
||||||
OtherFileIcon = 0x02,
|
|
||||||
//! Front cover image of the album
|
|
||||||
FrontCover = 0x03,
|
|
||||||
//! Back cover image of the album
|
|
||||||
BackCover = 0x04,
|
|
||||||
//! Inside leaflet page of the album
|
|
||||||
LeafletPage = 0x05,
|
|
||||||
//! Image from the album itself
|
|
||||||
Media = 0x06,
|
|
||||||
//! Picture of the lead artist or soloist
|
|
||||||
LeadArtist = 0x07,
|
|
||||||
//! Picture of the artist or performer
|
|
||||||
Artist = 0x08,
|
|
||||||
//! Picture of the conductor
|
|
||||||
Conductor = 0x09,
|
|
||||||
//! Picture of the band or orchestra
|
|
||||||
Band = 0x0A,
|
|
||||||
//! Picture of the composer
|
|
||||||
Composer = 0x0B,
|
|
||||||
//! Picture of the lyricist or text writer
|
|
||||||
Lyricist = 0x0C,
|
|
||||||
//! Picture of the recording location or studio
|
|
||||||
RecordingLocation = 0x0D,
|
|
||||||
//! Picture of the artists during recording
|
|
||||||
DuringRecording = 0x0E,
|
|
||||||
//! Picture of the artists during performance
|
|
||||||
DuringPerformance = 0x0F,
|
|
||||||
//! Picture from a movie or video related to the track
|
|
||||||
MovieScreenCapture = 0x10,
|
|
||||||
//! Picture of a large, coloured fish
|
|
||||||
ColouredFish = 0x11,
|
|
||||||
//! Illustration related to the track
|
|
||||||
Illustration = 0x12,
|
|
||||||
//! Logo of the band or performer
|
|
||||||
BandLogo = 0x13,
|
|
||||||
//! Logo of the publisher (record company)
|
|
||||||
PublisherLogo = 0x14
|
|
||||||
};
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Constructs an empty picture frame. The description, content and text
|
* Destroys the AttahcedPictureFrame instance.
|
||||||
* encoding should be set manually.
|
*/
|
||||||
*/
|
~AttachedPictureFrame() override;
|
||||||
AttachedPictureFrame();
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Constructs an AttachedPicture frame based on \a data.
|
* Returns a string containing the description and mime-type
|
||||||
*/
|
*/
|
||||||
explicit AttachedPictureFrame(const ByteVector &data);
|
String toString() const override;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Destroys the AttahcedPictureFrame instance.
|
* Returns the text encoding used for the description.
|
||||||
*/
|
*
|
||||||
virtual ~AttachedPictureFrame();
|
* \see setTextEncoding()
|
||||||
|
* \see description()
|
||||||
|
*/
|
||||||
|
String::Type textEncoding() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns a string containing the description and mime-type
|
* Set the text encoding used for the description.
|
||||||
*/
|
*
|
||||||
virtual String toString() const;
|
* \see description()
|
||||||
|
*/
|
||||||
|
void setTextEncoding(String::Type t);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the text encoding used for the description.
|
* Returns the mime type of the image.
|
||||||
*
|
* This should in most cases be "image/png" or "image/jpeg".
|
||||||
* \see setTextEncoding()
|
*/
|
||||||
* \see description()
|
String mimeType() const;
|
||||||
*/
|
|
||||||
String::Type textEncoding() const;
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Set the text encoding used for the description.
|
* Sets the mime type of the image.
|
||||||
*
|
* This should in most cases be "image/png" or "image/jpeg".
|
||||||
* \see description()
|
*/
|
||||||
*/
|
void setMimeType(const String &m);
|
||||||
void setTextEncoding(String::Type t);
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the mime type of the image. This should in most cases be
|
* Returns the type of the image.
|
||||||
* "image/png" or "image/jpeg".
|
*
|
||||||
*/
|
* \see Type
|
||||||
String mimeType() const;
|
* \see setType()
|
||||||
|
*/
|
||||||
|
Type type() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Sets the mime type of the image. This should in most cases be
|
* Sets the type for the image.
|
||||||
* "image/png" or "image/jpeg".
|
*
|
||||||
*/
|
* \see Type
|
||||||
void setMimeType(const String &m);
|
* \see type()
|
||||||
|
*/
|
||||||
|
void setType(Type t);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the type of the image.
|
* Returns a text description of the image.
|
||||||
*
|
*
|
||||||
* \see Type
|
* \see setDescription()
|
||||||
* \see setType()
|
* \see textEncoding()
|
||||||
*/
|
* \see setTextEncoding()
|
||||||
Type type() const;
|
*/
|
||||||
|
|
||||||
/*!
|
String description() const;
|
||||||
* Sets the type for the image.
|
|
||||||
*
|
|
||||||
* \see Type
|
|
||||||
* \see type()
|
|
||||||
*/
|
|
||||||
void setType(Type t);
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns a text description of the image.
|
* Sets a textual description of the image to \a desc.
|
||||||
*
|
*
|
||||||
* \see setDescription()
|
* \see description()
|
||||||
* \see textEncoding()
|
* \see textEncoding()
|
||||||
* \see setTextEncoding()
|
* \see setTextEncoding()
|
||||||
*/
|
*/
|
||||||
|
|
||||||
String description() const;
|
void setDescription(const String &desc);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Sets a textual description of the image to \a desc.
|
* Returns the image data as a ByteVector.
|
||||||
*
|
*
|
||||||
* \see description()
|
* \note ByteVector has a data() method that returns a const char * which should make it easy to export this data to external programs.
|
||||||
* \see textEncoding()
|
*
|
||||||
* \see setTextEncoding()
|
* \see setPicture()
|
||||||
*/
|
* \see mimeType()
|
||||||
|
*/
|
||||||
|
ByteVector picture() const;
|
||||||
|
|
||||||
void setDescription(const String &desc);
|
/*!
|
||||||
|
* Sets the image data to \a p. \a p should be of the type specified in this frame's mime-type specification.
|
||||||
|
*
|
||||||
|
* \see picture()
|
||||||
|
* \see mimeType()
|
||||||
|
* \see setMimeType()
|
||||||
|
*/
|
||||||
|
void setPicture(const ByteVector &p);
|
||||||
|
|
||||||
/*!
|
protected:
|
||||||
* Returns the image data as a ByteVector.
|
void parseFields(const ByteVector &data) override;
|
||||||
*
|
ByteVector renderFields() const override;
|
||||||
* \note ByteVector has a data() method that returns a const char * which
|
class AttachedPictureFramePrivate;
|
||||||
* should make it easy to export this data to external programs.
|
AttachedPictureFramePrivate *d;
|
||||||
*
|
|
||||||
* \see setPicture()
|
|
||||||
* \see mimeType()
|
|
||||||
*/
|
|
||||||
ByteVector picture() const;
|
|
||||||
|
|
||||||
/*!
|
private:
|
||||||
* Sets the image data to \a p. \a p should be of the type specified in
|
AttachedPictureFrame(const AttachedPictureFrame&);
|
||||||
* this frame's mime-type specification.
|
AttachedPictureFrame &operator=(const AttachedPictureFrame&);
|
||||||
*
|
explicit AttachedPictureFrame(const ByteVector &data, Header *h);
|
||||||
* \see picture()
|
};
|
||||||
* \see mimeType()
|
|
||||||
* \see setMimeType()
|
|
||||||
*/
|
|
||||||
void setPicture(const ByteVector &p);
|
|
||||||
|
|
||||||
protected:
|
//! support for ID3v2.2 PIC frames
|
||||||
virtual void parseFields(const ByteVector &data);
|
class TAGLIB_EXPORT AttachedPictureFrameV22 : public AttachedPictureFrame {
|
||||||
virtual ByteVector renderFields() const;
|
protected:
|
||||||
class AttachedPictureFramePrivate;
|
void parseFields(const ByteVector &data) override;
|
||||||
AttachedPictureFramePrivate *d;
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
AttachedPictureFrame(const AttachedPictureFrame &);
|
explicit AttachedPictureFrameV22(const ByteVector &data, Header *h);
|
||||||
AttachedPictureFrame &operator=(const AttachedPictureFrame &);
|
friend class FrameFactory;
|
||||||
AttachedPictureFrame(const ByteVector &data, Header *h);
|
};
|
||||||
|
|
||||||
};
|
} // namespace ID3v2
|
||||||
|
} // namespace TagLib
|
||||||
//! support for ID3v2.2 PIC frames
|
} // namespace Strawberry_TagLib
|
||||||
class TAGLIB_EXPORT AttachedPictureFrameV22 : public AttachedPictureFrame
|
|
||||||
{
|
|
||||||
protected:
|
|
||||||
virtual void parseFields(const ByteVector &data);
|
|
||||||
private:
|
|
||||||
AttachedPictureFrameV22(const ByteVector &data, Header *h);
|
|
||||||
friend class FrameFactory;
|
|
||||||
};
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
206
3rdparty/taglib/mpeg/id3v2/frames/chapterframe.cpp
vendored
206
3rdparty/taglib/mpeg/id3v2/frames/chapterframe.cpp
vendored
@@ -23,26 +23,24 @@
|
|||||||
* http://www.mozilla.org/MPL/ *
|
* http://www.mozilla.org/MPL/ *
|
||||||
***************************************************************************/
|
***************************************************************************/
|
||||||
|
|
||||||
#include <tbytevectorlist.h>
|
#include <cstdio>
|
||||||
#include <tpropertymap.h>
|
|
||||||
#include <tdebug.h>
|
#include "tbytevectorlist.h"
|
||||||
#include <stdio.h>
|
#include "tpropertymap.h"
|
||||||
|
#include "tdebug.h"
|
||||||
|
|
||||||
#include "chapterframe.h"
|
#include "chapterframe.h"
|
||||||
|
|
||||||
using namespace Strawberry_TagLib::TagLib;
|
using namespace Strawberry_TagLib::TagLib;
|
||||||
using namespace ID3v2;
|
using namespace ID3v2;
|
||||||
|
|
||||||
class ChapterFrame::ChapterFramePrivate
|
class ChapterFrame::ChapterFramePrivate {
|
||||||
{
|
public:
|
||||||
public:
|
explicit ChapterFramePrivate() : tagHeader(nullptr),
|
||||||
ChapterFramePrivate() :
|
startTime(0),
|
||||||
tagHeader(0),
|
endTime(0),
|
||||||
startTime(0),
|
startOffset(0),
|
||||||
endTime(0),
|
endOffset(0) {
|
||||||
startOffset(0),
|
|
||||||
endOffset(0)
|
|
||||||
{
|
|
||||||
embeddedFrameList.setAutoDelete(true);
|
embeddedFrameList.setAutoDelete(true);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -60,23 +58,14 @@ public:
|
|||||||
// public methods
|
// public methods
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
ChapterFrame::ChapterFrame(const ID3v2::Header *tagHeader, const ByteVector &data) :
|
ChapterFrame::ChapterFrame(const ID3v2::Header *tagHeader, const ByteVector &data) : ID3v2::Frame(data), d(new ChapterFramePrivate()) {
|
||||||
ID3v2::Frame(data),
|
|
||||||
d(new ChapterFramePrivate())
|
|
||||||
{
|
|
||||||
d->tagHeader = tagHeader;
|
d->tagHeader = tagHeader;
|
||||||
setData(data);
|
setData(data);
|
||||||
}
|
}
|
||||||
|
|
||||||
ChapterFrame::ChapterFrame(const ByteVector &elementID,
|
ChapterFrame::ChapterFrame(const ByteVector &elementID, unsigned int startTime, unsigned int endTime, unsigned int startOffset, unsigned int endOffset, const FrameList &embeddedFrames) : ID3v2::Frame("CHAP"), d(new ChapterFramePrivate()) {
|
||||||
unsigned int startTime, unsigned int endTime,
|
|
||||||
unsigned int startOffset, unsigned int endOffset,
|
// setElementID has a workaround for a previously silly API where you had to specifically include the null byte.
|
||||||
const FrameList &embeddedFrames) :
|
|
||||||
ID3v2::Frame("CHAP"),
|
|
||||||
d(new ChapterFramePrivate())
|
|
||||||
{
|
|
||||||
// setElementID has a workaround for a previously silly API where you had to
|
|
||||||
// specifically include the null byte.
|
|
||||||
|
|
||||||
setElementID(elementID);
|
setElementID(elementID);
|
||||||
|
|
||||||
@@ -85,92 +74,81 @@ ChapterFrame::ChapterFrame(const ByteVector &elementID,
|
|||||||
d->startOffset = startOffset;
|
d->startOffset = startOffset;
|
||||||
d->endOffset = endOffset;
|
d->endOffset = endOffset;
|
||||||
|
|
||||||
for(FrameList::ConstIterator it = embeddedFrames.begin();
|
for (FrameList::ConstIterator it = embeddedFrames.begin();
|
||||||
it != embeddedFrames.end(); ++it)
|
it != embeddedFrames.end();
|
||||||
|
++it)
|
||||||
addEmbeddedFrame(*it);
|
addEmbeddedFrame(*it);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ChapterFrame::~ChapterFrame()
|
ChapterFrame::~ChapterFrame() {
|
||||||
{
|
|
||||||
delete d;
|
delete d;
|
||||||
}
|
}
|
||||||
|
|
||||||
ByteVector ChapterFrame::elementID() const
|
ByteVector ChapterFrame::elementID() const {
|
||||||
{
|
|
||||||
return d->elementID;
|
return d->elementID;
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned int ChapterFrame::startTime() const
|
unsigned int ChapterFrame::startTime() const {
|
||||||
{
|
|
||||||
return d->startTime;
|
return d->startTime;
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned int ChapterFrame::endTime() const
|
unsigned int ChapterFrame::endTime() const {
|
||||||
{
|
|
||||||
return d->endTime;
|
return d->endTime;
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned int ChapterFrame::startOffset() const
|
unsigned int ChapterFrame::startOffset() const {
|
||||||
{
|
|
||||||
return d->startOffset;
|
return d->startOffset;
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned int ChapterFrame::endOffset() const
|
unsigned int ChapterFrame::endOffset() const {
|
||||||
{
|
|
||||||
return d->endOffset;
|
return d->endOffset;
|
||||||
}
|
}
|
||||||
|
|
||||||
void ChapterFrame::setElementID(const ByteVector &eID)
|
void ChapterFrame::setElementID(const ByteVector &eID) {
|
||||||
{
|
|
||||||
d->elementID = eID;
|
d->elementID = eID;
|
||||||
|
|
||||||
if(d->elementID.endsWith(char(0)))
|
if (d->elementID.endsWith(char(0)))
|
||||||
d->elementID = d->elementID.mid(0, d->elementID.size() - 1);
|
d->elementID = d->elementID.mid(0, d->elementID.size() - 1);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void ChapterFrame::setStartTime(const unsigned int &sT)
|
void ChapterFrame::setStartTime(const unsigned int &sT) {
|
||||||
{
|
|
||||||
d->startTime = sT;
|
d->startTime = sT;
|
||||||
}
|
}
|
||||||
|
|
||||||
void ChapterFrame::setEndTime(const unsigned int &eT)
|
void ChapterFrame::setEndTime(const unsigned int &eT) {
|
||||||
{
|
|
||||||
d->endTime = eT;
|
d->endTime = eT;
|
||||||
}
|
}
|
||||||
|
|
||||||
void ChapterFrame::setStartOffset(const unsigned int &sO)
|
void ChapterFrame::setStartOffset(const unsigned int &sO) {
|
||||||
{
|
|
||||||
d->startOffset = sO;
|
d->startOffset = sO;
|
||||||
}
|
}
|
||||||
|
|
||||||
void ChapterFrame::setEndOffset(const unsigned int &eO)
|
void ChapterFrame::setEndOffset(const unsigned int &eO) {
|
||||||
{
|
|
||||||
d->endOffset = eO;
|
d->endOffset = eO;
|
||||||
}
|
}
|
||||||
|
|
||||||
const FrameListMap &ChapterFrame::embeddedFrameListMap() const
|
const FrameListMap &ChapterFrame::embeddedFrameListMap() const {
|
||||||
{
|
|
||||||
return d->embeddedFrameListMap;
|
return d->embeddedFrameListMap;
|
||||||
}
|
}
|
||||||
|
|
||||||
const FrameList &ChapterFrame::embeddedFrameList() const
|
const FrameList &ChapterFrame::embeddedFrameList() const {
|
||||||
{
|
|
||||||
return d->embeddedFrameList;
|
return d->embeddedFrameList;
|
||||||
}
|
}
|
||||||
|
|
||||||
const FrameList &ChapterFrame::embeddedFrameList(const ByteVector &frameID) const
|
const FrameList &ChapterFrame::embeddedFrameList(const ByteVector &frameID) const {
|
||||||
{
|
|
||||||
return d->embeddedFrameListMap[frameID];
|
return d->embeddedFrameListMap[frameID];
|
||||||
}
|
}
|
||||||
|
|
||||||
void ChapterFrame::addEmbeddedFrame(Frame *frame)
|
void ChapterFrame::addEmbeddedFrame(Frame *frame) {
|
||||||
{
|
|
||||||
d->embeddedFrameList.append(frame);
|
d->embeddedFrameList.append(frame);
|
||||||
d->embeddedFrameListMap[frame->frameID()].append(frame);
|
d->embeddedFrameListMap[frame->frameID()].append(frame);
|
||||||
}
|
}
|
||||||
|
|
||||||
void ChapterFrame::removeEmbeddedFrame(Frame *frame, bool del)
|
void ChapterFrame::removeEmbeddedFrame(Frame *frame, bool del) {
|
||||||
{
|
|
||||||
// remove the frame from the frame list
|
// remove the frame from the frame list
|
||||||
FrameList::Iterator it = d->embeddedFrameList.find(frame);
|
FrameList::Iterator it = d->embeddedFrameList.find(frame);
|
||||||
d->embeddedFrameList.erase(it);
|
d->embeddedFrameList.erase(it);
|
||||||
@@ -180,100 +158,105 @@ void ChapterFrame::removeEmbeddedFrame(Frame *frame, bool del)
|
|||||||
d->embeddedFrameListMap[frame->frameID()].erase(it);
|
d->embeddedFrameListMap[frame->frameID()].erase(it);
|
||||||
|
|
||||||
// ...and delete as desired
|
// ...and delete as desired
|
||||||
if(del)
|
if (del)
|
||||||
delete frame;
|
delete frame;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void ChapterFrame::removeEmbeddedFrames(const ByteVector &id)
|
void ChapterFrame::removeEmbeddedFrames(const ByteVector &id) {
|
||||||
{
|
|
||||||
FrameList l = d->embeddedFrameListMap[id];
|
FrameList l = d->embeddedFrameListMap[id];
|
||||||
for(FrameList::ConstIterator it = l.begin(); it != l.end(); ++it)
|
for (FrameList::ConstIterator it = l.begin(); it != l.end(); ++it)
|
||||||
removeEmbeddedFrame(*it, true);
|
removeEmbeddedFrame(*it, true);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
String ChapterFrame::toString() const
|
String ChapterFrame::toString() const {
|
||||||
{
|
|
||||||
String s = String(d->elementID) +
|
|
||||||
": start time: " + String::number(d->startTime) +
|
|
||||||
", end time: " + String::number(d->endTime);
|
|
||||||
|
|
||||||
if(d->startOffset != 0xFFFFFFFF)
|
String s = String(d->elementID) +
|
||||||
|
": start time: " + String::number(d->startTime) +
|
||||||
|
", end time: " + String::number(d->endTime);
|
||||||
|
|
||||||
|
if (d->startOffset != 0xFFFFFFFF)
|
||||||
s += ", start offset: " + String::number(d->startOffset);
|
s += ", start offset: " + String::number(d->startOffset);
|
||||||
|
|
||||||
if(d->endOffset != 0xFFFFFFFF)
|
if (d->endOffset != 0xFFFFFFFF)
|
||||||
s += ", end offset: " + String::number(d->endOffset);
|
s += ", end offset: " + String::number(d->endOffset);
|
||||||
|
|
||||||
if(!d->embeddedFrameList.isEmpty()) {
|
if (!d->embeddedFrameList.isEmpty()) {
|
||||||
StringList frameIDs;
|
StringList frameIDs;
|
||||||
for(FrameList::ConstIterator it = d->embeddedFrameList.begin();
|
for (FrameList::ConstIterator it = d->embeddedFrameList.begin();
|
||||||
it != d->embeddedFrameList.end(); ++it)
|
it != d->embeddedFrameList.end();
|
||||||
|
++it)
|
||||||
frameIDs.append((*it)->frameID());
|
frameIDs.append((*it)->frameID());
|
||||||
s += ", sub-frames: [ " + frameIDs.toString(", ") + " ]";
|
s += ", sub-frames: [ " + frameIDs.toString(", ") + " ]";
|
||||||
}
|
}
|
||||||
|
|
||||||
return s;
|
return s;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
PropertyMap ChapterFrame::asProperties() const
|
PropertyMap ChapterFrame::asProperties() const {
|
||||||
{
|
|
||||||
PropertyMap map;
|
PropertyMap map;
|
||||||
|
|
||||||
map.unsupportedData().append(frameID() + String("/") + d->elementID);
|
map.unsupportedData().append(frameID() + String("/") + d->elementID);
|
||||||
|
|
||||||
return map;
|
return map;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ChapterFrame *ChapterFrame::findByElementID(const ID3v2::Tag *tag, const ByteVector &eID) // static
|
ChapterFrame *ChapterFrame::findByElementID(const ID3v2::Tag *tag, const ByteVector &eID) { // static
|
||||||
{
|
|
||||||
ID3v2::FrameList comments = tag->frameList("CHAP");
|
ID3v2::FrameList comments = tag->frameList("CHAP");
|
||||||
|
|
||||||
for(ID3v2::FrameList::ConstIterator it = comments.begin();
|
for (ID3v2::FrameList::ConstIterator it = comments.begin();
|
||||||
it != comments.end();
|
it != comments.end();
|
||||||
++it)
|
++it) {
|
||||||
{
|
|
||||||
ChapterFrame *frame = dynamic_cast<ChapterFrame *>(*it);
|
ChapterFrame *frame = dynamic_cast<ChapterFrame *>(*it);
|
||||||
if(frame && frame->elementID() == eID)
|
if (frame && frame->elementID() == eID)
|
||||||
return frame;
|
return frame;
|
||||||
}
|
}
|
||||||
|
|
||||||
return nullptr;
|
return nullptr;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void ChapterFrame::parseFields(const ByteVector &data)
|
void ChapterFrame::parseFields(const ByteVector &data) {
|
||||||
{
|
|
||||||
unsigned int size = data.size();
|
size_t size = data.size();
|
||||||
if(size < 18) {
|
if (size < 18) {
|
||||||
debug("A CHAP frame must contain at least 18 bytes (1 byte element ID "
|
debug("A CHAP frame must contain at least 18 bytes (1 byte element ID "
|
||||||
"terminated by null and 4x4 bytes for start and end time and offset).");
|
"terminated by null and 4x4 bytes for start and end time and offset).");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
int pos = 0;
|
size_t pos = 0;
|
||||||
unsigned int embPos = 0;
|
size_t embPos = 0;
|
||||||
d->elementID = readStringField(data, String::Latin1, &pos).data(String::Latin1);
|
d->elementID = readStringField(data, String::Latin1, pos).data(String::Latin1);
|
||||||
d->startTime = data.toUInt(pos, true);
|
d->startTime = data.toUInt32BE(pos);
|
||||||
pos += 4;
|
pos += 4;
|
||||||
d->endTime = data.toUInt(pos, true);
|
d->endTime = data.toUInt32BE(pos);
|
||||||
pos += 4;
|
pos += 4;
|
||||||
d->startOffset = data.toUInt(pos, true);
|
d->startOffset = data.toUInt32BE(pos);
|
||||||
pos += 4;
|
pos += 4;
|
||||||
d->endOffset = data.toUInt(pos, true);
|
d->endOffset = data.toUInt32BE(pos);
|
||||||
pos += 4;
|
pos += 4;
|
||||||
size -= pos;
|
size -= pos;
|
||||||
|
|
||||||
// Embedded frames are optional
|
// Embedded frames are optional
|
||||||
|
|
||||||
if(size < header()->size())
|
if (size < header()->size())
|
||||||
return;
|
return;
|
||||||
|
|
||||||
while(embPos < size - header()->size()) {
|
while (embPos < size - header()->size()) {
|
||||||
Frame *frame = FrameFactory::instance()->createFrame(data.mid(pos + embPos), d->tagHeader);
|
Frame *frame = FrameFactory::instance()->createFrame(data.mid(pos + embPos), d->tagHeader);
|
||||||
|
|
||||||
if(!frame)
|
if (!frame)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
// Checks to make sure that frame parsed correctly.
|
// Checks to make sure that frame parsed correctly.
|
||||||
if(frame->size() <= 0) {
|
if (frame->size() <= 0) {
|
||||||
delete frame;
|
delete frame;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@@ -281,29 +264,30 @@ void ChapterFrame::parseFields(const ByteVector &data)
|
|||||||
embPos += frame->size() + header()->size();
|
embPos += frame->size() + header()->size();
|
||||||
addEmbeddedFrame(frame);
|
addEmbeddedFrame(frame);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ByteVector ChapterFrame::renderFields() const
|
ByteVector ChapterFrame::renderFields() const {
|
||||||
{
|
|
||||||
ByteVector data;
|
ByteVector data;
|
||||||
|
|
||||||
data.append(d->elementID);
|
data.append(d->elementID);
|
||||||
data.append('\0');
|
data.append('\0');
|
||||||
data.append(ByteVector::fromUInt(d->startTime, true));
|
data.append(ByteVector::fromUInt32BE(d->startTime));
|
||||||
data.append(ByteVector::fromUInt(d->endTime, true));
|
data.append(ByteVector::fromUInt32BE(d->endTime));
|
||||||
data.append(ByteVector::fromUInt(d->startOffset, true));
|
data.append(ByteVector::fromUInt32BE(d->startOffset));
|
||||||
data.append(ByteVector::fromUInt(d->endOffset, true));
|
data.append(ByteVector::fromUInt32BE(d->endOffset));
|
||||||
FrameList l = d->embeddedFrameList;
|
FrameList l = d->embeddedFrameList;
|
||||||
for(FrameList::ConstIterator it = l.begin(); it != l.end(); ++it)
|
for (FrameList::ConstIterator it = l.begin(); it != l.end(); ++it)
|
||||||
data.append((*it)->render());
|
data.append((*it)->render());
|
||||||
|
|
||||||
return data;
|
return data;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ChapterFrame::ChapterFrame(const ID3v2::Header *tagHeader, const ByteVector &data, Header *h) :
|
ChapterFrame::ChapterFrame(const ID3v2::Header *tagHeader, const ByteVector &data, Header *h) : Frame(h), d(new ChapterFramePrivate()) {
|
||||||
Frame(h),
|
|
||||||
d(new ChapterFramePrivate())
|
|
||||||
{
|
|
||||||
d->tagHeader = tagHeader;
|
d->tagHeader = tagHeader;
|
||||||
parseFields(fieldData(data));
|
parseFields(fieldData(data));
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
354
3rdparty/taglib/mpeg/id3v2/frames/chapterframe.h
vendored
354
3rdparty/taglib/mpeg/id3v2/frames/chapterframe.h
vendored
@@ -32,220 +32,202 @@
|
|||||||
|
|
||||||
namespace Strawberry_TagLib {
|
namespace Strawberry_TagLib {
|
||||||
namespace TagLib {
|
namespace TagLib {
|
||||||
|
namespace ID3v2 {
|
||||||
|
|
||||||
namespace ID3v2 {
|
/*!
|
||||||
|
* This is an implementation of ID3v2 chapter frames.
|
||||||
|
* The purpose of this frame is to describe a single chapter within an audio file.
|
||||||
|
*/
|
||||||
|
|
||||||
/*!
|
//! An implementation of ID3v2 chapter frames
|
||||||
* This is an implementation of ID3v2 chapter frames. The purpose of this
|
|
||||||
* frame is to describe a single chapter within an audio file.
|
|
||||||
*/
|
|
||||||
|
|
||||||
//! An implementation of ID3v2 chapter frames
|
class TAGLIB_EXPORT ChapterFrame : public ID3v2::Frame {
|
||||||
|
friend class FrameFactory;
|
||||||
|
|
||||||
class TAGLIB_EXPORT ChapterFrame : public ID3v2::Frame
|
public:
|
||||||
{
|
/*!
|
||||||
friend class FrameFactory;
|
* Creates a chapter frame based on \a data.
|
||||||
|
* \a tagHeader is required as the internal frames are parsed based on the tag version.
|
||||||
|
*/
|
||||||
|
explicit ChapterFrame(const ID3v2::Header *tagHeader, const ByteVector &data);
|
||||||
|
|
||||||
public:
|
/*!
|
||||||
/*!
|
* Creates a chapter frame with the element ID \a elementID,
|
||||||
* Creates a chapter frame based on \a data. \a tagHeader is required as
|
* start time \a startTime, end time \a endTime, start offset \a startOffset,
|
||||||
* the internal frames are parsed based on the tag version.
|
* end offset \a endOffset and optionally a list of embedded frames,
|
||||||
*/
|
* whose ownership will then be taken over by this Frame, in \a embeededFrames;
|
||||||
ChapterFrame(const ID3v2::Header *tagHeader, const ByteVector &data);
|
*
|
||||||
|
* All times are in milliseconds.
|
||||||
|
*/
|
||||||
|
explicit ChapterFrame(const ByteVector &elementID, unsigned int startTime, unsigned int endTime, unsigned int startOffset, unsigned int endOffset, const FrameList &embeddedFrames = FrameList());
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Creates a chapter frame with the element ID \a elementID, start time
|
* Destroys the frame.
|
||||||
* \a startTime, end time \a endTime, start offset \a startOffset,
|
*/
|
||||||
* end offset \a endOffset and optionally a list of embedded frames,
|
~ChapterFrame() override;
|
||||||
* whose ownership will then be taken over by this Frame, in
|
|
||||||
* \a embeededFrames;
|
|
||||||
*
|
|
||||||
* All times are in milliseconds.
|
|
||||||
*/
|
|
||||||
ChapterFrame(const ByteVector &elementID,
|
|
||||||
unsigned int startTime, unsigned int endTime,
|
|
||||||
unsigned int startOffset, unsigned int endOffset,
|
|
||||||
const FrameList &embeddedFrames = FrameList());
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Destroys the frame.
|
* Returns the element ID of the frame.
|
||||||
*/
|
* Element ID is a null terminated string, however it's not human-readable.
|
||||||
virtual ~ChapterFrame();
|
*
|
||||||
|
* \see setElementID()
|
||||||
|
*/
|
||||||
|
ByteVector elementID() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the element ID of the frame. Element ID
|
* Returns time of chapter's start (in milliseconds).
|
||||||
* is a null terminated string, however it's not human-readable.
|
*
|
||||||
*
|
* \see setStartTime()
|
||||||
* \see setElementID()
|
*/
|
||||||
*/
|
unsigned int startTime() const;
|
||||||
ByteVector elementID() const;
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns time of chapter's start (in milliseconds).
|
* Returns time of chapter's end (in milliseconds).
|
||||||
*
|
*
|
||||||
* \see setStartTime()
|
* \see setEndTime()
|
||||||
*/
|
*/
|
||||||
unsigned int startTime() const;
|
unsigned int endTime() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns time of chapter's end (in milliseconds).
|
* Returns zero based byte offset (count of bytes from the beginning of the audio file) of chapter's start.
|
||||||
*
|
*
|
||||||
* \see setEndTime()
|
* \note If returned value is 0xFFFFFFFF, start time should be used instead.
|
||||||
*/
|
* \see setStartOffset()
|
||||||
unsigned int endTime() const;
|
*/
|
||||||
|
unsigned int startOffset() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns zero based byte offset (count of bytes from the beginning
|
* Returns zero based byte offset (count of bytes from the beginning of the audio file) of chapter's end.
|
||||||
* of the audio file) of chapter's start.
|
*
|
||||||
*
|
* \note If returned value is 0xFFFFFFFF, end time should be used instead.
|
||||||
* \note If returned value is 0xFFFFFFFF, start time should be used instead.
|
* \see setEndOffset()
|
||||||
* \see setStartOffset()
|
*/
|
||||||
*/
|
unsigned int endOffset() const;
|
||||||
unsigned int startOffset() const;
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns zero based byte offset (count of bytes from the beginning
|
* Sets the element ID of the frame to \a eID. If \a eID isn't null terminated, a null char is appended automatically.
|
||||||
* of the audio file) of chapter's end.
|
*
|
||||||
*
|
* \see elementID()
|
||||||
* \note If returned value is 0xFFFFFFFF, end time should be used instead.
|
*/
|
||||||
* \see setEndOffset()
|
void setElementID(const ByteVector &eID);
|
||||||
*/
|
|
||||||
unsigned int endOffset() const;
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Sets the element ID of the frame to \a eID. If \a eID isn't
|
* Sets time of chapter's start (in milliseconds) to \a sT.
|
||||||
* null terminated, a null char is appended automatically.
|
*
|
||||||
*
|
* \see startTime()
|
||||||
* \see elementID()
|
*/
|
||||||
*/
|
void setStartTime(const unsigned int &sT);
|
||||||
void setElementID(const ByteVector &eID);
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Sets time of chapter's start (in milliseconds) to \a sT.
|
* Sets time of chapter's end (in milliseconds) to \a eT.
|
||||||
*
|
*
|
||||||
* \see startTime()
|
* \see endTime()
|
||||||
*/
|
*/
|
||||||
void setStartTime(const unsigned int &sT);
|
void setEndTime(const unsigned int &eT);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Sets time of chapter's end (in milliseconds) to \a eT.
|
* Sets zero based byte offset (count of bytes from the beginning of the audio file) of chapter's start to \a sO.
|
||||||
*
|
*
|
||||||
* \see endTime()
|
* \see startOffset()
|
||||||
*/
|
*/
|
||||||
void setEndTime(const unsigned int &eT);
|
void setStartOffset(const unsigned int &sO);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Sets zero based byte offset (count of bytes from the beginning
|
* Sets zero based byte offset (count of bytes from the beginning of the audio file) of chapter's end to \a eO.
|
||||||
* of the audio file) of chapter's start to \a sO.
|
*
|
||||||
*
|
* \see endOffset()
|
||||||
* \see startOffset()
|
*/
|
||||||
*/
|
void setEndOffset(const unsigned int &eO);
|
||||||
void setStartOffset(const unsigned int &sO);
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Sets zero based byte offset (count of bytes from the beginning
|
* Returns a reference to the frame list map.
|
||||||
* of the audio file) of chapter's end to \a eO.
|
* This is an FrameListMap of all of the frames embedded in the CHAP frame.
|
||||||
*
|
*
|
||||||
* \see endOffset()
|
* This is the most convenient structure for accessing the CHAP frame's embedded frames.
|
||||||
*/
|
* Many frame types allow multiple instances of the same rame type so this is a map of lists.
|
||||||
void setEndOffset(const unsigned int &eO);
|
* In most cases however there will only be a single frame of a certain type.
|
||||||
|
*
|
||||||
|
* \warning You should not modify this data structure directly, instead
|
||||||
|
* use addEmbeddedFrame() and removeEmbeddedFrame().
|
||||||
|
*
|
||||||
|
* \see embeddedFrameList()
|
||||||
|
*/
|
||||||
|
const FrameListMap &embeddedFrameListMap() const;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns a reference to the frame list map. This is an FrameListMap of
|
* Returns a reference to the embedded frame list.
|
||||||
* all of the frames embedded in the CHAP frame.
|
* This is an FrameList of all of the frames embedded in the CHAP frame in the order that they were parsed.
|
||||||
*
|
*
|
||||||
* This is the most convenient structure for accessing the CHAP frame's
|
* This can be useful if for example you want iterate over the CHAP frame's
|
||||||
* embedded frames. Many frame types allow multiple instances of the same
|
* embedded frames in the order that they occur in the CHAP frame.
|
||||||
* frame type so this is a map of lists. In most cases however there will
|
*
|
||||||
* only be a single frame of a certain type.
|
* \warning You should not modify this data structure directly,
|
||||||
*
|
* instead use addEmbeddedFrame() and removeEmbeddedFrame().
|
||||||
* \warning You should not modify this data structure directly, instead
|
*/
|
||||||
* use addEmbeddedFrame() and removeEmbeddedFrame().
|
const FrameList &embeddedFrameList() const;
|
||||||
*
|
|
||||||
* \see embeddedFrameList()
|
|
||||||
*/
|
|
||||||
const FrameListMap &embeddedFrameListMap() const;
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns a reference to the embedded frame list. This is an FrameList
|
* Returns the embedded frame list for frames with the id \a frameID or an empty list if there are no embedded frames of that type.
|
||||||
* of all of the frames embedded in the CHAP frame in the order that they
|
* This is just a convenience and is equivalent to:
|
||||||
* were parsed.
|
*
|
||||||
*
|
* \code
|
||||||
* This can be useful if for example you want iterate over the CHAP frame's
|
* embeddedFrameListMap()[frameID];
|
||||||
* embedded frames in the order that they occur in the CHAP frame.
|
* \endcode
|
||||||
*
|
*
|
||||||
* \warning You should not modify this data structure directly, instead
|
* \see embeddedFrameListMap()
|
||||||
* use addEmbeddedFrame() and removeEmbeddedFrame().
|
*/
|
||||||
*/
|
const FrameList &embeddedFrameList(const ByteVector &frameID) const;
|
||||||
const FrameList &embeddedFrameList() const;
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Returns the embedded frame list for frames with the id \a frameID
|
* Add an embedded frame to the CHAP frame.
|
||||||
* or an empty list if there are no embedded frames of that type. This
|
* At this point the CHAP frame takes ownership of the embedded frame and will handle freeing its memory.
|
||||||
* is just a convenience and is equivalent to:
|
*
|
||||||
*
|
* \note Using this method will invalidate any pointers on the list returned by embeddedFrameList()
|
||||||
* \code
|
*/
|
||||||
* embeddedFrameListMap()[frameID];
|
void addEmbeddedFrame(Frame *frame);
|
||||||
* \endcode
|
|
||||||
*
|
|
||||||
* \see embeddedFrameListMap()
|
|
||||||
*/
|
|
||||||
const FrameList &embeddedFrameList(const ByteVector &frameID) const;
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Add an embedded frame to the CHAP frame. At this point the CHAP frame
|
* Remove an embedded frame from the CHAP frame.
|
||||||
* takes ownership of the embedded frame and will handle freeing its memory.
|
* If \a del is true the frame's memory will be freed; if it is false, it must be deleted by the user.
|
||||||
*
|
*
|
||||||
* \note Using this method will invalidate any pointers on the list
|
* \note Using this method will invalidate any pointers on the list returned by embeddedFrameList()
|
||||||
* returned by embeddedFrameList()
|
*/
|
||||||
*/
|
void removeEmbeddedFrame(Frame *frame, bool del = true);
|
||||||
void addEmbeddedFrame(Frame *frame);
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Remove an embedded frame from the CHAP frame. If \a del is true the frame's
|
* Remove all embedded frames of type \a id from the CHAP frame and free their memory.
|
||||||
* memory will be freed; if it is false, it must be deleted by the user.
|
*
|
||||||
*
|
* \note Using this method will invalidate any pointers on the list returned by embeddedFrameList()
|
||||||
* \note Using this method will invalidate any pointers on the list
|
*/
|
||||||
* returned by embeddedFrameList()
|
void removeEmbeddedFrames(const ByteVector &id);
|
||||||
*/
|
|
||||||
void removeEmbeddedFrame(Frame *frame, bool del = true);
|
|
||||||
|
|
||||||
/*!
|
String toString() const override;
|
||||||
* Remove all embedded frames of type \a id from the CHAP frame and free their
|
|
||||||
* memory.
|
|
||||||
*
|
|
||||||
* \note Using this method will invalidate any pointers on the list
|
|
||||||
* returned by embeddedFrameList()
|
|
||||||
*/
|
|
||||||
void removeEmbeddedFrames(const ByteVector &id);
|
|
||||||
|
|
||||||
virtual String toString() const;
|
PropertyMap asProperties() const;
|
||||||
|
|
||||||
PropertyMap asProperties() const;
|
/*!
|
||||||
|
* CHAP frames each have a unique element ID. This searches for a CHAP frame with the element ID \a eID and returns a pointer to it.
|
||||||
|
* This can be used to link CTOC and CHAP frames together.
|
||||||
|
*
|
||||||
|
* \see elementID()
|
||||||
|
*/
|
||||||
|
static ChapterFrame *findByElementID(const Tag *tag, const ByteVector &eID);
|
||||||
|
|
||||||
/*!
|
protected:
|
||||||
* CHAP frames each have a unique element ID. This searches for a CHAP
|
void parseFields(const ByteVector &data) override;
|
||||||
* frame with the element ID \a eID and returns a pointer to it. This
|
ByteVector renderFields() const override;
|
||||||
* can be used to link CTOC and CHAP frames together.
|
|
||||||
*
|
|
||||||
* \see elementID()
|
|
||||||
*/
|
|
||||||
static ChapterFrame *findByElementID(const Tag *tag, const ByteVector &eID);
|
|
||||||
|
|
||||||
protected:
|
private:
|
||||||
virtual void parseFields(const ByteVector &data);
|
explicit ChapterFrame(const ID3v2::Header *tagHeader, const ByteVector &data, Header *h);
|
||||||
virtual ByteVector renderFields() const;
|
ChapterFrame(const ChapterFrame&);
|
||||||
|
ChapterFrame &operator=(const ChapterFrame&);
|
||||||
|
|
||||||
private:
|
class ChapterFramePrivate;
|
||||||
ChapterFrame(const ID3v2::Header *tagHeader, const ByteVector &data, Header *h);
|
ChapterFramePrivate *d;
|
||||||
ChapterFrame(const ChapterFrame &);
|
};
|
||||||
ChapterFrame &operator=(const ChapterFrame &);
|
|
||||||
|
|
||||||
class ChapterFramePrivate;
|
} // namespace ID3v2
|
||||||
ChapterFramePrivate *d;
|
} // namespace TagLib
|
||||||
};
|
} // namespace Strawberry_TagLib
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user