mirror of
				https://github.com/twbs/bootstrap.git
				synced 2025-10-31 00:04:27 -04:00 
			
		
		
		
	
		
			
				
	
	
		
			515 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			515 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| /*!
 | |
|  * Bootstrap's Gruntfile
 | |
|  * http://getbootstrap.com
 | |
|  * Copyright 2013-2014 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 glob = require('glob');
 | |
|   var npmShrinkwrap = require('npm-shrinkwrap');
 | |
|   var mq4HoverShim = require('mq4-hover-shim');
 | |
| 
 | |
|   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 <%= pkg.license.type %> (<%= pkg.license.url %>)\n' +
 | |
|             ' */\n',
 | |
|     jqueryCheck: 'if (typeof jQuery === \'undefined\') {\n' +
 | |
|                  '  throw new Error(\'Bootstrap\\\'s JavaScript requires jQuery\')\n' +
 | |
|                  '}\n',
 | |
|     jqueryVersionCheck: '+function ($) {\n' +
 | |
|                         '  var version = $.fn.jquery.split(\' \')[0].split(\'.\')\n' +
 | |
|                         '  if ((version[0] < 2 && version[1] < 9) || (version[0] == 1 && version[1] == 9 && version[2] < 1)) {\n' +
 | |
|                         '    throw new Error(\'Bootstrap\\\'s JavaScript requires jQuery version 1.9.1 or higher\')\n' +
 | |
|                         '  }\n' +
 | |
|                         '}(jQuery);\n\n',
 | |
| 
 | |
|     // Task configuration.
 | |
|     clean: {
 | |
|       dist: 'dist',
 | |
|       docs: 'docs/dist'
 | |
|     },
 | |
| 
 | |
|     // JS build configuration
 | |
| 
 | |
