diff --git a/src/it/shingekinoshoujo/AndroidManifest.xml b/src/it/shingekinoshoujo/AndroidManifest.xml
new file mode 100644
index 000000000..30deb7f79
--- /dev/null
+++ b/src/it/shingekinoshoujo/AndroidManifest.xml
@@ -0,0 +1,2 @@
+
+
diff --git a/src/it/shingekinoshoujo/build.gradle b/src/it/shingekinoshoujo/build.gradle
new file mode 100644
index 000000000..4e08d6c94
--- /dev/null
+++ b/src/it/shingekinoshoujo/build.gradle
@@ -0,0 +1,11 @@
+apply plugin: 'com.android.application'
+apply plugin: 'kotlin-android'
+
+ext {
+ extName = 'Shingeki no Shoujo'
+ pkgNameSuffix = 'it.shingekinoshoujo'
+ extClass = '.ShingekiNoShoujo'
+ extVersionCode = 1
+}
+
+apply from: "$rootDir/common.gradle"
diff --git a/src/it/shingekinoshoujo/res/mipmap-hdpi/ic_launcher.png b/src/it/shingekinoshoujo/res/mipmap-hdpi/ic_launcher.png
new file mode 100644
index 000000000..524250f61
Binary files /dev/null and b/src/it/shingekinoshoujo/res/mipmap-hdpi/ic_launcher.png differ
diff --git a/src/it/shingekinoshoujo/res/mipmap-mdpi/ic_launcher.png b/src/it/shingekinoshoujo/res/mipmap-mdpi/ic_launcher.png
new file mode 100644
index 000000000..1a6248fdd
Binary files /dev/null and b/src/it/shingekinoshoujo/res/mipmap-mdpi/ic_launcher.png differ
diff --git a/src/it/shingekinoshoujo/res/mipmap-xhdpi/ic_launcher.png b/src/it/shingekinoshoujo/res/mipmap-xhdpi/ic_launcher.png
new file mode 100644
index 000000000..67abda1d2
Binary files /dev/null and b/src/it/shingekinoshoujo/res/mipmap-xhdpi/ic_launcher.png differ
diff --git a/src/it/shingekinoshoujo/res/mipmap-xxhdpi/ic_launcher.png b/src/it/shingekinoshoujo/res/mipmap-xxhdpi/ic_launcher.png
new file mode 100644
index 000000000..00c4a5221
Binary files /dev/null and b/src/it/shingekinoshoujo/res/mipmap-xxhdpi/ic_launcher.png differ
diff --git a/src/it/shingekinoshoujo/res/mipmap-xxxhdpi/ic_launcher.png b/src/it/shingekinoshoujo/res/mipmap-xxxhdpi/ic_launcher.png
new file mode 100644
index 000000000..529c550ce
Binary files /dev/null and b/src/it/shingekinoshoujo/res/mipmap-xxxhdpi/ic_launcher.png differ
diff --git a/src/it/shingekinoshoujo/res/web_hi_res_512.png b/src/it/shingekinoshoujo/res/web_hi_res_512.png
new file mode 100644
index 000000000..10bd3aab8
Binary files /dev/null and b/src/it/shingekinoshoujo/res/web_hi_res_512.png differ
diff --git a/src/it/shingekinoshoujo/src/eu/kanade/tachiyomi/extension/it/shingekinoshoujo/ShingekiNoShoujo.kt b/src/it/shingekinoshoujo/src/eu/kanade/tachiyomi/extension/it/shingekinoshoujo/ShingekiNoShoujo.kt
new file mode 100644
index 000000000..fa5d2b5f1
--- /dev/null
+++ b/src/it/shingekinoshoujo/src/eu/kanade/tachiyomi/extension/it/shingekinoshoujo/ShingekiNoShoujo.kt
@@ -0,0 +1,213 @@
+package eu.kanade.tachiyomi.extension.it.shingekinoshoujo
+
+import eu.kanade.tachiyomi.network.GET
+import eu.kanade.tachiyomi.source.model.Filter
+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.ParsedHttpSource
+import eu.kanade.tachiyomi.util.asJsoup
+import okhttp3.Headers
+import okhttp3.OkHttpClient
+import okhttp3.Request
+import okhttp3.Response
+import org.jsoup.nodes.Document
+import org.jsoup.nodes.Element
+
+class ShingekiNoShoujo : ParsedHttpSource() {
+
+ override val name = "Shingeki no Shoujo"
+ override val baseUrl = "https://shingekinoshoujo.to"
+ override val lang = "it"
+ override val supportsLatest = true
+ override val client: OkHttpClient = network.cloudflareClient
+
+ //region REQUESTS
+
+ override fun popularMangaRequest(page: Int): Request = GET("$baseUrl/", headers)
+ override fun latestUpdatesRequest(page: Int): Request = GET("$baseUrl/novita", headers)
+ override fun searchMangaRequest(page: Int, query: String, filters: FilterList): Request {
+ if (query.isNotEmpty()) {
+ return GET("$baseUrl/page/$page/?s=$query", headers)
+ } else {
+ (if (filters.isEmpty()) getFilterList() else filters).forEach { filter ->
+ when (filter) {
+ is GenreSelez -> {
+ return GET(
+ "$baseUrl/tag/${getGenreList().filter {
+ filter.values[filter.state] == it.name
+ }.map { it.id }[0]}/page/$page"
+ )
+ }
+ }
+ }
+ return GET(baseUrl, headers)
+ }
+ }
+ //endregion
+
+ //region CONTENTS INFO
+ private fun mangasParse(response: Response, selector: String, num: Int): MangasPage {
+ val document = response.asJsoup()
+
+ val mangas = document.select(selector).map { element ->
+ when (num) {
+ 1 -> popularMangaFromElement(element)
+ 2 -> latestUpdatesFromElement(element)
+ else -> searchMangaFromElement(element)
+ }
+ }
+ return MangasPage(
+ mangas,
+ !document.select(searchMangaNextPageSelector()).isNullOrEmpty()
+ )
+ }
+ override fun popularMangaParse(response: Response): MangasPage = mangasParse(response, popularMangaSelector(), 1)
+ override fun latestUpdatesParse(response: Response): MangasPage = mangasParse(response, latestUpdatesSelector(), 2)
+ override fun searchMangaParse(response: Response): MangasPage = mangasParse(response, searchMangaSelector(), 3)
+
+ override fun popularMangaSelector() = ".ultp-block-item > .ultp-block-content-wrap"
+ override fun latestUpdatesSelector() = ".spiffy-popup > a"
+ override fun searchMangaSelector() = "article[id^=post].format-standard:contains(Genere)"
+
+ override fun popularMangaFromElement(element: Element): SManga {
+ val manga = SManga.create().apply {
+ thumbnail_url = element.selectFirst("img").attr("src")
+ element.select("a").last().let {
+ setUrlWithoutDomain(it.attr("href"))
+ title = it.text()
+ }
+ }
+ return manga
+ }
+ override fun latestUpdatesFromElement(element: Element): SManga = SManga.create().apply {
+ thumbnail_url = element.select("img").first().attr("src")
+ setUrlWithoutDomain(element.attr("href"))
+ title = element.select("span").first().text()
+ }
+ override fun searchMangaFromElement(element: Element): SManga = SManga.create().apply {
+ thumbnail_url = element.select(".entry-thumbnail > img")?.attr("src") ?: ""
+ element.select("a").first().let {
+ setUrlWithoutDomain(it.attr("href"))
+ title = it.text()
+ }
+ }
+
+ override fun mangaDetailsParse(document: Document): SManga {
+ val statusElementText = document.select("blockquote").last().text().lowercase()
+ return SManga.create().apply {
+ thumbnail_url = document.select(".entry-thumbnail > img").attr("src")
+ status = when {
+ statusElementText.contains("in corso") -> SManga.ONGOING
+ statusElementText.contains("fine") -> SManga.COMPLETED
+ else -> SManga.UNKNOWN
+ }
+ author = document.select("span:has(strong:contains(Autore))")?.text()!!.substringAfter("Autore:").trim()
+ genre = document.select("span:has(strong:contains(Genere))")?.text()!!.substringAfter("Genere:").replace(".", "").trim()
+ description = document.select("p:has(strong:contains(Trama))")?.text()!!.substringAfter("Trama:").trim()
+ }
+ }
+ //endregion
+
+ //region NEXT SELECTOR - Not used
+
+ override fun popularMangaNextPageSelector() = latestUpdatesNextPageSelector()
+ override fun latestUpdatesNextPageSelector(): String? = null
+ override fun searchMangaNextPageSelector() = ".nav-previous:contains(Articoli meno recenti)"
+ //endregion
+
+ //region CHAPTER and PAGES
+
+ override fun chapterListParse(response: Response): List {
+ val document = response.asJsoup()
+ val chapters = mutableListOf()
+ document.select(chapterListSelector().replace("", document.location().toString().substringAfter("$baseUrl/").substringBefore("/"))).forEachIndexed { i, it ->
+ chapters.add(
+ SChapter.create().apply {
+ setUrlWithoutDomain(it.attr("href"))
+ name = it.text()
+ chapter_number = it.text().replace(Regex("OneShot|Prologo"), "0").filter { it.isDigit() }.let {
+ if (it.isEmpty()) "$i" else it
+ }.toFloat()
+ }
+ )
+ }
+
+ chapters.reverse()
+ return chapters
+ }
+ override fun chapterListSelector() = ".entry-content > blockquote > p > a[href], .entry-content > ul > li a[href*=]:not(:has(img))"
+ override fun chapterFromElement(element: Element) = throw Exception("Not used")
+
+ override fun pageListParse(document: Document): List {
+ val pages = mutableListOf()
+
+ document.select(".alignnone").forEachIndexed { i, it ->
+ pages.add(Page(i, "", it.attr("src")))
+ }
+ if (document.toString().contains("che state leggendo")) {
+ pages.add(Page(1, "", "https://i.imgur.com/l0eZuoO.png"))
+ } else if (pages.isEmpty()) {
+ pages.add(Page(1, "", "https://i.imgur.com/DPZ6K2q.png"))
+ }
+
+ return pages
+ }
+
+ override fun imageUrlParse(document: Document) = ""
+ override fun imageRequest(page: Page): Request {
+ val imgHeader = Headers.Builder().apply {
+ add("User-Agent", "Mozilla/5.0 (Linux; U; Android 4.1.1; en-gb; Build/KLP) AppleWebKit/534.30 (KHTML, like Gecko) Version/4.0 Safari/534.30")
+ add("Referer", baseUrl)
+ }.build()
+ return GET(page.imageUrl!!, imgHeader)
+ }
+ //endregion
+
+ //region FILTERS
+ private class Genre(name: String, val id: String = name) : Filter.CheckBox(name)
+ private class GenreSelez(genres: List) : Filter.Select(
+ "Genere",
+ genres.map {
+ it.name
+ }.toTypedArray(),
+ 0
+ )
+
+ override fun getFilterList() = FilterList(
+ Filter.Header("La ricerca testuale non accetta i filtri e viceversa"),
+ GenreSelez(getGenreList())
+ )
+
+ private fun getGenreList() = listOf(
+ Genre("Josei", "josei"),
+ Genre("Sportivo", "sportivo"),
+ Genre("Harlequin", "harlequin"),
+ Genre("Ufficio", "ufficio"),
+ Genre("Shoujo", "shoujo"),
+ Genre("Fujitani Yoko", "fujitani-yoko"),
+ Genre("Tsukishima Haru", "tsukishima-haru"),
+ Genre("Scolastico", "scolastico"),
+ Genre("Viaggio nel Tempo", "viaggio-nel-tempo"),
+ Genre("Vita Scolastica", "vita-scolastica"),
+ Genre("Seinen", "seinen"),
+ Genre("Drammatico", "drammatico"),
+ Genre("Commedia", "commedia"),
+ Genre("Drama", "drama"),
+ Genre("Oneshot", "oneshot"),
+ Genre("Mistero", "mistero"),
+ Genre("Saori", "saori"),
+ Genre("Fuji Momo", "fuji-momo"),
+ Genre("Azione", "azione"),
+ Genre("Sovrannaturale", "sovrannaturale"),
+ Genre("Vita Quotidiana", "vita-quotidiana"),
+ Genre("Storico", "storico"),
+ Genre("Shibano Yuka", "shibano-yuka"),
+ Genre("Fantasy", "fantasy"),
+ Genre("Smut", "smut"),
+ Genre("Psicologico", "psicologico")
+ )
+ //endregion
+}