worktree_picker.rs

  1use anyhow::Context as _;
  2use collections::HashSet;
  3use fuzzy::StringMatchCandidate;
  4
  5use git::repository::{Worktree as GitWorktree, validate_worktree_directory};
  6use gpui::{
  7    Action, App, AsyncWindowContext, Context, DismissEvent, Entity, EventEmitter, FocusHandle,
  8    Focusable, InteractiveElement, IntoElement, Modifiers, ModifiersChangedEvent, ParentElement,
  9    Render, SharedString, Styled, Subscription, Task, WeakEntity, Window, actions, rems,
 10};
 11use picker::{Picker, PickerDelegate, PickerEditorPosition};
 12use project::project_settings::ProjectSettings;
 13use project::{
 14    git_store::Repository,
 15    trusted_worktrees::{PathTrust, TrustedWorktrees},
 16};
 17use remote::{RemoteConnectionOptions, remote_client::ConnectionIdentifier};
 18use remote_connection::{RemoteConnectionModal, connect};
 19use settings::Settings;
 20use std::{path::PathBuf, sync::Arc};
 21use ui::{HighlightedLabel, KeyBinding, ListItem, ListItemSpacing, prelude::*};
 22use util::ResultExt;
 23use workspace::{ModalView, MultiWorkspace, Workspace, notifications::DetachAndPromptErr};
 24
 25use crate::git_panel::show_error_toast;
 26
 27actions!(
 28    git,
 29    [
 30        WorktreeFromDefault,
 31        WorktreeFromDefaultOnWindow,
 32        DeleteWorktree
 33    ]
 34);
 35
 36pub fn open(
 37    workspace: &mut Workspace,
 38    _: &zed_actions::git::Worktree,
 39    window: &mut Window,
 40    cx: &mut Context<Workspace>,
 41) {
 42    let repository = workspace.project().read(cx).active_repository(cx);
 43    let workspace_handle = workspace.weak_handle();
 44    workspace.toggle_modal(window, cx, |window, cx| {
 45        WorktreeList::new(repository, workspace_handle, rems(34.), window, cx)
 46    })
 47}
 48
 49pub fn create_embedded(
 50    repository: Option<Entity<Repository>>,
 51    workspace: WeakEntity<Workspace>,
 52    width: Rems,
 53    window: &mut Window,
 54    cx: &mut Context<WorktreeList>,
 55) -> WorktreeList {
 56    WorktreeList::new_embedded(repository, workspace, width, window, cx)
 57}
 58
 59pub struct WorktreeList {
 60    width: Rems,
 61    pub picker: Entity<Picker<WorktreeListDelegate>>,
 62    picker_focus_handle: FocusHandle,
 63    _subscription: Option<Subscription>,
 64    embedded: bool,
 65}
 66
 67impl WorktreeList {
 68    fn new(
 69        repository: Option<Entity<Repository>>,
 70        workspace: WeakEntity<Workspace>,
 71        width: Rems,
 72        window: &mut Window,
 73        cx: &mut Context<Self>,
 74    ) -> Self {
 75        let mut this = Self::new_inner(repository, workspace, width, false, window, cx);
 76        this._subscription = Some(cx.subscribe(&this.picker, |_, _, _, cx| {
 77            cx.emit(DismissEvent);
 78        }));
 79        this
 80    }
 81
 82    fn new_inner(
 83        repository: Option<Entity<Repository>>,
 84        workspace: WeakEntity<Workspace>,
 85        width: Rems,
 86        embedded: bool,
 87        window: &mut Window,
 88        cx: &mut Context<Self>,
 89    ) -> Self {
 90        let all_worktrees_request = repository
 91            .clone()
 92            .map(|repository| repository.update(cx, |repository, _| repository.worktrees()));
 93
 94        let default_branch_request = repository.clone().map(|repository| {
 95            repository.update(cx, |repository, _| repository.default_branch(false))
 96        });
 97
 98        cx.spawn_in(window, async move |this, cx| {
 99            let all_worktrees = all_worktrees_request
100                .context("No active repository")?
101                .await??;
102
103            let default_branch = default_branch_request
104                .context("No active repository")?
105                .await
106                .map(Result::ok)
107                .ok()
108                .flatten()
109                .flatten();
110
111            this.update_in(cx, |this, window, cx| {
112                this.picker.update(cx, |picker, cx| {
113                    picker.delegate.all_worktrees = Some(all_worktrees);
114                    picker.delegate.default_branch = default_branch;
115                    picker.refresh(window, cx);
116                })
117            })?;
118
119            anyhow::Ok(())
120        })
121        .detach_and_log_err(cx);
122
123        let delegate = WorktreeListDelegate::new(workspace, repository, window, cx);
124        let picker = cx.new(|cx| {
125            Picker::uniform_list(delegate, window, cx)
126                .show_scrollbar(true)
127                .modal(!embedded)
128        });
129        let picker_focus_handle = picker.focus_handle(cx);
130        picker.update(cx, |picker, _| {
131            picker.delegate.focus_handle = picker_focus_handle.clone();
132        });
133
134        Self {
135            picker,
136            picker_focus_handle,
137            width,
138            _subscription: None,
139            embedded,
140        }
141    }
142
143    fn new_embedded(
144        repository: Option<Entity<Repository>>,
145        workspace: WeakEntity<Workspace>,
146        width: Rems,
147        window: &mut Window,
148        cx: &mut Context<Self>,
149    ) -> Self {
150        let mut this = Self::new_inner(repository, workspace, width, true, window, cx);
151        this._subscription = Some(cx.subscribe(&this.picker, |_, _, _, cx| {
152            cx.emit(DismissEvent);
153        }));
154        this
155    }
156
157    pub fn handle_modifiers_changed(
158        &mut self,
159        ev: &ModifiersChangedEvent,
160        _: &mut Window,
161        cx: &mut Context<Self>,
162    ) {
163        self.picker
164            .update(cx, |picker, _| picker.delegate.modifiers = ev.modifiers)
165    }
166
167    pub fn handle_new_worktree(
168        &mut self,
169        replace_current_window: bool,
170        window: &mut Window,
171        cx: &mut Context<Self>,
172    ) {
173        self.picker.update(cx, |picker, cx| {
174            let ix = picker.delegate.selected_index();
175            let Some(entry) = picker.delegate.matches.get(ix) else {
176                return;
177            };
178            let Some(default_branch) = picker.delegate.default_branch.clone() else {
179                return;
180            };
181            if !entry.is_new {
182                return;
183            }
184            picker.delegate.create_worktree(
185                entry.worktree.branch(),
186                replace_current_window,
187                Some(default_branch.into()),
188                window,
189                cx,
190            );
191        })
192    }
193
194    pub fn handle_delete(
195        &mut self,
196        _: &DeleteWorktree,
197        window: &mut Window,
198        cx: &mut Context<Self>,
199    ) {
200        self.picker.update(cx, |picker, cx| {
201            picker
202                .delegate
203                .delete_at(picker.delegate.selected_index, window, cx)
204        })
205    }
206}
207impl ModalView for WorktreeList {}
208impl EventEmitter<DismissEvent> for WorktreeList {}
209
210impl Focusable for WorktreeList {
211    fn focus_handle(&self, _: &App) -> FocusHandle {
212        self.picker_focus_handle.clone()
213    }
214}
215
216impl Render for WorktreeList {
217    fn render(&mut self, _: &mut Window, cx: &mut Context<Self>) -> impl IntoElement {
218        v_flex()
219            .key_context("GitWorktreeSelector")
220            .w(self.width)
221            .on_modifiers_changed(cx.listener(Self::handle_modifiers_changed))
222            .on_action(cx.listener(|this, _: &WorktreeFromDefault, w, cx| {
223                this.handle_new_worktree(false, w, cx)
224            }))
225            .on_action(cx.listener(|this, _: &WorktreeFromDefaultOnWindow, w, cx| {
226                this.handle_new_worktree(true, w, cx)
227            }))
228            .on_action(cx.listener(|this, _: &DeleteWorktree, window, cx| {
229                this.handle_delete(&DeleteWorktree, window, cx)
230            }))
231            .child(self.picker.clone())
232            .when(!self.embedded, |el| {
233                el.on_mouse_down_out({
234                    cx.listener(move |this, _, window, cx| {
235                        this.picker.update(cx, |this, cx| {
236                            this.cancel(&Default::default(), window, cx);
237                        })
238                    })
239                })
240            })
241    }
242}
243
244#[derive(Debug, Clone)]
245struct WorktreeEntry {
246    worktree: GitWorktree,
247    positions: Vec<usize>,
248    is_new: bool,
249}
250
251pub struct WorktreeListDelegate {
252    matches: Vec<WorktreeEntry>,
253    all_worktrees: Option<Vec<GitWorktree>>,
254    workspace: WeakEntity<Workspace>,
255    repo: Option<Entity<Repository>>,
256    selected_index: usize,
257    last_query: String,
258    modifiers: Modifiers,
259    focus_handle: FocusHandle,
260    default_branch: Option<SharedString>,
261}
262
263impl WorktreeListDelegate {
264    fn new(
265        workspace: WeakEntity<Workspace>,
266        repo: Option<Entity<Repository>>,
267        _window: &mut Window,
268        cx: &mut Context<WorktreeList>,
269    ) -> Self {
270        Self {
271            matches: vec![],
272            all_worktrees: None,
273            workspace,
274            selected_index: 0,
275            repo,
276            last_query: Default::default(),
277            modifiers: Default::default(),
278            focus_handle: cx.focus_handle(),
279            default_branch: None,
280        }
281    }
282
283    fn create_worktree(
284        &self,
285        worktree_branch: &str,
286        replace_current_window: bool,
287        commit: Option<String>,
288        window: &mut Window,
289        cx: &mut Context<Picker<Self>>,
290    ) {
291        let Some(repo) = self.repo.clone() else {
292            return;
293        };
294
295        let branch = worktree_branch.to_string();
296        let workspace = self.workspace.clone();
297        cx.spawn_in(window, async move |_, cx| {
298            let (receiver, new_worktree_path) = repo.update(cx, |repo, cx| {
299                let worktree_directory_setting = ProjectSettings::get_global(cx)
300                    .git
301                    .worktree_directory
302                    .clone();
303                let original_repo = repo.original_repo_abs_path.clone();
304                let directory =
305                    validate_worktree_directory(&original_repo, &worktree_directory_setting)?;
306                let new_worktree_path = directory.join(&branch);
307                let receiver = repo.create_worktree(branch.clone(), directory, commit);
308                anyhow::Ok((receiver, new_worktree_path))
309            })?;
310            receiver.await??;
311
312            workspace.update(cx, |workspace, cx| {
313                if let Some(trusted_worktrees) = TrustedWorktrees::try_get_global(cx) {
314                    let repo_path = &repo.read(cx).snapshot().work_directory_abs_path;
315                    let project = workspace.project();
316                    if let Some((parent_worktree, _)) =
317                        project.read(cx).find_worktree(repo_path, cx)
318                    {
319                        let worktree_store = project.read(cx).worktree_store();
320                        trusted_worktrees.update(cx, |trusted_worktrees, cx| {
321                            if trusted_worktrees.can_trust(
322                                &worktree_store,
323                                parent_worktree.read(cx).id(),
324                                cx,
325                            ) {
326                                trusted_worktrees.trust(
327                                    &worktree_store,
328                                    HashSet::from_iter([PathTrust::AbsPath(
329                                        new_worktree_path.clone(),
330                                    )]),
331                                    cx,
332                                );
333                            }
334                        });
335                    }
336                }
337            })?;
338
339            let (connection_options, app_state, is_local) =
340                workspace.update(cx, |workspace, cx| {
341                    let project = workspace.project().clone();
342                    let connection_options = project.read(cx).remote_connection_options(cx);
343                    let app_state = workspace.app_state().clone();
344                    let is_local = project.read(cx).is_local();
345                    (connection_options, app_state, is_local)
346                })?;
347
348            if is_local {
349                workspace
350                    .update_in(cx, |workspace, window, cx| {
351                        workspace.open_workspace_for_paths(
352                            replace_current_window,
353                            vec![new_worktree_path],
354                            window,
355                            cx,
356                        )
357                    })?
358                    .await?;
359            } else if let Some(connection_options) = connection_options {
360                open_remote_worktree(
361                    connection_options,
362                    vec![new_worktree_path],
363                    app_state,
364                    workspace.clone(),
365                    replace_current_window,
366                    cx,
367                )
368                .await?;
369            }
370
371            anyhow::Ok(())
372        })
373        .detach_and_prompt_err("Failed to create worktree", window, cx, |e, _, _| {
374            let msg = e.to_string();
375            if msg.contains("git.worktree_directory") {
376                Some(format!("Invalid git.worktree_directory setting: {}", e))
377            } else {
378                Some(msg)
379            }
380        });
381    }
382
383    fn open_worktree(
384        &self,
385        worktree_path: &PathBuf,
386        replace_current_window: bool,
387        window: &mut Window,
388        cx: &mut Context<Picker<Self>>,
389    ) {
390        let workspace = self.workspace.clone();
391        let path = worktree_path.clone();
392
393        let Some((connection_options, app_state, is_local)) = workspace
394            .update(cx, |workspace, cx| {
395                let project = workspace.project().clone();
396                let connection_options = project.read(cx).remote_connection_options(cx);
397                let app_state = workspace.app_state().clone();
398                let is_local = project.read(cx).is_local();
399                (connection_options, app_state, is_local)
400            })
401            .log_err()
402        else {
403            return;
404        };
405
406        if is_local {
407            let open_task = workspace.update(cx, |workspace, cx| {
408                workspace.open_workspace_for_paths(replace_current_window, vec![path], window, cx)
409            });
410            cx.spawn(async move |_, _| {
411                open_task?.await?;
412                anyhow::Ok(())
413            })
414            .detach_and_prompt_err(
415                "Failed to open worktree",
416                window,
417                cx,
418                |e, _, _| Some(e.to_string()),
419            );
420        } else if let Some(connection_options) = connection_options {
421            cx.spawn_in(window, async move |_, cx| {
422                open_remote_worktree(
423                    connection_options,
424                    vec![path],
425                    app_state,
426                    workspace,
427                    replace_current_window,
428                    cx,
429                )
430                .await
431            })
432            .detach_and_prompt_err(
433                "Failed to open worktree",
434                window,
435                cx,
436                |e, _, _| Some(e.to_string()),
437            );
438        }
439
440        cx.emit(DismissEvent);
441    }
442
443    fn base_branch<'a>(&'a self, cx: &'a mut Context<Picker<Self>>) -> Option<&'a str> {
444        self.repo
445            .as_ref()
446            .and_then(|repo| repo.read(cx).branch.as_ref().map(|b| b.name()))
447    }
448
449    fn delete_at(&self, idx: usize, window: &mut Window, cx: &mut Context<Picker<Self>>) {
450        let Some(entry) = self.matches.get(idx).cloned() else {
451            return;
452        };
453        if entry.is_new {
454            return;
455        }
456        let Some(repo) = self.repo.clone() else {
457            return;
458        };
459        let workspace = self.workspace.clone();
460        let path = entry.worktree.path;
461
462        cx.spawn_in(window, async move |picker, cx| {
463            let result = repo
464                .update(cx, |repo, _| repo.remove_worktree(path.clone(), false))
465                .await?;
466
467            if let Err(e) = result {
468                log::error!("Failed to remove worktree: {}", e);
469                if let Some(workspace) = workspace.upgrade() {
470                    cx.update(|_window, cx| {
471                        show_error_toast(
472                            workspace,
473                            format!("worktree remove {}", path.display()),
474                            e,
475                            cx,
476                        )
477                    })?;
478                }
479                return Ok(());
480            }
481
482            picker.update_in(cx, |picker, _, cx| {
483                picker.delegate.matches.retain(|e| e.worktree.path != path);
484                if let Some(all_worktrees) = &mut picker.delegate.all_worktrees {
485                    all_worktrees.retain(|w| w.path != path);
486                }
487                if picker.delegate.matches.is_empty() {
488                    picker.delegate.selected_index = 0;
489                } else if picker.delegate.selected_index >= picker.delegate.matches.len() {
490                    picker.delegate.selected_index = picker.delegate.matches.len() - 1;
491                }
492                cx.notify();
493            })?;
494
495            anyhow::Ok(())
496        })
497        .detach();
498    }
499}
500
501async fn open_remote_worktree(
502    connection_options: RemoteConnectionOptions,
503    paths: Vec<PathBuf>,
504    app_state: Arc<workspace::AppState>,
505    workspace: WeakEntity<Workspace>,
506    replace_current_window: bool,
507    cx: &mut AsyncWindowContext,
508) -> anyhow::Result<()> {
509    let workspace_window = cx
510        .window_handle()
511        .downcast::<MultiWorkspace>()
512        .ok_or_else(|| anyhow::anyhow!("Window is not a Workspace window"))?;
513
514    let connect_task = workspace.update_in(cx, |workspace, window, cx| {
515        workspace.toggle_modal(window, cx, |window, cx| {
516            RemoteConnectionModal::new(&connection_options, Vec::new(), window, cx)
517        });
518
519        let prompt = workspace
520            .active_modal::<RemoteConnectionModal>(cx)
521            .expect("Modal just created")
522            .read(cx)
523            .prompt
524            .clone();
525
526        connect(
527            ConnectionIdentifier::setup(),
528            connection_options.clone(),
529            prompt,
530            window,
531            cx,
532        )
533        .prompt_err("Failed to connect", window, cx, |_, _, _| None)
534    })?;
535
536    let session = connect_task.await;
537
538    workspace
539        .update_in(cx, |workspace, _window, cx| {
540            if let Some(prompt) = workspace.active_modal::<RemoteConnectionModal>(cx) {
541                prompt.update(cx, |prompt, cx| prompt.finished(cx))
542            }
543        })
544        .ok();
545
546    let Some(Some(session)) = session else {
547        return Ok(());
548    };
549
550    let new_project: Entity<project::Project> = cx.update(|_, cx| {
551        project::Project::remote(
552            session,
553            app_state.client.clone(),
554            app_state.node_runtime.clone(),
555            app_state.user_store.clone(),
556            app_state.languages.clone(),
557            app_state.fs.clone(),
558            true,
559            cx,
560        )
561    })?;
562
563    let window_to_use = if replace_current_window {
564        workspace_window
565    } else {
566        let workspace_position = cx
567            .update(|_, cx| {
568                workspace::remote_workspace_position_from_db(connection_options.clone(), &paths, cx)
569            })?
570            .await
571            .context("fetching workspace position from db")?;
572
573        let mut options =
574            cx.update(|_, cx| (app_state.build_window_options)(workspace_position.display, cx))?;
575        options.window_bounds = workspace_position.window_bounds;
576
577        cx.open_window(options, |window, cx| {
578            let workspace = cx.new(|cx| {
579                let mut workspace =
580                    Workspace::new(None, new_project.clone(), app_state.clone(), window, cx);
581                workspace.centered_layout = workspace_position.centered_layout;
582                workspace
583            });
584            cx.new(|cx| MultiWorkspace::new(workspace, window, cx))
585        })?
586    };
587
588    workspace::open_remote_project_with_existing_connection(
589        connection_options,
590        new_project,
591        paths,
592        app_state,
593        window_to_use,
594        cx,
595    )
596    .await?;
597
598    Ok(())
599}
600
601impl PickerDelegate for WorktreeListDelegate {
602    type ListItem = ListItem;
603
604    fn placeholder_text(&self, _window: &mut Window, _cx: &mut App) -> Arc<str> {
605        "Select worktree…".into()
606    }
607
608    fn editor_position(&self) -> PickerEditorPosition {
609        PickerEditorPosition::Start
610    }
611
612    fn match_count(&self) -> usize {
613        self.matches.len()
614    }
615
616    fn selected_index(&self) -> usize {
617        self.selected_index
618    }
619
620    fn set_selected_index(
621        &mut self,
622        ix: usize,
623        _window: &mut Window,
624        _: &mut Context<Picker<Self>>,
625    ) {
626        self.selected_index = ix;
627    }
628
629    fn update_matches(
630        &mut self,
631        query: String,
632        window: &mut Window,
633        cx: &mut Context<Picker<Self>>,
634    ) -> Task<()> {
635        let Some(all_worktrees) = self.all_worktrees.clone() else {
636            return Task::ready(());
637        };
638
639        cx.spawn_in(window, async move |picker, cx| {
640            let mut matches: Vec<WorktreeEntry> = if query.is_empty() {
641                all_worktrees
642                    .into_iter()
643                    .map(|worktree| WorktreeEntry {
644                        worktree,
645                        positions: Vec::new(),
646                        is_new: false,
647                    })
648                    .collect()
649            } else {
650                let candidates = all_worktrees
651                    .iter()
652                    .enumerate()
653                    .map(|(ix, worktree)| StringMatchCandidate::new(ix, worktree.branch()))
654                    .collect::<Vec<StringMatchCandidate>>();
655                fuzzy::match_strings(
656                    &candidates,
657                    &query,
658                    true,
659                    true,
660                    10000,
661                    &Default::default(),
662                    cx.background_executor().clone(),
663                )
664                .await
665                .into_iter()
666                .map(|candidate| WorktreeEntry {
667                    worktree: all_worktrees[candidate.candidate_id].clone(),
668                    positions: candidate.positions,
669                    is_new: false,
670                })
671                .collect()
672            };
673            picker
674                .update(cx, |picker, _| {
675                    if !query.is_empty()
676                        && !matches
677                            .first()
678                            .is_some_and(|entry| entry.worktree.branch() == query)
679                    {
680                        let query = query.replace(' ', "-");
681                        matches.push(WorktreeEntry {
682                            worktree: GitWorktree {
683                                path: Default::default(),
684                                ref_name: format!("refs/heads/{query}").into(),
685                                sha: Default::default(),
686                            },
687                            positions: Vec::new(),
688                            is_new: true,
689                        })
690                    }
691                    let delegate = &mut picker.delegate;
692                    delegate.matches = matches;
693                    if delegate.matches.is_empty() {
694                        delegate.selected_index = 0;
695                    } else {
696                        delegate.selected_index =
697                            core::cmp::min(delegate.selected_index, delegate.matches.len() - 1);
698                    }
699                    delegate.last_query = query;
700                })
701                .log_err();
702        })
703    }
704
705    fn confirm(&mut self, secondary: bool, window: &mut Window, cx: &mut Context<Picker<Self>>) {
706        let Some(entry) = self.matches.get(self.selected_index()) else {
707            return;
708        };
709        if entry.is_new {
710            self.create_worktree(&entry.worktree.branch(), secondary, None, window, cx);
711        } else {
712            self.open_worktree(&entry.worktree.path, secondary, window, cx);
713        }
714
715        cx.emit(DismissEvent);
716    }
717
718    fn dismissed(&mut self, _: &mut Window, cx: &mut Context<Picker<Self>>) {
719        cx.emit(DismissEvent);
720    }
721
722    fn render_match(
723        &self,
724        ix: usize,
725        selected: bool,
726        _window: &mut Window,
727        cx: &mut Context<Picker<Self>>,
728    ) -> Option<Self::ListItem> {
729        let entry = &self.matches.get(ix)?;
730        let path = entry.worktree.path.to_string_lossy().to_string();
731        let sha = entry
732            .worktree
733            .sha
734            .clone()
735            .chars()
736            .take(7)
737            .collect::<String>();
738
739        let (branch_name, sublabel) = if entry.is_new {
740            (
741                Label::new(format!("Create Worktree: \"{}\"", entry.worktree.branch()))
742                    .truncate()
743                    .into_any_element(),
744                format!(
745                    "based off {}",
746                    self.base_branch(cx).unwrap_or("the current branch")
747                ),
748            )
749        } else {
750            let branch = entry.worktree.branch();
751            let branch_first_line = branch.lines().next().unwrap_or(branch);
752            let positions: Vec<_> = entry
753                .positions
754                .iter()
755                .copied()
756                .filter(|&pos| pos < branch_first_line.len())
757                .collect();
758
759            (
760                HighlightedLabel::new(branch_first_line.to_owned(), positions)
761                    .truncate()
762                    .into_any_element(),
763                path,
764            )
765        };
766
767        Some(
768            ListItem::new(format!("worktree-menu-{ix}"))
769                .inset(true)
770                .spacing(ListItemSpacing::Sparse)
771                .toggle_state(selected)
772                .child(
773                    v_flex()
774                        .w_full()
775                        .child(
776                            h_flex()
777                                .gap_2()
778                                .justify_between()
779                                .overflow_x_hidden()
780                                .child(branch_name)
781                                .when(!entry.is_new, |this| {
782                                    this.child(
783                                        Label::new(sha)
784                                            .size(LabelSize::Small)
785                                            .color(Color::Muted)
786                                            .buffer_font(cx)
787                                            .into_element(),
788                                    )
789                                }),
790                        )
791                        .child(
792                            Label::new(sublabel)
793                                .size(LabelSize::Small)
794                                .color(Color::Muted)
795                                .truncate()
796                                .into_any_element(),
797                        ),
798                ),
799        )
800    }
801
802    fn no_matches_text(&self, _window: &mut Window, _cx: &mut App) -> Option<SharedString> {
803        Some("No worktrees found".into())
804    }
805
806    fn render_footer(&self, _: &mut Window, cx: &mut Context<Picker<Self>>) -> Option<AnyElement> {
807        let focus_handle = self.focus_handle.clone();
808        let selected_entry = self.matches.get(self.selected_index);
809        let is_creating = selected_entry.is_some_and(|entry| entry.is_new);
810
811        let footer_container = h_flex()
812            .w_full()
813            .p_1p5()
814            .gap_0p5()
815            .justify_end()
816            .border_t_1()
817            .border_color(cx.theme().colors().border_variant);
818
819        if is_creating {
820            let from_default_button = self.default_branch.as_ref().map(|default_branch| {
821                Button::new(
822                    "worktree-from-default",
823                    format!("Create from: {default_branch}"),
824                )
825                .key_binding(
826                    KeyBinding::for_action_in(&WorktreeFromDefault, &focus_handle, cx)
827                        .map(|kb| kb.size(rems_from_px(12.))),
828                )
829                .on_click(|_, window, cx| {
830                    window.dispatch_action(WorktreeFromDefault.boxed_clone(), cx)
831                })
832            });
833
834            let current_branch = self.base_branch(cx).unwrap_or("current branch");
835
836            Some(
837                footer_container
838                    .when_some(from_default_button, |this, button| this.child(button))
839                    .child(
840                        Button::new(
841                            "worktree-from-current",
842                            format!("Create from: {current_branch}"),
843                        )
844                        .key_binding(
845                            KeyBinding::for_action_in(&menu::Confirm, &focus_handle, cx)
846                                .map(|kb| kb.size(rems_from_px(12.))),
847                        )
848                        .on_click(|_, window, cx| {
849                            window.dispatch_action(menu::Confirm.boxed_clone(), cx)
850                        }),
851                    )
852                    .into_any(),
853            )
854        } else {
855            Some(
856                footer_container
857                    .child(
858                        Button::new("delete-worktree", "Delete")
859                            .key_binding(
860                                KeyBinding::for_action_in(&DeleteWorktree, &focus_handle, cx)
861                                    .map(|kb| kb.size(rems_from_px(12.))),
862                            )
863                            .on_click(|_, window, cx| {
864                                window.dispatch_action(DeleteWorktree.boxed_clone(), cx)
865                            }),
866                    )
867                    .child(
868                        Button::new("open-in-new-window", "Open in New Window")
869                            .key_binding(
870                                KeyBinding::for_action_in(&menu::Confirm, &focus_handle, cx)
871                                    .map(|kb| kb.size(rems_from_px(12.))),
872                            )
873                            .on_click(|_, window, cx| {
874                                window.dispatch_action(menu::Confirm.boxed_clone(), cx)
875                            }),
876                    )
877                    .child(
878                        Button::new("open-in-window", "Open")
879                            .key_binding(
880                                KeyBinding::for_action_in(
881                                    &menu::SecondaryConfirm,
882                                    &focus_handle,
883                                    cx,
884                                )
885                                .map(|kb| kb.size(rems_from_px(12.))),
886                            )
887                            .on_click(|_, window, cx| {
888                                window.dispatch_action(menu::SecondaryConfirm.boxed_clone(), cx)
889                            }),
890                    )
891                    .into_any(),
892            )
893        }
894    }
895}