diff --git a/src/es/pelisforte/AndroidManifest.xml b/src/es/pelisforte/AndroidManifest.xml
new file mode 100644
index 000000000..568741e54
--- /dev/null
+++ b/src/es/pelisforte/AndroidManifest.xml
@@ -0,0 +1,2 @@
+
+
\ No newline at end of file
diff --git a/src/es/pelisforte/build.gradle b/src/es/pelisforte/build.gradle
new file mode 100644
index 000000000..37502ffa1
--- /dev/null
+++ b/src/es/pelisforte/build.gradle
@@ -0,0 +1,30 @@
+apply plugin: 'com.android.application'
+apply plugin: 'kotlin-android'
+apply plugin: 'kotlinx-serialization'
+
+ext {
+ extName = 'PelisForte'
+ pkgNameSuffix = 'es.pelisforte'
+ extClass = '.PelisForte'
+ extVersionCode = 1
+ libVersion = '13'
+}
+
+dependencies {
+ implementation(project(':lib-voe-extractor'))
+ implementation(project(':lib-burstcloud-extractor'))
+ implementation(project(':lib-mp4upload-extractor'))
+ implementation(project(':lib-streamwish-extractor'))
+ implementation(project(':lib-yourupload-extractor'))
+ implementation(project(':lib-fastream-extractor'))
+ implementation(project(':lib-upstream-extractor'))
+ implementation(project(':lib-filemoon-extractor'))
+ implementation(project(':lib-uqload-extractor'))
+ implementation(project(':lib-dood-extractor'))
+ implementation(project(':lib-streamtape-extractor'))
+ implementation(project(':lib-playlist-utils'))
+ implementation(project(':lib-streamlare-extractor'))
+ implementation(project(':lib-okru-extractor'))
+}
+
+apply from: "$rootDir/common.gradle"
diff --git a/src/es/pelisforte/res/mipmap-hdpi/ic_launcher.png b/src/es/pelisforte/res/mipmap-hdpi/ic_launcher.png
new file mode 100644
index 000000000..b295e339e
Binary files /dev/null and b/src/es/pelisforte/res/mipmap-hdpi/ic_launcher.png differ
diff --git a/src/es/pelisforte/res/mipmap-mdpi/ic_launcher.png b/src/es/pelisforte/res/mipmap-mdpi/ic_launcher.png
new file mode 100644
index 000000000..5f73a21ab
Binary files /dev/null and b/src/es/pelisforte/res/mipmap-mdpi/ic_launcher.png differ
diff --git a/src/es/pelisforte/res/mipmap-xhdpi/ic_launcher.png b/src/es/pelisforte/res/mipmap-xhdpi/ic_launcher.png
new file mode 100644
index 000000000..6fedde8b9
Binary files /dev/null and b/src/es/pelisforte/res/mipmap-xhdpi/ic_launcher.png differ
diff --git a/src/es/pelisforte/res/mipmap-xxhdpi/ic_launcher.png b/src/es/pelisforte/res/mipmap-xxhdpi/ic_launcher.png
new file mode 100644
index 000000000..bb4ce7347
Binary files /dev/null and b/src/es/pelisforte/res/mipmap-xxhdpi/ic_launcher.png differ
diff --git a/src/es/pelisforte/res/mipmap-xxxhdpi/ic_launcher.png b/src/es/pelisforte/res/mipmap-xxxhdpi/ic_launcher.png
new file mode 100644
index 000000000..e6841f553
Binary files /dev/null and b/src/es/pelisforte/res/mipmap-xxxhdpi/ic_launcher.png differ
diff --git a/src/es/pelisforte/src/eu/kanade/tachiyomi/animeextension/es/pelisforte/PelisForte.kt b/src/es/pelisforte/src/eu/kanade/tachiyomi/animeextension/es/pelisforte/PelisForte.kt
new file mode 100644
index 000000000..71ca96660
--- /dev/null
+++ b/src/es/pelisforte/src/eu/kanade/tachiyomi/animeextension/es/pelisforte/PelisForte.kt
@@ -0,0 +1,333 @@
+package eu.kanade.tachiyomi.animeextension.es.pelisforte
+
+import android.app.Application
+import android.content.SharedPreferences
+import androidx.preference.ListPreference
+import androidx.preference.PreferenceScreen
+import eu.kanade.tachiyomi.animesource.ConfigurableAnimeSource
+import eu.kanade.tachiyomi.animesource.model.AnimeFilter
+import eu.kanade.tachiyomi.animesource.model.AnimeFilterList
+import eu.kanade.tachiyomi.animesource.model.AnimesPage
+import eu.kanade.tachiyomi.animesource.model.SAnime
+import eu.kanade.tachiyomi.animesource.model.SEpisode
+import eu.kanade.tachiyomi.animesource.model.Video
+import eu.kanade.tachiyomi.animesource.online.AnimeHttpSource
+import eu.kanade.tachiyomi.lib.burstcloudextractor.BurstCloudExtractor
+import eu.kanade.tachiyomi.lib.doodextractor.DoodExtractor
+import eu.kanade.tachiyomi.lib.fastreamextractor.FastreamExtractor
+import eu.kanade.tachiyomi.lib.filemoonextractor.FilemoonExtractor
+import eu.kanade.tachiyomi.lib.mp4uploadextractor.Mp4uploadExtractor
+import eu.kanade.tachiyomi.lib.okruextractor.OkruExtractor
+import eu.kanade.tachiyomi.lib.streamlareextractor.StreamlareExtractor
+import eu.kanade.tachiyomi.lib.streamtapeextractor.StreamTapeExtractor
+import eu.kanade.tachiyomi.lib.streamwishextractor.StreamWishExtractor
+import eu.kanade.tachiyomi.lib.upstreamextractor.UpstreamExtractor
+import eu.kanade.tachiyomi.lib.uqloadextractor.UqloadExtractor
+import eu.kanade.tachiyomi.lib.voeextractor.VoeExtractor
+import eu.kanade.tachiyomi.lib.youruploadextractor.YourUploadExtractor
+import eu.kanade.tachiyomi.network.GET
+import eu.kanade.tachiyomi.util.asJsoup
+import okhttp3.HttpUrl.Companion.toHttpUrl
+import okhttp3.OkHttpClient
+import okhttp3.Request
+import okhttp3.Response
+import uy.kohesive.injekt.Injekt
+import uy.kohesive.injekt.api.get
+
+open class PelisForte : ConfigurableAnimeSource, AnimeHttpSource() {
+
+ override val name = "PelisForte"
+
+ override val baseUrl = "https://pelisforte.nu"
+
+ override val lang = "es"
+
+ override val supportsLatest = false
+
+ override val client: OkHttpClient = network.cloudflareClient
+
+ private val preferences: SharedPreferences by lazy {
+ Injekt.get().getSharedPreferences("source_$id", 0x0000)
+ }
+
+ companion object {
+ private const val PREF_LANGUAGE_KEY = "preferred_language"
+ private const val PREF_LANGUAGE_DEFAULT = "[LAT]"
+ private val LANGUAGE_LIST = arrayOf("[LAT]", "[SUB]", "[CAST]")
+
+ private const val PREF_QUALITY_KEY = "preferred_quality"
+ private const val PREF_QUALITY_DEFAULT = "1080"
+ private val QUALITY_LIST = arrayOf("1080", "720", "480", "360")
+
+ private const val PREF_SERVER_KEY = "preferred_server"
+ private const val PREF_SERVER_DEFAULT = "StreamWish"
+ private val SERVER_LIST = arrayOf(
+ "YourUpload", "BurstCloud", "Voe", "Mp4Upload", "Doodstream",
+ "Upload", "BurstCloud", "Upstream", "StreamTape", "Doodstream",
+ "Fastream", "Filemoon", "StreamWish", "Okru",
+ )
+ }
+
+ override fun popularAnimeRequest(page: Int) = GET("$baseUrl/ultimas-peliculas/page/$page", headers)
+
+ override fun popularAnimeParse(response: Response): AnimesPage {
+ val document = response.asJsoup()
+ val elements = document.select("#movies-a li[id*=post-]")
+ val nextPage = document.select(".pagination .nav-links .current ~ a:not(.page-link)").any()
+ val animeList = elements.map { element ->
+ SAnime.create().apply {
+ setUrlWithoutDomain(element.selectFirst("article > a")?.attr("abs:href") ?: "")
+ title = element.selectFirst("article .entry-header .entry-title")?.text() ?: ""
+ thumbnail_url = element.selectFirst("article .post-thumbnail figure img")?.getImageUrl()
+ }
+ }
+ return AnimesPage(animeList, nextPage)
+ }
+
+ protected open fun org.jsoup.nodes.Element.getImageUrl(): String? {
+ return if (hasAttr("srcset")) {
+ try {
+ fetchUrls(attr("abs:srcset")).maxOrNull()
+ } catch (_: Exception) {
+ attr("abs:src")
+ }
+ } else {
+ attr("abs:src")
+ }
+ }
+
+ override fun latestUpdatesRequest(page: Int) = popularAnimeRequest(page)
+
+ override fun latestUpdatesParse(response: Response) = popularAnimeParse(response)
+
+ override fun searchAnimeRequest(page: Int, query: String, filters: AnimeFilterList): Request {
+ val filterList = if (filters.isEmpty()) getFilterList() else filters
+ val genreFilter = filterList.find { it is GenreFilter } as GenreFilter
+
+ return when {
+ query.isNotBlank() -> GET("$baseUrl/page/$page?s=$query", headers)
+ genreFilter.state != 0 -> GET("$baseUrl/${genreFilter.toUriPart()}", headers)
+ else -> popularAnimeRequest(page)
+ }
+ }
+
+ override fun searchAnimeParse(response: Response) = popularAnimeParse(response)
+
+ override fun animeDetailsParse(response: Response): SAnime {
+ val document = response.asJsoup()
+ val animeDetails = SAnime.create().apply {
+ title = document.selectFirst(".alg-cr .entry-header .entry-title")?.text() ?: ""
+ description = document.select(".alg-cr .description").text()
+ thumbnail_url = document.selectFirst(".alg-cr .post-thumbnail img")?.getImageUrl()
+ genre = document.select(".genres a").joinToString { it.text() }
+ status = SAnime.UNKNOWN
+ }
+
+ document.select(".cast-lst li").map {
+ if (it.select("span").text().contains("Director", true)) {
+ animeDetails.author = it.selectFirst("p > a")?.text()
+ }
+ if (it.select("span").text().contains("Actores", true)) {
+ animeDetails.artist = it.selectFirst("p > a")?.text()
+ }
+ }
+ return animeDetails
+ }
+
+ override fun episodeListParse(response: Response): List {
+ return listOf(
+ SEpisode.create().apply {
+ setUrlWithoutDomain(response.request.url.toString())
+ name = "Película"
+ episode_number = 1F
+ },
+ )
+ }
+
+ private fun fetchUrls(text: String?): List {
+ if (text.isNullOrEmpty()) return listOf()
+ val linkRegex = "(http|ftp|https):\\/\\/([\\w_-]+(?:(?:\\.[\\w_-]+)+))([\\w.,@?^=%&:\\/~+#-]*[\\w@?^=%&\\/~+#-])".toRegex()
+ return linkRegex.findAll(text).map { it.value.trim().removeSurrounding("\"") }.toList()
+ }
+
+ override fun videoListParse(response: Response): List