天天书屋

https://www.tiku365.net

autobcb_admin (12020)05/08 05:19

该用户很懒,什么介绍也没有写!
二维码导入(APP尚未完成该功能)
{
    "bookSourceUrl": "https:\/\/www.tiku365.net",
    "bookSourceName": "天天书屋",
    "enabledExplore": true,
    "enabled": true,
    "bookSourceGroup": "",
    "author": "trae",
    "help": true,
    "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=\"https:\/\/www.tiku365.net\"\n    var header={\n        \"User-Agent\": \"Mozilla\/5.0 (Macintosh; Intel Mac OS X 10_15_7) 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    \/\/ 智能容器定位工具函数\n    function findBestContainer($htmlContainer, linkMatcher, minMatchCount) {\n        var $bestContainer = null;\n        var maxMatchCount = 0;\n        var checkedParents = new Set();\n        \n        $htmlContainer.find(\"a\").each(function() {\n            var href = $(this).attr('href');\n            if (!linkMatcher(href)) { return true; }\n            \n            var $parent = $(this).parent();\n            var depth = 0;\n            while ($parent.length > 0 && depth < 10) {\n                var parentKey = $parent[0].outerHTML.substring(0, Math.min(500, $parent[0].outerHTML.length));\n                if (!checkedParents.has(parentKey)) {\n                    checkedParents.add(parentKey);\n                    var count = $parent.find(\"a\").filter(function() { return linkMatcher($(this).attr('href')); }).length;\n                    if (count > maxMatchCount && count >= minMatchCount) { maxMatchCount = count; $bestContainer = $parent; }\n                }\n                $parent = $parent.parent();\n                depth++;\n            }\n        });\n        return $bestContainer;\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    function hasCaptchaChallenge(html) {\n        return html && (\n            html.indexOf('验证码') !== -1 ||\n            html.indexOf('请输入验证码') !== -1 ||\n            html.indexOf('yzm') !== -1 ||\n            html.indexOf('rdyzm') !== -1\n        );\n    }\n\n    \/*\n     * search(key, page) - 搜索功能\n     * 天天书屋的搜索需要验证码,使用浏览器模式\n     *\/\n    async function search(key, page) {\n        if(page > 1){\n            return \"[]\";\n        }\n\n        \/\/ 天天书屋搜索URL: GET \/sscc\/?q=关键词\n        var searchUrl = baseurl + \"\/sscc\/?q=\" + encodeURIComponent(key);\n\n        flutterBridge.log(\"搜索URL: \" + searchUrl);\n\n        \/\/ 使用webview打开搜索页面(可能需要用户手动输入验证码)\n        var htmlContent = await flutterBridge.webview(searchUrl, \"\", \"\", \"\", JSON.stringify(header));\n\n        \/\/ 检测是否有验证码\n        if(!htmlContent || hasCaptchaChallenge(htmlContent)){\n            flutterBridge.showToast(\"搜索需要验证码,请在浏览器中完成\");\n            var s = await flutterBridge.startBrowser(searchUrl, \"搜索验证\", 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(removeHTMLTags(htmlContent));\n\n        \/\/ 智能定位搜索结果容器\n        var linkMatcher = function(href) {\n            if (!href) return false;\n            \/\/ 天天书屋书籍链接格式: \/tiku\/数字\/\n            return \/\\\/tiku\\\/\\d+\/.test(href);\n        };\n        \n        var $resultContainer = findBestContainer($tempContainer, linkMatcher, 1);\n\n        if (!$resultContainer) {\n            $resultContainer = $tempContainer;\n        }\n\n        \/\/ 提取书籍信息\n        $resultContainer.find(\"a\").each(function() {\n            try{\n                var bookUrl = $(this).attr('href');\n                \n                if (!bookUrl || processedUrls.has(bookUrl)) { return true; }\n                if (!linkMatcher(bookUrl)) { return true; }\n                \n                processedUrls.add(bookUrl);\n                \n                var fullText = $(this).text().trim();\n                var bookName = fullText;\n                var author = \"\";\n                \n                \/\/ 提取作者(如果有)\n                var authorMatch = fullText.match(\/作者[::\\s]*(.+)$\/);\n                if(authorMatch) {\n                    author = authorMatch[1].trim();\n                    bookName = fullText.substring(0, fullText.indexOf('作者')).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\": \"\",\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        removeHTMLSafely($tempContainer);\n        flutterBridge.log(\"找到 \" + books.length + \" 本书\");\n        return JSON.stringify(books);\n    }\n\n    \/*\n     * info(bookurl) - 书籍详情\n     *\/\n    async function info(bookurl) {\n        var mheader={\n            ...header,\n            \"Referer\": baseurl\n        };\n\n        var htmlContent = await flutterBridge.webview(bookurl, \"\", \"\", \"\", JSON.stringify(mheader));\n\n        if(!htmlContent){\n            flutterBridge.showToast(\"获取信息失败\");\n            return JSON.stringify({});\n        }\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 = \"\";\n        var author = \"\";\n        var kind = \"\";\n        var intro = \"\";\n        var latestChapterTitle = \"\";\n\n        if(h1Text) {\n            name = h1Text;\n        }\n\n        \/\/ 提取作者、分类、状态等信息\n        $tempContainer.find(\"*\").each(function() {\n            var text = $(this).text().trim();\n            \n            \/\/ 作者\n            if(text.indexOf(\"作者\") !== -1 && !author && text.length < 50) {\n                var match = text.match(\/作者[::\\s]*(.+?)(?:\\s|$)\/);\n                if(match) { author = match[1].trim(); }\n            }\n            \n            \/\/ 分类和状态\n            if(text.indexOf(\"类别\") !== -1 && text.indexOf(\"状态\") !== -1 && !kind && text.length < 100) {\n                var kindMatch = text.match(\/类别[::\\s]*([^\\s状态]+)\\s*状态[::\\s]*(.+)\/);\n                if(kindMatch) { kind = kindMatch[1].trim(); }\n            }\n            \n            \/\/ 最新章节\n            if(text.indexOf(\"最新章节\") !== -1 && !latestChapterTitle && text.length < 100) {\n                var chapterMatch = text.match(\/最新章节[::\\s]*(.+)\/);\n                if(chapterMatch) { latestChapterTitle = chapterMatch[1].trim(); }\n            }\n        });\n\n        \/\/ 提取简介\n        $tempContainer.find(\"*\").each(function() {\n            if(intro) return false;\n            \n            var text = $(this).text().trim();\n            \n            \/\/ 查找包含\"简介\"的元素\n            if((text.indexOf(\"简介\") !== -1 || text.indexOf(\"介绍\") !== -1) && text.length > 20 && text.length < 2000) {\n                var idx = text.indexOf(\"简介\");\n                if(idx === -1) idx = text.indexOf(\"介绍\");\n                \n                if(idx !== -1) {\n                    var potentialIntro = text.substring(idx + 2).replace(\/[::]\\s*\/, \"\").trim();\n                    if(potentialIntro.length > 10 && potentialIntro.length < 1500) {\n                        intro = potentialIntro;\n                    }\n                }\n            }\n        });\n\n        var coverUrl = $tempContainer.find(\"img[src*='tikuimg']\").first().attr('src') || \"\";\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     * 页面结构:<dl>内有2个<dt>标题:\"最新章节\"+\"正文目录\",只需提取\"正文目录\"后的<dd>\n     *\/\n    async function chapter(tocUrl) {\n        var mheader={\n            ...header,\n            \"Referer\": baseurl\n        };\n\n        var htmlContent = await flutterBridge.webview(tocUrl, \"\", \"\", \"\", JSON.stringify(mheader));\n\n        if(!htmlContent){\n            flutterBridge.showToast(\"获取目录失败\");\n            return \"[]\";\n        }\n\n        flutterBridge.text(2, htmlContent);\n\n        var chapters = [];\n        var $tempContainer = parseHTMLSafely(removeHTMLTags(htmlContent));\n\n        \/\/ 天天书屋章节链接格式: \/ikus\/{书籍ID}\/{章节ID}.html\n        var chapterLinkMatcher = function(href) {\n            if (!href) return false;\n            return \/\\\/ikus\\\/\\d+\\\/\\d+\\.html\/.test(href);\n        };\n\n        \/\/ 核心策略:找到\"正文目录\"的<dt>标题,只提取其后的所有<dd>\n        var $catalogDt = null;\n        $tempContainer.find(\"dt\").each(function() {\n            var text = $(this).text();\n            if (text.indexOf(\"正文目录\") !== -1) {\n                $catalogDt = $(this);\n                return false; \/\/ 找到后停止遍历\n            }\n        });\n\n        if ($catalogDt) {\n            flutterBridge.log(\"找到'正文目录'标题\");\n\n            \/\/ 遍历该<dt>之后的所有兄弟元素(<dd>)\n            \/\/ 注意:这里的DD元素100%都是章节链接,不需要再用正则过滤!\n            var $nextSibling = $catalogDt.next();\n            while ($nextSibling.length > 0) {\n                \/\/ 检查是否是<dd>标签\n                if (!$nextSibling.is(\"dd\")) {\n                    break; \/\/ 遇到非<dd>元素就停止\n                }\n\n                try{\n                    var $link = $nextSibling.find(\"a\").first();\n                    if ($link.length === 0) {\n                        $nextSibling = $nextSibling.next();\n                        continue;\n                    }\n\n                    var chapterUrl = $link.attr('href');\n                    var chapterName = $link.text().trim();\n\n                    \/\/ 基本验证:URL和名称不能为空\n                    if(!chapterUrl || !chapterName || chapterName.length < 2){\n                        $nextSibling = $nextSibling.next();\n                        continue;\n                    }\n\n                    \/\/ 标准化URL\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                $nextSibling = $nextSibling.next();\n            }\n\n            flutterBridge.log(\"从'正文目录'提取了 \" + chapters.length + \" 个章节\");\n        } else {\n            \/\/ 兜底方案:如果没找到\"正文目录\"标题,使用URL去重模式\n            flutterBridge.log(\"未找到'正文目录'标题,使用去重模式\");\n\n            var processedUrls = new Set();\n\n            $tempContainer.find(\"a\").each(function(index) {\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;\n                    }\n\n                    if(!chapterLinkMatcher(chapterUrl)){\n                        return true;\n                    }\n\n                    var normalizedUrl = normalizeUrl(chapterUrl);\n\n                    if(processedUrls.has(normalizedUrl)){\n                        return true;\n                    }\n\n                    processedUrls.add(normalizedUrl);\n\n                    chapters.push({\n                        \"name\": chapterName,\n                        \"chapterId\": normalizedUrl,\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\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     * 天天书屋章节页面可能有验证码保护,且正文可能分多页\n     *\/\n    async function content(url) {\n        var mheader={\n            ...header,\n            \"Referer\": baseurl\n        };\n\n        \/\/ 获取第一页内容\n        var htmlContent = await flutterBridge.webview(url, \"\", \"\", \"\", JSON.stringify(mheader));\n\n        \/\/ 检查是否有验证码\n        if(!htmlContent || hasCaptchaChallenge(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 allContent = [];  \/\/ 存储所有页面内容\n        var currentUrl = url; \/\/ 当前页面URL\n        var maxPages = 50;    \/\/ 最大页数限制,防止死循环\n        var pageCount = 0;\n\n        while (currentUrl && pageCount < maxPages) {\n            pageCount++;\n\n            \/\/ 解析当前页面\n            var $tempContainer = parseHTMLSafely(removeHTMLTags(htmlContent));\n\n            \/\/ 从 #content 获取正文\n            var $contentElement = $tempContainer.find(\"#content\").first();\n            if($contentElement.length > 0) {\n                var pageText =await  flutterBridge.htmlToText($contentElement.html());\n                allContent.push(pageText);\n                flutterBridge.log(\"获取第 \" + pageCount + \" 页,长度: \" + pageText.length);\n            }\n\n            \/\/ 查找下一页链接(常见选择器)\n            var nextPageUrl = null;\n\n            \/\/ 方法1: 查找包含\"下一页\"文本的链接\n            $tempContainer.find(\"a\").each(function() {\n                var text = $(this).text().trim();\n                if ((text === \"下一页\" || text === \"下页\" || text === \"下一章 ›\") && !nextPageUrl) {\n                    nextPageUrl = $(this).attr('href');\n                }\n            });\n\n            \/\/ 方法2: 如果没找到,查找分页导航中的下一个数字链接\n            if (!nextPageUrl) {\n                \/\/ 查找当前激活的页码,然后找下一个\n                var $currentPage = $tempContainer.find(\".active, .current, .selected, [class*='page'][class*='active']\").first();\n                if ($currentPage.length > 0) {\n                    var $nextPageLink = $currentPage.next(\"a\");\n                    if ($nextPageLink.length > 0) {\n                        nextPageUrl = $nextPageLink.attr('href');\n                    }\n                }\n            }\n\n            removeHTMLSafely($tempContainer);\n\n            \/\/ 如果没有下一页链接,结束循环\n            if (!nextPageUrl) {\n                flutterBridge.log(\"没有更多页面,共 \" + pageCount + \" 页\");\n                break;\n            }\n\n            \/\/ 智能构建完整URL(处理相对路径)\n            if (nextPageUrl.startsWith('http')) {\n                currentUrl = nextPageUrl;  \/\/ 已经是完整URL\n            } else if (nextPageUrl.startsWith('\/')) {\n                \/\/ 相对于域名的路径,需要保留书籍路径\n                \/\/ 从currentUrl提取基础路径(如 \/ikus\/32515635305\/)\n                var pathParts = currentUrl.split('\/');\n                \/\/ 保留协议+域名+前3级路径(如 https:\/\/www.tiku365.net\/tikus\/32515635305)\n                var basePath = pathParts.slice(0, 5).join('\/');\n                currentUrl = basePath + nextPageUrl;\n            } else {\n                \/\/ 相对路径,基于当前页面所在目录\n                var lastSlashIndex = currentUrl.lastIndexOf('\/');\n                currentUrl = currentUrl.substring(0, lastSlashIndex + 1) + nextPageUrl;\n            }\n\n            \/\/ 避免重复访问同一URL(防止循环)\n            if (currentUrl === url && pageCount > 1) {\n                flutterBridge.log(\"检测到重复URL,停止翻页\");\n                break;\n            }\n\n            flutterBridge.log(\"正在获取第 \" + (pageCount + 1) + \" 页: \" + currentUrl);\n            htmlContent = await flutterBridge.webview(currentUrl, \"\", \"\", \"\", JSON.stringify(mheader));\n\n            if (!htmlContent) {\n                flutterBridge.log(\"获取第 \" + (pageCount + 1) + \" 页失败\");\n                break;\n            }\n        }\n\n        \/\/ 合并所有页面内容\n        var contenttxt = allContent.join(\"\\n\\n\");\n\n        \/\/ 清理内容\n        contenttxt = contenttxt.replace(\/天天书屋.*?www\\.tiku365\\.net[\\s\\S]*\/g, \"\");\n        contenttxt = contenttxt.replace(\/看啥都有.*$\/g, \"\");\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     *\/\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(\"书库\", \"\/all\/index.html\", 0);\n        push(\"排行榜\", \"\/rank\/\", 0);\n        push(\"完本小说\", \"\/finish\/\", 0);\n        push(\"都市重生\", \"\/tiku321\/\", 0);\n        push(\"玄幻奇幻\", \"\/tiku322\/\", 0);\n        push(\"仙侠武侠\", \"\/tiku323\/\", 0);\n        push(\"历史军事\", \"\/tiku324\/\", 0);\n        push(\"科幻未来\", \"\/tiku325\/\", 0);\n        push(\"游戏竞技\", \"\/tiku326\/\", 0);\n        push(\"灵异悬疑\", \"\/tiku327\/\", 0);\n        push(\"二次元\", \"\/tiku328\/\", 0);\n        push(\"女生言情\", \"\/tiku329\/\", 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 htmlContent = await flutterBridge.webview(u, \"\", \"\", \"\", JSON.stringify(mheader));\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        \/\/ 智能定位书籍列表容器\n        var linkMatcher = function(href) {\n            if (!href) return false;\n            return \/\\\/tiku\\\/\\d+\/.test(href);\n        };\n        \n        var $resultContainer = findBestContainer($tempContainer, linkMatcher, 1);\n\n        if (!$resultContainer) {\n            $resultContainer = $tempContainer;\n        }\n\n        \/\/ 提取书籍信息\n        $resultContainer.find(\"a\").each(function() {\n            try{\n                var bookUrl = $(this).attr('href');\n                \n                if (!bookUrl || processedUrls.has(bookUrl)) { return true; }\n                if (!linkMatcher(bookUrl)) { return true; }\n                \n                processedUrls.add(bookUrl);\n                \n                var fullText = $(this).text().trim();\n                var bookName = fullText;\n                var author = \"\";\n                \n                \/\/ 提取作者(如果有)\n                var authorMatch = fullText.match(\/作者[::\\s]*(.+)$\/);\n                if(authorMatch) {\n                    author = authorMatch[1].trim();\n                    bookName = fullText.substring(0, fullText.indexOf('作者')).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\": \"\",\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        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": "1778188793572"
}
广告