2016-11-18 18:25:02 -08:00
|
|
|
/**
|
|
|
|
* Copyright (c) 2015-present, Facebook, Inc.
|
|
|
|
*
|
2018-02-16 18:24:55 -08:00
|
|
|
* This source code is licensed under the MIT license found in the
|
|
|
|
* LICENSE file in the root directory of this source tree.
|
2018-05-11 12:43:49 -07:00
|
|
|
*
|
|
|
|
* @format
|
2016-11-18 18:25:02 -08:00
|
|
|
*/
|
2018-05-11 12:43:49 -07:00
|
|
|
|
2016-11-18 18:25:02 -08:00
|
|
|
'use strict';
|
|
|
|
|
|
|
|
const chalk = require('chalk');
|
|
|
|
const copyAndReplace = require('../util/copyAndReplace');
|
|
|
|
const path = require('path');
|
|
|
|
const prompt = require('./promptSync')();
|
|
|
|
const walk = require('../util/walk');
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Util for creating a new React Native project.
|
|
|
|
* Copy the project from a template and use the correct project name in
|
|
|
|
* all files.
|
|
|
|
* @param srcPath e.g. '/Users/martin/AwesomeApp/node_modules/react-native/local-cli/templates/HelloWorld'
|
|
|
|
* @param destPath e.g. '/Users/martin/AwesomeApp'
|
|
|
|
* @param newProjectName e.g. 'AwesomeApp'
|
2017-02-27 09:12:14 -08:00
|
|
|
* @param options e.g. {
|
|
|
|
* upgrade: true,
|
|
|
|
* force: false,
|
|
|
|
* displayName: 'Hello World',
|
|
|
|
* ignorePaths: ['template/file/to/ignore.md'],
|
|
|
|
* }
|
2016-11-18 18:25:02 -08:00
|
|
|
*/
|
2018-05-11 12:43:49 -07:00
|
|
|
function copyProjectTemplateAndReplace(
|
|
|
|
srcPath,
|
|
|
|
destPath,
|
|
|
|
newProjectName,
|
|
|
|
options,
|
|
|
|
) {
|
|
|
|
if (!srcPath) {
|
|
|
|
throw new Error('Need a path to copy from');
|
|
|
|
}
|
|
|
|
if (!destPath) {
|
|
|
|
throw new Error('Need a path to copy to');
|
|
|
|
}
|
|
|
|
if (!newProjectName) {
|
|
|
|
throw new Error('Need a project name');
|
|
|
|
}
|
2016-11-18 18:25:02 -08:00
|
|
|
|
2017-02-03 12:52:27 -08:00
|
|
|
options = options || {};
|
|
|
|
|
2016-11-18 18:25:02 -08:00
|
|
|
walk(srcPath).forEach(absoluteSrcFilePath => {
|
|
|
|
// 'react-native upgrade'
|
2017-02-03 12:52:27 -08:00
|
|
|
if (options.upgrade) {
|
2016-11-18 18:25:02 -08:00
|
|
|
// Don't upgrade these files
|
|
|
|
const fileName = path.basename(absoluteSrcFilePath);
|
|
|
|
// This also includes __tests__/index.*.js
|
2018-05-11 12:43:49 -07:00
|
|
|
if (fileName === 'index.ios.js') {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (fileName === 'index.android.js') {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (fileName === 'index.js') {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (fileName === 'App.js') {
|
|
|
|
return;
|
|
|
|
}
|
2016-11-18 18:25:02 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
const relativeFilePath = path.relative(srcPath, absoluteSrcFilePath);
|
2016-11-21 12:52:36 -08:00
|
|
|
const relativeRenamedPath = dotFilePath(relativeFilePath)
|
2016-11-18 18:25:02 -08:00
|
|
|
.replace(/HelloWorld/g, newProjectName)
|
|
|
|
.replace(/helloworld/g, newProjectName.toLowerCase());
|
|
|
|
|
2017-02-27 09:12:14 -08:00
|
|
|
// Templates may contain files that we don't want to copy.
|
|
|
|
// Examples:
|
|
|
|
// - Dummy package.json file included in the template only for publishing to npm
|
|
|
|
// - Docs specific to the template (.md files)
|
|
|
|
if (options.ignorePaths) {
|
|
|
|
if (!Array.isArray(options.ignorePaths)) {
|
|
|
|
throw new Error('options.ignorePaths must be an array');
|
|
|
|
}
|
2018-05-11 12:43:49 -07:00
|
|
|
if (
|
|
|
|
options.ignorePaths.some(ignorePath => ignorePath === relativeFilePath)
|
|
|
|
) {
|
2017-02-27 09:12:14 -08:00
|
|
|
// Skip copying this file
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-11-18 18:25:02 -08:00
|
|
|
let contentChangedCallback = null;
|
2018-05-11 12:43:49 -07:00
|
|
|
if (options.upgrade && !options.force) {
|
2016-11-18 18:25:02 -08:00
|
|
|
contentChangedCallback = (_, contentChanged) => {
|
|
|
|
return upgradeFileContentChangedCallback(
|
|
|
|
absoluteSrcFilePath,
|
|
|
|
relativeRenamedPath,
|
|
|
|
contentChanged,
|
|
|
|
);
|
|
|
|
};
|
|
|
|
}
|
|
|
|
copyAndReplace(
|
|
|
|
absoluteSrcFilePath,
|
|
|
|
path.resolve(destPath, relativeRenamedPath),
|
|
|
|
{
|
2017-02-03 12:52:27 -08:00
|
|
|
'Hello App Display Name': options.displayName || newProjectName,
|
2018-05-11 12:43:49 -07:00
|
|
|
HelloWorld: newProjectName,
|
|
|
|
helloworld: newProjectName.toLowerCase(),
|
2016-11-18 18:25:02 -08:00
|
|
|
},
|
|
|
|
contentChangedCallback,
|
|
|
|
);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2016-11-21 12:52:36 -08:00
|
|
|
/**
|
|
|
|
* There are various dotfiles in the templates folder in the RN repo. We want
|
|
|
|
* these to be ignored by tools when working with React Native itself.
|
|
|
|
* Example: _babelrc file is ignored by Babel, renamed to .babelrc inside
|
|
|
|
* a real app folder.
|
|
|
|
* This is especially important for .gitignore because npm has some special
|
|
|
|
* behavior of automatically renaming .gitignore to .npmignore.
|
|
|
|
*/
|
|
|
|
function dotFilePath(path) {
|
2018-05-11 12:43:49 -07:00
|
|
|
if (!path) {
|
|
|
|
return path;
|
|
|
|
}
|
2016-11-21 12:52:36 -08:00
|
|
|
return path
|
|
|
|
.replace('_gitignore', '.gitignore')
|
2016-11-22 17:48:12 -08:00
|
|
|
.replace('_gitattributes', '.gitattributes')
|
2016-11-21 12:52:36 -08:00
|
|
|
.replace('_babelrc', '.babelrc')
|
|
|
|
.replace('_flowconfig', '.flowconfig')
|
|
|
|
.replace('_buckconfig', '.buckconfig')
|
|
|
|
.replace('_watchmanconfig', '.watchmanconfig');
|
|
|
|
}
|
|
|
|
|
2016-11-18 18:25:02 -08:00
|
|
|
function upgradeFileContentChangedCallback(
|
|
|
|
absoluteSrcFilePath,
|
|
|
|
relativeDestPath,
|
2018-05-11 12:43:49 -07:00
|
|
|
contentChanged,
|
2016-11-18 18:25:02 -08:00
|
|
|
) {
|
|
|
|
if (contentChanged === 'new') {
|
|
|
|
console.log(chalk.bold('new') + ' ' + relativeDestPath);
|
|
|
|
return 'overwrite';
|
|
|
|
} else if (contentChanged === 'changed') {
|
2018-05-11 12:43:49 -07:00
|
|
|
console.log(
|
|
|
|
chalk.bold(relativeDestPath) +
|
|
|
|
' ' +
|
|
|
|
'has changed in the new version.\nDo you want to keep your ' +
|
|
|
|
relativeDestPath +
|
|
|
|
' or replace it with the ' +
|
|
|
|
'latest version?\nIf you ever made any changes ' +
|
|
|
|
"to this file, you'll probably want to keep it.\n" +
|
|
|
|
'You can see the new version here: ' +
|
|
|
|
absoluteSrcFilePath +
|
|
|
|
'\n' +
|
|
|
|
'Do you want to replace ' +
|
|
|
|
relativeDestPath +
|
|
|
|
'? ' +
|
|
|
|
'Answer y to replace, n to keep your version: ',
|
|
|
|
);
|
2016-11-18 18:25:02 -08:00
|
|
|
const answer = prompt();
|
|
|
|
if (answer === 'y') {
|
|
|
|
console.log('Replacing ' + relativeDestPath);
|
|
|
|
return 'overwrite';
|
|
|
|
} else {
|
|
|
|
console.log('Keeping your ' + relativeDestPath);
|
|
|
|
return 'keep';
|
|
|
|
}
|
|
|
|
} else if (contentChanged === 'identical') {
|
|
|
|
return 'keep';
|
|
|
|
} else {
|
2018-05-11 12:43:49 -07:00
|
|
|
throw new Error(
|
|
|
|
`Unknown file changed state: ${relativeDestPath}, ${contentChanged}`,
|
|
|
|
);
|
2016-11-18 18:25:02 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
module.exports = copyProjectTemplateAndReplace;
|