爱奇电子书

http://www.xqb8.org/

autobcb_admin (12020)3天前

爱奇电子书 作者 星辰

二维码导入(APP尚未完成该功能)
{
    "bookSourceUrl": "http:\/\/www.xqb8.org\/",
    "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<\/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.xqb8.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\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\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】过滤无效链接(适配新书籍链接规则:\/book\/数字\/)\r\n            if (href.includes('javascript:') || !href.includes('\/book\/') || href.includes('#')) {\r\n                return true; \/\/ continue\r\n            }\r\n\r\n            \/\/ 提取书籍ID - 适配新链接格式 \/book\/173271\/\r\n            var match = href.match(\/book\\\/(\\d+)\\\/?\/); \/\/ 匹配\/book\/数字\/ 或 \/book\/数字\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        \/\/ 1. 提取书名 - 适配新结构:h1直接包含书名(无子a标签)\r\n        var name = \"\";\r\n        var $titleElem = $tempContainer.find(\"h1\").first();\r\n        if ($titleElem.length > 0) {\r\n            \/\/ 直接提取h1文本(新结构h1无嵌套a标签)\r\n            name = $titleElem.text().trim();\r\n        }\r\n\r\n        \/\/ 2. 提取作者 - 精准匹配\"作者:\"后的链接,兼容原逻辑\r\n        var author = \"\";\r\n        var $authorLink = $tempContainer.find(\"p:contains('作者:') a[href*='\/author\/']\").first();\r\n        if ($authorLink.length > 0) {\r\n            author = $authorLink.text().trim();\r\n        }\r\n        \/\/ 备用方案:保留原逻辑防止结构变体\r\n        if (!author) {\r\n            $authorLink = $tempContainer.find(\"a[href*='\/author\/']\").first();\r\n            if ($authorLink.length > 0) {\r\n                author = $authorLink.text().trim();\r\n            }\r\n        }\r\n\r\n        \/\/ 3. 提取封面 - 精准定位#fmimg里的img,避免匹配其他图片\r\n        var coverUrl = \"\";\r\n        var $coverImg = $tempContainer.find(\"#fmimg img\").first();\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                    coverUrl = src;\r\n                } else if (src.startsWith('\/')) {\r\n                    coverUrl = baseurl + src;\r\n                }\r\n            }\r\n            \/\/ 备用:如果没有data-original,取src\r\n            if (!coverUrl) {\r\n                src = $coverImg.attr('src');\r\n                if (src && src.startsWith('\/')) {\r\n                    coverUrl = baseurl + src;\r\n                }\r\n            }\r\n        }\r\n\r\n        \/\/ 4. 提取分类 - 从#intro简介中提取(如\"玄幻\")\r\n        var kind = \"\";\r\n        var $introElem = $tempContainer.find(\"#intro\").first();\r\n        if ($introElem.length > 0) {\r\n            var introText = $introElem.text().trim();\r\n            \/\/ 匹配\"创作的xxx。\"格式提取分类(如《xxx》是xxx精心创作的玄幻。)\r\n            var kindMatch = introText.match(\/创作的([^。]+)。\/);\r\n            if (kindMatch && kindMatch[1]) {\r\n                kind = kindMatch[1].trim();\r\n            }\r\n        }\r\n\r\n        \/\/ 5. 提取简介 - 直接提取#intro的文本,清理多余换行\r\n        var intro = \"\";\r\n        if ($introElem.length > 0) {\r\n            intro = $introElem.text().trim().replace(\/\\s+\/g, \" \").replace(\/\\.\\.\\.\/g, \"\");\r\n        }\r\n\r\n        \/\/ 6. 提取字数 - 适配新结构的\"字数:408 万字\"格式(含空格)\r\n        var wordCount = \"\";\r\n        \/\/ 优先匹配visible-xs的字数行\r\n        var $wordXs = $tempContainer.find(\"p.visible-xs:contains('字数:')\").first();\r\n        if ($wordXs.length > 0) {\r\n            var wordText = $wordXs.text().trim();\r\n            var wordMatch = wordText.match(\/([\\d.]+)\\s*万字\/);\r\n            if (wordMatch) {\r\n                wordCount = wordMatch[1] + \"万字\"; \/\/ 统一格式为\"408万字\"(去掉空格)\r\n            }\r\n        }\r\n        \/\/ 备用:匹配hidden-xs里的字数\r\n        if (!wordCount) {\r\n            var $wordHidden = $tempContainer.find(\"p.hidden-xs\").first();\r\n            if ($wordHidden.length > 0) {\r\n                var wordText = $wordHidden.text().trim();\r\n                var wordMatch = wordText.match(\/([\\d.]+)\\s*万字\/);\r\n                if (wordMatch) {\r\n                    wordCount = wordMatch[1] + \"万字\";\r\n                }\r\n            }\r\n        }\r\n\r\n        \/\/ 7. 提取最新章节 - 精准匹配\"最新章节:\"后的链接\r\n        var latestChapterTitle = \"\";\r\n        \/\/ 优先匹配hidden-xs的最新章节\r\n        var $latestChapterLink = $tempContainer.find(\"p.hidden-xs:contains('最新章节:') a\").first();\r\n        if (!$latestChapterLink.length) {\r\n            \/\/ 备用:匹配visible-xs的lastchapter里的链接\r\n            $latestChapterLink = $tempContainer.find(\".lastchapter.visible-xs:contains('最新章节:') a\").first();\r\n        }\r\n        if ($latestChapterLink.length > 0) {\r\n            latestChapterTitle = $latestChapterLink.text().trim();\r\n        }\r\n\r\n        \/\/ 8. 提取目录链接 - 适配新结构:\/list\/书籍ID\/(替代原\/indexlist\/数字\/)\r\n        var tocUrl = bookurl;\r\n        \/\/ 从bookurl提取书籍ID(适配\/book\/173271\/格式)\r\n        var bookIdMatch = bookurl.match(\/book\\\/(\\d+)\\\/?\/);\r\n        if (bookIdMatch) {\r\n            var bookId = bookIdMatch[1];\r\n            tocUrl = baseurl + \"\/list\/\" + bookId + \"\/\";\r\n        }\r\n        \/\/ 备用:从页面的章节目录链接直接提取(更精准)\r\n        var $tocLink = $tempContainer.find(\"a.chapterlist\").first();\r\n        if ($tocLink.length > 0) {\r\n            var tocHref = $tocLink.attr('href');\r\n            if (tocHref) {\r\n                tocUrl = tocHref.startsWith('\/') ? baseurl + tocHref : tocHref;\r\n            }\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\nasync function chapter(tocUrl) {\r\n    try {\r\n        const allChapters = [];\r\n        \/\/ 仅构建并请求第一页URL(pageIndex=0)\r\n        const pageIndex = 0;\r\n        const pageUrl = buildPageUrl(tocUrl, pageIndex);\r\n        flutterBridge.log(\"获取第 \" + (pageIndex + 1) + \" 页: \" + pageUrl);\r\n\r\n        \/\/ 仅请求第一页数据\r\n        const result = await getChapterPage(pageUrl);\r\n\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        }\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               \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\n\/\/ 发现页分类函数\r\nasync function getfinds() {\r\n    try {\r\n        var sort = [\r\n            { \"title\": \"首页\", \"url\": baseurl + \"\/\", \"type\": 0 },\r\n            { \"title\": \"武侠小说\", \"url\": baseurl + \"\/html\/list-1-{{page}}.html\", \"type\": 0 },\r\n            { \"title\": \"言情小说\", \"url\": baseurl + \"\/html\/list-2-{{page}}.html\", \"type\": 0 },\r\n            { \"title\": \"玄幻小说\", \"url\": baseurl + \"\/html\/list-3-{{page}}.html\", \"type\": 0 },\r\n            { \"title\": \"都市小说\", \"url\": baseurl + \"\/html\/list-4-{{page}}.html\", \"type\": 0 },\r\n            { \"title\": \"穿越小说\", \"url\": baseurl + \"\/html\/list-5-{{page}}.html\", \"type\": 0 },\r\n            { \"title\": \"科幻小说\", \"url\": baseurl + \"\/html\/list-6-{{page}}.html\", \"type\": 0 },\r\n            { \"title\": \"网游小说\", \"url\": baseurl + \"\/html\/list-7-{{page}}.html\", \"type\": 0 },\r\n            { \"title\": \"同人小说\", \"url\": baseurl + \"\/html\/list-8-{{page}}.html\", \"type\": 0 },\r\n            { \"title\": \"历史小说\", \"url\": baseurl + \"\/html\/list-9-{{page}}.html\", \"type\": 0 },\r\n            { \"title\": \"惊悚小说\", \"url\": baseurl + \"\/html\/list-10-{{page}}.html\", \"type\": 0 },\r\n            { \"title\": \"重生小说\", \"url\": baseurl + \"\/html\/list-11-{{page}}.html\", \"type\": 0 },\r\n            { \"title\": \"耽美小说\", \"url\": baseurl + \"\/html\/list-12-{{page}}.html\", \"type\": 0 },\r\n            { \"title\": \"热门排行榜\", \"url\": baseurl + \"\/rank\/allvisit\/\", \"type\": 0 },\r\n            { \"title\": \"全本小说\", \"url\": baseurl + \"\/quanben\/html\/list-{{page}}\", \"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\nfunction 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            \/\/ 【核心修改1】拼接tocUrl:优先匹配\/book\/数字\/格式,兼容原txt数字.html格式\r\n            var numMatch = book.bookUrl.match(\/book\\\/(\\d+)\\\/?\/) || book.bookUrl.match(\/txt(\\d+)\\.html\/);\r\n            book.tocUrl = numMatch ? baseurl + \"\/list\/\" + numMatch[1] + \"\/\" : book.bookUrl; \/\/ 适配新toc路径\/list\/数字\/\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                \/\/ 备用:如果没有data-original,取src\r\n                if (!book.coverUrl) {\r\n                    src = $coverImg.attr('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            \/\/ 【核心修改2】拼接tocUrl:优先匹配\/book\/数字\/格式,兼容原txt数字.html和纯数字格式\r\n            var numMatch = book.bookUrl.match(\/book\\\/(\\d+)\\\/?\/) || book.bookUrl.match(\/txt(\\d+)\\.html\/) || book.bookUrl.match(\/\\\/(\\d+)\\\/\/);\r\n            book.tocUrl = numMatch ? baseurl + \"\/list\/\" + numMatch[1] + \"\/\" : book.bookUrl; \/\/ 适配新toc路径\/list\/数字\/\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": "1770977655916"
}
广告