mirror of
				https://github.com/qist/tvbox.git
				synced 2025-10-31 04:02:22 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			258 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			258 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| // 自动从 地址发布页 获取&跳转url地址
 | |
| import { Crypto, load, _ } from './lib/cat.js';
 | |
| 
 | |
| let key = '在线之家';
 | |
| let HOST = 'https://www.zxzj.site'; // 地址发布页
 | |
| let host = '';
 | |
| let siteKey = '';
 | |
| let siteType = 0;
 | |
| 
 | |
| const MOBILE_UA = 'Mozilla/5.0 (Linux; Android 11; M2007J3SC Build/RKQ1.200826.002; wv) AppleWebKit/537.36 (KHTML, like Gecko) Version/4.0 Chrome/77.0.3865.120 MQQBrowser/6.2 TBS/045714 Mobile Safari/537.36';
 | |
| 
 | |
| async function request(reqUrl, agentSp) {
 | |
|     let res = await req(reqUrl, {
 | |
|         method: 'get',
 | |
|         headers: {
 | |
|             'User-Agent': agentSp || MOBILE_UA,
 | |
|             'Referer': host
 | |
|         },
 | |
|     });
 | |
|     return res.content;
 | |
| }
 | |
| 
 | |
| // cfg = {skey: siteKey, ext: extend}
 | |
| async function init(cfg) {
 | |
|     siteKey = cfg.skey;
 | |
|     siteType = cfg.stype;
 | |
|     const html = await request(HOST);
 | |
|     const $ = load(html);
 | |
|     host = $('div.content-top > ul > li').find('a:first')[0].attribs.href;
 | |
|     console.debug('在线之家 跳转地址 =====>' + host); // js_debug.log
 | |
| }
 | |
| 
 | |
