var gulp = require("gulp"); var uglify = require("gulp-uglify"); var typescript = require("gulp-typescript"); var sourcemaps = require("gulp-sourcemaps"); var srcToVariable = require("gulp-content-to-variable"); var appendSrcToVariable = require("./gulp-appendSrcToVariable"); var addDtsExport = require("./gulp-addDtsExport"); var addModuleExports = require("./gulp-addModuleExports"); var merge2 = require("merge2"); var concat = require("gulp-concat"); var rename = require("gulp-rename"); var cleants = require('gulp-clean-ts-extends'); var changedInPlace = require('gulp-changed-in-place'); var runSequence = require('run-sequence'); var replace = require("gulp-replace"); var uncommentShader = require("./gulp-removeShaderComments"); var expect = require('gulp-expect-file'); var optimisejs = require('gulp-optimize-js'); var webserver = require('gulp-webserver'); var path = require('path'); var config = require("./config.json"); var customConfig = require("./custom.config.json"); var debug = require('gulp-debug'); var includeShadersStream; var shadersStream; var workersStream; var extendsSearchRegex = /var\s__extends[\s\S]+?\};/g; var decorateSearchRegex = /var\s__decorate[\s\S]+?\};/g; /** * TS configurations shared in the gulp file. */ var tsConfig = { noExternalResolve: true, target: 'ES5', declarationFiles: true, typescript: require('typescript'), experimentalDecorators: true, isolatedModules: false }; var tsProject = typescript.createProject(tsConfig); var externalTsConfig = { noExternalResolve: false, target: 'ES5', declarationFiles: true, typescript: require('typescript'), experimentalDecorators: true, isolatedModules: false }; /* * Shader Management. */ function shadersName(filename) { return path.basename(filename) .replace('.fragment', 'Pixel') .replace('.vertex', 'Vertex') .replace('.fx', 'Shader'); } function includeShadersName(filename) { return filename.replace('.fx', ''); } /* * Main necessary files stream Management. */ gulp.task("includeShaders", function (cb) { includeShadersStream = config.includeShadersDirectories.map(function (shadersDef) { return gulp.src(shadersDef.files). pipe(expect.real({ errorOnFailure: true }, shadersDef.files)). pipe(uncommentShader()). pipe(srcToVariable({ variableName: shadersDef.variable, asMap: true, namingCallback: includeShadersName })); }); cb(); }); gulp.task("shaders", ["includeShaders"], function (cb) { shadersStream = config.shadersDirectories.map(function (shadersDef) { return gulp.src(shadersDef.files). pipe(expect.real({ errorOnFailure: true }, shadersDef.files)). pipe(uncommentShader()). pipe(srcToVariable({ variableName: shadersDef.variable, asMap: true, namingCallback: shadersName })); }); cb(); }); gulp.task("workers", function (cb) { workersStream = config.workers.map(function (workerDef) { return gulp.src(workerDef.files). pipe(expect.real({ errorOnFailure: true }, workerDef.files)). pipe(uglify()). pipe(srcToVariable({ variableName: workerDef.variable })); }); cb(); }); /** * Build tasks to concat minify uflify optimise the BJS js in different flavor (workers...). */ gulp.task("buildCore", ["shaders"], function () { return merge2( gulp.src(config.core.files). pipe(expect.real({ errorOnFailure: true }, config.core.files)), shadersStream, includeShadersStream ) .pipe(concat(config.build.minCoreFilename)) .pipe(cleants()) .pipe(replace(extendsSearchRegex, "")) .pipe(replace(decorateSearchRegex, "")) .pipe(addModuleExports("BABYLON")) .pipe(uglify()) .pipe(optimisejs()) .pipe(gulp.dest(config.build.outputDirectory)); }); gulp.task("buildNoWorker", ["shaders"], function () { return merge2( gulp.src(config.core.files). pipe(expect.real({ errorOnFailure: true }, config.core.files)), gulp.src(config.extras.files). pipe(expect.real({ errorOnFailure: true }, config.extras.files)), shadersStream, includeShadersStream ) .pipe(concat(config.build.minNoWorkerFilename)) .pipe(cleants()) .pipe(replace(extendsSearchRegex, "")) .pipe(replace(decorateSearchRegex, "")) .pipe(addModuleExports("BABYLON")) .pipe(uglify()) .pipe(optimisejs()) .pipe(gulp.dest(config.build.outputDirectory)); }); gulp.task("build", ["workers", "shaders"], function () { return merge2( gulp.src(config.core.files). pipe(expect.real({ errorOnFailure: true }, config.core.files)), gulp.src(config.extras.files). pipe(expect.real({ errorOnFailure: true }, config.extras.files)), shadersStream, includeShadersStream, workersStream ) .pipe(concat(config.build.filename)) .pipe(cleants()) .pipe(replace(extendsSearchRegex, "")) .pipe(replace(decorateSearchRegex, "")) .pipe(addModuleExports("BABYLON")) .pipe(gulp.dest(config.build.outputDirectory)) .pipe(rename(config.build.minFilename)) .pipe(uglify()) .pipe(optimisejs()) .pipe(gulp.dest(config.build.outputDirectory)); }); gulp.task("build-custom", ["shaders"], function () { return merge2( gulp.src(customConfig.core.files). pipe(expect.real({ errorOnFailure: true }, customConfig.core.files)), shadersStream, includeShadersStream ) .pipe(concat(customConfig.build.filename)) .pipe(cleants()) .pipe(replace(extendsSearchRegex, "")) .pipe(replace(decorateSearchRegex, "")) .pipe(addModuleExports("BABYLON")) .pipe(gulp.dest(customConfig.build.outputDirectory)) .pipe(rename(customConfig.build.minFilename)) .pipe(uglify()) .pipe(optimisejs()) .pipe(gulp.dest(customConfig.build.outputDirectory)); }); /* * Compiles all typescript files and creating a js and a declaration file. */ gulp.task('typescript-compile', function () { var tsResult = gulp.src(config.core.typescript) .pipe(sourcemaps.init()) .pipe(typescript(tsProject)); //If this gulp task is running on travis, file the build! if (process.env.TRAVIS) { var error = false; tsResult.on('error', function () { error = true; }).on('end', function () { if (error) { console.log('Typescript compile failed'); process.exit(1); } }); } return merge2([ tsResult.dts .pipe(concat(config.build.declarationFilename)) //.pipe(addDtsExport("BABYLON")) .pipe(gulp.dest(config.build.outputDirectory)), tsResult.js .pipe(sourcemaps.write("./", { includeContent:false, sourceRoot: (filePath) => { return ''; } })) .pipe(gulp.dest(config.build.srcOutputDirectory)) ]) }); /** * Helper methods to build external library (mat, post processes, ...). */ var buildExternalLibraries = function(settings) { var tasks = settings.libraries.map(function (library) { return buildExternalLibrary(library, settings); }); return merge2(tasks); } var buildExternalLibrary= function(library, settings) { var tsProcess = gulp.src(library.files, {base: settings.build.srcOutputDirectory}) .pipe(sourcemaps.init()) .pipe(typescript(externalTsConfig)); var shader = gulp.src(library.shaderFiles || [], {base: settings.build.srcOutputDirectory} ) .pipe(uncommentShader()) .pipe(appendSrcToVariable("BABYLON.Effect.ShadersStore", shadersName, library.output + '.fx')) .pipe(gulp.dest(settings.build.srcOutputDirectory)); var dev = tsProcess.js.pipe(sourcemaps.write("./", { includeContent:false, sourceRoot: (filePath) => { return ''; } })) .pipe(gulp.dest(settings.build.srcOutputDirectory)); var outputDirectory = config.build.outputDirectory + settings.build.distOutputDirectory; var dist = merge2(tsProcess.js, shader) .pipe(concat(library.output)) .pipe(gulp.dest(outputDirectory)) .pipe(cleants()) .pipe(replace(extendsSearchRegex, "")) .pipe(replace(decorateSearchRegex, "")) .pipe(rename({extname: ".min.js"})) .pipe(uglify()) .pipe(optimisejs()) .pipe(gulp.dest(outputDirectory)); return merge2(dev, dist); } /** * The default task, concat and min the main BJS files. */ gulp.task('default', function (cb) { runSequence("buildNoWorker", "build", "buildCore", cb); }); /** * Build the releasable files. */ gulp.task("typescript", function (cb) { runSequence("typescript-compile", "default", cb); }); /** * Dynamic module creation. */ config.modules.map(function (module) { gulp.task(module, function () { return buildExternalLibraries(config[module]); }); }); gulp.task("typescript-libraries", config.modules, function () { }); /** * Do it all. */ gulp.task("typescript-all", function (cb) { runSequence("typescript", "typescript-libraries", cb); }); /** * Watch ts files and fire repective tasks. */ gulp.task('watch', [], function () { var tasks = [gulp.watch(config.core.typescript, ['typescript-compile'])]; config.modules.map(function (module) { config[module].libraries.map(function (library) { tasks.push(gulp.watch(library.files, function() { console.log(library.output); return buildExternalLibrary(library, config[module]) .pipe(debug()); })); tasks.push(gulp.watch(library.shaderFiles, function() { console.log(library.output); return buildExternalLibrary(library, config[module]) .pipe(debug()) })); }); }); return tasks; }); /** * Embedded local dev env management. */ gulp.task('deployLocalDev', function () { gulp.src('../../localDev/template/**.*') .pipe(gulp.dest('../../localDev/src/')); }); /** * Embedded webserver for test convenience. */ gulp.task('webserver', function () { gulp.src('../../.').pipe(webserver({ port: 1338, livereload: false     })); }); /** * Combine Webserver and Watch as long as vscode does not handle multi tasks. */ gulp.task('run', ['watch', 'webserver'], function () { });