初步修改加载机制

This commit is contained in:
unanmed 2023-05-31 12:09:47 +08:00
parent 6404c6f4d0
commit 39c655d355
13 changed files with 165 additions and 333 deletions

1
.gitignore vendored
View File

@ -34,3 +34,4 @@ dist.rar
index.cjs index.cjs
!public/swap/*.h5save !public/swap/*.h5save
_bundle _bundle
out

View File

@ -5,7 +5,7 @@
"type": "module", "type": "module",
"scripts": { "scripts": {
"dev": "ts-node-esm script/dev.ts", "dev": "ts-node-esm script/dev.ts",
"build": "vue-tsc && vite build && ts-node-esm script/build.ts", "build": "vue-tsc && vite build && ts-node-esm script/build.ts 0 1 1",
"build-gh": "vue-tsc && vite build --base=/HumanBreak/ && ts-node-esm script/build.ts 1", "build-gh": "vue-tsc && vite build --base=/HumanBreak/ && ts-node-esm script/build.ts 1",
"build-local": "vue-tsc && vite build --base=/ && ts-node-esm script/build.ts 1", "build-local": "vue-tsc && vite build --base=/ && ts-node-esm script/build.ts 1",
"preview": "vite preview", "preview": "vite preview",
@ -32,6 +32,7 @@
"@rollup/plugin-babel": "^6.0.3", "@rollup/plugin-babel": "^6.0.3",
"@rollup/plugin-commonjs": "^25.0.0", "@rollup/plugin-commonjs": "^25.0.0",
"@rollup/plugin-node-resolve": "^15.0.2", "@rollup/plugin-node-resolve": "^15.0.2",
"@rollup/plugin-replace": "^5.0.2",
"@rollup/plugin-terser": "^0.4.3", "@rollup/plugin-terser": "^0.4.3",
"@rollup/plugin-typescript": "^11.1.1", "@rollup/plugin-typescript": "^11.1.1",
"@types/babel__core": "^7.20.0", "@types/babel__core": "^7.20.0",

View File

@ -48,6 +48,9 @@ devDependencies:
'@rollup/plugin-node-resolve': '@rollup/plugin-node-resolve':
specifier: ^15.0.2 specifier: ^15.0.2
version: 15.0.2(rollup@3.23.0) version: 15.0.2(rollup@3.23.0)
'@rollup/plugin-replace':
specifier: ^5.0.2
version: 5.0.2(rollup@3.23.0)
'@rollup/plugin-terser': '@rollup/plugin-terser':
specifier: ^0.4.3 specifier: ^0.4.3
version: 0.4.3(rollup@3.23.0) version: 0.4.3(rollup@3.23.0)
@ -1735,6 +1738,20 @@ packages:
rollup: 3.23.0 rollup: 3.23.0
dev: true dev: true
/@rollup/plugin-replace@5.0.2(rollup@3.23.0):
resolution: {integrity: sha512-M9YXNekv/C/iHHK+cvORzfRYfPbq0RDD8r0G+bMiTXjNGKulPnCT9O3Ss46WfhI6ZOCgApOP7xAdmCQJ+U2LAA==}
engines: {node: '>=14.0.0'}
peerDependencies:
rollup: ^1.20.0||^2.0.0||^3.0.0
peerDependenciesMeta:
rollup:
optional: true
dependencies:
'@rollup/pluginutils': 5.0.2(rollup@3.23.0)
magic-string: 0.27.0
rollup: 3.23.0
dev: true
/@rollup/plugin-terser@0.4.3(rollup@3.23.0): /@rollup/plugin-terser@0.4.3(rollup@3.23.0):
resolution: {integrity: sha512-EF0oejTMtkyhrkwCdg0HJ0IpkcaVg1MMSf2olHb2Jp+1mnLM04OhjpJWGma4HobiDTF0WCyViWuvadyE9ch2XA==} resolution: {integrity: sha512-EF0oejTMtkyhrkwCdg0HJ0IpkcaVg1MMSf2olHb2Jp+1mnLM04OhjpJWGma4HobiDTF0WCyViWuvadyE9ch2XA==}
engines: {node: '>=14.0.0'} engines: {node: '>=14.0.0'}

View File

@ -218,6 +218,13 @@ function main() {
this.__VERSION_CODE__ = 510; this.__VERSION_CODE__ = 510;
this.timestamp = 0; this.timestamp = 0;
// 远程资源地址,在线游戏中,塔本体不包含任何资源,只包含源码,从而可以降低游戏本体的体积并平均分担资源包体积
// 从而可以优化加载并避免网站发布的大小限制
this.USE_RESORCE = false;
this.RESOURCE_URL = '';
this.RESOURCE_SYMBOL = '';
this.RESOURCE_INDEX = {};
} }
// >>>> body end // >>>> body end

View File

@ -8,6 +8,12 @@ import rollupBabel from '@rollup/plugin-babel';
import terser from '@rollup/plugin-terser'; import terser from '@rollup/plugin-terser';
import resolve from '@rollup/plugin-node-resolve'; import resolve from '@rollup/plugin-node-resolve';
import commonjs from '@rollup/plugin-commonjs'; import commonjs from '@rollup/plugin-commonjs';
import { splitResorce } from './resource.js';
import compressing from 'compressing';
const map = !!Number(process.argv[2]);
const resorce = !!Number(process.argv[3]);
const compress = !!Number(process.argv[4]);
(async function () { (async function () {
const timestamp = Date.now(); const timestamp = Date.now();
@ -44,7 +50,7 @@ import commonjs from '@rollup/plugin-commonjs';
}); });
}) })
); );
if (process.argv[2] !== '1') await fs.remove('./dist/maps/'); if (!map) await fs.remove('./dist/maps/');
// 在线查看什么都看不到,这编辑器难道还需要留着吗? // 在线查看什么都看不到,这编辑器难道还需要留着吗?
await fs.remove('./dist/_server'); await fs.remove('./dist/_server');
await fs.remove('./dist/editor.html'); await fs.remove('./dist/editor.html');
@ -60,13 +66,11 @@ import commonjs from '@rollup/plugin-commonjs';
const project = await fs.readdir('./public/project'); const project = await fs.readdir('./public/project');
const floors = await fs.readdir('./public/project/floors'); const floors = await fs.readdir('./public/project/floors');
const assets = await fs.readdir('./dist/assets/'); const assets = await fs.readdir('./dist/assets/');
const plugin = await fs.readdir('./public/project/plugin');
const all = [ const all = [
...libs.map(v => `./public/libs/${v}`), ...libs.map(v => `./public/libs/${v}`),
...project.map(v => `./public/project/${v}`), ...project.map(v => `./public/project/${v}`),
...floors.map(v => `./public/project/floors/${v}`), ...floors.map(v => `./public/project/floors/${v}`),
...assets.map(v => `./dist/assets/${v}`), ...assets.map(v => `./dist/assets/${v}`)
...plugin.map(v => `./public/project/plugin/${v}`)
]; ];
for await (const dir of all) { for await (const dir of all) {
const stat = await fs.stat(dir); const stat = await fs.stat(dir);
@ -112,7 +116,7 @@ import commonjs from '@rollup/plugin-commonjs';
}) })
]); ]);
} catch (e) { } catch (e) {
await fs.copy('./public/project/fonts', './dist/project/fonts'); console.log('字体压缩失败');
} }
// 3. 压缩js插件 // 3. 压缩js插件
@ -142,8 +146,6 @@ import commonjs from '@rollup/plugin-commonjs';
await fs.remove('./dist/project/plugin/'); await fs.remove('./dist/project/plugin/');
} catch (e) { } catch (e) {
console.log(e);
console.log('压缩插件失败'); console.log('压缩插件失败');
} }
@ -170,4 +172,15 @@ import commonjs from '@rollup/plugin-commonjs';
try { try {
await fs.copy('./LICENSE', './dist/LICENSE'); await fs.copy('./LICENSE', './dist/LICENSE');
} catch {} } catch {}
// 6. 资源分离
if (resorce) {
await splitResorce(compress);
}
// 7. 压缩本体
if (compress) {
await fs.ensureDir('./out');
await compressing.zip.compressDir('./dist', './out/dist.zip');
}
})(); })();

View File

@ -1,215 +0,0 @@
const hexcase = 0; /* hex output format. 0 - lowercase; 1 - uppercase */
const chrsz = 8; /* bits per input character. 8 - ASCII; 16 - Unicode */
// ---------- md5码
/*
* * A JavaScript implementation of the RSA Data Security, Inc. MD5 Message
* * Digest Algorithm, as defined in RFC 1321.
* * Version 2.1 Copyright (C) Paul Johnston 1999 - 2002.
* * Other contributors: Greg Holt, Andrew Kepert, Ydnar, Lostinet
* * Distributed under the BSD License
* * See http://pajhome.org.uk/crypt/md5 for more info.
* */
/*
* * Configurable letiables. You may need to tweak these to be compatible with
* * the server-side, but the defaults work in most cases.
* */
/*
* * These are the functions you'll usually want to call
* * They take string arguments and return either hex or base-64 encoded strings
* */
export function hex_md5(s: string) {
return binl2hex(core_md5(str2binl(s), s.length * chrsz));
}
/*
* * Calculate the MD5 of an array of little-endian words, and a bit length
* */
function core_md5(x: any[], len: number) {
/* append padding */
x[len >> 5] |= 0x80 << len % 32;
x[(((len + 64) >>> 9) << 4) + 14] = len;
let a = 1732584193;
let b = -271733879;
let c = -1732584194;
let d = 271733878;
for (let i = 0; i < x.length; i += 16) {
const olda = a;
const oldb = b;
const oldc = c;
const oldd = d;
a = md5_ff(a, b, c, d, x[i + 0], 7, -680876936);
d = md5_ff(d, a, b, c, x[i + 1], 12, -389564586);
c = md5_ff(c, d, a, b, x[i + 2], 17, 606105819);
b = md5_ff(b, c, d, a, x[i + 3], 22, -1044525330);
a = md5_ff(a, b, c, d, x[i + 4], 7, -176418897);
d = md5_ff(d, a, b, c, x[i + 5], 12, 1200080426);
c = md5_ff(c, d, a, b, x[i + 6], 17, -1473231341);
b = md5_ff(b, c, d, a, x[i + 7], 22, -45705983);
a = md5_ff(a, b, c, d, x[i + 8], 7, 1770035416);
d = md5_ff(d, a, b, c, x[i + 9], 12, -1958414417);
c = md5_ff(c, d, a, b, x[i + 10], 17, -42063);
b = md5_ff(b, c, d, a, x[i + 11], 22, -1990404162);
a = md5_ff(a, b, c, d, x[i + 12], 7, 1804603682);
d = md5_ff(d, a, b, c, x[i + 13], 12, -40341101);
c = md5_ff(c, d, a, b, x[i + 14], 17, -1502002290);
b = md5_ff(b, c, d, a, x[i + 15], 22, 1236535329);
a = md5_gg(a, b, c, d, x[i + 1], 5, -165796510);
d = md5_gg(d, a, b, c, x[i + 6], 9, -1069501632);
c = md5_gg(c, d, a, b, x[i + 11], 14, 643717713);
b = md5_gg(b, c, d, a, x[i + 0], 20, -373897302);
a = md5_gg(a, b, c, d, x[i + 5], 5, -701558691);
d = md5_gg(d, a, b, c, x[i + 10], 9, 38016083);
c = md5_gg(c, d, a, b, x[i + 15], 14, -660478335);
b = md5_gg(b, c, d, a, x[i + 4], 20, -405537848);
a = md5_gg(a, b, c, d, x[i + 9], 5, 568446438);
d = md5_gg(d, a, b, c, x[i + 14], 9, -1019803690);
c = md5_gg(c, d, a, b, x[i + 3], 14, -187363961);
b = md5_gg(b, c, d, a, x[i + 8], 20, 1163531501);
a = md5_gg(a, b, c, d, x[i + 13], 5, -1444681467);
d = md5_gg(d, a, b, c, x[i + 2], 9, -51403784);
c = md5_gg(c, d, a, b, x[i + 7], 14, 1735328473);
b = md5_gg(b, c, d, a, x[i + 12], 20, -1926607734);
a = md5_hh(a, b, c, d, x[i + 5], 4, -378558);
d = md5_hh(d, a, b, c, x[i + 8], 11, -2022574463);
c = md5_hh(c, d, a, b, x[i + 11], 16, 1839030562);
b = md5_hh(b, c, d, a, x[i + 14], 23, -35309556);
a = md5_hh(a, b, c, d, x[i + 1], 4, -1530992060);
d = md5_hh(d, a, b, c, x[i + 4], 11, 1272893353);
c = md5_hh(c, d, a, b, x[i + 7], 16, -155497632);
b = md5_hh(b, c, d, a, x[i + 10], 23, -1094730640);
a = md5_hh(a, b, c, d, x[i + 13], 4, 681279174);
d = md5_hh(d, a, b, c, x[i + 0], 11, -358537222);
c = md5_hh(c, d, a, b, x[i + 3], 16, -722521979);
b = md5_hh(b, c, d, a, x[i + 6], 23, 76029189);
a = md5_hh(a, b, c, d, x[i + 9], 4, -640364487);
d = md5_hh(d, a, b, c, x[i + 12], 11, -421815835);
c = md5_hh(c, d, a, b, x[i + 15], 16, 530742520);
b = md5_hh(b, c, d, a, x[i + 2], 23, -995338651);
a = md5_ii(a, b, c, d, x[i + 0], 6, -198630844);
d = md5_ii(d, a, b, c, x[i + 7], 10, 1126891415);
c = md5_ii(c, d, a, b, x[i + 14], 15, -1416354905);
b = md5_ii(b, c, d, a, x[i + 5], 21, -57434055);
a = md5_ii(a, b, c, d, x[i + 12], 6, 1700485571);
d = md5_ii(d, a, b, c, x[i + 3], 10, -1894986606);
c = md5_ii(c, d, a, b, x[i + 10], 15, -1051523);
b = md5_ii(b, c, d, a, x[i + 1], 21, -2054922799);
a = md5_ii(a, b, c, d, x[i + 8], 6, 1873313359);
d = md5_ii(d, a, b, c, x[i + 15], 10, -30611744);
c = md5_ii(c, d, a, b, x[i + 6], 15, -1560198380);
b = md5_ii(b, c, d, a, x[i + 13], 21, 1309151649);
a = md5_ii(a, b, c, d, x[i + 4], 6, -145523070);
d = md5_ii(d, a, b, c, x[i + 11], 10, -1120210379);
c = md5_ii(c, d, a, b, x[i + 2], 15, 718787259);
b = md5_ii(b, c, d, a, x[i + 9], 21, -343485551);
a = safe_add(a, olda);
b = safe_add(b, oldb);
c = safe_add(c, oldc);
d = safe_add(d, oldd);
}
return Array(a, b, c, d);
}
/*
* * These functions implement the four basic operations the algorithm uses.
* */
function md5_cmn(q: number, a: any, b: any, x: any, s: any, t: any) {
return safe_add(bit_rol(safe_add(safe_add(a, q), safe_add(x, t)), s), b);
}
function md5_ff(
a: number,
b: number,
c: number,
d: number,
x: any,
s: number,
t: number
) {
return md5_cmn((b & c) | (~b & d), a, b, x, s, t);
}
function md5_gg(
a: number,
b: number,
c: number,
d: number,
x: any,
s: number,
t: number
) {
return md5_cmn((b & d) | (c & ~d), a, b, x, s, t);
}
function md5_hh(
a: number,
b: number,
c: number,
d: number,
x: any,
s: number,
t: number
) {
return md5_cmn(b ^ c ^ d, a, b, x, s, t);
}
function md5_ii(
a: number,
b: number,
c: number,
d: number,
x: any,
s: number,
t: number
) {
return md5_cmn(c ^ (b | ~d), a, b, x, s, t);
}
/*
* * Add integers, wrapping at 2^32. This uses 16-bit operations internally
* * to work around bugs in some JS interpreters.
* */
function safe_add(x: number, y: number) {
const lsw = (x & 0xffff) + (y & 0xffff);
const msw = (x >> 16) + (y >> 16) + (lsw >> 16);
return (msw << 16) | (lsw & 0xffff);
}
/*
* * Bitwise rotate a 32-bit number to the left.
* */
function bit_rol(num: number, cnt: number) {
return (num << cnt) | (num >>> (32 - cnt));
}
/*
* * Convert a string to an array of little-endian words
* * If chrsz is ASCII, characters >255 have their hi-byte silently ignored.
* */
function str2binl(str: string) {
let bin = Array();
let mask = (1 << chrsz) - 1;
for (let i = 0; i < str.length * chrsz; i += chrsz)
bin[i >> 5] |= (str.charCodeAt(i / chrsz) & mask) << i % 32;
return bin;
}
/*
* * Convert an array of little-endian words to a hex string.
* */
function binl2hex(binarray: string | any[]) {
let hex_tab = hexcase ? '0123456789ABCDEF' : '0123456789abcdef';
let str = '';
for (let i = 0; i < binarray.length * 4; i++) {
str +=
hex_tab.charAt((binarray[i >> 2] >> ((i % 4) * 8 + 4)) & 0xf) +
hex_tab.charAt((binarray[i >> 2] >> ((i % 4) * 8)) & 0xf);
}
return str;
}

