libcosmic/widgets/src/labeled_item/imp.rs

168 lines
4.8 KiB
Rust

use gtk4::{prelude::*, Align, Label, Orientation, Widget};
use relm4::{ComponentParts, Sender, SimpleComponent};
use std::cell::RefCell;
#[derive(Debug)]
pub(crate) enum LabeledItemMessage {
Title(String),
Desc(Option<String>),
Align(Align),
Child(Widget),
}
#[track]
pub(crate) struct LabeledItem {
_title: String,
_desc: Option<String>,
_align: Align,
_child: Option<Widget>,
#[do_not_track]
_remove_child: RefCell<Option<Widget>>,
#[do_not_track]
_sender: Sender<LabeledItemMessage>,
}
impl LabeledItem {
pub fn title(&self) -> &str {
&self._title
}
pub fn description(&self) -> Option<&String> {
self._desc.as_ref()
}
pub fn alignment(&self) -> Align {
self._align
}
pub fn child(&self) -> Option<&Widget> {
self._child.as_ref()
}
pub fn set_title<S>(&self, title: S)
where
S: ToString,
{
self._sender
.send(LabeledItemMessage::Title(title.to_string()));
}
pub fn set_description<S>(&self, title: Option<S>)
where
S: ToString,
{
self._sender
.send(LabeledItemMessage::Desc(title.map(|s| s.to_string())));
}
pub fn set_alignment(&self, align: Align) {
self._sender.send(LabeledItemMessage::Align(align));
}
pub fn set_child(&self, child: Widget) {
self._sender.send(LabeledItemMessage::Child(child));
}
}
#[component(pub(crate))]
impl SimpleComponent for LabeledItem {
type Widgets = AppWidgets;
type InitParams = ();
type Input = LabeledItemMessage;
type Output = ();
view! {
base_box = gtk4::Box {
add_css_class: "labeled-item",
set_orientation: Orientation::Horizontal,
set_hexpand: true,
set_margin_start: 24,
set_margin_end: 24,
set_margin_top: 8,
set_margin_bottom: 8,
set_spacing: 16,
append: labeled_item_info = &gtk4::Box {
add_css_class: "labeled-item-info",
set_orientation: Orientation::Vertical,
set_hexpand: true,
set_spacing: 8,
set_valign: Align::Center,
&Label {
add_css_class: "labeled-item-title",
set_halign: Align::Start,
set_label: watch! { &model._title }
},
&Label {
add_css_class: "labeled-item-desc",
set_halign: Align::Start,
set_visible: watch! { model._desc.is_some() },
set_label: watch! { &model._desc.clone().unwrap_or_default() }
},
}
}
}
fn init_parts(
_init_params: Self::InitParams,
root: &Self::Root,
input: &Sender<Self::Input>,
_output: &Sender<Self::Output>,
) -> ComponentParts<Self> {
let model = LabeledItem {
_title: String::default(),
_desc: None,
_align: Align::Start,
_child: None,
_remove_child: RefCell::new(None),
_sender: input.clone(),
tracker: 0,
};
let widgets = view_output!();
ComponentParts { model, widgets }
}
fn update(
&mut self,
msg: Self::Input,
_input: &Sender<Self::Input>,
_ouput: &Sender<Self::Output>,
) {
self.reset();
match msg {
LabeledItemMessage::Title(title) => self.set__title(title),
LabeledItemMessage::Desc(desc) => self.set__desc(desc),
LabeledItemMessage::Align(align) => self.set__align(align),
LabeledItemMessage::Child(child) => {
*self._remove_child.borrow_mut() = self._child.take();
self.set__child(Some(child))
}
}
}
fn post_view() {
if let Some(child) = self._remove_child.borrow_mut().take() {
widgets.base_box.remove(&child);
}
if self.changed(LabeledItem::_child()) {
let child = self._child.as_ref().expect("there's no child??");
widgets.base_box.append(child);
}
if self.changed(LabeledItem::_align()) {
let child = self._child.as_ref().expect("set alignment without child");
match self._align {
Align::Start => {
widgets
.base_box
.reorder_child_after(&widgets.labeled_item_info, Some(child));
}
Align::End => {
widgets
.base_box
.reorder_child_after(child, Some(&widgets.labeled_item_info));
}
_ => unimplemented!(),
}
}
}
}