New source: en/Cutie Comics (#948)
CI / Prepare job (push) Successful in 3s Details
CI / Build multisrc modules (push) Successful in 6m6s Details
CI / Build individual modules (push) Successful in 39s Details
CI / Publish repo (push) Successful in 43s Details

* feat: Create Cutie Comics base

* feat: Implement popular manga page

* feat: Implement search manga page

* chore: Add isNsfw flag

* feat: Implement manga details page

* fix: Fix URL intent handler

* feat: Implement single-chapter "list"

* feat: Parse page list

* chore: Add source icon

* fix: Fix crash in URL intent handler

* refactor: Follow chapter name convention
This commit is contained in:
Claudemirovsky 2024-02-03 14:59:16 -03:00 committed by Draff
parent 742027746e
commit 4856f0b89e
9 changed files with 221 additions and 0 deletions

View File

@ -0,0 +1,22 @@
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android">
<application>
<activity
android:name=".en.cutiecomics.CutieComicsUrlActivity"
android:excludeFromRecents="true"
android:exported="true"
android:theme="@android:style/Theme.NoDisplay">
<intent-filter>
<action android:name="android.intent.action.VIEW" />
<category android:name="android.intent.category.DEFAULT" />
<category android:name="android.intent.category.BROWSABLE" />
<data
android:host="cutiecomics.com"
android:pathPattern="/..*"
android:scheme="https" />
</intent-filter>
</activity>
</application>
</manifest>

View File

@ -0,0 +1,8 @@
ext {
extName = 'Cutie Comics'
extClass = '.CutieComics'
extVersionCode = 1
isNsfw = true
}
apply from: "$rootDir/common.gradle"

Binary file not shown.

After

Width:  |  Height:  |  Size: 5.6 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 3.1 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 8.4 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 16 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 23 KiB

View File

@ -0,0 +1,150 @@
package eu.kanade.tachiyomi.extension.en.cutiecomics
import eu.kanade.tachiyomi.network.GET
import eu.kanade.tachiyomi.network.POST
import eu.kanade.tachiyomi.network.asObservableSuccess
import eu.kanade.tachiyomi.network.interceptor.rateLimitHost
import eu.kanade.tachiyomi.source.model.FilterList
import eu.kanade.tachiyomi.source.model.MangasPage
import eu.kanade.tachiyomi.source.model.Page
import eu.kanade.tachiyomi.source.model.SChapter
import eu.kanade.tachiyomi.source.model.SManga
import eu.kanade.tachiyomi.source.model.UpdateStrategy
import eu.kanade.tachiyomi.source.online.ParsedHttpSource
import eu.kanade.tachiyomi.util.asJsoup
import okhttp3.FormBody
import okhttp3.HttpUrl.Companion.toHttpUrl
import okhttp3.Request
import okhttp3.Response
import org.jsoup.nodes.Document
import org.jsoup.nodes.Element
import rx.Observable
class CutieComics : ParsedHttpSource() {
override val name = "Cutie Comics"
override val baseUrl = "https://cutiecomics.com"
override val lang = "en"
override val supportsLatest = false
override val client = network.client.newBuilder()
.rateLimitHost(baseUrl.toHttpUrl(), 2)
.build()
// ============================== Popular ===============================
override fun popularMangaRequest(page: Int) = GET("$baseUrl/page/$page", headers)
override fun popularMangaSelector() = "#dle-content > div.w25"
override fun popularMangaFromElement(element: Element) = SManga.create().apply {
with(element.selectFirst("strong.field-content > a")!!) {
title = ownText()
setUrlWithoutDomain(attr("href"))
}
thumbnail_url = element.selectFirst("a > img")?.absUrl("src")
}
override fun popularMangaNextPageSelector() = ".navigation > a > i.fa-angle-right"
// =============================== Latest ===============================
override fun latestUpdatesRequest(page: Int): Request {
throw UnsupportedOperationException()
}
override fun latestUpdatesSelector(): String {
throw UnsupportedOperationException()
}
override fun latestUpdatesFromElement(element: Element): SManga {
throw UnsupportedOperationException()
}
override fun latestUpdatesNextPageSelector(): String? {
throw UnsupportedOperationException()
}
// =============================== Search ===============================
override fun fetchSearchManga(page: Int, query: String, filters: FilterList): Observable<MangasPage> {
return if (query.startsWith(PREFIX_SEARCH)) { // URL intent handler
val id = query.removePrefix(PREFIX_SEARCH)
client.newCall(GET("$baseUrl/$id"))
.asObservableSuccess()
.map(::searchMangaByIdParse)
} else {
super.fetchSearchManga(page, query, filters)
}
}
private fun searchMangaByIdParse(response: Response): MangasPage {
val doc = response.asJsoup()
val details = mangaDetailsParse(doc)
.apply { setUrlWithoutDomain(doc.location()) }
return MangasPage(listOf(details), false)
}
override fun searchMangaRequest(page: Int, query: String, filters: FilterList): Request {
require(query.isNotBlank() && query.length >= 4) { "Invalid search! It should have at least 4 non-blank characters." }
val body = FormBody.Builder()
.add("do", "search")
.add("subaction", "search")
.add("full_search", "0")
.add("search_start", "$page")
.add("result_from", "${(page - 1) * 20 + 1}")
.add("story", query)
.build()
return POST("$baseUrl/index.php?do=search", headers, body)
}
override fun searchMangaSelector() = popularMangaSelector()
override fun searchMangaFromElement(element: Element) = popularMangaFromElement(element)
override fun searchMangaNextPageSelector() = popularMangaNextPageSelector()
// =========================== Manga Details ============================
override fun mangaDetailsParse(document: Document) = SManga.create().apply {
status = SManga.COMPLETED
update_strategy = UpdateStrategy.ONLY_FETCH_ONCE
title = document.selectFirst("h1#page-title")!!.text()
thumbnail_url = document.selectFirst("div.galery > img")?.absUrl("src")
genre = document.select("h3.field-label ~ span").joinToString { it.text() }
}
// ============================== Chapters ==============================
override fun fetchChapterList(manga: SManga): Observable<List<SChapter>> {
val chapter = SChapter.create().apply {
url = manga.url
chapter_number = 1F
name = "Chapter"
}
return Observable.just(listOf(chapter))
}
override fun chapterListSelector(): String {
throw UnsupportedOperationException()
}
override fun chapterFromElement(element: Element): SChapter {
throw UnsupportedOperationException()
}
// =============================== Pages ================================
override fun pageListParse(document: Document): List<Page> {
return document.select("div.galery > img").mapIndexed { index, item ->
Page(index, imageUrl = item.absUrl("src"))
}
}
override fun imageUrlParse(document: Document): String {
throw UnsupportedOperationException()
}
companion object {
const val PREFIX_SEARCH = "id:"
}
}

View File

@ -0,0 +1,41 @@
package eu.kanade.tachiyomi.extension.en.cutiecomics
import android.app.Activity
import android.content.ActivityNotFoundException
import android.content.Intent
import android.os.Bundle
import android.util.Log
import kotlin.system.exitProcess
/**
* Springboard that accepts https://cutiecomics.com/<item> intents
* and redirects them to the main Tachiyomi process.
*/
class CutieComicsUrlActivity : Activity() {
private val tag = javaClass.simpleName
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
val pathSegments = intent?.data?.pathSegments
if (pathSegments != null && pathSegments.isNotEmpty()) {
val item = pathSegments[0]
val mainIntent = Intent().apply {
action = "eu.kanade.tachiyomi.SEARCH"
putExtra("query", "${CutieComics.PREFIX_SEARCH}$item")
putExtra("filter", packageName)
}
try {
startActivity(mainIntent)
} catch (e: ActivityNotFoundException) {
Log.e(tag, e.toString())
}
} else {
Log.e(tag, "could not parse uri from intent $intent")
}
finish()
exitProcess(0)
}
}