package; import haxe.Json; import hxp.ArrayTools; import hxp.Haxelib; import hxp.HXML; import hxp.Log; import hxp.NDLL; import hxp.Path; import hxp.StringTools; import hxp.System; #if lime import lime.graphics.Image; #end import lime.tools.Architecture; import lime.tools.Asset; import lime.tools.AssetHelper; import lime.tools.AssetType; import lime.tools.CPPHelper; import lime.tools.DeploymentHelper; import lime.tools.HXProject; import lime.tools.Icon; import lime.tools.IconHelper; import lime.tools.Keystore; import lime.tools.Orientation; import lime.tools.Platform; import lime.tools.PlatformTarget; import lime.tools.ProjectHelper; import lime.tools.TVOSHelper; import sys.io.File; import sys.FileSystem; class TVOSPlatform extends PlatformTarget { public function new(command:String, _project:HXProject, targetFlags:Map) { super(command, _project, targetFlags); var defaults = new HXProject(); defaults.meta = { title: "MyApplication", description: "", packageName: "com.example.myapp", version: "1.0.0", company: "", companyUrl: "", buildNumber: null, companyId: "" }; defaults.app = { main: "Main", file: "MyApplication", path: "bin", preloader: "", swfVersion: 17, url: "", init: null }; defaults.window = { width: 800, height: 600, parameters: "{}", background: 0xFFFFFF, fps: 30, hardware: true, display: 0, resizable: true, borderless: false, orientation: Orientation.AUTO, vsync: false, fullscreen: false, allowHighDPI: true, alwaysOnTop: false, antialiasing: 0, allowShaders: true, requireShaders: false, depthBuffer: true, stencilBuffer: true, colorDepth: 32, maximized: false, minimized: false, hidden: false, title: "" }; defaults.architectures = [Architecture.ARM64]; defaults.window.width = 0; defaults.window.height = 0; defaults.window.fullscreen = true; defaults.window.requireShaders = true; for (i in 1...project.windows.length) { defaults.windows.push(defaults.window); } defaults.merge(project); project = defaults; for (excludeArchitecture in project.excludeArchitectures) { project.architectures.remove(excludeArchitecture); } targetDirectory = Path.combine(project.app.path, project.config.getString("tvos.output-directory", "tvos")); } public override function build():Void { if (project.targetFlags.exists("xcode") && System.hostPlatform == MAC) { System.runCommand("", "open", [targetDirectory + "/" + project.app.file + ".xcodeproj"]); } else { TVOSHelper.build(project, targetDirectory); if (noOutput) return; if (!project.targetFlags.exists("simulator")) { TVOSHelper.sign(project, targetDirectory + "/bin"); } } } public override function clean():Void { if (FileSystem.exists(targetDirectory)) { System.removeDirectory(targetDirectory); } } public override function deploy():Void { TVOSHelper.deploy(project, targetDirectory); } public override function display():Void { if (project.targetFlags.exists("output-file")) { Sys.println(Path.combine(targetDirectory, project.app.file + ".xcodeproj")); } else { Sys.println(getDisplayHXML().toString()); } } private function generateContext():Dynamic { // project = project.clone (); project.sources.unshift(""); project.sources = Path.relocatePaths(project.sources, Path.combine(targetDirectory, project.app.file + "/haxe")); // project.dependencies.push ("stdc++"); if (project.targetFlags.exists("xml")) { project.haxeflags.push("-xml " + targetDirectory + "/types.xml"); } if (project.targetFlags.exists("final")) { project.haxedefs.set("final", ""); } if (!project.config.exists("tvos.identity")) { project.config.set("tvos.identity", "tvOS Developer"); } var context = project.templateContext; context.HAS_ICON = false; context.HAS_LAUNCH_IMAGE = false; context.OBJC_ARC = false; context.KEY_STORE_IDENTITY = project.config.getString("tvos.identity"); context.linkedLibraries = []; for (dependency in project.dependencies) { if (!StringTools.endsWith(dependency.name, ".framework") && !StringTools.endsWith(dependency.name, ".tbd") && !StringTools.endsWith(dependency.path, ".framework") && !StringTools.endsWith(dependency.path, ".xcframework")) { if (dependency.path != "") { var name = Path.withoutDirectory(Path.withoutExtension(dependency.path)); project.config.push("tvos.linker-flags", "-force_load $SRCROOT/$PRODUCT_NAME/lib/$CURRENT_ARCH/" + Path.withoutDirectory(dependency.path)); if (StringTools.startsWith(name, "lib")) { name = name.substring(3, name.length); } context.linkedLibraries.push(name); } else if (dependency.name != "") { context.linkedLibraries.push(dependency.name); } } } var valid_archs = new Array(); var arm64 = false; var architectures = project.architectures; if (architectures == null || architectures.length == 0) { architectures = [Architecture.ARM64]; } for (architecture in project.architectures) { switch (architecture) { case ARM64: valid_archs.push("arm64"); arm64 = true; default: } } context.CURRENT_ARCHS = "( " + valid_archs.join(",") + ") "; valid_archs.push("i386"); context.VALID_ARCHS = valid_archs.join(" "); context.THUMB_SUPPORT = ""; var requiredCapabilities = []; requiredCapabilities.push({name: "arm64", value: true}); context.REQUIRED_CAPABILITY = requiredCapabilities; context.ARM64 = arm64; context.TARGET_DEVICES = switch (project.config.getString("tvos.device", "appletv")) { case "appletv": "3"; default: "3"; } context.DEPLOYMENT = project.config.getString("tvos.deployment", "9.0"); if (project.config.getString("tvos.compiler") == "llvm" || project.config.getString("tvos.compiler", "clang") == "clang") { context.OBJC_ARC = true; } context.IOS_COMPILER = project.config.getString("tvos.compiler", "clang"); context.CPP_BUILD_LIBRARY = project.config.getString("cpp.buildLibrary", "hxcpp"); context.CPP_CACHE_WORKAROUND = "unset HXCPP_COMPILE_CACHE;"; context.IOS_LINKER_FLAGS = ["-stdlib=libc++"].concat(project.config.getArrayString("tvos.linker-flags")); context.IOS_NON_EXEMPT_ENCRYPTION = project.config.getBool("tvos.non-exempt-encryption", true); switch (project.window.orientation) { case PORTRAIT: context.IOS_APP_ORIENTATION = "UIInterfaceOrientationPortraitUIInterfaceOrientationPortraitUpsideDown"; case LANDSCAPE: context.IOS_APP_ORIENTATION = "UIInterfaceOrientationLandscapeLeftUIInterfaceOrientationLandscapeRight"; case ALL: context.IOS_APP_ORIENTATION = "UIInterfaceOrientationLandscapeLeftUIInterfaceOrientationLandscapeRightUIInterfaceOrientationPortraitUIInterfaceOrientationPortraitUpsideDown"; // case "allButUpsideDown": // context.IOS_APP_ORIENTATION = "UIInterfaceOrientationLandscapeLeftUIInterfaceOrientationLandscapeRightUIInterfaceOrientationPortrait"; default: context.IOS_APP_ORIENTATION = "UIInterfaceOrientationLandscapeLeftUIInterfaceOrientationLandscapeRightUIInterfaceOrientationPortraitUIInterfaceOrientationPortraitUpsideDown"; } context.ADDL_PBX_BUILD_FILE = ""; context.ADDL_PBX_FILE_REFERENCE = ""; context.ADDL_PBX_FRAMEWORKS_BUILD_PHASE = ""; context.ADDL_PBX_FRAMEWORK_GROUP = ""; context.frameworkSearchPaths = []; for (dependency in project.dependencies) { var name = null; var path = null; var fileType = null; if (Path.extension(dependency.name) == "framework") { name = dependency.name; path = "/System/Library/Frameworks/" + dependency.name; fileType = "wrapper.framework"; } else if (Path.extension(dependency.name) == "tbd") { name = dependency.name; path = "usr/lib/" + dependency.name; fileType = "sourcecode.text-based-dylib-definition"; } else if (Path.extension(dependency.path) == "framework") { name = Path.withoutDirectory(dependency.path); path = Path.tryFullPath(dependency.path); fileType = "wrapper.framework"; } else if (Path.extension(dependency.path) == "xcframework") { name = Path.withoutDirectory(dependency.path); path = Path.tryFullPath(dependency.path); fileType = "wrapper.xcframework"; } if (name != null) { var frameworkID = "11C0000000000018" + StringTools.getUniqueID(); var fileID = "11C0000000000018" + StringTools.getUniqueID(); ArrayTools.addUnique(context.frameworkSearchPaths, Path.directory(path)); context.ADDL_PBX_BUILD_FILE += " " + frameworkID + " /* " + name + " in Frameworks */ = {isa = PBXBuildFile; fileRef = " + fileID + " /* " + name + " */; };\n"; context.ADDL_PBX_FILE_REFERENCE += " " + fileID + " /* " + name + " */ = {isa = PBXFileReference; lastKnownFileType = \"" + fileType + "\"; name = \"" + name + "\"; path = \"" + path + "\"; sourceTree = SDKROOT; };\n"; context.ADDL_PBX_FRAMEWORKS_BUILD_PHASE += " " + frameworkID + " /* " + name + " in Frameworks */,\n"; context.ADDL_PBX_FRAMEWORK_GROUP += " " + fileID + " /* " + name + " */,\n"; } } context.HXML_PATH = System.findTemplate(project.templatePaths, "tvos/PROJ/haxe/Build.hxml"); context.PRERENDERED_ICON = project.config.getBool("tvos.prerenderedIcon", false); var haxelibPath = project.environment.get("HAXELIB_PATH"); if (haxelibPath != null) { context.HAXELIB_PATH = 'export HAXELIB_PATH=$haxelibPath;'; } else { context.HAXELIB_PATH = ''; } return context; } private function getDisplayHXML():HXML { var path = targetDirectory + "/" + project.app.file + "/haxe/Build.hxml"; // try to use the existing .hxml file. however, if the project file was // modified more recently than the .hxml, then the .hxml cannot be // considered valid anymore. it may cause errors in editors like vscode. if (FileSystem.exists(path) && (project.projectFilePath == null || !FileSystem.exists(project.projectFilePath) || (FileSystem.stat(path).mtime.getTime() > FileSystem.stat(project.projectFilePath).mtime.getTime()))) { return File.getContent(path); } else { var context = project.templateContext; var hxml = HXML.fromString(context.HAXE_FLAGS); hxml.addClassName(context.APP_MAIN); hxml.cpp = "_"; hxml.define("tvos"); hxml.define("appletv"); hxml.noOutput = true; return hxml; } } public override function rebuild():Void { var arm64 = (command == "rebuild" || (project.architectures.indexOf(Architecture.ARM64) > -1 && !project.targetFlags.exists("simulator"))); var i386 = (command == "rebuild" || project.targetFlags.exists("simulator")); var x86_64 = (command == "rebuild" || project.targetFlags.exists("simulator")); var commands = []; if (arm64) commands.push([ "-Dtvos", "-Dappletvos", "-DHXCPP_CPP11", "-DHXCPP_ARM64", "-DOBJC_ARC", "-DENABLE_BITCODE" ]); if (i386) commands.push([ "-Dtvos", "-Dappletvsim", "-Dsimulator", "-DHXCPP_CPP11", "-DOBJC_ARC", "-DENABLE_BITCODE" ]); if (x86_64) commands.push([ "-Dtvos", "-Dappletvsim", "-Dsimulator", "-DHXCPP_M64", "-DHXCPP_CPP11", "-DOBJC_ARC", "-DENABLE_BITCODE" ]); CPPHelper.rebuild(project, commands); } public override function run():Void { if (project.targetFlags.exists("xcode")) return; TVOSHelper.launch(project, targetDirectory); } public override function update():Void { AssetHelper.processLibraries(project, targetDirectory); // project = project.clone (); for (asset in project.assets) { if (asset.embed && asset.sourcePath == "") { var path = Path.combine(targetDirectory + "/" + project.app.file + "/obj/tmp", asset.targetPath); System.mkdir(Path.directory(path)); AssetHelper.copyAsset(asset, path); asset.sourcePath = path; } } // var manifest = new Asset (); // manifest.id = "__manifest__"; // manifest.data = AssetHelper.createManifest (project).serialize (); // manifest.resourceName = manifest.flatName = manifest.targetPath = "manifest"; // manifest.type = AssetType.TEXT; // project.assets.push (manifest); var context = generateContext(); context.OUTPUT_DIR = targetDirectory; var projectDirectory = targetDirectory + "/" + project.app.file + "/"; System.mkdir(targetDirectory); System.mkdir(projectDirectory); System.mkdir(projectDirectory + "/haxe"); System.mkdir(projectDirectory + "/haxe/lime/installer"); var iconSizes:Array = [ {name: "Icon-Small.png", size: 29}, {name: "Icon-Small-40.png", size: 40}, {name: "Icon-Small-50.png", size: 50}, {name: "Icon.png", size: 57}, {name: "Icon-Small@2x.png", size: 58}, {name: "Icon-72.png", size: 72}, {name: "Icon-76.png", size: 76}, {name: "Icon-Small-40@2x.png", size: 80}, {name: "Icon-Small-50@2x.png", size: 100}, {name: "Icon@2x.png", size: 114}, {name: "Icon-60@2x.png", size: 120}, {name: "Icon-72@2x.png", size: 144}, {name: "Icon-76@2x.png", size: 152}, {name: "Icon-60@3x.png", size: 180}, ]; context.HAS_ICON = true; var iconPath = Path.combine(projectDirectory, "Images.xcassets/AppIcon.appiconset"); System.mkdir(iconPath); var icons = project.icons; if (icons.length == 0) { icons = [new Icon(System.findTemplate(project.templatePaths, "default/icon.svg"))]; } for (iconSize in iconSizes) { if (!IconHelper.createIcon(icons, iconSize.size, iconSize.size, Path.combine(iconPath, iconSize.name))) { context.HAS_ICON = false; } } var splashSizes:Array = [ {name: "Default.png", w: 320, h: 480}, // iPhone, portrait {name: "Default@2x.png", w: 640, h: 960}, // iPhone Retina, portrait {name: "Default-568h@2x.png", w: 640, h: 1136}, // iPhone 5, portrait {name: "Default-Portrait.png", w: 768, h: 1024}, // iPad, portrait {name: "Default-Landscape.png", w: 1024, h: 768}, // iPad, landscape {name: "Default-Portrait@2x.png", w: 1536, h: 2048}, // iPad Retina, portrait {name: "Default-Landscape@2x.png", w: 2048, h: 1536}, // iPad Retina, landscape {name: "Default-667h@2x.png", w: 750, h: 1334}, // iPhone 6, portrait {name: "Default-736h@3x.png", w: 1242, h: 2208}, // iPhone 6 Plus, portrait {name: "Default-736h-Landscape@3x.png", w: 2208, h: 1242}, // iPhone 6 Plus, landscape ]; var splashScreenPath = Path.combine(projectDirectory, "Images.xcassets/LaunchImage.launchimage"); System.mkdir(splashScreenPath); for (size in splashSizes) { var match = false; for (splashScreen in project.splashScreens) { if (splashScreen.width == size.w && splashScreen.height == size.h && Path.extension(splashScreen.path) == "png") { System.copyFile(splashScreen.path, Path.combine(splashScreenPath, size.name)); match = true; } } if (!match) { var imagePath = Path.combine(splashScreenPath, size.name); if (!FileSystem.exists(imagePath)) { #if (lime && lime_cffi && !macro) Log.info("", " - \x1b[1mGenerating image:\x1b[0m " + imagePath); var image = new Image(null, 0, 0, size.w, size.h, (0xFF << 24) | (project.window.background & 0xFFFFFF)); var bytes = image.encode(PNG); File.saveBytes(imagePath, bytes); #end } } } context.HAS_LAUNCH_IMAGE = true; System.mkdir(projectDirectory + "/resources"); System.mkdir(projectDirectory + "/haxe/build"); ProjectHelper.recursiveSmartCopyTemplate(project, "tvos/resources", projectDirectory + "/resources", context, true, false); ProjectHelper.recursiveSmartCopyTemplate(project, "tvos/PROJ/haxe", projectDirectory + "/haxe", context); ProjectHelper.recursiveSmartCopyTemplate(project, "haxe", projectDirectory + "/haxe", context); ProjectHelper.recursiveSmartCopyTemplate(project, "tvos/PROJ/Classes", projectDirectory + "/Classes", context); ProjectHelper.recursiveSmartCopyTemplate(project, "tvos/PROJ/Images.xcassets", projectDirectory + "/Images.xcassets", context); System.copyFileTemplate(project.templatePaths, "tvos/PROJ/PROJ-Entitlements.plist", projectDirectory + "/" + project.app.file + "-Entitlements.plist", context); System.copyFileTemplate(project.templatePaths, "tvos/PROJ/PROJ-Info.plist", projectDirectory + "/" + project.app.file + "-Info.plist", context); System.copyFileTemplate(project.templatePaths, "tvos/PROJ/PROJ-Prefix.pch", projectDirectory + "/" + project.app.file + "-Prefix.pch", context); ProjectHelper.recursiveSmartCopyTemplate(project, "tvos/PROJ.xcodeproj", targetDirectory + "/" + project.app.file + ".xcodeproj", context); // SWFHelper.generateSWFClasses (project, projectDirectory + "/haxe"); System.mkdir(projectDirectory + "/lib"); for (archID in 0...3) { var arch = ["arm64", "i386", "x86_64"][archID]; if (arch == "arm64" && !context.ARM64) continue; var libExt = [".appletvos-64.a", ".appletvsim.a", ".appletvsim-64.a"][archID]; System.mkdir(projectDirectory + "/lib/" + arch); System.mkdir(projectDirectory + "/lib/" + arch + "-debug"); for (ndll in project.ndlls) { // if (ndll.haxelib != null) { var releaseLib = NDLL.getLibraryPath(ndll, "AppleTV", "lib", libExt); Log.info("releaseLib: " + releaseLib); var debugLib = NDLL.getLibraryPath(ndll, "AppleTV", "lib", libExt, true); var releaseDest = projectDirectory + "/lib/" + arch + "/lib" + ndll.name + ".a"; Log.info("releaseDest: " + releaseDest); var debugDest = projectDirectory + "/lib/" + arch + "-debug/lib" + ndll.name + ".a"; if (!FileSystem.exists(releaseLib)) { releaseLib = NDLL.getLibraryPath(ndll, "AppleTV", "lib", ".appletvos-64.a"); Log.info("alternative releaseLib: " + releaseLib); debugLib = NDLL.getLibraryPath(ndll, "AppleTV", "lib", ".appletvos-64.a", true); } System.copyIfNewer(releaseLib, releaseDest); if (FileSystem.exists(debugLib) && debugLib != releaseLib) { System.copyIfNewer(debugLib, debugDest); } else if (FileSystem.exists(debugDest)) { FileSystem.deleteFile(debugDest); } // } } for (dependency in project.dependencies) { if (StringTools.endsWith(dependency.path, ".a")) { var fileName = Path.withoutDirectory(dependency.path); if (!StringTools.startsWith(fileName, "lib")) { fileName = "lib" + fileName; } copyIfNewer(dependency.path, projectDirectory + "/lib/" + arch + "/" + fileName); } } } System.mkdir(projectDirectory + "/assets"); for (asset in project.assets) { if (asset.type != AssetType.TEMPLATE) { var targetPath = Path.combine(projectDirectory + "/assets/", asset.resourceName); // var sourceAssetPath:String = projectDirectory + "haxe/" + asset.sourcePath; System.mkdir(Path.directory(targetPath)); AssetHelper.copyAssetIfNewer(asset, targetPath); // System.mkdir (Path.directory (sourceAssetPath)); // System.linkFile (flatAssetPath, sourceAssetPath, true, true); } else { var targetPath = Path.combine(projectDirectory, asset.targetPath); System.mkdir(Path.directory(targetPath)); AssetHelper.copyAsset(asset, targetPath, context); } } if (project.targetFlags.exists("xcode") && System.hostPlatform == MAC && command == "update") { System.runCommand("", "open", [targetDirectory + "/" + project.app.file + ".xcodeproj"]); } } /*private function updateLaunchImage () { var destination = buildDirectory + "/ios"; System.mkdir (destination); var has_launch_image = false; if (launchImages.length > 0) has_launch_image = true; for (launchImage in launchImages) { var splitPath = launchImage.name.split ("/"); var path = destination + "/" + splitPath[splitPath.length - 1]; System.copyFile (launchImage.name, path, context, false); } context.HAS_LAUNCH_IMAGE = has_launch_image; }*/ public override function watch():Void { var hxml = getDisplayHXML(); var dirs = hxml.getClassPaths(true); var outputPath = Path.combine(Sys.getCwd(), project.app.path); dirs = dirs.filter(function(dir) { return (!Path.startsWith(dir, outputPath)); }); var command = ProjectHelper.getCurrentCommand(); System.watch(command, dirs); } @ignore public override function install():Void {} @ignore public override function trace():Void {} @ignore public override function uninstall():Void {} } private typedef IconSize = { name:String, size:Int, } private typedef SplashSize = { name:String, w:Int, h:Int, }