]> git.tdb.fi Git - builder.git/blob - source/androidassetpackagingtool.cpp
Redesign how tools are run
[builder.git] / source / androidassetpackagingtool.cpp
1 #include <msp/core/algorithm.h>
2 #include <msp/fs/utils.h>
3 #include "androidassetpackagingtool.h"
4 #include "androidmanifestfile.h"
5 #include "androidresourcebundle.h"
6 #include "androidtools.h"
7 #include "component.h"
8 #include "externaltask.h"
9 #include "sourcepackage.h"
10
11 using namespace std;
12 using namespace Msp;
13
14 AndroidAssetPackagingTool::AndroidAssetPackagingTool(Builder &b, const AndroidSdk &s):
15         Tool(b, "AAPT"),
16         sdk(s)
17 {
18         if(sdk.get_root_dir().empty())
19                 problems.push_back("Android SDK not found");
20         else if(sdk.get_build_tools_dir().empty())
21                 problems.push_back("Android build-tools not found");
22         else
23                 set_command((sdk.get_build_tools_dir()/"aapt").str());
24
25         if(sdk.get_platform_jar().empty())
26                 problems.push_back("Android platform not found");
27
28         set_run(_run);
29 }
30
31 Target *AndroidAssetPackagingTool::create_target(const vector<Target *> &sources, const string &)
32 {
33         AndroidManifestFile *manifest = 0;
34         vector<FileTarget *> resources;
35         resources.reserve(sources.size());
36         for(Target *s: sources)
37         {
38                 if(AndroidManifestFile *m = dynamic_cast<AndroidManifestFile *>(s))
39                         manifest = m;
40                 else if(FileTarget *f = dynamic_cast<FileTarget *>(s))
41                         resources.push_back(f);
42         }
43
44         if(!manifest)
45                 throw invalid_argument("AndroidAssetPackagingTool::create_target");
46
47         AndroidResourceBundle *res = new AndroidResourceBundle(builder, *manifest->get_component(), *manifest, resources);
48         res->set_tool(*this);
49         return res;
50 }
51
52 Task *AndroidAssetPackagingTool::_run(const AndroidResourceBundle &res)
53 {
54         const AndroidAssetPackagingTool &tool = dynamic_cast<const AndroidAssetPackagingTool &>(*res.get_tool());
55
56         ExternalTask::Arguments argv;
57         argv.push_back(tool.get_executable()->get_path().str());
58         argv.push_back("package");
59
60         FS::Path work_dir = res.get_component()->get_package().get_source_directory();
61
62         argv.push_back("-I");
63         argv.push_back(tool.sdk.get_platform_jar().str());
64
65         argv.push_back("-F");
66         argv.push_back(FS::relative(res.get_path(), work_dir).str());
67
68         vector<FS::Path> resource_dirs;
69         resource_dirs.reserve(res.get_dependencies().size());
70         for(Target *d: res.get_dependencies())
71         {
72                 FileTarget *file = dynamic_cast<FileTarget *>(d);
73                 Target *real = d->get_real_target();
74
75                 if(dynamic_cast<AndroidManifestFile *>(real))
76                 {
77                         argv.push_back("-M");
78                         argv.push_back(FS::relative(file->get_path(), work_dir).str());
79                 }
80                 else if(real->get_package()==res.get_package())
81                 {
82                         const FS::Path &path = file->get_path();
83                         FS::Path res_dir = path.subpath(0, path.size()-2);
84                         if(!any_equals(resource_dirs, res_dir))
85                                 resource_dirs.push_back(res_dir);
86                 }
87         }
88
89         for(const FS::Path &d: resource_dirs)
90         {
91                 argv.push_back("-S");
92                 argv.push_back(FS::relative(d, work_dir).str());
93         }
94
95         return new ExternalTask(argv, work_dir);
96 }