worktree_picker.rs

  1use anyhow::Context as _;
  2use collections::HashSet;
  3use fuzzy::StringMatchCandidate;
  4
  5use git::repository::Worktree as GitWorktree;
  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 new_worktree_path =
304                    repo.path_for_new_linked_worktree(&branch, &worktree_directory_setting)?;
305                let receiver =
306                    repo.create_worktree(branch.clone(), new_worktree_path.clone(), commit);
307                anyhow::Ok((receiver, new_worktree_path))
308            })?;
309            receiver.await??;
310
311            workspace.update(cx, |workspace, cx| {
312                if let Some(trusted_worktrees) = TrustedWorktrees::try_get_global(cx) {
313                    let repo_path = &repo.read(cx).snapshot().work_directory_abs_path;
314                    let project = workspace.project();
315                    if let Some((parent_worktree, _)) =
316                        project.read(cx).find_worktree(repo_path, cx)
317                    {
318                        let worktree_store = project.read(cx).worktree_store();
319                        trusted_worktrees.update(cx, |trusted_worktrees, cx| {
320                            if trusted_worktrees.can_trust(
321                                &worktree_store,
322                                parent_worktree.read(cx).id(),
323                                cx,
324                            ) {
325                                trusted_worktrees.trust(
326                                    &worktree_store,
327                                    HashSet::from_iter([PathTrust::AbsPath(
328                                        new_worktree_path.clone(),
329                                    )]),
330                                    cx,
331                                );
332                            }
333                        });
334                    }
335                }
336            })?;
337
338            let (connection_options, app_state, is_local) =
339                workspace.update(cx, |workspace, cx| {
340                    let project = workspace.project().clone();
341                    let connection_options = project.read(cx).remote_connection_options(cx);
342                    let app_state = workspace.app_state().clone();
343                    let is_local = project.read(cx).is_local();
344                    (connection_options, app_state, is_local)
345                })?;
346
347            if is_local {
348                workspace
349                    .update_in(cx, |workspace, window, cx| {
350                        workspace.open_workspace_for_paths(
351                            replace_current_window,
352                            vec![new_worktree_path],
353                            window,
354                            cx,
355                        )
356                    })?
357                    .await?;
358            } else if let Some(connection_options) = connection_options {
359                open_remote_worktree(
360                    connection_options,
361                    vec![new_worktree_path],
362                    app_state,
363                    workspace.clone(),
364                    replace_current_window,
365                    cx,
366                )
367                .await?;
368            }
369
370            anyhow::Ok(())
371        })
372        .detach_and_prompt_err("Failed to create worktree", window, cx, |e, _, _| {
373            let msg = e.to_string();
374            if msg.contains("git.worktree_directory") {
375                Some(format!("Invalid git.worktree_directory setting: {}", e))
376            } else {
377                Some(msg)
378            }
379        });
380    }
381
382    fn open_worktree(
383        &self,
384        worktree_path: &PathBuf,
385        replace_current_window: bool,
386        window: &mut Window,
387        cx: &mut Context<Picker<Self>>,
388    ) {
389        let workspace = self.workspace.clone();
390        let path = worktree_path.clone();
391
392        let Some((connection_options, app_state, is_local)) = workspace
393            .update(cx, |workspace, cx| {
394                let project = workspace.project().clone();
395                let connection_options = project.read(cx).remote_connection_options(cx);
396                let app_state = workspace.app_state().clone();
397                let is_local = project.read(cx).is_local();
398                (connection_options, app_state, is_local)
399            })
400            .log_err()
401        else {
402            return;
403        };
404
405        if is_local {
406            let open_task = workspace.update(cx, |workspace, cx| {
407                workspace.open_workspace_for_paths(replace_current_window, vec![path], window, cx)
408            });
409            cx.spawn(async move |_, _| {
410                open_task?.await?;
411                anyhow::Ok(())
412            })
413            .detach_and_prompt_err(
414                "Failed to open worktree",
415                window,
416                cx,
417                |e, _, _| Some(e.to_string()),
418            );
419        } else if let Some(connection_options) = connection_options {
420            cx.spawn_in(window, async move |_, cx| {
421                open_remote_worktree(
422                    connection_options,
423                    vec![path],
424                    app_state,
425                    workspace,
426                    replace_current_window,
427                    cx,
428                )
429                .await
430            })
431            .detach_and_prompt_err(
432                "Failed to open worktree",
433                window,
434                cx,
435                |e, _, _| Some(e.to_string()),
436            );
437        }
438
439        cx.emit(DismissEvent);
440    }
441
442    fn base_branch<'a>(&'a self, cx: &'a mut Context<Picker<Self>>) -> Option<&'a str> {
443        self.repo
444            .as_ref()
445            .and_then(|repo| repo.read(cx).branch.as_ref().map(|b| b.name()))
446    }
447
448    fn delete_at(&self, idx: usize, window: &mut Window, cx: &mut Context<Picker<Self>>) {
449        let Some(entry) = self.matches.get(idx).cloned() else {
450            return;
451        };
452        if entry.is_new {
453            return;
454        }
455        let Some(repo) = self.repo.clone() else {
456            return;
457        };
458        let workspace = self.workspace.clone();
459        let path = entry.worktree.path;
460
461        cx.spawn_in(window, async move |picker, cx| {
462            let result = repo
463                .update(cx, |repo, _| repo.remove_worktree(path.clone(), false))
464                .await?;
465
466            if let Err(e) = result {
467                log::error!("Failed to remove worktree: {}", e);
468                if let Some(workspace) = workspace.upgrade() {
469                    cx.update(|_window, cx| {
470                        show_error_toast(
471                            workspace,
472                            format!("worktree remove {}", path.display()),
473                            e,
474                            cx,
475                        )
476                    })?;
477                }
478                return Ok(());
479            }
480
481            picker.update_in(cx, |picker, _, cx| {
482                picker.delegate.matches.retain(|e| e.worktree.path != path);
483                if let Some(all_worktrees) = &mut picker.delegate.all_worktrees {
484                    all_worktrees.retain(|w| w.path != path);
485                }
486                if picker.delegate.matches.is_empty() {
487                    picker.delegate.selected_index = 0;
488                } else if picker.delegate.selected_index >= picker.delegate.matches.len() {
489                    picker.delegate.selected_index = picker.delegate.matches.len() - 1;
490                }
491                cx.notify();
492            })?;
493
494            anyhow::Ok(())
495        })
496        .detach();
497    }
498}
499
500async fn open_remote_worktree(
501    connection_options: RemoteConnectionOptions,
502    paths: Vec<PathBuf>,
503    app_state: Arc<workspace::AppState>,
504    workspace: WeakEntity<Workspace>,
505    replace_current_window: bool,
506    cx: &mut AsyncWindowContext,
507) -> anyhow::Result<()> {
508    let workspace_window = cx
509        .window_handle()
510        .downcast::<MultiWorkspace>()
511        .ok_or_else(|| anyhow::anyhow!("Window is not a Workspace window"))?;
512
513    let connect_task = workspace.update_in(cx, |workspace, window, cx| {
514        workspace.toggle_modal(window, cx, |window, cx| {
515            RemoteConnectionModal::new(&connection_options, Vec::new(), window, cx)
516        });
517
518        let prompt = workspace
519            .active_modal::<RemoteConnectionModal>(cx)
520            .expect("Modal just created")
521            .read(cx)
522            .prompt
523            .clone();
524
525        connect(
526            ConnectionIdentifier::setup(),
527            connection_options.clone(),
528            prompt,
529            window,
530            cx,
531        )
532        .prompt_err("Failed to connect", window, cx, |_, _, _| None)
533    })?;
534
535    let session = connect_task.await;
536
537    workspace
538        .update_in(cx, |workspace, _window, cx| {
539            if let Some(prompt) = workspace.active_modal::<RemoteConnectionModal>(cx) {
540                prompt.update(cx, |prompt, cx| prompt.finished(cx))
541            }
542        })
543        .ok();
544
545    let Some(Some(session)) = session else {
546        return Ok(());
547    };
548
549    let new_project: Entity<project::Project> = cx.update(|_, cx| {
550        project::Project::remote(
551            session,
552            app_state.client.clone(),
553            app_state.node_runtime.clone(),
554            app_state.user_store.clone(),
555            app_state.languages.clone(),
556            app_state.fs.clone(),
557            true,
558            cx,
559        )
560    })?;
561
562    let window_to_use = if replace_current_window {
563        workspace_window
564    } else {
565        let workspace_position = cx
566            .update(|_, cx| {
567                workspace::remote_workspace_position_from_db(connection_options.clone(), &paths, cx)
568            })?
569            .await
570            .context("fetching workspace position from db")?;
571
572        let mut options =
573            cx.update(|_, cx| (app_state.build_window_options)(workspace_position.display, cx))?;
574        options.window_bounds = workspace_position.window_bounds;
575
576        cx.open_window(options, |window, cx| {
577            let workspace = cx.new(|cx| {
578                let mut workspace =
579                    Workspace::new(None, new_project.clone(), app_state.clone(), window, cx);
580                workspace.centered_layout = workspace_position.centered_layout;
581                workspace
582            });
583            cx.new(|cx| MultiWorkspace::new(workspace, window, cx))
584        })?
585    };
586
587    workspace::open_remote_project_with_existing_connection(
588        connection_options,
589        new_project,
590        paths,
591        app_state,
592        window_to_use,
593        cx,
594    )
595    .await?;
596
597    Ok(())
598}
599
600impl PickerDelegate for WorktreeListDelegate {
601    type ListItem = ListItem;
602
603    fn placeholder_text(&self, _window: &mut Window, _cx: &mut App) -> Arc<str> {
604        "Select worktree…".into()
605    }
606
607    fn editor_position(&self) -> PickerEditorPosition {
608        PickerEditorPosition::Start
609    }
610
611    fn match_count(&self) -> usize {
612        self.matches.len()
613    }
614
615    fn selected_index(&self) -> usize {
616        self.selected_index
617    }
618
619    fn set_selected_index(
620        &mut self,
621        ix: usize,
622        _window: &mut Window,
623        _: &mut Context<Picker<Self>>,
624    ) {
625        self.selected_index = ix;
626    }
627
628    fn update_matches(
629        &mut self,
630        query: String,
631        window: &mut Window,
632        cx: &mut Context<Picker<Self>>,
633    ) -> Task<()> {
634        let Some(all_worktrees) = self.all_worktrees.clone() else {
635            return Task::ready(());
636        };
637
638        cx.spawn_in(window, async move |picker, cx| {
639            let mut matches: Vec<WorktreeEntry> = if query.is_empty() {
640                all_worktrees
641                    .into_iter()
642                    .map(|worktree| WorktreeEntry {
643                        worktree,
644                        positions: Vec::new(),
645                        is_new: false,
646                    })
647                    .collect()
648            } else {
649                let candidates = all_worktrees
650                    .iter()
651                    .enumerate()
652                    .map(|(ix, worktree)| StringMatchCandidate::new(ix, worktree.branch()))
653                    .collect::<Vec<StringMatchCandidate>>();
654                fuzzy::match_strings(
655                    &candidates,
656                    &query,
657                    true,
658                    true,
659                    10000,
660                    &Default::default(),
661                    cx.background_executor().clone(),
662                )
663                .await
664                .into_iter()
665                .map(|candidate| WorktreeEntry {
666                    worktree: all_worktrees[candidate.candidate_id].clone(),
667                    positions: candidate.positions,
668                    is_new: false,
669                })
670                .collect()
671            };
672            picker
673                .update(cx, |picker, _| {
674                    if !query.is_empty()
675                        && !matches
676                            .first()
677                            .is_some_and(|entry| entry.worktree.branch() == query)
678                    {
679                        let query = query.replace(' ', "-");
680                        matches.push(WorktreeEntry {
681                            worktree: GitWorktree {
682                                path: Default::default(),
683                                ref_name: format!("refs/heads/{query}").into(),
684                                sha: Default::default(),
685                            },
686                            positions: Vec::new(),
687                            is_new: true,
688                        })
689                    }
690                    let delegate = &mut picker.delegate;
691                    delegate.matches = matches;
692                    if delegate.matches.is_empty() {
693                        delegate.selected_index = 0;
694                    } else {
695                        delegate.selected_index =
696                            core::cmp::min(delegate.selected_index, delegate.matches.len() - 1);
697                    }
698                    delegate.last_query = query;
699                })
700                .log_err();
701        })
702    }
703
704    fn confirm(&mut self, secondary: bool, window: &mut Window, cx: &mut Context<Picker<Self>>) {
705        let Some(entry) = self.matches.get(self.selected_index()) else {
706            return;
707        };
708        if entry.is_new {
709            self.create_worktree(&entry.worktree.branch(), secondary, None, window, cx);
710        } else {
711            self.open_worktree(&entry.worktree.path, secondary, window, cx);
712        }
713
714        cx.emit(DismissEvent);
715    }
716
717    fn dismissed(&mut self, _: &mut Window, cx: &mut Context<Picker<Self>>) {
718        cx.emit(DismissEvent);
719    }
720
721    fn render_match(
722        &self,
723        ix: usize,
724        selected: bool,
725        _window: &mut Window,
726        cx: &mut Context<Picker<Self>>,
727    ) -> Option<Self::ListItem> {
728        let entry = &self.matches.get(ix)?;
729        let path = entry.worktree.path.to_string_lossy().to_string();
730        let sha = entry
731            .worktree
732            .sha
733            .clone()
734            .chars()
735            .take(7)
736            .collect::<String>();
737
738        let (branch_name, sublabel) = if entry.is_new {
739            (
740                Label::new(format!("Create Worktree: \"{}\"", entry.worktree.branch()))
741                    .truncate()
742                    .into_any_element(),
743                format!(
744                    "based off {}",
745                    self.base_branch(cx).unwrap_or("the current branch")
746                ),
747            )
748        } else {
749            let branch = entry.worktree.branch();
750            let branch_first_line = branch.lines().next().unwrap_or(branch);
751            let positions: Vec<_> = entry
752                .positions
753                .iter()
754                .copied()
755                .filter(|&pos| pos < branch_first_line.len())
756                .collect();
757
758            (
759                HighlightedLabel::new(branch_first_line.to_owned(), positions)
760                    .truncate()
761                    .into_any_element(),
762                path,
763            )
764        };
765
766        Some(
767            ListItem::new(format!("worktree-menu-{ix}"))
768                .inset(true)
769                .spacing(ListItemSpacing::Sparse)
770                .toggle_state(selected)
771                .child(
772                    v_flex()
773                        .w_full()
774                        .child(
775                            h_flex()
776                                .gap_2()
777                                .justify_between()
778                                .overflow_x_hidden()
779                                .child(branch_name)
780                                .when(!entry.is_new, |this| {
781                                    this.child(
782                                        Label::new(sha)
783                                            .size(LabelSize::Small)
784                                            .color(Color::Muted)
785                                            .buffer_font(cx)
786                                            .into_element(),
787                                    )
788                                }),
789                        )
790                        .child(
791                            Label::new(sublabel)
792                                .size(LabelSize::Small)
793                                .color(Color::Muted)
794                                .truncate()
795                                .into_any_element(),
796                        ),
797                ),
798        )
799    }
800
801    fn no_matches_text(&self, _window: &mut Window, _cx: &mut App) -> Option<SharedString> {
802        Some("No worktrees found".into())
803    }
804
805    fn render_footer(&self, _: &mut Window, cx: &mut Context<Picker<Self>>) -> Option<AnyElement> {
806        let focus_handle = self.focus_handle.clone();
807        let selected_entry = self.matches.get(self.selected_index);
808        let is_creating = selected_entry.is_some_and(|entry| entry.is_new);
809
810        let footer_container = h_flex()
811            .w_full()
812            .p_1p5()
813            .gap_0p5()
814            .justify_end()
815            .border_t_1()
816            .border_color(cx.theme().colors().border_variant);
817
818        if is_creating {
819            let from_default_button = self.default_branch.as_ref().map(|default_branch| {
820                Button::new(
821                    "worktree-from-default",
822                    format!("Create from: {default_branch}"),
823                )
824                .key_binding(
825                    KeyBinding::for_action_in(&WorktreeFromDefault, &focus_handle, cx)
826                        .map(|kb| kb.size(rems_from_px(12.))),
827                )
828                .on_click(|_, window, cx| {
829                    window.dispatch_action(WorktreeFromDefault.boxed_clone(), cx)
830                })
831            });
832
833            let current_branch = self.base_branch(cx).unwrap_or("current branch");
834
835            Some(
836                footer_container
837                    .when_some(from_default_button, |this, button| this.child(button))
838                    .child(
839                        Button::new(
840                            "worktree-from-current",
841                            format!("Create from: {current_branch}"),
842                        )
843                        .key_binding(
844                            KeyBinding::for_action_in(&menu::Confirm, &focus_handle, cx)
845                                .map(|kb| kb.size(rems_from_px(12.))),
846                        )
847                        .on_click(|_, window, cx| {
848                            window.dispatch_action(menu::Confirm.boxed_clone(), cx)
849                        }),
850                    )
851                    .into_any(),
852            )
853        } else {
854            Some(
855                footer_container
856                    .child(
857                        Button::new("delete-worktree", "Delete")
858                            .key_binding(
859                                KeyBinding::for_action_in(&DeleteWorktree, &focus_handle, cx)
860                                    .map(|kb| kb.size(rems_from_px(12.))),
861                            )
862                            .on_click(|_, window, cx| {
863                                window.dispatch_action(DeleteWorktree.boxed_clone(), cx)
864                            }),
865                    )
866                    .child(
867                        Button::new("open-in-new-window", "Open in New Window")
868                            .key_binding(
869                                KeyBinding::for_action_in(&menu::Confirm, &focus_handle, cx)
870                                    .map(|kb| kb.size(rems_from_px(12.))),
871                            )
872                            .on_click(|_, window, cx| {
873                                window.dispatch_action(menu::Confirm.boxed_clone(), cx)
874                            }),
875                    )
876                    .child(
877                        Button::new("open-in-window", "Open")
878                            .key_binding(
879                                KeyBinding::for_action_in(
880                                    &menu::SecondaryConfirm,
881                                    &focus_handle,
882                                    cx,
883                                )
884                                .map(|kb| kb.size(rems_from_px(12.))),
885                            )
886                            .on_click(|_, window, cx| {
887                                window.dispatch_action(menu::SecondaryConfirm.boxed_clone(), cx)
888                            }),
889                    )
890                    .into_any(),
891            )
892        }
893    }
894}