1 /* jshint node: true */
3 module.exports = function(grunt) {
6 RegExp.quote = require('regexp-quote')
7 var btoa = require('btoa')
8 // Project configuration.
12 pkg: grunt.file.readJSON('package.json'),
14 ' * Bootstrap v<%= pkg.version %> by @fat and @mdo\n' +
15 ' * Copyright <%= grunt.template.today("yyyy") %> <%= pkg.author %>\n' +
16 ' * Licensed under <%= _.pluck(pkg.licenses, "url").join(", ") %>\n' +
18 ' * Designed and built with all the love in the world by @mdo and @fat.\n' +
20 jqueryCheck: 'if (typeof jQuery === "undefined") { throw new Error("Bootstrap requires jQuery") }\n\n',
22 // Task configuration.
29 jshintrc: 'js/.jshintrc'
38 src: ['js/tests/unit/*.js']
44 banner: '<%= banner %><%= jqueryCheck %>',
62 dest: 'dist/js/<%= pkg.name %>.js'
68 banner: '<%= banner %>',
72 src: ['<%= concat.bootstrap.dest %>'],
73 dest: 'dist/js/<%= pkg.name %>.min.js'
80 banner: '<%= banner %>'
83 src: ['less/bootstrap.less'],
84 dest: 'dist/css/<%= pkg.name %>.css'
90 src: ['less/bootstrap.less'],
91 dest: 'dist/css/<%= pkg.name %>.min.css'
94 src: ['less/theme.less'],
95 dest: 'dist/css/<%= pkg.name %>-theme.css'
101 src: ['less/theme.less'],
102 dest: 'dist/css/<%= pkg.name %>-theme.min.css'
116 inject: 'js/tests/unit/phantom.js'
118 files: ['js/tests/*.html']
138 "Bad value X-UA-Compatible for attribute http-equiv on element meta.",
139 "Element img is missing required attribute src."
143 src: ["_gh_pages/**/*.html"]
149 files: '<%= jshint.src.src %>',
150 tasks: ['jshint:src', 'qunit']
153 files: '<%= jshint.test.src %>',
154 tasks: ['jshint:test', 'qunit']
157 files: 'less/*.less',
164 pattern: (function () {
165 var old = grunt.option('oldver')
166 return old ? RegExp.quote(old) : old
168 replacement: grunt.option('newver'),
175 // These plugins provide necessary tasks.
176 grunt.loadNpmTasks('browserstack-runner');
177 grunt.loadNpmTasks('grunt-contrib-clean');
178 grunt.loadNpmTasks('grunt-contrib-concat');
179 grunt.loadNpmTasks('grunt-contrib-connect');
180 grunt.loadNpmTasks('grunt-contrib-copy');
181 grunt.loadNpmTasks('grunt-contrib-jshint');
182 grunt.loadNpmTasks('grunt-contrib-qunit');
183 grunt.loadNpmTasks('grunt-contrib-uglify');
184 grunt.loadNpmTasks('grunt-contrib-watch');
185 grunt.loadNpmTasks('grunt-html-validation');
186 grunt.loadNpmTasks('grunt-jekyll');
187 grunt.loadNpmTasks('grunt-recess');
188 grunt.loadNpmTasks('grunt-sed');
190 // Docs HTML validation task
191 grunt.registerTask('validate-html', ['jekyll', 'validation']);
194 var testSubtasks = ['dist-css', 'jshint', 'qunit', 'validate-html'];
195 // Only run BrowserStack tests under Travis
196 if (process.env.TRAVIS) {
197 // Only run BrowserStack tests if this is a mainline commit in twbs/bootstrap, or you have your own BrowserStack key
198 if ((process.env.TRAVIS_REPO_SLUG === 'twbs/bootstrap' && process.env.TRAVIS_PULL_REQUEST === 'false') || process.env.TWBS_HAVE_OWN_BROWSERSTACK_KEY) {
199 testSubtasks.push('browserstack_runner');
202 grunt.registerTask('test', testSubtasks);
204 // JS distribution task.
205 grunt.registerTask('dist-js', ['concat', 'uglify']);
207 // CSS distribution task.
208 grunt.registerTask('dist-css', ['recess']);
210 // Fonts distribution task.
211 grunt.registerTask('dist-fonts', ['copy']);
213 // Full distribution task.
214 grunt.registerTask('dist', ['clean', 'dist-css', 'dist-fonts', 'dist-js']);
217 grunt.registerTask('default', ['test', 'dist', 'build-customizer']);
219 // Version numbering task.
220 // grunt change-version-number --oldver=A.B.C --newver=X.Y.Z
221 // This can be overzealous, so its changes should always be manually reviewed!
222 grunt.registerTask('change-version-number', ['sed']);
224 // task for building customizer
225 grunt.registerTask('build-customizer', 'Add scripts/less files to customizer.', function () {
226 var fs = require('fs')
228 function getFiles(type) {
231 .filter(function (path) {
232 return type == 'fonts' ? true : new RegExp('\\.' + type + '$').test(path)
234 .forEach(function (path) {
235 var fullPath = type + '/' + path
236 return files[path] = (type == 'fonts' ? btoa(fs.readFileSync(fullPath)) : fs.readFileSync(fullPath, 'utf8'))
238 return 'var __' + type + ' = ' + JSON.stringify(files) + '\n'
241 var files = getFiles('js') + getFiles('less') + getFiles('fonts')
242 fs.writeFileSync('docs-assets/js/raw-files.js', files)