import gulp from "gulp";
import nunjucksRender from "gulp-nunjucks-render";
import plumber from "gulp-plumber";
import data from "gulp-data";
import cached from "gulp-cached";
import fs from "fs";
import del from "del";
import ws from "gulp-webserver";
import path from "path";
import gulpSass from "gulp-sass";
import dartSass from "dart-sass";
import sourcemaps from "gulp-sourcemaps";
import minificss from "gulp-minify-css";
import autoprefixer from "autoprefixer";
import postCss from "gulp-postcss";
import rename from "gulp-rename";
import dependents from "gulp-dependents";
import bro from "gulp-bro";
import babelify from "babelify";
import minify from "gulp-minify";
import imagemin from "gulp-imagemin";
const src = './src';
const dist = './dist';
const ass = '/assets';
const path_src = {
html: src + '/html',
css: src + ass + '/css',
images: src + ass + '/images',
js: src + ass + '/js',
}
const path_dist = {
html: dist,
css: dist + ass + '/css',
images: dist + ass + '/images',
js: dist + ass + '/js',
};
const onErrorHandler = (error) => console.log(error);
const html = () => {
const manageEnvironment = (environment) => {
environment.addFilter('tabIndent', (str, numOfIndents, firstLine) => {
str = str.replace(/^(?=.)/gm, new Array(numOfIndents + 1).join('\t'));
if(!firstLine) {
str = str.replace(/^\s+/, "");
}
return str;
});
};
const gnbJson = JSON.parse(fs.readFileSync(path_src.html + '/_templates/_json/_gnb.json'));
const json_all = {...gnbJson};
const datafile = () => {
return json_all;
}
return gulp.src([
path_src.html + '/**/*',
'!' + path_src.html + '/**/_*',
'!' + path_src.html + '/**/_*/**/*'
])
.pipe( plumber({errorHandler:onErrorHandler}) )
.pipe( data( datafile) )
.pipe( nunjucksRender({
envOptions: {
autoescape: false,
},
manageEnv: manageEnvironment,
path: [path_src.html],
}) )
.pipe( cached('html') )
.pipe( gulp.dest(path_dist.html) )
}
const css = () => {
const sass = gulpSass(dartSass);
const options = {
scss : {
outputStyle: "expanded",
indentType: "space",
indentWidth: 2,
precision: 8,
sourceComments: true,
compiler: dartSass,
},
postcss: [ autoprefixer({
overrideBrowserslist: 'last 2 versions',
}) ]
};
return gulp.src(
path_src.css + '/**/*.scss',
{ since: gulp.lastRun(css) }
)
.pipe( plumber({errorHandler:onErrorHandler}) )
.pipe( dependents() )
.pipe( sourcemaps.init() )
.pipe( sass(options.scss).on('error', sass.logError) )
.pipe( postCss(options.postcss) )
.pipe( sourcemaps.write() )
.pipe( gulp.dest(path_dist.css) )
.pipe( minificss() )
.pipe( rename({ suffix: '.min' }) )
.pipe( sourcemaps.write() )
.pipe( gulp.dest(path_dist.css) );
}
const js = () => {
return gulp.src([
path_src.js + '/main.js'
])
.pipe( sourcemaps.init({ loadMaps: true }) )
.pipe( bro({
transform: [
babelify.configure({ presets: ['@babel/preset-env'] }),
[ 'uglifyify', { global: true } ]
]
}) )
.pipe( sourcemaps.write('./') )
.pipe(minify({
ext: { min: '.min.js' },
ignoreFiles: ['-min.js']
}))
.pipe( gulp.dest(path_dist.js) );
}
const image = () => {
return gulp.src( path_src.images + '/**/*' )
.pipe( gulp.dest( path_dist.images ) );
}
const image_optimization = () => {
return gulp.src( path_src.images + '/**/*' )
.pipe( imagemin( { verbose:true } ) )
.pipe( gulp.dest( path_dist.images ) );
}
const clean = () => del([dist]);
const webserver = () => {
return gulp.src(dist)
.pipe(
ws({
port: 8300,
livereload: true,
open: true
})
);
}
const watch = () => {
const html_watcher = gulp.watch(path_src.html + "/**/*", html);
file_management(html_watcher, path_src.html, path_dist.html);
const scss_watcher = gulp.watch(path_src.css + "/**/*", css);
file_management(scss_watcher, path_src.css, path_dist.css);
const js_watcher = gulp.watch(path_src.js + "/**/*", js);
file_management(js_watcher, path_src.js, path_dist.js);
const image_watcher = gulp.watch(path_src.images + "/**/*", image);
file_management(image_watcher, path_src.images, path_dist.images);
}
const file_management = (watcher_target, src_path, dist_path) => {
watcher_target.on('unlink', (filepath) => {
const filePathFromSrc = path.relative(path.resolve(src_path), filepath);
const extension_type = filePathFromSrc.split('.')[filePathFromSrc.split('.').length-1];
if( extension_type === 'scss' ){
const destFilePath_css = path.resolve(dist_path, filePathFromSrc).replace('.scss','.css');
del.sync(destFilePath_css);
const destFilePath_minCss = path.resolve(dist_path, filePathFromSrc).replace('.scss','.min.css');
del.sync(destFilePath_minCss);
}
else if( extension_type === 'js' ){
const destFilePath_js = path.resolve(dist_path, filePathFromSrc);
del.sync(destFilePath_js);
const destFilePath_minJs = path.resolve(dist_path, filePathFromSrc).replace('.js','.min.js');
del.sync(destFilePath_minJs);
}
else if( extension_type === 'njk' ){
const destFilePath_html = path.resolve(dist_path, filePathFromSrc).replace('.njk','.html');
del.sync(destFilePath_html);
}
else{
const destFilePath = path.resolve(dist_path, filePathFromSrc);
del.sync(destFilePath);
}
});
}
const prepare = gulp.series([ clean, image ]);
const assets = gulp.series([ html, css, js ]);
const live = gulp.parallel([ webserver, watch ]);
export const build = gulp.series([ prepare, assets ]);
export const dev = gulp.series([ build, live ]);
export const image_min = gulp.series([ image_optimization ]);