refactor: 将 src/common src/core/common 移入 @motajs/legacy-common

This commit is contained in:
unanmed 2025-03-04 23:22:14 +08:00
parent a5e6f5b5f6
commit dc502ddd64
25 changed files with 256 additions and 201 deletions

View File

@ -0,0 +1,6 @@
{
"name": "@motajs/legacy-client",
"dependencies": {
"@motajs/common": "workspace:*"
}
}

View File

@ -0,0 +1,6 @@
{
"name": "@motajs/legacy-common",
"dependencies": {
"@motajs/common": "workspace:*"
}
}

View File

@ -1,9 +1,9 @@
import { EventEmitter } from './eventEmitter';
import { EventEmitter } from 'eventemitter3';
interface DisposableEvent<T> {
active: (value: T) => void;
dispose: (value: T) => void;
destroy: () => void;
active: [value: T];
dispose: [value: T];
destroy: [];
}
export class Disposable<T> extends EventEmitter<DisposableEvent<T>> {

View File

@ -0,0 +1,209 @@
import { isNil } from 'lodash-es';
export type Callable = (...params: any) => any;
export interface Listener<T extends (...params: any) => any> {
fn: T;
once?: boolean;
immediate?: boolean;
}
export interface ListenerOptions {
once: boolean;
immediate: boolean;
}
type EmitFn<F extends (...params: any) => any> = (
events: Listener<F>[],
...params: Parameters<F>
) => any;
type Key = number | string | symbol;
export class EventEmitter<T extends Record<keyof T, Callable> = {}> {
protected events: {
[x: Key]: Listener<Callable>[];
} = {};
private emitted: Set<string> = new Set();
protected emitter: {
[x: Key]: EmitFn<Callable> | undefined;
} = {};
/**
*
* @param event
* @param fn
* @param options
*/
on<K extends keyof T>(
event: K,
fn: T[K],
options?: Partial<ListenerOptions>
): void;
on(event: string, fn: Callable, options?: Partial<ListenerOptions>): void;
on(event: string, fn: Callable, options?: Partial<ListenerOptions>): void {
if (options?.immediate && this.emitted.has(event)) {
fn();
if (!options.once) {
this.events[event] ??= [];
this.events[event]?.push({
fn
});
}
return;
}
this.events[event] ??= [];
this.events[event]?.push({
fn,
once: options?.once
});
}
/**
*
* @param event
* @param fn
*/
off<K extends keyof T>(event: K, fn: T[K]): void;
off(event: string, fn: Callable): void;
off(event: string, fn: Callable): void {
const index = this.events[event]?.findIndex(v => v.fn === fn);
if (index === -1 || index === void 0) return;
this.events[event]?.splice(index, 1);
}
/**
*
* @param event
* @param fn
*/
once<K extends keyof T>(event: K, fn: T[K]): void;
once(event: string, fn: Callable): void;
once(event: string, fn: Callable): void {
this.on(event, fn, { once: true });
}
/**
*
* @param event
* @param params
*/
emit<K extends keyof T>(
event: K,
...params: Parameters<T[K]>
): ReturnType<T[K]>[];
emit(event: string, ...params: any[]): any[];
emit<K extends keyof T, R>(event: K, ...params: Parameters<T[K]>): R;
emit<R>(event: string, ...params: any[]): R;
emit(event: string, ...params: any[]): any[] {
this.emitted.add(event);
const events = (this.events[event] ??= []);
if (this.emitter[event]) {
const returns = this.emitter[event]!(events, ...params);
this.events[event] = events.filter(v => !v.once);
return returns;
} else {
const returns: ReturnType<Callable>[] = [];
for (let i = 0; i < events.length; i++) {
const e = events[i];
returns.push(e.fn(...(params as any)));
}
this.events[event] = events.filter(v => !v.once);
return returns;
}
}
/**
* (emitter)
* @param event
* @param fn
*/
// @ts-expect-error 无法推导
setEmitter<K extends keyof T>(event: K, fn?: EmitFn<T[K]>): void;
setEmitter(event: string, fn?: EmitFn<Callable>): void;
setEmitter(event: string, fn?: EmitFn<Callable>): void {
this.emitter[event] = fn;
}
/**
*
*/
removeAllListeners(): void;
/**
*
* @param event
*/
removeAllListeners(event: keyof T): void;
/**
*
* @param event
*/
removeAllListeners(event: string): void;
removeAllListeners(event?: string | keyof T) {
if (!isNil(event)) this.events[event] = [];
else this.events = {};
}
}
type IndexedSymbol = number | string | symbol;
export class IndexedEventEmitter<
T extends Record<keyof T, Callable>
> extends EventEmitter<T> {
private fnMap: {
[P in keyof T]?: Map<IndexedSymbol, T[P]>;
} = {};
/**
*
* @param event
* @param symbol
* @param fn
* @param options
*/
onIndex<K extends keyof T>(
event: K,
symbol: IndexedSymbol,
fn: T[K],
options: Partial<ListenerOptions>
) {
const map = this.ensureMap(event);
if (map.has(symbol)) {
console.warn(
`监听${String(event)}出错:已存在标识符为${String(
symbol
)}`
);
this.offIndex(event, symbol);
}
map.set(symbol, fn);
this.on(event, fn, options);
}
/**
*
* @param event
* @param symbol
* @param fn
*/
onceIndex<K extends keyof T>(event: K, symbol: IndexedSymbol, fn: T[K]) {
this.onIndex(event, symbol, fn, { once: true });
}
/**
*
* @param event
* @param symbol
*/
offIndex<K extends keyof T>(event: K, symbol: IndexedSymbol) {
const map = this.ensureMap(event);
const fn = map.get(symbol);
if (!fn) return;
this.off(event, fn);
}
private ensureMap<K extends keyof T>(event: K) {
return this.fnMap[event] ?? new Map<IndexedSymbol, T[K]>();
}
}

View File

@ -0,0 +1,3 @@
export * from './patch';
export * from './disposable';
export * from './eventEmitter';

View File

@ -0,0 +1,6 @@
{
"name": "@motajs/legacy-data",
"dependencies": {
"@motajs/common": "workspace:*"
}
}

View File

@ -1,75 +0,0 @@
import { EventEmitter } from 'eventemitter3';
interface MonoStoreEvent<T> {
change: [before: T | undefined, after: T | undefined];
}
/**
* 使
*
*/
export class MonoStore<T> extends EventEmitter<MonoStoreEvent<T>> {
list: Map<string, T> = new Map();
using?: T;
usingId?: string;
/**
* 使id的数据
* @param id 使id
*/
use(id: string) {
const before = this.using;
this.using = this.list.get(id);
this.usingId = id;
this.emit('change', before, this.using);
}
static toJSON(data: MonoStore<any>) {
return JSON.stringify({
now: data.usingId,
data: [...data.list]
});
}
static fromJSON<T>(data: string): MonoStore<T> {
const d = JSON.parse(data);
const arr: [string, T][] = d.data;
const store = new MonoStore<T>();
arr.forEach(([key, value]) => {
store.list.set(key, value);
});
if (d.now) store.use(d.now);
return store;
}
}
export namespace SerializeUtils {
interface StructSerializer<T> {
toJSON(data: T): string;
fromJSON(data: string): T;
}
/**
* Map键值对序列化函数使Map作为state使
*/
export const mapSerializer: StructSerializer<Map<any, any>> = {
toJSON(data) {
return JSON.stringify([...data]);
},
fromJSON(data) {
return new Map(JSON.parse(data));
}
};
/**
* Set集合序列化函数使Set作为state使
*/
export const setSerializer: StructSerializer<Set<any>> = {
toJSON(data) {
return JSON.stringify([...data]);
},
fromJSON(data) {
return new Set(JSON.parse(data));
}
};
}

View File

@ -101,7 +101,7 @@ export class EventEmitter<T extends Record<keyof T, Callable> = {}> {
emit(event: string, ...params: any[]): any[] {
this.emitted.add(event);
const events = (this.events[event] ??= []);
if (!!this.emitter[event]) {
if (this.emitter[event]) {
const returns = this.emitter[event]!(events, ...params);
this.events[event] = events.filter(v => !v.once);
return returns;
@ -121,7 +121,7 @@ export class EventEmitter<T extends Record<keyof T, Callable> = {}> {
* @param event
* @param fn
*/
// @ts-ignore
// @ts-expect-error 无法推导
setEmitter<K extends keyof T>(event: K, fn?: EmitFn<T[K]>): void;
setEmitter(event: string, fn?: EmitFn<Callable>): void;
setEmitter(event: string, fn?: EmitFn<Callable>): void {

View File

@ -1,8 +1,7 @@
import axios, { AxiosRequestConfig, ResponseType } from 'axios';
import { Disposable } from './disposable';
import { Disposable, EventEmitter } from '@motajs/legacy-common';
import { logger } from '@motajs/common';
import JSZip from 'jszip';
import { EventEmitter } from './eventEmitter';
type ProgressFn = (now: number, total: number) => void;

View File

@ -55,7 +55,6 @@ import EnemySpecial from '@/panel/enemySpecial.vue';
import EnemyTarget from '@/panel/enemyTarget.vue';
import KeyboardPanel from '@/panel/keyboard.vue';
import { MCGenerator } from './main/layout';
import { ResourceController } from './loader/controller';
import { logger } from '@motajs/common';
import { Danmaku } from './main/custom/danmaku';
import * as Shadow from './fx/shadow';
@ -75,7 +74,6 @@ Mota.register('class', 'MotaSetting', MotaSetting);
Mota.register('class', 'SettingDisplayer', SettingDisplayer);
Mota.register('class', 'UiController', UiController);
Mota.register('class', 'MComponent', MComponent);
Mota.register('class', 'ResourceController', ResourceController);
Mota.register('class', 'Danmaku', Danmaku);
// ----- 函数注册
Mota.register('fn', 'm', m);

View File

@ -1,30 +0,0 @@
import { EventEmitter } from '../common/eventEmitter';
interface ResourceControllerEvent<D = any, T = D> {
add: (uri: string, data: D) => void;
delete: (uri: string, content: T) => void;
}
export abstract class ResourceController<
D,
T = D
> extends EventEmitter<ResourceControllerEvent> {
list: Record<string, T> = {};
/**
*
* @param uri uri
* @param data
*/
abstract add(uri: string, data: D): void;
/**
*
* @param uri uri
*/
remove(uri: string) {
const content = this.list[uri];
delete this.list[uri];
// this.emit(uri, content);
}
}

View File

@ -1,5 +1,5 @@
import BoxAnimate from '@/components/boxAnimate.vue';
import { EventEmitter } from '@/core/common/eventEmitter';
import { EventEmitter } from 'eventemitter3';
import { logger } from '@motajs/common';
import { ResponseBase } from '@/core/interface';
import {
@ -53,9 +53,9 @@ interface PostLikeResponse extends ResponseBase {
}
interface DanmakuEvent {
showStart: (danmaku: Danmaku) => void;
showEnd: (danmaku: Danmaku) => void;
like: (liked: boolean, danmaku: Danmaku) => void;
showStart: [danmaku: Danmaku];
showEnd: [danmaku: Danmaku];
like: [liked: boolean, danmaku: Danmaku];
}
type SpecContentFn = (content: string, type: string) => VNode;

View File

@ -1,4 +1,4 @@
import { EventEmitter, Listener } from '@/core/common/eventEmitter';
import { EventEmitter, Listener } from '@motajs/legacy-common';
import { KeyCode } from '@/plugin/keyCodes';
import { gameKey } from './hotkey';
import { unwarpBinary } from './hotkey';

View File

@ -1,4 +1,4 @@
import { EventEmitter } from '@/core/common/eventEmitter';
import { EventEmitter } from '@motajs/legacy-common';
import { deleteWith, has } from '@/plugin/utils';
import { Component, nextTick, reactive, shallowReactive } from 'vue';
import { fixedUi } from '../init/ui';
@ -11,7 +11,6 @@ import type {
ToolbarItemMap,
ToolbarItemType
} from '../init/toolbar';
import { isMobile } from '@/plugin/use';
interface CustomToolbarEvent {
add: (item: ValueOf<ToolbarItemMap>) => void;

View File

@ -1,8 +1,6 @@
import { Component, shallowReactive } from 'vue';
import { Callable, EventEmitter } from '../../common/eventEmitter';
import { KeyCode } from '@/plugin/keyCodes';
import { EventEmitter } from '@motajs/legacy-common';
import { Hotkey } from './hotkey';
import { generateBinary } from '@/plugin/utils';
interface FocusEvent<T> {
focus: (before: T | null, after: T) => void;

View File

@ -1,5 +1,5 @@
import { FunctionalComponent, reactive } from 'vue';
import { EventEmitter } from '../common/eventEmitter';
import { EventEmitter } from '@motajs/legacy-common';
import { GameStorage } from './storage';
import { has, triggerFullscreen } from '@/plugin/utils';
import { createSettingComponents } from './init/settings';

View File

@ -1,13 +1,12 @@
import './system';
import '../plugin/game/index';
import * as damage from './enemy/damage';
import { EventEmitter, IndexedEventEmitter } from '@/core/common/eventEmitter';
import { EventEmitter, IndexedEventEmitter } from '@motajs/legacy-common';
import { specials } from './enemy/special';
import { gameListener, hook, loading } from './game';
import * as battle from './enemy/battle';
import * as hero from './state/hero';
import * as miscMechanism from './mechanism/misc';
import { registerPresetState } from './state/preset';
import { ItemState } from './state/item';
import {
BlockMover,
@ -52,8 +51,6 @@ loading.once('coreInit', () => {
Mota.Plugin.init();
});
registerPresetState();
window.addEventListener('beforeunload', () => {
core.checkAutosave();
});

View File

@ -1,64 +0,0 @@
import { MonoStore } from '@/common/struct';
import { GameState } from './state';
import { Hero, HeroState } from './hero';
export function registerPresetState() {
GameState.register<MonoStore<Hero<any>>>('hero', heroToJSON, heroFromJSON);
}
interface HeroSave {
x: number;
y: number;
floorId: FloorIds;
id: string;
items: [AllIdsOf<'items'>, number][];
state: {
status: any;
buffable: (string | number | symbol)[];
buffMap: [string | number | symbol, number][];
};
}
interface HeroSerializable {
now: string | null;
saves: HeroSave[];
}
function heroToJSON(data: MonoStore<Hero<any>>): string {
const now = data.usingId ?? null;
const saves: HeroSave[] = [...data.list.values()].map(v => {
return {
x: v.x,
y: v.y,
floorId: v.floorId,
id: v.id,
items: [...v.items],
state: {
status: v.state.status,
buffable: [...v.state.buffable],
buffMap: [...v.state.buffMap]
}
};
});
const obj: HeroSerializable = {
now,
saves
};
return JSON.stringify(obj);
}
function heroFromJSON(data: string): MonoStore<Hero<any>> {
const obj: HeroSerializable = JSON.parse(data);
const store = new MonoStore<Hero<any>>();
const saves: [string, Hero<any>][] = obj.saves.map(v => {
const state = new HeroState(v.state.status);
v.state.buffable.forEach(v => state.buffable.add(v));
v.state.buffMap.forEach(v => state.buffMap.set(v[0], v[1]));
const hero = new Hero(v.id, v.x, v.y, v.floorId, state);
v.items.forEach(v => hero.items.set(v[0], v[1]));
return [hero.id, hero];
});
store.list = new Map(saves);
if (obj.now) store.use(obj.now);
return store;
}

View File

@ -1,8 +1,8 @@
import type { Disposable } from '@/core/common/disposable';
import type {
Disposable,
EventEmitter,
IndexedEventEmitter
} from '@/core/common/eventEmitter';
} from '@motajs/legacy-common';
import type { loading } from './game';
import type { Hotkey } from '@/core/main/custom/hotkey';
import type { Keyboard } from '@/core/main/custom/keyboard';

View File

@ -1,4 +1,4 @@
import { Patch, PatchClass } from '@/common/patch';
import { Patch, PatchClass } from '@motajs/legacy-common';
import { audioPlayer, bgmController, soundPlayer } from '../audio';
import { mainSetting } from '@/core/main/setting';
import { sleep } from 'mutate-animate';

View File

@ -1,4 +1,4 @@
import { Patch } from '@/common/patch';
import { Patch } from '@motajs/legacy-common';
import { patchAudio } from './audio';
import { patchWeather } from './weather';
import { patchUI } from './ui';

View File

@ -1,4 +1,4 @@
import { Patch, PatchClass } from '@/common/patch';
import { Patch, PatchClass } from '@motajs/legacy-common';
import { TipStore } from '../render/components/tip';
export function patchUI() {

View File

@ -1,4 +1,4 @@
import { Patch, PatchClass } from '@/common/patch';
import { Patch, PatchClass } from '@motajs/legacy-common';
import { WeatherController } from '../weather';
import { isNil } from 'lodash-es';

View File

@ -5,6 +5,9 @@
"@motajs/common": "workspace:*",
"@motajs/render": "workspace:*",
"@motajs/system": "workspace:*",
"@motajs/system-ui": "workspace:*"
"@motajs/system-ui": "workspace:*",
"@motajs/legacy-common": "workspace:*",
"@motajs/legacy-client": "workspace:*",
"@motajs/legacy-data": "workspace:*"
}
}