ginka-generator/data/src/utils.ts

202 lines
6.1 KiB
TypeScript

import { readFile } from 'fs-extra';
import { join } from 'path';
import { BaseConfig, GinkaConfig, TowerInfo } from './types';
import { convertFloor } from './floor';
export interface DatasetMergable<T> {
datasetId: number;
data: Record<string, T>;
}
export interface FloorData {
map: number[][];
id: string;
config: BaseConfig;
}
export function mergeDataset<T>(
allowDuplicateKeys: boolean,
...datasets: DatasetMergable<T>[]
): DatasetMergable<T> {
if (datasets.length === 1) {
return datasets[0];
}
const usedKeys = new Set<string>();
const data: Record<string, T> = {};
datasets.forEach(v => {
for (const [key, value] of Object.entries(v.data)) {
if (usedKeys.has(key) && allowDuplicateKeys) {
const dataKey = `${v.datasetId}/${key}`;
data[dataKey] = value;
usedKeys.add(dataKey);
} else {
data[key] = value;
usedKeys.add(key);
}
}
});
const dataset: DatasetMergable<T> = {
datasetId: Math.floor(Math.random() * 1e12),
data: data
};
return dataset;
}
export function cosineSimilarity(vecA: number[], vecB: number[]): number {
if (vecA.length !== vecB.length) {
throw new Error('Vectors must have same dimension');
}
let dot = 0,
normA = 0,
normB = 0;
for (let i = 0; i < vecA.length; i++) {
dot += vecA[i] * vecB[i];
normA += vecA[i] ** 2;
normB += vecB[i] ** 2;
}
return dot / (Math.sqrt(normA) * Math.sqrt(normB));
}
export async function parseTowerInfo(
path: string,
configName: string
): Promise<TowerInfo> {
const dataFile = await readFile(join(path, 'data.js'), 'utf-8');
const data: any = JSON.parse(dataFile.split('\n').slice(1).join('\n'));
const configFile = await readFile(join(path, configName), 'utf-8');
return {
path: path,
name: data.firstData.name as string,
floorIds: data.main.floorIds as string[],
config: JSON.parse(configFile) as BaseConfig
};
}
export async function getAllFloors(...info: TowerInfo[]) {
const floorData = await Promise.all(
info.map(async tower => {
// 获取必要信息
const enemyFile = await readFile(
join(tower.path, 'enemys.js'),
'utf-8'
);
const mapFile = await readFile(
join(tower.path, 'maps.js'),
'utf-8'
);
const enemyMap = JSON.parse(
enemyFile.split('\n').slice(1).join('\n')
) as Record<string, any>;
const mapData = JSON.parse(
mapFile.split('\n').slice(1).join('\n')
) as Record<number, any>;
const enemyNumMap: Record<number, any> = {};
// 将怪物转化为数字映射
for (const [key, value] of Object.entries(mapData)) {
if (value.cls === 'enemys') {
enemyNumMap[parseInt(key)] = enemyMap[value.id];
}
}
return Promise.all(
tower.floorIds.map(async id => {
const floorFile = await readFile(
join(tower.path, 'floors', `${id}.js`),
'utf-8'
);
try {
const data = JSON.parse(
floorFile
// .replaceAll("'", '"')
.slice(floorFile.indexOf('=') + 1)
);
const map = data.map as number[][];
// 裁剪地图
const { clip } = tower.config;
const area = clip.special[id] ?? clip.defaults;
return convertFloor(
map,
area,
tower.config as GinkaConfig,
enemyNumMap
);
} catch (e) {
console.log(
`Error when processing '${tower.name}' '${id}'`
);
throw e;
}
})
);
})
);
const maps: Map<string, FloorData> = new Map();
floorData.forEach((tower, tid) => {
const name = info[tid].name;
tower.forEach((map, mid) => {
const floorId = info[tid].floorIds[mid];
maps.set(`${name}::${floorId}`, {
map,
id: floorId,
config: info[tid].config
});
});
});
return maps;
}
export function mergeFloorIds(...info: TowerInfo[]) {
const ids: string[] = [];
info.forEach(v => {
ids.push(...v.floorIds.map(id => `${v.name}:${id}`));
});
return ids;
}
export async function readOne(path: string) {
if (path.endsWith('.json')) {
return fromJSON(path);
} else {
return getAllFloors(await parseTowerInfo(path, 'minamo-config.json'));
}
}
export async function fromJSON(path: string) {
const file = await readFile(path, 'utf-8');
const data = JSON.parse(file) as Record<string, number[][]>;
const clip: Record<string, [number, number, number, number]> = {};
const config: BaseConfig = {
clip: {
defaults: [0, 0, 0, 0],
special: clip
}
};
const name = (Math.random() * 12).toFixed(0);
const floorMap = new Map<string, FloorData>();
for (const [key, value] of Object.entries(data)) {
const floorData: FloorData = {
map: value,
id: key,
config
};
floorMap.set(`${name}:${key}`, floorData);
}
return floorMap;
}
export function chooseFrom<T>(arr: T[], n: number): T[] {
const copy = arr.slice();
for (let i = copy.length - 1; i > 0; i--) {
let randIndex = Math.floor(Math.random() * (i + 1));
[copy[i], copy[randIndex]] = [copy[randIndex], copy[i]];
}
return copy.slice(0, n);
}