2014-01-21 01:00:52 +01:00
/ * !
* Bootstrap ' s Gruntfile
* http : //getbootstrap.com
* Copyright 2013 - 2014 Twitter , Inc .
* Licensed under MIT ( https : //github.com/twbs/bootstrap/blob/master/LICENSE)
* /
2013-08-04 01:03:38 +02:00
2013-12-07 01:51:59 +01:00
module . exports = function ( grunt ) {
2013-09-18 18:50:02 +02:00
'use strict' ;
2013-08-04 01:03:38 +02:00
2013-12-03 12:07:57 +01:00
// Force use of Unix newlines
grunt . util . linefeed = '\n' ;
2013-12-08 11:24:47 +01:00
RegExp . quote = function ( string ) {
2014-01-17 20:51:53 +01:00
return string . replace ( /[-\\^$*+?.()|[\]{}]/g , '\\$&' ) ;
} ;
2013-12-28 23:33:32 +01:00
2014-01-17 20:51:53 +01:00
var fs = require ( 'fs' ) ;
2014-01-19 15:04:29 +01:00
var path = require ( 'path' ) ;
2014-12-15 19:18:27 +01:00
var glob = require ( 'glob' ) ;
2014-06-19 19:47:55 +02:00
var npmShrinkwrap = require ( 'npm-shrinkwrap' ) ;
2015-02-05 06:15:01 +01:00
var mq4HoverShim = require ( 'mq4-hover-shim' ) ;
2014-12-10 22:51:43 +01:00
2014-08-28 03:24:23 +02:00
var generateCommonJSModule = require ( './grunt/bs-commonjs-generator.js' ) ;
2014-11-03 16:19:40 +01:00
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 ) ;
} ) ;
} ) ;
2013-12-28 23:33:32 +01:00
2013-08-04 01:03:38 +02:00
// Project configuration.
grunt . initConfig ( {
// Metadata.
pkg : grunt . file . readJSON ( 'package.json' ) ,
2013-09-11 04:21:44 +02:00
banner : '/*!\n' +
2014-01-28 12:16:13 +01:00
' * Bootstrap v<%= pkg.version %> (<%= pkg.homepage %>)\n' +
' * Copyright 2011-<%= grunt.template.today("yyyy") %> <%= pkg.author %>\n' +
2014-02-06 11:37:39 +01:00
' * Licensed under <%= pkg.license.type %> (<%= pkg.license.url %>)\n' +
2014-01-28 12:16:13 +01:00
' */\n' ,
2014-12-17 05:17:54 +01:00
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' ,
2013-08-04 01:03:38 +02:00
// Task configuration.
clean : {
2014-08-14 02:47:16 +02:00
dist : 'dist' ,
docs : 'docs/dist'
2013-08-04 01:03:38 +02:00
} ,
2015-05-07 21:48:22 +02:00
babel : {
2013-08-04 01:03:38 +02:00
options : {
2015-05-07 21:48:22 +02:00
sourceMap : true ,
modules : 'ignore'
2013-08-04 01:03:38 +02:00
} ,
2015-05-07 21:48:22 +02:00
dist : {
files : {
2015-05-10 22:47:11 +02:00
'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'
2015-05-07 21:48:22 +02:00
}
2013-08-04 01:03:38 +02:00
}
} ,
2013-08-08 08:06:29 +02:00
2013-12-07 01:51:38 +01:00
jscs : {
options : {
2014-02-28 16:29:36 +01:00
config : 'js/.jscsrc'
2013-12-07 01:51:38 +01:00
} ,
2014-01-20 22:06:13 +01:00
grunt : {
2015-05-07 21:48:22 +02:00
src : [ 'Gruntfile.js' , 'grunt/*.js' ]
2013-12-07 01:51:38 +01:00
} ,
2014-08-14 02:47:16 +02:00
core : {
2015-05-07 21:48:22 +02:00
src : 'js/*.js'
} ,
es6 : {
src : 'js/src/*.js'
2013-12-07 01:51:38 +01:00
} ,
test : {
2015-05-07 21:48:22 +02:00
src : 'js/tests/unit/*.js'
2014-01-06 05:52:37 +01:00
} ,
assets : {
2014-03-28 06:07:20 +01:00
options : {
requireCamelCaseOrUpperCaseIdentifiers : null
} ,
2015-05-07 21:48:22 +02:00
src : [ 'docs/assets/js/src/*.js' , 'docs/assets/js/*.js' , '!docs/assets/js/*.min.js' ]
2013-12-07 01:51:38 +01:00
}
} ,
2013-08-04 01:03:38 +02:00
concat : {
options : {
2014-10-22 20:52:15 +02:00
banner : '<%= banner %>\n<%= jqueryCheck %>\n<%= jqueryVersionCheck %>' ,
2013-08-04 01:03:38 +02:00
stripBanners : false
} ,
bootstrap : {
src : [
2015-05-06 22:34:14 +02:00
'js/transition.js' ,
2013-08-04 01:03:38 +02:00
'js/alert.js' ,
'js/button.js' ,
'js/carousel.js' ,
'js/collapse.js' ,
'js/dropdown.js' ,
'js/modal.js' ,
'js/tooltip.js' ,
'js/popover.js' ,
2015-05-06 22:34:14 +02:00
'js/scrollspy.js' ,
'js/tab.js' ,
'js/affix.js'
2013-08-04 01:03:38 +02:00
] ,
dest : 'dist/js/<%= pkg.name %>.js'
}
} ,
2013-08-08 08:06:29 +02:00
2015-05-06 22:34:14 +02:00
uglify : {
2014-06-15 16:41:24 +02:00
options : {
2015-05-06 22:34:14 +02:00
compress : {
warnings : false
2015-01-03 22:58:44 +01:00
} ,
2015-05-06 22:34:14 +02:00
mangle : true ,
preserveComments : 'some'
2014-06-15 16:41:24 +02:00
} ,
2015-05-06 22:34:14 +02:00
core : {
src : '<%= concat.bootstrap.dest %>' ,
2013-08-04 01:03:38 +02:00
dest : 'dist/js/<%= pkg.name %>.min.js'
2015-05-06 22:34:14 +02:00
} ,
customize : {
src : configBridge . paths . customizerJs ,
dest : 'docs/assets/js/customize.min.js'
2013-09-19 16:41:14 +02:00
} ,
2013-12-10 23:29:42 +01:00
docsJs : {
2014-11-03 16:19:40 +01:00
src : configBridge . paths . docsJs ,
2013-12-10 23:29:42 +01:00
dest : 'docs/assets/js/docs.min.js'
2013-08-04 01:03:38 +02:00
}
} ,
2014-03-10 00:09:36 +01:00
qunit : {
options : {
inject : 'js/tests/unit/phantom.js'
} ,
files : 'js/tests/index.html'
} ,
2014-12-09 04:02:25 +01:00
scsslint : {
scss : [ 'scss/*.scss' , '!scss/_normalize.scss' ] ,
options : {
config : 'scss/.scss-lint.yml' ,
reporterOutput : 'scss-lint-report.xml'
}
} ,
2015-01-01 10:41:56 +01:00
postcss : {
options : {
map : true ,
2015-02-17 02:18:16 +01:00
processors : [ mq4HoverShim . postprocessorFor ( { hoverSelectorPrefix : '.bs-true-hover ' } ) ]
2015-01-01 10:41:56 +01:00
} ,
core : {
src : 'dist/css/<%= pkg.name %>.css'
}
} ,
2014-02-22 11:17:58 +01:00
autoprefixer : {
options : {
2014-12-17 05:17:54 +01:00
browsers : [
'Android 2.3' ,
'Android >= 4' ,
2014-12-16 03:07:09 +01:00
'Chrome >= 35' ,
2014-12-17 05:17:54 +01:00
'Firefox >= 31' ,
'Explorer >= 9' ,
2014-12-16 03:17:25 +01:00
'iOS >= 7' ,
2014-12-17 05:17:54 +01:00
'Opera >= 12' ,
2014-12-16 03:12:30 +01:00
'Safari >= 7.1'
2014-12-17 05:17:54 +01:00
]
2014-02-22 11:17:58 +01:00
} ,
core : {
options : {
map : true
} ,
src : 'dist/css/<%= pkg.name %>.css'
} ,
docs : {
2014-12-01 05:29:47 +01:00
src : 'docs/assets/css/docs.min.css'
2014-02-22 11:17:58 +01:00
} ,
examples : {
expand : true ,
cwd : 'docs/examples/' ,
src : [ '**/*.css' ] ,
dest : 'docs/examples/'
}
} ,
2013-12-22 08:25:18 +01:00
cssmin : {
2014-03-11 18:48:14 +01:00
options : {
2015-03-09 05:31:58 +01:00
// 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
2014-03-27 18:45:25 +01:00
compatibility : 'ie8' ,
2014-06-25 00:53:10 +02:00
keepSpecialComments : '*' ,
noAdvanced : true
2014-03-11 18:48:14 +01:00
} ,
2014-12-01 05:29:47 +01:00
core : {
files : {
'dist/css/<%= pkg.name %>.min.css' : 'dist/css/<%= pkg.name %>.css'
}
2014-05-19 09:07:31 +02:00
} ,
2014-03-11 18:48:14 +01:00
docs : {
2014-12-01 05:29:47 +01:00
src : 'docs/assets/css/docs.min.css' ,
2014-01-29 20:50:10 +01:00
dest : 'docs/assets/css/docs.min.css'
2013-12-22 08:25:18 +01:00
}
} ,
2013-12-09 08:48:07 +01:00
usebanner : {
2014-03-11 18:48:14 +01:00
options : {
position : 'top' ,
banner : '<%= banner %>'
} ,
files : {
src : 'dist/css/*.css'
2013-12-09 04:09:27 +01:00
}
} ,
2013-12-09 08:48:07 +01:00
csscomb : {
2014-01-31 13:20:33 +01:00
options : {
2014-12-02 23:02:35 +01:00
config : 'scss/.csscomb.json'
2014-01-31 13:20:33 +01:00
} ,
dist : {
2014-03-11 18:48:14 +01:00
expand : true ,
cwd : 'dist/css/' ,
src : [ '*.css' , '!*.min.css' ] ,
dest : 'dist/css/'
2014-01-31 13:20:33 +01:00
} ,
examples : {
2014-02-10 20:09:59 +01:00
expand : true ,
cwd : 'docs/examples/' ,
2014-03-07 17:37:23 +01:00
src : '**/*.css' ,
2014-02-10 20:09:59 +01:00
dest : 'docs/examples/'
2014-02-27 09:12:52 +01:00
} ,
docs : {
2014-08-13 21:12:09 +02:00
src : 'docs/assets/css/src/docs.css' ,
dest : 'docs/assets/css/src/docs.css'
2013-08-04 01:03:38 +02:00
}
} ,
2013-08-18 09:36:51 +02:00
copy : {
2013-12-31 20:38:32 +01:00
docs : {
2015-02-20 10:22:06 +01:00
expand : true ,
cwd : 'dist/' ,
src : [
'**/*'
] ,
dest : 'docs/dist/'
2013-08-18 09:36:51 +02:00
}
} ,
2013-08-04 01:03:38 +02:00
connect : {
server : {
options : {
port : 3000 ,
base : '.'
2013-05-04 16:55:52 +02:00
}
2013-08-04 01:03:38 +02:00
}
} ,
2013-08-13 00:01:06 +02:00
jekyll : {
2014-11-14 14:00:02 +01:00
options : {
config : '_config.yml'
} ,
docs : { } ,
github : {
options : {
raw : 'github: true'
}
}
2013-08-13 00:01:06 +02:00
} ,
2015-01-20 00:08:34 +01:00
htmllint : {
2013-08-13 00:01:06 +02:00
options : {
2015-01-20 00:08:34 +01:00
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).'
2013-10-24 08:25:17 +02:00
]
2013-08-13 00:01:06 +02:00
} ,
2015-01-20 00:08:34 +01:00
src : '_gh_pages/**/*.html'
2013-08-13 00:01:06 +02:00
} ,
2013-08-04 01:03:38 +02:00
watch : {
src : {
2015-05-07 21:48:22 +02:00
files : '<%= jscs.core.src %>' ,
tasks : [ 'qunit' , 'concat' ]
2013-08-04 01:03:38 +02:00
} ,
test : {
2015-05-07 21:48:22 +02:00
files : '<%= jscs.test.src %>' ,
tasks : [ 'qunit' ]
2013-08-04 01:03:38 +02:00
} ,
2015-01-09 11:26:16 +01:00
sass : {
files : 'scss/**/*.scss' ,
tasks : 'sass-compile'
2014-12-01 05:29:47 +01:00
} ,
docs : {
2015-01-09 11:26:16 +01:00
files : 'docs/assets/scss/*.scss' ,
tasks : 'sass:docs'
2013-08-04 01:03:38 +02:00
}
2013-11-02 02:15:25 +01:00
} ,
sed : {
versionNumber : {
pattern : ( function ( ) {
2014-01-17 20:51:53 +01:00
var old = grunt . option ( 'oldver' ) ;
return old ? RegExp . quote ( old ) : old ;
2013-11-02 02:15:25 +01:00
} ) ( ) ,
replacement : grunt . option ( 'newver' ) ,
recursive : true
}
2013-12-04 03:42:31 +01:00
} ,
'saucelabs-qunit' : {
all : {
options : {
build : process . env . TRAVIS _JOB _ID ,
2014-01-16 00:55:47 +01:00
concurrency : 10 ,
2014-06-24 21:12:21 +02:00
maxRetries : 3 ,
2015-01-05 21:40:43 +01:00
maxPollRetries : 4 ,
2015-03-01 09:20:00 +01:00
urls : [ 'http://127.0.0.1:3000/js/tests/index.html?hidepassed' ] ,
2014-03-17 04:30:04 +01:00
browsers : grunt . file . readYAML ( 'grunt/sauce_browsers.yml' )
2013-12-04 03:42:31 +01:00
}
}
2014-01-20 22:06:13 +01:00
} ,
exec : {
npmUpdate : {
command : 'npm update'
2014-12-15 19:18:27 +01:00
} ,
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 ( ' && ' ) ;
}
2014-05-13 07:01:29 +02:00
}
2013-08-04 01:03:38 +02:00
}
} ) ;
2013-05-04 16:55:52 +02:00
2013-08-04 01:03:38 +02:00
// These plugins provide necessary tasks.
2014-12-15 19:18:27 +01:00
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' ] } ) ;
2014-03-07 07:46:15 +01:00
require ( 'time-grunt' ) ( grunt ) ;
2013-05-04 16:55:52 +02:00
2013-08-13 00:01:06 +02:00
// Docs HTML validation task
2015-01-20 00:08:34 +01:00
grunt . registerTask ( 'validate-html' , [ 'jekyll:docs' , 'htmllint' ] ) ;
2013-08-13 00:01:06 +02:00
2014-05-19 00:52:42 +02:00
var runSubset = function ( subset ) {
return ! process . env . TWBS _TEST || process . env . TWBS _TEST === subset ;
} ;
2014-05-19 00:53:40 +02:00
var isUndefOrNonZero = function ( val ) {
return val === undefined || val !== '0' ;
} ;
2014-05-19 00:52:42 +02:00
2013-08-04 01:03:38 +02:00
// Test task.
2013-12-15 04:09:44 +01:00
var testSubtasks = [ ] ;
// Skip core tests if running a different subset of the test suite
2014-11-20 00:43:23 +01:00
if ( runSubset ( 'core' ) &&
2014-12-01 05:29:47 +01:00
// Skip core tests if this is a Savage build
2014-12-15 19:18:27 +01:00
process . env . TRAVIS _REPO _SLUG !== 'twbs-savage/bootstrap' ) {
testSubtasks = testSubtasks . concat ( [ 'dist-css' , 'dist-js' , 'test-scss' , 'test-js' , 'docs' ] ) ;
2013-12-15 04:09:44 +01:00
}
// Skip HTML validation if running a different subset of the test suite
2014-05-19 00:53:40 +02:00
if ( runSubset ( 'validate-html' ) &&
// Skip HTML5 validator on Travis when [skip validator] is in the commit message
isUndefOrNonZero ( process . env . TWBS _DO _VALIDATOR ) ) {
2013-12-15 04:09:44 +01:00
testSubtasks . push ( 'validate-html' ) ;
}
2013-12-04 03:42:31 +01:00
// Only run Sauce Labs tests if there's a Sauce access key
2014-01-17 20:51:53 +01:00
if ( typeof process . env . SAUCE _ACCESS _KEY !== 'undefined' &&
2013-12-15 04:09:44 +01:00
// Skip Sauce if running a different subset of the test suite
2014-05-19 00:53:40 +02:00
runSubset ( 'sauce-js-unit' ) &&
// Skip Sauce on Travis when [skip sauce] is in the commit message
isUndefOrNonZero ( process . env . TWBS _DO _SAUCE ) ) {
2013-12-04 03:42:31 +01:00
testSubtasks . push ( 'connect' ) ;
testSubtasks . push ( 'saucelabs-qunit' ) ;
2013-08-06 09:39:35 +02:00
}
grunt . registerTask ( 'test' , testSubtasks ) ;
2015-05-07 21:48:22 +02:00
grunt . registerTask ( 'test-js' , [ 'jscs:core' , 'jscs:test' , 'jscs:grunt' , 'qunit' ] ) ;
2013-05-04 16:55:52 +02:00
2013-08-04 01:03:38 +02:00
// JS distribution task.
2015-05-06 22:34:14 +02:00
grunt . registerTask ( 'dist-js' , [ 'concat' , 'uglify:core' , 'commonjs' ] ) ;
2013-05-04 16:55:52 +02:00
2014-12-09 04:02:25 +01:00
grunt . registerTask ( 'test-scss' , [ 'scsslint:scss' ] ) ;
2013-08-04 01:03:38 +02:00
// CSS distribution task.
2014-12-15 19:18:27 +01:00
// Supported Compilers: sass (Ruby) and libsass.
( function ( sassCompilerName ) {
require ( './grunt/bs-sass-compile/' + sassCompilerName + '.js' ) ( grunt ) ;
} ) ( process . env . TWBS _SASS || 'libsass' ) ;
2014-12-02 23:02:35 +01:00
grunt . registerTask ( 'sass-compile' , [ 'sass:core' , 'sass:docs' ] ) ;
2014-12-15 19:18:27 +01:00
2015-01-01 10:41:56 +01:00
grunt . registerTask ( 'dist-css' , [ 'sass-compile' , 'postcss:core' , 'autoprefixer:core' , 'usebanner' , 'csscomb:dist' , 'cssmin:core' , 'cssmin:docs' ] ) ;
2013-08-18 09:36:51 +02:00
2013-08-04 01:03:38 +02:00
// Full distribution task.
2014-12-01 05:29:47 +01:00
grunt . registerTask ( 'dist' , [ 'clean:dist' , 'dist-css' , 'dist-js' ] ) ;
2013-05-04 16:55:52 +02:00
2013-08-04 01:03:38 +02:00
// Default task.
2014-12-01 05:29:47 +01:00
grunt . registerTask ( 'default' , [ 'clean:dist' , 'test' ] ) ;
2013-08-08 08:06:29 +02:00
2013-11-02 02:15:25 +01:00
// 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!
2014-01-19 15:04:29 +01:00
grunt . registerTask ( 'change-version-number' , 'sed' ) ;
2013-11-02 02:15:25 +01:00
2014-08-28 03:32:08 +02:00
grunt . registerTask ( 'commonjs' , 'Generate CommonJS entrypoint module in dist dir.' , function ( ) {
2014-08-28 03:39:41 +02:00
var srcFiles = grunt . config . get ( 'concat.bootstrap.src' ) ;
var destFilepath = 'dist/js/npm.js' ;
generateCommonJSModule ( grunt , srcFiles , destFilepath ) ;
2014-08-28 03:24:23 +02:00
} ) ;
2014-08-14 02:47:16 +02:00
// Docs task.
grunt . registerTask ( 'docs-css' , [ 'autoprefixer:docs' , 'autoprefixer:examples' , 'csscomb:docs' , 'csscomb:examples' , 'cssmin:docs' ] ) ;
2014-12-10 22:51:43 +01:00
grunt . registerTask ( 'docs-js' , [ 'uglify:docsJs' ] ) ;
2015-05-07 21:48:22 +02:00
grunt . registerTask ( 'lint-docs-js' , [ 'jscs:assets' ] ) ;
2014-12-02 23:02:35 +01:00
grunt . registerTask ( 'docs' , [ 'docs-css' , 'docs-js' , 'lint-docs-js' , 'clean:docs' , 'copy:docs' ] ) ;
2014-08-14 02:47:16 +02:00
2014-11-14 14:00:02 +01:00
grunt . registerTask ( 'docs-github' , [ 'jekyll:github' ] ) ;
2014-06-19 19:47:55 +02:00
// 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 ) {
2014-07-11 01:42:09 +02:00
grunt . fail . warn ( err ) ;
2014-06-19 19:47:55 +02:00
}
var dest = 'test-infra/npm-shrinkwrap.json' ;
fs . renameSync ( 'npm-shrinkwrap.json' , dest ) ;
grunt . log . writeln ( 'File ' + dest . cyan + ' updated.' ) ;
done ( ) ;
} ) ;
} ) ;
2014-12-15 19:18:27 +01:00
// 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' ] ) ;
2013-09-18 08:04:09 +02:00
} ;