281 lines
9.0 KiB
Rust
281 lines
9.0 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 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<RefCell<WindowState>>,
|
|
connection: Rc<Connection>,
|
|
event_queue: Rc<RefCell<EventQueue<WindowState>>>,
|
|
component_instance: Rc<ComponentInstance>,
|
|
event_loop: EventLoop<'static, ()>,
|
|
event_loop_handler: EventLoopHandler,
|
|
}
|
|
|
|
impl WindowingSystem {
|
|
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,
|
|
);
|
|
|
|
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<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(
|
|
event_queue: &Rc<RefCell<EventQueue<WindowState>>>,
|
|
state: &Rc<RefCell<WindowState>>,
|
|
) -> 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<RefCell<WindowState>>,
|
|
display: &WlDisplay,
|
|
) -> Result<FemtoVGRenderer> {
|
|
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<RefCell<WindowState>>,
|
|
display: &WlDisplay,
|
|
config: &WindowConfig,
|
|
) -> Result<Rc<ComponentInstance>> {
|
|
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<ComponentInstance> = 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<ComponentInstance> {
|
|
Rc::clone(&self.component_instance)
|
|
}
|
|
|
|
pub fn window(&self) -> Rc<FemtoVGWindow> {
|
|
Rc::clone(self.state().window().as_ref().unwrap())
|
|
}
|
|
|
|
pub fn state(&self) -> Ref<WindowState> {
|
|
self.state.borrow()
|
|
}
|
|
}
|