parent
94181a9eb1
commit
61409c94b2
17
src/ja/shonenjumpplus/build.gradle
Normal file
17
src/ja/shonenjumpplus/build.gradle
Normal file
@ -0,0 +1,17 @@
|
||||
apply plugin: 'com.android.application'
|
||||
apply plugin: 'kotlin-android'
|
||||
|
||||
ext {
|
||||
appName = 'Tachiyomi: Shonen Jump+'
|
||||
pkgNameSuffix = 'ja.shonenjumpplus'
|
||||
extClass = '.ShonenJumpPlus'
|
||||
extVersionCode = 1
|
||||
libVersion = '1.2'
|
||||
}
|
||||
|
||||
dependencies {
|
||||
compileOnly 'com.google.code.gson:gson:2.8.2'
|
||||
compileOnly 'com.github.salomonbrys.kotson:kotson:2.5.0'
|
||||
}
|
||||
|
||||
apply from: "$rootDir/common.gradle"
|
BIN
src/ja/shonenjumpplus/res/mipmap-hdpi/ic_launcher.png
Normal file
BIN
src/ja/shonenjumpplus/res/mipmap-hdpi/ic_launcher.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 4.1 KiB |
BIN
src/ja/shonenjumpplus/res/mipmap-mdpi/ic_launcher.png
Normal file
BIN
src/ja/shonenjumpplus/res/mipmap-mdpi/ic_launcher.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 2.2 KiB |
BIN
src/ja/shonenjumpplus/res/mipmap-xhdpi/ic_launcher.png
Normal file
BIN
src/ja/shonenjumpplus/res/mipmap-xhdpi/ic_launcher.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 5.3 KiB |
BIN
src/ja/shonenjumpplus/res/mipmap-xxhdpi/ic_launcher.png
Normal file
BIN
src/ja/shonenjumpplus/res/mipmap-xxhdpi/ic_launcher.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 10 KiB |
BIN
src/ja/shonenjumpplus/res/mipmap-xxxhdpi/ic_launcher.png
Normal file
BIN
src/ja/shonenjumpplus/res/mipmap-xxxhdpi/ic_launcher.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 14 KiB |
BIN
src/ja/shonenjumpplus/res/web_hi_res_512.png
Normal file
BIN
src/ja/shonenjumpplus/res/web_hi_res_512.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 55 KiB |
@ -0,0 +1,261 @@
|
||||
package eu.kanade.tachiyomi.extension.ja.shonenjumpplus
|
||||
|
||||
import android.graphics.Bitmap
|
||||
import android.graphics.BitmapFactory
|
||||
import android.graphics.Canvas
|
||||
import android.graphics.Rect
|
||||
import com.github.salomonbrys.kotson.get
|
||||
import com.github.salomonbrys.kotson.obj
|
||||
import com.github.salomonbrys.kotson.string
|
||||
import com.google.gson.JsonObject
|
||||
import com.google.gson.JsonParser
|
||||
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.*
|
||||
import org.jsoup.Jsoup
|
||||
import org.jsoup.nodes.Document
|
||||
import org.jsoup.nodes.Element
|
||||
import java.io.ByteArrayOutputStream
|
||||
import java.io.InputStream
|
||||
import java.lang.Exception
|
||||
import java.text.ParseException
|
||||
import java.text.SimpleDateFormat
|
||||
import java.util.*
|
||||
import kotlin.math.floor
|
||||
|
||||
class ShonenJumpPlus : ParsedHttpSource() {
|
||||
|
||||
override val name = "Shonen Jump+"
|
||||
|
||||
override val baseUrl = "https://shonenjumpplus.com"
|
||||
|
||||
override val lang = "ja"
|
||||
|
||||
override val supportsLatest = true
|
||||
|
||||
override val client: OkHttpClient = network.client.newBuilder()
|
||||
.addInterceptor { imageIntercept(it) }
|
||||
.build()
|
||||
|
||||
private val dayOfWeek: String
|
||||
get() = when (Calendar.getInstance().get(Calendar.DAY_OF_WEEK)) {
|
||||
Calendar.SUNDAY -> "sunday"
|
||||
Calendar.MONDAY -> "monday"
|
||||
Calendar.TUESDAY -> "tuesday"
|
||||
Calendar.WEDNESDAY -> "wednesday"
|
||||
Calendar.THURSDAY -> "thursday"
|
||||
Calendar.FRIDAY -> "friday"
|
||||
else -> "saturday"
|
||||
}
|
||||
|
||||
override fun headersBuilder(): Headers.Builder = Headers.Builder()
|
||||
.add("User-Agent", USER_AGENT)
|
||||
.add("Origin", baseUrl)
|
||||
.add("Referer", baseUrl)
|
||||
|
||||
override fun popularMangaRequest(page: Int): Request = GET("$baseUrl/series", headers)
|
||||
|
||||
override fun popularMangaSelector(): String = "ul.series-list li a"
|
||||
|
||||
override fun popularMangaFromElement(element: Element): SManga = SManga.create().apply {
|
||||
title = element.select("h2.series-list-title").first()!!.text()
|
||||
thumbnail_url = element.select("div.series-list-thumb img").first()!!.attr("data-src")
|
||||
setUrlWithoutDomain(element.attr("href"))
|
||||
}
|
||||
|
||||
override fun popularMangaNextPageSelector(): String? = null
|
||||
|
||||
override fun latestUpdatesRequest(page: Int): Request = popularMangaRequest(page)
|
||||
|
||||
override fun latestUpdatesSelector(): String = "h2.series-list-date-week.$dayOfWeek + ul.series-list li a"
|
||||
|
||||
override fun latestUpdatesFromElement(element: Element): SManga = popularMangaFromElement(element)
|
||||
|
||||
override fun latestUpdatesNextPageSelector(): String? = null
|
||||
|
||||
override fun searchMangaRequest(page: Int, query: String, filters: FilterList): Request {
|
||||
if (query.isNotEmpty()) {
|
||||
val url = HttpUrl.parse("$baseUrl/search")!!.newBuilder()
|
||||
.addQueryParameter("q", query)
|
||||
|
||||
return GET(url.toString(), headers)
|
||||
}
|
||||
|
||||
val path = arrayOf("", "oneshot", "finished")[(filters[0] as SeriesListMode).state]
|
||||
return GET("$baseUrl/series/$path", headers)
|
||||
}
|
||||
|
||||
override fun searchMangaParse(response: Response): MangasPage {
|
||||
if (response.request().url().toString().contains("search"))
|
||||
return super.searchMangaParse(response)
|
||||
|
||||
return popularMangaParse(response)
|
||||
}
|
||||
|
||||
override fun searchMangaSelector() = "ul.search-series-list li"
|
||||
|
||||
override fun searchMangaFromElement(element: Element): SManga = SManga.create().apply {
|
||||
title = element.select("div.title-box p.series-title").first()!!.text()
|
||||
thumbnail_url = element.select("div.thmb-container a img").first()!!.attr("src")
|
||||
setUrlWithoutDomain(element.select("div.thmb-container a").first()!!.attr("href"))
|
||||
}
|
||||
|
||||
override fun searchMangaNextPageSelector(): String? = null
|
||||
|
||||
override fun mangaDetailsParse(document: Document): SManga = SManga.create().apply {
|
||||
val infoElement = document.select("section.series-information div.series-header")
|
||||
|
||||
title = infoElement.select("h1.series-header-title").first()!!.text()
|
||||
author = infoElement.select("h2.series-header-author").first()!!.text()
|
||||
artist = author
|
||||
description = infoElement.select("p.series-header-description").first()!!.text()
|
||||
thumbnail_url = infoElement.select("div.series-header-image-wrapper img").first()!!.attr("data-src")
|
||||
}
|
||||
|
||||
override fun chapterListParse(response: Response): List<SChapter> {
|
||||
val document = response.asJsoup()
|
||||
val readableProductList = document.select("div.js-readable-product-list").first()!!
|
||||
val latestListEndpoint = HttpUrl.parse(readableProductList.attr("data-latest-list-endpoint"))!!
|
||||
val firstListEndpoint = HttpUrl.parse(readableProductList.attr("data-first-list-endpoint"))!!
|
||||
val numberSince = latestListEndpoint.queryParameter("number_since")!!.toInt()
|
||||
.coerceAtLeast(firstListEndpoint.queryParameter("number_since")!!.toInt())
|
||||
|
||||
val newHeaders = headers.newBuilder()
|
||||
.set("Referer", response.request().url().toString())
|
||||
.build()
|
||||
var readMoreEndpoint = firstListEndpoint.newBuilder()
|
||||
.setQueryParameter("number_since", numberSince.toString())
|
||||
.toString()
|
||||
|
||||
val chapters = mutableListOf<SChapter>()
|
||||
|
||||
var result = client.newCall(GET(readMoreEndpoint, newHeaders)).execute()
|
||||
|
||||
while (result.code() != 404) {
|
||||
val json = result.asJsonObject()
|
||||
readMoreEndpoint = json["nextUrl"].string
|
||||
val tempDocument = Jsoup.parse(json["html"].string)
|
||||
|
||||
chapters += tempDocument
|
||||
.select("ul.series-episode-list " + chapterListSelector())
|
||||
.map { element -> chapterFromElement(element, response.request().url().toString()) }
|
||||
|
||||
result = client.newCall(GET(readMoreEndpoint, newHeaders)).execute()
|
||||
}
|
||||
|
||||
return chapters
|
||||
}
|
||||
|
||||
override fun chapterListSelector() = "li.episode:has(span.series-episode-list-is-free)"
|
||||
|
||||
private fun chapterFromElement(element: Element, mangaUrl: String): SChapter {
|
||||
val info = element.select("a.series-episode-list-container").first() ?: element
|
||||
|
||||
return SChapter.create().apply {
|
||||
name = info.select("h4.series-episode-list-title").first()!!.text()
|
||||
date_upload = parseChapterDate(info.select("span.series-episode-list-date").first()?.text().orEmpty())
|
||||
scanlator = "集英社"
|
||||
setUrlWithoutDomain(if (info.tagName() == "a") info.attr("href") else mangaUrl)
|
||||
}
|
||||
}
|
||||
|
||||
override fun pageListRequest(chapter: SChapter): Request {
|
||||
val episodeId = chapter.url.substringAfterLast("/")
|
||||
return GET("$baseUrl/episode/$episodeId.json", headers)
|
||||
}
|
||||
|
||||
override fun pageListParse(response: Response): List<Page> {
|
||||
val json = response.asJsonObject()
|
||||
val pages = json["readableProduct"]["pageStructure"]["pages"].asJsonArray
|
||||
|
||||
return pages
|
||||
.filter { it["type"].string == "main" }
|
||||
.mapIndexed { i, pageObj ->
|
||||
val imageUrl = "${pageObj["src"].string}?width=${pageObj["width"].string}&height=${pageObj["height"].string}"
|
||||
Page(i, "", imageUrl)
|
||||
}
|
||||
}
|
||||
|
||||
override fun imageUrlParse(document: Document) = ""
|
||||
|
||||
private class SeriesListMode : Filter.Select<String>("一覧", arrayOf("ジャンプ+連載一覧", "ジャンプ+読切シリーズ", "連載終了作品"))
|
||||
|
||||
override fun getFilterList(): FilterList = FilterList(SeriesListMode())
|
||||
|
||||
override fun chapterFromElement(element: Element): SChapter = throw Exception("This method should not be called!")
|
||||
|
||||
override fun pageListParse(document: Document): List<Page> = throw Exception("This method should not be called!")
|
||||
|
||||
private fun imageIntercept(chain: Interceptor.Chain): Response {
|
||||
var request = chain.request()
|
||||
|
||||
if (!request.url().toString().startsWith(CDN_URL)) {
|
||||
return chain.proceed(request)
|
||||
}
|
||||
|
||||
val width = request.url().queryParameter("width")!!.toInt()
|
||||
val height = request.url().queryParameter("height")!!.toInt()
|
||||
|
||||
val newUrl = request.url().newBuilder()
|
||||
.removeAllQueryParameters("width")
|
||||
.removeAllQueryParameters("height")
|
||||
.build()
|
||||
request = request.newBuilder().url(newUrl).build()
|
||||
|
||||
val response = chain.proceed(request)
|
||||
val image = decodeImage(response.body()!!.byteStream(), width, height)
|
||||
val body = ResponseBody.create(MediaType.parse("image/png"), image)
|
||||
return response.newBuilder().body(body).build()
|
||||
}
|
||||
|
||||
private fun decodeImage(image: InputStream, width: Int, height: Int): ByteArray {
|
||||
val input = BitmapFactory.decodeStream(image)
|
||||
val cWidth = (floor(width.toDouble() / (DIVIDE_NUM * MULTIPLE)) * MULTIPLE).toInt()
|
||||
val cHeight = (floor(height.toDouble() / (DIVIDE_NUM * MULTIPLE)) * MULTIPLE).toInt()
|
||||
|
||||
val result = Bitmap.createBitmap(width, height, Bitmap.Config.ARGB_8888)
|
||||
val canvas = Canvas(result)
|
||||
|
||||
val imageRect = Rect(0, 0, width, height)
|
||||
canvas.drawBitmap(input, imageRect, imageRect, null)
|
||||
|
||||
for (e in 0 until DIVIDE_NUM * DIVIDE_NUM) {
|
||||
val x = e % DIVIDE_NUM * cWidth
|
||||
val y = (floor(e.toFloat() / DIVIDE_NUM) * cHeight).toInt()
|
||||
val cellSrc = Rect(x, y, x + cWidth, y + cHeight)
|
||||
|
||||
val row = floor(e.toFloat() / DIVIDE_NUM).toInt()
|
||||
val dstE = e % DIVIDE_NUM * DIVIDE_NUM + row
|
||||
val dstX = dstE % DIVIDE_NUM * cWidth
|
||||
val dstY = (floor(dstE.toFloat() / DIVIDE_NUM) * cHeight).toInt()
|
||||
val cellDst = Rect(dstX, dstY, dstX + cWidth, dstY + cHeight)
|
||||
canvas.drawBitmap(input, cellSrc, cellDst, null)
|
||||
}
|
||||
|
||||
val output = ByteArrayOutputStream()
|
||||
result.compress(Bitmap.CompressFormat.PNG, 100, output)
|
||||
return output.toByteArray()
|
||||
}
|
||||
|
||||
private fun parseChapterDate(date: String) : Long {
|
||||
return try {
|
||||
DATE_PARSER.parse(date).time
|
||||
} catch (e: ParseException) {
|
||||
0L
|
||||
}
|
||||
}
|
||||
|
||||
private fun Response.asJsonObject(): JsonObject = JSON_PARSER.parse(body()!!.string()).obj
|
||||
|
||||
companion object {
|
||||
private const val USER_AGENT = "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/69.0.3497.92 Safari/537.36"
|
||||
private val JSON_PARSER by lazy { JsonParser() }
|
||||
private val DATE_PARSER by lazy { SimpleDateFormat("yyyy/MM/dd", Locale.ENGLISH) }
|
||||
|
||||
private const val CDN_URL = "https://cdn-ak-img.shonenjumpplus.com"
|
||||
private const val DIVIDE_NUM = 4
|
||||
private const val MULTIPLE = 8
|
||||
}
|
||||
}
|
Loading…
x
Reference in New Issue
Block a user