diff --git a/assets/icons/file_diff.svg b/assets/icons/file_diff.svg
new file mode 100644
index 00000000000000..ff20f16c60907c
--- /dev/null
+++ b/assets/icons/file_diff.svg
@@ -0,0 +1 @@
+
diff --git a/assets/icons/thumbs_down.svg b/assets/icons/thumbs_down.svg
new file mode 100644
index 00000000000000..816273e9700a70
--- /dev/null
+++ b/assets/icons/thumbs_down.svg
@@ -0,0 +1 @@
+
diff --git a/assets/icons/thumbs_up.svg b/assets/icons/thumbs_up.svg
new file mode 100644
index 00000000000000..01fae76fe750df
--- /dev/null
+++ b/assets/icons/thumbs_up.svg
@@ -0,0 +1 @@
+
diff --git a/crates/ui/src/components/icon.rs b/crates/ui/src/components/icon.rs
index 1885c927361b42..8aaef3a69a2929 100644
--- a/crates/ui/src/components/icon.rs
+++ b/crates/ui/src/components/icon.rs
@@ -178,6 +178,7 @@ pub enum IconName {
File,
FileCode,
FileDoc,
+ FileDiff,
FileGeneric,
FileGit,
FileLock,
@@ -278,6 +279,8 @@ pub enum IconName {
Tab,
Terminal,
TextSnippet,
+ ThumbsUp,
+ ThumbsDown,
Trash,
TrashAlt,
Triangle,
diff --git a/crates/zeta/src/rate_completion_modal.rs b/crates/zeta/src/rate_completion_modal.rs
index 2d5650ba16e573..c32959b459bfae 100644
--- a/crates/zeta/src/rate_completion_modal.rs
+++ b/crates/zeta/src/rate_completion_modal.rs
@@ -7,7 +7,7 @@ use gpui::{
use language::{language_settings, OffsetRangeExt};
use settings::Settings;
use theme::ThemeSettings;
-use ui::{prelude::*, ListItem, ListItemSpacing};
+use ui::{prelude::*, List, ListItem, ListItemSpacing, TintColor};
use workspace::{ModalView, Workspace};
pub struct RateCompletionModal {
@@ -69,7 +69,7 @@ impl RateCompletionModal {
editor.set_show_wrap_guides(false, cx);
editor.set_show_indent_guides(false, cx);
editor.set_show_inline_completions(Some(false), cx);
- editor.set_placeholder_text("Your feedback about this completion...", cx);
+ editor.set_placeholder_text("Add your feedback about this completion…", cx);
editor
}),
});
@@ -134,94 +134,116 @@ impl RateCompletionModal {
};
let rated = self.zeta.read(cx).is_completion_rated(completion_id);
+
+ let border_color = cx.theme().colors().border;
+ let bg_color = cx.theme().colors().editor_background;
+
+ let label_container = || h_flex().pl_1().gap_1p5();
+
Some(
v_flex()
- .flex_1()
.size_full()
- .gap_2()
- .child(h_flex().justify_center().children(if rated {
- Some(
- Label::new("This completion was already rated")
- .color(Color::Muted)
- .size(LabelSize::Large),
- )
- } else if active_completion.completion.edits.is_empty() {
- Some(
- Label::new("This completion didn't produce any edits")
- .color(Color::Warning)
- .size(LabelSize::Large),
- )
- } else {
- None
- }))
+ .overflow_hidden()
.child(
- v_flex()
+ div()
.id("diff")
- .flex_1()
- .flex_basis(relative(0.75))
- .bg(cx.theme().colors().editor_background)
- .overflow_y_scroll()
- .p_2()
- .border_color(cx.theme().colors().border)
- .border_1()
- .rounded_lg()
+ .py_4()
+ .px_6()
+ .size_full()
+ .bg(bg_color)
+ .overflow_scroll()
.child(StyledText::new(diff).with_highlights(&text_style, diff_highlights)),
)
.child(
div()
- .flex_1()
- .flex_basis(relative(0.25))
- .bg(cx.theme().colors().editor_background)
- .border_color(cx.theme().colors().border)
- .border_1()
- .rounded_lg()
+ .h_40()
+ .bg(bg_color)
+ .border_t_1()
+ .border_color(border_color)
.child(active_completion.feedback_editor.clone()),
)
.child(
h_flex()
- .gap_2()
- .justify_end()
- .child(
- Button::new("bad", "👎 Bad Completion")
- .size(ButtonSize::Large)
- .disabled(rated)
- .label_size(LabelSize::Large)
- .color(Color::Error)
- .on_click({
- let completion = active_completion.completion.clone();
- let feedback_editor = active_completion.feedback_editor.clone();
- cx.listener(move |this, _, cx| {
- this.zeta.update(cx, |zeta, cx| {
- zeta.rate_completion(
- &completion,
- InlineCompletionRating::Negative,
- feedback_editor.read(cx).text(cx),
- cx,
- )
- })
- })
- }),
- )
+ .p_1()
+ .h_8()
+ .border_t_1()
+ .border_color(border_color)
+ .max_w_full()
+ .justify_between()
+ .children(if rated {
+ Some(
+ label_container()
+ .child(
+ Icon::new(IconName::Check)
+ .size(IconSize::Small)
+ .color(Color::Success),
+ )
+ .child(Label::new("Rated completion").color(Color::Muted)),
+ )
+ } else if active_completion.completion.edits.is_empty() {
+ Some(
+ label_container()
+ .child(
+ Icon::new(IconName::Warning)
+ .size(IconSize::Small)
+ .color(Color::Warning),
+ )
+ .child(Label::new("No edits produced").color(Color::Muted)),
+ )
+ } else {
+ Some(label_container())
+ })
.child(
- Button::new("good", "👍 Good Completion")
- .size(ButtonSize::Large)
- .disabled(rated)
- .label_size(LabelSize::Large)
- .color(Color::Success)
- .on_click({
- let completion = active_completion.completion.clone();
- let feedback_editor = active_completion.feedback_editor.clone();
- cx.listener(move |this, _, cx| {
- this.zeta.update(cx, |zeta, cx| {
- zeta.rate_completion(
- &completion,
- InlineCompletionRating::Positive,
- feedback_editor.read(cx).text(cx),
- cx,
- )
- })
- })
- }),
+ h_flex()
+ .gap_1()
+ .child(
+ Button::new("bad", "Bad Completion")
+ .style(ButtonStyle::Tinted(TintColor::Negative))
+ .icon(IconName::ThumbsDown)
+ .icon_size(IconSize::Small)
+ .icon_position(IconPosition::Start)
+ .icon_color(Color::Error)
+ .disabled(rated)
+ .on_click({
+ let completion = active_completion.completion.clone();
+ let feedback_editor =
+ active_completion.feedback_editor.clone();
+ cx.listener(move |this, _, cx| {
+ this.zeta.update(cx, |zeta, cx| {
+ zeta.rate_completion(
+ &completion,
+ InlineCompletionRating::Negative,
+ feedback_editor.read(cx).text(cx),
+ cx,
+ )
+ })
+ })
+ }),
+ )
+ .child(
+ Button::new("good", "Good Completion")
+ .style(ButtonStyle::Tinted(TintColor::Positive))
+ .icon(IconName::ThumbsUp)
+ .icon_size(IconSize::Small)
+ .icon_position(IconPosition::Start)
+ .icon_color(Color::Success)
+ .disabled(rated)
+ .on_click({
+ let completion = active_completion.completion.clone();
+ let feedback_editor =
+ active_completion.feedback_editor.clone();
+ cx.listener(move |this, _, cx| {
+ this.zeta.update(cx, |zeta, cx| {
+ zeta.rate_completion(
+ &completion,
+ InlineCompletionRating::Positive,
+ feedback_editor.read(cx).text(cx),
+ cx,
+ )
+ })
+ })
+ }),
+ ),
),
),
)
@@ -230,27 +252,38 @@ impl RateCompletionModal {
impl Render for RateCompletionModal {
fn render(&mut self, cx: &mut ViewContext) -> impl IntoElement {
+ let border_color = cx.theme().colors().border;
+
h_flex()
- .gap_2()
- .bg(cx.theme().colors().elevated_surface_background)
- .w(cx.viewport_size().width - px(256.))
- .h(cx.viewport_size().height - px(256.))
- .rounded_lg()
- .shadow_lg()
- .p_2()
.key_context("RateCompletionModal")
.track_focus(&self.focus_handle)
.on_action(cx.listener(Self::dismiss))
+ .bg(cx.theme().colors().elevated_surface_background)
+ .border_1()
+ .border_color(border_color)
+ .w(cx.viewport_size().width - px(320.))
+ .h(cx.viewport_size().height - px(300.))
+ .rounded_lg()
+ .shadow_lg()
.child(
div()
.id("completion_list")
+ .border_r_1()
+ .border_color(border_color)
.w_96()
.h_full()
+ .p_0p5()
.overflow_y_scroll()
.child(
- ui::List::new()
+ List::new()
.empty_message(
- "No completions, use the editor to generate some and rate them!",
+ div()
+ .p_2()
+ .child(
+ Label::new("No completions yet. Use the editor to generate some and rate them!")
+ .color(Color::Muted),
+ )
+ .into_any_element(),
)
.children(self.zeta.read(cx).recent_completions().cloned().map(
|completion| {
@@ -261,22 +294,26 @@ impl Render for RateCompletionModal {
let rated =
self.zeta.read(cx).is_completion_rated(completion.id);
ListItem::new(completion.id)
+ .inset(true)
.spacing(ListItemSpacing::Sparse)
.selected(selected)
- .end_slot(if rated {
+ .start_slot(if rated {
Icon::new(IconName::Check).color(Color::Success)
} else if completion.edits.is_empty() {
- Icon::new(IconName::Ellipsis).color(Color::Muted)
+ Icon::new(IconName::File).color(Color::Muted).size(IconSize::Small)
} else {
- Icon::new(IconName::Diff).color(Color::Muted)
+ Icon::new(IconName::FileDiff).color(Color::Accent).size(IconSize::Small)
})
.child(Label::new(
completion.path.to_string_lossy().to_string(),
- ))
+ ).size(LabelSize::Small))
.child(
- Label::new(format!("({})", completion.id))
- .color(Color::Muted)
- .size(LabelSize::XSmall),
+ div()
+ .overflow_hidden()
+ .text_ellipsis()
+ .child(Label::new(format!("({})", completion.id))
+ .color(Color::Muted)
+ .size(LabelSize::XSmall)),
)
.on_click(cx.listener(move |this, _, cx| {
this.select_completion(Some(completion.clone()), cx);