CopyManga: add user agent preference (#12348)
* CopyManga: add user agent preference * replace with vector icons * add version preference and more headers * reorder preferences * hot update referer
|
@ -6,7 +6,7 @@ ext {
|
||||||
extName = 'CopyManga'
|
extName = 'CopyManga'
|
||||||
pkgNameSuffix = 'zh.copymanga'
|
pkgNameSuffix = 'zh.copymanga'
|
||||||
extClass = '.CopyManga'
|
extClass = '.CopyManga'
|
||||||
extVersionCode = 29
|
extVersionCode = 30
|
||||||
}
|
}
|
||||||
|
|
||||||
dependencies {
|
dependencies {
|
||||||
|
|
Before Width: | Height: | Size: 5.7 KiB After Width: | Height: | Size: 4.0 KiB |
Before Width: | Height: | Size: 2.9 KiB After Width: | Height: | Size: 2.1 KiB |
Before Width: | Height: | Size: 8.9 KiB After Width: | Height: | Size: 4.9 KiB |
Before Width: | Height: | Size: 17 KiB After Width: | Height: | Size: 9.6 KiB |
Before Width: | Height: | Size: 26 KiB After Width: | Height: | Size: 12 KiB |
Before Width: | Height: | Size: 132 KiB After Width: | Height: | Size: 42 KiB |
|
@ -3,6 +3,7 @@ package eu.kanade.tachiyomi.extension.zh.copymanga
|
||||||
import android.app.Application
|
import android.app.Application
|
||||||
import android.content.SharedPreferences
|
import android.content.SharedPreferences
|
||||||
import android.util.Log
|
import android.util.Log
|
||||||
|
import androidx.preference.EditTextPreference
|
||||||
import androidx.preference.ListPreference
|
import androidx.preference.ListPreference
|
||||||
import androidx.preference.PreferenceScreen
|
import androidx.preference.PreferenceScreen
|
||||||
import androidx.preference.SwitchPreferenceCompat
|
import androidx.preference.SwitchPreferenceCompat
|
||||||
|
@ -49,12 +50,18 @@ class CopyManga : HttpSource(), ConfigurableSource {
|
||||||
.addInterceptor(NonblockingRateLimitInterceptor(2, 4)) // 2 requests per 4 seconds
|
.addInterceptor(NonblockingRateLimitInterceptor(2, 4)) // 2 requests per 4 seconds
|
||||||
.build()
|
.build()
|
||||||
|
|
||||||
override fun headersBuilder() = headersBuilder(preferences.getBoolean(OVERSEAS_CDN_PREF, false))
|
private fun Headers.Builder.setUserAgent(userAgent: String) = set("User-Agent", userAgent)
|
||||||
private fun headersBuilder(useOverseasCdn: Boolean) = Headers.Builder()
|
private fun Headers.Builder.setRegion(useOverseasCdn: Boolean) = set("region", if (useOverseasCdn) "0" else "1")
|
||||||
.add("User-Agent", System.getProperty("http.agent")!!)
|
private fun Headers.Builder.setReferer() = set("Referer", WWW_PREFIX + domain)
|
||||||
.add("region", if (useOverseasCdn) "0" else "1")
|
private fun Headers.Builder.setVersion(version: String) = set("version", version)
|
||||||
|
|
||||||
private var apiHeaders = headersBuilder().build()
|
private var apiHeaders = Headers.Builder()
|
||||||
|
.setUserAgent(preferences.getString(USER_AGENT_PREF, DEFAULT_USER_AGENT)!!)
|
||||||
|
.setRegion(preferences.getBoolean(OVERSEAS_CDN_PREF, false))
|
||||||
|
.setReferer()
|
||||||
|
.add("platform", "1")
|
||||||
|
.setVersion(preferences.getString(VERSION_PREF, DEFAULT_VERSION)!!)
|
||||||
|
.build()
|
||||||
|
|
||||||
private var useWebp = preferences.getBoolean(WEBP_PREF, true)
|
private var useWebp = preferences.getBoolean(WEBP_PREF, true)
|
||||||
|
|
||||||
|
@ -151,10 +158,15 @@ class CopyManga : HttpSource(), ConfigurableSource {
|
||||||
// 新版 API 中间是 /chapter2/ 并且返回值需要排序
|
// 新版 API 中间是 /chapter2/ 并且返回值需要排序
|
||||||
override fun pageListRequest(chapter: SChapter) = GET("$apiUrl/api/v3${chapter.url}", apiHeaders)
|
override fun pageListRequest(chapter: SChapter) = GET("$apiUrl/api/v3${chapter.url}", apiHeaders)
|
||||||
|
|
||||||
override fun pageListParse(response: Response): List<Page> =
|
override fun pageListParse(response: Response): List<Page> {
|
||||||
response.parseAs<ChapterPageListWrapperDto>().chapter.contents.mapIndexed { i, it ->
|
val result: ChapterPageListWrapperDto = response.parseAs()
|
||||||
|
if (result.show_app) {
|
||||||
|
throw Exception("访问受限,请尝试在插件设置中修改 User Agent")
|
||||||
|
}
|
||||||
|
return result.chapter.contents.mapIndexed { i, it ->
|
||||||
Page(i, imageUrl = it.url)
|
Page(i, imageUrl = it.url)
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
override fun imageUrlParse(response: Response) = throw UnsupportedOperationException("Not used.")
|
override fun imageUrlParse(response: Response) = throw UnsupportedOperationException("Not used.")
|
||||||
|
|
||||||
|
@ -226,6 +238,33 @@ class CopyManga : HttpSource(), ConfigurableSource {
|
||||||
preferences.edit().putString(DOMAIN_PREF, index).apply()
|
preferences.edit().putString(DOMAIN_PREF, index).apply()
|
||||||
domain = DOMAINS[index.toInt()]
|
domain = DOMAINS[index.toInt()]
|
||||||
apiUrl = API_PREFIX + domain
|
apiUrl = API_PREFIX + domain
|
||||||
|
apiHeaders = apiHeaders.newBuilder().setReferer().build()
|
||||||
|
true
|
||||||
|
}
|
||||||
|
}.let { screen.addPreference(it) }
|
||||||
|
|
||||||
|
EditTextPreference(screen.context).apply {
|
||||||
|
key = USER_AGENT_PREF
|
||||||
|
title = "User Agent"
|
||||||
|
summary = "可以在 Windows/macOS/iOS 上用浏览器访问 https://tool.lu/useragent 获取(不要使用 Windows Chrome 103)"
|
||||||
|
setDefaultValue(DEFAULT_USER_AGENT)
|
||||||
|
setOnPreferenceChangeListener { _, newValue ->
|
||||||
|
val userAgent = newValue as String
|
||||||
|
preferences.edit().putString(USER_AGENT_PREF, userAgent).apply()
|
||||||
|
apiHeaders = apiHeaders.newBuilder().setUserAgent(userAgent).build()
|
||||||
|
true
|
||||||
|
}
|
||||||
|
}.let { screen.addPreference(it) }
|
||||||
|
|
||||||
|
EditTextPreference(screen.context).apply {
|
||||||
|
key = VERSION_PREF
|
||||||
|
title = "网页版本号"
|
||||||
|
summary = "访问受限时按实际情况修改"
|
||||||
|
setDefaultValue(DEFAULT_VERSION)
|
||||||
|
setOnPreferenceChangeListener { _, newValue ->
|
||||||
|
val version = newValue as String
|
||||||
|
preferences.edit().putString(VERSION_PREF, version).apply()
|
||||||
|
apiHeaders = apiHeaders.newBuilder().setVersion(version).build()
|
||||||
true
|
true
|
||||||
}
|
}
|
||||||
}.let { screen.addPreference(it) }
|
}.let { screen.addPreference(it) }
|
||||||
|
@ -238,7 +277,7 @@ class CopyManga : HttpSource(), ConfigurableSource {
|
||||||
setOnPreferenceChangeListener { _, newValue ->
|
setOnPreferenceChangeListener { _, newValue ->
|
||||||
val useOverseasCdn = newValue as Boolean
|
val useOverseasCdn = newValue as Boolean
|
||||||
preferences.edit().putBoolean(OVERSEAS_CDN_PREF, useOverseasCdn).apply()
|
preferences.edit().putBoolean(OVERSEAS_CDN_PREF, useOverseasCdn).apply()
|
||||||
apiHeaders = headersBuilder(useOverseasCdn).build()
|
apiHeaders = apiHeaders.newBuilder().setRegion(useOverseasCdn).build()
|
||||||
true
|
true
|
||||||
}
|
}
|
||||||
}.let { screen.addPreference(it) }
|
}.let { screen.addPreference(it) }
|
||||||
|
@ -275,12 +314,16 @@ class CopyManga : HttpSource(), ConfigurableSource {
|
||||||
private const val OVERSEAS_CDN_PREF = "changeCDN"
|
private const val OVERSEAS_CDN_PREF = "changeCDN"
|
||||||
private const val SC_TITLE_PREF = "showSCTitle"
|
private const val SC_TITLE_PREF = "showSCTitle"
|
||||||
private const val WEBP_PREF = "webp"
|
private const val WEBP_PREF = "webp"
|
||||||
|
private const val USER_AGENT_PREF = "userAgent"
|
||||||
|
private const val VERSION_PREF = "version"
|
||||||
// private const val CHROME_VERSION_PREF = "chromeVersion" // default value was "103"
|
// private const val CHROME_VERSION_PREF = "chromeVersion" // default value was "103"
|
||||||
|
|
||||||
private const val WWW_PREFIX = "https://www."
|
private const val WWW_PREFIX = "https://www."
|
||||||
private const val API_PREFIX = "https://api."
|
private const val API_PREFIX = "https://api."
|
||||||
private val DOMAINS = arrayOf("copymanga.org", "copymanga.info", "copymanga.net")
|
private val DOMAINS = arrayOf("copymanga.org", "copymanga.info", "copymanga.net")
|
||||||
private val DOMAIN_INDICES = arrayOf("0", "1", "2")
|
private val DOMAIN_INDICES = arrayOf("0", "1", "2")
|
||||||
|
private const val DEFAULT_USER_AGENT = "Dart/2.16(dart:io)"
|
||||||
|
private const val DEFAULT_VERSION = "2022.06.22"
|
||||||
|
|
||||||
private const val PAGE_SIZE = 20
|
private const val PAGE_SIZE = 20
|
||||||
private const val CHAPTER_PAGE_SIZE = 500
|
private const val CHAPTER_PAGE_SIZE = 500
|
||||||
|
|
|
@ -114,7 +114,7 @@ class ChapterPageListDto(val contents: List<UrlDto>)
|
||||||
class UrlDto(val url: String)
|
class UrlDto(val url: String)
|
||||||
|
|
||||||
@Serializable
|
@Serializable
|
||||||
class ChapterPageListWrapperDto(val chapter: ChapterPageListDto)
|
class ChapterPageListWrapperDto(val chapter: ChapterPageListDto, val show_app: Boolean)
|
||||||
|
|
||||||
@Serializable
|
@Serializable
|
||||||
class ListDto<T>(
|
class ListDto<T>(
|
||||||
|
|