天域小说
http://www.tianyu8.org
autobcb_admin (12020)3天前
天域小说
{
"bookSourceUrl": "http:\/\/www.tianyu8.org",
"bookSourceName": "天域小说",
"enabledExplore": true,
"enabled": true,
"bookSourceGroup": "",
"author": "",
"help": false,
"html": "地址地址di'zhidi'zhdi'z<!DOCTYPE html>\r\n<html lang=\"en\">\r\n\r\n<head>\r\n <meta charset=\"UTF-8\">\r\n <title>天域小说<\/title>\r\n<body>\r\n\r\n<\/body>\r\n<script src=\"https:\/\/vc.jd.com\/web\/js\/jquery-3.1.1.min.js\"><\/script>\r\n<!--如果要引入外部 js 必须在书源代码的上面-->\r\n<script>\r\n var isCookieJar = false;\/\/ 不需要CookieJar请修改此处\r\n class FlutterJSBridge {\r\n constructor() {\r\n this.init(); \/\/前台webview 里必须删除这行\r\n }\r\n\r\n init() {\r\n if (window.flutter_inappwebview) {\r\n this.isReady = true;\r\n this.CookieJar();\r\n } else {\r\n window.addEventListener('flutterInAppWebViewPlatformReady', () => {\r\n this.isReady = true;\r\n console.log('JSBridge初始化完成');\r\n this.CookieJar();\r\n });\r\n }\r\n }\r\n\r\n \/\/通知原生页面初始化完成,仅在书源和tts生效,webview请勿使用,只有通知加载成功后才允许运行,否则会一直等待加载成功\r\n async CookieJar() {\r\n try {\r\n await window.flutter_inappwebview.callHandler('CookieJar', isCookieJar);\r\n } catch (error) {\r\n console.error('汇报完成准备失败:', error);\r\n }\r\n }\r\n\r\n \/\/获取应用编译版本\r\n async getbuildNumber() {\r\n try {\r\n return await window.flutter_inappwebview.callHandler('buildNumber');\r\n } catch (error) {\r\n return 0;\r\n }\r\n }\r\n\r\n \/\/获取应用版本\r\n async getversion() {\r\n try {\r\n return await window.flutter_inappwebview.callHandler('version');\r\n } catch (error) {\r\n return \"0.0.0\";\r\n }\r\n }\r\n\r\n \/\/获取设备唯一id\r\n async getDeviceid() {\r\n try {\r\n return await window.flutter_inappwebview.callHandler('id');\r\n } catch (error) {\r\n return \"\";\r\n }\r\n }\r\n\r\n \/\/获取设备平台 此处返回 windows、macos、ios、ohos、android\r\n async getDevice() {\r\n try {\r\n return await window.flutter_inappwebview.callHandler('device');\r\n } catch (error) {\r\n return \"\";\r\n }\r\n }\r\n\r\n \/\/输出日志,前台webview请勿使用\r\n \/\/str 为 String\r\n async log(str) {\r\n try {\r\n return await window.flutter_inappwebview.callHandler('log', str);\r\n } catch (error) {\r\n return false;\r\n }\r\n }\r\n\r\n \/\/书源调试时可输出 html 代码到前台\r\n \/\/type 0 搜索源码 , 1详情源码 ,2目录源码 ,3正文源码\r\n \/\/str 为 String\r\n \/\/type 为int\r\n async text(type, str) {\r\n try {\r\n return await window.flutter_inappwebview.callHandler('text', type, str);\r\n } catch (error) {\r\n return false;\r\n }\r\n }\r\n\r\n \/\/toast弹窗\r\n \/\/str 为 String\r\n async showToast(str) {\r\n try {\r\n return await window.flutter_inappwebview.callHandler('showToast', str);\r\n } catch (error) {\r\n return false;\r\n }\r\n }\r\n\r\n \/\/获取默认ua\r\n async getWebViewUA() {\r\n try {\r\n return await window.flutter_inappwebview.callHandler('getWebViewUA');\r\n } catch (error) {\r\n return \"\";\r\n }\r\n }\r\n\r\n \/\/通过url打开外部应用\r\n \/\/url 为 String\r\n async openurl(url) {\r\n try {\r\n return await window.flutter_inappwebview.callHandler('openurl', url, \"\");\r\n } catch (error) {\r\n return false;\r\n }\r\n }\r\n\r\n \/\/通过url打开外部应用并附带mimeType\r\n \/\/url 为 String\r\n \/\/mimeType 为 String\r\n async openurlwithMimeType(url, mimeType) {\r\n try {\r\n return await window.flutter_inappwebview.callHandler('openurl', url, mimeType);\r\n } catch (error) {\r\n return false;\r\n }\r\n }\r\n\r\n \/**\r\n * 使用webView访问网络\r\n * @param html 直接用webView载入的html, 如果html为空直接访问url\r\n * @param url html内如果有相对路径的资源不传入url访问不了\r\n * @param js 用来取返回值的js语句, 没有就返回整个源代码\r\n * @param body 当参数不为空的时候,会以post请求,此时请务必在 header 中带上content-type\r\n * @param header 请求的header头,此参数必须是json字符串\r\n * @return 返回js获取的内容\r\n *\/\r\n async webview(url, js, html, body, header) {\r\n try {\r\n return await window.flutter_inappwebview.callHandler('webview', url, js, html, body, header, \"\", \"\");\r\n } catch (error) {\r\n return \"\";\r\n }\r\n }\r\n\r\n \/**\r\n * overrideUrlRegex 为正则表达式\r\n * 使用方法和上面的一样\r\n * 但返回的内容为正则到的内容,如果无法正则到则返回 js 获取的内容,如果 js 为空则返回页面 html\r\n *\/\r\n async webViewGetOverrideUrl(url, js, html, body, header, overrideUrlRegex) {\r\n try {\r\n return await window.flutter_inappwebview.callHandler('webview', url, js, html, body, header, overrideUrlRegex, \"\");\r\n } catch (error) {\r\n return \"\";\r\n }\r\n }\r\n\r\n \/**\r\n * 使用webView获取资源url\r\n * urlregex 为正则表达式\r\n * 使用方法和上面的一样\r\n * 但返回的内容为正则到的内容,如果无法正则到则返回 js 获取的内容,如果 js 为空则返回页面 html\r\n *\/\r\n async webViewGetSource(url, js, html, body, header, urlregex) {\r\n try {\r\n return await window.flutter_inappwebview.callHandler('webview', url, js, html, body, header, \"\", urlregex);\r\n } catch (error) {\r\n return \"\";\r\n }\r\n }\r\n\r\n\r\n \/**\r\n * 启动前台 webview 访问链接并获取结束时的 html,可用于手工过盾\r\n * @param url 网址\r\n * @param title 标题\r\n * @param header 请求的header头,此参数必须是json字符串\r\n * @return 返回网页的内容\r\n *\/\r\n async startBrowser(url, title, header) {\r\n try {\r\n return await window.flutter_inappwebview.callHandler('startBrowser', url, title, header);\r\n } catch (error) {\r\n return \"\";\r\n }\r\n }\r\n\r\n \/\/专门为段评设置的半屏显示,不返回任何东西\r\n async startBrowserDp(url, title) {\r\n try {\r\n return await window.flutter_inappwebview.callHandler('startBrowserDp', url, title);\r\n } catch (error) {\r\n return \"\";\r\n }\r\n }\r\n\r\n \/\/仅前台webview可以使用,返回按钮,返回上一个页面\r\n async back() {\r\n try {\r\n return await window.flutter_inappwebview.callHandler('back');\r\n } catch (error) {\r\n return false;\r\n }\r\n }\r\n\r\n \/\/将 utf8字符串转到 gbk 并 url 编码\r\n async utf8ToGbkUrlEncoded(str) {\r\n try {\r\n return await window.flutter_inappwebview.callHandler('utf8ToGbkUrlEncoded', str);\r\n } catch (error) {\r\n return \"\";\r\n }\r\n }\r\n\r\n \/*\r\n * @param str为图片链接\r\n * @param header 请求的header头,此参数必须是json字符串\r\n * 此函数是让用户输入图片中的验证码,当链接为空则直接让用户输入验证码\r\n *\/\r\n async getVerificationCode(str, header) {\r\n try {\r\n return await window.flutter_inappwebview.callHandler('getVerificationCode', str, header);\r\n } catch (error) {\r\n return \"\";\r\n }\r\n }\r\n\r\n \/\/提交内容书本信息 json 后的字符串\r\n async addbook(book) {\r\n try {\r\n return await window.flutter_inappwebview.callHandler('addbook', book);\r\n } catch (error) {\r\n return \"\";\r\n }\r\n }\r\n\r\n }\r\n\r\n \/\/webview请勿使用\r\n \/\/以下提交的url,headers,body 都必须为字符串,headers必须为json字符串\r\n \/\/当followRedirects 为 false 时不处理重定向,当为 true 时会自动处理重定向 ,如不明白用途直接用 true 最佳\r\n \/\/ 以下所有参数除当followRedirects外均为 String\r\n class Http {\r\n constructor() { }\r\n\r\n \/*\r\n * 通用返回字段\r\n * method post get 或者 head\r\n * body 请求返回后的字节的 base64\r\n * headers map<String,List<String>> 可通过headers[\"\"]来或者\r\n * statusCode 状态码\r\n * statusMessage\r\n * data 返回后的字节 格式化后的内容\r\n *\/\r\n async Get(url, headers, followRedirects) {\r\n try {\r\n return await window.flutter_inappwebview.callHandler('http', \"get\", url, \"\", JSON.stringify(headers), followRedirects, \"\");\r\n } catch (error) {\r\n return null;\r\n }\r\n }\r\n\r\n async Head(url, headers, followRedirects) {\r\n try {\r\n return await window.flutter_inappwebview.callHandler('http', \"head\", url, \"\", JSON.stringify(headers), followRedirects, \"\");\r\n } catch (error) {\r\n return null;\r\n }\r\n }\r\n\r\n\r\n async Post(url, headers, body, contenttype, followRedirects) {\r\n try {\r\n return await window.flutter_inappwebview.callHandler('http', \"post\", url, body, JSON.stringify(headers), followRedirects, contenttype);\r\n } catch (error) {\r\n return null;\r\n }\r\n }\r\n }\r\n\r\n class Cache {\r\n constructor() { }\r\n async get(key) {\r\n try {\r\n return await window.flutter_inappwebview.callHandler('cache.get', key);\r\n } catch (error) {\r\n return null;\r\n }\r\n }\r\n\r\n async set(key, value) {\r\n try {\r\n return await window.flutter_inappwebview.callHandler('cache.set', key, value);\r\n } catch (error) {\r\n return null;\r\n }\r\n }\r\n\r\n async remove(key) {\r\n try {\r\n return await window.flutter_inappwebview.callHandler('cache.remove', key);\r\n } catch (error) {\r\n return null;\r\n }\r\n }\r\n\r\n \/\/如果登录为弹窗格式的,里面输入框输入的内容可以通过这个函数获取,默认返回的json格式或者为空,需要自行转换\r\n async getLoginInfo() {\r\n return await this.get(\"LoginInfo\")\r\n }\r\n\r\n \/\/将修改后的弹窗输入内容报错 ,必须 JSON.stringify,不然会出错\r\n async putLoginInfo(info) {\r\n return await this.set(\"LoginInfo\", info)\r\n }\r\n\r\n \/\/获取书本变量\r\n async getbookVariable(bookurl) {\r\n return await this.get(bookurl)\r\n }\r\n\r\n \/\/写入书本变量\r\n async setbookVariable(bookurl, value) {\r\n return await this.set(bookurl, value)\r\n }\r\n }\r\n\r\n class Cookie {\r\n constructor() { }\r\n\r\n \/\/通过url获取当前url的所有cookie\r\n async get(url) {\r\n try {\r\n return await window.flutter_inappwebview.callHandler('cookie.get', url);\r\n } catch (error) {\r\n return null;\r\n }\r\n }\r\n\r\n \/\/通过url删除当前url的所有cookie\r\n async remove(url) {\r\n try {\r\n return await window.flutter_inappwebview.callHandler('cookie.remove', url);\r\n } catch (error) {\r\n return null;\r\n }\r\n }\r\n\r\n\r\n \/\/通过url保存当前url的所有cookie\r\n async set(url, value) {\r\n try {\r\n return await window.flutter_inappwebview.callHandler('cookie.set', url, value);\r\n } catch (error) {\r\n return null;\r\n }\r\n }\r\n\r\n \/\/通过 url 获取单个 cookie 的值\r\n async getCookie(url, value) {\r\n try {\r\n return await window.flutter_inappwebview.callHandler('cookie.getCookie', url, value);\r\n } catch (error) {\r\n return null;\r\n }\r\n }\r\n }\r\n\r\n \/\/安全的创建一个 div 解析 html\r\n function parseHTMLSafely(htmlStr) {\r\n try {\r\n \/\/ 在函数作用域内创建独立的临时容器\r\n \/\/ 每个调用创建新的jQuery对象,互不影响\r\n var tempDiv = document.createElement('div');\r\n tempDiv.innerHTML = htmlStr;\r\n return $(tempDiv);\r\n } catch (e) {\r\n flutterBridge.log(\"HTML解析错误:\" + e.message);\r\n return $('<div>');\r\n }\r\n }\r\n\r\n \/\/parseHTMLSafely 创建的用完后必须删除\r\n function removeHTMLSafely(tempContainer) {\r\n try {\r\n tempContainer.innerHTML = '';\r\n if (tempContainer.parentNode) {\r\n tempContainer.parentNode.removeChild(tempContainer);\r\n }\r\n } catch (e) {\r\n flutterBridge.log(\"HTML移除失败:\" + e.message);\r\n }\r\n }\r\n\r\n \/\/移除 css js,创建parseHTMLSafely前如果用不上 cssjs 建议移除\r\n function removeHTMLTags(htmlString) {\r\n \/\/ 移除script标签\r\n let result = htmlString.replace(\/<script\\b[^<]*(?:(?!<\\\/script>)<[^<]*)*<\\\/script>\/gi, '');\r\n \/\/ 移除style标签\r\n result = result.replace(\/<style\\b[^<]*(?:(?!<\\\/style>)<[^<]*)*<\\\/style>\/gi, '');\r\n return result;\r\n }\r\n\r\n<\/script>\r\n<script>\r\n const flutterBridge = new FlutterJSBridge();\r\n const cache = new Cache();\r\n const http = new Http();\r\n const cookie = new Cookie();\r\n var baseurl = \"http:\/\/www.tianyu8.org\"\r\n var header = { \"User-Agent\": \"Mozilla\/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit\/537.36 (KHTML, like Gecko) Chrome\/131.0.0.0 Safari\/537.36\" };\r\n\r\n \/\/ 搜索函数\r\n async function search(key, page) {\r\n if (page > 1) {\r\n return \"[]\";\r\n }\r\n\r\n try {\r\n\r\n const indexHeader = { ...header, \"Referer\": baseurl };\r\n const indexRes = await http.Get(baseurl, JSON.stringify(indexHeader), true);\r\n const pageSource = indexRes.data;\r\n\r\n const actionReg = \/<form\\s+name=\"search\".*?action=\"(.*?)\".*?>\/i;\r\n const matchResult = pageSource.match(actionReg);\r\n const formAction = matchResult?.[1]?.trim() || '\/search1f.html';\r\n\r\n var url = baseurl + formAction + \"?searchkey=\" + encodeURIComponent(key);\r\n \r flutterBridge.log(\"获取到搜索地址:\" + url);\r\n var myheader = {\r\n ...header,\r\n \"Referer\": baseurl\r\n };\r\n var get = await http.Get(url, JSON.stringify(myheader), true);\r\n flutterBridge.text(0, get.data);\r\n\r\n return parseBookList(get.data);\r\n } catch (e) {\r\n flutterBridge.log(\"搜索出错:\" + e.message);\r\n return \"[]\";\r\n }\r\n }\r\n\/\/ 解析书籍列表(适配新的.item结构)\r\nfunction parseBookList(html) {\r\n try {\r\n \/\/ 安全解析HTML并移除多余标签,保留核心结构\r\n var $tempContainer = parseHTMLSafely(removeHTMLTags(html));\r\n var books = [];\r\n var bookUrls = new Set(); \/\/ 基于书籍链接去重,保证唯一性\r\n\r\n \/\/ 核心:遍历每一个书籍项.item,一个item对应一本书,精准匹配结构\r\n $tempContainer.find(\".item\").each(function () {\r\n var $item = $(this);\r\n var bookUrl = \"\";\r\n var name = \"\";\r\n\r\n \/\/ 1. 提取书籍链接和书名(dt下的a标签是核心,与image下的a链接一致)\r\n var $titleLink = $item.find(\"dl dt a\").first();\r\n if (!$titleLink.length) return true; \/\/ 无书名链接,跳过当前项\r\n var href = $titleLink.attr(\"href\");\r\n if (!href) return true; \/\/ 无href属性,跳过\r\n\r\n \/\/ 拼接完整书籍链接,兼容相对路径\/绝对路径,确保结尾以\/结束(与原逻辑一致)\r\n if (href.startsWith(\"http\")) {\r\n bookUrl = href;\r\n } else if (href.startsWith(\"\/\")) {\r\n bookUrl = baseurl + href;\r\n } else {\r\n return true; \/\/ 非标准路径,跳过\r\n }\r\n if (!bookUrl.endsWith(\"\/\")) bookUrl += \"\/\";\r\n\r\n \/\/ 链接去重:已存在则跳过\r\n if (bookUrls.has(bookUrl)) return true;\r\n bookUrls.add(bookUrl);\r\n\r\n \/\/ 提取书名并过滤无效名称(长度≥2)\r\n name = $titleLink.text().trim();\r\n if (!name || name.length < 2) return true;\r\n\r\n \/\/ 2. 提取书籍封面(image下的img标签,优先取src,兼容懒加载的data-original)\r\n var coverUrl = \"\";\r\n var $coverImg = $item.find(\".image img\").first();\r\n if ($coverImg.length) {\r\n var src =$coverImg.attr(\"data-original\");\r\n if (src) {\r\n coverUrl = src.startsWith(\"http\") ? src : (src.startsWith(\"\/\") ? baseurl + src : \"\");\r\n }\r\n }\r\n\r\n \/\/ 3. 提取书籍简介(dd标签,移除开头的\"简介:\"前缀,兼容中英文冒号)\r\n var intro = \"\";\r\n var $introDd = $item.find(\"dl dd\").first();\r\n if ($introDd.length) {\r\n intro = $introDd.text().trim();\r\n intro = intro.replace(\/^简介[::\\s]*\/, \"\"); \/\/ 去掉\"简介:\"\/\"简介:\"及后续空格\r\n }\r\n\r\n \/\/ 4. 提取作者(.btm下href包含\/author\/的a标签)\r\n var author = \"\";\r\n var $authorLink = $item.find(\".btm a[href*='\/author\/']\").first();\r\n if ($authorLink.length) {\r\n author = $authorLink.text().trim();\r\n }\r\n\r\n \/\/ 5. 提取字数(.btm下orange类的em标签)\r\n var wordCount = \"\";\r\n var $wordCountEm = $item.find(\".btm em.orange\").first();\r\n if ($wordCountEm.length) {\r\n wordCount = $wordCountEm.text().trim();\r\n }\r\n\r\n \/\/ 6. 提取更新时间(.btm下blue类的em标签,原结构无最新章节名,用更新时间替代更合理)\r\n var latestChapterTitle = \"\";\r\n var $updateTimeEm = $item.find(\".btm em.blue\").first();\r\n if ($updateTimeEm.length) {\r\n latestChapterTitle = $updateTimeEm.text().trim();\r\n }\r\n\r\n \/\/ 7. 组装书籍对象(字段与原函数完全一致,兼容调用方,分类kind原结构无则留空)\r\n var book = {\r\n \"bookUrl\": bookUrl, \/\/ 书籍主链接\r\n \"name\": name, \/\/ 书籍名称\r\n \"author\": author, \/\/ 作者\r\n \"kind\": \"\", \/\/ 原HTML结构无分类,留空可后续扩展\r\n \"coverUrl\": coverUrl, \/\/ 封面链接\r\n \"intro\": intro, \/\/ 书籍简介(已去前缀)\r\n \"tocUrl\": bookUrl, \/\/ 目录链接与主链接一致\r\n \"wordCount\": wordCount, \/\/ 书籍字数(替代原状态,更贴合新结构)\r\n \"type\": 0, \/\/ 原固定值,保留\r\n \"latestChapterTitle\": latestChapterTitle \/\/ 最新更新时间(原结构无章节名,合理替代)\r\n };\r\n\r\n books.push(book);\r\n });\r\n\r\n \/\/ 清理临时容器,避免内存占用\r\n removeHTMLSafely($tempContainer);\r\n \/\/ 返回JSON格式的书籍列表\r\n return JSON.stringify(books);\r\n } catch (e) {\r\n \/\/ 异常日志,与原函数一致\r\n flutterBridge.log(\"解析书籍列表出错:\" + e.message);\r\n \/\/ 异常时返回空数组的JSON\r\n return \"[]\";\r\n }\r\n}\r\n\r\n \/\/ 书籍详情函数\r\nasync function info(bookurl) {\r\n try {\r\n \/\/ 发起网络请求,获取书籍详情页HTML\r\n var get = await http.Get(bookurl, JSON.stringify(header), true);\r\n flutterBridge.text(1, get.data);\r\n\r\n \/\/ 安全解析HTML并清理标签,基于#maininfo根容器提取,避免无关元素干扰\r\n var $tempContainer = parseHTMLSafely(removeHTMLTags(get.data));\r\n var $mainInfo = $tempContainer.find(\"#maininfo\").first();\r\n \/\/ 无核心详情容器则直接返回空对象\r\n if (!$mainInfo.length) return \"{}\";\r\n\r\n \/\/ 1. 提取书名:从#info下的h1直接提取,结构固定精准\r\n var name = \"\";\r\n var $titleH1 = $mainInfo.find(\"#info h1\").first();\r\n if ($titleH1.length) {\r\n name = $titleH1.text().trim();\r\n }\r\n\r\n \/\/ 2. 提取作者:#info下href包含\/author\/的a标签,原结构a标签纯作者名无多余前缀\r\n var author = \"\";\r\n var $authorLink = $mainInfo.find(\"#info a[href*='\/author\/']\").first();\r\n if ($authorLink.length) {\r\n author = $authorLink.text().trim();\r\n }\r\n\r\n \/\/ 3. 提取封面:#sidebar #fmimg下的img,兼容懒加载src\/data-original,保留baseurl拼接\r\n var coverUrl = \"\";\r\n var $coverImg = $mainInfo.find(\"#sidebar #fmimg img\").first();\r\n if ($coverImg.length) {\r\n \/\/ 优先取src,无则取懒加载的data-original\r\n var src = $coverImg.attr(\"data-original\");\r\n if (src) {\r\n if (src.startsWith('http')) {\r\n coverUrl = src;\r\n } else if (src.startsWith('\/')) {\r\n coverUrl = baseurl + src;\r\n }\r\n }\r\n }\r\n\r\n \/\/ 4. 提取字数:匹配含「万字」的文本,兼容原结构「248 万字」(中间有空格)的格式\r\n var wordCount = \"\";\r\n $mainInfo.find(\"#info p\").each(function () {\r\n var text = $(this).text().trim();\r\n var wordMatch = text.match(\/(\\d+(?:\\.\\d+)?)\\s*万字\/);\r\n if (wordMatch) {\r\n wordCount = wordMatch[1] + \"万字\"; \/\/ 整理格式:去掉中间空格,统一为「248万字」\r\n return false; \/\/ 找到后终止遍历\r\n }\r\n });\r\n\r\n \/\/ 5. 提取书籍状态:连载\/已完结(新结构新增,补充至返回字段)\r\n var status = \"\";\r\n $mainInfo.find(\"#info p\").each(function () {\r\n var text = $(this).text().trim();\r\n if (text.includes(\"连载\") || text.includes(\"已完结\") || text.includes(\"断更\")) {\r\n status = text.match(\/(连载|已完结|断更)\/)[1];\r\n return false; \/\/ 找到后终止遍历\r\n }\r\n });\r\n\r\n \/\/ 6. 提取最新章节标题:优先#info内的最新章节链接,无则取.visible-xs的.lastchapter\r\n var latestChapterTitle = \"\";\r\n \/\/ 匹配rel=chapter且含.html的章节链接,精准定位最新章节\r\n var $latestChapter = $mainInfo.find(\"#info a[rel='chapter'][href*='.html']\").first();\r\n if (!$latestChapter.length) {\r\n $latestChapter = $mainInfo.find(\".lastchapter a[rel='chapter'][href*='.html']\").first();\r\n }\r\n if ($latestChapter.length) {\r\n latestChapterTitle = $latestChapter.text().trim();\r\n }\r\n\r\n \/\/ 7. 提取最后更新时间:新结构新增,补充至返回字段\r\n var lastUpdate = \"\";\r\n $mainInfo.find(\"#info p\").each(function () {\r\n var text = $(this).text().trim();\r\n if (text.startsWith(\"最后更新:\")) {\r\n lastUpdate = text.replace(\/^最后更新:\/, \"\").trim();\r\n return false; \/\/ 找到后终止遍历\r\n }\r\n });\r\n\r\n \/\/ 8. 提取简介:#intro标签直接提取,移除开头「简介:」\/「简介:」前缀\r\n var intro = \"\";\r\n var $introBox = $mainInfo.find(\"#intro\").first();\r\n if ($introBox.length) {\r\n intro = $introBox.text().trim();\r\n intro = intro.replace(\/^简介[::\\s]*\/, \"\"); \/\/ 兼容中英文冒号+空格\r\n }\r\n\r\n \/\/ 9. 提取分类:保留原函数的分类提取逻辑(新结构暂未显式展示分类,留空不影响)\r\n var kind = \"\";\r\n $mainInfo.find(\"a\").each(function () {\r\n var href = $(this).attr('href');\r\n var text = $(this).text().trim();\r\n \/\/ 匹配常见小说分类路径,找到后立即终止\r\n if (href && href.match(\/\\\/(xuanhuan|xianxia|dushi|lishi|junshi|kehuan|yanqing)\\\/\/) && text.length < 10) {\r\n kind = text;\r\n return false;\r\n }\r\n });\r\n\r\n \/\/ 10. 核心:按规则拼接tocUrl(原URL中间插入\/indexlist\/)\r\n var tocUrl = bookurl;\r\n try {\r\n \/\/ 先统一移除URL结尾的\/,避免拼接重复\r\n var pureBookUrl = bookurl.endsWith(\"\/\") ? bookurl.slice(0, -1) : bookurl;\r\n \/\/ 正则替换:匹配域名后的第一个\/,替换为\/indexlist\/(兼容http\/https)\r\n tocUrl = pureBookUrl.replace(\/^(https?:\\\/\\\/[^\\\/]+)\\\/(.+)$\/, \"$1\/indexlist\/$2\/\");\r\n } catch (e) {\r\n flutterBridge.log(\"拼接tocUrl出错:\" + e.message);\r\n tocUrl = bookurl; \/\/ 拼接失败则回退为原书籍URL\r\n }\r\n\r\n \/\/ 组装书籍详情对象:保留原字段+新增status\/lastUpdate,信息更完整\r\n var book = {\r\n \"bookUrl\": bookurl, \/\/ 原书籍URL\r\n \"name\": name, \/\/ 书名\r\n \"author\": author, \/\/ 作者\r\n \"kind\": kind, \/\/ 分类(新结构暂空)\r\n \"coverUrl\": coverUrl, \/\/ 封面链接(兼容懒加载)\r\n \"intro\": intro, \/\/ 书籍简介(已去前缀)\r\n \"tocUrl\": tocUrl, \/\/ 按规则拼接的目录URL\r\n \"wordCount\": wordCount, \/\/ 书籍字数(统一格式:xxx万字)\r\n \"status\": status, \/\/ 新增:书籍状态(连载\/已完结)\r\n \"type\": 0, \/\/ 原固定值,保留\r\n \"latestChapterTitle\": latestChapterTitle, \/\/ 最新章节标题\r\n \"lastUpdate\": lastUpdate \/\/ 新增:最后更新时间\r\n };\r\n\r\n \/\/ 清理临时容器,释放内存\r\n removeHTMLSafely($tempContainer);\r\n \/\/ 返回JSON格式的书籍详情\r\n return JSON.stringify(book);\r\n } catch (e) {\r\n \/\/ 异常捕获,打印日志并返回空对象\r\n flutterBridge.log(\"获取详情出错:\" + e.message);\r\n return \"{}\";\r\n }\r\n}\r\n\r\n \/\/ 章节列表函数(支持分页)\r\n \/\/ 章节列表函数(并发优化版)\r\nasync function chapter(tocUrl) {\r\n try {\r\n const allChapters = [];\r\n const maxConcurrent = 10; \/\/ 最大并发页数\r\n let currentPage = 0;\r\n let hasMore = true;\r\n\r\n while (hasMore) {\r\n \/\/ 构建当前批次要请求的页码\r\n const batchPromises = [];\r\n const batchPageNumbers = [];\r\n\r\n for (let i = 0; i < maxConcurrent && hasMore; i++) {\r\n const pageIndex = currentPage + i;\r\n const pageUrl = buildPageUrl(tocUrl, pageIndex);\r\n batchPromises.push(getChapterPage(pageUrl));\r\n batchPageNumbers.push(pageIndex);\r\n flutterBridge.log(\"预取第 \" + (pageIndex + 1) + \" 页: \" + pageUrl);\r\n }\r\n\r\n \/\/ 并发执行本批次\r\n const results = await Promise.all(batchPromises);\r\n\r\n \/\/ 处理结果\r\n let batchHasContent = false;\r\n for (let i = 0; i < results.length; i++) {\r\n const result = results[i];\r\n const pageIndex = batchPageNumbers[i];\r\n\r\n if (result.chapters.length > 0) {\r\n batchHasContent = true;\r\n allChapters.push(...result.chapters);\r\n flutterBridge.log(\"第 \" + (pageIndex + 1) + \" 页获取到 \" + result.chapters.length + \" 章\");\r\n } else {\r\n flutterBridge.log(\"第 \" + (pageIndex + 1) + \" 页无章节\");\r\n }\r\n\r\n \/\/ 只要最后一页还有下一页,就继续\r\n if (i === results.length - 1) {\r\n hasMore = result.hasNextPage;\r\n }\r\n }\r\n\r\n \/\/ 如果本批次没有任何内容,提前退出\r\n if (!batchHasContent) {\r\n flutterBridge.log(\"批次无有效章节,停止获取\");\r\n break;\r\n }\r\n\r\n currentPage += maxConcurrent;\r\n\r\n \/\/ 安全限制\r\n if (currentPage >= 100) {\r\n flutterBridge.log(\"已获取 100+ 页,强制停止\");\r\n break;\r\n }\r\n }\r\n\r\n \/\/ 重设 index\r\n for (let i = 0; i < allChapters.length; i++) {\r\n allChapters[i].index = i;\r\n }\r\n\r\n flutterBridge.log(\"总共获取 \" + allChapters.length + \" 章\");\r\n return JSON.stringify(allChapters);\r\n } catch (e) {\r\n flutterBridge.log(\"获取章节列表出错: \" + e.message);\r\n return \"[]\";\r\n }\r\n}\r\n\r\n\/\/ 辅助函数:构建分页 URL\r\nfunction buildPageUrl(tocUrl, pageIndex) {\r\n if (pageIndex === 0) {\r\n return tocUrl;\r\n } else {\r\n const baseUrl = tocUrl.endsWith('\/') ? tocUrl.slice(0, -1) : tocUrl;\r\n return baseUrl + \"\/\" + (pageIndex + 1) + \"\/\";\r\n }\r\n}\r\n\r\n \/\/ 获取单页章节\r\n async function getChapterPage(pageUrl) {\r\n try {\r\n var get = await http.Get(pageUrl, JSON.stringify(header), true);\r\n flutterBridge.text(2, get.data);\r\n\r\n var $tempContainer = parseHTMLSafely(removeHTMLTags(get.data));\r\n var chapters = [];\r\n var hasNextPage = false;\r\n\r\n var novelId = \"\";\r\n var match = pageUrl.match(\/\\\/(\\d+)\\\/?\/);\r\n if (match) {\r\n novelId = match[1];\r\n }\r\n\r\n var $listContainer = $tempContainer.find(\"#content_1\");\r\n var $chapterLinks = $listContainer.length > 0 ? $listContainer.find(\"a[href*='.html']\") : $tempContainer.find(\"a[href*='.html']\");\r\n\r\n var index = 0;\r\n $chapterLinks.each(function () {\r\n var $link = $(this);\r\n var chapterName = $link.text().trim();\r\n var href = $link.attr('href');\r\n\r\n if (!chapterName || !href || chapterName.length < 2) {\r\n return true;\r\n }\r\n\r\n if (chapterName.includes(\"下载\") || chapterName.includes(\"TXT\") || chapterName.includes(\"目录\") ||\r\n chapterName.includes(\"下一页\") || chapterName.includes(\"上一页\") || chapterName.includes(\"开始阅读\") ||\r\n chapterName.includes(\"最新章节\") || chapterName === \"全部\") {\r\n return true;\r\n }\r\n\r\n var chapterId = \"\";\r\n if (href.startsWith('http')) {\r\n chapterId = href;\r\n } else if (href.startsWith('\/')) {\r\n chapterId = baseurl + href;\r\n } else {\r\n chapterId = baseurl + \"\/\" + novelId + \"\/\" + href;\r\n }\r\n\r\n chapters.push({\r\n \"name\": chapterName,\r\n \"chapterId\": chapterId,\r\n \"index\": index,\r\n \"isPay\": false,\r\n \"isVip\": false,\r\n \"isVolume\": false,\r\n \"tag\": \"\"\r\n });\r\n index++;\r\n });\r\n\r\n $tempContainer.find(\"a\").each(function () {\r\n var text = $(this).text().trim();\r\n if (text === \"下一页\" || text.includes(\"下一页\")) {\r\n hasNextPage = true;\r\n return false;\r\n }\r\n });\r\n\r\n removeHTMLSafely($tempContainer);\r\n return { chapters: chapters, hasNextPage: hasNextPage };\r\n } catch (e) {\r\n flutterBridge.log(\"获取章节页出错:\" + e.message);\r\n return { chapters: [], hasNextPage: false };\r\n }\r\n }\r\n\r\n \/\/ 章节内容函数(支持分页章节)\r\n async function content(url) {\r\n try {\r\n var allContent = [];\r\n var currentUrl = url;\r\n var pageIndex = 1;\r\n var maxPages = 10; \/\/ 防止无限循环\r\n\r\n while (pageIndex <= maxPages) {\r\n var get = await http.Get(currentUrl, JSON.stringify(header), true);\r\n if (pageIndex === 1) {\r\n flutterBridge.text(3, get.data);\r\n }\r\n\r\n var $tempContainer = parseHTMLSafely(removeHTMLTags(get.data));\r\n\r\n \/\/ 使用正确的内容选择器: div.con\r\n var $content = $tempContainer.find(\"#booktxt\");\r\n\r\n if ($content.length > 0) {\r\n \/\/ 移除广告和无关元素\r\n $content.find(\"script, style, .ad, .adsbygoogle, .advertisement\").remove();\r\n\r\n \/\/ 提取段落内容\r\n var paragraphs = [];\r\n $content.find(\"p\").each(function () {\r\n var text = $(this).text().trim();\r\n if (text.length > 0) {\r\n paragraphs.push(text);\r\n }\r\n });\r\n\r\n \/\/ 如果没有p标签,直接获取文本\r\n if (paragraphs.length === 0) {\r\n var text = $content.text().trim();\r\n if (text.length > 0) {\r\n paragraphs.push(text);\r\n }\r\n }\r\n\r\n if (paragraphs.length > 0) {\r\n allContent.push(paragraphs.join('\\r\\n\\r\\n'));\r\n }\r\n } else {\r\n break;\r\n }\r\n\r\n \/\/ 检查是否有下一页\r\n var hasNextPage = false;\r\n $tempContainer.find(\"a\").each(function () {\r\n var text = $(this).text().trim();\r\n var href = $(this).attr('href');\r\n if ((text === \"下一页\" || text.includes(\"下一页\")) && href) {\r\n \/\/ 构建下一页URL\r\n if (href.startsWith('http')) {\r\n currentUrl = href;\r\n } else if (href.startsWith('\/')) {\r\n currentUrl = baseurl + href;\r\n } else {\r\n \/\/ 相对路径,需要基于当前URL构建\r\n var urlParts = currentUrl.split('\/');\r\n urlParts[urlParts.length - 1] = href;\r\n currentUrl = urlParts.join('\/');\r\n }\r\n hasNextPage = true;\r\n return false; \/\/ break\r\n }\r\n });\r\n\r\n removeHTMLSafely($tempContainer);\r\n\r\n if (!hasNextPage) {\r\n break;\r\n }\r\n\r\n pageIndex++;\r\n }\r\n\r\n \/\/ 合并所有分页内容\r\n var finalContent = allContent.join('\\r\\n\\r\\n');\r\n\r\n \/\/ 移除导航文本\r\n finalContent = finalContent.replace(\/上一章\/g, '');\r\n finalContent = finalContent.replace(\/目录\/g, '');\r\n finalContent = finalContent.replace(\/下一页\/g, '');\r\n finalContent = finalContent.replace(\/下一章\/g, '');\r\n finalContent = finalContent.replace(\/首页\/g, '');\r\n\r\n \/\/ 清理多余的空行\r\n finalContent = finalContent.replace(\/\\r\\n\\r\\n\\r\\n+\/g, '\\r\\n\\r\\n').trim();\r\n\r\n return finalContent;\r\n } catch (e) {\r\n flutterBridge.log(\"获取章节内容出错:\" + e.message);\r\n return \"\";\r\n }\r\n }\r\n \/\/获取发现\r\nasync function getfinds() {\r\n try {\r\n \/\/ 请求网站首页,获取包含nav的HTML结构\r\n var get = await http.Get(baseurl, JSON.stringify(header), true);\r\n var $tempContainer = parseHTMLSafely(removeHTMLTags(get.data));\r\n var sort = [];\r\n \/\/ 【修改1:移除.not(\"[href='\/']\"),保留首页链接,遍历所有导航a标签】\r\n var $navLinks = $tempContainer.find(\".nav ul li a\");\r\n\r\n \/\/ 遍历所有导航链接,解析每个分类的标题、URL\r\n $navLinks.each(function () {\r\n var $a = $(this);\r\n var title = $a.text().trim();\r\n var href = $a.attr(\"href\") || \"\";\r\n if (!title || !href) return true;\r\n\r\n \/\/ 拼接完整URL + 处理分页占位符:将\/1\/替换为\/{{page}}\/\r\n var fullUrl = baseurl + href;\r\n fullUrl = fullUrl.replace(\/\\\/1\\\/\/, \"\/{{page}}\/\");\r\n\r\n sort.push({\r\n \"title\": title,\r\n \"url\": fullUrl,\r\n \"type\": 0\r\n });\r\n });\r\n\r\n \/\/ 【修改2:移除解析结果的最后一个分类,slice(0, -1)不修改原数组,更安全】\r\n if (sort.length > 0) {\r\n sort = sort.slice(0, -1);\r\n }\r\n\r\n return JSON.stringify(sort);\r\n } catch (e) {\r\n flutterBridge.log(\"获取发现页分类出错:\" + (e.message || \"未知错误\"));\r\n return \"[]\";\r\n }\r\n}\r\n\r\n \/\/解析发现列表\r\n\r\n function parsefindList(html) {\r\n try {\r\n var $tempContainer = parseHTMLSafely(removeHTMLTags(html));\r\n var books = [];\r\n var bookUrls = new Set(); \/\/ 统一去重,两种结构共用\r\n\r\n $tempContainer.find(\".item\").each(function () {\r\n var $item = $(this);\r\n var book = {\r\n \"bookUrl\": \"\",\r\n \"name\": \"\",\r\n \"author\": \"\",\r\n \"kind\": \"\",\r\n \"coverUrl\": \"\",\r\n \"intro\": \"\",\r\n \"tocUrl\": \"\",\r\n \"wordCount\": \"\",\r\n \"type\": 0,\r\n \"latestChapterTitle\": \"\"\r\n };\r\n\r\n \/\/ 提取bookUrl并过滤\r\n var href = $item.find(\".image a\").first().attr('href');\r\n if (!href || href.includes('javascript:') || href.includes('#')) return true;\r\n \/\/ URL规范化\r\n book.bookUrl = href.startsWith('http') ? href : (href.startsWith('\/') ? baseurl + href : '');\r\n if (!book.bookUrl || bookUrls.has(book.bookUrl)) return true;\r\n bookUrls.add(book.bookUrl);\r\n\r\n \/\/ 拼接tocUrl:从txt数字.html提取数字\r\n var numMatch = book.bookUrl.match(\/txt(\\d+)\\.html\/);\r\n book.tocUrl = numMatch ? baseurl + \"\/indexlist\/\" + numMatch[1] + \"\/\" : book.bookUrl;\r\n\r\n \/\/ 提取书名并校验\r\n book.name = $item.find(\"dl dt a\").first().text().trim();\r\n if (!book.name || book.name.length < 2) return true;\r\n\r\n \/\/ 提取作者\/字数\/简介(新增简介)\r\n book.author = $item.find(\".btm a\").first().text().trim() || \"\";\r\n book.wordCount = $item.find(\".btm em.orange\").first().text().trim() || \"\";\r\n book.intro = $item.find(\"dl dd\").first().text().trim() || \"\"; \/\/ 新增:提取简介\r\n\r\n \/\/ 提取封面:和原parseBookList一致,取last(),优先data-original\r\n var $coverImg = $item.find(\".image img\").last();\r\n if ($coverImg.length > 0) {\r\n var src = $coverImg.attr('data-original');\r\n if (src) {\r\n book.coverUrl = src.startsWith('http') ? src : (src.startsWith('\/') ? baseurl + src : \"\");\r\n }\r\n }\r\n\r\n books.push(book);\r\n });\r\n\r\n $tempContainer.find(\"li:has(span.s1)\").each(function () {\r\n var $li = $(this);\r\n var book = {\r\n \"bookUrl\": \"\",\r\n \"name\": \"\",\r\n \"author\": \"\",\r\n \"kind\": \"\",\r\n \"coverUrl\": \"\",\r\n \"intro\": \"\",\r\n \"tocUrl\": \"\",\r\n \"wordCount\": \"\",\r\n \"type\": 0,\r\n \"latestChapterTitle\": \"\"\r\n };\r\n\r\n \/\/ 提取bookUrl(span.s2 a的链接)并过滤\r\n var $s2A = $li.find(\"span.s2 a\").first();\r\n var href = $s2A.attr('href');\r\n if (!href || href.includes('javascript:') || href.includes('#')) return true;\r\n \/\/ URL规范化\r\n book.bookUrl = href.startsWith('http') ? href : (href.startsWith('\/') ? baseurl + href : '');\r\n if (!book.bookUrl || bookUrls.has(book.bookUrl)) return true;\r\n bookUrls.add(book.bookUrl);\r\n\r\n \/\/ 拼接tocUrl:兼容txt数字.html 和 纯数字两种链接格式\r\n var numMatch = book.bookUrl.match(\/txt(\\d+)\\.html\/) || book.bookUrl.match(\/\\\/(\\d+)\\\/\/);\r\n book.tocUrl = numMatch ? baseurl + \"\/indexlist\/\" + numMatch[1] + \"\/\" : book.bookUrl;\r\n\r\n \/\/ 提取书名并校验\r\n book.name = $s2A.text().trim();\r\n if (!book.name || book.name.length < 2) return true;\r\n\r\n \/\/ 提取分类\/最新章节\/作者(结构2核心字段)\r\n book.kind = $li.find(\"span.s1\").first().text().trim() || \"\"; \/\/ 分类:如武侠\r\n book.latestChapterTitle = $li.find(\"span.s3 a\").first().text().trim() || \"\"; \/\/ 最新章节\r\n book.author = $li.find(\"span.s4\").first().text().trim() || \"\"; \/\/ 作者:如佚名\r\n\r\n \/\/ 无封面\/字数\/简介,留空;无其他字段,直接push\r\n books.push(book);\r\n });\r\n\r\n \/\/ 清理DOM+返回,和原parseBookList一致\r\n removeHTMLSafely($tempContainer);\r\n return JSON.stringify(books);\r\n } catch (e) {\r\n flutterBridge.log(\"解析发现页书籍列表出错:\" + (e.message || \"未知错误\"));\r\n return \"[]\";\r\n }\r\n }\r\n\r\n \/\/ 发现页书籍列表函数\r\n async function find(url, page) {\r\n try {\r\n \/\/ APP传入的URL包含{{page}}占位符,需要替换为实际页码\r\n if (url.includes('{{page}}')) {\r\n url = url.replace('{{page}}', page);\r\n }\r\n\r\n var get = await http.Get(url, JSON.stringify(header), true);\r\n\r\n return parsefindList(get.data);\r\n } catch (e) {\r\n flutterBridge.log(\"获取发现页出错:\" + e.message);\r\n return \"[]\";\r\n }\r\n }\r\n\r\n \/\/ 导出函数供APP调用\r\n window.search = search;\r\n window.info = info;\r\n window.chapter = chapter;\r\n window.content = content;\r\n window.getfinds = getfinds;\r\n window.find = find;\r\n\r\n<\/script>\r\n\r\n<\/html>",
"login": false,
"lastUpdateTime": "1770977621352"
}