remove some NSFW sources

This commit is contained in:
kodjomoustapha
2024-09-21 23:00:04 +01:00
parent 7bb9edb80b
commit a5655eb692
442 changed files with 2 additions and 7321 deletions

View File

@@ -1,347 +0,0 @@
const mangayomiSources = [{
"name": "Njav",
"lang": "all",
"baseUrl": "https://njav.tv/en",
"apiUrl": "",
"iconUrl": "https://njav.tv/assets/njav/images/favicon.png",
"typeSource": "single",
"isManga": false,
"isNsfw": true,
"version": "0.0.1",
"apiUrl": "",
"dateFormat": "",
"dateFormatLocale": "",
"pkgName": "anime/src/all/njav.js"
}];
class DefaultExtension extends MProvider {
dateStringToTimestamp(dateString) {
var parts = dateString.split('-');
var year = parseInt(parts[0]);
var month = parseInt(parts[1]) - 1;
var day = parseInt(parts[2]);
var date = new Date(year, month, day);
var timestamp = date.getTime();
return timestamp;
}
async request(url) {
const preference = new SharedPreferences();
const res = await new Client().get(preference.get("url") + "/" + preference.get("lang") + url);
return res.body;
}
async getItems(url) {
const res = await this.request(url);
const doc = new Document(res);
const elements = doc.select("div.box-item");
const items = [];
for (const element of elements) {
const cover = element.selectFirst("img").attr("data-src");
const info = element.selectFirst("div.detail a");
const url = info.attr("href");
const title = info.text;
items.push({
link: "/" + url,
imageUrl: cover,
name: title
});
}
return {
list: items,
hasNextPage: true
}
}
async getPopular(page) {
return await this.getItems(`/trending?page=${page}`);
}
async getLatestUpdates(page) {
return await this.getItems(`/new-release?page=${page}`);
}
async search(query, page, filters) {
if (query == "") {
var category, sort;
for (const filter of filters) {
if (filter["type"] == "CateFilter") {
category = filter["values"][filter["state"]]["value"];
} else if (filter["type"] == "SortFilter") {
sort = filter["values"][filter["state"]]["value"];
}
}
return await this.getItems(`/${category}?sort=${sort}&page=${page}`);
} else {
return await this.getItems(`/search?keyword=${query}&page=${page}`);
}
}
async getEpisodes(id, time) {
const res = await this.request(`/ajax/v/${id}/videos`);
const datas = JSON.parse(res);
const ep = [];
for (const data of datas["data"]["watch"]) {
ep.push({
name: data["name"],
url: data["url"],
dateUpload: time.toString()
});
}
return ep;
}
async getDetail(url) {
const res = await this.request(url);
const doc = new Document(res);
const body = doc.selectFirst("div#body");
const title = body.selectFirst("h1").text;
const cover = body.selectFirst("div#player").attr("data-poster");
const info = body.selectFirst("div.detail-item").select("div");
var desc;
try {
desc = body.selectFirst("div.description p").text;
} catch {
desc = "";
}
const updateTime = this.dateStringToTimestamp(info[1].select("span")[1].text);
var author;
try {
author = info[3].select("span")[1].text.replaceAll("\n", "");
} catch {
author = "Unknown";
}
var genres
try {
genres = info[4].selectFirst("span.genre").select("a").map(e => e.text);
} catch {
genres = [];
}
const id = body.selectFirst("div.container").attr("v-scope").slice(12, -3);
const eps = await this.getEpisodes(id, updateTime);
return {
name: title,
imageUrl: cover,
author: author,
genre: genres,
description: desc,
episodes: eps
};
}
async getVideoList(url) {
const res = await new Client().get(url);
const doc = new Document(res.body);
const str = doc.selectFirst("div#player").attr("v-scope").match(/, {([^']*)\)/)[1];
const data = JSON.parse("{" + str);
return [{
url: data["stream"],
originalUrl: data["stream"],
quality: "Origin",
headers: {
Referer: "https://javplayer.me/",
Origin: "https://javplayer.me"
}
}];
}
getFilterList() {
return [{
"type": "CateFilter",
"type_name": "SelectFilter",
"name": "Category",
"values": [{
"value": "recommended",
"name": "Recommended",
"type_name": "SelectOption"
},
{
"value": "censored",
"name": "Censored",
"type_name": "SelectOption"
},
{
"value": "uncensored",
"name": "Uncensored",
"type_name": "SelectOption"
},
{
"value": "uncensored-leaked",
"name": "Uncensored Leaked",
"type_name": "SelectOption"
},
{
"value": "vr",
"name": "VR",
"type_name": "SelectOption"
},
{
"value": "tags/fc2",
"name": "FC2",
"type_name": "SelectOption"
},
{
"value": "tags/heyzo",
"name": "HEYZO",
"type_name": "SelectOption"
},
{
"value": "tags/tokyo-hot",
"name": "Tokyo-Hot",
"type_name": "SelectOption"
},
{
"value": "tags/1pondo",
"name": "1pondo",
"type_name": "SelectOption"
},
{
"value": "tags/caribbeancom",
"name": "Caribbeancom",
"type_name": "SelectOption"
},
{
"value": "tags/caribbeancompr",
"name": "Caribbeancompr",
"type_name": "SelectOption"
},
{
"value": "tags/10musume",
"name": "10musume",
"type_name": "SelectOption"
},
{
"value": "tags/pacopacomama",
"name": "pacopacomama",
"type_name": "SelectOption"
},
{
"value": "tags/gachig",
"name": "Gachinco",
"type_name": "SelectOption"
},
{
"value": "tags/xxx-av",
"name": "XXX-AV",
"type_name": "SelectOption"
},
{
"value": "tags/c0930",
"name": "C0930",
"type_name": "SelectOption"
},
{
"value": "tags/h4610",
"name": "H4610",
"type_name": "SelectOption"
},
{
"value": "tags/h0930",
"name": "H0930",
"type_name": "SelectOption"
},
{
"value": "tags/siro",
"name": "SIRO",
"type_name": "SelectOption"
},
{
"value": "tags/259luxu",
"name": "LUXU",
"type_name": "SelectOption"
},
{
"value": "tags/200gana",
"name": "200GANA",
"type_name": "SelectOption"
},
{
"value": "tags/prestige-premium",
"name": "PRESTIGE PREMIUM",
"type_name": "SelectOption"
},
{
"value": "tags/s-cute",
"name": "S-CUTE",
"type_name": "SelectOption"
},
{
"value": "tags/261ara",
"name": "ARA",
"type_name": "SelectOption"
}
]
},
{
"type": "SortFilter",
"type_name": "SelectFilter",
"name": "Sort",
"values": [{
"value": "recent_update",
"name": "Recent Update",
"type_name": "SelectOption"
},
{
"value": "release_date",
"name": "Release date",
"type_name": "SelectOption"
},
{
"value": "trending",
"name": "Trending",
"type_name": "SelectOption"
},
{
"value": "most_viewed_today",
"name": "Most viewed today",
"type_name": "SelectOption"
},
{
"value": "most_viewed_week",
"name": "Most viewed by week",
"type_name": "SelectOption"
},
{
"value": "most_viewed_month",
"name": "Most viewed by month",
"type_name": "SelectOption"
},
{
"value": "most_viewed",
"name": "Most viewed",
"type_name": "SelectOption"
}, {
"value": "most_favourited",
"name": "Most favourited",
"type_name": "SelectOption"
}
]
}
];
}
getSourcePreferences() {
return [{
"key": "lang",
"listPreference": {
"title": "Language",
"summary": "",
"valueIndex": 0,
"entries": ["English", "繁體中文", "日本語", "한국의", "Melayu", "ไทย", "Deutsch", "Français", "Tiếng Việt"],
"entryValues": ["en", "zh", "ja", "ko", "ms", "th", "de", "fr", "vi"],
}
},
{
"key": "url",
"listPreference": {
"title": "Website Url",
"summary": "",
"valueIndex": 0,
"entries": ["njav", "missav", "javgo", "supjav"],
"entryValues": ["https://njav.xyz", "https://missav.li", "https://www.javgo.to", "https://supjav.pro"],
}
}
];
}
}

View File

@@ -1,117 +0,0 @@
const mangayomiSources = [{
"name": "VIVAMAXph",
"lang": "all",
"baseUrl": "https://vivamaxph.com/",
"apiUrl": "",
"iconUrl": "https://vivamaxph.com/wp-content/uploads/2024/02/logo2-1.png",
"typeSource": "single",
"isManga": false,
"isNsfw": true,
"version": "0.0.1",
"dateFormat": "",
"dateFormatLocale": "",
"pkgPath": "anime/src/all/vivamaxph.js"
}];
class DefaultExtension extends MProvider {
async getPopular(page) {
const baseUrl = this.source.baseUrl;
const res = await new Client().get(`${baseUrl}/page/${page}/?filter=most-viewed`);
const elements = new Document(res.body).select("article");
const list = [];
for (const element of elements){
const linkElement = element.selectFirst("a");
const name = element.selectFirst("a").attr("title");
const imageUrl = linkElement.selectFirst("img").attr("data-src");
const link = element.selectFirst("a").attr("href");
if (name && imageUrl && link) {
list.push({ name, imageUrl, link });
}
}
return {
list: list,
hasNextPage: true
}
}
async getLatestUpdates(page) {
const baseUrl = this.source.baseUrl;
const res = await new Client().get(`${baseUrl}/page/${page}/?filter=latest`);
const elements = new Document(res.body).select("article");
const list = [];
for (const element of elements){
const linkElement = element.selectFirst("a");
const name = element.selectFirst("a").attr("title");
const imageUrl = linkElement.selectFirst("img").attr("data-src");
const link = element.selectFirst("a").attr("href");
if (name && imageUrl && link) {
list.push({ name, imageUrl, link });
}
}
return {
list: list,
hasNextPage: true
};
}
async search(query, page, filters) {
const baseUrl = this.source.baseUrl;
const res = await new Client().get(`${baseUrl}?s=${query}`);
// console.log(res.body);
const elements = new Document(res.body).select("article");
const list = [];
for (const element of elements){
const linkElement = element.selectFirst("a");
const name = element.selectFirst("a").attr("title");
const imageUrl = linkElement.selectFirst("img").attr("data-src");
const link = element.selectFirst("a").attr("href");
if (name && imageUrl && link) {
list.push({ name, imageUrl, link });
}
}
return {
list: list,
hasNextPage: false
};
}
async getDetail(url) {
const baseUrl = this.source.baseUrl;
const res = await new Client().get(url);
const elements = new Document(res.body);
const movieURL = elements.selectFirst("meta[property='og:url']").attr("content");
const episodes = []
let episode = {
"name": "movie",
"url": `${movieURL}`
}
episodes.push(episode)
const name = elements.selectFirst("h1[itemprop='name']").text
const imageUrl = elements.selectFirst("meta[property='og:image']").attr("content");
return {
name, imageUrl, episodes
};
}
// For anime episode video list
async getVideoList(url) {
const res = await new Client().get(url);
const elements = new Document(res.body);
const videos = []
const redirectgs = elements.selectFirst("iframe").attr("src");
const body = (await new Client().get(redirectgs)).body;
const quality = "DoodStream";
const vids = await doodExtractor(redirectgs, quality);
for (const vid of vids) {
videos.push(vid);
}
return videos;
}
// For manga chapter pages
async getPageList() {
throw new Error("getPageList not implemented");
}
getFilterList() {
throw new Error("getFilterList not implemented");
}
getSourcePreferences() {
throw new Error("getSourcePreferences not implemented");
}
}

View File

@@ -1,179 +0,0 @@
const mangayomiSources = [{
"name": "奥斯卡资源站",
"lang": "zh",
"baseUrl": "https://aosikazy1.com",
"apiUrl": "",
"iconUrl": "https://aosikazy1.com/template/m1938pc3/image/favicon.ico",
"typeSource": "single",
"isManga": false,
"isNsfw": true,
"version": "0.0.1",
"dateFormat": "",
"dateFormatLocale": "",
"pkgPath": "anime/src/zh/aosikazy.js"
}];
class DefaultExtension extends MProvider {
dict = new Map([
[" ", " "],
[""", '"'],
["&lt;", "<"],
["&gt;", ">"],
["&amp;", "&"],
["&sdot;", "·"],
]);
text(content) {
if (!content) return "";
const str =
[...content.matchAll(/>([^<]+?)</g)]
.map((m) => m[1])
.join("")
.trim() || content;
return str.replace(/&[a-z]+;/g, (c) => this.dict.get(c) || c);
}
async request(url) {
const preference = new SharedPreferences();
return (await new Client({ 'useDartHttpClient': true }).get(preference.get("url") + "/api.php/provide/vod?ac=detail" + url, { "Referer": preference.get("url") })).body;
}
getHeaders(url) {
throw new Error("getHeaders not implemented");
}
async getPopular(page) {
// let genres = [];
// const gen = JSON.parse(await this.request("&ac=list"));
// gen.class.forEach((e) => {
// genres.push({
// type_name: "SelectOption",
// value: e.type_id,
// name: e.type_name
// });
// });
// console.log(genres)
const res = JSON.parse(await this.request(`&pg=${page}`));
return {
list: res.list.map((e) => ({
link: "&ids=" + e.vod_id,
imageUrl: e.vod_pic,
name: e.vod_name
})),
hasNextPage: true
};
}
async getLatestUpdates(page) {
const h = (new Date().getUTCHours() + 9) % 24;
const res = JSON.parse(await this.request(`&pg=${page}&h=${h || 24}`));
return {
list: res.list.map((e) => ({
link: "&ids=" + e.vod_id,
imageUrl: e.vod_pic,
name: e.vod_name
})),
hasNextPage: true
};
}
async search(query, page, filters) {
var categories;
for (const filter of filters) {
if (filter["type"] == "categories") {
categories = filter["values"][filter["state"]]["value"];
}
}
const res = JSON.parse(await this.request(`&wd=${query}&t=${categories ?? ""}&pg=${page}`));
return {
list: res.list.map((e) => ({
link: "&ids=" + e.vod_id,
imageUrl: e.vod_pic,
name: e.vod_name
})),
hasNextPage: true
};
}
async getDetail(url) {
let desc = "无";
const anime = JSON.parse(await this.request(url)).list[0];
const blurb = this.text(anime.vod_blurb);
const content = this.text(anime.vod_content);
desc = desc.length < blurb?.length ? blurb : desc;
desc = desc.length < content.length ? content : desc;
const urls = anime.vod_play_url
.split("#")
.filter((e) => e)
.map((e) => {
const s = e.split("$");
return { name: s[0], url: s[1] };
});
return {
name: anime.vod_name,
imageUrl: anime.vod_pic,
description: desc,
episodes: urls
};
}
// For anime episode video list
async getVideoList(url) {
return [{
url: url,
originalUrl: url,
quality: "HLS"
}];
}
// For manga chapter pages
async getPageList() {
throw new Error("getPageList not implemented");
}
getFilterList() {
return [{
type: "categories",
name: "影片類型",
type_name: "SelectFilter",
values: [
// { type_name: "SelectOption", value: "1", name: "电影" },
// { type_name: "SelectOption", value: "2", name: "连续剧" },
// { type_name: "SelectOption", value: "3", name: "综艺" },
{ type_name: "SelectOption", value: "", name: "全部" },
{ type_name: "SelectOption", value: "20", name: "国产视频" },
{ type_name: "SelectOption", value: "21", name: "中文字幕" },
{ type_name: "SelectOption", value: "22", name: "国产传媒" },
{ type_name: "SelectOption", value: "23", name: "日本有码" },
{ type_name: "SelectOption", value: "24", name: "日本无码" },
{ type_name: "SelectOption", value: "25", name: "欧美无码" },
{ type_name: "SelectOption", value: "26", name: "强奸乱伦" },
{ type_name: "SelectOption", value: "27", name: "制服诱惑" },
{ type_name: "SelectOption", value: "28", name: "国产主播" },
{ type_name: "SelectOption", value: "29", name: "激情动漫" },
{ type_name: "SelectOption", value: "30", name: "明星换脸" },
{ type_name: "SelectOption", value: "31", name: "抖阴视频" },
{ type_name: "SelectOption", value: "32", name: "女优明星" },
{ type_name: "SelectOption", value: "33", name: "视频一区" },
{ type_name: "SelectOption", value: "34", name: "视频二区" },
{ type_name: "SelectOption", value: "35", name: "网曝黑料" },
{ type_name: "SelectOption", value: "36", name: "视频三区" },
{ type_name: "SelectOption", value: "37", name: "伦理三级" },
{ type_name: "SelectOption", value: "38", name: "AV解说" },
{ type_name: "SelectOption", value: "39", name: "SM调教" },
{ type_name: "SelectOption", value: "40", name: "萝莉少女" },
{ type_name: "SelectOption", value: "41", name: "极品媚黑" },
{ type_name: "SelectOption", value: "42", name: "女同性恋" },
{ type_name: "SelectOption", value: "43", name: "网红头条" },
{ type_name: "SelectOption", value: "44", name: "视频四区" },
{ type_name: "SelectOption", value: "45", name: "人妖系列" },
{ type_name: "SelectOption", value: "46", name: "韩国主播" },
{ type_name: "SelectOption", value: "47", name: "VR视角" }
]
}];
}
getSourcePreferences() {
return [
{
"key": "url",
"listPreference": {
"title": "Website Url",
"summary": "",
"valueIndex": 0,
"entries": ["aosikazy1", "aosikazy2", "aosikazy3", "aosikazy4", "aosikazy5", "aosikazy6", "aosikazy7", "aosikazy8", "aosikazy9", "aosikazy10"],
"entryValues": ["https://aosikazy1.com", "https://aosikazy2.com", "https://aosikazy3.com", "https://aosikazy4.com", "https://aosikazy5.com", "https://aosikazy6.com", "https://aosikazy7.com", "https://aosikazy8.com", "https://aosikazy9.com", "https://aosikazy10.com"],
}
}
];
}
}

File diff suppressed because it is too large Load Diff

View File

@@ -1,745 +0,0 @@
const mangayomiSources = [{
"name": "Jable",
"lang": "zh",
"baseUrl": "https://jable.tv",
"apiUrl": "",
"iconUrl": "https://assets-cdn.jable.tv/assets/icon/favicon-32x32.png",
"typeSource": "single",
"isManga": false,
"isNsfw": true,
"version": "0.0.2",
"dateFormat": "",
"dateFormatLocale": "",
"pkgPath": "anime/src/zh/jable.js",
"hasCloudflare": true
}];
class DefaultExtension extends MProvider {
async getItems(url) {
const res = await new Client().get(this.source.baseUrl + url);
const doc = new Document(res.body);
const elements = doc.select("div.video-img-box");
const items = [];
for (const element of elements) {
const title = element.selectFirst("h6.title").text;
const cover = element.selectFirst("div.img-box a img").attr("data-src");
const url = element.selectFirst("a").attr("href");
items.push({
name: title,
imageUrl: cover,
link: url
});
}
return {
list: items,
hasNextPage: true
};
}
async getPopular(page) {
return await this.getItems(`/hot/${page}/`);
}
async getLatestUpdates(page) {
return await this.getItems(`/latest-updates/${page}/`);
}
async search(query, page, filters) {
if (query != "") {
return await this.getItems(`/search/${query}/?mode=async&function=get_block&block_id=list_videos_videos_list_search_result&q=${query}&sort_by=&from=${page}`);
}
var categories, sort;
for (const filter of filters) {
if (filter["type"] == "categories") {
categories = filter["values"][filter["state"]]["value"];
}
else if (filter["type"] == "sort") {
sort = filter["values"][filter["state"]]["value"];
}
}
return await this.getItems(`${categories}?mode=async&function=get_block&block_id=list_videos_common_videos_list&sort_by=${sort}&from=${page}`);
}
async getDetail(url) {
const res = await new Client().get(url);
const doc = new Document(res.body);
const title = doc.selectFirst("div.header-left h4").text;
const cover = doc.selectFirst("video#player").attr("poster");
const tags = doc.select("h5.tags a").map(e => e.text);
const actor = doc.selectFirst("div.models span").attr("title");
const vid_url = res.body.match(/hlsUrl\s*=\s*'([^']*)'/)[1];
return {
name: title,
imageUrl: cover,
genre: tags,
author: actor,
description: "",
episodes: [{
name: title,
url: vid_url
}]
};
}
async getVideoList(url) {
return [{
url: url,
originalUrl: url,
quality: "HLS"
}];
}
getFilterList() {
return [{
type: "categories",
name: "主題",
type_name: "SelectFilter",
values: [{
name: "角色劇情",
value: "/categories/roleplay/",
type_name: "SelectOption"
},
{
name: "中文字幕",
value: "/categories/chinese-subtitle/",
type_name: "SelectOption"
},
{
name: "制服誘惑",
value: "/categories/uniform/",
type_name: "SelectOption"
},
{
name: "直接開啪",
value: "/categories/sex-only/",
type_name: "SelectOption"
},
{
name: "絲襪美腿",
value: "/categories/pantyhose/",
type_name: "SelectOption"
},
{
name: "主奴調教",
value: "/categories/bdsm/",
type_name: "SelectOption"
},
{
name: "多P群交",
value: "/categories/groupsex/",
type_name: "SelectOption"
},
{
name: "男友視角",
value: "/categories/pov/",
type_name: "SelectOption"
},
{
name: "凌辱強暴",
value: "/categories/rape/",
type_name: "SelectOption"
},
{
name: "無碼解放",
value: "/categories/uncensored/",
type_name: "SelectOption"
},
{
name: "盜攝偷拍",
value: "/categories/hidden-cam/",
type_name: "SelectOption"
},
{
name: "女同歡愉",
value: "/categories/lesbian/",
type_name: "SelectOption"
},
{
name: "黑絲",
value: "/tags/black-pantyhose/",
type_name: "SelectOption"
},
{
name: "過膝襪",
value: "/tags/knee-socks/",
type_name: "SelectOption"
},
{
name: "運動裝",
value: "/tags/sportswear/",
type_name: "SelectOption"
},
{
name: "肉絲",
value: "/tags/flesh-toned-pantyhose/",
type_name: "SelectOption"
},
{
name: "絲襪",
value: "/tags/pantyhose/",
type_name: "SelectOption"
},
{
name: "眼鏡娘",
value: "/tags/glasses/",
type_name: "SelectOption"
},
{
name: "獸耳",
value: "/tags/kemonomimi/",
type_name: "SelectOption"
},
{
name: "漁網",
value: "/tags/fishnets/",
type_name: "SelectOption"
},
{
name: "水着",
value: "/tags/swimsuit/",
type_name: "SelectOption"
},
{
name: "校服",
value: "/tags/school-uniform/",
type_name: "SelectOption"
},
{
name: "旗袍",
value: "/tags/cheongsam/",
type_name: "SelectOption"
},
{
name: "婚紗",
value: "/tags/wedding-dress/",
type_name: "SelectOption"
},
{
name: "女僕",
value: "/tags/maid/",
type_name: "SelectOption"
},
{
name: "和服",
value: "/tags/kimono/",
type_name: "SelectOption"
},
{
name: "吊帶襪",
value: "/tags/stockings/",
type_name: "SelectOption"
},
{
name: "兔女郎",
value: "/tags/bunny-girl/",
type_name: "SelectOption"
},
{
name: "Cosplay",
value: "/tags/Cosplay/",
type_name: "SelectOption"
},
{
name: "黑肉",
value: "/tags/suntan/",
type_name: "SelectOption"
},
{
name: "長身",
value: "/tags/tall/",
type_name: "SelectOption"
},
{
name: "軟體",
value: "/tags/flexible-body/",
type_name: "SelectOption"
},
{
name: "貧乳",
value: "/tags/small-tits/",
type_name: "SelectOption"
},
{
name: "蘿莉",
value: "/tags/loli/",
type_name: "SelectOption"
},
{
name: "美腿",
value: "/tags/beautiful-leg/",
type_name: "SelectOption"
},
{
name: "美尻",
value: "/tags/beautiful-butt/",
type_name: "SelectOption"
},
{
name: "紋身",
value: "/tags/tattoo/",
type_name: "SelectOption"
},
{
name: "短髮",
value: "/tags/short-hair/",
type_name: "SelectOption"
},
{
name: "白虎",
value: "/tags/hairless-pussy/",
type_name: "SelectOption"
},
{
name: "熟女",
value: "/tags/mature-woman/",
type_name: "SelectOption"
},
{
name: "巨乳",
value: "/tags/big-tits/",
type_name: "SelectOption"
},
{
name: "少女",
value: "/tags/girl/",
type_name: "SelectOption"
},
{
name: "顏射",
value: "/tags/facial/",
type_name: "SelectOption"
},
{
name: "腳交",
value: "/tags/footjob/",
type_name: "SelectOption"
},
{
name: "肛交",
value: "/tags/anal-sex/",
type_name: "SelectOption"
},
{
name: "痙攣",
value: "/tags/spasms/",
type_name: "SelectOption"
},
{
name: "潮吹",
value: "/tags/squirting/",
type_name: "SelectOption"
},
{
name: "深喉",
value: "/tags/deep-throat/",
type_name: "SelectOption"
},
{
name: "接吻",
value: "/tags/kiss/",
type_name: "SelectOption"
},
{
name: "口爆",
value: "/tags/cum-in-mouth/",
type_name: "SelectOption"
},
{
name: "口交",
value: "/tags/blowjob/",
type_name: "SelectOption"
},
{
name: "乳交",
value: "/tags/tit-wank/",
type_name: "SelectOption"
},
{
name: "中出",
value: "/tags/creampie/",
type_name: "SelectOption"
},
{
name: "露出",
value: "/tags/outdoor/",
type_name: "SelectOption"
},
{
name: "輪姦",
value: "/tags/gang-rape/",
type_name: "SelectOption"
},
{
name: "調教",
value: "/tags/tune/",
type_name: "SelectOption"
},
{
name: "綑綁",
value: "/tags/bondage/",
type_name: "SelectOption"
},
{
name: "瞬間插入",
value: "/tags/quickie/",
type_name: "SelectOption"
},
{
name: "痴漢",
value: "/tags/chikan/",
type_name: "SelectOption"
},
{
name: "痴女",
value: "/tags/chizyo/",
type_name: "SelectOption"
},
{
name: "男M",
value: "/tags/masochism-guy/",
type_name: "SelectOption"
},
{
name: "泥醉",
value: "/tags/crapulence/",
type_name: "SelectOption"
},
{
name: "泡姬",
value: "/tags/soapland/",
type_name: "SelectOption"
},
{
name: "母乳",
value: "/tags/breast-milk/",
type_name: "SelectOption"
},
{
name: "放尿",
value: "/tags/piss/",
type_name: "SelectOption"
},
{
name: "按摩",
value: "/tags/massage/",
type_name: "SelectOption"
},
{
name: "強姦",
value: "/tags/rape/",
type_name: "SelectOption"
},
{
name: "多P",
value: "/tags/gangbang/",
type_name: "SelectOption"
},
{
name: "刑具",
value: "/tags/torture/",
type_name: "SelectOption"
},
{
name: "凌辱",
value: "/tags/insult/",
type_name: "SelectOption"
},
{
name: "一日十回",
value: "/tags/10-times-a-day/",
type_name: "SelectOption"
},
{
name: "3P",
value: "/tags/3p/",
type_name: "SelectOption"
},
{
name: "黑人",
value: "/tags/black/",
type_name: "SelectOption"
},
{
name: "醜男",
value: "/tags/ugly-man/",
type_name: "SelectOption"
},
{
name: "誘惑",
value: "/tags/temptation/",
type_name: "SelectOption"
},
{
name: "童貞",
value: "/tags/virginity/",
type_name: "SelectOption"
},
{
name: "時間停止",
value: "/tags/time-stop/",
type_name: "SelectOption"
},
{
name: "復仇",
value: "/tags/avenge/",
type_name: "SelectOption"
},
{
name: "年齡差",
value: "/tags/age-difference/",
type_name: "SelectOption"
},
{
name: "巨漢",
value: "/tags/giant/",
type_name: "SelectOption"
},
{
name: "媚藥",
value: "/tags/love-potion/",
type_name: "SelectOption"
},
{
name: "夫目前犯",
value: "/tags/sex-beside-husband/",
type_name: "SelectOption"
},
{
name: "出軌",
value: "/tags/affair/",
type_name: "SelectOption"
},
{
name: "催眠",
value: "/tags/hypnosis/",
type_name: "SelectOption"
},
{
name: "偷拍",
value: "/tags/hidden-cam/",
type_name: "SelectOption"
},
{
name: "不倫",
value: "/tags/incest/",
type_name: "SelectOption"
},
{
name: "下雨天",
value: "/tags/rainy-day/",
type_name: "SelectOption"
},
{
name: "NTR",
value: "/tags/ntr/",
type_name: "SelectOption"
},
{
name: "風俗娘",
value: "/tags/club-hostess-and-sex-worker/",
type_name: "SelectOption"
},
{
name: "醫生",
value: "/tags/doctor/",
type_name: "SelectOption"
},
{
name: "逃犯",
value: "/tags/fugitive/",
type_name: "SelectOption"
},
{
name: "護士",
value: "/tags/nurse/",
type_name: "SelectOption"
},
{
name: "老師",
value: "/tags/teacher/",
type_name: "SelectOption"
},
{
name: "空姐",
value: "/tags/flight-attendant/",
type_name: "SelectOption"
},
{
name: "球隊經理",
value: "/tags/team-manager/",
type_name: "SelectOption"
},
{
name: "未亡人",
value: "/tags/widow/",
type_name: "SelectOption"
},
{
name: "搜查官",
value: "/tags/detective/",
type_name: "SelectOption"
},
{
name: "情侶",
value: "/tags/couple/",
type_name: "SelectOption"
},
{
name: "家政婦",
value: "/tags/housewife/",
type_name: "SelectOption"
},
{
name: "家庭教師",
value: "/tags/private-teacher/",
type_name: "SelectOption"
},
{
name: "偶像",
value: "/tags/idol/",
type_name: "SelectOption"
},
{
name: "人妻",
value: "/tags/wife/",
type_name: "SelectOption"
},
{
name: "主播",
value: "/tags/female-anchor/",
type_name: "SelectOption"
},
{
name: "OL",
value: "/tags/ol/",
type_name: "SelectOption"
},
{
name: "魔鏡號",
value: "/tags/magic-mirror/",
type_name: "SelectOption"
},
{
name: "電車",
value: "/tags/tram/",
type_name: "SelectOption"
},
{
name: "處女",
value: "/tags/first-night/",
type_name: "SelectOption"
},
{
name: "監獄",
value: "/tags/prison/",
type_name: "SelectOption"
},
{
name: "溫泉",
value: "/tags/hot-spring/",
type_name: "SelectOption"
},
{
name: "洗浴場",
value: "/tags/bathing-place/",
type_name: "SelectOption"
},
{
name: "泳池",
value: "/tags/swimming-pool/",
type_name: "SelectOption"
},
{
name: "汽車",
value: "/tags/car/",
type_name: "SelectOption"
},
{
name: "廁所",
value: "/tags/toilet/",
type_name: "SelectOption"
},
{
name: "學校",
value: "/tags/school/",
type_name: "SelectOption"
},
{
name: "圖書館",
value: "/tags/library/",
type_name: "SelectOption"
},
{
name: "健身房",
value: "/tags/gym-room/",
type_name: "SelectOption"
},
{
name: "便利店",
value: "/tags/store/",
type_name: "SelectOption"
},
{
name: "錄像",
value: "/tags/video-recording/",
type_name: "SelectOption"
},
{
name: "處女作/引退作",
value: "/tags/debut-retires/",
type_name: "SelectOption"
},
{
name: "綜藝",
value: "/tags/variety-show/",
type_name: "SelectOption"
},
{
name: "節日主題",
value: "/tags/festival/",
type_name: "SelectOption"
},
{
name: "感謝祭",
value: "/tags/thanksgiving/",
type_name: "SelectOption"
},
{
name: "4小時以上",
value: "/tags/more-than-4-hours/",
type_name: "SelectOption"
}
]
},
{
type: "sort",
name: "排序",
type_name: "SelectFilter",
values: [{
name: "近期最佳",
value: "post_date_and_popularity",
type_name: "SelectOption"
},
{
name: "最近更新",
value: "post_date",
type_name: "SelectOption"
},
{
name: "最多觀看",
value: "video_viewed",
type_name: "SelectOption"
},
{
name: "最高收藏",
value: "most_favourited",
type_name: "SelectOption"
}
]
}]
}
getSourcePreferences() {
throw new Error("getSourcePreferences not implemented");
}
}

View File

@@ -1,256 +0,0 @@
const mangayomiSources = [{
"name": "禁漫天堂",
"lang": "zh",
"baseUrl": "https://18comic.vip",
"apiUrl": "",
"iconUrl": "https://cdn-msp.jmcomic.me/media/logo/new_logo.png?v=2024043002",
"typeSource": "single",
"isManga": true,
"isNsfw": true,
"version": "0.0.15",
"dateFormat": "",
"dateFormatLocale": "",
"pkgPath": "manga/src/zh/jmcomic.js"
}];
class DefaultExtension extends MProvider {
dateStringToTimestamp(dateString) {
var parts = dateString.split('-');
var year = parseInt(parts[0]);
var month = parseInt(parts[1]) - 1;
var day = parseInt(parts[2]);
var date = new Date(year, month, day);
var timestamp = date.getTime();
return timestamp;
}
getHeaders(url) {
throw new Error("getHeaders not implemented");
}
async getManga(url, p) {
const res = await new Client().get(this.source.baseUrl + url, {
Referer: this.source.baseUrl
});
const doc = new Document(res.body);
const manga = [];
const elements = doc.select(p);
for (const element of elements) {
var text = element.innerHtml;
text = text.slice(text.search("<noscript>"), -1);
const title = text.match(/title="(.*)" alt=/)[1];
var cover = text.match(/data-original="(.*)" title=/);
if (cover == null) {
cover = text.match(/src="(.*)" title=/);
}
const url = element.innerHtml.match(/<a href="(.*)"/)[1];
manga.push({
name: title,
link: url,
imageUrl: cover[1]
});
}
return {
list: manga,
hasNextPage: true
};
}
async getPopular(page) {
return await this.getManga(`/albums?t=a&o=mv&page=${page}`, "div.thumb-overlay-albums");
}
async getLatestUpdates(page) {
return await this.getManga(`/albums?t=a&o=mr&page=${page}`, "div.thumb-overlay-albums");
}
async search(query, page, filters) {
var type, time, sort;
for (const filter of filters) {
if (filter["type"] == "type") {
type = filter["values"][filter["state"]]["value"];
}
if (filter["type"] == "time") {
time = filter["values"][filter["state"]]["value"];
}
if (filter["type"] == "sort") {
sort = filter["values"][filter["state"]]["value"];
}
}
if (query != "") {
return await this.getManga(`/search/photos?search_query=${query}&t=a&o=mr`, "div.thumb-overlay");
}
return await this.getManga(`/albums${"/"+type}?t=${time}&o=${sort}&page=${page}`, "div.thumb-overlay-albums");
}
async getDetail(url) {
const res = await new Client().get(this.source.baseUrl + url, {
Referer: this.source.baseUrl
});
const doc = new Document(res.body);
const cover = doc.selectFirst("div.show_zoom").innerHtml.match(/data-cfsrc="(.*)" data-cfstyle/)[1];
const title = doc.selectFirst("h1").text;
const desc = doc.selectFirst("div#intro-block div.p-t-5").text;
const infos = doc.select("div#intro-block div.tag-block");
const tags = infos[2].select("a").map(e => e.text);
const author = infos[3].selectFirst("a").text;
const date_str = res.body.match(/更新日期 : (.*)\n<\/p>/)[1];
const chapters = [];
const elements = doc.selectFirst("div.episode ul").select("a");
if (elements.length == 0) {
chapters.push({
name: title,
dateUpload: this.dateStringToTimestamp(date_str).toString(),
url: url.replace("album", "photo")
});
} else {
for (const element of elements) {
const url = element.attr("href");
const title = element.selectFirst("li").text;
const date = element.selectFirst("span.hidden-xs").text;
chapters.push({
name: title.split("\n")[1],
dateUpload: this.dateStringToTimestamp(date).toString(),
url: url
});
}
}
chapters.reverse();
return {
name: title,
imageUrl: cover,
description: desc,
genre: tags,
author: author,
episodes: chapters
};
}
async getPageList(url) {
const res = await new Client().get(this.source.baseUrl + url);
const doc = new Document(res.body);
const elements = doc.select("div.scramble-page");
const pages = [];
for (const element of elements) {
var text = element.innerHtml;
text = text.slice(text.search("<noscript>"), -1);
const img = text.match(/data-original="(.*)" id/);
if (img != null) {
pages.push(img[1]);
}
}
return pages;
}
getFilterList() {
return [{
type: "sort",
name: "排序",
type_name: "SelectFilter",
values: [{
type_name: "SelectOption",
name: "最新",
value: "mr"
},
{
type_name: "SelectOption",
name: "最多订阅",
value: "mv"
},
{
type_name: "SelectOption",
name: "最多图片",
value: "mp"
},
{
type_name: "SelectOption",
name: "最高评分",
value: "tr"
},
{
type_name: "SelectOption",
name: "最多评论",
value: "md"
},
{
type_name: "SelectOption",
name: "最多爱心",
value: "tf"
},
]
}, {
type: "time",
name: "时间",
type_name: "SelectFilter",
values: [{
type_name: "SelectOption",
name: "全部",
value: "a"
},
{
type_name: "SelectOption",
name: "今天",
value: "t"
},
{
type_name: "SelectOption",
name: "这周",
value: "w"
},
{
type_name: "SelectOption",
name: "本月",
value: "m"
},
]
},
{
type: "type",
name: "分类",
type_name: "SelectFilter",
values: [{
type_name: "SelectOption",
name: "全部",
value: ""
},
{
type_name: "SelectOption",
name: "其他类",
value: "another"
},
{
type_name: "SelectOption",
name: "同人",
value: "doujin"
},
{
type_name: "SelectOption",
name: "韩漫",
value: "hanman"
},
{
type_name: "SelectOption",
name: "美漫",
value: "meiman"
},
{
type_name: "SelectOption",
name: "短篇",
value: "short"
},
{
type_name: "SelectOption",
name: "单本",
value: "single"
},
]
}
];
}
getSourcePreferences() {
throw new Error("getSourcePreferences not implemented");
}
}

View File

@@ -1,431 +0,0 @@
const mangayomiSources = [{
"name": "绅士漫画",
"lang": "zh",
"baseUrl": "https://www.wnacg.com",
"apiUrl": "",
"iconUrl": "https://www.wnacg.com/favicon.ico",
"typeSource": "single",
"isManga": true,
"isNsfw": true,
"version": "0.0.2",
"dateFormat": "",
"dateFormatLocale": "",
"pkgPath": "manga/src/zh/wnacg.js"
}];
class DefaultExtension extends MProvider {
url_strs = {
"dojinshi": {
"all": "albums-index-page-$-cate-5.html",
"chi": "albums-index-page-$-cate-1.html",
"jpa": "albums-index-page-$-cate-12.html",
"eng": "albums-index-page-$-cate-16.html"
},
"tankobon": {
"all": "albums-index-page-$-cate-6.html",
"chi": "albums-index-page-$-cate-9.html",
"jpa": "albums-index-page-$-cate-13.html",
"eng": "albums-index-page-$-cate-17.html"
},
"magazine": {
"all": "albums-index-page-$-cate-7.html",
"chi": "albums-index-page-$-cate-10.html",
"jpa": "albums-index-page-$-cate-14.html",
"eng": "albums-index-page-$-cate-18.html"
},
"korea": {
"all": "albums-index-page-$-cate-19.html",
"chi": "albums-index-page-$-cate-20.html",
"oth": "albums-index-page-$-cate-21.html"
},
"cg": {
"all": "albums-index-page-$-cate-2.html"
},
"cosplay": {
"all": "albums-index-page-$-cate-3.html"
},
"thridd": {
"all": "albums-index-page-$-cate-22.html"
},
"home": {
"all": ""
}
};
dateStringToTimestamp(dateString) {
var parts = dateString.split('-');
var year = parseInt(parts[0]);
var month = parseInt(parts[1]) - 1;
var day = parseInt(parts[2]);
var date = new Date(year, month, day);
var timestamp = date.getTime();
return timestamp;
}
async request(url_str, cookies) {
const headers = {
"User-Agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/118.0.0.0 Safari/537.36",
"Referer": this.source.baseUrl
};
if (cookies) {
const preference = new SharedPreferences();
headers["Cookie"] = `MPIC_bnS5=${preference.get("cookies")}`;
}
const res = await new Client().get(`${this.source.baseUrl}/${url_str}`, headers);
return res;
}
async getFavorites(page) {
const res = await this.request(`users-users_fav-page-${page}-c-0.html`, true);
const doc = new Document(res.body);
const elements = doc.select("div.asTB");
const mangas = [];
for (const element of elements) {
const url = element.selectFirst("div.box_cel p.l_title a").attr("href");
const title = element.selectFirst("div.box_cel p.l_title a").text;
const cover = "https:" + element.selectFirst("img").attr("src");
mangas.push({
"name": title,
"link": url,
"imageUrl": cover
});
}
return {
"list": mangas,
"hasNextPage": true
};
}
async getMangas(url_str, page) {
url_str = url_str.replace("$", page.toString());
const res = await this.request(url_str, false);
const doc = new Document(res.body);
const elements = doc.select("li.gallary_item");
const mangas = [];
for (const element of elements) {
const url = element.selectFirst("div.pic_box a").attr("href");
let title = element.selectFirst("div.pic_box a").attr("title");
title = title.replaceAll("<em>", "").replaceAll("</em>", "");
const cover = "https:" + element.selectFirst("div.pic_box a img").attr("src");
mangas.push({
"name": title,
"link": url,
"imageUrl": cover
});
}
return {
"list": mangas,
"hasNextPage": true
};
}
async getPopular(page) {
const result = await this.getMangas("", page);
result["hasNextPage"] = false;
return result;
}
async getLatestUpdates(page) {
return await this.getMangas("albums-index-page-$.html", page);
}
async search(query, page, filters) {
var url_str;
var jump = false;
if (query == "") {
var category;
var language;
var jump_page;
var djs_lang, tkb_lang, mgz_lang, kr_lang;
for (const filter of filters) {
if (filter["type"] == "CateFilter") {
category = filter["values"][filter["state"]]["value"];
}
else if (filter["type"] == "LangFilter") {
for (const lang_filter of filter["state"]) {
if (lang_filter["type"] == "DJSFilter") {
djs_lang = lang_filter["values"][lang_filter["state"]]["value"];
} else if (lang_filter["type"] == "TKBFilter") {
tkb_lang = lang_filter["values"][lang_filter["state"]]["value"];
} else if (lang_filter["type"] == "MGZFilter") {
mgz_lang = lang_filter["values"][lang_filter["state"]]["value"];
} else if (lang_filter["type"] == "KRFilter") {
kr_lang = lang_filter["values"][lang_filter["state"]]["value"];
}
}
}
else if (filter["type"] == "PageFilter") {
jump_page = parseInt(filter["state"]);
}
}
if (jump_page != 1) {
page = jump_page;
jump = true;
}
if (category == "dojinshi") {
language = djs_lang;
} else if (category == "tankobon") {
language = tkb_lang;
} else if (category == "magazine") {
language = mgz_lang;
} else if (category == "korea") {
language = kr_lang;
} else if (category == "fav") {
return await this.getFavorites(page);
} else {
language = "all";
}
url_str = this.url_strs[category][language];
} else {
url_str = `search/index.php?q=${query}&m=&syn=yes&f=_all&s=create_time_DESC&p=$`;
}
const result = await this.getMangas(url_str, page);
result["hasNextPage"] = !jump;
return result;
}
async getDetail(url) {
const res = await this.request(url.slice(1), false);
const doc = new Document(res.body);
const title = doc.selectFirst("h2").text;
var cover = doc.selectFirst("div.uwthumb img").attr("src");
if (cover[3] == "/") {
cover = "https:" + cover.substring(2, cover.length);
} else {
cover = "https:" + cover;
}
const desc_ = doc.select("div.uwconn label").map(e => e.text);
const desc = desc_.join("\n").replaceAll("+TAG\n", "").slice(0, -1) + doc.selectFirst("div.uwconn p").text;
const id = url.match(/-aid-(.+?).html/)[1];
const uploader = doc.selectFirst("div.uwuinfo p").text;
const tags = doc.select("div.addtags a.tagshow").map(e => e.text);
const uploaddate = this.dateStringToTimestamp(doc.selectFirst("div.info_col").text.replace("上傳於", ""));
return {
name: title,
imageUrl: cover,
description: desc,
author: uploader,
status: 1,
genre: tags,
episodes: [{
name: title,
url: `photos-gallery-aid-${id}.html`,
"dateUpload": uploaddate.toString()
}]
}
}
async getPageList(url) {
const res = await this.request(url, false);
const html = res.body;
const urls = [];
let urls_str = html.substring(html.search("imglist") + 12, html.search("喜歡紳士漫畫的同學請加入收藏哦!") + 17);
const url_list = urls_str.split("},{");
for (let url_str of url_list) {
urls.push("https:" + url_str.substring(url_str.search("img_host") + 11, url_str.search("\", ") - 1));
}
return urls.slice(0, -1);
}
getFilterList() {
return [{
"type": "HeaderFilter",
"name": "选择类别",
"type_name": "HeaderFilter"
},
{
"type": "CateFilter",
"type_name": "SelectFilter",
"name": "分类",
"values": [{
"value": "home",
"name": "主页",
"type_name": "SelectOption"
},
{
"value": "dojinshi",
"name": "同人志",
"type_name": "SelectOption"
},
{
"value": "tankobon",
"name": "单行本",
"type_name": "SelectOption"
},
{
"value": "magazine",
"name": "杂志&短篇",
"type_name": "SelectOption"
},
{
"value": "korea",
"name": "韩漫",
"type_name": "SelectOption"
},
{
"value": "cg",
"name": "CG畫集",
"type_name": "SelectOption"
},
{
"value": "cosplay",
"name": "Cosplay",
"type_name": "SelectOption"
},
{
"value": "thridd",
"name": "3D漫畫",
"type_name": "SelectOption"
},
{
"value": "fav",
"name": "收藏",
"type_name": "SelectOption"
}
]
},
{
"type": "SeparatorFilter",
"type_name": "SeparatorFilter"
},
{
"type": "HeaderFilter",
"name": "根据类别选择语言",
"type_name": "HeaderFilter"
},
{
"type": "LangFilter",
"name": "语言",
"type_name": "GroupFilter",
"state": [{
"type": "DJSFilter",
"type_name": "SelectFilter",
"name": "同人志",
"values": [{
"value": "all",
"name": "全部",
"type_name": "SelectOption"
},
{
"value": "chi",
"name": "漢化",
"type_name": "SelectOption"
},
{
"value": "jpa",
"name": "日語",
"type_name": "SelectOption"
},
{
"value": "eng",
"name": "English",
"type_name": "SelectOption"
}
]
},
{
"type": "TKBFilter",
"type_name": "SelectFilter",
"name": "单行本",
"values": [{
"value": "all",
"name": "全部",
"type_name": "SelectOption"
},
{
"value": "chi",
"name": "漢化",
"type_name": "SelectOption"
},
{
"value": "jpa",
"name": "日語",
"type_name": "SelectOption"
},
{
"value": "eng",
"name": "English",
"type_name": "SelectOption"
}
]
},
{
"type": "MGZFilter",
"type_name": "SelectFilter",
"name": "杂志&短篇",
"values": [{
"value": "all",
"name": "全部",
"type_name": "SelectOption"
},
{
"value": "chi",
"name": "漢化",
"type_name": "SelectOption"
},
{
"value": "jpa",
"name": "日語",
"type_name": "SelectOption"
},
{
"value": "eng",
"name": "English",
"type_name": "SelectOption"
}
]
},
{
"type": "KRFilter",
"type_name": "SelectFilter",
"name": "韩漫",
"values": [{
"value": "all",
"name": "全部",
"type_name": "SelectOption"
},
{
"value": "chi",
"name": "漢化",
"type_name": "SelectOption"
},
{
"value": "oth",
"name": "其他",
"type_name": "SelectOption"
}
]
},
]
}, {
"type": "SeparatorFilter",
"type_name": "SeparatorFilter"
},
{
"type": "HeaderFilter",
"name": "跳转页数",
"type_name": "HeaderFilter"
},
{
"type": "PageFilter",
"name": "页数",
"type_name": "TextFilter",
"state": "1"
}
];
}
getSourcePreferences() {
return [{
"key": "cookies",
"editTextPreference": {
"title": "用户Cookies",
"summary": "用于读取用户收藏的CookiesMPIC_bnS5",
"value": "",
"dialogTitle": "Cookies",
"dialogMessage": "",
}
}];
}
}