爱尚读书
http://www.aishangxs.net
autobcb_admin (12020)3天前
爱尚读书 作者 星辰
{
"bookSourceUrl": "http:\/\/www.aishangxs.net",
"bookSourceName": "爱尚读书",
"enabledExplore": true,
"enabled": true,
"bookSourceGroup": "雨落星辰",
"author": "星辰",
"help": false,
"html": "\r\n<!DOCTYPE html>\r\n<html lang=\"en\">\r\n\r\n<head>\r\n <meta charset=\"UTF-8\">\r\n <title>爱尚小说<\/title>\r\n<script type=\"text\/javascript\" nonce=\"dbab394703cd438cb8022acb0ce\" src=\"\/\/local.adguard.org?ts=1769152317581&type=content-script&dmn=svip.langge.cf&url=http%3A%2F%2Fsvip.langge.cf%2Fsudugu.html&app=msedge.exe&css=3&js=1&rel=1&rji=1&sbe=1&stealth=1&st-dnt\"><\/script><script type=\"text\/javascript\" nonce=\"dbab394703cd438cb8022acb0ce\" src=\"\/\/local.adguard.org?ts=1769152317581&name=AdGuard%20Extra&name=AdGuard%20Popup%20Blocker&type=user-script\"><\/script><\/head>\r\n\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.aishangxs.net\"\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 var url = baseurl + \"\/soso6ca15.html?searchkey=\" + encodeURIComponent(key);\r\n var myheader = {\r\n ...header,\r\n \"Referer\": baseurl\r\n };\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\r\n \/\/ 解析书籍列表\r\nfunction parseBookList(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 \/\/ 【核心修改1】遍历所有.item容器(新结构根节点),替代原h3 a遍历\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\": \"\", \/\/ 新结构存字数(如218万字),贴合原字段设计\r\n \"type\": 0,\r\n \"latestChapterTitle\": \"\"\r\n };\r\n\r\n \/\/ 1. 提取bookUrl:.item .image a 的href(新结构书籍链接)\r\n var $bookA = $item.find(\".image a\").first();\r\n var href = $bookA.attr('href');\r\n \/\/ 必须有href属性,无则跳过\r\n if (!href) {\r\n return true; \/\/ continue\r\n }\r\n \/\/ 【核心修改2】过滤无效链接(适配新书籍链接规则:\/read\/txt数字.html)\r\n if (href.includes('javascript:') || !href.includes('\/read\/txt') || href.includes('#')) {\r\n return true; \/\/ continue\r\n }\r\n\r\n \/\/ 提取书籍ID - 适配新链接格式 \/read\/txt112977.html\r\n var match = href.match(\/txt(\\d+)\\.html\/);\r\n if (!match) {\r\n return true; \/\/ 无有效书籍ID,跳过\r\n }\r\n var bookId = match[1];\r\n\r\n \/\/ URL规范化:补全baseurl、处理绝对\/相对路径(保留原逻辑)\r\n if (href.startsWith('http')) {\r\n book.bookUrl = href;\r\n } else if (href.startsWith('\/')) {\r\n book.bookUrl = baseurl + href;\r\n } else {\r\n return true; \/\/ 非有效路径,跳过\r\n }\r\n \/\/ 去重校验(保留原逻辑)\r\n if (bookUrls.has(book.bookUrl)) {\r\n return true; \/\/ continue\r\n }\r\n bookUrls.add(book.bookUrl);\r\n book.tocUrl = book.bookUrl; \/\/ 目录链接和书籍链接一致\r\n\r\n \/\/ 2. 提取书名:.item dl dt a 的文本(新结构书名位置)\r\n var $nameA = $item.find(\"dl dt a\").first();\r\n book.name = $nameA.text().trim();\r\n \/\/ 书名非空校验,过滤无效名称(保留原逻辑)\r\n if (!book.name || book.name.length < 2) {\r\n return true; \/\/ continue\r\n }\r\n\r\n \/\/ 3. 提取作者:.item .btm a 的文本(新结构作者位置,纯名无前缀)\r\n var $authorA = $item.find(\".btm a\").first();\r\n if ($authorA.length > 0) {\r\n book.author = $authorA.text().trim();\r\n }\r\n\r\n \/\/ 4. 提取字数:.item .btm em.orange 的文本(存到wordCount字段)\r\n var $wordEm = $item.find(\".btm em.orange\").first();\r\n if ($wordEm.length > 0) {\r\n book.wordCount = $wordEm.text().trim();\r\n }\r\n\r\n \/\/ 5. 提取封面:\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 if (src.startsWith('http')) {\r\n book.coverUrl = src;\r\n } else if (src.startsWith('\/')) {\r\n book.coverUrl = baseurl + src;\r\n }\r\n }\r\n }\r\n\r\n \/\/ kind\/latestChapterTitle\/intro:新结构无,按要求留空,无需处理\r\n\r\n \/\/ 有效书籍加入数组\r\n books.push(book);\r\n });\r\n\r\n \/\/ 清理临时DOM容器,防止内存泄漏(保留原逻辑)\r\n removeHTMLSafely($tempContainer);\r\n \/\/ 转JSON字符串返回,异常返回空数组(保留原逻辑)\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 info(bookurl) {\r\n try {\r\n var get = await http.Get(bookurl, JSON.stringify(header), true);\r\n flutterBridge.text(1, get.data);\r\n\r\n var $tempContainer = parseHTMLSafely(removeHTMLTags(get.data));\r\n\r\n \/\/ 提取书名 - 优先从h1 a标签提取(更准确)\r\n var name = \"\";\r\n var $titleLink = $tempContainer.find(\"h1 a\").first();\r\n if ($titleLink.length > 0) {\r\n \/\/ 直接从链接中提取书名\r\n name = $titleLink.text().trim();\r\n } else {\r\n \/\/ 备用方案:从h1标签提取\r\n var $titleElem = $tempContainer.find(\"h1\").first();\r\n if ($titleElem.length > 0) {\r\n \/\/ 只取h1的直接文本,不包括子元素\r\n name = $titleElem.clone().children().remove().end().text().trim();\r\n if (!name) {\r\n name = $titleElem.text().trim();\r\n }\r\n \/\/ 移除字数前缀(如 \"265.2万字\")\r\n name = name.replace(\/^[\\d.]+万字\/, '').trim();\r\n }\r\n }\r\n\r\n \/\/ 提取作者 - 查找作者链接\r\n var author = \"\";\r\n var $authorLink = $tempContainer.find(\"a[href*='\/author\/']\").first();\r\n if ($authorLink.length > 0) {\r\n author = $authorLink.text().trim().replace(\/^作者[:\\s]*\/, '');\r\n }\r\n\r\n \/\/ 提取封面\r\n var coverUrl = \"\";\r\n var $img = $tempContainer.find(\"img\").last();\r\n if ($img.length > 0) {\r\n var src = $img.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 \/\/ 提取分类 - 从链接中提取\r\n var kind = \"\";\r\n $tempContainer.find(\"a\").each(function () {\r\n var href = $(this).attr('href');\r\n var text = $(this).text().trim();\r\n \/\/ 查找分类链接(如\/xuanhuan\/, \/xianxia\/等)\r\n if (href && href.match(\/\\\/(xuanhuan|xianxia|dushi|lishi|junshi|kehuan|yanqing)\\\/\/) && text.length < 10) {\r\n kind = text;\r\n return false; \/\/ break\r\n }\r\n });\r\n\r\n \/\/ 提取简介 - 排除章节列表区域\r\n var intro = \"\";\r\n \/\/ 先移除#list区域,避免提取到章节列表\r\n var $introContainer = $tempContainer.clone();\r\n $introContainer.find(\"#list, #pages, .pages\").remove();\r\n\r\n \/\/ 查找包含较长文本的段落\r\n $introContainer.find(\"p, div\").each(function () {\r\n var text = $(this).text().trim();\r\n \/\/ 排除包含特定关键词的段落\r\n if (text.length > 50 && text.length < 500 &&\r\n !text.includes(\"最新章节\") &&\r\n !text.includes(\"作者\") &&\r\n !text.includes(\"更新时间\") &&\r\n !text.includes(\"第\") && \/\/ 排除章节标题\r\n !text.includes(\"章\")) {\r\n if (!intro || text.length > intro.length) {\r\n intro = text;\r\n }\r\n }\r\n });\r\n\r\n \/\/ 提取字数 - 查找包含\"万字\"的文本\r\n var wordCount = \"\";\r\n $tempContainer.find(\"*\").each(function () {\r\n var text = $(this).clone().children().remove().end().text().trim();\r\n if (text.match(\/^[\\d.]+万字$\/)) {\r\n wordCount = text;\r\n return false; \/\/ break\r\n }\r\n });\r\n\r\n \/\/ 提取最新章节\r\n var latestChapterTitle = \"\";\r\n var $chapterLinks = $tempContainer.find(\"a[href*='.html']\");\r\n if ($chapterLinks.length > 0) {\r\n latestChapterTitle = $chapterLinks.first().text().trim();\r\n }\r\n \/\/提取目录链接\r\n var tocUrl = bookurl; \r\n var numMatch = bookurl.match(\/txt(\\d+)\\.html\/); \r\n if (numMatch) {\r\n var bookNum = numMatch[1];\r\n tocUrl = baseurl + \"\/indexlist\/\" + bookNum + \"\/\"; \r\n }\r\n\r\n var book = {\r\n \"bookUrl\": bookurl,\r\n \"name\": name,\r\n \"author\": author,\r\n \"kind\": kind,\r\n \"coverUrl\": coverUrl,\r\n \"intro\": intro,\r\n \"tocUrl\": tocUrl,\r\n \"wordCount\": wordCount,\r\n \"type\": 0,\r\n \"latestChapterTitle\": latestChapterTitle,\r\n };\r\n\r\n removeHTMLSafely($tempContainer);\r\n return JSON.stringify(book);\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 chapter(tocUrl) {\r\n try {\r\n var allChapters = [];\r\n var pageIndex = 0;\r\n\r\n while (true) {\r\n \/\/ 修正分页URL格式: 第1页是基础URL, 第2页是p-2.html, 第3页是p-3.html\r\n var pageUrl;\r\n if (pageIndex === 0) {\r\n pageUrl = tocUrl;\r\n } else {\r\n \/\/ 移除末尾的\/,添加p-[page].html\r\n var baseUrl = tocUrl.endsWith('\/') ? tocUrl.slice(0, -1) : tocUrl;\r\n pageUrl = baseUrl + \"\/p-\" + (pageIndex + 1) + \".html\";\r\n }\r\n\r\n flutterBridge.log(\"正在获取第\" + (pageIndex + 1) + \"页: \" + pageUrl);\r\n\r\n var result = await getChapterPage(pageUrl);\r\n\r\n if (result.chapters.length > 0) {\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 break;\r\n }\r\n\r\n if (!result.hasNextPage) {\r\n flutterBridge.log(\"没有下一页,总共获取 \" + allChapters.length + \" 章\");\r\n break;\r\n }\r\n\r\n pageIndex++;\r\n if (pageIndex >= 100) {\r\n flutterBridge.log(\"目录页数超过100页,停止获取\");\r\n break;\r\n }\r\n }\r\n\r\n \/\/ 重新设置index\r\n for (var i = 0; i < allChapters.length; i++) {\r\n allChapters[i].index = i;\r\n }\r\n\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 \/\/ 获取单页章节\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(\"#list\");\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(\"#content\");\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\n\/\/ 发现页分类函数\r\nasync function getfinds() {\r\n try {\r\n var sort = [\r\n { \"title\": \"武侠小说\", \"url\": baseurl + \"\/html\/wuxiaxianxia\/list_{{page}}.html\", \"type\": 0 },\r\n { \"title\": \"言情小说\", \"url\": baseurl + \"\/html\/yanqing\/list_{{page}}.html\", \"type\": 0 },\r\n { \"title\": \"玄幻小说\", \"url\": baseurl + \"\/html\/xuanhuan\/list_{{page}}.html\", \"type\": 0 },\r\n { \"title\": \"都市小说\", \"url\": baseurl + \"\/html\/dushi\/list_{{page}}.html\", \"type\": 0 },\r\n { \"title\": \"穿越小说\", \"url\": baseurl + \"\/html\/chuanyue\/list_{{page}}.html\", \"type\": 0 },\r\n { \"title\": \"科幻小说\", \"url\": baseurl + \"\/html\/kehuanxiaoshuo\/list_{{page}}.html\", \"type\": 0 },\r\n { \"title\": \"网游小说\", \"url\": baseurl + \"\/html\/wangyou\/list_{{page}}.html\", \"type\": 0 },\r\n { \"title\": \"同人小说\", \"url\": baseurl + \"\/html\/tongren\/list_{{page}}.html\", \"type\": 0 },\r\n { \"title\": \"历史小说\", \"url\": baseurl + \"\/html\/lishi\/list_{{page}}.html\", \"type\": 0 },\r\n { \"title\": \"惊悚小说\", \"url\": baseurl + \"\/html\/jingsong\/list_{{page}}.html\", \"type\": 0 },\r\n { \"title\": \"重生小说\", \"url\": baseurl + \"\/html\/chongsheng\/list_{{page}}.html\", \"type\": 0 },\r\n { \"title\": \"耽美小说\", \"url\": baseurl + \"\/html\/danmei\/list_{{page}}.html\", \"type\": 0 },\r\n { \"title\": \"热门排行榜\", \"url\": baseurl + \"\/rank\/allvisit\/\", \"type\": 0 },\r\n { \"title\": \"全本小说\", \"url\": baseurl + \"\/quanben\/html\/{{page}}.html\", \"type\": 0 }\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": "1770977787605"
}