| async function home(filter) {
 | |
|     const html = await request(host);
 | |
|     const $ = load(html);
 | |
|     const class_parse = $('ul.stui-header__menu > li > a[href*=list]');
 | |
|     let classes = _.map(class_parse, (cls) => {
 | |
|         let typeId = cls.attribs['href'];
 | |
|         typeId = typeId.substring(typeId.lastIndexOf('/') + 1).replace('.html', '');
 | |
|         return {
 | |
|             type_id: typeId,
 | |
|             type_name: cls.children[0].data,
 | |
|         };
 | |
|     });
 | |
|     const filterObj = {
 | |
|         1: [{ key: 'class', name: '剧情', value: [{ n: '全部', v: '' }, { n: '喜剧', v: '喜剧' }, { n: '爱情', v: '爱情' }, { n: '恐怖', v: '恐怖' }, { n: '动作', v: '动作' }, { n: '科幻', v: '科幻' }, { n: '剧情', v: '剧情' }, { n: '战争', v: '战争' }, { n: '警匪', v: '警匪' }, { n: '犯罪', v: '犯罪' }, { n: '动画', v: '动画' }, { n: '奇幻', v: '奇幻' }, { n: '冒险', v: '冒险' }] }, { key: 'area', name: '地区', value: [{ n: '全部', v: '' }, { n: '大陆', v: '大陆' }, { n: '香港', v: '香港' }, { n: '台湾', v: '台湾' }, { n: '欧美', v: '欧美' }, { n: '韩国', v: '韩国' }, { n: '日本', v: '日本' }, { n: '泰国', v: '泰国' }, { n: '印度', v: '印度' }, { n: '俄罗斯', v: '俄罗斯' }, { n: '其他', v: '其他' }] }],
 | |
|         2: [{ key: 'class', name: '剧情', value: [{ n: '全部', v: '' }, { n: '剧情', v: '剧情' }, { n: '喜剧', v: '喜剧' }, { n: '爱情', v: '爱情' }, { n: '动作', v: '动作' }, { n: '悬疑', v: '悬疑' }, { n: '恐怖', v: '恐怖' }, { n: '奇幻', v: '奇幻' }, { n: '惊悚', v: '惊悚' }, { n: '犯罪', v: '犯罪' }, { n: '科幻', v: '科幻' }, { n: '音乐', v: '音乐' }, { n: '其他', v: '其他' }] }],
 | |
|         3: [{ key: 'class', name: '剧情', value: [{ n: '全部', v: '' }, { n: '剧情', v: '剧情' }, { n: '喜剧', v: '喜剧' }, { n: '爱情', v: '爱情' }, { n: '动作', v: '动作' }, { n: '悬疑', v: '悬疑' }, { n: '恐怖', v: '恐怖' }, { n: '奇幻', v: '奇幻' }, { n: '惊悚', v: '惊悚' }, { n: '犯罪', v: '犯罪' }, { n: '科幻', v: '科幻' }, { n: '音乐', v: '音乐' }, { n: '其他', v: '其他' }] }],
 | |
|         4: [{ key: 'class', name: '剧情', value: [{ n: '全部', v: '' }, { n: '剧情', v: '剧情' }, { n: '喜剧', v: '喜剧' }, { n: '爱情', v: '爱情' }, { n: '动作', v: '动作' }, { n: '悬疑', v: '悬疑' }, { n: '恐怖', v: '恐怖' }, { n: '奇幻', v: '奇幻' }, { n: '惊悚', v: '惊悚' }, { n: '犯罪', v: '犯罪' }, { n: '科幻', v: '科幻' }, { n: '音乐', v: '音乐' }, { n: '其他', v: '其他' }] }],
 | |
|         6: [{ key: 'class', name: '剧情', value: [{ n: '全部', v: '' }, { n: '情感', v: '情感' }, { n: '科幻', v: '科幻' }, { n: '热血', v: '热血' }, { n: '推理', v: '推理' }, { n: '搞笑', v: '搞笑' }, { n: '冒险', v: '冒险' }, { n: '萝莉', v: '萝莉' }, { n: '校园', v: '校园' }, { n: '动作', v: '动作' }, { n: '机战', v: '机战' }, { n: '运动', v: '运动' }, { n: '战争', v: '战争' }, { n: '少年', v: '少年' }] }, { key: 'area', name: '地区', value: [{ n: '全部', v: '' }, { n: '国产', v: '国产' }, { n: '日本', v: '日本' }, { n: '欧美', v: '欧美' }, { n: '其他', v: '其他' }] }]
 | |
|     };
 | |
|     let filYer = { key: 'year', name: '年份', value: [{ n: '全部', v: '' }, { n: '2023', v: '2023' }, { n: '2022', v: '2022' }, { n: '2021', v: '2021' }, { n: '2020', v: '2020' }, { n: '2019', v: '2019' }, { n: '2018', v: '2018' }, { n: '2017', v: '2017' }, { n: '2016', v: '2016' }, { n: '2015', v: '2015' }, { n: '2014', v: '2014' }, { n: '2013', v: '2013' }, { n: '2012', v: '2012' }, { n: '2011', v: '2011' }] };
 | |
|     filYer['init'] = filYer.value[0].v;
 | |
|     let filBy = { key: 'by', name: '排序', value: [{ n: '时间', v: 'time' }, { n: '人气', v: 'hits' }, { n: '评分', v: 'score' }] };
 | |
|     filBy['init'] = filBy.value[0].v;
 | |
|     return JSON.stringify({
 | |
|         class: _.map(classes, (cls) => {
 | |
|             if (filterObj[cls.type_id]) {
 | |
|                 filterObj[cls.type_id].push(filYer, filBy);
 | |
|                 filterObj[cls.type_id][0]['init'] = filterObj[cls.type_id][0].value[0].v;
 | |
|             } else {
 | |
|                 filterObj[cls.type_id] = [];
 | |
|                 filterObj[cls.type_id].push(filYer, filBy)
 | |
|             }
 | |
|             return cls;
 | |
|         }),
 | |
|         filters: filterObj,
 | |
|     });
 | |
| }
 | |
| 
 | |
| async function homeVod() {
 | |
|     const link = host + '/vodshow/1--hits---------2023.html';
 | |
|     const html = await request(link);
 | |
|     const $ = load(html);
 | |
|     const js2Base = await js2Proxy(true, siteType, siteKey, 'img/', {});
 | |
|     const items = $('ul.stui-vodlist > li');
 | |
|     let videos = _.map(items, (item) => {
 | |
|         const a = $(item).find('a:first')[0];
 | |
|         const remarks = $($(item).find('span.pic-text')[0]).text().trim();
 | |
|         return {
 | |
|             vod_id: a.attribs.href.replace(/.*?\/detail\/(.*).html/g, '$1'),
 | |
|             vod_name: a.attribs.title,
 | |
|             vod_pic: js2Base + base64Encode(a.attribs['data-original']),
 | |
|             vod_remarks: remarks || '',
 | |
|         };
 | |
|     });
 | |
|     return JSON.stringify({
 | |
|         list: videos,
 | |
|     });
 | |
| }
 | |
