mirror of
https://github.com/ionic-team/ionic-framework.git
synced 2025-08-17 02:31:34 +08:00
534 lines
14 KiB
JavaScript
534 lines
14 KiB
JavaScript
var buildConfig = require('./scripts/build/config');
|
|
var gulp = require('gulp');
|
|
var path = require('path');
|
|
var argv = require('yargs').argv;
|
|
var del = require('del');
|
|
var rename = require('gulp-rename');
|
|
var through2 = require('through2');
|
|
var runSequence = require('run-sequence');
|
|
var watch = require('gulp-watch');
|
|
var tsc = require('gulp-typescript');
|
|
var cache = require('gulp-cached');
|
|
var remember = require('gulp-remember');
|
|
var minimist = require('minimist');
|
|
var connect = require('gulp-connect');
|
|
|
|
var flagConfig = {
|
|
string: ['port', 'version', 'ngVersion', 'animations'],
|
|
boolean: ['dry-run'],
|
|
alias: {'p': 'port', 'v': 'version', 'a': 'ngVersion'},
|
|
default: { port: 8000 }
|
|
};
|
|
var flags = minimist(process.argv.slice(2), flagConfig);
|
|
|
|
var tscOptions = {
|
|
emitDecoratorMetadata: true,
|
|
experimentalDecorators: true,
|
|
target: "es5",
|
|
module: "commonjs",
|
|
declaration: true
|
|
}
|
|
|
|
var tscOptionsNoTypeCheck = {
|
|
emitDecoratorMetadata: true,
|
|
experimentalDecorators: true,
|
|
target: "es5",
|
|
module: "commonjs",
|
|
isolatedModules: true
|
|
}
|
|
|
|
var tscOptionsEs6 = {
|
|
emitDecoratorMetadata: true,
|
|
experimentalDecorators: true,
|
|
target: 'ES6',
|
|
isolatedModules: true
|
|
}
|
|
|
|
var tscReporter = {
|
|
error: function (error) {
|
|
// TODO
|
|
// suppress type errors until we convert everything to TS
|
|
// console.error(error.message);
|
|
}
|
|
};
|
|
|
|
// We use Babel to easily create named System.register modules
|
|
// See: https://github.com/Microsoft/TypeScript/issues/4801
|
|
// and https://github.com/ivogabe/gulp-typescript/issues/211
|
|
var babelOptions = {
|
|
modules: 'system',
|
|
moduleIds: true,
|
|
getModuleId: function(name) {
|
|
return 'ionic/' + name;
|
|
}
|
|
}
|
|
|
|
gulp.task('build', ['bundle.system', 'e2e.build', 'copy.web-animations', 'sass', 'fonts']);
|
|
|
|
gulp.task('clean.build', function(done) {
|
|
runSequence('clean', 'build', done);
|
|
});
|
|
|
|
gulp.task('watch', function(done) {
|
|
runSequence(
|
|
'build',
|
|
'serve',
|
|
function() {
|
|
watch([
|
|
'ionic/**/*.ts',
|
|
'!ionic/components/*/test/**/*',
|
|
'!ionic/util/test/*'
|
|
],
|
|
function(file) {
|
|
if (file.event === "unlink") {
|
|
deleteFile(file);
|
|
} else {
|
|
gulp.start('bundle.system');
|
|
}
|
|
}
|
|
);
|
|
|
|
watch('ionic/components/*/test/**/*', function(file) {
|
|
gulp.start('e2e.build');
|
|
});
|
|
|
|
watch('ionic/**/*.scss', function() {
|
|
gulp.start('sass');
|
|
});
|
|
|
|
done();
|
|
}
|
|
);
|
|
|
|
function deleteFile(file) {
|
|
var basePath = file.base.substring(0, file.base.lastIndexOf("ionic/"));
|
|
var relativePath = file.history[0].replace(file.base, '').replace('.ts', '.js');
|
|
|
|
var filePath = basePath + 'dist/' + relativePath;
|
|
var typingPath = filePath.replace('.js', '.d.ts');
|
|
|
|
delete cache.caches['no-typecheck'][file.history[0]];
|
|
remember.forget('no-typecheck', file.history[0]);
|
|
|
|
del([filePath, typingPath], function(){
|
|
gulp.start('bundle.system');
|
|
});
|
|
}
|
|
});
|
|
|
|
gulp.task('serve', function() {
|
|
connect.server({
|
|
port: flags.port,
|
|
livereload: {
|
|
port: 35700
|
|
}
|
|
});
|
|
});
|
|
|
|
gulp.task('clean', function(done) {
|
|
del(['dist/**', '!dist'], done);
|
|
});
|
|
|
|
function tsCompile(options, cacheName){
|
|
return gulp.src([
|
|
'ionic/**/*.ts',
|
|
'!ionic/components/*/test/**/*',
|
|
'!ionic/util/test/*'
|
|
])
|
|
.pipe(cache(cacheName, { optimizeMemory: true }))
|
|
.pipe(tsc(options, undefined, tscReporter))
|
|
.on('error', function(error) {
|
|
console.log(error.message);
|
|
this.emit('end');
|
|
});
|
|
}
|
|
|
|
gulp.task('transpile.no-typecheck', function(){
|
|
return tsCompile(tscOptionsNoTypeCheck, 'no-typecheck')
|
|
.pipe(gulp.dest('dist'));
|
|
});
|
|
|
|
gulp.task('transpile.typecheck', function(){
|
|
var merge = require('merge2');
|
|
|
|
var result = tsCompile(tscOptions, 'typecheck');
|
|
|
|
// merge definition and source streams
|
|
return merge([
|
|
result.dts,
|
|
result.js
|
|
])
|
|
.pipe(gulp.dest('dist'));
|
|
})
|
|
|
|
gulp.task('bundle.system', function(){
|
|
var babel = require('gulp-babel');
|
|
var concat = require('gulp-concat');
|
|
|
|
return tsCompile(tscOptionsEs6, 'system')
|
|
.pipe(babel(babelOptions))
|
|
.pipe(remember('system'))
|
|
.pipe(concat('ionic.system.js'))
|
|
.pipe(gulp.dest('dist/bundles'))
|
|
.pipe(connect.reload())
|
|
})
|
|
|
|
gulp.task('transpile', ['transpile.no-typecheck']);
|
|
|
|
gulp.task('bundle', ['bundle.cjs', 'bundle.system']);
|
|
|
|
gulp.task('bundle.cjs', ['transpile'], function(done){
|
|
//TODO
|
|
// if (flags.animations == 'polyfill') {
|
|
// prepend.push('window.Element.prototype.animate=undefined;');
|
|
// }
|
|
|
|
var config = require('./scripts/npm/ionic.webpack.config.js');
|
|
bundle({ config: config, stats: true });
|
|
|
|
// build minified bundle
|
|
var minConfig = require('./scripts/npm/ionic.min.webpack.config.js');
|
|
bundle({ config: minConfig, cb: finished, stats: true });
|
|
|
|
var outputPaths = [
|
|
config.output.path + path.sep + config.output.filename,
|
|
minConfig.output.path + path.sep + minConfig.output.filename
|
|
];
|
|
|
|
function finished(){
|
|
gulp.src(outputPaths)
|
|
.pipe(connect.reload())
|
|
.on('end', done);
|
|
}
|
|
})
|
|
|
|
function bundle(args) {
|
|
var webpack = require('webpack');
|
|
var path = require('path');
|
|
|
|
webpack(args.config, function(err, stats){
|
|
if (args.stats) {
|
|
var statsOptions = {
|
|
'colors': true,
|
|
'modules': false,
|
|
'chunks': false,
|
|
'exclude': ['node_module'],
|
|
'errorDetails': true
|
|
}
|
|
console.log(stats.toString(statsOptions));
|
|
}
|
|
|
|
args.cb && args.cb();
|
|
})
|
|
}
|
|
|
|
gulp.task('tests', function() {
|
|
return gulp.src('ionic/**/test/**/*.spec.ts')
|
|
.pipe(tsc(tscOptionsNoTypeCheck, undefined, tscReporter))
|
|
.pipe(rename(function(file) {
|
|
var regex = new RegExp(path.sep + 'test(' + path.sep + '|$)');
|
|
file.dirname = file.dirname.replace(regex, path.sep);
|
|
}))
|
|
.pipe(gulp.dest('dist/tests'))
|
|
})
|
|
|
|
gulp.task('e2e.build', function() {
|
|
var gulpif = require('gulp-if');
|
|
var merge = require('merge2');
|
|
var _ = require('lodash');
|
|
var fs = require('fs');
|
|
var VinylFile = require('vinyl');
|
|
|
|
var indexTemplate = _.template(
|
|
fs.readFileSync('scripts/e2e/e2e.template.html')
|
|
)({
|
|
buildConfig: buildConfig
|
|
})
|
|
var testTemplate = _.template(fs.readFileSync('scripts/e2e/e2e.template.js'));
|
|
|
|
var platforms = [
|
|
'android',
|
|
'ios',
|
|
];
|
|
|
|
// Get each test folder with gulp.src
|
|
var tsResult = gulp.src([
|
|
'ionic/components/*/test/*/**/*.ts',
|
|
'!ionic/components/*/test/*/**/*.spec.ts'
|
|
])
|
|
.pipe(cache('e2e.ts'))
|
|
.pipe(tsc(tscOptionsNoTypeCheck, undefined, tscReporter))
|
|
.on('error', function(error) {
|
|
console.log(error.message);
|
|
this.emit('end');
|
|
})
|
|
.pipe(gulpif(/index.js$/, createIndexHTML()))
|
|
.pipe(gulpif(/e2e.js$/, createPlatformTests()))
|
|
|
|
var testFiles = gulp.src([
|
|
'ionic/components/*/test/*/**/*',
|
|
'!ionic/components/*/test/*/**/*.ts'
|
|
])
|
|
.pipe(cache('e2e.files'))
|
|
|
|
return merge([
|
|
tsResult,
|
|
testFiles
|
|
])
|
|
.pipe(rename(function(file) {
|
|
var sep = path.sep;
|
|
file.dirname = file.dirname.replace(sep + 'test' + sep, sep);
|
|
}))
|
|
.pipe(gulp.dest('dist/e2e/'))
|
|
.pipe(connect.reload());
|
|
|
|
function createIndexHTML() {
|
|
return through2.obj(function(file, enc, next) {
|
|
this.push(new VinylFile({
|
|
base: file.base,
|
|
contents: new Buffer(indexTemplate),
|
|
path: path.join(path.dirname(file.path), 'index.html'),
|
|
}));
|
|
next(null, file);
|
|
});
|
|
}
|
|
|
|
function createPlatformTests(file) {
|
|
return through2.obj(function(file, enc, next) {
|
|
var self = this;
|
|
var relativePath = path.dirname(file.path.replace(/^.*?ionic(\/|\\)components(\/|\\)/, ''));
|
|
relativePath = relativePath.replace('/test/', '/');
|
|
var contents = file.contents.toString();
|
|
platforms.forEach(function(platform) {
|
|
var platformContents = testTemplate({
|
|
contents: contents,
|
|
buildConfig: buildConfig,
|
|
relativePath: relativePath,
|
|
platform: platform
|
|
});
|
|
self.push(new VinylFile({
|
|
base: file.base,
|
|
contents: new Buffer(platformContents),
|
|
path: file.path.replace(/e2e.js$/, platform + '.e2e.js')
|
|
}));
|
|
})
|
|
next();
|
|
})
|
|
}
|
|
});
|
|
|
|
gulp.task('e2e', ['e2e.build', 'bundle.system', 'copy.web-animations', 'sass', 'fonts']);
|
|
|
|
gulp.task('sass', function() {
|
|
var sass = require('gulp-sass');
|
|
var autoprefixer = require('gulp-autoprefixer');
|
|
var minifyCss = require('gulp-minify-css');
|
|
|
|
gulp.src([
|
|
'ionic/ionic.ios.scss',
|
|
'ionic/ionic.md.scss',
|
|
'ionic/ionic.scss'
|
|
])
|
|
.pipe(sass()
|
|
.on('error', sass.logError)
|
|
)
|
|
.pipe(autoprefixer(buildConfig.autoprefixer))
|
|
.pipe(gulp.dest('dist/bundles/'))
|
|
.pipe(minifyCss())
|
|
.pipe(rename({ extname: '.min.css' }))
|
|
.pipe(gulp.dest('dist/bundles/'));
|
|
});
|
|
|
|
gulp.task('fonts', function() {
|
|
return gulp.src(['ionic/**/*.ttf', 'ionic/**/*.woff'])
|
|
.pipe(gulp.dest('dist'))
|
|
.pipe(gulp.dest('dist/bundles'));
|
|
});
|
|
|
|
require('./scripts/snapshot/snapshot.task')(gulp, argv, buildConfig);
|
|
|
|
gulp.task('karma', ['tests'], function() {
|
|
var karma = require('karma').server;
|
|
return karma.start({ configFile: __dirname + '/scripts/karma/karma.conf.js' })
|
|
});
|
|
|
|
gulp.task('karma-watch', function() {
|
|
var karma = require('karma').server;
|
|
return karma.start({ configFile: __dirname + '/scripts/karma/karma-watch.conf.js' })
|
|
});
|
|
|
|
gulp.task('copy.ts', function() {
|
|
return gulp.src([
|
|
'ionic/**/*.ts',
|
|
'!ionic/components/*/test/**/*',
|
|
'!ionic/util/test/*'
|
|
])
|
|
.pipe(gulp.dest('dist/src/typescript'));
|
|
})
|
|
|
|
gulp.task('copy.scss', function() {
|
|
return gulp.src([
|
|
'ionic/**/*.scss',
|
|
'!ionic/components/*/test/**/*',
|
|
'!ionic/util/test/*'
|
|
])
|
|
.pipe(gulp.dest('dist'));
|
|
})
|
|
|
|
gulp.task('copy.web-animations', function() {
|
|
return gulp.src([
|
|
'scripts/resources/web-animations-js/web-animations.min.js'
|
|
])
|
|
.pipe(gulp.dest('dist/js'));
|
|
})
|
|
|
|
gulp.task('src.link', function(done) {
|
|
watch(['/ionic/**/*.ts', 'ionic/**/*.scss'], function(file) {
|
|
gulp.start('src');
|
|
});
|
|
})
|
|
|
|
gulp.task('src', function(done){
|
|
runSequence(
|
|
'clean',
|
|
['bundle', 'sass', 'fonts', 'copy.scss', 'copy.web-animations'],
|
|
'transpile.typecheck',
|
|
done
|
|
);
|
|
})
|
|
|
|
gulp.task('package', ['src'], function(){
|
|
return gulp.src([
|
|
'scripts/npm/.npmignore',
|
|
'scripts/npm/package.json',
|
|
'scripts/npm/README.md'
|
|
])
|
|
.pipe(gulp.dest('dist'));
|
|
});
|
|
|
|
require('./scripts/docs/gulp-tasks')(gulp, flags)
|
|
|
|
////////////////////////////////////////////////////
|
|
// Demos
|
|
// todo: move these to scripts/docs/gulp-tasks
|
|
////////////////////////////////////////////////////
|
|
|
|
gulp.task('build.demos', function(){
|
|
var gulpif = require('gulp-if');
|
|
var lazypipe = require('lazypipe');
|
|
var _ = require('lodash');
|
|
var fs = require('fs');
|
|
var VinylFile = require('vinyl');
|
|
|
|
var buildTest = lazypipe()
|
|
.pipe(tsc, tscOptions, undefined, tscReporter)
|
|
.pipe(babel, getBabelOptions('demos', 'common'))
|
|
// .pipe(babel, getBabelOptions('demos'))
|
|
|
|
var baseIndexTemplate = _.template(fs.readFileSync('scripts/demos/index.template.html'))();
|
|
var docsIndexTemplate = _.template(fs.readFileSync('scripts/demos/docs.index.template.html'))();
|
|
|
|
return gulp.src(['demos/**/*'])
|
|
.pipe(cache('demos', { optimizeMemory: true }))
|
|
.pipe(gulpif(/.ts$/, buildTest()))
|
|
.pipe(gulpif(/index.js$/, createIndexHTML())) //TSC changes .ts to .js
|
|
.pipe(gulp.dest('dist/demos'))
|
|
|
|
function createIndexHTML() {
|
|
return through2.obj(function(file, enc, next) {
|
|
var indexTemplate = baseIndexTemplate;
|
|
var customTemplateFp = file.path.split('/').slice(0, -1).join('/') + '/index.html';
|
|
if (file.path.indexOf('component-docs') > -1) {
|
|
indexTemplate = docsIndexTemplate;
|
|
} else if (fs.existsSync(customTemplateFp)) {
|
|
indexTemplate = _.template(fs.readFileSync(customTemplateFp))();
|
|
}
|
|
this.push(new VinylFile({
|
|
base: file.base,
|
|
contents: new Buffer(indexTemplate),
|
|
path: path.join(path.dirname(file.path), 'index.html'),
|
|
}));
|
|
next(null, file);
|
|
});
|
|
}
|
|
});
|
|
|
|
gulp.task('sass.demos:components', function() {
|
|
var sass = require('gulp-sass');
|
|
var autoprefixer = require('gulp-autoprefixer');
|
|
var concat = require('gulp-concat');
|
|
return gulp.src([
|
|
'demos/component-docs/app.scss',
|
|
])
|
|
.pipe(sass()
|
|
.on('error', sass.logError)
|
|
)
|
|
.pipe(concat('app.css'))
|
|
.pipe(autoprefixer(buildConfig.autoprefixer))
|
|
.pipe(gulp.dest('../ionic-site/docs/v2/demos/component-docs/'));
|
|
});
|
|
|
|
gulp.task('bundle.demos:api', ['build.demos'], function(done) {
|
|
return buildDemoBundle({demo: 'api'}, done);
|
|
});
|
|
|
|
gulp.task('bundle.demos:components', ['sass.demos:components', 'build.demos'], function(done) {
|
|
buildDemoBundle({demo: 'component-docs'}, done);
|
|
});
|
|
|
|
gulp.task('demos', ['bundle.demos:api', 'bundle.demos:components'], function() {
|
|
return gulp
|
|
.src([
|
|
'dist/demos/**/*',
|
|
'!dist/demos/**/*.scss',
|
|
])
|
|
.pipe(gulp.dest('../ionic-site/docs/v2/demos/'))
|
|
});
|
|
|
|
gulp.task('watch:demos', function() {
|
|
watch('demos/**/*', function() {
|
|
gulp.start('demos');
|
|
});
|
|
});
|
|
|
|
function buildDemoBundle(opts, done) {
|
|
var glob = require('glob');
|
|
var webpack = require('webpack');
|
|
var path = require('path');
|
|
|
|
var fp = 'dist/demos/'+opts.demo+'/index.js';
|
|
if (opts.demo == 'api') {
|
|
fp = "dist/demos/**/index.js";
|
|
}
|
|
|
|
return glob(fp, function(err, files){
|
|
var numTasks = files.length;
|
|
files.forEach(function(file){
|
|
var config = require('./scripts/demos/webpack.config.js');
|
|
|
|
// add our bundle entry, removing previous if necessary
|
|
// since config is cached
|
|
if (config.entry.length > 4) {
|
|
config.entry.pop();
|
|
}
|
|
config.entry.push('./' + file);
|
|
config.output = {
|
|
filename: path.dirname(file) + '/bundle.js'
|
|
}
|
|
|
|
webpack(config, function(err, stats){
|
|
// var statsOptions = {
|
|
// 'colors': true,
|
|
// 'modules': true,
|
|
// 'chunks': false,
|
|
// 'exclude': ['node_modules'],
|
|
// 'errorDetails': true
|
|
// }
|
|
// console.log(stats.toString(statsOptions));
|
|
if (--numTasks === 0) done();
|
|
})
|
|
})
|
|
|
|
});
|
|
}
|