forked from Experience-Monks/msdf-bmfont
-
Notifications
You must be signed in to change notification settings - Fork 51
/
Copy pathcli.js
executable file
·189 lines (175 loc) · 6.66 KB
/
cli.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
#!/usr/bin/env node
const pkg = require('./package.json');
const generateBMFont = require('./index');
const fs = require('fs');
const path = require('path');
const handlebars = require('handlebars');
const args = require('commander');
const updateNotifier = require('update-notifier');
const utils = require('./lib/utils');
updateNotifier({pkg}).notify();
let fontFile;
args
.version('msdf-bmfont-xml v' + pkg.version)
.usage('[options] <font-file>')
.arguments('<font_file>')
.description('Creates a BMFont compatible bitmap font of signed distance fields from a font file')
.option('-f, --output-type <format>', 'font file format: xml(default) | json | txt', /^(xml|json|txt)$/i, 'xml')
.option('-o, --filename <atlas_path>', 'filename of font textures (defaut: font-face) font filename always set to font-face name')
.option('-s, --font-size <fontSize>', 'font size for generated textures', 42)
.option('-i, --charset-file <charset>', 'user-specified charactors from text-file', fileExistValidate)
.option('-m, --texture-size <w,h>', 'ouput texture atlas size', (v) => {return v.split(',')}, [2048, 2048])
.option('-p, --texture-padding <n>', 'padding between glyphs', 1)
.option('-b, --border <n>', 'space between glyphs textures & edge', 0)
.option('-r, --distance-range <n>', 'distance range for SDF', 4)
.option('-t, --field-type <type>', 'msdf(default) | sdf | psdf', /^(msdf|sdf|psdf)$/i, 'msdf')
.option('-d, --round-decimal <digit>', 'rounded digits of the output font file.', 0)
.option('-v, --vector', 'generate svg vector file for debuging', false)
.option('-u, --reuse [file.cfg]', 'save/create config file for reusing settings', false)
.option(' --smart-size', 'shrink atlas to the smallest possible square', true)
.option(' --pot', 'atlas size shall be power of 2', false)
.option(' --square', 'atlas size shall be square', false)
.option(' --rot', 'allow 90-degree rotation while packing', false)
.option(' --rtl', 'use RTL(Arabic/Persian) charactors fix', false)
.action(function(file){
fontFile = fileExistValidate(file);
}).parse(process.argv);
//
// Initialize options
//
let opt = args.opts();
utils.roundAllValue(opt); // Parse all number from string
if (!fontFile) {
console.error('Must specify font-file, use: \'msdf-bmfont -h\' for more infomation');
process.exit(1);
}
const fontface = path.basename(fontFile, path.extname(fontFile));
const fontDir = path.dirname(fontFile);
//
// Set default value
//
// Note: somehow commander.js didn't parse boolean default value
// need to feed manually
//
opt.fontFile = fontFile;
if (typeof opt.reuse === 'boolean') {
opt.filename = utils.valueQueue([opt.filename, path.join(fontDir, fontface)]);
opt.vector = utils.valueQueue([opt.vector, false]);
opt.reuse = utils.valueQueue([opt.reuse, false]);
opt.smartSize = utils.valueQueue([opt.smartSize, false]);
opt.pot = utils.valueQueue([opt.pot, false]);
opt.square = utils.valueQueue([opt.square, false]);
opt.rot = utils.valueQueue([opt.rot, false]);
opt.rtl = utils.valueQueue([opt.rtl, false]);
} else {
opt.filename = utils.valueQueue([opt.filename, path.join("./", fontface)]);
}
//
// Display options
//
const keys = Object.keys(opt)
const padding = longestLength(keys) + 2;
console.log("\nUsing following settings");
console.log("========================================");
keys.forEach(key => {
if (typeof opt.reuse === 'string' && typeof opt[key] === 'undefined') {
console.log(pad(key, padding) + ": Defined in [" + opt.reuse + "]");
} else if (key === 'charsetFile' && typeof opt[key] === 'undefined') {
console.log(pad(key, padding) + ": Unspecified, fallback to ASC-II");
} else console.log(pad(key, padding) + ": " + opt[key]);
});
console.log("========================================");
//
// Validate
//
if (typeof opt.fontFile === 'undefined') {
console.error('No font file specified, aborting.... use -h for help');
process.exit(1);
}
if (typeof opt.reuse !== 'boolean') opt.reuse = fileValidate(opt.reuse);
fs.readFile(opt.charsetFile || '', 'utf8', (error, data) => {
if (error) {
console.warn('No valid charset file loaded, fallback to ASC-II');
}
if (data) opt.charset = data;
generateBMFont(opt.fontFile, opt, (error, textures, font) => {
if (error) throw error;
textures.forEach((texture, index) => {
if (opt.vector) {
const svgTemplate =
`<?xml version="1.0"?>
<!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN" "http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd">
<svg xmlns="http://www.w3.org/2000/svg" version="1.1" width="{{width}}" height="{{height}}">
{{{svgPath}}}
</svg>`;
const template = handlebars.compile(svgTemplate);
const content = template({
width: opt.textureWidth,
height: opt.textureHeight,
svgPath: texture.svg
});
fs.writeFile(`${texture.filename}.svg`, content , (err) => {
if (err) throw err;
console.log('wrote svg[', index, '] : ', `${texture.filename}.svg`);
});
}
fs.writeFile(`${texture.filename}.png`, texture.texture, (err) => {
if (err) throw err;
console.log('wrote spritesheet[', index, '] : ', `${texture.filename}.png`);
});
});
fs.writeFile(font.filename, font.data, (err) => {
if (err) throw err;
console.log('wrote font file : ', font.filename);
});
if(opt.reuse !== false) {
let cfgFileName = typeof opt.reuse === 'boolean' ? `${textures[0].filename}.cfg` : opt.reuse;
fs.writeFile(cfgFileName, JSON.stringify(font.settings, null, '\t'), (err) => {
if (err) throw err;
console.log('wrote cfg file : ', cfgFileName);
});
}
});
});
/**
* Pad `str` to `width`.
*
* @param {String} str
* @param {Number} width
* @return {String}
* @api private
*/
function pad(str, width) {
var len = Math.max(0, width - str.length);
return str + Array(len + 1).join(' ');
}
/**
* Return the largest length of string array.
*
* @param {Array.<String>} arr
* @return {Number}
* @api private
*/
function longestLength(arr) {
return arr.reduce((max, element) => {
return Math.max(max, element.length);
}, 0);
};
function fileExistValidate(filePath) {
try {
if(fs.statSync(filePath).isFile()) return path.normalize(filePath);
else {
console.error('File: ', filePath, ' not found! Aborting...');
process.exit(1);
}
} catch(err) {
console.error('File: ', filePath, ' not valid! Aborting...');
process.exit(1);
}
}
function fileValidate(filePath) {
if (require('is-invalid-path')(filePath)) {
console.error('File: ', filePath, ' not valid! Aborting...');
process.exit(1);
} else return path.normalize(filePath);
}