2015-11-04 23:49:03 +03:00
module . exports = function ( grunt ) {
2015-11-26 01:01:18 +03:00
'use strict' ;
2014-10-02 13:36:05 +04:00
2015-11-26 01:01:18 +03:00
// Force use of Unix newlines
grunt . util . linefeed = '\n' ;
2014-10-05 11:32:48 +04:00
2015-11-26 01:01:18 +03:00
RegExp . quote = function ( string ) {
return string . replace ( /[-\\^$*+?.()|[\]{}]/g , '\\$&' ) ;
} ;
var fs = require ( 'fs' ) ;
2015-11-20 23:24:39 +03:00
var path = require ( 'path' ) ;
2015-11-26 01:01:18 +03:00
var glob = require ( 'glob' ) ;
var isTravis = require ( 'is-travis' ) ;
var npmShrinkwrap = require ( 'npm-shrinkwrap' ) ;
var mq4HoverShim = require ( 'mq4-hover-shim' ) ;
var autoprefixer = require ( 'autoprefixer' ) ( {
browsers : [
'Android 2.3' ,
'Android >= 4' ,
'Chrome >= 35' ,
'Firefox >= 31' ,
// Note: Edge versions in Autoprefixer & Can I Use refer to the EdgeHTML rendering engine version,
// NOT the Edge app version shown in Edge's "About" screen.
// For example, at the time of writing, Edge 20 on an up-to-date system uses EdgeHTML 12.
// See also https://github.com/Fyrd/caniuse/issues/1928
'Edge >= 12' ,
'Explorer >= 9' ,
'iOS >= 7' ,
'Opera >= 12' ,
'Safari >= 7.1'
]
} ) ;
var generateCommonJSModule = require ( './grunt/bs-commonjs-generator.js' ) ;
var configBridge = grunt . file . readJSON ( './grunt/configBridge.json' , { encoding : 'utf8' } ) ;
2015-11-20 23:24:39 +03:00
Object . keys ( configBridge . paths ) . forEach ( function ( key ) {
configBridge . paths [ key ] . forEach ( function ( val , i , arr ) {
arr [ i ] = path . join ( './docs/assets' , val ) ;
} ) ;
} ) ;
2015-11-26 01:01:18 +03:00
// Project configuration.
2014-11-26 15:28:22 +03:00
grunt . initConfig ( {
2014-10-03 23:15:31 +04:00
2015-11-26 01:01:18 +03:00
// Metadata.
2015-11-21 00:17:08 +03:00
pkg : grunt . file . readJSON ( 'package.json' ) ,
2015-11-26 01:01:18 +03:00
banner : '/*!\n' +
' * Bootstrap Material Design v<%= pkg.version %> (<%= pkg.homepage %>)\n' +
' * Copyright 2014-<%= grunt.template.today("yyyy") %> <%= pkg.author %>\n' +
' * Licensed under MIT (https://github.com/FezVrasta/bootstrap-material-design/blob/master/LICENSE)\n' +
' */\n' ,
jqueryCheck : 'if (typeof jQuery === \'undefined\') {\n' +
' throw new Error(\'Bootstrap Material Design\\\'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) || (version[0] >= 3)) {\n' +
' throw new Error(\'Bootstrap Material Design\\\'s JavaScript requires at least jQuery v1.9.1 but less than v3.0.0\')\n' +
' }\n' +
'}(jQuery);\n\n' ,
2015-11-20 23:24:39 +03:00
2015-11-20 21:21:02 +03:00
// Task configuration.
clean : {
dist : 'dist' ,
docs : 'docs/dist'
} ,
2015-11-26 01:01:18 +03:00
// JS build configuration
lineremover : {
es6Import : {
files : {
'<%= concat.bootstrap.dest %>' : '<%= concat.bootstrap.dest %>'
} ,
2015-11-20 00:58:21 +03:00
options : {
2015-11-26 01:01:18 +03:00
exclusionPattern : /^(import|export)/g
2015-11-20 00:58:21 +03:00
}
}
} ,
2015-11-26 01:01:18 +03:00
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'
}
} ,
2015-11-21 00:17:08 +03:00
dist : {
options : {
2015-11-26 01:01:18 +03:00
modules : 'ignore'
2015-11-21 00:17:08 +03:00
} ,
2015-11-26 01:01:18 +03:00
files : {
'<%= concat.bootstrap.dest %>' : '<%= concat.bootstrap.dest %>'
}
} ,
umd : {
2015-11-07 02:41:39 +03:00
options : {
2015-11-26 01:01:18 +03:00
modules : 'umd'
2015-11-07 02:41:39 +03:00
} ,
2015-11-26 01:01:18 +03:00
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'
}
2015-11-07 02:41:39 +03:00
}
} ,
2015-11-26 01:01:18 +03:00
eslint : {
2015-11-07 02:41:39 +03:00
options : {
2015-11-26 01:01:18 +03:00
configFile : 'js/.eslintrc'
2015-11-07 02:41:39 +03:00
} ,
2015-11-26 01:01:18 +03:00
target : 'js/src/*.js'
2015-11-07 02:41:39 +03:00
} ,
2015-11-20 00:58:21 +03:00
2015-11-26 01:01:18 +03:00
jscs : {
options : {
config : 'js/.jscsrc'
} ,
grunt : {
src : [ 'Gruntfile.js' , 'grunt/*.js' ]
} ,
core : {
src : 'js/src/*.js'
} ,
test : {
src : 'js/tests/unit/*.js'
} ,
assets : {
2015-11-03 04:17:00 +03:00
options : {
2015-11-26 01:01:18 +03:00
requireCamelCaseOrUpperCaseIdentifiers : null
} ,
src : [ 'docs/assets/js/src/*.js' , 'docs/assets/js/*.js' , '!docs/assets/js/*.min.js' ]
}
} ,
2015-11-05 01:54:57 +03:00
2015-11-26 01:01:18 +03:00
stamp : {
options : {
banner : '<%= banner %>\n<%= jqueryCheck %>\n<%= jqueryVersionCheck %>\n+function ($) {\n' ,
footer : '\n}(jQuery);'
} ,
bootstrap : {
files : {
src : '<%= concat.bootstrap.dest %>'
2015-11-03 04:17:00 +03:00
}
2015-11-02 22:42:29 +03:00
}
} ,
2014-12-04 15:17:45 +03:00
2015-11-26 01:01:18 +03:00
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'
2015-11-03 02:55:59 +03:00
}
} ,
2015-11-26 01:01:18 +03:00
uglify : {
options : {
compress : {
warnings : false
2014-10-03 23:15:31 +04:00
} ,
2015-11-26 01:01:18 +03:00
mangle : true ,
preserveComments : /^!|@preserve|@license|@cc_on/i
2015-03-09 19:10:44 +03:00
} ,
2015-11-26 01:01:18 +03:00
core : {
src : '<%= concat.bootstrap.dest %>' ,
dest : 'dist/js/<%= pkg.name %>.min.js'
2015-11-21 01:18:58 +03:00
} ,
2015-11-26 01:01:18 +03:00
docsJs : {
src : configBridge . paths . docsJs ,
dest : 'docs/assets/js/docs.min.js'
2014-11-26 15:28:22 +03:00
}
} ,
2015-11-26 01:01:18 +03:00
qunit : {
2014-11-26 15:28:22 +03:00
options : {
2015-11-26 01:01:18 +03:00
inject : 'js/tests/unit/phantom.js'
2014-11-26 15:28:22 +03:00
} ,
2015-11-26 01:01:18 +03:00
files : 'js/tests/index.html'
} ,
// CSS build configuration
scsslint : {
options : {
bundleExec : true ,
config : 'scss/.scss-lint.yml' ,
reporterOutput : null
2015-03-09 19:10:44 +03:00
} ,
2015-11-26 01:01:18 +03:00
src : [ 'scss/*.scss' , '!scss/_normalize.scss' ]
} ,
postcss : {
core : {
options : {
map : true ,
processors : [
mq4HoverShim . postprocessorFor ( { hoverSelectorPrefix : '.bs-true-hover ' } ) ,
autoprefixer
]
} ,
src : 'dist/css/*.css'
2015-11-20 23:24:39 +03:00
} ,
docs : {
2015-11-26 01:01:18 +03:00
options : {
processors : [
autoprefixer
]
} ,
src : 'docs/assets/css/docs.min.css'
2015-11-20 23:24:39 +03:00
} ,
examples : {
2015-11-26 01:01:18 +03:00
options : {
processors : [
autoprefixer
]
} ,
2015-11-20 23:24:39 +03:00
expand : true ,
cwd : 'docs/examples/' ,
src : [ '**/*.css' ] ,
dest : 'docs/examples/'
}
} ,
2015-11-20 21:21:02 +03:00
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
2015-11-26 01:01:18 +03:00
compatibility : 'ie9' ,
2015-11-20 21:21:02 +03:00
keepSpecialComments : '*' ,
sourceMap : true ,
2015-11-26 01:01:18 +03:00
noAdvanced : true
2015-03-09 19:10:44 +03:00
} ,
2015-11-26 01:01:18 +03:00
core : {
files : [
{
expand : true ,
cwd : 'dist/css' ,
src : [ '*.css' , '!*.min.css' ] ,
dest : 'dist/css' ,
ext : '.min.css'
}
]
2015-11-20 21:21:02 +03:00
} ,
docs : {
2015-11-26 01:01:18 +03:00
src : 'docs/assets/css/docs.min.css' ,
2015-11-20 21:21:02 +03:00
dest : 'docs/assets/css/docs.min.css'
2014-12-04 15:30:23 +03:00
}
} ,
2015-11-26 01:01:18 +03:00
csscomb : {
options : {
config : 'scss/.csscomb.json'
2014-12-04 15:17:45 +03:00
} ,
2015-11-26 01:01:18 +03:00
dist : {
expand : true ,
cwd : 'dist/css/' ,
src : [ '*.css' , '!*.min.css' ] ,
dest : 'dist/css/'
2014-11-26 15:28:22 +03:00
} ,
2015-11-26 01:01:18 +03:00
examples : {
2014-11-26 15:28:22 +03:00
expand : true ,
2015-11-26 01:01:18 +03:00
cwd : 'docs/examples/' ,
src : '**/*.css' ,
dest : 'docs/examples/'
2015-11-20 23:24:39 +03:00
} ,
2015-11-26 01:01:18 +03:00
docs : {
src : 'docs/assets/css/src/docs.css' ,
dest : 'docs/assets/css/src/docs.css'
}
} ,
copy : {
2015-11-20 23:24:39 +03:00
docs : {
expand : true ,
cwd : 'dist/' ,
src : [
'**/*'
] ,
dest : 'docs/dist/'
2014-11-26 15:28:22 +03:00
}
} ,
2015-11-26 01:01:18 +03:00
connect : {
server : {
options : {
port : 3000 ,
base : '.'
2014-12-04 15:17:45 +03:00
}
}
} ,
2015-11-26 01:01:18 +03:00
jekyll : {
2014-12-05 11:25:06 +03:00
options : {
2015-11-26 01:01:18 +03:00
bundleExec : true ,
config : '_config.yml' ,
incremental : false
2014-12-05 11:25:06 +03:00
} ,
2015-11-26 01:01:18 +03:00
docs : { } ,
github : {
2014-12-05 11:25:06 +03:00
options : {
2015-11-26 01:01:18 +03:00
raw : 'github: true'
2014-12-05 11:25:06 +03:00
}
}
} ,
2015-11-26 01:01:18 +03:00
htmllint : {
2014-11-26 15:28:22 +03:00
options : {
2015-11-26 01:01:18 +03:00
ignore : [
'Element “img” is missing required attribute “src”.' ,
'Attribute “autocomplete” is only allowed when the input type is “color”, “date”, “datetime”, “datetime-local”, “email”, “month”, “number”, “password”, “range”, “search”, “tel”, “text”, “time”, “url”, or “week”.' ,
'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).' ,
'The “datetime” input type is not supported in all browsers. Please be sure to test, and consider using a polyfill.'
2014-11-26 15:28:22 +03:00
]
2015-11-20 23:24:39 +03:00
} ,
2015-11-26 01:01:18 +03:00
src : [ '_gh_pages/**/*.html' , 'js/tests/visual/*.html' ]
2014-11-26 15:28:22 +03:00
} ,
2015-11-20 23:24:39 +03:00
2015-11-26 01:01:18 +03:00
watch : {
src : {
files : '<%= jscs.core.src %>' ,
tasks : [ 'babel:dev' ]
2015-11-20 23:24:39 +03:00
} ,
2015-11-26 01:01:18 +03:00
sass : {
files : 'scss/**/*.scss' ,
tasks : [ 'dist-css' , 'docs' ]
2015-11-20 23:24:39 +03:00
} ,
2015-11-26 01:01:18 +03:00
docs : {
files : 'docs/assets/scss/**/*.scss' ,
tasks : [ 'dist-css' , 'docs' ]
2015-11-20 23:24:39 +03:00
}
} ,
2015-11-26 01:01:18 +03:00
sed : {
versionNumber : {
pattern : ( function ( ) {
var old = grunt . option ( 'oldver' ) ;
return old ? RegExp . quote ( old ) : old ;
} ) ( ) ,
replacement : grunt . option ( 'newver' ) ,
recursive : true
2014-11-26 15:28:22 +03:00
}
2014-12-15 18:16:33 +03:00
} ,
2015-11-21 00:17:08 +03:00
2015-11-26 01:01:18 +03:00
'saucelabs-qunit' : {
all : {
2015-11-21 00:17:08 +03:00
options : {
2015-11-26 01:01:18 +03:00
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' )
}
2015-11-21 00:17:08 +03:00
}
} ,
2014-12-15 18:16:33 +03:00
exec : {
2015-11-26 01:01:18 +03:00
npmUpdate : {
command : 'npm update'
}
} ,
buildcontrol : {
options : {
dir : '_gh_pages' ,
commit : true ,
push : true ,
message : 'Built %sourceName% from commit %sourceCommit% on branch %sourceBranch%'
2014-12-15 18:16:33 +03:00
} ,
2015-11-26 01:01:18 +03:00
pages : {
options : {
remote : 'git@github.com:twbs/derpstrap.git' ,
branch : 'gh-pages'
}
2014-12-15 18:16:33 +03:00
}
2014-11-26 15:28:22 +03:00
}
} ) ;
2015-11-20 00:58:21 +03:00
2015-11-26 01:01:18 +03:00
// 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 ) ;
2015-11-20 23:24:39 +03:00
2015-11-20 00:58:21 +03:00
// Docs HTML validation task
grunt . registerTask ( 'validate-html' , [ 'jekyll:docs' , 'htmllint' ] ) ;
2015-11-26 01:01:18 +03:00
var runSubset = function ( subset ) {
return ! process . env . BMD _TEST || process . env . BMD _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' ) &&
isTravis &&
// Skip HTML5 validator when [skip validator] is in the commit message
isUndefOrNonZero ( process . env . BMD _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 . BMD _DO _SAUCE ) ) {
testSubtasks . push ( 'babel:dev' ) ;
testSubtasks . push ( 'connect' ) ;
testSubtasks . push ( 'saucelabs-qunit' ) ;
}
grunt . registerTask ( 'test' , testSubtasks ) ;
grunt . registerTask ( 'test-js' , [ 'eslint' , 'jscs:core' , 'jscs:test' , 'jscs:grunt' , 'qunit' ] ) ;
// JS distribution task.
grunt . registerTask ( 'dist-js' , [ 'babel:dev' , 'concat' , 'lineremover' , 'babel:dist' , 'stamp' , 'uglify:core' , 'commonjs' ] ) ;
grunt . registerTask ( 'test-scss' , [ 'scsslint' ] ) ;
// CSS distribution task.
// Supported Compilers: sass (Ruby) and libsass.
( function ( sassCompilerName ) {
require ( './grunt/bs-sass-compile/' + sassCompilerName + '.js' ) ( grunt ) ;
} ) ( process . env . BMD _SASS || 'libsass' ) ;
// grunt.registerTask('sass-compile', ['sass:core', 'sass:extras', 'sass:docs']);
grunt . registerTask ( 'sass-compile' , [ 'sass:core' , 'sass:docs' ] ) ;
grunt . registerTask ( 'dist-css' , [ 'sass-compile' , 'postcss:core' , 'csscomb:dist' , 'cssmin:core' , 'cssmin:docs' ] ) ;
// Full distribution task.
grunt . registerTask ( 'dist' , [ 'clean:dist' , 'dist-css' , 'dist-js' ] ) ;
2014-11-26 15:28:22 +03:00
2015-11-20 21:21:02 +03:00
// Default task.
2015-11-26 01:01:18 +03:00
grunt . registerTask ( 'default' , [ 'clean:dist' , 'test' ] ) ;
2014-11-26 15:28:22 +03:00
2015-11-26 01:01:18 +03: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!
grunt . registerTask ( 'change-version-number' , 'sed' ) ;
2014-10-03 23:15:31 +04:00
2015-11-26 01:01:18 +03:00
grunt . registerTask ( 'commonjs' , [ 'babel:umd' , 'npm-js' ] ) ;
2014-12-15 18:16:33 +03:00
2015-11-26 01:01:18 +03:00
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 ) ;
} ) ;
2015-11-20 23:24:39 +03:00
// Docs task.
2015-11-26 01:01:18 +03:00
grunt . registerTask ( 'docs-css' , [ 'postcss:docs' , 'postcss: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 ( 'prep-release' , [ 'dist' , 'docs' , 'jekyll:github' , 'htmlmin' , 'compress' ] ) ;
// Publish to GitHub
grunt . registerTask ( 'publish' , [ 'buildcontrol:pages' ] ) ;
// 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 = 'grunt/npm-shrinkwrap.json' ;
fs . renameSync ( 'npm-shrinkwrap.json' , dest ) ;
grunt . log . writeln ( 'File ' + dest . cyan + ' updated.' ) ;
done ( ) ;
} ) ;
} ) ;
2014-10-02 13:36:05 +04:00
} ;