小黎听书

小黎听书

autobcb_admin (12020)04/11 04:11

小黎听书

二维码导入(APP尚未完成该功能)
{
    "bookSourceUrl": "小黎听书",
    "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<!-- 没用到jq请去掉-->\n <script src=\"https:\/\/cdnjs.cloudflare.com\/ajax\/libs\/crypto-js\/4.1.1\/crypto-js.min.js\"><\/script>\n<script src=\"https:\/\/vc.jd.com\/web\/js\/jquery-3.1.1.min.js\"><\/script>\n<script>\n    var isCookieJar=false;\/\/ 不需要CookieJar请修改此处\n    class FlutterJSBridge {\n        constructor() {\n            this.init(); \/\/前台webview 里必须删除这行\n        }\n\n        init() {\n            if (window.flutter_inappwebview) {\n                this.isReady = true;\n                this.CookieJar();\n            } else {\n                window.addEventListener('flutterInAppWebViewPlatformReady', () => {\n                    this.isReady = true;\n                    console.log('JSBridge初始化完成');\n                    this.CookieJar();\n                });\n            }\n        }\n\n        \/\/通知原生页面初始化完成,仅在书源和tts生效,webview请勿使用,只有通知加载成功后才允许运行,否则会一直等待加载成功\n        async CookieJar() {\n            try {\n                await window.flutter_inappwebview.callHandler('CookieJar', isCookieJar);\n            } catch (error) {\n                console.error('汇报完成准备失败:', error);\n            }\n        }\n\n        \/\/获取应用编译版本\n        async getbuildNumber() {\n            try {\n                return await window.flutter_inappwebview.callHandler('buildNumber');\n            } catch (error) {\n                return  0;\n            }\n        }\n\n        \/\/获取应用版本\n        async getversion() {\n            try {\n                return await window.flutter_inappwebview.callHandler('version');\n            } catch (error) {\n                return  \"0.0.0\";\n            }\n        }\n\n        \/\/将html转换成正文格式\n        async htmlToText(str) {\n            try {\n                return await window.flutter_inappwebview.callHandler('htmlToText',str);\n            } catch (error) {\n                return  \"\";\n            }\n        }\n\n        \/\/将简体字转成繁体字\n        async toTraditional(str) {\n            try {\n                return await window.flutter_inappwebview.callHandler('toTraditional',str);\n            } catch (error) {\n                return  \"\";\n            }\n        }\n\n\n        \/\/将繁体字转成简体字\n        async toSimplified(str) {\n            try {\n                return await window.flutter_inappwebview.callHandler('toSimplified',str);\n            } catch (error) {\n                return  \"\";\n            }\n        }\n\n        \/\/播放朗读引擎仅tts源生效\n        async voice() {\n            try {\n                return await window.flutter_inappwebview.callHandler('voice');\n            } catch (error) {\n                return  \"\";\n            }\n        }\n\n\n        \/\/获取设备唯一id\n        async getDeviceid() {\n            try {\n                return await window.flutter_inappwebview.callHandler('id');\n            } catch (error) {\n                return  \"\";\n            }\n        }\n\n        \/\/获取设备平台 此处返回 windows、macos、ios、ohos、android\n        async getDevice() {\n            try {\n                return await window.flutter_inappwebview.callHandler('device');\n            } catch (error) {\n                return  \"\";\n            }\n        }\n\n        \/\/获取轻悦时光登录用户名,没登录返回为空\n        async getLoginUser() {\n            try {\n                return await window.flutter_inappwebview.callHandler('getLoginUser');\n            } catch (error) {\n                return  \"\";\n            }\n        }\n\n        \/\/输出日志,前台webview请勿使用\n        \/\/str 为 String\n        async log(str) {\n            try {\n                return await window.flutter_inappwebview.callHandler('log',str);\n            } catch (error) {\n                return  false;\n            }\n        }\n\n        \/\/书源调试时可输出 html 代码到前台\n        \/\/type 0 搜索源码 , 1详情源码 ,2目录源码 ,3正文源码\n        \/\/str 为 String\n        \/\/type 为int\n        async text(type,str) {\n            try {\n                return await window.flutter_inappwebview.callHandler('text',type,str);\n            } catch (error) {\n                return  false;\n            }\n        }\n\n        \/\/toast弹窗,显示3秒\n        \/\/str 为 String\n        async showToast(str) {\n            try {\n                return await window.flutter_inappwebview.callHandler('showToast',str);\n            } catch (error) {\n                return  false;\n            }\n        }\n\n        \/\/长toast弹窗,显示10秒\n        \/\/str 为 String\n        async showLongToast(str) {\n            try {\n                return await window.flutter_inappwebview.callHandler('showLongToast',str);\n            } catch (error) {\n                return  false;\n            }\n        }\n\n        \/\/webview 里禁止使用,webview请使用js获取ua (navigator.userAgent)\n        \/\/获取默认ua\n        async getWebViewUA() {\n            try {\n                return await window.flutter_inappwebview.callHandler('getWebViewUA');\n            } catch (error) {\n                return  \"\";\n            }\n        }\n\n        \/\/通过url打开外部应用\n        \/\/url 为 String\n        async openurl(url) {\n            try {\n                return await window.flutter_inappwebview.callHandler('openurl',url,\"\");\n            } catch (error) {\n                return  false;\n            }\n        }\n\n        \/\/通过url打开外部应用并附带mimeType\n        \/\/url 为 String\n        \/\/mimeType 为 String\n        async openurlwithMimeType(url,mimeType) {\n            try {\n                return await window.flutter_inappwebview.callHandler('openurl',url,mimeType);\n            } catch (error) {\n                return  false;\n            }\n        }\n\n        \/**\n         * 使用webView访问网络\n         * @param html 直接用webView载入的html, 如果html为空直接访问url\n         * @param url html内如果有相对路径的资源不传入url访问不了\n         * @param js 用来取返回值的js语句, 没有就返回整个源代码\n         * @param body 当参数不为空的时候,会以post请求,此时请务必在 header 中带上content-type\n         * @param header 请求的header头,此参数必须是json字符串\n         * @return 返回js获取的内容\n         *\/\n        async webview(url,js,html,body,header) {\n            try {\n                return await window.flutter_inappwebview.callHandler('webview',url,js,html,body,header,\"\",\"\");\n            } catch (error) {\n                return  \"\";\n            }\n        }\n\n        \/**\n         * overrideUrlRegex 为正则表达式\n         * 使用方法和上面的一样\n         * 但返回的内容为正则到的内容,如果无法正则到则返回 js 获取的内容,如果 js 为空则返回页面 html\n         *\/\n        async webViewGetOverrideUrl(url,js,html,body,header,overrideUrlRegex) {\n            try {\n                return await window.flutter_inappwebview.callHandler('webview',url,js,html,body,header,overrideUrlRegex,\"\");\n            } catch (error) {\n                return  \"\";\n            }\n        }\n\n        \/**\n         * 使用webView获取资源url\n         * urlregex 为正则表达式\n         * 使用方法和上面的一样\n         * 但返回的内容为正则到的内容,如果无法正则到则返回 js 获取的内容,如果 js 为空则返回页面 html\n         *\/\n        async webViewGetSource(url,js,html,body,header,urlregex) {\n            try {\n                return await window.flutter_inappwebview.callHandler('webview',url,js,html,body,header,\"\",urlregex);\n            } catch (error) {\n                return  \"\";\n            }\n        }\n\n        \/**\n         * 使用webView拦截 ajax\n         * ajaxregex 为正则表达式,通过 ajax 匹配 path\n         * 匹配成功返回 ajax 的结果 失败返回 html\n         *\/\n        async webViewGetAjax(url,html,body,header,ajaxregex) {\n            try {\n                return await window.flutter_inappwebview.callHandler('webviewajax',url,html,body,header,ajaxregex);\n            } catch (error) {\n                return  \"\";\n            }\n        }\n\n\n\n        \/**\n         * 启动前台 webview 访问链接并获取结束时的 html,可用于手工过盾\n         * @param url 网址\n         * @param title 标题\n         * @param header 请求的header头,此参数必须是json字符串\n         * @return 返回网页的内容\n         *\/\n        async startBrowser(url,title,header) {\n            try {\n                return await window.flutter_inappwebview.callHandler('startBrowser',url,title,header);\n            } catch (error) {\n                return  \"\";\n            }\n        }\n\n        \/**\n         * 启动前台 webview 并对每次打开的 url 进行拦截\n         * @param url 网址\n         * @param title 标题\n         * @param header 请求的header头,此参数必须是json字符串\n         *\/\n        async startBrowserWithShouldOverrideUrlLoading(url,title,header) {\n            try {\n                return await window.flutter_inappwebview.callHandler('startBrowserWithShouldOverrideUrlLoading',url,title,header);\n            } catch (error) {\n                return  \"\";\n            }\n        }\n\n        \/\/专门为段评设置的半屏显示,不返回任何东西\n        async startBrowserDp(url,title) {\n            try {\n                return await window.flutter_inappwebview.callHandler('startBrowserDp',url,title);\n            } catch (error) {\n                return  \"\";\n            }\n        }\n\n        \/\/仅前台webview可以使用,返回按钮,返回上一个页面\n        async back() {\n            try {\n                return await window.flutter_inappwebview.callHandler('back');\n            } catch (error) {\n                return  false;\n            }\n        }\n\n        \/\/将 utf8字符串转到 gbk 并 url 编码\n        async utf8ToGbkUrlEncoded(str) {\n            try {\n                return await window.flutter_inappwebview.callHandler('utf8ToGbkUrlEncoded',str);\n            } catch (error) {\n                return  \"\";\n            }\n        }\n\n        \/*\n        * @param str为图片链接\n        * @param header 请求的header头,此参数必须是json字符串\n        * 此函数是让用户输入图片中的验证码,当链接为空则直接让用户输入验证码\n        *\/\n        async getVerificationCode(str,header) {\n            try {\n                return await window.flutter_inappwebview.callHandler('getVerificationCode',str,header);\n            } catch (error) {\n                return  \"\";\n            }\n        }\n\n        \/\/提交内容bookUrl,我会调用书源 info 函数来获取这本书的信息\n        async addbook(bookUrl) {\n            try {\n                return await window.flutter_inappwebview.callHandler('addbook',bookUrl);\n            } catch (error) {\n                return  \"\";\n            }\n        }\n\n\n        \/\/获取书本当前阅读章节index\n        async getdurChapterIndex(bookUrl) {\n            try {\n                return await window.flutter_inappwebview.callHandler('getdurChapterIndex',bookUrl);\n            } catch (error) {\n                return  0;\n            }\n        }\n\n        \/\/utf8 字符串转base64\n        async base64encode(str) {\n            try {\n                return await window.flutter_inappwebview.callHandler('base64encode',str);\n            } catch (error) {\n                return  \"\";\n            }\n        }\n\n        \/\/base64 转utf8字符串\n        async base64decode(str) {\n            try {\n                return await window.flutter_inappwebview.callHandler('base64decode',str);\n            } catch (error) {\n                return  \"\";\n            }\n        }\n\n\n\n    }\n\n    \/\/webview下isCookieJar必定true 会自动处理cookie\n    \/\/以下提交的url,headers,body 都必须为字符串,headers必须为json字符串\n    \/\/当followRedirects 为 false 时不处理重定向,当为 true 时会自动处理重定向 ,如不明白用途直接用 true 最佳\n    \/\/ 以下所有参数除当followRedirects外均为 String\n    \/\/ 如果需要使用http2协议 请在url 前添加 http2:\/\/ ,例如 http2:\/\/baidu.com\n    \/\/ 如果https一直被盾拦截 ,可以使用https2协议\n    class Http {\n        constructor() {\n            \/*\n             * 速率限制配置\n             * requestTimestamps: 存储请求时间戳的数组\n             * rateLimit: 速率限制,单位时间内最大请求数\n             * rateLimitWindow: 速率限制窗口,单位毫秒\n             *\/\n            this.open = false; \/\/ 是否开启速率限制\n            this.requestTimestamps = []; \/\/ 存储请求时间戳的数组\n            this.rateLimit = 5; \/\/ 速率限制,1000毫秒内最多5次请求\n            this.rateLimitWindow = 1000; \/\/ 速率限制窗口,1000毫秒\n        }\n\n        \/*\n         * 检查速率限制\n         * 实现方法:\n         * 1. 获取当前时间戳\n         * 2. 过滤掉超出时间窗口的时间戳\n         * 3. 检查是否超过速率限制\n         * 4. 如果超过限制,计算需要等待的时间并等待\n         * 5. 递归检查速率限制\n         * 6. 将当前时间戳添加到数组中\n         *\/\n        async checkRateLimit() {\n            if(!this.open) return;\n            const now = Date.now();\n            \/\/ 过滤掉超出时间窗口的时间戳\n            this.requestTimestamps = this.requestTimestamps.filter(timestamp => now - timestamp < this.rateLimitWindow);\n            \/\/ 检查是否超过速率限制\n            if (this.requestTimestamps.length >= this.rateLimit) {\n                \/\/ 计算需要等待的时间\n                const oldestTimestamp = this.requestTimestamps[0];\n                const waitTime = this.rateLimitWindow - (now - oldestTimestamp);\n                \/\/ 等待到速率限制可用\n                await new Promise(resolve => setTimeout(resolve, waitTime));\n                \/\/ 递归检查速率限制\n                return this.checkRateLimit();\n            }\n            \/\/ 将当前时间戳添加到数组中\n            this.requestTimestamps.push(now);\n        }\n\n\n        \/*\n         * 通用返回字段\n         * method post get 或者 head\n         * body 请求返回后的字节的 base64\n         * headers  map<String,List<String>> 可通过headers[\"\"]来或者\n         * statusCode 状态码\n         * statusMessage\n         * data 返回后的字节 格式化后的内容\n         *\/\n        async Get(url,headers,followRedirects) {\n            try {\n                await this.checkRateLimit();\n                return await window.flutter_inappwebview.callHandler('http',\"get\",url,\"\",JSON.stringify(headers),followRedirects,\"\");\n            } catch (error) {\n                return  null;\n            }\n        }\n\n        async Head(url,headers,followRedirects) {\n            try {\n                await this.checkRateLimit();\n                return await window.flutter_inappwebview.callHandler('http',\"head\",url,\"\",JSON.stringify(headers),followRedirects,\"\");\n            } catch (error) {\n                return  null;\n            }\n        }\n\n\n        async Post(url,headers,body,contenttype,followRedirects) {\n            try {\n                await this.checkRateLimit();\n                return await window.flutter_inappwebview.callHandler('http',\"post\",url,body,JSON.stringify(headers),followRedirects,contenttype);\n            } catch (error) {\n                return  null;\n            }\n        }\n    }\n\n    class Cache {\n        constructor() {}\n        async get(key) {\n            try {\n                return await window.flutter_inappwebview.callHandler('cache.get',key);\n            } catch (error) {\n                return  null;\n            }\n        }\n\n        async set(key,value) {\n            try {\n                return await window.flutter_inappwebview.callHandler('cache.set',key,value);\n            } catch (error) {\n                return  null;\n            }\n        }\n\n        async remove(key) {\n            try {\n                return await window.flutter_inappwebview.callHandler('cache.remove',key);\n            } catch (error) {\n                return  null;\n            }\n        }\n\n        \/\/如果登录为弹窗格式的,里面输入框输入的内容可以通过这个函数获取,默认返回的json格式或者为空,需要自行转换\n        async getLoginInfo(){\n            return await  this.get(\"LoginInfo\")\n        }\n\n        \/\/将修改后的弹窗输入内容报错 ,必须 JSON.stringify,不然会出错\n        async putLoginInfo(info){\n            return await  this.set(\"LoginInfo\",info)\n        }\n\n        \/\/获取书本变量\n        async getbookVariable(bookurl){\n            return await  this.get(bookurl)\n        }\n\n        \/\/写入书本变量\n        async setbookVariable(bookurl,value){\n            return await  this.set(bookurl,value)\n        }\n    }\n\n    class Cookie {\n        constructor() {}\n\n        \/\/通过url获取当前url的所有cookie\n        async get(url) {\n            try {\n                return await window.flutter_inappwebview.callHandler('cookie.get',url);\n            } catch (error) {\n                return  null;\n            }\n        }\n\n        \/\/通过url删除当前url的所有cookie\n        async remove(url) {\n            try {\n                return await window.flutter_inappwebview.callHandler('cookie.remove',url);\n            } catch (error) {\n                return  null;\n            }\n        }\n\n\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 = \"小黎听书\";\n    \n    \/\/ 加密相关函数\n    function md5(str) {\n        return CryptoJS.MD5(str).toString(CryptoJS.enc.Hex);\n    }\n    \n    \/\/ DES解密函数\n    function decode(result) {\n        try {\n            let data = JSON.parse(result).body.encData;\n            let key = CryptoJS.enc.Utf8.parse(\"rThYzbEp\");\n            let decrypted = CryptoJS.DES.decrypt(\n                data, \n                key, \n                {\n                    mode: CryptoJS.mode.ECB,\n                    padding: CryptoJS.pad.Pkcs7\n                }\n            );\n            return decrypted.toString(CryptoJS.enc.Utf8);\n        } catch (e) {\n            flutterBridge.log(\"DES解密错误:\" + e.message);\n            return result;\n        }\n    }\n    \n    function getHeader() {\n        let t13 = String(Math.floor(new Date().getTime()))\n        return JSON.stringify({\n            \"Content-Type\": \"application\/json\",\n            \"appId\": \"com.funny.syzs\",\n            \"appVersion\": \"1\",\n            \"time\": t13,\n            \"sign\": md5(\"qwer\" + t13 + \"com.funny.syzs\" + \"1\" + \"rewq\")\n        });\n    }\n    \n    \/\/ 获取应用配置\n    async function appconfig() {\n        try {\n            let appService = await http.Get(\"https:\/\/umeng-log-v5.gz.bcebos.com\/cloud\/app\/com.funny.syzs\/appService.json\", getHeader(), true);\n            if (appService && appService.statusCode === 200) {\n                var html1=JSON.parse(appService.data);\n                if(html1.code == 1001){\n                    await cache.set(\"appService\", decode(appService.data));\n                }else{\n                    await cache.set(\"appService\", appService.data);\n                }\n            }\n            let update = await http.Get(\"https:\/\/umeng-log-v5.gz.bcebos.com\/cloud\/app\/com.funny.syzs\/update.json\", getHeader(), true);\n            if (update && update.statusCode === 200) {\n                var html1=JSON.parse(update.data);\n                if(html1.code == 1001){\n                    await cache.set(\"update\", decode(update.data));\n                }else{\n                    await cache.set(\"update\", update.data);\n                }\n            }\n        } catch (e) {\n            flutterBridge.log(\"appconfig错误:\" + e.message);\n        }\n    }\n    \n    \/\/ 获取服务配置\n    async function getS() {\n        let appService = await cache.get(\"appService\");\n        if (!appService) {\n            await appconfig();\n            appService = await cache.get(\"appService\");\n        }\n        return JSON.parse(appService || \"{}\");\n    }\n    \n    \/\/ 获取更新配置\n    async function getU() {\n        let update = await cache.get(\"update\");\n        if (!update) {\n            await appconfig();\n            update = await cache.get(\"update\");\n        }\n        return JSON.parse(update || \"{}\");\n    }\n    \n    \/\/ 获取API数据\n    async function get1(path) {\n        try {\n            let appService = await getS();\n            let url = path;\n            \n            if (!path.includes(\"http\")) {\n                url = appService.businessStaticAddress + path;\n                let result = await http.Get(url, getHeader(), true);\n                if (!result || result.statusCode !== 200) {\n                    url = appService.businessDynamicAddress + path;\n                }\n            }\n\n            let result = await http.Get(url, getHeader(), true);\n            if (!result || result.statusCode !== 200) {\n                return null;\n            }\n            \n            let data = JSON.parse(result.data);\n            if (data.code == 1001) {\n                \/\/ 需要解密\n                return decode(result.data);\n            }\n            return result.data;\n        } catch (e) {\n            flutterBridge.log(\"get1错误:\" + e.message);\n            return null;\n        }\n    }\n    \n    async function search(key, page) {\n        try {\n            await appconfig();\n            let appService = await getS();\n            let searchUrl = appService.searchAddress + \"api\/search\/v1\/albums\";\n\n            \n            \n            let body = JSON.stringify({\"key\": key, \"page\": page});\n            let result = await http.Post(searchUrl, getHeader(), body, \"application\/json\", true);\n            \n            if (!result || result.statusCode !== 200) {\n                return \"[]\";\n            }\n\n            \n            let data = JSON.parse(\tdecode(result.data));\n            let books = [];\n            \n            let items = data.items || [];\n            for (let item of items) {\n                let book = {\n                    name: (item.title || \"\").replace(\"<word>\",\"\").replace(\"<\/word>\",\"\").replace(\"<word>\",\"\").replace(\"<\/word>\",\"\"),\n                    author: (item.announcerName || (item.announcers ? item.announcers.join(\",\") : \"\")).replace(\"<word>\",\"\").replace(\"<\/word>\",\"\").replace(\"<word>\",\"\").replace(\"<\/word>\",\"\"),\n                    intro: item.desc || \"\",\n                    kind: (item.category || item.categories ? (item.category || item.categories) + \",\" : \"\") + (item.finishStatus == \"0\" ? \"连载\" : \"完结\"),\n                    bookUrl: `${await flutterBridge.base64encode(item.albumId || item.id)}`,\n                    coverUrl: appService.staticResource + (item.coverPath || \"\"),\n                    lastChapter: item.latestTrackTitle || \"\",\n                    type:1\n                };\n                books.push(book);\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 id = await flutterBridge.base64decode(bookurl);\n\n            let data = await get1(\"api\/business\/v1\/album\/info\/\" + id + \".json\");\n            \n            if (!data) {\n                return JSON.stringify({});\n            }\n\n            flutterBridge.text(1,data)\n            \n            let bookInfo = JSON.parse(data);\n            let appService = await getS();\n            var categories=\"\";\n            if(bookInfo.category ){\n                try{\n                    bookInfo.category.forEach((e)=>{\n                        categories=categories+e.categoryName+\",\";\n                    })\n                }catch (e){\n\n                }\n            }\n            categories=categories+(bookInfo.finishStatus === \"0\" ? \"连载\" : \"完结\");\n            author=\"\";\n            if(bookInfo.announcerNam){\n                author= bookInfo.announcerNam\n            }\n            if(bookInfo.announcers){\n                try{\n                    bookInfo.announcers.forEach((e)=>{\n                        if(author == \"\"){\n                            author+=e.announcerName ;\n                        }else{\n                            author=author+\",\"+e.announcerName ;\n                        }\n\n                    })\n                }catch (e){\n\n                }\n            }\n            let book = {\n                name: bookInfo.title || \"\",\n                author:author,\n                intro: bookInfo.desc || \"\",\n                kind: categories,\n                lastChapter: bookInfo.latestTrackTitle || \"\",\n                coverUrl: appService.staticResource + (bookInfo.coverPath || \"\"),\n                tocUrl: `${await flutterBridge.base64encode(bookInfo.id || \"\")}`,\n                bookUrl: bookurl,\n                type:1\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 id = await flutterBridge.base64decode(tocUrl);\n            let data = await get1(\"api\/business\/v1\/album\/tracks\/\" + id + \".json\");\n            \n            if (!data) {\n                return \"[]\";\n            }\n            flutterBridge.text(2,data)\n            let chapters = JSON.parse(data);\n            let chapterList = [];\n            \n            for (let i = 0; i < chapters.length; i++) {\n                let chapter = chapters[i];\n                let chapterItem = {\n                    name: chapter.title || \"\",\n                    chapterId: `${await flutterBridge.base64encode(chapter.albumId + \"❌\" + chapter.id)}`,\n                    index: i\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 decodedData = await flutterBridge.base64decode(url);\n            let [albumId, id] = decodedData.split(\"❌\");\n            \n            let data = await get1(\"api\/business\/v1\/album\/analysis\/\" + albumId + \"\/\" + id + \".json\");\n            if (!data) {\n                return \"\";\n            }\n            \n            let html1 = JSON.parse(data);\n            let appService = await getS();\n            let update = await getU();\n            \n            if (html1.playType === \"self\") {\n                let t13 = String(Math.floor(Date.now()));\n                let t11 = t13.substring(0, 11);\n                let sign1 = \"$$\" + md5(md5(\"bOLpce2OsGWj7cJu6VXEscK7QyXawZzg\" + html1.accessPath)) + \"$$\";\n                let sign2 = \"$$\" + md5(md5(\"i*Tc^pJ~Wp^.sCNtkhIMZcF*275o4Pvk\" + html1.accessPath)) + \"$$\";\n                \n                let header = {\n                    \"Icy-MetaData\": \"1\",\n                    \"Accept-Encoding\": \"identity\",\n                    \"initStatus\": \"true\",\n                    \"version\": update.latestVersionName || \"1.0.0\",\n                    \"psign\": \"9f05e35446d7d6d886ecafdf7d689db2\",\n                    \"userId\": \"b012b709e395a2da6d5774fc586cf842\",\n                    \"clientTime\": t11 + \"99\",\n                    \"appChannel\": \"normal\",\n                    \"versionCode\": update.latestVersionCode || \"1\",\n                    \"sign2\": sign2,\n                    \"rewardTime\": \"0\",\n                    \"sign1\": sign1,\n                    \"sign3\": \"99\" + \"35288721\" + t11,\n                    \"freeTime\": t13,\n                    \"appKey\": \"com.funny.syzs\",\n                    \"sTime\": t13,\n                    \"User-Agent\": \"okhttp\/4.11.0\"\n                };\n                \n                return appService.trackResource + html1.accessPath + \",\" + JSON.stringify({\"headers\": header});\n            } else if (html1.playType === \"lanren\") {\n                let extras = html1.extras;\n                let apiUrl = extras.apiUrl2;\n                let lanrenUrl = apiUrl + \"\/ajax\/getPlayPath?entityId=\" + html1.sourceAlbumId + \"&entityType=3&opType=1&sections=[\" + html1.sourceExtras.section + \"]&type=0&section=\" + html1.sourceExtras.section + \"&id=\" + html1.sourceTrackId;\n                \n                let result = await http.Get(lanrenUrl, {}, true);\n                if (result && result.statusCode === 200) {\n                    let lanrenData = JSON.parse(result.data);\n                    if (lanrenData.list && lanrenData.list.length > 0) {\n                        return lanrenData.list[0].path;\n                    }\n                }\n            }\n            \n            return \"\";\n        } catch (e) {\n            flutterBridge.log(\"正文错误:\" + e.message);\n            return \"\";\n        }\n    }\n\n    async function getfinds() {\n        let finds = [];\n        \n        finds.push({ title: \"首页\", url: await flutterBridge.base64encode(\"home❌1\") });\n        finds.push({ title: \"有声小说\", url: await flutterBridge.base64encode(\"6603a267516ee217034fb43bad787bf6❌1\") });\n        finds.push({ title: \"相声评书\", url: await flutterBridge.base64encode(\"779df3afa5521d3fa13575919e3cbfaa❌1\") });\n        finds.push({ title: \"脱口秀\", url: await flutterBridge.base64encode(\"75f156a8d034bf065dab01fe38368d2e❌1\") });\n        finds.push({ title: \"广播剧\", url: await flutterBridge.base64encode(\"b9aa4406aacb55ae5602c053d02bafde❌1\") });\n        finds.push({ title: \"历史\", url: await flutterBridge.base64encode(\"c827d8dbb76ce33219f64c6749893159❌1\") });\n        finds.push({ title: \"悬疑\", url: await flutterBridge.base64encode(\"7c1753ed36d64f7fabf6dd43bf165075❌1\") });\n        finds.push({ title: \"其他\", url: await flutterBridge.base64encode(\"0d98c74797e49d00bcc4c17c9d557a2b❌1\") });\n        finds.push({ title: \"更新\", url: await flutterBridge.base64encode(\"update❌1\") });\n        finds.push({ title: \"新书\", url: await flutterBridge.base64encode(\"new❌1\") });\n        \n        return JSON.stringify(finds);\n    }\n\n    async function find(url, page) {\n        try {\n            let decodedData = await flutterBridge.base64decode(url);\n            let [type, pageNum] = decodedData.split(\"❌\");\n            \n            let data;\n            if (type === \"home\") {\n                data = await get1(\"api\/business\/v1\/lib\/home\/tab\/01k4eymbkwwdq6rwrc1j26hdx1\/groups.json\");\n            } else if (type === \"update\") {\n                data = await get1(\"api\/business\/v1\/lib\/update\/album\/\" + pageNum + \".json\");\n            } else if (type === \"new\") {\n                data = await get1(\"api\/business\/v1\/lib\/new\/album\/\" + pageNum + \".json\");\n            } else {\n                data = await get1(\"api\/business\/v1\/album\/list\/category\/\" + type + \"\/\" + pageNum + \".json\");\n            }\n            \n            if (!data) {\n                return \"[]\";\n            }\n            flutterBridge.text(0,data)\n            \n            let jsonData = JSON.parse(data);\n            let books = [];\n            let items = jsonData.items || (jsonData[0] && jsonData[0].albums ? jsonData[0].albums : []);\n            \n            let appService = await getS();\n            \n            for (let item of items) {\n                let book = {\n                    name: item.title || \"\",\n                    author: item.announcers,\n                    intro: item.desc || \"\",\n                    kind: (item.category || item.categories ? (item.category || item.categories) + \",\" : \"\") + (item.finishStatus == \"0\" ? \"连载\" : \"完结\"),\n                    bookUrl: `${await flutterBridge.base64encode(item.albumId || item.id)}`,\n                    coverUrl: appService.staticResource + (item.coverPath || \"\"),\n                    lastChapter: item.latestTrackTitle || \"\",\n                    type:1\n                };\n                books.push(book);\n            }\n            \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来源:小黎听书APP\\n说明:有声书书源,包含有声小说、相声评书、脱口秀、广播剧等内容\";\n    }\n\n<\/script>\n\n<\/html>\n",
    "login": false,
    "lastUpdateTime": "1775851905523"
}
广告