use anyhow::Result;
use std::fs;
use std::path::Path;
pub fn create_macos_xcode_project(project_dir: &Path, app_name: &str) -> Result<()> {
crate::xcode_macos::create_macos_xcode_project(project_dir, app_name)
}
pub fn create_xcode_project(project_dir: &Path, app_name: &str) -> Result<()> {
let ios_dir = project_dir.join("platforms/ios");
let project_name = format!("{}.xcodeproj", app_name);
let project_path = ios_dir.join(&project_name);
fs::create_dir_all(&project_path)?;
fs::create_dir_all(project_path.join("project.xcworkspace"))?;
let pbxproj = generate_pbxproj(app_name)?;
fs::write(project_path.join("project.pbxproj"), pbxproj)?;
let workspace_contents = r#"<?xml version="1.0" encoding="UTF-8"?>
<Workspace
version = "1.0">
<FileRef
location = "self:">
</FileRef>
</Workspace>
"#;
fs::write(
project_path.join("project.xcworkspace/contents.xcworkspacedata"),
workspace_contents,
)?;
Ok(())
}
fn generate_pbxproj(app_name: &str) -> Result<String> {
let bundle_id = format!("com.example.{}", app_name.replace("-", ""));
let app_name_pascal = to_pascal_case(app_name);
let module_name = app_name.replace("-", "_");
let shell_script = format!(
"# Copy the Rust FFI dylib into the app bundle\n\
TARGET_DIR=\"\"\n\
if [ \"${{PLATFORM_NAME}}\" = \"iphoneos\" ]; then\n\
TARGET_DIR=\"aarch64-apple-ios\"\n\
elif [ \"${{PLATFORM_NAME}}\" = \"iphonesimulator\" ]; then\n\
if [ \"${{CURRENT_ARCH}}\" = \"arm64\" ]; then\n\
TARGET_DIR=\"aarch64-apple-ios-sim\"\n\
else\n\
TARGET_DIR=\"x86_64-apple-ios\"\n\
fi\n\
fi\n\
\n\
if [ -z \"$TARGET_DIR\" ]; then\n\
echo \"Warning: Unknown PLATFORM_NAME=${{PLATFORM_NAME}} CURRENT_ARCH=${{CURRENT_ARCH}}\"\n\
exit 0\n\
fi\n\
\n\
FFI_LIB=\"${{PROJECT_DIR}}/../../target/$TARGET_DIR/${{CONFIGURATION}}/lib{module_name}_ffi.dylib\"\n\
\n\
if [ -f \"$FFI_LIB\" ]; then\n\
echo \"Copying FFI library: $FFI_LIB\"\n\
cp \"$FFI_LIB\" \"${{BUILT_PRODUCTS_DIR}}/${{PRODUCT_NAME}}.app/\"\n\
echo \"FFI library copied successfully\"\n\
else\n\
echo \"Warning: FFI library not found at $FFI_LIB\"\n\
fi\n",
module_name = module_name
);
let escaped_shell_script = shell_script
.replace('\\', "\\\\")
.replace('"', "\\\"")
.replace('\n', "\\n");
let pbxproj = format!(r#"// !$*UTF8*$!
{{
archiveVersion = 1;
classes = {{
}};
objectVersion = 56;
objects = {{
/* Begin PBXBuildFile section */
A1000001000000000000001 /* {app_name_pascal}App.swift in Sources */ = {{isa = PBXBuildFile; fileRef = A2000001000000000000001 /* {app_name_pascal}App.swift */; }};
A1000002000000000000001 /* ContentView.swift in Sources */ = {{isa = PBXBuildFile; fileRef = A2000002000000000000001 /* ContentView.swift */; }};
A1000003000000000000001 /* AppState.swift in Sources */ = {{isa = PBXBuildFile; fileRef = A2000003000000000000001 /* AppState.swift */; }};
A1000004000000000000001 /* Assets.xcassets in Resources */ = {{isa = PBXBuildFile; fileRef = A2000004000000000000001 /* Assets.xcassets */; }};
A1000007000000000000001 /* {module_name}_ffi.swift in Sources */ = {{isa = PBXBuildFile; fileRef = A2000007000000000000001 /* {module_name}_ffi.swift */; }};
/* End PBXBuildFile section */
/* Begin PBXFileReference section */
A3000001000000000000001 /* {app_name}.app */ = {{isa = PBXFileReference; explicitFileType = wrapper.application; includeInIndex = 0; path = {app_name}.app; sourceTree = BUILT_PRODUCTS_DIR; }};
A2000001000000000000001 /* {app_name_pascal}App.swift */ = {{isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = {app_name_pascal}App.swift; sourceTree = "<group>"; }};
A2000002000000000000001 /* ContentView.swift */ = {{isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ContentView.swift; sourceTree = "<group>"; }};
A2000003000000000000001 /* AppState.swift */ = {{isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AppState.swift; sourceTree = "<group>"; }};
A2000004000000000000001 /* Assets.xcassets */ = {{isa = PBXFileReference; lastKnownFileType = folder.assetcatalog; path = Assets.xcassets; sourceTree = "<group>"; }};
A2000005000000000000001 /* Info.plist */ = {{isa = PBXFileReference; lastKnownFileType = text.plist.xml; path = Info.plist; sourceTree = "<group>"; }};
A2000006000000000000001 /* BridgingHeader.h */ = {{isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = BridgingHeader.h; sourceTree = "<group>"; }};
A2000007000000000000001 /* {module_name}_ffi.swift */ = {{isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = {module_name}_ffi.swift; sourceTree = "<group>"; }};
/* End PBXFileReference section */
/* Begin PBXFrameworksBuildPhase section */
A4000001000000000000001 /* Frameworks */ = {{
isa = PBXFrameworksBuildPhase;
buildActionMask = 2147483647;
files = (
);
runOnlyForDeploymentPostprocessing = 0;
}};
/* End PBXFrameworksBuildPhase section */
/* Begin PBXGroup section */
A5000001000000000000001 = {{
isa = PBXGroup;
children = (
A5000002000000000000001 /* {app_name} */,
A5000003000000000000001 /* Products */,
);
sourceTree = "<group>";
}};
A5000002000000000000001 /* {app_name} */ = {{
isa = PBXGroup;
children = (
A2000001000000000000001 /* {app_name_pascal}App.swift */,
A2000002000000000000001 /* ContentView.swift */,
A2000003000000000000001 /* AppState.swift */,
A2000007000000000000001 /* {module_name}_ffi.swift */,
A2000004000000000000001 /* Assets.xcassets */,
A2000005000000000000001 /* Info.plist */,
A2000006000000000000001 /* BridgingHeader.h */,
);
sourceTree = "<group>";
}};
A5000003000000000000001 /* Products */ = {{
isa = PBXGroup;
children = (
A3000001000000000000001 /* {app_name}.app */,
);
name = Products;
sourceTree = "<group>";
}};
/* End PBXGroup section */
/* Begin PBXNativeTarget section */
A6000001000000000000001 /* {app_name} */ = {{
isa = PBXNativeTarget;
buildConfigurationList = A7000001000000000000001 /* Build configuration list for PBXNativeTarget "{app_name}" */;
buildPhases = (
A8000001000000000000001 /* Sources */,
A4000001000000000000001 /* Frameworks */,
A9000001000000000000001 /* Resources */,
AB000001000000000000001 /* Copy FFI Library */,
);
buildRules = (
);
dependencies = (
);
name = {app_name};
productName = {app_name};
productReference = A3000001000000000000001 /* {app_name}.app */;
productType = "com.apple.product-type.application";
}};
/* End PBXNativeTarget section */
/* Begin PBXProject section */
A0000001000000000000001 /* Project object */ = {{
isa = PBXProject;
attributes = {{
BuildIndependentTargetsInParallel = 1;
LastSwiftUpdateCheck = 1500;
LastUpgradeCheck = 1500;
TargetAttributes = {{
A6000001000000000000001 = {{
CreatedOnToolsVersion = 15.0;
}};
}};
}};
buildConfigurationList = A7000002000000000000001 /* Build configuration list for PBXProject "{app_name}" */;
compatibilityVersion = "Xcode 14.0";
developmentRegion = en;
hasScannedForEncodings = 0;
knownRegions = (
en,
Base,
);
mainGroup = A5000001000000000000001;
productRefGroup = A5000003000000000000001 /* Products */;
projectDirPath = "";
projectRoot = "";
targets = (
A6000001000000000000001 /* {app_name} */,
);
}};
/* End PBXProject section */
/* Begin PBXResourcesBuildPhase section */
A9000001000000000000001 /* Resources */ = {{
isa = PBXResourcesBuildPhase;
buildActionMask = 2147483647;
files = (
A1000004000000000000001 /* Assets.xcassets in Resources */,
);
runOnlyForDeploymentPostprocessing = 0;
}};
/* End PBXResourcesBuildPhase section */
/* Begin PBXSourcesBuildPhase section */
A8000001000000000000001 /* Sources */ = {{
isa = PBXSourcesBuildPhase;
buildActionMask = 2147483647;
files = (
A1000002000000000000001 /* ContentView.swift in Sources */,
A1000001000000000000001 /* {app_name_pascal}App.swift in Sources */,
A1000003000000000000001 /* AppState.swift in Sources */,
A1000007000000000000001 /* {module_name}_ffi.swift in Sources */,
);
runOnlyForDeploymentPostprocessing = 0;
}};
/* End PBXSourcesBuildPhase section */
/* Begin PBXShellScriptBuildPhase section */
AB000001000000000000001 /* Copy FFI Library */ = {{
isa = PBXShellScriptBuildPhase;
buildActionMask = 2147483647;
files = (
);
inputFileListPaths = (
);
inputPaths = (
);
name = "Copy FFI Library";
outputFileListPaths = (
);
outputPaths = (
);
runOnlyForDeploymentPostprocessing = 0;
shellPath = /bin/sh;
shellScript = "{escaped_shell_script}";
}};
/* End PBXShellScriptBuildPhase section */
/* Begin XCBuildConfiguration section */
A7000003000000000000001 /* Debug */ = {{
isa = XCBuildConfiguration;
buildSettings = {{
ALWAYS_SEARCH_USER_PATHS = NO;
ASSETCATALOG_COMPILER_GENERATE_SWIFT_ASSET_SYMBOL_EXTENSIONS = YES;
CLANG_ANALYZER_NONNULL = YES;
CLANG_ANALYZER_NUMBER_OBJECT_CONVERSION = YES_AGGRESSIVE;
CLANG_CXX_LANGUAGE_STANDARD = "gnu++20";
CLANG_ENABLE_MODULES = YES;
CLANG_ENABLE_OBJC_ARC = YES;
CLANG_ENABLE_OBJC_WEAK = YES;
CLANG_WARN_BLOCK_CAPTURE_AUTORELEASING = YES;
CLANG_WARN_BOOL_CONVERSION = YES;
CLANG_WARN_COMMA = YES;
CLANG_WARN_CONSTANT_CONVERSION = YES;
CLANG_WARN_DEPRECATED_OBJC_IMPLEMENTATIONS = YES;
CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR;
CLANG_WARN_DOCUMENTATION_COMMENTS = YES;
CLANG_WARN_EMPTY_BODY = YES;
CLANG_WARN_ENUM_CONVERSION = YES;
CLANG_WARN_INFINITE_RECURSION = YES;
CLANG_WARN_INT_CONVERSION = YES;
CLANG_WARN_NON_LITERAL_NULL_CONVERSION = YES;
CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES;
CLANG_WARN_OBJC_LITERAL_CONVERSION = YES;
CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR;
CLANG_WARN_QUOTED_INCLUDE_IN_FRAMEWORK_HEADER = YES;
CLANG_WARN_RANGE_LOOP_ANALYSIS = YES;
CLANG_WARN_STRICT_PROTOTYPES = YES;
CLANG_WARN_SUSPICIOUS_MOVE = YES;
CLANG_WARN_UNGUARDED_AVAILABILITY = YES_AGGRESSIVE;
CLANG_WARN_UNREACHABLE_CODE = YES;
CLANG_WARN__DUPLICATE_METHOD_MATCH = YES;
COPY_PHASE_STRIP = NO;
DEBUG_INFORMATION_FORMAT = dwarf;
ENABLE_STRICT_OBJC_MSGSEND = YES;
ENABLE_TESTABILITY = YES;
ENABLE_USER_SCRIPT_SANDBOXING = YES;
GCC_C_LANGUAGE_STANDARD = gnu17;
GCC_DYNAMIC_NO_PIC = NO;
GCC_NO_COMMON_BLOCKS = YES;
GCC_OPTIMIZATION_LEVEL = 0;
GCC_PREPROCESSOR_DEFINITIONS = (
"DEBUG=1",
"$(inherited)",
);
GCC_WARN_64_TO_32_BIT_CONVERSION = YES;
GCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR;
GCC_WARN_UNDECLARED_SELECTOR = YES;
GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE;
GCC_WARN_UNUSED_FUNCTION = YES;
GCC_WARN_UNUSED_VARIABLE = YES;
IPHONEOS_DEPLOYMENT_TARGET = 16.0;
LOCALIZATION_PREFERS_STRING_CATALOGS = YES;
MTL_ENABLE_DEBUG_INFO = INCLUDE_SOURCE;
MTL_FAST_MATH = YES;
ONLY_ACTIVE_ARCH = YES;
SDKROOT = iphoneos;
SWIFT_ACTIVE_COMPILATION_CONDITIONS = "DEBUG $(inherited)";
SWIFT_OPTIMIZATION_LEVEL = "-Onone";
}};
name = Debug;
}};
A7000004000000000000001 /* Release */ = {{
isa = XCBuildConfiguration;
buildSettings = {{
ALWAYS_SEARCH_USER_PATHS = NO;
ASSETCATALOG_COMPILER_GENERATE_SWIFT_ASSET_SYMBOL_EXTENSIONS = YES;
CLANG_ANALYZER_NONNULL = YES;
CLANG_ANALYZER_NUMBER_OBJECT_CONVERSION = YES_AGGRESSIVE;
CLANG_CXX_LANGUAGE_STANDARD = "gnu++20";
CLANG_ENABLE_MODULES = YES;
CLANG_ENABLE_OBJC_ARC = YES;
CLANG_ENABLE_OBJC_WEAK = YES;
CLANG_WARN_BLOCK_CAPTURE_AUTORELEASING = YES;
CLANG_WARN_BOOL_CONVERSION = YES;
CLANG_WARN_COMMA = YES;
CLANG_WARN_CONSTANT_CONVERSION = YES;
CLANG_WARN_DEPRECATED_OBJC_IMPLEMENTATIONS = YES;
CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR;
CLANG_WARN_DOCUMENTATION_COMMENTS = YES;
CLANG_WARN_EMPTY_BODY = YES;
CLANG_WARN_ENUM_CONVERSION = YES;
CLANG_WARN_INFINITE_RECURSION = YES;
CLANG_WARN_INT_CONVERSION = YES;
CLANG_WARN_NON_LITERAL_NULL_CONVERSION = YES;
CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES;
CLANG_WARN_OBJC_LITERAL_CONVERSION = YES;
CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR;
CLANG_WARN_QUOTED_INCLUDE_IN_FRAMEWORK_HEADER = YES;
CLANG_WARN_RANGE_LOOP_ANALYSIS = YES;
CLANG_WARN_STRICT_PROTOTYPES = YES;
CLANG_WARN_SUSPICIOUS_MOVE = YES;
CLANG_WARN_UNGUARDED_AVAILABILITY = YES_AGGRESSIVE;
CLANG_WARN_UNREACHABLE_CODE = YES;
CLANG_WARN__DUPLICATE_METHOD_MATCH = YES;
COPY_PHASE_STRIP = NO;
DEBUG_INFORMATION_FORMAT = "dwarf-with-dsym";
ENABLE_NS_ASSERTIONS = NO;
ENABLE_STRICT_OBJC_MSGSEND = YES;
ENABLE_USER_SCRIPT_SANDBOXING = YES;
GCC_C_LANGUAGE_STANDARD = gnu17;
GCC_NO_COMMON_BLOCKS = YES;
GCC_WARN_64_TO_32_BIT_CONVERSION = YES;
GCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR;
GCC_WARN_UNDECLARED_SELECTOR = YES;
GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE;
GCC_WARN_UNUSED_FUNCTION = YES;
GCC_WARN_UNUSED_VARIABLE = YES;
IPHONEOS_DEPLOYMENT_TARGET = 16.0;
LOCALIZATION_PREFERS_STRING_CATALOGS = YES;
MTL_ENABLE_DEBUG_INFO = NO;
MTL_FAST_MATH = YES;
SDKROOT = iphoneos;
SWIFT_COMPILATION_MODE = wholemodule;
VALIDATE_PRODUCT = YES;
}};
name = Release;
}};
A7000005000000000000001 /* Debug */ = {{
isa = XCBuildConfiguration;
buildSettings = {{
ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon;
ASSETCATALOG_COMPILER_GLOBAL_ACCENT_COLOR_NAME = AccentColor;
CODE_SIGN_STYLE = Automatic;
CURRENT_PROJECT_VERSION = 1;
DEVELOPMENT_ASSET_PATHS = "";
ENABLE_PREVIEWS = YES;
GENERATE_INFOPLIST_FILE = NO;
INFOPLIST_FILE = Info.plist;
INFOPLIST_KEY_UIApplicationSceneManifest_Generation = YES;
INFOPLIST_KEY_UIApplicationSupportsIndirectInputEvents = YES;
INFOPLIST_KEY_UILaunchScreen_Generation = YES;
INFOPLIST_KEY_UISupportedInterfaceOrientations_iPad = "UIInterfaceOrientationPortrait UIInterfaceOrientationPortraitUpsideDown UIInterfaceOrientationLandscapeLeft UIInterfaceOrientationLandscapeRight";
INFOPLIST_KEY_UISupportedInterfaceOrientations_iPhone = "UIInterfaceOrientationPortrait UIInterfaceOrientationLandscapeLeft UIInterfaceOrientationLandscapeRight";
LD_RUNPATH_SEARCH_PATHS = (
"$(inherited)",
"@executable_path/Frameworks",
"@executable_path",
);
LIBRARY_SEARCH_PATHS = (
"$(inherited)",
"$(PROJECT_DIR)/../../target/aarch64-apple-ios-sim/debug",
"$(PROJECT_DIR)/../../target/aarch64-apple-ios/debug",
"$(PROJECT_DIR)/../../target/x86_64-apple-ios/debug",
);
MARKETING_VERSION = 1.0;
OTHER_LDFLAGS = "-l{module_name}_ffi";
PRODUCT_BUNDLE_IDENTIFIER = {bundle_id};
PRODUCT_NAME = "$(TARGET_NAME)";
SWIFT_EMIT_LOC_STRINGS = YES;
SWIFT_OBJC_BRIDGING_HEADER = BridgingHeader.h;
SWIFT_VERSION = 5.0;
TARGETED_DEVICE_FAMILY = "1,2";
}};
name = Debug;
}};
A7000006000000000000001 /* Release */ = {{
isa = XCBuildConfiguration;
buildSettings = {{
ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon;
ASSETCATALOG_COMPILER_GLOBAL_ACCENT_COLOR_NAME = AccentColor;
CODE_SIGN_STYLE = Automatic;
CURRENT_PROJECT_VERSION = 1;
DEVELOPMENT_ASSET_PATHS = "";
ENABLE_PREVIEWS = YES;
GENERATE_INFOPLIST_FILE = NO;
INFOPLIST_FILE = Info.plist;
INFOPLIST_KEY_UIApplicationSceneManifest_Generation = YES;
INFOPLIST_KEY_UIApplicationSupportsIndirectInputEvents = YES;
INFOPLIST_KEY_UILaunchScreen_Generation = YES;
INFOPLIST_KEY_UISupportedInterfaceOrientations_iPad = "UIInterfaceOrientationPortrait UIInterfaceOrientationPortraitUpsideDown UIInterfaceOrientationLandscapeLeft UIInterfaceOrientationLandscapeRight";
INFOPLIST_KEY_UISupportedInterfaceOrientations_iPhone = "UIInterfaceOrientationPortrait UIInterfaceOrientationLandscapeLeft UIInterfaceOrientationLandscapeRight";
LD_RUNPATH_SEARCH_PATHS = (
"$(inherited)",
"@executable_path/Frameworks",
"@executable_path",
);
LIBRARY_SEARCH_PATHS = (
"$(inherited)",
"$(PROJECT_DIR)/../../target/aarch64-apple-ios-sim/release",
"$(PROJECT_DIR)/../../target/aarch64-apple-ios/release",
"$(PROJECT_DIR)/../../target/x86_64-apple-ios/release",
);
MARKETING_VERSION = 1.0;
OTHER_LDFLAGS = "-l{module_name}_ffi";
PRODUCT_BUNDLE_IDENTIFIER = {bundle_id};
PRODUCT_NAME = "$(TARGET_NAME)";
SWIFT_EMIT_LOC_STRINGS = YES;
SWIFT_OBJC_BRIDGING_HEADER = BridgingHeader.h;
SWIFT_VERSION = 5.0;
TARGETED_DEVICE_FAMILY = "1,2";
}};
name = Release;
}};
/* End XCBuildConfiguration section */
/* Begin XCConfigurationList section */
A7000001000000000000001 /* Build configuration list for PBXNativeTarget "{app_name}" */ = {{
isa = XCConfigurationList;
buildConfigurations = (
A7000005000000000000001 /* Debug */,
A7000006000000000000001 /* Release */,
);
defaultConfigurationIsVisible = 0;
defaultConfigurationName = Release;
}};
A7000002000000000000001 /* Build configuration list for PBXProject "{app_name}" */ = {{
isa = XCConfigurationList;
buildConfigurations = (
A7000003000000000000001 /* Debug */,
A7000004000000000000001 /* Release */,
);
defaultConfigurationIsVisible = 0;
defaultConfigurationName = Release;
}};
/* End XCConfigurationList section */
}};
rootObject = A0000001000000000000001 /* Project object */;
}}
"#, app_name = app_name, app_name_pascal = app_name_pascal, bundle_id = bundle_id, escaped_shell_script = escaped_shell_script);
Ok(pbxproj)
}
fn to_pascal_case(s: &str) -> String {
s.split('-')
.map(|word| {
let mut chars = word.chars();
match chars.next() {
None => String::new(),
Some(first) => first.to_uppercase().collect::<String>() + chars.as_str(),
}
})
.collect()
}