gulpfile.js 10.7 KB
Newer Older
C
codecalm 已提交
1
const gulp = require('gulp'),
C
codecalm 已提交
2
	debug = require('gulp-debug'),
C
codecalm 已提交
3
	clean = require('gulp-clean'),
4
	sass = require('gulp-sass')(require('sass')),
C
codecalm 已提交
5
	postcss = require('gulp-postcss'),
C
codecalm 已提交
6
	header = require('gulp-header'),
C
codecalm 已提交
7
	cleanCSS = require('gulp-clean-css'),
C
codecalm 已提交
8
	rtlcss = require('gulp-rtlcss'),
C
codecalm 已提交
9
	minifyJS = require('gulp-minify'),
C
codecalm 已提交
10
	rename = require('gulp-rename'),
C
codecalm 已提交
11
	purgecss = require('gulp-purgecss'),
C
codecalm 已提交
12
	rollupStream = require('@rollup/stream'),
C
codecalm 已提交
13 14
	rollupBabel = require('rollup-plugin-babel'),
	rollupCleanup = require('rollup-plugin-cleanup'),
P
Paweł Kuna 已提交
15
	{ nodeResolve } = require('@rollup/plugin-node-resolve'),
C
codecalm 已提交
16
	rollupCommonjs = require('@rollup/plugin-commonjs'),
C
codecalm 已提交
17
	rollupReplace = require('@rollup/plugin-replace'),
C
codecalm 已提交
18 19
	vinylSource = require('vinyl-source-stream'),
	vinylBuffer = require('vinyl-buffer'),
C
codecalm 已提交
20
	critical = require('critical').stream,
C
codecalm 已提交
21
	browserSync = require('browser-sync'),
C
codecalm 已提交
22
	glob = require('glob'),
C
codecalm 已提交
23
	spawn = require('cross-spawn'),
C
codecalm 已提交
24 25 26
	fs = require('fs'),
	path = require('path'),
	YAML = require('yaml'),
C
codecalm 已提交
27
	yargs = require('yargs/yargs'),
C
codecalm 已提交
28 29
	cp = require('child_process'),
	pkg = require('./package.json'),
C
codecalm 已提交
30
	year = new Date().getFullYear(),
P
Paweł Kuna 已提交
31
	argv = yargs(process.argv).argv
C
codecalm 已提交
32

C
codecalm 已提交
33 34 35
let BUILD = false,
	distDir = './.tmp',
	demoDir = './.tmp',
P
Paweł Kuna 已提交
36
	srcDir = './src'
C
codecalm 已提交
37

C
codecalm 已提交
38 39 40
/**
 * Enable BUILD mode and set directories
 */
C
codecalm 已提交
41
gulp.task('build-on', (cb) => {
P
Paweł Kuna 已提交
42 43 44
	BUILD = true
	distDir = './dist'
	demoDir = './demo'
C
codecalm 已提交
45

P
Paweł Kuna 已提交
46 47
	cb()
})
C
codecalm 已提交
48 49 50 51

/**
 * Return banner added to CSS and JS dist files
 */
C
codecalm 已提交
52 53 54 55 56 57 58 59 60
const getBanner = () => {
	return `/*!
* Tabler v${pkg.version} (${pkg.homepage})
* @version ${pkg.version}
* @link ${pkg.homepage}
* Copyright 2018-${year} The Tabler Authors
* Copyright 2018-${year} codecalm.net Paweł Kuna
* Licensed under MIT (https://github.com/tabler/tabler/blob/master/LICENSE)
*/
P
Paweł Kuna 已提交
61 62
`
}
C
codecalm 已提交
63

C
codecalm 已提交
64 65 66
/**
 * Array.flat polyfill
 */
