[{"bookSourceUrl":"https:\/\/www.bqg128.cc\/","bookSourceName":"笔趣阁128","enabledExplore":true,"enabled":true,"bookSourceGroup":"","author":"","help":false,"html":"<!DOCTYPE html>\n<html lang=\"en\">\n<head>\n    <meta charset=\"UTF-8\">\n    <title>Bqg128<\/title>\n<\/head>\n<body>\n\n<\/body>\n<script src=\"https:\/\/vc.jd.com\/web\/js\/jquery-3.1.1.min.js\"><\/script>\n<!--如果要引入外部 js 必须在书源代码的上面-->\n<script>\n  var isCookieJar=true;\/\/ 不需要CookieJar请修改此处\n  class FlutterJSBridge {\n    constructor() {\n      this.init(); \/\/前台webview 里必须删除这行\n    }\n\n    init() {\n      if (window.flutter_inappwebview) {\n        this.isReady = true;\n        this.CookieJar();\n      } else {\n        window.addEventListener('flutterInAppWebViewPlatformReady', () => {\n          this.isReady = true;\n          console.log('JSBridge初始化完成');\n          this.CookieJar();\n        });\n      }\n    }\n\n    \/\/通知原生页面初始化完成，仅在书源和tts生效，webview请勿使用，只有通知加载成功后才允许运行，否则会一直等待加载成功\n    async CookieJar() {\n      try {\n        await window.flutter_inappwebview.callHandler('CookieJar', isCookieJar);\n      } catch (error) {\n        console.error('汇报完成准备失败:', error);\n      }\n    }\n\n    \/\/获取应用编译版本\n    async getbuildNumber() {\n      try {\n        return await window.flutter_inappwebview.callHandler('buildNumber');\n      } catch (error) {\n        return 0;\n      }\n    }\n\n    \/\/获取应用版本\n    async getversion() {\n      try {\n        return await window.flutter_inappwebview.callHandler('version');\n      } catch (error) {\n        return \"0.0.0\";\n      }\n    }\n    \n    \/\/将html转换成正文格式        \n    async htmlToText(str) {\n      try {\n          return await window.flutter_inappwebview.callHandler('htmlToText',str);\n        } catch (error) {\n          return \"\";\n        }\n    }\n    \n    \/\/将简体字转成繁体字\n    async toTraditional(str) {\n      try {\n        return await window.flutter_inappwebview.callHandler('toTraditional',str);\n      } catch (error) {\n        return \"\";\n      }\n    }\n    \n    \n    \/\/将繁体字转成简体字\n    async toSimplified(str) {\n      try {\n        return await window.flutter_inappwebview.callHandler('toSimplified',str);\n      } catch (error) {\n        return \"\";\n      }\n    }\n\n    \/\/播放朗读引擎仅tts源生效\n    async voice() {\n      try {\n        return await window.flutter_inappwebview.callHandler('voice');\n      } catch (error) {\n        return \"\";\n      }\n    }\n   \n\n    \/\/获取设备唯一id\n    async getDeviceid() {\n      try {\n        return await window.flutter_inappwebview.callHandler('id');\n      } catch (error) {\n        return \"\";\n      }\n    }\n\n    \/\/获取设备平台 此处返回 windows、macos、ios、ohos、android\n    async getDevice() {\n      try {\n        return await window.flutter_inappwebview.callHandler('device');\n      } catch (error) {\n        return \"\";\n      }\n    }\n    \n    \/\/获取轻悦时光登录用户名，没登录返回为空\n    async getLoginUser() {\n      try {\n        return await window.flutter_inappwebview.callHandler('getLoginUser');\n      } catch (error) {\n        return \"\";\n      }\n    }\n\n    \/\/输出日志,前台webview请勿使用\n    \/\/str 为 String\n    async log(str) {\n      try {\n        return await window.flutter_inappwebview.callHandler('log',str);\n      } catch (error) {\n        return false;\n      }\n    }\n\n    \/\/书源调试时可输出 html 代码到前台\n    \/\/type 0 搜索源码 ， 1详情源码 ，2目录源码 ，3正文源码\n    \/\/str 为 String\n    \/\/type 为int\n    async text(type,str) {\n      try {\n        return await window.flutter_inappwebview.callHandler('text',type,str);\n      } catch (error) {\n        return false;\n      }\n    }\n\n    \/\/toast弹窗，显示3秒\n    \/\/str 为 String\n    async showToast(str) {\n      try {\n        return await window.flutter_inappwebview.callHandler('showToast',str);\n      } catch (error) {\n        return false;\n      }\n    }\n    \n    \/\/长toast弹窗，显示10秒\n    \/\/str 为 String\n    async showLongToast(str) {\n      try {\n        return await window.flutter_inappwebview.callHandler('showLongToast',str);\n      } catch (error) {\n        return false;\n      }\n    }\n\n    \/\/webview 里禁止使用，webview请使用js获取ua （navigator.userAgent）\n    \/\/获取默认ua\n    async getWebViewUA() {\n      try {\n        return await window.flutter_inappwebview.callHandler('getWebViewUA');\n      } catch (error) {\n        return \"\";\n      }\n    }\n\n    \/\/通过url打开外部应用\n    \/\/url 为 String\n    async openurl(url) {\n      try {\n        return await window.flutter_inappwebview.callHandler('openurl',url,\"\");\n      } catch (error) {\n        return false;\n      }\n    }\n\n    \/\/通过url打开外部应用并附带mimeType\n    \/\/url 为 String\n    \/\/mimeType 为 String\n    async openurlwithMimeType(url,mimeType) {\n      try {\n        return await window.flutter_inappwebview.callHandler('openurl',url,mimeType);\n      } catch (error) {\n        return false;\n      }\n    }\n\n    \/**\n     * 使用webView访问网络\n     * @param html 直接用webView载入的html, 如果html为空直接访问url\n     * @param url html内如果有相对路径的资源不传入url访问不了\n     * @param js 用来取返回值的js语句, 没有就返回整个源代码\n     * @param body 当参数不为空的时候，会以post请求，此时请务必在 header 中带上content-type\n     * @param header 请求的header头，此参数必须是json字符串\n     * @return 返回js获取的内容\n     *\/\n    async webview(url,js,html,body,header) {\n      try {\n        return await window.flutter_inappwebview.callHandler('webview',url,js,html,body,header,\"\",\"\");\n      } catch (error) {\n        return \"\";\n      }\n    }\n\n    \/**\n     * overrideUrlRegex 为正则表达式\n     * 使用方法和上面的一样\n     * 但返回的内容为正则到的内容，如果无法正则到则返回 js 获取的内容，如果 js 为空则返回页面 html\n     *\/\n    async webViewGetOverrideUrl(url,js,html,body,header,overrideUrlRegex) {\n      try {\n        return await window.flutter_inappwebview.callHandler('webview',url,js,html,body,header,overrideUrlRegex,\"\");\n      } catch (error) {\n        return \"\";\n      }\n    }\n\n    \/**\n     * 使用webView获取资源url\n     * urlregex 为正则表达式\n     * 使用方法和上面的一样\n     * 但返回的内容为正则到的内容，如果无法正则到则返回 js 获取的内容，如果 js 为空则返回页面 html\n     *\/\n    async webViewGetSource(url,js,html,body,header,urlregex) {\n      try {\n        return await window.flutter_inappwebview.callHandler('webview',url,js,html,body,header,\"\",urlregex);\n      } catch (error) {\n        return \"\";\n      }\n    }\n    \n    \/**\n     * 使用webView拦截 ajax\n     * ajaxregex 为正则表达式，通过 ajax 匹配 path\n     * 匹配成功返回 ajax 的结果 失败返回 html\n     *\/\n    async webViewGetAjax(url,html,body,header,ajaxregex) {\n      try {\n        return await window.flutter_inappwebview.callHandler('webviewajax',url,html,body,header,ajaxregex);\n      } catch (error) {\n        return \"\";\n      }\n    }\n\n\n\n    \/**\n     * 启动前台 webview 访问链接并获取结束时的 html，可用于手工过盾\n     * @param url 网址\n     * @param title 标题\n     * @param header 请求的header头，此参数必须是json字符串\n     * @return 返回网页的内容\n     *\/\n    async startBrowser(url,title,header) {\n      try {\n        return await window.flutter_inappwebview.callHandler('startBrowser',url,title,header);\n      } catch (error) {\n        return \"\";\n      }\n    }\n    \n    \/**\n     * 启动前台 webview 并对每次打开的 url 进行拦截\n     * @param url 网址\n     * @param title 标题\n     * @param header 请求的header头，此参数必须是json字符串\n     *\/\n    async startBrowserWithShouldOverrideUrlLoading(url,title,header) {\n      try {\n        return await window.flutter_inappwebview.callHandler('startBrowserWithShouldOverrideUrlLoading',url,title,header);\n      } catch (error) {\n        return \"\";\n      }\n    }\n\n    \/\/专门为段评设置的半屏显示，不返回任何东西\n    async startBrowserDp(url,title) {\n      try {\n        return await window.flutter_inappwebview.callHandler('startBrowserDp',url,title);\n      } catch (error) {\n        return \"\";\n      }\n    }\n\n    \/\/仅前台webview可以使用，返回按钮，返回上一个页面\n    async back() {\n      try {\n        return await window.flutter_inappwebview.callHandler('back');\n      } catch (error) {\n        return false;\n      }\n    }\n\n    \/\/将 utf8字符串转到 gbk 并 url编码\n    async utf8ToGbkUrlEncoded(str) {\n      try {\n        return await window.flutter_inappwebview.callHandler('utf8ToGbkUrlEncoded',str);\n      } catch (error) {\n        return \"\";\n      }\n    }\n\n    \/*\n    * @param str为图片链接 \n    * @param header 请求的header头，此参数必须是json字符串\n    * 此函数是让用户输入图片中的验证码，当链接为空则直接让用户输入验证码\n    *\/\n    async getVerificationCode(str,header) {\n      try {\n        return await window.flutter_inappwebview.callHandler('getVerificationCode',str,header);\n      } catch (error) {\n        return \"\";\n      }\n    }\n    \n    \/\/提交内容bookUrl,我会调用书源 info 函数来获取这本书的信息\n    async addbook(bookUrl) {\n      try {\n        return await window.flutter_inappwebview.callHandler('addbook',bookUrl);\n      } catch (error) {\n        return \"\";\n      }\n    }\n    \n    \n    \/\/获取书本当前阅读章节index\n    async getdurChapterIndex(bookUrl) {\n      try {\n        return await window.flutter_inappwebview.callHandler('getdurChapterIndex',bookUrl);\n      } catch (error) {\n        return 0;\n      }\n    }\n    \n    \/\/utf8 字符串转base64\n    async base64encode(str) {\n      try {\n        return await window.flutter_inappwebview.callHandler('base64encode',str);\n      } catch (error) {\n        return \"\";\n      }\n    }\n    \n    \/\/base64 转utf8字符串\n    async base64decode(str) {\n      try {\n        return await window.flutter_inappwebview.callHandler('base64decode',str);\n      } catch (error) {\n        return \"\";\n      }\n    }\n    \n    \n\n  }\n\n  \/\/webview下isCookieJar必定true 会自动处理cookie\n  \/\/以下提交的url，headers,body 都必须为字符串,headers必须为json字符串\n  \/\/当followRedirects 为 false 时不处理重定向，当为 true 时会自动处理重定向 ，如不明白用途直接用 true 最佳\n  \/\/ 以下所有参数除当followRedirects外均为 String\n  \/\/ 如果需要使用http2协议请在url前添加 http2:\/\/ ，例如 http2:\/\/baidu.com\n  \/\/ 如果https一直被盾拦截 ，可以使用https2协议\n  class Http {\n    constructor() {\n      \/*\n       * 速率限制配置\n       * requestTimestamps: 存储请求时间戳的数组\n       * rateLimit: 速率限制，单位时间内最大请求数\n       * rateLimitWindow: 速率限制窗口，单位毫秒\n       *\/\n      this.open = false; \/\/ 是否开启速率限制\n      this.requestTimestamps = []; \/\/ 存储请求时间戳的数组\n      this.rateLimit = 5; \/\/ 速率限制，1000毫秒内最多5次请求\n      this.rateLimitWindow = 1000; \/\/ 速率限制窗口，1000毫秒\n    }\n\n    \/*\n     * 检查速率限制\n     * 实现方法：\n     * 1. 获取当前时间戳\n     * 2. 过滤掉超出时间窗口的时间戳\n     * 3. 检查是否超过速率限制\n     * 4. 如果超过限制，计算需要等待的时间并等待\n     * 5. 递归检查速率限制\n     * 6. 将当前时间戳添加到数组中\n     *\/\n    async checkRateLimit() {\n      if(!this.open) return;\n      const now = Date.now();\n      \/\/ 过滤掉超出时间窗口的时间戳\n      this.requestTimestamps = this.requestTimestamps.filter(timestamp => now - timestamp < this.rateLimitWindow);\n      \/\/ 检查是否超过速率限制\n      if (this.requestTimestamps.length >= this.rateLimit) {\n        \/\/ 计算需要等待的时间\n        const oldestTimestamp = this.requestTimestamps[0];\n        const waitTime = this.rateLimitWindow - (now - oldestTimestamp);\n        \/\/ 等待到速率限制可用\n        await new Promise(resolve => setTimeout(resolve, waitTime));\n        \/\/ 递归检查速率限制\n        return this.checkRateLimit();\n      }\n      \/\/ 将当前时间戳添加到数组中\n      this.requestTimestamps.push(now);\n    }\n\n\n    \/*\n     * 通用返回字段\n     * method post get 或者 head\n     * body 请求返回后的字节的 base64\n     * headers map<String,List<String>> 可通过headers[\"\"]来或者\n     * statusCode 状态码\n     * statusMessage\n     * data 返回后的字节格式化后的内容\n     *\/\n    async Get(url,headers,followRedirects) {\n      try {\n        await this.checkRateLimit();\n        return await window.flutter_inappwebview.callHandler('http',\"get\",url,\"\",JSON.stringify(headers),followRedirects,\"\");\n      } catch (error) {\n        return null;\n      }\n    }\n\n    async Head(url,headers,followRedirects) {\n      try {\n        await this.checkRateLimit();\n        return await window.flutter_inappwebview.callHandler('http',\"head\",url,\"\",JSON.stringify(headers),followRedirects,\"\");\n      } catch (error) {\n        return null;\n      }\n    }\n\n    \n    async Post(url,headers,body,contenttype,followRedirects) {\n      try {\n        await this.checkRateLimit();\n        return await window.flutter_inappwebview.callHandler('http',\"post\",url,body,JSON.stringify(headers),followRedirects,contenttype);\n      } catch (error) {\n        return null;\n      }\n    }\n  }\n\n  class Cache {\n    constructor() {}\n    async get(key) {\n      try {\n        return await window.flutter_inappwebview.callHandler('cache.get',key);\n      } catch (error) {\n        return null;\n      }\n    }\n\n    async set(key,value) {\n      try {\n        return await window.flutter_inappwebview.callHandler('cache.set',key,value);\n      } catch (error) {\n        return null;\n      }\n    }\n\n    async remove(key) {\n      try {\n        return await window.flutter_inappwebview.callHandler('cache.remove',key);\n      } catch (error) {\n        return null;\n      }\n    }\n\n    \/\/如果登录为弹窗格式的，里面输入框输入的内容可以通过这个函数获取，默认返回的json格式或者为空，需要自行转换\n    async getLoginInfo(){\n      return await this.get(\"LoginInfo\")\n    }\n\n    \/\/将修改后的弹窗输入内容报错 ，必须 JSON.stringify，不然会出错\n    async putLoginInfo(info){\n      return await this.set(\"LoginInfo\",info)\n    }\n   \n    \/\/获取书本变量\n    async getbookVariable(bookurl){\n      return await this.get(bookurl)\n    }\n    \n    \/\/写入书本变量\n    async setbookVariable(bookurl,value){\n      return await this.set(bookurl,value)\n    }\n  }\n\n  class Cookie {\n    constructor() {}\n\n    \/\/通过url获取当前url的所有cookie\n    async get(url) {\n      try {\n        return await window.flutter_inappwebview.callHandler('cookie.get',url);\n      } catch (error) {\n        return null;\n      }\n    }\n\n    \/\/通过url删除当前url的所有cookie\n    async remove(url) {\n      try {\n        return await window.flutter_inappwebview.callHandler('cookie.remove',url);\n      } catch (error) {\n        return null;\n      }\n    }\n\n    \/\/通过url保存当前url的所有cookie\n    async set(url,value) {\n      try {\n        return await window.flutter_inappwebview.callHandler('cookie.set',url,value);\n      } catch (error) {\n        return null;\n      }\n    }\n    \n    \/\/设置单独一个cookie\n    async setCookie(url,key,value) {\n      try {\n        return await window.flutter_inappwebview.callHandler('cookie.setcookie',url,key,value);\n      } catch (error) {\n        return null;\n      }\n    }\n\n    \/\/通过 url 获取单个 cookie 的值\n    async getCookie(url,value) {\n      try {\n        return await window.flutter_inappwebview.callHandler('cookie.getCookie',url,value);\n      } catch (error) {\n        return null;\n      }\n    }\n  }\n\n  \/\/安全的创建一个 div 解析 html\n  function parseHTMLSafely(htmlStr) {\n    try {\n      \/\/ 在函数作用域内创建独立的临时容器\n      \/\/ 每个调用创建新的jQuery对象，互不影响\n      var tempDiv = document.createElement('div');\n      tempDiv.innerHTML = htmlStr;\n      return $(tempDiv);\n    } catch (e) {\n      flutterBridge.log(\"HTML解析错误:\"+e.message);\n      return $('<div>');\n    }\n  }\n\n  \/\/parseHTMLSafely 创建的用完后必须删除\n  function removeHTMLSafely(tempContainer) {\n    try {\n      tempContainer.innerHTML = '';\n      if (tempContainer.parentNode) {\n        tempContainer.parentNode.removeChild(tempContainer);\n      }\n    } catch (e) {\n      flutterBridge.log(\"HTML移除失败:\"+e.message);\n    }\n  }\n\n  \/\/移除 css js，创建parseHTMLSafely前如果用不上cssjs建议移除\n  function removeHTMLTags(htmlString) {\n    \/\/ 移除script标签\n    let result = htmlString.replace(\/<script\\b[^<]*(?:(?!<\\\/script>)<[^<]*)*<\\\/script>\/gi, '');\n    \/\/ 移除style标签\n    result = result.replace(\/<style\\b[^<]*(?:(?!<\\\/style>)<[^<]*)*<\\\/style>\/gi, '');\n    return result;\n  }\n\n<\/script>\n\n<script>\n    const flutterBridge = new FlutterJSBridge();\n    const cache = new Cache();\n    const http = new Http();\n    const cookie = new Cookie();\n    var baseurl=\"https:\/\/www.bqg128.cc\"\n    var header={\n        \"User-Agent\": \"Mozilla\/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit\/537.36 (KHTML, like Gecko) Chrome\/142.0.0.0 Safari\/537.36 Edg\/142.0.0.0\",\n        \"Referer\": baseurl\n    };\n\n    function isCloudflareChallenge(html) {\n        return html && html.indexOf('Just a moment') !== -1;\n    }\n\n    \/\/===========================================\n    \/\/ 📚 智能容器定位工具函数（新增功能）\n    \/\/===========================================\n    \n    \/\/ 找到包含最多符合规则链接的上层容器\n    function findBestContainer($htmlContainer, linkMatcher, minMatchCount) {\n        var $bestContainer = null;\n        var maxMatchCount = 0;\n        var checkedParents = new Set();\n        \n        var $allLinks = $htmlContainer.find(\"a\");\n        $allLinks.each(function() {\n            var $link = $(this);\n            var href = $link.attr('href');\n            \n            \/\/ 检查链接是否匹配规则\n            if (!linkMatcher(href)) {\n                return true; \/\/ continue\n            }\n            \n            var $parent = $link.parent();\n            var depth = 0;\n            var maxDepth = 10;\n            \n            \/\/ 向上查找，找到包含最多匹配链接的容器\n            while ($parent.length > 0 && depth < maxDepth) {\n                var parentKey = $parent[0].outerHTML.substring(0, Math.min(500, $parent[0].outerHTML.length));\n                \n                if (!checkedParents.has(parentKey)) {\n                    checkedParents.add(parentKey);\n                    \n                    var count = $parent.find(\"a\").filter(function() {\n                        return linkMatcher($(this).attr('href'));\n                    }).length;\n                    \n                    if (count > maxMatchCount && count >= minMatchCount) {\n                        maxMatchCount = count;\n                        $bestContainer = $parent;\n                    }\n                }\n                \n                $parent = $parent.parent();\n                depth++;\n            }\n        });\n        \n        return $bestContainer;\n    }\n\n    \/\/ 找到包含指定文本区域的合适容器\n    function findContainerByMarker($htmlContainer, markerTexts, maxDepth) {\n        var $resultContainer = null;\n        \n        for (var i = 0; i < markerTexts.length; i++) {\n            var marker = markerTexts[i];\n            \n            $htmlContainer.find(\"*\").each(function() {\n                var $elem = $(this);\n                var text = $elem.text().trim();\n                \n                if (text.indexOf(marker) !== -1) {\n                    var $parent = $elem.parent();\n                    var depth = 0;\n                    \n                    while ($parent.length > 0 && depth < maxDepth) {\n                        $resultContainer = $parent;\n                        $parent = $parent.parent();\n                        depth++;\n                    }\n                    return false;\n                }\n            });\n            \n            if ($resultContainer) {\n                break;\n            }\n        }\n        \n        return $resultContainer;\n    }\n\n    \/\/ 提取相对URL，处理baseurl\n    function normalizeUrl(url) {\n        if (!url) return \"\";\n        if (url.startsWith('http')) return url;\n        if (url.startsWith('\/')) return baseurl + url;\n        return baseurl + '\/' + url;\n    }\n\n    \/*\n     * ============================================\n     * TODO: search(key, page) - 搜索功能\n     * ============================================\n     * 分析步骤（必须实际访问网站！）：\n     * 1. 使用 curl 访问网站主页，查找搜索表单\n     * 2. 分析搜索表单的 action URL 和 method（GET还是POST）\n     * 3. 查找搜索参数名（如 searchkey, keyword, q 等）\n     * 4. 测试搜索请求，查看搜索结果HTML\n     * 5. 分析搜索结果中的书籍容器和字段\n     *\n     * 示例（实际测试过）：\n     * curl -s \"https:\/\/www.22biqu.com\" | grep -i \"form\\|input\\|search\"\n     * curl -s -X POST \"https:\/\/www.22biqu.com\/ss\/\" -d \"searchkey=凡人修仙传\"\n     *\/\n    async function search(key, page) {\n        if(page > 1){\n            return \"[]\";\n        }\n\n        \/*\n         * TODO: 根据实际网站修改以下内容\n         * 1. 修改搜索URL（可能是GET或POST）\n         * 2. 修改body参数名（searchkey, keyword, q 等）\n         * 3. 修改linkMatcher（确定哪些链接是书籍链接）\n         * 4. 调整minMatchCount（最少需要找到几本书）\n         *\/\n        var searchUrl = baseurl + \"\/s\";\n        var searchParams = \"q=\" + encodeURIComponent(key);\n        var fullSearchUrl = searchUrl + \"?\" + searchParams;\n\n        flutterBridge.log(\"搜索URL: \" + fullSearchUrl);\n\n        \/\/ 📌 选择：有反爬用 webview，简单网站用 http\n        \/\/ var get = await http.Get(fullSearchUrl, JSON.stringify(header), true);\n        \/\/ var htmlContent = get ? get.data : \"\";\n        \n        var htmlContent = await flutterBridge.webview(fullSearchUrl, \"\", \"\", \"\", JSON.stringify(header));\n\n        if(!htmlContent || isCloudflareChallenge(htmlContent)){\n            flutterBridge.showToast(\"如果没有出现验证请手动搜索\");\n            var s = await flutterBridge.startBrowser(fullSearchUrl, \"验证\", JSON.stringify(header));\n            if(s){\n                htmlContent = s;\n            }\n        }\n\n        if(!htmlContent){\n            flutterBridge.showToast(\"搜索请求失败，请在浏览器中完成验证\");\n            return \"[]\";\n        }\n\n        flutterBridge.text(0, htmlContent);\n\n        var books = [];\n        var processedUrls = new Set();\n        var $tempContainer = parseHTMLSafely(htmlContent);\n\n        \/\/ 🔍 步骤一：智能定位搜索结果容器（新增功能）\n        \/\/ TODO: 修改 linkMatcher - 根据实际网站的书籍链接特征\n        var linkMatcher = function(href) {\n            if (!href) return false;\n            \/\/ 示例：链接包含 \/book\/ 等特征\n            return href.indexOf('\/book\/') !== -1;\n        };\n        \n        var minMatchCount = 3;\n        var $resultContainer = findBestContainer($tempContainer, linkMatcher, minMatchCount);\n        \n        if (!$resultContainer) {\n            $resultContainer = $tempContainer;\n            flutterBridge.log(\"警告：未找到专门的搜索结果容器，使用全页搜索\");\n        } else {\n            flutterBridge.log(\"找到搜索结果容器\");\n        }\n\n        \/\/ 🔍 步骤二：在找到的容器内提取书籍\n        var index = 0;\n        $resultContainer.find(\".bookbox\").each(function() {\n            try{\n                var $bookBox = $(this);\n                var $bookInfo = $bookBox.find(\".bookinfo\");\n                var $bookImg = $bookBox.find(\".bookimg\");\n                \n                \/\/ 提取书籍链接\n                var $bookLink = $bookInfo.find(\".bookname a\");\n                var bookUrl = $bookLink.attr('href');\n                \n                if (!bookUrl || processedUrls.has(bookUrl)) {\n                    return true; \/\/ continue\n                }\n                \n                if (!linkMatcher(bookUrl)) {\n                    return true;\n                }\n                \n                processedUrls.add(bookUrl);\n                \n                \/\/ 提取书籍信息\n                var bookName = $bookLink.text().trim();\n                var author = $bookInfo.find(\".author\").text().trim().replace(\/^作者：\/, \"\").trim();\n                var coverUrl = $bookImg.find(\"img\").attr('src') || \"\";\n                var intro = $bookInfo.find(\".uptime\").text().trim();\n\n                if(bookName && bookName.length > 1){\n                    bookUrl = normalizeUrl(bookUrl);\n                    coverUrl = normalizeUrl(coverUrl);\n\n                    var book={\n                        \"bookUrl\": bookUrl,\n                        \"name\": bookName,\n                        \"author\": author,\n                        \"kind\": \"\",\n                        \"coverUrl\": coverUrl,\n                        \"intro\": intro,\n                        \"tocUrl\": bookUrl,\n                        \"wordCount\": \"\",\n                        \"type\": 0,\n                        \"latestChapterTitle\": \"\"\n                    };\n                    books.push(book);\n                    index++;\n                }\n            }catch(e){\n                flutterBridge.log(\"解析书籍信息出错: \" + e.message);\n            }\n        });\n\n        removeHTMLSafely($tempContainer);\n        flutterBridge.log(\"找到 \" + books.length + \" 本书\");\n        return JSON.stringify(books);\n    }\n\n    \/*\n     * ============================================\n     * TODO: info(bookurl) - 书籍详情\n     * ============================================\n     * 分析步骤：\n     * 1. curl 访问书籍详情页\n     * 2. 查找书名位置（通常在 h1 或特定class中）\n     * 3. 查找作者、分类、封面、简介的位置\n     * 4. 查找最新章节信息\n     *\n     * 示例：\n     * curl -s \"https:\/\/www.22biqu.com\/biqu105084\/\" | head -200\n     *\/\n    async function info(bookurl) {\n        var mheader={\n            ...header,\n            \"Referer\": baseurl\n        };\n\n        \/\/ 📌 选择：有反爬用 webview，简单网站用 http\n        \/\/ var get = await http.Get(bookurl, JSON.stringify(mheader), true);\n        \/\/ var htmlContent = get ? get.data : \"\";\n        \n        var htmlContent = await flutterBridge.webview(bookurl, \"\", \"\", \"\", JSON.stringify(mheader));\n\n        if(!htmlContent || isCloudflareChallenge(htmlContent)){\n            flutterBridge.showToast(\"如果没有出现验证请手动打开书籍\");\n            var s = await flutterBridge.startBrowser(bookurl, \"验证\", JSON.stringify(mheader));\n            if(s){\n                htmlContent = s;\n            }\n        }\n\n        if(!htmlContent){\n            flutterBridge.showToast(\"获取信息失败，请在浏览器中完成验证\");\n            return JSON.stringify({});\n        }\n\n        flutterBridge.text(1, htmlContent);\n\n        var $tempContainer = parseHTMLSafely(htmlContent);\n\n        \/\/ 提取基本信息\n        var name = \"\";\n        var author = \"\";\n        var kind = \"\";\n        var coverUrl = \"\";\n        var intro = \"\";\n        var latestChapterTitle = \"\";\n\n        \/\/ 书名：优先找 h1\/h2\n        name = $tempContainer.find(\"h1\").first().text().trim();\n        if (!name) {\n            name = $tempContainer.find(\"h2\").first().text().trim();\n        }\n        if (!name) {\n            name = $tempContainer.find(\".book-title, .title\").first().text().trim();\n        }\n\n        \/\/ 作者和分类\n        $tempContainer.find(\"p, span, div\").each(function() {\n            var text = $(this).text();\n            if(text.indexOf(\"作者\") !== -1 && !author){\n                author = text.replace(\/[^:：]+[：:]\\s*\/, \"\").trim();\n            }\n            if(text.indexOf(\"分类\") !== -1 || text.indexOf(\"类别\") !== -1){\n                kind = text.replace(\/[^:：]+[：:]\\s*\/, \"\").trim();\n            }\n        });\n\n        \/\/ 如果没找到明确的\"作者\"标记，尝试其他方式\n        if (!author) {\n            $tempContainer.find(\"span, p\").each(function() {\n                var text = $(this).text().trim();\n                if (text && text.length > 0 && text.length < 30 && !author) {\n                    if (text.indexOf(\"万字\") === -1 && text.indexOf(\"作者\") === -1 && text.indexOf(\"分类\") === -1) {\n                        author = text;\n                    }\n                }\n            });\n        }\n\n        \/\/ 封面\n        var possibleCoverSelectors = [\n            \".cover img\", \".book-cover img\", \"#bookimg img\", \n            \"img[src*='book']\", \"img[src*='cover']\", \"img\"\n        ];\n        for (var i = 0; i < possibleCoverSelectors.length && !coverUrl; i++) {\n            var $img = $tempContainer.find(possibleCoverSelectors[i]).first();\n            if ($img.length > 0) {\n                coverUrl = $img.attr('src') || \"\";\n            }\n        }\n\n        \/\/ 简介\n        var possibleIntroSelectors = [\".intro\", \".desc\", \".book-intro\", \".summary\", \"p\"];\n        for (var j = 0; j < possibleIntroSelectors.length && !intro; j++) {\n            var text = $tempContainer.find(possibleIntroSelectors[j]).first().text().trim();\n            if (text && text.length > 20) {\n                intro = text;\n            }\n        }\n\n        \/\/ 最新章节\n        var possibleChapterSelectors = [\".latest a\", \".update a\", \".new-chapter a\", \"a\"];\n        for (var k = 0; k < possibleChapterSelectors.length && !latestChapterTitle; k++) {\n            $tempContainer.find(possibleChapterSelectors[k]).each(function() {\n                var text = $(this).text().trim();\n                if (text && (text.indexOf(\"章\") !== -1 || text.indexOf(\"节\") !== -1 || text.length > 3)) {\n                    latestChapterTitle = text;\n                    return false;\n                }\n            });\n        }\n\n        coverUrl = normalizeUrl(coverUrl);\n\n        var book={\n            \"bookUrl\": bookurl,\n            \"name\": name,\n            \"author\": author,\n            \"kind\": kind,\n            \"coverUrl\": coverUrl,\n            \"intro\": intro,\n            \"tocUrl\": bookurl,\n            \"wordCount\": \"\",\n            \"type\": 0,\n            \"latestChapterTitle\": latestChapterTitle\n        };\n\n        removeHTMLSafely($tempContainer);\n        return JSON.stringify(book);\n    }\n\n    \/*\n     * ============================================\n     * TODO: chapter(tocUrl) - 章节列表\n     * ============================================\n     * 分析步骤：\n     * 1. 先检查是否有专门的章节列表页面\n     * 2. 查找标记文本（如\"章节目录\"、\"目录\"等）\n     * 3. 找到包含最多章节链接的容器\n     * 4. 在找到的容器内提取所有章节\n     *\n     * 常见特征：\n     * - 章节链接通常包含 \/chapter\/、\/read\/ 等路径\n     * - 章节名称通常包含\"章\"、\"节\"、\"第\"等\n     *\/\n    async function chapter(tocUrl) {\n        var mheader={\n            ...header,\n            \"Referer\": baseurl\n        };\n\n        var chapters = [];\n        var processedUrls = new Set();\n        var chapterIndex = 0;\n        var currentPage = 1;\n        var hasMorePages = true;\n\n        while (hasMorePages) {\n            \/\/ 构建当前页的URL\n            var chapterListUrl = tocUrl;\n            if (currentPage > 1) {\n                \/\/ 第二页及以后使用 _2.html 格式\n                chapterListUrl = tocUrl.replace(\/\\\/$\/, '') + \"_\" + currentPage + \".html\";\n            }\n\n            \/\/ 📌 选择：有反爬用 webview，简单网站用 http\n            \/\/ var get = await http.Get(chapterListUrl, JSON.stringify(mheader), true);\n            \/\/ var htmlContent = get ? get.data : \"\";\n            \n            var htmlContent = await flutterBridge.webview(chapterListUrl, \"\", \"\", \"\", JSON.stringify(mheader));\n\n            if(!htmlContent || isCloudflareChallenge(htmlContent)){\n                flutterBridge.showToast(\"如果没有出现验证请手动打开目录\");\n                var s = await flutterBridge.startBrowser(chapterListUrl, \"验证\", JSON.stringify(mheader));\n                if(s){\n                    htmlContent = s;\n                }\n            }\n\n            if(!htmlContent){\n                flutterBridge.showToast(\"获取目录失败，请在浏览器中完成验证\");\n                break;\n            }\n\n            flutterBridge.text(2, htmlContent);\n\n            var $tempContainer = parseHTMLSafely(htmlContent);\n\n            \/\/ 🎯 步骤一：定位章节目录区域\n            var $chapterContainer = null;\n            \n            \/\/ 方法1: 查找包含章节链接的dl元素（在.listmain中）\n            var $chapterDl = $tempContainer.find(\".listmain dl\").filter(function() {\n                var $links = $(this).find(\"a\");\n                var chapterLinkCount = 0;\n                $links.each(function() {\n                    var href = $(this).attr('href');\n                    if (href && \/\\\/book\\\/\\d+\\\/\\d+\\.html$\/.test(href)) {\n                        chapterLinkCount++;\n                    }\n                });\n                return chapterLinkCount > 5;\n            }).first();\n            \n            if ($chapterDl.length > 0) {\n                $chapterContainer = $chapterDl;\n                flutterBridge.log(\"找到章节列表容器: .listmain dl\");\n            } else {\n                $chapterContainer = $tempContainer;\n                flutterBridge.log(\"警告：未找到专门的章节列表容器，使用全页搜索\");\n            }\n\n            \/\/ 🎯 步骤二：在定位好的容器内提取章节\n            var pageChapters = 0;\n            $chapterContainer.find(\"dd a\").each(function() {\n                try{\n                    var $element = $(this);\n                    var chapterUrl = $element.attr('href');\n                    var chapterName = $element.text().trim();\n\n                    if(!chapterUrl || !chapterName || chapterName.length < 2){\n                        return true; \/\/ continue\n                    }\n\n                    if(processedUrls.has(chapterUrl)){\n                        return true;\n                    }\n\n                    \/\/ 验证章节链接格式\n                    var isChapterLink = \/\\\/book\\\/\\d+\\\/\\d+\\.html$\/.test(chapterUrl);\n                    if (!isChapterLink) {\n                        return true;\n                    }\n\n                    \/\/ 验证章节名称\n                    var isChapterName = chapterName.indexOf(\"章\") !== -1 || \n                                        chapterName.indexOf(\"节\") !== -1 ||\n                                        chapterName.indexOf(\"第\") !== -1 ||\n                                        \/^\\d+\/.test(chapterName);\n                    if (!isChapterName) {\n                        return true;\n                    }\n\n                    \/\/ 排除非章节内容（如作者信息、网站链接等）\n                    if (chapterName.indexOf(\"作者：\") !== -1 || \n                        chapterName.indexOf(\"笔趣阁\") !== -1 ||\n                        chapterName.indexOf(\"完本感言\") !== -1) {\n                        return true;\n                    }\n\n                    processedUrls.add(chapterUrl);\n                    \n                    \/\/ 清理章节名称\n                    var cleanedChapterName = chapterName.replace(\/\\s*\\([^)]*\\)$\/, '').trim();\n\n                    chapterUrl = normalizeUrl(chapterUrl);\n\n                    var chapter={\n                        \"name\": cleanedChapterName,\n                        \"chapterId\": chapterUrl,\n                        \"index\": chapterIndex,\n                        \"isPay\": false,\n                        \"isVip\": false,\n                        \"isVolume\": false,\n                        \"tag\": \"\"\n                    };\n                    chapters.push(chapter);\n                    chapterIndex++;\n                    pageChapters++;\n                }catch(e){\n                    flutterBridge.log(\"解析章节出错: \" + e.message);\n                }\n            });\n\n            \/\/ 检查是否有下一页\n            var $nextPageLink = $tempContainer.find(\"a\").filter(function() {\n                return $(this).text().indexOf(\"下一页\") !== -1;\n            });\n            \n            hasMorePages = $nextPageLink.length > 0 && pageChapters > 0;\n            currentPage++;\n\n            removeHTMLSafely($tempContainer);\n        }\n\n        flutterBridge.log(\"找到 \" + chapters.length + \" 个章节\");\n        return JSON.stringify(chapters);\n    }\n\n    \/*\n     * ============================================\n     * TODO: content(url) - 章节内容\n     * ============================================\n     * 分析步骤：\n     * 1. curl 访问章节页\n     * 2. 查找正文内容容器\n     * 3. 清理HTML标签，保留换行\n     * 4. 智能过滤广告和无关内容\n     *\/\n    async function content(url) {\n        var mheader={\n            ...header,\n            \"Referer\": baseurl\n        };\n\n        \/\/ 📌 选择：有反爬用 webview，简单网站用 http\n        \/\/ var get = await http.Get(url, JSON.stringify(mheader), true);\n        \/\/ var htmlContent = get ? get.data : \"\";\n        \n        var htmlContent = await flutterBridge.webview(url, \"\", \"\", \"\", JSON.stringify(mheader));\n\n        if(!htmlContent || isCloudflareChallenge(htmlContent)){\n            flutterBridge.showToast(\"如果没有出现验证请手动打开章节\");\n            var s = await flutterBridge.startBrowser(url, \"验证\", JSON.stringify(mheader));\n            if(s){\n                htmlContent = s;\n            }\n        }\n\n        if(!htmlContent){\n            flutterBridge.showToast(\"获取内容失败，请在浏览器中完成验证\");\n            return \"\";\n        }\n\n        flutterBridge.text(3, htmlContent);\n\n        var $tempContainer = parseHTMLSafely(htmlContent);\n        var contenttxt = \"\";\n\n        \/\/ 🔍 智能定位内容容器（新增功能）\n        var possibleContentSelectors = [\n            \"#content\", \".content\", \"#chapterContent\", \".novel-content\", \n            \"#novelcontent\", \".read-content\", \".article-content\"\n        ];\n        \n        var $contentContainer = null;\n        for (var i = 0; i < possibleContentSelectors.length && !$contentContainer; i++) {\n            var $found = $tempContainer.find(possibleContentSelectors[i]);\n            if ($found.length > 0) {\n                var textLength = $found.text().trim().length;\n                if (textLength > 100) {\n                    $contentContainer = $found;\n                }\n            }\n        }\n        \n        \/\/ 如果没找到，查找文本最多的div\n        if (!$contentContainer) {\n            var maxTextLength = 0;\n            $tempContainer.find(\"div, article\").each(function() {\n                var $div = $(this);\n                var textLength = $div.text().trim().length;\n                var pCount = $div.find(\"p\").length;\n                \n                \/\/ 评分：文本长度 + p标签数量*权重\n                var score = textLength + pCount * 100;\n                \n                if (score > maxTextLength && textLength > 200) {\n                    maxTextLength = score;\n                    $contentContainer = $div;\n                }\n            });\n        }\n\n        if ($contentContainer) {\n            \/\/ 提取内容，保留段落结构\n            $contentContainer.find(\"p\").each(function() {\n                var pText = $(this).text().trim();\n                if (pText) {\n                    \/\/ TODO: 过滤广告关键词\n                    var isAd = pText.indexOf(\"广告\") !== -1 || \n                              pText.indexOf(\"APP\") !== -1 ||\n                              pText.indexOf(\"最新网址\") !== -1;\n                    if (!isAd) {\n                        contenttxt += pText + \"\\r\\n\\r\\n\";\n                    }\n                }\n            });\n            \n            \/\/ 如果没有p标签，直接提取文本\n            if (!contenttxt) {\n                contenttxt = $contentContainer.text().trim();\n                \/\/ 适当添加换行\n                contenttxt = contenttxt.replace(\/(?<=[。！？!?])\\s*\/g, \"\\r\\n\\r\\n\");\n            }\n        }\n\n        \/\/ 兜底方案\n        if (!contenttxt || contenttxt.length < 50) {\n            contenttxt = $tempContainer.text().trim();\n        }\n\n        removeHTMLSafely($tempContainer);\n\n        if(!contenttxt || contenttxt.length < 50){\n            flutterBridge.showToast(\"获取内容失败\");\n            return \"\";\n        }\n\n        return contenttxt.trim();\n    }\n\n    \/*\n     * ============================================\n     * TODO: getfinds() - 发现\/分类\n     * ============================================\n     * 返回网站的所有分类\n     * URL格式使用 {{page}} 表示分页\n     *\n     * 分析步骤：\n     * 1. 查看主页的分类导航\n     * 2. 记录每个分类的名称和URL格式\n     * 3. URL中用 {{page}} 替换页码\n     *\/\n    async function getfinds() {\n        var result = [];\n        var push = (title, url, type) => result.push({\n            title: title,\n            url: url,\n            type: type || 0\n        });\n\n        \/\/ TODO: 修改为实际分类\n        push(\"发现\", \"\", 0);\n\n        \/\/ 实际分类：\n        push(\"玄幻\", \"\/#\/xuanhuan\", 0);\n        push(\"武侠\", \"\/#\/wuxia\", 0);\n        push(\"都市\", \"\/#\/dushi\", 0);\n        push(\"历史\", \"\/#\/lishi\", 0);\n        push(\"网游\", \"\/#\/wangyou\", 0);\n        push(\"科幻\", \"\/#\/kehuan\", 0);\n        push(\"女生\", \"\/#\/mm\", 0);\n        push(\"完本\", \"\/#\/finish\", 0);\n        push(\"排行榜\", \"\/#\/top\", 0);\n\n        return JSON.stringify(result);\n    }\n\n    \/*\n     * ============================================\n     * TODO: find(url, page) - 分类浏览\n     * ============================================\n     * 浏览某个分类的书籍列表\n     * 复用搜索结果的解析逻辑\n     *\/\n    async function find(url, page) {\n        \/\/ 分类页面使用哈希路由，不需要分页参数\n        var u = url;\n        if(!u.startsWith('http')){\n            u = baseurl + u;\n        }\n\n        flutterBridge.log(\"分类URL: \" + u);\n\n        var mheader={\n            ...header,\n            \"Referer\": baseurl\n        };\n\n        \/\/ 📌 选择：有反爬用 webview，简单网站用 http\n        \/\/ var get = await http.Get(u, JSON.stringify(mheader), true);\n        \/\/ var htmlContent = get ? get.data : \"\";\n        \n        var htmlContent = await flutterBridge.webview(u, \"\", \"\", \"\", JSON.stringify(mheader));\n\n        if(!htmlContent || isCloudflareChallenge(htmlContent)){\n            flutterBridge.showToast(\"如果没有出现验证请手动打开分类\");\n            var s = await flutterBridge.startBrowser(u, \"验证\", JSON.stringify(mheader));\n            if(s){\n                htmlContent = s;\n            }\n        }\n\n        if(!htmlContent){\n            flutterBridge.showToast(\"获取分类失败，请在浏览器中完成验证\");\n            return \"[]\";\n        }\n\n        flutterBridge.text(0, htmlContent);\n\n        var books = [];\n        var processedUrls = new Set();\n        var $tempContainer = parseHTMLSafely(htmlContent);\n\n        \/\/ 直接查找 .hot .item 容器\n        var $hotItems = $tempContainer.find(\".hot .item\");\n        flutterBridge.log(\"找到 \" + $hotItems.length + \" 个书籍项\");\n\n        \/\/ 提取书籍\n        var index = 0;\n        $hotItems.each(function() {\n            try{\n                var $item = $(this);\n                var $image = $item.find(\".image\");\n                var $dl = $item.find(\"dl\");\n                \n                \/\/ 提取书籍链接\n                var $bookLink = $dl.find(\"dt a\");\n                var bookUrl = $bookLink.attr('href');\n                \n                if (!bookUrl || processedUrls.has(bookUrl)) {\n                    return true; \/\/ continue\n                }\n                \n                \/\/ 处理哈希路由格式的链接\n                if (bookUrl.startsWith('#\/book\/')) {\n                    bookUrl = bookUrl.replace('#', '');\n                }\n                \n                \/\/ 验证书籍链接\n                if (!bookUrl || !bookUrl.includes('\/book\/')) {\n                    return true;\n                }\n                \n                processedUrls.add(bookUrl);\n                \n                \/\/ 提取书籍信息\n                var bookName = $bookLink.text().trim();\n                var author = $dl.find(\"dt span\").text().trim();\n                var coverUrl = $image.find(\"img\").attr('src') || \"\";\n                var intro = $dl.find(\"dd\").text().trim();\n\n                if(bookName && bookName.length > 1){\n                    bookUrl = normalizeUrl(bookUrl);\n                    coverUrl = normalizeUrl(coverUrl);\n\n                    var book={\n                        \"bookUrl\": bookUrl,\n                        \"name\": bookName,\n                        \"author\": author,\n                        \"kind\": \"\",\n                        \"coverUrl\": coverUrl,\n                        \"intro\": intro,\n                        \"tocUrl\": bookUrl,\n                        \"wordCount\": \"\",\n                        \"type\": 0,\n                        \"latestChapterTitle\": \"\"\n                    };\n                    books.push(book);\n                    index++;\n                }\n            }catch(e){\n                flutterBridge.log(\"解析书籍信息出错: \" + e.message);\n            }\n        });\n\n        removeHTMLSafely($tempContainer);\n        flutterBridge.log(\"找到 \" + books.length + \" 本书\");\n        return JSON.stringify(books);\n    }\n\n    async function getloginurl(){\n        return baseurl;\n    }\n\n<\/script>\n<\/html>\n","login":false,"lastUpdateTime":"1777193352601"}]