project.rs 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180
  1. // Copyright 2019-2021 Tauri Programme within The Commons Conservancy
  2. // SPDX-License-Identifier: Apache-2.0
  3. // SPDX-License-Identifier: MIT
  4. use crate::{helpers::template, Result};
  5. use anyhow::Context;
  6. use cargo_mobile::{
  7. android::{
  8. config::{Config, Metadata},
  9. target::Target,
  10. },
  11. config::app::DEFAULT_ASSET_DIR,
  12. os,
  13. target::TargetTrait as _,
  14. util::{
  15. self,
  16. cli::{Report, TextWrapper},
  17. prefix_path,
  18. },
  19. };
  20. use handlebars::Handlebars;
  21. use include_dir::{include_dir, Dir};
  22. use std::{ffi::OsStr, fs, path::Path};
  23. const TEMPLATE_DIR: Dir<'_> = include_dir!("templates/mobile/android");
  24. pub fn gen(
  25. config: &Config,
  26. metadata: &Metadata,
  27. (handlebars, mut map): (Handlebars, template::JsonMap),
  28. wrapper: &TextWrapper,
  29. ) -> Result<()> {
  30. println!("Installing Android toolchains...");
  31. Target::install_all().with_context(|| "failed to run rustup")?;
  32. println!("Generating Android Studio project...");
  33. let dest = config.project_dir();
  34. let asset_packs = metadata.asset_packs().unwrap_or_default();
  35. map.insert(
  36. "root-dir-rel",
  37. Path::new(&os::replace_path_separator(
  38. util::relativize_path(
  39. config.app().root_dir(),
  40. config.project_dir().join(config.app().name()),
  41. )
  42. .into_os_string(),
  43. )),
  44. );
  45. map.insert("root-dir", config.app().root_dir());
  46. map.insert("targets", Target::all().values().collect::<Vec<_>>());
  47. map.insert("target-names", Target::all().keys().collect::<Vec<_>>());
  48. map.insert(
  49. "arches",
  50. Target::all()
  51. .values()
  52. .map(|target| target.arch)
  53. .collect::<Vec<_>>(),
  54. );
  55. map.insert("android-app-plugins", metadata.app_plugins());
  56. map.insert(
  57. "android-project-dependencies",
  58. metadata.project_dependencies(),
  59. );
  60. map.insert("android-app-dependencies", metadata.app_dependencies());
  61. map.insert(
  62. "android-app-dependencies-platform",
  63. metadata.app_dependencies_platform(),
  64. );
  65. map.insert(
  66. "has-code",
  67. metadata.project_dependencies().is_some()
  68. || metadata.app_dependencies().is_some()
  69. || metadata.app_dependencies_platform().is_some(),
  70. );
  71. map.insert(
  72. "asset-packs",
  73. asset_packs
  74. .iter()
  75. .map(|p| p.name.as_str())
  76. .collect::<Vec<_>>(),
  77. );
  78. map.insert("windows", cfg!(windows));
  79. let domain = config.app().reverse_domain().replace('.', "/");
  80. let package_path = format!("java/{}/{}", domain, config.app().name());
  81. map.insert("package-path", &package_path);
  82. let mut created_dirs = Vec::new();
  83. template::render_with_generator(
  84. &handlebars,
  85. map.inner(),
  86. &TEMPLATE_DIR,
  87. &dest,
  88. &mut |path| {
  89. let mut iter = path.iter();
  90. let root = iter.next().unwrap().to_str().unwrap();
  91. let path_without_root: std::path::PathBuf = iter.collect();
  92. let path = match (
  93. root,
  94. path.extension().and_then(|o| o.to_str()),
  95. path_without_root.strip_prefix("src/main"),
  96. ) {
  97. ("app" | "buildSrc", Some("kt"), Ok(path)) => {
  98. let parent = path.parent().unwrap();
  99. let file_name = path.file_name().unwrap();
  100. let out_dir = dest
  101. .join(root)
  102. .join("src/main")
  103. .join(&package_path)
  104. .join(parent);
  105. out_dir.join(file_name)
  106. }
  107. _ => dest.join(path),
  108. };
  109. let parent = path.parent().unwrap().to_path_buf();
  110. if !created_dirs.contains(&parent) {
  111. fs::create_dir_all(&parent)?;
  112. created_dirs.push(parent);
  113. }
  114. let mut options = fs::OpenOptions::new();
  115. #[cfg(unix)]
  116. if path.file_name().unwrap() == OsStr::new("gradlew") {
  117. use std::os::unix::fs::OpenOptionsExt;
  118. options.mode(0o755);
  119. }
  120. options.create_new(true).write(true).open(path)
  121. },
  122. )
  123. .with_context(|| "failed to process template")?;
  124. if !asset_packs.is_empty() {
  125. Report::action_request(
  126. "When running from Android Studio, you must first set your deployment option to \"APK from app bundle\".",
  127. "Android Studio will not be able to find your asset packs otherwise. The option can be found under \"Run > Edit Configurations > Deploy\"."
  128. ).print(wrapper);
  129. }
  130. let source_dest = dest.join("app");
  131. for source in metadata.app_sources() {
  132. let source_src = config.app().root_dir().join(&source);
  133. let source_file = source_src
  134. .file_name()
  135. .ok_or_else(|| anyhow::anyhow!("asset source {} is invalid", source_src.display()))?;
  136. fs::copy(&source_src, source_dest.join(source_file)).map_err(|cause| {
  137. anyhow::anyhow!(
  138. "failed to copy {} to {}: {}",
  139. source_src.display(),
  140. source_dest.display(),
  141. cause
  142. )
  143. })?;
  144. }
  145. let dest = prefix_path(dest, "app/src/main/");
  146. fs::create_dir_all(&dest).map_err(|cause| {
  147. anyhow::anyhow!(
  148. "failed to create directory at {}: {}",
  149. dest.display(),
  150. cause
  151. )
  152. })?;
  153. let asset_dir = dest.join(DEFAULT_ASSET_DIR);
  154. if !asset_dir.is_dir() {
  155. fs::create_dir_all(&asset_dir).map_err(|cause| {
  156. anyhow::anyhow!(
  157. "failed to create asset dir {path}: {cause}",
  158. path = asset_dir.display()
  159. )
  160. })?;
  161. }
  162. Ok(())
  163. }