C
codecalm 已提交
67 68
if (!Array.prototype.flat) {
	Object.defineProperty(Array.prototype, 'flat', {
C
codecalm 已提交
69 70
		value: function (depth = 1) {
			return this.reduce(function (flat, toFlatten) {
P
Paweł Kuna 已提交
71 72
				return flat.concat((Array.isArray(toFlatten) && (depth > 1)) ? toFlatten.flat(depth - 1) : toFlatten)
			}, [])
C
codecalm 已提交
73
		}
P
Paweł Kuna 已提交
74
	})
C
codecalm 已提交
75 76
}

C
codecalm 已提交
77
/**
C
codecalm 已提交
78
 * Import tabler-icons form npm and generate Jekyll `.yml` data files
C
codecalm 已提交
79
 */
C
codecalm 已提交
80 81
gulp.task('svg-icons', (cb) => {
	const prepareSvgFile = (svg) => {
P
Paweł Kuna 已提交
82 83
		return svg.replace(/\n/g, '').replace(/>\s+</g, '><')
	}
C
codecalm 已提交
84

C
codecalm 已提交
85
	const generateIconsYml = (dir, filename) => {
P
Paweł Kuna 已提交
86 87
		const files = glob.sync(dir)
		let svgList = {}
C
codecalm 已提交
88

C
codecalm 已提交
89
		files.forEach((file) => {
P
Paweł Kuna 已提交
90 91 92
			const basename = path.basename(file, '.svg')
			svgList[basename] = prepareSvgFile(fs.readFileSync(file).toString())
		})
C
codecalm 已提交
93

P
Paweł Kuna 已提交
94 95
		fs.writeFileSync(filename, YAML.stringify(svgList))
	}
C
codecalm 已提交
96

P
Paweł Kuna 已提交
97
	generateIconsYml("./node_modules/@tabler/icons/icons/*.svg", `${srcDir}/pages/_data/icons.yml`)
C
codecalm 已提交
98

P
Paweł Kuna 已提交
99 100
	cb()
})
C
codecalm 已提交
101

C
codecalm 已提交
102 103 104
/**
 * Check unused Jekyll partials
 */
