2014-01-21 08:00:52 +08:00
/ * !
* Bootstrap ' s Gruntfile
2016-10-04 00:55:59 +08:00
* https : //getbootstrap.com
2016-05-21 00:04:31 +08:00
* Copyright 2013 - 2016 The Bootstrap Authors
2016-01-02 04:31:38 +08:00
* Copyright 2013 - 2016 Twitter , Inc .
2014-01-21 08:00:52 +08:00
* Licensed under MIT ( https : //github.com/twbs/bootstrap/blob/master/LICENSE)
* /
2013-08-04 07:03:38 +08:00
2013-12-07 08:51:59 +08:00
module . exports = function ( grunt ) {
2013-09-19 00:50:02 +08:00
'use strict' ;
2013-08-04 07:03:38 +08:00
2013-12-03 19:07:57 +08:00
// Force use of Unix newlines
grunt . util . linefeed = '\n' ;
2013-12-08 18:24:47 +08:00
RegExp . quote = function ( string ) {
2014-01-18 03:51:53 +08:00
return string . replace ( /[-\\^$*+?.()|[\]{}]/g , '\\$&' ) ;
} ;
2013-12-29 06:33:32 +08:00
2014-01-18 03:51:53 +08:00
var fs = require ( 'fs' ) ;
2014-01-19 22:04:29 +08:00
var path = require ( 'path' ) ;
2015-09-26 06:32:42 +08:00
var isTravis = require ( 'is-travis' ) ;
2014-12-11 05:51:43 +08:00
2014-11-03 23:19:40 +08: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 07:42:13 +08:00
arr [ i ] = path . join ( './docs' , val ) ;
2014-11-03 23:19:40 +08:00
} ) ;
} ) ;
2013-12-29 06:33:32 +08:00
2013-08-04 07:03:38 +08:00
// Project configuration.
grunt . initConfig ( {
// Metadata.
pkg : grunt . file . readJSON ( 'package.json' ) ,
2013-09-11 10:21:44 +08:00
banner : '/*!\n' +
2014-01-28 19:16:13 +08:00
' * Bootstrap v<%= pkg.version %> (<%= pkg.homepage %>)\n' +
' * Copyright 2011-<%= grunt.template.today("yyyy") %> <%= pkg.author %>\n' +
2015-08-10 14:16:43 +08:00
' * Licensed under MIT (https://github.com/twbs/bootstrap/blob/master/LICENSE)\n' +
2014-01-28 19:16:13 +08:00
' */\n' ,
2014-12-17 12:17:54 +08:00
jqueryCheck : 'if (typeof jQuery === \'undefined\') {\n' +
2016-11-27 10:46:28 +08:00
' throw new Error(\'Bootstrap\\\'s JavaScript requires jQuery. jQuery must be included before Bootstrap\\\'s JavaScript.\')\n' +
2014-12-17 12:17:54 +08:00
'}\n' ,
jqueryVersionCheck : '+function ($) {\n' +
' var version = $.fn.jquery.split(\' \')[0].split(\'.\')\n' +
2016-06-29 13:19:46 +08: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 12:17:54 +08:00
' }\n' +
'}(jQuery);\n\n' ,
2013-08-04 07:03:38 +08:00
// Task configuration.
clean : {
2014-08-14 08:47:16 +08:00
dist : 'dist' ,
docs : 'docs/dist'
2013-08-04 07:03:38 +08:00
} ,
2015-05-13 07:52:54 +08:00
// JS build configuration
2015-05-08 03:48:22 +08:00
babel : {
2015-05-14 03:48:34 +08:00
dev : {
2015-05-13 07:52:54 +08:00
options : {
2016-10-06 21:21:05 +08:00
sourceMap : true
2014-02-07 20:34:17 +08:00
} ,
2015-05-08 03:48:22 +08:00
files : {
2015-05-12 03:05:35 +08: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-12 03:29:06 +08:00
'js/dist/scrollspy.js' : 'js/src/scrollspy.js' ,
2015-05-12 14:32:37 +08:00
'js/dist/tab.js' : 'js/src/tab.js' ,
2015-05-13 05:28:11 +08:00
'js/dist/tooltip.js' : 'js/src/tooltip.js' ,
'js/dist/popover.js' : 'js/src/popover.js'
2015-05-08 03:48:22 +08:00
}
2013-08-04 07:03:38 +08:00
} ,
2015-05-13 07:52:54 +08:00
dist : {
2014-03-17 15:07:21 +08:00
options : {
2016-10-06 21:21:05 +08:00
extends : '../../js/.babelrc'
2014-03-17 15:07:21 +08:00
} ,
2015-05-13 07:52:54 +08:00
files : {
'<%= concat.bootstrap.dest %>' : '<%= concat.bootstrap.dest %>'
}
2013-08-04 07:03:38 +08:00
}
} ,
2013-08-08 14:06:29 +08:00
2015-05-13 07:52:54 +08:00
stamp : {
2013-08-04 07:03:38 +08:00
options : {
2016-10-06 21:21:05 +08:00
banner : '<%= banner %>\n<%= jqueryCheck %>\n<%= jqueryVersionCheck %>\n+function () {\n' ,
footer : '\n}();'
2013-08-04 07:03:38 +08:00
} ,
bootstrap : {
2015-05-13 07:52:54 +08:00
files : {
src : '<%= concat.bootstrap.dest %>'
}
2013-08-04 07:03:38 +08:00
}
} ,
2013-08-08 14:06:29 +08:00
2013-08-04 07:03:38 +08:00
concat : {
2014-06-15 22:41:24 +08:00
options : {
2016-04-02 15:03:04 +08:00
// Custom function to remove all export and import statements
process : function ( src ) {
return src . replace ( /^(export|import).*/gm , '' ) ;
2016-10-06 21:21:05 +08:00
}
2014-06-15 22:41:24 +08:00
} ,
2013-08-04 07:03:38 +08:00
bootstrap : {
2015-01-04 05:58:44 +08:00
src : [
2015-05-13 07:52:54 +08: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-04 05:58:44 +08:00
] ,
2013-08-04 07:03:38 +08:00
dest : 'dist/js/<%= pkg.name %>.js'
2015-01-04 05:58:44 +08:00
}
} ,
uglify : {
options : {
2015-05-07 04:34:14 +08:00
compress : {
warnings : false
2015-01-04 05:58:44 +08:00
} ,
2015-05-07 04:34:14 +08:00
mangle : true ,
2015-11-16 15:32:59 +08:00
preserveComments : /^!|@preserve|@license|@cc_on/i
2013-09-19 22:41:14 +08:00
} ,
2015-05-07 04:34:14 +08:00
core : {
src : '<%= concat.bootstrap.dest %>' ,
2013-08-04 07:03:38 +08:00
dest : 'dist/js/<%= pkg.name %>.min.js'
2015-05-07 04:34:14 +08:00
} ,
2013-12-11 06:29:42 +08:00
docsJs : {
2014-11-03 23:19:40 +08:00
src : configBridge . paths . docsJs ,
2013-12-11 06:29:42 +08:00
dest : 'docs/assets/js/docs.min.js'
2013-08-04 07:03:38 +08:00
}
} ,
2014-03-10 07:09:36 +08:00
qunit : {
options : {
inject : 'js/tests/unit/phantom.js'
} ,
files : 'js/tests/index.html'
} ,
2015-05-13 07:52:54 +08:00
// CSS build configuration
2014-12-09 11:02:25 +08:00
scsslint : {
options : {
2015-09-06 17:16:01 +08:00
bundleExec : true ,
2015-09-30 06:48:56 +08:00
config : 'scss/.scss-lint.yml' ,
2015-06-19 15:14:52 +08:00
reporterOutput : null
} ,
2016-01-04 10:09:09 +08:00
core : {
src : [ 'scss/*.scss' , '!scss/_normalize.scss' ]
} ,
docs : {
2016-02-10 05:34:47 +08:00
src : [ 'docs/assets/scss/*.scss' , '!docs/assets/scss/docs.scss' ]
2016-01-04 10:09:09 +08:00
}
2014-12-09 11:02:25 +08:00
} ,
2013-12-22 15:25:18 +08:00
cssmin : {
2014-03-12 01:48:14 +08:00
options : {
2015-06-05 09:05:59 +08:00
sourceMap : true ,
2016-10-03 10:28:52 +08:00
// sourceMapInlineSources: true,
2015-12-08 13:51:04 +08:00
advanced : false
2014-03-12 01:48:14 +08:00
} ,
2014-12-01 12:29:47 +08:00
core : {
2015-06-19 01:19:50 +08:00
files : [
{
expand : true ,
cwd : 'dist/css' ,
src : [ '*.css' , '!*.min.css' ] ,
dest : 'dist/css' ,
ext : '.min.css'
}
]
2014-05-19 15:07:31 +08:00
} ,
2014-03-12 01:48:14 +08:00
docs : {
2016-07-27 12:09:39 +08:00
files : [
{
expand : true ,
cwd : 'docs/assets/css' ,
src : [ '*.css' , '!*.min.css' ] ,
dest : 'docs/assets/css' ,
ext : '.min.css'
}
]
2013-12-22 15:25:18 +08:00
}
} ,
2013-08-18 15:36:51 +08:00
copy : {
2014-01-01 03:38:32 +08:00
docs : {
2015-02-20 17:22:06 +08:00
expand : true ,
cwd : 'dist/' ,
src : [
'**/*'
] ,
dest : 'docs/dist/'
2013-08-18 15:36:51 +08:00
}
} ,
2013-08-04 07:03:38 +08:00
connect : {
server : {
options : {
port : 3000 ,
base : '.'
2013-05-04 22:55:52 +08:00
}
2013-08-04 07:03:38 +08:00
}
} ,
2013-08-13 06:01:06 +08:00
jekyll : {
2014-11-14 21:00:02 +08:00
options : {
2015-08-25 13:43:47 +08:00
bundleExec : true ,
2015-10-28 13:20:47 +08:00
config : '_config.yml' ,
incremental : false
2014-11-14 21:00:02 +08:00
} ,
docs : { } ,
github : {
options : {
raw : 'github: true'
}
}
2013-08-13 06:01:06 +08:00
} ,
2015-01-20 07:08:34 +08:00
htmllint : {
2013-08-13 06:01:06 +08:00
options : {
2015-01-20 07:08:34 +08:00
ignore : [
2016-06-29 13:54:25 +08: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 07:08:34 +08:00
'Attribute “autocomplete” not allowed on element “button” at this point.' ,
2015-05-28 06:49:55 +08: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 13:54:25 +08: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 07:52:12 +08: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 13:54:25 +08:00
'The “datetime” input type is not supported in all browsers. Please be sure to test, and consider using a polyfill.' ,
2016-01-07 07:52:12 +08: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 20:55:26 +08:00
'The “week” input type is not supported in all browsers. Please be sure to test, and consider using a polyfill.'
2013-10-24 14:25:17 +08:00
]
2013-08-13 06:01:06 +08:00
} ,
2015-11-19 17:49:12 +08:00
src : [ '_gh_pages/**/*.html' , 'js/tests/visual/*.html' ]
2013-08-13 06:01:06 +08:00
} ,
2013-08-04 07:03:38 +08:00
watch : {
src : {
2016-06-05 08:46:36 +08:00
files : '<%= concat.bootstrap.src %>' ,
2015-05-14 03:48:34 +08:00
tasks : [ 'babel:dev' ]
2013-08-04 07:03:38 +08:00
} ,
2015-01-09 18:26:16 +08:00
sass : {
files : 'scss/**/*.scss' ,
2015-08-13 11:14:14 +08:00
tasks : [ 'dist-css' , 'docs' ]
2015-08-15 13:45:55 +08:00
} ,
docs : {
files : 'docs/assets/scss/**/*.scss' ,
tasks : [ 'dist-css' , 'docs' ]
2013-08-04 07:03:38 +08:00
}
2013-11-02 09:15:25 +08:00
} ,
2013-12-04 10:42:31 +08:00
'saucelabs-qunit' : {
all : {
options : {
build : process . env . TRAVIS _JOB _ID ,
2014-01-16 07:55:47 +08:00
concurrency : 10 ,
2014-06-25 03:12:21 +08:00
maxRetries : 3 ,
2015-01-06 04:40:43 +08:00
maxPollRetries : 4 ,
2015-03-01 16:20:00 +08:00
urls : [ 'http://127.0.0.1:3000/js/tests/index.html?hidepassed' ] ,
2014-03-17 11:30:04 +08:00
browsers : grunt . file . readYAML ( 'grunt/sauce_browsers.yml' )
2013-12-04 10:42:31 +08:00
}
}
2014-01-21 05:06:13 +08:00
} ,
exec : {
2016-06-21 07:18:21 +08:00
postcss : {
command : 'npm run postcss'
} ,
'postcss-docs' : {
command : 'npm run postcss-docs'
2016-06-28 05:19:10 +08:00
} ,
2016-07-15 07:22:56 +08:00
htmlhint : {
command : 'npm run htmlhint'
2016-06-21 07:18:21 +08:00
}
2015-08-19 13:47:26 +08: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-17 00:05:29 +08: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 07:03:38 +08:00
}
2015-11-17 00:05:29 +08:00
2013-08-04 07:03:38 +08:00
} ) ;
2013-05-04 22:55:52 +08:00
2013-08-04 07:03:38 +08:00
// These plugins provide necessary tasks.
2014-12-16 02:18:27 +08: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 14:46:15 +08:00
require ( 'time-grunt' ) ( grunt ) ;
2013-05-04 22:55:52 +08:00
2013-08-13 06:01:06 +08:00
// Docs HTML validation task
2016-07-15 07:22:56 +08:00
grunt . registerTask ( 'validate-html' , [ 'jekyll:docs' , 'htmllint' , 'exec:htmlhint' ] ) ;
2013-08-13 06:01:06 +08:00
2014-05-19 06:52:42 +08:00
var runSubset = function ( subset ) {
return ! process . env . TWBS _TEST || process . env . TWBS _TEST === subset ;
} ;
2014-05-19 06:53:40 +08:00
var isUndefOrNonZero = function ( val ) {
return val === undefined || val !== '0' ;
} ;
2014-05-19 06:52:42 +08:00
2013-08-04 07:03:38 +08:00
// Test task.
2013-12-15 11:09:44 +08:00
var testSubtasks = [ ] ;
// Skip core tests if running a different subset of the test suite
2014-11-20 07:43:23 +08:00
if ( runSubset ( 'core' ) &&
2014-12-01 12:29:47 +08:00
// Skip core tests if this is a Savage build
2014-12-16 02:18:27 +08:00
process . env . TRAVIS _REPO _SLUG !== 'twbs-savage/bootstrap' ) {
2016-06-05 08:46:36 +08:00
testSubtasks = testSubtasks . concat ( [ 'dist-css' , 'dist-js' , 'test-scss' , 'qunit' , 'docs' ] ) ;
2013-12-15 11:09:44 +08:00
}
// Skip HTML validation if running a different subset of the test suite
2014-05-19 06:53:40 +08:00
if ( runSubset ( 'validate-html' ) &&
2015-09-26 06:32:42 +08:00
isTravis &&
// Skip HTML5 validator when [skip validator] is in the commit message
2014-05-19 06:53:40 +08:00
isUndefOrNonZero ( process . env . TWBS _DO _VALIDATOR ) ) {
2013-12-15 11:09:44 +08:00
testSubtasks . push ( 'validate-html' ) ;
}
2013-12-04 10:42:31 +08:00
// Only run Sauce Labs tests if there's a Sauce access key
2014-01-18 03:51:53 +08:00
if ( typeof process . env . SAUCE _ACCESS _KEY !== 'undefined' &&
2013-12-15 11:09:44 +08:00
// Skip Sauce if running a different subset of the test suite
2016-06-28 05:19:10 +08:00
runSubset ( 'sauce-js-unit' ) ) {
2016-12-05 08:40:27 +08:00
testSubtasks = testSubtasks . concat ( [ 'dist' , 'docs-css' , 'docs-js' , 'clean:docs' , 'copy:docs' ] ) ;
2016-06-28 05:19:10 +08:00
// 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 15:39:35 +08:00
}
grunt . registerTask ( 'test' , testSubtasks ) ;
2013-05-04 22:55:52 +08:00
2013-08-04 07:03:38 +08:00
// JS distribution task.
2016-06-11 00:28:03 +08:00
grunt . registerTask ( 'dist-js' , [ 'babel:dev' , 'concat' , 'babel:dist' , 'stamp' , 'uglify:core' ] ) ;
2013-05-04 22:55:52 +08:00
2016-01-04 10:09:09 +08:00
grunt . registerTask ( 'test-scss' , [ 'scsslint:core' ] ) ;
2014-12-09 11:02:25 +08:00
2013-08-04 07:03:38 +08:00
// CSS distribution task.
2014-12-16 02:18:27 +08: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 12:45:08 +08:00
// grunt.registerTask('sass-compile', ['sass:core', 'sass:extras', 'sass:docs']);
2016-09-12 13:29:45 +08:00
grunt . registerTask ( 'sass-compile' , [ 'sass:core' , 'sass:extras' , 'sass:docs' ] ) ;
2014-12-16 02:18:27 +08:00
2016-06-21 07:18:21 +08:00
grunt . registerTask ( 'dist-css' , [ 'sass-compile' , 'exec:postcss' , 'cssmin:core' , 'cssmin:docs' ] ) ;
2013-08-18 15:36:51 +08:00
2013-08-04 07:03:38 +08:00
// Full distribution task.
2014-12-01 12:29:47 +08:00
grunt . registerTask ( 'dist' , [ 'clean:dist' , 'dist-css' , 'dist-js' ] ) ;
2013-05-04 22:55:52 +08:00
2013-08-04 07:03:38 +08:00
// Default task.
2014-12-01 12:29:47 +08:00
grunt . registerTask ( 'default' , [ 'clean:dist' , 'test' ] ) ;
2013-08-08 14:06:29 +08:00
2014-08-14 08:47:16 +08:00
// Docs task.
2016-06-21 07:18:21 +08:00
grunt . registerTask ( 'docs-css' , [ 'cssmin:docs' , 'exec:postcss-docs' ] ) ;
2016-01-04 16:22:03 +08:00
grunt . registerTask ( 'lint-docs-css' , [ 'scsslint:docs' ] ) ;
2014-12-11 05:51:43 +08:00
grunt . registerTask ( 'docs-js' , [ 'uglify:docsJs' ] ) ;
2016-06-05 08:46:36 +08:00
grunt . registerTask ( 'docs' , [ 'lint-docs-css' , 'docs-css' , 'docs-js' , 'clean:docs' , 'copy:docs' ] ) ;
2015-12-09 10:25:11 +08:00
grunt . registerTask ( 'docs-github' , [ 'jekyll:github' ] ) ;
2014-08-14 08:47:16 +08:00
2015-12-07 04:11:10 +08:00
grunt . registerTask ( 'prep-release' , [ 'dist' , 'docs' , 'docs-github' , 'compress' ] ) ;
2014-11-14 21:00:02 +08:00
2015-08-19 13:49:26 +08:00
// Publish to GitHub
grunt . registerTask ( 'publish' , [ 'buildcontrol:pages' ] ) ;
2013-09-18 14:04:09 +08:00
} ;