diff --git a/src/ru/allhentai/build.gradle b/src/ru/allhentai/build.gradle new file mode 100644 index 000000000..4fcfee0eb --- /dev/null +++ b/src/ru/allhentai/build.gradle @@ -0,0 +1,17 @@ +apply plugin: 'com.android.application' +apply plugin: 'kotlin-android' + +ext { + extName = 'AllHentai' + pkgNameSuffix = 'ru.allhentai' + extClass = '.AllHentai' + extVersionCode = 1 + libVersion = '1.2' + containsNsfw = true +} + +dependencies { + implementation project(':lib-ratelimit') +} + +apply from: "$rootDir/common.gradle" diff --git a/src/ru/allhentai/res/mipmap-hdpi/ic_launcher.png b/src/ru/allhentai/res/mipmap-hdpi/ic_launcher.png new file mode 100644 index 000000000..359610927 Binary files /dev/null and b/src/ru/allhentai/res/mipmap-hdpi/ic_launcher.png differ diff --git a/src/ru/allhentai/res/mipmap-mdpi/ic_launcher.png b/src/ru/allhentai/res/mipmap-mdpi/ic_launcher.png new file mode 100644 index 000000000..c23a8e98d Binary files /dev/null and b/src/ru/allhentai/res/mipmap-mdpi/ic_launcher.png differ diff --git a/src/ru/allhentai/res/mipmap-xhdpi/ic_launcher.png b/src/ru/allhentai/res/mipmap-xhdpi/ic_launcher.png new file mode 100644 index 000000000..6b6ce16c8 Binary files /dev/null and b/src/ru/allhentai/res/mipmap-xhdpi/ic_launcher.png differ diff --git a/src/ru/allhentai/res/mipmap-xxhdpi/ic_launcher.png b/src/ru/allhentai/res/mipmap-xxhdpi/ic_launcher.png new file mode 100644 index 000000000..ecbb34fd2 Binary files /dev/null and b/src/ru/allhentai/res/mipmap-xxhdpi/ic_launcher.png differ diff --git a/src/ru/allhentai/res/mipmap-xxxhdpi/ic_launcher.png b/src/ru/allhentai/res/mipmap-xxxhdpi/ic_launcher.png new file mode 100644 index 000000000..d45ddd771 Binary files /dev/null and b/src/ru/allhentai/res/mipmap-xxxhdpi/ic_launcher.png differ diff --git a/src/ru/allhentai/res/web_hi_res_512.png b/src/ru/allhentai/res/web_hi_res_512.png new file mode 100644 index 000000000..2bb4c600f Binary files /dev/null and b/src/ru/allhentai/res/web_hi_res_512.png differ diff --git a/src/ru/allhentai/src/eu/kanade/tachiyomi/extension/ru/allhentai/AllHentai.kt b/src/ru/allhentai/src/eu/kanade/tachiyomi/extension/ru/allhentai/AllHentai.kt new file mode 100644 index 000000000..b85af6d28 --- /dev/null +++ b/src/ru/allhentai/src/eu/kanade/tachiyomi/extension/ru/allhentai/AllHentai.kt @@ -0,0 +1,403 @@ +package eu.kanade.tachiyomi.extension.ru.allhentai + +import eu.kanade.tachiyomi.lib.ratelimit.RateLimitInterceptor +import eu.kanade.tachiyomi.network.GET +import eu.kanade.tachiyomi.network.asObservableSuccess +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 eu.kanade.tachiyomi.source.online.ParsedHttpSource +import eu.kanade.tachiyomi.util.asJsoup +import okhttp3.Headers +import okhttp3.HttpUrl +import okhttp3.OkHttpClient +import okhttp3.Request +import okhttp3.Response +import org.jsoup.nodes.Document +import org.jsoup.nodes.Element +import rx.Observable +import java.text.ParseException +import java.text.SimpleDateFormat +import java.util.Locale +import java.util.regex.Pattern + +class AllHentai : ParsedHttpSource() { + + override val name = "AllHentai" + + override val baseUrl = "http://allhentai.ru" + + override val lang = "ru" + + override val supportsLatest = true + + private val rateLimitInterceptor = RateLimitInterceptor(2) + + override val client: OkHttpClient = network.client.newBuilder() + .addNetworkInterceptor(rateLimitInterceptor).build() + + override fun popularMangaSelector() = "div.tile" + + override fun latestUpdatesSelector() = "div.tile" + + override fun popularMangaRequest(page: Int): Request = + GET("$baseUrl/list?sortType=rate&offset=${70 * (page - 1)}&max=70", headers) + + override fun latestUpdatesRequest(page: Int): Request = + GET("$baseUrl/list?sortType=updated&offset=${70 * (page - 1)}&max=70", headers) + + override fun popularMangaFromElement(element: Element): SManga { + val manga = SManga.create() + manga.thumbnail_url = element.select("img.lazy").first()?.attr("data-original") + element.select("h3 > a").first().let { + manga.setUrlWithoutDomain(it.attr("href")) + manga.title = it.attr("title") + } + return manga + } + + override fun latestUpdatesFromElement(element: Element): SManga = + popularMangaFromElement(element) + + override fun popularMangaNextPageSelector() = "a.nextLink" + + override fun latestUpdatesNextPageSelector() = "a.nextLink" + + override fun searchMangaRequest(page: Int, query: String, filters: FilterList): Request { + val url = HttpUrl.parse("$baseUrl/search/advanced")!!.newBuilder() + (if (filters.isEmpty()) getFilterList() else filters).forEach { filter -> + when (filter) { + is GenreList -> filter.state.forEach { genre -> + if (genre.state != Filter.TriState.STATE_IGNORE) { + url.addQueryParameter(genre.id, arrayOf("=", "=in", "=ex")[genre.state]) + } + } + } + } + if (query.isNotEmpty()) { + url.addQueryParameter("q", query) + } + return GET(url.toString().replace("=%3D", "="), headers) + } + + override fun searchMangaSelector() = popularMangaSelector() + + override fun searchMangaFromElement(element: Element): SManga = popularMangaFromElement(element) + + // max 200 results + override fun searchMangaNextPageSelector(): Nothing? = null + + override fun mangaDetailsParse(document: Document): SManga { + val infoElement = document.select("div.leftContent").first() + val rawCategory = infoElement.select("span.elem_category").text() + val category = if (rawCategory.isNotEmpty()) { + rawCategory.toLowerCase() + } else { + "манга" + } + + val manga = SManga.create() + manga.author = infoElement.select("span.elem_author").first()?.text() + manga.artist = infoElement.select("span.elem_illustrator").first()?.text() + manga.genre = infoElement.select("span.elem_genre").text().split(",").plusElement(category).joinToString { it.trim() } + manga.description = infoElement.select("div.manga-description").text() + manga.status = parseStatus(infoElement.html()) + manga.thumbnail_url = infoElement.select("img").attr("data-full") + return manga + } + + private fun parseStatus(element: String): Int = when { + element.contains("Запрещена публикация произведения по копирайту") -> SManga.LICENSED + element.contains("

Сингл") || element.contains("Перевод: завершен") -> SManga.COMPLETED + element.contains("Перевод: продолжается") -> SManga.ONGOING + else -> SManga.UNKNOWN + } + + override fun fetchChapterList(manga: SManga): Observable> { + return if (manga.status != SManga.LICENSED) { + client.newCall(chapterListRequest(manga)) + .asObservableSuccess() + .map { response -> + chapterListParse(response, manga) + } + } else { + Observable.error(java.lang.Exception("Licensed - No chapters to show")) + } + } + + private fun chapterListParse(response: Response, manga: SManga): List { + val document = response.asJsoup() + return document.select(chapterListSelector()).map { chapterFromElement(it, manga) } + } + + override fun chapterListSelector() = "div.chapters-link > table > tbody > tr:has(td > a)" + + private fun chapterFromElement(element: Element, manga: SManga): SChapter { + val urlElement = element.select("a").first() + val urlText = urlElement.text() + + val chapter = SChapter.create() + chapter.setUrlWithoutDomain(urlElement.attr("href") + "?mtr=1") + + chapter.name = urlText.removeSuffix(" новое").trim() + if (manga.title.length > 25) { + for (word in manga.title.split(' ')) { + chapter.name = chapter.name.removePrefix(word).trim() + } + } + val dots = chapter.name.indexOf("…") + val numbers = chapter.name.findAnyOf(IntRange(0, 9).map { it.toString() })?.first ?: 0 + + if (dots in 0 until numbers) { + chapter.name = chapter.name.substringAfter("…").trim() + } + + chapter.date_upload = element.select("td.hidden-xxs").last()?.text()?.let { + try { + SimpleDateFormat("dd.MM.yy", Locale.US).parse(it).time + } catch (e: ParseException) { + SimpleDateFormat("dd/MM/yy", Locale.US).parse(it).time + } + } ?: 0 + return chapter + } + + override fun chapterFromElement(element: Element): SChapter { + throw Exception("Not used") + } + + override fun prepareNewChapter(chapter: SChapter, manga: SManga) { + val basic = Regex("""\s*([0-9]+)(\s-\s)([0-9]+)\s*""") + val extra = Regex("""\s*([0-9]+\sЭкстра)\s*""") + val single = Regex("""\s*Сингл\s*""") + when { + basic.containsMatchIn(chapter.name) -> { + basic.find(chapter.name)?.let { + val number = it.groups[3]?.value!! + chapter.chapter_number = number.toFloat() + } + } + extra.containsMatchIn(chapter.name) -> // Extra chapters doesn't contain chapter number + chapter.chapter_number = -2f + single.containsMatchIn(chapter.name) -> // Oneshoots, doujinshi and other mangas with one chapter + chapter.chapter_number = 1f + } + } + + override fun pageListParse(response: Response): List { + val html = response.body()!!.string() + val beginIndex = html.indexOf("rm_h.init( [") + val endIndex = html.indexOf(");", beginIndex) + val trimmedHtml = html.substring(beginIndex, endIndex) + + val p = Pattern.compile("'.*?','.*?',\".*?\"") + val m = p.matcher(trimmedHtml) + + val pages = mutableListOf() + + var i = 0 + while (m.find()) { + val urlParts = m.group().replace("[\"\']+".toRegex(), "").split(',') + val url = if (urlParts[1].isEmpty() && urlParts[2].startsWith("/static/")) { + baseUrl + urlParts[2] + } else { + if (urlParts[1].endsWith("/manga/")) { + urlParts[0] + urlParts[2] + } else { + urlParts[1] + urlParts[0] + urlParts[2] + } + } + pages.add(Page(i++, "", url)) + } + return pages + } + + override fun pageListParse(document: Document): List { + throw Exception("Not used") + } + + override fun imageUrlParse(document: Document) = "" + + override fun imageRequest(page: Page): Request { + val imgHeader = Headers.Builder().apply { + add("User-Agent", "Mozilla/5.0 (Windows NT 6.3; WOW64)") + add("Referer", baseUrl) + }.build() + return GET(page.imageUrl!!, imgHeader) + } + + private class Genre(name: String, val id: String) : Filter.TriState(name) + private class GenreList(genres: List) : Filter.Group("Genres", genres) + + /* [...document.querySelectorAll("tr.advanced_option:nth-child(1) > td:nth-child(3) span.js-link")] + * .map(el => `Genre("${el.textContent.trim()}", $"{el.getAttribute('onclick') + * .substr(31,el.getAttribute('onclick').length-33)"})`).join(',\n') + * on https://readmanga.me/search/advanced + */ + override fun getFilterList() = FilterList( + GenreList(getGenreList()) + ) + + private fun getGenreList() = listOf( + Genre("3D", "el_626"), + Genre("ahegao", "el_855"), + Genre("footfuck", "el_912"), + Genre("gender bender", "el_89"), + Genre("handjob", "el_1254"), + Genre("megane", "el_962"), + Genre("Mind break", "el_705"), + Genre("netori", "el_1356"), + Genre("paizuri (titsfuck)", "el_1027"), + Genre("scat", "el_1221"), + Genre("tomboy", "el_881"), + Genre("x-ray", "el_1992"), + Genre("алкоголь", "el_1000"), + Genre("анал", "el_828"), + Genre("андроид", "el_1752"), + Genre("анилингус", "el_1037"), + Genre("арт", "el_1190"), + Genre("бдсм", "el_78"), + Genre("Без текста", "el_3157"), + Genre("без трусиков", "el_993"), + Genre("без цензуры", "el_888"), + Genre("беременность", "el_922"), + Genre("бикини", "el_1126"), + Genre("близнецы", "el_1092"), + Genre("боди-арт", "el_1130"), + Genre("Больница", "el_289"), + Genre("большая грудь", "el_837"), + Genre("Большая попка", "el_3156"), + Genre("борьба", "el_72"), + Genre("буккакэ", "el_82"), + Genre("в бассейне", "el_3599"), + Genre("в ванной", "el_878"), + Genre("в государственном учреждении", "el_86"), + Genre("в общественном месте", "el_866"), + Genre("в первый раз", "el_811"), + Genre("в транспорте", "el_3246"), + Genre("в цвете", "el_290"), + Genre("вампиры", "el_1250"), + Genre("веб", "el_1104"), + Genre("вибратор", "el_867"), + Genre("втроем", "el_3711"), + Genre("гарем", "el_87"), + Genre("гипноз", "el_1423"), + Genre("глубокий минет", "el_3555"), + Genre("горячий источник", "el_1209"), + Genre("групповой секс", "el_88"), + Genre("гяру и гангуро", "el_844"), + Genre("двойное проникновение", "el_911"), + Genre("Девочки волшебницы", "el_292"), + Genre("девчонки", "el_875"), + Genre("демоны", "el_1139"), + Genre("дилдо", "el_868"), + Genre("додзинси", "el_92"), + Genre("Домохозяйка", "el_300"), + Genre("драма", "el_95"), + Genre("дыра в стене", "el_1420"), + Genre("жестокость", "el_883"), + Genre("золотой дождь", "el_1007"), + Genre("зомби", "el_1099"), + Genre("зрелые женщины", "el_1441"), + Genre("Измена", "el_291"), + Genre("изнасилование", "el_124"), + Genre("инопланетяне", "el_990"), + Genre("инцест", "el_85"), + Genre("исполнение желаний", "el_909"), + Genre("исторический", "el_93"), + Genre("камера", "el_869"), + Genre("колготки", "el_849"), + Genre("комикс", "el_1003"), + Genre("косплей", "el_1024"), + Genre("кремпай", "el_3709"), + Genre("куннилингус", "el_5383"), + Genre("купальники", "el_845"), + Genre("латекс и кожа", "el_1047"), + Genre("магия", "el_1128"), + Genre("маленькая грудь", "el_870"), + Genre("мастурбация", "el_882"), + Genre("медсестра", "el_5688"), + Genre("мейдочки", "el_994"), + Genre("Мерзкий дядька", "el_2145"), + Genre("милф", "el_5679"), + Genre("много девушек", "el_860"), + Genre("много спермы", "el_1020"), + Genre("молоко", "el_1029"), + Genre("монстрдевушки", "el_1022"), + Genre("монстры", "el_917"), + Genre("мочеиспускание", "el_1193"), + Genre("мужчина крепкого телосложения", "el_5715"), + Genre("на природе", "el_842"), + Genre("наблюдение", "el_928"), + Genre("научная фантастика", "el_76"), + Genre("не бритая киска", "el_4237"), + Genre("не бритые подмышки", "el_4238"), + Genre("Нетораре", "el_303"), + Genre("обмен телами", "el_5120"), + Genre("обычный секс", "el_1012"), + Genre("огромная грудь", "el_1207"), + Genre("огромный член", "el_884"), + Genre("омораси", "el_81"), + Genre("оральный секс", "el_853"), + Genre("орки", "el_3247"), + Genre("парень пассив", "el_861"), + Genre("парни", "el_874"), + Genre("переодевание", "el_1026"), + Genre("пляж", "el_846"), + Genre("повседневность", "el_90"), + Genre("подглядывание", "el_978"), + Genre("подчинение", "el_885"), + Genre("похищение", "el_1183"), + Genre("превозмогание", "el_71"), + Genre("принуждение", "el_929"), + Genre("прозрачная одежда", "el_924"), + Genre("проституция", "el_3563"), + Genre("психические отклонения", "el_886"), + Genre("публично", "el_1045"), + Genre("пьяные", "el_2055"), + Genre("рабыни", "el_1433"), + Genre("романтика", "el_74"), + Genre("сверхъестественное", "el_634"), + Genre("секс игрушки", "el_871"), + Genre("сексуально возбужденная", "el_925"), + Genre("сибари", "el_80"), + Genre("сильный", "el_913"), + Genre("слабая", "el_455"), + Genre("спортивная форма", "el_891"), + Genre("спящие", "el_972"), + Genre("страпон", "el_872"), + Genre("Суккуб", "el_677"), + Genre("темнокожие", "el_611"), + Genre("тентакли", "el_69"), + Genre("толстушки", "el_1036"), + Genre("трагедия", "el_1321"), + Genre("трап", "el_859"), + Genre("ужасы", "el_75"), + Genre("униформа", "el_1008"), + Genre("ушастые", "el_991"), + Genre("фантазии", "el_1124"), + Genre("фемдом", "el_873"), + Genre("Фестиваль", "el_1269"), + Genre("фетиш", "el_1137"), + Genre("фистинг", "el_821"), + Genre("фурри", "el_91"), + Genre("футанари", "el_77"), + Genre("футанари имеет парня", "el_1426"), + Genre("фэнтези", "el_70"), + Genre("цельный купальник", "el_1257"), + Genre("цундере", "el_850"), + Genre("чикан", "el_1059"), + Genre("чулки", "el_889"), + Genre("шлюха", "el_763"), + Genre("эксгибиционизм", "el_813"), + Genre("Эльфы", "el_286"), + Genre("эччи", "el_798"), + Genre("юмор", "el_73"), + Genre("юные", "el_1162"), + Genre("юри", "el_84"), + Genre("яндере", "el_823"), + Genre("яой", "el_83") + ) +}