Add MangaShow.Me (Korean source) (#775)
Add MangaShow.Me (Korean source)
This commit is contained in:
		
							parent
							
								
									4599a37c4a
								
							
						
					
					
						commit
						8983e8e688
					
				
							
								
								
									
										16
									
								
								src/ko/mangashowme/build.gradle
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								src/ko/mangashowme/build.gradle
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,16 @@
 | 
			
		||||
apply plugin: 'com.android.application'
 | 
			
		||||
apply plugin: 'kotlin-android'
 | 
			
		||||
 | 
			
		||||
ext {
 | 
			
		||||
    appName = 'Tachiyomi: MangaShow.Me'
 | 
			
		||||
    pkgNameSuffix = 'ko.mangashowme'
 | 
			
		||||
    extClass = '.MangaShowMe'
 | 
			
		||||
    extVersionCode = 1
 | 
			
		||||
    libVersion = '1.2'
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
dependencies {
 | 
			
		||||
    compileOnly project(':duktape-stub')
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
apply from: "$rootDir/common.gradle"
 | 
			
		||||
@ -0,0 +1,193 @@
 | 
			
		||||
package eu.kanade.tachiyomi.extension.ko.mangashowme
 | 
			
		||||
 | 
			
		||||
import eu.kanade.tachiyomi.network.GET
 | 
			
		||||
import eu.kanade.tachiyomi.source.model.Filter
 | 
			
		||||
import eu.kanade.tachiyomi.source.model.FilterList
 | 
			
		||||
import okhttp3.HttpUrl
 | 
			
		||||
import okhttp3.Request
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
// TODO: Completely Implement/Update Filters(Genre/Artist).
 | 
			
		||||
// private class TextField(name: String, val key: String) : Filter.Text(name)
 | 
			
		||||
private class SearchCheckBox(val id: Int, name: String) : Filter.CheckBox(name)
 | 
			
		||||
 | 
			
		||||
private class SearchFieldMatch : Filter.Select<String>("Search Match", arrayOf("Not Set", "AND", "OR"))
 | 
			
		||||
private class SearchTagMatch : Filter.Select<String>("Tag Match", arrayOf("AND", "OR"))
 | 
			
		||||
private class SearchGenresList(genres: List<SearchCheckBox>) : Filter.Group<SearchCheckBox>("Genres", genres)
 | 
			
		||||
private class SearchNamingList(naming: List<SearchCheckBox>) : Filter.Group<SearchCheckBox>("Naming", naming)
 | 
			
		||||
private class SearchStatusList(status: List<SearchCheckBox>) : Filter.Group<SearchCheckBox>("Status", status)
 | 
			
		||||
 | 
			
		||||
private fun searchNaming() = listOf(
 | 
			
		||||
        SearchCheckBox(0, "ㄱ"),
 | 
			
		||||
        SearchCheckBox(1, "ㄲ"),
 | 
			
		||||
        SearchCheckBox(2, "ㄴ"),
 | 
			
		||||
        SearchCheckBox(3, "ㄷ"),
 | 
			
		||||
        SearchCheckBox(4, "ㄸ"),
 | 
			
		||||
        SearchCheckBox(5, "ㄹ"),
 | 
			
		||||
        SearchCheckBox(6, "ㅁ"),
 | 
			
		||||
        SearchCheckBox(7, "ㅂ"),
 | 
			
		||||
        SearchCheckBox(8, "ㅃ"),
 | 
			
		||||
        SearchCheckBox(9, "ㅅ"),
 | 
			
		||||
        SearchCheckBox(10, "ㅆ"),
 | 
			
		||||
        SearchCheckBox(11, "ㅇ"),
 | 
			
		||||
        SearchCheckBox(12, "ㅈ"),
 | 
			
		||||
        SearchCheckBox(13, "ㅉ"),
 | 
			
		||||
        SearchCheckBox(14, "ㅊ"),
 | 
			
		||||
        SearchCheckBox(15, "ㅋ"),
 | 
			
		||||
        SearchCheckBox(16, "ㅌ"),
 | 
			
		||||
        SearchCheckBox(17, "ㅍ"),
 | 
			
		||||
        SearchCheckBox(18, "ㅎ"),
 | 
			
		||||
        SearchCheckBox(19, "A-Z"),
 | 
			
		||||
        SearchCheckBox(20, "0-9")
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
private fun searchStatus() = listOf(
 | 
			
		||||
        SearchCheckBox(0, "미분류"),
 | 
			
		||||
        SearchCheckBox(1, "주간"),
 | 
			
		||||
        SearchCheckBox(2, "격주"),
 | 
			
		||||
        SearchCheckBox(3, "월간"),
 | 
			
		||||
        SearchCheckBox(4, "격월/비정기"),
 | 
			
		||||
        SearchCheckBox(5, "단편"),
 | 
			
		||||
        SearchCheckBox(6, "단행본"),
 | 
			
		||||
        SearchCheckBox(7, "완결")
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
private fun searchGenres() = listOf(
 | 
			
		||||
        SearchCheckBox(0, "17"),
 | 
			
		||||
        SearchCheckBox(0, "BL"),
 | 
			
		||||
        SearchCheckBox(0, "SF"),
 | 
			
		||||
        SearchCheckBox(0, "TS"),
 | 
			
		||||
        SearchCheckBox(0, "개그"),
 | 
			
		||||
        SearchCheckBox(0, "게임"),
 | 
			
		||||
        SearchCheckBox(0, "공포"),
 | 
			
		||||
        SearchCheckBox(0, "도박"),
 | 
			
		||||
        SearchCheckBox(0, "드라마"),
 | 
			
		||||
        SearchCheckBox(0, "라노벨"),
 | 
			
		||||
        SearchCheckBox(0, "러브코미디"),
 | 
			
		||||
        SearchCheckBox(0, "로맨스"),
 | 
			
		||||
        SearchCheckBox(0, "먹방"),
 | 
			
		||||
        SearchCheckBox(0, "백합"),
 | 
			
		||||
        SearchCheckBox(0, "붕탁"),
 | 
			
		||||
        SearchCheckBox(0, "순정"),
 | 
			
		||||
        SearchCheckBox(0, "스릴러"),
 | 
			
		||||
        SearchCheckBox(0, "스포츠"),
 | 
			
		||||
        SearchCheckBox(0, "시대"),
 | 
			
		||||
        SearchCheckBox(0, "애니화"),
 | 
			
		||||
        SearchCheckBox(0, "액션"),
 | 
			
		||||
        SearchCheckBox(0, "역사"),
 | 
			
		||||
        SearchCheckBox(0, "요리"),
 | 
			
		||||
        SearchCheckBox(0, "음악"),
 | 
			
		||||
        SearchCheckBox(0, "이세계"),
 | 
			
		||||
        SearchCheckBox(0, "일상"),
 | 
			
		||||
        SearchCheckBox(0, "전생"),
 | 
			
		||||
        SearchCheckBox(0, "추리"),
 | 
			
		||||
        SearchCheckBox(0, "판타지"),
 | 
			
		||||
        SearchCheckBox(0, "학원"),
 | 
			
		||||
        SearchCheckBox(0, "호러")
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
fun getFilters() = FilterList(
 | 
			
		||||
        SearchNamingList(searchNaming()),
 | 
			
		||||
        SearchStatusList(searchStatus()),
 | 
			
		||||
        SearchGenresList(searchGenres()),
 | 
			
		||||
        Filter.Separator(),
 | 
			
		||||
        SearchFieldMatch(),
 | 
			
		||||
        SearchTagMatch()
 | 
			
		||||
        //Filter.Separator(),
 | 
			
		||||
        //TextField("Author/Artist (Accurate full name)", "author")
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
fun searchComplexFilterMangaRequestBuilder(baseUrl: String, page: Int, query: String, filters: FilterList): Request {
 | 
			
		||||
    // normal search function.
 | 
			
		||||
    fun normalSearch(state: Int = 0): Request {
 | 
			
		||||
        val url = HttpUrl.parse("$baseUrl/bbs/search.php?url=$baseUrl/bbs/search.php")!!.newBuilder()
 | 
			
		||||
 | 
			
		||||
        if (state > 0) {
 | 
			
		||||
            url.addQueryParameter("sop", arrayOf("and", "or")[state - 1])
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        url.addQueryParameter("stx", query)
 | 
			
		||||
 | 
			
		||||
        if (page > 1) {
 | 
			
		||||
            url.addQueryParameter("page", "${page - 1}")
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        return GET(url.toString())
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    val nameFilter = mutableListOf<Int>()
 | 
			
		||||
    val statusFilter = mutableListOf<Int>()
 | 
			
		||||
    val genresFilter = mutableListOf<String>()
 | 
			
		||||
    var matchFieldFilter = 0
 | 
			
		||||
    var matchTagFilter = 1
 | 
			
		||||
 | 
			
		||||
    filters.forEach { filter ->
 | 
			
		||||
        when (filter) {
 | 
			
		||||
            is SearchFieldMatch -> {
 | 
			
		||||
                matchFieldFilter = filter.state
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    filters.forEach { filter ->
 | 
			
		||||
        when (filter) {
 | 
			
		||||
            is SearchTagMatch -> {
 | 
			
		||||
                if (filter.state > 0) {
 | 
			
		||||
                    matchTagFilter = filter.state + 1
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            is SearchNamingList -> {
 | 
			
		||||
                filter.state.forEach {
 | 
			
		||||
                    if (it.state) {
 | 
			
		||||
                        nameFilter.add(it.id)
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            is SearchStatusList -> {
 | 
			
		||||
                filter.state.forEach {
 | 
			
		||||
                    if (it.state) {
 | 
			
		||||
                        statusFilter.add(it.id)
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            is SearchGenresList -> {
 | 
			
		||||
                filter.state.forEach {
 | 
			
		||||
                    if (it.state) {
 | 
			
		||||
                        genresFilter.add(it.name)
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
//            is TextField -> {
 | 
			
		||||
//                if (type == 4 && filter.key == "author") {
 | 
			
		||||
//                    if (filter.key.length > 1) {
 | 
			
		||||
//                        return GET("$baseUrl/bbs/page.php?hid=manga_list&sfl=4&stx=${filter.state}")
 | 
			
		||||
//                    }
 | 
			
		||||
//                }
 | 
			
		||||
//            }
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // If Query is over 2 length, just go to normal search
 | 
			
		||||
    if (query.length > 1) {
 | 
			
		||||
        return normalSearch(matchFieldFilter)
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    if (nameFilter.isEmpty() && statusFilter.isEmpty() && genresFilter.isEmpty()) {
 | 
			
		||||
        return GET("$baseUrl/bbs/page.php?hid=manga_list")
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    val url = HttpUrl.parse("$baseUrl/bbs/page.php?hid=manga_list")!!.newBuilder()
 | 
			
		||||
    url.addQueryParameter("search_type", matchTagFilter.toString())
 | 
			
		||||
    url.addQueryParameter("_1", nameFilter.joinToString(","))
 | 
			
		||||
    url.addQueryParameter("_2", statusFilter.joinToString(","))
 | 
			
		||||
    url.addQueryParameter("_3", genresFilter.joinToString(","))
 | 
			
		||||
    if (page > 1) {
 | 
			
		||||
        url.addQueryParameter("page", "${page - 1}")
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    return GET(url.toString())
 | 
			
		||||
}
 | 
			
		||||
@ -0,0 +1,166 @@
 | 
			
		||||
package eu.kanade.tachiyomi.extension.ko.mangashowme
 | 
			
		||||
 | 
			
		||||
import android.graphics.Bitmap
 | 
			
		||||
import android.graphics.BitmapFactory
 | 
			
		||||
import android.graphics.Canvas
 | 
			
		||||
import android.graphics.Rect
 | 
			
		||||
import eu.kanade.tachiyomi.network.GET
 | 
			
		||||
import okhttp3.*
 | 
			
		||||
import java.io.ByteArrayOutputStream
 | 
			
		||||
import java.io.InputStream
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 * `v1` means url padding of image host.
 | 
			
		||||
 *  It's not need now, but it remains in this code for sometime.
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
internal class ImageDecoder(private val version: String, scripts: String) {
 | 
			
		||||
    private  val cnt = substringBetween(scripts, "var view_cnt = ", ";")
 | 
			
		||||
            .toIntOrNull() ?: 0
 | 
			
		||||
    private val chapter = substringBetween(scripts, "var chapter = ", ";")
 | 
			
		||||
            .toIntOrNull() ?: 0
 | 
			
		||||
 | 
			
		||||
    fun request(url: String): String {
 | 
			
		||||
        return when (version) {
 | 
			
		||||
            "v1" -> decodeVersion1ImageUrl(cnt, chapter, url)
 | 
			
		||||
            else -> url
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private fun decodeVersion1ImageUrl(cnt: Int, chapter: Int, url: String): String {
 | 
			
		||||
        return HttpUrl.parse(url)!!.newBuilder()
 | 
			
		||||
                .addQueryParameter("cnt", cnt.toString())
 | 
			
		||||
                .addQueryParameter("ch", chapter.toString())
 | 
			
		||||
                .addQueryParameter("ver", "v1")
 | 
			
		||||
                .addQueryParameter("type", "ImageDecodeRequest")
 | 
			
		||||
                .build()!!.toString()
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
internal class ImageDecoderInterceptor : Interceptor {
 | 
			
		||||
    override fun intercept(chain: Interceptor.Chain): Response {
 | 
			
		||||
        val req = chain.request()
 | 
			
		||||
        val url = req.url().toString()
 | 
			
		||||
        return if (url.contains("ImageDecodeRequest")) {
 | 
			
		||||
            try {
 | 
			
		||||
                val reqUrl = HttpUrl.parse(url)!!
 | 
			
		||||
 | 
			
		||||
                val viewCnt = reqUrl.queryParameter("cnt")!!
 | 
			
		||||
                val version = reqUrl.queryParameter("ver")!!
 | 
			
		||||
                val chapter = reqUrl.queryParameter("ch")!!
 | 
			
		||||
                val imageUrl = url.split("?").first()
 | 
			
		||||
 | 
			
		||||
                val response = chain.proceed(GET(imageUrl))
 | 
			
		||||
                val res = response.body()!!.byteStream().use {
 | 
			
		||||
                    decodeImageRequest(version, chapter, viewCnt, it)
 | 
			
		||||
                }
 | 
			
		||||
 | 
			
		||||
                val rb = ResponseBody.create(MediaType.parse("image/png"), res)
 | 
			
		||||
                response.newBuilder().body(rb).build()
 | 
			
		||||
            } catch (e: Exception) {
 | 
			
		||||
                e.printStackTrace()
 | 
			
		||||
                chain.proceed(req)
 | 
			
		||||
            }
 | 
			
		||||
        } else {
 | 
			
		||||
            chain.proceed(req)
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /*
 | 
			
		||||
     * `decodeV1ImageNative` is modified version of
 | 
			
		||||
     *  https://github.com/junheah/MangaViewAndroid/blob/master/app/src/main/java/ml/melun/mangaview/Downloader.java#L213-L245
 | 
			
		||||
     *
 | 
			
		||||
     * MIT License
 | 
			
		||||
     *
 | 
			
		||||
     * Copyright (c) 2019 junheah
 | 
			
		||||
     */
 | 
			
		||||
    private fun decodeV1ImageNative(input: Bitmap, chapter: Int, view_cnt: Int, half: Int = 0, CX: Int = MangaShowMe.V1_CX, CY: Int = MangaShowMe.V1_CY): Bitmap {
 | 
			
		||||
        if (view_cnt == 0) return input
 | 
			
		||||
        val viewCnt = view_cnt / 10
 | 
			
		||||
 | 
			
		||||
        //decode image
 | 
			
		||||
        val order = Array(CX * CY) { IntArray(2) }
 | 
			
		||||
        val oSize = order.size - 1
 | 
			
		||||
 | 
			
		||||
        for (i in 0..oSize) {
 | 
			
		||||
            order[i][0] = i
 | 
			
		||||
            order[i][1] = decoderRandom(chapter, viewCnt, i)
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        java.util.Arrays.sort(order) { a, b -> java.lang.Double.compare(a[1].toDouble(), b[1].toDouble()) }
 | 
			
		||||
 | 
			
		||||
        //create new bitmap
 | 
			
		||||
        val outputWidth = if (half == 0) input.width else input.width / 2
 | 
			
		||||
        val output = Bitmap.createBitmap(outputWidth, input.height, Bitmap.Config.ARGB_8888)
 | 
			
		||||
        val canvas = Canvas(output)
 | 
			
		||||
 | 
			
		||||
        val rowWidth = input.width / CX
 | 
			
		||||
        val rowHeight = input.height / CY
 | 
			
		||||
 | 
			
		||||
        for (i in 0..oSize) {
 | 
			
		||||
            val o = order[i]
 | 
			
		||||
            val ox = i % CX
 | 
			
		||||
            val oy = i / CX
 | 
			
		||||
            val tx = o[0] % CX
 | 
			
		||||
            val ty = o[0] / CX
 | 
			
		||||
            val sx = if (half == 2) -input.width / 2 else 0
 | 
			
		||||
 | 
			
		||||
            val srcX = ox * rowWidth
 | 
			
		||||
            val srcY = oy * rowHeight
 | 
			
		||||
            val destX = (tx * rowWidth) + sx
 | 
			
		||||
            val destY = ty * rowHeight
 | 
			
		||||
 | 
			
		||||
            canvas.drawBitmap(input,
 | 
			
		||||
                    Rect(srcX, srcY, srcX + rowWidth, srcY + rowHeight),
 | 
			
		||||
                    Rect(destX, destY, destX + rowWidth, destY + rowHeight),
 | 
			
		||||
                    null)
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        return output
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /*
 | 
			
		||||
     * `decodeRandom` is modified version of
 | 
			
		||||
     *  https://github.com/junheah/MangaViewAndroid/blob/master/app/src/main/java/ml/melun/mangaview/Downloader.java#L213-L245
 | 
			
		||||
     *
 | 
			
		||||
     * MIT License
 | 
			
		||||
     *
 | 
			
		||||
     * Copyright (c) 2019 junheah
 | 
			
		||||
     */
 | 
			
		||||
    private fun decoderRandom(chapter: Int, view_cnt: Int, index: Int): Int {
 | 
			
		||||
        if (chapter < 554714) {
 | 
			
		||||
            val x = 10000 * Math.sin((view_cnt + index).toDouble())
 | 
			
		||||
            return Math.floor(100000 * (x - Math.floor(x))).toInt()
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        val seed = view_cnt + index + 1
 | 
			
		||||
        val t = 100 * Math.sin((10 * seed).toDouble())
 | 
			
		||||
        val n = 1000 * Math.cos((13 * seed).toDouble())
 | 
			
		||||
        val a = 10000 * Math.tan((14 * seed).toDouble())
 | 
			
		||||
 | 
			
		||||
        return (Math.floor(100 * (t - Math.floor(t))) +
 | 
			
		||||
                Math.floor(1000 * (n - Math.floor(n))) +
 | 
			
		||||
                Math.floor(10000 * (a - Math.floor(a)))).toInt()
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private fun decodeImageRequest(version: String, chapter: String, view_cnt: String, img: InputStream): ByteArray {
 | 
			
		||||
        return when (version) {
 | 
			
		||||
            "v1" -> decodeV1Image(chapter, view_cnt, img)
 | 
			
		||||
            else -> img.readBytes()
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private fun decodeV1Image(chapter: String, view_cnt: String, img: InputStream): ByteArray {
 | 
			
		||||
        val decoded = BitmapFactory.decodeStream(img)
 | 
			
		||||
        val result = decodeV1ImageNative(decoded, chapter.toInt(), view_cnt.toInt())
 | 
			
		||||
 | 
			
		||||
        val output = ByteArrayOutputStream()
 | 
			
		||||
        result.compress(Bitmap.CompressFormat.PNG, 100, output)
 | 
			
		||||
        return output.toByteArray()
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
private fun substringBetween(target: String, prefix: String, suffix: String): String = {
 | 
			
		||||
    target.substringAfter(prefix).substringBefore(suffix)
 | 
			
		||||
}()
 | 
			
		||||
@ -0,0 +1,238 @@
 | 
			
		||||
package eu.kanade.tachiyomi.extension.ko.mangashowme
 | 
			
		||||
 | 
			
		||||
import android.annotation.SuppressLint
 | 
			
		||||
import eu.kanade.tachiyomi.network.GET
 | 
			
		||||
import eu.kanade.tachiyomi.source.model.*
 | 
			
		||||
import eu.kanade.tachiyomi.source.online.ParsedHttpSource
 | 
			
		||||
import eu.kanade.tachiyomi.util.asJsoup
 | 
			
		||||
import okhttp3.OkHttpClient
 | 
			
		||||
import okhttp3.Request
 | 
			
		||||
import okhttp3.Response
 | 
			
		||||
import org.json.JSONArray
 | 
			
		||||
import org.jsoup.nodes.Document
 | 
			
		||||
import org.jsoup.nodes.Element
 | 
			
		||||
import org.jsoup.select.Elements
 | 
			
		||||
import java.text.SimpleDateFormat
 | 
			
		||||
import java.util.*
 | 
			
		||||
import java.util.concurrent.TimeUnit
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * MangaShow.Me Source
 | 
			
		||||
 *
 | 
			
		||||
 * PS. There's no Popular section. It's just a list of manga. Also not latest updates.
 | 
			
		||||
 *     `manga_list` returns latest 'added' manga. not a chapter updates.
 | 
			
		||||
 **/
 | 
			
		||||
class MangaShowMe : ParsedHttpSource() {
 | 
			
		||||
    override val name = "MangaShow.Me"
 | 
			
		||||
    override val baseUrl = "https://mangashow.me"
 | 
			
		||||
    override val lang: String = "ko"
 | 
			
		||||
 | 
			
		||||
    // Latest updates currently returns duplicate manga as it separates manga into chapters
 | 
			
		||||
    override val supportsLatest = false
 | 
			
		||||
    override val client: OkHttpClient = network.cloudflareClient.newBuilder()
 | 
			
		||||
            .connectTimeout(10, TimeUnit.SECONDS)
 | 
			
		||||
            .readTimeout(30, TimeUnit.SECONDS)
 | 
			
		||||
            .addInterceptor(ImageDecoderInterceptor())
 | 
			
		||||
            .addInterceptor { chain ->
 | 
			
		||||
                val response = chain.proceed(chain.request())
 | 
			
		||||
                if (response.code() == 503) {
 | 
			
		||||
                    val body = response.body().toString()
 | 
			
		||||
                    if (body.contains("console.log(\"503\")") || body.contains("console.log('503')"))
 | 
			
		||||
                        throw Exception("Try again.\nServer returns 503 Service Unavailable.")
 | 
			
		||||
                }
 | 
			
		||||
                response
 | 
			
		||||
            }
 | 
			
		||||
            .build()!!
 | 
			
		||||
 | 
			
		||||
    //override fun popularMangaSelector() = "div.basic-post-gallery > div >  div.post-row"
 | 
			
		||||
    override fun popularMangaSelector() = "div.manga-list-gallery > div > div.post-row"
 | 
			
		||||
 | 
			
		||||
    override fun popularMangaFromElement(element: Element): SManga {
 | 
			
		||||
        val linkElement = element.select("a")
 | 
			
		||||
        val titleElement = element.select(".manga-subject > a").first()
 | 
			
		||||
 | 
			
		||||
        val manga = SManga.create()
 | 
			
		||||
        manga.url = urlTitleEscape(linkElement.attr("href"))
 | 
			
		||||
        manga.title = titleElement.text()
 | 
			
		||||
        manga.thumbnail_url = urlFinder(element.select(".img-wrap-back").attr("style"))
 | 
			
		||||
        return manga
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    override fun popularMangaNextPageSelector() = "ul.pagination > li:not(.disabled)"
 | 
			
		||||
 | 
			
		||||
    // Do not add page parameter if page is 1 to prevent tracking.
 | 
			
		||||
    override fun popularMangaRequest(page: Int) = GET("$baseUrl/bbs/page.php?hid=manga_list" +
 | 
			
		||||
            if (page > 1) "&page=${page - 1}" else "")
 | 
			
		||||
 | 
			
		||||
    override fun popularMangaParse(response: Response): MangasPage {
 | 
			
		||||
        val document = response.asJsoup()
 | 
			
		||||
 | 
			
		||||
        val mangas = document.select(popularMangaSelector()).map { element ->
 | 
			
		||||
            popularMangaFromElement(element)
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        val hasNextPage = try {
 | 
			
		||||
            !document.select(popularMangaNextPageSelector()).last().hasClass("active")
 | 
			
		||||
        } catch (_: Exception) {
 | 
			
		||||
            false
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        return MangasPage(mangas, hasNextPage)
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    override fun searchMangaSelector() = popularMangaSelector()
 | 
			
		||||
    override fun searchMangaFromElement(element: Element) = popularMangaFromElement(element)
 | 
			
		||||
    override fun searchMangaNextPageSelector() = popularMangaSelector()
 | 
			
		||||
    override fun searchMangaParse(response: Response) = popularMangaParse(response)
 | 
			
		||||
    override fun searchMangaRequest(page: Int, query: String, filters: FilterList): Request = searchComplexFilterMangaRequestBuilder(baseUrl, page, query, filters)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    override fun mangaDetailsParse(document: Document): SManga {
 | 
			
		||||
        val info = document.select("div.left-info").first()
 | 
			
		||||
        val thumbnailElement = info.select("div.manga-thumbnail").first()
 | 
			
		||||
        val publishTypeText = thumbnailElement.select("a.publish_type").text() ?: ""
 | 
			
		||||
        val authorText = thumbnailElement.select("a.author").text() ?: ""
 | 
			
		||||
        val mangaLike = info.select("div.recommend > i.fa").first().text() ?: "0"
 | 
			
		||||
        val mangaChaptersLike = mangaElementsSum(document.select("div.addedAt i.fa.fa-thumbs-up > span"))
 | 
			
		||||
        val mangaComments = mangaElementsSum(document.select("div.addedAt i.fa.fa-comment > span"))
 | 
			
		||||
        val genres = mutableListOf<String>()
 | 
			
		||||
        document.select("div.left-info > .manga-tags > a.tag").forEach {
 | 
			
		||||
            genres.add(it.text())
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        val manga = SManga.create()
 | 
			
		||||
        manga.title = info.select("div.red").text()
 | 
			
		||||
        // They using background-image style tag for cover. extract url from style attribute.
 | 
			
		||||
        manga.thumbnail_url = urlFinder(thumbnailElement.attr("style"))
 | 
			
		||||
        // Only title and thumbnail are provided now.
 | 
			
		||||
        // TODO: Implement description when site supports it.
 | 
			
		||||
        manga.description = "\nMangaShow.Me doesn't provide manga description currently.\n" +
 | 
			
		||||
                "\n\uD83D\uDCDD: ${if (publishTypeText.trim().isBlank()) "Unknown" else publishTypeText}" +
 | 
			
		||||
                "\n\uD83D\uDCAC: $mangaComments" +
 | 
			
		||||
                "\n👍: $mangaLike ($mangaChaptersLike)"
 | 
			
		||||
        manga.author = authorText
 | 
			
		||||
        manga.genre = genres.joinToString(", ")
 | 
			
		||||
        manga.status = parseStatus(publishTypeText)
 | 
			
		||||
        return manga
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private fun parseStatus(status: String) = when (status.trim()) {
 | 
			
		||||
        "주간", "격주", "월간", "격월/비정기", "단행본" -> SManga.ONGOING
 | 
			
		||||
        "단편", "완결" -> SManga.COMPLETED
 | 
			
		||||
        // "미분류", "" -> SManga.UNKNOWN
 | 
			
		||||
        else -> SManga.UNKNOWN
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private fun mangaElementsSum(element: Elements?): String {
 | 
			
		||||
        if (element.isNullOrEmpty()) return "0"
 | 
			
		||||
        return try {
 | 
			
		||||
            String.format("%,d", element.map {
 | 
			
		||||
                it.text().toInt()
 | 
			
		||||
            }.sum())
 | 
			
		||||
        } catch (_: Exception) {
 | 
			
		||||
            "0"
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    override fun chapterListSelector() = "div.manga-detail-list > div.chapter-list > .slot"
 | 
			
		||||
 | 
			
		||||
    override fun chapterFromElement(element: Element): SChapter {
 | 
			
		||||
        val linkElement = element.select("a")
 | 
			
		||||
        val rawName = linkElement.select("div.title").last()
 | 
			
		||||
 | 
			
		||||
        val chapter = SChapter.create()
 | 
			
		||||
        chapter.url = linkElement.attr("href")
 | 
			
		||||
        chapter.chapter_number = parseChapterNumber(rawName.text())
 | 
			
		||||
        chapter.name = rawName.ownText().trim()
 | 
			
		||||
        chapter.date_upload = parseChapterDate(element.select("div.addedAt").text().split(" ").first())
 | 
			
		||||
        return chapter
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private fun parseChapterNumber(name: String): Float {
 | 
			
		||||
        try {
 | 
			
		||||
            if (name.contains("[단편]")) return 1f
 | 
			
		||||
            // `특별` means `Special`, so It can be buggy. so pad `편`(Chapter) to prevent false return
 | 
			
		||||
            if (name.contains("번외") || name.contains("특별편")) return -2f
 | 
			
		||||
            val regex = Regex("([0-9]+)(?:[-.]([0-9]+))?(?:화)")
 | 
			
		||||
            val (ch_primal, ch_second) = regex.find(name)!!.destructured
 | 
			
		||||
            return (ch_primal + if (ch_second.isBlank()) "" else ".$ch_second").toFloatOrNull() ?: -1f
 | 
			
		||||
        } catch (e: Exception) {
 | 
			
		||||
            e.printStackTrace()
 | 
			
		||||
            return -1f
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    @SuppressLint("SimpleDateFormat")
 | 
			
		||||
    private fun parseChapterDate(date: String): Long {
 | 
			
		||||
        val calendar = Calendar.getInstance()
 | 
			
		||||
 | 
			
		||||
        // MangaShow.Me doesn't provide uploaded year now(18/12/15).
 | 
			
		||||
        // If received month is bigger then current month, set last year.
 | 
			
		||||
        // TODO: Fix years due to lack of info.
 | 
			
		||||
        return try {
 | 
			
		||||
            val month = date.trim().split('-').first().toInt()
 | 
			
		||||
            val currYear = calendar.get(Calendar.YEAR)
 | 
			
		||||
            val year = if (month > calendar.get(Calendar.MONTH) + 1) // Before December now, // and Retrieved month is December == 2018.
 | 
			
		||||
                currYear - 1 else currYear
 | 
			
		||||
            SimpleDateFormat("yyyy-MM-dd").parse("$year-$date").time
 | 
			
		||||
        } catch (e: Exception) {
 | 
			
		||||
            e.printStackTrace()
 | 
			
		||||
            0
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    // They are using full url in every links.
 | 
			
		||||
    // There's possibility to using another domain for serve manga(s). Like marumaru.
 | 
			
		||||
    override fun pageListRequest(chapter: SChapter) = GET(chapter.url, headers)
 | 
			
		||||
 | 
			
		||||
    override fun pageListParse(document: Document): List<Page> {
 | 
			
		||||
        val pages = mutableListOf<Page>()
 | 
			
		||||
 | 
			
		||||
        try {
 | 
			
		||||
            val element = document.select("div.col-md-9.at-col.at-main script")
 | 
			
		||||
            val imageUrl = element.html().substringAfter("var img_list = [").substringBefore("];")
 | 
			
		||||
            val imageUrls = JSONArray("[$imageUrl]")
 | 
			
		||||
            val decoder = ImageDecoder("v1", element.html())
 | 
			
		||||
 | 
			
		||||
            (0 until imageUrls.length())
 | 
			
		||||
                    .map { imageUrls.getString(it) }
 | 
			
		||||
                    .forEach { pages.add(Page(pages.size, "", decoder.request(it))) }
 | 
			
		||||
        } catch (e: Exception) {
 | 
			
		||||
            e.printStackTrace()
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        return pages
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    // Latest not supported
 | 
			
		||||
    override fun latestUpdatesSelector() = throw UnsupportedOperationException("This method should not be called!")
 | 
			
		||||
    override fun latestUpdatesFromElement(element: Element) = throw UnsupportedOperationException("This method should not be called!")
 | 
			
		||||
    override fun latestUpdatesRequest(page: Int) = throw UnsupportedOperationException("This method should not be called!")
 | 
			
		||||
    override fun latestUpdatesNextPageSelector() = throw UnsupportedOperationException("This method should not be called!")
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    //We are able to get the image URL directly from the page list
 | 
			
		||||
    override fun imageUrlParse(document: Document) = throw UnsupportedOperationException("This method should not be called!")
 | 
			
		||||
 | 
			
		||||
    private fun urlFinder(style: String): String {
 | 
			
		||||
        // val regex = Regex("(https?:)?//[-a-zA-Z0-9@:%._\\\\+~#=]{2,256}\\.[a-z]{2,6}\\b([-a-zA-Z0-9@:%_\\\\+.~#?&/=]*)")
 | 
			
		||||
        // return regex.find(style)!!.value
 | 
			
		||||
        return style.substringAfter("background-image:url(").substringBefore(")")
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // Some title contains `&` and `#` which can cause a error.
 | 
			
		||||
    private fun urlTitleEscape(title: String): String {
 | 
			
		||||
        val url = title.split("&manga_name=")
 | 
			
		||||
        return "${url[0]}&manga_name=" +
 | 
			
		||||
                url[1].replace("&", "%26").replace("#", "%23")
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    override fun getFilterList() = getFilters()
 | 
			
		||||
 | 
			
		||||
    companion object {
 | 
			
		||||
        internal const val V1_CX = 5
 | 
			
		||||
        internal const val V1_CY = 5
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user