53
script/resource.ts Normal file
View File

@ -0,0 +1,53 @@
import fs from 'fs-extra';
import { uniqueSymbol } from './utils.js';
import { extname, resolve } from 'path';
type ResorceType =
| 'bgm'
| 'sound'
| 'autotiles'
| 'images'
| 'materials'
| 'tilesets'
| 'animates'
| 'fonts';
let resorceIndex = 0;
const compress: ResorceType[] = [
'sound',
'animates',
'autotiles',
'images',
'materials',
'tilesets'
];
const SYMBOL = uniqueSymbol();
const MAX_SIZE = 100 * (1 << 20);
let totalSize = 0;
export async function splitResorce(compress: boolean = false) {
const folder = await fs.stat('./dist');
totalSize = folder.size;
if (totalSize < MAX_SIZE) return;
await fs.ensureDir('./dist-resource');
await doSplit(compress);
}
async function sortDir(dir: string, ext?: string[]) {
const path = await fs.readdir(dir);
const stats: fs.Stats[] = [];
for await (const one of path) {
if (ext && !ext.includes(extname(one))) continue;
const stat = await fs.stat(resolve(dir, one));
if (!stat.isFile()) continue;
stats.push(stat);
}
return stats.sort((a, b) => b.size - a.size);
}
async function doSplit(compress: boolean) {}

