import base from './base.js' import fetch from 'node-fetch' import lodash from 'lodash' import puppeteer from '../../../lib/puppeteer/puppeteer.js' import common from '../../../lib/common/common.js' import gsCfg from '../model/gsCfg.js' const _path = process.cwd() let emoticon export default class MysNews extends base { constructor (e) { super(e) this.model = 'mysNews' } async getNews () { let type = 1 let typeName = '公告' if (this.e.msg.includes('资讯')) { type = '3' typeName = '资讯' } if (this.e.msg.includes('活动')) { type = '2' typeName = '活动' } const res = await this.postData('getNewsList', { gids: 6, page_size: 20, type }) if (!res) return const data = res.data.list if (data.length == 0) { return true } const page = this.e.msg.replace(/#|#|星铁|星穹|铁道|公告|资讯|活动/g, '').trim() || 1 if (page > data.length) { await this.e.reply('目前只查前20条最新的公告,请输入1-20之间的整数。') return true } const postId = data[page - 1].post.post_id const param = await this.newsDetail(postId) const img = await this.rander(param) return await this.replyMsg(img, `崩坏星穹铁道${typeName}:${param.data.post.subject}`) } async rander (param) { const pageHeight = 7000 await puppeteer.browserInit() if (!puppeteer.browser) return false const savePath = puppeteer.dealTpl('mysNews', param) if (!savePath) return false const page = await puppeteer.browser.newPage() try { await page.goto(`file://${_path}${lodash.trim(savePath, '.')}`, { timeout: 120000 }) const body = await page.$('#container') || await page.$('body') const boundingBox = await body.boundingBox() const num = Math.round(boundingBox.height / pageHeight) || 1 if (num > 1) { await page.setViewport({ width: boundingBox.width, height: pageHeight + 100 }) } const img = [] for (let i = 1; i <= num; i++) { const randData = { type: 'jpeg', quality: 90 } if (i != 1 && i == num) { await page.setViewport({ width: boundingBox.width, height: parseInt(boundingBox.height) - pageHeight * (num - 1) }) } if (i != 1 && i <= num) { await page.evaluate(() => window.scrollBy(0, 7000)) } let buff if (num == 1) { buff = await body.screenshot(randData) } else { buff = await page.screenshot(randData) } if (num > 2) await common.sleep(200) puppeteer.renderNum++ /** 计算图片大小 */ const kb = (buff.length / 1024).toFixed(2) + 'kb' logger.mark(`[图片生成][${this.model}][${puppeteer.renderNum}次] ${kb}`) img.push(segment.image(buff)) } await page.close().catch((err) => logger.error(err)) if (num > 1) { logger.mark(`[图片生成][${this.model}] 处理完成`) } return img } catch (error) { logger.error(`图片生成失败:${this.model}:${error}`) /** 关闭浏览器 */ if (puppeteer.browser) { await puppeteer.browser.close().catch((err) => logger.error(err)) } puppeteer.browser = false } } async newsDetail (postId) { const res = await this.postData('getPostFull', { gids: 6, read: 1, post_id: postId }) if (!res) return const data = await this.detalData(res.data.post) return { ...this.screenData, saveId: postId, dataConent: data.post.content, data } } postApi (type, data) { let host = 'https://bbs-api-static.mihoyo.com/' let param = [] lodash.forEach(data, (v, i) => param.push(`${i}=${v}`)) param = param.join('&') switch (type) { // 搜索 case 'searchPosts': host = 'https://bbs-api.mihoyo.com/post/wapi/searchPosts?' break // 帖子详情 case 'getPostFull': host += 'post/wapi/getPostFull?' break // 公告列表 case 'getNewsList': host += 'post/wapi/getNewsList?' break case 'emoticon': host += 'misc/api/emoticon_set?' break } return host + param } async postData (type, data) { const url = this.postApi(type, data) const headers = { Referer: 'https://bbs.mihoyo.com/', 'User-Agent': 'Mozilla/5.0 (Windows NT 6.1; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/104.0.0.0 Safari/537.36' } let response try { response = await fetch(url, { method: 'get', headers }) } catch (error) { logger.error(error.toString()) return false } if (!response.ok) { logger.error(`[米游社接口错误][${type}] ${response.status} ${response.statusText}`) return false } const res = await response.json() return res } async detalData (data) { let json try { json = JSON.parse(data.post.content) } catch (error) { } if (typeof json == 'object') { if (json.imgs && json.imgs.length > 0) { for (const val of json.imgs) { data.post.content = `