Compare commits
27 Commits
v6.1.3
...
feature/do
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
5df55d1fe9 | ||
|
|
fbb267e11c | ||
|
|
5740af05fa | ||
|
|
ae2cc1dffc | ||
|
|
a5b9712e9f | ||
|
|
c013e6e4f4 | ||
|
|
0249faa3f6 | ||
|
|
9c52423dc0 | ||
|
|
1f7e5458ae | ||
|
|
b4d487b398 | ||
|
|
0281f1eadb | ||
|
|
1bd9b655f9 | ||
|
|
ed87292921 | ||
|
|
861be7614e | ||
|
|
717fe8748a | ||
|
|
c7a1312cd6 | ||
|
|
b2927854d4 | ||
|
|
cfda150630 | ||
|
|
4fa1382ce9 | ||
|
|
43075c52d1 | ||
|
|
87942747fc | ||
|
|
bb6cd73acd | ||
|
|
6790e5b0d4 | ||
|
|
845c356a73 | ||
|
|
34499ea77d | ||
|
|
6210864280 | ||
|
|
19084419c7 |
@@ -16,8 +16,8 @@ android {
|
||||
applicationId 'org.koitharu.kotatsu'
|
||||
minSdk = 21
|
||||
targetSdk = 34
|
||||
versionCode = 581
|
||||
versionName = '6.1.3'
|
||||
versionCode = 584
|
||||
versionName = '6.1.6'
|
||||
generatedDensities = []
|
||||
testInstrumentationRunner "org.koitharu.kotatsu.HiltTestRunner"
|
||||
ksp {
|
||||
@@ -81,7 +81,7 @@ afterEvaluate {
|
||||
}
|
||||
dependencies {
|
||||
//noinspection GradleDependency
|
||||
implementation('com.github.KotatsuApp:kotatsu-parsers:0004be15ba') {
|
||||
implementation('com.github.KotatsuApp:kotatsu-parsers:400a90464e') {
|
||||
exclude group: 'org.json', module: 'json'
|
||||
}
|
||||
|
||||
@@ -90,7 +90,7 @@ dependencies {
|
||||
|
||||
implementation 'androidx.appcompat:appcompat:1.6.1'
|
||||
implementation 'androidx.core:core-ktx:1.12.0'
|
||||
implementation 'androidx.activity:activity-ktx:1.7.2'
|
||||
implementation 'androidx.activity:activity-ktx:1.8.0'
|
||||
implementation 'androidx.fragment:fragment-ktx:1.6.1'
|
||||
implementation 'androidx.lifecycle:lifecycle-viewmodel-ktx:2.6.2'
|
||||
implementation 'androidx.lifecycle:lifecycle-livedata-ktx:2.6.2'
|
||||
@@ -102,7 +102,7 @@ dependencies {
|
||||
implementation 'androidx.viewpager2:viewpager2:1.1.0-beta02'
|
||||
implementation 'androidx.preference:preference-ktx:1.2.1'
|
||||
implementation 'androidx.biometric:biometric-ktx:1.2.0-alpha05'
|
||||
implementation 'com.google.android.material:material:1.9.0'
|
||||
implementation 'com.google.android.material:material:1.10.0'
|
||||
implementation 'androidx.lifecycle:lifecycle-common-java8:2.6.2'
|
||||
|
||||
// TODO https://issuetracker.google.com/issues/254846063
|
||||
@@ -120,13 +120,13 @@ dependencies {
|
||||
|
||||
implementation 'com.squareup.okhttp3:okhttp:4.11.0'
|
||||
implementation 'com.squareup.okhttp3:okhttp-dnsoverhttps:4.11.0'
|
||||
implementation 'com.squareup.okio:okio:3.5.0'
|
||||
implementation 'com.squareup.okio:okio:3.6.0'
|
||||
|
||||
implementation 'com.hannesdorfmann:adapterdelegates4-kotlin-dsl:4.3.2'
|
||||
implementation 'com.hannesdorfmann:adapterdelegates4-kotlin-dsl-viewbinding:4.3.2'
|
||||
|
||||
implementation 'com.google.dagger:hilt-android:2.48'
|
||||
kapt 'com.google.dagger:hilt-compiler:2.48'
|
||||
implementation 'com.google.dagger:hilt-android:2.48.1'
|
||||
kapt 'com.google.dagger:hilt-compiler:2.48.1'
|
||||
implementation 'androidx.hilt:hilt-work:1.0.0'
|
||||
kapt 'androidx.hilt:hilt-compiler:1.0.0'
|
||||
|
||||
@@ -155,6 +155,6 @@ dependencies {
|
||||
androidTestImplementation 'androidx.room:room-testing:2.5.2'
|
||||
androidTestImplementation 'com.squareup.moshi:moshi-kotlin:1.15.0'
|
||||
|
||||
androidTestImplementation 'com.google.dagger:hilt-android-testing:2.48'
|
||||
kaptAndroidTest 'com.google.dagger:hilt-android-compiler:2.48'
|
||||
androidTestImplementation 'com.google.dagger:hilt-android-testing:2.48.1'
|
||||
kaptAndroidTest 'com.google.dagger:hilt-android-compiler:2.48.1'
|
||||
}
|
||||
|
||||
@@ -51,6 +51,28 @@ abstract class TagsDao {
|
||||
)
|
||||
abstract suspend fun findTags(query: String, limit: Int): List<TagEntity>
|
||||
|
||||
@Query(
|
||||
"""
|
||||
SELECT tags.* FROM manga_tags
|
||||
LEFT JOIN tags ON tags.tag_id = manga_tags.tag_id
|
||||
WHERE manga_tags.manga_id IN (SELECT manga_id FROM manga_tags WHERE tag_id = :tagId)
|
||||
GROUP BY tags.tag_id
|
||||
ORDER BY COUNT(manga_id) DESC;
|
||||
""",
|
||||
)
|
||||
abstract suspend fun findRelatedTags(tagId: Long): List<TagEntity>
|
||||
|
||||
@Query(
|
||||
"""
|
||||
SELECT tags.* FROM manga_tags
|
||||
LEFT JOIN tags ON tags.tag_id = manga_tags.tag_id
|
||||
WHERE manga_tags.manga_id IN (SELECT manga_id FROM manga_tags WHERE tag_id IN (:ids))
|
||||
GROUP BY tags.tag_id
|
||||
ORDER BY COUNT(manga_id) DESC;
|
||||
""",
|
||||
)
|
||||
abstract suspend fun findRelatedTags(ids: Set<Long>): List<TagEntity>
|
||||
|
||||
@Upsert
|
||||
abstract suspend fun upsert(tags: Iterable<TagEntity>)
|
||||
}
|
||||
|
||||
@@ -19,6 +19,8 @@ fun TagEntity.toMangaTag() = MangaTag(
|
||||
|
||||
fun Collection<TagEntity>.toMangaTags() = mapToSet(TagEntity::toMangaTag)
|
||||
|
||||
fun Collection<TagEntity>.toMangaTagsList() = map(TagEntity::toMangaTag)
|
||||
|
||||
fun MangaEntity.toManga(tags: Set<MangaTag>) = Manga(
|
||||
id = this.id,
|
||||
title = this.title,
|
||||
|
||||
@@ -1,6 +1,9 @@
|
||||
package org.koitharu.kotatsu.core.network
|
||||
|
||||
import androidx.collection.ArraySet
|
||||
import dagger.Lazy
|
||||
import kotlinx.coroutines.Dispatchers
|
||||
import kotlinx.coroutines.runInterruptible
|
||||
import okhttp3.Interceptor
|
||||
import okhttp3.Request
|
||||
import okhttp3.Response
|
||||
@@ -13,6 +16,7 @@ import org.koitharu.kotatsu.core.parser.MangaRepository
|
||||
import org.koitharu.kotatsu.core.parser.RemoteMangaRepository
|
||||
import org.koitharu.kotatsu.core.prefs.AppSettings
|
||||
import org.koitharu.kotatsu.parsers.model.MangaSource
|
||||
import java.util.EnumMap
|
||||
import javax.inject.Inject
|
||||
import javax.inject.Singleton
|
||||
|
||||
@@ -22,9 +26,15 @@ class MirrorSwitchInterceptor @Inject constructor(
|
||||
private val settings: AppSettings,
|
||||
) : Interceptor {
|
||||
|
||||
private val locks = EnumMap<MangaSource, Any>(MangaSource::class.java)
|
||||
private val blacklist = EnumMap<MangaSource, MutableSet<String>>(MangaSource::class.java)
|
||||
|
||||
val isEnabled: Boolean
|
||||
get() = settings.isMirrorSwitchingAvailable
|
||||
|
||||
override fun intercept(chain: Interceptor.Chain): Response {
|
||||
val request = chain.request()
|
||||
if (!settings.isMirrorSwitchingAvailable) {
|
||||
if (!isEnabled) {
|
||||
return chain.proceed(request)
|
||||
}
|
||||
return try {
|
||||
@@ -43,6 +53,30 @@ class MirrorSwitchInterceptor @Inject constructor(
|
||||
}
|
||||
}
|
||||
|
||||
suspend fun trySwitchMirror(repository: RemoteMangaRepository): Boolean = runInterruptible(Dispatchers.Default) {
|
||||
if (!isEnabled) {
|
||||
return@runInterruptible false
|
||||
}
|
||||
val mirrors = repository.getAvailableMirrors()
|
||||
if (mirrors.size <= 1) {
|
||||
return@runInterruptible false
|
||||
}
|
||||
synchronized(obtainLock(repository.source)) {
|
||||
val currentMirror = repository.domain
|
||||
addToBlacklist(repository.source, currentMirror)
|
||||
val newMirror = mirrors.firstOrNull { x ->
|
||||
x != currentMirror && !isBlacklisted(repository.source, x)
|
||||
} ?: return@synchronized false
|
||||
repository.domain = newMirror
|
||||
true
|
||||
}
|
||||
}
|
||||
|
||||
fun rollback(repository: RemoteMangaRepository, oldMirror: String) = synchronized(obtainLock(repository.source)) {
|
||||
blacklist[repository.source]?.remove(oldMirror)
|
||||
repository.domain = oldMirror
|
||||
}
|
||||
|
||||
private fun trySwitchMirror(request: Request, chain: Interceptor.Chain): Response? {
|
||||
val source = request.tag(MangaSource::class.java) ?: return null
|
||||
val repository = mangaRepositoryFactoryLazy.get().create(source) as? RemoteMangaRepository ?: return null
|
||||
@@ -50,7 +84,9 @@ class MirrorSwitchInterceptor @Inject constructor(
|
||||
if (mirrors.isEmpty()) {
|
||||
return null
|
||||
}
|
||||
return tryMirrors(repository, mirrors, chain, request)
|
||||
return synchronized(obtainLock(repository.source)) {
|
||||
tryMirrors(repository, mirrors, chain, request)
|
||||
}
|
||||
}
|
||||
|
||||
private fun tryMirrors(
|
||||
@@ -66,7 +102,7 @@ class MirrorSwitchInterceptor @Inject constructor(
|
||||
}
|
||||
val urlBuilder = url.newBuilder()
|
||||
for (mirror in mirrors) {
|
||||
if (mirror == currentDomain) {
|
||||
if (mirror == currentDomain || isBlacklisted(repository.source, mirror)) {
|
||||
continue
|
||||
}
|
||||
val newHost = hostOf(url.host, mirror) ?: continue
|
||||
@@ -75,6 +111,7 @@ class MirrorSwitchInterceptor @Inject constructor(
|
||||
.build()
|
||||
val response = chain.proceed(newRequest)
|
||||
if (response.isFailed) {
|
||||
addToBlacklist(repository.source, mirror)
|
||||
response.closeQuietly()
|
||||
} else {
|
||||
repository.domain = mirror
|
||||
@@ -104,4 +141,18 @@ class MirrorSwitchInterceptor @Inject constructor(
|
||||
private fun ResponseBody.copy(): ResponseBody {
|
||||
return source().readByteArray().toResponseBody(contentType())
|
||||
}
|
||||
|
||||
private fun obtainLock(source: MangaSource): Any = locks.getOrPut(source) {
|
||||
Any()
|
||||
}
|
||||
|
||||
private fun isBlacklisted(source: MangaSource, domain: String): Boolean {
|
||||
return blacklist[source]?.contains(domain) == true
|
||||
}
|
||||
|
||||
private fun addToBlacklist(source: MangaSource, domain: String) {
|
||||
blacklist.getOrPut(source) {
|
||||
ArraySet(2)
|
||||
}.add(domain)
|
||||
}
|
||||
}
|
||||
|
||||
@@ -76,12 +76,18 @@ class AppShortcutManager @Inject constructor(
|
||||
}
|
||||
}
|
||||
|
||||
suspend fun requestPinShortcut(manga: Manga): Boolean {
|
||||
return ShortcutManagerCompat.requestPinShortcut(context, buildShortcutInfo(manga), null)
|
||||
suspend fun requestPinShortcut(manga: Manga): Boolean = try {
|
||||
ShortcutManagerCompat.requestPinShortcut(context, buildShortcutInfo(manga), null)
|
||||
} catch (e: IllegalStateException) {
|
||||
e.printStackTraceDebug()
|
||||
false
|
||||
}
|
||||
|
||||
suspend fun requestPinShortcut(source: MangaSource): Boolean {
|
||||
return ShortcutManagerCompat.requestPinShortcut(context, buildShortcutInfo(source), null)
|
||||
suspend fun requestPinShortcut(source: MangaSource): Boolean = try {
|
||||
ShortcutManagerCompat.requestPinShortcut(context, buildShortcutInfo(source), null)
|
||||
} catch (e: IllegalStateException) {
|
||||
e.printStackTraceDebug()
|
||||
false
|
||||
}
|
||||
|
||||
@VisibleForTesting
|
||||
|
||||
@@ -17,7 +17,7 @@ import org.koitharu.kotatsu.parsers.MangaLoaderContext
|
||||
import org.koitharu.kotatsu.parsers.config.MangaSourceConfig
|
||||
import org.koitharu.kotatsu.parsers.model.MangaSource
|
||||
import java.lang.ref.WeakReference
|
||||
import java.util.*
|
||||
import java.util.Locale
|
||||
import javax.inject.Inject
|
||||
import javax.inject.Singleton
|
||||
import kotlin.coroutines.resume
|
||||
@@ -50,7 +50,7 @@ class MangaLoaderContextImpl @Inject constructor(
|
||||
}
|
||||
|
||||
override fun encodeBase64(data: ByteArray): String {
|
||||
return Base64.encodeToString(data, Base64.NO_PADDING)
|
||||
return Base64.encodeToString(data, Base64.NO_WRAP)
|
||||
}
|
||||
|
||||
override fun decodeBase64(data: String): ByteArray {
|
||||
|
||||
@@ -2,6 +2,7 @@ package org.koitharu.kotatsu.core.parser
|
||||
|
||||
import androidx.annotation.AnyThread
|
||||
import org.koitharu.kotatsu.core.cache.ContentCache
|
||||
import org.koitharu.kotatsu.core.network.MirrorSwitchInterceptor
|
||||
import org.koitharu.kotatsu.local.data.LocalMangaRepository
|
||||
import org.koitharu.kotatsu.parsers.MangaLoaderContext
|
||||
import org.koitharu.kotatsu.parsers.model.Manga
|
||||
@@ -43,6 +44,7 @@ interface MangaRepository {
|
||||
private val localMangaRepository: LocalMangaRepository,
|
||||
private val loaderContext: MangaLoaderContext,
|
||||
private val contentCache: ContentCache,
|
||||
private val mirrorSwitchInterceptor: MirrorSwitchInterceptor,
|
||||
) {
|
||||
|
||||
private val cache = EnumMap<MangaSource, WeakReference<RemoteMangaRepository>>(MangaSource::class.java)
|
||||
@@ -55,7 +57,11 @@ interface MangaRepository {
|
||||
cache[source]?.get()?.let { return it }
|
||||
return synchronized(cache) {
|
||||
cache[source]?.get()?.let { return it }
|
||||
val repository = RemoteMangaRepository(MangaParser(source, loaderContext), contentCache)
|
||||
val repository = RemoteMangaRepository(
|
||||
parser = MangaParser(source, loaderContext),
|
||||
cache = contentCache,
|
||||
mirrorSwitchInterceptor = mirrorSwitchInterceptor,
|
||||
)
|
||||
cache[source] = WeakReference(repository)
|
||||
repository
|
||||
}
|
||||
|
||||
@@ -13,11 +13,13 @@ import okhttp3.Response
|
||||
import org.koitharu.kotatsu.BuildConfig
|
||||
import org.koitharu.kotatsu.core.cache.ContentCache
|
||||
import org.koitharu.kotatsu.core.cache.SafeDeferred
|
||||
import org.koitharu.kotatsu.core.network.MirrorSwitchInterceptor
|
||||
import org.koitharu.kotatsu.core.prefs.SourceSettings
|
||||
import org.koitharu.kotatsu.core.util.ext.processLifecycleScope
|
||||
import org.koitharu.kotatsu.parsers.MangaParser
|
||||
import org.koitharu.kotatsu.parsers.MangaParserAuthProvider
|
||||
import org.koitharu.kotatsu.parsers.config.ConfigKey
|
||||
import org.koitharu.kotatsu.parsers.exception.ParseException
|
||||
import org.koitharu.kotatsu.parsers.model.Favicons
|
||||
import org.koitharu.kotatsu.parsers.model.Manga
|
||||
import org.koitharu.kotatsu.parsers.model.MangaChapter
|
||||
@@ -31,6 +33,7 @@ import org.koitharu.kotatsu.parsers.util.runCatchingCancellable
|
||||
class RemoteMangaRepository(
|
||||
private val parser: MangaParser,
|
||||
private val cache: ContentCache,
|
||||
private val mirrorSwitchInterceptor: MirrorSwitchInterceptor,
|
||||
) : MangaRepository, Interceptor {
|
||||
|
||||
override val source: MangaSource
|
||||
@@ -66,11 +69,15 @@ class RemoteMangaRepository(
|
||||
}
|
||||
|
||||
override suspend fun getList(offset: Int, query: String): List<Manga> {
|
||||
return parser.getList(offset, query)
|
||||
return mirrorSwitchInterceptor.withMirrorSwitching {
|
||||
parser.getList(offset, query)
|
||||
}
|
||||
}
|
||||
|
||||
override suspend fun getList(offset: Int, tags: Set<MangaTag>?, sortOrder: SortOrder?): List<Manga> {
|
||||
return parser.getList(offset, tags, sortOrder)
|
||||
return mirrorSwitchInterceptor.withMirrorSwitching {
|
||||
parser.getList(offset, tags, sortOrder)
|
||||
}
|
||||
}
|
||||
|
||||
override suspend fun getDetails(manga: Manga): Manga = getDetails(manga, withCache = true)
|
||||
@@ -78,17 +85,25 @@ class RemoteMangaRepository(
|
||||
override suspend fun getPages(chapter: MangaChapter): List<MangaPage> {
|
||||
cache.getPages(source, chapter.url)?.let { return it }
|
||||
val pages = asyncSafe {
|
||||
parser.getPages(chapter).distinctById()
|
||||
mirrorSwitchInterceptor.withMirrorSwitching {
|
||||
parser.getPages(chapter).distinctById()
|
||||
}
|
||||
}
|
||||
cache.putPages(source, chapter.url, pages)
|
||||
return pages.await()
|
||||
}
|
||||
|
||||
override suspend fun getPageUrl(page: MangaPage): String = parser.getPageUrl(page)
|
||||
override suspend fun getPageUrl(page: MangaPage): String = mirrorSwitchInterceptor.withMirrorSwitching {
|
||||
parser.getPageUrl(page)
|
||||
}
|
||||
|
||||
override suspend fun getTags(): Set<MangaTag> = parser.getTags()
|
||||
override suspend fun getTags(): Set<MangaTag> = mirrorSwitchInterceptor.withMirrorSwitching {
|
||||
parser.getTags()
|
||||
}
|
||||
|
||||
suspend fun getFavicons(): Favicons = parser.getFavicons()
|
||||
suspend fun getFavicons(): Favicons = mirrorSwitchInterceptor.withMirrorSwitching {
|
||||
parser.getFavicons()
|
||||
}
|
||||
|
||||
override suspend fun getRelated(seed: Manga): List<Manga> {
|
||||
cache.getRelatedManga(source, seed.url)?.let { return it }
|
||||
@@ -105,7 +120,9 @@ class RemoteMangaRepository(
|
||||
}
|
||||
cache.getDetails(source, manga.url)?.let { return it }
|
||||
val details = asyncSafe {
|
||||
parser.getDetails(manga)
|
||||
mirrorSwitchInterceptor.withMirrorSwitching {
|
||||
parser.getDetails(manga)
|
||||
}
|
||||
}
|
||||
cache.putDetails(source, manga.url, details)
|
||||
return details.await()
|
||||
@@ -155,4 +172,33 @@ class RemoteMangaRepository(
|
||||
}
|
||||
return result
|
||||
}
|
||||
|
||||
private suspend fun <R> MirrorSwitchInterceptor.withMirrorSwitching(block: suspend () -> R): R {
|
||||
if (!isEnabled) {
|
||||
return block()
|
||||
}
|
||||
val initialMirror = domain
|
||||
val result = runCatchingCancellable {
|
||||
block()
|
||||
}
|
||||
if (result.isValidResult()) {
|
||||
return result.getOrThrow()
|
||||
}
|
||||
return if (trySwitchMirror(this@RemoteMangaRepository)) {
|
||||
val newResult = runCatchingCancellable {
|
||||
block()
|
||||
}
|
||||
if (newResult.isValidResult()) {
|
||||
return newResult.getOrThrow()
|
||||
} else {
|
||||
rollback(this@RemoteMangaRepository, initialMirror)
|
||||
return result.getOrThrow()
|
||||
}
|
||||
} else {
|
||||
result.getOrThrow()
|
||||
}
|
||||
}
|
||||
|
||||
private fun Result<*>.isValidResult() = exceptionOrNull() !is ParseException
|
||||
&& (getOrNull() as? Collection<*>)?.isEmpty() != true
|
||||
}
|
||||
|
||||
@@ -270,6 +270,9 @@ class AppSettings @Inject constructor(@ApplicationContext context: Context) {
|
||||
val isReaderSliderEnabled: Boolean
|
||||
get() = prefs.getBoolean(KEY_READER_SLIDER, true)
|
||||
|
||||
val isReaderKeepScreenOn: Boolean
|
||||
get() = prefs.getBoolean(KEY_READER_SCREEN_ON, true)
|
||||
|
||||
val isImagesProxyEnabled: Boolean
|
||||
get() = prefs.getBoolean(KEY_IMAGES_PROXY, false)
|
||||
|
||||
@@ -460,6 +463,7 @@ class AppSettings @Inject constructor(@ApplicationContext context: Context) {
|
||||
const val KEY_READER_BAR = "reader_bar"
|
||||
const val KEY_READER_SLIDER = "reader_slider"
|
||||
const val KEY_READER_BACKGROUND = "reader_background"
|
||||
const val KEY_READER_SCREEN_ON = "reader_screen_on"
|
||||
const val KEY_SHORTCUTS = "dynamic_shortcuts"
|
||||
const val KEY_READER_TAPS_LTR = "reader_taps_ltr"
|
||||
const val KEY_LOCAL_LIST_ORDER = "local_order"
|
||||
|
||||
@@ -65,6 +65,9 @@ class SlidingBottomNavigationView @JvmOverloads constructor(
|
||||
}
|
||||
|
||||
fun show() {
|
||||
if (currentState == STATE_UP) {
|
||||
return
|
||||
}
|
||||
currentAnimator?.cancel()
|
||||
clearAnimation()
|
||||
|
||||
@@ -77,6 +80,9 @@ class SlidingBottomNavigationView @JvmOverloads constructor(
|
||||
}
|
||||
|
||||
fun hide() {
|
||||
if (currentState == STATE_DOWN) {
|
||||
return
|
||||
}
|
||||
currentAnimator?.cancel()
|
||||
clearAnimation()
|
||||
|
||||
@@ -117,6 +123,7 @@ class SlidingBottomNavigationView @JvmOverloads constructor(
|
||||
}
|
||||
|
||||
internal class SavedState : AbsSavedState {
|
||||
|
||||
var currentState = STATE_UP
|
||||
var translationY = 0F
|
||||
|
||||
|
||||
@@ -160,21 +160,22 @@ class DetailsFragment :
|
||||
}
|
||||
|
||||
when (manga.state) {
|
||||
MangaState.FINISHED -> {
|
||||
infoLayout.textViewState.apply {
|
||||
textAndVisible = resources.getString(R.string.state_finished)
|
||||
drawableTop = ContextCompat.getDrawable(context, R.drawable.ic_state_finished)
|
||||
}
|
||||
MangaState.FINISHED -> infoLayout.textViewState.apply {
|
||||
textAndVisible = resources.getString(R.string.state_finished)
|
||||
drawableTop = ContextCompat.getDrawable(context, R.drawable.ic_state_finished)
|
||||
}
|
||||
|
||||
MangaState.ONGOING -> {
|
||||
infoLayout.textViewState.apply {
|
||||
textAndVisible = resources.getString(R.string.state_ongoing)
|
||||
drawableTop = ContextCompat.getDrawable(context, R.drawable.ic_state_ongoing)
|
||||
}
|
||||
MangaState.ONGOING -> infoLayout.textViewState.apply {
|
||||
textAndVisible = resources.getString(R.string.state_ongoing)
|
||||
drawableTop = ContextCompat.getDrawable(context, R.drawable.ic_state_ongoing)
|
||||
}
|
||||
|
||||
else -> infoLayout.textViewState.isVisible = false
|
||||
MangaState.ABANDONED -> infoLayout.textViewState.apply {
|
||||
textAndVisible = resources.getString(R.string.state_abandoned)
|
||||
drawableTop = ContextCompat.getDrawable(context, R.drawable.ic_state_abandoned)
|
||||
}
|
||||
|
||||
null -> infoLayout.textViewState.isVisible = false
|
||||
}
|
||||
if (manga.source == MangaSource.LOCAL) {
|
||||
infoLayout.textViewSource.isVisible = false
|
||||
|
||||
@@ -131,7 +131,7 @@ class FilterCoordinator @Inject constructor(
|
||||
observeState(),
|
||||
observeAvailableTags(),
|
||||
) { state, available ->
|
||||
val chips = createChipsList(state, available.orEmpty())
|
||||
val chips = createChipsList(state, available.orEmpty(), 8)
|
||||
FilterHeaderModel(chips, state.sortOrder, state.tags.isNotEmpty())
|
||||
}
|
||||
|
||||
@@ -157,11 +157,16 @@ class FilterCoordinator @Inject constructor(
|
||||
private suspend fun createChipsList(
|
||||
filterState: FilterState,
|
||||
availableTags: Set<MangaTag>,
|
||||
limit: Int,
|
||||
): List<ChipsView.ChipModel> {
|
||||
val selectedTags = filterState.tags.toMutableSet()
|
||||
var tags = searchRepository.getTagsSuggestion("", 6, repository.source)
|
||||
if (tags.isEmpty()) {
|
||||
tags = availableTags.take(6)
|
||||
var tags = if (selectedTags.isEmpty()) {
|
||||
searchRepository.getTagsSuggestion("", limit, repository.source)
|
||||
} else {
|
||||
searchRepository.getTagsSuggestion(selectedTags).take(limit)
|
||||
}
|
||||
if (tags.size < limit) {
|
||||
tags = tags + availableTags.take(limit - tags.size)
|
||||
}
|
||||
if (tags.isEmpty() && selectedTags.isEmpty()) {
|
||||
return emptyList()
|
||||
|
||||
@@ -34,6 +34,9 @@ abstract class MangaListViewModel(
|
||||
)
|
||||
val onDownloadStarted = MutableEventFlow<Unit>()
|
||||
|
||||
val isIncognitoModeEnabled: Boolean
|
||||
get() = settings.isIncognitoModeEnabled
|
||||
|
||||
open fun onUpdateFilter(tags: Set<MangaTag>) = Unit
|
||||
|
||||
abstract fun onRefresh()
|
||||
|
||||
@@ -83,6 +83,7 @@ class MainActivity : BaseActivity<ActivityMainBinding>(), AppBarOwner, BottomNav
|
||||
private val viewModel by viewModels<MainViewModel>()
|
||||
private val searchSuggestionViewModel by viewModels<SearchSuggestionViewModel>()
|
||||
private val closeSearchCallback = CloseSearchCallback()
|
||||
private val appUpdateDialog = AppUpdateDialog(this)
|
||||
private lateinit var navigationDelegate: MainNavigationDelegate
|
||||
private lateinit var appUpdateBadge: OptionsMenuBadgeHelper
|
||||
|
||||
@@ -111,7 +112,6 @@ class MainActivity : BaseActivity<ActivityMainBinding>(), AppBarOwner, BottomNav
|
||||
onFocusChangeListener = this@MainActivity
|
||||
searchSuggestionListener = this@MainActivity
|
||||
}
|
||||
window.statusBarColor = ContextCompat.getColor(this, R.color.dim_statusbar)
|
||||
|
||||
viewBinding.fab?.setOnClickListener(this)
|
||||
viewBinding.navRail?.headerView?.setOnClickListener(this)
|
||||
@@ -143,6 +143,9 @@ class MainActivity : BaseActivity<ActivityMainBinding>(), AppBarOwner, BottomNav
|
||||
viewModel.onFirstStart.observeEvent(this) {
|
||||
OnboardDialogFragment.show(supportFragmentManager)
|
||||
}
|
||||
viewModel.isIncognitoMode.observe(this) {
|
||||
adjustSearchUI(isSearchOpened(), false)
|
||||
}
|
||||
searchSuggestionViewModel.isIncognitoModeEnabled.observe(this, this::onIncognitoModeChanged)
|
||||
}
|
||||
|
||||
@@ -198,8 +201,7 @@ class MainActivity : BaseActivity<ActivityMainBinding>(), AppBarOwner, BottomNav
|
||||
|
||||
R.id.action_app_update -> {
|
||||
viewModel.appUpdate.value?.also {
|
||||
AppUpdateDialog(this)
|
||||
.show(it)
|
||||
appUpdateDialog.show(it)
|
||||
} != null
|
||||
}
|
||||
|
||||
@@ -312,13 +314,11 @@ class MainActivity : BaseActivity<ActivityMainBinding>(), AppBarOwner, BottomNav
|
||||
|
||||
private fun onSearchOpened() {
|
||||
adjustSearchUI(isOpened = true, animate = true)
|
||||
closeSearchCallback.isEnabled = true
|
||||
}
|
||||
|
||||
private fun onSearchClosed() {
|
||||
viewBinding.searchView.hideKeyboard()
|
||||
adjustSearchUI(isOpened = false, animate = true)
|
||||
closeSearchCallback.isEnabled = false
|
||||
}
|
||||
|
||||
private fun isSearchOpened(): Boolean {
|
||||
@@ -379,7 +379,11 @@ class MainActivity : BaseActivity<ActivityMainBinding>(), AppBarOwner, BottomNav
|
||||
adjustFabVisibility(isSearchOpened = isOpened)
|
||||
supportActionBar?.apply {
|
||||
setHomeAsUpIndicator(
|
||||
if (isOpened) materialR.drawable.abc_ic_ab_back_material else materialR.drawable.abc_ic_search_api_material,
|
||||
when {
|
||||
isOpened -> materialR.drawable.abc_ic_ab_back_material
|
||||
viewModel.isIncognitoMode.value -> R.drawable.ic_incognito
|
||||
else -> materialR.drawable.abc_ic_search_api_material
|
||||
},
|
||||
)
|
||||
setHomeActionContentDescription(
|
||||
if (isOpened) R.string.back else R.string.search,
|
||||
@@ -389,6 +393,7 @@ class MainActivity : BaseActivity<ActivityMainBinding>(), AppBarOwner, BottomNav
|
||||
if (isOpened) R.string.search_hint else R.string.search_manga,
|
||||
)
|
||||
bottomNav?.showOrHide(!isOpened)
|
||||
closeSearchCallback.isEnabled = isOpened
|
||||
}
|
||||
|
||||
private fun requestNotificationsPermission() {
|
||||
|
||||
@@ -72,7 +72,7 @@ class MainNavigationDelegate(
|
||||
}
|
||||
|
||||
override fun handleOnBackPressed() {
|
||||
navBar.selectedItemId = R.id.nav_history
|
||||
navBar.selectedItemId = firstItem()?.itemId ?: return
|
||||
}
|
||||
|
||||
fun onCreate(lifecycleOwner: LifecycleOwner, savedInstanceState: Bundle?) {
|
||||
@@ -171,7 +171,7 @@ class MainNavigationDelegate(
|
||||
}
|
||||
|
||||
private fun onFragmentChanged(fragment: Fragment, fromUser: Boolean) {
|
||||
isEnabled = fragment !is HistoryListFragment
|
||||
isEnabled = getItemId(fragment) != firstItem()?.itemId
|
||||
listeners.forEach { it.onFragmentChanged(fragment, fromUser) }
|
||||
}
|
||||
|
||||
|
||||
@@ -13,6 +13,7 @@ import org.koitharu.kotatsu.core.exceptions.EmptyHistoryException
|
||||
import org.koitharu.kotatsu.core.github.AppUpdateRepository
|
||||
import org.koitharu.kotatsu.core.prefs.AppSettings
|
||||
import org.koitharu.kotatsu.core.prefs.NavItem
|
||||
import org.koitharu.kotatsu.core.prefs.observeAsStateFlow
|
||||
import org.koitharu.kotatsu.core.ui.BaseViewModel
|
||||
import org.koitharu.kotatsu.core.util.ext.MutableEventFlow
|
||||
import org.koitharu.kotatsu.core.util.ext.call
|
||||
@@ -43,6 +44,12 @@ class MainViewModel @Inject constructor(
|
||||
initialValue = false,
|
||||
)
|
||||
|
||||
val isIncognitoMode = settings.observeAsStateFlow(
|
||||
scope = viewModelScope + Dispatchers.Default,
|
||||
key = AppSettings.KEY_INCOGNITO_MODE,
|
||||
valueProducer = { isIncognitoModeEnabled },
|
||||
)
|
||||
|
||||
val appUpdate = appUpdateRepository.observeAvailableUpdate()
|
||||
|
||||
val counters = combine(
|
||||
|
||||
@@ -13,11 +13,11 @@ import org.koitharu.kotatsu.core.parser.MangaDataRepository
|
||||
import org.koitharu.kotatsu.core.parser.MangaRepository
|
||||
import org.koitharu.kotatsu.core.prefs.AppSettings
|
||||
import org.koitharu.kotatsu.core.prefs.ReaderMode
|
||||
import org.koitharu.kotatsu.core.util.ext.printStackTraceDebug
|
||||
import org.koitharu.kotatsu.parsers.model.Manga
|
||||
import org.koitharu.kotatsu.parsers.model.MangaPage
|
||||
import org.koitharu.kotatsu.parsers.util.runCatchingCancellable
|
||||
import org.koitharu.kotatsu.reader.ui.ReaderState
|
||||
import org.koitharu.kotatsu.core.util.ext.printStackTraceDebug
|
||||
import java.io.InputStream
|
||||
import java.util.zip.ZipFile
|
||||
import javax.inject.Inject
|
||||
|
||||
@@ -103,7 +103,7 @@ class ReaderActivity :
|
||||
override fun onCreate(savedInstanceState: Bundle?) {
|
||||
super.onCreate(savedInstanceState)
|
||||
setContentView(ActivityReaderBinding.inflate(layoutInflater))
|
||||
readerManager = ReaderManager(supportFragmentManager, R.id.container)
|
||||
readerManager = ReaderManager(supportFragmentManager, viewBinding.container)
|
||||
supportActionBar?.setDisplayHomeAsUpEnabled(true)
|
||||
touchHelper = GridTouchHelper(this, this)
|
||||
scrollTimer = scrollTimerFactory.create(this, this)
|
||||
@@ -137,6 +137,7 @@ class ReaderActivity :
|
||||
onLoadingStateChanged(viewModel.isLoading.value)
|
||||
}
|
||||
viewModel.isScreenshotsBlockEnabled.observe(this, this::setWindowSecure)
|
||||
viewModel.isKeepScreenOnEnabled.observe(this, this::setKeepScreenOn)
|
||||
viewModel.isInfoBarEnabled.observe(this, ::onReaderBarChanged)
|
||||
viewModel.isBookmarkAdded.observe(this, this::onBookmarkStateChanged)
|
||||
viewModel.onShowToast.observeEvent(this) { msgId ->
|
||||
@@ -304,6 +305,14 @@ class ReaderActivity :
|
||||
}
|
||||
}
|
||||
|
||||
private fun setKeepScreenOn(isKeep: Boolean) {
|
||||
if (isKeep) {
|
||||
window.addFlags(WindowManager.LayoutParams.FLAG_KEEP_SCREEN_ON)
|
||||
} else {
|
||||
window.clearFlags(WindowManager.LayoutParams.FLAG_KEEP_SCREEN_ON)
|
||||
}
|
||||
}
|
||||
|
||||
private fun setUiIsVisible(isUiVisible: Boolean) {
|
||||
if (viewBinding.appbarTop.isVisible != isUiVisible) {
|
||||
if (isAnimationsEnabled) {
|
||||
|
||||
@@ -1,10 +1,12 @@
|
||||
package org.koitharu.kotatsu.reader.ui
|
||||
|
||||
import androidx.annotation.IdRes
|
||||
import androidx.fragment.app.FragmentContainerView
|
||||
import androidx.fragment.app.FragmentManager
|
||||
import androidx.fragment.app.commit
|
||||
import org.koitharu.kotatsu.R
|
||||
import org.koitharu.kotatsu.core.prefs.ReaderMode
|
||||
import org.koitharu.kotatsu.reader.ui.pager.BaseReaderFragment
|
||||
import org.koitharu.kotatsu.reader.ui.pager.doublepage.DoublePageReaderFragment
|
||||
import org.koitharu.kotatsu.reader.ui.pager.reversed.ReversedReaderFragment
|
||||
import org.koitharu.kotatsu.reader.ui.pager.standard.PagerReaderFragment
|
||||
import org.koitharu.kotatsu.reader.ui.pager.webtoon.WebtoonReaderFragment
|
||||
@@ -12,19 +14,24 @@ import java.util.EnumMap
|
||||
|
||||
class ReaderManager(
|
||||
private val fragmentManager: FragmentManager,
|
||||
@IdRes private val containerResId: Int,
|
||||
private val container: FragmentContainerView,
|
||||
) {
|
||||
|
||||
private val modeMap = EnumMap<ReaderMode, Class<out BaseReaderFragment<*>>>(ReaderMode::class.java)
|
||||
|
||||
init {
|
||||
modeMap[ReaderMode.STANDARD] = PagerReaderFragment::class.java
|
||||
val isTablet = container.resources.getBoolean(R.bool.is_tablet)
|
||||
modeMap[ReaderMode.STANDARD] = if (isTablet) {
|
||||
DoublePageReaderFragment::class.java
|
||||
} else {
|
||||
PagerReaderFragment::class.java
|
||||
}
|
||||
modeMap[ReaderMode.REVERSED] = ReversedReaderFragment::class.java
|
||||
modeMap[ReaderMode.WEBTOON] = WebtoonReaderFragment::class.java
|
||||
}
|
||||
|
||||
val currentReader: BaseReaderFragment<*>?
|
||||
get() = fragmentManager.findFragmentById(containerResId) as? BaseReaderFragment<*>
|
||||
get() = fragmentManager.findFragmentById(container.id) as? BaseReaderFragment<*>
|
||||
|
||||
val currentMode: ReaderMode?
|
||||
get() {
|
||||
@@ -36,14 +43,14 @@ class ReaderManager(
|
||||
val readerClass = requireNotNull(modeMap[newMode])
|
||||
fragmentManager.commit {
|
||||
setReorderingAllowed(true)
|
||||
replace(containerResId, readerClass, null, null)
|
||||
replace(container.id, readerClass, null, null)
|
||||
}
|
||||
}
|
||||
|
||||
fun replace(reader: BaseReaderFragment<*>) {
|
||||
/*fun replace(reader: BaseReaderFragment<*>) {
|
||||
fragmentManager.commit {
|
||||
setReorderingAllowed(true)
|
||||
replace(containerResId, reader)
|
||||
}
|
||||
}
|
||||
}*/
|
||||
}
|
||||
|
||||
@@ -113,6 +113,12 @@ class ReaderViewModel @Inject constructor(
|
||||
valueProducer = { isReaderBarEnabled },
|
||||
)
|
||||
|
||||
val isKeepScreenOnEnabled = settings.observeAsStateFlow(
|
||||
scope = viewModelScope + Dispatchers.Default,
|
||||
key = AppSettings.KEY_READER_SCREEN_ON,
|
||||
valueProducer = { isReaderKeepScreenOn },
|
||||
)
|
||||
|
||||
val isWebtoonZoomEnabled = settings.observeAsStateFlow(
|
||||
scope = viewModelScope + Dispatchers.Default,
|
||||
key = AppSettings.KEY_WEBTOON_ZOOM,
|
||||
|
||||
@@ -0,0 +1,48 @@
|
||||
package org.koitharu.kotatsu.reader.ui.pager.doublepage
|
||||
|
||||
import android.graphics.PointF
|
||||
import android.view.Gravity
|
||||
import android.widget.FrameLayout
|
||||
import androidx.lifecycle.LifecycleOwner
|
||||
import com.davemorrissey.labs.subscaleview.SubsamplingScaleImageView
|
||||
import org.koitharu.kotatsu.core.exceptions.resolve.ExceptionResolver
|
||||
import org.koitharu.kotatsu.core.os.NetworkState
|
||||
import org.koitharu.kotatsu.databinding.ItemPageBinding
|
||||
import org.koitharu.kotatsu.reader.domain.PageLoader
|
||||
import org.koitharu.kotatsu.reader.ui.config.ReaderSettings
|
||||
import org.koitharu.kotatsu.reader.ui.pager.ReaderPage
|
||||
import org.koitharu.kotatsu.reader.ui.pager.standard.PageHolder
|
||||
|
||||
class DoublePageHolder(
|
||||
owner: LifecycleOwner,
|
||||
binding: ItemPageBinding,
|
||||
loader: PageLoader,
|
||||
settings: ReaderSettings,
|
||||
networkState: NetworkState,
|
||||
exceptionResolver: ExceptionResolver,
|
||||
) : PageHolder(owner, binding, loader, settings, networkState, exceptionResolver) {
|
||||
|
||||
private val isEven: Boolean
|
||||
get() = bindingAdapterPosition and 1 == 0
|
||||
|
||||
override fun onBind(data: ReaderPage) {
|
||||
super.onBind(data)
|
||||
(binding.textViewNumber.layoutParams as FrameLayout.LayoutParams)
|
||||
.gravity = (if (isEven) Gravity.START else Gravity.END) or Gravity.BOTTOM
|
||||
}
|
||||
|
||||
override fun onImageShowing(settings: ReaderSettings) {
|
||||
with(binding.ssiv) {
|
||||
maxScale = 2f * maxOf(
|
||||
width / sWidth.toFloat(),
|
||||
height / sHeight.toFloat(),
|
||||
)
|
||||
binding.ssiv.colorFilter = settings.colorFilter?.toColorFilter()
|
||||
minimumScaleType = SubsamplingScaleImageView.SCALE_TYPE_CENTER_INSIDE
|
||||
setScaleAndCenter(
|
||||
minScale,
|
||||
PointF(if (isEven) sWidth.toFloat() else 0f, 0f),
|
||||
)
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,19 @@
|
||||
package org.koitharu.kotatsu.reader.ui.pager.doublepage
|
||||
|
||||
import android.content.Context
|
||||
import android.util.AttributeSet
|
||||
import androidx.recyclerview.widget.LinearLayoutManager
|
||||
import androidx.recyclerview.widget.RecyclerView
|
||||
|
||||
class DoublePageLayoutManager(
|
||||
context: Context,
|
||||
attrs: AttributeSet?,
|
||||
defStyleAttr: Int,
|
||||
defStyleRes: Int,
|
||||
) : LinearLayoutManager(context, attrs, defStyleAttr, defStyleRes) {
|
||||
|
||||
override fun checkLayoutParams(lp: RecyclerView.LayoutParams?): Boolean {
|
||||
lp?.width = width / 2
|
||||
return super.checkLayoutParams(lp)
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,128 @@
|
||||
package org.koitharu.kotatsu.reader.ui.pager.doublepage
|
||||
|
||||
import android.os.Bundle
|
||||
import android.view.LayoutInflater
|
||||
import android.view.ViewGroup
|
||||
import androidx.recyclerview.widget.LinearLayoutManager
|
||||
import androidx.recyclerview.widget.RecyclerView
|
||||
import com.google.android.material.snackbar.Snackbar
|
||||
import dagger.hilt.android.AndroidEntryPoint
|
||||
import kotlinx.coroutines.async
|
||||
import kotlinx.coroutines.coroutineScope
|
||||
import kotlinx.coroutines.yield
|
||||
import org.koitharu.kotatsu.R
|
||||
import org.koitharu.kotatsu.core.os.NetworkState
|
||||
import org.koitharu.kotatsu.core.util.ext.firstVisibleItemPosition
|
||||
import org.koitharu.kotatsu.databinding.FragmentReaderDoubleBinding
|
||||
import org.koitharu.kotatsu.parsers.util.toIntUp
|
||||
import org.koitharu.kotatsu.reader.domain.PageLoader
|
||||
import org.koitharu.kotatsu.reader.ui.ReaderState
|
||||
import org.koitharu.kotatsu.reader.ui.pager.BaseReaderAdapter
|
||||
import org.koitharu.kotatsu.reader.ui.pager.BaseReaderFragment
|
||||
import org.koitharu.kotatsu.reader.ui.pager.ReaderPage
|
||||
import javax.inject.Inject
|
||||
import kotlin.math.absoluteValue
|
||||
|
||||
@AndroidEntryPoint
|
||||
class DoublePageReaderFragment : BaseReaderFragment<FragmentReaderDoubleBinding>() {
|
||||
|
||||
@Inject
|
||||
lateinit var networkState: NetworkState
|
||||
|
||||
@Inject
|
||||
lateinit var pageLoader: PageLoader
|
||||
|
||||
override fun onCreateViewBinding(
|
||||
inflater: LayoutInflater,
|
||||
container: ViewGroup?,
|
||||
) = FragmentReaderDoubleBinding.inflate(inflater, container, false)
|
||||
|
||||
override fun onViewBindingCreated(
|
||||
binding: FragmentReaderDoubleBinding,
|
||||
savedInstanceState: Bundle?,
|
||||
) {
|
||||
super.onViewBindingCreated(binding, savedInstanceState)
|
||||
with(binding.recyclerView) {
|
||||
adapter = readerAdapter
|
||||
addOnScrollListener(PageScrollListener())
|
||||
DoublePageSnapHelper().attachToRecyclerView(this)
|
||||
}
|
||||
}
|
||||
|
||||
override fun onDestroyView() {
|
||||
requireViewBinding().recyclerView.adapter = null
|
||||
super.onDestroyView()
|
||||
}
|
||||
|
||||
override suspend fun onPagesChanged(pages: List<ReaderPage>, pendingState: ReaderState?) =
|
||||
coroutineScope {
|
||||
val items = async {
|
||||
requireAdapter().setItems(pages)
|
||||
yield()
|
||||
}
|
||||
if (pendingState != null) {
|
||||
val position = pages.indexOfFirst {
|
||||
it.chapterId == pendingState.chapterId && it.index == pendingState.page
|
||||
}
|
||||
items.await()
|
||||
if (position != -1) {
|
||||
requireViewBinding().recyclerView.firstVisibleItemPosition = position or 1
|
||||
notifyPageChanged(position)
|
||||
} else {
|
||||
Snackbar.make(requireView(), R.string.not_found_404, Snackbar.LENGTH_SHORT)
|
||||
.show()
|
||||
}
|
||||
} else {
|
||||
items.await()
|
||||
}
|
||||
}
|
||||
|
||||
override fun onCreateAdapter() = DoublePagesAdapter(
|
||||
lifecycleOwner = viewLifecycleOwner,
|
||||
loader = pageLoader,
|
||||
settings = viewModel.readerSettings,
|
||||
networkState = networkState,
|
||||
exceptionResolver = exceptionResolver,
|
||||
)
|
||||
|
||||
override fun switchPageBy(delta: Int) {
|
||||
switchPageTo((requireViewBinding().recyclerView.currentItem() + delta) or 1, delta.absoluteValue > 1)
|
||||
}
|
||||
|
||||
override fun switchPageTo(position: Int, smooth: Boolean) {
|
||||
requireViewBinding().recyclerView.firstVisibleItemPosition = position or 1
|
||||
}
|
||||
|
||||
override fun getCurrentState(): ReaderState? = viewBinding?.run {
|
||||
val adapter = recyclerView.adapter as? BaseReaderAdapter<*>
|
||||
val page = adapter?.getItemOrNull(recyclerView.currentItem()) ?: return@run null
|
||||
ReaderState(
|
||||
chapterId = page.chapterId,
|
||||
page = page.index,
|
||||
scroll = 0,
|
||||
)
|
||||
}
|
||||
|
||||
private fun notifyPageChanged(page: Int) {
|
||||
viewModel.onCurrentPageChanged(page)
|
||||
}
|
||||
|
||||
private fun RecyclerView.currentItem(): Int {
|
||||
val lm = layoutManager as LinearLayoutManager
|
||||
return ((lm.findFirstVisibleItemPosition() + lm.findLastVisibleItemPosition()) / 2f).toIntUp()
|
||||
}
|
||||
|
||||
private inner class PageScrollListener : RecyclerView.OnScrollListener() {
|
||||
|
||||
private var lastPage = RecyclerView.NO_POSITION
|
||||
|
||||
override fun onScrolled(recyclerView: RecyclerView, dx: Int, dy: Int) {
|
||||
super.onScrolled(recyclerView, dx, dy)
|
||||
val page = recyclerView.currentItem()
|
||||
if (page != lastPage) {
|
||||
lastPage = page
|
||||
notifyPageChanged(page)
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,280 @@
|
||||
package org.koitharu.kotatsu.reader.ui.pager.doublepage
|
||||
|
||||
import android.util.DisplayMetrics
|
||||
import android.view.View
|
||||
import android.view.animation.Interpolator
|
||||
import android.widget.Scroller
|
||||
import androidx.core.view.ViewCompat
|
||||
import androidx.recyclerview.widget.GridLayoutManager
|
||||
import androidx.recyclerview.widget.LinearLayoutManager
|
||||
import androidx.recyclerview.widget.LinearSmoothScroller
|
||||
import androidx.recyclerview.widget.OrientationHelper
|
||||
import androidx.recyclerview.widget.RecyclerView
|
||||
import androidx.recyclerview.widget.RecyclerView.SmoothScroller.ScrollVectorProvider
|
||||
import androidx.recyclerview.widget.SnapHelper
|
||||
import kotlin.math.abs
|
||||
import kotlin.math.max
|
||||
import kotlin.math.roundToInt
|
||||
|
||||
class DoublePageSnapHelper : SnapHelper() {
|
||||
|
||||
private lateinit var recyclerView: RecyclerView
|
||||
|
||||
// Total number of items in a block of view in the RecyclerView
|
||||
private var blockSize = 2
|
||||
|
||||
// Maximum number of positions to move on a fling.
|
||||
private var maxPositionsToMove = 0
|
||||
|
||||
// Width of a RecyclerView item if orientation is horizontal; height of the item if vertical
|
||||
private var itemDimension = 0
|
||||
|
||||
// Maxim blocks to move during most vigorous fling.
|
||||
private val maxFlingBlocks = 2
|
||||
|
||||
// When snapping, used to determine direction of snap.
|
||||
private var priorFirstPosition = RecyclerView.NO_POSITION
|
||||
|
||||
// Our private scroller
|
||||
private var scroller: Scroller? = null
|
||||
|
||||
// Horizontal/vertical layout helper
|
||||
private lateinit var orientationHelper: OrientationHelper
|
||||
|
||||
// LTR/RTL helper
|
||||
private lateinit var layoutDirectionHelper: LayoutDirectionHelper
|
||||
|
||||
private val snapInterpolator = Interpolator { input ->
|
||||
var t = input
|
||||
t -= 1.0f
|
||||
t * t * t + 1.0f
|
||||
}
|
||||
|
||||
@Throws(IllegalStateException::class)
|
||||
override fun attachToRecyclerView(target: RecyclerView?) {
|
||||
if (target != null) {
|
||||
recyclerView = target
|
||||
val layoutManager = recyclerView.layoutManager as LinearLayoutManager
|
||||
check(layoutManager.canScrollHorizontally()) { "RecyclerView must be scrollable" }
|
||||
orientationHelper = OrientationHelper.createHorizontalHelper(layoutManager)
|
||||
layoutDirectionHelper = LayoutDirectionHelper(ViewCompat.getLayoutDirection(recyclerView))
|
||||
scroller = Scroller(target.context, snapInterpolator)
|
||||
initItemDimensionIfNeeded(layoutManager)
|
||||
}
|
||||
super.attachToRecyclerView(recyclerView)
|
||||
}
|
||||
|
||||
override fun calculateDistanceToFinalSnap(
|
||||
layoutManager: RecyclerView.LayoutManager,
|
||||
targetView: View
|
||||
): IntArray {
|
||||
val out = IntArray(2)
|
||||
if (layoutManager.canScrollHorizontally()) {
|
||||
out[0] = layoutDirectionHelper.getScrollToAlignView(targetView)
|
||||
}
|
||||
if (layoutManager.canScrollVertically()) {
|
||||
out[1] = layoutDirectionHelper.getScrollToAlignView(targetView)
|
||||
}
|
||||
return out
|
||||
}
|
||||
|
||||
// We are flinging and need to know where we are heading.
|
||||
override fun findTargetSnapPosition(
|
||||
layoutManager: RecyclerView.LayoutManager,
|
||||
velocityX: Int, velocityY: Int
|
||||
): Int {
|
||||
val lm = layoutManager as LinearLayoutManager
|
||||
initItemDimensionIfNeeded(layoutManager)
|
||||
scroller!!.fling(0, 0, velocityX, velocityY, Int.MIN_VALUE, Int.MAX_VALUE, Int.MIN_VALUE, Int.MAX_VALUE)
|
||||
if (velocityX != 0) {
|
||||
return layoutDirectionHelper
|
||||
.getPositionsToMove(lm, scroller!!.finalX, itemDimension)
|
||||
}
|
||||
return if (velocityY != 0) {
|
||||
layoutDirectionHelper
|
||||
.getPositionsToMove(lm, scroller!!.finalY, itemDimension)
|
||||
} else RecyclerView.NO_POSITION
|
||||
}
|
||||
|
||||
// We have scrolled to the neighborhood where we will snap. Determine the snap position.
|
||||
override fun findSnapView(layoutManager: RecyclerView.LayoutManager): View? {
|
||||
// Snap to a view that is either 1) toward the bottom of the data and therefore on screen,
|
||||
// or, 2) toward the top of the data and may be off-screen.
|
||||
val snapPos: Int = calcTargetPosition(layoutManager as LinearLayoutManager)
|
||||
return if (snapPos == RecyclerView.NO_POSITION) null else layoutManager.findViewByPosition(snapPos)
|
||||
}
|
||||
|
||||
// Does the heavy lifting for findSnapView.
|
||||
private fun calcTargetPosition(layoutManager: LinearLayoutManager): Int {
|
||||
val snapPos: Int
|
||||
val firstVisiblePos = layoutManager.findFirstVisibleItemPosition()
|
||||
if (firstVisiblePos == RecyclerView.NO_POSITION) {
|
||||
return RecyclerView.NO_POSITION
|
||||
}
|
||||
initItemDimensionIfNeeded(layoutManager)
|
||||
if (firstVisiblePos >= priorFirstPosition) {
|
||||
// Scrolling toward bottom of data
|
||||
val firstCompletePosition = layoutManager.findFirstCompletelyVisibleItemPosition()
|
||||
snapPos = if (firstCompletePosition != RecyclerView.NO_POSITION
|
||||
&& firstCompletePosition % blockSize == 0
|
||||
) {
|
||||
firstCompletePosition
|
||||
} else {
|
||||
roundDownToBlockSize(firstVisiblePos + blockSize)
|
||||
}
|
||||
} else {
|
||||
// Scrolling toward top of data
|
||||
snapPos = roundDownToBlockSize(firstVisiblePos)
|
||||
// Check to see if target view exists. If it doesn't, force a smooth scroll.
|
||||
// SnapHelper only snaps to existing views and will not scroll to a non-existent one.
|
||||
// If limiting fling to single block, then the following is not needed since the
|
||||
// views are likely to be in the RecyclerView pool.
|
||||
if (layoutManager.findViewByPosition(snapPos) == null) {
|
||||
val toScroll: IntArray = layoutDirectionHelper.calculateDistanceToScroll(layoutManager, snapPos)
|
||||
recyclerView.smoothScrollBy(toScroll[0], toScroll[1], snapInterpolator)
|
||||
}
|
||||
}
|
||||
priorFirstPosition = firstVisiblePos
|
||||
return snapPos
|
||||
}
|
||||
|
||||
private fun initItemDimensionIfNeeded(layoutManager: RecyclerView.LayoutManager) {
|
||||
if (itemDimension != 0) {
|
||||
return
|
||||
}
|
||||
val child: View = layoutManager.getChildAt(0) ?: return
|
||||
if (layoutManager.canScrollHorizontally()) {
|
||||
itemDimension = child.width
|
||||
blockSize = getSpanCount(layoutManager) * (recyclerView.width / itemDimension)
|
||||
} else if (layoutManager.canScrollVertically()) {
|
||||
itemDimension = child.height
|
||||
blockSize = getSpanCount(layoutManager) * (recyclerView.height / itemDimension)
|
||||
}
|
||||
maxPositionsToMove = blockSize * maxFlingBlocks
|
||||
}
|
||||
|
||||
private fun getSpanCount(layoutManager: RecyclerView.LayoutManager): Int {
|
||||
return if (layoutManager is GridLayoutManager) layoutManager.spanCount else 1
|
||||
}
|
||||
|
||||
private fun roundDownToBlockSize(trialPosition: Int): Int {
|
||||
return trialPosition - trialPosition % blockSize
|
||||
}
|
||||
|
||||
private fun roundUpToBlockSize(trialPosition: Int): Int {
|
||||
return roundDownToBlockSize(trialPosition + blockSize - 1)
|
||||
}
|
||||
|
||||
override fun createScroller(layoutManager: RecyclerView.LayoutManager): RecyclerView.SmoothScroller? {
|
||||
return if (layoutManager !is ScrollVectorProvider) {
|
||||
null
|
||||
} else object : LinearSmoothScroller(recyclerView.context) {
|
||||
override fun onTargetFound(targetView: View, state: RecyclerView.State, action: Action) {
|
||||
val snapDistances = calculateDistanceToFinalSnap(
|
||||
recyclerView.layoutManager!!,
|
||||
targetView,
|
||||
)
|
||||
val dx = snapDistances[0]
|
||||
val dy = snapDistances[1]
|
||||
val time = calculateTimeForDeceleration(
|
||||
max(abs(dx.toDouble()), abs(dy.toDouble()))
|
||||
.toInt(),
|
||||
)
|
||||
if (time > 0) {
|
||||
action.update(dx, dy, time, snapInterpolator)
|
||||
}
|
||||
}
|
||||
|
||||
override fun calculateSpeedPerPixel(displayMetrics: DisplayMetrics): Float {
|
||||
return 40f / displayMetrics.densityDpi
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/*
|
||||
Helper class that handles calculations for LTR and RTL layouts.
|
||||
*/
|
||||
private inner class LayoutDirectionHelper(direction: Int) {
|
||||
|
||||
// Is the layout an RTL one?
|
||||
private val mIsRTL: Boolean
|
||||
|
||||
init {
|
||||
mIsRTL = direction == View.LAYOUT_DIRECTION_RTL
|
||||
}
|
||||
|
||||
/*
|
||||
Calculate the amount of scroll needed to align the target view with the layout edge.
|
||||
*/
|
||||
fun getScrollToAlignView(targetView: View?): Int {
|
||||
return if (mIsRTL) orientationHelper.getDecoratedEnd(targetView) - recyclerView.width else orientationHelper.getDecoratedStart(
|
||||
targetView,
|
||||
)
|
||||
}
|
||||
|
||||
/**
|
||||
* Calculate the distance to final snap position when the view corresponding to the snap
|
||||
* position is not currently available.
|
||||
*
|
||||
* @param layoutManager LinearLayoutManager or descendant class
|
||||
* @param targetPos - Adapter position to snap to
|
||||
* @return int[2] {x-distance in pixels, y-distance in pixels}
|
||||
*/
|
||||
fun calculateDistanceToScroll(layoutManager: LinearLayoutManager, targetPos: Int): IntArray {
|
||||
val out = IntArray(2)
|
||||
val firstVisiblePos = layoutManager.findFirstVisibleItemPosition()
|
||||
if (layoutManager.canScrollHorizontally()) {
|
||||
if (targetPos <= firstVisiblePos) { // scrolling toward top of data
|
||||
if (mIsRTL) {
|
||||
val lastView = layoutManager.findViewByPosition(layoutManager.findLastVisibleItemPosition())
|
||||
out[0] = (orientationHelper.getDecoratedEnd(lastView)
|
||||
+ (firstVisiblePos - targetPos) * itemDimension)
|
||||
} else {
|
||||
val firstView = layoutManager.findViewByPosition(firstVisiblePos)
|
||||
out[0] = (orientationHelper.getDecoratedStart(firstView)
|
||||
- (firstVisiblePos - targetPos) * itemDimension)
|
||||
}
|
||||
}
|
||||
}
|
||||
if (layoutManager.canScrollVertically()) {
|
||||
if (targetPos <= firstVisiblePos) { // scrolling toward top of data
|
||||
val firstView = layoutManager.findViewByPosition(firstVisiblePos)
|
||||
out[1] = firstView!!.top - (firstVisiblePos - targetPos) * itemDimension
|
||||
}
|
||||
}
|
||||
return out
|
||||
}
|
||||
|
||||
/*
|
||||
Calculate the number of positions to move in the RecyclerView given a scroll amount
|
||||
and the size of the items to be scrolled. Return integral multiple of mBlockSize not
|
||||
equal to zero.
|
||||
*/
|
||||
fun getPositionsToMove(llm: LinearLayoutManager, scroll: Int, itemSize: Int): Int {
|
||||
var positionsToMove: Int
|
||||
positionsToMove = roundUpToBlockSize(abs((scroll.toDouble()) / itemSize).roundToInt())
|
||||
if (positionsToMove < blockSize) {
|
||||
// Must move at least one block
|
||||
positionsToMove = blockSize
|
||||
} else if (positionsToMove > maxPositionsToMove) {
|
||||
// Clamp number of positions to move, so we don't get wild flinging.
|
||||
positionsToMove = maxPositionsToMove
|
||||
}
|
||||
if (scroll < 0) {
|
||||
positionsToMove *= -1
|
||||
}
|
||||
if (mIsRTL) {
|
||||
positionsToMove *= -1
|
||||
}
|
||||
return if (layoutDirectionHelper.isDirectionToBottom(scroll < 0)) {
|
||||
// Scrolling toward the bottom of data.
|
||||
roundDownToBlockSize(llm.findFirstVisibleItemPosition()) + positionsToMove
|
||||
} else roundDownToBlockSize(llm.findLastVisibleItemPosition()) + positionsToMove
|
||||
// Scrolling toward the top of the data.
|
||||
}
|
||||
|
||||
fun isDirectionToBottom(velocityNegative: Boolean): Boolean {
|
||||
return if (mIsRTL) velocityNegative else !velocityNegative
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,35 @@
|
||||
package org.koitharu.kotatsu.reader.ui.pager.doublepage
|
||||
|
||||
import android.view.LayoutInflater
|
||||
import android.view.ViewGroup
|
||||
import androidx.lifecycle.LifecycleOwner
|
||||
import org.koitharu.kotatsu.core.exceptions.resolve.ExceptionResolver
|
||||
import org.koitharu.kotatsu.core.os.NetworkState
|
||||
import org.koitharu.kotatsu.databinding.ItemPageBinding
|
||||
import org.koitharu.kotatsu.reader.domain.PageLoader
|
||||
import org.koitharu.kotatsu.reader.ui.config.ReaderSettings
|
||||
import org.koitharu.kotatsu.reader.ui.pager.BaseReaderAdapter
|
||||
|
||||
class DoublePagesAdapter(
|
||||
private val lifecycleOwner: LifecycleOwner,
|
||||
loader: PageLoader,
|
||||
settings: ReaderSettings,
|
||||
networkState: NetworkState,
|
||||
exceptionResolver: ExceptionResolver,
|
||||
) : BaseReaderAdapter<DoublePageHolder>(loader, settings, networkState, exceptionResolver) {
|
||||
|
||||
override fun onCreateViewHolder(
|
||||
parent: ViewGroup,
|
||||
loader: PageLoader,
|
||||
settings: ReaderSettings,
|
||||
networkState: NetworkState,
|
||||
exceptionResolver: ExceptionResolver,
|
||||
) = DoublePageHolder(
|
||||
owner = lifecycleOwner,
|
||||
binding = ItemPageBinding.inflate(LayoutInflater.from(parent.context), parent, false),
|
||||
loader = loader,
|
||||
settings = settings,
|
||||
networkState = networkState,
|
||||
exceptionResolver = exceptionResolver,
|
||||
)
|
||||
}
|
||||
@@ -8,6 +8,7 @@ import android.view.View
|
||||
import androidx.appcompat.view.ActionMode
|
||||
import androidx.appcompat.widget.SearchView
|
||||
import androidx.core.view.MenuProvider
|
||||
import androidx.core.view.inputmethod.EditorInfoCompat
|
||||
import androidx.fragment.app.viewModels
|
||||
import dagger.hilt.android.AndroidEntryPoint
|
||||
import org.koitharu.kotatsu.R
|
||||
@@ -118,6 +119,13 @@ class RemoteListFragment : MangaListFragment(), FilterOwner {
|
||||
|
||||
override fun onMenuItemActionExpand(item: MenuItem): Boolean {
|
||||
(activity as? AppBarOwner)?.appBar?.setExpanded(false, true)
|
||||
(item.actionView as? SearchView)?.run {
|
||||
imeOptions = if (viewModel.isIncognitoModeEnabled) {
|
||||
imeOptions or EditorInfoCompat.IME_FLAG_NO_PERSONALIZED_LEARNING
|
||||
} else {
|
||||
imeOptions and EditorInfoCompat.IME_FLAG_NO_PERSONALIZED_LEARNING.inv()
|
||||
}
|
||||
}
|
||||
return true
|
||||
}
|
||||
|
||||
|
||||
@@ -10,8 +10,10 @@ import kotlinx.coroutines.currentCoroutineContext
|
||||
import kotlinx.coroutines.isActive
|
||||
import kotlinx.coroutines.withContext
|
||||
import org.koitharu.kotatsu.core.db.MangaDatabase
|
||||
import org.koitharu.kotatsu.core.db.entity.toEntity
|
||||
import org.koitharu.kotatsu.core.db.entity.toManga
|
||||
import org.koitharu.kotatsu.core.db.entity.toMangaTag
|
||||
import org.koitharu.kotatsu.core.db.entity.toMangaTagsList
|
||||
import org.koitharu.kotatsu.core.prefs.AppSettings
|
||||
import org.koitharu.kotatsu.explore.data.MangaSourcesRepository
|
||||
import org.koitharu.kotatsu.parsers.model.ContentType
|
||||
@@ -19,6 +21,7 @@ import org.koitharu.kotatsu.parsers.model.Manga
|
||||
import org.koitharu.kotatsu.parsers.model.MangaSource
|
||||
import org.koitharu.kotatsu.parsers.model.MangaTag
|
||||
import org.koitharu.kotatsu.parsers.util.levenshteinDistance
|
||||
import org.koitharu.kotatsu.parsers.util.mapToSet
|
||||
import org.koitharu.kotatsu.search.ui.MangaSuggestionsProvider
|
||||
import javax.inject.Inject
|
||||
|
||||
@@ -93,8 +96,17 @@ class MangaSearchRepository @Inject constructor(
|
||||
query.isNotEmpty() -> db.tagsDao.findTags("%$query%", limit)
|
||||
source != null -> db.tagsDao.findPopularTags(source.name, limit)
|
||||
else -> db.tagsDao.findPopularTags(limit)
|
||||
}.map {
|
||||
it.toMangaTag()
|
||||
}.toMangaTagsList()
|
||||
}
|
||||
|
||||
suspend fun getTagsSuggestion(tags: Set<MangaTag>): List<MangaTag> {
|
||||
val ids = tags.mapToSet { it.toEntity().id }
|
||||
return if (ids.size == 1) {
|
||||
db.tagsDao.findRelatedTags(ids.first())
|
||||
} else {
|
||||
db.tagsDao.findRelatedTags(ids)
|
||||
}.mapNotNull { x ->
|
||||
if (x.id in ids) null else x.toMangaTag()
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@@ -59,7 +59,6 @@ class MultiSearchActivity :
|
||||
override fun onCreate(savedInstanceState: Bundle?) {
|
||||
super.onCreate(savedInstanceState)
|
||||
setContentView(ActivitySearchMultiBinding.inflate(layoutInflater))
|
||||
window.statusBarColor = ContextCompat.getColor(this, R.color.dim_statusbar)
|
||||
title = viewModel.query
|
||||
|
||||
val itemCLickListener = OnListItemClickListener<MultiSearchListModel> { item, view ->
|
||||
|
||||
@@ -29,6 +29,7 @@ import org.koitharu.kotatsu.databinding.ActivitySettingsBinding
|
||||
import org.koitharu.kotatsu.main.ui.owners.AppBarOwner
|
||||
import org.koitharu.kotatsu.parsers.model.MangaSource
|
||||
import org.koitharu.kotatsu.settings.about.AboutSettingsFragment
|
||||
import org.koitharu.kotatsu.settings.about.AppUpdateDialog
|
||||
import org.koitharu.kotatsu.settings.sources.SourceSettingsFragment
|
||||
import org.koitharu.kotatsu.settings.sources.SourcesManageFragment
|
||||
import org.koitharu.kotatsu.settings.tracker.TrackerSettingsFragment
|
||||
@@ -41,6 +42,8 @@ class SettingsActivity :
|
||||
AppBarOwner,
|
||||
FragmentManager.OnBackStackChangedListener {
|
||||
|
||||
val appUpdateDialog = AppUpdateDialog(this)
|
||||
|
||||
override val appBar: AppBarLayout
|
||||
get() = viewBinding.appbar
|
||||
|
||||
|
||||
@@ -20,6 +20,7 @@ import org.koitharu.kotatsu.core.ui.BasePreferenceFragment
|
||||
import org.koitharu.kotatsu.core.util.ShareHelper
|
||||
import org.koitharu.kotatsu.core.util.ext.observe
|
||||
import org.koitharu.kotatsu.core.util.ext.observeEvent
|
||||
import org.koitharu.kotatsu.settings.SettingsActivity
|
||||
import javax.inject.Inject
|
||||
|
||||
@AndroidEntryPoint
|
||||
@@ -76,7 +77,7 @@ class AboutSettingsFragment : BasePreferenceFragment(R.string.about) {
|
||||
Snackbar.make(listView, R.string.no_update_available, Snackbar.LENGTH_SHORT).show()
|
||||
return
|
||||
}
|
||||
AppUpdateDialog(context ?: return).show(version)
|
||||
(activity as SettingsActivity).appUpdateDialog.show(version)
|
||||
}
|
||||
|
||||
private fun openLink(url: String, title: CharSequence?) {
|
||||
|
||||
@@ -1,10 +1,14 @@
|
||||
package org.koitharu.kotatsu.settings.about
|
||||
|
||||
import android.Manifest
|
||||
import android.app.DownloadManager
|
||||
import android.content.Context
|
||||
import android.content.Intent
|
||||
import android.os.Build
|
||||
import android.os.Environment
|
||||
import android.widget.Toast
|
||||
import androidx.activity.result.contract.ActivityResultContracts
|
||||
import androidx.appcompat.app.AppCompatActivity
|
||||
import androidx.core.net.toUri
|
||||
import androidx.core.text.buildSpannedString
|
||||
import com.google.android.material.dialog.MaterialAlertDialogBuilder
|
||||
@@ -14,19 +18,32 @@ import org.koitharu.kotatsu.core.github.AppVersion
|
||||
import org.koitharu.kotatsu.core.util.FileSize
|
||||
import com.google.android.material.R as materialR
|
||||
|
||||
class AppUpdateDialog(private val context: Context) {
|
||||
class AppUpdateDialog(private val activity: AppCompatActivity) {
|
||||
|
||||
private lateinit var latestVersion: AppVersion
|
||||
|
||||
private val permissionRequest = activity.registerForActivityResult(
|
||||
ActivityResultContracts.RequestPermission(),
|
||||
) {
|
||||
if (it) {
|
||||
downloadUpdateImpl()
|
||||
} else {
|
||||
openInBrowser()
|
||||
}
|
||||
}
|
||||
|
||||
fun show(version: AppVersion) {
|
||||
latestVersion = version
|
||||
val message = buildSpannedString {
|
||||
append(context.getString(R.string.new_version_s, version.name))
|
||||
append(activity.getString(R.string.new_version_s, version.name))
|
||||
appendLine()
|
||||
append(context.getString(R.string.size_s, FileSize.BYTES.format(context, version.apkSize)))
|
||||
append(activity.getString(R.string.size_s, FileSize.BYTES.format(activity, version.apkSize)))
|
||||
appendLine()
|
||||
appendLine()
|
||||
append(Markwon.create(context).toMarkdown(version.description))
|
||||
append(Markwon.create(activity).toMarkdown(version.description))
|
||||
}
|
||||
MaterialAlertDialogBuilder(
|
||||
context,
|
||||
activity,
|
||||
materialR.style.ThemeOverlay_Material3_MaterialAlertDialog_Centered,
|
||||
)
|
||||
.setTitle(R.string.app_update_available)
|
||||
@@ -34,24 +51,38 @@ class AppUpdateDialog(private val context: Context) {
|
||||
.setIcon(R.drawable.ic_app_update)
|
||||
.setNeutralButton(R.string.open_in_browser) { _, _ ->
|
||||
val intent = Intent(Intent.ACTION_VIEW, version.url.toUri())
|
||||
context.startActivity(Intent.createChooser(intent, context.getString(R.string.open_in_browser)))
|
||||
activity.startActivity(Intent.createChooser(intent, activity.getString(R.string.open_in_browser)))
|
||||
}.setPositiveButton(R.string.update) { _, _ ->
|
||||
downloadUpdate(version)
|
||||
downloadUpdate()
|
||||
}.setNegativeButton(android.R.string.cancel, null)
|
||||
.setCancelable(false)
|
||||
.create()
|
||||
.show()
|
||||
}
|
||||
|
||||
private fun downloadUpdate(version: AppVersion) {
|
||||
private fun downloadUpdate() {
|
||||
if (Build.VERSION.SDK_INT < Build.VERSION_CODES.Q) {
|
||||
permissionRequest.launch(Manifest.permission.WRITE_EXTERNAL_STORAGE)
|
||||
} else {
|
||||
downloadUpdateImpl()
|
||||
}
|
||||
}
|
||||
|
||||
private fun downloadUpdateImpl() {
|
||||
val version = latestVersion
|
||||
val url = version.apkUrl.toUri()
|
||||
val dm = context.getSystemService(Context.DOWNLOAD_SERVICE) as DownloadManager
|
||||
val dm = activity.getSystemService(Context.DOWNLOAD_SERVICE) as DownloadManager
|
||||
val request = DownloadManager.Request(url)
|
||||
.setTitle("${context.getString(R.string.app_name)} v${version.name}")
|
||||
.setTitle("${activity.getString(R.string.app_name)} v${version.name}")
|
||||
.setDestinationInExternalPublicDir(Environment.DIRECTORY_DOWNLOADS, url.lastPathSegment)
|
||||
.setNotificationVisibility(DownloadManager.Request.VISIBILITY_VISIBLE_NOTIFY_COMPLETED)
|
||||
.setMimeType("application/vnd.android.package-archive")
|
||||
dm.enqueue(request)
|
||||
Toast.makeText(context, R.string.download_started, Toast.LENGTH_SHORT).show()
|
||||
Toast.makeText(activity, R.string.download_started, Toast.LENGTH_SHORT).show()
|
||||
}
|
||||
|
||||
private fun openInBrowser() {
|
||||
val intent = Intent(Intent.ACTION_VIEW, latestVersion.url.toUri())
|
||||
activity.startActivity(Intent.createChooser(intent, activity.getString(R.string.open_in_browser)))
|
||||
}
|
||||
}
|
||||
|
||||
@@ -1,5 +1,6 @@
|
||||
package org.koitharu.kotatsu.suggestions.ui
|
||||
|
||||
import android.annotation.SuppressLint
|
||||
import android.app.PendingIntent
|
||||
import android.content.Context
|
||||
import android.content.pm.ServiceInfo
|
||||
@@ -10,6 +11,7 @@ import androidx.core.app.NotificationCompat
|
||||
import androidx.core.app.NotificationManagerCompat
|
||||
import androidx.core.app.PendingIntentCompat
|
||||
import androidx.core.text.HtmlCompat
|
||||
import androidx.core.text.bold
|
||||
import androidx.core.text.buildSpannedString
|
||||
import androidx.core.text.parseAsHtml
|
||||
import androidx.hilt.work.HiltWorker
|
||||
@@ -48,6 +50,7 @@ import org.koitharu.kotatsu.core.prefs.AppSettings
|
||||
import org.koitharu.kotatsu.core.util.ext.almostEquals
|
||||
import org.koitharu.kotatsu.core.util.ext.asArrayList
|
||||
import org.koitharu.kotatsu.core.util.ext.awaitUniqueWorkInfoByName
|
||||
import org.koitharu.kotatsu.core.util.ext.checkNotificationPermission
|
||||
import org.koitharu.kotatsu.core.util.ext.flatten
|
||||
import org.koitharu.kotatsu.core.util.ext.printStackTraceDebug
|
||||
import org.koitharu.kotatsu.core.util.ext.sanitize
|
||||
@@ -163,7 +166,7 @@ class SuggestionsWorker @AssistedInject constructor(
|
||||
.sortedBy { it.relevance }
|
||||
.take(MAX_RESULTS)
|
||||
suggestionRepository.replace(suggestions)
|
||||
if (appSettings.isSuggestionsNotificationAvailable) {
|
||||
if (appSettings.isSuggestionsNotificationAvailable && applicationContext.checkNotificationPermission()) {
|
||||
for (i in 0..3) {
|
||||
try {
|
||||
val manga = suggestions[Random.nextInt(0, suggestions.size / 3)]
|
||||
@@ -221,10 +224,8 @@ class SuggestionsWorker @AssistedInject constructor(
|
||||
e.printStackTraceDebug()
|
||||
}.getOrDefault(emptyList())
|
||||
|
||||
@SuppressLint("MissingPermission")
|
||||
private suspend fun showNotification(manga: Manga) {
|
||||
if (!notificationManager.areNotificationsEnabled()) {
|
||||
return
|
||||
}
|
||||
val channel = NotificationChannelCompat.Builder(MANGA_CHANNEL_ID, NotificationManagerCompat.IMPORTANCE_DEFAULT)
|
||||
.setName(applicationContext.getString(R.string.suggestions))
|
||||
.setDescription(applicationContext.getString(R.string.suggestions_summary))
|
||||
@@ -255,17 +256,19 @@ class SuggestionsWorker @AssistedInject constructor(
|
||||
style.bigText(
|
||||
buildSpannedString {
|
||||
append(tagsText)
|
||||
appendLine()
|
||||
append(description)
|
||||
val chaptersCount = manga.chapters?.size ?: 0
|
||||
appendLine()
|
||||
append(
|
||||
applicationContext.resources.getQuantityString(
|
||||
R.plurals.chapters,
|
||||
chaptersCount,
|
||||
chaptersCount,
|
||||
),
|
||||
)
|
||||
bold {
|
||||
append(
|
||||
applicationContext.resources.getQuantityString(
|
||||
R.plurals.chapters,
|
||||
chaptersCount,
|
||||
chaptersCount,
|
||||
),
|
||||
)
|
||||
}
|
||||
appendLine()
|
||||
append(description)
|
||||
},
|
||||
)
|
||||
style.setBigContentTitle(title)
|
||||
|
||||
@@ -13,10 +13,11 @@ import org.koitharu.kotatsu.core.db.MangaDatabase
|
||||
import org.koitharu.kotatsu.core.db.entity.MangaEntity
|
||||
import org.koitharu.kotatsu.core.db.entity.toManga
|
||||
import org.koitharu.kotatsu.core.model.FavouriteCategory
|
||||
import org.koitharu.kotatsu.core.model.isLocal
|
||||
import org.koitharu.kotatsu.core.util.ext.mapItems
|
||||
import org.koitharu.kotatsu.favourites.data.toFavouriteCategory
|
||||
import org.koitharu.kotatsu.local.data.LocalMangaRepository
|
||||
import org.koitharu.kotatsu.parsers.model.Manga
|
||||
import org.koitharu.kotatsu.parsers.model.MangaSource
|
||||
import org.koitharu.kotatsu.parsers.util.mapToSet
|
||||
import org.koitharu.kotatsu.tracker.data.TrackEntity
|
||||
import org.koitharu.kotatsu.tracker.data.TrackLogEntity
|
||||
@@ -27,12 +28,14 @@ import org.koitharu.kotatsu.tracker.domain.model.TrackingLogItem
|
||||
import java.util.Date
|
||||
import java.util.concurrent.atomic.AtomicBoolean
|
||||
import javax.inject.Inject
|
||||
import javax.inject.Provider
|
||||
|
||||
private const val NO_ID = 0L
|
||||
|
||||
@Reusable
|
||||
class TrackingRepository @Inject constructor(
|
||||
private val db: MangaDatabase,
|
||||
private val localMangaRepositoryProvider: Provider<LocalMangaRepository>,
|
||||
) {
|
||||
|
||||
private var isGcCalled = AtomicBoolean(false)
|
||||
@@ -66,12 +69,17 @@ class TrackingRepository @Inject constructor(
|
||||
val idSet = HashSet<Long>()
|
||||
val result = ArrayList<MangaTracking>(mangaList.size)
|
||||
for (item in mangaList) {
|
||||
if (item.source == MangaSource.LOCAL || !idSet.add(item.id)) {
|
||||
val manga = if (item.isLocal) {
|
||||
localMangaRepositoryProvider.get().getRemoteManga(item) ?: continue
|
||||
} else {
|
||||
item
|
||||
}
|
||||
if (!idSet.add(manga.id)) {
|
||||
continue
|
||||
}
|
||||
val track = tracks[item.id]?.lastOrNull()
|
||||
val track = tracks[manga.id]?.lastOrNull()
|
||||
result += MangaTracking(
|
||||
manga = item,
|
||||
manga = manga,
|
||||
lastChapterId = track?.lastChapterId ?: NO_ID,
|
||||
lastCheck = track?.lastCheck?.takeUnless { it == 0L }?.let(::Date),
|
||||
)
|
||||
|
||||
@@ -6,7 +6,6 @@ import com.hannesdorfmann.adapterdelegates4.dsl.adapterDelegateViewBinding
|
||||
import org.koitharu.kotatsu.R
|
||||
import org.koitharu.kotatsu.core.ui.list.OnListItemClickListener
|
||||
import org.koitharu.kotatsu.core.util.ext.enqueueWith
|
||||
import org.koitharu.kotatsu.core.util.ext.isBold
|
||||
import org.koitharu.kotatsu.core.util.ext.newImageRequest
|
||||
import org.koitharu.kotatsu.core.util.ext.source
|
||||
import org.koitharu.kotatsu.databinding.ItemFeedBinding
|
||||
@@ -26,8 +25,9 @@ fun feedItemAD(
|
||||
}
|
||||
|
||||
bind {
|
||||
binding.textViewTitle.isBold = item.isNew
|
||||
binding.textViewSummary.isBold = item.isNew
|
||||
val alpha = if (item.isNew) 1f else 0.5f
|
||||
binding.textViewTitle.alpha = alpha
|
||||
binding.textViewSummary.alpha = alpha
|
||||
binding.imageViewCover.newImageRequest(lifecycleOwner, item.imageUrl)?.run {
|
||||
placeholder(R.drawable.ic_placeholder)
|
||||
fallback(R.drawable.ic_placeholder)
|
||||
|
||||
@@ -1,4 +1,4 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<selector xmlns:android="http://schemas.android.com/apk/res/android">
|
||||
<item android:alpha="0.6" android:color="?attr/colorSurface" />
|
||||
<item android:alpha="0.6" android:color="?android:colorBackground" />
|
||||
</selector>
|
||||
|
||||
11
app/src/main/res/drawable/ic_state_abandoned.xml
Normal file
11
app/src/main/res/drawable/ic_state_abandoned.xml
Normal file
@@ -0,0 +1,11 @@
|
||||
<vector
|
||||
xmlns:android="http://schemas.android.com/apk/res/android"
|
||||
android:width="24dp"
|
||||
android:height="24dp"
|
||||
android:tint="?attr/colorControlNormal"
|
||||
android:viewportWidth="24"
|
||||
android:viewportHeight="24">
|
||||
<path
|
||||
android:fillColor="@android:color/white"
|
||||
android:pathData="M19,6.41L17.59,5L12,10.59L6.41,5L5,6.41L10.59,12L5,17.59L6.41,19L12,13.41L17.59,19L19,17.59L13.41,12L19,6.41Z" />
|
||||
</vector>
|
||||
@@ -4,8 +4,7 @@
|
||||
xmlns:app="http://schemas.android.com/apk/res-auto"
|
||||
xmlns:tools="http://schemas.android.com/tools"
|
||||
android:layout_width="match_parent"
|
||||
android:layout_height="match_parent"
|
||||
android:keepScreenOn="true">
|
||||
android:layout_height="match_parent">
|
||||
|
||||
<androidx.fragment.app.FragmentContainerView
|
||||
android:id="@+id/container"
|
||||
|
||||
10
app/src/main/res/layout/fragment_reader_double.xml
Normal file
10
app/src/main/res/layout/fragment_reader_double.xml
Normal file
@@ -0,0 +1,10 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<androidx.recyclerview.widget.RecyclerView
|
||||
xmlns:android="http://schemas.android.com/apk/res/android"
|
||||
xmlns:app="http://schemas.android.com/apk/res-auto"
|
||||
android:id="@+id/recyclerView"
|
||||
android:layout_width="match_parent"
|
||||
android:layout_height="match_parent"
|
||||
android:defaultFocusHighlightEnabled="false"
|
||||
android:orientation="horizontal"
|
||||
app:layoutManager="org.koitharu.kotatsu.reader.ui.pager.doublepage.DoublePageLayoutManager" />
|
||||
@@ -30,7 +30,7 @@
|
||||
android:layout_marginEnd="8dp"
|
||||
android:ellipsize="end"
|
||||
android:maxLines="1"
|
||||
android:textAppearance="?attr/textAppearanceBodyLarge"
|
||||
android:textAppearance="?attr/textAppearanceTitleSmall"
|
||||
app:layout_constraintBottom_toTopOf="@+id/textView_summary"
|
||||
app:layout_constraintEnd_toEndOf="parent"
|
||||
app:layout_constraintStart_toEndOf="@+id/imageView_cover"
|
||||
@@ -45,7 +45,7 @@
|
||||
android:layout_marginEnd="8dp"
|
||||
android:ellipsize="end"
|
||||
android:maxLines="1"
|
||||
android:textAppearance="?attr/textAppearanceBodyMedium"
|
||||
android:textAppearance="?attr/textAppearanceBodySmall"
|
||||
app:layout_constraintBottom_toBottomOf="@+id/imageView_cover"
|
||||
app:layout_constraintEnd_toEndOf="parent"
|
||||
app:layout_constraintStart_toEndOf="@+id/imageView_cover"
|
||||
|
||||
@@ -31,7 +31,7 @@
|
||||
android:layout_marginEnd="8dp"
|
||||
android:ellipsize="end"
|
||||
android:maxLines="1"
|
||||
android:textAppearance="?attr/textAppearanceBodyLarge"
|
||||
android:textAppearance="?attr/textAppearanceTitleSmall"
|
||||
app:layout_constraintBottom_toTopOf="@+id/textView_subtitle"
|
||||
app:layout_constraintEnd_toEndOf="parent"
|
||||
app:layout_constraintStart_toEndOf="@+id/imageView_cover"
|
||||
@@ -46,7 +46,7 @@
|
||||
android:layout_marginEnd="8dp"
|
||||
android:ellipsize="end"
|
||||
android:maxLines="1"
|
||||
android:textAppearance="?attr/textAppearanceBodyMedium"
|
||||
android:textAppearance="?attr/textAppearanceBodySmall"
|
||||
app:layout_constraintBottom_toBottomOf="@+id/imageView_cover"
|
||||
app:layout_constraintEnd_toEndOf="parent"
|
||||
app:layout_constraintStart_toEndOf="@+id/imageView_cover"
|
||||
|
||||
@@ -40,8 +40,7 @@
|
||||
android:layout_marginEnd="12dp"
|
||||
android:ellipsize="end"
|
||||
android:maxLines="2"
|
||||
android:textAppearance="?attr/textAppearanceTitleLarge"
|
||||
android:textSize="20sp"
|
||||
android:textAppearance="?attr/textAppearanceTitleMedium"
|
||||
app:layout_constraintEnd_toEndOf="parent"
|
||||
app:layout_constraintStart_toEndOf="@+id/imageView_cover"
|
||||
app:layout_constraintTop_toTopOf="parent"
|
||||
@@ -57,7 +56,7 @@
|
||||
android:ellipsize="end"
|
||||
android:gravity="center_vertical"
|
||||
android:maxLines="2"
|
||||
android:textAppearance="?attr/textAppearanceSubtitle1"
|
||||
android:textAppearance="?attr/textAppearanceBodySmall"
|
||||
app:layout_constraintEnd_toEndOf="parent"
|
||||
app:layout_constraintStart_toEndOf="@+id/imageView_cover"
|
||||
app:layout_constraintTop_toBottomOf="@+id/textView_title"
|
||||
|
||||
@@ -479,4 +479,10 @@
|
||||
<string name="directories">Каталогі</string>
|
||||
<string name="main_screen_sections">Раздзелы галоўнага экрана</string>
|
||||
<string name="to_top">Уверх</string>
|
||||
<string name="zoom_in">Павялічыць</string>
|
||||
<string name="reader_zoom_buttons_summary">Ці паказваць кнопкі кіравання маштабаваннем у правым ніжнім куце</string>
|
||||
<string name="reader_zoom_buttons">Паказаць кнопкі маштабавання</string>
|
||||
<string name="zoom_out">Зменшыць</string>
|
||||
<string name="keep_screen_on">Трымаць экран уключаным</string>
|
||||
<string name="keep_screen_on_summary">Ня выключаць экран падчас чытання мангі</string>
|
||||
</resources>
|
||||
@@ -483,4 +483,6 @@
|
||||
<string name="reader_zoom_buttons_summary">Mostrar o no los botones de control del zoom en la esquina inferior derecha</string>
|
||||
<string name="reader_zoom_buttons">Mostrar los botones del zoom</string>
|
||||
<string name="zoom_out">Alejar</string>
|
||||
<string name="keep_screen_on">Mantener pantalla encendida</string>
|
||||
<string name="keep_screen_on_summary">No apagues la pantalla mientras lees manga</string>
|
||||
</resources>
|
||||
@@ -473,4 +473,14 @@
|
||||
<string name="too_many_requests_message">Masyadong maraming request. Subukang ulit mamaya</string>
|
||||
<string name="default_section">Default na seksyon</string>
|
||||
<string name="manga_list">Listahan ng Manga</string>
|
||||
<string name="zoom_in">Mag-zoom paloob</string>
|
||||
<string name="reader_zoom_buttons_summary">Kung magpapakita ng mga button ng kontrol ng zoom sa kanang sulok sa ibaba</string>
|
||||
<string name="on_device">Sa device</string>
|
||||
<string name="moved_to_top">Nailipat sa itaas</string>
|
||||
<string name="items_limit_exceeded">Wala nang mga aytem na pwedeng idagdag</string>
|
||||
<string name="directories">Mga Directory</string>
|
||||
<string name="reader_zoom_buttons">Ipakita ang mga button ng pag-zoom</string>
|
||||
<string name="main_screen_sections">Mga pangunahing seksyon ng screen</string>
|
||||
<string name="zoom_out">Mag-zoom palabas</string>
|
||||
<string name="to_top">Sa taas</string>
|
||||
</resources>
|
||||
@@ -479,4 +479,8 @@
|
||||
<string name="main_screen_sections">メイン画面のセクション</string>
|
||||
<string name="moved_to_top">上部に移動しました</string>
|
||||
<string name="to_top">最上部へ移動</string>
|
||||
<string name="zoom_in">ズームイン</string>
|
||||
<string name="reader_zoom_buttons_summary">右下にズームコントロールボタンを表示するかどうか</string>
|
||||
<string name="reader_zoom_buttons">ズームボタンを表示</string>
|
||||
<string name="zoom_out">ズームアウト</string>
|
||||
</resources>
|
||||
@@ -1,27 +1,32 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<resources>
|
||||
<plurals name="items">
|
||||
<item quantity="one">%1$d item</item>
|
||||
<item quantity="other">%1$d itens</item>
|
||||
</plurals>
|
||||
<plurals name="chapters">
|
||||
<item quantity="one">%1$d capítulo</item>
|
||||
<item quantity="other">%1$d capítulos</item>
|
||||
</plurals>
|
||||
<plurals name="new_chapters">
|
||||
<item quantity="one">%1$d novo capítulo</item>
|
||||
<item quantity="other">%1$d novos capítulos</item>
|
||||
</plurals>
|
||||
<plurals name="minutes_ago">
|
||||
<item quantity="one">%1$d minuto atrás</item>
|
||||
<item quantity="other">%1$d minutos atrás</item>
|
||||
</plurals>
|
||||
<plurals name="hours_ago">
|
||||
<item quantity="one">%1$d hora atrás</item>
|
||||
<item quantity="other">%1$d horas atrás</item>
|
||||
</plurals>
|
||||
<plurals name="days_ago">
|
||||
<item quantity="one">%1$d dia atrás</item>
|
||||
<item quantity="other">%1$d dias atrás</item>
|
||||
</plurals>
|
||||
</resources>
|
||||
<plurals name="items">
|
||||
<item quantity="one">%1$d item</item>
|
||||
<item quantity="other">%1$d itens</item>
|
||||
</plurals>
|
||||
<plurals name="chapters">
|
||||
<item quantity="one">%1$d capítulo</item>
|
||||
<item quantity="other">%1$d capítulos</item>
|
||||
</plurals>
|
||||
<plurals name="new_chapters">
|
||||
<item quantity="one">%1$d novo capítulo</item>
|
||||
<item quantity="other">%1$d novos capítulos</item>
|
||||
</plurals>
|
||||
<plurals name="minutes_ago">
|
||||
<item quantity="one">%1$d minuto atrás</item>
|
||||
<item quantity="other">%1$d minutos atrás</item>
|
||||
</plurals>
|
||||
<plurals name="hours_ago">
|
||||
<item quantity="one">%1$d hora atrás</item>
|
||||
<item quantity="other">%1$d horas atrás</item>
|
||||
</plurals>
|
||||
<plurals name="days_ago">
|
||||
<item quantity="one">%1$d dia atrás</item>
|
||||
<item quantity="other">%1$d dias atrás</item>
|
||||
</plurals>
|
||||
<plurals name="months_ago">
|
||||
<item quantity="one">%1$d mês atrás</item>
|
||||
<item quantity="many">%1$d meses atrás</item>
|
||||
<item quantity="other"></item>
|
||||
</plurals>
|
||||
</resources>
|
||||
@@ -90,7 +90,7 @@
|
||||
<string name="text_empty_holder_primary">Tá meio vazio aqui…</string>
|
||||
<string name="text_search_holder_secondary">Tente reformular a consulta.</string>
|
||||
<string name="text_history_holder_primary">O que você ler será exibido aqui</string>
|
||||
<string name="text_history_holder_secondary">Encontre o que ler no menu lateral.</string>
|
||||
<string name="text_history_holder_secondary">Encontre o que ler na aba «Explorar»</string>
|
||||
<string name="text_local_holder_primary">Salve algo primeiro</string>
|
||||
<string name="manga_shelf">Prateleira</string>
|
||||
<string name="recent_manga">Recente</string>
|
||||
@@ -281,7 +281,7 @@
|
||||
<string name="show_reading_indicators">Mostrar indicadores de progresso de leitura</string>
|
||||
<string name="data_deletion">Exclusão de dados</string>
|
||||
<string name="show_reading_indicators_summary">Mostrar percentual de leitura no histórico e nos favoritos</string>
|
||||
<string name="exclude_nsfw_from_history_summary">Mangás marcados como +18 nunca serão adicionados ao histórico e seu progresso não sera salvo</string>
|
||||
<string name="exclude_nsfw_from_history_summary">Mangás marcados como +18 não serão adicionados ao histórico e o progresso não será salvo</string>
|
||||
<string name="show_all">Mostrar tudo</string>
|
||||
<string name="clear_all_history">Limpar todo o histórico</string>
|
||||
<string name="last_2_hours">Ultimas 2 horas</string>
|
||||
@@ -361,14 +361,14 @@
|
||||
<string name="theme_name_rikka">Rikka</string>
|
||||
<string name="theme_name_sakura">Sakura</string>
|
||||
<string name="source_disabled">Fonte desativada</string>
|
||||
<string name="enable_logging_summary">Gravar algumas ações para propósitos de depuração</string>
|
||||
<string name="enable_logging_summary">Gravar algumas ações para propósitos de depuração. Não ative se você não sabe o que está fazendo</string>
|
||||
<string name="theme_name_mamimi">Mamimi</string>
|
||||
<string name="theme_name_miku">Miku</string>
|
||||
<string name="theme_name_kanade">Kanade</string>
|
||||
<string name="scrobbling_empty_hint">Para acompanhar o progresso da leitura, selecione Menu → Rastrear na tela de detalhes do mangá.</string>
|
||||
<string name="clear_new_chapters_counters">Também informações claras sobre novos capítulos</string>
|
||||
<string name="sync_auth_hint">Você pode entrar em uma conta existente ou criar uma nova</string>
|
||||
<string name="allow_unstable_updates_summary">Propor atualizações para versões beta do aplicativo</string>
|
||||
<string name="allow_unstable_updates_summary">Receber notificações sobre versões beta</string>
|
||||
<string name="comics_archive_import_description">Você pode selecionar um ou mais arquivos .cbz ou .zip, cada arquivo será reconhecido como um mangá separado.</string>
|
||||
<string name="folder_with_images_import_description">Você pode selecionar um diretório com arquivos ou imagens. Cada arquivo (ou subdiretório) será reconhecido como um capítulo.</string>
|
||||
<string name="speed">Velocidade</string>
|
||||
@@ -429,7 +429,7 @@
|
||||
<string name="sync_host_description">Você pode usar um servidor de sincronização auto-hospedado ou um padrão. Não mude isso se não tiver certeza do que está fazendo.</string>
|
||||
<string name="ignore_ssl_errors">Ignorar erros de SSL</string>
|
||||
<string name="mirror_switching">Escolher espelho automaticamente</string>
|
||||
<string name="mirror_switching_summary">Alternar domínios automaticamente para fontes remotas em caso de erros, se espelhos estiverem disponíveis</string>
|
||||
<string name="mirror_switching_summary">Troca automática de domínios para fontes de mangá em caso de erros, se houver espelhos disponíveis</string>
|
||||
<string name="pause">Pausar</string>
|
||||
<string name="suggestions_enable_prompt">Quer receber sugestões personalizadas de mangás\?</string>
|
||||
<string name="suggestion_manga">Sugestão: %s</string>
|
||||
@@ -443,4 +443,44 @@
|
||||
<string name="pick_custom_directory">Escolha um diretório personalizado</string>
|
||||
<string name="no_access_to_file">Você não tem acesso a esse arquivo ou diretório</string>
|
||||
<string name="local_manga_directories">Diretórios locais de mangá</string>
|
||||
<string name="languages">Idiomas</string>
|
||||
<string name="zoom_in">Aumentar o zoom</string>
|
||||
<string name="captcha_required_summary">%s requer que um captcha seja resolvido para funcionar</string>
|
||||
<string name="progress">Progresso</string>
|
||||
<string name="error_corrupted_file">Dados inválidos ou o arquivo está corrompido</string>
|
||||
<string name="related_manga_summary">Mostra uma lista de mangás relacionados. Em alguns casos, ela pode estar incorreta ou ausente</string>
|
||||
<string name="reader_zoom_buttons_summary">Se deseja mostrar os botões de controle de zoom no canto inferior direito</string>
|
||||
<string name="tracker_wifi_only_summary">Não procurar por novos capítulos usando dados móveis</string>
|
||||
<string name="order_added">Adicionado</string>
|
||||
<string name="on_device">No dispositivo</string>
|
||||
<string name="moved_to_top">Movido pra cima</string>
|
||||
<string name="data_not_restored_text">Certifique-se de ter selecionado o arquivo de backup correto</string>
|
||||
<string name="view_list">Ver lista</string>
|
||||
<string name="unknown">Desconhecido</string>
|
||||
<string name="in_progress">Em andamento</string>
|
||||
<string name="items_limit_exceeded">Não é possível adicionar mais itens</string>
|
||||
<string name="data_not_restored">Os dados não foram restaurados</string>
|
||||
<string name="directories">Diretórios</string>
|
||||
<string name="manage_categories">Gerenciar categorias</string>
|
||||
<string name="color_light">Claro</string>
|
||||
<string name="search_hint">Coloque o nome do mangá, gênero ou fonte</string>
|
||||
<string name="description">Descrição</string>
|
||||
<string name="reader_zoom_buttons">Mostrar botões de zoom</string>
|
||||
<string name="main_screen_sections">Seções da tela principal</string>
|
||||
<string name="advanced">Avançado</string>
|
||||
<string name="color_dark">Escuro</string>
|
||||
<string name="too_many_requests_message">Muitas tentativas. Tente de novo mais tarde</string>
|
||||
<string name="related_manga">Mangá relacionado</string>
|
||||
<string name="suggestions_wifi_only_summary">Não atualize sugestões usando dados móveis</string>
|
||||
<string name="default_section">Seção padrão</string>
|
||||
<string name="background">Fundo</string>
|
||||
<string name="zoom_out">Diminuir o zoom</string>
|
||||
<string name="voice_search">Pesquisa por voz</string>
|
||||
<string name="manga_list">Lista de mangás</string>
|
||||
<string name="disable_nsfw">Desativar NSFW (+18)</string>
|
||||
<string name="color_white">Branco</string>
|
||||
<string name="to_top">Para cima</string>
|
||||
<string name="show">Mostrar</string>
|
||||
<string name="color_black">Preto</string>
|
||||
<string name="this_month">Esse mês</string>
|
||||
</resources>
|
||||
38
app/src/main/res/values-pt/plurals.xml
Normal file
38
app/src/main/res/values-pt/plurals.xml
Normal file
@@ -0,0 +1,38 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<resources>
|
||||
<plurals name="minutes_ago">
|
||||
<item quantity="one">%1$d minuto atrás</item>
|
||||
<item quantity="many">%1$d minutos atrás</item>
|
||||
<item quantity="other"></item>
|
||||
</plurals>
|
||||
<plurals name="items">
|
||||
<item quantity="one">%1$d item</item>
|
||||
<item quantity="many">%1$d itens</item>
|
||||
<item quantity="other"></item>
|
||||
</plurals>
|
||||
<plurals name="chapters">
|
||||
<item quantity="one">%1$d capítulo</item>
|
||||
<item quantity="many">%1$d capítulos</item>
|
||||
<item quantity="other"></item>
|
||||
</plurals>
|
||||
<plurals name="new_chapters">
|
||||
<item quantity="one">%1$d novo capítulo</item>
|
||||
<item quantity="many">%1$d novos capítulos</item>
|
||||
<item quantity="other"></item>
|
||||
</plurals>
|
||||
<plurals name="months_ago">
|
||||
<item quantity="one">%1$d mês atrás</item>
|
||||
<item quantity="many">%1$d meses atrás</item>
|
||||
<item quantity="other"></item>
|
||||
</plurals>
|
||||
<plurals name="days_ago">
|
||||
<item quantity="one">%1$d dia atrás</item>
|
||||
<item quantity="many">%1$d dias atrás</item>
|
||||
<item quantity="other"></item>
|
||||
</plurals>
|
||||
<plurals name="hours_ago">
|
||||
<item quantity="one">%1$d hora atrás</item>
|
||||
<item quantity="many">%1$d horas atrás</item>
|
||||
<item quantity="other"></item>
|
||||
</plurals>
|
||||
</resources>
|
||||
@@ -164,7 +164,7 @@
|
||||
<string name="vibration">Vibração</string>
|
||||
<string name="cannot_find_available_storage">Sem armazenamento disponível</string>
|
||||
<string name="favourites_categories">Categorias favoritas</string>
|
||||
<string name="text_history_holder_secondary">Encontre o que ler no menu lateral.</string>
|
||||
<string name="text_history_holder_secondary">Encontre o que ler na aba «Explorar»</string>
|
||||
<string name="text_local_holder_secondary">Salve-o de fontes online ou importe fiheiros.</string>
|
||||
<string name="recent_manga">Recente</string>
|
||||
<string name="other_storage">Outro armazenamento</string>
|
||||
@@ -313,7 +313,7 @@
|
||||
<string name="disable_all">Desativar tudo</string>
|
||||
<string name="report">Relatório</string>
|
||||
<string name="show_reading_indicators_summary">Mostrar porcentagem lida no histórico e favoritos</string>
|
||||
<string name="exclude_nsfw_from_history_summary">Mangás marcados como NSFW nunca serão adicionados ao seu histórico e seu progresso não será salvo</string>
|
||||
<string name="exclude_nsfw_from_history_summary">Os mangás marcados como NSFW nunca serão adicionados ao histórico e o seu progresso não será guardado</string>
|
||||
<string name="history_cleared">Histórico deletado</string>
|
||||
<string name="manage">Gerenciar</string>
|
||||
<string name="no_bookmarks_yet">Ainda não há marcadores</string>
|
||||
@@ -351,7 +351,7 @@
|
||||
<string name="theme_name_mamimi">Mamimi</string>
|
||||
<string name="nothing_here">Não há nada aqui</string>
|
||||
<string name="services">Serviços</string>
|
||||
<string name="enable_logging_summary">Gravar algumas ações para propósitos de depuração</string>
|
||||
<string name="enable_logging_summary">Gravar algumas ações para propósitos de depuração. Não ative se você não sabe o que está fazendo</string>
|
||||
<string name="allow_unstable_updates">Permitir atualizações instáveis</string>
|
||||
<string name="download_started">Descarga iniciada</string>
|
||||
<string name="share_logs">Compartilhar registos</string>
|
||||
@@ -370,7 +370,7 @@
|
||||
<string name="scrobbling_empty_hint">Para acompanhar o progresso da leitura, selecione Menu → Vá até a tela de detalhes do mangá.</string>
|
||||
<string name="details_button_tip">Pressione e segure o botão Ler para ver mais opções</string>
|
||||
<string name="webtoon_zoom">Zoom Webtoon</string>
|
||||
<string name="allow_unstable_updates_summary">Propor atualizações para versões beta do aplicativo</string>
|
||||
<string name="allow_unstable_updates_summary">Receber notificações sobre versões instáveis</string>
|
||||
<string name="got_it">Entendi</string>
|
||||
<string name="sources_reorder_tip">Toque e segure em um item para reordená-lo</string>
|
||||
<string name="comics_archive_import_description">Você pode selecionar um ou mais arquivos .cbz ou .zip, cada arquivo será reconhecido como um mangá separado.</string>
|
||||
@@ -392,7 +392,7 @@
|
||||
<string name="server_address">Endereço do servidor</string>
|
||||
<string name="ignore_ssl_errors">Ignorar erros de SSL</string>
|
||||
<string name="mirror_switching">Escolher espelho automaticamente</string>
|
||||
<string name="mirror_switching_summary">Alternar domínios automaticamente para fontes remotas em caso de erros, se espelhos estiverem disponíveis</string>
|
||||
<string name="mirror_switching_summary">Troca automática de domínios para fontes de mangá em caso de erro, se houver espelhos disponíveis</string>
|
||||
<string name="pause">Pausa</string>
|
||||
<string name="downloads_wifi_only">Baixe apenas via Wi-Fi</string>
|
||||
<string name="downloads_wifi_only_summary">Interrompa o download ao mudar para uma rede móvel</string>
|
||||
@@ -431,4 +431,56 @@
|
||||
<string name="enable">Activar</string>
|
||||
<string name="no_thanks">Não obrigado</string>
|
||||
<string name="remove_completed">Remoção concluída</string>
|
||||
<string name="download_option_all_unread">Todos os capítulos não lidos</string>
|
||||
<string name="clear_source_cookies_summary">Limpa os cookies apenas para o domínio especificado. Na maioria dos casos, a autorização será invalidada</string>
|
||||
<string name="download_option_manual_selection">Selecionar os capítulos manualmente</string>
|
||||
<string name="download_option_all_unread_b">Todos os capítulos não lidos (%s)</string>
|
||||
<string name="download_option_first_n_chapters">Primeiro %s</string>
|
||||
<string name="custom_directory">Diretório personalizado</string>
|
||||
<string name="download_option_next_unread_n_chapters">Próximo não lido %s</string>
|
||||
<string name="download_option_all_chapters">Todos os capítulos com tradução %s</string>
|
||||
<string name="languages">Línguas</string>
|
||||
<string name="zoom_in">Aumentar o zoom</string>
|
||||
<string name="captcha_required_summary">%s requer que um captcha seja resolvido para funcionar corretamente</string>
|
||||
<string name="progress">Progresso</string>
|
||||
<string name="error_corrupted_file">São devolvidos dados inválidos ou o ficheiro está corrompido</string>
|
||||
<string name="pick_custom_directory">Escolher diretório personalizado</string>
|
||||
<string name="related_manga_summary">Mostrar uma lista de mangas relacionadas. Em alguns casos, pode estar incorrecta ou em falta</string>
|
||||
<string name="reader_zoom_buttons_summary">Mostrar ou não os botões de controlo do zoom no canto inferior direito</string>
|
||||
<string name="tracker_wifi_only_summary">Não verificar a existência de novos capítulos utilizando ligações de rede com contador</string>
|
||||
<string name="order_added">Adicionado</string>
|
||||
<string name="on_device">No dispositivo</string>
|
||||
<string name="download_option_whole_manga">Todo o mangá</string>
|
||||
<string name="moved_to_top">Movido para o topo</string>
|
||||
<string name="data_not_restored_text">Certifique-se de que seleccionou o ficheiro de cópia de segurança correto</string>
|
||||
<string name="view_list">Ver lista</string>
|
||||
<string name="unknown">Desconhecido</string>
|
||||
<string name="in_progress">Em curso</string>
|
||||
<string name="items_limit_exceeded">Não podem ser adicionados mais itens</string>
|
||||
<string name="data_not_restored">Os dados não foram restaurados</string>
|
||||
<string name="directories">Directórios</string>
|
||||
<string name="local_manga_directories">Directórios locais de manga</string>
|
||||
<string name="manage_categories">Gerir categorias</string>
|
||||
<string name="color_light">Claro</string>
|
||||
<string name="search_hint">Introduzir o título da manga, o género ou o nome da fonte</string>
|
||||
<string name="description">Descrição</string>
|
||||
<string name="reader_zoom_buttons">Mostrar botões de zoom</string>
|
||||
<string name="main_screen_sections">Secções do ecrã principal</string>
|
||||
<string name="advanced">Avançado</string>
|
||||
<string name="color_dark">Escuro</string>
|
||||
<string name="too_many_requests_message">Demasiados pedidos. Tentar novamente mais tarde</string>
|
||||
<string name="related_manga">Mangá relacionado</string>
|
||||
<string name="suggestions_wifi_only_summary">Não atualizar sugestões utilizando ligações de rede com contador</string>
|
||||
<string name="default_section">Secção predefinida</string>
|
||||
<string name="background">Fundo</string>
|
||||
<string name="no_access_to_file">Não tem acesso a este ficheiro ou diretório</string>
|
||||
<string name="zoom_out">Reduzir o zoom</string>
|
||||
<string name="voice_search">Pesquisa por voz</string>
|
||||
<string name="manga_list">Lista de mangás</string>
|
||||
<string name="disable_nsfw">Desativar NSFW</string>
|
||||
<string name="color_white">Branco</string>
|
||||
<string name="to_top">Para cima</string>
|
||||
<string name="show">Mostrar</string>
|
||||
<string name="color_black">Preto</string>
|
||||
<string name="this_month">Este mês</string>
|
||||
</resources>
|
||||
@@ -483,4 +483,6 @@
|
||||
<string name="reader_zoom_buttons_summary">Показывать или нет кнопки управления масштабом в правом нижнем углу</string>
|
||||
<string name="reader_zoom_buttons">Отображать кнопки масштабирования</string>
|
||||
<string name="zoom_out">Отдалить</string>
|
||||
<string name="keep_screen_on">Держать экран включённым</string>
|
||||
<string name="keep_screen_on_summary">Не выключать экран во время чтения манги</string>
|
||||
</resources>
|
||||
@@ -479,4 +479,10 @@
|
||||
<string name="directories">Каталоги</string>
|
||||
<string name="main_screen_sections">Розділи головного екрана</string>
|
||||
<string name="to_top">Вгору</string>
|
||||
<string name="zoom_in">Збільшити</string>
|
||||
<string name="reader_zoom_buttons_summary">Чи показувати елементи керування масштабуванням у нижньому правому куті</string>
|
||||
<string name="reader_zoom_buttons">Показати кнопки масштабування</string>
|
||||
<string name="zoom_out">Зменшити</string>
|
||||
<string name="keep_screen_on">Тримати екран увімкненим</string>
|
||||
<string name="keep_screen_on_summary">Не вимикати екран під час читання манги</string>
|
||||
</resources>
|
||||
@@ -1,10 +0,0 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<resources>
|
||||
|
||||
<style name="Base.V23.Kotatsu" parent="Base.Theme.Kotatsu">
|
||||
<item name="android:statusBarColor">@android:color/transparent</item>
|
||||
</style>
|
||||
|
||||
<style name="Theme.Kotatsu" parent="Base.V23.Kotatsu" />
|
||||
|
||||
</resources>
|
||||
@@ -1,7 +1,7 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<resources>
|
||||
|
||||
<style name="Base.V27.Kotatsu" parent="Base.V23.Kotatsu">
|
||||
<style name="Base.V27.Kotatsu" parent="Base.Theme.Kotatsu">
|
||||
<item name="android:navigationBarColor">@color/navigation_bar_scrim</item>
|
||||
<item name="android:windowLightNavigationBar">@bool/light_navigation_bar</item>
|
||||
</style>
|
||||
|
||||
@@ -477,4 +477,8 @@
|
||||
<string name="to_top">置顶</string>
|
||||
<string name="show">显示</string>
|
||||
<string name="color_black">黑色</string>
|
||||
<string name="zoom_in">放大</string>
|
||||
<string name="reader_zoom_buttons_summary">是否在右下角显示缩放按钮</string>
|
||||
<string name="reader_zoom_buttons">显示缩放按钮</string>
|
||||
<string name="zoom_out">缩小</string>
|
||||
</resources>
|
||||
@@ -489,4 +489,7 @@
|
||||
<string name="zoom_in">Zoom in</string>
|
||||
<string name="reader_zoom_buttons">Show zoom buttons</string>
|
||||
<string name="reader_zoom_buttons_summary">Whether to show zoom control buttons in the bottom right corner</string>
|
||||
<string name="keep_screen_on">Keep screen on</string>
|
||||
<string name="keep_screen_on_summary">Do not turn the screen off while you\'re reading manga</string>
|
||||
<string name="state_abandoned">Dropped</string>
|
||||
</resources>
|
||||
|
||||
@@ -54,7 +54,7 @@
|
||||
|
||||
<!-- Themes -->
|
||||
<item name="android:windowLightStatusBar" tools:targetApi="M">@bool/light_status_bar</item>
|
||||
<item name="android:statusBarColor">@color/dim</item>
|
||||
<item name="android:statusBarColor">@color/dim_statusbar</item>
|
||||
<item name="android:navigationBarColor">@color/surface_amoled</item>
|
||||
<item name="android:navigationBarDividerColor" tools:targetApi="o_mr1">@null</item>
|
||||
<item name="android:enforceNavigationBarContrast" tools:targetApi="Q">false</item>
|
||||
|
||||
@@ -78,13 +78,19 @@
|
||||
android:summary="@string/show_pages_numbers_summary"
|
||||
android:title="@string/show_pages_numbers" />
|
||||
|
||||
<SwitchPreferenceCompat
|
||||
android:defaultValue="true"
|
||||
android:key="reader_screen_on"
|
||||
android:summary="@string/keep_screen_on_summary"
|
||||
android:title="@string/keep_screen_on"
|
||||
app:allowDividerAbove="true" />
|
||||
|
||||
<ListPreference
|
||||
android:defaultValue="allow"
|
||||
android:entries="@array/screenshots_policy"
|
||||
android:entryValues="@array/values_screenshots_policy"
|
||||
android:key="screenshots_policy"
|
||||
android:title="@string/screenshots_policy"
|
||||
app:allowDividerAbove="true"
|
||||
app:useSimpleSummaryProvider="true" />
|
||||
|
||||
<ListPreference
|
||||
|
||||
@@ -4,9 +4,9 @@ buildscript {
|
||||
mavenCentral()
|
||||
}
|
||||
dependencies {
|
||||
classpath 'com.android.tools.build:gradle:8.1.1'
|
||||
classpath 'com.android.tools.build:gradle:8.1.2'
|
||||
classpath 'org.jetbrains.kotlin:kotlin-gradle-plugin:1.9.10'
|
||||
classpath 'com.google.dagger:hilt-android-gradle-plugin:2.48'
|
||||
classpath 'com.google.dagger:hilt-android-gradle-plugin:2.48.1'
|
||||
classpath 'com.google.devtools.ksp:symbol-processing-gradle-plugin:1.9.10-1.0.13'
|
||||
}
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user