167 lines
4.5 KiB
Rust
167 lines
4.5 KiB
Rust
use {
|
|
crate::{
|
|
client::Client,
|
|
cursor::KnownCursor,
|
|
fixed::Fixed,
|
|
ifs::wl_seat::{NodeSeatState, WlSeatGlobal},
|
|
rect::Rect,
|
|
render::{Renderer, Texture},
|
|
state::State,
|
|
text,
|
|
tree::{FindTreeResult, FoundNode, Node, NodeId, NodeVisitor, ToplevelData, ToplevelNode},
|
|
utils::{clonecell::CloneCell, errorfmt::ErrorFmt},
|
|
},
|
|
jay_config::Direction,
|
|
std::{cell::Cell, ops::Deref, rc::Rc},
|
|
};
|
|
|
|
tree_id!(PlaceholderNodeId);
|
|
|
|
pub struct PlaceholderNode {
|
|
id: PlaceholderNodeId,
|
|
toplevel: ToplevelData,
|
|
destroyed: Cell<bool>,
|
|
pub texture: CloneCell<Option<Rc<Texture>>>,
|
|
}
|
|
|
|
impl PlaceholderNode {
|
|
pub fn new_for(state: &Rc<State>, node: Rc<dyn ToplevelNode>) -> Self {
|
|
Self {
|
|
id: state.node_ids.next(),
|
|
toplevel: ToplevelData::new(
|
|
state,
|
|
node.tl_data().title.borrow_mut().clone(),
|
|
node.node_client(),
|
|
),
|
|
destroyed: Default::default(),
|
|
texture: Default::default(),
|
|
}
|
|
}
|
|
|
|
pub fn is_destroyed(&self) -> bool {
|
|
self.destroyed.get()
|
|
}
|
|
|
|
pub fn update_texture(&self) {
|
|
self.texture.set(None);
|
|
if let Some(ctx) = self.toplevel.state.render_ctx.get() {
|
|
let rect = self.toplevel.pos.get();
|
|
if rect.width() != 0 && rect.height() != 0 {
|
|
let font = format!("monospace {}", rect.width() / 10);
|
|
match text::render_fitting(
|
|
&ctx,
|
|
rect.height(),
|
|
&font,
|
|
"Fullscreen",
|
|
self.toplevel.state.theme.colors.unfocused_title_text.get(),
|
|
false,
|
|
) {
|
|
Ok(t) => {
|
|
self.texture.set(Some(t));
|
|
}
|
|
Err(e) => {
|
|
log::warn!("Could not render fullscreen texture: {}", ErrorFmt(e));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
impl Node for PlaceholderNode {
|
|
fn node_id(&self) -> NodeId {
|
|
self.id.into()
|
|
}
|
|
|
|
fn node_seat_state(&self) -> &NodeSeatState {
|
|
&self.toplevel.seat_state
|
|
}
|
|
|
|
fn node_visit(self: Rc<Self>, visitor: &mut dyn NodeVisitor) {
|
|
visitor.visit_placeholder(&self);
|
|
}
|
|
|
|
fn node_visit_children(&self, _visitor: &mut dyn NodeVisitor) {
|
|
// nothing
|
|
}
|
|
|
|
fn node_visible(&self) -> bool {
|
|
self.toplevel.visible.get()
|
|
}
|
|
|
|
fn node_absolute_position(&self) -> Rect {
|
|
self.toplevel.pos.get()
|
|
}
|
|
|
|
fn node_do_focus(self: Rc<Self>, seat: &Rc<WlSeatGlobal>, _direction: Direction) {
|
|
seat.focus_toplevel(self.clone());
|
|
}
|
|
|
|
fn node_active_changed(&self, active: bool) {
|
|
self.toplevel.active.set(active);
|
|
if let Some(parent) = self.toplevel.parent.get() {
|
|
parent.node_child_active_changed(self, active, 1);
|
|
}
|
|
}
|
|
|
|
fn node_find_tree_at(&self, _x: i32, _y: i32, _tree: &mut Vec<FoundNode>) -> FindTreeResult {
|
|
FindTreeResult::AcceptsInput
|
|
}
|
|
|
|
fn node_render(&self, renderer: &mut Renderer, x: i32, y: i32) {
|
|
renderer.render_placeholder(self, x, y);
|
|
}
|
|
|
|
fn node_client(&self) -> Option<Rc<Client>> {
|
|
self.toplevel.client.clone()
|
|
}
|
|
|
|
fn node_toplevel(self: Rc<Self>) -> Option<Rc<dyn ToplevelNode>> {
|
|
Some(self)
|
|
}
|
|
|
|
fn node_on_pointer_enter(self: Rc<Self>, seat: &Rc<WlSeatGlobal>, _x: Fixed, _y: Fixed) {
|
|
seat.set_known_cursor(KnownCursor::Default);
|
|
seat.enter_toplevel(self.clone());
|
|
}
|
|
|
|
fn node_is_placeholder(&self) -> bool {
|
|
true
|
|
}
|
|
}
|
|
|
|
impl ToplevelNode for PlaceholderNode {
|
|
tl_node_impl!();
|
|
|
|
fn tl_data(&self) -> &ToplevelData {
|
|
&self.toplevel
|
|
}
|
|
|
|
fn tl_default_focus_child(&self) -> Option<Rc<dyn Node>> {
|
|
None
|
|
}
|
|
|
|
fn tl_change_extents(self: Rc<Self>, rect: &Rect) {
|
|
self.toplevel.pos.set(*rect);
|
|
if let Some(p) = self.toplevel.parent.get() {
|
|
p.node_child_size_changed(self.deref(), rect.width(), rect.height());
|
|
}
|
|
self.update_texture();
|
|
}
|
|
|
|
fn tl_close(self: Rc<Self>) {
|
|
let slf = self.clone();
|
|
self.toplevel.state.run_toplevel.schedule(move || {
|
|
slf.tl_destroy();
|
|
});
|
|
}
|
|
|
|
fn tl_set_visible(&self, visible: bool) {
|
|
self.toplevel.visible.set(visible);
|
|
}
|
|
|
|
fn tl_destroy(&self) {
|
|
self.toplevel.destroy_node(self);
|
|
self.destroyed.set(true);
|
|
}
|
|
}
|