gulpfile.js 9.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331
  1. const fs = require('fs');
  2. const pkg = require('./package.json')
  3. const glob = require('glob')
  4. const yargs = require('yargs')
  5. const through = require('through2');
  6. const qunit = require('node-qunit-puppeteer')
  7. const {rollup} = require('rollup')
  8. const terser = require('@rollup/plugin-terser')
  9. const babel = require('@rollup/plugin-babel').default
  10. const commonjs = require('@rollup/plugin-commonjs')
  11. const resolve = require('@rollup/plugin-node-resolve').default
  12. const sass = require('sass')
  13. const gulp = require('gulp')
  14. const zip = require('gulp-zip')
  15. const header = require('gulp-header-comment')
  16. const eslint = require('gulp-eslint')
  17. const minify = require('gulp-clean-css')
  18. const connect = require('gulp-connect')
  19. const autoprefixer = require('gulp-autoprefixer')
  20. const root = yargs.argv.root || '.'
  21. const port = yargs.argv.port || 8000
  22. const host = yargs.argv.host || 'localhost'
  23. const cssLicense = `
  24. reveal.js ${pkg.version}
  25. ${pkg.homepage}
  26. MIT licensed
  27. Copyright (C) 2011-2024 Hakim El Hattab, https://hakim.se
  28. `;
  29. const jsLicense = `/*!
  30. * reveal.js ${pkg.version}
  31. * ${pkg.homepage}
  32. * MIT licensed
  33. *
  34. * Copyright (C) 2011-2024 Hakim El Hattab, https://hakim.se
  35. */\n`;
  36. // Prevents warnings from opening too many test pages
  37. process.setMaxListeners(20);
  38. const babelConfig = {
  39. babelHelpers: 'bundled',
  40. ignore: ['node_modules'],
  41. compact: false,
  42. extensions: ['.js', '.html'],
  43. plugins: [
  44. 'transform-html-import-to-string'
  45. ],
  46. presets: [[
  47. '@babel/preset-env',
  48. {
  49. corejs: 3,
  50. useBuiltIns: 'usage',
  51. modules: false
  52. }
  53. ]]
  54. };
  55. // Our ES module bundle only targets newer browsers with
  56. // module support. Browsers are targeted explicitly instead
  57. // of using the "esmodule: true" target since that leads to
  58. // polyfilling older browsers and a larger bundle.
  59. const babelConfigESM = JSON.parse( JSON.stringify( babelConfig ) );
  60. babelConfigESM.presets[0][1].targets = { browsers: [
  61. 'last 2 Chrome versions',
  62. 'last 2 Safari versions',
  63. 'last 2 iOS versions',
  64. 'last 2 Firefox versions',
  65. 'last 2 Edge versions',
  66. ] };
  67. let cache = {};
  68. // Creates a bundle with broad browser support, exposed
  69. // as UMD
  70. gulp.task('js-es5', () => {
  71. return rollup({
  72. cache: cache.umd,
  73. input: 'js/index.js',
  74. plugins: [
  75. resolve(),
  76. commonjs(),
  77. babel( babelConfig ),
  78. terser()
  79. ]
  80. }).then( bundle => {
  81. cache.umd = bundle.cache;
  82. return bundle.write({
  83. name: 'Reveal',
  84. file: './dist/reveal.js',
  85. format: 'umd',
  86. banner: jsLicense,
  87. sourcemap: true
  88. });
  89. });
  90. })
  91. // Creates an ES module bundle
  92. gulp.task('js-es6', () => {
  93. return rollup({
  94. cache: cache.esm,
  95. input: 'js/index.js',
  96. plugins: [
  97. resolve(),
  98. commonjs(),
  99. babel( babelConfigESM ),
  100. terser()
  101. ]
  102. }).then( bundle => {
  103. cache.esm = bundle.cache;
  104. return bundle.write({
  105. file: './dist/reveal.esm.js',
  106. format: 'es',
  107. banner: jsLicense,
  108. sourcemap: true
  109. });
  110. });
  111. })
  112. gulp.task('js', gulp.parallel('js-es5', 'js-es6'));
  113. // Creates a UMD and ES module bundle for each of our
  114. // built-in plugins
  115. gulp.task('plugins', () => {
  116. return Promise.all([
  117. { name: 'RevealHighlight', input: './plugin/highlight/plugin.js', output: './plugin/highlight/highlight' },
  118. { name: 'RevealMarkdown', input: './plugin/markdown/plugin.js', output: './plugin/markdown/markdown' },
  119. { name: 'RevealSearch', input: './plugin/search/plugin.js', output: './plugin/search/search' },
  120. { name: 'RevealNotes', input: './plugin/notes/plugin.js', output: './plugin/notes/notes' },
  121. { name: 'RevealZoom', input: './plugin/zoom/plugin.js', output: './plugin/zoom/zoom' },
  122. { name: 'RevealMath', input: './plugin/math/plugin.js', output: './plugin/math/math' },
  123. ].map( plugin => {
  124. return rollup({
  125. cache: cache[plugin.input],
  126. input: plugin.input,
  127. plugins: [
  128. resolve(),
  129. commonjs(),
  130. babel({
  131. ...babelConfig,
  132. ignore: [/node_modules\/(?!(highlight\.js|marked)\/).*/],
  133. }),
  134. terser()
  135. ]
  136. }).then( bundle => {
  137. cache[plugin.input] = bundle.cache;
  138. bundle.write({
  139. file: plugin.output + '.esm.js',
  140. name: plugin.name,
  141. format: 'es'
  142. })
  143. bundle.write({
  144. file: plugin.output + '.js',
  145. name: plugin.name,
  146. format: 'umd'
  147. })
  148. });
  149. } ));
  150. })
  151. // a custom pipeable step to transform Sass to CSS
  152. function compileSass() {
  153. return through.obj( ( vinylFile, encoding, callback ) => {
  154. const transformedFile = vinylFile.clone();
  155. sass.render({
  156. silenceDeprecations: ['legacy-js-api'],
  157. data: transformedFile.contents.toString(),
  158. file: transformedFile.path,
  159. }, ( err, result ) => {
  160. if( err ) {
  161. callback(err);
  162. }
  163. else {
  164. transformedFile.extname = '.css';
  165. transformedFile.contents = result.css;
  166. callback( null, transformedFile );
  167. }
  168. });
  169. });
  170. }
  171. gulp.task('css-themes', () => gulp.src(['./css/theme/source/*.{sass,scss}'])
  172. .pipe(compileSass())
  173. .pipe(gulp.dest('./dist/theme')))
  174. gulp.task('css-core', () => gulp.src(['css/reveal.scss'])
  175. .pipe(compileSass())
  176. .pipe(autoprefixer())
  177. .pipe(minify({compatibility: 'ie9'}))
  178. .pipe(header(cssLicense))
  179. .pipe(gulp.dest('./dist')))
  180. gulp.task('css', gulp.parallel('css-themes', 'css-core'))
  181. gulp.task('qunit', () => {
  182. let serverConfig = {
  183. root,
  184. port: 8009,
  185. host: 'localhost',
  186. name: 'test-server'
  187. }
  188. let server = connect.server( serverConfig )
  189. let testFiles = glob.sync('test/*.html' )
  190. let totalTests = 0;
  191. let failingTests = 0;
  192. let tests = Promise.all( testFiles.map( filename => {
  193. return new Promise( ( resolve, reject ) => {
  194. qunit.runQunitPuppeteer({
  195. targetUrl: `http://${serverConfig.host}:${serverConfig.port}/${filename}`,
  196. timeout: 20000,
  197. redirectConsole: false,
  198. puppeteerArgs: ['--allow-file-access-from-files', '--no-sandbox']
  199. })
  200. .then(result => {
  201. if( result.stats.failed > 0 ) {
  202. console.log(`${'!'} ${filename} [${result.stats.passed}/${result.stats.total}] in ${result.stats.runtime}ms`.red);
  203. // qunit.printResultSummary(result, console);
  204. qunit.printFailedTests(result, console);
  205. }
  206. else {
  207. console.log(`${'✔'} ${filename} [${result.stats.passed}/${result.stats.total}] in ${result.stats.runtime}ms`.green);
  208. }
  209. totalTests += result.stats.total;
  210. failingTests += result.stats.failed;
  211. resolve();
  212. })
  213. .catch(error => {
  214. console.error(error);
  215. reject();
  216. });
  217. } )
  218. } ) );
  219. return new Promise( ( resolve, reject ) => {
  220. tests.then( () => {
  221. if( failingTests > 0 ) {
  222. reject( new Error(`${failingTests}/${totalTests} tests failed`.red) );
  223. }
  224. else {
  225. console.log(`${'✔'} Passed ${totalTests} tests`.green.bold);
  226. resolve();
  227. }
  228. } )
  229. .catch( () => {
  230. reject();
  231. } )
  232. .finally( () => {
  233. server.close();
  234. } );
  235. } );
  236. } )
  237. gulp.task('eslint', () => gulp.src(['./js/**', 'gulpfile.js'])
  238. .pipe(eslint())
  239. .pipe(eslint.format()))
  240. gulp.task('test', gulp.series( 'eslint', 'qunit' ))
  241. gulp.task('default', gulp.series(gulp.parallel('js', 'css', 'plugins'), 'test'))
  242. gulp.task('build', gulp.parallel('js', 'css', 'plugins'))
  243. gulp.task('package', gulp.series(async () => {
  244. let dirs = [
  245. './index.html',
  246. './dist/**',
  247. './plugin/**',
  248. './*/*.md'
  249. ];
  250. if (fs.existsSync('./lib')) dirs.push('./lib/**');
  251. if (fs.existsSync('./images')) dirs.push('./images/**');
  252. if (fs.existsSync('./slides')) dirs.push('./slides/**');
  253. return gulp.src( dirs, { base: './', encoding: false } )
  254. .pipe(zip('reveal-js-presentation.zip')).pipe(gulp.dest('./'))
  255. }))
  256. gulp.task('reload', () => gulp.src(['index.html'])
  257. .pipe(connect.reload()));
  258. gulp.task('serve', () => {
  259. connect.server({
  260. root: root,
  261. port: port,
  262. host: host,
  263. livereload: true
  264. })
  265. const slidesRoot = root.endsWith('/') ? root : root + '/'
  266. gulp.watch([
  267. slidesRoot + '**/*.html',
  268. slidesRoot + '**/*.md',
  269. `!${slidesRoot}**/node_modules/**`, // ignore node_modules
  270. ], gulp.series('reload'))
  271. gulp.watch(['js/**'], gulp.series('js', 'reload', 'eslint'))
  272. gulp.watch(['plugin/**/plugin.js', 'plugin/**/*.html'], gulp.series('plugins', 'reload'))
  273. gulp.watch([
  274. 'css/theme/source/**/*.{sass,scss}',
  275. 'css/theme/template/*.{sass,scss}',
  276. ], gulp.series('css-themes', 'reload'))
  277. gulp.watch([
  278. 'css/*.scss',
  279. 'css/print/*.{sass,scss,css}'
  280. ], gulp.series('css-core', 'reload'))
  281. gulp.watch(['test/*.html'], gulp.series('test'))
  282. })