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 config::WindowConfig; use log::{debug, info}; use slint::{platform::femtovg_renderer::FemtoVGRenderer, ComponentHandle, LogicalPosition}; use slint_interpreter::ComponentInstance; use smithay_client_toolkit::reexports::{ calloop::{EventLoop, LoopHandle}, protocols_wlr::layer_shell::v1::client::{ zwlr_layer_shell_v1::ZwlrLayerShellV1, zwlr_layer_surface_v1::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, }; pub mod builder; mod config; mod event_loop; mod macros; mod state; pub struct WindowingSystem { state: Rc>, connection: Rc, event_queue: Rc>>, component_instance: Rc, event_loop: EventLoop<'static, ()>, event_loop_handler: EventLoopHandler, } impl WindowingSystem { fn new(config: &WindowConfig) -> Result { 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, ); Self::wait_for_configure(&event_queue, &state)?; let component_instance = Self::initialize_renderer_and_ui(&state, &display, config)?; let event_loop = EventLoop::try_new().context("Failed to create event loop")?; let event_loop_handler = EventLoopHandler::new( Rc::clone(&event_queue), Rc::clone(&connection), Rc::clone(&state), ); Ok(Self { state, connection, event_queue, component_instance, event_loop, event_loop_handler, }) } fn initialize_registry(connection: &Connection) -> Result { registry_queue_init::(connection) .map(|(global_list, _)| global_list) .context("Failed to initialize registry") } fn bind_globals( global_list: &GlobalList, queue_handle: &QueueHandle, ) -> 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, state: &Rc>, 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, 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( event_queue: &Rc>>, state: &Rc>, ) -> Result<()> { info!("Waiting for surface to be configured..."); let mut state = state.borrow_mut(); 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 create_renderer( state: &Rc>, display: &WlDisplay, ) -> Result { let state_borrow = 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(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_renderer_and_ui( state: &Rc>, display: &WlDisplay, config: &WindowConfig, ) -> Result> { let renderer = Self::create_renderer(state, display)?; let component_definition = config .component_definition .as_ref() .ok_or_else(|| anyhow::anyhow!("Component definition not set"))?; let femtovg_window = FemtoVGWindow::new(renderer); let size = state.borrow().size(); info!("Initializing UI with size: {:?}", size); femtovg_window.set_size(slint::WindowSize::Physical(size)); femtovg_window.set_scale_factor(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 = Rc::new(component_definition.create()?); slint_component .show() .map_err(|e| anyhow::anyhow!("Failed to show component: {:?}", e))?; state.borrow_mut().set_window(Rc::clone(&femtovg_window)); Ok(slint_component) } pub fn initialize_event_loop_handler(&mut self) { let event_loop_handler = EventLoopHandler::new( Rc::clone(&self.event_queue), Rc::clone(&self.connection), Rc::clone(&self.state), ); self.event_loop_handler = event_loop_handler; } pub fn setup_event_sources(&self) -> Result<()> { let loop_handle = self.event_loop.handle(); self.event_loop_handler .setup_wayland_event_source(&loop_handle)?; Ok(()) } pub fn event_loop_handle(&self) -> LoopHandle<'static, ()> { 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 { Rc::clone(&self.component_instance) } pub fn window(&self) -> Rc { Rc::clone(self.state().window().as_ref().unwrap()) } pub fn state(&self) -> Ref { self.state.borrow() } }