* git://repo.or.cz/git-gui: git-gui: update french translation git-gui: update Japanese translation git-gui: fix shortcut for menu "Commit/Revert Changes" git-gui: Quote git path when starting another gui in a submodule git-gui: update Italian translation git-gui: Update Swedish translation (520t0f0u) git-gui: use themed tk widgets with Tk 8.5 git-gui: Update German translation (12 new or changed strings). git-gui: Update translation template git-gui: Remove unused icon file_parttick git-gui: use different icon for new and modified files in the index git-gui: set GIT_DIR and GIT_WORK_TREE after setup git-gui: update shortcut tools to use _gitworktree git-gui: handle bare repos correctly git-gui: handle non-standard worktree locations git-gui: Support applying a range of changes at once git-gui: Add a special diff popup menu for submodules git-gui: Use git diff --submodule when available
		
			
				
	
	
		
			634 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			Tcl
		
	
	
	
	
	
			
		
		
	
	
			634 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			Tcl
		
	
	
	
	
	
# 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]
 | 
						|
		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 {$pos_x >= 0} {append g +}
 | 
						|
	append g $pos_x
 | 
						|
	if {$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 {}
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
}
 |