summaryrefslogtreecommitdiffstats
path: root/winit
diff options
context:
space:
mode:
authorLibravatar Maja Kądziołka <maya@compilercrim.es>2024-08-11 22:33:17 +0200
committerLibravatar Maja Kądziołka <maya@compilercrim.es>2024-08-11 22:33:17 +0200
commitf92e01e913480e1450696f3d37af4bff09f661d0 (patch)
treed370a90803671ceabc5678b4622970190d7ccd54 /winit
parent1c8850023f2bdeae02ec061a49aa76dbb91262ad (diff)
downloadiced-f92e01e913480e1450696f3d37af4bff09f661d0.tar.gz
iced-f92e01e913480e1450696f3d37af4bff09f661d0.tar.bz2
iced-f92e01e913480e1450696f3d37af4bff09f661d0.zip
iced_winit: drop Clipboard before Window
Fixes #2482, avoids nasal daemons
Diffstat (limited to 'winit')
-rw-r--r--winit/src/clipboard.rs31
-rw-r--r--winit/src/program.rs2
2 files changed, 24 insertions, 9 deletions
diff --git a/winit/src/clipboard.rs b/winit/src/clipboard.rs
index 5237ca01..f8b90777 100644
--- a/winit/src/clipboard.rs
+++ b/winit/src/clipboard.rs
@@ -1,6 +1,8 @@
//! Access the clipboard.
use crate::core::clipboard::Kind;
+use winit::window::Window;
+use std::sync::Arc;
/// A buffer for short-term storage and transfer within and between
/// applications.
@@ -10,18 +12,31 @@ pub struct Clipboard {
}
enum State {
- Connected(window_clipboard::Clipboard),
+ Connected {
+ clipboard: window_clipboard::Clipboard,
+ // Held until drop to satisfy the safety invariants of
+ // `window_clipboard::Clipboard`.
+ //
+ // Note that the field ordering is load-bearing.
+ #[allow(dead_code)]
+ window: Arc<Window>,
+ },
Unavailable,
}
impl Clipboard {
/// Creates a new [`Clipboard`] for the given window.
- pub fn connect(window: &winit::window::Window) -> Clipboard {
+ pub fn connect(window: Arc<Window>) -> Clipboard {
#[allow(unsafe_code)]
- let state = unsafe { window_clipboard::Clipboard::connect(window) }
- .ok()
- .map(State::Connected)
- .unwrap_or(State::Unavailable);
+ // SAFETY: The window handle will stay alive throughout the entire
+ // lifetime of the `window_clipboard::Clipboard` because we hold
+ // the `Arc<Window>` together with `State`, and enum variant fields
+ // get dropped in declaration order.
+ let clipboard = unsafe { window_clipboard::Clipboard::connect(&window) };
+ let state = match clipboard {
+ Ok(clipboard) => State::Connected { clipboard, window },
+ Err(_) => State::Unavailable,
+ };
Clipboard { state }
}
@@ -37,7 +52,7 @@ impl Clipboard {
/// Reads the current content of the [`Clipboard`] as text.
pub fn read(&self, kind: Kind) -> Option<String> {
match &self.state {
- State::Connected(clipboard) => match kind {
+ State::Connected { clipboard, .. } => match kind {
Kind::Standard => clipboard.read().ok(),
Kind::Primary => clipboard.read_primary().and_then(Result::ok),
},
@@ -48,7 +63,7 @@ impl Clipboard {
/// Writes the given text contents to the [`Clipboard`].
pub fn write(&mut self, kind: Kind, contents: String) {
match &mut self.state {
- State::Connected(clipboard) => {
+ State::Connected { clipboard, .. } => {
let result = match kind {
Kind::Standard => clipboard.write(contents),
Kind::Primary => {
diff --git a/winit/src/program.rs b/winit/src/program.rs
index 3d709b7e..139b2b8f 100644
--- a/winit/src/program.rs
+++ b/winit/src/program.rs
@@ -307,7 +307,7 @@ where
}
};
- let clipboard = Clipboard::connect(&window);
+ let clipboard = Clipboard::connect(window.clone());
let finish_boot = async move {
let mut compositor =