diff --git a/app/src/google/java/ani/dantotsu/others/AppUpdater.kt b/app/src/google/java/ani/dantotsu/others/AppUpdater.kt index f6f8e8caf6..36b4c35d8e 100644 --- a/app/src/google/java/ani/dantotsu/others/AppUpdater.kt +++ b/app/src/google/java/ani/dantotsu/others/AppUpdater.kt @@ -29,7 +29,6 @@ import ani.dantotsu.util.Logger import kotlinx.coroutines.Dispatchers import kotlinx.coroutines.MainScope import kotlinx.coroutines.launch -import kotlinx.coroutines.time.delay import kotlinx.serialization.SerialName import kotlinx.serialization.Serializable import kotlinx.serialization.json.JsonArray diff --git a/app/src/main/AndroidManifest.xml b/app/src/main/AndroidManifest.xml index 644d218a8b..fe6f1a90e1 100644 --- a/app/src/main/AndroidManifest.xml +++ b/app/src/main/AndroidManifest.xml @@ -394,11 +394,10 @@ - - + - + PrefName.MangaExtensionRepos to "Manga" + "aniyomi" -> PrefName.AnimeExtensionRepos to "Anime" + "novelyomi" -> PrefName.NovelExtensionRepos to "Novel" + else -> throw Exception("Invalid scheme") } val savedRepos: Set = PrefManager.getVal(prefName) val newRepos = savedRepos.toMutableSet() AddRepositoryBottomSheet.addRepoWarning(this) { newRepos.add(url) PrefManager.setVal(prefName, newRepos) - toast("${if (uri.scheme == "tachiyomi") "Manga" else "Anime"} Extension Repo added") + toast("$name Extension Repo added") } return } @@ -488,9 +489,9 @@ class MainActivity : AppCompatActivity() { return@passwordAlertDialog } if (PreferencePackager.unpack(decryptedJson)) { - val intent = Intent(this, this.javaClass) + val newIntent = Intent(this, this.javaClass) this.finish() - startActivity(intent) + startActivity(newIntent) } } else { toast("Password cannot be empty") @@ -499,9 +500,9 @@ class MainActivity : AppCompatActivity() { } else if (name.endsWith(".ani")) { val decryptedJson = jsonString.toString(Charsets.UTF_8) if (PreferencePackager.unpack(decryptedJson)) { - val intent = Intent(this, this.javaClass) + val newIntent = Intent(this, this.javaClass) this.finish() - startActivity(intent) + startActivity(newIntent) } } else { toast("Invalid file type") diff --git a/app/src/main/java/ani/dantotsu/parsers/novel/NovelExtension.kt b/app/src/main/java/ani/dantotsu/parsers/novel/NovelExtension.kt index 42595aaf9d..3261ca77db 100644 --- a/app/src/main/java/ani/dantotsu/parsers/novel/NovelExtension.kt +++ b/app/src/main/java/ani/dantotsu/parsers/novel/NovelExtension.kt @@ -26,6 +26,7 @@ sealed class NovelExtension { override val pkgName: String, override val versionName: String, override val versionCode: Long, + var repository: String, val sources: List, val iconUrl: String, ) : NovelExtension() diff --git a/app/src/main/java/ani/dantotsu/parsers/novel/NovelExtensionGithubApi.kt b/app/src/main/java/ani/dantotsu/parsers/novel/NovelExtensionGithubApi.kt deleted file mode 100644 index 0a225c9d75..0000000000 --- a/app/src/main/java/ani/dantotsu/parsers/novel/NovelExtensionGithubApi.kt +++ /dev/null @@ -1,186 +0,0 @@ -package ani.dantotsu.parsers.novel - - -import android.content.Context -import ani.dantotsu.settings.saving.PrefManager -import ani.dantotsu.settings.saving.PrefName -import ani.dantotsu.util.Logger -import eu.kanade.tachiyomi.extension.ExtensionUpdateNotifier -import eu.kanade.tachiyomi.extension.anime.model.AnimeExtension -import eu.kanade.tachiyomi.extension.anime.model.AnimeLoadResult -import eu.kanade.tachiyomi.extension.util.ExtensionLoader -import eu.kanade.tachiyomi.network.GET -import eu.kanade.tachiyomi.network.NetworkHelper -import eu.kanade.tachiyomi.network.awaitSuccess -import eu.kanade.tachiyomi.network.parseAs -import kotlinx.serialization.Serializable -import kotlinx.serialization.json.Json -import tachiyomi.core.util.lang.withIOContext -import uy.kohesive.injekt.injectLazy -import java.util.Date -import kotlin.time.Duration.Companion.days - -class NovelExtensionGithubApi { - - private val networkService: NetworkHelper by injectLazy() - private val novelExtensionManager: NovelExtensionManager by injectLazy() - private val json: Json by injectLazy() - - private val lastExtCheck: Long = PrefManager.getVal(PrefName.NovelLastExtCheck) - - private var requiresFallbackSource = false - - suspend fun findExtensions(): List { - return withIOContext { - val githubResponse = if (requiresFallbackSource) { - null - } else { - try { - networkService.client - .newCall(GET("${REPO_URL_PREFIX}index.min.json")) - .awaitSuccess() - } catch (e: Throwable) { - Logger.log("Failed to get extensions from GitHub") - requiresFallbackSource = true - null - } - } - - val response = githubResponse ?: run { - Logger.log("using fallback source") - networkService.client - .newCall(GET("${FALLBACK_REPO_URL_PREFIX}index.min.json")) - .awaitSuccess() - } - - Logger.log("response: $response") - - val extensions = with(json) { - response - .parseAs>() - .toExtensions() - } - - // Sanity check - a small number of extensions probably means something broke - // with the repo generator - /*if (extensions.size < 10) { //TODO: uncomment when more extensions are added - throw Exception() - }*/ - Logger.log("extensions: $extensions") - extensions - } - } - - suspend fun checkForUpdates( - context: Context, - fromAvailableExtensionList: Boolean = false - ): List? { - // Limit checks to once a day at most - if (fromAvailableExtensionList && Date().time < lastExtCheck + 1.days.inWholeMilliseconds) { - return null - } - - val extensions = if (fromAvailableExtensionList) { - novelExtensionManager.availableExtensionsFlow.value - } else { - findExtensions().also { - PrefManager.setVal(PrefName.NovelLastExtCheck, Date().time) - } - } - - val installedExtensions = ExtensionLoader.loadNovelExtensions(context) - .filterIsInstance() - .map { it.extension } - - val extensionsWithUpdate = mutableListOf() - for (installedExt in installedExtensions) { - val pkgName = installedExt.pkgName - val availableExt = extensions.find { it.pkgName == pkgName } ?: continue - - val hasUpdatedVer = availableExt.versionCode > installedExt.versionCode - val hasUpdate = installedExt.isUnofficial.not() && (hasUpdatedVer) - if (hasUpdate) { - extensionsWithUpdate.add(installedExt) - } - } - - if (extensionsWithUpdate.isNotEmpty()) { - ExtensionUpdateNotifier(context).promptUpdates(extensionsWithUpdate.map { it.name }) - } - - return extensionsWithUpdate - } - - private fun List.toExtensions(): List { - return mapNotNull { extension -> - val sources = extension.sources?.map { source -> - NovelExtensionSourceJsonObject( - source.id, - source.lang, - source.name, - source.baseUrl, - ) - } - val iconUrl = "${REPO_URL_PREFIX}icon/${extension.pkg}.png" - NovelExtension.Available( - extension.name, - extension.pkg, - extension.apk, - extension.code, - sources?.toSources() ?: emptyList(), - iconUrl, - ) - } - } - - private fun List.toSources(): List { - return map { source -> - AvailableNovelSources( - source.id, - source.lang, - source.name, - source.baseUrl, - ) - } - } - - fun getApkUrl(extension: NovelExtension.Available): String { - return "${getUrlPrefix()}apk/${extension.pkgName}.apk" - } - - private fun getUrlPrefix(): String { - return if (requiresFallbackSource) { - FALLBACK_REPO_URL_PREFIX - } else { - REPO_URL_PREFIX - } - } -} - -private const val REPO_URL_PREFIX = - "https://raw.githubusercontent.com/dannovels/novel-extensions/main/" -private const val FALLBACK_REPO_URL_PREFIX = - "https://gcore.jsdelivr.net/gh/dannovels/novel-extensions@latest/" - -@Serializable -private data class NovelExtensionJsonObject( - val name: String, - val pkg: String, - val apk: String, - val lang: String, - val code: Long, - val version: String, - val nsfw: Int, - val hasReadme: Int = 0, - val hasChangelog: Int = 0, - val sources: List?, -) - -@Serializable -private data class NovelExtensionSourceJsonObject( - val id: Long, - val lang: String, - val name: String, - val baseUrl: String, -) - diff --git a/app/src/main/java/ani/dantotsu/parsers/novel/NovelExtensionManager.kt b/app/src/main/java/ani/dantotsu/parsers/novel/NovelExtensionManager.kt index aab307b41f..5ef1aa7d35 100644 --- a/app/src/main/java/ani/dantotsu/parsers/novel/NovelExtensionManager.kt +++ b/app/src/main/java/ani/dantotsu/parsers/novel/NovelExtensionManager.kt @@ -6,6 +6,7 @@ import ani.dantotsu.media.MediaType import ani.dantotsu.snackString import ani.dantotsu.util.Logger import eu.kanade.tachiyomi.extension.InstallStep +import eu.kanade.tachiyomi.extension.api.ExtensionGithubApi import eu.kanade.tachiyomi.extension.util.ExtensionInstallReceiver import eu.kanade.tachiyomi.extension.util.ExtensionInstaller import eu.kanade.tachiyomi.extension.util.ExtensionLoader @@ -22,7 +23,7 @@ class NovelExtensionManager(private val context: Context) { /** * API where all the available Novel extensions can be found. */ - private val api = NovelExtensionGithubApi() + private val api = ExtensionGithubApi() /** * The installer which installs, updates and uninstalls the Novel extensions. @@ -70,7 +71,7 @@ class NovelExtensionManager(private val context: Context) { */ suspend fun findAvailableExtensions() { val extensions: List = try { - api.findExtensions() + api.findNovelExtensions() } catch (e: Exception) { Logger.log("Error finding extensions: ${e.message}") withUIContext { snackString("Failed to get Novel extensions list") } @@ -119,7 +120,7 @@ class NovelExtensionManager(private val context: Context) { * @param extension The anime extension to be installed. */ fun installExtension(extension: NovelExtension.Available): Observable { - return installer.downloadAndInstall(api.getApkUrl(extension), extension.pkgName, + return installer.downloadAndInstall(api.getNovelApkUrl(extension), extension.pkgName, extension.name, MediaType.NOVEL) } diff --git a/app/src/main/java/ani/dantotsu/settings/AddRepositoryBottomSheet.kt b/app/src/main/java/ani/dantotsu/settings/AddRepositoryBottomSheet.kt index 4581f8cb91..30dbdaba37 100644 --- a/app/src/main/java/ani/dantotsu/settings/AddRepositoryBottomSheet.kt +++ b/app/src/main/java/ani/dantotsu/settings/AddRepositoryBottomSheet.kt @@ -2,6 +2,7 @@ package ani.dantotsu.settings import android.content.Context import android.os.Bundle +import android.view.HapticFeedbackConstants import android.view.KeyEvent import android.view.LayoutInflater import android.view.View @@ -10,29 +11,52 @@ import android.view.inputmethod.EditorInfo import androidx.recyclerview.widget.LinearLayoutManager import ani.dantotsu.BottomSheetDialogFragment import ani.dantotsu.R +import ani.dantotsu.copyToClipboard import ani.dantotsu.databinding.BottomSheetAddRepositoryBinding import ani.dantotsu.databinding.ItemRepoBinding import ani.dantotsu.media.MediaType +import ani.dantotsu.parsers.novel.NovelExtensionManager +import ani.dantotsu.settings.saving.PrefManager +import ani.dantotsu.settings.saving.PrefName import ani.dantotsu.util.customAlertDialog import com.xwray.groupie.GroupieAdapter import com.xwray.groupie.viewbinding.BindableItem +import eu.kanade.tachiyomi.extension.anime.AnimeExtensionManager +import eu.kanade.tachiyomi.extension.manga.MangaExtensionManager +import kotlinx.coroutines.CoroutineScope +import kotlinx.coroutines.Dispatchers +import kotlinx.coroutines.launch +import uy.kohesive.injekt.Injekt +import uy.kohesive.injekt.api.get class RepoItem( val url: String, - val onRemove: (String) -> Unit + private val mediaType: MediaType, + val onRemove: (String, MediaType) -> Unit ) :BindableItem() { override fun getLayout() = R.layout.item_repo override fun bind(viewBinding: ItemRepoBinding, position: Int) { - viewBinding.repoNameTextView.text = url + viewBinding.repoNameTextView.text = url.cleanShownUrl() viewBinding.repoDeleteImageView.setOnClickListener { - onRemove(url) + onRemove(url, mediaType) + } + viewBinding.repoCopyImageView.setOnClickListener { + viewBinding.repoCopyImageView.performHapticFeedback(HapticFeedbackConstants.LONG_PRESS) + copyToClipboard(url, true) } } override fun initializeViewBinding(view: View): ItemRepoBinding { return ItemRepoBinding.bind(view) } + + private fun String.cleanShownUrl(): String { + return this + .removePrefix("https://raw.githubusercontent.com/") + .replace("index.min.json", "") + .removeSuffix("/") + } } class AddRepositoryBottomSheet : BottomSheetDialogFragment() { @@ -41,7 +65,7 @@ class AddRepositoryBottomSheet : BottomSheetDialogFragment() { private var mediaType: MediaType = MediaType.ANIME private var onRepositoryAdded: ((String, MediaType) -> Unit)? = null private var repositories: MutableList = mutableListOf() - private var onRepositoryRemoved: ((String) -> Unit)? = null + private var onRepositoryRemoved: ((String, MediaType) -> Unit)? = null private var adapter: GroupieAdapter = GroupieAdapter() override fun onCreateView( @@ -62,24 +86,19 @@ class AddRepositoryBottomSheet : BottomSheetDialogFragment() { LinearLayoutManager.VERTICAL, false ) - adapter.addAll(repositories.map { RepoItem(it, ::onRepositoryRemoved) }) + adapter.addAll(repositories.map { RepoItem(it, mediaType, ::onRepositoryRemoved) }) binding.repositoryInput.hint = when(mediaType) { MediaType.ANIME -> getString(R.string.anime_add_repository) MediaType.MANGA -> getString(R.string.manga_add_repository) - else -> "" + MediaType.NOVEL -> getString(R.string.novel_add_repository) } binding.addButton.setOnClickListener { val input = binding.repositoryInput.text.toString() val error = isValidUrl(input) if (error == null) { - context?.let { context -> - addRepoWarning(context) { - onRepositoryAdded?.invoke(input, mediaType) - dismiss() - } - } + acceptUrl(input) } else { binding.repositoryInput.error = error } @@ -96,12 +115,7 @@ class AddRepositoryBottomSheet : BottomSheetDialogFragment() { if (url.isNotBlank()) { val error = isValidUrl(url) if (error == null) { - context?.let { context -> - addRepoWarning(context) { - onRepositoryAdded?.invoke(url, mediaType) - dismiss() - } - } + acceptUrl(url) return@setOnEditorActionListener true } else { binding.repositoryInput.error = error @@ -112,20 +126,62 @@ class AddRepositoryBottomSheet : BottomSheetDialogFragment() { } } - private fun onRepositoryRemoved(url: String) { - onRepositoryRemoved?.invoke(url) - repositories.remove(url) - adapter.update(repositories.map { RepoItem(it, ::onRepositoryRemoved) }) + private fun acceptUrl(url: String) { + val finalUrl = getRepoUrl(url) + context?.let { context -> + addRepoWarning(context) { + onRepositoryAdded?.invoke(finalUrl, mediaType) + dismiss() + } + } } - private fun isValidUrl(url: String): String? { - if (!url.startsWith("https://") && !url.startsWith("http://")) - return "URL must start with http:// or https://" - if (!url.removeSuffix("/").endsWith("index.min.json")) - return "URL must end with index.min.json" + private fun isValidUrl(input: String): String? { + if (input.startsWith("http://") || input.startsWith("https://")) { + if (!input.removeSuffix("/").endsWith("index.min.json")) { + return "URL must end with index.min.json" + } + return null + } + + val parts = input.split("/") + if (parts.size !in 2..3) { + return "Must be a full URL or in format: username/repo[/branch]" + } + + val username = parts[0] + val repo = parts[1] + val branch = if (parts.size == 3) parts[2] else "repo" + + if (username.isBlank() || repo.isBlank()) { + return "Username and repository name cannot be empty" + } + if (parts.size == 3 && branch.isBlank()) { + return "Branch name cannot be empty" + } + return null } + private fun getRepoUrl(input: String): String { + if (input.startsWith("http://") || input.startsWith("https://")) { + return input + } + + val parts = input.split("/") + val username = parts[0] + val repo = parts[1] + val branch = if (parts.size == 3) parts[2] else "repo" + + return "https://raw.githubusercontent.com/$username/$repo/$branch/index.min.json" + } + + private fun onRepositoryRemoved(url: String, mediaType: MediaType) { + onRepositoryRemoved?.invoke(url, mediaType) + repositories.remove(url) + adapter.update(repositories.map { RepoItem(it, mediaType, ::onRepositoryRemoved) }) + } + override fun onDestroyView() { super.onDestroyView() _binding = null @@ -142,11 +198,81 @@ class AddRepositoryBottomSheet : BottomSheetDialogFragment() { .setNegButton(R.string.cancel) { } .show() } + + fun addRepo(input: String, mediaType: MediaType) { + val validLink = if (input.contains("github.com") && input.contains("blob")) { + input.replace("github.com", "raw.githubusercontent.com") + .replace("/blob/", "/") + } else input + + when (mediaType) { + MediaType.ANIME -> { + val anime = + PrefManager.getVal>(PrefName.AnimeExtensionRepos) + .plus(validLink) + PrefManager.setVal(PrefName.AnimeExtensionRepos, anime) + CoroutineScope(Dispatchers.IO).launch { + Injekt.get().findAvailableExtensions() + } + } + MediaType.MANGA -> { + val manga = + PrefManager.getVal>(PrefName.MangaExtensionRepos) + .plus(validLink) + PrefManager.setVal(PrefName.MangaExtensionRepos, manga) + CoroutineScope(Dispatchers.IO).launch { + Injekt.get().findAvailableExtensions() + } + } + MediaType.NOVEL -> { + val novel = + PrefManager.getVal>(PrefName.NovelExtensionRepos) + .plus(validLink) + PrefManager.setVal(PrefName.NovelExtensionRepos, novel) + CoroutineScope(Dispatchers.IO).launch { + Injekt.get().findAvailableExtensions() + } + } + } + } + + fun removeRepo(input: String, mediaType: MediaType) { + when (mediaType) { + MediaType.ANIME -> { + val anime = + PrefManager.getVal>(PrefName.AnimeExtensionRepos) + .minus(input) + PrefManager.setVal(PrefName.AnimeExtensionRepos, anime) + CoroutineScope(Dispatchers.IO).launch { + Injekt.get().findAvailableExtensions() + } + } + MediaType.MANGA -> { + val manga = + PrefManager.getVal>(PrefName.MangaExtensionRepos) + .minus(input) + PrefManager.setVal(PrefName.MangaExtensionRepos, manga) + CoroutineScope(Dispatchers.IO).launch { + Injekt.get().findAvailableExtensions() + } + } + MediaType.NOVEL -> { + val novel = + PrefManager.getVal>(PrefName.NovelExtensionRepos) + .minus(input) + PrefManager.setVal(PrefName.NovelExtensionRepos, novel) + CoroutineScope(Dispatchers.IO).launch { + Injekt.get().findAvailableExtensions() + } + } + } + } + fun newInstance( mediaType: MediaType, repositories: List, onRepositoryAdded: (String, MediaType) -> Unit, - onRepositoryRemoved: (String) -> Unit + onRepositoryRemoved: (String, MediaType) -> Unit ): AddRepositoryBottomSheet { return AddRepositoryBottomSheet().apply { this.mediaType = mediaType diff --git a/app/src/main/java/ani/dantotsu/settings/ExtensionsActivity.kt b/app/src/main/java/ani/dantotsu/settings/ExtensionsActivity.kt index dd11df3a64..de9dbbf056 100644 --- a/app/src/main/java/ani/dantotsu/settings/ExtensionsActivity.kt +++ b/app/src/main/java/ani/dantotsu/settings/ExtensionsActivity.kt @@ -1,18 +1,12 @@ package ani.dantotsu.settings -import android.app.AlertDialog import android.content.Intent import android.os.Bundle import android.text.Editable import android.text.TextWatcher -import android.view.HapticFeedbackConstants -import android.view.KeyEvent -import android.view.LayoutInflater import android.view.View import android.view.ViewGroup -import android.view.inputmethod.EditorInfo import android.widget.AutoCompleteTextView -import android.widget.EditText import androidx.appcompat.app.AppCompatActivity import androidx.core.content.ContextCompat import androidx.core.view.updateLayoutParams @@ -20,10 +14,7 @@ import androidx.fragment.app.Fragment import androidx.viewpager2.adapter.FragmentStateAdapter import androidx.viewpager2.widget.ViewPager2 import ani.dantotsu.R -import ani.dantotsu.copyToClipboard import ani.dantotsu.databinding.ActivityExtensionsBinding -import ani.dantotsu.databinding.DialogRepositoriesBinding -import ani.dantotsu.databinding.ItemRepositoryBinding import ani.dantotsu.initActivity import ani.dantotsu.media.MediaType import ani.dantotsu.navBarHeight @@ -37,20 +28,11 @@ import ani.dantotsu.themes.ThemeManager import ani.dantotsu.util.customAlertDialog import com.google.android.material.tabs.TabLayout import com.google.android.material.tabs.TabLayoutMediator -import eu.kanade.tachiyomi.extension.anime.AnimeExtensionManager -import eu.kanade.tachiyomi.extension.manga.MangaExtensionManager -import kotlinx.coroutines.CoroutineScope -import kotlinx.coroutines.Dispatchers -import kotlinx.coroutines.launch -import uy.kohesive.injekt.injectLazy import java.util.Locale class ExtensionsActivity : AppCompatActivity() { lateinit var binding: ActivityExtensionsBinding - private val animeExtensionManager: AnimeExtensionManager by injectLazy() - private val mangaExtensionManager: MangaExtensionManager by injectLazy() - override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState) @@ -124,6 +106,9 @@ class ExtensionsActivity : AppCompatActivity() { if (tab.text?.contains("Manga") == true) { generateRepositoryButton(MediaType.MANGA) } + if (tab.text?.contains("Novels") == true) { + generateRepositoryButton(MediaType.NOVEL) + } } override fun onTabUnselected(tab: TabLayout.Tab) { @@ -199,136 +184,28 @@ class ExtensionsActivity : AppCompatActivity() { } } - private fun processUserInput(input: String, mediaType: MediaType) { - val entry = if (input.endsWith("/") || input.endsWith("index.min.json")) - input.substring(0, input.lastIndexOf("/")) else input - if (mediaType == MediaType.ANIME) { - val anime = - PrefManager.getVal>(PrefName.AnimeExtensionRepos).plus(entry) - PrefManager.setVal(PrefName.AnimeExtensionRepos, anime) - CoroutineScope(Dispatchers.IO).launch { - animeExtensionManager.findAvailableExtensions() - } - } - if (mediaType == MediaType.MANGA) { - val manga = - PrefManager.getVal>(PrefName.MangaExtensionRepos).plus(entry) - PrefManager.setVal(PrefName.MangaExtensionRepos, manga) - CoroutineScope(Dispatchers.IO).launch { - mangaExtensionManager.findAvailableExtensions() - } - } - } - - private fun getSavedRepositories(repoInventory: ViewGroup, type: MediaType) { - repoInventory.removeAllViews() - val prefName: PrefName? = when (type) { - MediaType.ANIME -> { - PrefName.AnimeExtensionRepos - } - - MediaType.MANGA -> { - PrefName.MangaExtensionRepos - } - - else -> { - null - } - } - prefName?.let { repoList -> - PrefManager.getVal>(repoList).forEach { item -> - val view = ItemRepositoryBinding.inflate( - LayoutInflater.from(repoInventory.context), repoInventory, true - ) - view.repositoryItem.text = item.removePrefix("https://raw.githubusercontent.com") - view.repositoryItem.setOnClickListener { - customAlertDialog().apply { - setTitle(R.string.rem_repository) - setMessage(item) - setPosButton(R.string.ok) { - val repos = PrefManager.getVal>(prefName).minus(item) - PrefManager.setVal(prefName, repos) - repoInventory.removeView(view.root) - CoroutineScope(Dispatchers.IO).launch { - when (type) { - MediaType.ANIME -> { - animeExtensionManager.findAvailableExtensions() - } - - MediaType.MANGA -> { - mangaExtensionManager.findAvailableExtensions() - } - - else -> {} - } - } - } - setNegButton(R.string.cancel) - show() - } - } - view.repositoryItem.setOnLongClickListener { - it.performHapticFeedback(HapticFeedbackConstants.LONG_PRESS) - copyToClipboard(item, true) - true + private fun generateRepositoryButton(type: MediaType) { + binding.openSettingsButton.setOnClickListener { + val repos: Set = when (type) { + MediaType.ANIME -> { + PrefManager.getVal(PrefName.AnimeExtensionRepos) } - } - } - } - private fun processEditorAction(editText: EditText, mediaType: MediaType) { - editText.setOnEditorActionListener { textView, action, keyEvent -> - if (action == EditorInfo.IME_ACTION_SEARCH || action == EditorInfo.IME_ACTION_DONE || - (keyEvent?.action == KeyEvent.ACTION_UP - && keyEvent.keyCode == KeyEvent.KEYCODE_ENTER) - ) { - return@setOnEditorActionListener if (textView.text.isNullOrBlank()) { - false - } else { - processUserInput(textView.text.toString(), mediaType) - true + MediaType.MANGA -> { + PrefManager.getVal(PrefName.MangaExtensionRepos) } - } - false - } - } - - private fun generateRepositoryButton(type: MediaType) { - val hintResource: Int? = when (type) { - MediaType.ANIME -> { - R.string.anime_add_repository - } - - MediaType.MANGA -> { - R.string.manga_add_repository - } - else -> { - null - } - } - hintResource?.let { res -> - binding.openSettingsButton.setOnClickListener { - val dialogView = DialogRepositoriesBinding.inflate( - LayoutInflater.from(binding.openSettingsButton.context), null, false - ) - dialogView.repositoryTextBox.hint = getString(res) - dialogView.repoInventory.apply { - getSavedRepositories(this, type) - } - processEditorAction(dialogView.repositoryTextBox, type) - customAlertDialog().apply { - setTitle(R.string.edit_repositories) - setCustomView(dialogView.root) - setPosButton(R.string.add_list) { - if (!dialogView.repositoryTextBox.text.isNullOrBlank()) { - processUserInput(dialogView.repositoryTextBox.text.toString(), type) - } - } - setNegButton(R.string.close) - show() + MediaType.NOVEL -> { + PrefManager.getVal(PrefName.NovelExtensionRepos) } } + AddRepositoryBottomSheet.newInstance( + type, + repos.toList(), + AddRepositoryBottomSheet::addRepo, + AddRepositoryBottomSheet::removeRepo + + ).show(supportFragmentManager, "add_repo") } } } diff --git a/app/src/main/java/ani/dantotsu/settings/SettingsExtensionsActivity.kt b/app/src/main/java/ani/dantotsu/settings/SettingsExtensionsActivity.kt index 59add57808..8a84d53d42 100644 --- a/app/src/main/java/ani/dantotsu/settings/SettingsExtensionsActivity.kt +++ b/app/src/main/java/ani/dantotsu/settings/SettingsExtensionsActivity.kt @@ -1,14 +1,10 @@ package ani.dantotsu.settings -import android.app.AlertDialog import android.content.Intent import android.os.Bundle import android.view.HapticFeedbackConstants -import android.view.KeyEvent import android.view.LayoutInflater import android.view.ViewGroup -import android.view.inputmethod.EditorInfo -import android.widget.EditText import androidx.appcompat.app.AppCompatActivity import androidx.core.content.ContextCompat import androidx.core.view.isVisible @@ -32,9 +28,6 @@ import ani.dantotsu.util.customAlertDialog import eu.kanade.domain.base.BasePreferences import eu.kanade.tachiyomi.extension.anime.AnimeExtensionManager import eu.kanade.tachiyomi.extension.manga.MangaExtensionManager -import kotlinx.coroutines.CoroutineScope -import kotlinx.coroutines.Dispatchers -import kotlinx.coroutines.launch import uy.kohesive.injekt.Injekt import uy.kohesive.injekt.api.get import uy.kohesive.injekt.injectLazy @@ -42,8 +35,7 @@ import uy.kohesive.injekt.injectLazy class SettingsExtensionsActivity : AppCompatActivity() { private lateinit var binding: ActivitySettingsExtensionsBinding private val extensionInstaller = Injekt.get().extensionInstaller() - private val animeExtensionManager: AnimeExtensionManager by injectLazy() - private val mangaExtensionManager: MangaExtensionManager by injectLazy() + override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState) ThemeManager(this).applyTheme() @@ -61,7 +53,7 @@ class SettingsExtensionsActivity : AppCompatActivity() { } fun setExtensionOutput(repoInventory: ViewGroup, type: MediaType) { repoInventory.removeAllViews() - val prefName: PrefName? = when (type) { + val prefName: PrefName = when (type) { MediaType.ANIME -> { PrefName.AnimeExtensionRepos } @@ -70,74 +62,24 @@ class SettingsExtensionsActivity : AppCompatActivity() { PrefName.MangaExtensionRepos } - else -> { - null - } - } - prefName?.let { repoList -> - PrefManager.getVal>(repoList).forEach { item -> - val view = ItemRepositoryBinding.inflate( - LayoutInflater.from(repoInventory.context), repoInventory, true - ) - view.repositoryItem.text = - item.removePrefix("https://raw.githubusercontent.com/") - view.repositoryItem.setOnClickListener { - context.customAlertDialog().apply { - setTitle(R.string.rem_repository) - setMessage(item) - setPosButton(R.string.ok) { - val repos = PrefManager.getVal>(repoList).minus(item) - PrefManager.setVal(repoList, repos) - setExtensionOutput(repoInventory, type) - CoroutineScope(Dispatchers.IO).launch { - when (type) { - MediaType.ANIME -> { - animeExtensionManager.findAvailableExtensions() - } - MediaType.MANGA -> { - mangaExtensionManager.findAvailableExtensions() - } - else -> {} - } - } - } - setNegButton(R.string.cancel) - show() - } - } - view.repositoryItem.setOnLongClickListener { - it.performHapticFeedback(HapticFeedbackConstants.LONG_PRESS) - copyToClipboard(item, true) - true - } + MediaType.NOVEL -> { + PrefName.NovelExtensionRepos } - repoInventory.isVisible = repoInventory.childCount > 0 } - } + PrefManager.getVal>(prefName).forEach { item -> + val view = ItemRepositoryBinding.inflate( + LayoutInflater.from(repoInventory.context), repoInventory, true + ) + view.repositoryItem.text = + item.removePrefix("https://raw.githubusercontent.com/") - fun processUserInput(input: String, mediaType: MediaType, view: ViewGroup) { - val validLink = if (input.contains("github.com") && input.contains("blob")) { - input.replace("github.com", "raw.githubusercontent.com") - .replace("/blob/", "/") - } else input - if (mediaType == MediaType.ANIME) { - val anime = - PrefManager.getVal>(PrefName.AnimeExtensionRepos).plus(validLink) - PrefManager.setVal(PrefName.AnimeExtensionRepos, anime) - CoroutineScope(Dispatchers.IO).launch { - animeExtensionManager.findAvailableExtensions() - } - setExtensionOutput(view, MediaType.ANIME) - } - if (mediaType == MediaType.MANGA) { - val manga = - PrefManager.getVal>(PrefName.MangaExtensionRepos).plus(validLink) - PrefManager.setVal(PrefName.MangaExtensionRepos, manga) - CoroutineScope(Dispatchers.IO).launch { - mangaExtensionManager.findAvailableExtensions() + view.repositoryItem.setOnLongClickListener { + it.performHapticFeedback(HapticFeedbackConstants.LONG_PRESS) + copyToClipboard(item, true) + true } - setExtensionOutput(view, MediaType.MANGA) } + repoInventory.isVisible = repoInventory.childCount > 0 } settingsRecyclerView.adapter = SettingsAdapter( @@ -148,17 +90,18 @@ class SettingsExtensionsActivity : AppCompatActivity() { desc = getString(R.string.anime_add_repository_desc), icon = R.drawable.ic_github, onClick = { - val animeRepos = PrefManager.getVal>(PrefName.AnimeExtensionRepos) + val animeRepos = + PrefManager.getVal>(PrefName.AnimeExtensionRepos) AddRepositoryBottomSheet.newInstance( MediaType.ANIME, animeRepos.toList(), onRepositoryAdded = { input, mediaType -> - processUserInput(input, mediaType, it.attachView) + AddRepositoryBottomSheet.addRepo(input, mediaType) + setExtensionOutput(it.attachView, mediaType) }, - onRepositoryRemoved = { item -> - val repos = PrefManager.getVal>(PrefName.AnimeExtensionRepos).minus(item) - PrefManager.setVal(PrefName.AnimeExtensionRepos, repos) - setExtensionOutput(it.attachView, MediaType.ANIME) + onRepositoryRemoved = { item, mediaType -> + AddRepositoryBottomSheet.removeRepo(item, mediaType) + setExtensionOutput(it.attachView, mediaType) } ).show(supportFragmentManager, "add_repo") }, @@ -172,17 +115,18 @@ class SettingsExtensionsActivity : AppCompatActivity() { desc = getString(R.string.manga_add_repository_desc), icon = R.drawable.ic_github, onClick = { - val mangaRepos = PrefManager.getVal>(PrefName.MangaExtensionRepos) + val mangaRepos = + PrefManager.getVal>(PrefName.MangaExtensionRepos) AddRepositoryBottomSheet.newInstance( MediaType.MANGA, mangaRepos.toList(), onRepositoryAdded = { input, mediaType -> - processUserInput(input, mediaType, it.attachView) + AddRepositoryBottomSheet.addRepo(input, mediaType) + setExtensionOutput(it.attachView, mediaType) }, - onRepositoryRemoved = { item -> - val repos = PrefManager.getVal>(PrefName.MangaExtensionRepos).minus(item) - PrefManager.setVal(PrefName.MangaExtensionRepos, repos) - setExtensionOutput(it.attachView, MediaType.MANGA) + onRepositoryRemoved = { item, mediaType -> + AddRepositoryBottomSheet.removeRepo(item, mediaType) + setExtensionOutput(it.attachView, mediaType) } ).show(supportFragmentManager, "add_repo") }, @@ -190,6 +134,31 @@ class SettingsExtensionsActivity : AppCompatActivity() { setExtensionOutput(it.attachView, MediaType.MANGA) } ), + Settings( + type = 1, + name = getString(R.string.novel_add_repository), + desc = getString(R.string.novel_add_repository_desc), + icon = R.drawable.ic_github, + onClick = { + val novelRepos = + PrefManager.getVal>(PrefName.NovelExtensionRepos) + AddRepositoryBottomSheet.newInstance( + MediaType.NOVEL, + novelRepos.toList(), + onRepositoryAdded = { input, mediaType -> + AddRepositoryBottomSheet.addRepo(input, mediaType) + setExtensionOutput(it.attachView, mediaType) + }, + onRepositoryRemoved = { item, mediaType -> + AddRepositoryBottomSheet.removeRepo(item, mediaType) + setExtensionOutput(it.attachView, mediaType) + } + ).show(supportFragmentManager, "add_repo") + }, + attach = { + setExtensionOutput(it.attachView, MediaType.NOVEL) + } + ), Settings( type = 1, name = getString(R.string.extension_test), @@ -217,7 +186,10 @@ class SettingsExtensionsActivity : AppCompatActivity() { setTitle(R.string.user_agent) setCustomView(dialogView.root) setPosButton(R.string.ok) { - PrefManager.setVal(PrefName.DefaultUserAgent, editText.text.toString()) + PrefManager.setVal( + PrefName.DefaultUserAgent, + editText.text.toString() + ) } setNeutralButton(R.string.reset) { PrefManager.removeVal(PrefName.DefaultUserAgent) @@ -247,7 +219,7 @@ class SettingsExtensionsActivity : AppCompatActivity() { ProxyDialogFragment().show(supportFragmentManager, "dialog") } ), - Settings( + Settings( type = 2, name = getString(R.string.force_legacy_installer), desc = getString(R.string.force_legacy_installer_desc), diff --git a/app/src/main/java/ani/dantotsu/settings/saving/Preferences.kt b/app/src/main/java/ani/dantotsu/settings/saving/Preferences.kt index 659e399b1a..866871b390 100644 --- a/app/src/main/java/ani/dantotsu/settings/saving/Preferences.kt +++ b/app/src/main/java/ani/dantotsu/settings/saving/Preferences.kt @@ -32,6 +32,7 @@ enum class PrefName(val data: Pref) { //TODO: Split this into multiple files ), AnimeExtensionRepos(Pref(Location.General, Set::class, setOf())), MangaExtensionRepos(Pref(Location.General, Set::class, setOf())), + NovelExtensionRepos(Pref(Location.General, Set::class, setOf())), AnimeSourcesOrder(Pref(Location.General, List::class, listOf())), AnimeSearchHistory(Pref(Location.General, Set::class, setOf())), MangaSourcesOrder(Pref(Location.General, List::class, listOf())), diff --git a/app/src/main/java/eu/kanade/tachiyomi/extension/api/ExtensionGithubApi.kt b/app/src/main/java/eu/kanade/tachiyomi/extension/api/ExtensionGithubApi.kt index d0a7af3321..c2771afdec 100644 --- a/app/src/main/java/eu/kanade/tachiyomi/extension/api/ExtensionGithubApi.kt +++ b/app/src/main/java/eu/kanade/tachiyomi/extension/api/ExtensionGithubApi.kt @@ -1,5 +1,7 @@ package eu.kanade.tachiyomi.extension.api +import ani.dantotsu.parsers.novel.AvailableNovelSources +import ani.dantotsu.parsers.novel.NovelExtension import ani.dantotsu.settings.saving.PrefManager import ani.dantotsu.settings.saving.PrefName import ani.dantotsu.util.Logger @@ -192,6 +194,92 @@ internal class ExtensionGithubApi { return "${extension.repository}/apk/${extension.apkName}" } + suspend fun findNovelExtensions(): List { + return withIOContext { + + val extensions: ArrayList = arrayListOf() + + val repos = + PrefManager.getVal>(PrefName.NovelExtensionRepos).toMutableList() + + repos.forEach { + val repoUrl = if (it.contains("index.min.json")) { + it + } else { + "$it${if (it.endsWith('/')) "" else "/"}index.min.json" + } + try { + val githubResponse = try { + networkService.client + .newCall(GET(repoUrl)) + .awaitSuccess() + } catch (e: Throwable) { + Logger.log("Failed to get repo: $repoUrl") + Logger.log(e) + null + } + + val response = githubResponse ?: run { + networkService.client + .newCall(GET(fallbackRepoUrl(it) + "/index.min.json")) + .awaitSuccess() + } + + val repoExtensions = with(json) { + response + .parseAs>() + .toNovelExtensions(it) + } + + extensions.addAll(repoExtensions) + } catch (e: Throwable) { + Logger.log("Failed to get extensions from GitHub") + Logger.log(e) + } + } + + extensions + } + } + + private fun List.toNovelExtensions(repository: String): List { + return mapNotNull { extension -> + val sources = extension.sources?.map { source -> + ExtensionSourceJsonObject( + source.id, + source.lang, + source.name, + source.baseUrl, + ) + } + val iconUrl = "${repository.removeSuffix("/index.min.json")}/icon/${extension.pkg}.png" + NovelExtension.Available( + extension.name, + extension.pkg, + extension.apk, + extension.code, + repository, + sources?.toNovelSources() ?: emptyList(), + iconUrl, + ) + } + } + + private fun List.toNovelSources(): List { + return map { source -> + AvailableNovelSources( + source.id, + source.lang, + source.name, + source.baseUrl, + ) + } + } + + fun getNovelApkUrl(extension: NovelExtension.Available): String { + return "${extension.repository}/apk/${extension.pkgName}.apk" + } + private fun fallbackRepoUrl(repoUrl: String): String? { var fallbackRepoUrl = "https://gcore.jsdelivr.net/gh/" val strippedRepoUrl = repoUrl diff --git a/app/src/main/res/layout/item_repo.xml b/app/src/main/res/layout/item_repo.xml index 658110b5be..6cbf4cb543 100644 --- a/app/src/main/res/layout/item_repo.xml +++ b/app/src/main/res/layout/item_repo.xml @@ -25,6 +25,8 @@ android:layout_width="40dp" android:layout_height="40dp" android:layout_marginEnd="10dp" + android:scaleX="0.8" + android:scaleY="0.8" android:layout_gravity="center_vertical" android:layout_marginStart="3dp" android:background="?android:attr/selectableItemBackground" @@ -32,4 +34,18 @@ app:tint="?attr/colorOnBackground" tools:ignore="ContentDescription" /> + + \ No newline at end of file diff --git a/app/src/main/res/values/strings.xml b/app/src/main/res/values/strings.xml index 504436d0c5..c695eda9bc 100644 --- a/app/src/main/res/values/strings.xml +++ b/app/src/main/res/values/strings.xml @@ -896,6 +896,7 @@ Non quae tempore quo provident laudantium qui illo dolor vel quia dolor et exerc Add Anime Repo Add Manga Repo + Add Novel Repo Edit repositories Remove repository? @@ -963,6 +964,7 @@ Non quae tempore quo provident laudantium qui illo dolor vel quia dolor et exerc Show only adult content in the explore page Add Anime Extensions from various sources Add Manga Extensions from various sources + Add Novel Extensions from various sources Change your default user agent Use the legacy installer to install extensions (For older android phones) Don\'t load icons of extensions on the extension page @@ -1089,7 +1091,7 @@ Non quae tempore quo provident laudantium qui illo dolor vel quia dolor et exerc Subtitle Stroke Bottom Margin Add Repository - A repository link should look like this: https://raw.githubusercontent.com/username/repo/branch/index.min.json + A repository link should look like this: https://raw.githubusercontent.com/username/repo/branch/index.min.json\nOr: username/repo/branch Current Repositories Warning: Extensions from the repository can run arbitrary code on your device. Only use repositories you trust. \n\nBy adding a repository, you agree to: \n\n1. Not use the app for viewing or distributing copyrighted content. \n2. Not use the app for any illegal activities. \n3. Not use the app for any activities that violate the terms of service of the content providers. \n\nThe app or it\'s maintainer are not affiliated in any way with extension providers. The developers are not responsible for any damages caused by the app. \n\nBy adding a repository, you agree to these terms. Privacy Policy