View File

@ -1,91 +0,0 @@
import axios from 'axios';
import fs from 'fs/promises';
import fss from 'fs';
import FormData from 'form-data';
import compressing from 'compressing';
import { hex_md5 } from './md_5.js';
const id = 1000; // 用户id
const password = 'password'; // 密码
(async function () {
// 压缩
await compressing.zip.compressDir('./dist', './dist.zip');
const name = (
(() => {
const data = fss.readFileSync('./public/project/data.js', 'utf-8');
const json = JSON.parse(
data
.split(/(\n|\r\n)/)
.slice(1)
.join('\n')
);
return json;
})() as { firstData: Record<string, string> }
).firstData.name;
const pass = hex_md5(password);
// 登录
const res = await axios.postForm(
'https://h5mota.com/backend/user/login.php',
{
nid: id,
npass: pass
}
);
if (res.data.code === 0) console.log('登录成功');
else return console.log('登录失败');
const data = await axios.post(
'https://h5mota.com/up2cos/getData.php',
void 0,
{
headers: {
'Access-Control-Allow-Origin': '*',
Cookie: `id=${id}; password=${pass}`
}
}
);
if (data.data.code !== 1) return console.log('获取自助更新信息失败');
const tower = data.data.list.find((v: { name: string }) => v.name === name);
if (!tower) return console.log('你还没有发过这个塔');
const stat = await fs.stat('./dist.zip');
let size = '';
size = `${stat.size}B`;
if (stat.size > 1024) size = `${(stat.size / 1024).toFixed(2)}KB`;
if (stat.size > 1024 ** 2) size = `${(stat.size / 1024 ** 2).toFixed(2)}MB`;
if (stat.size > 1024 ** 3) size = `${(stat.size / 1024 ** 3).toFixed(2)}GB`;
const stream = fss.createReadStream('./dist.zip');
const form = new FormData();
form.append('type', 'upload');
form.append('name', name);
form.append('comment', tower.text);
form.append('file', stream);
const headers = form.getHeaders(); // 获取headers
const [err, length] = await new Promise(
(res: (v: [Error | null, number]) => void) =>
form.getLength((e, l) => res([e, l]))
);
if (err) return console.log('获取content-length失败');
headers['content-length'] = length;
headers['cookie'] = `id=${id}; password=${pass}`;
console.log(`开始上传, 文件大小: ${size}`);
const ans = await axios.post(
'https://h5mota.com/up2cos/setData.php',
form,
{
headers
}
);
if (ans.data.error === 0 && ans.data.upload === true && ans.data.code === 1)
console.log('上传成功');
else throw new Error(`上传失败, 失败信息: ${ans.data}`);
})();

