mirror of
https://github.com/rustdesk/rustdesk.git
synced 2026-03-26 22:51:03 +03:00
feat: cursor, linux (#12822)
* feat: cursor, linux Signed-off-by: fufesou <linlong1266@gmail.com> * refact: cursor, text, white background Signed-off-by: fufesou <linlong1266@gmail.com> --------- Signed-off-by: fufesou <linlong1266@gmail.com>
This commit is contained in:
426
src/whiteboard/linux.rs
Normal file
426
src/whiteboard/linux.rs
Normal file
@@ -0,0 +1,426 @@
|
||||
use super::{
|
||||
server::{Ripple, EVENT_PROXY},
|
||||
win_linux::{create_font_face, draw_text},
|
||||
Cursor, CustomEvent,
|
||||
};
|
||||
use hbb_common::{bail, log, tokio::sync::mpsc::unbounded_channel, ResultType};
|
||||
use softbuffer::{Context, Surface};
|
||||
use std::{
|
||||
collections::HashMap,
|
||||
ffi::{c_int, c_short, c_ulong, c_ushort},
|
||||
num::NonZeroU32,
|
||||
sync::Arc,
|
||||
time::Instant,
|
||||
};
|
||||
use tiny_skia::{Color, FillRule, Paint, PathBuilder, PixmapMut, Stroke, Transform};
|
||||
use ttf_parser::Face;
|
||||
use winit::raw_window_handle::{
|
||||
DisplayHandle, HasDisplayHandle, HasWindowHandle, RawDisplayHandle, RawWindowHandle,
|
||||
};
|
||||
use winit::{
|
||||
application::ApplicationHandler,
|
||||
dpi::{PhysicalPosition, PhysicalSize},
|
||||
event::WindowEvent,
|
||||
event_loop::{ActiveEventLoop, EventLoop},
|
||||
platform::x11::{WindowAttributesExtX11, WindowType},
|
||||
window::{Window, WindowId, WindowLevel},
|
||||
};
|
||||
|
||||
enum _XDisplay {}
|
||||
type Display = _XDisplay;
|
||||
|
||||
type XID = c_ulong;
|
||||
type XserverRegion = XID;
|
||||
|
||||
#[derive(Debug, Clone, Copy, PartialEq)]
|
||||
#[repr(C)]
|
||||
pub struct XRectangle {
|
||||
pub x: c_short,
|
||||
pub y: c_short,
|
||||
pub width: c_ushort,
|
||||
pub height: c_ushort,
|
||||
}
|
||||
|
||||
#[link(name = "Xfixes")]
|
||||
extern "C" {
|
||||
fn XFixesCreateRegion(
|
||||
dpy: *mut Display,
|
||||
rectangles: *mut XRectangle,
|
||||
nrectangles: c_int,
|
||||
) -> XserverRegion;
|
||||
fn XFixesDestroyRegion(dpy: *mut Display, region: XserverRegion) -> ();
|
||||
fn XFixesSetWindowShapeRegion(
|
||||
dpy: *mut Display,
|
||||
win: XID,
|
||||
shape_kind: c_int,
|
||||
x_off: c_int,
|
||||
y_off: c_int,
|
||||
region: XserverRegion,
|
||||
) -> ();
|
||||
}
|
||||
|
||||
const SHAPE_INPUT: std::ffi::c_int = 2;
|
||||
|
||||
pub fn run() {
|
||||
let event_loop = match EventLoop::<(String, CustomEvent)>::with_user_event().build() {
|
||||
Ok(el) => el,
|
||||
Err(e) => {
|
||||
log::error!("Failed to create event loop: {}", e);
|
||||
return;
|
||||
}
|
||||
};
|
||||
|
||||
let event_loop_proxy = event_loop.create_proxy();
|
||||
EVENT_PROXY.write().unwrap().replace(event_loop_proxy);
|
||||
|
||||
let (tx_exit, rx_exit) = unbounded_channel();
|
||||
std::thread::spawn(move || {
|
||||
super::server::start_ipc(rx_exit);
|
||||
});
|
||||
|
||||
let mut app = match WhiteboardApplication::new(&event_loop) {
|
||||
Ok(app) => app,
|
||||
Err(e) => {
|
||||
log::error!("Failed to create whiteboard application: {}", e);
|
||||
tx_exit.send(()).ok();
|
||||
return;
|
||||
}
|
||||
};
|
||||
|
||||
if let Err(e) = event_loop.run_app(&mut app) {
|
||||
log::error!("Failed to run app: {}", e);
|
||||
tx_exit.send(()).ok();
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
struct WindowState {
|
||||
window: Arc<Window>,
|
||||
// NOTE: This surface must be dropped before the `Window`.
|
||||
surface: Surface<DisplayHandle<'static>, Arc<Window>>,
|
||||
ripples: Vec<Ripple>,
|
||||
last_cursors: HashMap<String, Cursor>,
|
||||
}
|
||||
|
||||
struct WhiteboardApplication {
|
||||
windows: Vec<WindowState>,
|
||||
// Drawing context.
|
||||
//
|
||||
// With OpenGL it could be EGLDisplay.
|
||||
context: Option<Context<DisplayHandle<'static>>>,
|
||||
face: Option<Face<'static>>,
|
||||
close_requested: bool,
|
||||
}
|
||||
|
||||
impl WhiteboardApplication {
|
||||
fn new<T>(event_loop: &EventLoop<T>) -> ResultType<Self> {
|
||||
// https://github.com/rust-windowing/winit/blob/f6893a4390dfe6118ce4b33458d458fd3efd3025/examples/window.rs#L91
|
||||
// SAFETY: we drop the context right before the event loop is stopped, thus making it safe.
|
||||
let context = match Context::new(unsafe {
|
||||
std::mem::transmute::<DisplayHandle<'_>, DisplayHandle<'static>>(
|
||||
event_loop.display_handle()?,
|
||||
)
|
||||
}) {
|
||||
Ok(ctx) => Some(ctx),
|
||||
Err(e) => {
|
||||
bail!("Failed to create context: {}", e);
|
||||
}
|
||||
};
|
||||
let face = match create_font_face() {
|
||||
Ok(face) => Some(face),
|
||||
Err(err) => {
|
||||
log::error!("Failed to create font face: {}", err);
|
||||
None
|
||||
}
|
||||
};
|
||||
Ok(Self {
|
||||
windows: Vec::new(),
|
||||
context,
|
||||
face,
|
||||
close_requested: false,
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
impl ApplicationHandler<(String, CustomEvent)> for WhiteboardApplication {
|
||||
fn user_event(&mut self, _event_loop: &ActiveEventLoop, (k, evt): (String, CustomEvent)) {
|
||||
match evt {
|
||||
CustomEvent::Cursor(cursor) => {
|
||||
if let Some(state) = self.windows.first_mut() {
|
||||
if cursor.btns != 0 {
|
||||
state.ripples.push(Ripple {
|
||||
x: cursor.x,
|
||||
y: cursor.y,
|
||||
start_time: Instant::now(),
|
||||
});
|
||||
}
|
||||
state.last_cursors.insert(k, cursor);
|
||||
state.window.request_redraw();
|
||||
}
|
||||
}
|
||||
CustomEvent::Exit => {
|
||||
self.close_requested = true;
|
||||
}
|
||||
_ => {}
|
||||
}
|
||||
}
|
||||
|
||||
fn resumed(&mut self, event_loop: &ActiveEventLoop) {
|
||||
let (x, y, w, h) = match super::server::get_displays_rect() {
|
||||
Ok(r) => r,
|
||||
Err(err) => {
|
||||
log::error!("Failed to get displays rect: {}", err);
|
||||
self.close_requested = true;
|
||||
return;
|
||||
}
|
||||
};
|
||||
|
||||
let window_attributes = Window::default_attributes()
|
||||
.with_title("RustDesk whiteboard")
|
||||
.with_inner_size(PhysicalSize::new(w, h))
|
||||
.with_position(PhysicalPosition::new(x, y))
|
||||
.with_decorations(false)
|
||||
.with_transparent(true)
|
||||
.with_window_level(WindowLevel::AlwaysOnTop)
|
||||
.with_x11_window_type(vec![WindowType::Dock])
|
||||
.with_override_redirect(true);
|
||||
|
||||
let window = match event_loop.create_window(window_attributes) {
|
||||
Ok(window) => Arc::new(window),
|
||||
Err(e) => {
|
||||
log::error!("Failed to create window: {}", e);
|
||||
self.close_requested = true;
|
||||
return;
|
||||
}
|
||||
};
|
||||
|
||||
let display = match window.display_handle() {
|
||||
Ok(d) => d,
|
||||
Err(e) => {
|
||||
log::error!("Failed to get display handle: {}", e);
|
||||
self.close_requested = true;
|
||||
return;
|
||||
}
|
||||
};
|
||||
let rwh = match window.window_handle() {
|
||||
Ok(w) => w,
|
||||
Err(e) => {
|
||||
log::error!("Failed to get window handle: {}", e);
|
||||
self.close_requested = true;
|
||||
return;
|
||||
}
|
||||
};
|
||||
|
||||
// Both the following block and `window.set_cursor_hittest(false)` in `draw()` are necessary to ensure cursor events are properly passed through the window.
|
||||
// These issues may be related to winit X11 handling.
|
||||
// https://github.com/rust-windowing/winit/issues/3509
|
||||
// https://github.com/rust-windowing/winit/issues/4120
|
||||
// If either block is removed, cursor events may not be passed through as expected.
|
||||
// If you update winit, please revisit this workaround.
|
||||
match (rwh.as_raw(), display.as_raw()) {
|
||||
(RawWindowHandle::Xlib(xlib_window), RawDisplayHandle::Xlib(xlib_display)) => {
|
||||
unsafe {
|
||||
let xwindow = xlib_window.window;
|
||||
if let Some(display_ptr) = xlib_display.display {
|
||||
let xdisplay = display_ptr.as_ptr() as *mut Display;
|
||||
// Mouse event passthrough
|
||||
let empty_region = XFixesCreateRegion(xdisplay, std::ptr::null_mut(), 0);
|
||||
if empty_region == 0 {
|
||||
log::error!("XFixesCreateRegion failed: returned null region");
|
||||
} else {
|
||||
XFixesSetWindowShapeRegion(
|
||||
xdisplay,
|
||||
xwindow,
|
||||
SHAPE_INPUT,
|
||||
0,
|
||||
0,
|
||||
empty_region,
|
||||
);
|
||||
XFixesDestroyRegion(xdisplay, empty_region);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
_ => {
|
||||
log::error!("Unsupported windowing system for shape extension");
|
||||
self.close_requested = true;
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
let Some(ctx) = self.context.as_ref() else {
|
||||
// unreachable
|
||||
self.close_requested = true;
|
||||
return;
|
||||
};
|
||||
|
||||
let surface = match Surface::new(ctx, window.clone()) {
|
||||
Ok(s) => s,
|
||||
Err(e) => {
|
||||
log::error!("Failed to create surface: {}", e);
|
||||
self.close_requested = true;
|
||||
return;
|
||||
}
|
||||
};
|
||||
|
||||
let state = WindowState {
|
||||
window,
|
||||
surface,
|
||||
ripples: Vec::new(),
|
||||
last_cursors: HashMap::new(),
|
||||
};
|
||||
|
||||
self.windows.push(state);
|
||||
}
|
||||
|
||||
fn window_event(
|
||||
&mut self,
|
||||
_event_loop: &ActiveEventLoop,
|
||||
window_id: WindowId,
|
||||
event: WindowEvent,
|
||||
) {
|
||||
match event {
|
||||
WindowEvent::CloseRequested => {
|
||||
self.close_requested = true;
|
||||
}
|
||||
WindowEvent::RedrawRequested => {
|
||||
let Some(state) = self.windows.iter_mut().find(|w| w.window.id() == window_id)
|
||||
else {
|
||||
log::error!("No window found for id: {:?}", window_id);
|
||||
return;
|
||||
};
|
||||
if let Err(err) = state.draw(&self.face) {
|
||||
log::error!("Failed to draw window: {}", err);
|
||||
}
|
||||
}
|
||||
_ => (),
|
||||
}
|
||||
}
|
||||
|
||||
fn about_to_wait(&mut self, event_loop: &ActiveEventLoop) {
|
||||
if !self.close_requested {
|
||||
for state in self.windows.iter() {
|
||||
state.window.request_redraw();
|
||||
}
|
||||
} else {
|
||||
event_loop.exit();
|
||||
}
|
||||
}
|
||||
|
||||
fn exiting(&mut self, _event_loop: &ActiveEventLoop) {
|
||||
// We must drop the context here.
|
||||
self.context = None;
|
||||
}
|
||||
}
|
||||
|
||||
impl WindowState {
|
||||
fn draw(&mut self, face: &Option<Face<'static>>) -> ResultType<()> {
|
||||
let (width, height) = {
|
||||
let size = self.window.inner_size();
|
||||
(size.width, size.height)
|
||||
};
|
||||
|
||||
let (Some(width), Some(height)) = (NonZeroU32::new(width), NonZeroU32::new(height)) else {
|
||||
bail!("Invalid window size, {width}x{height}")
|
||||
};
|
||||
if let Err(e) = self.surface.resize(width, height) {
|
||||
bail!("Failed to resize surface: {}", e);
|
||||
}
|
||||
|
||||
let mut buffer = match self.surface.buffer_mut() {
|
||||
Ok(buf) => buf,
|
||||
Err(e) => {
|
||||
bail!("Failed to get buffer: {}", e);
|
||||
}
|
||||
};
|
||||
|
||||
let Some(mut pixmap) = PixmapMut::from_bytes(
|
||||
bytemuck::cast_slice_mut(&mut buffer),
|
||||
width.get(),
|
||||
height.get(),
|
||||
) else {
|
||||
bail!("Failed to create pixmap from buffer");
|
||||
};
|
||||
pixmap.fill(Color::TRANSPARENT);
|
||||
|
||||
Ripple::retain_active(&mut self.ripples);
|
||||
for ripple in &self.ripples {
|
||||
let (radius, alpha) = ripple.get_radius_alpha();
|
||||
|
||||
let mut ripple_paint = Paint::default();
|
||||
// Note: The real color is bgra here.
|
||||
ripple_paint.set_color_rgba8(64, 64, 255, (alpha * 128.0) as u8);
|
||||
ripple_paint.anti_alias = true;
|
||||
|
||||
let mut ripple_pb = PathBuilder::new();
|
||||
ripple_pb.push_circle(ripple.x, ripple.y, radius);
|
||||
if let Some(path) = ripple_pb.finish() {
|
||||
pixmap.fill_path(
|
||||
&path,
|
||||
&ripple_paint,
|
||||
FillRule::Winding,
|
||||
Transform::identity(),
|
||||
None,
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
for cursor in self.last_cursors.values() {
|
||||
let (x, y) = (cursor.x, cursor.y);
|
||||
let size = 1.5f32;
|
||||
|
||||
let mut pb = PathBuilder::new();
|
||||
pb.move_to(x, y);
|
||||
pb.line_to(x, y + 16.0 * size);
|
||||
pb.line_to(x + 4.0 * size, y + 13.0 * size);
|
||||
pb.line_to(x + 7.0 * size, y + 20.0 * size);
|
||||
pb.line_to(x + 9.0 * size, y + 19.0 * size);
|
||||
pb.line_to(x + 6.0 * size, y + 12.0 * size);
|
||||
pb.line_to(x + 11.0 * size, y + 12.0 * size);
|
||||
pb.close();
|
||||
|
||||
if let Some(path) = pb.finish() {
|
||||
let mut arrow_paint = Paint::default();
|
||||
let rgba = super::argb_to_rgba(cursor.argb);
|
||||
arrow_paint.set_color_rgba8(rgba.2, rgba.1, rgba.0, rgba.3);
|
||||
arrow_paint.anti_alias = true;
|
||||
pixmap.fill_path(
|
||||
&path,
|
||||
&arrow_paint,
|
||||
FillRule::Winding,
|
||||
Transform::identity(),
|
||||
None,
|
||||
);
|
||||
|
||||
let mut black_paint = Paint::default();
|
||||
black_paint.set_color_rgba8(0, 0, 0, 255);
|
||||
black_paint.anti_alias = true;
|
||||
let mut stroke = Stroke::default();
|
||||
stroke.width = 1.0f32;
|
||||
pixmap.stroke_path(&path, &black_paint, &stroke, Transform::identity(), None);
|
||||
|
||||
face.as_ref().map(|face| {
|
||||
draw_text(
|
||||
&mut pixmap,
|
||||
face,
|
||||
&cursor.text,
|
||||
x + 24.0 * size,
|
||||
y + 24.0 * size,
|
||||
&arrow_paint,
|
||||
14.0f32,
|
||||
);
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
self.window.pre_present_notify();
|
||||
|
||||
if let Err(e) = buffer.present() {
|
||||
log::error!("Failed to present buffer: {}", e);
|
||||
}
|
||||
|
||||
self.window.set_cursor_hittest(false).ok();
|
||||
|
||||
Ok(())
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user