mirror of
https://github.com/gedoor/legado.git
synced 2025-08-10 00:52:30 +00:00
优化
This commit is contained in:
1879
app/schemas/io.legado.app.data.AppDatabase/67.json
Normal file
1879
app/schemas/io.legado.app.data.AppDatabase/67.json
Normal file
File diff suppressed because it is too large
Load Diff
@@ -19,9 +19,11 @@ object SourceHelp {
|
||||
private val list18Plus by lazy {
|
||||
try {
|
||||
return@lazy String(appCtx.assets.open("18PlusList.txt").readBytes())
|
||||
.splitNotBlank("\n")
|
||||
.splitNotBlank("\n").map {
|
||||
EncoderUtils.base64Decode(it)
|
||||
}.toHashSet()
|
||||
} catch (e: Exception) {
|
||||
return@lazy arrayOf<String>()
|
||||
return@lazy hashSetOf<String>()
|
||||
}
|
||||
}
|
||||
|
||||
@@ -32,42 +34,42 @@ object SourceHelp {
|
||||
}
|
||||
|
||||
fun insertRssSource(vararg rssSources: RssSource) {
|
||||
rssSources.forEach { rssSource ->
|
||||
if (is18Plus(rssSource.sourceUrl)) {
|
||||
handler.post {
|
||||
appCtx.toastOnUi("${rssSource.sourceName}是18+网址,禁止导入.")
|
||||
}
|
||||
} else {
|
||||
appDb.rssSourceDao.insert(rssSource)
|
||||
val rssSourcesGroup = rssSources.groupBy {
|
||||
is18Plus(it.sourceUrl)
|
||||
}
|
||||
rssSourcesGroup[true]?.forEach {
|
||||
handler.post {
|
||||
appCtx.toastOnUi("${it.sourceName}是18+网址,禁止导入.")
|
||||
}
|
||||
}
|
||||
rssSourcesGroup[false]?.let {
|
||||
appDb.rssSourceDao.insert(*it.toTypedArray())
|
||||
}
|
||||
}
|
||||
|
||||
fun insertBookSource(vararg bookSources: BookSource) {
|
||||
bookSources.forEach { bookSource ->
|
||||
if (is18Plus(bookSource.bookSourceUrl)) {
|
||||
handler.post {
|
||||
appCtx.toastOnUi("${bookSource.bookSourceName}是18+网址,禁止导入.")
|
||||
}
|
||||
} else {
|
||||
appDb.bookSourceDao.insert(bookSource)
|
||||
val bookSourcesGroup = bookSources.groupBy {
|
||||
is18Plus(it.bookSourceUrl)
|
||||
}
|
||||
bookSourcesGroup[true]?.forEach {
|
||||
handler.post {
|
||||
appCtx.toastOnUi("${it.bookSourceName}是18+网址,禁止导入.")
|
||||
}
|
||||
}
|
||||
bookSourcesGroup[false]?.let {
|
||||
appDb.bookSourceDao.insert(*it.toTypedArray())
|
||||
}
|
||||
}
|
||||
|
||||
private fun is18Plus(url: String?): Boolean {
|
||||
url ?: return false
|
||||
val baseUrl = NetworkUtils.getBaseUrl(url)
|
||||
baseUrl ?: return false
|
||||
if (AppConst.isPlayChannel) return false
|
||||
val baseUrl = NetworkUtils.getBaseUrl(url) ?: return false
|
||||
kotlin.runCatching {
|
||||
val host = baseUrl.split("//", ".")
|
||||
val base64Url = EncoderUtils.base64Encode("${host[host.lastIndex - 1]}.${host.last()}")
|
||||
list18Plus.forEach {
|
||||
if (base64Url == it) {
|
||||
return true
|
||||
}
|
||||
val host = baseUrl.split("//", ".").let {
|
||||
if (it.size > 2) "${it[it.lastIndex - 1]}.${it.last()}" else return false
|
||||
}
|
||||
return list18Plus.contains(host)
|
||||
}
|
||||
return false
|
||||
}
|
||||
|
||||
@@ -307,7 +307,6 @@ object ReadBook : CoroutineScope by MainScope() {
|
||||
contentLoadFinish(book, chapter, it, upContent, resetPageOffset) {
|
||||
success?.invoke()
|
||||
}
|
||||
removeLoading(chapter.index)
|
||||
} ?: download(this, chapter, resetPageOffset = resetPageOffset)
|
||||
} ?: removeLoading(index)
|
||||
}.onError {
|
||||
@@ -320,26 +319,24 @@ object ReadBook : CoroutineScope by MainScope() {
|
||||
/**
|
||||
* 下载正文
|
||||
*/
|
||||
private fun download(index: Int) {
|
||||
private fun downloadIndex(index: Int) {
|
||||
if (index < 0) return
|
||||
if (index > chapterSize - 1) {
|
||||
upToc()
|
||||
return
|
||||
}
|
||||
book?.let { book ->
|
||||
if (book.isLocal) return
|
||||
if (addLoading(index)) {
|
||||
Coroutine.async {
|
||||
appDb.bookChapterDao.getChapter(book.bookUrl, index)?.let { chapter ->
|
||||
if (BookHelp.hasContent(book, chapter)) {
|
||||
removeLoading(chapter.index)
|
||||
} else {
|
||||
download(this, chapter, false)
|
||||
}
|
||||
} ?: removeLoading(index)
|
||||
}.onError {
|
||||
removeLoading(index)
|
||||
}
|
||||
val book = book ?: return
|
||||
if (addLoading(index)) {
|
||||
try {
|
||||
appDb.bookChapterDao.getChapter(book.bookUrl, index)?.let { chapter ->
|
||||
if (BookHelp.hasContent(book, chapter)) {
|
||||
removeLoading(chapter.index)
|
||||
} else {
|
||||
download(this, chapter, false)
|
||||
}
|
||||
} ?: removeLoading(index)
|
||||
} catch (e: Exception) {
|
||||
removeLoading(index)
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -353,20 +350,17 @@ object ReadBook : CoroutineScope by MainScope() {
|
||||
resetPageOffset: Boolean,
|
||||
success: (() -> Unit)? = null
|
||||
) {
|
||||
val book = book
|
||||
val book = book ?: return removeLoading(chapter.index)
|
||||
val bookSource = bookSource
|
||||
if (book != null && bookSource != null) {
|
||||
if (bookSource != null) {
|
||||
CacheBook.getOrCreate(bookSource, book).download(scope, chapter)
|
||||
} else if (book != null) {
|
||||
} else {
|
||||
val msg = if (book.isLocal) "无内容" else "没有书源"
|
||||
contentLoadFinish(
|
||||
book, chapter, "加载正文失败\n$msg", resetPageOffset = resetPageOffset
|
||||
) {
|
||||
success?.invoke()
|
||||
}
|
||||
removeLoading(chapter.index)
|
||||
} else {
|
||||
removeLoading(chapter.index)
|
||||
}
|
||||
}
|
||||
|
||||
@@ -415,10 +409,12 @@ object ReadBook : CoroutineScope by MainScope() {
|
||||
curPageChanged()
|
||||
callBack?.contentLoadFinish()
|
||||
}
|
||||
|
||||
-1 -> {
|
||||
prevTextChapter = textChapter
|
||||
if (upContent) callBack?.upContent(offset, resetPageOffset)
|
||||
}
|
||||
|
||||
1 -> {
|
||||
nextTextChapter = textChapter
|
||||
if (upContent) callBack?.upContent(offset, resetPageOffset)
|
||||
@@ -464,18 +460,17 @@ object ReadBook : CoroutineScope by MainScope() {
|
||||
|
||||
fun saveRead() {
|
||||
Coroutine.async {
|
||||
book?.let { book ->
|
||||
book.lastCheckCount = 0
|
||||
book.durChapterTime = System.currentTimeMillis()
|
||||
book.durChapterIndex = durChapterIndex
|
||||
book.durChapterPos = durChapterPos
|
||||
appDb.bookChapterDao.getChapter(book.bookUrl, durChapterIndex)?.let {
|
||||
book.durChapterTitle = it.getDisplayTitle(
|
||||
ContentProcessor.get(book.name, book.origin).getTitleReplaceRules()
|
||||
)
|
||||
}
|
||||
appDb.bookDao.update(book)
|
||||
val book = book ?: return@async
|
||||
book.lastCheckCount = 0
|
||||
book.durChapterTime = System.currentTimeMillis()
|
||||
book.durChapterIndex = durChapterIndex
|
||||
book.durChapterPos = durChapterPos
|
||||
appDb.bookChapterDao.getChapter(book.bookUrl, durChapterIndex)?.let {
|
||||
book.durChapterTitle = it.getDisplayTitle(
|
||||
ContentProcessor.get(book.name, book.origin).getTitleReplaceRules()
|
||||
)
|
||||
}
|
||||
appDb.bookDao.update(book)
|
||||
}
|
||||
}
|
||||
|
||||
@@ -483,6 +478,7 @@ object ReadBook : CoroutineScope by MainScope() {
|
||||
* 预下载
|
||||
*/
|
||||
private fun preDownload() {
|
||||
if (book?.isLocal == true) return
|
||||
if (AppConfig.preDownloadNum < 2) {
|
||||
return
|
||||
}
|
||||
@@ -491,12 +487,12 @@ object ReadBook : CoroutineScope by MainScope() {
|
||||
val maxChapterIndex = durChapterIndex + AppConfig.preDownloadNum
|
||||
for (i in durChapterIndex.plus(2)..maxChapterIndex) {
|
||||
delay(1000)
|
||||
download(i)
|
||||
downloadIndex(i)
|
||||
}
|
||||
val minChapterIndex = durChapterIndex - min(5, AppConfig.preDownloadNum)
|
||||
for (i in durChapterIndex.minus(2) downTo minChapterIndex) {
|
||||
delay(1000)
|
||||
download(i)
|
||||
downloadIndex(i)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -27,6 +27,7 @@ import io.legado.app.databinding.DialogBookChangeSourceBinding
|
||||
import io.legado.app.help.config.AppConfig
|
||||
import io.legado.app.lib.dialogs.alert
|
||||
import io.legado.app.lib.theme.primaryColor
|
||||
import io.legado.app.ui.book.read.ReadBookActivity
|
||||
import io.legado.app.ui.book.source.edit.BookSourceEditActivity
|
||||
import io.legado.app.ui.book.source.manage.BookSourceActivity
|
||||
import io.legado.app.ui.widget.dialog.WaitDialog
|
||||
@@ -86,7 +87,7 @@ class ChangeBookSourceDialog() : BaseDialogFragment(R.layout.dialog_book_change_
|
||||
|
||||
override fun onFragmentCreated(view: View, savedInstanceState: Bundle?) {
|
||||
binding.toolBar.setBackgroundColor(primaryColor)
|
||||
viewModel.initData(arguments)
|
||||
viewModel.initData(arguments, callBack?.oldBook, activity is ReadBookActivity)
|
||||
showTitle()
|
||||
initMenu()
|
||||
initRecyclerView()
|
||||
@@ -212,19 +213,23 @@ class ChangeBookSourceDialog() : BaseDialogFragment(R.layout.dialog_book_change_
|
||||
item.isChecked = !item.isChecked
|
||||
viewModel.refresh()
|
||||
}
|
||||
|
||||
R.id.menu_load_info -> {
|
||||
AppConfig.changeSourceLoadInfo = !item.isChecked
|
||||
item.isChecked = !item.isChecked
|
||||
}
|
||||
|
||||
R.id.menu_load_toc -> {
|
||||
AppConfig.changeSourceLoadToc = !item.isChecked
|
||||
item.isChecked = !item.isChecked
|
||||
}
|
||||
|
||||
R.id.menu_load_word_count -> {
|
||||
AppConfig.changeSourceLoadWordCount = !item.isChecked
|
||||
item.isChecked = !item.isChecked
|
||||
viewModel.onLoadWordCountChecked(item.isChecked)
|
||||
}
|
||||
|
||||
R.id.menu_start_stop -> viewModel.startOrStopSearch()
|
||||
R.id.menu_source_manage -> startActivity<BookSourceActivity>()
|
||||
R.id.menu_refresh_list -> viewModel.startRefreshList()
|
||||
@@ -306,7 +311,7 @@ class ChangeBookSourceDialog() : BaseDialogFragment(R.layout.dialog_book_change_
|
||||
}
|
||||
|
||||
override fun setBookScore(searchBook: SearchBook, score: Int) {
|
||||
viewModel.setBookScore(searchBook,score)
|
||||
viewModel.setBookScore(searchBook, score)
|
||||
}
|
||||
|
||||
override fun getBookScore(searchBook: SearchBook): Int {
|
||||
|
||||
@@ -16,12 +16,14 @@ import io.legado.app.data.entities.BookChapter
|
||||
import io.legado.app.data.entities.BookSource
|
||||
import io.legado.app.data.entities.SearchBook
|
||||
import io.legado.app.exception.NoStackTraceException
|
||||
import io.legado.app.help.book.BookHelp
|
||||
import io.legado.app.help.config.AppConfig
|
||||
import io.legado.app.help.config.SourceConfig
|
||||
import io.legado.app.help.coroutine.CompositeCoroutine
|
||||
import io.legado.app.help.coroutine.Coroutine
|
||||
import io.legado.app.model.ReadBook
|
||||
import io.legado.app.model.webBook.WebBook
|
||||
import io.legado.app.ui.book.read.ReadBookActivity
|
||||
import io.legado.app.utils.getPrefBoolean
|
||||
import io.legado.app.utils.postEvent
|
||||
import io.legado.app.utils.toastOnUi
|
||||
@@ -44,6 +46,8 @@ open class ChangeBookSourceViewModel(application: Application) : BaseViewModel(a
|
||||
var searchFinishCallback: ((isEmpty: Boolean) -> Unit)? = null
|
||||
var name: String = ""
|
||||
var author: String = ""
|
||||
private var fromReadBookActivity = false
|
||||
private var oldBook: Book? = null
|
||||
private var tasks = CompositeCoroutine()
|
||||
private var screenKey: String = ""
|
||||
private var bookSourceList = arrayListOf<BookSource>()
|
||||
@@ -114,7 +118,7 @@ open class ChangeBookSourceViewModel(application: Application) : BaseViewModel(a
|
||||
}
|
||||
|
||||
@CallSuper
|
||||
open fun initData(arguments: Bundle?) {
|
||||
open fun initData(arguments: Bundle?, book: Book?, fromReadBookActivity: Boolean) {
|
||||
arguments?.let { bundle ->
|
||||
bundle.getString("name")?.let {
|
||||
name = it
|
||||
@@ -122,6 +126,8 @@ open class ChangeBookSourceViewModel(application: Application) : BaseViewModel(a
|
||||
bundle.getString("author")?.let {
|
||||
author = it.replace(AppPattern.authorRegex, "")
|
||||
}
|
||||
this.fromReadBookActivity = fromReadBookActivity
|
||||
oldBook = book
|
||||
}
|
||||
}
|
||||
|
||||
@@ -180,22 +186,24 @@ open class ChangeBookSourceViewModel(application: Application) : BaseViewModel(a
|
||||
val task = Coroutine.async(scope = viewModelScope, context = searchPool!!) {
|
||||
val resultBooks = WebBook.searchBookAwait(source, name)
|
||||
resultBooks.forEach { searchBook ->
|
||||
if (searchBook.name == name) {
|
||||
if ((AppConfig.changeSourceCheckAuthor && searchBook.author.contains(author))
|
||||
|| !AppConfig.changeSourceCheckAuthor
|
||||
) {
|
||||
when {
|
||||
searchBook.latestChapterTitle.isNullOrEmpty() &&
|
||||
(AppConfig.changeSourceLoadInfo || AppConfig.changeSourceLoadToc) -> {
|
||||
loadBookInfo(source, searchBook.toBook())
|
||||
}
|
||||
searchBook.chapterWordCountText.isNullOrBlank() && AppConfig.changeSourceLoadWordCount -> {
|
||||
loadBookToc(source, searchBook.toBook())
|
||||
}
|
||||
else -> {
|
||||
searchCallback?.searchSuccess(searchBook)
|
||||
}
|
||||
}
|
||||
if (searchBook.name != name) {
|
||||
return@forEach
|
||||
}
|
||||
if (AppConfig.changeSourceCheckAuthor && !searchBook.author.contains(author)) {
|
||||
return@forEach
|
||||
}
|
||||
when {
|
||||
searchBook.latestChapterTitle.isNullOrEmpty() &&
|
||||
(AppConfig.changeSourceLoadInfo || AppConfig.changeSourceLoadToc) -> {
|
||||
loadBookInfo(source, searchBook.toBook())
|
||||
}
|
||||
|
||||
AppConfig.changeSourceLoadWordCount -> {
|
||||
loadBookToc(source, searchBook.toBook())
|
||||
}
|
||||
|
||||
else -> {
|
||||
searchCallback?.searchSuccess(searchBook)
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -211,7 +219,7 @@ open class ChangeBookSourceViewModel(application: Application) : BaseViewModel(a
|
||||
|
||||
private suspend fun loadBookInfo(source: BookSource, book: Book) {
|
||||
WebBook.getBookInfoAwait(source, book)
|
||||
if (context.getPrefBoolean(PreferKey.changeSourceLoadToc)) {
|
||||
if (AppConfig.changeSourceLoadToc || AppConfig.changeSourceLoadWordCount) {
|
||||
loadBookToc(source, book)
|
||||
} else {
|
||||
//从详情页里获取最新章节
|
||||
@@ -224,7 +232,7 @@ open class ChangeBookSourceViewModel(application: Application) : BaseViewModel(a
|
||||
val chapters = WebBook.getChapterListAwait(source, book).getOrThrow()
|
||||
tocMap[book.bookUrl] = chapters
|
||||
bookMap[book.bookUrl] = book
|
||||
if (context.getPrefBoolean(PreferKey.changeSourceLoadWordCount)) {
|
||||
if (AppConfig.changeSourceLoadWordCount) {
|
||||
loadBookWordCount(source, book, chapters)
|
||||
} else {
|
||||
val searchBook = book.toSearchBook()
|
||||
@@ -237,16 +245,26 @@ open class ChangeBookSourceViewModel(application: Application) : BaseViewModel(a
|
||||
book: Book,
|
||||
chapters: List<BookChapter>
|
||||
) = coroutineScope {
|
||||
val chapterIndex = ReadBook.curTextChapter?.chapter?.index ?: (chapters.size - 1)
|
||||
val chapterIndex = if (fromReadBookActivity) {
|
||||
oldBook?.let {
|
||||
BookHelp.getDurChapter(
|
||||
it.durChapterIndex,
|
||||
it.durChapterTitle,
|
||||
chapters,
|
||||
it.totalChapterNum
|
||||
)
|
||||
} ?: chapters.lastIndex
|
||||
} else chapters.lastIndex
|
||||
val bookChapter = chapters.getOrNull(chapterIndex)
|
||||
val startTime = System.currentTimeMillis()
|
||||
val pair = try {
|
||||
if (bookChapter == null) throw NoStackTraceException("章节缺失,总章节数${chapters.size}")
|
||||
if (!isActive) return@coroutineScope
|
||||
WebBook.getContentAwait(source, book, bookChapter, null, false).length.let {
|
||||
val nextChapterUrl = chapters.getOrNull(chapterIndex + 1)?.url
|
||||
WebBook.getContentAwait(source, book, bookChapter, nextChapterUrl, false).length.let {
|
||||
it to "第${chapterIndex + 1}章 字数:${it}"
|
||||
}
|
||||
} catch (t: Throwable) {
|
||||
if (t is CancellationException) throw t
|
||||
-1 to "第${chapterIndex + 1}章 获取字数失败:${t.localizedMessage}"
|
||||
}
|
||||
val endTime = System.currentTimeMillis()
|
||||
|
||||
@@ -32,6 +32,7 @@ import io.legado.app.help.config.AppConfig
|
||||
import io.legado.app.lib.dialogs.alert
|
||||
import io.legado.app.lib.theme.elevation
|
||||
import io.legado.app.lib.theme.primaryColor
|
||||
import io.legado.app.ui.book.read.ReadBookActivity
|
||||
import io.legado.app.ui.book.source.edit.BookSourceEditActivity
|
||||
import io.legado.app.ui.book.source.manage.BookSourceActivity
|
||||
import io.legado.app.ui.widget.recycler.VerticalDivider
|
||||
@@ -112,7 +113,7 @@ class ChangeChapterSourceDialog() : BaseDialogFragment(R.layout.dialog_chapter_c
|
||||
|
||||
override fun onFragmentCreated(view: View, savedInstanceState: Bundle?) {
|
||||
binding.toolBar.setBackgroundColor(primaryColor)
|
||||
viewModel.initData(arguments)
|
||||
viewModel.initData(arguments, callBack?.oldBook, activity is ReadBookActivity)
|
||||
showTitle()
|
||||
initMenu()
|
||||
initView()
|
||||
|
||||
@@ -15,8 +15,8 @@ class ChangeChapterSourceViewModel(application: Application) :
|
||||
var chapterIndex: Int = 0
|
||||
var chapterTitle: String = ""
|
||||
|
||||
override fun initData(arguments: Bundle?) {
|
||||
super.initData(arguments)
|
||||
override fun initData(arguments: Bundle?, book: Book?, fromReadBookActivity: Boolean) {
|
||||
super.initData(arguments, book, fromReadBookActivity)
|
||||
arguments?.let { bundle ->
|
||||
bundle.getString("chapterTitle")?.let {
|
||||
chapterTitle = it
|
||||
|
||||
Reference in New Issue
Block a user