🔰晋江文学

https://m.jjwxc.net/channel/

autobcb_admin (12020)20小时前

vip 建议自己实现,这里只支持免费章节

二维码导入(APP尚未完成该功能)
{
    "bookSourceUrl": "https:\/\/m.jjwxc.net\/channel\/",
    "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  \n  const baseurl = \"https:\/\/m.jjwxc.net\";\n  const apiUrl = \"https:\/\/android.jjwxc.net\";\n  const appUrl = \"https:\/\/app.jjwxc.org\";\n  \n  async function search(key, page) {\n    try {\n      let headers = {\n        \"User-Agent\": await flutterBridge.getWebViewUA(),\n        \"versiontype\": \"reading\"\n      };\n      \n      let searchUrl = `${apiUrl}\/androidapi\/search?versionCode=191&keyword=${encodeURIComponent(key)}&type=1&page=${page}&searchType=8&sortMode=DESC`;\n      let result = await http.Get(searchUrl, headers, true);\n      \n      if (!result || result.statusCode !== 200) {\n        return \"[]\";\n      }\n\n      \n      let data = JSON.parse(result.data);\n      let items = data.items || [];\n      \n      \/\/ 同时获取作者搜索结果\n      let authorUrl = `${appUrl}\/androidapi\/search?versionCode=191&keyword=${encodeURIComponent(key)}&type=2&page=${page}&searchType=8&sortMode=DESC`;\n      let authorResult = await http.Get(authorUrl, headers, true);\n      if (authorResult && authorResult.statusCode === 200) {\n        let authorData = JSON.parse(authorResult.data);\n        if (authorData.items) {\n          items = items.concat(authorData.items);\n        }\n      }\n      \n      let books = [];\n      for (let item of items) {\n        if (!item.novelid) continue;\n        \n        let bookUrl = `https:\/\/www.jjwxc.net\/onebook.php?novelid=${item.novelid}`;\n        let bookInfoResult = await http.Get(bookUrl, headers, true);\n        \n        if (bookInfoResult && bookInfoResult.statusCode === 200) {\n          flutterBridge.text(0,bookInfoResult.data)\n          let html = parseHTMLSafely(bookInfoResult.data);\n          let book = {\n            name: html.find('h1').text().trim(),\n            author: html.find('span[itemprop=\"author\"]').text().trim(),\n            bookUrl: `http:\/\/app-cdn.jjwxc.net\/androidapi\/novelbasicinfo?novelId=${item.novelid}`,\n            coverUrl: html.find('.noveldefaultimage').attr('src') || `https:\/\/i9-static.jjwxc.net\/novelimage.php?novelid=${item.novelid}`,\n            intro: html.find('#novelintro').text().trim(),\n            kind: html.find('a[style=\"text-decoration:none;color: red;\"]').text().trim().replace(\/\\s\/g, ','),\n            wordCount: html.find('span[itemprop=\"wordCount\"]').text().trim().replace(\/字\/, ''),\n            status: html.find('span[itemprop=\"updataStatus\"]').text().trim()\n          };\n          \n          if (book.name) {\n            books.push(book);\n          }\n          \n          removeHTMLSafely(html);\n        }\n      }\n      \n      return JSON.stringify(books);\n    } catch (e) {\n      flutterBridge.log(\"搜索错误:\" + e.message);\n      return \"[]\";\n    }\n  }\n\n  async function info(bookurl) {\n    try {\n      let headers = {\n        \"User-Agent\": await flutterBridge.getWebViewUA(),\n        \"versiontype\": \"reading\"\n      };\n      \n      let result = await http.Get(bookurl, headers, true);\n      if (!result || result.statusCode !== 200) {\n        return JSON.stringify({});\n      }\n      \n      let data = JSON.parse(result.data);\n      let novelId = bookurl.match(\/novelId=(\\d+)\/)[1];\n      \n      \/\/ 获取最新章节信息\n      let chapterUrl = `${apiUrl}\/androidapi\/chapterList?novelId=${novelId}&more=0&whole=1`;\n      let chapterResult = await http.Get(chapterUrl, headers, true);\n      let lastChapter = \"\";\n      \n      if (chapterResult && chapterResult.statusCode === 200) {\n        let chapterData = JSON.parse(chapterResult.data);\n        let chapters = chapterData.chapterlist || [];\n        if (chapters.length > 0) {\n          let last = chapters[chapters.length - 1];\n          lastChapter = last.chaptername + ' • ' + last.chapterdate;\n        }\n      }\n      \n      \/\/ 获取其他信息\n      let otherInfoUrl = `${appUrl}\/androidapi\/getnovelOtherInfo?novelId=${novelId}&type=novelbasicinfo&versionCode=163`;\n      let otherInfoResult = await http.Get(otherInfoUrl, headers, true);\n      let novelIntro = \"\";\n      let leave = \"\";\n      \n      if (otherInfoResult && otherInfoResult.statusCode === 200) {\n        let otherData = JSON.parse(otherInfoResult.data);\n        novelIntro = otherData.novelIntro || \"\";\n        if (otherData.novelLeave) {\n          leave = '————————•————————\\n' + otherData.novelLeave.leaveDateBack + '\\n' + otherData.novelLeave.leaveContent + '\\n' + otherData.novelLeave.leaveDate;\n        }\n      }\n      \n      let book = {\n        name: data.novelName || \"\",\n        author: data.authorName || \"\",\n        coverUrl: data.originalCover || data.novelCover || `https:\/\/i9-static.jjwxc.net\/novelimage.php?novelid=${novelId}`,\n        intro: novelIntro + '\\n' + leave,\n        kind: (data.novelStep === '2' ? '已完结' : '连载中') + ',' + (data.novelClass || \"\"),\n        wordCount: data.novelSizeformat || data.novelsizeformat || \"\",\n        lastChapter: lastChapter,\n        bookUrl: bookurl,\n        tocUrl: `data:token;base64,${await flutterBridge.base64encode(`token=&novelId=${novelId}`)}`\n      };\n      \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    try {\n      let headers = {\n        \"User-Agent\": await flutterBridge.getWebViewUA(),\n        \"versiontype\": \"reading\"\n      };\n      \n      let base64Data = tocUrl.replace('data:token;base64,', '');\n      let decodedData = await flutterBridge.base64decode(base64Data);\n      let params = new URLSearchParams(decodedData);\n      let novelId = params.get('novelId');\n      let token = params.get('token') || \"\";\n      \n      let chapterUrl = `${apiUrl}\/androidapi\/chapterList?novelId=${novelId}&more=0&whole=1`;\n      if (token) {\n        chapterUrl += `&token=${token}`;\n      }\n      \n      let result = await http.Get(chapterUrl, headers, true);\n      if (!result || result.statusCode !== 200) {\n        return \"[]\";\n      }\n      \n      let data = JSON.parse(result.data);\n      let chapters = data.chapterlist || [];\n      let chapterList = [];\n      \n      for (let i = 0; i < chapters.length; i++) {\n        let chapter = chapters[i];\n        var vip =  chapter.isvip !== '0';\n        var type =  chapter.chaptertype === '1';\n        var point=chapter.point !== '0'\n        var ispay=false;\n        try{\n          var chapterMessage =!\/VIP|番外\/.test(chapter.chapterMessage)\n\n          if(chapterMessage||point){\n            ispay=true\n          }else{\n            ispay=false\n          }\n        }catch (e){\n          \n        }\n        let chapterItem = {\n          name: chapter.chaptertype === '1' ? `📖 ${chapter.chaptername} 📖` : chapter.chaptername,\n          chapterId: chapter.chaptertype === '1' ? \"\" : `${apiUrl}\/androidapi\/chapterContent?novelId=${novelId}&chapterId=${chapter.chapterid}`,\n          index: i,\n          isVolume: chapter.chaptertype === '1',\n          isVip: !type?(vip?true:false):false,\n          isPay: ispay\n        };\n        \n        chapterList.push(chapterItem);\n      }\n      \n      return JSON.stringify(chapterList);\n    } catch (e) {\n      flutterBridge.log(\"目录错误:\" + e.message);\n      return \"[]\";\n    }\n  }\n\n  async function content(url, bookurl) {\n    try {\n      let headers = {\n        \"User-Agent\": await flutterBridge.getWebViewUA(),\n        \"versiontype\": \"reading\"\n      };\n      flutterBridge.log(url)\n      let result = await http.Get(url, headers, true);\n      if (!result || result.statusCode !== 200) {\n        return \"\";\n      }\n      flutterBridge.text(3,result.data)\n      \n      let data = JSON.parse(result.data);\n      let content = data.content || \"\";\n      \n      \/\/ 处理内容格式\n      content = content.replace(\/<br\\\/>\/g, '\\n');\n      content = content.replace(\/&nbsp;\/g, ' ');\n      \n      return content;\n    } catch (e) {\n      flutterBridge.log(\"正文错误:\" + e.message);\n      return \"\";\n    }\n  }\n\n  async function getfinds() {\n    let finds = [];\n    \n    const baseApi = \"https:\/\/app.jjwxc.org\/\";\n    const topBase = `${baseApi}app.jjwxc\/android\/reading\/TopOneHundred\/getRankData?versionCode=389&type=`;\n    const token = await cache.get(\"token\") || \"\";\n    \n    \/\/ 生成日期链接\n    function generateDateLinks() {\n      let links = [];\n      links.push(`今  天::https:\/\/app-cdn.jjwxc.com\/bookstore\/favObservationByDate?day=${getFormattedDate(0)}&use_cdn=1&version=21`);\n      links.push(`明  天::https:\/\/app-cdn.jjwxc.com\/bookstore\/favObservationByDate?day=${getFormattedDate(1)}&use_cdn=1&version=21`);\n      for (let i = 1; i <= 32; i++) {\n        links.push(`前${i}天::https:\/\/app-cdn.jjwxc.com\/bookstore\/favObservationByDate?day=${getFormattedDate(-i)}&use_cdn=1&version=21`);\n      }\n      return links;\n    }\n    \n    function getFormattedDate(days) {\n      let date = new Date();\n      date.setDate(date.getDate() + days);\n      return date.toISOString().split('T')[0];\n    }\n    \n    \/\/ 基础发现项\n   \/* let all = [\n      \"🍒 其他::::1\",\n      `点击签到::${baseApi}androidapi\/signin,{\"method\":\"POST\",\"body\":\"versionCode=313&sign={{enbody = String(Date.now())+':'+String(token);encode(enbody)}}\"}`,\n      `已购书籍::https:\/\/app.jjwxc.org\/app.jjwxc\/android\/reading\/VipService\/vipNovelList?versionCode=467&offset={{100*(page-1)}}&limit=100&notconcerns=0&get_total={{page}},{\"headers\":{\"versionCode\": \"467\",\"sign\":\"{{enbody='{\\\"token\\\":\\\"'+String(token)+'\\\",\\\"time\\\":\"+parseInt(Date.now()\/1000)+}\";encode(enbody)}}\"}}`,\n      `今日限免::${baseApi}bookstore\/getFullPage?channel=novelfree`,\n      `🆕新书千字榜::data:favObservationByDate;base64,筛选,{\"type\":\"jjwx\"}::1`\n    ].concat(generateDateLinks()).concat([\n      \"⬆️ TOP 100::::1\",\n      `言情::${topBase}yq,{\"headers\":{\"versionCode\": \"389\"}}::5`,\n      `纯爱::${topBase}dm,{\"headers\":{\"versionCode\": \"389\"}}::5`,\n      `衍生::${topBase}tr,{\"headers\":{\"versionCode\": \"389\"}}::5`,\n      `无CP+::${topBase}nocp_plus,{\"headers\":{\"versionCode\": \"389\"}}::5`,\n      `百合::${topBase}bh,{\"headers\":{\"versionCode\": \"389\"}}::5`\n    ]);\n    \n    \/\/ 分类数据\n    const categories = [\"古代言情\",\"现代言情\",\"幻想现言\",\"古代穿越\",\"奇幻言情\",\"未来游戏悬疑\",\"都市纯爱\",\"现代幻纯\",\"古代纯爱\",\"百合\",\"无CP\",\"衍生无CP\",\"衍生纯爱\",\"衍生言情\",\"二次元言情\",\"多元\",\"男主无CP\",\"女主无CP\",\"未来幻纯\"];\n    *\/\n    \/\/ 解析分类\n    function parseCategory(data) {\n      const parts = data.split('::');\n      const title = parts[0];\n      const url = parts[1] || '';\n      return { title, url };\n    }\n    \n    \/\/ 添加基础发现项\n   \/* all.forEach(item => {\n      const { title, url } = parseCategory(item);\n      finds.push({ title, url });\n    });*\/\n    \n    \/\/ 添加分类\n   \/* categories.forEach((category, index) => {\n      finds.push({ title: `🌸 ${category}`, url: `${baseApi}bookstore\/getFullPage?channelBody=%7B%22natural_${index + 1}%22%3A%7B%22offset%22%3A%220%22%2C%22limit%22%3A%2225%22%7D%7D&versionCode=148` });\n    });*\/\n    \n    \/\/ 添加标签\n    const tags = [\"轻松::262::::bq\",\"爽文::137::::bq\",\"甜文::124::::bq\",\"正剧::263::::bq\",\"穿书::134::::bq\",\"强强::19::::bq\",\"系统::122::::bq\",\"成长::183::::bq\",\"重生::75::::bq\",\"快穿::125::::bq\",\"直播::142::::bq\",\"逆袭::184::::bq\",\"励志::121::::bq\",\"打脸::138::::bq\",\"女强::82::::bq\",\"异能::72::::bq\",\"基建::237::::bq\",\"救赎::329::::bq\",\"女配::96::::bq\",\"暗恋::314::::bq\",\"末世::81::::bq\",\"惊悚::29::::bq\",\"婚恋::78::::bq\",\"科幻::28::::bq\",\"炮灰::305::::bq\",\"开挂::344::::bq\",\"经营::180::::bq\",\"脑洞::273::::bq\",\"烧脑::269::::bq\",\"都市::30::::bq\",\"校园::185::::bq\",\"综漫::6::::bq\",\"星际::135::::bq\",\"日常::268::::bq\",\"柯南::241::::bq\",\"年下::21::::bq\",\"咒回::240::::bq\",\"沙雕::266::::bq\",\"美食::99::::bq\",\"生子::20::::bq\",\"群像::356::::bq\",\"萌宠::205::::bq\",\"文野::208::::bq\",\"治愈::265::::bq\",\"团宠::287::::bq\",\"爆笑::261::::bq\",\"西幻::143::::bq\",\"魔幻::27::::bq\",\"综艺::335::::bq\",\"热血::267::::bq\",\"废土::281::::bq\",\"清穿::18::::bq\",\"朝堂::178::::bq\",\"虫族::260::::bq\",\"玄学::206::::bq\",\"萌娃::302::::bq\",\"钓系::298::::bq\",\"电竞::328::::bq\",\"港风::282::::bq\",\"全息::276::::bq\",\"宫斗::74::::bq\",\"剧透::343::::bq\",\"科举::174::::bq\",\"机甲::97::::bq\",\"虐文::42::::bq\",\"网王::1::::bq\",\"原神::371::::bq\",\"古早::275::::bq\",\"师徒::292::::bq\",\"家教::12::::bq\",\"江湖::35::::bq\",\"忠犬::312::::bq\",\"武侠::11::::bq\",\"七五::8::::bq\",\"宅斗::73::::bq\",\"腹黑::290::::bq\",\"权谋::326::::bq\",\"失忆::320::::bq\",\"猎人::5::::bq\",\"洪荒::67::::bq\",\"卡牌::357::::bq\",\"职场::98::::bq\",\"御姐::310::::bq\",\"盲盒::345::::bq\",\"吐槽::270::::bq\",\"替身::286::::bq\",\"预知::346::::bq\",\"创业::330::::bq\",\"足球::368::::bq\",\"BE::317::::bq\",\"HE::318::::bq\",\"ABO::259::::bq\",\"APP::338::::bq\",\"NPC::293::::bq\",\"哨向::369::::bq\",\"田园::58::::bq\",\"中二::375::::bq\",\"群穿::352::::bq\",\"奇谭::197::::bq\",\"囤货::322::::bq\",\"位面::350::::bq\",\"学霸::309::::bq\",\"傲娇::294::::bq\",\"民国::61::::bq\",\"鬼灭::254::::bq\",\"开荒::331::::bq\",\"商战::123::::bq\",\"转生::362::::bq\",\"锦鲤::349::::bq\",\"齐神::201::::bq\",\"血族::69::::bq\",\"网红::141::::bq\",\"现实::271::::bq\",\"扶贫::324::::bq\",\"封神::215::::bq\",\"冰山::296::::bq\"];\n    \n    finds.push({ title: \"🏷 热门标签\", url: \"\" });\n    tags.forEach(tag => {\n      const parts = tag.split('::');\n      const title = parts[0];\n      const tagId = parts[1];\n      const tagType = parts[3] || \"bq\";\n      \n      let url = \"\";\n      if (tagType === \"bq\") {\n        url = `${baseApi}search\/getSearchForKeyWords?versionCode=313&offset={{(page-1)*20}}&limit=20&removetags=0&fw=0&yc=0&xx=1&sd=0&fg=0&mainview=0&fbsj=0&collectionTypes=ors&isfinish=0&sortType=0&token=${token}&bq=${tagId}`;\n      } else if (tagType === \"lx\") {\n        url = `${baseApi}search\/getSearchForKeyWords?versionCode=313&offset={{(page-1)*20}}&limit=20&removetags=0&fw=0&yc=0&xx=1&sd=0&fg=0&mainview=0&fbsj=0&collectionTypes=ors&isfinish=0&sortType=0&token=${token}&lx=${tagId}`;\n      }\n      \n      finds.push({ title, url });\n    });\n    \n    return JSON.stringify(finds);\n  }\n\n  async function find(url, page) {\n    try {\n      let headers = {\n        \"User-Agent\": await flutterBridge.getWebViewUA(),\n        \"versiontype\": \"reading\"\n      };\n      \n      let result = await http.Get(\n              url.replaceAll(\"{{(page-1)*20}}\",(page-1)*20).replaceAll(\"{{page}}\",page).replaceAll(\"{{100*(page-1)}}\",100*(page-1)),\n              headers, true);\n      if (!result || result.statusCode !== 200) {\n        return \"[]\";\n      }\n      flutterBridge.log(url.replaceAll(\"{{(page-1)*20}}\",(page-1)*20).replaceAll(\"{{page}}\",page).replaceAll(\"{{100*(page-1)}}\",100*(page-1)))\n      flutterBridge.text(0,result.data)\n\n      let data = JSON.parse(result.data);\n      let items = data.data.items || data.items;\n      flutterBridge.log(items.length)\n\n      let books = [];\n      for (let item of items) {\n        if (!item.novelid) continue;\n\n        let bookUrl = `https:\/\/www.jjwxc.net\/onebook.php?novelid=${item.novelid}`;\n        let bookInfoResult = await http.Get(bookUrl, headers, true);\n\n        if (bookInfoResult && bookInfoResult.statusCode === 200) {\n          flutterBridge.text(0,bookInfoResult.data)\n          let html = parseHTMLSafely(bookInfoResult.data);\n          let book = {\n            name: html.find('h1').text().trim(),\n            author: html.find('span[itemprop=\"author\"]').text().trim(),\n            bookUrl: `http:\/\/app-cdn.jjwxc.net\/androidapi\/novelbasicinfo?novelId=${item.novelid}`,\n            coverUrl: html.find('.noveldefaultimage').attr('src') || `https:\/\/i9-static.jjwxc.net\/novelimage.php?novelid=${item.novelid}`,\n            intro: html.find('#novelintro').text().trim(),\n            kind: html.find('a[style=\"text-decoration:none;color: red;\"]').text().trim().replace(\/\\s\/g, ','),\n            wordCount: html.find('span[itemprop=\"wordCount\"]').text().trim().replace(\/字\/, ''),\n            status: html.find('span[itemprop=\"updataStatus\"]').text().trim()\n          };\n\n          if (book.name) {\n            books.push(book);\n          }\n\n          removeHTMLSafely(html);\n        }\n      }\n\n      return JSON.stringify(books);\n    } catch (e) {\n      flutterBridge.log(\"发现错误:\" + e.message);\n      return \"[]\";\n    }\n  }\n\n  async function getloginurl() {\n    return baseurl;\n  }\n\n  async function login() {\n    \/\/ 登录逻辑\n  }\n\n  async function pay(bookurl, url) {\n    \/\/ 支付逻辑\n  }\n\n  async function imagedecrypt(url, image) {\n    \/\/ 图片解密\n  }\n\n  async function shouldOverrideUrlLoading(url) {\n    return true;\n  }\n\n  async function gethelp() {\n    return \"晋江文学城书源使用说明:\\n1. 请先登录获取token\\n2. 部分内容需要登录后才能访问\\n3. 支持搜索、发现、详情、目录、内容等功能\";\n  }\n\n<\/script>\n\n<\/html>",
    "login": true,
    "lastUpdateTime": "1773924872313"
}
广告