1use std::sync::Arc;
  2
  3use client::TelemetrySettings;
  4use fs::Fs;
  5use gpui::{Action, App, IntoElement};
  6use settings::{BaseKeymap, Settings, update_settings_file};
  7use theme::{
  8    Appearance, SystemAppearance, ThemeMode, ThemeName, ThemeRegistry, ThemeSelection,
  9    ThemeSettings,
 10};
 11use ui::{
 12    Divider, ParentElement as _, StatefulInteractiveElement, SwitchField, TintColor,
 13    ToggleButtonGroup, ToggleButtonGroupSize, ToggleButtonSimple, ToggleButtonWithIcon, prelude::*,
 14    rems_from_px,
 15};
 16use vim_mode_setting::VimModeSetting;
 17
 18use crate::{
 19    ImportCursorSettings, ImportVsCodeSettings, SettingsImportState,
 20    theme_preview::{ThemePreviewStyle, ThemePreviewTile},
 21};
 22
 23const LIGHT_THEMES: [&str; 3] = ["One Light", "Ayu Light", "Gruvbox Light"];
 24const DARK_THEMES: [&str; 3] = ["One Dark", "Ayu Dark", "Gruvbox Dark"];
 25const FAMILY_NAMES: [SharedString; 3] = [
 26    SharedString::new_static("One"),
 27    SharedString::new_static("Ayu"),
 28    SharedString::new_static("Gruvbox"),
 29];
 30
 31fn get_theme_family_themes(theme_name: &str) -> Option<(&'static str, &'static str)> {
 32    for i in 0..LIGHT_THEMES.len() {
 33        if LIGHT_THEMES[i] == theme_name || DARK_THEMES[i] == theme_name {
 34            return Some((LIGHT_THEMES[i], DARK_THEMES[i]));
 35        }
 36    }
 37    None
 38}
 39
 40fn render_theme_section(tab_index: &mut isize, cx: &mut App) -> impl IntoElement {
 41    let theme_selection = ThemeSettings::get_global(cx).theme.clone();
 42    let system_appearance = theme::SystemAppearance::global(cx);
 43
 44    let theme_mode = theme_selection
 45        .mode()
 46        .unwrap_or_else(|| match *system_appearance {
 47            Appearance::Light => ThemeMode::Light,
 48            Appearance::Dark => ThemeMode::Dark,
 49        });
 50
 51    return v_flex()
 52        .gap_2()
 53        .child(
 54            h_flex().justify_between().child(Label::new("Theme")).child(
 55                ToggleButtonGroup::single_row(
 56                    "theme-selector-onboarding-dark-light",
 57                    [ThemeMode::Light, ThemeMode::Dark, ThemeMode::System].map(|mode| {
 58                        const MODE_NAMES: [SharedString; 3] = [
 59                            SharedString::new_static("Light"),
 60                            SharedString::new_static("Dark"),
 61                            SharedString::new_static("System"),
 62                        ];
 63                        ToggleButtonSimple::new(
 64                            MODE_NAMES[mode as usize].clone(),
 65                            move |_, _, cx| {
 66                                write_mode_change(mode, cx);
 67
 68                                telemetry::event!(
 69                                    "Welcome Theme mode Changed",
 70                                    from = theme_mode,
 71                                    to = mode
 72                                );
 73                            },
 74                        )
 75                    }),
 76                )
 77                .size(ToggleButtonGroupSize::Medium)
 78                .tab_index(tab_index)
 79                .selected_index(theme_mode as usize)
 80                .style(ui::ToggleButtonGroupStyle::Outlined)
 81                .width(rems_from_px(3. * 64.)),
 82            ),
 83        )
 84        .child(
 85            h_flex()
 86                .gap_4()
 87                .justify_between()
 88                .children(render_theme_previews(tab_index, &theme_selection, cx)),
 89        );
 90
 91    fn render_theme_previews(
 92        tab_index: &mut isize,
 93        theme_selection: &ThemeSelection,
 94        cx: &mut App,
 95    ) -> [impl IntoElement; 3] {
 96        let system_appearance = SystemAppearance::global(cx);
 97        let theme_registry = ThemeRegistry::global(cx);
 98
 99        let theme_seed = 0xBEEF as f32;
100        let theme_mode = theme_selection
101            .mode()
102            .unwrap_or_else(|| match *system_appearance {
103                Appearance::Light => ThemeMode::Light,
104                Appearance::Dark => ThemeMode::Dark,
105            });
106        let appearance = match theme_mode {
107            ThemeMode::Light => Appearance::Light,
108            ThemeMode::Dark => Appearance::Dark,
109            ThemeMode::System => *system_appearance,
110        };
111        let current_theme_name: SharedString = theme_selection.name(appearance).0.into();
112
113        let theme_names = match appearance {
114            Appearance::Light => LIGHT_THEMES,
115            Appearance::Dark => DARK_THEMES,
116        };
117
118        let themes = theme_names.map(|theme| theme_registry.get(theme).unwrap());
119
120        [0, 1, 2].map(|index| {
121            let theme = &themes[index];
122            let is_selected = theme.name == current_theme_name;
123            let name = theme.name.clone();
124            let colors = cx.theme().colors();
125
126            v_flex()
127                .w_full()
128                .items_center()
129                .gap_1()
130                .child(
131                    h_flex()
132                        .id(name)
133                        .relative()
134                        .w_full()
135                        .border_2()
136                        .border_color(colors.border_transparent)
137                        .rounded(ThemePreviewTile::ROOT_RADIUS)
138                        .map(|this| {
139                            if is_selected {
140                                this.border_color(colors.border_selected)
141                            } else {
142                                this.opacity(0.8).hover(|s| s.border_color(colors.border))
143                            }
144                        })
145                        .tab_index({
146                            *tab_index += 1;
147                            *tab_index - 1
148                        })
149                        .focus(|mut style| {
150                            style.border_color = Some(colors.border_focused);
151                            style
152                        })
153                        .on_click({
154                            let theme_name = theme.name.clone();
155                            let current_theme_name = current_theme_name.clone();
156
157                            move |_, _, cx| {
158                                write_theme_change(theme_name.clone(), theme_mode, cx);
159                                telemetry::event!(
160                                    "Welcome Theme Changed",
161                                    from = current_theme_name,
162                                    to = theme_name
163                                );
164                            }
165                        })
166                        .map(|this| {
167                            if theme_mode == ThemeMode::System {
168                                let (light, dark) = (
169                                    theme_registry.get(LIGHT_THEMES[index]).unwrap(),
170                                    theme_registry.get(DARK_THEMES[index]).unwrap(),
171                                );
172                                this.child(
173                                    ThemePreviewTile::new(light, theme_seed)
174                                        .style(ThemePreviewStyle::SideBySide(dark)),
175                                )
176                            } else {
177                                this.child(
178                                    ThemePreviewTile::new(theme.clone(), theme_seed)
179                                        .style(ThemePreviewStyle::Bordered),
180                                )
181                            }
182                        }),
183                )
184                .child(
185                    Label::new(FAMILY_NAMES[index].clone())
186                        .color(Color::Muted)
187                        .size(LabelSize::Small),
188                )
189        })
190    }
191
192    fn write_mode_change(mode: ThemeMode, cx: &mut App) {
193        let fs = <dyn Fs>::global(cx);
194        update_settings_file(fs, cx, move |settings, _cx| {
195            theme::set_mode(settings, mode);
196        });
197    }
198
199    fn write_theme_change(theme: impl Into<Arc<str>>, theme_mode: ThemeMode, cx: &mut App) {
200        let fs = <dyn Fs>::global(cx);
201        let theme = theme.into();
202        update_settings_file(fs, cx, move |settings, cx| {
203            if theme_mode == ThemeMode::System {
204                let (light_theme, dark_theme) =
205                    get_theme_family_themes(&theme).unwrap_or((theme.as_ref(), theme.as_ref()));
206
207                settings.theme.theme = Some(settings::ThemeSelection::Dynamic {
208                    mode: ThemeMode::System,
209                    light: ThemeName(light_theme.into()),
210                    dark: ThemeName(dark_theme.into()),
211                });
212            } else {
213                let appearance = *SystemAppearance::global(cx);
214                theme::set_theme(settings, theme, appearance);
215            }
216        });
217    }
218}
219
220fn render_telemetry_section(tab_index: &mut isize, cx: &App) -> impl IntoElement {
221    let fs = <dyn Fs>::global(cx);
222
223    v_flex()
224        .gap_4()
225        .child(
226            SwitchField::new(
227                "onboarding-telemetry-metrics",
228                None::<&str>,
229                Some("Help improve Zed by sending anonymous usage data".into()),
230                if TelemetrySettings::get_global(cx).metrics {
231                    ui::ToggleState::Selected
232                } else {
233                    ui::ToggleState::Unselected
234                },
235                {
236                    let fs = fs.clone();
237                    move |selection, _, cx| {
238                        let enabled = match selection {
239                            ToggleState::Selected => true,
240                            ToggleState::Unselected => false,
241                            ToggleState::Indeterminate => {
242                                return;
243                            }
244                        };
245
246                        update_settings_file(fs.clone(), cx, move |setting, _| {
247                            setting.telemetry.get_or_insert_default().metrics = Some(enabled);
248                        });
249
250                        // This telemetry event shouldn't fire when it's off. If it does we'll be alerted
251                        // and can fix it in a timely manner to respect a user's choice.
252                        telemetry::event!(
253                            "Welcome Page Telemetry Metrics Toggled",
254                            options = if enabled { "on" } else { "off" }
255                        );
256                    }
257                },
258            )
259            .tab_index({
260                *tab_index += 1;
261                *tab_index
262            }),
263        )
264        .child(
265            SwitchField::new(
266                "onboarding-telemetry-crash-reports",
267                None::<&str>,
268                Some(
269                    "Help fix Zed by sending crash reports so we can fix critical issues fast"
270                        .into(),
271                ),
272                if TelemetrySettings::get_global(cx).diagnostics {
273                    ui::ToggleState::Selected
274                } else {
275                    ui::ToggleState::Unselected
276                },
277                {
278                    let fs = fs.clone();
279                    move |selection, _, cx| {
280                        let enabled = match selection {
281                            ToggleState::Selected => true,
282                            ToggleState::Unselected => false,
283                            ToggleState::Indeterminate => {
284                                return;
285                            }
286                        };
287
288                        update_settings_file(fs.clone(), cx, move |setting, _| {
289                            setting.telemetry.get_or_insert_default().diagnostics = Some(enabled);
290                        });
291
292                        // This telemetry event shouldn't fire when it's off. If it does we'll be alerted
293                        // and can fix it in a timely manner to respect a user's choice.
294                        telemetry::event!(
295                            "Welcome Page Telemetry Diagnostics Toggled",
296                            options = if enabled { "on" } else { "off" }
297                        );
298                    }
299                },
300            )
301            .tab_index({
302                *tab_index += 1;
303                *tab_index
304            }),
305        )
306}
307
308fn render_base_keymap_section(tab_index: &mut isize, cx: &mut App) -> impl IntoElement {
309    let base_keymap = match BaseKeymap::get_global(cx) {
310        BaseKeymap::VSCode => Some(0),
311        BaseKeymap::JetBrains => Some(1),
312        BaseKeymap::SublimeText => Some(2),
313        BaseKeymap::Atom => Some(3),
314        BaseKeymap::Emacs => Some(4),
315        BaseKeymap::Cursor => Some(5),
316        BaseKeymap::TextMate | BaseKeymap::None => None,
317    };
318
319    return v_flex().gap_2().child(Label::new("Base Keymap")).child(
320        ToggleButtonGroup::two_rows(
321            "base_keymap_selection",
322            [
323                ToggleButtonWithIcon::new("VS Code", IconName::EditorVsCode, |_, _, cx| {
324                    write_keymap_base(BaseKeymap::VSCode, cx);
325                }),
326                ToggleButtonWithIcon::new("JetBrains", IconName::EditorJetBrains, |_, _, cx| {
327                    write_keymap_base(BaseKeymap::JetBrains, cx);
328                }),
329                ToggleButtonWithIcon::new("Sublime Text", IconName::EditorSublime, |_, _, cx| {
330                    write_keymap_base(BaseKeymap::SublimeText, cx);
331                }),
332            ],
333            [
334                ToggleButtonWithIcon::new("Atom", IconName::EditorAtom, |_, _, cx| {
335                    write_keymap_base(BaseKeymap::Atom, cx);
336                }),
337                ToggleButtonWithIcon::new("Emacs", IconName::EditorEmacs, |_, _, cx| {
338                    write_keymap_base(BaseKeymap::Emacs, cx);
339                }),
340                ToggleButtonWithIcon::new("Cursor", IconName::EditorCursor, |_, _, cx| {
341                    write_keymap_base(BaseKeymap::Cursor, cx);
342                }),
343            ],
344        )
345        .when_some(base_keymap, |this, base_keymap| {
346            this.selected_index(base_keymap)
347        })
348        .full_width()
349        .tab_index(tab_index)
350        .size(ui::ToggleButtonGroupSize::Medium)
351        .style(ui::ToggleButtonGroupStyle::Outlined),
352    );
353
354    fn write_keymap_base(keymap_base: BaseKeymap, cx: &App) {
355        let fs = <dyn Fs>::global(cx);
356
357        update_settings_file(fs, cx, move |setting, _| {
358            setting.base_keymap = Some(keymap_base.into());
359        });
360
361        telemetry::event!("Welcome Keymap Changed", keymap = keymap_base);
362    }
363}
364
365fn render_vim_mode_switch(tab_index: &mut isize, cx: &mut App) -> impl IntoElement {
366    let toggle_state = if VimModeSetting::get_global(cx).0 {
367        ui::ToggleState::Selected
368    } else {
369        ui::ToggleState::Unselected
370    };
371    SwitchField::new(
372        "onboarding-vim-mode",
373        Some("Vim Mode"),
374        Some("Coming from Neovim? Use our first-class implementation of Vim Mode".into()),
375        toggle_state,
376        {
377            let fs = <dyn Fs>::global(cx);
378            move |&selection, _, cx| {
379                let vim_mode = match selection {
380                    ToggleState::Selected => true,
381                    ToggleState::Unselected => false,
382                    ToggleState::Indeterminate => {
383                        return;
384                    }
385                };
386                update_settings_file(fs.clone(), cx, move |setting, _| {
387                    setting.vim_mode = Some(vim_mode);
388                });
389
390                telemetry::event!(
391                    "Welcome Vim Mode Toggled",
392                    options = if vim_mode { "on" } else { "off" },
393                );
394            }
395        },
396    )
397    .tab_index({
398        *tab_index += 1;
399        *tab_index - 1
400    })
401}
402
403fn render_setting_import_button(
404    tab_index: isize,
405    label: SharedString,
406    action: &dyn Action,
407    imported: bool,
408) -> impl IntoElement + 'static {
409    let action = action.boxed_clone();
410
411    Button::new(label.clone(), label.clone())
412        .style(ButtonStyle::OutlinedGhost)
413        .size(ButtonSize::Medium)
414        .label_size(LabelSize::Small)
415        .selected_style(ButtonStyle::Tinted(TintColor::Accent))
416        .toggle_state(imported)
417        .tab_index(tab_index)
418        .when(imported, |this| {
419            this.icon(IconName::Check)
420                .icon_size(IconSize::Small)
421                .color(Color::Success)
422        })
423        .on_click(move |_, window, cx| {
424            telemetry::event!("Welcome Import Settings", import_source = label,);
425            window.dispatch_action(action.boxed_clone(), cx);
426        })
427}
428
429fn render_import_settings_section(tab_index: &mut isize, cx: &mut App) -> impl IntoElement {
430    let import_state = SettingsImportState::global(cx);
431    let imports: [(SharedString, &dyn Action, bool); 2] = [
432        (
433            "VS Code".into(),
434            &ImportVsCodeSettings { skip_prompt: false },
435            import_state.vscode,
436        ),
437        (
438            "Cursor".into(),
439            &ImportCursorSettings { skip_prompt: false },
440            import_state.cursor,
441        ),
442    ];
443
444    let [vscode, cursor] = imports.map(|(label, action, imported)| {
445        *tab_index += 1;
446        render_setting_import_button(*tab_index - 1, label, action, imported)
447    });
448
449    h_flex()
450        .gap_2()
451        .flex_wrap()
452        .justify_between()
453        .child(
454            v_flex()
455                .gap_0p5()
456                .max_w_5_6()
457                .child(Label::new("Import Settings"))
458                .child(
459                    Label::new("Automatically pull your settings from other editors")
460                        .color(Color::Muted),
461                ),
462        )
463        .child(h_flex().gap_1().child(vscode).child(cursor))
464}
465
466pub(crate) fn render_basics_page(cx: &mut App) -> impl IntoElement {
467    let mut tab_index = 0;
468    v_flex()
469        .id("basics-page")
470        .gap_6()
471        .child(render_theme_section(&mut tab_index, cx))
472        .child(render_base_keymap_section(&mut tab_index, cx))
473        .child(render_import_settings_section(&mut tab_index, cx))
474        .child(render_vim_mode_switch(&mut tab_index, cx))
475        .child(Divider::horizontal().color(ui::DividerColor::BorderVariant))
476        .child(render_telemetry_section(&mut tab_index, cx))
477}