Yeoman中的工作流是什么,如何处理Sass文件?

39

我尝试使用yeoman,但我不知道如何在它中使用我的自定义sass文件。

通过 grunt server,Sass文件会被监视并编译到 .tmp/styles/ 中,但除了 <link rel="stylesheet" href="styles/main.css"> 之外,没有对已编译样式表的引用。

那么,在开发期间使用已编译的sass文件的推荐方法是什么呢?例如,如果我将我的样式 app/styles/my.sass 更改为 .tmp/styles/my.css,这将被覆盖,而且它在服务器(localhost:9000)之外。因此,无法在 index.html 中链接它。

通过 grunt build,一切都包括 my.sass 在内的内容都在 main.css 中,但在开发期间,我不知道如何在 index.html 中使用自己的sass文件。

你可以给我一些简单的例子吗?

这是默认的yeoman安装。我做了以下操作:

  1. yo angular test
  2. 我添加了 app/styles/style.sass
  3. grunt server:这将 style.sass 编译为 .tmp/styles/style.css
  4. 现在我不知道如何在html中包含 style.css

(很抱歉,也许这是一个愚蠢的问题,但我对yeoman和grunt都很陌生)

这是来自yeoman的Gruntfile.js:

'use strict';
var lrSnippet = require('grunt-contrib-livereload/lib/utils').livereloadSnippet;
var mountFolder = function (connect, dir) {
  return connect.static(require('path').resolve(dir));
};

module.exports = function (grunt) {
  // load all grunt tasks
  require('matchdep').filterDev('grunt-*').forEach(grunt.loadNpmTasks);

  // configurable paths
  var yeomanConfig = {
    app: 'app',
    dist: 'dist'
  };

  try {
    yeomanConfig.app = require('./component.json').appPath || yeomanConfig.app;
  } catch (e) {}

  grunt.initConfig({
    yeoman: yeomanConfig,
    watch: {
      coffee: {
        files: ['<%= yeoman.app %>/scripts/{,*/}*.coffee'],
        tasks: ['coffee:dist']
      },
      coffeeTest: {
        files: ['test/spec/{,*/}*.coffee'],
        tasks: ['coffee:test']
      },
      compass: {
        files: ['<%= yeoman.app %>/styles/{,*/}*.{scss,sass}'],
        tasks: ['compass']
      },
      livereload: {
        files: [
          '<%= yeoman.app %>/{,*/}*.html',
          '{.tmp,<%= yeoman.app %>}/styles/{,*/}*.css',
          '{.tmp,<%= yeoman.app %>}/scripts/{,*/}*.js',
          '<%= yeoman.app %>/images/{,*/}*.{png,jpg,jpeg,gif,webp}'
        ],
        tasks: ['livereload']
      }
    },
    connect: {
      livereload: {
        options: {
          port: 9000,
          // Change this to '0.0.0.0' to access the server from outside.
          hostname: 'localhost',
          middleware: function (connect) {
            return [
              lrSnippet,
              mountFolder(connect, '.tmp'),
              mountFolder(connect, yeomanConfig.app)
            ];
          }
        }
      },
      test: {
        options: {
          port: 9000,
          middleware: function (connect) {
            return [
              mountFolder(connect, '.tmp'),
              mountFolder(connect, 'test')
            ];
          }
        }
      }
    },
    open: {
      server: {
        url: 'http://localhost:<%= connect.livereload.options.port %>'
      }
    },
    clean: {
      dist: ['.tmp', '<%= yeoman.dist %>/*'],
      server: '.tmp'
    },
    jshint: {
      options: {
        jshintrc: '.jshintrc'
      },
      all: [
        'Gruntfile.js',
        '<%= yeoman.app %>/scripts/{,*/}*.js'
      ]
    },
    karma: {
      unit: {
        configFile: 'karma.conf.js',
        singleRun: true
      }
    },
    coffee: {
      dist: {
        files: {
          '.tmp/scripts/coffee.js': '<%= yeoman.app %>/scripts/*.coffee'
        }
      },
      test: {
        files: [{
          expand: true,
          cwd: '.tmp/spec',
          src: '*.coffee',
          dest: 'test/spec'
        }]
      }
    },
    compass: {
      options: {
        sassDir: '<%= yeoman.app %>/styles',
        cssDir: '.tmp/styles',
        imagesDir: '<%= yeoman.app %>/images',
        javascriptsDir: '<%= yeoman.app %>/scripts',
        fontsDir: '<%= yeoman.app %>/styles/fonts',
        importPath: '<%= yeoman.app %>/components',
        relativeAssets: true
      },
      dist: {},
      server: {
        options: {
          debugInfo: true
        }
      }
    },
    concat: {
      dist: {
        files: {
          '<%= yeoman.dist %>/scripts/scripts.js': [
            '.tmp/scripts/{,*/}*.js',
            '<%= yeoman.app %>/scripts/{,*/}*.js'
          ]
        }
      }
    },
    useminPrepare: {
      html: '<%= yeoman.app %>/index.html',
      options: {
        dest: '<%= yeoman.dist %>'
      }
    },
    usemin: {
      html: ['<%= yeoman.dist %>/{,*/}*.html'],
      css: ['<%= yeoman.dist %>/styles/{,*/}*.css'],
      options: {
        dirs: ['<%= yeoman.dist %>']
      }
    },
    imagemin: {
      dist: {
        files: [{
          expand: true,
          cwd: '<%= yeoman.app %>/images',
          src: '{,*/}*.{png,jpg,jpeg}',
          dest: '<%= yeoman.dist %>/images'
        }]
      }
    },
    cssmin: {
      dist: {
        files: {
          '<%= yeoman.dist %>/styles/main.css': [
            '.tmp/styles/{,*/}*.css',
            '<%= yeoman.app %>/styles/{,*/}*.css'
          ]
        }
      }
    },
    htmlmin: {
      dist: {
        options: {
          /*removeCommentsFromCDATA: true,
          // https://github.com/yeoman/grunt-usemin/issues/44
          //collapseWhitespace: true,
          collapseBooleanAttributes: true,
          removeAttributeQuotes: true,
          removeRedundantAttributes: true,
          useShortDoctype: true,
          removeEmptyAttributes: true,
          removeOptionalTags: true*/
        },
        files: [{
          expand: true,
          cwd: '<%= yeoman.app %>',
          src: ['*.html', 'views/*.html'],
          dest: '<%= yeoman.dist %>'
        }]
      }
    },
    cdnify: {
      dist: {
        html: ['<%= yeoman.dist %>/*.html']
      }
    },
    ngmin: {
      dist: {
        files: [{
          expand: true,
          cwd: '<%= yeoman.dist %>/scripts',
          src: '*.js',
          dest: '<%= yeoman.dist %>/scripts'
        }]
      }
    },
    uglify: {
      dist: {
        files: {
          '<%= yeoman.dist %>/scripts/scripts.js': [
            '<%= yeoman.dist %>/scripts/scripts.js'
          ],
        }
      }
    },
    copy: {
      dist: {
        files: [{
          expand: true,
          dot: true,
          cwd: '<%= yeoman.app %>',
          dest: '<%= yeoman.dist %>',
          src: [
            '*.{ico,txt}',
            '.htaccess',
            'components/**/*',
            'images/{,*/}*.{gif,webp}'
          ]
        }]
      }
    }
  });

  grunt.renameTask('regarde', 'watch');
  // remove when mincss task is renamed
  grunt.renameTask('mincss', 'cssmin');

  grunt.registerTask('server', [
    'clean:server',
    'coffee:dist',
    'compass:server',
    'livereload-start',
    'connect:livereload',
    'open',
    'watch'
  ]);

  grunt.registerTask('test', [
    'clean:server',
    'coffee',
    'compass',
    'connect:test',
    'karma'
  ]);

  grunt.registerTask('build', [
    'clean:dist',
    'jshint',
    'test',
    'coffee',
    'compass:dist',
    'useminPrepare',
    'imagemin',
    'cssmin',
    'htmlmin',
    'concat',
    'copy',
    'cdnify',
    'usemin',
    'ngmin',
    'uglify'
  ]);

  grunt.registerTask('default', ['build']);
};

