海洋听书
http://www.ychy.org
autobcb_admin (12020)3小时前
似乎是喜马拉雅的资源
{
"bookSourceUrl": "http:\/\/www.ychy.org",
"bookSourceName": "海洋听书",
"enabledExplore": true,
"enabled": true,
"bookSourceGroup": "",
"html": "<!DOCTYPE html>\n<html lang=\"en\">\n<head>\n <meta charset=\"UTF-8\">\n <title>海洋听书<\/title>\n<\/head>\n<body>\n\n<\/body>\n<script src=\"https:\/\/vc.jd.com\/web\/js\/jquery-3.1.1.min.js\"><\/script>\n<!--如果要引入外部 js 必须在书源代码的上面-->\n<script>\n var isCookieJar=true;\/\/ 不需要CookieJar请修改此处\n class FlutterJSBridge {\n constructor() {\n this.isReady = false;\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 \/\/获取设备唯一 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 async log(str) {\n try {\n return await window.flutter_inappwebview.callHandler('log',str);\n } catch (error) {\n return false;\n }\n }\n\n async text(type,str) {\n try {\n return await window.flutter_inappwebview.callHandler('text',type,str);\n } catch (error) {\n return false;\n }\n }\n\n \/\/toast弹窗\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 \/\/获取默认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 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 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 * 使用webView获取跳转url\n * overrideUrlRegex 为正则表达式\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 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 \/**\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 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 async utf8ToGbkUrlEncoded(str) {\n try {\n return await window.flutter_inappwebview.callHandler('utf8ToGbkUrlEncoded',str);\n } catch (error) {\n return \"\";\n }\n }\n\n async getVerificationCode(str,header) {\n try {\n return await window.flutter_inappwebview.callHandler('getVerificationCode',str,header);\n } catch (error) {\n return \"\";\n }\n }\n\n }\n\n \/\/webview请勿使用\n class Http {\n constructor() {}\n\n \/\/以下提交的url,headers,body 都必须为字符串,headers必须为json字符串\n\n \/\/当followRedirects 为 false 时不处理重定向,当为 true 时会自动处理重定向 ,如不明白用途直接用 true 最佳\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 \/\/注意post 请求是务必在 header 中带上content-type\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 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 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 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\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<script>\n const flutterBridge = new FlutterJSBridge();\n const cache = new Cache();\n const http = new Http();\n const cookie = new Cookie();\n var baseurl=\"http:\/\/www.ychy.org\"\n var contenttype=\"application\/x-www-form-urlencoded\"\n var header={\n \"User-Agent\": \"Mozilla\/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit\/537.36 (KHTML, like Gecko) Chrome\/142.0.0.0 Safari\/537.36 Edg\/142.0.0.0\",\n \"Referer\" : baseurl\n };\n async function search(key,page) {\n if(page > 1){\n return \"[]\";\n }\n var url=baseurl+\"\/so.asp\";\n var body=\"searchword=\"+(await flutterBridge.utf8ToGbkUrlEncoded(key))+\"&Submit=++\"\n var get= await http.Post(url,JSON.stringify(header),body,contenttype,true);\n flutterBridge.text(0,get.data )\n var books=[];\n var html = $.parseHTML( get.data );\n $(html).find(\".ItemListbody\").find(\"li\").each(function(index) {\n var $element = $(this);\n var img = $element.find(\"img\").attr('src');\n var book={\n \"bookUrl\":baseurl+$element.find(\"a\").eq(1).attr('href'),\n \"name\": $element.find(\"span\").text(),\n \"author\": $element.find(\"dd\").eq(1).text(),\n \"kind\":$element.find(\"dd\").eq(3).text(),\n \"coverUrl\":img,\n \"intro\":$element.find(\"dd\").eq(9).text(),\n \"tocUrl\":baseurl+$element.find(\"a\").eq(1).attr('href'),\n \"wordCount\":\"\",\n \"type\":1,\n \"latestChapterTitle\":$element.find(\"dd\").eq(2).text(),\n }\n books.push(book);\n });\n return JSON.stringify(books);\n }\n\n async function info(bookurl) {\n var get=await http.Get(bookurl,JSON.stringify(header),true);\n flutterBridge.text(1,get.data )\n var html = $.parseHTML( get.data );\n var $html=$(html)\n var book={\n \"bookUrl\":bookurl,\n \"name\": $html.find(\".content_title\").find(\"h1\").text(),\n \"author\": $html.find(\".content_center\").find(\"a\").eq(1).text(),\n \"kind\": $html.find(\".content_center\").find(\"a\").eq(2).text(),\n \"coverUrl\": $html.find(\".content_right\").find(\"img\").attr('src'),\n \"intro\":$html.find(\".txtbox\").html(),\n \"tocUrl\":bookurl,\n \"wordCount\":\"\",\n \"type\":1,\n \"latestChapterTitle\":$html.find(\".playlist\").find(\"a\").eq($html.find(\".playlist\").find(\"a\").length-1).text(),\n }\n return JSON.stringify(book);\n }\n\n async function chapter(tocUrl) {\n var get = await http.Get(tocUrl, JSON.stringify(header), true);\n flutterBridge.text(2,get.data )\n var chapters=[];\n var html = $.parseHTML( get.data );\n $(html).find(\".playlist\").find(\"li\").each(function(index) {\n var chapter={\n \"name\" :$(this).find(\"a\").text(),\n \"chapterId\":baseurl+$(this).find(\"a\").attr('href'),\n \"index\" :index,\n \"isPay\":false,\n \"isVip\":false,\n \"isVolume\":false,\n \"tag\":\"\"\n };\n chapters.push(chapter);\n });\n return JSON.stringify(chapters);\n }\n\n async function content(url) {\n var re=await flutterBridge.webViewGetSource(url,\"\",\"\",\"\",JSON.stringify(header),\".*\\\\.(mp3|m4a).*\");\n flutterBridge.text(3,re )\n if(re.length > 1000){\n flutterBridge.log(\"webview 正则资源失败,可能是鸿蒙 bug,没关系我还有 jq\")\n var html = $.parseHTML( re);\n return $(html).find(\"audio\").attr('src');\n }\n return re;\n }\n\n async function getfinds() {\n var sort=[{\n title: \"发现\",\n url: \"\",\n type: 0,\n }];\n \n var str=\"盗墓探险::\/list\/45-{{page}}.html\\n\" +\n \"官场刑侦::\/list\/14-{{page}}.html\\n\" +\n \"网络玄幻::\/list\/52-{{page}}.html\\n\" +\n \"历史军事::\/list\/15-{{page}}.html\\n\" +\n \"人物传记::\/list\/16-{{page}}.html\\n\" +\n \"儿童读物::\/list\/4-{{page}}.html\\n\" +\n \"恐怖悬疑::\/list\/17-{{page}}.html\\n\" +\n \"都市言情::\/list\/13-{{page}}.html\\n\" +\n \"职场商战::\/list\/81-{{page}}.html\\n\" +\n \"传统武侠::\/list\/12-{{page}}.html\\n\" +\n \"相声戏曲::\/list\/7-{{page}}.html\\n\" +\n \"百家讲坛::\/list\/32-{{page}}.html\\n\" +\n \"经典评书::\/list\/3-{{page}}.html\\n\" +\n \" 广 播 剧 ::\/list\/18-{{page}}.html\\n\" +\n \"文艺戏曲::\/list\/53-{{page}}.html\";\n \n \/\/ 解析分类字符串\n var categories = str.split('\\n');\n categories.forEach(function(category) {\n if (category.trim() !== '') {\n \/\/ 按::分割获取标题和URL\n var parts = category.split('::');\n if (parts.length === 2) {\n \/\/ 添加到sort数组,去除标题中的多余空格\n sort.push({\n title: parts[0].trim(),\n url: parts[1].trim(),\n type: 0 \/\/ 保持type为0,与第一个元素一致\n });\n }\n }\n });\n\n return JSON.stringify(sort);\n }\n\n async function find(url,page) {\n var u=baseurl+url.replace(\"{{page}}\",page)\n flutterBridge.log(u)\n var get=await http.Get(u,JSON.stringify(header),true);\n flutterBridge.text(0,get.data )\n var html = $.parseHTML( get.data );\n var books=[];\n try {\n $(html).find(\".bx_channel_test\").each(function(index) {\n var $element = $(this);\n var img = $element.find(\"img\").attr('src');\n var book={\n \"bookUrl\":baseurl+$element.find(\"a\").eq(0).attr('href'),\n \"name\": $element.find(\".bx_channel_testname\").find(\"a\").text(),\n \"author\": $element.find(\"p\").eq(2).text().replaceAll(\"作者:\",\"\"),\n \"kind\":\"\",\n \"coverUrl\":img,\n \"intro\":$element.find(\"p\").eq(3).text(),\n \"tocUrl\":baseurl+$element.find(\"a\").eq(0).attr('href'),\n \"wordCount\":\"\",\n \"type\":1,\n \"latestChapterTitle\":$element.find(\"p\").eq(0).text(),\n }\n books.push(book);\n });\n }catch (e){\n\n }\n return JSON.stringify(books)\n }\n\n \/\/返回http开头的则任务登录链接会跳webview,其他的会按照json解析显示弹窗\n async function getloginurl(){\n return baseurl;\n }\n\n<\/script>\n<\/html>",
"login": true,
"lastUpdateTime": "1767339875740"
}