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