gulpfile.js 9.3 KB

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