From 2d8a869c89a63f00619a4fcf09323470a236bd5f Mon Sep 17 00:00:00 2001 From: entityJY Date: Thu, 28 Aug 2025 00:07:32 -0700 Subject: [PATCH] Add webnoveltranslations --- .../novel/src/en/webnoveltranslations.js | 145 ++++++++++++++++++ 1 file changed, 145 insertions(+) create mode 100644 javascript/novel/src/en/webnoveltranslations.js diff --git a/javascript/novel/src/en/webnoveltranslations.js b/javascript/novel/src/en/webnoveltranslations.js new file mode 100644 index 00000000..5f0c5322 --- /dev/null +++ b/javascript/novel/src/en/webnoveltranslations.js @@ -0,0 +1,145 @@ +const mangayomiSources = [{ + "name": "Web Novel Translations", + "lang": "en", + "baseUrl": "https://webnoveltranslations.com", + "apiUrl": "", + "iconUrl": "", + "typeSource": "single", + "itemType": 2, + "version": "0.0.1", + "pkgPath": "", + "notes": "" +}]; + +class DefaultExtension extends MProvider { + + mangaListFromPage(res) { + const doc = new Document(res.body); + const mangaElements = doc.select(".row.c-tabs-item__content"); + const list = []; + for (const element of mangaElements) { + const name = element.selectFirst("h3")?.text.trim(); + const imageUrl = element.selectFirst("img").getSrc; + const link = element.selectFirst(".tab-thumb.c-image-hover > a").getHref; + list.push({ name, imageUrl, link }); + } + const hasNextPage = false; + return { list: list, hasNextPage }; + } + + getHeaders(url) { + throw new Error("getHeaders not implemented"); + } + + async getPopular(page) { + let url = `${this.source.baseUrl}/?s=&post_type=wp-manga`; + const res = await new Client().get(url, this.headers); + return this.mangaListFromPage(res); + } + + get supportsLatest() { + throw new Error("supportsLatest not implemented"); + } + + async getLatestUpdates(page) { + throw new Error("getLatestUpdates not implemented"); + let url = this.source.baseUrl; + const res = await new Client().get(url, this.headers); + const doc = new Document(res.body); + const mangaElements = doc.select("#loop-content > div"); + const list = []; + for (const element of mangaElements) { + const name = element.selectFirst("div.post-title.font-title")?.text.trim(); + const imageUrl = element.selectFirst("img").getSrc; + const link = element.selectFirst(".item-summary > a").getHref; + list.push({ name, imageUrl, link }); + } + const hasNextPage = false; + return { list: list, hasNextPage }; + } + async search(query, page, filters) { + let url = `${this.source.baseUrl}/?s=${query}&post_type=wp-manga`; + const res = await new Client().get(url, this.headers); + return this.mangaListFromPage(res); + } + async getDetail(url) { + const client = new Client(); + // const res = await client.get(this.source.baseUrl + url, this.headers); + const res = await client.get(url, this.headers); + const doc = new Document(res.body); + const main = doc.selectFirst('.site-content'); + + let description = ""; + for (const element of doc.select(".summary__content > p")) { + description += element.text; + } + + const genre = doc.select("div.genres-content > a").map((el) => el.text.trim()); + + const author = doc.selectFirst("div.author-content > a").text.trim(); + + //const status = doc.selectFirst("div.post-status > .summary-content")?.text.trim(); + const status = 5; + + + const chapterRes = await client.post(url + "ajax/chapters/?t=1", {"x-requested-with": "XMLHttpRequest"}); + const chapterDoc = new Document(chapterRes.body); + + let chapters = []; + for (const chapter of chapterDoc.select("li.wp-manga-chapter ")) { + chapters.push({ + name: chapter.selectFirst("a").text.trim(), + url: chapter.selectFirst("a").getHref, + dateUpload: chapter.selectFirst('i').text, + //dateUpload: "", + scanlator: null, + }); + } + + console.log(chapters[0]); + console.log(chapters[30]); + + return { + description, + genre, + author, + status, + chapters, + }; + } + // For novel html content + async getHtmlContent(name, url) { + const client = await new Client(); + const res = await client.get(url); + + const html = await this.cleanHtmlContent(res.body); + + return html; + } + // Clean html up for reader + async cleanHtmlContent(html) { + const doc = new Document(html); + const title = doc.selectFirst("#chapter-heading")?.text.trim() || ""; + + const content = doc.select("#novel-chapter-container.text-left > p"); + let chapterContent = ""; + for (const line of content) { + chapterContent += "

" + line.text + "

"; + }; + return `

${title}



${chapterContent}`; + } + // For anime episode video list + async getVideoList(url) { + throw new Error("getVideoList not implemented"); + } + // For manga chapter pages + async getPageList(url) { + throw new Error("getPageList not implemented"); + } + getFilterList() { + throw new Error("getFilterList not implemented"); + } + getSourcePreferences() { + throw new Error("getSourcePreferences not implemented"); + } +}