gulpfile.js 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267
  1. const pkg = require('./package.json')
  2. const path = require('path')
  3. const glob = require('glob')
  4. const yargs = require('yargs')
  5. const colors = require('colors')
  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')
  10. const commonjs = require('@rollup/plugin-commonjs')
  11. const resolve = require('@rollup/plugin-node-resolve')
  12. const gulp = require('gulp')
  13. const tap = require('gulp-tap')
  14. const zip = require('gulp-zip')
  15. const sass = require('gulp-sass')
  16. const header = require('gulp-header')
  17. const eslint = require('gulp-eslint')
  18. const minify = require('gulp-clean-css')
  19. const connect = require('gulp-connect')
  20. const autoprefixer = require('gulp-autoprefixer')
  21. const root = yargs.argv.root || '.'
  22. const port = yargs.argv.port || 8000
  23. const banner = `/*!
  24. * reveal.js ${pkg.version} (${new Date().toDateString()})
  25. * ${pkg.homepage}
  26. * MIT licensed
  27. *
  28. * Copyright (C) 2020 Hakim El Hattab, https://hakim.se
  29. */\n`
  30. // Prevents warnings from opening too many test pages
  31. process.setMaxListeners(20);
  32. const babelConfig = {
  33. exclude: 'node_modules/**',
  34. compact: false,
  35. extensions: ['.js', '.html'],
  36. plugins: ['transform-html-import-to-string'],
  37. presets: [[
  38. '@babel/preset-env',
  39. {
  40. corejs: 3,
  41. useBuiltIns: 'entry',
  42. modules: false
  43. }
  44. ]]
  45. };
  46. const rollupConfig = {
  47. plugins: [
  48. babel( babelConfig ),
  49. resolve(),
  50. commonjs(),
  51. terser()
  52. ]
  53. };
  54. // Our ES module bundle only needs to support modern
  55. // browser features
  56. const babelConfigESM = JSON.parse( JSON.stringify( babelConfig ) );
  57. babelConfigESM.presets[0][1].targets = { esmodules: true };
  58. const rollupConfigESM = {
  59. plugins: [
  60. babel( babelConfigESM ),
  61. resolve(),
  62. commonjs(),
  63. terser()
  64. ]
  65. };
  66. // Creates a bundle with broad browser support, exposed
  67. // as UMD
  68. gulp.task('js-es5', () => {
  69. return rollup({
  70. input: 'js/index.js',
  71. ...rollupConfig
  72. }).then( bundle => {
  73. return bundle.write({
  74. name: 'Reveal',
  75. file: './dist/reveal.js',
  76. format: 'umd',
  77. banner: banner,
  78. sourcemap: true
  79. });
  80. });
  81. })
  82. // Creates an ES module bundle
  83. gulp.task('js-es6', () => {
  84. return rollup({
  85. input: 'js/index.js',
  86. ...rollupConfigESM
  87. }).then( bundle => {
  88. return bundle.write({
  89. file: './dist/reveal.esm.js',
  90. format: 'es',
  91. banner: banner,
  92. sourcemap: true
  93. });
  94. });
  95. })
  96. gulp.task('js', gulp.parallel('js-es5', 'js-es6'));
  97. // Creates a UMD and ES module bundle for each of our
  98. // built-in plugins
  99. gulp.task('plugins', () => {
  100. return Promise.all([
  101. { name: 'RevealHighlight', input: './plugin/highlight/plugin.js', output: './plugin/highlight/highlight' },
  102. { name: 'RevealMarkdown', input: './plugin/markdown/plugin.js', output: './plugin/markdown/markdown' },
  103. { name: 'RevealSearch', input: './plugin/search/plugin.js', output: './plugin/search/search' },
  104. { name: 'RevealNotes', input: './plugin/notes/plugin.js', output: './plugin/notes/notes' },
  105. { name: 'RevealZoom', input: './plugin/zoom/plugin.js', output: './plugin/zoom/zoom' },
  106. { name: 'RevealMath', input: './plugin/math/plugin.js', output: './plugin/math/math' },
  107. ].map( plugin => {
  108. return rollup({
  109. input: plugin.input,
  110. ...rollupConfig
  111. }).then( bundle => {
  112. bundle.write({
  113. file: plugin.output + '.esm.js',
  114. name: plugin.name,
  115. format: 'es'
  116. })
  117. bundle.write({
  118. file: plugin.output + '.js',
  119. name: plugin.name,
  120. format: 'umd'
  121. })
  122. });
  123. } ));
  124. })
  125. gulp.task('css-themes', () => gulp.src(['./css/theme/source/*.{sass,scss}'])
  126. .pipe(sass())
  127. .pipe(gulp.dest('./dist/theme')))
  128. gulp.task('css-core', () => gulp.src(['css/reveal.scss'])
  129. .pipe(sass())
  130. .pipe(autoprefixer())
  131. .pipe(minify({compatibility: 'ie9'}))
  132. .pipe(header(banner))
  133. .pipe(gulp.dest('./dist')))
  134. gulp.task('css', gulp.parallel('css-themes', 'css-core'))
  135. gulp.task('qunit', () => {
  136. let serverConfig = {
  137. root,
  138. port: 8009,
  139. host: '0.0.0.0',
  140. name: 'test-server'
  141. }
  142. let server = connect.server( serverConfig )
  143. let testFiles = glob.sync('test/*.html' )
  144. let totalTests = 0;
  145. let failingTests = 0;
  146. let tests = Promise.all( testFiles.map( filename => {
  147. return new Promise( ( resolve, reject ) => {
  148. qunit.runQunitPuppeteer({
  149. targetUrl: `http://${serverConfig.host}:${serverConfig.port}/${filename}`,
  150. timeout: 20000,
  151. redirectConsole: false,
  152. puppeteerArgs: ['--allow-file-access-from-files']
  153. })
  154. .then(result => {
  155. if( result.stats.failed > 0 ) {
  156. console.log(`${'!'} ${filename} [${result.stats.passed}/${result.stats.total}] in ${result.stats.runtime}ms`.red);
  157. // qunit.printResultSummary(result, console);
  158. qunit.printFailedTests(result, console);
  159. }
  160. else {
  161. console.log(`${'✔'} ${filename} [${result.stats.passed}/${result.stats.total}] in ${result.stats.runtime}ms`.green);
  162. }
  163. totalTests += result.stats.total;
  164. failingTests += result.stats.failed;
  165. resolve();
  166. })
  167. .catch(error => {
  168. console.error(error);
  169. reject();
  170. });
  171. } )
  172. } ) );
  173. return new Promise( ( resolve, reject ) => {
  174. tests.then( () => {
  175. if( failingTests > 0 ) {
  176. reject( new Error(`${failingTests}/${totalTests} tests failed`.red) );
  177. }
  178. else {
  179. console.log(`${'✔'} Passed ${totalTests} tests`.green.bold);
  180. resolve();
  181. }
  182. } )
  183. .catch( () => {
  184. reject();
  185. } )
  186. .finally( () => {
  187. server.close();
  188. } );
  189. } );
  190. } )
  191. gulp.task('eslint', () => gulp.src(['./js/**', 'gulpfile.js'])
  192. .pipe(eslint())
  193. .pipe(eslint.format()))
  194. gulp.task('test', gulp.series( 'eslint', 'qunit' ))
  195. gulp.task('default', gulp.series(gulp.parallel('js', 'css', 'plugins'), 'test'))
  196. gulp.task('build', gulp.parallel('js', 'css', 'plugins'))
  197. gulp.task('package', gulp.series('default', () =>
  198. gulp.src([
  199. './index.html',
  200. './dist/**',
  201. './lib/**',
  202. './images/**',
  203. './plugin/**',
  204. './**.md'
  205. ]).pipe(zip('reveal-js-presentation.zip')).pipe(gulp.dest('./'))
  206. ))
  207. gulp.task('serve', () => {
  208. connect.server({
  209. root: root,
  210. port: port,
  211. host: '0.0.0.0',
  212. livereload: true
  213. })
  214. gulp.watch(['js/**'], gulp.series('js', 'test'))
  215. gulp.watch(['plugin/**/plugin.js'], gulp.series('plugins'))
  216. gulp.watch(['test/*.html'], gulp.series('test'))
  217. gulp.watch([
  218. 'css/theme/source/*.{sass,scss}',
  219. 'css/theme/template/*.{sass,scss}',
  220. ], gulp.series('css-themes'))
  221. gulp.watch([
  222. 'css/*.scss',
  223. 'css/print/*.{sass,scss,css}'
  224. ], gulp.series('css-core'))
  225. })