| 
 | |
| async function category(tid, pg, filter, extend) {
 | |
|     if (pg <= 0 || typeof (pg) == 'undefined') pg = 1;
 | |
|     const link = host + '/vodshow/' + tid + '-' + (extend.area || '') + '-' + (extend.by || 'time') + '-' + (extend.class || '') + '-' + (extend.lang || '') + '-' + (extend.letter || '') + '---' + pg + '---' + (extend.year || '') + '.html';
 | |
|     const html = await request(link);
 | |
|     const $ = load(html);
 | |
|     const js2Base = await js2Proxy(true, siteType, siteKey, 'img/', {});
 | |
|     const items = $('ul.stui-vodlist > li');
 | |
|     let videos = _.map(items, (item) => {
 | |
|         const a = $(item).find('a:first')[0];
 | |
|         const remarks = $($(item).find('span.pic-text')[0]).text().trim();
 | |
|         return {
 | |
|             vod_id: a.attribs.href.replace(/.*?\/detail\/(.*).html/g, '$1'),
 | |
|             vod_name: a.attribs.title,
 | |
|             vod_pic: js2Base + base64Encode(a.attribs['data-original']),
 | |
|             vod_remarks: remarks || '',
 | |
|         };
 | |
|     });
 | |
|     const hasMore = $('ul.stui-page__item > li > a:contains(下一页)').length > 0;
 | |
|     const pgCount = hasMore ? parseInt(pg) + 1 : parseInt(pg);
 | |
|     return JSON.stringify({
 | |
|         page: parseInt(pg),
 | |
|         pagecount: pgCount,
 | |
|         limit: 24,
 | |
|         total: 24 * pgCount,
 | |
|         list: videos,
 | |
|     });
 | |
| }
 | |
| 
 | |
| async function detail(id) {
 | |
|     const html = await request(host + '/detail/' + id + '.html');
 | |
|     const $ = load(html);
 | |
|     const js2Base = await js2Proxy(true, siteType, siteKey, 'img/', {});
 | |
|     const vod = {
 | |
|         vod_id: id,
 | |
|         vod_name: $('h1:first').text().trim(),
 | |
|         vod_type: $('.stui-content__detail p:first a').text(),
 | |
|         vod_actor: $('.stui-content__detail p:nth-child(3)').text().replace('主演:', ''),
 | |
|         vod_director: $('.stui-content__detail p:nth-child(4)').text().replace('导演:', ''),
 | |
|         vod_pic: js2Base + base64Encode($('.stui-content__thumb img:first').attr('data-original')),
 | |
|         vod_remarks: $('.stui-content__detail p:nth-child(5)').text() || '',
 | |
|         vod_content: $('span.detail-content').text().trim(),
 | |
|     };
 | |
|     let playMap = {};
 | |
|     const tabs = $('div.stui-vodlist__head > h3');
 | |
|     const playlists = $('ul.stui-content__playlist');
 | |
|     _.each(tabs, (tab, i) => {
 | |
|         const from = tab.children[0].data;
 | |
|         let list = playlists[i];
 | |
|         list = $(list).find('a');
 | |
|         _.each(list, (it) => {
 | |
|             const title = it.children[0].data;
 | |
|             const playUrl = it.attribs.href;
 | |
|             if (title.length == 0) title = it.children[0].data.trim();
 | |
|             if (!playMap.hasOwnProperty(from)) {
 | |
|                 playMap[from] = [];
 | |
|             }
 | |
|             playMap[from].push(title + '$' + playUrl);
 | |
|         });
 | |
|     });
 | |
|     vod.vod_play_from = _.keys(playMap).join('$$$');
 | |
|     const urls = _.values(playMap);
 | |
|     let vod_play_url = _.map(urls, (urlist) => {
 | |
|         return urlist.join('#');
 | |
|     });
 | |
|     vod.vod_play_url = vod_play_url.join('$$$');
 | |
|     return JSON.stringify({
 | |
|         list: [vod],
 | |
|     });
 | |
| }
 | |
