add a Dark Science extension (#3435)
* add a Dark Science extension * add a work-around to not have the app think there’s missing chapters * applied some changes suggested in PR #3435 * rework chapter fetching to not block, fix URL invariant for SManga * cleanup --------- Co-authored-by: AwkwardPeak7 <48650614+AwkwardPeak7@users.noreply.github.com>
This commit is contained in:
		
							parent
							
								
									13474b911f
								
							
						
					
					
						commit
						cb1d65d02c
					
				
							
								
								
									
										7
									
								
								src/en/darkscience/build.gradle
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								src/en/darkscience/build.gradle
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,7 @@
 | 
			
		||||
ext {
 | 
			
		||||
    extName = 'Dark Science'
 | 
			
		||||
    extClass = '.DarkScience'
 | 
			
		||||
    extVersionCode = 1
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
apply from: "$rootDir/common.gradle"
 | 
			
		||||
							
								
								
									
										
											BIN
										
									
								
								src/en/darkscience/res/mipmap-hdpi/ic_launcher.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								src/en/darkscience/res/mipmap-hdpi/ic_launcher.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| 
		 After Width: | Height: | Size: 9.3 KiB  | 
							
								
								
									
										
											BIN
										
									
								
								src/en/darkscience/res/mipmap-mdpi/ic_launcher.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								src/en/darkscience/res/mipmap-mdpi/ic_launcher.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| 
		 After Width: | Height: | Size: 4.5 KiB  | 
							
								
								
									
										
											BIN
										
									
								
								src/en/darkscience/res/mipmap-xhdpi/ic_launcher.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								src/en/darkscience/res/mipmap-xhdpi/ic_launcher.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| 
		 After Width: | Height: | Size: 15 KiB  | 
							
								
								
									
										
											BIN
										
									
								
								src/en/darkscience/res/mipmap-xxhdpi/ic_launcher.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								src/en/darkscience/res/mipmap-xxhdpi/ic_launcher.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| 
		 After Width: | Height: | Size: 31 KiB  | 
							
								
								
									
										
											BIN
										
									
								
								src/en/darkscience/res/mipmap-xxxhdpi/ic_launcher.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								src/en/darkscience/res/mipmap-xxxhdpi/ic_launcher.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| 
		 After Width: | Height: | Size: 52 KiB  | 
@ -0,0 +1,130 @@
 | 
			
		||||
package eu.kanade.tachiyomi.extension.en.darkscience
 | 
			
		||||
 | 
			
		||||
import eu.kanade.tachiyomi.network.GET
 | 
			
		||||
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.online.HttpSource
 | 
			
		||||
import eu.kanade.tachiyomi.util.asJsoup
 | 
			
		||||
import okhttp3.Request
 | 
			
		||||
import okhttp3.Response
 | 
			
		||||
import org.jsoup.nodes.Document
 | 
			
		||||
import rx.Observable
 | 
			
		||||
import java.text.SimpleDateFormat
 | 
			
		||||
import java.util.Locale
 | 
			
		||||
 | 
			
		||||
class DarkScience : HttpSource() {
 | 
			
		||||
    override val name = "Dark Science"
 | 
			
		||||
    override val baseUrl = "https://dresdencodak.com"
 | 
			
		||||
    override val lang = "en"
 | 
			
		||||
    override val supportsLatest = false
 | 
			
		||||
 | 
			
		||||
    private fun initTheManga(manga: SManga): SManga = manga.apply {
 | 
			
		||||
        url = "/category/darkscience/"
 | 
			
		||||
        thumbnail_url = "https://dresdencodak.com/wp-content/uploads/2019/03/DC_CastIcon_Kimiko.png"
 | 
			
		||||
        title = name
 | 
			
		||||
        author = "Sen (A. Senna Diaz)"
 | 
			
		||||
        artist = "Sen (A. Senna Diaz)"
 | 
			
		||||
        description = """Scientist Kimiko Ross has a problem:
 | 
			
		||||
        | her money’s gone and a bank exploded her house. With no place
 | 
			
		||||
        | else to go, she travels to Nephilopolis, the city of giants –
 | 
			
		||||
        | built from the ruins of an ancient war and a fading memory of
 | 
			
		||||
        | tomorrow.\n Follow our cyborg hero as she attempts to survive the
 | 
			
		||||
        | bureaucratic behemoth with a little “help” from her “friends.”
 | 
			
		||||
        | And what exactly is Dark Science anyway?\nSupport the comic on
 | 
			
		||||
        | Patreon: https://www.patreon.com/dresdencodak
 | 
			
		||||
        """.trimMargin()
 | 
			
		||||
        genre = "Science Fiction, Mystery, LGBT+"
 | 
			
		||||
        status = SManga.ONGOING
 | 
			
		||||
        initialized = true
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    override fun fetchPopularManga(page: Int): Observable<MangasPage> = Observable.just(
 | 
			
		||||
        MangasPage(
 | 
			
		||||
            listOf(initTheManga(SManga.create())),
 | 
			
		||||
            false,
 | 
			
		||||
        ),
 | 
			
		||||
    )
 | 
			
		||||
 | 
			
		||||
    override fun fetchMangaDetails(manga: SManga): Observable<SManga> =
 | 
			
		||||
        Observable.just(initTheManga(manga))
 | 
			
		||||
 | 
			
		||||
    override fun fetchChapterList(manga: SManga): Observable<List<SChapter>> {
 | 
			
		||||
        val chapters = mutableListOf<SChapter>()
 | 
			
		||||
 | 
			
		||||
        var archivePage: Document? = client.newCall(GET(baseUrl + manga.url, headers))
 | 
			
		||||
            .execute().asJsoup()
 | 
			
		||||
 | 
			
		||||
        var chLast = 0.0F
 | 
			
		||||
 | 
			
		||||
        while (archivePage != null) {
 | 
			
		||||
            val nextArchivePageUrl = archivePage.selectFirst("""#nav-below .nav-previous > a""")
 | 
			
		||||
                ?.attr("href")
 | 
			
		||||
            val nextArchivePage = if (nextArchivePageUrl != null) {
 | 
			
		||||
                client.newCall(GET(nextArchivePageUrl, headers)).execute().asJsoup()
 | 
			
		||||
            } else { null }
 | 
			
		||||
 | 
			
		||||
            archivePage.select("""#content article header > h2 > a""").forEach {
 | 
			
		||||
                val chTitle = it.text()
 | 
			
		||||
                val chLink = it.attr("href")
 | 
			
		||||
                val chNum = chapterNumberRegex.find(chTitle)
 | 
			
		||||
                    ?.groupValues?.getOrNull(1)?.toFloatOrNull()
 | 
			
		||||
                    ?: (chLast + 0.01F)
 | 
			
		||||
 | 
			
		||||
                chapters.add(
 | 
			
		||||
                    SChapter.create().apply {
 | 
			
		||||
                        name = chTitle
 | 
			
		||||
                        chapter_number = chNum
 | 
			
		||||
                        date_upload = getDate(chLink)
 | 
			
		||||
                        setUrlWithoutDomain(chLink)
 | 
			
		||||
                    },
 | 
			
		||||
                )
 | 
			
		||||
 | 
			
		||||
                // This is a hack to make the app not think there’s missing chapters after
 | 
			
		||||
                // a title page.
 | 
			
		||||
                chLast = chNum
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            archivePage = nextArchivePage
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        return Observable.just(chapters)
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private fun getDate(url: String): Long {
 | 
			
		||||
        return try {
 | 
			
		||||
            dateFormat.parse(
 | 
			
		||||
                chapterDateRegex.find(url)!!.groupValues[1],
 | 
			
		||||
            )!!.time
 | 
			
		||||
        } catch (_: Exception) {
 | 
			
		||||
            0L
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    override fun pageListParse(response: Response): List<Page> =
 | 
			
		||||
        listOf(
 | 
			
		||||
            Page(
 | 
			
		||||
                0,
 | 
			
		||||
                imageUrl = response.asJsoup()
 | 
			
		||||
                    .selectFirst("article.post img.aligncenter")!!
 | 
			
		||||
                    .attr("src"),
 | 
			
		||||
            ),
 | 
			
		||||
        )
 | 
			
		||||
 | 
			
		||||
    override fun imageUrlParse(response: Response): String = throw UnsupportedOperationException()
 | 
			
		||||
    override fun popularMangaRequest(page: Int): Request = throw UnsupportedOperationException()
 | 
			
		||||
    override fun popularMangaParse(response: Response): MangasPage = throw UnsupportedOperationException()
 | 
			
		||||
    override fun mangaDetailsParse(response: Response): SManga = throw UnsupportedOperationException()
 | 
			
		||||
    override fun fetchSearchManga(page: Int, query: String, filters: FilterList): Observable<MangasPage> = throw UnsupportedOperationException()
 | 
			
		||||
    override fun searchMangaRequest(page: Int, query: String, filters: FilterList): Request = throw UnsupportedOperationException()
 | 
			
		||||
    override fun searchMangaParse(response: Response): MangasPage = throw UnsupportedOperationException()
 | 
			
		||||
    override fun latestUpdatesRequest(page: Int): Request = throw UnsupportedOperationException()
 | 
			
		||||
    override fun latestUpdatesParse(response: Response): MangasPage = throw UnsupportedOperationException()
 | 
			
		||||
    override fun chapterListParse(response: Response): List<SChapter> = throw UnsupportedOperationException()
 | 
			
		||||
 | 
			
		||||
    private val dateFormat = SimpleDateFormat("yyyy/MM/dd", Locale.US)
 | 
			
		||||
    private val chapterDateRegex = """/(\d\d\d\d/\d\d/\d\d)/""".toRegex()
 | 
			
		||||
    private val chapterNumberRegex = """Dark Science #(\d+)""".toRegex()
 | 
			
		||||
}
 | 
			
		||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user