mirror of
https://github.com/kodjodevf/mangayomi-extensions.git
synced 2026-02-14 10:51:17 +00:00
remove some NSFW sources
This commit is contained in:
@@ -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"],
|
||||
}
|
||||
}
|
||||
];
|
||||
}
|
||||
}
|
||||
@@ -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");
|
||||
}
|
||||
}
|
||||
@@ -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([
|
||||
[" ", " "],
|
||||
[""", '"'],
|
||||
["<", "<"],
|
||||
[">", ">"],
|
||||
["&", "&"],
|
||||
["⋅", "·"],
|
||||
]);
|
||||
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
@@ -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");
|
||||
}
|
||||
}
|
||||
@@ -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");
|
||||
}
|
||||
}
|
||||
@@ -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": "用于读取用户收藏的Cookies(MPIC_bnS5)",
|
||||
"value": "",
|
||||
"dialogTitle": "Cookies",
|
||||
"dialogMessage": "",
|
||||
}
|
||||
}];
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user