// Copyright (c) 2014-2015, THUNDERBEAST GAMES LLC All rights reserved // Please see LICENSE.md in repository root for license information // https://github.com/AtomicGameEngine/AtomicGameEngine #include #include #include #include #include "../ToolSystem.h" #include "../ToolEnvironment.h" #include "../Subprocess/SubprocessSystem.h" #include "../Project/Project.h" #include "../Project/ProjectBuildSettings.h" #include "../Platform/PlatformAndroid.h" #include "AndroidProjectGenerator.h" #include "BuildSystem.h" #include "BuildEvents.h" #include "BuildAndroid.h" namespace ToolCore { BuildAndroid::BuildAndroid(Context* context, Project* project) : BuildBase(context, project) { ToolSystem* toolSystem = GetSubsystem(); // this cast isn't great platformAndroid_ = (PlatformAndroid*) toolSystem->GetPlatformByID(PLATFORMID_ANDROID); } BuildAndroid::~BuildAndroid() { } void BuildAndroid::SendBuildFailure(const String& message) { VariantMap buildError; buildError[BuildFailed::P_PLATFORMID] = PLATFORMID_ANDROID; buildError[BuildFailed::P_MESSAGE] = message; SendEvent(E_BUILDFAILED, buildError); } void BuildAndroid::HandleADBStartActivityComplete(StringHash eventType, VariantMap& eventData) { } // adb shell am start -n com.package.name/com.package.name.ActivityName void BuildAndroid::RunADBStartActivity() { SubprocessSystem* subs = GetSubsystem(); String adbCommand = platformAndroid_->GetADBCommand(); ToolSystem* toolSystem = GetSubsystem(); Project* project = toolSystem->GetProject(); AndroidBuildSettings* settings = project->GetBuildSettings()->GetAndroidBuildSettings(); String stringArgs; const char* cpackage = settings->GetPackageName().CString(); stringArgs.AppendWithFormat("shell am start -n %s/%s.AtomicGameEngine",cpackage, cpackage); Vector args = stringArgs.Split(' '); currentBuildPhase_ = ADBStartActivity; Subprocess* subprocess = subs->Launch(adbCommand, args, buildPath_); if (!subprocess) { SendBuildFailure("BuildFailed::RunStartActivity"); return; } VariantMap buildOutput; buildOutput[BuildOutput::P_TEXT] = "\n\nStarting Android Activity\n\n"; SendEvent(E_BUILDOUTPUT, buildOutput); SubscribeToEvent(subprocess, E_SUBPROCESSCOMPLETE, HANDLER(BuildAndroid, HandleADBStartActivityComplete)); SubscribeToEvent(subprocess, E_SUBPROCESSOUTPUT, HANDLER(BuildBase, HandleSubprocessOutputEvent)); } void BuildAndroid::HandleRunADBInstallComplete(StringHash eventType, VariantMap& eventData) { int code = eventData[SubprocessComplete::P_RETCODE].GetInt(); if (!code) { RunADBStartActivity(); } else { BuildSystem* buildSystem = GetSubsystem(); buildSystem->BuildComplete(PLATFORMID_ANDROID, buildPath_, false); } } void BuildAndroid::RunADBInstall() { SubprocessSystem* subs = GetSubsystem(); String adbCommand = platformAndroid_->GetADBCommand(); Vector args = String("install -r ./bin/Atomic-debug-unaligned.apk").Split(' '); currentBuildPhase_ = ADBInstall; Subprocess* subprocess = subs->Launch(adbCommand, args, buildPath_); if (!subprocess) { SendBuildFailure("BuildFailed::RunADBInstall"); return; } VariantMap buildOutput; buildOutput[BuildOutput::P_TEXT] = "\n\nInstalling on Android Device\n\n"; SendEvent(E_BUILDOUTPUT, buildOutput); SubscribeToEvent(subprocess, E_SUBPROCESSCOMPLETE, HANDLER(BuildAndroid, HandleRunADBInstallComplete)); SubscribeToEvent(subprocess, E_SUBPROCESSOUTPUT, HANDLER(BuildBase, HandleSubprocessOutputEvent)); } void BuildAndroid::HandleADBListDevicesComplete(StringHash eventType, VariantMap& eventData) { BuildSystem* buildSystem = GetSubsystem(); int code = eventData[SubprocessComplete::P_RETCODE].GetInt(); if (!code) { // check if we have any devices attached, otherwise adb install // will hang looking for devices bool noDevices = true; if (deviceListText_.Length()) { MemoryBuffer reader(deviceListText_.CString(), deviceListText_.Length() + 1); while (!reader.IsEof()) { String line = reader.ReadLine(); if (line.Length() && line[0] >= '0' && line[0] <= '9') { noDevices = false; break; } } } if (!noDevices) RunADBInstall(); else { // can't proceed, though success buildSystem->BuildComplete(PLATFORMID_ANDROID, buildPath_); } } else { buildSystem->BuildComplete(PLATFORMID_ANDROID, buildPath_, false); } } void BuildAndroid::HandleADBListDevicesOutputEvent(StringHash eventType, VariantMap& eventData) { // E_SUBPROCESSOUTPUT const String& text = eventData[SubprocessOutput::P_TEXT].GetString(); deviceListText_ += text; // convert to a build output event and forward to subscribers VariantMap buildOutputData; buildOutputData[BuildOutput::P_TEXT] = text; SendEvent(E_BUILDOUTPUT, buildOutputData); } void BuildAndroid::RunADBListDevices() { ToolSystem* toolSystem = GetSubsystem(); SubprocessSystem* subs = GetSubsystem(); String adbCommand = platformAndroid_->GetADBCommand(); deviceListText_.Clear(); Vector args = String("devices").Split(' '); currentBuildPhase_ = ADBListDevices; Subprocess* subprocess = subs->Launch(adbCommand, args, ""); if (!subprocess) { SendBuildFailure("BuildFailed::RunADBListDevices"); return; } VariantMap buildOutput; buildOutput[BuildOutput::P_TEXT] = "\n\nListing Android Devices\n\n"; SendEvent(E_BUILDOUTPUT, buildOutput); SubscribeToEvent(subprocess, E_SUBPROCESSCOMPLETE, HANDLER(BuildAndroid, HandleADBListDevicesComplete)); SubscribeToEvent(subprocess, E_SUBPROCESSOUTPUT, HANDLER(BuildAndroid, HandleADBListDevicesOutputEvent)); } void BuildAndroid::HandleAntDebugComplete(StringHash eventType, VariantMap& eventData) { int code = eventData[SubprocessComplete::P_RETCODE].GetInt(); if (!code) { RunADBListDevices(); } else { BuildSystem* buildSystem = GetSubsystem(); buildSystem->BuildComplete(PLATFORMID_ANDROID, buildPath_, false); } } void BuildAndroid::RunAntDebug() { ToolEnvironment* tenv = GetSubsystem(); SubprocessSystem* subs = GetSubsystem(); ToolPrefs* tprefs = tenv->GetToolPrefs(); Poco::Process::Env env; #ifdef ATOMIC_PLATFORM_OSX String antCommand = tprefs->GetAntPath(); Vector args; args.Push("debug"); #else // C:\ProgramData\Oracle\Java\javapath; Vector args; String antCommand = "cmd"; String antPath = prefs->GetAntPath() + "/ant.bat"; env["JAVA_HOME"] = prefs->GetJDKRootPath().CString(); // ant is a batch file on windows, so have to run with cmd /c args.Push("/c"); args.Push("\"" + antPath + "\""); args.Push("debug"); #endif currentBuildPhase_ = AntBuildDebug; Subprocess* subprocess = subs->Launch(antCommand, args, buildPath_, env); if (!subprocess) { SendBuildFailure("BuildFailed::RunAntDebug"); return; } VariantMap buildOutput; buildOutput[BuildOutput::P_TEXT] = "Starting Android Deployment\n\n"; SendEvent(E_BUILDOUTPUT, buildOutput); SubscribeToEvent(subprocess, E_SUBPROCESSCOMPLETE, HANDLER(BuildAndroid, HandleAntDebugComplete)); SubscribeToEvent(subprocess, E_SUBPROCESSOUTPUT, HANDLER(BuildBase, HandleSubprocessOutputEvent)); } void BuildAndroid::Initialize() { ToolSystem* tsystem = GetSubsystem(); Project* project = tsystem->GetProject(); Vector defaultResourcePaths; GetDefaultResourcePaths(defaultResourcePaths); String projectResources = project->GetResourcePath(); for (unsigned i = 0; i < defaultResourcePaths.Size(); i++) { AddResourceDir(defaultResourcePaths[i]); } // TODO: smart filtering of cache AddResourceDir(project->GetProjectPath() + "Cache/"); AddResourceDir(projectResources); BuildResourceEntries(); } void BuildAndroid::Build(const String& buildPath) { ToolEnvironment* tenv = GetSubsystem(); ToolSystem* tsystem = GetSubsystem(); Project* project = tsystem->GetProject(); buildPath_ = AddTrailingSlash(buildPath) + GetBuildSubfolder(); Initialize(); //generate manifest file String manifest; for (unsigned i = 0; i < resourceEntries_.Size(); i++) { BuildResourceEntry* entry = resourceEntries_[i]; manifest += entry->packagePath_; if ( i != resourceEntries_.Size() - 1) manifest += ";"; } FileSystem* fileSystem = GetSubsystem(); if (fileSystem->DirExists(buildPath_)) fileSystem->RemoveDir(buildPath_, true); String buildSourceDir = tenv->GetToolDataDir(); String androidProject = buildSourceDir + "Deployment/Android"; // Copy the base android project fileSystem->CopyDir(androidProject, buildPath_); Vector defaultResourcePaths; GetDefaultResourcePaths(defaultResourcePaths); String projectResources = project->GetResourcePath(); for (unsigned i = 0; i < defaultResourcePaths.Size(); i++) { fileSystem->CopyDir(defaultResourcePaths[i], buildPath_ + "/assets/" + GetFileName(RemoveTrailingSlash(defaultResourcePaths[i]))); } fileSystem->CopyDir(project->GetProjectPath() + "Cache/", buildPath_ + "/assets/Cache"); fileSystem->CopyDir(projectResources, buildPath_ + "/assets/AtomicResources"); // write the manifest SharedPtr mfile(new File(context_, buildPath_ + "/assets/AtomicManifest", FILE_WRITE)); mfile->WriteString(manifest); mfile->Close(); AndroidProjectGenerator gen(context_); gen.SetBuildPath(buildPath_); if (!gen.Generate()) { SendBuildFailure(gen.GetErrorText()); return; } RunAntDebug(); } }