| # git-gui revision chooser |
| # Copyright (C) 2006, 2007 Shawn Pearce |
| |
| class choose_rev { |
| |
| image create photo ::choose_rev::img_find -data {R0lGODlhEAAQAIYAAPwCBCQmJDw+PBQSFAQCBMza3NTm5MTW1HyChOT29Ozq7MTq7Kze5Kzm7Oz6/NTy9Iza5GzGzKzS1Nzy9Nz29Kzq9HTGzHTK1Lza3AwKDLzu9JTi7HTW5GTCzITO1Mzq7Hza5FTK1ESyvHzKzKzW3DQyNDyqtDw6PIzW5HzGzAT+/Dw+RKyurNTOzMTGxMS+tJSGdATCxHRydLSqpLymnLSijBweHERCRNze3Pz69PTy9Oze1OTSxOTGrMSqlLy+vPTu5OzSvMymjNTGvNS+tMy2pMyunMSefAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAACH5BAEAAAAALAAAAAAQABAAAAe4gACCAAECA4OIiAIEBQYHBAKJgwIICQoLDA0IkZIECQ4PCxARCwSSAxITFA8VEBYXGBmJAQYLGhUbHB0eH7KIGRIMEBAgISIjJKaIJQQLFxERIialkieUGigpKRoIBCqJKyyLBwvJAioEyoICLS4v6QQwMQQyLuqLli8zNDU2BCf1lN3AkUPHDh49fAQAAEnGD1MCCALZEaSHkIUMBQS8wWMIkSJGhBzBmFEGgRsBUqpMiSgdAD+BAAAh/mhDcmVhdGVkIGJ5IEJNUFRvR0lGIFBybyB2ZXJzaW9uIDIuNQ0KqSBEZXZlbENvciAxOTk3LDE5OTguIEFsbCByaWdodHMgcmVzZXJ2ZWQuDQpodHRwOi8vd3d3LmRldmVsY29yLmNvbQA7} |
| |
| field w ; # our megawidget path |
| field w_list ; # list of currently filtered specs |
| field w_filter ; # filter entry for $w_list |
| |
| field c_expr {}; # current revision expression |
| field filter ""; # current filter string |
| field revtype head; # type of revision chosen |
| field cur_specs [list]; # list of specs for $revtype |
| field spec_head ; # list of all head specs |
| field spec_trck ; # list of all tracking branch specs |
| field spec_tag ; # list of all tag specs |
| field tip_data ; # array of tip commit info by refname |
| field log_last ; # array of reflog date by refname |
| |
| field tooltip_wm {} ; # Current tooltip toplevel, if open |
| field tooltip_t {} ; # Text widget in $tooltip_wm |
| field tooltip_timer {} ; # Current timer event for our tooltip |
| |
| proc new {path {title {}}} { |
| return [_new $path 0 $title] |
| } |
| |
| proc new_unmerged {path {title {}}} { |
| return [_new $path 1 $title] |
| } |
| |
| constructor _new {path unmerged_only title} { |
| global current_branch is_detached use_ttk NS |
| |
| if {![info exists ::all_remotes]} { |
| load_all_remotes |
| } |
| |
| set w $path |
| |
| if {$title ne {}} { |
| ${NS}::labelframe $w -text $title |
| } else { |
| ${NS}::frame $w |
| } |
| bind $w <Destroy> [cb _delete %W] |
| |
| if {$is_detached} { |
| ${NS}::radiobutton $w.detachedhead_r \ |
| -text [mc "This Detached Checkout"] \ |
| -value HEAD \ |
| -variable @revtype |
| if {!$use_ttk} {$w.detachedhead_r configure -anchor w} |
| grid $w.detachedhead_r -sticky we -padx {0 5} -columnspan 2 |
| } |
| |
| ${NS}::radiobutton $w.expr_r \ |
| -text [mc "Revision Expression:"] \ |
| -value expr \ |
| -variable @revtype |
| ${NS}::entry $w.expr_t \ |
| -width 50 \ |
| -textvariable @c_expr \ |
| -validate key \ |
| -validatecommand [cb _validate %d %S] |
| grid $w.expr_r $w.expr_t -sticky we -padx {0 5} |
| |
| ${NS}::frame $w.types |
| ${NS}::radiobutton $w.types.head_r \ |
| -text [mc "Local Branch"] \ |
| -value head \ |
| -variable @revtype |
| pack $w.types.head_r -side left |
| ${NS}::radiobutton $w.types.trck_r \ |
| -text [mc "Tracking Branch"] \ |
| -value trck \ |
| -variable @revtype |
| pack $w.types.trck_r -side left |
| ${NS}::radiobutton $w.types.tag_r \ |
| -text [mc "Tag"] \ |
| -value tag \ |
| -variable @revtype |
| pack $w.types.tag_r -side left |
| set w_filter $w.types.filter |
| ${NS}::entry $w_filter \ |
| -width 12 \ |
| -textvariable @filter \ |
| -validate key \ |
| -validatecommand [cb _filter %P] |
| pack $w_filter -side right |
| pack [${NS}::label $w.types.filter_icon \ |
| -image ::choose_rev::img_find \ |
| ] -side right |
| grid $w.types -sticky we -padx {0 5} -columnspan 2 |
| |
| if {$use_ttk} { |
| ttk::frame $w.list -style SListbox.TFrame -padding 2 |
| } else { |
| frame $w.list |
| } |
| set w_list $w.list.l |
| listbox $w_list \ |
| -font font_diff \ |
| -width 50 \ |
| -height 10 \ |
| -selectmode browse \ |
| -exportselection false \ |
| -xscrollcommand [cb _sb_set $w.list.sbx h] \ |
| -yscrollcommand [cb _sb_set $w.list.sby v] |
| if {$use_ttk} { |
| $w_list configure -relief flat -highlightthickness 0 -borderwidth 0 |
| } |
| pack $w_list -fill both -expand 1 |
| grid $w.list -sticky nswe -padx {20 5} -columnspan 2 |
| bind $w_list <Any-Motion> [cb _show_tooltip @%x,%y] |
| bind $w_list <Any-Enter> [cb _hide_tooltip] |
| bind $w_list <Any-Leave> [cb _hide_tooltip] |
| bind $w_list <Destroy> [cb _hide_tooltip] |
| |
| grid columnconfigure $w 1 -weight 1 |
| if {$is_detached} { |
| grid rowconfigure $w 3 -weight 1 |
| } else { |
| grid rowconfigure $w 2 -weight 1 |
| } |
| |
| trace add variable @revtype write [cb _select] |
| bind $w_filter <Key-Return> [list focus $w_list]\;break |
| bind $w_filter <Key-Down> [list focus $w_list] |
| |
| set fmt list |
| append fmt { %(refname)} |
| append fmt { [list} |
| append fmt { %(objecttype)} |
| append fmt { %(objectname)} |
| append fmt { [concat %(taggername) %(authorname)]} |
| append fmt { [reformat_date [concat %(taggerdate) %(authordate)]]} |
| append fmt { %(subject)} |
| append fmt {] [list} |
| append fmt { %(*objecttype)} |
| append fmt { %(*objectname)} |
| append fmt { %(*authorname)} |
| append fmt { [reformat_date %(*authordate)]} |
| append fmt { %(*subject)} |
| append fmt {]} |
| set all_refn [list] |
| set fr_fd [git_read for-each-ref \ |
| --tcl \ |
| --sort=-taggerdate \ |
| --format=$fmt \ |
| refs/heads \ |
| refs/remotes \ |
| refs/tags \ |
| ] |
| fconfigure $fr_fd -translation lf -encoding utf-8 |
| while {[gets $fr_fd line] > 0} { |
| set line [eval $line] |
| if {[lindex $line 1 0] eq {tag}} { |
| if {[lindex $line 2 0] eq {commit}} { |
| set sha1 [lindex $line 2 1] |
| } else { |
| continue |
| } |
| } elseif {[lindex $line 1 0] eq {commit}} { |
| set sha1 [lindex $line 1 1] |
| } else { |
| continue |
| } |
| set refn [lindex $line 0] |
| set tip_data($refn) [lrange $line 1 end] |
| lappend cmt_refn($sha1) $refn |
| lappend all_refn $refn |
| } |
| close $fr_fd |
| |
| if {$unmerged_only} { |
| set fr_fd [git_read rev-list --all ^$::HEAD] |
| while {[gets $fr_fd sha1] > 0} { |
| if {[catch {set rlst $cmt_refn($sha1)}]} continue |
| foreach refn $rlst { |
| set inc($refn) 1 |
| } |
| } |
| close $fr_fd |
| } else { |
| foreach refn $all_refn { |
| set inc($refn) 1 |
| } |
| } |
| |
| set spec_head [list] |
| foreach name [load_all_heads] { |
| set refn refs/heads/$name |
| if {[info exists inc($refn)]} { |
| lappend spec_head [list $name $refn] |
| } |
| } |
| |
| set spec_trck [list] |
| foreach spec [all_tracking_branches] { |
| set refn [lindex $spec 0] |
| if {[info exists inc($refn)]} { |
| regsub ^refs/(heads|remotes)/ $refn {} name |
| lappend spec_trck [concat $name $spec] |
| } |
| } |
| |
| set spec_tag [list] |
| foreach name [load_all_tags] { |
| set refn refs/tags/$name |
| if {[info exists inc($refn)]} { |
| lappend spec_tag [list $name $refn] |
| } |
| } |
| |
| if {$is_detached} { set revtype HEAD |
| } elseif {[llength $spec_head] > 0} { set revtype head |
| } elseif {[llength $spec_trck] > 0} { set revtype trck |
| } elseif {[llength $spec_tag ] > 0} { set revtype tag |
| } else { set revtype expr |
| } |
| |
| if {$revtype eq {head} && $current_branch ne {}} { |
| set i 0 |
| foreach spec $spec_head { |
| if {[lindex $spec 0] eq $current_branch} { |
| $w_list selection clear 0 end |
| $w_list selection set $i |
| break |
| } |
| incr i |
| } |
| } |
| |
| return $this |
| } |
| |
| method none {text} { |
| global NS use_ttk |
| if {![winfo exists $w.none_r]} { |
| ${NS}::radiobutton $w.none_r \ |
| -value none \ |
| -variable @revtype |
| if {!$use_ttk} {$w.none_r configure -anchor w} |
| grid $w.none_r -sticky we -padx {0 5} -columnspan 2 |
| } |
| $w.none_r configure -text $text |
| } |
| |
| method get {} { |
| switch -- $revtype { |
| head - |
| trck - |
| tag { |
| set i [$w_list curselection] |
| if {$i ne {}} { |
| return [lindex $cur_specs $i 0] |
| } else { |
| return {} |
| } |
| } |
| |
| HEAD { return HEAD } |
| expr { return $c_expr } |
| none { return {} } |
| default { error "unknown type of revision" } |
| } |
| } |
| |
| method pick_tracking_branch {} { |
| set revtype trck |
| } |
| |
| method focus_filter {} { |
| if {[$w_filter cget -state] eq {normal}} { |
| focus $w_filter |
| } |
| } |
| |
| method bind_listbox {event script} { |
| bind $w_list $event $script |
| } |
| |
| method get_local_branch {} { |
| if {$revtype eq {head}} { |
| return [_expr $this] |
| } else { |
| return {} |
| } |
| } |
| |
| method get_tracking_branch {} { |
| set i [$w_list curselection] |
| if {$i eq {} || $revtype ne {trck}} { |
| return {} |
| } |
| return [lrange [lindex $cur_specs $i] 1 end] |
| } |
| |
| method get_commit {} { |
| set e [_expr $this] |
| if {$e eq {}} { |
| return {} |
| } |
| return [git rev-parse --verify "$e^0"] |
| } |
| |
| method commit_or_die {} { |
| if {[catch {set new [get_commit $this]} err]} { |
| |
| # Cleanup the not-so-friendly error from rev-parse. |
| # |
| regsub {^fatal:\s*} $err {} err |
| if {$err eq {Needed a single revision}} { |
| set err {} |
| } |
| |
| set top [winfo toplevel $w] |
| set msg [strcat [mc "Invalid revision: %s" [get $this]] "\n\n$err"] |
| tk_messageBox \ |
| -icon error \ |
| -type ok \ |
| -title [wm title $top] \ |
| -parent $top \ |
| -message $msg |
| error $msg |
| } |
| return $new |
| } |
| |
| method _expr {} { |
| switch -- $revtype { |
| head - |
| trck - |
| tag { |
| set i [$w_list curselection] |
| if {$i ne {}} { |
| return [lindex $cur_specs $i 1] |
| } else { |
| error [mc "No revision selected."] |
| } |
| } |
| |
| expr { |
| if {$c_expr ne {}} { |
| return $c_expr |
| } else { |
| error [mc "Revision expression is empty."] |
| } |
| } |
| HEAD { return HEAD } |
| none { return {} } |
| default { error "unknown type of revision" } |
| } |
| } |
| |
| method _validate {d S} { |
| if {$d == 1} { |
| if {[regexp {\s} $S]} { |
| return 0 |
| } |
| if {[string length $S] > 0} { |
| set revtype expr |
| } |
| } |
| return 1 |
| } |
| |
| method _filter {P} { |
| if {[regexp {\s} $P]} { |
| return 0 |
| } |
| _rebuild $this $P |
| return 1 |
| } |
| |
| method _select {args} { |
| _rebuild $this $filter |
| focus_filter $this |
| } |
| |
| method _rebuild {pat} { |
| set ste normal |
| switch -- $revtype { |
| head { set new $spec_head } |
| trck { set new $spec_trck } |
| tag { set new $spec_tag } |
| expr - |
| HEAD - |
| none { |
| set new [list] |
| set ste disabled |
| } |
| } |
| |
| if {[$w_list cget -state] eq {disabled}} { |
| $w_list configure -state normal |
| } |
| $w_list delete 0 end |
| |
| if {$pat ne {}} { |
| set pat *${pat}* |
| } |
| set cur_specs [list] |
| foreach spec $new { |
| set txt [lindex $spec 0] |
| if {$pat eq {} || [string match $pat $txt]} { |
| lappend cur_specs $spec |
| $w_list insert end $txt |
| } |
| } |
| if {$cur_specs ne {}} { |
| $w_list selection clear 0 end |
| $w_list selection set 0 |
| } |
| |
| if {[$w_filter cget -state] ne $ste} { |
| $w_list configure -state $ste |
| $w_filter configure -state $ste |
| } |
| } |
| |
| method _delete {current} { |
| if {$current eq $w} { |
| delete_this |
| } |
| } |
| |
| method _sb_set {sb orient first last} { |
| global NS |
| set old_focus [focus -lastfor $w] |
| |
| if {$first == 0 && $last == 1} { |
| if {[winfo exists $sb]} { |
| destroy $sb |
| if {$old_focus ne {}} { |
| update |
| focus $old_focus |
| } |
| } |
| return |
| } |
| |
| if {![winfo exists $sb]} { |
| if {$orient eq {h}} { |
| ${NS}::scrollbar $sb -orient h -command [list $w_list xview] |
| pack $sb -fill x -side bottom -before $w_list |
| } else { |
| ${NS}::scrollbar $sb -orient v -command [list $w_list yview] |
| pack $sb -fill y -side right -before $w_list |
| } |
| if {$old_focus ne {}} { |
| update |
| focus $old_focus |
| } |
| } |
| |
| catch {$sb set $first $last} |
| } |
| |
| method _show_tooltip {pos} { |
| if {$tooltip_wm ne {}} { |
| _open_tooltip $this |
| } elseif {$tooltip_timer eq {}} { |
| set tooltip_timer [after 1000 [cb _open_tooltip]] |
| } |
| } |
| |
| method _open_tooltip {} { |
| global remote_url |
| |
| set tooltip_timer {} |
| set pos_x [winfo pointerx $w_list] |
| set pos_y [winfo pointery $w_list] |
| if {[winfo containing $pos_x $pos_y] ne $w_list} { |
| _hide_tooltip $this |
| return |
| } |
| |
| set pos @[join [list \ |
| [expr {$pos_x - [winfo rootx $w_list]}] \ |
| [expr {$pos_y - [winfo rooty $w_list]}]] ,] |
| set lno [$w_list index $pos] |
| if {$lno eq {}} { |
| _hide_tooltip $this |
| return |
| } |
| |
| set spec [lindex $cur_specs $lno] |
| set refn [lindex $spec 1] |
| if {$refn eq {}} { |
| _hide_tooltip $this |
| return |
| } |
| |
| if {$tooltip_wm eq {}} { |
| set tooltip_wm [toplevel $w_list.tooltip -borderwidth 1] |
| catch {wm attributes $tooltip_wm -type tooltip} |
| wm overrideredirect $tooltip_wm 1 |
| wm transient $tooltip_wm [winfo toplevel $w_list] |
| set tooltip_t $tooltip_wm.label |
| text $tooltip_t \ |
| -takefocus 0 \ |
| -highlightthickness 0 \ |
| -relief flat \ |
| -borderwidth 0 \ |
| -wrap none \ |
| -background lightyellow \ |
| -foreground black |
| $tooltip_t tag conf section_header -font font_uibold |
| bind $tooltip_wm <Escape> [cb _hide_tooltip] |
| pack $tooltip_t |
| } else { |
| $tooltip_t conf -state normal |
| $tooltip_t delete 0.0 end |
| } |
| |
| set data $tip_data($refn) |
| if {[lindex $data 0 0] eq {tag}} { |
| set tag [lindex $data 0] |
| if {[lindex $data 1 0] eq {commit}} { |
| set cmit [lindex $data 1] |
| } else { |
| set cmit {} |
| } |
| } elseif {[lindex $data 0 0] eq {commit}} { |
| set tag {} |
| set cmit [lindex $data 0] |
| } |
| |
| $tooltip_t insert end [lindex $spec 0] |
| set last [_reflog_last $this [lindex $spec 1]] |
| if {$last ne {}} { |
| $tooltip_t insert end "\n" |
| $tooltip_t insert end [mc "Updated"] |
| $tooltip_t insert end " $last" |
| } |
| $tooltip_t insert end "\n" |
| |
| if {$tag ne {}} { |
| $tooltip_t insert end "\n" |
| $tooltip_t insert end [mc "Tag"] section_header |
| $tooltip_t insert end " [lindex $tag 1]\n" |
| $tooltip_t insert end [lindex $tag 2] |
| $tooltip_t insert end " ([lindex $tag 3])\n" |
| $tooltip_t insert end [lindex $tag 4] |
| $tooltip_t insert end "\n" |
| } |
| |
| if {$cmit ne {}} { |
| $tooltip_t insert end "\n" |
| $tooltip_t insert end [mc "Commit@@noun"] section_header |
| $tooltip_t insert end " [lindex $cmit 1]\n" |
| $tooltip_t insert end [lindex $cmit 2] |
| $tooltip_t insert end " ([lindex $cmit 3])\n" |
| $tooltip_t insert end [lindex $cmit 4] |
| } |
| |
| if {[llength $spec] > 2} { |
| $tooltip_t insert end "\n" |
| $tooltip_t insert end [mc "Remote"] section_header |
| $tooltip_t insert end " [lindex $spec 2]\n" |
| $tooltip_t insert end [mc "URL"] |
| $tooltip_t insert end " $remote_url([lindex $spec 2])\n" |
| $tooltip_t insert end [mc "Branch"] |
| $tooltip_t insert end " [lindex $spec 3]" |
| } |
| |
| $tooltip_t conf -state disabled |
| _position_tooltip $this |
| } |
| |
| method _reflog_last {name} { |
| if {[info exists reflog_last($name)]} { |
| return reflog_last($name) |
| } |
| |
| set last {} |
| if {[catch {set last [file mtime [gitdir $name]]}] |
| && ![catch {set g [open [gitdir logs $name] r]}]} { |
| fconfigure $g -translation binary |
| while {[gets $g line] >= 0} { |
| if {[regexp {> ([1-9][0-9]*) } $line line when]} { |
| set last $when |
| } |
| } |
| close $g |
| } |
| |
| if {$last ne {}} { |
| set last [format_date $last] |
| } |
| set reflog_last($name) $last |
| return $last |
| } |
| |
| method _position_tooltip {} { |
| set max_h [lindex [split [$tooltip_t index end] .] 0] |
| set max_w 0 |
| for {set i 1} {$i <= $max_h} {incr i} { |
| set c [lindex [split [$tooltip_t index "$i.0 lineend"] .] 1] |
| if {$c > $max_w} {set max_w $c} |
| } |
| $tooltip_t conf -width $max_w -height $max_h |
| |
| set req_w [winfo reqwidth $tooltip_t] |
| set req_h [winfo reqheight $tooltip_t] |
| set pos_x [expr {[winfo pointerx .] + 5}] |
| set pos_y [expr {[winfo pointery .] + 10}] |
| |
| set g "${req_w}x${req_h}" |
| if {[tk windowingsystem] eq "win32" || $pos_x >= 0} {append g +} |
| append g $pos_x |
| if {[tk windowingsystem] eq "win32" || $pos_y >= 0} {append g +} |
| append g $pos_y |
| |
| wm geometry $tooltip_wm $g |
| raise $tooltip_wm |
| } |
| |
| method _hide_tooltip {} { |
| if {$tooltip_wm ne {}} { |
| destroy $tooltip_wm |
| set tooltip_wm {} |
| } |
| if {$tooltip_timer ne {}} { |
| after cancel $tooltip_timer |
| set tooltip_timer {} |
| } |
| } |
| |
| } |