ui_input: TextField
-> SingleLineInput
(#28031)
- Rename `TextField` -> `SingleLineInput` - Add a component preview for `SingleLineInput` - Apply `SingleLineInput` to the AddContextServerModal Release Notes: - N/A --------- Co-authored-by: Agus Zubiaga <hi@aguz.me> Co-authored-by: Danilo Leal <daniloleal09@gmail.com> Co-authored-by: Danilo Leal <67129314+danilo-leal@users.noreply.github.com>
This commit is contained in:
parent
315f1bf168
commit
2086f7d85b
3
Cargo.lock
generated
3
Cargo.lock
generated
@ -116,6 +116,7 @@ dependencies = [
|
|||||||
"time",
|
"time",
|
||||||
"time_format",
|
"time_format",
|
||||||
"ui",
|
"ui",
|
||||||
|
"ui_input",
|
||||||
"util",
|
"util",
|
||||||
"uuid",
|
"uuid",
|
||||||
"vim_mode_setting",
|
"vim_mode_setting",
|
||||||
@ -15310,8 +15311,10 @@ dependencies = [
|
|||||||
name = "ui_input"
|
name = "ui_input"
|
||||||
version = "0.1.0"
|
version = "0.1.0"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
|
"component",
|
||||||
"editor",
|
"editor",
|
||||||
"gpui",
|
"gpui",
|
||||||
|
"linkme",
|
||||||
"settings",
|
"settings",
|
||||||
"theme",
|
"theme",
|
||||||
"ui",
|
"ui",
|
||||||
|
@ -81,6 +81,7 @@ theme.workspace = true
|
|||||||
time.workspace = true
|
time.workspace = true
|
||||||
time_format.workspace = true
|
time_format.workspace = true
|
||||||
ui.workspace = true
|
ui.workspace = true
|
||||||
|
ui_input.workspace = true
|
||||||
util.workspace = true
|
util.workspace = true
|
||||||
uuid.workspace = true
|
uuid.workspace = true
|
||||||
vim_mode_setting.workspace = true
|
vim_mode_setting.workspace = true
|
||||||
|
@ -1,17 +1,17 @@
|
|||||||
use context_server::{ContextServerSettings, ServerCommand, ServerConfig};
|
use context_server::{ContextServerSettings, ServerCommand, ServerConfig};
|
||||||
use editor::Editor;
|
|
||||||
use gpui::{DismissEvent, Entity, EventEmitter, FocusHandle, Focusable, WeakEntity, prelude::*};
|
use gpui::{DismissEvent, Entity, EventEmitter, FocusHandle, Focusable, WeakEntity, prelude::*};
|
||||||
use serde_json::json;
|
use serde_json::json;
|
||||||
use settings::update_settings_file;
|
use settings::update_settings_file;
|
||||||
use ui::{Modal, ModalFooter, ModalHeader, Section, Tooltip, prelude::*};
|
use ui::{Modal, ModalFooter, ModalHeader, Section, Tooltip, prelude::*};
|
||||||
|
use ui_input::SingleLineInput;
|
||||||
use workspace::{ModalView, Workspace};
|
use workspace::{ModalView, Workspace};
|
||||||
|
|
||||||
use crate::AddContextServer;
|
use crate::AddContextServer;
|
||||||
|
|
||||||
pub struct AddContextServerModal {
|
pub struct AddContextServerModal {
|
||||||
workspace: WeakEntity<Workspace>,
|
workspace: WeakEntity<Workspace>,
|
||||||
name_editor: Entity<Editor>,
|
name_editor: Entity<SingleLineInput>,
|
||||||
command_editor: Entity<Editor>,
|
command_editor: Entity<SingleLineInput>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl AddContextServerModal {
|
impl AddContextServerModal {
|
||||||
@ -33,15 +33,10 @@ impl AddContextServerModal {
|
|||||||
window: &mut Window,
|
window: &mut Window,
|
||||||
cx: &mut Context<Self>,
|
cx: &mut Context<Self>,
|
||||||
) -> Self {
|
) -> Self {
|
||||||
let name_editor = cx.new(|cx| Editor::single_line(window, cx));
|
let name_editor =
|
||||||
let command_editor = cx.new(|cx| Editor::single_line(window, cx));
|
cx.new(|cx| SingleLineInput::new(window, cx, "Your server name").label("Name"));
|
||||||
|
let command_editor = cx.new(|cx| {
|
||||||
name_editor.update(cx, |editor, cx| {
|
SingleLineInput::new(window, cx, "Command").label("Command to run the context server")
|
||||||
editor.set_placeholder_text("Context server name", cx);
|
|
||||||
});
|
|
||||||
|
|
||||||
command_editor.update(cx, |editor, cx| {
|
|
||||||
editor.set_placeholder_text("Command to run the context server", cx);
|
|
||||||
});
|
});
|
||||||
|
|
||||||
Self {
|
Self {
|
||||||
@ -52,8 +47,22 @@ impl AddContextServerModal {
|
|||||||
}
|
}
|
||||||
|
|
||||||
fn confirm(&mut self, cx: &mut Context<Self>) {
|
fn confirm(&mut self, cx: &mut Context<Self>) {
|
||||||
let name = self.name_editor.read(cx).text(cx).trim().to_string();
|
let name = self
|
||||||
let command = self.command_editor.read(cx).text(cx).trim().to_string();
|
.name_editor
|
||||||
|
.read(cx)
|
||||||
|
.editor()
|
||||||
|
.read(cx)
|
||||||
|
.text(cx)
|
||||||
|
.trim()
|
||||||
|
.to_string();
|
||||||
|
let command = self
|
||||||
|
.command_editor
|
||||||
|
.read(cx)
|
||||||
|
.editor()
|
||||||
|
.read(cx)
|
||||||
|
.text(cx)
|
||||||
|
.trim()
|
||||||
|
.to_string();
|
||||||
|
|
||||||
if name.is_empty() || command.is_empty() {
|
if name.is_empty() || command.is_empty() {
|
||||||
return;
|
return;
|
||||||
@ -104,8 +113,8 @@ impl EventEmitter<DismissEvent> for AddContextServerModal {}
|
|||||||
|
|
||||||
impl Render for AddContextServerModal {
|
impl Render for AddContextServerModal {
|
||||||
fn render(&mut self, _window: &mut Window, cx: &mut Context<Self>) -> impl IntoElement {
|
fn render(&mut self, _window: &mut Window, cx: &mut Context<Self>) -> impl IntoElement {
|
||||||
let is_name_empty = self.name_editor.read(cx).text(cx).trim().is_empty();
|
let is_name_empty = self.name_editor.read(cx).is_empty(cx);
|
||||||
let is_command_empty = self.command_editor.read(cx).text(cx).trim().is_empty();
|
let is_command_empty = self.command_editor.read(cx).is_empty(cx);
|
||||||
|
|
||||||
div()
|
div()
|
||||||
.elevation_3(cx)
|
.elevation_3(cx)
|
||||||
@ -122,18 +131,8 @@ impl Render for AddContextServerModal {
|
|||||||
.header(ModalHeader::new().headline("Add Context Server"))
|
.header(ModalHeader::new().headline("Add Context Server"))
|
||||||
.section(
|
.section(
|
||||||
Section::new()
|
Section::new()
|
||||||
.child(
|
.child(self.name_editor.clone())
|
||||||
v_flex()
|
.child(self.command_editor.clone()),
|
||||||
.gap_1()
|
|
||||||
.child(Label::new("Name"))
|
|
||||||
.child(self.name_editor.clone()),
|
|
||||||
)
|
|
||||||
.child(
|
|
||||||
v_flex()
|
|
||||||
.gap_1()
|
|
||||||
.child(Label::new("Command"))
|
|
||||||
.child(self.command_editor.clone()),
|
|
||||||
),
|
|
||||||
)
|
)
|
||||||
.footer(
|
.footer(
|
||||||
ModalFooter::new()
|
ModalFooter::new()
|
||||||
|
@ -12,8 +12,10 @@ workspace = true
|
|||||||
path = "src/ui_input.rs"
|
path = "src/ui_input.rs"
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
|
component.workspace = true
|
||||||
editor.workspace = true
|
editor.workspace = true
|
||||||
gpui.workspace = true
|
gpui.workspace = true
|
||||||
|
linkme.workspace = true
|
||||||
settings.workspace = true
|
settings.workspace = true
|
||||||
theme.workspace = true
|
theme.workspace = true
|
||||||
ui.workspace = true
|
ui.workspace = true
|
||||||
|
@ -5,20 +5,14 @@
|
|||||||
//! It can't be located in the `ui` crate because it depends on `editor`.
|
//! It can't be located in the `ui` crate because it depends on `editor`.
|
||||||
//!
|
//!
|
||||||
|
|
||||||
|
use component::{ComponentPreview, example_group, single_example};
|
||||||
use editor::{Editor, EditorElement, EditorStyle};
|
use editor::{Editor, EditorElement, EditorStyle};
|
||||||
use gpui::{App, Entity, FocusHandle, Focusable, FontStyle, Hsla, TextStyle};
|
use gpui::{App, Entity, FocusHandle, Focusable, FontStyle, Hsla, TextStyle};
|
||||||
use settings::Settings;
|
use settings::Settings;
|
||||||
use theme::ThemeSettings;
|
use theme::ThemeSettings;
|
||||||
use ui::prelude::*;
|
use ui::prelude::*;
|
||||||
|
|
||||||
#[derive(Debug, Clone, Copy, PartialEq)]
|
pub struct SingleLineInputStyle {
|
||||||
pub enum FieldLabelLayout {
|
|
||||||
Hidden,
|
|
||||||
Inline,
|
|
||||||
Stacked,
|
|
||||||
}
|
|
||||||
|
|
||||||
pub struct TextFieldStyle {
|
|
||||||
text_color: Hsla,
|
text_color: Hsla,
|
||||||
background_color: Hsla,
|
background_color: Hsla,
|
||||||
border_color: Hsla,
|
border_color: Hsla,
|
||||||
@ -27,11 +21,13 @@ pub struct TextFieldStyle {
|
|||||||
/// A Text Field that can be used to create text fields like search inputs, form fields, etc.
|
/// A Text Field that can be used to create text fields like search inputs, form fields, etc.
|
||||||
///
|
///
|
||||||
/// It wraps a single line [`Editor`] and allows for common field properties like labels, placeholders, icons, etc.
|
/// It wraps a single line [`Editor`] and allows for common field properties like labels, placeholders, icons, etc.
|
||||||
pub struct TextField {
|
#[derive(IntoComponent)]
|
||||||
|
#[component(scope = "Input")]
|
||||||
|
pub struct SingleLineInput {
|
||||||
/// An optional label for the text field.
|
/// An optional label for the text field.
|
||||||
///
|
///
|
||||||
/// Its position is determined by the [`FieldLabelLayout`].
|
/// Its position is determined by the [`FieldLabelLayout`].
|
||||||
label: SharedString,
|
label: Option<SharedString>,
|
||||||
/// The placeholder text for the text field.
|
/// The placeholder text for the text field.
|
||||||
placeholder: SharedString,
|
placeholder: SharedString,
|
||||||
/// Exposes the underlying [`Model<Editor>`] to allow for customizing the editor beyond the provided API.
|
/// Exposes the underlying [`Model<Editor>`] to allow for customizing the editor beyond the provided API.
|
||||||
@ -42,25 +38,18 @@ pub struct TextField {
|
|||||||
///
|
///
|
||||||
/// For example, a magnifying glass icon in a search field.
|
/// For example, a magnifying glass icon in a search field.
|
||||||
start_icon: Option<IconName>,
|
start_icon: Option<IconName>,
|
||||||
/// The layout of the label relative to the text field.
|
|
||||||
with_label: FieldLabelLayout,
|
|
||||||
/// Whether the text field is disabled.
|
/// Whether the text field is disabled.
|
||||||
disabled: bool,
|
disabled: bool,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Focusable for TextField {
|
impl Focusable for SingleLineInput {
|
||||||
fn focus_handle(&self, cx: &App) -> FocusHandle {
|
fn focus_handle(&self, cx: &App) -> FocusHandle {
|
||||||
self.editor.focus_handle(cx)
|
self.editor.focus_handle(cx)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl TextField {
|
impl SingleLineInput {
|
||||||
pub fn new(
|
pub fn new(window: &mut Window, cx: &mut App, placeholder: impl Into<SharedString>) -> Self {
|
||||||
window: &mut Window,
|
|
||||||
cx: &mut App,
|
|
||||||
label: impl Into<SharedString>,
|
|
||||||
placeholder: impl Into<SharedString>,
|
|
||||||
) -> Self {
|
|
||||||
let placeholder_text = placeholder.into();
|
let placeholder_text = placeholder.into();
|
||||||
|
|
||||||
let editor = cx.new(|cx| {
|
let editor = cx.new(|cx| {
|
||||||
@ -70,11 +59,10 @@ impl TextField {
|
|||||||
});
|
});
|
||||||
|
|
||||||
Self {
|
Self {
|
||||||
label: label.into(),
|
label: None,
|
||||||
placeholder: placeholder_text,
|
placeholder: placeholder_text,
|
||||||
editor,
|
editor,
|
||||||
start_icon: None,
|
start_icon: None,
|
||||||
with_label: FieldLabelLayout::Hidden,
|
|
||||||
disabled: false,
|
disabled: false,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -84,8 +72,8 @@ impl TextField {
|
|||||||
self
|
self
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn with_label(mut self, layout: FieldLabelLayout) -> Self {
|
pub fn label(mut self, label: impl Into<SharedString>) -> Self {
|
||||||
self.with_label = layout;
|
self.label = Some(label.into());
|
||||||
self
|
self
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -95,25 +83,29 @@ impl TextField {
|
|||||||
.update(cx, |editor, _| editor.set_read_only(disabled))
|
.update(cx, |editor, _| editor.set_read_only(disabled))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn is_empty(&self, cx: &App) -> bool {
|
||||||
|
self.editor().read(cx).text(cx).trim().is_empty()
|
||||||
|
}
|
||||||
|
|
||||||
pub fn editor(&self) -> &Entity<Editor> {
|
pub fn editor(&self) -> &Entity<Editor> {
|
||||||
&self.editor
|
&self.editor
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Render for TextField {
|
impl Render for SingleLineInput {
|
||||||
fn render(&mut self, _: &mut Window, cx: &mut Context<Self>) -> impl IntoElement {
|
fn render(&mut self, _: &mut Window, cx: &mut Context<Self>) -> impl IntoElement {
|
||||||
let settings = ThemeSettings::get_global(cx);
|
let settings = ThemeSettings::get_global(cx);
|
||||||
let theme_color = cx.theme().colors();
|
let theme_color = cx.theme().colors();
|
||||||
|
|
||||||
let mut style = TextFieldStyle {
|
let mut style = SingleLineInputStyle {
|
||||||
text_color: theme_color.text,
|
text_color: theme_color.text,
|
||||||
background_color: theme_color.ghost_element_background,
|
background_color: theme_color.editor_background,
|
||||||
border_color: theme_color.border,
|
border_color: theme_color.border_variant,
|
||||||
};
|
};
|
||||||
|
|
||||||
if self.disabled {
|
if self.disabled {
|
||||||
style.text_color = theme_color.text_disabled;
|
style.text_color = theme_color.text_disabled;
|
||||||
style.background_color = theme_color.ghost_element_disabled;
|
style.background_color = theme_color.editor_background;
|
||||||
style.border_color = theme_color.border_disabled;
|
style.border_color = theme_color.border_disabled;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -123,8 +115,8 @@ impl Render for TextField {
|
|||||||
// }
|
// }
|
||||||
|
|
||||||
let text_style = TextStyle {
|
let text_style = TextStyle {
|
||||||
font_family: settings.buffer_font.family.clone(),
|
font_family: settings.ui_font.family.clone(),
|
||||||
font_features: settings.buffer_font.features.clone(),
|
font_features: settings.ui_font.features.clone(),
|
||||||
font_size: rems(0.875).into(),
|
font_size: rems(0.875).into(),
|
||||||
font_weight: settings.buffer_font.weight,
|
font_weight: settings.buffer_font.weight,
|
||||||
font_style: FontStyle::Normal,
|
font_style: FontStyle::Normal,
|
||||||
@ -140,13 +132,13 @@ impl Render for TextField {
|
|||||||
..Default::default()
|
..Default::default()
|
||||||
};
|
};
|
||||||
|
|
||||||
div()
|
v_flex()
|
||||||
.id(self.placeholder.clone())
|
.id(self.placeholder.clone())
|
||||||
.group("text-field")
|
|
||||||
.w_full()
|
.w_full()
|
||||||
.when(self.with_label == FieldLabelLayout::Stacked, |this| {
|
.gap_1()
|
||||||
|
.when_some(self.label.clone(), |this, label| {
|
||||||
this.child(
|
this.child(
|
||||||
Label::new(self.label.clone())
|
Label::new(label)
|
||||||
.size(LabelSize::Default)
|
.size(LabelSize::Default)
|
||||||
.color(if self.disabled {
|
.color(if self.disabled {
|
||||||
Color::Disabled
|
Color::Disabled
|
||||||
@ -156,35 +148,37 @@ impl Render for TextField {
|
|||||||
)
|
)
|
||||||
})
|
})
|
||||||
.child(
|
.child(
|
||||||
v_flex().w_full().child(
|
h_flex()
|
||||||
h_flex()
|
.px_2()
|
||||||
.w_full()
|
.py_1()
|
||||||
.flex_grow()
|
.bg(style.background_color)
|
||||||
.gap_2()
|
.text_color(style.text_color)
|
||||||
.when(self.with_label == FieldLabelLayout::Inline, |this| {
|
.rounded_md()
|
||||||
this.child(Label::new(self.label.clone()).size(LabelSize::Default))
|
.border_1()
|
||||||
})
|
.border_color(style.border_color)
|
||||||
.child(
|
.min_w_48()
|
||||||
h_flex()
|
.w_full()
|
||||||
.px_2()
|
.flex_grow()
|
||||||
.py_1()
|
.when_some(self.start_icon, |this, icon| {
|
||||||
.bg(style.background_color)
|
this.gap_1()
|
||||||
.text_color(style.text_color)
|
.child(Icon::new(icon).size(IconSize::Small).color(Color::Muted))
|
||||||
.rounded_lg()
|
})
|
||||||
.border_1()
|
.child(EditorElement::new(&self.editor, editor_style)),
|
||||||
.border_color(style.border_color)
|
|
||||||
.min_w_48()
|
|
||||||
.w_full()
|
|
||||||
.flex_grow()
|
|
||||||
.gap_1()
|
|
||||||
.when_some(self.start_icon, |this, icon| {
|
|
||||||
this.child(
|
|
||||||
Icon::new(icon).size(IconSize::Small).color(Color::Muted),
|
|
||||||
)
|
|
||||||
})
|
|
||||||
.child(EditorElement::new(&self.editor, editor_style)),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl ComponentPreview for SingleLineInput {
|
||||||
|
fn preview(window: &mut Window, cx: &mut App) -> AnyElement {
|
||||||
|
let input_1 =
|
||||||
|
cx.new(|cx| SingleLineInput::new(window, cx, "placeholder").label("Some Label"));
|
||||||
|
|
||||||
|
v_flex()
|
||||||
|
.gap_6()
|
||||||
|
.children(vec![example_group(vec![single_example(
|
||||||
|
"Default",
|
||||||
|
div().child(input_1.clone()).into_any_element(),
|
||||||
|
)])])
|
||||||
|
.into_any_element()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user