C
codecalm 已提交
105
gulp.task('unused-files', (cb) => {
P
Paweł Kuna 已提交
106
	let foundFiles = []
C
codecalm 已提交
107

C
codecalm 已提交
108
	glob.sync(`${srcDir}/pages/**/*.{html,md}`).forEach((file) => {
P
Paweł Kuna 已提交
109
		let fileContent = fs.readFileSync(file)
C
codecalm 已提交
110

C
codecalm 已提交
111
		fileContent.toString().replace(/\{% include(_cached)? ([a-z0-9\/_-]+\.html)/g, (f, c, filename) => {
P
Paweł Kuna 已提交
112
			filename = `${srcDir}/pages/_includes/${filename}`
C
codecalm 已提交
113 114

			if (!foundFiles.includes(filename)) {
P
Paweł Kuna 已提交
115
				foundFiles.push(filename)
C
codecalm 已提交
116
			}
P
Paweł Kuna 已提交
117 118
		})
	})
C
codecalm 已提交
119

P
Paweł Kuna 已提交
120
	let includeFiles = glob.sync(`${srcDir}/pages/_includes/**/*.html`)
C
codecalm 已提交
121

C
codecalm 已提交
122
	includeFiles.forEach((file) => {
C
codecalm 已提交
123
		if (!foundFiles.includes(file)) {
P
Paweł Kuna 已提交
124
			console.log('file', file)
C
codecalm 已提交
125
		}
P
Paweł Kuna 已提交
126
	})
C
codecalm 已提交
127

P
Paweł Kuna 已提交
128 129
	cb()
})
C
codecalm 已提交
130

C
codecalm 已提交
131
/**
C
codecalm 已提交
132
 * Clean `dist` folder before build
C
codecalm 已提交
133
 */
C
codecalm 已提交
134
gulp.task('clean-dirs', () => {
C
codecalm 已提交
135
	return gulp
C
codecalm 已提交
136
		.src(`{${distDir}/*,${demoDir}/*}`, { read: false })
P
Paweł Kuna 已提交
137 138
		.pipe(clean())
})
C
codecalm 已提交
139

C
codecalm 已提交
140
gulp.task('clean-jekyll', (cb) => {
C
codecalm 已提交
141
	return spawn('bundle', ['exec', 'jekyll', 'clean'], { stdio: 'inherit' })
P
Paweł Kuna 已提交
142 143
		.on('close', cb)
})
C
codecalm 已提交
144

C
codecalm 已提交
145
/**
C
codecalm 已提交
146
 * Compile SASS to CSS and move it to dist directory
C
codecalm 已提交
147
 */
C
codecalm 已提交
148
gulp.task('sass', () => {
P
Paweł Kuna 已提交
149
	const g = gulp
150
		.src(argv.withPlugins || BUILD ? `${srcDir}/scss/!(_)*.scss` : `${srcDir}/scss/+(tabler|demo).scss`)
C
codecalm 已提交
151
		.pipe(debug())
C
codecalm 已提交
152 153
		.pipe(sass({
			style: 'expanded',
C
codecalm 已提交
154
			precision: 7,
C
codecalm 已提交
155 156
			importer: (url, prev, done) => {
				if (url[0] === '~') {
P
Paweł Kuna 已提交
157
					url = path.resolve('node_modules', url.substr(1))
C
codecalm 已提交
158 159
				}

P
Paweł Kuna 已提交
160
				return { file: url }
C
codecalm 已提交
161 162 163 164 165 166 167 168
			},
		}).on('error', sass.logError))
		.pipe(postcss([
			require('autoprefixer'),
		]))
		.pipe(gulp.dest(`${distDir}/css/`))
		.pipe(browserSync.reload({
			stream: true,
C
codecalm 已提交
169
		}))
P
Paweł Kuna 已提交
170 171 172 173 174 175 176 177 178 179 180

	if (BUILD) {
		g
			.pipe(rtlcss())
			.pipe(rename((path) => {
				path.basename += '.rtl'
			}))
			.pipe(gulp.dest(`${distDir}/css/`))
	}

	return g
P
Paweł Kuna 已提交
181
})
C
codecalm 已提交
182

C
codecalm 已提交
183 184 185
/**
 * CSS minify
 */
P
Paweł Kuna 已提交
186
gulp.task('css-minify', function () {
C
codecalm 已提交
187 188 189 190
	return gulp.src(`${distDir}/css/!(*.min).css`)
		.pipe(debug())
		.pipe(cleanCSS())
		.pipe(rename((path) => {
P
Paweł Kuna 已提交
191
			path.basename += '.min'
C
codecalm 已提交
192
		}))
P
Paweł Kuna 已提交
193 194
		.pipe(gulp.dest(`${distDir}/css/`))
})
C
codecalm 已提交
195

C
codecalm 已提交
196 197 198
/**
 * Compile JS files to dist directory
 */
P
Paweł Kuna 已提交
199 200 201 202 203 204 205
let cache = {}

const compileJs = function (name, mjs = false) {
	if (!cache[name]) {
		cache[name] = null
	}

C
codecalm 已提交
206
	const g = rollupStream({
P
Paweł Kuna 已提交
207 208
		input: `${srcDir}/js/${name}.js`,
		cache: cache[name],
C
codecalm 已提交
209
		output: {
P
Paweł Kuna 已提交
210 211 212
			name: `${name}.js`,
			format: mjs ? 'es' : 'umd',
			...(mjs ? { exports: 'named' } : {})
C
codecalm 已提交
213 214
		},
		plugins: [
C
codecalm 已提交
215 216
			rollupReplace({
				'process.env.NODE_ENV': JSON.stringify(BUILD ? 'production' : 'development'),
217
				preventAssignment: false
C
codecalm 已提交
218
			}),
C
codecalm 已提交
219 220 221 222 223 224 225 226 227
			rollupBabel({
				exclude: 'node_modules/**'
			}),
			nodeResolve(),
			rollupCommonjs(),
			rollupCleanup()
		]
	})
		.on('bundle', (bundle) => {
P
Paweł Kuna 已提交
228
			cache[name] = bundle
C
codecalm 已提交
229
		})
P
Paweł Kuna 已提交
230
		.pipe(vinylSource(`${name}.js`))
C
codecalm 已提交
231
		.pipe(vinylBuffer())
C
codecalm 已提交
232
		.pipe(rename((path) => {
P
Paweł Kuna 已提交
233
			path.dirname = ''
C
codecalm 已提交
234 235 236 237
		}))
		.pipe(gulp.dest(`${distDir}/js/`))
		.pipe(browserSync.reload({
			stream: true,
P
Paweł Kuna 已提交
238
		}))
C
codecalm 已提交
239 240 241 242 243 244 245 246

	if (BUILD) {
		g.pipe(minifyJS({
			ext: {
				src: '.js',
				min: '.min.js'
			},
		}))
P
Paweł Kuna 已提交
247
			.pipe(gulp.dest(`${distDir}/js/`))
C
codecalm 已提交
248 249
	}

P
Paweł Kuna 已提交
250 251 252 253 254 255 256 257 258 259 260 261 262
	return g
}

/**
 * Compile JS files to dist directory
 */
gulp.task('js', () => {
	return compileJs('tabler')
})

gulp.task('js-demo', () => {
	return compileJs('demo')
})
C
codecalm 已提交
263

264 265 266
/**
 * Compile JS module files to dist directory
 */
P
Paweł Kuna 已提交
267 268 269 270 271
gulp.task('mjs', () => {
	return compileJs('tabler.esm', true)
})

let cacheEsm
272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294
gulp.task('mjs', () => {
	const g = rollupStream({
		input: `${srcDir}/js/tabler.esm.js`,
		cache: cacheEsm,
		output: {
			name: 'tabler.esm.js',
			format: 'es',
			exports: 'named'
		},
		plugins: [
			rollupReplace({
				'process.env.NODE_ENV': JSON.stringify(BUILD ? 'production' : 'development'),
				preventAssignment: false
			}),
			rollupBabel({
				exclude: 'node_modules/**'
			}),
			nodeResolve(),
			rollupCommonjs(),
			rollupCleanup()
		]
	})
		.on('bundle', (bundle) => {
P
Paweł Kuna 已提交
295
			cacheEsm = bundle
296 297 298 299
		})
		.pipe(vinylSource('tabler.esm.js'))
		.pipe(vinylBuffer())
		.pipe(rename((path) => {
P
Paweł Kuna 已提交
300
			path.dirname = ''
301 302 303 304
		}))
		.pipe(gulp.dest(`${distDir}/js/`))
		.pipe(browserSync.reload({
			stream: true,
P
Paweł Kuna 已提交
305
		}))
306 307 308 309 310 311 312 313

	if (BUILD) {
		g.pipe(minifyJS({
			ext: {
				src: '.js',
				min: '.min.js'
			},
		}))
P
Paweł Kuna 已提交
314
			.pipe(gulp.dest(`${distDir}/js/`))
315 316
	}

P
Paweł Kuna 已提交
317 318
	return g
})
319

C
codecalm 已提交
320 321 322
/**
 * Watch Jekyll files and build it to demo directory
 */
C
codecalm 已提交
323
gulp.task('watch-jekyll', (cb) => {
P
Paweł Kuna 已提交
324
	browserSync.notify('Building Jekyll')
C
codecalm 已提交
325
	return spawn('bundle', ['exec', 'jekyll', 'build', '--watch', '--destination', demoDir, '--trace'], { stdio: 'inherit' })
P
Paweł Kuna 已提交
326 327
		.on('close', cb)
})
C
codecalm 已提交
328

C
codecalm 已提交
329 330 331
/**
 * Build Jekyll files do demo directory
 */
C
codecalm 已提交
332
gulp.task('build-jekyll', (cb) => {
P
Paweł Kuna 已提交
333
	var env = Object.create(process.env)
C
codecalm 已提交
334

P
Paweł Kuna 已提交
335 336 337 338
	if (argv.preview) {
		env.JEKYLL_ENV = 'preview'
	} else {
		env.JEKYLL_ENV = 'production'
C
codecalm 已提交
339
	}
C
codecalm 已提交
340

P
Paweł Kuna 已提交
341 342 343 344 345 346
	return spawn('bundle', ['exec', 'jekyll', 'build', '--destination', demoDir, '--trace'], {
		env: env,
		stdio: 'inherit'
	})
		.on('close', cb)
})
C
codecalm 已提交
347

C
sitemap  
codecalm 已提交
348 349 350
gulp.task('build-cleanup', () => {
	return gulp
		.src(`${demoDir}/redirects.json`, { read: false, allowEmpty: true })
P
Paweł Kuna 已提交
351 352
		.pipe(clean())
})
C
sitemap  
codecalm 已提交
353

C
codecalm 已提交
354
gulp.task('build-purgecss', (cb) => {
P
Paweł Kuna 已提交
355
	if (argv.preview) {
C
codecalm 已提交
356 357 358 359 360 361 362
		return gulp.src('demo/dist/{libs,css}/**/*.css')
			.pipe(purgecss({
				content: ['demo/**/*.html']
			}))
			.pipe(gulp.dest('demo/dist/css'))
	}

P
Paweł Kuna 已提交
363 364
	cb()
})
C
codecalm 已提交
365 366

gulp.task('build-critical', (cb) => {
P
Paweł Kuna 已提交
367
	if (argv.preview) {
C
codecalm 已提交
368 369 370 371 372 373
		return gulp
			.src('demo/**/*.html')
			.pipe(
				critical({
					base: 'demo/',
					inline: true,
C
codecalm 已提交
374 375 376 377 378 379 380
					css: ['demo/dist/css/tabler.css'],
					ignore: {
						atrule: ['@font-face', '@import'],
						decl: (node, value) => {
							/url\(/.test(value)
						},
					},
C
codecalm 已提交
381 382 383
				})
			)
			.on('error', err => {
P
Paweł Kuna 已提交
384
				console.log(err.message)
C
codecalm 已提交
385
			})
P
Paweł Kuna 已提交
386
			.pipe(gulp.dest('demo'))
C
codecalm 已提交
387 388
	}

P
Paweł Kuna 已提交
389 390
	cb()
})
C
codecalm 已提交
391

C
codecalm 已提交
392 393 394
/**
 * Watch JS and SCSS files
 */
C
codecalm 已提交
395
gulp.task('watch', (cb) => {
P
Paweł Kuna 已提交
396 397 398 399
	gulp.watch('./src/scss/**/*.scss', gulp.series('sass'))
	gulp.watch('./src/js/**/*.js', gulp.parallel('js', 'mjs', 'js-demo'))
	cb()
})
C
codecalm 已提交
400

C
codecalm 已提交
401 402 403
/**
 * Create BrowserSync server
 */
C
codecalm 已提交
404
gulp.task('browser-sync', () => {
C
codecalm 已提交
405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421
	browserSync({
		watch: true,
		server: {
			baseDir: demoDir,
			routes: {
				'/node_modules': 'node_modules',
				'/dist/css': `${distDir}/css`,
				'/dist/js': `${distDir}/js`,
				'/dist/img': `${srcDir}/img`,
				'/static': `${srcDir}/static`,
			},
		},
		port: 3000,
		open: false,
		host: 'localhost',
		notify: false,
		reloadOnRestart: true
P
Paweł Kuna 已提交
422 423
	})
})
C
codecalm 已提交
424

C
codecalm 已提交
425 426 427
/**
 * Copy libs used in tabler from npm to dist directory
 */
C
codecalm 已提交
428
gulp.task('copy-libs', (cb) => {
P
Paweł Kuna 已提交
429
	const allLibs = require(`${srcDir}/pages/_data/libs`)
C
codecalm 已提交
430

P
Paweł Kuna 已提交
431
	let files = []
C
codecalm 已提交
432

C
codecalm 已提交
433
	Object.keys(allLibs.js).forEach((lib) => {
P
Paweł Kuna 已提交
434 435
		files.push(Array.isArray(allLibs.js[lib]) ? allLibs.js[lib] : [allLibs.js[lib]])
	})
C
codecalm 已提交
436

C
codecalm 已提交
437
	Object.keys(allLibs.css).forEach((lib) => {
P
Paweł Kuna 已提交
438 439
		files.push(Array.isArray(allLibs.css[lib]) ? allLibs.css[lib] : [allLibs.css[lib]])
	})
C
codecalm 已提交
440

P
Paweł Kuna 已提交
441
	files = files.flat()
C
codecalm 已提交
442

C
codecalm 已提交
443 444
	files.forEach((file) => {
		if (!file.match(/^https?/)) {
P
Paweł Kuna 已提交
445 446
			let dirname = path.dirname(file).replace('@', '')
			let cmd = `mkdir -p "dist/libs/${dirname}" && cp -r node_modules/${file} ${distDir}/libs/${file.replace('@', '')}`
C
codecalm 已提交
447 448 449

			cp.exec(cmd)
		}
P
Paweł Kuna 已提交
450
	})
C
codecalm 已提交
451

P
Paweł Kuna 已提交
452 453
	cb()
})
C
codecalm 已提交
454

C
codecalm 已提交
455 456 457
/**
 * Copy static files (flags, payments images, etc) to dist directory
 */
C
codecalm 已提交
458
gulp.task('copy-images', () => {
C
codecalm 已提交
459 460
	return gulp
		.src(`${srcDir}/img/**/*`)
P
Paweł Kuna 已提交
461 462
		.pipe(gulp.dest(`${distDir}/img`))
})
C
codecalm 已提交
463

C
codecalm 已提交
464 465 466
/**
 * Copy static files (demo images, etc) to demo directory
 */
C
codecalm 已提交
467
gulp.task('copy-static', () => {
C
codecalm 已提交
468 469
	return gulp
		.src(`${srcDir}/static/**/*`)
P
Paweł Kuna 已提交
470 471
		.pipe(gulp.dest(`${demoDir}/static`))
})
C
codecalm 已提交
472

C
codecalm 已提交
473 474 475
/**
 * Copy Tabler dist files to demo directory
 */
C
codecalm 已提交
476
gulp.task('copy-dist', () => {
C
codecalm 已提交
477 478
	return gulp
		.src(`${distDir}/**/*`)
P
Paweł Kuna 已提交
479 480
		.pipe(gulp.dest(`${demoDir}/dist/`))
})
C
codecalm 已提交
481

C
codecalm 已提交
482 483 484
/**
 * Add banner to build JS and CSS files
 */
C
codecalm 已提交
485 486 487 488
gulp.task('add-banner', () => {
	return gulp.src(`${distDir}/{css,js}/**/*.{js,css}`)
		.pipe(header(getBanner()))
		.pipe(gulp.dest(`${distDir}`))
P
Paweł Kuna 已提交
489
})
C
codecalm 已提交
490

P
Paweł Kuna 已提交
491
gulp.task('clean', gulp.series('clean-dirs', 'clean-jekyll'))
C
codecalm 已提交
492

P
Paweł Kuna 已提交
493
gulp.task('start', gulp.series('clean', 'sass', 'js', 'js-demo', 'mjs', 'build-jekyll', gulp.parallel('watch-jekyll', 'watch', 'browser-sync')))
C
codecalm 已提交
494

P
Paweł Kuna 已提交
495 496 497
gulp.task('build-core', gulp.series('build-on', 'clean', 'sass', 'css-minify', 'js', 'js-demo', 'mjs', 'copy-images', 'copy-libs', 'add-banner'))
gulp.task('build-demo', gulp.series('build-on', 'build-jekyll', 'copy-static', 'copy-dist', 'build-cleanup', 'build-purgecss'/*, 'build-critical'*/))
gulp.task('build', gulp.series('build-core', 'build-demo'))