[关闭]
@ranger-01 2016-01-18T20:19:39.000000Z 字数 9100 阅读 732

Grunt

Grunt


Grunt

Why use a task runner?

In one word: automation. The less work you have to do when performing repetitive tasks like minification, compilation, unit testing, linting, etc, the easier your job becomes. After you've configured it through a Gruntfile, a task runner can do most of that mundane work for you—and your team—with basically zero effort.

Why use Grunt?

The Grunt ecosystem is huge and it's growing every day. With literally hundreds of plugins to choose from, you can use Grunt to automate just about anything with a minimum of effort. If someone hasn't already built what you need, authoring and publishing your own Grunt plugin to npm is a breeze. See how to get started.

package.jeson

What does this mean for you? Well, first you should understand the difference between the two. In the simplest terms, the tilde matches the most recent minor version (the middle number). ~1.2.3 will match all 1.2.x versions but will miss 1.3.0. This has been the default behavior of ‘–save’ since the start, and you are probably already comfortable seeing it in your package.json. The caret, on the other hand, is more relaxed. It will update you to the most recent major version (the first number). ^1.2.3 will match any 1.x.x release including 1.3.0, but will hold off on 2.0.0.
https://github.com/npm/node-semver#caret-ranges-123-025-004

Gruntfile.js

  1. wrapper function

    module.exports = function(grunt) {
    }
  2. configuration
    The configuration object for a task lives as a property on the configuration object, that's named the same as the task. So the "concat" task goes in our config object under the "concat" key.

    grunt.initConfig({
    });
  3. the entire Gruntfile.js

    1. module.exports = function(grunt) {
    2. grunt.initConfig({
    3. pkg: grunt.file.readJSON('package.json'),
    4. concat: {
    5. options: {
    6. separator: ';'
    7. },
    8. dist: {
    9. src: ['src/**/*.js'],
    10. dest: 'dist/<%= pkg.name %>.js'
    11. }
    12. },
    13. uglify: {
    14. options: {
    15. banner: '/*! <%= pkg.name %> <%= grunt.template.today("dd-mm-yyyy") %> */\n'
    16. },
    17. dist: {
    18. files: {
    19. 'dist/<%= pkg.name %>.min.js': ['<%= concat.dist.dest %>']
    20. }
    21. }
    22. },
    23. qunit: {
    24. files: ['test/**/*.html']
    25. },
    26. jshint: {
    27. files: ['Gruntfile.js', 'src/**/*.js', 'test/**/*.js'],
    28. options: {
    29. // options here to override JSHint defaults
    30. globals: {
    31. jQuery: true,
    32. console: true,
    33. module: true,
    34. document: true
    35. }
    36. }
    37. },
    38. watch: {
    39. files: ['<%= jshint.files %>'],
    40. tasks: ['jshint', 'qunit']
    41. }
    42. });
    43. grunt.loadNpmTasks('grunt-contrib-uglify');
    44. grunt.loadNpmTasks('grunt-contrib-jshint');
    45. grunt.loadNpmTasks('grunt-contrib-qunit');
    46. grunt.loadNpmTasks('grunt-contrib-watch');
    47. grunt.loadNpmTasks('grunt-contrib-concat');
    48. grunt.registerTask('test', ['jshint', 'qunit']);
    49. grunt.registerTask('default', ['jshint', 'qunit', 'concat', 'uglify']);
    50. };

Plugin

wiredep

  1. Inject Bower packages into your source code with Grunt.
  2. This plug-in uses wiredep, which takes a look at all of the components you have, then determines the best order to inject your scripts in to your HTML file.
    Putting script tags that aren't managed by grunt-wiredep is not advised, as anything between <!-- bower:js --> and <!-- endbower --> will be overwritten with each command.
  3. example:
  1. For JavaScript dependencies, pop this in your HTML file:
  2. <!-- bower:js -->
  3. <!-- endbower -->
  4. Install a Bower component:
  5. bower install jquery --save
  6. Call the Grunt task:
  7. grunt wiredep
  8. You're in business!
  9. <!-- bower:js -->
  10. <script src="bower_components/jquery/jquery.js"></script>
  11. <!-- endbower -->

