笔趣阁
https://www.biquge123.uk
autobcb_admin (12020)04/25 16:30
笔趣阁
{
"bookSourceUrl": "https:\/\/www.biquge123.uk",
"bookSourceName": "笔趣阁",
"enabledExplore": true,
"enabled": true,
"bookSourceGroup": "",
"author": "",
"help": false,
"html": "<!DOCTYPE html>\n<html lang=\"en\">\n<head>\n <meta charset=\"UTF-8\">\n <title>Biquge123<\/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.biquge123.uk\"\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 var searchUrl = baseurl + \"\/search?keyword=\" + encodeURIComponent(key);\n\n flutterBridge.log(\"搜索URL: \" + searchUrl);\n\n var htmlContent = await flutterBridge.webview(searchUrl, \"\", \"\", \"\", JSON.stringify(header));\n\n if(!htmlContent || isCloudflareChallenge(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(htmlContent);\n\n var linkMatcher = function(href) {\n if (!href) return false;\n return \/^\\\/\\d+\\\/?$\/.test(href);\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 var index = 0;\n $resultContainer.find(\"a\").each(function() {\n try{\n var $element = $(this);\n var bookUrl = $element.attr('href');\n \n if (!bookUrl || processedUrls.has(bookUrl)) {\n return true;\n }\n \n if (!linkMatcher(bookUrl)) {\n return true;\n }\n \n processedUrls.add(bookUrl);\n \n var bookName = $element.find(\".hot_name, .book-name, .name\").text().trim();\n if (!bookName) {\n bookName = $element.text().trim();\n }\n \n var author = \"\";\n var authorElement = $element.find(\".author, .book-author\");\n if (authorElement.length > 0) {\n author = authorElement.text().trim();\n } else {\n $element.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) {\n author = text;\n }\n }\n });\n }\n \n var coverUrl = $element.find(\"img\").attr('src') || \"\";\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\": \"\",\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, .hot_name\").first().text().trim();\n }\n\n \/\/ 作者提取:根据实际HTML结构\n var $infoCt = $tempContainer.find(\".info_ct\");\n if ($infoCt.length > 0) {\n $infoCt.find(\"div\").each(function() {\n var text = $(this).text();\n if (text.indexOf(\"作者:\") !== -1) {\n author = text.replace(\"作者:\", \"\").trim();\n }\n });\n }\n \n \/\/ 如果没找到,尝试其他方式\n if (!author) {\n $tempContainer.find(\"div, p, span\").each(function() {\n var text = $(this).text();\n if (text.indexOf(\"作者:\") !== -1) {\n author = text.replace(\"作者:\", \"\").trim();\n }\n });\n }\n\n \/\/ 分类\n $tempContainer.find(\"p, span, div\").each(function() {\n var text = $(this).text();\n if(text.indexOf(\"分类\") !== -1 || text.indexOf(\"类别\") !== -1){\n kind = text.replace(\/[^::]+[::]\\s*\/, \"\").trim();\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 \/\/ 简介提取:根据实际HTML结构\n var $des = $tempContainer.find(\".des\");\n if ($des.length > 0) {\n var desText = $des.text().trim();\n \/\/ 移除\"简介:\"前缀\n intro = desText.replace(\"简介:\", \"\").trim();\n }\n \n \/\/ 如果没找到,尝试其他选择器\n if (!intro) {\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 > 50) {\n intro = text;\n }\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 chapterListUrl = tocUrl;\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 return \"[]\";\n }\n\n flutterBridge.text(2, htmlContent);\n\n var chapters = [];\n var processedUrls = new Set();\n var $tempContainer = parseHTMLSafely(htmlContent);\n\n var $chapterContainer = null;\n \n \/\/ 先尝试找到完整的章节目录容器,避免包含最新章节\n var markers = [\"章节目录\", \"目录\"];\n var $markerContainer = findContainerByMarker($tempContainer, markers, 5);\n if ($markerContainer) {\n $chapterContainer = $markerContainer;\n }\n \n if (!$chapterContainer) {\n var linkMatcher = function(href) {\n if (!href) return false;\n return \/^\\\/\\d+\\\/\\d+$\/.test(href);\n };\n \n var minMatchCount = 10; \/\/ 增加最小匹配数量,确保找到完整的章节列表\n $chapterContainer = findBestContainer($tempContainer, linkMatcher, minMatchCount);\n }\n \n if (!$chapterContainer) {\n $chapterContainer = $tempContainer;\n flutterBridge.log(\"警告:未找到专门的章节列表容器,使用全页搜索\");\n } else {\n flutterBridge.log(\"找到章节列表容器\");\n }\n\n var chapterIndex = 0;\n var chapterLinks = [];\n \n \/\/ 先收集所有章节链接\n $chapterContainer.find(\"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;\n }\n\n if(processedUrls.has(chapterUrl)){\n return true;\n }\n\n var isChapterLink = \/^\\\/\\d+\\\/\\d+$\/.test(chapterUrl);\n \n if (!isChapterLink) {\n return true;\n }\n\n var isChapterName = chapterName.indexOf(\"章\") !== -1 || \n chapterName.indexOf(\"节\") !== -1 ||\n chapterName.indexOf(\"第\") !== -1 ||\n \/^\\d+\/.test(chapterName);\n \n if (!isChapterName) {\n return true;\n }\n\n processedUrls.add(chapterUrl);\n \n var dateIndex = chapterName.lastIndexOf(\"202\");\n if (dateIndex > 0) {\n chapterName = chapterName.substring(0, dateIndex).trim();\n }\n\n chapterUrl = normalizeUrl(chapterUrl);\n\n chapterLinks.push({\n \"name\": chapterName,\n \"chapterId\": chapterUrl,\n \"url\": chapterUrl\n });\n }catch(e){\n flutterBridge.log(\"解析章节出错: \" + e.message);\n }\n });\n \n \/\/ 过滤掉可能的最新章节(通常在页面顶部,数量较少)\n \/\/ 只保留完整的章节列表\n if (chapterLinks.length > 20) {\n \/\/ 如果链接数量很多,说明是完整的章节列表\n chapters = chapterLinks;\n } else {\n \/\/ 否则可能是最新章节,需要重新查找\n flutterBridge.log(\"章节数量过少,可能只获取到了最新章节\");\n chapters = chapterLinks;\n }\n \n \/\/ 按章节URL中的数字排序,确保顺序正确\n chapters.sort(function(a, b) {\n var numA = parseInt(a.url.match(\/\\\/\\d+\\\/(\\d+)\/)[1]);\n var numB = parseInt(b.url.match(\/\\\/\\d+\\\/(\\d+)\/)[1]);\n return numA - numB;\n });\n \n \/\/ 设置正确的索引\n for (var i = 0; i < chapters.length; i++) {\n chapters[i].index = i;\n chapters[i].isPay = false;\n chapters[i].isVip = false;\n chapters[i].isVolume = false;\n chapters[i].tag = \"\";\n }\n\n removeHTMLSafely($tempContainer);\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 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 var possibleContentSelectors = [\n \".article\", \"#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 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 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 $contentContainer.find(\"p\").each(function() {\n var pText = $(this).text().trim();\n if (pText) {\n var isAd = pText.indexOf(\"广告\") !== -1 || \n pText.indexOf(\"APP\") !== -1 ||\n pText.indexOf(\"最新网址\") !== -1 ||\n pText.indexOf(\"笔趣阁\") !== -1;\n if (!isAd) {\n contenttxt += pText + \"\\r\\n\\r\\n\";\n }\n }\n });\n \n if (!contenttxt) {\n contenttxt = $contentContainer.text().trim();\n contenttxt = contenttxt.replace(\/(?<=[。!?!?])\\s*\/g, \"\\r\\n\\r\\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 push(\"发现\", \"\", 0);\n push(\"玄幻奇幻\", \"\/xh\", 0);\n push(\"武侠修真\", \"\/xz\", 0);\n push(\"都市言情\", \"\/ds\", 0);\n push(\"穿越历史\", \"\/cy\", 0);\n push(\"网游竞技\", \"\/wy\", 0);\n push(\"科幻灵异\", \"\/kh\", 0);\n push(\"完结小说\", \"\/wj\", 0);\n push(\"排行榜\", \"\/ph\", 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 var u = url;\n if(page > 1) {\n u = url + \"?page=\" + page;\n }\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 || 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 var linkMatcher = function(href) {\n if (!href) return false;\n return \/^\\\/\\d+\\\/?$\/.test(href);\n };\n \n var minMatchCount = 3;\n var $resultContainer = findBestContainer($tempContainer, linkMatcher, minMatchCount);\n \n if (!$resultContainer) {\n $resultContainer = $tempContainer;\n flutterBridge.log(\"警告:未找到专门的分类结果容器,使用全页搜索\");\n }\n\n var index = 0;\n $resultContainer.find(\"a\").each(function() {\n try{\n var $element = $(this);\n var bookUrl = $element.attr('href');\n \n if (!bookUrl || processedUrls.has(bookUrl)) {\n return true;\n }\n \n if (!linkMatcher(bookUrl)) {\n return true;\n }\n \n processedUrls.add(bookUrl);\n \n var bookName = $element.find(\".hot_name, .book-name, .name\").text().trim();\n if (!bookName) {\n bookName = $element.text().trim();\n }\n \n var author = \"\";\n var authorElement = $element.find(\".author, .book-author\");\n if (authorElement.length > 0) {\n author = authorElement.text().trim();\n } else {\n $element.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) {\n author = text;\n }\n }\n });\n }\n \n var coverUrl = $element.find(\"img\").attr('src') || \"\";\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\": \"\",\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": "1777105852725"
}