小说狂人

https://czbooks.net

autobcb_admin (12020)3天前

小说狂人 作者 清词

二维码导入(APP尚未完成该功能)
{
    "bookSourceUrl": "https:\/\/czbooks.net",
    "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>小说狂人<\/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    \/\/将简体字转成繁体字\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    \/\/输出日志,前台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  \n  var baseurl = \"http:\/\/czbooks.net\";\n  var header = {\n    \"User-Agent\": navigator.userAgent,\n    \"Accept\": \"text\/html,application\/xhtml+xml,application\/xml;q=0.9,image\/webp,image\/apng,*\/*;q=0.8\",\n    \"Accept-Language\": \"zh-CN,zh;q=0.9,en;q=0.8\",\n    \"Accept-Encoding\": \"gzip, deflate\",\n    \"Referer\": \"http:\/\/czbooks.net\/\"\n  };\n\n  async function method(result) {\n    if (!result) return result;\n    result = result.replace(\/著\/g, '着').replace(\/仿真\/g, '模拟');\n    try {\n      result = await flutterBridge.toSimplified(result);\n    } catch (error) {\n      flutterBridge.log(\"繁体转简体失败: \" + error);\n    }\n    return result;\n  }\n\n  function cleanChapterTitle(title) {\n    if (!title) return '';\n    let cleaned = title\n      .replace(\"••\", \"\")\n      .replace(\/^(\\d+).第\/, '第')\n      .replace(\/^(\\d+)[、.]第.+章\/, '第$1章')\n      .replace(\/^(\\d+)、\\d+、\/, '第$1章 ')\n      .replace(\/^(\\d+)、\\d+\/, '第$1章')\n      .replace(\/^(第.+章)\\s?\\d+\/, '$1')\n      .replace(\/^(\\d+)、\/, '第$1章 ')\n      .replace(\/^(第.+章)\\s?第.+章\/, '$1')\n      .replace(\/第\\s(.+)\\s章\/, '第$1章')\n      .replace(\/.*(chapter|Chapter)\\s?(\\d+)\\s?\/, '$1 $2 ')\n      .replace(\/\\(.+\\)\/, '')\n      .replace(\/\\[|。\/g, '');\n    return cleaned.trim();\n  }\n\n  function cleanHtmlContent(html) {\n  if (!html) return '';\n  \n  html = removeHTMLTags(html);\n  \n  html = html.replace(\/<br\\s*\\\/?>\/gi, '\\n')\n            .replace(\/<br\\s*\\\/?>\\s*\/gi, '\\n')\n            .replace(\/<\\\/p>\/gi, '\\n')\n            .replace(\/<p[^>]*>\/gi, '')\n            .replace(\/<\\\/div>\/gi, '\\n')\n            .replace(\/<div[^>]*>\/gi, '')\n            .replace(\/<\\\/h[1-6]>\/gi, '\\n')\n            .replace(\/<h[1-6][^>]*>\/gi, '')\n            .replace(\/&nbsp;\/g, ' ')\n            .replace(\/&lt;\/g, '<')\n            .replace(\/&gt;\/g, '>')\n            .replace(\/&amp;\/g, '&')\n            .replace(\/&quot;\/g, '\"')\n            .replace(\/&#39;\/g, \"'\")\n            .replace(\/&#160;\/g, ' ')\n            .replace(\/&#xa0;\/g, ' ')\n            .replace(\/\\r\\n\/g, '\\n')\n            .replace(\/\\r\/g, '\\n');\n  \n  html = html.replace(\/<[^>]+>\/g, '');\n  \n  html = html.replace(\/[ \\t]+\/g, ' ')\n            .replace(\/^\\s+|\\s+$\/g, '')\n            .replace(\/\\n{3,}\/g, '\\n\\n')\n            .replace(\/[ \\t]+\\n\/g, '\\n')\n            .replace(\/\\n[ \\t]+\/g, '\\n')\n            .replace(\/\\s*\\n\\s*\/g, '\\n');\n  \n  return html.trim();\n}\n\n  async function search(key, page) {\n    var url = baseurl + \"\/s\/\" + encodeURIComponent(key) + \"?q=\" + encodeURIComponent(key);\n    if (page > 1) {\n      url += \"&page=\" + page;\n    }\n    \n    var get = await http.Get(url, header, true);\n    var result = get.data;\n    \n    if (!result) return \"[]\";\n    \n    result = await method(result);\n    flutterBridge.text(0, result);\n    \n    var books = [];\n    var $temp = parseHTMLSafely(result);\n    \n    $temp.find(\"li.novel-item-wrapper\").each(function() {\n      var $this = $(this);\n      var $a = $this.find(\"a\").eq(0);\n      var $img = $this.find(\"img\");\n      var $title = $this.find(\".novel-item-title\");\n      var $author = $this.find(\".novel-item-author a\");\n      var $date = $this.find(\".novel-item-date\");\n      var $nav = $this.find(\".nav\");\n      var $chapter = $this.find(\".novel-item-newest-chapter a\");\n      \n      var navText = $nav.text() || \"\";\n      var navParts = navText.split(\/\\s+\/);\n      var collection = navParts.length > 0 ? navParts[0] : \"\";\n      var view = navParts.length > 1 ? navParts[1] : \"\";\n      \n      var chapterText = $chapter.text() || \"\";\n      var dateText = $date.text() || \"\";\n      var lastChapter = cleanChapterTitle(chapterText + \"•\" + dateText);\n      \n      var bookUrl = $a.attr('href') || '';\n      if (bookUrl && !bookUrl.startsWith('http')) {\n        if (bookUrl.startsWith('\/\/')) {\n          bookUrl = 'http:' + bookUrl;\n        } else if (bookUrl.startsWith('\/')) {\n          bookUrl = baseurl + bookUrl;\n        } else {\n          bookUrl = baseurl + '\/' + bookUrl;\n        }\n      }\n      \n      var book = {\n        \"bookUrl\": bookUrl,\n        \"name\": ($title.text() || '').trim(),\n        \"author\": ($author.text() || '').trim(),\n        \"kind\": ($date.text() || '').trim(),\n        \"coverUrl\": ($img.attr('src') || '') + \",\" + JSON.stringify({\"headers\": header}),\n        \"intro\": \"收藏:\" + collection + \"\\n观看:\" + view,\n        \"tocUrl\": bookUrl,\n        \"wordCount\": \"\",\n        \"type\": 0,\n        \"latestChapterTitle\": lastChapter\n      };\n      \n      if (book.bookUrl && book.name) {\n        books.push(book);\n      }\n    });\n    \n    removeHTMLSafely($temp[0]);\n    return JSON.stringify(books);\n  }\n\n  async function info(bookurl) {\n    if (!bookurl.startsWith('http')) {\n      if (bookurl.startsWith('\/\/')) {\n        bookurl = 'http:' + bookurl;\n      } else if (bookurl.startsWith('\/')) {\n        bookurl = baseurl + bookurl;\n      } else {\n        bookurl = baseurl + '\/' + bookurl;\n      }\n    }\n    \n    var get = await http.Get(bookurl, header, true);\n    var result = get.data;\n    \n    if (!result) {\n      return JSON.stringify({\n        \"bookUrl\": bookurl,\n        \"name\": \"\",\n        \"author\": \"\",\n        \"kind\": \"\",\n        \"coverUrl\": \"\",\n        \"intro\": \"\",\n        \"tocUrl\": bookurl,\n        \"wordCount\": \"\",\n        \"type\": 0,\n        \"latestChapterTitle\": \"\"\n      });\n    }\n    \n    result = await method(result);\n    flutterBridge.text(1, result);\n    \n    var $temp = parseHTMLSafely(result);\n    \n    var $title = $temp.find(\"span.title\");\n    var $author = $temp.find(\"span.author a\");\n    var $thumbnail = $temp.find(\".thumbnail img\");\n    var $description = $temp.find(\".description\");\n    \n    var name = ($title.text() || '').replace(\/《|》\/g, '').trim();\n    var author = ($author.text() || '').trim();\n    \n    var coverUrl = $thumbnail.attr('src') || '';\n    if (coverUrl && !coverUrl.startsWith('http')) {\n      if (coverUrl.startsWith('\/\/')) {\n        coverUrl = 'https:' + coverUrl;\n      } else if (coverUrl.startsWith('\/')) {\n        coverUrl = baseurl + coverUrl;\n      }\n    }\n    coverUrl += \",\" + JSON.stringify({\"headers\": header});\n    \n    var introHtml = $description.html() || \"\";\n    introHtml = cleanHtmlContent(introHtml);\n    \n    var tableData = {};\n    $temp.find(\"table tbody tr\").each(function() {\n      var $tr = $(this);\n      var $tds = $tr.find(\"td\");\n      if ($tds.length >= 2) {\n        var key = ($tds.eq(0).text() || '').trim();\n        var value = ($tds.eq(1).text() || '').trim();\n        tableData[key] = value;\n      }\n    });\n    \n    var tags = [];\n    $temp.find(\"ul.hashtag li.item\").slice(0, 4).each(function() {\n      var tag = $(this).find(\"a\").text().trim();\n      if (tag) tags.push(tag);\n    });\n    \n    var $lastChapterLink = $temp.find(\"#chapter-list a\").last();\n    var lastChapterText = cleanChapterTitle($lastChapterLink.text() || \"\");\n    \n    var comments = \"\";\n    $temp.find(\"li.comment-item\").slice(0, 4).each(function() {\n      var comment = $(this).text().trim();\n      if (comment) {\n        comments += \"<p>👤 用户名:\" + comment + \"<\/p>\";\n      }\n    });\n    \n    if (comments) {\n      comments = comments\n        .replace(\/\\s(..)前\/g, ' \\n⏰ 时间:$1前')\n        .replace(\/\\s(\\d+)\\s\/g, ' 👍🏻 点赞:$1 \\n📖 评论:')\n        .replace(\/\\s・\/g, ' 📘章节:');\n      comments = cleanHtmlContent(comments);\n    }\n    \n    var intro = \"\";\n    if (tableData['最近更新']) {\n      intro += `🕰  更新:${tableData['最近更新']}\\n\\n`;\n    }\n    if (tableData['阅读']) {\n      intro += `👁  阅读:${tableData['阅读']}\\n\\n`;\n    }\n    if (tableData['收藏']) {\n      intro += `❤️  收藏:${tableData['收藏']}\\n\\n`;\n    }\n    if (tags.length > 0) {\n      intro += `🏷  标签:${tags.join('·')}\\n\\n`;\n    }\n    if (lastChapterText) {\n      intro += `☘️  最新:${lastChapterText}\\n\\n`;\n    }\n    intro += `📜  简介:\\n${introHtml}`;\n    \n    if (comments) {\n      intro += `\\n\\n🍚  书评:\\n${comments}`;\n    }\n    \n    var kind = \"\";\n    if (tableData['分类']) kind += tableData['分类'] + \"\\n\";\n    if (tableData['状态']) kind += tableData['状态'] + \"\\n\";\n    if (tableData['最近更新']) kind += tableData['最近更新'];\n    \n    var book = {\n      \"bookUrl\": bookurl,\n      \"name\": name,\n      \"author\": author,\n      \"kind\": kind.trim(),\n      \"coverUrl\": coverUrl,\n      \"intro\": intro,\n      \"tocUrl\": bookurl,\n      \"wordCount\": \"\",\n      \"type\": 0,\n      \"latestChapterTitle\": cleanChapterTitle(`${lastChapterText}•${tableData['最近更新'] || ''}`)\n    };\n    \n    removeHTMLSafely($temp[0]);\n    return JSON.stringify(book);\n  }\n\n  async function chapter(tocUrl, bookurl) {\n    if (!tocUrl.startsWith('http')) {\n      if (tocUrl.startsWith('\/\/')) {\n        tocUrl = 'http:' + tocUrl;\n      } else if (tocUrl.startsWith('\/')) {\n        tocUrl = baseurl + tocUrl;\n      } else {\n        tocUrl = baseurl + '\/' + tocUrl;\n      }\n    }\n    \n    var get = await http.Get(tocUrl, header, true);\n    var result = get.data;\n    \n    if (!result) return \"[]\";\n    \n    result = await method(result);\n    flutterBridge.text(2, result);\n    \n    var chapters = [];\n    var $temp = parseHTMLSafely(result);\n    \n    var bookName = \"\";\n    var $title = $temp.find(\"span.title\");\n    if ($title.length > 0) {\n      bookName = $title.text().replace(\/《|》\/g, '').trim();\n    }\n    \n    var currentIndex = 0;\n    $temp.find(\".volume, .chapter-list > li\").each(function() {\n      var $this = $(this);\n      \n      if ($this.hasClass(\"volume\")) {\n        var volumeName = $this.text().trim();\n        if (bookName && volumeName.includes(bookName)) {\n          volumeName = volumeName.replace(bookName, '').trim();\n        }\n        \n        if (volumeName) {\n          chapters.push({\n            \"name\": `🏮〖${volumeName}〗🏮`,\n            \"chapterId\": \"\",\n            \"index\": currentIndex,\n            \"isPay\": false,\n            \"isVip\": false,\n            \"isVolume\": true,\n            \"tag\": \"\"\n          });\n          currentIndex++;\n        }\n      } else {\n        $this.find(\"a\").each(function() {\n          var $a = $(this);\n          var chapterUrl = $a.attr('href') || '';\n          if (chapterUrl && !chapterUrl.startsWith('http')) {\n            if (chapterUrl.startsWith('\/\/')) {\n              chapterUrl = 'http:' + chapterUrl;\n            } else if (chapterUrl.startsWith('\/')) {\n              chapterUrl = baseurl + chapterUrl;\n            } else {\n              chapterUrl = baseurl + '\/' + chapterUrl;\n            }\n          }\n          \n          var chapter = {\n            \"name\": $a.text().trim(),\n            \"chapterId\": chapterUrl,\n            \"index\": currentIndex,\n            \"isPay\": false,\n            \"isVip\": false,\n            \"isVolume\": false,\n            \"tag\": \"\"\n          };\n          chapters.push(chapter);\n          currentIndex++;\n        });\n      }\n    });\n    \n    removeHTMLSafely($temp[0]);\n    return JSON.stringify(chapters);\n  }\n\n  async function content(url, bookurl) {\n    if (!url.startsWith('http')) {\n      if (url.startsWith('\/\/')) {\n        url = 'http:' + url;\n      } else if (url.startsWith('\/')) {\n        url = baseurl + url;\n      } else {\n        url = baseurl + '\/' + url;\n      }\n    }\n    \n    var get = await http.Get(url, header, true);\n    var result = get.data;\n    \n    if (!result) return \"\";\n    \n    result = await method(result);\n    flutterBridge.text(3, result);\n    \n    var content = \"\";\n    var $temp = parseHTMLSafely(result);\n    \n    $temp.find(\".content\").each(function() {\n      var html = $(this).html() || \"\";\n      content += html;\n    });\n    \n    if (content) {\n      content = cleanHtmlContent(content);\n      content = content\n        .replace(\/着名\/g, '著名')\n        .replace(\/土着\/g, '土著')\n        .replace(\/原着\/g, '原著')\n        .replace(\/名着\/g, '名著')\n        .replace(\/着作\/g, '著作');\n    }\n    \n    removeHTMLSafely($temp[0]);\n    return content;\n  }\n\n  async function getfinds() {\n    var finds = [];\n    \n    finds.push({\n      \"title\": \"༺ˇ»`ʚ排行榜ɞ´«ˇ༻\",\n      \"url\": \"\",\n      \"type\": 0,\n      \"width\": 3\n    });\n    \n    [[\"精选日榜\", \"day\"], [\"精选周榜\", \"week\"], [\"精选总榜\", \"total\"]].forEach(([title, uri]) => {\n      finds.push({\n        \"title\": title,\n        \"url\": `${baseurl}\/c\/normal\/${uri}\/{{page}}`,\n        \"type\": 0,\n        \"width\": 0\n      });\n    });\n    \n    [[\"人气日榜\", \"day\"], [\"人气周榜\", \"week\"], [\"人气总榜\", \"total\"]].forEach(([title, uri]) => {\n      finds.push({\n        \"title\": title,\n        \"url\": `${baseurl}\/c\/all\/${uri}\/{{page}}`,\n        \"type\": 0,\n        \"width\": 0\n      });\n    });\n    \n    [[\"收藏日榜\", \"day\"], [\"收藏周榜\", \"week\"], [\"收藏总榜\", \"total\"]].forEach(([title, uri]) => {\n      finds.push({\n        \"title\": title,\n        \"url\": `${baseurl}\/c\/favorite\/${uri}\/{{page}}`,\n        \"type\": 0,\n        \"width\": 0\n      });\n    });\n    \n    [[\"完结日榜\", \"day\"], [\"完结周榜\", \"week\"], [\"完结总榜\", \"total\"]].forEach(([title, uri]) => {\n      finds.push({\n        \"title\": title,\n        \"url\": `${baseurl}\/c\/finish\/${uri}\/{{page}}`,\n        \"type\": 0,\n        \"width\": 0\n      });\n    });\n    \n    [[\"工口日榜\", \"day\"], [\"工口周榜\", \"week\"], [\"工口总榜\", \"total\"]].forEach(([title, uri]) => {\n      finds.push({\n        \"title\": title,\n        \"url\": `${baseurl}\/c\/hot\/${uri}\/{{page}}`,\n        \"type\": 0,\n        \"width\": 0\n      });\n    });\n    \n    [[\"原创最新\", \"new\"], [\"原创日榜\", \"day\"], [\"原创周榜\", \"week\"], [\"原创总榜\", \"total\"], [\"------\", \"\"], [\"原创完本\", \"finish\"]].forEach(([title, uri]) => {\n      if (title === \"------\") {\n        finds.push({\n          \"title\": title,\n          \"url\": \"\",\n          \"type\": 0,\n          \"width\": 3\n        });\n      } else {\n        finds.push({\n          \"title\": title,\n          \"url\": `${baseurl}\/c\/creator\/${uri}\/{{page}}`,\n          \"type\": 0,\n          \"width\": 0\n        });\n      }\n    });\n    \n    finds.push({\n      \"title\": \"༺ˇ»`ʚ分类ɞ´«ˇ༻\",\n      \"url\": \"\",\n      \"type\": 0,\n      \"width\": 3\n    });\n    \n    const categories = [\n      [\"玄幻\", \"xuanhuan\"],\n      [\"言情\", \"yanqing\"],\n      [\"仙武\", \"xianxia\"],\n      [\"军史\", \"lishi\"],\n      [\"科游\", \"wangyou\"],\n      [\"灵异\", \"lingyi\"],\n      [\"女同\", \"tongren\"],\n      [\"原同\", \"erciyuan\"],\n      [\"耽美\", \"danmei\"],\n      [\"百合\", \"baihe\"],\n      [\"日系\", \"japan\"],\n      [\"奇冒\", \"fanatsy\"],\n      [\"电视\", \"drama\"],\n      [\"情工\", \"herotic\"],\n      [\"耽工\", \"blerotic\"],\n      [\"经典\", \"classicbook\"],\n      [\"推理\", \"suspense\"],\n      [\"女性\", \"girl\"],\n      [\"短篇\", \"short\"]\n    ];\n    \n    categories.forEach(([name, uri]) => {\n      [[\"最新\", \"new\"], [\"日榜\", \"day\"], [\"周榜\", \"week\"], [\"总榜\", \"total\"], [\"------\", \"\"], [\"完本\", \"finish\"]].forEach(([type, typeUri]) => {\n        if (type === \"------\") {\n          finds.push({\n            \"title\": \"------\",\n            \"url\": \"\",\n            \"type\": 0,\n            \"width\": 3\n          });\n        } else {\n          finds.push({\n            \"title\": `${name}${type}`,\n            \"url\": `${baseurl}\/c\/${uri}\/${typeUri}\/{{page}}`,\n            \"type\": 0,\n            \"width\": 0\n          });\n        }\n      });\n    });\n    \n    return JSON.stringify(finds);\n  }\n\n  async function find(url, page) {\n    if (!url || url === \"\") return \"[]\";\n    \n    var actualUrl = url.replace(\"{{page}}\", page);\n    var fullUrl = actualUrl.startsWith('http') ? actualUrl : baseurl + actualUrl;\n    \n    var get = await http.Get(fullUrl, header, true);\n    var result = get.data;\n    \n    if (!result) return \"[]\";\n    \n    result = await method(result);\n    flutterBridge.text(0, result);\n    \n    var books = [];\n    var $temp = parseHTMLSafely(result);\n    \n    $temp.find(\"li.novel-item-wrapper\").each(function() {\n      var $this = $(this);\n      var $a = $this.find(\"a\").eq(0);\n      var $img = $a.find(\"img\");\n      var $title = $this.find(\".novel-item-title\");\n      var $author = $this.find(\".novel-item-author a\");\n      var $date = $this.find(\".novel-item-date\");\n      var $nav = $this.find(\".nav\");\n      var $chapter = $this.find(\".novel-item-newest-chapter a\");\n      \n      var navText = $nav.text() || \"\";\n      var navParts = navText.split(\/\\s+\/);\n      var collection = navParts.length > 0 ? navParts[0] : \"\";\n      var view = navParts.length > 1 ? navParts[1] : \"\";\n      \n      var chapterText = $chapter.text() || \"\";\n      var dateText = $date.text() || \"\";\n      var lastChapter = cleanChapterTitle(chapterText + \"•\" + dateText);\n      \n      var bookUrl = $a.attr('href') || '';\n      if (bookUrl && !bookUrl.startsWith('http')) {\n        if (bookUrl.startsWith('\/\/')) {\n          bookUrl = 'http:' + bookUrl;\n        } else if (bookUrl.startsWith('\/')) {\n          bookUrl = baseurl + bookUrl;\n        } else {\n          bookUrl = baseurl + '\/' + bookUrl;\n        }\n      }\n      \n      var book = {\n        \"bookUrl\": bookUrl,\n        \"name\": ($title.text() || '').trim(),\n        \"author\": ($author.text() || '').trim(),\n        \"kind\": ($date.text() || '').trim(),\n        \"coverUrl\": ($img.attr('data-src') || $img.attr('src') || '') + \",\" + JSON.stringify({\"headers\": header}),\n        \"intro\": \"收藏:\" + collection + \"\\n观看:\" + view,\n        \"tocUrl\": bookUrl,\n        \"wordCount\": \"\",\n        \"type\": 0,\n        \"latestChapterTitle\": lastChapter\n      };\n      \n      if (book.bookUrl && book.name) {\n        books.push(book);\n      }\n    });\n    \n    removeHTMLSafely($temp[0]);\n    return JSON.stringify(books);\n  }\n\n  async function getloginurl() {\n    return baseurl;\n  }\n\n  async function login() {\n  }\n\n  async function pay(bookurl, url) {\n  }\n\n  async function imagedecrypt(url, image) {\n    return [];\n  }\n\n  async function shouldOverrideUrlLoading(url) {\n    return true;\n  }\n\n  async function gethelp(){\n    return \"\";\n  }\n<\/script>\n<\/html>",
    "login": false,
    "lastUpdateTime": "1770978068190"
}
广告