3
script/utils.ts Normal file
View File

@ -0,0 +1,3 @@
export function uniqueSymbol() {
return Math.ceil(Math.random() * 0xefffffff + 0x10000000).toString(16);
}

30
src/core/loader/load.ts Normal file
View File

@ -0,0 +1,30 @@
import axios, { AxiosRequestConfig } from 'axios';
class LoadTask<T> {
loaded: boolean = false;
promise?: Promise<T>;
url: string;
config?: AxiosRequestConfig<T>;
constructor(url: string, config?: AxiosRequestConfig<T>) {
this.url = url;
this.config = config;
}
load() {
if (this.promise) return this.promise;
return (this.promise = axios.get(this.url, this.config));
}
static list: Promise<any>[] = [];
static push(...tasks: LoadTask<any>[]) {
this.list.push(...tasks.map(v => v.load()));
}
static onEnd<T extends any[] = any[]>(): Promise<T> {
return Promise.all(LoadTask.list) as Promise<T>;
}
}
export default function load() {}

View File

@ -71,5 +71,15 @@
"desc": [ "desc": [
"开启后,会在画面内以类似状态栏的盒子的形式显示当前已学习的怪物技能。" "开启后,会在画面内以类似状态栏的盒子的形式显示当前已学习的怪物技能。"
] ]
},
"betterLoad": {
"text": "优化加载",
"desc": [
"开启后游戏将对加载进行优化,缩短进入游戏时的加载时长,而在游戏中对资源进行部分性按需加载,从而对加载进行优化。",
"该设置不会影响你的正常游戏。",
"<br>",
"<br>",
"注:修改后刷新页面起效。"
]
} }
} }

