lib / choose_rev.tclon commit git-gui: Cleanup bindings within merge dialog (9feefbd)
   1# git-gui revision chooser
   2# Copyright (C) 2006, 2007 Shawn Pearce
   3
   4class choose_rev {
   5
   6image create photo ::choose_rev::img_find -data {R0lGODlhEAAQAIYAAPwCBCQmJDw+PBQSFAQCBMza3NTm5MTW1HyChOT29Ozq7MTq7Kze5Kzm7Oz6/NTy9Iza5GzGzKzS1Nzy9Nz29Kzq9HTGzHTK1Lza3AwKDLzu9JTi7HTW5GTCzITO1Mzq7Hza5FTK1ESyvHzKzKzW3DQyNDyqtDw6PIzW5HzGzAT+/Dw+RKyurNTOzMTGxMS+tJSGdATCxHRydLSqpLymnLSijBweHERCRNze3Pz69PTy9Oze1OTSxOTGrMSqlLy+vPTu5OzSvMymjNTGvNS+tMy2pMyunMSefAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAACH5BAEAAAAALAAAAAAQABAAAAe4gACCAAECA4OIiAIEBQYHBAKJgwIICQoLDA0IkZIECQ4PCxARCwSSAxITFA8VEBYXGBmJAQYLGhUbHB0eH7KIGRIMEBAgISIjJKaIJQQLFxERIialkieUGigpKRoIBCqJKyyLBwvJAioEyoICLS4v6QQwMQQyLuqLli8zNDU2BCf1lN3AkUPHDh49fAQAAEnGD1MCCALZEaSHkIUMBQS8wWMIkSJGhBzBmFEGgRsBUqpMiSgdAD+BAAAh/mhDcmVhdGVkIGJ5IEJNUFRvR0lGIFBybyB2ZXJzaW9uIDIuNQ0KqSBEZXZlbENvciAxOTk3LDE5OTguIEFsbCByaWdodHMgcmVzZXJ2ZWQuDQpodHRwOi8vd3d3LmRldmVsY29yLmNvbQA7}
   7
   8field w               ; # our megawidget path
   9field w_list          ; # list of currently filtered specs
  10field w_filter        ; # filter entry for $w_list
  11
  12field c_expr        {}; # current revision expression
  13field filter          ; # current filter string
  14field revtype     head; # type of revision chosen
  15field cur_specs [list]; # list of specs for $revtype
  16field spec_head       ; # list of all head specs
  17field spec_trck       ; # list of all tracking branch specs
  18field spec_tag        ; # list of all tag specs
  19field tip_data        ; # array of tip commit info by refname
  20field log_last        ; # array of reflog date by refname
  21
  22field tooltip_wm        {} ; # Current tooltip toplevel, if open
  23field tooltip_t         {} ; # Text widget in $tooltip_wm
  24field tooltip_timer     {} ; # Current timer event for our tooltip
  25
  26proc new {path {title {}}} {
  27        return [_new $path 0 $title]
  28}
  29
  30proc new_unmerged {path {title {}}} {
  31        return [_new $path 1 $title]
  32}
  33
  34constructor _new {path unmerged_only title} {
  35        global current_branch is_detached
  36
  37        set w $path
  38
  39        if {$title ne {}} {
  40                labelframe $w -text $title
  41        } else {
  42                frame $w
  43        }
  44        bind $w <Destroy> [cb _delete %W]
  45
  46        if {$is_detached} {
  47                radiobutton $w.detachedhead_r \
  48                        -anchor w \
  49                        -text {This Detached Checkout} \
  50                        -value HEAD \
  51                        -variable @revtype
  52                grid $w.detachedhead_r -sticky we -padx {0 5} -columnspan 2
  53        }
  54
  55        radiobutton $w.expr_r \
  56                -text {Revision Expression:} \
  57                -value expr \
  58                -variable @revtype
  59        entry $w.expr_t \
  60                -borderwidth 1 \
  61                -relief sunken \
  62                -width 50 \
  63                -textvariable @c_expr \
  64                -validate key \
  65                -validatecommand [cb _validate %d %S]
  66        grid $w.expr_r $w.expr_t -sticky we -padx {0 5}
  67
  68        frame $w.types
  69        radiobutton $w.types.head_r \
  70                -text {Local Branch} \
  71                -value head \
  72                -variable @revtype
  73        pack $w.types.head_r -side left
  74        radiobutton $w.types.trck_r \
  75                -text {Tracking Branch} \
  76                -value trck \
  77                -variable @revtype
  78        pack $w.types.trck_r -side left
  79        radiobutton $w.types.tag_r \
  80                -text {Tag} \
  81                -value tag \
  82                -variable @revtype
  83        pack $w.types.tag_r -side left
  84        set w_filter $w.types.filter
  85        entry $w_filter \
  86                -borderwidth 1 \
  87                -relief sunken \
  88                -width 12 \
  89                -textvariable @filter \
  90                -validate key \
  91                -validatecommand [cb _filter %P]
  92        pack $w_filter -side right
  93        pack [label $w.types.filter_icon \
  94                -image ::choose_rev::img_find \
  95                ] -side right
  96        grid $w.types -sticky we -padx {0 5} -columnspan 2
  97
  98        frame $w.list
  99        set w_list $w.list.l
 100        listbox $w_list \
 101                -font font_diff \
 102                -width 50 \
 103                -height 10 \
 104                -selectmode browse \
 105                -exportselection false \
 106                -xscrollcommand [cb _sb_set $w.list.sbx h] \
 107                -yscrollcommand [cb _sb_set $w.list.sby v]
 108        pack $w_list -fill both -expand 1
 109        grid $w.list -sticky nswe -padx {20 5} -columnspan 2
 110        bind $w_list <Any-Motion>  [cb _show_tooltip @%x,%y]
 111        bind $w_list <Any-Enter>   [cb _hide_tooltip]
 112        bind $w_list <Any-Leave>   [cb _hide_tooltip]
 113        bind $w_list <Destroy>     [cb _hide_tooltip]
 114
 115        grid columnconfigure $w 1 -weight 1
 116        if {$is_detached} {
 117                grid rowconfigure $w 3 -weight 1
 118        } else {
 119                grid rowconfigure $w 2 -weight 1
 120        }
 121
 122        trace add variable @revtype write [cb _select]
 123        bind $w_filter <Key-Return> [list focus $w_list]\;break
 124        bind $w_filter <Key-Down>   [list focus $w_list]
 125
 126        set fmt list
 127        append fmt { %(refname)}
 128        append fmt { [list}
 129        append fmt { %(objecttype)}
 130        append fmt { %(objectname)}
 131        append fmt { [concat %(taggername) %(authorname)]}
 132        append fmt { [concat %(taggerdate) %(authordate)]}
 133        append fmt { %(subject)}
 134        append fmt {] [list}
 135        append fmt { %(*objecttype)}
 136        append fmt { %(*objectname)}
 137        append fmt { %(*authorname)}
 138        append fmt { %(*authordate)}
 139        append fmt { %(*subject)}
 140        append fmt {]}
 141        set all_refn [list]
 142        set fr_fd [git_read for-each-ref \
 143                --tcl \
 144                --sort=-taggerdate \
 145                --format=$fmt \
 146                refs/heads \
 147                refs/remotes \
 148                refs/tags \
 149                ]
 150        fconfigure $fr_fd -translation lf -encoding utf-8
 151        while {[gets $fr_fd line] > 0} {
 152                set line [eval $line]
 153                if {[lindex $line 1 0] eq {tag}} {
 154                        if {[lindex $line 2 0] eq {commit}} {
 155                                set sha1 [lindex $line 2 1]
 156                        } else {
 157                                continue
 158                        }
 159                } elseif {[lindex $line 1 0] eq {commit}} {
 160                        set sha1 [lindex $line 1 1]
 161                } else {
 162                        continue
 163                }
 164                set refn [lindex $line 0]
 165                set tip_data($refn) [lrange $line 1 end]
 166                lappend cmt_refn($sha1) $refn
 167                lappend all_refn $refn
 168        }
 169        close $fr_fd
 170
 171        if {$unmerged_only} {
 172                set fr_fd [git_read rev-list --all ^$::HEAD]
 173                while {[gets $fr_fd sha1] > 0} {
 174                        if {[catch {set rlst $cmt_refn($sha1)}]} continue
 175                        foreach refn $rlst {
 176                                set inc($refn) 1
 177                        }
 178                }
 179                close $fr_fd
 180        } else {
 181                foreach refn $all_refn {
 182                        set inc($refn) 1
 183                }
 184        }
 185
 186        set spec_head [list]
 187        foreach name [load_all_heads] {
 188                set refn refs/heads/$name
 189                if {[info exists inc($refn)]} {
 190                        lappend spec_head [list $name $refn]
 191                }
 192        }
 193
 194        set spec_trck [list]
 195        foreach spec [all_tracking_branches] {
 196                set refn [lindex $spec 0]
 197                if {[info exists inc($refn)]} {
 198                        regsub ^refs/(heads|remotes)/ $refn {} name
 199                        lappend spec_trck [concat $name $spec]
 200                }
 201        }
 202
 203        set spec_tag [list]
 204        foreach name [load_all_tags] {
 205                set refn refs/tags/$name
 206                if {[info exists inc($refn)]} {
 207                        lappend spec_tag [list $name $refn]
 208                }
 209        }
 210
 211                  if {$is_detached}             { set revtype HEAD
 212        } elseif {[llength $spec_head] > 0} { set revtype head
 213        } elseif {[llength $spec_trck] > 0} { set revtype trck
 214        } elseif {[llength $spec_tag ] > 0} { set revtype tag
 215        } else {                              set revtype expr
 216        }
 217
 218        if {$revtype eq {head} && $current_branch ne {}} {
 219                set i 0
 220                foreach spec $spec_head {
 221                        if {[lindex $spec 0] eq $current_branch} {
 222                                $w_list selection clear 0 end
 223                                $w_list selection set $i
 224                                break
 225                        }
 226                        incr i
 227                }
 228        }
 229
 230        return $this
 231}
 232
 233method none {text} {
 234        if {![winfo exists $w.none_r]} {
 235                radiobutton $w.none_r \
 236                        -anchor w \
 237                        -value none \
 238                        -variable @revtype
 239                grid $w.none_r -sticky we -padx {0 5} -columnspan 2
 240        }
 241        $w.none_r configure -text $text
 242}
 243
 244method get {} {
 245        switch -- $revtype {
 246        head -
 247        trck -
 248        tag  {
 249                set i [$w_list curselection]
 250                if {$i ne {}} {
 251                        return [lindex $cur_specs $i 0]
 252                } else {
 253                        return {}
 254                }
 255        }
 256
 257        HEAD { return HEAD                     }
 258        expr { return $c_expr                  }
 259        none { return {}                       }
 260        default { error "unknown type of revision" }
 261        }
 262}
 263
 264method pick_tracking_branch {} {
 265        set revtype trck
 266}
 267
 268method focus_filter {} {
 269        if {[$w_filter cget -state] eq {normal}} {
 270                focus $w_filter
 271        }
 272}
 273
 274method bind_listbox {event script}  {
 275        bind $w_list $event $script
 276}
 277
 278method get_local_branch {} {
 279        if {$revtype eq {head}} {
 280                return [_expr $this]
 281        } else {
 282                return {}
 283        }
 284}
 285
 286method get_tracking_branch {} {
 287        set i [$w_list curselection]
 288        if {$i eq {} || $revtype ne {trck}} {
 289                return {}
 290        }
 291        return [lrange [lindex $cur_specs $i] 1 end]
 292}
 293
 294method get_commit {} {
 295        set e [_expr $this]
 296        if {$e eq {}} {
 297                return {}
 298        }
 299        return [git rev-parse --verify "$e^0"]
 300}
 301
 302method commit_or_die {} {
 303        if {[catch {set new [get_commit $this]} err]} {
 304
 305                # Cleanup the not-so-friendly error from rev-parse.
 306                #
 307                regsub {^fatal:\s*} $err {} err
 308                if {$err eq {Needed a single revision}} {
 309                        set err {}
 310                }
 311
 312                set top [winfo toplevel $w]
 313                set msg "Invalid revision: [get $this]\n\n$err"
 314                tk_messageBox \
 315                        -icon error \
 316                        -type ok \
 317                        -title [wm title $top] \
 318                        -parent $top \
 319                        -message $msg
 320                error $msg
 321        }
 322        return $new
 323}
 324
 325method _expr {} {
 326        switch -- $revtype {
 327        head -
 328        trck -
 329        tag  {
 330                set i [$w_list curselection]
 331                if {$i ne {}} {
 332                        return [lindex $cur_specs $i 1]
 333                } else {
 334                        error "No revision selected."
 335                }
 336        }
 337
 338        expr {
 339                if {$c_expr ne {}} {
 340                        return $c_expr
 341                } else {
 342                        error "Revision expression is empty."
 343                }
 344        }
 345        HEAD { return HEAD                     }
 346        none { return {}                       }
 347        default { error "unknown type of revision"      }
 348        }
 349}
 350
 351method _validate {d S} {
 352        if {$d == 1} {
 353                if {[regexp {\s} $S]} {
 354                        return 0
 355                }
 356                if {[string length $S] > 0} {
 357                        set revtype expr
 358                }
 359        }
 360        return 1
 361}
 362
 363method _filter {P} {
 364        if {[regexp {\s} $P]} {
 365                return 0
 366        }
 367        _rebuild $this $P
 368        return 1
 369}
 370
 371method _select {args} {
 372        _rebuild $this $filter
 373        focus_filter $this
 374}
 375
 376method _rebuild {pat} {
 377        set ste normal
 378        switch -- $revtype {
 379        head { set new $spec_head }
 380        trck { set new $spec_trck }
 381        tag  { set new $spec_tag  }
 382        expr -
 383        HEAD -
 384        none {
 385                set new [list]
 386                set ste disabled
 387        }
 388        }
 389
 390        if {[$w_list cget -state] eq {disabled}} {
 391                $w_list configure -state normal
 392        }
 393        $w_list delete 0 end
 394
 395        if {$pat ne {}} {
 396                set pat *${pat}*
 397        }
 398        set cur_specs [list]
 399        foreach spec $new {
 400                set txt [lindex $spec 0]
 401                if {$pat eq {} || [string match $pat $txt]} {
 402                        lappend cur_specs $spec
 403                        $w_list insert end $txt
 404                }
 405        }
 406        if {$cur_specs ne {}} {
 407                $w_list selection clear 0 end
 408                $w_list selection set 0
 409        }
 410
 411        if {[$w_filter cget -state] ne $ste} {
 412                $w_list   configure -state $ste
 413                $w_filter configure -state $ste
 414        }
 415}
 416
 417method _delete {current} {
 418        if {$current eq $w} {
 419                delete_this
 420        }
 421}
 422
 423method _sb_set {sb orient first last} {
 424        set old_focus [focus -lastfor $w]
 425
 426        if {$first == 0 && $last == 1} {
 427                if {[winfo exists $sb]} {
 428                        destroy $sb
 429                        if {$old_focus ne {}} {
 430                                update
 431                                focus $old_focus
 432                        }
 433                }
 434                return
 435        }
 436
 437        if {![winfo exists $sb]} {
 438                if {$orient eq {h}} {
 439                        scrollbar $sb -orient h -command [list $w_list xview]
 440                        pack $sb -fill x -side bottom -before $w_list
 441                } else {
 442                        scrollbar $sb -orient v -command [list $w_list yview]
 443                        pack $sb -fill y -side right -before $w_list
 444                }
 445                if {$old_focus ne {}} {
 446                        update
 447                        focus $old_focus
 448                }
 449        }
 450        $sb set $first $last
 451}
 452
 453method _show_tooltip {pos} {
 454        if {$tooltip_wm ne {}} {
 455                _open_tooltip $this
 456        } elseif {$tooltip_timer eq {}} {
 457                set tooltip_timer [after 1000 [cb _open_tooltip]]
 458        }
 459}
 460
 461method _open_tooltip {} {
 462        global remote_url
 463
 464        set tooltip_timer {}
 465        set pos_x [winfo pointerx $w_list]
 466        set pos_y [winfo pointery $w_list]
 467        if {[winfo containing $pos_x $pos_y] ne $w_list} {
 468                _hide_tooltip $this
 469                return
 470        }
 471
 472        set pos @[join [list \
 473                [expr {$pos_x - [winfo rootx $w_list]}] \
 474                [expr {$pos_y - [winfo rooty $w_list]}]] ,]
 475        set lno [$w_list index $pos]
 476        if {$lno eq {}} {
 477                _hide_tooltip $this
 478                return
 479        }
 480
 481        set spec [lindex $cur_specs $lno]
 482        set refn [lindex $spec 1]
 483        if {$refn eq {}} {
 484                _hide_tooltip $this
 485                return
 486        }
 487
 488        if {$tooltip_wm eq {}} {
 489                set tooltip_wm [toplevel $w_list.tooltip -borderwidth 1]
 490                wm overrideredirect $tooltip_wm 1
 491                wm transient $tooltip_wm [winfo toplevel $w_list]
 492                set tooltip_t $tooltip_wm.label
 493                text $tooltip_t \
 494                        -takefocus 0 \
 495                        -highlightthickness 0 \
 496                        -relief flat \
 497                        -borderwidth 0 \
 498                        -wrap none \
 499                        -background lightyellow \
 500                        -foreground black
 501                $tooltip_t tag conf section_header -font font_uibold
 502                bind $tooltip_wm <Escape> [cb _hide_tooltip]
 503                pack $tooltip_t
 504        } else {
 505                $tooltip_t conf -state normal
 506                $tooltip_t delete 0.0 end
 507        }
 508
 509        set data $tip_data($refn)
 510        if {[lindex $data 0 0] eq {tag}} {
 511                set tag  [lindex $data 0]
 512                if {[lindex $data 1 0] eq {commit}} {
 513                        set cmit [lindex $data 1]
 514                } else {
 515                        set cmit {}
 516                }
 517        } elseif {[lindex $data 0 0] eq {commit}} {
 518                set tag  {}
 519                set cmit [lindex $data 0]
 520        }
 521
 522        $tooltip_t insert end [lindex $spec 0]
 523        set last [_reflog_last $this [lindex $spec 1]]
 524        if {$last ne {}} {
 525                $tooltip_t insert end "\n"
 526                $tooltip_t insert end "updated"
 527                $tooltip_t insert end " $last"
 528        }
 529        $tooltip_t insert end "\n"
 530
 531        if {$tag ne {}} {
 532                $tooltip_t insert end "\n"
 533                $tooltip_t insert end "tag" section_header
 534                $tooltip_t insert end "  [lindex $tag 1]\n"
 535                $tooltip_t insert end [lindex $tag 2]
 536                $tooltip_t insert end " ([lindex $tag 3])\n"
 537                $tooltip_t insert end [lindex $tag 4]
 538                $tooltip_t insert end "\n"
 539        }
 540
 541        if {$cmit ne {}} {
 542                $tooltip_t insert end "\n"
 543                $tooltip_t insert end "commit" section_header
 544                $tooltip_t insert end "  [lindex $cmit 1]\n"
 545                $tooltip_t insert end [lindex $cmit 2]
 546                $tooltip_t insert end " ([lindex $cmit 3])\n"
 547                $tooltip_t insert end [lindex $cmit 4]
 548        }
 549
 550        if {[llength $spec] > 2} {
 551                $tooltip_t insert end "\n"
 552                $tooltip_t insert end "remote" section_header
 553                $tooltip_t insert end "  [lindex $spec 2]\n"
 554                $tooltip_t insert end "url"
 555                $tooltip_t insert end " $remote_url([lindex $spec 2])\n"
 556                $tooltip_t insert end "branch"
 557                $tooltip_t insert end " [lindex $spec 3]"
 558        }
 559
 560        $tooltip_t conf -state disabled
 561        _position_tooltip $this
 562}
 563
 564method _reflog_last {name} {
 565        if {[info exists reflog_last($name)]} {
 566                return reflog_last($name)
 567        }
 568
 569        set last {}
 570        if {[catch {set last [file mtime [gitdir $name]]}]
 571        && ![catch {set g [open [gitdir logs $name] r]}]} {
 572                fconfigure $g -translation binary
 573                while {[gets $g line] >= 0} {
 574                        if {[regexp {> ([1-9][0-9]*) } $line line when]} {
 575                                set last $when
 576                        }
 577                }
 578                close $g
 579        }
 580
 581        if {$last ne {}} {
 582                set last [clock format $last -format {%a %b %e %H:%M:%S %Y}]
 583        }
 584        set reflog_last($name) $last
 585        return $last
 586}
 587
 588method _position_tooltip {} {
 589        set max_h [lindex [split [$tooltip_t index end] .] 0]
 590        set max_w 0
 591        for {set i 1} {$i <= $max_h} {incr i} {
 592                set c [lindex [split [$tooltip_t index "$i.0 lineend"] .] 1]
 593                if {$c > $max_w} {set max_w $c}
 594        }
 595        $tooltip_t conf -width $max_w -height $max_h
 596
 597        set req_w [winfo reqwidth  $tooltip_t]
 598        set req_h [winfo reqheight $tooltip_t]
 599        set pos_x [expr {[winfo pointerx .] +  5}]
 600        set pos_y [expr {[winfo pointery .] + 10}]
 601
 602        set g "${req_w}x${req_h}"
 603        if {$pos_x >= 0} {append g +}
 604        append g $pos_x
 605        if {$pos_y >= 0} {append g +}
 606        append g $pos_y
 607
 608        wm geometry $tooltip_wm $g
 609        raise $tooltip_wm
 610}
 611
 612method _hide_tooltip {} {
 613        if {$tooltip_wm ne {}} {
 614                destroy $tooltip_wm
 615                set tooltip_wm {}
 616        }
 617        if {$tooltip_timer ne {}} {
 618                after cancel $tooltip_timer
 619                set tooltip_timer {}
 620        }
 621}
 622
 623}