Skip to content

Commit 76286b9

Browse files
committed
Experimental Android support for eframe
1 parent 4e101d2 commit 76286b9

File tree

7 files changed

+178
-0
lines changed

7 files changed

+178
-0
lines changed

Cargo.lock

+38
Original file line numberDiff line numberDiff line change
@@ -188,6 +188,23 @@ version = "0.1.1"
188188
source = "registry+https://github.com/rust-lang/crates.io-index"
189189
checksum = "e999941b234f3131b00bc13c22d06e8c5ff726d1b6318ac7eb276997bbb4fef0"
190190

191+
[[package]]
192+
name = "android_log-sys"
193+
version = "0.3.1"
194+
source = "registry+https://github.com/rust-lang/crates.io-index"
195+
checksum = "5ecc8056bf6ab9892dcd53216c83d1597487d7dacac16c8df6b877d127df9937"
196+
197+
[[package]]
198+
name = "android_logger"
199+
version = "0.14.1"
200+
source = "registry+https://github.com/rust-lang/crates.io-index"
201+
checksum = "05b07e8e73d720a1f2e4b6014766e6039fd2e96a4fa44e2a78d0e1fa2ff49826"
202+
dependencies = [
203+
"android_log-sys",
204+
"env_filter",
205+
"log",
206+
]
207+
191208
[[package]]
192209
name = "android_system_properties"
193210
version = "0.1.5"
@@ -1477,6 +1494,16 @@ dependencies = [
14771494
"syn 2.0.48",
14781495
]
14791496

1497+
[[package]]
1498+
name = "env_filter"
1499+
version = "0.1.2"
1500+
source = "registry+https://github.com/rust-lang/crates.io-index"
1501+
checksum = "4f2c92ceda6ceec50f43169f9ee8424fe2db276791afde7b2cd8bc084cb376ab"
1502+
dependencies = [
1503+
"log",
1504+
"regex",
1505+
]
1506+
14801507
[[package]]
14811508
name = "env_logger"
14821509
version = "0.10.0"
@@ -2091,6 +2118,17 @@ version = "0.4.1"
20912118
source = "registry+https://github.com/rust-lang/crates.io-index"
20922119
checksum = "95505c38b4572b2d910cecb0281560f54b440a19336cbbcb27bf6ce6adc6f5a8"
20932120

2121+
[[package]]
2122+
name = "hello_android"
2123+
version = "0.1.0"
2124+
dependencies = [
2125+
"android_logger",
2126+
"eframe",
2127+
"egui_extras",
2128+
"log",
2129+
"winit",
2130+
]
2131+
20942132
[[package]]
20952133
name = "hello_world"
20962134
version = "0.1.0"

crates/eframe/src/epi.rs

+16
Original file line numberDiff line numberDiff line change
@@ -364,6 +364,16 @@ pub struct NativeOptions {
364364
///
365365
/// Defaults to true.
366366
pub dithering: bool,
367+
368+
/// Android application for `winit`'s event loop.
369+
///
370+
/// This value is required on Android to correctly create the event loop. See
371+
/// [`EventLoopBuilder::build`] and [`with_android_app`] for details.
372+
///
373+
/// [`EventLoopBuilder::build`]: winit::event_loop::EventLoopBuilder::build
374+
/// [`with_android_app`]: winit::platform::android::EventLoopBuilderExtAndroid::with_android_app
375+
#[cfg(target_os = "android")]
376+
pub android_app: Option<winit::platform::android::activity::AndroidApp>,
367377
}
368378

369379
#[cfg(not(target_arch = "wasm32"))]
@@ -383,6 +393,9 @@ impl Clone for NativeOptions {
383393

384394
persistence_path: self.persistence_path.clone(),
385395

396+
#[cfg(target_os = "android")]
397+
android_app: self.android_app.clone(),
398+
386399
..*self
387400
}
388401
}
@@ -424,6 +437,9 @@ impl Default for NativeOptions {
424437
persistence_path: None,
425438

426439
dithering: true,
440+
441+
#[cfg(target_os = "android")]
442+
android_app: None,
427443
}
428444
}
429445
}

crates/eframe/src/native/run.rs

