HumanBreak/src/plugin/utils.ts

257 lines
6.4 KiB
TypeScript
Raw Normal View History

2023-02-05 18:17:10 +08:00
/// <reference path="../types/core.d.ts" />
2022-12-29 12:02:48 +08:00
import { message } from 'ant-design-vue';
import { MessageApi } from 'ant-design-vue/lib/message';
2022-11-16 23:01:23 +08:00
import { isNil } from 'lodash';
2023-02-21 23:22:03 +08:00
import { Animation, sleep, TimingFn } from 'mutate-animate';
2022-11-21 20:00:34 +08:00
import { ComputedRef, ref } from 'vue';
2022-11-19 18:15:42 +08:00
import { EVENT_KEY_CODE_MAP } from './keyCodes';
import axios from 'axios';
import { decompressFromBase64 } from 'lz-string';
2022-11-16 23:01:23 +08:00
2022-11-21 20:00:34 +08:00
type CanParseCss = keyof {
[P in keyof CSSStyleDeclaration as CSSStyleDeclaration[P] extends string
? P extends string
? P
: never
: never]: CSSStyleDeclaration[P];
};
2022-11-16 23:01:23 +08:00
export default function init() {
return {
has,
getDamageColor,
parseCss,
tip,
changeLocalStorage,
swapChapter
};
2022-11-16 23:01:23 +08:00
}
/**
* undefined或null
* @param value
*/
export function has<T>(value: T): value is NonNullable<T> {
return !isNil(value);
}
/**
*
* @param damage
*/
export function getDamageColor(damage: number): Color {
2022-11-16 23:01:23 +08:00
if (typeof damage !== 'number') return '#f00';
if (damage === 0) return '#2f2';
if (damage < 0) return '#7f7';
if (damage < core.status.hero.hp / 3) return '#fff';
if (damage < (core.status.hero.hp * 2) / 3) return '#ff4';
2022-12-30 14:14:28 +08:00
if (damage < core.status.hero.hp) return '#f93';
return '#f22';
2022-11-16 23:01:23 +08:00
}
2022-11-19 11:30:14 +08:00
/**
*
* @param canvas
* @param w
* @param h
*/
export function setCanvasSize(
canvas: HTMLCanvasElement,
w: number,
h: number
): void {
canvas.width = w;
canvas.height = h;
canvas.style.width = `${w}px`;
canvas.style.height = `${h}px`;
}
2022-11-19 18:15:42 +08:00
/**
* keycode对应的键
* @param key
*/
export function keycode(key: number) {
return EVENT_KEY_CODE_MAP[key];
}
2022-11-21 20:00:34 +08:00
/**
* css字符串为CSSStyleDeclaration对象
* @param css css字符串
*/
export function parseCss(css: string): Partial<Record<CanParseCss, string>> {
2023-02-05 18:17:10 +08:00
const str = css.replace(/[\n\s\t]*/g, '').replace(/;*/g, ';');
2022-11-21 20:00:34 +08:00
const styles = str.split(';');
const res: Partial<Record<CanParseCss, string>> = {};
for (const one of styles) {
const [key, data] = one.split(':');
2022-11-30 16:42:44 +08:00
const cssKey = key.replace(/\-([a-z])/g, (str, $1) =>
2022-11-21 20:00:34 +08:00
$1.toUpperCase()
) as CanParseCss;
res[cssKey] = data;
}
return res;
}
/**
* 使
* @param str
2022-11-30 16:42:44 +08:00
* @param time 1true时
2022-11-21 20:00:34 +08:00
* @param timing 线线
2022-11-30 16:42:44 +08:00
* @param avr
2022-11-21 20:00:34 +08:00
* @returns
*/
export function type(
str: string,
time: number = 1000,
2022-11-30 16:42:44 +08:00
timing: TimingFn = n => n,
avr: boolean = false
2022-11-21 20:00:34 +08:00
): Ref<string> {
const toShow = eval('`' + str + '`') as string;
2022-11-30 16:42:44 +08:00
if (typeof toShow !== 'string') {
throw new TypeError('Error str type in typing!');
}
2022-12-29 00:26:12 +08:00
if (toShow.startsWith('!!html')) return ref(toShow);
2022-11-30 16:42:44 +08:00
if (avr) time *= toShow.length;
2022-11-21 20:00:34 +08:00
const ani = new Animation();
const content = ref('');
const all = toShow.length;
const fn = (time: number) => {
if (!has(time)) return;
const now = ani.x;
content.value = toShow.slice(0, Math.floor(now));
if (Math.floor(now) === all) {
ani.ticker.destroy();
2022-11-30 16:42:44 +08:00
content.value = toShow;
2022-11-21 20:00:34 +08:00
}
};
ani.ticker.add(fn);
ani.mode(timing).time(time).move(all, 0);
2022-12-31 19:49:43 +08:00
setTimeout(() => ani.ticker.destroy(), time + 100);
2022-11-21 20:00:34 +08:00
return content;
}
2022-12-29 12:02:48 +08:00
export function tip(
type: Exclude<keyof MessageApi, 'open' | 'config' | 'destroy'>,
text: string
) {
message[type]({
content: text,
class: 'antdv-message'
});
}
2022-12-30 23:28:27 +08:00
/**
*
* @param str
*/
export function splitText(str: string[]) {
return str
.map((v, i, a) => {
if (/^\d+\./.test(v)) return `${'&nbsp;'.repeat(12)}${v}`;
else if (
(has(a[i - 1]) && v !== '<br>' && a[i - 1] === '<br>') ||
i === 0
) {
return `${'&nbsp;'.repeat(8)}${v}`;
} else return v;
})
.join('');
}
2023-01-15 20:09:02 +08:00
/**
*
* @param cb
*/
export function nextFrame(cb: (time: number) => void) {
requestAnimationFrame(() => {
requestAnimationFrame(cb);
});
}
2023-01-23 14:38:50 +08:00
/**
*
* @param canvas
* @param name
*/
export function downloadCanvasImage(
canvas: HTMLCanvasElement,
name: string
): void {
const data = canvas.toDataURL('image/png');
download(data, name);
}
/**
*
* @param content
* @param name
*/
export function download(content: string, name: string) {
const a = document.createElement('a');
a.download = `${name}.png`;
a.href = content;
document.body.appendChild(a);
a.click();
document.body.removeChild(a);
}
2023-02-21 23:22:03 +08:00
/**
*
* @param funcs
* @param interval
*/
2023-02-21 23:27:40 +08:00
export async function doByInterval(
funcs: (() => void)[],
interval: number,
awaitFirst: boolean = false
) {
2023-02-21 23:22:03 +08:00
for await (const fn of funcs) {
2023-02-21 23:27:40 +08:00
if (awaitFirst) {
await sleep(interval);
}
2023-02-21 23:22:03 +08:00
fn();
2023-02-21 23:27:40 +08:00
if (!awaitFirst) {
await sleep(interval);
}
2023-02-21 23:22:03 +08:00
}
}
2023-02-23 23:10:57 +08:00
/**
*
* @param name
* @param fn
* @param defaultValue
*/
export function changeLocalStorage<T>(
name: string,
fn: (data: T) => T,
defaultValue?: T
) {
const now = core.getLocalStorage(name, defaultValue);
const to = fn(now);
core.setLocalStorage(name, to);
}
export async function swapChapter(chapter: number, hard: number) {
const h = hard === 2 ? 'hard' : 'easy';
const save = await axios.get(
`${import.meta.env.BASE_URL}swap/${chapter}.${h}.h5save`,
{
responseType: 'text',
responseEncoding: 'utf-8'
}
);
const data = JSON.parse(decompressFromBase64(save.data));
core.loadData(data.data, () => {
core.removeFlag('__fromLoad__');
core.drawTip('读档成功');
});
}