> There is already a branch selection dialog (hit F2 to bring it up). > Could you please have a look whether it can be reused here? Because... Ah, I didn't consider that! I'll investigate using something similar. > At a minimum, there should be a confirmation when a > transition from a branch to a detached head happens That sounds like a good idea to me too, will fix. > I find it irritating that it is possible to check out a detached head > with Ctrl-O, but not via the context menu. > > ... > > I propose that you do not include the ability to check out a detached > head in this commit, but make it a new feature once this series has settled. I agree that the asymmetry between the keyboard bind and the context menu is somewhat unfortunate. But I also think that there is a pretty good justification it, and because of this I'd like to keep the feature in this PR. The row context menu has a bit too many entries already, to the point where it is a bit hard to use. We really only what to put the most important or commonly used commands here. This is an inherit limitation for GUI menus: the screen real estate is limited. So there is a valid reason not to include the Checkout command. For keyboard binds however these is no such screen real-estate issue. On the contrary: For keyboard binds it is an artificial asymmetry without any technical justification to NOT be able to checkout the commit of a selected row. Apart from this it will unfortunately be hard for me to take the time to create a separate PR. Do you accept this motivation for having the "checkout detached head" feature stay in this PR? > I find it irritating that it is possible to check out a detached head > with Ctrl-O, but not via the context menu. > > Playing around a bit with this feature, it may be a bit too easy to > check out a different commit with a simple key press and without > confirmation. At a minimum, there should be a confirmation when a > transition from a branch to a detached head happens, because otherwise > inexperienced users end up in inconvenient territories. > > I propose that you do not include the ability to check out a detached > head in this commit, but make it a new feature once this series has settled. On 2023-07-05 19:29, Johannes Sixt wrote: > Am 03.07.23 um 20:45 schrieb Jens Lidestrom via GitGitGadget: >> From: Jens Lidestrom <jens@xxxxxxxxxxxx> >> >> This also introduces the ability to check out detatched heads. This >> shouldn't result any problems, because gitk already works with >> detatched heads if they are checked out using the terminal. > > I find it irritating that it is possible to check out a detached head > with Ctrl-O, but not via the context menu. > > Playing around a bit with this feature, it may be a bit too easy to > check out a different commit with a simple key press and without > confirmation. At a minimum, there should be a confirmation when a > transition from a branch to a detached head happens, because otherwise > inexperienced users end up in inconvenient territories. > > I propose that you do not include the ability to check out a detached > head in this commit, but make it a new feature once this series has settled. > >> >> Signed-off-by: Jens Lidestrom <jens@xxxxxxxxxxxx> >> --- >> gitk-git/gitk | 125 +++++++++++++++++++++++++++++++++++++++----------- >> 1 file changed, 98 insertions(+), 27 deletions(-) >> >> diff --git a/gitk-git/gitk b/gitk-git/gitk >> index 642cd7f652a..8364033ad58 100755 >> --- a/gitk-git/gitk >> +++ b/gitk-git/gitk >> @@ -2691,6 +2691,7 @@ proc makewindow {} { >> bind $ctext <Button-1> {focus %W} >> bind $ctext <<Selection>> rehighlight_search_results >> bind . <$M1B-t> {resethead [selected_line_id]} >> + bind . <$M1B-o> {checkout [selected_line_heads] [selected_line_id]} >> for {set i 1} {$i < 10} {incr i} { >> bind . <$M1B-Key-$i> [list go_to_parent $i] >> } >> @@ -2732,7 +2733,7 @@ proc makewindow {} { >> >> set headctxmenu .headctxmenu >> makemenu $headctxmenu { >> - {mc "Check out this branch" command cobranch} >> + {mc "Check out this branch" command {checkout [list $headmenuhead] $headmenuid}} >> {mc "Rename this branch" command mvbranch} >> {mc "Remove this branch" command rmbranch} >> {mc "Copy branch name" command {clipboard clear; clipboard append $headmenuhead}} >> @@ -3183,6 +3184,7 @@ proc keys {} { >> [mc "<%s-minus> Decrease font size" $M1T] >> [mc "<F5> Update"] >> [mc "<%s-T> Reset current branch to selected commit" $M1T] >> +[mc "<%s-O> Check out selected commit" $M1T] >> " \ >> -justify left -bg $bgcolor -border 2 -relief groove >> pack $w.m -side top -fill both -padx 2 -pady 2 >> @@ -9978,25 +9980,93 @@ proc headmenu {x y id head} { >> tk_popup $headctxmenu $x $y >> } >> >> -proc cobranch {} { >> - global headmenuid headmenuhead headids >> +proc checkout {heads_on_commit id_to_checkout} { >> + global headids mainhead >> global showlocalchanges >> + global sel_ix confirm_ok NS >> >> # check the tree is clean first?? >> - set newhead $headmenuhead >> + >> + # Filter out remote branches if local branch is also present >> + foreach remote_ix [lsearch -all $heads_on_commit "remotes/*"] { >> + set remote_head [lindex $heads_on_commit $remote_ix] >> + set local_head [string range $remote_head [expr [string last / $remote_head] + 1] end] >> + if {$local_head in $heads_on_commit} { >> + set heads_on_commit [lreplace $heads_on_commit $remote_ix $remote_ix] >> + } >> + } >> + >> + if {[llength $heads_on_commit] == 1 && [lindex $heads_on_commit 0] eq $mainhead} { >> + # Only the currently active branch >> + return >> + } >> + >> + # Filter out mainhead >> + set mainhead_ix [lsearch $heads_on_commit $mainhead] >> + if {$mainhead_ix != -1} { >> + set heads_on_commit [lreplace $heads_on_commit $mainhead_ix $mainhead_ix] >> + } >> + set nr_heads_on_commit [llength $heads_on_commit] >> + >> + # The number of heads on the commit determines how to select what to checkout >> + if {$nr_heads_on_commit == 0} { >> + set head_to_checkout "" >> + set ref_to_checkout $id_to_checkout >> + } elseif {$nr_heads_on_commit == 1} { >> + set head_to_checkout [lindex $heads_on_commit 0] >> + set ref_to_checkout $head_to_checkout >> + } else { >> + # Branch selection dialog >> + >> + set confirm_ok 0 >> + >> + set w ".selectbranch" >> + ttk_toplevel $w >> + make_transient $w . >> + wm title $w [mc "Check out branch"] > > There is already a branch selection dialog (hit F2 to bring it up). > Could you please have a look whether it can be reused here? Because... > >> + ${NS}::label $w.m -text [mc "Check out which branch?"] >> + pack $w.m -side top -fill x -padx 20 -pady 20 >> + ${NS}::frame $w.f >> + >> + set sel_ix 0 >> + for {set i 0} {$i < $nr_heads_on_commit} {incr i} { >> + ${NS}::radiobutton $w.f.id_$i -value $i -variable sel_ix \ >> + -text [lindex $heads_on_commit $i] > > ... presenting a list of items in the form of radio buttons is a very > non-standard user-interface, IMO. I would have expected a list widget in > which an entry is highlighted. Also, if the list is long, a list would > have a scroll bar, but a radio button group is limited to the available > screen height. > >> + bind $w.f.id_$i <Key-Up> "set sel_ix [expr ($i - 1) % $nr_heads_on_commit]" >> + bind $w.f.id_$i <Key-Down> "set sel_ix [expr ($i + 1) % $nr_heads_on_commit]" >> + grid $w.f.id_$i -sticky w -padx 20 >> + } >> + >> + pack $w.f -side top -fill x -padx 4 >> + ${NS}::button $w.ok -text [mc OK] -command "set confirm_ok 1; destroy $w" >> + bind $w <Key-Return> "set confirm_ok 1; destroy $w" >> + pack $w.ok -side left -fill x -padx 20 -pady 20 >> + ${NS}::button $w.cancel -text [mc Cancel] -command "destroy $w" >> + bind $w <Key-Escape> [list destroy $w] >> + pack $w.cancel -side right -fill x -padx 20 -pady 20 >> + bind $w <Visibility> "grab $w; focus $w.f.id_$sel_ix" >> + >> + tkwait window $w >> + >> + if {!$confirm_ok} return >> + >> + set head_to_checkout [lindex $heads_on_commit $sel_ix] >> + set ref_to_checkout $head_to_checkout >> + } >> + >> + # Handle remote branches >> set command [list | git checkout] >> - if {[string match "remotes/*" $newhead]} { >> - set remote $newhead >> - set newhead [string range $newhead [expr [string last / $newhead] + 1] end] >> - # The following check is redundant - the menu option should >> - # be disabled to begin with... >> - if {[info exists headids($newhead)]} { >> - error_popup [mc "A local branch named %s exists already" $newhead] >> + if {[string match "remotes/*" $head_to_checkout]} { >> + set remote $head_to_checkout >> + set head_to_checkout [string range $head_to_checkout [expr [string last / $head_to_checkout] + 1] end] >> + set ref_to_checkout $head_to_checkout >> + if {[info exists headids($head_to_checkout)]} { >> + error_popup [mc "A local branch named %s exists already" $head_to_checkout] >> return >> } >> - lappend command -b $newhead --track $remote >> + lappend command -b $head_to_checkout --track $remote >> } else { >> - lappend command $newhead >> + lappend command $ref_to_checkout >> } >> lappend command 2>@1 >> nowbusy checkout [mc "Checking out"] >> @@ -10011,11 +10081,11 @@ proc cobranch {} { >> dodiffindex >> } >> } else { >> - filerun $fd [list readcheckoutstat $fd $newhead $headmenuid] >> + filerun $fd [list readcheckoutstat $fd $head_to_checkout $ref_to_checkout $id_to_checkout] >> } >> } >> >> -proc readcheckoutstat {fd newhead newheadid} { >> +proc readcheckoutstat {fd head_to_checkout ref_to_checkout id_to_checkout} { >> global mainhead mainheadid headids idheads showlocalchanges progresscoords >> global viewmainheadid curview >> >> @@ -10033,18 +10103,19 @@ proc readcheckoutstat {fd newhead newheadid} { >> error_popup $err >> return >> } >> - set oldmainid $mainheadid >> - if {! [info exists headids($newhead)]} { >> - set headids($newhead) $newheadid >> - lappend idheads($newheadid) $newhead >> - addedhead $newheadid $newhead >> - } >> - set mainhead $newhead >> - set mainheadid $newheadid >> - set viewmainheadid($curview) $newheadid >> - redrawtags $oldmainid >> - redrawtags $newheadid >> - selbyid $newheadid >> + set old_main_id $mainheadid >> + >> + if {$head_to_checkout ne "" && ! [info exists headids($head_to_checkout)]} { >> + set headids($head_to_checkout) $id_to_checkout >> + lappend idheads($id_to_checkout) $head_to_checkout >> + addedhead $id_to_checkout $head_to_checkout >> + } >> + set mainhead $ref_to_checkout >> + set mainheadid $id_to_checkout >> + set viewmainheadid($curview) $id_to_checkout >> + redrawtags $old_main_id >> + redrawtags $id_to_checkout >> + selbyid $id_to_checkout >> if {$showlocalchanges} { >> dodiffindex >> } >