use crate::{
ActiveTheme, Collapsible, Icon, IconName, Side, Sizable, StyledExt,
button::{Button, ButtonVariants},
h_flex,
scroll::ScrollableElement,
v_flex,
};
use gpui::{
AnyElement, App, ClickEvent, EdgesRefinement, InteractiveElement as _, IntoElement,
ParentElement, Pixels, RenderOnce, StyleRefinement, Styled, Window, div,
prelude::FluentBuilder, px,
};
use std::rc::Rc;
mod footer;
mod group;
mod header;
mod menu;
pub use footer::*;
pub use group::*;
pub use header::*;
pub use menu::*;
const DEFAULT_WIDTH: Pixels = px(255.);
const COLLAPSED_WIDTH: Pixels = px(48.);
#[derive(IntoElement)]
pub struct Sidebar<E: Collapsible + IntoElement + 'static> {
style: StyleRefinement,
content: Vec<E>,
header: Option<AnyElement>,
footer: Option<AnyElement>,
side: Side,
collapsible: bool,
collapsed: bool,
}
impl<E: Collapsible + IntoElement> Sidebar<E> {
pub fn new(side: Side) -> Self {
Self {
style: StyleRefinement::default(),
content: vec![],
header: None,
footer: None,
side,
collapsible: true,
collapsed: false,
}
}
pub fn left() -> Self {
Self::new(Side::Left)
}
pub fn right() -> Self {
Self::new(Side::Right)
}
pub fn collapsible(mut self, collapsible: bool) -> Self {
self.collapsible = collapsible;
self
}
pub fn collapsed(mut self, collapsed: bool) -> Self {
self.collapsed = collapsed;
self
}
pub fn header(mut self, header: impl IntoElement) -> Self {
self.header = Some(header.into_any_element());
self
}
pub fn footer(mut self, footer: impl IntoElement) -> Self {
self.footer = Some(footer.into_any_element());
self
}
pub fn child(mut self, child: E) -> Self {
self.content.push(child);
self
}
pub fn children(mut self, children: impl IntoIterator<Item = E>) -> Self {
self.content.extend(children);
self
}
}
#[derive(IntoElement)]
pub struct SidebarToggleButton {
btn: Button,
collapsed: bool,
side: Side,
on_click: Option<Rc<dyn Fn(&ClickEvent, &mut Window, &mut App)>>,
}
impl SidebarToggleButton {
fn new(side: Side) -> Self {
Self {
btn: Button::new("collapse").ghost().small(),
collapsed: false,
side,
on_click: None,
}
}
pub fn left() -> Self {
Self::new(Side::Left)
}
pub fn right() -> Self {
Self::new(Side::Right)
}
pub fn side(mut self, side: Side) -> Self {
self.side = side;
self
}
pub fn collapsed(mut self, collapsed: bool) -> Self {
self.collapsed = collapsed;
self
}
pub fn on_click(
mut self,
on_click: impl Fn(&ClickEvent, &mut Window, &mut App) + 'static,
) -> Self {
self.on_click = Some(Rc::new(on_click));
self
}
}
impl RenderOnce for SidebarToggleButton {
fn render(self, _window: &mut Window, _cx: &mut App) -> impl IntoElement {
let collapsed = self.collapsed;
let on_click = self.on_click.clone();
let icon = if collapsed {
if self.side.is_left() {
IconName::PanelLeftOpen
} else {
IconName::PanelRightOpen
}
} else {
if self.side.is_left() {
IconName::PanelLeftClose
} else {
IconName::PanelRightClose
}
};
self.btn
.when_some(on_click, |this, on_click| {
this.on_click(move |ev, window, cx| {
on_click(ev, window, cx);
})
})
.icon(Icon::new(icon).size_4())
}
}
impl<E: Collapsible + IntoElement> Styled for Sidebar<E> {
fn style(&mut self) -> &mut StyleRefinement {
&mut self.style
}
}
impl<E: Collapsible + IntoElement> RenderOnce for Sidebar<E> {
fn render(mut self, _: &mut Window, cx: &mut App) -> impl IntoElement {
self.style.padding = EdgesRefinement::default();
v_flex()
.id("sidebar")
.w(DEFAULT_WIDTH)
.flex_shrink_0()
.h_full()
.overflow_hidden()
.relative()
.bg(cx.theme().sidebar)
.text_color(cx.theme().sidebar_foreground)
.border_color(cx.theme().sidebar_border)
.map(|this| match self.side {
Side::Left => this.border_r_1(),
Side::Right => this.border_l_1(),
})
.refine_style(&self.style)
.when(self.collapsed, |this| this.w(COLLAPSED_WIDTH).gap_2())
.when_some(self.header.take(), |this, header| {
this.child(
h_flex()
.id("header")
.pt_3()
.px_3()
.gap_2()
.when(self.collapsed, |this| this.pt_2().px_2())
.child(header),
)
})
.child(
v_flex().id("content").flex_1().min_h_0().child(
v_flex()
.id("inner")
.p_3()
.when(self.collapsed, |this| this.p_2())
.children(
self.content.into_iter().enumerate().map(|(ix, c)| {
div().id(ix).mt_3().child(c.collapsed(self.collapsed))
}),
)
.overflow_y_scrollbar(),
),
)
.when_some(self.footer.take(), |this, footer| {
this.child(
h_flex()
.id("footer")
.pb_3()
.px_3()
.gap_2()
.when(self.collapsed, |this| this.pt_2().px_2())
.child(footer),
)
})
}
}