| 
 | |
| async function play(flag, id, flags) {
 | |
|     const html = await request(host + id);
 | |
|     const mhtml = html.match(/r player_.*?=(.*?)</)[1];
 | |
|     const json = JSON.parse(mhtml);
 | |
|     const url = json.url;
 | |
|     const from = json.from;
 | |
|     if (json.encrypt == '1') {
 | |
|         url = unescape(url)
 | |
|     } else if (json.encrypt == '2') {
 | |
|         url = unescape(base64Decode(url))
 | |
|     }
 | |
|     if (url.indexOf('m3u8') >= 0 || url.indexOf('mp4') >= 0) {
 | |
|         // console.debug('在线之家url =====>' + url); // js_debug.log
 | |
|         return JSON.stringify({
 | |
|             parse: 0,
 | |
|             url: url,
 | |
|         });
 | |
|     } else if (from.indexOf('line3') >= 0 || from.indexOf('line5') >= 0) {
 | |
|         const ifrwy = await request(url);
 | |
|         const code = ifrwy.match(/var url = '(.*?)'/)[1].split('').reverse().join('');
 | |
|         let temp = '';
 | |
|         for (let i = 0x0; i < code.length; i = i + 0x2) {
 | |
|             temp += String.fromCharCode(parseInt(code[i] + code[i + 0x1], 0x10))
 | |
|         }
 | |
|         const purl = temp.substring(0x0, (temp.length - 0x7) / 0x2) + temp.substring((temp.length - 0x7) / 0x2 + 0x7);
 | |
|         // console.debug('在线之家purl =====>' + purl); // js_debug.log
 | |
|         return JSON.stringify({
 | |
|             parse: 0,
 | |
|             url: purl,
 | |
|         });
 | |
|     } else {
 | |
|         // console.debug('在线之家url =====>' + '空'); // js_debug.log
 | |
|         return '{}';
 | |
|     }
 | |
| }
 | |
| 
 | |
| async function search(wd, quick) {
 | |
|     var data = JSON.parse(await request(host + '/index.php/ajax/suggest?mid=1&wd=' + wd + '&limit=50')).list;
 | |
|     const js2Base = await js2Proxy(true, siteType, siteKey, 'img/', {});
 | |
|     let videos = _.map(data, (it) => {
 | |
|         return {
 | |
|             vod_id: it.id,
 | |
|             vod_name: it.name,
 | |
|             vod_pic: js2Base + base64Encode(it.pic),
 | |
|             vod_remarks: '',
 | |
|         }
 | |
|     });
 | |
|     return JSON.stringify({
 | |
|         list: videos,
 | |
|         limit: 50,
 | |
|     });
 | |
| }
 | |
| 
 | |
| function base64Encode(text) {
 | |
|     return Crypto.enc.Base64.stringify(Crypto.enc.Utf8.parse(text));
 | |
| }
 | |
| 
 | |
| function base64Decode(text) {
 | |
|     return Crypto.enc.Utf8.stringify(Crypto.enc.Base64.parse(text));
 | |
| }
 | |
| 
 | |
| async function proxy(segments, headers) {
 | |
|     let what = segments[0];
 | |
|     let url = base64Decode(segments[1]);
 | |
|     if (what == 'img') {
 | |
|         var resp = await req(url, {
 | |
|             buffer: 2,
 | |
|             headers: {
 | |
|                 Referer: 'https://api.douban.com/',
 | |
|                 'User-Agent': MOBILE_UA,
 | |
|             },
 | |
|         });
 | |
|         return JSON.stringify({
 | |
|             code: resp.code,
 | |
|             buffer: 2,
 | |
|             content: resp.content,
 | |
|             headers: resp.headers,
 | |
|         });
 | |
|     }
 | |
|     return JSON.stringify({
 | |
|         code: 500,
 | |
|         content: '',
 | |
|     });
 | |
| }
 | |
| 
 | |
| export function __jsEvalReturn() {
 | |
|     return {
 | |
|         init: init,
 | |
|         home: home,
 | |
|         homeVod: homeVod,
 | |
|         category: category,
 | |
|         detail: detail,
 | |
|         play: play,
 | |
|         proxy: proxy,
 | |
|         search: search,
 | |
|     };
 | |
| } |