mirror of
https://github.com/italicsjenga/winit-sonoma-fix.git
synced 2025-01-23 02:16:33 +11:00
0d366ffbda
This re-works the portable `run()` API that consumes the `EventLoop` and runs the loop on the calling thread until the app exits. This can be supported across _all_ platforms and compared to the previous `run() -> !` API is now able to return a `Result` status on all platforms except iOS and Web. Fixes: #2709 By moving away from `run() -> !` we stop calling `std::process::exit()` internally as a means to kill the process without returning which means it's possible to return an exit status and applications can return from their `main()` function normally. This also fixes Android support where an Activity runs in a thread but we can't assume to have full ownership of the process (other services could be running in separate threads). Additionally all examples have generally been updated so that `main()` returns a `Result` from `run()` Fixes: #2709
88 lines
3.4 KiB
Rust
88 lines
3.4 KiB
Rust
#[cfg(any(x11_platform, macos_platform, windows_platform))]
|
|
#[path = "util/fill.rs"]
|
|
mod fill;
|
|
|
|
#[cfg(any(x11_platform, macos_platform, windows_platform))]
|
|
fn main() -> Result<(), impl std::error::Error> {
|
|
use std::collections::HashMap;
|
|
|
|
use raw_window_handle::HasRawWindowHandle;
|
|
use winit::{
|
|
dpi::{LogicalPosition, LogicalSize, Position},
|
|
event::{ElementState, Event, KeyEvent, WindowEvent},
|
|
event_loop::{ControlFlow, EventLoop, EventLoopWindowTarget},
|
|
window::{Window, WindowBuilder, WindowId},
|
|
};
|
|
|
|
fn spawn_child_window(
|
|
parent: &Window,
|
|
event_loop: &EventLoopWindowTarget<()>,
|
|
windows: &mut HashMap<WindowId, Window>,
|
|
) {
|
|
let parent = parent.raw_window_handle();
|
|
let mut builder = WindowBuilder::new()
|
|
.with_title("child window")
|
|
.with_inner_size(LogicalSize::new(200.0f32, 200.0f32))
|
|
.with_position(Position::Logical(LogicalPosition::new(0.0, 0.0)))
|
|
.with_visible(true);
|
|
// `with_parent_window` is unsafe. Parent window must be a valid window.
|
|
builder = unsafe { builder.with_parent_window(Some(parent)) };
|
|
let child_window = builder.build(event_loop).unwrap();
|
|
|
|
let id = child_window.id();
|
|
windows.insert(id, child_window);
|
|
println!("child window created with id: {id:?}");
|
|
}
|
|
|
|
let mut windows = HashMap::new();
|
|
|
|
let event_loop: EventLoop<()> = EventLoop::new();
|
|
let parent_window = WindowBuilder::new()
|
|
.with_title("parent window")
|
|
.with_position(Position::Logical(LogicalPosition::new(0.0, 0.0)))
|
|
.with_inner_size(LogicalSize::new(640.0f32, 480.0f32))
|
|
.build(&event_loop)
|
|
.unwrap();
|
|
|
|
println!("parent window: {parent_window:?})");
|
|
|
|
event_loop.run(move |event: Event<'_, ()>, event_loop, control_flow| {
|
|
*control_flow = ControlFlow::Wait;
|
|
|
|
if let Event::WindowEvent { event, window_id } = event {
|
|
match event {
|
|
WindowEvent::CloseRequested => {
|
|
windows.clear();
|
|
*control_flow = ControlFlow::Exit;
|
|
}
|
|
WindowEvent::CursorEntered { device_id: _ } => {
|
|
// On x11, println when the cursor entered in a window even if the child window is created
|
|
// by some key inputs.
|
|
// the child windows are always placed at (0, 0) with size (200, 200) in the parent window,
|
|
// so we also can see this log when we move the cursor arround (200, 200) in parent window.
|
|
println!("cursor entered in the window {window_id:?}");
|
|
}
|
|
WindowEvent::KeyboardInput {
|
|
event:
|
|
KeyEvent {
|
|
state: ElementState::Pressed,
|
|
..
|
|
},
|
|
..
|
|
} => {
|
|
spawn_child_window(&parent_window, event_loop, &mut windows);
|
|
}
|
|
_ => (),
|
|
}
|
|
} else if let Event::RedrawRequested(wid) = event {
|
|
if let Some(window) = windows.get(&wid) {
|
|
fill::fill_window(window);
|
|
}
|
|
}
|
|
})
|
|
}
|
|
|
|
#[cfg(not(any(x11_platform, macos_platform, windows_platform)))]
|
|
fn main() {
|
|
panic!("This example is supported only on x11, macOS, and Windows.");
|
|
}
|