Compare commits
338 Commits
v6.5.2
...
feature/st
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
5d1a2fcf77 | ||
|
|
876675445d | ||
|
|
f7a70680bd | ||
|
|
8e82db441c | ||
|
|
f2626c668d | ||
|
|
4694215ccc | ||
|
|
096f5b15dc | ||
|
|
101d357eff | ||
|
|
11cd5609bb | ||
|
|
fda59996aa | ||
|
|
20461112d2 | ||
|
|
f98bb87d6e | ||
|
|
c451952a1e | ||
|
|
f8cbc9692f | ||
|
|
9f3113363b | ||
|
|
dba36838d4 | ||
|
|
f6de1b02d7 | ||
|
|
d6b8e2fd9e | ||
|
|
5227240478 | ||
|
|
8f65ea6535 | ||
|
|
7d7a6eadd2 | ||
|
|
40f1ad3181 | ||
|
|
a28c9447d7 | ||
|
|
a84cf97982 | ||
|
|
3a8eb58fd1 | ||
|
|
5d75e9af4a | ||
|
|
d4684e7462 | ||
|
|
c0a2f0b533 | ||
|
|
40867dd2b6 | ||
|
|
c3294e6459 | ||
|
|
5139feb51a | ||
|
|
6b1240fccb | ||
|
|
e00a5b7505 | ||
|
|
2c07d2c8e1 | ||
|
|
45c3c05f01 | ||
|
|
e97a745713 | ||
|
|
2dc4de0a3c | ||
|
|
3cf2c58058 | ||
|
|
1e19f32fc5 | ||
|
|
99e4359523 | ||
|
|
04868488cc | ||
|
|
2b3b406b84 | ||
|
|
7ab3c75232 | ||
|
|
61f7755465 | ||
|
|
9389015ab9 | ||
|
|
bc56a94aa6 | ||
|
|
7cfcaec6dd | ||
|
|
39c7ae31cd | ||
|
|
9349eccc0c | ||
|
|
8204934359 | ||
|
|
b5497c571e | ||
|
|
35a2ac4b04 | ||
|
|
b4d52f1367 | ||
|
|
325a8be484 | ||
|
|
f39ccb6223 | ||
|
|
6cb6c891dd | ||
|
|
8cc04b0f7a | ||
|
|
258dbf3dc3 | ||
|
|
e7af4e8450 | ||
|
|
0c25c61858 | ||
|
|
abc3e45907 | ||
|
|
bd98d8eded | ||
|
|
2e81f41073 | ||
|
|
5cccebc416 | ||
|
|
c668ffd555 | ||
|
|
a0f77b715f | ||
|
|
2831843a25 | ||
|
|
86c1aa11b0 | ||
|
|
d71514ec7a | ||
|
|
92ed320f57 | ||
|
|
2de1fe8b77 | ||
|
|
cebc3cd9e8 | ||
|
|
6c0e2e2b90 | ||
|
|
b4bd923ce8 | ||
|
|
813561fd3b | ||
|
|
4107336132 | ||
|
|
30d9d87c17 | ||
|
|
c4b5be657d | ||
|
|
8a763b2b9f | ||
|
|
c783378022 | ||
|
|
c4355f16e8 | ||
|
|
522dfc2418 | ||
|
|
06d03e3ddd | ||
|
|
9dc8c7959d | ||
|
|
db219020ca | ||
|
|
c04edcb76c | ||
|
|
936fc2e4ae | ||
|
|
cbed866665 | ||
|
|
ac568b6361 | ||
|
|
84157f988d | ||
|
|
6f6339f0f8 | ||
|
|
a7019b9096 | ||
|
|
867e3f10ca | ||
|
|
fb2cf04d75 | ||
|
|
3ed44ba0d6 | ||
|
|
b78104a0f1 | ||
|
|
e4ee93f77c | ||
|
|
c6e8da5f23 | ||
|
|
376de7cce3 | ||
|
|
bec2195971 | ||
|
|
722ac4ecc7 | ||
|
|
516c1c02a6 | ||
|
|
0cb7e71781 | ||
|
|
36a74f32df | ||
|
|
0e4ef32642 | ||
|
|
3125cac4c8 | ||
|
|
5d9016d1bc | ||
|
|
c5eeb89d10 | ||
|
|
4f8f43cab1 | ||
|
|
4cbff308ce | ||
|
|
d786ab7deb | ||
|
|
c823d402ff | ||
|
|
12e68db41f | ||
|
|
96717321d2 | ||
|
|
044b5590ef | ||
|
|
00112ebb44 | ||
|
|
00dde80fdf | ||
|
|
f1dfc4ebd6 | ||
|
|
5426edd83a | ||
|
|
2a500eb2cb | ||
|
|
2310ed06c1 | ||
|
|
68ed7a09d6 | ||
|
|
6cdb56e740 | ||
|
|
35fb78c924 | ||
|
|
8c3b5d7f53 | ||
|
|
af6592a8df | ||
|
|
9efb82d887 | ||
|
|
f8722ddc73 | ||
|
|
656ac97153 | ||
|
|
4fc23f8f54 | ||
|
|
e1f325993f | ||
|
|
4c3938a1fd | ||
|
|
530dfa8cde | ||
|
|
58d1c3de26 | ||
|
|
ba2ed6a2ef | ||
|
|
2d909854fb | ||
|
|
cba694bedd | ||
|
|
e5cf1be91a | ||
|
|
72a1dd8227 | ||
|
|
8558b00dca | ||
|
|
8e9175d5f0 | ||
|
|
eae40d9b90 | ||
|
|
2d61209696 | ||
|
|
d24754f2a0 | ||
|
|
54ef02ad88 | ||
|
|
e2a82920b6 | ||
|
|
d494030d50 | ||
|
|
73369f9a6d | ||
|
|
cc1da6e8da | ||
|
|
668a5bd040 | ||
|
|
8efa8bc0d2 | ||
|
|
6e6c70a770 | ||
|
|
413605b520 | ||
|
|
bdf23a0d62 | ||
|
|
4c5d26d4b4 | ||
|
|
3b7ad7f28d | ||
|
|
331af45a29 | ||
|
|
d349bd30c9 | ||
|
|
3349e3abc5 | ||
|
|
4aa31ead67 | ||
|
|
113da3b6c1 | ||
|
|
8b027e2f45 | ||
|
|
9c462b1a3a | ||
|
|
a5bc8c1e9e | ||
|
|
ebb77c68cc | ||
|
|
74ddf86ebe | ||
|
|
12d2fdaf3e | ||
|
|
8cfc97c795 | ||
|
|
3855ca802e | ||
|
|
9db427275f | ||
|
|
3a38644089 | ||
|
|
60a34ec092 | ||
|
|
acd79f12e3 | ||
|
|
461d7ed578 | ||
|
|
5374ac390c | ||
|
|
913a67a652 | ||
|
|
e7a920e43a | ||
|
|
9668b3ef5f | ||
|
|
9581f937de | ||
|
|
44ef6f6dbf | ||
|
|
af11697133 | ||
|
|
09ff356790 | ||
|
|
92ea50d6b6 | ||
|
|
077107e9a7 | ||
|
|
ae57561591 | ||
|
|
2379efc191 | ||
|
|
edca0e5334 | ||
|
|
a4e2675d61 | ||
|
|
892f95a7a6 | ||
|
|
95aaa967a8 | ||
|
|
5687ca6e96 | ||
|
|
d0ee185d2e | ||
|
|
21a3ac0902 | ||
|
|
1382ab7933 | ||
|
|
aabdd281f3 | ||
|
|
131a0ffcaa | ||
|
|
4194609929 | ||
|
|
889b799d8d | ||
|
|
6f7f3dc5e2 | ||
|
|
72187e7da0 | ||
|
|
f881cc439a | ||
|
|
ccdebf6789 | ||
|
|
4252ebd24d | ||
|
|
4db61d3c04 | ||
|
|
cd0575a524 | ||
|
|
6eb2608f88 | ||
|
|
39e21ff93c | ||
|
|
5ec2eab6b8 | ||
|
|
850f6c2f3e | ||
|
|
ec53eb9c70 | ||
|
|
cdd76f723f | ||
|
|
e7c9d1943d | ||
|
|
b1240e7efa | ||
|
|
a0a72b1192 | ||
|
|
5d9a59d577 | ||
|
|
83cb35fe6e | ||
|
|
0fff53ae47 | ||
|
|
a95017a5f0 | ||
|
|
9251823d9a | ||
|
|
ce8f87272b | ||
|
|
db1ddf539c | ||
|
|
d56fc674ab | ||
|
|
a37e8825b0 | ||
|
|
c9fcc0f0f8 | ||
|
|
2450544454 | ||
|
|
f6a510653e | ||
|
|
5990da587c | ||
|
|
91e3d2f5db | ||
|
|
971c683746 | ||
|
|
15e9aaab26 | ||
|
|
da2ad40adf | ||
|
|
af5716a8ce | ||
|
|
a98202e15e | ||
|
|
d6887e2d75 | ||
|
|
ba6afd44dd | ||
|
|
0b55c4d037 | ||
|
|
2a5300a634 | ||
|
|
59bfa929fd | ||
|
|
c5d88f8700 | ||
|
|
a1120ea709 | ||
|
|
796af6b811 | ||
|
|
eafd878413 | ||
|
|
9baf2bfcd9 | ||
|
|
0b4dd5beef | ||
|
|
12047a85c7 | ||
|
|
48808f8a7d | ||
|
|
18e573b6b8 | ||
|
|
c80dc08d6c | ||
|
|
61b5b8aa73 | ||
|
|
5e79809326 | ||
|
|
dcbd7c2117 | ||
|
|
1e134b109a | ||
|
|
7f9b6a67af | ||
|
|
79448bb01d | ||
|
|
b94f9e4b01 | ||
|
|
ae8b48d733 | ||
|
|
313013dccd | ||
|
|
c36d23ec06 | ||
|
|
ebe71476d1 | ||
|
|
ca2ae9bc83 | ||
|
|
f2898aba85 | ||
|
|
1b73f19ae1 | ||
|
|
4b4aea0410 | ||
|
|
b9f8e3978a | ||
|
|
89eed10508 | ||
|
|
799c0910ea | ||
|
|
ed6802344a | ||
|
|
2d94688742 | ||
|
|
d0b44050f5 | ||
|
|
486ae12d41 | ||
|
|
cb36f085d7 | ||
|
|
5af42e5b6b | ||
|
|
160fa2c001 | ||
|
|
f9ba87b8cf | ||
|
|
ee3cf08545 | ||
|
|
e0497b357b | ||
|
|
6de08cd4fe | ||
|
|
4760f1ea35 | ||
|
|
c64c4643bf | ||
|
|
7bfcdb387c | ||
|
|
503fcf65fb | ||
|
|
54fb79dc98 | ||
|
|
ea4c048029 | ||
|
|
badc826cd3 | ||
|
|
f5ece8124e | ||
|
|
accdc41d6c | ||
|
|
819730984e | ||
|
|
01c404f9e5 | ||
|
|
1fad686733 | ||
|
|
396be6008d | ||
|
|
42f7846167 | ||
|
|
dca56a43ee | ||
|
|
cc91e56e1b | ||
|
|
627cf73d72 | ||
|
|
514870f71c | ||
|
|
adffa800e8 | ||
|
|
3acca44b5e | ||
|
|
c7da4feb8f | ||
|
|
baee9bee0e | ||
|
|
ec41d36508 | ||
|
|
8b63d227a7 | ||
|
|
c9b48c8207 | ||
|
|
6d7ce5205e | ||
|
|
5a02d534c9 | ||
|
|
6128e5b699 | ||
|
|
717a0ad4fb | ||
|
|
dee94ac0c4 | ||
|
|
9eec9a9957 | ||
|
|
a4966b4661 | ||
|
|
58e570601d | ||
|
|
7247cba855 | ||
|
|
d6012f9ddd | ||
|
|
2eedd0b4a8 | ||
|
|
5e6da9bb1c | ||
|
|
2f2a5b868d | ||
|
|
3f2e32dcc2 | ||
|
|
004109a6bc | ||
|
|
6159ee36c4 | ||
|
|
3b7d83dd6f | ||
|
|
877a018ced | ||
|
|
2e80b330e9 | ||
|
|
42ca38e693 | ||
|
|
d2fc3354af | ||
|
|
2a870e6167 | ||
|
|
393a9c2791 | ||
|
|
4c69839076 | ||
|
|
e37455e790 | ||
|
|
36259ba901 | ||
|
|
5b041b9a49 | ||
|
|
1734e888d6 | ||
|
|
9108646cea | ||
|
|
fd01367601 | ||
|
|
cb64740349 | ||
|
|
6fdcaf0d02 | ||
|
|
56de725cf1 | ||
|
|
7a2ad47405 | ||
|
|
41551451b0 | ||
|
|
d5c24cd5c8 |
1
.gitignore
vendored
1
.gitignore
vendored
@@ -10,6 +10,7 @@
|
|||||||
/.idea/compiler.xml
|
/.idea/compiler.xml
|
||||||
/.idea/workspace.xml
|
/.idea/workspace.xml
|
||||||
/.idea/navEditor.xml
|
/.idea/navEditor.xml
|
||||||
|
/.idea/ktlint-plugin.xml
|
||||||
/.idea/assetWizardSettings.xml
|
/.idea/assetWizardSettings.xml
|
||||||
/.idea/kotlinScripting.xml
|
/.idea/kotlinScripting.xml
|
||||||
/.idea/kotlinc.xml
|
/.idea/kotlinc.xml
|
||||||
|
|||||||
@@ -1,11 +1,12 @@
|
|||||||
## Kotatsu contribution guidelines
|
## Kotatsu contribution guidelines
|
||||||
|
|
||||||
- If you want to fix bug or implement a new feature, that already mention in the [issues](https://github.com/KotatsuApp/Kotatsu/issues), please, assign this issue to you and/or comment about it.
|
+ If you want to **fix bugs** or **implement new features** that **already have an [issue card](https://github.com/KotatsuApp/Kotatsu/issues):** please assign this issue to you and/or comment about it.
|
||||||
- Whether you have to implement new feature, please, open an issue or discussion regarding it to ensure it will be accepted.
|
+ If you want to **implement a new feature:** open an issue or discussion regarding it to ensure it will be accepted.
|
||||||
- Translations have to be managed using the [Weblate](https://hosted.weblate.org/engage/kotatsu/) platform.
|
+ **Translations** have to be managed using the [Weblate](https://hosted.weblate.org/engage/kotatsu/) platform.
|
||||||
- In case you want to add a new manga source, refer to the [parsers repository](https://github.com/KotatsuApp/kotatsu-parsers).
|
+ In case you want to **add a new manga source,** refer to the [parsers repository](https://github.com/KotatsuApp/kotatsu-parsers).
|
||||||
|
|
||||||
Refactoring or some dev-faces improvements are also might be accepted, however please stick to the following principles:
|
**Refactoring** or some **dev-faces improvements** might also be accepted. However, please stick to the following principles:
|
||||||
- Performance matters. In the case of choosing between source code beauty and performance, performance should be a priority.
|
|
||||||
- Please, do not modify readme and other information files (except for typos).
|
+ **Performance matters.** In the case of choosing between source code beauty and performance, performance should be a priority.
|
||||||
- Avoid adding new dependencies unless required. APK size is important.
|
+ Please, **do not modify readme and other information files** (except for typos).
|
||||||
|
+ **Avoid adding new dependencies** unless required. APK size is important.
|
||||||
|
|||||||
@@ -19,7 +19,7 @@ Kotatsu is a free and open source manga reader for Android.
|
|||||||
* Tablet-optimized Material You UI
|
* Tablet-optimized Material You UI
|
||||||
* Standard and Webtoon-optimized reader
|
* Standard and Webtoon-optimized reader
|
||||||
* Notifications about new chapters with updates feed
|
* Notifications about new chapters with updates feed
|
||||||
* Integration with manga tracking services: Shikimori, AniList, MyAnimeList
|
* Integration with manga tracking services: Shikimori, AniList, MyAnimeList, Kitsu
|
||||||
* Password/fingerprint protect access to the app
|
* Password/fingerprint protect access to the app
|
||||||
* History and favourites [synchronization](https://github.com/KotatsuApp/kotatsu-syncserver) across devices
|
* History and favourites [synchronization](https://github.com/KotatsuApp/kotatsu-syncserver) across devices
|
||||||
|
|
||||||
|
|||||||
@@ -16,8 +16,8 @@ android {
|
|||||||
applicationId 'org.koitharu.kotatsu'
|
applicationId 'org.koitharu.kotatsu'
|
||||||
minSdk = 21
|
minSdk = 21
|
||||||
targetSdk = 34
|
targetSdk = 34
|
||||||
versionCode = 608
|
versionCode = 626
|
||||||
versionName = '6.5.2'
|
versionName = '6.7.4'
|
||||||
generatedDensities = []
|
generatedDensities = []
|
||||||
testInstrumentationRunner 'org.koitharu.kotatsu.HiltTestRunner'
|
testInstrumentationRunner 'org.koitharu.kotatsu.HiltTestRunner'
|
||||||
ksp {
|
ksp {
|
||||||
@@ -82,29 +82,30 @@ afterEvaluate {
|
|||||||
}
|
}
|
||||||
dependencies {
|
dependencies {
|
||||||
//noinspection GradleDependency
|
//noinspection GradleDependency
|
||||||
implementation('com.github.KotatsuApp:kotatsu-parsers:4a0e7221b0') {
|
implementation('com.github.KotatsuApp:kotatsu-parsers:103f578c61') {
|
||||||
exclude group: 'org.json', module: 'json'
|
exclude group: 'org.json', module: 'json'
|
||||||
}
|
}
|
||||||
|
|
||||||
coreLibraryDesugaring 'com.android.tools:desugar_jdk_libs:2.0.4'
|
coreLibraryDesugaring 'com.android.tools:desugar_jdk_libs:2.0.4'
|
||||||
implementation 'org.jetbrains.kotlin:kotlin-stdlib:1.9.21'
|
implementation 'org.jetbrains.kotlin:kotlin-stdlib:1.9.22'
|
||||||
implementation 'org.jetbrains.kotlinx:kotlinx-coroutines-android:1.7.3'
|
implementation 'org.jetbrains.kotlinx:kotlinx-coroutines-android:1.8.0'
|
||||||
|
|
||||||
implementation 'androidx.appcompat:appcompat:1.6.1'
|
implementation 'androidx.appcompat:appcompat:1.6.1'
|
||||||
implementation 'androidx.core:core-ktx:1.12.0'
|
implementation 'androidx.core:core-ktx:1.12.0'
|
||||||
implementation 'androidx.activity:activity-ktx:1.8.2'
|
implementation 'androidx.activity:activity-ktx:1.8.2'
|
||||||
implementation 'androidx.fragment:fragment-ktx:1.6.2'
|
implementation 'androidx.fragment:fragment-ktx:1.6.2'
|
||||||
implementation 'androidx.lifecycle:lifecycle-viewmodel-ktx:2.6.2'
|
implementation 'androidx.collection:collection:1.4.0'
|
||||||
implementation 'androidx.lifecycle:lifecycle-service:2.6.2'
|
implementation 'androidx.lifecycle:lifecycle-viewmodel-ktx:2.7.0'
|
||||||
implementation 'androidx.lifecycle:lifecycle-process:2.6.2'
|
implementation 'androidx.lifecycle:lifecycle-service:2.7.0'
|
||||||
|
implementation 'androidx.lifecycle:lifecycle-process:2.7.0'
|
||||||
implementation 'androidx.constraintlayout:constraintlayout:2.1.4'
|
implementation 'androidx.constraintlayout:constraintlayout:2.1.4'
|
||||||
implementation 'androidx.swiperefreshlayout:swiperefreshlayout:1.1.0'
|
implementation 'androidx.swiperefreshlayout:swiperefreshlayout:1.1.0'
|
||||||
implementation 'androidx.recyclerview:recyclerview:1.3.2'
|
implementation 'androidx.recyclerview:recyclerview:1.3.2'
|
||||||
implementation 'androidx.viewpager2:viewpager2:1.1.0-beta02'
|
implementation 'androidx.viewpager2:viewpager2:1.1.0-beta02'
|
||||||
implementation 'androidx.preference:preference-ktx:1.2.1'
|
implementation 'androidx.preference:preference-ktx:1.2.1'
|
||||||
implementation 'androidx.biometric:biometric-ktx:1.2.0-alpha05'
|
implementation 'androidx.biometric:biometric-ktx:1.2.0-alpha05'
|
||||||
implementation 'com.google.android.material:material:1.11.0'
|
implementation 'com.google.android.material:material:1.12.0-alpha03'
|
||||||
implementation 'androidx.lifecycle:lifecycle-common-java8:2.6.2'
|
implementation 'androidx.lifecycle:lifecycle-common-java8:2.7.0'
|
||||||
|
|
||||||
implementation 'androidx.work:work-runtime:2.9.0'
|
implementation 'androidx.work:work-runtime:2.9.0'
|
||||||
//noinspection GradleDependency
|
//noinspection GradleDependency
|
||||||
@@ -120,41 +121,45 @@ dependencies {
|
|||||||
|
|
||||||
implementation 'com.squareup.okhttp3:okhttp:4.12.0'
|
implementation 'com.squareup.okhttp3:okhttp:4.12.0'
|
||||||
implementation 'com.squareup.okhttp3:okhttp-dnsoverhttps:4.12.0'
|
implementation 'com.squareup.okhttp3:okhttp-dnsoverhttps:4.12.0'
|
||||||
implementation 'com.squareup.okio:okio:3.7.0'
|
implementation 'com.squareup.okio:okio:3.8.0'
|
||||||
|
|
||||||
implementation 'com.hannesdorfmann:adapterdelegates4-kotlin-dsl:4.3.2'
|
implementation 'com.hannesdorfmann:adapterdelegates4-kotlin-dsl:4.3.2'
|
||||||
implementation 'com.hannesdorfmann:adapterdelegates4-kotlin-dsl-viewbinding:4.3.2'
|
implementation 'com.hannesdorfmann:adapterdelegates4-kotlin-dsl-viewbinding:4.3.2'
|
||||||
|
|
||||||
implementation 'com.google.dagger:hilt-android:2.50'
|
implementation 'com.google.dagger:hilt-android:2.51'
|
||||||
kapt 'com.google.dagger:hilt-compiler:2.50'
|
kapt 'com.google.dagger:hilt-compiler:2.51'
|
||||||
implementation 'androidx.hilt:hilt-work:1.1.0'
|
implementation 'androidx.hilt:hilt-work:1.2.0'
|
||||||
kapt 'androidx.hilt:hilt-compiler:1.1.0'
|
kapt 'androidx.hilt:hilt-compiler:1.2.0'
|
||||||
|
|
||||||
implementation 'io.coil-kt:coil-base:2.5.0'
|
implementation 'io.coil-kt:coil-base:2.6.0'
|
||||||
implementation 'io.coil-kt:coil-svg:2.5.0'
|
implementation 'io.coil-kt:coil-svg:2.6.0'
|
||||||
implementation 'com.github.KotatsuApp:subsampling-scale-image-view:02e6d6cfe9'
|
implementation 'com.github.KotatsuApp:subsampling-scale-image-view:02e6d6cfe9'
|
||||||
implementation 'com.github.solkin:disk-lru-cache:1.4'
|
implementation 'com.github.solkin:disk-lru-cache:1.4'
|
||||||
implementation 'io.noties.markwon:core:4.6.2'
|
implementation 'io.noties.markwon:core:4.6.2'
|
||||||
|
|
||||||
implementation 'ch.acra:acra-http:5.11.3'
|
implementation 'ch.acra:acra-http:5.11.3'
|
||||||
implementation 'ch.acra:acra-dialog:5.11.3'
|
implementation 'ch.acra:acra-dialog:5.11.3'
|
||||||
|
compileOnly 'com.google.auto.service:auto-service-annotations:1.1.1'
|
||||||
|
ksp 'dev.zacsweers.autoservice:auto-service-ksp:1.1.0'
|
||||||
|
|
||||||
debugImplementation 'com.squareup.leakcanary:leakcanary-android:2.12'
|
implementation 'org.conscrypt:conscrypt-android:2.5.2'
|
||||||
|
|
||||||
|
debugImplementation 'com.squareup.leakcanary:leakcanary-android:2.13'
|
||||||
|
|
||||||
testImplementation 'junit:junit:4.13.2'
|
testImplementation 'junit:junit:4.13.2'
|
||||||
testImplementation 'org.json:json:20231013'
|
testImplementation 'org.json:json:20240205'
|
||||||
testImplementation 'org.jetbrains.kotlinx:kotlinx-coroutines-test:1.7.3'
|
testImplementation 'org.jetbrains.kotlinx:kotlinx-coroutines-test:1.8.0'
|
||||||
|
|
||||||
androidTestImplementation 'androidx.test:runner:1.5.2'
|
androidTestImplementation 'androidx.test:runner:1.5.2'
|
||||||
androidTestImplementation 'androidx.test:rules:1.5.0'
|
androidTestImplementation 'androidx.test:rules:1.5.0'
|
||||||
androidTestImplementation 'androidx.test:core-ktx:1.5.0'
|
androidTestImplementation 'androidx.test:core-ktx:1.5.0'
|
||||||
androidTestImplementation 'androidx.test.ext:junit-ktx:1.1.5'
|
androidTestImplementation 'androidx.test.ext:junit-ktx:1.1.5'
|
||||||
|
|
||||||
androidTestImplementation 'org.jetbrains.kotlinx:kotlinx-coroutines-test:1.7.3'
|
androidTestImplementation 'org.jetbrains.kotlinx:kotlinx-coroutines-test:1.8.0'
|
||||||
|
|
||||||
androidTestImplementation 'androidx.room:room-testing:2.6.1'
|
androidTestImplementation 'androidx.room:room-testing:2.6.1'
|
||||||
androidTestImplementation 'com.squareup.moshi:moshi-kotlin:1.15.0'
|
androidTestImplementation 'com.squareup.moshi:moshi-kotlin:1.15.1'
|
||||||
|
|
||||||
androidTestImplementation 'com.google.dagger:hilt-android-testing:2.50'
|
androidTestImplementation 'com.google.dagger:hilt-android-testing:2.51'
|
||||||
kaptAndroidTest 'com.google.dagger:hilt-android-compiler:2.50'
|
kaptAndroidTest 'com.google.dagger:hilt-android-compiler:2.51'
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -57,6 +57,7 @@ class AppShortcutManagerTest {
|
|||||||
page = 4,
|
page = 4,
|
||||||
scroll = 2,
|
scroll = 2,
|
||||||
percent = 0.3f,
|
percent = 0.3f,
|
||||||
|
force = false,
|
||||||
)
|
)
|
||||||
awaitUpdate()
|
awaitUpdate()
|
||||||
|
|
||||||
|
|||||||
@@ -7,7 +7,9 @@ import dagger.hilt.android.testing.HiltAndroidRule
|
|||||||
import dagger.hilt.android.testing.HiltAndroidTest
|
import dagger.hilt.android.testing.HiltAndroidTest
|
||||||
import kotlinx.coroutines.flow.first
|
import kotlinx.coroutines.flow.first
|
||||||
import kotlinx.coroutines.test.runTest
|
import kotlinx.coroutines.test.runTest
|
||||||
import org.junit.Assert.*
|
import org.junit.Assert.assertEquals
|
||||||
|
import org.junit.Assert.assertNull
|
||||||
|
import org.junit.Assert.assertTrue
|
||||||
import org.junit.Before
|
import org.junit.Before
|
||||||
import org.junit.Rule
|
import org.junit.Rule
|
||||||
import org.junit.Test
|
import org.junit.Test
|
||||||
@@ -61,6 +63,7 @@ class AppBackupAgentTest {
|
|||||||
page = 3,
|
page = 3,
|
||||||
scroll = 40,
|
scroll = 40,
|
||||||
percent = 0.2f,
|
percent = 0.2f,
|
||||||
|
force = false,
|
||||||
)
|
)
|
||||||
val history = checkNotNull(historyRepository.getOne(SampleData.manga))
|
val history = checkNotNull(historyRepository.getOne(SampleData.manga))
|
||||||
|
|
||||||
|
|||||||
@@ -36,7 +36,7 @@ class KotatsuApp : BaseApp() {
|
|||||||
FragmentStrictMode.defaultPolicy = FragmentStrictMode.Policy.Builder()
|
FragmentStrictMode.defaultPolicy = FragmentStrictMode.Policy.Builder()
|
||||||
.penaltyDeath()
|
.penaltyDeath()
|
||||||
.detectFragmentReuse()
|
.detectFragmentReuse()
|
||||||
// .detectWrongFragmentContainer() FIXME: migrate to ViewPager2
|
.detectWrongFragmentContainer()
|
||||||
.detectRetainInstanceUsage()
|
.detectRetainInstanceUsage()
|
||||||
.detectSetUserVisibleHint()
|
.detectSetUserVisibleHint()
|
||||||
.detectFragmentTagUsage()
|
.detectFragmentTagUsage()
|
||||||
|
|||||||
@@ -10,6 +10,8 @@ class CurlLoggingInterceptor(
|
|||||||
private val curlOptions: String? = null
|
private val curlOptions: String? = null
|
||||||
) : Interceptor {
|
) : Interceptor {
|
||||||
|
|
||||||
|
private val escapeRegex = Regex("([\\[\\]\"])")
|
||||||
|
|
||||||
override fun intercept(chain: Interceptor.Chain): Response {
|
override fun intercept(chain: Interceptor.Chain): Response {
|
||||||
val request = chain.request()
|
val request = chain.request()
|
||||||
var isCompressed = false
|
var isCompressed = false
|
||||||
@@ -40,7 +42,7 @@ class CurlLoggingInterceptor(
|
|||||||
if (isCompressed) {
|
if (isCompressed) {
|
||||||
curlCmd.append(" --compressed")
|
curlCmd.append(" --compressed")
|
||||||
}
|
}
|
||||||
curlCmd.append(" \"").append(request.url).append('"')
|
curlCmd.append(" \"").append(request.url.toString().escape()).append('"')
|
||||||
|
|
||||||
log("---cURL (" + request.url + ")")
|
log("---cURL (" + request.url + ")")
|
||||||
log(curlCmd.toString())
|
log(curlCmd.toString())
|
||||||
@@ -48,7 +50,12 @@ class CurlLoggingInterceptor(
|
|||||||
return chain.proceed(request)
|
return chain.proceed(request)
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun String.escape() = replace("\"", "\\\"")
|
private fun String.escape() = replace(escapeRegex) { match ->
|
||||||
|
"\\" + match.value
|
||||||
|
}
|
||||||
|
// .replace("\"", "\\\"")
|
||||||
|
// .replace("[", "\\[")
|
||||||
|
// .replace("]", "\\]")
|
||||||
|
|
||||||
private fun log(msg: String) {
|
private fun log(msg: String) {
|
||||||
Log.d("CURL", msg)
|
Log.d("CURL", msg)
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
@@ -2,7 +2,7 @@ package org.koitharu.kotatsu.bookmarks.data
|
|||||||
|
|
||||||
import org.koitharu.kotatsu.bookmarks.domain.Bookmark
|
import org.koitharu.kotatsu.bookmarks.domain.Bookmark
|
||||||
import org.koitharu.kotatsu.parsers.model.Manga
|
import org.koitharu.kotatsu.parsers.model.Manga
|
||||||
import java.util.Date
|
import java.time.Instant
|
||||||
|
|
||||||
fun BookmarkEntity.toBookmark(manga: Manga) = Bookmark(
|
fun BookmarkEntity.toBookmark(manga: Manga) = Bookmark(
|
||||||
manga = manga,
|
manga = manga,
|
||||||
@@ -11,7 +11,7 @@ fun BookmarkEntity.toBookmark(manga: Manga) = Bookmark(
|
|||||||
page = page,
|
page = page,
|
||||||
scroll = scroll,
|
scroll = scroll,
|
||||||
imageUrl = imageUrl,
|
imageUrl = imageUrl,
|
||||||
createdAt = Date(createdAt),
|
createdAt = Instant.ofEpochMilli(createdAt),
|
||||||
percent = percent,
|
percent = percent,
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -22,7 +22,7 @@ fun Bookmark.toEntity() = BookmarkEntity(
|
|||||||
page = page,
|
page = page,
|
||||||
scroll = scroll,
|
scroll = scroll,
|
||||||
imageUrl = imageUrl,
|
imageUrl = imageUrl,
|
||||||
createdAt = createdAt.time,
|
createdAt = createdAt.toEpochMilli(),
|
||||||
percent = percent,
|
percent = percent,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|||||||
@@ -4,7 +4,7 @@ import org.koitharu.kotatsu.list.ui.model.ListModel
|
|||||||
import org.koitharu.kotatsu.local.data.hasImageExtension
|
import org.koitharu.kotatsu.local.data.hasImageExtension
|
||||||
import org.koitharu.kotatsu.parsers.model.Manga
|
import org.koitharu.kotatsu.parsers.model.Manga
|
||||||
import org.koitharu.kotatsu.parsers.model.MangaPage
|
import org.koitharu.kotatsu.parsers.model.MangaPage
|
||||||
import java.util.Date
|
import java.time.Instant
|
||||||
|
|
||||||
data class Bookmark(
|
data class Bookmark(
|
||||||
val manga: Manga,
|
val manga: Manga,
|
||||||
@@ -13,7 +13,7 @@ data class Bookmark(
|
|||||||
val page: Int,
|
val page: Int,
|
||||||
val scroll: Int,
|
val scroll: Int,
|
||||||
val imageUrl: String,
|
val imageUrl: String,
|
||||||
val createdAt: Date,
|
val createdAt: Instant,
|
||||||
val percent: Float,
|
val percent: Float,
|
||||||
) : ListModel {
|
) : ListModel {
|
||||||
|
|
||||||
|
|||||||
@@ -13,7 +13,6 @@ import org.koitharu.kotatsu.list.ui.adapter.emptyStateListAD
|
|||||||
import org.koitharu.kotatsu.list.ui.adapter.listHeaderAD
|
import org.koitharu.kotatsu.list.ui.adapter.listHeaderAD
|
||||||
import org.koitharu.kotatsu.list.ui.adapter.loadingFooterAD
|
import org.koitharu.kotatsu.list.ui.adapter.loadingFooterAD
|
||||||
import org.koitharu.kotatsu.list.ui.adapter.loadingStateAD
|
import org.koitharu.kotatsu.list.ui.adapter.loadingStateAD
|
||||||
import org.koitharu.kotatsu.list.ui.model.ListHeader
|
|
||||||
import org.koitharu.kotatsu.list.ui.model.ListModel
|
import org.koitharu.kotatsu.list.ui.model.ListModel
|
||||||
|
|
||||||
class BookmarksAdapter(
|
class BookmarksAdapter(
|
||||||
@@ -32,13 +31,6 @@ class BookmarksAdapter(
|
|||||||
}
|
}
|
||||||
|
|
||||||
override fun getSectionText(context: Context, position: Int): CharSequence? {
|
override fun getSectionText(context: Context, position: Int): CharSequence? {
|
||||||
val list = items
|
return findHeader(position)?.getText(context)
|
||||||
for (i in (0..position).reversed()) {
|
|
||||||
val item = list.getOrNull(i) ?: continue
|
|
||||||
if (item is ListHeader) {
|
|
||||||
return item.getText(context)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return null
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -14,7 +14,6 @@ import androidx.core.view.isVisible
|
|||||||
import androidx.core.view.updatePadding
|
import androidx.core.view.updatePadding
|
||||||
import org.koitharu.kotatsu.R
|
import org.koitharu.kotatsu.R
|
||||||
import org.koitharu.kotatsu.core.ui.BaseActivity
|
import org.koitharu.kotatsu.core.ui.BaseActivity
|
||||||
import org.koitharu.kotatsu.core.util.ext.catchingWebViewUnavailability
|
|
||||||
import org.koitharu.kotatsu.databinding.ActivityBrowserBinding
|
import org.koitharu.kotatsu.databinding.ActivityBrowserBinding
|
||||||
import org.koitharu.kotatsu.parsers.network.UserAgents
|
import org.koitharu.kotatsu.parsers.network.UserAgents
|
||||||
import com.google.android.material.R as materialR
|
import com.google.android.material.R as materialR
|
||||||
@@ -26,7 +25,7 @@ class BrowserActivity : BaseActivity<ActivityBrowserBinding>(), BrowserCallback
|
|||||||
|
|
||||||
override fun onCreate(savedInstanceState: Bundle?) {
|
override fun onCreate(savedInstanceState: Bundle?) {
|
||||||
super.onCreate(savedInstanceState)
|
super.onCreate(savedInstanceState)
|
||||||
if (!catchingWebViewUnavailability { setContentView(ActivityBrowserBinding.inflate(layoutInflater)) }) {
|
if (!setContentViewWebViewSafe { ActivityBrowserBinding.inflate(layoutInflater) }) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
supportActionBar?.run {
|
supportActionBar?.run {
|
||||||
|
|||||||
@@ -13,6 +13,7 @@ import org.koitharu.kotatsu.R
|
|||||||
import org.koitharu.kotatsu.core.exceptions.CloudFlareProtectedException
|
import org.koitharu.kotatsu.core.exceptions.CloudFlareProtectedException
|
||||||
import org.koitharu.kotatsu.core.util.ext.checkNotificationPermission
|
import org.koitharu.kotatsu.core.util.ext.checkNotificationPermission
|
||||||
import org.koitharu.kotatsu.parsers.model.ContentType
|
import org.koitharu.kotatsu.parsers.model.ContentType
|
||||||
|
import org.koitharu.kotatsu.parsers.model.MangaSource
|
||||||
|
|
||||||
class CaptchaNotifier(
|
class CaptchaNotifier(
|
||||||
private val context: Context,
|
private val context: Context,
|
||||||
@@ -58,6 +59,10 @@ class CaptchaNotifier(
|
|||||||
manager.notify(TAG, exception.source.hashCode(), notification)
|
manager.notify(TAG, exception.source.hashCode(), notification)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fun dismiss(source: MangaSource) {
|
||||||
|
NotificationManagerCompat.from(context).cancel(TAG, source.hashCode())
|
||||||
|
}
|
||||||
|
|
||||||
override fun onError(request: ImageRequest, result: ErrorResult) {
|
override fun onError(request: ImageRequest, result: ErrorResult) {
|
||||||
super.onError(request, result)
|
super.onError(request, result)
|
||||||
val e = result.throwable
|
val e = result.throwable
|
||||||
|
|||||||
@@ -27,7 +27,6 @@ import org.koitharu.kotatsu.core.network.CommonHeaders
|
|||||||
import org.koitharu.kotatsu.core.network.cookies.MutableCookieJar
|
import org.koitharu.kotatsu.core.network.cookies.MutableCookieJar
|
||||||
import org.koitharu.kotatsu.core.ui.BaseActivity
|
import org.koitharu.kotatsu.core.ui.BaseActivity
|
||||||
import org.koitharu.kotatsu.core.util.TaggedActivityResult
|
import org.koitharu.kotatsu.core.util.TaggedActivityResult
|
||||||
import org.koitharu.kotatsu.core.util.ext.catchingWebViewUnavailability
|
|
||||||
import org.koitharu.kotatsu.databinding.ActivityBrowserBinding
|
import org.koitharu.kotatsu.databinding.ActivityBrowserBinding
|
||||||
import org.koitharu.kotatsu.parsers.network.UserAgents
|
import org.koitharu.kotatsu.parsers.network.UserAgents
|
||||||
import javax.inject.Inject
|
import javax.inject.Inject
|
||||||
@@ -45,13 +44,7 @@ class CloudFlareActivity : BaseActivity<ActivityBrowserBinding>(), CloudFlareCal
|
|||||||
|
|
||||||
override fun onCreate(savedInstanceState: Bundle?) {
|
override fun onCreate(savedInstanceState: Bundle?) {
|
||||||
super.onCreate(savedInstanceState)
|
super.onCreate(savedInstanceState)
|
||||||
if (!catchingWebViewUnavailability {
|
if (!setContentViewWebViewSafe { ActivityBrowserBinding.inflate(layoutInflater) }) {
|
||||||
setContentView(
|
|
||||||
ActivityBrowserBinding.inflate(
|
|
||||||
layoutInflater,
|
|
||||||
),
|
|
||||||
)
|
|
||||||
}) {
|
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
supportActionBar?.run {
|
supportActionBar?.run {
|
||||||
|
|||||||
@@ -2,6 +2,7 @@ package org.koitharu.kotatsu.core
|
|||||||
|
|
||||||
import android.app.Application
|
import android.app.Application
|
||||||
import android.content.Context
|
import android.content.Context
|
||||||
|
import android.os.Build
|
||||||
import androidx.annotation.WorkerThread
|
import androidx.annotation.WorkerThread
|
||||||
import androidx.appcompat.app.AppCompatDelegate
|
import androidx.appcompat.app.AppCompatDelegate
|
||||||
import androidx.hilt.work.HiltWorkerFactory
|
import androidx.hilt.work.HiltWorkerFactory
|
||||||
@@ -19,6 +20,7 @@ import org.acra.config.httpSender
|
|||||||
import org.acra.data.StringFormat
|
import org.acra.data.StringFormat
|
||||||
import org.acra.ktx.initAcra
|
import org.acra.ktx.initAcra
|
||||||
import org.acra.sender.HttpSender
|
import org.acra.sender.HttpSender
|
||||||
|
import org.conscrypt.Conscrypt
|
||||||
import org.koitharu.kotatsu.BuildConfig
|
import org.koitharu.kotatsu.BuildConfig
|
||||||
import org.koitharu.kotatsu.R
|
import org.koitharu.kotatsu.R
|
||||||
import org.koitharu.kotatsu.core.db.MangaDatabase
|
import org.koitharu.kotatsu.core.db.MangaDatabase
|
||||||
@@ -27,6 +29,7 @@ import org.koitharu.kotatsu.core.prefs.AppSettings
|
|||||||
import org.koitharu.kotatsu.core.util.WorkServiceStopHelper
|
import org.koitharu.kotatsu.core.util.WorkServiceStopHelper
|
||||||
import org.koitharu.kotatsu.core.util.ext.processLifecycleScope
|
import org.koitharu.kotatsu.core.util.ext.processLifecycleScope
|
||||||
import org.koitharu.kotatsu.settings.work.WorkScheduleManager
|
import org.koitharu.kotatsu.settings.work.WorkScheduleManager
|
||||||
|
import java.security.Security
|
||||||
import javax.inject.Inject
|
import javax.inject.Inject
|
||||||
import javax.inject.Provider
|
import javax.inject.Provider
|
||||||
|
|
||||||
@@ -52,7 +55,7 @@ open class BaseApp : Application(), Configuration.Provider {
|
|||||||
lateinit var appValidator: AppValidator
|
lateinit var appValidator: AppValidator
|
||||||
|
|
||||||
@Inject
|
@Inject
|
||||||
lateinit var workScheduleManager: Provider<WorkScheduleManager>
|
lateinit var workScheduleManager: WorkScheduleManager
|
||||||
|
|
||||||
@Inject
|
@Inject
|
||||||
lateinit var workManagerProvider: Provider<WorkManager>
|
lateinit var workManagerProvider: Provider<WorkManager>
|
||||||
@@ -66,6 +69,10 @@ open class BaseApp : Application(), Configuration.Provider {
|
|||||||
super.onCreate()
|
super.onCreate()
|
||||||
AppCompatDelegate.setDefaultNightMode(settings.theme)
|
AppCompatDelegate.setDefaultNightMode(settings.theme)
|
||||||
AppCompatDelegate.setApplicationLocales(settings.appLocales)
|
AppCompatDelegate.setApplicationLocales(settings.appLocales)
|
||||||
|
// TLS 1.3 support for Android < 10
|
||||||
|
if (Build.VERSION.SDK_INT < Build.VERSION_CODES.Q) {
|
||||||
|
Security.insertProviderAt(Conscrypt.newProvider(), 1)
|
||||||
|
}
|
||||||
setupActivityLifecycleCallbacks()
|
setupActivityLifecycleCallbacks()
|
||||||
processLifecycleScope.launch {
|
processLifecycleScope.launch {
|
||||||
val isOriginalApp = withContext(Dispatchers.Default) {
|
val isOriginalApp = withContext(Dispatchers.Default) {
|
||||||
@@ -76,7 +83,7 @@ open class BaseApp : Application(), Configuration.Provider {
|
|||||||
processLifecycleScope.launch(Dispatchers.Default) {
|
processLifecycleScope.launch(Dispatchers.Default) {
|
||||||
setupDatabaseObservers()
|
setupDatabaseObservers()
|
||||||
}
|
}
|
||||||
workScheduleManager.get().init()
|
workScheduleManager.init()
|
||||||
WorkServiceStopHelper(workManagerProvider).setup()
|
WorkServiceStopHelper(workManagerProvider).setup()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -0,0 +1,24 @@
|
|||||||
|
package org.koitharu.kotatsu.core
|
||||||
|
|
||||||
|
import android.content.Context
|
||||||
|
import com.google.auto.service.AutoService
|
||||||
|
import org.acra.builder.ReportBuilder
|
||||||
|
import org.acra.config.CoreConfiguration
|
||||||
|
import org.acra.config.ReportingAdministrator
|
||||||
|
|
||||||
|
@AutoService(ReportingAdministrator::class)
|
||||||
|
class ErrorReportingAdmin : ReportingAdministrator {
|
||||||
|
|
||||||
|
override fun shouldStartCollecting(
|
||||||
|
context: Context,
|
||||||
|
config: CoreConfiguration,
|
||||||
|
reportBuilder: ReportBuilder
|
||||||
|
): Boolean {
|
||||||
|
return reportBuilder.exception?.isDeadOs() != true
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun Throwable.isDeadOs(): Boolean {
|
||||||
|
val className = javaClass.simpleName
|
||||||
|
return className == "DeadSystemException" || className == "DeadSystemRuntimeException" || cause?.isDeadOs() == true
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -55,7 +55,7 @@ class BackupRepository @Inject constructor(
|
|||||||
var offset = 0
|
var offset = 0
|
||||||
val entry = BackupEntry(BackupEntry.Name.FAVOURITES, JSONArray())
|
val entry = BackupEntry(BackupEntry.Name.FAVOURITES, JSONArray())
|
||||||
while (true) {
|
while (true) {
|
||||||
val favourites = db.getFavouritesDao().findAll(offset, PAGE_SIZE)
|
val favourites = db.getFavouritesDao().findAllRaw(offset, PAGE_SIZE)
|
||||||
if (favourites.isEmpty()) {
|
if (favourites.isEmpty()) {
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -5,10 +5,10 @@ import kotlinx.coroutines.Dispatchers
|
|||||||
import kotlinx.coroutines.runInterruptible
|
import kotlinx.coroutines.runInterruptible
|
||||||
import okio.Closeable
|
import okio.Closeable
|
||||||
import org.koitharu.kotatsu.R
|
import org.koitharu.kotatsu.R
|
||||||
import org.koitharu.kotatsu.core.util.ext.format
|
|
||||||
import org.koitharu.kotatsu.core.zip.ZipOutput
|
import org.koitharu.kotatsu.core.zip.ZipOutput
|
||||||
import java.io.File
|
import java.io.File
|
||||||
import java.util.Date
|
import java.time.LocalDate
|
||||||
|
import java.time.format.DateTimeFormatter
|
||||||
import java.util.Locale
|
import java.util.Locale
|
||||||
import java.util.zip.Deflater
|
import java.util.zip.Deflater
|
||||||
|
|
||||||
@@ -39,7 +39,7 @@ suspend fun BackupZipOutput(context: Context): BackupZipOutput = runInterruptibl
|
|||||||
val filename = buildString {
|
val filename = buildString {
|
||||||
append(context.getString(R.string.app_name).replace(' ', '_').lowercase(Locale.ROOT))
|
append(context.getString(R.string.app_name).replace(' ', '_').lowercase(Locale.ROOT))
|
||||||
append('_')
|
append('_')
|
||||||
append(Date().format("ddMMyyyy"))
|
append(LocalDate.now().format(DateTimeFormatter.ofPattern("ddMMyyyy")))
|
||||||
append(".bk.zip")
|
append(".bk.zip")
|
||||||
}
|
}
|
||||||
BackupZipOutput(File(dir, filename))
|
BackupZipOutput(File(dir, filename))
|
||||||
|
|||||||
@@ -54,6 +54,7 @@ class JsonDeserializer(private val json: JSONObject) {
|
|||||||
page = json.getInt("page"),
|
page = json.getInt("page"),
|
||||||
scroll = json.getDouble("scroll").toFloat(),
|
scroll = json.getDouble("scroll").toFloat(),
|
||||||
percent = json.getFloatOrDefault("percent", -1f),
|
percent = json.getFloatOrDefault("percent", -1f),
|
||||||
|
chaptersCount = json.getIntOrDefault("chapters", -1),
|
||||||
deletedAt = 0L,
|
deletedAt = 0L,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|||||||
@@ -41,6 +41,7 @@ class JsonSerializer private constructor(private val json: JSONObject) {
|
|||||||
put("page", e.page)
|
put("page", e.page)
|
||||||
put("scroll", e.scroll)
|
put("scroll", e.scroll)
|
||||||
put("percent", e.percent)
|
put("percent", e.percent)
|
||||||
|
put("chapters", e.chaptersCount)
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|||||||
@@ -30,6 +30,7 @@ import org.koitharu.kotatsu.core.db.migrations.Migration14To15
|
|||||||
import org.koitharu.kotatsu.core.db.migrations.Migration15To16
|
import org.koitharu.kotatsu.core.db.migrations.Migration15To16
|
||||||
import org.koitharu.kotatsu.core.db.migrations.Migration16To17
|
import org.koitharu.kotatsu.core.db.migrations.Migration16To17
|
||||||
import org.koitharu.kotatsu.core.db.migrations.Migration17To18
|
import org.koitharu.kotatsu.core.db.migrations.Migration17To18
|
||||||
|
import org.koitharu.kotatsu.core.db.migrations.Migration18To19
|
||||||
import org.koitharu.kotatsu.core.db.migrations.Migration1To2
|
import org.koitharu.kotatsu.core.db.migrations.Migration1To2
|
||||||
import org.koitharu.kotatsu.core.db.migrations.Migration2To3
|
import org.koitharu.kotatsu.core.db.migrations.Migration2To3
|
||||||
import org.koitharu.kotatsu.core.db.migrations.Migration3To4
|
import org.koitharu.kotatsu.core.db.migrations.Migration3To4
|
||||||
@@ -48,20 +49,22 @@ import org.koitharu.kotatsu.history.data.HistoryDao
|
|||||||
import org.koitharu.kotatsu.history.data.HistoryEntity
|
import org.koitharu.kotatsu.history.data.HistoryEntity
|
||||||
import org.koitharu.kotatsu.scrobbling.common.data.ScrobblingDao
|
import org.koitharu.kotatsu.scrobbling.common.data.ScrobblingDao
|
||||||
import org.koitharu.kotatsu.scrobbling.common.data.ScrobblingEntity
|
import org.koitharu.kotatsu.scrobbling.common.data.ScrobblingEntity
|
||||||
|
import org.koitharu.kotatsu.stats.data.StatsDao
|
||||||
|
import org.koitharu.kotatsu.stats.data.StatsEntity
|
||||||
import org.koitharu.kotatsu.suggestions.data.SuggestionDao
|
import org.koitharu.kotatsu.suggestions.data.SuggestionDao
|
||||||
import org.koitharu.kotatsu.suggestions.data.SuggestionEntity
|
import org.koitharu.kotatsu.suggestions.data.SuggestionEntity
|
||||||
import org.koitharu.kotatsu.tracker.data.TrackEntity
|
import org.koitharu.kotatsu.tracker.data.TrackEntity
|
||||||
import org.koitharu.kotatsu.tracker.data.TrackLogEntity
|
import org.koitharu.kotatsu.tracker.data.TrackLogEntity
|
||||||
import org.koitharu.kotatsu.tracker.data.TracksDao
|
import org.koitharu.kotatsu.tracker.data.TracksDao
|
||||||
|
|
||||||
const val DATABASE_VERSION = 18
|
const val DATABASE_VERSION = 19
|
||||||
|
|
||||||
@Database(
|
@Database(
|
||||||
entities = [
|
entities = [
|
||||||
MangaEntity::class, TagEntity::class, HistoryEntity::class, MangaTagsEntity::class,
|
MangaEntity::class, TagEntity::class, HistoryEntity::class, MangaTagsEntity::class,
|
||||||
FavouriteCategoryEntity::class, FavouriteEntity::class, MangaPrefsEntity::class,
|
FavouriteCategoryEntity::class, FavouriteEntity::class, MangaPrefsEntity::class,
|
||||||
TrackEntity::class, TrackLogEntity::class, SuggestionEntity::class, BookmarkEntity::class,
|
TrackEntity::class, TrackLogEntity::class, SuggestionEntity::class, BookmarkEntity::class,
|
||||||
ScrobblingEntity::class, MangaSourceEntity::class,
|
ScrobblingEntity::class, MangaSourceEntity::class, StatsEntity::class,
|
||||||
],
|
],
|
||||||
version = DATABASE_VERSION,
|
version = DATABASE_VERSION,
|
||||||
)
|
)
|
||||||
@@ -90,6 +93,8 @@ abstract class MangaDatabase : RoomDatabase() {
|
|||||||
abstract fun getScrobblingDao(): ScrobblingDao
|
abstract fun getScrobblingDao(): ScrobblingDao
|
||||||
|
|
||||||
abstract fun getSourcesDao(): MangaSourcesDao
|
abstract fun getSourcesDao(): MangaSourcesDao
|
||||||
|
|
||||||
|
abstract fun getStatsDao(): StatsDao
|
||||||
}
|
}
|
||||||
|
|
||||||
fun getDatabaseMigrations(context: Context): Array<Migration> = arrayOf(
|
fun getDatabaseMigrations(context: Context): Array<Migration> = arrayOf(
|
||||||
@@ -110,6 +115,7 @@ fun getDatabaseMigrations(context: Context): Array<Migration> = arrayOf(
|
|||||||
Migration15To16(),
|
Migration15To16(),
|
||||||
Migration16To17(context),
|
Migration16To17(context),
|
||||||
Migration17To18(),
|
Migration17To18(),
|
||||||
|
Migration18To19(),
|
||||||
)
|
)
|
||||||
|
|
||||||
fun MangaDatabase(context: Context): MangaDatabase = Room
|
fun MangaDatabase(context: Context): MangaDatabase = Room
|
||||||
|
|||||||
@@ -31,6 +31,16 @@ abstract class TagsDao {
|
|||||||
)
|
)
|
||||||
abstract suspend fun findPopularTags(source: String, limit: Int): List<TagEntity>
|
abstract suspend fun findPopularTags(source: String, limit: Int): List<TagEntity>
|
||||||
|
|
||||||
|
@Query(
|
||||||
|
"""SELECT tags.* FROM tags
|
||||||
|
LEFT JOIN manga_tags ON tags.tag_id = manga_tags.tag_id
|
||||||
|
WHERE tags.source = :source
|
||||||
|
GROUP BY tags.title
|
||||||
|
ORDER BY COUNT(manga_id) ASC
|
||||||
|
LIMIT :limit""",
|
||||||
|
)
|
||||||
|
abstract suspend fun findRareTags(source: String, limit: Int): List<TagEntity>
|
||||||
|
|
||||||
@Query(
|
@Query(
|
||||||
"""SELECT tags.* FROM tags
|
"""SELECT tags.* FROM tags
|
||||||
LEFT JOIN manga_tags ON tags.tag_id = manga_tags.tag_id
|
LEFT JOIN manga_tags ON tags.tag_id = manga_tags.tag_id
|
||||||
|
|||||||
@@ -0,0 +1,12 @@
|
|||||||
|
package org.koitharu.kotatsu.core.db.migrations
|
||||||
|
|
||||||
|
import androidx.room.migration.Migration
|
||||||
|
import androidx.sqlite.db.SupportSQLiteDatabase
|
||||||
|
|
||||||
|
class Migration18To19 : Migration(18, 19) {
|
||||||
|
|
||||||
|
override fun migrate(db: SupportSQLiteDatabase) {
|
||||||
|
db.execSQL("ALTER TABLE history ADD COLUMN `chapters` INTEGER NOT NULL DEFAULT -1")
|
||||||
|
db.execSQL("CREATE TABLE IF NOT EXISTS `stats` (`manga_id` INTEGER NOT NULL, `started_at` INTEGER NOT NULL, `duration` INTEGER NOT NULL, `pages` INTEGER NOT NULL, PRIMARY KEY(`manga_id`, `started_at`), FOREIGN KEY(`manga_id`) REFERENCES `history`(`manga_id`) ON UPDATE NO ACTION ON DELETE CASCADE )")
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,13 +1,13 @@
|
|||||||
package org.koitharu.kotatsu.core.exceptions
|
package org.koitharu.kotatsu.core.exceptions
|
||||||
|
|
||||||
import okio.IOException
|
import okio.IOException
|
||||||
import java.util.Date
|
import java.time.Instant
|
||||||
|
import java.time.temporal.ChronoUnit
|
||||||
|
|
||||||
class TooManyRequestExceptions(
|
class TooManyRequestExceptions(
|
||||||
val url: String,
|
val url: String,
|
||||||
val retryAt: Date?,
|
val retryAt: Instant?,
|
||||||
) : IOException() {
|
) : IOException() {
|
||||||
|
|
||||||
val retryAfter: Long
|
val retryAfter: Long
|
||||||
get() = if (retryAt == null) 0 else (retryAt.time - System.currentTimeMillis()).coerceAtLeast(0)
|
get() = retryAt?.until(Instant.now(), ChronoUnit.MILLIS) ?: 0
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -20,8 +20,9 @@ import org.koitharu.kotatsu.parsers.util.runCatchingCancellable
|
|||||||
import org.koitharu.kotatsu.core.util.ext.printStackTraceDebug
|
import org.koitharu.kotatsu.core.util.ext.printStackTraceDebug
|
||||||
import java.io.File
|
import java.io.File
|
||||||
import java.io.FileOutputStream
|
import java.io.FileOutputStream
|
||||||
import java.text.SimpleDateFormat
|
import java.time.LocalDateTime
|
||||||
import java.util.Date
|
import java.time.format.DateTimeFormatter
|
||||||
|
import java.time.format.FormatStyle
|
||||||
import java.util.Locale
|
import java.util.Locale
|
||||||
import java.util.concurrent.ConcurrentLinkedQueue
|
import java.util.concurrent.ConcurrentLinkedQueue
|
||||||
|
|
||||||
@@ -41,11 +42,7 @@ class FileLogger(
|
|||||||
}
|
}
|
||||||
val isEnabled: Boolean
|
val isEnabled: Boolean
|
||||||
get() = settings.isLoggingEnabled
|
get() = settings.isLoggingEnabled
|
||||||
private val dateFormat = SimpleDateFormat.getDateTimeInstance(
|
private val dateTimeFormatter = DateTimeFormatter.ofLocalizedDateTime(FormatStyle.SHORT).withLocale(Locale.ROOT)
|
||||||
SimpleDateFormat.SHORT,
|
|
||||||
SimpleDateFormat.SHORT,
|
|
||||||
Locale.ROOT,
|
|
||||||
)
|
|
||||||
private val buffer = ConcurrentLinkedQueue<String>()
|
private val buffer = ConcurrentLinkedQueue<String>()
|
||||||
private val mutex = Mutex()
|
private val mutex = Mutex()
|
||||||
private var flushJob: Job? = null
|
private var flushJob: Job? = null
|
||||||
@@ -55,7 +52,7 @@ class FileLogger(
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
val text = buildString {
|
val text = buildString {
|
||||||
append(dateFormat.format(Date()))
|
append(dateTimeFormatter.format(LocalDateTime.now()))
|
||||||
append(": ")
|
append(": ")
|
||||||
if (e != null) {
|
if (e != null) {
|
||||||
append("E!")
|
append("E!")
|
||||||
|
|||||||
@@ -5,7 +5,7 @@ import kotlinx.parcelize.Parcelize
|
|||||||
import org.koitharu.kotatsu.list.domain.ListSortOrder
|
import org.koitharu.kotatsu.list.domain.ListSortOrder
|
||||||
import org.koitharu.kotatsu.list.ui.ListModelDiffCallback
|
import org.koitharu.kotatsu.list.ui.ListModelDiffCallback
|
||||||
import org.koitharu.kotatsu.list.ui.model.ListModel
|
import org.koitharu.kotatsu.list.ui.model.ListModel
|
||||||
import java.util.Date
|
import java.time.Instant
|
||||||
|
|
||||||
@Parcelize
|
@Parcelize
|
||||||
data class FavouriteCategory(
|
data class FavouriteCategory(
|
||||||
@@ -13,7 +13,7 @@ data class FavouriteCategory(
|
|||||||
val title: String,
|
val title: String,
|
||||||
val sortKey: Int,
|
val sortKey: Int,
|
||||||
val order: ListSortOrder,
|
val order: ListSortOrder,
|
||||||
val createdAt: Date,
|
val createdAt: Instant,
|
||||||
val isTrackingEnabled: Boolean,
|
val isTrackingEnabled: Boolean,
|
||||||
val isVisibleInLibrary: Boolean,
|
val isVisibleInLibrary: Boolean,
|
||||||
) : Parcelable, ListModel {
|
) : Parcelable, ListModel {
|
||||||
|
|||||||
@@ -3,15 +3,20 @@ package org.koitharu.kotatsu.core.model
|
|||||||
import android.net.Uri
|
import android.net.Uri
|
||||||
import androidx.annotation.DrawableRes
|
import androidx.annotation.DrawableRes
|
||||||
import androidx.annotation.StringRes
|
import androidx.annotation.StringRes
|
||||||
|
import androidx.collection.MutableObjectIntMap
|
||||||
import androidx.core.os.LocaleListCompat
|
import androidx.core.os.LocaleListCompat
|
||||||
import org.koitharu.kotatsu.R
|
import org.koitharu.kotatsu.R
|
||||||
import org.koitharu.kotatsu.core.util.ext.iterator
|
import org.koitharu.kotatsu.core.util.ext.iterator
|
||||||
import org.koitharu.kotatsu.details.ui.model.ChapterListItem
|
import org.koitharu.kotatsu.details.ui.model.ChapterListItem
|
||||||
|
import org.koitharu.kotatsu.parsers.model.ContentRating
|
||||||
import org.koitharu.kotatsu.parsers.model.Manga
|
import org.koitharu.kotatsu.parsers.model.Manga
|
||||||
import org.koitharu.kotatsu.parsers.model.MangaChapter
|
import org.koitharu.kotatsu.parsers.model.MangaChapter
|
||||||
import org.koitharu.kotatsu.parsers.model.MangaSource
|
import org.koitharu.kotatsu.parsers.model.MangaSource
|
||||||
import org.koitharu.kotatsu.parsers.model.MangaState
|
import org.koitharu.kotatsu.parsers.model.MangaState
|
||||||
import org.koitharu.kotatsu.parsers.util.mapToSet
|
import org.koitharu.kotatsu.parsers.util.mapToSet
|
||||||
|
import java.text.DecimalFormat
|
||||||
|
import java.text.DecimalFormatSymbols
|
||||||
|
import com.google.android.material.R as materialR
|
||||||
|
|
||||||
@JvmName("mangaIds")
|
@JvmName("mangaIds")
|
||||||
fun Collection<Manga>.ids() = mapToSet { it.id }
|
fun Collection<Manga>.ids() = mapToSet { it.id }
|
||||||
@@ -27,12 +32,14 @@ fun Collection<ChapterListItem>.countChaptersByBranch(): Int {
|
|||||||
if (size <= 1) {
|
if (size <= 1) {
|
||||||
return size
|
return size
|
||||||
}
|
}
|
||||||
val acc = HashMap<String?, Int>()
|
val acc = MutableObjectIntMap<String?>()
|
||||||
for (item in this) {
|
for (item in this) {
|
||||||
val branch = item.chapter.branch
|
val branch = item.chapter.branch
|
||||||
acc[branch] = (acc[branch] ?: 0) + 1
|
acc[branch] = acc.getOrDefault(branch, 0) + 1
|
||||||
}
|
}
|
||||||
return acc.values.max()
|
var max = 0
|
||||||
|
acc.forEachValue { x -> if (x > max) max = x }
|
||||||
|
return max
|
||||||
}
|
}
|
||||||
|
|
||||||
@get:StringRes
|
@get:StringRes
|
||||||
@@ -42,15 +49,25 @@ val MangaState.titleResId: Int
|
|||||||
MangaState.FINISHED -> R.string.state_finished
|
MangaState.FINISHED -> R.string.state_finished
|
||||||
MangaState.ABANDONED -> R.string.state_abandoned
|
MangaState.ABANDONED -> R.string.state_abandoned
|
||||||
MangaState.PAUSED -> R.string.state_paused
|
MangaState.PAUSED -> R.string.state_paused
|
||||||
|
MangaState.UPCOMING -> R.string.state_upcoming
|
||||||
}
|
}
|
||||||
|
|
||||||
@get:DrawableRes
|
@get:DrawableRes
|
||||||
val MangaState.iconResId: Int
|
val MangaState.iconResId: Int
|
||||||
get() = when (this) {
|
get() = when (this) {
|
||||||
MangaState.ONGOING -> R.drawable.ic_state_ongoing
|
MangaState.ONGOING -> R.drawable.ic_play
|
||||||
MangaState.FINISHED -> R.drawable.ic_state_finished
|
MangaState.FINISHED -> R.drawable.ic_state_finished
|
||||||
MangaState.ABANDONED -> R.drawable.ic_state_abandoned
|
MangaState.ABANDONED -> R.drawable.ic_state_abandoned
|
||||||
MangaState.PAUSED -> R.drawable.ic_action_pause
|
MangaState.PAUSED -> R.drawable.ic_action_pause
|
||||||
|
MangaState.UPCOMING -> materialR.drawable.ic_clock_black_24dp
|
||||||
|
}
|
||||||
|
|
||||||
|
@get:StringRes
|
||||||
|
val ContentRating.titleResId: Int
|
||||||
|
get() = when (this) {
|
||||||
|
ContentRating.SAFE -> R.string.rating_safe
|
||||||
|
ContentRating.SUGGESTIVE -> R.string.rating_suggestive
|
||||||
|
ContentRating.ADULT -> R.string.rating_adult
|
||||||
}
|
}
|
||||||
|
|
||||||
fun Manga.findChapter(id: Long): MangaChapter? {
|
fun Manga.findChapter(id: Long): MangaChapter? {
|
||||||
@@ -101,3 +118,16 @@ val Manga.appUrl: Uri
|
|||||||
.appendQueryParameter("name", title)
|
.appendQueryParameter("name", title)
|
||||||
.appendQueryParameter("url", url)
|
.appendQueryParameter("url", url)
|
||||||
.build()
|
.build()
|
||||||
|
|
||||||
|
private val chaptersNumberFormat = DecimalFormat("#.#").also { f ->
|
||||||
|
f.decimalFormatSymbols = DecimalFormatSymbols.getInstance().also {
|
||||||
|
it.decimalSeparator = '.'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fun MangaChapter.formatNumber(): String? {
|
||||||
|
if (number <= 0f) {
|
||||||
|
return null
|
||||||
|
}
|
||||||
|
return chaptersNumberFormat.format(number.toDouble())
|
||||||
|
}
|
||||||
|
|||||||
@@ -2,14 +2,14 @@ package org.koitharu.kotatsu.core.model
|
|||||||
|
|
||||||
import android.os.Parcelable
|
import android.os.Parcelable
|
||||||
import kotlinx.parcelize.Parcelize
|
import kotlinx.parcelize.Parcelize
|
||||||
import java.util.*
|
import java.time.Instant
|
||||||
|
|
||||||
@Parcelize
|
@Parcelize
|
||||||
data class MangaHistory(
|
data class MangaHistory(
|
||||||
val createdAt: Date,
|
val createdAt: Instant,
|
||||||
val updatedAt: Date,
|
val updatedAt: Instant,
|
||||||
val chapterId: Long,
|
val chapterId: Long,
|
||||||
val page: Int,
|
val page: Int,
|
||||||
val scroll: Int,
|
val scroll: Int,
|
||||||
val percent: Float,
|
val percent: Float,
|
||||||
) : Parcelable
|
) : Parcelable
|
||||||
|
|||||||
@@ -19,7 +19,8 @@ data class ParcelableChapter(
|
|||||||
MangaChapter(
|
MangaChapter(
|
||||||
id = parcel.readLong(),
|
id = parcel.readLong(),
|
||||||
name = parcel.readString().orEmpty(),
|
name = parcel.readString().orEmpty(),
|
||||||
number = parcel.readInt(),
|
number = parcel.readFloat(),
|
||||||
|
volume = parcel.readInt(),
|
||||||
url = parcel.readString().orEmpty(),
|
url = parcel.readString().orEmpty(),
|
||||||
scanlator = parcel.readString(),
|
scanlator = parcel.readString(),
|
||||||
uploadDate = parcel.readLong(),
|
uploadDate = parcel.readLong(),
|
||||||
@@ -31,7 +32,8 @@ data class ParcelableChapter(
|
|||||||
override fun ParcelableChapter.write(parcel: Parcel, flags: Int) = with(chapter) {
|
override fun ParcelableChapter.write(parcel: Parcel, flags: Int) = with(chapter) {
|
||||||
parcel.writeLong(id)
|
parcel.writeLong(id)
|
||||||
parcel.writeString(name)
|
parcel.writeString(name)
|
||||||
parcel.writeInt(number)
|
parcel.writeFloat(number)
|
||||||
|
parcel.writeInt(volume)
|
||||||
parcel.writeString(url)
|
parcel.writeString(url)
|
||||||
parcel.writeString(scanlator)
|
parcel.writeString(scanlator)
|
||||||
parcel.writeLong(uploadDate)
|
parcel.writeLong(uploadDate)
|
||||||
|
|||||||
@@ -7,6 +7,7 @@ import coil.request.ErrorResult
|
|||||||
import coil.request.ImageResult
|
import coil.request.ImageResult
|
||||||
import coil.request.SuccessResult
|
import coil.request.SuccessResult
|
||||||
import coil.size.Dimension
|
import coil.size.Dimension
|
||||||
|
import coil.size.isOriginal
|
||||||
import okhttp3.HttpUrl
|
import okhttp3.HttpUrl
|
||||||
import okhttp3.HttpUrl.Companion.toHttpUrlOrNull
|
import okhttp3.HttpUrl.Companion.toHttpUrlOrNull
|
||||||
import okhttp3.OkHttpClient
|
import okhttp3.OkHttpClient
|
||||||
@@ -46,11 +47,13 @@ class ImageProxyInterceptor @Inject constructor(
|
|||||||
.scheme("https")
|
.scheme("https")
|
||||||
.host("wsrv.nl")
|
.host("wsrv.nl")
|
||||||
.addQueryParameter("url", url.toString())
|
.addQueryParameter("url", url.toString())
|
||||||
.addQueryParameter("fit", "outside")
|
|
||||||
.addQueryParameter("we", null)
|
.addQueryParameter("we", null)
|
||||||
val size = request.sizeResolver.size()
|
val size = request.sizeResolver.size()
|
||||||
(size.height as? Dimension.Pixels)?.let { newUrl.addQueryParameter("h", it.toString()) }
|
if (!size.isOriginal) {
|
||||||
(size.width as? Dimension.Pixels)?.let { newUrl.addQueryParameter("w", it.toString()) }
|
newUrl.addQueryParameter("crop", "cover")
|
||||||
|
(size.height as? Dimension.Pixels)?.let { newUrl.addQueryParameter("h", it.toString()) }
|
||||||
|
(size.width as? Dimension.Pixels)?.let { newUrl.addQueryParameter("w", it.toString()) }
|
||||||
|
}
|
||||||
|
|
||||||
val newRequest = request.newBuilder()
|
val newRequest = request.newBuilder()
|
||||||
.data(newUrl.build())
|
.data(newUrl.build())
|
||||||
|
|||||||
@@ -4,15 +4,11 @@ import okhttp3.Interceptor
|
|||||||
import okhttp3.Response
|
import okhttp3.Response
|
||||||
import okhttp3.internal.closeQuietly
|
import okhttp3.internal.closeQuietly
|
||||||
import org.koitharu.kotatsu.core.exceptions.TooManyRequestExceptions
|
import org.koitharu.kotatsu.core.exceptions.TooManyRequestExceptions
|
||||||
import java.text.SimpleDateFormat
|
import java.time.Instant
|
||||||
import java.util.Date
|
import java.time.ZonedDateTime
|
||||||
import java.util.Locale
|
import java.time.format.DateTimeFormatter
|
||||||
import java.util.concurrent.TimeUnit
|
|
||||||
|
|
||||||
class RateLimitInterceptor : Interceptor {
|
class RateLimitInterceptor : Interceptor {
|
||||||
|
|
||||||
private val dateFormat = SimpleDateFormat("EEE, dd MMM yyyy HH:mm:ss ZZZ", Locale.ENGLISH)
|
|
||||||
|
|
||||||
override fun intercept(chain: Interceptor.Chain): Response {
|
override fun intercept(chain: Interceptor.Chain): Response {
|
||||||
val response = chain.proceed(chain.request())
|
val response = chain.proceed(chain.request())
|
||||||
if (response.code == 429) {
|
if (response.code == 429) {
|
||||||
@@ -27,10 +23,8 @@ class RateLimitInterceptor : Interceptor {
|
|||||||
return response
|
return response
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun String.parseRetryDate(): Date? {
|
private fun String.parseRetryDate(): Instant? {
|
||||||
toIntOrNull()?.let {
|
return toLongOrNull()?.let { Instant.now().plusSeconds(it) }
|
||||||
return Date(System.currentTimeMillis() + TimeUnit.SECONDS.toMillis(it.toLong()))
|
?: ZonedDateTime.parse(this, DateTimeFormatter.RFC_1123_DATE_TIME).toInstant()
|
||||||
}
|
|
||||||
return dateFormat.parse(this)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,16 +1,9 @@
|
|||||||
package org.koitharu.kotatsu.core.os
|
package org.koitharu.kotatsu.core.os
|
||||||
|
|
||||||
import android.annotation.SuppressLint
|
|
||||||
import android.content.Context
|
import android.content.Context
|
||||||
import android.content.pm.PackageManager
|
import android.content.pm.PackageManager
|
||||||
|
import androidx.core.content.pm.PackageInfoCompat
|
||||||
import dagger.hilt.android.qualifiers.ApplicationContext
|
import dagger.hilt.android.qualifiers.ApplicationContext
|
||||||
import org.koitharu.kotatsu.core.util.ext.printStackTraceDebug
|
|
||||||
import org.koitharu.kotatsu.parsers.util.byte2HexFormatted
|
|
||||||
import java.io.ByteArrayInputStream
|
|
||||||
import java.io.InputStream
|
|
||||||
import java.security.MessageDigest
|
|
||||||
import java.security.cert.CertificateFactory
|
|
||||||
import java.security.cert.X509Certificate
|
|
||||||
import javax.inject.Inject
|
import javax.inject.Inject
|
||||||
import javax.inject.Singleton
|
import javax.inject.Singleton
|
||||||
|
|
||||||
@@ -18,29 +11,13 @@ import javax.inject.Singleton
|
|||||||
class AppValidator @Inject constructor(
|
class AppValidator @Inject constructor(
|
||||||
@ApplicationContext private val context: Context,
|
@ApplicationContext private val context: Context,
|
||||||
) {
|
) {
|
||||||
|
@Suppress("NewApi")
|
||||||
val isOriginalApp by lazy {
|
val isOriginalApp by lazy {
|
||||||
getCertificateSHA1Fingerprint() == CERT_SHA1
|
val certificates = mapOf(CERT_SHA256.hexToByteArray() to PackageManager.CERT_INPUT_SHA256)
|
||||||
|
PackageInfoCompat.hasSignatures(context.packageManager, context.packageName, certificates, false)
|
||||||
}
|
}
|
||||||
|
|
||||||
@Suppress("DEPRECATION")
|
|
||||||
@SuppressLint("PackageManagerGetSignatures")
|
|
||||||
private fun getCertificateSHA1Fingerprint(): String? = runCatching {
|
|
||||||
val packageInfo = context.packageManager.getPackageInfo(context.packageName, PackageManager.GET_SIGNATURES)
|
|
||||||
val signatures = requireNotNull(packageInfo?.signatures)
|
|
||||||
val cert: ByteArray = signatures.first().toByteArray()
|
|
||||||
val input: InputStream = ByteArrayInputStream(cert)
|
|
||||||
val cf = CertificateFactory.getInstance("X509")
|
|
||||||
val c = cf.generateCertificate(input) as X509Certificate
|
|
||||||
val md: MessageDigest = MessageDigest.getInstance("SHA1")
|
|
||||||
val publicKey: ByteArray = md.digest(c.encoded)
|
|
||||||
return publicKey.byte2HexFormatted()
|
|
||||||
}.onFailure { error ->
|
|
||||||
error.printStackTraceDebug()
|
|
||||||
}.getOrNull()
|
|
||||||
|
|
||||||
private companion object {
|
private companion object {
|
||||||
|
private const val CERT_SHA256 = "67e15100bb809301783edcb6348fa3bbf83034d91e62868a91053dbd70db3f18"
|
||||||
private const val CERT_SHA1 = "2C:19:C7:E8:07:61:2B:8E:94:51:1B:FD:72:67:07:64:5D:C2:58:AE"
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -5,6 +5,7 @@ import org.koitharu.kotatsu.core.cache.ContentCache
|
|||||||
import org.koitharu.kotatsu.core.network.MirrorSwitchInterceptor
|
import org.koitharu.kotatsu.core.network.MirrorSwitchInterceptor
|
||||||
import org.koitharu.kotatsu.local.data.LocalMangaRepository
|
import org.koitharu.kotatsu.local.data.LocalMangaRepository
|
||||||
import org.koitharu.kotatsu.parsers.MangaLoaderContext
|
import org.koitharu.kotatsu.parsers.MangaLoaderContext
|
||||||
|
import org.koitharu.kotatsu.parsers.model.ContentRating
|
||||||
import org.koitharu.kotatsu.parsers.model.Manga
|
import org.koitharu.kotatsu.parsers.model.Manga
|
||||||
import org.koitharu.kotatsu.parsers.model.MangaChapter
|
import org.koitharu.kotatsu.parsers.model.MangaChapter
|
||||||
import org.koitharu.kotatsu.parsers.model.MangaListFilter
|
import org.koitharu.kotatsu.parsers.model.MangaListFilter
|
||||||
@@ -28,10 +29,16 @@ interface MangaRepository {
|
|||||||
|
|
||||||
val states: Set<MangaState>
|
val states: Set<MangaState>
|
||||||
|
|
||||||
|
val contentRatings: Set<ContentRating>
|
||||||
|
|
||||||
var defaultSortOrder: SortOrder
|
var defaultSortOrder: SortOrder
|
||||||
|
|
||||||
val isMultipleTagsSupported: Boolean
|
val isMultipleTagsSupported: Boolean
|
||||||
|
|
||||||
|
val isTagsExclusionSupported: Boolean
|
||||||
|
|
||||||
|
val isSearchSupported: Boolean
|
||||||
|
|
||||||
suspend fun getList(offset: Int, filter: MangaListFilter?): List<Manga>
|
suspend fun getList(offset: Int, filter: MangaListFilter?): List<Manga>
|
||||||
|
|
||||||
suspend fun getDetails(manga: Manga): Manga
|
suspend fun getDetails(manga: Manga): Manga
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
package org.koitharu.kotatsu.core.parser
|
package org.koitharu.kotatsu.core.parser
|
||||||
|
|
||||||
import android.util.Log
|
import android.util.Log
|
||||||
|
import androidx.collection.MutableLongSet
|
||||||
import coil.request.CachePolicy
|
import coil.request.CachePolicy
|
||||||
import kotlinx.coroutines.CoroutineDispatcher
|
import kotlinx.coroutines.CoroutineDispatcher
|
||||||
import kotlinx.coroutines.CoroutineScope
|
import kotlinx.coroutines.CoroutineScope
|
||||||
@@ -21,6 +22,7 @@ import org.koitharu.kotatsu.parsers.MangaParser
|
|||||||
import org.koitharu.kotatsu.parsers.MangaParserAuthProvider
|
import org.koitharu.kotatsu.parsers.MangaParserAuthProvider
|
||||||
import org.koitharu.kotatsu.parsers.config.ConfigKey
|
import org.koitharu.kotatsu.parsers.config.ConfigKey
|
||||||
import org.koitharu.kotatsu.parsers.exception.ParseException
|
import org.koitharu.kotatsu.parsers.exception.ParseException
|
||||||
|
import org.koitharu.kotatsu.parsers.model.ContentRating
|
||||||
import org.koitharu.kotatsu.parsers.model.Favicons
|
import org.koitharu.kotatsu.parsers.model.Favicons
|
||||||
import org.koitharu.kotatsu.parsers.model.Manga
|
import org.koitharu.kotatsu.parsers.model.Manga
|
||||||
import org.koitharu.kotatsu.parsers.model.MangaChapter
|
import org.koitharu.kotatsu.parsers.model.MangaChapter
|
||||||
@@ -49,6 +51,9 @@ class RemoteMangaRepository(
|
|||||||
override val states: Set<MangaState>
|
override val states: Set<MangaState>
|
||||||
get() = parser.availableStates
|
get() = parser.availableStates
|
||||||
|
|
||||||
|
override val contentRatings: Set<ContentRating>
|
||||||
|
get() = parser.availableContentRating
|
||||||
|
|
||||||
override var defaultSortOrder: SortOrder
|
override var defaultSortOrder: SortOrder
|
||||||
get() = getConfig().defaultSortOrder ?: sortOrders.first()
|
get() = getConfig().defaultSortOrder ?: sortOrders.first()
|
||||||
set(value) {
|
set(value) {
|
||||||
@@ -58,6 +63,12 @@ class RemoteMangaRepository(
|
|||||||
override val isMultipleTagsSupported: Boolean
|
override val isMultipleTagsSupported: Boolean
|
||||||
get() = parser.isMultipleTagsSupported
|
get() = parser.isMultipleTagsSupported
|
||||||
|
|
||||||
|
override val isSearchSupported: Boolean
|
||||||
|
get() = parser.isSearchSupported
|
||||||
|
|
||||||
|
override val isTagsExclusionSupported: Boolean
|
||||||
|
get() = parser.isTagsExclusionSupported
|
||||||
|
|
||||||
var domain: String
|
var domain: String
|
||||||
get() = parser.domain
|
get() = parser.domain
|
||||||
set(value) {
|
set(value) {
|
||||||
@@ -179,7 +190,7 @@ class RemoteMangaRepository(
|
|||||||
return emptyList()
|
return emptyList()
|
||||||
}
|
}
|
||||||
val result = ArrayList<MangaPage>(size)
|
val result = ArrayList<MangaPage>(size)
|
||||||
val set = HashSet<Long>(size)
|
val set = MutableLongSet(size)
|
||||||
for (page in this) {
|
for (page in this) {
|
||||||
if (set.add(page.id)) {
|
if (set.add(page.id)) {
|
||||||
result.add(page)
|
result.add(page)
|
||||||
@@ -216,6 +227,5 @@ class RemoteMangaRepository(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun Result<*>.isValidResult() = exceptionOrNull() !is ParseException
|
private fun Result<*>.isValidResult() = isSuccess && (getOrNull() as? Collection<*>)?.isEmpty() != true
|
||||||
&& (getOrNull() as? Collection<*>)?.isEmpty() != true
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -11,6 +11,7 @@ import androidx.appcompat.app.AppCompatDelegate
|
|||||||
import androidx.collection.ArraySet
|
import androidx.collection.ArraySet
|
||||||
import androidx.core.content.edit
|
import androidx.core.content.edit
|
||||||
import androidx.core.os.LocaleListCompat
|
import androidx.core.os.LocaleListCompat
|
||||||
|
import androidx.documentfile.provider.DocumentFile
|
||||||
import androidx.preference.PreferenceManager
|
import androidx.preference.PreferenceManager
|
||||||
import dagger.hilt.android.qualifiers.ApplicationContext
|
import dagger.hilt.android.qualifiers.ApplicationContext
|
||||||
import org.json.JSONArray
|
import org.json.JSONArray
|
||||||
@@ -26,6 +27,7 @@ import org.koitharu.kotatsu.explore.data.SourcesSortOrder
|
|||||||
import org.koitharu.kotatsu.list.domain.ListSortOrder
|
import org.koitharu.kotatsu.list.domain.ListSortOrder
|
||||||
import org.koitharu.kotatsu.parsers.model.SortOrder
|
import org.koitharu.kotatsu.parsers.model.SortOrder
|
||||||
import org.koitharu.kotatsu.parsers.util.find
|
import org.koitharu.kotatsu.parsers.util.find
|
||||||
|
import org.koitharu.kotatsu.parsers.util.isNumeric
|
||||||
import org.koitharu.kotatsu.parsers.util.mapNotNullToSet
|
import org.koitharu.kotatsu.parsers.util.mapNotNullToSet
|
||||||
import org.koitharu.kotatsu.parsers.util.mapToSet
|
import org.koitharu.kotatsu.parsers.util.mapToSet
|
||||||
import org.koitharu.kotatsu.reader.domain.ReaderColorFilter
|
import org.koitharu.kotatsu.reader.domain.ReaderColorFilter
|
||||||
@@ -70,6 +72,9 @@ class AppSettings @Inject constructor(@ApplicationContext context: Context) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
val isNavLabelsVisible: Boolean
|
||||||
|
get() = prefs.getBoolean(KEY_NAV_LABELS, true)
|
||||||
|
|
||||||
var gridSize: Int
|
var gridSize: Int
|
||||||
get() = prefs.getInt(KEY_GRID_SIZE, 100)
|
get() = prefs.getInt(KEY_GRID_SIZE, 100)
|
||||||
set(value) = prefs.edit { putInt(KEY_GRID_SIZE, value) }
|
set(value) = prefs.edit { putInt(KEY_GRID_SIZE, value) }
|
||||||
@@ -101,14 +106,21 @@ class AppSettings @Inject constructor(@ApplicationContext context: Context) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
val readerPageSwitch: Set<String>
|
var isReaderDoubleOnLandscape: Boolean
|
||||||
get() = prefs.getStringSet(KEY_READER_SWITCHERS, null) ?: setOf(PAGE_SWITCH_TAPS)
|
get() = prefs.getBoolean(KEY_READER_DOUBLE_PAGES, false)
|
||||||
|
set(value) = prefs.edit { putBoolean(KEY_READER_DOUBLE_PAGES, value) }
|
||||||
|
|
||||||
|
val isReaderVolumeButtonsEnabled: Boolean
|
||||||
|
get() = prefs.getBoolean(KEY_READER_VOLUME_BUTTONS, false)
|
||||||
|
|
||||||
val isReaderZoomButtonsEnabled: Boolean
|
val isReaderZoomButtonsEnabled: Boolean
|
||||||
get() = prefs.getBoolean(KEY_READER_ZOOM_BUTTONS, false)
|
get() = prefs.getBoolean(KEY_READER_ZOOM_BUTTONS, false)
|
||||||
|
|
||||||
val isReaderTapsAdaptive: Boolean
|
val isReaderControlAlwaysLTR: Boolean
|
||||||
get() = !prefs.getBoolean(KEY_READER_TAPS_LTR, false)
|
get() = prefs.getBoolean(KEY_READER_CONTROL_LTR, false)
|
||||||
|
|
||||||
|
val isReaderFullscreenEnabled: Boolean
|
||||||
|
get() = prefs.getBoolean(KEY_READER_FULLSCREEN, true)
|
||||||
|
|
||||||
val isReaderOptimizationEnabled: Boolean
|
val isReaderOptimizationEnabled: Boolean
|
||||||
get() = prefs.getBoolean(KEY_READER_OPTIMIZE, false)
|
get() = prefs.getBoolean(KEY_READER_OPTIMIZE, false)
|
||||||
@@ -180,11 +192,13 @@ class AppSettings @Inject constructor(@ApplicationContext context: Context) {
|
|||||||
var appPassword: String?
|
var appPassword: String?
|
||||||
get() = prefs.getString(KEY_APP_PASSWORD, null)
|
get() = prefs.getString(KEY_APP_PASSWORD, null)
|
||||||
set(value) = prefs.edit {
|
set(value) = prefs.edit {
|
||||||
if (value != null) putString(KEY_APP_PASSWORD, value) else remove(
|
if (value != null) putString(KEY_APP_PASSWORD, value) else remove(KEY_APP_PASSWORD)
|
||||||
KEY_APP_PASSWORD,
|
|
||||||
)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var isAppPasswordNumeric: Boolean
|
||||||
|
get() = prefs.getBoolean(KEY_APP_PASSWORD_NUMERIC, false)
|
||||||
|
set(value) = prefs.edit { putBoolean(KEY_APP_PASSWORD_NUMERIC, value) }
|
||||||
|
|
||||||
val isLoggingEnabled: Boolean
|
val isLoggingEnabled: Boolean
|
||||||
get() = prefs.getBoolean(KEY_LOGGING_ENABLED, false)
|
get() = prefs.getBoolean(KEY_LOGGING_ENABLED, false)
|
||||||
|
|
||||||
@@ -204,6 +218,9 @@ class AppSettings @Inject constructor(@ApplicationContext context: Context) {
|
|||||||
val isUnstableUpdatesAllowed: Boolean
|
val isUnstableUpdatesAllowed: Boolean
|
||||||
get() = prefs.getBoolean(KEY_UPDATES_UNSTABLE, false)
|
get() = prefs.getBoolean(KEY_UPDATES_UNSTABLE, false)
|
||||||
|
|
||||||
|
val defaultDetailsTab: Int
|
||||||
|
get() = prefs.getString(KEY_DETAILS_TAB, null)?.toIntOrNull()?.coerceIn(0, 1) ?: 0
|
||||||
|
|
||||||
val isContentPrefetchEnabled: Boolean
|
val isContentPrefetchEnabled: Boolean
|
||||||
get() {
|
get() {
|
||||||
if (isBackgroundNetworkRestricted()) {
|
if (isBackgroundNetworkRestricted()) {
|
||||||
@@ -263,6 +280,9 @@ class AppSettings @Inject constructor(@ApplicationContext context: Context) {
|
|||||||
val isDownloadsWiFiOnly: Boolean
|
val isDownloadsWiFiOnly: Boolean
|
||||||
get() = prefs.getBoolean(KEY_DOWNLOADS_WIFI, false)
|
get() = prefs.getBoolean(KEY_DOWNLOADS_WIFI, false)
|
||||||
|
|
||||||
|
val preferredDownloadFormat: DownloadFormat
|
||||||
|
get() = prefs.getEnumValue(KEY_DOWNLOADS_FORMAT, DownloadFormat.AUTOMATIC)
|
||||||
|
|
||||||
var isSuggestionsEnabled: Boolean
|
var isSuggestionsEnabled: Boolean
|
||||||
get() = prefs.getBoolean(KEY_SUGGESTIONS, false)
|
get() = prefs.getBoolean(KEY_SUGGESTIONS, false)
|
||||||
set(value) = prefs.edit { putBoolean(KEY_SUGGESTIONS, value) }
|
set(value) = prefs.edit { putBoolean(KEY_SUGGESTIONS, value) }
|
||||||
@@ -295,13 +315,13 @@ class AppSettings @Inject constructor(@ApplicationContext context: Context) {
|
|||||||
get() = prefs.getBoolean(KEY_READER_SCREEN_ON, true)
|
get() = prefs.getBoolean(KEY_READER_SCREEN_ON, true)
|
||||||
|
|
||||||
var readerColorFilter: ReaderColorFilter?
|
var readerColorFilter: ReaderColorFilter?
|
||||||
get() {
|
get() = runCatching {
|
||||||
val brightness = prefs.getFloat(KEY_CF_BRIGHTNESS, ReaderColorFilter.EMPTY.brightness)
|
val brightness = prefs.getFloat(KEY_CF_BRIGHTNESS, ReaderColorFilter.EMPTY.brightness)
|
||||||
val contrast = prefs.getFloat(KEY_CF_CONTRAST, ReaderColorFilter.EMPTY.contrast)
|
val contrast = prefs.getFloat(KEY_CF_CONTRAST, ReaderColorFilter.EMPTY.contrast)
|
||||||
val inverted = prefs.getBoolean(KEY_CF_INVERTED, ReaderColorFilter.EMPTY.isInverted)
|
val inverted = prefs.getBoolean(KEY_CF_INVERTED, ReaderColorFilter.EMPTY.isInverted)
|
||||||
val grayscale = prefs.getBoolean(KEY_CF_GRAYSCALE, ReaderColorFilter.EMPTY.isGrayscale)
|
val grayscale = prefs.getBoolean(KEY_CF_GRAYSCALE, ReaderColorFilter.EMPTY.isGrayscale)
|
||||||
return ReaderColorFilter(brightness, contrast, inverted, grayscale).takeUnless { it.isEmpty }
|
ReaderColorFilter(brightness, contrast, inverted, grayscale).takeUnless { it.isEmpty }
|
||||||
}
|
}.getOrNull()
|
||||||
set(value) {
|
set(value) {
|
||||||
prefs.edit {
|
prefs.edit {
|
||||||
val cf = value ?: ReaderColorFilter.EMPTY
|
val cf = value ?: ReaderColorFilter.EMPTY
|
||||||
@@ -344,15 +364,23 @@ class AppSettings @Inject constructor(@ApplicationContext context: Context) {
|
|||||||
set(value) = prefs.edit { putEnumValue(KEY_LOCAL_LIST_ORDER, value) }
|
set(value) = prefs.edit { putEnumValue(KEY_LOCAL_LIST_ORDER, value) }
|
||||||
|
|
||||||
var historySortOrder: ListSortOrder
|
var historySortOrder: ListSortOrder
|
||||||
get() = prefs.getEnumValue(KEY_HISTORY_ORDER, ListSortOrder.UPDATED)
|
get() = prefs.getEnumValue(KEY_HISTORY_ORDER, ListSortOrder.LAST_READ)
|
||||||
set(value) = prefs.edit { putEnumValue(KEY_HISTORY_ORDER, value) }
|
set(value) = prefs.edit { putEnumValue(KEY_HISTORY_ORDER, value) }
|
||||||
|
|
||||||
|
var allFavoritesSortOrder: ListSortOrder
|
||||||
|
get() = prefs.getEnumValue(KEY_FAVORITES_ORDER, ListSortOrder.NEWEST)
|
||||||
|
set(value) = prefs.edit { putEnumValue(KEY_FAVORITES_ORDER, value) }
|
||||||
|
|
||||||
val isRelatedMangaEnabled: Boolean
|
val isRelatedMangaEnabled: Boolean
|
||||||
get() = prefs.getBoolean(KEY_RELATED_MANGA, true)
|
get() = prefs.getBoolean(KEY_RELATED_MANGA, true)
|
||||||
|
|
||||||
val isWebtoonZoomEnable: Boolean
|
val isWebtoonZoomEnable: Boolean
|
||||||
get() = prefs.getBoolean(KEY_WEBTOON_ZOOM, true)
|
get() = prefs.getBoolean(KEY_WEBTOON_ZOOM, true)
|
||||||
|
|
||||||
|
@get:FloatRange(from = 0.0, to = 0.5)
|
||||||
|
val defaultWebtoonZoomOut: Float
|
||||||
|
get() = prefs.getInt(KEY_WEBTOON_ZOOM_OUT, 0).coerceIn(0, 50) / 100f
|
||||||
|
|
||||||
@get:FloatRange(from = 0.0, to = 1.0)
|
@get:FloatRange(from = 0.0, to = 1.0)
|
||||||
var readerAutoscrollSpeed: Float
|
var readerAutoscrollSpeed: Float
|
||||||
get() = prefs.getFloat(KEY_READER_AUTOSCROLL_SPEED, 0f)
|
get() = prefs.getFloat(KEY_READER_AUTOSCROLL_SPEED, 0f)
|
||||||
@@ -388,6 +416,15 @@ class AppSettings @Inject constructor(@ApplicationContext context: Context) {
|
|||||||
get() = prefs.getString(KEY_BACKUP_PERIODICAL_OUTPUT, null)?.toUriOrNull()
|
get() = prefs.getString(KEY_BACKUP_PERIODICAL_OUTPUT, null)?.toUriOrNull()
|
||||||
set(value) = prefs.edit { putString(KEY_BACKUP_PERIODICAL_OUTPUT, value?.toString()) }
|
set(value) = prefs.edit { putString(KEY_BACKUP_PERIODICAL_OUTPUT, value?.toString()) }
|
||||||
|
|
||||||
|
val isReadingTimeEstimationEnabled: Boolean
|
||||||
|
get() = prefs.getBoolean(KEY_READING_TIME, true)
|
||||||
|
|
||||||
|
val isPagesSavingAskEnabled: Boolean
|
||||||
|
get() = prefs.getBoolean(KEY_PAGES_SAVE_ASK, true)
|
||||||
|
|
||||||
|
val isStatsEnabled: Boolean
|
||||||
|
get() = prefs.getBoolean(KEY_STATS_ENABLED, false)
|
||||||
|
|
||||||
fun isTipEnabled(tip: String): Boolean {
|
fun isTipEnabled(tip: String): Boolean {
|
||||||
return prefs.getStringSet(KEY_TIPS_CLOSED, emptySet())?.contains(tip) != true
|
return prefs.getStringSet(KEY_TIPS_CLOSED, emptySet())?.contains(tip) != true
|
||||||
}
|
}
|
||||||
@@ -400,6 +437,15 @@ class AppSettings @Inject constructor(@ApplicationContext context: Context) {
|
|||||||
prefs.edit { putStringSet(KEY_TIPS_CLOSED, closedTips + tip) }
|
prefs.edit { putStringSet(KEY_TIPS_CLOSED, closedTips + tip) }
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fun getPagesSaveDir(context: Context): DocumentFile? =
|
||||||
|
prefs.getString(KEY_PAGES_SAVE_DIR, null)?.toUriOrNull()?.let {
|
||||||
|
DocumentFile.fromTreeUri(context, it)?.takeIf { it.canWrite() }
|
||||||
|
}
|
||||||
|
|
||||||
|
fun setPagesSaveDir(uri: Uri?) {
|
||||||
|
prefs.edit { putString(KEY_PAGES_SAVE_DIR, uri?.toString()) }
|
||||||
|
}
|
||||||
|
|
||||||
fun subscribe(listener: SharedPreferences.OnSharedPreferenceChangeListener) {
|
fun subscribe(listener: SharedPreferences.OnSharedPreferenceChangeListener) {
|
||||||
prefs.registerOnSharedPreferenceChangeListener(listener)
|
prefs.registerOnSharedPreferenceChangeListener(listener)
|
||||||
}
|
}
|
||||||
@@ -446,7 +492,6 @@ class AppSettings @Inject constructor(@ApplicationContext context: Context) {
|
|||||||
|
|
||||||
companion object {
|
companion object {
|
||||||
|
|
||||||
const val PAGE_SWITCH_TAPS = "taps"
|
|
||||||
const val PAGE_SWITCH_VOLUME_KEYS = "volume"
|
const val PAGE_SWITCH_VOLUME_KEYS = "volume"
|
||||||
|
|
||||||
const val TRACK_HISTORY = "history"
|
const val TRACK_HISTORY = "history"
|
||||||
@@ -469,8 +514,11 @@ class AppSettings @Inject constructor(@ApplicationContext context: Context) {
|
|||||||
const val KEY_GRID_SIZE = "grid_size"
|
const val KEY_GRID_SIZE = "grid_size"
|
||||||
const val KEY_REMOTE_SOURCES = "remote_sources"
|
const val KEY_REMOTE_SOURCES = "remote_sources"
|
||||||
const val KEY_LOCAL_STORAGE = "local_storage"
|
const val KEY_LOCAL_STORAGE = "local_storage"
|
||||||
const val KEY_READER_SWITCHERS = "reader_switchers"
|
const val KEY_READER_DOUBLE_PAGES = "reader_double_pages"
|
||||||
const val KEY_READER_ZOOM_BUTTONS = "reader_zoom_buttons"
|
const val KEY_READER_ZOOM_BUTTONS = "reader_zoom_buttons"
|
||||||
|
const val KEY_READER_CONTROL_LTR = "reader_taps_ltr"
|
||||||
|
const val KEY_READER_FULLSCREEN = "reader_fullscreen"
|
||||||
|
const val KEY_READER_VOLUME_BUTTONS = "reader_volume_buttons"
|
||||||
const val KEY_TRACKER_ENABLED = "tracker_enabled"
|
const val KEY_TRACKER_ENABLED = "tracker_enabled"
|
||||||
const val KEY_TRACKER_WIFI_ONLY = "tracker_wifi"
|
const val KEY_TRACKER_WIFI_ONLY = "tracker_wifi"
|
||||||
const val KEY_TRACK_SOURCES = "track_sources"
|
const val KEY_TRACK_SOURCES = "track_sources"
|
||||||
@@ -486,6 +534,7 @@ class AppSettings @Inject constructor(@ApplicationContext context: Context) {
|
|||||||
const val KEY_READER_MODE = "reader_mode"
|
const val KEY_READER_MODE = "reader_mode"
|
||||||
const val KEY_READER_MODE_DETECT = "reader_mode_detect"
|
const val KEY_READER_MODE_DETECT = "reader_mode_detect"
|
||||||
const val KEY_APP_PASSWORD = "app_password"
|
const val KEY_APP_PASSWORD = "app_password"
|
||||||
|
const val KEY_APP_PASSWORD_NUMERIC = "app_password_num"
|
||||||
const val KEY_PROTECT_APP = "protect_app"
|
const val KEY_PROTECT_APP = "protect_app"
|
||||||
const val KEY_PROTECT_APP_BIOMETRIC = "protect_app_bio"
|
const val KEY_PROTECT_APP_BIOMETRIC = "protect_app_bio"
|
||||||
const val KEY_APP_VERSION = "app_version"
|
const val KEY_APP_VERSION = "app_version"
|
||||||
@@ -511,7 +560,9 @@ class AppSettings @Inject constructor(@ApplicationContext context: Context) {
|
|||||||
const val KEY_SHIKIMORI = "shikimori"
|
const val KEY_SHIKIMORI = "shikimori"
|
||||||
const val KEY_ANILIST = "anilist"
|
const val KEY_ANILIST = "anilist"
|
||||||
const val KEY_MAL = "mal"
|
const val KEY_MAL = "mal"
|
||||||
|
const val KEY_KITSU = "kitsu"
|
||||||
const val KEY_DOWNLOADS_WIFI = "downloads_wifi"
|
const val KEY_DOWNLOADS_WIFI = "downloads_wifi"
|
||||||
|
const val KEY_DOWNLOADS_FORMAT = "downloads_format"
|
||||||
const val KEY_ALL_FAVOURITES_VISIBLE = "all_favourites_visible"
|
const val KEY_ALL_FAVOURITES_VISIBLE = "all_favourites_visible"
|
||||||
const val KEY_DOH = "doh"
|
const val KEY_DOH = "doh"
|
||||||
const val KEY_EXIT_CONFIRM = "exit_confirm"
|
const val KEY_EXIT_CONFIRM = "exit_confirm"
|
||||||
@@ -523,11 +574,13 @@ class AppSettings @Inject constructor(@ApplicationContext context: Context) {
|
|||||||
const val KEY_READER_BACKGROUND = "reader_background"
|
const val KEY_READER_BACKGROUND = "reader_background"
|
||||||
const val KEY_READER_SCREEN_ON = "reader_screen_on"
|
const val KEY_READER_SCREEN_ON = "reader_screen_on"
|
||||||
const val KEY_SHORTCUTS = "dynamic_shortcuts"
|
const val KEY_SHORTCUTS = "dynamic_shortcuts"
|
||||||
const val KEY_READER_TAPS_LTR = "reader_taps_ltr"
|
const val KEY_READER_TAP_ACTIONS = "reader_tap_actions"
|
||||||
const val KEY_READER_OPTIMIZE = "reader_optimize"
|
const val KEY_READER_OPTIMIZE = "reader_optimize"
|
||||||
const val KEY_LOCAL_LIST_ORDER = "local_order"
|
const val KEY_LOCAL_LIST_ORDER = "local_order"
|
||||||
const val KEY_HISTORY_ORDER = "history_order"
|
const val KEY_HISTORY_ORDER = "history_order"
|
||||||
|
const val KEY_FAVORITES_ORDER = "fav_order"
|
||||||
const val KEY_WEBTOON_ZOOM = "webtoon_zoom"
|
const val KEY_WEBTOON_ZOOM = "webtoon_zoom"
|
||||||
|
const val KEY_WEBTOON_ZOOM_OUT = "webtoon_zoom_out"
|
||||||
const val KEY_PREFETCH_CONTENT = "prefetch_content"
|
const val KEY_PREFETCH_CONTENT = "prefetch_content"
|
||||||
const val KEY_APP_LOCALE = "app_locale"
|
const val KEY_APP_LOCALE = "app_locale"
|
||||||
const val KEY_LOGGING_ENABLED = "logging"
|
const val KEY_LOGGING_ENABLED = "logging"
|
||||||
@@ -551,6 +604,7 @@ class AppSettings @Inject constructor(@ApplicationContext context: Context) {
|
|||||||
const val KEY_DISABLE_NSFW = "no_nsfw"
|
const val KEY_DISABLE_NSFW = "no_nsfw"
|
||||||
const val KEY_RELATED_MANGA = "related_manga"
|
const val KEY_RELATED_MANGA = "related_manga"
|
||||||
const val KEY_NAV_MAIN = "nav_main"
|
const val KEY_NAV_MAIN = "nav_main"
|
||||||
|
const val KEY_NAV_LABELS = "nav_labels"
|
||||||
const val KEY_32BIT_COLOR = "enhanced_colors"
|
const val KEY_32BIT_COLOR = "enhanced_colors"
|
||||||
const val KEY_SOURCES_ORDER = "sources_sort_order"
|
const val KEY_SOURCES_ORDER = "sources_sort_order"
|
||||||
const val KEY_SOURCES_CATALOG = "sources_catalog"
|
const val KEY_SOURCES_CATALOG = "sources_catalog"
|
||||||
@@ -559,8 +613,11 @@ class AppSettings @Inject constructor(@ApplicationContext context: Context) {
|
|||||||
const val KEY_CF_INVERTED = "cf_inverted"
|
const val KEY_CF_INVERTED = "cf_inverted"
|
||||||
const val KEY_CF_GRAYSCALE = "cf_grayscale"
|
const val KEY_CF_GRAYSCALE = "cf_grayscale"
|
||||||
const val KEY_IGNORE_DOZE = "ignore_dose"
|
const val KEY_IGNORE_DOZE = "ignore_dose"
|
||||||
|
const val KEY_DETAILS_TAB = "details_tab"
|
||||||
// About
|
const val KEY_READING_TIME = "reading_time"
|
||||||
|
const val KEY_PAGES_SAVE_DIR = "pages_dir"
|
||||||
|
const val KEY_PAGES_SAVE_ASK = "pages_dir_ask"
|
||||||
|
const val KEY_STATS_ENABLED = "stats_on"
|
||||||
const val KEY_APP_UPDATE = "app_update"
|
const val KEY_APP_UPDATE = "app_update"
|
||||||
const val KEY_APP_TRANSLATION = "about_app_translation"
|
const val KEY_APP_TRANSLATION = "about_app_translation"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,8 @@
|
|||||||
|
package org.koitharu.kotatsu.core.prefs
|
||||||
|
|
||||||
|
enum class DownloadFormat {
|
||||||
|
|
||||||
|
AUTOMATIC,
|
||||||
|
SINGLE_CBZ,
|
||||||
|
MULTIPLE_CBZ,
|
||||||
|
}
|
||||||
@@ -4,13 +4,12 @@ import androidx.annotation.DrawableRes
|
|||||||
import androidx.annotation.IdRes
|
import androidx.annotation.IdRes
|
||||||
import androidx.annotation.StringRes
|
import androidx.annotation.StringRes
|
||||||
import org.koitharu.kotatsu.R
|
import org.koitharu.kotatsu.R
|
||||||
import org.koitharu.kotatsu.list.ui.model.ListModel
|
|
||||||
|
|
||||||
enum class NavItem(
|
enum class NavItem(
|
||||||
@IdRes val id: Int,
|
@IdRes val id: Int,
|
||||||
@StringRes val title: Int,
|
@StringRes val title: Int,
|
||||||
@DrawableRes val icon: Int,
|
@DrawableRes val icon: Int,
|
||||||
) : ListModel {
|
) {
|
||||||
|
|
||||||
HISTORY(R.id.nav_history, R.string.history, R.drawable.ic_history_selector),
|
HISTORY(R.id.nav_history, R.string.history, R.drawable.ic_history_selector),
|
||||||
FAVORITES(R.id.nav_favorites, R.string.favourites, R.drawable.ic_favourites_selector),
|
FAVORITES(R.id.nav_favorites, R.string.favourites, R.drawable.ic_favourites_selector),
|
||||||
@@ -21,10 +20,6 @@ enum class NavItem(
|
|||||||
BOOKMARKS(R.id.nav_bookmarks, R.string.bookmarks, R.drawable.ic_bookmark_selector),
|
BOOKMARKS(R.id.nav_bookmarks, R.string.bookmarks, R.drawable.ic_bookmark_selector),
|
||||||
;
|
;
|
||||||
|
|
||||||
override fun areItemsTheSame(other: ListModel): Boolean {
|
|
||||||
return other is NavItem && ordinal == other.ordinal
|
|
||||||
}
|
|
||||||
|
|
||||||
fun isAvailable(settings: AppSettings): Boolean = when (this) {
|
fun isAvailable(settings: AppSettings): Boolean = when (this) {
|
||||||
SUGGESTIONS -> settings.isSuggestionsEnabled
|
SUGGESTIONS -> settings.isSuggestionsEnabled
|
||||||
FEED -> settings.isTrackerEnabled
|
FEED -> settings.isTrackerEnabled
|
||||||
|
|||||||
@@ -4,7 +4,9 @@ enum class ReaderMode(val id: Int) {
|
|||||||
|
|
||||||
STANDARD(1),
|
STANDARD(1),
|
||||||
REVERSED(3),
|
REVERSED(3),
|
||||||
WEBTOON(2);
|
VERTICAL(4),
|
||||||
|
WEBTOON(2),
|
||||||
|
;
|
||||||
|
|
||||||
companion object {
|
companion object {
|
||||||
|
|
||||||
|
|||||||
@@ -8,6 +8,7 @@ import android.os.Bundle
|
|||||||
import android.view.KeyEvent
|
import android.view.KeyEvent
|
||||||
import android.view.View
|
import android.view.View
|
||||||
import android.view.ViewGroup
|
import android.view.ViewGroup
|
||||||
|
import android.widget.Toast
|
||||||
import androidx.annotation.CallSuper
|
import androidx.annotation.CallSuper
|
||||||
import androidx.appcompat.app.AppCompatActivity
|
import androidx.appcompat.app.AppCompatActivity
|
||||||
import androidx.appcompat.view.ActionMode
|
import androidx.appcompat.view.ActionMode
|
||||||
@@ -29,6 +30,7 @@ import org.koitharu.kotatsu.core.ui.util.ActionModeDelegate
|
|||||||
import org.koitharu.kotatsu.core.ui.util.BaseActivityEntryPoint
|
import org.koitharu.kotatsu.core.ui.util.BaseActivityEntryPoint
|
||||||
import org.koitharu.kotatsu.core.ui.util.WindowInsetsDelegate
|
import org.koitharu.kotatsu.core.ui.util.WindowInsetsDelegate
|
||||||
import org.koitharu.kotatsu.core.util.ext.getThemeColor
|
import org.koitharu.kotatsu.core.util.ext.getThemeColor
|
||||||
|
import org.koitharu.kotatsu.core.util.ext.isWebViewUnavailable
|
||||||
|
|
||||||
@Suppress("LeakingThis")
|
@Suppress("LeakingThis")
|
||||||
abstract class BaseActivity<B : ViewBinding> :
|
abstract class BaseActivity<B : ViewBinding> :
|
||||||
@@ -164,6 +166,21 @@ abstract class BaseActivity<B : ViewBinding> :
|
|||||||
intent?.putExtra(EXTRA_DATA, intent.data)
|
intent?.putExtra(EXTRA_DATA, intent.data)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
protected fun setContentViewWebViewSafe(viewBindingProducer: () -> B): Boolean {
|
||||||
|
return try {
|
||||||
|
setContentView(viewBindingProducer())
|
||||||
|
true
|
||||||
|
} catch (e: Exception) {
|
||||||
|
if (e.isWebViewUnavailable()) {
|
||||||
|
Toast.makeText(this, R.string.web_view_unavailable, Toast.LENGTH_LONG).show()
|
||||||
|
finishAfterTransition()
|
||||||
|
false
|
||||||
|
} else {
|
||||||
|
throw e
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
companion object {
|
companion object {
|
||||||
|
|
||||||
const val EXTRA_DATA = "data"
|
const val EXTRA_DATA = "data"
|
||||||
|
|||||||
@@ -29,7 +29,6 @@ abstract class BaseFullscreenActivity<B : ViewBinding> :
|
|||||||
WindowManager.LayoutParams.LAYOUT_IN_DISPLAY_CUTOUT_MODE_SHORT_EDGES
|
WindowManager.LayoutParams.LAYOUT_IN_DISPLAY_CUTOUT_MODE_SHORT_EDGES
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// insetsControllerCompat.systemBarsBehavior = WindowInsetsControllerCompat.BEHAVIOR_SHOW_TRANSIENT_BARS_BY_SWIPE
|
|
||||||
systemUiController.setSystemUiVisible(true)
|
systemUiController.setSystemUiVisible(true)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -10,6 +10,7 @@ import kotlinx.coroutines.flow.FlowCollector
|
|||||||
import org.koitharu.kotatsu.core.util.ContinuationResumeRunnable
|
import org.koitharu.kotatsu.core.util.ContinuationResumeRunnable
|
||||||
import org.koitharu.kotatsu.list.ui.ListModelDiffCallback
|
import org.koitharu.kotatsu.list.ui.ListModelDiffCallback
|
||||||
import org.koitharu.kotatsu.list.ui.adapter.ListItemType
|
import org.koitharu.kotatsu.list.ui.adapter.ListItemType
|
||||||
|
import org.koitharu.kotatsu.list.ui.model.ListHeader
|
||||||
import org.koitharu.kotatsu.list.ui.model.ListModel
|
import org.koitharu.kotatsu.list.ui.model.ListModel
|
||||||
import kotlin.coroutines.suspendCoroutine
|
import kotlin.coroutines.suspendCoroutine
|
||||||
|
|
||||||
@@ -28,11 +29,23 @@ open class BaseListAdapter<T : ListModel> : AsyncListDifferDelegationAdapter<T>(
|
|||||||
return this
|
return this
|
||||||
}
|
}
|
||||||
|
|
||||||
fun addListListener(listListener: ListListener<T>) {
|
fun addListListener(listListener: ListListener<T>): BaseListAdapter<T> {
|
||||||
differ.addListListener(listListener)
|
differ.addListListener(listListener)
|
||||||
|
return this
|
||||||
}
|
}
|
||||||
|
|
||||||
fun removeListListener(listListener: ListListener<T>) {
|
fun removeListListener(listListener: ListListener<T>) {
|
||||||
differ.removeListListener(listListener)
|
differ.removeListListener(listListener)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fun findHeader(position: Int): ListHeader? {
|
||||||
|
val snapshot = items
|
||||||
|
for (i in (0..position).reversed()) {
|
||||||
|
val item = snapshot.getOrNull(i) ?: continue
|
||||||
|
if (item is ListHeader) {
|
||||||
|
return item
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return null
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,5 +1,7 @@
|
|||||||
package org.koitharu.kotatsu.core.ui
|
package org.koitharu.kotatsu.core.ui
|
||||||
|
|
||||||
|
import android.content.ActivityNotFoundException
|
||||||
|
import android.content.Intent
|
||||||
import android.os.Bundle
|
import android.os.Bundle
|
||||||
import android.view.View
|
import android.view.View
|
||||||
import androidx.annotation.CallSuper
|
import androidx.annotation.CallSuper
|
||||||
@@ -8,7 +10,9 @@ import androidx.core.graphics.Insets
|
|||||||
import androidx.core.view.updatePadding
|
import androidx.core.view.updatePadding
|
||||||
import androidx.preference.PreferenceFragmentCompat
|
import androidx.preference.PreferenceFragmentCompat
|
||||||
import androidx.recyclerview.widget.RecyclerView
|
import androidx.recyclerview.widget.RecyclerView
|
||||||
|
import com.google.android.material.snackbar.Snackbar
|
||||||
import dagger.hilt.android.AndroidEntryPoint
|
import dagger.hilt.android.AndroidEntryPoint
|
||||||
|
import org.koitharu.kotatsu.R
|
||||||
import org.koitharu.kotatsu.core.prefs.AppSettings
|
import org.koitharu.kotatsu.core.prefs.AppSettings
|
||||||
import org.koitharu.kotatsu.core.ui.util.RecyclerViewOwner
|
import org.koitharu.kotatsu.core.ui.util.RecyclerViewOwner
|
||||||
import org.koitharu.kotatsu.core.ui.util.WindowInsetsDelegate
|
import org.koitharu.kotatsu.core.ui.util.WindowInsetsDelegate
|
||||||
@@ -62,4 +66,12 @@ abstract class BasePreferenceFragment(@StringRes private val titleId: Int) :
|
|||||||
protected fun setTitle(title: CharSequence?) {
|
protected fun setTitle(title: CharSequence?) {
|
||||||
(activity as? SettingsActivity)?.setSectionTitle(title)
|
(activity as? SettingsActivity)?.setSectionTitle(title)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
protected fun startActivitySafe(intent: Intent) {
|
||||||
|
try {
|
||||||
|
startActivity(intent)
|
||||||
|
} catch (_: ActivityNotFoundException) {
|
||||||
|
Snackbar.make(listView, R.string.operation_not_supported, Snackbar.LENGTH_SHORT).show()
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -68,6 +68,13 @@ abstract class BaseViewModel : ViewModel() {
|
|||||||
errorEvent.call(error)
|
errorEvent.call(error)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
protected inline suspend fun <T> withLoading(block: () -> T): T = try {
|
||||||
|
loadingCounter.increment()
|
||||||
|
block()
|
||||||
|
} finally {
|
||||||
|
loadingCounter.decrement()
|
||||||
|
}
|
||||||
|
|
||||||
protected fun MutableStateFlow<Int>.increment() = update { it + 1 }
|
protected fun MutableStateFlow<Int>.increment() = update { it + 1 }
|
||||||
|
|
||||||
protected fun MutableStateFlow<Int>.decrement() = update { it - 1 }
|
protected fun MutableStateFlow<Int>.decrement() = update { it - 1 }
|
||||||
|
|||||||
@@ -68,6 +68,14 @@ class RecyclerViewAlertDialog private constructor(
|
|||||||
return this
|
return this
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fun setNeutralButton(
|
||||||
|
@StringRes textId: Int,
|
||||||
|
listener: DialogInterface.OnClickListener,
|
||||||
|
): Builder<T> {
|
||||||
|
delegate.setNeutralButton(textId, listener)
|
||||||
|
return this
|
||||||
|
}
|
||||||
|
|
||||||
fun setCancelable(isCancelable: Boolean): Builder<T> {
|
fun setCancelable(isCancelable: Boolean): Builder<T> {
|
||||||
delegate.setCancelable(isCancelable)
|
delegate.setCancelable(isCancelable)
|
||||||
return this
|
return this
|
||||||
|
|||||||
@@ -12,11 +12,10 @@ import android.graphics.RectF
|
|||||||
import android.graphics.drawable.Drawable
|
import android.graphics.drawable.Drawable
|
||||||
import androidx.annotation.StyleRes
|
import androidx.annotation.StyleRes
|
||||||
import androidx.core.content.withStyledAttributes
|
import androidx.core.content.withStyledAttributes
|
||||||
import androidx.core.graphics.ColorUtils
|
|
||||||
import androidx.core.graphics.withClip
|
import androidx.core.graphics.withClip
|
||||||
import com.google.android.material.color.MaterialColors
|
import com.google.android.material.color.MaterialColors
|
||||||
import org.koitharu.kotatsu.R
|
import org.koitharu.kotatsu.R
|
||||||
import kotlin.math.absoluteValue
|
import org.koitharu.kotatsu.core.util.KotatsuColors
|
||||||
|
|
||||||
class FaviconDrawable(
|
class FaviconDrawable(
|
||||||
context: Context,
|
context: Context,
|
||||||
@@ -44,7 +43,7 @@ class FaviconDrawable(
|
|||||||
}
|
}
|
||||||
paint.textAlign = Paint.Align.CENTER
|
paint.textAlign = Paint.Align.CENTER
|
||||||
paint.isFakeBoldText = true
|
paint.isFakeBoldText = true
|
||||||
colorForeground = MaterialColors.harmonize(colorOfString(name), colorBackground)
|
colorForeground = MaterialColors.harmonize(KotatsuColors.random(name), colorBackground)
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun draw(canvas: Canvas) {
|
override fun draw(canvas: Canvas) {
|
||||||
@@ -104,9 +103,4 @@ class FaviconDrawable(
|
|||||||
paint.getTextBounds(text, 0, text.length, tempRect)
|
paint.getTextBounds(text, 0, text.length, tempRect)
|
||||||
return testTextSize * width / tempRect.width()
|
return testTextSize * width / tempRect.width()
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun colorOfString(str: String): Int {
|
|
||||||
val hue = (str.hashCode() % 360).absoluteValue.toFloat()
|
|
||||||
return ColorUtils.HSLToColor(floatArrayOf(hue, 0.5f, 0.5f))
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -38,4 +38,12 @@ abstract class BoundsScrollListener(
|
|||||||
firstVisibleItemPosition: Int,
|
firstVisibleItemPosition: Int,
|
||||||
visibleItemCount: Int
|
visibleItemCount: Int
|
||||||
) = Unit
|
) = Unit
|
||||||
|
|
||||||
|
fun invalidate(recyclerView: RecyclerView) {
|
||||||
|
onScrolled(recyclerView, 0, 0)
|
||||||
|
}
|
||||||
|
|
||||||
|
fun postInvalidate(recyclerView: RecyclerView) = recyclerView.post {
|
||||||
|
invalidate(recyclerView)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -91,7 +91,7 @@ abstract class AbstractSelectionItemDecoration : RecyclerView.ItemDecoration() {
|
|||||||
canvas.restoreToCount(checkpoint)
|
canvas.restoreToCount(checkpoint)
|
||||||
}
|
}
|
||||||
|
|
||||||
protected open fun getItemId(parent: RecyclerView, child: View) = parent.getChildItemId(child)
|
abstract fun getItemId(parent: RecyclerView, child: View): Long
|
||||||
|
|
||||||
protected open fun onDrawBackground(
|
protected open fun onDrawBackground(
|
||||||
canvas: Canvas,
|
canvas: Canvas,
|
||||||
|
|||||||
@@ -4,8 +4,10 @@ import android.content.Context
|
|||||||
import android.util.AttributeSet
|
import android.util.AttributeSet
|
||||||
import android.view.ViewGroup
|
import android.view.ViewGroup
|
||||||
import androidx.annotation.AttrRes
|
import androidx.annotation.AttrRes
|
||||||
|
import androidx.core.view.ancestors
|
||||||
import androidx.core.view.isVisible
|
import androidx.core.view.isVisible
|
||||||
import androidx.recyclerview.widget.RecyclerView
|
import androidx.recyclerview.widget.RecyclerView
|
||||||
|
import androidx.viewpager2.widget.ViewPager2
|
||||||
import org.koitharu.kotatsu.R
|
import org.koitharu.kotatsu.R
|
||||||
|
|
||||||
class FastScrollRecyclerView @JvmOverloads constructor(
|
class FastScrollRecyclerView @JvmOverloads constructor(
|
||||||
@@ -15,6 +17,16 @@ class FastScrollRecyclerView @JvmOverloads constructor(
|
|||||||
) : RecyclerView(context, attrs, defStyleAttr) {
|
) : RecyclerView(context, attrs, defStyleAttr) {
|
||||||
|
|
||||||
val fastScroller = FastScroller(context, attrs)
|
val fastScroller = FastScroller(context, attrs)
|
||||||
|
var isVP2BugWorkaroundEnabled = false
|
||||||
|
set(value) {
|
||||||
|
field = value
|
||||||
|
if (value && isAttachedToWindow) {
|
||||||
|
checkIfInVP2()
|
||||||
|
} else if (!value) {
|
||||||
|
applyVP2Workaround = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
private var applyVP2Workaround = false
|
||||||
|
|
||||||
var isFastScrollerEnabled: Boolean = true
|
var isFastScrollerEnabled: Boolean = true
|
||||||
set(value) {
|
set(value) {
|
||||||
@@ -43,10 +55,29 @@ class FastScrollRecyclerView @JvmOverloads constructor(
|
|||||||
override fun onAttachedToWindow() {
|
override fun onAttachedToWindow() {
|
||||||
super.onAttachedToWindow()
|
super.onAttachedToWindow()
|
||||||
fastScroller.attachRecyclerView(this)
|
fastScroller.attachRecyclerView(this)
|
||||||
|
if (isVP2BugWorkaroundEnabled) {
|
||||||
|
checkIfInVP2()
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun onDetachedFromWindow() {
|
override fun onDetachedFromWindow() {
|
||||||
fastScroller.detachRecyclerView()
|
fastScroller.detachRecyclerView()
|
||||||
super.onDetachedFromWindow()
|
super.onDetachedFromWindow()
|
||||||
|
applyVP2Workaround = false
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun isLayoutRequested(): Boolean {
|
||||||
|
return if (applyVP2Workaround) false else super.isLayoutRequested()
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun requestLayout() {
|
||||||
|
super.requestLayout()
|
||||||
|
if (applyVP2Workaround && parent?.isLayoutRequested == true) {
|
||||||
|
parent?.requestLayout()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun checkIfInVP2() {
|
||||||
|
applyVP2Workaround = ancestors.any { it is ViewPager2 } == true
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -2,9 +2,8 @@ package org.koitharu.kotatsu.core.ui.model
|
|||||||
|
|
||||||
import android.content.res.Resources
|
import android.content.res.Resources
|
||||||
import org.koitharu.kotatsu.R
|
import org.koitharu.kotatsu.R
|
||||||
import org.koitharu.kotatsu.core.util.ext.daysDiff
|
import java.time.LocalDate
|
||||||
import org.koitharu.kotatsu.core.util.ext.format
|
import java.time.format.DateTimeFormatter
|
||||||
import java.util.Date
|
|
||||||
|
|
||||||
sealed class DateTimeAgo {
|
sealed class DateTimeAgo {
|
||||||
|
|
||||||
@@ -74,32 +73,22 @@ sealed class DateTimeAgo {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
class Absolute(private val date: Date) : DateTimeAgo() {
|
data class Absolute(private val date: LocalDate) : DateTimeAgo() {
|
||||||
|
|
||||||
private val day = date.daysDiff(0)
|
|
||||||
|
|
||||||
override fun format(resources: Resources): String {
|
override fun format(resources: Resources): String {
|
||||||
return if (date.time == 0L) {
|
return if (date == EPOCH_DATE) {
|
||||||
resources.getString(R.string.unknown)
|
resources.getString(R.string.unknown)
|
||||||
} else {
|
} else {
|
||||||
date.format("d MMMM")
|
date.format(formatter)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun equals(other: Any?): Boolean {
|
override fun toString() = "abs_${date.toEpochDay()}"
|
||||||
if (this === other) return true
|
|
||||||
if (javaClass != other?.javaClass) return false
|
|
||||||
|
|
||||||
other as Absolute
|
companion object {
|
||||||
|
// TODO: Use Java 9's LocalDate.EPOCH.
|
||||||
return day == other.day
|
private val EPOCH_DATE = LocalDate.of(1970, 1, 1)
|
||||||
|
private val formatter = DateTimeFormatter.ofPattern("d MMMM")
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun hashCode(): Int {
|
|
||||||
return day
|
|
||||||
}
|
|
||||||
|
|
||||||
override fun toString() = "abs_$day"
|
|
||||||
}
|
}
|
||||||
|
|
||||||
object LongAgo : DateTimeAgo() {
|
object LongAgo : DateTimeAgo() {
|
||||||
|
|||||||
@@ -12,4 +12,5 @@ val SortOrder.titleRes: Int
|
|||||||
SortOrder.RATING -> R.string.by_rating
|
SortOrder.RATING -> R.string.by_rating
|
||||||
SortOrder.NEWEST -> R.string.newest
|
SortOrder.NEWEST -> R.string.newest
|
||||||
SortOrder.ALPHABETICAL -> R.string.by_name
|
SortOrder.ALPHABETICAL -> R.string.by_name
|
||||||
|
SortOrder.ALPHABETICAL_DESC -> R.string.by_name_reverse
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,38 @@
|
|||||||
|
package org.koitharu.kotatsu.core.ui.util
|
||||||
|
|
||||||
|
import android.view.MenuItem
|
||||||
|
import android.view.View
|
||||||
|
import androidx.appcompat.widget.PopupMenu
|
||||||
|
import androidx.core.view.MenuProvider
|
||||||
|
import org.koitharu.kotatsu.core.util.ext.setOnContextClickListenerCompat
|
||||||
|
|
||||||
|
class PopupMenuMediator(
|
||||||
|
private val provider: MenuProvider,
|
||||||
|
) : View.OnLongClickListener, PopupMenu.OnMenuItemClickListener, PopupMenu.OnDismissListener {
|
||||||
|
|
||||||
|
override fun onLongClick(v: View): Boolean {
|
||||||
|
val menu = PopupMenu(v.context, v)
|
||||||
|
provider.onCreateMenu(menu.menu, menu.menuInflater)
|
||||||
|
provider.onPrepareMenu(menu.menu)
|
||||||
|
if (!menu.menu.hasVisibleItems()) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
menu.setOnMenuItemClickListener(this)
|
||||||
|
menu.setOnDismissListener(this)
|
||||||
|
menu.show()
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun onMenuItemClick(item: MenuItem): Boolean {
|
||||||
|
return provider.onMenuItemSelected(item)
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun onDismiss(menu: PopupMenu) {
|
||||||
|
provider.onMenuClosed(menu.menu)
|
||||||
|
}
|
||||||
|
|
||||||
|
fun attach(view: View) {
|
||||||
|
view.setOnLongClickListener(this)
|
||||||
|
view.setOnContextClickListenerCompat(this)
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -7,12 +7,16 @@ import org.koitharu.kotatsu.R
|
|||||||
|
|
||||||
class ReversibleActionObserver(
|
class ReversibleActionObserver(
|
||||||
private val snackbarHost: View,
|
private val snackbarHost: View,
|
||||||
|
private val snackbarAnchor: View? = null,
|
||||||
) : FlowCollector<ReversibleAction> {
|
) : FlowCollector<ReversibleAction> {
|
||||||
|
|
||||||
override suspend fun emit(value: ReversibleAction) {
|
override suspend fun emit(value: ReversibleAction) {
|
||||||
val handle = value.handle
|
val handle = value.handle
|
||||||
val length = if (handle == null) Snackbar.LENGTH_SHORT else Snackbar.LENGTH_LONG
|
val length = if (handle == null) Snackbar.LENGTH_SHORT else Snackbar.LENGTH_LONG
|
||||||
val snackbar = Snackbar.make(snackbarHost, value.stringResId, length)
|
val snackbar = Snackbar.make(snackbarHost, value.stringResId, length)
|
||||||
|
if (snackbarAnchor?.isShown == true) {
|
||||||
|
snackbar.anchorView = snackbarAnchor
|
||||||
|
}
|
||||||
if (handle != null) {
|
if (handle != null) {
|
||||||
snackbar.setAction(R.string.undo) { handle.reverseAsync() }
|
snackbar.setAction(R.string.undo) { handle.reverseAsync() }
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -108,10 +108,9 @@ class ChipsView @JvmOverloads constructor(
|
|||||||
chip.setChipDrawable(drawable)
|
chip.setChipDrawable(drawable)
|
||||||
chip.isCheckedIconVisible = true
|
chip.isCheckedIconVisible = true
|
||||||
chip.isChipIconVisible = false
|
chip.isChipIconVisible = false
|
||||||
chip.setCheckedIconResource(R.drawable.ic_check)
|
|
||||||
chip.isCloseIconVisible = onChipCloseClickListener != null
|
chip.isCloseIconVisible = onChipCloseClickListener != null
|
||||||
chip.setOnCloseIconClickListener(chipOnCloseListener)
|
chip.setOnCloseIconClickListener(chipOnCloseListener)
|
||||||
chip.setEnsureMinTouchTargetSize(false)
|
chip.setEnsureMinTouchTargetSize(false) // TODO remove
|
||||||
chip.setOnClickListener(chipOnClickListener)
|
chip.setOnClickListener(chipOnClickListener)
|
||||||
addView(chip)
|
addView(chip)
|
||||||
return chip
|
return chip
|
||||||
|
|||||||
@@ -1,8 +1,8 @@
|
|||||||
package org.koitharu.kotatsu.core.ui.widgets
|
package org.koitharu.kotatsu.core.ui.widgets
|
||||||
|
|
||||||
import android.content.Context
|
import android.content.Context
|
||||||
import android.util.ArrayMap
|
|
||||||
import android.util.AttributeSet
|
import android.util.AttributeSet
|
||||||
|
import androidx.collection.MutableScatterMap
|
||||||
import com.google.android.material.slider.Slider
|
import com.google.android.material.slider.Slider
|
||||||
import kotlin.math.cbrt
|
import kotlin.math.cbrt
|
||||||
import kotlin.math.pow
|
import kotlin.math.pow
|
||||||
@@ -12,7 +12,7 @@ class CubicSlider @JvmOverloads constructor(
|
|||||||
attrs: AttributeSet? = null,
|
attrs: AttributeSet? = null,
|
||||||
) : Slider(context, attrs) {
|
) : Slider(context, attrs) {
|
||||||
|
|
||||||
private val changeListeners = ArrayMap<OnChangeListener, OnChangeListenerMapper>(1)
|
private val changeListeners = MutableScatterMap<OnChangeListener, OnChangeListenerMapper>(1)
|
||||||
|
|
||||||
override fun setValue(value: Float) {
|
override fun setValue(value: Float) {
|
||||||
super.setValue(value.unmap())
|
super.setValue(value.unmap())
|
||||||
|
|||||||
@@ -1,397 +0,0 @@
|
|||||||
package org.koitharu.kotatsu.core.ui.widgets
|
|
||||||
|
|
||||||
import android.animation.ValueAnimator
|
|
||||||
import android.content.Context
|
|
||||||
import android.graphics.Canvas
|
|
||||||
import android.graphics.Color
|
|
||||||
import android.graphics.CornerPathEffect
|
|
||||||
import android.graphics.Paint
|
|
||||||
import android.graphics.Rect
|
|
||||||
import android.graphics.RectF
|
|
||||||
import android.graphics.Typeface
|
|
||||||
import android.os.Build
|
|
||||||
import android.os.Parcelable
|
|
||||||
import android.text.Layout
|
|
||||||
import android.text.StaticLayout
|
|
||||||
import android.text.TextDirectionHeuristic
|
|
||||||
import android.text.TextDirectionHeuristics
|
|
||||||
import android.text.TextPaint
|
|
||||||
import android.util.AttributeSet
|
|
||||||
import android.view.View
|
|
||||||
import androidx.annotation.RequiresApi
|
|
||||||
import androidx.interpolator.view.animation.FastOutSlowInInterpolator
|
|
||||||
import org.koitharu.kotatsu.R
|
|
||||||
import org.koitharu.kotatsu.core.util.ext.draw
|
|
||||||
import org.koitharu.kotatsu.core.util.ext.getAnimationDuration
|
|
||||||
import org.koitharu.kotatsu.core.util.ext.resolveDp
|
|
||||||
import org.koitharu.kotatsu.core.util.ext.resolveSp
|
|
||||||
|
|
||||||
class PieChart @JvmOverloads constructor(
|
|
||||||
context: Context,
|
|
||||||
attrs: AttributeSet? = null,
|
|
||||||
defStyleAttr: Int = 0
|
|
||||||
) : View(context, attrs, defStyleAttr), PieChartInterface {
|
|
||||||
|
|
||||||
private var marginTextFirst: Float = context.resources.resolveDp(DEFAULT_MARGIN_TEXT_1)
|
|
||||||
private var marginTextSecond: Float = context.resources.resolveDp(DEFAULT_MARGIN_TEXT_2)
|
|
||||||
private var marginTextThird: Float = context.resources.resolveDp(DEFAULT_MARGIN_TEXT_3)
|
|
||||||
private var marginSmallCircle: Float = context.resources.resolveDp(DEFAULT_MARGIN_SMALL_CIRCLE)
|
|
||||||
private val marginText: Float = marginTextFirst + marginTextSecond
|
|
||||||
private val circleRect = RectF()
|
|
||||||
private var circleStrokeWidth: Float = context.resources.resolveDp(6f)
|
|
||||||
private var circleRadius: Float = 0f
|
|
||||||
private var circlePadding: Float = context.resources.resolveDp(8f)
|
|
||||||
private var circlePaintRoundSize: Boolean = true
|
|
||||||
private var circleSectionSpace: Float = 3f
|
|
||||||
private var circleCenterX: Float = 0f
|
|
||||||
private var circleCenterY: Float = 0f
|
|
||||||
private var numberTextPaint: TextPaint = TextPaint()
|
|
||||||
private var descriptionTextPain: TextPaint = TextPaint()
|
|
||||||
private var amountTextPaint: TextPaint = TextPaint()
|
|
||||||
private var textStartX: Float = 0f
|
|
||||||
private var textStartY: Float = 0f
|
|
||||||
private var textHeight: Int = 0
|
|
||||||
private var textCircleRadius: Float = context.resources.resolveDp(4f)
|
|
||||||
private var textAmountStr: String = ""
|
|
||||||
private var textAmountY: Float = 0f
|
|
||||||
private var textAmountXNumber: Float = 0f
|
|
||||||
private var textAmountXDescription: Float = 0f
|
|
||||||
private var textAmountYDescription: Float = 0f
|
|
||||||
private var totalAmount: Int = 0
|
|
||||||
private var pieChartColors: List<String> = listOf()
|
|
||||||
private var percentageCircleList: List<PieChartModel> = listOf()
|
|
||||||
private var textRowList: MutableList<StaticLayout> = mutableListOf()
|
|
||||||
private var dataList: List<Pair<Int, String>> = listOf()
|
|
||||||
private var animationSweepAngle: Int = 0
|
|
||||||
|
|
||||||
init {
|
|
||||||
var textAmountSize: Float = context.resources.resolveSp(22f)
|
|
||||||
var textNumberSize: Float = context.resources.resolveSp(20f)
|
|
||||||
var textDescriptionSize: Float = context.resources.resolveSp(14f)
|
|
||||||
var textAmountColor: Int = Color.WHITE
|
|
||||||
var textNumberColor: Int = Color.WHITE
|
|
||||||
var textDescriptionColor: Int = Color.GRAY
|
|
||||||
|
|
||||||
if (attrs != null) {
|
|
||||||
val typeArray = context.obtainStyledAttributes(attrs, R.styleable.PieChart)
|
|
||||||
|
|
||||||
val colorResId = typeArray.getResourceId(R.styleable.PieChart_pieChartColors, 0)
|
|
||||||
pieChartColors = typeArray.resources.getStringArray(colorResId).toList()
|
|
||||||
|
|
||||||
marginTextFirst = typeArray.getDimension(R.styleable.PieChart_pieChartMarginTextFirst, marginTextFirst)
|
|
||||||
marginTextSecond = typeArray.getDimension(R.styleable.PieChart_pieChartMarginTextSecond, marginTextSecond)
|
|
||||||
marginTextThird = typeArray.getDimension(R.styleable.PieChart_pieChartMarginTextThird, marginTextThird)
|
|
||||||
marginSmallCircle =
|
|
||||||
typeArray.getDimension(R.styleable.PieChart_pieChartMarginSmallCircle, marginSmallCircle)
|
|
||||||
|
|
||||||
circleStrokeWidth =
|
|
||||||
typeArray.getDimension(R.styleable.PieChart_pieChartCircleStrokeWidth, circleStrokeWidth)
|
|
||||||
circlePadding = typeArray.getDimension(R.styleable.PieChart_pieChartCirclePadding, circlePadding)
|
|
||||||
circlePaintRoundSize =
|
|
||||||
typeArray.getBoolean(R.styleable.PieChart_pieChartCirclePaintRoundSize, circlePaintRoundSize)
|
|
||||||
circleSectionSpace = typeArray.getFloat(R.styleable.PieChart_pieChartCircleSectionSpace, circleSectionSpace)
|
|
||||||
|
|
||||||
textCircleRadius = typeArray.getDimension(R.styleable.PieChart_pieChartTextCircleRadius, textCircleRadius)
|
|
||||||
textAmountSize = typeArray.getDimension(R.styleable.PieChart_pieChartTextAmountSize, textAmountSize)
|
|
||||||
textNumberSize = typeArray.getDimension(R.styleable.PieChart_pieChartTextNumberSize, textNumberSize)
|
|
||||||
textDescriptionSize =
|
|
||||||
typeArray.getDimension(R.styleable.PieChart_pieChartTextDescriptionSize, textDescriptionSize)
|
|
||||||
textAmountColor = typeArray.getColor(R.styleable.PieChart_pieChartTextAmountColor, textAmountColor)
|
|
||||||
textNumberColor = typeArray.getColor(R.styleable.PieChart_pieChartTextNumberColor, textNumberColor)
|
|
||||||
textDescriptionColor =
|
|
||||||
typeArray.getColor(R.styleable.PieChart_pieChartTextDescriptionColor, textDescriptionColor)
|
|
||||||
textAmountStr = typeArray.getString(R.styleable.PieChart_pieChartTextAmount) ?: ""
|
|
||||||
|
|
||||||
typeArray.recycle()
|
|
||||||
}
|
|
||||||
|
|
||||||
circlePadding += circleStrokeWidth
|
|
||||||
|
|
||||||
// Инициализация кистей View
|
|
||||||
initPaints(amountTextPaint, textAmountSize, textAmountColor)
|
|
||||||
initPaints(numberTextPaint, textNumberSize, textNumberColor)
|
|
||||||
initPaints(descriptionTextPain, textDescriptionSize, textDescriptionColor, true)
|
|
||||||
}
|
|
||||||
|
|
||||||
@RequiresApi(Build.VERSION_CODES.M)
|
|
||||||
override fun onMeasure(widthMeasureSpec: Int, heightMeasureSpec: Int) {
|
|
||||||
textRowList.clear()
|
|
||||||
|
|
||||||
val initSizeWidth = resolveDefaultSize(widthMeasureSpec, DEFAULT_VIEW_SIZE_WIDTH)
|
|
||||||
|
|
||||||
val textTextWidth = (initSizeWidth * TEXT_WIDTH_PERCENT)
|
|
||||||
val initSizeHeight = calculateViewHeight(heightMeasureSpec, textTextWidth.toInt())
|
|
||||||
|
|
||||||
textStartX = initSizeWidth - textTextWidth.toFloat()
|
|
||||||
textStartY = initSizeHeight.toFloat() / 2 - textHeight / 2
|
|
||||||
|
|
||||||
calculateCircleRadius(initSizeWidth, initSizeHeight)
|
|
||||||
|
|
||||||
setMeasuredDimension(initSizeWidth, initSizeHeight)
|
|
||||||
}
|
|
||||||
|
|
||||||
override fun onDraw(canvas: Canvas) {
|
|
||||||
super.onDraw(canvas)
|
|
||||||
|
|
||||||
drawCircle(canvas)
|
|
||||||
drawText(canvas)
|
|
||||||
}
|
|
||||||
|
|
||||||
override fun onRestoreInstanceState(state: Parcelable?) {
|
|
||||||
val pieChartState = state as? PieChartState
|
|
||||||
super.onRestoreInstanceState(pieChartState?.superState ?: state)
|
|
||||||
|
|
||||||
dataList = pieChartState?.dataList ?: listOf()
|
|
||||||
}
|
|
||||||
|
|
||||||
override fun onSaveInstanceState(): Parcelable {
|
|
||||||
val superState = super.onSaveInstanceState()
|
|
||||||
return PieChartState(superState, dataList)
|
|
||||||
}
|
|
||||||
|
|
||||||
override fun setDataChart(list: List<Pair<Int, String>>) {
|
|
||||||
dataList = list
|
|
||||||
calculatePercentageOfData()
|
|
||||||
}
|
|
||||||
|
|
||||||
override fun startAnimation() {
|
|
||||||
val animator = ValueAnimator.ofInt(0, 360).apply {
|
|
||||||
duration = context.getAnimationDuration(android.R.integer.config_longAnimTime)
|
|
||||||
interpolator = FastOutSlowInInterpolator()
|
|
||||||
addUpdateListener { valueAnimator ->
|
|
||||||
animationSweepAngle = valueAnimator.animatedValue as Int
|
|
||||||
invalidate()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
animator.start()
|
|
||||||
}
|
|
||||||
|
|
||||||
private fun drawCircle(canvas: Canvas) {
|
|
||||||
for (percent in percentageCircleList) {
|
|
||||||
if (animationSweepAngle > percent.percentToStartAt + percent.percentOfCircle) {
|
|
||||||
canvas.drawArc(circleRect, percent.percentToStartAt, percent.percentOfCircle, false, percent.paint)
|
|
||||||
} else if (animationSweepAngle > percent.percentToStartAt) {
|
|
||||||
canvas.drawArc(
|
|
||||||
circleRect,
|
|
||||||
percent.percentToStartAt,
|
|
||||||
animationSweepAngle - percent.percentToStartAt,
|
|
||||||
false,
|
|
||||||
percent.paint,
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private fun drawText(canvas: Canvas) {
|
|
||||||
var textBuffY = textStartY
|
|
||||||
textRowList.forEachIndexed { index, staticLayout ->
|
|
||||||
if (index % 2 == 0) {
|
|
||||||
staticLayout.draw(canvas, textStartX + marginSmallCircle + textCircleRadius, textBuffY)
|
|
||||||
canvas.drawCircle(
|
|
||||||
textStartX + marginSmallCircle / 2,
|
|
||||||
textBuffY + staticLayout.height / 2 + textCircleRadius / 2,
|
|
||||||
textCircleRadius,
|
|
||||||
Paint().apply { color = Color.parseColor(pieChartColors[(index / 2) % pieChartColors.size]) },
|
|
||||||
)
|
|
||||||
textBuffY += staticLayout.height + marginTextFirst
|
|
||||||
} else {
|
|
||||||
staticLayout.draw(canvas, textStartX, textBuffY)
|
|
||||||
textBuffY += staticLayout.height + marginTextSecond
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
canvas.drawText(totalAmount.toString(), textAmountXNumber, textAmountY, amountTextPaint)
|
|
||||||
canvas.drawText(textAmountStr, textAmountXDescription, textAmountYDescription, descriptionTextPain)
|
|
||||||
}
|
|
||||||
|
|
||||||
private fun initPaints(textPaint: TextPaint, textSize: Float, textColor: Int, isDescription: Boolean = false) {
|
|
||||||
textPaint.color = textColor
|
|
||||||
textPaint.textSize = textSize
|
|
||||||
textPaint.isAntiAlias = true
|
|
||||||
|
|
||||||
if (!isDescription) textPaint.typeface = Typeface.create(Typeface.DEFAULT, Typeface.BOLD)
|
|
||||||
}
|
|
||||||
|
|
||||||
private fun resolveDefaultSize(spec: Int, defValue: Int): Int {
|
|
||||||
return when (MeasureSpec.getMode(spec)) {
|
|
||||||
MeasureSpec.UNSPECIFIED -> resources.resolveDp(defValue)
|
|
||||||
else -> MeasureSpec.getSize(spec)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@RequiresApi(Build.VERSION_CODES.M)
|
|
||||||
private fun calculateViewHeight(heightMeasureSpec: Int, textWidth: Int): Int {
|
|
||||||
val initSizeHeight = resolveDefaultSize(heightMeasureSpec, DEFAULT_VIEW_SIZE_HEIGHT)
|
|
||||||
textHeight = (dataList.size * marginText + getTextViewHeight(textWidth)).toInt()
|
|
||||||
|
|
||||||
val textHeightWithPadding = textHeight + paddingTop + paddingBottom
|
|
||||||
return if (textHeightWithPadding > initSizeHeight) textHeightWithPadding else initSizeHeight
|
|
||||||
}
|
|
||||||
|
|
||||||
private fun calculateCircleRadius(width: Int, height: Int) {
|
|
||||||
val circleViewWidth = (width * CIRCLE_WIDTH_PERCENT)
|
|
||||||
circleRadius = if (circleViewWidth > height) {
|
|
||||||
(height.toFloat() - circlePadding) / 2
|
|
||||||
} else {
|
|
||||||
circleViewWidth.toFloat() / 2
|
|
||||||
}
|
|
||||||
|
|
||||||
with(circleRect) {
|
|
||||||
left = circlePadding
|
|
||||||
top = height / 2 - circleRadius
|
|
||||||
right = circleRadius * 2 + circlePadding
|
|
||||||
bottom = height / 2 + circleRadius
|
|
||||||
}
|
|
||||||
|
|
||||||
circleCenterX = (circleRadius * 2 + circlePadding + circlePadding) / 2
|
|
||||||
circleCenterY = (height / 2 + circleRadius + (height / 2 - circleRadius)) / 2
|
|
||||||
|
|
||||||
textAmountY = circleCenterY
|
|
||||||
|
|
||||||
val sizeTextAmountNumber = getWidthOfAmountText(
|
|
||||||
totalAmount.toString(),
|
|
||||||
amountTextPaint,
|
|
||||||
)
|
|
||||||
|
|
||||||
textAmountXNumber = circleCenterX - sizeTextAmountNumber.width() / 2
|
|
||||||
textAmountXDescription = circleCenterX - getWidthOfAmountText(textAmountStr, descriptionTextPain).width() / 2
|
|
||||||
textAmountYDescription = circleCenterY + sizeTextAmountNumber.height() + marginTextThird
|
|
||||||
}
|
|
||||||
|
|
||||||
@RequiresApi(Build.VERSION_CODES.M)
|
|
||||||
private fun getTextViewHeight(maxWidth: Int): Int {
|
|
||||||
var textHeight = 0
|
|
||||||
dataList.forEach {
|
|
||||||
val textLayoutNumber = getMultilineText(
|
|
||||||
text = it.first.toString(),
|
|
||||||
textPaint = numberTextPaint,
|
|
||||||
width = maxWidth,
|
|
||||||
)
|
|
||||||
val textLayoutDescription = getMultilineText(
|
|
||||||
text = it.second,
|
|
||||||
textPaint = descriptionTextPain,
|
|
||||||
width = maxWidth,
|
|
||||||
)
|
|
||||||
textRowList.apply {
|
|
||||||
add(textLayoutNumber)
|
|
||||||
add(textLayoutDescription)
|
|
||||||
}
|
|
||||||
textHeight += textLayoutNumber.height + textLayoutDescription.height
|
|
||||||
}
|
|
||||||
|
|
||||||
return textHeight
|
|
||||||
}
|
|
||||||
|
|
||||||
private fun calculatePercentageOfData() {
|
|
||||||
totalAmount = dataList.fold(0) { res, value -> res + value.first }
|
|
||||||
|
|
||||||
var startAt = circleSectionSpace
|
|
||||||
percentageCircleList = dataList.mapIndexed { index, pair ->
|
|
||||||
var percent = pair.first * 100 / totalAmount.toFloat() - circleSectionSpace
|
|
||||||
percent = if (percent < 0f) 0f else percent
|
|
||||||
|
|
||||||
val resultModel = PieChartModel(
|
|
||||||
percentOfCircle = percent,
|
|
||||||
percentToStartAt = startAt,
|
|
||||||
colorOfLine = Color.parseColor(pieChartColors[index % pieChartColors.size]),
|
|
||||||
stroke = circleStrokeWidth,
|
|
||||||
paintRound = circlePaintRoundSize,
|
|
||||||
)
|
|
||||||
if (percent != 0f) startAt += percent + circleSectionSpace
|
|
||||||
resultModel
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private fun getWidthOfAmountText(text: String, textPaint: TextPaint): Rect {
|
|
||||||
val bounds = Rect()
|
|
||||||
textPaint.getTextBounds(text, 0, text.length, bounds)
|
|
||||||
return bounds
|
|
||||||
}
|
|
||||||
|
|
||||||
@RequiresApi(Build.VERSION_CODES.M)
|
|
||||||
private fun getMultilineText(
|
|
||||||
text: CharSequence,
|
|
||||||
textPaint: TextPaint,
|
|
||||||
width: Int,
|
|
||||||
start: Int = 0,
|
|
||||||
end: Int = text.length,
|
|
||||||
alignment: Layout.Alignment = Layout.Alignment.ALIGN_NORMAL,
|
|
||||||
textDir: TextDirectionHeuristic = TextDirectionHeuristics.LTR,
|
|
||||||
spacingMult: Float = 1f,
|
|
||||||
spacingAdd: Float = 0f
|
|
||||||
): StaticLayout {
|
|
||||||
|
|
||||||
return StaticLayout.Builder
|
|
||||||
.obtain(text, start, end, textPaint, width)
|
|
||||||
.setAlignment(alignment)
|
|
||||||
.setTextDirection(textDir)
|
|
||||||
.setLineSpacing(spacingAdd, spacingMult)
|
|
||||||
.build()
|
|
||||||
}
|
|
||||||
|
|
||||||
companion object {
|
|
||||||
private const val DEFAULT_MARGIN_TEXT_1 = 2f
|
|
||||||
private const val DEFAULT_MARGIN_TEXT_2 = 10f
|
|
||||||
private const val DEFAULT_MARGIN_TEXT_3 = 2f
|
|
||||||
private const val DEFAULT_MARGIN_SMALL_CIRCLE = 12f
|
|
||||||
|
|
||||||
private const val TEXT_WIDTH_PERCENT = 0.40
|
|
||||||
private const val CIRCLE_WIDTH_PERCENT = 0.50
|
|
||||||
|
|
||||||
const val DEFAULT_VIEW_SIZE_HEIGHT = 150
|
|
||||||
const val DEFAULT_VIEW_SIZE_WIDTH = 250
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
interface PieChartInterface {
|
|
||||||
|
|
||||||
fun setDataChart(list: List<Pair<Int, String>>)
|
|
||||||
|
|
||||||
fun startAnimation()
|
|
||||||
}
|
|
||||||
|
|
||||||
data class PieChartModel(
|
|
||||||
var percentOfCircle: Float = 0f,
|
|
||||||
var percentToStartAt: Float = 0f,
|
|
||||||
var colorOfLine: Int = 0,
|
|
||||||
var stroke: Float = 0f,
|
|
||||||
var paint: Paint = Paint(),
|
|
||||||
var paintRound: Boolean = true
|
|
||||||
) {
|
|
||||||
|
|
||||||
init {
|
|
||||||
if (percentOfCircle < 0 || percentOfCircle > 100) {
|
|
||||||
percentOfCircle = 100f
|
|
||||||
}
|
|
||||||
|
|
||||||
percentOfCircle = 360 * percentOfCircle / 100
|
|
||||||
|
|
||||||
if (percentToStartAt < 0 || percentToStartAt > 100) {
|
|
||||||
percentToStartAt = 0f
|
|
||||||
}
|
|
||||||
|
|
||||||
percentToStartAt = 360 * percentToStartAt / 100
|
|
||||||
|
|
||||||
if (colorOfLine == 0) {
|
|
||||||
colorOfLine = Color.parseColor("#000000")
|
|
||||||
}
|
|
||||||
|
|
||||||
paint = Paint()
|
|
||||||
paint.color = colorOfLine
|
|
||||||
paint.isAntiAlias = true
|
|
||||||
paint.style = Paint.Style.STROKE
|
|
||||||
paint.strokeWidth = stroke
|
|
||||||
paint.isDither = true
|
|
||||||
|
|
||||||
if (paintRound) {
|
|
||||||
paint.strokeJoin = Paint.Join.ROUND
|
|
||||||
paint.strokeCap = Paint.Cap.ROUND
|
|
||||||
paint.pathEffect = CornerPathEffect(8f)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
class PieChartState(
|
|
||||||
superSavedState: Parcelable?,
|
|
||||||
val dataList: List<Pair<Int, String>>
|
|
||||||
) : View.BaseSavedState(superSavedState), Parcelable
|
|
||||||
@@ -11,6 +11,7 @@ import android.view.View
|
|||||||
import android.view.ViewOutlineProvider
|
import android.view.ViewOutlineProvider
|
||||||
import androidx.annotation.ColorInt
|
import androidx.annotation.ColorInt
|
||||||
import androidx.annotation.FloatRange
|
import androidx.annotation.FloatRange
|
||||||
|
import androidx.collection.MutableFloatList
|
||||||
import androidx.interpolator.view.animation.FastOutSlowInInterpolator
|
import androidx.interpolator.view.animation.FastOutSlowInInterpolator
|
||||||
import org.koitharu.kotatsu.core.util.ext.getAnimationDuration
|
import org.koitharu.kotatsu.core.util.ext.getAnimationDuration
|
||||||
import org.koitharu.kotatsu.core.util.ext.isAnimationsEnabled
|
import org.koitharu.kotatsu.core.util.ext.isAnimationsEnabled
|
||||||
@@ -25,7 +26,7 @@ class SegmentedBarView @JvmOverloads constructor(
|
|||||||
|
|
||||||
private val paint = Paint(Paint.ANTI_ALIAS_FLAG)
|
private val paint = Paint(Paint.ANTI_ALIAS_FLAG)
|
||||||
private val segmentsData = ArrayList<Segment>()
|
private val segmentsData = ArrayList<Segment>()
|
||||||
private val segmentsSizes = ArrayList<Float>()
|
private val segmentsSizes = MutableFloatList()
|
||||||
private var cornerSize = 0f
|
private var cornerSize = 0f
|
||||||
private var scaleFactor = 1f
|
private var scaleFactor = 1f
|
||||||
private var scaleAnimator: ValueAnimator? = null
|
private var scaleAnimator: ValueAnimator? = null
|
||||||
|
|||||||
@@ -9,10 +9,8 @@ import androidx.fragment.app.FragmentManager
|
|||||||
import androidx.fragment.app.FragmentManager.FragmentLifecycleCallbacks
|
import androidx.fragment.app.FragmentManager.FragmentLifecycleCallbacks
|
||||||
import org.acra.ACRA
|
import org.acra.ACRA
|
||||||
import org.koitharu.kotatsu.core.ui.DefaultActivityLifecycleCallbacks
|
import org.koitharu.kotatsu.core.ui.DefaultActivityLifecycleCallbacks
|
||||||
import java.text.DateFormat
|
import java.time.LocalTime
|
||||||
import java.text.SimpleDateFormat
|
import java.time.temporal.ChronoUnit
|
||||||
import java.util.Date
|
|
||||||
import java.util.Locale
|
|
||||||
import java.util.WeakHashMap
|
import java.util.WeakHashMap
|
||||||
import javax.inject.Inject
|
import javax.inject.Inject
|
||||||
import javax.inject.Singleton
|
import javax.inject.Singleton
|
||||||
@@ -20,7 +18,6 @@ import javax.inject.Singleton
|
|||||||
@Singleton
|
@Singleton
|
||||||
class AcraScreenLogger @Inject constructor() : FragmentLifecycleCallbacks(), DefaultActivityLifecycleCallbacks {
|
class AcraScreenLogger @Inject constructor() : FragmentLifecycleCallbacks(), DefaultActivityLifecycleCallbacks {
|
||||||
|
|
||||||
private val timeFormat = SimpleDateFormat.getTimeInstance(DateFormat.DEFAULT, Locale.ROOT)
|
|
||||||
private val keys = WeakHashMap<Any, String>()
|
private val keys = WeakHashMap<Any, String>()
|
||||||
|
|
||||||
override fun onFragmentAttached(fm: FragmentManager, f: Fragment, context: Context) {
|
override fun onFragmentAttached(fm: FragmentManager, f: Fragment, context: Context) {
|
||||||
@@ -47,11 +44,10 @@ class AcraScreenLogger @Inject constructor() : FragmentLifecycleCallbacks(), Def
|
|||||||
}
|
}
|
||||||
|
|
||||||
private fun Any.key() = keys.getOrPut(this) {
|
private fun Any.key() = keys.getOrPut(this) {
|
||||||
"${time()}: ${javaClass.simpleName}"
|
val time = LocalTime.now().truncatedTo(ChronoUnit.SECONDS)
|
||||||
|
"$time: ${javaClass.simpleName}"
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun time() = timeFormat.format(Date())
|
|
||||||
|
|
||||||
@Suppress("DEPRECATION")
|
@Suppress("DEPRECATION")
|
||||||
private fun Bundle?.contentToString() = this?.keySet()?.joinToString { k ->
|
private fun Bundle?.contentToString() = this?.keySet()?.joinToString { k ->
|
||||||
val v = get(k)
|
val v = get(k)
|
||||||
|
|||||||
@@ -1,70 +0,0 @@
|
|||||||
package org.koitharu.kotatsu.core.util
|
|
||||||
|
|
||||||
import android.content.Context
|
|
||||||
import android.view.GestureDetector
|
|
||||||
import android.view.MotionEvent
|
|
||||||
import kotlin.math.roundToInt
|
|
||||||
|
|
||||||
class GridTouchHelper(
|
|
||||||
context: Context,
|
|
||||||
private val listener: OnGridTouchListener,
|
|
||||||
) : GestureDetector.SimpleOnGestureListener() {
|
|
||||||
|
|
||||||
private val detector = GestureDetector(context, this)
|
|
||||||
private val width = context.resources.displayMetrics.widthPixels
|
|
||||||
private val height = context.resources.displayMetrics.heightPixels
|
|
||||||
private var isDispatching = false
|
|
||||||
|
|
||||||
init {
|
|
||||||
detector.setIsLongpressEnabled(true)
|
|
||||||
detector.setOnDoubleTapListener(this)
|
|
||||||
}
|
|
||||||
|
|
||||||
fun dispatchTouchEvent(event: MotionEvent) {
|
|
||||||
if (event.actionMasked == MotionEvent.ACTION_DOWN) {
|
|
||||||
isDispatching = listener.onProcessTouch(event.rawX.toInt(), event.rawY.toInt())
|
|
||||||
}
|
|
||||||
detector.onTouchEvent(event)
|
|
||||||
}
|
|
||||||
|
|
||||||
override fun onSingleTapConfirmed(event: MotionEvent): Boolean {
|
|
||||||
if (!isDispatching) {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
val xIndex = (event.rawX * 2f / width).roundToInt()
|
|
||||||
val yIndex = (event.rawY * 2f / height).roundToInt()
|
|
||||||
listener.onGridTouch(
|
|
||||||
when (xIndex) {
|
|
||||||
0 -> AREA_LEFT
|
|
||||||
1 -> {
|
|
||||||
when (yIndex) {
|
|
||||||
0 -> AREA_TOP
|
|
||||||
1 -> AREA_CENTER
|
|
||||||
2 -> AREA_BOTTOM
|
|
||||||
else -> return false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
2 -> AREA_RIGHT
|
|
||||||
else -> return false
|
|
||||||
},
|
|
||||||
)
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
|
|
||||||
companion object {
|
|
||||||
|
|
||||||
const val AREA_CENTER = 1
|
|
||||||
const val AREA_LEFT = 2
|
|
||||||
const val AREA_RIGHT = 3
|
|
||||||
const val AREA_TOP = 4
|
|
||||||
const val AREA_BOTTOM = 5
|
|
||||||
}
|
|
||||||
|
|
||||||
interface OnGridTouchListener {
|
|
||||||
|
|
||||||
fun onGridTouch(area: Int)
|
|
||||||
|
|
||||||
fun onProcessTouch(rawX: Int, rawY: Int): Boolean
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -0,0 +1,55 @@
|
|||||||
|
package org.koitharu.kotatsu.core.util
|
||||||
|
|
||||||
|
import android.content.Context
|
||||||
|
import androidx.annotation.AttrRes
|
||||||
|
import androidx.annotation.ColorInt
|
||||||
|
import androidx.core.graphics.ColorUtils
|
||||||
|
import com.google.android.material.R
|
||||||
|
import com.google.android.material.color.MaterialColors
|
||||||
|
import org.koitharu.kotatsu.core.util.ext.getThemeColor
|
||||||
|
import org.koitharu.kotatsu.parsers.model.Manga
|
||||||
|
import kotlin.math.absoluteValue
|
||||||
|
|
||||||
|
object KotatsuColors {
|
||||||
|
|
||||||
|
@ColorInt
|
||||||
|
fun segmentColor(context: Context, @AttrRes resId: Int): Int {
|
||||||
|
val colorHex = String.format("%06x", context.getThemeColor(resId))
|
||||||
|
val hue = getHue(colorHex)
|
||||||
|
val color = ColorUtils.HSLToColor(floatArrayOf(hue, 0.5f, 0.5f))
|
||||||
|
val backgroundColor = context.getThemeColor(R.attr.colorSurfaceContainerHigh)
|
||||||
|
return MaterialColors.harmonize(color, backgroundColor)
|
||||||
|
}
|
||||||
|
|
||||||
|
@ColorInt
|
||||||
|
fun random(seed: Any): Int {
|
||||||
|
val hue = (seed.hashCode() % 360).absoluteValue.toFloat()
|
||||||
|
return ColorUtils.HSLToColor(floatArrayOf(hue, 0.5f, 0.5f))
|
||||||
|
}
|
||||||
|
|
||||||
|
@ColorInt
|
||||||
|
fun ofManga(context: Context, manga: Manga?): Int {
|
||||||
|
val color = if (manga != null) {
|
||||||
|
val hue = (manga.id.absoluteValue % 360).toFloat()
|
||||||
|
ColorUtils.HSLToColor(floatArrayOf(hue, 0.5f, 0.5f))
|
||||||
|
} else {
|
||||||
|
context.getThemeColor(R.attr.colorSurface)
|
||||||
|
}
|
||||||
|
val backgroundColor = context.getThemeColor(R.attr.colorSurfaceContainerHigh)
|
||||||
|
return MaterialColors.harmonize(color, backgroundColor)
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun getHue(hex: String): Float {
|
||||||
|
val r = (hex.substring(0, 2).toInt(16)).toFloat()
|
||||||
|
val g = (hex.substring(2, 4).toInt(16)).toFloat()
|
||||||
|
val b = (hex.substring(4, 6).toInt(16)).toFloat()
|
||||||
|
|
||||||
|
var hue = 0F
|
||||||
|
if ((r >= g) && (g >= b)) {
|
||||||
|
hue = 60 * (g - b) / (r - b)
|
||||||
|
} else if ((g > r) && (r >= b)) {
|
||||||
|
hue = 60 * (2 - (r - b) / (g - b))
|
||||||
|
}
|
||||||
|
return hue
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -17,6 +17,7 @@ import android.content.SyncResult
|
|||||||
import android.content.pm.PackageManager
|
import android.content.pm.PackageManager
|
||||||
import android.content.pm.ResolveInfo
|
import android.content.pm.ResolveInfo
|
||||||
import android.database.SQLException
|
import android.database.SQLException
|
||||||
|
import android.graphics.Bitmap
|
||||||
import android.graphics.Color
|
import android.graphics.Color
|
||||||
import android.net.Uri
|
import android.net.Uri
|
||||||
import android.os.Build
|
import android.os.Build
|
||||||
@@ -26,9 +27,9 @@ import android.provider.Settings
|
|||||||
import android.view.View
|
import android.view.View
|
||||||
import android.view.ViewPropertyAnimator
|
import android.view.ViewPropertyAnimator
|
||||||
import android.view.Window
|
import android.view.Window
|
||||||
import android.widget.Toast
|
|
||||||
import androidx.activity.result.ActivityResultLauncher
|
import androidx.activity.result.ActivityResultLauncher
|
||||||
import androidx.annotation.IntegerRes
|
import androidx.annotation.IntegerRes
|
||||||
|
import androidx.annotation.WorkerThread
|
||||||
import androidx.core.app.ActivityOptionsCompat
|
import androidx.core.app.ActivityOptionsCompat
|
||||||
import androidx.core.app.NotificationManagerCompat
|
import androidx.core.app.NotificationManagerCompat
|
||||||
import androidx.core.content.ContextCompat
|
import androidx.core.content.ContextCompat
|
||||||
@@ -37,6 +38,7 @@ import androidx.lifecycle.Lifecycle
|
|||||||
import androidx.lifecycle.coroutineScope
|
import androidx.lifecycle.coroutineScope
|
||||||
import androidx.work.CoroutineWorker
|
import androidx.work.CoroutineWorker
|
||||||
import com.google.android.material.elevation.ElevationOverlayProvider
|
import com.google.android.material.elevation.ElevationOverlayProvider
|
||||||
|
import kotlinx.coroutines.Dispatchers
|
||||||
import kotlinx.coroutines.channels.awaitClose
|
import kotlinx.coroutines.channels.awaitClose
|
||||||
import kotlinx.coroutines.channels.trySendBlocking
|
import kotlinx.coroutines.channels.trySendBlocking
|
||||||
import kotlinx.coroutines.delay
|
import kotlinx.coroutines.delay
|
||||||
@@ -45,7 +47,9 @@ import kotlinx.coroutines.flow.callbackFlow
|
|||||||
import kotlinx.coroutines.flow.distinctUntilChanged
|
import kotlinx.coroutines.flow.distinctUntilChanged
|
||||||
import kotlinx.coroutines.flow.flow
|
import kotlinx.coroutines.flow.flow
|
||||||
import kotlinx.coroutines.launch
|
import kotlinx.coroutines.launch
|
||||||
|
import kotlinx.coroutines.runInterruptible
|
||||||
import okio.IOException
|
import okio.IOException
|
||||||
|
import okio.use
|
||||||
import org.json.JSONException
|
import org.json.JSONException
|
||||||
import org.jsoup.internal.StringUtil.StringJoiner
|
import org.jsoup.internal.StringUtil.StringJoiner
|
||||||
import org.koitharu.kotatsu.BuildConfig
|
import org.koitharu.kotatsu.BuildConfig
|
||||||
@@ -53,6 +57,7 @@ import org.koitharu.kotatsu.R
|
|||||||
import org.koitharu.kotatsu.parsers.util.runCatchingCancellable
|
import org.koitharu.kotatsu.parsers.util.runCatchingCancellable
|
||||||
import org.xmlpull.v1.XmlPullParser
|
import org.xmlpull.v1.XmlPullParser
|
||||||
import org.xmlpull.v1.XmlPullParserException
|
import org.xmlpull.v1.XmlPullParserException
|
||||||
|
import java.io.File
|
||||||
import kotlin.math.roundToLong
|
import kotlin.math.roundToLong
|
||||||
|
|
||||||
val Context.activityManager: ActivityManager?
|
val Context.activityManager: ActivityManager?
|
||||||
@@ -210,23 +215,23 @@ fun Context.findActivity(): Activity? = when (this) {
|
|||||||
else -> null
|
else -> null
|
||||||
}
|
}
|
||||||
|
|
||||||
inline fun Activity.catchingWebViewUnavailability(block: () -> Unit): Boolean {
|
|
||||||
return try {
|
|
||||||
block()
|
|
||||||
true
|
|
||||||
} catch (e: Exception) {
|
|
||||||
if (e.isWebViewUnavailable()) {
|
|
||||||
Toast.makeText(this, R.string.web_view_unavailable, Toast.LENGTH_LONG).show()
|
|
||||||
finishAfterTransition()
|
|
||||||
false
|
|
||||||
} else {
|
|
||||||
throw e
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fun Context.checkNotificationPermission(): Boolean = if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.TIRAMISU) {
|
fun Context.checkNotificationPermission(): Boolean = if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.TIRAMISU) {
|
||||||
ContextCompat.checkSelfPermission(this, Manifest.permission.POST_NOTIFICATIONS) == PackageManager.PERMISSION_GRANTED
|
ContextCompat.checkSelfPermission(this, Manifest.permission.POST_NOTIFICATIONS) == PackageManager.PERMISSION_GRANTED
|
||||||
} else {
|
} else {
|
||||||
NotificationManagerCompat.from(this).areNotificationsEnabled()
|
NotificationManagerCompat.from(this).areNotificationsEnabled()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@WorkerThread
|
||||||
|
suspend fun Bitmap.compressToPNG(output: File) = runInterruptible(Dispatchers.IO) {
|
||||||
|
output.outputStream().use { os ->
|
||||||
|
if (!compress(Bitmap.CompressFormat.PNG, 100, os)) {
|
||||||
|
throw IOException("Failed to encode bitmap into PNG format")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fun Context.ensureRamAtLeast(requiredSize: Long) {
|
||||||
|
if (ramAvailable < requiredSize) {
|
||||||
|
throw IllegalStateException("Not enough free memory")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -24,11 +24,15 @@ inline fun <reified T : Parcelable> Intent.getParcelableExtraCompat(key: String)
|
|||||||
}
|
}
|
||||||
|
|
||||||
inline fun <reified T : Serializable> Intent.getSerializableExtraCompat(key: String): T? {
|
inline fun <reified T : Serializable> Intent.getSerializableExtraCompat(key: String): T? {
|
||||||
return getSerializableExtra(key) as T?
|
return if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.UPSIDE_DOWN_CAKE) {
|
||||||
|
getSerializableExtra(key, T::class.java)
|
||||||
|
} else {
|
||||||
|
getSerializableExtra(key) as T?
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
inline fun <reified T : Serializable> Bundle.getSerializableCompat(key: String): T? {
|
inline fun <reified T : Serializable> Bundle.getSerializableCompat(key: String): T? {
|
||||||
return if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.TIRAMISU) {
|
return if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.UPSIDE_DOWN_CAKE) {
|
||||||
getSerializable(key, T::class.java)
|
getSerializable(key, T::class.java)
|
||||||
} else {
|
} else {
|
||||||
getSerializable(key) as T?
|
getSerializable(key) as T?
|
||||||
|
|||||||
@@ -25,7 +25,7 @@ fun ImageView.newImageRequest(lifecycleOwner: LifecycleOwner, data: Any?): Image
|
|||||||
}
|
}
|
||||||
// disposeImageRequest()
|
// disposeImageRequest()
|
||||||
return ImageRequest.Builder(context)
|
return ImageRequest.Builder(context)
|
||||||
.data(data)
|
.data(data?.takeUnless { it == "" })
|
||||||
.lifecycle(lifecycleOwner)
|
.lifecycle(lifecycleOwner)
|
||||||
.crossfade(context)
|
.crossfade(context)
|
||||||
.target(this)
|
.target(this)
|
||||||
|
|||||||
@@ -1,30 +1,32 @@
|
|||||||
package org.koitharu.kotatsu.core.util.ext
|
package org.koitharu.kotatsu.core.util.ext
|
||||||
|
|
||||||
import android.annotation.SuppressLint
|
import org.koitharu.kotatsu.core.ui.model.DateTimeAgo
|
||||||
import android.text.format.DateUtils
|
import java.time.Instant
|
||||||
import java.text.SimpleDateFormat
|
import java.time.LocalDate
|
||||||
import java.util.*
|
import java.time.LocalDateTime
|
||||||
import java.util.concurrent.TimeUnit
|
import java.time.ZoneId
|
||||||
|
import java.time.temporal.ChronoUnit
|
||||||
|
|
||||||
@SuppressLint("SimpleDateFormat")
|
fun calculateTimeAgo(instant: Instant, showMonths: Boolean = false): DateTimeAgo {
|
||||||
fun Date.format(pattern: String): String = SimpleDateFormat(pattern).format(this)
|
// TODO: Use Java 9's LocalDate.ofInstant().
|
||||||
|
val localDate = LocalDateTime.ofInstant(instant, ZoneId.systemDefault()).toLocalDate()
|
||||||
|
val now = LocalDate.now()
|
||||||
|
val diffDays = localDate.until(now, ChronoUnit.DAYS)
|
||||||
|
|
||||||
fun Date.formatRelative(minResolution: Long): CharSequence = DateUtils.getRelativeTimeSpanString(
|
return when {
|
||||||
time, System.currentTimeMillis(), minResolution,
|
diffDays == 0L -> {
|
||||||
)
|
if (instant.until(Instant.now(), ChronoUnit.MINUTES) < 3) DateTimeAgo.JustNow
|
||||||
|
else DateTimeAgo.Today
|
||||||
fun Date.daysDiff(other: Long): Int {
|
}
|
||||||
val thisDay = time / TimeUnit.DAYS.toMillis(1L)
|
diffDays == 1L -> DateTimeAgo.Yesterday
|
||||||
val otherDay = other / TimeUnit.DAYS.toMillis(1L)
|
diffDays < 6 -> DateTimeAgo.DaysAgo(diffDays.toInt())
|
||||||
return (thisDay - otherDay).toInt()
|
else -> {
|
||||||
}
|
val diffMonths = localDate.until(now, ChronoUnit.MONTHS)
|
||||||
|
if (showMonths && diffMonths <= 6) {
|
||||||
fun Date.startOfDay(): Long {
|
DateTimeAgo.MonthsAgo(diffMonths.toInt())
|
||||||
val calendar = Calendar.getInstance()
|
} else {
|
||||||
calendar.time = this
|
DateTimeAgo.Absolute(localDate)
|
||||||
calendar[Calendar.HOUR_OF_DAY] = 0
|
}
|
||||||
calendar[Calendar.MINUTE] = 0
|
}
|
||||||
calendar[Calendar.SECOND] = 0
|
}
|
||||||
calendar[Calendar.MILLISECOND] = 0
|
|
||||||
return calendar.timeInMillis
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -19,6 +19,7 @@ import java.nio.file.attribute.BasicFileAttributes
|
|||||||
import java.util.zip.ZipEntry
|
import java.util.zip.ZipEntry
|
||||||
import java.util.zip.ZipFile
|
import java.util.zip.ZipFile
|
||||||
import kotlin.io.path.ExperimentalPathApi
|
import kotlin.io.path.ExperimentalPathApi
|
||||||
|
import kotlin.io.path.PathWalkOption
|
||||||
import kotlin.io.path.readAttributes
|
import kotlin.io.path.readAttributes
|
||||||
import kotlin.io.path.walk
|
import kotlin.io.path.walk
|
||||||
|
|
||||||
@@ -72,7 +73,7 @@ fun ContentResolver.resolveName(uri: Uri): String? {
|
|||||||
}
|
}
|
||||||
|
|
||||||
suspend fun File.computeSize(): Long = runInterruptible(Dispatchers.IO) {
|
suspend fun File.computeSize(): Long = runInterruptible(Dispatchers.IO) {
|
||||||
walkCompat().sumOf { it.length() }
|
walkCompat(includeDirectories = false).sumOf { it.length() }
|
||||||
}
|
}
|
||||||
|
|
||||||
fun File.children() = FileSequence(this)
|
fun File.children() = FileSequence(this)
|
||||||
@@ -87,10 +88,16 @@ val File.creationTime
|
|||||||
}
|
}
|
||||||
|
|
||||||
@OptIn(ExperimentalPathApi::class)
|
@OptIn(ExperimentalPathApi::class)
|
||||||
fun File.walkCompat() = if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
|
fun File.walkCompat(includeDirectories: Boolean): Sequence<File> = if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
|
||||||
// Use lazy loading on Android 8.0 and later
|
// Use lazy loading on Android 8.0 and later
|
||||||
toPath().walk().map { it.toFile() }
|
val walk = if (includeDirectories) {
|
||||||
|
toPath().walk(PathWalkOption.INCLUDE_DIRECTORIES)
|
||||||
|
} else {
|
||||||
|
toPath().walk()
|
||||||
|
}
|
||||||
|
walk.map { it.toFile() }
|
||||||
} else {
|
} else {
|
||||||
// Directories are excluded by default in Path.walk(), so do it here as well
|
// Directories are excluded by default in Path.walk(), so do it here as well
|
||||||
walk().filter { it.isFile }
|
val walk = walk()
|
||||||
|
if (includeDirectories) walk else walk.filter { it.isFile }
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -11,8 +11,10 @@ import kotlinx.coroutines.flow.flow
|
|||||||
import kotlinx.coroutines.flow.map
|
import kotlinx.coroutines.flow.map
|
||||||
import kotlinx.coroutines.flow.onCompletion
|
import kotlinx.coroutines.flow.onCompletion
|
||||||
import kotlinx.coroutines.flow.onEach
|
import kotlinx.coroutines.flow.onEach
|
||||||
|
import kotlinx.coroutines.flow.transform
|
||||||
import kotlinx.coroutines.flow.transformLatest
|
import kotlinx.coroutines.flow.transformLatest
|
||||||
import org.koitharu.kotatsu.R
|
import org.koitharu.kotatsu.R
|
||||||
|
import java.util.concurrent.atomic.AtomicInteger
|
||||||
|
|
||||||
fun <T> Flow<T>.onFirst(action: suspend (T) -> Unit): Flow<T> {
|
fun <T> Flow<T>.onFirst(action: suspend (T) -> Unit): Flow<T> {
|
||||||
var isFirstCall = true
|
var isFirstCall = true
|
||||||
@@ -37,6 +39,14 @@ fun <T> Flow<T>.onEachWhile(action: suspend (T) -> Boolean): Flow<T> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fun <T> Flow<T>.onEachIndexed(action: suspend (index: Int, T) -> Unit): Flow<T> {
|
||||||
|
val counter = AtomicInteger(0)
|
||||||
|
return transform { value ->
|
||||||
|
action(counter.getAndIncrement(), value)
|
||||||
|
return@transform emit(value)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
inline fun <T, R> Flow<List<T>>.mapItems(crossinline transform: (T) -> R): Flow<List<R>> {
|
inline fun <T, R> Flow<List<T>>.mapItems(crossinline transform: (T) -> R): Flow<List<R>> {
|
||||||
return map { list -> list.map(transform) }
|
return map { list -> list.map(transform) }
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,18 +3,18 @@ package org.koitharu.kotatsu.core.util.ext
|
|||||||
import android.annotation.SuppressLint
|
import android.annotation.SuppressLint
|
||||||
import android.content.Context
|
import android.content.Context
|
||||||
import android.content.res.Resources
|
import android.content.res.Resources
|
||||||
import android.util.TypedValue
|
|
||||||
import androidx.annotation.Px
|
import androidx.annotation.Px
|
||||||
|
import androidx.core.util.TypedValueCompat
|
||||||
import kotlin.math.roundToInt
|
import kotlin.math.roundToInt
|
||||||
|
|
||||||
@Px
|
@Px
|
||||||
fun Resources.resolveDp(dp: Int) = (dp * displayMetrics.density).roundToInt()
|
fun Resources.resolveDp(dp: Int) = resolveDp(dp.toFloat()).roundToInt()
|
||||||
|
|
||||||
@Px
|
@Px
|
||||||
fun Resources.resolveDp(dp: Float) = dp * displayMetrics.density
|
fun Resources.resolveDp(dp: Float) = TypedValueCompat.dpToPx(dp, displayMetrics)
|
||||||
|
|
||||||
@Px
|
@Px
|
||||||
fun Resources.resolveSp(sp: Float) = TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_SP, sp, displayMetrics)
|
fun Resources.resolveSp(sp: Float) = TypedValueCompat.spToPx(sp, displayMetrics)
|
||||||
|
|
||||||
@SuppressLint("DiscouragedApi")
|
@SuppressLint("DiscouragedApi")
|
||||||
fun Context.getSystemBoolean(resName: String, fallback: Boolean): Boolean {
|
fun Context.getSystemBoolean(resName: String, fallback: Boolean): Boolean {
|
||||||
|
|||||||
@@ -8,9 +8,11 @@ import androidx.annotation.AttrRes
|
|||||||
import androidx.annotation.ColorInt
|
import androidx.annotation.ColorInt
|
||||||
import androidx.annotation.FloatRange
|
import androidx.annotation.FloatRange
|
||||||
import androidx.annotation.Px
|
import androidx.annotation.Px
|
||||||
|
import androidx.annotation.StyleRes
|
||||||
import androidx.core.content.ContextCompat
|
import androidx.core.content.ContextCompat
|
||||||
import androidx.core.content.res.use
|
import androidx.core.content.res.use
|
||||||
import androidx.core.graphics.ColorUtils
|
import androidx.core.graphics.ColorUtils
|
||||||
|
import com.google.android.material.R as materialR
|
||||||
|
|
||||||
fun Context.getThemeDrawable(
|
fun Context.getThemeDrawable(
|
||||||
@AttrRes resId: Int,
|
@AttrRes resId: Int,
|
||||||
@@ -75,3 +77,7 @@ fun TypedArray.getDrawableCompat(context: Context, index: Int): Drawable? {
|
|||||||
val resId = getResourceId(index, 0)
|
val resId = getResourceId(index, 0)
|
||||||
return if (resId != 0) ContextCompat.getDrawable(context, resId) else null
|
return if (resId != 0) ContextCompat.getDrawable(context, resId) else null
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@get:StyleRes
|
||||||
|
val DIALOG_THEME_CENTERED: Int
|
||||||
|
inline get() = materialR.style.ThemeOverlay_Material3_MaterialAlertDialog_Centered
|
||||||
|
|||||||
@@ -2,7 +2,6 @@ package org.koitharu.kotatsu.core.util.ext
|
|||||||
|
|
||||||
import android.content.ActivityNotFoundException
|
import android.content.ActivityNotFoundException
|
||||||
import android.content.res.Resources
|
import android.content.res.Resources
|
||||||
import android.util.AndroidRuntimeException
|
|
||||||
import androidx.annotation.DrawableRes
|
import androidx.annotation.DrawableRes
|
||||||
import androidx.collection.arraySetOf
|
import androidx.collection.arraySetOf
|
||||||
import coil.network.HttpException
|
import coil.network.HttpException
|
||||||
@@ -115,8 +114,8 @@ private val reportableExceptions = arraySetOf<Class<*>>(
|
|||||||
)
|
)
|
||||||
|
|
||||||
fun Throwable.isWebViewUnavailable(): Boolean {
|
fun Throwable.isWebViewUnavailable(): Boolean {
|
||||||
return (this is AndroidRuntimeException && message?.contains("WebView") == true) ||
|
val trace = stackTraceToString()
|
||||||
cause?.isWebViewUnavailable() == true
|
return trace.contains("android.webkit.WebView.<init>")
|
||||||
}
|
}
|
||||||
|
|
||||||
@Suppress("FunctionName")
|
@Suppress("FunctionName")
|
||||||
|
|||||||
@@ -1,32 +1,33 @@
|
|||||||
package org.koitharu.kotatsu.core.util.ext
|
package org.koitharu.kotatsu.core.util.ext
|
||||||
|
|
||||||
import android.app.Activity
|
|
||||||
import android.graphics.Rect
|
import android.graphics.Rect
|
||||||
import android.os.Build
|
import android.os.Build
|
||||||
import android.view.View
|
import android.view.View
|
||||||
import android.view.View.MeasureSpec
|
import android.view.View.MeasureSpec
|
||||||
import android.view.ViewGroup
|
import android.view.ViewGroup
|
||||||
import android.view.inputmethod.InputMethodManager
|
|
||||||
import android.widget.Checkable
|
import android.widget.Checkable
|
||||||
|
import androidx.appcompat.widget.ActionMenuView
|
||||||
|
import androidx.appcompat.widget.Toolbar
|
||||||
|
import androidx.core.view.SoftwareKeyboardControllerCompat
|
||||||
import androidx.core.view.children
|
import androidx.core.view.children
|
||||||
import androidx.core.view.descendants
|
import androidx.core.view.descendants
|
||||||
import androidx.core.view.isVisible
|
import androidx.core.view.isVisible
|
||||||
import androidx.recyclerview.widget.RecyclerView
|
import androidx.recyclerview.widget.RecyclerView
|
||||||
import androidx.recyclerview.widget.RecyclerView.ViewHolder
|
import androidx.recyclerview.widget.RecyclerView.ViewHolder
|
||||||
|
import androidx.swiperefreshlayout.widget.CircularProgressDrawable
|
||||||
import androidx.viewpager2.widget.ViewPager2
|
import androidx.viewpager2.widget.ViewPager2
|
||||||
|
import com.google.android.material.button.MaterialButton
|
||||||
import com.google.android.material.progressindicator.BaseProgressIndicator
|
import com.google.android.material.progressindicator.BaseProgressIndicator
|
||||||
import com.google.android.material.slider.Slider
|
import com.google.android.material.slider.Slider
|
||||||
import com.google.android.material.tabs.TabLayout
|
import com.google.android.material.tabs.TabLayout
|
||||||
import kotlin.math.roundToInt
|
import kotlin.math.roundToInt
|
||||||
|
|
||||||
fun View.hideKeyboard() {
|
fun View.hideKeyboard() {
|
||||||
val imm = context.getSystemService(Activity.INPUT_METHOD_SERVICE) as InputMethodManager
|
SoftwareKeyboardControllerCompat(this).hide()
|
||||||
imm.hideSoftInputFromWindow(this.windowToken, 0)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fun View.showKeyboard() {
|
fun View.showKeyboard() {
|
||||||
val imm = context.getSystemService(Activity.INPUT_METHOD_SERVICE) as InputMethodManager
|
SoftwareKeyboardControllerCompat(this).show()
|
||||||
imm.showSoftInput(this, 0)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fun View.hasGlobalPoint(x: Int, y: Int): Boolean {
|
fun View.hasGlobalPoint(x: Int, y: Int): Boolean {
|
||||||
@@ -153,3 +154,18 @@ fun View.setOnContextClickListenerCompat(listener: View.OnLongClickListener) {
|
|||||||
setOnContextClickListener(listener::onLongClick)
|
setOnContextClickListener(listener::onLongClick)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
val Toolbar.menuView: ActionMenuView?
|
||||||
|
get() {
|
||||||
|
menu // to call ensureMenu()
|
||||||
|
return children.firstNotNullOfOrNull { it as? ActionMenuView }
|
||||||
|
}
|
||||||
|
|
||||||
|
fun MaterialButton.setProgressIcon() {
|
||||||
|
val progressDrawable = CircularProgressDrawable(context)
|
||||||
|
progressDrawable.strokeWidth = resources.resolveDp(2f)
|
||||||
|
progressDrawable.setColorSchemeColors(currentTextColor)
|
||||||
|
progressDrawable.setTintList(textColors)
|
||||||
|
icon = progressDrawable
|
||||||
|
progressDrawable.start()
|
||||||
|
}
|
||||||
|
|||||||
@@ -1,6 +1,8 @@
|
|||||||
package org.koitharu.kotatsu.core.util.progress
|
package org.koitharu.kotatsu.core.util.progress
|
||||||
|
|
||||||
import android.os.SystemClock
|
import android.os.SystemClock
|
||||||
|
import androidx.collection.IntList
|
||||||
|
import androidx.collection.MutableIntList
|
||||||
import java.util.concurrent.TimeUnit
|
import java.util.concurrent.TimeUnit
|
||||||
import kotlin.math.roundToInt
|
import kotlin.math.roundToInt
|
||||||
import kotlin.math.roundToLong
|
import kotlin.math.roundToLong
|
||||||
@@ -10,7 +12,7 @@ private const val NO_TIME = -1L
|
|||||||
|
|
||||||
class TimeLeftEstimator {
|
class TimeLeftEstimator {
|
||||||
|
|
||||||
private var times = ArrayList<Int>()
|
private var times = MutableIntList()
|
||||||
private var lastTick: Tick? = null
|
private var lastTick: Tick? = null
|
||||||
private val tooLargeTime = TimeUnit.DAYS.toMillis(1)
|
private val tooLargeTime = TimeUnit.DAYS.toMillis(1)
|
||||||
|
|
||||||
@@ -50,6 +52,15 @@ class TimeLeftEstimator {
|
|||||||
return if (etl == NO_TIME) NO_TIME else System.currentTimeMillis() + etl
|
return if (etl == NO_TIME) NO_TIME else System.currentTimeMillis() + etl
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private fun IntList.average(): Double {
|
||||||
|
if (isEmpty()) {
|
||||||
|
return 0.0
|
||||||
|
}
|
||||||
|
var acc = 0L
|
||||||
|
forEach { acc += it }
|
||||||
|
return acc / size.toDouble()
|
||||||
|
}
|
||||||
|
|
||||||
private class Tick(
|
private class Tick(
|
||||||
@JvmField val value: Int,
|
@JvmField val value: Int,
|
||||||
@JvmField val total: Int,
|
@JvmField val total: Int,
|
||||||
|
|||||||
@@ -0,0 +1,22 @@
|
|||||||
|
package org.koitharu.kotatsu.details.data
|
||||||
|
|
||||||
|
import android.content.res.Resources
|
||||||
|
import org.koitharu.kotatsu.R
|
||||||
|
|
||||||
|
data class ReadingTime(
|
||||||
|
val minutes: Int,
|
||||||
|
val hours: Int,
|
||||||
|
val isContinue: Boolean,
|
||||||
|
) {
|
||||||
|
|
||||||
|
fun format(resources: Resources): String = when {
|
||||||
|
hours == 0 && minutes == 0 -> resources.getString(R.string.less_than_minute)
|
||||||
|
hours == 0 -> resources.getQuantityString(R.plurals.minutes, minutes, minutes)
|
||||||
|
minutes == 0 -> resources.getQuantityString(R.plurals.hours, hours, hours)
|
||||||
|
else -> resources.getString(
|
||||||
|
R.string.remaining_time_pattern,
|
||||||
|
resources.getQuantityString(R.plurals.hours, hours, hours),
|
||||||
|
resources.getQuantityString(R.plurals.minutes, minutes, minutes),
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,53 @@
|
|||||||
|
package org.koitharu.kotatsu.details.domain
|
||||||
|
|
||||||
|
import org.koitharu.kotatsu.core.model.MangaHistory
|
||||||
|
import org.koitharu.kotatsu.core.model.findById
|
||||||
|
import org.koitharu.kotatsu.core.prefs.AppSettings
|
||||||
|
import org.koitharu.kotatsu.details.data.MangaDetails
|
||||||
|
import org.koitharu.kotatsu.details.data.ReadingTime
|
||||||
|
import org.koitharu.kotatsu.stats.data.StatsRepository
|
||||||
|
import java.util.concurrent.TimeUnit
|
||||||
|
import javax.inject.Inject
|
||||||
|
import kotlin.math.roundToInt
|
||||||
|
|
||||||
|
class ReadingTimeUseCase @Inject constructor(
|
||||||
|
private val settings: AppSettings,
|
||||||
|
private val statsRepository: StatsRepository,
|
||||||
|
) {
|
||||||
|
|
||||||
|
suspend fun invoke(manga: MangaDetails?, branch: String?, history: MangaHistory?): ReadingTime? {
|
||||||
|
if (!settings.isReadingTimeEstimationEnabled) {
|
||||||
|
return null
|
||||||
|
}
|
||||||
|
val chapters = manga?.chapters?.get(branch)
|
||||||
|
if (chapters.isNullOrEmpty()) {
|
||||||
|
return null
|
||||||
|
}
|
||||||
|
val isOnHistoryBranch = history != null && chapters.findById(history.chapterId) != null
|
||||||
|
// Impossible task, I guess. Good luck on this.
|
||||||
|
var averageTimeSec: Int = 20 /* pages */ * getSecondsPerPage(manga.id) * chapters.size
|
||||||
|
if (isOnHistoryBranch) {
|
||||||
|
averageTimeSec = (averageTimeSec * (1f - checkNotNull(history).percent)).roundToInt()
|
||||||
|
}
|
||||||
|
if (averageTimeSec < 60) {
|
||||||
|
return null
|
||||||
|
}
|
||||||
|
return ReadingTime(
|
||||||
|
minutes = (averageTimeSec / 60) % 60,
|
||||||
|
hours = averageTimeSec / 3600,
|
||||||
|
isContinue = isOnHistoryBranch,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
private suspend fun getSecondsPerPage(mangaId: Long): Int {
|
||||||
|
var time = if (settings.isStatsEnabled) {
|
||||||
|
TimeUnit.MILLISECONDS.toSeconds(statsRepository.getTimePerPage(mangaId)).toInt()
|
||||||
|
} else {
|
||||||
|
0
|
||||||
|
}
|
||||||
|
if (time == 0) {
|
||||||
|
time = 10 // default
|
||||||
|
}
|
||||||
|
return time
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -6,12 +6,17 @@ import android.view.View
|
|||||||
import android.view.View.OnLayoutChangeListener
|
import android.view.View.OnLayoutChangeListener
|
||||||
import androidx.activity.OnBackPressedCallback
|
import androidx.activity.OnBackPressedCallback
|
||||||
import androidx.appcompat.view.ActionMode
|
import androidx.appcompat.view.ActionMode
|
||||||
|
import androidx.viewpager2.widget.ViewPager2
|
||||||
import com.google.android.material.bottomsheet.BottomSheetBehavior
|
import com.google.android.material.bottomsheet.BottomSheetBehavior
|
||||||
|
import com.google.android.material.tabs.TabLayout
|
||||||
import org.koitharu.kotatsu.core.ui.util.ActionModeListener
|
import org.koitharu.kotatsu.core.ui.util.ActionModeListener
|
||||||
import org.koitharu.kotatsu.core.util.ext.doOnExpansionsChanged
|
import org.koitharu.kotatsu.core.util.ext.doOnExpansionsChanged
|
||||||
|
import org.koitharu.kotatsu.core.util.ext.setTabsEnabled
|
||||||
|
|
||||||
class ChaptersBottomSheetMediator(
|
class ChaptersBottomSheetMediator(
|
||||||
private val behavior: BottomSheetBehavior<*>,
|
private val behavior: BottomSheetBehavior<*>,
|
||||||
|
private val pager: ViewPager2,
|
||||||
|
private val tabLayout: TabLayout,
|
||||||
) : OnBackPressedCallback(false),
|
) : OnBackPressedCallback(false),
|
||||||
ActionModeListener,
|
ActionModeListener,
|
||||||
OnLayoutChangeListener, View.OnGenericMotionListener {
|
OnLayoutChangeListener, View.OnGenericMotionListener {
|
||||||
@@ -73,7 +78,7 @@ class ChaptersBottomSheetMediator(
|
|||||||
|
|
||||||
fun lock() {
|
fun lock() {
|
||||||
lockCounter++
|
lockCounter++
|
||||||
behavior.isDraggable = lockCounter <= 0
|
updateLock()
|
||||||
}
|
}
|
||||||
|
|
||||||
fun unlock() {
|
fun unlock() {
|
||||||
@@ -81,6 +86,12 @@ class ChaptersBottomSheetMediator(
|
|||||||
if (lockCounter < 0) {
|
if (lockCounter < 0) {
|
||||||
lockCounter = 0
|
lockCounter = 0
|
||||||
}
|
}
|
||||||
|
updateLock()
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun updateLock() {
|
||||||
behavior.isDraggable = lockCounter <= 0
|
behavior.isDraggable = lockCounter <= 0
|
||||||
|
pager.isUserInputEnabled = lockCounter <= 0
|
||||||
|
tabLayout.setTabsEnabled(lockCounter <= 0)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,10 +1,14 @@
|
|||||||
package org.koitharu.kotatsu.details.ui
|
package org.koitharu.kotatsu.details.ui
|
||||||
|
|
||||||
|
import android.content.Context
|
||||||
|
import org.koitharu.kotatsu.R
|
||||||
import org.koitharu.kotatsu.bookmarks.domain.Bookmark
|
import org.koitharu.kotatsu.bookmarks.domain.Bookmark
|
||||||
import org.koitharu.kotatsu.core.model.MangaHistory
|
import org.koitharu.kotatsu.core.model.MangaHistory
|
||||||
import org.koitharu.kotatsu.details.data.MangaDetails
|
import org.koitharu.kotatsu.details.data.MangaDetails
|
||||||
import org.koitharu.kotatsu.details.ui.model.ChapterListItem
|
import org.koitharu.kotatsu.details.ui.model.ChapterListItem
|
||||||
import org.koitharu.kotatsu.details.ui.model.toListItem
|
import org.koitharu.kotatsu.details.ui.model.toListItem
|
||||||
|
import org.koitharu.kotatsu.list.ui.model.ListHeader
|
||||||
|
import org.koitharu.kotatsu.list.ui.model.ListModel
|
||||||
import org.koitharu.kotatsu.parsers.util.mapToSet
|
import org.koitharu.kotatsu.parsers.util.mapToSet
|
||||||
|
|
||||||
fun MangaDetails.mapChapters(
|
fun MangaDetails.mapChapters(
|
||||||
@@ -61,3 +65,22 @@ fun MangaDetails.mapChapters(
|
|||||||
}
|
}
|
||||||
return result
|
return result
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fun List<ChapterListItem>.withVolumeHeaders(context: Context): List<ListModel> {
|
||||||
|
var prevVolume = 0
|
||||||
|
val result = ArrayList<ListModel>((size * 1.4).toInt())
|
||||||
|
for (item in this) {
|
||||||
|
val chapter = item.chapter
|
||||||
|
if (chapter.volume != prevVolume) {
|
||||||
|
val text = if (chapter.volume == 0) {
|
||||||
|
context.getString(R.string.volume_unknown)
|
||||||
|
} else {
|
||||||
|
context.getString(R.string.volume_, chapter.volume)
|
||||||
|
}
|
||||||
|
result.add(ListHeader(text))
|
||||||
|
prevVolume = chapter.volume
|
||||||
|
}
|
||||||
|
result.add(item)
|
||||||
|
}
|
||||||
|
return result
|
||||||
|
}
|
||||||
|
|||||||
@@ -4,7 +4,9 @@ import android.content.Context
|
|||||||
import android.content.Intent
|
import android.content.Intent
|
||||||
import android.graphics.Color
|
import android.graphics.Color
|
||||||
import android.os.Bundle
|
import android.os.Bundle
|
||||||
|
import android.text.style.DynamicDrawableSpan
|
||||||
import android.text.style.ForegroundColorSpan
|
import android.text.style.ForegroundColorSpan
|
||||||
|
import android.text.style.ImageSpan
|
||||||
import android.text.style.RelativeSizeSpan
|
import android.text.style.RelativeSizeSpan
|
||||||
import android.transition.AutoTransition
|
import android.transition.AutoTransition
|
||||||
import android.transition.Slide
|
import android.transition.Slide
|
||||||
@@ -22,13 +24,14 @@ import androidx.appcompat.widget.PopupMenu
|
|||||||
import androidx.core.graphics.Insets
|
import androidx.core.graphics.Insets
|
||||||
import androidx.core.text.buildSpannedString
|
import androidx.core.text.buildSpannedString
|
||||||
import androidx.core.text.inSpans
|
import androidx.core.text.inSpans
|
||||||
|
import androidx.core.view.MenuHost
|
||||||
import androidx.core.view.isGone
|
import androidx.core.view.isGone
|
||||||
import androidx.core.view.isVisible
|
import androidx.core.view.isVisible
|
||||||
import androidx.core.view.updateLayoutParams
|
import androidx.core.view.updateLayoutParams
|
||||||
import androidx.core.view.updatePadding
|
import androidx.core.view.updatePadding
|
||||||
import com.google.android.material.bottomsheet.BottomSheetBehavior
|
import com.google.android.material.bottomsheet.BottomSheetBehavior
|
||||||
import com.google.android.material.snackbar.BaseTransientBottomBar
|
|
||||||
import com.google.android.material.snackbar.Snackbar
|
import com.google.android.material.snackbar.Snackbar
|
||||||
|
import com.google.android.material.tabs.TabLayoutMediator
|
||||||
import dagger.hilt.android.AndroidEntryPoint
|
import dagger.hilt.android.AndroidEntryPoint
|
||||||
import kotlinx.coroutines.flow.FlowCollector
|
import kotlinx.coroutines.flow.FlowCollector
|
||||||
import kotlinx.coroutines.flow.filterNotNull
|
import kotlinx.coroutines.flow.filterNotNull
|
||||||
@@ -37,15 +40,19 @@ import org.koitharu.kotatsu.core.exceptions.resolve.SnackbarErrorObserver
|
|||||||
import org.koitharu.kotatsu.core.model.parcelable.ParcelableManga
|
import org.koitharu.kotatsu.core.model.parcelable.ParcelableManga
|
||||||
import org.koitharu.kotatsu.core.os.AppShortcutManager
|
import org.koitharu.kotatsu.core.os.AppShortcutManager
|
||||||
import org.koitharu.kotatsu.core.parser.MangaIntent
|
import org.koitharu.kotatsu.core.parser.MangaIntent
|
||||||
|
import org.koitharu.kotatsu.core.prefs.AppSettings
|
||||||
import org.koitharu.kotatsu.core.ui.BaseActivity
|
import org.koitharu.kotatsu.core.ui.BaseActivity
|
||||||
import org.koitharu.kotatsu.core.ui.util.MenuInvalidator
|
import org.koitharu.kotatsu.core.ui.util.MenuInvalidator
|
||||||
|
import org.koitharu.kotatsu.core.ui.util.ReversibleActionObserver
|
||||||
import org.koitharu.kotatsu.core.util.ext.doOnExpansionsChanged
|
import org.koitharu.kotatsu.core.util.ext.doOnExpansionsChanged
|
||||||
import org.koitharu.kotatsu.core.util.ext.getAnimationDuration
|
import org.koitharu.kotatsu.core.util.ext.getAnimationDuration
|
||||||
import org.koitharu.kotatsu.core.util.ext.getThemeColor
|
import org.koitharu.kotatsu.core.util.ext.getThemeColor
|
||||||
import org.koitharu.kotatsu.core.util.ext.isAnimationsEnabled
|
import org.koitharu.kotatsu.core.util.ext.isAnimationsEnabled
|
||||||
import org.koitharu.kotatsu.core.util.ext.measureHeight
|
import org.koitharu.kotatsu.core.util.ext.measureHeight
|
||||||
|
import org.koitharu.kotatsu.core.util.ext.menuView
|
||||||
import org.koitharu.kotatsu.core.util.ext.observe
|
import org.koitharu.kotatsu.core.util.ext.observe
|
||||||
import org.koitharu.kotatsu.core.util.ext.observeEvent
|
import org.koitharu.kotatsu.core.util.ext.observeEvent
|
||||||
|
import org.koitharu.kotatsu.core.util.ext.recyclerView
|
||||||
import org.koitharu.kotatsu.core.util.ext.setNavigationBarTransparentCompat
|
import org.koitharu.kotatsu.core.util.ext.setNavigationBarTransparentCompat
|
||||||
import org.koitharu.kotatsu.core.util.ext.setNavigationIconSafe
|
import org.koitharu.kotatsu.core.util.ext.setNavigationIconSafe
|
||||||
import org.koitharu.kotatsu.core.util.ext.setOnContextClickListenerCompat
|
import org.koitharu.kotatsu.core.util.ext.setOnContextClickListenerCompat
|
||||||
@@ -54,6 +61,7 @@ import org.koitharu.kotatsu.databinding.ActivityDetailsBinding
|
|||||||
import org.koitharu.kotatsu.details.service.MangaPrefetchService
|
import org.koitharu.kotatsu.details.service.MangaPrefetchService
|
||||||
import org.koitharu.kotatsu.details.ui.model.ChapterListItem
|
import org.koitharu.kotatsu.details.ui.model.ChapterListItem
|
||||||
import org.koitharu.kotatsu.details.ui.model.HistoryInfo
|
import org.koitharu.kotatsu.details.ui.model.HistoryInfo
|
||||||
|
import org.koitharu.kotatsu.details.ui.pager.DetailsPagerAdapter
|
||||||
import org.koitharu.kotatsu.download.ui.worker.DownloadStartedObserver
|
import org.koitharu.kotatsu.download.ui.worker.DownloadStartedObserver
|
||||||
import org.koitharu.kotatsu.main.ui.owners.NoModalBottomSheetOwner
|
import org.koitharu.kotatsu.main.ui.owners.NoModalBottomSheetOwner
|
||||||
import org.koitharu.kotatsu.parsers.model.Manga
|
import org.koitharu.kotatsu.parsers.model.Manga
|
||||||
@@ -74,10 +82,18 @@ class DetailsActivity :
|
|||||||
@Inject
|
@Inject
|
||||||
lateinit var appShortcutManager: AppShortcutManager
|
lateinit var appShortcutManager: AppShortcutManager
|
||||||
|
|
||||||
|
@Inject
|
||||||
|
lateinit var settings: AppSettings
|
||||||
|
|
||||||
private var buttonTip: WeakReference<ButtonTip>? = null
|
private var buttonTip: WeakReference<ButtonTip>? = null
|
||||||
|
|
||||||
private val viewModel: DetailsViewModel by viewModels()
|
private val viewModel: DetailsViewModel by viewModels()
|
||||||
private lateinit var chaptersMenuProvider: ChaptersMenuProvider
|
|
||||||
|
val secondaryMenuHost: MenuHost
|
||||||
|
get() = viewBinding.toolbarChapters ?: this
|
||||||
|
|
||||||
|
var bottomSheetMediator: ChaptersBottomSheetMediator? = null
|
||||||
|
private set
|
||||||
|
|
||||||
override fun onCreate(savedInstanceState: Bundle?) {
|
override fun onCreate(savedInstanceState: Bundle?) {
|
||||||
super.onCreate(savedInstanceState)
|
super.onCreate(savedInstanceState)
|
||||||
@@ -93,24 +109,22 @@ class DetailsActivity :
|
|||||||
|
|
||||||
if (viewBinding.layoutBottom != null) {
|
if (viewBinding.layoutBottom != null) {
|
||||||
val behavior = BottomSheetBehavior.from(checkNotNull(viewBinding.layoutBottom))
|
val behavior = BottomSheetBehavior.from(checkNotNull(viewBinding.layoutBottom))
|
||||||
val bsMediator = ChaptersBottomSheetMediator(behavior)
|
val bsMediator = ChaptersBottomSheetMediator(behavior, viewBinding.pager, viewBinding.tabs)
|
||||||
actionModeDelegate.addListener(bsMediator)
|
actionModeDelegate.addListener(bsMediator)
|
||||||
checkNotNull(viewBinding.layoutBsHeader).addOnLayoutChangeListener(bsMediator)
|
checkNotNull(viewBinding.layoutBsHeader).addOnLayoutChangeListener(bsMediator)
|
||||||
onBackPressedDispatcher.addCallback(bsMediator)
|
onBackPressedDispatcher.addCallback(bsMediator)
|
||||||
chaptersMenuProvider = ChaptersMenuProvider(viewModel, bsMediator)
|
bottomSheetMediator = bsMediator
|
||||||
behavior.doOnExpansionsChanged(::onChaptersSheetStateChanged)
|
behavior.doOnExpansionsChanged(::onChaptersSheetStateChanged)
|
||||||
viewBinding.toolbarChapters?.setNavigationOnClickListener {
|
viewBinding.toolbarChapters?.setNavigationOnClickListener {
|
||||||
behavior.state = BottomSheetBehavior.STATE_COLLAPSED
|
behavior.state = BottomSheetBehavior.STATE_COLLAPSED
|
||||||
}
|
}
|
||||||
viewBinding.toolbarChapters?.setOnGenericMotionListener(bsMediator)
|
viewBinding.toolbarChapters?.setOnGenericMotionListener(bsMediator)
|
||||||
} else {
|
|
||||||
chaptersMenuProvider = ChaptersMenuProvider(viewModel, null)
|
|
||||||
addMenuProvider(chaptersMenuProvider)
|
|
||||||
}
|
}
|
||||||
onBackPressedDispatcher.addCallback(chaptersMenuProvider)
|
initPager()
|
||||||
|
|
||||||
viewModel.manga.filterNotNull().observe(this, ::onMangaUpdated)
|
viewModel.manga.filterNotNull().observe(this, ::onMangaUpdated)
|
||||||
viewModel.onMangaRemoved.observeEvent(this, ::onMangaRemoved)
|
viewModel.onMangaRemoved.observeEvent(this, ::onMangaRemoved)
|
||||||
|
viewModel.newChaptersCount.observe(this, ::onNewChaptersChanged)
|
||||||
viewModel.onError.observeEvent(
|
viewModel.onError.observeEvent(
|
||||||
this,
|
this,
|
||||||
SnackbarErrorObserver(
|
SnackbarErrorObserver(
|
||||||
@@ -124,21 +138,22 @@ class DetailsActivity :
|
|||||||
},
|
},
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
viewModel.onShowToast.observeEvent(this) {
|
viewModel.onActionDone.observeEvent(
|
||||||
makeSnackbar(getString(it), Snackbar.LENGTH_SHORT).show()
|
this,
|
||||||
}
|
ReversibleActionObserver(viewBinding.containerDetails, viewBinding.layoutBottom),
|
||||||
|
)
|
||||||
viewModel.onShowTip.observeEvent(this) { showTip() }
|
viewModel.onShowTip.observeEvent(this) { showTip() }
|
||||||
viewModel.historyInfo.observe(this, ::onHistoryChanged)
|
viewModel.historyInfo.observe(this, ::onHistoryChanged)
|
||||||
viewModel.selectedBranch.observe(this) {
|
viewModel.selectedBranch.observe(this) {
|
||||||
viewBinding.toolbarChapters?.subtitle = it
|
viewBinding.toolbarChapters?.subtitle = it
|
||||||
viewBinding.textViewSubtitle?.textAndVisible = it
|
viewBinding.textViewSubtitle?.textAndVisible = it
|
||||||
}
|
}
|
||||||
viewModel.isChaptersReversed.observe(
|
val chaptersMenuInvalidator = MenuInvalidator(viewBinding.toolbarChapters ?: this)
|
||||||
this,
|
viewModel.isChaptersReversed.observe(this, chaptersMenuInvalidator)
|
||||||
MenuInvalidator(viewBinding.toolbarChapters ?: this),
|
viewModel.isChaptersEmpty.observe(this, chaptersMenuInvalidator)
|
||||||
)
|
|
||||||
val menuInvalidator = MenuInvalidator(this)
|
val menuInvalidator = MenuInvalidator(this)
|
||||||
viewModel.favouriteCategories.observe(this, menuInvalidator)
|
viewModel.favouriteCategories.observe(this, menuInvalidator)
|
||||||
|
viewModel.isStatsEnabled.observe(this, menuInvalidator)
|
||||||
viewModel.remoteManga.observe(this, menuInvalidator)
|
viewModel.remoteManga.observe(this, menuInvalidator)
|
||||||
viewModel.branches.observe(this) {
|
viewModel.branches.observe(this) {
|
||||||
viewBinding.buttonDropdown.isVisible = it.size > 1
|
viewBinding.buttonDropdown.isVisible = it.size > 1
|
||||||
@@ -153,7 +168,7 @@ class DetailsActivity :
|
|||||||
DetailsMenuProvider(
|
DetailsMenuProvider(
|
||||||
activity = this,
|
activity = this,
|
||||||
viewModel = viewModel,
|
viewModel = viewModel,
|
||||||
snackbarHost = viewBinding.containerChapters,
|
snackbarHost = viewBinding.pager,
|
||||||
appShortcutManager = appShortcutManager,
|
appShortcutManager = appShortcutManager,
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
@@ -176,6 +191,9 @@ class DetailsActivity :
|
|||||||
buttonTip = null
|
buttonTip = null
|
||||||
val menu = PopupMenu(v.context, v)
|
val menu = PopupMenu(v.context, v)
|
||||||
menu.inflate(R.menu.popup_read)
|
menu.inflate(R.menu.popup_read)
|
||||||
|
menu.menu.findItem(R.id.action_forget)?.isVisible = viewModel.historyInfo.value.run {
|
||||||
|
!isIncognitoMode && history != null
|
||||||
|
}
|
||||||
menu.setOnMenuItemClickListener(this)
|
menu.setOnMenuItemClickListener(this)
|
||||||
menu.setForceShowIcon(true)
|
menu.setForceShowIcon(true)
|
||||||
menu.show()
|
menu.show()
|
||||||
@@ -192,6 +210,11 @@ class DetailsActivity :
|
|||||||
true
|
true
|
||||||
}
|
}
|
||||||
|
|
||||||
|
R.id.action_forget -> {
|
||||||
|
viewModel.removeFromHistory()
|
||||||
|
true
|
||||||
|
}
|
||||||
|
|
||||||
R.id.action_pages_thumbs -> {
|
R.id.action_pages_thumbs -> {
|
||||||
val history = viewModel.historyInfo.value.history
|
val history = viewModel.historyInfo.value.history
|
||||||
PagesThumbnailsSheet.show(
|
PagesThumbnailsSheet.show(
|
||||||
@@ -217,12 +240,11 @@ class DetailsActivity :
|
|||||||
TransitionManager.beginDelayedTransition(toolbar, transition)
|
TransitionManager.beginDelayedTransition(toolbar, transition)
|
||||||
}
|
}
|
||||||
if (isExpanded) {
|
if (isExpanded) {
|
||||||
toolbar.addMenuProvider(chaptersMenuProvider)
|
|
||||||
toolbar.setNavigationIconSafe(materialR.drawable.abc_ic_clear_material)
|
toolbar.setNavigationIconSafe(materialR.drawable.abc_ic_clear_material)
|
||||||
} else {
|
} else {
|
||||||
toolbar.removeMenuProvider(chaptersMenuProvider)
|
|
||||||
toolbar.navigationIcon = null
|
toolbar.navigationIcon = null
|
||||||
}
|
}
|
||||||
|
toolbar.menuView?.isVisible = isExpanded
|
||||||
viewBinding.buttonRead.isGone = isExpanded
|
viewBinding.buttonRead.isGone = isExpanded
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -293,11 +315,35 @@ class DetailsActivity :
|
|||||||
viewBinding.textViewTitle?.text = text
|
viewBinding.textViewTitle?.text = text
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private fun onNewChaptersChanged(count: Int) {
|
||||||
|
val tab = viewBinding.tabs.getTabAt(0) ?: return
|
||||||
|
if (count == 0) {
|
||||||
|
tab.removeBadge()
|
||||||
|
} else {
|
||||||
|
val badge = tab.orCreateBadge
|
||||||
|
badge.horizontalOffsetWithText = -resources.getDimensionPixelOffset(R.dimen.margin_small)
|
||||||
|
badge.number = count
|
||||||
|
badge.isVisible = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
private fun showBranchPopupMenu(v: View) {
|
private fun showBranchPopupMenu(v: View) {
|
||||||
val menu = PopupMenu(v.context, v)
|
val menu = PopupMenu(v.context, v)
|
||||||
val branches = viewModel.branches.value
|
val branches = viewModel.branches.value
|
||||||
for ((i, branch) in branches.withIndex()) {
|
for ((i, branch) in branches.withIndex()) {
|
||||||
val title = buildSpannedString {
|
val title = buildSpannedString {
|
||||||
|
if (branch.isCurrent) {
|
||||||
|
inSpans(
|
||||||
|
ImageSpan(
|
||||||
|
this@DetailsActivity,
|
||||||
|
R.drawable.ic_current_chapter,
|
||||||
|
DynamicDrawableSpan.ALIGN_BASELINE,
|
||||||
|
),
|
||||||
|
) {
|
||||||
|
append(' ')
|
||||||
|
}
|
||||||
|
append(' ')
|
||||||
|
}
|
||||||
append(branch.name ?: getString(R.string.system_default))
|
append(branch.name ?: getString(R.string.system_default))
|
||||||
append(' ')
|
append(' ')
|
||||||
append(' ')
|
append(' ')
|
||||||
@@ -326,9 +372,8 @@ class DetailsActivity :
|
|||||||
val manga = viewModel.manga.value ?: return
|
val manga = viewModel.manga.value ?: return
|
||||||
val chapterId = viewModel.historyInfo.value.history?.chapterId
|
val chapterId = viewModel.historyInfo.value.history?.chapterId
|
||||||
if (chapterId != null && manga.chapters?.none { x -> x.id == chapterId } == true) {
|
if (chapterId != null && manga.chapters?.none { x -> x.id == chapterId } == true) {
|
||||||
val snackbar =
|
Snackbar.make(viewBinding.containerDetails, R.string.chapter_is_missing, Snackbar.LENGTH_SHORT)
|
||||||
makeSnackbar(getString(R.string.chapter_is_missing), Snackbar.LENGTH_SHORT)
|
.show()
|
||||||
snackbar.show()
|
|
||||||
} else {
|
} else {
|
||||||
startActivity(
|
startActivity(
|
||||||
IntentBuilder(this)
|
IntentBuilder(this)
|
||||||
@@ -343,6 +388,15 @@ class DetailsActivity :
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private fun initPager() {
|
||||||
|
val adapter = DetailsPagerAdapter(this)
|
||||||
|
viewBinding.pager.recyclerView?.isNestedScrollingEnabled = false
|
||||||
|
viewBinding.pager.offscreenPageLimit = 1
|
||||||
|
viewBinding.pager.adapter = adapter
|
||||||
|
TabLayoutMediator(viewBinding.tabs, viewBinding.pager, adapter).attach()
|
||||||
|
viewBinding.pager.setCurrentItem(settings.defaultDetailsTab, false)
|
||||||
|
}
|
||||||
|
|
||||||
private fun showBottomSheet(isVisible: Boolean) {
|
private fun showBottomSheet(isVisible: Boolean) {
|
||||||
val view = viewBinding.layoutBottom ?: return
|
val view = viewBinding.layoutBottom ?: return
|
||||||
if (view.isVisible == isVisible) return
|
if (view.isVisible == isVisible) return
|
||||||
@@ -353,17 +407,6 @@ class DetailsActivity :
|
|||||||
view.isVisible = isVisible
|
view.isVisible = isVisible
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun makeSnackbar(
|
|
||||||
text: CharSequence,
|
|
||||||
@BaseTransientBottomBar.Duration duration: Int,
|
|
||||||
): Snackbar {
|
|
||||||
val sb = Snackbar.make(viewBinding.containerDetails, text, duration)
|
|
||||||
if (viewBinding.layoutBottom?.isVisible == true) {
|
|
||||||
sb.anchorView = viewBinding.toolbarChapters
|
|
||||||
}
|
|
||||||
return sb
|
|
||||||
}
|
|
||||||
|
|
||||||
private class PrefetchObserver(
|
private class PrefetchObserver(
|
||||||
private val context: Context,
|
private val context: Context,
|
||||||
) : FlowCollector<List<ChapterListItem>?> {
|
) : FlowCollector<List<ChapterListItem>?> {
|
||||||
|
|||||||
@@ -10,8 +10,6 @@ import android.widget.Toast
|
|||||||
import androidx.appcompat.widget.PopupMenu
|
import androidx.appcompat.widget.PopupMenu
|
||||||
import androidx.core.content.ContextCompat
|
import androidx.core.content.ContextCompat
|
||||||
import androidx.core.graphics.Insets
|
import androidx.core.graphics.Insets
|
||||||
import androidx.core.text.buildSpannedString
|
|
||||||
import androidx.core.text.color
|
|
||||||
import androidx.core.text.method.LinkMovementMethodCompat
|
import androidx.core.text.method.LinkMovementMethodCompat
|
||||||
import androidx.core.view.isGone
|
import androidx.core.view.isGone
|
||||||
import androidx.core.view.isVisible
|
import androidx.core.view.isVisible
|
||||||
@@ -23,13 +21,14 @@ import coil.request.SuccessResult
|
|||||||
import coil.util.CoilUtils
|
import coil.util.CoilUtils
|
||||||
import com.google.android.material.chip.Chip
|
import com.google.android.material.chip.Chip
|
||||||
import dagger.hilt.android.AndroidEntryPoint
|
import dagger.hilt.android.AndroidEntryPoint
|
||||||
import kotlinx.coroutines.flow.combine
|
|
||||||
import kotlinx.coroutines.flow.filterNotNull
|
import kotlinx.coroutines.flow.filterNotNull
|
||||||
import org.koitharu.kotatsu.R
|
import org.koitharu.kotatsu.R
|
||||||
import org.koitharu.kotatsu.bookmarks.domain.Bookmark
|
import org.koitharu.kotatsu.bookmarks.domain.Bookmark
|
||||||
import org.koitharu.kotatsu.bookmarks.ui.adapter.BookmarksAdapter
|
import org.koitharu.kotatsu.bookmarks.ui.adapter.BookmarksAdapter
|
||||||
import org.koitharu.kotatsu.bookmarks.ui.sheet.BookmarksSheet
|
import org.koitharu.kotatsu.bookmarks.ui.sheet.BookmarksSheet
|
||||||
import org.koitharu.kotatsu.core.model.countChaptersByBranch
|
import org.koitharu.kotatsu.core.model.countChaptersByBranch
|
||||||
|
import org.koitharu.kotatsu.core.model.iconResId
|
||||||
|
import org.koitharu.kotatsu.core.model.titleResId
|
||||||
import org.koitharu.kotatsu.core.ui.BaseFragment
|
import org.koitharu.kotatsu.core.ui.BaseFragment
|
||||||
import org.koitharu.kotatsu.core.ui.BaseListAdapter
|
import org.koitharu.kotatsu.core.ui.BaseListAdapter
|
||||||
import org.koitharu.kotatsu.core.ui.image.CoverSizeResolver
|
import org.koitharu.kotatsu.core.ui.image.CoverSizeResolver
|
||||||
@@ -40,7 +39,6 @@ import org.koitharu.kotatsu.core.util.FileSize
|
|||||||
import org.koitharu.kotatsu.core.util.ext.crossfade
|
import org.koitharu.kotatsu.core.util.ext.crossfade
|
||||||
import org.koitharu.kotatsu.core.util.ext.drawableTop
|
import org.koitharu.kotatsu.core.util.ext.drawableTop
|
||||||
import org.koitharu.kotatsu.core.util.ext.enqueueWith
|
import org.koitharu.kotatsu.core.util.ext.enqueueWith
|
||||||
import org.koitharu.kotatsu.core.util.ext.getThemeColor
|
|
||||||
import org.koitharu.kotatsu.core.util.ext.ifNullOrEmpty
|
import org.koitharu.kotatsu.core.util.ext.ifNullOrEmpty
|
||||||
import org.koitharu.kotatsu.core.util.ext.isTextTruncated
|
import org.koitharu.kotatsu.core.util.ext.isTextTruncated
|
||||||
import org.koitharu.kotatsu.core.util.ext.observe
|
import org.koitharu.kotatsu.core.util.ext.observe
|
||||||
@@ -50,6 +48,7 @@ import org.koitharu.kotatsu.core.util.ext.scaleUpActivityOptionsOf
|
|||||||
import org.koitharu.kotatsu.core.util.ext.showOrHide
|
import org.koitharu.kotatsu.core.util.ext.showOrHide
|
||||||
import org.koitharu.kotatsu.core.util.ext.textAndVisible
|
import org.koitharu.kotatsu.core.util.ext.textAndVisible
|
||||||
import org.koitharu.kotatsu.databinding.FragmentDetailsBinding
|
import org.koitharu.kotatsu.databinding.FragmentDetailsBinding
|
||||||
|
import org.koitharu.kotatsu.details.data.ReadingTime
|
||||||
import org.koitharu.kotatsu.details.ui.model.ChapterListItem
|
import org.koitharu.kotatsu.details.ui.model.ChapterListItem
|
||||||
import org.koitharu.kotatsu.details.ui.model.HistoryInfo
|
import org.koitharu.kotatsu.details.ui.model.HistoryInfo
|
||||||
import org.koitharu.kotatsu.details.ui.related.RelatedMangaActivity
|
import org.koitharu.kotatsu.details.ui.related.RelatedMangaActivity
|
||||||
@@ -63,10 +62,10 @@ import org.koitharu.kotatsu.list.ui.adapter.mangaGridItemAD
|
|||||||
import org.koitharu.kotatsu.list.ui.model.ListModel
|
import org.koitharu.kotatsu.list.ui.model.ListModel
|
||||||
import org.koitharu.kotatsu.list.ui.model.MangaItemModel
|
import org.koitharu.kotatsu.list.ui.model.MangaItemModel
|
||||||
import org.koitharu.kotatsu.list.ui.size.StaticItemSizeResolver
|
import org.koitharu.kotatsu.list.ui.size.StaticItemSizeResolver
|
||||||
|
import org.koitharu.kotatsu.local.ui.info.LocalInfoDialog
|
||||||
import org.koitharu.kotatsu.main.ui.owners.NoModalBottomSheetOwner
|
import org.koitharu.kotatsu.main.ui.owners.NoModalBottomSheetOwner
|
||||||
import org.koitharu.kotatsu.parsers.model.Manga
|
import org.koitharu.kotatsu.parsers.model.Manga
|
||||||
import org.koitharu.kotatsu.parsers.model.MangaSource
|
import org.koitharu.kotatsu.parsers.model.MangaSource
|
||||||
import org.koitharu.kotatsu.parsers.model.MangaState
|
|
||||||
import org.koitharu.kotatsu.parsers.model.MangaTag
|
import org.koitharu.kotatsu.parsers.model.MangaTag
|
||||||
import org.koitharu.kotatsu.reader.ui.ReaderActivity
|
import org.koitharu.kotatsu.reader.ui.ReaderActivity
|
||||||
import org.koitharu.kotatsu.scrobbling.common.domain.model.ScrobblingInfo
|
import org.koitharu.kotatsu.scrobbling.common.domain.model.ScrobblingInfo
|
||||||
@@ -74,7 +73,6 @@ import org.koitharu.kotatsu.scrobbling.common.ui.selector.ScrobblingSelectorShee
|
|||||||
import org.koitharu.kotatsu.search.ui.MangaListActivity
|
import org.koitharu.kotatsu.search.ui.MangaListActivity
|
||||||
import org.koitharu.kotatsu.search.ui.SearchActivity
|
import org.koitharu.kotatsu.search.ui.SearchActivity
|
||||||
import javax.inject.Inject
|
import javax.inject.Inject
|
||||||
import com.google.android.material.R as materialR
|
|
||||||
|
|
||||||
@AndroidEntryPoint
|
@AndroidEntryPoint
|
||||||
class DetailsFragment :
|
class DetailsFragment :
|
||||||
@@ -105,6 +103,7 @@ class DetailsFragment :
|
|||||||
binding.buttonScrobblingMore.setOnClickListener(this)
|
binding.buttonScrobblingMore.setOnClickListener(this)
|
||||||
binding.buttonRelatedMore.setOnClickListener(this)
|
binding.buttonRelatedMore.setOnClickListener(this)
|
||||||
binding.infoLayout.textViewSource.setOnClickListener(this)
|
binding.infoLayout.textViewSource.setOnClickListener(this)
|
||||||
|
binding.infoLayout.textViewSize.setOnClickListener(this)
|
||||||
binding.textViewDescription.addOnLayoutChangeListener(this)
|
binding.textViewDescription.addOnLayoutChangeListener(this)
|
||||||
binding.textViewDescription.viewTreeObserver.addOnDrawListener(this)
|
binding.textViewDescription.viewTreeObserver.addOnDrawListener(this)
|
||||||
binding.textViewDescription.movementMethod = LinkMovementMethodCompat.getInstance()
|
binding.textViewDescription.movementMethod = LinkMovementMethodCompat.getInstance()
|
||||||
@@ -121,7 +120,8 @@ class DetailsFragment :
|
|||||||
viewModel.description.observe(viewLifecycleOwner, ::onDescriptionChanged)
|
viewModel.description.observe(viewLifecycleOwner, ::onDescriptionChanged)
|
||||||
viewModel.localSize.observe(viewLifecycleOwner, ::onLocalSizeChanged)
|
viewModel.localSize.observe(viewLifecycleOwner, ::onLocalSizeChanged)
|
||||||
viewModel.relatedManga.observe(viewLifecycleOwner, ::onRelatedMangaChanged)
|
viewModel.relatedManga.observe(viewLifecycleOwner, ::onRelatedMangaChanged)
|
||||||
combine(viewModel.chapters, viewModel.newChaptersCount, ::Pair).observe(viewLifecycleOwner, ::onChaptersChanged)
|
viewModel.chapters.observe(viewLifecycleOwner, ::onChaptersChanged)
|
||||||
|
viewModel.readingTime.observe(viewLifecycleOwner, ::onReadingTimeChanged)
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun onItemClick(item: Bookmark, view: View) {
|
override fun onItemClick(item: Bookmark, view: View) {
|
||||||
@@ -181,28 +181,13 @@ class DetailsFragment :
|
|||||||
ratingBar.isVisible = false
|
ratingBar.isVisible = false
|
||||||
}
|
}
|
||||||
|
|
||||||
when (manga.state) {
|
infoLayout.textViewState.apply {
|
||||||
MangaState.FINISHED -> infoLayout.textViewState.apply {
|
manga.state?.let { state ->
|
||||||
textAndVisible = resources.getString(R.string.state_finished)
|
textAndVisible = resources.getString(state.titleResId)
|
||||||
drawableTop = ContextCompat.getDrawable(context, R.drawable.ic_state_finished)
|
drawableTop = ContextCompat.getDrawable(context, state.iconResId)
|
||||||
|
} ?: run {
|
||||||
|
isVisible = false
|
||||||
}
|
}
|
||||||
|
|
||||||
MangaState.ONGOING -> infoLayout.textViewState.apply {
|
|
||||||
textAndVisible = resources.getString(R.string.state_ongoing)
|
|
||||||
drawableTop = ContextCompat.getDrawable(context, R.drawable.ic_state_ongoing)
|
|
||||||
}
|
|
||||||
|
|
||||||
MangaState.ABANDONED -> infoLayout.textViewState.apply {
|
|
||||||
textAndVisible = resources.getString(R.string.state_abandoned)
|
|
||||||
drawableTop = ContextCompat.getDrawable(context, R.drawable.ic_state_abandoned)
|
|
||||||
}
|
|
||||||
|
|
||||||
MangaState.PAUSED -> infoLayout.textViewState.apply {
|
|
||||||
textAndVisible = resources.getString(R.string.state_paused)
|
|
||||||
drawableTop = ContextCompat.getDrawable(context, R.drawable.ic_action_pause)
|
|
||||||
}
|
|
||||||
|
|
||||||
null -> infoLayout.textViewState.isVisible = false
|
|
||||||
}
|
}
|
||||||
if (manga.source == MangaSource.LOCAL) {
|
if (manga.source == MangaSource.LOCAL) {
|
||||||
infoLayout.textViewSource.isVisible = false
|
infoLayout.textViewSource.isVisible = false
|
||||||
@@ -218,8 +203,7 @@ class DetailsFragment :
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun onChaptersChanged(data: Pair<List<ChapterListItem>?, Int>) {
|
private fun onChaptersChanged(chapters: List<ChapterListItem>?) {
|
||||||
val (chapters, newChapters) = data
|
|
||||||
val infoLayout = requireViewBinding().infoLayout
|
val infoLayout = requireViewBinding().infoLayout
|
||||||
if (chapters.isNullOrEmpty()) {
|
if (chapters.isNullOrEmpty()) {
|
||||||
infoLayout.textViewChapters.isVisible = false
|
infoLayout.textViewChapters.isVisible = false
|
||||||
@@ -227,22 +211,23 @@ class DetailsFragment :
|
|||||||
val count = chapters.countChaptersByBranch()
|
val count = chapters.countChaptersByBranch()
|
||||||
infoLayout.textViewChapters.isVisible = true
|
infoLayout.textViewChapters.isVisible = true
|
||||||
val chaptersText = resources.getQuantityString(R.plurals.chapters, count, count)
|
val chaptersText = resources.getQuantityString(R.plurals.chapters, count, count)
|
||||||
infoLayout.textViewChapters.text = if (newChapters == 0) {
|
infoLayout.textViewChapters.text = chaptersText
|
||||||
chaptersText
|
|
||||||
} else {
|
|
||||||
buildSpannedString {
|
|
||||||
append(chaptersText)
|
|
||||||
append(' ')
|
|
||||||
color(infoLayout.textViewChapters.context.getThemeColor(materialR.attr.colorError)) {
|
|
||||||
append("(+")
|
|
||||||
append(newChapters.toString())
|
|
||||||
append(')')
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private fun onReadingTimeChanged(time: ReadingTime?) {
|
||||||
|
val binding = viewBinding ?: return
|
||||||
|
if (time == null) {
|
||||||
|
binding.approximateReadTimeLayout.isVisible = false
|
||||||
|
return
|
||||||
|
}
|
||||||
|
binding.approximateReadTime.text = time.format(resources)
|
||||||
|
binding.approximateReadTimeTitle.setText(
|
||||||
|
if (time.isContinue) R.string.approximate_remaining_time else R.string.approximate_reading_time
|
||||||
|
)
|
||||||
|
binding.approximateReadTimeLayout.isVisible = true
|
||||||
|
}
|
||||||
|
|
||||||
private fun onDescriptionChanged(description: CharSequence?) {
|
private fun onDescriptionChanged(description: CharSequence?) {
|
||||||
val tv = requireViewBinding().textViewDescription
|
val tv = requireViewBinding().textViewDescription
|
||||||
if (description.isNullOrBlank()) {
|
if (description.isNullOrBlank()) {
|
||||||
@@ -341,6 +326,10 @@ class DetailsFragment :
|
|||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
R.id.textView_size -> {
|
||||||
|
LocalInfoDialog.show(parentFragmentManager, manga)
|
||||||
|
}
|
||||||
|
|
||||||
R.id.imageView_cover -> {
|
R.id.imageView_cover -> {
|
||||||
startActivity(
|
startActivity(
|
||||||
ImageActivity.newIntent(
|
ImageActivity.newIntent(
|
||||||
|
|||||||
@@ -23,6 +23,7 @@ import org.koitharu.kotatsu.favourites.ui.categories.select.FavoriteSheet
|
|||||||
import org.koitharu.kotatsu.parsers.model.MangaSource
|
import org.koitharu.kotatsu.parsers.model.MangaSource
|
||||||
import org.koitharu.kotatsu.scrobbling.common.ui.selector.ScrobblingSelectorSheet
|
import org.koitharu.kotatsu.scrobbling.common.ui.selector.ScrobblingSelectorSheet
|
||||||
import org.koitharu.kotatsu.search.ui.multi.MultiSearchActivity
|
import org.koitharu.kotatsu.search.ui.multi.MultiSearchActivity
|
||||||
|
import org.koitharu.kotatsu.stats.ui.sheet.MangaStatsSheet
|
||||||
|
|
||||||
class DetailsMenuProvider(
|
class DetailsMenuProvider(
|
||||||
private val activity: FragmentActivity,
|
private val activity: FragmentActivity,
|
||||||
@@ -43,6 +44,7 @@ class DetailsMenuProvider(
|
|||||||
menu.findItem(R.id.action_shortcut).isVisible = ShortcutManagerCompat.isRequestPinShortcutSupported(activity)
|
menu.findItem(R.id.action_shortcut).isVisible = ShortcutManagerCompat.isRequestPinShortcutSupported(activity)
|
||||||
menu.findItem(R.id.action_scrobbling).isVisible = viewModel.isScrobblingAvailable
|
menu.findItem(R.id.action_scrobbling).isVisible = viewModel.isScrobblingAvailable
|
||||||
menu.findItem(R.id.action_online).isVisible = viewModel.remoteManga.value != null
|
menu.findItem(R.id.action_online).isVisible = viewModel.remoteManga.value != null
|
||||||
|
menu.findItem(R.id.action_stats).isVisible = viewModel.isStatsEnabled.value
|
||||||
menu.findItem(R.id.action_favourite).setIcon(
|
menu.findItem(R.id.action_favourite).setIcon(
|
||||||
if (viewModel.favouriteCategories.value) R.drawable.ic_heart else R.drawable.ic_heart_outline,
|
if (viewModel.favouriteCategories.value) R.drawable.ic_heart else R.drawable.ic_heart_outline,
|
||||||
)
|
)
|
||||||
@@ -101,6 +103,12 @@ class DetailsMenuProvider(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
R.id.action_stats -> {
|
||||||
|
viewModel.manga.value?.let {
|
||||||
|
MangaStatsSheet.show(activity.supportFragmentManager, it)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
R.id.action_scrobbling -> {
|
R.id.action_scrobbling -> {
|
||||||
viewModel.manga.value?.let {
|
viewModel.manga.value?.let {
|
||||||
ScrobblingSelectorSheet.show(activity.supportFragmentManager, it, null)
|
ScrobblingSelectorSheet.show(activity.supportFragmentManager, it, null)
|
||||||
|
|||||||
@@ -21,15 +21,18 @@ import kotlinx.coroutines.flow.mapLatest
|
|||||||
import kotlinx.coroutines.flow.stateIn
|
import kotlinx.coroutines.flow.stateIn
|
||||||
import kotlinx.coroutines.flow.update
|
import kotlinx.coroutines.flow.update
|
||||||
import kotlinx.coroutines.plus
|
import kotlinx.coroutines.plus
|
||||||
|
import okio.FileNotFoundException
|
||||||
import org.koitharu.kotatsu.R
|
import org.koitharu.kotatsu.R
|
||||||
import org.koitharu.kotatsu.bookmarks.domain.Bookmark
|
import org.koitharu.kotatsu.bookmarks.domain.Bookmark
|
||||||
import org.koitharu.kotatsu.bookmarks.domain.BookmarksRepository
|
import org.koitharu.kotatsu.bookmarks.domain.BookmarksRepository
|
||||||
|
import org.koitharu.kotatsu.core.model.findById
|
||||||
import org.koitharu.kotatsu.core.model.getPreferredBranch
|
import org.koitharu.kotatsu.core.model.getPreferredBranch
|
||||||
import org.koitharu.kotatsu.core.parser.MangaIntent
|
import org.koitharu.kotatsu.core.parser.MangaIntent
|
||||||
import org.koitharu.kotatsu.core.prefs.AppSettings
|
import org.koitharu.kotatsu.core.prefs.AppSettings
|
||||||
import org.koitharu.kotatsu.core.prefs.ListMode
|
import org.koitharu.kotatsu.core.prefs.ListMode
|
||||||
import org.koitharu.kotatsu.core.prefs.observeAsStateFlow
|
import org.koitharu.kotatsu.core.prefs.observeAsStateFlow
|
||||||
import org.koitharu.kotatsu.core.ui.BaseViewModel
|
import org.koitharu.kotatsu.core.ui.BaseViewModel
|
||||||
|
import org.koitharu.kotatsu.core.ui.util.ReversibleAction
|
||||||
import org.koitharu.kotatsu.core.util.ext.MutableEventFlow
|
import org.koitharu.kotatsu.core.util.ext.MutableEventFlow
|
||||||
import org.koitharu.kotatsu.core.util.ext.call
|
import org.koitharu.kotatsu.core.util.ext.call
|
||||||
import org.koitharu.kotatsu.core.util.ext.computeSize
|
import org.koitharu.kotatsu.core.util.ext.computeSize
|
||||||
@@ -40,6 +43,7 @@ import org.koitharu.kotatsu.details.domain.BranchComparator
|
|||||||
import org.koitharu.kotatsu.details.domain.DetailsInteractor
|
import org.koitharu.kotatsu.details.domain.DetailsInteractor
|
||||||
import org.koitharu.kotatsu.details.domain.DetailsLoadUseCase
|
import org.koitharu.kotatsu.details.domain.DetailsLoadUseCase
|
||||||
import org.koitharu.kotatsu.details.domain.ProgressUpdateUseCase
|
import org.koitharu.kotatsu.details.domain.ProgressUpdateUseCase
|
||||||
|
import org.koitharu.kotatsu.details.domain.ReadingTimeUseCase
|
||||||
import org.koitharu.kotatsu.details.domain.RelatedMangaUseCase
|
import org.koitharu.kotatsu.details.domain.RelatedMangaUseCase
|
||||||
import org.koitharu.kotatsu.details.ui.model.ChapterListItem
|
import org.koitharu.kotatsu.details.ui.model.ChapterListItem
|
||||||
import org.koitharu.kotatsu.details.ui.model.HistoryInfo
|
import org.koitharu.kotatsu.details.ui.model.HistoryInfo
|
||||||
@@ -74,13 +78,14 @@ class DetailsViewModel @Inject constructor(
|
|||||||
private val extraProvider: ListExtraProvider,
|
private val extraProvider: ListExtraProvider,
|
||||||
private val detailsLoadUseCase: DetailsLoadUseCase,
|
private val detailsLoadUseCase: DetailsLoadUseCase,
|
||||||
private val progressUpdateUseCase: ProgressUpdateUseCase,
|
private val progressUpdateUseCase: ProgressUpdateUseCase,
|
||||||
|
private val readingTimeUseCase: ReadingTimeUseCase,
|
||||||
) : BaseViewModel() {
|
) : BaseViewModel() {
|
||||||
|
|
||||||
private val intent = MangaIntent(savedStateHandle)
|
private val intent = MangaIntent(savedStateHandle)
|
||||||
private val mangaId = intent.mangaId
|
private val mangaId = intent.mangaId
|
||||||
private var loadingJob: Job
|
private var loadingJob: Job
|
||||||
|
|
||||||
val onShowToast = MutableEventFlow<Int>()
|
val onActionDone = MutableEventFlow<ReversibleAction>()
|
||||||
val onShowTip = MutableEventFlow<Unit>()
|
val onShowTip = MutableEventFlow<Unit>()
|
||||||
val onSelectChapter = MutableEventFlow<Long>()
|
val onSelectChapter = MutableEventFlow<Long>()
|
||||||
val onDownloadStarted = MutableEventFlow<Unit>()
|
val onDownloadStarted = MutableEventFlow<Unit>()
|
||||||
@@ -95,6 +100,10 @@ class DetailsViewModel @Inject constructor(
|
|||||||
val favouriteCategories = interactor.observeIsFavourite(mangaId)
|
val favouriteCategories = interactor.observeIsFavourite(mangaId)
|
||||||
.stateIn(viewModelScope + Dispatchers.Default, SharingStarted.Eagerly, false)
|
.stateIn(viewModelScope + Dispatchers.Default, SharingStarted.Eagerly, false)
|
||||||
|
|
||||||
|
val isStatsEnabled = settings.observeAsStateFlow(viewModelScope + Dispatchers.Default, AppSettings.KEY_STATS_ENABLED) {
|
||||||
|
isStatsEnabled
|
||||||
|
}
|
||||||
|
|
||||||
val remoteManga = MutableStateFlow<Manga?>(null)
|
val remoteManga = MutableStateFlow<Manga?>(null)
|
||||||
|
|
||||||
val newChaptersCount = details.flatMapLatest { d ->
|
val newChaptersCount = details.flatMapLatest { d ->
|
||||||
@@ -167,10 +176,21 @@ class DetailsViewModel @Inject constructor(
|
|||||||
val branches: StateFlow<List<MangaBranch>> = combine(
|
val branches: StateFlow<List<MangaBranch>> = combine(
|
||||||
details,
|
details,
|
||||||
selectedBranch,
|
selectedBranch,
|
||||||
) { m, b ->
|
history,
|
||||||
(m?.chapters ?: return@combine emptyList())
|
) { m, b, h ->
|
||||||
.map { x -> MangaBranch(x.key, x.value.size, x.key == b) }
|
val c = m?.chapters
|
||||||
.sortedWith(BranchComparator())
|
if (c.isNullOrEmpty()) {
|
||||||
|
return@combine emptyList()
|
||||||
|
}
|
||||||
|
val currentBranch = h?.let { m.allChapters.findById(it.chapterId) }?.branch
|
||||||
|
c.map { x ->
|
||||||
|
MangaBranch(
|
||||||
|
name = x.key,
|
||||||
|
count = x.value.size,
|
||||||
|
isSelected = x.key == b,
|
||||||
|
isCurrent = h != null && x.key == currentBranch,
|
||||||
|
)
|
||||||
|
}.sortedWith(BranchComparator())
|
||||||
}.stateIn(viewModelScope + Dispatchers.Default, SharingStarted.Eagerly, emptyList())
|
}.stateIn(viewModelScope + Dispatchers.Default, SharingStarted.Eagerly, emptyList())
|
||||||
|
|
||||||
val isChaptersEmpty: StateFlow<Boolean> = details.map {
|
val isChaptersEmpty: StateFlow<Boolean> = details.map {
|
||||||
@@ -198,6 +218,14 @@ class DetailsViewModel @Inject constructor(
|
|||||||
(if (reversed) list.asReversed() else list).filterSearch(query)
|
(if (reversed) list.asReversed() else list).filterSearch(query)
|
||||||
}.stateIn(viewModelScope, SharingStarted.Eagerly, emptyList())
|
}.stateIn(viewModelScope, SharingStarted.Eagerly, emptyList())
|
||||||
|
|
||||||
|
val readingTime = combine(
|
||||||
|
details,
|
||||||
|
selectedBranch,
|
||||||
|
history,
|
||||||
|
) { m, b, h ->
|
||||||
|
readingTimeUseCase.invoke(m, b, h)
|
||||||
|
}.stateIn(viewModelScope, SharingStarted.Lazily, null)
|
||||||
|
|
||||||
val selectedBranchValue: String?
|
val selectedBranchValue: String?
|
||||||
get() = selectedBranch.value
|
get() = selectedBranch.value
|
||||||
|
|
||||||
@@ -234,7 +262,7 @@ class DetailsViewModel @Inject constructor(
|
|||||||
fun deleteLocal() {
|
fun deleteLocal() {
|
||||||
val m = details.value?.local?.manga
|
val m = details.value?.local?.manga
|
||||||
if (m == null) {
|
if (m == null) {
|
||||||
onShowToast.call(R.string.file_not_found)
|
errorEvent.call(FileNotFoundException())
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
launchLoadingJob(Dispatchers.Default) {
|
launchLoadingJob(Dispatchers.Default) {
|
||||||
@@ -246,7 +274,7 @@ class DetailsViewModel @Inject constructor(
|
|||||||
fun removeBookmark(bookmark: Bookmark) {
|
fun removeBookmark(bookmark: Bookmark) {
|
||||||
launchJob(Dispatchers.Default) {
|
launchJob(Dispatchers.Default) {
|
||||||
bookmarksRepository.removeBookmark(bookmark)
|
bookmarksRepository.removeBookmark(bookmark)
|
||||||
onShowToast.call(R.string.bookmark_removed)
|
onActionDone.call(ReversibleAction(R.string.bookmark_removed, null))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -296,6 +324,7 @@ class DetailsViewModel @Inject constructor(
|
|||||||
page = 0,
|
page = 0,
|
||||||
scroll = 0,
|
scroll = 0,
|
||||||
percent = percent,
|
percent = percent,
|
||||||
|
force = true,
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -322,6 +351,13 @@ class DetailsViewModel @Inject constructor(
|
|||||||
settings.closeTip(DetailsActivity.TIP_BUTTON)
|
settings.closeTip(DetailsActivity.TIP_BUTTON)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fun removeFromHistory() {
|
||||||
|
launchJob(Dispatchers.Default) {
|
||||||
|
val handle = historyRepository.delete(setOf(mangaId))
|
||||||
|
onActionDone.call(ReversibleAction(R.string.removed_from_history, handle))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
private fun doLoad() = launchLoadingJob(Dispatchers.Default) {
|
private fun doLoad() = launchLoadingJob(Dispatchers.Default) {
|
||||||
detailsLoadUseCase.invoke(intent)
|
detailsLoadUseCase.invoke(intent)
|
||||||
.onEachWhile {
|
.onEachWhile {
|
||||||
|
|||||||
@@ -8,6 +8,7 @@ import org.koitharu.kotatsu.core.ui.dialog.RecyclerViewAlertDialog
|
|||||||
import org.koitharu.kotatsu.core.ui.list.OnListItemClickListener
|
import org.koitharu.kotatsu.core.ui.list.OnListItemClickListener
|
||||||
import org.koitharu.kotatsu.download.ui.dialog.DownloadOption
|
import org.koitharu.kotatsu.download.ui.dialog.DownloadOption
|
||||||
import org.koitharu.kotatsu.download.ui.dialog.downloadOptionAD
|
import org.koitharu.kotatsu.download.ui.dialog.downloadOptionAD
|
||||||
|
import org.koitharu.kotatsu.settings.SettingsActivity
|
||||||
|
|
||||||
class DownloadDialogHelper(
|
class DownloadDialogHelper(
|
||||||
private val host: View,
|
private val host: View,
|
||||||
@@ -57,6 +58,9 @@ class DownloadDialogHelper(
|
|||||||
.setCancelable(true)
|
.setCancelable(true)
|
||||||
.setTitle(R.string.download)
|
.setTitle(R.string.download)
|
||||||
.setNegativeButton(android.R.string.cancel)
|
.setNegativeButton(android.R.string.cancel)
|
||||||
|
.setNeutralButton(R.string.settings) { _, _ ->
|
||||||
|
host.context.startActivity(SettingsActivity.newDownloadsSettingsIntent(host.context))
|
||||||
|
}
|
||||||
.setItems(options)
|
.setItems(options)
|
||||||
.create()
|
.create()
|
||||||
.also { it.show() }
|
.also { it.show() }
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
package org.koitharu.kotatsu.details.ui.adapter
|
package org.koitharu.kotatsu.details.ui.adapter
|
||||||
|
|
||||||
|
import android.graphics.Typeface
|
||||||
import androidx.core.content.ContextCompat
|
import androidx.core.content.ContextCompat
|
||||||
import androidx.core.view.isVisible
|
import androidx.core.view.isVisible
|
||||||
import com.hannesdorfmann.adapterdelegates4.dsl.adapterDelegateViewBinding
|
import com.hannesdorfmann.adapterdelegates4.dsl.adapterDelegateViewBinding
|
||||||
@@ -7,15 +8,16 @@ import org.koitharu.kotatsu.R
|
|||||||
import org.koitharu.kotatsu.core.ui.list.AdapterDelegateClickListenerAdapter
|
import org.koitharu.kotatsu.core.ui.list.AdapterDelegateClickListenerAdapter
|
||||||
import org.koitharu.kotatsu.core.ui.list.OnListItemClickListener
|
import org.koitharu.kotatsu.core.ui.list.OnListItemClickListener
|
||||||
import org.koitharu.kotatsu.core.util.ext.drawableStart
|
import org.koitharu.kotatsu.core.util.ext.drawableStart
|
||||||
import org.koitharu.kotatsu.core.util.ext.getThemeColor
|
import org.koitharu.kotatsu.core.util.ext.getThemeColorStateList
|
||||||
import org.koitharu.kotatsu.core.util.ext.textAndVisible
|
import org.koitharu.kotatsu.core.util.ext.textAndVisible
|
||||||
import org.koitharu.kotatsu.databinding.ItemChapterBinding
|
import org.koitharu.kotatsu.databinding.ItemChapterBinding
|
||||||
import org.koitharu.kotatsu.details.ui.model.ChapterListItem
|
import org.koitharu.kotatsu.details.ui.model.ChapterListItem
|
||||||
import com.google.android.material.R as materialR
|
import org.koitharu.kotatsu.list.ui.model.ListModel
|
||||||
|
import com.google.android.material.R as MR
|
||||||
|
|
||||||
fun chapterListItemAD(
|
fun chapterListItemAD(
|
||||||
clickListener: OnListItemClickListener<ChapterListItem>,
|
clickListener: OnListItemClickListener<ChapterListItem>,
|
||||||
) = adapterDelegateViewBinding<ChapterListItem, ChapterListItem, ItemChapterBinding>(
|
) = adapterDelegateViewBinding<ChapterListItem, ListModel, ItemChapterBinding>(
|
||||||
{ inflater, parent -> ItemChapterBinding.inflate(inflater, parent, false) },
|
{ inflater, parent -> ItemChapterBinding.inflate(inflater, parent, false) },
|
||||||
) {
|
) {
|
||||||
|
|
||||||
@@ -26,31 +28,38 @@ fun chapterListItemAD(
|
|||||||
bind { payloads ->
|
bind { payloads ->
|
||||||
if (payloads.isEmpty()) {
|
if (payloads.isEmpty()) {
|
||||||
binding.textViewTitle.text = item.chapter.name
|
binding.textViewTitle.text = item.chapter.name
|
||||||
binding.textViewNumber.text = item.chapter.number.toString()
|
binding.textViewDescription.textAndVisible = item.description
|
||||||
binding.textViewDescription.textAndVisible = item.description()
|
|
||||||
}
|
}
|
||||||
when {
|
when {
|
||||||
item.isCurrent -> {
|
item.isCurrent -> {
|
||||||
binding.textViewNumber.setBackgroundResource(R.drawable.bg_badge_primary)
|
binding.textViewTitle.drawableStart = ContextCompat.getDrawable(context, R.drawable.ic_current_chapter)
|
||||||
binding.textViewNumber.setTextColor(context.getThemeColor(materialR.attr.colorOnPrimary))
|
binding.textViewTitle.setTextColor(context.getThemeColorStateList(android.R.attr.textColorPrimary))
|
||||||
|
binding.textViewDescription.setTextColor(context.getThemeColorStateList(android.R.attr.textColorPrimary))
|
||||||
|
binding.textViewTitle.typeface = Typeface.DEFAULT_BOLD
|
||||||
|
binding.textViewDescription.typeface = Typeface.DEFAULT_BOLD
|
||||||
}
|
}
|
||||||
|
|
||||||
item.isUnread -> {
|
item.isUnread -> {
|
||||||
binding.textViewNumber.setBackgroundResource(R.drawable.bg_badge_default)
|
binding.textViewTitle.drawableStart = if (item.isNew) {
|
||||||
binding.textViewNumber.setTextColor(context.getThemeColor(materialR.attr.colorOnTertiary))
|
ContextCompat.getDrawable(context, R.drawable.ic_new)
|
||||||
|
} else {
|
||||||
|
null
|
||||||
|
}
|
||||||
|
binding.textViewTitle.setTextColor(context.getThemeColorStateList(android.R.attr.textColorPrimary))
|
||||||
|
binding.textViewDescription.setTextColor(context.getThemeColorStateList(MR.attr.colorOutline))
|
||||||
|
binding.textViewTitle.typeface = Typeface.DEFAULT
|
||||||
|
binding.textViewDescription.typeface = Typeface.DEFAULT
|
||||||
}
|
}
|
||||||
|
|
||||||
else -> {
|
else -> {
|
||||||
binding.textViewNumber.setBackgroundResource(R.drawable.bg_badge_outline)
|
binding.textViewTitle.drawableStart = null
|
||||||
binding.textViewNumber.setTextColor(context.getThemeColor(android.R.attr.textColorTertiary))
|
binding.textViewTitle.setTextColor(context.getThemeColorStateList(android.R.attr.textColorHint))
|
||||||
|
binding.textViewDescription.setTextColor(context.getThemeColorStateList(android.R.attr.textColorHint))
|
||||||
|
binding.textViewTitle.typeface = Typeface.DEFAULT
|
||||||
|
binding.textViewDescription.typeface = Typeface.DEFAULT
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
binding.imageViewBookmarked.isVisible = item.isBookmarked
|
binding.imageViewBookmarked.isVisible = item.isBookmarked
|
||||||
binding.imageViewDownloaded.isVisible = item.isDownloaded
|
binding.imageViewDownloaded.isVisible = item.isDownloaded
|
||||||
binding.textViewTitle.drawableStart = if (item.isNew) {
|
|
||||||
ContextCompat.getDrawable(context, R.drawable.ic_new)
|
|
||||||
} else {
|
|
||||||
null
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -5,22 +5,20 @@ import org.koitharu.kotatsu.core.ui.BaseListAdapter
|
|||||||
import org.koitharu.kotatsu.core.ui.list.OnListItemClickListener
|
import org.koitharu.kotatsu.core.ui.list.OnListItemClickListener
|
||||||
import org.koitharu.kotatsu.core.ui.list.fastscroll.FastScroller
|
import org.koitharu.kotatsu.core.ui.list.fastscroll.FastScroller
|
||||||
import org.koitharu.kotatsu.details.ui.model.ChapterListItem
|
import org.koitharu.kotatsu.details.ui.model.ChapterListItem
|
||||||
|
import org.koitharu.kotatsu.list.ui.adapter.ListItemType
|
||||||
|
import org.koitharu.kotatsu.list.ui.adapter.listHeaderAD
|
||||||
|
import org.koitharu.kotatsu.list.ui.model.ListModel
|
||||||
|
|
||||||
class ChaptersAdapter(
|
class ChaptersAdapter(
|
||||||
onItemClickListener: OnListItemClickListener<ChapterListItem>,
|
onItemClickListener: OnListItemClickListener<ChapterListItem>,
|
||||||
) : BaseListAdapter<ChapterListItem>(), FastScroller.SectionIndexer {
|
) : BaseListAdapter<ListModel>(), FastScroller.SectionIndexer {
|
||||||
|
|
||||||
init {
|
init {
|
||||||
setHasStableIds(true)
|
addDelegate(ListItemType.CHAPTER, chapterListItemAD(onItemClickListener))
|
||||||
delegatesManager.addDelegate(chapterListItemAD(onItemClickListener))
|
addDelegate(ListItemType.HEADER, listHeaderAD(null))
|
||||||
}
|
|
||||||
|
|
||||||
override fun getItemId(position: Int): Long {
|
|
||||||
return items[position].chapter.id
|
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun getSectionText(context: Context, position: Int): CharSequence? {
|
override fun getSectionText(context: Context, position: Int): CharSequence? {
|
||||||
val item = items.getOrNull(position) ?: return null
|
return findHeader(position)?.getText(context)
|
||||||
return item.chapter.number.toString()
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -9,7 +9,9 @@ import android.view.View
|
|||||||
import androidx.core.graphics.ColorUtils
|
import androidx.core.graphics.ColorUtils
|
||||||
import androidx.recyclerview.widget.RecyclerView
|
import androidx.recyclerview.widget.RecyclerView
|
||||||
import org.koitharu.kotatsu.core.ui.list.decor.AbstractSelectionItemDecoration
|
import org.koitharu.kotatsu.core.ui.list.decor.AbstractSelectionItemDecoration
|
||||||
|
import org.koitharu.kotatsu.core.util.ext.getItem
|
||||||
import org.koitharu.kotatsu.core.util.ext.getThemeColor
|
import org.koitharu.kotatsu.core.util.ext.getThemeColor
|
||||||
|
import org.koitharu.kotatsu.details.ui.model.ChapterListItem
|
||||||
import com.google.android.material.R as materialR
|
import com.google.android.material.R as materialR
|
||||||
|
|
||||||
class ChaptersSelectionDecoration(context: Context) : AbstractSelectionItemDecoration() {
|
class ChaptersSelectionDecoration(context: Context) : AbstractSelectionItemDecoration() {
|
||||||
@@ -25,6 +27,12 @@ class ChaptersSelectionDecoration(context: Context) : AbstractSelectionItemDecor
|
|||||||
paint.style = Paint.Style.FILL
|
paint.style = Paint.Style.FILL
|
||||||
}
|
}
|
||||||
|
|
||||||
|
override fun getItemId(parent: RecyclerView, child: View): Long {
|
||||||
|
val holder = parent.getChildViewHolder(child) ?: return RecyclerView.NO_ID
|
||||||
|
val item = holder.getItem(ChapterListItem::class.java) ?: return RecyclerView.NO_ID
|
||||||
|
return item.chapter.id
|
||||||
|
}
|
||||||
|
|
||||||
override fun onDrawBackground(
|
override fun onDrawBackground(
|
||||||
canvas: Canvas,
|
canvas: Canvas,
|
||||||
parent: RecyclerView,
|
parent: RecyclerView,
|
||||||
|
|||||||
@@ -1,6 +1,8 @@
|
|||||||
package org.koitharu.kotatsu.details.ui.model
|
package org.koitharu.kotatsu.details.ui.model
|
||||||
|
|
||||||
import android.text.format.DateUtils
|
import android.text.format.DateUtils
|
||||||
|
import org.jsoup.internal.StringUtil.StringJoiner
|
||||||
|
import org.koitharu.kotatsu.core.model.formatNumber
|
||||||
import org.koitharu.kotatsu.list.ui.model.ListModel
|
import org.koitharu.kotatsu.list.ui.model.ListModel
|
||||||
import org.koitharu.kotatsu.parsers.model.MangaChapter
|
import org.koitharu.kotatsu.parsers.model.MangaChapter
|
||||||
|
|
||||||
@@ -10,6 +12,14 @@ data class ChapterListItem(
|
|||||||
private val uploadDateMs: Long,
|
private val uploadDateMs: Long,
|
||||||
) : ListModel {
|
) : ListModel {
|
||||||
|
|
||||||
|
var description: String? = null
|
||||||
|
private set
|
||||||
|
get() {
|
||||||
|
if (field != null) return field
|
||||||
|
field = buildDescription()
|
||||||
|
return field
|
||||||
|
}
|
||||||
|
|
||||||
var uploadDate: CharSequence? = null
|
var uploadDate: CharSequence? = null
|
||||||
private set
|
private set
|
||||||
get() {
|
get() {
|
||||||
@@ -38,13 +48,20 @@ data class ChapterListItem(
|
|||||||
val isNew: Boolean
|
val isNew: Boolean
|
||||||
get() = hasFlag(FLAG_NEW)
|
get() = hasFlag(FLAG_NEW)
|
||||||
|
|
||||||
fun description(): CharSequence? {
|
private fun buildDescription(): String {
|
||||||
val scanlator = chapter.scanlator?.takeUnless { it.isBlank() }
|
val joiner = StringJoiner(" • ")
|
||||||
return when {
|
chapter.formatNumber()?.let {
|
||||||
uploadDate != null && scanlator != null -> "$uploadDate • $scanlator"
|
joiner.add("#").append(it)
|
||||||
scanlator != null -> scanlator
|
|
||||||
else -> uploadDate
|
|
||||||
}
|
}
|
||||||
|
uploadDate?.let { date ->
|
||||||
|
joiner.add(date.toString())
|
||||||
|
}
|
||||||
|
chapter.scanlator?.let { scanlator ->
|
||||||
|
if (scanlator.isNotBlank()) {
|
||||||
|
joiner.add(scanlator)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return joiner.complete()
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun hasFlag(flag: Int): Boolean {
|
private fun hasFlag(flag: Int): Boolean {
|
||||||
|
|||||||
@@ -7,6 +7,7 @@ data class MangaBranch(
|
|||||||
val name: String?,
|
val name: String?,
|
||||||
val count: Int,
|
val count: Int,
|
||||||
val isSelected: Boolean,
|
val isSelected: Boolean,
|
||||||
|
val isCurrent: Boolean,
|
||||||
) : ListModel {
|
) : ListModel {
|
||||||
|
|
||||||
override fun areItemsTheSame(other: ListModel): Boolean {
|
override fun areItemsTheSame(other: ListModel): Boolean {
|
||||||
|
|||||||
@@ -0,0 +1,32 @@
|
|||||||
|
package org.koitharu.kotatsu.details.ui.pager
|
||||||
|
|
||||||
|
import androidx.fragment.app.Fragment
|
||||||
|
import androidx.fragment.app.FragmentActivity
|
||||||
|
import androidx.viewpager2.adapter.FragmentStateAdapter
|
||||||
|
import com.google.android.material.tabs.TabLayout
|
||||||
|
import com.google.android.material.tabs.TabLayoutMediator
|
||||||
|
import org.koitharu.kotatsu.R
|
||||||
|
import org.koitharu.kotatsu.details.ui.pager.chapters.ChaptersFragment
|
||||||
|
import org.koitharu.kotatsu.details.ui.pager.pages.PagesFragment
|
||||||
|
|
||||||
|
class DetailsPagerAdapter(activity: FragmentActivity) : FragmentStateAdapter(activity),
|
||||||
|
TabLayoutMediator.TabConfigurationStrategy {
|
||||||
|
|
||||||
|
override fun getItemCount(): Int = 2
|
||||||
|
|
||||||
|
override fun createFragment(position: Int): Fragment = when (position) {
|
||||||
|
0 -> ChaptersFragment()
|
||||||
|
1 -> PagesFragment()
|
||||||
|
else -> throw IllegalArgumentException("Invalid position $position")
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun onConfigureTab(tab: TabLayout.Tab, position: Int) {
|
||||||
|
tab.setText(
|
||||||
|
when (position) {
|
||||||
|
0 -> R.string.chapters
|
||||||
|
1 -> R.string.pages
|
||||||
|
else -> 0
|
||||||
|
},
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,6 +1,5 @@
|
|||||||
package org.koitharu.kotatsu.details.ui
|
package org.koitharu.kotatsu.details.ui.pager.chapters
|
||||||
|
|
||||||
import android.os.Build
|
|
||||||
import android.os.Bundle
|
import android.os.Bundle
|
||||||
import android.view.LayoutInflater
|
import android.view.LayoutInflater
|
||||||
import android.view.Menu
|
import android.view.Menu
|
||||||
@@ -11,7 +10,11 @@ import androidx.appcompat.view.ActionMode
|
|||||||
import androidx.core.graphics.Insets
|
import androidx.core.graphics.Insets
|
||||||
import androidx.core.view.isVisible
|
import androidx.core.view.isVisible
|
||||||
import androidx.fragment.app.activityViewModels
|
import androidx.fragment.app.activityViewModels
|
||||||
|
import androidx.lifecycle.Lifecycle
|
||||||
import com.google.android.material.snackbar.Snackbar
|
import com.google.android.material.snackbar.Snackbar
|
||||||
|
import kotlinx.coroutines.Dispatchers
|
||||||
|
import kotlinx.coroutines.flow.flowOn
|
||||||
|
import kotlinx.coroutines.flow.map
|
||||||
import org.koitharu.kotatsu.R
|
import org.koitharu.kotatsu.R
|
||||||
import org.koitharu.kotatsu.core.ui.BaseFragment
|
import org.koitharu.kotatsu.core.ui.BaseFragment
|
||||||
import org.koitharu.kotatsu.core.ui.list.ListSelectionController
|
import org.koitharu.kotatsu.core.ui.list.ListSelectionController
|
||||||
@@ -20,9 +23,15 @@ import org.koitharu.kotatsu.core.util.RecyclerViewScrollCallback
|
|||||||
import org.koitharu.kotatsu.core.util.ext.observe
|
import org.koitharu.kotatsu.core.util.ext.observe
|
||||||
import org.koitharu.kotatsu.core.util.ext.observeEvent
|
import org.koitharu.kotatsu.core.util.ext.observeEvent
|
||||||
import org.koitharu.kotatsu.databinding.FragmentChaptersBinding
|
import org.koitharu.kotatsu.databinding.FragmentChaptersBinding
|
||||||
|
import org.koitharu.kotatsu.details.ui.ChaptersMenuProvider
|
||||||
|
import org.koitharu.kotatsu.details.ui.DetailsActivity
|
||||||
|
import org.koitharu.kotatsu.details.ui.DetailsViewModel
|
||||||
import org.koitharu.kotatsu.details.ui.adapter.ChaptersAdapter
|
import org.koitharu.kotatsu.details.ui.adapter.ChaptersAdapter
|
||||||
import org.koitharu.kotatsu.details.ui.adapter.ChaptersSelectionDecoration
|
import org.koitharu.kotatsu.details.ui.adapter.ChaptersSelectionDecoration
|
||||||
import org.koitharu.kotatsu.details.ui.model.ChapterListItem
|
import org.koitharu.kotatsu.details.ui.model.ChapterListItem
|
||||||
|
import org.koitharu.kotatsu.details.ui.withVolumeHeaders
|
||||||
|
import org.koitharu.kotatsu.list.ui.adapter.TypedListSpacingDecoration
|
||||||
|
import org.koitharu.kotatsu.list.ui.model.ListModel
|
||||||
import org.koitharu.kotatsu.local.ui.LocalChaptersRemoveService
|
import org.koitharu.kotatsu.local.ui.LocalChaptersRemoveService
|
||||||
import org.koitharu.kotatsu.parsers.model.MangaSource
|
import org.koitharu.kotatsu.parsers.model.MangaSource
|
||||||
import org.koitharu.kotatsu.reader.ui.ReaderActivity.IntentBuilder
|
import org.koitharu.kotatsu.reader.ui.ReaderActivity.IntentBuilder
|
||||||
@@ -54,21 +63,29 @@ class ChaptersFragment :
|
|||||||
callback = this,
|
callback = this,
|
||||||
)
|
)
|
||||||
with(binding.recyclerViewChapters) {
|
with(binding.recyclerViewChapters) {
|
||||||
|
addItemDecoration(TypedListSpacingDecoration(context, true))
|
||||||
checkNotNull(selectionController).attachToRecyclerView(this)
|
checkNotNull(selectionController).attachToRecyclerView(this)
|
||||||
setHasFixedSize(true)
|
setHasFixedSize(true)
|
||||||
|
isNestedScrollingEnabled = false
|
||||||
adapter = chaptersAdapter
|
adapter = chaptersAdapter
|
||||||
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
|
|
||||||
scrollIndicators = if (resources.getBoolean(R.bool.is_tablet)) 0 else View.SCROLL_INDICATOR_TOP
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
viewModel.isLoading.observe(viewLifecycleOwner, this::onLoadingStateChanged)
|
viewModel.isLoading.observe(viewLifecycleOwner, this::onLoadingStateChanged)
|
||||||
viewModel.chapters.observe(viewLifecycleOwner, this::onChaptersChanged)
|
viewModel.chapters
|
||||||
|
.map { it.withVolumeHeaders(requireContext()) }
|
||||||
|
.flowOn(Dispatchers.Default)
|
||||||
|
.observe(viewLifecycleOwner, this::onChaptersChanged)
|
||||||
viewModel.isChaptersEmpty.observe(viewLifecycleOwner) {
|
viewModel.isChaptersEmpty.observe(viewLifecycleOwner) {
|
||||||
binding.textViewHolder.isVisible = it
|
binding.textViewHolder.isVisible = it
|
||||||
}
|
}
|
||||||
viewModel.onSelectChapter.observeEvent(viewLifecycleOwner) {
|
viewModel.onSelectChapter.observeEvent(viewLifecycleOwner) {
|
||||||
selectionController?.onItemLongClick(it)
|
selectionController?.onItemLongClick(it)
|
||||||
}
|
}
|
||||||
|
val detailsActivity = activity as? DetailsActivity
|
||||||
|
if (detailsActivity != null) {
|
||||||
|
val menuProvider = ChaptersMenuProvider(viewModel, detailsActivity.bottomSheetMediator)
|
||||||
|
activity?.onBackPressedDispatcher?.addCallback(menuProvider)
|
||||||
|
detailsActivity.secondaryMenuHost.addMenuProvider(menuProvider, viewLifecycleOwner, Lifecycle.State.RESUMED)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun onDestroyView() {
|
override fun onDestroyView() {
|
||||||
@@ -77,6 +94,17 @@ class ChaptersFragment :
|
|||||||
super.onDestroyView()
|
super.onDestroyView()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
override fun onPause() {
|
||||||
|
// required for BottomSheetBehavior
|
||||||
|
requireViewBinding().recyclerViewChapters.isNestedScrollingEnabled = false
|
||||||
|
super.onPause()
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun onResume() {
|
||||||
|
requireViewBinding().recyclerViewChapters.isNestedScrollingEnabled = true
|
||||||
|
super.onResume()
|
||||||
|
}
|
||||||
|
|
||||||
override fun onItemClick(item: ChapterListItem, view: View) {
|
override fun onItemClick(item: ChapterListItem, view: View) {
|
||||||
if (selectionController?.onItemClick(item.chapter.id) == true) {
|
if (selectionController?.onItemClick(item.chapter.id) == true) {
|
||||||
return
|
return
|
||||||
@@ -126,6 +154,9 @@ class ChaptersFragment :
|
|||||||
val buffer = HashSet<Long>()
|
val buffer = HashSet<Long>()
|
||||||
var isAdding = false
|
var isAdding = false
|
||||||
for (x in items) {
|
for (x in items) {
|
||||||
|
if (x !is ChapterListItem) {
|
||||||
|
continue
|
||||||
|
}
|
||||||
if (x.chapter.id in ids) {
|
if (x.chapter.id in ids) {
|
||||||
isAdding = true
|
isAdding = true
|
||||||
if (buffer.isNotEmpty()) {
|
if (buffer.isNotEmpty()) {
|
||||||
@@ -141,7 +172,13 @@ class ChaptersFragment :
|
|||||||
}
|
}
|
||||||
|
|
||||||
R.id.action_select_all -> {
|
R.id.action_select_all -> {
|
||||||
val ids = chaptersAdapter?.items?.map { it.chapter.id } ?: return false
|
val ids = chaptersAdapter?.items?.mapNotNull {
|
||||||
|
if (it is ChapterListItem) {
|
||||||
|
it.chapter.id
|
||||||
|
} else {
|
||||||
|
null
|
||||||
|
}
|
||||||
|
} ?: return false
|
||||||
controller.addAll(ids)
|
controller.addAll(ids)
|
||||||
true
|
true
|
||||||
}
|
}
|
||||||
@@ -165,7 +202,15 @@ class ChaptersFragment :
|
|||||||
override fun onPrepareActionMode(controller: ListSelectionController, mode: ActionMode, menu: Menu): Boolean {
|
override fun onPrepareActionMode(controller: ListSelectionController, mode: ActionMode, menu: Menu): Boolean {
|
||||||
val selectedIds = selectionController?.peekCheckedIds() ?: return false
|
val selectedIds = selectionController?.peekCheckedIds() ?: return false
|
||||||
val allItems = chaptersAdapter?.items.orEmpty()
|
val allItems = chaptersAdapter?.items.orEmpty()
|
||||||
val items = allItems.withIndex().filter { (_, x) -> x.chapter.id in selectedIds }
|
val items = allItems.withIndex().mapNotNull<IndexedValue<ListModel>, IndexedValue<ChapterListItem>> { x ->
|
||||||
|
val value = x.value
|
||||||
|
@Suppress("UNCHECKED_CAST")
|
||||||
|
if (value is ChapterListItem && value.chapter.id in selectedIds) {
|
||||||
|
x as IndexedValue<ChapterListItem>
|
||||||
|
} else {
|
||||||
|
null
|
||||||
|
}
|
||||||
|
}
|
||||||
var canSave = true
|
var canSave = true
|
||||||
var canDelete = true
|
var canDelete = true
|
||||||
items.forEach { (_, x) ->
|
items.forEach { (_, x) ->
|
||||||
@@ -189,15 +234,15 @@ class ChaptersFragment :
|
|||||||
}
|
}
|
||||||
|
|
||||||
override fun onSelectionChanged(controller: ListSelectionController, count: Int) {
|
override fun onSelectionChanged(controller: ListSelectionController, count: Int) {
|
||||||
requireViewBinding().recyclerViewChapters.invalidateItemDecorations()
|
viewBinding?.recyclerViewChapters?.invalidateItemDecorations()
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun onWindowInsetsChanged(insets: Insets) = Unit
|
override fun onWindowInsetsChanged(insets: Insets) = Unit
|
||||||
|
|
||||||
private fun onChaptersChanged(list: List<ChapterListItem>) {
|
private fun onChaptersChanged(list: List<ListModel>) {
|
||||||
val adapter = chaptersAdapter ?: return
|
val adapter = chaptersAdapter ?: return
|
||||||
if (adapter.itemCount == 0) {
|
if (adapter.itemCount == 0) {
|
||||||
val position = list.indexOfFirst { it.isCurrent } - 1
|
val position = list.indexOfFirst { it is ChapterListItem && it.isCurrent } - 1
|
||||||
if (position > 0) {
|
if (position > 0) {
|
||||||
val offset = (resources.getDimensionPixelSize(R.dimen.chapter_list_item_height) * 0.6).roundToInt()
|
val offset = (resources.getDimensionPixelSize(R.dimen.chapter_list_item_height) * 0.6).roundToInt()
|
||||||
adapter.setItems(
|
adapter.setItems(
|
||||||
@@ -0,0 +1,204 @@
|
|||||||
|
package org.koitharu.kotatsu.details.ui.pager.pages
|
||||||
|
|
||||||
|
import android.os.Bundle
|
||||||
|
import android.view.LayoutInflater
|
||||||
|
import android.view.View
|
||||||
|
import android.view.ViewGroup
|
||||||
|
import androidx.core.graphics.Insets
|
||||||
|
import androidx.core.view.isInvisible
|
||||||
|
import androidx.core.view.isVisible
|
||||||
|
import androidx.fragment.app.activityViewModels
|
||||||
|
import androidx.fragment.app.viewModels
|
||||||
|
import androidx.recyclerview.widget.GridLayoutManager
|
||||||
|
import androidx.recyclerview.widget.RecyclerView
|
||||||
|
import coil.ImageLoader
|
||||||
|
import dagger.hilt.android.AndroidEntryPoint
|
||||||
|
import kotlinx.coroutines.Dispatchers
|
||||||
|
import kotlinx.coroutines.flow.combine
|
||||||
|
import kotlinx.coroutines.flow.flowOn
|
||||||
|
import org.koitharu.kotatsu.R
|
||||||
|
import org.koitharu.kotatsu.core.exceptions.resolve.SnackbarErrorObserver
|
||||||
|
import org.koitharu.kotatsu.core.prefs.AppSettings
|
||||||
|
import org.koitharu.kotatsu.core.ui.BaseFragment
|
||||||
|
import org.koitharu.kotatsu.core.ui.list.BoundsScrollListener
|
||||||
|
import org.koitharu.kotatsu.core.ui.list.OnListItemClickListener
|
||||||
|
import org.koitharu.kotatsu.core.util.RecyclerViewScrollCallback
|
||||||
|
import org.koitharu.kotatsu.core.util.ext.observe
|
||||||
|
import org.koitharu.kotatsu.core.util.ext.observeEvent
|
||||||
|
import org.koitharu.kotatsu.core.util.ext.showOrHide
|
||||||
|
import org.koitharu.kotatsu.databinding.FragmentPagesBinding
|
||||||
|
import org.koitharu.kotatsu.details.ui.DetailsViewModel
|
||||||
|
import org.koitharu.kotatsu.list.ui.MangaListSpanResolver
|
||||||
|
import org.koitharu.kotatsu.list.ui.adapter.ListItemType
|
||||||
|
import org.koitharu.kotatsu.list.ui.adapter.TypedListSpacingDecoration
|
||||||
|
import org.koitharu.kotatsu.list.ui.model.ListModel
|
||||||
|
import org.koitharu.kotatsu.reader.ui.ReaderActivity.IntentBuilder
|
||||||
|
import org.koitharu.kotatsu.reader.ui.ReaderState
|
||||||
|
import org.koitharu.kotatsu.reader.ui.thumbnails.PageThumbnail
|
||||||
|
import org.koitharu.kotatsu.reader.ui.thumbnails.adapter.PageThumbnailAdapter
|
||||||
|
import javax.inject.Inject
|
||||||
|
import kotlin.math.roundToInt
|
||||||
|
|
||||||
|
@AndroidEntryPoint
|
||||||
|
class PagesFragment :
|
||||||
|
BaseFragment<FragmentPagesBinding>(),
|
||||||
|
OnListItemClickListener<PageThumbnail> {
|
||||||
|
|
||||||
|
private val detailsViewModel by activityViewModels<DetailsViewModel>()
|
||||||
|
private val viewModel by viewModels<PagesViewModel>()
|
||||||
|
|
||||||
|
@Inject
|
||||||
|
lateinit var coil: ImageLoader
|
||||||
|
|
||||||
|
@Inject
|
||||||
|
lateinit var settings: AppSettings
|
||||||
|
|
||||||
|
private var thumbnailsAdapter: PageThumbnailAdapter? = null
|
||||||
|
private var spanResolver: MangaListSpanResolver? = null
|
||||||
|
private var scrollListener: ScrollListener? = null
|
||||||
|
|
||||||
|
private val spanSizeLookup = SpanSizeLookup()
|
||||||
|
|
||||||
|
override fun onCreate(savedInstanceState: Bundle?) {
|
||||||
|
super.onCreate(savedInstanceState)
|
||||||
|
combine(
|
||||||
|
detailsViewModel.details,
|
||||||
|
detailsViewModel.history,
|
||||||
|
detailsViewModel.selectedBranch,
|
||||||
|
) { details, history, branch ->
|
||||||
|
if (details != null && (details.isLoaded || details.chapters.isNotEmpty())) {
|
||||||
|
PagesViewModel.State(details.filterChapters(branch), history, branch)
|
||||||
|
} else {
|
||||||
|
null
|
||||||
|
}
|
||||||
|
}.flowOn(Dispatchers.Default)
|
||||||
|
.observe(this, viewModel::updateState)
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun onCreateViewBinding(inflater: LayoutInflater, container: ViewGroup?): FragmentPagesBinding {
|
||||||
|
return FragmentPagesBinding.inflate(inflater, container, false)
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun onViewBindingCreated(binding: FragmentPagesBinding, savedInstanceState: Bundle?) {
|
||||||
|
super.onViewBindingCreated(binding, savedInstanceState)
|
||||||
|
spanResolver = MangaListSpanResolver(binding.root.resources)
|
||||||
|
thumbnailsAdapter = PageThumbnailAdapter(
|
||||||
|
coil = coil,
|
||||||
|
lifecycleOwner = viewLifecycleOwner,
|
||||||
|
clickListener = this@PagesFragment,
|
||||||
|
)
|
||||||
|
with(binding.recyclerView) {
|
||||||
|
addItemDecoration(TypedListSpacingDecoration(context, false))
|
||||||
|
adapter = thumbnailsAdapter
|
||||||
|
setHasFixedSize(true)
|
||||||
|
isNestedScrollingEnabled = false
|
||||||
|
addOnLayoutChangeListener(spanResolver)
|
||||||
|
spanResolver?.setGridSize(settings.gridSize / 100f, this)
|
||||||
|
addOnScrollListener(ScrollListener().also { scrollListener = it })
|
||||||
|
(layoutManager as GridLayoutManager).let {
|
||||||
|
it.spanSizeLookup = spanSizeLookup
|
||||||
|
it.spanCount = checkNotNull(spanResolver).spanCount
|
||||||
|
}
|
||||||
|
}
|
||||||
|
detailsViewModel.isChaptersEmpty.observe(viewLifecycleOwner, ::onNoChaptersChanged)
|
||||||
|
viewModel.thumbnails.observe(viewLifecycleOwner, ::onThumbnailsChanged)
|
||||||
|
viewModel.onError.observeEvent(viewLifecycleOwner, SnackbarErrorObserver(binding.recyclerView, this))
|
||||||
|
viewModel.isLoading.observe(viewLifecycleOwner) { binding.progressBar.showOrHide(it) }
|
||||||
|
viewModel.isLoadingUp.observe(viewLifecycleOwner) { binding.progressBarTop.showOrHide(it) }
|
||||||
|
viewModel.isLoadingDown.observe(viewLifecycleOwner) { binding.progressBarBottom.showOrHide(it) }
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun onDestroyView() {
|
||||||
|
spanResolver = null
|
||||||
|
scrollListener = null
|
||||||
|
thumbnailsAdapter = null
|
||||||
|
spanSizeLookup.invalidateCache()
|
||||||
|
super.onDestroyView()
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun onPause() {
|
||||||
|
// required for BottomSheetBehavior
|
||||||
|
requireViewBinding().recyclerView.isNestedScrollingEnabled = false
|
||||||
|
super.onPause()
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun onResume() {
|
||||||
|
requireViewBinding().recyclerView.isNestedScrollingEnabled = true
|
||||||
|
super.onResume()
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun onWindowInsetsChanged(insets: Insets) = Unit
|
||||||
|
|
||||||
|
override fun onItemClick(item: PageThumbnail, view: View) {
|
||||||
|
val manga = detailsViewModel.manga.value ?: return
|
||||||
|
val state = ReaderState(item.page.chapterId, item.page.index, 0)
|
||||||
|
val intent = IntentBuilder(view.context).manga(manga).state(state).build()
|
||||||
|
startActivity(intent)
|
||||||
|
}
|
||||||
|
|
||||||
|
private suspend fun onThumbnailsChanged(list: List<ListModel>) {
|
||||||
|
val adapter = thumbnailsAdapter ?: return
|
||||||
|
if (adapter.itemCount == 0) {
|
||||||
|
var position = list.indexOfFirst { it is PageThumbnail && it.isCurrent }
|
||||||
|
if (position > 0) {
|
||||||
|
val spanCount = spanResolver?.spanCount ?: 0
|
||||||
|
val offset = if (position > spanCount + 1) {
|
||||||
|
(resources.getDimensionPixelSize(R.dimen.manga_list_details_item_height) * 0.6).roundToInt()
|
||||||
|
} else {
|
||||||
|
position = 0
|
||||||
|
0
|
||||||
|
}
|
||||||
|
val scrollCallback = RecyclerViewScrollCallback(requireViewBinding().recyclerView, position, offset)
|
||||||
|
adapter.emit(list)
|
||||||
|
scrollCallback.run()
|
||||||
|
} else {
|
||||||
|
adapter.emit(list)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
adapter.emit(list)
|
||||||
|
}
|
||||||
|
spanSizeLookup.invalidateCache()
|
||||||
|
viewBinding?.recyclerView?.let {
|
||||||
|
scrollListener?.postInvalidate(it)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun onNoChaptersChanged(isNoChapters: Boolean) {
|
||||||
|
with(viewBinding ?: return) {
|
||||||
|
textViewHolder.isVisible = isNoChapters
|
||||||
|
recyclerView.isInvisible = isNoChapters
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private inner class ScrollListener : BoundsScrollListener(3, 3) {
|
||||||
|
|
||||||
|
override fun onScrolledToStart(recyclerView: RecyclerView) {
|
||||||
|
viewModel.loadPrevChapter()
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun onScrolledToEnd(recyclerView: RecyclerView) {
|
||||||
|
viewModel.loadNextChapter()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private inner class SpanSizeLookup : GridLayoutManager.SpanSizeLookup() {
|
||||||
|
|
||||||
|
init {
|
||||||
|
isSpanIndexCacheEnabled = true
|
||||||
|
isSpanGroupIndexCacheEnabled = true
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun getSpanSize(position: Int): Int {
|
||||||
|
val total = (viewBinding?.recyclerView?.layoutManager as? GridLayoutManager)?.spanCount ?: return 1
|
||||||
|
return when (thumbnailsAdapter?.getItemViewType(position)) {
|
||||||
|
ListItemType.PAGE_THUMB.ordinal -> 1
|
||||||
|
else -> total
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fun invalidateCache() {
|
||||||
|
invalidateSpanGroupIndexCache()
|
||||||
|
invalidateSpanIndexCache()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,115 @@
|
|||||||
|
package org.koitharu.kotatsu.details.ui.pager.pages
|
||||||
|
|
||||||
|
import dagger.hilt.android.lifecycle.HiltViewModel
|
||||||
|
import kotlinx.coroutines.Dispatchers
|
||||||
|
import kotlinx.coroutines.Job
|
||||||
|
import kotlinx.coroutines.flow.MutableStateFlow
|
||||||
|
import kotlinx.coroutines.flow.collectLatest
|
||||||
|
import org.koitharu.kotatsu.core.model.MangaHistory
|
||||||
|
import org.koitharu.kotatsu.core.ui.BaseViewModel
|
||||||
|
import org.koitharu.kotatsu.core.util.ext.firstNotNull
|
||||||
|
import org.koitharu.kotatsu.details.data.MangaDetails
|
||||||
|
import org.koitharu.kotatsu.list.ui.model.ListHeader
|
||||||
|
import org.koitharu.kotatsu.list.ui.model.ListModel
|
||||||
|
import org.koitharu.kotatsu.reader.domain.ChaptersLoader
|
||||||
|
import org.koitharu.kotatsu.reader.ui.thumbnails.PageThumbnail
|
||||||
|
import javax.inject.Inject
|
||||||
|
|
||||||
|
@HiltViewModel
|
||||||
|
class PagesViewModel @Inject constructor(
|
||||||
|
private val chaptersLoader: ChaptersLoader,
|
||||||
|
) : BaseViewModel() {
|
||||||
|
|
||||||
|
private var loadingJob: Job? = null
|
||||||
|
private var loadingPrevJob: Job? = null
|
||||||
|
private var loadingNextJob: Job? = null
|
||||||
|
|
||||||
|
private val state = MutableStateFlow<State?>(null)
|
||||||
|
val thumbnails = MutableStateFlow<List<ListModel>>(emptyList())
|
||||||
|
val isLoadingUp = MutableStateFlow(false)
|
||||||
|
val isLoadingDown = MutableStateFlow(false)
|
||||||
|
|
||||||
|
init {
|
||||||
|
loadingJob = launchLoadingJob(Dispatchers.Default) {
|
||||||
|
val firstState = state.firstNotNull()
|
||||||
|
doInit(firstState)
|
||||||
|
launchJob(Dispatchers.Default) {
|
||||||
|
state.collectLatest {
|
||||||
|
if (it != null) {
|
||||||
|
doInit(it)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fun updateState(newState: State?) {
|
||||||
|
if (newState != null) {
|
||||||
|
state.value = newState
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fun loadPrevChapter() {
|
||||||
|
if (loadingJob?.isActive == true || loadingPrevJob?.isActive == true) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
loadingPrevJob = loadPrevNextChapter(isNext = false)
|
||||||
|
}
|
||||||
|
|
||||||
|
fun loadNextChapter() {
|
||||||
|
if (loadingJob?.isActive == true || loadingNextJob?.isActive == true) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
loadingNextJob = loadPrevNextChapter(isNext = true)
|
||||||
|
}
|
||||||
|
|
||||||
|
private suspend fun doInit(state: State) {
|
||||||
|
chaptersLoader.init(state.details)
|
||||||
|
val initialChapterId = state.history?.chapterId ?: state.details.allChapters.firstOrNull()?.id ?: return
|
||||||
|
if (!chaptersLoader.hasPages(initialChapterId)) {
|
||||||
|
chaptersLoader.loadSingleChapter(initialChapterId)
|
||||||
|
}
|
||||||
|
updateList(state.history)
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun loadPrevNextChapter(isNext: Boolean): Job = launchJob(Dispatchers.Default) {
|
||||||
|
val indicator = if (isNext) isLoadingDown else isLoadingUp
|
||||||
|
indicator.value = true
|
||||||
|
try {
|
||||||
|
val currentState = state.firstNotNull()
|
||||||
|
val currentId = (if (isNext) chaptersLoader.last() else chaptersLoader.first()).chapterId
|
||||||
|
chaptersLoader.loadPrevNextChapter(currentState.details, currentId, isNext)
|
||||||
|
updateList(currentState.history)
|
||||||
|
} finally {
|
||||||
|
indicator.value = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun updateList(history: MangaHistory?) {
|
||||||
|
val snapshot = chaptersLoader.snapshot()
|
||||||
|
val pages = buildList(snapshot.size + chaptersLoader.size + 2) {
|
||||||
|
var previousChapterId = 0L
|
||||||
|
for (page in snapshot) {
|
||||||
|
if (page.chapterId != previousChapterId) {
|
||||||
|
chaptersLoader.peekChapter(page.chapterId)?.let {
|
||||||
|
add(ListHeader(it.name))
|
||||||
|
}
|
||||||
|
previousChapterId = page.chapterId
|
||||||
|
}
|
||||||
|
this += PageThumbnail(
|
||||||
|
isCurrent = history?.let {
|
||||||
|
page.chapterId == it.chapterId && page.index == it.page
|
||||||
|
} ?: false,
|
||||||
|
page = page,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
thumbnails.value = pages
|
||||||
|
}
|
||||||
|
|
||||||
|
data class State(
|
||||||
|
val details: MangaDetails,
|
||||||
|
val history: MangaHistory?,
|
||||||
|
val branch: String?
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -4,7 +4,7 @@ import androidx.work.Data
|
|||||||
import org.koitharu.kotatsu.history.data.PROGRESS_NONE
|
import org.koitharu.kotatsu.history.data.PROGRESS_NONE
|
||||||
import org.koitharu.kotatsu.local.domain.model.LocalManga
|
import org.koitharu.kotatsu.local.domain.model.LocalManga
|
||||||
import org.koitharu.kotatsu.parsers.model.Manga
|
import org.koitharu.kotatsu.parsers.model.Manga
|
||||||
import java.util.Date
|
import java.time.Instant
|
||||||
|
|
||||||
data class DownloadState(
|
data class DownloadState(
|
||||||
val manga: Manga,
|
val manga: Manga,
|
||||||
@@ -72,7 +72,7 @@ data class DownloadState(
|
|||||||
|
|
||||||
fun getEta(data: Data): Long = data.getLong(DATA_ETA, -1L)
|
fun getEta(data: Data): Long = data.getLong(DATA_ETA, -1L)
|
||||||
|
|
||||||
fun getTimestamp(data: Data): Date = Date(data.getLong(DATA_TIMESTAMP, 0L))
|
fun getTimestamp(data: Data): Instant = Instant.ofEpochMilli(data.getLong(DATA_TIMESTAMP, 0L))
|
||||||
|
|
||||||
fun getDownloadedChapters(data: Data): Int = data.getInt(DATA_CHAPTERS, 0)
|
fun getDownloadedChapters(data: Data): Int = data.getInt(DATA_CHAPTERS, 0)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -8,7 +8,7 @@ import org.koitharu.kotatsu.download.ui.list.chapters.DownloadChapter
|
|||||||
import org.koitharu.kotatsu.list.ui.ListModelDiffCallback
|
import org.koitharu.kotatsu.list.ui.ListModelDiffCallback
|
||||||
import org.koitharu.kotatsu.list.ui.model.ListModel
|
import org.koitharu.kotatsu.list.ui.model.ListModel
|
||||||
import org.koitharu.kotatsu.parsers.model.Manga
|
import org.koitharu.kotatsu.parsers.model.Manga
|
||||||
import java.util.Date
|
import java.time.Instant
|
||||||
import java.util.UUID
|
import java.util.UUID
|
||||||
|
|
||||||
data class DownloadItemModel(
|
data class DownloadItemModel(
|
||||||
@@ -21,7 +21,7 @@ data class DownloadItemModel(
|
|||||||
val max: Int,
|
val max: Int,
|
||||||
val progress: Int,
|
val progress: Int,
|
||||||
val eta: Long,
|
val eta: Long,
|
||||||
val timestamp: Date,
|
val timestamp: Instant,
|
||||||
val chaptersDownloaded: Int,
|
val chaptersDownloaded: Int,
|
||||||
val isExpanded: Boolean,
|
val isExpanded: Boolean,
|
||||||
val chapters: StateFlow<List<DownloadChapter>?>,
|
val chapters: StateFlow<List<DownloadChapter>?>,
|
||||||
|
|||||||
@@ -7,6 +7,7 @@ import android.view.MenuItem
|
|||||||
import androidx.core.view.MenuProvider
|
import androidx.core.view.MenuProvider
|
||||||
import com.google.android.material.dialog.MaterialAlertDialogBuilder
|
import com.google.android.material.dialog.MaterialAlertDialogBuilder
|
||||||
import org.koitharu.kotatsu.R
|
import org.koitharu.kotatsu.R
|
||||||
|
import org.koitharu.kotatsu.core.util.ext.DIALOG_THEME_CENTERED
|
||||||
import org.koitharu.kotatsu.settings.SettingsActivity
|
import org.koitharu.kotatsu.settings.SettingsActivity
|
||||||
|
|
||||||
class DownloadsMenuProvider(
|
class DownloadsMenuProvider(
|
||||||
@@ -41,10 +42,8 @@ class DownloadsMenuProvider(
|
|||||||
}
|
}
|
||||||
|
|
||||||
private fun confirmCancelAll() {
|
private fun confirmCancelAll() {
|
||||||
MaterialAlertDialogBuilder(
|
MaterialAlertDialogBuilder(context, DIALOG_THEME_CENTERED)
|
||||||
context,
|
.setTitle(R.string.cancel_all)
|
||||||
com.google.android.material.R.style.ThemeOverlay_Material3_MaterialAlertDialog_Centered,
|
|
||||||
).setTitle(R.string.cancel_all)
|
|
||||||
.setMessage(R.string.cancel_all_downloads_confirm)
|
.setMessage(R.string.cancel_all_downloads_confirm)
|
||||||
.setIcon(R.drawable.ic_cancel_multiple)
|
.setIcon(R.drawable.ic_cancel_multiple)
|
||||||
.setNegativeButton(android.R.string.cancel, null)
|
.setNegativeButton(android.R.string.cancel, null)
|
||||||
@@ -54,10 +53,8 @@ class DownloadsMenuProvider(
|
|||||||
}
|
}
|
||||||
|
|
||||||
private fun confirmRemoveCompleted() {
|
private fun confirmRemoveCompleted() {
|
||||||
MaterialAlertDialogBuilder(
|
MaterialAlertDialogBuilder(context, DIALOG_THEME_CENTERED)
|
||||||
context,
|
.setTitle(R.string.remove_completed)
|
||||||
com.google.android.material.R.style.ThemeOverlay_Material3_MaterialAlertDialog_Centered,
|
|
||||||
).setTitle(R.string.remove_completed)
|
|
||||||
.setMessage(R.string.remove_completed_downloads_confirm)
|
.setMessage(R.string.remove_completed_downloads_confirm)
|
||||||
.setIcon(R.drawable.ic_clear_all)
|
.setIcon(R.drawable.ic_clear_all)
|
||||||
.setNegativeButton(android.R.string.cancel, null)
|
.setNegativeButton(android.R.string.cancel, null)
|
||||||
|
|||||||
@@ -21,6 +21,7 @@ import kotlinx.coroutines.plus
|
|||||||
import kotlinx.coroutines.sync.Mutex
|
import kotlinx.coroutines.sync.Mutex
|
||||||
import kotlinx.coroutines.sync.withLock
|
import kotlinx.coroutines.sync.withLock
|
||||||
import org.koitharu.kotatsu.R
|
import org.koitharu.kotatsu.R
|
||||||
|
import org.koitharu.kotatsu.core.model.formatNumber
|
||||||
import org.koitharu.kotatsu.core.parser.MangaDataRepository
|
import org.koitharu.kotatsu.core.parser.MangaDataRepository
|
||||||
import org.koitharu.kotatsu.core.parser.MangaRepository
|
import org.koitharu.kotatsu.core.parser.MangaRepository
|
||||||
import org.koitharu.kotatsu.core.parser.RemoteMangaRepository
|
import org.koitharu.kotatsu.core.parser.RemoteMangaRepository
|
||||||
@@ -28,8 +29,8 @@ import org.koitharu.kotatsu.core.ui.BaseViewModel
|
|||||||
import org.koitharu.kotatsu.core.ui.model.DateTimeAgo
|
import org.koitharu.kotatsu.core.ui.model.DateTimeAgo
|
||||||
import org.koitharu.kotatsu.core.ui.util.ReversibleAction
|
import org.koitharu.kotatsu.core.ui.util.ReversibleAction
|
||||||
import org.koitharu.kotatsu.core.util.ext.MutableEventFlow
|
import org.koitharu.kotatsu.core.util.ext.MutableEventFlow
|
||||||
|
import org.koitharu.kotatsu.core.util.ext.calculateTimeAgo
|
||||||
import org.koitharu.kotatsu.core.util.ext.call
|
import org.koitharu.kotatsu.core.util.ext.call
|
||||||
import org.koitharu.kotatsu.core.util.ext.daysDiff
|
|
||||||
import org.koitharu.kotatsu.core.util.ext.isEmpty
|
import org.koitharu.kotatsu.core.util.ext.isEmpty
|
||||||
import org.koitharu.kotatsu.download.domain.DownloadState
|
import org.koitharu.kotatsu.download.domain.DownloadState
|
||||||
import org.koitharu.kotatsu.download.ui.list.chapters.DownloadChapter
|
import org.koitharu.kotatsu.download.ui.list.chapters.DownloadChapter
|
||||||
@@ -44,10 +45,8 @@ import org.koitharu.kotatsu.local.domain.model.LocalManga
|
|||||||
import org.koitharu.kotatsu.parsers.model.Manga
|
import org.koitharu.kotatsu.parsers.model.Manga
|
||||||
import org.koitharu.kotatsu.parsers.util.mapToSet
|
import org.koitharu.kotatsu.parsers.util.mapToSet
|
||||||
import org.koitharu.kotatsu.parsers.util.runCatchingCancellable
|
import org.koitharu.kotatsu.parsers.util.runCatchingCancellable
|
||||||
import java.util.Date
|
|
||||||
import java.util.LinkedList
|
import java.util.LinkedList
|
||||||
import java.util.UUID
|
import java.util.UUID
|
||||||
import java.util.concurrent.TimeUnit
|
|
||||||
import javax.inject.Inject
|
import javax.inject.Inject
|
||||||
|
|
||||||
@HiltViewModel
|
@HiltViewModel
|
||||||
@@ -225,7 +224,7 @@ class DownloadsViewModel @Inject constructor(
|
|||||||
WorkInfo.State.ENQUEUED -> queued += item
|
WorkInfo.State.ENQUEUED -> queued += item
|
||||||
|
|
||||||
else -> {
|
else -> {
|
||||||
val date = timeAgo(item.timestamp)
|
val date = calculateTimeAgo(item.timestamp)
|
||||||
if (prevDate != date) {
|
if (prevDate != date) {
|
||||||
destination += ListHeader(date)
|
destination += ListHeader(date)
|
||||||
}
|
}
|
||||||
@@ -275,19 +274,6 @@ class DownloadsViewModel @Inject constructor(
|
|||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun timeAgo(date: Date): DateTimeAgo {
|
|
||||||
val diff = (System.currentTimeMillis() - date.time).coerceAtLeast(0L)
|
|
||||||
val diffMinutes = TimeUnit.MILLISECONDS.toMinutes(diff).toInt()
|
|
||||||
val diffDays = -date.daysDiff(System.currentTimeMillis())
|
|
||||||
return when {
|
|
||||||
diffMinutes < 3 -> DateTimeAgo.JustNow
|
|
||||||
diffDays < 1 -> DateTimeAgo.Today
|
|
||||||
diffDays == 1 -> DateTimeAgo.Yesterday
|
|
||||||
diffDays < 6 -> DateTimeAgo.DaysAgo(diffDays)
|
|
||||||
else -> DateTimeAgo.Absolute(date)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private fun emptyStateList() = listOf(
|
private fun emptyStateList() = listOf(
|
||||||
EmptyState(
|
EmptyState(
|
||||||
icon = R.drawable.ic_empty_common,
|
icon = R.drawable.ic_empty_common,
|
||||||
@@ -321,7 +307,7 @@ class DownloadsViewModel @Inject constructor(
|
|||||||
return chapters.mapNotNullTo(ArrayList(size)) {
|
return chapters.mapNotNullTo(ArrayList(size)) {
|
||||||
if (chapterIds == null || it.id in chapterIds) {
|
if (chapterIds == null || it.id in chapterIds) {
|
||||||
DownloadChapter(
|
DownloadChapter(
|
||||||
number = it.number,
|
number = it.formatNumber(),
|
||||||
name = it.name,
|
name = it.name,
|
||||||
isDownloaded = it.id in localChapters,
|
isDownloaded = it.id in localChapters,
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -4,7 +4,7 @@ import org.koitharu.kotatsu.list.ui.ListModelDiffCallback
|
|||||||
import org.koitharu.kotatsu.list.ui.model.ListModel
|
import org.koitharu.kotatsu.list.ui.model.ListModel
|
||||||
|
|
||||||
data class DownloadChapter(
|
data class DownloadChapter(
|
||||||
val number: Int,
|
val number: String?,
|
||||||
val name: String,
|
val name: String,
|
||||||
val isDownloaded: Boolean,
|
val isDownloaded: Boolean,
|
||||||
) : ListModel {
|
) : ListModel {
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
package org.koitharu.kotatsu.download.ui.worker
|
package org.koitharu.kotatsu.download.ui.worker
|
||||||
|
|
||||||
|
import androidx.collection.MutableObjectLongMap
|
||||||
import kotlinx.coroutines.delay
|
import kotlinx.coroutines.delay
|
||||||
import org.koitharu.kotatsu.core.parser.MangaRepository
|
import org.koitharu.kotatsu.core.parser.MangaRepository
|
||||||
import org.koitharu.kotatsu.core.parser.RemoteMangaRepository
|
import org.koitharu.kotatsu.core.parser.RemoteMangaRepository
|
||||||
@@ -9,7 +10,7 @@ class DownloadSlowdownDispatcher(
|
|||||||
private val mangaRepositoryFactory: MangaRepository.Factory,
|
private val mangaRepositoryFactory: MangaRepository.Factory,
|
||||||
private val defaultDelay: Long,
|
private val defaultDelay: Long,
|
||||||
) {
|
) {
|
||||||
private val timeMap = HashMap<MangaSource, Long>()
|
private val timeMap = MutableObjectLongMap<MangaSource>()
|
||||||
|
|
||||||
suspend fun delay(source: MangaSource) {
|
suspend fun delay(source: MangaSource) {
|
||||||
val repo = mangaRepositoryFactory.create(source) as? RemoteMangaRepository ?: return
|
val repo = mangaRepositoryFactory.create(source) as? RemoteMangaRepository ?: return
|
||||||
@@ -17,7 +18,7 @@ class DownloadSlowdownDispatcher(
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
val lastRequest = synchronized(timeMap) {
|
val lastRequest = synchronized(timeMap) {
|
||||||
val res = timeMap[source] ?: 0L
|
val res = timeMap.getOrDefault(source, 0L)
|
||||||
timeMap[source] = System.currentTimeMillis()
|
timeMap[source] = System.currentTimeMillis()
|
||||||
res
|
res
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -91,6 +91,7 @@ class DownloadWorker @AssistedInject constructor(
|
|||||||
private val localMangaRepository: LocalMangaRepository,
|
private val localMangaRepository: LocalMangaRepository,
|
||||||
private val mangaDataRepository: MangaDataRepository,
|
private val mangaDataRepository: MangaDataRepository,
|
||||||
private val mangaRepositoryFactory: MangaRepository.Factory,
|
private val mangaRepositoryFactory: MangaRepository.Factory,
|
||||||
|
private val settings: AppSettings,
|
||||||
@LocalStorageChanges private val localStorageChanges: MutableSharedFlow<LocalManga?>,
|
@LocalStorageChanges private val localStorageChanges: MutableSharedFlow<LocalManga?>,
|
||||||
notificationFactoryFactory: DownloadNotificationFactory.Factory,
|
notificationFactoryFactory: DownloadNotificationFactory.Factory,
|
||||||
) : CoroutineWorker(appContext, params) {
|
) : CoroutineWorker(appContext, params) {
|
||||||
@@ -182,7 +183,7 @@ class DownloadWorker @AssistedInject constructor(
|
|||||||
}
|
}
|
||||||
val repo = mangaRepositoryFactory.create(manga.source)
|
val repo = mangaRepositoryFactory.create(manga.source)
|
||||||
val mangaDetails = if (manga.chapters.isNullOrEmpty()) repo.getDetails(manga) else manga
|
val mangaDetails = if (manga.chapters.isNullOrEmpty()) repo.getDetails(manga) else manga
|
||||||
output = LocalMangaOutput.getOrCreate(destination, mangaDetails)
|
output = LocalMangaOutput.getOrCreate(destination, mangaDetails, settings.preferredDownloadFormat)
|
||||||
val coverUrl = mangaDetails.largeCoverUrl.ifNullOrEmpty { mangaDetails.coverUrl }
|
val coverUrl = mangaDetails.largeCoverUrl.ifNullOrEmpty { mangaDetails.coverUrl }
|
||||||
if (coverUrl.isNotEmpty()) {
|
if (coverUrl.isNotEmpty()) {
|
||||||
downloadFile(coverUrl, destination, repo.source).let { file ->
|
downloadFile(coverUrl, destination, repo.source).let { file ->
|
||||||
@@ -193,12 +194,12 @@ class DownloadWorker @AssistedInject constructor(
|
|||||||
val chapters = getChapters(mangaDetails, includedIds)
|
val chapters = getChapters(mangaDetails, includedIds)
|
||||||
for ((chapterIndex, chapter) in chapters.withIndex()) {
|
for ((chapterIndex, chapter) in chapters.withIndex()) {
|
||||||
checkIsPaused()
|
checkIsPaused()
|
||||||
if (chaptersToSkip.remove(chapter.id)) {
|
if (chaptersToSkip.remove(chapter.value.id)) {
|
||||||
publishState(currentState.copy(downloadedChapters = currentState.downloadedChapters + 1))
|
publishState(currentState.copy(downloadedChapters = currentState.downloadedChapters + 1))
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
val pages = runFailsafe {
|
val pages = runFailsafe {
|
||||||
repo.getPages(chapter)
|
repo.getPages(chapter.value)
|
||||||
} ?: continue
|
} ?: continue
|
||||||
val pageCounter = AtomicInteger(0)
|
val pageCounter = AtomicInteger(0)
|
||||||
channelFlow {
|
channelFlow {
|
||||||
@@ -237,7 +238,7 @@ class DownloadWorker @AssistedInject constructor(
|
|||||||
),
|
),
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
if (output.flushChapter(chapter)) {
|
if (output.flushChapter(chapter.value)) {
|
||||||
runCatchingCancellable {
|
runCatchingCancellable {
|
||||||
localStorageChanges.emit(LocalMangaInput.of(output.rootFile).getManga())
|
localStorageChanges.emit(LocalMangaInput.of(output.rootFile).getManga())
|
||||||
}.onFailure(Throwable::printStackTraceDebug)
|
}.onFailure(Throwable::printStackTraceDebug)
|
||||||
@@ -377,19 +378,26 @@ class DownloadWorker @AssistedInject constructor(
|
|||||||
private fun getChapters(
|
private fun getChapters(
|
||||||
manga: Manga,
|
manga: Manga,
|
||||||
includedIds: LongArray?,
|
includedIds: LongArray?,
|
||||||
): List<MangaChapter> {
|
): List<IndexedValue<MangaChapter>> {
|
||||||
val chapters = checkNotNull(manga.chapters) {
|
val chapters = checkNotNull(manga.chapters) { "Chapters list must not be null" }
|
||||||
"Chapters list must not be null"
|
val chaptersIdsSet = includedIds?.toMutableSet()
|
||||||
}.toMutableList()
|
val result = ArrayList<IndexedValue<MangaChapter>>((chaptersIdsSet ?: chapters).size)
|
||||||
if (includedIds != null) {
|
val counters = HashMap<String?, Int>()
|
||||||
val chaptersIdsSet = includedIds.toMutableSet()
|
for (chapter in chapters) {
|
||||||
chapters.retainAll { x -> chaptersIdsSet.remove(x.id) }
|
val index = counters[chapter.branch] ?: 0
|
||||||
|
counters[chapter.branch] = index + 1
|
||||||
|
if (chaptersIdsSet != null && !chaptersIdsSet.remove(chapter.id)) {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
result.add(IndexedValue(index, chapter))
|
||||||
|
}
|
||||||
|
if (chaptersIdsSet != null) {
|
||||||
check(chaptersIdsSet.isEmpty()) {
|
check(chaptersIdsSet.isEmpty()) {
|
||||||
"${chaptersIdsSet.size} of ${includedIds.size} requested chapters not found in manga"
|
"${chaptersIdsSet.size} of ${includedIds.size} requested chapters not found in manga"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
check(chapters.isNotEmpty()) { "Chapters list must not be empty" }
|
check(result.isNotEmpty()) { "Chapters list must not be empty" }
|
||||||
return chapters
|
return result
|
||||||
}
|
}
|
||||||
|
|
||||||
private suspend inline fun <T> withMangaLock(manga: Manga, block: () -> T) = try {
|
private suspend inline fun <T> withMangaLock(manga: Manga, block: () -> T) = try {
|
||||||
|
|||||||
@@ -57,7 +57,7 @@ class MangaSourcesRepository @Inject constructor(
|
|||||||
observeIsNsfwDisabled(),
|
observeIsNsfwDisabled(),
|
||||||
dao.observeEnabled(SourcesSortOrder.MANUAL),
|
dao.observeEnabled(SourcesSortOrder.MANUAL),
|
||||||
) { skipNsfw, sources ->
|
) { skipNsfw, sources ->
|
||||||
sources.count { skipNsfw || !MangaSource(it.source).isNsfw() }
|
sources.count { !skipNsfw || !MangaSource(it.source).isNsfw() }
|
||||||
}.distinctUntilChanged()
|
}.distinctUntilChanged()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -76,12 +76,9 @@ class ExploreRepository @Inject constructor(
|
|||||||
}
|
}
|
||||||
val list = repository.getList(
|
val list = repository.getList(
|
||||||
offset = 0,
|
offset = 0,
|
||||||
filter = MangaListFilter.Advanced(
|
filter = MangaListFilter.Advanced.Builder(order)
|
||||||
sortOrder = order,
|
.tags(setOfNotNull(tag))
|
||||||
tags = setOfNotNull(tag),
|
.build(),
|
||||||
locale = null,
|
|
||||||
states = emptySet(),
|
|
||||||
),
|
|
||||||
).asArrayList()
|
).asArrayList()
|
||||||
if (settings.isSuggestionsExcludeNsfw) {
|
if (settings.isSuggestionsExcludeNsfw) {
|
||||||
list.removeAll { it.isNsfw }
|
list.removeAll { it.isNsfw }
|
||||||
|
|||||||
@@ -111,7 +111,7 @@ class ExploreFragment :
|
|||||||
}
|
}
|
||||||
|
|
||||||
override fun onListHeaderClick(item: ListHeader, view: View) {
|
override fun onListHeaderClick(item: ListHeader, view: View) {
|
||||||
startActivity(SettingsActivity.newManageSourcesIntent(view.context))
|
startActivity(Intent(view.context, SourcesCatalogActivity::class.java))
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun onPrimaryButtonClick(tipView: TipView) {
|
override fun onPrimaryButtonClick(tipView: TipView) {
|
||||||
|
|||||||
@@ -29,7 +29,6 @@ import org.koitharu.kotatsu.list.ui.model.EmptyHint
|
|||||||
import org.koitharu.kotatsu.list.ui.model.ListHeader
|
import org.koitharu.kotatsu.list.ui.model.ListHeader
|
||||||
import org.koitharu.kotatsu.list.ui.model.ListModel
|
import org.koitharu.kotatsu.list.ui.model.ListModel
|
||||||
import org.koitharu.kotatsu.list.ui.model.LoadingState
|
import org.koitharu.kotatsu.list.ui.model.LoadingState
|
||||||
import org.koitharu.kotatsu.list.ui.model.TipModel
|
|
||||||
import org.koitharu.kotatsu.parsers.model.Manga
|
import org.koitharu.kotatsu.parsers.model.Manga
|
||||||
import org.koitharu.kotatsu.parsers.model.MangaSource
|
import org.koitharu.kotatsu.parsers.model.MangaSource
|
||||||
import org.koitharu.kotatsu.parsers.util.runCatchingCancellable
|
import org.koitharu.kotatsu.parsers.util.runCatchingCancellable
|
||||||
@@ -126,24 +125,18 @@ class ExploreViewModel @Inject constructor(
|
|||||||
randomLoading: Boolean,
|
randomLoading: Boolean,
|
||||||
newSources: Set<MangaSource>,
|
newSources: Set<MangaSource>,
|
||||||
): List<ListModel> {
|
): List<ListModel> {
|
||||||
val result = ArrayList<ListModel>(sources.size + 4)
|
val result = ArrayList<ListModel>(sources.size + 3)
|
||||||
result += ExploreButtons(randomLoading)
|
result += ExploreButtons(randomLoading)
|
||||||
if (recommendation != null) {
|
if (recommendation != null) {
|
||||||
result += ListHeader(R.string.suggestions)
|
result += ListHeader(R.string.suggestions)
|
||||||
result += RecommendationsItem(recommendation)
|
result += RecommendationsItem(recommendation)
|
||||||
}
|
}
|
||||||
if (sources.isNotEmpty()) {
|
if (sources.isNotEmpty()) {
|
||||||
result += ListHeader(R.string.remote_sources, R.string.manage)
|
result += ListHeader(
|
||||||
if (newSources.isNotEmpty()) {
|
textRes = R.string.remote_sources,
|
||||||
result += TipModel(
|
buttonTextRes = R.string.catalog,
|
||||||
key = TIP_NEW_SOURCES,
|
badge = if (newSources.isNotEmpty()) "" else null,
|
||||||
title = R.string.new_sources_text,
|
)
|
||||||
text = R.string.new_sources_text,
|
|
||||||
icon = R.drawable.ic_explore_normal,
|
|
||||||
primaryButtonText = R.string.manage,
|
|
||||||
secondaryButtonText = R.string.discard,
|
|
||||||
)
|
|
||||||
}
|
|
||||||
sources.mapTo(result) { MangaSourceItem(it, isGrid) }
|
sources.mapTo(result) { MangaSourceItem(it, isGrid) }
|
||||||
} else {
|
} else {
|
||||||
result += EmptyHint(
|
result += EmptyHint(
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user