import fs from 'fs/promises'; import fss from 'fs'; import fse from 'fs-extra'; import Fontmin from 'fontmin'; import * as babel from '@babel/core'; import * as rollup from 'rollup'; (async function () { // 1. 去除未使用的文件 const data = (() => { 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 { main: Record }; const main = data.main; try { const data = [ ['./dist/project/floors', '.js', 'floorIds'], ['./dist/project/bgms', '', 'bgms'], ['./dist/project/sounds', '', 'sounds'], ['./dist/project/images', '', 'images'], ['./dist/project/animates', '.animate', 'animates'], ['./dist/project/tilesets', '', 'tilesets'], ['./dist/project/fonts', '.ttf', 'fonts'] ]; await Promise.all( data.map(async v => { const all = await fs.readdir(`${v[0]}`); const data = main[v[2]].map(vv => vv + v[1]); all.forEach(async vv => { if (!data.includes(vv)) { await fs.rm(`${v[0]}/${vv}`); } }); }) ); if (process.argv[2] !== '1') await fse.remove('./dist/maps/'); // 在线查看什么都看不到,这编辑器难道还需要留着吗? await fse.remove('./dist/_server'); await fse.remove('./dist/editor.html'); await fse.remove('./dist/server.cjs'); } catch {} // 2. 压缩字体 try { // 获取要压缩的文字列表,libs & projects下的所有js文件 let texts = ``; const exclude = `\n \t`; const libs = await fs.readdir('./public/libs'); const project = await fs.readdir('./public/project'); const floors = await fs.readdir('./public/project/floors'); const assets = await fs.readdir('./dist/assets/'); const plugin = await fs.readdir('./public/project/plugin'); const all = [ ...libs.map(v => `./public/libs/${v}`), ...project.map(v => `./public/project/${v}`), ...floors.map(v => `./public/project/floors/${v}`), ...assets.map(v => `./dist/assets/${v}`), ...plugin.map(v => `./public/project/plugin/${v}`) ]; for await (const dir of all) { const stat = await fs.stat(dir); if (!stat.isFile()) continue; if (dir.endsWith('.ttf')) continue; const file = await fs.readFile(dir, 'utf-8'); for (let i = 0; i < file.length; i++) { const char = file[i]; if (!texts.includes(char) && !exclude.includes(char)) texts += char; } } // 获取所有字体(直接压缩字体会报错 const fonts = main.fonts; await Promise.all([ ...fonts.map(v => (async () => { const fontmin = new Fontmin(); fontmin .src(`./public/project/fonts/${v}.ttf`) .dest('./dist/project/fonts') .use( Fontmin.glyph({ text: texts }) ); await new Promise(res => { fontmin.run(err => { if (err) throw err; res(''); }); }); })() ) ]); } catch (e) { await fse.copy('./public/project/fonts', './dist/project/fonts'); } // 3. 压缩js插件 try { const build = await rollup.rollup({ input: 'public/project/plugin/index.js' }); const code = await build.generate({ format: 'iife', name: 'CorePlugin' }); const compressed = babel.transformSync(code.output[0].code)?.code!; await fs.writeFile('./dist/project/plugin.min.js', compressed, 'utf-8'); await fse.remove('./dist/project/plugin/'); } catch (e) { console.log('压缩插件失败'); } // 4. 压缩main.js try { // 先获取不能压缩的部分 const main = (await fs.readFile('./public/main.js', 'utf-8')).replace( /this.pluginUseCompress\s*=\s*false\;/, 'this.pluginUseCompress = true;' ); const endIndex = main.indexOf('// >>>> body end'); const nonCompress = main.slice(0, endIndex); const needCompress = main.slice(endIndex + 17); const compressed = babel.transformSync(needCompress)?.code; await fs.writeFile('./dist/main.js', nonCompress + compressed, 'utf-8'); } catch { console.log('main.js压缩失败'); } // 5. 杂项 try { await fse.copy('./LICENSE', './dist/LICENSE'); } catch {} })();