1#![deny(clippy::all, clippy::if_not_else, clippy::enum_glob_use)]
7use std::borrow::Cow;
8use std::ffi::c_void;
9use std::io::Result;
10use std::sync::mpsc::{self, Receiver};
11
12use sctk::reexports::calloop::channel::{self, Sender};
13use sctk::reexports::client::backend::Backend;
14use sctk::reexports::client::protocol::wl_surface::WlSurface;
15use sctk::reexports::client::Connection;
16
17#[cfg(feature = "dnd")]
18pub mod dnd;
19pub mod mime;
20mod state;
21pub mod text;
22mod worker;
23
24use mime::{AllowedMimeTypes, AsMimeTypes, MimeType};
25use state::Target;
26use text::Text;
27
28pub type SimpleClipboard = Clipboard<WlSurface>;
29
30pub struct Clipboard<T> {
32 request_sender: Sender<worker::Command<T>>,
33 request_receiver: Receiver<Result<(Vec<u8>, MimeType)>>,
34 clipboard_thread: Option<std::thread::JoinHandle<()>>,
35 connection: Connection,
36}
37
38impl<T: 'static + Send + Clone> Clipboard<T> {
39 pub unsafe fn new(display: *mut c_void) -> Self {
47 let backend = unsafe { Backend::from_foreign_display(display.cast()) };
48 let connection = Connection::from_backend(backend);
49
50 let (request_sender, rx_chan) = channel::channel();
52 let (clipboard_reply_sender, request_receiver) = mpsc::channel();
54
55 let name = String::from("smithay-clipboard");
56 let clipboard_thread =
57 worker::spawn(name, connection.clone(), rx_chan, clipboard_reply_sender);
58
59 Self { request_receiver, request_sender, clipboard_thread, connection }
60 }
61
62 pub fn load<D: AllowedMimeTypes + 'static>(&self) -> Result<D> {
66 self.load_inner(Target::Clipboard, D::allowed())
67 }
68
69 pub fn load_text(&self) -> Result<String> {
73 self.load::<Text>().map(|t| t.0)
74 }
75
76 pub fn load_primary<D: AllowedMimeTypes + 'static>(&self) -> Result<D> {
81 self.load_inner(Target::Primary, D::allowed())
82 }
83
84 pub fn load_primary_text(&self) -> Result<String> {
88 self.load_primary::<Text>().map(|t| t.0)
89 }
90
91 pub fn load_mime<D: TryFrom<(Vec<u8>, MimeType)>>(
95 &self,
96 allowed: impl Into<Cow<'static, [MimeType]>>,
97 ) -> Result<D> {
98 self.load_inner(Target::Clipboard, allowed).and_then(|d| {
99 D::try_from(d).map_err(|_| {
100 std::io::Error::new(
101 std::io::ErrorKind::Other,
102 "Failed to load data of the requested type.",
103 )
104 })
105 })
106 }
107
108 pub fn load_primary_mime<D: TryFrom<(Vec<u8>, MimeType)>>(
112 &self,
113 allowed: impl Into<Cow<'static, [MimeType]>>,
114 ) -> Result<D> {
115 self.load_inner(Target::Primary, allowed).and_then(|d| {
116 D::try_from(d).map_err(|_| {
117 std::io::Error::new(
118 std::io::ErrorKind::Other,
119 "Failed to load data of the requested type.",
120 )
121 })
122 })
123 }
124
125 pub fn store<D: AsMimeTypes + Send + 'static>(&self, data: D) {
129 self.store_inner(data, Target::Clipboard);
130 }
131
132 pub fn store_text<D: Into<String>>(&self, text: D) {
136 self.store(Text(text.into()));
137 }
138
139 pub fn store_primary<D: AsMimeTypes + Send + 'static>(&self, data: D) {
144 self.store_inner(data, Target::Primary);
145 }
146
147 pub fn store_primary_text<D: Into<String>>(&self, text: D) {
151 self.store_primary(Text(text.into()));
152 }
153
154 fn load_inner<D: TryFrom<(Vec<u8>, MimeType)> + 'static>(
155 &self,
156 target: Target,
157 allowed: impl Into<Cow<'static, [MimeType]>>,
158 ) -> Result<D> {
159 let _ = self.request_sender.send(worker::Command::Load(allowed.into(), target));
160
161 match self.request_receiver.recv() {
162 Ok(res) => res.and_then(|(data, mime)| {
163 D::try_from((data, mime)).map_err(|_| {
164 std::io::Error::new(
165 std::io::ErrorKind::Other,
166 "Failed to load data of the requested type.",
167 )
168 })
169 }),
170 Err(_) => Err(std::io::Error::new(std::io::ErrorKind::Other, "clipboard is dead.")),
173 }
174 }
175
176 fn store_inner<D: AsMimeTypes + Send + 'static>(&self, data: D, target: Target) {
177 let request = worker::Command::Store(Box::new(data), target);
178 let _ = self.request_sender.send(request);
179 }
180}
181
182impl<T> Drop for Clipboard<T> {
183 fn drop(&mut self) {
184 let _ = self.request_sender.send(worker::Command::Exit);
186 if let Some(clipboard_thread) = self.clipboard_thread.take() {
187 let _ = clipboard_thread.join();
188 }
189 }
190}