mirror of
				https://github.com/unanmed/HumanBreak.git
				synced 2025-10-25 15:52:59 +08:00 
			
		
		
		
	重写资源分离
This commit is contained in:
		
							parent
							
								
									6ecf201beb
								
							
						
					
					
						commit
						ec0a7f6785
					
				
							
								
								
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @ -36,3 +36,4 @@ index.cjs | |||||||
| _bundle | _bundle | ||||||
| out | out | ||||||
| dist-resource | dist-resource | ||||||
|  | _temp | ||||||
| @ -1,6 +1,7 @@ | |||||||
| interface MotaConfig { | interface MotaConfig { | ||||||
|     name: string; |     name: string; | ||||||
|     resourceName?: string; |     resourceName?: string; | ||||||
|  |     zip?: Record<string, string[]>; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| function defineConfig(config: MotaConfig): MotaConfig { | function defineConfig(config: MotaConfig): MotaConfig { | ||||||
| @ -10,5 +11,16 @@ function defineConfig(config: MotaConfig): MotaConfig { | |||||||
| export default defineConfig({ | export default defineConfig({ | ||||||
|     // 这里修改塔的name,请保持与全塔属性的完全相同,否则发布之后可能无法进行游玩
 |     // 这里修改塔的name,请保持与全塔属性的完全相同,否则发布之后可能无法进行游玩
 | ||||||
|     name: 'HumanBreak', |     name: 'HumanBreak', | ||||||
|     resourceName: 'HumanBreakRes' |     resourceName: 'HumanBreakRes', | ||||||
|  |     zip: { | ||||||
|  |         'resource.zip': [ | ||||||
|  |             'autotiles/*', | ||||||
|  |             'tilesets/*', | ||||||
|  |             'materials/*', | ||||||
|  |             'images/*', | ||||||
|  |             'animates/*', | ||||||
|  |             'sounds/*', | ||||||
|  |             'fonts/*' | ||||||
|  |         ] | ||||||
|  |     } | ||||||
| }); | }); | ||||||
|  | |||||||
| @ -5,9 +5,9 @@ | |||||||
|   "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 0 1 1", |     "build": "vue-tsc && vite build && ts-node-esm script/build.ts dist", | ||||||
|     "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 gh", | ||||||
|     "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 local", | ||||||
|     "preview": "vite preview", |     "preview": "vite preview", | ||||||
|     "update": "ts-node-esm script/update.ts", |     "update": "ts-node-esm script/update.ts", | ||||||
|     "declare": "ts-node-esm script/declare.ts", |     "declare": "ts-node-esm script/declare.ts", | ||||||
|  | |||||||
| @ -220,7 +220,7 @@ function main() { | |||||||
| 
 | 
 | ||||||
|     // 远程资源地址,在线游戏中,塔本体不包含任何资源,只包含源码,从而可以降低游戏本体的体积并平均分担资源包体积
 |     // 远程资源地址,在线游戏中,塔本体不包含任何资源,只包含源码,从而可以降低游戏本体的体积并平均分担资源包体积
 | ||||||
|     // 从而可以优化加载并避免网站发布的大小限制
 |     // 从而可以优化加载并避免网站发布的大小限制
 | ||||||
|     this.USE_RESOURCE = false; |     this.RESOURCE_TYPE = 'dev'; | ||||||
|     this.RESOURCE_URL = ''; |     this.RESOURCE_URL = ''; | ||||||
|     this.RESOURCE_SYMBOL = ''; |     this.RESOURCE_SYMBOL = ''; | ||||||
|     this.RESOURCE_INDEX = {}; |     this.RESOURCE_INDEX = {}; | ||||||
|  | |||||||
| @ -11,9 +11,10 @@ import commonjs from '@rollup/plugin-commonjs'; | |||||||
| import { splitResorce } from './resource.js'; | import { splitResorce } from './resource.js'; | ||||||
| import compressing from 'compressing'; | import compressing from 'compressing'; | ||||||
| 
 | 
 | ||||||
| const map = !!Number(process.argv[2]); | const type = process.argv[2]; | ||||||
| const resorce = !!Number(process.argv[3]); | const map = false; | ||||||
| const compress = !!Number(process.argv[4]); | const resorce = type !== 'dev'; | ||||||
|  | const compress = type === 'dist'; | ||||||
| 
 | 
 | ||||||
| (async function () { | (async function () { | ||||||
|     const timestamp = Date.now(); |     const timestamp = Date.now(); | ||||||
| @ -175,7 +176,7 @@ const compress = !!Number(process.argv[4]); | |||||||
| 
 | 
 | ||||||
|     // 6. 资源分离
 |     // 6. 资源分离
 | ||||||
|     if (resorce) { |     if (resorce) { | ||||||
|         await splitResorce(compress); |         await splitResorce(type); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     // 7. 压缩
 |     // 7. 压缩
 | ||||||
|  | |||||||
| @ -1,174 +1,231 @@ | |||||||
| import fs from 'fs-extra'; | import fs from 'fs-extra'; | ||||||
| import { uniqueSymbol } from './utils.js'; | import { uniqueSymbol } from './utils.js'; | ||||||
| import { basename, extname, resolve } from 'path'; | import { resolve } from 'path'; | ||||||
| import { dirname } from 'path'; |  | ||||||
| import motaConfig from '../mota.config.js'; | import motaConfig from '../mota.config.js'; | ||||||
| 
 | import compressing from 'compressing'; | ||||||
| type ResorceType = |  | ||||||
|     | 'bgms' |  | ||||||
|     | 'sounds' |  | ||||||
|     | 'autotiles' |  | ||||||
|     | 'images' |  | ||||||
|     | 'materials' |  | ||||||
|     | 'tilesets' |  | ||||||
|     | 'animates' |  | ||||||
|     | 'fonts'; |  | ||||||
| 
 | 
 | ||||||
| const SYMBOL = uniqueSymbol(); | const SYMBOL = uniqueSymbol(); | ||||||
| const MAX_SIZE = 100 * (1 << 20) - 20 * (1 << 10); | const MAX_SIZE = 100 * (1 << 20) - 20 * (1 << 10); | ||||||
| const baseDir = './dist'; | const sourceIndex: Record<string, string> = {}; | ||||||
| 
 | const toMove: Stats[] = []; | ||||||
| let totalSize = 0; | const all = [ | ||||||
|  |     'bgms', | ||||||
|  |     'sounds', | ||||||
|  |     'autotiles', | ||||||
|  |     'images', | ||||||
|  |     'materials', | ||||||
|  |     'tilesets', | ||||||
|  |     'animates', | ||||||
|  |     'fonts' | ||||||
|  | ]; | ||||||
| 
 | 
 | ||||||
| type Stats = fs.Stats & { name?: string }; | type Stats = fs.Stats & { name?: string }; | ||||||
| 
 | 
 | ||||||
| export async function splitResorce(compress: boolean = false) { | export async function splitResorce(type: string) { | ||||||
|     await fs.ensureDir('./dist-resource'); |     await fs.ensureDir('./dist-resource'); | ||||||
|     await fs.emptyDir('./dist-resource'); |     await fs.emptyDir('./dist-resource'); | ||||||
|     const folder = await fs.stat('./dist'); |     await readySplit(); | ||||||
|     totalSize = folder.size; |  | ||||||
|     if (totalSize < MAX_SIZE) return; |  | ||||||
| 
 | 
 | ||||||
|     await doSplit(compress); |     await zipResource(); | ||||||
|  |     await split(type === 'dist' ? MAX_SIZE : void 0); | ||||||
|  | 
 | ||||||
|  |     await endSplit(type); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| async function sortDir(dir: string, ext?: string[]) { | async function readySplit() { | ||||||
|     const path = await fs.readdir(dir); |     await fs.ensureDir('./_temp'); | ||||||
|     const stats: Stats[] = []; |     await fs.emptyDir('./_temp'); | ||||||
| 
 |     await fs.ensureDir('./_temp/origin'); | ||||||
|     for await (const one of path) { |     await copyAll(); | ||||||
|         if (ext && !ext.includes(extname(one))) continue; |  | ||||||
|         if (one === 'bg.jpg') continue; |  | ||||||
|         const stat = await fs.stat(resolve(dir, one)); |  | ||||||
|         if (!stat.isFile()) continue; |  | ||||||
|         const status: Stats = { |  | ||||||
|             ...stat, |  | ||||||
|             name: one |  | ||||||
|         }; |  | ||||||
|         stats.push(status); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     return stats.sort((a, b) => b.size - a.size); |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| async function calResourceSize() { | async function endSplit(type: string) { | ||||||
|     return ( |     await rewriteMain(type); | ||||||
|         await Promise.all( |     await fs.emptyDir('./_temp'); | ||||||
|             [ |     await fs.rmdir('./_temp'); | ||||||
|                 'animates', |  | ||||||
|                 'autotiles', |  | ||||||
|                 'bgms', |  | ||||||
|                 'fonts', |  | ||||||
|                 'images', |  | ||||||
|                 'materials', |  | ||||||
|                 'sounds', |  | ||||||
|                 'tilesets' |  | ||||||
|             ].map(v => fs.stat(resolve('./dist/project/', v))) |  | ||||||
|         ) |  | ||||||
|     ).reduce((pre, cur) => pre + cur.size, 0); |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| async function doSplit(compress: boolean) { | async function zipResource() { | ||||||
|     let size = await calResourceSize(); |     const zip = motaConfig.zip; | ||||||
|     await fs.emptyDir('./dist-resource'); |     if (!zip) return; | ||||||
|     const priority: ResorceType[] = [ |     for await (const [name, files] of Object.entries(zip)) { | ||||||
|         'materials', |         const stream = new compressing.zip.Stream(); | ||||||
|         'tilesets', |         const dirs: string[] = []; | ||||||
|         'autotiles', |  | ||||||
|         'animates', |  | ||||||
|         'images', |  | ||||||
|         'sounds', |  | ||||||
|         'fonts', |  | ||||||
|         'bgms' |  | ||||||
|     ]; |  | ||||||
|     const dirInfo: Record<ResorceType, Stats[]> = Object.fromEntries( |  | ||||||
|         await Promise.all( |  | ||||||
|             priority.map(async v => [ |  | ||||||
|                 v, |  | ||||||
|                 await sortDir(resolve(baseDir, 'project', v)) |  | ||||||
|             ]) |  | ||||||
|         ) |  | ||||||
|     ); |  | ||||||
| 
 | 
 | ||||||
|     let currSize = 0; |         for await (const file of files) { | ||||||
|     const length = Object.fromEntries( |             if (/^.+\/\*$/.test(file)) { | ||||||
|         priority.map(v => [v, dirInfo[v].length]) |                 const dir = file.split('/')[0]; | ||||||
|     ); |                 dirs.push(dir); | ||||||
|     const soruceIndex: Record<string, number> = {}; |                 await fs.copy(`./_temp/origin/${dir}`, `./_temp/${dir}`); | ||||||
|     const split = async (index: number): Promise<void> => { |  | ||||||
|         size -= currSize; |  | ||||||
|         currSize = 0; |  | ||||||
| 
 |  | ||||||
|         const cut: string[] = []; |  | ||||||
| 
 |  | ||||||
|         const mapped: ResorceType[] = []; |  | ||||||
|         while (1) { |  | ||||||
|             const toCut = priority.find( |  | ||||||
|                 v => dirInfo[v].length > 0 && !mapped.includes(v) |  | ||||||
|             ); |  | ||||||
|             if (!toCut) break; |  | ||||||
| 
 |  | ||||||
|             mapped.push(toCut); |  | ||||||
|             const l = dirInfo[toCut].length; |  | ||||||
|             const data: string[] = []; |  | ||||||
| 
 |  | ||||||
|             let pass = 0; |  | ||||||
|             while (1) { |  | ||||||
|                 const stats = dirInfo[toCut]; |  | ||||||
|                 const stat = stats[pass]; |  | ||||||
|                 if (!stat) { |  | ||||||
|                     break; |  | ||||||
|                 } |  | ||||||
|                 if (currSize + stat.size >= MAX_SIZE) { |  | ||||||
|                     pass++; |  | ||||||
|                     continue; |  | ||||||
|                 } |  | ||||||
|                 data.push(`${toCut}/${stat.name}`); |  | ||||||
|                 stats.splice(pass, 1); |  | ||||||
|                 currSize += stat.size; |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             if (l === length[toCut] && dirInfo[toCut].length === 0) { |  | ||||||
|                 soruceIndex[`${toCut}.*`] = index; |  | ||||||
|             } else { |             } else { | ||||||
|                 data.map(v => (soruceIndex[v] = index)); |                 const [dir, name] = file.split('/'); | ||||||
|             } |                 if (dirs.includes(dir)) dirs.push(dir); | ||||||
|             cut.push(...data); |                 await fs.ensureDir(`./_temp/${dir}`); | ||||||
|         } |                 await fs.copyFile( | ||||||
| 
 |                     `./_temp/origin/${dir}/${name}`, | ||||||
|         const dir = `./dist-resource/${index}`; |                     `./_temp/${dir}/${name}` | ||||||
|         await fs.ensureDir(dir); |  | ||||||
|         await Promise.all(priority.map(v => fs.ensureDir(resolve(dir, v)))); |  | ||||||
| 
 |  | ||||||
|         await Promise.all( |  | ||||||
|             cut.map(v => |  | ||||||
|                 fs.move( |  | ||||||
|                     resolve('./dist/project', v), |  | ||||||
|                     resolve( |  | ||||||
|                         dir, |  | ||||||
|                         dirname(v), |  | ||||||
|                         `${basename(v).split('.')[0]}-${SYMBOL}${extname(v)}` |  | ||||||
|                     ) |  | ||||||
|                 ) |  | ||||||
|             ) |  | ||||||
|                 ); |                 ); | ||||||
|  |             } | ||||||
|  |         } | ||||||
| 
 | 
 | ||||||
|         // 生成可发布结构
 |         dirs.forEach(v => stream.addEntry(`./_temp/${v}`)); | ||||||
|         await generatePublishStructure(dir, index); |         const dest = fs.createWriteStream(`./_temp/${name}`); | ||||||
| 
 |         await new Promise<void>(res => | ||||||
|         if (Object.values(dirInfo).every(v => v.length === 0)) return; |             stream.pipe(dest).on('finish', () => { | ||||||
|         else return split(index + 1); |                 res(); | ||||||
|     }; |             }) | ||||||
| 
 |         ); | ||||||
|     await split(0); |         const stat = await fs.stat(`./_temp/${name}`); | ||||||
| 
 |         toMove.push({ ...stat, name: `./_temp/${name}` }); | ||||||
|     await rewriteMain(soruceIndex); |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| async function rewriteMain(sourceIndex: Record<string, number>) { | async function getRemainReource() { | ||||||
|  |     const zip = motaConfig.zip; | ||||||
|  |     if (!zip) return; | ||||||
|  |     const values = Object.values(zip); | ||||||
|  |     for await (const one of all) { | ||||||
|  |         if (values.some(v => v.includes(`${one}/*`))) continue; | ||||||
|  |         const list = await fs.readdir(`./_temp/origin/${one}`); | ||||||
|  |         for await (const name of list) { | ||||||
|  |             if (!values.some(vv => vv.includes(`${one}/${name}`))) { | ||||||
|  |                 const stat = await fs.stat(`./_temp/origin/${one}/${name}`); | ||||||
|  |                 toMove.push({ | ||||||
|  |                     ...stat, | ||||||
|  |                     name: `./_temp/origin/${one}/${name}` | ||||||
|  |                 }); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |     toMove.sort((a, b) => { | ||||||
|  |         if (a.name?.endsWith('.zip') && b.name?.endsWith('.zip')) { | ||||||
|  |             return b.size - a.size; | ||||||
|  |         } | ||||||
|  |         if (a.name?.endsWith('.zip')) return -1; | ||||||
|  |         if (b.name?.endsWith('.zip')) return 1; | ||||||
|  |         return b.size - a.size; | ||||||
|  |     }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | async function split(max?: number) { | ||||||
|  |     await getRemainReource(); | ||||||
|  | 
 | ||||||
|  |     const doSplit = async (index: string | number) => { | ||||||
|  |         const base = | ||||||
|  |             typeof index === 'string' ? index : `./dist-resource/${index}`; | ||||||
|  | 
 | ||||||
|  |         await fs.ensureDir(base); | ||||||
|  |         await generatePublishStructure( | ||||||
|  |             base, | ||||||
|  |             typeof index === 'string' ? 0 : index | ||||||
|  |         ); | ||||||
|  | 
 | ||||||
|  |         let size = (await fs.stat(base)).size; | ||||||
|  |         // 计算出要移动多少资源
 | ||||||
|  |         const res = (() => { | ||||||
|  |             if (!max) return toMove.splice(0, toMove.length); | ||||||
|  |             let remain = max - size; | ||||||
|  |             for (let i = 0; i < toMove.length; i++) { | ||||||
|  |                 const ele = toMove[i]; | ||||||
|  |                 remain -= ele.size; | ||||||
|  |                 if (remain <= 0) { | ||||||
|  |                     return toMove.splice(0, i); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             return toMove.splice(0, toMove.length); | ||||||
|  |         })(); | ||||||
|  | 
 | ||||||
|  |         if (base.endsWith('dist')) { | ||||||
|  |             await fs.ensureDir(resolve(base, 'resource')); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         // 执行移动
 | ||||||
|  |         await Promise.all( | ||||||
|  |             res.map(async v => { | ||||||
|  |                 if (!v.name) return; | ||||||
|  |                 // 压缩包
 | ||||||
|  |                 if (v.name.endsWith('.zip')) { | ||||||
|  |                     const [, , name] = v.name.split('/'); | ||||||
|  |                     const split = name.split('.'); | ||||||
|  |                     const target = `${split | ||||||
|  |                         .slice(0, -1) | ||||||
|  |                         .join('.')}-${SYMBOL}.${split.at(-1)}`;
 | ||||||
|  |                     if (base.endsWith('dist')) { | ||||||
|  |                         await fs.ensureDir(resolve(base, 'resource/zip')); | ||||||
|  |                         return fs.copyFile( | ||||||
|  |                             v.name, | ||||||
|  |                             resolve(base, 'resource', 'zip', target) | ||||||
|  |                         ); | ||||||
|  |                     } else { | ||||||
|  |                         await fs.ensureDir(resolve(base, 'zip')); | ||||||
|  |                         return fs.copyFile( | ||||||
|  |                             v.name, | ||||||
|  |                             resolve(base, 'zip', target) | ||||||
|  |                         ); | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|  |                 // 非压缩包
 | ||||||
|  |                 if (!v.name.endsWith('.zip')) { | ||||||
|  |                     const [, , , type, name] = v.name.split('/'); | ||||||
|  |                     const split = name.split('.'); | ||||||
|  |                     const target = `${split | ||||||
|  |                         .slice(0, -1) | ||||||
|  |                         .join('.')}-${SYMBOL}.${split.at(-1)}`;
 | ||||||
|  |                     if (base.endsWith('dist')) { | ||||||
|  |                         await fs.ensureDir(resolve(base, 'resource', type)); | ||||||
|  |                     } else { | ||||||
|  |                         await fs.ensureDir(resolve(base, type)); | ||||||
|  |                     } | ||||||
|  |                     if (base.endsWith('dist')) { | ||||||
|  |                         return fs.copyFile( | ||||||
|  |                             v.name, | ||||||
|  |                             resolve(base, 'resource', type, target) | ||||||
|  |                         ); | ||||||
|  |                     } else { | ||||||
|  |                         return fs.copyFile(v.name, resolve(base, type, target)); | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |             }) | ||||||
|  |         ); | ||||||
|  | 
 | ||||||
|  |         // 标记资源索引
 | ||||||
|  |         res.forEach(v => { | ||||||
|  |             if (!v.name) return; | ||||||
|  |             // 压缩包
 | ||||||
|  |             if (v.name.endsWith('.zip')) { | ||||||
|  |                 const [, , name] = v.name.split('/'); | ||||||
|  |                 sourceIndex[`zip.${name}`] = index.toString(); | ||||||
|  |             } | ||||||
|  |             // 非压缩包
 | ||||||
|  |             if (!v.name.endsWith('.zip')) { | ||||||
|  |                 const [, , , type, name] = v.name.split('/'); | ||||||
|  |                 sourceIndex[`${type}.${name}`] = index.toString(); | ||||||
|  |             } | ||||||
|  |         }); | ||||||
|  | 
 | ||||||
|  |         if (toMove.length > 0) { | ||||||
|  |             await doSplit(typeof index === 'string' ? 0 : index + 1); | ||||||
|  |         } | ||||||
|  |     }; | ||||||
|  |     await doSplit('dist'); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | async function copyAll() { | ||||||
|  |     await Promise.all( | ||||||
|  |         all.map(v => { | ||||||
|  |             return fs.move(`./dist/project/${v}`, `./_temp/origin/${v}`); | ||||||
|  |         }) | ||||||
|  |     ); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | async function rewriteMain(type: string) { | ||||||
|     const main = await fs.readFile('./dist/main.js', 'utf-8'); |     const main = await fs.readFile('./dist/main.js', 'utf-8'); | ||||||
|     const res = main |     const res = main | ||||||
|         .replace(/this\.USE_RESOURCE\s*\=\s*false/, 'this.USE_RESOURCE = true') |         .replace( | ||||||
|  |             /this\.RESOURCE_TYPE\s*\=\s*.*;/, | ||||||
|  |             `this.RESOURCE_TYPE = '${type}';` | ||||||
|  |         ) | ||||||
|         .replace( |         .replace( | ||||||
|             /this\.RESOURCE_URL\s*\=\s*'.*'/, |             /this\.RESOURCE_URL\s*\=\s*'.*'/, | ||||||
|             `this.RESOURCE_URL = '/games/${motaConfig.resourceName}'` |             `this.RESOURCE_URL = '/games/${motaConfig.resourceName}'` | ||||||
| @ -184,32 +241,26 @@ async function rewriteMain(sourceIndex: Record<string, number>) { | |||||||
|     await fs.writeFile('./dist/main.js', res, 'utf-8'); |     await fs.writeFile('./dist/main.js', res, 'utf-8'); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | /** | ||||||
|  |  * 生成可发布目录 | ||||||
|  |  */ | ||||||
| async function generatePublishStructure(dir: string, index: number) { | async function generatePublishStructure(dir: string, index: number) { | ||||||
|     await fs.mkdir(resolve(dir, 'libs')); |     await fs.ensureDir(resolve(dir, 'libs')); | ||||||
|     await fs.mkdir(resolve(dir, 'libs/thirdparty')); |     await fs.ensureDir(resolve(dir, 'libs/thirdparty')); | ||||||
|     await fs.mkdir(resolve(dir, 'project')); |     await fs.ensureDir(resolve(dir, 'project')); | ||||||
|     await Promise.all( |     await Promise.all( | ||||||
|         [ |         all.map(v => { | ||||||
|             'autotiles', |             fs.ensureDir(resolve(dir, 'project', v)); | ||||||
|             'images', |             fs.emptyDir(resolve(dir, 'project', v)); | ||||||
|             'materials', |         }) | ||||||
|             'animates', |  | ||||||
|             'fonts', |  | ||||||
|             'floors', |  | ||||||
|             'tilesets', |  | ||||||
|             'sounds', |  | ||||||
|             'bgms' |  | ||||||
|         ].map(v => fs.mkdir(resolve(dir, 'project', v))) |  | ||||||
|     ); |     ); | ||||||
| 
 | 
 | ||||||
|  |     if (!dir.endsWith('dist')) { | ||||||
|         await fs.writeFile( |         await fs.writeFile( | ||||||
|             resolve(dir, 'project/icons.js'), |             resolve(dir, 'project/icons.js'), | ||||||
|             `var icons_4665ee12_3a1f_44a4_bea3_0fccba634dc1 = 
 |             `var icons_4665ee12_3a1f_44a4_bea3_0fccba634dc1 = 
 | ||||||
| { | {"autotile": {}} | ||||||
|     "autotile": { | `,
 | ||||||
| 
 |  | ||||||
|     } |  | ||||||
| }`,
 |  | ||||||
|             'utf-8' |             'utf-8' | ||||||
|         ); |         ); | ||||||
|         await fs.writeFile( |         await fs.writeFile( | ||||||
| @ -219,10 +270,7 @@ async function generatePublishStructure(dir: string, index: number) { | |||||||
|         ); |         ); | ||||||
|         await fs.writeFile(resolve(dir, 'libs/none.js'), '"none"', 'utf-8'); |         await fs.writeFile(resolve(dir, 'libs/none.js'), '"none"', 'utf-8'); | ||||||
| 
 | 
 | ||||||
|     await fs.copyFile( |         await fs.copyFile('./script/template/main.js', resolve(dir, 'main.js')); | ||||||
|         './script/template/main.js', |  | ||||||
|         resolve(dir, 'project/main.js') |  | ||||||
|     ); |  | ||||||
|         const data = await fs.readFile('./script/template/data.js', 'utf-8'); |         const data = await fs.readFile('./script/template/data.js', 'utf-8'); | ||||||
|         await fs.writeFile( |         await fs.writeFile( | ||||||
|             resolve(dir, 'project/data.js'), |             resolve(dir, 'project/data.js'), | ||||||
| @ -233,6 +281,7 @@ async function generatePublishStructure(dir: string, index: number) { | |||||||
|             './script/template/lz-string.min.js', |             './script/template/lz-string.min.js', | ||||||
|             resolve(dir, 'libs/thirdparty/lz-string.min.js') |             resolve(dir, 'libs/thirdparty/lz-string.min.js') | ||||||
|         ); |         ); | ||||||
|  |     } | ||||||
| 
 | 
 | ||||||
|     await Promise.all( |     await Promise.all( | ||||||
|         ['animates', 'images', 'materials', 'sounds', 'tilesets'].map(v => { |         ['animates', 'images', 'materials', 'sounds', 'tilesets'].map(v => { | ||||||
|  | |||||||
| @ -3,7 +3,11 @@ import { Resource, getTypeByResource } from './resource'; | |||||||
| 
 | 
 | ||||||
| const info = resource; | const info = resource; | ||||||
| 
 | 
 | ||||||
|  | /** | ||||||
|  |  * 构建游戏包后的加载 | ||||||
|  |  */ | ||||||
| export function readyAllResource() { | export function readyAllResource() { | ||||||
|  |     if (main.RESOURCE_TYPE === 'dev') return readyDevResource(); | ||||||
|     info.resource.forEach(v => { |     info.resource.forEach(v => { | ||||||
|         const type = getTypeByResource(v); |         const type = getTypeByResource(v); | ||||||
|         if (type === 'zip') { |         if (type === 'zip') { | ||||||
| @ -13,3 +17,8 @@ export function readyAllResource() { | |||||||
|         } |         } | ||||||
|     }); |     }); | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * 开发时的加载 | ||||||
|  |  */ | ||||||
|  | function readyDevResource() {} | ||||||
|  | |||||||
| @ -88,21 +88,34 @@ export class Resource< | |||||||
|         const name = (this.name = resolve.slice(1, -1).join('.')); |         const name = (this.name = resolve.slice(1, -1).join('.')); | ||||||
|         const ext = (this.ext = '.' + resolve.at(-1)); |         const ext = (this.ext = '.' + resolve.at(-1)); | ||||||
| 
 | 
 | ||||||
|         if (!main.USE_RESOURCE) { |         const distBase = import.meta.env.BASE_URL; | ||||||
|             return `/games/${core.data.firstData.name}/project/${type}/${name}${ext}`; |  | ||||||
|         } |  | ||||||
| 
 | 
 | ||||||
|         const base = main.RESOURCE_URL; |         const base = main.RESOURCE_URL; | ||||||
|         const indexes = main.RESOURCE_INDEX; |         const indexes = main.RESOURCE_INDEX; | ||||||
|         const symbol = main.RESOURCE_SYMBOL; |         const symbol = main.RESOURCE_SYMBOL; | ||||||
|  |         const t = main.RESOURCE_TYPE; | ||||||
| 
 | 
 | ||||||
|  |         if (t === 'dist') { | ||||||
|             if (has(indexes[`${type}.*`])) { |             if (has(indexes[`${type}.*`])) { | ||||||
|                 const i = indexes[`${type}.*`]; |                 const i = indexes[`${type}.*`]; | ||||||
|  |                 if (i !== 'dist') { | ||||||
|                     return `${base}${i}/${type}/${name}-${symbol}${ext}`; |                     return `${base}${i}/${type}/${name}-${symbol}${ext}`; | ||||||
|  |                 } else { | ||||||
|  |                     return `${distBase}resource/${type}/${name}-${symbol}${ext}`; | ||||||
|  |                 } | ||||||
|             } else { |             } else { | ||||||
|                 const i = indexes[`${type}.${name}${ext}`]; |                 const i = indexes[`${type}.${name}${ext}`]; | ||||||
|             const index = has(i) ? i : 0; |                 const index = has(i) ? i : '0'; | ||||||
|  |                 if (i !== 'dist') { | ||||||
|                     return `${base}${index}/${type}/${name}-${symbol}${ext}`; |                     return `${base}${index}/${type}/${name}-${symbol}${ext}`; | ||||||
|  |                 } else { | ||||||
|  |                     return `${distBase}resource/${type}/${name}-${symbol}${ext}`; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } else if (t === 'gh' || t === 'local') { | ||||||
|  |             return `${distBase}resource/${type}/${name}-${symbol}${ext}`; | ||||||
|  |         } else { | ||||||
|  |             return `${distBase}project/${type}/${name}${ext}`; | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  | |||||||
							
								
								
									
										2
									
								
								src/types/core.d.ts
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								src/types/core.d.ts
									
									
									
									
										vendored
									
									
								
							| @ -1259,7 +1259,7 @@ interface Main extends MainData { | |||||||
|     readonly RESOURCE_INDEX: Record<string, string>; |     readonly RESOURCE_INDEX: Record<string, string>; | ||||||
|     readonly RESOURCE_URL: string; |     readonly RESOURCE_URL: string; | ||||||
|     readonly RESOURCE_SYMBOL: string; |     readonly RESOURCE_SYMBOL: string; | ||||||
|     readonly USE_RESOURCE: boolean; |     readonly RESOURCE_TYPE: 'dev' | 'dist' | 'gh' | 'local'; | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|      * 初始化游戏 |      * 初始化游戏 | ||||||
|  | |||||||
| @ -19,8 +19,7 @@ | |||||||
|         "src/**/*.d.ts", |         "src/**/*.d.ts", | ||||||
|         "src/**/*.tsx", |         "src/**/*.tsx", | ||||||
|         "src/**/*.vue", |         "src/**/*.vue", | ||||||
|         "mota.config.ts", |         "mota.config.ts" | ||||||
|         "script/**/*.ts" |  | ||||||
|     ], |     ], | ||||||
|     "references": [{ "path": "./tsconfig.node.json" }] |     "references": [{ "path": "./tsconfig.node.json" }] | ||||||
| } | } | ||||||
|  | |||||||
| @ -4,7 +4,8 @@ | |||||||
|         "composite": true, |         "composite": true, | ||||||
|         "module": "ESNext", |         "module": "ESNext", | ||||||
|         "moduleResolution": "Node", |         "moduleResolution": "Node", | ||||||
|         "allowSyntheticDefaultImports": true |         "allowSyntheticDefaultImports": true, | ||||||
|  |         "strict": true | ||||||
|     }, |     }, | ||||||
|     "include": ["vite.config.ts", "mota.config.ts"] |     "include": ["vite.config.ts", "mota.config.ts", "script/**/*.ts"] | ||||||
| } | } | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user