Format everything and add rustfmt to travis (#951)

* Format everything and add rustfmt to travis

* Remove extern crate winit from examples and add force_multiline_blocks

* Format the code properly

* Fix inconsistent period in PULL_REQUEST_TEMPLATE.md

* Only run rustfmt on nightly

* Travis fixings
This commit is contained in:
Osspial 2019-06-21 11:33:15 -04:00 committed by GitHub
parent b1b5aefc4b
commit e2c84725de
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
109 changed files with 4787 additions and 3679 deletions

View file

@ -1,4 +1,5 @@
- [ ] Tested on all platforms changed - [ ] Tested on all platforms changed
- [ ] `cargo fmt` has been run on this branch
- [ ] Added an entry to `CHANGELOG.md` if knowledge of this change could be valuable to users - [ ] Added an entry to `CHANGELOG.md` if knowledge of this change could be valuable to users
- [ ] Updated documentation to reflect any user-facing changes, including notes of platform-specific behavior - [ ] Updated documentation to reflect any user-facing changes, including notes of platform-specific behavior
- [ ] Created an example program if it would help users understand this functionality - [ ] Created an example program if it would help users understand this functionality

View file

@ -45,8 +45,11 @@ matrix:
install: install:
- rustup self update - rustup self update
- rustup target add $TARGET; true - rustup target add $TARGET; true
- rustup install nightly
- rustup component add rustfmt --toolchain nightly
script: script:
- cargo +nightly fmt --all -- --check
- cargo build --target $TARGET --verbose - cargo build --target $TARGET --verbose
- cargo build --target $TARGET --features serde --verbose - cargo build --target $TARGET --features serde --verbose
# Running iOS apps on OSX requires the simulator so we skip that for now # Running iOS apps on OSX requires the simulator so we skip that for now

View file

@ -31,10 +31,11 @@ show something on the window you need to use the platform-specific getters provi
another library. another library.
```rust ```rust
extern crate winit; use winit::{
use winit::window::WindowBuilder; event::{Event, WindowEvent},
use winit::event::{Event, WindowEvent}; event_loop::{ControlFlow, EventLoop},
use winit::event_loop::{EventLoop, ControlFlow}; window::WindowBuilder,
};
fn main() { fn main() {
let event_loop = EventLoop::new(); let event_loop = EventLoop::new();

View file

@ -1,8 +1,8 @@
extern crate winit; use winit::{
event::{ElementState, Event, KeyboardInput, WindowEvent},
use winit::window::{WindowBuilder, CursorIcon}; event_loop::{ControlFlow, EventLoop},
use winit::event::{Event, WindowEvent, ElementState, KeyboardInput}; window::{CursorIcon, WindowBuilder},
use winit::event_loop::{EventLoop, ControlFlow}; };
fn main() { fn main() {
let event_loop = EventLoop::new(); let event_loop = EventLoop::new();
@ -14,7 +14,18 @@ fn main() {
event_loop.run(move |event, _, control_flow| { event_loop.run(move |event, _, control_flow| {
match event { match event {
Event::WindowEvent { event: WindowEvent::KeyboardInput { input: KeyboardInput { state: ElementState::Pressed, .. }, .. }, .. } => { Event::WindowEvent {
event:
WindowEvent::KeyboardInput {
input:
KeyboardInput {
state: ElementState::Pressed,
..
},
..
},
..
} => {
println!("Setting cursor to \"{:?}\"", CURSORS[cursor_idx]); println!("Setting cursor to \"{:?}\"", CURSORS[cursor_idx]);
window.set_cursor_icon(CURSORS[cursor_idx]); window.set_cursor_icon(CURSORS[cursor_idx]);
if cursor_idx < CURSORS.len() - 1 { if cursor_idx < CURSORS.len() - 1 {
@ -23,26 +34,52 @@ fn main() {
cursor_idx = 0; cursor_idx = 0;
} }
}, },
Event::WindowEvent { event: WindowEvent::CloseRequested, .. } => { Event::WindowEvent {
event: WindowEvent::CloseRequested,
..
} => {
*control_flow = ControlFlow::Exit; *control_flow = ControlFlow::Exit;
return; return;
}, },
_ => () _ => (),
} }
}); });
} }
const CURSORS: &[CursorIcon] = &[ const CURSORS: &[CursorIcon] = &[
CursorIcon::Default, CursorIcon::Crosshair, CursorIcon::Hand, CursorIcon::Default,
CursorIcon::Arrow, CursorIcon::Move, CursorIcon::Text, CursorIcon::Crosshair,
CursorIcon::Wait, CursorIcon::Help, CursorIcon::Progress, CursorIcon::Hand,
CursorIcon::NotAllowed, CursorIcon::ContextMenu, CursorIcon::Cell, CursorIcon::Arrow,
CursorIcon::VerticalText, CursorIcon::Alias, CursorIcon::Copy, CursorIcon::Move,
CursorIcon::NoDrop, CursorIcon::Grab, CursorIcon::Grabbing, CursorIcon::Text,
CursorIcon::AllScroll, CursorIcon::ZoomIn, CursorIcon::ZoomOut, CursorIcon::Wait,
CursorIcon::EResize, CursorIcon::NResize, CursorIcon::NeResize, CursorIcon::Help,
CursorIcon::NwResize, CursorIcon::SResize, CursorIcon::SeResize, CursorIcon::Progress,
CursorIcon::SwResize, CursorIcon::WResize, CursorIcon::EwResize, CursorIcon::NotAllowed,
CursorIcon::NsResize, CursorIcon::NeswResize, CursorIcon::NwseResize, CursorIcon::ContextMenu,
CursorIcon::ColResize, CursorIcon::RowResize CursorIcon::Cell,
CursorIcon::VerticalText,
CursorIcon::Alias,
CursorIcon::Copy,
CursorIcon::NoDrop,
CursorIcon::Grab,
CursorIcon::Grabbing,
CursorIcon::AllScroll,
CursorIcon::ZoomIn,
CursorIcon::ZoomOut,
CursorIcon::EResize,
CursorIcon::NResize,
CursorIcon::NeResize,
CursorIcon::NwResize,
CursorIcon::SResize,
CursorIcon::SeResize,
CursorIcon::SwResize,
CursorIcon::WResize,
CursorIcon::EwResize,
CursorIcon::NsResize,
CursorIcon::NeswResize,
CursorIcon::NwseResize,
CursorIcon::ColResize,
CursorIcon::RowResize,
]; ];

View file

@ -1,8 +1,8 @@
extern crate winit; use winit::{
event::{ElementState, Event, KeyboardInput, WindowEvent},
use winit::window::WindowBuilder; event_loop::{ControlFlow, EventLoop},
use winit::event::{Event, WindowEvent, ElementState, KeyboardInput}; window::WindowBuilder,
use winit::event_loop::{EventLoop, ControlFlow}; };
fn main() { fn main() {
let event_loop = EventLoop::new(); let event_loop = EventLoop::new();
@ -18,12 +18,13 @@ fn main() {
match event { match event {
WindowEvent::CloseRequested => *control_flow = ControlFlow::Exit, WindowEvent::CloseRequested => *control_flow = ControlFlow::Exit,
WindowEvent::KeyboardInput { WindowEvent::KeyboardInput {
input: KeyboardInput { input:
state: ElementState::Released, KeyboardInput {
virtual_keycode: Some(key), state: ElementState::Released,
modifiers, virtual_keycode: Some(key),
.. modifiers,
}, ..
},
.. ..
} => { } => {
use winit::event::VirtualKeyCode::*; use winit::event::VirtualKeyCode::*;
@ -33,7 +34,7 @@ fn main() {
H => window.set_cursor_visible(modifiers.shift), H => window.set_cursor_visible(modifiers.shift),
_ => (), _ => (),
} }
} },
_ => (), _ => (),
} }
} }

View file

@ -1,10 +1,10 @@
extern crate winit;
use std::io::{self, Write}; use std::io::{self, Write};
use winit::monitor::MonitorHandle; use winit::{
use winit::window::WindowBuilder; event::{ElementState, Event, KeyboardInput, VirtualKeyCode, WindowEvent},
use winit::event::{Event, WindowEvent, VirtualKeyCode, ElementState, KeyboardInput}; event_loop::{ControlFlow, EventLoop},
use winit::event_loop::{EventLoop, ControlFlow}; monitor::MonitorHandle,
window::WindowBuilder,
};
fn main() { fn main() {
let event_loop = EventLoop::new(); let event_loop = EventLoop::new();
@ -24,7 +24,7 @@ fn main() {
let num = num.trim().parse().ok().expect("Please enter a number"); let num = num.trim().parse().ok().expect("Please enter a number");
match num { match num {
2 => macos_use_simple_fullscreen = true, 2 => macos_use_simple_fullscreen = true,
_ => {} _ => {},
} }
// Prompt for monitor when using native fullscreen // Prompt for monitor when using native fullscreen
@ -54,59 +54,69 @@ fn main() {
*control_flow = ControlFlow::Wait; *control_flow = ControlFlow::Wait;
match event { match event {
Event::WindowEvent { event, .. } => match event { Event::WindowEvent { event, .. } => {
WindowEvent::CloseRequested => *control_flow = ControlFlow::Exit, match event {
WindowEvent::KeyboardInput { WindowEvent::CloseRequested => *control_flow = ControlFlow::Exit,
input: WindowEvent::KeyboardInput {
KeyboardInput { input:
virtual_keycode: Some(virtual_code), KeyboardInput {
state, virtual_keycode: Some(virtual_code),
.. state,
}, ..
.. },
} => match (virtual_code, state) { ..
(VirtualKeyCode::Escape, _) => *control_flow = ControlFlow::Exit, } => {
(VirtualKeyCode::F, ElementState::Pressed) => { match (virtual_code, state) {
#[cfg(target_os = "macos")] (VirtualKeyCode::Escape, _) => *control_flow = ControlFlow::Exit,
{ (VirtualKeyCode::F, ElementState::Pressed) => {
if macos_use_simple_fullscreen { #[cfg(target_os = "macos")]
use winit::platform::macos::WindowExtMacOS; {
if WindowExtMacOS::set_simple_fullscreen(&window, !is_fullscreen) { if macos_use_simple_fullscreen {
is_fullscreen = !is_fullscreen; use winit::platform::macos::WindowExtMacOS;
if WindowExtMacOS::set_simple_fullscreen(
&window,
!is_fullscreen,
) {
is_fullscreen = !is_fullscreen;
}
return;
}
} }
return;
}
}
is_fullscreen = !is_fullscreen; is_fullscreen = !is_fullscreen;
if !is_fullscreen { if !is_fullscreen {
window.set_fullscreen(None); window.set_fullscreen(None);
} else { } else {
window.set_fullscreen(Some(window.current_monitor())); window.set_fullscreen(Some(window.current_monitor()));
} }
} },
(VirtualKeyCode::S, ElementState::Pressed) => { (VirtualKeyCode::S, ElementState::Pressed) => {
println!("window.fullscreen {:?}", window.fullscreen()); println!("window.fullscreen {:?}", window.fullscreen());
#[cfg(target_os = "macos")] #[cfg(target_os = "macos")]
{ {
use winit::platform::macos::WindowExtMacOS; use winit::platform::macos::WindowExtMacOS;
println!("window.simple_fullscreen {:?}", WindowExtMacOS::simple_fullscreen(&window)); println!(
"window.simple_fullscreen {:?}",
WindowExtMacOS::simple_fullscreen(&window)
);
}
},
(VirtualKeyCode::M, ElementState::Pressed) => {
is_maximized = !is_maximized;
window.set_maximized(is_maximized);
},
(VirtualKeyCode::D, ElementState::Pressed) => {
decorations = !decorations;
window.set_decorations(decorations);
},
_ => (),
} }
} },
(VirtualKeyCode::M, ElementState::Pressed) => {
is_maximized = !is_maximized;
window.set_maximized(is_maximized);
}
(VirtualKeyCode::D, ElementState::Pressed) => {
decorations = !decorations;
window.set_decorations(decorations);
}
_ => (), _ => (),
}, }
_ => (),
}, },
_ => {} _ => {},
} }
}); });
} }
@ -123,7 +133,10 @@ fn prompt_for_monitor(event_loop: &EventLoop<()>) -> MonitorHandle {
let mut num = String::new(); let mut num = String::new();
io::stdin().read_line(&mut num).unwrap(); io::stdin().read_line(&mut num).unwrap();
let num = num.trim().parse().ok().expect("Please enter a number"); let num = num.trim().parse().ok().expect("Please enter a number");
let monitor = event_loop.available_monitors().nth(num).expect("Please enter a valid ID"); let monitor = event_loop
.available_monitors()
.nth(num)
.expect("Please enter a valid ID");
println!("Using {:?}", monitor.name()); println!("Using {:?}", monitor.name());

View file

@ -1,8 +1,8 @@
extern crate winit; use winit::{
event::{Event, KeyboardInput, WindowEvent},
use winit::window::WindowBuilder; event_loop::{ControlFlow, EventLoop},
use winit::event::{Event, WindowEvent, KeyboardInput}; window::WindowBuilder,
use winit::event_loop::{EventLoop, ControlFlow}; };
fn main() { fn main() {
let event_loop = EventLoop::new(); let event_loop = EventLoop::new();
@ -15,60 +15,66 @@ fn main() {
let mut close_requested = false; let mut close_requested = false;
event_loop.run(move |event, _, control_flow| { event_loop.run(move |event, _, control_flow| {
use winit::event::ElementState::Released; use winit::event::{
use winit::event::VirtualKeyCode::{N, Y}; ElementState::Released,
VirtualKeyCode::{N, Y},
};
*control_flow = ControlFlow::Wait; *control_flow = ControlFlow::Wait;
match event { match event {
Event::WindowEvent { event, .. } => match event { Event::WindowEvent { event, .. } => {
WindowEvent::CloseRequested => { match event {
// `CloseRequested` is sent when the close button on the window is pressed (or WindowEvent::CloseRequested => {
// through whatever other mechanisms the window manager provides for closing a // `CloseRequested` is sent when the close button on the window is pressed (or
// window). If you don't handle this event, the close button won't actually do // through whatever other mechanisms the window manager provides for closing a
// anything. // window). If you don't handle this event, the close button won't actually do
// anything.
// A common thing to do here is prompt the user if they have unsaved work. // A common thing to do here is prompt the user if they have unsaved work.
// Creating a proper dialog box for that is far beyond the scope of this // Creating a proper dialog box for that is far beyond the scope of this
// example, so here we'll just respond to the Y and N keys. // example, so here we'll just respond to the Y and N keys.
println!("Are you ready to bid your window farewell? [Y/N]"); println!("Are you ready to bid your window farewell? [Y/N]");
close_requested = true; close_requested = true;
// In applications where you can safely close the window without further // In applications where you can safely close the window without further
// action from the user, this is generally where you'd handle cleanup before // action from the user, this is generally where you'd handle cleanup before
// closing the window. How to close the window is detailed in the handler for // closing the window. How to close the window is detailed in the handler for
// the Y key. // the Y key.
} },
WindowEvent::KeyboardInput { WindowEvent::KeyboardInput {
input: input:
KeyboardInput { KeyboardInput {
virtual_keycode: Some(virtual_code), virtual_keycode: Some(virtual_code),
state: Released, state: Released,
.. ..
}, },
.. ..
} => match virtual_code { } => {
Y => { match virtual_code {
if close_requested { Y => {
// This is where you'll want to do any cleanup you need. if close_requested {
println!("Buh-bye!"); // This is where you'll want to do any cleanup you need.
println!("Buh-bye!");
// For a single-window application like this, you'd normally just // For a single-window application like this, you'd normally just
// break out of the event loop here. If you wanted to keep running the // break out of the event loop here. If you wanted to keep running the
// event loop (i.e. if it's a multi-window application), you need to // event loop (i.e. if it's a multi-window application), you need to
// drop the window. That closes it, and results in `Destroyed` being // drop the window. That closes it, and results in `Destroyed` being
// sent. // sent.
*control_flow = ControlFlow::Exit; *control_flow = ControlFlow::Exit;
}
},
N => {
if close_requested {
println!("Your window will continue to stay by your side.");
close_requested = false;
}
},
_ => (),
} }
} },
N => {
if close_requested {
println!("Your window will continue to stay by your side.");
close_requested = false;
}
}
_ => (), _ => (),
}, }
_ => (),
}, },
_ => (), _ => (),
} }

View file

@ -1,16 +1,14 @@
extern crate winit; use winit::{
dpi::LogicalSize,
use winit::dpi::LogicalSize; event::{Event, WindowEvent},
use winit::window::WindowBuilder; event_loop::{ControlFlow, EventLoop},
use winit::event::{Event, WindowEvent}; window::WindowBuilder,
use winit::event_loop::{EventLoop, ControlFlow}; };
fn main() { fn main() {
let event_loop = EventLoop::new(); let event_loop = EventLoop::new();
let window = WindowBuilder::new() let window = WindowBuilder::new().build(&event_loop).unwrap();
.build(&event_loop)
.unwrap();
window.set_min_inner_size(Some(LogicalSize::new(400.0, 200.0))); window.set_min_inner_size(Some(LogicalSize::new(400.0, 200.0)));
window.set_max_inner_size(Some(LogicalSize::new(800.0, 400.0))); window.set_max_inner_size(Some(LogicalSize::new(800.0, 400.0)));
@ -19,8 +17,10 @@ fn main() {
println!("{:?}", event); println!("{:?}", event);
match event { match event {
Event::WindowEvent { event: WindowEvent::CloseRequested, .. } => Event::WindowEvent {
*control_flow = ControlFlow::Exit, event: WindowEvent::CloseRequested,
..
} => *control_flow = ControlFlow::Exit,
_ => *control_flow = ControlFlow::Wait, _ => *control_flow = ControlFlow::Wait,
} }
}); });

View file

@ -1,6 +1,4 @@
extern crate winit; use winit::{event_loop::EventLoop, window::WindowBuilder};
use winit::event_loop::EventLoop;
use winit::window::WindowBuilder;
fn main() { fn main() {
let event_loop = EventLoop::new(); let event_loop = EventLoop::new();

View file

@ -1,11 +1,10 @@
extern crate env_logger; extern crate env_logger;
extern crate winit;
use std::{collections::HashMap, sync::mpsc, thread, time::Duration}; use std::{collections::HashMap, sync::mpsc, thread, time::Duration};
use winit::{ use winit::{
event::{ElementState, Event, KeyboardInput, VirtualKeyCode, WindowEvent}, event::{ElementState, Event, KeyboardInput, VirtualKeyCode, WindowEvent},
event_loop::{ControlFlow, EventLoop}, window::{CursorIcon, WindowBuilder}, event_loop::{ControlFlow, EventLoop},
window::{CursorIcon, WindowBuilder},
}; };
const WINDOW_COUNT: usize = 3; const WINDOW_COUNT: usize = 3;
@ -25,26 +24,34 @@ fn main() {
thread::spawn(move || { thread::spawn(move || {
while let Ok(event) = rx.recv() { while let Ok(event) = rx.recv() {
match event { match event {
WindowEvent::KeyboardInput { input: KeyboardInput { WindowEvent::KeyboardInput {
state: ElementState::Released, input:
virtual_keycode: Some(key), KeyboardInput {
modifiers, state: ElementState::Released,
virtual_keycode: Some(key),
modifiers,
..
},
.. ..
}, .. } => { } => {
window.set_title(&format!("{:?}", key)); window.set_title(&format!("{:?}", key));
let state = !modifiers.shift; let state = !modifiers.shift;
use self::VirtualKeyCode::*; use self::VirtualKeyCode::*;
match key { match key {
A => window.set_always_on_top(state), A => window.set_always_on_top(state),
C => window.set_cursor_icon(match state { C => {
true => CursorIcon::Progress, window.set_cursor_icon(match state {
false => CursorIcon::Default, true => CursorIcon::Progress,
}), false => CursorIcon::Default,
})
},
D => window.set_decorations(!state), D => window.set_decorations(!state),
F => window.set_fullscreen(match state { F => {
true => Some(window.current_monitor()), window.set_fullscreen(match state {
false => None, true => Some(window.current_monitor()),
}), false => None,
})
},
G => window.set_cursor_grab(state).unwrap(), G => window.set_cursor_grab(state).unwrap(),
H => window.set_cursor_visible(!state), H => window.set_cursor_visible(!state),
I => { I => {
@ -54,28 +61,40 @@ fn main() {
println!("-> outer_size : {:?}", window.outer_size()); println!("-> outer_size : {:?}", window.outer_size());
println!("-> inner_size : {:?}", window.inner_size()); println!("-> inner_size : {:?}", window.inner_size());
}, },
L => window.set_min_inner_size(match state { L => {
true => Some(WINDOW_SIZE.into()), window.set_min_inner_size(match state {
false => None, true => Some(WINDOW_SIZE.into()),
}), false => None,
})
},
M => window.set_maximized(state), M => window.set_maximized(state),
P => window.set_outer_position({ P => {
let mut position = window.outer_position().unwrap(); window.set_outer_position({
let sign = if state { 1.0 } else { -1.0 }; let mut position = window.outer_position().unwrap();
position.x += 10.0 * sign; let sign = if state { 1.0 } else { -1.0 };
position.y += 10.0 * sign; position.x += 10.0 * sign;
position position.y += 10.0 * sign;
}), position
})
},
Q => window.request_redraw(), Q => window.request_redraw(),
R => window.set_resizable(state), R => window.set_resizable(state),
S => window.set_inner_size(match state { S => {
true => (WINDOW_SIZE.0 + 100, WINDOW_SIZE.1 + 100), window.set_inner_size(
false => WINDOW_SIZE, match state {
}.into()), true => (WINDOW_SIZE.0 + 100, WINDOW_SIZE.1 + 100),
W => window.set_cursor_position(( false => WINDOW_SIZE,
WINDOW_SIZE.0 as i32 / 2, }
WINDOW_SIZE.1 as i32 / 2, .into(),
).into()).unwrap(), )
},
W => {
window
.set_cursor_position(
(WINDOW_SIZE.0 as i32 / 2, WINDOW_SIZE.1 as i32 / 2).into(),
)
.unwrap()
},
Z => { Z => {
window.set_visible(false); window.set_visible(false);
thread::sleep(Duration::from_secs(1)); thread::sleep(Duration::from_secs(1));
@ -99,16 +118,23 @@ fn main() {
match event { match event {
WindowEvent::CloseRequested WindowEvent::CloseRequested
| WindowEvent::Destroyed | WindowEvent::Destroyed
| WindowEvent::KeyboardInput { input: KeyboardInput { | WindowEvent::KeyboardInput {
virtual_keycode: Some(VirtualKeyCode::Escape), input:
.. }, .. } => { KeyboardInput {
virtual_keycode: Some(VirtualKeyCode::Escape),
..
},
..
} => {
window_senders.remove(&window_id); window_senders.remove(&window_id);
}, },
_ => if let Some(tx) = window_senders.get(&window_id) { _ => {
tx.send(event).unwrap(); if let Some(tx) = window_senders.get(&window_id) {
tx.send(event).unwrap();
}
}, },
} }
} },
_ => (), _ => (),
} }
}) })

View file

@ -1,9 +1,9 @@
extern crate winit;
use std::collections::HashMap; use std::collections::HashMap;
use winit::window::Window; use winit::{
use winit::event::{Event, WindowEvent, ElementState, KeyboardInput}; event::{ElementState, Event, KeyboardInput, WindowEvent},
use winit::event_loop::{EventLoop, ControlFlow}; event_loop::{ControlFlow, EventLoop},
window::Window,
};
fn main() { fn main() {
let event_loop = EventLoop::new(); let event_loop = EventLoop::new();
@ -29,13 +29,20 @@ fn main() {
*control_flow = ControlFlow::Exit; *control_flow = ControlFlow::Exit;
} }
}, },
WindowEvent::KeyboardInput { input: KeyboardInput { state: ElementState::Pressed, .. }, .. } => { WindowEvent::KeyboardInput {
input:
KeyboardInput {
state: ElementState::Pressed,
..
},
..
} => {
let window = Window::new(&event_loop).unwrap(); let window = Window::new(&event_loop).unwrap();
windows.insert(window.id(), window); windows.insert(window.id(), window);
}, },
_ => () _ => (),
} }
} },
_ => (), _ => (),
} }
}) })

View file

@ -1,7 +1,8 @@
extern crate winit; use winit::{
use winit::window::WindowBuilder; event::{Event, WindowEvent},
use winit::event::{Event, WindowEvent}; event_loop::{ControlFlow, EventLoop},
use winit::event_loop::{EventLoop, ControlFlow}; window::WindowBuilder,
};
fn main() { fn main() {
let event_loop: EventLoop<i32> = EventLoop::new_user_event(); let event_loop: EventLoop<i32> = EventLoop::new_user_event();
@ -26,8 +27,10 @@ fn main() {
event_loop.run(move |event, _, control_flow| { event_loop.run(move |event, _, control_flow| {
println!("{:?}", event); println!("{:?}", event);
match event { match event {
Event::WindowEvent { event: WindowEvent::CloseRequested, .. } => Event::WindowEvent {
*control_flow = ControlFlow::Exit, event: WindowEvent::CloseRequested,
..
} => *control_flow = ControlFlow::Exit,
_ => *control_flow = ControlFlow::Wait, _ => *control_flow = ControlFlow::Wait,
} }
}); });

View file

@ -1,9 +1,10 @@
extern crate winit; use std::time::{Duration, Instant};
use std::time::{Instant, Duration};
use winit::window::WindowBuilder; use winit::{
use winit::event::{Event, WindowEvent}; event::{Event, WindowEvent},
use winit::event_loop::{EventLoop, ControlFlow}; event_loop::{ControlFlow, EventLoop},
window::WindowBuilder,
};
fn main() { fn main() {
let event_loop = EventLoop::new(); let event_loop = EventLoop::new();
@ -25,7 +26,7 @@ fn main() {
window.request_redraw(); window.request_redraw();
*control_flow = ControlFlow::WaitUntil(Instant::now() + Duration::new(1, 0)) *control_flow = ControlFlow::WaitUntil(Instant::now() + Duration::new(1, 0))
}, },
_ => () _ => (),
} }
}); });
} }

View file

@ -1,7 +1,8 @@
extern crate winit; use winit::{
use winit::window::WindowBuilder; event::{ElementState, Event, KeyboardInput, VirtualKeyCode, WindowEvent},
use winit::event::{Event, WindowEvent, VirtualKeyCode, ElementState, KeyboardInput}; event_loop::{ControlFlow, EventLoop},
use winit::event_loop::{EventLoop, ControlFlow}; window::WindowBuilder,
};
fn main() { fn main() {
let event_loop = EventLoop::new(); let event_loop = EventLoop::new();
@ -18,22 +19,24 @@ fn main() {
event_loop.run(move |event, _, control_flow| { event_loop.run(move |event, _, control_flow| {
*control_flow = ControlFlow::Wait; *control_flow = ControlFlow::Wait;
match event { match event {
Event::WindowEvent { event, .. } => match event { Event::WindowEvent { event, .. } => {
WindowEvent::CloseRequested => *control_flow = ControlFlow::Exit, match event {
WindowEvent::KeyboardInput { WindowEvent::CloseRequested => *control_flow = ControlFlow::Exit,
input: WindowEvent::KeyboardInput {
KeyboardInput { input:
virtual_keycode: Some(VirtualKeyCode::Space), KeyboardInput {
state: ElementState::Released, virtual_keycode: Some(VirtualKeyCode::Space),
.. state: ElementState::Released,
}, ..
.. },
} => { ..
resizable = !resizable; } => {
println!("Resizable: {}", resizable); resizable = !resizable;
window.set_resizable(resizable); println!("Resizable: {}", resizable);
window.set_resizable(resizable);
},
_ => (),
} }
_ => (),
}, },
_ => (), _ => (),
}; };

View file

@ -1,8 +1,9 @@
extern crate winit;
use std::time::{Duration, Instant}; use std::time::{Duration, Instant};
use winit::window::WindowBuilder; use winit::{
use winit::event::{Event, WindowEvent, StartCause}; event::{Event, StartCause, WindowEvent},
use winit::event_loop::{EventLoop, ControlFlow}; event_loop::{ControlFlow, EventLoop},
window::WindowBuilder,
};
fn main() { fn main() {
let event_loop = EventLoop::new(); let event_loop = EventLoop::new();
@ -18,9 +19,10 @@ fn main() {
println!("{:?}", event); println!("{:?}", event);
match event { match event {
Event::NewEvents(StartCause::Init) => Event::NewEvents(StartCause::Init) => {
*control_flow = ControlFlow::WaitUntil(Instant::now() + timer_length), *control_flow = ControlFlow::WaitUntil(Instant::now() + timer_length)
Event::NewEvents(StartCause::ResumeTimeReached{..}) => { },
Event::NewEvents(StartCause::ResumeTimeReached { .. }) => {
*control_flow = ControlFlow::WaitUntil(Instant::now() + timer_length); *control_flow = ControlFlow::WaitUntil(Instant::now() + timer_length);
println!("\nTimer\n"); println!("\nTimer\n");
}, },
@ -28,7 +30,7 @@ fn main() {
event: WindowEvent::CloseRequested, event: WindowEvent::CloseRequested,
.. ..
} => *control_flow = ControlFlow::Exit, } => *control_flow = ControlFlow::Exit,
_ => () _ => (),
} }
}); });
} }

View file

@ -1,14 +1,17 @@
extern crate winit; use winit::{
use winit::window::WindowBuilder; event::{Event, WindowEvent},
use winit::event::{Event, WindowEvent}; event_loop::{ControlFlow, EventLoop},
use winit::event_loop::{EventLoop, ControlFlow}; window::WindowBuilder,
};
fn main() { fn main() {
let event_loop = EventLoop::new(); let event_loop = EventLoop::new();
let window = WindowBuilder::new().with_decorations(false) let window = WindowBuilder::new()
.with_transparent(true) .with_decorations(false)
.build(&event_loop).unwrap(); .with_transparent(true)
.build(&event_loop)
.unwrap();
window.set_title("A fantastic window!"); window.set_title("A fantastic window!");
@ -16,8 +19,10 @@ fn main() {
println!("{:?}", event); println!("{:?}", event);
match event { match event {
Event::WindowEvent { event: WindowEvent::CloseRequested, .. } => Event::WindowEvent {
*control_flow = ControlFlow::Exit, event: WindowEvent::CloseRequested,
..
} => *control_flow = ControlFlow::Exit,
_ => *control_flow = ControlFlow::Wait, _ => *control_flow = ControlFlow::Wait,
} }
}); });

View file

@ -1,5 +1,3 @@
extern crate winit;
use winit::event_loop::EventLoop; use winit::event_loop::EventLoop;
fn main() { fn main() {

View file

@ -1,7 +1,8 @@
extern crate winit; use winit::{
use winit::window::WindowBuilder; event::{Event, WindowEvent},
use winit::event::{Event, WindowEvent}; event_loop::{ControlFlow, EventLoop},
use winit::event_loop::{EventLoop, ControlFlow}; window::WindowBuilder,
};
fn main() { fn main() {
let event_loop = EventLoop::new(); let event_loop = EventLoop::new();

View file

@ -1,13 +1,12 @@
extern crate winit;
extern crate image; extern crate image;
use std::path::Path; use std::path::Path;
use winit::window::{WindowBuilder, Icon}; use winit::{
use winit::event::Event; event::Event,
use winit::event_loop::{EventLoop, ControlFlow}; event_loop::{ControlFlow, EventLoop},
window::{Icon, WindowBuilder},
};
fn main() { fn main() {
// You'll have to choose an icon size at your own discretion. On X11, the desired size varies // You'll have to choose an icon size at your own discretion. On X11, the desired size varies
// by WM, and on Windows, you still have to account for screen scaling. Here we use 32px, // by WM, and on Windows, you still have to account for screen scaling. Here we use 32px,
// since it seems to work well enough in most cases. Be careful about going too high, or // since it seems to work well enough in most cases. Be careful about going too high, or
@ -43,8 +42,6 @@ fn main() {
match event { match event {
CloseRequested => *control_flow = ControlFlow::Exit, CloseRequested => *control_flow = ControlFlow::Exit,
DroppedFile(path) => { DroppedFile(path) => {
window.set_window_icon(Some(load_icon(&path))); window.set_window_icon(Some(load_icon(&path)));
}, },
_ => (), _ => (),

View file

@ -1,9 +1,9 @@
extern crate winit; use winit::{
event::{Event, WindowEvent},
use winit::window::WindowBuilder; event_loop::{ControlFlow, EventLoop},
use winit::event::{Event, WindowEvent}; platform::desktop::EventLoopExtDesktop,
use winit::event_loop::{EventLoop, ControlFlow}; window::WindowBuilder,
use winit::platform::desktop::EventLoopExtDesktop; };
fn main() { fn main() {
let mut event_loop = EventLoop::new(); let mut event_loop = EventLoop::new();

7
rustfmt.toml Normal file
View file

@ -0,0 +1,7 @@
merge_imports=true
match_block_trailing_comma=true
force_explicit_abi=true
format_macro_matchers=true
use_field_init_shorthand=true
format_code_in_doc_comments=true
force_multiline_blocks=true

View file

@ -1,4 +1,3 @@
//! DPI is important, so read the docs for this module if you don't want to be confused. //! DPI is important, so read the docs for this module if you don't want to be confused.
//! //!
//! Originally, `winit` dealt entirely in physical pixels (excluding unintentional inconsistencies), but now all //! Originally, `winit` dealt entirely in physical pixels (excluding unintentional inconsistencies), but now all

View file

@ -1,5 +1,4 @@
use std::fmt; use std::{error, fmt};
use std::error;
use crate::platform_impl; use crate::platform_impl;
@ -30,20 +29,14 @@ impl NotSupportedError {
#[inline] #[inline]
#[allow(dead_code)] #[allow(dead_code)]
pub(crate) fn new() -> NotSupportedError { pub(crate) fn new() -> NotSupportedError {
NotSupportedError { NotSupportedError { _marker: () }
_marker: ()
}
} }
} }
impl OsError { impl OsError {
#[allow(dead_code)] #[allow(dead_code)]
pub(crate) fn new(line: u32, file: &'static str, error: platform_impl::OsError) -> OsError { pub(crate) fn new(line: u32, file: &'static str, error: platform_impl::OsError) -> OsError {
OsError { OsError { line, file, error }
line,
file,
error,
}
} }
} }
@ -51,12 +44,15 @@ impl OsError {
macro_rules! os_error { macro_rules! os_error {
($error:expr) => {{ ($error:expr) => {{
crate::error::OsError::new(line!(), file!(), $error) crate::error::OsError::new(line!(), file!(), $error)
}} }};
} }
impl fmt::Display for OsError { impl fmt::Display for OsError {
fn fmt(&self, formatter: &mut fmt::Formatter<'_>) -> Result<(), fmt::Error> { fn fmt(&self, formatter: &mut fmt::Formatter<'_>) -> Result<(), fmt::Error> {
formatter.pad(&format!("os error at {}:{}: {}", self.file, self.line, self.error)) formatter.pad(&format!(
"os error at {}:{}: {}",
self.file, self.line, self.error
))
} }
} }

View file

@ -4,12 +4,13 @@
//! processed and used to modify the program state. For more details, see the root-level documentation. //! processed and used to modify the program state. For more details, see the root-level documentation.
//! //!
//! [event_loop_run]: ../event_loop/struct.EventLoop.html#method.run //! [event_loop_run]: ../event_loop/struct.EventLoop.html#method.run
use std::time::Instant; use std::{path::PathBuf, time::Instant};
use std::path::PathBuf;
use crate::dpi::{LogicalPosition, LogicalSize}; use crate::{
use crate::window::WindowId; dpi::{LogicalPosition, LogicalSize},
use crate::platform_impl; platform_impl,
window::WindowId,
};
/// Describes a generic event. /// Describes a generic event.
#[derive(Clone, Debug, PartialEq)] #[derive(Clone, Debug, PartialEq)]
@ -47,8 +48,8 @@ impl<T> Event<T> {
use self::Event::*; use self::Event::*;
match self { match self {
UserEvent(_) => Err(self), UserEvent(_) => Err(self),
WindowEvent{window_id, event} => Ok(WindowEvent{window_id, event}), WindowEvent { window_id, event } => Ok(WindowEvent { window_id, event }),
DeviceEvent{device_id, event} => Ok(DeviceEvent{device_id, event}), DeviceEvent { device_id, event } => Ok(DeviceEvent { device_id, event }),
NewEvents(cause) => Ok(NewEvents(cause)), NewEvents(cause) => Ok(NewEvents(cause)),
EventsCleared => Ok(EventsCleared), EventsCleared => Ok(EventsCleared),
LoopDestroyed => Ok(LoopDestroyed), LoopDestroyed => Ok(LoopDestroyed),
@ -65,14 +66,14 @@ pub enum StartCause {
/// guaranteed to be equal to or after the requested resume time. /// guaranteed to be equal to or after the requested resume time.
ResumeTimeReached { ResumeTimeReached {
start: Instant, start: Instant,
requested_resume: Instant requested_resume: Instant,
}, },
/// Sent if the OS has new events to send to the window, after a wait was requested. Contains /// Sent if the OS has new events to send to the window, after a wait was requested. Contains
/// the moment the wait was requested and the resume time, if requested. /// the moment the wait was requested and the resume time, if requested.
WaitCancelled { WaitCancelled {
start: Instant, start: Instant,
requested_resume: Option<Instant> requested_resume: Option<Instant>,
}, },
/// Sent if the event loop is being resumed after the loop's control flow was set to /// Sent if the event loop is being resumed after the loop's control flow was set to
@ -80,7 +81,7 @@ pub enum StartCause {
Poll, Poll,
/// Sent once, immediately after `run` is called. Indicates that the loop was just initialized. /// Sent once, immediately after `run` is called. Indicates that the loop was just initialized.
Init Init,
} }
/// Describes an event from a `Window`. /// Describes an event from a `Window`.
@ -125,7 +126,10 @@ pub enum WindowEvent {
Focused(bool), Focused(bool),
/// An event from the keyboard has been received. /// An event from the keyboard has been received.
KeyboardInput { device_id: DeviceId, input: KeyboardInput }, KeyboardInput {
device_id: DeviceId,
input: KeyboardInput,
},
/// The cursor has moved on the window. /// The cursor has moved on the window.
CursorMoved { CursorMoved {
@ -135,7 +139,7 @@ pub enum WindowEvent {
/// limited by the display area and it may have been transformed by the OS to implement effects such as cursor /// limited by the display area and it may have been transformed by the OS to implement effects such as cursor
/// acceleration, it should not be used to implement non-cursor-like interactions such as 3D camera control. /// acceleration, it should not be used to implement non-cursor-like interactions such as 3D camera control.
position: LogicalPosition, position: LogicalPosition,
modifiers: ModifiersState modifiers: ModifiersState,
}, },
/// The cursor has entered the window. /// The cursor has entered the window.
@ -145,21 +149,38 @@ pub enum WindowEvent {
CursorLeft { device_id: DeviceId }, CursorLeft { device_id: DeviceId },
/// A mouse wheel movement or touchpad scroll occurred. /// A mouse wheel movement or touchpad scroll occurred.
MouseWheel { device_id: DeviceId, delta: MouseScrollDelta, phase: TouchPhase, modifiers: ModifiersState }, MouseWheel {
device_id: DeviceId,
delta: MouseScrollDelta,
phase: TouchPhase,
modifiers: ModifiersState,
},
/// An mouse button press has been received. /// An mouse button press has been received.
MouseInput { device_id: DeviceId, state: ElementState, button: MouseButton, modifiers: ModifiersState }, MouseInput {
device_id: DeviceId,
state: ElementState,
button: MouseButton,
modifiers: ModifiersState,
},
/// Touchpad pressure event. /// Touchpad pressure event.
/// ///
/// At the moment, only supported on Apple forcetouch-capable macbooks. /// At the moment, only supported on Apple forcetouch-capable macbooks.
/// The parameters are: pressure level (value between 0 and 1 representing how hard the touchpad /// The parameters are: pressure level (value between 0 and 1 representing how hard the touchpad
/// is being pressed) and stage (integer representing the click level). /// is being pressed) and stage (integer representing the click level).
TouchpadPressure { device_id: DeviceId, pressure: f32, stage: i64 }, TouchpadPressure {
device_id: DeviceId,
pressure: f32,
stage: i64,
},
/// Motion on some analog axis. May report data redundant to other, more specific events. /// Motion on some analog axis. May report data redundant to other, more specific events.
AxisMotion { device_id: DeviceId, axis: AxisId, value: f64 }, AxisMotion {
device_id: DeviceId,
axis: AxisId,
value: f64,
},
/// The OS or application has requested that the window be redrawn. /// The OS or application has requested that the window be redrawn.
RedrawRequested, RedrawRequested,
@ -229,11 +250,19 @@ pub enum DeviceEvent {
/// Motion on some analog axis. This event will be reported for all arbitrary input devices /// Motion on some analog axis. This event will be reported for all arbitrary input devices
/// that winit supports on this platform, including mouse devices. If the device is a mouse /// that winit supports on this platform, including mouse devices. If the device is a mouse
/// device then this will be reported alongside the MouseMotion event. /// device then this will be reported alongside the MouseMotion event.
Motion { axis: AxisId, value: f64 }, Motion {
axis: AxisId,
value: f64,
},
Button { button: ButtonId, state: ElementState }, Button {
button: ButtonId,
state: ElementState,
},
Key(KeyboardInput), Key(KeyboardInput),
Text { codepoint: char }, Text {
codepoint: char,
},
} }
/// Describes a keyboard input event. /// Describes a keyboard input event.
@ -259,7 +288,7 @@ pub struct KeyboardInput {
/// ///
/// This is tracked internally to avoid tracking errors arising from modifier key state changes when events from /// This is tracked internally to avoid tracking errors arising from modifier key state changes when events from
/// this device are not being delivered to the application, e.g. due to keyboard focus being elsewhere. /// this device are not being delivered to the application, e.g. due to keyboard focus being elsewhere.
pub modifiers: ModifiersState pub modifiers: ModifiersState,
} }
/// Describes touch-screen input state. /// Describes touch-screen input state.
@ -269,7 +298,7 @@ pub enum TouchPhase {
Started, Started,
Moved, Moved,
Ended, Ended,
Cancelled Cancelled,
} }
/// Represents touch event /// Represents touch event
@ -293,7 +322,7 @@ pub struct Touch {
pub phase: TouchPhase, pub phase: TouchPhase,
pub location: LogicalPosition, pub location: LogicalPosition,
/// unique identifier of a finger. /// unique identifier of a finger.
pub id: u64 pub id: u64,
} }
/// Hardware-dependent keyboard scan code. /// Hardware-dependent keyboard scan code.
@ -327,19 +356,19 @@ pub enum MouseButton {
#[derive(Debug, Clone, Copy, PartialEq)] #[derive(Debug, Clone, Copy, PartialEq)]
#[cfg_attr(feature = "serde", derive(Serialize, Deserialize))] #[cfg_attr(feature = "serde", derive(Serialize, Deserialize))]
pub enum MouseScrollDelta { pub enum MouseScrollDelta {
/// Amount in lines or rows to scroll in the horizontal /// Amount in lines or rows to scroll in the horizontal
/// and vertical directions. /// and vertical directions.
/// ///
/// Positive values indicate movement forward /// Positive values indicate movement forward
/// (away from the user) or rightwards. /// (away from the user) or rightwards.
LineDelta(f32, f32), LineDelta(f32, f32),
/// Amount in pixels to scroll in the horizontal and /// Amount in pixels to scroll in the horizontal and
/// vertical direction. /// vertical direction.
/// ///
/// Scroll events are expressed as a PixelDelta if /// Scroll events are expressed as a PixelDelta if
/// supported by the device (eg. a touchpad) and /// supported by the device (eg. a touchpad) and
/// platform. /// platform.
PixelDelta(LogicalPosition), PixelDelta(LogicalPosition),
} }
/// Symbolic name for a keyboard key. /// Symbolic name for a keyboard key.
@ -499,7 +528,7 @@ pub enum VirtualKeyCode {
Multiply, Multiply,
Mute, Mute,
MyComputer, MyComputer,
NavigateForward, // also called "Prior" NavigateForward, // also called "Prior"
NavigateBackward, // also called "Next" NavigateBackward, // also called "Next"
NextTrack, NextTrack,
NoConvert, NoConvert,
@ -557,5 +586,5 @@ pub struct ModifiersState {
/// The "logo" key /// The "logo" key
/// ///
/// This is the "windows" key on PC and "command" key on Mac. /// This is the "windows" key on PC and "command" key on Mac.
pub logo: bool pub logo: bool,
} }

View file

@ -9,13 +9,13 @@
//! [create_proxy]: ./struct.EventLoop.html#method.create_proxy //! [create_proxy]: ./struct.EventLoop.html#method.create_proxy
//! [event_loop_proxy]: ./struct.EventLoopProxy.html //! [event_loop_proxy]: ./struct.EventLoopProxy.html
//! [send_event]: ./struct.EventLoopProxy.html#method.send_event //! [send_event]: ./struct.EventLoopProxy.html#method.send_event
use std::{fmt, error}; use std::{error, fmt, ops::Deref, time::Instant};
use std::time::Instant;
use std::ops::Deref;
use crate::platform_impl; use crate::{
use crate::event::Event; event::Event,
use crate::monitor::{AvailableMonitorsIter, MonitorHandle}; monitor::{AvailableMonitorsIter, MonitorHandle},
platform_impl,
};
/// Provides a way to retrieve events from the system and from the windows that were registered to /// Provides a way to retrieve events from the system and from the windows that were registered to
/// the events loop. /// the events loop.
@ -32,7 +32,7 @@ use crate::monitor::{AvailableMonitorsIter, MonitorHandle};
/// `EventLoopProxy` allows you to wake up an `EventLoop` from an other thread. /// `EventLoopProxy` allows you to wake up an `EventLoop` from an other thread.
pub struct EventLoop<T: 'static> { pub struct EventLoop<T: 'static> {
pub(crate) event_loop: platform_impl::EventLoop<T>, pub(crate) event_loop: platform_impl::EventLoop<T>,
pub(crate) _marker: ::std::marker::PhantomData<*mut ()> // Not Send nor Sync pub(crate) _marker: ::std::marker::PhantomData<*mut ()>, // Not Send nor Sync
} }
/// Target that associates windows with an `EventLoop`. /// Target that associates windows with an `EventLoop`.
@ -42,7 +42,7 @@ pub struct EventLoop<T: 'static> {
/// take `&EventLoop`. /// take `&EventLoop`.
pub struct EventLoopWindowTarget<T: 'static> { pub struct EventLoopWindowTarget<T: 'static> {
pub(crate) p: platform_impl::EventLoopWindowTarget<T>, pub(crate) p: platform_impl::EventLoopWindowTarget<T>,
pub(crate) _marker: ::std::marker::PhantomData<*mut ()> // Not Send nor Sync pub(crate) _marker: ::std::marker::PhantomData<*mut ()>, // Not Send nor Sync
} }
impl<T> fmt::Debug for EventLoop<T> { impl<T> fmt::Debug for EventLoop<T> {
@ -82,7 +82,7 @@ pub enum ControlFlow {
/// Send a `LoopDestroyed` event and stop the event loop. This variant is *sticky* - once set, /// Send a `LoopDestroyed` event and stop the event loop. This variant is *sticky* - once set,
/// `control_flow` cannot be changed from `Exit`, and any future attempts to do so will result /// `control_flow` cannot be changed from `Exit`, and any future attempts to do so will result
/// in the `control_flow` parameter being reset to `Exit`. /// in the `control_flow` parameter being reset to `Exit`.
Exit Exit,
} }
impl Default for ControlFlow { impl Default for ControlFlow {
@ -133,7 +133,8 @@ impl<T> EventLoop<T> {
/// [`ControlFlow`]: ./enum.ControlFlow.html /// [`ControlFlow`]: ./enum.ControlFlow.html
#[inline] #[inline]
pub fn run<F>(self, event_handler: F) -> ! pub fn run<F>(self, event_handler: F) -> !
where F: 'static + FnMut(Event<T>, &EventLoopWindowTarget<T>, &mut ControlFlow) where
F: 'static + FnMut(Event<T>, &EventLoopWindowTarget<T>, &mut ControlFlow),
{ {
self.event_loop.run(event_handler) self.event_loop.run(event_handler)
} }
@ -149,13 +150,17 @@ impl<T> EventLoop<T> {
#[inline] #[inline]
pub fn available_monitors(&self) -> impl Iterator<Item = MonitorHandle> { pub fn available_monitors(&self) -> impl Iterator<Item = MonitorHandle> {
let data = self.event_loop.available_monitors(); let data = self.event_loop.available_monitors();
AvailableMonitorsIter{ data: data.into_iter() } AvailableMonitorsIter {
data: data.into_iter(),
}
} }
/// Returns the primary monitor of the system. /// Returns the primary monitor of the system.
#[inline] #[inline]
pub fn primary_monitor(&self) -> MonitorHandle { pub fn primary_monitor(&self) -> MonitorHandle {
MonitorHandle { inner: self.event_loop.primary_monitor() } MonitorHandle {
inner: self.event_loop.primary_monitor(),
}
} }
} }
@ -205,4 +210,3 @@ impl error::Error for EventLoopClosed {
"Tried to wake up a closed `EventLoop`" "Tried to wake up a closed `EventLoop`"
} }
} }

View file

@ -1,5 +1,4 @@
use std::{fmt, mem}; use std::{error::Error, fmt, mem};
use std::error::Error;
#[repr(C)] #[repr(C)]
#[derive(Debug)] #[derive(Debug)]
@ -17,9 +16,7 @@ pub(crate) const PIXEL_SIZE: usize = mem::size_of::<Pixel>();
pub enum BadIcon { pub enum BadIcon {
/// Produced when the length of the `rgba` argument isn't divisible by 4, thus `rgba` can't be /// Produced when the length of the `rgba` argument isn't divisible by 4, thus `rgba` can't be
/// safely interpreted as 32bpp RGBA pixels. /// safely interpreted as 32bpp RGBA pixels.
ByteCountNotDivisibleBy4 { ByteCountNotDivisibleBy4 { byte_count: usize },
byte_count: usize,
},
/// Produced when the number of pixels (`rgba.len() / 4`) isn't equal to `width * height`. /// Produced when the number of pixels (`rgba.len() / 4`) isn't equal to `width * height`.
/// At least one of your arguments is incorrect. /// At least one of your arguments is incorrect.
DimensionsVsPixelCount { DimensionsVsPixelCount {
@ -76,7 +73,9 @@ impl Icon {
/// `rgba.len() / 4`. Otherwise, this will return a `BadIcon` error. /// `rgba.len() / 4`. Otherwise, this will return a `BadIcon` error.
pub fn from_rgba(rgba: Vec<u8>, width: u32, height: u32) -> Result<Self, BadIcon> { pub fn from_rgba(rgba: Vec<u8>, width: u32, height: u32) -> Result<Self, BadIcon> {
if rgba.len() % PIXEL_SIZE != 0 { if rgba.len() % PIXEL_SIZE != 0 {
return Err(BadIcon::ByteCountNotDivisibleBy4 { byte_count: rgba.len() }); return Err(BadIcon::ByteCountNotDivisibleBy4 {
byte_count: rgba.len(),
});
} }
let pixel_count = rgba.len() / PIXEL_SIZE; let pixel_count = rgba.len() / PIXEL_SIZE;
if pixel_count != (width * height) as usize { if pixel_count != (width * height) as usize {
@ -87,7 +86,11 @@ impl Icon {
pixel_count, pixel_count,
}) })
} else { } else {
Ok(Icon { rgba, width, height }) Ok(Icon {
rgba,
width,
height,
})
} }
} }
} }

View file

@ -33,14 +33,19 @@
//! is emitted and the entire program terminates. //! is emitted and the entire program terminates.
//! //!
//! ```no_run //! ```no_run
//! use winit::event_loop::ControlFlow; //! use winit::{
//! use winit::event::{Event, WindowEvent}; //! event::{Event, WindowEvent},
//! event_loop::ControlFlow,
//! };
//! # use winit::event_loop::EventLoop; //! # use winit::event_loop::EventLoop;
//! # let event_loop = EventLoop::new(); //! # let event_loop = EventLoop::new();
//! //!
//! event_loop.run(move |event, _, control_flow| { //! event_loop.run(move |event, _, control_flow| {
//! match event { //! match event {
//! Event::WindowEvent { event: WindowEvent::CloseRequested, .. } => { //! Event::WindowEvent {
//! event: WindowEvent::CloseRequested,
//! ..
//! } => {
//! println!("The close button was pressed; stopping"); //! println!("The close button was pressed; stopping");
//! *control_flow = ControlFlow::Exit //! *control_flow = ControlFlow::Exit
//! }, //! },
@ -101,8 +106,8 @@ pub mod error;
pub mod event; pub mod event;
pub mod event_loop; pub mod event_loop;
mod icon; mod icon;
pub mod monitor;
mod platform_impl; mod platform_impl;
pub mod window; pub mod window;
pub mod monitor;
pub mod platform; pub mod platform;

View file

@ -12,8 +12,10 @@
//! [window_get]: ../window/struct.Window.html#method.available_monitors //! [window_get]: ../window/struct.Window.html#method.available_monitors
use std::collections::vec_deque::IntoIter as VecDequeIter; use std::collections::vec_deque::IntoIter as VecDequeIter;
use crate::platform_impl; use crate::{
use crate::dpi::{PhysicalPosition, PhysicalSize}; dpi::{PhysicalPosition, PhysicalSize},
platform_impl,
};
/// An iterator over all available monitors. /// An iterator over all available monitors.
/// ///
@ -90,7 +92,7 @@ impl VideoMode {
/// [`Window`]: ../window/struct.Window.html /// [`Window`]: ../window/struct.Window.html
#[derive(Debug, Clone)] #[derive(Debug, Clone)]
pub struct MonitorHandle { pub struct MonitorHandle {
pub(crate) inner: platform_impl::MonitorHandle pub(crate) inner: platform_impl::MonitorHandle,
} }
impl MonitorHandle { impl MonitorHandle {

View file

@ -1,9 +1,7 @@
#![cfg(any(target_os = "android"))] #![cfg(any(target_os = "android"))]
use crate::{EventLoop, Window, WindowBuilder};
use std::os::raw::c_void; use std::os::raw::c_void;
use crate::EventLoop;
use crate::Window;
use crate::WindowBuilder;
/// Additional methods on `EventLoop` that are specific to Android. /// Additional methods on `EventLoop` that are specific to Android.
pub trait EventLoopExtAndroid { pub trait EventLoopExtAndroid {
@ -30,9 +28,6 @@ impl WindowExtAndroid for Window {
} }
/// Additional methods on `WindowBuilder` that are specific to Android. /// Additional methods on `WindowBuilder` that are specific to Android.
pub trait WindowBuilderExtAndroid { pub trait WindowBuilderExtAndroid {}
} impl WindowBuilderExtAndroid for WindowBuilder {}
impl WindowBuilderExtAndroid for WindowBuilder {
}

View file

@ -4,8 +4,10 @@
target_os = "linux", target_os = "dragonfly", target_os = "freebsd", target_os = "netbsd", target_os = "openbsd" target_os = "linux", target_os = "dragonfly", target_os = "freebsd", target_os = "netbsd", target_os = "openbsd"
))] ))]
use crate::event::Event; use crate::{
use crate::event_loop::{EventLoop, EventLoopWindowTarget, ControlFlow}; event::Event,
event_loop::{ControlFlow, EventLoop, EventLoopWindowTarget},
};
/// Additional methods on `EventLoop` that are specific to desktop platforms. /// Additional methods on `EventLoop` that are specific to desktop platforms.
pub trait EventLoopExtDesktop { pub trait EventLoopExtDesktop {
@ -27,14 +29,16 @@ pub trait EventLoopExtDesktop {
/// ///
/// You are strongly encouraged to use `run`, unless the use of this is absolutely necessary. /// You are strongly encouraged to use `run`, unless the use of this is absolutely necessary.
fn run_return<F>(&mut self, event_handler: F) fn run_return<F>(&mut self, event_handler: F)
where F: FnMut(Event<Self::UserEvent>, &EventLoopWindowTarget<Self::UserEvent>, &mut ControlFlow); where
F: FnMut(Event<Self::UserEvent>, &EventLoopWindowTarget<Self::UserEvent>, &mut ControlFlow);
} }
impl<T> EventLoopExtDesktop for EventLoop<T> { impl<T> EventLoopExtDesktop for EventLoop<T> {
type UserEvent = T; type UserEvent = T;
fn run_return<F>(&mut self, event_handler: F) fn run_return<F>(&mut self, event_handler: F)
where F: FnMut(Event<T>, &EventLoopWindowTarget<T>, &mut ControlFlow) where
F: FnMut(Event<T>, &EventLoopWindowTarget<T>, &mut ControlFlow),
{ {
self.event_loop.run_return(event_handler) self.event_loop.run_return(event_handler)
} }

View file

@ -2,9 +2,11 @@
use std::os::raw::c_void; use std::os::raw::c_void;
use crate::event_loop::EventLoop; use crate::{
use crate::monitor::MonitorHandle; event_loop::EventLoop,
use crate::window::{Window, WindowBuilder}; monitor::MonitorHandle,
window::{Window, WindowBuilder},
};
/// Additional methods on `EventLoop` that are specific to iOS. /// Additional methods on `EventLoop` that are specific to iOS.
pub trait EventLoopExtIOS { pub trait EventLoopExtIOS {

View file

@ -2,9 +2,11 @@
use std::os::raw::c_void; use std::os::raw::c_void;
use crate::dpi::LogicalSize; use crate::{
use crate::monitor::MonitorHandle; dpi::LogicalSize,
use crate::window::{Window, WindowBuilder}; monitor::MonitorHandle,
window::{Window, WindowBuilder},
};
/// Additional methods on `Window` that are specific to MacOS. /// Additional methods on `Window` that are specific to MacOS.
pub trait WindowExtMacOS { pub trait WindowExtMacOS {
@ -97,7 +99,8 @@ pub trait WindowBuilderExtMacOS {
/// Sets the activation policy for the window being built. /// Sets the activation policy for the window being built.
fn with_activation_policy(self, activation_policy: ActivationPolicy) -> WindowBuilder; fn with_activation_policy(self, activation_policy: ActivationPolicy) -> WindowBuilder;
/// Enables click-and-drag behavior for the entire window, not just the titlebar. /// Enables click-and-drag behavior for the entire window, not just the titlebar.
fn with_movable_by_window_background(self, movable_by_window_background: bool) -> WindowBuilder; fn with_movable_by_window_background(self, movable_by_window_background: bool)
-> WindowBuilder;
/// Makes the titlebar transparent and allows the content to appear behind it. /// Makes the titlebar transparent and allows the content to appear behind it.
fn with_titlebar_transparent(self, titlebar_transparent: bool) -> WindowBuilder; fn with_titlebar_transparent(self, titlebar_transparent: bool) -> WindowBuilder;
/// Hides the window title. /// Hides the window title.
@ -120,7 +123,10 @@ impl WindowBuilderExtMacOS for WindowBuilder {
} }
#[inline] #[inline]
fn with_movable_by_window_background(mut self, movable_by_window_background: bool) -> WindowBuilder { fn with_movable_by_window_background(
mut self,
movable_by_window_background: bool,
) -> WindowBuilder {
self.platform_specific.movable_by_window_background = movable_by_window_background; self.platform_specific.movable_by_window_background = movable_by_window_background;
self self
} }

View file

@ -1,29 +1,26 @@
#![cfg(any(target_os = "linux", target_os = "dragonfly", target_os = "freebsd", target_os = "netbsd", target_os = "openbsd"))] #![cfg(any(target_os = "linux", target_os = "dragonfly", target_os = "freebsd", target_os = "netbsd", target_os = "openbsd"))]
use std::os::raw; use std::{os::raw, ptr, sync::Arc};
use std::ptr;
use std::sync::Arc;
use smithay_client_toolkit::window::{ButtonState, Theme}; use smithay_client_toolkit::window::{ButtonState, Theme};
use crate::dpi::LogicalSize; use crate::{
use crate::event_loop::EventLoop; dpi::LogicalSize,
use crate::monitor::MonitorHandle; event_loop::EventLoop,
use crate::window::{Window, WindowBuilder}; monitor::MonitorHandle,
window::{Window, WindowBuilder},
};
use crate::platform_impl::{ use crate::platform_impl::{
EventLoop as LinuxEventLoop, x11::{ffi::XVisualInfo, XConnection},
Window as LinuxWindow, EventLoop as LinuxEventLoop, Window as LinuxWindow,
}; };
use crate::platform_impl::x11::XConnection;
use crate::platform_impl::x11::ffi::XVisualInfo;
// TODO: stupid hack so that glutin can do its work // TODO: stupid hack so that glutin can do its work
#[doc(hidden)] #[doc(hidden)]
pub use crate::platform_impl::x11; pub use crate::platform_impl::x11;
pub use crate::platform_impl::XNotSupported; pub use crate::platform_impl::{x11::util::WindowType as XWindowType, XNotSupported};
pub use crate::platform_impl::x11::util::WindowType as XWindowType;
/// Theme for wayland client side decorations /// Theme for wayland client side decorations
/// ///
@ -97,11 +94,13 @@ impl Theme for WaylandThemeObject {
pub trait EventLoopExtUnix { pub trait EventLoopExtUnix {
/// Builds a new `EventLoops` that is forced to use X11. /// Builds a new `EventLoops` that is forced to use X11.
fn new_x11() -> Result<Self, XNotSupported> fn new_x11() -> Result<Self, XNotSupported>
where Self: Sized; where
Self: Sized;
/// Builds a new `EventLoop` that is forced to use Wayland. /// Builds a new `EventLoop` that is forced to use Wayland.
fn new_wayland() -> Self fn new_wayland() -> Self
where Self: Sized; where
Self: Sized;
/// True if the `EventLoop` uses Wayland. /// True if the `EventLoop` uses Wayland.
fn is_wayland(&self) -> bool; fn is_wayland(&self) -> bool;
@ -123,12 +122,12 @@ pub trait EventLoopExtUnix {
impl<T> EventLoopExtUnix for EventLoop<T> { impl<T> EventLoopExtUnix for EventLoop<T> {
#[inline] #[inline]
fn new_x11() -> Result<Self, XNotSupported> { fn new_x11() -> Result<Self, XNotSupported> {
LinuxEventLoop::new_x11().map(|ev| LinuxEventLoop::new_x11().map(|ev| {
EventLoop { EventLoop {
event_loop: ev, event_loop: ev,
_marker: ::std::marker::PhantomData, _marker: ::std::marker::PhantomData,
} }
) })
} }
#[inline] #[inline]
@ -136,7 +135,7 @@ impl<T> EventLoopExtUnix for EventLoop<T> {
EventLoop { EventLoop {
event_loop: match LinuxEventLoop::new_wayland() { event_loop: match LinuxEventLoop::new_wayland() {
Ok(e) => e, Ok(e) => e,
Err(_) => panic!() // TODO: propagate Err(_) => panic!(), // TODO: propagate
}, },
_marker: ::std::marker::PhantomData, _marker: ::std::marker::PhantomData,
} }
@ -157,7 +156,7 @@ impl<T> EventLoopExtUnix for EventLoop<T> {
fn xlib_xconnection(&self) -> Option<Arc<XConnection>> { fn xlib_xconnection(&self) -> Option<Arc<XConnection>> {
match self.event_loop { match self.event_loop {
LinuxEventLoop::X(ref e) => Some(e.x_connection().clone()), LinuxEventLoop::X(ref e) => Some(e.x_connection().clone()),
_ => None _ => None,
} }
} }
@ -165,7 +164,7 @@ impl<T> EventLoopExtUnix for EventLoop<T> {
fn wayland_display(&self) -> Option<*mut raw::c_void> { fn wayland_display(&self) -> Option<*mut raw::c_void> {
match self.event_loop { match self.event_loop {
LinuxEventLoop::Wayland(ref e) => Some(e.display().get_display_ptr() as *mut _), LinuxEventLoop::Wayland(ref e) => Some(e.display().get_display_ptr() as *mut _),
_ => None _ => None,
} }
} }
} }
@ -231,7 +230,7 @@ impl WindowExtUnix for Window {
fn xlib_window(&self) -> Option<raw::c_ulong> { fn xlib_window(&self) -> Option<raw::c_ulong> {
match self.window { match self.window {
LinuxWindow::X(ref w) => Some(w.xlib_window()), LinuxWindow::X(ref w) => Some(w.xlib_window()),
_ => None _ => None,
} }
} }
@ -239,7 +238,7 @@ impl WindowExtUnix for Window {
fn xlib_display(&self) -> Option<*mut raw::c_void> { fn xlib_display(&self) -> Option<*mut raw::c_void> {
match self.window { match self.window {
LinuxWindow::X(ref w) => Some(w.xlib_display()), LinuxWindow::X(ref w) => Some(w.xlib_display()),
_ => None _ => None,
} }
} }
@ -247,7 +246,7 @@ impl WindowExtUnix for Window {
fn xlib_screen_id(&self) -> Option<raw::c_int> { fn xlib_screen_id(&self) -> Option<raw::c_int> {
match self.window { match self.window {
LinuxWindow::X(ref w) => Some(w.xlib_screen_id()), LinuxWindow::X(ref w) => Some(w.xlib_screen_id()),
_ => None _ => None,
} }
} }
@ -256,7 +255,7 @@ impl WindowExtUnix for Window {
fn xlib_xconnection(&self) -> Option<Arc<XConnection>> { fn xlib_xconnection(&self) -> Option<Arc<XConnection>> {
match self.window { match self.window {
LinuxWindow::X(ref w) => Some(w.xlib_xconnection()), LinuxWindow::X(ref w) => Some(w.xlib_xconnection()),
_ => None _ => None,
} }
} }
@ -264,7 +263,7 @@ impl WindowExtUnix for Window {
fn xcb_connection(&self) -> Option<*mut raw::c_void> { fn xcb_connection(&self) -> Option<*mut raw::c_void> {
match self.window { match self.window {
LinuxWindow::X(ref w) => Some(w.xcb_connection()), LinuxWindow::X(ref w) => Some(w.xcb_connection()),
_ => None _ => None,
} }
} }
@ -279,7 +278,7 @@ impl WindowExtUnix for Window {
fn wayland_surface(&self) -> Option<*mut raw::c_void> { fn wayland_surface(&self) -> Option<*mut raw::c_void> {
match self.window { match self.window {
LinuxWindow::Wayland(ref w) => Some(w.surface().as_ref().c_ptr() as *mut _), LinuxWindow::Wayland(ref w) => Some(w.surface().as_ref().c_ptr() as *mut _),
_ => None _ => None,
} }
} }
@ -287,7 +286,7 @@ impl WindowExtUnix for Window {
fn wayland_display(&self) -> Option<*mut raw::c_void> { fn wayland_display(&self) -> Option<*mut raw::c_void> {
match self.window { match self.window {
LinuxWindow::Wayland(ref w) => Some(w.display().as_ref().c_ptr() as *mut _), LinuxWindow::Wayland(ref w) => Some(w.display().as_ref().c_ptr() as *mut _),
_ => None _ => None,
} }
} }
@ -295,7 +294,7 @@ impl WindowExtUnix for Window {
fn set_wayland_theme(&self, theme: WaylandTheme) { fn set_wayland_theme(&self, theme: WaylandTheme) {
match self.window { match self.window {
LinuxWindow::Wayland(ref w) => w.set_theme(WaylandThemeObject(theme)), LinuxWindow::Wayland(ref w) => w.set_theme(WaylandThemeObject(theme)),
_ => {} _ => {},
} }
} }
@ -334,9 +333,8 @@ pub trait WindowBuilderExtUnix {
impl WindowBuilderExtUnix for WindowBuilder { impl WindowBuilderExtUnix for WindowBuilder {
#[inline] #[inline]
fn with_x11_visual<T>(mut self, visual_infos: *const T) -> WindowBuilder { fn with_x11_visual<T>(mut self, visual_infos: *const T) -> WindowBuilder {
self.platform_specific.visual_infos = Some( self.platform_specific.visual_infos =
unsafe { ptr::read(visual_infos as *const XVisualInfo) } Some(unsafe { ptr::read(visual_infos as *const XVisualInfo) });
);
self self
} }

View file

@ -5,17 +5,21 @@ use std::os::raw::c_void;
use libc; use libc;
use winapi::shared::windef::HWND; use winapi::shared::windef::HWND;
use crate::event::DeviceId; use crate::{
use crate::monitor::MonitorHandle; event::DeviceId,
use crate::event_loop::EventLoop; event_loop::EventLoop,
use crate::window::{Icon, Window, WindowBuilder}; monitor::MonitorHandle,
use crate::platform_impl::EventLoop as WindowsEventLoop; platform_impl::EventLoop as WindowsEventLoop,
window::{Icon, Window, WindowBuilder},
};
/// Additional methods on `EventLoop` that are specific to Windows. /// Additional methods on `EventLoop` that are specific to Windows.
pub trait EventLoopExtWindows { pub trait EventLoopExtWindows {
/// By default, winit on Windows will attempt to enable process-wide DPI awareness. If that's /// By default, winit on Windows will attempt to enable process-wide DPI awareness. If that's
/// undesirable, you can create an `EventLoop` using this function instead. /// undesirable, you can create an `EventLoop` using this function instead.
fn new_dpi_unaware() -> Self where Self: Sized; fn new_dpi_unaware() -> Self
where
Self: Sized;
} }
impl<T> EventLoopExtWindows for EventLoop<T> { impl<T> EventLoopExtWindows for EventLoop<T> {

View file

@ -9,32 +9,32 @@ use std::os::raw;
#[link(name = "android")] #[link(name = "android")]
#[link(name = "EGL")] #[link(name = "EGL")]
#[link(name = "GLESv2")] #[link(name = "GLESv2")]
extern {} extern "C" {}
/** /**
* asset_manager.h ** asset_manager.h
*/ **/
pub type AAssetManager = raw::c_void; pub type AAssetManager = raw::c_void;
/** /**
* native_window.h ** native_window.h
*/ **/
pub type ANativeWindow = raw::c_void; pub type ANativeWindow = raw::c_void;
extern { extern "C" {
pub fn ANativeWindow_getHeight(window: *const ANativeWindow) -> libc::int32_t; pub fn ANativeWindow_getHeight(window: *const ANativeWindow) -> libc::int32_t;
pub fn ANativeWindow_getWidth(window: *const ANativeWindow) -> libc::int32_t; pub fn ANativeWindow_getWidth(window: *const ANativeWindow) -> libc::int32_t;
} }
/** /**
* native_activity.h ** native_activity.h
*/ **/
pub type JavaVM = (); pub type JavaVM = ();
pub type JNIEnv = (); pub type JNIEnv = ();
pub type jobject = *const libc::c_void; pub type jobject = *const libc::c_void;
pub type AInputQueue = (); // FIXME: wrong pub type AInputQueue = (); // FIXME: wrong
pub type ARect = (); // FIXME: wrong pub type ARect = (); // FIXME: wrong
#[repr(C)] #[repr(C)]
pub struct ANativeActivity { pub struct ANativeActivity {
@ -52,43 +52,56 @@ pub struct ANativeActivity {
#[repr(C)] #[repr(C)]
pub struct ANativeActivityCallbacks { pub struct ANativeActivityCallbacks {
pub onStart: extern fn(*mut ANativeActivity), pub onStart: extern "C" fn(*mut ANativeActivity),
pub onResume: extern fn(*mut ANativeActivity), pub onResume: extern "C" fn(*mut ANativeActivity),
pub onSaveInstanceState: extern fn(*mut ANativeActivity, *mut libc::size_t), pub onSaveInstanceState: extern "C" fn(*mut ANativeActivity, *mut libc::size_t),
pub onPause: extern fn(*mut ANativeActivity), pub onPause: extern "C" fn(*mut ANativeActivity),
pub onStop: extern fn(*mut ANativeActivity), pub onStop: extern "C" fn(*mut ANativeActivity),
pub onDestroy: extern fn(*mut ANativeActivity), pub onDestroy: extern "C" fn(*mut ANativeActivity),
pub onWindowFocusChanged: extern fn(*mut ANativeActivity, libc::c_int), pub onWindowFocusChanged: extern "C" fn(*mut ANativeActivity, libc::c_int),
pub onNativeWindowCreated: extern fn(*mut ANativeActivity, *const ANativeWindow), pub onNativeWindowCreated: extern "C" fn(*mut ANativeActivity, *const ANativeWindow),
pub onNativeWindowResized: extern fn(*mut ANativeActivity, *const ANativeWindow), pub onNativeWindowResized: extern "C" fn(*mut ANativeActivity, *const ANativeWindow),
pub onNativeWindowRedrawNeeded: extern fn(*mut ANativeActivity, *const ANativeWindow), pub onNativeWindowRedrawNeeded: extern "C" fn(*mut ANativeActivity, *const ANativeWindow),
pub onNativeWindowDestroyed: extern fn(*mut ANativeActivity, *const ANativeWindow), pub onNativeWindowDestroyed: extern "C" fn(*mut ANativeActivity, *const ANativeWindow),
pub onInputQueueCreated: extern fn(*mut ANativeActivity, *mut AInputQueue), pub onInputQueueCreated: extern "C" fn(*mut ANativeActivity, *mut AInputQueue),
pub onInputQueueDestroyed: extern fn(*mut ANativeActivity, *mut AInputQueue), pub onInputQueueDestroyed: extern "C" fn(*mut ANativeActivity, *mut AInputQueue),
pub onContentRectChanged: extern fn(*mut ANativeActivity, *const ARect), pub onContentRectChanged: extern "C" fn(*mut ANativeActivity, *const ARect),
pub onConfigurationChanged: extern fn(*mut ANativeActivity), pub onConfigurationChanged: extern "C" fn(*mut ANativeActivity),
pub onLowMemory: extern fn(*mut ANativeActivity), pub onLowMemory: extern "C" fn(*mut ANativeActivity),
} }
/** /**
* looper.h ** looper.h
*/ **/
pub type ALooper = (); pub type ALooper = ();
#[link(name = "android")] #[link(name = "android")]
extern { extern "C" {
pub fn ALooper_forThread() -> *const ALooper; pub fn ALooper_forThread() -> *const ALooper;
pub fn ALooper_acquire(looper: *const ALooper); pub fn ALooper_acquire(looper: *const ALooper);
pub fn ALooper_release(looper: *const ALooper); pub fn ALooper_release(looper: *const ALooper);
pub fn ALooper_prepare(opts: libc::c_int) -> *const ALooper; pub fn ALooper_prepare(opts: libc::c_int) -> *const ALooper;
pub fn ALooper_pollOnce(timeoutMillis: libc::c_int, outFd: *mut libc::c_int, pub fn ALooper_pollOnce(
outEvents: *mut libc::c_int, outData: *mut *mut libc::c_void) -> libc::c_int; timeoutMillis: libc::c_int,
pub fn ALooper_pollAll(timeoutMillis: libc::c_int, outFd: *mut libc::c_int, outFd: *mut libc::c_int,
outEvents: *mut libc::c_int, outData: *mut *mut libc::c_void) -> libc::c_int; outEvents: *mut libc::c_int,
outData: *mut *mut libc::c_void,
) -> libc::c_int;
pub fn ALooper_pollAll(
timeoutMillis: libc::c_int,
outFd: *mut libc::c_int,
outEvents: *mut libc::c_int,
outData: *mut *mut libc::c_void,
) -> libc::c_int;
pub fn ALooper_wake(looper: *const ALooper); pub fn ALooper_wake(looper: *const ALooper);
pub fn ALooper_addFd(looper: *const ALooper, fd: libc::c_int, ident: libc::c_int, pub fn ALooper_addFd(
events: libc::c_int, callback: ALooper_callbackFunc, data: *mut libc::c_void) looper: *const ALooper,
-> libc::c_int; fd: libc::c_int,
ident: libc::c_int,
events: libc::c_int,
callback: ALooper_callbackFunc,
data: *mut libc::c_void,
) -> libc::c_int;
pub fn ALooper_removeFd(looper: *const ALooper, fd: libc::c_int) -> libc::c_int; pub fn ALooper_removeFd(looper: *const ALooper, fd: libc::c_int) -> libc::c_int;
} }
@ -105,4 +118,5 @@ pub const ALOOPER_EVENT_ERROR: libc::c_int = 1 << 2;
pub const ALOOPER_EVENT_HANGUP: libc::c_int = 1 << 3; pub const ALOOPER_EVENT_HANGUP: libc::c_int = 1 << 3;
pub const ALOOPER_EVENT_INVALID: libc::c_int = 1 << 4; pub const ALOOPER_EVENT_INVALID: libc::c_int = 1 << 4;
pub type ALooper_callbackFunc = extern fn(libc::c_int, libc::c_int, *mut libc::c_void) -> libc::c_int; pub type ALooper_callbackFunc =
extern "C" fn(libc::c_int, libc::c_int, *mut libc::c_void) -> libc::c_int;

View file

@ -4,28 +4,22 @@ extern crate android_glue;
mod ffi; mod ffi;
use std::cell::RefCell; use std::{
use std::collections::VecDeque; cell::RefCell,
use std::fmt; collections::VecDeque,
use std::os::raw::c_void; fmt,
use std::sync::mpsc::{Receiver, channel}; os::raw::c_void,
sync::mpsc::{channel, Receiver},
};
use { use crate::{
crate::CreationError, error::{ExternalError, NotSupportedError},
crate::Event, events::{Touch, TouchPhase},
crate::LogicalPosition, window::MonitorHandle as RootMonitorHandle,
crate::LogicalSize, CreationError, CursorIcon, Event, LogicalPosition, LogicalSize, PhysicalPosition, PhysicalSize,
crate::CursorIcon, WindowAttributes, WindowEvent, WindowId as RootWindowId,
crate::PhysicalPosition,
crate::PhysicalSize,
crate::WindowAttributes,
crate::WindowEvent,
crate::WindowId as RootWindowId,
}; };
use CreationError::OsError; use CreationError::OsError;
use crate::error::{ExternalError, NotSupportedError};
use crate::events::{Touch, TouchPhase};
use crate::window::MonitorHandle as RootMonitorHandle;
pub type OsError = std::io::Error; pub type OsError = std::io::Error;
@ -60,10 +54,11 @@ impl EventLoop {
} }
pub fn poll_events<F>(&mut self, mut callback: F) pub fn poll_events<F>(&mut self, mut callback: F)
where F: FnMut(::Event) where
F: FnMut(::Event),
{ {
while let Ok(event) = self.event_rx.try_recv() { while let Ok(event) = self.event_rx.try_recv() {
let e = match event{ let e = match event {
android_glue::Event::EventMotion(motion) => { android_glue::Event::EventMotion(motion) => {
let dpi_factor = MonitorHandle.hidpi_factor(); let dpi_factor = MonitorHandle.hidpi_factor();
let location = LogicalPosition::from_physical( let location = LogicalPosition::from_physical(
@ -99,8 +94,7 @@ impl EventLoop {
} }
Some(Event::Suspended(true)) Some(Event::Suspended(true))
}, },
android_glue::Event::WindowResized | android_glue::Event::WindowResized | android_glue::Event::ConfigChanged => {
android_glue::Event::ConfigChanged => {
// Activity Orientation changed or resized. // Activity Orientation changed or resized.
let native_window = unsafe { android_glue::native_window() }; let native_window = unsafe { android_glue::native_window() };
if native_window.is_null() { if native_window.is_null() {
@ -121,19 +115,15 @@ impl EventLoop {
window_id: RootWindowId(WindowId), window_id: RootWindowId(WindowId),
event: WindowEvent::Redraw, event: WindowEvent::Redraw,
}) })
} },
android_glue::Event::Wake => { android_glue::Event::Wake => Some(Event::Awakened),
Some(Event::Awakened) _ => None,
}
_ => {
None
}
}; };
if let Some(event) = e { if let Some(event) = e {
callback(event); callback(event);
} }
}; }
} }
pub fn set_suspend_callback(&self, cb: Option<Box<dyn Fn(bool) -> ()>>) { pub fn set_suspend_callback(&self, cb: Option<Box<dyn Fn(bool) -> ()>>) {
@ -141,7 +131,8 @@ impl EventLoop {
} }
pub fn run_forever<F>(&mut self, mut callback: F) pub fn run_forever<F>(&mut self, mut callback: F)
where F: FnMut(::Event) -> ::ControlFlow, where
F: FnMut(::Event) -> ::ControlFlow,
{ {
// Yeah that's a very bad implementation. // Yeah that's a very bad implementation.
loop { loop {
@ -229,7 +220,8 @@ impl MonitorHandle {
( (
ffi::ANativeWindow_getWidth(window) as f64, ffi::ANativeWindow_getWidth(window) as f64,
ffi::ANativeWindow_getHeight(window) as f64, ffi::ANativeWindow_getHeight(window) as f64,
).into() )
.into()
} }
} }
@ -251,10 +243,11 @@ pub struct PlatformSpecificWindowBuilderAttributes;
pub struct PlatformSpecificHeadlessBuilderAttributes; pub struct PlatformSpecificHeadlessBuilderAttributes;
impl Window { impl Window {
pub fn new(_: &EventLoop, win_attribs: WindowAttributes, pub fn new(
_: PlatformSpecificWindowBuilderAttributes) _: &EventLoop,
-> Result<Window, CreationError> win_attribs: WindowAttributes,
{ _: PlatformSpecificWindowBuilderAttributes,
) -> Result<Window, CreationError> {
let native_window = unsafe { android_glue::native_window() }; let native_window = unsafe { android_glue::native_window() };
if native_window.is_null() { if native_window.is_null() {
return Err(OsError(format!("Android's native window is null"))); return Err(OsError(format!("Android's native window is null")));
@ -406,7 +399,9 @@ impl Window {
#[inline] #[inline]
pub fn current_monitor(&self) -> RootMonitorHandle { pub fn current_monitor(&self) -> RootMonitorHandle {
RootMonitorHandle { inner: MonitorHandle } RootMonitorHandle {
inner: MonitorHandle,
}
} }
#[inline] #[inline]

View file

@ -1,8 +1,8 @@
#![allow(dead_code, non_camel_case_types, non_snake_case)] #![allow(dead_code, non_camel_case_types, non_snake_case)]
use std::os::raw::{c_int, c_char, c_void, c_ulong, c_double, c_long, c_ushort};
#[cfg(test)] #[cfg(test)]
use std::mem; use std::mem;
use std::os::raw::{c_char, c_double, c_int, c_long, c_ulong, c_ushort, c_void};
pub type EM_BOOL = c_int; pub type EM_BOOL = c_int;
pub type EM_UTF8 = c_char; pub type EM_UTF8 = c_char;
@ -71,30 +71,45 @@ pub const DOM_KEY_LOCATION_NUMPAD: c_ulong = 0x03;
pub type em_callback_func = Option<unsafe extern "C" fn()>; pub type em_callback_func = Option<unsafe extern "C" fn()>;
pub type em_key_callback_func = Option<unsafe extern "C" fn( pub type em_key_callback_func = Option<
eventType: c_int, unsafe extern "C" fn(
keyEvent: *const EmscriptenKeyboardEvent, eventType: c_int,
userData: *mut c_void) -> EM_BOOL>; keyEvent: *const EmscriptenKeyboardEvent,
userData: *mut c_void,
) -> EM_BOOL,
>;
pub type em_mouse_callback_func = Option<unsafe extern "C" fn( pub type em_mouse_callback_func = Option<
eventType: c_int, unsafe extern "C" fn(
mouseEvent: *const EmscriptenMouseEvent, eventType: c_int,
userData: *mut c_void) -> EM_BOOL>; mouseEvent: *const EmscriptenMouseEvent,
userData: *mut c_void,
) -> EM_BOOL,
>;
pub type em_pointerlockchange_callback_func = Option<unsafe extern "C" fn( pub type em_pointerlockchange_callback_func = Option<
eventType: c_int, unsafe extern "C" fn(
pointerlockChangeEvent: *const EmscriptenPointerlockChangeEvent, eventType: c_int,
userData: *mut c_void) -> EM_BOOL>; pointerlockChangeEvent: *const EmscriptenPointerlockChangeEvent,
userData: *mut c_void,
) -> EM_BOOL,
>;
pub type em_fullscreenchange_callback_func = Option<unsafe extern "C" fn( pub type em_fullscreenchange_callback_func = Option<
eventType: c_int, unsafe extern "C" fn(
fullscreenChangeEvent: *const EmscriptenFullscreenChangeEvent, eventType: c_int,
userData: *mut c_void) -> EM_BOOL>; fullscreenChangeEvent: *const EmscriptenFullscreenChangeEvent,
userData: *mut c_void,
) -> EM_BOOL,
>;
pub type em_touch_callback_func = Option<unsafe extern "C" fn( pub type em_touch_callback_func = Option<
eventType: c_int, unsafe extern "C" fn(
touchEvent: *const EmscriptenTouchEvent, eventType: c_int,
userData: *mut c_void) -> EM_BOOL>; touchEvent: *const EmscriptenTouchEvent,
userData: *mut c_void,
) -> EM_BOOL,
>;
#[repr(C)] #[repr(C)]
pub struct EmscriptenFullscreenChangeEvent { pub struct EmscriptenFullscreenChangeEvent {
@ -136,7 +151,9 @@ fn bindgen_test_layout_EmscriptenKeyboardEvent() {
assert_eq!(mem::align_of::<EmscriptenKeyboardEvent>(), 8usize); assert_eq!(mem::align_of::<EmscriptenKeyboardEvent>(), 8usize);
} }
impl Clone for EmscriptenKeyboardEvent { impl Clone for EmscriptenKeyboardEvent {
fn clone(&self) -> Self { *self } fn clone(&self) -> Self {
*self
}
} }
#[repr(C)] #[repr(C)]
@ -219,96 +236,128 @@ fn bindgen_test_layout_EmscriptenPointerlockChangeEvent() {
} }
extern "C" { extern "C" {
pub fn emscripten_set_canvas_size( pub fn emscripten_set_canvas_size(width: c_int, height: c_int) -> EMSCRIPTEN_RESULT;
width: c_int, height: c_int)
-> EMSCRIPTEN_RESULT;
pub fn emscripten_get_canvas_size( pub fn emscripten_get_canvas_size(
width: *mut c_int, height: *mut c_int, width: *mut c_int,
is_fullscreen: *mut c_int) height: *mut c_int,
-> EMSCRIPTEN_RESULT; is_fullscreen: *mut c_int,
) -> EMSCRIPTEN_RESULT;
pub fn emscripten_set_element_css_size( pub fn emscripten_set_element_css_size(
target: *const c_char, width: c_double, target: *const c_char,
height: c_double) -> EMSCRIPTEN_RESULT; width: c_double,
height: c_double,
) -> EMSCRIPTEN_RESULT;
pub fn emscripten_get_element_css_size( pub fn emscripten_get_element_css_size(
target: *const c_char, width: *mut c_double, target: *const c_char,
height: *mut c_double) -> EMSCRIPTEN_RESULT; width: *mut c_double,
height: *mut c_double,
) -> EMSCRIPTEN_RESULT;
pub fn emscripten_request_pointerlock( pub fn emscripten_request_pointerlock(
target: *const c_char, deferUntilInEventHandler: EM_BOOL) target: *const c_char,
-> EMSCRIPTEN_RESULT; deferUntilInEventHandler: EM_BOOL,
) -> EMSCRIPTEN_RESULT;
pub fn emscripten_exit_pointerlock() -> EMSCRIPTEN_RESULT; pub fn emscripten_exit_pointerlock() -> EMSCRIPTEN_RESULT;
pub fn emscripten_request_fullscreen( pub fn emscripten_request_fullscreen(
target: *const c_char, deferUntilInEventHandler: EM_BOOL) target: *const c_char,
-> EMSCRIPTEN_RESULT; deferUntilInEventHandler: EM_BOOL,
) -> EMSCRIPTEN_RESULT;
pub fn emscripten_exit_fullscreen() -> EMSCRIPTEN_RESULT; pub fn emscripten_exit_fullscreen() -> EMSCRIPTEN_RESULT;
pub fn emscripten_set_keydown_callback( pub fn emscripten_set_keydown_callback(
target: *const c_char, userData: *mut c_void, target: *const c_char,
useCapture: EM_BOOL, callback: em_key_callback_func) userData: *mut c_void,
-> EMSCRIPTEN_RESULT; useCapture: EM_BOOL,
callback: em_key_callback_func,
) -> EMSCRIPTEN_RESULT;
pub fn emscripten_set_keyup_callback( pub fn emscripten_set_keyup_callback(
target: *const c_char, userData: *mut c_void, target: *const c_char,
useCapture: EM_BOOL, callback: em_key_callback_func) userData: *mut c_void,
-> EMSCRIPTEN_RESULT; useCapture: EM_BOOL,
callback: em_key_callback_func,
) -> EMSCRIPTEN_RESULT;
pub fn emscripten_set_mousemove_callback( pub fn emscripten_set_mousemove_callback(
target: *const c_char, user_data: *mut c_void, target: *const c_char,
use_capture: EM_BOOL, callback: em_mouse_callback_func) user_data: *mut c_void,
-> EMSCRIPTEN_RESULT; use_capture: EM_BOOL,
callback: em_mouse_callback_func,
) -> EMSCRIPTEN_RESULT;
pub fn emscripten_set_mousedown_callback( pub fn emscripten_set_mousedown_callback(
target: *const c_char, user_data: *mut c_void, target: *const c_char,
use_capture: EM_BOOL, callback: em_mouse_callback_func) user_data: *mut c_void,
-> EMSCRIPTEN_RESULT; use_capture: EM_BOOL,
callback: em_mouse_callback_func,
) -> EMSCRIPTEN_RESULT;
pub fn emscripten_set_mouseup_callback( pub fn emscripten_set_mouseup_callback(
target: *const c_char, user_data: *mut c_void, target: *const c_char,
use_capture: EM_BOOL, callback: em_mouse_callback_func) user_data: *mut c_void,
-> EMSCRIPTEN_RESULT; use_capture: EM_BOOL,
callback: em_mouse_callback_func,
) -> EMSCRIPTEN_RESULT;
pub fn emscripten_hide_mouse(); pub fn emscripten_hide_mouse();
pub fn emscripten_get_device_pixel_ratio() -> f64; pub fn emscripten_get_device_pixel_ratio() -> f64;
pub fn emscripten_set_pointerlockchange_callback( pub fn emscripten_set_pointerlockchange_callback(
target: *const c_char, userData: *mut c_void, useCapture: EM_BOOL, target: *const c_char,
callback: em_pointerlockchange_callback_func) -> EMSCRIPTEN_RESULT; userData: *mut c_void,
useCapture: EM_BOOL,
callback: em_pointerlockchange_callback_func,
) -> EMSCRIPTEN_RESULT;
pub fn emscripten_set_fullscreenchange_callback( pub fn emscripten_set_fullscreenchange_callback(
target: *const c_char, userData: *mut c_void, useCapture: EM_BOOL, target: *const c_char,
callback: em_fullscreenchange_callback_func) -> EMSCRIPTEN_RESULT; userData: *mut c_void,
useCapture: EM_BOOL,
callback: em_fullscreenchange_callback_func,
) -> EMSCRIPTEN_RESULT;
pub fn emscripten_asm_const(code: *const c_char); pub fn emscripten_asm_const(code: *const c_char);
pub fn emscripten_set_main_loop( pub fn emscripten_set_main_loop(
func: em_callback_func, fps: c_int, simulate_infinite_loop: EM_BOOL); func: em_callback_func,
fps: c_int,
simulate_infinite_loop: EM_BOOL,
);
pub fn emscripten_cancel_main_loop(); pub fn emscripten_cancel_main_loop();
pub fn emscripten_set_touchstart_callback( pub fn emscripten_set_touchstart_callback(
target: *const c_char, userData: *mut c_void, target: *const c_char,
useCapture: c_int, callback: em_touch_callback_func) userData: *mut c_void,
-> EMSCRIPTEN_RESULT; useCapture: c_int,
callback: em_touch_callback_func,
) -> EMSCRIPTEN_RESULT;
pub fn emscripten_set_touchend_callback( pub fn emscripten_set_touchend_callback(
target: *const c_char, userData: *mut c_void, target: *const c_char,
useCapture: c_int, callback: em_touch_callback_func) userData: *mut c_void,
-> EMSCRIPTEN_RESULT; useCapture: c_int,
callback: em_touch_callback_func,
) -> EMSCRIPTEN_RESULT;
pub fn emscripten_set_touchmove_callback( pub fn emscripten_set_touchmove_callback(
target: *const c_char, userData: *mut c_void, target: *const c_char,
useCapture: c_int, callback: em_touch_callback_func) userData: *mut c_void,
-> EMSCRIPTEN_RESULT; useCapture: c_int,
callback: em_touch_callback_func,
) -> EMSCRIPTEN_RESULT;
pub fn emscripten_set_touchcancel_callback( pub fn emscripten_set_touchcancel_callback(
target: *const c_char, userData: *mut c_void, target: *const c_char,
useCapture: c_int, callback: em_touch_callback_func) userData: *mut c_void,
-> EMSCRIPTEN_RESULT; useCapture: c_int,
callback: em_touch_callback_func,
) -> EMSCRIPTEN_RESULT;
} }

View file

@ -2,16 +2,23 @@
mod ffi; mod ffi;
use std::{mem, ptr, str}; use std::{
use std::cell::RefCell; cell::RefCell,
use std::collections::VecDeque; collections::VecDeque,
use std::os::raw::{c_char, c_void, c_double, c_ulong, c_int}; mem,
use std::sync::atomic::{AtomicBool, Ordering}; os::raw::{c_char, c_double, c_int, c_ulong, c_void},
use std::sync::{Mutex, Arc}; ptr, str,
sync::{
atomic::{AtomicBool, Ordering},
Arc, Mutex,
},
};
use crate::dpi::{LogicalPosition, LogicalSize, PhysicalPosition, PhysicalSize}; use crate::{
use crate::error::{ExternalError, NotSupportedError}; dpi::{LogicalPosition, LogicalSize, PhysicalPosition, PhysicalSize},
use crate::window::MonitorHandle as RootMonitorHandle; error::{ExternalError, NotSupportedError},
window::MonitorHandle as RootMonitorHandle,
};
const DOCUMENT_NAME: &'static str = "#document\0"; const DOCUMENT_NAME: &'static str = "#document\0";
@ -68,14 +75,22 @@ impl MonitorHandle {
thread_local!(static MAIN_LOOP_CALLBACK: RefCell<*mut c_void> = RefCell::new(ptr::null_mut())); thread_local!(static MAIN_LOOP_CALLBACK: RefCell<*mut c_void> = RefCell::new(ptr::null_mut()));
// Used to assign a callback to emscripten main loop // Used to assign a callback to emscripten main loop
pub fn set_main_loop_callback<F>(callback : F) where F : FnMut() { pub fn set_main_loop_callback<F>(callback: F)
where
F: FnMut(),
{
MAIN_LOOP_CALLBACK.with(|log| { MAIN_LOOP_CALLBACK.with(|log| {
*log.borrow_mut() = &callback as *const _ as *mut c_void; *log.borrow_mut() = &callback as *const _ as *mut c_void;
}); });
unsafe { ffi::emscripten_set_main_loop(Some(wrapper::<F>), 0, 1); } unsafe {
ffi::emscripten_set_main_loop(Some(wrapper::<F>), 0, 1);
}
unsafe extern "C" fn wrapper<F>() where F : FnMut() { unsafe extern "C" fn wrapper<F>()
where
F: FnMut(),
{
MAIN_LOOP_CALLBACK.with(|z| { MAIN_LOOP_CALLBACK.with(|z| {
let closure = *z.borrow_mut() as *mut F; let closure = *z.borrow_mut() as *mut F;
(*closure)(); (*closure)();
@ -128,7 +143,8 @@ impl EventLoop {
} }
pub fn poll_events<F>(&self, mut callback: F) pub fn poll_events<F>(&self, mut callback: F)
where F: FnMut(::Event) where
F: FnMut(::Event),
{ {
let ref mut window = *self.window.lock().unwrap(); let ref mut window = *self.window.lock().unwrap();
if let &mut Some(ref mut window) = window { if let &mut Some(ref mut window) = window {
@ -139,17 +155,22 @@ impl EventLoop {
} }
pub fn run_forever<F>(&self, mut callback: F) pub fn run_forever<F>(&self, mut callback: F)
where F: FnMut(::Event) -> ::ControlFlow where
F: FnMut(::Event) -> ::ControlFlow,
{ {
self.interrupted.store(false, Ordering::Relaxed); self.interrupted.store(false, Ordering::Relaxed);
// TODO: handle control flow // TODO: handle control flow
set_main_loop_callback(|| { set_main_loop_callback(|| {
self.poll_events(|e| { callback(e); }); self.poll_events(|e| {
callback(e);
});
::std::thread::sleep(::std::time::Duration::from_millis(5)); ::std::thread::sleep(::std::time::Duration::from_millis(5));
if self.interrupted.load(Ordering::Relaxed) { if self.interrupted.load(Ordering::Relaxed) {
unsafe { ffi::emscripten_cancel_main_loop(); } unsafe {
ffi::emscripten_cancel_main_loop();
}
} }
}); });
} }
@ -190,15 +211,15 @@ fn show_mouse() {
// } // }
// styleSheet.insertRule('canvas.emscripten { border: none; cursor: auto; }', 0); // styleSheet.insertRule('canvas.emscripten { border: none; cursor: auto; }', 0);
unsafe { unsafe {
ffi::emscripten_asm_const(b"var styleSheet = document.styleSheets[0]; var rules = styleSheet.cssRules; for (var i = 0; i < rules.length; i++) { if (rules[i].cssText.substr(0, 6) == 'canvas') { styleSheet.deleteRule(i); i--; } } styleSheet.insertRule('canvas.emscripten { border: none; cursor: auto; }', 0);\0".as_ptr() as *const c_char); ffi::emscripten_asm_const(b"var styleSheet = document.styleSheets[0]; var rules = styleSheet.cssRules; for (var i = 0; i < rules.length; i++) { if (rules[i].cssText.substr(0, 6) == 'canvas') { styleSheet.deleteRule(i); i--; } } styleSheet.insertRule('canvas.emscripten { border: none; cursor: auto; }', 0);\0".as_ptr() as *const c_char);
} }
} }
extern "C" fn mouse_callback( extern "C" fn mouse_callback(
event_type: c_int, event_type: c_int,
event: *const ffi::EmscriptenMouseEvent, event: *const ffi::EmscriptenMouseEvent,
event_queue: *mut c_void) -> ffi::EM_BOOL event_queue: *mut c_void,
{ ) -> ffi::EM_BOOL {
unsafe { unsafe {
let queue: &Mutex<VecDeque<::Event>> = mem::transmute(event_queue); let queue: &Mutex<VecDeque<::Event>> = mem::transmute(event_queue);
@ -221,18 +242,18 @@ extern "C" fn mouse_callback(
event: ::WindowEvent::CursorMoved { event: ::WindowEvent::CursorMoved {
device_id: ::DeviceId(DeviceId), device_id: ::DeviceId(DeviceId),
position, position,
modifiers: modifiers, modifiers,
} },
}); });
queue.lock().unwrap().push_back(::Event::DeviceEvent { queue.lock().unwrap().push_back(::Event::DeviceEvent {
device_id: ::DeviceId(DeviceId), device_id: ::DeviceId(DeviceId),
event: ::DeviceEvent::MouseMotion { event: ::DeviceEvent::MouseMotion {
delta: ((*event).movementX as f64, (*event).movementY as f64), delta: ((*event).movementX as f64, (*event).movementY as f64),
} },
}); });
}, },
mouse_input @ ffi::EMSCRIPTEN_EVENT_MOUSEDOWN | mouse_input @ ffi::EMSCRIPTEN_EVENT_MOUSEDOWN
mouse_input @ ffi::EMSCRIPTEN_EVENT_MOUSEUP => { | mouse_input @ ffi::EMSCRIPTEN_EVENT_MOUSEUP => {
let button = match (*event).button { let button = match (*event).button {
0 => ::MouseButton::Left, 0 => ::MouseButton::Left,
1 => ::MouseButton::Middle, 1 => ::MouseButton::Middle,
@ -248,14 +269,13 @@ extern "C" fn mouse_callback(
window_id: ::WindowId(WindowId(0)), window_id: ::WindowId(WindowId(0)),
event: ::WindowEvent::MouseInput { event: ::WindowEvent::MouseInput {
device_id: ::DeviceId(DeviceId), device_id: ::DeviceId(DeviceId),
state: state, state,
button: button, button,
modifiers: modifiers, modifiers,
} },
}) })
}, },
_ => { _ => {},
}
} }
} }
ffi::EM_FALSE ffi::EM_FALSE
@ -264,8 +284,8 @@ extern "C" fn mouse_callback(
extern "C" fn keyboard_callback( extern "C" fn keyboard_callback(
event_type: c_int, event_type: c_int,
event: *const ffi::EmscriptenKeyboardEvent, event: *const ffi::EmscriptenKeyboardEvent,
event_queue: *mut c_void) -> ffi::EM_BOOL event_queue: *mut c_void,
{ ) -> ffi::EM_BOOL {
unsafe { unsafe {
let queue: &Mutex<VecDeque<::Event>> = mem::transmute(event_queue); let queue: &Mutex<VecDeque<::Event>> = mem::transmute(event_queue);
@ -305,18 +325,17 @@ extern "C" fn keyboard_callback(
}, },
}); });
}, },
_ => { _ => {},
}
} }
} }
ffi::EM_FALSE ffi::EM_FALSE
} }
extern fn touch_callback( extern "C" fn touch_callback(
event_type: c_int, event_type: c_int,
event: *const ffi::EmscriptenTouchEvent, event: *const ffi::EmscriptenTouchEvent,
event_queue: *mut c_void) -> ffi::EM_BOOL event_queue: *mut c_void,
{ ) -> ffi::EM_BOOL {
unsafe { unsafe {
let queue: &Mutex<VecDeque<::Event>> = mem::transmute(event_queue); let queue: &Mutex<VecDeque<::Event>> = mem::transmute(event_queue);
@ -356,8 +375,8 @@ extern fn touch_callback(
unsafe extern "C" fn fullscreen_callback( unsafe extern "C" fn fullscreen_callback(
_eventType: c_int, _eventType: c_int,
_fullscreenChangeEvent: *const ffi::EmscriptenFullscreenChangeEvent, _fullscreenChangeEvent: *const ffi::EmscriptenFullscreenChangeEvent,
_userData: *mut c_void) -> ffi::EM_BOOL _userData: *mut c_void,
{ ) -> ffi::EM_BOOL {
ffi::emscripten_request_fullscreen(ptr::null(), ffi::EM_TRUE); ffi::emscripten_request_fullscreen(ptr::null(), ffi::EM_TRUE);
ffi::EM_FALSE ffi::EM_FALSE
} }
@ -367,8 +386,8 @@ unsafe extern "C" fn fullscreen_callback(
unsafe extern "C" fn pointerlockchange_callback( unsafe extern "C" fn pointerlockchange_callback(
_eventType: c_int, _eventType: c_int,
_pointerlockChangeEvent: *const ffi::EmscriptenPointerlockChangeEvent, _pointerlockChangeEvent: *const ffi::EmscriptenPointerlockChangeEvent,
_userData: *mut c_void) -> ffi::EM_BOOL _userData: *mut c_void,
{ ) -> ffi::EM_BOOL {
ffi::emscripten_request_pointerlock(ptr::null(), ffi::EM_TRUE); ffi::emscripten_request_pointerlock(ptr::null(), ffi::EM_TRUE);
ffi::EM_FALSE ffi::EM_FALSE
} }
@ -381,12 +400,15 @@ fn em_try(res: ffi::EMSCRIPTEN_RESULT) -> Result<(), String> {
} }
impl Window { impl Window {
pub fn new(event_loop: &EventLoop, attribs: ::WindowAttributes, pub fn new(
_pl_attribs: PlatformSpecificWindowBuilderAttributes) event_loop: &EventLoop,
-> Result<Window, ::CreationError> attribs: ::WindowAttributes,
{ _pl_attribs: PlatformSpecificWindowBuilderAttributes,
) -> Result<Window, ::CreationError> {
if event_loop.window.lock().unwrap().is_some() { if event_loop.window.lock().unwrap().is_some() {
return Err(::CreationError::OsError("Cannot create another window".to_owned())); return Err(::CreationError::OsError(
"Cannot create another window".to_owned(),
));
} }
let w = Window2 { let w = Window2 {
@ -400,35 +422,87 @@ impl Window {
window: Arc::new(w), window: Arc::new(w),
}; };
// TODO: set up more event callbacks // TODO: set up more event callbacks
unsafe { unsafe {
em_try(ffi::emscripten_set_mousemove_callback(DOCUMENT_NAME.as_ptr() as *const c_char, mem::transmute(&*window.window.events), ffi::EM_FALSE, Some(mouse_callback))) em_try(ffi::emscripten_set_mousemove_callback(
.map_err(|e| ::CreationError::OsError(format!("emscripten error: {}", e)))?; DOCUMENT_NAME.as_ptr() as *const c_char,
em_try(ffi::emscripten_set_mousedown_callback(DOCUMENT_NAME.as_ptr() as *const c_char, mem::transmute(&*window.window.events), ffi::EM_FALSE, Some(mouse_callback))) mem::transmute(&*window.window.events),
.map_err(|e| ::CreationError::OsError(format!("emscripten error: {}", e)))?; ffi::EM_FALSE,
em_try(ffi::emscripten_set_mouseup_callback(DOCUMENT_NAME.as_ptr() as *const c_char, mem::transmute(&*window.window.events), ffi::EM_FALSE, Some(mouse_callback))) Some(mouse_callback),
.map_err(|e| ::CreationError::OsError(format!("emscripten error: {}", e)))?; ))
em_try(ffi::emscripten_set_keydown_callback(DOCUMENT_NAME.as_ptr() as *const c_char, mem::transmute(&*window.window.events), ffi::EM_FALSE, Some(keyboard_callback))) .map_err(|e| ::CreationError::OsError(format!("emscripten error: {}", e)))?;
.map_err(|e| ::CreationError::OsError(format!("emscripten error: {}", e)))?; em_try(ffi::emscripten_set_mousedown_callback(
em_try(ffi::emscripten_set_keyup_callback(DOCUMENT_NAME.as_ptr() as *const c_char, mem::transmute(&*window.window.events), ffi::EM_FALSE, Some(keyboard_callback))) DOCUMENT_NAME.as_ptr() as *const c_char,
.map_err(|e| ::CreationError::OsError(format!("emscripten error: {}", e)))?; mem::transmute(&*window.window.events),
em_try(ffi::emscripten_set_touchstart_callback(DOCUMENT_NAME.as_ptr() as *const c_char, mem::transmute(&*window.window.events), ffi::EM_FALSE, Some(touch_callback))) ffi::EM_FALSE,
.map_err(|e| ::CreationError::OsError(format!("emscripten error: {}", e)))?; Some(mouse_callback),
em_try(ffi::emscripten_set_touchend_callback(DOCUMENT_NAME.as_ptr() as *const c_char, mem::transmute(&*window.window.events), ffi::EM_FALSE, Some(touch_callback))) ))
.map_err(|e| ::CreationError::OsError(format!("emscripten error: {}", e)))?; .map_err(|e| ::CreationError::OsError(format!("emscripten error: {}", e)))?;
em_try(ffi::emscripten_set_touchmove_callback(DOCUMENT_NAME.as_ptr() as *const c_char, mem::transmute(&*window.window.events), ffi::EM_FALSE, Some(touch_callback))) em_try(ffi::emscripten_set_mouseup_callback(
.map_err(|e| ::CreationError::OsError(format!("emscripten error: {}", e)))?; DOCUMENT_NAME.as_ptr() as *const c_char,
em_try(ffi::emscripten_set_touchcancel_callback(DOCUMENT_NAME.as_ptr() as *const c_char, mem::transmute(&*window.window.events), ffi::EM_FALSE, Some(touch_callback))) mem::transmute(&*window.window.events),
.map_err(|e| ::CreationError::OsError(format!("emscripten error: {}", e)))?; ffi::EM_FALSE,
Some(mouse_callback),
))
.map_err(|e| ::CreationError::OsError(format!("emscripten error: {}", e)))?;
em_try(ffi::emscripten_set_keydown_callback(
DOCUMENT_NAME.as_ptr() as *const c_char,
mem::transmute(&*window.window.events),
ffi::EM_FALSE,
Some(keyboard_callback),
))
.map_err(|e| ::CreationError::OsError(format!("emscripten error: {}", e)))?;
em_try(ffi::emscripten_set_keyup_callback(
DOCUMENT_NAME.as_ptr() as *const c_char,
mem::transmute(&*window.window.events),
ffi::EM_FALSE,
Some(keyboard_callback),
))
.map_err(|e| ::CreationError::OsError(format!("emscripten error: {}", e)))?;
em_try(ffi::emscripten_set_touchstart_callback(
DOCUMENT_NAME.as_ptr() as *const c_char,
mem::transmute(&*window.window.events),
ffi::EM_FALSE,
Some(touch_callback),
))
.map_err(|e| ::CreationError::OsError(format!("emscripten error: {}", e)))?;
em_try(ffi::emscripten_set_touchend_callback(
DOCUMENT_NAME.as_ptr() as *const c_char,
mem::transmute(&*window.window.events),
ffi::EM_FALSE,
Some(touch_callback),
))
.map_err(|e| ::CreationError::OsError(format!("emscripten error: {}", e)))?;
em_try(ffi::emscripten_set_touchmove_callback(
DOCUMENT_NAME.as_ptr() as *const c_char,
mem::transmute(&*window.window.events),
ffi::EM_FALSE,
Some(touch_callback),
))
.map_err(|e| ::CreationError::OsError(format!("emscripten error: {}", e)))?;
em_try(ffi::emscripten_set_touchcancel_callback(
DOCUMENT_NAME.as_ptr() as *const c_char,
mem::transmute(&*window.window.events),
ffi::EM_FALSE,
Some(touch_callback),
))
.map_err(|e| ::CreationError::OsError(format!("emscripten error: {}", e)))?;
} }
if attribs.fullscreen.is_some() { if attribs.fullscreen.is_some() {
unsafe { unsafe {
em_try(ffi::emscripten_request_fullscreen(ptr::null(), ffi::EM_TRUE)) em_try(ffi::emscripten_request_fullscreen(
.map_err(|e| ::CreationError::OsError(e))?; ptr::null(),
em_try(ffi::emscripten_set_fullscreenchange_callback(ptr::null(), 0 as *mut c_void, ffi::EM_FALSE, Some(fullscreen_callback))) ffi::EM_TRUE,
.map_err(|e| ::CreationError::OsError(e))?; ))
.map_err(|e| ::CreationError::OsError(e))?;
em_try(ffi::emscripten_set_fullscreenchange_callback(
ptr::null(),
0 as *mut c_void,
ffi::EM_FALSE,
Some(fullscreen_callback),
))
.map_err(|e| ::CreationError::OsError(e))?;
} }
} else if let Some(size) = attribs.inner_size { } else if let Some(size) = attribs.inner_size {
window.set_inner_size(size); window.set_inner_size(size);
@ -444,8 +518,7 @@ impl Window {
} }
#[inline] #[inline]
pub fn set_title(&self, _title: &str) { pub fn set_title(&self, _title: &str) {}
}
#[inline] #[inline]
pub fn outer_position(&self) -> Option<LogicalPosition> { pub fn outer_position(&self) -> Option<LogicalPosition> {
@ -458,8 +531,7 @@ impl Window {
} }
#[inline] #[inline]
pub fn set_outer_position(&self, _: LogicalPosition) { pub fn set_outer_position(&self, _: LogicalPosition) {}
}
#[inline] #[inline]
pub fn inner_size(&self) -> Option<LogicalSize> { pub fn inner_size(&self) -> Option<LogicalSize> {
@ -532,7 +604,9 @@ impl Window {
#[inline] #[inline]
pub fn set_cursor_grab(&self, grab: bool) -> Result<(), ExternalError> { pub fn set_cursor_grab(&self, grab: bool) -> Result<(), ExternalError> {
let mut grabbed_lock = self.window.cursor_grabbed.lock().unwrap(); let mut grabbed_lock = self.window.cursor_grabbed.lock().unwrap();
if grab == *grabbed_lock { return Ok(()); } if grab == *grabbed_lock {
return Ok(());
}
unsafe { unsafe {
if grab { if grab {
em_try(ffi::emscripten_set_pointerlockchange_callback( em_try(ffi::emscripten_set_pointerlockchange_callback(
@ -541,7 +615,10 @@ impl Window {
ffi::EM_FALSE, ffi::EM_FALSE,
Some(pointerlockchange_callback), Some(pointerlockchange_callback),
))?; ))?;
em_try(ffi::emscripten_request_pointerlock(ptr::null(), ffi::EM_TRUE))?; em_try(ffi::emscripten_request_pointerlock(
ptr::null(),
ffi::EM_TRUE,
))?;
} else { } else {
em_try(ffi::emscripten_set_pointerlockchange_callback( em_try(ffi::emscripten_set_pointerlockchange_callback(
ptr::null(), ptr::null(),
@ -559,7 +636,9 @@ impl Window {
#[inline] #[inline]
pub fn set_cursor_visible(&self, visible: bool) { pub fn set_cursor_visible(&self, visible: bool) {
let mut visible_lock = self.window.cursor_visible.lock().unwrap(); let mut visible_lock = self.window.cursor_visible.lock().unwrap();
if visible == *visible_lock { return; } if visible == *visible_lock {
return;
}
if visible { if visible {
show_mouse(); show_mouse();
} else { } else {
@ -615,7 +694,9 @@ impl Window {
#[inline] #[inline]
pub fn current_monitor(&self) -> RootMonitorHandle { pub fn current_monitor(&self) -> RootMonitorHandle {
RootMonitorHandle { inner: MonitorHandle } RootMonitorHandle {
inner: MonitorHandle,
}
} }
#[inline] #[inline]
@ -646,21 +727,37 @@ impl Drop for Window {
// Exit fullscreen if on // Exit fullscreen if on
if self.window.is_fullscreen { if self.window.is_fullscreen {
ffi::emscripten_set_fullscreenchange_callback(ptr::null(), 0 as *mut c_void, ffi::EM_FALSE, None); ffi::emscripten_set_fullscreenchange_callback(
ptr::null(),
0 as *mut c_void,
ffi::EM_FALSE,
None,
);
ffi::emscripten_exit_fullscreen(); ffi::emscripten_exit_fullscreen();
} }
// Delete callbacks // Delete callbacks
ffi::emscripten_set_keydown_callback(DOCUMENT_NAME.as_ptr() as *const c_char, 0 as *mut c_void, ffi::EM_FALSE,None); ffi::emscripten_set_keydown_callback(
ffi::emscripten_set_keyup_callback(DOCUMENT_NAME.as_ptr() as *const c_char, 0 as *mut c_void, ffi::EM_FALSE,None); DOCUMENT_NAME.as_ptr() as *const c_char,
0 as *mut c_void,
ffi::EM_FALSE,
None,
);
ffi::emscripten_set_keyup_callback(
DOCUMENT_NAME.as_ptr() as *const c_char,
0 as *mut c_void,
ffi::EM_FALSE,
None,
);
} }
} }
} }
fn error_to_str(code: ffi::EMSCRIPTEN_RESULT) -> &'static str { fn error_to_str(code: ffi::EMSCRIPTEN_RESULT) -> &'static str {
match code { match code {
ffi::EMSCRIPTEN_RESULT_SUCCESS | ffi::EMSCRIPTEN_RESULT_DEFERRED ffi::EMSCRIPTEN_RESULT_SUCCESS | ffi::EMSCRIPTEN_RESULT_DEFERRED => {
=> "Internal error in the library (success detected as failure)", "Internal error in the library (success detected as failure)"
},
ffi::EMSCRIPTEN_RESULT_NOT_SUPPORTED => "Not supported", ffi::EMSCRIPTEN_RESULT_NOT_SUPPORTED => "Not supported",
ffi::EMSCRIPTEN_RESULT_FAILED_NOT_DEFERRED => "Failed not deferred", ffi::EMSCRIPTEN_RESULT_FAILED_NOT_DEFERRED => "Failed not deferred",
@ -670,7 +767,7 @@ fn error_to_str(code: ffi::EMSCRIPTEN_RESULT) -> &'static str {
ffi::EMSCRIPTEN_RESULT_FAILED => "Failed", ffi::EMSCRIPTEN_RESULT_FAILED => "Failed",
ffi::EMSCRIPTEN_RESULT_NO_DATA => "No data", ffi::EMSCRIPTEN_RESULT_NO_DATA => "No data",
_ => "Undocumented error" _ => "Undocumented error",
} }
} }
@ -679,7 +776,9 @@ fn key_translate(input: [ffi::EM_UTF8; ffi::EM_HTML5_SHORT_STRING_LEN_BYTES]) ->
let maybe_key = unsafe { str::from_utf8(mem::transmute::<_, &[u8]>(slice)) }; let maybe_key = unsafe { str::from_utf8(mem::transmute::<_, &[u8]>(slice)) };
let key = match maybe_key { let key = match maybe_key {
Ok(key) => key, Ok(key) => key,
Err(_) => { return 0; }, Err(_) => {
return 0;
},
}; };
if key.chars().count() == 1 { if key.chars().count() == 1 {
key.as_bytes()[0] key.as_bytes()[0]
@ -688,28 +787,35 @@ fn key_translate(input: [ffi::EM_UTF8; ffi::EM_HTML5_SHORT_STRING_LEN_BYTES]) ->
} }
} }
fn key_translate_virt(input: [ffi::EM_UTF8; ffi::EM_HTML5_SHORT_STRING_LEN_BYTES], fn key_translate_virt(
location: c_ulong) -> Option<::VirtualKeyCode> input: [ffi::EM_UTF8; ffi::EM_HTML5_SHORT_STRING_LEN_BYTES],
{ location: c_ulong,
) -> Option<::VirtualKeyCode> {
let slice = &input[0..input.iter().take_while(|x| **x != 0).count()]; let slice = &input[0..input.iter().take_while(|x| **x != 0).count()];
let maybe_key = unsafe { str::from_utf8(mem::transmute::<_, &[u8]>(slice)) }; let maybe_key = unsafe { str::from_utf8(mem::transmute::<_, &[u8]>(slice)) };
let key = match maybe_key { let key = match maybe_key {
Ok(key) => key, Ok(key) => key,
Err(_) => { return None; }, Err(_) => {
return None;
},
}; };
use VirtualKeyCode::*; use VirtualKeyCode::*;
match key { match key {
"Alt" => match location { "Alt" => {
ffi::DOM_KEY_LOCATION_LEFT => Some(LAlt), match location {
ffi::DOM_KEY_LOCATION_RIGHT => Some(RAlt), ffi::DOM_KEY_LOCATION_LEFT => Some(LAlt),
_ => None, ffi::DOM_KEY_LOCATION_RIGHT => Some(RAlt),
_ => None,
}
}, },
"AltGraph" => None, "AltGraph" => None,
"CapsLock" => None, "CapsLock" => None,
"Control" => match location { "Control" => {
ffi::DOM_KEY_LOCATION_LEFT => Some(LControl), match location {
ffi::DOM_KEY_LOCATION_RIGHT => Some(RControl), ffi::DOM_KEY_LOCATION_LEFT => Some(LControl),
_ => None, ffi::DOM_KEY_LOCATION_RIGHT => Some(RControl),
_ => None,
}
}, },
"Fn" => None, "Fn" => None,
"FnLock" => None, "FnLock" => None,
@ -717,18 +823,22 @@ fn key_translate_virt(input: [ffi::EM_UTF8; ffi::EM_HTML5_SHORT_STRING_LEN_BYTES
"Meta" => None, "Meta" => None,
"NumLock" => Some(Numlock), "NumLock" => Some(Numlock),
"ScrollLock" => Some(Scroll), "ScrollLock" => Some(Scroll),
"Shift" => match location { "Shift" => {
ffi::DOM_KEY_LOCATION_LEFT => Some(LShift), match location {
ffi::DOM_KEY_LOCATION_RIGHT => Some(RShift), ffi::DOM_KEY_LOCATION_LEFT => Some(LShift),
_ => None, ffi::DOM_KEY_LOCATION_RIGHT => Some(RShift),
_ => None,
}
}, },
"Super" => None, "Super" => None,
"Symbol" => None, "Symbol" => None,
"SymbolLock" => None, "SymbolLock" => None,
"Enter" => match location { "Enter" => {
ffi::DOM_KEY_LOCATION_NUMPAD => Some(NumpadEnter), match location {
_ => Some(Return), ffi::DOM_KEY_LOCATION_NUMPAD => Some(NumpadEnter),
_ => Some(Return),
}
}, },
"Tab" => Some(Tab), "Tab" => Some(Tab),
" " => Some(Space), " " => Some(Space),
@ -1053,45 +1163,65 @@ fn key_translate_virt(input: [ffi::EM_UTF8; ffi::EM_HTML5_SHORT_STRING_LEN_BYTES
"Divide" => Some(Divide), "Divide" => Some(Divide),
"Subtract" | "-" => Some(Subtract), "Subtract" | "-" => Some(Subtract),
"Separator" => None, "Separator" => None,
"0" => match location { "0" => {
ffi::DOM_KEY_LOCATION_NUMPAD => Some(Numpad0), match location {
_ => Some(Key0), ffi::DOM_KEY_LOCATION_NUMPAD => Some(Numpad0),
_ => Some(Key0),
}
}, },
"1" => match location { "1" => {
ffi::DOM_KEY_LOCATION_NUMPAD => Some(Numpad1), match location {
_ => Some(Key1), ffi::DOM_KEY_LOCATION_NUMPAD => Some(Numpad1),
_ => Some(Key1),
}
}, },
"2" => match location { "2" => {
ffi::DOM_KEY_LOCATION_NUMPAD => Some(Numpad2), match location {
_ => Some(Key2), ffi::DOM_KEY_LOCATION_NUMPAD => Some(Numpad2),
_ => Some(Key2),
}
}, },
"3" => match location { "3" => {
ffi::DOM_KEY_LOCATION_NUMPAD => Some(Numpad3), match location {
_ => Some(Key3), ffi::DOM_KEY_LOCATION_NUMPAD => Some(Numpad3),
_ => Some(Key3),
}
}, },
"4" => match location { "4" => {
ffi::DOM_KEY_LOCATION_NUMPAD => Some(Numpad4), match location {
_ => Some(Key4), ffi::DOM_KEY_LOCATION_NUMPAD => Some(Numpad4),
_ => Some(Key4),
}
}, },
"5" => match location { "5" => {
ffi::DOM_KEY_LOCATION_NUMPAD => Some(Numpad5), match location {
_ => Some(Key5), ffi::DOM_KEY_LOCATION_NUMPAD => Some(Numpad5),
_ => Some(Key5),
}
}, },
"6" => match location { "6" => {
ffi::DOM_KEY_LOCATION_NUMPAD => Some(Numpad6), match location {
_ => Some(Key6), ffi::DOM_KEY_LOCATION_NUMPAD => Some(Numpad6),
_ => Some(Key6),
}
}, },
"7" => match location { "7" => {
ffi::DOM_KEY_LOCATION_NUMPAD => Some(Numpad7), match location {
_ => Some(Key7), ffi::DOM_KEY_LOCATION_NUMPAD => Some(Numpad7),
_ => Some(Key7),
}
}, },
"8" => match location { "8" => {
ffi::DOM_KEY_LOCATION_NUMPAD => Some(Numpad8), match location {
_ => Some(Key8), ffi::DOM_KEY_LOCATION_NUMPAD => Some(Numpad8),
_ => Some(Key8),
}
}, },
"9" => match location { "9" => {
ffi::DOM_KEY_LOCATION_NUMPAD => Some(Numpad9), match location {
_ => Some(Key9), ffi::DOM_KEY_LOCATION_NUMPAD => Some(Numpad9),
_ => Some(Key9),
}
}, },
"A" | "a" => Some(A), "A" | "a" => Some(A),
@ -1124,13 +1254,17 @@ fn key_translate_virt(input: [ffi::EM_UTF8; ffi::EM_HTML5_SHORT_STRING_LEN_BYTES
"'" => Some(Apostrophe), "'" => Some(Apostrophe),
"\\" => Some(Backslash), "\\" => Some(Backslash),
":" => Some(Colon), ":" => Some(Colon),
"," => match location { "," => {
ffi::DOM_KEY_LOCATION_NUMPAD => Some(NumpadComma), match location {
_ => Some(Comma), ffi::DOM_KEY_LOCATION_NUMPAD => Some(NumpadComma),
_ => Some(Comma),
}
}, },
"=" => match location { "=" => {
ffi::DOM_KEY_LOCATION_NUMPAD => Some(NumpadEquals), match location {
_ => Some(Equals), ffi::DOM_KEY_LOCATION_NUMPAD => Some(NumpadEquals),
_ => Some(Equals),
}
}, },
"{" => Some(LBracket), "{" => Some(LBracket),
"." => Some(Period), "." => Some(Period),

View file

@ -1,26 +1,23 @@
use std::{mem, ptr}; use std::{
use std::cell::{RefCell, RefMut}; cell::{RefCell, RefMut},
use std::mem::ManuallyDrop; mem::{self, ManuallyDrop},
use std::os::raw::c_void; os::raw::c_void,
use std::time::Instant; ptr,
time::Instant,
};
use crate::event::{Event, StartCause}; use crate::{
use crate::event_loop::ControlFlow; event::{Event, StartCause},
event_loop::ControlFlow,
};
use crate::platform_impl::platform::event_loop::{EventHandler, Never}; use crate::platform_impl::platform::{
use crate::platform_impl::platform::ffi::{ event_loop::{EventHandler, Never},
id, ffi::{
CFAbsoluteTimeGetCurrent, id, kCFRunLoopCommonModes, CFAbsoluteTimeGetCurrent, CFRelease, CFRunLoopAddTimer,
CFRelease, CFRunLoopGetMain, CFRunLoopRef, CFRunLoopTimerCreate, CFRunLoopTimerInvalidate,
CFRunLoopAddTimer, CFRunLoopTimerRef, CFRunLoopTimerSetNextFireDate, NSUInteger,
CFRunLoopGetMain, },
CFRunLoopRef,
CFRunLoopTimerCreate,
CFRunLoopTimerInvalidate,
CFRunLoopTimerRef,
CFRunLoopTimerSetNextFireDate,
kCFRunLoopCommonModes,
NSUInteger,
}; };
macro_rules! bug { macro_rules! bug {
@ -62,13 +59,19 @@ enum AppStateImpl {
impl Drop for AppStateImpl { impl Drop for AppStateImpl {
fn drop(&mut self) { fn drop(&mut self) {
match self { match self {
&mut AppStateImpl::NotLaunched { ref mut queued_windows, .. } | &mut AppStateImpl::NotLaunched {
&mut AppStateImpl::Launching { ref mut queued_windows, .. } => unsafe { ref mut queued_windows,
..
}
| &mut AppStateImpl::Launching {
ref mut queued_windows,
..
} => unsafe {
for &mut window in queued_windows { for &mut window in queued_windows {
let () = msg_send![window, release]; let () = msg_send![window, release];
} }
} },
_ => {} _ => {},
} }
} }
} }
@ -88,7 +91,9 @@ impl AppState {
static mut APP_STATE: RefCell<Option<AppState>> = RefCell::new(None); static mut APP_STATE: RefCell<Option<AppState>> = RefCell::new(None);
if cfg!(debug_assertions) { if cfg!(debug_assertions) {
assert_main_thread!("bug in winit: `AppState::get_mut()` can only be called on the main thread"); assert_main_thread!(
"bug in winit: `AppState::get_mut()` can only be called on the main thread"
);
} }
let mut guard = APP_STATE.borrow_mut(); let mut guard = APP_STATE.borrow_mut();
@ -108,9 +113,7 @@ impl AppState {
} }
init_guard(&mut guard) init_guard(&mut guard)
} }
RefMut::map(guard, |state| { RefMut::map(guard, |state| state.as_mut().unwrap())
state.as_mut().unwrap()
})
} }
// requires main thread and window is a UIWindow // requires main thread and window is a UIWindow
@ -118,16 +121,23 @@ impl AppState {
pub unsafe fn set_key_window(window: id) { pub unsafe fn set_key_window(window: id) {
let mut this = AppState::get_mut(); let mut this = AppState::get_mut();
match &mut this.app_state { match &mut this.app_state {
&mut AppStateImpl::NotLaunched { ref mut queued_windows, .. } => { &mut AppStateImpl::NotLaunched {
ref mut queued_windows,
..
} => {
queued_windows.push(window); queued_windows.push(window);
msg_send![window, retain]; msg_send![window, retain];
return; return;
} },
&mut AppStateImpl::ProcessingEvents { .. } => {}, &mut AppStateImpl::ProcessingEvents { .. } => {},
&mut AppStateImpl::InUserCallback { .. } => {}, &mut AppStateImpl::InUserCallback { .. } => {},
&mut AppStateImpl::Terminated => panic!("Attempt to create a `Window` \ &mut AppStateImpl::Terminated => {
after the app has terminated"), panic!(
app_state => unreachable!("unexpected state: {:#?}", app_state), // all other cases should be impossible "Attempt to create a `Window` \
after the app has terminated"
)
},
app_state => unreachable!("unexpected state: {:#?}", app_state), /* all other cases should be impossible */
} }
drop(this); drop(this);
msg_send![window, makeKeyAndVisible] msg_send![window, makeKeyAndVisible]
@ -144,15 +154,22 @@ impl AppState {
let windows = ptr::read(queued_windows); let windows = ptr::read(queued_windows);
let events = ptr::read(queued_events); let events = ptr::read(queued_events);
(windows, events) (windows, events)
} },
_ => panic!("winit iOS expected the app to be in a `NotLaunched` \ _ => {
state, but was not - please file an issue"), panic!(
"winit iOS expected the app to be in a `NotLaunched` \
state, but was not - please file an issue"
)
},
}; };
ptr::write(&mut this.app_state, AppStateImpl::Launching { ptr::write(
queued_windows, &mut this.app_state,
queued_events, AppStateImpl::Launching {
queued_event_handler, queued_windows,
}); queued_events,
queued_event_handler,
},
);
} }
// requires main thread // requires main thread
@ -163,10 +180,12 @@ impl AppState {
ref mut queued_windows, ref mut queued_windows,
.. ..
} => mem::replace(queued_windows, Vec::new()), } => mem::replace(queued_windows, Vec::new()),
_ => panic!( _ => {
"winit iOS expected the app to be in a `Launching` \ panic!(
state, but was not - please file an issue" "winit iOS expected the app to be in a `Launching` \
), state, but was not - please file an issue"
)
},
}; };
// have to drop RefMut because the window setup code below can trigger new events // have to drop RefMut because the window setup code below can trigger new events
drop(this); drop(this);
@ -188,11 +207,11 @@ impl AppState {
let screen: id = msg_send![window, screen]; let screen: id = msg_send![window, screen];
let () = msg_send![screen, retain]; let () = msg_send![screen, retain];
let () = msg_send![window, setScreen:0 as id]; let () = msg_send![window, setScreen:0 as id];
let () = msg_send![window, setScreen:screen]; let () = msg_send![window, setScreen: screen];
let () = msg_send![screen, release]; let () = msg_send![screen, release];
let controller: id = msg_send![window, rootViewController]; let controller: id = msg_send![window, rootViewController];
let () = msg_send![window, setRootViewController:ptr::null::<()>()]; let () = msg_send![window, setRootViewController:ptr::null::<()>()];
let () = msg_send![window, setRootViewController:controller]; let () = msg_send![window, setRootViewController: controller];
let () = msg_send![window, makeKeyAndVisible]; let () = msg_send![window, makeKeyAndVisible];
} }
let () = msg_send![window, release]; let () = msg_send![window, release];
@ -209,14 +228,21 @@ impl AppState {
let events = ptr::read(queued_events); let events = ptr::read(queued_events);
let event_handler = ptr::read(queued_event_handler); let event_handler = ptr::read(queued_event_handler);
(windows, events, event_handler) (windows, events, event_handler)
} },
_ => panic!("winit iOS expected the app to be in a `Launching` \ _ => {
state, but was not - please file an issue"), panic!(
"winit iOS expected the app to be in a `Launching` \
state, but was not - please file an issue"
)
},
}; };
ptr::write(&mut this.app_state, AppStateImpl::ProcessingEvents { ptr::write(
event_handler, &mut this.app_state,
active_control_flow: ControlFlow::Poll, AppStateImpl::ProcessingEvents {
}); event_handler,
active_control_flow: ControlFlow::Poll,
},
);
drop(this); drop(this);
let events = std::iter::once(Event::NewEvents(StartCause::Init)).chain(events); let events = std::iter::once(Event::NewEvents(StartCause::Init)).chain(events);
@ -238,69 +264,79 @@ impl AppState {
// AppState::did_finish_launching handles the special transition `Init` // AppState::did_finish_launching handles the special transition `Init`
pub unsafe fn handle_wakeup_transition() { pub unsafe fn handle_wakeup_transition() {
let mut this = AppState::get_mut(); let mut this = AppState::get_mut();
let event = match this.control_flow { let event =
ControlFlow::Poll => { match this.control_flow {
let event_handler = match &mut this.app_state { ControlFlow::Poll => {
&mut AppStateImpl::NotLaunched { .. } | let event_handler = match &mut this.app_state {
&mut AppStateImpl::Launching { .. } => return, &mut AppStateImpl::NotLaunched { .. }
&mut AppStateImpl::PollFinished { | &mut AppStateImpl::Launching { .. } => return,
ref mut waiting_event_handler, &mut AppStateImpl::PollFinished {
} => ptr::read(waiting_event_handler), ref mut waiting_event_handler,
_ => bug!("`EventHandler` unexpectedly started polling"), } => ptr::read(waiting_event_handler),
}; _ => bug!("`EventHandler` unexpectedly started polling"),
ptr::write(&mut this.app_state, AppStateImpl::ProcessingEvents { };
event_handler, ptr::write(
active_control_flow: ControlFlow::Poll, &mut this.app_state,
}); AppStateImpl::ProcessingEvents {
Event::NewEvents(StartCause::Poll) event_handler,
} active_control_flow: ControlFlow::Poll,
ControlFlow::Wait => { },
let (event_handler, start) = match &mut this.app_state { );
&mut AppStateImpl::NotLaunched { .. } | Event::NewEvents(StartCause::Poll)
&mut AppStateImpl::Launching { .. } => return, },
&mut AppStateImpl::Waiting { ControlFlow::Wait => {
ref mut waiting_event_handler, let (event_handler, start) = match &mut this.app_state {
ref mut start, &mut AppStateImpl::NotLaunched { .. }
} => (ptr::read(waiting_event_handler), *start), | &mut AppStateImpl::Launching { .. } => return,
_ => bug!("`EventHandler` unexpectedly woke up"), &mut AppStateImpl::Waiting {
}; ref mut waiting_event_handler,
ptr::write(&mut this.app_state, AppStateImpl::ProcessingEvents { ref mut start,
event_handler, } => (ptr::read(waiting_event_handler), *start),
active_control_flow: ControlFlow::Wait, _ => bug!("`EventHandler` unexpectedly woke up"),
}); };
Event::NewEvents(StartCause::WaitCancelled { ptr::write(
start, &mut this.app_state,
requested_resume: None, AppStateImpl::ProcessingEvents {
}) event_handler,
} active_control_flow: ControlFlow::Wait,
ControlFlow::WaitUntil(requested_resume) => { },
let (event_handler, start) = match &mut this.app_state { );
&mut AppStateImpl::NotLaunched { .. } |
&mut AppStateImpl::Launching { .. } => return,
&mut AppStateImpl::Waiting {
ref mut waiting_event_handler,
ref mut start,
} => (ptr::read(waiting_event_handler), *start),
_ => bug!("`EventHandler` unexpectedly woke up"),
};
ptr::write(&mut this.app_state, AppStateImpl::ProcessingEvents {
event_handler,
active_control_flow: ControlFlow::WaitUntil(requested_resume),
});
if Instant::now() >= requested_resume {
Event::NewEvents(StartCause::ResumeTimeReached {
start,
requested_resume,
})
} else {
Event::NewEvents(StartCause::WaitCancelled { Event::NewEvents(StartCause::WaitCancelled {
start, start,
requested_resume: Some(requested_resume), requested_resume: None,
}) })
} },
} ControlFlow::WaitUntil(requested_resume) => {
ControlFlow::Exit => bug!("unexpected controlflow `Exit`"), let (event_handler, start) = match &mut this.app_state {
}; &mut AppStateImpl::NotLaunched { .. }
| &mut AppStateImpl::Launching { .. } => return,
&mut AppStateImpl::Waiting {
ref mut waiting_event_handler,
ref mut start,
} => (ptr::read(waiting_event_handler), *start),
_ => bug!("`EventHandler` unexpectedly woke up"),
};
ptr::write(
&mut this.app_state,
AppStateImpl::ProcessingEvents {
event_handler,
active_control_flow: ControlFlow::WaitUntil(requested_resume),
},
);
if Instant::now() >= requested_resume {
Event::NewEvents(StartCause::ResumeTimeReached {
start,
requested_resume,
})
} else {
Event::NewEvents(StartCause::WaitCancelled {
start,
requested_resume: Some(requested_resume),
})
}
},
ControlFlow::Exit => bug!("unexpected controlflow `Exit`"),
};
drop(this); drop(this);
AppState::handle_nonuser_event(event) AppState::handle_nonuser_event(event)
} }
@ -328,8 +364,8 @@ impl AppState {
.. ..
} => { } => {
queued_events.extend(events); queued_events.extend(events);
return return;
} },
&mut AppStateImpl::ProcessingEvents { &mut AppStateImpl::ProcessingEvents {
ref mut event_handler, ref mut event_handler,
ref mut active_control_flow, ref mut active_control_flow,
@ -338,9 +374,12 @@ impl AppState {
| &mut AppStateImpl::Waiting { .. } | &mut AppStateImpl::Waiting { .. }
| &mut AppStateImpl::Terminated => bug!("unexpected attempted to process an event"), | &mut AppStateImpl::Terminated => bug!("unexpected attempted to process an event"),
}; };
ptr::write(&mut this.app_state, AppStateImpl::InUserCallback { ptr::write(
queued_events: Vec::new(), &mut this.app_state,
}); AppStateImpl::InUserCallback {
queued_events: Vec::new(),
},
);
drop(this); drop(this);
for event in events { for event in events {
@ -360,7 +399,7 @@ impl AppState {
active_control_flow, active_control_flow,
}; };
this.control_flow = control_flow; this.control_flow = control_flow;
break break;
} }
drop(this); drop(this);
for event in queued_events { for event in queued_events {
@ -384,9 +423,12 @@ impl AppState {
| &mut AppStateImpl::Waiting { .. } | &mut AppStateImpl::Waiting { .. }
| &mut AppStateImpl::Terminated => bug!("unexpected attempted to process an event"), | &mut AppStateImpl::Terminated => bug!("unexpected attempted to process an event"),
}; };
ptr::write(&mut this.app_state, AppStateImpl::InUserCallback { ptr::write(
queued_events: Vec::new(), &mut this.app_state,
}); AppStateImpl::InUserCallback {
queued_events: Vec::new(),
},
);
drop(this); drop(this);
event_handler.handle_user_events(&mut control_flow); event_handler.handle_user_events(&mut control_flow);
@ -404,7 +446,7 @@ impl AppState {
active_control_flow, active_control_flow,
}; };
this.control_flow = control_flow; this.control_flow = control_flow;
break break;
} }
drop(this); drop(this);
for event in queued_events { for event in queued_events {
@ -419,7 +461,7 @@ impl AppState {
let mut this = AppState::get_mut(); let mut this = AppState::get_mut();
match &mut this.app_state { match &mut this.app_state {
&mut AppStateImpl::NotLaunched { .. } | &mut AppStateImpl::Launching { .. } => return, &mut AppStateImpl::NotLaunched { .. } | &mut AppStateImpl::Launching { .. } => return,
&mut AppStateImpl::ProcessingEvents { .. } => {} &mut AppStateImpl::ProcessingEvents { .. } => {},
_ => unreachable!(), _ => unreachable!(),
}; };
drop(this); drop(this);
@ -432,7 +474,12 @@ impl AppState {
&mut AppStateImpl::ProcessingEvents { &mut AppStateImpl::ProcessingEvents {
ref mut event_handler, ref mut event_handler,
ref mut active_control_flow, ref mut active_control_flow,
} => (ManuallyDrop::new(ptr::read(event_handler)), *active_control_flow), } => {
(
ManuallyDrop::new(ptr::read(event_handler)),
*active_control_flow,
)
},
_ => unreachable!(), _ => unreachable!(),
}; };
@ -457,7 +504,8 @@ impl AppState {
) )
}, },
(ControlFlow::WaitUntil(old_instant), ControlFlow::WaitUntil(new_instant)) (ControlFlow::WaitUntil(old_instant), ControlFlow::WaitUntil(new_instant))
if old_instant == new_instant => { if old_instant == new_instant =>
{
let start = Instant::now(); let start = Instant::now();
ptr::write( ptr::write(
&mut this.app_state, &mut this.app_state,
@ -503,7 +551,7 @@ impl AppState {
// it is not possible to quit an iOS app gracefully and programatically // it is not possible to quit an iOS app gracefully and programatically
warn!("`ControlFlow::Exit` ignored on iOS"); warn!("`ControlFlow::Exit` ignored on iOS");
this.control_flow = old this.control_flow = old
} },
} }
} }
@ -511,7 +559,11 @@ impl AppState {
let mut this = unsafe { AppState::get_mut() }; let mut this = unsafe { AppState::get_mut() };
let mut old = mem::replace(&mut this.app_state, AppStateImpl::Terminated); let mut old = mem::replace(&mut this.app_state, AppStateImpl::Terminated);
let mut control_flow = this.control_flow; let mut control_flow = this.control_flow;
if let AppStateImpl::ProcessingEvents { ref mut event_handler, .. } = old { if let AppStateImpl::ProcessingEvents {
ref mut event_handler,
..
} = old
{
drop(this); drop(this);
event_handler.handle_nonuser_event(Event::LoopDestroyed, &mut control_flow) event_handler.handle_nonuser_event(Event::LoopDestroyed, &mut control_flow)
} else { } else {
@ -535,7 +587,7 @@ impl Drop for EventLoopWaker {
impl EventLoopWaker { impl EventLoopWaker {
fn new(rl: CFRunLoopRef) -> EventLoopWaker { fn new(rl: CFRunLoopRef) -> EventLoopWaker {
extern fn wakeup_main_loop(_timer: CFRunLoopTimerRef, _info: *mut c_void) {} extern "C" fn wakeup_main_loop(_timer: CFRunLoopTimerRef, _info: *mut c_void) {}
unsafe { unsafe {
// create a timer with a 1microsec interval (1ns does not work) to mimic polling. // create a timer with a 1microsec interval (1ns does not work) to mimic polling.
// it is initially setup with a first fire time really far into the // it is initially setup with a first fire time really far into the

View file

@ -1,50 +1,33 @@
use std::{mem, ptr}; use std::{
use std::collections::VecDeque; collections::VecDeque,
use std::ffi::c_void; ffi::c_void,
use std::fmt::{self, Debug, Formatter}; fmt::{self, Debug, Formatter},
use std::marker::PhantomData; marker::PhantomData,
use std::sync::mpsc::{self, Sender, Receiver}; mem, ptr,
sync::mpsc::{self, Receiver, Sender},
use crate::event::Event;
use crate::event_loop::{
ControlFlow,
EventLoopWindowTarget as RootEventLoopWindowTarget,
EventLoopClosed,
}; };
use crate::platform::ios::Idiom;
use crate::platform_impl::platform::app_state::AppState; use crate::{
use crate::platform_impl::platform::ffi::{ event::Event,
id, event_loop::{
nil, ControlFlow, EventLoopClosed, EventLoopWindowTarget as RootEventLoopWindowTarget,
CFIndex, },
CFRelease, platform::ios::Idiom,
CFRunLoopActivity, };
CFRunLoopAddObserver,
CFRunLoopAddSource, use crate::platform_impl::platform::{
CFRunLoopGetMain, app_state::AppState,
CFRunLoopObserverCreate, ffi::{
CFRunLoopObserverRef, id, kCFRunLoopAfterWaiting, kCFRunLoopBeforeWaiting, kCFRunLoopCommonModes,
CFRunLoopSourceContext, kCFRunLoopDefaultMode, kCFRunLoopEntry, kCFRunLoopExit, nil, CFIndex, CFRelease,
CFRunLoopSourceCreate, CFRunLoopActivity, CFRunLoopAddObserver, CFRunLoopAddSource, CFRunLoopGetMain,
CFRunLoopSourceInvalidate, CFRunLoopObserverCreate, CFRunLoopObserverRef, CFRunLoopSourceContext,
CFRunLoopSourceRef, CFRunLoopSourceCreate, CFRunLoopSourceInvalidate, CFRunLoopSourceRef,
CFRunLoopSourceSignal, CFRunLoopSourceSignal, CFRunLoopWakeUp, NSOperatingSystemVersion, NSString,
CFRunLoopWakeUp, UIApplicationMain, UIUserInterfaceIdiom,
kCFRunLoopCommonModes, },
kCFRunLoopDefaultMode, monitor, view, MonitorHandle,
kCFRunLoopEntry,
kCFRunLoopBeforeWaiting,
kCFRunLoopAfterWaiting,
kCFRunLoopExit,
NSOperatingSystemVersion,
NSString,
UIApplicationMain,
UIUserInterfaceIdiom,
}; };
use crate::platform_impl::platform::monitor;
use crate::platform_impl::platform::MonitorHandle;
use crate::platform_impl::platform::view;
pub struct EventLoopWindowTarget<T: 'static> { pub struct EventLoopWindowTarget<T: 'static> {
receiver: Receiver<T>, receiver: Receiver<T>,
@ -67,8 +50,11 @@ impl<T: 'static> EventLoop<T> {
static mut SINGLETON_INIT: bool = false; static mut SINGLETON_INIT: bool = false;
unsafe { unsafe {
assert_main_thread!("`EventLoop` can only be created on the main thread on iOS"); assert_main_thread!("`EventLoop` can only be created on the main thread on iOS");
assert!(!SINGLETON_INIT, "Only one `EventLoop` is supported on iOS. \ assert!(
`EventLoopProxy` might be helpful"); !SINGLETON_INIT,
"Only one `EventLoop` is supported on iOS. \
`EventLoopProxy` might be helpful"
);
SINGLETON_INIT = true; SINGLETON_INIT = true;
view::create_delegate_class(); view::create_delegate_class();
} }
@ -92,25 +78,34 @@ impl<T: 'static> EventLoop<T> {
capabilities, capabilities,
}, },
_marker: PhantomData, _marker: PhantomData,
} },
} }
} }
pub fn run<F>(self, event_handler: F) -> ! pub fn run<F>(self, event_handler: F) -> !
where where
F: 'static + FnMut(Event<T>, &RootEventLoopWindowTarget<T>, &mut ControlFlow) F: 'static + FnMut(Event<T>, &RootEventLoopWindowTarget<T>, &mut ControlFlow),
{ {
unsafe { unsafe {
let application: *mut c_void = msg_send![class!(UIApplication), sharedApplication]; let application: *mut c_void = msg_send![class!(UIApplication), sharedApplication];
assert_eq!(application, ptr::null_mut(), "\ assert_eq!(
`EventLoop` cannot be `run` after a call to `UIApplicationMain` on iOS\n\ application,
Note: `EventLoop::run` calls `UIApplicationMain` on iOS"); ptr::null_mut(),
"\
`EventLoop` cannot be `run` after a call to `UIApplicationMain` on iOS\n\
Note: `EventLoop::run` calls `UIApplicationMain` on iOS"
);
AppState::will_launch(Box::new(EventLoopHandler { AppState::will_launch(Box::new(EventLoopHandler {
f: event_handler, f: event_handler,
event_loop: self.window_target, event_loop: self.window_target,
})); }));
UIApplicationMain(0, ptr::null(), nil, NSString::alloc(nil).init_str("AppDelegate")); UIApplicationMain(
0,
ptr::null(),
nil,
NSString::alloc(nil).init_str("AppDelegate"),
);
unreachable!() unreachable!()
} }
} }
@ -121,16 +116,12 @@ impl<T: 'static> EventLoop<T> {
pub fn available_monitors(&self) -> VecDeque<MonitorHandle> { pub fn available_monitors(&self) -> VecDeque<MonitorHandle> {
// guaranteed to be on main thread // guaranteed to be on main thread
unsafe { unsafe { monitor::uiscreens() }
monitor::uiscreens()
}
} }
pub fn primary_monitor(&self) -> MonitorHandle { pub fn primary_monitor(&self) -> MonitorHandle {
// guaranteed to be on main thread // guaranteed to be on main thread
unsafe { unsafe { monitor::main_uiscreen() }
monitor::main_uiscreen()
}
} }
pub fn window_target(&self) -> &RootEventLoopWindowTarget<T> { pub fn window_target(&self) -> &RootEventLoopWindowTarget<T> {
@ -142,9 +133,7 @@ impl<T: 'static> EventLoop<T> {
impl<T: 'static> EventLoop<T> { impl<T: 'static> EventLoop<T> {
pub fn idiom(&self) -> Idiom { pub fn idiom(&self) -> Idiom {
// guaranteed to be on main thread // guaranteed to be on main thread
unsafe { unsafe { self::get_idiom() }
self::get_idiom()
}
} }
} }
@ -183,18 +172,12 @@ impl<T> EventLoopProxy<T> {
// we want all the members of context to be zero/null, except one // we want all the members of context to be zero/null, except one
let mut context: CFRunLoopSourceContext = mem::zeroed(); let mut context: CFRunLoopSourceContext = mem::zeroed();
context.perform = event_loop_proxy_handler; context.perform = event_loop_proxy_handler;
let source = CFRunLoopSourceCreate( let source =
ptr::null_mut(), CFRunLoopSourceCreate(ptr::null_mut(), CFIndex::max_value() - 1, &mut context);
CFIndex::max_value() - 1,
&mut context,
);
CFRunLoopAddSource(rl, source, kCFRunLoopCommonModes); CFRunLoopAddSource(rl, source, kCFRunLoopCommonModes);
CFRunLoopWakeUp(rl); CFRunLoopWakeUp(rl);
EventLoopProxy { EventLoopProxy { sender, source }
sender,
source,
}
} }
} }
@ -213,7 +196,7 @@ impl<T> EventLoopProxy<T> {
fn setup_control_flow_observers() { fn setup_control_flow_observers() {
unsafe { unsafe {
// begin is queued with the highest priority to ensure it is processed before other observers // begin is queued with the highest priority to ensure it is processed before other observers
extern fn control_flow_begin_handler( extern "C" fn control_flow_begin_handler(
_: CFRunLoopObserverRef, _: CFRunLoopObserverRef,
activity: CFRunLoopActivity, activity: CFRunLoopActivity,
_: *mut c_void, _: *mut c_void,
@ -230,7 +213,7 @@ fn setup_control_flow_observers() {
// end is queued with the lowest priority to ensure it is processed after other observers // end is queued with the lowest priority to ensure it is processed after other observers
// without that, LoopDestroyed will get sent after EventsCleared // without that, LoopDestroyed will get sent after EventsCleared
extern fn control_flow_end_handler( extern "C" fn control_flow_end_handler(
_: CFRunLoopObserverRef, _: CFRunLoopObserverRef,
activity: CFRunLoopActivity, activity: CFRunLoopActivity,
_: *mut c_void, _: *mut c_void,
@ -282,7 +265,8 @@ struct EventLoopHandler<F, T: 'static> {
impl<F, T: 'static> Debug for EventLoopHandler<F, T> { impl<F, T: 'static> Debug for EventLoopHandler<F, T> {
fn fmt(&self, formatter: &mut fmt::Formatter<'_>) -> fmt::Result { fn fmt(&self, formatter: &mut fmt::Formatter<'_>) -> fmt::Result {
formatter.debug_struct("EventLoopHandler") formatter
.debug_struct("EventLoopHandler")
.field("event_loop", &self.event_loop) .field("event_loop", &self.event_loop)
.finish() .finish()
} }
@ -303,11 +287,7 @@ where
fn handle_user_events(&mut self, control_flow: &mut ControlFlow) { fn handle_user_events(&mut self, control_flow: &mut ControlFlow) {
for event in self.event_loop.p.receiver.try_iter() { for event in self.event_loop.p.receiver.try_iter() {
(self.f)( (self.f)(Event::UserEvent(event), &self.event_loop, control_flow);
Event::UserEvent(event),
&self.event_loop,
control_flow,
);
} }
} }
} }
@ -325,7 +305,10 @@ pub struct Capabilities {
impl From<NSOperatingSystemVersion> for Capabilities { impl From<NSOperatingSystemVersion> for Capabilities {
fn from(os_version: NSOperatingSystemVersion) -> Capabilities { fn from(os_version: NSOperatingSystemVersion) -> Capabilities {
assert!(os_version.major >= 8, "`winit` current requires iOS version 8 or greater"); assert!(
os_version.major >= 8,
"`winit` current requires iOS version 8 or greater"
);
let supports_safe_area = os_version.major >= 11; let supports_safe_area = os_version.major >= 11;

View file

@ -1,11 +1,8 @@
#![allow(non_camel_case_types, non_snake_case, non_upper_case_globals)] #![allow(non_camel_case_types, non_snake_case, non_upper_case_globals)]
use std::ffi::CString; use std::{ffi::CString, ops::BitOr, os::raw::*};
use std::ops::BitOr;
use std::os::raw::*;
use objc::{Encode, Encoding}; use objc::{runtime::Object, Encode, Encoding};
use objc::runtime::Object;
use crate::platform::ios::{Idiom, ValidOrientations}; use crate::platform::ios::{Idiom, ValidOrientations};
@ -87,7 +84,9 @@ pub struct UIEdgeInsets {
pub struct UIUserInterfaceIdiom(NSInteger); pub struct UIUserInterfaceIdiom(NSInteger);
unsafe impl Encode for UIUserInterfaceIdiom { unsafe impl Encode for UIUserInterfaceIdiom {
fn encode() -> Encoding { NSInteger::encode() } fn encode() -> Encoding {
NSInteger::encode()
}
} }
impl UIUserInterfaceIdiom { impl UIUserInterfaceIdiom {
@ -128,7 +127,9 @@ impl Into<Idiom> for UIUserInterfaceIdiom {
pub struct UIInterfaceOrientationMask(NSUInteger); pub struct UIInterfaceOrientationMask(NSUInteger);
unsafe impl Encode for UIInterfaceOrientationMask { unsafe impl Encode for UIInterfaceOrientationMask {
fn encode() -> Encoding { NSUInteger::encode() } fn encode() -> Encoding {
NSUInteger::encode()
}
} }
impl UIInterfaceOrientationMask { impl UIInterfaceOrientationMask {
@ -136,9 +137,12 @@ impl UIInterfaceOrientationMask {
pub const PortraitUpsideDown: UIInterfaceOrientationMask = UIInterfaceOrientationMask(1 << 2); pub const PortraitUpsideDown: UIInterfaceOrientationMask = UIInterfaceOrientationMask(1 << 2);
pub const LandscapeLeft: UIInterfaceOrientationMask = UIInterfaceOrientationMask(1 << 4); pub const LandscapeLeft: UIInterfaceOrientationMask = UIInterfaceOrientationMask(1 << 4);
pub const LandscapeRight: UIInterfaceOrientationMask = UIInterfaceOrientationMask(1 << 3); pub const LandscapeRight: UIInterfaceOrientationMask = UIInterfaceOrientationMask(1 << 3);
pub const Landscape: UIInterfaceOrientationMask = UIInterfaceOrientationMask(Self::LandscapeLeft.0 | Self::LandscapeRight.0); pub const Landscape: UIInterfaceOrientationMask =
pub const AllButUpsideDown: UIInterfaceOrientationMask = UIInterfaceOrientationMask(Self::Landscape.0 | Self::Portrait.0); UIInterfaceOrientationMask(Self::LandscapeLeft.0 | Self::LandscapeRight.0);
pub const All: UIInterfaceOrientationMask = UIInterfaceOrientationMask(Self::AllButUpsideDown.0 | Self::PortraitUpsideDown.0); pub const AllButUpsideDown: UIInterfaceOrientationMask =
UIInterfaceOrientationMask(Self::Landscape.0 | Self::Portrait.0);
pub const All: UIInterfaceOrientationMask =
UIInterfaceOrientationMask(Self::AllButUpsideDown.0 | Self::PortraitUpsideDown.0);
} }
impl BitOr for UIInterfaceOrientationMask { impl BitOr for UIInterfaceOrientationMask {
@ -155,18 +159,23 @@ impl UIInterfaceOrientationMask {
idiom: Idiom, idiom: Idiom,
) -> UIInterfaceOrientationMask { ) -> UIInterfaceOrientationMask {
match (valid_orientations, idiom) { match (valid_orientations, idiom) {
(ValidOrientations::LandscapeAndPortrait, Idiom::Phone) => UIInterfaceOrientationMask::AllButUpsideDown, (ValidOrientations::LandscapeAndPortrait, Idiom::Phone) => {
UIInterfaceOrientationMask::AllButUpsideDown
},
(ValidOrientations::LandscapeAndPortrait, _) => UIInterfaceOrientationMask::All, (ValidOrientations::LandscapeAndPortrait, _) => UIInterfaceOrientationMask::All,
(ValidOrientations::Landscape, _) => UIInterfaceOrientationMask::Landscape, (ValidOrientations::Landscape, _) => UIInterfaceOrientationMask::Landscape,
(ValidOrientations::Portrait, Idiom::Phone) => UIInterfaceOrientationMask::Portrait, (ValidOrientations::Portrait, Idiom::Phone) => UIInterfaceOrientationMask::Portrait,
(ValidOrientations::Portrait, _) => UIInterfaceOrientationMask::Portrait | UIInterfaceOrientationMask::PortraitUpsideDown, (ValidOrientations::Portrait, _) => {
UIInterfaceOrientationMask::Portrait
| UIInterfaceOrientationMask::PortraitUpsideDown
},
} }
} }
} }
#[link(name = "UIKit", kind = "framework")] #[link(name = "UIKit", kind = "framework")]
#[link(name = "CoreFoundation", kind = "framework")] #[link(name = "CoreFoundation", kind = "framework")]
extern { extern "C" {
pub static kCFRunLoopDefaultMode: CFRunLoopMode; pub static kCFRunLoopDefaultMode: CFRunLoopMode;
pub static kCFRunLoopCommonModes: CFRunLoopMode; pub static kCFRunLoopCommonModes: CFRunLoopMode;
@ -203,15 +212,8 @@ extern {
callout: CFRunLoopTimerCallBack, callout: CFRunLoopTimerCallBack,
context: *mut CFRunLoopTimerContext, context: *mut CFRunLoopTimerContext,
) -> CFRunLoopTimerRef; ) -> CFRunLoopTimerRef;
pub fn CFRunLoopAddTimer( pub fn CFRunLoopAddTimer(rl: CFRunLoopRef, timer: CFRunLoopTimerRef, mode: CFRunLoopMode);
rl: CFRunLoopRef, pub fn CFRunLoopTimerSetNextFireDate(timer: CFRunLoopTimerRef, fireDate: CFAbsoluteTime);
timer: CFRunLoopTimerRef,
mode: CFRunLoopMode,
);
pub fn CFRunLoopTimerSetNextFireDate(
timer: CFRunLoopTimerRef,
fireDate: CFAbsoluteTime,
);
pub fn CFRunLoopTimerInvalidate(time: CFRunLoopTimerRef); pub fn CFRunLoopTimerInvalidate(time: CFRunLoopTimerRef);
pub fn CFRunLoopSourceCreate( pub fn CFRunLoopSourceCreate(
@ -219,11 +221,7 @@ extern {
order: CFIndex, order: CFIndex,
context: *mut CFRunLoopSourceContext, context: *mut CFRunLoopSourceContext,
) -> CFRunLoopSourceRef; ) -> CFRunLoopSourceRef;
pub fn CFRunLoopAddSource( pub fn CFRunLoopAddSource(rl: CFRunLoopRef, source: CFRunLoopSourceRef, mode: CFRunLoopMode);
rl: CFRunLoopRef,
source: CFRunLoopSourceRef,
mode: CFRunLoopMode,
);
pub fn CFRunLoopSourceInvalidate(source: CFRunLoopSourceRef); pub fn CFRunLoopSourceInvalidate(source: CFRunLoopSourceRef);
pub fn CFRunLoopSourceSignal(source: CFRunLoopSourceRef); pub fn CFRunLoopSourceSignal(source: CFRunLoopSourceRef);
@ -259,15 +257,9 @@ pub const kCFRunLoopBeforeWaiting: CFRunLoopActivity = 1 << 5;
pub const kCFRunLoopAfterWaiting: CFRunLoopActivity = 1 << 6; pub const kCFRunLoopAfterWaiting: CFRunLoopActivity = 1 << 6;
pub const kCFRunLoopExit: CFRunLoopActivity = 1 << 7; pub const kCFRunLoopExit: CFRunLoopActivity = 1 << 7;
pub type CFRunLoopObserverCallBack = extern "C" fn( pub type CFRunLoopObserverCallBack =
observer: CFRunLoopObserverRef, extern "C" fn(observer: CFRunLoopObserverRef, activity: CFRunLoopActivity, info: *mut c_void);
activity: CFRunLoopActivity, pub type CFRunLoopTimerCallBack = extern "C" fn(timer: CFRunLoopTimerRef, info: *mut c_void);
info: *mut c_void,
);
pub type CFRunLoopTimerCallBack = extern "C" fn(
timer: CFRunLoopTimerRef,
info: *mut c_void,
);
pub enum CFRunLoopObserverContext {} pub enum CFRunLoopObserverContext {}
pub enum CFRunLoopTimerContext {} pub enum CFRunLoopTimerContext {}
@ -299,11 +291,11 @@ pub trait NSString: Sized {
impl NSString for id { impl NSString for id {
unsafe fn initWithUTF8String_(self, c_string: *const c_char) -> id { unsafe fn initWithUTF8String_(self, c_string: *const c_char) -> id {
msg_send![self, initWithUTF8String:c_string as id] msg_send![self, initWithUTF8String: c_string as id]
} }
unsafe fn stringByAppendingString_(self, other: id) -> id { unsafe fn stringByAppendingString_(self, other: id) -> id {
msg_send![self, stringByAppendingString:other] msg_send![self, stringByAppendingString: other]
} }
unsafe fn init_str(self, string: &str) -> id { unsafe fn init_str(self, string: &str) -> id {

View file

@ -26,7 +26,6 @@
//! fn start_inner() { //! fn start_inner() {
//! ... //! ...
//! } //! }
//!
//! ``` //! ```
//! //!
//! Compile project and then drag resulting .a into Xcode project. Add winit.h to xcode. //! Compile project and then drag resulting .a into Xcode project. Add winit.h to xcode.
@ -78,12 +77,10 @@ mod window;
use std::fmt; use std::fmt;
pub use self::event_loop::{EventLoop, EventLoopProxy, EventLoopWindowTarget}; pub use self::{
pub use self::monitor::MonitorHandle; event_loop::{EventLoop, EventLoopProxy, EventLoopWindowTarget},
pub use self::window::{ monitor::MonitorHandle,
PlatformSpecificWindowBuilderAttributes, window::{PlatformSpecificWindowBuilderAttributes, Window, WindowId},
Window,
WindowId,
}; };
#[derive(Debug, Copy, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)] #[derive(Debug, Copy, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
@ -108,7 +105,7 @@ pub enum OsError {}
impl fmt::Display for OsError { impl fmt::Display for OsError {
fn fmt(&self, _: &mut fmt::Formatter<'_>) -> fmt::Result { fn fmt(&self, _: &mut fmt::Formatter<'_>) -> fmt::Result {
match self { match self {
_ => unreachable!() _ => unreachable!(),
} }
} }
} }

View file

@ -4,10 +4,14 @@ use std::{
ops::{Deref, DerefMut}, ops::{Deref, DerefMut},
}; };
use crate::dpi::{PhysicalPosition, PhysicalSize}; use crate::{
use crate::monitor::VideoMode; dpi::{PhysicalPosition, PhysicalSize},
monitor::VideoMode,
};
use crate::platform_impl::platform::ffi::{id, nil, CGFloat, CGRect, CGSize, NSInteger, NSUInteger}; use crate::platform_impl::platform::ffi::{
id, nil, CGFloat, CGRect, CGSize, NSInteger, NSUInteger,
};
pub struct Inner { pub struct Inner {
uiscreen: id, uiscreen: id,
@ -30,7 +34,9 @@ impl Deref for MonitorHandle {
fn deref(&self) -> &Inner { fn deref(&self) -> &Inner {
unsafe { unsafe {
assert_main_thread!("`MonitorHandle` methods can only be run on the main thread on iOS"); assert_main_thread!(
"`MonitorHandle` methods can only be run on the main thread on iOS"
);
} }
&self.inner &self.inner
} }
@ -39,7 +45,9 @@ impl Deref for MonitorHandle {
impl DerefMut for MonitorHandle { impl DerefMut for MonitorHandle {
fn deref_mut(&mut self) -> &mut Inner { fn deref_mut(&mut self) -> &mut Inner {
unsafe { unsafe {
assert_main_thread!("`MonitorHandle` methods can only be run on the main thread on iOS"); assert_main_thread!(
"`MonitorHandle` methods can only be run on the main thread on iOS"
);
} }
&mut self.inner &mut self.inner
} }
@ -89,7 +97,9 @@ impl MonitorHandle {
assert_main_thread!("`MonitorHandle` can only be cloned on the main thread on iOS"); assert_main_thread!("`MonitorHandle` can only be cloned on the main thread on iOS");
let () = msg_send![uiscreen, retain]; let () = msg_send![uiscreen, retain];
} }
MonitorHandle { inner: Inner { uiscreen } } MonitorHandle {
inner: Inner { uiscreen },
}
} }
} }
@ -180,7 +190,7 @@ pub unsafe fn uiscreens() -> VecDeque<MonitorHandle> {
loop { loop {
let screen: id = msg_send![screens_enum, nextObject]; let screen: id = msg_send![screens_enum, nextObject];
if screen == nil { if screen == nil {
break result break result;
} }
result.push_back(MonitorHandle::retained_new(screen)); result.push_back(MonitorHandle::retained_new(screen));
} }

View file

@ -1,31 +1,23 @@
use std::collections::HashMap; use std::collections::HashMap;
use objc::declare::ClassDecl; use objc::{
use objc::runtime::{BOOL, Class, NO, Object, Sel, YES}; declare::ClassDecl,
runtime::{Class, Object, Sel, BOOL, NO, YES},
use crate::event::{
DeviceId as RootDeviceId,
Event,
Touch,
TouchPhase,
WindowEvent
}; };
use crate::platform::ios::MonitorHandleExtIOS;
use crate::window::{WindowAttributes, WindowId as RootWindowId};
use crate::platform_impl::platform::app_state::AppState; use crate::{
use crate::platform_impl::platform::DeviceId; event::{DeviceId as RootDeviceId, Event, Touch, TouchPhase, WindowEvent},
use crate::platform_impl::platform::event_loop; platform::ios::MonitorHandleExtIOS,
use crate::platform_impl::platform::ffi::{ window::{WindowAttributes, WindowId as RootWindowId},
id, };
nil,
CGFloat, use crate::platform_impl::platform::{
CGPoint, app_state::AppState,
CGRect, event_loop,
UIInterfaceOrientationMask, ffi::{id, nil, CGFloat, CGPoint, CGRect, UIInterfaceOrientationMask, UITouchPhase},
UITouchPhase, window::PlatformSpecificWindowBuilderAttributes,
DeviceId,
}; };
use crate::platform_impl::platform::window::{PlatformSpecificWindowBuilderAttributes};
// requires main thread // requires main thread
unsafe fn get_view_class(root_view_class: &'static Class) -> &'static Class { unsafe fn get_view_class(root_view_class: &'static Class) -> &'static Class {
@ -40,10 +32,13 @@ unsafe fn get_view_class(root_view_class: &'static Class) -> &'static Class {
classes.entry(root_view_class).or_insert_with(move || { classes.entry(root_view_class).or_insert_with(move || {
let uiview_class = class!(UIView); let uiview_class = class!(UIView);
let is_uiview: BOOL = msg_send![root_view_class, isSubclassOfClass:uiview_class]; let is_uiview: BOOL = msg_send![root_view_class, isSubclassOfClass: uiview_class];
assert_eq!(is_uiview, YES, "`root_view_class` must inherit from `UIView`"); assert_eq!(
is_uiview, YES,
"`root_view_class` must inherit from `UIView`"
);
extern fn draw_rect(object: &Object, _: Sel, rect: CGRect) { extern "C" fn draw_rect(object: &Object, _: Sel, rect: CGRect) {
unsafe { unsafe {
let window: id = msg_send![object, window]; let window: id = msg_send![object, window];
AppState::handle_nonuser_event(Event::WindowEvent { AppState::handle_nonuser_event(Event::WindowEvent {
@ -55,13 +50,14 @@ unsafe fn get_view_class(root_view_class: &'static Class) -> &'static Class {
} }
} }
extern fn layout_subviews(object: &Object, _: Sel) { extern "C" fn layout_subviews(object: &Object, _: Sel) {
unsafe { unsafe {
let window: id = msg_send![object, window]; let window: id = msg_send![object, window];
let bounds: CGRect = msg_send![window, bounds]; let bounds: CGRect = msg_send![window, bounds];
let screen: id = msg_send![window, screen]; let screen: id = msg_send![window, screen];
let screen_space: id = msg_send![screen, coordinateSpace]; let screen_space: id = msg_send![screen, coordinateSpace];
let screen_frame: CGRect = msg_send![object, convertRect:bounds toCoordinateSpace:screen_space]; let screen_frame: CGRect =
msg_send![object, convertRect:bounds toCoordinateSpace:screen_space];
let size = crate::dpi::LogicalSize { let size = crate::dpi::LogicalSize {
width: screen_frame.size.width, width: screen_frame.size.width,
height: screen_frame.size.height, height: screen_frame.size.height,
@ -78,10 +74,14 @@ unsafe fn get_view_class(root_view_class: &'static Class) -> &'static Class {
let mut decl = ClassDecl::new(&format!("WinitUIView{}", ID), root_view_class) let mut decl = ClassDecl::new(&format!("WinitUIView{}", ID), root_view_class)
.expect("Failed to declare class `WinitUIView`"); .expect("Failed to declare class `WinitUIView`");
ID += 1; ID += 1;
decl.add_method(sel!(drawRect:), decl.add_method(
draw_rect as extern fn(&Object, Sel, CGRect)); sel!(drawRect:),
decl.add_method(sel!(layoutSubviews), draw_rect as extern "C" fn(&Object, Sel, CGRect),
layout_subviews as extern fn(&Object, Sel)); );
decl.add_method(
sel!(layoutSubviews),
layout_subviews as extern "C" fn(&Object, Sel),
);
decl.register() decl.register()
}) })
} }
@ -92,33 +92,39 @@ unsafe fn get_view_controller_class() -> &'static Class {
if CLASS.is_none() { if CLASS.is_none() {
let uiviewcontroller_class = class!(UIViewController); let uiviewcontroller_class = class!(UIViewController);
extern fn set_prefers_status_bar_hidden(object: &mut Object, _: Sel, hidden: BOOL) { extern "C" fn set_prefers_status_bar_hidden(object: &mut Object, _: Sel, hidden: BOOL) {
unsafe { unsafe {
object.set_ivar::<BOOL>("_prefers_status_bar_hidden", hidden); object.set_ivar::<BOOL>("_prefers_status_bar_hidden", hidden);
let () = msg_send![object, setNeedsStatusBarAppearanceUpdate]; let () = msg_send![object, setNeedsStatusBarAppearanceUpdate];
} }
} }
extern fn prefers_status_bar_hidden(object: &Object, _: Sel) -> BOOL { extern "C" fn prefers_status_bar_hidden(object: &Object, _: Sel) -> BOOL {
unsafe { unsafe { *object.get_ivar::<BOOL>("_prefers_status_bar_hidden") }
*object.get_ivar::<BOOL>("_prefers_status_bar_hidden")
}
} }
extern fn set_supported_orientations(object: &mut Object, _: Sel, orientations: UIInterfaceOrientationMask) { extern "C" fn set_supported_orientations(
object: &mut Object,
_: Sel,
orientations: UIInterfaceOrientationMask,
) {
unsafe { unsafe {
object.set_ivar::<UIInterfaceOrientationMask>("_supported_orientations", orientations); object.set_ivar::<UIInterfaceOrientationMask>(
"_supported_orientations",
orientations,
);
let () = msg_send![class!(UIViewController), attemptRotationToDeviceOrientation]; let () = msg_send![class!(UIViewController), attemptRotationToDeviceOrientation];
} }
} }
extern fn supported_orientations(object: &Object, _: Sel) -> UIInterfaceOrientationMask { extern "C" fn supported_orientations(
unsafe { object: &Object,
*object.get_ivar::<UIInterfaceOrientationMask>("_supported_orientations") _: Sel,
} ) -> UIInterfaceOrientationMask {
unsafe { *object.get_ivar::<UIInterfaceOrientationMask>("_supported_orientations") }
} }
extern fn should_autorotate(_: &Object, _: Sel) -> BOOL { extern "C" fn should_autorotate(_: &Object, _: Sel) -> BOOL {
YES YES
} }
@ -126,16 +132,27 @@ unsafe fn get_view_controller_class() -> &'static Class {
.expect("Failed to declare class `WinitUIViewController`"); .expect("Failed to declare class `WinitUIViewController`");
decl.add_ivar::<BOOL>("_prefers_status_bar_hidden"); decl.add_ivar::<BOOL>("_prefers_status_bar_hidden");
decl.add_ivar::<UIInterfaceOrientationMask>("_supported_orientations"); decl.add_ivar::<UIInterfaceOrientationMask>("_supported_orientations");
decl.add_method(sel!(setPrefersStatusBarHidden:), decl.add_method(
set_prefers_status_bar_hidden as extern fn(&mut Object, Sel, BOOL)); sel!(setPrefersStatusBarHidden:),
decl.add_method(sel!(prefersStatusBarHidden), set_prefers_status_bar_hidden as extern "C" fn(&mut Object, Sel, BOOL),
prefers_status_bar_hidden as extern fn(&Object, Sel) -> BOOL); );
decl.add_method(sel!(setSupportedInterfaceOrientations:), decl.add_method(
set_supported_orientations as extern fn(&mut Object, Sel, UIInterfaceOrientationMask)); sel!(prefersStatusBarHidden),
decl.add_method(sel!(supportedInterfaceOrientations), prefers_status_bar_hidden as extern "C" fn(&Object, Sel) -> BOOL,
supported_orientations as extern fn(&Object, Sel) -> UIInterfaceOrientationMask); );
decl.add_method(sel!(shouldAutorotate), decl.add_method(
should_autorotate as extern fn(&Object, Sel) -> BOOL); sel!(setSupportedInterfaceOrientations:),
set_supported_orientations
as extern "C" fn(&mut Object, Sel, UIInterfaceOrientationMask),
);
decl.add_method(
sel!(supportedInterfaceOrientations),
supported_orientations as extern "C" fn(&Object, Sel) -> UIInterfaceOrientationMask,
);
decl.add_method(
sel!(shouldAutorotate),
should_autorotate as extern "C" fn(&Object, Sel) -> BOOL,
);
CLASS = Some(decl.register()); CLASS = Some(decl.register());
} }
CLASS.unwrap() CLASS.unwrap()
@ -147,7 +164,7 @@ unsafe fn get_window_class() -> &'static Class {
if CLASS.is_none() { if CLASS.is_none() {
let uiwindow_class = class!(UIWindow); let uiwindow_class = class!(UIWindow);
extern fn become_key_window(object: &Object, _: Sel) { extern "C" fn become_key_window(object: &Object, _: Sel) {
unsafe { unsafe {
AppState::handle_nonuser_event(Event::WindowEvent { AppState::handle_nonuser_event(Event::WindowEvent {
window_id: RootWindowId(object.into()), window_id: RootWindowId(object.into()),
@ -157,7 +174,7 @@ unsafe fn get_window_class() -> &'static Class {
} }
} }
extern fn resign_key_window(object: &Object, _: Sel) { extern "C" fn resign_key_window(object: &Object, _: Sel) {
unsafe { unsafe {
AppState::handle_nonuser_event(Event::WindowEvent { AppState::handle_nonuser_event(Event::WindowEvent {
window_id: RootWindowId(object.into()), window_id: RootWindowId(object.into()),
@ -167,7 +184,7 @@ unsafe fn get_window_class() -> &'static Class {
} }
} }
extern fn handle_touches(object: &Object, _: Sel, touches: id, _:id) { extern "C" fn handle_touches(object: &Object, _: Sel, touches: id, _: id) {
unsafe { unsafe {
let uiscreen = msg_send![object, screen]; let uiscreen = msg_send![object, screen];
let touches_enum: id = msg_send![touches, objectEnumerator]; let touches_enum: id = msg_send![touches, objectEnumerator];
@ -175,9 +192,9 @@ unsafe fn get_window_class() -> &'static Class {
loop { loop {
let touch: id = msg_send![touches_enum, nextObject]; let touch: id = msg_send![touches_enum, nextObject];
if touch == nil { if touch == nil {
break break;
} }
let location: CGPoint = msg_send![touch, locationInView:nil]; let location: CGPoint = msg_send![touch, locationInView: nil];
let touch_id = touch as u64; let touch_id = touch as u64;
let phase: UITouchPhase = msg_send![touch, phase]; let phase: UITouchPhase = msg_send![touch, phase];
let phase = match phase { let phase = match phase {
@ -203,16 +220,20 @@ unsafe fn get_window_class() -> &'static Class {
} }
} }
extern fn set_content_scale_factor(object: &mut Object, _: Sel, hidpi_factor: CGFloat) { extern "C" fn set_content_scale_factor(object: &mut Object, _: Sel, hidpi_factor: CGFloat) {
unsafe { unsafe {
let () = msg_send![super(object, class!(UIWindow)), setContentScaleFactor:hidpi_factor]; let () = msg_send![
super(object, class!(UIWindow)),
setContentScaleFactor: hidpi_factor
];
let view_controller: id = msg_send![object, rootViewController]; let view_controller: id = msg_send![object, rootViewController];
let view: id = msg_send![view_controller, view]; let view: id = msg_send![view_controller, view];
let () = msg_send![view, setContentScaleFactor:hidpi_factor]; let () = msg_send![view, setContentScaleFactor: hidpi_factor];
let bounds: CGRect = msg_send![object, bounds]; let bounds: CGRect = msg_send![object, bounds];
let screen: id = msg_send![object, screen]; let screen: id = msg_send![object, screen];
let screen_space: id = msg_send![screen, coordinateSpace]; let screen_space: id = msg_send![screen, coordinateSpace];
let screen_frame: CGRect = msg_send![object, convertRect:bounds toCoordinateSpace:screen_space]; let screen_frame: CGRect =
msg_send![object, convertRect:bounds toCoordinateSpace:screen_space];
let size = crate::dpi::LogicalSize { let size = crate::dpi::LogicalSize {
width: screen_frame.size.width, width: screen_frame.size.width,
height: screen_frame.size.height, height: screen_frame.size.height,
@ -221,32 +242,47 @@ unsafe fn get_window_class() -> &'static Class {
std::iter::once(Event::WindowEvent { std::iter::once(Event::WindowEvent {
window_id: RootWindowId(object.into()), window_id: RootWindowId(object.into()),
event: WindowEvent::HiDpiFactorChanged(hidpi_factor as _), event: WindowEvent::HiDpiFactorChanged(hidpi_factor as _),
}).chain(std::iter::once(Event::WindowEvent { })
.chain(std::iter::once(Event::WindowEvent {
window_id: RootWindowId(object.into()), window_id: RootWindowId(object.into()),
event: WindowEvent::Resized(size), event: WindowEvent::Resized(size),
})) })),
); );
} }
} }
let mut decl = ClassDecl::new("WinitUIWindow", uiwindow_class) let mut decl = ClassDecl::new("WinitUIWindow", uiwindow_class)
.expect("Failed to declare class `WinitUIWindow`"); .expect("Failed to declare class `WinitUIWindow`");
decl.add_method(sel!(becomeKeyWindow), decl.add_method(
become_key_window as extern fn(&Object, Sel)); sel!(becomeKeyWindow),
decl.add_method(sel!(resignKeyWindow), become_key_window as extern "C" fn(&Object, Sel),
resign_key_window as extern fn(&Object, Sel)); );
decl.add_method(
sel!(resignKeyWindow),
resign_key_window as extern "C" fn(&Object, Sel),
);
decl.add_method(sel!(touchesBegan:withEvent:), decl.add_method(
handle_touches as extern fn(this: &Object, _: Sel, _: id, _:id)); sel!(touchesBegan:withEvent:),
decl.add_method(sel!(touchesMoved:withEvent:), handle_touches as extern "C" fn(this: &Object, _: Sel, _: id, _: id),
handle_touches as extern fn(this: &Object, _: Sel, _: id, _:id)); );
decl.add_method(sel!(touchesEnded:withEvent:), decl.add_method(
handle_touches as extern fn(this: &Object, _: Sel, _: id, _:id)); sel!(touchesMoved:withEvent:),
decl.add_method(sel!(touchesCancelled:withEvent:), handle_touches as extern "C" fn(this: &Object, _: Sel, _: id, _: id),
handle_touches as extern fn(this: &Object, _: Sel, _: id, _:id)); );
decl.add_method(
sel!(touchesEnded:withEvent:),
handle_touches as extern "C" fn(this: &Object, _: Sel, _: id, _: id),
);
decl.add_method(
sel!(touchesCancelled:withEvent:),
handle_touches as extern "C" fn(this: &Object, _: Sel, _: id, _: id),
);
decl.add_method(sel!(setContentScaleFactor:), decl.add_method(
set_content_scale_factor as extern fn(&mut Object, Sel, CGFloat)); sel!(setContentScaleFactor:),
set_content_scale_factor as extern "C" fn(&mut Object, Sel, CGFloat),
);
CLASS = Some(decl.register()); CLASS = Some(decl.register());
} }
@ -263,9 +299,9 @@ pub unsafe fn create_view(
let view: id = msg_send![class, alloc]; let view: id = msg_send![class, alloc];
assert!(!view.is_null(), "Failed to create `UIView` instance"); assert!(!view.is_null(), "Failed to create `UIView` instance");
let view: id = msg_send![view, initWithFrame:frame]; let view: id = msg_send![view, initWithFrame: frame];
assert!(!view.is_null(), "Failed to initialize `UIView` instance"); assert!(!view.is_null(), "Failed to initialize `UIView` instance");
let () = msg_send![view, setMultipleTouchEnabled:YES]; let () = msg_send![view, setMultipleTouchEnabled: YES];
view view
} }
@ -279,9 +315,15 @@ pub unsafe fn create_view_controller(
let class = get_view_controller_class(); let class = get_view_controller_class();
let view_controller: id = msg_send![class, alloc]; let view_controller: id = msg_send![class, alloc];
assert!(!view_controller.is_null(), "Failed to create `UIViewController` instance"); assert!(
!view_controller.is_null(),
"Failed to create `UIViewController` instance"
);
let view_controller: id = msg_send![view_controller, init]; let view_controller: id = msg_send![view_controller, init];
assert!(!view_controller.is_null(), "Failed to initialize `UIViewController` instance"); assert!(
!view_controller.is_null(),
"Failed to initialize `UIViewController` instance"
);
let status_bar_hidden = if window_attributes.decorations { let status_bar_hidden = if window_attributes.decorations {
NO NO
} else { } else {
@ -292,9 +334,15 @@ pub unsafe fn create_view_controller(
platform_attributes.valid_orientations, platform_attributes.valid_orientations,
idiom, idiom,
); );
let () = msg_send![view_controller, setPrefersStatusBarHidden:status_bar_hidden]; let () = msg_send![
let () = msg_send![view_controller, setSupportedInterfaceOrientations:supported_orientations]; view_controller,
let () = msg_send![view_controller, setView:view]; setPrefersStatusBarHidden: status_bar_hidden
];
let () = msg_send![
view_controller,
setSupportedInterfaceOrientations: supported_orientations
];
let () = msg_send![view_controller, setView: view];
view_controller view_controller
} }
@ -309,11 +357,14 @@ pub unsafe fn create_window(
let window: id = msg_send![class, alloc]; let window: id = msg_send![class, alloc];
assert!(!window.is_null(), "Failed to create `UIWindow` instance"); assert!(!window.is_null(), "Failed to create `UIWindow` instance");
let window: id = msg_send![window, initWithFrame:frame]; let window: id = msg_send![window, initWithFrame: frame];
assert!(!window.is_null(), "Failed to initialize `UIWindow` instance"); assert!(
let () = msg_send![window, setRootViewController:view_controller]; !window.is_null(),
"Failed to initialize `UIWindow` instance"
);
let () = msg_send![window, setRootViewController: view_controller];
if let Some(hidpi_factor) = platform_attributes.hidpi_factor { if let Some(hidpi_factor) = platform_attributes.hidpi_factor {
let () = msg_send![window, setContentScaleFactor:hidpi_factor as CGFloat]; let () = msg_send![window, setContentScaleFactor: hidpi_factor as CGFloat];
} }
if let &Some(ref monitor) = &window_attributes.fullscreen { if let &Some(ref monitor) = &window_attributes.fullscreen {
let () = msg_send![window, setScreen:monitor.ui_screen()]; let () = msg_send![window, setScreen:monitor.ui_screen()];
@ -323,29 +374,25 @@ pub unsafe fn create_window(
} }
pub fn create_delegate_class() { pub fn create_delegate_class() {
extern fn did_finish_launching(_: &mut Object, _: Sel, _: id, _: id) -> BOOL { extern "C" fn did_finish_launching(_: &mut Object, _: Sel, _: id, _: id) -> BOOL {
unsafe { unsafe {
AppState::did_finish_launching(); AppState::did_finish_launching();
} }
YES YES
} }
extern fn did_become_active(_: &Object, _: Sel, _: id) { extern "C" fn did_become_active(_: &Object, _: Sel, _: id) {
unsafe { unsafe { AppState::handle_nonuser_event(Event::Suspended(false)) }
AppState::handle_nonuser_event(Event::Suspended(false))
}
} }
extern fn will_resign_active(_: &Object, _: Sel, _: id) { extern "C" fn will_resign_active(_: &Object, _: Sel, _: id) {
unsafe { unsafe { AppState::handle_nonuser_event(Event::Suspended(true)) }
AppState::handle_nonuser_event(Event::Suspended(true))
}
} }
extern fn will_enter_foreground(_: &Object, _: Sel, _: id) {} extern "C" fn will_enter_foreground(_: &Object, _: Sel, _: id) {}
extern fn did_enter_background(_: &Object, _: Sel, _: id) {} extern "C" fn did_enter_background(_: &Object, _: Sel, _: id) {}
extern fn will_terminate(_: &Object, _: Sel, _: id) { extern "C" fn will_terminate(_: &Object, _: Sel, _: id) {
unsafe { unsafe {
let app: id = msg_send![class!(UIApplication), sharedApplication]; let app: id = msg_send![class!(UIApplication), sharedApplication];
let windows: id = msg_send![app, windows]; let windows: id = msg_send![app, windows];
@ -354,9 +401,9 @@ pub fn create_delegate_class() {
loop { loop {
let window: id = msg_send![windows_enum, nextObject]; let window: id = msg_send![windows_enum, nextObject];
if window == nil { if window == nil {
break break;
} }
let is_winit_window: BOOL = msg_send![window, isKindOfClass:class!(WinitUIWindow)]; let is_winit_window: BOOL = msg_send![window, isKindOfClass: class!(WinitUIWindow)];
if is_winit_window == YES { if is_winit_window == YES {
events.push(Event::WindowEvent { events.push(Event::WindowEvent {
window_id: RootWindowId(window.into()), window_id: RootWindowId(window.into()),
@ -370,23 +417,36 @@ pub fn create_delegate_class() {
} }
let ui_responder = class!(UIResponder); let ui_responder = class!(UIResponder);
let mut decl = ClassDecl::new("AppDelegate", ui_responder).expect("Failed to declare class `AppDelegate`"); let mut decl =
ClassDecl::new("AppDelegate", ui_responder).expect("Failed to declare class `AppDelegate`");
unsafe { unsafe {
decl.add_method(sel!(application:didFinishLaunchingWithOptions:), decl.add_method(
did_finish_launching as extern fn(&mut Object, Sel, id, id) -> BOOL); sel!(application:didFinishLaunchingWithOptions:),
did_finish_launching as extern "C" fn(&mut Object, Sel, id, id) -> BOOL,
);
decl.add_method(sel!(applicationDidBecomeActive:), decl.add_method(
did_become_active as extern fn(&Object, Sel, id)); sel!(applicationDidBecomeActive:),
decl.add_method(sel!(applicationWillResignActive:), did_become_active as extern "C" fn(&Object, Sel, id),
will_resign_active as extern fn(&Object, Sel, id)); );
decl.add_method(sel!(applicationWillEnterForeground:), decl.add_method(
will_enter_foreground as extern fn(&Object, Sel, id)); sel!(applicationWillResignActive:),
decl.add_method(sel!(applicationDidEnterBackground:), will_resign_active as extern "C" fn(&Object, Sel, id),
did_enter_background as extern fn(&Object, Sel, id)); );
decl.add_method(
sel!(applicationWillEnterForeground:),
will_enter_foreground as extern "C" fn(&Object, Sel, id),
);
decl.add_method(
sel!(applicationDidEnterBackground:),
did_enter_background as extern "C" fn(&Object, Sel, id),
);
decl.add_method(sel!(applicationWillTerminate:), decl.add_method(
will_terminate as extern fn(&Object, Sel, id)); sel!(applicationWillTerminate:),
will_terminate as extern "C" fn(&Object, Sel, id),
);
decl.register(); decl.register();
} }

View file

@ -3,35 +3,21 @@ use std::{
ops::{Deref, DerefMut}, ops::{Deref, DerefMut},
}; };
use objc::runtime::{Class, NO, Object, YES}; use objc::runtime::{Class, Object, NO, YES};
use crate::dpi::{self, LogicalPosition, LogicalSize}; use crate::{
use crate::error::{ExternalError, NotSupportedError, OsError as RootOsError}; dpi::{self, LogicalPosition, LogicalSize},
use crate::icon::Icon; error::{ExternalError, NotSupportedError, OsError as RootOsError},
use crate::monitor::MonitorHandle as RootMonitorHandle; icon::Icon,
use crate::platform::ios::{MonitorHandleExtIOS, ValidOrientations}; monitor::MonitorHandle as RootMonitorHandle,
use crate::window::{ platform::ios::{MonitorHandleExtIOS, ValidOrientations},
CursorIcon, platform_impl::platform::{
WindowAttributes,
};
use crate::platform_impl::{
platform::{
app_state::AppState, app_state::AppState,
event_loop, event_loop,
ffi::{ ffi::{id, CGFloat, CGPoint, CGRect, CGSize, UIEdgeInsets, UIInterfaceOrientationMask},
id, monitor, view, EventLoopWindowTarget, MonitorHandle,
CGFloat,
CGPoint,
CGRect,
CGSize,
UIEdgeInsets,
UIInterfaceOrientationMask,
},
monitor,
view,
EventLoopWindowTarget,
MonitorHandle
}, },
window::{CursorIcon, WindowAttributes},
}; };
pub struct Inner { pub struct Inner {
@ -59,10 +45,10 @@ impl Inner {
pub fn set_visible(&self, visible: bool) { pub fn set_visible(&self, visible: bool) {
match visible { match visible {
true => unsafe { true => unsafe {
let () = msg_send![self.window, setHidden:NO]; let () = msg_send![self.window, setHidden: NO];
}, },
false => unsafe { false => unsafe {
let () = msg_send![self.window, setHidden:YES]; let () = msg_send![self.window, setHidden: YES];
}, },
} }
} }
@ -104,7 +90,7 @@ impl Inner {
size: screen_frame.size, size: screen_frame.size,
}; };
let bounds = self.from_screen_space(new_screen_frame); let bounds = self.from_screen_space(new_screen_frame);
let () = msg_send![self.window, setBounds:bounds]; let () = msg_send![self.window, setBounds: bounds];
} }
} }
@ -181,10 +167,10 @@ impl Inner {
// this is pretty slow on iOS, so avoid doing it if we can // this is pretty slow on iOS, so avoid doing it if we can
if uiscreen != current { if uiscreen != current {
let () = msg_send![self.window, setScreen:uiscreen]; let () = msg_send![self.window, setScreen: uiscreen];
} }
let () = msg_send![self.window, setFrame:bounds]; let () = msg_send![self.window, setFrame: bounds];
} },
None => warn!("`Window::set_fullscreen(None)` ignored on iOS"), None => warn!("`Window::set_fullscreen(None)` ignored on iOS"),
} }
} }
@ -213,7 +199,10 @@ impl Inner {
pub fn set_decorations(&self, decorations: bool) { pub fn set_decorations(&self, decorations: bool) {
unsafe { unsafe {
let status_bar_hidden = if decorations { NO } else { YES }; let status_bar_hidden = if decorations { NO } else { YES };
let () = msg_send![self.view_controller, setPrefersStatusBarHidden:status_bar_hidden]; let () = msg_send![
self.view_controller,
setPrefersStatusBarHidden: status_bar_hidden
];
} }
} }
@ -232,20 +221,18 @@ impl Inner {
pub fn current_monitor(&self) -> RootMonitorHandle { pub fn current_monitor(&self) -> RootMonitorHandle {
unsafe { unsafe {
let uiscreen: id = msg_send![self.window, screen]; let uiscreen: id = msg_send![self.window, screen];
RootMonitorHandle { inner: MonitorHandle::retained_new(uiscreen) } RootMonitorHandle {
inner: MonitorHandle::retained_new(uiscreen),
}
} }
} }
pub fn available_monitors(&self) -> VecDeque<MonitorHandle> { pub fn available_monitors(&self) -> VecDeque<MonitorHandle> {
unsafe { unsafe { monitor::uiscreens() }
monitor::uiscreens()
}
} }
pub fn primary_monitor(&self) -> MonitorHandle { pub fn primary_monitor(&self) -> MonitorHandle {
unsafe { unsafe { monitor::main_uiscreen() }
monitor::main_uiscreen()
}
} }
pub fn id(&self) -> WindowId { pub fn id(&self) -> WindowId {
@ -306,23 +293,35 @@ impl Window {
// TODO: transparency, visible // TODO: transparency, visible
unsafe { unsafe {
let screen = window_attributes.fullscreen let screen = window_attributes
.fullscreen
.as_ref() .as_ref()
.map(|screen| screen.ui_screen() as _) .map(|screen| screen.ui_screen() as _)
.unwrap_or_else(|| monitor::main_uiscreen().ui_screen()); .unwrap_or_else(|| monitor::main_uiscreen().ui_screen());
let screen_bounds: CGRect = msg_send![screen, bounds]; let screen_bounds: CGRect = msg_send![screen, bounds];
let frame = match window_attributes.inner_size { let frame = match window_attributes.inner_size {
Some(dim) => CGRect { Some(dim) => {
origin: screen_bounds.origin, CGRect {
size: CGSize { width: dim.width, height: dim.height }, origin: screen_bounds.origin,
size: CGSize {
width: dim.width,
height: dim.height,
},
}
}, },
None => screen_bounds, None => screen_bounds,
}; };
let view = view::create_view(&window_attributes, &platform_attributes, frame.clone()); let view = view::create_view(&window_attributes, &platform_attributes, frame.clone());
let view_controller = view::create_view_controller(&window_attributes, &platform_attributes, view); let view_controller =
let window = view::create_window(&window_attributes, &platform_attributes, frame, view_controller); view::create_view_controller(&window_attributes, &platform_attributes, view);
let window = view::create_window(
&window_attributes,
&platform_attributes,
frame,
view_controller,
);
let supports_safe_area = event_loop.capabilities().supports_safe_area; let supports_safe_area = event_loop.capabilities().supports_safe_area;
@ -342,23 +341,38 @@ impl Window {
// WindowExtIOS // WindowExtIOS
impl Inner { impl Inner {
pub fn ui_window(&self) -> id { self.window } pub fn ui_window(&self) -> id {
pub fn ui_view_controller(&self) -> id { self.view_controller } self.window
pub fn ui_view(&self) -> id { self.view } }
pub fn ui_view_controller(&self) -> id {
self.view_controller
}
pub fn ui_view(&self) -> id {
self.view
}
pub fn set_hidpi_factor(&self, hidpi_factor: f64) { pub fn set_hidpi_factor(&self, hidpi_factor: f64) {
unsafe { unsafe {
assert!(dpi::validate_hidpi_factor(hidpi_factor), "`WindowExtIOS::set_hidpi_factor` received an invalid hidpi factor"); assert!(
dpi::validate_hidpi_factor(hidpi_factor),
"`WindowExtIOS::set_hidpi_factor` received an invalid hidpi factor"
);
let hidpi_factor = hidpi_factor as CGFloat; let hidpi_factor = hidpi_factor as CGFloat;
let () = msg_send![self.view, setContentScaleFactor:hidpi_factor]; let () = msg_send![self.view, setContentScaleFactor: hidpi_factor];
} }
} }
pub fn set_valid_orientations(&self, valid_orientations: ValidOrientations) { pub fn set_valid_orientations(&self, valid_orientations: ValidOrientations) {
unsafe { unsafe {
let idiom = event_loop::get_idiom(); let idiom = event_loop::get_idiom();
let supported_orientations = UIInterfaceOrientationMask::from_valid_orientations_idiom(valid_orientations, idiom); let supported_orientations = UIInterfaceOrientationMask::from_valid_orientations_idiom(
msg_send![self.view_controller, setSupportedInterfaceOrientations:supported_orientations]; valid_orientations,
idiom,
);
msg_send![
self.view_controller,
setSupportedInterfaceOrientations: supported_orientations
];
} }
} }
} }
@ -411,14 +425,18 @@ impl Inner {
let screen_frame = self.to_screen_space(bounds); let screen_frame = self.to_screen_space(bounds);
let status_bar_frame: CGRect = { let status_bar_frame: CGRect = {
let app: id = msg_send![class!(UIApplication), sharedApplication]; let app: id = msg_send![class!(UIApplication), sharedApplication];
assert!(!app.is_null(), "`Window::get_inner_position` cannot be called before `EventLoop::run` on iOS"); assert!(
!app.is_null(),
"`Window::get_inner_position` cannot be called before `EventLoop::run` on iOS"
);
msg_send![app, statusBarFrame] msg_send![app, statusBarFrame]
}; };
let (y, height) = if screen_frame.origin.y > status_bar_frame.size.height { let (y, height) = if screen_frame.origin.y > status_bar_frame.size.height {
(screen_frame.origin.y, screen_frame.size.height) (screen_frame.origin.y, screen_frame.size.height)
} else { } else {
let y = status_bar_frame.size.height; let y = status_bar_frame.size.height;
let height = screen_frame.size.height - (status_bar_frame.size.height - screen_frame.origin.y); let height = screen_frame.size.height
- (status_bar_frame.size.height - screen_frame.origin.y);
(y, height) (y, height)
}; };
CGRect { CGRect {
@ -429,7 +447,7 @@ impl Inner {
size: CGSize { size: CGSize {
width: screen_frame.size.width, width: screen_frame.size.width,
height, height,
} },
} }
} }
} }
@ -453,13 +471,17 @@ unsafe impl Sync for WindowId {}
impl From<&Object> for WindowId { impl From<&Object> for WindowId {
fn from(window: &Object) -> WindowId { fn from(window: &Object) -> WindowId {
WindowId { window: window as *const _ as _ } WindowId {
window: window as *const _ as _,
}
} }
} }
impl From<&mut Object> for WindowId { impl From<&mut Object> for WindowId {
fn from(window: &mut Object) -> WindowId { fn from(window: &mut Object) -> WindowId {
WindowId { window: window as _ } WindowId {
window: window as _,
}
} }
} }

View file

@ -1,13 +1,13 @@
#![cfg(any(target_os = "linux", target_os = "dragonfly", target_os = "freebsd", target_os = "netbsd", target_os = "openbsd"))] #![cfg(any(target_os = "linux", target_os = "dragonfly", target_os = "freebsd", target_os = "netbsd", target_os = "openbsd"))]
#![allow(dead_code)] #![allow(dead_code)]
use std::os::raw::{c_void, c_char, c_int}; use std::os::raw::{c_char, c_int, c_void};
pub const RTLD_LAZY: c_int = 0x001; pub const RTLD_LAZY: c_int = 0x001;
pub const RTLD_NOW: c_int = 0x002; pub const RTLD_NOW: c_int = 0x002;
#[link(name ="dl")] #[link(name = "dl")]
extern { extern "C" {
pub fn dlopen(filename: *const c_char, flag: c_int) -> *mut c_void; pub fn dlopen(filename: *const c_char, flag: c_int) -> *mut c_void;
pub fn dlerror() -> *mut c_char; pub fn dlerror() -> *mut c_char;
pub fn dlsym(handle: *mut c_void, symbol: *const c_char) -> *mut c_void; pub fn dlsym(handle: *mut c_void, symbol: *const c_char) -> *mut c_void;

View file

@ -1,24 +1,21 @@
#![cfg(any(target_os = "linux", target_os = "dragonfly", target_os = "freebsd", target_os = "netbsd", target_os = "openbsd"))] #![cfg(any(target_os = "linux", target_os = "dragonfly", target_os = "freebsd", target_os = "netbsd", target_os = "openbsd"))]
use std::collections::VecDeque; use std::{collections::VecDeque, env, ffi::CStr, fmt, mem, os::raw::*, sync::Arc};
use std::{env, mem, fmt};
use std::ffi::CStr;
use std::os::raw::*;
use std::sync::Arc;
use parking_lot::Mutex; use parking_lot::Mutex;
use smithay_client_toolkit::reexports::client::ConnectError; use smithay_client_toolkit::reexports::client::ConnectError;
use crate::dpi::{LogicalPosition, LogicalSize, PhysicalPosition, PhysicalSize};
use crate::icon::Icon;
use crate::error::{ExternalError, NotSupportedError, OsError as RootOsError};
use crate::event::Event;
use crate::event_loop::{EventLoopClosed, ControlFlow, EventLoopWindowTarget as RootELW};
use crate::monitor::{MonitorHandle as RootMonitorHandle, VideoMode};
use crate::window::{WindowAttributes, CursorIcon};
use self::x11::{XConnection, XError};
use self::x11::ffi::XVisualInfo;
pub use self::x11::XNotSupported; pub use self::x11::XNotSupported;
use self::x11::{ffi::XVisualInfo, XConnection, XError};
use crate::{
dpi::{LogicalPosition, LogicalSize, PhysicalPosition, PhysicalSize},
error::{ExternalError, NotSupportedError, OsError as RootOsError},
event::Event,
event_loop::{ControlFlow, EventLoopClosed, EventLoopWindowTarget as RootELW},
icon::Icon,
monitor::{MonitorHandle as RootMonitorHandle, VideoMode},
window::{CursorIcon, WindowAttributes},
};
mod dlopen; mod dlopen;
pub mod wayland; pub mod wayland;
@ -43,14 +40,13 @@ pub struct PlatformSpecificWindowBuilderAttributes {
pub override_redirect: bool, pub override_redirect: bool,
pub x11_window_type: x11::util::WindowType, pub x11_window_type: x11::util::WindowType,
pub gtk_theme_variant: Option<String>, pub gtk_theme_variant: Option<String>,
pub app_id: Option<String> pub app_id: Option<String>,
} }
lazy_static!( lazy_static! {
pub static ref X11_BACKEND: Mutex<Result<Arc<XConnection>, XNotSupported>> = { pub static ref X11_BACKEND: Mutex<Result<Arc<XConnection>, XNotSupported>> =
Mutex::new(XConnection::new(Some(x_error_callback)).map(Arc::new)) { Mutex::new(XConnection::new(Some(x_error_callback)).map(Arc::new)) };
}; }
);
#[derive(Debug, Clone)] #[derive(Debug, Clone)]
pub enum OsError { pub enum OsError {
@ -269,7 +265,7 @@ impl Window {
pub fn set_cursor_icon(&self, cursor: CursorIcon) { pub fn set_cursor_icon(&self, cursor: CursorIcon) {
match self { match self {
&Window::X(ref w) => w.set_cursor_icon(cursor), &Window::X(ref w) => w.set_cursor_icon(cursor),
&Window::Wayland(ref w) => w.set_cursor_icon(cursor) &Window::Wayland(ref w) => w.set_cursor_icon(cursor),
} }
} }
@ -291,7 +287,7 @@ impl Window {
#[inline] #[inline]
pub fn hidpi_factor(&self) -> f64 { pub fn hidpi_factor(&self) -> f64 {
match self { match self {
&Window::X(ref w) => w.hidpi_factor(), &Window::X(ref w) => w.hidpi_factor(),
&Window::Wayland(ref w) => w.hidpi_factor() as f64, &Window::Wayland(ref w) => w.hidpi_factor() as f64,
} }
@ -317,8 +313,13 @@ impl Window {
pub fn fullscreen(&self) -> Option<RootMonitorHandle> { pub fn fullscreen(&self) -> Option<RootMonitorHandle> {
match self { match self {
&Window::X(ref w) => w.fullscreen(), &Window::X(ref w) => w.fullscreen(),
&Window::Wayland(ref w) => w.fullscreen() &Window::Wayland(ref w) => {
.map(|monitor_id| RootMonitorHandle { inner: MonitorHandle::Wayland(monitor_id) }) w.fullscreen().map(|monitor_id| {
RootMonitorHandle {
inner: MonitorHandle::Wayland(monitor_id),
}
})
},
} }
} }
@ -326,7 +327,7 @@ impl Window {
pub fn set_fullscreen(&self, monitor: Option<RootMonitorHandle>) { pub fn set_fullscreen(&self, monitor: Option<RootMonitorHandle>) {
match self { match self {
&Window::X(ref w) => w.set_fullscreen(monitor), &Window::X(ref w) => w.set_fullscreen(monitor),
&Window::Wayland(ref w) => w.set_fullscreen(monitor) &Window::Wayland(ref w) => w.set_fullscreen(monitor),
} }
} }
@ -334,7 +335,7 @@ impl Window {
pub fn set_decorations(&self, decorations: bool) { pub fn set_decorations(&self, decorations: bool) {
match self { match self {
&Window::X(ref w) => w.set_decorations(decorations), &Window::X(ref w) => w.set_decorations(decorations),
&Window::Wayland(ref w) => w.set_decorations(decorations) &Window::Wayland(ref w) => w.set_decorations(decorations),
} }
} }
@ -373,22 +374,36 @@ impl Window {
#[inline] #[inline]
pub fn current_monitor(&self) -> RootMonitorHandle { pub fn current_monitor(&self) -> RootMonitorHandle {
match self { match self {
&Window::X(ref window) => RootMonitorHandle { inner: MonitorHandle::X(window.current_monitor()) }, &Window::X(ref window) => {
&Window::Wayland(ref window) => RootMonitorHandle { inner: MonitorHandle::Wayland(window.current_monitor()) }, RootMonitorHandle {
inner: MonitorHandle::X(window.current_monitor()),
}
},
&Window::Wayland(ref window) => {
RootMonitorHandle {
inner: MonitorHandle::Wayland(window.current_monitor()),
}
},
} }
} }
#[inline] #[inline]
pub fn available_monitors(&self) -> VecDeque<MonitorHandle> { pub fn available_monitors(&self) -> VecDeque<MonitorHandle> {
match self { match self {
&Window::X(ref window) => window.available_monitors() &Window::X(ref window) => {
.into_iter() window
.map(MonitorHandle::X) .available_monitors()
.collect(), .into_iter()
&Window::Wayland(ref window) => window.available_monitors() .map(MonitorHandle::X)
.into_iter() .collect()
.map(MonitorHandle::Wayland) },
.collect(), &Window::Wayland(ref window) => {
window
.available_monitors()
.into_iter()
.map(MonitorHandle::Wayland)
.collect()
},
} }
} }
@ -401,7 +416,6 @@ impl Window {
} }
} }
unsafe extern "C" fn x_error_callback( unsafe extern "C" fn x_error_callback(
display: *mut x11::ffi::Display, display: *mut x11::ffi::Display,
event: *mut x11::ffi::XErrorEvent, event: *mut x11::ffi::XErrorEvent,
@ -432,10 +446,9 @@ unsafe extern "C" fn x_error_callback(
0 0
} }
pub enum EventLoop<T: 'static> { pub enum EventLoop<T: 'static> {
Wayland(wayland::EventLoop<T>), Wayland(wayland::EventLoop<T>),
X(x11::EventLoop<T>) X(x11::EventLoop<T>),
} }
#[derive(Clone)] #[derive(Clone)]
@ -444,7 +457,7 @@ pub enum EventLoopProxy<T: 'static> {
Wayland(wayland::EventLoopProxy<T>), Wayland(wayland::EventLoopProxy<T>),
} }
impl<T:'static> EventLoop<T> { impl<T: 'static> EventLoop<T> {
pub fn new() -> EventLoop<T> { pub fn new() -> EventLoop<T> {
if let Ok(env_var) = env::var(BACKEND_PREFERENCE_ENV_VAR) { if let Ok(env_var) = env::var(BACKEND_PREFERENCE_ENV_VAR) {
match env_var.as_str() { match env_var.as_str() {
@ -453,13 +466,14 @@ impl<T:'static> EventLoop<T> {
return EventLoop::new_x11().expect("Failed to initialize X11 backend"); return EventLoop::new_x11().expect("Failed to initialize X11 backend");
}, },
"wayland" => { "wayland" => {
return EventLoop::new_wayland() return EventLoop::new_wayland().expect("Failed to initialize Wayland backend");
.expect("Failed to initialize Wayland backend"); },
_ => {
panic!(
"Unknown environment variable value for {}, try one of `x11`,`wayland`",
BACKEND_PREFERENCE_ENV_VAR,
)
}, },
_ => panic!(
"Unknown environment variable value for {}, try one of `x11`,`wayland`",
BACKEND_PREFERENCE_ENV_VAR,
),
} }
} }
@ -475,15 +489,13 @@ impl<T:'static> EventLoop<T> {
let err_string = format!( let err_string = format!(
"Failed to initialize any backend! Wayland status: {:?} X11 status: {:?}", "Failed to initialize any backend! Wayland status: {:?} X11 status: {:?}",
wayland_err, wayland_err, x11_err,
x11_err,
); );
panic!(err_string); panic!(err_string);
} }
pub fn new_wayland() -> Result<EventLoop<T>, ConnectError> { pub fn new_wayland() -> Result<EventLoop<T>, ConnectError> {
wayland::EventLoop::new() wayland::EventLoop::new().map(EventLoop::Wayland)
.map(EventLoop::Wayland)
} }
pub fn new_x11() -> Result<EventLoop<T>, XNotSupported> { pub fn new_x11() -> Result<EventLoop<T>, XNotSupported> {
@ -499,17 +511,19 @@ impl<T:'static> EventLoop<T> {
#[inline] #[inline]
pub fn available_monitors(&self) -> VecDeque<MonitorHandle> { pub fn available_monitors(&self) -> VecDeque<MonitorHandle> {
match *self { match *self {
EventLoop::Wayland(ref evlp) => evlp EventLoop::Wayland(ref evlp) => {
.available_monitors() evlp.available_monitors()
.into_iter() .into_iter()
.map(MonitorHandle::Wayland) .map(MonitorHandle::Wayland)
.collect(), .collect()
EventLoop::X(ref evlp) => evlp },
.x_connection() EventLoop::X(ref evlp) => {
.available_monitors() evlp.x_connection()
.into_iter() .available_monitors()
.map(MonitorHandle::X) .into_iter()
.collect(), .map(MonitorHandle::X)
.collect()
},
} }
} }
@ -529,20 +543,22 @@ impl<T:'static> EventLoop<T> {
} }
pub fn run_return<F>(&mut self, callback: F) pub fn run_return<F>(&mut self, callback: F)
where F: FnMut(crate::event::Event<T>, &RootELW<T>, &mut ControlFlow) where
F: FnMut(crate::event::Event<T>, &RootELW<T>, &mut ControlFlow),
{ {
match *self { match *self {
EventLoop::Wayland(ref mut evlp) => evlp.run_return(callback), EventLoop::Wayland(ref mut evlp) => evlp.run_return(callback),
EventLoop::X(ref mut evlp) => evlp.run_return(callback) EventLoop::X(ref mut evlp) => evlp.run_return(callback),
} }
} }
pub fn run<F>(self, callback: F) -> ! pub fn run<F>(self, callback: F) -> !
where F: 'static + FnMut(crate::event::Event<T>, &RootELW<T>, &mut ControlFlow) where
F: 'static + FnMut(crate::event::Event<T>, &RootELW<T>, &mut ControlFlow),
{ {
match self { match self {
EventLoop::Wayland(evlp) => evlp.run(callback), EventLoop::Wayland(evlp) => evlp.run(callback),
EventLoop::X(evlp) => evlp.run(callback) EventLoop::X(evlp) => evlp.run(callback),
} }
} }
@ -557,7 +573,7 @@ impl<T:'static> EventLoop<T> {
pub fn window_target(&self) -> &crate::event_loop::EventLoopWindowTarget<T> { pub fn window_target(&self) -> &crate::event_loop::EventLoopWindowTarget<T> {
match *self { match *self {
EventLoop::Wayland(ref evl) => evl.window_target(), EventLoop::Wayland(ref evl) => evl.window_target(),
EventLoop::X(ref evl) => evl.window_target() EventLoop::X(ref evl) => evl.window_target(),
} }
} }
} }
@ -573,12 +589,16 @@ impl<T: 'static> EventLoopProxy<T> {
pub enum EventLoopWindowTarget<T> { pub enum EventLoopWindowTarget<T> {
Wayland(wayland::EventLoopWindowTarget<T>), Wayland(wayland::EventLoopWindowTarget<T>),
X(x11::EventLoopWindowTarget<T>) X(x11::EventLoopWindowTarget<T>),
} }
fn sticky_exit_callback<T, F>( fn sticky_exit_callback<T, F>(
evt: Event<T>, target: &RootELW<T>, control_flow: &mut ControlFlow, callback: &mut F evt: Event<T>,
) where F: FnMut(Event<T>, &RootELW<T>, &mut ControlFlow) target: &RootELW<T>,
control_flow: &mut ControlFlow,
callback: &mut F,
) where
F: FnMut(Event<T>, &RootELW<T>, &mut ControlFlow),
{ {
// make ControlFlow::Exit sticky by providing a dummy // make ControlFlow::Exit sticky by providing a dummy
// control flow reference if it is already Exit. // control flow reference if it is already Exit.

View file

@ -1,25 +1,30 @@
use std::cell::RefCell; use std::{
use std::collections::VecDeque; cell::RefCell,
use std::fmt; collections::VecDeque,
use std::rc::Rc; fmt,
use std::sync::{Arc, Mutex}; rc::Rc,
use std::time::Instant; sync::{Arc, Mutex},
time::Instant,
use crate::event_loop::{ControlFlow, EventLoopClosed, EventLoopWindowTarget as RootELW}; };
use crate::event::ModifiersState;
use crate::dpi::{PhysicalPosition, PhysicalSize}; use crate::{
use crate::platform_impl::platform::sticky_exit_callback; dpi::{PhysicalPosition, PhysicalSize},
use crate::monitor::VideoMode; event::ModifiersState,
event_loop::{ControlFlow, EventLoopClosed, EventLoopWindowTarget as RootELW},
use super::window::WindowStore; monitor::VideoMode,
use super::WindowId; platform_impl::platform::sticky_exit_callback,
};
use smithay_client_toolkit::output::OutputMgr;
use smithay_client_toolkit::reexports::client::protocol::{ use super::{window::WindowStore, WindowId};
wl_keyboard, wl_output, wl_pointer, wl_registry, wl_seat, wl_touch,
use smithay_client_toolkit::{
output::OutputMgr,
reexports::client::{
protocol::{wl_keyboard, wl_output, wl_pointer, wl_registry, wl_seat, wl_touch},
ConnectError, Display, EventQueue, GlobalEvent,
},
Environment,
}; };
use smithay_client_toolkit::reexports::client::{ConnectError, Display, EventQueue, GlobalEvent};
use smithay_client_toolkit::Environment;
pub struct WindowEventsSink { pub struct WindowEventsSink {
buffer: VecDeque<(crate::event::WindowEvent, crate::window::WindowId)>, buffer: VecDeque<(crate::event::WindowEvent, crate::window::WindowId)>,
@ -33,7 +38,10 @@ impl WindowEventsSink {
} }
pub fn send_event(&mut self, evt: crate::event::WindowEvent, wid: WindowId) { pub fn send_event(&mut self, evt: crate::event::WindowEvent, wid: WindowId) {
self.buffer.push_back((evt, crate::window::WindowId(crate::platform_impl::WindowId::Wayland(wid)))); self.buffer.push_back((
evt,
crate::window::WindowId(crate::platform_impl::WindowId::Wayland(wid)),
));
} }
fn empty_with<F, T>(&mut self, mut callback: F) fn empty_with<F, T>(&mut self, mut callback: F)
@ -41,7 +49,10 @@ impl WindowEventsSink {
F: FnMut(crate::event::Event<T>), F: FnMut(crate::event::Event<T>),
{ {
for (evt, wid) in self.buffer.drain(..) { for (evt, wid) in self.buffer.drain(..) {
callback(crate::event::Event::WindowEvent { event: evt, window_id: wid}) callback(crate::event::Event::WindowEvent {
event: evt,
window_id: wid,
})
} }
} }
} }
@ -58,8 +69,10 @@ pub struct EventLoop<T: 'static> {
pending_user_events: Rc<RefCell<VecDeque<T>>>, pending_user_events: Rc<RefCell<VecDeque<T>>>,
_user_source: ::calloop::Source<::calloop::channel::Channel<T>>, _user_source: ::calloop::Source<::calloop::channel::Channel<T>>,
user_sender: ::calloop::channel::Sender<T>, user_sender: ::calloop::channel::Sender<T>,
_kbd_source: ::calloop::Source<::calloop::channel::Channel<(crate::event::WindowEvent, super::WindowId)>>, _kbd_source: ::calloop::Source<
window_target: RootELW<T> ::calloop::channel::Channel<(crate::event::WindowEvent, super::WindowId)>,
>,
window_target: RootELW<T>,
} }
// A handle that can be sent across threads and used to wake up the `EventLoop`. // A handle that can be sent across threads and used to wake up the `EventLoop`.
@ -67,7 +80,7 @@ pub struct EventLoop<T: 'static> {
// We should only try and wake up the `EventLoop` if it still exists, so we hold Weak ptrs. // We should only try and wake up the `EventLoop` if it still exists, so we hold Weak ptrs.
#[derive(Clone)] #[derive(Clone)]
pub struct EventLoopProxy<T: 'static> { pub struct EventLoopProxy<T: 'static> {
user_sender: ::calloop::channel::Sender<T> user_sender: ::calloop::channel::Sender<T>,
} }
pub struct EventLoopWindowTarget<T> { pub struct EventLoopWindowTarget<T> {
@ -83,7 +96,7 @@ pub struct EventLoopWindowTarget<T> {
pub display: Arc<Display>, pub display: Arc<Display>,
// The list of seats // The list of seats
pub seats: Arc<Mutex<Vec<(u32, wl_seat::WlSeat)>>>, pub seats: Arc<Mutex<Vec<(u32, wl_seat::WlSeat)>>>,
_marker: ::std::marker::PhantomData<T> _marker: ::std::marker::PhantomData<T>,
} }
impl<T: 'static> EventLoopProxy<T> { impl<T: 'static> EventLoopProxy<T> {
@ -105,11 +118,14 @@ impl<T: 'static> EventLoop<T> {
let (kbd_sender, kbd_channel) = ::calloop::channel::channel(); let (kbd_sender, kbd_channel) = ::calloop::channel::channel();
let kbd_sink = sink.clone(); let kbd_sink = sink.clone();
let kbd_source = inner_loop.handle().insert_source(kbd_channel, move |evt, &mut()| { let kbd_source = inner_loop
if let ::calloop::channel::Event::Msg((evt, wid)) = evt { .handle()
kbd_sink.lock().unwrap().send_event(evt, wid); .insert_source(kbd_channel, move |evt, &mut ()| {
} if let ::calloop::channel::Event::Msg((evt, wid)) = evt {
}).unwrap(); kbd_sink.lock().unwrap().send_event(evt, wid);
}
})
.unwrap();
let mut seat_manager = SeatManager { let mut seat_manager = SeatManager {
sink: sink.clone(), sink: sink.clone(),
@ -123,7 +139,11 @@ impl<T: 'static> EventLoop<T> {
&mut event_queue, &mut event_queue,
move |event, registry| { move |event, registry| {
match event { match event {
GlobalEvent::New { id, ref interface, version } => { GlobalEvent::New {
id,
ref interface,
version,
} => {
if interface == "wl_seat" { if interface == "wl_seat" {
seat_manager.add_seat(id, version, registry) seat_manager.add_seat(id, version, registry)
} }
@ -135,20 +155,27 @@ impl<T: 'static> EventLoop<T> {
}, },
} }
}, },
).unwrap(); )
.unwrap();
let source = inner_loop.handle().insert_source(event_queue, |(), &mut ()| {}).unwrap(); let source = inner_loop
.handle()
.insert_source(event_queue, |(), &mut ()| {})
.unwrap();
let pending_user_events = Rc::new(RefCell::new(VecDeque::new())); let pending_user_events = Rc::new(RefCell::new(VecDeque::new()));
let pending_user_events2 = pending_user_events.clone(); let pending_user_events2 = pending_user_events.clone();
let (user_sender, user_channel) = ::calloop::channel::channel(); let (user_sender, user_channel) = ::calloop::channel::channel();
let user_source = inner_loop.handle().insert_source(user_channel, move |evt, &mut()| { let user_source = inner_loop
if let ::calloop::channel::Event::Msg(msg) = evt { .handle()
pending_user_events2.borrow_mut().push_back(msg); .insert_source(user_channel, move |evt, &mut ()| {
} if let ::calloop::channel::Event::Msg(msg) = evt {
}).unwrap(); pending_user_events2.borrow_mut().push_back(msg);
}
})
.unwrap();
Ok(EventLoop { Ok(EventLoop {
inner_loop, inner_loop,
@ -167,28 +194,30 @@ impl<T: 'static> EventLoop<T> {
cleanup_needed: Arc::new(Mutex::new(false)), cleanup_needed: Arc::new(Mutex::new(false)),
seats, seats,
display, display,
_marker: ::std::marker::PhantomData _marker: ::std::marker::PhantomData,
}), }),
_marker: ::std::marker::PhantomData _marker: ::std::marker::PhantomData,
} },
}) })
} }
pub fn create_proxy(&self) -> EventLoopProxy<T> { pub fn create_proxy(&self) -> EventLoopProxy<T> {
EventLoopProxy { EventLoopProxy {
user_sender: self.user_sender.clone() user_sender: self.user_sender.clone(),
} }
} }
pub fn run<F>(mut self, callback: F) -> ! pub fn run<F>(mut self, callback: F) -> !
where F: 'static + FnMut(crate::event::Event<T>, &RootELW<T>, &mut ControlFlow) where
F: 'static + FnMut(crate::event::Event<T>, &RootELW<T>, &mut ControlFlow),
{ {
self.run_return(callback); self.run_return(callback);
::std::process::exit(0); ::std::process::exit(0);
} }
pub fn run_return<F>(&mut self, mut callback: F) pub fn run_return<F>(&mut self, mut callback: F)
where F: FnMut(crate::event::Event<T>, &RootELW<T>, &mut ControlFlow) where
F: FnMut(crate::event::Event<T>, &RootELW<T>, &mut ControlFlow),
{ {
// send pending events to the server // send pending events to the server
self.display.flush().expect("Wayland connection lost."); self.display.flush().expect("Wayland connection lost.");
@ -198,7 +227,11 @@ impl<T: 'static> EventLoop<T> {
let sink = self.sink.clone(); let sink = self.sink.clone();
let user_events = self.pending_user_events.clone(); let user_events = self.pending_user_events.clone();
callback(crate::event::Event::NewEvents(crate::event::StartCause::Init), &self.window_target, &mut control_flow); callback(
crate::event::Event::NewEvents(crate::event::StartCause::Init),
&self.window_target,
&mut control_flow,
);
loop { loop {
self.post_dispatch_triggers(); self.post_dispatch_triggers();
@ -207,7 +240,12 @@ impl<T: 'static> EventLoop<T> {
{ {
let mut guard = sink.lock().unwrap(); let mut guard = sink.lock().unwrap();
guard.empty_with(|evt| { guard.empty_with(|evt| {
sticky_exit_callback(evt, &self.window_target, &mut control_flow, &mut callback); sticky_exit_callback(
evt,
&self.window_target,
&mut control_flow,
&mut callback,
);
}); });
} }
// empty user events // empty user events
@ -218,7 +256,7 @@ impl<T: 'static> EventLoop<T> {
crate::event::Event::UserEvent(evt), crate::event::Event::UserEvent(evt),
&self.window_target, &self.window_target,
&mut control_flow, &mut control_flow,
&mut callback &mut callback,
); );
} }
} }
@ -228,7 +266,12 @@ impl<T: 'static> EventLoop<T> {
{ {
let mut guard = sink.lock().unwrap(); let mut guard = sink.lock().unwrap();
guard.empty_with(|evt| { guard.empty_with(|evt| {
sticky_exit_callback(evt, &self.window_target, &mut control_flow, &mut callback); sticky_exit_callback(
evt,
&self.window_target,
&mut control_flow,
&mut callback,
);
}); });
} }
// send Events cleared // send Events cleared
@ -237,7 +280,7 @@ impl<T: 'static> EventLoop<T> {
crate::event::Event::EventsCleared, crate::event::Event::EventsCleared,
&self.window_target, &self.window_target,
&mut control_flow, &mut control_flow,
&mut callback &mut callback,
); );
} }
@ -248,18 +291,24 @@ impl<T: 'static> EventLoop<T> {
ControlFlow::Exit => break, ControlFlow::Exit => break,
ControlFlow::Poll => { ControlFlow::Poll => {
// non-blocking dispatch // non-blocking dispatch
self.inner_loop.dispatch(Some(::std::time::Duration::from_millis(0)), &mut ()).unwrap(); self.inner_loop
callback(crate::event::Event::NewEvents(crate::event::StartCause::Poll), &self.window_target, &mut control_flow); .dispatch(Some(::std::time::Duration::from_millis(0)), &mut ())
.unwrap();
callback(
crate::event::Event::NewEvents(crate::event::StartCause::Poll),
&self.window_target,
&mut control_flow,
);
}, },
ControlFlow::Wait => { ControlFlow::Wait => {
self.inner_loop.dispatch(None, &mut ()).unwrap(); self.inner_loop.dispatch(None, &mut ()).unwrap();
callback( callback(
crate::event::Event::NewEvents(crate::event::StartCause::WaitCancelled { crate::event::Event::NewEvents(crate::event::StartCause::WaitCancelled {
start: Instant::now(), start: Instant::now(),
requested_resume: None requested_resume: None,
}), }),
&self.window_target, &self.window_target,
&mut control_flow &mut control_flow,
); );
}, },
ControlFlow::WaitUntil(deadline) => { ControlFlow::WaitUntil(deadline) => {
@ -274,28 +323,36 @@ impl<T: 'static> EventLoop<T> {
let now = Instant::now(); let now = Instant::now();
if now < deadline { if now < deadline {
callback( callback(
crate::event::Event::NewEvents(crate::event::StartCause::WaitCancelled { crate::event::Event::NewEvents(
start, crate::event::StartCause::WaitCancelled {
requested_resume: Some(deadline) start,
}), requested_resume: Some(deadline),
},
),
&self.window_target, &self.window_target,
&mut control_flow &mut control_flow,
); );
} else { } else {
callback( callback(
crate::event::Event::NewEvents(crate::event::StartCause::ResumeTimeReached { crate::event::Event::NewEvents(
start, crate::event::StartCause::ResumeTimeReached {
requested_resume: deadline start,
}), requested_resume: deadline,
},
),
&self.window_target, &self.window_target,
&mut control_flow &mut control_flow,
); );
} }
}, },
} }
} }
callback(crate::event::Event::LoopDestroyed, &self.window_target, &mut control_flow); callback(
crate::event::Event::LoopDestroyed,
&self.window_target,
&mut control_flow,
);
} }
pub fn primary_monitor(&self) -> MonitorHandle { pub fn primary_monitor(&self) -> MonitorHandle {
@ -324,7 +381,7 @@ impl<T> EventLoop<T> {
let mut sink = self.sink.lock().unwrap(); let mut sink = self.sink.lock().unwrap();
let window_target = match self.window_target.p { let window_target = match self.window_target.p {
crate::platform_impl::EventLoopWindowTarget::Wayland(ref wt) => wt, crate::platform_impl::EventLoopWindowTarget::Wayland(ref wt) => wt,
_ => unreachable!() _ => unreachable!(),
}; };
// prune possible dead windows // prune possible dead windows
{ {
@ -355,7 +412,10 @@ impl<T> EventLoop<T> {
} }
} }
if let Some(dpi) = new_dpi { if let Some(dpi) = new_dpi {
sink.send_event(crate::event::WindowEvent::HiDpiFactorChanged(dpi as f64), wid); sink.send_event(
crate::event::WindowEvent::HiDpiFactorChanged(dpi as f64),
wid,
);
} }
if refresh { if refresh {
sink.send_event(crate::event::WindowEvent::RedrawRequested, wid); sink.send_event(crate::event::WindowEvent::RedrawRequested, wid);
@ -376,7 +436,7 @@ struct SeatManager {
sink: Arc<Mutex<WindowEventsSink>>, sink: Arc<Mutex<WindowEventsSink>>,
store: Arc<Mutex<WindowStore>>, store: Arc<Mutex<WindowStore>>,
seats: Arc<Mutex<Vec<(u32, wl_seat::WlSeat)>>>, seats: Arc<Mutex<Vec<(u32, wl_seat::WlSeat)>>>,
kbd_sender: ::calloop::channel::Sender<(crate::event::WindowEvent, super::WindowId)> kbd_sender: ::calloop::channel::Sender<(crate::event::WindowEvent, super::WindowId)>,
} }
impl SeatManager { impl SeatManager {
@ -394,9 +454,7 @@ impl SeatManager {
}; };
let seat = registry let seat = registry
.bind(min(version, 5), id, move |seat| { .bind(min(version, 5), id, move |seat| {
seat.implement_closure(move |event, seat| { seat.implement_closure(move |event, seat| seat_data.receive(event, seat), ())
seat_data.receive(event, seat)
}, ())
}) })
.unwrap(); .unwrap();
self.store.lock().unwrap().new_seat(&seat); self.store.lock().unwrap().new_seat(&seat);
@ -479,7 +537,7 @@ impl SeatData {
} }
} }
}, },
_ => unreachable!() _ => unreachable!(),
} }
} }
} }
@ -566,7 +624,8 @@ impl MonitorHandle {
}) { }) {
Some(Some((w, h))) => (w as u32, h as u32), Some(Some((w, h))) => (w as u32, h as u32),
_ => (0, 0), _ => (0, 0),
}.into() }
.into()
} }
pub fn position(&self) -> PhysicalPosition { pub fn position(&self) -> PhysicalPosition {
@ -584,16 +643,17 @@ impl MonitorHandle {
} }
#[inline] #[inline]
pub fn video_modes(&self) -> impl Iterator<Item = VideoMode> pub fn video_modes(&self) -> impl Iterator<Item = VideoMode> {
{
self.mgr self.mgr
.with_info(&self.proxy, |_, info| info.modes.clone()) .with_info(&self.proxy, |_, info| info.modes.clone())
.unwrap_or(vec![]) .unwrap_or(vec![])
.into_iter() .into_iter()
.map(|x| VideoMode { .map(|x| {
size: (x.dimensions.0 as u32, x.dimensions.1 as u32), VideoMode {
refresh_rate: (x.refresh_rate as f32 / 1000.0).round() as u16, size: (x.dimensions.0 as u32, x.dimensions.1 as u32),
bit_depth: 32 refresh_rate: (x.refresh_rate as f32 / 1000.0).round() as u16,
bit_depth: 32,
}
}) })
} }
} }
@ -614,9 +674,11 @@ pub fn primary_monitor(outputs: &OutputMgr) -> MonitorHandle {
pub fn available_monitors(outputs: &OutputMgr) -> VecDeque<MonitorHandle> { pub fn available_monitors(outputs: &OutputMgr) -> VecDeque<MonitorHandle> {
outputs.with_all(|list| { outputs.with_all(|list| {
list.iter() list.iter()
.map(|&(_, ref proxy, _)| MonitorHandle { .map(|&(_, ref proxy, _)| {
proxy: proxy.clone(), MonitorHandle {
mgr: outputs.clone(), proxy: proxy.clone(),
mgr: outputs.clone(),
}
}) })
.collect() .collect()
}) })

View file

@ -1,10 +1,12 @@
use std::sync::{Arc, Mutex}; use std::sync::{Arc, Mutex};
use super::{make_wid, DeviceId}; use super::{make_wid, DeviceId};
use smithay_client_toolkit::keyboard::{ use smithay_client_toolkit::{
self, map_keyboard_auto_with_repeat, Event as KbEvent, KeyRepeatEvent, KeyRepeatKind, keyboard::{
self, map_keyboard_auto_with_repeat, Event as KbEvent, KeyRepeatEvent, KeyRepeatKind,
},
reexports::client::protocol::{wl_keyboard, wl_seat},
}; };
use smithay_client_toolkit::reexports::client::protocol::{wl_keyboard, wl_seat};
use crate::event::{ElementState, KeyboardInput, ModifiersState, VirtualKeyCode, WindowEvent}; use crate::event::{ElementState, KeyboardInput, ModifiersState, VirtualKeyCode, WindowEvent};
@ -23,78 +25,92 @@ pub fn init_keyboard(
let ret = map_keyboard_auto_with_repeat( let ret = map_keyboard_auto_with_repeat(
seat, seat,
KeyRepeatKind::System, KeyRepeatKind::System,
move |evt: KbEvent<'_>, _| match evt { move |evt: KbEvent<'_>, _| {
KbEvent::Enter { surface, .. } => { match evt {
let wid = make_wid(&surface); KbEvent::Enter { surface, .. } => {
my_sink.send((WindowEvent::Focused(true), wid)).unwrap(); let wid = make_wid(&surface);
*target.lock().unwrap() = Some(wid); my_sink.send((WindowEvent::Focused(true), wid)).unwrap();
} *target.lock().unwrap() = Some(wid);
KbEvent::Leave { surface, .. } => { },
let wid = make_wid(&surface); KbEvent::Leave { surface, .. } => {
my_sink.send((WindowEvent::Focused(false), wid)).unwrap(); let wid = make_wid(&surface);
*target.lock().unwrap() = None; my_sink.send((WindowEvent::Focused(false), wid)).unwrap();
} *target.lock().unwrap() = None;
KbEvent::Key { },
rawkey, KbEvent::Key {
keysym, rawkey,
state, keysym,
utf8, state,
.. utf8,
} => { ..
if let Some(wid) = *target.lock().unwrap() { } => {
let state = match state { if let Some(wid) = *target.lock().unwrap() {
wl_keyboard::KeyState::Pressed => ElementState::Pressed, let state = match state {
wl_keyboard::KeyState::Released => ElementState::Released, wl_keyboard::KeyState::Pressed => ElementState::Pressed,
_ => unreachable!(), wl_keyboard::KeyState::Released => ElementState::Released,
}; _ => unreachable!(),
let vkcode = key_to_vkey(rawkey, keysym); };
my_sink.send( let vkcode = key_to_vkey(rawkey, keysym);
(WindowEvent::KeyboardInput { my_sink
device_id: crate::event::DeviceId(crate::platform_impl::DeviceId::Wayland(DeviceId)), .send((
input: KeyboardInput { WindowEvent::KeyboardInput {
state: state, device_id: crate::event::DeviceId(
scancode: rawkey, crate::platform_impl::DeviceId::Wayland(DeviceId),
virtual_keycode: vkcode, ),
modifiers: modifiers_tracker.lock().unwrap().clone(), input: KeyboardInput {
}, state,
}, scancode: rawkey,
wid) virtual_keycode: vkcode,
).unwrap(); modifiers: modifiers_tracker.lock().unwrap().clone(),
// send char event only on key press, not release },
if let ElementState::Released = state { },
return; wid,
} ))
if let Some(txt) = utf8 { .unwrap();
for chr in txt.chars() { // send char event only on key press, not release
my_sink.send((WindowEvent::ReceivedCharacter(chr), wid)).unwrap(); if let ElementState::Released = state {
return;
}
if let Some(txt) = utf8 {
for chr in txt.chars() {
my_sink
.send((WindowEvent::ReceivedCharacter(chr), wid))
.unwrap();
}
} }
} }
} },
KbEvent::RepeatInfo { .. } => { /* Handled by smithay client toolkit */ },
KbEvent::Modifiers {
modifiers: event_modifiers,
} => *modifiers_tracker.lock().unwrap() = event_modifiers.into(),
} }
KbEvent::RepeatInfo { .. } => { /* Handled by smithay client toolkit */ }
KbEvent::Modifiers { modifiers: event_modifiers } => {
*modifiers_tracker.lock().unwrap() = event_modifiers.into()
},
}, },
move |repeat_event: KeyRepeatEvent, _| { move |repeat_event: KeyRepeatEvent, _| {
if let Some(wid) = *repeat_target.lock().unwrap() { if let Some(wid) = *repeat_target.lock().unwrap() {
let state = ElementState::Pressed; let state = ElementState::Pressed;
let vkcode = key_to_vkey(repeat_event.rawkey, repeat_event.keysym); let vkcode = key_to_vkey(repeat_event.rawkey, repeat_event.keysym);
repeat_sink.send(( repeat_sink
WindowEvent::KeyboardInput { .send((
device_id: crate::event::DeviceId(crate::platform_impl::DeviceId::Wayland(DeviceId)), WindowEvent::KeyboardInput {
input: KeyboardInput { device_id: crate::event::DeviceId(
state: state, crate::platform_impl::DeviceId::Wayland(DeviceId),
scancode: repeat_event.rawkey, ),
virtual_keycode: vkcode, input: KeyboardInput {
modifiers: my_modifiers.lock().unwrap().clone(), state,
scancode: repeat_event.rawkey,
virtual_keycode: vkcode,
modifiers: my_modifiers.lock().unwrap().clone(),
},
}, },
}, wid,
wid) ))
).unwrap(); .unwrap();
if let Some(txt) = repeat_event.utf8 { if let Some(txt) = repeat_event.utf8 {
for chr in txt.chars() { for chr in txt.chars() {
repeat_sink.send((WindowEvent::ReceivedCharacter(chr), wid)).unwrap(); repeat_sink
.send((WindowEvent::ReceivedCharacter(chr), wid))
.unwrap();
} }
} }
} }
@ -116,42 +132,54 @@ pub fn init_keyboard(
let my_sink = sink; let my_sink = sink;
// } // }
seat.get_keyboard(|keyboard| { seat.get_keyboard(|keyboard| {
keyboard.implement_closure(move |evt, _| match evt { keyboard.implement_closure(
wl_keyboard::Event::Enter { surface, .. } => { move |evt, _| {
let wid = make_wid(&surface); match evt {
my_sink.send((WindowEvent::Focused(true), wid)).unwrap(); wl_keyboard::Event::Enter { surface, .. } => {
target = Some(wid); let wid = make_wid(&surface);
} my_sink.send((WindowEvent::Focused(true), wid)).unwrap();
wl_keyboard::Event::Leave { surface, .. } => { target = Some(wid);
let wid = make_wid(&surface); },
my_sink.send((WindowEvent::Focused(false), wid)).unwrap(); wl_keyboard::Event::Leave { surface, .. } => {
target = None; let wid = make_wid(&surface);
} my_sink.send((WindowEvent::Focused(false), wid)).unwrap();
wl_keyboard::Event::Key { key, state, .. } => { target = None;
if let Some(wid) = target { },
let state = match state { wl_keyboard::Event::Key { key, state, .. } => {
wl_keyboard::KeyState::Pressed => ElementState::Pressed, if let Some(wid) = target {
wl_keyboard::KeyState::Released => ElementState::Released, let state = match state {
_ => unreachable!() wl_keyboard::KeyState::Pressed => ElementState::Pressed,
}; wl_keyboard::KeyState::Released => ElementState::Released,
my_sink.send(( _ => unreachable!(),
WindowEvent::KeyboardInput { };
device_id: crate::event::DeviceId(crate::platform_impl::DeviceId::Wayland(DeviceId)), my_sink
input: KeyboardInput { .send((
state: state, WindowEvent::KeyboardInput {
scancode: key, device_id: crate::event::DeviceId(
virtual_keycode: None, crate::platform_impl::DeviceId::Wayland(
modifiers: ModifiersState::default(), DeviceId,
}, ),
}, ),
wid, input: KeyboardInput {
)).unwrap(); state,
scancode: key,
virtual_keycode: None,
modifiers: ModifiersState::default(),
},
},
wid,
))
.unwrap();
}
},
_ => (),
} }
} },
_ => (), (),
}, ()) )
}).unwrap() })
} .unwrap()
},
} }
} }

View file

@ -1,15 +1,19 @@
#![cfg(any(target_os = "linux", target_os = "dragonfly", target_os = "freebsd", #![cfg(any(target_os = "linux", target_os = "dragonfly", target_os = "freebsd",
target_os = "netbsd", target_os = "openbsd"))] target_os = "netbsd", target_os = "openbsd"))]
pub use self::window::Window; pub use self::{
pub use self::event_loop::{EventLoop, EventLoopWindowTarget, EventLoopProxy, WindowEventsSink, MonitorHandle}; event_loop::{
EventLoop, EventLoopProxy, EventLoopWindowTarget, MonitorHandle, WindowEventsSink,
},
window::Window,
};
use smithay_client_toolkit::reexports::client::protocol::wl_surface; use smithay_client_toolkit::reexports::client::protocol::wl_surface;
mod event_loop; mod event_loop;
mod keyboard;
mod pointer; mod pointer;
mod touch; mod touch;
mod keyboard;
mod window; mod window;
#[derive(Debug, Copy, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)] #[derive(Debug, Copy, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]

View file

@ -1,13 +1,15 @@
use std::sync::{Arc, Mutex}; use std::sync::{Arc, Mutex};
use crate::event::{ElementState, MouseButton, MouseScrollDelta, TouchPhase, WindowEvent, ModifiersState}; use crate::event::{
ElementState, ModifiersState, MouseButton, MouseScrollDelta, TouchPhase, WindowEvent,
};
use super::DeviceId; use super::{event_loop::WindowEventsSink, window::WindowStore, DeviceId};
use super::event_loop::WindowEventsSink;
use super::window::WindowStore;
use smithay_client_toolkit::reexports::client::protocol::wl_pointer::{self, Event as PtrEvent, WlPointer}; use smithay_client_toolkit::reexports::client::protocol::{
use smithay_client_toolkit::reexports::client::protocol::wl_seat; wl_pointer::{self, Event as PtrEvent, WlPointer},
wl_seat,
};
pub fn implement_pointer( pub fn implement_pointer(
seat: &wl_seat::WlSeat, seat: &wl_seat::WlSeat,
@ -21,171 +23,195 @@ pub fn implement_pointer(
let mut axis_state = TouchPhase::Ended; let mut axis_state = TouchPhase::Ended;
seat.get_pointer(|pointer| { seat.get_pointer(|pointer| {
pointer.implement_closure(move |evt, pointer| { pointer.implement_closure(
let mut sink = sink.lock().unwrap(); move |evt, pointer| {
let store = store.lock().unwrap(); let mut sink = sink.lock().unwrap();
match evt { let store = store.lock().unwrap();
PtrEvent::Enter { match evt {
surface, PtrEvent::Enter {
surface_x, surface,
surface_y, surface_x,
.. surface_y,
} => { ..
let wid = store.find_wid(&surface); } => {
if let Some(wid) = wid { let wid = store.find_wid(&surface);
mouse_focus = Some(wid); if let Some(wid) = wid {
sink.send_event( mouse_focus = Some(wid);
WindowEvent::CursorEntered {
device_id: crate::event::DeviceId(crate::platform_impl::DeviceId::Wayland(DeviceId)),
},
wid,
);
sink.send_event(
WindowEvent::CursorMoved {
device_id: crate::event::DeviceId(crate::platform_impl::DeviceId::Wayland(DeviceId)),
position: (surface_x, surface_y).into(),
modifiers: modifiers_tracker.lock().unwrap().clone(),
},
wid,
);
}
}
PtrEvent::Leave { surface, .. } => {
mouse_focus = None;
let wid = store.find_wid(&surface);
if let Some(wid) = wid {
sink.send_event(
WindowEvent::CursorLeft {
device_id: crate::event::DeviceId(crate::platform_impl::DeviceId::Wayland(DeviceId)),
},
wid,
);
}
}
PtrEvent::Motion {
surface_x,
surface_y,
..
} => {
if let Some(wid) = mouse_focus {
sink.send_event(
WindowEvent::CursorMoved {
device_id: crate::event::DeviceId(crate::platform_impl::DeviceId::Wayland(DeviceId)),
position: (surface_x, surface_y).into(),
modifiers: modifiers_tracker.lock().unwrap().clone(),
},
wid,
);
}
}
PtrEvent::Button { button, state, .. } => {
if let Some(wid) = mouse_focus {
let state = match state {
wl_pointer::ButtonState::Pressed => ElementState::Pressed,
wl_pointer::ButtonState::Released => ElementState::Released,
_ => unreachable!()
};
let button = match button {
0x110 => MouseButton::Left,
0x111 => MouseButton::Right,
0x112 => MouseButton::Middle,
// TODO figure out the translation ?
_ => return,
};
sink.send_event(
WindowEvent::MouseInput {
device_id: crate::event::DeviceId(crate::platform_impl::DeviceId::Wayland(DeviceId)),
state: state,
button: button,
modifiers: modifiers_tracker.lock().unwrap().clone(),
},
wid,
);
}
}
PtrEvent::Axis { axis, value, .. } => {
if let Some(wid) = mouse_focus {
if pointer.as_ref().version() < 5 {
let (mut x, mut y) = (0.0, 0.0);
// old seat compatibility
match axis {
// wayland vertical sign convention is the inverse of winit
wl_pointer::Axis::VerticalScroll => y -= value as f32,
wl_pointer::Axis::HorizontalScroll => x += value as f32,
_ => unreachable!()
}
sink.send_event( sink.send_event(
WindowEvent::MouseWheel { WindowEvent::CursorEntered {
device_id: crate::event::DeviceId(crate::platform_impl::DeviceId::Wayland(DeviceId)), device_id: crate::event::DeviceId(
delta: MouseScrollDelta::PixelDelta((x as f64, y as f64).into()), crate::platform_impl::DeviceId::Wayland(DeviceId),
phase: TouchPhase::Moved, ),
modifiers: modifiers_tracker.lock().unwrap().clone(),
}, },
wid, wid,
); );
} else {
let (mut x, mut y) = axis_buffer.unwrap_or((0.0, 0.0));
match axis {
// wayland vertical sign convention is the inverse of winit
wl_pointer::Axis::VerticalScroll => y -= value as f32,
wl_pointer::Axis::HorizontalScroll => x += value as f32,
_ => unreachable!()
}
axis_buffer = Some((x, y));
axis_state = match axis_state {
TouchPhase::Started | TouchPhase::Moved => TouchPhase::Moved,
_ => TouchPhase::Started,
}
}
}
}
PtrEvent::Frame => {
let axis_buffer = axis_buffer.take();
let axis_discrete_buffer = axis_discrete_buffer.take();
if let Some(wid) = mouse_focus {
if let Some((x, y)) = axis_discrete_buffer {
sink.send_event( sink.send_event(
WindowEvent::MouseWheel { WindowEvent::CursorMoved {
device_id: crate::event::DeviceId(crate::platform_impl::DeviceId::Wayland(DeviceId)), device_id: crate::event::DeviceId(
delta: MouseScrollDelta::LineDelta(x as f32, y as f32), crate::platform_impl::DeviceId::Wayland(DeviceId),
phase: axis_state, ),
modifiers: modifiers_tracker.lock().unwrap().clone(), position: (surface_x, surface_y).into(),
},
wid,
);
} else if let Some((x, y)) = axis_buffer {
sink.send_event(
WindowEvent::MouseWheel {
device_id: crate::event::DeviceId(crate::platform_impl::DeviceId::Wayland(DeviceId)),
delta: MouseScrollDelta::PixelDelta((x as f64, y as f64).into()),
phase: axis_state,
modifiers: modifiers_tracker.lock().unwrap().clone(), modifiers: modifiers_tracker.lock().unwrap().clone(),
}, },
wid, wid,
); );
} }
} },
PtrEvent::Leave { surface, .. } => {
mouse_focus = None;
let wid = store.find_wid(&surface);
if let Some(wid) = wid {
sink.send_event(
WindowEvent::CursorLeft {
device_id: crate::event::DeviceId(
crate::platform_impl::DeviceId::Wayland(DeviceId),
),
},
wid,
);
}
},
PtrEvent::Motion {
surface_x,
surface_y,
..
} => {
if let Some(wid) = mouse_focus {
sink.send_event(
WindowEvent::CursorMoved {
device_id: crate::event::DeviceId(
crate::platform_impl::DeviceId::Wayland(DeviceId),
),
position: (surface_x, surface_y).into(),
modifiers: modifiers_tracker.lock().unwrap().clone(),
},
wid,
);
}
},
PtrEvent::Button { button, state, .. } => {
if let Some(wid) = mouse_focus {
let state = match state {
wl_pointer::ButtonState::Pressed => ElementState::Pressed,
wl_pointer::ButtonState::Released => ElementState::Released,
_ => unreachable!(),
};
let button = match button {
0x110 => MouseButton::Left,
0x111 => MouseButton::Right,
0x112 => MouseButton::Middle,
// TODO figure out the translation ?
_ => return,
};
sink.send_event(
WindowEvent::MouseInput {
device_id: crate::event::DeviceId(
crate::platform_impl::DeviceId::Wayland(DeviceId),
),
state,
button,
modifiers: modifiers_tracker.lock().unwrap().clone(),
},
wid,
);
}
},
PtrEvent::Axis { axis, value, .. } => {
if let Some(wid) = mouse_focus {
if pointer.as_ref().version() < 5 {
let (mut x, mut y) = (0.0, 0.0);
// old seat compatibility
match axis {
// wayland vertical sign convention is the inverse of winit
wl_pointer::Axis::VerticalScroll => y -= value as f32,
wl_pointer::Axis::HorizontalScroll => x += value as f32,
_ => unreachable!(),
}
sink.send_event(
WindowEvent::MouseWheel {
device_id: crate::event::DeviceId(
crate::platform_impl::DeviceId::Wayland(DeviceId),
),
delta: MouseScrollDelta::PixelDelta(
(x as f64, y as f64).into(),
),
phase: TouchPhase::Moved,
modifiers: modifiers_tracker.lock().unwrap().clone(),
},
wid,
);
} else {
let (mut x, mut y) = axis_buffer.unwrap_or((0.0, 0.0));
match axis {
// wayland vertical sign convention is the inverse of winit
wl_pointer::Axis::VerticalScroll => y -= value as f32,
wl_pointer::Axis::HorizontalScroll => x += value as f32,
_ => unreachable!(),
}
axis_buffer = Some((x, y));
axis_state = match axis_state {
TouchPhase::Started | TouchPhase::Moved => TouchPhase::Moved,
_ => TouchPhase::Started,
}
}
}
},
PtrEvent::Frame => {
let axis_buffer = axis_buffer.take();
let axis_discrete_buffer = axis_discrete_buffer.take();
if let Some(wid) = mouse_focus {
if let Some((x, y)) = axis_discrete_buffer {
sink.send_event(
WindowEvent::MouseWheel {
device_id: crate::event::DeviceId(
crate::platform_impl::DeviceId::Wayland(DeviceId),
),
delta: MouseScrollDelta::LineDelta(x as f32, y as f32),
phase: axis_state,
modifiers: modifiers_tracker.lock().unwrap().clone(),
},
wid,
);
} else if let Some((x, y)) = axis_buffer {
sink.send_event(
WindowEvent::MouseWheel {
device_id: crate::event::DeviceId(
crate::platform_impl::DeviceId::Wayland(DeviceId),
),
delta: MouseScrollDelta::PixelDelta(
(x as f64, y as f64).into(),
),
phase: axis_state,
modifiers: modifiers_tracker.lock().unwrap().clone(),
},
wid,
);
}
}
},
PtrEvent::AxisSource { .. } => (),
PtrEvent::AxisStop { .. } => {
axis_state = TouchPhase::Ended;
},
PtrEvent::AxisDiscrete { axis, discrete } => {
let (mut x, mut y) = axis_discrete_buffer.unwrap_or((0, 0));
match axis {
// wayland vertical sign convention is the inverse of winit
wl_pointer::Axis::VerticalScroll => y -= discrete,
wl_pointer::Axis::HorizontalScroll => x += discrete,
_ => unreachable!(),
}
axis_discrete_buffer = Some((x, y));
axis_state = match axis_state {
TouchPhase::Started | TouchPhase::Moved => TouchPhase::Moved,
_ => TouchPhase::Started,
}
},
_ => unreachable!(),
} }
PtrEvent::AxisSource { .. } => (), },
PtrEvent::AxisStop { .. } => { (),
axis_state = TouchPhase::Ended; )
} })
PtrEvent::AxisDiscrete { axis, discrete } => { .unwrap()
let (mut x, mut y) = axis_discrete_buffer.unwrap_or((0, 0));
match axis {
// wayland vertical sign convention is the inverse of winit
wl_pointer::Axis::VerticalScroll => y -= discrete,
wl_pointer::Axis::HorizontalScroll => x += discrete,
_ => unreachable!()
}
axis_discrete_buffer = Some((x, y));
axis_state = match axis_state {
TouchPhase::Started | TouchPhase::Moved => TouchPhase::Moved,
_ => TouchPhase::Started,
}
},
_ => unreachable!()
}
}, ())
}).unwrap()
} }

View file

@ -2,12 +2,12 @@ use std::sync::{Arc, Mutex};
use crate::event::{TouchPhase, WindowEvent}; use crate::event::{TouchPhase, WindowEvent};
use super::{DeviceId, WindowId}; use super::{event_loop::WindowEventsSink, window::WindowStore, DeviceId, WindowId};
use super::event_loop::WindowEventsSink;
use super::window::WindowStore;
use smithay_client_toolkit::reexports::client::protocol::wl_touch::{Event as TouchEvent, WlTouch}; use smithay_client_toolkit::reexports::client::protocol::{
use smithay_client_toolkit::reexports::client::protocol::wl_seat; wl_seat,
wl_touch::{Event as TouchEvent, WlTouch},
};
struct TouchPoint { struct TouchPoint {
wid: WindowId, wid: WindowId,
@ -22,75 +22,89 @@ pub(crate) fn implement_touch(
) -> WlTouch { ) -> WlTouch {
let mut pending_ids = Vec::new(); let mut pending_ids = Vec::new();
seat.get_touch(|touch| { seat.get_touch(|touch| {
touch.implement_closure(move |evt, _| { touch.implement_closure(
let mut sink = sink.lock().unwrap(); move |evt, _| {
let store = store.lock().unwrap(); let mut sink = sink.lock().unwrap();
match evt { let store = store.lock().unwrap();
TouchEvent::Down { match evt {
surface, id, x, y, .. TouchEvent::Down {
} => { surface, id, x, y, ..
let wid = store.find_wid(&surface); } => {
if let Some(wid) = wid { let wid = store.find_wid(&surface);
sink.send_event( if let Some(wid) = wid {
WindowEvent::Touch(crate::event::Touch { sink.send_event(
device_id: crate::event::DeviceId(crate::platform_impl::DeviceId::Wayland(DeviceId)), WindowEvent::Touch(crate::event::Touch {
phase: TouchPhase::Started, device_id: crate::event::DeviceId(
location: (x, y).into(), crate::platform_impl::DeviceId::Wayland(DeviceId),
id: id as u64, ),
}), phase: TouchPhase::Started,
wid, location: (x, y).into(),
); id: id as u64,
pending_ids.push(TouchPoint { }),
wid: wid, wid,
location: (x, y), );
id: id, pending_ids.push(TouchPoint {
}); wid,
} location: (x, y),
id,
});
}
},
TouchEvent::Up { id, .. } => {
let idx = pending_ids.iter().position(|p| p.id == id);
if let Some(idx) = idx {
let pt = pending_ids.remove(idx);
sink.send_event(
WindowEvent::Touch(crate::event::Touch {
device_id: crate::event::DeviceId(
crate::platform_impl::DeviceId::Wayland(DeviceId),
),
phase: TouchPhase::Ended,
location: pt.location.into(),
id: id as u64,
}),
pt.wid,
);
}
},
TouchEvent::Motion { id, x, y, .. } => {
let pt = pending_ids.iter_mut().find(|p| p.id == id);
if let Some(pt) = pt {
pt.location = (x, y);
sink.send_event(
WindowEvent::Touch(crate::event::Touch {
device_id: crate::event::DeviceId(
crate::platform_impl::DeviceId::Wayland(DeviceId),
),
phase: TouchPhase::Moved,
location: (x, y).into(),
id: id as u64,
}),
pt.wid,
);
}
},
TouchEvent::Frame => (),
TouchEvent::Cancel => {
for pt in pending_ids.drain(..) {
sink.send_event(
WindowEvent::Touch(crate::event::Touch {
device_id: crate::event::DeviceId(
crate::platform_impl::DeviceId::Wayland(DeviceId),
),
phase: TouchPhase::Cancelled,
location: pt.location.into(),
id: pt.id as u64,
}),
pt.wid,
);
}
},
_ => unreachable!(),
} }
TouchEvent::Up { id, .. } => { },
let idx = pending_ids.iter().position(|p| p.id == id); (),
if let Some(idx) = idx { )
let pt = pending_ids.remove(idx); })
sink.send_event( .unwrap()
WindowEvent::Touch(crate::event::Touch {
device_id: crate::event::DeviceId(crate::platform_impl::DeviceId::Wayland(DeviceId)),
phase: TouchPhase::Ended,
location: pt.location.into(),
id: id as u64,
}),
pt.wid,
);
}
}
TouchEvent::Motion { id, x, y, .. } => {
let pt = pending_ids.iter_mut().find(|p| p.id == id);
if let Some(pt) = pt {
pt.location = (x, y);
sink.send_event(
WindowEvent::Touch(crate::event::Touch {
device_id: crate::event::DeviceId(crate::platform_impl::DeviceId::Wayland(DeviceId)),
phase: TouchPhase::Moved,
location: (x, y).into(),
id: id as u64,
}),
pt.wid,
);
}
}
TouchEvent::Frame => (),
TouchEvent::Cancel => for pt in pending_ids.drain(..) {
sink.send_event(
WindowEvent::Touch(crate::event::Touch {
device_id: crate::event::DeviceId(crate::platform_impl::DeviceId::Wayland(DeviceId)),
phase: TouchPhase::Cancelled,
location: pt.location.into(),
id: pt.id as u64,
}),
pt.wid,
);
},
_ => unreachable!()
}
}, ())
}).unwrap()
} }

View file

@ -1,18 +1,29 @@
use std::collections::VecDeque; use std::{
use std::io::{Seek, SeekFrom, Write}; collections::VecDeque,
use std::sync::{Arc, Mutex, Weak}; io::{Seek, SeekFrom, Write},
sync::{Arc, Mutex, Weak},
};
use crate::dpi::{LogicalPosition, LogicalSize}; use crate::{
use crate::error::{ExternalError, NotSupportedError, OsError as RootOsError}; dpi::{LogicalPosition, LogicalSize},
use crate::platform_impl::{MonitorHandle as PlatformMonitorHandle, PlatformSpecificWindowBuilderAttributes as PlAttributes}; error::{ExternalError, NotSupportedError, OsError as RootOsError},
use crate::monitor::MonitorHandle as RootMonitorHandle; monitor::MonitorHandle as RootMonitorHandle,
use crate::window::{WindowAttributes, CursorIcon}; platform_impl::{
MonitorHandle as PlatformMonitorHandle,
PlatformSpecificWindowBuilderAttributes as PlAttributes,
},
window::{CursorIcon, WindowAttributes},
};
use smithay_client_toolkit::surface::{get_dpi_factor, get_outputs}; use smithay_client_toolkit::{
use smithay_client_toolkit::window::{ConceptFrame, Event as WEvent, State as WState, Window as SWindow, Theme}; output::OutputMgr,
use smithay_client_toolkit::reexports::client::{Display, NewProxy}; reexports::client::{
use smithay_client_toolkit::reexports::client::protocol::{wl_seat, wl_surface, wl_subsurface, wl_shm}; protocol::{wl_seat, wl_shm, wl_subsurface, wl_surface},
use smithay_client_toolkit::output::OutputMgr; Display, NewProxy,
},
surface::{get_dpi_factor, get_outputs},
window::{ConceptFrame, Event as WEvent, State as WState, Theme, Window as SWindow},
};
use super::{make_wid, EventLoopWindowTarget, MonitorHandle, WindowId}; use super::{make_wid, EventLoopWindowTarget, MonitorHandle, WindowId};
use crate::platform_impl::platform::wayland::event_loop::{available_monitors, primary_monitor}; use crate::platform_impl::platform::wayland::event_loop::{available_monitors, primary_monitor};
@ -32,7 +43,11 @@ pub struct Window {
} }
impl Window { impl Window {
pub fn new<T>(evlp: &EventLoopWindowTarget<T>, attributes: WindowAttributes, pl_attribs: PlAttributes) -> Result<Window, RootOsError> { pub fn new<T>(
evlp: &EventLoopWindowTarget<T>,
attributes: WindowAttributes,
pl_attribs: PlAttributes,
) -> Result<Window, RootOsError> {
let (width, height) = attributes.inner_size.map(Into::into).unwrap_or((800, 600)); let (width, height) = attributes.inner_size.map(Into::into).unwrap_or((800, 600));
// Create the window // Create the window
let size = Arc::new(Mutex::new((width, height))); let size = Arc::new(Mutex::new((width, height)));
@ -71,47 +86,50 @@ impl Window {
&evlp.env, &evlp.env,
bg_surface.clone(), bg_surface.clone(),
(width, height), (width, height),
move |event| match event { move |event| {
WEvent::Configure { new_size, states } => { match event {
let mut store = window_store.lock().unwrap(); WEvent::Configure { new_size, states } => {
let is_fullscreen = states.contains(&WState::Fullscreen); let mut store = window_store.lock().unwrap();
let is_fullscreen = states.contains(&WState::Fullscreen);
for window in &mut store.windows { for window in &mut store.windows {
if window.surface.as_ref().equals(&my_surface.as_ref()) { if window.surface.as_ref().equals(&my_surface.as_ref()) {
window.newsize = new_size; window.newsize = new_size;
*(window.need_refresh.lock().unwrap()) = true; *(window.need_refresh.lock().unwrap()) = true;
*(window.fullscreen.lock().unwrap()) = is_fullscreen; *(window.fullscreen.lock().unwrap()) = is_fullscreen;
*(window.need_frame_refresh.lock().unwrap()) = true; *(window.need_frame_refresh.lock().unwrap()) = true;
if !window.configured { if !window.configured {
// this is our first configure event, display ourselves ! // this is our first configure event, display ourselves !
window.configured = true; window.configured = true;
my_bg_surface.attach(Some(&buffer), 0, 0); my_bg_surface.attach(Some(&buffer), 0, 0);
my_bg_surface.commit(); my_bg_surface.commit();
}
return;
} }
return;
} }
} },
} WEvent::Refresh => {
WEvent::Refresh => { let store = window_store.lock().unwrap();
let store = window_store.lock().unwrap(); for window in &store.windows {
for window in &store.windows { if window.surface.as_ref().equals(&my_surface.as_ref()) {
if window.surface.as_ref().equals(&my_surface.as_ref()) { *(window.need_frame_refresh.lock().unwrap()) = true;
*(window.need_frame_refresh.lock().unwrap()) = true; return;
return; }
} }
} },
} WEvent::Close => {
WEvent::Close => { let mut store = window_store.lock().unwrap();
let mut store = window_store.lock().unwrap(); for window in &mut store.windows {
for window in &mut store.windows { if window.surface.as_ref().equals(&my_surface.as_ref()) {
if window.surface.as_ref().equals(&my_surface.as_ref()) { window.closed = true;
window.closed = true; return;
return; }
} }
} },
} }
}, },
).unwrap(); )
.unwrap();
if let Some(app_id) = pl_attribs.app_id { if let Some(app_id) = pl_attribs.app_id {
frame.set_app_id(app_id); frame.set_app_id(app_id);
@ -166,11 +184,11 @@ impl Window {
Ok(Window { Ok(Window {
display: evlp.display.clone(), display: evlp.display.clone(),
_bg_surface: bg_surface, _bg_surface: bg_surface,
user_surface: user_surface, user_surface,
_user_subsurface: user_subsurface, _user_subsurface: user_subsurface,
frame: frame, frame,
outputs: evlp.env.outputs.clone(), outputs: evlp.env.outputs.clone(),
size: size, size,
kill_switch: (kill_switch, evlp.cleanup_needed.clone()), kill_switch: (kill_switch, evlp.cleanup_needed.clone()),
need_frame_refresh, need_frame_refresh,
need_refresh, need_refresh,
@ -231,12 +249,18 @@ impl Window {
#[inline] #[inline]
pub fn set_min_inner_size(&self, dimensions: Option<LogicalSize>) { pub fn set_min_inner_size(&self, dimensions: Option<LogicalSize>) {
self.frame.lock().unwrap().set_min_size(dimensions.map(Into::into)); self.frame
.lock()
.unwrap()
.set_min_size(dimensions.map(Into::into));
} }
#[inline] #[inline]
pub fn set_max_inner_size(&self, dimensions: Option<LogicalSize>) { pub fn set_max_inner_size(&self, dimensions: Option<LogicalSize>) {
self.frame.lock().unwrap().set_max_size(dimensions.map(Into::into)); self.frame
.lock()
.unwrap()
.set_max_size(dimensions.map(Into::into));
} }
#[inline] #[inline]
@ -412,7 +436,16 @@ impl WindowStore {
pub fn for_each<F>(&mut self, mut f: F) pub fn for_each<F>(&mut self, mut f: F)
where where
F: FnMut(Option<(u32, u32)>, &mut (u32, u32), Option<i32>, bool, bool, bool, WindowId, Option<&mut SWindow<ConceptFrame>>), F: FnMut(
Option<(u32, u32)>,
&mut (u32, u32),
Option<i32>,
bool,
bool,
bool,
WindowId,
Option<&mut SWindow<ConceptFrame>>,
),
{ {
for window in &mut self.windows { for window in &mut self.windows {
let opt_arc = window.frame.upgrade(); let opt_arc = window.frame.upgrade();
@ -435,4 +468,3 @@ impl WindowStore {
} }
} }
} }

View file

@ -1,8 +1,10 @@
use std::io; use std::{
use std::sync::Arc; io,
use std::path::{Path, PathBuf}; os::raw::*,
use std::str::Utf8Error; path::{Path, PathBuf},
use std::os::raw::*; str::Utf8Error,
sync::Arc,
};
use percent_encoding::percent_decode; use percent_encoding::percent_decode;
@ -127,13 +129,15 @@ impl Dnd {
DndState::Accepted => (1, self.atoms.action_private as c_long), DndState::Accepted => (1, self.atoms.action_private as c_long),
DndState::Rejected => (0, self.atoms.none as c_long), DndState::Rejected => (0, self.atoms.none as c_long),
}; };
self.xconn.send_client_msg( self.xconn
target_window, .send_client_msg(
target_window, target_window,
self.atoms.status, target_window,
None, self.atoms.status,
[this_window as c_long, accepted, 0, 0, action], None,
).flush() [this_window as c_long, accepted, 0, 0, action],
)
.flush()
} }
pub unsafe fn send_finished( pub unsafe fn send_finished(
@ -146,24 +150,23 @@ impl Dnd {
DndState::Accepted => (1, self.atoms.action_private as c_long), DndState::Accepted => (1, self.atoms.action_private as c_long),
DndState::Rejected => (0, self.atoms.none as c_long), DndState::Rejected => (0, self.atoms.none as c_long),
}; };
self.xconn.send_client_msg( self.xconn
target_window, .send_client_msg(
target_window, target_window,
self.atoms.finished, target_window,
None, self.atoms.finished,
[this_window as c_long, accepted, action, 0, 0], None,
).flush() [this_window as c_long, accepted, action, 0, 0],
)
.flush()
} }
pub unsafe fn get_type_list( pub unsafe fn get_type_list(
&self, &self,
source_window: c_ulong, source_window: c_ulong,
) -> Result<Vec<ffi::Atom>, util::GetPropertyError> { ) -> Result<Vec<ffi::Atom>, util::GetPropertyError> {
self.xconn.get_property( self.xconn
source_window, .get_property(source_window, self.atoms.type_list, ffi::XA_ATOM)
self.atoms.type_list,
ffi::XA_ATOM,
)
} }
pub unsafe fn convert_selection(&self, window: c_ulong, time: c_ulong) { pub unsafe fn convert_selection(&self, window: c_ulong, time: c_ulong) {
@ -181,11 +184,8 @@ impl Dnd {
&self, &self,
window: c_ulong, window: c_ulong,
) -> Result<Vec<c_uchar>, util::GetPropertyError> { ) -> Result<Vec<c_uchar>, util::GetPropertyError> {
self.xconn.get_property( self.xconn
window, .get_property(window, self.atoms.selection, self.atoms.uri_list)
self.atoms.selection,
self.atoms.uri_list,
)
} }
pub fn parse_data(&self, data: &mut Vec<c_uchar>) -> Result<Vec<PathBuf>, DndDataParseError> { pub fn parse_data(&self, data: &mut Vec<c_uchar>) -> Result<Vec<PathBuf>, DndDataParseError> {

View file

@ -1,20 +1,18 @@
use std::cell::RefCell; use std::{cell::RefCell, collections::HashMap, ptr, rc::Rc, slice};
use std::collections::HashMap;
use std::ptr;
use std::rc::Rc;
use std::slice;
use libc::{c_int, c_uint, c_ulong, c_char, c_long}; use libc::{c_char, c_int, c_long, c_uint, c_ulong};
use super::{ use super::{
mkdid, mkwid, get_xtarget, DeviceId, WindowId, Device, ImeReceiver, XExtension, events, ffi, get_xtarget, mkdid, mkwid, monitor, util, Device, DeviceId, DeviceInfo, Dnd,
monitor, ffi, UnownedWindow, ScrollOrientation, GenericEventCookie, DndState, GenericEventCookie, ImeReceiver, ScrollOrientation, UnownedWindow, WindowId,
events, util, DndState, Dnd, DeviceInfo XExtension,
}; };
use crate::event_loop::EventLoopWindowTarget as RootELW; use crate::{
use crate::event::{DeviceEvent, Event, KeyboardInput, ModifiersState, WindowEvent}; dpi::{LogicalPosition, LogicalSize},
use crate::dpi::{LogicalPosition,LogicalSize}; event::{DeviceEvent, Event, KeyboardInput, ModifiersState, WindowEvent},
event_loop::EventLoopWindowTarget as RootELW,
};
pub(super) struct EventProcessor<T: 'static> { pub(super) struct EventProcessor<T: 'static> {
pub(super) dnd: Dnd, pub(super) dnd: Dnd,
@ -22,7 +20,7 @@ pub(super) struct EventProcessor<T: 'static> {
pub(super) randr_event_offset: c_int, pub(super) randr_event_offset: c_int,
pub(super) devices: RefCell<HashMap<DeviceId, Device>>, pub(super) devices: RefCell<HashMap<DeviceId, Device>>,
pub(super) xi2ext: XExtension, pub(super) xi2ext: XExtension,
pub(super) target: Rc<RootELW<T>> pub(super) target: Rc<RootELW<T>>,
} }
impl<T: 'static> EventProcessor<T> { impl<T: 'static> EventProcessor<T> {
@ -37,12 +35,14 @@ impl<T: 'static> EventProcessor<T> {
} }
fn with_window<F, Ret>(&self, window_id: ffi::Window, callback: F) -> Option<Ret> fn with_window<F, Ret>(&self, window_id: ffi::Window, callback: F) -> Option<Ret>
where F: Fn(&UnownedWindow) -> Ret where
F: Fn(&UnownedWindow) -> Ret,
{ {
let mut deleted = false; let mut deleted = false;
let window_id = WindowId(window_id); let window_id = WindowId(window_id);
let wt = get_xtarget(&self.target); let wt = get_xtarget(&self.target);
let result = wt.windows let result = wt
.windows
.borrow() .borrow()
.get(&window_id) .get(&window_id)
.and_then(|window| { .and_then(|window| {
@ -62,7 +62,7 @@ impl<T: 'static> EventProcessor<T> {
self.with_window(window_id, |_| ()).is_some() self.with_window(window_id, |_| ()).is_some()
} }
pub(super) unsafe fn poll_one_event(&mut self, event_ptr : *mut ffi::XEvent) -> bool { pub(super) unsafe fn poll_one_event(&mut self, event_ptr: *mut ffi::XEvent) -> bool {
let wt = get_xtarget(&self.target); let wt = get_xtarget(&self.target);
// This function is used to poll and remove a single event // This function is used to poll and remove a single event
// from the Xlib event queue in a non-blocking, atomic way. // from the Xlib event queue in a non-blocking, atomic way.
@ -73,7 +73,8 @@ impl<T: 'static> EventProcessor<T> {
unsafe extern "C" fn predicate( unsafe extern "C" fn predicate(
_display: *mut ffi::Display, _display: *mut ffi::Display,
_event: *mut ffi::XEvent, _event: *mut ffi::XEvent,
_arg : *mut c_char) -> c_int { _arg: *mut c_char,
) -> c_int {
// This predicate always returns "true" (1) to accept all events // This predicate always returns "true" (1) to accept all events
1 1
} }
@ -82,32 +83,42 @@ impl<T: 'static> EventProcessor<T> {
wt.xconn.display, wt.xconn.display,
event_ptr, event_ptr,
Some(predicate), Some(predicate),
std::ptr::null_mut()); std::ptr::null_mut(),
);
result != 0 result != 0
} }
pub(super) fn process_event<F>(&mut self, xev: &mut ffi::XEvent, mut callback: F) pub(super) fn process_event<F>(&mut self, xev: &mut ffi::XEvent, mut callback: F)
where F: FnMut(Event<T>) where
F: FnMut(Event<T>),
{ {
let wt = get_xtarget(&self.target); let wt = get_xtarget(&self.target);
// XFilterEvent tells us when an event has been discarded by the input method. // XFilterEvent tells us when an event has been discarded by the input method.
// Specifically, this involves all of the KeyPress events in compose/pre-edit sequences, // Specifically, this involves all of the KeyPress events in compose/pre-edit sequences,
// along with an extra copy of the KeyRelease events. This also prevents backspace and // along with an extra copy of the KeyRelease events. This also prevents backspace and
// arrow keys from being detected twice. // arrow keys from being detected twice.
if ffi::True == unsafe { (wt.xconn.xlib.XFilterEvent)( if ffi::True
xev, == unsafe {
{ let xev: &ffi::XAnyEvent = xev.as_ref(); xev.window } (wt.xconn.xlib.XFilterEvent)(xev, {
) } { let xev: &ffi::XAnyEvent = xev.as_ref();
xev.window
})
}
{
return; return;
} }
let event_type = xev.get_type(); let event_type = xev.get_type();
match event_type { match event_type {
ffi::MappingNotify => { ffi::MappingNotify => {
unsafe { (wt.xconn.xlib.XRefreshKeyboardMapping)(xev.as_mut()); } unsafe {
wt.xconn.check_errors().expect("Failed to call XRefreshKeyboardMapping"); (wt.xconn.xlib.XRefreshKeyboardMapping)(xev.as_mut());
} }
wt.xconn
.check_errors()
.expect("Failed to call XRefreshKeyboardMapping");
},
ffi::ClientMessage => { ffi::ClientMessage => {
let client_msg: &ffi::XClientMessageEvent = xev.as_ref(); let client_msg: &ffi::XClientMessageEvent = xev.as_ref();
@ -116,7 +127,10 @@ impl<T: 'static> EventProcessor<T> {
let window_id = mkwid(window); let window_id = mkwid(window);
if client_msg.data.get_long(0) as ffi::Atom == wt.wm_delete_window { if client_msg.data.get_long(0) as ffi::Atom == wt.wm_delete_window {
callback(Event::WindowEvent { window_id, event: WindowEvent::CloseRequested }); callback(Event::WindowEvent {
window_id,
event: WindowEvent::CloseRequested,
});
} else if client_msg.message_type == self.dnd.atoms.enter { } else if client_msg.message_type == self.dnd.atoms.enter {
let source_window = client_msg.data.get_long(0) as c_ulong; let source_window = client_msg.data.get_long(0) as c_ulong;
let flags = client_msg.data.get_long(1); let flags = client_msg.data.get_long(1);
@ -127,10 +141,11 @@ impl<T: 'static> EventProcessor<T> {
let type_list = vec![ let type_list = vec![
client_msg.data.get_long(2) as c_ulong, client_msg.data.get_long(2) as c_ulong,
client_msg.data.get_long(3) as c_ulong, client_msg.data.get_long(3) as c_ulong,
client_msg.data.get_long(4) as c_ulong client_msg.data.get_long(4) as c_ulong,
]; ];
self.dnd.type_list = Some(type_list); self.dnd.type_list = Some(type_list);
} else if let Ok(more_types) = unsafe { self.dnd.get_type_list(source_window) } { } else if let Ok(more_types) = unsafe { self.dnd.get_type_list(source_window) }
{
self.dnd.type_list = Some(more_types); self.dnd.type_list = Some(more_types);
} }
} else if client_msg.message_type == self.dnd.atoms.position { } else if client_msg.message_type == self.dnd.atoms.position {
@ -178,18 +193,21 @@ impl<T: 'static> EventProcessor<T> {
// This results in the `SelectionNotify` event below // This results in the `SelectionNotify` event below
self.dnd.convert_selection(window, time); self.dnd.convert_selection(window, time);
} }
self.dnd.send_status(window, source_window, DndState::Accepted) self.dnd
.send_status(window, source_window, DndState::Accepted)
.expect("Failed to send `XdndStatus` message."); .expect("Failed to send `XdndStatus` message.");
} }
} else { } else {
unsafe { unsafe {
self.dnd.send_status(window, source_window, DndState::Rejected) self.dnd
.send_status(window, source_window, DndState::Rejected)
.expect("Failed to send `XdndStatus` message."); .expect("Failed to send `XdndStatus` message.");
} }
self.dnd.reset(); self.dnd.reset();
} }
} else if client_msg.message_type == self.dnd.atoms.drop { } else if client_msg.message_type == self.dnd.atoms.drop {
let (source_window, state) = if let Some(source_window) = self.dnd.source_window { let (source_window, state) = if let Some(source_window) = self.dnd.source_window
{
if let Some(Ok(ref path_list)) = self.dnd.result { if let Some(Ok(ref path_list)) = self.dnd.result {
for path in path_list { for path in path_list {
callback(Event::WindowEvent { callback(Event::WindowEvent {
@ -206,7 +224,8 @@ impl<T: 'static> EventProcessor<T> {
(source_window, DndState::Rejected) (source_window, DndState::Rejected)
}; };
unsafe { unsafe {
self.dnd.send_finished(window, source_window, state) self.dnd
.send_finished(window, source_window, state)
.expect("Failed to send `XdndFinished` message."); .expect("Failed to send `XdndFinished` message.");
} }
self.dnd.reset(); self.dnd.reset();
@ -217,7 +236,7 @@ impl<T: 'static> EventProcessor<T> {
event: WindowEvent::HoveredFileCancelled, event: WindowEvent::HoveredFileCancelled,
}); });
} }
} },
ffi::SelectionNotify => { ffi::SelectionNotify => {
let xsel: &ffi::XSelectionEvent = xev.as_ref(); let xsel: &ffi::XSelectionEvent = xev.as_ref();
@ -244,7 +263,7 @@ impl<T: 'static> EventProcessor<T> {
self.dnd.result = result; self.dnd.result = result;
} }
} },
ffi::ConfigureNotify => { ffi::ConfigureNotify => {
#[derive(Debug, Default)] #[derive(Debug, Default)]
@ -274,15 +293,22 @@ impl<T: 'static> EventProcessor<T> {
let mut shared_state_lock = window.shared_state.lock(); let mut shared_state_lock = window.shared_state.lock();
let (mut resized, moved) = { let (mut resized, moved) = {
let resized = util::maybe_change(&mut shared_state_lock.size, new_inner_size); let resized =
util::maybe_change(&mut shared_state_lock.size, new_inner_size);
let moved = if is_synthetic { let moved = if is_synthetic {
util::maybe_change(&mut shared_state_lock.inner_position, new_inner_position) util::maybe_change(
&mut shared_state_lock.inner_position,
new_inner_position,
)
} else { } else {
// Detect when frame extents change. // Detect when frame extents change.
// Since this isn't synthetic, as per the notes above, this position is relative to the // Since this isn't synthetic, as per the notes above, this position is relative to the
// parent window. // parent window.
let rel_parent = new_inner_position; let rel_parent = new_inner_position;
if util::maybe_change(&mut shared_state_lock.inner_position_rel_parent, rel_parent) { if util::maybe_change(
&mut shared_state_lock.inner_position_rel_parent,
rel_parent,
) {
// This ensures we process the next `Moved`. // This ensures we process the next `Moved`.
shared_state_lock.inner_position = None; shared_state_lock.inner_position = None;
// Extra insurance against stale frame extents. // Extra insurance against stale frame extents.
@ -297,18 +323,22 @@ impl<T: 'static> EventProcessor<T> {
let new_outer_position = if moved || shared_state_lock.position.is_none() { let new_outer_position = if moved || shared_state_lock.position.is_none() {
// We need to convert client area position to window position. // We need to convert client area position to window position.
let frame_extents = shared_state_lock.frame_extents let frame_extents = shared_state_lock
.frame_extents
.as_ref() .as_ref()
.cloned() .cloned()
.unwrap_or_else(|| { .unwrap_or_else(|| {
let frame_extents = wt.xconn.get_frame_extents_heuristic(xwindow, wt.root); let frame_extents =
wt.xconn.get_frame_extents_heuristic(xwindow, wt.root);
shared_state_lock.frame_extents = Some(frame_extents.clone()); shared_state_lock.frame_extents = Some(frame_extents.clone());
frame_extents frame_extents
}); });
let outer = frame_extents.inner_pos_to_outer(new_inner_position.0, new_inner_position.1); let outer = frame_extents
.inner_pos_to_outer(new_inner_position.0, new_inner_position.1);
shared_state_lock.position = Some(outer); shared_state_lock.position = Some(outer);
if moved { if moved {
let logical_position = LogicalPosition::from_physical(outer, monitor.hidpi_factor); let logical_position =
LogicalPosition::from_physical(outer, monitor.hidpi_factor);
events.moved = Some(WindowEvent::Moved(logical_position)); events.moved = Some(WindowEvent::Moved(logical_position));
} }
outer outer
@ -319,12 +349,13 @@ impl<T: 'static> EventProcessor<T> {
if is_synthetic { if is_synthetic {
// If we don't use the existing adjusted value when available, then the user can screw up the // If we don't use the existing adjusted value when available, then the user can screw up the
// resizing by dragging across monitors *without* dropping the window. // resizing by dragging across monitors *without* dropping the window.
let (width, height) = shared_state_lock.dpi_adjusted let (width, height) = shared_state_lock
.dpi_adjusted
.unwrap_or_else(|| (xev.width as f64, xev.height as f64)); .unwrap_or_else(|| (xev.width as f64, xev.height as f64));
let last_hidpi_factor = shared_state_lock.guessed_dpi let last_hidpi_factor =
.take() shared_state_lock.guessed_dpi.take().unwrap_or_else(|| {
.unwrap_or_else(|| { shared_state_lock
shared_state_lock.last_monitor .last_monitor
.as_ref() .as_ref()
.map(|last_monitor| last_monitor.hidpi_factor) .map(|last_monitor| last_monitor.hidpi_factor)
.unwrap_or(1.0) .unwrap_or(1.0)
@ -337,7 +368,8 @@ impl<T: 'static> EventProcessor<T> {
new_hidpi_factor new_hidpi_factor
}; };
if last_hidpi_factor != new_hidpi_factor { if last_hidpi_factor != new_hidpi_factor {
events.dpi_changed = Some(WindowEvent::HiDpiFactorChanged(new_hidpi_factor)); events.dpi_changed =
Some(WindowEvent::HiDpiFactorChanged(new_hidpi_factor));
let (new_width, new_height, flusher) = window.adjust_for_dpi( let (new_width, new_height, flusher) = window.adjust_for_dpi(
last_hidpi_factor, last_hidpi_factor,
new_hidpi_factor, new_hidpi_factor,
@ -357,7 +389,10 @@ impl<T: 'static> EventProcessor<T> {
// WMs constrain the window size, making the resize fail. This would cause an endless stream of // WMs constrain the window size, making the resize fail. This would cause an endless stream of
// XResizeWindow requests, making Xorg, the winit client, and the WM consume 100% of CPU. // XResizeWindow requests, making Xorg, the winit client, and the WM consume 100% of CPU.
if let Some(adjusted_size) = shared_state_lock.dpi_adjusted { if let Some(adjusted_size) = shared_state_lock.dpi_adjusted {
let rounded_size = (adjusted_size.0.round() as u32, adjusted_size.1.round() as u32); let rounded_size = (
adjusted_size.0.round() as u32,
adjusted_size.1.round() as u32,
);
if new_inner_size == rounded_size || !util::wm_name_is_one_of(&["Xfwm4"]) { if new_inner_size == rounded_size || !util::wm_name_is_one_of(&["Xfwm4"]) {
// When this finally happens, the event will not be synthetic. // When this finally happens, the event will not be synthetic.
shared_state_lock.dpi_adjusted = None; shared_state_lock.dpi_adjusted = None;
@ -374,7 +409,8 @@ impl<T: 'static> EventProcessor<T> {
} }
if resized { if resized {
let logical_size = LogicalSize::from_physical(new_inner_size, monitor.hidpi_factor); let logical_size =
LogicalSize::from_physical(new_inner_size, monitor.hidpi_factor);
events.resized = Some(WindowEvent::Resized(logical_size)); events.resized = Some(WindowEvent::Resized(logical_size));
} }
@ -393,7 +429,7 @@ impl<T: 'static> EventProcessor<T> {
callback(Event::WindowEvent { window_id, event }); callback(Event::WindowEvent { window_id, event });
} }
} }
} },
ffi::ReparentNotify => { ffi::ReparentNotify => {
let xev: &ffi::XReparentEvent = xev.as_ref(); let xev: &ffi::XReparentEvent = xev.as_ref();
@ -408,7 +444,7 @@ impl<T: 'static> EventProcessor<T> {
self.with_window(xev.window, |window| { self.with_window(xev.window, |window| {
window.invalidate_cached_frame_extents(); window.invalidate_cached_frame_extents();
}); });
} },
ffi::DestroyNotify => { ffi::DestroyNotify => {
let xev: &ffi::XDestroyWindowEvent = xev.as_ref(); let xev: &ffi::XDestroyWindowEvent = xev.as_ref();
@ -427,8 +463,11 @@ impl<T: 'static> EventProcessor<T> {
.remove_context(window) .remove_context(window)
.expect("Failed to destroy input context"); .expect("Failed to destroy input context");
callback(Event::WindowEvent { window_id, event: WindowEvent::Destroyed }); callback(Event::WindowEvent {
} window_id,
event: WindowEvent::Destroyed,
});
},
ffi::Expose => { ffi::Expose => {
let xev: &ffi::XExposeEvent = xev.as_ref(); let xev: &ffi::XExposeEvent = xev.as_ref();
@ -436,8 +475,11 @@ impl<T: 'static> EventProcessor<T> {
let window = xev.window; let window = xev.window;
let window_id = mkwid(window); let window_id = mkwid(window);
callback(Event::WindowEvent { window_id, event: WindowEvent::RedrawRequested }); callback(Event::WindowEvent {
} window_id,
event: WindowEvent::RedrawRequested,
});
},
ffi::KeyPress | ffi::KeyRelease => { ffi::KeyPress | ffi::KeyRelease => {
use crate::event::ElementState::{Pressed, Released}; use crate::event::ElementState::{Pressed, Released};
@ -493,7 +535,7 @@ impl<T: 'static> EventProcessor<T> {
virtual_keycode, virtual_keycode,
modifiers, modifiers,
}, },
} },
}); });
} }
@ -512,20 +554,29 @@ impl<T: 'static> EventProcessor<T> {
callback(event); callback(event);
} }
} }
} },
ffi::GenericEvent => { ffi::GenericEvent => {
let guard = if let Some(e) = GenericEventCookie::from_event(&wt.xconn, *xev) { e } else { return }; let guard = if let Some(e) = GenericEventCookie::from_event(&wt.xconn, *xev) {
e
} else {
return;
};
let xev = &guard.cookie; let xev = &guard.cookie;
if self.xi2ext.opcode != xev.extension { if self.xi2ext.opcode != xev.extension {
return; return;
} }
use crate::event::WindowEvent::{Focused, CursorEntered, MouseInput, CursorLeft, CursorMoved, MouseWheel, AxisMotion}; use crate::event::{
use crate::event::ElementState::{Pressed, Released}; ElementState::{Pressed, Released},
use crate::event::MouseButton::{Left, Right, Middle, Other}; MouseButton::{Left, Middle, Other, Right},
use crate::event::MouseScrollDelta::LineDelta; MouseScrollDelta::LineDelta,
use crate::event::{Touch, TouchPhase}; Touch, TouchPhase,
WindowEvent::{
AxisMotion, CursorEntered, CursorLeft, CursorMoved, Focused, MouseInput,
MouseWheel,
},
};
match xev.evtype { match xev.evtype {
ffi::XI_ButtonPress | ffi::XI_ButtonRelease => { ffi::XI_ButtonPress | ffi::XI_ButtonRelease => {
@ -545,66 +596,76 @@ impl<T: 'static> EventProcessor<T> {
Released Released
}; };
match xev.detail as u32 { match xev.detail as u32 {
ffi::Button1 => callback(Event::WindowEvent { ffi::Button1 => {
window_id, callback(Event::WindowEvent {
event: MouseInput { window_id,
device_id, event: MouseInput {
state, device_id,
button: Left, state,
modifiers, button: Left,
}, modifiers,
}), },
ffi::Button2 => callback(Event::WindowEvent { })
window_id, },
event: MouseInput { ffi::Button2 => {
device_id, callback(Event::WindowEvent {
state, window_id,
button: Middle, event: MouseInput {
modifiers, device_id,
}, state,
}), button: Middle,
ffi::Button3 => callback(Event::WindowEvent { modifiers,
window_id, },
event: MouseInput { })
device_id, },
state, ffi::Button3 => {
button: Right, callback(Event::WindowEvent {
modifiers, window_id,
}, event: MouseInput {
}), device_id,
state,
button: Right,
modifiers,
},
})
},
// Suppress emulated scroll wheel clicks, since we handle the real motion events for those. // Suppress emulated scroll wheel clicks, since we handle the real motion events for those.
// In practice, even clicky scroll wheels appear to be reported by evdev (and XInput2 in // In practice, even clicky scroll wheels appear to be reported by evdev (and XInput2 in
// turn) as axis motion, so we don't otherwise special-case these button presses. // turn) as axis motion, so we don't otherwise special-case these button presses.
4 | 5 | 6 | 7 => if xev.flags & ffi::XIPointerEmulated == 0 { 4 | 5 | 6 | 7 => {
callback(Event::WindowEvent { if xev.flags & ffi::XIPointerEmulated == 0 {
window_id, callback(Event::WindowEvent {
event: MouseWheel { window_id,
device_id, event: MouseWheel {
delta: match xev.detail { device_id,
4 => LineDelta(0.0, 1.0), delta: match xev.detail {
5 => LineDelta(0.0, -1.0), 4 => LineDelta(0.0, 1.0),
6 => LineDelta(-1.0, 0.0), 5 => LineDelta(0.0, -1.0),
7 => LineDelta(1.0, 0.0), 6 => LineDelta(-1.0, 0.0),
_ => unreachable!(), 7 => LineDelta(1.0, 0.0),
_ => unreachable!(),
},
phase: TouchPhase::Moved,
modifiers,
}, },
phase: TouchPhase::Moved, });
modifiers, }
},
});
}, },
x => callback(Event::WindowEvent { x => {
window_id, callback(Event::WindowEvent {
event: MouseInput { window_id,
device_id, event: MouseInput {
state, device_id,
button: Other(x as u8), state,
modifiers, button: Other(x as u8),
}, modifiers,
}), },
})
},
} }
} },
ffi::XI_Motion => { ffi::XI_Motion => {
let xev: &ffi::XIDeviceEvent = unsafe { &*(xev.data as *const _) }; let xev: &ffi::XIDeviceEvent = unsafe { &*(xev.data as *const _) };
let device_id = mkdid(xev.deviceid); let device_id = mkdid(xev.deviceid);
@ -618,9 +679,8 @@ impl<T: 'static> EventProcessor<T> {
util::maybe_change(&mut shared_state_lock.cursor_pos, new_cursor_pos) util::maybe_change(&mut shared_state_lock.cursor_pos, new_cursor_pos)
}); });
if cursor_moved == Some(true) { if cursor_moved == Some(true) {
let dpi_factor = self.with_window(xev.event, |window| { let dpi_factor =
window.hidpi_factor() self.with_window(xev.event, |window| window.hidpi_factor());
});
if let Some(dpi_factor) = dpi_factor { if let Some(dpi_factor) = dpi_factor {
let position = LogicalPosition::from_physical( let position = LogicalPosition::from_physical(
(xev.event_x as f64, xev.event_y as f64), (xev.event_x as f64, xev.event_y as f64),
@ -644,7 +704,12 @@ impl<T: 'static> EventProcessor<T> {
// More gymnastics, for self.devices // More gymnastics, for self.devices
let mut events = Vec::new(); let mut events = Vec::new();
{ {
let mask = unsafe { slice::from_raw_parts(xev.valuators.mask, xev.valuators.mask_len as usize) }; let mask = unsafe {
slice::from_raw_parts(
xev.valuators.mask,
xev.valuators.mask_len as usize,
)
};
let mut devices = self.devices.borrow_mut(); let mut devices = self.devices.borrow_mut();
let physical_device = match devices.get_mut(&DeviceId(xev.sourceid)) { let physical_device = match devices.get_mut(&DeviceId(xev.sourceid)) {
Some(device) => device, Some(device) => device,
@ -652,10 +717,14 @@ impl<T: 'static> EventProcessor<T> {
}; };
let mut value = xev.valuators.values; let mut value = xev.valuators.values;
for i in 0..xev.valuators.mask_len*8 { for i in 0..xev.valuators.mask_len * 8 {
if ffi::XIMaskIsSet(mask, i) { if ffi::XIMaskIsSet(mask, i) {
let x = unsafe { *value }; let x = unsafe { *value };
if let Some(&mut (_, ref mut info)) = physical_device.scroll_axes.iter_mut().find(|&&mut (axis, _)| axis == i) { if let Some(&mut (_, ref mut info)) = physical_device
.scroll_axes
.iter_mut()
.find(|&&mut (axis, _)| axis == i)
{
let delta = (x - info.position) / info.increment; let delta = (x - info.position) / info.increment;
info.position = x; info.position = x;
events.push(Event::WindowEvent { events.push(Event::WindowEvent {
@ -663,9 +732,13 @@ impl<T: 'static> EventProcessor<T> {
event: MouseWheel { event: MouseWheel {
device_id, device_id,
delta: match info.orientation { delta: match info.orientation {
ScrollOrientation::Horizontal => LineDelta(delta as f32, 0.0), ScrollOrientation::Horizontal => {
LineDelta(delta as f32, 0.0)
},
// X11 vertical scroll coordinates are opposite to winit's // X11 vertical scroll coordinates are opposite to winit's
ScrollOrientation::Vertical => LineDelta(0.0, -delta as f32), ScrollOrientation::Vertical => {
LineDelta(0.0, -delta as f32)
},
}, },
phase: TouchPhase::Moved, phase: TouchPhase::Moved,
modifiers, modifiers,
@ -688,7 +761,7 @@ impl<T: 'static> EventProcessor<T> {
for event in events { for event in events {
callback(event); callback(event);
} }
} },
ffi::XI_Enter => { ffi::XI_Enter => {
let xev: &ffi::XIEnterEvent = unsafe { &*(xev.data as *const _) }; let xev: &ffi::XIEnterEvent = unsafe { &*(xev.data as *const _) };
@ -704,7 +777,8 @@ impl<T: 'static> EventProcessor<T> {
// presumably some other WMs. On those, `XI_Enter` doesn't include // presumably some other WMs. On those, `XI_Enter` doesn't include
// the physical device ID, so both `sourceid` and `deviceid` are // the physical device ID, so both `sourceid` and `deviceid` are
// the virtual device. // the virtual device.
|| device_info.attachment == xev.sourceid { || device_info.attachment == xev.sourceid
{
let device_id = DeviceId(device_info.deviceid); let device_id = DeviceId(device_info.deviceid);
if let Some(device) = devices.get_mut(&device_id) { if let Some(device) = devices.get_mut(&device_id) {
device.reset_scroll_position(device_info); device.reset_scroll_position(device_info);
@ -717,9 +791,9 @@ impl<T: 'static> EventProcessor<T> {
event: CursorEntered { device_id }, event: CursorEntered { device_id },
}); });
if let Some(dpi_factor) = self.with_window(xev.event, |window| { if let Some(dpi_factor) =
window.hidpi_factor() self.with_window(xev.event, |window| window.hidpi_factor())
}) { {
let position = LogicalPosition::from_physical( let position = LogicalPosition::from_physical(
(xev.event_x as f64, xev.event_y as f64), (xev.event_x as f64, xev.event_y as f64),
dpi_factor, dpi_factor,
@ -732,7 +806,8 @@ impl<T: 'static> EventProcessor<T> {
// This needs to only be done after confirming the window still exists, // This needs to only be done after confirming the window still exists,
// since otherwise we risk getting a `BadWindow` error if the window was // since otherwise we risk getting a `BadWindow` error if the window was
// dropped with queued events. // dropped with queued events.
let modifiers = wt.xconn let modifiers = wt
.xconn
.query_pointer(xev.event, xev.deviceid) .query_pointer(xev.event, xev.deviceid)
.expect("Failed to query pointer device") .expect("Failed to query pointer device")
.get_modifier_state(); .get_modifier_state();
@ -746,7 +821,7 @@ impl<T: 'static> EventProcessor<T> {
}, },
}); });
} }
} },
ffi::XI_Leave => { ffi::XI_Leave => {
let xev: &ffi::XILeaveEvent = unsafe { &*(xev.data as *const _) }; let xev: &ffi::XILeaveEvent = unsafe { &*(xev.data as *const _) };
@ -756,19 +831,20 @@ impl<T: 'static> EventProcessor<T> {
if !window_closed { if !window_closed {
callback(Event::WindowEvent { callback(Event::WindowEvent {
window_id: mkwid(xev.event), window_id: mkwid(xev.event),
event: CursorLeft { device_id: mkdid(xev.deviceid) }, event: CursorLeft {
device_id: mkdid(xev.deviceid),
},
}); });
} }
} },
ffi::XI_FocusIn => { ffi::XI_FocusIn => {
let xev: &ffi::XIFocusInEvent = unsafe { &*(xev.data as *const _) }; let xev: &ffi::XIFocusInEvent = unsafe { &*(xev.data as *const _) };
let dpi_factor = match self.with_window(xev.event, |window| { let dpi_factor =
window.hidpi_factor() match self.with_window(xev.event, |window| window.hidpi_factor()) {
}) { Some(dpi_factor) => dpi_factor,
Some(dpi_factor) => dpi_factor, None => return,
None => return, };
};
let window_id = mkwid(xev.event); let window_id = mkwid(xev.event);
wt.ime wt.ime
@ -776,11 +852,15 @@ impl<T: 'static> EventProcessor<T> {
.focus(xev.event) .focus(xev.event)
.expect("Failed to focus input context"); .expect("Failed to focus input context");
callback(Event::WindowEvent { window_id, event: Focused(true) }); callback(Event::WindowEvent {
window_id,
event: Focused(true),
});
// The deviceid for this event is for a keyboard instead of a pointer, // The deviceid for this event is for a keyboard instead of a pointer,
// so we have to do a little extra work. // so we have to do a little extra work.
let pointer_id = self.devices let pointer_id = self
.devices
.borrow() .borrow()
.get(&DeviceId(xev.deviceid)) .get(&DeviceId(xev.deviceid))
.map(|device| device.attachment) .map(|device| device.attachment)
@ -796,12 +876,14 @@ impl<T: 'static> EventProcessor<T> {
device_id: mkdid(pointer_id), device_id: mkdid(pointer_id),
position, position,
modifiers: ModifiersState::from(xev.mods), modifiers: ModifiersState::from(xev.mods),
} },
}); });
} },
ffi::XI_FocusOut => { ffi::XI_FocusOut => {
let xev: &ffi::XIFocusOutEvent = unsafe { &*(xev.data as *const _) }; let xev: &ffi::XIFocusOutEvent = unsafe { &*(xev.data as *const _) };
if !self.window_exists(xev.event) { return; } if !self.window_exists(xev.event) {
return;
}
wt.ime wt.ime
.borrow_mut() .borrow_mut()
.unfocus(xev.event) .unfocus(xev.event)
@ -810,7 +892,7 @@ impl<T: 'static> EventProcessor<T> {
window_id: mkwid(xev.event), window_id: mkwid(xev.event),
event: Focused(false), event: Focused(false),
}) })
} },
ffi::XI_TouchBegin | ffi::XI_TouchUpdate | ffi::XI_TouchEnd => { ffi::XI_TouchBegin | ffi::XI_TouchUpdate | ffi::XI_TouchEnd => {
let xev: &ffi::XIDeviceEvent = unsafe { &*(xev.data as *const _) }; let xev: &ffi::XIDeviceEvent = unsafe { &*(xev.data as *const _) };
@ -819,11 +901,10 @@ impl<T: 'static> EventProcessor<T> {
ffi::XI_TouchBegin => TouchPhase::Started, ffi::XI_TouchBegin => TouchPhase::Started,
ffi::XI_TouchUpdate => TouchPhase::Moved, ffi::XI_TouchUpdate => TouchPhase::Moved,
ffi::XI_TouchEnd => TouchPhase::Ended, ffi::XI_TouchEnd => TouchPhase::Ended,
_ => unreachable!() _ => unreachable!(),
}; };
let dpi_factor = self.with_window(xev.event, |window| { let dpi_factor =
window.hidpi_factor() self.with_window(xev.event, |window| window.hidpi_factor());
});
if let Some(dpi_factor) = dpi_factor { if let Some(dpi_factor) = dpi_factor {
let location = LogicalPosition::from_physical( let location = LogicalPosition::from_physical(
(xev.event_x as f64, xev.event_y as f64), (xev.event_x as f64, xev.event_y as f64),
@ -839,31 +920,39 @@ impl<T: 'static> EventProcessor<T> {
}), }),
}) })
} }
} },
ffi::XI_RawButtonPress | ffi::XI_RawButtonRelease => { ffi::XI_RawButtonPress | ffi::XI_RawButtonRelease => {
let xev: &ffi::XIRawEvent = unsafe { &*(xev.data as *const _) }; let xev: &ffi::XIRawEvent = unsafe { &*(xev.data as *const _) };
if xev.flags & ffi::XIPointerEmulated == 0 { if xev.flags & ffi::XIPointerEmulated == 0 {
callback(Event::DeviceEvent { device_id: mkdid(xev.deviceid), event: DeviceEvent::Button { callback(Event::DeviceEvent {
button: xev.detail as u32, device_id: mkdid(xev.deviceid),
state: match xev.evtype { event: DeviceEvent::Button {
ffi::XI_RawButtonPress => Pressed, button: xev.detail as u32,
ffi::XI_RawButtonRelease => Released, state: match xev.evtype {
_ => unreachable!(), ffi::XI_RawButtonPress => Pressed,
ffi::XI_RawButtonRelease => Released,
_ => unreachable!(),
},
}, },
}}); });
} }
} },
ffi::XI_RawMotion => { ffi::XI_RawMotion => {
let xev: &ffi::XIRawEvent = unsafe { &*(xev.data as *const _) }; let xev: &ffi::XIRawEvent = unsafe { &*(xev.data as *const _) };
let did = mkdid(xev.deviceid); let did = mkdid(xev.deviceid);
let mask = unsafe { slice::from_raw_parts(xev.valuators.mask, xev.valuators.mask_len as usize) }; let mask = unsafe {
slice::from_raw_parts(
xev.valuators.mask,
xev.valuators.mask_len as usize,
)
};
let mut value = xev.raw_values; let mut value = xev.raw_values;
let mut mouse_delta = (0.0, 0.0); let mut mouse_delta = (0.0, 0.0);
let mut scroll_delta = (0.0, 0.0); let mut scroll_delta = (0.0, 0.0);
for i in 0..xev.valuators.mask_len*8 { for i in 0..xev.valuators.mask_len * 8 {
if ffi::XIMaskIsSet(mask, i) { if ffi::XIMaskIsSet(mask, i) {
let x = unsafe { *value }; let x = unsafe { *value };
// We assume that every XInput2 device with analog axes is a pointing device emitting // We assume that every XInput2 device with analog axes is a pointing device emitting
@ -875,24 +964,31 @@ impl<T: 'static> EventProcessor<T> {
3 => scroll_delta.1 = x as f32, 3 => scroll_delta.1 = x as f32,
_ => {}, _ => {},
} }
callback(Event::DeviceEvent { device_id: did, event: DeviceEvent::Motion { callback(Event::DeviceEvent {
axis: i as u32, device_id: did,
value: x, event: DeviceEvent::Motion {
}}); axis: i as u32,
value: x,
},
});
value = unsafe { value.offset(1) }; value = unsafe { value.offset(1) };
} }
} }
if mouse_delta != (0.0, 0.0) { if mouse_delta != (0.0, 0.0) {
callback(Event::DeviceEvent { device_id: did, event: DeviceEvent::MouseMotion { callback(Event::DeviceEvent {
delta: mouse_delta, device_id: did,
}}); event: DeviceEvent::MouseMotion { delta: mouse_delta },
});
} }
if scroll_delta != (0.0, 0.0) { if scroll_delta != (0.0, 0.0) {
callback(Event::DeviceEvent { device_id: did, event: DeviceEvent::MouseWheel { callback(Event::DeviceEvent {
delta: LineDelta(scroll_delta.0, scroll_delta.1), device_id: did,
}}); event: DeviceEvent::MouseWheel {
delta: LineDelta(scroll_delta.0, scroll_delta.1),
},
});
} }
} },
ffi::XI_RawKeyPress | ffi::XI_RawKeyRelease => { ffi::XI_RawKeyPress | ffi::XI_RawKeyRelease => {
let xev: &ffi::XIRawEvent = unsafe { &*(xev.data as *const _) }; let xev: &ffi::XIRawEvent = unsafe { &*(xev.data as *const _) };
@ -905,7 +1001,9 @@ impl<T: 'static> EventProcessor<T> {
let device_id = xev.sourceid; let device_id = xev.sourceid;
let keycode = xev.detail; let keycode = xev.detail;
if keycode < 8 { return; } if keycode < 8 {
return;
}
let scancode = (keycode - 8) as u32; let scancode = (keycode - 8) as u32;
let keysym = unsafe { let keysym = unsafe {
@ -915,7 +1013,9 @@ impl<T: 'static> EventProcessor<T> {
0, 0,
) )
}; };
wt.xconn.check_errors().expect("Failed to lookup raw keysym"); wt.xconn
.check_errors()
.expect("Failed to lookup raw keysym");
let virtual_keycode = events::keysym_to_element(keysym as c_uint); let virtual_keycode = events::keysym_to_element(keysym as c_uint);
@ -933,23 +1033,32 @@ impl<T: 'static> EventProcessor<T> {
modifiers: ModifiersState::default(), modifiers: ModifiersState::default(),
}), }),
}); });
} },
ffi::XI_HierarchyChanged => { ffi::XI_HierarchyChanged => {
let xev: &ffi::XIHierarchyEvent = unsafe { &*(xev.data as *const _) }; let xev: &ffi::XIHierarchyEvent = unsafe { &*(xev.data as *const _) };
for info in unsafe { slice::from_raw_parts(xev.info, xev.num_info as usize) } { for info in
unsafe { slice::from_raw_parts(xev.info, xev.num_info as usize) }
{
if 0 != info.flags & (ffi::XISlaveAdded | ffi::XIMasterAdded) { if 0 != info.flags & (ffi::XISlaveAdded | ffi::XIMasterAdded) {
self.init_device(info.deviceid); self.init_device(info.deviceid);
callback(Event::DeviceEvent { device_id: mkdid(info.deviceid), event: DeviceEvent::Added }); callback(Event::DeviceEvent {
} else if 0 != info.flags & (ffi::XISlaveRemoved | ffi::XIMasterRemoved) { device_id: mkdid(info.deviceid),
callback(Event::DeviceEvent { device_id: mkdid(info.deviceid), event: DeviceEvent::Removed }); event: DeviceEvent::Added,
});
} else if 0 != info.flags & (ffi::XISlaveRemoved | ffi::XIMasterRemoved)
{
callback(Event::DeviceEvent {
device_id: mkdid(info.deviceid),
event: DeviceEvent::Removed,
});
let mut devices = self.devices.borrow_mut(); let mut devices = self.devices.borrow_mut();
devices.remove(&DeviceId(info.deviceid)); devices.remove(&DeviceId(info.deviceid));
} }
} }
} },
_ => {} _ => {},
} }
}, },
_ => { _ => {
@ -972,10 +1081,11 @@ impl<T: 'static> EventProcessor<T> {
callback(Event::WindowEvent { callback(Event::WindowEvent {
window_id: mkwid(window_id.0), window_id: mkwid(window_id.0),
event: WindowEvent::HiDpiFactorChanged( event: WindowEvent::HiDpiFactorChanged(
new_monitor.hidpi_factor new_monitor.hidpi_factor,
), ),
}); });
let (width, height) = window.inner_size_physical(); let (width, height) =
window.inner_size_physical();
let (_, _, flusher) = window.adjust_for_dpi( let (_, _, flusher) = window.adjust_for_dpi(
prev_monitor.hidpi_factor, prev_monitor.hidpi_factor,
new_monitor.hidpi_factor, new_monitor.hidpi_factor,

View file

@ -1,6 +1,6 @@
use libc;
use super::ffi; use super::ffi;
use crate::event::VirtualKeyCode; use crate::event::VirtualKeyCode;
use libc;
pub fn keysym_to_element(keysym: libc::c_uint) -> Option<VirtualKeyCode> { pub fn keysym_to_element(keysym: libc::c_uint) -> Option<VirtualKeyCode> {
Some(match keysym { Some(match keysym {
@ -1003,6 +1003,6 @@ pub fn keysym_to_element(keysym: libc::c_uint) -> Option<VirtualKeyCode> {
ffi::XF86XK_Copy => VirtualKeyCode::Copy, ffi::XF86XK_Copy => VirtualKeyCode::Copy,
ffi::XF86XK_Paste => VirtualKeyCode::Paste, ffi::XF86XK_Paste => VirtualKeyCode::Paste,
ffi::XF86XK_Cut => VirtualKeyCode::Cut, ffi::XF86XK_Cut => VirtualKeyCode::Cut,
_ => return None _ => return None,
}) })
} }

View file

@ -1,9 +1,4 @@
pub use x11_dl::keysym::*; pub use x11_dl::{
pub use x11_dl::xcursor::*; error::OpenError, keysym::*, xcursor::*, xinput::*, xinput2::*, xlib::*, xlib_xcb::*,
pub use x11_dl::xlib::*; xrandr::*, xrender::*,
pub use x11_dl::xinput::*; };
pub use x11_dl::xinput2::*;
pub use x11_dl::xlib_xcb::*;
pub use x11_dl::error::OpenError;
pub use x11_dl::xrandr::*;
pub use x11_dl::xrender::*;

View file

@ -1,13 +1,12 @@
use std::ptr; use std::{collections::HashMap, os::raw::c_char, ptr, sync::Arc};
use std::sync::Arc;
use std::collections::HashMap;
use std::os::raw::c_char;
use super::{ffi, XConnection, XError}; use super::{ffi, XConnection, XError};
use super::inner::{close_im, ImeInner}; use super::{
use super::input_method::PotentialInputMethods; context::{ImeContext, ImeContextCreationError},
use super::context::{ImeContextCreationError, ImeContext}; inner::{close_im, ImeInner},
input_method::PotentialInputMethods,
};
pub unsafe fn xim_set_callback( pub unsafe fn xim_set_callback(
xconn: &Arc<XConnection>, xconn: &Arc<XConnection>,
@ -17,12 +16,7 @@ pub unsafe fn xim_set_callback(
) -> Result<(), XError> { ) -> Result<(), XError> {
// It's advisable to wrap variadic FFI functions in our own functions, as we want to minimize // It's advisable to wrap variadic FFI functions in our own functions, as we want to minimize
// access that isn't type-checked. // access that isn't type-checked.
(xconn.xlib.XSetIMValues)( (xconn.xlib.XSetIMValues)(xim, field, callback, ptr::null_mut::<()>());
xim,
field,
callback,
ptr::null_mut::<()>(),
);
xconn.check_errors() xconn.check_errors()
} }
@ -107,18 +101,14 @@ unsafe fn replace_im(inner: *mut ImeInner) -> Result<(), ReplaceImError> {
let _ = close_im(xconn, new_im.im); let _ = close_im(xconn, new_im.im);
} }
result result
}.map_err(ReplaceImError::SetDestroyCallbackFailed)?; }
.map_err(ReplaceImError::SetDestroyCallbackFailed)?;
let mut new_contexts = HashMap::new(); let mut new_contexts = HashMap::new();
for (window, old_context) in (*inner).contexts.iter() { for (window, old_context) in (*inner).contexts.iter() {
let spot = old_context.as_ref().map(|old_context| old_context.ic_spot); let spot = old_context.as_ref().map(|old_context| old_context.ic_spot);
let new_context = { let new_context = {
let result = ImeContext::new( let result = ImeContext::new(xconn, new_im.im, *window, spot);
xconn,
new_im.im,
*window,
spot,
);
if result.is_err() { if result.is_err() {
let _ = close_im(xconn, new_im.im); let _ = close_im(xconn, new_im.im);
} }
@ -137,7 +127,7 @@ unsafe fn replace_im(inner: *mut ImeInner) -> Result<(), ReplaceImError> {
Ok(()) Ok(())
} }
pub unsafe extern fn xim_instantiate_callback( pub unsafe extern "C" fn xim_instantiate_callback(
_display: *mut ffi::Display, _display: *mut ffi::Display,
client_data: ffi::XPointer, client_data: ffi::XPointer,
// This field is unsupplied. // This field is unsupplied.
@ -160,7 +150,7 @@ pub unsafe extern fn xim_instantiate_callback(
// This callback is triggered when the input method is closed on the server end. When this // This callback is triggered when the input method is closed on the server end. When this
// happens, XCloseIM/XDestroyIC doesn't need to be called, as the resources have already been // happens, XCloseIM/XDestroyIC doesn't need to be called, as the resources have already been
// free'd (attempting to do so causes our connection to freeze). // free'd (attempting to do so causes our connection to freeze).
pub unsafe extern fn xim_destroy_callback( pub unsafe extern "C" fn xim_destroy_callback(
_xim: ffi::XIM, _xim: ffi::XIM,
client_data: ffi::XPointer, client_data: ffi::XPointer,
// This field is unsupplied. // This field is unsupplied.

View file

@ -1,6 +1,8 @@
use std::ptr; use std::{
use std::sync::Arc; os::raw::{c_short, c_void},
use std::os::raw::{c_short, c_void}; ptr,
sync::Arc,
};
use super::{ffi, util, XConnection, XError}; use super::{ffi, util, XConnection, XError};
@ -22,7 +24,8 @@ unsafe fn create_pre_edit_attr<'a>(
ic_spot, ic_spot,
ptr::null_mut::<()>(), ptr::null_mut::<()>(),
), ),
).expect("XVaCreateNestedList returned NULL") )
.expect("XVaCreateNestedList returned NULL")
} }
// WARNING: this struct doesn't destroy its XIC resource when dropped. // WARNING: this struct doesn't destroy its XIC resource when dropped.
@ -49,7 +52,9 @@ impl ImeContext {
}; };
let ic = ic.ok_or(ImeContextCreationError::Null)?; let ic = ic.ok_or(ImeContextCreationError::Null)?;
xconn.check_errors().map_err(ImeContextCreationError::XError)?; xconn
.check_errors()
.map_err(ImeContextCreationError::XError)?;
Ok(ImeContext { Ok(ImeContext {
ic, ic,

View file

@ -1,12 +1,8 @@
use std::mem; use std::{collections::HashMap, mem, ptr, sync::Arc};
use std::ptr;
use std::sync::Arc;
use std::collections::HashMap;
use super::{ffi, XConnection, XError}; use super::{ffi, XConnection, XError};
use super::input_method::PotentialInputMethods; use super::{context::ImeContext, input_method::PotentialInputMethods};
use super::context::ImeContext;
pub unsafe fn close_im(xconn: &Arc<XConnection>, im: ffi::XIM) -> Result<(), XError> { pub unsafe fn close_im(xconn: &Arc<XConnection>, im: ffi::XIM) -> Result<(), XError> {
(xconn.xlib.XCloseIM)(im); (xconn.xlib.XCloseIM)(im);
@ -33,10 +29,7 @@ pub struct ImeInner {
} }
impl ImeInner { impl ImeInner {
pub fn new( pub fn new(xconn: Arc<XConnection>, potential_input_methods: PotentialInputMethods) -> Self {
xconn: Arc<XConnection>,
potential_input_methods: PotentialInputMethods,
) -> Self {
ImeInner { ImeInner {
xconn, xconn,
im: ptr::null_mut(), im: ptr::null_mut(),

View file

@ -1,9 +1,11 @@
use std::env; use std::{
use std::fmt; env,
use std::ptr; ffi::{CStr, CString, IntoStringError},
use std::sync::Arc; fmt,
use std::os::raw::c_char; os::raw::c_char,
use std::ffi::{CStr, CString, IntoStringError}; ptr,
sync::Arc,
};
use parking_lot::Mutex; use parking_lot::Mutex;
@ -13,10 +15,7 @@ lazy_static! {
static ref GLOBAL_LOCK: Mutex<()> = Default::default(); static ref GLOBAL_LOCK: Mutex<()> = Default::default();
} }
unsafe fn open_im( unsafe fn open_im(xconn: &Arc<XConnection>, locale_modifiers: &CStr) -> Option<ffi::XIM> {
xconn: &Arc<XConnection>,
locale_modifiers: &CStr,
) -> Option<ffi::XIM> {
let _lock = GLOBAL_LOCK.lock(); let _lock = GLOBAL_LOCK.lock();
// XSetLocaleModifiers returns... // XSetLocaleModifiers returns...
@ -97,11 +96,9 @@ unsafe fn get_xim_servers(xconn: &Arc<XConnection>) -> Result<Vec<String>, GetXi
let root = (xconn.xlib.XDefaultRootWindow)(xconn.display); let root = (xconn.xlib.XDefaultRootWindow)(xconn.display);
let mut atoms: Vec<ffi::Atom> = xconn.get_property( let mut atoms: Vec<ffi::Atom> = xconn
root, .get_property(root, servers_atom, ffi::XA_ATOM)
servers_atom, .map_err(GetXimServersError::GetPropertyError)?;
ffi::XA_ATOM,
).map_err(GetXimServersError::GetPropertyError)?;
let mut names: Vec<*const c_char> = Vec::with_capacity(atoms.len()); let mut names: Vec<*const c_char> = Vec::with_capacity(atoms.len());
(xconn.xlib.XGetAtomNames)( (xconn.xlib.XGetAtomNames)(
@ -135,15 +132,12 @@ impl InputMethodName {
pub fn from_string(string: String) -> Self { pub fn from_string(string: String) -> Self {
let c_string = CString::new(string.clone()) let c_string = CString::new(string.clone())
.expect("String used to construct CString contained null byte"); .expect("String used to construct CString contained null byte");
InputMethodName { InputMethodName { c_string, string }
c_string,
string,
}
} }
pub fn from_str(string: &str) -> Self { pub fn from_str(string: &str) -> Self {
let c_string = CString::new(string) let c_string =
.expect("String used to construct CString contained null byte"); CString::new(string).expect("String used to construct CString contained null byte");
InputMethodName { InputMethodName {
c_string, c_string,
string: string.to_owned(), string: string.to_owned(),

View file

@ -1,20 +1,24 @@
// Important: all XIM calls need to happen from the same thread! // Important: all XIM calls need to happen from the same thread!
mod callbacks;
mod context;
mod inner; mod inner;
mod input_method; mod input_method;
mod context;
mod callbacks;
use std::sync::Arc; use std::sync::{
use std::sync::mpsc::{Receiver, Sender}; mpsc::{Receiver, Sender},
Arc,
};
use super::{ffi, util, XConnection, XError}; use super::{ffi, util, XConnection, XError};
use self::inner::{close_im, ImeInner};
use self::input_method::PotentialInputMethods;
use self::context::ImeContext;
use self::callbacks::*;
pub use self::context::ImeContextCreationError; pub use self::context::ImeContextCreationError;
use self::{
callbacks::*,
context::ImeContext,
inner::{close_im, ImeInner},
input_method::PotentialInputMethods,
};
pub type ImeReceiver = Receiver<(ffi::Window, i16, i16)>; pub type ImeReceiver = Receiver<(ffi::Window, i16, i16)>;
pub type ImeSender = Sender<(ffi::Window, i16, i16)>; pub type ImeSender = Sender<(ffi::Window, i16, i16)>;
@ -37,10 +41,7 @@ impl Ime {
let potential_input_methods = PotentialInputMethods::new(&xconn); let potential_input_methods = PotentialInputMethods::new(&xconn);
let (mut inner, client_data) = { let (mut inner, client_data) = {
let mut inner = Box::new(ImeInner::new( let mut inner = Box::new(ImeInner::new(xconn, potential_input_methods));
xconn,
potential_input_methods,
));
let inner_ptr = Box::into_raw(inner); let inner_ptr = Box::into_raw(inner);
let client_data = inner_ptr as _; let client_data = inner_ptr as _;
let destroy_callback = ffi::XIMCallback { let destroy_callback = ffi::XIMCallback {
@ -54,9 +55,12 @@ impl Ime {
let xconn = Arc::clone(&inner.xconn); let xconn = Arc::clone(&inner.xconn);
let input_method = inner.potential_input_methods.open_im(&xconn, Some(&|| { let input_method = inner.potential_input_methods.open_im(
let _ = unsafe { set_instantiate_callback(&xconn, client_data) }; &xconn,
})); Some(&|| {
let _ = unsafe { set_instantiate_callback(&xconn, client_data) };
}),
);
let is_fallback = input_method.is_fallback(); let is_fallback = input_method.is_fallback();
if let Some(input_method) = input_method.ok() { if let Some(input_method) = input_method.ok() {
@ -84,19 +88,12 @@ impl Ime {
// Ok(_) indicates that nothing went wrong internally // Ok(_) indicates that nothing went wrong internally
// Ok(true) indicates that the action was actually performed // Ok(true) indicates that the action was actually performed
// Ok(false) indicates that the action is not presently applicable // Ok(false) indicates that the action is not presently applicable
pub fn create_context(&mut self, window: ffi::Window) pub fn create_context(&mut self, window: ffi::Window) -> Result<bool, ImeContextCreationError> {
-> Result<bool, ImeContextCreationError>
{
let context = if self.is_destroyed() { let context = if self.is_destroyed() {
// Create empty entry in map, so that when IME is rebuilt, this window has a context. // Create empty entry in map, so that when IME is rebuilt, this window has a context.
None None
} else { } else {
Some(unsafe { ImeContext::new( Some(unsafe { ImeContext::new(&self.inner.xconn, self.inner.im, window, None) }?)
&self.inner.xconn,
self.inner.im,
window,
None,
) }?)
}; };
self.inner.contexts.insert(window, context); self.inner.contexts.insert(window, context);
Ok(!self.is_destroyed()) Ok(!self.is_destroyed())

View file

@ -1,39 +1,47 @@
#![cfg(any(target_os = "linux", target_os = "dragonfly", target_os = "freebsd", target_os = "netbsd", target_os = "openbsd"))] #![cfg(any(target_os = "linux", target_os = "dragonfly", target_os = "freebsd", target_os = "netbsd", target_os = "openbsd"))]
pub mod ffi; mod dnd;
mod event_processor;
mod events; mod events;
pub mod ffi;
mod ime;
mod monitor; mod monitor;
pub mod util;
mod window; mod window;
mod xdisplay; mod xdisplay;
mod dnd;
mod ime;
pub mod util;
mod event_processor;
pub use self::monitor::MonitorHandle; pub use self::{
pub use self::window::UnownedWindow; monitor::MonitorHandle,
pub use self::xdisplay::{XConnection, XNotSupported, XError}; window::UnownedWindow,
xdisplay::{XConnection, XError, XNotSupported},
};
use std::{mem, slice}; use std::{
use std::cell::RefCell; cell::RefCell,
use std::collections::{VecDeque, HashMap, HashSet}; collections::{HashMap, HashSet, VecDeque},
use std::ffi::CStr; ffi::CStr,
use std::ops::Deref; mem,
use std::os::raw::*; ops::Deref,
use std::rc::Rc; os::raw::*,
use std::sync::{Arc, mpsc, Weak, Mutex}; rc::Rc,
slice,
sync::{mpsc, Arc, Mutex, Weak},
};
use libc::{self, setlocale, LC_CTYPE}; use libc::{self, setlocale, LC_CTYPE};
use crate::error::OsError as RootOsError; use self::{
use crate::event_loop::{ControlFlow, EventLoopClosed, EventLoopWindowTarget as RootELW}; dnd::{Dnd, DndState},
use crate::event::{WindowEvent, Event}; event_processor::EventProcessor,
use crate::platform_impl::PlatformSpecificWindowBuilderAttributes; ime::{Ime, ImeCreationError, ImeReceiver, ImeSender},
use crate::platform_impl::platform::sticky_exit_callback; };
use crate::window::{WindowAttributes}; use crate::{
use self::dnd::{Dnd, DndState}; error::OsError as RootOsError,
use self::ime::{ImeReceiver, ImeSender, ImeCreationError, Ime}; event::{Event, WindowEvent},
use self::event_processor::EventProcessor; event_loop::{ControlFlow, EventLoopClosed, EventLoopWindowTarget as RootELW},
platform_impl::{platform::sticky_exit_callback, PlatformSpecificWindowBuilderAttributes},
window::WindowAttributes,
};
pub struct EventLoopWindowTarget<T> { pub struct EventLoopWindowTarget<T> {
xconn: Arc<XConnection>, xconn: Arc<XConnection>,
@ -43,7 +51,7 @@ pub struct EventLoopWindowTarget<T> {
ime: RefCell<Ime>, ime: RefCell<Ime>,
windows: RefCell<HashMap<WindowId, Weak<UnownedWindow>>>, windows: RefCell<HashMap<WindowId, Weak<UnownedWindow>>>,
pending_redraws: Arc<Mutex<HashSet<WindowId>>>, pending_redraws: Arc<Mutex<HashSet<WindowId>>>,
_marker: ::std::marker::PhantomData<T> _marker: ::std::marker::PhantomData<T>,
} }
pub struct EventLoop<T: 'static> { pub struct EventLoop<T: 'static> {
@ -53,7 +61,7 @@ pub struct EventLoop<T: 'static> {
pending_user_events: Rc<RefCell<VecDeque<T>>>, pending_user_events: Rc<RefCell<VecDeque<T>>>,
user_sender: ::calloop::channel::Sender<T>, user_sender: ::calloop::channel::Sender<T>,
pending_events: Rc<RefCell<VecDeque<Event<T>>>>, pending_events: Rc<RefCell<VecDeque<Event<T>>>>,
target: Rc<RootELW<T>> target: Rc<RootELW<T>>,
} }
#[derive(Clone)] #[derive(Clone)]
@ -73,7 +81,9 @@ impl<T: 'static> EventLoop<T> {
let (ime_sender, ime_receiver) = mpsc::channel(); let (ime_sender, ime_receiver) = mpsc::channel();
// Input methods will open successfully without setting the locale, but it won't be // Input methods will open successfully without setting the locale, but it won't be
// possible to actually commit pre-edit sequences. // possible to actually commit pre-edit sequences.
unsafe { setlocale(LC_CTYPE, b"\0".as_ptr() as *const _); } unsafe {
setlocale(LC_CTYPE, b"\0".as_ptr() as *const _);
}
let ime = RefCell::new({ let ime = RefCell::new({
let result = Ime::new(Arc::clone(&xconn)); let result = Ime::new(Arc::clone(&xconn));
if let Err(ImeCreationError::OpenFailure(ref state)) = result { if let Err(ImeCreationError::OpenFailure(ref state)) = result {
@ -82,7 +92,8 @@ impl<T: 'static> EventLoop<T> {
result.expect("Failed to set input method destruction callback") result.expect("Failed to set input method destruction callback")
}); });
let randr_event_offset = xconn.select_xrandr_input(root) let randr_event_offset = xconn
.select_xrandr_input(root)
.expect("Failed to query XRandR extension"); .expect("Failed to query XRandR extension");
let xi2ext = unsafe { let xi2ext = unsafe {
@ -96,7 +107,8 @@ impl<T: 'static> EventLoop<T> {
b"XInputExtension\0".as_ptr() as *const c_char, b"XInputExtension\0".as_ptr() as *const c_char,
&mut result.opcode as *mut c_int, &mut result.opcode as *mut c_int,
&mut result.first_event_id as *mut c_int, &mut result.first_event_id as *mut c_int,
&mut result.first_error_id as *mut c_int); &mut result.first_error_id as *mut c_int,
);
if res == ffi::False { if res == ffi::False {
panic!("X server missing XInput extension"); panic!("X server missing XInput extension");
} }
@ -110,18 +122,18 @@ impl<T: 'static> EventLoop<T> {
xconn.display, xconn.display,
&mut xinput_major_ver, &mut xinput_major_ver,
&mut xinput_minor_ver, &mut xinput_minor_ver,
) != ffi::Success as libc::c_int { ) != ffi::Success as libc::c_int
{
panic!( panic!(
"X server has XInput extension {}.{} but does not support XInput2", "X server has XInput extension {}.{} but does not support XInput2",
xinput_major_ver, xinput_major_ver, xinput_minor_ver,
xinput_minor_ver,
); );
} }
} }
xconn.update_cached_wm_info(root); xconn.update_cached_wm_info(root);
let target = Rc::new(RootELW{ let target = Rc::new(RootELW {
p: super::EventLoopWindowTarget::X(EventLoopWindowTarget { p: super::EventLoopWindowTarget::X(EventLoopWindowTarget {
ime, ime,
root, root,
@ -132,7 +144,7 @@ impl<T: 'static> EventLoop<T> {
wm_delete_window, wm_delete_window,
pending_redraws: Default::default(), pending_redraws: Default::default(),
}), }),
_marker: ::std::marker::PhantomData _marker: ::std::marker::PhantomData,
}); });
// A calloop event loop to drive us // A calloop event loop to drive us
@ -144,11 +156,14 @@ impl<T: 'static> EventLoop<T> {
let (user_sender, user_channel) = ::calloop::channel::channel(); let (user_sender, user_channel) = ::calloop::channel::channel();
let _user_source = inner_loop.handle().insert_source(user_channel, move |evt, &mut()| { let _user_source = inner_loop
if let ::calloop::channel::Event::Msg(msg) = evt { .handle()
pending_user_events2.borrow_mut().push_back(msg); .insert_source(user_channel, move |evt, &mut ()| {
} if let ::calloop::channel::Event::Msg(msg) = evt {
}).unwrap(); pending_user_events2.borrow_mut().push_back(msg);
}
})
.unwrap();
// Handle X11 events // Handle X11 events
let pending_events: Rc<RefCell<VecDeque<_>>> = Default::default(); let pending_events: Rc<RefCell<VecDeque<_>>> = Default::default();
@ -164,20 +179,20 @@ impl<T: 'static> EventLoop<T> {
// Register for device hotplug events // Register for device hotplug events
// (The request buffer is flushed during `init_device`) // (The request buffer is flushed during `init_device`)
get_xtarget(&target).xconn.select_xinput_events( get_xtarget(&target)
root, .xconn
ffi::XIAllDevices, .select_xinput_events(root, ffi::XIAllDevices, ffi::XI_HierarchyChangedMask)
ffi::XI_HierarchyChangedMask, .queue();
).queue();
processor.init_device(ffi::XIAllDevices); processor.init_device(ffi::XIAllDevices);
// Setup the X11 event source // Setup the X11 event source
let mut x11_events = ::calloop::generic::Generic::from_raw_fd(get_xtarget(&target).xconn.x11_fd); let mut x11_events =
::calloop::generic::Generic::from_raw_fd(get_xtarget(&target).xconn.x11_fd);
x11_events.set_interest(::calloop::mio::Ready::readable()); x11_events.set_interest(::calloop::mio::Ready::readable());
let _x11_source = inner_loop.handle().insert_source( let _x11_source = inner_loop
x11_events, .handle()
{ .insert_source(x11_events, {
let pending_events = pending_events.clone(); let pending_events = pending_events.clone();
let mut callback = move |event| { let mut callback = move |event| {
pending_events.borrow_mut().push_back(event); pending_events.borrow_mut().push_back(event);
@ -191,8 +206,8 @@ impl<T: 'static> EventLoop<T> {
} }
} }
} }
} })
).unwrap(); .unwrap();
let result = EventLoop { let result = EventLoop {
inner_loop, inner_loop,
@ -201,7 +216,7 @@ impl<T: 'static> EventLoop<T> {
_user_source, _user_source,
user_sender, user_sender,
pending_user_events, pending_user_events,
target target,
}; };
result result
@ -224,7 +239,8 @@ impl<T: 'static> EventLoop<T> {
} }
pub fn run_return<F>(&mut self, mut callback: F) pub fn run_return<F>(&mut self, mut callback: F)
where F: FnMut(Event<T>, &RootELW<T>, &mut ControlFlow) where
F: FnMut(Event<T>, &RootELW<T>, &mut ControlFlow),
{ {
let mut control_flow = ControlFlow::default(); let mut control_flow = ControlFlow::default();
let wt = get_xtarget(&self.target); let wt = get_xtarget(&self.target);
@ -246,7 +262,7 @@ impl<T: 'static> EventLoop<T> {
crate::event::Event::UserEvent(evt), crate::event::Event::UserEvent(evt),
&self.target, &self.target,
&mut control_flow, &mut control_flow,
&mut callback &mut callback,
); );
} }
} }
@ -257,11 +273,11 @@ impl<T: 'static> EventLoop<T> {
sticky_exit_callback( sticky_exit_callback(
Event::WindowEvent { Event::WindowEvent {
window_id: crate::window::WindowId(super::WindowId::X(wid)), window_id: crate::window::WindowId(super::WindowId::X(wid)),
event: WindowEvent::RedrawRequested event: WindowEvent::RedrawRequested,
}, },
&self.target, &self.target,
&mut control_flow, &mut control_flow,
&mut callback &mut callback,
); );
} }
} }
@ -271,29 +287,37 @@ impl<T: 'static> EventLoop<T> {
crate::event::Event::EventsCleared, crate::event::Event::EventsCleared,
&self.target, &self.target,
&mut control_flow, &mut control_flow,
&mut callback &mut callback,
); );
} }
// flush the X11 connection // flush the X11 connection
unsafe { (wt.xconn.xlib.XFlush)(wt.xconn.display); } unsafe {
(wt.xconn.xlib.XFlush)(wt.xconn.display);
}
match control_flow { match control_flow {
ControlFlow::Exit => break, ControlFlow::Exit => break,
ControlFlow::Poll => { ControlFlow::Poll => {
// non-blocking dispatch // non-blocking dispatch
self.inner_loop.dispatch(Some(::std::time::Duration::from_millis(0)), &mut ()).unwrap(); self.inner_loop
callback(crate::event::Event::NewEvents(crate::event::StartCause::Poll), &self.target, &mut control_flow); .dispatch(Some(::std::time::Duration::from_millis(0)), &mut ())
.unwrap();
callback(
crate::event::Event::NewEvents(crate::event::StartCause::Poll),
&self.target,
&mut control_flow,
);
}, },
ControlFlow::Wait => { ControlFlow::Wait => {
self.inner_loop.dispatch(None, &mut ()).unwrap(); self.inner_loop.dispatch(None, &mut ()).unwrap();
callback( callback(
crate::event::Event::NewEvents(crate::event::StartCause::WaitCancelled { crate::event::Event::NewEvents(crate::event::StartCause::WaitCancelled {
start: ::std::time::Instant::now(), start: ::std::time::Instant::now(),
requested_resume: None requested_resume: None,
}), }),
&self.target, &self.target,
&mut control_flow &mut control_flow,
); );
}, },
ControlFlow::WaitUntil(deadline) => { ControlFlow::WaitUntil(deadline) => {
@ -308,32 +332,41 @@ impl<T: 'static> EventLoop<T> {
let now = std::time::Instant::now(); let now = std::time::Instant::now();
if now < deadline { if now < deadline {
callback( callback(
crate::event::Event::NewEvents(crate::event::StartCause::WaitCancelled { crate::event::Event::NewEvents(
start, crate::event::StartCause::WaitCancelled {
requested_resume: Some(deadline) start,
}), requested_resume: Some(deadline),
},
),
&self.target, &self.target,
&mut control_flow &mut control_flow,
); );
} else { } else {
callback( callback(
crate::event::Event::NewEvents(crate::event::StartCause::ResumeTimeReached { crate::event::Event::NewEvents(
start, crate::event::StartCause::ResumeTimeReached {
requested_resume: deadline start,
}), requested_resume: deadline,
},
),
&self.target, &self.target,
&mut control_flow &mut control_flow,
); );
} }
} },
} }
} }
callback(crate::event::Event::LoopDestroyed, &self.target, &mut control_flow); callback(
crate::event::Event::LoopDestroyed,
&self.target,
&mut control_flow,
);
} }
pub fn run<F>(mut self, callback: F) -> ! pub fn run<F>(mut self, callback: F) -> !
where F: 'static + FnMut(Event<T>, &RootELW<T>, &mut ControlFlow) where
F: 'static + FnMut(Event<T>, &RootELW<T>, &mut ControlFlow),
{ {
self.run_return(callback); self.run_return(callback);
::std::process::exit(0); ::std::process::exit(0);
@ -342,10 +375,10 @@ impl<T: 'static> EventLoop<T> {
fn get_xtarget<T>(rt: &RootELW<T>) -> &EventLoopWindowTarget<T> { fn get_xtarget<T>(rt: &RootELW<T>) -> &EventLoopWindowTarget<T> {
if let super::EventLoopWindowTarget::X(ref target) = rt.p { if let super::EventLoopWindowTarget::X(ref target) = rt.p {
target target
} else { } else {
unreachable!(); unreachable!();
} }
} }
impl<T: 'static> EventLoopProxy<T> { impl<T: 'static> EventLoopProxy<T> {
@ -365,19 +398,17 @@ impl<'a> DeviceInfo<'a> {
unsafe { unsafe {
let mut count = mem::uninitialized(); let mut count = mem::uninitialized();
let info = (xconn.xinput2.XIQueryDevice)(xconn.display, device, &mut count); let info = (xconn.xinput2.XIQueryDevice)(xconn.display, device, &mut count);
xconn.check_errors() xconn.check_errors().ok().and_then(|_| {
.ok() if info.is_null() || count == 0 {
.and_then(|_| { None
if info.is_null() || count == 0 { } else {
None Some(DeviceInfo {
} else { xconn,
Some(DeviceInfo { info,
xconn, count: count as usize,
info, })
count: count as usize, }
}) })
}
})
} }
} }
} }
@ -428,10 +459,11 @@ impl Window {
pub fn new<T>( pub fn new<T>(
event_loop: &EventLoopWindowTarget<T>, event_loop: &EventLoopWindowTarget<T>,
attribs: WindowAttributes, attribs: WindowAttributes,
pl_attribs: PlatformSpecificWindowBuilderAttributes pl_attribs: PlatformSpecificWindowBuilderAttributes,
) -> Result<Self, RootOsError> { ) -> Result<Self, RootOsError> {
let window = Arc::new(UnownedWindow::new(&event_loop, attribs, pl_attribs)?); let window = Arc::new(UnownedWindow::new(&event_loop, attribs, pl_attribs)?);
event_loop.windows event_loop
.windows
.borrow_mut() .borrow_mut()
.insert(window.id(), Arc::downgrade(&window)); .insert(window.id(), Arc::downgrade(&window));
Ok(Window(window)) Ok(Window(window))
@ -454,11 +486,14 @@ impl Drop for Window {
/// extract the cookie from a GenericEvent XEvent and release the cookie data once it has been processed /// extract the cookie from a GenericEvent XEvent and release the cookie data once it has been processed
struct GenericEventCookie<'a> { struct GenericEventCookie<'a> {
xconn: &'a XConnection, xconn: &'a XConnection,
cookie: ffi::XGenericEventCookie cookie: ffi::XGenericEventCookie,
} }
impl<'a> GenericEventCookie<'a> { impl<'a> GenericEventCookie<'a> {
fn from_event<'b>(xconn: &'b XConnection, event: ffi::XEvent) -> Option<GenericEventCookie<'b>> { fn from_event<'b>(
xconn: &'b XConnection,
event: ffi::XEvent,
) -> Option<GenericEventCookie<'b>> {
unsafe { unsafe {
let mut cookie: ffi::XGenericEventCookie = From::from(event); let mut cookie: ffi::XGenericEventCookie = From::from(event);
if (xconn.xlib.XGetEventData)(xconn.display, &mut cookie) == ffi::True { if (xconn.xlib.XGetEventData)(xconn.display, &mut cookie) == ffi::True {
@ -485,8 +520,12 @@ struct XExtension {
first_error_id: c_int, first_error_id: c_int,
} }
fn mkwid(w: ffi::Window) -> crate::window::WindowId { crate::window::WindowId(crate::platform_impl::WindowId::X(WindowId(w))) } fn mkwid(w: ffi::Window) -> crate::window::WindowId {
fn mkdid(w: c_int) -> crate::event::DeviceId { crate::event::DeviceId(crate::platform_impl::DeviceId::X(DeviceId(w))) } crate::window::WindowId(crate::platform_impl::WindowId::X(WindowId(w)))
}
fn mkdid(w: c_int) -> crate::event::DeviceId {
crate::event::DeviceId(crate::platform_impl::DeviceId::X(DeviceId(w)))
}
#[derive(Debug)] #[derive(Debug)]
struct Device { struct Device {
@ -525,32 +564,39 @@ impl Device {
| ffi::XI_RawKeyPressMask | ffi::XI_RawKeyPressMask
| ffi::XI_RawKeyReleaseMask; | ffi::XI_RawKeyReleaseMask;
// The request buffer is flushed when we poll for events // The request buffer is flushed when we poll for events
wt.xconn.select_xinput_events(wt.root, info.deviceid, mask).queue(); wt.xconn
.select_xinput_events(wt.root, info.deviceid, mask)
.queue();
// Identify scroll axes // Identify scroll axes
for class_ptr in Device::classes(info) { for class_ptr in Device::classes(info) {
let class = unsafe { &**class_ptr }; let class = unsafe { &**class_ptr };
match class._type { match class._type {
ffi::XIScrollClass => { ffi::XIScrollClass => {
let info = unsafe { mem::transmute::<&ffi::XIAnyClassInfo, &ffi::XIScrollClassInfo>(class) }; let info = unsafe {
scroll_axes.push((info.number, ScrollAxis { mem::transmute::<&ffi::XIAnyClassInfo, &ffi::XIScrollClassInfo>(class)
increment: info.increment, };
orientation: match info.scroll_type { scroll_axes.push((
ffi::XIScrollTypeHorizontal => ScrollOrientation::Horizontal, info.number,
ffi::XIScrollTypeVertical => ScrollOrientation::Vertical, ScrollAxis {
_ => { unreachable!() } increment: info.increment,
orientation: match info.scroll_type {
ffi::XIScrollTypeHorizontal => ScrollOrientation::Horizontal,
ffi::XIScrollTypeVertical => ScrollOrientation::Vertical,
_ => unreachable!(),
},
position: 0.0,
}, },
position: 0.0, ));
})); },
} _ => {},
_ => {}
} }
} }
} }
let mut device = Device { let mut device = Device {
name: name.into_owned(), name: name.into_owned(),
scroll_axes: scroll_axes, scroll_axes,
attachment: info.attachment, attachment: info.attachment,
}; };
device.reset_scroll_position(info); device.reset_scroll_position(info);
@ -563,12 +609,18 @@ impl Device {
let class = unsafe { &**class_ptr }; let class = unsafe { &**class_ptr };
match class._type { match class._type {
ffi::XIValuatorClass => { ffi::XIValuatorClass => {
let info = unsafe { mem::transmute::<&ffi::XIAnyClassInfo, &ffi::XIValuatorClassInfo>(class) }; let info = unsafe {
if let Some(&mut (_, ref mut axis)) = self.scroll_axes.iter_mut().find(|&&mut (axis, _)| axis == info.number) { mem::transmute::<&ffi::XIAnyClassInfo, &ffi::XIValuatorClassInfo>(class)
};
if let Some(&mut (_, ref mut axis)) = self
.scroll_axes
.iter_mut()
.find(|&&mut (axis, _)| axis == info.number)
{
axis.position = info.value; axis.position = info.value;
} }
} },
_ => {} _ => {},
} }
} }
} }
@ -576,11 +628,18 @@ impl Device {
#[inline] #[inline]
fn physical_device(info: &ffi::XIDeviceInfo) -> bool { fn physical_device(info: &ffi::XIDeviceInfo) -> bool {
info._use == ffi::XISlaveKeyboard || info._use == ffi::XISlavePointer || info._use == ffi::XIFloatingSlave info._use == ffi::XISlaveKeyboard
|| info._use == ffi::XISlavePointer
|| info._use == ffi::XIFloatingSlave
} }
#[inline] #[inline]
fn classes(info: &ffi::XIDeviceInfo) -> &[*const ffi::XIAnyClassInfo] { fn classes(info: &ffi::XIDeviceInfo) -> &[*const ffi::XIAnyClassInfo] {
unsafe { slice::from_raw_parts(info.classes as *const *const ffi::XIAnyClassInfo, info.num_classes as usize) } unsafe {
slice::from_raw_parts(
info.classes as *const *const ffi::XIAnyClassInfo,
info.num_classes as usize,
)
}
} }
} }

View file

@ -2,16 +2,16 @@ use std::os::raw::*;
use parking_lot::Mutex; use parking_lot::Mutex;
use crate::dpi::{PhysicalPosition, PhysicalSize}; use super::{
use crate::monitor::VideoMode; ffi::{
use super::{util, XConnection, XError}; RRCrtcChangeNotifyMask, RROutputPropertyNotifyMask, RRScreenChangeNotifyMask, True, Window,
use super::ffi::{ XRRScreenResources,
RRCrtcChangeNotifyMask, },
RROutputPropertyNotifyMask, util, XConnection, XError,
RRScreenChangeNotifyMask, };
True, use crate::{
Window, dpi::{PhysicalPosition, PhysicalSize},
XRRScreenResources, monitor::VideoMode,
}; };
// Used to test XRandR < 1.5 code path. This should always be committed as false. // Used to test XRandR < 1.5 code path. This should always be committed as false.
@ -160,7 +160,8 @@ impl XConnection {
// videowalls. // videowalls.
let xrandr_1_5 = self.xrandr_1_5.as_ref().unwrap(); let xrandr_1_5 = self.xrandr_1_5.as_ref().unwrap();
let mut monitor_count = 0; let mut monitor_count = 0;
let monitors = (xrandr_1_5.XRRGetMonitors)(self.display, root, 1, &mut monitor_count); let monitors =
(xrandr_1_5.XRRGetMonitors)(self.display, root, 1, &mut monitor_count);
assert!(monitor_count >= 0); assert!(monitor_count >= 0);
available = Vec::with_capacity(monitor_count as usize); available = Vec::with_capacity(monitor_count as usize);
for monitor_index in 0..monitor_count { for monitor_index in 0..monitor_count {
@ -173,7 +174,8 @@ impl XConnection {
monitor_index as u32, monitor_index as u32,
monitor.into(), monitor.into(),
is_primary, is_primary,
).map(|monitor_id| available.push(monitor_id)); )
.map(|monitor_id| available.push(monitor_id));
} }
(xrandr_1_5.XRRFreeMonitors)(monitors); (xrandr_1_5.XRRFreeMonitors)(monitors);
} else { } else {
@ -190,13 +192,8 @@ impl XConnection {
let crtc = util::MonitorRepr::from(crtc); let crtc = util::MonitorRepr::from(crtc);
let is_primary = crtc.get_output() == primary; let is_primary = crtc.get_output() == primary;
has_primary |= is_primary; has_primary |= is_primary;
MonitorHandle::from_repr( MonitorHandle::from_repr(self, resources, crtc_id as u32, crtc, is_primary)
self, .map(|monitor_id| available.push(monitor_id));
resources,
crtc_id as u32,
crtc,
is_primary,
).map(|monitor_id| available.push(monitor_id));
} }
(self.xrandr.XRRFreeCrtcInfo)(crtc); (self.xrandr.XRRFreeCrtcInfo)(crtc);
} }
@ -244,13 +241,8 @@ impl XConnection {
if version_lock.is_none() { if version_lock.is_none() {
let mut major = 0; let mut major = 0;
let mut minor = 0; let mut minor = 0;
let has_extension = unsafe { let has_extension =
(self.xrandr.XRRQueryVersion)( unsafe { (self.xrandr.XRRQueryVersion)(self.display, &mut major, &mut minor) };
self.display,
&mut major,
&mut minor,
)
};
if has_extension != True { if has_extension != True {
panic!("[winit] XRandR extension not available."); panic!("[winit] XRandR extension not available.");
} }
@ -261,11 +253,7 @@ impl XConnection {
let mut event_offset = 0; let mut event_offset = 0;
let mut error_offset = 0; let mut error_offset = 0;
let status = unsafe { let status = unsafe {
(self.xrandr.XRRQueryExtension)( (self.xrandr.XRRQueryExtension)(self.display, &mut event_offset, &mut error_offset)
self.display,
&mut event_offset,
&mut error_offset,
)
}; };
if status != True { if status != True {
@ -273,9 +261,7 @@ impl XConnection {
unreachable!("[winit] `XRRQueryExtension` failed but no error was received."); unreachable!("[winit] `XRRQueryExtension` failed but no error was received.");
} }
let mask = RRCrtcChangeNotifyMask let mask = RRCrtcChangeNotifyMask | RROutputPropertyNotifyMask | RRScreenChangeNotifyMask;
| RROutputPropertyNotifyMask
| RRScreenChangeNotifyMask;
unsafe { (self.xrandr.XRRSelectInput)(self.display, root, mask) }; unsafe { (self.xrandr.XRRSelectInput)(self.display, root, mask) };
Ok(event_offset) Ok(event_offset)

View file

@ -1,7 +1,9 @@
use std::collections::HashMap; use std::{
use std::ffi::{CStr, CString}; collections::HashMap,
use std::fmt::Debug; ffi::{CStr, CString},
use std::os::raw::*; fmt::Debug,
os::raw::*,
};
use parking_lot::Mutex; use parking_lot::Mutex;
@ -21,11 +23,9 @@ impl XConnection {
if let Some(atom) = cached_atom { if let Some(atom) = cached_atom {
atom atom
} else { } else {
let atom = unsafe { (self.xlib.XInternAtom)( let atom = unsafe {
self.display, (self.xlib.XInternAtom)(self.display, name.as_ptr() as *const c_char, ffi::False)
name.as_ptr() as *const c_char, };
ffi::False,
) };
if atom == 0 { if atom == 0 {
let msg = format!( let msg = format!(
"`XInternAtom` failed, which really shouldn't happen. Atom: {:?}, Error: {:#?}", "`XInternAtom` failed, which really shouldn't happen. Atom: {:?}, Error: {:#?}",
@ -52,7 +52,7 @@ impl XConnection {
// Note: this doesn't use caching, for the sake of simplicity. // Note: this doesn't use caching, for the sake of simplicity.
// If you're dealing with this many atoms, you'll usually want to cache them locally anyway. // If you're dealing with this many atoms, you'll usually want to cache them locally anyway.
pub unsafe fn get_atoms(&self, names: &[*mut c_char]) -> Result<Vec<ffi::Atom>, XError> { pub unsafe fn get_atoms(&self, names: &[*mut c_char]) -> Result<Vec<ffi::Atom>, XError> {
let mut atoms = Vec::with_capacity(names.len()); let mut atoms = Vec::with_capacity(names.len());
(self.xlib.XInternAtoms)( (self.xlib.XInternAtoms)(
self.display, self.display,

View file

@ -24,7 +24,7 @@ impl XConnection {
pub fn send_client_msg( pub fn send_client_msg(
&self, &self,
window: c_ulong, // The window this is "about"; not necessarily this window window: c_ulong, // The window this is "about"; not necessarily this window
target_window: c_ulong, // The window we're sending to target_window: c_ulong, // The window we're sending to
message_type: ffi::Atom, message_type: ffi::Atom,
event_mask: Option<c_long>, event_mask: Option<c_long>,
@ -45,7 +45,7 @@ impl XConnection {
// to send more than one message worth of data. // to send more than one message worth of data.
pub fn send_client_msg_multi<T: Formattable>( pub fn send_client_msg_multi<T: Formattable>(
&self, &self,
window: c_ulong, // The window this is "about"; not necessarily this window window: c_ulong, // The window this is "about"; not necessarily this window
target_window: c_ulong, // The window we're sending to target_window: c_ulong, // The window we're sending to
message_type: ffi::Atom, message_type: ffi::Atom,
event_mask: Option<c_long>, event_mask: Option<c_long>,

View file

@ -1,6 +1,4 @@
use std::fmt::Debug; use std::{fmt::Debug, mem, os::raw::*};
use std::mem;
use std::os::raw::*;
// This isn't actually the number of the bits in the format. // This isn't actually the number of the bits in the format.
// X11 does a match on this value to determine which type to call sizeof on. // X11 does a match on this value to determine which type to call sizeof on.
@ -50,9 +48,21 @@ pub trait Formattable: Debug + Clone + Copy + PartialEq + PartialOrd {
} }
// You might be surprised by the absence of c_int, but not as surprised as X11 would be by the presence of it. // You might be surprised by the absence of c_int, but not as surprised as X11 would be by the presence of it.
impl Formattable for c_schar { const FORMAT: Format = Format::Char; } impl Formattable for c_schar {
impl Formattable for c_uchar { const FORMAT: Format = Format::Char; } const FORMAT: Format = Format::Char;
impl Formattable for c_short { const FORMAT: Format = Format::Short; } }
impl Formattable for c_ushort { const FORMAT: Format = Format::Short; } impl Formattable for c_uchar {
impl Formattable for c_long { const FORMAT: Format = Format::Long; } const FORMAT: Format = Format::Char;
impl Formattable for c_ulong { const FORMAT: Format = Format::Long; } }
impl Formattable for c_short {
const FORMAT: Format = Format::Short;
}
impl Formattable for c_ushort {
const FORMAT: Format = Format::Short;
}
impl Formattable for c_long {
const FORMAT: Format = Format::Long;
}
impl Formattable for c_ulong {
const FORMAT: Format = Format::Long;
}

View file

@ -16,7 +16,12 @@ impl AaRect {
pub fn new((x, y): (i32, i32), (width, height): (u32, u32)) -> Self { pub fn new((x, y): (i32, i32), (width, height): (u32, u32)) -> Self {
let (x, y) = (x as i64, y as i64); let (x, y) = (x as i64, y as i64);
let (width, height) = (width as i64, height as i64); let (width, height) = (width as i64, height as i64);
AaRect { x, y, width, height } AaRect {
x,
y,
width,
height,
}
} }
pub fn contains_point(&self, x: i64, y: i64) -> bool { pub fn contains_point(&self, x: i64, y: i64) -> bool {
@ -73,7 +78,12 @@ pub struct FrameExtents {
impl FrameExtents { impl FrameExtents {
pub fn new(left: c_ulong, right: c_ulong, top: c_ulong, bottom: c_ulong) -> Self { pub fn new(left: c_ulong, right: c_ulong, top: c_ulong, bottom: c_ulong) -> Self {
FrameExtents { left, right, top, bottom } FrameExtents {
left,
right,
top,
bottom,
}
} }
pub fn from_border(border: c_ulong) -> Self { pub fn from_border(border: c_ulong) -> Self {
@ -116,13 +126,20 @@ impl FrameExtentsHeuristic {
pub fn inner_pos_to_outer(&self, x: i32, y: i32) -> (i32, i32) { pub fn inner_pos_to_outer(&self, x: i32, y: i32) -> (i32, i32) {
use self::FrameExtentsHeuristicPath::*; use self::FrameExtentsHeuristicPath::*;
if self.heuristic_path != UnsupportedBordered { if self.heuristic_path != UnsupportedBordered {
(x - self.frame_extents.left as i32, y - self.frame_extents.top as i32) (
x - self.frame_extents.left as i32,
y - self.frame_extents.top as i32,
)
} else { } else {
(x, y) (x, y)
} }
} }
pub fn inner_pos_to_outer_logical(&self, mut logical: LogicalPosition, factor: f64) -> LogicalPosition { pub fn inner_pos_to_outer_logical(
&self,
mut logical: LogicalPosition,
factor: f64,
) -> LogicalPosition {
use self::FrameExtentsHeuristicPath::*; use self::FrameExtentsHeuristicPath::*;
if self.heuristic_path != UnsupportedBordered { if self.heuristic_path != UnsupportedBordered {
let frame_extents = self.frame_extents.as_logical(factor); let frame_extents = self.frame_extents.as_logical(factor);
@ -135,15 +152,23 @@ impl FrameExtentsHeuristic {
pub fn inner_size_to_outer(&self, width: u32, height: u32) -> (u32, u32) { pub fn inner_size_to_outer(&self, width: u32, height: u32) -> (u32, u32) {
( (
width.saturating_add( width.saturating_add(
self.frame_extents.left.saturating_add(self.frame_extents.right) as u32 self.frame_extents
.left
.saturating_add(self.frame_extents.right) as u32,
), ),
height.saturating_add( height.saturating_add(
self.frame_extents.top.saturating_add(self.frame_extents.bottom) as u32 self.frame_extents
.top
.saturating_add(self.frame_extents.bottom) as u32,
), ),
) )
} }
pub fn inner_size_to_outer_logical(&self, mut logical: LogicalSize, factor: f64) -> LogicalSize { pub fn inner_size_to_outer_logical(
&self,
mut logical: LogicalSize,
factor: f64,
) -> LogicalSize {
let frame_extents = self.frame_extents.as_logical(factor); let frame_extents = self.frame_extents.as_logical(factor);
logical.width += frame_extents.left + frame_extents.right; logical.width += frame_extents.left + frame_extents.right;
logical.height += frame_extents.top + frame_extents.bottom; logical.height += frame_extents.top + frame_extents.bottom;
@ -153,7 +178,11 @@ impl FrameExtentsHeuristic {
impl XConnection { impl XConnection {
// This is adequate for inner_position // This is adequate for inner_position
pub fn translate_coords(&self, window: ffi::Window, root: ffi::Window) -> Result<TranslatedCoords, XError> { pub fn translate_coords(
&self,
window: ffi::Window,
root: ffi::Window,
) -> Result<TranslatedCoords, XError> {
let mut translated_coords: TranslatedCoords = unsafe { mem::uninitialized() }; let mut translated_coords: TranslatedCoords = unsafe { mem::uninitialized() };
unsafe { unsafe {
(self.xlib.XTranslateCoordinates)( (self.xlib.XTranslateCoordinates)(
@ -201,11 +230,9 @@ impl XConnection {
// Of the WMs tested, xmonad, i3, dwm, IceWM (1.3.x and earlier), and blackbox don't // Of the WMs tested, xmonad, i3, dwm, IceWM (1.3.x and earlier), and blackbox don't
// support this. As this is part of EWMH (Extended Window Manager Hints), it's likely to // support this. As this is part of EWMH (Extended Window Manager Hints), it's likely to
// be unsupported by many smaller WMs. // be unsupported by many smaller WMs.
let extents: Option<Vec<c_ulong>> = self.get_property( let extents: Option<Vec<c_ulong>> = self
window, .get_property(window, extents_atom, ffi::XA_CARDINAL)
extents_atom, .ok();
ffi::XA_CARDINAL,
).ok();
extents.and_then(|extents| { extents.and_then(|extents| {
if extents.len() >= 4 { if extents.len() >= 4 {
@ -228,11 +255,9 @@ impl XConnection {
return None; return None;
} }
let client_list: Option<Vec<ffi::Window>> = self.get_property( let client_list: Option<Vec<ffi::Window>> = self
root, .get_property(root, client_list_atom, ffi::XA_WINDOW)
client_list_atom, .ok();
ffi::XA_WINDOW,
).ok();
client_list.map(|client_list| client_list.contains(&window)) client_list.map(|client_list| client_list.contains(&window))
} }
@ -264,7 +289,11 @@ impl XConnection {
self.check_errors().map(|_| parent) self.check_errors().map(|_| parent)
} }
fn climb_hierarchy(&self, window: ffi::Window, root: ffi::Window) -> Result<ffi::Window, XError> { fn climb_hierarchy(
&self,
window: ffi::Window,
root: ffi::Window,
) -> Result<ffi::Window, XError> {
let mut outer_window = window; let mut outer_window = window;
loop { loop {
let candidate = self.get_parent_window(outer_window)?; let candidate = self.get_parent_window(outer_window)?;
@ -276,7 +305,11 @@ impl XConnection {
Ok(outer_window) Ok(outer_window)
} }
pub fn get_frame_extents_heuristic(&self, window: ffi::Window, root: ffi::Window) -> FrameExtentsHeuristic { pub fn get_frame_extents_heuristic(
&self,
window: ffi::Window,
root: ffi::Window,
) -> FrameExtentsHeuristic {
use self::FrameExtentsHeuristicPath::*; use self::FrameExtentsHeuristicPath::*;
// Position relative to root window. // Position relative to root window.
@ -284,15 +317,16 @@ impl XConnection {
// isn't nested are outlined in the comments throghout this function, but in addition to // isn't nested are outlined in the comments throghout this function, but in addition to
// that, fullscreen windows often aren't nested. // that, fullscreen windows often aren't nested.
let (inner_y_rel_root, child) = { let (inner_y_rel_root, child) = {
let coords = self.translate_coords(window, root).expect("Failed to translate window coordinates"); let coords = self
( .translate_coords(window, root)
coords.y_rel_root, .expect("Failed to translate window coordinates");
coords.child, (coords.y_rel_root, coords.child)
)
}; };
let (width, height, border) = { let (width, height, border) = {
let inner_geometry = self.get_geometry(window).expect("Failed to get inner window geometry"); let inner_geometry = self
.get_geometry(window)
.expect("Failed to get inner window geometry");
( (
inner_geometry.width, inner_geometry.width,
inner_geometry.height, inner_geometry.height,
@ -343,9 +377,13 @@ impl XConnection {
// If the position value we have is for a nested window used as the client area, we'll // If the position value we have is for a nested window used as the client area, we'll
// just climb up the hierarchy and get the geometry of the outermost window we're // just climb up the hierarchy and get the geometry of the outermost window we're
// nested in. // nested in.
let outer_window = self.climb_hierarchy(window, root).expect("Failed to climb window hierarchy"); let outer_window = self
.climb_hierarchy(window, root)
.expect("Failed to climb window hierarchy");
let (outer_y, outer_width, outer_height) = { let (outer_y, outer_width, outer_height) = {
let outer_geometry = self.get_geometry(outer_window).expect("Failed to get outer window geometry"); let outer_geometry = self
.get_geometry(outer_window)
.expect("Failed to get outer window geometry");
( (
outer_geometry.y_rel_parent, outer_geometry.y_rel_parent,
outer_geometry.width, outer_geometry.width,
@ -364,12 +402,8 @@ impl XConnection {
let top = offset_y; let top = offset_y;
let bottom = diff_y.saturating_sub(offset_y); let bottom = diff_y.saturating_sub(offset_y);
let frame_extents = FrameExtents::new( let frame_extents =
left.into(), FrameExtents::new(left.into(), right.into(), top.into(), bottom.into());
right.into(),
top.into(),
bottom.into(),
);
FrameExtentsHeuristic { FrameExtentsHeuristic {
frame_extents, frame_extents,
heuristic_path: UnsupportedNested, heuristic_path: UnsupportedNested,

View file

@ -99,7 +99,9 @@ pub struct NormalHints<'a> {
impl<'a> NormalHints<'a> { impl<'a> NormalHints<'a> {
pub fn new(xconn: &'a XConnection) -> Self { pub fn new(xconn: &'a XConnection) -> Self {
NormalHints { size_hints: xconn.alloc_size_hints() } NormalHints {
size_hints: xconn.alloc_size_hints(),
}
} }
pub fn has_flag(&self, flag: c_long) -> bool { pub fn has_flag(&self, flag: c_long) -> bool {
@ -130,7 +132,11 @@ impl<'a> NormalHints<'a> {
} }
pub fn get_max_size(&self) -> Option<(u32, u32)> { pub fn get_max_size(&self) -> Option<(u32, u32)> {
self.getter(ffi::PMaxSize, &self.size_hints.max_width, &self.size_hints.max_height) self.getter(
ffi::PMaxSize,
&self.size_hints.max_width,
&self.size_hints.max_height,
)
} }
pub fn set_max_size(&mut self, max_size: Option<(u32, u32)>) { pub fn set_max_size(&mut self, max_size: Option<(u32, u32)>) {
@ -144,7 +150,11 @@ impl<'a> NormalHints<'a> {
} }
pub fn get_min_size(&self) -> Option<(u32, u32)> { pub fn get_min_size(&self) -> Option<(u32, u32)> {
self.getter(ffi::PMinSize, &self.size_hints.min_width, &self.size_hints.min_height) self.getter(
ffi::PMinSize,
&self.size_hints.min_width,
&self.size_hints.min_height,
)
} }
pub fn set_min_size(&mut self, min_size: Option<(u32, u32)>) { pub fn set_min_size(&mut self, min_size: Option<(u32, u32)>) {
@ -158,7 +168,11 @@ impl<'a> NormalHints<'a> {
} }
pub fn get_resize_increments(&self) -> Option<(u32, u32)> { pub fn get_resize_increments(&self) -> Option<(u32, u32)> {
self.getter(ffi::PResizeInc, &self.size_hints.width_inc, &self.size_hints.height_inc) self.getter(
ffi::PResizeInc,
&self.size_hints.width_inc,
&self.size_hints.height_inc,
)
} }
pub fn set_resize_increments(&mut self, resize_increments: Option<(u32, u32)>) { pub fn set_resize_increments(&mut self, resize_increments: Option<(u32, u32)>) {
@ -172,7 +186,11 @@ impl<'a> NormalHints<'a> {
} }
pub fn get_base_size(&self) -> Option<(u32, u32)> { pub fn get_base_size(&self) -> Option<(u32, u32)> {
self.getter(ffi::PBaseSize, &self.size_hints.base_width, &self.size_hints.base_height) self.getter(
ffi::PBaseSize,
&self.size_hints.base_width,
&self.size_hints.base_height,
)
} }
pub fn set_base_size(&mut self, base_size: Option<(u32, u32)>) { pub fn set_base_size(&mut self, base_size: Option<(u32, u32)>) {
@ -187,7 +205,10 @@ impl<'a> NormalHints<'a> {
} }
impl XConnection { impl XConnection {
pub fn get_wm_hints(&self, window: ffi::Window) -> Result<XSmartPointer<'_, ffi::XWMHints>, XError> { pub fn get_wm_hints(
&self,
window: ffi::Window,
) -> Result<XSmartPointer<'_, ffi::XWMHints>, XError> {
let wm_hints = unsafe { (self.xlib.XGetWMHints)(self.display, window) }; let wm_hints = unsafe { (self.xlib.XGetWMHints)(self.display, window) };
self.check_errors()?; self.check_errors()?;
let wm_hints = if wm_hints.is_null() { let wm_hints = if wm_hints.is_null() {
@ -198,13 +219,13 @@ impl XConnection {
Ok(wm_hints) Ok(wm_hints)
} }
pub fn set_wm_hints(&self, window: ffi::Window, wm_hints: XSmartPointer<'_, ffi::XWMHints>) -> Flusher<'_> { pub fn set_wm_hints(
&self,
window: ffi::Window,
wm_hints: XSmartPointer<'_, ffi::XWMHints>,
) -> Flusher<'_> {
unsafe { unsafe {
(self.xlib.XSetWMHints)( (self.xlib.XSetWMHints)(self.display, window, wm_hints.ptr);
self.display,
window,
wm_hints.ptr,
);
} }
Flusher::new(self) Flusher::new(self)
} }
@ -223,13 +244,13 @@ impl XConnection {
self.check_errors().map(|_| NormalHints { size_hints }) self.check_errors().map(|_| NormalHints { size_hints })
} }
pub fn set_normal_hints(&self, window: ffi::Window, normal_hints: NormalHints<'_>) -> Flusher<'_> { pub fn set_normal_hints(
&self,
window: ffi::Window,
normal_hints: NormalHints<'_>,
) -> Flusher<'_> {
unsafe { unsafe {
(self.xlib.XSetWMNormalHints)( (self.xlib.XSetWMNormalHints)(self.display, window, normal_hints.size_hints.ptr);
self.display,
window,
normal_hints.size_hints.ptr,
);
} }
Flusher::new(self) Flusher::new(self)
} }

View file

@ -1,5 +1,5 @@
use crate::window::{Icon, Pixel, PIXEL_SIZE};
use super::*; use super::*;
use crate::window::{Icon, Pixel, PIXEL_SIZE};
impl Pixel { impl Pixel {
pub fn to_packed_argb(&self) -> Cardinal { pub fn to_packed_argb(&self) -> Cardinal {

View file

@ -55,7 +55,12 @@ impl<'a> Drop for PointerState<'a> {
} }
impl XConnection { impl XConnection {
pub fn select_xinput_events(&self, window: c_ulong, device_id: c_int, mask: i32) -> Flusher<'_> { pub fn select_xinput_events(
&self,
window: c_ulong,
device_id: c_int,
mask: i32,
) -> Flusher<'_> {
let mut event_mask = ffi::XIEventMask { let mut event_mask = ffi::XIEventMask {
deviceid: device_id, deviceid: device_id,
mask: &mask as *const _ as *mut c_uchar, mask: &mask as *const _ as *mut c_uchar,
@ -74,14 +79,7 @@ impl XConnection {
#[allow(dead_code)] #[allow(dead_code)]
pub fn select_xkb_events(&self, device_id: c_uint, mask: c_ulong) -> Option<Flusher<'_>> { pub fn select_xkb_events(&self, device_id: c_uint, mask: c_ulong) -> Option<Flusher<'_>> {
let status = unsafe { let status = unsafe { (self.xlib.XkbSelectEvents)(self.display, device_id, mask, mask) };
(self.xlib.XkbSelectEvents)(
self.display,
device_id,
mask,
mask,
)
};
if status == ffi::True { if status == ffi::True {
Some(Flusher::new(self)) Some(Flusher::new(self))
} else { } else {
@ -89,7 +87,11 @@ impl XConnection {
} }
} }
pub fn query_pointer(&self, window: ffi::Window, device_id: c_int) -> Result<PointerState<'_>, XError> { pub fn query_pointer(
&self,
window: ffi::Window,
device_id: c_int,
) -> Result<PointerState<'_>, XError> {
unsafe { unsafe {
let mut pointer_state: PointerState<'_> = mem::uninitialized(); let mut pointer_state: PointerState<'_> = mem::uninitialized();
pointer_state.xconn = self; pointer_state.xconn = self;

View file

@ -12,10 +12,7 @@ impl<'a, T> XSmartPointer<'a, T> {
// Returns None if ptr is null. // Returns None if ptr is null.
pub fn new(xconn: &'a XConnection, ptr: *mut T) -> Option<Self> { pub fn new(xconn: &'a XConnection, ptr: *mut T) -> Option<Self> {
if !ptr.is_null() { if !ptr.is_null() {
Some(XSmartPointer { Some(XSmartPointer { xconn, ptr })
xconn,
ptr,
})
} else { } else {
None None
} }

View file

@ -13,22 +13,12 @@ mod randr;
mod window_property; mod window_property;
mod wm; mod wm;
pub use self::atom::*; pub use self::{
pub use self::client_msg::*; atom::*, client_msg::*, format::*, geometry::*, hint::*, icon::*, input::*, memory::*,
pub use self::format::*; randr::*, window_property::*, wm::*,
pub use self::geometry::*; };
pub use self::hint::*;
pub use self::icon::*;
pub use self::input::*;
pub use self::memory::*;
pub use self::randr::*;
pub use self::window_property::*;
pub use self::wm::*;
use std::mem; use std::{mem, ops::BitAnd, os::raw::*, ptr};
use std::ptr;
use std::ops::BitAnd;
use std::os::raw::*;
use super::{ffi, XConnection, XError}; use super::{ffi, XConnection, XError};
@ -48,8 +38,8 @@ pub fn maybe_change<T: PartialEq>(field: &mut Option<T>, value: T) -> bool {
} }
pub fn has_flag<T>(bitset: T, flag: T) -> bool pub fn has_flag<T>(bitset: T, flag: T) -> bool
where T: where
Copy + PartialEq + BitAnd<T, Output = T> T: Copy + PartialEq + BitAnd<T, Output = T>,
{ {
bitset & flag == flag bitset & flag == flag
} }

View file

@ -1,9 +1,7 @@
use std::{env, slice}; use std::{env, slice, str::FromStr};
use std::str::FromStr;
use crate::monitor::VideoMode;
use crate::dpi::validate_hidpi_factor;
use super::*; use super::*;
use crate::{dpi::validate_hidpi_factor, monitor::VideoMode};
pub fn calc_dpi_factor( pub fn calc_dpi_factor(
(width_px, height_px): (u32, u32), (width_px, height_px): (u32, u32),
@ -29,9 +27,7 @@ pub fn calc_dpi_factor(
return 1.0; return 1.0;
} }
let ppmm = ( let ppmm = ((width_px as f64 * height_px as f64) / (width_mm as f64 * height_mm as f64)).sqrt();
(width_px as f64 * height_px as f64) / (width_mm as f64 * height_mm as f64)
).sqrt();
// Quantize 1/12 step size // Quantize 1/12 step size
let dpi_factor = ((ppmm * (12.0 * 25.4 / 96.0)).round() / 12.0).max(1.0); let dpi_factor = ((ppmm * (12.0 * 25.4 / 96.0)).round() / 12.0).max(1.0);
assert!(validate_hidpi_factor(dpi_factor)); assert!(validate_hidpi_factor(dpi_factor));
@ -88,7 +84,7 @@ impl XConnection {
return None; return None;
} }
if let Ok(res) = ::std::ffi::CStr::from_ptr(resource_manager_str).to_str() { if let Ok(res) = ::std::ffi::CStr::from_ptr(resource_manager_str).to_str() {
let name : &str = "Xft.dpi:\t"; let name: &str = "Xft.dpi:\t";
for pair in res.split("\n") { for pair in res.split("\n") {
if pair.starts_with(&name) { if pair.starts_with(&name) {
let res = &pair[name.len()..]; let res = &pair[name.len()..];
@ -103,11 +99,8 @@ impl XConnection {
resources: *mut ffi::XRRScreenResources, resources: *mut ffi::XRRScreenResources,
repr: &MonitorRepr, repr: &MonitorRepr,
) -> Option<(String, f64, Vec<VideoMode>)> { ) -> Option<(String, f64, Vec<VideoMode>)> {
let output_info = (self.xrandr.XRRGetOutputInfo)( let output_info =
self.display, (self.xrandr.XRRGetOutputInfo)(self.display, resources, repr.get_output());
resources,
repr.get_output(),
);
if output_info.is_null() { if output_info.is_null() {
// When calling `XRRGetOutputInfo` on a virtual monitor (versus a physical display) // When calling `XRRGetOutputInfo` on a virtual monitor (versus a physical display)
// it's possible for it to return null. // it's possible for it to return null.
@ -152,7 +145,10 @@ impl XConnection {
} else { } else {
calc_dpi_factor( calc_dpi_factor(
repr.size(), repr.size(),
((*output_info).mm_width as u64, (*output_info).mm_height as u64), (
(*output_info).mm_width as u64,
(*output_info).mm_height as u64,
),
) )
}; };

View file

@ -84,10 +84,8 @@ impl XConnection {
if !buf.is_null() { if !buf.is_null() {
offset += PROPERTY_BUFFER_SIZE; offset += PROPERTY_BUFFER_SIZE;
let new_data = std::slice::from_raw_parts( let new_data =
buf as *mut T, std::slice::from_raw_parts(buf as *mut T, quantity_returned as usize);
quantity_returned as usize,
);
/*println!( /*println!(
"XGetWindowProperty prop:{:?} fmt:{:02} len:{:02} off:{:02} out:{:02}, buf:{:?}", "XGetWindowProperty prop:{:?} fmt:{:02} len:{:02} off:{:02} out:{:02}, buf:{:?}",
property, property,

View file

@ -28,11 +28,8 @@ impl XConnection {
fn get_supported_hints(&self, root: ffi::Window) -> Vec<ffi::Atom> { fn get_supported_hints(&self, root: ffi::Window) -> Vec<ffi::Atom> {
let supported_atom = unsafe { self.get_atom_unchecked(b"_NET_SUPPORTED\0") }; let supported_atom = unsafe { self.get_atom_unchecked(b"_NET_SUPPORTED\0") };
self.get_property( self.get_property(root, supported_atom, ffi::XA_ATOM)
root, .unwrap_or_else(|_| Vec::with_capacity(0))
supported_atom,
ffi::XA_ATOM,
).unwrap_or_else(|_| Vec::with_capacity(0))
} }
fn get_wm_name(&self, root: ffi::Window) -> Option<String> { fn get_wm_name(&self, root: ffi::Window) -> Option<String> {
@ -61,15 +58,9 @@ impl XConnection {
// Querying this property on the root window will give us the ID of a child window created by // Querying this property on the root window will give us the ID of a child window created by
// the WM. // the WM.
let root_window_wm_check = { let root_window_wm_check = {
let result = self.get_property( let result = self.get_property(root, check_atom, ffi::XA_WINDOW);
root,
check_atom,
ffi::XA_WINDOW,
);
let wm_check = result let wm_check = result.ok().and_then(|wm_check| wm_check.get(0).cloned());
.ok()
.and_then(|wm_check| wm_check.get(0).cloned());
if let Some(wm_check) = wm_check { if let Some(wm_check) = wm_check {
wm_check wm_check
@ -81,15 +72,9 @@ impl XConnection {
// Querying the same property on the child window we were given, we should get this child // Querying the same property on the child window we were given, we should get this child
// window's ID again. // window's ID again.
let child_window_wm_check = { let child_window_wm_check = {
let result = self.get_property( let result = self.get_property(root_window_wm_check, check_atom, ffi::XA_WINDOW);
root_window_wm_check,
check_atom,
ffi::XA_WINDOW,
);
let wm_check = result let wm_check = result.ok().and_then(|wm_check| wm_check.get(0).cloned());
.ok()
.and_then(|wm_check| wm_check.get(0).cloned());
if let Some(wm_check) = wm_check { if let Some(wm_check) = wm_check {
wm_check wm_check
@ -107,11 +92,7 @@ impl XConnection {
let wm_name = { let wm_name = {
let utf8_string_atom = unsafe { self.get_atom_unchecked(b"UTF8_STRING\0") }; let utf8_string_atom = unsafe { self.get_atom_unchecked(b"UTF8_STRING\0") };
let result = self.get_property( let result = self.get_property(root_window_wm_check, wm_name_atom, utf8_string_atom);
root_window_wm_check,
wm_name_atom,
utf8_string_atom,
);
// IceWM requires this. IceWM was also the only WM tested that returns a null-terminated // IceWM requires this. IceWM was also the only WM tested that returns a null-terminated
// string. For more fun trivia, IceWM is also unique in including version and uname // string. For more fun trivia, IceWM is also unique in including version and uname
@ -126,15 +107,12 @@ impl XConnection {
}; };
if no_utf8 { if no_utf8 {
self.get_property( self.get_property(root_window_wm_check, wm_name_atom, ffi::XA_STRING)
root_window_wm_check,
wm_name_atom,
ffi::XA_STRING,
)
} else { } else {
result result
} }
}.ok(); }
.ok();
wm_name.and_then(|wm_name| String::from_utf8(wm_name).ok()) wm_name.and_then(|wm_name| String::from_utf8(wm_name).ok())
} }

View file

@ -1,23 +1,20 @@
use std::{cmp, env, mem}; use std::{cmp, collections::HashSet, env, ffi::CString, mem, os::raw::*, path::Path, sync::Arc};
use std::collections::HashSet;
use std::ffi::CString;
use std::os::raw::*;
use std::path::Path;
use std::sync::Arc;
use libc; use libc;
use parking_lot::Mutex; use parking_lot::Mutex;
use crate::error::{ExternalError, NotSupportedError, OsError as RootOsError}; use crate::{
use crate::window::{Icon, CursorIcon, WindowAttributes}; dpi::{LogicalPosition, LogicalSize},
use crate::dpi::{LogicalPosition, LogicalSize}; error::{ExternalError, NotSupportedError, OsError as RootOsError},
use crate::platform_impl::MonitorHandle as PlatformMonitorHandle; monitor::MonitorHandle as RootMonitorHandle,
use crate::platform_impl::{OsError, PlatformSpecificWindowBuilderAttributes}; platform_impl::{
use crate::platform_impl::x11::ime::ImeContextCreationError; x11::{ime::ImeContextCreationError, MonitorHandle as X11MonitorHandle},
use crate::platform_impl::x11::MonitorHandle as X11MonitorHandle; MonitorHandle as PlatformMonitorHandle, OsError, PlatformSpecificWindowBuilderAttributes,
use crate::monitor::MonitorHandle as RootMonitorHandle; },
window::{CursorIcon, Icon, WindowAttributes},
};
use super::{ffi, util, ImeSender, XConnection, XError, WindowId, EventLoopWindowTarget}; use super::{ffi, util, EventLoopWindowTarget, ImeSender, WindowId, XConnection, XError};
unsafe extern "C" fn visibility_predicate( unsafe extern "C" fn visibility_predicate(
_display: *mut ffi::Display, _display: *mut ffi::Display,
@ -60,9 +57,9 @@ unsafe impl Sync for UnownedWindow {}
pub struct UnownedWindow { pub struct UnownedWindow {
pub xconn: Arc<XConnection>, // never changes pub xconn: Arc<XConnection>, // never changes
xwindow: ffi::Window, // never changes xwindow: ffi::Window, // never changes
root: ffi::Window, // never changes root: ffi::Window, // never changes
screen_id: i32, // never changes screen_id: i32, // never changes
cursor: Mutex<CursorIcon>, cursor: Mutex<CursorIcon>,
cursor_grabbed: Mutex<bool>, cursor_grabbed: Mutex<bool>,
cursor_visible: Mutex<bool>, cursor_visible: Mutex<bool>,
@ -89,7 +86,8 @@ impl UnownedWindow {
} }
} }
dpi_factor.unwrap_or_else(|| { dpi_factor.unwrap_or_else(|| {
xconn.query_pointer(root, util::VIRTUAL_CORE_POINTER) xconn
.query_pointer(root, util::VIRTUAL_CORE_POINTER)
.ok() .ok()
.and_then(|pointer_state| { .and_then(|pointer_state| {
let (x, y) = (pointer_state.root_x as i64, pointer_state.root_y as i64); let (x, y) = (pointer_state.root_x as i64, pointer_state.root_y as i64);
@ -110,17 +108,18 @@ impl UnownedWindow {
info!("Guessed window DPI factor: {}", dpi_factor); info!("Guessed window DPI factor: {}", dpi_factor);
let max_inner_size: Option<(u32, u32)> = window_attrs.max_inner_size.map(|size| { let max_inner_size: Option<(u32, u32)> = window_attrs
size.to_physical(dpi_factor).into() .max_inner_size
}); .map(|size| size.to_physical(dpi_factor).into());
let min_inner_size: Option<(u32, u32)> = window_attrs.min_inner_size.map(|size| { let min_inner_size: Option<(u32, u32)> = window_attrs
size.to_physical(dpi_factor).into() .min_inner_size
}); .map(|size| size.to_physical(dpi_factor).into());
let dimensions = { let dimensions = {
// x11 only applies constraints when the window is actively resized // x11 only applies constraints when the window is actively resized
// by the user, so we have to manually apply the initial constraints // by the user, so we have to manually apply the initial constraints
let mut dimensions: (u32, u32) = window_attrs.inner_size let mut dimensions: (u32, u32) = window_attrs
.inner_size
.or_else(|| Some((800, 600).into())) .or_else(|| Some((800, 600).into()))
.map(|size| size.to_physical(dpi_factor)) .map(|size| size.to_physical(dpi_factor))
.map(Into::into) .map(Into::into)
@ -133,7 +132,10 @@ impl UnownedWindow {
dimensions.0 = cmp::max(dimensions.0, min.0); dimensions.0 = cmp::max(dimensions.0, min.0);
dimensions.1 = cmp::max(dimensions.1, min.1); dimensions.1 = cmp::max(dimensions.1, min.1);
} }
debug!("Calculated physical dimensions: {}x{}", dimensions.0, dimensions.1); debug!(
"Calculated physical dimensions: {}x{}",
dimensions.0, dimensions.1
);
dimensions dimensions
}; };
@ -150,7 +152,9 @@ impl UnownedWindow {
let visual = vi.visual; let visual = vi.visual;
(xconn.xlib.XCreateColormap)(xconn.display, root, visual, ffi::AllocNone) (xconn.xlib.XCreateColormap)(xconn.display, root, visual, ffi::AllocNone)
} }
} else { 0 }; } else {
0
};
swa.event_mask = ffi::ExposureMask swa.event_mask = ffi::ExposureMask
| ffi::StructureNotifyMask | ffi::StructureNotifyMask
| ffi::VisibilityChangeMask | ffi::VisibilityChangeMask
@ -220,7 +224,9 @@ impl UnownedWindow {
// title to determine placement/etc., so doing this after mapping would cause the WM to // title to determine placement/etc., so doing this after mapping would cause the WM to
// act on the wrong title state. // act on the wrong title state.
window.set_title_inner(&window_attrs.title).queue(); window.set_title_inner(&window_attrs.title).queue();
window.set_decorations_inner(window_attrs.decorations).queue(); window
.set_decorations_inner(window_attrs.decorations)
.queue();
{ {
// Enable drag and drop (TODO: extend API to make this toggleable) // Enable drag and drop (TODO: extend API to make this toggleable)
@ -234,15 +240,16 @@ impl UnownedWindow {
util::PropMode::Replace, util::PropMode::Replace,
version, version,
) )
}.queue(); }
.queue();
// WM_CLASS must be set *before* mapping the window, as per ICCCM! // WM_CLASS must be set *before* mapping the window, as per ICCCM!
{ {
let (class, instance) = if let Some((instance, class)) = pl_attribs.class { let (class, instance) = if let Some((instance, class)) = pl_attribs.class {
let instance = CString::new(instance.as_str()) let instance = CString::new(instance.as_str())
.expect("`WM_CLASS` instance contained null byte"); .expect("`WM_CLASS` instance contained null byte");
let class = CString::new(class.as_str()) let class =
.expect("`WM_CLASS` class contained null byte"); CString::new(class.as_str()).expect("`WM_CLASS` class contained null byte");
(instance, class) (instance, class)
} else { } else {
let class = env::args() let class = env::args()
@ -269,12 +276,8 @@ impl UnownedWindow {
(*class_hint).res_class = instance.as_ptr() as *mut c_char; (*class_hint).res_class = instance.as_ptr() as *mut c_char;
unsafe { unsafe {
(xconn.xlib.XSetClassHint)( (xconn.xlib.XSetClassHint)(xconn.display, window.xwindow, class_hint.ptr);
xconn.display, } //.queue();
window.xwindow,
class_hint.ptr,
);
}//.queue();
} }
window.set_pid().map(|flusher| flusher.queue()); window.set_pid().map(|flusher| flusher.queue());
@ -289,9 +292,11 @@ impl UnownedWindow {
// set size hints // set size hints
{ {
let mut min_inner_size = window_attrs.min_inner_size let mut min_inner_size = window_attrs
.min_inner_size
.map(|size| size.to_physical(dpi_factor)); .map(|size| size.to_physical(dpi_factor));
let mut max_inner_size = window_attrs.max_inner_size let mut max_inner_size = window_attrs
.max_inner_size
.map(|size| size.to_physical(dpi_factor)); .map(|size| size.to_physical(dpi_factor));
if !window_attrs.resizable { if !window_attrs.resizable {
if util::wm_name_is_one_of(&["Xfwm4"]) { if util::wm_name_is_one_of(&["Xfwm4"]) {
@ -328,13 +333,13 @@ impl UnownedWindow {
&event_loop.wm_delete_window as *const ffi::Atom as *mut ffi::Atom, &event_loop.wm_delete_window as *const ffi::Atom as *mut ffi::Atom,
1, 1,
); );
}//.queue(); } //.queue();
// Set visibility (map window) // Set visibility (map window)
if window_attrs.visible { if window_attrs.visible {
unsafe { unsafe {
(xconn.xlib.XMapRaised)(xconn.display, window.xwindow); (xconn.xlib.XMapRaised)(xconn.display, window.xwindow);
}//.queue(); } //.queue();
} }
// Attempt to make keyboard input repeat detectable // Attempt to make keyboard input repeat detectable
@ -346,7 +351,9 @@ impl UnownedWindow {
&mut supported_ptr, &mut supported_ptr,
); );
if supported_ptr == ffi::False { if supported_ptr == ffi::False {
return Err(os_error!(OsError::XMisc("`XkbSetDetectableAutoRepeat` failed"))); return Err(os_error!(OsError::XMisc(
"`XkbSetDetectableAutoRepeat` failed"
)));
} }
} }
@ -366,16 +373,18 @@ impl UnownedWindow {
| ffi::XI_TouchEndMask; | ffi::XI_TouchEndMask;
mask mask
}; };
xconn.select_xinput_events(window.xwindow, ffi::XIAllMasterDevices, mask).queue(); xconn
.select_xinput_events(window.xwindow, ffi::XIAllMasterDevices, mask)
.queue();
{ {
let result = event_loop.ime let result = event_loop.ime.borrow_mut().create_context(window.xwindow);
.borrow_mut()
.create_context(window.xwindow);
if let Err(err) = result { if let Err(err) = result {
let e = match err { let e = match err {
ImeContextCreationError::XError(err) => OsError::XError(err), ImeContextCreationError::XError(err) => OsError::XError(err),
ImeContextCreationError::Null => OsError::XMisc("IME Context creation failed"), ImeContextCreationError::Null => {
OsError::XMisc("IME Context creation failed")
},
}; };
return Err(os_error!(e)); return Err(os_error!(e));
} }
@ -386,10 +395,14 @@ impl UnownedWindow {
window.set_maximized_inner(window_attrs.maximized).queue(); window.set_maximized_inner(window_attrs.maximized).queue();
} }
if window_attrs.fullscreen.is_some() { if window_attrs.fullscreen.is_some() {
window.set_fullscreen_inner(window_attrs.fullscreen.clone()).queue(); window
.set_fullscreen_inner(window_attrs.fullscreen.clone())
.queue();
} }
if window_attrs.always_on_top { if window_attrs.always_on_top {
window.set_always_on_top_inner(window_attrs.always_on_top).queue(); window
.set_always_on_top_inner(window_attrs.always_on_top)
.queue();
} }
if window_attrs.visible { if window_attrs.visible {
@ -397,7 +410,8 @@ impl UnownedWindow {
// XSetInputFocus generates an error if the window is not visible, so we wait // XSetInputFocus generates an error if the window is not visible, so we wait
// until we receive VisibilityNotify. // until we receive VisibilityNotify.
let mut event = mem::uninitialized(); let mut event = mem::uninitialized();
(xconn.xlib.XIfEvent)( // This will flush the request buffer IF it blocks. (xconn.xlib.XIfEvent)(
// This will flush the request buffer IF it blocks.
xconn.display, xconn.display,
&mut event as *mut ffi::XEvent, &mut event as *mut ffi::XEvent,
Some(visibility_predicate), Some(visibility_predicate),
@ -414,7 +428,8 @@ impl UnownedWindow {
} }
// We never want to give the user a broken window, since by then, it's too late to handle. // We never want to give the user a broken window, since by then, it's too late to handle.
xconn.sync_with_server() xconn
.sync_with_server()
.map(|_| window) .map(|_| window)
.map_err(|x_err| os_error!(OsError::XError(x_err))) .map_err(|x_err| os_error!(OsError::XError(x_err)))
} }
@ -439,18 +454,22 @@ impl UnownedWindow {
const MAXHOSTNAMELEN: usize = 256; const MAXHOSTNAMELEN: usize = 256;
let mut hostname: [c_char; MAXHOSTNAMELEN] = mem::uninitialized(); let mut hostname: [c_char; MAXHOSTNAMELEN] = mem::uninitialized();
let status = libc::gethostname(hostname.as_mut_ptr(), hostname.len()); let status = libc::gethostname(hostname.as_mut_ptr(), hostname.len());
if status != 0 { return None; } if status != 0 {
return None;
}
hostname[MAXHOSTNAMELEN - 1] = '\0' as c_char; // a little extra safety hostname[MAXHOSTNAMELEN - 1] = '\0' as c_char; // a little extra safety
let hostname_length = libc::strlen(hostname.as_ptr()); let hostname_length = libc::strlen(hostname.as_ptr());
(hostname, hostname_length as usize) (hostname, hostname_length as usize)
}; };
self.xconn.change_property( self.xconn
self.xwindow, .change_property(
pid_atom, self.xwindow,
ffi::XA_CARDINAL, pid_atom,
util::PropMode::Replace, ffi::XA_CARDINAL,
&[libc::getpid() as util::Cardinal], util::PropMode::Replace,
).queue(); &[libc::getpid() as util::Cardinal],
)
.queue();
let flusher = self.xconn.change_property( let flusher = self.xconn.change_property(
self.xwindow, self.xwindow,
client_machine_atom, client_machine_atom,
@ -489,13 +508,19 @@ impl UnownedWindow {
#[inline] #[inline]
pub fn set_urgent(&self, is_urgent: bool) { pub fn set_urgent(&self, is_urgent: bool) {
let mut wm_hints = self.xconn.get_wm_hints(self.xwindow).expect("`XGetWMHints` failed"); let mut wm_hints = self
.xconn
.get_wm_hints(self.xwindow)
.expect("`XGetWMHints` failed");
if is_urgent { if is_urgent {
(*wm_hints).flags |= ffi::XUrgencyHint; (*wm_hints).flags |= ffi::XUrgencyHint;
} else { } else {
(*wm_hints).flags &= !ffi::XUrgencyHint; (*wm_hints).flags &= !ffi::XUrgencyHint;
} }
self.xconn.set_wm_hints(self.xwindow, wm_hints).flush().expect("Failed to set urgency hint"); self.xconn
.set_wm_hints(self.xwindow, wm_hints)
.flush()
.expect("Failed to set urgency hint");
} }
fn set_netwm( fn set_netwm(
@ -520,7 +545,8 @@ impl UnownedWindow {
} }
fn set_fullscreen_hint(&self, fullscreen: bool) -> util::Flusher<'_> { fn set_fullscreen_hint(&self, fullscreen: bool) -> util::Flusher<'_> {
let fullscreen_atom = unsafe { self.xconn.get_atom_unchecked(b"_NET_WM_STATE_FULLSCREEN\0") }; let fullscreen_atom =
unsafe { self.xconn.get_atom_unchecked(b"_NET_WM_STATE_FULLSCREEN\0") };
self.set_netwm(fullscreen.into(), (fullscreen_atom as c_long, 0, 0, 0)) self.set_netwm(fullscreen.into(), (fullscreen_atom as c_long, 0, 0, 0))
} }
@ -533,13 +559,16 @@ impl UnownedWindow {
} }
flusher flusher
}, },
Some(RootMonitorHandle { inner: PlatformMonitorHandle::X(monitor) }) => { Some(RootMonitorHandle {
inner: PlatformMonitorHandle::X(monitor),
}) => {
let window_position = self.outer_position_physical(); let window_position = self.outer_position_physical();
self.shared_state.lock().restore_position = Some(window_position); self.shared_state.lock().restore_position = Some(window_position);
let monitor_origin: (i32, i32) = monitor.position().into(); let monitor_origin: (i32, i32) = monitor.position().into();
self.set_position_inner(monitor_origin.0, monitor_origin.1).queue(); self.set_position_inner(monitor_origin.0, monitor_origin.1)
.queue();
self.set_fullscreen_hint(true) self.set_fullscreen_hint(true)
} },
_ => unreachable!(), _ => unreachable!(),
} }
} }
@ -567,17 +596,15 @@ impl UnownedWindow {
#[inline] #[inline]
pub fn current_monitor(&self) -> X11MonitorHandle { pub fn current_monitor(&self) -> X11MonitorHandle {
let monitor = self.shared_state let monitor = self.shared_state.lock().last_monitor.as_ref().cloned();
.lock() monitor.unwrap_or_else(|| {
.last_monitor let monitor = self
.as_ref() .xconn
.cloned(); .get_monitor_for_window(Some(self.get_rect()))
monitor .to_owned();
.unwrap_or_else(|| { self.shared_state.lock().last_monitor = Some(monitor.clone());
let monitor = self.xconn.get_monitor_for_window(Some(self.get_rect())).to_owned(); monitor
self.shared_state.lock().last_monitor = Some(monitor.clone()); })
monitor
})
} }
pub fn available_monitors(&self) -> Vec<X11MonitorHandle> { pub fn available_monitors(&self) -> Vec<X11MonitorHandle> {
@ -589,9 +616,18 @@ impl UnownedWindow {
} }
fn set_maximized_inner(&self, maximized: bool) -> util::Flusher<'_> { fn set_maximized_inner(&self, maximized: bool) -> util::Flusher<'_> {
let horz_atom = unsafe { self.xconn.get_atom_unchecked(b"_NET_WM_STATE_MAXIMIZED_HORZ\0") }; let horz_atom = unsafe {
let vert_atom = unsafe { self.xconn.get_atom_unchecked(b"_NET_WM_STATE_MAXIMIZED_VERT\0") }; self.xconn
self.set_netwm(maximized.into(), (horz_atom as c_long, vert_atom as c_long, 0, 0)) .get_atom_unchecked(b"_NET_WM_STATE_MAXIMIZED_HORZ\0")
};
let vert_atom = unsafe {
self.xconn
.get_atom_unchecked(b"_NET_WM_STATE_MAXIMIZED_VERT\0")
};
self.set_netwm(
maximized.into(),
(horz_atom as c_long, vert_atom as c_long, 0, 0),
)
} }
#[inline] #[inline]
@ -638,10 +674,10 @@ impl UnownedWindow {
util::PropMode::Replace, util::PropMode::Replace,
&[ &[
util::MWM_HINTS_DECORATIONS, // flags util::MWM_HINTS_DECORATIONS, // flags
0, // functions 0, // functions
decorations as c_ulong, // decorations decorations as c_ulong, // decorations
0, // input mode 0, // input mode
0, // status 0, // status
], ],
) )
} }
@ -695,7 +731,9 @@ impl UnownedWindow {
match icon { match icon {
Some(icon) => self.set_icon_inner(icon), Some(icon) => self.set_icon_inner(icon),
None => self.unset_icon_inner(), None => self.unset_icon_inner(),
}.flush().expect("Failed to set icons"); }
.flush()
.expect("Failed to set icons");
} }
#[inline] #[inline]
@ -703,19 +741,23 @@ impl UnownedWindow {
match visible { match visible {
true => unsafe { true => unsafe {
(self.xconn.xlib.XMapRaised)(self.xconn.display, self.xwindow); (self.xconn.xlib.XMapRaised)(self.xconn.display, self.xwindow);
self.xconn.flush_requests() self.xconn
.flush_requests()
.expect("Failed to call XMapRaised"); .expect("Failed to call XMapRaised");
}, },
false => unsafe { false => unsafe {
(self.xconn.xlib.XUnmapWindow)(self.xconn.display, self.xwindow); (self.xconn.xlib.XUnmapWindow)(self.xconn.display, self.xwindow);
self.xconn.flush_requests() self.xconn
.flush_requests()
.expect("Failed to call XUnmapWindow"); .expect("Failed to call XUnmapWindow");
} },
} }
} }
fn update_cached_frame_extents(&self) { fn update_cached_frame_extents(&self) {
let extents = self.xconn.get_frame_extents_heuristic(self.xwindow, self.root); let extents = self
.xconn
.get_frame_extents_heuristic(self.xwindow, self.root);
(*self.shared_state.lock()).frame_extents = Some(extents); (*self.shared_state.lock()).frame_extents = Some(extents);
} }
@ -749,7 +791,8 @@ impl UnownedWindow {
pub(crate) fn inner_position_physical(&self) -> (i32, i32) { pub(crate) fn inner_position_physical(&self) -> (i32, i32) {
// This should be okay to unwrap since the only error XTranslateCoordinates can return // This should be okay to unwrap since the only error XTranslateCoordinates can return
// is BadWindow, and if the window handle is bad we have bigger problems. // is BadWindow, and if the window handle is bad we have bigger problems.
self.xconn.translate_coords(self.xwindow, self.root) self.xconn
.translate_coords(self.xwindow, self.root)
.map(|coords| (coords.x_rel_root, coords.y_rel_root)) .map(|coords| (coords.x_rel_root, coords.y_rel_root))
.unwrap() .unwrap()
} }
@ -773,12 +816,7 @@ impl UnownedWindow {
} }
} }
unsafe { unsafe {
(self.xconn.xlib.XMoveWindow)( (self.xconn.xlib.XMoveWindow)(self.xconn.display, self.xwindow, x as c_int, y as c_int);
self.xconn.display,
self.xwindow,
x as c_int,
y as c_int,
);
} }
util::Flusher::new(&self.xconn) util::Flusher::new(&self.xconn)
} }
@ -798,7 +836,8 @@ impl UnownedWindow {
pub(crate) fn inner_size_physical(&self) -> (u32, u32) { pub(crate) fn inner_size_physical(&self) -> (u32, u32) {
// This should be okay to unwrap since the only error XGetGeometry can return // This should be okay to unwrap since the only error XGetGeometry can return
// is BadWindow, and if the window handle is bad we have bigger problems. // is BadWindow, and if the window handle is bad we have bigger problems.
self.xconn.get_geometry(self.xwindow) self.xconn
.get_geometry(self.xwindow)
.map(|geo| (geo.width, geo.height)) .map(|geo| (geo.width, geo.height))
.unwrap() .unwrap()
} }
@ -840,7 +879,8 @@ impl UnownedWindow {
height as c_uint, height as c_uint,
); );
self.xconn.flush_requests() self.xconn.flush_requests()
}.expect("Failed to call `XResizeWindow`"); }
.expect("Failed to call `XResizeWindow`");
} }
#[inline] #[inline]
@ -851,11 +891,14 @@ impl UnownedWindow {
} }
fn update_normal_hints<F>(&self, callback: F) -> Result<(), XError> fn update_normal_hints<F>(&self, callback: F) -> Result<(), XError>
where F: FnOnce(&mut util::NormalHints<'_>) -> () where
F: FnOnce(&mut util::NormalHints<'_>) -> (),
{ {
let mut normal_hints = self.xconn.get_normal_hints(self.xwindow)?; let mut normal_hints = self.xconn.get_normal_hints(self.xwindow)?;
callback(&mut normal_hints); callback(&mut normal_hints);
self.xconn.set_normal_hints(self.xwindow, normal_hints).flush() self.xconn
.set_normal_hints(self.xwindow, normal_hints)
.flush()
} }
pub(crate) fn set_min_inner_size_physical(&self, dimensions: Option<(u32, u32)>) { pub(crate) fn set_min_inner_size_physical(&self, dimensions: Option<(u32, u32)>) {
@ -866,9 +909,8 @@ impl UnownedWindow {
#[inline] #[inline]
pub fn set_min_inner_size(&self, logical_dimensions: Option<LogicalSize>) { pub fn set_min_inner_size(&self, logical_dimensions: Option<LogicalSize>) {
self.shared_state.lock().min_inner_size = logical_dimensions; self.shared_state.lock().min_inner_size = logical_dimensions;
let physical_dimensions = logical_dimensions.map(|logical_dimensions| { let physical_dimensions = logical_dimensions
logical_dimensions.to_physical(self.hidpi_factor()).into() .map(|logical_dimensions| logical_dimensions.to_physical(self.hidpi_factor()).into());
});
self.set_min_inner_size_physical(physical_dimensions); self.set_min_inner_size_physical(physical_dimensions);
} }
@ -880,9 +922,8 @@ impl UnownedWindow {
#[inline] #[inline]
pub fn set_max_inner_size(&self, logical_dimensions: Option<LogicalSize>) { pub fn set_max_inner_size(&self, logical_dimensions: Option<LogicalSize>) {
self.shared_state.lock().max_inner_size = logical_dimensions; self.shared_state.lock().max_inner_size = logical_dimensions;
let physical_dimensions = logical_dimensions.map(|logical_dimensions| { let physical_dimensions = logical_dimensions
logical_dimensions.to_physical(self.hidpi_factor()).into() .map(|logical_dimensions| logical_dimensions.to_physical(self.hidpi_factor()).into());
});
self.set_max_inner_size_physical(physical_dimensions); self.set_max_inner_size_physical(physical_dimensions);
} }
@ -910,7 +951,8 @@ impl UnownedWindow {
normal_hints.set_min_size(min_size); normal_hints.set_min_size(min_size);
normal_hints.set_resize_increments(resize_increments); normal_hints.set_resize_increments(resize_increments);
normal_hints.set_base_size(base_size); normal_hints.set_base_size(base_size);
}).expect("Failed to update normal hints"); })
.expect("Failed to update normal hints");
unsafe { unsafe {
(self.xconn.xlib.XResizeWindow)( (self.xconn.xlib.XResizeWindow)(
self.xconn.display, self.xconn.display,
@ -933,7 +975,10 @@ impl UnownedWindow {
let (logical_min, logical_max) = if resizable { let (logical_min, logical_max) = if resizable {
let shared_state_lock = self.shared_state.lock(); let shared_state_lock = self.shared_state.lock();
(shared_state_lock.min_inner_size, shared_state_lock.max_inner_size) (
shared_state_lock.min_inner_size,
shared_state_lock.max_inner_size,
)
} else { } else {
let window_size = Some(self.inner_size()); let window_size = Some(self.inner_size());
(window_size.clone(), window_size) (window_size.clone(), window_size)
@ -949,7 +994,8 @@ impl UnownedWindow {
self.update_normal_hints(|normal_hints| { self.update_normal_hints(|normal_hints| {
normal_hints.set_min_size(min_inner_size); normal_hints.set_min_size(min_inner_size);
normal_hints.set_max_size(max_inner_size); normal_hints.set_max_size(max_inner_size);
}).expect("Failed to call `XSetWMNormalHints`"); })
.expect("Failed to call `XSetWMNormalHints`");
} }
#[inline] #[inline]
@ -974,9 +1020,7 @@ impl UnownedWindow {
#[inline] #[inline]
pub fn xcb_connection(&self) -> *mut c_void { pub fn xcb_connection(&self) -> *mut c_void {
unsafe { unsafe { (self.xconn.xlib_xcb.XGetXCBConnection)(self.xconn.display) as *mut _ }
(self.xconn.xlib_xcb.XGetXCBConnection)(self.xconn.display) as *mut _
}
} }
fn load_cursor(&self, name: &[u8]) -> ffi::Cursor { fn load_cursor(&self, name: &[u8]) -> ffi::Cursor {
@ -999,13 +1043,9 @@ impl UnownedWindow {
} }
fn get_cursor(&self, cursor: CursorIcon) -> ffi::Cursor { fn get_cursor(&self, cursor: CursorIcon) -> ffi::Cursor {
let load = |name: &[u8]| { let load = |name: &[u8]| self.load_cursor(name);
self.load_cursor(name)
};
let loadn = |names: &[&[u8]]| { let loadn = |names: &[&[u8]]| self.load_first_existing_cursor(names);
self.load_first_existing_cursor(names)
};
// Try multiple names in some cases where the name // Try multiple names in some cases where the name
// differs on the desktop environments or themes. // differs on the desktop environments or themes.
@ -1030,7 +1070,6 @@ impl UnownedWindow {
CursorIcon::NoDrop => loadn(&[b"no-drop\0", b"circle\0"]), CursorIcon::NoDrop => loadn(&[b"no-drop\0", b"circle\0"]),
CursorIcon::NotAllowed => load(b"crossed_circle\0"), CursorIcon::NotAllowed => load(b"crossed_circle\0"),
// Resize cursors // Resize cursors
CursorIcon::EResize => load(b"right_side\0"), CursorIcon::EResize => load(b"right_side\0"),
CursorIcon::NResize => load(b"top_side\0"), CursorIcon::NResize => load(b"top_side\0"),
@ -1063,7 +1102,9 @@ impl UnownedWindow {
if cursor != 0 { if cursor != 0 {
(self.xconn.xlib.XFreeCursor)(self.xconn.display, cursor); (self.xconn.xlib.XFreeCursor)(self.xconn.display, cursor);
} }
self.xconn.flush_requests().expect("Failed to set or free the cursor"); self.xconn
.flush_requests()
.expect("Failed to set or free the cursor");
} }
} }
@ -1081,13 +1122,7 @@ impl UnownedWindow {
fn create_empty_cursor(&self) -> Option<ffi::Cursor> { fn create_empty_cursor(&self) -> Option<ffi::Cursor> {
let data = 0; let data = 0;
let pixmap = unsafe { let pixmap = unsafe {
(self.xconn.xlib.XCreateBitmapFromData)( (self.xconn.xlib.XCreateBitmapFromData)(self.xconn.display, self.xwindow, &data, 1, 1)
self.xconn.display,
self.xwindow,
&data,
1,
1,
)
}; };
if pixmap == 0 { if pixmap == 0 {
// Failed to allocate // Failed to allocate
@ -1116,7 +1151,9 @@ impl UnownedWindow {
#[inline] #[inline]
pub fn set_cursor_grab(&self, grab: bool) -> Result<(), ExternalError> { pub fn set_cursor_grab(&self, grab: bool) -> Result<(), ExternalError> {
let mut grabbed_lock = self.cursor_grabbed.lock(); let mut grabbed_lock = self.cursor_grabbed.lock();
if grab == *grabbed_lock { return Ok(()); } if grab == *grabbed_lock {
return Ok(());
}
unsafe { unsafe {
// We ungrab before grabbing to prevent passive grabs from causing `AlreadyGrabbed`. // We ungrab before grabbing to prevent passive grabs from causing `AlreadyGrabbed`.
// Therefore, this is common to both codepaths. // Therefore, this is common to both codepaths.
@ -1128,8 +1165,7 @@ impl UnownedWindow {
self.xconn.display, self.xconn.display,
self.xwindow, self.xwindow,
ffi::True, ffi::True,
( (ffi::ButtonPressMask
ffi::ButtonPressMask
| ffi::ButtonReleaseMask | ffi::ButtonReleaseMask
| ffi::EnterWindowMask | ffi::EnterWindowMask
| ffi::LeaveWindowMask | ffi::LeaveWindowMask
@ -1141,8 +1177,7 @@ impl UnownedWindow {
| ffi::Button4MotionMask | ffi::Button4MotionMask
| ffi::Button5MotionMask | ffi::Button5MotionMask
| ffi::ButtonMotionMask | ffi::ButtonMotionMask
| ffi::KeymapStateMask | ffi::KeymapStateMask) as c_uint,
) as c_uint,
ffi::GrabModeAsync, ffi::GrabModeAsync,
ffi::GrabModeAsync, ffi::GrabModeAsync,
self.xwindow, self.xwindow,
@ -1153,14 +1188,20 @@ impl UnownedWindow {
match result { match result {
ffi::GrabSuccess => Ok(()), ffi::GrabSuccess => Ok(()),
ffi::AlreadyGrabbed => Err("Cursor could not be grabbed: already grabbed by another client"), ffi::AlreadyGrabbed => {
Err("Cursor could not be grabbed: already grabbed by another client")
},
ffi::GrabInvalidTime => Err("Cursor could not be grabbed: invalid time"), ffi::GrabInvalidTime => Err("Cursor could not be grabbed: invalid time"),
ffi::GrabNotViewable => Err("Cursor could not be grabbed: grab location not viewable"), ffi::GrabNotViewable => {
Err("Cursor could not be grabbed: grab location not viewable")
},
ffi::GrabFrozen => Err("Cursor could not be grabbed: frozen by another client"), ffi::GrabFrozen => Err("Cursor could not be grabbed: frozen by another client"),
_ => unreachable!(), _ => unreachable!(),
}.map_err(|err| ExternalError::Os(os_error!(OsError::XMisc(err)))) }
.map_err(|err| ExternalError::Os(os_error!(OsError::XMisc(err))))
} else { } else {
self.xconn.flush_requests() self.xconn
.flush_requests()
.map_err(|err| ExternalError::Os(os_error!(OsError::XError(err)))) .map_err(|err| ExternalError::Os(os_error!(OsError::XError(err))))
}; };
if result.is_ok() { if result.is_ok() {
@ -1172,11 +1213,14 @@ impl UnownedWindow {
#[inline] #[inline]
pub fn set_cursor_visible(&self, visible: bool) { pub fn set_cursor_visible(&self, visible: bool) {
let mut visible_lock = self.cursor_visible.lock(); let mut visible_lock = self.cursor_visible.lock();
if visible == *visible_lock {return; } if visible == *visible_lock {
return;
}
let cursor = if visible { let cursor = if visible {
self.get_cursor(*self.cursor.lock()) self.get_cursor(*self.cursor.lock())
} else { } else {
self.create_empty_cursor().expect("Failed to create empty cursor") self.create_empty_cursor()
.expect("Failed to create empty cursor")
}; };
*visible_lock = visible; *visible_lock = visible;
drop(visible_lock); drop(visible_lock);
@ -1190,29 +1234,25 @@ impl UnownedWindow {
pub fn set_cursor_position_physical(&self, x: i32, y: i32) -> Result<(), ExternalError> { pub fn set_cursor_position_physical(&self, x: i32, y: i32) -> Result<(), ExternalError> {
unsafe { unsafe {
(self.xconn.xlib.XWarpPointer)( (self.xconn.xlib.XWarpPointer)(self.xconn.display, 0, self.xwindow, 0, 0, 0, 0, x, y);
self.xconn.display, self.xconn
0, .flush_requests()
self.xwindow, .map_err(|e| ExternalError::Os(os_error!(OsError::XError(e))))
0,
0,
0,
0,
x,
y,
);
self.xconn.flush_requests().map_err(|e| ExternalError::Os(os_error!(OsError::XError(e))))
} }
} }
#[inline] #[inline]
pub fn set_cursor_position(&self, logical_position: LogicalPosition) -> Result<(), ExternalError> { pub fn set_cursor_position(
&self,
logical_position: LogicalPosition,
) -> Result<(), ExternalError> {
let (x, y) = logical_position.to_physical(self.hidpi_factor()).into(); let (x, y) = logical_position.to_physical(self.hidpi_factor()).into();
self.set_cursor_position_physical(x, y) self.set_cursor_position_physical(x, y)
} }
pub(crate) fn set_ime_position_physical(&self, x: i32, y: i32) { pub(crate) fn set_ime_position_physical(&self, x: i32, y: i32) {
let _ = self.ime_sender let _ = self
.ime_sender
.lock() .lock()
.send((self.xwindow, x as i16, y as i16)); .send((self.xwindow, x as i16, y as i16));
} }
@ -1224,10 +1264,15 @@ impl UnownedWindow {
} }
#[inline] #[inline]
pub fn id(&self) -> WindowId { WindowId(self.xwindow) } pub fn id(&self) -> WindowId {
WindowId(self.xwindow)
}
#[inline] #[inline]
pub fn request_redraw(&self) { pub fn request_redraw(&self) {
self.pending_redraws.lock().unwrap().insert(WindowId(self.xwindow)); self.pending_redraws
.lock()
.unwrap()
.insert(WindowId(self.xwindow));
} }
} }

View file

@ -1,7 +1,4 @@
use std::ptr; use std::{error::Error, fmt, os::raw::c_int, ptr};
use std::fmt;
use std::error::Error;
use std::os::raw::c_int;
use libc; use libc;
use parking_lot::Mutex; use parking_lot::Mutex;
@ -27,7 +24,8 @@ pub struct XConnection {
unsafe impl Send for XConnection {} unsafe impl Send for XConnection {}
unsafe impl Sync for XConnection {} unsafe impl Sync for XConnection {}
pub type XErrorHandler = Option<unsafe extern fn(*mut ffi::Display, *mut ffi::XErrorEvent) -> libc::c_int>; pub type XErrorHandler =
Option<unsafe extern "C" fn(*mut ffi::Display, *mut ffi::XErrorEvent) -> libc::c_int>;
impl XConnection { impl XConnection {
pub fn new(error_handler: XErrorHandler) -> Result<XConnection, XNotSupported> { pub fn new(error_handler: XErrorHandler) -> Result<XConnection, XNotSupported> {
@ -53,9 +51,7 @@ impl XConnection {
}; };
// Get X11 socket file descriptor // Get X11 socket file descriptor
let fd = unsafe { let fd = unsafe { (xlib.XConnectionNumber)(display) };
(xlib.XConnectionNumber)(display)
};
Ok(XConnection { Ok(XConnection {
xlib, xlib,
@ -120,8 +116,11 @@ impl Error for XError {
impl fmt::Display for XError { impl fmt::Display for XError {
fn fmt(&self, formatter: &mut fmt::Formatter<'_>) -> Result<(), fmt::Error> { fn fmt(&self, formatter: &mut fmt::Formatter<'_>) -> Result<(), fmt::Error> {
write!(formatter, "X error: {} (code: {}, request code: {}, minor code: {})", write!(
self.description, self.error_code, self.request_code, self.minor_code) formatter,
"X error: {} (code: {}, request code: {}, minor code: {})",
self.description, self.error_code, self.request_code, self.minor_code
)
} }
} }
@ -131,7 +130,7 @@ pub enum XNotSupported {
/// Failed to load one or several shared libraries. /// Failed to load one or several shared libraries.
LibraryOpenError(ffi::OpenError), LibraryOpenError(ffi::OpenError),
/// Connecting to the X server with `XOpenDisplay` failed. /// Connecting to the X server with `XOpenDisplay` failed.
XOpenDisplayFailed, // TODO: add better message XOpenDisplayFailed, // TODO: add better message
} }
impl From<ffi::OpenError> for XNotSupported { impl From<ffi::OpenError> for XNotSupported {
@ -154,7 +153,7 @@ impl Error for XNotSupported {
fn cause(&self) -> Option<&dyn Error> { fn cause(&self) -> Option<&dyn Error> {
match *self { match *self {
XNotSupported::LibraryOpenError(ref err) => Some(err), XNotSupported::LibraryOpenError(ref err) => Some(err),
_ => None _ => None,
} }
} }
} }

View file

@ -1,10 +1,18 @@
use std::collections::VecDeque; use std::collections::VecDeque;
use cocoa::{appkit::{self, NSEvent}, base::id}; use cocoa::{
use objc::{declare::ClassDecl, runtime::{Class, Object, Sel}}; appkit::{self, NSEvent},
base::id,
};
use objc::{
declare::ClassDecl,
runtime::{Class, Object, Sel},
};
use crate::event::{DeviceEvent, Event}; use crate::{
use crate::platform_impl::platform::{app_state::AppState, DEVICE_ID, util}; event::{DeviceEvent, Event},
platform_impl::platform::{app_state::AppState, util, DEVICE_ID},
};
pub struct AppClass(pub *const Class); pub struct AppClass(pub *const Class);
unsafe impl Send for AppClass {} unsafe impl Send for AppClass {}
@ -17,7 +25,7 @@ lazy_static! {
decl.add_method( decl.add_method(
sel!(sendEvent:), sel!(sendEvent:),
send_event as extern fn(&Object, Sel, id), send_event as extern "C" fn(&Object, Sel, id),
); );
AppClass(decl.register()) AppClass(decl.register())
@ -27,23 +35,25 @@ lazy_static! {
// Normally, holding Cmd + any key never sends us a `keyUp` event for that key. // Normally, holding Cmd + any key never sends us a `keyUp` event for that key.
// Overriding `sendEvent:` like this fixes that. (https://stackoverflow.com/a/15294196) // Overriding `sendEvent:` like this fixes that. (https://stackoverflow.com/a/15294196)
// Fun fact: Firefox still has this bug! (https://bugzilla.mozilla.org/show_bug.cgi?id=1299553) // Fun fact: Firefox still has this bug! (https://bugzilla.mozilla.org/show_bug.cgi?id=1299553)
extern fn send_event(this: &Object, _sel: Sel, event: id) { extern "C" fn send_event(this: &Object, _sel: Sel, event: id) {
unsafe { unsafe {
// For posterity, there are some undocumented event types // For posterity, there are some undocumented event types
// (https://github.com/servo/cocoa-rs/issues/155) // (https://github.com/servo/cocoa-rs/issues/155)
// but that doesn't really matter here. // but that doesn't really matter here.
let event_type = event.eventType(); let event_type = event.eventType();
let modifier_flags = event.modifierFlags(); let modifier_flags = event.modifierFlags();
if event_type == appkit::NSKeyUp && util::has_flag( if event_type == appkit::NSKeyUp
modifier_flags, && util::has_flag(
appkit::NSEventModifierFlags::NSCommandKeyMask, modifier_flags,
) { appkit::NSEventModifierFlags::NSCommandKeyMask,
)
{
let key_window: id = msg_send![this, keyWindow]; let key_window: id = msg_send![this, keyWindow];
let _: () = msg_send![key_window, sendEvent:event]; let _: () = msg_send![key_window, sendEvent: event];
} else { } else {
maybe_dispatch_device_event(event); maybe_dispatch_device_event(event);
let superclass = util::superclass(this); let superclass = util::superclass(this);
let _: () = msg_send![super(this, superclass), sendEvent:event]; let _: () = msg_send![super(this, superclass), sendEvent: event];
} }
} }
} }
@ -51,10 +61,10 @@ extern fn send_event(this: &Object, _sel: Sel, event: id) {
unsafe fn maybe_dispatch_device_event(event: id) { unsafe fn maybe_dispatch_device_event(event: id) {
let event_type = event.eventType(); let event_type = event.eventType();
match event_type { match event_type {
appkit::NSMouseMoved | appkit::NSMouseMoved
appkit::NSLeftMouseDragged | | appkit::NSLeftMouseDragged
appkit::NSOtherMouseDragged | | appkit::NSOtherMouseDragged
appkit::NSRightMouseDragged => { | appkit::NSRightMouseDragged => {
let mut events = VecDeque::with_capacity(3); let mut events = VecDeque::with_capacity(3);
let delta_x = event.deltaX() as f64; let delta_x = event.deltaX() as f64;
@ -63,21 +73,29 @@ unsafe fn maybe_dispatch_device_event(event: id) {
if delta_x != 0.0 { if delta_x != 0.0 {
events.push_back(Event::DeviceEvent { events.push_back(Event::DeviceEvent {
device_id: DEVICE_ID, device_id: DEVICE_ID,
event: DeviceEvent::Motion { axis: 0, value: delta_x }, event: DeviceEvent::Motion {
axis: 0,
value: delta_x,
},
}); });
} }
if delta_y != 0.0 { if delta_y != 0.0 {
events.push_back(Event::DeviceEvent { events.push_back(Event::DeviceEvent {
device_id: DEVICE_ID, device_id: DEVICE_ID,
event: DeviceEvent::Motion { axis: 1, value: delta_y }, event: DeviceEvent::Motion {
axis: 1,
value: delta_y,
},
}); });
} }
if delta_x != 0.0 || delta_y != 0.0 { if delta_x != 0.0 || delta_y != 0.0 {
events.push_back(Event::DeviceEvent { events.push_back(Event::DeviceEvent {
device_id: DEVICE_ID, device_id: DEVICE_ID,
event: DeviceEvent::MouseMotion { delta: (delta_x, delta_y) }, event: DeviceEvent::MouseMotion {
delta: (delta_x, delta_y),
},
}); });
} }

View file

@ -1,5 +1,8 @@
use cocoa::base::id; use cocoa::base::id;
use objc::{runtime::{Class, Object, Sel, BOOL, YES}, declare::ClassDecl}; use objc::{
declare::ClassDecl,
runtime::{Class, Object, Sel, BOOL, YES},
};
use crate::platform_impl::platform::app_state::AppState; use crate::platform_impl::platform::app_state::AppState;
@ -14,41 +17,41 @@ lazy_static! {
decl.add_method( decl.add_method(
sel!(applicationDidFinishLaunching:), sel!(applicationDidFinishLaunching:),
did_finish_launching as extern fn(&Object, Sel, id) -> BOOL, did_finish_launching as extern "C" fn(&Object, Sel, id) -> BOOL,
); );
decl.add_method( decl.add_method(
sel!(applicationDidBecomeActive:), sel!(applicationDidBecomeActive:),
did_become_active as extern fn(&Object, Sel, id), did_become_active as extern "C" fn(&Object, Sel, id),
); );
decl.add_method( decl.add_method(
sel!(applicationWillResignActive:), sel!(applicationWillResignActive:),
will_resign_active as extern fn(&Object, Sel, id), will_resign_active as extern "C" fn(&Object, Sel, id),
); );
decl.add_method( decl.add_method(
sel!(applicationWillEnterForeground:), sel!(applicationWillEnterForeground:),
will_enter_foreground as extern fn(&Object, Sel, id), will_enter_foreground as extern "C" fn(&Object, Sel, id),
); );
decl.add_method( decl.add_method(
sel!(applicationDidEnterBackground:), sel!(applicationDidEnterBackground:),
did_enter_background as extern fn(&Object, Sel, id), did_enter_background as extern "C" fn(&Object, Sel, id),
); );
decl.add_method( decl.add_method(
sel!(applicationWillTerminate:), sel!(applicationWillTerminate:),
will_terminate as extern fn(&Object, Sel, id), will_terminate as extern "C" fn(&Object, Sel, id),
); );
AppDelegateClass(decl.register()) AppDelegateClass(decl.register())
}; };
} }
extern fn did_finish_launching(_: &Object, _: Sel, _: id) -> BOOL { extern "C" fn did_finish_launching(_: &Object, _: Sel, _: id) -> BOOL {
trace!("Triggered `didFinishLaunching`"); trace!("Triggered `didFinishLaunching`");
AppState::launched(); AppState::launched();
trace!("Completed `didFinishLaunching`"); trace!("Completed `didFinishLaunching`");
YES YES
} }
extern fn did_become_active(_: &Object, _: Sel, _: id) { extern "C" fn did_become_active(_: &Object, _: Sel, _: id) {
trace!("Triggered `didBecomeActive`"); trace!("Triggered `didBecomeActive`");
/*unsafe { /*unsafe {
HANDLER.lock().unwrap().handle_nonuser_event(Event::Suspended(false)) HANDLER.lock().unwrap().handle_nonuser_event(Event::Suspended(false))
@ -56,7 +59,7 @@ extern fn did_become_active(_: &Object, _: Sel, _: id) {
trace!("Completed `didBecomeActive`"); trace!("Completed `didBecomeActive`");
} }
extern fn will_resign_active(_: &Object, _: Sel, _: id) { extern "C" fn will_resign_active(_: &Object, _: Sel, _: id) {
trace!("Triggered `willResignActive`"); trace!("Triggered `willResignActive`");
/*unsafe { /*unsafe {
HANDLER.lock().unwrap().handle_nonuser_event(Event::Suspended(true)) HANDLER.lock().unwrap().handle_nonuser_event(Event::Suspended(true))
@ -64,17 +67,17 @@ extern fn will_resign_active(_: &Object, _: Sel, _: id) {
trace!("Completed `willResignActive`"); trace!("Completed `willResignActive`");
} }
extern fn will_enter_foreground(_: &Object, _: Sel, _: id) { extern "C" fn will_enter_foreground(_: &Object, _: Sel, _: id) {
trace!("Triggered `willEnterForeground`"); trace!("Triggered `willEnterForeground`");
trace!("Completed `willEnterForeground`"); trace!("Completed `willEnterForeground`");
} }
extern fn did_enter_background(_: &Object, _: Sel, _: id) { extern "C" fn did_enter_background(_: &Object, _: Sel, _: id) {
trace!("Triggered `didEnterBackground`"); trace!("Triggered `didEnterBackground`");
trace!("Completed `didEnterBackground`"); trace!("Completed `didEnterBackground`");
} }
extern fn will_terminate(_: &Object, _: Sel, _: id) { extern "C" fn will_terminate(_: &Object, _: Sel, _: id) {
trace!("Triggered `willTerminate`"); trace!("Triggered `willTerminate`");
/*unsafe { /*unsafe {
let app: id = msg_send![class!(UIApplication), sharedApplication]; let app: id = msg_send![class!(UIApplication), sharedApplication];

View file

@ -1,7 +1,13 @@
use std::{ use std::{
collections::VecDeque, fmt::{self, Debug}, collections::VecDeque,
hint::unreachable_unchecked, mem, fmt::{self, Debug},
sync::{atomic::{AtomicBool, Ordering}, Mutex, MutexGuard}, time::Instant, hint::unreachable_unchecked,
mem,
sync::{
atomic::{AtomicBool, Ordering},
Mutex, MutexGuard,
},
time::Instant,
}; };
use cocoa::{appkit::NSApp, base::nil}; use cocoa::{appkit::NSApp, base::nil};
@ -9,9 +15,9 @@ use cocoa::{appkit::NSApp, base::nil};
use crate::{ use crate::{
event::{Event, StartCause, WindowEvent}, event::{Event, StartCause, WindowEvent},
event_loop::{ControlFlow, EventLoopWindowTarget as RootWindowTarget}, event_loop::{ControlFlow, EventLoopWindowTarget as RootWindowTarget},
platform_impl::platform::{observer::EventLoopWaker, util::Never},
window::WindowId, window::WindowId,
}; };
use crate::platform_impl::platform::{observer::EventLoopWaker, util::Never};
lazy_static! { lazy_static! {
static ref HANDLER: Handler = Default::default(); static ref HANDLER: Handler = Default::default();
@ -39,7 +45,8 @@ struct EventLoopHandler<F, T: 'static> {
impl<F, T> Debug for EventLoopHandler<F, T> { impl<F, T> Debug for EventLoopHandler<F, T> {
fn fmt(&self, formatter: &mut fmt::Formatter<'_>) -> fmt::Result { fn fmt(&self, formatter: &mut fmt::Formatter<'_>) -> fmt::Result {
formatter.debug_struct("EventLoopHandler") formatter
.debug_struct("EventLoopHandler")
.field("window_target", &self.window_target) .field("window_target", &self.window_target)
.finish() .finish()
} }
@ -51,11 +58,7 @@ where
T: 'static, T: 'static,
{ {
fn handle_nonuser_event(&mut self, event: Event<Never>, control_flow: &mut ControlFlow) { fn handle_nonuser_event(&mut self, event: Event<Never>, control_flow: &mut ControlFlow) {
(self.callback)( (self.callback)(event.userify(), &self.window_target, control_flow);
event.userify(),
&self.window_target,
control_flow,
);
self.will_exit |= *control_flow == ControlFlow::Exit; self.will_exit |= *control_flow == ControlFlow::Exit;
if self.will_exit { if self.will_exit {
*control_flow = ControlFlow::Exit; *control_flow = ControlFlow::Exit;
@ -65,11 +68,7 @@ where
fn handle_user_events(&mut self, control_flow: &mut ControlFlow) { fn handle_user_events(&mut self, control_flow: &mut ControlFlow) {
let mut will_exit = self.will_exit; let mut will_exit = self.will_exit;
for event in self.window_target.p.receiver.try_iter() { for event in self.window_target.p.receiver.try_iter() {
(self.callback)( (self.callback)(Event::UserEvent(event), &self.window_target, control_flow);
Event::UserEvent(event),
&self.window_target,
control_flow,
);
will_exit |= *control_flow == ControlFlow::Exit; will_exit |= *control_flow == ControlFlow::Exit;
if will_exit { if will_exit {
*control_flow = ControlFlow::Exit; *control_flow = ControlFlow::Exit;
@ -167,18 +166,13 @@ impl Handler {
fn handle_nonuser_event(&self, event: Event<Never>) { fn handle_nonuser_event(&self, event: Event<Never>) {
if let Some(ref mut callback) = *self.callback.lock().unwrap() { if let Some(ref mut callback) = *self.callback.lock().unwrap() {
callback.handle_nonuser_event( callback.handle_nonuser_event(event, &mut *self.control_flow.lock().unwrap());
event,
&mut *self.control_flow.lock().unwrap(),
);
} }
} }
fn handle_user_events(&self) { fn handle_user_events(&self) {
if let Some(ref mut callback) = *self.callback.lock().unwrap() { if let Some(ref mut callback) = *self.callback.lock().unwrap() {
callback.handle_user_events( callback.handle_user_events(&mut *self.control_flow.lock().unwrap());
&mut *self.control_flow.lock().unwrap(),
);
} }
} }
} }
@ -213,13 +207,17 @@ impl AppState {
} }
pub fn wakeup() { pub fn wakeup() {
if !HANDLER.is_ready() { return } if !HANDLER.is_ready() {
return;
}
let start = HANDLER.get_start_time().unwrap(); let start = HANDLER.get_start_time().unwrap();
let cause = match HANDLER.get_control_flow_and_update_prev() { let cause = match HANDLER.get_control_flow_and_update_prev() {
ControlFlow::Poll => StartCause::Poll, ControlFlow::Poll => StartCause::Poll,
ControlFlow::Wait => StartCause::WaitCancelled { ControlFlow::Wait => {
start, StartCause::WaitCancelled {
requested_resume: None, start,
requested_resume: None,
}
}, },
ControlFlow::WaitUntil(requested_resume) => { ControlFlow::WaitUntil(requested_resume) => {
if Instant::now() >= requested_resume { if Instant::now() >= requested_resume {
@ -234,7 +232,7 @@ impl AppState {
} }
} }
}, },
ControlFlow::Exit => StartCause::Poll,//panic!("unexpected `ControlFlow::Exit`"), ControlFlow::Exit => StartCause::Poll, //panic!("unexpected `ControlFlow::Exit`"),
}; };
HANDLER.set_in_callback(true); HANDLER.set_in_callback(true);
HANDLER.handle_nonuser_event(Event::NewEvents(cause)); HANDLER.handle_nonuser_event(Event::NewEvents(cause));
@ -278,7 +276,9 @@ impl AppState {
} }
pub fn cleared() { pub fn cleared() {
if !HANDLER.is_ready() { return } if !HANDLER.is_ready() {
return;
}
if !HANDLER.get_in_callback() { if !HANDLER.get_in_callback() {
HANDLER.set_in_callback(true); HANDLER.set_in_callback(true);
HANDLER.handle_user_events(); HANDLER.handle_user_events();
@ -295,8 +295,8 @@ impl AppState {
HANDLER.set_in_callback(false); HANDLER.set_in_callback(false);
} }
if HANDLER.should_exit() { if HANDLER.should_exit() {
let _: () = unsafe { msg_send![NSApp(), stop:nil] }; let _: () = unsafe { msg_send![NSApp(), stop: nil] };
return return;
} }
HANDLER.update_start_time(); HANDLER.update_start_time();
match HANDLER.get_old_and_new_control_flow() { match HANDLER.get_old_and_new_control_flow() {

View file

@ -1,12 +1,14 @@
use std::os::raw::c_ushort; use std::os::raw::c_ushort;
use cocoa::{appkit::{NSEvent, NSEventModifierFlags}, base::id}; use cocoa::{
appkit::{NSEvent, NSEventModifierFlags},
use crate::event::{ base::id,
ElementState, KeyboardInput, };
ModifiersState, VirtualKeyCode, WindowEvent,
use crate::{
event::{ElementState, KeyboardInput, ModifiersState, VirtualKeyCode, WindowEvent},
platform_impl::platform::DEVICE_ID,
}; };
use crate::platform_impl::platform::DEVICE_ID;
pub fn char_to_keycode(c: char) -> Option<VirtualKeyCode> { pub fn char_to_keycode(c: char) -> Option<VirtualKeyCode> {
// We only translate keys that are affected by keyboard layout. // We only translate keys that are affected by keyboard layout.
@ -57,9 +59,9 @@ pub fn char_to_keycode(c: char) -> Option<VirtualKeyCode> {
'-' | '_' => VirtualKeyCode::Minus, '-' | '_' => VirtualKeyCode::Minus,
']' | '}' => VirtualKeyCode::RBracket, ']' | '}' => VirtualKeyCode::RBracket,
'[' | '{' => VirtualKeyCode::LBracket, '[' | '{' => VirtualKeyCode::LBracket,
'\''| '"' => VirtualKeyCode::Apostrophe, '\'' | '"' => VirtualKeyCode::Apostrophe,
';' | ':' => VirtualKeyCode::Semicolon, ';' | ':' => VirtualKeyCode::Semicolon,
'\\'| '|' => VirtualKeyCode::Backslash, '\\' | '|' => VirtualKeyCode::Backslash,
',' | '<' => VirtualKeyCode::Comma, ',' | '<' => VirtualKeyCode::Comma,
'/' | '?' => VirtualKeyCode::Slash, '/' | '?' => VirtualKeyCode::Slash,
'.' | '>' => VirtualKeyCode::Period, '.' | '>' => VirtualKeyCode::Period,
@ -198,7 +200,6 @@ pub fn scancode_to_keycode(scancode: c_ushort) -> Option<VirtualKeyCode> {
0x7d => VirtualKeyCode::Down, 0x7d => VirtualKeyCode::Down,
0x7e => VirtualKeyCode::Up, 0x7e => VirtualKeyCode::Up,
//0x7f => unkown, //0x7f => unkown,
0xa => VirtualKeyCode::Caret, 0xa => VirtualKeyCode::Caret,
_ => return None, _ => return None,
}) })
@ -215,16 +216,14 @@ pub fn check_function_keys(string: &String) -> Option<VirtualKeyCode> {
0xf71a => VirtualKeyCode::F23, 0xf71a => VirtualKeyCode::F23,
0xf71b => VirtualKeyCode::F24, 0xf71b => VirtualKeyCode::F24,
_ => return None, _ => return None,
}) });
} }
None None
} }
pub fn event_mods(event: id) -> ModifiersState { pub fn event_mods(event: id) -> ModifiersState {
let flags = unsafe { let flags = unsafe { NSEvent::modifierFlags(event) };
NSEvent::modifierFlags(event)
};
ModifiersState { ModifiersState {
shift: flags.contains(NSEventModifierFlags::NSShiftKeyMask), shift: flags.contains(NSEventModifierFlags::NSShiftKeyMask),
ctrl: flags.contains(NSEventModifierFlags::NSControlKeyMask), ctrl: flags.contains(NSEventModifierFlags::NSControlKeyMask),
@ -238,9 +237,7 @@ pub fn get_scancode(event: cocoa::base::id) -> c_ushort {
// and there is no easy way to navtively retrieve the layout-dependent character. // and there is no easy way to navtively retrieve the layout-dependent character.
// In winit, we use keycode to refer to the key's character, and so this function aligns // In winit, we use keycode to refer to the key's character, and so this function aligns
// AppKit's terminology with ours. // AppKit's terminology with ours.
unsafe { unsafe { msg_send![event, keyCode] }
msg_send![event, keyCode]
}
} }
pub unsafe fn modifier_event( pub unsafe fn modifier_event(
@ -249,7 +246,8 @@ pub unsafe fn modifier_event(
was_key_pressed: bool, was_key_pressed: bool,
) -> Option<WindowEvent> { ) -> Option<WindowEvent> {
if !was_key_pressed && NSEvent::modifierFlags(ns_event).contains(keymask) if !was_key_pressed && NSEvent::modifierFlags(ns_event).contains(keymask)
|| was_key_pressed && !NSEvent::modifierFlags(ns_event).contains(keymask) { || was_key_pressed && !NSEvent::modifierFlags(ns_event).contains(keymask)
{
let state = if was_key_pressed { let state = if was_key_pressed {
ElementState::Released ElementState::Released
} else { } else {

View file

@ -1,17 +1,24 @@
use std::{ use std::{
collections::VecDeque, mem, os::raw::c_void, process, ptr, sync::mpsc, marker::PhantomData collections::VecDeque, marker::PhantomData, mem, os::raw::c_void, process, ptr, sync::mpsc,
}; };
use cocoa::{appkit::NSApp, base::{id, nil}, foundation::NSAutoreleasePool}; use cocoa::{
appkit::NSApp,
base::{id, nil},
foundation::NSAutoreleasePool,
};
use crate::{ use crate::{
event::Event, event::Event,
event_loop::{ControlFlow, EventLoopClosed, EventLoopWindowTarget as RootWindowTarget}, event_loop::{ControlFlow, EventLoopClosed, EventLoopWindowTarget as RootWindowTarget},
}; platform_impl::platform::{
use crate::platform_impl::platform::{ app::APP_CLASS,
app::APP_CLASS, app_delegate::APP_DELEGATE_CLASS, app_delegate::APP_DELEGATE_CLASS,
app_state::AppState, monitor::{self, MonitorHandle}, app_state::AppState,
observer::*, util::IdRef, monitor::{self, MonitorHandle},
observer::*,
util::IdRef,
},
}; };
pub struct EventLoopWindowTarget<T: 'static> { pub struct EventLoopWindowTarget<T: 'static> {
@ -75,7 +82,8 @@ impl<T> EventLoop<T> {
} }
pub fn run<F>(self, callback: F) -> ! pub fn run<F>(self, callback: F) -> !
where F: 'static + FnMut(Event<T>, &RootWindowTarget<T>, &mut ControlFlow), where
F: 'static + FnMut(Event<T>, &RootWindowTarget<T>, &mut ControlFlow),
{ {
unsafe { unsafe {
let _pool = NSAutoreleasePool::new(nil); let _pool = NSAutoreleasePool::new(nil);
@ -89,7 +97,8 @@ impl<T> EventLoop<T> {
} }
pub fn run_return<F>(&mut self, _callback: F) pub fn run_return<F>(&mut self, _callback: F)
where F: FnMut(Event<T>, &RootWindowTarget<T>, &mut ControlFlow), where
F: FnMut(Event<T>, &RootWindowTarget<T>, &mut ControlFlow),
{ {
unimplemented!(); unimplemented!();
} }
@ -119,11 +128,8 @@ impl<T> Proxy<T> {
let rl = CFRunLoopGetMain(); let rl = CFRunLoopGetMain();
let mut context: CFRunLoopSourceContext = mem::zeroed(); let mut context: CFRunLoopSourceContext = mem::zeroed();
context.perform = event_loop_proxy_handler; context.perform = event_loop_proxy_handler;
let source = CFRunLoopSourceCreate( let source =
ptr::null_mut(), CFRunLoopSourceCreate(ptr::null_mut(), CFIndex::max_value() - 1, &mut context);
CFIndex::max_value() - 1,
&mut context,
);
CFRunLoopAddSource(rl, source, kCFRunLoopCommonModes); CFRunLoopAddSource(rl, source, kCFRunLoopCommonModes);
CFRunLoopWakeUp(rl); CFRunLoopWakeUp(rl);

View file

@ -2,8 +2,10 @@
#![allow(dead_code, non_snake_case, non_upper_case_globals)] #![allow(dead_code, non_snake_case, non_upper_case_globals)]
use cocoa::base::id; use cocoa::{
use cocoa::foundation::{NSInteger, NSUInteger}; base::id,
foundation::{NSInteger, NSUInteger},
};
use objc; use objc;
pub const NSNotFound: NSInteger = NSInteger::max_value(); pub const NSNotFound: NSInteger = NSInteger::max_value();
@ -53,11 +55,11 @@ impl NSMutableAttributedString for id {
} }
unsafe fn initWithString(self, string: id) -> id { unsafe fn initWithString(self, string: id) -> id {
msg_send![self, initWithString:string] msg_send![self, initWithString: string]
} }
unsafe fn initWithAttributedString(self, string: id) -> id { unsafe fn initWithAttributedString(self, string: id) -> id {
msg_send![self, initWithAttributedString:string] msg_send![self, initWithAttributedString: string]
} }
unsafe fn string(self) -> id { unsafe fn string(self) -> id {

View file

@ -15,16 +15,13 @@ mod window_delegate;
use std::{fmt, ops::Deref, sync::Arc}; use std::{fmt, ops::Deref, sync::Arc};
use crate::{
event::DeviceId as RootDeviceId, window::WindowAttributes,
error::OsError as RootOsError,
};
pub use self::{ pub use self::{
event_loop::{EventLoop, EventLoopWindowTarget, Proxy as EventLoopProxy}, event_loop::{EventLoop, EventLoopWindowTarget, Proxy as EventLoopProxy},
monitor::MonitorHandle, monitor::MonitorHandle,
window::{ window::{Id as WindowId, PlatformSpecificWindowBuilderAttributes, UnownedWindow},
Id as WindowId, PlatformSpecificWindowBuilderAttributes, UnownedWindow, };
}, use crate::{
error::OsError as RootOsError, event::DeviceId as RootDeviceId, window::WindowAttributes,
}; };
#[derive(Debug, Copy, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)] #[derive(Debug, Copy, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
@ -48,7 +45,7 @@ pub struct Window {
#[derive(Debug)] #[derive(Debug)]
pub enum OsError { pub enum OsError {
CGError(core_graphics::base::CGError), CGError(core_graphics::base::CGError),
CreationError(&'static str) CreationError(&'static str),
} }
unsafe impl Send for Window {} unsafe impl Send for Window {}

View file

@ -11,9 +11,11 @@ use core_video_sys::{
CVDisplayLinkGetNominalOutputVideoRefreshPeriod, CVDisplayLinkRelease, CVDisplayLinkGetNominalOutputVideoRefreshPeriod, CVDisplayLinkRelease,
}; };
use crate::dpi::{PhysicalPosition, PhysicalSize}; use crate::{
use crate::monitor::VideoMode; dpi::{PhysicalPosition, PhysicalSize},
use crate::platform_impl::platform::util::IdRef; monitor::VideoMode,
platform_impl::platform::util::IdRef,
};
#[derive(Clone, PartialEq)] #[derive(Clone, PartialEq)]
pub struct MonitorHandle(CGDirectDisplayID); pub struct MonitorHandle(CGDirectDisplayID);
@ -79,10 +81,7 @@ impl MonitorHandle {
let display = CGDisplay::new(display_id); let display = CGDisplay::new(display_id);
let height = display.pixels_high(); let height = display.pixels_high();
let width = display.pixels_wide(); let width = display.pixels_wide();
PhysicalSize::from_logical( PhysicalSize::from_logical((width as f64, height as f64), self.hidpi_factor())
(width as f64, height as f64),
self.hidpi_factor(),
)
} }
#[inline] #[inline]

View file

@ -1,9 +1,9 @@
use std::{self, ptr, os::raw::*, time::Instant}; use std::{self, os::raw::*, ptr, time::Instant};
use crate::platform_impl::platform::app_state::AppState; use crate::platform_impl::platform::app_state::AppState;
#[link(name = "CoreFoundation", kind = "framework")] #[link(name = "CoreFoundation", kind = "framework")]
extern { extern "C" {
pub static kCFRunLoopDefaultMode: CFRunLoopMode; pub static kCFRunLoopDefaultMode: CFRunLoopMode;
pub static kCFRunLoopCommonModes: CFRunLoopMode; pub static kCFRunLoopCommonModes: CFRunLoopMode;
@ -33,15 +33,8 @@ extern {
callout: CFRunLoopTimerCallBack, callout: CFRunLoopTimerCallBack,
context: *mut CFRunLoopTimerContext, context: *mut CFRunLoopTimerContext,
) -> CFRunLoopTimerRef; ) -> CFRunLoopTimerRef;
pub fn CFRunLoopAddTimer( pub fn CFRunLoopAddTimer(rl: CFRunLoopRef, timer: CFRunLoopTimerRef, mode: CFRunLoopMode);
rl: CFRunLoopRef, pub fn CFRunLoopTimerSetNextFireDate(timer: CFRunLoopTimerRef, fireDate: CFAbsoluteTime);
timer: CFRunLoopTimerRef,
mode: CFRunLoopMode,
);
pub fn CFRunLoopTimerSetNextFireDate(
timer: CFRunLoopTimerRef,
fireDate: CFAbsoluteTime,
);
pub fn CFRunLoopTimerInvalidate(time: CFRunLoopTimerRef); pub fn CFRunLoopTimerInvalidate(time: CFRunLoopTimerRef);
pub fn CFRunLoopSourceCreate( pub fn CFRunLoopSourceCreate(
@ -49,11 +42,7 @@ extern {
order: CFIndex, order: CFIndex,
context: *mut CFRunLoopSourceContext, context: *mut CFRunLoopSourceContext,
) -> CFRunLoopSourceRef; ) -> CFRunLoopSourceRef;
pub fn CFRunLoopAddSource( pub fn CFRunLoopAddSource(rl: CFRunLoopRef, source: CFRunLoopSourceRef, mode: CFRunLoopMode);
rl: CFRunLoopRef,
source: CFRunLoopSourceRef,
mode: CFRunLoopMode,
);
#[allow(dead_code)] #[allow(dead_code)]
pub fn CFRunLoopSourceInvalidate(source: CFRunLoopSourceRef); pub fn CFRunLoopSourceInvalidate(source: CFRunLoopSourceRef);
pub fn CFRunLoopSourceSignal(source: CFRunLoopSourceRef); pub fn CFRunLoopSourceSignal(source: CFRunLoopSourceRef);
@ -98,15 +87,9 @@ pub const kCFRunLoopAfterWaiting: CFRunLoopActivity = 1 << 6;
#[allow(non_upper_case_globals)] #[allow(non_upper_case_globals)]
pub const kCFRunLoopExit: CFRunLoopActivity = 1 << 7; pub const kCFRunLoopExit: CFRunLoopActivity = 1 << 7;
pub type CFRunLoopObserverCallBack = extern "C" fn( pub type CFRunLoopObserverCallBack =
observer: CFRunLoopObserverRef, extern "C" fn(observer: CFRunLoopObserverRef, activity: CFRunLoopActivity, info: *mut c_void);
activity: CFRunLoopActivity, pub type CFRunLoopTimerCallBack = extern "C" fn(timer: CFRunLoopTimerRef, info: *mut c_void);
info: *mut c_void,
);
pub type CFRunLoopTimerCallBack = extern "C" fn(
timer: CFRunLoopTimerRef,
info: *mut c_void
);
pub enum CFRunLoopObserverContext {} pub enum CFRunLoopObserverContext {}
pub enum CFRunLoopTimerContext {} pub enum CFRunLoopTimerContext {}
@ -127,7 +110,7 @@ pub struct CFRunLoopSourceContext {
} }
// begin is queued with the highest priority to ensure it is processed before other observers // begin is queued with the highest priority to ensure it is processed before other observers
extern fn control_flow_begin_handler( extern "C" fn control_flow_begin_handler(
_: CFRunLoopObserverRef, _: CFRunLoopObserverRef,
activity: CFRunLoopActivity, activity: CFRunLoopActivity,
_: *mut c_void, _: *mut c_void,
@ -146,7 +129,7 @@ extern fn control_flow_begin_handler(
// end is queued with the lowest priority to ensure it is processed after other observers // end is queued with the lowest priority to ensure it is processed after other observers
// without that, LoopDestroyed would get sent after EventsCleared // without that, LoopDestroyed would get sent after EventsCleared
extern fn control_flow_end_handler( extern "C" fn control_flow_end_handler(
_: CFRunLoopObserverRef, _: CFRunLoopObserverRef,
activity: CFRunLoopActivity, activity: CFRunLoopActivity,
_: *mut c_void, _: *mut c_void,
@ -158,7 +141,7 @@ extern fn control_flow_end_handler(
AppState::cleared(); AppState::cleared();
//trace!("Completed `CFRunLoopBeforeWaiting`"); //trace!("Completed `CFRunLoopBeforeWaiting`");
}, },
kCFRunLoopExit => (),//unimplemented!(), // not expected to ever happen kCFRunLoopExit => (), //unimplemented!(), // not expected to ever happen
_ => unreachable!(), _ => unreachable!(),
} }
} }
@ -179,7 +162,7 @@ impl RunLoop {
let observer = CFRunLoopObserverCreate( let observer = CFRunLoopObserverCreate(
ptr::null_mut(), ptr::null_mut(),
flags, flags,
TRUE, // Indicates we want this to run repeatedly TRUE, // Indicates we want this to run repeatedly
priority, // The lower the value, the sooner this will run priority, // The lower the value, the sooner this will run
handler, handler,
ptr::null_mut(), ptr::null_mut(),
@ -204,7 +187,6 @@ pub fn setup_control_flow_observers() {
} }
} }
pub struct EventLoopWaker { pub struct EventLoopWaker {
timer: CFRunLoopTimerRef, timer: CFRunLoopTimerRef,
} }
@ -220,7 +202,7 @@ impl Drop for EventLoopWaker {
impl Default for EventLoopWaker { impl Default for EventLoopWaker {
fn default() -> EventLoopWaker { fn default() -> EventLoopWaker {
extern fn wakeup_main_loop(_timer: CFRunLoopTimerRef, _info: *mut c_void) {} extern "C" fn wakeup_main_loop(_timer: CFRunLoopTimerRef, _info: *mut c_void) {}
unsafe { unsafe {
// create a timer with a 1µs interval (1ns does not work) to mimic polling. // create a timer with a 1µs interval (1ns does not work) to mimic polling.
// it is initially setup with a first fire time really far into the // it is initially setup with a first fire time really far into the
@ -257,8 +239,8 @@ impl EventLoopWaker {
unsafe { unsafe {
let current = CFAbsoluteTimeGetCurrent(); let current = CFAbsoluteTimeGetCurrent();
let duration = instant - now; let duration = instant - now;
let fsecs = duration.subsec_nanos() as f64 / 1_000_000_000.0 let fsecs =
+ duration.as_secs() as f64; duration.subsec_nanos() as f64 / 1_000_000_000.0 + duration.as_secs() as f64;
CFRunLoopTimerSetNextFireDate(self.timer, current + fsecs) CFRunLoopTimerSetNextFireDate(self.timer, current + fsecs)
} }
} }

View file

@ -10,8 +10,10 @@ use cocoa::{
}; };
use dispatch::ffi::{dispatch_async_f, dispatch_get_main_queue, dispatch_sync_f}; use dispatch::ffi::{dispatch_async_f, dispatch_get_main_queue, dispatch_sync_f};
use crate::dpi::LogicalSize; use crate::{
use crate::platform_impl::platform::{ffi, util::IdRef, window::SharedState}; dpi::LogicalSize,
platform_impl::platform::{ffi, util::IdRef, window::SharedState},
};
unsafe fn set_style_mask(ns_window: id, ns_view: id, mask: NSWindowStyleMask) { unsafe fn set_style_mask(ns_window: id, ns_view: id, mask: NSWindowStyleMask) {
ns_window.setStyleMask_(mask); ns_window.setStyleMask_(mask);
@ -26,15 +28,15 @@ struct SetStyleMaskData {
mask: NSWindowStyleMask, mask: NSWindowStyleMask,
} }
impl SetStyleMaskData { impl SetStyleMaskData {
fn new_ptr( fn new_ptr(ns_window: id, ns_view: id, mask: NSWindowStyleMask) -> *mut Self {
ns_window: id, Box::into_raw(Box::new(SetStyleMaskData {
ns_view: id, ns_window,
mask: NSWindowStyleMask, ns_view,
) -> *mut Self { mask,
Box::into_raw(Box::new(SetStyleMaskData { ns_window, ns_view, mask })) }))
} }
} }
extern fn set_style_mask_callback(context: *mut c_void) { extern "C" fn set_style_mask_callback(context: *mut c_void) {
unsafe { unsafe {
let context_ptr = context as *mut SetStyleMaskData; let context_ptr = context as *mut SetStyleMaskData;
{ {
@ -70,14 +72,11 @@ struct SetContentSizeData {
size: LogicalSize, size: LogicalSize,
} }
impl SetContentSizeData { impl SetContentSizeData {
fn new_ptr( fn new_ptr(ns_window: id, size: LogicalSize) -> *mut Self {
ns_window: id,
size: LogicalSize,
) -> *mut Self {
Box::into_raw(Box::new(SetContentSizeData { ns_window, size })) Box::into_raw(Box::new(SetContentSizeData { ns_window, size }))
} }
} }
extern fn set_content_size_callback(context: *mut c_void) { extern "C" fn set_content_size_callback(context: *mut c_void) {
unsafe { unsafe {
let context_ptr = context as *mut SetContentSizeData; let context_ptr = context as *mut SetContentSizeData;
{ {
@ -109,14 +108,11 @@ struct SetFrameTopLeftPointData {
point: NSPoint, point: NSPoint,
} }
impl SetFrameTopLeftPointData { impl SetFrameTopLeftPointData {
fn new_ptr( fn new_ptr(ns_window: id, point: NSPoint) -> *mut Self {
ns_window: id,
point: NSPoint,
) -> *mut Self {
Box::into_raw(Box::new(SetFrameTopLeftPointData { ns_window, point })) Box::into_raw(Box::new(SetFrameTopLeftPointData { ns_window, point }))
} }
} }
extern fn set_frame_top_left_point_callback(context: *mut c_void) { extern "C" fn set_frame_top_left_point_callback(context: *mut c_void) {
unsafe { unsafe {
let context_ptr = context as *mut SetFrameTopLeftPointData; let context_ptr = context as *mut SetFrameTopLeftPointData;
{ {
@ -142,14 +138,11 @@ struct SetLevelData {
level: ffi::NSWindowLevel, level: ffi::NSWindowLevel,
} }
impl SetLevelData { impl SetLevelData {
fn new_ptr( fn new_ptr(ns_window: id, level: ffi::NSWindowLevel) -> *mut Self {
ns_window: id,
level: ffi::NSWindowLevel,
) -> *mut Self {
Box::into_raw(Box::new(SetLevelData { ns_window, level })) Box::into_raw(Box::new(SetLevelData { ns_window, level }))
} }
} }
extern fn set_level_callback(context: *mut c_void) { extern "C" fn set_level_callback(context: *mut c_void) {
unsafe { unsafe {
let context_ptr = context as *mut SetLevelData; let context_ptr = context as *mut SetLevelData;
{ {
@ -190,7 +183,7 @@ impl ToggleFullScreenData {
})) }))
} }
} }
extern fn toggle_full_screen_callback(context: *mut c_void) { extern "C" fn toggle_full_screen_callback(context: *mut c_void) {
unsafe { unsafe {
let context_ptr = context as *mut ToggleFullScreenData; let context_ptr = context as *mut ToggleFullScreenData;
{ {
@ -227,12 +220,7 @@ pub unsafe fn toggle_full_screen_async(
not_fullscreen: bool, not_fullscreen: bool,
shared_state: Weak<Mutex<SharedState>>, shared_state: Weak<Mutex<SharedState>>,
) { ) {
let context = ToggleFullScreenData::new_ptr( let context = ToggleFullScreenData::new_ptr(ns_window, ns_view, not_fullscreen, shared_state);
ns_window,
ns_view,
not_fullscreen,
shared_state,
);
dispatch_async_f( dispatch_async_f(
dispatch_get_main_queue(), dispatch_get_main_queue(),
context as *mut _, context as *mut _,
@ -325,7 +313,7 @@ impl OrderOutData {
Box::into_raw(Box::new(OrderOutData { ns_window })) Box::into_raw(Box::new(OrderOutData { ns_window }))
} }
} }
extern fn order_out_callback(context: *mut c_void) { extern "C" fn order_out_callback(context: *mut c_void) {
unsafe { unsafe {
let context_ptr = context as *mut OrderOutData; let context_ptr = context as *mut OrderOutData;
{ {
@ -354,7 +342,7 @@ impl MakeKeyAndOrderFrontData {
Box::into_raw(Box::new(MakeKeyAndOrderFrontData { ns_window })) Box::into_raw(Box::new(MakeKeyAndOrderFrontData { ns_window }))
} }
} }
extern fn make_key_and_order_front_callback(context: *mut c_void) { extern "C" fn make_key_and_order_front_callback(context: *mut c_void) {
unsafe { unsafe {
let context_ptr = context as *mut MakeKeyAndOrderFrontData; let context_ptr = context as *mut MakeKeyAndOrderFrontData;
{ {
@ -415,7 +403,7 @@ impl CloseData {
Box::into_raw(Box::new(CloseData { ns_window })) Box::into_raw(Box::new(CloseData { ns_window }))
} }
} }
extern fn close_callback(context: *mut c_void) { extern "C" fn close_callback(context: *mut c_void) {
unsafe { unsafe {
let context_ptr = context as *mut CloseData; let context_ptr = context as *mut CloseData;
{ {
@ -431,9 +419,5 @@ extern fn close_callback(context: *mut c_void) {
// thread. Though, it's a good idea to look into that more... // thread. Though, it's a good idea to look into that more...
pub unsafe fn close_async(ns_window: id) { pub unsafe fn close_async(ns_window: id) {
let context = CloseData::new_ptr(ns_window); let context = CloseData::new_ptr(ns_window);
dispatch_async_f( dispatch_async_f(dispatch_get_main_queue(), context as *mut _, close_callback);
dispatch_get_main_queue(),
context as *mut _,
close_callback,
);
} }

View file

@ -1,5 +1,6 @@
use cocoa::{ use cocoa::{
appkit::NSImage, base::{id, nil}, appkit::NSImage,
base::{id, nil},
foundation::{NSDictionary, NSPoint, NSString}, foundation::{NSDictionary, NSPoint, NSString},
}; };
use objc::runtime::Sel; use objc::runtime::Sel;
@ -22,7 +23,9 @@ impl From<CursorIcon> for Cursor {
CursorIcon::VerticalText => Cursor::Native("IBeamCursorForVerticalLayout"), CursorIcon::VerticalText => Cursor::Native("IBeamCursorForVerticalLayout"),
CursorIcon::Copy => Cursor::Native("dragCopyCursor"), CursorIcon::Copy => Cursor::Native("dragCopyCursor"),
CursorIcon::Alias => Cursor::Native("dragLinkCursor"), CursorIcon::Alias => Cursor::Native("dragLinkCursor"),
CursorIcon::NotAllowed | CursorIcon::NoDrop => Cursor::Native("operationNotAllowedCursor"), CursorIcon::NotAllowed | CursorIcon::NoDrop => {
Cursor::Native("operationNotAllowedCursor")
},
CursorIcon::ContextMenu => Cursor::Native("contextualMenuCursor"), CursorIcon::ContextMenu => Cursor::Native("contextualMenuCursor"),
CursorIcon::Crosshair => Cursor::Native("crosshairCursor"), CursorIcon::Crosshair => Cursor::Native("crosshairCursor"),
CursorIcon::EResize => Cursor::Native("resizeRightCursor"), CursorIcon::EResize => Cursor::Native("resizeRightCursor"),
@ -52,7 +55,9 @@ impl From<CursorIcon> for Cursor {
// https://bugs.eclipse.org/bugs/show_bug.cgi?id=522349 // https://bugs.eclipse.org/bugs/show_bug.cgi?id=522349
// This is the wrong semantics for `Wait`, but it's the same as // This is the wrong semantics for `Wait`, but it's the same as
// what's used in Safari and Chrome. // what's used in Safari and Chrome.
CursorIcon::Wait | CursorIcon::Progress => Cursor::Undocumented("busyButClickableCursor"), CursorIcon::Wait | CursorIcon::Progress => {
Cursor::Undocumented("busyButClickableCursor")
},
// For the rest, we can just snatch the cursors from WebKit... // For the rest, we can just snatch the cursors from WebKit...
// They fit the style of the native cursors, and will seem // They fit the style of the native cursors, and will seem
@ -75,18 +80,18 @@ impl Cursor {
match self { match self {
Cursor::Native(cursor_name) => { Cursor::Native(cursor_name) => {
let sel = Sel::register(cursor_name); let sel = Sel::register(cursor_name);
msg_send![class!(NSCursor), performSelector:sel] msg_send![class!(NSCursor), performSelector: sel]
}, },
Cursor::Undocumented(cursor_name) => { Cursor::Undocumented(cursor_name) => {
let class = class!(NSCursor); let class = class!(NSCursor);
let sel = Sel::register(cursor_name); let sel = Sel::register(cursor_name);
let sel = if msg_send![class, respondsToSelector:sel] { let sel = if msg_send![class, respondsToSelector: sel] {
sel sel
} else { } else {
warn!("Cursor `{}` appears to be invalid", cursor_name); warn!("Cursor `{}` appears to be invalid", cursor_name);
sel!(arrowCursor) sel!(arrowCursor)
}; };
msg_send![class, performSelector:sel] msg_send![class, performSelector: sel]
}, },
Cursor::WebKit(cursor_name) => load_webkit_cursor(cursor_name), Cursor::WebKit(cursor_name) => load_webkit_cursor(cursor_name),
} }
@ -104,27 +109,15 @@ pub unsafe fn load_webkit_cursor(cursor_name: &str) -> id {
let key_x = NSString::alloc(nil).init_str("hotx"); let key_x = NSString::alloc(nil).init_str("hotx");
let key_y = NSString::alloc(nil).init_str("hoty"); let key_y = NSString::alloc(nil).init_str("hoty");
let cursor_path: id = msg_send![cursor_root, let cursor_path: id = msg_send![cursor_root, stringByAppendingPathComponent: cursor_name];
stringByAppendingPathComponent:cursor_name let pdf_path: id = msg_send![cursor_path, stringByAppendingPathComponent: cursor_pdf];
]; let info_path: id = msg_send![cursor_path, stringByAppendingPathComponent: cursor_plist];
let pdf_path: id = msg_send![cursor_path,
stringByAppendingPathComponent:cursor_pdf
];
let info_path: id = msg_send![cursor_path,
stringByAppendingPathComponent:cursor_plist
];
let image = NSImage::alloc(nil).initByReferencingFile_(pdf_path); let image = NSImage::alloc(nil).initByReferencingFile_(pdf_path);
let info = NSDictionary::dictionaryWithContentsOfFile_( let info = NSDictionary::dictionaryWithContentsOfFile_(nil, info_path);
nil,
info_path,
);
let x = info.valueForKey_(key_x); let x = info.valueForKey_(key_x);
let y = info.valueForKey_(key_y); let y = info.valueForKey_(key_y);
let point = NSPoint::new( let point = NSPoint::new(msg_send![x, doubleValue], msg_send![y, doubleValue]);
msg_send![x, doubleValue],
msg_send![y, doubleValue],
);
let cursor: id = msg_send![class!(NSCursor), alloc]; let cursor: id = msg_send![class!(NSCursor), alloc];
msg_send![cursor, msg_send![cursor,
initWithImage:image initWithImage:image

View file

@ -1,10 +1,9 @@
mod r#async; mod r#async;
mod cursor; mod cursor;
pub use self::{r#async::*, cursor::*}; pub use self::{cursor::*, r#async::*};
use std::ops::Deref; use std::ops::{BitAnd, Deref};
use std::ops::BitAnd;
use cocoa::{ use cocoa::{
appkit::{NSApp, NSWindowStyleMask}, appkit::{NSApp, NSWindowStyleMask},
@ -12,7 +11,7 @@ use cocoa::{
foundation::{NSAutoreleasePool, NSRect, NSUInteger}, foundation::{NSAutoreleasePool, NSRect, NSUInteger},
}; };
use core_graphics::display::CGDisplay; use core_graphics::display::CGDisplay;
use objc::runtime::{BOOL, Class, Object, Sel, YES}; use objc::runtime::{Class, Object, Sel, BOOL, YES};
use crate::platform_impl::platform::ffi; use crate::platform_impl::platform::ffi;
@ -21,8 +20,8 @@ use crate::platform_impl::platform::ffi;
pub enum Never {} pub enum Never {}
pub fn has_flag<T>(bitset: T, flag: T) -> bool pub fn has_flag<T>(bitset: T, flag: T) -> bool
where T: where
Copy + PartialEq + BitAnd<T, Output = T> T: Copy + PartialEq + BitAnd<T, Output = T>,
{ {
bitset & flag == flag bitset & flag == flag
} }
@ -48,7 +47,11 @@ impl IdRef {
} }
pub fn non_nil(self) -> Option<IdRef> { pub fn non_nil(self) -> Option<IdRef> {
if self.0 == nil { None } else { Some(self) } if self.0 == nil {
None
} else {
Some(self)
}
} }
} }
@ -94,16 +97,16 @@ pub unsafe fn superclass<'a>(this: &'a Object) -> &'a Class {
pub unsafe fn create_input_context(view: id) -> IdRef { pub unsafe fn create_input_context(view: id) -> IdRef {
let input_context: id = msg_send![class!(NSTextInputContext), alloc]; let input_context: id = msg_send![class!(NSTextInputContext), alloc];
let input_context: id = msg_send![input_context, initWithClient:view]; let input_context: id = msg_send![input_context, initWithClient: view];
IdRef::new(input_context) IdRef::new(input_context)
} }
#[allow(dead_code)] #[allow(dead_code)]
pub unsafe fn open_emoji_picker() { pub unsafe fn open_emoji_picker() {
let () = msg_send![NSApp(), orderFrontCharacterPalette:nil]; let () = msg_send![NSApp(), orderFrontCharacterPalette: nil];
} }
pub extern fn yes(_: &Object, _: Sel) -> BOOL { pub extern "C" fn yes(_: &Object, _: Sel) -> BOOL {
YES YES
} }
@ -120,4 +123,3 @@ pub unsafe fn toggle_style_mask(window: id, view: id, mask: NSWindowStyleMask, o
// If we don't do this, key handling will break. Therefore, never call `setStyleMask` directly! // If we don't do this, key handling will break. Therefore, never call `setStyleMask` directly!
window.makeFirstResponder_(view); window.makeFirstResponder_(view);
} }

View file

@ -1,26 +1,39 @@
use std::{ use std::{
boxed::Box, collections::VecDeque, os::raw::*, slice, str, boxed::Box,
collections::VecDeque,
os::raw::*,
slice, str,
sync::{Arc, Mutex, Weak}, sync::{Arc, Mutex, Weak},
}; };
use cocoa::{ use cocoa::{
appkit::{NSApp, NSEvent, NSEventModifierFlags, NSEventPhase, NSView, NSWindow}, appkit::{NSApp, NSEvent, NSEventModifierFlags, NSEventPhase, NSView, NSWindow},
base::{id, nil}, foundation::{NSPoint, NSRect, NSSize, NSString, NSUInteger}, base::{id, nil},
foundation::{NSPoint, NSRect, NSSize, NSString, NSUInteger},
};
use objc::{
declare::ClassDecl,
runtime::{Class, Object, Protocol, Sel, BOOL, NO, YES},
}; };
use objc::{declare::ClassDecl, runtime::{BOOL, Class, NO, Object, Protocol, Sel, YES}};
use crate::{ use crate::{
event::{ event::{
DeviceEvent, ElementState, Event, KeyboardInput, MouseButton, DeviceEvent, ElementState, Event, KeyboardInput, MouseButton, MouseScrollDelta, TouchPhase,
MouseScrollDelta, TouchPhase, VirtualKeyCode, WindowEvent, VirtualKeyCode, WindowEvent,
},
platform_impl::platform::{
app_state::AppState,
event::{
char_to_keycode, check_function_keys, event_mods, get_scancode, modifier_event,
scancode_to_keycode,
},
ffi::*,
util::{self, IdRef},
window::get_window_id,
DEVICE_ID,
}, },
window::WindowId, window::WindowId,
}; };
use crate::platform_impl::platform::{
app_state::AppState, DEVICE_ID,
event::{check_function_keys, event_mods, modifier_event, char_to_keycode, get_scancode, scancode_to_keycode},
util::{self, IdRef}, ffi::*, window::get_window_id,
};
#[derive(Default)] #[derive(Default)]
struct Modifiers { struct Modifiers {
@ -54,17 +67,18 @@ pub fn new_view(ns_window: id) -> (IdRef, Weak<Mutex<util::Cursor>>) {
// This is free'd in `dealloc` // This is free'd in `dealloc`
let state_ptr = Box::into_raw(Box::new(state)) as *mut c_void; let state_ptr = Box::into_raw(Box::new(state)) as *mut c_void;
let ns_view: id = msg_send![VIEW_CLASS.0, alloc]; let ns_view: id = msg_send![VIEW_CLASS.0, alloc];
(IdRef::new(msg_send![ns_view, initWithWinit:state_ptr]), cursor_access) (
IdRef::new(msg_send![ns_view, initWithWinit: state_ptr]),
cursor_access,
)
} }
} }
pub unsafe fn set_ime_position(ns_view: id, input_context: id, x: f64, y: f64) { pub unsafe fn set_ime_position(ns_view: id, input_context: id, x: f64, y: f64) {
let state_ptr: *mut c_void = *(*ns_view).get_mut_ivar("winitState"); let state_ptr: *mut c_void = *(*ns_view).get_mut_ivar("winitState");
let state = &mut *(state_ptr as *mut ViewState); let state = &mut *(state_ptr as *mut ViewState);
let content_rect = NSWindow::contentRectForFrameRect_( let content_rect =
state.ns_window, NSWindow::contentRectForFrameRect_(state.ns_window, NSWindow::frame(state.ns_window));
NSWindow::frame(state.ns_window),
);
let base_x = content_rect.origin.x as f64; let base_x = content_rect.origin.x as f64;
let base_y = (content_rect.origin.y + content_rect.size.height) as f64; let base_y = (content_rect.origin.y + content_rect.size.height) as f64;
state.ime_spot = Some((base_x + x, base_y - y)); state.ime_spot = Some((base_x + x, base_y - y));
@ -79,161 +93,148 @@ lazy_static! {
static ref VIEW_CLASS: ViewClass = unsafe { static ref VIEW_CLASS: ViewClass = unsafe {
let superclass = class!(NSView); let superclass = class!(NSView);
let mut decl = ClassDecl::new("WinitView", superclass).unwrap(); let mut decl = ClassDecl::new("WinitView", superclass).unwrap();
decl.add_method( decl.add_method(sel!(dealloc), dealloc as extern "C" fn(&Object, Sel));
sel!(dealloc),
dealloc as extern fn(&Object, Sel),
);
decl.add_method( decl.add_method(
sel!(initWithWinit:), sel!(initWithWinit:),
init_with_winit as extern fn(&Object, Sel, *mut c_void) -> id, init_with_winit as extern "C" fn(&Object, Sel, *mut c_void) -> id,
); );
decl.add_method( decl.add_method(
sel!(viewDidMoveToWindow), sel!(viewDidMoveToWindow),
view_did_move_to_window as extern fn(&Object, Sel), view_did_move_to_window as extern "C" fn(&Object, Sel),
); );
decl.add_method( decl.add_method(
sel!(drawRect:), sel!(drawRect:),
draw_rect as extern fn(&Object, Sel, id), draw_rect as extern "C" fn(&Object, Sel, id),
); );
decl.add_method( decl.add_method(
sel!(acceptsFirstResponder), sel!(acceptsFirstResponder),
accepts_first_responder as extern fn(&Object, Sel) -> BOOL, accepts_first_responder as extern "C" fn(&Object, Sel) -> BOOL,
); );
decl.add_method( decl.add_method(
sel!(touchBar), sel!(touchBar),
touch_bar as extern fn(&Object, Sel) -> BOOL, touch_bar as extern "C" fn(&Object, Sel) -> BOOL,
); );
decl.add_method( decl.add_method(
sel!(resetCursorRects), sel!(resetCursorRects),
reset_cursor_rects as extern fn(&Object, Sel), reset_cursor_rects as extern "C" fn(&Object, Sel),
); );
decl.add_method( decl.add_method(
sel!(hasMarkedText), sel!(hasMarkedText),
has_marked_text as extern fn(&Object, Sel) -> BOOL, has_marked_text as extern "C" fn(&Object, Sel) -> BOOL,
); );
decl.add_method( decl.add_method(
sel!(markedRange), sel!(markedRange),
marked_range as extern fn(&Object, Sel) -> NSRange, marked_range as extern "C" fn(&Object, Sel) -> NSRange,
); );
decl.add_method( decl.add_method(
sel!(selectedRange), sel!(selectedRange),
selected_range as extern fn(&Object, Sel) -> NSRange, selected_range as extern "C" fn(&Object, Sel) -> NSRange,
); );
decl.add_method( decl.add_method(
sel!(setMarkedText:selectedRange:replacementRange:), sel!(setMarkedText:selectedRange:replacementRange:),
set_marked_text as extern fn(&mut Object, Sel, id, NSRange, NSRange), set_marked_text as extern "C" fn(&mut Object, Sel, id, NSRange, NSRange),
);
decl.add_method(
sel!(unmarkText),
unmark_text as extern fn(&Object, Sel),
); );
decl.add_method(sel!(unmarkText), unmark_text as extern "C" fn(&Object, Sel));
decl.add_method( decl.add_method(
sel!(validAttributesForMarkedText), sel!(validAttributesForMarkedText),
valid_attributes_for_marked_text as extern fn(&Object, Sel) -> id, valid_attributes_for_marked_text as extern "C" fn(&Object, Sel) -> id,
); );
decl.add_method( decl.add_method(
sel!(attributedSubstringForProposedRange:actualRange:), sel!(attributedSubstringForProposedRange:actualRange:),
attributed_substring_for_proposed_range as extern fn(&Object, Sel, NSRange, *mut c_void) -> id, attributed_substring_for_proposed_range
as extern "C" fn(&Object, Sel, NSRange, *mut c_void) -> id,
); );
decl.add_method( decl.add_method(
sel!(insertText:replacementRange:), sel!(insertText:replacementRange:),
insert_text as extern fn(&Object, Sel, id, NSRange), insert_text as extern "C" fn(&Object, Sel, id, NSRange),
); );
decl.add_method( decl.add_method(
sel!(characterIndexForPoint:), sel!(characterIndexForPoint:),
character_index_for_point as extern fn(&Object, Sel, NSPoint) -> NSUInteger, character_index_for_point as extern "C" fn(&Object, Sel, NSPoint) -> NSUInteger,
); );
decl.add_method( decl.add_method(
sel!(firstRectForCharacterRange:actualRange:), sel!(firstRectForCharacterRange:actualRange:),
first_rect_for_character_range as extern fn(&Object, Sel, NSRange, *mut c_void) -> NSRect, first_rect_for_character_range
as extern "C" fn(&Object, Sel, NSRange, *mut c_void) -> NSRect,
); );
decl.add_method( decl.add_method(
sel!(doCommandBySelector:), sel!(doCommandBySelector:),
do_command_by_selector as extern fn(&Object, Sel, Sel), do_command_by_selector as extern "C" fn(&Object, Sel, Sel),
);
decl.add_method(
sel!(keyDown:),
key_down as extern fn(&Object, Sel, id),
);
decl.add_method(
sel!(keyUp:),
key_up as extern fn(&Object, Sel, id),
); );
decl.add_method(sel!(keyDown:), key_down as extern "C" fn(&Object, Sel, id));
decl.add_method(sel!(keyUp:), key_up as extern "C" fn(&Object, Sel, id));
decl.add_method( decl.add_method(
sel!(flagsChanged:), sel!(flagsChanged:),
flags_changed as extern fn(&Object, Sel, id), flags_changed as extern "C" fn(&Object, Sel, id),
); );
decl.add_method( decl.add_method(
sel!(insertTab:), sel!(insertTab:),
insert_tab as extern fn(&Object, Sel, id), insert_tab as extern "C" fn(&Object, Sel, id),
); );
decl.add_method( decl.add_method(
sel!(insertBackTab:), sel!(insertBackTab:),
insert_back_tab as extern fn(&Object, Sel, id), insert_back_tab as extern "C" fn(&Object, Sel, id),
); );
decl.add_method( decl.add_method(
sel!(mouseDown:), sel!(mouseDown:),
mouse_down as extern fn(&Object, Sel, id), mouse_down as extern "C" fn(&Object, Sel, id),
);
decl.add_method(
sel!(mouseUp:),
mouse_up as extern fn(&Object, Sel, id),
); );
decl.add_method(sel!(mouseUp:), mouse_up as extern "C" fn(&Object, Sel, id));
decl.add_method( decl.add_method(
sel!(rightMouseDown:), sel!(rightMouseDown:),
right_mouse_down as extern fn(&Object, Sel, id), right_mouse_down as extern "C" fn(&Object, Sel, id),
); );
decl.add_method( decl.add_method(
sel!(rightMouseUp:), sel!(rightMouseUp:),
right_mouse_up as extern fn(&Object, Sel, id), right_mouse_up as extern "C" fn(&Object, Sel, id),
); );
decl.add_method( decl.add_method(
sel!(otherMouseDown:), sel!(otherMouseDown:),
other_mouse_down as extern fn(&Object, Sel, id), other_mouse_down as extern "C" fn(&Object, Sel, id),
); );
decl.add_method( decl.add_method(
sel!(otherMouseUp:), sel!(otherMouseUp:),
other_mouse_up as extern fn(&Object, Sel, id), other_mouse_up as extern "C" fn(&Object, Sel, id),
); );
decl.add_method( decl.add_method(
sel!(mouseMoved:), sel!(mouseMoved:),
mouse_moved as extern fn(&Object, Sel, id), mouse_moved as extern "C" fn(&Object, Sel, id),
); );
decl.add_method( decl.add_method(
sel!(mouseDragged:), sel!(mouseDragged:),
mouse_dragged as extern fn(&Object, Sel, id), mouse_dragged as extern "C" fn(&Object, Sel, id),
); );
decl.add_method( decl.add_method(
sel!(rightMouseDragged:), sel!(rightMouseDragged:),
right_mouse_dragged as extern fn(&Object, Sel, id), right_mouse_dragged as extern "C" fn(&Object, Sel, id),
); );
decl.add_method( decl.add_method(
sel!(otherMouseDragged:), sel!(otherMouseDragged:),
other_mouse_dragged as extern fn(&Object, Sel, id), other_mouse_dragged as extern "C" fn(&Object, Sel, id),
); );
decl.add_method( decl.add_method(
sel!(mouseEntered:), sel!(mouseEntered:),
mouse_entered as extern fn(&Object, Sel, id), mouse_entered as extern "C" fn(&Object, Sel, id),
); );
decl.add_method( decl.add_method(
sel!(mouseExited:), sel!(mouseExited:),
mouse_exited as extern fn(&Object, Sel, id), mouse_exited as extern "C" fn(&Object, Sel, id),
); );
decl.add_method( decl.add_method(
sel!(scrollWheel:), sel!(scrollWheel:),
scroll_wheel as extern fn(&Object, Sel, id), scroll_wheel as extern "C" fn(&Object, Sel, id),
); );
decl.add_method( decl.add_method(
sel!(pressureChangeWithEvent:), sel!(pressureChangeWithEvent:),
pressure_change_with_event as extern fn(&Object, Sel, id), pressure_change_with_event as extern "C" fn(&Object, Sel, id),
); );
decl.add_method( decl.add_method(
sel!(_wantsKeyDownForEvent:), sel!(_wantsKeyDownForEvent:),
wants_key_down_for_event as extern fn(&Object, Sel, id) -> BOOL, wants_key_down_for_event as extern "C" fn(&Object, Sel, id) -> BOOL,
); );
decl.add_method( decl.add_method(
sel!(cancelOperation:), sel!(cancelOperation:),
cancel_operation as extern fn(&Object, Sel, id), cancel_operation as extern "C" fn(&Object, Sel, id),
); );
decl.add_ivar::<*mut c_void>("winitState"); decl.add_ivar::<*mut c_void>("winitState");
decl.add_ivar::<id>("markedText"); decl.add_ivar::<id>("markedText");
@ -243,7 +244,7 @@ lazy_static! {
}; };
} }
extern fn dealloc(this: &Object, _sel: Sel) { extern "C" fn dealloc(this: &Object, _sel: Sel) {
unsafe { unsafe {
let state: *mut c_void = *this.get_ivar("winitState"); let state: *mut c_void = *this.get_ivar("winitState");
let marked_text: id = *this.get_ivar("markedText"); let marked_text: id = *this.get_ivar("markedText");
@ -252,21 +253,20 @@ extern fn dealloc(this: &Object, _sel: Sel) {
} }
} }
extern fn init_with_winit(this: &Object, _sel: Sel, state: *mut c_void) -> id { extern "C" fn init_with_winit(this: &Object, _sel: Sel, state: *mut c_void) -> id {
unsafe { unsafe {
let this: id = msg_send![this, init]; let this: id = msg_send![this, init];
if this != nil { if this != nil {
(*this).set_ivar("winitState", state); (*this).set_ivar("winitState", state);
let marked_text = <id as NSMutableAttributedString>::init( let marked_text =
NSMutableAttributedString::alloc(nil), <id as NSMutableAttributedString>::init(NSMutableAttributedString::alloc(nil));
);
(*this).set_ivar("markedText", marked_text); (*this).set_ivar("markedText", marked_text);
} }
this this
} }
} }
extern fn view_did_move_to_window(this: &Object, _sel: Sel) { extern "C" fn view_did_move_to_window(this: &Object, _sel: Sel) {
trace!("Triggered `viewDidMoveToWindow`"); trace!("Triggered `viewDidMoveToWindow`");
unsafe { unsafe {
let rect: NSRect = msg_send![this, visibleRect]; let rect: NSRect = msg_send![this, visibleRect];
@ -280,7 +280,7 @@ extern fn view_did_move_to_window(this: &Object, _sel: Sel) {
trace!("Completed `viewDidMoveToWindow`"); trace!("Completed `viewDidMoveToWindow`");
} }
extern fn draw_rect(this: &Object, _sel: Sel, rect: id) { extern "C" fn draw_rect(this: &Object, _sel: Sel, rect: id) {
unsafe { unsafe {
let state_ptr: *mut c_void = *this.get_ivar("winitState"); let state_ptr: *mut c_void = *this.get_ivar("winitState");
let state = &mut *(state_ptr as *mut ViewState); let state = &mut *(state_ptr as *mut ViewState);
@ -288,22 +288,22 @@ extern fn draw_rect(this: &Object, _sel: Sel, rect: id) {
AppState::queue_redraw(WindowId(get_window_id(state.ns_window))); AppState::queue_redraw(WindowId(get_window_id(state.ns_window)));
let superclass = util::superclass(this); let superclass = util::superclass(this);
let () = msg_send![super(this, superclass), drawRect:rect]; let () = msg_send![super(this, superclass), drawRect: rect];
} }
} }
extern fn accepts_first_responder(_this: &Object, _sel: Sel) -> BOOL { extern "C" fn accepts_first_responder(_this: &Object, _sel: Sel) -> BOOL {
YES YES
} }
// This is necessary to prevent a beefy terminal error on MacBook Pros: // This is necessary to prevent a beefy terminal error on MacBook Pros:
// IMKInputSession [0x7fc573576ff0 presentFunctionRowItemTextInputViewWithEndpoint:completionHandler:] : [self textInputContext]=0x7fc573558e10 *NO* NSRemoteViewController to client, NSError=Error Domain=NSCocoaErrorDomain Code=4099 "The connection from pid 0 was invalidated from this process." UserInfo={NSDebugDescription=The connection from pid 0 was invalidated from this process.}, com.apple.inputmethod.EmojiFunctionRowItem // IMKInputSession [0x7fc573576ff0 presentFunctionRowItemTextInputViewWithEndpoint:completionHandler:] : [self textInputContext]=0x7fc573558e10 *NO* NSRemoteViewController to client, NSError=Error Domain=NSCocoaErrorDomain Code=4099 "The connection from pid 0 was invalidated from this process." UserInfo={NSDebugDescription=The connection from pid 0 was invalidated from this process.}, com.apple.inputmethod.EmojiFunctionRowItem
// TODO: Add an API extension for using `NSTouchBar` // TODO: Add an API extension for using `NSTouchBar`
extern fn touch_bar(_this: &Object, _sel: Sel) -> BOOL { extern "C" fn touch_bar(_this: &Object, _sel: Sel) -> BOOL {
NO NO
} }
extern fn reset_cursor_rects(this: &Object, _sel: Sel) { extern "C" fn reset_cursor_rects(this: &Object, _sel: Sel) {
unsafe { unsafe {
let state_ptr: *mut c_void = *this.get_ivar("winitState"); let state_ptr: *mut c_void = *this.get_ivar("winitState");
let state = &mut *(state_ptr as *mut ViewState); let state = &mut *(state_ptr as *mut ViewState);
@ -317,8 +317,7 @@ extern fn reset_cursor_rects(this: &Object, _sel: Sel) {
} }
} }
extern "C" fn has_marked_text(this: &Object, _sel: Sel) -> BOOL {
extern fn has_marked_text(this: &Object, _sel: Sel) -> BOOL {
unsafe { unsafe {
trace!("Triggered `hasMarkedText`"); trace!("Triggered `hasMarkedText`");
let marked_text: id = *this.get_ivar("markedText"); let marked_text: id = *this.get_ivar("markedText");
@ -327,7 +326,7 @@ extern fn has_marked_text(this: &Object, _sel: Sel) -> BOOL {
} }
} }
extern fn marked_range(this: &Object, _sel: Sel) -> NSRange { extern "C" fn marked_range(this: &Object, _sel: Sel) -> NSRange {
unsafe { unsafe {
trace!("Triggered `markedRange`"); trace!("Triggered `markedRange`");
let marked_text: id = *this.get_ivar("markedText"); let marked_text: id = *this.get_ivar("markedText");
@ -341,13 +340,13 @@ extern fn marked_range(this: &Object, _sel: Sel) -> NSRange {
} }
} }
extern fn selected_range(_this: &Object, _sel: Sel) -> NSRange { extern "C" fn selected_range(_this: &Object, _sel: Sel) -> NSRange {
trace!("Triggered `selectedRange`"); trace!("Triggered `selectedRange`");
trace!("Completed `selectedRange`"); trace!("Completed `selectedRange`");
util::EMPTY_RANGE util::EMPTY_RANGE
} }
extern fn set_marked_text( extern "C" fn set_marked_text(
this: &mut Object, this: &mut Object,
_sel: Sel, _sel: Sel,
string: id, string: id,
@ -359,7 +358,7 @@ extern fn set_marked_text(
let marked_text_ref: &mut id = this.get_mut_ivar("markedText"); let marked_text_ref: &mut id = this.get_mut_ivar("markedText");
let _: () = msg_send![(*marked_text_ref), release]; let _: () = msg_send![(*marked_text_ref), release];
let marked_text = NSMutableAttributedString::alloc(nil); let marked_text = NSMutableAttributedString::alloc(nil);
let has_attr = msg_send![string, isKindOfClass:class!(NSAttributedString)]; let has_attr = msg_send![string, isKindOfClass: class!(NSAttributedString)];
if has_attr { if has_attr {
marked_text.initWithAttributedString(string); marked_text.initWithAttributedString(string);
} else { } else {
@ -370,7 +369,7 @@ extern fn set_marked_text(
trace!("Completed `setMarkedText`"); trace!("Completed `setMarkedText`");
} }
extern fn unmark_text(this: &Object, _sel: Sel) { extern "C" fn unmark_text(this: &Object, _sel: Sel) {
trace!("Triggered `unmarkText`"); trace!("Triggered `unmarkText`");
unsafe { unsafe {
let marked_text: id = *this.get_ivar("markedText"); let marked_text: id = *this.get_ivar("markedText");
@ -382,13 +381,13 @@ extern fn unmark_text(this: &Object, _sel: Sel) {
trace!("Completed `unmarkText`"); trace!("Completed `unmarkText`");
} }
extern fn valid_attributes_for_marked_text(_this: &Object, _sel: Sel) -> id { extern "C" fn valid_attributes_for_marked_text(_this: &Object, _sel: Sel) -> id {
trace!("Triggered `validAttributesForMarkedText`"); trace!("Triggered `validAttributesForMarkedText`");
trace!("Completed `validAttributesForMarkedText`"); trace!("Completed `validAttributesForMarkedText`");
unsafe { msg_send![class!(NSArray), array] } unsafe { msg_send![class!(NSArray), array] }
} }
extern fn attributed_substring_for_proposed_range( extern "C" fn attributed_substring_for_proposed_range(
_this: &Object, _this: &Object,
_sel: Sel, _sel: Sel,
_range: NSRange, _range: NSRange,
@ -399,13 +398,13 @@ extern fn attributed_substring_for_proposed_range(
nil nil
} }
extern fn character_index_for_point(_this: &Object, _sel: Sel, _point: NSPoint) -> NSUInteger { extern "C" fn character_index_for_point(_this: &Object, _sel: Sel, _point: NSPoint) -> NSUInteger {
trace!("Triggered `characterIndexForPoint`"); trace!("Triggered `characterIndexForPoint`");
trace!("Completed `characterIndexForPoint`"); trace!("Completed `characterIndexForPoint`");
0 0
} }
extern fn first_rect_for_character_range( extern "C" fn first_rect_for_character_range(
this: &Object, this: &Object,
_sel: Sel, _sel: Sel,
_range: NSRange, _range: NSRange,
@ -425,20 +424,17 @@ extern fn first_rect_for_character_range(
(x, y) (x, y)
}); });
trace!("Completed `firstRectForCharacterRange`"); trace!("Completed `firstRectForCharacterRange`");
NSRect::new( NSRect::new(NSPoint::new(x as _, y as _), NSSize::new(0.0, 0.0))
NSPoint::new(x as _, y as _),
NSSize::new(0.0, 0.0),
)
} }
} }
extern fn insert_text(this: &Object, _sel: Sel, string: id, _replacement_range: NSRange) { extern "C" fn insert_text(this: &Object, _sel: Sel, string: id, _replacement_range: NSRange) {
trace!("Triggered `insertText`"); trace!("Triggered `insertText`");
unsafe { unsafe {
let state_ptr: *mut c_void = *this.get_ivar("winitState"); let state_ptr: *mut c_void = *this.get_ivar("winitState");
let state = &mut *(state_ptr as *mut ViewState); let state = &mut *(state_ptr as *mut ViewState);
let has_attr = msg_send![string, isKindOfClass:class!(NSAttributedString)]; let has_attr = msg_send![string, isKindOfClass: class!(NSAttributedString)];
let characters = if has_attr { let characters = if has_attr {
// This is a *mut NSAttributedString // This is a *mut NSAttributedString
msg_send![string, string] msg_send![string, string]
@ -447,10 +443,8 @@ extern fn insert_text(this: &Object, _sel: Sel, string: id, _replacement_range:
string string
}; };
let slice = slice::from_raw_parts( let slice =
characters.UTF8String() as *const c_uchar, slice::from_raw_parts(characters.UTF8String() as *const c_uchar, characters.len());
characters.len(),
);
let string = str::from_utf8_unchecked(slice); let string = str::from_utf8_unchecked(slice);
state.is_key_down = true; state.is_key_down = true;
@ -470,7 +464,7 @@ extern fn insert_text(this: &Object, _sel: Sel, string: id, _replacement_range:
trace!("Completed `insertText`"); trace!("Completed `insertText`");
} }
extern fn do_command_by_selector(this: &Object, _sel: Sel, command: Sel) { extern "C" fn do_command_by_selector(this: &Object, _sel: Sel, command: Sel) {
trace!("Triggered `doCommandBySelector`"); trace!("Triggered `doCommandBySelector`");
// Basically, we're sent this message whenever a keyboard event that doesn't generate a "human readable" character // Basically, we're sent this message whenever a keyboard event that doesn't generate a "human readable" character
// happens, i.e. newlines, tabs, and Ctrl+C. // happens, i.e. newlines, tabs, and Ctrl+C.
@ -513,10 +507,8 @@ fn get_characters(event: id, ignore_modifiers: bool) -> String {
}; };
assert_ne!(characters, nil); assert_ne!(characters, nil);
let slice = slice::from_raw_parts( let slice =
characters.UTF8String() as *const c_uchar, slice::from_raw_parts(characters.UTF8String() as *const c_uchar, characters.len());
characters.len(),
);
let string = str::from_utf8_unchecked(slice); let string = str::from_utf8_unchecked(slice);
string.to_owned() string.to_owned()
@ -528,15 +520,15 @@ fn retrieve_keycode(event: id) -> Option<VirtualKeyCode> {
#[inline] #[inline]
fn get_code(ev: id, raw: bool) -> Option<VirtualKeyCode> { fn get_code(ev: id, raw: bool) -> Option<VirtualKeyCode> {
let characters = get_characters(ev, raw); let characters = get_characters(ev, raw);
characters.chars().next().map_or(None, |c| char_to_keycode(c)) characters
.chars()
.next()
.map_or(None, |c| char_to_keycode(c))
} }
// Cmd switches Roman letters for Dvorak-QWERTY layout, so we try modified characters first. // Cmd switches Roman letters for Dvorak-QWERTY layout, so we try modified characters first.
// If we don't get a match, then we fall back to unmodified characters. // If we don't get a match, then we fall back to unmodified characters.
let code = get_code(event, false) let code = get_code(event, false).or_else(|| get_code(event, true));
.or_else(|| {
get_code(event, true)
});
// We've checked all layout related keys, so fall through to scancode. // We've checked all layout related keys, so fall through to scancode.
// Reaching this code means that the key is layout-independent (e.g. Backspace, Return). // Reaching this code means that the key is layout-independent (e.g. Backspace, Return).
@ -546,14 +538,11 @@ fn retrieve_keycode(event: id) -> Option<VirtualKeyCode> {
// in characters property. // in characters property.
code.or_else(|| { code.or_else(|| {
let scancode = get_scancode(event); let scancode = get_scancode(event);
scancode_to_keycode(scancode) scancode_to_keycode(scancode).or_else(|| check_function_keys(&get_characters(event, true)))
.or_else(|| {
check_function_keys(&get_characters(event, true))
})
}) })
} }
extern fn key_down(this: &Object, _sel: Sel, event: id) { extern "C" fn key_down(this: &Object, _sel: Sel, event: id) {
trace!("Triggered `keyDown`"); trace!("Triggered `keyDown`");
unsafe { unsafe {
let state_ptr: *mut c_void = *this.get_ivar("winitState"); let state_ptr: *mut c_void = *this.get_ivar("winitState");
@ -601,14 +590,14 @@ extern fn key_down(this: &Object, _sel: Sel, event: id) {
// Some keys (and only *some*, with no known reason) don't trigger `insertText`, while others do... // Some keys (and only *some*, with no known reason) don't trigger `insertText`, while others do...
// So, we don't give repeats the opportunity to trigger that, since otherwise our hack will cause some // So, we don't give repeats the opportunity to trigger that, since otherwise our hack will cause some
// keys to generate twice as many characters. // keys to generate twice as many characters.
let array: id = msg_send![class!(NSArray), arrayWithObject:event]; let array: id = msg_send![class!(NSArray), arrayWithObject: event];
let _: () = msg_send![this, interpretKeyEvents:array]; let _: () = msg_send![this, interpretKeyEvents: array];
} }
} }
trace!("Completed `keyDown`"); trace!("Completed `keyDown`");
} }
extern fn key_up(this: &Object, _sel: Sel, event: id) { extern "C" fn key_up(this: &Object, _sel: Sel, event: id) {
trace!("Triggered `keyUp`"); trace!("Triggered `keyUp`");
unsafe { unsafe {
let state_ptr: *mut c_void = *this.get_ivar("winitState"); let state_ptr: *mut c_void = *this.get_ivar("winitState");
@ -637,7 +626,7 @@ extern fn key_up(this: &Object, _sel: Sel, event: id) {
trace!("Completed `keyUp`"); trace!("Completed `keyUp`");
} }
extern fn flags_changed(this: &Object, _sel: Sel, event: id) { extern "C" fn flags_changed(this: &Object, _sel: Sel, event: id) {
trace!("Triggered `flagsChanged`"); trace!("Triggered `flagsChanged`");
unsafe { unsafe {
let state_ptr: *mut c_void = *this.get_ivar("winitState"); let state_ptr: *mut c_void = *this.get_ivar("winitState");
@ -691,31 +680,31 @@ extern fn flags_changed(this: &Object, _sel: Sel, event: id) {
trace!("Completed `flagsChanged`"); trace!("Completed `flagsChanged`");
} }
extern fn insert_tab(this: &Object, _sel: Sel, _sender: id) { extern "C" fn insert_tab(this: &Object, _sel: Sel, _sender: id) {
unsafe { unsafe {
let window: id = msg_send![this, window]; let window: id = msg_send![this, window];
let first_responder: id = msg_send![window, firstResponder]; let first_responder: id = msg_send![window, firstResponder];
let this_ptr = this as *const _ as *mut _; let this_ptr = this as *const _ as *mut _;
if first_responder == this_ptr { if first_responder == this_ptr {
let (): _ = msg_send![window, selectNextKeyView:this]; let (): _ = msg_send![window, selectNextKeyView: this];
} }
} }
} }
extern fn insert_back_tab(this: &Object, _sel: Sel, _sender: id) { extern "C" fn insert_back_tab(this: &Object, _sel: Sel, _sender: id) {
unsafe { unsafe {
let window: id = msg_send![this, window]; let window: id = msg_send![this, window];
let first_responder: id = msg_send![window, firstResponder]; let first_responder: id = msg_send![window, firstResponder];
let this_ptr = this as *const _ as *mut _; let this_ptr = this as *const _ as *mut _;
if first_responder == this_ptr { if first_responder == this_ptr {
let (): _ = msg_send![window, selectPreviousKeyView:this]; let (): _ = msg_send![window, selectPreviousKeyView: this];
} }
} }
} }
// Allows us to receive Cmd-. (the shortcut for closing a dialog) // Allows us to receive Cmd-. (the shortcut for closing a dialog)
// https://bugs.eclipse.org/bugs/show_bug.cgi?id=300620#c6 // https://bugs.eclipse.org/bugs/show_bug.cgi?id=300620#c6
extern fn cancel_operation(this: &Object, _sel: Sel, _sender: id) { extern "C" fn cancel_operation(this: &Object, _sel: Sel, _sender: id) {
trace!("Triggered `cancelOperation`"); trace!("Triggered `cancelOperation`");
unsafe { unsafe {
let state_ptr: *mut c_void = *this.get_ivar("winitState"); let state_ptr: *mut c_void = *this.get_ivar("winitState");
@ -764,27 +753,27 @@ fn mouse_click(this: &Object, event: id, button: MouseButton, button_state: Elem
} }
} }
extern fn mouse_down(this: &Object, _sel: Sel, event: id) { extern "C" fn mouse_down(this: &Object, _sel: Sel, event: id) {
mouse_click(this, event, MouseButton::Left, ElementState::Pressed); mouse_click(this, event, MouseButton::Left, ElementState::Pressed);
} }
extern fn mouse_up(this: &Object, _sel: Sel, event: id) { extern "C" fn mouse_up(this: &Object, _sel: Sel, event: id) {
mouse_click(this, event, MouseButton::Left, ElementState::Released); mouse_click(this, event, MouseButton::Left, ElementState::Released);
} }
extern fn right_mouse_down(this: &Object, _sel: Sel, event: id) { extern "C" fn right_mouse_down(this: &Object, _sel: Sel, event: id) {
mouse_click(this, event, MouseButton::Right, ElementState::Pressed); mouse_click(this, event, MouseButton::Right, ElementState::Pressed);
} }
extern fn right_mouse_up(this: &Object, _sel: Sel, event: id) { extern "C" fn right_mouse_up(this: &Object, _sel: Sel, event: id) {
mouse_click(this, event, MouseButton::Right, ElementState::Released); mouse_click(this, event, MouseButton::Right, ElementState::Released);
} }
extern fn other_mouse_down(this: &Object, _sel: Sel, event: id) { extern "C" fn other_mouse_down(this: &Object, _sel: Sel, event: id) {
mouse_click(this, event, MouseButton::Middle, ElementState::Pressed); mouse_click(this, event, MouseButton::Middle, ElementState::Pressed);
} }
extern fn other_mouse_up(this: &Object, _sel: Sel, event: id) { extern "C" fn other_mouse_up(this: &Object, _sel: Sel, event: id) {
mouse_click(this, event, MouseButton::Middle, ElementState::Released); mouse_click(this, event, MouseButton::Middle, ElementState::Released);
} }
@ -801,9 +790,10 @@ fn mouse_motion(this: &Object, event: id) {
let view_rect = NSView::frame(view); let view_rect = NSView::frame(view);
if view_point.x.is_sign_negative() if view_point.x.is_sign_negative()
|| view_point.y.is_sign_negative() || view_point.y.is_sign_negative()
|| view_point.x > view_rect.size.width || view_point.x > view_rect.size.width
|| view_point.y > view_rect.size.height { || view_point.y > view_rect.size.height
{
// Point is outside of the client area (view) // Point is outside of the client area (view)
return; return;
} }
@ -824,23 +814,23 @@ fn mouse_motion(this: &Object, event: id) {
} }
} }
extern fn mouse_moved(this: &Object, _sel: Sel, event: id) { extern "C" fn mouse_moved(this: &Object, _sel: Sel, event: id) {
mouse_motion(this, event); mouse_motion(this, event);
} }
extern fn mouse_dragged(this: &Object, _sel: Sel, event: id) { extern "C" fn mouse_dragged(this: &Object, _sel: Sel, event: id) {
mouse_motion(this, event); mouse_motion(this, event);
} }
extern fn right_mouse_dragged(this: &Object, _sel: Sel, event: id) { extern "C" fn right_mouse_dragged(this: &Object, _sel: Sel, event: id) {
mouse_motion(this, event); mouse_motion(this, event);
} }
extern fn other_mouse_dragged(this: &Object, _sel: Sel, event: id) { extern "C" fn other_mouse_dragged(this: &Object, _sel: Sel, event: id) {
mouse_motion(this, event); mouse_motion(this, event);
} }
extern fn mouse_entered(this: &Object, _sel: Sel, event: id) { extern "C" fn mouse_entered(this: &Object, _sel: Sel, event: id) {
trace!("Triggered `mouseEntered`"); trace!("Triggered `mouseEntered`");
unsafe { unsafe {
let state_ptr: *mut c_void = *this.get_ivar("winitState"); let state_ptr: *mut c_void = *this.get_ivar("winitState");
@ -848,7 +838,9 @@ extern fn mouse_entered(this: &Object, _sel: Sel, event: id) {
let enter_event = Event::WindowEvent { let enter_event = Event::WindowEvent {
window_id: WindowId(get_window_id(state.ns_window)), window_id: WindowId(get_window_id(state.ns_window)),
event: WindowEvent::CursorEntered { device_id: DEVICE_ID }, event: WindowEvent::CursorEntered {
device_id: DEVICE_ID,
},
}; };
let move_event = { let move_event = {
@ -866,7 +858,7 @@ extern fn mouse_entered(this: &Object, _sel: Sel, event: id) {
device_id: DEVICE_ID, device_id: DEVICE_ID,
position: (x, y).into(), position: (x, y).into(),
modifiers: event_mods(event), modifiers: event_mods(event),
} },
} }
}; };
@ -876,7 +868,7 @@ extern fn mouse_entered(this: &Object, _sel: Sel, event: id) {
trace!("Completed `mouseEntered`"); trace!("Completed `mouseEntered`");
} }
extern fn mouse_exited(this: &Object, _sel: Sel, _event: id) { extern "C" fn mouse_exited(this: &Object, _sel: Sel, _event: id) {
trace!("Triggered `mouseExited`"); trace!("Triggered `mouseExited`");
unsafe { unsafe {
let state_ptr: *mut c_void = *this.get_ivar("winitState"); let state_ptr: *mut c_void = *this.get_ivar("winitState");
@ -884,7 +876,9 @@ extern fn mouse_exited(this: &Object, _sel: Sel, _event: id) {
let window_event = Event::WindowEvent { let window_event = Event::WindowEvent {
window_id: WindowId(get_window_id(state.ns_window)), window_id: WindowId(get_window_id(state.ns_window)),
event: WindowEvent::CursorLeft { device_id: DEVICE_ID }, event: WindowEvent::CursorLeft {
device_id: DEVICE_ID,
},
}; };
AppState::queue_event(window_event); AppState::queue_event(window_event);
@ -892,7 +886,7 @@ extern fn mouse_exited(this: &Object, _sel: Sel, _event: id) {
trace!("Completed `mouseExited`"); trace!("Completed `mouseExited`");
} }
extern fn scroll_wheel(this: &Object, _sel: Sel, event: id) { extern "C" fn scroll_wheel(this: &Object, _sel: Sel, event: id) {
trace!("Triggered `scrollWheel`"); trace!("Triggered `scrollWheel`");
unsafe { unsafe {
let delta = { let delta = {
@ -904,7 +898,9 @@ extern fn scroll_wheel(this: &Object, _sel: Sel, event: id) {
} }
}; };
let phase = match event.phase() { let phase = match event.phase() {
NSEventPhase::NSEventPhaseMayBegin | NSEventPhase::NSEventPhaseBegan => TouchPhase::Started, NSEventPhase::NSEventPhaseMayBegin | NSEventPhase::NSEventPhaseBegan => {
TouchPhase::Started
},
NSEventPhase::NSEventPhaseEnded => TouchPhase::Ended, NSEventPhase::NSEventPhaseEnded => TouchPhase::Ended,
_ => TouchPhase::Moved, _ => TouchPhase::Moved,
}; };
@ -933,7 +929,7 @@ extern fn scroll_wheel(this: &Object, _sel: Sel, event: id) {
trace!("Completed `scrollWheel`"); trace!("Completed `scrollWheel`");
} }
extern fn pressure_change_with_event(this: &Object, _sel: Sel, event: id) { extern "C" fn pressure_change_with_event(this: &Object, _sel: Sel, event: id) {
trace!("Triggered `pressureChangeWithEvent`"); trace!("Triggered `pressureChangeWithEvent`");
unsafe { unsafe {
let state_ptr: *mut c_void = *this.get_ivar("winitState"); let state_ptr: *mut c_void = *this.get_ivar("winitState");
@ -959,6 +955,6 @@ extern fn pressure_change_with_event(this: &Object, _sel: Sel, event: id) {
// Allows us to receive Ctrl-Tab and Ctrl-Esc. // Allows us to receive Ctrl-Tab and Ctrl-Esc.
// Note that this *doesn't* help with any missing Cmd inputs. // Note that this *doesn't* help with any missing Cmd inputs.
// https://github.com/chromium/chromium/blob/a86a8a6bcfa438fa3ac2eba6f02b3ad1f8e0756f/ui/views/cocoa/bridged_content_view.mm#L816 // https://github.com/chromium/chromium/blob/a86a8a6bcfa438fa3ac2eba6f02b3ad1f8e0756f/ui/views/cocoa/bridged_content_view.mm#L816
extern fn wants_key_down_for_event(_this: &Object, _sel: Sel, _event: id) -> BOOL { extern "C" fn wants_key_down_for_event(_this: &Object, _sel: Sel, _event: id) -> BOOL {
YES YES
} }

View file

@ -1,34 +1,44 @@
use std::{ use std::{
collections::VecDeque, f64, os::raw::c_void, collections::VecDeque,
sync::{Arc, atomic::{Ordering, AtomicBool}, Mutex, Weak}, f64,
os::raw::c_void,
sync::{
atomic::{AtomicBool, Ordering},
Arc, Mutex, Weak,
},
}; };
use cocoa::{ use cocoa::{
appkit::{ appkit::{
self, CGFloat, NSApp, NSApplication, NSApplicationActivationPolicy, self, CGFloat, NSApp, NSApplication, NSApplicationActivationPolicy,
NSColor, NSRequestUserAttentionType, NSScreen, NSView, NSWindow, NSApplicationPresentationOptions, NSColor, NSRequestUserAttentionType, NSScreen, NSView,
NSWindowButton, NSWindowStyleMask, NSApplicationPresentationOptions NSWindow, NSWindowButton, NSWindowStyleMask,
}, },
base::{id, nil}, base::{id, nil},
foundation::{NSAutoreleasePool, NSDictionary, NSPoint, NSRect, NSSize, NSString}, foundation::{NSAutoreleasePool, NSDictionary, NSPoint, NSRect, NSSize, NSString},
}; };
use core_graphics::display::CGDisplay; use core_graphics::display::CGDisplay;
use objc::{runtime::{Class, Object, Sel, BOOL, YES, NO}, declare::ClassDecl}; use objc::{
declare::ClassDecl,
runtime::{Class, Object, Sel, BOOL, NO, YES},
};
use crate::{ use crate::{
dpi::{LogicalPosition, LogicalSize}, icon::Icon, dpi::{LogicalPosition, LogicalSize},
error::{ExternalError, NotSupportedError, OsError as RootOsError}, error::{ExternalError, NotSupportedError, OsError as RootOsError},
icon::Icon,
monitor::MonitorHandle as RootMonitorHandle, monitor::MonitorHandle as RootMonitorHandle,
window::{ platform::macos::{ActivationPolicy, WindowExtMacOS},
CursorIcon, WindowAttributes, WindowId as RootWindowId, platform_impl::platform::{
app_state::AppState,
ffi,
monitor::{self, MonitorHandle},
util::{self, IdRef},
view::{self, new_view},
window_delegate::new_delegate,
OsError,
}, },
}; window::{CursorIcon, WindowAttributes, WindowId as RootWindowId},
use crate::platform::macos::{ActivationPolicy, WindowExtMacOS};
use crate::platform_impl::platform::{
OsError,
app_state::AppState, ffi, monitor::{self, MonitorHandle},
util::{self, IdRef}, view::{self, new_view},
window_delegate::new_delegate,
}; };
#[derive(Debug, Copy, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)] #[derive(Debug, Copy, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
@ -112,7 +122,8 @@ fn create_window(
let frame = match screen { let frame = match screen {
Some(screen) => appkit::NSScreen::frame(screen), Some(screen) => appkit::NSScreen::frame(screen),
None => { None => {
let (width, height) = attrs.inner_size let (width, height) = attrs
.inner_size
.map(|logical| (logical.width, logical.height)) .map(|logical| (logical.width, logical.height))
.unwrap_or_else(|| (800.0, 600.0)); .unwrap_or_else(|| (800.0, 600.0));
NSRect::new(NSPoint::new(0.0, 0.0), NSSize::new(width, height)) NSRect::new(NSPoint::new(0.0, 0.0), NSSize::new(width, height))
@ -122,18 +133,16 @@ fn create_window(
let mut masks = if !attrs.decorations && !screen.is_some() { let mut masks = if !attrs.decorations && !screen.is_some() {
// Resizable UnownedWindow without a titlebar or borders // Resizable UnownedWindow without a titlebar or borders
// if decorations is set to false, ignore pl_attrs // if decorations is set to false, ignore pl_attrs
NSWindowStyleMask::NSBorderlessWindowMask NSWindowStyleMask::NSBorderlessWindowMask | NSWindowStyleMask::NSResizableWindowMask
| NSWindowStyleMask::NSResizableWindowMask
} else if pl_attrs.titlebar_hidden { } else if pl_attrs.titlebar_hidden {
// if the titlebar is hidden, ignore other pl_attrs // if the titlebar is hidden, ignore other pl_attrs
NSWindowStyleMask::NSBorderlessWindowMask | NSWindowStyleMask::NSBorderlessWindowMask | NSWindowStyleMask::NSResizableWindowMask
NSWindowStyleMask::NSResizableWindowMask
} else { } else {
// default case, resizable window with titlebar and titlebar buttons // default case, resizable window with titlebar and titlebar buttons
NSWindowStyleMask::NSClosableWindowMask | NSWindowStyleMask::NSClosableWindowMask
NSWindowStyleMask::NSMiniaturizableWindowMask | | NSWindowStyleMask::NSMiniaturizableWindowMask
NSWindowStyleMask::NSResizableWindowMask | | NSWindowStyleMask::NSResizableWindowMask
NSWindowStyleMask::NSTitledWindowMask | NSWindowStyleMask::NSTitledWindowMask
}; };
if !attrs.resizable { if !attrs.resizable {
@ -171,7 +180,7 @@ fn create_window(
NSWindowButton::NSWindowZoomButton, NSWindowButton::NSWindowZoomButton,
] { ] {
let button = ns_window.standardWindowButton_(*titlebar_button); let button = ns_window.standardWindowButton_(*titlebar_button);
let _: () = msg_send![button, setHidden:YES]; let _: () = msg_send![button, setHidden: YES];
} }
} }
if pl_attrs.movable_by_window_background { if pl_attrs.movable_by_window_background {
@ -179,7 +188,10 @@ fn create_window(
} }
if attrs.always_on_top { if attrs.always_on_top {
let _: () = msg_send![*ns_window, setLevel:ffi::NSWindowLevel::NSFloatingWindowLevel]; let _: () = msg_send![
*ns_window,
setLevel: ffi::NSWindowLevel::NSFloatingWindowLevel
];
} }
if let Some(increments) = pl_attrs.resize_increments { if let Some(increments) = pl_attrs.resize_increments {
@ -206,8 +218,14 @@ lazy_static! {
static ref WINDOW_CLASS: WindowClass = unsafe { static ref WINDOW_CLASS: WindowClass = unsafe {
let window_superclass = class!(NSWindow); let window_superclass = class!(NSWindow);
let mut decl = ClassDecl::new("WinitWindow", window_superclass).unwrap(); let mut decl = ClassDecl::new("WinitWindow", window_superclass).unwrap();
decl.add_method(sel!(canBecomeMainWindow), util::yes as extern fn(&Object, Sel) -> BOOL); decl.add_method(
decl.add_method(sel!(canBecomeKeyWindow), util::yes as extern fn(&Object, Sel) -> BOOL); sel!(canBecomeMainWindow),
util::yes as extern "C" fn(&Object, Sel) -> BOOL,
);
decl.add_method(
sel!(canBecomeKeyWindow),
util::yes as extern "C" fn(&Object, Sel) -> BOOL,
);
WindowClass(decl.register()) WindowClass(decl.register())
}; };
} }
@ -242,14 +260,14 @@ impl From<WindowAttributes> for SharedState {
// identical, resulting in a no-op. // identical, resulting in a no-op.
fullscreen: None, fullscreen: None,
maximized: attribs.maximized, maximized: attribs.maximized,
.. Default::default() ..Default::default()
} }
} }
} }
pub struct UnownedWindow { pub struct UnownedWindow {
pub ns_window: IdRef, // never changes pub ns_window: IdRef, // never changes
pub ns_view: IdRef, // never changes pub ns_view: IdRef, // never changes
input_context: IdRef, // never changes input_context: IdRef, // never changes
pub shared_state: Arc<Mutex<SharedState>>, pub shared_state: Arc<Mutex<SharedState>>,
decorations: AtomicBool, decorations: AtomicBool,
@ -298,15 +316,20 @@ impl UnownedWindow {
ns_app.activateIgnoringOtherApps_(YES); ns_app.activateIgnoringOtherApps_(YES);
win_attribs.min_inner_size.map(|dim| set_min_inner_size(*ns_window, dim)); win_attribs
win_attribs.max_inner_size.map(|dim| set_max_inner_size(*ns_window, dim)); .min_inner_size
.map(|dim| set_min_inner_size(*ns_window, dim));
win_attribs
.max_inner_size
.map(|dim| set_max_inner_size(*ns_window, dim));
use cocoa::foundation::NSArray; use cocoa::foundation::NSArray;
// register for drag and drop operations. // register for drag and drop operations.
let () = msg_send![*ns_window, registerForDraggedTypes:NSArray::arrayWithObject( let () = msg_send![
nil, *ns_window,
appkit::NSFilenamesPboardType, registerForDraggedTypes:
)]; NSArray::arrayWithObject(nil, appkit::NSFilenamesPboardType)
];
} }
// Since `win_attribs` is put into a mutex below, we'll just copy these // Since `win_attribs` is put into a mutex below, we'll just copy these
@ -364,19 +387,11 @@ impl UnownedWindow {
} }
fn set_style_mask_async(&self, mask: NSWindowStyleMask) { fn set_style_mask_async(&self, mask: NSWindowStyleMask) {
unsafe { util::set_style_mask_async( unsafe { util::set_style_mask_async(*self.ns_window, *self.ns_view, mask) };
*self.ns_window,
*self.ns_view,
mask,
) };
} }
fn set_style_mask_sync(&self, mask: NSWindowStyleMask) { fn set_style_mask_sync(&self, mask: NSWindowStyleMask) {
unsafe { util::set_style_mask_sync( unsafe { util::set_style_mask_sync(*self.ns_window, *self.ns_view, mask) };
*self.ns_window,
*self.ns_view,
mask,
) };
} }
pub fn id(&self) -> Id { pub fn id(&self) -> Id {
@ -405,20 +420,19 @@ impl UnownedWindow {
Ok(( Ok((
frame_rect.origin.x as f64, frame_rect.origin.x as f64,
util::bottom_left_to_top_left(frame_rect), util::bottom_left_to_top_left(frame_rect),
).into()) )
.into())
} }
pub fn inner_position(&self) -> Result<LogicalPosition, NotSupportedError> { pub fn inner_position(&self) -> Result<LogicalPosition, NotSupportedError> {
let content_rect = unsafe { let content_rect = unsafe {
NSWindow::contentRectForFrameRect_( NSWindow::contentRectForFrameRect_(*self.ns_window, NSWindow::frame(*self.ns_window))
*self.ns_window,
NSWindow::frame(*self.ns_window),
)
}; };
Ok(( Ok((
content_rect.origin.x as f64, content_rect.origin.x as f64,
util::bottom_left_to_top_left(content_rect), util::bottom_left_to_top_left(content_rect),
).into()) )
.into())
} }
pub fn set_outer_position(&self, position: LogicalPosition) { pub fn set_outer_position(&self, position: LogicalPosition) {
@ -529,7 +543,10 @@ impl UnownedWindow {
} }
#[inline] #[inline]
pub fn set_cursor_position(&self, cursor_position: LogicalPosition) -> Result<(), ExternalError> { pub fn set_cursor_position(
&self,
cursor_position: LogicalPosition,
) -> Result<(), ExternalError> {
let window_position = self.inner_position().unwrap(); let window_position = self.inner_position().unwrap();
let point = appkit::CGPoint { let point = appkit::CGPoint {
x: (cursor_position.x + window_position.x) as CGFloat, x: (cursor_position.x + window_position.x) as CGFloat,
@ -548,8 +565,8 @@ impl UnownedWindow {
// we make it resizable temporalily. // we make it resizable temporalily.
let curr_mask = unsafe { self.ns_window.styleMask() }; let curr_mask = unsafe { self.ns_window.styleMask() };
let required = NSWindowStyleMask::NSTitledWindowMask let required =
| NSWindowStyleMask::NSResizableWindowMask; NSWindowStyleMask::NSTitledWindowMask | NSWindowStyleMask::NSResizableWindowMask;
let needs_temp_mask = !curr_mask.contains(required); let needs_temp_mask = !curr_mask.contains(required);
if needs_temp_mask { if needs_temp_mask {
self.set_style_mask_sync(required); self.set_style_mask_sync(required);
@ -566,7 +583,8 @@ impl UnownedWindow {
} }
fn saved_style(&self, shared_state: &mut SharedState) -> NSWindowStyleMask { fn saved_style(&self, shared_state: &mut SharedState) -> NSWindowStyleMask {
let base_mask = shared_state.saved_style let base_mask = shared_state
.saved_style
.take() .take()
.unwrap_or_else(|| unsafe { self.ns_window.styleMask() }); .unwrap_or_else(|| unsafe { self.ns_window.styleMask() });
if shared_state.resizable { if shared_state.resizable {
@ -620,7 +638,7 @@ impl UnownedWindow {
pub fn set_fullscreen(&self, monitor: Option<RootMonitorHandle>) { pub fn set_fullscreen(&self, monitor: Option<RootMonitorHandle>) {
let shared_state_lock = self.shared_state.lock().unwrap(); let shared_state_lock = self.shared_state.lock().unwrap();
if shared_state_lock.is_simple_fullscreen { if shared_state_lock.is_simple_fullscreen {
return return;
} }
let not_fullscreen = { let not_fullscreen = {
@ -639,12 +657,14 @@ impl UnownedWindow {
current.is_none() current.is_none()
}; };
unsafe { util::toggle_full_screen_async( unsafe {
*self.ns_window, util::toggle_full_screen_async(
*self.ns_view, *self.ns_window,
not_fullscreen, *self.ns_view,
Arc::downgrade(&self.shared_state), not_fullscreen,
) }; Arc::downgrade(&self.shared_state),
)
};
} }
#[inline] #[inline]
@ -664,7 +684,9 @@ impl UnownedWindow {
// If we're in fullscreen mode, we wait to apply decoration changes // If we're in fullscreen mode, we wait to apply decoration changes
// until we're in `window_did_exit_fullscreen`. // until we're in `window_did_exit_fullscreen`.
if fullscreen { return } if fullscreen {
return;
}
let new_mask = { let new_mask = {
let mut new_mask = if decorations { let mut new_mask = if decorations {
@ -727,7 +749,9 @@ impl UnownedWindow {
let key = IdRef::new(NSString::alloc(nil).init_str("NSScreenNumber")); let key = IdRef::new(NSString::alloc(nil).init_str("NSScreenNumber"));
let value = NSDictionary::valueForKey_(desc, *key); let value = NSDictionary::valueForKey_(desc, *key);
let display_id = msg_send![value, unsignedIntegerValue]; let display_id = msg_send![value, unsignedIntegerValue];
RootMonitorHandle { inner: MonitorHandle::new(display_id) } RootMonitorHandle {
inner: MonitorHandle::new(display_id),
}
} }
} }
@ -779,7 +803,10 @@ impl WindowExtMacOS for UnownedWindow {
let is_simple_fullscreen = shared_state_lock.is_simple_fullscreen; let is_simple_fullscreen = shared_state_lock.is_simple_fullscreen;
// Do nothing if native fullscreen is active. // Do nothing if native fullscreen is active.
if is_native_fullscreen || (fullscreen && is_simple_fullscreen) || (!fullscreen && !is_simple_fullscreen) { if is_native_fullscreen
|| (fullscreen && is_simple_fullscreen)
|| (!fullscreen && !is_simple_fullscreen)
{
return false; return false;
} }
@ -799,7 +826,12 @@ impl WindowExtMacOS for UnownedWindow {
app.setPresentationOptions_(presentation_options); app.setPresentationOptions_(presentation_options);
// Hide the titlebar // Hide the titlebar
util::toggle_style_mask(*self.ns_window, *self.ns_view, NSWindowStyleMask::NSTitledWindowMask, false); util::toggle_style_mask(
*self.ns_window,
*self.ns_view,
NSWindowStyleMask::NSTitledWindowMask,
false,
);
// Set the window frame to the screen frame size // Set the window frame to the screen frame size
let screen = self.ns_window.screen(); let screen = self.ns_window.screen();
@ -807,8 +839,18 @@ impl WindowExtMacOS for UnownedWindow {
NSWindow::setFrame_display_(*self.ns_window, screen_frame, YES); NSWindow::setFrame_display_(*self.ns_window, screen_frame, YES);
// Fullscreen windows can't be resized, minimized, or moved // Fullscreen windows can't be resized, minimized, or moved
util::toggle_style_mask(*self.ns_window, *self.ns_view, NSWindowStyleMask::NSMiniaturizableWindowMask, false); util::toggle_style_mask(
util::toggle_style_mask(*self.ns_window, *self.ns_view, NSWindowStyleMask::NSResizableWindowMask, false); *self.ns_window,
*self.ns_view,
NSWindowStyleMask::NSMiniaturizableWindowMask,
false,
);
util::toggle_style_mask(
*self.ns_window,
*self.ns_view,
NSWindowStyleMask::NSResizableWindowMask,
false,
);
NSWindow::setMovable_(*self.ns_window, NO); NSWindow::setMovable_(*self.ns_window, NO);
true true

View file

@ -1,20 +1,33 @@
use std::{f64, os::raw::c_void, sync::{Arc, Weak}}; use std::{
f64,
os::raw::c_void,
sync::{Arc, Weak},
};
use cocoa::{ use cocoa::{
appkit::{self, NSView, NSWindow}, base::{id, nil}, appkit::{self, NSView, NSWindow},
base::{id, nil},
foundation::NSAutoreleasePool, foundation::NSAutoreleasePool,
}; };
use objc::{runtime::{Class, Object, Sel, BOOL, YES, NO}, declare::ClassDecl}; use objc::{
declare::ClassDecl,
runtime::{Class, Object, Sel, BOOL, NO, YES},
};
use crate::{dpi::LogicalSize, event::{Event, WindowEvent}, window::WindowId}; use crate::{
use crate::platform_impl::platform::{ dpi::LogicalSize,
app_state::AppState, util::{self, IdRef}, event::{Event, WindowEvent},
window::{get_window_id, UnownedWindow}, platform_impl::platform::{
app_state::AppState,
util::{self, IdRef},
window::{get_window_id, UnownedWindow},
},
window::WindowId,
}; };
pub struct WindowDelegateState { pub struct WindowDelegateState {
ns_window: IdRef, // never changes ns_window: IdRef, // never changes
ns_view: IdRef, // never changes ns_view: IdRef, // never changes
window: Weak<UnownedWindow>, window: Weak<UnownedWindow>,
@ -33,10 +46,7 @@ pub struct WindowDelegateState {
} }
impl WindowDelegateState { impl WindowDelegateState {
pub fn new( pub fn new(window: &Arc<UnownedWindow>, initial_fullscreen: bool) -> Self {
window: &Arc<UnownedWindow>,
initial_fullscreen: bool,
) -> Self {
let dpi_factor = window.hidpi_factor(); let dpi_factor = window.hidpi_factor();
let mut delegate_state = WindowDelegateState { let mut delegate_state = WindowDelegateState {
@ -57,11 +67,10 @@ impl WindowDelegateState {
} }
fn with_window<F, T>(&mut self, callback: F) -> Option<T> fn with_window<F, T>(&mut self, callback: F) -> Option<T>
where F: FnOnce(&UnownedWindow) -> T where
F: FnOnce(&UnownedWindow) -> T,
{ {
self.window self.window.upgrade().map(|ref window| callback(window))
.upgrade()
.map(|ref window| callback(window))
} }
pub fn emit_event(&mut self, event: WindowEvent) { pub fn emit_event(&mut self, event: WindowEvent) {
@ -100,7 +109,7 @@ pub fn new_delegate(window: &Arc<UnownedWindow>, initial_fullscreen: bool) -> Id
// This is free'd in `dealloc` // This is free'd in `dealloc`
let state_ptr = Box::into_raw(Box::new(state)) as *mut c_void; let state_ptr = Box::into_raw(Box::new(state)) as *mut c_void;
let delegate: id = msg_send![WINDOW_DELEGATE_CLASS.0, alloc]; let delegate: id = msg_send![WINDOW_DELEGATE_CLASS.0, alloc];
IdRef::new(msg_send![delegate, initWithWinit:state_ptr]) IdRef::new(msg_send![delegate, initWithWinit: state_ptr])
} }
} }
@ -113,83 +122,81 @@ lazy_static! {
let superclass = class!(NSResponder); let superclass = class!(NSResponder);
let mut decl = ClassDecl::new("WinitWindowDelegate", superclass).unwrap(); let mut decl = ClassDecl::new("WinitWindowDelegate", superclass).unwrap();
decl.add_method( decl.add_method(sel!(dealloc), dealloc as extern "C" fn(&Object, Sel));
sel!(dealloc),
dealloc as extern fn(&Object, Sel),
);
decl.add_method( decl.add_method(
sel!(initWithWinit:), sel!(initWithWinit:),
init_with_winit as extern fn(&Object, Sel, *mut c_void) -> id, init_with_winit as extern "C" fn(&Object, Sel, *mut c_void) -> id,
); );
decl.add_method( decl.add_method(
sel!(windowShouldClose:), sel!(windowShouldClose:),
window_should_close as extern fn(&Object, Sel, id) -> BOOL, window_should_close as extern "C" fn(&Object, Sel, id) -> BOOL,
); );
decl.add_method( decl.add_method(
sel!(windowWillClose:), sel!(windowWillClose:),
window_will_close as extern fn(&Object, Sel, id), window_will_close as extern "C" fn(&Object, Sel, id),
); );
decl.add_method( decl.add_method(
sel!(windowDidResize:), sel!(windowDidResize:),
window_did_resize as extern fn(&Object, Sel, id), window_did_resize as extern "C" fn(&Object, Sel, id),
); );
decl.add_method( decl.add_method(
sel!(windowDidMove:), sel!(windowDidMove:),
window_did_move as extern fn(&Object, Sel, id)); window_did_move as extern "C" fn(&Object, Sel, id),
);
decl.add_method( decl.add_method(
sel!(windowDidChangeScreen:), sel!(windowDidChangeScreen:),
window_did_change_screen as extern fn(&Object, Sel, id), window_did_change_screen as extern "C" fn(&Object, Sel, id),
); );
decl.add_method( decl.add_method(
sel!(windowDidChangeBackingProperties:), sel!(windowDidChangeBackingProperties:),
window_did_change_backing_properties as extern fn(&Object, Sel, id), window_did_change_backing_properties as extern "C" fn(&Object, Sel, id),
); );
decl.add_method( decl.add_method(
sel!(windowDidBecomeKey:), sel!(windowDidBecomeKey:),
window_did_become_key as extern fn(&Object, Sel, id), window_did_become_key as extern "C" fn(&Object, Sel, id),
); );
decl.add_method( decl.add_method(
sel!(windowDidResignKey:), sel!(windowDidResignKey:),
window_did_resign_key as extern fn(&Object, Sel, id), window_did_resign_key as extern "C" fn(&Object, Sel, id),
); );
decl.add_method( decl.add_method(
sel!(draggingEntered:), sel!(draggingEntered:),
dragging_entered as extern fn(&Object, Sel, id) -> BOOL, dragging_entered as extern "C" fn(&Object, Sel, id) -> BOOL,
); );
decl.add_method( decl.add_method(
sel!(prepareForDragOperation:), sel!(prepareForDragOperation:),
prepare_for_drag_operation as extern fn(&Object, Sel, id) -> BOOL, prepare_for_drag_operation as extern "C" fn(&Object, Sel, id) -> BOOL,
); );
decl.add_method( decl.add_method(
sel!(performDragOperation:), sel!(performDragOperation:),
perform_drag_operation as extern fn(&Object, Sel, id) -> BOOL, perform_drag_operation as extern "C" fn(&Object, Sel, id) -> BOOL,
); );
decl.add_method( decl.add_method(
sel!(concludeDragOperation:), sel!(concludeDragOperation:),
conclude_drag_operation as extern fn(&Object, Sel, id), conclude_drag_operation as extern "C" fn(&Object, Sel, id),
); );
decl.add_method( decl.add_method(
sel!(draggingExited:), sel!(draggingExited:),
dragging_exited as extern fn(&Object, Sel, id), dragging_exited as extern "C" fn(&Object, Sel, id),
); );
decl.add_method( decl.add_method(
sel!(windowDidEnterFullScreen:), sel!(windowDidEnterFullScreen:),
window_did_enter_fullscreen as extern fn(&Object, Sel, id), window_did_enter_fullscreen as extern "C" fn(&Object, Sel, id),
); );
decl.add_method( decl.add_method(
sel!(windowWillEnterFullScreen:), sel!(windowWillEnterFullScreen:),
window_will_enter_fullscreen as extern fn(&Object, Sel, id), window_will_enter_fullscreen as extern "C" fn(&Object, Sel, id),
); );
decl.add_method( decl.add_method(
sel!(windowDidExitFullScreen:), sel!(windowDidExitFullScreen:),
window_did_exit_fullscreen as extern fn(&Object, Sel, id), window_did_exit_fullscreen as extern "C" fn(&Object, Sel, id),
); );
decl.add_method( decl.add_method(
sel!(windowDidFailToEnterFullScreen:), sel!(windowDidFailToEnterFullScreen:),
window_did_fail_to_enter_fullscreen as extern fn(&Object, Sel, id), window_did_fail_to_enter_fullscreen as extern "C" fn(&Object, Sel, id),
); );
decl.add_ivar::<*mut c_void>("winitState"); decl.add_ivar::<*mut c_void>("winitState");
@ -207,47 +214,47 @@ fn with_state<F: FnOnce(&mut WindowDelegateState) -> T, T>(this: &Object, callba
callback(state_ptr); callback(state_ptr);
} }
extern fn dealloc(this: &Object, _sel: Sel) { extern "C" fn dealloc(this: &Object, _sel: Sel) {
with_state(this, |state| unsafe { with_state(this, |state| unsafe {
Box::from_raw(state as *mut WindowDelegateState); Box::from_raw(state as *mut WindowDelegateState);
}); });
} }
extern fn init_with_winit(this: &Object, _sel: Sel, state: *mut c_void) -> id { extern "C" fn init_with_winit(this: &Object, _sel: Sel, state: *mut c_void) -> id {
unsafe { unsafe {
let this: id = msg_send![this, init]; let this: id = msg_send![this, init];
if this != nil { if this != nil {
(*this).set_ivar("winitState", state); (*this).set_ivar("winitState", state);
with_state(&*this, |state| { with_state(&*this, |state| {
let () = msg_send![*state.ns_window, setDelegate:this]; let () = msg_send![*state.ns_window, setDelegate: this];
}); });
} }
this this
} }
} }
extern fn window_should_close(this: &Object, _: Sel, _: id) -> BOOL { extern "C" fn window_should_close(this: &Object, _: Sel, _: id) -> BOOL {
trace!("Triggered `windowShouldClose:`"); trace!("Triggered `windowShouldClose:`");
with_state(this, |state| state.emit_event(WindowEvent::CloseRequested)); with_state(this, |state| state.emit_event(WindowEvent::CloseRequested));
trace!("Completed `windowShouldClose:`"); trace!("Completed `windowShouldClose:`");
NO NO
} }
extern fn window_will_close(this: &Object, _: Sel, _: id) { extern "C" fn window_will_close(this: &Object, _: Sel, _: id) {
trace!("Triggered `windowWillClose:`"); trace!("Triggered `windowWillClose:`");
with_state(this, |state| unsafe { with_state(this, |state| unsafe {
// `setDelegate:` retains the previous value and then autoreleases it // `setDelegate:` retains the previous value and then autoreleases it
let pool = NSAutoreleasePool::new(nil); let pool = NSAutoreleasePool::new(nil);
// Since El Capitan, we need to be careful that delegate methods can't // Since El Capitan, we need to be careful that delegate methods can't
// be called after the window closes. // be called after the window closes.
let () = msg_send![*state.ns_window, setDelegate:nil]; let () = msg_send![*state.ns_window, setDelegate: nil];
pool.drain(); pool.drain();
state.emit_event(WindowEvent::Destroyed); state.emit_event(WindowEvent::Destroyed);
}); });
trace!("Completed `windowWillClose:`"); trace!("Completed `windowWillClose:`");
} }
extern fn window_did_resize(this: &Object, _: Sel, _: id) { extern "C" fn window_did_resize(this: &Object, _: Sel, _: id) {
trace!("Triggered `windowDidResize:`"); trace!("Triggered `windowDidResize:`");
with_state(this, |state| { with_state(this, |state| {
state.emit_resize_event(); state.emit_resize_event();
@ -257,7 +264,7 @@ extern fn window_did_resize(this: &Object, _: Sel, _: id) {
} }
// This won't be triggered if the move was part of a resize. // This won't be triggered if the move was part of a resize.
extern fn window_did_move(this: &Object, _: Sel, _: id) { extern "C" fn window_did_move(this: &Object, _: Sel, _: id) {
trace!("Triggered `windowDidMove:`"); trace!("Triggered `windowDidMove:`");
with_state(this, |state| { with_state(this, |state| {
state.emit_move_event(); state.emit_move_event();
@ -265,12 +272,10 @@ extern fn window_did_move(this: &Object, _: Sel, _: id) {
trace!("Completed `windowDidMove:`"); trace!("Completed `windowDidMove:`");
} }
extern fn window_did_change_screen(this: &Object, _: Sel, _: id) { extern "C" fn window_did_change_screen(this: &Object, _: Sel, _: id) {
trace!("Triggered `windowDidChangeScreen:`"); trace!("Triggered `windowDidChangeScreen:`");
with_state(this, |state| { with_state(this, |state| {
let dpi_factor = unsafe { let dpi_factor = unsafe { NSWindow::backingScaleFactor(*state.ns_window) } as f64;
NSWindow::backingScaleFactor(*state.ns_window)
} as f64;
if state.previous_dpi_factor != dpi_factor { if state.previous_dpi_factor != dpi_factor {
state.previous_dpi_factor = dpi_factor; state.previous_dpi_factor = dpi_factor;
state.emit_event(WindowEvent::HiDpiFactorChanged(dpi_factor)); state.emit_event(WindowEvent::HiDpiFactorChanged(dpi_factor));
@ -281,12 +286,10 @@ extern fn window_did_change_screen(this: &Object, _: Sel, _: id) {
} }
// This will always be called before `window_did_change_screen`. // This will always be called before `window_did_change_screen`.
extern fn window_did_change_backing_properties(this: &Object, _:Sel, _:id) { extern "C" fn window_did_change_backing_properties(this: &Object, _: Sel, _: id) {
trace!("Triggered `windowDidChangeBackingProperties:`"); trace!("Triggered `windowDidChangeBackingProperties:`");
with_state(this, |state| { with_state(this, |state| {
let dpi_factor = unsafe { let dpi_factor = unsafe { NSWindow::backingScaleFactor(*state.ns_window) } as f64;
NSWindow::backingScaleFactor(*state.ns_window)
} as f64;
if state.previous_dpi_factor != dpi_factor { if state.previous_dpi_factor != dpi_factor {
state.previous_dpi_factor = dpi_factor; state.previous_dpi_factor = dpi_factor;
state.emit_event(WindowEvent::HiDpiFactorChanged(dpi_factor)); state.emit_event(WindowEvent::HiDpiFactorChanged(dpi_factor));
@ -296,7 +299,7 @@ extern fn window_did_change_backing_properties(this: &Object, _:Sel, _:id) {
trace!("Completed `windowDidChangeBackingProperties:`"); trace!("Completed `windowDidChangeBackingProperties:`");
} }
extern fn window_did_become_key(this: &Object, _: Sel, _: id) { extern "C" fn window_did_become_key(this: &Object, _: Sel, _: id) {
trace!("Triggered `windowDidBecomeKey:`"); trace!("Triggered `windowDidBecomeKey:`");
with_state(this, |state| { with_state(this, |state| {
// TODO: center the cursor if the window had mouse grab when it // TODO: center the cursor if the window had mouse grab when it
@ -306,7 +309,7 @@ extern fn window_did_become_key(this: &Object, _: Sel, _: id) {
trace!("Completed `windowDidBecomeKey:`"); trace!("Completed `windowDidBecomeKey:`");
} }
extern fn window_did_resign_key(this: &Object, _: Sel, _: id) { extern "C" fn window_did_resign_key(this: &Object, _: Sel, _: id) {
trace!("Triggered `windowDidResignKey:`"); trace!("Triggered `windowDidResignKey:`");
with_state(this, |state| { with_state(this, |state| {
state.emit_event(WindowEvent::Focused(false)); state.emit_event(WindowEvent::Focused(false));
@ -315,11 +318,10 @@ extern fn window_did_resign_key(this: &Object, _: Sel, _: id) {
} }
/// Invoked when the dragged image enters destination bounds or frame /// Invoked when the dragged image enters destination bounds or frame
extern fn dragging_entered(this: &Object, _: Sel, sender: id) -> BOOL { extern "C" fn dragging_entered(this: &Object, _: Sel, sender: id) -> BOOL {
trace!("Triggered `draggingEntered:`"); trace!("Triggered `draggingEntered:`");
use cocoa::appkit::NSPasteboard; use cocoa::{appkit::NSPasteboard, foundation::NSFastEnumeration};
use cocoa::foundation::NSFastEnumeration;
use std::path::PathBuf; use std::path::PathBuf;
let pb: id = unsafe { msg_send![sender, draggingPasteboard] }; let pb: id = unsafe { msg_send![sender, draggingPasteboard] };
@ -337,25 +339,24 @@ extern fn dragging_entered(this: &Object, _: Sel, sender: id) -> BOOL {
state.emit_event(WindowEvent::HoveredFile(PathBuf::from(path))); state.emit_event(WindowEvent::HoveredFile(PathBuf::from(path)));
}); });
} }
}; }
trace!("Completed `draggingEntered:`"); trace!("Completed `draggingEntered:`");
YES YES
} }
/// Invoked when the image is released /// Invoked when the image is released
extern fn prepare_for_drag_operation(_: &Object, _: Sel, _: id) -> BOOL { extern "C" fn prepare_for_drag_operation(_: &Object, _: Sel, _: id) -> BOOL {
trace!("Triggered `prepareForDragOperation:`"); trace!("Triggered `prepareForDragOperation:`");
trace!("Completed `prepareForDragOperation:`"); trace!("Completed `prepareForDragOperation:`");
YES YES
} }
/// Invoked after the released image has been removed from the screen /// Invoked after the released image has been removed from the screen
extern fn perform_drag_operation(this: &Object, _: Sel, sender: id) -> BOOL { extern "C" fn perform_drag_operation(this: &Object, _: Sel, sender: id) -> BOOL {
trace!("Triggered `performDragOperation:`"); trace!("Triggered `performDragOperation:`");
use cocoa::appkit::NSPasteboard; use cocoa::{appkit::NSPasteboard, foundation::NSFastEnumeration};
use cocoa::foundation::NSFastEnumeration;
use std::path::PathBuf; use std::path::PathBuf;
let pb: id = unsafe { msg_send![sender, draggingPasteboard] }; let pb: id = unsafe { msg_send![sender, draggingPasteboard] };
@ -373,38 +374,42 @@ extern fn perform_drag_operation(this: &Object, _: Sel, sender: id) -> BOOL {
state.emit_event(WindowEvent::DroppedFile(PathBuf::from(path))); state.emit_event(WindowEvent::DroppedFile(PathBuf::from(path)));
}); });
} }
}; }
trace!("Completed `performDragOperation:`"); trace!("Completed `performDragOperation:`");
YES YES
} }
/// Invoked when the dragging operation is complete /// Invoked when the dragging operation is complete
extern fn conclude_drag_operation(_: &Object, _: Sel, _: id) { extern "C" fn conclude_drag_operation(_: &Object, _: Sel, _: id) {
trace!("Triggered `concludeDragOperation:`"); trace!("Triggered `concludeDragOperation:`");
trace!("Completed `concludeDragOperation:`"); trace!("Completed `concludeDragOperation:`");
} }
/// Invoked when the dragging operation is cancelled /// Invoked when the dragging operation is cancelled
extern fn dragging_exited(this: &Object, _: Sel, _: id) { extern "C" fn dragging_exited(this: &Object, _: Sel, _: id) {
trace!("Triggered `draggingExited:`"); trace!("Triggered `draggingExited:`");
with_state(this, |state| state.emit_event(WindowEvent::HoveredFileCancelled)); with_state(this, |state| {
state.emit_event(WindowEvent::HoveredFileCancelled)
});
trace!("Completed `draggingExited:`"); trace!("Completed `draggingExited:`");
} }
/// Invoked when before enter fullscreen /// Invoked when before enter fullscreen
extern fn window_will_enter_fullscreen(this: &Object, _: Sel, _: id) { extern "C" fn window_will_enter_fullscreen(this: &Object, _: Sel, _: id) {
trace!("Triggered `windowWillEnterFullscreen:`"); trace!("Triggered `windowWillEnterFullscreen:`");
with_state(this, |state| state.with_window(|window| { with_state(this, |state| {
trace!("Locked shared state in `window_will_enter_fullscreen`"); state.with_window(|window| {
window.shared_state.lock().unwrap().maximized = window.is_zoomed(); trace!("Locked shared state in `window_will_enter_fullscreen`");
trace!("Unlocked shared state in `window_will_enter_fullscreen`"); window.shared_state.lock().unwrap().maximized = window.is_zoomed();
})); trace!("Unlocked shared state in `window_will_enter_fullscreen`");
})
});
trace!("Completed `windowWillEnterFullscreen:`"); trace!("Completed `windowWillEnterFullscreen:`");
} }
/// Invoked when entered fullscreen /// Invoked when entered fullscreen
extern fn window_did_enter_fullscreen(this: &Object, _: Sel, _: id) { extern "C" fn window_did_enter_fullscreen(this: &Object, _: Sel, _: id) {
trace!("Triggered `windowDidEnterFullscreen:`"); trace!("Triggered `windowDidEnterFullscreen:`");
with_state(this, |state| { with_state(this, |state| {
state.with_window(|window| { state.with_window(|window| {
@ -419,11 +424,13 @@ extern fn window_did_enter_fullscreen(this: &Object, _: Sel, _: id) {
} }
/// Invoked when exited fullscreen /// Invoked when exited fullscreen
extern fn window_did_exit_fullscreen(this: &Object, _: Sel, _: id) { extern "C" fn window_did_exit_fullscreen(this: &Object, _: Sel, _: id) {
trace!("Triggered `windowDidExitFullscreen:`"); trace!("Triggered `windowDidExitFullscreen:`");
with_state(this, |state| state.with_window(|window| { with_state(this, |state| {
window.restore_state_from_fullscreen(); state.with_window(|window| {
})); window.restore_state_from_fullscreen();
})
});
trace!("Completed `windowDidExitFullscreen:`"); trace!("Completed `windowDidExitFullscreen:`");
} }
@ -443,15 +450,17 @@ extern fn window_did_exit_fullscreen(this: &Object, _: Sel, _: id) {
/// due to being in the midst of handling some other animation or user gesture. /// due to being in the midst of handling some other animation or user gesture.
/// This method indicates that there was an error, and you should clean up any /// This method indicates that there was an error, and you should clean up any
/// work you may have done to prepare to enter full-screen mode. /// work you may have done to prepare to enter full-screen mode.
extern fn window_did_fail_to_enter_fullscreen(this: &Object, _: Sel, _: id) { extern "C" fn window_did_fail_to_enter_fullscreen(this: &Object, _: Sel, _: id) {
trace!("Triggered `windowDidFailToEnterFullscreen:`"); trace!("Triggered `windowDidFailToEnterFullscreen:`");
with_state(this, |state| { with_state(this, |state| {
if state.initial_fullscreen { if state.initial_fullscreen {
let _: () = unsafe { msg_send![*state.ns_window, let _: () = unsafe {
performSelector:sel!(toggleFullScreen:) msg_send![*state.ns_window,
withObject:nil performSelector:sel!(toggleFullScreen:)
afterDelay: 0.5 withObject:nil
] }; afterDelay: 0.5
]
};
} else { } else {
state.with_window(|window| window.restore_state_from_fullscreen()); state.with_window(|window| window.restore_state_from_fullscreen());
} }

View file

@ -1,26 +1,40 @@
pub use self::platform::*; pub use self::platform::*;
#[cfg(target_os = "windows")] #[cfg(target_os = "windows")]
#[path="windows/mod.rs"] #[path = "windows/mod.rs"]
mod platform; mod platform;
#[cfg(any(target_os = "linux", target_os = "dragonfly", target_os = "freebsd", target_os = "netbsd", target_os = "openbsd"))] #[cfg(any(
#[path="linux/mod.rs"] target_os = "linux",
target_os = "dragonfly",
target_os = "freebsd",
target_os = "netbsd",
target_os = "openbsd"
))]
#[path = "linux/mod.rs"]
mod platform; mod platform;
#[cfg(target_os = "macos")] #[cfg(target_os = "macos")]
#[path="macos/mod.rs"] #[path = "macos/mod.rs"]
mod platform; mod platform;
#[cfg(target_os = "android")] #[cfg(target_os = "android")]
#[path="android/mod.rs"] #[path = "android/mod.rs"]
mod platform; mod platform;
#[cfg(target_os = "ios")] #[cfg(target_os = "ios")]
#[path="ios/mod.rs"] #[path = "ios/mod.rs"]
mod platform; mod platform;
#[cfg(target_os = "emscripten")] #[cfg(target_os = "emscripten")]
#[path="emscripten/mod.rs"] #[path = "emscripten/mod.rs"]
mod platform; mod platform;
#[cfg(all(not(target_os = "ios"), not(target_os = "windows"), not(target_os = "linux"), #[cfg(all(
not(target_os = "macos"), not(target_os = "android"), not(target_os = "dragonfly"), not(target_os = "ios"),
not(target_os = "freebsd"), not(target_os = "netbsd"), not(target_os = "openbsd"), not(target_os = "windows"),
not(target_os = "emscripten")))] not(target_os = "linux"),
not(target_os = "macos"),
not(target_os = "android"),
not(target_os = "dragonfly"),
not(target_os = "freebsd"),
not(target_os = "netbsd"),
not(target_os = "openbsd"),
not(target_os = "emscripten")
))]
compile_error!("The platform you're compiling for is not supported by winit"); compile_error!("The platform you're compiling for is not supported by winit");

View file

@ -1,45 +1,43 @@
#![allow(non_snake_case, unused_unsafe)] #![allow(non_snake_case, unused_unsafe)]
use std::mem; use std::{
use std::os::raw::c_void; mem,
use std::sync::{Once, ONCE_INIT}; os::raw::c_void,
sync::{Once, ONCE_INIT},
};
use winapi::shared::minwindef::{BOOL, UINT, FALSE}; use winapi::{
use winapi::shared::windef::{ shared::{
DPI_AWARENESS_CONTEXT, minwindef::{BOOL, FALSE, UINT},
DPI_AWARENESS_CONTEXT_PER_MONITOR_AWARE, windef::{DPI_AWARENESS_CONTEXT, DPI_AWARENESS_CONTEXT_PER_MONITOR_AWARE, HMONITOR, HWND},
HMONITOR, winerror::S_OK,
HWND, },
um::{
libloaderapi::{GetProcAddress, LoadLibraryA},
shellscalingapi::{
MDT_EFFECTIVE_DPI, MONITOR_DPI_TYPE, PROCESS_DPI_AWARENESS,
PROCESS_PER_MONITOR_DPI_AWARE,
},
wingdi::{GetDeviceCaps, LOGPIXELSX},
winnt::{HRESULT, LPCSTR},
winuser::{self, MONITOR_DEFAULTTONEAREST},
},
}; };
use winapi::shared::winerror::S_OK;
use winapi::um::libloaderapi::{GetProcAddress, LoadLibraryA};
use winapi::um::shellscalingapi::{
MDT_EFFECTIVE_DPI,
MONITOR_DPI_TYPE,
PROCESS_DPI_AWARENESS,
PROCESS_PER_MONITOR_DPI_AWARE,
};
use winapi::um::wingdi::{GetDeviceCaps, LOGPIXELSX};
use winapi::um::winnt::{HRESULT, LPCSTR};
use winapi::um::winuser::{self, MONITOR_DEFAULTTONEAREST};
const DPI_AWARENESS_CONTEXT_PER_MONITOR_AWARE_V2: DPI_AWARENESS_CONTEXT = -4isize as _; const DPI_AWARENESS_CONTEXT_PER_MONITOR_AWARE_V2: DPI_AWARENESS_CONTEXT = -4isize as _;
type SetProcessDPIAware = unsafe extern "system" fn () -> BOOL; type SetProcessDPIAware = unsafe extern "system" fn() -> BOOL;
type SetProcessDpiAwareness = unsafe extern "system" fn ( type SetProcessDpiAwareness = unsafe extern "system" fn(value: PROCESS_DPI_AWARENESS) -> HRESULT;
value: PROCESS_DPI_AWARENESS, type SetProcessDpiAwarenessContext =
) -> HRESULT; unsafe extern "system" fn(value: DPI_AWARENESS_CONTEXT) -> BOOL;
type SetProcessDpiAwarenessContext = unsafe extern "system" fn ( type GetDpiForWindow = unsafe extern "system" fn(hwnd: HWND) -> UINT;
value: DPI_AWARENESS_CONTEXT, type GetDpiForMonitor = unsafe extern "system" fn(
) -> BOOL;
type GetDpiForWindow = unsafe extern "system" fn (hwnd: HWND) -> UINT;
type GetDpiForMonitor = unsafe extern "system" fn (
hmonitor: HMONITOR, hmonitor: HMONITOR,
dpi_type: MONITOR_DPI_TYPE, dpi_type: MONITOR_DPI_TYPE,
dpi_x: *mut UINT, dpi_x: *mut UINT,
dpi_y: *mut UINT, dpi_y: *mut UINT,
) -> HRESULT; ) -> HRESULT;
type EnableNonClientDpiScaling = unsafe extern "system" fn (hwnd: HWND) -> BOOL; type EnableNonClientDpiScaling = unsafe extern "system" fn(hwnd: HWND) -> BOOL;
// Helper function to dynamically load function pointer. // Helper function to dynamically load function pointer.
// `library` and `function` must be zero-terminated. // `library` and `function` must be zero-terminated.
@ -65,53 +63,48 @@ macro_rules! get_function {
($lib:expr, $func:ident) => { ($lib:expr, $func:ident) => {
get_function_impl(concat!($lib, '\0'), concat!(stringify!($func), '\0')) get_function_impl(concat!($lib, '\0'), concat!(stringify!($func), '\0'))
.map(|f| unsafe { mem::transmute::<*const _, $func>(f) }) .map(|f| unsafe { mem::transmute::<*const _, $func>(f) })
} };
} }
lazy_static! { lazy_static! {
static ref GET_DPI_FOR_WINDOW: Option<GetDpiForWindow> = get_function!( static ref GET_DPI_FOR_WINDOW: Option<GetDpiForWindow> =
"user32.dll", get_function!("user32.dll", GetDpiForWindow);
GetDpiForWindow static ref GET_DPI_FOR_MONITOR: Option<GetDpiForMonitor> =
); get_function!("shcore.dll", GetDpiForMonitor);
static ref GET_DPI_FOR_MONITOR: Option<GetDpiForMonitor> = get_function!( static ref ENABLE_NON_CLIENT_DPI_SCALING: Option<EnableNonClientDpiScaling> =
"shcore.dll", get_function!("user32.dll", EnableNonClientDpiScaling);
GetDpiForMonitor
);
static ref ENABLE_NON_CLIENT_DPI_SCALING: Option<EnableNonClientDpiScaling> = get_function!(
"user32.dll",
EnableNonClientDpiScaling
);
} }
pub fn become_dpi_aware(enable: bool) { pub fn become_dpi_aware(enable: bool) {
if !enable { return; } if !enable {
return;
}
static ENABLE_DPI_AWARENESS: Once = ONCE_INIT; static ENABLE_DPI_AWARENESS: Once = ONCE_INIT;
ENABLE_DPI_AWARENESS.call_once(|| { unsafe { ENABLE_DPI_AWARENESS.call_once(|| {
if let Some(SetProcessDpiAwarenessContext) = get_function!( unsafe {
"user32.dll", if let Some(SetProcessDpiAwarenessContext) =
SetProcessDpiAwarenessContext get_function!("user32.dll", SetProcessDpiAwarenessContext)
) { {
// We are on Windows 10 Anniversary Update (1607) or later. // We are on Windows 10 Anniversary Update (1607) or later.
if SetProcessDpiAwarenessContext(DPI_AWARENESS_CONTEXT_PER_MONITOR_AWARE_V2) if SetProcessDpiAwarenessContext(DPI_AWARENESS_CONTEXT_PER_MONITOR_AWARE_V2)
== FALSE { == FALSE
// V2 only works with Windows 10 Creators Update (1703). Try using the older {
// V1 if we can't set V2. // V2 only works with Windows 10 Creators Update (1703). Try using the older
SetProcessDpiAwarenessContext(DPI_AWARENESS_CONTEXT_PER_MONITOR_AWARE); // V1 if we can't set V2.
SetProcessDpiAwarenessContext(DPI_AWARENESS_CONTEXT_PER_MONITOR_AWARE);
}
} else if let Some(SetProcessDpiAwareness) =
get_function!("shcore.dll", SetProcessDpiAwareness)
{
// We are on Windows 8.1 or later.
SetProcessDpiAwareness(PROCESS_PER_MONITOR_DPI_AWARE);
} else if let Some(SetProcessDPIAware) = get_function!("user32.dll", SetProcessDPIAware)
{
// We are on Vista or later.
SetProcessDPIAware();
} }
} else if let Some(SetProcessDpiAwareness) = get_function!(
"shcore.dll",
SetProcessDpiAwareness
) {
// We are on Windows 8.1 or later.
SetProcessDpiAwareness(PROCESS_PER_MONITOR_DPI_AWARE);
} else if let Some(SetProcessDPIAware) = get_function!(
"user32.dll",
SetProcessDPIAware
) {
// We are on Vista or later.
SetProcessDPIAware();
} }
} }); });
} }
pub fn enable_non_client_dpi_scaling(hwnd: HWND) { pub fn enable_non_client_dpi_scaling(hwnd: HWND) {
@ -132,7 +125,7 @@ pub fn get_monitor_dpi(hmonitor: HMONITOR) -> Option<u32> {
// MSDN says that "the values of *dpiX and *dpiY are identical. You only need to // MSDN says that "the values of *dpiX and *dpiY are identical. You only need to
// record one of the values to determine the DPI and respond appropriately". // record one of the values to determine the DPI and respond appropriately".
// https://msdn.microsoft.com/en-us/library/windows/desktop/dn280510(v=vs.85).aspx // https://msdn.microsoft.com/en-us/library/windows/desktop/dn280510(v=vs.85).aspx
return Some(dpi_x as u32) return Some(dpi_x as u32);
} }
} }
} }

View file

@ -1,23 +1,31 @@
use std::ffi::OsString; use std::{
use std::os::windows::ffi::OsStringExt; ffi::OsString,
use std::path::PathBuf; mem,
use std::sync::atomic::{AtomicUsize, Ordering}; os::windows::ffi::OsStringExt,
use std::{mem, ptr}; path::PathBuf,
ptr,
sync::atomic::{AtomicUsize, Ordering},
};
use winapi::ctypes::c_void; use winapi::{
use winapi::shared::guiddef::REFIID; ctypes::c_void,
use winapi::shared::minwindef::{DWORD, MAX_PATH, UINT, ULONG}; shared::{
use winapi::shared::windef::{HWND, POINTL}; guiddef::REFIID,
use winapi::shared::winerror::S_OK; minwindef::{DWORD, MAX_PATH, UINT, ULONG},
use winapi::um::objidl::IDataObject; windef::{HWND, POINTL},
use winapi::um::oleidl::{DROPEFFECT_COPY, DROPEFFECT_NONE, IDropTarget, IDropTargetVtbl}; winerror::S_OK,
use winapi::um::winnt::HRESULT; },
use winapi::um::{shellapi, unknwnbase}; um::{
objidl::IDataObject,
oleidl::{IDropTarget, IDropTargetVtbl, DROPEFFECT_COPY, DROPEFFECT_NONE},
shellapi, unknwnbase,
winnt::HRESULT,
},
};
use crate::platform_impl::platform::WindowId; use crate::platform_impl::platform::WindowId;
use crate::event::Event; use crate::{event::Event, window::WindowId as SuperWindowId};
use crate::window::WindowId as SuperWindowId;
#[repr(C)] #[repr(C)]
pub struct FileDropHandlerData { pub struct FileDropHandlerData {
@ -26,7 +34,7 @@ pub struct FileDropHandlerData {
window: HWND, window: HWND,
send_event: Box<dyn Fn(Event<()>)>, send_event: Box<dyn Fn(Event<()>)>,
cursor_effect: DWORD, cursor_effect: DWORD,
hovered_is_valid: bool, // If the currently hovered item is not valid there must not be any `HoveredFileCancelled` emitted hovered_is_valid: bool, /* If the currently hovered item is not valid there must not be any `HoveredFileCancelled` emitted */
} }
pub struct FileDropHandler { pub struct FileDropHandler {
@ -155,16 +163,25 @@ impl FileDropHandler {
&mut *(this as *mut _) &mut *(this as *mut _)
} }
unsafe fn iterate_filenames<F>(data_obj: *const IDataObject, callback: F) -> Option<shellapi::HDROP> unsafe fn iterate_filenames<F>(
data_obj: *const IDataObject,
callback: F,
) -> Option<shellapi::HDROP>
where where
F: Fn(PathBuf), F: Fn(PathBuf),
{ {
use winapi::ctypes::wchar_t; use winapi::{
use winapi::shared::winerror::{SUCCEEDED, DV_E_FORMATETC}; ctypes::wchar_t,
use winapi::shared::wtypes::{CLIPFORMAT, DVASPECT_CONTENT}; shared::{
use winapi::um::objidl::{FORMATETC, TYMED_HGLOBAL}; winerror::{DV_E_FORMATETC, SUCCEEDED},
use winapi::um::shellapi::DragQueryFileW; wtypes::{CLIPFORMAT, DVASPECT_CONTENT},
use winapi::um::winuser::CF_HDROP; },
um::{
objidl::{FORMATETC, TYMED_HGLOBAL},
shellapi::DragQueryFileW,
winuser::CF_HDROP,
},
};
let mut drop_format = FORMATETC { let mut drop_format = FORMATETC {
cfFormat: CF_HDROP as CLIPFORMAT, cfFormat: CF_HDROP as CLIPFORMAT,

View file

@ -1,16 +1,19 @@
use std::{char, ptr}; use std::{
use std::os::raw::c_int; char,
use std::sync::atomic::{AtomicBool, AtomicPtr, Ordering}; os::raw::c_int,
ptr,
sync::atomic::{AtomicBool, AtomicPtr, Ordering},
};
use crate::event::{ScanCode, ModifiersState, VirtualKeyCode}; use crate::event::{ModifiersState, ScanCode, VirtualKeyCode};
use winapi::shared::minwindef::{WPARAM, LPARAM, UINT, HKL, HKL__}; use winapi::{
use winapi::um::winuser; shared::minwindef::{HKL, HKL__, LPARAM, UINT, WPARAM},
um::winuser,
};
fn key_pressed(vkey: c_int) -> bool { fn key_pressed(vkey: c_int) -> bool {
unsafe { unsafe { (winuser::GetKeyState(vkey) & (1 << 15)) == (1 << 15) }
(winuser::GetKeyState(vkey) & (1 << 15)) == (1 << 15)
}
} }
pub fn get_key_mods() -> ModifiersState { pub fn get_key_mods() -> ModifiersState {
@ -26,9 +29,19 @@ pub fn get_key_mods() -> ModifiersState {
unsafe fn get_char(keyboard_state: &[u8; 256], v_key: u32, hkl: HKL) -> Option<char> { unsafe fn get_char(keyboard_state: &[u8; 256], v_key: u32, hkl: HKL) -> Option<char> {
let mut unicode_bytes = [0u16; 5]; let mut unicode_bytes = [0u16; 5];
let len = winuser::ToUnicodeEx(v_key, 0, keyboard_state.as_ptr(), unicode_bytes.as_mut_ptr(), unicode_bytes.len() as _, 0, hkl); let len = winuser::ToUnicodeEx(
v_key,
0,
keyboard_state.as_ptr(),
unicode_bytes.as_mut_ptr(),
unicode_bytes.len() as _,
0,
hkl,
);
if len >= 1 { if len >= 1 {
char::decode_utf16(unicode_bytes.into_iter().cloned()).next().and_then(|c| c.ok()) char::decode_utf16(unicode_bytes.into_iter().cloned())
.next()
.and_then(|c| c.ok())
} else { } else {
None None
} }
@ -239,7 +252,7 @@ pub fn vkey_to_winit_vkey(vkey: c_int) -> Option<VirtualKeyCode> {
winuser::VK_OEM_5 => map_text_keys(vkey), winuser::VK_OEM_5 => map_text_keys(vkey),
winuser::VK_OEM_6 => map_text_keys(vkey), winuser::VK_OEM_6 => map_text_keys(vkey),
winuser::VK_OEM_7 => map_text_keys(vkey), winuser::VK_OEM_7 => map_text_keys(vkey),
/*winuser::VK_OEM_8 => Some(VirtualKeyCode::Oem_8), */ /* winuser::VK_OEM_8 => Some(VirtualKeyCode::Oem_8), */
winuser::VK_OEM_102 => Some(VirtualKeyCode::OEM102), winuser::VK_OEM_102 => Some(VirtualKeyCode::OEM102),
/*winuser::VK_PROCESSKEY => Some(VirtualKeyCode::Processkey), /*winuser::VK_PROCESSKEY => Some(VirtualKeyCode::Processkey),
winuser::VK_PACKET => Some(VirtualKeyCode::Packet), winuser::VK_PACKET => Some(VirtualKeyCode::Packet),
@ -252,49 +265,61 @@ pub fn vkey_to_winit_vkey(vkey: c_int) -> Option<VirtualKeyCode> {
winuser::VK_NONAME => Some(VirtualKeyCode::Noname), winuser::VK_NONAME => Some(VirtualKeyCode::Noname),
winuser::VK_PA1 => Some(VirtualKeyCode::Pa1), winuser::VK_PA1 => Some(VirtualKeyCode::Pa1),
winuser::VK_OEM_CLEAR => Some(VirtualKeyCode::Oem_clear),*/ winuser::VK_OEM_CLEAR => Some(VirtualKeyCode::Oem_clear),*/
_ => None _ => None,
} }
} }
pub fn handle_extended_keys(vkey: c_int, mut scancode: UINT, extended: bool) -> Option<(c_int, UINT)> { pub fn handle_extended_keys(
vkey: c_int,
mut scancode: UINT,
extended: bool,
) -> Option<(c_int, UINT)> {
// Welcome to hell https://blog.molecular-matters.com/2011/09/05/properly-handling-keyboard-input/ // Welcome to hell https://blog.molecular-matters.com/2011/09/05/properly-handling-keyboard-input/
let vkey = match vkey { let vkey = match vkey {
winuser::VK_SHIFT => unsafe { winuser::MapVirtualKeyA( winuser::VK_SHIFT => unsafe {
scancode, winuser::MapVirtualKeyA(scancode, winuser::MAPVK_VSC_TO_VK_EX) as _
winuser::MAPVK_VSC_TO_VK_EX,
) as _ },
winuser::VK_CONTROL => if extended {
winuser::VK_RCONTROL
} else {
winuser::VK_LCONTROL
}, },
winuser::VK_MENU => if extended { winuser::VK_CONTROL => {
winuser::VK_RMENU if extended {
} else { winuser::VK_RCONTROL
winuser::VK_LMENU } else {
winuser::VK_LCONTROL
}
}, },
_ => match scancode { winuser::VK_MENU => {
// This is only triggered when using raw input. Without this check, we get two events whenever VK_PAUSE is if extended {
// pressed, the first one having scancode 0x1D but vkey VK_PAUSE... winuser::VK_RMENU
0x1D if vkey == winuser::VK_PAUSE => return None, } else {
// ...and the second having scancode 0x45 but an unmatched vkey! winuser::VK_LMENU
0x45 => winuser::VK_PAUSE, }
// VK_PAUSE and VK_SCROLL have the same scancode when using modifiers, alongside incorrect vkey values. },
0x46 => { _ => {
if extended { match scancode {
scancode = 0x45; // This is only triggered when using raw input. Without this check, we get two events whenever VK_PAUSE is
winuser::VK_PAUSE // pressed, the first one having scancode 0x1D but vkey VK_PAUSE...
} else { 0x1D if vkey == winuser::VK_PAUSE => return None,
winuser::VK_SCROLL // ...and the second having scancode 0x45 but an unmatched vkey!
} 0x45 => winuser::VK_PAUSE,
}, // VK_PAUSE and VK_SCROLL have the same scancode when using modifiers, alongside incorrect vkey values.
_ => vkey, 0x46 => {
if extended {
scancode = 0x45;
winuser::VK_PAUSE
} else {
winuser::VK_SCROLL
}
},
_ => vkey,
}
}, },
}; };
Some((vkey, scancode)) Some((vkey, scancode))
} }
pub fn process_key_params(wparam: WPARAM, lparam: LPARAM) -> Option<(ScanCode, Option<VirtualKeyCode>)> { pub fn process_key_params(
wparam: WPARAM,
lparam: LPARAM,
) -> Option<(ScanCode, Option<VirtualKeyCode>)> {
let scancode = ((lparam >> 16) & 0xff) as UINT; let scancode = ((lparam >> 16) & 0xff) as UINT;
let extended = (lparam & 0x01000000) != 0; let extended = (lparam & 0x01000000) != 0;
handle_extended_keys(wparam as _, scancode, extended) handle_extended_keys(wparam as _, scancode, extended)
@ -304,7 +329,9 @@ pub fn process_key_params(wparam: WPARAM, lparam: LPARAM) -> Option<(ScanCode, O
// This is needed as windows doesn't properly distinguish // This is needed as windows doesn't properly distinguish
// some virtual key codes for different keyboard layouts // some virtual key codes for different keyboard layouts
fn map_text_keys(win_virtual_key: i32) -> Option<VirtualKeyCode> { fn map_text_keys(win_virtual_key: i32) -> Option<VirtualKeyCode> {
let char_key = unsafe { winuser::MapVirtualKeyA(win_virtual_key as u32, winuser::MAPVK_VK_TO_CHAR) } & 0x7FFF; let char_key =
unsafe { winuser::MapVirtualKeyA(win_virtual_key as u32, winuser::MAPVK_VK_TO_CHAR) }
& 0x7FFF;
match char::from_u32(char_key) { match char::from_u32(char_key) {
Some(';') => Some(VirtualKeyCode::Semicolon), Some(';') => Some(VirtualKeyCode::Semicolon),
Some('/') => Some(VirtualKeyCode::Slash), Some('/') => Some(VirtualKeyCode::Slash),
@ -313,6 +340,6 @@ fn map_text_keys(win_virtual_key: i32) -> Option<VirtualKeyCode> {
Some(']') => Some(VirtualKeyCode::RBracket), Some(']') => Some(VirtualKeyCode::RBracket),
Some('\'') => Some(VirtualKeyCode::Apostrophe), Some('\'') => Some(VirtualKeyCode::Apostrophe),
Some('\\') => Some(VirtualKeyCode::Backslash), Some('\\') => Some(VirtualKeyCode::Backslash),
_ => None _ => None,
} }
} }

File diff suppressed because it is too large Load diff

View file

@ -1,13 +1,15 @@
use std::{mem, ptr, io}; use std::{io, mem, os::windows::ffi::OsStrExt, path::Path, ptr};
use std::os::windows::ffi::OsStrExt;
use std::path::Path;
use winapi::ctypes::{c_int, wchar_t}; use winapi::{
use winapi::shared::minwindef::{BYTE, LPARAM, WPARAM}; ctypes::{c_int, wchar_t},
use winapi::shared::windef::{HICON, HWND}; shared::{
use winapi::um::winuser; minwindef::{BYTE, LPARAM, WPARAM},
windef::{HICON, HWND},
},
um::winuser,
};
use crate::icon::{Pixel, PIXEL_SIZE, Icon}; use crate::icon::{Icon, Pixel, PIXEL_SIZE};
impl Pixel { impl Pixel {
fn to_bgra(&mut self) { fn to_bgra(&mut self) {
@ -103,11 +105,6 @@ impl Drop for WinIcon {
pub fn unset_for_window(hwnd: HWND, icon_type: IconType) { pub fn unset_for_window(hwnd: HWND, icon_type: IconType) {
unsafe { unsafe {
winuser::SendMessageW( winuser::SendMessageW(hwnd, winuser::WM_SETICON, icon_type as WPARAM, 0 as LPARAM);
hwnd,
winuser::WM_SETICON,
icon_type as WPARAM,
0 as LPARAM,
);
} }
} }

View file

@ -1,14 +1,14 @@
#![cfg(target_os = "windows")] #![cfg(target_os = "windows")]
use winapi; use winapi::{self, shared::windef::HWND};
use winapi::shared::windef::HWND;
pub use self::event_loop::{EventLoop, EventLoopWindowTarget, EventLoopProxy}; pub use self::{
pub use self::monitor::MonitorHandle; event_loop::{EventLoop, EventLoopProxy, EventLoopWindowTarget},
pub use self::window::Window; monitor::MonitorHandle,
window::Window,
};
use crate::window::Icon; use crate::{event::DeviceId as RootDeviceId, window::Icon};
use crate::event::DeviceId as RootDeviceId;
#[derive(Clone, Default)] #[derive(Clone, Default)]
pub struct PlatformSpecificWindowBuilderAttributes { pub struct PlatformSpecificWindowBuilderAttributes {

Some files were not shown because too many files have changed in this diff Show more