mirror of
https://github.com/TeamNewPipe/NewPipe
synced 2025-10-02 15:00:51 +02:00
Compare commits
1819 Commits
Author | SHA1 | Date | |
---|---|---|---|
![]() |
f24fab0fa2 | ||
![]() |
261620fd13 | ||
![]() |
9af6effad9 | ||
![]() |
92602916dd | ||
![]() |
6c7b90e1c3 | ||
![]() |
d8e57144f7 | ||
![]() |
e69c45a246 | ||
![]() |
3f87a6d714 | ||
![]() |
841124b1f3 | ||
![]() |
257a878ef4 | ||
![]() |
07d3f82912 | ||
![]() |
7d3eb4f5a6 | ||
![]() |
0efcc55373 | ||
![]() |
eafceb8a6c | ||
![]() |
5ac8318e71 | ||
![]() |
7981f9dc91 | ||
![]() |
a9bf04cbc6 | ||
![]() |
4b5591d884 | ||
![]() |
c08197f025 | ||
![]() |
5d22eabfa3 | ||
![]() |
c3a38e384a | ||
![]() |
0d3dd94559 | ||
![]() |
9cdaa37519 | ||
![]() |
202a319d69 | ||
![]() |
198384b2ed | ||
![]() |
9d8b070f1e | ||
![]() |
28ba2d5008 | ||
![]() |
37ddd63d27 | ||
![]() |
d7f8b8c1e0 | ||
![]() |
9e70c5bbea | ||
![]() |
4dd572063e | ||
![]() |
db9cf95648 | ||
![]() |
df6bae4712 | ||
![]() |
56cb8209b8 | ||
![]() |
aba9c2e113 | ||
![]() |
3d25008739 | ||
![]() |
9437f057d0 | ||
![]() |
917d6089a7 | ||
![]() |
f19e99e3ad | ||
![]() |
2d0fb05fa6 | ||
![]() |
c617f2dfad | ||
![]() |
747c5fc89a | ||
![]() |
4f662ef203 | ||
![]() |
eb274ad88f | ||
![]() |
bd7b41be9b | ||
![]() |
076720bfff | ||
![]() |
2aadae407e | ||
![]() |
b16bb07774 | ||
![]() |
81dd083388 | ||
![]() |
92d4cef1e2 | ||
![]() |
d2cd79dcf5 | ||
![]() |
0ef8b391e4 | ||
![]() |
0fbd105977 | ||
![]() |
5c23af541e | ||
![]() |
db021ff78b | ||
![]() |
31294c6f54 | ||
![]() |
03d3495759 | ||
![]() |
4644e105a8 | ||
![]() |
af5002508d | ||
![]() |
9416d2b9c7 | ||
![]() |
f60f5d5a6c | ||
![]() |
26d00f87a8 | ||
![]() |
6c33ea423c | ||
![]() |
ad354aca4e | ||
![]() |
ce6a5ebf0a | ||
![]() |
d66ef38e8d | ||
![]() |
1b6c49f621 | ||
![]() |
0262c83815 | ||
![]() |
c212934130 | ||
![]() |
ac30e47e59 | ||
![]() |
d640057453 | ||
![]() |
0cea136d4d | ||
![]() |
9b3d93e734 | ||
![]() |
ca3782ac62 | ||
![]() |
dc9ffb7fd5 | ||
![]() |
b21b3a55fe | ||
![]() |
e62bd85a7a | ||
![]() |
fed221b008 | ||
![]() |
935298d159 | ||
![]() |
909919250c | ||
![]() |
d24f278f7c | ||
![]() |
ca408a495c | ||
![]() |
ee99719137 | ||
![]() |
5fb0729eae | ||
![]() |
a1db3187cd | ||
![]() |
488a3cf1ff | ||
![]() |
1253a620a4 | ||
![]() |
64547fc4a7 | ||
![]() |
d643d140cf | ||
![]() |
222e2e3242 | ||
![]() |
ea7b6daf26 | ||
![]() |
14cfab495d | ||
![]() |
ed4b4a1a3c | ||
![]() |
511c552188 | ||
![]() |
8c650219eb | ||
![]() |
189b779abe | ||
![]() |
c26074b184 | ||
![]() |
6c70779ff1 | ||
![]() |
73897481ac | ||
![]() |
a62c523ed5 | ||
![]() |
0f46f57b52 | ||
![]() |
bc4d65a20c | ||
![]() |
f1e9a44ad9 | ||
![]() |
61972c4719 | ||
![]() |
9183c9c220 | ||
![]() |
1ce59f13ff | ||
![]() |
1bac5db6d5 | ||
![]() |
58f2c4d8e6 | ||
![]() |
2c2c61b2fc | ||
![]() |
98b04a4a83 | ||
![]() |
3ac64c44f4 | ||
![]() |
3375847302 | ||
![]() |
14043c86f5 | ||
![]() |
030117780a | ||
![]() |
971c9fe5a1 | ||
![]() |
77c6d3d576 | ||
![]() |
6edbfe2a6f | ||
![]() |
d0a3125df4 | ||
![]() |
d8c76d4c21 | ||
![]() |
a1cc0897df | ||
![]() |
e88a90f242 | ||
![]() |
1ae54f6f8c | ||
![]() |
8a4cb484fa | ||
![]() |
93defbf341 | ||
![]() |
ca1089da9a | ||
![]() |
dd07c7db0e | ||
![]() |
dfe932e37a | ||
![]() |
568cf9c259 | ||
![]() |
b4ef20e785 | ||
![]() |
3a56dabf42 | ||
![]() |
3ac4899b96 | ||
![]() |
b7b228d9ce | ||
![]() |
45339fd6d2 | ||
![]() |
1856a5a82f | ||
![]() |
b2aa703b62 | ||
![]() |
b31897d65d | ||
![]() |
a92f776ebe | ||
![]() |
b84ed675f5 | ||
![]() |
21eb98a52c | ||
![]() |
75dd8d492b | ||
![]() |
338893ded4 | ||
![]() |
b459900040 | ||
![]() |
d31300f4f9 | ||
![]() |
f2285c0b19 | ||
![]() |
684cb81974 | ||
![]() |
9db272f30e | ||
![]() |
c4a5e8dc86 | ||
![]() |
df4dd0122f | ||
![]() |
8fed18b2ac | ||
![]() |
ecabbb57e6 | ||
![]() |
8d1d4092aa | ||
![]() |
6185c4ddcf | ||
![]() |
3ccbbccd10 | ||
![]() |
30e0ccc77b | ||
![]() |
50f7b72b09 | ||
![]() |
a5828c7949 | ||
![]() |
22a5e72470 | ||
![]() |
5d14dca818 | ||
![]() |
f97cb821f8 | ||
![]() |
d02dce5562 | ||
![]() |
c6bd42843b | ||
![]() |
43e4fbfcd0 | ||
![]() |
ac8430cbba | ||
![]() |
86d619be30 | ||
![]() |
fec7672598 | ||
![]() |
44d2744a8c | ||
![]() |
22bfbe96c8 | ||
![]() |
c5d1271894 | ||
![]() |
050d3058f9 | ||
![]() |
2ae99afa21 | ||
![]() |
b094c9190f | ||
![]() |
1a9e1e6f7c | ||
![]() |
8c94926693 | ||
![]() |
880a176e65 | ||
![]() |
8de1b5f3d9 | ||
![]() |
178f4546fc | ||
![]() |
7564a49344 | ||
![]() |
6c5c42c2b5 | ||
![]() |
b1653b359e | ||
![]() |
11098afab5 | ||
![]() |
a64051e0f1 | ||
![]() |
db8ac4a9ae | ||
![]() |
890474a635 | ||
![]() |
c7941a85ed | ||
![]() |
fab5f26e09 | ||
![]() |
a39b10eee9 | ||
![]() |
97b933a990 | ||
![]() |
4926e90514 | ||
![]() |
6dca975844 | ||
![]() |
e31743770a | ||
![]() |
d8aab62d75 | ||
![]() |
f90603a18f | ||
![]() |
5fce9facbe | ||
![]() |
32b48d5cdb | ||
![]() |
3e4f0d682b | ||
![]() |
98449ddfe0 | ||
![]() |
26040d4e9f | ||
![]() |
83cea5e1ee | ||
![]() |
3e73a5dbd3 | ||
![]() |
dd424a4cb3 | ||
![]() |
0c79f5cce3 | ||
![]() |
794c3703e5 | ||
![]() |
b674006fcc | ||
![]() |
559c397b2f | ||
![]() |
646698f1ed | ||
![]() |
c9b938ae55 | ||
![]() |
990c220fa0 | ||
![]() |
38641d4edf | ||
![]() |
450072bd23 | ||
![]() |
25eb93fae0 | ||
![]() |
bcf6f60571 | ||
![]() |
2076b8f1d7 | ||
![]() |
f19cfb75e6 | ||
![]() |
ceaacc771d | ||
![]() |
48067e3285 | ||
![]() |
003855a94c | ||
![]() |
3599ab3caf | ||
![]() |
e7a26b436d | ||
![]() |
ab94bc18dc | ||
![]() |
36e91ea155 | ||
![]() |
6035be9ce6 | ||
![]() |
67499bdc65 | ||
![]() |
222c8fdb62 | ||
![]() |
9d648bad51 | ||
![]() |
1a62b9a161 | ||
![]() |
33021e8fe0 | ||
![]() |
5a2ae4c3e3 | ||
![]() |
9eaef84cc8 | ||
![]() |
fcc4d655f5 | ||
![]() |
8d0ac4f5f0 | ||
![]() |
d56cf003f8 | ||
![]() |
34d0c0b1ba | ||
![]() |
9bbbffbe7a | ||
![]() |
ccd0f7d9cc | ||
![]() |
ccb9bceecc | ||
![]() |
c1a67ff1f8 | ||
![]() |
bfda8dcc02 | ||
![]() |
8746e7c9ad | ||
![]() |
e2aa36d083 | ||
![]() |
ff90f257cc | ||
![]() |
9b84046865 | ||
![]() |
51434a39f8 | ||
![]() |
3e2031be7c | ||
![]() |
ea4e8805b7 | ||
![]() |
f62bfeac08 | ||
![]() |
b9042c37f9 | ||
![]() |
49a58dcab1 | ||
![]() |
e2acbeddc2 | ||
![]() |
449b17d830 | ||
![]() |
7ed460ce02 | ||
![]() |
9f4a7e664f | ||
![]() |
38eb75b13f | ||
![]() |
0ad56874b4 | ||
![]() |
c1168693fa | ||
![]() |
feb8c27f1f | ||
![]() |
c20ebd66e5 | ||
![]() |
4d78d530dc | ||
![]() |
22b20c15de | ||
![]() |
c69b224c65 | ||
![]() |
805d328d6c | ||
![]() |
edcb692f78 | ||
![]() |
f33586f062 | ||
![]() |
b8293f134d | ||
![]() |
7e9bcff0f3 | ||
![]() |
eba3b32708 | ||
![]() |
d61bf26e17 | ||
![]() |
2e662b5745 | ||
![]() |
9714fa369b | ||
![]() |
0e8acba08e | ||
![]() |
a18e588e55 | ||
![]() |
bf55e3c0cc | ||
![]() |
486d114e64 | ||
![]() |
ac5c060a98 | ||
![]() |
86a9d197cb | ||
![]() |
e911dbb9d4 | ||
![]() |
e26d123f67 | ||
![]() |
70aac81900 | ||
![]() |
28e78d98f6 | ||
![]() |
fabc5ae032 | ||
![]() |
e40882455d | ||
![]() |
f1483e8c8e | ||
![]() |
bd40e2c3ff | ||
![]() |
9479498713 | ||
![]() |
6bd8523ec2 | ||
![]() |
22e3cddd91 | ||
![]() |
c864b15c34 | ||
![]() |
069654c5f9 | ||
![]() |
f3d4d4747a | ||
![]() |
5bbb0cd666 | ||
![]() |
7b5ba3bdc2 | ||
![]() |
5ac17ee6b8 | ||
![]() |
d647555e3a | ||
![]() |
26e22f97ee | ||
![]() |
84976a65e0 | ||
![]() |
fef9d541ed | ||
![]() |
ad5535af81 | ||
![]() |
cdcfb4ffce | ||
![]() |
5403ac8893 | ||
![]() |
9b26457781 | ||
![]() |
939cc56951 | ||
![]() |
23309e6fdf | ||
![]() |
3f60c961d9 | ||
![]() |
9e7f07e196 | ||
![]() |
eda4439ee8 | ||
![]() |
ad02558ade | ||
![]() |
0d1901cfe5 | ||
![]() |
ce493d1ae2 | ||
![]() |
09312ecd1d | ||
![]() |
2f274d5f52 | ||
![]() |
6dbf226365 | ||
![]() |
6401b5f54d | ||
![]() |
ac2dd81d39 | ||
![]() |
6784522195 | ||
![]() |
f42d077f30 | ||
![]() |
eaca47ebc6 | ||
![]() |
4d57223847 | ||
![]() |
d9ab96236b | ||
![]() |
44b2e62eef | ||
![]() |
7d5d4df761 | ||
![]() |
da7716db60 | ||
![]() |
862c5d342b | ||
![]() |
6167d5dbfc | ||
![]() |
2856fb029f | ||
![]() |
8fb945312a | ||
![]() |
25d6806ebd | ||
![]() |
51070d9afd | ||
![]() |
1048caa496 | ||
![]() |
0cd7ac05aa | ||
![]() |
2793c42d91 | ||
![]() |
2eaa7288e4 | ||
![]() |
f9341bea79 | ||
![]() |
0df8d13020 | ||
![]() |
d27622de1e | ||
![]() |
47c3da131c | ||
![]() |
3a608bb582 | ||
![]() |
64f9228ee3 | ||
![]() |
c926482b3c | ||
![]() |
9562972c42 | ||
![]() |
b462b7fcc4 | ||
![]() |
b34f9d7fd3 | ||
![]() |
57732c3e5f | ||
![]() |
eb1f56488f | ||
![]() |
5825843f68 | ||
![]() |
dd56a5d869 | ||
![]() |
86f82c0e61 | ||
![]() |
8ae45240b2 | ||
![]() |
6d845af7f1 | ||
![]() |
733ebb8caf | ||
![]() |
695a194467 | ||
![]() |
8a56257ade | ||
![]() |
45fea983b9 | ||
![]() |
642b499e70 | ||
![]() |
5c5575bb72 | ||
![]() |
4d5dc0d39c | ||
![]() |
fba0a8036b | ||
![]() |
d82274f5d4 | ||
![]() |
3cf81230b2 | ||
![]() |
17bab456e4 | ||
![]() |
e1cc006db7 | ||
![]() |
7e95dd3c76 | ||
![]() |
ff3ce46a26 | ||
![]() |
4d61c2c5e0 | ||
![]() |
8589ee14ff | ||
![]() |
1727387c9c | ||
![]() |
e1146e4655 | ||
![]() |
afaba7ccdf | ||
![]() |
ccb3ceae4f | ||
![]() |
d0d93f6d2d | ||
![]() |
67f091c731 | ||
![]() |
c29b0645bd | ||
![]() |
96dac0f979 | ||
![]() |
fda9b59129 | ||
![]() |
fa3aebb7b1 | ||
![]() |
988251deb6 | ||
![]() |
d99a389c49 | ||
![]() |
7508f9d3bb | ||
![]() |
b2512d7aee | ||
![]() |
eb74e1e3b2 | ||
![]() |
bd91e82b17 | ||
![]() |
bae60acfe7 | ||
![]() |
426cefe8ee | ||
![]() |
a36fe3ef21 | ||
![]() |
ee0756c7c0 | ||
![]() |
44b96121e4 | ||
![]() |
8add777b34 | ||
![]() |
6932b15144 | ||
![]() |
c09edda797 | ||
![]() |
0e86010565 | ||
![]() |
fa5896ee5b | ||
![]() |
6fe3fdce11 | ||
![]() |
90769a12b1 | ||
![]() |
3c6d27b504 | ||
![]() |
6ef25eb861 | ||
![]() |
ec28e972bb | ||
![]() |
d1a9033525 | ||
![]() |
2d5bc3ada8 | ||
![]() |
506ffb9885 | ||
![]() |
8ef702fa07 | ||
![]() |
1d49b725b6 | ||
![]() |
204feb97ce | ||
![]() |
e12389a748 | ||
![]() |
9fc38b5bb8 | ||
![]() |
907e5a10d6 | ||
![]() |
670591e221 | ||
![]() |
8006a7d578 | ||
![]() |
8b9078b82e | ||
![]() |
56d4edd3ae | ||
![]() |
a7f5ea865e | ||
![]() |
2a62144abd | ||
![]() |
c7ea7a4f65 | ||
![]() |
7ea090ba8d | ||
![]() |
9294e353d3 | ||
![]() |
71c50c58ee | ||
![]() |
e322299fda | ||
![]() |
06602a568a | ||
![]() |
54ac5e8940 | ||
![]() |
e2341363d4 | ||
![]() |
a0f0529ad8 | ||
![]() |
c371d863d9 | ||
![]() |
16d8c75953 | ||
![]() |
fa48c0e76b | ||
![]() |
2d17cfaf2b | ||
![]() |
f379de0a3a | ||
![]() |
d3fcb0aa6a | ||
![]() |
046740f10b | ||
![]() |
ffae2eda83 | ||
![]() |
8334b8fe46 | ||
![]() |
c6c6e58d28 | ||
![]() |
2ff0d6665c | ||
![]() |
dc6108c970 | ||
![]() |
6f52f9b4af | ||
![]() |
b6eb896f0b | ||
![]() |
adf861c36d | ||
![]() |
4e1bcdf209 | ||
![]() |
d107fe19f7 | ||
![]() |
fa03ff51d4 | ||
![]() |
f6c6536b48 | ||
![]() |
38e4249182 | ||
![]() |
2dd8c0beee | ||
![]() |
6a35494ef1 | ||
![]() |
bf1c42e085 | ||
![]() |
f1aa3d8c90 | ||
![]() |
0ef6bf8067 | ||
![]() |
9d63e2ae97 | ||
![]() |
fdd8060296 | ||
![]() |
ebbb23ffbb | ||
![]() |
0e413109bd | ||
![]() |
f7e29c1e00 | ||
![]() |
42f777d49a | ||
![]() |
efb8ea4c25 | ||
![]() |
52bf5690c0 | ||
![]() |
607dff9263 | ||
![]() |
0510db75fb | ||
![]() |
cf3e53eb71 | ||
![]() |
b8865e925d | ||
![]() |
2e9a860aaa | ||
![]() |
859eecabc0 | ||
![]() |
7a0253631b | ||
![]() |
c12a60c1ad | ||
![]() |
df9dbd6130 | ||
![]() |
a98e745116 | ||
![]() |
61f1b10a06 | ||
![]() |
4bcb2a5c9d | ||
![]() |
7e48648f9e | ||
![]() |
e4bef056e6 | ||
![]() |
bcc97d1aa7 | ||
![]() |
49f9d36718 | ||
![]() |
f9cb258a5e | ||
![]() |
803b8eab28 | ||
![]() |
08bbfc50ee | ||
![]() |
97381a4908 | ||
![]() |
4e6722f201 | ||
![]() |
a29e2116a7 | ||
![]() |
7a15acc546 | ||
![]() |
0265de2137 | ||
![]() |
595b9910f5 | ||
![]() |
f4d4d2cc35 | ||
![]() |
b4fb4d7dcf | ||
![]() |
1a375fb523 | ||
![]() |
658c0ff5c4 | ||
![]() |
515be677a9 | ||
![]() |
d694c5f511 | ||
![]() |
66c753f3a3 | ||
![]() |
7047b62442 | ||
![]() |
6092f06d46 | ||
![]() |
4671b956b3 | ||
![]() |
552ba6999a | ||
![]() |
b86bc4455f | ||
![]() |
b6d36ee117 | ||
![]() |
5cc73555cd | ||
![]() |
219922cd82 | ||
![]() |
5d57f864dc | ||
![]() |
c6ee2816db | ||
![]() |
0aa898b13f | ||
![]() |
b22e82c8ce | ||
![]() |
63dee1e1ac | ||
![]() |
91c87ac301 | ||
![]() |
7124d9bca5 | ||
![]() |
08127e5806 | ||
![]() |
6417bd91ef | ||
![]() |
cde5f7d12e | ||
![]() |
395c9587b6 | ||
![]() |
17197ad670 | ||
![]() |
42371a6e8d | ||
![]() |
d7aae849f6 | ||
![]() |
4fe9413662 | ||
![]() |
37bf51cebf | ||
![]() |
eee9cb8b87 | ||
![]() |
1724bca5f0 | ||
![]() |
ae6581fb55 | ||
![]() |
0c632307ea | ||
![]() |
479887eb84 | ||
![]() |
af280a7343 | ||
![]() |
802b26e870 | ||
![]() |
0ab86937d2 | ||
![]() |
3ab06bf383 | ||
![]() |
5db0cc5241 | ||
![]() |
a588ec084b | ||
![]() |
5660b5ddc6 | ||
![]() |
27fbe69033 | ||
![]() |
1672ecb892 | ||
![]() |
784e01347c | ||
![]() |
5cc21a831b | ||
![]() |
dcb9380b50 | ||
![]() |
b1957773bb | ||
![]() |
c4f4583f20 | ||
![]() |
440cdbdb23 | ||
![]() |
58bfde33a9 | ||
![]() |
7c33e49ef6 | ||
![]() |
3bf318e2b7 | ||
![]() |
f56193ac18 | ||
![]() |
31efa7a8f8 | ||
![]() |
372932abaf | ||
![]() |
4c6dd2cdf2 | ||
![]() |
6d04e39151 | ||
![]() |
afa257e79a | ||
![]() |
05f8ee9747 | ||
![]() |
818ae03928 | ||
![]() |
97ad3c1e6d | ||
![]() |
97555645f8 | ||
![]() |
612228bb73 | ||
![]() |
e350f43adc | ||
![]() |
2257f3484d | ||
![]() |
6e75d41956 | ||
![]() |
9883a38698 | ||
![]() |
07256e2e34 | ||
![]() |
43674ae80a | ||
![]() |
f4ba8df02b | ||
![]() |
c066ebd76f | ||
![]() |
6e382c64a4 | ||
![]() |
99ebf03876 | ||
![]() |
995d79e373 | ||
![]() |
0cd153ab61 | ||
![]() |
81e76f260c | ||
![]() |
b24baa68ba | ||
![]() |
d4c1b8d321 | ||
![]() |
1e53d6bfab | ||
![]() |
5931cd6af7 | ||
![]() |
a1be03543c | ||
![]() |
b1a5547de2 | ||
![]() |
93571961ee | ||
![]() |
ee4942dfd7 | ||
![]() |
cb24347b23 | ||
![]() |
146b7be825 | ||
![]() |
50203c5f87 | ||
![]() |
b188073fb0 | ||
![]() |
8aef24be1e | ||
![]() |
fb25f6c7ac | ||
![]() |
fbd983217d | ||
![]() |
ce21fe2087 | ||
![]() |
17cd395712 | ||
![]() |
bfe9de05cd | ||
![]() |
91c60df0e9 | ||
![]() |
ad065e9281 | ||
![]() |
b1429366da | ||
![]() |
8bf7af2e74 | ||
![]() |
2003f51d49 | ||
![]() |
ce83fd9a10 | ||
![]() |
eacbaa3680 | ||
![]() |
98c65fb9b7 | ||
![]() |
44a71d8565 | ||
![]() |
badd4d3207 | ||
![]() |
0f517b803b | ||
![]() |
c2d11e786f | ||
![]() |
b0efe49e29 | ||
![]() |
2d029b9f76 | ||
![]() |
8ad917cff0 | ||
![]() |
4e478c65d3 | ||
![]() |
a469086915 | ||
![]() |
bf05ff6048 | ||
![]() |
a817d8cbf9 | ||
![]() |
4a19c78fa5 | ||
![]() |
e8bb7da906 | ||
![]() |
523477fc2b | ||
![]() |
c730426be0 | ||
![]() |
57d6c97203 | ||
![]() |
fce17aa1d4 | ||
![]() |
01abc244b1 | ||
![]() |
7bedacf5ad | ||
![]() |
552a1d0464 | ||
![]() |
8dde25532a | ||
![]() |
f29fa939ab | ||
![]() |
614bdb33b4 | ||
![]() |
71761675cf | ||
![]() |
d9194aa859 | ||
![]() |
f15081a474 | ||
![]() |
2f99ff4a0c | ||
![]() |
3a7d26aa46 | ||
![]() |
3f35bc593c | ||
![]() |
e5e708d781 | ||
![]() |
d694561980 | ||
![]() |
8d6d18e875 | ||
![]() |
072e27ed27 | ||
![]() |
6d64215614 | ||
![]() |
33f5ed5b14 | ||
![]() |
27f509c8e0 | ||
![]() |
890b3e13c9 | ||
![]() |
b730cb099f | ||
![]() |
fc94f184d2 | ||
![]() |
cbf6540889 | ||
![]() |
40804a7fb3 | ||
![]() |
d4101c4f43 | ||
![]() |
409bebd5bc | ||
![]() |
8e3ad69adb | ||
![]() |
c8e46d9e21 | ||
![]() |
c56241ffc1 | ||
![]() |
be62a2bfc5 | ||
![]() |
5cb7771484 | ||
![]() |
6675d3e2cd | ||
![]() |
8ecbe4c8ad | ||
![]() |
edb75c4bab | ||
![]() |
54b21c716a | ||
![]() |
4704274b87 | ||
![]() |
78547aa119 | ||
![]() |
3887231c73 | ||
![]() |
8a29cfbb7e | ||
![]() |
a01d6eaf72 | ||
![]() |
69fc571b56 | ||
![]() |
4cfd9c322b | ||
![]() |
4326354ca6 | ||
![]() |
4208c852e1 | ||
![]() |
7330b4532e | ||
![]() |
1e0f6f9e41 | ||
![]() |
216e2367c6 | ||
![]() |
d2dce8801b | ||
![]() |
5b8bb9f678 | ||
![]() |
2076f146cf | ||
![]() |
910c10f554 | ||
![]() |
04e974b326 | ||
![]() |
e7abeb5ad9 | ||
![]() |
f4416fe007 | ||
![]() |
510591ef0f | ||
![]() |
a5e89d1dd1 | ||
![]() |
5d4f2b7862 | ||
![]() |
e3815e40d2 | ||
![]() |
6dccfb4774 | ||
![]() |
1ccc1f4c1a | ||
![]() |
042809620a | ||
![]() |
cb4c8abd94 | ||
![]() |
e86302f5b9 | ||
![]() |
de080d5811 | ||
![]() |
21b7045f93 | ||
![]() |
627301f83d | ||
![]() |
607ca84fcc | ||
![]() |
a7f36248d0 | ||
![]() |
d008d15167 | ||
![]() |
607dc436bd | ||
![]() |
4384948f6c | ||
![]() |
5e05e9ec93 | ||
![]() |
ac1fe66cf9 | ||
![]() |
2a18eacf62 | ||
![]() |
af42e32ae6 | ||
![]() |
12b93d6637 | ||
![]() |
930c971035 | ||
![]() |
06f20c66f8 | ||
![]() |
7c875a8541 | ||
![]() |
f85e19c75d | ||
![]() |
5e2aa51627 | ||
![]() |
75a44fb30a | ||
![]() |
86732b6ae4 | ||
![]() |
0713f55e9c | ||
![]() |
5c32d73409 | ||
![]() |
5e13a1735d | ||
![]() |
6a1fbb00d9 | ||
![]() |
20c3badfac | ||
![]() |
7817cfe0c1 | ||
![]() |
9ed823b5a5 | ||
![]() |
c6a5dedf0a | ||
![]() |
01c9ab36b7 | ||
![]() |
27f5bdeef1 | ||
![]() |
723898f87d | ||
![]() |
bc05cc1445 | ||
![]() |
dcf4e43e28 | ||
![]() |
3868c53908 | ||
![]() |
a71c693ca3 | ||
![]() |
691f93f01c | ||
![]() |
4cff749186 | ||
![]() |
e1ac1547fd | ||
![]() |
e53bd505fb | ||
![]() |
cfa926542e | ||
![]() |
79097eca47 | ||
![]() |
d1741e40e3 | ||
![]() |
5d0528d195 | ||
![]() |
a9ea06f753 | ||
![]() |
62e121c12c | ||
![]() |
02ef05160f | ||
![]() |
6e66c013c0 | ||
![]() |
dcb11f01e1 | ||
![]() |
8a0e4b577c | ||
![]() |
b57f420261 | ||
![]() |
7d1790abe3 | ||
![]() |
1fc494571b | ||
![]() |
e6d97bc773 | ||
![]() |
0e53323fb7 | ||
![]() |
eb4764d2b2 | ||
![]() |
298a91adbf | ||
![]() |
2cb9912039 | ||
![]() |
e52bfe4335 | ||
![]() |
761a249e05 | ||
![]() |
c42df3a0c2 | ||
![]() |
3d359b7a98 | ||
![]() |
deef6417ad | ||
![]() |
f55a8deb97 | ||
![]() |
333506e00b | ||
![]() |
bbc1642b90 | ||
![]() |
8209eda27a | ||
![]() |
b13f7a599b | ||
![]() |
cb0f700be1 | ||
![]() |
7b6d6b466a | ||
![]() |
76f97e5c2e | ||
![]() |
4669a1ab57 | ||
![]() |
b1ad0edbe1 | ||
![]() |
51a695d047 | ||
![]() |
396e2d14f3 | ||
![]() |
245479c339 | ||
![]() |
092215f47a | ||
![]() |
fcb46db718 | ||
![]() |
60c58c8b9c | ||
![]() |
124a2839b5 | ||
![]() |
ededfe10ab | ||
![]() |
81895c20d6 | ||
![]() |
46fabe065c | ||
![]() |
7ac338756a | ||
![]() |
640b8edd78 | ||
![]() |
c5d98752fa | ||
![]() |
a6a5bef447 | ||
![]() |
042885c155 | ||
![]() |
cbb9dcf7d0 | ||
![]() |
9b080800e1 | ||
![]() |
6effbf50a8 | ||
![]() |
398f9aa19a | ||
![]() |
935d89747f | ||
![]() |
21c2fbfd39 | ||
![]() |
bd337f3aac | ||
![]() |
4a673eee81 | ||
![]() |
cebf349f9a | ||
![]() |
3683deb51c | ||
![]() |
99ee076db9 | ||
![]() |
04f759041f | ||
![]() |
75f89059e7 | ||
![]() |
a81f31156d | ||
![]() |
f706452e67 | ||
![]() |
b0126afbcf | ||
![]() |
5b8393ff89 | ||
![]() |
d2235da06a | ||
![]() |
a87f6a0791 | ||
![]() |
7e7cfb79a4 | ||
![]() |
fb43a5265c | ||
![]() |
439a814133 | ||
![]() |
d9dfcc04bf | ||
![]() |
0cfac137b7 | ||
![]() |
4575ee805a | ||
![]() |
67f70ce2cc | ||
![]() |
2f641ffb13 | ||
![]() |
50f92269c2 | ||
![]() |
f220f397ae | ||
![]() |
cdb4096124 | ||
![]() |
01938af65b | ||
![]() |
47a1fca32f | ||
![]() |
321342cf6d | ||
![]() |
086e9beb59 | ||
![]() |
3519d4b219 | ||
![]() |
9034b9a9ae | ||
![]() |
90ba8440a0 | ||
![]() |
4988b37d6f | ||
![]() |
ad4799ee60 | ||
![]() |
35229f8ae5 | ||
![]() |
7a011d9e75 | ||
![]() |
3d86835979 | ||
![]() |
4073306538 | ||
![]() |
b86aa28d6a | ||
![]() |
d43cee29f2 | ||
![]() |
acad468b4a | ||
![]() |
4cbe842cfa | ||
![]() |
d93e227190 | ||
![]() |
a9cf424998 | ||
![]() |
6d8fdf46d5 | ||
![]() |
781f98ef91 | ||
![]() |
1313f685da | ||
![]() |
4779a993d3 | ||
![]() |
342b2ae5dc | ||
![]() |
ce8ae40206 | ||
![]() |
d0704f621f | ||
![]() |
c6da4043ed | ||
![]() |
1aa3761d1a | ||
![]() |
ff769caf82 | ||
![]() |
be6bc68b56 | ||
![]() |
feb3d11f63 | ||
![]() |
d2f9b063b2 | ||
![]() |
315089c361 | ||
![]() |
e95df0dbd5 | ||
![]() |
8fed029ee3 | ||
![]() |
522daf5aff | ||
![]() |
e948eebe84 | ||
![]() |
783d4e7e8a | ||
![]() |
1ce2198621 | ||
![]() |
f521def4a5 | ||
![]() |
75202921a1 | ||
![]() |
4ef8b93344 | ||
![]() |
881b191b8d | ||
![]() |
b39e071d1e | ||
![]() |
84cb3a1060 | ||
![]() |
f4ea3980c2 | ||
![]() |
ef73720a5e | ||
![]() |
d8cdc57702 | ||
![]() |
b52dee37f4 | ||
![]() |
21bd9f09da | ||
![]() |
44f24e58f6 | ||
![]() |
860c4d045a | ||
![]() |
3b0c96f654 | ||
![]() |
65b744472b | ||
![]() |
26489b0f00 | ||
![]() |
414abad05f | ||
![]() |
735d9a5391 | ||
![]() |
50571449cb | ||
![]() |
a31aacd115 | ||
![]() |
f5b57cc0da | ||
![]() |
b7006a8f2c | ||
![]() |
82bb467a2a | ||
![]() |
3a44e92432 | ||
![]() |
e60db5f928 | ||
![]() |
78485287a4 | ||
![]() |
48b6f01b13 | ||
![]() |
39e04de208 | ||
![]() |
68d5b59693 | ||
![]() |
4ef01ef745 | ||
![]() |
573fa8870c | ||
![]() |
88d354b08b | ||
![]() |
0ff65b5496 | ||
![]() |
14e0dcb085 | ||
![]() |
e008fd21a4 | ||
![]() |
4638149ad0 | ||
![]() |
e386bdd6b3 | ||
![]() |
decb167ba9 | ||
![]() |
dd9557c13e | ||
![]() |
708d7162fb | ||
![]() |
1ee1b522f1 | ||
![]() |
d5a500c037 | ||
![]() |
3e02c65bc0 | ||
![]() |
d10f9a5f25 | ||
![]() |
17e7214d25 | ||
![]() |
22774592db | ||
![]() |
7c2aa6e69d | ||
![]() |
d1dbcda88e | ||
![]() |
3e05508cf9 | ||
![]() |
1acdefd358 | ||
![]() |
23143797f9 | ||
![]() |
3364bf268b | ||
![]() |
2025d6305e | ||
![]() |
639ad7698d | ||
![]() |
4eaff51ba2 | ||
![]() |
1fb30bc3d9 | ||
![]() |
6b66f40bcb | ||
![]() |
a3cd531cc8 | ||
![]() |
55c84192be | ||
![]() |
aa9018b97b | ||
![]() |
1e79c146a7 | ||
![]() |
cb8545e33d | ||
![]() |
4a484c535b | ||
![]() |
ee417e41ea | ||
![]() |
3cdc1fcaee | ||
![]() |
0a023a61f9 | ||
![]() |
73f81c5b52 | ||
![]() |
bc4acbb7e1 | ||
![]() |
26d07ea2c6 | ||
![]() |
5380c8352d | ||
![]() |
e85e91183e | ||
![]() |
b48c251b36 | ||
![]() |
181a14ce59 | ||
![]() |
a208a22bc2 | ||
![]() |
b9ea7ce066 | ||
![]() |
f2f275512d | ||
![]() |
5150c2ee62 | ||
![]() |
644766b5a6 | ||
![]() |
ca679f5932 | ||
![]() |
7f7145e8de | ||
![]() |
aa1878c15a | ||
![]() |
e7d23176b7 | ||
![]() |
31218c2a8c | ||
![]() |
06374c82fd | ||
![]() |
8efcc8f80f | ||
![]() |
2d6317bd24 | ||
![]() |
157b064214 | ||
![]() |
0ece4851d2 | ||
![]() |
f1f5996975 | ||
![]() |
0a2dbc4688 | ||
![]() |
13587d7ab3 | ||
![]() |
0fcef064fb | ||
![]() |
19b8796cbc | ||
![]() |
15fb60a845 | ||
![]() |
5c202f04e7 | ||
![]() |
bc6fdf81d2 | ||
![]() |
3194a2bf2c | ||
![]() |
72d1e5131f | ||
![]() |
7721098551 | ||
![]() |
0b7593ad28 | ||
![]() |
a68823491c | ||
![]() |
f563bc4210 | ||
![]() |
43e7be9b86 | ||
![]() |
27131d15dd | ||
![]() |
fb1a290bd9 | ||
![]() |
ef16145695 | ||
![]() |
4fbd1182c2 | ||
![]() |
2d39e65b5c | ||
![]() |
8e96b675fa | ||
![]() |
adb6943420 | ||
![]() |
eae7babf93 | ||
![]() |
7d5e18c05b | ||
![]() |
cbe001efd6 | ||
![]() |
86b783fb0f | ||
![]() |
ccc27b48df | ||
![]() |
a32391f560 | ||
![]() |
aed0348802 | ||
![]() |
1b66446c0d | ||
![]() |
2bc0c8a483 | ||
![]() |
3f7e02e305 | ||
![]() |
dd0d666003 | ||
![]() |
81859a37de | ||
![]() |
180bb581a3 | ||
![]() |
2f31779af4 | ||
![]() |
ee6d512165 | ||
![]() |
1470fdc057 | ||
![]() |
93bbaf187e | ||
![]() |
f1e43007f1 | ||
![]() |
155436b85d | ||
![]() |
f3e029c3f6 | ||
![]() |
90d6416f55 | ||
![]() |
af2a2e45af | ||
![]() |
e79eda9c5c | ||
![]() |
b338d9dbcf | ||
![]() |
7fb9345344 | ||
![]() |
77b488568b | ||
![]() |
fcf650c6eb | ||
![]() |
0c21023ad8 | ||
![]() |
8f35a56ec8 | ||
![]() |
95ba1873e4 | ||
![]() |
8b8652d44c | ||
![]() |
2515b8167f | ||
![]() |
09dd044f3d | ||
![]() |
ace0ed9667 | ||
![]() |
cf03708da2 | ||
![]() |
d4670bf6fa | ||
![]() |
feea448a24 | ||
![]() |
71ad54652b | ||
![]() |
8e1deda7b0 | ||
![]() |
752f985e13 | ||
![]() |
89e3219e06 | ||
![]() |
429dddc6c9 | ||
![]() |
981174a490 | ||
![]() |
201f7e9848 | ||
![]() |
bb0d8ad58a | ||
![]() |
29e64f7c1a | ||
![]() |
4819ebd56e | ||
![]() |
3b603b0637 | ||
![]() |
baa63249d1 | ||
![]() |
ee347e3081 | ||
![]() |
f96bc95053 | ||
![]() |
e1df4757e4 | ||
![]() |
4fc37a7321 | ||
![]() |
2a45a13f73 | ||
![]() |
067b15c300 | ||
![]() |
8a1c283542 | ||
![]() |
93d1e8b2ff | ||
![]() |
c60d5b54fa | ||
![]() |
ef180f082e | ||
![]() |
f52741cc37 | ||
![]() |
2a2661f066 | ||
![]() |
b521903138 | ||
![]() |
acbd699d95 | ||
![]() |
6b8928becb | ||
![]() |
e393bdb1e5 | ||
![]() |
bba6b96765 | ||
![]() |
740116356c | ||
![]() |
2f6e4fa4a3 | ||
![]() |
fb3f6721b2 | ||
![]() |
4e7bd21e5c | ||
![]() |
219c2030b9 | ||
![]() |
b75fdb4566 | ||
![]() |
4584b14a31 | ||
![]() |
814ddb5932 | ||
![]() |
6ea0f6290a | ||
![]() |
c796fe1fe6 | ||
![]() |
a452a164e6 | ||
![]() |
0d9dd69b19 | ||
![]() |
d7b73c18f1 | ||
![]() |
07f66c0e45 | ||
![]() |
d449acbf86 | ||
![]() |
fce416ba76 | ||
![]() |
cb6bfe8556 | ||
![]() |
d7b31e1d25 | ||
![]() |
85057376d6 | ||
![]() |
c43ac7c869 | ||
![]() |
b2657315f1 | ||
![]() |
de5ed9717c | ||
![]() |
e17a6cbb9f | ||
![]() |
8e783b774b | ||
![]() |
733663f40d | ||
![]() |
4b2a792a62 | ||
![]() |
f7aa171d01 | ||
![]() |
5eafefb683 | ||
![]() |
5d1b02a856 | ||
![]() |
4acda3d9ae | ||
![]() |
643e10ace2 | ||
![]() |
7b64a232de | ||
![]() |
d7472d837d | ||
![]() |
94b473ab4b | ||
![]() |
4a05bbb6c8 | ||
![]() |
0343659b35 | ||
![]() |
f38eadbe30 | ||
![]() |
eb29a53ac5 | ||
![]() |
bc71e260e2 | ||
![]() |
ddf23a3443 | ||
![]() |
d41b248d1c | ||
![]() |
a025b25933 | ||
![]() |
c9a52a6088 | ||
![]() |
0276dca406 | ||
![]() |
3bb95ad44c | ||
![]() |
0a6572c282 | ||
![]() |
3937067be1 | ||
![]() |
48e4eb44f2 | ||
![]() |
c78cc6f2fd | ||
![]() |
73a71e0f5c | ||
![]() |
93605774f0 | ||
![]() |
2834e5d78f | ||
![]() |
dd467b4d63 | ||
![]() |
1fa541776b | ||
![]() |
f6f67c7b0a | ||
![]() |
9b3f19c19b | ||
![]() |
f4a9ec15e8 | ||
![]() |
006c4ecb02 | ||
![]() |
1c752b0e18 | ||
![]() |
f84aff63e3 | ||
![]() |
ae8121b680 | ||
![]() |
882fbf9275 | ||
![]() |
0a1743251e | ||
![]() |
3403a127c1 | ||
![]() |
9c5ca9f09d | ||
![]() |
73eea5608a | ||
![]() |
f48aeb91f4 | ||
![]() |
0bda964ece | ||
![]() |
14f5d54b50 | ||
![]() |
105ac2f6ff | ||
![]() |
160560f1fd | ||
![]() |
deeb667d6f | ||
![]() |
78123ff6f5 | ||
![]() |
9b1fdff22f | ||
![]() |
0275502796 | ||
![]() |
8af475e319 | ||
![]() |
2202c8f09e | ||
![]() |
adf309d3a8 | ||
![]() |
3071314586 | ||
![]() |
88e1df840d | ||
![]() |
23c1b66f6c | ||
![]() |
b0318a1cce | ||
![]() |
c130a66e4d | ||
![]() |
3386ba6d1b | ||
![]() |
9275569fa6 | ||
![]() |
2b23dfd0a6 | ||
![]() |
2a13d9990e | ||
![]() |
c9669b51c6 | ||
![]() |
486c180b3c | ||
![]() |
9eb5bf9b87 | ||
![]() |
953a89f3a1 | ||
![]() |
d638fa1434 | ||
![]() |
e6d700288c | ||
![]() |
371f14cdc9 | ||
![]() |
0733ae2404 | ||
![]() |
b1731ebd49 | ||
![]() |
342b3191ac | ||
![]() |
cd39445245 | ||
![]() |
92eac67367 | ||
![]() |
0e31a0c704 | ||
![]() |
d60c117a70 | ||
![]() |
69ccad5998 | ||
![]() |
b6d22320e6 | ||
![]() |
d3bf948dba | ||
![]() |
5de3d96b31 | ||
![]() |
d30dd64322 | ||
![]() |
386df10a5a | ||
![]() |
af147de547 | ||
![]() |
23cd0e5a5e | ||
![]() |
c5a566657c | ||
![]() |
ce2018c864 | ||
![]() |
472ab46af2 | ||
![]() |
7f87d45bb5 | ||
![]() |
e9f7ab18bb | ||
![]() |
fba83a8afe | ||
![]() |
05089abddc | ||
![]() |
ccd70aac51 | ||
![]() |
5df8445d04 | ||
![]() |
8c43674fa4 | ||
![]() |
f162316a6b | ||
![]() |
670596ed88 | ||
![]() |
6b3eb716c4 | ||
![]() |
6a5180d94c | ||
![]() |
83faaedfcc | ||
![]() |
d98d790a7a | ||
![]() |
36b5833a3a | ||
![]() |
5e86781a79 | ||
![]() |
6a780504b4 | ||
![]() |
a0d8212136 | ||
![]() |
0e1e6a9d62 | ||
![]() |
812282a332 | ||
![]() |
a8a4c9e97f | ||
![]() |
24c293e335 | ||
![]() |
0a596df497 | ||
![]() |
3d66c6572b | ||
![]() |
f45769cbb2 | ||
![]() |
ef51f93c6f | ||
![]() |
35af68f148 | ||
![]() |
ff21430b43 | ||
![]() |
fcb67f5119 | ||
![]() |
0a41fbd185 | ||
![]() |
d32aaf488f | ||
![]() |
1254798013 | ||
![]() |
c72d2a2308 | ||
![]() |
703181655b | ||
![]() |
da4a1c5bf0 | ||
![]() |
1130bd502e | ||
![]() |
27ea4ee679 | ||
![]() |
56e3b66d06 | ||
![]() |
7d4768e151 | ||
![]() |
7ec1011610 | ||
![]() |
e1bbd2055c | ||
![]() |
8a19547d9f | ||
![]() |
6e6922dab8 | ||
![]() |
e009ade922 | ||
![]() |
90e15bcab9 | ||
![]() |
2a040cea4b | ||
![]() |
72f2a7f8db | ||
![]() |
8a8022afe6 | ||
![]() |
b692bec310 | ||
![]() |
536c01c70d | ||
![]() |
ec8e14e977 | ||
![]() |
85b34f8809 | ||
![]() |
22951a56a5 | ||
![]() |
46ad84b101 | ||
![]() |
5efb77e520 | ||
![]() |
76903102b8 | ||
![]() |
c8e26b429c | ||
![]() |
55c1310f74 | ||
![]() |
b8278d91e0 | ||
![]() |
b032502148 | ||
![]() |
bec1a4dd1a | ||
![]() |
4dfb9e7977 | ||
![]() |
2bfa165cdc | ||
![]() |
658666276d | ||
![]() |
62f91b9084 | ||
![]() |
719140ab78 | ||
![]() |
0471fd8145 | ||
![]() |
a079a0c901 | ||
![]() |
ac2fa74c8f | ||
![]() |
4c10ef65f5 | ||
![]() |
cfa697fab2 | ||
![]() |
a09b9d3e4d | ||
![]() |
c470909f19 | ||
![]() |
5e59cfcf9d | ||
![]() |
a099fe35d2 | ||
![]() |
bcfd8a2450 | ||
![]() |
8ed9d71e14 | ||
![]() |
004c2fa55a | ||
![]() |
3dd63d03cb | ||
![]() |
cceedab864 | ||
![]() |
b494b2ea39 | ||
![]() |
0b29cf086b | ||
![]() |
11d33097f7 | ||
![]() |
3ae61645de | ||
![]() |
4711befffa | ||
![]() |
5673d53a20 | ||
![]() |
90ca4a5e92 | ||
![]() |
ad252956ab | ||
![]() |
1b2c091c39 | ||
![]() |
9031bc0c7b | ||
![]() |
1d85e0ea63 | ||
![]() |
458774aadb | ||
![]() |
ae89f7bea3 | ||
![]() |
fd77b8552b | ||
![]() |
bae9f5e844 | ||
![]() |
e3f3d90b68 | ||
![]() |
646fa877ba | ||
![]() |
7f3bd8aec2 | ||
![]() |
4501203a7a | ||
![]() |
06292bceb2 | ||
![]() |
f94f14ab65 | ||
![]() |
7145c68e03 | ||
![]() |
d1b0cd74be | ||
![]() |
bac3825c87 | ||
![]() |
fc1d283414 | ||
![]() |
c0652daa97 | ||
![]() |
553903bd9d | ||
![]() |
a43ec25b7e | ||
![]() |
bb2a66fd02 | ||
![]() |
71ac830bfa | ||
![]() |
82bce80c62 | ||
![]() |
67ddf78e18 | ||
![]() |
8285df0f3f | ||
![]() |
bdb45295b9 | ||
![]() |
f330ee8f8d | ||
![]() |
7331e4a7f2 | ||
![]() |
51252d3b61 | ||
![]() |
dcdb2c323e | ||
![]() |
c4ac901c67 | ||
![]() |
a7ce072ca2 | ||
![]() |
a9b427b877 | ||
![]() |
67a9f3a4ad | ||
![]() |
1a4905f36a | ||
![]() |
cbb1fde7b0 | ||
![]() |
b86bd019a7 | ||
![]() |
6a0bada9d2 | ||
![]() |
a708278cf0 | ||
![]() |
119462cbc9 | ||
![]() |
0324a4928c | ||
![]() |
895a2a56b5 | ||
![]() |
d9e616beee | ||
![]() |
aa5d5d2b6d | ||
![]() |
85dc555358 | ||
![]() |
15b4a7d055 | ||
![]() |
696c94050d | ||
![]() |
b222614c4a | ||
![]() |
edff694bf3 | ||
![]() |
be430a6ac0 | ||
![]() |
937d40c5f7 | ||
![]() |
d3979676ab | ||
![]() |
6716262a28 | ||
![]() |
06d8bafce6 | ||
![]() |
f814755908 | ||
![]() |
f355fd2551 | ||
![]() |
ea84c62d76 | ||
![]() |
acda71cebb | ||
![]() |
67dcd2e5c6 | ||
![]() |
c996644613 | ||
![]() |
171c3e492d | ||
![]() |
8834195cc6 | ||
![]() |
ed57e72fa1 | ||
![]() |
7e84c3e167 | ||
![]() |
4adc33471b | ||
![]() |
31d07cc1e2 | ||
![]() |
a349a66d5a | ||
![]() |
7d427b4cc4 | ||
![]() |
6c7e54868d | ||
![]() |
9484a5e2ee | ||
![]() |
3cbd2057e3 | ||
![]() |
71ff1cf713 | ||
![]() |
c737d891bc | ||
![]() |
82a53343fc | ||
![]() |
03fdc60018 | ||
![]() |
42d19d98ad | ||
![]() |
c9915bba18 | ||
![]() |
a275d7ff50 | ||
![]() |
74199c8624 | ||
![]() |
238bff1fee | ||
![]() |
de534b58c5 | ||
![]() |
111a0f9f17 | ||
![]() |
50392ed67d | ||
![]() |
3de9da0528 | ||
![]() |
ece93cadd5 | ||
![]() |
7219c8d33c | ||
![]() |
b0a09c7876 | ||
![]() |
1d017d3cbc | ||
![]() |
5fa56f2aca | ||
![]() |
5ded1b139f | ||
![]() |
2abb0db272 | ||
![]() |
674c041c13 | ||
![]() |
03c89a69a2 | ||
![]() |
dbed5e2a62 | ||
![]() |
d7519903a1 | ||
![]() |
ab14f0d7ac | ||
![]() |
3b6e43b7ca | ||
![]() |
7d3b7af874 | ||
![]() |
2acc348062 | ||
![]() |
33ba8acd05 | ||
![]() |
0c56af7090 | ||
![]() |
4375ecaad7 | ||
![]() |
90c538d4b6 | ||
![]() |
1ffa8329d0 | ||
![]() |
b404b7974a | ||
![]() |
de44b580f9 | ||
![]() |
2a630cd745 | ||
![]() |
1ccccc5e39 | ||
![]() |
aec851acfa | ||
![]() |
ad5ac479fd | ||
![]() |
67b82a404c | ||
![]() |
f7f86a2f62 | ||
![]() |
6e42b4af27 | ||
![]() |
fe9ed8af76 | ||
![]() |
c4c3f369c0 | ||
![]() |
9b9bf40e47 | ||
![]() |
14a888a610 | ||
![]() |
f3d777c65c | ||
![]() |
d494b6c934 | ||
![]() |
7294220727 | ||
![]() |
665b9087b1 | ||
![]() |
f1691050cd | ||
![]() |
169b6acd24 | ||
![]() |
35b3bf2edb | ||
![]() |
8f73c8c98b | ||
![]() |
94ea8c21eb | ||
![]() |
b6a19c3e2d | ||
![]() |
21455d58a6 | ||
![]() |
239e9bd3db | ||
![]() |
0832171a2b | ||
![]() |
1edfcc629c | ||
![]() |
9a2a8698ef | ||
![]() |
54c31422d9 | ||
![]() |
a33464284c | ||
![]() |
4842caf426 | ||
![]() |
02b6b4d8eb | ||
![]() |
4b0d7c7d88 | ||
![]() |
559b53acc3 | ||
![]() |
4233c18dbb | ||
![]() |
adec2c9fcc | ||
![]() |
6ad2406262 | ||
![]() |
bfeb8ca104 | ||
![]() |
5614edfa2f | ||
![]() |
946f7b2305 | ||
![]() |
dfcf435f88 | ||
![]() |
0e734c267c | ||
![]() |
faadcec4d1 | ||
![]() |
eb81784818 | ||
![]() |
327010f76b | ||
![]() |
371669dcb6 | ||
![]() |
fb702b93ca | ||
![]() |
50c8453785 | ||
![]() |
14fb5ee6c4 | ||
![]() |
46c2343ec6 | ||
![]() |
f5c226362f | ||
![]() |
9114c1157d | ||
![]() |
a4f4230275 | ||
![]() |
46a6147c08 | ||
![]() |
0d26923d79 | ||
![]() |
b57c93cf03 | ||
![]() |
ec19e4bc8f | ||
![]() |
cb68a9c2ce | ||
![]() |
b51ded1580 | ||
![]() |
4b6fb5bfeb | ||
![]() |
9b8175b5be | ||
![]() |
3ecfc622e2 | ||
![]() |
d8d2ee4e09 | ||
![]() |
be478cb088 | ||
![]() |
c187b4487f | ||
![]() |
f6d21e74cb | ||
![]() |
bde4c4bd25 | ||
![]() |
d9aaceea95 | ||
![]() |
676d64a24a | ||
![]() |
8ae1768f71 | ||
![]() |
212705e7e3 | ||
![]() |
b3f6524e5c | ||
![]() |
5ca0a0adf2 | ||
![]() |
669d2c44c9 | ||
![]() |
604bc5b4c1 | ||
![]() |
88268ae569 | ||
![]() |
140fb86401 | ||
![]() |
2b281c4357 | ||
![]() |
7b6dbfb456 | ||
![]() |
b3c49ac86b | ||
![]() |
d309fd9c97 | ||
![]() |
f3a280dcb6 | ||
![]() |
43894687a2 | ||
![]() |
8dd8928dab | ||
![]() |
282a4d15c2 | ||
![]() |
83a2ffc67e | ||
![]() |
e9a6e3fae0 | ||
![]() |
97944b9793 | ||
![]() |
0b5d995b6d | ||
![]() |
1781a9fe9a | ||
![]() |
a0479bf7ca | ||
![]() |
2affa31edf | ||
![]() |
0bd040a851 | ||
![]() |
616c6a1607 | ||
![]() |
a5176fbf80 | ||
![]() |
f2b76b7582 | ||
![]() |
f4847f31ae | ||
![]() |
05adcf2c12 | ||
![]() |
ea301be3c1 | ||
![]() |
88e9785b01 | ||
![]() |
78efabfc1c | ||
![]() |
9d3e22200c | ||
![]() |
183f9701fd | ||
![]() |
ebbba134fe | ||
![]() |
c257be8176 | ||
![]() |
afb0aea660 | ||
![]() |
960bf46f44 | ||
![]() |
16db799e88 | ||
![]() |
f8f2cdcfcc | ||
![]() |
a1f7862f96 | ||
![]() |
3ff9284f2a | ||
![]() |
edf8f27c0f | ||
![]() |
a720953ff3 | ||
![]() |
f79dd26a82 | ||
![]() |
36b5fce4dd | ||
![]() |
89537322fd | ||
![]() |
2229ce1fe9 | ||
![]() |
f257b2177a | ||
![]() |
056c1fd43e | ||
![]() |
acacd3134f | ||
![]() |
1fe0ba1a6e | ||
![]() |
e639f4b5de | ||
![]() |
5a234bd989 | ||
![]() |
661227266e | ||
![]() |
e01556fd9a | ||
![]() |
5cb53ffc18 | ||
![]() |
e7178626ab | ||
![]() |
7f0948b0ed | ||
![]() |
4baee67781 | ||
![]() |
7061f38abe | ||
![]() |
2aa259c0b2 | ||
![]() |
a0f74e715a | ||
![]() |
c13e761c3f | ||
![]() |
2837e44bab | ||
![]() |
d41e3bb41e | ||
![]() |
4160bbb8c4 | ||
![]() |
404a20f280 | ||
![]() |
2241146b9f | ||
![]() |
bcb26c5721 | ||
![]() |
a681d8c1ba | ||
![]() |
0515b74f75 | ||
![]() |
7eeb96d88e | ||
![]() |
5b6c73ef06 | ||
![]() |
c70b866d16 | ||
![]() |
b84f2874dc | ||
![]() |
c372b5529b | ||
![]() |
a43156c38d | ||
![]() |
a15112febc | ||
![]() |
1c6a677a39 | ||
![]() |
9bf749a2c8 | ||
![]() |
f6c9d9df20 | ||
![]() |
5386e0ded9 | ||
![]() |
8fab405a3a | ||
![]() |
ee409f3ca9 | ||
![]() |
bda9beacaa | ||
![]() |
5d8c7e5733 | ||
![]() |
a68c763125 | ||
![]() |
31ac89d9d6 | ||
![]() |
8b258cbbe4 | ||
![]() |
53676fc0fd | ||
![]() |
42c561af9e | ||
![]() |
fae777c14c | ||
![]() |
1af6dc614a | ||
![]() |
2986004638 | ||
![]() |
d6f7b4706b | ||
![]() |
b40dd3e5c0 | ||
![]() |
2f4097ca9d | ||
![]() |
c8a91fbb25 | ||
![]() |
0ec259a5fc | ||
![]() |
2fda99bdb7 | ||
![]() |
1bb4ac4e9c | ||
![]() |
f376b98c09 | ||
![]() |
7382fdb2d2 | ||
![]() |
aead9592cf | ||
![]() |
bedacf29d9 | ||
![]() |
570e9a8307 | ||
![]() |
fcb94ec603 | ||
![]() |
69886ed58b | ||
![]() |
798235cd21 | ||
![]() |
b38b6b6f35 | ||
![]() |
2dba13c52e | ||
![]() |
ff366cb2c5 | ||
![]() |
a87861a993 | ||
![]() |
65476356c9 | ||
![]() |
8af6667f3e | ||
![]() |
b546df7b95 | ||
![]() |
20c1f12da8 | ||
![]() |
e65ff201af | ||
![]() |
4f367a3dcd | ||
![]() |
d7093bce4d | ||
![]() |
79e3e59cae | ||
![]() |
f1b292df93 | ||
![]() |
0cd383b8dd | ||
![]() |
22c404a667 | ||
![]() |
ba11a59d89 | ||
![]() |
06020851a9 | ||
![]() |
fd5cbde18c | ||
![]() |
384398a1e2 | ||
![]() |
2bdba4ba8b | ||
![]() |
db4179a530 | ||
![]() |
b905b74dc2 | ||
![]() |
ff05c36856 | ||
![]() |
40ea5eb53d | ||
![]() |
02f48ccc7f | ||
![]() |
72eaff148c | ||
![]() |
8b60397f06 | ||
![]() |
18d019c62a | ||
![]() |
27527b18e1 | ||
![]() |
ae9aa2662a | ||
![]() |
3bfb593b21 | ||
![]() |
8556e99241 | ||
![]() |
e885822a34 | ||
![]() |
8ebb564a79 | ||
![]() |
7dc176edcb | ||
![]() |
5167fe078b | ||
![]() |
bc7188c8a8 | ||
![]() |
5a05cb96be | ||
![]() |
0258726f0a | ||
![]() |
2fa9aa04f4 | ||
![]() |
a5f9927459 | ||
![]() |
61b422502b | ||
![]() |
1e57b5ea49 | ||
![]() |
12ce915e8e | ||
![]() |
a32273af91 | ||
![]() |
96a327af17 | ||
![]() |
1910e81ad9 | ||
![]() |
80593e774c | ||
![]() |
c092fc8e18 | ||
![]() |
579efa15c7 | ||
![]() |
37ff4e9aeb | ||
![]() |
4e4a7de5d4 | ||
![]() |
2d1bc6436a | ||
![]() |
65726d75cc | ||
![]() |
d63c0a32eb | ||
![]() |
e49c4162e5 | ||
![]() |
ffc5ad5ce7 | ||
![]() |
2dbfc28d69 | ||
![]() |
1ac7b2b8cb | ||
![]() |
7d047e612e | ||
![]() |
24f2999669 | ||
![]() |
b08728b645 | ||
![]() |
00dee43a1e | ||
![]() |
21e300b9f3 | ||
![]() |
158f0aa2d9 | ||
![]() |
10fb763d66 | ||
![]() |
4aa2c1c2c2 | ||
![]() |
bdf044d264 | ||
![]() |
6049a1f2f5 | ||
![]() |
a9fea9f606 | ||
![]() |
11002e9d45 | ||
![]() |
e7a0b850df | ||
![]() |
736c902f3c | ||
![]() |
a67ff564d0 | ||
![]() |
2a778383b1 | ||
![]() |
36457400e7 | ||
![]() |
1d629e7b2e | ||
![]() |
5f764ab8f5 | ||
![]() |
a8c9edbc3f | ||
![]() |
0a5bffe826 | ||
![]() |
9c9b6bc0d6 | ||
![]() |
562f7e7e41 | ||
![]() |
594c55afa6 | ||
![]() |
0abf97e999 | ||
![]() |
b811aec773 | ||
![]() |
ca47f566dc | ||
![]() |
65674f7fd4 | ||
![]() |
dcc510ff6c | ||
![]() |
a4fe43a964 | ||
![]() |
3dab4c07cf | ||
![]() |
ee3248ea5d | ||
![]() |
7be5ec0521 | ||
![]() |
04e90cc279 | ||
![]() |
ec8d488249 | ||
![]() |
41fdafac45 | ||
![]() |
d2e2622279 | ||
![]() |
9476bd6527 | ||
![]() |
24a06ea6f6 | ||
![]() |
2567f8eefb | ||
![]() |
fa5f5ce251 | ||
![]() |
728a61756a | ||
![]() |
736ccbe376 | ||
![]() |
fa2b226b9e | ||
![]() |
e8c5ae194d | ||
![]() |
cc2feab37e | ||
![]() |
83b084a90b | ||
![]() |
e2ac0722c8 | ||
![]() |
c3efb40b8e | ||
![]() |
03d7a416f3 | ||
![]() |
e6e812fdb0 | ||
![]() |
b34160eeec | ||
![]() |
d01aeab242 | ||
![]() |
9904e01252 | ||
![]() |
7f068b691b | ||
![]() |
59558efed1 | ||
![]() |
a88e19a8ed | ||
![]() |
0c17f0825b | ||
![]() |
9384d2523a | ||
![]() |
0e13172a89 | ||
![]() |
92f34452b5 | ||
![]() |
db54929584 | ||
![]() |
ba23cafb18 | ||
![]() |
a2e189767b | ||
![]() |
0d236fd678 | ||
![]() |
9979b160c7 | ||
![]() |
65b332c395 | ||
![]() |
b867ac8cc7 | ||
![]() |
9a0ff24ffe | ||
![]() |
9540a12b6f | ||
![]() |
fdfb53c05c | ||
![]() |
3ec979cc40 | ||
![]() |
9ea08c8a4b | ||
![]() |
629eada5c3 | ||
![]() |
a1220c77da | ||
![]() |
b4668367c6 | ||
![]() |
77da40e507 | ||
![]() |
b3b2748bb7 | ||
![]() |
26e8143616 | ||
![]() |
1444fe5468 | ||
![]() |
ac431e3ece | ||
![]() |
563a4137bd | ||
![]() |
19cbcd0c1d | ||
![]() |
8803b60b28 | ||
![]() |
e9f59ae769 | ||
![]() |
5cf3bee336 | ||
![]() |
060fe835c7 | ||
![]() |
2688ea8f59 | ||
![]() |
8eb61cf752 | ||
![]() |
6628901d46 | ||
![]() |
5a31882be3 | ||
![]() |
a42da09d6c | ||
![]() |
658cf5c873 | ||
![]() |
b8c752b740 | ||
![]() |
9151ae7081 | ||
![]() |
a1a894f722 | ||
![]() |
3352ee3151 | ||
![]() |
da3533a430 | ||
![]() |
0bb7f9becf | ||
![]() |
40c64ee2d8 | ||
![]() |
66651f7111 | ||
![]() |
c7d0bd5dec | ||
![]() |
9ce0a9d49d | ||
![]() |
2070b353c2 | ||
![]() |
1d91f3b91b | ||
![]() |
8b0aebfddb | ||
![]() |
467905d7b0 | ||
![]() |
cc81921bcb | ||
![]() |
8600c04ff3 | ||
![]() |
3f31445f12 | ||
![]() |
8a33371f37 | ||
![]() |
1164bd7183 | ||
![]() |
589fcd09c0 | ||
![]() |
c5d49016d4 | ||
![]() |
8886b12151 | ||
![]() |
336ffd7cf0 | ||
![]() |
3088778a9f | ||
![]() |
4c1de83b24 | ||
![]() |
ee2fbfc2d1 | ||
![]() |
dff7fe722b | ||
![]() |
eb6dac2e9f | ||
![]() |
255760de16 | ||
![]() |
a2373b817a | ||
![]() |
72a9940863 | ||
![]() |
46e088b5f3 | ||
![]() |
a3468b51e2 | ||
![]() |
34f19c4268 | ||
![]() |
1d2c616ce0 | ||
![]() |
99e0f0c3e4 | ||
![]() |
1a92dfb019 | ||
![]() |
cc7f27fb53 | ||
![]() |
e8402008bc | ||
![]() |
c1a302834c | ||
![]() |
762f374f93 | ||
![]() |
e21d2bd511 | ||
![]() |
d936ca6b89 | ||
![]() |
88ac821070 | ||
![]() |
c20837d5f5 | ||
![]() |
81a4c66f92 | ||
![]() |
e4dfb02cb0 | ||
![]() |
0abaab4880 | ||
![]() |
a1aaa52c2a | ||
![]() |
b3a509ad14 | ||
![]() |
ad0f58090f | ||
![]() |
43c4e619c2 | ||
![]() |
427397ba7b | ||
![]() |
b51abf1ea6 | ||
![]() |
76e082159d | ||
![]() |
d36c371c1d | ||
![]() |
a5b2100f8a | ||
![]() |
fe19780f06 | ||
![]() |
83f1d7af82 | ||
![]() |
1916616b07 | ||
![]() |
0a6a684acc | ||
![]() |
6d9aecd500 | ||
![]() |
4d25db2e11 | ||
![]() |
77d5714059 | ||
![]() |
76c59cbdea | ||
![]() |
212f7dfc93 | ||
![]() |
9ba37ce34c | ||
![]() |
6c439bfbc4 | ||
![]() |
352d0db08b | ||
![]() |
be8ce1fce5 | ||
![]() |
f6356e576a | ||
![]() |
83a34a8ba1 | ||
![]() |
fb7a855eda | ||
![]() |
9c1d778623 | ||
![]() |
1bad2a023d | ||
![]() |
999da51e99 | ||
![]() |
ea4b965eeb | ||
![]() |
33160e83cb | ||
![]() |
3e5e7f49cc | ||
![]() |
cc02b01c2b | ||
![]() |
243e5391db | ||
![]() |
230ad5c04f | ||
![]() |
289cfaa407 | ||
![]() |
0798745c16 | ||
![]() |
094695a7ff | ||
![]() |
86f041b803 | ||
![]() |
00e65153f4 | ||
![]() |
b12f0490f3 | ||
![]() |
42a2bc8a9a | ||
![]() |
8adca3725d | ||
![]() |
b57d4b3048 | ||
![]() |
9c7aa241e4 | ||
![]() |
c3ec9b2ad7 | ||
![]() |
195070f8f5 | ||
![]() |
cbfe91f36f | ||
![]() |
738e2ac344 | ||
![]() |
ba0be665ae | ||
![]() |
0ba6f8b39f | ||
![]() |
2773f5fbc8 | ||
![]() |
263a816c3b | ||
![]() |
e7d148336b | ||
![]() |
829059ea01 | ||
![]() |
622d698ff8 | ||
![]() |
f09b04dce0 | ||
![]() |
59f8583895 | ||
![]() |
f506fc0478 | ||
![]() |
880676d670 | ||
![]() |
6485327b97 | ||
![]() |
5773152ed3 | ||
![]() |
e88312659b | ||
![]() |
c4d0ba549f | ||
![]() |
cb41afb11f | ||
![]() |
6d27aea9f2 | ||
![]() |
39e1f9cb76 | ||
![]() |
8fb7d64f79 | ||
![]() |
08fdef4870 | ||
![]() |
aa0196b9d0 | ||
![]() |
a3426f92ac | ||
![]() |
d50d4254c5 | ||
![]() |
50cdadc4a2 | ||
![]() |
5aa9b6cb12 | ||
![]() |
44fc8d80e0 | ||
![]() |
817fa57bfe | ||
![]() |
668e2da01b | ||
![]() |
7f3982d153 | ||
![]() |
f62ae930c7 | ||
![]() |
d0808ce159 | ||
![]() |
7b19dadbf5 | ||
![]() |
43ab0283d9 | ||
![]() |
c27e9d5901 | ||
![]() |
e0d21627bb | ||
![]() |
1b1dd6ef88 | ||
![]() |
10700007d5 | ||
![]() |
c5ec8d04c1 | ||
![]() |
490b250db6 | ||
![]() |
0630423c8e | ||
![]() |
c2e06517e1 | ||
![]() |
b01ae33d1e | ||
![]() |
a55ee32058 | ||
![]() |
c3941d5bec | ||
![]() |
6020dc2b2d | ||
![]() |
7ab41e0c3a | ||
![]() |
c0a75f5b98 | ||
![]() |
efd4db40ef | ||
![]() |
3c3fe7bf83 | ||
![]() |
268762166a | ||
![]() |
53a1833e26 | ||
![]() |
1ff8b5fb9f | ||
![]() |
225b43ca3c | ||
![]() |
75a58d6381 | ||
![]() |
62814f083e | ||
![]() |
6f9deea873 | ||
![]() |
d3160eed9d | ||
![]() |
9b4a07de34 | ||
![]() |
d31eeac49e | ||
![]() |
84c5d27416 | ||
![]() |
17d77aa31f | ||
![]() |
388ec3e3d3 | ||
![]() |
f0829f9ef3 | ||
![]() |
81f481833c | ||
![]() |
a74c4168f3 | ||
![]() |
776dbc34f7 | ||
![]() |
168ac91ab8 | ||
![]() |
9bd26798b6 | ||
![]() |
4ae81a2de4 | ||
![]() |
3c314ced0a | ||
![]() |
ba9d0d7707 | ||
![]() |
38946e4b0f | ||
![]() |
f71242a036 | ||
![]() |
960fd9be38 | ||
![]() |
40844dcd76 | ||
![]() |
420d28c713 | ||
![]() |
5bbd6afaf1 | ||
![]() |
77a06c7604 | ||
![]() |
1f4f87d3bd | ||
![]() |
2e8d86575e | ||
![]() |
ef0659f436 | ||
![]() |
e18e69966f | ||
![]() |
e7b4b88055 | ||
![]() |
de65d1e1fc | ||
![]() |
7ea0862f95 | ||
![]() |
efc7049dfd | ||
![]() |
629549d76f | ||
![]() |
756fb795d6 | ||
![]() |
13d1974a5b | ||
![]() |
d3168a9022 | ||
![]() |
059378eedf | ||
![]() |
e973868a90 | ||
![]() |
2a0e5d6835 | ||
![]() |
5537abe2c3 | ||
![]() |
5eae235b3c | ||
![]() |
bfc7718a21 | ||
![]() |
405d6bee78 |
47
.github/CONTRIBUTING.md
vendored
47
.github/CONTRIBUTING.md
vendored
@@ -5,41 +5,64 @@ PLEASE READ THESE GUIDELINES CAREFULLY BEFORE ANY CONTRIBUTION!
|
||||
|
||||
## Crash reporting
|
||||
|
||||
Do not report crashes in the GitHub issue tracker. NewPipe has an automated crash report system that will ask you to send a report via e-mail when a crash occurs. This contains all the data we need for debugging, and allows you to even add a comment to it. You'll see exactly what is sent, the system is 100% transparent.
|
||||
Do not report crashes in the GitHub issue tracker. NewPipe has an automated crash report system that will ask you to
|
||||
send a report via e-mail when a crash occurs. This contains all the data we need for debugging, and allows you to even
|
||||
add a comment to it. You'll see exactly what is sent, the system is 100% transparent.
|
||||
|
||||
## Issue reporting/feature requests
|
||||
|
||||
* Search the [existing issues](https://github.com/TeamNewPipe/NewPipe/issues) first to make sure your issue/feature hasn't been reported/requested before
|
||||
* Search the [existing issues](https://github.com/TeamNewPipe/NewPipe/issues) first to make sure your issue/feature
|
||||
hasn't been reported/requested before
|
||||
* Check whether your issue/feature is already fixed/implemented
|
||||
* Check if the issue still exists in the latest release/beta version
|
||||
* If you are an Android/Java developer, you are always welcome to fix/implement an issue/a feature yourself. PRs welcome!
|
||||
* We use English for development. Issues in other languages will be closed and ignored.
|
||||
* Please only add *one* issue at a time. Do not put multiple issues into one thread.
|
||||
* When reporting a bug please give us a context, and a description how to reproduce it.
|
||||
* Issues that only contain a generated bug report, but no description might be closed.
|
||||
|
||||
## Bug Fixing
|
||||
* If you want to help NewPipe to become free of bugs (this is our utopic goal for NewPipe), you can send us an email to tnp@newpipe.schabi.org to let me know that you intend to help. We'll send you further instructions. You may, on request, register at our [Sentry](https://sentry.schabi.org) instance (see section "Crash reporting" for more information.
|
||||
* If you want to help NewPipe to become free of bugs (this is our utopic goal for NewPipe), you can send us an email to
|
||||
tnp@newpipe.schabi.org to let me know that you intend to help. We'll send you further instructions. You may, on request,
|
||||
register at our [Sentry](https://sentry.schabi.org) instance (see section "Crash reporting" for more information.
|
||||
|
||||
## Translation
|
||||
|
||||
* NewPipe can be translated via [Weblate](https://hosted.weblate.org/projects/newpipe/strings/). You can log in there with your GitHub account.
|
||||
* NewPipe can be translated via [Weblate](https://hosted.weblate.org/projects/newpipe/strings/). You can log in there
|
||||
with your GitHub account.
|
||||
|
||||
## Code contribution
|
||||
|
||||
* Stick to NewPipe's style conventions (well, just look the other code and then do it the same way :))
|
||||
* Do not bring non-free software (e.g., binary blobs) into the project. Also, make sure you do not introduce Google libraries.
|
||||
* Do not bring non-free software (e.g., binary blobs) into the project. Also, make sure you do not introduce Google
|
||||
libraries.
|
||||
* Stick to [F-Droid contribution guidelines](https://f-droid.org/wiki/page/Inclusion_Policy)
|
||||
* Make changes on a separate branch, not on the master branch. This is commonly known as *feature branch workflow*. You may then send your changes as a pull request on GitHub. Patches to the email address mentioned in this document might not be considered, GitHub is the primary platform.
|
||||
* When submitting changes, you confirm that your code is licensed under the terms of the [GNU General Public License v3](https://www.gnu.org/licenses/gpl-3.0.html).
|
||||
* Please test (compile and run) your code before you submit changes! Ideally, provide test feedback in the PR description. Untested code will **not** be merged!
|
||||
* Make changes on a separate branch, not on the master branch. This is commonly known as *feature branch workflow*. You
|
||||
may then send your changes as a pull request on GitHub. Patches to the email address mentioned in this document might
|
||||
not be considered, GitHub is the primary platform. (This only affects you if you are a member of TeamNewPipe)
|
||||
* When submitting changes, you confirm that your code is licensed under the terms of the
|
||||
[GNU General Public License v3](https://www.gnu.org/licenses/gpl-3.0.html).
|
||||
* Please test (compile and run) your code before you submit changes! Ideally, provide test feedback in the PR
|
||||
description. Untested code will **not** be merged!
|
||||
* Try to figure out yourself why builds on our CI fail.
|
||||
* Make sure your PR is up-to-date with the rest of the code. Often, a simple click on "Update branch" will do the job, but if not, you are asked to merge the master branch manually and resolve the problems on your own. That will make the maintainers' jobs way easier.
|
||||
* Please show intention to maintain your features and code after you contributed it. Unmaintained code is a hassle for the core developers, and just adds work. If you do not intend to maintain features you contributed, please think again about submission, or clearly state that in the description of your PR.
|
||||
* Make sure your PR is up-to-date with the rest of the code. Often, a simple click on "Update branch" will do the job,
|
||||
but if not, you are asked to merge the master branch manually and resolve the problems on your own. That will make the
|
||||
maintainers' jobs way easier.
|
||||
* Please show intention to maintain your features and code after you contributed it. Unmaintained code is a hassle for
|
||||
the core developers, and just adds work. If you do not intend to maintain features you contributed, please think again
|
||||
about submission, or clearly state that in the description of your PR.
|
||||
* Respond yourselves if someone requests changes or otherwise raises issues about your PRs.
|
||||
* Check if your contributions align with the [fdroid inclusion guidelines](https://f-droid.org/en/docs/Inclusion_Policy/).
|
||||
* Check if your submission can be build with the current fdroid build server setup.
|
||||
* Send PR that only cover one specific issue/solution/bug. Do not send PRs that are huge and consists of multiple
|
||||
independent solutions.
|
||||
|
||||
## Communication
|
||||
|
||||
* WE DO NOW HAVE A MAILING LIST: [newpipe@list.schabi.org](https://list.schabi.org/cgi-bin/mailman/listinfo/newpipe).
|
||||
* There is an IRC channel on Freenode which is regularly visited by the core team and other developers: [#newpipe](irc:irc.freenode.net/newpipe). [Click here for Webchat](https://webchat.freenode.net/?channels=newpipe)!
|
||||
* If you want to get in touch with the core team or one of our other contributors you can send an email to tnp(at)schabi.org. Please do not send issue reports, they will be ignored and remain unanswered! Use the GitHub issue tracker described above!
|
||||
* There is an IRC channel on Freenode which is regularly visited by the core team and other developers:
|
||||
[#newpipe](irc:irc.freenode.net/newpipe). [Click here for Webchat](https://webchat.freenode.net/?channels=newpipe)!
|
||||
* If you want to get in touch with the core team or one of our other contributors you can send an email to
|
||||
tnp(at)schabi.org. Please do not send issue reports, they will be ignored and remain unanswered! Use the GitHub issue
|
||||
tracker described above!
|
||||
* Feel free to post suggestions, changes, ideas etc. on GitHub, IRC or the mailing list!
|
||||
|
1
.github/ISSUE_TEMPLATE.md
vendored
1
.github/ISSUE_TEMPLATE.md
vendored
@@ -1,2 +1,3 @@
|
||||
- [ ] I carefully read the [contribution guidelines](https://github.com/TeamNewPipe/NewPipe/blob/HEAD/.github/CONTRIBUTING.md) and agree to them.
|
||||
- [ ] I checked if the issue/feature exists in the latest version.
|
||||
- [ ] I did use the [incredible bugreport to markdown converter](https://teamnewpipe.github.io/CrashReportToMarkdown/) to paste bug reports.
|
||||
|
@@ -5,13 +5,13 @@ android:
|
||||
components:
|
||||
# The BuildTools version used by NewPipe
|
||||
- tools
|
||||
- build-tools-27.0.1
|
||||
- build-tools-28.0.3
|
||||
|
||||
# The SDK version used to compile NewPipe
|
||||
- android-27
|
||||
- android-28
|
||||
|
||||
before_install:
|
||||
- yes | sdkmanager "platforms;android-27"
|
||||
- yes | sdkmanager "platforms;android-28"
|
||||
script: ./gradlew -Dorg.gradle.jvmargs=-Xmx1536m assembleDebug lintDebug testDebugUnitTest
|
||||
|
||||
licenses:
|
||||
|
Binary file not shown.
@@ -1,170 +0,0 @@
|
||||
import java.io.BufferedReader;
|
||||
import java.io.File;
|
||||
import java.io.FileReader;
|
||||
import java.io.IOException;
|
||||
import java.util.regex.*;
|
||||
import java.util.Arrays;
|
||||
import java.util.List;
|
||||
import java.util.ArrayList;
|
||||
import java.nio.file.Files;
|
||||
import java.nio.charset.Charset;
|
||||
|
||||
public final class CheckTranslations {
|
||||
|
||||
private static boolean debug = false;
|
||||
private static boolean plurals = false;
|
||||
private static boolean empty = false;
|
||||
private static boolean remove = false;
|
||||
private static int checks = 0;
|
||||
private static int matches = 0;
|
||||
private static int changes = 0;
|
||||
private static Pattern p, pb, pe, e, o;
|
||||
|
||||
/**
|
||||
* Search translated strings.xml files for empty item / plural tags
|
||||
* and remove them.
|
||||
* @param args directories which contain string.xml files (in any subdirectory)
|
||||
* -e option to find all empty string tags
|
||||
* -p option to find all empty plurals and item tags
|
||||
* -r option to remove all occurrences from the files
|
||||
* -d option to see more details
|
||||
*/
|
||||
public static void main(String[] args) {
|
||||
if (args.length < 1 || (args[0].equals("-d") && args.length < 2)) {
|
||||
System.out.println("Not enough arguments");
|
||||
return;
|
||||
}
|
||||
for (int i = 0; i < args.length; i++) {
|
||||
switch (args[i]) {
|
||||
case "-d":
|
||||
debug = true;
|
||||
break;
|
||||
case "-p":
|
||||
plurals = true;
|
||||
break;
|
||||
case "-e":
|
||||
empty = true;
|
||||
break;
|
||||
case "-r":
|
||||
remove = true;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
if (!plurals && !empty) {
|
||||
plurals = true;
|
||||
empty = true;
|
||||
}
|
||||
|
||||
p = Pattern.compile("(<item quantity=\")(zero|one|two|three|few|many|other)(\"></item>|\"/>)");
|
||||
pb = Pattern.compile("(<plurals[\\sa-zA-Z=\"]*>)");
|
||||
pe = Pattern.compile("(</plurals>)");
|
||||
e = Pattern.compile("(<string[\\sa-z_\\\"=]*)((><\\/string>|\\/>){1})");
|
||||
o = Pattern.compile("(<item quantity=\"other\">)[^</>]*(<\\/item>)");
|
||||
|
||||
for (int i = 0; i < args.length; i++) {
|
||||
if (!args[i].equals("-d") && !args[i].equals("-p") && !args[i].equals("-e") && !args[i].equals("-r")) {
|
||||
File f = new File(args[i]);
|
||||
if (f.exists() && !f.isDirectory()) {
|
||||
checkFile(f);
|
||||
} else if (f.isDirectory()) {
|
||||
checkFiles(f.listFiles());
|
||||
} else {
|
||||
System.out.println("'" + args[i] + "' does not exist!");
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
System.out.println(checks + " files were checked.");
|
||||
System.out.println(matches + " corrupt lines detected.");
|
||||
if (remove) {
|
||||
System.out.println(matches + " corrupt lines removed and " + changes + " lines fixed.");
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
private static void checkFiles(File[] f) {
|
||||
for (int i = 0; i < f.length; i++) {
|
||||
if (f[i].exists() && !f[i].isDirectory()) {
|
||||
if (f[i].toString().contains("strings.xml")) {
|
||||
checkFile(f[i]);
|
||||
}
|
||||
} else if (f[i].isDirectory()) {
|
||||
checkFiles(f[i].listFiles());
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private static void checkFile(File f) {
|
||||
// Do not check our original English strings to cause no unwanted changes
|
||||
// Btw. there should not be empty plural/item tags
|
||||
if (f.toString().contains("values/strings.xml")) {
|
||||
return;
|
||||
}
|
||||
if (debug) System.out.println("Checking " + f.toString());
|
||||
checks++;
|
||||
|
||||
|
||||
List<String> lines = new ArrayList<String>();
|
||||
boolean checkFailed = false;
|
||||
boolean otherDetected = false;
|
||||
boolean inPlurals = false;
|
||||
try (BufferedReader br = new BufferedReader(new FileReader(f))) {
|
||||
String line;
|
||||
int ln = 0;
|
||||
while ((line = br.readLine()) != null) {
|
||||
ln++;
|
||||
if (plurals && p.matcher(line).find()) {
|
||||
matches++;
|
||||
if (debug) System.out.println(" Line " + ln + " was " + ((remove) ? "removed" : "detected") + ": '" + line + "'");
|
||||
checkFailed = true;
|
||||
} else if (empty && e.matcher(line).find()) {
|
||||
matches++;
|
||||
checkFailed = true;
|
||||
if (debug) System.out.println(" Line " + ln + " was " + ((remove) ? "removed" : "detected") + ": '" + line + "'");
|
||||
} else {
|
||||
if (remove) lines.add(line);
|
||||
}
|
||||
}
|
||||
br.close();
|
||||
int pluralsLine = 0;
|
||||
for (int i = 0; i < lines.size(); i++) {
|
||||
if (o.matcher(lines.get(i)).find()) {
|
||||
otherDetected = true;
|
||||
}
|
||||
if (plurals && pb.matcher(lines.get(i)).find()) {
|
||||
inPlurals = true;
|
||||
pluralsLine = i;
|
||||
} else if (plurals && pe.matcher(lines.get(i)).find()) {
|
||||
inPlurals = false;
|
||||
if (!otherDetected) {
|
||||
boolean b = false;
|
||||
check: for(int j = pluralsLine; j < i; j++) {
|
||||
if (lines.get(j).contains("many")) {
|
||||
b = true;
|
||||
pluralsLine = j;
|
||||
break check;
|
||||
}
|
||||
}
|
||||
if (remove && b) {
|
||||
if (debug) System.out.println(" Line " + (pluralsLine + 1) + " was " + ((remove) ? "changed" : "detected") + ": '" + lines.get(pluralsLine) + "'");
|
||||
lines.set(pluralsLine, lines.get(pluralsLine).replace("many", "other"));
|
||||
changes++;
|
||||
checkFailed = true;
|
||||
} else if (debug) {
|
||||
if (debug) System.out.println(" WARNING: Line " + (i + 1) + " - No <item quantity=\"other\"> found!");
|
||||
}
|
||||
}
|
||||
otherDetected = false;
|
||||
}
|
||||
|
||||
}
|
||||
if (remove && checkFailed) {
|
||||
Files.write(f.toPath(), lines, Charset.forName("UTF-8"));
|
||||
}
|
||||
} catch (IOException e) {
|
||||
System.out.println(e);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@@ -1,38 +1,32 @@
|
||||
apply plugin: 'com.android.application'
|
||||
|
||||
android {
|
||||
compileSdkVersion 27
|
||||
buildToolsVersion '27.0.1'
|
||||
compileSdkVersion 28
|
||||
buildToolsVersion '28.0.3'
|
||||
|
||||
defaultConfig {
|
||||
applicationId "org.schabi.newpipe"
|
||||
minSdkVersion 15
|
||||
targetSdkVersion 27
|
||||
versionCode 47
|
||||
versionName "0.11.6"
|
||||
minSdkVersion 19
|
||||
targetSdkVersion 28
|
||||
versionCode 720
|
||||
versionName "0.16.0"
|
||||
|
||||
testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
|
||||
vectorDrawables.useSupportLibrary = true
|
||||
}
|
||||
|
||||
buildTypes {
|
||||
release {
|
||||
minifyEnabled true
|
||||
shrinkResources true
|
||||
proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
|
||||
}
|
||||
|
||||
debug {
|
||||
multiDexEnabled true
|
||||
|
||||
debuggable true
|
||||
applicationIdSuffix ".debug"
|
||||
}
|
||||
beta {
|
||||
minifyEnabled true
|
||||
shrinkResources true
|
||||
proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
|
||||
|
||||
applicationIdSuffix ".beta"
|
||||
}
|
||||
}
|
||||
|
||||
lintOptions {
|
||||
@@ -41,6 +35,7 @@ android {
|
||||
// but continue the build even when errors are found:
|
||||
abortOnError false
|
||||
}
|
||||
|
||||
compileOptions {
|
||||
sourceCompatibility JavaVersion.VERSION_1_8
|
||||
targetCompatibility JavaVersion.VERSION_1_8
|
||||
@@ -48,45 +43,61 @@ android {
|
||||
}
|
||||
|
||||
ext {
|
||||
supportLibVersion = '27.0.2'
|
||||
supportLibVersion = '28.0.0'
|
||||
exoPlayerLibVersion = '2.8.4' //2.9.0
|
||||
roomDbLibVersion = '1.1.1'
|
||||
leakCanaryLibVersion = '1.5.4' //1.6.1
|
||||
okHttpLibVersion = '3.11.0'
|
||||
icepickLibVersion = '3.2.0'
|
||||
stethoLibVersion = '1.5.0'
|
||||
}
|
||||
dependencies {
|
||||
androidTestImplementation('com.android.support.test.espresso:espresso-core:2.2.2') {
|
||||
exclude module: 'support-annotations'
|
||||
}
|
||||
|
||||
implementation 'com.github.TeamNewPipe:NewPipeExtractor:7fd21ec08581d'
|
||||
dependencies {
|
||||
androidTestImplementation('com.android.support.test.espresso:espresso-core:3.0.2', {
|
||||
exclude module: 'support-annotations'
|
||||
})
|
||||
|
||||
implementation 'com.github.TeamNewPipe:NewPipeExtractor:8de53111d9'
|
||||
|
||||
testImplementation 'junit:junit:4.12'
|
||||
testImplementation 'org.mockito:mockito-core:1.10.19'
|
||||
testImplementation 'org.mockito:mockito-core:2.23.0'
|
||||
|
||||
implementation "com.android.support:appcompat-v7:$supportLibVersion"
|
||||
implementation "com.android.support:support-v4:$supportLibVersion"
|
||||
implementation "com.android.support:design:$supportLibVersion"
|
||||
implementation "com.android.support:recyclerview-v7:$supportLibVersion"
|
||||
implementation "com.android.support:preference-v14:$supportLibVersion"
|
||||
implementation "com.android.support:appcompat-v7:${supportLibVersion}"
|
||||
implementation "com.android.support:support-v4:${supportLibVersion}"
|
||||
implementation "com.android.support:design:${supportLibVersion}"
|
||||
implementation "com.android.support:recyclerview-v7:${supportLibVersion}"
|
||||
implementation "com.android.support:preference-v14:${supportLibVersion}"
|
||||
implementation "com.android.support:cardview-v7:${supportLibVersion}"
|
||||
implementation 'com.android.support.constraint:constraint-layout:1.1.3'
|
||||
|
||||
implementation 'com.google.code.gson:gson:2.8.2'
|
||||
implementation 'ch.acra:acra:4.9.2'
|
||||
implementation 'ch.acra:acra:4.9.2' //4.11
|
||||
|
||||
implementation 'com.nostra13.universalimageloader:universal-image-loader:1.9.5'
|
||||
implementation 'de.hdodenhof:circleimageview:2.2.0'
|
||||
implementation 'com.github.nirhart:ParallaxScroll:dd53d1f9d1'
|
||||
implementation 'com.nononsenseapps:filepicker:3.0.1'
|
||||
implementation 'com.google.android.exoplayer:exoplayer:r2.5.4'
|
||||
implementation 'com.nononsenseapps:filepicker:4.2.1'
|
||||
|
||||
debugImplementation 'com.facebook.stetho:stetho:1.5.0'
|
||||
debugImplementation 'com.facebook.stetho:stetho-urlconnection:1.5.0'
|
||||
debugImplementation 'com.android.support:multidex:1.0.2'
|
||||
implementation "com.google.android.exoplayer:exoplayer:${exoPlayerLibVersion}"
|
||||
implementation "com.google.android.exoplayer:extension-mediasession:${exoPlayerLibVersion}"
|
||||
|
||||
implementation 'io.reactivex.rxjava2:rxjava:2.1.7'
|
||||
implementation 'io.reactivex.rxjava2:rxandroid:2.0.1'
|
||||
implementation 'com.jakewharton.rxbinding2:rxbinding:2.0.0'
|
||||
debugImplementation "com.facebook.stetho:stetho:${stethoLibVersion}"
|
||||
debugImplementation "com.facebook.stetho:stetho-urlconnection:${stethoLibVersion}"
|
||||
debugImplementation 'com.android.support:multidex:1.0.3'
|
||||
|
||||
implementation 'android.arch.persistence.room:runtime:1.0.0'
|
||||
implementation 'android.arch.persistence.room:rxjava2:1.0.0'
|
||||
annotationProcessor 'android.arch.persistence.room:compiler:1.0.0'
|
||||
implementation 'io.reactivex.rxjava2:rxjava:2.2.2'
|
||||
implementation 'io.reactivex.rxjava2:rxandroid:2.1.0'
|
||||
implementation 'com.jakewharton.rxbinding2:rxbinding:2.1.1'
|
||||
|
||||
implementation 'frankiesardo:icepick:3.2.0'
|
||||
annotationProcessor 'frankiesardo:icepick-processor:3.2.0'
|
||||
implementation "android.arch.persistence.room:runtime:${roomDbLibVersion}"
|
||||
implementation "android.arch.persistence.room:rxjava2:${roomDbLibVersion}"
|
||||
annotationProcessor "android.arch.persistence.room:compiler:${roomDbLibVersion}"
|
||||
|
||||
implementation "frankiesardo:icepick:${icepickLibVersion}"
|
||||
annotationProcessor "frankiesardo:icepick-processor:${icepickLibVersion}"
|
||||
|
||||
debugImplementation "com.squareup.leakcanary:leakcanary-android:${leakCanaryLibVersion}"
|
||||
releaseImplementation "com.squareup.leakcanary:leakcanary-android-no-op:${leakCanaryLibVersion}"
|
||||
|
||||
implementation "com.squareup.okhttp3:okhttp:${okHttpLibVersion}"
|
||||
debugImplementation "com.facebook.stetho:stetho-okhttp3:${stethoLibVersion}"
|
||||
}
|
||||
|
13
app/proguard-rules.pro
vendored
13
app/proguard-rules.pro
vendored
@@ -35,3 +35,16 @@
|
||||
@icepick.* <fields>;
|
||||
}
|
||||
-keepnames class * { @icepick.State *;}
|
||||
|
||||
# Rules for OkHttp. Copy paste from https://github.com/square/okhttp
|
||||
-dontwarn okhttp3.**
|
||||
-dontwarn okio.**
|
||||
-dontwarn javax.annotation.**
|
||||
# A resource is loaded with a relative path so the package of this class must be preserved.
|
||||
-keepnames class okhttp3.internal.publicsuffix.PublicSuffixDatabase
|
||||
-keepclassmembers class * implements java.io.Serializable {
|
||||
static final long serialVersionUID;
|
||||
!static !transient <fields>;
|
||||
private void writeObject(java.io.ObjectOutputStream);
|
||||
private void readObject(java.io.ObjectInputStream);
|
||||
}
|
||||
|
@@ -1,13 +0,0 @@
|
||||
package org.schabi.newpipe;
|
||||
|
||||
import android.app.Application;
|
||||
import android.test.ApplicationTestCase;
|
||||
|
||||
/**
|
||||
* <a href="http://d.android.com/tools/testing/testing_android.html">Testing Fundamentals</a>
|
||||
*/
|
||||
public class ApplicationTest extends ApplicationTestCase<Application> {
|
||||
public ApplicationTest() {
|
||||
super(Application.class);
|
||||
}
|
||||
}
|
@@ -1,10 +0,0 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
|
||||
xmlns:tools="http://schemas.android.com/tools">
|
||||
|
||||
<application
|
||||
android:label="NewPipe Beta"
|
||||
tools:replace="android:label">
|
||||
</application>
|
||||
|
||||
</manifest>
|
Binary file not shown.
Before Width: | Height: | Size: 5.8 KiB |
Binary file not shown.
Before Width: | Height: | Size: 3.0 KiB |
Binary file not shown.
Before Width: | Height: | Size: 8.1 KiB |
Binary file not shown.
Before Width: | Height: | Size: 13 KiB |
Binary file not shown.
Before Width: | Height: | Size: 19 KiB |
@@ -1,9 +1,26 @@
|
||||
package org.schabi.newpipe;
|
||||
|
||||
import android.content.Context;
|
||||
import android.content.SharedPreferences;
|
||||
import android.preference.PreferenceManager;
|
||||
import android.support.annotation.NonNull;
|
||||
import android.support.multidex.MultiDex;
|
||||
|
||||
import com.facebook.stetho.Stetho;
|
||||
import com.facebook.stetho.okhttp3.StethoInterceptor;
|
||||
import com.squareup.leakcanary.AndroidHeapDumper;
|
||||
import com.squareup.leakcanary.DefaultLeakDirectoryProvider;
|
||||
import com.squareup.leakcanary.HeapDumper;
|
||||
import com.squareup.leakcanary.LeakCanary;
|
||||
import com.squareup.leakcanary.LeakDirectoryProvider;
|
||||
import com.squareup.leakcanary.RefWatcher;
|
||||
|
||||
import org.schabi.newpipe.extractor.Downloader;
|
||||
|
||||
import java.io.File;
|
||||
import java.util.concurrent.TimeUnit;
|
||||
|
||||
import okhttp3.OkHttpClient;
|
||||
|
||||
public class DebugApp extends App {
|
||||
private static final String TAG = DebugApp.class.toString();
|
||||
@@ -17,10 +34,15 @@ public class DebugApp extends App {
|
||||
@Override
|
||||
public void onCreate() {
|
||||
super.onCreate();
|
||||
|
||||
initStetho();
|
||||
}
|
||||
|
||||
@Override
|
||||
protected Downloader getDownloader() {
|
||||
return org.schabi.newpipe.Downloader.init(new OkHttpClient.Builder()
|
||||
.addNetworkInterceptor(new StethoInterceptor()));
|
||||
}
|
||||
|
||||
private void initStetho() {
|
||||
// Create an InitializerBuilder
|
||||
Stetho.InitializerBuilder initializerBuilder =
|
||||
@@ -42,4 +64,41 @@ public class DebugApp extends App {
|
||||
// Initialize Stetho with the Initializer
|
||||
Stetho.initialize(initializer);
|
||||
}
|
||||
|
||||
@Override
|
||||
protected boolean isDisposedRxExceptionsReported() {
|
||||
return PreferenceManager.getDefaultSharedPreferences(this)
|
||||
.getBoolean(getString(R.string.allow_disposed_exceptions_key), false);
|
||||
}
|
||||
|
||||
@Override
|
||||
protected RefWatcher installLeakCanary() {
|
||||
return LeakCanary.refWatcher(this)
|
||||
.heapDumper(new ToggleableHeapDumper(this))
|
||||
// give each object 10 seconds to be gc'ed, before leak canary gets nosy on it
|
||||
.watchDelay(10, TimeUnit.SECONDS)
|
||||
.buildAndInstall();
|
||||
}
|
||||
|
||||
public static class ToggleableHeapDumper implements HeapDumper {
|
||||
private final HeapDumper dumper;
|
||||
private final SharedPreferences preferences;
|
||||
private final String dumpingAllowanceKey;
|
||||
|
||||
ToggleableHeapDumper(@NonNull final Context context) {
|
||||
LeakDirectoryProvider leakDirectoryProvider = new DefaultLeakDirectoryProvider(context);
|
||||
this.dumper = new AndroidHeapDumper(context, leakDirectoryProvider);
|
||||
this.preferences = PreferenceManager.getDefaultSharedPreferences(context);
|
||||
this.dumpingAllowanceKey = context.getString(R.string.allow_heap_dumping_key);
|
||||
}
|
||||
|
||||
private boolean isDumpingAllowed() {
|
||||
return preferences.getBoolean(dumpingAllowanceKey, false);
|
||||
}
|
||||
|
||||
@Override
|
||||
public File dumpHeap() {
|
||||
return isDumpingAllowed() ? dumper.dumpHeap() : HeapDumper.RETRY_LATER;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@@ -9,6 +9,7 @@
|
||||
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE"/>
|
||||
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
|
||||
<uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW"/>
|
||||
<uses-permission android:name="android.permission.FOREGROUND_SERVICE"/>
|
||||
|
||||
<application
|
||||
android:name=".App"
|
||||
@@ -28,15 +29,19 @@
|
||||
</intent-filter>
|
||||
</activity>
|
||||
|
||||
<activity
|
||||
android:name=".player.old.PlayVideoActivity"
|
||||
android:configChanges="orientation|keyboardHidden|screenSize"
|
||||
android:theme="@style/OldVideoPlayerTheme"
|
||||
tools:ignore="UnusedAttribute"/>
|
||||
<receiver android:name="android.support.v4.media.session.MediaButtonReceiver" >
|
||||
<intent-filter>
|
||||
<action android:name="android.intent.action.MEDIA_BUTTON" />
|
||||
</intent-filter>
|
||||
</receiver>
|
||||
|
||||
<service
|
||||
android:name=".player.BackgroundPlayer"
|
||||
android:exported="false"/>
|
||||
android:exported="false">
|
||||
<intent-filter>
|
||||
<action android:name="android.intent.action.MEDIA_BUTTON" />
|
||||
</intent-filter>
|
||||
</service>
|
||||
|
||||
<activity
|
||||
android:name=".player.BackgroundPlayerActivity"
|
||||
@@ -66,9 +71,8 @@
|
||||
android:name=".about.AboutActivity"
|
||||
android:label="@string/title_activity_about"/>
|
||||
|
||||
<activity
|
||||
android:name=".history.HistoryActivity"
|
||||
android:label="@string/title_activity_history"/>
|
||||
<service android:name=".local.subscription.services.SubscriptionsImportService"/>
|
||||
<service android:name=".local.subscription.services.SubscriptionsExportService"/>
|
||||
|
||||
<activity
|
||||
android:name=".PanicResponderActivity"
|
||||
@@ -117,13 +121,17 @@
|
||||
android:grantUriPermissions="true">
|
||||
<meta-data
|
||||
android:name="android.support.FILE_PROVIDER_PATHS"
|
||||
android:resource="@xml/provider_paths"/>
|
||||
android:resource="@xml/nnf_provider_paths"/>
|
||||
</provider>
|
||||
|
||||
<activity
|
||||
android:name=".RouterActivity"
|
||||
android:excludeFromRecents="true"
|
||||
android:label="@string/preferred_open_action_share_menu_title"
|
||||
android:taskAffinity=""
|
||||
android:theme="@style/RouterActivityThemeDark">
|
||||
|
||||
<!-- Youtube filter -->
|
||||
<intent-filter>
|
||||
<action android:name="android.intent.action.VIEW"/>
|
||||
<action android:name="android.media.action.MEDIA_PLAY_FROM_SEARCH"/>
|
||||
@@ -169,9 +177,86 @@
|
||||
<category android:name="android.intent.category.DEFAULT"/>
|
||||
<category android:name="android.intent.category.BROWSABLE"/>
|
||||
|
||||
<data android:scheme="http"/>
|
||||
<data android:scheme="https"/>
|
||||
<data android:host="www.youtube-nocookie.com"/>
|
||||
<data android:pathPrefix="/embed/"/>
|
||||
</intent-filter>
|
||||
<intent-filter>
|
||||
<action android:name="android.intent.action.VIEW"/>
|
||||
<action android:name="android.media.action.MEDIA_PLAY_FROM_SEARCH"/>
|
||||
<action android:name="android.nfc.action.NDEF_DISCOVERED"/>
|
||||
|
||||
<category android:name="android.intent.category.DEFAULT"/>
|
||||
<category android:name="android.intent.category.BROWSABLE"/>
|
||||
|
||||
<data android:scheme="vnd.youtube"/>
|
||||
<data android:scheme="vnd.youtube.launch"/>
|
||||
</intent-filter>
|
||||
|
||||
<!-- Hooktube filter -->
|
||||
<intent-filter>
|
||||
<action android:name="android.intent.action.VIEW"/>
|
||||
<action android:name="android.media.action.MEDIA_PLAY_FROM_SEARCH"/>
|
||||
<action android:name="android.nfc.action.NDEF_DISCOVERED"/>
|
||||
|
||||
<category android:name="android.intent.category.DEFAULT"/>
|
||||
<category android:name="android.intent.category.BROWSABLE"/>
|
||||
|
||||
<data android:scheme="http"/>
|
||||
<data android:scheme="https"/>
|
||||
<data android:host="hooktube.com"/>
|
||||
<data android:host="*.hooktube.com"/>
|
||||
<!-- video prefix -->
|
||||
<data android:pathPrefix="/v/"/>
|
||||
<data android:pathPrefix="/embed/"/>
|
||||
<data android:pathPrefix="/watch"/>
|
||||
<!-- channel prefix -->
|
||||
<data android:pathPrefix="/channel/"/>
|
||||
<data android:pathPrefix="/user/"/>
|
||||
</intent-filter>
|
||||
|
||||
<!-- Invidious filter -->
|
||||
<intent-filter>
|
||||
<action android:name="android.intent.action.VIEW"/>
|
||||
<action android:name="android.media.action.MEDIA_PLAY_FROM_SEARCH"/>
|
||||
<action android:name="android.nfc.action.NDEF_DISCOVERED"/>
|
||||
|
||||
<category android:name="android.intent.category.DEFAULT"/>
|
||||
<category android:name="android.intent.category.BROWSABLE"/>
|
||||
|
||||
<data android:scheme="http"/>
|
||||
<data android:scheme="https"/>
|
||||
<data android:host="invidio.us"/>
|
||||
<data android:host="www.invidio.us"/>
|
||||
<!-- video prefix -->
|
||||
<data android:pathPrefix="/embed/"/>
|
||||
<data android:pathPrefix="/watch"/>
|
||||
<!-- channel prefix -->
|
||||
<data android:pathPrefix="/channel/"/>
|
||||
<data android:pathPrefix="/user/"/>
|
||||
<!-- playlist prefix -->
|
||||
<data android:pathPrefix="/playlist"/>
|
||||
</intent-filter>
|
||||
|
||||
<!-- Soundcloud filter -->
|
||||
<intent-filter>
|
||||
<action android:name="android.intent.action.VIEW"/>
|
||||
<action android:name="android.media.action.MEDIA_PLAY_FROM_SEARCH"/>
|
||||
<action android:name="android.nfc.action.NDEF_DISCOVERED"/>
|
||||
|
||||
<category android:name="android.intent.category.DEFAULT"/>
|
||||
<category android:name="android.intent.category.BROWSABLE"/>
|
||||
|
||||
<data android:scheme="http"/>
|
||||
<data android:scheme="https"/>
|
||||
<data android:host="soundcloud.com"/>
|
||||
<data android:host="m.soundcloud.com"/>
|
||||
<data android:host="www.soundcloud.com"/>
|
||||
<data android:pathPrefix="/"/>
|
||||
</intent-filter>
|
||||
|
||||
<!-- Share filter -->
|
||||
<intent-filter>
|
||||
<action android:name="android.intent.action.SEND"/>
|
||||
<category android:name="android.intent.category.DEFAULT"/>
|
||||
@@ -180,68 +265,7 @@
|
||||
</activity>
|
||||
|
||||
<service
|
||||
android:name=".RouterPlayerActivity$FetcherService"
|
||||
android:name=".RouterActivity$FetcherService"
|
||||
android:exported="false"/>
|
||||
|
||||
<activity
|
||||
android:name=".RouterPlayerActivity"
|
||||
android:excludeFromRecents="true"
|
||||
android:label="@string/preferred_player_share_menu_title"
|
||||
android:taskAffinity=""
|
||||
android:theme="@style/RouterActivityThemeDark">
|
||||
<intent-filter>
|
||||
<action android:name="android.intent.action.VIEW"/>
|
||||
<action android:name="android.media.action.MEDIA_PLAY_FROM_SEARCH"/>
|
||||
<action android:name="android.nfc.action.NDEF_DISCOVERED"/>
|
||||
|
||||
<category android:name="android.intent.category.DEFAULT"/>
|
||||
<category android:name="android.intent.category.BROWSABLE"/>
|
||||
|
||||
<data android:scheme="http"/>
|
||||
<data android:scheme="https"/>
|
||||
<data android:host="youtube.com"/>
|
||||
<data android:host="m.youtube.com"/>
|
||||
<data android:host="www.youtube.com"/>
|
||||
<!-- video prefix -->
|
||||
<data android:pathPrefix="/v/"/>
|
||||
<data android:pathPrefix="/embed/"/>
|
||||
<data android:pathPrefix="/watch"/>
|
||||
<data android:pathPrefix="/attribution_link"/>
|
||||
<!-- channel prefix -->
|
||||
<data android:pathPrefix="/channel/"/>
|
||||
<data android:pathPrefix="/user/"/>
|
||||
<!-- playlist prefix -->
|
||||
<data android:pathPrefix="/playlist"/>
|
||||
</intent-filter>
|
||||
<intent-filter>
|
||||
<action android:name="android.intent.action.VIEW"/>
|
||||
<action android:name="android.media.action.MEDIA_PLAY_FROM_SEARCH"/>
|
||||
<action android:name="android.nfc.action.NDEF_DISCOVERED"/>
|
||||
|
||||
<category android:name="android.intent.category.DEFAULT"/>
|
||||
<category android:name="android.intent.category.BROWSABLE"/>
|
||||
|
||||
<data android:scheme="http"/>
|
||||
<data android:scheme="https"/>
|
||||
<data android:host="youtu.be"/>
|
||||
<data android:pathPrefix="/"/>
|
||||
</intent-filter>
|
||||
<intent-filter>
|
||||
<action android:name="android.intent.action.VIEW"/>
|
||||
<action android:name="android.media.action.MEDIA_PLAY_FROM_SEARCH"/>
|
||||
<action android:name="android.nfc.action.NDEF_DISCOVERED"/>
|
||||
|
||||
<category android:name="android.intent.category.DEFAULT"/>
|
||||
<category android:name="android.intent.category.BROWSABLE"/>
|
||||
|
||||
<data android:scheme="vnd.youtube"/>
|
||||
<data android:scheme="vnd.youtube.launch"/>
|
||||
</intent-filter>
|
||||
<intent-filter>
|
||||
<action android:name="android.intent.action.SEND"/>
|
||||
<category android:name="android.intent.category.DEFAULT"/>
|
||||
<data android:mimeType="text/plain"/>
|
||||
</intent-filter>
|
||||
</activity>
|
||||
</application>
|
||||
</manifest>
|
||||
</manifest>
|
||||
|
@@ -0,0 +1,116 @@
|
||||
package android.support.design.widget;
|
||||
|
||||
import android.animation.ValueAnimator;
|
||||
import android.content.Context;
|
||||
import android.support.annotation.NonNull;
|
||||
import android.support.design.animation.AnimationUtils;
|
||||
import android.util.AttributeSet;
|
||||
import android.view.View;
|
||||
|
||||
// check this https://github.com/ToDou/appbarlayout-spring-behavior/blob/master/appbarspring/src/main/java/android/support/design/widget/AppBarFlingFixBehavior.java
|
||||
public final class FlingBehavior extends AppBarLayout.Behavior {
|
||||
|
||||
private ValueAnimator mOffsetAnimator;
|
||||
private static final int MAX_OFFSET_ANIMATION_DURATION = 600; // ms
|
||||
|
||||
public FlingBehavior() {
|
||||
}
|
||||
|
||||
public FlingBehavior(Context context, AttributeSet attrs) {
|
||||
super(context, attrs);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onNestedPreScroll(CoordinatorLayout coordinatorLayout, AppBarLayout child, View target, int dx, int dy, int[] consumed, int type) {
|
||||
if (dy != 0) {
|
||||
int val = child.getBottom();
|
||||
if (val != 0) {
|
||||
int min, max;
|
||||
if (dy < 0) {
|
||||
// We're scrolling down
|
||||
} else {
|
||||
// We're scrolling up
|
||||
if (mOffsetAnimator != null && mOffsetAnimator.isRunning()) {
|
||||
mOffsetAnimator.cancel();
|
||||
}
|
||||
min = -child.getUpNestedPreScrollRange();
|
||||
max = 0;
|
||||
consumed[1] = scroll(coordinatorLayout, child, dy, min, max);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean onNestedPreFling(@NonNull CoordinatorLayout coordinatorLayout, @NonNull AppBarLayout child, @NonNull View target, float velocityX, float velocityY) {
|
||||
|
||||
if (velocityY != 0) {
|
||||
if (velocityY < 0) {
|
||||
// We're flinging down
|
||||
int val = child.getBottom();
|
||||
if (val != 0) {
|
||||
final int targetScroll =
|
||||
+child.getDownNestedPreScrollRange();
|
||||
animateOffsetTo(coordinatorLayout, child, targetScroll, velocityY);
|
||||
}
|
||||
|
||||
} else {
|
||||
// We're flinging up
|
||||
int val = child.getBottom();
|
||||
if (val != 0) {
|
||||
final int targetScroll = -child.getUpNestedPreScrollRange();
|
||||
if (getTopBottomOffsetForScrollingSibling() > targetScroll) {
|
||||
animateOffsetTo(coordinatorLayout, child, targetScroll, velocityY);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return super.onNestedPreFling(coordinatorLayout, child, target, velocityX, velocityY);
|
||||
}
|
||||
|
||||
private void animateOffsetTo(final CoordinatorLayout coordinatorLayout,
|
||||
final AppBarLayout child, final int offset, float velocity) {
|
||||
final int distance = Math.abs(getTopBottomOffsetForScrollingSibling() - offset);
|
||||
|
||||
final int duration;
|
||||
velocity = Math.abs(velocity);
|
||||
if (velocity > 0) {
|
||||
duration = 3 * Math.round(1000 * (distance / velocity));
|
||||
} else {
|
||||
final float distanceRatio = (float) distance / child.getHeight();
|
||||
duration = (int) ((distanceRatio + 1) * 150);
|
||||
}
|
||||
|
||||
animateOffsetWithDuration(coordinatorLayout, child, offset, duration);
|
||||
}
|
||||
|
||||
private void animateOffsetWithDuration(final CoordinatorLayout coordinatorLayout,
|
||||
final AppBarLayout child, final int offset, final int duration) {
|
||||
final int currentOffset = getTopBottomOffsetForScrollingSibling();
|
||||
if (currentOffset == offset) {
|
||||
if (mOffsetAnimator != null && mOffsetAnimator.isRunning()) {
|
||||
mOffsetAnimator.cancel();
|
||||
}
|
||||
return;
|
||||
}
|
||||
|
||||
if (mOffsetAnimator == null) {
|
||||
mOffsetAnimator = new ValueAnimator();
|
||||
mOffsetAnimator.setInterpolator(AnimationUtils.DECELERATE_INTERPOLATOR);
|
||||
mOffsetAnimator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {
|
||||
@Override
|
||||
public void onAnimationUpdate(ValueAnimator animator) {
|
||||
setHeaderTopBottomOffset(coordinatorLayout, child,
|
||||
(Integer) animator.getAnimatedValue());
|
||||
}
|
||||
});
|
||||
} else {
|
||||
mOffsetAnimator.cancel();
|
||||
}
|
||||
|
||||
mOffsetAnimator.setDuration(Math.min(duration, MAX_OFFSET_ANIMATION_DURATION));
|
||||
mOffsetAnimator.setIntValues(currentOffset, offset);
|
||||
mOffsetAnimator.start();
|
||||
}
|
||||
}
|
@@ -1,24 +1,29 @@
|
||||
package org.schabi.newpipe;
|
||||
|
||||
import android.app.AlarmManager;
|
||||
import android.annotation.TargetApi;
|
||||
import android.app.Application;
|
||||
import android.app.NotificationChannel;
|
||||
import android.app.NotificationManager;
|
||||
import android.app.PendingIntent;
|
||||
import android.content.Context;
|
||||
import android.content.Intent;
|
||||
import android.os.Build;
|
||||
import android.preference.PreferenceManager;
|
||||
import android.support.annotation.Nullable;
|
||||
import android.util.Log;
|
||||
|
||||
import com.nostra13.universalimageloader.cache.memory.impl.LRULimitedMemoryCache;
|
||||
import com.nostra13.universalimageloader.core.ImageLoader;
|
||||
import com.nostra13.universalimageloader.core.ImageLoaderConfiguration;
|
||||
import com.squareup.leakcanary.LeakCanary;
|
||||
import com.squareup.leakcanary.RefWatcher;
|
||||
|
||||
import org.acra.ACRA;
|
||||
import org.acra.config.ACRAConfiguration;
|
||||
import org.acra.config.ACRAConfigurationException;
|
||||
import org.acra.config.ConfigurationBuilder;
|
||||
import org.acra.sender.ReportSenderFactory;
|
||||
import org.schabi.newpipe.extractor.Downloader;
|
||||
import org.schabi.newpipe.extractor.NewPipe;
|
||||
import org.schabi.newpipe.extractor.utils.Localization;
|
||||
import org.schabi.newpipe.report.AcraReportSenderFactory;
|
||||
import org.schabi.newpipe.report.ErrorActivity;
|
||||
import org.schabi.newpipe.report.UserAction;
|
||||
@@ -29,9 +34,13 @@ import org.schabi.newpipe.util.StateSaver;
|
||||
import java.io.IOException;
|
||||
import java.io.InterruptedIOException;
|
||||
import java.net.SocketException;
|
||||
import java.util.Collections;
|
||||
import java.util.List;
|
||||
|
||||
import io.reactivex.annotations.NonNull;
|
||||
import io.reactivex.exceptions.CompositeException;
|
||||
import io.reactivex.exceptions.MissingBackpressureException;
|
||||
import io.reactivex.exceptions.OnErrorNotImplementedException;
|
||||
import io.reactivex.exceptions.UndeliverableException;
|
||||
import io.reactivex.functions.Consumer;
|
||||
import io.reactivex.plugins.RxJavaPlugins;
|
||||
@@ -56,9 +65,12 @@ import io.reactivex.plugins.RxJavaPlugins;
|
||||
|
||||
public class App extends Application {
|
||||
protected static final String TAG = App.class.toString();
|
||||
private RefWatcher refWatcher;
|
||||
private static App app;
|
||||
|
||||
@SuppressWarnings("unchecked")
|
||||
private static final Class<? extends ReportSenderFactory>[] reportSenderFactoryClasses = new Class[]{AcraReportSenderFactory.class};
|
||||
private static final Class<? extends ReportSenderFactory>[]
|
||||
reportSenderFactoryClasses = new Class[]{AcraReportSenderFactory.class};
|
||||
|
||||
@Override
|
||||
protected void attachBaseContext(Context base) {
|
||||
@@ -71,54 +83,105 @@ public class App extends Application {
|
||||
public void onCreate() {
|
||||
super.onCreate();
|
||||
|
||||
if (LeakCanary.isInAnalyzerProcess(this)) {
|
||||
// This process is dedicated to LeakCanary for heap analysis.
|
||||
// You should not init your app in this process.
|
||||
return;
|
||||
}
|
||||
refWatcher = installLeakCanary();
|
||||
|
||||
app = this;
|
||||
|
||||
// Initialize settings first because others inits can use its values
|
||||
SettingsActivity.initSettings(this);
|
||||
|
||||
NewPipe.init(Downloader.getInstance());
|
||||
NewPipeDatabase.init(this);
|
||||
NewPipe.init(getDownloader(),
|
||||
org.schabi.newpipe.util.Localization.getPreferredExtractorLocal(this));
|
||||
StateSaver.init(this);
|
||||
initNotificationChannel();
|
||||
|
||||
// Initialize image loader
|
||||
ImageLoaderConfiguration config = new ImageLoaderConfiguration.Builder(this).build();
|
||||
ImageLoader.getInstance().init(config);
|
||||
ImageLoader.getInstance().init(getImageLoaderConfigurations(10, 50));
|
||||
|
||||
configureRxJavaErrorHandler();
|
||||
|
||||
// Check for new version
|
||||
new CheckForNewAppVersionTask().execute();
|
||||
}
|
||||
|
||||
protected Downloader getDownloader() {
|
||||
return org.schabi.newpipe.Downloader.init(null);
|
||||
}
|
||||
|
||||
private void configureRxJavaErrorHandler() {
|
||||
// https://github.com/ReactiveX/RxJava/wiki/What's-different-in-2.0#error-handling
|
||||
RxJavaPlugins.setErrorHandler(new Consumer<Throwable>() {
|
||||
@Override
|
||||
public void accept(@NonNull Throwable throwable) throws Exception {
|
||||
Log.e(TAG, "RxJavaPlugins.ErrorHandler called with -> : throwable = [" + throwable.getClass().getName() + "]");
|
||||
public void accept(@NonNull Throwable throwable) {
|
||||
Log.e(TAG, "RxJavaPlugins.ErrorHandler called with -> : " +
|
||||
"throwable = [" + throwable.getClass().getName() + "]");
|
||||
|
||||
if (throwable instanceof UndeliverableException) {
|
||||
// As UndeliverableException is a wrapper, get the cause of it to get the "real" exception
|
||||
throwable = throwable.getCause();
|
||||
}
|
||||
|
||||
final List<Throwable> errors;
|
||||
if (throwable instanceof CompositeException) {
|
||||
for (Throwable element : ((CompositeException) throwable).getExceptions()) {
|
||||
if (checkThrowable(element)) return;
|
||||
errors = ((CompositeException) throwable).getExceptions();
|
||||
} else {
|
||||
errors = Collections.singletonList(throwable);
|
||||
}
|
||||
|
||||
for (final Throwable error : errors) {
|
||||
if (isThrowableIgnored(error)) return;
|
||||
if (isThrowableCritical(error)) {
|
||||
reportException(error);
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
if (checkThrowable(throwable)) return;
|
||||
// Out-of-lifecycle exceptions should only be reported if a debug user wishes so,
|
||||
// When exception is not reported, log it
|
||||
if (isDisposedRxExceptionsReported()) {
|
||||
reportException(throwable);
|
||||
} else {
|
||||
Log.e(TAG, "RxJavaPlugin: Undeliverable Exception received: ", throwable);
|
||||
}
|
||||
}
|
||||
|
||||
private boolean isThrowableIgnored(@NonNull final Throwable throwable) {
|
||||
// Don't crash the application over a simple network problem
|
||||
return ExtractorHelper.hasAssignableCauseThrowable(throwable,
|
||||
IOException.class, SocketException.class, // network api cancellation
|
||||
InterruptedException.class, InterruptedIOException.class); // blocking code disposed
|
||||
}
|
||||
|
||||
private boolean isThrowableCritical(@NonNull final Throwable throwable) {
|
||||
// Though these exceptions cannot be ignored
|
||||
return ExtractorHelper.hasAssignableCauseThrowable(throwable,
|
||||
NullPointerException.class, IllegalArgumentException.class, // bug in app
|
||||
OnErrorNotImplementedException.class, MissingBackpressureException.class,
|
||||
IllegalStateException.class); // bug in operator
|
||||
}
|
||||
|
||||
private void reportException(@NonNull final Throwable throwable) {
|
||||
// Throw uncaught exception that will trigger the report system
|
||||
Thread.currentThread().getUncaughtExceptionHandler()
|
||||
.uncaughtException(Thread.currentThread(), throwable);
|
||||
}
|
||||
|
||||
private boolean checkThrowable(@NonNull Throwable throwable) {
|
||||
// Don't crash the application over a simple network problem
|
||||
return ExtractorHelper.hasAssignableCauseThrowable(throwable,
|
||||
IOException.class, SocketException.class, InterruptedException.class, InterruptedIOException.class);
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
private ImageLoaderConfiguration getImageLoaderConfigurations(final int memoryCacheSizeMb,
|
||||
final int diskCacheSizeMb) {
|
||||
return new ImageLoaderConfiguration.Builder(this)
|
||||
.memoryCache(new LRULimitedMemoryCache(memoryCacheSizeMb * 1024 * 1024))
|
||||
.diskCacheSize(diskCacheSizeMb * 1024 * 1024)
|
||||
.imageDownloader(new ImageDownloader(getApplicationContext()))
|
||||
.build();
|
||||
}
|
||||
|
||||
private void initACRA() {
|
||||
try {
|
||||
final ACRAConfiguration acraConfig = new ConfigurationBuilder(this)
|
||||
@@ -128,7 +191,11 @@ public class App extends Application {
|
||||
ACRA.init(this, acraConfig);
|
||||
} catch (ACRAConfigurationException ace) {
|
||||
ace.printStackTrace();
|
||||
ErrorActivity.reportError(this, ace, null, null, ErrorActivity.ErrorInfo.make(UserAction.SOMETHING_ELSE, "none",
|
||||
ErrorActivity.reportError(this,
|
||||
ace,
|
||||
null,
|
||||
null,
|
||||
ErrorActivity.ErrorInfo.make(UserAction.SOMETHING_ELSE, "none",
|
||||
"Could not initialize ACRA crash report", R.string.app_ui_crash));
|
||||
}
|
||||
}
|
||||
@@ -148,8 +215,51 @@ public class App extends Application {
|
||||
NotificationChannel mChannel = new NotificationChannel(id, name, importance);
|
||||
mChannel.setDescription(description);
|
||||
|
||||
NotificationManager mNotificationManager = (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE);
|
||||
NotificationManager mNotificationManager =
|
||||
(NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE);
|
||||
mNotificationManager.createNotificationChannel(mChannel);
|
||||
|
||||
setUpUpdateNotificationChannel(importance);
|
||||
}
|
||||
|
||||
/**
|
||||
* Set up notification channel for app update.
|
||||
* @param importance
|
||||
*/
|
||||
@TargetApi(Build.VERSION_CODES.O)
|
||||
private void setUpUpdateNotificationChannel(int importance) {
|
||||
|
||||
final String appUpdateId
|
||||
= getString(R.string.app_update_notification_channel_id);
|
||||
final CharSequence appUpdateName
|
||||
= getString(R.string.app_update_notification_channel_name);
|
||||
final String appUpdateDescription
|
||||
= getString(R.string.app_update_notification_channel_description);
|
||||
|
||||
NotificationChannel appUpdateChannel
|
||||
= new NotificationChannel(appUpdateId, appUpdateName, importance);
|
||||
appUpdateChannel.setDescription(appUpdateDescription);
|
||||
|
||||
NotificationManager appUpdateNotificationManager
|
||||
= (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE);
|
||||
appUpdateNotificationManager.createNotificationChannel(appUpdateChannel);
|
||||
}
|
||||
|
||||
@Nullable
|
||||
public static RefWatcher getRefWatcher(Context context) {
|
||||
final App application = (App) context.getApplicationContext();
|
||||
return application.refWatcher;
|
||||
}
|
||||
|
||||
protected RefWatcher installLeakCanary() {
|
||||
return RefWatcher.DISABLED;
|
||||
}
|
||||
|
||||
protected boolean isDisposedRxExceptionsReported() {
|
||||
return false;
|
||||
}
|
||||
|
||||
public static App getApp() {
|
||||
return app;
|
||||
}
|
||||
}
|
||||
|
@@ -1,28 +1,36 @@
|
||||
package org.schabi.newpipe;
|
||||
|
||||
import android.content.Context;
|
||||
import android.content.res.TypedArray;
|
||||
import android.os.Bundle;
|
||||
import android.support.annotation.AttrRes;
|
||||
import android.support.annotation.NonNull;
|
||||
import android.support.v4.app.Fragment;
|
||||
import android.support.v4.app.FragmentManager;
|
||||
import android.support.v7.app.AppCompatActivity;
|
||||
import android.util.Log;
|
||||
import android.view.View;
|
||||
|
||||
import com.nostra13.universalimageloader.core.DisplayImageOptions;
|
||||
import com.nostra13.universalimageloader.core.ImageLoader;
|
||||
import com.nostra13.universalimageloader.core.display.FadeInBitmapDisplayer;
|
||||
import com.squareup.leakcanary.RefWatcher;
|
||||
|
||||
import icepick.Icepick;
|
||||
import icepick.State;
|
||||
|
||||
public abstract class BaseFragment extends Fragment {
|
||||
protected final String TAG = getClass().getSimpleName() + "@" + Integer.toHexString(hashCode());
|
||||
protected boolean DEBUG = MainActivity.DEBUG;
|
||||
protected final boolean DEBUG = MainActivity.DEBUG;
|
||||
|
||||
protected AppCompatActivity activity;
|
||||
public static final ImageLoader imageLoader = ImageLoader.getInstance();
|
||||
|
||||
//These values are used for controlling framgents when they are part of the frontpage
|
||||
@State
|
||||
protected boolean useAsFrontPage = false;
|
||||
protected boolean mIsVisibleToUser = false;
|
||||
|
||||
public void useAsFrontPage(boolean value) {
|
||||
useAsFrontPage = value;
|
||||
}
|
||||
|
||||
/*//////////////////////////////////////////////////////////////////////////
|
||||
// Fragment's Lifecycle
|
||||
//////////////////////////////////////////////////////////////////////////*/
|
||||
@@ -67,6 +75,20 @@ public abstract class BaseFragment extends Fragment {
|
||||
protected void onRestoreInstanceState(@NonNull Bundle savedInstanceState) {
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onDestroy() {
|
||||
super.onDestroy();
|
||||
|
||||
RefWatcher refWatcher = App.getRefWatcher(getActivity());
|
||||
if (refWatcher != null) refWatcher.watch(this);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void setUserVisibleHint(boolean isVisibleToUser) {
|
||||
super.setUserVisibleHint(isVisibleToUser);
|
||||
mIsVisibleToUser = isVisibleToUser;
|
||||
}
|
||||
|
||||
/*//////////////////////////////////////////////////////////////////////////
|
||||
// Init
|
||||
//////////////////////////////////////////////////////////////////////////*/
|
||||
@@ -78,33 +100,20 @@ public abstract class BaseFragment extends Fragment {
|
||||
}
|
||||
|
||||
/*//////////////////////////////////////////////////////////////////////////
|
||||
// DisplayImageOptions default configurations
|
||||
// Utils
|
||||
//////////////////////////////////////////////////////////////////////////*/
|
||||
|
||||
public static final DisplayImageOptions BASE_OPTIONS =
|
||||
new DisplayImageOptions.Builder().cacheInMemory(true).build();
|
||||
public void setTitle(String title) {
|
||||
if (DEBUG) Log.d(TAG, "setTitle() called with: title = [" + title + "]");
|
||||
if((!useAsFrontPage || mIsVisibleToUser)
|
||||
&& (activity != null && activity.getSupportActionBar() != null)) {
|
||||
activity.getSupportActionBar().setTitle(title);
|
||||
}
|
||||
}
|
||||
|
||||
public static final DisplayImageOptions DISPLAY_AVATAR_OPTIONS =
|
||||
new DisplayImageOptions.Builder()
|
||||
.cloneFrom(BASE_OPTIONS)
|
||||
.showImageOnLoading(R.drawable.buddy)
|
||||
.showImageForEmptyUri(R.drawable.buddy)
|
||||
.showImageOnFail(R.drawable.buddy)
|
||||
.build();
|
||||
|
||||
public static final DisplayImageOptions DISPLAY_THUMBNAIL_OPTIONS =
|
||||
new DisplayImageOptions.Builder()
|
||||
.cloneFrom(BASE_OPTIONS)
|
||||
.displayer(new FadeInBitmapDisplayer(250))
|
||||
.showImageForEmptyUri(R.drawable.dummy_thumbnail)
|
||||
.showImageOnFail(R.drawable.dummy_thumbnail)
|
||||
.build();
|
||||
|
||||
public static final DisplayImageOptions DISPLAY_BANNER_OPTIONS =
|
||||
new DisplayImageOptions.Builder()
|
||||
.cloneFrom(BASE_OPTIONS)
|
||||
.showImageOnLoading(R.drawable.channel_banner)
|
||||
.showImageForEmptyUri(R.drawable.channel_banner)
|
||||
.showImageOnFail(R.drawable.channel_banner)
|
||||
.build();
|
||||
protected FragmentManager getFM() {
|
||||
return getParentFragment() == null
|
||||
? getFragmentManager()
|
||||
: getParentFragment().getFragmentManager();
|
||||
}
|
||||
}
|
||||
|
@@ -0,0 +1,230 @@
|
||||
package org.schabi.newpipe;
|
||||
|
||||
import android.app.Application;
|
||||
import android.app.PendingIntent;
|
||||
import android.content.Intent;
|
||||
import android.content.SharedPreferences;
|
||||
import android.content.pm.PackageInfo;
|
||||
import android.content.pm.PackageManager;
|
||||
import android.content.pm.Signature;
|
||||
import android.net.Uri;
|
||||
import android.os.AsyncTask;
|
||||
import android.preference.PreferenceManager;
|
||||
import android.support.v4.app.NotificationCompat;
|
||||
import android.support.v4.app.NotificationManagerCompat;
|
||||
|
||||
import org.json.JSONException;
|
||||
import org.json.JSONObject;
|
||||
import org.schabi.newpipe.report.ErrorActivity;
|
||||
import org.schabi.newpipe.report.UserAction;
|
||||
|
||||
import java.io.BufferedReader;
|
||||
import java.io.ByteArrayInputStream;
|
||||
import java.io.IOException;
|
||||
import java.io.InputStream;
|
||||
import java.io.InputStreamReader;
|
||||
import java.net.HttpURLConnection;
|
||||
import java.net.URL;
|
||||
import java.security.MessageDigest;
|
||||
import java.security.NoSuchAlgorithmException;
|
||||
import java.security.cert.CertificateEncodingException;
|
||||
import java.security.cert.CertificateException;
|
||||
import java.security.cert.CertificateFactory;
|
||||
import java.security.cert.X509Certificate;
|
||||
import java.util.concurrent.TimeUnit;
|
||||
|
||||
import okhttp3.OkHttpClient;
|
||||
import okhttp3.Request;
|
||||
import okhttp3.RequestBody;
|
||||
import okhttp3.Response;
|
||||
|
||||
/**
|
||||
* AsyncTask to check if there is a newer version of the NewPipe github apk available or not.
|
||||
* If there is a newer version we show a notification, informing the user. On tapping
|
||||
* the notification, the user will be directed to the download link.
|
||||
*/
|
||||
public class CheckForNewAppVersionTask extends AsyncTask<Void, Void, String> {
|
||||
|
||||
private static final Application app = App.getApp();
|
||||
private static final String GITHUB_APK_SHA1 = "B0:2E:90:7C:1C:D6:FC:57:C3:35:F0:88:D0:8F:50:5F:94:E4:D2:15";
|
||||
private static final String newPipeApiUrl = "https://newpipe.schabi.org/api/data.json";
|
||||
private static final int timeoutPeriod = 30;
|
||||
|
||||
private SharedPreferences mPrefs;
|
||||
private OkHttpClient client;
|
||||
|
||||
@Override
|
||||
protected void onPreExecute() {
|
||||
|
||||
mPrefs = PreferenceManager.getDefaultSharedPreferences(app);
|
||||
|
||||
// Check if user has enabled/ disabled update checking
|
||||
// and if the current apk is a github one or not.
|
||||
if (!mPrefs.getBoolean(app.getString(R.string.update_app_key), true)
|
||||
|| !isGithubApk()) {
|
||||
this.cancel(true);
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
protected String doInBackground(Void... voids) {
|
||||
|
||||
// Make a network request to get latest NewPipe data.
|
||||
if (client == null) {
|
||||
|
||||
client = new OkHttpClient
|
||||
.Builder()
|
||||
.readTimeout(timeoutPeriod, TimeUnit.SECONDS)
|
||||
.build();
|
||||
}
|
||||
|
||||
Request request = new Request.Builder()
|
||||
.url(newPipeApiUrl)
|
||||
.build();
|
||||
|
||||
try {
|
||||
Response response = client.newCall(request).execute();
|
||||
return response.body().string();
|
||||
} catch (IOException ex) {
|
||||
ErrorActivity.reportError(app, ex, null, null,
|
||||
ErrorActivity.ErrorInfo.make(UserAction.SOMETHING_ELSE, "none",
|
||||
"app update API fail", R.string.app_ui_crash));
|
||||
}
|
||||
|
||||
return null;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onPostExecute(String response) {
|
||||
|
||||
// Parse the json from the response.
|
||||
if (response != null) {
|
||||
|
||||
try {
|
||||
JSONObject mainObject = new JSONObject(response);
|
||||
JSONObject flavoursObject = mainObject.getJSONObject("flavors");
|
||||
JSONObject githubObject = flavoursObject.getJSONObject("github");
|
||||
JSONObject githubStableObject = githubObject.getJSONObject("stable");
|
||||
|
||||
String versionName = githubStableObject.getString("version");
|
||||
String versionCode = githubStableObject.getString("version_code");
|
||||
String apkLocationUrl = githubStableObject.getString("apk");
|
||||
|
||||
compareAppVersionAndShowNotification(versionName, apkLocationUrl, versionCode);
|
||||
|
||||
} catch (JSONException ex) {
|
||||
ErrorActivity.reportError(app, ex, null, null,
|
||||
ErrorActivity.ErrorInfo.make(UserAction.SOMETHING_ELSE, "none",
|
||||
"could not parse app update JSON data", R.string.app_ui_crash));
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Method to compare the current and latest available app version.
|
||||
* If a newer version is available, we show the update notification.
|
||||
* @param versionName
|
||||
* @param apkLocationUrl
|
||||
*/
|
||||
private void compareAppVersionAndShowNotification(String versionName,
|
||||
String apkLocationUrl,
|
||||
String versionCode) {
|
||||
|
||||
int NOTIFICATION_ID = 2000;
|
||||
|
||||
if (BuildConfig.VERSION_CODE < Integer.valueOf(versionCode)) {
|
||||
|
||||
// A pending intent to open the apk location url in the browser.
|
||||
Intent intent = new Intent(Intent.ACTION_VIEW, Uri.parse(apkLocationUrl));
|
||||
PendingIntent pendingIntent
|
||||
= PendingIntent.getActivity(app, 0, intent, 0);
|
||||
|
||||
NotificationCompat.Builder notificationBuilder = new NotificationCompat
|
||||
.Builder(app, app.getString(R.string.app_update_notification_channel_id))
|
||||
.setSmallIcon(R.drawable.ic_newpipe_update)
|
||||
.setVisibility(NotificationCompat.VISIBILITY_PUBLIC)
|
||||
.setContentIntent(pendingIntent)
|
||||
.setAutoCancel(true)
|
||||
.setContentTitle(app.getString(R.string.app_update_notification_content_title))
|
||||
.setContentText(app.getString(R.string.app_update_notification_content_text)
|
||||
+ " " + versionName);
|
||||
|
||||
NotificationManagerCompat notificationManager = NotificationManagerCompat.from(app);
|
||||
notificationManager.notify(NOTIFICATION_ID, notificationBuilder.build());
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Method to get the apk's SHA1 key.
|
||||
* https://stackoverflow.com/questions/9293019/get-certificate-fingerprint-from-android-app#22506133
|
||||
*/
|
||||
private static String getCertificateSHA1Fingerprint() {
|
||||
|
||||
PackageManager pm = app.getPackageManager();
|
||||
String packageName = app.getPackageName();
|
||||
int flags = PackageManager.GET_SIGNATURES;
|
||||
PackageInfo packageInfo = null;
|
||||
|
||||
try {
|
||||
packageInfo = pm.getPackageInfo(packageName, flags);
|
||||
} catch (PackageManager.NameNotFoundException ex) {
|
||||
ErrorActivity.reportError(app, ex, null, null,
|
||||
ErrorActivity.ErrorInfo.make(UserAction.SOMETHING_ELSE, "none",
|
||||
"Could not find package info", R.string.app_ui_crash));
|
||||
}
|
||||
|
||||
Signature[] signatures = packageInfo.signatures;
|
||||
byte[] cert = signatures[0].toByteArray();
|
||||
InputStream input = new ByteArrayInputStream(cert);
|
||||
|
||||
CertificateFactory cf = null;
|
||||
X509Certificate c = null;
|
||||
|
||||
try {
|
||||
cf = CertificateFactory.getInstance("X509");
|
||||
c = (X509Certificate) cf.generateCertificate(input);
|
||||
} catch (CertificateException ex) {
|
||||
ErrorActivity.reportError(app, ex, null, null,
|
||||
ErrorActivity.ErrorInfo.make(UserAction.SOMETHING_ELSE, "none",
|
||||
"Certificate error", R.string.app_ui_crash));
|
||||
}
|
||||
|
||||
String hexString = null;
|
||||
|
||||
try {
|
||||
MessageDigest md = MessageDigest.getInstance("SHA1");
|
||||
byte[] publicKey = md.digest(c.getEncoded());
|
||||
hexString = byte2HexFormatted(publicKey);
|
||||
} catch (NoSuchAlgorithmException ex1) {
|
||||
ErrorActivity.reportError(app, ex1, null, null,
|
||||
ErrorActivity.ErrorInfo.make(UserAction.SOMETHING_ELSE, "none",
|
||||
"Could not retrieve SHA1 key", R.string.app_ui_crash));
|
||||
} catch (CertificateEncodingException ex2) {
|
||||
ErrorActivity.reportError(app, ex2, null, null,
|
||||
ErrorActivity.ErrorInfo.make(UserAction.SOMETHING_ELSE, "none",
|
||||
"Could not retrieve SHA1 key", R.string.app_ui_crash));
|
||||
}
|
||||
|
||||
return hexString;
|
||||
}
|
||||
|
||||
private static String byte2HexFormatted(byte[] arr) {
|
||||
|
||||
StringBuilder str = new StringBuilder(arr.length * 2);
|
||||
|
||||
for (int i = 0; i < arr.length; i++) {
|
||||
String h = Integer.toHexString(arr[i]);
|
||||
int l = h.length();
|
||||
if (l == 1) h = "0" + h;
|
||||
if (l > 2) h = h.substring(l - 2, l);
|
||||
str.append(h.toUpperCase());
|
||||
if (i < (arr.length - 1)) str.append(':');
|
||||
}
|
||||
return str.toString();
|
||||
}
|
||||
|
||||
public static boolean isGithubApk() {
|
||||
|
||||
return getCertificateSHA1Fingerprint().equals(GITHUB_APK_SHA1);
|
||||
}
|
||||
}
|
File diff suppressed because it is too large
Load Diff
46
app/src/main/java/org/schabi/newpipe/ImageDownloader.java
Normal file
46
app/src/main/java/org/schabi/newpipe/ImageDownloader.java
Normal file
@@ -0,0 +1,46 @@
|
||||
package org.schabi.newpipe;
|
||||
|
||||
import android.annotation.SuppressLint;
|
||||
import android.content.Context;
|
||||
import android.content.SharedPreferences;
|
||||
import android.content.res.Resources;
|
||||
import android.preference.PreferenceManager;
|
||||
|
||||
import com.nostra13.universalimageloader.core.download.BaseImageDownloader;
|
||||
|
||||
import org.schabi.newpipe.extractor.NewPipe;
|
||||
|
||||
import java.io.IOException;
|
||||
import java.io.InputStream;
|
||||
|
||||
public class ImageDownloader extends BaseImageDownloader {
|
||||
private final Resources resources;
|
||||
private final SharedPreferences preferences;
|
||||
private final String downloadThumbnailKey;
|
||||
|
||||
public ImageDownloader(Context context) {
|
||||
super(context);
|
||||
this.resources = context.getResources();
|
||||
this.preferences = PreferenceManager.getDefaultSharedPreferences(context);
|
||||
this.downloadThumbnailKey = context.getString(R.string.download_thumbnail_key);
|
||||
}
|
||||
|
||||
private boolean isDownloadingThumbnail() {
|
||||
return preferences.getBoolean(downloadThumbnailKey, true);
|
||||
}
|
||||
|
||||
@SuppressLint("ResourceType")
|
||||
@Override
|
||||
public InputStream getStream(String imageUri, Object extra) throws IOException {
|
||||
if (isDownloadingThumbnail()) {
|
||||
return super.getStream(imageUri, extra);
|
||||
} else {
|
||||
return resources.openRawResource(R.drawable.dummy_thumbnail_dark);
|
||||
}
|
||||
}
|
||||
|
||||
protected InputStream getStreamFromNetwork(String imageUri, Object extra) throws IOException {
|
||||
final Downloader downloader = (Downloader) NewPipe.getDownloader();
|
||||
return downloader.stream(imageUri);
|
||||
}
|
||||
}
|
File diff suppressed because it is too large
Load Diff
@@ -7,25 +7,36 @@ import android.support.annotation.NonNull;
|
||||
import org.schabi.newpipe.database.AppDatabase;
|
||||
|
||||
import static org.schabi.newpipe.database.AppDatabase.DATABASE_NAME;
|
||||
import static org.schabi.newpipe.database.Migrations.MIGRATION_11_12;
|
||||
|
||||
public final class NewPipeDatabase {
|
||||
|
||||
private static AppDatabase databaseInstance;
|
||||
private static volatile AppDatabase databaseInstance;
|
||||
|
||||
private NewPipeDatabase() {
|
||||
//no instance
|
||||
}
|
||||
|
||||
public static void init(Context context) {
|
||||
databaseInstance = Room.databaseBuilder(context.getApplicationContext(),
|
||||
AppDatabase.class, DATABASE_NAME
|
||||
).build();
|
||||
private static AppDatabase getDatabase(Context context) {
|
||||
return Room
|
||||
.databaseBuilder(context.getApplicationContext(), AppDatabase.class, DATABASE_NAME)
|
||||
.addMigrations(MIGRATION_11_12)
|
||||
.fallbackToDestructiveMigration()
|
||||
.build();
|
||||
}
|
||||
|
||||
@NonNull
|
||||
public static AppDatabase getInstance() {
|
||||
if (databaseInstance == null) throw new RuntimeException("Database not initialized");
|
||||
public static AppDatabase getInstance(@NonNull Context context) {
|
||||
AppDatabase result = databaseInstance;
|
||||
if (result == null) {
|
||||
synchronized (NewPipeDatabase.class) {
|
||||
result = databaseInstance;
|
||||
if (result == null) {
|
||||
databaseInstance = (result = getDatabase(context));
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return databaseInstance;
|
||||
return result;
|
||||
}
|
||||
}
|
||||
|
@@ -85,7 +85,7 @@ public class ReCaptchaActivity extends AppCompatActivity {
|
||||
}
|
||||
|
||||
private class ReCaptchaWebViewClient extends WebViewClient {
|
||||
private Activity context;
|
||||
private final Activity context;
|
||||
private String mCookies;
|
||||
|
||||
ReCaptchaWebViewClient(Activity ctx) {
|
||||
@@ -107,7 +107,7 @@ public class ReCaptchaActivity extends AppCompatActivity {
|
||||
// find cookies : s_gl & goojf and Add cookies to Downloader
|
||||
if (find_access_cookies(cookies)) {
|
||||
// Give cookies to Downloader class
|
||||
Downloader.setCookies(mCookies);
|
||||
Downloader.getInstance().setCookies(mCookies);
|
||||
|
||||
// Closing activity and return to parent
|
||||
setResult(RESULT_OK);
|
||||
|
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
@@ -32,7 +32,6 @@ public class AboutActivity extends AppCompatActivity {
|
||||
new SoftwareComponent("Giga Get", "2014", "Peter Cai", "https://github.com/PaperAirplane-Dev-Team/GigaGet", StandardLicenses.GPL2),
|
||||
new SoftwareComponent("NewPipe Extractor", "2017", "Christian Schabesberger", "https://github.com/TeamNewPipe/NewPipeExtractor", StandardLicenses.GPL3),
|
||||
new SoftwareComponent("Jsoup", "2017", "Jonathan Hedley", "https://github.com/jhy/jsoup", StandardLicenses.MIT),
|
||||
new SoftwareComponent("Google Gson", "2008", "Google Inc", "https://github.com/google/gson", StandardLicenses.APACHE2),
|
||||
new SoftwareComponent("Rhino", "2015", "Mozilla", "https://www.mozilla.org/rhino/", StandardLicenses.MPL2),
|
||||
new SoftwareComponent("ACRA", "2013", "Kevin Gaudin", "http://www.acra.ch", StandardLicenses.APACHE2),
|
||||
new SoftwareComponent("Universal Image Loader", "2011 - 2015", "Sergey Tarasevich", "https://github.com/nostra13/Android-Universal-Image-Loader", StandardLicenses.APACHE2),
|
||||
@@ -129,47 +128,31 @@ public class AboutActivity extends AppCompatActivity {
|
||||
public View onCreateView(LayoutInflater inflater, ViewGroup container,
|
||||
Bundle savedInstanceState) {
|
||||
View rootView = inflater.inflate(R.layout.fragment_about, container, false);
|
||||
Context context = this.getContext();
|
||||
|
||||
TextView version = rootView.findViewById(R.id.app_version);
|
||||
version.setText(BuildConfig.VERSION_NAME);
|
||||
|
||||
View githubLink = rootView.findViewById(R.id.github_link);
|
||||
githubLink.setOnClickListener(new OnGithubLinkClickListener());
|
||||
githubLink.setOnClickListener(nv -> openWebsite(context.getString(R.string.github_url), context));
|
||||
|
||||
View donationLink = rootView.findViewById(R.id.donation_link);
|
||||
donationLink.setOnClickListener(new OnDonationLinkClickListener());
|
||||
donationLink.setOnClickListener(v -> openWebsite(context.getString(R.string.donation_url), context));
|
||||
|
||||
View websiteLink = rootView.findViewById(R.id.website_link);
|
||||
websiteLink.setOnClickListener(new OnWebsiteLinkClickListener());
|
||||
websiteLink.setOnClickListener(nv -> openWebsite(context.getString(R.string.website_url), context));
|
||||
|
||||
View privacyPolicyLink = rootView.findViewById(R.id.privacy_policy_link);
|
||||
privacyPolicyLink.setOnClickListener(v -> openWebsite(context.getString(R.string.privacy_policy_url), context));
|
||||
|
||||
return rootView;
|
||||
}
|
||||
|
||||
private static class OnGithubLinkClickListener implements View.OnClickListener {
|
||||
@Override
|
||||
public void onClick(final View view) {
|
||||
final Context context = view.getContext();
|
||||
Intent intent = new Intent(Intent.ACTION_VIEW, Uri.parse(context.getString(R.string.github_url)));
|
||||
context.startActivity(intent);
|
||||
}
|
||||
private void openWebsite(String url, Context context) {
|
||||
Intent intent = new Intent(Intent.ACTION_VIEW, Uri.parse(url));
|
||||
context.startActivity(intent);
|
||||
}
|
||||
|
||||
private static class OnDonationLinkClickListener implements View.OnClickListener {
|
||||
@Override
|
||||
public void onClick(final View view) {
|
||||
final Context context = view.getContext();
|
||||
Intent intent = new Intent(Intent.ACTION_VIEW, Uri.parse(context.getString(R.string.donation_url)));
|
||||
context.startActivity(intent);
|
||||
}
|
||||
}
|
||||
|
||||
private static class OnWebsiteLinkClickListener implements View.OnClickListener {
|
||||
@Override
|
||||
public void onClick(final View view) {
|
||||
final Context context = view.getContext();
|
||||
Intent intent = new Intent(Intent.ACTION_VIEW, Uri.parse(context.getString(R.string.website_url)));
|
||||
context.startActivity(intent);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
|
@@ -1,5 +1,6 @@
|
||||
package org.schabi.newpipe.about;
|
||||
|
||||
import android.app.Activity;
|
||||
import android.content.Context;
|
||||
import android.content.Intent;
|
||||
import android.net.Uri;
|
||||
@@ -39,7 +40,7 @@ public class LicenseFragment extends Fragment {
|
||||
* @param license the license to show
|
||||
*/
|
||||
public static void showLicense(Context context, License license) {
|
||||
new LicenseFragmentHelper().execute(context, license);
|
||||
new LicenseFragmentHelper((Activity) context).execute(license);
|
||||
}
|
||||
|
||||
@Override
|
||||
|
@@ -1,8 +1,10 @@
|
||||
package org.schabi.newpipe.about;
|
||||
|
||||
import android.app.Activity;
|
||||
import android.content.Context;
|
||||
import android.content.DialogInterface;
|
||||
import android.os.AsyncTask;
|
||||
import android.support.annotation.Nullable;
|
||||
import android.support.v7.app.AlertDialog;
|
||||
import android.webkit.WebView;
|
||||
import org.schabi.newpipe.R;
|
||||
@@ -10,26 +12,46 @@ import org.schabi.newpipe.util.ThemeHelper;
|
||||
|
||||
import java.io.BufferedReader;
|
||||
import java.io.InputStreamReader;
|
||||
import java.lang.ref.WeakReference;
|
||||
|
||||
public class LicenseFragmentHelper extends AsyncTask<Object, Void, Integer> {
|
||||
|
||||
private Context context;
|
||||
final WeakReference<Activity> weakReference;
|
||||
private License license;
|
||||
|
||||
public LicenseFragmentHelper(@Nullable Activity activity) {
|
||||
weakReference = new WeakReference<>(activity);
|
||||
}
|
||||
|
||||
@Nullable
|
||||
private Activity getActivity() {
|
||||
Activity activity = weakReference.get();
|
||||
|
||||
if (activity != null && activity.isFinishing()) {
|
||||
return null;
|
||||
} else {
|
||||
return activity;
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
protected Integer doInBackground(Object... objects) {
|
||||
context = (Context) objects[0];
|
||||
license = (License) objects[1];
|
||||
license = (License) objects[0];
|
||||
return 1;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onPostExecute(Integer result){
|
||||
String webViewData = getFormattedLicense(context, license);
|
||||
AlertDialog.Builder alert = new AlertDialog.Builder(context);
|
||||
protected void onPostExecute(Integer result) {
|
||||
Activity activity = getActivity();
|
||||
if (activity == null) {
|
||||
return;
|
||||
}
|
||||
|
||||
String webViewData = getFormattedLicense(activity, license);
|
||||
AlertDialog.Builder alert = new AlertDialog.Builder(activity);
|
||||
alert.setTitle(license.getName());
|
||||
|
||||
WebView wv = new WebView(context);
|
||||
WebView wv = new WebView(activity);
|
||||
wv.loadData(webViewData, "text/html; charset=UTF-8", null);
|
||||
|
||||
alert.setView(wv);
|
||||
@@ -55,18 +77,18 @@ public class LicenseFragmentHelper extends AsyncTask<Object, Void, Integer> {
|
||||
throw new NullPointerException("license is null");
|
||||
}
|
||||
|
||||
String licenseContent = "";
|
||||
StringBuilder licenseContent = new StringBuilder();
|
||||
String webViewData;
|
||||
try {
|
||||
BufferedReader in = new BufferedReader(new InputStreamReader(context.getAssets().open(license.getFilename()), "UTF-8"));
|
||||
String str;
|
||||
while ((str = in.readLine()) != null) {
|
||||
licenseContent += str;
|
||||
licenseContent.append(str);
|
||||
}
|
||||
in.close();
|
||||
|
||||
// split the HTML file and insert the stylesheet into the HEAD of the file
|
||||
String[] insert = licenseContent.split("</head>");
|
||||
String[] insert = licenseContent.toString().split("</head>");
|
||||
webViewData = insert[0] + "<style type=\"text/css\">"
|
||||
+ getLicenseStylesheet(context) + "</style></head>"
|
||||
+ insert[1];
|
||||
|
@@ -4,23 +4,52 @@ import android.arch.persistence.room.Database;
|
||||
import android.arch.persistence.room.RoomDatabase;
|
||||
import android.arch.persistence.room.TypeConverters;
|
||||
|
||||
import org.schabi.newpipe.database.history.Converters;
|
||||
import org.schabi.newpipe.database.history.dao.SearchHistoryDAO;
|
||||
import org.schabi.newpipe.database.history.dao.WatchHistoryDAO;
|
||||
import org.schabi.newpipe.database.history.dao.StreamHistoryDAO;
|
||||
import org.schabi.newpipe.database.history.model.SearchHistoryEntry;
|
||||
import org.schabi.newpipe.database.history.model.WatchHistoryEntry;
|
||||
import org.schabi.newpipe.database.history.model.StreamHistoryEntity;
|
||||
import org.schabi.newpipe.database.playlist.dao.PlaylistDAO;
|
||||
import org.schabi.newpipe.database.playlist.dao.PlaylistRemoteDAO;
|
||||
import org.schabi.newpipe.database.playlist.dao.PlaylistStreamDAO;
|
||||
import org.schabi.newpipe.database.playlist.model.PlaylistEntity;
|
||||
import org.schabi.newpipe.database.playlist.model.PlaylistRemoteEntity;
|
||||
import org.schabi.newpipe.database.playlist.model.PlaylistStreamEntity;
|
||||
import org.schabi.newpipe.database.stream.dao.StreamDAO;
|
||||
import org.schabi.newpipe.database.stream.dao.StreamStateDAO;
|
||||
import org.schabi.newpipe.database.stream.model.StreamEntity;
|
||||
import org.schabi.newpipe.database.stream.model.StreamStateEntity;
|
||||
import org.schabi.newpipe.database.subscription.SubscriptionDAO;
|
||||
import org.schabi.newpipe.database.subscription.SubscriptionEntity;
|
||||
|
||||
import static org.schabi.newpipe.database.Migrations.DB_VER_12_0;
|
||||
|
||||
@TypeConverters({Converters.class})
|
||||
@Database(entities = {SubscriptionEntity.class, WatchHistoryEntry.class, SearchHistoryEntry.class}, version = 1, exportSchema = false)
|
||||
@Database(
|
||||
entities = {
|
||||
SubscriptionEntity.class, SearchHistoryEntry.class,
|
||||
StreamEntity.class, StreamHistoryEntity.class, StreamStateEntity.class,
|
||||
PlaylistEntity.class, PlaylistStreamEntity.class, PlaylistRemoteEntity.class
|
||||
},
|
||||
version = DB_VER_12_0,
|
||||
exportSchema = false
|
||||
)
|
||||
public abstract class AppDatabase extends RoomDatabase {
|
||||
|
||||
public static final String DATABASE_NAME = "newpipe.db";
|
||||
|
||||
public abstract SubscriptionDAO subscriptionDAO();
|
||||
|
||||
public abstract WatchHistoryDAO watchHistoryDAO();
|
||||
|
||||
public abstract SearchHistoryDAO searchHistoryDAO();
|
||||
|
||||
public abstract StreamDAO streamDAO();
|
||||
|
||||
public abstract StreamHistoryDAO streamHistoryDAO();
|
||||
|
||||
public abstract StreamStateDAO streamStateDAO();
|
||||
|
||||
public abstract PlaylistDAO playlistDAO();
|
||||
|
||||
public abstract PlaylistStreamDAO playlistStreamDAO();
|
||||
|
||||
public abstract PlaylistRemoteDAO playlistRemoteDAO();
|
||||
}
|
||||
|
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user