读书阁

https://m.wzlygd.com

autobcb_admin (12020)3天前

读书阁 作者 星辰

二维码导入(APP尚未完成该功能)
{
    "bookSourceUrl": "https:\/\/m.wzlygd.com",
    "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<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 = \"https:\/\/m.wzlygd.com\"\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\nasync function search(key, page) {\r\n    \/\/ 页码大于1直接返回空数组,保持原逻辑\r\n    if (page > 1) {\r\n        return \"[]\";\r\n    }\r\n\r\n    try {\r\n        const url = baseurl+\"\/search\/\"; \r\n        const myheader = {\r\n            ...header,\r\n            \"Referer\": baseurl\r\n        };\r\n       \r\n        const body = \"searchkey=\"+encodeURIComponent(key)+\"&submit=1\";\r\n        const contentType = \"application\/x-www-form-urlencoded\";\r\n        const res = await http.Post(url, JSON.stringify(myheader), body, contentType, true);\r\n        \r\n        if (!res) {\r\n            flutterBridge.log(\"搜索请求失败:桥接调用异常\");\r\n            return \"[]\";\r\n        }\r\n\r\n        flutterBridge.text(0, res.data);\r\n        return parseBookList(res.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        \/\/ 遍历所有.bookbox容器,每个容器对应一本书(真实结构)\r\n        $tempContainer.find(\".bookbox\").each(function () {\r\n            var $bookBox = $(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            \/\/ 1. 提取bookUrl:.bookbox .bookimg a 的href(精准匹配真实结构)\r\n            var $bookA = $bookBox.find(\".bookimg a\").first();\r\n            var href = $bookA.attr('href');\r\n            if (!href) {\r\n                return true; \/\/ 无链接,跳过当前bookbox\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            if (!book.bookUrl.endsWith('\/')) {\r\n                book.bookUrl += '\/';\r\n            }\r\n            \/\/ 去重校验\r\n            if (bookUrls.has(book.bookUrl)) {\r\n                return true;\r\n            }\r\n            bookUrls.add(book.bookUrl);\r\n            book.tocUrl = book.bookUrl; \/\/ 目录链接和书籍链接一致\r\n\r\n            \/\/ 2. 提取name:.bookbox .bookinfo h4.bookname a 的文本(纯书名,无嵌套干扰)\r\n            var $bookNameA = $bookBox.find(\".bookinfo h4.bookname a\").first();\r\n            book.name = $bookNameA.text().trim();\r\n            \/\/ 书名非空校验,过滤无效名称\r\n            if (!book.name || book.name.length < 2) {\r\n                return true;\r\n            }\r\n\r\n            \/\/ 3. 提取author:第一个.author,移除「作者:」前缀(核心修复!)\r\n            var $authorItems = $bookBox.find(\".bookinfo .author\"); \/\/ 找到所有.authordiv\r\n            if ($authorItems.length >= 1) {\r\n                var authorText = $authorItems.eq(0).text().trim();\r\n                book.author = authorText.replace(\/^作者[::\\s]*\/, \"\"); \/\/ 兼容中英文冒号\r\n            }\r\n\r\n            \/\/ 4. 提取kind:第二个.author,移除「类型:」前缀(核心修复!)\r\n            if ($authorItems.length >= 2) {\r\n                var kindText = $authorItems.eq(1).text().trim();\r\n                book.kind = kindText.replace(\/^类型[::\\s]*\/, \"\"); \/\/ 兼容中英文冒号\r\n            }\r\n\r\n            \/\/ 5. 提取latestChapterTitle:.update a 的文本(纯章节名,去掉「最后章节:」)\r\n            var $updateA = $bookBox.find(\".bookinfo .update a\").first();\r\n            if ($updateA.length > 0) {\r\n                book.latestChapterTitle = $updateA.text().trim();\r\n            }\r\n\r\n            \/\/ 6. 提取coverUrl:.bookimg img 的src(精准封面链接,保留原URL补全逻辑)\r\n            var $bookImg = $bookBox.find(\".bookimg img\").first();\r\n            if ($bookImg.length > 0) {\r\n                var src = $bookImg.attr('src');\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            \/\/ 7. 提取状态(已完结\/连载中):从span提取,无则留空(保留原逻辑)\r\n            $bookBox.find(\"span\").each(function () {\r\n                var text = $(this).text().trim();\r\n                if (text === '已完结' || text === '连载中') {\r\n                    book.wordCount = text;\r\n                }\r\n            });\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, header, true);\r\n        \r\n        if (!get) {\r\n            flutterBridge.log(\"获取书籍详情失败:请求返回空\");\r\n            return \"{}\";\r\n        }\r\n\r\n        flutterBridge.text(1, get.data);\r\n        var $tempContainer = parseHTMLSafely(removeHTMLTags(get.data));\r\n        \/\/ 初始化书籍对象,默认值为空\r\n        var book = {\r\n            \"bookUrl\": bookurl,\r\n            \"name\": \"\",\r\n            \"author\": \"\",\r\n            \"kind\": \"\",\r\n            \"coverUrl\": \"\",\r\n            \"intro\": \"\",\r\n            \"tocUrl\": bookurl,\r\n            \"wordCount\": \"\", \/\/ 存状态:连载\/已完结(原结构无字数字段)\r\n            \"type\": 0,\r\n            \"latestChapterTitle\": \"\" \/\/ 移除原尾逗号,兼容低版本JS\r\n        };\r\n\r\n        \/\/ 1. 提取书名 - 保留原h1精准逻辑(详情页书名通用位置,兜底)\r\n        var $titleLink = $tempContainer.find(\"h1 a\").first();\r\n        if ($titleLink.length > 0) {\r\n            book.name = $titleLink.text().trim();\r\n        } else {\r\n            var $titleElem = $tempContainer.find(\"h1\").first();\r\n            if ($titleElem.length > 0) {\r\n                \/\/ 只取h1直接文本,排除子元素\r\n                var pureText = $titleElem.clone().children().remove().end().text().trim();\r\n                book.name = pureText ? pureText : $titleElem.text().trim();\r\n                \/\/ 移除可能的字数前缀\r\n                book.name = book.name.replace(\/^[\\d.]+万字\/, '').trim();\r\n            }\r\n        }\r\n\r\n        \/\/ 2. 提取作者 - 【精准匹配】.synopsisArea_detail b.author,失败则回退原逻辑\r\n        var $authorTag = $tempContainer.find(\".synopsisArea .synopsisArea_detail b.author\").first();\r\n        if ($authorTag.length > 0) {\r\n            var authorText = $authorTag.text().trim();\r\n            book.author = authorText.replace(\/^作者[::\\s]*\/, \"\"); \/\/ 兼容中英文冒号\r\n        } else {\r\n            \/\/ 兜底:原通用逻辑(找作者链接)\r\n            var $authorLink = $tempContainer.find(\"a[href*='\/zuozhe\/']\").first();\r\n            if ($authorLink.length > 0) {\r\n                book.author = $authorLink.text().trim().replace(\/^作者[:\\s]*\/, '');\r\n            }\r\n        }\r\n\r\n        \/\/ 3. 提取封面 - 【精准匹配】.synopsisArea_detail img,失败则回退原逻辑\r\n        var $coverImg = $tempContainer.find(\".synopsisArea .synopsisArea_detail img\").first();\r\n        if ($coverImg.length > 0) {\r\n            var src = $coverImg.attr('src');\r\n            if (src) {\r\n                book.coverUrl = src.startsWith('http') ? src : (src.startsWith('\/') ? baseurl + src : \"\");\r\n            }\r\n        } else {\r\n            \/\/ 兜底:原通用逻辑(找页面第一个img)\r\n            var $img = $tempContainer.find(\"img\").first();\r\n            if ($img.length > 0) {\r\n                var fallSrc = $img.attr('src');\r\n                if (fallSrc) {\r\n                    book.coverUrl = fallSrc.startsWith('http') ? fallSrc : (fallSrc.startsWith('\/') ? baseurl + fallSrc : \"\");\r\n                }\r\n            }\r\n        }\r\n\r\n        \/\/ 4. 提取分类 - 【精准匹配】.synopsisArea_detail p.sort,移除类别前缀\r\n        var $sortTag = $tempContainer.find(\".synopsisArea .synopsisArea_detail p.sort\").first();\r\n        if ($sortTag.length > 0) {\r\n            var kindText = $sortTag.text().trim();\r\n            book.kind = kindText.replace(\/^类别[::\\s]*\/, \"\"); \/\/ 兼容中英文冒号\r\n        } else {\r\n            \/\/ 兜底:原通用逻辑(从分类链接提取)\r\n            $tempContainer.find(\"a\").each(function () {\r\n                var href = $(this).attr('href');\r\n                var text = $(this).text().trim();\r\n                if (href && href.match(\/\\\/(xuanhuan|xianxia|dushi|lishi|junshi|kehuan|yanqing)\\\/\/) && text.length < 10) {\r\n                    book.kind = text;\r\n                    return false; \/\/ break循环\r\n                }\r\n            });\r\n        }\r\n\r\n        \/\/ 5. 提取状态(存wordCount)- 【精准匹配】.synopsisArea_detail 包含“状态:”的p标签\r\n        $tempContainer.find(\".synopsisArea .synopsisArea_detail p\").each(function () {\r\n            var statusText = $(this).text().trim();\r\n            if (statusText.startsWith('状态:')) {\r\n                book.wordCount = statusText.replace(\/^状态[::\\s]*\/, \"\");\r\n                return false; \/\/ break循环\r\n            }\r\n        });\r\n\r\n        \/\/ 6. 提取最新章节 - 【精准匹配】.synopsisArea_detail p.lastchapter a\r\n        var $lastChapterA = $tempContainer.find(\".synopsisArea .synopsisArea_detail p.lastchapter a\").first();\r\n        if ($lastChapterA.length > 0) {\r\n            book.latestChapterTitle = $lastChapterA.text().trim();\r\n        } else {\r\n            \/\/ 兜底:原通用逻辑(找.html章节链接)\r\n            var $chapterLinks = $tempContainer.find(\"a[href*='.html']\");\r\n            if ($chapterLinks.length > 0) {\r\n                book.latestChapterTitle = $chapterLinks.first().text().trim();\r\n            }\r\n        }\r\n\r\n        \/\/ 7. 提取简介 - 【精准匹配】.synopsisArea .review,过滤“相关推荐”段落,拼接有效内容\r\n        var introArr = [];\r\n        var $reviewContainer = $tempContainer.find(\".synopsisArea .review\").first();\r\n        if ($reviewContainer.length > 0) {\r\n            $reviewContainer.find(\"p\").each(function () {\r\n                var pText = $(this).text().trim();\r\n                \/\/ 过滤:空文本、包含“相关推荐”的段落\r\n                if (!pText || pText.includes(\"相关推荐:\")) {\r\n                    return true; \/\/ continue\r\n                }\r\n                introArr.push(pText);\r\n            });\r\n            \/\/ 拼接简介,用换行分隔(更贴合原文格式)\r\n            book.intro = introArr.join(\"\\n\");\r\n        } else {\r\n            \/\/ 兜底:原通用简介提取逻辑\r\n            var $introContainer = $tempContainer.clone();\r\n            $introContainer.find(\"#list, #pages, .pages\").remove();\r\n            $introContainer.find(\"p, div\").each(function () {\r\n                var text = $(this).text().trim();\r\n                if (text.length > 50 && text.length < 500 &&\r\n                    !text.includes(\"最新章节\") && !text.includes(\"作者\") &&\r\n                    !text.includes(\"更新时间\") && !text.includes(\"第\") && !text.includes(\"章\")) {\r\n                    if (!book.intro || text.length > book.intro.length) {\r\n                        book.intro = text;\r\n                    }\r\n                }\r\n            });\r\n        }\r\n\r\n        \/\/ 清理临时DOM容器,防止内存泄漏\r\n        removeHTMLSafely($tempContainer);\r\n        \/\/ 转JSON字符串返回\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                var pageUrl;\r\n                if (pageIndex === 0) {\r\n                    pageUrl = tocUrl;\r\n                } else {\r\n                    var baseUrl = tocUrl.endsWith('\/') ? tocUrl.slice(0, -1) : tocUrl;\r\n                    pageUrl = baseUrl + \"\/\" + (pageIndex + 1) + \"\/\";\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(\".directoryArea\").last();\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(\"div.Readarea\").first();\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\n\r\n\/\/ 解析发现页书籍列表 - 适配.hot_sale容器,结构\/逻辑完全仿照parseBookList\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        \/\/ 遍历所有.hot_sale容器,每个容器对应一本书(精准匹配目标结构)\r\n        $tempContainer.find(\".hot_sale\").each(function () {\r\n            var $hotSale = $(this);\r\n            \/\/ 书籍对象结构和parseBookList完全一致,无则留空\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            \/\/ 1. 提取bookUrl:.hot_sale a 的href,沿用原URL规范化逻辑\r\n            var $bookA = $hotSale.find(\"a\").first();\r\n            var href = $bookA.attr('href');\r\n            if (!href) {\r\n                return true; \/\/ 无链接,跳过当前容器\r\n            }\r\n            \/\/ 补全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            \/\/ 确保URL以\/结尾,和parseBookList保持一致\r\n            if (!book.bookUrl.endsWith('\/')) {\r\n                book.bookUrl += '\/';\r\n            }\r\n            \/\/ 去重校验,避免重复书籍\r\n            if (bookUrls.has(book.bookUrl)) {\r\n                return true;\r\n            }\r\n            bookUrls.add(book.bookUrl);\r\n            book.tocUrl = book.bookUrl; \/\/ 目录链接和书籍链接一致\r\n\r\n            \/\/ 2. 提取name:.hot_sale a p.title 的文本,非空校验\r\n            var $titleP = $bookA.find(\"p.title\").first();\r\n            book.name = $titleP.text().trim();\r\n            \/\/ 过滤无效书名(长度<2),和parseBookList校验规则一致\r\n            if (!book.name || book.name.length < 2) {\r\n                return true;\r\n            }\r\n\r\n            \/\/ 3. 提取author:.hot_sale a p.author,移除「作者:」前缀+剔除末尾(日期)后缀\r\n            var $authorP = $bookA.find(\"p.author\").first();\r\n            if ($authorP.length > 0) {\r\n                var authorText = $authorP.text().trim();\r\n                authorText = authorText.replace(\/^作者[::\\s]*\/, \"\"); \/\/ 移除前缀,兼容中英文冒号\r\n                authorText = authorText.replace(\/(.*?)$\/, \"\"); \/\/ 剔除末尾(日期)后缀,如(2026-01-23)\r\n                book.author = authorText.trim(); \/\/ 二次去空格,避免残留空白\r\n            }\r\n\r\n            \/\/ 4. 提取intro:.hot_sale a p.review,移除「简介:」前缀\r\n            var $reviewP = $bookA.find(\"p.review\").first();\r\n            if ($reviewP.length > 0) {\r\n                var introText = $reviewP.text().trim();\r\n                introText = introText.replace(\/^简介[::\\s]*\/, \"\"); \/\/ 移除前缀,兼容中英文冒号\r\n                book.intro = introText;\r\n            }\r\n\r\n            \/\/ 以下字段:coverUrl\/kind\/wordCount\/latestChapterTitle\r\n            \/\/ 目标结构中无,按要求直接留空,无需额外处理\r\n\r\n            \/\/ 有效书籍加入数组\r\n            books.push(book);\r\n        });\r\n\r\n        \/\/ 清理临时DOM容器,防止内存泄漏,和parseBookList保持一致\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        return \"[]\";\r\n    }\r\n}\r\nasync function getfinds() {\r\n    try {\r\n        var sort = [\r\n            \/\/ 书库核心分类 - 对齐你给的 \/quanben\/sort\/数字\/{{page}} 规则\r\n            { \"title\": \"全部分类\", \"url\": baseurl + \"\/quanben\/sort\/\", \"type\": 0 },\r\n            { \"title\": \"玄幻小说\", \"url\": baseurl + \"\/quanben\/sort\/1\/{{page}}\", \"type\": 0 },\r\n            { \"title\": \"修真小说\", \"url\": baseurl + \"\/quanben\/sort\/2\/{{page}}\", \"type\": 0 },\r\n            { \"title\": \"都市小说\", \"url\": baseurl + \"\/quanben\/sort\/3\/{{page}}\", \"type\": 0 },\r\n            { \"title\": \"穿越小说\", \"url\": baseurl + \"\/quanben\/sort\/4\/{{page}}\", \"type\": 0 },\r\n            { \"title\": \"网游小说\", \"url\": baseurl + \"\/quanben\/sort\/5\/{{page}}\", \"type\": 0 },\r\n            { \"title\": \"科幻小说\", \"url\": baseurl + \"\/quanben\/sort\/6\/{{page}}\", \"type\": 0 },\r\n            { \"title\": \"其他小说\", \"url\": baseurl + \"\/quanben\/sort\/7\/{{page}}\", \"type\": 0 },\r\n            { \"title\": \"现代言情\", \"url\": baseurl + \"\/quanben\/sort\/8\/{{page}}\", \"type\": 0 },\r\n            { \"title\": \"古代言情\", \"url\": baseurl + \"\/quanben\/sort\/9\/{{page}}\", \"type\": 0 },\r\n            { \"title\": \"短故事\",     \"url\": baseurl + \"\/quanben\/sort\/10\/{{page}}\", \"type\": 0 },\r\n            { \"title\": \"历史小说\", \"url\": baseurl + \"\/quanben\/sort\/11\/{{page}}\", \"type\": 0 },\r\n            { \"title\": \"军事小说\", \"url\": baseurl + \"\/quanben\/sort\/12\/{{page}}\", \"type\": 0 },\r\n            { \"title\": \"最新更新\", \"url\": baseurl + \"\/zuixin\/{{page}}\", \"type\": 0 },\r\n\r\n            \/\/ 排行榜分类 - 对齐你给的 \/rank\/xxx\/ 规则(无分页,直接写死路径)\r\n            { \"title\": \"总排行榜\", \"url\": baseurl + \"\/rank\/allvisit\/\", \"type\": 0 },\r\n            { \"title\": \"月排行榜\", \"url\": baseurl + \"\/rank\/monthvisit\/\", \"type\": 0 },\r\n            { \"title\": \"周排行榜\", \"url\": baseurl + \"\/rank\/weekvisit\/\", \"type\": 0 },\r\n            { \"title\": \"收 藏 榜\", \"url\": baseurl + \"\/rank\/goodnum\/\", \"type\": 0 },\r\n\r\n            \/\/ 完本分类 - 对应核心分类数字,用 \/wanjie\/数字\/{{page}} 规则,逻辑统一\r\n            { \"title\": \"完本玄幻\", \"url\": baseurl + \"\/wanjie\/1\/{{page}}\", \"type\": 0 },\r\n            { \"title\": \"完本修真\", \"url\": baseurl + \"\/wanjie\/2\/{{page}}\", \"type\": 0 },\r\n            { \"title\": \"完本都市\", \"url\": baseurl + \"\/wanjie\/3\/{{page}}\", \"type\": 0 },\r\n            { \"title\": \"完本穿越\", \"url\": baseurl + \"\/wanjie\/4\/{{page}}\", \"type\": 0 },\r\n            { \"title\": \"完本网游\", \"url\": baseurl + \"\/wanjie\/5\/{{page}}\", \"type\": 0 },\r\n            { \"title\": \"完本科幻\", \"url\": baseurl + \"\/wanjie\/6\/{{page}}\", \"type\": 0 },\r\n            { \"title\": \"完本言情\", \"url\": baseurl + \"\/wanjie\/8\/{{page}}\", \"type\": 0 },\r\n            { \"title\": \"完本历史\", \"url\": baseurl + \"\/wanjie\/11\/{{page}}\", \"type\": 0 },\r\n            { \"title\": \"完结小说\", \"url\": baseurl + \"\/wanjie\/{{page}}\", \"type\": 0 }\r\n        ];\r\n        \/\/ 按原函数格式,直接返回JSON字符串\r\n        return JSON.stringify(sort);\r\n    } catch (e) {\r\n        \/\/ 保持和search\/info等函数一致的异常处理,兜底返回空数组\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": "1770977855546"
}
广告