feat(clipboard): add clipboard operation options and alternative x11 xclip backend. #882

This commit is contained in:
Federico Terzi 2021-11-21 19:38:43 +01:00
parent c4f4f438d3
commit 41b72acdf1
7 changed files with 282 additions and 27 deletions

View File

@ -24,7 +24,7 @@ use std::{
path::PathBuf, path::PathBuf,
}; };
use crate::Clipboard; use crate::{Clipboard, ClipboardOperationOptions};
use anyhow::Result; use anyhow::Result;
use log::error; use log::error;
use thiserror::Error; use thiserror::Error;
@ -38,7 +38,7 @@ impl CocoaClipboard {
} }
impl Clipboard for CocoaClipboard { impl Clipboard for CocoaClipboard {
fn get_text(&self) -> Option<String> { fn get_text(&self, _: &ClipboardOperationOptions) -> Option<String> {
let mut buffer: [i8; 2048] = [0; 2048]; let mut buffer: [i8; 2048] = [0; 2048];
let native_result = let native_result =
unsafe { ffi::clipboard_get_text(buffer.as_mut_ptr(), (buffer.len() - 1) as i32) }; unsafe { ffi::clipboard_get_text(buffer.as_mut_ptr(), (buffer.len() - 1) as i32) };
@ -50,7 +50,7 @@ impl Clipboard for CocoaClipboard {
} }
} }
fn set_text(&self, text: &str) -> anyhow::Result<()> { fn set_text(&self, text: &str, _: &ClipboardOperationOptions) -> anyhow::Result<()> {
let string = CString::new(text)?; let string = CString::new(text)?;
let native_result = unsafe { ffi::clipboard_set_text(string.as_ptr()) }; let native_result = unsafe { ffi::clipboard_set_text(string.as_ptr()) };
if native_result > 0 { if native_result > 0 {
@ -60,7 +60,11 @@ impl Clipboard for CocoaClipboard {
} }
} }
fn set_image(&self, image_path: &std::path::Path) -> anyhow::Result<()> { fn set_image(
&self,
image_path: &std::path::Path,
_: &ClipboardOperationOptions,
) -> anyhow::Result<()> {
if !image_path.exists() || !image_path.is_file() { if !image_path.exists() || !image_path.is_file() {
return Err(CocoaClipboardError::ImageNotFound(image_path.to_path_buf()).into()); return Err(CocoaClipboardError::ImageNotFound(image_path.to_path_buf()).into());
} }
@ -75,7 +79,12 @@ impl Clipboard for CocoaClipboard {
} }
} }
fn set_html(&self, html: &str, fallback_text: Option<&str>) -> anyhow::Result<()> { fn set_html(
&self,
html: &str,
fallback_text: Option<&str>,
_: &ClipboardOperationOptions,
) -> anyhow::Result<()> {
let html_string = CString::new(html)?; let html_string = CString::new(html)?;
let fallback_string = CString::new(fallback_text.unwrap_or_default())?; let fallback_string = CString::new(fallback_text.unwrap_or_default())?;
let fallback_ptr = if fallback_text.is_some() { let fallback_ptr = if fallback_text.is_some() {

View File

@ -37,10 +37,28 @@ mod wayland;
mod cocoa; mod cocoa;
pub trait Clipboard { pub trait Clipboard {
fn get_text(&self) -> Option<String>; fn get_text(&self, options: &ClipboardOperationOptions) -> Option<String>;
fn set_text(&self, text: &str) -> Result<()>; fn set_text(&self, text: &str, options: &ClipboardOperationOptions) -> Result<()>;
fn set_image(&self, image_path: &Path) -> Result<()>; fn set_image(&self, image_path: &Path, options: &ClipboardOperationOptions) -> Result<()>;
fn set_html(&self, html: &str, fallback_text: Option<&str>) -> Result<()>; fn set_html(
&self,
html: &str,
fallback_text: Option<&str>,
options: &ClipboardOperationOptions,
) -> Result<()>;
}
#[allow(dead_code)]
pub struct ClipboardOperationOptions {
pub use_xclip_backend: bool,
}
impl Default for ClipboardOperationOptions {
fn default() -> Self {
Self {
use_xclip_backend: false,
}
}
} }
#[allow(dead_code)] #[allow(dead_code)]
@ -74,8 +92,8 @@ pub fn get_clipboard(_: ClipboardOptions) -> Result<Box<dyn Clipboard>> {
#[cfg(target_os = "linux")] #[cfg(target_os = "linux")]
#[cfg(not(feature = "wayland"))] #[cfg(not(feature = "wayland"))]
pub fn get_clipboard(_: ClipboardOptions) -> Result<Box<dyn Clipboard>> { pub fn get_clipboard(_: ClipboardOptions) -> Result<Box<dyn Clipboard>> {
info!("using X11NativeClipboard"); info!("using X11Clipboard");
Ok(Box::new(x11::native::X11NativeClipboard::new()?)) Ok(Box::new(x11::X11Clipboard::new()?))
} }
#[cfg(target_os = "linux")] #[cfg(target_os = "linux")]

View File

@ -24,7 +24,7 @@ use std::{
process::Stdio, process::Stdio,
}; };
use crate::{Clipboard, ClipboardOptions}; use crate::{Clipboard, ClipboardOperationOptions, ClipboardOptions};
use anyhow::Result; use anyhow::Result;
use log::{error, warn}; use log::{error, warn};
use std::process::Command; use std::process::Command;
@ -74,7 +74,7 @@ impl WaylandFallbackClipboard {
} }
impl Clipboard for WaylandFallbackClipboard { impl Clipboard for WaylandFallbackClipboard {
fn get_text(&self) -> Option<String> { fn get_text(&self, _: &ClipboardOperationOptions) -> Option<String> {
let timeout = std::time::Duration::from_millis(self.command_timeout); let timeout = std::time::Duration::from_millis(self.command_timeout);
match Command::new("wl-paste") match Command::new("wl-paste")
.arg("--no-newline") .arg("--no-newline")
@ -116,11 +116,15 @@ impl Clipboard for WaylandFallbackClipboard {
} }
} }
fn set_text(&self, text: &str) -> anyhow::Result<()> { fn set_text(&self, text: &str, _: &ClipboardOperationOptions) -> anyhow::Result<()> {
self.invoke_command_with_timeout(&mut Command::new("wl-copy"), text.as_bytes(), "wl-copy") self.invoke_command_with_timeout(&mut Command::new("wl-copy"), text.as_bytes(), "wl-copy")
} }
fn set_image(&self, image_path: &std::path::Path) -> anyhow::Result<()> { fn set_image(
&self,
image_path: &std::path::Path,
_: &ClipboardOperationOptions,
) -> anyhow::Result<()> {
if !image_path.exists() || !image_path.is_file() { if !image_path.exists() || !image_path.is_file() {
return Err(WaylandFallbackClipboardError::ImageNotFound(image_path.to_path_buf()).into()); return Err(WaylandFallbackClipboardError::ImageNotFound(image_path.to_path_buf()).into());
} }
@ -137,7 +141,12 @@ impl Clipboard for WaylandFallbackClipboard {
) )
} }
fn set_html(&self, html: &str, _fallback_text: Option<&str>) -> anyhow::Result<()> { fn set_html(
&self,
html: &str,
_fallback_text: Option<&str>,
_: &ClipboardOperationOptions,
) -> anyhow::Result<()> {
self.invoke_command_with_timeout( self.invoke_command_with_timeout(
&mut Command::new("wl-copy").arg("--type").arg("text/html"), &mut Command::new("wl-copy").arg("--type").arg("text/html"),
html.as_bytes(), html.as_bytes(),

View File

@ -21,7 +21,7 @@ mod ffi;
use std::{ffi::CString, path::PathBuf}; use std::{ffi::CString, path::PathBuf};
use crate::Clipboard; use crate::{Clipboard, ClipboardOperationOptions};
use anyhow::Result; use anyhow::Result;
use log::error; use log::error;
use thiserror::Error; use thiserror::Error;
@ -36,7 +36,7 @@ impl Win32Clipboard {
} }
impl Clipboard for Win32Clipboard { impl Clipboard for Win32Clipboard {
fn get_text(&self) -> Option<String> { fn get_text(&self, _: &ClipboardOperationOptions) -> Option<String> {
let mut buffer: [u16; 2048] = [0; 2048]; let mut buffer: [u16; 2048] = [0; 2048];
let native_result = let native_result =
unsafe { ffi::clipboard_get_text(buffer.as_mut_ptr(), (buffer.len() - 1) as i32) }; unsafe { ffi::clipboard_get_text(buffer.as_mut_ptr(), (buffer.len() - 1) as i32) };
@ -48,7 +48,7 @@ impl Clipboard for Win32Clipboard {
} }
} }
fn set_text(&self, text: &str) -> anyhow::Result<()> { fn set_text(&self, text: &str, _: &ClipboardOperationOptions) -> anyhow::Result<()> {
let string = U16CString::from_str(text)?; let string = U16CString::from_str(text)?;
let native_result = unsafe { ffi::clipboard_set_text(string.as_ptr()) }; let native_result = unsafe { ffi::clipboard_set_text(string.as_ptr()) };
if native_result > 0 { if native_result > 0 {
@ -58,7 +58,11 @@ impl Clipboard for Win32Clipboard {
} }
} }
fn set_image(&self, image_path: &std::path::Path) -> anyhow::Result<()> { fn set_image(
&self,
image_path: &std::path::Path,
_: &ClipboardOperationOptions,
) -> anyhow::Result<()> {
if !image_path.exists() || !image_path.is_file() { if !image_path.exists() || !image_path.is_file() {
return Err(Win32ClipboardError::ImageNotFound(image_path.to_path_buf()).into()); return Err(Win32ClipboardError::ImageNotFound(image_path.to_path_buf()).into());
} }
@ -73,7 +77,12 @@ impl Clipboard for Win32Clipboard {
} }
} }
fn set_html(&self, html: &str, fallback_text: Option<&str>) -> anyhow::Result<()> { fn set_html(
&self,
html: &str,
fallback_text: Option<&str>,
_: &ClipboardOperationOptions,
) -> anyhow::Result<()> {
let html_descriptor = generate_html_descriptor(html); let html_descriptor = generate_html_descriptor(html);
let html_string = CString::new(html_descriptor)?; let html_string = CString::new(html_descriptor)?;
let fallback_string = U16CString::from_str(fallback_text.unwrap_or_default())?; let fallback_string = U16CString::from_str(fallback_text.unwrap_or_default())?;

View File

@ -17,4 +17,66 @@
* along with espanso. If not, see <https://www.gnu.org/licenses/>. * along with espanso. If not, see <https://www.gnu.org/licenses/>.
*/ */
pub(crate) mod native; use anyhow::Result;
use crate::{Clipboard, ClipboardOperationOptions};
mod native;
mod xclip;
pub(crate) struct X11Clipboard {
native_backend: native::X11NativeClipboard,
xclip_backend: xclip::XClipClipboard,
}
impl X11Clipboard {
pub fn new() -> Result<Self> {
Ok(Self {
native_backend: native::X11NativeClipboard::new()?,
xclip_backend: xclip::XClipClipboard::new(),
})
}
}
impl Clipboard for X11Clipboard {
fn get_text(&self, options: &ClipboardOperationOptions) -> Option<String> {
if options.use_xclip_backend {
self.xclip_backend.get_text(options)
} else {
self.native_backend.get_text(options)
}
}
fn set_text(&self, text: &str, options: &ClipboardOperationOptions) -> anyhow::Result<()> {
if options.use_xclip_backend {
self.xclip_backend.set_text(text, options)
} else {
self.native_backend.set_text(text, options)
}
}
fn set_image(
&self,
image_path: &std::path::Path,
options: &ClipboardOperationOptions,
) -> anyhow::Result<()> {
if options.use_xclip_backend {
self.xclip_backend.set_image(image_path, options)
} else {
self.native_backend.set_image(image_path, options)
}
}
fn set_html(
&self,
html: &str,
fallback_text: Option<&str>,
options: &ClipboardOperationOptions,
) -> anyhow::Result<()> {
if options.use_xclip_backend {
self.xclip_backend.set_html(html, fallback_text, options)
} else {
self.native_backend.set_html(html, fallback_text, options)
}
}
}

View File

@ -23,7 +23,7 @@ use std::{
path::PathBuf, path::PathBuf,
}; };
use crate::Clipboard; use crate::{Clipboard, ClipboardOperationOptions};
use anyhow::Result; use anyhow::Result;
use std::os::raw::c_char; use std::os::raw::c_char;
use thiserror::Error; use thiserror::Error;
@ -39,7 +39,7 @@ impl X11NativeClipboard {
} }
impl Clipboard for X11NativeClipboard { impl Clipboard for X11NativeClipboard {
fn get_text(&self) -> Option<String> { fn get_text(&self, _: &ClipboardOperationOptions) -> Option<String> {
let mut buffer: [c_char; 2048] = [0; 2048]; let mut buffer: [c_char; 2048] = [0; 2048];
let native_result = let native_result =
unsafe { ffi::clipboard_x11_get_text(buffer.as_mut_ptr(), (buffer.len() - 1) as i32) }; unsafe { ffi::clipboard_x11_get_text(buffer.as_mut_ptr(), (buffer.len() - 1) as i32) };
@ -51,7 +51,7 @@ impl Clipboard for X11NativeClipboard {
} }
} }
fn set_text(&self, text: &str) -> anyhow::Result<()> { fn set_text(&self, text: &str, _: &ClipboardOperationOptions) -> anyhow::Result<()> {
let string = CString::new(text)?; let string = CString::new(text)?;
let native_result = unsafe { ffi::clipboard_x11_set_text(string.as_ptr()) }; let native_result = unsafe { ffi::clipboard_x11_set_text(string.as_ptr()) };
if native_result > 0 { if native_result > 0 {
@ -61,7 +61,11 @@ impl Clipboard for X11NativeClipboard {
} }
} }
fn set_image(&self, image_path: &std::path::Path) -> anyhow::Result<()> { fn set_image(
&self,
image_path: &std::path::Path,
_: &ClipboardOperationOptions,
) -> anyhow::Result<()> {
if !image_path.exists() || !image_path.is_file() { if !image_path.exists() || !image_path.is_file() {
return Err(X11NativeClipboardError::ImageNotFound(image_path.to_path_buf()).into()); return Err(X11NativeClipboardError::ImageNotFound(image_path.to_path_buf()).into());
} }
@ -80,7 +84,12 @@ impl Clipboard for X11NativeClipboard {
} }
} }
fn set_html(&self, html: &str, fallback_text: Option<&str>) -> anyhow::Result<()> { fn set_html(
&self,
html: &str,
fallback_text: Option<&str>,
_: &ClipboardOperationOptions,
) -> anyhow::Result<()> {
let html_string = CString::new(html)?; let html_string = CString::new(html)?;
let fallback_string = CString::new(fallback_text.unwrap_or_default())?; let fallback_string = CString::new(fallback_text.unwrap_or_default())?;
let fallback_ptr = if fallback_text.is_some() { let fallback_ptr = if fallback_text.is_some() {

View File

@ -0,0 +1,139 @@
/*
* This file is part of espanso.
*
* Copyright (C) 2019-2021 Federico Terzi
*
* espanso is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* espanso is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with espanso. If not, see <https://www.gnu.org/licenses/>.
*/
use anyhow::bail;
use log::error;
use std::io::Write;
use std::process::{Command, Stdio};
use crate::{Clipboard, ClipboardOperationOptions};
pub struct XClipClipboard {
is_xclip_available: bool,
}
impl XClipClipboard {
pub fn new() -> Self {
let command = Command::new("xclipz").arg("-h").output();
let is_xclip_available = command
.map(|output| output.status.success())
.unwrap_or(false);
Self { is_xclip_available }
}
}
impl Clipboard for XClipClipboard {
fn get_text(&self, _: &ClipboardOperationOptions) -> Option<String> {
if !self.is_xclip_available {
error!("attempted to use XClipClipboard, but `xclip` command can't be called");
return None;
}
match Command::new("xclip").args(&["-o", "-sel", "clip"]).output() {
Ok(output) => {
if output.status.success() {
let s = String::from_utf8_lossy(&output.stdout);
return Some(s.to_string());
}
}
Err(error) => {
error!("xclip reported an error: {}", error);
}
}
None
}
fn set_text(&self, text: &str, _: &ClipboardOperationOptions) -> anyhow::Result<()> {
if !self.is_xclip_available {
bail!("attempted to use XClipClipboard, but `xclip` command can't be called");
}
let mut child = Command::new("xclip")
.args(&["-sel", "clip"])
.stdin(Stdio::piped())
.spawn()?;
let stdin = child.stdin.as_mut();
if let Some(input) = stdin {
input.write_all(text.as_bytes())?;
child.wait()?;
}
Ok(())
}
fn set_image(
&self,
image_path: &std::path::Path,
_: &ClipboardOperationOptions,
) -> anyhow::Result<()> {
if !self.is_xclip_available {
bail!("attempted to use XClipClipboard, but `xclip` command can't be called");
}
let extension = image_path.extension();
let mime = match extension {
Some(ext) => {
let ext = ext.to_string_lossy().to_lowercase();
match ext.as_ref() {
"png" => "image/png",
"jpg" | "jpeg" => "image/jpeg",
"gif" => "image/gif",
"svg" => "image/svg",
_ => "image/png",
}
}
None => "image/png",
};
let image_path = image_path.to_string_lossy();
Command::new("xclip")
.args(&["-selection", "clipboard", "-t", mime, "-i", &image_path])
.spawn()?;
Ok(())
}
fn set_html(
&self,
html: &str,
_: Option<&str>,
_: &ClipboardOperationOptions,
) -> anyhow::Result<()> {
if !self.is_xclip_available {
bail!("attempted to use XClipClipboard, but `xclip` command can't be called");
}
let mut child = Command::new("xclip")
.args(&["-sel", "clip", "-t", "text/html"])
.stdin(Stdio::piped())
.spawn()?;
let stdin = child.stdin.as_mut();
if let Some(input) = stdin {
input.write_all(html.as_bytes())?;
child.wait()?;
}
Ok(())
}
}