useminPrepare

$ npm install grunt-usemin --save-dev

  1. introduction
    usemin replaces the references of scripts, stylesheets and other assets within HTML files dynamically with optimized versions of them. To do this usemin exports 2 built-in tasks called useminPrepare and usemin and utilizes a couple of other Grunt plugins for the optimization process. usemin does this by generating the subtasks for these Grunt plugins dynamically.

  2. The built-in tasks of usemin:

    • useminPrepare:
      prepares the configuration to transform specific blocks in the scrutinized file into a single line, targeting an optimized version of the files. This is done by generating subtasks called generated for each of the optimization steps handled by the Grunt plugins listed below.

    • usemin:
      replaces the blocks by the file they reference, and replaces all references to assets by their revisioned version if it is found on the disk. This target modifies the files it is working on.

    • Grunt plugins which usemin can use to optimize files:

      $ npm install grunt-contrib-concat grunt-contrib-uglify grunt-contrib-cssmin grunt-filerev --save-dev

      • concat concatenates files (usually JS or CSS).
      • uglify minifies JS files.
      • cssmin minifies CSS files.
      • filerev revisions static assets through a file content hash.
  3. The useminPrepare task
    Internally, the task parses your HTML markup to find each of these blocks, and initializes the corresponding Grunt config for the concat / uglify tasks when type=js, the concat / cssmin tasks when type=css.

    1. <!-- build:js js/app.js -->
    2. <script src="js/app.js"></script>
    3. <script src="js/controllers/thing-controller.js"></script>
    4. <script src="js/models/thing-model.js"></script>
    5. <script src="js/views/thing-view.js"></script>
    6. <!-- endbuild -->
    7. The produced configuration will look like:
    8. {
    9. concat: {
    10. generated: {
    11. files: [
    12. {
    13. dest: '.tmp/concat/js/app.js',
    14. src: [
    15. 'app/js/app.js',
    16. 'app/js/controllers/thing-controller.js',
    17. 'app/js/models/thing-model.js',
    18. 'app/js/views/thing-view.js'
    19. ]
    20. }
    21. ]
    22. }
    23. },
    24. uglify: {
    25. generated: {
    26. files: [
    27. {
    28. dest: 'dist/js/app.js',
    29. src: [ '.tmp/concat/js/app.js' ]
    30. }
    31. ]
    32. }
    33. }
    34. }
  4. The usemin task
    The usemin task has 2 actions:

    • First it replaces all the blocks with a single "summary" line, pointing to a file creating by the transformation flow.
    • Then it looks for references to assets (i.e. images, scripts, ...), and tries to replace them with their revved version if it can find one on disk
    1. usemin: {
    2. js: '*.js',
    3. options: {
    4. assetsDirs: 'images',
    5. patterns: {
    6. js: [
    7. [/(image\.png)/, 'Replacing reference to image.png']
    8. ]
    9. }
    10. }
    11. }
  5. summary

    • useminPrepare
      useminPrepare is trying to prepare the right configuration for the pipeline of actions that are going to be applied on the blocks (for example concatenation and uglify-cation). As such it needs to have the input directory, temporary directories (staging) and destination directory. The files referenced in the block are either absolute or relative (/images/foo.png or ../../images/foo.png). Absolute files references are looked in a given set of search path (input), which by default is set to the directory where the html/css file examined is located (can be overridden per block, or more generally through root option). Relative files references are also looked at from location of the examined file, unless stated otherwise.

    • usemin
      usemin target replaces references to images, scripts, css, ... in the furnished files (html, css, ...). These references may be either absolute (i.e. /images/foo.png) or relative (i.e. image/foo.png or ../images/foo.png). When the reference is absolute a set of asset search paths should be looked at under the destination directory (for example, using the previous example, and searchpath equal to ['assets'], usemin would try to find either a revved version of the image of the image below the assets directory: for example dest/assets/images/foo.1223443.png). When the reference is relative, by default the referenced item is looked in the path relative to the current file location in the destination directory (e.g. with the preceding example, if the file is build/bar/index.html, then transformed index.html will be in dist/bar, and usemin will look for dist/bar/../images/foo.32323.png).

