Repository: gedoor/legado
Branch: master
Commit: 0486da3c0255
Files: 1746
Total size: 11.0 MB
Directory structure:
gitextract_9ap85jiu/
├── .github/
│ ├── ISSUE_TEMPLATE/
│ │ ├── 01-bugReport.yml
│ │ ├── 02-featureRequest.yml
│ │ └── config.yml
│ ├── dependabot.yml
│ ├── scripts/
│ │ ├── cronet.sh
│ │ ├── lzy_web.py
│ │ └── tg_bot.py
│ └── workflows/
│ ├── autoupdatefork.yml
│ ├── cronet.yml
│ ├── legado.jks
│ ├── release.yml
│ ├── stale.yml
│ ├── test.yml
│ └── web.yml
├── .gitignore
├── CHANGELOG.md
├── English.md
├── LICENSE
├── README.md
├── api.md
├── app/
│ ├── .gitignore
│ ├── build.gradle
│ ├── cronet-proguard-rules.pro
│ ├── cronetlib/
│ │ ├── cronet_api.jar
│ │ ├── cronet_impl_common_java.jar
│ │ ├── cronet_impl_native_java.jar
│ │ ├── cronet_impl_platform_java.jar
│ │ └── cronet_shared_java.jar
│ ├── download.gradle
│ ├── google-services.json
│ ├── proguard-rules.pro
│ ├── schemas/
│ │ └── io.legado.app.data.AppDatabase/
│ │ ├── 1.json
│ │ ├── 10.json
│ │ ├── 11.json
│ │ ├── 12.json
│ │ ├── 13.json
│ │ ├── 14.json
│ │ ├── 15.json
│ │ ├── 16.json
│ │ ├── 17.json
│ │ ├── 18.json
│ │ ├── 19.json
│ │ ├── 2.json
│ │ ├── 20.json
│ │ ├── 21.json
│ │ ├── 22.json
│ │ ├── 23.json
│ │ ├── 24.json
│ │ ├── 25.json
│ │ ├── 26.json
│ │ ├── 27.json
│ │ ├── 28.json
│ │ ├── 29.json
│ │ ├── 3.json
│ │ ├── 30.json
│ │ ├── 31.json
│ │ ├── 32.json
│ │ ├── 33.json
│ │ ├── 34.json
│ │ ├── 35.json
│ │ ├── 36.json
│ │ ├── 37.json
│ │ ├── 38.json
│ │ ├── 39.json
│ │ ├── 4.json
│ │ ├── 40.json
│ │ ├── 41.json
│ │ ├── 42.json
│ │ ├── 43.json
│ │ ├── 44.json
│ │ ├── 45.json
│ │ ├── 46.json
│ │ ├── 47.json
│ │ ├── 48.json
│ │ ├── 49.json
│ │ ├── 5.json
│ │ ├── 50.json
│ │ ├── 51.json
│ │ ├── 52.json
│ │ ├── 53.json
│ │ ├── 54.json
│ │ ├── 55.json
│ │ ├── 56.json
│ │ ├── 57.json
│ │ ├── 58.json
│ │ ├── 59.json
│ │ ├── 6.json
│ │ ├── 60.json
│ │ ├── 61.json
│ │ ├── 62.json
│ │ ├── 63.json
│ │ ├── 64.json
│ │ ├── 65.json
│ │ ├── 66.json
│ │ ├── 67.json
│ │ ├── 68.json
│ │ ├── 69.json
│ │ ├── 7.json
│ │ ├── 70.json
│ │ ├── 71.json
│ │ ├── 72.json
│ │ ├── 73.json
│ │ ├── 74.json
│ │ ├── 75.json
│ │ ├── 8.json
│ │ └── 9.json
│ └── src/
│ ├── androidTest/
│ │ └── java/
│ │ └── io/
│ │ └── legado/
│ │ └── app/
│ │ ├── AndroidJsTest.kt
│ │ ├── ExampleInstrumentedTest.kt
│ │ ├── HttpTest.kt
│ │ ├── HttpTtsTest.kt
│ │ ├── MigrationTest.kt
│ │ └── UpdateTest.kt
│ ├── debug/
│ │ └── res/
│ │ ├── values/
│ │ │ └── strings.xml
│ │ └── values-zh/
│ │ └── strings.xml
│ ├── main/
│ │ ├── AndroidManifest.xml
│ │ ├── assets/
│ │ │ ├── 18PlusList.txt
│ │ │ ├── LICENSE.md
│ │ │ ├── cronet.json
│ │ │ ├── defaultData/
│ │ │ │ ├── bookSources.json
│ │ │ │ ├── coverRule.json
│ │ │ │ ├── dictRules.json
│ │ │ │ ├── directLinkUpload.json
│ │ │ │ ├── httpTTS.json
│ │ │ │ ├── keyboardAssists.json
│ │ │ │ ├── readConfig.json
│ │ │ │ ├── rssSources.json
│ │ │ │ ├── themeConfig.json
│ │ │ │ └── txtTocRule.json
│ │ │ ├── disclaimer.md
│ │ │ ├── epub/
│ │ │ │ ├── chapter.html
│ │ │ │ ├── cover.html
│ │ │ │ ├── fonts.css
│ │ │ │ ├── intro.html
│ │ │ │ └── main.css
│ │ │ ├── privacyPolicy.md
│ │ │ ├── storageHelp.md
│ │ │ ├── updateLog.md
│ │ │ └── web/
│ │ │ ├── assets/
│ │ │ │ ├── css/
│ │ │ │ │ └── main.css
│ │ │ │ └── js/
│ │ │ │ ├── dist.js
│ │ │ │ └── md5.js
│ │ │ ├── help/
│ │ │ │ ├── index.html
│ │ │ │ ├── js/
│ │ │ │ │ ├── main.js
│ │ │ │ │ ├── marked-highlight.umd.js
│ │ │ │ │ └── require.js
│ │ │ │ └── md/
│ │ │ │ ├── ExtensionContentType.md
│ │ │ │ ├── SourceMBookHelp.md
│ │ │ │ ├── SourceMRssHelp.md
│ │ │ │ ├── appHelp.md
│ │ │ │ ├── debugHelp.md
│ │ │ │ ├── dictRuleHelp.md
│ │ │ │ ├── httpTTSHelp.md
│ │ │ │ ├── jsHelp.md
│ │ │ │ ├── readMenuHelp.md
│ │ │ │ ├── regexHelp.md
│ │ │ │ ├── replaceRuleHelp.md
│ │ │ │ ├── ruleHelp.md
│ │ │ │ ├── txtTocRuleHelp.md
│ │ │ │ ├── webDavBookHelp.md
│ │ │ │ ├── webDavHelp.md
│ │ │ │ └── xpathHelp.md
│ │ │ ├── index.html
│ │ │ ├── uploadBook/
│ │ │ │ ├── css/
│ │ │ │ │ └── wifi_send.css
│ │ │ │ ├── index.html
│ │ │ │ └── js/
│ │ │ │ ├── common.js
│ │ │ │ └── html5_fun.js
│ │ │ └── vue/
│ │ │ ├── assets/
│ │ │ │ ├── BookChapter-BsiFtdIw.css
│ │ │ │ ├── BookChapter-Cs3stH93.js
│ │ │ │ ├── BookShelf-00b2QCsd.css
│ │ │ │ ├── BookShelf-DIQtBULC.js
│ │ │ │ ├── index-CrxHVQK7.css
│ │ │ │ ├── index-Wr40-hHf.js
│ │ │ │ ├── loading-C4J6hIxs.js
│ │ │ │ ├── loading-DkQYEuap.css
│ │ │ │ ├── vendor-CXe1BRiH.css
│ │ │ │ └── vendor-KSDcS24u.js
│ │ │ └── index.html
│ │ ├── java/
│ │ │ └── io/
│ │ │ └── legado/
│ │ │ └── app/
│ │ │ ├── App.kt
│ │ │ ├── README.md
│ │ │ ├── api/
│ │ │ │ ├── ReaderProvider.kt
│ │ │ │ ├── ReturnData.kt
│ │ │ │ ├── ShortCuts.kt
│ │ │ │ └── controller/
│ │ │ │ ├── BookController.kt
│ │ │ │ ├── BookSourceController.kt
│ │ │ │ ├── ReplaceRuleController.kt
│ │ │ │ └── RssSourceController.kt
│ │ │ ├── base/
│ │ │ │ ├── AppContextWrapper.kt
│ │ │ │ ├── BaseActivity.kt
│ │ │ │ ├── BaseDialogFragment.kt
│ │ │ │ ├── BaseFragment.kt
│ │ │ │ ├── BasePrefDialogFragment.kt
│ │ │ │ ├── BaseService.kt
│ │ │ │ ├── BaseViewModel.kt
│ │ │ │ ├── README.md
│ │ │ │ ├── VMBaseActivity.kt
│ │ │ │ ├── VMBaseFragment.kt
│ │ │ │ └── adapter/
│ │ │ │ ├── DiffRecyclerAdapter.kt
│ │ │ │ ├── ItemAnimation.kt
│ │ │ │ ├── ItemViewHolder.kt
│ │ │ │ ├── RecyclerAdapter.kt
│ │ │ │ └── animations/
│ │ │ │ ├── AlphaInAnimation.kt
│ │ │ │ ├── BaseAnimation.kt
│ │ │ │ ├── ScaleInAnimation.kt
│ │ │ │ ├── SlideInBottomAnimation.kt
│ │ │ │ ├── SlideInLeftAnimation.kt
│ │ │ │ └── SlideInRightAnimation.kt
│ │ │ ├── constant/
│ │ │ │ ├── AppConst.kt
│ │ │ │ ├── AppLog.kt
│ │ │ │ ├── AppPattern.kt
│ │ │ │ ├── BookSourceType.kt
│ │ │ │ ├── BookType.kt
│ │ │ │ ├── EventBus.kt
│ │ │ │ ├── IntentAction.kt
│ │ │ │ ├── NotificationId.kt
│ │ │ │ ├── PageAnim.kt
│ │ │ │ ├── PreferKey.kt
│ │ │ │ ├── SourceType.kt
│ │ │ │ ├── Status.kt
│ │ │ │ └── Theme.kt
│ │ │ ├── data/
│ │ │ │ ├── AppDatabase.kt
│ │ │ │ ├── DatabaseMigrations.kt
│ │ │ │ ├── README.md
│ │ │ │ ├── dao/
│ │ │ │ │ ├── BookChapterDao.kt
│ │ │ │ │ ├── BookDao.kt
│ │ │ │ │ ├── BookGroupDao.kt
│ │ │ │ │ ├── BookSourceDao.kt
│ │ │ │ │ ├── BookmarkDao.kt
│ │ │ │ │ ├── CacheDao.kt
│ │ │ │ │ ├── CookieDao.kt
│ │ │ │ │ ├── DictRuleDao.kt
│ │ │ │ │ ├── HttpTTSDao.kt
│ │ │ │ │ ├── KeyboardAssistsDao.kt
│ │ │ │ │ ├── ReadRecordDao.kt
│ │ │ │ │ ├── ReplaceRuleDao.kt
│ │ │ │ │ ├── RssArticleDao.kt
│ │ │ │ │ ├── RssReadRecordDao.kt
│ │ │ │ │ ├── RssSourceDao.kt
│ │ │ │ │ ├── RssStarDao.kt
│ │ │ │ │ ├── RuleSubDao.kt
│ │ │ │ │ ├── SearchBookDao.kt
│ │ │ │ │ ├── SearchKeywordDao.kt
│ │ │ │ │ ├── ServerDao.kt
│ │ │ │ │ └── TxtTocRuleDao.kt
│ │ │ │ └── entities/
│ │ │ │ ├── BaseBook.kt
│ │ │ │ ├── BaseRssArticle.kt
│ │ │ │ ├── BaseSource.kt
│ │ │ │ ├── Book.kt
│ │ │ │ ├── BookChapter.kt
│ │ │ │ ├── BookChapterReview.kt
│ │ │ │ ├── BookGroup.kt
│ │ │ │ ├── BookProgress.kt
│ │ │ │ ├── BookSource.kt
│ │ │ │ ├── BookSourcePart.kt
│ │ │ │ ├── Bookmark.kt
│ │ │ │ ├── Cache.kt
│ │ │ │ ├── Cookie.kt
│ │ │ │ ├── DictRule.kt
│ │ │ │ ├── HttpTTS.kt
│ │ │ │ ├── KeyboardAssist.kt
│ │ │ │ ├── ReadRecord.kt
│ │ │ │ ├── ReadRecordShow.kt
│ │ │ │ ├── ReplaceRule.kt
│ │ │ │ ├── RssArticle.kt
│ │ │ │ ├── RssReadRecord.kt
│ │ │ │ ├── RssSource.kt
│ │ │ │ ├── RssStar.kt
│ │ │ │ ├── RuleSub.kt
│ │ │ │ ├── SearchBook.kt
│ │ │ │ ├── SearchKeyword.kt
│ │ │ │ ├── Server.kt
│ │ │ │ ├── TxtTocRule.kt
│ │ │ │ └── rule/
│ │ │ │ ├── BookInfoRule.kt
│ │ │ │ ├── BookListRule.kt
│ │ │ │ ├── ContentRule.kt
│ │ │ │ ├── ExploreKind.kt
│ │ │ │ ├── ExploreRule.kt
│ │ │ │ ├── FlexChildStyle.kt
│ │ │ │ ├── ReviewRule.kt
│ │ │ │ ├── RowUi.kt
│ │ │ │ ├── SearchRule.kt
│ │ │ │ └── TocRule.kt
│ │ │ ├── exception/
│ │ │ │ ├── ConcurrentException.kt
│ │ │ │ ├── ContentEmptyException.kt
│ │ │ │ ├── EmptyFileException.kt
│ │ │ │ ├── InvalidBooksDirException.kt
│ │ │ │ ├── NoBooksDirException.kt
│ │ │ │ ├── NoStackTraceException.kt
│ │ │ │ ├── RegexTimeoutException.kt
│ │ │ │ └── TocEmptyException.kt
│ │ │ ├── help/
│ │ │ │ ├── AppFreezeMonitor.kt
│ │ │ │ ├── AppWebDav.kt
│ │ │ │ ├── CacheManager.kt
│ │ │ │ ├── ConcurrentRateLimiter.kt
│ │ │ │ ├── CrashHandler.kt
│ │ │ │ ├── DefaultData.kt
│ │ │ │ ├── DirectLinkUpload.kt
│ │ │ │ ├── DispatchersMonitor.kt
│ │ │ │ ├── EventMessage.kt
│ │ │ │ ├── ExecutorService.kt
│ │ │ │ ├── IntentData.kt
│ │ │ │ ├── IntentHelp.kt
│ │ │ │ ├── JsEncodeUtils.kt
│ │ │ │ ├── JsExtensions.kt
│ │ │ │ ├── LauncherIconHelp.kt
│ │ │ │ ├── LayoutManager.kt
│ │ │ │ ├── LifecycleHelp.kt
│ │ │ │ ├── MediaHelp.kt
│ │ │ │ ├── PaintPool.kt
│ │ │ │ ├── README.md
│ │ │ │ ├── ReplaceAnalyzer.kt
│ │ │ │ ├── RuleBigDataHelp.kt
│ │ │ │ ├── RuleComplete.kt
│ │ │ │ ├── TTS.kt
│ │ │ │ ├── book/
│ │ │ │ │ ├── BookContent.kt
│ │ │ │ │ ├── BookExtensions.kt
│ │ │ │ │ ├── BookHelp.kt
│ │ │ │ │ ├── ContentHelp.kt
│ │ │ │ │ └── ContentProcessor.kt
│ │ │ │ ├── config/
│ │ │ │ │ ├── AppConfig.kt
│ │ │ │ │ ├── LocalConfig.kt
│ │ │ │ │ ├── ReadBookConfig.kt
│ │ │ │ │ ├── ReadTipConfig.kt
│ │ │ │ │ ├── SourceConfig.kt
│ │ │ │ │ └── ThemeConfig.kt
│ │ │ │ ├── coroutine/
│ │ │ │ │ ├── ActivelyCancelException.kt
│ │ │ │ │ ├── CompositeCoroutine.kt
│ │ │ │ │ ├── Coroutine.kt
│ │ │ │ │ └── CoroutineContainer.kt
│ │ │ │ ├── crypto/
│ │ │ │ │ ├── AsymmetricCrypto.kt
│ │ │ │ │ ├── README.md
│ │ │ │ │ ├── Sign.kt
│ │ │ │ │ └── SymmetricCryptoAndroid.kt
│ │ │ │ ├── exoplayer/
│ │ │ │ │ ├── ExoPlayerHelper.kt
│ │ │ │ │ └── InputStreamDataSource.kt
│ │ │ │ ├── glide/
│ │ │ │ │ ├── AsyncRecycleBitmapPool.kt
│ │ │ │ │ ├── BlurTransformation.kt
│ │ │ │ │ ├── FilePathLoader.kt
│ │ │ │ │ ├── GlideHeaders.kt
│ │ │ │ │ ├── ImageLoader.kt
│ │ │ │ │ ├── LegadoDataUrlLoader.kt
│ │ │ │ │ ├── LegadoGlideModule.kt
│ │ │ │ │ ├── OkHttpModeLoaderFactory.kt
│ │ │ │ │ ├── OkHttpModelLoader.kt
│ │ │ │ │ ├── OkHttpStreamFetcher.kt
│ │ │ │ │ └── progress/
│ │ │ │ │ ├── OnProgressListener.kt
│ │ │ │ │ ├── ProgressManager.kt
│ │ │ │ │ └── ProgressResponseBody.kt
│ │ │ │ ├── http/
│ │ │ │ │ ├── BackstageWebView.kt
│ │ │ │ │ ├── CookieManager.kt
│ │ │ │ │ ├── CookieStore.kt
│ │ │ │ │ ├── Cronet.kt
│ │ │ │ │ ├── DecompressInterceptor.kt
│ │ │ │ │ ├── HttpHelper.kt
│ │ │ │ │ ├── ObsoleteUrlFactory.kt
│ │ │ │ │ ├── OkHttpExceptionInterceptor.kt
│ │ │ │ │ ├── OkHttpUtils.kt
│ │ │ │ │ ├── OkhttpUncaughtExceptionHandler.kt
│ │ │ │ │ ├── RequestMethod.kt
│ │ │ │ │ ├── SSLHelper.kt
│ │ │ │ │ ├── StrResponse.kt
│ │ │ │ │ └── api/
│ │ │ │ │ └── CookieManagerInterface.kt
│ │ │ │ ├── rhino/
│ │ │ │ │ └── NativeBaseSource.kt
│ │ │ │ ├── source/
│ │ │ │ │ ├── BaseSourceExtensions.kt
│ │ │ │ │ ├── BookSourceExtensions.kt
│ │ │ │ │ ├── RssSourceExtensions.kt
│ │ │ │ │ ├── SourceHelp.kt
│ │ │ │ │ └── SourceVerificationHelp.kt
│ │ │ │ ├── storage/
│ │ │ │ │ ├── Backup.kt
│ │ │ │ │ ├── BackupAES.kt
│ │ │ │ │ ├── BackupConfig.kt
│ │ │ │ │ ├── ImportOldData.kt
│ │ │ │ │ └── Restore.kt
│ │ │ │ └── update/
│ │ │ │ ├── AppReleaseInfo.kt
│ │ │ │ ├── AppUpdate.kt
│ │ │ │ └── AppUpdateGitHub.kt
│ │ │ ├── lib/
│ │ │ │ ├── README.md
│ │ │ │ ├── aliyun/
│ │ │ │ │ └── ALiYun.kt
│ │ │ │ ├── cronet/
│ │ │ │ │ ├── AbsCallBack.kt
│ │ │ │ │ ├── BodyUploadProvider.kt
│ │ │ │ │ ├── CallbackResult.kt
│ │ │ │ │ ├── CallbackStep.kt
│ │ │ │ │ ├── CronetCoroutineInterceptor.kt
│ │ │ │ │ ├── CronetHelper.kt
│ │ │ │ │ ├── CronetInterceptor.kt
│ │ │ │ │ ├── CronetLoader.kt
│ │ │ │ │ ├── LargeBodyUploadProvider.kt
│ │ │ │ │ ├── NewCallBack.kt
│ │ │ │ │ └── OldCallback.kt
│ │ │ │ ├── dialogs/
│ │ │ │ │ ├── AlertBuilder.kt
│ │ │ │ │ ├── AndroidAlertBuilder.kt
│ │ │ │ │ ├── AndroidDialogs.kt
│ │ │ │ │ ├── AndroidSelectors.kt
│ │ │ │ │ └── SelectItem.kt
│ │ │ │ ├── icu4j/
│ │ │ │ │ ├── CharsetDetector.java
│ │ │ │ │ ├── CharsetMatch.java
│ │ │ │ │ ├── CharsetRecog_2022.java
│ │ │ │ │ ├── CharsetRecog_UTF8.java
│ │ │ │ │ ├── CharsetRecog_Unicode.java
│ │ │ │ │ ├── CharsetRecog_mbcs.java
│ │ │ │ │ ├── CharsetRecog_sbcs.java
│ │ │ │ │ └── CharsetRecognizer.java
│ │ │ │ ├── mobi/
│ │ │ │ │ ├── KF6Book.kt
│ │ │ │ │ ├── KF8Book.kt
│ │ │ │ │ ├── MobiBook.kt
│ │ │ │ │ ├── MobiReader.kt
│ │ │ │ │ ├── PDBFile.kt
│ │ │ │ │ ├── decompress/
│ │ │ │ │ │ ├── CDICData.kt
│ │ │ │ │ │ ├── Decompressor.kt
│ │ │ │ │ │ ├── HuffcdicDecompressor.kt
│ │ │ │ │ │ ├── Lz77Decompressor.kt
│ │ │ │ │ │ └── PlainDecompressor.kt
│ │ │ │ │ ├── entities/
│ │ │ │ │ │ ├── ExthRecordType.kt
│ │ │ │ │ │ ├── FdstHeader.kt
│ │ │ │ │ │ ├── Fragment.kt
│ │ │ │ │ │ ├── IndexData.kt
│ │ │ │ │ │ ├── IndexEntry.kt
│ │ │ │ │ │ ├── IndexTag.kt
│ │ │ │ │ │ ├── IndxHeader.kt
│ │ │ │ │ │ ├── KF6Section.kt
│ │ │ │ │ │ ├── KF8Header.kt
│ │ │ │ │ │ ├── KF8Pos.kt
│ │ │ │ │ │ ├── KF8Resource.kt
│ │ │ │ │ │ ├── KF8Section.kt
│ │ │ │ │ │ ├── MobiEntryHeaders.kt
│ │ │ │ │ │ ├── MobiHeader.kt
│ │ │ │ │ │ ├── MobiMetadata.kt
│ │ │ │ │ │ ├── NCX.kt
│ │ │ │ │ │ ├── PalmDocHeader.kt
│ │ │ │ │ │ ├── Ptagx.kt
│ │ │ │ │ │ ├── Skeleton.kt
│ │ │ │ │ │ ├── TOC.kt
│ │ │ │ │ │ ├── TagxHeader.kt
│ │ │ │ │ │ └── TagxTag.kt
│ │ │ │ │ └── utils/
│ │ │ │ │ ├── BitwiseExtensions.kt
│ │ │ │ │ └── ByteBufferExtensions.kt
│ │ │ │ ├── permission/
│ │ │ │ │ ├── OnErrorCallback.kt
│ │ │ │ │ ├── OnPermissionsDeniedCallback.kt
│ │ │ │ │ ├── OnPermissionsGrantedCallback.kt
│ │ │ │ │ ├── OnPermissionsResultCallback.kt
│ │ │ │ │ ├── OnRequestPermissionsResultCallback.kt
│ │ │ │ │ ├── PermissionActivity.kt
│ │ │ │ │ ├── Permissions.kt
│ │ │ │ │ ├── PermissionsCompat.kt
│ │ │ │ │ ├── Request.kt
│ │ │ │ │ ├── RequestManager.kt
│ │ │ │ │ └── RequestPlugins.kt
│ │ │ │ ├── prefs/
│ │ │ │ │ ├── ColorPreference.kt
│ │ │ │ │ ├── EditTextPreference.kt
│ │ │ │ │ ├── EditTextPreferenceDialog.kt
│ │ │ │ │ ├── IconListPreference.kt
│ │ │ │ │ ├── ListPreferenceDialog.kt
│ │ │ │ │ ├── MultiSelectListPreferenceDialog.kt
│ │ │ │ │ ├── NameListPreference.kt
│ │ │ │ │ ├── Preference.kt
│ │ │ │ │ ├── PreferenceCategory.kt
│ │ │ │ │ ├── SwitchPreference.kt
│ │ │ │ │ └── fragment/
│ │ │ │ │ └── PreferenceFragment.kt
│ │ │ │ ├── theme/
│ │ │ │ │ ├── MaterialValueHelper.kt
│ │ │ │ │ ├── Selector.kt
│ │ │ │ │ ├── ThemeStore.kt
│ │ │ │ │ ├── ThemeStoreInterface.kt
│ │ │ │ │ ├── ThemeStorePrefKeys.kt
│ │ │ │ │ ├── ThemeUtils.kt
│ │ │ │ │ ├── TintHelper.kt
│ │ │ │ │ ├── ViewUtils.kt
│ │ │ │ │ └── view/
│ │ │ │ │ ├── ThemeBottomNavigationVIew.kt
│ │ │ │ │ ├── ThemeCheckBox.kt
│ │ │ │ │ ├── ThemeEditText.kt
│ │ │ │ │ ├── ThemeProgressBar.kt
│ │ │ │ │ ├── ThemeRadioButton.kt
│ │ │ │ │ ├── ThemeRadioNoButton.kt
│ │ │ │ │ ├── ThemeSeekBar.kt
│ │ │ │ │ └── ThemeSwitch.kt
│ │ │ │ └── webdav/
│ │ │ │ ├── Authorization.kt
│ │ │ │ ├── WebDav.kt
│ │ │ │ ├── WebDavException.kt
│ │ │ │ └── WebDavFile.kt
│ │ │ ├── model/
│ │ │ │ ├── AudioPlay.kt
│ │ │ │ ├── BookCover.kt
│ │ │ │ ├── CacheBook.kt
│ │ │ │ ├── CheckSource.kt
│ │ │ │ ├── Debug.kt
│ │ │ │ ├── Download.kt
│ │ │ │ ├── ImageProvider.kt
│ │ │ │ ├── README.md
│ │ │ │ ├── ReadAloud.kt
│ │ │ │ ├── ReadBook.kt
│ │ │ │ ├── ReadManga.kt
│ │ │ │ ├── SharedJsScope.kt
│ │ │ │ ├── analyzeRule/
│ │ │ │ │ ├── AnalyzeByJSonPath.kt
│ │ │ │ │ ├── AnalyzeByJSoup.kt
│ │ │ │ │ ├── AnalyzeByRegex.kt
│ │ │ │ │ ├── AnalyzeByXPath.kt
│ │ │ │ │ ├── AnalyzeRule.kt
│ │ │ │ │ ├── AnalyzeUrl.kt
│ │ │ │ │ ├── CustomUrl.kt
│ │ │ │ │ ├── QueryTTF.java
│ │ │ │ │ ├── RuleAnalyzer.kt
│ │ │ │ │ ├── RuleData.kt
│ │ │ │ │ └── RuleDataInterface.kt
│ │ │ │ ├── localBook/
│ │ │ │ │ ├── BaseLocalBookParse.kt
│ │ │ │ │ ├── EpubFile.kt
│ │ │ │ │ ├── LocalBook.kt
│ │ │ │ │ ├── MobiFile.kt
│ │ │ │ │ ├── PdfFile.kt
│ │ │ │ │ ├── README.md
│ │ │ │ │ ├── TextFile.kt
│ │ │ │ │ └── UmdFile.kt
│ │ │ │ ├── remote/
│ │ │ │ │ ├── RemoteBook.kt
│ │ │ │ │ ├── RemoteBookManager.kt
│ │ │ │ │ └── RemoteBookWebDav.kt
│ │ │ │ ├── rss/
│ │ │ │ │ ├── Rss.kt
│ │ │ │ │ ├── RssParserByRule.kt
│ │ │ │ │ └── RssParserDefault.kt
│ │ │ │ └── webBook/
│ │ │ │ ├── BookChapterList.kt
│ │ │ │ ├── BookContent.kt
│ │ │ │ ├── BookInfo.kt
│ │ │ │ ├── BookList.kt
│ │ │ │ ├── SearchModel.kt
│ │ │ │ └── WebBook.kt
│ │ │ ├── receiver/
│ │ │ │ ├── MediaButtonReceiver.kt
│ │ │ │ ├── NetworkChangedListener.kt
│ │ │ │ ├── SharedReceiverActivity.kt
│ │ │ │ └── TimeBatteryReceiver.kt
│ │ │ ├── service/
│ │ │ │ ├── AudioPlayService.kt
│ │ │ │ ├── BaseReadAloudService.kt
│ │ │ │ ├── CacheBookService.kt
│ │ │ │ ├── CheckSourceService.kt
│ │ │ │ ├── DownloadService.kt
│ │ │ │ ├── ExportBookService.kt
│ │ │ │ ├── HttpReadAloudService.kt
│ │ │ │ ├── README.md
│ │ │ │ ├── TTSReadAloudService.kt
│ │ │ │ ├── WebService.kt
│ │ │ │ └── WebTileService.kt
│ │ │ ├── ui/
│ │ │ │ ├── README.md
│ │ │ │ ├── about/
│ │ │ │ │ ├── AboutActivity.kt
│ │ │ │ │ ├── AboutFragment.kt
│ │ │ │ │ ├── AppLogDialog.kt
│ │ │ │ │ ├── CrashLogsDialog.kt
│ │ │ │ │ ├── ReadRecordActivity.kt
│ │ │ │ │ └── UpdateDialog.kt
│ │ │ │ ├── association/
│ │ │ │ │ ├── AddToBookshelfDialog.kt
│ │ │ │ │ ├── BaseAssociationViewModel.kt
│ │ │ │ │ ├── FileAssociationActivity.kt
│ │ │ │ │ ├── FileAssociationViewModel.kt
│ │ │ │ │ ├── ImportBookSourceDialog.kt
│ │ │ │ │ ├── ImportBookSourceViewModel.kt
│ │ │ │ │ ├── ImportDictRuleDialog.kt
│ │ │ │ │ ├── ImportDictRuleViewModel.kt
│ │ │ │ │ ├── ImportHttpTtsDialog.kt
│ │ │ │ │ ├── ImportHttpTtsViewModel.kt
│ │ │ │ │ ├── ImportReplaceRuleDialog.kt
│ │ │ │ │ ├── ImportReplaceRuleViewModel.kt
│ │ │ │ │ ├── ImportRssSourceDialog.kt
│ │ │ │ │ ├── ImportRssSourceViewModel.kt
│ │ │ │ │ ├── ImportThemeDialog.kt
│ │ │ │ │ ├── ImportThemeViewModel.kt
│ │ │ │ │ ├── ImportTxtTocRuleDialog.kt
│ │ │ │ │ ├── ImportTxtTocRuleViewModel.kt
│ │ │ │ │ ├── OnLineImportActivity.kt
│ │ │ │ │ ├── OnLineImportViewModel.kt
│ │ │ │ │ ├── OpenUrlConfirmActivity.kt
│ │ │ │ │ ├── OpenUrlConfirmDialog.kt
│ │ │ │ │ ├── OpenUrlConfirmViewModel.kt
│ │ │ │ │ ├── VerificationCodeActivity.kt
│ │ │ │ │ ├── VerificationCodeDialog.kt
│ │ │ │ │ └── VerificationCodeViewModel.kt
│ │ │ │ ├── book/
│ │ │ │ │ ├── audio/
│ │ │ │ │ │ ├── AudioPlayActivity.kt
│ │ │ │ │ │ ├── AudioPlayViewModel.kt
│ │ │ │ │ │ └── TimerSliderPopup.kt
│ │ │ │ │ ├── bookmark/
│ │ │ │ │ │ ├── AllBookmarkActivity.kt
│ │ │ │ │ │ ├── AllBookmarkViewModel.kt
│ │ │ │ │ │ ├── BookmarkAdapter.kt
│ │ │ │ │ │ ├── BookmarkDecoration.kt
│ │ │ │ │ │ └── BookmarkDialog.kt
│ │ │ │ │ ├── cache/
│ │ │ │ │ │ ├── CacheActivity.kt
│ │ │ │ │ │ ├── CacheAdapter.kt
│ │ │ │ │ │ └── CacheViewModel.kt
│ │ │ │ │ ├── changecover/
│ │ │ │ │ │ ├── ChangeCoverDialog.kt
│ │ │ │ │ │ ├── ChangeCoverViewModel.kt
│ │ │ │ │ │ └── CoverAdapter.kt
│ │ │ │ │ ├── changesource/
│ │ │ │ │ │ ├── ChangeBookSourceAdapter.kt
│ │ │ │ │ │ ├── ChangeBookSourceDialog.kt
│ │ │ │ │ │ ├── ChangeBookSourceViewModel.kt
│ │ │ │ │ │ ├── ChangeChapterSourceAdapter.kt
│ │ │ │ │ │ ├── ChangeChapterSourceDialog.kt
│ │ │ │ │ │ ├── ChangeChapterSourceViewModel.kt
│ │ │ │ │ │ └── ChangeChapterTocAdapter.kt
│ │ │ │ │ ├── explore/
│ │ │ │ │ │ ├── ExploreShowActivity.kt
│ │ │ │ │ │ ├── ExploreShowAdapter.kt
│ │ │ │ │ │ └── ExploreShowViewModel.kt
│ │ │ │ │ ├── group/
│ │ │ │ │ │ ├── GroupEditDialog.kt
│ │ │ │ │ │ ├── GroupManageDialog.kt
│ │ │ │ │ │ ├── GroupSelectDialog.kt
│ │ │ │ │ │ └── GroupViewModel.kt
│ │ │ │ │ ├── import/
│ │ │ │ │ │ ├── BaseImportBookActivity.kt
│ │ │ │ │ │ ├── local/
│ │ │ │ │ │ │ ├── ImportBook.kt
│ │ │ │ │ │ │ ├── ImportBookActivity.kt
│ │ │ │ │ │ │ ├── ImportBookAdapter.kt
│ │ │ │ │ │ │ └── ImportBookViewModel.kt
│ │ │ │ │ │ └── remote/
│ │ │ │ │ │ ├── RemoteBookActivity.kt
│ │ │ │ │ │ ├── RemoteBookAdapter.kt
│ │ │ │ │ │ ├── RemoteBookSort.kt
│ │ │ │ │ │ ├── RemoteBookViewModel.kt
│ │ │ │ │ │ ├── ServerConfigDialog.kt
│ │ │ │ │ │ ├── ServerConfigViewModel.kt
│ │ │ │ │ │ ├── ServersDialog.kt
│ │ │ │ │ │ └── ServersViewModel.kt
│ │ │ │ │ ├── info/
│ │ │ │ │ │ ├── BookInfoActivity.kt
│ │ │ │ │ │ ├── BookInfoViewModel.kt
│ │ │ │ │ │ └── edit/
│ │ │ │ │ │ ├── BookInfoEditActivity.kt
│ │ │ │ │ │ └── BookInfoEditViewModel.kt
│ │ │ │ │ ├── manage/
│ │ │ │ │ │ ├── BookAdapter.kt
│ │ │ │ │ │ ├── BookshelfManageActivity.kt
│ │ │ │ │ │ ├── BookshelfManageViewModel.kt
│ │ │ │ │ │ └── SourcePickerDialog.kt
│ │ │ │ │ ├── manga/
│ │ │ │ │ │ ├── ReadMangaActivity.kt
│ │ │ │ │ │ ├── ReadMangaViewModel.kt
│ │ │ │ │ │ ├── config/
│ │ │ │ │ │ │ ├── MangaColorFilterConfig.kt
│ │ │ │ │ │ │ ├── MangaColorFilterDialog.kt
│ │ │ │ │ │ │ ├── MangaEpaperDialog.kt
│ │ │ │ │ │ │ ├── MangaFooterConfig.kt
│ │ │ │ │ │ │ └── MangaFooterSettingDialog.kt
│ │ │ │ │ │ ├── entities/
│ │ │ │ │ │ │ ├── BaseMangaPage.kt
│ │ │ │ │ │ │ ├── EpaperTransformation.kt
│ │ │ │ │ │ │ ├── GrayscaleTransformation.kt
│ │ │ │ │ │ │ ├── MangaChapter.kt
│ │ │ │ │ │ │ ├── MangaContent.kt
│ │ │ │ │ │ │ ├── MangaPage.kt
│ │ │ │ │ │ │ └── ReaderLoading.kt
│ │ │ │ │ │ └── recyclerview/
│ │ │ │ │ │ ├── GestureDetectorWithLongTap.kt
│ │ │ │ │ │ ├── MangaAdapter.kt
│ │ │ │ │ │ ├── MangaLayoutManager.kt
│ │ │ │ │ │ ├── MangaVH.kt
│ │ │ │ │ │ ├── ScrollTimer.kt
│ │ │ │ │ │ ├── WebtoonFrame.kt
│ │ │ │ │ │ └── WebtoonRecyclerView.kt
│ │ │ │ │ ├── read/
│ │ │ │ │ │ ├── BaseReadBookActivity.kt
│ │ │ │ │ │ ├── ContentEditDialog.kt
│ │ │ │ │ │ ├── EffectiveReplacesDialog.kt
│ │ │ │ │ │ ├── MangaMenu.kt
│ │ │ │ │ │ ├── ReadBookActivity.kt
│ │ │ │ │ │ ├── ReadBookViewModel.kt
│ │ │ │ │ │ ├── ReadMenu.kt
│ │ │ │ │ │ ├── SearchMenu.kt
│ │ │ │ │ │ ├── TextActionMenu.kt
│ │ │ │ │ │ ├── config/
│ │ │ │ │ │ │ ├── AutoReadDialog.kt
│ │ │ │ │ │ │ ├── BgAdapter.kt
│ │ │ │ │ │ │ ├── BgTextConfigDialog.kt
│ │ │ │ │ │ │ ├── ChineseConverter.kt
│ │ │ │ │ │ │ ├── ClickActionConfigDialog.kt
│ │ │ │ │ │ │ ├── HttpTtsEditDialog.kt
│ │ │ │ │ │ │ ├── HttpTtsEditViewModel.kt
│ │ │ │ │ │ │ ├── MoreConfigDialog.kt
│ │ │ │ │ │ │ ├── PaddingConfigDialog.kt
│ │ │ │ │ │ │ ├── PageKeyDialog.kt
│ │ │ │ │ │ │ ├── ReadAloudConfigDialog.kt
│ │ │ │ │ │ │ ├── ReadAloudDialog.kt
│ │ │ │ │ │ │ ├── ReadStyleDialog.kt
│ │ │ │ │ │ │ ├── SpeakEngineDialog.kt
│ │ │ │ │ │ │ ├── SpeakEngineViewModel.kt
│ │ │ │ │ │ │ ├── TextFontWeightConverter.kt
│ │ │ │ │ │ │ └── TipConfigDialog.kt
│ │ │ │ │ │ └── page/
│ │ │ │ │ │ ├── AutoPager.kt
│ │ │ │ │ │ ├── ContentTextView.kt
│ │ │ │ │ │ ├── PageView.kt
│ │ │ │ │ │ ├── ReadView.kt
│ │ │ │ │ │ ├── api/
│ │ │ │ │ │ │ ├── DataSource.kt
│ │ │ │ │ │ │ └── PageFactory.kt
│ │ │ │ │ │ ├── delegate/
│ │ │ │ │ │ │ ├── CoverPageDelegate.kt
│ │ │ │ │ │ │ ├── HorizontalPageDelegate.kt
│ │ │ │ │ │ │ ├── NoAnimPageDelegate.kt
│ │ │ │ │ │ │ ├── PageDelegate.kt
│ │ │ │ │ │ │ ├── ScrollPageDelegate.kt
│ │ │ │ │ │ │ ├── SimulationPageDelegate.kt
│ │ │ │ │ │ │ └── SlidePageDelegate.kt
│ │ │ │ │ │ ├── entities/
│ │ │ │ │ │ │ ├── PageDirection.kt
│ │ │ │ │ │ │ ├── TextChapter.kt
│ │ │ │ │ │ │ ├── TextLine.kt
│ │ │ │ │ │ │ ├── TextPage.kt
│ │ │ │ │ │ │ ├── TextParagraph.kt
│ │ │ │ │ │ │ ├── TextPos.kt
│ │ │ │ │ │ │ └── column/
│ │ │ │ │ │ │ ├── BaseColumn.kt
│ │ │ │ │ │ │ ├── ButtonColumn.kt
│ │ │ │ │ │ │ ├── ImageColumn.kt
│ │ │ │ │ │ │ ├── ReviewColumn.kt
│ │ │ │ │ │ │ └── TextColumn.kt
│ │ │ │ │ │ └── provider/
│ │ │ │ │ │ ├── ChapterProvider.kt
│ │ │ │ │ │ ├── LayoutProgressListener.kt
│ │ │ │ │ │ ├── TextChapterLayout.kt
│ │ │ │ │ │ ├── TextMeasure.kt
│ │ │ │ │ │ ├── TextPageFactory.kt
│ │ │ │ │ │ └── ZhLayout.kt
│ │ │ │ │ ├── search/
│ │ │ │ │ │ ├── BookAdapter.kt
│ │ │ │ │ │ ├── HistoryKeyAdapter.kt
│ │ │ │ │ │ ├── SearchActivity.kt
│ │ │ │ │ │ ├── SearchAdapter.kt
│ │ │ │ │ │ ├── SearchScope.kt
│ │ │ │ │ │ ├── SearchScopeDialog.kt
│ │ │ │ │ │ └── SearchViewModel.kt
│ │ │ │ │ ├── searchContent/
│ │ │ │ │ │ ├── SearchContentActivity.kt
│ │ │ │ │ │ ├── SearchContentAdapter.kt
│ │ │ │ │ │ ├── SearchContentViewModel.kt
│ │ │ │ │ │ └── SearchResult.kt
│ │ │ │ │ ├── source/
│ │ │ │ │ │ ├── debug/
│ │ │ │ │ │ │ ├── BookSourceDebugActivity.kt
│ │ │ │ │ │ │ ├── BookSourceDebugAdapter.kt
│ │ │ │ │ │ │ └── BookSourceDebugModel.kt
│ │ │ │ │ │ ├── edit/
│ │ │ │ │ │ │ ├── BookSourceEditActivity.kt
│ │ │ │ │ │ │ ├── BookSourceEditAdapter.kt
│ │ │ │ │ │ │ └── BookSourceEditViewModel.kt
│ │ │ │ │ │ └── manage/
│ │ │ │ │ │ ├── BookSourceActivity.kt
│ │ │ │ │ │ ├── BookSourceAdapter.kt
│ │ │ │ │ │ ├── BookSourceSort.kt
│ │ │ │ │ │ ├── BookSourceViewModel.kt
│ │ │ │ │ │ └── GroupManageDialog.kt
│ │ │ │ │ └── toc/
│ │ │ │ │ ├── BookmarkAdapter.kt
│ │ │ │ │ ├── BookmarkFragment.kt
│ │ │ │ │ ├── ChapterListAdapter.kt
│ │ │ │ │ ├── ChapterListFragment.kt
│ │ │ │ │ ├── TocActivity.kt
│ │ │ │ │ ├── TocActivityResult.kt
│ │ │ │ │ ├── TocViewModel.kt
│ │ │ │ │ └── rule/
│ │ │ │ │ ├── TxtTocRuleActivity.kt
│ │ │ │ │ ├── TxtTocRuleAdapter.kt
│ │ │ │ │ ├── TxtTocRuleDialog.kt
│ │ │ │ │ ├── TxtTocRuleEditDialog.kt
│ │ │ │ │ └── TxtTocRuleViewModel.kt
│ │ │ │ ├── browser/
│ │ │ │ │ ├── WebViewActivity.kt
│ │ │ │ │ └── WebViewModel.kt
│ │ │ │ ├── config/
│ │ │ │ │ ├── BackupConfigFragment.kt
│ │ │ │ │ ├── CheckSourceConfig.kt
│ │ │ │ │ ├── ConfigActivity.kt
│ │ │ │ │ ├── ConfigTag.kt
│ │ │ │ │ ├── ConfigViewModel.kt
│ │ │ │ │ ├── CoverConfigFragment.kt
│ │ │ │ │ ├── CoverRuleConfigDialog.kt
│ │ │ │ │ ├── DirectLinkUploadConfig.kt
│ │ │ │ │ ├── OtherConfigFragment.kt
│ │ │ │ │ ├── ThemeConfigFragment.kt
│ │ │ │ │ ├── ThemeListDialog.kt
│ │ │ │ │ └── WelcomeConfigFragment.kt
│ │ │ │ ├── dict/
│ │ │ │ │ ├── DictDialog.kt
│ │ │ │ │ ├── DictViewModel.kt
│ │ │ │ │ └── rule/
│ │ │ │ │ ├── DictRuleActivity.kt
│ │ │ │ │ ├── DictRuleAdapter.kt
│ │ │ │ │ ├── DictRuleEditDialog.kt
│ │ │ │ │ └── DictRuleViewModel.kt
│ │ │ │ ├── file/
│ │ │ │ │ ├── FileManageActivity.kt
│ │ │ │ │ ├── FileManageViewModel.kt
│ │ │ │ │ ├── FilePickerDialog.kt
│ │ │ │ │ ├── FilePickerViewModel.kt
│ │ │ │ │ ├── HandleFileActivity.kt
│ │ │ │ │ ├── HandleFileContract.kt
│ │ │ │ │ ├── HandleFileViewModel.kt
│ │ │ │ │ └── utils/
│ │ │ │ │ └── FilePickerIcon.java
│ │ │ │ ├── font/
│ │ │ │ │ ├── FontAdapter.kt
│ │ │ │ │ └── FontSelectDialog.kt
│ │ │ │ ├── login/
│ │ │ │ │ ├── SourceLoginActivity.kt
│ │ │ │ │ ├── SourceLoginDialog.kt
│ │ │ │ │ ├── SourceLoginViewModel.kt
│ │ │ │ │ └── WebViewLoginFragment.kt
│ │ │ │ ├── main/
│ │ │ │ │ ├── MainActivity.kt
│ │ │ │ │ ├── MainFragmentInterface.kt
│ │ │ │ │ ├── MainViewModel.kt
│ │ │ │ │ ├── bookshelf/
│ │ │ │ │ │ ├── BaseBookshelfFragment.kt
│ │ │ │ │ │ ├── BookshelfViewModel.kt
│ │ │ │ │ │ ├── style1/
│ │ │ │ │ │ │ ├── BookshelfFragment1.kt
│ │ │ │ │ │ │ └── books/
│ │ │ │ │ │ │ ├── BaseBooksAdapter.kt
│ │ │ │ │ │ │ ├── BooksAdapterGrid.kt
│ │ │ │ │ │ │ ├── BooksAdapterList.kt
│ │ │ │ │ │ │ └── BooksFragment.kt
│ │ │ │ │ │ └── style2/
│ │ │ │ │ │ ├── BaseBooksAdapter.kt
│ │ │ │ │ │ ├── BooksAdapterGrid.kt
│ │ │ │ │ │ ├── BooksAdapterList.kt
│ │ │ │ │ │ └── BookshelfFragment2.kt
│ │ │ │ │ ├── explore/
│ │ │ │ │ │ ├── ExploreAdapter.kt
│ │ │ │ │ │ ├── ExploreDiffItemCallBack.kt
│ │ │ │ │ │ ├── ExploreFragment.kt
│ │ │ │ │ │ └── ExploreViewModel.kt
│ │ │ │ │ ├── my/
│ │ │ │ │ │ └── MyFragment.kt
│ │ │ │ │ └── rss/
│ │ │ │ │ ├── RssAdapter.kt
│ │ │ │ │ ├── RssFragment.kt
│ │ │ │ │ └── RssViewModel.kt
│ │ │ │ ├── qrcode/
│ │ │ │ │ ├── QrCodeActivity.kt
│ │ │ │ │ ├── QrCodeFragment.kt
│ │ │ │ │ ├── QrCodeResult.kt
│ │ │ │ │ └── ScanResultCallback.kt
│ │ │ │ ├── replace/
│ │ │ │ │ ├── GroupManageDialog.kt
│ │ │ │ │ ├── ReplaceRuleActivity.kt
│ │ │ │ │ ├── ReplaceRuleAdapter.kt
│ │ │ │ │ ├── ReplaceRuleViewModel.kt
│ │ │ │ │ └── edit/
│ │ │ │ │ ├── ReplaceEditActivity.kt
│ │ │ │ │ └── ReplaceEditViewModel.kt
│ │ │ │ ├── rss/
│ │ │ │ │ ├── article/
│ │ │ │ │ │ ├── BaseRssArticlesAdapter.kt
│ │ │ │ │ │ ├── ReadRecordDialog.kt
│ │ │ │ │ │ ├── RssArticlesAdapter.kt
│ │ │ │ │ │ ├── RssArticlesAdapter1.kt
│ │ │ │ │ │ ├── RssArticlesAdapter2.kt
│ │ │ │ │ │ ├── RssArticlesFragment.kt
│ │ │ │ │ │ ├── RssArticlesViewModel.kt
│ │ │ │ │ │ ├── RssSortActivity.kt
│ │ │ │ │ │ └── RssSortViewModel.kt
│ │ │ │ │ ├── favorites/
│ │ │ │ │ │ ├── RssFavoritesActivity.kt
│ │ │ │ │ │ ├── RssFavoritesAdapter.kt
│ │ │ │ │ │ ├── RssFavoritesDialog.kt
│ │ │ │ │ │ ├── RssFavoritesFragment.kt
│ │ │ │ │ │ └── RssFavoritesViewModel.kt
│ │ │ │ │ ├── read/
│ │ │ │ │ │ ├── ReadRssActivity.kt
│ │ │ │ │ │ ├── ReadRssViewModel.kt
│ │ │ │ │ │ ├── RssJsExtensions.kt
│ │ │ │ │ │ └── VisibleWebView.kt
│ │ │ │ │ ├── source/
│ │ │ │ │ │ ├── debug/
│ │ │ │ │ │ │ ├── RssSourceDebugActivity.kt
│ │ │ │ │ │ │ ├── RssSourceDebugAdapter.kt
│ │ │ │ │ │ │ └── RssSourceDebugModel.kt
│ │ │ │ │ │ ├── edit/
│ │ │ │ │ │ │ ├── RssSourceEditActivity.kt
│ │ │ │ │ │ │ ├── RssSourceEditAdapter.kt
│ │ │ │ │ │ │ └── RssSourceEditViewModel.kt
│ │ │ │ │ │ └── manage/
│ │ │ │ │ │ ├── GroupManageDialog.kt
│ │ │ │ │ │ ├── RssSourceActivity.kt
│ │ │ │ │ │ ├── RssSourceAdapter.kt
│ │ │ │ │ │ └── RssSourceViewModel.kt
│ │ │ │ │ └── subscription/
│ │ │ │ │ ├── RuleSubActivity.kt
│ │ │ │ │ └── RuleSubAdapter.kt
│ │ │ │ ├── welcome/
│ │ │ │ │ └── WelcomeActivity.kt
│ │ │ │ └── widget/
│ │ │ │ ├── BatteryView.kt
│ │ │ │ ├── DetailSeekBar.kt
│ │ │ │ ├── LabelsBar.kt
│ │ │ │ ├── NoChildScrollNestedScrollView.kt
│ │ │ │ ├── PopupAction.kt
│ │ │ │ ├── ReaderInfoBarView.kt
│ │ │ │ ├── SearchView.kt
│ │ │ │ ├── SelectActionBar.kt
│ │ │ │ ├── ShadowLayout.kt
│ │ │ │ ├── TitleBar.kt
│ │ │ │ ├── anima/
│ │ │ │ │ ├── RefreshProgressBar.kt
│ │ │ │ │ ├── RotateLoading.kt
│ │ │ │ │ └── explosion_field/
│ │ │ │ │ ├── ExplosionAnimator.kt
│ │ │ │ │ ├── ExplosionField.kt
│ │ │ │ │ ├── ExplosionView.kt
│ │ │ │ │ ├── OnAnimatorListener.kt
│ │ │ │ │ └── Utils.kt
│ │ │ │ ├── checkbox/
│ │ │ │ │ └── SmoothCheckBox.kt
│ │ │ │ ├── code/
│ │ │ │ │ ├── CodeView.kt
│ │ │ │ │ ├── CodeViewExtensions.kt
│ │ │ │ │ └── KeywordTokenizer.kt
│ │ │ │ ├── dialog/
│ │ │ │ │ ├── CodeDialog.kt
│ │ │ │ │ ├── PhotoDialog.kt
│ │ │ │ │ ├── TextDialog.kt
│ │ │ │ │ ├── TextListDialog.kt
│ │ │ │ │ ├── UrlOptionDialog.kt
│ │ │ │ │ ├── VariableDialog.kt
│ │ │ │ │ └── WaitDialog.kt
│ │ │ │ ├── dynamiclayout/
│ │ │ │ │ ├── DynamicFrameLayout.kt
│ │ │ │ │ └── ViewSwitcher.kt
│ │ │ │ ├── image/
│ │ │ │ │ ├── ArcView.kt
│ │ │ │ │ ├── CircleImageView.kt
│ │ │ │ │ ├── CoverImageView.kt
│ │ │ │ │ ├── FilletImageView.kt
│ │ │ │ │ ├── ImageButton.kt
│ │ │ │ │ ├── PhotoView.kt
│ │ │ │ │ └── photo/
│ │ │ │ │ ├── Info.kt
│ │ │ │ │ └── RotateGestureDetector.kt
│ │ │ │ ├── keyboard/
│ │ │ │ │ ├── KeyboardAssistsConfig.kt
│ │ │ │ │ └── KeyboardToolPop.kt
│ │ │ │ ├── number/
│ │ │ │ │ └── NumberPickerDialog.kt
│ │ │ │ ├── recycler/
│ │ │ │ │ ├── DividerNoLast.kt
│ │ │ │ │ ├── DragSelectTouchHelper.kt
│ │ │ │ │ ├── HeaderAdapterDataObserver.kt
│ │ │ │ │ ├── ItemTouchCallback.kt
│ │ │ │ │ ├── LoadMoreView.kt
│ │ │ │ │ ├── NoChildScrollLinearLayoutManager.kt
│ │ │ │ │ ├── RecyclerViewAtPager2.kt
│ │ │ │ │ ├── UpLinearLayoutManager.kt
│ │ │ │ │ ├── VerticalDivider.kt
│ │ │ │ │ ├── ViewPager2Container.kt
│ │ │ │ │ └── scroller/
│ │ │ │ │ ├── FastScrollRecyclerView.kt
│ │ │ │ │ ├── FastScrollStateChangeListener.kt
│ │ │ │ │ └── FastScroller.kt
│ │ │ │ ├── seekbar/
│ │ │ │ │ ├── SeekBarChangeListener.kt
│ │ │ │ │ ├── VerticalSeekBar.kt
│ │ │ │ │ └── VerticalSeekBarWrapper.kt
│ │ │ │ └── text/
│ │ │ │ ├── AccentBgTextView.kt
│ │ │ │ ├── AccentStrokeTextView.kt
│ │ │ │ ├── AccentTextView.kt
│ │ │ │ ├── AutoCompleteTextView.kt
│ │ │ │ ├── BadgeView.kt
│ │ │ │ ├── BevelLabelView.kt
│ │ │ │ ├── EditEntity.kt
│ │ │ │ ├── MultilineTextView.kt
│ │ │ │ ├── PrimaryTextView.kt
│ │ │ │ ├── ScrollMultiAutoCompleteTextView.kt
│ │ │ │ ├── ScrollTextView.kt
│ │ │ │ ├── SecondaryTextView.kt
│ │ │ │ ├── StrokeTextView.kt
│ │ │ │ └── TextInputLayout.kt
│ │ │ ├── utils/
│ │ │ │ ├── ACache.kt
│ │ │ │ ├── ActivityExtensions.kt
│ │ │ │ ├── ActivityResult.kt
│ │ │ │ ├── ActivityResultContracts.kt
│ │ │ │ ├── AlphanumComparator.kt
│ │ │ │ ├── AnimationExtensions.kt
│ │ │ │ ├── ArchiveUtils.kt
│ │ │ │ ├── AsyncFileHandler.kt
│ │ │ │ ├── BitmapUtils.kt
│ │ │ │ ├── BookChapterExtensions.kt
│ │ │ │ ├── ByteArrayExtensions.kt
│ │ │ │ ├── ChineseUtils.kt
│ │ │ │ ├── CollectionExtensions.kt
│ │ │ │ ├── ColorUtils.kt
│ │ │ │ ├── ConfigurationExtensions.kt
│ │ │ │ ├── ConflateLiveData.kt
│ │ │ │ ├── ConstraintModify.kt
│ │ │ │ ├── ContextExtensions.kt
│ │ │ │ ├── ConvertExtensions.kt
│ │ │ │ ├── CookieManagerExtensions.kt
│ │ │ │ ├── CoroutineExtensions.kt
│ │ │ │ ├── CustomExportUtils.kt
│ │ │ │ ├── Debounce.kt
│ │ │ │ ├── DebugLog.kt
│ │ │ │ ├── DialogExtensions.kt
│ │ │ │ ├── DocumentUtils.kt
│ │ │ │ ├── DrawableUtils.kt
│ │ │ │ ├── EncoderUtils.kt
│ │ │ │ ├── EncodingDetect.kt
│ │ │ │ ├── EventBusExtensions.kt
│ │ │ │ ├── FileDocExtensions.kt
│ │ │ │ ├── FileExtensions.kt
│ │ │ │ ├── FileUtils.kt
│ │ │ │ ├── FlowExtensions.kt
│ │ │ │ ├── FragmentExtensions.kt
│ │ │ │ ├── GsonExtensions.kt
│ │ │ │ ├── HandlerUtils.kt
│ │ │ │ ├── HtmlFormatter.kt
│ │ │ │ ├── ImageUtils.kt
│ │ │ │ ├── InputStreamExtensions.kt
│ │ │ │ ├── IntentExtensions.kt
│ │ │ │ ├── IntentType.kt
│ │ │ │ ├── JsURL.kt
│ │ │ │ ├── JsonExtensions.kt
│ │ │ │ ├── JsoupExtensions.kt
│ │ │ │ ├── LogUtils.kt
│ │ │ │ ├── MD5Utils.kt
│ │ │ │ ├── MapExtensions.kt
│ │ │ │ ├── MenuExtensions.kt
│ │ │ │ ├── MenuItemExtensions.kt
│ │ │ │ ├── MutableLiveDataExtensions.kt
│ │ │ │ ├── NavigationViewUtils.kt
│ │ │ │ ├── NetworkUtils.kt
│ │ │ │ ├── PaintExtensions.kt
│ │ │ │ ├── ParcelFileDescriptorChannel.kt
│ │ │ │ ├── PreferencesExtensions.kt
│ │ │ │ ├── QRCodeUtils.kt
│ │ │ │ ├── RandomColor.kt
│ │ │ │ ├── RealPathUtil.kt
│ │ │ │ ├── RecyclerViewExtensions.kt
│ │ │ │ ├── RegexExtensions.kt
│ │ │ │ ├── RequestManagerExtensions.kt
│ │ │ │ ├── Snackbars.kt
│ │ │ │ ├── StringExtensions.kt
│ │ │ │ ├── StringUtils.kt
│ │ │ │ ├── SvgUtils.kt
│ │ │ │ ├── SyncedRenderer.kt
│ │ │ │ ├── SystemUtils.kt
│ │ │ │ ├── Throttle.kt
│ │ │ │ ├── ThrowableExtensions.kt
│ │ │ │ ├── TimeUtils.kt
│ │ │ │ ├── ToastUtils.kt
│ │ │ │ ├── ToolBarExtensions.kt
│ │ │ │ ├── UriExtensions.kt
│ │ │ │ ├── UrlUtil.kt
│ │ │ │ ├── Utf8BomUtils.kt
│ │ │ │ ├── ViewExtensions.kt
│ │ │ │ ├── WebSettingsExtensions.kt
│ │ │ │ ├── WindowInsetsExtensions.kt
│ │ │ │ ├── canvasrecorder/
│ │ │ │ │ ├── BaseCanvasRecorder.kt
│ │ │ │ │ ├── CanvasRecorder.kt
│ │ │ │ │ ├── CanvasRecorderApi23Impl.kt
│ │ │ │ │ ├── CanvasRecorderApi29Impl.kt
│ │ │ │ │ ├── CanvasRecorderExtensions.kt
│ │ │ │ │ ├── CanvasRecorderFactory.kt
│ │ │ │ │ ├── CanvasRecorderImpl.kt
│ │ │ │ │ ├── CanvasRecorderLocked.kt
│ │ │ │ │ └── pools/
│ │ │ │ │ ├── CanvasPool.kt
│ │ │ │ │ ├── PicturePool.kt
│ │ │ │ │ └── RenderNodePool.kt
│ │ │ │ ├── compress/
│ │ │ │ │ ├── LibArchiveUtils.kt
│ │ │ │ │ └── ZipUtils.kt
│ │ │ │ ├── objectpool/
│ │ │ │ │ ├── BaseObjectPool.kt
│ │ │ │ │ ├── BaseSafeObjectPool.kt
│ │ │ │ │ ├── ObjectPool.kt
│ │ │ │ │ ├── ObjectPoolExtensions.kt
│ │ │ │ │ └── ObjectPoolLocked.kt
│ │ │ │ └── viewbindingdelegate/
│ │ │ │ ├── ActivityViewBindings.kt
│ │ │ │ ├── FragmentViewBindings.kt
│ │ │ │ └── ViewBindingProperty.kt
│ │ │ └── web/
│ │ │ ├── HttpServer.kt
│ │ │ ├── ReadMe.md
│ │ │ ├── WebSocketServer.kt
│ │ │ ├── socket/
│ │ │ │ ├── BookSearchWebSocket.kt
│ │ │ │ ├── BookSourceDebugWebSocket.kt
│ │ │ │ └── RssSourceDebugWebSocket.kt
│ │ │ └── utils/
│ │ │ └── AssetsWeb.kt
│ │ └── res/
│ │ ├── anim/
│ │ │ ├── anim_none.xml
│ │ │ ├── anim_readbook_bottom_in.xml
│ │ │ ├── anim_readbook_bottom_out.xml
│ │ │ ├── anim_readbook_top_in.xml
│ │ │ └── anim_readbook_top_out.xml
│ │ ├── color/
│ │ │ └── selector_image.xml
│ │ ├── drawable/
│ │ │ ├── bg_chapter_item_divider.xml
│ │ │ ├── bg_edit.xml
│ │ │ ├── bg_eink_border_bottom.xml
│ │ │ ├── bg_eink_border_dialog.xml
│ │ │ ├── bg_eink_border_top.xml
│ │ │ ├── bg_find_book_group.xml
│ │ │ ├── bg_gradient.xml
│ │ │ ├── bg_img_border.xml
│ │ │ ├── bg_item_focused_on_tv.xml
│ │ │ ├── bg_popup_menu.xml
│ │ │ ├── bg_prefs_color.xml
│ │ │ ├── bg_searchview.xml
│ │ │ ├── bg_shadow_bottom.xml
│ │ │ ├── bg_shadow_bottom_night.xml
│ │ │ ├── bg_shadow_top.xml
│ │ │ ├── bg_shadow_top_night.xml
│ │ │ ├── bg_textfield_search.xml
│ │ │ ├── fastscroll_bubble.xml
│ │ │ ├── fastscroll_handle.xml
│ │ │ ├── fastscroll_track.xml
│ │ │ ├── ic_add.xml
│ │ │ ├── ic_add_online.xml
│ │ │ ├── ic_arrange.xml
│ │ │ ├── ic_arrow_back.xml
│ │ │ ├── ic_arrow_down.xml
│ │ │ ├── ic_arrow_drop_down.xml
│ │ │ ├── ic_arrow_drop_up.xml
│ │ │ ├── ic_arrow_right.xml
│ │ │ ├── ic_author.xml
│ │ │ ├── ic_auto_page.xml
│ │ │ ├── ic_auto_page_stop.xml
│ │ │ ├── ic_backup.xml
│ │ │ ├── ic_baseline_close.xml
│ │ │ ├── ic_baseline_sort_24.xml
│ │ │ ├── ic_book_has.xml
│ │ │ ├── ic_book_last.xml
│ │ │ ├── ic_bookmark.xml
│ │ │ ├── ic_bottom_books.xml
│ │ │ ├── ic_bottom_books_e.xml
│ │ │ ├── ic_bottom_books_s.xml
│ │ │ ├── ic_bottom_explore.xml
│ │ │ ├── ic_bottom_explore_e.xml
│ │ │ ├── ic_bottom_explore_s.xml
│ │ │ ├── ic_bottom_person.xml
│ │ │ ├── ic_bottom_person_e.xml
│ │ │ ├── ic_bottom_person_s.xml
│ │ │ ├── ic_bottom_rss_feed.xml
│ │ │ ├── ic_bottom_rss_feed_e.xml
│ │ │ ├── ic_bottom_rss_feed_s.xml
│ │ │ ├── ic_brightness.xml
│ │ │ ├── ic_brightness_auto.xml
│ │ │ ├── ic_bubble_chart.xml
│ │ │ ├── ic_bug_report.xml
│ │ │ ├── ic_cfg_about.xml
│ │ │ ├── ic_cfg_backup.xml
│ │ │ ├── ic_cfg_donate.xml
│ │ │ ├── ic_cfg_other.xml
│ │ │ ├── ic_cfg_replace.xml
│ │ │ ├── ic_cfg_source.xml
│ │ │ ├── ic_cfg_theme.xml
│ │ │ ├── ic_cfg_web.xml
│ │ │ ├── ic_chapter_list.xml
│ │ │ ├── ic_check.xml
│ │ │ ├── ic_check_source.xml
│ │ │ ├── ic_clear_all.xml
│ │ │ ├── ic_copy.xml
│ │ │ ├── ic_create_folder_outline.xml
│ │ │ ├── ic_cursor_left.xml
│ │ │ ├── ic_cursor_right.xml
│ │ │ ├── ic_daytime.xml
│ │ │ ├── ic_divider.xml
│ │ │ ├── ic_download.xml
│ │ │ ├── ic_download_line.xml
│ │ │ ├── ic_edit.xml
│ │ │ ├── ic_exchange.xml
│ │ │ ├── ic_exchange_order.xml
│ │ │ ├── ic_exit.xml
│ │ │ ├── ic_expand_less.xml
│ │ │ ├── ic_expand_more.xml
│ │ │ ├── ic_export.xml
│ │ │ ├── ic_fast_forward.xml
│ │ │ ├── ic_fast_rewind.xml
│ │ │ ├── ic_find_replace.xml
│ │ │ ├── ic_folder.xml
│ │ │ ├── ic_folder_open.xml
│ │ │ ├── ic_folder_outline.xml
│ │ │ ├── ic_groups.xml
│ │ │ ├── ic_help.xml
│ │ │ ├── ic_history.xml
│ │ │ ├── ic_image.xml
│ │ │ ├── ic_import.xml
│ │ │ ├── ic_interface_setting.xml
│ │ │ ├── ic_launcher1.xml
│ │ │ ├── ic_launcher1_b.xml
│ │ │ ├── ic_launcher2.xml
│ │ │ ├── ic_launcher3.xml
│ │ │ ├── ic_launcher4.xml
│ │ │ ├── ic_launcher4_b.xml
│ │ │ ├── ic_launcher5.xml
│ │ │ ├── ic_launcher5_b.xml
│ │ │ ├── ic_launcher6.xml
│ │ │ ├── ic_launcher7.xml
│ │ │ ├── ic_launcher7_b.xml
│ │ │ ├── ic_lock_outline.xml
│ │ │ ├── ic_menu.xml
│ │ │ ├── ic_more.xml
│ │ │ ├── ic_more_vert.xml
│ │ │ ├── ic_network_check.xml
│ │ │ ├── ic_outline_cloud_24.xml
│ │ │ ├── ic_outline_delete.xml
│ │ │ ├── ic_pause_24dp.xml
│ │ │ ├── ic_pause_outline_24dp.xml
│ │ │ ├── ic_play_24dp.xml
│ │ │ ├── ic_play_mode_list_end_stop.xml
│ │ │ ├── ic_play_mode_list_loop.xml
│ │ │ ├── ic_play_mode_random.xml
│ │ │ ├── ic_play_mode_single_loop.xml
│ │ │ ├── ic_play_outline_24dp.xml
│ │ │ ├── ic_praise.xml
│ │ │ ├── ic_read_aloud.xml
│ │ │ ├── ic_reduce.xml
│ │ │ ├── ic_refresh_black_24dp.xml
│ │ │ ├── ic_refresh_white_24dp.xml
│ │ │ ├── ic_restore.xml
│ │ │ ├── ic_save.xml
│ │ │ ├── ic_scan.xml
│ │ │ ├── ic_scoring.xml
│ │ │ ├── ic_screen.xml
│ │ │ ├── ic_search.xml
│ │ │ ├── ic_search_hint.xml
│ │ │ ├── ic_settings.xml
│ │ │ ├── ic_share.xml
│ │ │ ├── ic_skip_next.xml
│ │ │ ├── ic_skip_previous.xml
│ │ │ ├── ic_sort.xml
│ │ │ ├── ic_star.xml
│ │ │ ├── ic_star_border.xml
│ │ │ ├── ic_stop_black_24dp.xml
│ │ │ ├── ic_storage_black_24dp.xml
│ │ │ ├── ic_swap_horiz.xml
│ │ │ ├── ic_time_add_24dp.xml
│ │ │ ├── ic_timer_black_24dp.xml
│ │ │ ├── ic_toc.xml
│ │ │ ├── ic_translate.xml
│ │ │ ├── ic_update.xml
│ │ │ ├── ic_view_quilt.xml
│ │ │ ├── ic_visibility_off.xml
│ │ │ ├── ic_volume_up.xml
│ │ │ ├── ic_web_outline.xml
│ │ │ ├── ic_web_service_noti.xml
│ │ │ ├── recyclerview_divider_horizontal.xml
│ │ │ ├── recyclerview_divider_vertical.xml
│ │ │ ├── selector_btn_accent_bg.xml
│ │ │ ├── selector_circle_btn_bg.xml
│ │ │ ├── selector_common_bg.xml
│ │ │ ├── selector_fillet_btn_bg.xml
│ │ │ ├── selector_tv_black.xml
│ │ │ ├── shape_card_view.xml
│ │ │ ├── shape_circle.xml
│ │ │ ├── shape_fillet_btn.xml
│ │ │ ├── shape_fillet_btn_press.xml
│ │ │ ├── shape_pop_checkaddshelf_bg.xml
│ │ │ ├── shape_radius_10dp.xml
│ │ │ ├── shape_radius_1dp.xml
│ │ │ ├── shape_space_divider.xml
│ │ │ ├── shape_text_cursor.xml
│ │ │ └── shape_translucent_card.xml
│ │ ├── layout/
│ │ │ ├── activity_about.xml
│ │ │ ├── activity_all_bookmark.xml
│ │ │ ├── activity_arrange_book.xml
│ │ │ ├── activity_audio_play.xml
│ │ │ ├── activity_book_info.xml
│ │ │ ├── activity_book_info_edit.xml
│ │ │ ├── activity_book_read.xml
│ │ │ ├── activity_book_search.xml
│ │ │ ├── activity_book_source.xml
│ │ │ ├── activity_book_source_edit.xml
│ │ │ ├── activity_cache_book.xml
│ │ │ ├── activity_chapter_list.xml
│ │ │ ├── activity_config.xml
│ │ │ ├── activity_dict_rule.xml
│ │ │ ├── activity_donate.xml
│ │ │ ├── activity_explore_show.xml
│ │ │ ├── activity_file_manage.xml
│ │ │ ├── activity_import_book.xml
│ │ │ ├── activity_main.xml
│ │ │ ├── activity_manga.xml
│ │ │ ├── activity_qrcode_capture.xml
│ │ │ ├── activity_read_record.xml
│ │ │ ├── activity_replace_edit.xml
│ │ │ ├── activity_replace_rule.xml
│ │ │ ├── activity_rss_artivles.xml
│ │ │ ├── activity_rss_favorites.xml
│ │ │ ├── activity_rss_read.xml
│ │ │ ├── activity_rss_source.xml
│ │ │ ├── activity_rss_source_edit.xml
│ │ │ ├── activity_rule_sub.xml
│ │ │ ├── activity_search_content.xml
│ │ │ ├── activity_source_debug.xml
│ │ │ ├── activity_source_login.xml
│ │ │ ├── activity_translucence.xml
│ │ │ ├── activity_txt_toc_rule.xml
│ │ │ ├── activity_web_view.xml
│ │ │ ├── activity_welcome.xml
│ │ │ ├── dialog_add_to_bookshelf.xml
│ │ │ ├── dialog_auto_read.xml
│ │ │ ├── dialog_book_change_source.xml
│ │ │ ├── dialog_book_group_edit.xml
│ │ │ ├── dialog_book_group_picker.xml
│ │ │ ├── dialog_bookmark.xml
│ │ │ ├── dialog_bookshelf_config.xml
│ │ │ ├── dialog_change_cover.xml
│ │ │ ├── dialog_chapter_change_source.xml
│ │ │ ├── dialog_check_source_config.xml
│ │ │ ├── dialog_click_action_config.xml
│ │ │ ├── dialog_code_view.xml
│ │ │ ├── dialog_content_edit.xml
│ │ │ ├── dialog_cover_rule_config.xml
│ │ │ ├── dialog_custom_group.xml
│ │ │ ├── dialog_dict.xml
│ │ │ ├── dialog_dict_rule_edit.xml
│ │ │ ├── dialog_direct_link_upload_config.xml
│ │ │ ├── dialog_download_choice.xml
│ │ │ ├── dialog_edit_text.xml
│ │ │ ├── dialog_file_chooser.xml
│ │ │ ├── dialog_font_select.xml
│ │ │ ├── dialog_http_tts_edit.xml
│ │ │ ├── dialog_image_blurring.xml
│ │ │ ├── dialog_login.xml
│ │ │ ├── dialog_manga_color_filter.xml
│ │ │ ├── dialog_manga_epaper.xml
│ │ │ ├── dialog_manga_footer_setting.xml
│ │ │ ├── dialog_multiple_edit_text.xml
│ │ │ ├── dialog_number_picker.xml
│ │ │ ├── dialog_open_url_confirm.xml
│ │ │ ├── dialog_page_key.xml
│ │ │ ├── dialog_photo_view.xml
│ │ │ ├── dialog_progressbar_view.xml
│ │ │ ├── dialog_read_aloud.xml
│ │ │ ├── dialog_read_bg_text.xml
│ │ │ ├── dialog_read_book_style.xml
│ │ │ ├── dialog_read_padding.xml
│ │ │ ├── dialog_recycler_view.xml
│ │ │ ├── dialog_rss_favorite_config.xml
│ │ │ ├── dialog_rule_sub_edit.xml
│ │ │ ├── dialog_search_scope.xml
│ │ │ ├── dialog_select_section_export.xml
│ │ │ ├── dialog_simulated_reading.xml
│ │ │ ├── dialog_source_picker.xml
│ │ │ ├── dialog_text_view.xml
│ │ │ ├── dialog_tip_config.xml
│ │ │ ├── dialog_toc_regex.xml
│ │ │ ├── dialog_toc_regex_edit.xml
│ │ │ ├── dialog_update.xml
│ │ │ ├── dialog_url_option_edit.xml
│ │ │ ├── dialog_variable.xml
│ │ │ ├── dialog_verification_code_view.xml
│ │ │ ├── dialog_wait.xml
│ │ │ ├── dialog_webdav_server.xml
│ │ │ ├── fragment_bookmark.xml
│ │ │ ├── fragment_books.xml
│ │ │ ├── fragment_bookshelf1.xml
│ │ │ ├── fragment_bookshelf2.xml
│ │ │ ├── fragment_chapter_list.xml
│ │ │ ├── fragment_explore.xml
│ │ │ ├── fragment_my_config.xml
│ │ │ ├── fragment_rss.xml
│ │ │ ├── fragment_rss_articles.xml
│ │ │ ├── fragment_web_view_login.xml
│ │ │ ├── item_1line_text.xml
│ │ │ ├── item_1line_text_and_del.xml
│ │ │ ├── item_app_log.xml
│ │ │ ├── item_arrange_book.xml
│ │ │ ├── item_bg_image.xml
│ │ │ ├── item_book_file_import.xml
│ │ │ ├── item_book_group_manage.xml
│ │ │ ├── item_book_manga_edge.xml
│ │ │ ├── item_book_manga_page.xml
│ │ │ ├── item_book_source.xml
│ │ │ ├── item_bookmark.xml
│ │ │ ├── item_bookshelf_grid.xml
│ │ │ ├── item_bookshelf_grid_group.xml
│ │ │ ├── item_bookshelf_list.xml
│ │ │ ├── item_bookshelf_list_group.xml
│ │ │ ├── item_change_source.xml
│ │ │ ├── item_chapter_list.xml
│ │ │ ├── item_check_box.xml
│ │ │ ├── item_cover.xml
│ │ │ ├── item_dict_rule.xml
│ │ │ ├── item_download.xml
│ │ │ ├── item_file.xml
│ │ │ ├── item_file_picker.xml
│ │ │ ├── item_fillet_text.xml
│ │ │ ├── item_find_book.xml
│ │ │ ├── item_font.xml
│ │ │ ├── item_group_manage.xml
│ │ │ ├── item_group_select.xml
│ │ │ ├── item_http_tts.xml
│ │ │ ├── item_icon_preference.xml
│ │ │ ├── item_import_book.xml
│ │ │ ├── item_log.xml
│ │ │ ├── item_path_picker.xml
│ │ │ ├── item_radio_button.xml
│ │ │ ├── item_read_record.xml
│ │ │ ├── item_read_style.xml
│ │ │ ├── item_replace_rule.xml
│ │ │ ├── item_rss.xml
│ │ │ ├── item_rss_article.xml
│ │ │ ├── item_rss_article_1.xml
│ │ │ ├── item_rss_article_2.xml
│ │ │ ├── item_rss_read_record.xml
│ │ │ ├── item_rss_source.xml
│ │ │ ├── item_rule_sub.xml
│ │ │ ├── item_search.xml
│ │ │ ├── item_search_list.xml
│ │ │ ├── item_server_select.xml
│ │ │ ├── item_source_edit.xml
│ │ │ ├── item_source_edit_check_box.xml
│ │ │ ├── item_source_import.xml
│ │ │ ├── item_text.xml
│ │ │ ├── item_theme_config.xml
│ │ │ ├── item_toc_regex.xml
│ │ │ ├── item_txt_toc_rule.xml
│ │ │ ├── popup_action.xml
│ │ │ ├── popup_action_menu.xml
│ │ │ ├── popup_keyboard_tool.xml
│ │ │ ├── popup_seek_bar.xml
│ │ │ ├── view_action_button.xml
│ │ │ ├── view_book_page.xml
│ │ │ ├── view_detail_seek_bar.xml
│ │ │ ├── view_dynamic.xml
│ │ │ ├── view_error.xml
│ │ │ ├── view_fastscroller.xml
│ │ │ ├── view_icon.xml
│ │ │ ├── view_load_more.xml
│ │ │ ├── view_loading.xml
│ │ │ ├── view_manga_menu.xml
│ │ │ ├── view_navigation_badge.xml
│ │ │ ├── view_preference.xml
│ │ │ ├── view_preference_category.xml
│ │ │ ├── view_read_menu.xml
│ │ │ ├── view_refresh_recycler.xml
│ │ │ ├── view_search.xml
│ │ │ ├── view_search_menu.xml
│ │ │ ├── view_select_action_bar.xml
│ │ │ ├── view_tab_layout.xml
│ │ │ ├── view_tab_layout_min.xml
│ │ │ ├── view_title_bar.xml
│ │ │ ├── view_title_bar_dark.xml
│ │ │ └── view_toast.xml
│ │ ├── layout-land/
│ │ │ └── activity_book_info.xml
│ │ ├── menu/
│ │ │ ├── about.xml
│ │ │ ├── app_log.xml
│ │ │ ├── app_update.xml
│ │ │ ├── audio_play.xml
│ │ │ ├── backup_restore.xml
│ │ │ ├── book_cache.xml
│ │ │ ├── book_cache_download.xml
│ │ │ ├── book_group_manage.xml
│ │ │ ├── book_info.xml
│ │ │ ├── book_info_edit.xml
│ │ │ ├── book_manga.xml
│ │ │ ├── book_read.xml
│ │ │ ├── book_read_change_source.xml
│ │ │ ├── book_read_record.xml
│ │ │ ├── book_read_refresh.xml
│ │ │ ├── book_read_source.xml
│ │ │ ├── book_remote.xml
│ │ │ ├── book_search.xml
│ │ │ ├── book_search_scope.xml
│ │ │ ├── book_source.xml
│ │ │ ├── book_source_debug.xml
│ │ │ ├── book_source_item.xml
│ │ │ ├── book_source_sel.xml
│ │ │ ├── book_toc.xml
│ │ │ ├── bookmark.xml
│ │ │ ├── bookshelf_manage.xml
│ │ │ ├── bookshelf_menage_sel.xml
│ │ │ ├── change_cover.xml
│ │ │ ├── change_source.xml
│ │ │ ├── change_source_item.xml
│ │ │ ├── code_edit.xml
│ │ │ ├── content_edit.xml
│ │ │ ├── content_search.xml
│ │ │ ├── content_select_action.xml
│ │ │ ├── crash_log.xml
│ │ │ ├── dialog_text.xml
│ │ │ ├── dict_rule.xml
│ │ │ ├── dict_rule_edit.xml
│ │ │ ├── dict_rule_sel.xml
│ │ │ ├── direct_link_upload_config.xml
│ │ │ ├── explore_item.xml
│ │ │ ├── file_chooser.xml
│ │ │ ├── file_long_click.xml
│ │ │ ├── font_select.xml
│ │ │ ├── group_manage.xml
│ │ │ ├── import_book.xml
│ │ │ ├── import_book_sel.xml
│ │ │ ├── import_replace.xml
│ │ │ ├── import_source.xml
│ │ │ ├── keyboard_assists_config.xml
│ │ │ ├── main_bnv.xml
│ │ │ ├── main_bookshelf.xml
│ │ │ ├── main_explore.xml
│ │ │ ├── main_my.xml
│ │ │ ├── main_rss.xml
│ │ │ ├── open_url_confirm.xml
│ │ │ ├── qr_code_scan.xml
│ │ │ ├── replace_edit.xml
│ │ │ ├── replace_rule.xml
│ │ │ ├── replace_rule_item.xml
│ │ │ ├── replace_rule_sel.xml
│ │ │ ├── rss_articles.xml
│ │ │ ├── rss_favorites.xml
│ │ │ ├── rss_main_item.xml
│ │ │ ├── rss_read.xml
│ │ │ ├── rss_read_record.xml
│ │ │ ├── rss_source.xml
│ │ │ ├── rss_source_debug.xml
│ │ │ ├── rss_source_item.xml
│ │ │ ├── rss_source_sel.xml
│ │ │ ├── save.xml
│ │ │ ├── search_view.xml
│ │ │ ├── server_config.xml
│ │ │ ├── servers.xml
│ │ │ ├── source_edit.xml
│ │ │ ├── source_login.xml
│ │ │ ├── source_picker.xml
│ │ │ ├── source_sub_item.xml
│ │ │ ├── source_subscription.xml
│ │ │ ├── source_webview_login.xml
│ │ │ ├── speak_engine.xml
│ │ │ ├── speak_engine_edit.xml
│ │ │ ├── theme_config.xml
│ │ │ ├── theme_list.xml
│ │ │ ├── txt_toc_rule.xml
│ │ │ ├── txt_toc_rule_edit.xml
│ │ │ ├── txt_toc_rule_item.xml
│ │ │ ├── txt_toc_rule_sel.xml
│ │ │ ├── verification_code.xml
│ │ │ └── web_view.xml
│ │ ├── mipmap-anydpi-v26/
│ │ │ ├── ic_launcher.xml
│ │ │ ├── launcher1.xml
│ │ │ ├── launcher2.xml
│ │ │ ├── launcher3.xml
│ │ │ ├── launcher4.xml
│ │ │ ├── launcher5.xml
│ │ │ └── launcher6.xml
│ │ ├── values/
│ │ │ ├── array_values.xml
│ │ │ ├── arrays.xml
│ │ │ ├── attrs.xml
│ │ │ ├── colors.xml
│ │ │ ├── colors_material_design.xml
│ │ │ ├── dimens.xml
│ │ │ ├── ids.xml
│ │ │ ├── non_translat.xml
│ │ │ ├── strings.xml
│ │ │ └── styles.xml
│ │ ├── values-es-rES/
│ │ │ ├── arrays.xml
│ │ │ └── strings.xml
│ │ ├── values-ja-rJP/
│ │ │ └── strings.xml
│ │ ├── values-night/
│ │ │ ├── colors.xml
│ │ │ └── styles.xml
│ │ ├── values-pt-rBR/
│ │ │ ├── arrays.xml
│ │ │ └── strings.xml
│ │ ├── values-vi/
│ │ │ ├── arrays.xml
│ │ │ └── strings.xml
│ │ ├── values-zh/
│ │ │ ├── arrays.xml
│ │ │ └── strings.xml
│ │ ├── values-zh-rHK/
│ │ │ ├── arrays.xml
│ │ │ └── strings.xml
│ │ ├── values-zh-rTW/
│ │ │ ├── arrays.xml
│ │ │ └── strings.xml
│ │ └── xml/
│ │ ├── about.xml
│ │ ├── file_paths.xml
│ │ ├── network_security_config.xml
│ │ ├── pref_config_aloud.xml
│ │ ├── pref_config_backup.xml
│ │ ├── pref_config_cover.xml
│ │ ├── pref_config_other.xml
│ │ ├── pref_config_read.xml
│ │ ├── pref_config_theme.xml
│ │ ├── pref_config_welcome.xml
│ │ ├── pref_main.xml
│ │ └── spen_remote_actions.xml
│ └── test/
│ └── java/
│ └── io/
│ └── legado/
│ └── app/
│ ├── ExampleUnitTest.kt
│ └── JsTest.kt
├── avd.bat
├── avd.sh
├── build.gradle
├── gradle/
│ ├── libs.versions.toml
│ └── wrapper/
│ ├── gradle-wrapper.jar
│ └── gradle-wrapper.properties
├── gradle.properties
├── gradlew
├── gradlew.bat
├── modules/
│ ├── book/
│ │ ├── .gitignore
│ │ ├── build.gradle
│ │ ├── consumer-rules.pro
│ │ └── src/
│ │ └── main/
│ │ ├── AndroidManifest.xml
│ │ ├── java/
│ │ │ └── me/
│ │ │ └── ag2s/
│ │ │ ├── base/
│ │ │ │ ├── PfdHelper.java
│ │ │ │ └── ThrowableUtils.java
│ │ │ ├── epublib/
│ │ │ │ ├── Constants.java
│ │ │ │ ├── browsersupport/
│ │ │ │ │ ├── NavigationEvent.java
│ │ │ │ │ ├── NavigationEventListener.java
│ │ │ │ │ ├── NavigationHistory.java
│ │ │ │ │ ├── Navigator.java
│ │ │ │ │ └── package-info.java
│ │ │ │ ├── domain/
│ │ │ │ │ ├── Author.java
│ │ │ │ │ ├── Date.java
│ │ │ │ │ ├── EpubBook.java
│ │ │ │ │ ├── EpubResourceProvider.java
│ │ │ │ │ ├── FileResourceProvider.java
│ │ │ │ │ ├── Guide.java
│ │ │ │ │ ├── GuideReference.java
│ │ │ │ │ ├── Identifier.java
│ │ │ │ │ ├── LazyResource.java
│ │ │ │ │ ├── LazyResourceProvider.java
│ │ │ │ │ ├── ManifestItemProperties.java
│ │ │ │ │ ├── ManifestItemRefProperties.java
│ │ │ │ │ ├── ManifestProperties.java
│ │ │ │ │ ├── MediaType.java
│ │ │ │ │ ├── MediaTypes.java
│ │ │ │ │ ├── Metadata.java
│ │ │ │ │ ├── Relator.java
│ │ │ │ │ ├── Resource.java
│ │ │ │ │ ├── ResourceInputStream.java
│ │ │ │ │ ├── ResourceReference.java
│ │ │ │ │ ├── Resources.java
│ │ │ │ │ ├── Spine.java
│ │ │ │ │ ├── SpineReference.java
│ │ │ │ │ ├── TOCReference.java
│ │ │ │ │ ├── TableOfContents.java
│ │ │ │ │ └── TitledResourceReference.java
│ │ │ │ ├── epub/
│ │ │ │ │ ├── BookProcessor.java
│ │ │ │ │ ├── BookProcessorPipeline.java
│ │ │ │ │ ├── DOMUtil.java
│ │ │ │ │ ├── EpubProcessorSupport.java
│ │ │ │ │ ├── EpubReader.java
│ │ │ │ │ ├── EpubWriter.java
│ │ │ │ │ ├── EpubWriterProcessor.java
│ │ │ │ │ ├── HtmlProcessor.java
│ │ │ │ │ ├── NCXDocumentV2.java
│ │ │ │ │ ├── NCXDocumentV3.java
│ │ │ │ │ ├── PackageDocumentBase.java
│ │ │ │ │ ├── PackageDocumentMetadataReader.java
│ │ │ │ │ ├── PackageDocumentMetadataWriter.java
│ │ │ │ │ ├── PackageDocumentReader.java
│ │ │ │ │ ├── PackageDocumentWriter.java
│ │ │ │ │ └── ResourcesLoader.java
│ │ │ │ └── util/
│ │ │ │ ├── CollectionUtil.java
│ │ │ │ ├── IOUtil.java
│ │ │ │ ├── NoCloseOutputStream.java
│ │ │ │ ├── NoCloseWriter.java
│ │ │ │ ├── ResourceUtil.java
│ │ │ │ ├── StringUtil.java
│ │ │ │ ├── URLEncodeUtil.java
│ │ │ │ ├── commons/
│ │ │ │ │ └── io/
│ │ │ │ │ ├── BOMInputStream.java
│ │ │ │ │ ├── ByteOrderMark.java
│ │ │ │ │ ├── IOConsumer.java
│ │ │ │ │ ├── ProxyInputStream.java
│ │ │ │ │ ├── XmlStreamReader.java
│ │ │ │ │ └── XmlStreamReaderException.java
│ │ │ │ └── zip/
│ │ │ │ ├── AndroidZipEntry.java
│ │ │ │ ├── AndroidZipFile.java
│ │ │ │ ├── ZipConstants.java
│ │ │ │ ├── ZipEntryWrapper.java
│ │ │ │ ├── ZipException.java
│ │ │ │ └── ZipFileWrapper.java
│ │ │ └── umdlib/
│ │ │ ├── domain/
│ │ │ │ ├── UmdBook.java
│ │ │ │ ├── UmdChapters.java
│ │ │ │ ├── UmdCover.java
│ │ │ │ ├── UmdEnd.java
│ │ │ │ └── UmdHeader.java
│ │ │ ├── tool/
│ │ │ │ ├── StreamReader.java
│ │ │ │ ├── UmdUtils.java
│ │ │ │ └── WrapOutputStream.java
│ │ │ └── umd/
│ │ │ └── UmdReader.java
│ │ └── resources/
│ │ ├── dtd/
│ │ │ ├── openebook.org/
│ │ │ │ └── dtds/
│ │ │ │ └── oeb-1.2/
│ │ │ │ ├── oeb12.ent
│ │ │ │ └── oebpkg12.dtd
│ │ │ ├── www.daisy.org/
│ │ │ │ └── z3986/
│ │ │ │ └── 2005/
│ │ │ │ └── ncx-2005-1.dtd
│ │ │ └── www.w3.org/
│ │ │ └── TR/
│ │ │ ├── ruby/
│ │ │ │ └── xhtml-ruby-1.mod
│ │ │ ├── xhtml-modularization/
│ │ │ │ └── DTD/
│ │ │ │ ├── xhtml-arch-1.mod
│ │ │ │ ├── xhtml-attribs-1.mod
│ │ │ │ ├── xhtml-base-1.mod
│ │ │ │ ├── xhtml-bdo-1.mod
│ │ │ │ ├── xhtml-blkphras-1.mod
│ │ │ │ ├── xhtml-blkpres-1.mod
│ │ │ │ ├── xhtml-blkstruct-1.mod
│ │ │ │ ├── xhtml-charent-1.mod
│ │ │ │ ├── xhtml-csismap-1.mod
│ │ │ │ ├── xhtml-datatypes-1.mod
│ │ │ │ ├── xhtml-datatypes-1.mod.1
│ │ │ │ ├── xhtml-edit-1.mod
│ │ │ │ ├── xhtml-events-1.mod
│ │ │ │ ├── xhtml-form-1.mod
│ │ │ │ ├── xhtml-framework-1.mod
│ │ │ │ ├── xhtml-hypertext-1.mod
│ │ │ │ ├── xhtml-image-1.mod
│ │ │ │ ├── xhtml-inlphras-1.mod
│ │ │ │ ├── xhtml-inlpres-1.mod
│ │ │ │ ├── xhtml-inlstruct-1.mod
│ │ │ │ ├── xhtml-inlstyle-1.mod
│ │ │ │ ├── xhtml-lat1.ent
│ │ │ │ ├── xhtml-link-1.mod
│ │ │ │ ├── xhtml-list-1.mod
│ │ │ │ ├── xhtml-meta-1.mod
│ │ │ │ ├── xhtml-notations-1.mod
│ │ │ │ ├── xhtml-object-1.mod
│ │ │ │ ├── xhtml-param-1.mod
│ │ │ │ ├── xhtml-pres-1.mod
│ │ │ │ ├── xhtml-qname-1.mod
│ │ │ │ ├── xhtml-script-1.mod
│ │ │ │ ├── xhtml-special.ent
│ │ │ │ ├── xhtml-ssismap-1.mod
│ │ │ │ ├── xhtml-struct-1.mod
│ │ │ │ ├── xhtml-style-1.mod
│ │ │ │ ├── xhtml-symbol.ent
│ │ │ │ ├── xhtml-symbol.ent.1
│ │ │ │ ├── xhtml-table-1.mod
│ │ │ │ ├── xhtml-text-1.mod
│ │ │ │ └── xhtml11-model-1.mod
│ │ │ ├── xhtml1/
│ │ │ │ └── DTD/
│ │ │ │ ├── xhtml-lat1.ent
│ │ │ │ ├── xhtml-special.ent
│ │ │ │ ├── xhtml-symbol.ent
│ │ │ │ ├── xhtml1-strict.dtd
│ │ │ │ └── xhtml1-transitional.dtd
│ │ │ └── xhtml11/
│ │ │ └── DTD/
│ │ │ └── xhtml11.dtd
│ │ └── log4j.properties
│ ├── rhino/
│ │ ├── build.gradle
│ │ ├── consumer-rules.pro
│ │ ├── lib/
│ │ │ └── rhino-1.7.14.jar
│ │ └── src/
│ │ └── main/
│ │ ├── AndroidManifest.xml
│ │ └── java/
│ │ └── com/
│ │ └── script/
│ │ ├── AbstractScriptEngine.kt
│ │ ├── Bindings.kt
│ │ ├── Compilable.kt
│ │ ├── CompiledScript.kt
│ │ ├── Invocable.kt
│ │ ├── RhinoContextFactory.kt
│ │ ├── ScriptBindings.kt
│ │ ├── ScriptBindingsExtensions.kt
│ │ ├── ScriptContext.kt
│ │ ├── ScriptEngine.kt
│ │ ├── ScriptException.kt
│ │ ├── SimpleBindings.kt
│ │ ├── SimpleScriptContext.kt
│ │ └── rhino/
│ │ ├── ClassNameMatcher.kt
│ │ ├── CollectionExtensions.kt
│ │ ├── ExternalScriptable.kt
│ │ ├── InterfaceImplementor.kt
│ │ ├── JSAdapter.kt
│ │ ├── JavaAdapter.kt
│ │ ├── JavaObjectWrapFactory.kt
│ │ ├── ProtectedNativeJavaClass.kt
│ │ ├── ReadOnlyJavaObject.kt
│ │ ├── RhinoClassShutter.kt
│ │ ├── RhinoCompiledScript.kt
│ │ ├── RhinoContext.kt
│ │ ├── RhinoErrors.kt
│ │ ├── RhinoExtensions.kt
│ │ ├── RhinoScriptEngine.kt
│ │ ├── RhinoTopLevel.kt
│ │ ├── RhinoWrapFactory.kt
│ │ └── VMBridgeReflect.kt
│ └── web/
│ ├── .browserslistrc
│ ├── .editorconfig
│ ├── .gitignore
│ ├── .prettierignore
│ ├── .prettierrc.json
│ ├── LICENSE
│ ├── README.md
│ ├── env.d.ts
│ ├── eslint.config.mjs
│ ├── index.html
│ ├── package.json
│ ├── scripts/
│ │ └── sync.js
│ ├── src/
│ │ ├── App.vue
│ │ ├── api/
│ │ │ ├── api.ts
│ │ │ ├── axios.ts
│ │ │ └── index.ts
│ │ ├── assets/
│ │ │ ├── bookshelf.css
│ │ │ ├── code.css
│ │ │ ├── fonts/
│ │ │ │ ├── iconfont.css
│ │ │ │ ├── popfont.css
│ │ │ │ └── shelffont.css
│ │ │ ├── kbd.css
│ │ │ └── sourceeditor.css
│ │ ├── auto-imports.d.ts
│ │ ├── book.d.ts
│ │ ├── components/
│ │ │ ├── BookItems.vue
│ │ │ ├── CatalogItem.vue
│ │ │ ├── ChapterContent.vue
│ │ │ ├── PopCatalog.vue
│ │ │ ├── ReadSettings.vue
│ │ │ ├── SourceDebug.vue
│ │ │ ├── SourceHelp.vue
│ │ │ ├── SourceItem.vue
│ │ │ ├── SourceJson.vue
│ │ │ ├── SourceList.vue
│ │ │ ├── SourceTabForm.vue
│ │ │ ├── SourceTabTools.vue
│ │ │ └── ToolBar.vue
│ │ ├── components.d.ts
│ │ ├── config/
│ │ │ ├── bookSourceEditConfig.ts
│ │ │ ├── rssSourceEditConfig.ts
│ │ │ ├── sourceConfig.d.ts
│ │ │ └── themeConfig.ts
│ │ ├── hooks/
│ │ │ ├── loading.css
│ │ │ └── loading.ts
│ │ ├── main.ts
│ │ ├── pages/
│ │ │ ├── bookshelf/
│ │ │ │ ├── README.md
│ │ │ │ ├── index.html
│ │ │ │ └── main.js
│ │ │ └── source/
│ │ │ ├── README.md
│ │ │ ├── index.html
│ │ │ └── main.js
│ │ ├── plugins/
│ │ │ ├── jump.d.ts
│ │ │ └── jump.js
│ │ ├── router/
│ │ │ ├── bookRouter.ts
│ │ │ ├── index.ts
│ │ │ └── sourceRouter.ts
│ │ ├── source.d.ts
│ │ ├── store/
│ │ │ ├── bookStore.ts
│ │ │ ├── connectionStore.ts
│ │ │ ├── index.ts
│ │ │ └── sourceStore.ts
│ │ ├── utils/
│ │ │ ├── souce.ts
│ │ │ └── utils.ts
│ │ ├── views/
│ │ │ ├── BookChapter.vue
│ │ │ ├── BookShelf.vue
│ │ │ └── SourceEditor.vue
│ │ └── web.d.ts
│ ├── tsconfig.app.json
│ ├── tsconfig.json
│ ├── tsconfig.node.json
│ └── vite.config.ts
├── package.json
└── settings.gradle
================================================
FILE CONTENTS
================================================
================================================
FILE: .github/ISSUE_TEMPLATE/01-bugReport.yml
================================================
name: BUG提交 / BUG Report
description: Report bugs to developers
labels: ["BUG"]
body:
- type: checkboxes
attributes:
label: 确认 / Assignments
description: 提交issue请确保完成以下前提,否则该issue可能被忽略 / Make sure you read checkboxs below
options:
- label: 搜索现有issues,不存在相似或相关的issue / No similar or related issues
required: true
- label: 最新[测试版](https://github.com/gedoor/legado/releases/beta)依然存在此问题 / Latest beta app does not work
required: true
- label: 此问题和Xposed、Lsposed、Magisk、手机主题、浏览器插件、无障碍服务等无关 / Make sure your machine is not touched by hook frameworks, plugins, accessibility etc
required: true
- type: textarea
attributes:
label: 问题描述 / Describe Bugs
validations:
required: true
- type: textarea
attributes:
label: 复现步骤 / How to reproduce
validations:
required: true
- type: checkboxes
attributes:
label: 确认 / Assignment
options:
- label: 已经提交复现所需要的附加资料 / Submit additions related with bugs
required: true
- type: textarea
attributes:
label: 其他信息 / Additions
description: |
反馈WEB书架前端问题时请提供浏览器版本信息,如Edge 129.0.2792.89
placeholder: "请用```将提交的内容包裹"
- type: textarea
attributes:
label: 日志提交 / Relevant log output
description: |
阅读日志位于我的-关于-崩溃日志、保存日志、书架-右上角-日志,或者自行使用log工具抓取日志
如果崩溃日志中包含`java.lang.OutOfMemoryError`,请安装最新测试版,在其他设置里打开记录堆转储,复现崩溃后去关于那里点保存日志,然后去备份目录里将heapDump文件夹里的文件打包压缩一下上传上来
placeholder: "请用```将日志内容包裹"
- type: input
attributes:
label: 阅读版本 / Legado version
placeholder: "3.22.110823"
validations:
required: true
- type: input
attributes:
label: Android版本 / Android version
placeholder: "Android 12"
validations:
required: true
- type: input
attributes:
label: 机型 / Model
placeholder: "Redmi K30 Pro"
validations:
required: true
================================================
FILE: .github/ISSUE_TEMPLATE/02-featureRequest.yml
================================================
name: 功能请求 / Features
description: Request new features
labels: ["需求"]
body:
- type: checkboxes
attributes:
label: 确认 / Assignments
description: 提交issue请确保完成以下前提,否则该issue可能被忽略 / Make sure you read checkbox below
options:
- label: 搜索现有issues,不存在相似或相关的issue / No related requests
required: true
- label: 最新[测试版](https://github.com/gedoor/legado/releases/beta)没有此功能 / Latest beta app does not this feature
required: true
- type: textarea
attributes:
label: 功能描述 / Features
placeholder: 请清晰的、详细的描述你想要的功能
validations:
required: true
- type: textarea
attributes:
label: 期望实现方式 / How to implement
placeholder: 阅读应该如何实现该功能
validations:
required: true
- type: input
attributes:
label: 阅读版本 / Legado version
placeholder: "3.22.110823"
validations:
required: true
- type: textarea
attributes:
label: 附加信息 / Additions
placeholder: 其他的与功能相关的附加信息
- type: textarea
attributes:
label: 效果演示 / Demo
placeholder: 可以手绘一些草图,或者提供可借鉴的图片
================================================
FILE: .github/ISSUE_TEMPLATE/config.yml
================================================
blank_issues_enabled: false
contact_links:
- name: 简繁转化
url: https://github.com/liuyueyi/quick-chinese-transfer/issues/new
about: 简繁转化问题请优先到quick-chinese-transfer反馈
- name: 讨论 / Discussions
url: https://github.com/gedoor/legado/discussions
about: Please ask and answer questions here.
- name: 常见问题 / Wiki
url: https://github.com/gedoor/legado/wiki
about: Read wiki if your are new here.
================================================
FILE: .github/dependabot.yml
================================================
# To get started with Dependabot version updates, you'll need to specify which
# package ecosystems to update and where the package manifests are located.
# Please see the documentation for all configuration options:
# https://docs.github.com/github/administering-a-repository/configuration-options-for-dependency-updates
version: 2
registries:
maven-google:
type: maven-repository
# url: https://maven.google.com
url: https://dl.google.com/dl/android/maven2/
password: dummy
username: dummy
maven-central:
type: maven-repository
url: https://repo1.maven.org/maven2/
password: dummy
username: dummy
maven-jitpack:
type: maven-repository
url: https://jitpack.io
password: dummy
username: dummy
updates:
- package-ecosystem: gradle
directory: "/"
schedule:
interval: "weekly"
registries: "*"
open-pull-requests-limit: 20
groups:
kotlin_KSP:
patterns:
- "org.jetbrains.kotlin:*"
- "com.google.devtools.ksp"
# Maintain dependencies for GitHub Actions
- package-ecosystem: "github-actions"
directory: "/"
schedule:
interval: "weekly"
- package-ecosystem: npm
directory: "modules/web"
schedule:
interval: "weekly"
================================================
FILE: .github/scripts/cronet.sh
================================================
#!/usr/bin/env bash
#分支Stable Dev Beta
branch=$1
#api 最大偏移
max_offset=$2
[ -z $1 ] && branch=Stable
[ -z $2 ] && max_offset=3
[ -z $GITHUB_ENV ] && echo "Error: Unexpected github workflow environment" && exit
offset=0
function fetch_version() {
# 获取最新cronet版本
lastest_cronet_version=`curl -s "https://chromiumdash.appspot.com/fetch_releases?channel=$branch&platform=Android&num=1&offset=$offset" | jq .[0].version -r`
echo "lastest_cronet_version: $lastest_cronet_version"
#lastest_cronet_version=100.0.4845.0
lastest_cronet_main_version=${lastest_cronet_version%%\.*}.0.0.0
check_version_exit
}
function check_version_exit() {
# 检查版本是否存在
local jar_url="https://storage.googleapis.com/chromium-cronet/android/$lastest_cronet_version/Release/cronet/cronet_api.jar"
statusCode=$(curl -s -I -w %{http_code} "$jar_url" -o /dev/null)
if [ $statusCode == "404" ]; then
echo "storage.googleapis.com return 404 for cronet $lastest_cronet_version"
if [[ $max_offset > $offset ]]; then
offset=$(expr $offset + 1)
echo "retry with offset $offset"
fetch_version
else
exit
fi
fi
}
function version_compare() {
# 版本号比较 本地版本小于远程版本时返回0
local local_version=$1
local remote_version=$2
if [[ $local_version == $remote_version ]]; then
return 1
fi
if [[ $(printf '%s\n' "$1" "$2" | sort -V | head -n1) == $remote_version ]]; then
return 1
else
return 0
fi
}
# 添加变量到github env
function write_github_env_variable() {
echo "$1=$2" >> $GITHUB_ENV
}
function sync_proguard_rules() {
local raw_github_git="https://raw.githubusercontent.com/chromium/chromium/$lastest_cronet_version"
local proguard_paths=(
components/cronet/android/cronet_combined_impl_native_proguard_golden.cfg
)
local proguard_rules_path="$GITHUB_WORKSPACE/app/cronet-proguard-rules.pro"
rm -f $proguard_rules_path
echo "fetch cronet proguard rules from upstream $raw_github_git"
for path in ${proguard_paths[@]}
do
echo "fetching $path ..."
curl "$raw_github_git/$path" >> $proguard_rules_path
done
}
##########
# 获取本地cronet版本
path=$GITHUB_WORKSPACE/gradle.properties
current_cronet_version=`cat $path | grep CronetVersion | sed s/CronetVersion=//`
echo "current_cronet_version: $current_cronet_version"
echo "fetch $branch release info from https://chromiumdash.appspot.com ..."
fetch_version
if version_compare $current_cronet_version $lastest_cronet_version; then
# 更新gradle.properties
sed -i s/CronetVersion=.*/CronetVersion=$lastest_cronet_version/ $path
sed -i s/CronetMainVersion=.*/CronetMainVersion=$lastest_cronet_main_version/ $path
# 更新cronet_proguard_rules.pro
sync_proguard_rules
# 更新cronet版本
sed -i "s/## cronet版本: .*/## cronet版本: $lastest_cronet_version/" $GITHUB_WORKSPACE/app/src/main/assets/updateLog.md
# 生成pull request信息
write_github_env_variable PR_TITLE "Bump cronet from $current_cronet_version to $lastest_cronet_version"
write_github_env_variable PR_BODY "Changes in the [Git log](https://chromium.googlesource.com/chromium/src/+log/$current_cronet_version..$lastest_cronet_version)"
# 生成cronet flag
write_github_env_variable cronet ok
fi
================================================
FILE: .github/scripts/lzy_web.py
================================================
import requests, os, datetime, sys
# Cookie 中 phpdisk_info 的值
cookie_phpdisk_info = os.environ.get('phpdisk_info')
# Cookie 中 ylogin 的值
cookie_ylogin = os.environ.get('ylogin')
# 请求头
headers = {
'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/89.0.4389.72 Safari/537.36 Edg/89.0.774.45',
'Accept-Language': 'zh-CN,zh;q=0.9',
'Referer': 'https://pc.woozooo.com/account.php?action=login'
}
# 小饼干
cookie = {
'ylogin': cookie_ylogin,
'phpdisk_info': cookie_phpdisk_info
}
# 日志打印
def log(msg):
utc_time = datetime.datetime.utcnow()
china_time = utc_time + datetime.timedelta(hours=8)
print(f"[{china_time.strftime('%Y.%m.%d %H:%M:%S')}] {msg}")
# 检查是否已登录
def login_by_cookie():
url_account = "https://pc.woozooo.com/account.php"
if cookie['phpdisk_info'] is None:
log('ERROR: 请指定 Cookie 中 phpdisk_info 的值!')
return False
if cookie['ylogin'] is None:
log('ERROR: 请指定 Cookie 中 ylogin 的值!')
return False
res = requests.get(url_account, headers=headers, cookies=cookie, verify=True)
if '网盘用户登录' in res.text:
log('ERROR: 登录失败,请更新Cookie')
return False
else:
log('登录成功')
return True
# 上传文件
def upload_file(file_dir, folder_id):
file_name = os.path.basename(file_dir)
url_upload = "https://up.woozooo.com/fileup.php"
headers['Referer'] = f'https://up.woozooo.com/mydisk.php?item=files&action=index&u={cookie_ylogin}'
post_data = {
"task": "1",
"folder_id": folder_id,
"id": "WU_FILE_0",
"name": file_name,
}
files = {'upload_file': (file_name, open(file_dir, "rb"), 'application/octet-stream')}
res = requests.post(url_upload, data=post_data, files=files, headers=headers, cookies=cookie, timeout=120).json()
log(f"{file_dir} -> {res['info']}")
return res['zt'] == 1
# 上传文件夹内的文件
def upload_folder(folder_dir, folder_id):
file_list = sorted(os.listdir(folder_dir), reverse=True)
for file in file_list:
path = os.path.join(folder_dir, file)
if os.path.isfile(path):
upload_file(path, folder_id)
else:
upload_folder(path, folder_id)
# 上传
def upload(dir, folder_id):
if dir is None:
log('ERROR: 请指定上传的文件路径')
return
if folder_id is None:
log('ERROR: 请指定蓝奏云的文件夹id')
return
if os.path.isfile(dir):
upload_file(dir, str(folder_id))
else:
upload_folder(dir, str(folder_id))
if __name__ == '__main__':
argv = sys.argv[1:]
if len(argv) != 2:
log('ERROR: 参数错误,请以这种格式重新尝试\npython lzy_web.py 需上传的路径 蓝奏云文件夹id')
# 需上传的路径
upload_path = argv[0]
# 蓝奏云文件夹id
lzy_folder_id = argv[1]
if login_by_cookie():
upload(upload_path, lzy_folder_id)
================================================
FILE: .github/scripts/tg_bot.py
================================================
import os, sys, telebot
# 上传文件
def upload_file(tb, chat_id, file_dir):
doc = open(file_dir, 'rb')
tb.send_document(chat_id, doc)
# 上传文件夹内的文件
def upload_folder(tb, chat_id, folder_dir):
file_list = sorted(os.listdir(folder_dir))
for file in file_list:
path = os.path.join(folder_dir, file)
if os.path.isfile(path):
upload_file(tb, chat_id, path)
else:
upload_folder(tb, chat_id, path)
# 上传
def upload(tb, chat_id, dir):
if tb is None:
log('ERROR: 输入正确的token')
return
if chat_id is None:
log('ERROR: 输入正确的chat_id')
return
if dir is None:
log('ERROR: 请指定上传的文件路径')
return
if os.path.isfile(dir):
upload_file(tb, chat_id, dir)
else:
upload_folder(tb, chat_id, dir)
if __name__ == '__main__':
argv = sys.argv[1:]
if len(argv) != 3:
log('ERROR: 参数错误,请以这种格式重新尝试\npython tg_bot.py $token $chat_id 待上传的路径')
# Token
TOKEN = argv[0]
# chat_id
chat_id = argv[1]
# 待上传文件的路径
upload_path = argv[2]
#创建连接
tb = telebot.TeleBot(TOKEN)
#开始上传
upload(tb, chat_id, upload_path)
================================================
FILE: .github/workflows/autoupdatefork.yml
================================================
#更新fork
name: update fork
on:
schedule:
- cron: '0 16 * * *' #0点定时任务
jobs:
build:
runs-on: ubuntu-latest
if: ${{ github.event.repository.owner.id == github.event.sender.id && github.actor != 'gedoor' }}
steps:
- name: Checkout
uses: actions/checkout@v4
- name: Set env
run: |
git config --global user.email "github-actions@github.com"
git config --global user.name "github-actions"
- name: Update fork
run: |
git remote add upstream https://github.com/gedoor/legado.git
git remote -v
git fetch upstream
git checkout master
git merge upstream/master
git push origin master
================================================
FILE: .github/workflows/cronet.yml
================================================
name: Update Cronet
on:
schedule:
# 周一北京时间9点
- cron: 0 1 * * 1
workflow_dispatch:
jobs:
build:
runs-on: ubuntu-latest
if: ${{ github.repository == 'gedoor/legado' }}
steps:
- uses: actions/checkout@v4
- name: Check Cronet Updates
run: source .github/scripts/cronet.sh
- name: Set up JDK 17
if: ${{ env.cronet == 'ok' }}
uses: actions/setup-java@v4
with:
distribution: 'temurin'
java-version: 17
- uses: gradle/actions/setup-gradle@v4
if: ${{ env.cronet == 'ok' }}
- name: Download Cronet
if: ${{ env.cronet == 'ok' }}
run: |
chmod +x gradlew
./gradlew app:downloadCronet
- name: Create Pull Request
if: ${{ env.cronet == 'ok' }}
uses: peter-evans/create-pull-request@v7
continue-on-error: true
with:
token: ${{ secrets.ACTIONS_TOKEN }}
title: ${{ env.PR_TITLE }}
commit-message: |
${{ env.PR_TITLE }}
- ${{ env.PR_BODY }}
body: ${{ env.PR_BODY }}
branch: cronet
delete-branch: true
add-paths: |
*cronet*jar
*cronet.json
*updateLog.md
*gradle.properties
*cronet-proguard-rules.pro
================================================
FILE: .github/workflows/release.yml
================================================
name: Release Build
on:
# push:
# branches:
# - master
# paths:
# - 'CHANGELOG.md'
workflow_dispatch:
jobs:
prepare:
runs-on: ubuntu-latest
if: github.event_name != 'pull_request' && github.ref == 'refs/heads/master'
outputs:
version: ${{ steps.set-ver.outputs.version }}
play: ${{ steps.check.outputs.play }}
sign: ${{ steps.check.outputs.sign }}
steps:
- id: set-ver
run: |
echo "version=$(date -d "8 hour" -u +3.%y.%m%d%H)" >> $GITHUB_OUTPUT
- id: check
run: |
if [ ! -z "${{ secrets.RELEASE_KEY_STORE }}" ]; then
echo "sign=yes" >> $GITHUB_OUTPUT
fi
if [ ! -z "${{ secrets.SERVICE_ACCOUNT_JSON }}" ]; then
echo "play=yes" >> $GITHUB_OUTPUT
fi
build:
needs: prepare
if: ${{ needs.prepare.outputs.sign }}
strategy:
matrix:
product: [ app, google ]
fail-fast: false
runs-on: ubuntu-latest
env:
product: ${{ matrix.product }}
VERSION: ${{ needs.prepare.outputs.version }}
play: ${{ needs.prepare.outputs.play }}
steps:
- uses: actions/checkout@v4
- name: Set up JDK 17
uses: actions/setup-java@v4
with:
distribution: 'temurin'
java-version: 17
- name: Release Apk Sign
run: |
# not use this output
# echo "KeyStore=yes" >> $GITHUB_OUTPUT
echo -e "\n" >> gradle.properties
echo RELEASE_KEY_ALIAS='${{ secrets.RELEASE_KEY_ALIAS }}' >> gradle.properties
echo RELEASE_KEY_PASSWORD='${{ secrets.RELEASE_KEY_PASSWORD }}' >> gradle.properties
echo RELEASE_STORE_PASSWORD='${{ secrets.RELEASE_STORE_PASSWORD }}' >> gradle.properties
echo RELEASE_STORE_FILE='./key.jks' >> gradle.properties
echo ${{ secrets.RELEASE_KEY_STORE }} | base64 --decode > $GITHUB_WORKSPACE/app/key.jks
- name: Unify Version Name
run: |
echo "统一版本号"
sed "/def version/c def version = \"${{ env.VERSION }}\"" $GITHUB_WORKSPACE/app/build.gradle -i
- name: Set up Gradle
uses: gradle/actions/setup-gradle@v4
- name: Build With Gradle
run: |
echo "开始进行${{ env.product }}构建"
chmod +x gradlew
./gradlew assemble${{ env.product }}release --build-cache --parallel --daemon --warning-mode all
- name: Organize the Files
run: |
mkdir -p ${{ github.workspace }}/apk/
cp -rf ${{ github.workspace }}/app/build/outputs/apk/*/*/*.apk ${{ github.workspace }}/apk/
- name: Upload App To Artifact
uses: actions/upload-artifact@v4
with:
name: legado_${{ env.product }}
path: ${{ github.workspace }}/apk/*.apk
- name: Release
if: ${{ env.product == 'app' }}
uses: softprops/action-gh-release@v2
with:
name: legado_app_${{ env.VERSION }}
tag_name: ${{ env.VERSION }}
body_path: ${{ github.workspace }}/CHANGELOG.md
draft: false
prerelease: false
files: ${{ github.workspace }}/apk/legado_app_*.apk
env:
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
- name: Prepare For GooglePlay
if: env.product == 'google' && env.play == 'yes'
run: |
mkdir -p ReleaseNotes
ln -s ${{ github.workspace }}/CHANGELOG.md ReleaseNotes/whatsnew-en-US
ln -s ${{ github.workspace }}/CHANGELOG.md ReleaseNotes/whatsnew-zh-CN
- name: Release To GooglePlay
if: env.product == 'google' && env.play == 'yes'
uses: r0adkll/upload-google-play@v1
with:
serviceAccountJsonPlainText: ${{ secrets.SERVICE_ACCOUNT_JSON }}
packageName: io.legado.play
releaseFiles: ${{ github.workspace }}/apk/legado_google_*.apk
track: production
whatsNewDirectory: ${{ github.workspace }}/ReleaseNotes
- name: Push Assets To "release" Branch
if: ${{ github.actor == 'gedoor' }}
run: |
cd $GITHUB_WORKSPACE/apk/
git init
git checkout -b release
git config --global user.name "github-actions[bot]"
git config --global user.email "41898282+github-actions[bot]@users.noreply.github.com"
git remote add origin "https://${{ github.actor }}:${{ secrets.ACTIONS_TOKEN }}@github.com/${{ github.actor }}/release"
git add *.apk
git commit -m "${{ env.VERSION }}"
git push -f -u origin release
- name: Purge Jsdelivr Cache
if: ${{ github.actor == 'gedoor' }}
run: |
result=$(curl -s https://purge.jsdelivr.net/gh/${{ github.actor }}/release@release/)
if echo $result |grep -q 'success.*true'; then
echo "jsdelivr缓存更新成功"
else
echo $result
fi
================================================
FILE: .github/workflows/stale.yml
================================================
# This workflow warns and then closes issues and PRs that have had no activity for a specified amount of time.
#
# You can adjust the behavior by modifying this file.
# For more information, see:
# https://github.com/actions/stale
name: closeStaleIssue
on:
schedule:
# 每5天北京时间9点
- cron: '30 1 1/5 * *'
workflow_dispatch:
jobs:
stale:
runs-on: ubuntu-latest
if: ${{ github.repository == 'gedoor/legado' }}
permissions:
issues: write
steps:
- uses: actions/stale@v9
with:
repo-token: ${{ secrets.GITHUB_TOKEN }}
stale-issue-message: '由于长期没有状态更新,该问题将于5天后自动关闭。如有需要可重新打开。'
days-before-stale: 30
days-before-close: 5
operations-per-run: 100
================================================
FILE: .github/workflows/test.yml
================================================
name: Test Build
on:
push:
branches:
- master
paths:
- '**'
# - '!**/assets/**'
# - '!**.md'
- '!**/ISSUE_TEMPLATE/**'
- '!**/modules/web/**'
pull_request:
paths-ignore:
- '**/modules/web/**'
workflow_run:
workflows: [Build Web]
branches: [master]
types:
- completed
workflow_dispatch:
concurrency:
group: ${{ github.workflow }}-${{ github.ref }}
cancel-in-progress: true
jobs:
prepare:
runs-on: ubuntu-latest
if: ${{ !startsWith(github.event.head_commit.message, 'Merge pull request') }}
outputs:
version: ${{ steps.set-ver.outputs.version }}
versionL: ${{ steps.set-ver.outputs.versionL }}
lanzou: ${{ steps.check.outputs.lanzou }}
telegram: ${{ steps.check.outputs.telegram }}
steps:
- id: set-ver
run: |
echo "version=$(date -d "8 hour" -u +3.%y.%m%d%H)" >> $GITHUB_OUTPUT
echo "versionL=$(date -d "8 hour" -u +3.%y.%m%d%H%M)" >> $GITHUB_OUTPUT
- id: check
run: |
if [ ${{ secrets.LANZOU_ID }} ]; then
echo "lanzou=yes" >> $GITHUB_OUTPUT
fi
if [ ${{ secrets.BOT_TOKEN }} ]; then
echo "telegram=yes" >> $GITHUB_OUTPUT
fi
build:
needs: prepare
strategy:
matrix:
product: [ app ]
type: [ release, releaseA ]
fail-fast: false
runs-on: ubuntu-latest
env:
product: ${{ matrix.product }}
type: ${{ matrix.type }}
VERSION: ${{ needs.prepare.outputs.version }}
VERSIONL: ${{ needs.prepare.outputs.versionL }}
steps:
- uses: actions/checkout@v4
with:
fetch-depth: 0
- name: Set up JDK 17
uses: actions/setup-java@v4
with:
distribution: 'temurin'
java-version: 17
- name: Clear 18PlusList.txt
run: |
echo "清空18PlusList.txt"
echo "">$GITHUB_WORKSPACE/app/src/main/assets/18PlusList.txt
- name: Release Apk Sign
run: |
echo "给apk增加签名"
cp $GITHUB_WORKSPACE/.github/workflows/legado.jks $GITHUB_WORKSPACE/app/legado.jks
sed '$a\RELEASE_STORE_FILE=./legado.jks' $GITHUB_WORKSPACE/gradle.properties -i
sed '$a\RELEASE_KEY_ALIAS=legado' $GITHUB_WORKSPACE/gradle.properties -i
sed '$a\RELEASE_STORE_PASSWORD=gedoor_legado' $GITHUB_WORKSPACE/gradle.properties -i
sed '$a\RELEASE_KEY_PASSWORD=gedoor_legado' $GITHUB_WORKSPACE/gradle.properties -i
- name: Set up Gradle
uses: gradle/actions/setup-gradle@v4
- name: Build With Gradle
continue-on-error: true
run: |
if [ ${{ env.type }} == 'release' ]; then
typeName="原包名"
else
typeName="共存"
sed "s/'.release'/'.releaseA'/" $GITHUB_WORKSPACE/app/build.gradle -i
sed 's/.release/.releaseA/' $GITHUB_WORKSPACE/app/google-services.json -i
fi
echo "统一版本号"
sed "/def version/c def version = \"${{ env.VERSION }}\"" $GITHUB_WORKSPACE/app/build.gradle -i
echo "开始${{ env.product }}$typeName构建"
chmod +x gradlew
./gradlew assemble${{ env.product }}release --build-cache --parallel --daemon --warning-mode all
echo "修改APK文件名"
mkdir -p ${{ github.workspace }}/apk/
for file in `ls ${{ github.workspace }}/app/build/outputs/apk/*/*/*.apk`; do
mv "$file" ${{ github.workspace }}/apk/legado_${{ env.product }}_${{ env.VERSIONL }}_$typeName.apk
done
echo "移动mapping文件"
mkdir -p ${{ github.workspace }}/mapping/
for file in `ls ${{ github.workspace }}/app/build/outputs/mapping/*/mapping.txt`; do
mv "$file" ${{ github.workspace }}/mapping/mapping.txt
done
- name: Move Missing Rules Files
run: |
echo "移动missing_rules.txt文件"
mkdir -p ${{ github.workspace }}/mapping/
for file in `ls ${{ github.workspace }}/app/build/outputs/mapping/*/missing_rules.txt`; do
mv "$file" ${{ github.workspace }}/mapping/missing_rules.txt
done
- name: Upload Missing Rules File To Artifact
uses: actions/upload-artifact@v4
with:
name: legado.${{ env.product }}.${{ env.type }}.mapping.missing_rules
if-no-files-found: ignore
path: ${{ github.workspace }}/mapping/missing_rules.txt
- name: Check Build production
run: |
if [ ! -d ${{ github.workspace }}/apk ]; then
echo "Build production not found! Check gradle logs."
exit 1
fi
cd ${{ github.workspace }}/apk/
if [ ! -e legado_*.apk ]; then
echo "Build production not found! Check gradle logs."
exit 1
fi
- name: Upload App To Artifact
uses: actions/upload-artifact@v4
with:
name: legado.${{ env.product }}.${{ env.type }}
if-no-files-found: ignore
path: ${{ github.workspace }}/apk/*.apk
- name: Upload Mapping File To Artifact
uses: actions/upload-artifact@v4
with:
name: legado.${{ env.product }}.${{ env.type }}.mapping
if-no-files-found: ignore
path: ${{ github.workspace }}/mapping/mapping.txt
prerelease:
needs: [ prepare, build ]
if: github.event_name != 'pull_request' && github.repository == 'gedoor/legado'
runs-on: ubuntu-latest
env:
VERSION: ${{ needs.prepare.outputs.version }}
steps:
- uses: actions/checkout@v4
- uses: actions/download-artifact@v4
with:
path: apk/
- working-directory: apk/
run: |
mv */*.apk .
rm -rf */
for file in `ls *.apk`; do
if [[ "$file" == *原包名* ]]; then
mv "$file" $(echo $file | sed s/原包名/release/)
else
mv "$file" $(echo $file | sed s/共存/releaseA/)
fi
done
- name: Delete Pre-Release
run: |
if gh release view beta &>/dev/null; then
gh release delete beta -y
fi
env:
GH_TOKEN: ${{ github.token }}
- name: Create or update beta tag
uses: richardsimko/update-tag@v1
with:
tag_name: beta
env:
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
- name: Publish Pre-Release
uses: ncipollo/release-action@v1
with:
name: legado_app_${{ env.VERSION }}
tag: "beta"
body: |
此版本为测试版,签名与正式版不同,可能存在不稳定情况,升级前请务必备份好数据。
releaseA 为共存版本,可同时安装使用,功能没有区别。
prerelease: true
artifacts: ${{ github.workspace }}/apk/*.apk
lanzou:
needs: [ prepare, build ]
if: ${{ github.event_name != 'pull_request' && needs.prepare.outputs.lanzou == 'yes' }}
runs-on: ubuntu-latest
env:
# 登录蓝奏云后在控制台运行document.cookie
ylogin: ${{ secrets.LANZOU_ID }}
phpdisk_info: ${{ secrets.LANZOU_PSD }}
# 蓝奏云里的文件夹ID(阅读3测试版:2670621)
LANZOU_FOLDER_ID: ${{ secrets.LANZOU_FOLDER_ID }}
#蓝奏云分享链接
LANZOU_URL: ${{ secrets.LANZOU_URL }}
steps:
- uses: actions/checkout@v4
- uses: actions/download-artifact@v4
with:
path: apk/
- working-directory: apk/
run: mv */*.apk . ;rm -rf */
- name: Upload To Lanzou
continue-on-error: true
run: |
path="$GITHUB_WORKSPACE/apk/"
python3 $GITHUB_WORKSPACE/.github/scripts/lzy_web.py "$path" "$LANZOU_FOLDER_ID"
echo "[$(date -u -d '+8 hour' '+%Y.%m.%d %H:%M:%S')] 分享链接: $LANZOU_URL"
test_Branch:
needs: [ prepare, build ]
runs-on: ubuntu-latest
if: ${{ github.event_name != 'pull_request' && github.actor == 'gedoor' }}
steps:
- uses: actions/checkout@v4
- uses: actions/download-artifact@v4
with:
path: apk/
- working-directory: apk/
run: mv */*.apk . ;rm -rf */
- name: Push To "test" Branch
run: |
cd $GITHUB_WORKSPACE/apk/
git init
git checkout -b test
git config --global user.name "github-actions[bot]"
git config --global user.email "41898282+github-actions[bot]@users.noreply.github.com"
git remote add origin "https://${{ github.actor }}:${{ secrets.ACTIONS_TOKEN }}@github.com/${{ github.actor }}/release"
git add *.apk
git commit -m "${{ needs.prepare.outputs.versionL }}"
git push -f -u origin test
telegram:
needs: [ prepare, build ]
if: ${{ github.event_name != 'pull_request' && needs.prepare.outputs.telegram == 'yes' }}
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v4
- uses: actions/download-artifact@v4
with:
path: apk/
- working-directory: apk/
run: |
for file in `ls */*.apk`; do
mv "$file" "$(echo "$file"|sed -e 's#.*\/##g' -e "s/_/ /g" -e 's/legado/阅读/')"
done
rm -rf */
- name: Post to channel
uses: xireiki/channel-post@v1
with:
chat_id: ${{ secrets.CHANNEL_ID }}
bot_token: ${{ secrets.BOT_TOKEN }}
context: "#阅读 #Legado #Beta ${{ needs.prepare.outputs.versionL }}"
path: apk/*
method: sendFile
================================================
FILE: .github/workflows/web.yml
================================================
name: Build Web
on:
push:
branches:
- master
paths:
- '**/modules/web/**'
pull_request:
paths:
- '**/modules/web/**'
workflow_dispatch:
env:
UPSTREAM_REPOSITORY: gedoor/legado
jobs:
build:
runs-on: ubuntu-latest
steps:
- name: Checkout
uses: actions/checkout@v4
- name: Install Node.js
uses: actions/setup-node@v4
with:
node-version: 22
- uses: pnpm/action-setup@v4
name: Install pnpm
id: pnpm-install
with:
version: 9
run_install: false
- name: Get pnpm store directory
id: pnpm-cache
shell: bash
run: |
echo "STORE_PATH=$(pnpm store path)" >> $GITHUB_OUTPUT
- uses: actions/cache@v4
name: Setup pnpm cache
with:
path: ${{ steps.pnpm-cache.outputs.STORE_PATH }}
key: ${{ runner.os }}-pnpm-store-${{ hashFiles('**/web/package.json') }}
restore-keys: |
${{ runner.os }}-pnpm-store-
- name: Build and move files
working-directory: modules/web
run: |
pnpm i
pnpm build
version="v$(date -d "8 hour" -u +3.%y.%m%d%H)"
echo "APP_VER=$version" >> $GITHUB_ENV
- name: push changes
if: ${{ github.event_name != 'pull_request' && github.repository == env.UPSTREAM_REPOSITORY }}
uses: stefanzweifel/git-auto-commit-action@v5.1.0
with:
commit_message: Bump web ${{ env.APP_VER}}
file_pattern: app/src/main/assets/web/vue/
================================================
FILE: .gitignore
================================================
*.iml
.gradle
local.properties
.DS_Store
/build
build/
/captures
.externalNativeBuild
/release
/tmp
node_modules/
/app/app
/app/google
/app/gradle.properties
package-lock.json
.idea/
# Kotlin 2.0
.kotlin/
================================================
FILE: CHANGELOG.md
================================================
**2022/10/02**
* 更新cronet: 106.0.5249.79
* 正文选择菜单朗读按钮长按可切换朗读选择内容和从选择开始处一直朗读
* 源编辑输入框设置最大行数12,在行数特别多的时候更容易滚动到其它输入
* 修复某些情况下无法搜索到标题的bug,净化规则较多的可能会降低搜索速度 by Xwite
* 修复文件类书源换源后阅读bug by Xwite
* Cronet 支持DnsHttpsSvcb by g2s20150909
* 修复web进度同步问题 by 821938089
* 启用混淆以减小app大小 有bug请带日志反馈
* 其它一些优化
================================================
FILE: English.md
================================================
# [English](English.md) [中文](README.md)
[](https://play.google.com/store/apps/details?id=io.legado.play.release)
Legado / 开源阅读




Legado / 开源阅读





s;s++)if(t=b.relative[e[s].type])c=[be(we(c),t)];else{if((t=b.filter[e[s].type].apply(null,e[s].matches))[k]){for(n=++s;r>n&&!b.relative[e[n].type];n++);return Ce(s>1&&we(c),s>1&&xe(e.slice(0,s-1).concat({value:" "===e[s-2].type?"*":""})).replace(B,"$1"),t,n>s&&Ee(e.slice(s,n)),r>n&&Ee(e=e.slice(n)),r>n&&xe(e))}c.push(t)}return we(c)}var e,d,b,o,i,h,f,g,w,u,l,T,C,a,E,v,s,c,y,k="sizzle"+1*new Date,m=n.document,S=0,r=0,p=ue(),x=ue(),N=ue(),A=ue(),D=function(e,t){return e===t&&(l=!0),0},j={}.hasOwnProperty,t=[],q=t.pop,L=t.push,H=t.push,O=t.slice,P=function(e,t){for(var n=0,r=e.length;r>n;n++)if(e[n]===t)return n;return-1},R="checked|selected|async|autofocus|autoplay|controls|defer|disabled|hidden|ismap|loop|multiple|open|readonly|required|scoped",M="[\\x20\\t\\r\\n\\f]",I="(?:\\\\.|[\\w-]|[^\x00-\\xa0])+",W="\\["+M+"*("+I+")(?:"+M+"*([*^$|!~]?=)"+M+"*(?:'((?:\\\\.|[^\\\\'])*)'|\"((?:\\\\.|[^\\\\\"])*)\"|("+I+"))|)"+M+"*\\]",$=":("+I+")(?:\\((('((?:\\\\.|[^\\\\'])*)'|\"((?:\\\\.|[^\\\\\"])*)\")|((?:\\\\.|[^\\\\()[\\]]|"+W+")*)|.*)\\)|)",F=new RegExp(M+"+","g"),B=new RegExp("^"+M+"+|((?:^|[^\\\\])(?:\\\\.)*)"+M+"+$","g"),_=new RegExp("^"+M+"*,"+M+"*"),z=new RegExp("^"+M+"*([>+~]|"+M+")"+M+"*"),U=new RegExp(M+"|>"),X=new RegExp($),V=new RegExp("^"+I+"$"),G={ID:new RegExp("^#("+I+")"),CLASS:new RegExp("^\\.("+I+")"),TAG:new RegExp("^("+I+"|[*])"),ATTR:new RegExp("^"+W),PSEUDO:new RegExp("^"+$),CHILD:new RegExp("^:(only|first|last|nth|nth-last)-(child|of-type)(?:\\("+M+"*(even|odd|(([+-]|)(\\d*)n|)"+M+"*(?:([+-]|)"+M+"*(\\d+)|))"+M+"*\\)|)","i"),bool:new RegExp("^(?:"+R+")$","i"),needsContext:new RegExp("^"+M+"*[>+~]|:(even|odd|eq|gt|lt|nth|first|last)(?:\\("+M+"*((?:-\\d)?\\d*)"+M+"*\\)|)(?=[^-]|$)","i")},Y=/HTML$/i,Q=/^(?:input|select|textarea|button)$/i,J=/^h\d$/i,K=/^[^{]+\{\s*\[native \w/,Z=/^(?:#([\w-]+)|(\w+)|\.([\w-]+))$/,ee=/[+~]/,te=new RegExp("\\\\([\\da-f]{1,6}"+M+"?|("+M+")|.)","ig"),ne=function(e,t,n){var r="0x"+t-65536;return r!=r||n?t:0>r?String.fromCharCode(r+65536):String.fromCharCode(r>>10|55296,1023&r|56320)},re=/([\0-\x1f\x7f]|^-?\d)|^-$|[^\0-\x1f\x7f-\uFFFF\w-]/g,ie=function(e,t){return t?"\x00"===e?"\ufffd":e.slice(0,-1)+"\\"+e.charCodeAt(e.length-1).toString(16)+" ":"\\"+e},oe=function(){T()},ae=be(function(e){return!0===e.disabled&&"fieldset"===e.nodeName.toLowerCase()},{dir:"parentNode",next:"legend"});try{H.apply(t=O.call(m.childNodes),m.childNodes),t[m.childNodes.length].nodeType}catch(e){H={apply:t.length?function(e,t){L.apply(e,O.call(t))}:function(e,t){for(var n=e.length,r=0;e[n++]=t[r++];);e.length=n-1}}}for(e in d=se.support={},i=se.isXML=function(e){var t=e.namespaceURI,n=(e.ownerDocument||e).documentElement;return!Y.test(t||n&&n.nodeName||"HTML")},T=se.setDocument=function(e){var t,n,r=e?e.ownerDocument||e:m;return r!==C&&9===r.nodeType&&r.documentElement&&(a=(C=r).documentElement,E=!i(C),m!==C&&(n=C.defaultView)&&n.top!==n&&(n.addEventListener?n.addEventListener("unload",oe,!1):n.attachEvent&&n.attachEvent("onunload",oe)),d.attributes=ce(function(e){return e.className="i",!e.getAttribute("className")}),d.getElementsByTagName=ce(function(e){return e.appendChild(C.createComment("")),!e.getElementsByTagName("*").length}),d.getElementsByClassName=K.test(C.getElementsByClassName),d.getById=ce(function(e){return a.appendChild(e).id=k,!C.getElementsByName||!C.getElementsByName(k).length}),d.getById?(b.filter.ID=function(e){var t=e.replace(te,ne);return function(e){return e.getAttribute("id")===t}},b.find.ID=function(e,t){if("undefined"!=typeof t.getElementById&&E){var n=t.getElementById(e);return n?[n]:[]}}):(b.filter.ID=function(e){var n=e.replace(te,ne);return function(e){var t="undefined"!=typeof e.getAttributeNode&&e.getAttributeNode("id");return t&&t.value===n}},b.find.ID=function(e,t){if("undefined"!=typeof t.getElementById&&E){var n,r,i,o=t.getElementById(e);if(o){if((n=o.getAttributeNode("id"))&&n.value===e)return[o];for(i=t.getElementsByName(e),r=0;o=i[r++];)if((n=o.getAttributeNode("id"))&&n.value===e)return[o]}return[]}}),b.find.TAG=d.getElementsByTagName?function(e,t){return"undefined"!=typeof t.getElementsByTagName?t.getElementsByTagName(e):d.qsa?t.querySelectorAll(e):void 0}:function(e,t){var n,r=[],i=0,o=t.getElementsByTagName(e);if("*"===e){for(;n=o[i++];)1===n.nodeType&&r.push(n);return r}return o},b.find.CLASS=d.getElementsByClassName&&function(e,t){return"undefined"!=typeof t.getElementsByClassName&&E?t.getElementsByClassName(e):void 0},s=[],v=[],(d.qsa=K.test(C.querySelectorAll))&&(ce(function(e){a.appendChild(e).innerHTML="",e.querySelectorAll("[msallowcapture^='']").length&&v.push("[*^$]="+M+"*(?:''|\"\")"),e.querySelectorAll("[selected]").length||v.push("\\["+M+"*(?:value|"+R+")"),e.querySelectorAll("[id~="+k+"-]").length||v.push("~="),e.querySelectorAll(":checked").length||v.push(":checked"),e.querySelectorAll("a#"+k+"+*").length||v.push(".#.+[+~]")}),ce(function(e){e.innerHTML="";var t=C.createElement("input");t.setAttribute("type","hidden"),e.appendChild(t).setAttribute("name","D"),e.querySelectorAll("[name=d]").length&&v.push("name"+M+"*[*^$|!~]?="),2!==e.querySelectorAll(":enabled").length&&v.push(":enabled",":disabled"),a.appendChild(e).disabled=!0,2!==e.querySelectorAll(":disabled").length&&v.push(":enabled",":disabled"),e.querySelectorAll("*,:x"),v.push(",.*:")})),(d.matchesSelector=K.test(c=a.matches||a.webkitMatchesSelector||a.mozMatchesSelector||a.oMatchesSelector||a.msMatchesSelector))&&ce(function(e){d.disconnectedMatch=c.call(e,"*"),c.call(e,"[s!='']:x"),s.push("!=",$)}),v=v.length&&new RegExp(v.join("|")),s=s.length&&new RegExp(s.join("|")),t=K.test(a.compareDocumentPosition),y=t||K.test(a.contains)?function(e,t){var n=9===e.nodeType?e.documentElement:e,r=t&&t.parentNode;return e===r||!(!r||1!==r.nodeType||!(n.contains?n.contains(r):e.compareDocumentPosition&&16&e.compareDocumentPosition(r)))}:function(e,t){if(t)for(;t=t.parentNode;)if(t===e)return!0;return!1},D=t?function(e,t){if(e===t)return l=!0,0;var n=!e.compareDocumentPosition-!t.compareDocumentPosition;return n||(1&(n=(e.ownerDocument||e)===(t.ownerDocument||t)?e.compareDocumentPosition(t):1)||!d.sortDetached&&t.compareDocumentPosition(e)===n?e===C||e.ownerDocument===m&&y(m,e)?-1:t===C||t.ownerDocument===m&&y(m,t)?1:u?P(u,e)-P(u,t):0:4&n?-1:1)}:function(e,t){if(e===t)return l=!0,0;var n,r=0,i=e.parentNode,o=t.parentNode,a=[e],s=[t];if(!i||!o)return e===C?-1:t===C?1:i?-1:o?1:u?P(u,e)-P(u,t):0;if(i===o)return pe(e,t);for(n=e;n=n.parentNode;)a.unshift(n);for(n=t;n=n.parentNode;)s.unshift(n);for(;a[r]===s[r];)r++;return r?pe(a[r],s[r]):a[r]===m?-1:s[r]===m?1:0}),C},se.matches=function(e,t){return se(e,null,null,t)},se.matchesSelector=function(e,t){if((e.ownerDocument||e)!==C&&T(e),d.matchesSelector&&E&&!A[t+" "]&&(!s||!s.test(t))&&(!v||!v.test(t)))try{var n=c.call(e,t);if(n||d.disconnectedMatch||e.document&&11!==e.document.nodeType)return n}catch(e){A(t,!0)}return 0","
"],col:[2,"
"],tr:[2,"","
"],td:[3,"
"],_default:[0,"",""]};ge.optgroup=ge.option,ge.tbody=ge.tfoot=ge.colgroup=ge.caption=ge.thead,ge.th=ge.td;var me,xe,be=/<|?\w+;/;me=E.createDocumentFragment().appendChild(E.createElement("div")),(xe=E.createElement("input")).setAttribute("type","radio"),xe.setAttribute("checked","checked"),xe.setAttribute("name","t"),me.appendChild(xe),y.checkClone=me.cloneNode(!0).cloneNode(!0).lastChild.checked,me.innerHTML="",y.noCloneChecked=!!me.cloneNode(!0).lastChild.defaultValue;var Te=/^key/,Ce=/^(?:mouse|pointer|contextmenu|drag|drop)|click/,Ee=/^([^.]*)(?:\.(.+)|)/;k.event={global:{},add:function(t,e,n,r,i){var o,a,s,u,l,c,f,p,d,h,g,v=Q.get(t);if(v)for(n.handler&&(n=(o=n).handler,i=o.selector),i&&k.find.matchesSelector(ie,i),n.guid||(n.guid=k.guid++),(u=v.events)||(u=v.events={}),(a=v.handle)||(a=v.handle=function(e){return"undefined"!=typeof k&&k.event.triggered!==e.type?k.event.dispatch.apply(t,arguments):void 0}),l=(e=(e||"").match(R)||[""]).length;l--;)d=g=(s=Ee.exec(e[l])||[])[1],h=(s[2]||"").split(".").sort(),d&&(f=k.event.special[d]||{},d=(i?f.delegateType:f.bindType)||d,f=k.event.special[d]||{},c=k.extend({type:d,origType:g,data:r,handler:n,guid:n.guid,selector:i,needsContext:i&&k.expr.match.needsContext.test(i),namespace:h.join(".")},o),(p=u[d])||((p=u[d]=[]).delegateCount=0,f.setup&&!1!==f.setup.call(t,r,h,a)||t.addEventListener&&t.addEventListener(d,a)),f.add&&(f.add.call(t,c),c.handler.guid||(c.handler.guid=n.guid)),i?p.splice(p.delegateCount++,0,c):p.push(c),k.event.global[d]=!0)},remove:function(e,t,n,r,i){var o,a,s,u,l,c,f,p,d,h,g,v=Q.hasData(e)&&Q.get(e);if(v&&(u=v.events)){for(l=(t=(t||"").match(R)||[""]).length;l--;)if(d=g=(s=Ee.exec(t[l])||[])[1],h=(s[2]||"").split(".").sort(),d){for(f=k.event.special[d]||{},p=u[d=(r?f.delegateType:f.bindType)||d]||[],s=s[2]&&new RegExp("(^|\\.)"+h.join("\\.(?:.*\\.|)")+"(\\.|$)"),a=o=p.length;o--;)c=p[o],!i&&g!==c.origType||n&&n.guid!==c.guid||s&&!s.test(c.namespace)||r&&r!==c.selector&&("**"!==r||!c.selector)||(p.splice(o,1),c.selector&&p.delegateCount--,f.remove&&f.remove.call(e,c));a&&!p.length&&(f.teardown&&!1!==f.teardown.call(e,h,v.handle)||k.removeEvent(e,d,v.handle),delete u[d])}else for(d in u)k.event.remove(e,d+t[l],n,r,!0);k.isEmptyObject(u)&&Q.remove(e,"handle events")}},dispatch:function(e){var t,n,r,i,o,a,s=k.event.fix(e),u=new Array(arguments.length),l=(Q.get(this,"events")||{})[s.type]||[],c=k.event.special[s.type]||{};for(u[0]=s,t=1;t","