西德书屋
https://m.xide8.com/
autobcb_admin (12020)03/07 03:24
西德书屋 支持搜索
{
"bookSourceUrl": "https:\/\/m.xide8.com\/",
"bookSourceName": "西德书屋",
"enabledExplore": true,
"enabled": true,
"bookSourceGroup": "",
"author": "trae",
"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<!-- 没用到jq请去掉-->\n<script src=\"https:\/\/vc.jd.com\/web\/js\/jquery-3.1.1.min.js\"><\/script>\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弹窗\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 \/\/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 * 通用返回字段\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 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 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 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\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 const baseurl = \"https:\/\/m.xide8.com\";\n const header = {\n \"User-Agent\": \"Mozilla\/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit\/537.36 (KHTML, like Gecko) Chrome\/91.0.4472.124 Safari\/537.36\"\n };\n\n async function search(key, page) {\n let url = baseurl + \"\/search\/\";\n \n var getResult = await http.Get(url, header, true);\n if (!getResult || getResult.statusCode !== 200) {\n return \"[]\";\n }\n \n let htmlContent = getResult.data;\n \n let searchJs = `\n <script>\n window.onload = function() {\n document.getElementById('keyword').value = '${key}';\n const element = document.querySelector('.btn');\n if (element) {\n element.click();\n }\n };\n <`+`\/script>\n `;\n \n htmlContent = htmlContent.replace('<\/head>', searchJs + '<\/head>');\n \n var result = await flutterBridge.webview(url, \"\", htmlContent, \"\", JSON.stringify(header));\n \n if (!result) {\n return \"[]\";\n }\n flutterBridge.text(0,result)\n \n try {\n var html = parseHTMLSafely(result);\n var books = [];\n \n html.find(\".search-card\").each(function() {\n let book = {};\n let $this = $(this);\n \n let statusSpan = $this.find(\".sc-title span.gray\");\n if (statusSpan.length > 0) {\n book.kind = statusSpan.text().replace(\/[\\[\\]]\/g, \"\");\n }\n \n let bookLink = $this.find(\".sc-title span a[target='_blank']\");\n if (bookLink.length > 0) {\n book.name = bookLink.text();\n book.bookUrl = baseurl + bookLink.attr(\"href\");\n }\n \n let authorLink = $this.find(\".sc-title a.green\");\n if (authorLink.length > 0) {\n book.author = authorLink.text();\n }\n \n let introDiv = $this.find(\".sc-content\");\n if (introDiv.length > 0) {\n book.intro = introDiv.text().trim();\n }\n \n let lastChapterSpan = $this.find(\".sc-time span:first\");\n if (lastChapterSpan.length > 0) {\n book.lastChapter = lastChapterSpan.text();\n }\n \n if (book.name && book.bookUrl) {\n books.push(book);\n }\n });\n \n removeHTMLSafely(html);\n return JSON.stringify(books);\n } catch (e) {\n flutterBridge.log(\"搜索解析错误:\" + e.message);\n return \"[]\";\n }\n }\n\n async function info(bookurl) {\n var result = await http.Get(bookurl, JSON.stringify(header), true);\n if (!result || result.statusCode !== 200) {\n return JSON.stringify({});\n }\n \n try {\n var html = parseHTMLSafely(result.data);\n var book = {};\n \n book.bookUrl = bookurl;\n \n let titleDiv = html.find(\".BGsectionOne-top-right .title\");\n if (titleDiv.length > 0) {\n book.name = titleDiv.text().trim();\n }\n \n let authorLink = html.find(\".BGsectionOne-top-right .author a\");\n if (authorLink.length > 0) {\n book.author = authorLink.text().trim();\n }\n \n let img = html.find(\".BGsectionOne-top-left img\");\n if (img.length > 0) {\n let coverUrl = img.attr(\"data-lazy\") || img.attr(\"src\");\n if (coverUrl && !coverUrl.startsWith(\"http\")) {\n coverUrl = \"https:\" + coverUrl;\n }\n book.coverUrl = coverUrl;\n }\n \n let introDiv = html.find(\".sectionTwo-content\");\n if (introDiv.length > 0) {\n book.intro = introDiv.text().trim();\n }\n \n let kindSpan = html.find(\".age span\");\n if (kindSpan.length > 0) {\n book.kind = kindSpan.text().trim();\n }\n \n let lastChapterLink = html.find(\".chaptercount a\");\n if (lastChapterLink.length > 0) {\n book.lastChapter = \"共\" + lastChapterLink.text().match(\/\\d+\/)[0] + \"章\";\n book.tocUrl = baseurl + lastChapterLink.attr(\"href\");\n }\n \n let statusSpan = html.find(\".time span\");\n if (statusSpan.length > 0) {\n book.status = statusSpan.text().trim();\n }\n \n removeHTMLSafely(html);\n return JSON.stringify(book);\n } catch (e) {\n flutterBridge.log(\"详情解析错误:\" + e.message);\n return JSON.stringify({});\n }\n }\n\n async function chapter(tocUrl, bookurl) {\n var chapters = [];\n var currentPage = 1;\n var totalPages = 1;\n \n flutterBridge.log(\"目录开始获取:\" + tocUrl);\n \n var result = await http.Get(tocUrl, JSON.stringify(header), true);\n if (!result || result.statusCode !== 200) {\n flutterBridge.log(\"目录第一页获取失败:\" + (result ? result.statusCode : \"null\"));\n return \"[]\";\n }\n \n flutterBridge.log(\"目录第一页获取成功,数据长度:\" + result.data.length);\n \n try {\n var html = parseHTMLSafely(result.data);\n \n let scriptText = result.data;\n let pageMatch = scriptText.match(\/i\\s*<=\\s*(\\d+)\/);\n if (pageMatch) {\n totalPages = parseInt(pageMatch[1]);\n flutterBridge.log(\"检测到总页数:\" + totalPages);\n } else {\n flutterBridge.log(\"未检测到总页数,使用默认值1\");\n }\n \n let firstPageCount = 0;\n html.find(\".BGsectionThree-content li a\").each(function() {\n let chapter = {};\n chapter.name = $(this).text().trim();\n chapter.chapterId = baseurl + $(this).attr(\"href\");\n chapters.push(chapter);\n firstPageCount++;\n });\n \n flutterBridge.log(\"第一页章节数:\" + firstPageCount);\n removeHTMLSafely(html);\n } catch (e) {\n flutterBridge.log(\"目录第一页解析错误:\" + e.message);\n }\n \n currentPage = 2;\n flutterBridge.log(\"开始获取第2页到第\" + totalPages + \"页\");\n \n while (currentPage <= totalPages) {\n var pageUrl = tocUrl.replace(\"\/ml.html\", \"\/ml_\" + currentPage + \".html\");\n flutterBridge.log(\"获取第\" + currentPage + \"页:\" + pageUrl);\n\n var result = await http.Get(pageUrl, JSON.stringify(header), true);\n if (!result || result.statusCode !== 200) {\n flutterBridge.log(\"第\" + currentPage + \"页获取失败:\" + (result ? result.statusCode : \"null\"));\n break;\n }\n \n flutterBridge.log(\"第\" + currentPage + \"页获取成功\");\n \n try {\n var html = parseHTMLSafely(result.data);\n \n let pageCount = 0;\n html.find(\".BGsectionThree-content li a\").each(function() {\n let chapter = {};\n chapter.name = $(this).text().trim();\n chapter.chapterId = baseurl + $(this).attr(\"href\");\n chapters.push(chapter);\n pageCount++;\n });\n \n flutterBridge.log(\"第\" + currentPage + \"页章节数:\" + pageCount);\n removeHTMLSafely(html);\n } catch (e) {\n flutterBridge.log(\"第\" + currentPage + \"页解析错误:\" + e.message);\n break;\n }\n \n currentPage++;\n }\n \n flutterBridge.log(\"目录获取完成,总章节数:\" + chapters.length);\n \n for (let i = 0; i < chapters.length; i++) {\n chapters[i].index = i;\n }\n \n return JSON.stringify(chapters);\n }\n\n async function content(u1, bookurl) {\n flutterBridge.log(u1)\n var result1 = await http.Get(u1, JSON.stringify(header), true);\n if(result1 == null){\n result1 = await http.Get(u1, JSON.stringify(header), true);\n }\n flutterBridge.text(3,result1.data)\n var url=\"https:\"+result1.data.match(\/initTxt\\(\"([^\"]+)\",\"[^\"]+\"\\)\/)[1];\n flutterBridge.log(url);\n var result = await http.Get(url, JSON.stringify(header), true);\n if(result == null){\n result = await http.Get(url, JSON.stringify(header), true);\n }\n flutterBridge.text(3,result.data)\n var x=String(result.data.match(\/_txt_call\\(([\\s\\S]+\\})\\)\/)[1]);\n var r = eval('(' + x + ')');\n if(r.content!=null){\n var e = r.replace;\n for (var n in e) {\n var i = new RegExp(e[n], \"ig\");\n r.content = r.content.replace(i, n)\n }\n r.content = r.content.replace(\/&#x([0-9a-fA-F]+);\/g, function(match, hex) {\n return String.fromCharCode(parseInt(hex, 16));\n });\n return r.content\n }else{\n return \"章节加载失败,或者内容正在手打中,请稍后访问~\";\n }\n }\n\n async function getfinds() {\n var finds = [];\n \n finds.push({\n title: \"都市好书\",\n url: baseurl + \"\/\"\n });\n \n finds.push({\n title: \"言情小说\",\n url: baseurl + \"\/yanqingxiaoshuo\/\"\n });\n \n finds.push({\n title: \"都市小说\",\n url: baseurl + \"\/dushixiaoshuo\/\"\n });\n \n finds.push({\n title: \"仙侠小说\",\n url: baseurl + \"\/xianxiaxiaoshuo\/\"\n });\n \n finds.push({\n title: \"完本小说\",\n url: baseurl + \"\/over.html\"\n });\n \n return JSON.stringify(finds);\n }\n\n async function find(url, page) {\n var result = await http.Get(url, JSON.stringify(header), true);\n if (!result || result.statusCode !== 200) {\n return \"[]\";\n }\n \n try {\n var html = parseHTMLSafely(result.data);\n var books = [];\n \n html.find(\"a.col.xs-4.text-center\").each(function() {\n let book = {};\n let $this = $(this);\n \n let nameDiv = $this.find(\".text-hide.f13\");\n if (nameDiv.length > 0) {\n book.title = nameDiv.text().trim();\n }\n \n let href = $this.attr(\"href\");\n if (href) {\n book.bookUrl = baseurl + href;\n }\n \n let img = $this.find(\"img\");\n if (img.length > 0) {\n let coverUrl = img.attr(\"data-lazy\") || img.attr(\"src\");\n if (coverUrl && !coverUrl.startsWith(\"http\")) {\n coverUrl = \"https:\" + coverUrl;\n }\n book.coverUrl = coverUrl;\n }\n \n if (book.title && book.bookUrl) {\n books.push(book);\n }\n });\n \n if (books.length === 0) {\n html.find(\".book_img_pic a\").each(function() {\n let book = {};\n let href = $(this).attr(\"href\");\n if (href) {\n book.bookUrl = baseurl + href;\n book.title = $(this).attr(\"title\") || $(this).find(\"img\").attr(\"alt\");\n \n let img = $(this).find(\"img\");\n if (img.length > 0) {\n let coverUrl = img.attr(\"data-lazy\") || img.attr(\"src\");\n if (coverUrl && !coverUrl.startsWith(\"http\")) {\n coverUrl = \"https:\" + coverUrl;\n }\n book.coverUrl = coverUrl;\n }\n \n if (book.title && book.bookUrl) {\n books.push(book);\n }\n }\n });\n }\n \n removeHTMLSafely(html);\n return JSON.stringify(books);\n } catch (e) {\n flutterBridge.log(\"发现解析错误:\" + e.message);\n return \"[]\";\n }\n }\n\n\n \/\/返回http开头的则任务登录链接会跳webview,其他的会按照json解析显示弹窗\n async function getloginurl(){\n return baseurl;\n }\n\n \/\/如果登录 url 为非 http 开头的弹窗界面,每次修改完弹窗就会执行此函数\n async function login(){\n\n }\n\n async function pay(bookurl,url){\n\n }\n\n \/\/ url 为图片的url,如果需要传递参数可以在图片后接json字符串,例如:http:\/\/127.0.0.1,{'headers':{'a':'b'}}\n \/\/图片解密,image 为加密的图片的base64,执行的js必须是字符串所以这参数只能base64转码\n \/\/这个函数得返回byteList List<int> ,并且能直接被Uint8List.fromList(byteList)接受\n async function imagedecrypt(url,image){\n\n }\n \/\/ 当调用startBrowserWithShouldOverrideUrlLoading时必须有此函数\n \/\/ url 为每次打开的 url\n \/\/ 返回 false 则会取消打开这个网页\n async function shouldOverrideUrlLoading(url){\n return true;\n }\n\n\n \/\/帮助内容,开启帮助后点击帮助将会显示函数反馈的内容\n \/\/当前函数有三种反馈方式\n \/\/1. http 开头的链接\n \/\/2. 纯文字\n \/\/3. @html: 开头的 html 内容\n async function gethelp(){\n return \"\";\n }\n\n<\/script>\n\n<\/html>\n",
"login": false,
"lastUpdateTime": "1772825051970"
}