【问题标题】:Gulp browser-sync only works onceGulp 浏览器同步只工作一次
【发布时间】:2015-04-22 14:41:33
【问题描述】:

我正在我的一个项目中试用 Gulp,我想像以前使用 Grunt watch 一样运行它。意思是,一旦完成所有操作,它就必须查看更少的文件和 js 文件、lint、合并、编译和刷新浏览器。

我设法让它与 gulp-browser-sync 一起工作,但由于某种原因它只能工作一次。我对我的 .less 文件进行了更改,然后浏览器重新加载。然后,第二次更改,它确实编译但没有重新加载发生。

这是日志:

[BS] Serving files from: ./
[09:47:26] Starting 'css-clean'...
[09:47:26] Finished 'css-clean' after 16 ms
[09:47:26] Starting 'styles'...
[BS] 1 file changed (styles.min.css)
[09:47:27] Finished 'styles' after 624 ms
[09:47:27] Starting 'styles-watch'...
[BS] Reloading Browsers...

当我第二次点击保存时:

[09:47:31] Starting 'css-clean'...
[09:47:31] Finished 'css-clean' after 3.39 ms
[09:47:31] Starting 'styles'...
[BS] 1 file changed (styles.min.css)
[09:47:32] Finished 'styles' after 362 ms

至于 JS,它一直都在工作。没有任何问题,即使样式任务完成后,JS 更改仍然会正确触发重新加载。确实只有样式有问题。

这是 gulpfile.js

var gulp = require('gulp'),
    autoprefixer = require('gulp-autoprefixer'),
    less = require('gulp-less'),
    minifycss = require('gulp-minify-css'),
    concat = require('gulp-concat'),
    jshint = require('gulp-jshint'),
    uglify = require('gulp-uglify'),
    rename = require('gulp-rename'),
    notify = require('gulp-notify'),
    path = require('path'),
    streamqueue = require('streamqueue'),
    clean = require('gulp-clean'),
    browserSync = require('browser-sync').create(),
    reload = browserSync.reload;


// Clean the CSS folder
gulp.task('css-clean', function () {
    return gulp.src(['dist/css'], {read: false})
        .pipe(clean());
});

// Clean the CSS folder
gulp.task('js-clean', function () {
    return gulp.src(['dist/js'], {read: false})
        .pipe(clean());
});


// Generate the CSS styles, clean before hand
gulp.task('styles', ['css-clean'], function() {
  return streamqueue({ objectMode: true },
            gulp.src(['./bower_components/uikit/less/uikit.less']),
            gulp.src(['./src/less/main.less'])
        )
    .pipe(concat('styles.less'))
    .pipe(less({
      paths: [ path.join(__dirname, 'less', 'includes') ]
    }))
    .pipe(autoprefixer('last 2 version', 'safari 5', 'ie 8', 'ie 9', 'opera 12.1', 'ios 6', 'android 4'))
    .pipe(gulp.dest('dist/css'))
    .pipe(rename({suffix: '.min'}))
    .pipe(minifycss())
    .pipe(gulp.dest('dist/css'))
    .pipe(browserSync.reload({stream:true}));
});

// create a task that ensures the `styles` task is complete before
// reloading browsers
gulp.task('styles-watch', ['styles'], browserSync.reload);



// Lint Task
gulp.task('lint', function() {
    return gulp.src('src/js/*.js')
        .pipe(jshint())
        .pipe(jshint.reporter('default'));
});


// Generate the scripts, clean before hand
gulp.task('scripts', ['js-clean', 'lint'], function() {
  return streamqueue({ objectMode: true },
            gulp.src(['./bower_components/jquery/dist/jquery.js']),
            gulp.src(['./bower_components/modernizer/modernizr.js']),
            gulp.src(['./bower_components/uikit/js/uikit.js']),
            gulp.src(['./src/js/plugin.js']),
            gulp.src(['./src/js/main.js'])
        )
    .pipe(concat('scripts.js'))
    .pipe(gulp.dest('dist/js'))
    .pipe(rename({suffix: '.min'}))
    .pipe(uglify())
    .pipe(gulp.dest('dist/js'))
    .pipe(browserSync.reload({stream:true}));
});

// create a task that ensures the `scripts` task is complete before
// reloading browsers
gulp.task('scripts-watch', ['scripts'], browserSync.reload);


// Lint Task
gulp.task('alldone', ['scripts'], function() {
    return gulp.src('src/js/*.js')
               .pipe(notify({ message: 'Gulp tasks are completed!!' }));
});


