package eu.kanade.tachiyomi.extension.id.komikcast import eu.kanade.tachiyomi.lib.ratelimit.RateLimitInterceptor import eu.kanade.tachiyomi.multisrc.wpmangastream.WPMangaStream import eu.kanade.tachiyomi.network.GET import eu.kanade.tachiyomi.source.model.Filter import eu.kanade.tachiyomi.source.model.FilterList import eu.kanade.tachiyomi.source.model.Page import eu.kanade.tachiyomi.source.model.SChapter import eu.kanade.tachiyomi.source.model.SManga import okhttp3.HttpUrl import okhttp3.OkHttpClient import okhttp3.Request import org.jsoup.nodes.Document import org.jsoup.nodes.Element import java.util.concurrent.TimeUnit class KomikCast : WPMangaStream("Komik Cast", "https://komikcast.com", "id") { // Formerly "Komik Cast (WP Manga Stream)" override val id = 972717448578983812 private val rateLimitInterceptor = RateLimitInterceptor(4) override val client: OkHttpClient = network.cloudflareClient.newBuilder() .connectTimeout(10, TimeUnit.SECONDS) .readTimeout(30, TimeUnit.SECONDS) .addNetworkInterceptor(rateLimitInterceptor) .build() override fun popularMangaSelector() = "div.list-update_item" override fun popularMangaRequest(page: Int): Request { return GET("$baseUrl/daftar-komik/page/$page/?order=popular", headers) } override fun latestUpdatesRequest(page: Int): Request { return GET("$baseUrl/komik/page/$page/", headers) } override fun searchMangaRequest(page: Int, query: String, filters: FilterList): Request { val url = if (query.isNotBlank()) { val url = HttpUrl.parse("$baseUrl/page/$page")!!.newBuilder() val pattern = "\\s+".toRegex() val q = query.replace(pattern, "+") if (query.isNotEmpty()) { url.addQueryParameter("s", q) } else { url.addQueryParameter("s", "") } url.toString() } else { val url = HttpUrl.parse("$baseUrl/daftar-komik/page/$page")!!.newBuilder() var orderBy: String (if (filters.isEmpty()) getFilterList() else filters).forEach { filter -> when (filter) { is StatusFilter -> url.addQueryParameter("status", arrayOf("", "ongoing", "completed")[filter.state]) is GenreListFilter -> { val genreInclude = mutableListOf() filter.state.forEach { if (it.state == 1) { genreInclude.add(it.id) } } if (genreInclude.isNotEmpty()) { genreInclude.forEach { genre -> url.addQueryParameter("genre[]", genre) } } } is SortByFilter -> { orderBy = filter.toUriPart() url.addQueryParameter("order", orderBy) } } } url.toString() } return GET(url, headers) } override fun popularMangaFromElement(element: Element): SManga { val manga = SManga.create() manga.thumbnail_url = element.select("div.list-update_item-image img").imgAttr() element.select("a").first().let { manga.setUrlWithoutDomain(it.attr("href")) manga.title = it.attr("title") } return manga } override fun mangaDetailsParse(document: Document): SManga { return SManga.create().apply { document.select("div.komik_info").firstOrNull()?.let { infoElement -> genre = infoElement.select(".komik_info-content-genre a").joinToString { it.text() } status = parseStatus(infoElement.select("span:contains(Status:)").firstOrNull()?.ownText()) author = infoElement.select("span:contains(Author:)").firstOrNull()?.ownText() artist = infoElement.select("span:contains(Author:)").firstOrNull()?.ownText() description = infoElement.select("div.komik_info-description-sinopsis p").joinToString("\n") { it.text() } thumbnail_url = infoElement.select("div.komik_info-content-thumbnail img").imgAttr() // add series type(manga/manhwa/manhua/other) thinggy to genre document.select(seriesTypeSelector).firstOrNull()?.ownText()?.let { if (it.isEmpty().not() && genre!!.contains(it, true).not()) { genre += if (genre!!.isEmpty()) it else ", $it" } } // add alternative name to manga description document.select(altNameSelector).firstOrNull()?.ownText()?.let { if (it.isEmpty().not() && it !="N/A" && it != "-") { description += when { description!!.isEmpty() -> altName + it else -> "\n\n$altName" + it } } } } } } override val seriesTypeSelector = "span:contains(Type) a" override val altNameSelector = ".komik_info-content-native" override fun chapterListSelector() = "div.komik_info-chapters li" override fun chapterFromElement(element: Element): SChapter { val urlElement = element.select("a").first() val chapter = SChapter.create() chapter.setUrlWithoutDomain(urlElement.attr("href")) chapter.name = urlElement.text() chapter.date_upload = element.select(".chapter-link-time").firstOrNull()?.text()?.let { parseChapterDate(it) } ?: 0 return chapter } override fun pageListParse(document: Document): List { return document.select("div#chapter_body .main-reading-area img.size-full") .mapIndexed { i, img -> Page(i, "", img.attr("abs:Src")) } } override fun getFilterList() = FilterList( Filter.Header("NOTE: Ignored if using text search!"), Filter.Separator(), SortByFilter(), Filter.Separator(), StatusFilter(), Filter.Separator(), GenreListFilter(getGenreList()) ) }