Files
venera-configs/baozi.js
2024-10-18 13:16:27 +08:00

318 lines
12 KiB
JavaScript
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

class Baozi extends ComicSource {
// 此漫画源的名称
name = "包子漫画"
// 唯一标识符
key = "baozi"
version = "1.0.0"
minAppVersion = "1.0.0"
// 更新链接
url = "https://raw.githubusercontent.com/venera-app/venera-configs/refs/heads/main/baozi.js"
/// APP启动时或者添加/更新漫画源时执行此函数
init() {}
/// 账号
/// 设置为null禁用账号功能
account = {
/// 登录
/// 返回任意值表示登录成功
login: async (account, pwd) => {
let res = await Network.post("https://cn.baozimh.com/api/bui/signin", {
'content-type': 'multipart/form-data; boundary=----WebKitFormBoundaryFUNUxpOwyUaDop8s'
}, "------WebKitFormBoundaryFUNUxpOwyUaDop8s\r\nContent-Disposition: form-data; name=\"username\"\r\n\r\n" + account + "\r\n------WebKitFormBoundaryFUNUxpOwyUaDop8s\r\nContent-Disposition: form-data; name=\"password\"\r\n\r\n" + pwd + "\r\n------WebKitFormBoundaryFUNUxpOwyUaDop8s--\r\n")
if(res.status !== 200) {
throw "Invalid status code: " + res.status
}
let json = JSON.parse(res.body)
let token = json.data
Network.setCookies(this.baseUrl, [
Cookie('TSID', token, '.baozimh.com'),
])
return 'ok'
},
// 退出登录时将会调用此函数
logout: () => {
Network.deleteCookies('.baozimh.com')
},
registerWebsite: `https://cn.baozimh.com/user/signup`
}
lang = 'cn'
get baseUrl(){
return `https://${this.lang}.baozimh.com`
}
parseComic(e) {
let url = e.querySelector("a").attributes['href']
let id = url.split("/").pop()
let title = e.querySelector("h3").text.trim()
let cover = e.querySelector("a > amp-img").attributes["src"]
let tags = e.querySelectorAll("div.tabs > span").map(e => e.text.trim())
let description = e.querySelector("small").text.trim()
return {
id: id,
title: title,
cover: cover,
tags: tags,
description: description
}
}
parseJsonComic(e) {
return {
id: e.comic_id,
title: e.name,
subTitle: e.author,
cover: `https://static-tw.baozimh.com/cover/${e.topic_img}?w=285&h=375&q=100`,
tags: e.type_names,
}
}
/// 探索页面
/// 一个漫画源可以有多个探索页面
explore = [
{
/// 标题
/// 标题同时用作标识符, 不能重复
title: "包子漫画",
/// singlePageWithMultiPart 或者 multiPageComicList
type: "singlePageWithMultiPart",
load: async () => {
var res = await Network.get(this.baseUrl)
if (res.status !== 200) {
throw "Invalid status code: " + res.status
}
let document = new HtmlDocument(res.body)
let parts = document.querySelectorAll("div.index-recommend-items")
let result = {}
for (let part of parts) {
let title = part.querySelector("div.catalog-title").text.trim()
let comics = part.querySelectorAll("div.comics-card").map(e => this.parseComic(e))
if(comics.length > 0) {
result[title] = comics
}
}
return result
}
}
]
/// 分类页面
/// 一个漫画源只能有一个分类页面, 也可以没有, 设置为null禁用分类页面
category = {
/// 标题, 同时为标识符, 不能与其他漫画源的分类页面重复
title: "包子漫画",
parts: [
{
name: "类型",
// fixed 或者 random
// random用于分类数量相当多时, 随机显示其中一部分
type: "fixed",
// 如果类型为random, 需要提供此字段, 表示同时显示的数量
// randomNumber: 5,
categories: ['全部', '恋爱', '纯爱', '古风', '异能', '悬疑', '剧情', '科幻', '奇幻', '玄幻', '穿越', '冒险', '推理', '武侠', '格斗', '战争', '热血', '搞笑', '大女主', '都市', '总裁', '后宫', '日常', '韩漫', '少年', '其它'],
// category或者search
// 如果为category, 点击后将进入分类漫画页面, 使用下方的`categoryComics`加载漫画
// 如果为search, 将进入搜索页面
itemType: "category",
// 若提供, 数量需要和`categories`一致, `categoryComics.load`方法将会收到此参数
categoryParams: ['all', 'lianai', 'chunai', 'gufeng', 'yineng', 'xuanyi', 'juqing', 'kehuan', 'qihuan', 'xuanhuan', 'chuanyue', 'mouxian', 'tuili', 'wuxia', 'gedou', 'zhanzheng', 'rexie', 'gaoxiao', 'danuzhu', 'dushi', 'zongcai', 'hougong', 'richang', 'hanman', 'shaonian', 'qita']
}
],
enableRankingPage: false,
}
/// 分类漫画页面, 即点击分类标签后进入的页面
categoryComics = {
load: async (category, param, options, page) => {
let res = await Network.get(`${this.baseUrl}/api/bzmhq/amp_comic_list?type=${param}&region=${options[0]}&state=${options[1]}&filter=%2a&page=${page}&limit=36&language=${this.lang}&__amp_source_origin=https%3A%2F%2F${this.lang}.baozimh.com`)
if (res.status !== 200) {
throw "Invalid status code: " + res.status
}
let maxPage = null
let json = JSON.parse(res.body)
if(!json.next) {
maxPage = page
}
return {
comics: json.items.map(e => this.parseJsonComic(e)),
maxPage: maxPage
}
},
// 提供选项
optionList: [
{
options: [
"all-全部",
"cn-国漫",
"jp-日本",
"kr-韩国",
"en-欧美",
],
},
{
options: [
"all-全部",
"serial-连载中",
"pub-已完结",
],
},
],
}
/// 搜索
search = {
load: async (keyword, options, page) => {
let res = await Network.get(`${this.baseUrl}/search?q=${keyword}`)
if (res.status !== 200) {
throw "Invalid status code: " + res.status
}
let document = new HtmlDocument(res.body)
let comics = document.querySelectorAll("div.comics-card").map(e => this.parseComic(e))
return {
comics: comics,
maxPage: 1
}
},
// 提供选项
optionList: []
}
/// 收藏
favorites = {
/// 是否为多收藏夹
multiFolder: false,
/// 添加或者删除收藏
addOrDelFavorite: async (comicId, folderId, isAdding) => {
if(!isAdding) {
let res = await Network.post(`${this.baseUrl}/user/operation_v2?op=del_bookmark&comic_id=${comicId}`)
if (!res.status || res.status >= 400) {
throw "Invalid status code: " + res.status
}
return 'ok'
} else {
let res = await Network.post(`${this.baseUrl}/user/operation_v2?op=set_bookmark&comic_id=${comicId}&chapter_slot=0`)
if (!res.status || res.status >= 400) {
throw "Invalid status code: " + res.status
}
return 'ok'
}
},
// 加载收藏夹, 仅当multiFolder为true时有效
// 当comicId不为null时, 需要同时返回包含该漫画的收藏夹
loadFolders: null,
/// 加载漫画
loadComics: async (page, folder) => {
let res = await Network.get(`${this.baseUrl}/user/my_bookshelf`)
if (res.status !== 200) {
throw "Invalid status code: " + res.status
}
let document = new HtmlDocument(res.body)
function parseComic(e) {
let title = e.querySelector("h4 > a").text.trim()
let url = e.querySelector("h4 > a").attributes['href']
let id = url.split("/").pop()
let author = e.querySelector("div.info > ul").children[1].text.split("")[1].trim()
let description = e.querySelector("div.info > ul").children[4].children[0].text.trim()
return {
id: id,
title: title,
subTitle: author,
description: description,
cover: e.querySelector("amp-img").attributes['src']
}
}
let comics = document.querySelectorAll("div.bookshelf-items").map(e => parseComic(e))
return {
comics: comics,
maxPage: 1
}
}
}
/// 单个漫画相关
comic = {
// 加载漫画信息
loadInfo: async (id) => {
let res = await Network.get(`${this.baseUrl}/comic/${id}`)
if (res.status !== 200) {
throw "Invalid status code: " + res.status
}
let document = new HtmlDocument(res.body)
let title = document.querySelector("h1.comics-detail__title").text.trim()
let cover = document.querySelector("div.l-content > div > div > amp-img").attributes['src']
let author = document.querySelector("h2.comics-detail__author").text.trim()
let tags = document.querySelectorAll("div.tag-list > span").map(e => e.text.trim())
let updateTime = document.querySelector("div.supporting-text > div > span > em").text.trim().replace('(', '').replace(')', '')
let description = document.querySelector("p.comics-detail__desc").text.trim()
let chapters = new Map()
let i = 0
for(let c of document.querySelectorAll("div#chapter-items > div.comics-chapters > a > div > span")) {
chapters.set(i.toString(), c.text.trim())
i++
}
for(let c of document.querySelectorAll("div#chapters_other_list > div.comics-chapters > a > div > span")) {
chapters.set(i.toString(), c.text.trim())
i++
}
let recommend = []
for(let c of document.querySelectorAll("div.recommend--item")) {
if(c.querySelectorAll("div.tag-comic").length > 0) {
let title = c.querySelector("span").text.trim()
let cover = c.querySelector("amp-img").attributes['src']
let url = c.querySelector("a").attributes['href']
let id = url.split("/").pop()
recommend.push({
id: id,
title: title,
cover: cover
})
}
}
return {
title: title,
cover: cover,
description: description,
tags: {
"作者": [author],
"更新": [updateTime],
"标签": tags
},
chapters: chapters,
recommend: recommend
}
},
// 获取章节图片
loadEp: async (comicId, epId) => {
let res = await Network.get(`https://${this.lang}.czmanga.com/comic/chapter/${comicId}/0_${epId}.html`)
if (res.status !== 200) {
throw "Invalid status code: " + res.status
}
let document = new HtmlDocument(res.body)
let images = document.querySelectorAll("ul.comic-contain > div > amp-img").map(e => e.attributes['src'])
console.log(images)
return {
images: images
}
},
}
}