concurrent

Run grunt tasks concurrently

autoprefixer

This project has been deprecated in favour of grunt-postcss.
Apply several post-processors to your CSS using PostCSS.

ngtemplates

  1. introduction
    Speed up your AngularJS app by automatically minifying, combining, and automatically caching your HTML templates with $templateCache.

    Here's an example of the output created by this task from multiple .html files:

    1. angular.module('app').run(["$templateCache", function($templateCache) {
    2. $templateCache.put("home.html",
    3. // contents for home.html ...
    4. );
    5. ...
    6. $templateCache.put("src/app/templates/button.html",
    7. // contents for button.html
    8. );
    9. }]);

    Then, when you use ng-include or templateUrl with $routeProvider, the template is already loaded without an extra AJAX request!

  2. example
    Using grunt-usemin

    usemin
    Path to usemin target
    This should be the output path of the compiled JS indicated in your HTML, such as path/to/output.js shown here.

Using the following HTML as an example:

  1. <!-- build:js dist/vendors.js -->
  2. <script src="bower_components/angular/angular.js"></script>
  3. <script src="bower_components/angular-resource/angular-resource.js"></script>
  4. <!-- endbuild -->

Do not use the concat option, even though grunt-usemin generates a concat.generated object behind the scenes. Instead, use the usemin option to indicate the anticipated output filepath from grunt-usemin.

  1. ngtemplates: {
  2. app: {
  3. src: '**.html',
  4. dest: 'template.js',
  5. options: {
  6. usemin: 'dist/vendors.js' // <~~ This came from the <!-- build:js --> block
  7. }
  8. }
  9. }

concat

ngAnnotate

Add, remove and rebuild AngularJS dependency injection annotations.

  1. grunt.initConfig({
  2. ngAnnotate: {
  3. options: {
  4. singleQuotes: true,
  5. },
  6. app1: {
  7. files: {
  8. 'a.js': ['a.js'],
  9. 'c.js': ['b.js'],
  10. 'f.js': ['d.js', 'e.js'],
  11. },
  12. },
  13. app2: {
  14. files: [
  15. {
  16. expand: true,
  17. src: ['f.js'],
  18. ext: '.annotated.js', // Dest filepaths will have this extension.
  19. extDot: 'last', // Extensions in filenames begin after the last dot
  20. },
  21. ],
  22. },
  23. app3: {
  24. files: [
  25. {
  26. expand: true,
  27. src: ['g.js'],
  28. rename: function (dest, src) { return src + '-annotated'; },
  29. },
  30. ],
  31. },
  32. },
  33. });

After executing grunt ngAnnotate, you'll get file a.js annotated and saved under the same name, file b.js annotated and saved as c.js and files d.js and e.js concatenated, annotated and saved as f.js. Annotations will be saved using single quotes.
An annotated version of the f.js file will be saved as f.annotated.js and an annotated version of the g.js file will be saved as g.js-annotated.

copy

cssmin

uglify

filerev

This task will revision your files based on its contents. You should then set the files to expire far into the future for better caching and it will only update when it changes.

  1. grunt.initConfig({
  2. filerev: {
  3. options: {
  4. algorithm: 'md5',
  5. length: 8
  6. },
  7. images: {
  8. src: 'img/**/*.{jpg,jpeg,gif,png,webp}'
  9. }
  10. },
  11. })

usemin

htmlmin

添加新批注
在作者公开此批注前,只有你和作者可见。
回复批注