| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430 | /*! * Bootstrap's Gruntfile * https://getbootstrap.com/ * Copyright 2013-2019 Twitter, Inc. * Licensed under MIT (https://github.com/twbs/bootstrap/blob/master/LICENSE) */module.exports = function (grunt) {  'use strict';  // Force use of Unix newlines  grunt.util.linefeed = '\n';  RegExp.quote = function (string) {    return string.replace(/[-\\^$*+?.()|[\]{}]/g, '\\$&');  };  var fs = require('fs');  var path = require('path');  var generateGlyphiconsData = require('./grunt/bs-glyphicons-data-generator.js');  var BsLessdocParser = require('./grunt/bs-lessdoc-parser.js');  var getLessVarsData = function () {    var filePath = path.join(__dirname, 'less/variables.less');    var fileContent = fs.readFileSync(filePath, { encoding: 'utf8' });    var parser = new BsLessdocParser(fileContent);    return { sections: parser.parseFile() };  };  var generateRawFiles = require('./grunt/bs-raw-files-generator.js');  var generateCommonJSModule = require('./grunt/bs-commonjs-generator.js');  var configBridge = grunt.file.readJSON('./grunt/configBridge.json', { encoding: 'utf8' });  Object.keys(configBridge.paths).forEach(function (key) {    configBridge.paths[key].forEach(function (val, i, arr) {      arr[i] = path.join('./docs/assets', val);    });  });  // Project configuration.  grunt.initConfig({    // Metadata.    pkg: grunt.file.readJSON('package.json'),    banner: '/*!\n' +            ' * Bootstrap v<%= pkg.version %> (<%= pkg.homepage %>)\n' +            ' * Copyright 2011-<%= grunt.template.today("yyyy") %> <%= pkg.author %>\n' +            ' * Licensed under the <%= pkg.license %> license\n' +            ' */\n',    jqueryCheck: configBridge.config.jqueryCheck.join('\n'),    jqueryVersionCheck: configBridge.config.jqueryVersionCheck.join('\n'),    // Task configuration.    clean: {      dist: 'dist',      docs: 'docs/dist'    },    jshint: {      options: {        jshintrc: 'js/.jshintrc'      },      grunt: {        options: {          jshintrc: 'grunt/.jshintrc'        },        src: ['Gruntfile.js', 'package.js', 'grunt/*.js']      },      core: {        src: 'js/*.js'      },      test: {        options: {          jshintrc: 'js/tests/unit/.jshintrc'        },        src: 'js/tests/unit/*.js'      },      assets: {        src: ['docs/assets/js/src/*.js', 'docs/assets/js/*.js', '!docs/assets/js/*.min.js']      }    },    jscs: {      options: {        config: 'js/.jscsrc'      },      grunt: {        src: '<%= jshint.grunt.src %>'      },      core: {        src: '<%= jshint.core.src %>'      },      test: {        src: '<%= jshint.test.src %>'      },      assets: {        options: {          requireCamelCaseOrUpperCaseIdentifiers: null        },        src: '<%= jshint.assets.src %>'      }    },    concat: {      options: {        banner: '<%= banner %>\n<%= jqueryCheck %>\n<%= jqueryVersionCheck %>',        stripBanners: false      },      core: {        src: [          'js/transition.js',          'js/alert.js',          'js/button.js',          'js/carousel.js',          'js/collapse.js',          'js/dropdown.js',          'js/modal.js',          'js/tooltip.js',          'js/popover.js',          'js/scrollspy.js',          'js/tab.js',          'js/affix.js'        ],        dest: 'dist/js/<%= pkg.name %>.js'      }    },    uglify: {      options: {        compress: true,        mangle: true,        ie8: true,        output: {          comments: /^!|@preserve|@license|@cc_on/i        }      },      core: {        src: '<%= concat.core.dest %>',        dest: 'dist/js/<%= pkg.name %>.min.js'      },      customize: {        src: configBridge.paths.customizerJs,        dest: 'docs/assets/js/customize.min.js'      },      docs: {        src: configBridge.paths.docsJs,        dest: 'docs/assets/js/docs.min.js'      }    },    less: {      options: {        ieCompat: true,        strictMath: true,        sourceMap: true,        outputSourceFiles: true      },      core: {        options: {          sourceMapURL: '<%= pkg.name %>.css.map',          sourceMapFilename: 'dist/css/<%= pkg.name %>.css.map'        },        src: 'less/bootstrap.less',        dest: 'dist/css/<%= pkg.name %>.css'      },      theme: {        options: {          sourceMapURL: '<%= pkg.name %>-theme.css.map',          sourceMapFilename: 'dist/css/<%= pkg.name %>-theme.css.map'        },        src: 'less/theme.less',        dest: 'dist/css/<%= pkg.name %>-theme.css'      },      docs: {        options: {          sourceMapURL: 'docs.css.map',          sourceMapFilename: 'docs/assets/css/docs.css.map'        },        src: 'docs/assets/less/docs.less',        dest: 'docs/assets/css/docs.css'      },      docsIe: {        options: {          sourceMap: false        },        src: 'docs/assets/less/ie10-viewport-bug-workaround.less',        dest: 'docs/assets/css/ie10-viewport-bug-workaround.css'      }    },    postcss: {      options: {        map: {          inline: false,          sourcesContent: true        },        processors: [          require('autoprefixer')(configBridge.config.autoprefixer)        ]      },      core: {        src: 'dist/css/<%= pkg.name %>.css'      },      theme: {        src: 'dist/css/<%= pkg.name %>-theme.css'      },      docs: {        src: 'docs/assets/css/docs.css'      },      examples: {        options: {          map: false        },        expand: true,        cwd: 'docs/examples/',        src: ['**/*.css'],        dest: 'docs/examples/'      }    },    stylelint: {      options: {        configFile: 'grunt/.stylelintrc',        reportNeedlessDisables: false      },      dist: [        'less/**/*.less'      ],      docs: [        'docs/assets/less/**/*.less'      ],      examples: [        'docs/examples/**/*.css'      ]    },    cssmin: {      options: {        compatibility: 'ie8',        sourceMap: true,        sourceMapInlineSources: true,        level: {          1: {            specialComments: 'all'          }        }      },      core: {        src: 'dist/css/<%= pkg.name %>.css',        dest: 'dist/css/<%= pkg.name %>.min.css'      },      theme: {        src: 'dist/css/<%= pkg.name %>-theme.css',        dest: 'dist/css/<%= pkg.name %>-theme.min.css'      },      docs: {        src: 'docs/assets/css/docs.css',        dest: 'docs/assets/css/docs.min.css'      }    },    copy: {      fonts: {        expand: true,        src: 'fonts/**',        dest: 'dist/'      },      docs: {        expand: true,        cwd: 'dist/',        src: [          '**/*'        ],        dest: 'docs/dist/'      }    },    connect: {      server: {        options: {          port: 3000,          base: '.'        }      }    },    jekyll: {      options: {        bundleExec: true,        config: '_config.yml',        incremental: false      },      docs: {},      github: {        options: {          raw: 'github: true'        }      }    },    pug: {      options: {        pretty: true,        data: getLessVarsData      },      customizerVars: {        src: 'docs/_pug/customizer-variables.pug',        dest: 'docs/_includes/customizer-variables.html'      },      customizerNav: {        src: 'docs/_pug/customizer-nav.pug',        dest: 'docs/_includes/nav/customize.html'      }    },    htmllint: {      options: {        ignore: [          'Element "img" is missing required attribute "src".'        ],        noLangDetect: true      },      src: ['_gh_pages/**/*.html', 'js/tests/**/*.html']    },    watch: {      src: {        files: '<%= jshint.core.src %>',        tasks: ['jshint:core', 'exec:karma', 'concat']      },      test: {        files: '<%= jshint.test.src %>',        tasks: ['jshint:test', 'exec:karma']      },      less: {        files: 'less/**/*.less',        tasks: ['less', 'copy']      },      docs: {        files: 'docs/assets/less/**/*.less',        tasks: ['less']      }    },    exec: {      browserstack: {        command: 'cross-env BROWSER=true karma start grunt/karma.conf.js'      },      karma: {        command: 'karma start grunt/karma.conf.js'      }    }  });  // These plugins provide necessary tasks.  require('load-grunt-tasks')(grunt, { scope: 'devDependencies' });  require('time-grunt')(grunt);  // Docs HTML validation task  grunt.registerTask('validate-html', ['jekyll:docs', 'htmllint']);  var runSubset = function (subset) {    return !process.env.TWBS_TEST || process.env.TWBS_TEST === subset;  };  var isUndefOrNonZero = function (val) {    return typeof val === 'undefined' || val !== '0';  };  // Test task.  var testSubtasks = [];  // Skip core tests if running a different subset of the test suite  if (runSubset('core')) {    testSubtasks = testSubtasks.concat(['dist-css', 'dist-js', 'stylelint:dist', 'test-js', 'docs']);  }  // Skip HTML validation if running a different subset of the test suite  if (runSubset('validate-html') &&      // Skip HTML5 validator on Travis when [skip validator] is in the commit message      isUndefOrNonZero(process.env.TWBS_DO_VALIDATOR)) {    testSubtasks.push('validate-html');  }  // Only run BrowserStack tests if there's a BrowserStack access key  if (typeof process.env.BROWSER_STACK_USERNAME !== 'undefined' &&      // Skip BrowserStack if running a different subset of the test suite      runSubset('browserstack') &&      // Skip BrowserStack on Travis when [skip browserstack] is in the commit message      isUndefOrNonZero(process.env.TWBS_DO_BROWSERSTACK)) {    testSubtasks.push('exec:browserstack');  }  grunt.registerTask('test', testSubtasks);  grunt.registerTask('test-js', ['jshint:core', 'jshint:test', 'jshint:grunt', 'jscs:core', 'jscs:test', 'jscs:grunt', 'exec:karma']);  // JS distribution task.  grunt.registerTask('dist-js', ['concat', 'uglify:core', 'commonjs']);  // CSS distribution task.  grunt.registerTask('dist-css', ['less:core', 'less:theme', 'postcss:core', 'postcss:theme', 'cssmin:core', 'cssmin:theme']);  // Full distribution task.  grunt.registerTask('dist', ['clean:dist', 'dist-css', 'copy:fonts', 'dist-js']);  // Default task.  grunt.registerTask('default', ['clean:dist', 'copy:fonts', 'test']);  grunt.registerTask('build-glyphicons-data', function () {    generateGlyphiconsData.call(this, grunt);  });  // task for building customizer  grunt.registerTask('build-customizer', ['build-customizer-html', 'build-raw-files']);  grunt.registerTask('build-customizer-html', 'pug');  grunt.registerTask('build-raw-files', 'Add scripts/less files to customizer.', function () {    var banner = grunt.template.process('<%= banner %>');    generateRawFiles(grunt, banner);  });  grunt.registerTask('commonjs', 'Generate CommonJS entrypoint module in dist dir.', function () {    var srcFiles = grunt.config.get('concat.core.src');    var destFilepath = 'dist/js/npm.js';    generateCommonJSModule(grunt, srcFiles, destFilepath);  });  // Docs task.  grunt.registerTask('docs-css', ['less:docs', 'less:docsIe', 'postcss:docs', 'postcss:examples', 'cssmin:docs']);  grunt.registerTask('lint-docs-css', ['stylelint:docs', 'stylelint:examples']);  grunt.registerTask('docs-js', ['uglify:docs', 'uglify:customize']);  grunt.registerTask('lint-docs-js', ['jshint:assets', 'jscs:assets']);  grunt.registerTask('docs', ['docs-css', 'lint-docs-css', 'docs-js', 'lint-docs-js', 'clean:docs', 'copy:docs', 'build-glyphicons-data', 'build-customizer']);  grunt.registerTask('prep-release', ['dist', 'docs', 'jekyll:github']);};
 |