gulpfile.js 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528
  1. var gulp = require("gulp");
  2. var uglify = require("gulp-uglify");
  3. var typescript = require("gulp-typescript");
  4. var sourcemaps = require("gulp-sourcemaps");
  5. var srcToVariable = require("gulp-content-to-variable");
  6. var appendSrcToVariable = require("./gulp-appendSrcToVariable");
  7. var addDtsExport = require("./gulp-addDtsExport");
  8. var addModuleExports = require("./gulp-addModuleExports");
  9. var merge2 = require("merge2");
  10. var concat = require("gulp-concat");
  11. var rename = require("gulp-rename");
  12. var cleants = require('gulp-clean-ts-extends');
  13. var changedInPlace = require('gulp-changed-in-place');
  14. var runSequence = require('run-sequence');
  15. var replace = require("gulp-replace");
  16. var uncommentShader = require("./gulp-removeShaderComments");
  17. var expect = require('gulp-expect-file');
  18. var optimisejs = require('gulp-optimize-js');
  19. var webserver = require('gulp-webserver');
  20. var path = require('path');
  21. var sass = require('gulp-sass');
  22. var webpack = require('webpack-stream');
  23. var zip = require('gulp-zip');
  24. var config = require("./config.json");
  25. var del = require('del');
  26. var debug = require('gulp-debug');
  27. var includeShadersStream;
  28. var shadersStream;
  29. var workersStream;
  30. var extendsSearchRegex = /var\s__extends[\s\S]+?\}\)\(\);/g;
  31. var decorateSearchRegex = /var\s__decorate[\s\S]+?\};/g;
  32. var referenceSearchRegex = /\/\/\/ <reference.*/g;
  33. /**
  34. * TS configurations shared in the gulp file.
  35. */
  36. var tsConfig = {
  37. noResolve: true,
  38. target: 'ES5',
  39. declarationFiles: true,
  40. typescript: require('typescript'),
  41. experimentalDecorators: true,
  42. isolatedModules: false,
  43. noImplicitAny: true,
  44. noImplicitReturns: true
  45. };
  46. var tsProject = typescript.createProject(tsConfig);
  47. var externalTsConfig = {
  48. noResolve: false,
  49. target: 'ES5',
  50. declarationFiles: true,
  51. typescript: require('typescript'),
  52. experimentalDecorators: true,
  53. isolatedModules: false,
  54. noImplicitAny: true,
  55. noImplicitReturns: true
  56. };
  57. function processDependency(kind, dependency, filesToLoad) {
  58. if (dependency.dependUpon) {
  59. for (var i = 0; i < dependency.dependUpon.length; i++) {
  60. var dependencyName = dependency.dependUpon[i];
  61. var parent = config.workloads[dependencyName];
  62. processDependency(kind, parent, filesToLoad);
  63. }
  64. }
  65. var content = dependency[kind];
  66. if (!content) {
  67. return;
  68. }
  69. for (var i = 0; i < content.length; i++) {
  70. var file = content[i];
  71. if (filesToLoad.indexOf(file) === -1) {
  72. filesToLoad.push(file);
  73. }
  74. }
  75. }
  76. function determineFilesToProcess(kind) {
  77. var currentConfig = config.build.currentConfig;
  78. var buildConfiguration = config.buildConfigurations[currentConfig];
  79. var filesToLoad = [];
  80. for (var index = 0; index < buildConfiguration.length; index++) {
  81. var dependencyName = buildConfiguration[index];
  82. var dependency = config.workloads[dependencyName];
  83. if (kind === "directFiles" && !dependency) {
  84. filesToLoad.push("../../dist/preview release/" + dependencyName);
  85. }
  86. else if (dependency) {
  87. processDependency(kind, dependency, filesToLoad);
  88. }
  89. }
  90. if (kind === "shaderIncludes") {
  91. for (var index = 0; index < filesToLoad.length; index++) {
  92. filesToLoad[index] = "../../src/Shaders/ShadersInclude/" + filesToLoad[index] + ".fx";
  93. }
  94. } else if (kind === "shaders") {
  95. for (var index = 0; index < filesToLoad.length; index++) {
  96. var name = filesToLoad[index];
  97. filesToLoad[index] = "../../src/Shaders/" + filesToLoad[index] + ".fx";
  98. }
  99. }
  100. return filesToLoad;
  101. }
  102. /*
  103. * Shader Management.
  104. */
  105. function shadersName(filename) {
  106. return path.basename(filename)
  107. .replace('.fragment', 'Pixel')
  108. .replace('.vertex', 'Vertex')
  109. .replace('.fx', 'Shader');
  110. }
  111. function includeShadersName(filename) {
  112. return path.basename(filename).replace('.fx', '');
  113. }
  114. /*
  115. * Main necessary files stream Management.
  116. */
  117. gulp.task("includeShaders", function (cb) {
  118. var filesToProcess = determineFilesToProcess("shaderIncludes");
  119. includeShadersStream = gulp.src(filesToProcess).
  120. pipe(expect.real({ errorOnFailure: true }, filesToProcess)).
  121. pipe(uncommentShader()).
  122. pipe(srcToVariable({
  123. variableName: "BABYLON.Effect.IncludesShadersStore", asMap: true, namingCallback: includeShadersName
  124. }));
  125. cb();
  126. });
  127. gulp.task("shaders", ["includeShaders"], function (cb) {
  128. var filesToProcess = determineFilesToProcess("shaders");
  129. shadersStream = gulp.src(filesToProcess).
  130. pipe(expect.real({ errorOnFailure: true }, filesToProcess)).
  131. pipe(uncommentShader()).
  132. pipe(srcToVariable({
  133. variableName: "BABYLON.Effect.ShadersStore", asMap: true, namingCallback: shadersName
  134. }));
  135. cb();
  136. });
  137. gulp.task("workers", function (cb) {
  138. workersStream = config.workers.map(function (workerDef) {
  139. return gulp.src(workerDef.files).
  140. pipe(expect.real({ errorOnFailure: true }, workerDef.files)).
  141. pipe(uglify()).
  142. pipe(srcToVariable({
  143. variableName: workerDef.variable
  144. }));
  145. });
  146. cb();
  147. });
  148. /**
  149. * Build tasks to concat minify uflify optimise the BJS js in different flavor (workers...).
  150. */
  151. gulp.task("buildWorker", ["workers", "shaders"], function () {
  152. var filesToProcess = determineFilesToProcess("files");
  153. return merge2(
  154. gulp.src(filesToProcess).
  155. pipe(expect.real({ errorOnFailure: true }, filesToProcess)),
  156. shadersStream,
  157. includeShadersStream,
  158. workersStream
  159. )
  160. .pipe(concat(config.build.minWorkerFilename))
  161. .pipe(cleants())
  162. .pipe(replace(extendsSearchRegex, ""))
  163. .pipe(replace(decorateSearchRegex, ""))
  164. .pipe(addModuleExports("BABYLON"))
  165. .pipe(uglify())
  166. .pipe(optimisejs())
  167. .pipe(gulp.dest(config.build.outputDirectory));
  168. });
  169. gulp.task("build", ["shaders"], function () {
  170. var filesToProcess = determineFilesToProcess("files");
  171. var directFilesToProcess = determineFilesToProcess("directFiles");
  172. return merge2(
  173. gulp.src(filesToProcess).
  174. pipe(expect.real({ errorOnFailure: true }, filesToProcess)),
  175. shadersStream,
  176. includeShadersStream,
  177. gulp.src(directFilesToProcess)
  178. )
  179. .pipe(concat(config.build.filename))
  180. .pipe(cleants())
  181. .pipe(replace(extendsSearchRegex, ""))
  182. .pipe(replace(decorateSearchRegex, ""))
  183. .pipe(addModuleExports("BABYLON"))
  184. .pipe(gulp.dest(config.build.outputDirectory))
  185. .pipe(rename(config.build.minFilename))
  186. .pipe(uglify())
  187. .pipe(optimisejs())
  188. .pipe(gulp.dest(config.build.outputDirectory));
  189. });
  190. /*
  191. * Compiles all typescript files and creating a js and a declaration file.
  192. */
  193. gulp.task('typescript-compile', function () {
  194. var tsResult = gulp.src(config.typescript)
  195. .pipe(sourcemaps.init())
  196. .pipe(tsProject());
  197. //If this gulp task is running on travis, file the build!
  198. if (process.env.TRAVIS) {
  199. var error = false;
  200. tsResult.on('error', function () {
  201. error = true;
  202. }).on('end', function () {
  203. if (error) {
  204. console.log('Typescript compile failed');
  205. process.exit(1);
  206. }
  207. });
  208. }
  209. return merge2([
  210. tsResult.dts
  211. .pipe(concat(config.build.declarationFilename))
  212. .pipe(gulp.dest(config.build.outputDirectory)),
  213. tsResult.dts
  214. .pipe(concat(config.build.declarationModuleFilename))
  215. .pipe(addDtsExport("BABYLON", "babylonjs"))
  216. .pipe(gulp.dest(config.build.outputDirectory)),
  217. tsResult.js
  218. .pipe(sourcemaps.write("./",
  219. {
  220. includeContent: false,
  221. sourceRoot: (filePath) => {
  222. return '';
  223. }
  224. }))
  225. .pipe(gulp.dest(config.build.srcOutputDirectory))
  226. ])
  227. });
  228. /**
  229. * Helper methods to build external library (mat, post processes, ...).
  230. */
  231. var buildExternalLibraries = function (settings) {
  232. var tasks = settings.libraries.map(function (library) {
  233. return buildExternalLibrary(library, settings, false);
  234. });
  235. let mergedTasks = merge2(tasks);
  236. if (settings.build.buildAsModule) {
  237. mergedTasks.on('end', function () {
  238. //generate js file list
  239. let files = settings.libraries.filter(function (lib) {
  240. return !lib.doNotIncludeInBundle;
  241. }).map(function (lib) {
  242. return config.build.outputDirectory + settings.build.distOutputDirectory + lib.output;
  243. });
  244. var outputDirectory = config.build.outputDirectory + settings.build.distOutputDirectory;
  245. let srcTask = gulp.src(files)
  246. .pipe(concat(settings.build.outputFilename + '.js'))
  247. .pipe(replace(extendsSearchRegex, ""))
  248. .pipe(replace(decorateSearchRegex, ""))
  249. .pipe(replace(referenceSearchRegex, ""))
  250. .pipe(addModuleExports(settings.build.moduleDeclaration, true, settings.build.extendsRoot))
  251. .pipe(gulp.dest(outputDirectory))
  252. .pipe(cleants())
  253. .pipe(rename({ extname: ".min.js" }))
  254. .pipe(uglify())
  255. .pipe(optimisejs())
  256. .pipe(gulp.dest(outputDirectory));
  257. let dtsFiles = files.map(function (filename) {
  258. return filename.replace(".js", ".d.ts");
  259. });
  260. let dtsTask = gulp.src(dtsFiles)
  261. .pipe(concat(settings.build.outputFilename + '.module.d.ts'))
  262. .pipe(replace(referenceSearchRegex, ""))
  263. .pipe(addDtsExport(settings.build.moduleDeclaration, settings.build.moduleName, true, settings.build.extendsRoot))
  264. .pipe(gulp.dest(outputDirectory));
  265. return merge2([srcTask, dtsTask]);
  266. });
  267. }
  268. return mergedTasks;
  269. }
  270. var buildExternalLibrary = function (library, settings, watch) {
  271. var tsProcess = gulp.src(library.files, { base: settings.build.srcOutputDirectory })
  272. .pipe(sourcemaps.init())
  273. .pipe(typescript(externalTsConfig));
  274. var includeShader = gulp.src(library.shadersIncludeFiles || [], { base: settings.build.srcOutputDirectory })
  275. .pipe(uncommentShader())
  276. .pipe(appendSrcToVariable("BABYLON.Effect.IncludesShadersStore", includeShadersName, library.output + '.include.fx'))
  277. .pipe(gulp.dest(settings.build.srcOutputDirectory));
  278. var shader = gulp.src(library.shaderFiles || [], { base: settings.build.srcOutputDirectory })
  279. .pipe(uncommentShader())
  280. .pipe(appendSrcToVariable("BABYLON.Effect.ShadersStore", shadersName, library.output + '.fx'))
  281. .pipe(gulp.dest(settings.build.srcOutputDirectory));
  282. var dev = tsProcess.js
  283. .pipe(sourcemaps.write("./", {
  284. includeContent: false,
  285. sourceRoot: (filePath) => {
  286. return '';
  287. }
  288. })).pipe(gulp.dest(settings.build.srcOutputDirectory));
  289. var outputDirectory = config.build.outputDirectory + settings.build.distOutputDirectory;
  290. var css = gulp.src(library.sassFiles || [])
  291. .pipe(sass().on('error', sass.logError))
  292. .pipe(concat(library.output.replace(".js", ".css")))
  293. .pipe(gulp.dest(outputDirectory));
  294. if (watch) {
  295. return merge2([shader, includeShader, dev, css]);
  296. }
  297. else {
  298. /*if (library.bundle) {
  299. // Don't remove extends and decorate functions
  300. var code = merge2([tsProcess.js, shader, includeShader])
  301. .pipe(concat(library.output));
  302. if (library.buildAsModule) {
  303. code = code.pipe(addModuleExports(library.moduleDeclaration, true))
  304. }
  305. code.pipe(gulp.dest(outputDirectory))
  306. .pipe(cleants())
  307. .pipe(rename({ extname: ".min.js" }))
  308. .pipe(uglify())
  309. .pipe(optimisejs())
  310. .pipe(gulp.dest(outputDirectory));
  311. } else {*/
  312. var code = merge2([tsProcess.js, shader, includeShader])
  313. .pipe(concat(library.output))
  314. if (library.buildAsModule) {
  315. code = code.pipe(replace(extendsSearchRegex, ""))
  316. .pipe(replace(decorateSearchRegex, ""))
  317. .pipe(addModuleExports(library.moduleDeclaration, true, library.extendsRoot))
  318. }
  319. code = code.pipe(gulp.dest(outputDirectory))
  320. .pipe(cleants())
  321. .pipe(rename({ extname: ".min.js" }))
  322. .pipe(uglify())
  323. .pipe(optimisejs())
  324. .pipe(gulp.dest(outputDirectory));
  325. /*}*/
  326. var dts = tsProcess.dts
  327. .pipe(concat(library.output))
  328. .pipe(replace(referenceSearchRegex, ""))
  329. .pipe(rename({ extname: ".d.ts" }))
  330. .pipe(gulp.dest(outputDirectory));
  331. var waitAll;
  332. if (library.buildAsModule) {
  333. var dts2 = tsProcess.dts
  334. .pipe(concat(library.output))
  335. .pipe(replace(referenceSearchRegex, ""))
  336. .pipe(addDtsExport(library.moduleDeclaration, library.moduleName, true, library.extendsRoot))
  337. .pipe(rename({ extname: ".module.d.ts" }))
  338. .pipe(gulp.dest(outputDirectory));
  339. waitAll = merge2([dev, code, css, dts, dts2]);
  340. } else {
  341. waitAll = merge2([dev, code, css, dts]);
  342. }
  343. if (library.webpack) {
  344. return waitAll.on('end', function () {
  345. webpack(require(library.webpack))
  346. .pipe(rename(library.output.replace(".js", ".bundle.js")))
  347. .pipe(gulp.dest(outputDirectory))
  348. });
  349. }
  350. else {
  351. return waitAll;
  352. }
  353. }
  354. }
  355. /**
  356. * The default task, concat and min the main BJS files.
  357. */
  358. gulp.task("default", ["typescript-all"], function () {
  359. });
  360. gulp.task("mainBuild", function (cb) {
  361. runSequence("buildWorker", "build", cb);
  362. });
  363. /**
  364. * Build the releasable files.
  365. */
  366. gulp.task("typescript", function (cb) {
  367. runSequence("typescript-compile", "mainBuild", cb);
  368. });
  369. /**
  370. * Dynamic module creation.
  371. */
  372. config.modules.map(function (module) {
  373. gulp.task(module, function () {
  374. return buildExternalLibraries(config[module]);
  375. });
  376. });
  377. gulp.task("typescript-libraries", config.modules, function () {
  378. });
  379. /**
  380. * Dynamic custom configurations.
  381. */
  382. config.buildConfigurations.distributed.map(function (customConfiguration) {
  383. gulp.task(customConfiguration, function (cb) {
  384. config.build.currentConfig = customConfiguration;
  385. config.build.outputDirectory = config.build.outputCustomConfigurationsDirectory + "/" + customConfiguration;
  386. runSequence("typescript-compile", "build", cb);
  387. });
  388. });
  389. gulp.task("typescript-customConfigurations", function (cb) {
  390. runSequence(config.buildConfigurations.distributed, cb);
  391. });
  392. /**
  393. * Custom build with full path file control; used by profile.html
  394. */
  395. gulp.task("build-custom", function (cb) {
  396. runSequence("typescript-compile", "build", cb);
  397. });
  398. /**
  399. * Do it all.
  400. */
  401. gulp.task("typescript-all", function (cb) {
  402. runSequence("typescript", "typescript-libraries", "typescript-customConfigurations", cb);
  403. });
  404. /**
  405. * Watch ts files and fire repective tasks.
  406. */
  407. gulp.task('watch', [], function () {
  408. var tasks = [gulp.watch(config.typescript, ['typescript-compile'])];
  409. config.modules.map(function (module) {
  410. config[module].libraries.map(function (library) {
  411. tasks.push(gulp.watch(library.files, function () {
  412. console.log(library.output);
  413. return buildExternalLibrary(library, config[module], true)
  414. .pipe(debug());
  415. }));
  416. tasks.push(gulp.watch(library.shaderFiles, function () {
  417. console.log(library.output);
  418. return buildExternalLibrary(library, config[module], true)
  419. .pipe(debug())
  420. }));
  421. tasks.push(gulp.watch(library.sassFiles, function () {
  422. console.log(library.output);
  423. return buildExternalLibrary(library, config[module], true)
  424. .pipe(debug())
  425. }));
  426. });
  427. });
  428. return tasks;
  429. });
  430. /**
  431. * Embedded local dev env management.
  432. */
  433. gulp.task('deployLocalDev', function () {
  434. gulp.src('../../localDev/template/**.*')
  435. .pipe(gulp.dest('../../localDev/src/'));
  436. });
  437. /**
  438. * Embedded webserver for test convenience.
  439. */
  440. gulp.task('webserver', function () {
  441. gulp.src('../../.').pipe(webserver({
  442. host: '0.0.0.0',
  443. port: 1338,
  444. livereload: false
  445. }));
  446. });
  447. /**
  448. * Combine Webserver and Watch as long as vscode does not handle multi tasks.
  449. */
  450. gulp.task('run', ['watch', 'webserver'], function () {
  451. });
  452. gulp.task("zip-blender", function () {
  453. return gulp.src('../../Exporters/Blender/src/**')
  454. .pipe(zip('Blender2Babylon-5.4.zip'))
  455. .pipe(gulp.dest('../../Exporters/Blender'));
  456. });
  457. gulp.task('clean-JS-MAP', function () {
  458. return del([
  459. '../../src/**/*.js.map', '../../src/**/*.js'
  460. ], { force: true });
  461. });