// Static server
gulp.task('browsersync', function() {
    browserSync.init({
        server: {
            baseDir: "./"
        }
    });

    gulp.watch("src/less/*.less", ['styles-watch']);
    gulp.watch('src/js/*.js', ['lint', 'scripts-watch']);
    gulp.watch("*.html").on('change', reload);
});


// Default Task
gulp.task('default', ['css-clean', 'js-clean', 'styles', 'lint', 'scripts', 'alldone']);

// Build Task
gulp.task('build',   ['css-clean', 'js-clean', 'styles', 'lint', 'scripts', 'alldone']);

感谢您的帮助!

【问题讨论】:

    标签: javascript gulp gulp-watch gulp-less


    【解决方案1】:

    对于 Gulp 4 中的我来说,匿名函数并没有解决问题,我所做的是将 browserSync.reload() 包装在一个带有 done 回调的函数中:

    function reload(done) {
      browserSync.reload();
      done();
    }
    
    gulp.watch(['scripts/**/*.js','!scripts/main.min.js'], gulp.series(buildScripts, reload));
    

    【讨论】:

    • 是的!这结束了很多试验和错误。谢谢
    • 这个简单的调整很容易实现,而且效果很好。谢谢保罗
    【解决方案2】:

    直接使用怎么样

    .pipe(browserSync.stream())
    

    我曾经在更改仅重新加载一次的“.jade”文件时遇到同样的问题。然后我将 browserSync.reload 包装在一个匿名函数中,例如

    gulp.task('templates-watch', ['templates'], function() {
        browserSync.reload();
    });
    

    它对我有用。我不知道 browserSync.reload 是否有什么特别之处。不如试一试。 :)

    【讨论】:

    • 匿名函数让一切变得不同! +100 万
    • 那是因为 browserSync.reload 函数需要参数。 github.com/BrowserSync/browser-sync/issues/…
    • 是的,出于某种原因,如果您直接将其用作回调,则它只能工作一次!有人可能应该向 browsersync 报告这一点
    【解决方案3】:

    我知道这不是正确的做事方式,但经过反复试验,我发现在 browserSync.reload() 之后使用未定义的函数

    gulp.watch("./**/*.php", function () {
        browserSync.reload();
        done();
    });
    

    导致我的日志中出现引用错误,触发重新加载浏览器:

    ReferenceError: done is not defined 
        at ...
        at ...
        at asyncRunner (...)
        at processTicksAndRejections (...)
    [Browsersync] Reloading Browsers...
    

    【讨论】:

      【解决方案4】:

      您确定浏览器中的 CSS 没有更改吗?无需重新加载页面即可加载新样式。至少我的 gulp 设置是这样做的。

      尝试更改 .less 文件,看看更改是否在浏览器中实际可见。

      【讨论】:

      • 它没有更新,因为我的测试正在更改导航栏的背景颜色,并且在我第二次自己刷新页面之前我在浏览器中看不到它。
      【解决方案5】:

      我的情况有点不同,但可能相似到足以对您或其他人有所帮助。我有这样的 gulp + browserSync 设置:

      .task('serve', ['clean', 'lint', 'sass', 'js', 'server'], function () {
        return gulp.watch([paths.js, paths.html, paths.sass], 
          ['lint', 'sass', 'js', browserSync.reload]);
      })
      

      这将启动并打开页面一次。如果我对 HTML 页面进行更改并保存它,我可以看到 lint、sass、js 和 browserSync.reload 运行,但浏览器没有刷新。我的 HTML 非常简单;这个:

      <!doctype html>
      <html>
      TEST
      </html>
      

      我终于把 HTML 改成了这个,它开始工作了:

      <!doctype html>
      <html>
      <head>
          <title>TEST</title>
      </head>
      <body>
      HELLO-TEST
      </body>
      </html>
      

      【讨论】:

        【解决方案6】:

        我面临同样的问题。看起来 node_modules 中的“asyncDone”功能存在问题。这是文件路径:/node_modules/glob-watcher/index.js

        以下只是临时修复的肮脏技巧。有时您可能会遇到问题。但对我有用。

        function onChange() {
          console.log("onChange running: ", running);
        
          /** existing code */
        
          // keeping this function at it worked for me.
          setTimeout(() => {
             runComplete({msg: "test"});
          }, 2000);
        }
        

        【讨论】:

          猜你喜欢
          • 1970-01-01
          • 2014-09-28
          • 1970-01-01
          • 1970-01-01
          • 2015-04-15
          • 2017-10-31
          • 1970-01-01
          • 2015-03-11
          • 1970-01-01
          相关资源
          最近更新 更多