我能看一下你的Gruntfile吗? - Allan Kimmer Jensen
应该在你的问题中提到;-) 回答只用于回答原始问题,而不是评论。 - Allan Kimmer Jensen
抱歉,我已经将问题中的内容添加了。 - MarekLi
3个回答

14

您寻找的内容已经记录在:https://github.com/yeoman/grunt-usemin

只需将您的CSS导入包装在注释块中,类似于处理JavaScript文件的方式即可。

<!-- build:css styles/main.css -->
<link rel="stylesheet" href="styles/base.css">
<link rel="stylesheet" href="styles/modules.css">
<link rel="stylesheet" href="styles/layout.css">
<!-- endbuild -->

确保你的生成器是最新的,并且你的Grunt任务都已设置好。执行这个魔术的是'useminPrepare'


我不明白为什么 useminPrepare 通常在 server 任务中从未被调用。在生产环境中(构建任务)是很重要的,需要理解这一点。 - Andy

6
实际的开发魔力正在发生在
mountFolder(connect, '.tmp'),

这是关于grunt-contrib-watch插件使用connect-livereload中间件的说明。

使用该插件后,本地服务器也会提供.tmp文件夹的内容,因此您可以引用styles/main.css,并得到.tmp/styles/main.css作为返回结果。

useminPrepare通常不在server任务中调用。


1
不错!我从你的回答中理解了grunt工作流程。livereload: { options: { open: true, base: [ '.tmp', '<%= yeoman.app %>' ] } }, - rahmat

5
您只需要像这样引用它:
<link rel="stylesheet" href="styles/style.css">

当运行服务器时,Yeoman / grunt 将知道它应该从 temp 文件夹中获取 sass 文件。


不是默认设置。但是您可以添加一个grunt任务,例如 grunt-text-replace 用于在文本文件中查找/替换。但是为什么您想要在开发期间在index.html中包含不同的sass文件呢? - acme
我不想要它。但是在yeoman默认的grunt build中,所有css和.sass文件都会从app/style合并到dist/styles/main.css中,因此在grunt build之后,除了main.css之外,其他css和sass文件都是不必要的。也许我没有理解yeoman构建过程的意图,因为如果需要grunt-text-replace,在我看来,yeoman的制作者应该在grunt build期间使用它。因此,这就是我上面的问题的另一种表述方式:yeoman是否旨在使用我的自定义css、sass文件? - MarekLi

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接