Как заказать вложенные файлы с помощью Gulp?

My Angular приложение имеет следующую структуру

- src/app/main/
    |
    -main.js
    -main.controller.js
    -index.html

Я использую Gulp. По завершении сборки файлы *.js вводятся в неправильном порядке в index.html. Файл main.js зависит от main.controller.js, поэтому main.controller.js должен быть введен до main.js.

<!-- build:js({.tmp/serve,.tmp/partials,src}) scripts/app.js -->
<!-- inject:js -->
<script src="app/main/main.js"></script>
<script src="app/main/main.controller.js"></script>
<script src="app/index.js"></script>
<!-- endinject -->

Это мой gulpfile.js

'use strict';

var gulp = require('gulp');
var gutil = require('gulp-util');
var wrench = require('wrench');

var options = {
  src: 'src',
  dist: 'dist',
  tmp: '.tmp',
  e2e: 'e2e',
  errorHandler: function(title) {
    return function(err) {
      gutil.log(gutil.colors.red('[' + title + ']'), err.toString());
      this.emit('end');
    };
  },
  wiredep: {
    directory: 'bower_components',
    exclude: [/jquery/, /bootstrap-sass-official\/.*\.js/, /bootstrap\.css/]
  }
};

wrench.readdirSyncRecursive('./gulp').filter(function(file) {
  return (/\.(js|coffee)$/i).test(file);
}).map(function(file) {
  require('./gulp/' + file)(options);
});

gulp.task('default', ['clean'], function () {
    gulp.start('build');
});

Ответ 1

Используйте gulp-angular-filesort в файле inject.js

https://www.npmjs.com/package/gulp-angular-filesort

Чтобы работать корректно, каждый файл angular должен иметь уникальное имя синтаксис модуля и сеттера (с помощью скобок), т.е. angular.module('myModule', []).

Пример inject.js:

module.exports = function(options) {
  gulp.task('inject', ['scripts', 'styles'], function () {
    var injectStyles = gulp.src([
      options.tmp + '/serve/app/**/*.css',
      '!' + options.tmp + '/serve/app/vendor.css'
    ], { read: false });

    var injectScripts = gulp.src([
      options.src + '/app/**/*.js',
      '!' + options.src + '/app/**/*.spec.js',
      '!' + options.src + '/app/**/*.mock.js'
    ])
    .pipe($.angularFilesort()).on('error', options.errorHandler('AngularFilesort'));

    var injectOptions = {
      ignorePath: [options.src, options.tmp + '/serve'],
      addRootSlash: false
    };

    return gulp.src(options.src + '/*.html')
      .pipe($.inject(injectStyles, injectOptions))
      .pipe($.inject(injectScripts, injectOptions))
      .pipe(wiredep(options.wiredep))
      .pipe(gulp.dest(options.tmp + '/serve'));

  });
};

Ответ 2

Это позволяет мне сначала определить важные файлы js в исходном массиве Gulp:

var target = gulp.src('app/index.html');
var sources = gulp.src(['app/scripts/core/app.js','app/scripts/**/*.js'], {read: false});
target.pipe(inject(sources, {})).pipe(gulp.dest('app'));

Ответ 3

Я исправил эту проблему, используя gulp-natural-sort и stream-series. gulp-natural-sort упорядочивает ваши файлы в последовательном порядке и stream-series принимает несколько потоков файлов и помещает их друг в друга.

Мой код выглядит следующим образом:

var series = require('stream-series');
var naturalSort = require('gulp-natural-sort');

gulp.task('inject', function() {
    log('Injecting custom scripts to index.html');

    var theseComeFirst = gulp.src(["somepath/first*.js", {read: false}).pipe(naturalSort());

    var theseComeSecond = gulp.src("somepath/second*.js", {read: false}).pipe(naturalSort());

    return gulp
        .src("*.html")
        .pipe($.plumber({errorHandler: handleError}))
        .pipe($.inject(series(theseComeFirst, theseComeSecond), {relative: true}))
        .pipe(gulp.dest('client/'));
});

Вы можете просто использовать серию потоков, если вам все равно, находятся ли другие в порядке или нет. К сожалению, inject() будет помещать файлы в группу серий в несколько случайном порядке каждый раз, когда он запускается, что дает мне jeebees heebee.