[{"bookSourceUrl":"http:\/\/www.shenpinwu.co\/","bookSourceName":"神书网","enabledExplore":true,"enabled":true,"bookSourceGroup":"","author":"trae","help":false,"html":"<!DOCTYPE html>\n<html lang=\"zh-CN\">\n<head>\n    <meta charset=\"UTF-8\">\n    <title>神书网<\/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<script>\n  var isCookieJar=true;\n  class FlutterJSBridge {\n    constructor() {\n      this.init();\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    async CookieJar() {\n      try {\n        await window.flutter_inappwebview.callHandler('CookieJar', isCookieJar);\n      } catch (error) {\n        console.error('汇报完成准备失败:', error);\n      }\n    }\n\n    async getbuildNumber() {\n      try { return await window.flutter_inappwebview.callHandler('buildNumber'); } catch (error) { return 0; }\n    }\n\n    async getversion() {\n      try { return await window.flutter_inappwebview.callHandler('version'); } catch (error) { return \"0.0.0\"; }\n    }\n    \n    async htmlToText(str) {\n      try { return await window.flutter_inappwebview.callHandler('htmlToText',str); } catch (error) { return \"\"; }\n    }\n    \n    async toTraditional(str) {\n      try { return await window.flutter_inappwebview.callHandler('toTraditional',str); } catch (error) { return \"\"; }\n    }\n    \n    async toSimplified(str) {\n      try { return await window.flutter_inappwebview.callHandler('toSimplified',str); } catch (error) { return \"\"; }\n    }\n\n    async voice() {\n      try { return await window.flutter_inappwebview.callHandler('voice'); } catch (error) { return \"\"; }\n    }\n   \n    async getDeviceid() {\n      try { return await window.flutter_inappwebview.callHandler('id'); } catch (error) { return \"\"; }\n    }\n\n    async getDevice() {\n      try { return await window.flutter_inappwebview.callHandler('device'); } catch (error) { return \"\"; }\n    }\n    \n    async getLoginUser() {\n      try { return await window.flutter_inappwebview.callHandler('getLoginUser'); } catch (error) { return \"\"; }\n    }\n\n    async log(str) {\n      try { return await window.flutter_inappwebview.callHandler('log',str); } catch (error) { return false; }\n    }\n\n    async text(type,str) {\n      try { return await window.flutter_inappwebview.callHandler('text',type,str); } catch (error) { return false; }\n    }\n\n    async showToast(str) {\n      try { return await window.flutter_inappwebview.callHandler('showToast',str); } catch (error) { return false; }\n    }\n    \n    async showLongToast(str) {\n      try { return await window.flutter_inappwebview.callHandler('showLongToast',str); } catch (error) { return false; }\n    }\n\n    async getWebViewUA() {\n      try { return await window.flutter_inappwebview.callHandler('getWebViewUA'); } catch (error) { return \"\"; }\n    }\n\n    async openurl(url) {\n      try { return await window.flutter_inappwebview.callHandler('openurl',url,\"\"); } catch (error) { return false; }\n    }\n\n    async openurlwithMimeType(url,mimeType) {\n      try { return await window.flutter_inappwebview.callHandler('openurl',url,mimeType); } catch (error) { return false; }\n    }\n\n    async webview(url,js,html,body,header) {\n      try { return await window.flutter_inappwebview.callHandler('webview',url,js,html,body,header,\"\",\"\"); } catch (error) { return \"\"; }\n    }\n\n    async webViewGetOverrideUrl(url,js,html,body,header,overrideUrlRegex) {\n      try { return await window.flutter_inappwebview.callHandler('webview',url,js,html,body,header,overrideUrlRegex,\"\"); } catch (error) { return \"\"; }\n    }\n\n    async webViewGetSource(url,js,html,body,header,urlregex) {\n      try { return await window.flutter_inappwebview.callHandler('webview',url,js,html,body,header,\"\",urlregex); } catch (error) { return \"\"; }\n    }\n    \n    async webViewGetAjax(url,html,body,header,ajaxregex) {\n      try { return await window.flutter_inappwebview.callHandler('webviewajax',url,html,body,header,ajaxregex); } catch (error) { return \"\"; }\n    }\n\n    async startBrowser(url,title,header) {\n      try { return await window.flutter_inappwebview.callHandler('startBrowser',url,title,header); } catch (error) { return \"\"; }\n    }\n    \n    async startBrowserWithShouldOverrideUrlLoading(url,title,header) {\n      try { return await window.flutter_inappwebview.callHandler('startBrowserWithShouldOverrideUrlLoading',url,title,header); } catch (error) { return \"\"; }\n    }\n\n    async startBrowserDp(url,title) {\n      try { return await window.flutter_inappwebview.callHandler('startBrowserDp',url,title); } catch (error) { return \"\"; }\n    }\n\n    async back() {\n      try { return await window.flutter_inappwebview.callHandler('back'); } catch (error) { return false; }\n    }\n\n    async utf8ToGbkUrlEncoded(str) {\n      try { return await window.flutter_inappwebview.callHandler('utf8ToGbkUrlEncoded',str); } catch (error) { return \"\"; }\n    }\n\n    async getVerificationCode(str,header) {\n      try { return await window.flutter_inappwebview.callHandler('getVerificationCode',str,header); } catch (error) { return \"\"; }\n    }\n    \n    async addbook(bookUrl) {\n      try { return await window.flutter_inappwebview.callHandler('addbook',bookUrl); } catch (error) { return \"\"; }\n    }\n    \n    async getdurChapterIndex(bookUrl) {\n      try { return await window.flutter_inappwebview.callHandler('getdurChapterIndex',bookUrl); } catch (error) { return 0; }\n    }\n    \n    async base64encode(str) {\n      try { return await window.flutter_inappwebview.callHandler('base64encode',str); } catch (error) { return \"\"; }\n    }\n    \n    async base64decode(str) {\n      try { return await window.flutter_inappwebview.callHandler('base64decode',str); } catch (error) { return \"\"; }\n    }\n  }\n\n  class Http {\n    constructor() {\n      this.open = false;\n      this.requestTimestamps = [];\n      this.rateLimit = 5;\n      this.rateLimitWindow = 1000;\n    }\n\n    async checkRateLimit() {\n      if(!this.open) return;\n      const now = Date.now();\n      this.requestTimestamps = this.requestTimestamps.filter(timestamp => now - timestamp < this.rateLimitWindow);\n      if (this.requestTimestamps.length >= this.rateLimit) {\n        const oldestTimestamp = this.requestTimestamps[0];\n        const waitTime = this.rateLimitWindow - (now - oldestTimestamp);\n        await new Promise(resolve => setTimeout(resolve, waitTime));\n        return this.checkRateLimit();\n      }\n      this.requestTimestamps.push(now);\n    }\n\n    async Get(url,headers,followRedirects) {\n      try { await this.checkRateLimit(); return await window.flutter_inappwebview.callHandler('http',\"get\",url,\"\",JSON.stringify(headers),followRedirects,\"\"); } catch (error) { return null; }\n    }\n\n    async Head(url,headers,followRedirects) {\n      try { await this.checkRateLimit(); return await window.flutter_inappwebview.callHandler('http',\"head\",url,\"\",JSON.stringify(headers),followRedirects,\"\"); } catch (error) { return null; }\n    }\n    \n    async Post(url,headers,body,contenttype,followRedirects) {\n      try { await this.checkRateLimit(); return await window.flutter_inappwebview.callHandler('http',\"post\",url,body,JSON.stringify(headers),followRedirects,contenttype); } catch (error) { return null; }\n    }\n  }\n\n  class Cache {\n    constructor() {}\n    async get(key) { try { return await window.flutter_inappwebview.callHandler('cache.get',key); } catch (error) { return null; } }\n    async set(key,value) { try { return await window.flutter_inappwebview.callHandler('cache.set',key,value); } catch (error) { return null; } }\n    async remove(key) { try { return await window.flutter_inappwebview.callHandler('cache.remove',key); } catch (error) { return null; } }\n    async getLoginInfo(){ return await this.get(\"LoginInfo\") }\n    async putLoginInfo(info){ return await this.set(\"LoginInfo\",info) }\n    async getbookVariable(bookurl){ return await this.get(bookurl) }\n    async setbookVariable(bookurl,value){ return await this.set(bookurl,value) }\n  }\n\n  class Cookie {\n    constructor() {}\n    async get(url) { try { return await window.flutter_inappwebview.callHandler('cookie.get',url); } catch (error) { return null; } }\n    async remove(url) { try { return await window.flutter_inappwebview.callHandler('cookie.remove',url); } catch (error) { return null; } }\n    async set(url,value) { try { return await window.flutter_inappwebview.callHandler('cookie.set',url,value); } catch (error) { return null; } }\n    async setCookie(url,key,value) { try { return await window.flutter_inappwebview.callHandler('cookie.setcookie',url,key,value); } catch (error) { return null; } }\n    async getCookie(url,value) { try { return await window.flutter_inappwebview.callHandler('cookie.getCookie',url,value); } catch (error) { return null; } }\n  }\n\n  function parseHTMLSafely(htmlStr) {\n    try { var tempDiv = document.createElement('div'); tempDiv.innerHTML = htmlStr; return $(tempDiv); } catch (e) { flutterBridge.log(\"HTML解析错误:\"+e.message); return $('<div>'); }\n  }\n\n  function removeHTMLSafely(tempContainer) {\n    try { tempContainer.innerHTML = ''; if (tempContainer.parentNode) { tempContainer.parentNode.removeChild(tempContainer); } } catch (e) { flutterBridge.log(\"HTML移除失败:\"+e.message); }\n  }\n\n  function removeHTMLTags(htmlString) {\n    let result = htmlString.replace(\/<script\\b[^<]*(?:(?!<\\\/script>)<[^<]*)*<\\\/script>\/gi, '');\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=\"http:\/\/www.shenpinwu.co\"\n    var header={\n        \"User-Agent\": \"Mozilla\/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit\/537.36 (KHTML, like Gecko) Chrome\/120.0.0.0 Safari\/537.36\",\n        \"Referer\": baseurl,\n        \"Accept\": \"text\/html,application\/xhtml+xml,application\/xml;q=0.9,*\/*;q=0.8\"\n    };\n\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     * search(key, page) - 搜索功能\n     * 神书网搜索URL: \/soso67fb4.html?searchkey={URL编码关键词}\n     *\/\n    async function search(key, page) {\n        if(page > 1){\n            return \"[]\";\n        }\n\n        \/\/ 获取首页HTML，从中提取搜索表单的action地址\n        var homeResult = await flutterBridge.webview(baseurl + \"\/\", \"\", \"\", \"\",header);\n        \n        if(!homeResult ){\n            flutterBridge.showToast(\"获取首页失败\");\n            return \"[]\";\n        }\n        \n        var homeHtml = homeResult;\n         flutterBridge.text(0, homeHtml);\n        \/\/ 从HTML中提取搜索表单的action属性\n        \/\/ 表单格式: <form ... action=\"\/soso67fb4.html?method=get\">\n        var formActionMatch = homeHtml.match(\/<form[^>]*action=[\"']([^\"']*soso[^\"']*)[\"'][^>]*>\/i);\n        \n        if(!formActionMatch){\n            flutterBridge.log(\"未能在首页找到搜索表单action\");\n            return \"[]\";\n        }\n        \n        var formAction = formActionMatch[1];\n        var encodedKey = encodeURIComponent(key);\n        \n        \/\/ 构建完整搜索URL（注意：action可能已包含参数，需要正确拼接）\n        var realSearchUrl = \"\";\n        if(formAction.indexOf('?') !== -1){\n            realSearchUrl = baseurl + formAction + \"&searchkey=\" + encodedKey;\n        } else {\n            realSearchUrl = baseurl + formAction + \"?searchkey=\" + encodedKey;\n        }\n\n        flutterBridge.log(\"真实搜索URL: \" + realSearchUrl);\n        await cookie.remove(baseurl);\n        \n        \/\/ 使用webview加载（因为可能有JS重定向）\n        var htmlContent =  await  flutterBridge.webview(realSearchUrl, \"\", \"\", \"\",header);\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(removeHTMLTags(htmlContent));\n\n        \/\/ 神书网书籍链接格式: \/{分类ID}_{书籍ID}\/\n        \/\/ 示例: \/222_222264\/ 或 \/201_201175\/\n        var bookLinkMatcher = function(href) {\n            if (!href) return false;\n            return \/^\\\/\\d+_\\d+\\\/$\/.test(href);\n        };\n\n        \/\/ 基于真实HTML结构提取搜索结果\n        \/\/ 结构: div.item > div.image > a[href] (书籍链接)\n        \/\/       + dl > dt > a (书名) + dd (简介) + div.btm > a[author] + em(字数\/时间)\n        \n        \/\/ 方法1: 优先查找 .item 容器\n        $tempContainer.find(\".item\").each(function() {\n            try{\n                var $item = $(this);\n                \n                \/\/ 从 image 区域获取书籍URL\n                var bookUrl = $item.find(\".image a\").first().attr('href');\n                \n                if (!bookUrl || !bookLinkMatcher(bookUrl) || processedUrls.has(bookUrl)) { \n                    return true; \n                }\n                \n                processedUrls.add(bookUrl);\n                \n                \/\/ 书名（优先从dt标签获取）\n                var bookName = $item.find(\"dt a\").first().text().trim();\n                if(!bookName) {\n                    bookName = $item.find(\".image a\").first().attr('title') || \"\";\n                }\n                \n                \/\/ 作者（\/author\/xxx\/ 格式的链接）\n                var author = \"\";\n                $item.find(\"a[href*='\/author\/']\").each(function(){\n                    if(!author) {\n                        author = $(this).text().trim();\n                    }\n                });\n                \n                \/\/ 简介（dd标签内容）\n                var intro = $item.find(\"dd\").first().text().trim();\n                if(intro && intro.length > 200) {\n                    intro = intro.substring(0, 200);  \/\/ 限制长度\n                }\n                \n                \/\/ 字数（em标签）\n                var wordCount = \"\";\n                $item.find(\"em.orange, em[class*='orange']\").each(function(){\n                    if(!wordCount) wordCount = $(this).text().trim();\n                });\n                \n                if(bookName && bookName.length > 1){\n                    bookUrl = normalizeUrl(bookUrl);\n\n                    var book={\n                        \"bookUrl\": bookUrl,\n                        \"name\": bookName,\n                        \"author\": author,\n                        \"kind\": \"\",\n                        \"coverUrl\": \"\",\n                        \"intro\": intro,\n                        \"tocUrl\": bookUrl,\n                        \"wordCount\": wordCount,\n                        \"type\": 0,\n                        \"latestChapterTitle\": \"\"\n                    };\n                    books.push(book);\n                }\n            }catch(e){\n                flutterBridge.log(\"解析.item出错: \" + e.message);\n            }\n        });\n\n        \/\/ 方法2: 如果没找到.item容器，使用通用匹配作为备选\n        if(books.length === 0) {\n            $tempContainer.find(\"a\").each(function() {\n                try{\n                    var bookUrl = $(this).attr('href');\n                    \n                    if (!bookUrl || processedUrls.has(bookUrl)) { return true; }\n                    if (!bookLinkMatcher(bookUrl)) { return true; }\n                    \n                    processedUrls.add(bookUrl);\n                    \n                    var bookName = $(this).attr('title') || $(this).text().trim();\n\n                    if(bookName && bookName.length > 1){\n                        bookUrl = normalizeUrl(bookUrl);\n\n                        var book={\n                            \"bookUrl\": bookUrl,\n                            \"name\": bookName,\n                            \"author\": \"\",\n                            \"kind\": \"\",\n                            \"coverUrl\": \"\",\n                            \"intro\": \"\",\n                            \"tocUrl\": bookUrl,\n                            \"wordCount\": \"\",\n                            \"type\": 0,\n                            \"latestChapterTitle\": \"\"\n                        };\n                        books.push(book);\n                    }\n                }catch(e){\n                    flutterBridge.log(\"解析书籍信息出错: \" + e.message);\n                }\n            });\n        }\n\n        removeHTMLSafely($tempContainer);\n        flutterBridge.log(\"找到 \" + books.length + \" 本书\");\n        return JSON.stringify(books);\n    }\n\n    \/*\n     * info(bookurl) - 书籍详情\n     * URL格式: http:\/\/www.shenpinwu.co\/{cat}_{book_id}\/\n     *\/\n    async function info(bookurl) {\n        var mheader={\n            ...header,\n            \"Referer\": baseurl\n        };\n\n        var htmlResult = await http.Get(bookurl, mheader, true);\n\n        if(!htmlResult || !htmlResult.data){\n            flutterBridge.showToast(\"获取信息失败\");\n            return JSON.stringify({});\n        }\n\n        var htmlContent = htmlResult.data;\n\n        flutterBridge.text(1, htmlContent);\n\n        var $tempContainer = parseHTMLSafely(removeHTMLTags(htmlContent));\n\n        \/\/ 从h1提取书名\n        var h1Text = $tempContainer.find(\"h1\").first().text().trim();\n        \n        var name = h1Text || \"\";\n        var author = \"\";\n        var kind = \"\";\n        var intro = \"\";\n        var latestChapterTitle = \"\";\n        var coverUrl = \"\";\n\n        \/\/ 提取封面图\n        coverUrl = $tempContainer.find(\"img\").first().attr('src') || \"\";\n\n        \/\/ 遍历页面元素提取详细信息\n        $tempContainer.find(\"*\").each(function() {\n            var text = $(this).text().trim();\n\n            \/\/ 作者\n            if(!author && text.length < 30) {\n                var authorMatch = text.match(\/作者?[：:\\s]*(.+?)(?:\\s|$)\/);\n                if(authorMatch && authorMatch[1].length > 0 && authorMatch[1].length < 15) { \n                    author = authorMatch[1].trim(); \n                }\n            }\n\n            \/\/ 分类\n            if(!kind && text.length < 15) {\n                var categories = ['玄幻', '武侠', '都市', '历史', '科幻', '游戏', '女生', '其他'];\n                for(var i=0; i<categories.length; i++) {\n                    if(text.indexOf(categories[i]) !== -1) {\n                        kind = categories[i];\n                        break;\n                    }\n                }\n            }\n\n            \/\/ 最新章节\n            if(!latestChapterTitle && text.length < 60 && name) {\n                var chapterMatch = text.match(\/(第\\s*\\d+\\s*[章节篇])\/);\n                if(chapterMatch && text.indexOf(name) === -1) { \n                    latestChapterTitle = chapterMatch[1]; \n                }\n            }\n        });\n\n        \/\/ 提取简介（查找较长的文本块）\n        $tempContainer.find(\"p, div\").each(function() {\n            if(intro) return false;\n            \n            var text = $(this).text().trim();\n            \n            if(text.length > 50 && text.length < 2000 && \n               text.indexOf(name) === -1 &&\n               text.indexOf('加入书架') === -1 &&\n               text.indexOf('马上阅读') === -1 &&\n               text.indexOf('推荐') === -1 &&\n               text.indexOf('催更') === -1) {\n                intro = text;\n            }\n        });\n\n        var book={\n            \"bookUrl\": bookurl,\n            \"name\": name,\n            \"author\": author,\n            \"kind\": kind,\n            \"coverUrl\": normalizeUrl(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     * chapter(tocUrl) - 章节列表\n     * 从书籍详情页提取所有章节链接\n     * 注意：页面有2个区域：最新章节(上) + 全部章节目录(下)\n     * 全部章节目录默认只显示部分章节，需点击\"展开更多\"才能看到全部\n     *\/\n    async function chapter(tocUrl) {\n        var mheader={\n            ...header,\n            \"Referer\": baseurl\n        };\n\n        \/\/ 第1步：先用http.Get获取页面\n        var htmlResult = await http.Get(tocUrl, mheader, true);\n\n        if(!htmlResult || !htmlResult.data){\n            flutterBridge.showToast(\"获取目录失败\");\n            return \"[]\";\n        }\n\n        var htmlContent = htmlResult.data;\n        \n        \/\/ 第2步：检查是否需要展开全部章节（如果章节数太少）\n        var $tempCheck = parseHTMLSafely(removeHTMLTags(htmlContent));\n        var initialChapterCount = 0;\n        var hasExpandBtn = false;\n        \n        \/\/ 统计可见的章节数量\n        $tempCheck.find(\"a\").each(function() {\n            var href = $(this).attr('href');\n            if (href && \/^\\\/\\d+_\\d+\\\/\\d+\\.html$\/.test(href)) {\n                initialChapterCount++;\n            }\n        });\n        \n        \/\/ 检查是否有\"展开更多\"按钮\n        if ($tempCheck.find(\"*:contains('展开'), *:contains('更多'), *:contains('查看全部')\").length > 0) {\n            hasExpandBtn = true;\n        }\n        \n        removeHTMLSafely($tempCheck);\n        \n        flutterBridge.log(\"初始章节数: \" + initialChapterCount + \", 有展开按钮: \" + hasExpandBtn);\n        \n        \/\/ 第3步：如果章节数少或有展开按钮，用webview加载并点击展开\n        if (initialChapterCount < 50 || hasExpandBtn) {\n            flutterBridge.log(\"使用webview展开全部章节...\");\n            \n            var expandJs = `\n                setTimeout(function(){\n                    \/\/ 点击\"展开更多\"、\"点击展开\"、\"查看全部\"等按钮\n                    var buttons = document.querySelectorAll('a, div, span, button');\n                    for(var i=0; i<buttons.length; i++){\n                        var text = (buttons[i].textContent || '').toLowerCase();\n                        if((text.indexOf('展开') !== -1 || \n                            text.indexOf('更多') !== -1 || \n                            text.indexOf('查看全部') !== -1) &&\n                           text.length < 20) {\n                            buttons[i].click();\n                            break;\n                        }\n                    }\n                }, 500);\n            `;\n            \n            var expandedContent = await flutterBridge.webview(\n                tocUrl,\n                \"\",\n                htmlContent + \"<script>\" + expandJs + \"<\"+\"\/script>\",\n                \"\",\n                JSON.stringify(mheader)\n            );\n            \n            if(expandedContent && expandedContent.length > htmlContent.length) {\n                htmlContent = expandedContent;\n                flutterBridge.log(\"展开成功，新内容长度: \" + htmlContent.length);\n            } else {\n                flutterBridge.log(\"展开失败或无变化，使用原始内容\");\n            }\n        }\n\n        flutterBridge.text(2, htmlContent);\n\n        var chapters = [];\n        var processedUrls = new Set();\n        var $tempContainer = parseHTMLSafely(removeHTMLTags(htmlContent));\n\n        \/\/ 神书网章节链接格式: \/{cat}_{book_id}\/{chapter_id}.html\n        var chapterLinkMatcher = function(href) {\n            if (!href) return false;\n            return \/^\\\/\\d+_\\d+\\\/\\d+\\.html$\/.test(href);\n        };\n\n        \/\/ 核心策略：找到\"全部章节目录\"的<dt>标题，只遍历其后的兄弟节点\n        var $catalogDt = null;\n\n        \/\/ 查找包含\"全部章节目录\"文本的<dt>元素\n        $tempContainer.find(\"dt\").each(function() {\n            var text = $(this).text().trim();\n\n            if ((text.indexOf('全部章节') !== -1 ||\n                 text.indexOf('正文目录') !== -1 ||\n                 text.indexOf('章节目录') !== -1) &&\n                text.length < 50) {\n\n                $catalogDt = $(this);\n                flutterBridge.log(\"找到'全部章节目录'标题: \" + text.substring(0, 50));\n                return false;  \/\/ 找到后停止\n            }\n        });\n\n        if ($catalogDt && $catalogDt.length > 0) {\n            \/\/ 从该<dt>之后开始遍历兄弟节点\n            var $nextSibling = $catalogDt.next();\n\n            while ($nextSibling.length > 0) {\n                \/\/ 检查是否是<a>标签（章节链接）\n                if ($nextSibling.is(\"a\")) {\n                    try{\n                        var chapterUrl = $nextSibling.attr('href');\n                        var chapterName = $nextSibling.find(\"dd\").first().text().trim();\n\n                        if(!chapterName){\n                            chapterName = $nextSibling.text().trim();\n                        }\n\n                        if(!chapterUrl || !chapterName || chapterName.length < 2){\n                            $nextSibling = $nextSibling.next();\n                            continue;\n                        }\n\n                        if(processedUrls.has(chapterUrl)){\n                            $nextSibling = $nextSibling.next();\n                            continue;\n                        }\n\n                        if(!chapterLinkMatcher(chapterUrl)){\n                            $nextSibling = $nextSibling.next();\n                            continue;\n                        }\n\n                        processedUrls.add(chapterUrl);\n                        chapterUrl = normalizeUrl(chapterUrl);\n\n                        chapters.push({\n                            \"name\": chapterName,\n                            \"chapterId\": chapterUrl,\n                            \"index\": chapters.length,\n                            \"isPay\": false,\n                            \"isVip\": false,\n                            \"isVolume\": false,\n                            \"tag\": \"\"\n                        });\n                    }catch(e){\n                        flutterBridge.log(\"解析章节出错: \" + e.message);\n                    }\n                }\n                \/\/ 如果遇到另一个<dt>，检查是否是结束标记\n                else if ($nextSibling.is(\"dt\")) {\n                    var dtText = $nextSibling.text().trim();\n                    \n                    \/\/ 如果是\"点击展开更多\"等按钮，跳过继续\n                    if (dtText.indexOf('展开') !== -1 ||\n                        dtText.indexOf('更多') !== -1 ||\n                        dtText.indexOf('点击') !== -1) {\n                        flutterBridge.log(\"跳过展开按钮: \" + dtText.substring(0, 30));\n                        $nextSibling = $nextSibling.next();\n                        continue;\n                    }\n                    \n                    \/\/ 遇到其他<dt>标题，可能到了另一个区域，停止\n                    flutterBridge.log(\"遇到其他<dt>标题，停止遍历: \" + dtText.substring(0, 30));\n                    break;\n                }\n\n                $nextSibling = $nextSibling.next();\n            }\n\n            flutterBridge.log(\"从'全部章节目录'提取了 \" + chapters.length + \" 个章节\");\n        } else {\n            \/\/ 兜底方案：提取所有章节并用URL去重（按URL数字排序确保正序）\n            flutterBridge.log(\"未找到'全部章节目录'标题，使用去重+排序模式\");\n\n            var allChapters = [];\n\n            $tempContainer.find(\"a\").each(function(index) {\n                try{\n                    var chapterUrl = $(this).attr('href');\n                    var chapterName = $(this).text().trim();\n\n                    if(!chapterUrl || !chapterName || chapterName.length < 2){\n                        return true;\n                    }\n\n                    if(processedUrls.has(chapterUrl)){\n                        return true;\n                    }\n\n                    if(!chapterLinkMatcher(chapterUrl)){\n                        return true;\n                    }\n\n                    processedUrls.add(chapterUrl);\n                    chapterUrl = normalizeUrl(chapterUrl);\n\n                    \/\/ 提取章节号用于排序\n                    var chapterNumMatch = chapterUrl.match(\/\\\/(\\d+)\\.html$\/);\n                    var chapterNum = chapterNumMatch ? parseInt(chapterNumMatch[1]) : 999999;\n\n                    allChapters.push({\n                        name: chapterName,\n                        chapterId: chapterUrl,\n                        num: chapterNum\n                    });\n                }catch(e){\n                    flutterBridge.log(\"解析章节出错: \" + e.message);\n                }\n            });\n\n            \/\/ 按章节号正序排列（第1章在最前）\n            allChapters.sort(function(a, b) { return a.num - b.num; });\n\n            allChapters.forEach(function(ch, idx) {\n                chapters.push({\n                    \"name\": ch.name,\n                    \"chapterId\": ch.chapterId,\n                    \"index\": idx,\n                    \"isPay\": false,\n                    \"isVip\": false,\n                    \"isVolume\": false,\n                    \"tag\": \"\"\n                });\n            });\n\n            flutterBridge.log(\"去重+排序模式找到 \" + chapters.length + \" 个章节\");\n        }\n\n        removeHTMLSafely($tempContainer);\n        flutterBridge.log(\"总共返回 \" + chapters.length + \" 个章节\");\n        return JSON.stringify(chapters);\n    }\n\n    \/*\n     * content(url) - 章节内容\n     * URL格式: \/{cat}_{book_id}\/{chapter_id}.html\n     * 注意：正文可能分多页，需要自动获取所有页面并合并\n     *\/\n    async function content(url) {\n        var mheader={\n            ...header,\n            \"Referer\": baseurl\n        };\n\n        var htmlResult = await http.Get(url, mheader, true);\n\n        if(!htmlResult || !htmlResult.data){\n            flutterBridge.showToast(\"获取内容失败\");\n            return \"\";\n        }\n\n        var htmlContent = htmlResult.data;\n\n        flutterBridge.text(3, htmlContent);\n\n        var allContent = [];  \/\/ 存储所有页面内容\n        var currentUrl = url; \/\/ 当前页面URL\n        var maxPages = 20;    \/\/ 最大页数限制\n        var pageCount = 0;\n\n        while (currentUrl && pageCount < maxPages) {\n            pageCount++;\n\n            \/\/ 解析当前页面\n            var $tempContainer = parseHTMLSafely(removeHTMLTags(htmlContent));\n\n            \/\/ 从正文容器获取内容\n            var $contentElement = $tempContainer.find(\".content\").first();\n            if ($contentElement.length === 0) {\n                var $h1 = $tempContainer.find(\"h1\").first();\n                if ($h1.length > 0) {\n                    $contentElement = $h1.nextAll(\"p, div\");\n                }\n                \n                if ($contentElement.length === 0) {\n                    var maxLength = 0;\n                    $tempContainer.find(\"div, p\").each(function() {\n                        var text = $(this).text().trim();\n                        if(text.length > maxLength && text.length > 200 &&\n                           text.indexOf('首页') === -1 && \n                           text.indexOf('目录') === -1 &&\n                           text.indexOf('下一页') === -1) {\n                            maxLength = text.length;\n                            $contentElement = $(this);\n                        }\n                    });\n                }\n            }\n\n            if($contentElement.length > 0) {\n                var pageText = $contentElement.html();\n                if (pageText) {\n                    allContent.push(await flutterBridge.htmlToText(pageText));\n                    flutterBridge.log(\"获取第 \" + pageCount + \" 页，长度: \" + pageText.length);\n                }\n            }\n\n            \/\/ 查找下一页链接\n            var nextPageUrl = null;\n            \n            $tempContainer.find(\"a\").each(function() {\n                var text = $(this).text().trim();\n                var href = $(this).attr('href');\n                \n                if ((text === '下页' || text === '下一页' || text === '下一章(→)') && !nextPageUrl && href) {\n                    nextPageUrl = href;\n                }\n                \n                if (!nextPageUrl && href && href.indexOf('page=') !== -1) {\n                    nextPageUrl = href;\n                }\n            });\n\n            removeHTMLSafely($tempContainer);\n\n            if (!nextPageUrl) {\n                flutterBridge.log(\"没有更多页面，共 \" + pageCount + \" 页\");\n                break;\n            }\n\n            \/\/ 构建完整URL并获取下一页\n            if (nextPageUrl.startsWith('\/')) {\n                var pathParts = url.split('\/');\n                basePath = pathParts.slice(0, 3).join('\/');\n                currentUrl = basePath + nextPageUrl;\n            } else if (nextPageUrl.startsWith('http')) {\n                currentUrl = nextPageUrl;\n            } else {\n                var lastSlashIndex = url.lastIndexOf('\/');\n                currentUrl = url.substring(0, lastSlashIndex + 1) + nextPageUrl;\n            }\n\n            if (currentUrl === url && pageCount > 1) {\n                flutterBridge.log(\"检测到重复URL，停止翻页\");\n                break;\n            }\n\n            flutterBridge.log(\"正在获取第 \" + (pageCount + 1) + \" 页: \" + currentUrl);\n            \n            var nextResult = await http.Get(currentUrl, mheader, true);\n            if(!nextResult || !nextResult.data){\n                flutterBridge.log(\"获取第 \" + (pageCount + 1) + \" 页失败\");\n                break;\n            }\n            htmlContent = nextResult.data;\n        }\n\n        \/\/ 合并所有页面内容\n        var contenttxt = allContent.join(\"\\n\\n\");\n\n        \/\/ 清理内容\n        contenttxt = contenttxt.replace(\/\\n{3,}\/g, \"\\n\\n\");\n        contenttxt = contenttxt.trim();\n\n        \/\/ 限制最大长度\n        if(contenttxt.length > 50000) {\n            contenttxt = contenttxt.substring(0, 50000);\n        }\n\n        if(!contenttxt || contenttxt.length < 30){\n            flutterBridge.showToast(\"获取内容失败\");\n            return \"\";\n        }\n\n        return contenttxt;\n    }\n\n    \/*\n     * getfinds() - 发现\/分类\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        push(\"排行\", \"\/rank\/\", 0);\n        push(\"书库\", \"\/bookcase\/\", 0);\n        push(\"玄幻魔法\", \"\/xuanhuan\/\", 0);\n        push(\"武侠修真\", \"\/wuxia\/\", 0);\n        push(\"都市言情\", \"\/dushi\/\", 0);\n        push(\"历史军事\", \"\/lishi\/\", 0);\n        push(\"科幻灵异\", \"\/kehuan\/\", 0);\n        push(\"游戏竞技\", \"\/youxi\/\", 0);\n        push(\"女生频道\", \"\/nvsheng\/\", 0);\n        push(\"其他类型\", \"\/qita\/\", 0);\n        push(\"全本完结\", \"\/quanben\/\", 0);\n\n        return JSON.stringify(result);\n    }\n\n    \/*\n     * find(url, page) - 分类浏览\n     *\/\n    async function find(url, page) {\n        var u = url.replaceAll(\"{{page}}\", page);\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        var htmlResult = await http.Get(u, mheader, true);\n\n        if(!htmlResult || !htmlResult.data){\n            flutterBridge.showToast(\"获取分类失败\");\n            return \"[]\";\n        }\n\n        var htmlContent = htmlResult.data;\n\n        flutterBridge.text(0, htmlContent);\n\n        var books = [];\n        var processedUrls = new Set();\n        var $tempContainer = parseHTMLSafely(removeHTMLTags(htmlContent));\n\n        \/\/ 神书网书籍链接格式: \/{分类ID}_{书籍ID}\/\n        var bookLinkMatcher = function(href) {\n            if (!href) return false;\n            return \/^\\\/\\d+_\\d+\\\/$\/.test(href);\n        };\n\n        \/\/ 基于真实HTML结构提取书籍列表\n        \/\/ 结构: div.item > div.image > a[href] (封面\/链接)\n        \/\/       + dl > dt > a (书名) + dd (简介) + div.btm > a[author] + em(字数\/时间)\n        \n        $tempContainer.find(\".item\").each(function() {\n            try{\n                var $item = $(this);\n                \n                \/\/ 从 image 区域获取书籍URL和封面\n                var $imageLink = $item.find(\".image a\").first();\n                var bookUrl = $imageLink.attr('href');\n                var coverUrl = $imageLink.find(\"img\").first().attr('data-original') || \n                               $imageLink.find(\"img\").first().attr('src') || \"\";\n                \n                if (!bookUrl || !bookLinkMatcher(bookUrl) || processedUrls.has(bookUrl)) { \n                    return true; \n                }\n                \n                processedUrls.add(bookUrl);\n                \n                \/\/ 书名（从dt标签的a获取）\n                var bookName = $item.find(\"dt a\").first().attr('title') || \n                               $item.find(\"dt a\").first().text().trim();\n                \n                \/\/ 作者（从.btm中的a[包含\/author\/]获取）\n                var author = \"\";\n                $item.find(\"a[href*='\/author\/']\").each(function(){\n                    if(!author) {\n                        author = $(this).text().trim();\n                    }\n                });\n                \n                \/\/ 简介（dd标签内容）\n                var intro = $item.find(\"dd\").first().text().trim();\n                if(intro && intro.length > 200) {\n                    intro = intro.substring(0, 200);  \/\/ 限制长度\n                }\n                \n                \/\/ 字数和时间\n                var wordCount = \"\";\n                var updateTime = \"\";\n                $item.find(\"em.orange, em[class*='orange']\").each(function(){\n                    if(!wordCount) wordCount = $(this).text().trim();\n                });\n                $item.find(\"em.blue, em[class*='blue']\").each(function(){\n                    if(!updateTime) updateTime = $(this).text().trim();\n                });\n                \n                if(bookName && bookName.length > 1){\n                    bookUrl = normalizeUrl(bookUrl);\n\n                    var book={\n                        \"bookUrl\": bookUrl,\n                        \"name\": bookName,\n                        \"author\": author,\n                        \"kind\": \"\",\n                        \"coverUrl\": normalizeUrl(coverUrl),\n                        \"intro\": intro,\n                        \"tocUrl\": bookUrl,\n                        \"wordCount\": wordCount + (updateTime ? \" | \" + updateTime : \"\"),\n                        \"type\": 0,\n                        \"latestChapterTitle\": \"\"\n                    };\n                    books.push(book);\n                }\n            }catch(e){\n                flutterBridge.log(\"解析.item出错: \" + e.message);\n            }\n        });\n\n        \/\/ 如果没找到.item容器，使用通用匹配作为备选\n        if(books.length === 0) {\n            flutterBridge.log(\"未找到.item容器，使用通用匹配\");\n            \n            $tempContainer.find(\"a\").each(function() {\n                try{\n                    var bookUrl = $(this).attr('href');\n                    \n                    if (!bookUrl || processedUrls.has(bookUrl)) { return true; }\n                    if (!bookLinkMatcher(bookUrl)) { return true; }\n                    \n                    processedUrls.add(bookUrl);\n                    \n                    var bookName = $(this).attr('title') || $(this).text().trim();\n                    \n                    if(bookName && bookName.length > 1){\n                        bookUrl = normalizeUrl(bookUrl);\n\n                        var book={\n                            \"bookUrl\": bookUrl,\n                            \"name\": bookName,\n                            \"author\": \"\",\n                            \"kind\": \"\",\n                            \"coverUrl\": \"\",\n                            \"intro\": \"\",\n                            \"tocUrl\": bookUrl,\n                            \"wordCount\": \"\",\n                            \"type\": 0,\n                            \"latestChapterTitle\": \"\"\n                        };\n                        books.push(book);\n                    }\n                }catch(e){\n                    flutterBridge.log(\"解析书籍信息出错: \" + e.message);\n                }\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>","login":false,"lastUpdateTime":"1778347572129"}]