在 Cordova 的 plugin.xml 支持之前,我已经实施了一种解决方法,希望将来,一旦此类条目中的 embed 属性将具有相同的效果:<framework embed="true" src="..." />,目前,此属性没有帮助,因此有以下解决方法。
以下解决方案使用 Cordova 版本 5.3.3 工作。
首先,确保将框架条目添加到plugin.xml:
<framework src="pointToYour/File.framework" embed="true" />
embed="true" 暂时不起作用,但还是添加它。
我们要创建一个钩子,在你的 plugin.xml 中声明它:
<hook type="after_platform_add" src="hooks/embedframework/addEmbedded.js" />
接下来,我们在钩子的代码中需要一个特定的节点模块,该模块是node-xcode。
安装node-xcode(必须是0.8.7以上版本):
npm i xcode
最后是钩子本身的代码 -
addEmbedded.js 文件:
'use strict';
const xcode = require('xcode'),
fs = require('fs'),
path = require('path');
module.exports = function(context) {
if(process.length >=5 && process.argv[1].indexOf('cordova') == -1) {
if(process.argv[4] != 'ios') {
return; // plugin only meant to work for ios platform.
}
}
function fromDir(startPath,filter, rec, multiple){
if (!fs.existsSync(startPath)){
console.log("no dir ", startPath);
return;
}
const files=fs.readdirSync(startPath);
var resultFiles = []
for(var i=0;i<files.length;i++){
var filename=path.join(startPath,files[i]);
var stat = fs.lstatSync(filename);
if (stat.isDirectory() && rec){
fromDir(filename,filter); //recurse
}
if (filename.indexOf(filter)>=0) {
if (multiple) {
resultFiles.push(filename);
} else {
return filename;
}
}
}
if(multiple) {
return resultFiles;
}
}
function getFileIdAndRemoveFromFrameworks(myProj, fileBasename) {
var fileId = '';
const pbxFrameworksBuildPhaseObjFiles = myProj.pbxFrameworksBuildPhaseObj(myProj.getFirstTarget().uuid).files;
for(var i=0; i<pbxFrameworksBuildPhaseObjFiles.length;i++) {
var frameworkBuildPhaseFile = pbxFrameworksBuildPhaseObjFiles[i];
if(frameworkBuildPhaseFile.comment && frameworkBuildPhaseFile.comment.indexOf(fileBasename) != -1) {
fileId = frameworkBuildPhaseFile.value;
pbxFrameworksBuildPhaseObjFiles.splice(i,1); // MUST remove from frameworks build phase or else CodeSignOnCopy won't do anything.
break;
}
}
return fileId;
}
function getFileRefFromName(myProj, fName) {
const fileReferences = myProj.hash.project.objects['PBXFileReference'];
var fileRef = '';
for(var ref in fileReferences) {
if(ref.indexOf('_comment') == -1) {
var tmpFileRef = fileReferences[ref];
if(tmpFileRef.name && tmpFileRef.name.indexOf(fName) != -1) {
fileRef = ref;
break;
}
}
}
return fileRef;
}
const xcodeProjPath = fromDir('platforms/ios','.xcodeproj', false);
const projectPath = xcodeProjPath + '/project.pbxproj';
const myProj = xcode.project(projectPath);
function addRunpathSearchBuildProperty(proj, build) {
const LD_RUNPATH_SEARCH_PATHS = proj.getBuildProperty("LD_RUNPATH_SEARCH_PATHS", build);
if(!LD_RUNPATH_SEARCH_PATHS) {
proj.addBuildProperty("LD_RUNPATH_SEARCH_PATHS", "\"$(inherited) @executable_path/Frameworks\"", build);
} else if(LD_RUNPATH_SEARCH_PATHS.indexOf("@executable_path/Frameworks") == -1) {
var newValue = LD_RUNPATH_SEARCH_PATHS.substr(0,LD_RUNPATH_SEARCH_PATHS.length-1);
newValue += ' @executable_path/Frameworks\"';
proj.updateBuildProperty("LD_RUNPATH_SEARCH_PATHS", newValue, build);
}
}
myProj.parseSync();
addRunpathSearchBuildProperty(myProj, "Debug");
addRunpathSearchBuildProperty(myProj, "Release");
// unquote (remove trailing ")
var projectName = myProj.getFirstTarget().firstTarget.name.substr(1);
projectName = projectName.substr(0, projectName.length-1); //Removing the char " at beginning and the end.
const groupName = 'Embed Frameworks ' + context.opts.plugin.id;
const pluginPathInPlatformIosDir = projectName + '/Plugins/' + context.opts.plugin.id;
process.chdir('./platforms/ios');
const frameworkFilesToEmbed = fromDir(pluginPathInPlatformIosDir ,'.framework', false, true);
process.chdir('../../');
if(!frameworkFilesToEmbed.length) return;
myProj.addBuildPhase(frameworkFilesToEmbed, 'PBXCopyFilesBuildPhase', groupName, myProj.getFirstTarget().uuid, 'frameworks');
for(var frmFileFullPath of frameworkFilesToEmbed) {
var justFrameworkFile = path.basename(frmFileFullPath);
var fileRef = getFileRefFromName(myProj, justFrameworkFile);
var fileId = getFileIdAndRemoveFromFrameworks(myProj, justFrameworkFile);
// Adding PBXBuildFile for embedded frameworks
var file = {
uuid: fileId,
basename: justFrameworkFile,
settings: {
ATTRIBUTES: ["CodeSignOnCopy", "RemoveHeadersOnCopy"]
},
fileRef:fileRef,
group:groupName
};
myProj.addToPbxBuildFileSection(file);
// Adding to Frameworks as well (separate PBXBuildFile)
var newFrameworkFileEntry = {
uuid: myProj.generateUuid(),
basename: justFrameworkFile,
fileRef:fileRef,
group: "Frameworks"
};
myProj.addToPbxBuildFileSection(newFrameworkFileEntry);
myProj.addToPbxFrameworksBuildPhase(newFrameworkFileEntry);
}
fs.writeFileSync(projectPath, myProj.writeSync());
console.log('Embedded Frameworks In ' + context.opts.plugin.id);
};
这个钩子实际上做了什么:
- 创建一个以您的插件 ID 命名的“构建阶段”,配置为“复制文件”,该副本的目标是“框架”。
- 查找您的 .framework 文件并将其添加到上述构建阶段,然后将其嵌入。
- 设置一个名为
LD_RUNPATH_SEARCH_PATHS 的 Xcode 构建属性以在 "@executable_path/Frameworks" 中查找嵌入式框架(这是嵌入式框架将在“复制文件”->“框架”构建阶段之后被复制到的位置
- 通过为您的 .framework 文件设置“CodeSignOnCopy”和“RemoveHeadersOnCopy”来配置 ATTRIBUTES 键。
- 从 FrameworksBuildPhase 中删除您的 .framework 文件并将它们重新添加到 FrameworksBuildPhase 作为新的分离的 PBXBuildFiles(相同的 PBXFileReference),必须这样做才能使“CodeSignOnCopy”具有任何含义,而不是删除它,如果您用 Xcode 打开项目,在构建阶段你不会发现它会签名的复选标记。
更新1:钩子代码,修改:
- 挂钩会自动查找您的 .framework 文件,无需编辑挂钩。
- 添加了一项重要修改,为您的 .framework 文件设置 ATTRIBUTES "CodeSignOnCopy" 和 "RemoveHeadersOnCopy"。
- 改进了钩子,使其能够在多个插件使用此钩子的情况下工作。
更新 2
- 由于我的pull request 已被接受,我不再需要安装我自己的 fork。
- 改进的挂钩代码。
更新 3 (19/09/2016)
根据 Max Whaler 的建议修改了钩子脚本,因为我在 Xcode 8 上遇到了同样的问题。
最后说明
将应用上传到 AppStore 后,如果由于不支持的架构(i386 等...)而导致验证失败,请尝试以下 Cordova 插件(仅挂钩,无本机代码):zcordova-plugin-archtrim