diff options
| -rw-r--r-- | .github/icons.png | bin | 953644 -> 1121834 bytes | |||
| -rw-r--r-- | gulpfile.js | 663 | ||||
| -rw-r--r-- | package.json | 2 | 
3 files changed, 309 insertions, 356 deletions
| diff --git a/.github/icons.png b/.github/icons.pngBinary files differ index 7a1f6e65..c0f163a0 100644 --- a/.github/icons.png +++ b/.github/icons.png diff --git a/gulpfile.js b/gulpfile.js index 5f501dc8..fddb1d94 100644 --- a/gulpfile.js +++ b/gulpfile.js @@ -16,67 +16,67 @@ const gulp = require('gulp'),  	argv = require('minimist')(process.argv.slice(2)),  	svgParse = require('parse-svg-path'),  	svgpath = require('svgpath'), -	svgr = require('@svgr/core').default; +	svgr = require('@svgr/core').default  const compileOptions = {  	includeIcons: [],  	strokeWidth: null,  	fontForge: "fontforge" -}; +}  if (fs.existsSync('./compile-options.json')) {  	try { -		const tempOptions = require('./compile-options.json'); +		const tempOptions = require('./compile-options.json')  		if (typeof tempOptions !== "object") { -			throw "Compile options file does not contain an json object"; +			throw "Compile options file does not contain an json object"  		}  		if (typeof tempOptions.includeIcons !== "undefined") {  			if (!Array.isArray(tempOptions.includeIcons)) { -				throw "property inludeIcons is not an array"; +				throw "property inludeIcons is not an array"  			} -			compileOptions.includeIcons = tempOptions.includeIcons; +			compileOptions.includeIcons = tempOptions.includeIcons  		}  		if (typeof tempOptions.includeCategories !== "undefined") {  			if (typeof tempOptions.includeCategories === "string") { -				tempOptions.includeCategories = tempOptions.includeCategories.split(' '); +				tempOptions.includeCategories = tempOptions.includeCategories.split(' ')  			} -            if (!Array.isArray(tempOptions.includeCategories)) { -				throw "property includeCategories is not an array or string"; +			if (!Array.isArray(tempOptions.includeCategories)) { +				throw "property includeCategories is not an array or string"  			} -            const tags = Object.entries(require('./tags.json')); -        	tempOptions.includeCategories.forEach(function (category) { -                category = category.charAt(0).toUpperCase() + category.slice(1); +			const tags = Object.entries(require('./tags.json')) +			tempOptions.includeCategories.forEach(function (category) { +				category = category.charAt(0).toUpperCase() + category.slice(1)  				for (const [icon, data] of tags) { -                    if (data.category === category && compileOptions.includeIcons.indexOf(icon) === -1) { -                        compileOptions.includeIcons.push(icon); -                    } -                } -			}); +					if (data.category === category && compileOptions.includeIcons.indexOf(icon) === -1) { +						compileOptions.includeIcons.push(icon) +					} +				} +			})  		}  		if (typeof tempOptions.excludeIcons !== "undefined") { -            if (!Array.isArray(tempOptions.excludeIcons)) { -				throw "property excludeIcons is not an array"; +			if (!Array.isArray(tempOptions.excludeIcons)) { +				throw "property excludeIcons is not an array"  			} -            compileOptions.includeIcons = compileOptions.includeIcons.filter(function (icon) { -                return tempOptions.excludeIcons.indexOf(icon) === -1; -            }); -        } +			compileOptions.includeIcons = compileOptions.includeIcons.filter(function (icon) { +				return tempOptions.excludeIcons.indexOf(icon) === -1 +			}) +		}  		if (typeof tempOptions.strokeWidth !== "undefined") {  			if (typeof tempOptions.strokeWidth !== "string" && typeof tempOptions.strokeWidth !== "number") { -				throw "property strokeWidth is not a string or number"; +				throw "property strokeWidth is not a string or number"  			} -			compileOptions.strokeWidth = tempOptions.strokeWidth.toString(); +			compileOptions.strokeWidth = tempOptions.strokeWidth.toString()  		}  		if (typeof tempOptions.fontForge !== "undefined") {  			if (typeof tempOptions.fontForge !== "string") { -				throw "property fontForge is not a string"; +				throw "property fontForge is not a string"  			} -		  	compileOptions.fontForge = tempOptions.fontForge; +			compileOptions.fontForge = tempOptions.fontForge  		}  	} catch (error) { @@ -85,208 +85,163 @@ if (fs.existsSync('./compile-options.json')) {  } -async function asyncForEach(array, callback) { +async function asyncForEach (array, callback) {  	for (let index = 0; index < array.length; index++) { -		await callback(array[index], index, array); +		await callback(array[index], index, array)  	}  }  const svgToPng = async (filePath, destination) => { -	filePath = path.join(__dirname, filePath); - -	const htmlFilePath = path.join("file:", filePath); -	const browser = await puppeteer.launch(); -	const page = await browser.newPage(); - -	await page.setViewport({ -		height: 24, -		width: 24, -		deviceScaleFactor: 10 -	}); +	filePath = path.join(__dirname, filePath) -	await page.goto(htmlFilePath); - -	await page.screenshot({ -		path: path.join(__dirname, destination), -		omitBackground: true, -		fullPage: true -	}); - -	await browser.close(); - -	return page; -}; +	await cp.exec(`rsvg-convert -h 240 ${filePath} > ${destination}`) +}  const createScreenshot = async (filePath) => { -	try { -		filePath = path.join(__dirname, filePath); - -		const fileName = filePath.replace('.svg', ''); -		const htmlFilePath = path.join("file:", filePath); -		const browser = await puppeteer.launch(); -		const page = await browser.newPage(); - -		await page.setViewport({ -			height: 10, -			width: 10, -			deviceScaleFactor: 2 -		}); - -		await page.goto(htmlFilePath); - -		await page.screenshot({ -			path: `${fileName}.png`, -			omitBackground: false, -			fullPage: true -		}); - -		await browser.close(); -	} catch (error) { -		console.error(error); -		throw Error(error); -	} -}; - +	await cp.exec(`rsvg-convert -x 2 -y 2 ${filePath} > ${filePath.replace('.svg', '.png')}`) +	await cp.exec(`rsvg-convert -x 4 -y 4 ${filePath} > ${filePath.replace('.svg', '@2x.png')}`) +}  const printChangelog = function (newIcons, modifiedIcons, renamedIcons, pretty = false) {  	if (newIcons.length > 0) {  		if (pretty) { -			console.log(`### ${newIcons.length} new icons:`); +			console.log(`### ${newIcons.length} new icons:`)  			newIcons.forEach(function (icon, i) { -				console.log(`- \`${icon}\``); -			}); +				console.log(`- \`${icon}\``) +			})  		} else { -			let str = ''; -			str += `${newIcons.length} new icons: `; +			let str = '' +			str += `${newIcons.length} new icons: `  			newIcons.forEach(function (icon, i) { -				str += `\`${icon}\``; +				str += `\`${icon}\``  				if ((i + 1) <= newIcons.length - 1) {  					str += ', '  				} -			}); +			}) -			console.log(str); +			console.log(str)  		} -		console.log(''); +		console.log('')  	}  	if (modifiedIcons.length > 0) { -		let str = ''; -		str += `Fixed icons: `; +		let str = '' +		str += `Fixed icons: `  		modifiedIcons.forEach(function (icon, i) { -			str += `\`${icon}\``; +			str += `\`${icon}\``  			if ((i + 1) <= modifiedIcons.length - 1) {  				str += ', '  			} -		}); +		}) -		console.log(str); -		console.log(''); +		console.log(str) +		console.log('')  	}  	if (renamedIcons.length > 0) { -		console.log(`Renamed icons: `); +		console.log(`Renamed icons: `)  		renamedIcons.forEach(function (icon, i) { -			console.log(`- \`${icon[0]}\` renamed to \`${icon[1]}\``); -		}); +			console.log(`- \`${icon[0]}\` renamed to \`${icon[1]}\``) +		})  	} -}; +}  const generateIconsPreview = function (files, destFile, cb, columnsCount = 19, paddingOuter = 7) {  	const padding = 20, -		iconSize = 24; +		iconSize = 24  	const iconsCount = files.length,  		rowsCount = Math.ceil(iconsCount / columnsCount),  		width = columnsCount * (iconSize + padding) + 2 * paddingOuter - padding, -		height = rowsCount * (iconSize + padding) + 2 * paddingOuter - padding; +		height = rowsCount * (iconSize + padding) + 2 * paddingOuter - padding  	let svgContentSymbols = '',  		svgContentIcons = '',  		x = paddingOuter, -		y = paddingOuter; +		y = paddingOuter  	files.forEach(function (file, i) { -		let name = path.basename(file, '.svg'); +		let name = path.basename(file, '.svg')  		let svgFile = fs.readFileSync(file), -			svgFileContent = svgFile.toString(); +			svgFileContent = svgFile.toString()  		svgFileContent = svgFileContent  			.replace('<svg xmlns="http://www.w3.org/2000/svg"', `<symbol id="${name}"`)  			.replace(' width="24" height="24"', '')  			.replace('</svg>', '</symbol>') -			.replace(/\n\s+/g, ''); +			.replace(/\n\s+/g, '') -		svgContentSymbols += `\t${svgFileContent}\n`; -		svgContentIcons += `\t<use xlink:href="#${name}" x="${x}" y="${y}" width="${iconSize}" height="${iconSize}" />\n`; +		svgContentSymbols += `\t${svgFileContent}\n` +		svgContentIcons += `\t<use xlink:href="#${name}" x="${x}" y="${y}" width="${iconSize}" height="${iconSize}" />\n` -		x += padding + iconSize; +		x += padding + iconSize  		if (i % columnsCount === columnsCount - 1) { -			x = paddingOuter; -			y += padding + iconSize; +			x = paddingOuter +			y += padding + iconSize  		} -	}); +	}) -	const svgContent = `<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" viewBox="0 0 ${width} ${height}" width="${width}" height="${height}" style="color: #354052"><rect x="0" y="0" width="${width}" height="${height}" fill="#fff"></rect>\n${svgContentSymbols}\n${svgContentIcons}\n</svg>`; +	const svgContent = `<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" viewBox="0 0 ${width} ${height}" width="${width}" height="${height}" style="color: #354052"><rect x="0" y="0" width="${width}" height="${height}" fill="#fff"></rect>\n${svgContentSymbols}\n${svgContentIcons}\n</svg>` -	fs.writeFileSync(destFile, svgContent); -	createScreenshot(destFile); +	fs.writeFileSync(destFile, svgContent) +	createScreenshot(destFile) -	cb(); -}; +	cb() +}  //*********************************************************************************************  gulp.task('iconfont-prepare', function (cb) {  	cp.exec('mkdir -p icons-outlined/ && rm -fd ./icons-outlined/* && mkdir -p && rm -fd ./iconfont/*', function () { -		cb(); -	}); -}); +		cb() +	}) +})  gulp.task('iconfont-clean', function (cb) {  	cp.exec('rm -rf ./icons-outlined', function () { -		cb(); -	}); -}); +		cb() +	}) +})  gulp.task('iconfont-svg-outline', function (cb) {  	cp.exec('mkdir -p icons-outlined/ && rm -fd ./icons-outlined/*', async () => { -		let files = glob.sync("./icons/*.svg"); +		let files = glob.sync("./icons/*.svg") -		let iconfontUnicode = {}; +		let iconfontUnicode = {}  		if (fs.existsSync('./.build/iconfont-unicode.json')) { -			iconfontUnicode = require('./.build/iconfont-unicode'); +			iconfontUnicode = require('./.build/iconfont-unicode')  		}  		await asyncForEach(files, async function (file) { -			const name = path.basename(file, '.svg'); +			const name = path.basename(file, '.svg') -			if (compileOptions.includeIcons.length==0 || compileOptions.includeIcons.indexOf(name)>=0) { +			if (compileOptions.includeIcons.length === 0 || compileOptions.includeIcons.indexOf(name) >= 0) { -				unicode = iconfontUnicode[name]; +				unicode = iconfontUnicode[name] -				await console.log('Stroke for:', file, unicode); +				await console.log('Stroke for:', file, unicode) -				let strokedSVG = fs.readFileSync(file).toString(); +				let strokedSVG = fs.readFileSync(file).toString()  				strokedSVG = strokedSVG  					.replace('width="24"', 'width="1000"') -					.replace('height="24"', 'height="1000"'); +					.replace('height="24"', 'height="1000"') +  				if (compileOptions.strokeWidth) { -					strokedSVG = strokedSVG.replace('stroke-width="2"', `stroke-width="${compileOptions.strokeWidth}"`); +					strokedSVG = strokedSVG.replace('stroke-width="2"', `stroke-width="${compileOptions.strokeWidth}"`)  				}  				await outlineStroke(strokedSVG, { @@ -298,52 +253,52 @@ gulp.task('iconfont-svg-outline', function (cb) {  					color: 'black'  				}).then(outlined => {  					if (unicode) { -						fs.writeFileSync(`icons-outlined/u${unicode.toUpperCase()}-${name}.svg`, outlined); +						fs.writeFileSync(`icons-outlined/u${unicode.toUpperCase()}-${name}.svg`, outlined)  					} else { -						fs.writeFileSync(`icons-outlined/${name}.svg`, outlined); +						fs.writeFileSync(`icons-outlined/${name}.svg`, outlined)  					} -				}).catch(error => console.log(error)); +				}).catch(error => console.log(error))  			} -		}); +		}) -		cb(); -	}); -}); +		cb() +	}) +}) -gulp.task('iconfont-optimize', function() { +gulp.task('iconfont-optimize', function () {  	return gulp.src('icons-outlined/*')  		.pipe(svgo()) -		.pipe(gulp.dest('icons-outlined')); -}); +		.pipe(gulp.dest('icons-outlined')) +}) -gulp.task('iconfont-fix-outline', function(cb) { -	var fontForge= compileOptions.fontForge; +gulp.task('iconfont-fix-outline', function (cb) { +	var fontForge = compileOptions.fontForge  	// correct svg outline directions in a child process using fontforge -	const generate = cp.spawn(fontForge, ["-lang=py", "-script", "./fix-outline.py"], { stdio: 'inherit' }); +	const generate = cp.spawn(fontForge, ["-lang=py", "-script", "./fix-outline.py"], { stdio: 'inherit' })  	generate.on("close", function (code) { -		console.log(`Correcting svg outline directions exited with code ${code}`); +		console.log(`Correcting svg outline directions exited with code ${code}`)  		if (!code) { -			cb(); +			cb()  		} -	}); -}); +	}) +})  gulp.task('iconfont', function () { -	let maxUnicode = 59905; +	let maxUnicode = 59905  	if (fs.existsSync('./.build/iconfont-unicode.json')) { -		const iconfontUnicode = require('./.build/iconfont-unicode'); +		const iconfontUnicode = require('./.build/iconfont-unicode')  		for (const name in iconfontUnicode) { -			const unicode = parseInt(iconfontUnicode[name], 16); +			const unicode = parseInt(iconfontUnicode[name], 16) -			maxUnicode = Math.max(maxUnicode, unicode); +			maxUnicode = Math.max(maxUnicode, unicode)  		}  	} -	maxUnicode = maxUnicode + 1; +	maxUnicode = maxUnicode + 1  	return gulp.src(['icons-outlined/*.svg'])  		.pipe(iconfont({ @@ -358,134 +313,134 @@ gulp.task('iconfont', function () {  		}))  		.on('glyphs', function (glyphs, options) {  			//glyphs json -			let glyphsObject = {}; +			let glyphsObject = {}  			//sort glypht  			glyphs = glyphs.sort(function (a, b) {  				return ('' + a.name).localeCompare(b.name) -			}); +			})  			glyphs.forEach(function (glyph) { -				glyphsObject[glyph.name] = glyph.unicode[0].codePointAt(0).toString(16); -			}); +				glyphsObject[glyph.name] = glyph.unicode[0].codePointAt(0).toString(16) +			}) -			fs.writeFileSync(`./.build/iconfont-unicode.json`, JSON.stringify(glyphsObject)); +			fs.writeFileSync(`./.build/iconfont-unicode.json`, JSON.stringify(glyphsObject))  			//css -			options['glyphs'] = glyphs; -			options['v'] = p.version; +			options['glyphs'] = glyphs +			options['v'] = p.version -			const compiled = template(fs.readFileSync('.build/iconfont.scss').toString()); -			const result = compiled(options); +			const compiled = template(fs.readFileSync('.build/iconfont.scss').toString()) +			const result = compiled(options) -			fs.writeFileSync('iconfont/tabler-icons.scss', result); +			fs.writeFileSync('iconfont/tabler-icons.scss', result)  			//html -			const compiledHtml = template(fs.readFileSync('.build/iconfont.html').toString()); -			const resultHtml = compiledHtml(options); +			const compiledHtml = template(fs.readFileSync('.build/iconfont.html').toString()) +			const resultHtml = compiledHtml(options) -			fs.writeFileSync('iconfont/tabler-icons.html', resultHtml); +			fs.writeFileSync('iconfont/tabler-icons.html', resultHtml)  		}) -		.pipe(gulp.dest('iconfont/fonts')); -}); +		.pipe(gulp.dest('iconfont/fonts')) +})  gulp.task('iconfont-css', function (cb) {  	sass.render({  		file: 'iconfont/tabler-icons.scss',  		outputStyle: 'expanded'  	}, function (err, result) { -		fs.writeFileSync('iconfont/tabler-icons.css', result.css); +		fs.writeFileSync('iconfont/tabler-icons.css', result.css) -		const cleanOutput = new cleanCSS({}).minify(result.css); -		fs.writeFileSync('iconfont/tabler-icons.min.css', cleanOutput.styles); +		const cleanOutput = new cleanCSS({}).minify(result.css) +		fs.writeFileSync('iconfont/tabler-icons.min.css', cleanOutput.styles) -		cb(); -	}); -}); +		cb() +	}) +}) -gulp.task('update-tags-unicode', function(cb) { +gulp.task('update-tags-unicode', function (cb) {  	let tags = require('./tags.json'), -		unicodes = require('./.build/iconfont-unicode.json'); +		unicodes = require('./.build/iconfont-unicode.json') -	for(let i in tags) { +	for (let i in tags) {  		tags[i] = {  			...tags[i],  			unicode: unicodes[i],  		}  	} -	console.log('tags', tags); +	console.log('tags', tags) -	fs.writeFileSync(`tags.json`, JSON.stringify(tags, null, 2)); +	fs.writeFileSync(`tags.json`, JSON.stringify(tags, null, 2)) -	cb(); -}); +	cb() +}) -gulp.task('build-iconfont', gulp.series('iconfont-prepare', 'iconfont-svg-outline', 'iconfont-fix-outline', 'iconfont-optimize', 'iconfont', 'iconfont-css', 'iconfont-clean', 'update-tags-unicode')); +gulp.task('build-iconfont', gulp.series('iconfont-prepare', 'iconfont-svg-outline', 'iconfont-fix-outline', 'iconfont-optimize', 'iconfont', 'iconfont-css', 'iconfont-clean', 'update-tags-unicode'))  gulp.task('build-zip', function () { -	const version = p.version; +	const version = p.version  	return gulp.src('{icons/**/*,icons-png/**/*,icons-react/**/*,iconfont/**/*,tabler-sprite.svg,tabler-sprite-nostroke.svg}')  		.pipe(zip(`tabler-icons-${version}.zip`))  		.pipe(gulp.dest('packages')) -}); +})  gulp.task('build-jekyll', function (cb) { -	const jekyll = cp.spawn("bundle", ["exec", "jekyll", "build"], { stdio: 'inherit' }); +	const jekyll = cp.spawn("bundle", ["exec", "jekyll", "build"], { stdio: 'inherit' })  	jekyll.on("close", function (code) { -		console.log(`Jekyll build exited with code ${code}`); +		console.log(`Jekyll build exited with code ${code}`)  		if (!code) { -			cb(); +			cb()  		} -	}); -}); +	}) +})  gulp.task('build-copy', function (cb) {  	cp.exec('mkdir -p icons/ && rm -fd ./icons/* && cp ./_site/icons/* ./icons && cp ./_site/tags.json .', function () { -		cb(); -	}); -}); +		cb() +	}) +})  gulp.task('clean-png', function (cb) {  	cp.exec('rm -fd ./icons-png/*', function () { -		cb(); -	}); -}); +		cb() +	}) +})  gulp.task('icons-sprite', function (cb) {  	glob("_site/icons/*.svg", {}, function (er, files) { -		let svgContent = ''; +		let svgContent = ''  		files.forEach(function (file, i) {  			let name = path.basename(file, '.svg'),  				svgFile = fs.readFileSync(file), -				svgFileContent = svgFile.toString(); +				svgFileContent = svgFile.toString()  			svgFileContent = svgFileContent  				.replace(/<svg[^>]+>/g, '')  				.replace(/<\/svg>/g, '')  				.replace(/\n+/g, '')  				.replace(/>\s+</g, '><') -				.trim(); +				.trim()  			svgContent += `<symbol id="tabler-${name}" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">${svgFileContent}</symbol>` -		}); +		}) -		let svg = `<svg xmlns="http://www.w3.org/2000/svg"><defs>${svgContent}</defs></svg>`; +		let svg = `<svg xmlns="http://www.w3.org/2000/svg"><defs>${svgContent}</defs></svg>` -		fs.writeFileSync('tabler-sprite.svg', svg); -		fs.writeFileSync('tabler-sprite-nostroke.svg', svg.replace(/stroke-width="2"\s/g, '')); -		cb(); -	}); -}); +		fs.writeFileSync('tabler-sprite.svg', svg) +		fs.writeFileSync('tabler-sprite-nostroke.svg', svg.replace(/stroke-width="2"\s/g, '')) +		cb() +	}) +})  gulp.task('icons-preview', function (cb) {  	glob("icons/*.svg", {}, function (er, files) { -		generateIconsPreview(files, '.github/icons.svg', cb); -	}); -}); +		generateIconsPreview(files, '.github/icons.svg', cb) +	}) +})  gulp.task('icons-stroke', gulp.series('build-jekyll', function (cb) { @@ -496,11 +451,11 @@ gulp.task('icons-stroke', gulp.series('build-jekyll', function (cb) {  		paddingOuter = 3,  		iconSize = 32,  		width = 914, -		height = iconSize + paddingOuter * 2; +		height = iconSize + paddingOuter * 2  	let svgContentSymbols = '',  		svgContentIcons = '', -		x = paddingOuter; +		x = paddingOuter  	strokes.forEach(function (stroke) {  		let svgFileContentStroked = svgFileContent @@ -508,42 +463,42 @@ gulp.task('icons-stroke', gulp.series('build-jekyll', function (cb) {  			.replace(' width="24" height="24"', '')  			.replace(' stroke-width="2"', ` stroke-width="${stroke}"`)  			.replace('</svg>', '</symbol>') -			.replace(/\n\s+/g, ''); +			.replace(/\n\s+/g, '') -		svgContentSymbols += `\t${svgFileContentStroked}\n`; -		svgContentIcons += `\t<use xlink:href="#icon-${stroke}" x="${x}" y="${paddingOuter}" width="${iconSize}" height="${iconSize}" />\n`; +		svgContentSymbols += `\t${svgFileContentStroked}\n` +		svgContentIcons += `\t<use xlink:href="#icon-${stroke}" x="${x}" y="${paddingOuter}" width="${iconSize}" height="${iconSize}" />\n` -		x += padding + iconSize; -	}); +		x += padding + iconSize +	}) -	const svgContent = `<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" viewBox="0 0 ${width} ${height}" width="${width}" height="${height}" style="color: #354052"><rect x="0" y="0" width="${width}" height="${height}" fill="#fff"></rect>\n${svgContentSymbols}\n${svgContentIcons}\n</svg>`; +	const svgContent = `<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" viewBox="0 0 ${width} ${height}" width="${width}" height="${height}" style="color: #354052"><rect x="0" y="0" width="${width}" height="${height}" fill="#fff"></rect>\n${svgContentSymbols}\n${svgContentIcons}\n</svg>` -	fs.writeFileSync('.github/icons-stroke.svg', svgContent); -	createScreenshot('.github/icons-stroke.svg'); -	cb(); -})); +	fs.writeFileSync('.github/icons-stroke.svg', svgContent) +	createScreenshot('.github/icons-stroke.svg') +	cb() +}))  gulp.task('optimize', function (cb) {  	const addFloats = function (n1, n2) {  		return Math.round((parseFloat(n1) + parseFloat(n2)) * 1000) / 1000 -	}; +	} -	const optimizePath = function(path) { +	const optimizePath = function (path) {  		let transformed = svgpath(path)  			.rel()  			.round(3) -			.toString(); +			.toString() -		return svgParse(transformed).map(function(a){ -			return a.join(' '); -		}).join(' '); -	}; +		return svgParse(transformed).map(function (a) { +			return a.join(' ') +		}).join(' ') +	}  	glob("src/_icons/*.svg", {}, function (er, files) {  		files.forEach(function (file, i) {  			let svgFile = fs.readFileSync(file), -				svgFileContent = svgFile.toString(); +				svgFileContent = svgFile.toString()  			svgFileContent = svgFileContent  				.replace(/><\/(polyline|line|rect|circle|path)>/g, '/>') @@ -551,275 +506,273 @@ gulp.task('optimize', function (cb) {  				.replace(/\s?\/>/g, ' />')  				.replace(/\n\s*<(line|circle|path|polyline|rect)/g, "\n  <$1")  				.replace(/polyline points="([0-9.]+)\s([0-9.]+)\s([0-9.]+)\s([0-9.]+)"/g, 'line x1="$1" y1="$2" x2="$3" y2="$4"') -				.replace(/<path d="([^"]+)"/g, function(f, r1) { -					r1 = optimizePath(r1); +				.replace(/<path d="([^"]+)"/g, function (f, r1) { +					r1 = optimizePath(r1) -					return `<path d="${r1}"`; +					return `<path d="${r1}"`  				})  				.replace(/d="m/g, 'd="M')  				.replace(/([Aa])\s?([0-9.]+)\s([0-9.]+)\s([0-9.]+)\s?([0-1])\s?([0-1])\s?(-?[0-9.]+)\s?(-?[0-9.]+)/gi, '$1$2 $3 $4 $5 $6 $7 $8')  				.replace(/\n\n+/g, "\n")  				.replace(/<path d="M([0-9.]*) ([0-9.]*)l\s?([-0-9.]*) ([-0-9.]*)"/g, function (f, r1, r2, r3, r4) { -					return `<line x1="${r1}" y1="${r2}" x2="${addFloats(r1, r3)}" y2="${addFloats(r2, r4)}"`; +					return `<line x1="${r1}" y1="${r2}" x2="${addFloats(r1, r3)}" y2="${addFloats(r2, r4)}"`  				})  				.replace(/<path d="M([0-9.]*) ([0-9.]*)v\s?([-0-9.]*)"/g, function (f, r1, r2, r3) { -					return `<line x1="${r1}" y1="${r2}" x2="${r1}" y2="${addFloats(r2, r3)}"`; +					return `<line x1="${r1}" y1="${r2}" x2="${r1}" y2="${addFloats(r2, r3)}"`  				})  				.replace(/<path d="M([0-9.]*) ([0-9.]*)h\s?([-0-9.]*)"/g, function (f, r1, r2, r3) { -					return `<line x1="${r1}" y1="${r2}" x2="${addFloats(r1, r3)}" y2="${r2}"`; +					return `<line x1="${r1}" y1="${r2}" x2="${addFloats(r1, r3)}" y2="${r2}"`  				})  				.replace(/<path d="([^"]+)"/g, function (f, r1) {  					r1 = r1  						.replace(/ -0\./g, " -.")  						.replace(/ 0\./g, " .")  						.replace(/\s([a-z])/gi, "$1") -						.replace(/([a-z])\s/gi, "$1"); -					return `<path d="${r1}"`; +						.replace(/([a-z])\s/gi, "$1") +					return `<path d="${r1}"`  				}) -			;  			if (svgFile.toString() !== svgFileContent) { -				fs.writeFileSync(file, svgFileContent); +				fs.writeFileSync(file, svgFileContent)  			} -		}); - -		cb(); -	}); -}); +		}) +		cb() +	}) +})  gulp.task('changelog-commit', function (cb) {  	cp.exec('git status', function (err, ret) { -		let newIcons = [], modifiedIcons = [], renamedIcons = []; +		let newIcons = [], modifiedIcons = [], renamedIcons = []  		ret.replace(/new file:\s+src\/_icons\/([a-z0-9-]+)\.svg/g, function (m, fileName) { -			newIcons.push(fileName); -		}); +			newIcons.push(fileName) +		})  		ret.replace(/modified:\s+src\/_icons\/([a-z0-9-]+)\.svg/g, function (m, fileName) { -			modifiedIcons.push(fileName); -		}); +			modifiedIcons.push(fileName) +		})  		ret.replace(/renamed:\s+src\/_icons\/([a-z0-9-]+).svg -> src\/_icons\/([a-z0-9-]+).svg/g, function (m, fileNameBefore, fileNameAfter) { -			renamedIcons.push([fileNameBefore, fileNameAfter]); -		}); +			renamedIcons.push([fileNameBefore, fileNameAfter]) +		})  		modifiedIcons = modifiedIcons.filter(function (el) { -			return newIcons.indexOf(el) < 0; -		}); +			return newIcons.indexOf(el) < 0 +		}) -		printChangelog(newIcons, modifiedIcons, renamedIcons); +		printChangelog(newIcons, modifiedIcons, renamedIcons) -		cb(); -	}); -}); +		cb() +	}) +})  gulp.task('changelog', function (cb) { -	const version = argv['latest-tag'] || `v${p.version}`; +	const version = argv['latest-tag'] || `v${p.version}`  	if (version) {  		cp.exec(`git diff ${version} HEAD --name-status`, function (err, ret) { -			let newIcons = [], modifiedIcons = [], renamedIcons = []; +			let newIcons = [], modifiedIcons = [], renamedIcons = []  			ret.replace(/A\s+src\/_icons\/([a-z0-9-]+)\.svg/g, function (m, fileName) { -				newIcons.push(fileName); -			}); +				newIcons.push(fileName) +			})  			ret.replace(/M\s+src\/_icons\/([a-z0-9-]+)\.svg/g, function (m, fileName) { -				modifiedIcons.push(fileName); -			}); +				modifiedIcons.push(fileName) +			})  			ret.replace(/R[0-9]+\s+src\/_icons\/([a-z0-9-]+)\.svg\s+src\/_icons\/([a-z0-9-]+).svg/g, function (m, fileNameBefore, fileNameAfter) { -				renamedIcons.push([fileNameBefore, fileNameAfter]); -			}); +				renamedIcons.push([fileNameBefore, fileNameAfter]) +			})  			modifiedIcons = modifiedIcons.filter(function (el) { -				return newIcons.indexOf(el) < 0; -			}); +				return newIcons.indexOf(el) < 0 +			}) -			printChangelog(newIcons, modifiedIcons, renamedIcons, true); +			printChangelog(newIcons, modifiedIcons, renamedIcons, true) -			cb(); -		}); +			cb() +		})  	} -}); +})  gulp.task('changelog-image', function (cb) {  	const version = argv['latest-version'] || `${p.version}`, -		newVersion = argv['new-version'] || `${p.version}`; +		newVersion = argv['new-version'] || `${p.version}`  	if (version) {  		cp.exec(`git diff v${version} HEAD --name-status`, function (err, ret) { -			let newIcons = []; +			let newIcons = []  			ret.replace(/[A]\s+src\/_icons\/([a-z0-9-]+)\.svg/g, function (m, fileName) { -				newIcons.push(fileName); -			}); +				newIcons.push(fileName) +			})  			newIcons = newIcons.map(function (icon) { -				return `./icons/${icon}.svg`; -			}); +				return `./icons/${icon}.svg` +			})  			if (newIcons.length > 0) { -				generateIconsPreview(newIcons, `.github/tabler-icons-${newVersion}.svg`, cb, 6, 24); +				generateIconsPreview(newIcons, `.github/tabler-icons-${newVersion}.svg`, cb, 6, 24)  			} else { -				cb(); +				cb()  			} -		}); +		})  	} else { -		cb(); +		cb()  	} -}); +})  gulp.task('svg-to-png', gulp.series('build-jekyll', 'clean-png', async (cb) => { -	let files = glob.sync("./icons/*.svg"); +	let files = glob.sync("./icons/*.svg")  	await asyncForEach(files, async function (file, i) { -		let name = path.basename(file, '.svg'); +		let name = path.basename(file, '.svg') -		console.log('name', name); +		console.log('name', name) -		await svgToPng(file, `icons-png/${name}.png`); -	}); +		await svgToPng(file, `icons-png/${name}.png`) +	}) -	cb(); -})); +	cb() +}))  gulp.task('clean-react', function (cb) {  	cp.exec('rm -fd ./icons-react/* && mkdir icons-react/icons-js', function () { -		cb(); -	}); -}); +		cb() +	}) +})  gulp.task('svg-to-react', gulp.series('clean-react', async function (cb) { -	let files = glob.sync("./icons/*.svg"); +	let files = glob.sync("./icons/*.svg")  	const camelize = function (str) { -		str = str.replace(/-/g, ' '); +		str = str.replace(/-/g, ' ')  		return str.replace(/(?:^\w|[A-Z]|\b\w)/g, function (word, index) { -			return word.toUpperCase(); -		}).replace(/\s+/g, ''); -	}; +			return word.toUpperCase() +		}).replace(/\s+/g, '') +	}  	const componentName = function (file) { -		file = path.basename(file, '.svg'); -		file = camelize(`Icon ${file}`); +		file = path.basename(file, '.svg') +		file = camelize(`Icon ${file}`) -		return file; -	}; +		return file +	} -	const optimizeSvgCode = function(svgCode) { -		return svgCode.replace('<path stroke="none" d="M0 0h24v24H0z"/>', ''); -	}; +	const optimizeSvgCode = function (svgCode) { +		return svgCode.replace('<path stroke="none" d="M0 0h24v24H0z"/>', '') +	}  	let indexCode = '', -		indexDCode = `import { FC, SVGAttributes } from 'react';\n\ninterface TablerIconProps extends SVGAttributes<SVGElement> { color?: string; size?: string | number; stroke?: string | number; }\n\ntype TablerIcon = FC<TablerIconProps>;\n\n`; +		indexDCode = `import { FC, SVGAttributes } from 'react';\n\ninterface TablerIconProps extends SVGAttributes<SVGElement> { color?: string; size?: string | number; stroke?: string | number; }\n\ntype TablerIcon = FC<TablerIconProps>;\n\n`  	await asyncForEach(files, async function (file) {  		const svgCode = optimizeSvgCode(fs.readFileSync(file).toString()),  			fileName = path.basename(file, '.svg') + '.js', -			iconComponentName = componentName(file); +			iconComponentName = componentName(file)  		await svgr(svgCode, {  			icon: false,  			svgProps: { width: '{size}', height: '{size}', strokeWidth: '{stroke}', stroke: '{color}' },  			template: require('./.build/svgr-template')  		}, { componentName: iconComponentName }).then(jsCode => { -			fs.writeFileSync('icons-react/icons-js/' + fileName, jsCode); -			indexCode += `export { default as ${iconComponentName} } from './icons-js/${fileName}';\n`; -			indexDCode += `export const ${iconComponentName}: TablerIcon;\n`; -		}); +			fs.writeFileSync('icons-react/icons-js/' + fileName, jsCode) +			indexCode += `export { default as ${iconComponentName} } from './icons-js/${fileName}';\n` +			indexDCode += `export const ${iconComponentName}: TablerIcon;\n` +		}) -		fs.writeFileSync('icons-react/index.js', indexCode); -		fs.writeFileSync('icons-react/index.d.ts', indexDCode); -	}); +		fs.writeFileSync('icons-react/index.js', indexCode) +		fs.writeFileSync('icons-react/index.d.ts', indexDCode) +	}) -	cb(); -})); +	cb() +})) -const setVersions = function(version, files) { -	for(const i in files) { -		const file = files[i]; +const setVersions = function (version, files) { +	for (const i in files) { +		const file = files[i]  		if (fs.existsSync(`src/_icons/${file}.svg`)) { -			let svgFile = fs.readFileSync(`src/_icons/${file}.svg`).toString(); +			let svgFile = fs.readFileSync(`src/_icons/${file}.svg`).toString() -			if(!svgFile.match(/version: ([0-9.]+)/i)) { -				svgFile = svgFile.replace(/---\n<svg>/i, function(m){ -					return `version: "${version}"\n${m}`; -				}); +			if (!svgFile.match(/version: ([0-9.]+)/i)) { +				svgFile = svgFile.replace(/---\n<svg>/i, function (m) { +					return `version: "${version}"\n${m}` +				}) -				fs.writeFileSync(`src/_icons/${file}.svg`, svgFile); +				fs.writeFileSync(`src/_icons/${file}.svg`, svgFile)  			} else { -				console.log(`File ${file} already has version`); +				console.log(`File ${file} already has version`)  			}  		} else { -			console.log(`File ${file} doesn't exists`); +			console.log(`File ${file} doesn't exists`)  		}  	} -}; +}  gulp.task('update-icons-version', function (cb) {  	const version = argv['latest-version'] || `${p.version}`, -		newVersion = argv['new-version'] || `${p.version}`; +		newVersion = argv['new-version'] || `${p.version}`  	if (version) {  		cp.exec(`git diff v${version} HEAD --name-status`, function (err, ret) { -			let newIcons = []; +			let newIcons = []  			ret.replace(/[A]\s+src\/_icons\/([a-z0-9-]+)\.svg/g, function (m, fileName) { -				newIcons.push(fileName); -			}); +				newIcons.push(fileName) +			}) -			if(newIcons.length) { -				setVersions(newVersion.replace(/\.0$/, ''), newIcons); +			if (newIcons.length) { +				setVersions(newVersion.replace(/\.0$/, ''), newIcons)  			} -		}); +		})  	} -	cb(); -}); +	cb() +}) -gulp.task('import-tags', function(cb) { +gulp.task('import-tags', function (cb) {  	fs.createReadStream('./_import.csv')  		.pipe(csv({  			headers: false,  			separator: "\t"  		}))  		.on('data', (row) => { -			console.log(row[0], row[1]); +			console.log(row[0], row[1]) -			const filename = `src/_icons/${row[0]}.svg`; +			const filename = `src/_icons/${row[0]}.svg` -			let data = fs.readFileSync(filename).toString(); -			data = data.replace(/(---[\s\S]+?---)/, function(m, headerContent){ +			let data = fs.readFileSync(filename).toString() +			data = data.replace(/(---[\s\S]+?---)/, function (m, headerContent) { -				headerContent = headerContent.replace(/tags: .*\n/, ''); -				headerContent = headerContent.replace(/---/, `---\ntags: [${row[1]}]`); +				headerContent = headerContent.replace(/tags: .*\n/, '') +				headerContent = headerContent.replace(/---/, `---\ntags: [${row[1]}]`) -				return headerContent; -			}); +				return headerContent +			}) -			fs.writeFileSync(filename, data); +			fs.writeFileSync(filename, data)  		})  		.on('end', () => { -			console.log('CSV file successfully processed'); -		}); -	cb(); -}); +			console.log('CSV file successfully processed') +		}) +	cb() +})  gulp.task("build-react", function (cb) {  	cp.exec("npm run build-react", function () { -		cb(); -	}); -}); +		cb() +	}) +}) -gulp.task('build', gulp.series('optimize', 'update-icons-version', 'build-jekyll', 'build-copy', 'icons-sprite', 'svg-to-react', 'build-react', 'icons-preview', 'svg-to-png', 'build-iconfont', 'changelog-image', 'build-zip')); +gulp.task('build', gulp.series('optimize', 'update-icons-version', 'build-jekyll', 'build-copy', 'icons-sprite', 'svg-to-react', 'build-react', 'icons-preview', 'svg-to-png', 'build-iconfont', 'changelog-image', 'build-zip')) diff --git a/package.json b/package.json index c5c26c60..b4c538e6 100644 --- a/package.json +++ b/package.json @@ -77,7 +77,7 @@      "gulp-zip": "5.0.2",      "lodash.template": "4.5.0",      "minimist": "1.2.5", -    "node-sass": "4.14.1", +    "node-sass": "7.0.1",      "parse-svg-path": "^0.1.2",      "puppeteer": "5.3.0",      "release-it": "14.12.5", | 
