Finish building out adapters and names

Document core traits
Add start for a component storybook
This commit is contained in:
Mikayla 2023-08-19 14:40:05 -07:00
parent bd3ab82dac
commit e946b0a2ec
No known key found for this signature in database
8 changed files with 213 additions and 153 deletions

11
Cargo.lock generated
View file

@ -1562,6 +1562,17 @@ dependencies = [
"workspace", "workspace",
] ]
[[package]]
name = "component_test"
version = "0.1.0"
dependencies = [
"gpui",
"settings",
"theme",
"util",
"workspace",
]
[[package]] [[package]]
name = "concurrent-queue" name = "concurrent-queue"
version = "2.2.0" version = "2.2.0"

View file

@ -13,6 +13,7 @@ members = [
"crates/collab_ui", "crates/collab_ui",
"crates/collections", "crates/collections",
"crates/command_palette", "crates/command_palette",
"crates/component_test",
"crates/context_menu", "crates/context_menu",
"crates/copilot", "crates/copilot",
"crates/copilot_button", "crates/copilot_button",

View file

@ -17,8 +17,8 @@ use gpui::{
actions, actions,
elements::{ elements::{
Canvas, ChildView, Component, Empty, Flex, Image, Label, List, ListOffset, ListState, Canvas, ChildView, Component, Empty, Flex, Image, Label, List, ListOffset, ListState,
MouseEventHandler, Orientation, OverlayPositionMode, Padding, ParentElement, Stack, MouseEventHandler, Orientation, OverlayPositionMode, Padding, ParentElement, SafeStylable,
StyleableComponent, Svg, Stack, Svg,
}, },
geometry::{ geometry::{
rect::RectF, rect::RectF,
@ -1633,11 +1633,8 @@ impl CollabPanel {
}) })
.align_children_center() .align_children_center()
.styleable_component() .styleable_component()
.disclosable( .disclosable(disclosed, Box::new(ToggleCollapsed { channel_id }))
disclosed, .with_id(channel_id as usize)
Box::new(ToggleCollapsed { channel_id }),
channel_id as usize,
)
.with_style(theme.disclosure.clone()) .with_style(theme.disclosure.clone())
.element() .element()
.constrained() .constrained()

View file

@ -72,7 +72,7 @@ impl View for TestView {
TextStyle::for_color(Color::blue()), TextStyle::for_color(Color::blue()),
) )
.with_style(ButtonStyle::fill(Color::yellow())) .with_style(ButtonStyle::fill(Color::yellow()))
.stateful_element(), .element(),
) )
.with_child( .with_child(
ToggleableButton::new(self.is_doubling, move |_, v: &mut Self, cx| { ToggleableButton::new(self.is_doubling, move |_, v: &mut Self, cx| {
@ -84,7 +84,7 @@ impl View for TestView {
inactive: ButtonStyle::fill(Color::red()), inactive: ButtonStyle::fill(Color::red()),
active: ButtonStyle::fill(Color::green()), active: ButtonStyle::fill(Color::green()),
}) })
.stateful_element(), .element(),
) )
.expanded() .expanded()
.contained() .contained()

View file

@ -230,25 +230,25 @@ pub trait Element<V: View>: 'static {
MouseEventHandler::for_child::<Tag>(self.into_any(), region_id) MouseEventHandler::for_child::<Tag>(self.into_any(), region_id)
} }
fn stateful_component(self) -> ElementAdapter<V> fn component(self) -> StatelessElementAdapter
where where
Self: Sized, Self: Sized,
{ {
ElementAdapter::new(self.into_any()) StatelessElementAdapter::new(self.into_any())
} }
fn component(self) -> DynamicElementAdapter fn stateful_component(self) -> StatefulElementAdapter<V>
where where
Self: Sized, Self: Sized,
{ {
DynamicElementAdapter::new(self.into_any()) StatefulElementAdapter::new(self.into_any())
} }
fn styleable_component(self) -> StylableAdapter<DynamicElementAdapter> fn styleable_component(self) -> StylableAdapter<StatelessElementAdapter>
where where
Self: Sized, Self: Sized,
{ {
DynamicElementAdapter::new(self.into_any()).stylable() StatelessElementAdapter::new(self.into_any()).stylable()
} }
} }

View file

