2014-01-21 01:00:52 +01:00
/ * !
* Bootstrap ' s Gruntfile
* http : //getbootstrap.com
2016-05-20 18:04:31 +02:00
* Copyright 2013 - 2016 The Bootstrap Authors
2016-01-01 21:31:38 +01:00
* Copyright 2013 - 2016 Twitter , Inc .
2014-01-21 01:00:52 +01:00
* 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' ) ;
2015-09-26 00:32:42 +02:00
var isTravis = require ( 'is-travis' ) ;
2014-12-10 22:51:43 +01:00
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 ) {
Fix docs asset file paths in /grunt/configBridge.json (#20178)
Previously, when running the docs locally, the site, rooted at:
http://localhost:9001/
would reference docs assets using relative URLs such as:
/../assets/js/vendor/anchor.min.js
which is equivalent to:
http://localhost:9001/../assets/js/vendor/anchor.min.js
which is nonsense, since the root directory has no parent directory.
Apparently browsers silently ignore this extra '..', hence why this wasn't noticed until now.
But if you adjust Jekyll's `baseurl` setting, this mistake causes incorrect URLs to get generated.
This commit corrects the problem by removing the extra '../' from the paths.
These paths are also referenced in the Gruntfile, where the fix actually allows us to simplify the code.
Previously, in the Gruntfile, we were doing, e.g.:
path.join('./docs/assets', '../assets/js/vendor/anchor.min.js')
which calculates to:
./docs/assets/../assets/js/vendor/anchor.min.js
which can be simplified to:
./docs/assets/js/vendor/anchor.min.js
So we can remove the '/assets' suffix from the left argument
and the '../' prefix from the right argument
and still obtain the same result.
2016-06-27 01:42:13 +02:00
arr [ i ] = path . join ( './docs' , val ) ;
2014-11-03 16:19:40 +01:00
} ) ;
} ) ;
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' +
2015-08-10 08:16:43 +02:00
' * Licensed under MIT (https://github.com/twbs/bootstrap/blob/master/LICENSE)\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' +
2016-06-29 07:19:46 +02:00
' if ((version[0] < 2 && version[1] < 9) || (version[0] == 1 && version[1] == 9 && version[2] < 1) || (version[0] >= 4)) {\n' +
' throw new Error(\'Bootstrap\\\'s JavaScript requires at least jQuery v1.9.1 but less than v4.0.0\')\n' +
2014-12-17 05:17:54 +01:00
' }\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-13 01:52:54 +02:00
// JS build configuration
2015-05-07 21:48:22 +02:00
babel : {
2015-05-13 21:48:34 +02:00
dev : {
2015-05-13 01:52:54 +02:00
options : {
sourceMap : true ,
modules : 'ignore'
2014-02-07 13:34:17 +01:00
} ,
2015-05-07 21:48:22 +02:00
files : {
2015-05-11 21:05:35 +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' ,
'js/dist/modal.js' : 'js/src/modal.js' ,
2015-05-11 21:29:06 +02:00
'js/dist/scrollspy.js' : 'js/src/scrollspy.js' ,
2015-05-12 08:32:37 +02:00
'js/dist/tab.js' : 'js/src/tab.js' ,
2015-05-12 23:28:11 +02:00
'js/dist/tooltip.js' : 'js/src/tooltip.js' ,
'js/dist/popover.js' : 'js/src/popover.js'
2015-05-07 21:48:22 +02:00
}
2013-08-04 01:03:38 +02:00
} ,
2015-05-13 01:52:54 +02:00
dist : {
2014-03-17 08:07:21 +01:00
options : {
2015-05-13 01:52:54 +02:00
modules : 'ignore'
2014-03-17 08:07:21 +01:00
} ,
2015-05-13 01:52:54 +02:00
files : {
'<%= concat.bootstrap.dest %>' : '<%= concat.bootstrap.dest %>'
}
2013-08-04 01:03:38 +02:00
}
} ,
2013-08-08 08:06:29 +02:00
2015-05-13 01:52:54 +02:00
stamp : {
2013-08-04 01:03:38 +02:00
options : {
2015-05-13 01:52:54 +02:00
banner : '<%= banner %>\n<%= jqueryCheck %>\n<%= jqueryVersionCheck %>\n+function ($) {\n' ,
footer : '\n}(jQuery);'
2013-08-04 01:03:38 +02:00
} ,
bootstrap : {
2015-05-13 01:52:54 +02:00
files : {
src : '<%= concat.bootstrap.dest %>'
}
2013-08-04 01:03:38 +02:00
}
} ,
2013-08-08 08:06:29 +02:00
2013-08-04 01:03:38 +02:00
concat : {
2014-06-15 16:41:24 +02:00
options : {
2016-04-02 09:03:04 +02:00
// Custom function to remove all export and import statements
process : function ( src ) {
return src . replace ( /^(export|import).*/gm , '' ) ;
} ,
2013-08-04 01:03:38 +02:00
stripBanners : false
2014-06-15 16:41:24 +02:00
} ,
2013-08-04 01:03:38 +02:00
bootstrap : {
2015-01-03 22:58:44 +01:00
src : [
2015-05-13 01:52:54 +02:00
'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'
2015-01-03 22:58:44 +01:00
] ,
2013-08-04 01:03:38 +02:00
dest : 'dist/js/<%= pkg.name %>.js'
2015-01-03 22:58:44 +01:00
}
} ,
uglify : {
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 ,
2015-11-16 08:32:59 +01:00
preserveComments : /^!|@preserve|@license|@cc_on/i
2013-09-19 16:41:14 +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
} ,
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'
} ,
2015-05-13 01:52:54 +02:00
// CSS build configuration
2014-12-09 04:02:25 +01:00
scsslint : {
options : {
2015-09-06 11:16:01 +02:00
bundleExec : true ,
2015-09-30 00:48:56 +02:00
config : 'scss/.scss-lint.yml' ,
2015-06-19 09:14:52 +02:00
reporterOutput : null
} ,
2016-01-04 03:09:09 +01:00
core : {
src : [ 'scss/*.scss' , '!scss/_normalize.scss' ]
} ,
docs : {
2016-02-09 22:34:47 +01:00
src : [ 'docs/assets/scss/*.scss' , '!docs/assets/scss/docs.scss' ]
2016-01-04 03:09:09 +01:00
}
2014-12-09 04:02:25 +01:00
} ,
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
2015-08-25 14:37:19 +02:00
compatibility : 'ie9' ,
2014-06-25 00:53:10 +02:00
keepSpecialComments : '*' ,
2015-06-05 03:05:59 +02:00
sourceMap : true ,
2016-10-03 04:28:52 +02:00
// sourceMapInlineSources: true,
2015-12-08 06:51:04 +01:00
advanced : false
2014-03-11 18:48:14 +01:00
} ,
2014-12-01 05:29:47 +01:00
core : {
2015-06-18 19:19:50 +02:00
files : [
{
expand : true ,
cwd : 'dist/css' ,
src : [ '*.css' , '!*.min.css' ] ,
dest : 'dist/css' ,
ext : '.min.css'
}
]
2014-05-19 09:07:31 +02:00
} ,
2014-03-11 18:48:14 +01:00
docs : {
2016-07-27 06:09:39 +02:00
files : [
{
expand : true ,
cwd : 'docs/assets/css' ,
src : [ '*.css' , '!*.min.css' ] ,
dest : 'docs/assets/css' ,
ext : '.min.css'
}
]
2013-12-22 08:25:18 +01: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 : {
2015-08-25 07:43:47 +02:00
bundleExec : true ,
2015-10-28 06:20:47 +01:00
config : '_config.yml' ,
incremental : false
2014-11-14 14:00:02 +01:00
} ,
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 : [
2016-06-29 07:54:25 +02:00
'Attribute “autocomplete” is only allowed when the input type is “color”, “date”, “datetime”, “datetime-local”, “email”, “hidden”, “month”, “number”, “password”, “range”, “search”, “tel”, “text”, “time”, “url”, or “week”.' ,
2015-01-20 00:08:34 +01:00
'Attribute “autocomplete” not allowed on element “button” at this point.' ,
2015-05-28 00:49:55 +02:00
'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).' ,
2016-06-29 07:54:25 +02:00
'Element “div” not allowed as child of element “progress” in this context. (Suppressing further errors from this subtree.)' ,
'Element “img” is missing required attribute “src”.' ,
2016-01-07 00:52:12 +01:00
'The “color” input type is not supported in all browsers. Please be sure to test, and consider using a polyfill.' ,
'The “date” input type is not supported in all browsers. Please be sure to test, and consider using a polyfill.' ,
2016-06-29 07:54:25 +02:00
'The “datetime” input type is not supported in all browsers. Please be sure to test, and consider using a polyfill.' ,
2016-01-07 00:52:12 +01:00
'The “datetime-local” input type is not supported in all browsers. Please be sure to test, and consider using a polyfill.' ,
'The “month” input type is not supported in all browsers. Please be sure to test, and consider using a polyfill.' ,
'The “time” input type is not supported in all browsers. Please be sure to test, and consider using a polyfill.' ,
2016-01-26 13:55:26 +01:00
'The “week” input type is not supported in all browsers. Please be sure to test, and consider using a polyfill.'
2013-10-24 08:25:17 +02:00
]
2013-08-13 00:01:06 +02:00
} ,
2015-11-19 10:49:12 +01:00
src : [ '_gh_pages/**/*.html' , 'js/tests/visual/*.html' ]
2013-08-13 00:01:06 +02:00
} ,
2013-08-04 01:03:38 +02:00
watch : {
src : {
2016-06-05 02:46:36 +02:00
files : '<%= concat.bootstrap.src %>' ,
2015-05-13 21:48:34 +02:00
tasks : [ 'babel:dev' ]
2013-08-04 01:03:38 +02:00
} ,
2015-01-09 11:26:16 +01:00
sass : {
files : 'scss/**/*.scss' ,
2015-08-13 05:14:14 +02:00
tasks : [ 'dist-css' , 'docs' ]
2015-08-15 07:45:55 +02:00
} ,
docs : {
files : 'docs/assets/scss/**/*.scss' ,
tasks : [ 'dist-css' , 'docs' ]
2013-08-04 01:03:38 +02:00
}
2013-11-02 02:15:25 +01:00
} ,
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 : {
2016-06-21 01:18:21 +02:00
postcss : {
command : 'npm run postcss'
} ,
'postcss-docs' : {
command : 'npm run postcss-docs'
2016-06-27 23:19:10 +02:00
} ,
2016-07-15 01:22:56 +02:00
htmlhint : {
command : 'npm run htmlhint'
} ,
2016-06-27 23:19:10 +02:00
'upload-preview' : {
command : './grunt/upload-preview.sh'
2016-06-21 01:18:21 +02:00
}
2015-08-19 07:47:26 +02:00
} ,
buildcontrol : {
options : {
dir : '_gh_pages' ,
commit : true ,
push : true ,
message : 'Built %sourceName% from commit %sourceCommit% on branch %sourceBranch%'
} ,
pages : {
options : {
remote : 'git@github.com:twbs/derpstrap.git' ,
branch : 'gh-pages'
}
}
2015-11-16 17:05:29 +01:00
} ,
compress : {
main : {
options : {
archive : 'bootstrap-<%= pkg.version %>-dist.zip' ,
mode : 'zip' ,
level : 9 ,
pretty : true
} ,
files : [
{
expand : true ,
cwd : 'dist/' ,
src : [ '**' ] ,
dest : 'bootstrap-<%= pkg.version %>-dist'
}
]
}
2013-08-04 01:03:38 +02:00
}
2015-11-16 17:05:29 +01: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
2016-07-15 01:22:56 +02:00
grunt . registerTask ( 'validate-html' , [ 'jekyll:docs' , 'htmllint' , 'exec:htmlhint' ] ) ;
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' ) {
2016-06-05 02:46:36 +02:00
testSubtasks = testSubtasks . concat ( [ 'dist-css' , 'dist-js' , 'test-scss' , 'qunit' , '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' ) &&
2015-09-26 00:32:42 +02:00
isTravis &&
// Skip HTML5 validator when [skip validator] is in the commit message
2014-05-19 00:53:40 +02:00
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
2016-06-27 23:19:10 +02:00
runSubset ( 'sauce-js-unit' ) ) {
testSubtasks = testSubtasks . concat ( [ 'dist' , 'docs-css' , 'docs-js' , 'clean:docs' , 'copy:docs' , 'exec:upload-preview' ] ) ;
// Skip Sauce on Travis when [skip sauce] is in the commit message
if ( isUndefOrNonZero ( process . env . TWBS _DO _SAUCE ) ) {
testSubtasks . push ( 'connect' ) ;
testSubtasks . push ( 'saucelabs-qunit' ) ;
}
2013-08-06 09:39:35 +02:00
}
grunt . registerTask ( 'test' , testSubtasks ) ;
2013-05-04 16:55:52 +02:00
2013-08-04 01:03:38 +02:00
// JS distribution task.
2016-06-10 18:28:03 +02:00
grunt . registerTask ( 'dist-js' , [ 'babel:dev' , 'concat' , 'babel:dist' , 'stamp' , 'uglify:core' ] ) ;
2013-05-04 16:55:52 +02:00
2016-01-04 03:09:09 +01:00
grunt . registerTask ( 'test-scss' , [ 'scsslint:core' ] ) ;
2014-12-09 04:02:25 +01:00
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' ) ;
2015-08-10 06:45:08 +02:00
// grunt.registerTask('sass-compile', ['sass:core', 'sass:extras', 'sass:docs']);
2016-09-12 07:29:45 +02:00
grunt . registerTask ( 'sass-compile' , [ 'sass:core' , 'sass:extras' , 'sass:docs' ] ) ;
2014-12-15 19:18:27 +01:00
2016-06-21 01:18:21 +02:00
grunt . registerTask ( 'dist-css' , [ 'sass-compile' , 'exec:postcss' , '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
2014-08-14 02:47:16 +02:00
// Docs task.
2016-06-21 01:18:21 +02:00
grunt . registerTask ( 'docs-css' , [ 'cssmin:docs' , 'exec:postcss-docs' ] ) ;
2016-01-04 09:22:03 +01:00
grunt . registerTask ( 'lint-docs-css' , [ 'scsslint:docs' ] ) ;
2014-12-10 22:51:43 +01:00
grunt . registerTask ( 'docs-js' , [ 'uglify:docsJs' ] ) ;
2016-06-05 02:46:36 +02:00
grunt . registerTask ( 'docs' , [ 'lint-docs-css' , 'docs-css' , 'docs-js' , 'clean:docs' , 'copy:docs' ] ) ;
2015-12-09 03:25:11 +01:00
grunt . registerTask ( 'docs-github' , [ 'jekyll:github' ] ) ;
2014-08-14 02:47:16 +02:00
2015-12-06 21:11:10 +01:00
grunt . registerTask ( 'prep-release' , [ 'dist' , 'docs' , 'docs-github' , 'compress' ] ) ;
2014-11-14 14:00:02 +01:00
2015-08-19 07:49:26 +02:00
// Publish to GitHub
grunt . registerTask ( 'publish' , [ 'buildcontrol:pages' ] ) ;
2013-09-18 08:04:09 +02:00
} ;