#!/usr/bin/env node /* Licensed to the Apache Software Foundation (ASF) under one or more contributor license agreements. See the NOTICE file distributed with this work for additional information regarding copyright ownership. The ASF licenses this file to you under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License. */ var shell = require('shelljs'); var Q = require('q'); var path = require('path'); var fs = require('fs'); var xcode = require('xcode'); var xmlescape = require('xml-escape'); var ROOT = path.join(__dirname, '..', '..'); var events = require('cordova-common').events; function updateSubprojectHelp () { console.log('Updates the subproject path of the CordovaLib entry to point to this script\'s version of Cordova.'); console.log('Usage: CordovaVersion/bin/update_cordova_project path/to/your/app.xcodeproj [path/to/CordovaLib.xcodeproj]'); } function copyJsAndCordovaLib (projectPath, projectName, use_shared, config) { shell.cp('-f', path.join(ROOT, 'CordovaLib', 'cordova.js'), path.join(projectPath, 'www')); shell.cp('-rf', path.join(ROOT, 'cordova-js-src'), path.join(projectPath, 'platform_www')); shell.cp('-f', path.join(ROOT, 'CordovaLib', 'cordova.js'), path.join(projectPath, 'platform_www')); try { const stats = fs.lstatSync(path.join(projectPath, 'CordovaLib')); if (stats.isSymbolicLink()) { fs.unlinkSync(path.join(projectPath, 'CordovaLib')); } else { shell.rm('-rf', path.join(projectPath, 'CordovaLib')); } } catch (e) { } if (use_shared) { update_cordova_subproject([path.join(projectPath, projectName + '.xcodeproj', 'project.pbxproj'), config]); // Symlink not used in project file, but is currently required for plugman because // it reads the VERSION file from it (instead of using the cordova/version script // like it should). fs.symlinkSync(path.join(ROOT, 'CordovaLib'), path.join(projectPath, 'CordovaLib')); } else { var r = path.join(projectPath, projectName); shell.mkdir('-p', path.join(projectPath, 'CordovaLib', 'CordovaLib.xcodeproj')); shell.cp('-f', path.join(r, '.gitignore'), projectPath); shell.cp('-rf', path.join(ROOT, 'CordovaLib', 'Classes'), path.join(projectPath, 'CordovaLib')); shell.cp('-f', path.join(ROOT, 'CordovaLib', 'VERSION'), path.join(projectPath, 'CordovaLib')); shell.cp('-f', path.join(ROOT, 'CordovaLib', 'cordova.js'), path.join(projectPath, 'CordovaLib')); shell.cp('-f', path.join(ROOT, 'CordovaLib', 'CordovaLib_Prefix.pch'), path.join(projectPath, 'CordovaLib')); shell.cp('-f', path.join(ROOT, 'CordovaLib', 'CordovaLib.xcodeproj', 'project.pbxproj'), path.join(projectPath, 'CordovaLib', 'CordovaLib.xcodeproj')); update_cordova_subproject([path.join(r + '.xcodeproj', 'project.pbxproj'), path.join(projectPath, 'CordovaLib', 'CordovaLib.xcodeproj', 'project.pbxproj'), config]); } } function copyScripts (projectPath, projectName) { var srcScriptsDir = path.join(ROOT, 'bin', 'templates', 'scripts', 'cordova'); var destScriptsDir = path.join(projectPath, 'cordova'); // Delete old scripts directory. shell.rm('-rf', destScriptsDir); // Copy in the new ones. var binDir = path.join(ROOT, 'bin'); shell.cp('-r', srcScriptsDir, projectPath); let nodeModulesDir = path.join(ROOT, 'node_modules'); if (fs.existsSync(nodeModulesDir)) shell.cp('-r', nodeModulesDir, destScriptsDir); // Copy the check_reqs script shell.cp(path.join(binDir, 'check_reqs*'), destScriptsDir); // Copy the version scripts shell.cp(path.join(binDir, 'apple_ios_version'), destScriptsDir); shell.cp(path.join(binDir, 'apple_osx_version'), destScriptsDir); shell.cp(path.join(binDir, 'apple_xcode_version'), destScriptsDir); // TODO: the two files being edited on-the-fly here are shared between // platform and project-level commands. the below `sed` is updating the // `require` path for the two libraries. if there's a better way to share // modules across both the repo and generated projects, we should make sure // to remove/update this. var path_regex = /templates\/scripts\/cordova\//; shell.sed('-i', path_regex, '', path.join(destScriptsDir, 'check_reqs')); shell.sed('-i', path_regex, '', path.join(destScriptsDir, 'apple_ios_version')); shell.sed('-i', path_regex, '', path.join(destScriptsDir, 'apple_osx_version')); shell.sed('-i', path_regex, '', path.join(destScriptsDir, 'apple_xcode_version')); // CB-11792 do a token replace for __PROJECT_NAME__ in .xcconfig var project_name_esc = projectName.replace(/&/g, '\\&'); shell.sed('-i', /__PROJECT_NAME__/g, project_name_esc, path.join(destScriptsDir, 'build-debug.xcconfig')); shell.sed('-i', /__PROJECT_NAME__/g, project_name_esc, path.join(destScriptsDir, 'build-release.xcconfig')); // Make sure they are executable (sometimes zipping them can remove executable bit) shell.find(destScriptsDir).forEach(function (entry) { shell.chmod(755, entry); }); } /* * Copy project template files into cordova project. * * @param {String} project_path path to cordova project * @param {String} project_name name of cordova project * @param {String} project_template_dir path to cordova-ios template directory * @parm {BOOL} use_cli true if cli project */ function copyTemplateFiles (project_path, project_name, project_template_dir, package_name) { var r = path.join(project_path, project_name); shell.rm('-rf', path.join(r + '.xcodeproj')); shell.cp('-rf', path.join(project_template_dir, '__TEMP__.xcodeproj'), project_path); shell.mv('-f', path.join(project_path, '__TEMP__.xcodeproj'), path.join(r + '.xcodeproj')); shell.rm('-rf', path.join(project_path, project_name + '.xcworkspace')); shell.cp('-rf', path.join(project_template_dir, '__TEMP__.xcworkspace'), project_path); shell.mv('-f', path.join(project_path, '__TEMP__.xcworkspace'), path.join(r + '.xcworkspace')); shell.mv('-f', path.join(r + '.xcworkspace', 'xcshareddata', 'xcschemes', '__PROJECT_NAME__.xcscheme'), path.join(r + '.xcworkspace', 'xcshareddata', 'xcschemes', project_name + '.xcscheme')); shell.rm('-rf', r); shell.cp('-rf', path.join(project_template_dir, '__PROJECT_NAME__'), project_path); shell.mv('-f', path.join(project_path, '__PROJECT_NAME__'), r); shell.mv('-f', path.join(r, '__PROJECT_NAME__-Info.plist'), path.join(r, project_name + '-Info.plist')); shell.mv('-f', path.join(r, '__PROJECT_NAME__-Prefix.pch'), path.join(r, project_name + '-Prefix.pch')); shell.mv('-f', path.join(r, 'gitignore'), path.join(r, '.gitignore')); /* replace __PROJECT_NAME__ and __PROJECT_ID__ with ACTIVITY and ID strings, respectively, in: * * - ./__PROJECT_NAME__.xcodeproj/project.pbxproj * - ./__PROJECT_NAME__/Classes/AppDelegate.h * - ./__PROJECT_NAME__/Classes/AppDelegate.m * - ./__PROJECT_NAME__/Classes/MainViewController.h * - ./__PROJECT_NAME__/Classes/MainViewController.m * - ./__PROJECT_NAME__/Resources/main.m * - ./__PROJECT_NAME__/Resources/__PROJECT_NAME__-info.plist * - ./__PROJECT_NAME__/Resources/__PROJECT_NAME__-Prefix.plist */ // https://issues.apache.org/jira/browse/CB-12402 - Encode XML characters properly var project_name_xml_esc = xmlescape(project_name); shell.sed('-i', /__PROJECT_NAME__/g, project_name_xml_esc, path.join(r + '.xcworkspace', 'contents.xcworkspacedata')); shell.sed('-i', /__PROJECT_NAME__/g, project_name_xml_esc, path.join(r + '.xcworkspace', 'xcshareddata', 'xcschemes', project_name + '.xcscheme')); var project_name_esc = project_name.replace(/&/g, '\\&'); shell.sed('-i', /__PROJECT_NAME__/g, project_name_esc, path.join(r + '.xcodeproj', 'project.pbxproj')); shell.sed('-i', /__PROJECT_ID__/g, package_name, path.join(r + '.xcodeproj', 'project.pbxproj')); shell.sed('-i', /__PROJECT_NAME__/g, project_name_esc, path.join(r, 'Classes', 'AppDelegate.h')); shell.sed('-i', /__PROJECT_NAME__/g, project_name_esc, path.join(r, 'Classes', 'AppDelegate.m')); shell.sed('-i', /__PROJECT_NAME__/g, project_name_esc, path.join(r, 'Classes', 'MainViewController.h')); shell.sed('-i', /__PROJECT_NAME__/g, project_name_esc, path.join(r, 'Classes', 'MainViewController.m')); shell.sed('-i', /__PROJECT_NAME__/g, project_name_esc, path.join(r, 'main.m')); shell.sed('-i', /__PROJECT_NAME__/g, project_name_esc, path.join(r, project_name + '-Info.plist')); shell.sed('-i', /__PROJECT_NAME__/g, project_name_esc, path.join(r, project_name + '-Prefix.pch')); } function AbsParentPath (_path) { return path.resolve(path.dirname(_path)); } function AbsProjectPath (relative_path) { var absolute_path = path.resolve(relative_path); if (/.pbxproj$/.test(absolute_path)) { absolute_path = AbsParentPath(absolute_path); } else if (!(/.xcodeproj$/.test(absolute_path))) { throw new Error('The following is not a valid path to an Xcode project: ' + absolute_path); } return absolute_path; } function relpath (_path, start) { start = start || process.cwd(); return path.relative(path.resolve(start), path.resolve(_path)); } /* * Creates a new iOS project with the following options: * * - --link (optional): Link directly against the shared copy of the CordovaLib instead of a copy of it * - --cli (optional): Use the CLI-project template * - : Path to your new Cordova iOS project * - : Package name, following reverse-domain style convention * - : Project name * - : Path to a project template (override) * */ exports.createProject = function (project_path, package_name, project_name, opts, config) { package_name = package_name || 'my.cordova.project'; project_name = project_name || 'CordovaExample'; var use_shared = !!opts.link; var bin_dir = path.join(ROOT, 'bin'); var project_parent = path.dirname(project_path); var project_template_dir = opts.customTemplate || path.join(bin_dir, 'templates', 'project'); // check that project path doesn't exist if (fs.existsSync(project_path)) { return Q.reject('Project already exists'); } // check that parent directory does exist so cp -r will not fail if (!fs.existsSync(project_parent)) { return Q.reject('Parent directory "' + project_parent + '" of given project path does not exist'); } events.emit('log', 'Creating Cordova project for the iOS platform:'); events.emit('log', '\tPath: ' + path.relative(process.cwd(), project_path)); events.emit('log', '\tPackage: ' + package_name); events.emit('log', '\tName: ' + project_name); events.emit('verbose', 'Copying iOS template project to ' + project_path); // create the project directory and copy over files shell.mkdir(project_path); shell.cp('-rf', path.join(project_template_dir, 'www'), project_path); // Copy project template files copyTemplateFiles(project_path, project_name, project_template_dir, package_name); // Copy xcconfig files shell.cp('-rf', path.join(project_template_dir, '*.xcconfig'), project_path); // CordovaLib stuff copyJsAndCordovaLib(project_path, project_name, use_shared, config); copyScripts(project_path, project_name); events.emit('log', generateDoneMessage('create', use_shared)); return Q.resolve(); }; exports.updateProject = function (projectPath, opts) { var errorString = 'An in-place platform update is not supported. \n' + 'The `platforms` folder is always treated as a build artifact.\n' + 'To update your platform, you have to remove, then add your ios platform again.\n' + 'Make sure you save your plugins beforehand using `cordova plugin save`, and save a copy of the platform first if you had manual changes in it.\n' + '\tcordova plugin save\n' + '\tcordova platform rm ios\n' + '\tcordova platform add ios\n' ; return Q.reject(errorString); }; function generateDoneMessage (type, link) { var pkg = require('../../package'); var msg = 'iOS project ' + (type === 'update' ? 'updated ' : 'created ') + 'with ' + pkg.name + '@' + pkg.version; if (link) { msg += ' and has a linked CordovaLib'; } return msg; } function update_cordova_subproject (argv) { if (argv.length < 1 || argv.length > 3) { updateSubprojectHelp(); throw new Error('Usage error for update_cordova_subproject'); } var projectPath = AbsProjectPath(argv[0]); var cordovaLibXcodePath; var projectConfig; if (argv.length < 3) { cordovaLibXcodePath = path.join(ROOT, 'CordovaLib', 'CordovaLib.xcodeproj'); projectConfig = argv[1]; } else { cordovaLibXcodePath = AbsProjectPath(argv[1]); projectConfig = argv[2]; } var parentProjectPath = AbsParentPath(projectPath); var subprojectPath = relpath(cordovaLibXcodePath, parentProjectPath); var REGEX = /(.+PBXFileReference.+wrapper.pb-project.+)(path = .+?;)(.*)(sourceTree.+;)(.+)/; var newLine; var lines = shell.grep('CordovaLib.xcodeproj', path.join(projectPath, 'project.pbxproj')); var found = false; subprojectPath = subprojectPath.replace(/\\/g, '/'); lines = lines.split('\n'); for (var i = 0; i < lines.length; ++i) { if (lines[i].match(REGEX)) { found = true; newLine = lines[i].replace(/path = .+?;/, 'path = ' + subprojectPath + ';'); newLine = newLine.replace(/sourceTree.+?;/, 'sourceTree = \"\";'); /* eslint no-useless-escape : 0 */ if (!newLine.match('name')) { newLine = newLine.replace('path = ', 'name = CordovaLib.xcodeproj; path = '); } shell.sed('-i', lines[i], newLine, path.join(projectPath, 'project.pbxproj')); } } if (!found) { throw new Error('Entry not found in project file for sub-project: ' + subprojectPath); } var wkWebViewOnly = projectConfig.getPreference('WKWebViewOnly') === 'true'; if (wkWebViewOnly) { var pbxPath = path.join(cordovaLibXcodePath, 'project.pbxproj'); var xcodeproj = xcode.project(pbxPath); xcodeproj.parseSync(); xcodeproj.updateBuildProperty('WK_WEB_VIEW_ONLY', '1'); fs.writeFileSync(pbxPath, xcodeproj.writeSync()); } } exports.updateSubprojectHelp = updateSubprojectHelp; exports.update_cordova_subproject = update_cordova_subproject;