@ -9,14 +9,15 @@ use crate::{
use super::Empty; use super::Empty;
/// The core stateless component trait, simply rendering an element tree
pub trait Component { pub trait Component {
fn render<V: View>(self, v: &mut V, cx: &mut ViewContext<V>) -> AnyElement<V>; fn render<V: View>(self, cx: &mut ViewContext<V>) -> AnyElement<V>;
fn element<V: View>(self) -> StatefulAdapter<V, Self> fn element<V: View>(self) -> ComponentAdapter<V, Self>
where where
Self: Sized, Self: Sized,
{ {
StatefulAdapter::new(self) ComponentAdapter::new(self)
} }
fn stylable(self) -> StylableAdapter<Self> fn stylable(self) -> StylableAdapter<Self>
@ -25,8 +26,46 @@ pub trait Component {
{ {
StylableAdapter::new(self) StylableAdapter::new(self)
} }
fn stateful<V: View>(self) -> StatefulAdapter<Self, V>
where
Self: Sized,
{
StatefulAdapter::new(self)
}
} }
/// Allows a a component's styles to be rebound in a simple way.
pub trait Stylable: Component {
type Style: Clone;
fn with_style(self, style: Self::Style) -> Self;
}
/// This trait models the typestate pattern for a component's style,
/// enforcing at compile time that a component is only usable after
/// it has been styled while still allowing for late binding of the
/// styling information
pub trait SafeStylable {
type Style: Clone;
type Output: Component;
fn with_style(self, style: Self::Style) -> Self::Output;
}
/// All stylable components can trivially implement SafeStylable
impl<C: Stylable> SafeStylable for C {
type Style = C::Style;
type Output = C;
fn with_style(self, style: Self::Style) -> Self::Output {
self.with_style(style)
}
}
/// Allows converting an unstylable component into a stylable one
/// by using `()` as the style type
pub struct StylableAdapter<C: Component> { pub struct StylableAdapter<C: Component> {
component: C, component: C,
} }
@ -37,7 +76,7 @@ impl<C: Component> StylableAdapter<C> {
} }
} }
impl<C: Component> StyleableComponent for StylableAdapter<C> { impl<C: Component> SafeStylable for StylableAdapter<C> {
type Style = (); type Style = ();
type Output = C; type Output = C;
@ -47,36 +86,61 @@ impl<C: Component> StyleableComponent for StylableAdapter<C> {
} }
} }
pub trait StyleableComponent { /// This is a secondary trait for components that can be styled
/// which rely on their view's state. This is useful for components that, for example,
/// want to take click handler callbacks Unfortunately, the generic bound on the
/// Component trait makes it incompatible with the stateless components above.
// So let's just replicate them for now
pub trait StatefulComponent<V: View> {
fn render(self, v: &mut V, cx: &mut ViewContext<V>) -> AnyElement<V>;
fn element(self) -> ComponentAdapter<V, Self>
where
Self: Sized,
{
ComponentAdapter::new(self)
}
fn styleable(self) -> StatefulStylableAdapter<Self, V>
where
Self: Sized,
{
StatefulStylableAdapter::new(self)
}
fn stateless(self) -> StatelessElementAdapter
where
Self: Sized + 'static,
{
StatelessElementAdapter::new(self.element().into_any())
}
}
/// It is trivial to convert stateless components to stateful components, so lets
/// do so en masse. Note that the reverse is impossible without a helper.
impl<V: View, C: Component> StatefulComponent<V> for C {
fn render(self, _: &mut V, cx: &mut ViewContext<V>) -> AnyElement<V> {
self.render(cx)
}
}
/// Same as stylable, but generic over a view type
pub trait StatefulStylable<V: View>: StatefulComponent<V> {
type Style: Clone; type Style: Clone;
type Output: Component;
fn with_style(self, style: Self::Style) -> Self::Output; fn stateful_with_style(self, style: Self::Style) -> Self;
} }
impl Component for () { /// Same as SafeStylable, but generic over a view type
fn render<V: View>(self, _: &mut V, _: &mut ViewContext<V>) -> AnyElement<V> { pub trait StatefulSafeStylable<V: View> {
Empty::new().into_any()
}
}
impl StyleableComponent for () {
type Style = ();
type Output = ();
fn with_style(self, _: Self::Style) -> Self::Output {
()
}
}
pub trait StatefulStyleableComponent<V: View> {
type Style: Clone; type Style: Clone;
type Output: StatefulComponent<V>; type Output: StatefulComponent<V>;
fn stateful_with_style(self, style: Self::Style) -> Self::Output; fn stateful_with_style(self, style: Self::Style) -> Self::Output;
} }
impl<V: View, C: StyleableComponent> StatefulStyleableComponent<V> for C { /// Converting from stateless to stateful
impl<V: View, C: SafeStylable> StatefulSafeStylable<V> for C {
type Style = C::Style; type Style = C::Style;
type Output = C::Output; type Output = C::Output;
@ -86,31 +150,29 @@ impl<V: View, C: StyleableComponent> StatefulStyleableComponent<V> for C {
} }
} }
pub trait StatefulComponent<V: View> { // A helper for converting stateless components into stateful ones
fn render(self, v: &mut V, cx: &mut ViewContext<V>) -> AnyElement<V>; pub struct StatefulAdapter<C, V> {
component: C,
phantom: std::marker::PhantomData<V>,
}
fn stateful_element(self) -> StatefulAdapter<V, Self> impl<C: Component, V: View> StatefulAdapter<C, V> {
where pub fn new(component: C) -> Self {
Self: Sized, Self {
{ component,
StatefulAdapter::new(self) phantom: std::marker::PhantomData,
} }
fn stateful_styleable(self) -> StatefulStylableAdapter<Self, V>
where
Self: Sized,
{
StatefulStylableAdapter::new(self)
} }
} }
impl<V: View, C: Component> StatefulComponent<V> for C { impl<C: Component, V: View> StatefulComponent<V> for StatefulAdapter<C, V> {
fn render(self, v: &mut V, cx: &mut ViewContext<V>) -> AnyElement<V> { fn render(self, _: &mut V, cx: &mut ViewContext<V>) -> AnyElement<V> {
self.render(v, cx) self.component.render(cx)
} }
} }
// StylableComponent -> Component // A helper for converting stateful but style-less components into stylable ones
// by using `()` as the style type
pub struct StatefulStylableAdapter<C: StatefulComponent<V>, V: View> { pub struct StatefulStylableAdapter<C: StatefulComponent<V>, V: View> {
component: C, component: C,
phantom: std::marker::PhantomData<V>, phantom: std::marker::PhantomData<V>,
@ -125,9 +187,7 @@ impl<C: StatefulComponent<V>, V: View> StatefulStylableAdapter<C, V> {
} }
} }
impl<C: StatefulComponent<V>, V: View> StatefulStyleableComponent<V> impl<C: StatefulComponent<V>, V: View> StatefulSafeStylable<V> for StatefulStylableAdapter<C, V> {
for StatefulStylableAdapter<C, V>
{
type Style = (); type Style = ();
type Output = C; type Output = C;
@ -137,37 +197,37 @@ impl<C: StatefulComponent<V>, V: View> StatefulStyleableComponent<V>
} }
} }
// Element -> GeneralComponent /// A way of erasing the view generic from an element, useful
/// for wrapping up an explicit element tree into stateless
pub struct DynamicElementAdapter { /// components
pub struct StatelessElementAdapter {
element: Box<dyn Any>, element: Box<dyn Any>,
} }
impl DynamicElementAdapter { impl StatelessElementAdapter {
pub fn new<V: View>(element: AnyElement<V>) -> Self { pub fn new<V: View>(element: AnyElement<V>) -> Self {
DynamicElementAdapter { StatelessElementAdapter {
element: Box::new(element) as Box<dyn Any>, element: Box::new(element) as Box<dyn Any>,
} }
} }
} }
impl Component for DynamicElementAdapter { impl Component for StatelessElementAdapter {
fn render<V: View>(self, _: &mut V, _: &mut ViewContext<V>) -> AnyElement<V> { fn render<V: View>(self, _: &mut ViewContext<V>) -> AnyElement<V> {
let element = self *self
.element .element
.downcast::<AnyElement<V>>() .downcast::<AnyElement<V>>()
.expect("Don't move elements out of their view :("); .expect("Don't move elements out of their view :(")
*element
} }
} }
// Element -> Component // For converting elements into stateful components
pub struct ElementAdapter<V: View> { pub struct StatefulElementAdapter<V: View> {
element: AnyElement<V>, element: AnyElement<V>,
_phantom: std::marker::PhantomData<V>, _phantom: std::marker::PhantomData<V>,
} }
impl<V: View> ElementAdapter<V> { impl<V: View> StatefulElementAdapter<V> {
pub fn new(element: AnyElement<V>) -> Self { pub fn new(element: AnyElement<V>) -> Self {
Self { Self {
element, element,
@ -176,20 +236,35 @@ impl<V: View> ElementAdapter<V> {
} }
} }
impl<V: View> StatefulComponent<V> for ElementAdapter<V> { impl<V: View> StatefulComponent<V> for StatefulElementAdapter<V> {
fn render(self, _: &mut V, _: &mut ViewContext<V>) -> AnyElement<V> { fn render(self, _: &mut V, _: &mut ViewContext<V>) -> AnyElement<V> {
self.element self.element
} }
} }
// Component -> Element /// A convenient shorthand for creating an empty component.
pub struct StatefulAdapter<V: View, E> { impl Component for () {
fn render<V: View>(self, _: &mut ViewContext<V>) -> AnyElement<V> {
Empty::new().into_any()
}
}
impl Stylable for () {
type Style = ();
fn with_style(self, _: Self::Style) -> Self {
()
}
}
// For converting components back into Elements
pub struct ComponentAdapter<V: View, E> {
component: Option<E>, component: Option<E>,
element: Option<AnyElement<V>>, element: Option<AnyElement<V>>,
phantom: PhantomData<V>, phantom: PhantomData<V>,
} }
impl<E, V: View> StatefulAdapter<V, E> { impl<E, V: View> ComponentAdapter<V, E> {
pub fn new(e: E) -> Self { pub fn new(e: E) -> Self {
Self { Self {
component: Some(e), component: Some(e),
@ -199,7 +274,7 @@ impl<E, V: View> StatefulAdapter<V, E> {
} }
} }
impl<V: View, C: StatefulComponent<V> + 'static> Element<V> for StatefulAdapter<V, C> { impl<V: View, C: StatefulComponent<V> + 'static> Element<V> for ComponentAdapter<V, C> {
type LayoutState = (); type LayoutState = ();
type PaintState = (); type PaintState = ();
@ -262,6 +337,7 @@ impl<V: View, C: StatefulComponent<V> + 'static> Element<V> for StatefulAdapter<
) -> serde_json::Value { ) -> serde_json::Value {
serde_json::json!({ serde_json::json!({
"type": "ComponentAdapter", "type": "ComponentAdapter",
"component": std::any::type_name::<C>(),
"child": self.element.as_ref().map(|el| el.debug(view, cx)), "child": self.element.as_ref().map(|el| el.debug(view, cx)),
}) })
} }

View file

@ -2,15 +2,13 @@ use bitflags::bitflags;
pub use buffer_search::BufferSearchBar; pub use buffer_search::BufferSearchBar;
use gpui::{ use gpui::{
actions, actions,
elements::{Component, StyleableComponent, TooltipStyle}, elements::{Component, SafeStylable, TooltipStyle},
Action, AnyElement, AppContext, Element, View, Action, AnyElement, AppContext, Element, View,
}; };
pub use mode::SearchMode; pub use mode::SearchMode;
use project::search::SearchQuery; use project::search::SearchQuery;
pub use project_search::{ProjectSearchBar, ProjectSearchView}; pub use project_search::{ProjectSearchBar, ProjectSearchView};
use theme::components::{ use theme::components::{action_button::Button, svg::Svg, ComponentExt, ToggleIconButtonStyle};
action_button::ActionButton, svg::Svg, ComponentExt, ToggleIconButtonStyle,
};
pub mod buffer_search; pub mod buffer_search;
mod history; mod history;
@ -91,7 +89,7 @@ impl SearchOptions {
tooltip_style: TooltipStyle, tooltip_style: TooltipStyle,
button_style: ToggleIconButtonStyle, button_style: ToggleIconButtonStyle,
) -> AnyElement<V> { ) -> AnyElement<V> {
ActionButton::new_dynamic(self.to_toggle_action()) Button::dynamic_action(self.to_toggle_action())
.with_tooltip(format!("Toggle {}", self.label()), tooltip_style) .with_tooltip(format!("Toggle {}", self.label()), tooltip_style)
.with_contents(Svg::new(self.icon())) .with_contents(Svg::new(self.icon()))
.toggleable(active) .toggleable(active)

View file

@ -1,4 +1,4 @@
use gpui::{elements::StyleableComponent, Action}; use gpui::{elements::SafeStylable, Action};
use crate::{Interactive, Toggleable}; use crate::{Interactive, Toggleable};
@ -6,44 +6,34 @@ use self::{action_button::ButtonStyle, disclosure::Disclosable, svg::SvgStyle, t
pub type ToggleIconButtonStyle = Toggleable<Interactive<ButtonStyle<SvgStyle>>>; pub type ToggleIconButtonStyle = Toggleable<Interactive<ButtonStyle<SvgStyle>>>;
pub trait ComponentExt<C: StyleableComponent> { pub trait ComponentExt<C: SafeStylable> {
fn toggleable(self, active: bool) -> Toggle<C, ()>; fn toggleable(self, active: bool) -> Toggle<C, ()>;
fn disclosable( fn disclosable(self, disclosed: Option<bool>, action: Box<dyn Action>) -> Disclosable<C, ()>;
self,
disclosed: Option<bool>,
action: Box<dyn Action>,
id: usize,
) -> Disclosable<C, ()>;
} }
impl<C: StyleableComponent> ComponentExt<C> for C { impl<C: SafeStylable> ComponentExt<C> for C {
fn toggleable(self, active: bool) -> Toggle<C, ()> { fn toggleable(self, active: bool) -> Toggle<C, ()> {
Toggle::new(self, active) Toggle::new(self, active)
} }
/// Some(True) => disclosed => content is visible /// Some(True) => disclosed => content is visible
/// Some(false) => closed => content is hidden /// Some(false) => closed => content is hidden
/// None => No disclosure button, but reserve spacing /// None => No disclosure button, but reserve disclosure spacing
fn disclosable( fn disclosable(self, disclosed: Option<bool>, action: Box<dyn Action>) -> Disclosable<C, ()> {
self, Disclosable::new(disclosed, self, action)
disclosed: Option<bool>,
action: Box<dyn Action>,
id: usize,
) -> Disclosable<C, ()> {
Disclosable::new(disclosed, self, action, id)
} }
} }
pub mod disclosure { pub mod disclosure {
use gpui::{ use gpui::{
elements::{Component, Empty, Flex, ParentElement, StyleableComponent}, elements::{Component, Empty, Flex, ParentElement, SafeStylable},
Action, Element, Action, Element,
}; };
use schemars::JsonSchema; use schemars::JsonSchema;
use serde_derive::Deserialize; use serde_derive::Deserialize;
use super::{action_button::ActionButton, svg::Svg, ComponentExt, ToggleIconButtonStyle}; use super::{action_button::Button, svg::Svg, ComponentExt, ToggleIconButtonStyle};
#[derive(Clone, Default, Deserialize, JsonSchema)] #[derive(Clone, Default, Deserialize, JsonSchema)]
pub struct DisclosureStyle<S> { pub struct DisclosureStyle<S> {
@ -72,19 +62,24 @@ pub mod disclosure {
disclosed: Option<bool>, disclosed: Option<bool>,
content: C, content: C,
action: Box<dyn Action>, action: Box<dyn Action>,
id: usize,
) -> Disclosable<C, ()> { ) -> Disclosable<C, ()> {
Disclosable { Disclosable {
disclosed, disclosed,
content, content,
action, action,
id, id: 0,
style: (), style: (),
} }
} }
} }
impl<C: StyleableComponent> StyleableComponent for Disclosable<C, ()> { impl<C> Disclosable<C, ()> {
pub fn with_id(self, id: usize) -> Disclosable<C, ()> {
Disclosable { id, ..self }
}
}
impl<C: SafeStylable> SafeStylable for Disclosable<C, ()> {
type Style = DisclosureStyle<C::Style>; type Style = DisclosureStyle<C::Style>;
type Output = Disclosable<C, Self::Style>; type Output = Disclosable<C, Self::Style>;
@ -100,15 +95,11 @@ pub mod disclosure {
} }
} }
impl<C: StyleableComponent> Component for Disclosable<C, DisclosureStyle<C::Style>> { impl<C: SafeStylable> Component for Disclosable<C, DisclosureStyle<C::Style>> {
fn render<V: gpui::View>( fn render<V: gpui::View>(self, cx: &mut gpui::ViewContext<V>) -> gpui::AnyElement<V> {
self,
v: &mut V,
cx: &mut gpui::ViewContext<V>,
) -> gpui::AnyElement<V> {
Flex::row() Flex::row()
.with_child(if let Some(disclosed) = self.disclosed { .with_child(if let Some(disclosed) = self.disclosed {
ActionButton::new_dynamic(self.action) Button::dynamic_action(self.action)
.with_id(self.id) .with_id(self.id)
.with_contents(Svg::new(if disclosed { .with_contents(Svg::new(if disclosed {
"icons/file_icons/chevron_down.svg" "icons/file_icons/chevron_down.svg"
@ -131,7 +122,7 @@ pub mod disclosure {
.with_child( .with_child(
self.content self.content
.with_style(self.style.content) .with_style(self.style.content)
.render(v, cx) .render(cx)
.flex(1., true), .flex(1., true),
) )
.align_children_center() .align_children_center()
@ -141,7 +132,7 @@ pub mod disclosure {
} }
pub mod toggle { pub mod toggle {
use gpui::elements::{Component, StyleableComponent}; use gpui::elements::{Component, SafeStylable};
use crate::Toggleable; use crate::Toggleable;
@ -151,7 +142,7 @@ pub mod toggle {
component: C, component: C,
} }
impl<C: StyleableComponent> Toggle<C, ()> { impl<C: SafeStylable> Toggle<C, ()> {
pub fn new(component: C, active: bool) -> Self { pub fn new(component: C, active: bool) -> Self {
Toggle { Toggle {
active, active,
@ -161,7 +152,7 @@ pub mod toggle {
} }
} }
impl<C: StyleableComponent> StyleableComponent for Toggle<C, ()> { impl<C: SafeStylable> SafeStylable for Toggle<C, ()> {
type Style = Toggleable<C::Style>; type Style = Toggleable<C::Style>;
type Output = Toggle<C, Self::Style>; type Output = Toggle<C, Self::Style>;
@ -175,15 +166,11 @@ pub mod toggle {
} }
} }
impl<C: StyleableComponent> Component for Toggle<C, Toggleable<C::Style>> { impl<C: SafeStylable> Component for Toggle<C, Toggleable<C::Style>> {
fn render<V: gpui::View>( fn render<V: gpui::View>(self, cx: &mut gpui::ViewContext<V>) -> gpui::AnyElement<V> {
self,
v: &mut V,
cx: &mut gpui::ViewContext<V>,
) -> gpui::AnyElement<V> {
self.component self.component
.with_style(self.style.in_state(self.active).clone()) .with_style(self.style.in_state(self.active).clone())
.render(v, cx) .render(cx)
} }
} }
} }
@ -192,9 +179,7 @@ pub mod action_button {
use std::borrow::Cow; use std::borrow::Cow;
use gpui::{ use gpui::{
elements::{ elements::{Component, ContainerStyle, MouseEventHandler, SafeStylable, TooltipStyle},
Component, ContainerStyle, MouseEventHandler, StyleableComponent, TooltipStyle,
},
platform::{CursorStyle, MouseButton}, platform::{CursorStyle, MouseButton},
Action, Element, TypeTag, View, Action, Element, TypeTag, View,
}; };
@ -216,7 +201,7 @@ pub mod action_button {
contents: C, contents: C,
} }
pub struct ActionButton<C, S> { pub struct Button<C, S> {
action: Box<dyn Action>, action: Box<dyn Action>,
tooltip: Option<(Cow<'static, str>, TooltipStyle)>, tooltip: Option<(Cow<'static, str>, TooltipStyle)>,
tag: TypeTag, tag: TypeTag,
@ -225,8 +210,8 @@ pub mod action_button {
style: Interactive<S>, style: Interactive<S>,
} }
impl ActionButton<(), ()> { impl Button<(), ()> {
pub fn new_dynamic(action: Box<dyn Action>) -> Self { pub fn dynamic_action(action: Box<dyn Action>) -> Self {
Self { Self {
contents: (), contents: (),
tag: action.type_tag(), tag: action.type_tag(),
@ -237,8 +222,8 @@ pub mod action_button {
} }
} }
pub fn new<A: Action + Clone>(action: A) -> Self { pub fn action<A: Action + Clone>(action: A) -> Self {
Self::new_dynamic(Box::new(action)) Self::dynamic_action(Box::new(action))
} }
pub fn with_tooltip( pub fn with_tooltip(
@ -255,8 +240,8 @@ pub mod action_button {
self self
} }
pub fn with_contents<C: StyleableComponent>(self, contents: C) -> ActionButton<C, ()> { pub fn with_contents<C: SafeStylable>(self, contents: C) -> Button<C, ()> {
ActionButton { Button {
action: self.action, action: self.action,
tag: self.tag, tag: self.tag,
style: self.style, style: self.style,
@ -267,12 +252,12 @@ pub mod action_button {
} }
} }
impl<C: StyleableComponent> StyleableComponent for ActionButton<C, ()> { impl<C: SafeStylable> SafeStylable for Button<C, ()> {
type Style = Interactive<ButtonStyle<C::Style>>; type Style = Interactive<ButtonStyle<C::Style>>;
type Output = ActionButton<C, ButtonStyle<C::Style>>; type Output = Button<C, ButtonStyle<C::Style>>;
fn with_style(self, style: Self::Style) -> Self::Output { fn with_style(self, style: Self::Style) -> Self::Output {
ActionButton { Button {
action: self.action, action: self.action,
tag: self.tag, tag: self.tag,
contents: self.contents, contents: self.contents,
@ -283,14 +268,14 @@ pub mod action_button {
} }
} }
impl<C: StyleableComponent> Component for ActionButton<C, ButtonStyle<C::Style>> { impl<C: SafeStylable> Component for Button<C, ButtonStyle<C::Style>> {
fn render<V: View>(self, v: &mut V, cx: &mut gpui::ViewContext<V>) -> gpui::AnyElement<V> { fn render<V: View>(self, cx: &mut gpui::ViewContext<V>) -> gpui::AnyElement<V> {
let mut button = MouseEventHandler::new_dynamic(self.tag, self.id, cx, |state, cx| { let mut button = MouseEventHandler::new_dynamic(self.tag, self.id, cx, |state, cx| {
let style = self.style.style_for(state); let style = self.style.style_for(state);
let mut contents = self let mut contents = self
.contents .contents
.with_style(style.contents.to_owned()) .with_style(style.contents.to_owned())
.render(v, cx) .render(cx)
.contained() .contained()
.with_style(style.container) .with_style(style.container)
.constrained(); .constrained();
@ -335,7 +320,7 @@ pub mod svg {
use std::borrow::Cow; use std::borrow::Cow;
use gpui::{ use gpui::{
elements::{Component, Empty, StyleableComponent}, elements::{Component, Empty, SafeStylable},
Element, Element,
}; };
use schemars::JsonSchema; use schemars::JsonSchema;
@ -417,7 +402,7 @@ pub mod svg {
} }
} }
impl StyleableComponent for Svg<()> { impl SafeStylable for Svg<()> {
type Style = SvgStyle; type Style = SvgStyle;
type Output = Svg<SvgStyle>; type Output = Svg<SvgStyle>;
@ -431,11 +416,7 @@ pub mod svg {
} }
impl Component for Svg<SvgStyle> { impl Component for Svg<SvgStyle> {
fn render<V: gpui::View>( fn render<V: gpui::View>(self, _: &mut gpui::ViewContext<V>) -> gpui::AnyElement<V> {
self,
_: &mut V,
_: &mut gpui::ViewContext<V>,
) -> gpui::AnyElement<V> {
if let Some(path) = self.path { if let Some(path) = self.path {
gpui::elements::Svg::new(path) gpui::elements::Svg::new(path)
.with_color(self.style.color) .with_color(self.style.color)
@ -455,7 +436,7 @@ pub mod label {
use std::borrow::Cow; use std::borrow::Cow;
use gpui::{ use gpui::{
elements::{Component, LabelStyle, StyleableComponent}, elements::{Component, LabelStyle, SafeStylable},
Element, Element,
}; };
@ -473,7 +454,7 @@ pub mod label {
} }
} }
impl StyleableComponent for Label<()> { impl SafeStylable for Label<()> {
type Style = LabelStyle; type Style = LabelStyle;
type Output = Label<LabelStyle>; type Output = Label<LabelStyle>;
@ -487,11 +468,7 @@ pub mod label {
} }
impl Component for Label<LabelStyle> { impl Component for Label<LabelStyle> {
fn render<V: gpui::View>( fn render<V: gpui::View>(self, _: &mut gpui::ViewContext<V>) -> gpui::AnyElement<V> {
self,
_: &mut V,
_: &mut gpui::ViewContext<V>,
) -> gpui::AnyElement<V> {
gpui::elements::Label::new(self.text, self.style).into_any() gpui::elements::Label::new(self.text, self.style).into_any()
} }
} }