+11
Original file line numberDiff line numberDiff line change
@@ -17,9 +17,20 @@ use crate::{
1717

1818
// ----------------------------------------------------------------------------
1919
fn create_event_loop(native_options: &mut epi::NativeOptions) -> Result<EventLoop<UserEvent>> {
20+
#[cfg(target_os = "android")]
21+
use winit::platform::android::EventLoopBuilderExtAndroid as _;
22+
2023
crate::profile_function!();
2124
let mut builder = winit::event_loop::EventLoop::with_user_event();
2225

26+
#[cfg(target_os = "android")]
27+
let mut builder =
28+
builder.with_android_app(native_options.android_app.take().ok_or_else(|| {
29+
crate::Error::AppCreation(Box::from(
30+
"`NativeOptions` is missing required `android_app`",
31+
))
32+
})?);
33+
2334
if let Some(hook) = std::mem::take(&mut native_options.event_loop_builder) {
2435
hook(&mut builder);
2536
}

examples/hello_android/Cargo.toml

+29
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,29 @@
1+
[package]
2+
name = "hello_android"
3+
version = "0.1.0"
4+
authors = ["Emil Ernerfeldt <emil.ernerfeldt@gmail.com>"]
5+
license = "MIT OR Apache-2.0"
6+
edition = "2021"
7+
rust-version = "1.76"
8+
publish = false
9+
10+
# `unsafe_code` is required for `#[no_mangle]`, disable workspace lints to workaround lint error.
11+
# [lints]
12+
# workspace = true
13+
14+
[lib]
15+
crate-type = ["cdylib"]
16+
17+
18+
[dependencies]
19+
eframe = { workspace = true, features = [
20+
"default",
21+
"android-native-activity",
22+
] }
23+
24+
# For image support:
25+
egui_extras = { workspace = true, features = ["default", "image"] }
26+
27+
log = { workspace = true }
28+
winit = { workspace = true }
29+
android_logger = "0.14"

examples/hello_android/README.md

+20
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,20 @@
1+
Hello world example for Android.
2+
3+
Use `cargo-apk` to build and run. Requires a patch to workaround [an upstream bug](https://github.com/rust-mobile/cargo-subcommand/issues/29).
4+
5+
One-time setup:
6+
7+
```sh
8+
cargo install \
9+
--git https://github.com/parasyte/cargo-apk.git \
10+
--rev 282639508eeed7d73f2e1eaeea042da2716436d5 \
11+
cargo-apk
12+
```
13+
14+
Build and run:
15+
16+
```sh
17+
cargo apk run -p hello_android
18+
```
19+
20+
![](screenshot.png)

examples/hello_android/screenshot.png

117 KB
Loading

examples/hello_android/src/lib.rs

+64
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,64 @@
1+
#![allow(rustdoc::missing_crate_level_docs)] // it's an example
2+
3+
use android_logger::Config;
4+
use eframe::egui;
5+
use log::LevelFilter;
6+
use winit::platform::android::activity::AndroidApp;
7+
8+
#[no_mangle]
9+
fn android_main(app: AndroidApp) {
10+
// Log to android output
11+
android_logger::init_once(Config::default().with_max_level(LevelFilter::Info));
12+
13+
let options = eframe::NativeOptions {
14+
android_app: Some(app),
15+
..Default::default()
16+
};
17+
eframe::run_native(
18+
"My egui App",
19+
options,
20+
Box::new(|cc| {
21+
// This gives us image support:
22+
egui_extras::install_image_loaders(&cc.egui_ctx);
23+
24+
Ok(Box::<MyApp>::default())
25+
}),
26+
)
27+
.unwrap()
28+
}
29+
30+
struct MyApp {
31+
name: String,
32+
age: u32,
33+
}
34+
35+
impl Default for MyApp {
36+
fn default() -> Self {
37+
Self {
38+
name: "Arthur".to_owned(),
39+
age: 42,
40+
}
41+
}
42+
}
43+
44+
impl eframe::App for MyApp {
45+
fn update(&mut self, ctx: &egui::Context, _frame: &mut eframe::Frame) {
46+
egui::CentralPanel::default().show(ctx, |ui| {
47+
ui.heading("My egui Application");
48+
ui.horizontal(|ui| {
49+
let name_label = ui.label("Your name: ");
50+
ui.text_edit_singleline(&mut self.name)
51+
.labelled_by(name_label.id);
52+
});
53+
ui.add(egui::Slider::new(&mut self.age, 0..=120).text("age"));
54+
if ui.button("Increment").clicked() {
55+
self.age += 1;
56+
}
57+
ui.label(format!("Hello '{}', age {}", self.name, self.age));
58+
59+
ui.image(egui::include_image!(
60+
"../../../crates/egui/assets/ferris.png"
61+
));
62+
});
63+
}
64+
}

0 commit comments

Comments
 (0)