use std::rc::{Rc, Weak};
use gpui::{
div, prelude::FluentBuilder as _, px, AlignItems, AnyElement, AnyView, App, Axis, Div, Element,
ElementId, FocusHandle, InteractiveElement as _, IntoElement, ParentElement, Pixels, Rems,
RenderOnce, SharedString, Styled, Window,
};
use crate::{h_flex, v_flex, ActiveTheme as _, AxisExt, Sizable, Size, StyledExt};
pub fn v_form() -> Form {
Form::vertical()
}
pub fn h_form() -> Form {
Form::horizontal()
}
pub fn form_field() -> FormField {
FormField::new()
}
#[derive(IntoElement)]
pub struct Form {
fields: Vec<FormField>,
props: FieldProps,
}
#[derive(Clone, Copy)]
struct FieldProps {
size: Size,
label_width: Option<Pixels>,
label_text_size: Option<Rems>,
layout: Axis,
gap: Option<Pixels>,
column: u16,
}
impl Default for FieldProps {
fn default() -> Self {
Self {
label_width: Some(px(140.)),
label_text_size: None,
layout: Axis::Vertical,
size: Size::default(),
gap: None,
column: 1,
}
}
}
impl Form {
fn new() -> Self {
Self {
props: FieldProps::default(),
fields: Vec::new(),
}
}
pub fn horizontal() -> Self {
Self::new().layout(Axis::Horizontal)
}
pub fn vertical() -> Self {
Self::new().layout(Axis::Vertical)
}
pub fn layout(mut self, layout: Axis) -> Self {
self.props.layout = layout;
self
}
pub fn label_width(mut self, width: Pixels) -> Self {
self.props.label_width = Some(width);
self
}
pub fn label_text_size(mut self, size: Rems) -> Self {
self.props.label_text_size = Some(size);
self
}
pub fn gap(mut self, gap: Pixels) -> Self {
self.props.gap = Some(gap);
self
}
pub fn child(mut self, field: impl Into<FormField>) -> Self {
self.fields.push(field.into());
self
}
pub fn children(mut self, fields: impl IntoIterator<Item = FormField>) -> Self {
self.fields.extend(fields);
self
}
pub fn column(mut self, column: u16) -> Self {
self.props.column = column;
self
}
}
impl Sizable for Form {
fn with_size(mut self, size: impl Into<Size>) -> Self {
self.props.size = size.into();
self
}
}
pub enum FieldBuilder {
String(SharedString),
Element(Rc<dyn Fn(&mut Window, &mut App) -> AnyElement>),
View(AnyView),
}
impl Default for FieldBuilder {
fn default() -> Self {
Self::String(SharedString::default())
}
}
impl From<AnyView> for FieldBuilder {
fn from(view: AnyView) -> Self {
Self::View(view)
}
}
impl RenderOnce for FieldBuilder {
fn render(self, window: &mut Window, cx: &mut App) -> impl IntoElement {
match self {
FieldBuilder::String(value) => value.into_any_element(),
FieldBuilder::Element(builder) => builder(window, cx),
FieldBuilder::View(view) => view.into_any(),
}
}
}
impl From<&'static str> for FieldBuilder {
fn from(value: &'static str) -> Self {
Self::String(value.into())
}
}
impl From<String> for FieldBuilder {
fn from(value: String) -> Self {
Self::String(value.into())
}
}
impl From<SharedString> for FieldBuilder {
fn from(value: SharedString) -> Self {
Self::String(value)
}
}
#[derive(IntoElement)]
pub struct FormField {
id: ElementId,
form: Weak<Form>,
label: Option<FieldBuilder>,
no_label_indent: bool,
focus_handle: Option<FocusHandle>,
description: Option<FieldBuilder>,
child: Div,
visible: bool,
required: bool,
align_items: Option<AlignItems>,
props: FieldProps,
col_span: u16,
col_start: Option<i16>,
col_end: Option<i16>,
}
impl FormField {
pub fn new() -> Self {
Self {
id: 0.into(),
form: Weak::new(),
label: None,
description: None,
child: div(),
visible: true,
required: false,
no_label_indent: false,
focus_handle: None,
align_items: None,
props: FieldProps::default(),
col_span: 1,
col_start: None,
col_end: None,
}
}
pub fn label(mut self, label: impl Into<FieldBuilder>) -> Self {
self.label = Some(label.into());
self
}
pub fn no_label_indent(mut self) -> Self {
self.no_label_indent = true;
self
}
pub fn label_fn<F, E>(mut self, label: F) -> Self
where
E: IntoElement,
F: Fn(&mut Window, &mut App) -> E + 'static,
{
self.label = Some(FieldBuilder::Element(Rc::new(move |window, cx| {
label(window, cx).into_any_element()
})));
self
}
pub fn description(mut self, description: impl Into<FieldBuilder>) -> Self {
self.description = Some(description.into());
self
}
pub fn description_fn<F, E>(mut self, description: F) -> Self
where
E: IntoElement,
F: Fn(&mut Window, &mut App) -> E + 'static,
{
self.description = Some(FieldBuilder::Element(Rc::new(move |window, cx| {
description(window, cx).into_any_element()
})));
self
}
pub fn visible(mut self, visible: bool) -> Self {
self.visible = visible;
self
}
pub fn required(mut self, required: bool) -> Self {
self.required = required;
self
}
pub fn track_focus(mut self, focus_handle: &FocusHandle) -> Self {
self.focus_handle = Some(focus_handle.clone());
self
}
pub fn parent(mut self, form: &Rc<Form>) -> Self {
self.form = Rc::downgrade(form);
self
}
fn props(mut self, ix: usize, props: FieldProps) -> Self {
self.id = ix.into();
self.props = props;
self
}
pub fn items_start(mut self) -> Self {
self.align_items = Some(AlignItems::Start);
self
}
pub fn items_end(mut self) -> Self {
self.align_items = Some(AlignItems::End);
self
}
pub fn items_center(mut self) -> Self {
self.align_items = Some(AlignItems::Center);
self
}
pub fn col_span(mut self, col_span: u16) -> Self {
self.col_span = col_span;
self
}
pub fn col_start(mut self, col_start: i16) -> Self {
self.col_start = Some(col_start);
self
}
pub fn col_end(mut self, col_end: i16) -> Self {
self.col_end = Some(col_end);
self
}
}
impl ParentElement for FormField {
fn extend(&mut self, elements: impl IntoIterator<Item = AnyElement>) {
self.child.extend(elements);
}
}
impl RenderOnce for FormField {
fn render(self, window: &mut Window, cx: &mut App) -> impl IntoElement {
let layout = self.props.layout;
let label_width = if layout.is_vertical() {
None
} else {
self.props.label_width
};
let has_label = !self.no_label_indent;
#[inline]
fn wrap_div(layout: Axis) -> Div {
if layout.is_vertical() {
v_flex()
} else {
h_flex()
}
}
#[inline]
fn wrap_label(label_width: Option<Pixels>) -> Div {
div().when_some(label_width, |this, width| this.w(width).flex_shrink_0())
}
let gap = match self.props.gap {
Some(v) => v,
None => match self.props.size {
Size::Large => px(8.),
Size::XSmall | Size::Small => px(4.),
_ => px(4.),
},
};
let inner_gap = if layout.is_horizontal() {
gap
} else {
gap / 2.
};
v_flex()
.flex_1()
.gap(gap / 2.)
.col_span(self.col_span)
.when_some(self.col_start, |this, start| this.col_start(start))
.when_some(self.col_end, |this, end| this.col_end(end))
.child(
wrap_div(layout)
.id(self.id)
.gap(inner_gap)
.when_some(self.align_items, |this, align| {
this.map(|this| match align {
AlignItems::Start => this.items_start(),
AlignItems::End => this.items_end(),
AlignItems::Center => this.items_center(),
AlignItems::Baseline => this.items_baseline(),
_ => this,
})
})
.when(has_label, |this| {
this.child(
wrap_label(label_width)
.text_sm()
.when_some(self.props.label_text_size, |this, size| {
this.text_size(size)
})
.font_medium()
.flex()
.flex_row()
.gap_1()
.items_center()
.when_some(self.label, |this, builder| {
this.child(
h_flex()
.gap_1()
.child(
div()
.overflow_x_hidden()
.child(builder.render(window, cx)),
)
.when(self.required, |this| {
this.child(
div().text_color(cx.theme().danger).child("*"),
)
}),
)
}),
)
})
.child(
div()
.w_full()
.flex_1()
.overflow_x_hidden()
.child(self.child),
),
)
.child(
wrap_div(layout)
.gap(inner_gap)
.when(has_label && layout.is_horizontal(), |this| {
this.child(
wrap_label(label_width),
)
})
.when_some(self.description, |this, builder| {
this.child(
div()
.text_xs()
.text_color(cx.theme().muted_foreground)
.child(builder.render(window, cx)),
)
}),
)
}
}
impl RenderOnce for Form {
fn render(self, _window: &mut Window, _cx: &mut App) -> impl IntoElement {
let props = self.props;
let gap = match props.size {
Size::XSmall | Size::Small => px(6.),
Size::Large => px(12.),
_ => px(8.),
};
v_flex()
.w_full()
.gap_x(gap * 3.)
.gap_y(gap)
.grid()
.grid_cols(props.column)
.children(
self.fields
.into_iter()
.enumerate()
.map(|(ix, field)| field.props(ix, props)),
)
}
}