402 lines
12 KiB
Rust
402 lines
12 KiB
Rust
use self::{event_loop::EventLoopHandler, state::WindowState};
|
|
use crate::{
|
|
bind_globals,
|
|
rendering::{
|
|
egl_context::EGLContext, femtovg_window::FemtoVGWindow, slint_platform::CustomSlintPlatform,
|
|
},
|
|
};
|
|
use anyhow::{Context, Result};
|
|
use log::{debug, info};
|
|
use slint::{platform::femtovg_renderer::FemtoVGRenderer, ComponentHandle, LogicalPosition};
|
|
use slint_interpreter::{ComponentDefinition, ComponentInstance};
|
|
use smithay_client_toolkit::reexports::{
|
|
calloop::{self, EventLoop},
|
|
protocols_wlr::layer_shell::v1::client::{
|
|
zwlr_layer_shell_v1::{self, ZwlrLayerShellV1},
|
|
zwlr_layer_surface_v1::{Anchor, KeyboardInteractivity, ZwlrLayerSurfaceV1},
|
|
},
|
|
};
|
|
use std::{
|
|
cell::{Ref, RefCell},
|
|
rc::Rc,
|
|
};
|
|
use wayland_client::{
|
|
globals::{registry_queue_init, GlobalList},
|
|
protocol::{
|
|
wl_compositor::WlCompositor, wl_display::WlDisplay, wl_output::WlOutput, wl_seat::WlSeat,
|
|
wl_surface::WlSurface,
|
|
},
|
|
Connection, EventQueue, Proxy, QueueHandle,
|
|
};
|
|
|
|
mod event_loop;
|
|
mod macros;
|
|
mod state;
|
|
|
|
pub struct WindowConfig {
|
|
height: u32,
|
|
layer: zwlr_layer_shell_v1::Layer,
|
|
margin: (i32, i32, i32, i32),
|
|
anchor: Anchor,
|
|
keyboard_interactivity: KeyboardInteractivity,
|
|
exclusive_zone: i32,
|
|
scale_factor: f32,
|
|
namespace: String,
|
|
component_definition: Option<ComponentDefinition>,
|
|
}
|
|
|
|
impl Default for WindowConfig {
|
|
fn default() -> Self {
|
|
Self {
|
|
height: 30,
|
|
layer: zwlr_layer_shell_v1::Layer::Top,
|
|
margin: (0, 0, 0, 0),
|
|
anchor: Anchor::Top | Anchor::Left | Anchor::Right,
|
|
keyboard_interactivity: KeyboardInteractivity::OnDemand,
|
|
exclusive_zone: -1,
|
|
namespace: "layer-shika".to_owned(),
|
|
scale_factor: 1.0,
|
|
component_definition: None,
|
|
}
|
|
}
|
|
}
|
|
|
|
pub struct WindowingSystemBuilder {
|
|
config: WindowConfig,
|
|
}
|
|
|
|
impl Default for WindowingSystemBuilder {
|
|
fn default() -> Self {
|
|
Self::new()
|
|
}
|
|
}
|
|
|
|
impl WindowingSystemBuilder {
|
|
#[inline]
|
|
#[must_use]
|
|
pub fn new() -> Self {
|
|
Self {
|
|
config: WindowConfig::default(),
|
|
}
|
|
}
|
|
|
|
#[must_use]
|
|
pub const fn with_height(mut self, height: u32) -> Self {
|
|
self.config.height = height;
|
|
self
|
|
}
|
|
|
|
#[must_use]
|
|
pub const fn with_layer(mut self, layer: zwlr_layer_shell_v1::Layer) -> Self {
|
|
self.config.layer = layer;
|
|
self
|
|
}
|
|
|
|
#[must_use]
|
|
pub const fn with_margin(mut self, top: i32, right: i32, bottom: i32, left: i32) -> Self {
|
|
self.config.margin = (top, right, bottom, left);
|
|
self
|
|
}
|
|
|
|
#[must_use]
|
|
pub const fn with_anchor(mut self, anchor: Anchor) -> Self {
|
|
self.config.anchor = anchor;
|
|
self
|
|
}
|
|
|
|
#[must_use]
|
|
pub const fn with_keyboard_interactivity(
|
|
mut self,
|
|
interactivity: KeyboardInteractivity,
|
|
) -> Self {
|
|
self.config.keyboard_interactivity = interactivity;
|
|
self
|
|
}
|
|
|
|
#[must_use]
|
|
pub const fn with_exclusive_zone(mut self, zone: i32) -> Self {
|
|
self.config.exclusive_zone = zone;
|
|
self
|
|
}
|
|
|
|
#[must_use]
|
|
pub fn with_namespace(mut self, namespace: String) -> Self {
|
|
self.config.namespace = namespace;
|
|
self
|
|
}
|
|
|
|
#[must_use]
|
|
pub const fn with_scale_factor(mut self, scale_factor: f32) -> Self {
|
|
self.config.scale_factor = scale_factor;
|
|
self
|
|
}
|
|
|
|
#[must_use]
|
|
pub fn with_component_definition(mut self, component: ComponentDefinition) -> Self {
|
|
self.config.component_definition = Some(component);
|
|
self
|
|
}
|
|
|
|
pub fn build(self) -> Result<WindowingSystem<'static>> {
|
|
if self.config.component_definition.is_none() {
|
|
return Err(anyhow::anyhow!("Slint component not set"));
|
|
}
|
|
|
|
WindowingSystem::new(self.config)
|
|
}
|
|
}
|
|
|
|
pub struct WindowingSystem<'a> {
|
|
state: Rc<RefCell<WindowState>>,
|
|
connection: Rc<Connection>,
|
|
event_queue: Rc<RefCell<EventQueue<WindowState>>>,
|
|
component_instance: Option<Rc<ComponentInstance>>,
|
|
display: WlDisplay,
|
|
config: WindowConfig,
|
|
event_loop: EventLoop<'a, ()>,
|
|
event_loop_handler: Option<EventLoopHandler>,
|
|
}
|
|
|
|
impl<'a> WindowingSystem<'a> {
|
|
fn new(config: WindowConfig) -> Result<Self> {
|
|
info!("Initializing WindowingSystem");
|
|
let connection = Rc::new(Connection::connect_to_env()?);
|
|
let state = Rc::new(RefCell::new(WindowState::new(&config)));
|
|
let display = connection.display();
|
|
let event_queue = Rc::new(RefCell::new(connection.new_event_queue()));
|
|
let global_list = Self::initialize_registry(&connection)?;
|
|
let (compositor, output, layer_shell, seat) =
|
|
Self::bind_globals(&global_list, &event_queue.borrow().handle())?;
|
|
|
|
Self::setup_surface(
|
|
&compositor,
|
|
&output,
|
|
&layer_shell,
|
|
&seat,
|
|
&event_queue.borrow().handle(),
|
|
&state,
|
|
&config,
|
|
);
|
|
|
|
let event_loop = EventLoop::try_new().context("Failed to create event loop")?;
|
|
|
|
let mut system = Self {
|
|
state,
|
|
connection,
|
|
event_queue,
|
|
component_instance: None,
|
|
display,
|
|
config,
|
|
event_loop,
|
|
event_loop_handler: None,
|
|
};
|
|
|
|
system.wait_for_configure()?;
|
|
system.initialize_renderer_and_ui()?;
|
|
|
|
Ok(system)
|
|
}
|
|
|
|
fn initialize_registry(connection: &Connection) -> Result<GlobalList> {
|
|
registry_queue_init::<WindowState>(connection)
|
|
.map(|(global_list, _)| global_list)
|
|
.context("Failed to initialize registry")
|
|
}
|
|
|
|
fn bind_globals(
|
|
global_list: &GlobalList,
|
|
queue_handle: &QueueHandle<WindowState>,
|
|
) -> Result<(WlCompositor, WlOutput, ZwlrLayerShellV1, WlSeat)> {
|
|
bind_globals!(
|
|
global_list,
|
|
queue_handle,
|
|
(WlCompositor, compositor, 1..=1),
|
|
(WlOutput, output, 1..=1),
|
|
(ZwlrLayerShellV1, layer_shell, 1..=1),
|
|
(WlSeat, seat, 1..=1)
|
|
)
|
|
}
|
|
|
|
fn setup_surface(
|
|
compositor: &WlCompositor,
|
|
output: &WlOutput,
|
|
layer_shell: &ZwlrLayerShellV1,
|
|
seat: &WlSeat,
|
|
queue_handle: &QueueHandle<WindowState>,
|
|
state: &Rc<RefCell<WindowState>>,
|
|
config: &WindowConfig,
|
|
) {
|
|
let surface = Rc::new(compositor.create_surface(queue_handle, ()));
|
|
let layer_surface = Rc::new(layer_shell.get_layer_surface(
|
|
&surface,
|
|
Some(output),
|
|
config.layer,
|
|
config.namespace.clone(),
|
|
queue_handle,
|
|
(),
|
|
));
|
|
|
|
let pointer = Rc::new(seat.get_pointer(queue_handle, ()));
|
|
|
|
let mut state = state.borrow_mut();
|
|
state.set_surface(Rc::clone(&surface));
|
|
state.set_layer_surface(Rc::clone(&layer_surface));
|
|
state.set_pointer(pointer);
|
|
|
|
Self::configure_layer_surface(&layer_surface, &surface, config);
|
|
}
|
|
|
|
fn configure_layer_surface(
|
|
layer_surface: &Rc<ZwlrLayerSurfaceV1>,
|
|
surface: &WlSurface,
|
|
config: &WindowConfig,
|
|
) {
|
|
layer_surface.set_anchor(config.anchor);
|
|
layer_surface.set_margin(
|
|
config.margin.0,
|
|
config.margin.1,
|
|
config.margin.2,
|
|
config.margin.3,
|
|
);
|
|
|
|
layer_surface.set_exclusive_zone(config.exclusive_zone);
|
|
layer_surface.set_keyboard_interactivity(config.keyboard_interactivity);
|
|
layer_surface.set_size(1, config.height);
|
|
surface.commit();
|
|
}
|
|
|
|
fn wait_for_configure(&self) -> Result<()> {
|
|
info!("Waiting for surface to be configured...");
|
|
let mut state = self.state.borrow_mut();
|
|
self.event_queue
|
|
.borrow_mut()
|
|
.blocking_dispatch(&mut state)
|
|
.context("Failed to dispatch events")?;
|
|
info!("Blocking dispatch completed");
|
|
let size = state.output_size();
|
|
if size.width > 1 && size.height > 1 {
|
|
info!("Configured output size: {:?}", size);
|
|
} else {
|
|
return Err(anyhow::anyhow!("Invalid output size: {:?}", size));
|
|
}
|
|
debug!("Surface configuration complete");
|
|
Ok(())
|
|
}
|
|
|
|
fn initialize_renderer_and_ui(&mut self) -> Result<()> {
|
|
let renderer = self.create_renderer()?;
|
|
let component_definition = self
|
|
.config
|
|
.component_definition
|
|
.as_ref()
|
|
.ok_or_else(|| anyhow::anyhow!("Component definition not set"))?;
|
|
let (window, component_instance) =
|
|
self.initialize_slint_ui(renderer, component_definition)?;
|
|
|
|
self.state.borrow_mut().set_window(window);
|
|
self.component_instance = Some(component_instance);
|
|
|
|
Ok(())
|
|
}
|
|
|
|
fn create_renderer(&self) -> Result<FemtoVGRenderer> {
|
|
let state_borrow = self.state.borrow();
|
|
let size = state_borrow.size();
|
|
let surface = state_borrow.surface().unwrap();
|
|
|
|
debug!("Creating EGL context with size: {:?}", size);
|
|
let context = EGLContext::builder()
|
|
.with_display_id(self.display.id())
|
|
.with_surface_id(surface.id())
|
|
.with_size(size)
|
|
.build()
|
|
.map_err(|e| anyhow::anyhow!("Failed to create EGL context: {:?}", e))?;
|
|
|
|
debug!("Creating FemtoVGRenderer");
|
|
FemtoVGRenderer::new(context).context("Failed to create FemtoVGRenderer")
|
|
}
|
|
|
|
fn initialize_slint_ui(
|
|
&self,
|
|
renderer: FemtoVGRenderer,
|
|
component_definition: &ComponentDefinition,
|
|
) -> Result<(Rc<FemtoVGWindow>, Rc<ComponentInstance>)> {
|
|
let femtovg_window = FemtoVGWindow::new(renderer);
|
|
let size = self.state.borrow().size();
|
|
info!("Initializing UI with size: {:?}", size);
|
|
femtovg_window.set_size(slint::WindowSize::Physical(size));
|
|
femtovg_window.set_scale_factor(self.config.scale_factor);
|
|
femtovg_window.set_position(LogicalPosition::new(0., 0.));
|
|
|
|
debug!("Setting up custom Slint platform");
|
|
let platform = CustomSlintPlatform::new(&femtovg_window);
|
|
slint::platform::set_platform(Box::new(platform))
|
|
.map_err(|e| anyhow::anyhow!("Failed to set platform: {:?}", e))?;
|
|
|
|
debug!("Creating Slint component instance");
|
|
let slint_component: Rc<ComponentInstance> = Rc::new(component_definition.create()?);
|
|
|
|
slint_component
|
|
.show()
|
|
.map_err(|e| anyhow::anyhow!("Failed to show component: {:?}", e))?;
|
|
|
|
Ok((femtovg_window, slint_component))
|
|
}
|
|
|
|
pub fn initialize_event_loop_handler(&mut self) {
|
|
let event_loop_handler = EventLoopHandler::new(
|
|
Rc::downgrade(self.state.borrow().window().as_ref().unwrap()),
|
|
Rc::downgrade(&self.event_queue),
|
|
Rc::downgrade(&self.connection),
|
|
Rc::downgrade(&self.state),
|
|
);
|
|
|
|
self.event_loop_handler = Some(event_loop_handler);
|
|
}
|
|
|
|
pub fn setup_event_sources(&self) -> Result<()> {
|
|
let loop_handle = self.event_loop.handle();
|
|
let event_loop_handler = self
|
|
.event_loop_handler
|
|
.as_ref()
|
|
.ok_or_else(|| anyhow::anyhow!("EventLoopHandler not initialized"))?;
|
|
|
|
event_loop_handler.setup_wayland_event_source(&loop_handle)?;
|
|
|
|
Ok(())
|
|
}
|
|
|
|
pub fn event_loop_handle(&self) -> calloop::LoopHandle<'a, ()> {
|
|
self.event_loop.handle()
|
|
}
|
|
|
|
pub fn run(&mut self) -> Result<()> {
|
|
info!("Starting WindowingSystem main loop");
|
|
self.initialize_event_loop_handler();
|
|
self.setup_event_sources()?;
|
|
if let Some(window) = &self.state.borrow().window() {
|
|
window.render_frame_if_dirty();
|
|
}
|
|
|
|
self.event_loop
|
|
.run(None, &mut (), |()| {})
|
|
.map_err(|e| anyhow::anyhow!("Failed to run event loop: {}", e))
|
|
}
|
|
|
|
pub fn component_instance(&self) -> Rc<ComponentInstance> {
|
|
Rc::clone(self.component_instance.as_ref().unwrap())
|
|
}
|
|
|
|
pub fn window(&self) -> Rc<FemtoVGWindow> {
|
|
Rc::clone(self.state().window().as_ref().unwrap())
|
|
}
|
|
|
|
pub fn state(&self) -> Ref<WindowState> {
|
|
self.state.borrow()
|
|
}
|
|
|
|
pub const fn display(&self) -> &WlDisplay {
|
|
&self.display
|
|
}
|
|
}
|