秋风书屋
https://www.qiufengshuwu.com/
autobcb_admin (12020)04/25 06:19
秋风书屋 ai
{
"bookSourceUrl": "https:\/\/www.qiufengshuwu.com\/",
"bookSourceName": "秋风书屋",
"enabledExplore": true,
"enabled": true,
"bookSourceGroup": "",
"author": "",
"help": true,
"html": "<!DOCTYPE html>\n<html lang=\"en\">\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<!--如果要引入外部 js 必须在书源代码的上面-->\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 {\n return await window.flutter_inappwebview.callHandler('buildNumber');\n } catch (error) {\n return 0;\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 async htmlToText(str) {\n try {\n return await window.flutter_inappwebview.callHandler('htmlToText', str);\n } catch (error) {\n return \"\";\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 async toSimplified(str) {\n try {\n return await window.flutter_inappwebview.callHandler('toSimplified', str);\n } catch (error) {\n return \"\";\n }\n }\n\n async voice() {\n try {\n return await window.flutter_inappwebview.callHandler('voice');\n } catch (error) {\n return \"\";\n }\n }\n\n async getDeviceid() {\n try {\n return await window.flutter_inappwebview.callHandler('id');\n } catch (error) {\n return \"\";\n }\n }\n\n async getDevice() {\n try {\n return await window.flutter_inappwebview.callHandler('device');\n } catch (error) {\n return \"\";\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 async log(str) {\n try {\n return await window.flutter_inappwebview.callHandler('log', str);\n } catch (error) {\n return false;\n }\n }\n\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 async showToast(str) {\n try {\n return await window.flutter_inappwebview.callHandler('showToast', str);\n } catch (error) {\n return false;\n }\n }\n\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 async getWebViewUA() {\n try {\n return await window.flutter_inappwebview.callHandler('getWebViewUA');\n } catch (error) {\n return \"\";\n }\n }\n\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 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 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 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 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 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 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 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 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 async back() {\n try {\n return await window.flutter_inappwebview.callHandler('back');\n } catch (error) {\n return false;\n }\n }\n\n async utf8ToGbkUrlEncoded(str) {\n try {\n return await window.flutter_inappwebview.callHandler('utf8ToGbkUrlEncoded', str);\n } catch (error) {\n return \"\";\n }\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 async addbook(bookUrl) {\n try {\n return await window.flutter_inappwebview.callHandler('addbook', bookUrl);\n } catch (error) {\n return \"\";\n }\n }\n\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 async base64encode(str) {\n try {\n return await window.flutter_inappwebview.callHandler('base64encode', str);\n } catch (error) {\n return \"\";\n }\n }\n\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 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 {\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 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\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 async getLoginInfo() {\n return await this.get(\"LoginInfo\");\n }\n\n async putLoginInfo(info) {\n return await this.set(\"LoginInfo\", info);\n }\n\n async getbookVariable(bookurl) {\n return await this.get(bookurl);\n }\n\n async setbookVariable(bookurl, value) {\n return await this.set(bookurl, value);\n }\n }\n\n class Cookie {\n constructor() {}\n\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 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 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 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 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 function parseHTMLSafely(htmlStr) {\n try {\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 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 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.qiufengshuwu.com\";\n var header = {\n \"User-Agent\": \"\",\n \"Referer\": baseurl\n };\n\n async function init() {\n var drive = await flutterBridge.getDevice();\n if (drive == \"ohos\") {\n header[\"User-Agent\"] = \"Mozilla\/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit\/537.36 (KHTML, like Gecko) Chrome\/143.0.0.0 Safari\/537.36 Edg\/143.0.0.0\";\n } else {\n header[\"User-Agent\"] = await flutterBridge.getWebViewUA();\n }\n flutterBridge.CookieJar();\n }\n\n function isCloudflareChallenge(html) {\n return html && (\n html.indexOf('challenge') !== -1 ||\n html.indexOf('cloudflare') !== -1 ||\n html.indexOf('Just a moment') !== -1 ||\n html.indexOf('Ray ID') !== -1\n );\n }\n\n async function search(key, page) {\n if (page > 1) {\n return \"[]\";\n }\n\n var searchUrl = baseurl + \"\/s.html\";\n var body = \"s=\" + encodeURIComponent(key) + \"&type=articlename\";\n \n\n flutterBridge.log(\"搜索URL: \" + searchUrl);\n\n var htmlContent = await flutterBridge.webview(searchUrl, \"\", \"\", body, JSON.stringify(header));\n\n \/\/ 检查是否有问题,遇到Cloudflare就用startBrowser让用户手动验证\n if (!htmlContent || isCloudflareChallenge(htmlContent) || (get && get.statusCode != 200)) {\n flutterBridge.showToast(\"如果没有出现验证请手动搜索\");\n var s = await flutterBridge.startBrowser(searchUrl, \"验证\", JSON.stringify(header));\n if (s) {\n htmlContent = s;\n }\n }\n\n \/\/ 再检查\n if (!htmlContent) {\n flutterBridge.showToast(\"搜索请求失败,请在浏览器中完成验证\");\n return \"[]\";\n }\n\n flutterBridge.text(0, htmlContent);\n\n var books = [];\n var $tempContainer = parseHTMLSafely(removeHTMLTags(htmlContent));\n\n \/\/ 解析 .sone(搜索结果页)\n $tempContainer.find(\".sone\").each(function (index) {\n try {\n var $element = $(this);\n var bookUrl = $element.find(\"a\").attr('href');\n var bookName = $element.find(\"a\").text().trim();\n var author = $element.find(\".author a\").text().trim();\n\n if (bookUrl && bookName) {\n if (!bookUrl.startsWith('http')) {\n bookUrl = baseurl + bookUrl;\n }\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 \/\/ 解析 .article(分类页)\n $tempContainer.find(\".article\").each(function (index) {\n try {\n var $element = $(this);\n var bookUrl = $element.find(\"a\").attr('href');\n var bookName = $element.find(\"h6 a\").text().trim();\n var author = $element.find(\".author a\").text().trim();\n var coverUrl = $element.find(\"img\").attr('data-src') || $element.find(\"img\").attr('src') || \"\";\n\n if (bookUrl && bookName) {\n if (!bookUrl.startsWith('http')) {\n bookUrl = baseurl + bookUrl;\n }\n if (coverUrl && !coverUrl.startsWith('http')) {\n coverUrl = baseurl + coverUrl;\n }\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 }\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 info(bookurl) {\n var mheader = {\n ...header,\n \"Referer\": baseurl\n };\n\n var htmlContent = \"\";\n\n \/\/ 直接用 webview\n \/\/ webview(url, js, html, body, header)\n htmlContent = await flutterBridge.webview(bookurl, \"\", \"\", \"\", JSON.stringify(mheader));\n\n if (!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(removeHTMLTags(htmlContent));\n\n \/\/ 从详情页正确获取信息\n var name = $tempContainer.find(\".cataloginfo h3\").text().trim();\n var author = \"\";\n var coverUrl = $tempContainer.find(\".pic img\").attr('src') || \"\";\n var intro = $tempContainer.find(\"#intro p\").text().trim();\n var latestChapterTitle = \"\";\n var tocUrl = bookurl; \/\/ 默认为bookurl\n\n \/\/ 获取作者\n $tempContainer.find(\".infotype p\").each(function () {\n var text = $(this).text();\n if (text.indexOf(\"作者名称\") !== -1) {\n var authorLink = $(this).find(\"a\").text().trim();\n if (authorLink) {\n author = authorLink;\n }\n }\n });\n\n \/\/ 获取目录URL\n $tempContainer.find(\"a\").each(function () {\n var linkText = $(this).text().trim();\n var linkUrl = $(this).attr('href');\n if (linkUrl && (linkText.indexOf(\"章节目录\") !== -1 || linkText.indexOf(\"查看全部章节\") !== -1)) {\n if (!linkUrl.startsWith('http')) {\n tocUrl = baseurl + linkUrl;\n } else {\n tocUrl = linkUrl;\n }\n }\n });\n\n if (coverUrl && !coverUrl.startsWith('http')) {\n coverUrl = baseurl + coverUrl;\n }\n\n var book = {\n \"bookUrl\": bookurl,\n \"name\": name,\n \"author\": author,\n \"kind\": \"\",\n \"coverUrl\": coverUrl,\n \"intro\": intro,\n \"tocUrl\": tocUrl,\n \"wordCount\": \"\",\n \"type\": 0,\n \"latestChapterTitle\": latestChapterTitle\n };\n\n removeHTMLSafely($tempContainer);\n return JSON.stringify(book);\n }\n\n async function chapter(tocUrl) {\n var mheader = {\n ...header,\n \"Referer\": baseurl\n };\n\n var htmlContent = \"\";\n\n \/\/ 直接用 webview\n htmlContent = await flutterBridge.webview(tocUrl, \"\", \"\", \"\", JSON.stringify(mheader));\n\n if (!htmlContent) {\n flutterBridge.showToast(\"如果没有出现验证请手动打开书籍目录\");\n var s = await flutterBridge.startBrowser(tocUrl, \"验证\", 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 $tempContainer = parseHTMLSafely(removeHTMLTags(htmlContent));\n\n var chapterIndex = 0;\n\n \/\/ 使用正确的选择器解析目录\n $tempContainer.find(\"ul.chapters li a\").each(function (index) {\n try {\n var chapterUrl = $(this).attr('href');\n var chapterName = $(this).text().trim();\n\n if (chapterUrl && chapterName) {\n \/\/ 确保链接是完整的\n var finalChapterUrl = chapterUrl;\n if (!chapterUrl.startsWith('http')) {\n finalChapterUrl = baseurl + chapterUrl;\n }\n\n var chapter = {\n \"name\": chapterName,\n \"chapterId\": finalChapterUrl,\n \"index\": chapterIndex,\n \"isPay\": false,\n \"isVip\": false,\n \"isVolume\": false,\n \"tag\": \"\"\n };\n chapters.push(chapter);\n chapterIndex++;\n }\n } catch (e) {\n flutterBridge.log(\"解析章节出错:\" + e.message);\n }\n });\n\n removeHTMLSafely($tempContainer);\n flutterBridge.log(\"找到 \" + chapters.length + \" 个章节\");\n return JSON.stringify(chapters);\n }\n\n async function content(url) {\n var mheader = {\n ...header,\n \"Referer\": baseurl\n };\n\n var htmlContent = \"\";\n\n \/\/ 直接用 webview\n htmlContent = await flutterBridge.webview(url, \"\", \"\", \"\", JSON.stringify(mheader));\n\n if (!htmlContent) {\n flutterBridge.showToast(\"如果没有出现验证请手动打开章节\");\n var s = await flutterBridge.startBrowser(url, \"验证\", JSON.stringify(mheader));\n if (s) {\n htmlContent = s;\n }\n }\n\n if (!htmlContent) {\n flutterBridge.showToast(\"获取内容失败,请在浏览器中完成验证\");\n return \"\";\n }\n\n flutterBridge.text(3, htmlContent);\n\n var $tempContainer = parseHTMLSafely(htmlContent);\n var contenttxt = \"\";\n\n \/\/ 使用正确的选择器解析正文\n var $contentDiv = $tempContainer.find(\"#novelcontent\");\n \n \/\/ 获取所有 <p> 标签并拼接\n $contentDiv.find(\"p\").each(function (index) {\n var pText = $(this).text().trim();\n if (pText) {\n contenttxt += pText + \"\\n\\n\"; \/\/ 每个段落加两个换行\n }\n });\n\n removeHTMLSafely($tempContainer);\n\n if (!contenttxt || contenttxt.length < 10) {\n flutterBridge.showToast(\"获取内容失败\");\n return \"\";\n }\n\n return contenttxt.trim();\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(\"发现\", \"\", 0);\n push(\"热门小说\", baseurl + \"\/sort\/1_{{page}}\/\", 0);\n push(\"言情小说\", baseurl + \"\/sort\/2_{{page}}\/\", 0);\n push(\"中国文学\", baseurl + \"\/sort\/3_{{page}}\/\", 0);\n push(\"外国文学\", baseurl + \"\/sort\/4_{{page}}\/\", 0);\n push(\"小小说\", baseurl + \"\/sort\/5_{{page}}\/\", 0);\n push(\"原创小说\", baseurl + \"\/sort\/6_{{page}}\/\", 0);\n push(\"历史小说\", baseurl + \"\/sort\/7_{{page}}\/\", 0);\n push(\"精选推荐\", baseurl + \"\/sort\/8_{{page}}\/\", 0);\n push(\"排行\", baseurl + \"\/top\/\", 0);\n push(\"新书\", baseurl + \"\/top\/postdate_1\/\", 0);\n push(\"完本\", baseurl + \"\/full\/1\/\", 0);\n\n return JSON.stringify(result);\n }\n\n async function find(url, page) {\n var u = url;\n \n \/\/ 先替换 {{page}}\n u = u.replaceAll('{{page}}', page);\n \n if (page > 1 && u.indexOf('\/sort\/') !== -1) {\n var parts = u.split('\/');\n if (parts.length > 4) {\n parts[4] = page + \"_1\";\n u = parts.join('\/');\n }\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 = \"\";\n\n \/\/ 直接用 webview\n htmlContent = await flutterBridge.webview(u, \"\", \"\", \"\", JSON.stringify(mheader));\n\n if (!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 $tempContainer = parseHTMLSafely(removeHTMLTags(htmlContent));\n\n \/\/ 解析 .sone(搜索结果页)\n $tempContainer.find(\".sone\").each(function (index) {\n try {\n var $element = $(this);\n var bookUrl = $element.find(\"a\").attr('href');\n var bookName = $element.find(\"a\").text().trim();\n var author = $element.find(\".author a\").text().trim();\n\n if (bookUrl && bookName) {\n if (!bookUrl.startsWith('http')) {\n bookUrl = baseurl + bookUrl;\n }\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 \/\/ 解析 .article(分类页)\n $tempContainer.find(\".article\").each(function (index) {\n try {\n var $element = $(this);\n var bookUrl = $element.find(\"a\").attr('href');\n var bookName = $element.find(\"h6 a\").text().trim();\n var author = $element.find(\".author a\").text().trim();\n var coverUrl = $element.find(\"img\").attr('data-src') || $element.find(\"img\").attr('src') || \"\";\n\n if (bookUrl && bookName) {\n if (!bookUrl.startsWith('http')) {\n bookUrl = baseurl + bookUrl;\n }\n if (coverUrl && !coverUrl.startsWith('http')) {\n coverUrl = baseurl + coverUrl;\n }\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 }\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": "1777069163651"
}