gulpfile.js 8.6 KB

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