-
Notifications
You must be signed in to change notification settings - Fork 142
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Showing
1 changed file
with
152 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,152 @@ | ||
// ==MiruExtension== | ||
// @name 哔哩轻小说 | ||
// @version v0.0.3 | ||
// @author hualiang | ||
// @lang zh | ||
// @icon https://www.bilinovel.com/favicon.ico | ||
// @license MIT | ||
// @package bilinovel.com | ||
// @type fikushon | ||
// @webSite https://www.bilinovel.com | ||
// ==/MiruExtension== | ||
export default class extends Extension { | ||
dict = { "“": "「", "”": "」", "‘": "『", "’": "』", "": "的", "": "一", "": "是", "": "了", "": "我", "": "不", "": "人", | ||
"": "在", "": "他", "": "有", "": "这", "": "个", "": "上", "": "们", "": "来", "": "到", "": "时", "": "大", | ||
"": "地", "": "为", "": "子", "": "中", "": "你", "": "说", "": "生", "": "国", "": "年", "": "着", "": "就", | ||
"": "那", "": "和", "": "要", "": "她", "": "出", "": "也", "": "得", "": "里", "": "后", "": "自", "": "以", | ||
"": "会", "": "家", "": "可", "": "下", "": "而", "": "过", "": "天", "": "去", "": "能", "": "对", "": "小", | ||
"": "多", "": "然", "": "于", "": "心", "": "学", "": "么", "": "之", "": "都", "": "好", "": "看", "": "起", | ||
"": "发", "": "当", "": "没", "": "成", "": "只", "": "如", "": "事", "": "把", "": "还", "": "用", "": "第", | ||
"": "样", "": "道", "": "想", "": "作", "": "种", "": "开", "": "美", "": "乳", "": "阴", "": "液", "": "茎", | ||
"": "欲", "": "呻", "": "肉", "": "交", "": "性", "": "胸", "": "私", "": "穴", "": "淫", "": "臀", "": "舔", | ||
"": "射", "": "脱", "": "裸", "": "骚", "": "唇", " ": " ", "<": "<", ">": ">", "&": "&", "⋅": '·', | ||
}; | ||
|
||
async load() { | ||
this.querySelector = async (content, selector) => { | ||
const res = await this.querySelectorAll(content, selector); | ||
return res === null ? null : res[0]; | ||
}; | ||
this.text = (element) => { | ||
return [...element.content.matchAll(/>([^<\n]+?)</g)].map((m) => m[1].trim()).join(""); | ||
}; | ||
this.filter = (element) => { // 59392 | ||
if (element.content.startsWith("<img")) { | ||
const match = [...element.content.matchAll(/src="(.*?)"/g)]; | ||
return `【Miru 暂不支持查看插图${match.length ? ":" + match[match.length - 1][1] : ""}】`; | ||
} | ||
return this.text(element).replace(/&[a-z]+;|./g, (c) => this.dict[c] || c); | ||
}; | ||
this.handle = async (url, count = 10) => { | ||
try { | ||
const response = await this.request(url, { headers: { "Accept-Language": "zh-cn", Accept: "*/*" } }); | ||
const row = await this.querySelectorAll(response, "#acontentz > p, img"); | ||
return row.map(this.filter); | ||
} catch (error) { | ||
if (count > 0) { | ||
console.log(`Retry ${count} times: ${url}`); | ||
await new Promise((resolve) => setTimeout(resolve, 1000)); | ||
return this.handle(url, count - 1); | ||
} else { | ||
throw error; | ||
} | ||
} | ||
}; | ||
} | ||
|
||
async latest(page) { | ||
const res = await this.request(`/top/lastupdate/${page}.html`); | ||
const list = await this.querySelectorAll(res, ".book-li > a"); | ||
const tasks = list.map(async (e) => { | ||
const img = await this.querySelector(e.content, "img"); | ||
const cover = await img.getAttributeText("data-src"); | ||
const title = await img.getAttributeText("alt"); | ||
const update = this.text(await this.querySelector(e.content, ".book-author")); | ||
const url = `${e.content.match(/href="(.*?)"/)[1]}|${title}|${cover}`; | ||
return { title, url, cover, update }; | ||
}); | ||
return await Promise.all(tasks); | ||
} | ||
|
||
async search(kw, page) { | ||
const res = await this.request(`/search/${encodeURI(kw)}_${page}.html`); | ||
const total = await this.querySelector(res, "#pagelink > span"); | ||
if (total) { | ||
if (parseInt(this.text(total).split("/")[1].slice(0, -1)) < page) { | ||
return []; | ||
} | ||
const list = await this.querySelectorAll(res, ".book-li > a"); | ||
const tasks = list.map(async (e) => { | ||
const img = await this.querySelector(e.content, "img"); | ||
const cover = await img.getAttributeText("data-src"); | ||
const title = await img.getAttributeText("alt"); | ||
let author = await this.querySelector(e.content, ".book-author"); | ||
author = author.content.match(/svg>(.*)<\/span>/)[1].trim(); | ||
if (author.indexOf("hot") > 0) { | ||
author = author.match(/class="hot">(.*)<\/span>/)[1].trim() + author.split("</span>")[1]; | ||
} | ||
const url = `${e.content.match(/href="(.*?)"/)[1]}|${title}|${cover}`; | ||
return { title, url, cover, update: author }; | ||
}); | ||
return await Promise.all(tasks); | ||
} | ||
// 此时已经在详情页 | ||
const head = await this.querySelector(res, "head"); | ||
const desc = await this.querySelector(res, "#bookSummary > content"); | ||
const title = await this.getAttributeText(head.content, "meta[property='og:title']", "content"); | ||
const cover = await this.getAttributeText(head.content, "meta[property='og:image']", "content"); | ||
const url = await this.getAttributeText(head.content, "meta[property='og:novel:read_url']", "content"); | ||
return [ | ||
{ | ||
title, | ||
cover, | ||
url: `${url.substring(25)}|${title}|${cover}|${this.text(desc).replace("<br>", "")}`, | ||
update: await this.getAttributeText(head.content, "meta[property='og:novel:author']", "content"), | ||
}, | ||
]; | ||
} | ||
|
||
async detail(string) { | ||
let promise; | ||
const data = string.split("|"); | ||
if (data.length == 3) { | ||
promise = (async (data) => { | ||
const res = await this.request(data[0]); | ||
const desc = await this.querySelector(res, "#bookSummary > content"); | ||
data.push(this.text(desc).replace("<br>", "")); | ||
}).call(this, data); | ||
data[0] = `${data[0].slice(0, -5)}/catalog`; | ||
} | ||
const catalog = await this.request(data[0]); | ||
const volumes = await this.querySelectorAll(catalog, ".volume-chapters"); | ||
const episodes = volumes.map(async (volume) => { | ||
const title = this.text(await this.querySelector(volume.content, ".chapter-bar > h3")); | ||
const urls = await this.querySelectorAll(volume.content, ".chapter-li-a"); | ||
const tasks = await Promise.all( | ||
urls.map(async (url) => ({ name: this.text(url), url: await url.getAttributeText("href") })) | ||
); | ||
return { title, urls: tasks }; | ||
}); | ||
const newLocal = await Promise.all(episodes); | ||
if (promise) await promise; | ||
return { title: data[1], cover: data[2], desc: data[3], episodes: newLocal }; | ||
} | ||
|
||
async watch(url) { | ||
url = url.slice(0, -5); | ||
const res = await this.request(`${url}_2.html`, { headers: { "Accept-Language": "zh-cn", Accept: "*/*" } }); | ||
const subtitle = this.text(await this.querySelector(res, "#apage h1")); | ||
const total = parseInt(subtitle.split("/")[1].slice(0, -1)) || 1; | ||
let tasks = []; | ||
for (let i = 1; i <= total; i++) { | ||
if (i != 2) { | ||
tasks.push(this.handle(`${url}_${i}.html`)); | ||
} | ||
} | ||
tasks = await Promise.all(tasks); | ||
if (total > 1) { | ||
tasks.splice(1, 0, (await this.querySelectorAll(res, "#acontentz > p")).map(this.filter)); | ||
} | ||
return { title: "Test", subtitle: subtitle.split("(")[0], content: tasks.flat() }; | ||
} | ||
} |