|     lineremover: {
 | |
|       es6Import: {
 | |
|         files: {
 | |
|           '<%= concat.bootstrap.dest %>': '<%= concat.bootstrap.dest %>'
 | |
|         },
 | |
|         options: {
 | |
|           exclusionPattern: /^(import|export)/g
 | |
|         }
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     babel: {
 | |
|       dev: {
 | |
|         options: {
 | |
|           sourceMap: true,
 | |
|           modules: 'ignore'
 | |
|         },
 | |
|         files: {
 | |
|           'js/dist/util.js'      : 'js/src/util.js',
 | |
|           'js/dist/alert.js'     : 'js/src/alert.js',
 | |
|           'js/dist/button.js'    : 'js/src/button.js',
 | |
|           'js/dist/carousel.js'  : 'js/src/carousel.js',
 | |
|           'js/dist/collapse.js'  : 'js/src/collapse.js',
 | |
|           'js/dist/dropdown.js'  : 'js/src/dropdown.js',
 | |
|           'js/dist/modal.js'     : 'js/src/modal.js',
 | |
|           'js/dist/scrollspy.js' : 'js/src/scrollspy.js',
 | |
|           'js/dist/tab.js'       : 'js/src/tab.js',
 | |
|           'js/dist/tooltip.js'   : 'js/src/tooltip.js',
 | |
|           'js/dist/popover.js'   : 'js/src/popover.js'
 | |
|         }
 | |
|       },
 | |
|       dist: {
 | |
|         options: {
 | |
|           modules: 'ignore'
 | |
|         },
 | |
|         files: {
 | |
|           '<%= concat.bootstrap.dest %>' : '<%= concat.bootstrap.dest %>'
 | |
|         }
 | |
|       },
 | |
|       umd: {
 | |
|         options: {
 | |
|           modules: 'umd'
 | |
|         },
 | |
|         files: {
 | |
|           'dist/js/umd/util.js'      : 'js/src/util.js',
 | |
|           'dist/js/umd/alert.js'     : 'js/src/alert.js',
 | |
|           'dist/js/umd/button.js'    : 'js/src/button.js',
 | |
|           'dist/js/umd/carousel.js'  : 'js/src/carousel.js',
 | |
|           'dist/js/umd/collapse.js'  : 'js/src/collapse.js',
 | |
|           'dist/js/umd/dropdown.js'  : 'js/src/dropdown.js',
 | |
|           'dist/js/umd/modal.js'     : 'js/src/modal.js',
 | |
|           'dist/js/umd/scrollspy.js' : 'js/src/scrollspy.js',
 | |
|           'dist/js/umd/tab.js'       : 'js/src/tab.js',
 | |
|           'dist/js/umd/tooltip.js'   : 'js/src/tooltip.js',
 | |
|           'dist/js/umd/popover.js'   : 'js/src/popover.js'
 | |
|         }
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     jscs: {
 | |
|       options: {
 | |
|         config: 'js/.jscsrc'
 | |
|       },
 | |
|       grunt: {
 | |
|         src: ['Gruntfile.js', 'grunt/*.js']
 | |
|       },
 | |
|       core: {
 | |
|         src: 'js/src/*.js'
 | |
|       },
 | |
|       test: {
 | |
|         src: 'js/tests/unit/*.js'
 | |
|       },
 | |
|       assets: {
 | |
|         options: {
 | |
|           requireCamelCaseOrUpperCaseIdentifiers: null
 | |
|         },
 | |
|         src: ['docs/assets/js/src/*.js', 'docs/assets/js/*.js', '!docs/assets/js/*.min.js']
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     stamp: {
 | |
|       options: {
 | |
|         banner: '<%= banner %>\n<%= jqueryCheck %>\n<%= jqueryVersionCheck %>\n+function ($) {\n',
 | |
|         footer: '\n}(jQuery);'
 | |
|       },
 | |
|       bootstrap: {
 | |
|         files: {
 | |
|           src: '<%= concat.bootstrap.dest %>'
 | |
|         }
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     concat: {
 | |
|       options: {
 | |
|         stripBanners: false
 | |
|       },
 | |
|       bootstrap: {
 | |
|         src: [
 | |
|           'js/src/util.js',
 | |
|           'js/src/alert.js',
 | |
|           'js/src/button.js',
 | |
|           'js/src/carousel.js',
 | |
|           'js/src/collapse.js',
 | |
|           'js/src/dropdown.js',
 | |
|           'js/src/modal.js',
 | |
|           'js/src/scrollspy.js',
 | |
|           'js/src/tab.js',
 | |
|           'js/src/tooltip.js',
 | |
|           'js/src/popover.js'
 | |
|         ],
 | |
|         dest: 'dist/js/<%= pkg.name %>.js'
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     uglify: {
 | |
|       options: {
 | |
|         compress: {
 | |
|           warnings: false
 | |
|         },
 | |
|         mangle: true,
 | |
|         preserveComments: 'some'
 | |
|       },
 | |
|       core: {
 | |
|         src: '<%= concat.bootstrap.dest %>',
 | |
|         dest: 'dist/js/<%= pkg.name %>.min.js'
 | |
|       },
 | |
|       customize: {
 | |
|         src: configBridge.paths.customizerJs,
 | |
|         dest: 'docs/assets/js/customize.min.js'
 | |
|       },
 | |
|       docsJs: {
 | |
|         src: configBridge.paths.docsJs,
 | |
|         dest: 'docs/assets/js/docs.min.js'
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     qunit: {
 | |
|       options: {
 | |
|         inject: 'js/tests/unit/phantom.js'
 | |
|       },
 | |
|       files: 'js/tests/index.html'
 | |
|     },
 | |
| 
 | |
| 
 | |
|     // CSS build configuration
 | |
| 
 | |
|     scsslint: {
 | |
|       scss: ['scss/*.scss', '!scss/_normalize.scss'],
 | |
|       options: {
 | |
|         config: 'scss/.scss-lint.yml',
 | |
|         reporterOutput: 'scss-lint-report.xml'
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     postcss: {
 | |
|       options: {
 | |
|         map: true,
 | |
|         processors: [mq4HoverShim.postprocessorFor({ hoverSelectorPrefix: '.bs-true-hover ' })]
 | |
|       },
 | |
|       core: {
 | |
|         src: 'dist/css/<%= pkg.name %>.css'
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     autoprefixer: {
 | |
|       options: {
 | |
|         browsers: [
 | |
|           'Android 2.3',
 | |
|           'Android >= 4',
 | |
|           'Chrome >= 35',
 | |
|           'Firefox >= 31',
 | |
|           'Explorer >= 9',
 | |
|           'iOS >= 7',
 | |
|           'Opera >= 12',
 | |
|           'Safari >= 7.1'
 | |
|         ]
 | |
|       },
 | |
|       core: {
 | |
|         options: {
 | |
|           map: true
 | |
|         },
 | |
|         src: 'dist/css/<%= pkg.name %>.css'
 | |
|       },
 | |
|       docs: {
 | |
|         src: 'docs/assets/css/docs.min.css'
 | |
|       },
 | |
|       examples: {
 | |
|         expand: true,
 | |
|         cwd: 'docs/examples/',
 | |
|         src: ['**/*.css'],
 | |
|         dest: 'docs/examples/'
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     cssmin: {
 | |
|       options: {
 | |
|         // TODO: disable `zeroUnits` optimization once clean-css 3.2 is released
 | |
|         //    and then simplify the fix for https://github.com/twbs/bootstrap/issues/14837 accordingly
 | |
|         compatibility: 'ie8',
 | |
|         keepSpecialComments: '*',
 | |
|         noAdvanced: true
 | |
|       },
 | |
|       core: {
 | |
|         files: {
 | |
|           'dist/css/<%= pkg.name %>.min.css': 'dist/css/<%= pkg.name %>.css'
 | |
|         }
 | |
|       },
 | |
|       docs: {
 | |
|         src: 'docs/assets/css/docs.min.css',
 | |
|         dest: 'docs/assets/css/docs.min.css'
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     usebanner: {
 | |
|       options: {
 | |
|         position: 'top',
 | |
|         banner: '<%= banner %>'
 | |
|       },
 | |
|       files: {
 | |
|         src: 'dist/css/*.css'
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     csscomb: {
 | |
|       options: {
 | |
|         config: 'scss/.csscomb.json'
 | |
|       },
 | |
|       dist: {
 | |
|         expand: true,
 | |
|         cwd: 'dist/css/',
 | |
|         src: ['*.css', '!*.min.css'],
 | |
|         dest: 'dist/css/'
 | |
|       },
 | |
|       examples: {
 | |
|         expand: true,
 | |
|         cwd: 'docs/examples/',
 | |
|         src: '**/*.css',
 | |
|         dest: 'docs/examples/'
 | |
|       },
 | |
|       docs: {
 | |
|         src: 'docs/assets/css/src/docs.css',
 | |
|         dest: 'docs/assets/css/src/docs.css'
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     copy: {
 | |
|       docs: {
 | |
|         expand: true,
 | |
|         cwd: 'dist/',
 | |
|         src: [
 | |
|           '**/*'
 | |
|         ],
 | |
|         dest: 'docs/dist/'
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     connect: {
 | |
|       server: {
 | |
|         options: {
 | |
|           port: 3000,
 | |
|           base: '.'
 | |
|         }
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     jekyll: {
 | |
|       options: {
 | |
|         config: '_config.yml'
 | |
|       },
 | |
|       docs: {},
 | |
|       github: {
 | |
|         options: {
 | |
|           raw: 'github: true'
 | |
|         }
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     htmllint: {
 | |
|       options: {
 | |
|         ignore: [
 | |
|           'Element “img” is missing required attribute “src”.',
 | |
|           'Bad value “X-UA-Compatible” for attribute “http-equiv” on element “meta”.',
 | |
|           'Attribute “autocomplete” not allowed on element “input” at this point.',
 | |
|           'Attribute “autocomplete” not allowed on element “button” at this point.',
 | |
|           'Element “div” not allowed as child of element “progress” in this context. (Suppressing further errors from this subtree.)',
 | |
|           'Consider using the “h1” element as a top-level heading only (all “h1” elements are treated as top-level headings by many screen readers and other tools).'
 | |
|         ]
 | |
|       },
 | |
|       src: '_gh_pages/**/*.html'
 | |
|     },
 | |
| 
 | |
|     watch: {
 | |
|       src: {
 | |
|         files: '<%= jscs.core.src %>',
 | |
|         tasks: ['babel:dev']
 | |
|       },
 | |
|       test: {
 | |
|         files: '<%= jscs.test.src %>',
 | |
|         tasks: ['qunit']
 | |
|       },
 | |
|       sass: {
 | |
|         files: 'scss/**/*.scss',
 | |
|         tasks: 'sass-compile'
 | |
|       },
 | |
|       docs: {
 | |
|         files: 'docs/assets/scss/*.scss',
 | |
|         tasks: 'sass:docs'
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     sed: {
 | |
|       versionNumber: {
 | |
|         pattern: (function () {
 | |
|           var old = grunt.option('oldver');
 | |
|           return old ? RegExp.quote(old) : old;
 | |
|         })(),
 | |
|         replacement: grunt.option('newver'),
 | |
|         recursive: true
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     'saucelabs-qunit': {
 | |
|       all: {
 | |
|         options: {
 | |
|           build: process.env.TRAVIS_JOB_ID,
 | |
|           concurrency: 10,
 | |
|           maxRetries: 3,
 | |
|           maxPollRetries: 4,
 | |
|           urls: ['http://127.0.0.1:3000/js/tests/index.html?hidepassed'],
 | |
|           browsers: grunt.file.readYAML('grunt/sauce_browsers.yml')
 | |
|         }
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     exec: {
 | |
|       npmUpdate: {
 | |
|         command: 'npm update'
 | |
|       },
 | |
|       bundleUpdate: {
 | |
|         command: function () {
 | |
|           // Update dev gems and all the test gemsets
 | |
|           return 'bundle update && ' + glob.sync('test-infra/gemfiles/*.gemfile').map(function (gemfile) {
 | |
|             return 'BUNDLE_GEMFILE=' + gemfile + ' bundle update';
 | |
|           }).join(' && ');
 | |
|         }
 | |
|       }
 | |
|     }
 | |
|   });
 | |
| 
 | |
| 
 | |
|   // These plugins provide necessary tasks.
 | |
|   require('load-grunt-tasks')(grunt, { scope: 'devDependencies',
 | |
|     // Exclude Sass compilers. We choose the one to load later on.
 | |
|     pattern: ['grunt-*', '!grunt-sass', '!grunt-contrib-sass'] });
 | |
|   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 val === undefined || val !== '0';
 | |
|   };
 | |
| 
 | |
|   // Test task.
 | |
|   var testSubtasks = [];
 | |
|   // Skip core tests if running a different subset of the test suite
 | |
|   if (runSubset('core') &&
 | |
|     // Skip core tests if this is a Savage build
 | |
|     process.env.TRAVIS_REPO_SLUG !== 'twbs-savage/bootstrap') {
 | |
|     testSubtasks = testSubtasks.concat(['dist-css', 'dist-js', 'test-scss', '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 Sauce Labs tests if there's a Sauce access key
 | |
|   if (typeof process.env.SAUCE_ACCESS_KEY !== 'undefined' &&
 | |
|       // Skip Sauce if running a different subset of the test suite
 | |
|       runSubset('sauce-js-unit') &&
 | |
|       // Skip Sauce on Travis when [skip sauce] is in the commit message
 | |
|       isUndefOrNonZero(process.env.TWBS_DO_SAUCE)) {
 | |
|     testSubtasks.push('connect');
 | |
|     testSubtasks.push('saucelabs-qunit');
 | |
|   }
 | |
|   grunt.registerTask('test', testSubtasks);
 | |
|   grunt.registerTask('test-js', ['jscs:core', 'jscs:test', 'jscs:grunt', 'qunit']);
 | |
| 
 | |
|   // JS distribution task.
 | |
|   grunt.registerTask('dist-js', ['concat', 'lineremover', 'babel:dist', 'stamp', 'uglify:core', 'commonjs']);
 | |
| 
 | |
|   grunt.registerTask('test-scss', ['scsslint:scss']);
 | |
| 
 | |
|   // CSS distribution task.
 | |
|   // Supported Compilers: sass (Ruby) and libsass.
 | |
|   (function (sassCompilerName) {
 | |
|     require('./grunt/bs-sass-compile/' + sassCompilerName + '.js')(grunt);
 | |
|   })(process.env.TWBS_SASS || 'libsass');
 | |
|   grunt.registerTask('sass-compile', ['sass:core', 'sass:docs']);
 | |
| 
 | |
|   grunt.registerTask('dist-css', ['sass-compile', 'postcss:core', 'autoprefixer:core', 'usebanner', 'csscomb:dist', 'cssmin:core', 'cssmin:docs']);
 | |
| 
 | |
|   // Full distribution task.
 | |
|   grunt.registerTask('dist', ['clean:dist', 'dist-css', 'dist-js']);
 | |
| 
 | |
|   // Default task.
 | |
|   grunt.registerTask('default', ['clean:dist', 'test']);
 | |
| 
 | |
|   // Version numbering task.
 | |
|   // grunt change-version-number --oldver=A.B.C --newver=X.Y.Z
 | |
|   // This can be overzealous, so its changes should always be manually reviewed!
 | |
|   grunt.registerTask('change-version-number', 'sed');
 | |
| 
 | |
|   grunt.registerTask('commonjs', ['babel:umd', 'npm-js']);
 | |
| 
 | |
|   grunt.registerTask('npm-js', 'Generate npm-js entrypoint module in dist dir.', function () {
 | |
|     var srcFiles = Object.keys(grunt.config.get('babel.umd.files')).map(function (filename) {
 | |
|       return './' + path.join('umd', path.basename(filename))
 | |
|     })
 | |
|     var destFilepath = 'dist/js/npm.js';
 | |
|     generateCommonJSModule(grunt, srcFiles, destFilepath);
 | |
|   });
 | |
| 
 | |
|   // Docs task.
 | |
|   grunt.registerTask('docs-css', ['autoprefixer:docs', 'autoprefixer:examples', 'csscomb:docs', 'csscomb:examples', 'cssmin:docs']);
 | |
|   grunt.registerTask('docs-js', ['uglify:docsJs']);
 | |
|   grunt.registerTask('lint-docs-js', ['jscs:assets']);
 | |
|   grunt.registerTask('docs', ['docs-css', 'docs-js', 'lint-docs-js', 'clean:docs', 'copy:docs']);
 | |
| 
 | |
|   grunt.registerTask('docs-github', ['jekyll:github']);
 | |
| 
 | |
|   // Task for updating the cached npm packages used by the Travis build (which are controlled by test-infra/npm-shrinkwrap.json).
 | |
|   // This task should be run and the updated file should be committed whenever Bootstrap's dependencies change.
 | |
|   grunt.registerTask('update-shrinkwrap', ['exec:npmUpdate', '_update-shrinkwrap']);
 | |
|   grunt.registerTask('_update-shrinkwrap', function () {
 | |
|     var done = this.async();
 | |
|     npmShrinkwrap({ dev: true, dirname: __dirname }, function (err) {
 | |
|       if (err) {
 | |
|         grunt.fail.warn(err);
 | |
|       }
 | |
|       var dest = 'test-infra/npm-shrinkwrap.json';
 | |
|       fs.renameSync('npm-shrinkwrap.json', dest);
 | |
|       grunt.log.writeln('File ' + dest.cyan + ' updated.');
 | |
|       done();
 | |
|     });
 | |
|   });
 | |
|   // Task for updating the cached RubyGem packages used by the Travis build (which are controlled by test-infra/Gemfile.lock).
 | |
|   // This task should be run and the updated file should be committed whenever Bootstrap's RubyGem dependencies change.
 | |
|   grunt.registerTask('update-gemfile-lock', ['exec:bundleUpdate']);
 | |
| };
 |