class Baozi extends ComicSource { // 此漫画源的名称 name = "包子漫画" // 唯一标识符 key = "baozi" version = "1.0.2" minAppVersion = "1.0.0" // 更新链接 url = "https://cdn.jsdelivr.net/gh/venera-app/venera-configs@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, [ new Cookie({name: 'TSID', value: token, domain: '.baozimh.com'}), ]) return 'ok' }, // 退出登录时将会调用此函数 logout: () => { Network.deleteCookies('.baozimh.com') }, registerWebsite: `https://cn.baozimh.com/user/signup` } get baseUrl(){ return `https://www.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}®ion=${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://www.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 } }, } }