View File

@ -18,6 +18,7 @@ import completion, { floors } from './plugin/completion';
import path from './plugin/fx/path'; import path from './plugin/fx/path';
import gameCanvas from './plugin/fx/gameCanvas'; import gameCanvas from './plugin/fx/gameCanvas';
import noise from './plugin/fx/noise'; import noise from './plugin/fx/noise';
import load from './core/loader/load';
function forward() { function forward() {
const toForward: any[] = [ const toForward: any[] = [
@ -44,7 +45,6 @@ function forward() {
]; ];
// 初始化所有插件并转发到core上 // 初始化所有插件并转发到core上
(async function () {
for (const data of toForward) { for (const data of toForward) {
for (const name in data) { for (const name in data) {
const d = data[name as keyof typeof data]; const d = data[name as keyof typeof data];
@ -61,7 +61,7 @@ function forward() {
} }
console.log('插件转发完成!'); console.log('插件转发完成!');
})(); load();
Object.values(floors).forEach((v, i) => { Object.values(floors).forEach((v, i) => {
const from = core.floorIds.indexOf(v[0]); const from = core.floorIds.indexOf(v[0]);

View File

@ -19,7 +19,6 @@ interface EnemyInfo {
atkBuff: number; atkBuff: number;
defBuff: number; defBuff: number;
hpBuff: number; hpBuff: number;
together: number;
enemy: Enemy; enemy: Enemy;
} }
@ -202,7 +201,6 @@ export class DamageEnemy<T extends EnemyIds = EnemyIds> {
atkBuff: 0, atkBuff: 0,
defBuff: 0, defBuff: 0,
hpBuff: 0, hpBuff: 0,
together: 0,
enemy: this.enemy enemy: this.enemy
}; };
this.needCalculate = true; this.needCalculate = true;
@ -322,9 +320,13 @@ export class DamageEnemy<T extends EnemyIds = EnemyIds> {
const square7: HaloFn[] = []; const square7: HaloFn[] = [];
const square5: HaloFn[] = []; const square5: HaloFn[] = [];
// 抱团
if (speical.includes(8)) { if (speical.includes(8)) {
square5.push((e, enemy) => { square5.push((e, enemy) => {
if (e === this.info) e.together += enemy.together ?? 0; if (e.special.includes(8)) {
e.atkBuff += enemy.together ?? 0;
e.defBuff += enemy.together ?? 0;
}
}); });
this.providedHalo.push(8); this.providedHalo.push(8);
} }
@ -424,6 +426,7 @@ export class DamageEnemy<T extends EnemyIds = EnemyIds> {
calMapDamage(damage?: Record<string, MapDamage>) { calMapDamage(damage?: Record<string, MapDamage>) {
damage ??= {}; damage ??= {};
if (!has(this.x) || !has(this.y)) return damage;
return damage; return damage;
} }
@ -436,7 +439,7 @@ const realStatus: (keyof HeroStatus)[] = ['atk', 'def'];
/** /**
* *
*/ */
const skills = [ const skills: [unlock: string, condition: string][] = [
['bladeOn', 'blade'], ['bladeOn', 'blade'],
['shieldOn', 'shield'] ['shieldOn', 'shield']
]; ];