2 # Tcl ignores the next line -*- tcl -*- \
3 if test "z$*" = zversion \
4 ||
test "z$*" = z--version
; \
6 echo 'git-gui version @@GITGUI_VERSION@@'; \
11 set appvers
{@@GITGUI_VERSION@@
}
13 Copyright ©
2006, 2007 Shawn Pearce
, et. al.
15 This program is free software
; you can redistribute it and
/or modify
16 it under the terms of the GNU General Public License as published by
17 the Free Software Foundation
; either version
2 of the License
, or
18 (at your option
) any later version.
20 This program is distributed
in the hope that it will be useful
,
21 but WITHOUT ANY WARRANTY
; without even the implied warranty of
22 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
23 GNU General Public License
for more details.
25 You should have received a copy of the GNU General Public License
26 along with this program
; if not
, write to the Free Software
27 Foundation
, Inc.
, 59 Temple Place
, Suite
330, Boston
, MA
02111-1307 USA
}
29 ######################################################################
31 ## Tcl/Tk sanity check
33 if {[catch
{package require Tcl
8.4} err
]
34 ||
[catch
{package require Tk
8.4} err
]
40 -title "git-gui: fatal error" \
45 ######################################################################
47 ## configure our library
49 set oguilib
{@@GITGUI_LIBDIR@@
}
50 set oguirel
{@@GITGUI_RELATIVE@@
}
51 if {$oguirel eq
{1}} {
52 set oguilib
[file dirname [file dirname [file normalize
$argv0]]]
53 set oguilib
[file join $oguilib share git-gui lib
]
54 } elseif
{[string match @@
* $oguirel]} {
55 set oguilib
[file join [file dirname [file normalize
$argv0]] lib
]
57 set idx
[file join $oguilib tclIndex
]
60 if {[gets
$fd] eq
{# Autogenerated by git-gui Makefile}} {
62 while {[gets
$fd n
] >= 0} {
63 if {$n ne
{} && ![string match
#* $n]} {
75 if {[lsearch
-exact $loaded $p] >= 0} continue
77 source [file join $oguilib $p]
82 set auto_path
[concat
[list
$oguilib] $auto_path]
84 unset -nocomplain oguilib oguirel idx fd
86 if {![catch
{set _verbose
$env(GITGUI_VERBOSE
)}]} {
88 rename auto_load real__auto_load
89 proc auto_load
{name args
} {
90 puts stderr
"auto_load $name"
91 return [uplevel
1 real__auto_load
$name $args]
93 rename
source real__source
95 puts stderr
"source $name"
96 uplevel
1 real__source
$name
100 ######################################################################
104 set _appname
[lindex
[file split $argv0] end
]
120 return [eval [concat
[list
file join $_gitdir] $args]]
123 proc gitexec
{args
} {
125 if {$_gitexec eq
{}} {
126 if {[catch
{set _gitexec
[git
--exec-path]} err
]} {
127 error
"Git not installed?\n\n$err"
133 return [eval [concat
[list
file join $_gitexec] $args]]
142 global tcl_platform tk_library
143 if {[tk windowingsystem
] eq
{aqua
}} {
151 if {$tcl_platform(platform
) eq
{windows
}} {
158 global tcl_platform _iscygwin
159 if {$_iscygwin eq
{}} {
160 if {$tcl_platform(platform
) eq
{windows
}} {
161 if {[catch
{set p
[exec cygpath
--windir]} err
]} {
173 proc is_enabled
{option
} {
174 global enabled_options
175 if {[catch
{set on
$enabled_options($option)}]} {return 0}
179 proc enable_option
{option
} {
180 global enabled_options
181 set enabled_options
($option) 1
184 proc disable_option
{option
} {
185 global enabled_options
186 set enabled_options
($option) 0
189 ######################################################################
193 proc is_many_config
{name
} {
194 switch
-glob -- $name {
203 proc is_config_true
{name
} {
205 if {[catch
{set v
$repo_config($name)}]} {
207 } elseif
{$v eq
{true
} ||
$v eq
{1} ||
$v eq
{yes}} {
214 proc load_config
{include_global
} {
215 global repo_config global_config default_config
217 array
unset global_config
218 if {$include_global} {
220 set fd_rc
[open
"| git config --global --list" r
]
221 while {[gets
$fd_rc line
] >= 0} {
222 if {[regexp
{^
([^
=]+)=(.
*)$
} $line line name value
]} {
223 if {[is_many_config
$name]} {
224 lappend global_config
($name) $value
226 set global_config
($name) $value
234 array
unset repo_config
236 set fd_rc
[open
"| git config --list" r
]
237 while {[gets
$fd_rc line
] >= 0} {
238 if {[regexp
{^
([^
=]+)=(.
*)$
} $line line name value
]} {
239 if {[is_many_config
$name]} {
240 lappend repo_config
($name) $value
242 set repo_config
($name) $value
249 foreach name
[array names default_config
] {
250 if {[catch
{set v
$global_config($name)}]} {
251 set global_config
($name) $default_config($name)
253 if {[catch
{set v
$repo_config($name)}]} {
254 set repo_config
($name) $default_config($name)
259 ######################################################################
264 return [eval exec git
$args]
267 auto_load tk_optionMenu
268 rename tk_optionMenu real__tkOptionMenu
269 proc tk_optionMenu
{w varName args
} {
270 set m
[eval real__tkOptionMenu
$w $varName $args]
271 $m configure
-font font_ui
272 $w configure
-font font_ui
276 ######################################################################
283 if {[catch
{set v
[git
--version]} err
]} {
284 catch
{wm withdraw .
}
285 error_popup
"Cannot determine Git version:
289 [appname] requires Git $req_maj.$req_min or later."
292 if {[regexp
{^git version
(\d
+)\.
(\d
+)} $v _junk act_maj act_min
]} {
293 if {$act_maj < $req_maj
294 ||
($act_maj == $req_maj && $act_min < $req_min)} {
295 catch
{wm withdraw .
}
296 error_popup
"[appname] requires Git $req_maj.$req_min or later.
302 catch
{wm withdraw .
}
303 error_popup
"Cannot parse Git version string:\n\n$v"
306 unset -nocomplain v _junk act_maj act_min req_maj req_min
308 ######################################################################
313 set _gitdir
$env(GIT_DIR
)
317 set _gitdir
[git rev-parse
--git-dir]
318 set _prefix
[git rev-parse
--show-prefix]
320 catch
{wm withdraw .
}
321 error_popup
"Cannot find the git directory:\n\n$err"
324 if {![file isdirectory
$_gitdir] && [is_Cygwin
]} {
325 catch
{set _gitdir
[exec cygpath
--unix $_gitdir]}
327 if {![file isdirectory
$_gitdir]} {
328 catch
{wm withdraw .
}
329 error_popup
"Git directory not found:\n\n$_gitdir"
332 if {[lindex
[file split $_gitdir] end
] ne
{.git
}} {
333 catch
{wm withdraw .
}
334 error_popup
"Cannot use funny .git directory:\n\n$_gitdir"
337 if {[catch
{cd [file dirname $_gitdir]} err
]} {
338 catch
{wm withdraw .
}
339 error_popup
"No working directory [file dirname $_gitdir]:\n\n$err"
342 set _reponame
[lindex
[file split \
343 [file normalize
[file dirname $_gitdir]]] \
346 ######################################################################
350 set current_diff_path
{}
351 set current_diff_side
{}
352 set diff_actions
[list
]
353 set ui_status_value
{Initializing...
}
357 set MERGE_HEAD
[list
]
360 set current_branch
{}
361 set current_diff_path
{}
362 set selected_commit_type new
364 ######################################################################
372 set disable_on_lock
[list
]
373 set index_lock_type none
375 proc lock_index
{type} {
376 global index_lock_type disable_on_lock
378 if {$index_lock_type eq
{none
}} {
379 set index_lock_type
$type
380 foreach w
$disable_on_lock {
381 uplevel
#0 $w disabled
384 } elseif
{$index_lock_type eq
"begin-$type"} {
385 set index_lock_type
$type
391 proc unlock_index
{} {
392 global index_lock_type disable_on_lock
394 set index_lock_type none
395 foreach w
$disable_on_lock {
400 ######################################################################
404 proc repository_state
{ctvar hdvar mhvar
} {
405 global current_branch
406 upvar
$ctvar ct
$hdvar hd
$mhvar mh
410 if {[catch
{set current_branch
[git symbolic-ref HEAD
]}]} {
411 set current_branch
{}
413 regsub ^refs
/((heads|tags|remotes
)/)? \
419 if {[catch
{set hd
[git rev-parse
--verify HEAD
]}]} {
425 set merge_head
[gitdir MERGE_HEAD
]
426 if {[file exists
$merge_head]} {
428 set fd_mh
[open
$merge_head r
]
429 while {[gets
$fd_mh line
] >= 0} {
440 global PARENT empty_tree
442 set p
[lindex
$PARENT 0]
446 if {$empty_tree eq
{}} {
447 set empty_tree
[git mktree
<< {}]
452 proc rescan
{after
{honor_trustmtime
1}} {
453 global HEAD PARENT MERGE_HEAD commit_type
454 global ui_index ui_workdir ui_status_value ui_comm
455 global rescan_active file_states
458 if {$rescan_active > 0 ||
![lock_index
read]} return
460 repository_state newType newHEAD newMERGE_HEAD
461 if {[string match amend
* $commit_type]
462 && $newType eq
{normal
}
463 && $newHEAD eq
$HEAD} {
467 set MERGE_HEAD
$newMERGE_HEAD
468 set commit_type
$newType
471 array
unset file_states
473 if {![$ui_comm edit modified
]
474 ||
[string trim
[$ui_comm get
0.0 end
]] eq
{}} {
475 if {[string match amend
* $commit_type]} {
476 } elseif
{[load_message GITGUI_MSG
]} {
477 } elseif
{[load_message MERGE_MSG
]} {
478 } elseif
{[load_message SQUASH_MSG
]} {
481 $ui_comm edit modified false
484 if {[is_enabled branch
]} {
489 if {$honor_trustmtime && $repo_config(gui.trustmtime
) eq
{true
}} {
490 rescan_stage2
{} $after
493 set ui_status_value
{Refreshing
file status...
}
494 set cmd
[list git update-index
]
496 lappend cmd
--unmerged
497 lappend cmd
--ignore-missing
498 lappend cmd
--refresh
499 set fd_rf
[open
"| $cmd" r
]
500 fconfigure
$fd_rf -blocking 0 -translation binary
501 fileevent
$fd_rf readable \
502 [list rescan_stage2
$fd_rf $after]
506 proc rescan_stage2
{fd after
} {
507 global ui_status_value
508 global rescan_active buf_rdi buf_rdf buf_rlo
512 if {![eof
$fd]} return
516 set ls_others
[list | git ls-files
--others -z \
517 --exclude-per-directory=.gitignore
]
518 set info_exclude
[gitdir info exclude
]
519 if {[file readable
$info_exclude]} {
520 lappend ls_others
"--exclude-from=$info_exclude"
528 set ui_status_value
{Scanning
for modified files ...
}
529 set fd_di
[open
"| git diff-index --cached -z [PARENT]" r
]
530 set fd_df
[open
"| git diff-files -z" r
]
531 set fd_lo
[open
$ls_others r
]
533 fconfigure
$fd_di -blocking 0 -translation binary
-encoding binary
534 fconfigure
$fd_df -blocking 0 -translation binary
-encoding binary
535 fconfigure
$fd_lo -blocking 0 -translation binary
-encoding binary
536 fileevent
$fd_di readable
[list read_diff_index
$fd_di $after]
537 fileevent
$fd_df readable
[list read_diff_files
$fd_df $after]
538 fileevent
$fd_lo readable
[list read_ls_others
$fd_lo $after]
541 proc load_message
{file} {
545 if {[file isfile
$f]} {
546 if {[catch
{set fd
[open
$f r
]}]} {
549 set content
[string trim
[read $fd]]
551 regsub
-all -line {[ \r\t]+$
} $content {} content
552 $ui_comm delete
0.0 end
553 $ui_comm insert end
$content
559 proc read_diff_index
{fd after
} {
562 append buf_rdi
[read $fd]
564 set n
[string length
$buf_rdi]
566 set z1
[string first
"\0" $buf_rdi $c]
569 set z2
[string first
"\0" $buf_rdi $z1]
573 set i
[split [string range
$buf_rdi $c [expr {$z1 - 2}]] { }]
574 set p
[string range
$buf_rdi $z1 [expr {$z2 - 1}]]
576 [encoding convertfrom
$p] \
578 [list
[lindex
$i 0] [lindex
$i 2]] \
584 set buf_rdi
[string range
$buf_rdi $c end
]
589 rescan_done
$fd buf_rdi
$after
592 proc read_diff_files
{fd after
} {
595 append buf_rdf
[read $fd]
597 set n
[string length
$buf_rdf]
599 set z1
[string first
"\0" $buf_rdf $c]
602 set z2
[string first
"\0" $buf_rdf $z1]
606 set i
[split [string range
$buf_rdf $c [expr {$z1 - 2}]] { }]
607 set p
[string range
$buf_rdf $z1 [expr {$z2 - 1}]]
609 [encoding convertfrom
$p] \
612 [list
[lindex
$i 0] [lindex
$i 2]]
617 set buf_rdf
[string range
$buf_rdf $c end
]
622 rescan_done
$fd buf_rdf
$after
625 proc read_ls_others
{fd after
} {
628 append buf_rlo
[read $fd]
629 set pck
[split $buf_rlo "\0"]
630 set buf_rlo
[lindex
$pck end
]
631 foreach p
[lrange
$pck 0 end-1
] {
632 merge_state
[encoding convertfrom
$p] ?O
634 rescan_done
$fd buf_rlo
$after
637 proc rescan_done
{fd buf after
} {
638 global rescan_active current_diff_path
639 global file_states repo_config
642 if {![eof
$fd]} return
645 if {[incr rescan_active
-1] > 0} return
650 if {$current_diff_path ne
{}} reshow_diff
654 proc prune_selection
{} {
655 global file_states selected_paths
657 foreach path
[array names selected_paths
] {
658 if {[catch
{set still_here
$file_states($path)}]} {
659 unset selected_paths
($path)
664 ######################################################################
668 proc mapicon
{w state path
} {
671 if {[catch
{set r
$all_icons($state$w)}]} {
672 puts
"error: no icon for $w state={$state} $path"
678 proc mapdesc
{state path
} {
681 if {[catch
{set r
$all_descs($state)}]} {
682 puts
"error: no desc for state={$state} $path"
688 proc escape_path
{path
} {
689 regsub
-all {\\} $path "\\\\" path
690 regsub
-all "\n" $path "\\n" path
694 proc short_path
{path
} {
695 return [escape_path
[lindex
[file split $path] end
]]
699 set null_sha1
[string repeat
0 40]
701 proc merge_state
{path new_state
{head_info
{}} {index_info
{}}} {
702 global file_states next_icon_id null_sha1
704 set s0
[string index
$new_state 0]
705 set s1
[string index
$new_state 1]
707 if {[catch
{set info
$file_states($path)}]} {
709 set icon n
[incr next_icon_id
]
711 set state
[lindex
$info 0]
712 set icon
[lindex
$info 1]
713 if {$head_info eq
{}} {set head_info
[lindex
$info 2]}
714 if {$index_info eq
{}} {set index_info
[lindex
$info 3]}
717 if {$s0 eq
{?
}} {set s0
[string index
$state 0]} \
718 elseif
{$s0 eq
{_
}} {set s0 _
}
720 if {$s1 eq
{?
}} {set s1
[string index
$state 1]} \
721 elseif
{$s1 eq
{_
}} {set s1 _
}
723 if {$s0 eq
{A
} && $s1 eq
{_
} && $head_info eq
{}} {
724 set head_info
[list
0 $null_sha1]
725 } elseif
{$s0 ne
{_
} && [string index
$state 0] eq
{_
}
726 && $head_info eq
{}} {
727 set head_info
$index_info
730 set file_states
($path) [list
$s0$s1 $icon \
731 $head_info $index_info \
736 proc display_file_helper
{w path icon_name old_m new_m
} {
740 set lno
[lsearch
-sorted -exact $file_lists($w) $path]
742 set file_lists
($w) [lreplace
$file_lists($w) $lno $lno]
744 $w conf
-state normal
745 $w delete
$lno.0 [expr {$lno + 1}].0
746 $w conf
-state disabled
748 } elseif
{$old_m eq
{_
} && $new_m ne
{_
}} {
749 lappend file_lists
($w) $path
750 set file_lists
($w) [lsort
-unique $file_lists($w)]
751 set lno
[lsearch
-sorted -exact $file_lists($w) $path]
753 $w conf
-state normal
754 $w image create
$lno.0 \
755 -align center
-padx 5 -pady 1 \
757 -image [mapicon
$w $new_m $path]
758 $w insert
$lno.1 "[escape_path $path]\n"
759 $w conf
-state disabled
760 } elseif
{$old_m ne
$new_m} {
761 $w conf
-state normal
762 $w image conf
$icon_name -image [mapicon
$w $new_m $path]
763 $w conf
-state disabled
767 proc display_file
{path state
} {
768 global file_states selected_paths
769 global ui_index ui_workdir
771 set old_m
[merge_state
$path $state]
772 set s
$file_states($path)
773 set new_m
[lindex
$s 0]
774 set icon_name
[lindex
$s 1]
776 set o
[string index
$old_m 0]
777 set n
[string index
$new_m 0]
784 display_file_helper
$ui_index $path $icon_name $o $n
786 if {[string index
$old_m 0] eq
{U
}} {
789 set o
[string index
$old_m 1]
791 if {[string index
$new_m 0] eq
{U
}} {
794 set n
[string index
$new_m 1]
796 display_file_helper
$ui_workdir $path $icon_name $o $n
798 if {$new_m eq
{__
}} {
799 unset file_states
($path)
800 catch
{unset selected_paths
($path)}
804 proc display_all_files_helper
{w path icon_name m
} {
807 lappend file_lists
($w) $path
808 set lno
[expr {[lindex
[split [$w index end
] .
] 0] - 1}]
809 $w image create end \
810 -align center
-padx 5 -pady 1 \
812 -image [mapicon
$w $m $path]
813 $w insert end
"[escape_path $path]\n"
816 proc display_all_files
{} {
817 global ui_index ui_workdir
818 global file_states file_lists
821 $ui_index conf
-state normal
822 $ui_workdir conf
-state normal
824 $ui_index delete
0.0 end
825 $ui_workdir delete
0.0 end
828 set file_lists
($ui_index) [list
]
829 set file_lists
($ui_workdir) [list
]
831 foreach path
[lsort
[array names file_states
]] {
832 set s
$file_states($path)
834 set icon_name
[lindex
$s 1]
836 set s
[string index
$m 0]
837 if {$s ne
{U
} && $s ne
{_
}} {
838 display_all_files_helper
$ui_index $path \
842 if {[string index
$m 0] eq
{U
}} {
845 set s
[string index
$m 1]
848 display_all_files_helper
$ui_workdir $path \
853 $ui_index conf
-state disabled
854 $ui_workdir conf
-state disabled
857 ######################################################################
862 #define mask_width 14
863 #define mask_height 15
864 static unsigned char mask_bits
[] = {
865 0xfe, 0x1f, 0xfe, 0x1f, 0xfe, 0x1f, 0xfe, 0x1f, 0xfe, 0x1f, 0xfe, 0x1f,
866 0xfe, 0x1f, 0xfe, 0x1f, 0xfe, 0x1f, 0xfe, 0x1f, 0xfe, 0x1f, 0xfe, 0x1f,
867 0xfe, 0x1f, 0xfe, 0x1f, 0xfe, 0x1f};
870 image create bitmap file_plain
-background white
-foreground black
-data {
871 #define plain_width 14
872 #define plain_height 15
873 static unsigned char plain_bits
[] = {
874 0xfe, 0x01, 0x02, 0x03, 0x02, 0x05, 0x02, 0x09, 0x02, 0x1f, 0x02, 0x10,
875 0x02, 0x10, 0x02, 0x10, 0x02, 0x10, 0x02, 0x10, 0x02, 0x10, 0x02, 0x10,
876 0x02, 0x10, 0x02, 0x10, 0xfe, 0x1f};
877 } -maskdata $filemask
879 image create bitmap file_mod
-background white
-foreground blue
-data {
881 #define mod_height 15
882 static unsigned char mod_bits
[] = {
883 0xfe, 0x01, 0x02, 0x03, 0x7a, 0x05, 0x02, 0x09, 0x7a, 0x1f, 0x02, 0x10,
884 0xfa, 0x17, 0x02, 0x10, 0xfa, 0x17, 0x02, 0x10, 0xfa, 0x17, 0x02, 0x10,
885 0xfa, 0x17, 0x02, 0x10, 0xfe, 0x1f};
886 } -maskdata $filemask
888 image create bitmap file_fulltick
-background white
-foreground "#007000" -data {
889 #define file_fulltick_width 14
890 #define file_fulltick_height 15
891 static unsigned char file_fulltick_bits
[] = {
892 0xfe, 0x01, 0x02, 0x1a, 0x02, 0x0c, 0x02, 0x0c, 0x02, 0x16, 0x02, 0x16,
893 0x02, 0x13, 0x00, 0x13, 0x86, 0x11, 0x8c, 0x11, 0xd8, 0x10, 0xf2, 0x10,
894 0x62, 0x10, 0x02, 0x10, 0xfe, 0x1f};
895 } -maskdata $filemask
897 image create bitmap file_parttick
-background white
-foreground "#005050" -data {
898 #define parttick_width 14
899 #define parttick_height 15
900 static unsigned char parttick_bits
[] = {
901 0xfe, 0x01, 0x02, 0x03, 0x7a, 0x05, 0x02, 0x09, 0x7a, 0x1f, 0x02, 0x10,
902 0x7a, 0x14, 0x02, 0x16, 0x02, 0x13, 0x8a, 0x11, 0xda, 0x10, 0x72, 0x10,
903 0x22, 0x10, 0x02, 0x10, 0xfe, 0x1f};
904 } -maskdata $filemask
906 image create bitmap file_question
-background white
-foreground black
-data {
907 #define file_question_width 14
908 #define file_question_height 15
909 static unsigned char file_question_bits
[] = {
910 0xfe, 0x01, 0x02, 0x02, 0xe2, 0x04, 0xf2, 0x09, 0x1a, 0x1b, 0x0a, 0x13,
911 0x82, 0x11, 0xc2, 0x10, 0x62, 0x10, 0x62, 0x10, 0x02, 0x10, 0x62, 0x10,
912 0x62, 0x10, 0x02, 0x10, 0xfe, 0x1f};
913 } -maskdata $filemask
915 image create bitmap file_removed
-background white
-foreground red
-data {
916 #define file_removed_width 14
917 #define file_removed_height 15
918 static unsigned char file_removed_bits
[] = {
919 0xfe, 0x01, 0x02, 0x03, 0x02, 0x05, 0x02, 0x09, 0x02, 0x1f, 0x02, 0x10,
920 0x1a, 0x16, 0x32, 0x13, 0xe2, 0x11, 0xc2, 0x10, 0xe2, 0x11, 0x32, 0x13,
921 0x1a, 0x16, 0x02, 0x10, 0xfe, 0x1f};
922 } -maskdata $filemask
924 image create bitmap file_merge
-background white
-foreground blue
-data {
925 #define file_merge_width 14
926 #define file_merge_height 15
927 static unsigned char file_merge_bits
[] = {
928 0xfe, 0x01, 0x02, 0x03, 0x62, 0x05, 0x62, 0x09, 0x62, 0x1f, 0x62, 0x10,
929 0xfa, 0x11, 0xf2, 0x10, 0x62, 0x10, 0x02, 0x10, 0xfa, 0x17, 0x02, 0x10,
930 0xfa, 0x17, 0x02, 0x10, 0xfe, 0x1f};
931 } -maskdata $filemask
934 #define file_width 18
935 #define file_height 18
936 static unsigned char file_bits
[] = {
937 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xf8, 0x03, 0x00,
938 0x0c, 0x03, 0x00, 0x04, 0xfe, 0x00, 0x06, 0x80, 0x00, 0xff, 0x9f, 0x00,
939 0x03, 0x98, 0x00, 0x02, 0x90, 0x00, 0x06, 0xb0, 0x00, 0x04, 0xa0, 0x00,
940 0x0c, 0xe0, 0x00, 0x08, 0xc0, 0x00, 0xf8, 0xff, 0x00, 0x00, 0x00, 0x00,
941 0x00, 0x00, 0x00, 0x00, 0x00, 0x00};
943 image create bitmap file_dir
-background white
-foreground blue \
944 -data $file_dir_data -maskdata $file_dir_data
947 set file_uplevel_data
{
950 static unsigned char up_bits
[] = {
951 0x80, 0x00, 0xc0, 0x01, 0xe0, 0x03, 0xf0, 0x07, 0xf8, 0x0f, 0xfc, 0x1f,
952 0xfe, 0x3f, 0xc0, 0x01, 0xc0, 0x01, 0xc0, 0x01, 0xc0, 0x01, 0xc0, 0x01,
953 0xc0, 0x01, 0xc0, 0x01, 0x00, 0x00};
955 image create bitmap file_uplevel
-background white
-foreground red \
956 -data $file_uplevel_data -maskdata $file_uplevel_data
957 unset file_uplevel_data
959 set ui_index .vpane.files.index.list
960 set ui_workdir .vpane.files.workdir.list
962 set all_icons
(_
$ui_index) file_plain
963 set all_icons
(A
$ui_index) file_fulltick
964 set all_icons
(M
$ui_index) file_fulltick
965 set all_icons
(D
$ui_index) file_removed
966 set all_icons
(U
$ui_index) file_merge
968 set all_icons
(_
$ui_workdir) file_plain
969 set all_icons
(M
$ui_workdir) file_mod
970 set all_icons
(D
$ui_workdir) file_question
971 set all_icons
(U
$ui_workdir) file_merge
972 set all_icons
(O
$ui_workdir) file_plain
974 set max_status_desc
0
978 {_M
"Modified, not staged"}
979 {M_
"Staged for commit"}
980 {MM
"Portions staged for commit"}
981 {MD
"Staged for commit, missing"}
983 {_O
"Untracked, not staged"}
984 {A_
"Staged for commit"}
985 {AM
"Portions staged for commit"}
986 {AD
"Staged for commit, missing"}
989 {D_
"Staged for removal"}
990 {DO
"Staged for removal, still present"}
992 {U_
"Requires merge resolution"}
993 {UU
"Requires merge resolution"}
994 {UM
"Requires merge resolution"}
995 {UD
"Requires merge resolution"}
997 if {$max_status_desc < [string length
[lindex
$i 1]]} {
998 set max_status_desc
[string length
[lindex
$i 1]]
1000 set all_descs
([lindex
$i 0]) [lindex
$i 1]
1004 ######################################################################
1008 proc bind_button3
{w cmd
} {
1009 bind $w <Any-Button-3
> $cmd
1011 bind $w <Control-Button-1
> $cmd
1015 proc scrollbar2many
{list mode args
} {
1016 foreach w
$list {eval $w $mode $args}
1019 proc many2scrollbar
{list mode sb top bottom
} {
1020 $sb set $top $bottom
1021 foreach w
$list {$w $mode moveto
$top}
1024 proc incr_font_size
{font
{amt
1}} {
1025 set sz
[font configure
$font -size]
1027 font configure
$font -size $sz
1028 font configure
${font}bold
-size $sz
1029 font configure
${font}italic
-size $sz
1032 ######################################################################
1036 set starting_gitk_msg
{Starting gitk... please
wait...
}
1038 proc do_gitk
{revs
} {
1039 global env ui_status_value starting_gitk_msg
1041 # -- Always start gitk through whatever we were loaded with. This
1042 # lets us bypass using shell process on Windows systems.
1044 set cmd
[list
[info nameofexecutable
]]
1045 set exe
[gitexec gitk
]
1052 if {! [file exists
$exe]} {
1053 error_popup
"Unable to start gitk:\n\n$exe does not exist"
1056 set ui_status_value
$starting_gitk_msg
1058 if {$ui_status_value eq
$starting_gitk_msg} {
1059 set ui_status_value
{Ready.
}
1068 global ui_comm is_quitting repo_config commit_type
1070 if {$is_quitting} return
1073 if {[winfo exists
$ui_comm]} {
1074 # -- Stash our current commit buffer.
1076 set save
[gitdir GITGUI_MSG
]
1077 set msg
[string trim
[$ui_comm get
0.0 end
]]
1078 regsub
-all -line {[ \r\t]+$
} $msg {} msg
1079 if {(![string match amend
* $commit_type]
1080 ||
[$ui_comm edit modified
])
1083 set fd
[open
$save w
]
1084 puts
-nonewline $fd $msg
1088 catch
{file delete
$save}
1091 # -- Stash our current window geometry into this repository.
1093 set cfg_geometry
[list
]
1094 lappend cfg_geometry
[wm geometry .
]
1095 lappend cfg_geometry
[lindex
[.vpane sash coord
0] 1]
1096 lappend cfg_geometry
[lindex
[.vpane.files sash coord
0] 0]
1097 if {[catch
{set rc_geometry
$repo_config(gui.geometry
)}]} {
1100 if {$cfg_geometry ne
$rc_geometry} {
1101 catch
{git config gui.geometry
$cfg_geometry}
1109 rescan
{set ui_status_value
{Ready.
}}
1116 proc toggle_or_diff
{w x y
} {
1117 global file_states file_lists current_diff_path ui_index ui_workdir
1118 global last_clicked selected_paths
1120 set pos
[split [$w index @
$x,$y] .
]
1121 set lno
[lindex
$pos 0]
1122 set col [lindex
$pos 1]
1123 set path
[lindex
$file_lists($w) [expr {$lno - 1}]]
1129 set last_clicked
[list
$w $lno]
1130 array
unset selected_paths
1131 $ui_index tag remove in_sel
0.0 end
1132 $ui_workdir tag remove in_sel
0.0 end
1135 if {$current_diff_path eq
$path} {
1136 set after
{reshow_diff
;}
1140 if {$w eq
$ui_index} {
1142 "Unstaging [short_path $path] from commit" \
1144 [concat
$after {set ui_status_value
{Ready.
}}]
1145 } elseif
{$w eq
$ui_workdir} {
1147 "Adding [short_path $path]" \
1149 [concat
$after {set ui_status_value
{Ready.
}}]
1152 show_diff
$path $w $lno
1156 proc add_one_to_selection
{w x y
} {
1157 global file_lists last_clicked selected_paths
1159 set lno
[lindex
[split [$w index @
$x,$y] .
] 0]
1160 set path
[lindex
$file_lists($w) [expr {$lno - 1}]]
1166 if {$last_clicked ne
{}
1167 && [lindex
$last_clicked 0] ne
$w} {
1168 array
unset selected_paths
1169 [lindex
$last_clicked 0] tag remove in_sel
0.0 end
1172 set last_clicked
[list
$w $lno]
1173 if {[catch
{set in_sel
$selected_paths($path)}]} {
1177 unset selected_paths
($path)
1178 $w tag remove in_sel
$lno.0 [expr {$lno + 1}].0
1180 set selected_paths
($path) 1
1181 $w tag add in_sel
$lno.0 [expr {$lno + 1}].0
1185 proc add_range_to_selection
{w x y
} {
1186 global file_lists last_clicked selected_paths
1188 if {[lindex
$last_clicked 0] ne
$w} {
1189 toggle_or_diff
$w $x $y
1193 set lno
[lindex
[split [$w index @
$x,$y] .
] 0]
1194 set lc
[lindex
$last_clicked 1]
1203 foreach path
[lrange
$file_lists($w) \
1204 [expr {$begin - 1}] \
1205 [expr {$end - 1}]] {
1206 set selected_paths
($path) 1
1208 $w tag add in_sel
$begin.0 [expr {$end + 1}].0
1211 ######################################################################
1215 set cursor_ptr arrow
1216 font create font_diff
-family Courier
-size 10
1220 eval font configure font_ui
[font actual
[.dummy cget
-font]]
1224 font create font_uiitalic
1225 font create font_uibold
1226 font create font_diffbold
1227 font create font_diffitalic
1229 foreach class
{Button Checkbutton Entry Label
1230 Labelframe Listbox Menu Message
1231 Radiobutton Spinbox Text
} {
1232 option add
*$class.font font_ui
1244 proc apply_config
{} {
1245 global repo_config font_descs
1247 foreach option
$font_descs {
1248 set name
[lindex
$option 0]
1249 set font
[lindex
$option 1]
1251 foreach
{cn cv
} $repo_config(gui.
$name) {
1252 font configure
$font $cn $cv
1255 error_popup
"Invalid font specified in gui.$name:\n\n$err"
1257 foreach
{cn cv
} [font configure
$font] {
1258 font configure
${font}bold
$cn $cv
1259 font configure
${font}italic
$cn $cv
1261 font configure
${font}bold
-weight bold
1262 font configure
${font}italic
-slant italic
1266 set default_config
(merge.summary
) false
1267 set default_config
(merge.verbosity
) 2
1268 set default_config
(user.name
) {}
1269 set default_config
(user.email
) {}
1271 set default_config
(gui.trustmtime
) false
1272 set default_config
(gui.diffcontext
) 5
1273 set default_config
(gui.newbranchtemplate
) {}
1274 set default_config
(gui.fontui
) [font configure font_ui
]
1275 set default_config
(gui.fontdiff
) [font configure font_diff
]
1277 {fontui font_ui
{Main Font
}}
1278 {fontdiff font_diff
{Diff
/Console Font
}}
1283 ######################################################################
1285 ## feature option selection
1287 if {[regexp
{^git-
(.
+)$
} [appname
] _junk subcommand
]} {
1292 if {$subcommand eq
{gui.sh
}} {
1295 if {$subcommand eq
{gui
} && [llength
$argv] > 0} {
1296 set subcommand
[lindex
$argv 0]
1297 set argv
[lrange
$argv 1 end
]
1300 enable_option multicommit
1301 enable_option branch
1302 enable_option transport
1304 switch
-- $subcommand {
1307 disable_option multicommit
1308 disable_option branch
1309 disable_option transport
1312 enable_option singlecommit
1314 disable_option multicommit
1315 disable_option branch
1316 disable_option transport
1320 ######################################################################
1328 menu .mbar
-tearoff 0
1329 .mbar add cascade
-label Repository
-menu .mbar.repository
1330 .mbar add cascade
-label Edit
-menu .mbar.edit
1331 if {[is_enabled branch
]} {
1332 .mbar add cascade
-label Branch
-menu .mbar.branch
1334 if {[is_enabled multicommit
] ||
[is_enabled singlecommit
]} {
1335 .mbar add cascade
-label Commit
-menu .mbar.commit
1337 if {[is_enabled transport
]} {
1338 .mbar add cascade
-label Merge
-menu .mbar.merge
1339 .mbar add cascade
-label Fetch
-menu .mbar.fetch
1340 .mbar add cascade
-label Push
-menu .mbar.push
1342 . configure
-menu .mbar
1344 # -- Repository Menu
1346 menu .mbar.repository
1348 .mbar.repository add
command \
1349 -label {Browse Current Branch
} \
1350 -command {browser
::new
$current_branch}
1351 trace add variable current_branch
write ".mbar.repository entryconf [.mbar.repository index last] -label \"Browse \$current_branch\" ;#"
1352 .mbar.repository add separator
1354 .mbar.repository add
command \
1355 -label {Visualize Current Branch
} \
1356 -command {do_gitk
$current_branch}
1357 trace add variable current_branch
write ".mbar.repository entryconf [.mbar.repository index last] -label \"Visualize \$current_branch\" ;#"
1358 .mbar.repository add
command \
1359 -label {Visualize All Branches
} \
1360 -command {do_gitk
--all}
1361 .mbar.repository add separator
1363 if {[is_enabled multicommit
]} {
1364 .mbar.repository add
command -label {Database Statistics
} \
1367 .mbar.repository add
command -label {Compress Database
} \
1370 .mbar.repository add
command -label {Verify Database
} \
1371 -command do_fsck_objects
1373 .mbar.repository add separator
1376 .mbar.repository add
command \
1377 -label {Create Desktop Icon
} \
1378 -command do_cygwin_shortcut
1379 } elseif
{[is_Windows
]} {
1380 .mbar.repository add
command \
1381 -label {Create Desktop Icon
} \
1382 -command do_windows_shortcut
1383 } elseif
{[is_MacOSX
]} {
1384 .mbar.repository add
command \
1385 -label {Create Desktop Icon
} \
1386 -command do_macosx_app
1390 .mbar.repository add
command -label Quit \
1397 .mbar.edit add
command -label Undo \
1398 -command {catch
{[focus
] edit undo
}} \
1400 .mbar.edit add
command -label Redo \
1401 -command {catch
{[focus
] edit redo
}} \
1403 .mbar.edit add separator
1404 .mbar.edit add
command -label Cut \
1405 -command {catch
{tk_textCut
[focus
]}} \
1407 .mbar.edit add
command -label Copy \
1408 -command {catch
{tk_textCopy
[focus
]}} \
1410 .mbar.edit add
command -label Paste \
1411 -command {catch
{tk_textPaste
[focus
]; [focus
] see insert
}} \
1413 .mbar.edit add
command -label Delete \
1414 -command {catch
{[focus
] delete sel.first sel.last
}} \
1416 .mbar.edit add separator
1417 .mbar.edit add
command -label {Select All
} \
1418 -command {catch
{[focus
] tag add sel
0.0 end
}} \
1423 if {[is_enabled branch
]} {
1426 .mbar.branch add
command -label {Create...
} \
1427 -command do_create_branch \
1429 lappend disable_on_lock
[list .mbar.branch entryconf \
1430 [.mbar.branch index last
] -state]
1432 .mbar.branch add
command -label {Delete...
} \
1433 -command do_delete_branch
1434 lappend disable_on_lock
[list .mbar.branch entryconf \
1435 [.mbar.branch index last
] -state]
1437 .mbar.branch add
command -label {Reset...
} \
1438 -command merge
::reset_hard
1439 lappend disable_on_lock
[list .mbar.branch entryconf \
1440 [.mbar.branch index last
] -state]
1445 if {[is_enabled multicommit
] ||
[is_enabled singlecommit
]} {
1448 .mbar.commit add radiobutton \
1449 -label {New Commit
} \
1450 -command do_select_commit_type \
1451 -variable selected_commit_type \
1453 lappend disable_on_lock \
1454 [list .mbar.commit entryconf
[.mbar.commit index last
] -state]
1456 .mbar.commit add radiobutton \
1457 -label {Amend Last Commit
} \
1458 -command do_select_commit_type \
1459 -variable selected_commit_type \
1461 lappend disable_on_lock \
1462 [list .mbar.commit entryconf
[.mbar.commit index last
] -state]
1464 .mbar.commit add separator
1466 .mbar.commit add
command -label Rescan \
1467 -command do_rescan \
1469 lappend disable_on_lock \
1470 [list .mbar.commit entryconf
[.mbar.commit index last
] -state]
1472 .mbar.commit add
command -label {Add To Commit
} \
1473 -command do_add_selection
1474 lappend disable_on_lock \
1475 [list .mbar.commit entryconf
[.mbar.commit index last
] -state]
1477 .mbar.commit add
command -label {Add Existing To Commit
} \
1478 -command do_add_all \
1480 lappend disable_on_lock \
1481 [list .mbar.commit entryconf
[.mbar.commit index last
] -state]
1483 .mbar.commit add
command -label {Unstage From Commit
} \
1484 -command do_unstage_selection
1485 lappend disable_on_lock \
1486 [list .mbar.commit entryconf
[.mbar.commit index last
] -state]
1488 .mbar.commit add
command -label {Revert Changes
} \
1489 -command do_revert_selection
1490 lappend disable_on_lock \
1491 [list .mbar.commit entryconf
[.mbar.commit index last
] -state]
1493 .mbar.commit add separator
1495 .mbar.commit add
command -label {Sign Off
} \
1496 -command do_signoff \
1499 .mbar.commit add
command -label Commit \
1500 -command do_commit \
1501 -accelerator $M1T-Return
1502 lappend disable_on_lock \
1503 [list .mbar.commit entryconf
[.mbar.commit index last
] -state]
1508 if {[is_enabled branch
]} {
1510 .mbar.merge add
command -label {Local Merge...
} \
1511 -command merge
::dialog
1512 lappend disable_on_lock \
1513 [list .mbar.merge entryconf
[.mbar.merge index last
] -state]
1514 .mbar.merge add
command -label {Abort Merge...
} \
1515 -command merge
::reset_hard
1516 lappend disable_on_lock \
1517 [list .mbar.merge entryconf
[.mbar.merge index last
] -state]
1523 if {[is_enabled transport
]} {
1527 .mbar.push add
command -label {Push...
} \
1528 -command do_push_anywhere \
1533 # -- Apple Menu (Mac OS X only)
1535 .mbar add cascade
-label Apple
-menu .mbar.apple
1538 .mbar.apple add
command -label "About [appname]" \
1540 .mbar.apple add
command -label "Options..." \
1545 .mbar.edit add separator
1546 .mbar.edit add
command -label {Options...
} \
1551 if {[is_Cygwin
] && [file exists
/usr
/local
/miga
/lib
/gui-miga
]} {
1553 global ui_status_value
1554 if {![lock_index update
]} return
1555 set cmd
[list sh
--login -c "/usr/local/miga/lib/gui-miga \"[pwd]\""]
1556 set miga_fd
[open
"|$cmd" r
]
1557 fconfigure
$miga_fd -blocking 0
1558 fileevent
$miga_fd readable
[list miga_done
$miga_fd]
1559 set ui_status_value
{Running miga...
}
1561 proc miga_done
{fd
} {
1566 rescan
[list
set ui_status_value
{Ready.
}]
1569 .mbar add cascade
-label Tools
-menu .mbar.tools
1571 .mbar.tools add
command -label "Migrate" \
1573 lappend disable_on_lock \
1574 [list .mbar.tools entryconf
[.mbar.tools index last
] -state]
1580 .mbar add cascade
-label Help
-menu .mbar.
help
1584 .mbar.
help add
command -label "About [appname]" \
1589 catch
{set browser
$repo_config(instaweb.browser
)}
1590 set doc_path
[file dirname [gitexec
]]
1591 set doc_path
[file join $doc_path Documentation index.html
]
1594 set doc_path
[exec cygpath
--mixed $doc_path]
1597 if {$browser eq
{}} {
1600 } elseif
{[is_Cygwin
]} {
1601 set program_files
[file dirname [exec cygpath
--windir]]
1602 set program_files
[file join $program_files {Program Files
}]
1603 set firefox
[file join $program_files {Mozilla Firefox
} firefox.exe
]
1604 set ie
[file join $program_files {Internet Explorer
} IEXPLORE.EXE
]
1605 if {[file exists
$firefox]} {
1606 set browser
$firefox
1607 } elseif
{[file exists
$ie]} {
1610 unset program_files firefox ie
1614 if {[file isfile
$doc_path]} {
1615 set doc_url
"file:$doc_path"
1617 set doc_url
{http
://www.kernel.org
/pub
/software
/scm
/git
/docs
/}
1620 if {$browser ne
{}} {
1621 .mbar.
help add
command -label {Online Documentation
} \
1622 -command [list
exec $browser $doc_url &]
1624 unset browser doc_path doc_url
1626 # -- Standard bindings
1628 wm protocol . WM_DELETE_WINDOW do_quit
1629 bind all
<$M1B-Key-q> do_quit
1630 bind all
<$M1B-Key-Q> do_quit
1631 bind all
<$M1B-Key-w> {destroy
[winfo toplevel
%W
]}
1632 bind all
<$M1B-Key-W> {destroy
[winfo toplevel
%W
]}
1634 set subcommand_args
{}
1636 puts stderr
"usage: $::argv0 $::subcommand $::subcommand_args"
1640 # -- Not a normal commit type invocation? Do that instead!
1642 switch
-- $subcommand {
1644 set subcommand_args
{rev?
}
1645 switch
[llength
$argv] {
1647 set current_branch
[git symbolic-ref HEAD
]
1648 regsub ^refs
/((heads|tags|remotes
)/)? \
1649 $current_branch {} current_branch
1652 set current_branch
[lindex
$argv 0]
1656 browser
::new
$current_branch
1660 set subcommand_args
{rev? path?
}
1665 if {$is_path ||
[file exists
$_prefix$a]} {
1666 if {$path ne
{}} usage
1669 } elseif
{$a eq
{--}} {
1671 if {$head ne
{}} usage
1676 } elseif
{$head eq
{}} {
1677 if {$head ne
{}} usage
1686 set current_branch
[git symbolic-ref HEAD
]
1687 regsub ^refs
/((heads|tags|remotes
)/)? \
1688 $current_branch {} current_branch
1690 set current_branch
$head
1693 if {$path eq
{}} usage
1694 blame
::new
$head $path
1699 if {[llength
$argv] != 0} {
1700 puts
-nonewline stderr
"usage: $argv0"
1701 if {$subcommand ne
{gui
} && [appname
] ne
"git-$subcommand"} {
1702 puts
-nonewline stderr
" $subcommand"
1707 # fall through to setup UI for commits
1710 puts stderr
"usage: $argv0 \[{blame|browser|citool}\]"
1721 -text {Current Branch
:} \
1725 -textvariable current_branch \
1728 pack .branch.l1
-side left
1729 pack .branch.cb
-side left
-fill x
1730 pack .branch
-side top
-fill x
1732 # -- Main Window Layout
1734 panedwindow .vpane
-orient vertical
1735 panedwindow .vpane.files
-orient horizontal
1736 .vpane add .vpane.files
-sticky nsew
-height 100 -width 200
1737 pack .vpane
-anchor n
-side top
-fill both
-expand 1
1739 # -- Index File List
1741 frame .vpane.files.index
-height 100 -width 200
1742 label .vpane.files.index.title
-text {Staged Changes
(Will Be Committed
)} \
1743 -background lightgreen
1744 text
$ui_index -background white
-borderwidth 0 \
1745 -width 20 -height 10 \
1747 -cursor $cursor_ptr \
1748 -xscrollcommand {.vpane.files.index.sx
set} \
1749 -yscrollcommand {.vpane.files.index.sy
set} \
1751 scrollbar .vpane.files.index.sx
-orient h
-command [list
$ui_index xview
]
1752 scrollbar .vpane.files.index.sy
-orient v
-command [list
$ui_index yview
]
1753 pack .vpane.files.index.title
-side top
-fill x
1754 pack .vpane.files.index.sx
-side bottom
-fill x
1755 pack .vpane.files.index.sy
-side right
-fill y
1756 pack
$ui_index -side left
-fill both
-expand 1
1757 .vpane.files add .vpane.files.index
-sticky nsew
1759 # -- Working Directory File List
1761 frame .vpane.files.workdir
-height 100 -width 200
1762 label .vpane.files.workdir.title
-text {Unstaged Changes
(Will Not Be Committed
)} \
1763 -background lightsalmon
1764 text
$ui_workdir -background white
-borderwidth 0 \
1765 -width 20 -height 10 \
1767 -cursor $cursor_ptr \
1768 -xscrollcommand {.vpane.files.workdir.sx
set} \
1769 -yscrollcommand {.vpane.files.workdir.sy
set} \
1771 scrollbar .vpane.files.workdir.sx
-orient h
-command [list
$ui_workdir xview
]
1772 scrollbar .vpane.files.workdir.sy
-orient v
-command [list
$ui_workdir yview
]
1773 pack .vpane.files.workdir.title
-side top
-fill x
1774 pack .vpane.files.workdir.sx
-side bottom
-fill x
1775 pack .vpane.files.workdir.sy
-side right
-fill y
1776 pack
$ui_workdir -side left
-fill both
-expand 1
1777 .vpane.files add .vpane.files.workdir
-sticky nsew
1779 foreach i
[list
$ui_index $ui_workdir] {
1780 $i tag conf in_diff
-background lightgray
1781 $i tag conf in_sel
-background lightgray
1785 # -- Diff and Commit Area
1787 frame .vpane.lower
-height 300 -width 400
1788 frame .vpane.lower.commarea
1789 frame .vpane.lower.
diff -relief sunken
-borderwidth 1
1790 pack .vpane.lower.commarea
-side top
-fill x
1791 pack .vpane.lower.
diff -side bottom
-fill both
-expand 1
1792 .vpane add .vpane.lower
-sticky nsew
1794 # -- Commit Area Buttons
1796 frame .vpane.lower.commarea.buttons
1797 label .vpane.lower.commarea.buttons.l
-text {} \
1800 pack .vpane.lower.commarea.buttons.l
-side top
-fill x
1801 pack .vpane.lower.commarea.buttons
-side left
-fill y
1803 button .vpane.lower.commarea.buttons.rescan
-text {Rescan
} \
1805 pack .vpane.lower.commarea.buttons.rescan
-side top
-fill x
1806 lappend disable_on_lock \
1807 {.vpane.lower.commarea.buttons.rescan conf
-state}
1809 button .vpane.lower.commarea.buttons.incall
-text {Add Existing
} \
1811 pack .vpane.lower.commarea.buttons.incall
-side top
-fill x
1812 lappend disable_on_lock \
1813 {.vpane.lower.commarea.buttons.incall conf
-state}
1815 button .vpane.lower.commarea.buttons.signoff
-text {Sign Off
} \
1817 pack .vpane.lower.commarea.buttons.signoff
-side top
-fill x
1819 button .vpane.lower.commarea.buttons.commit
-text {Commit
} \
1821 pack .vpane.lower.commarea.buttons.commit
-side top
-fill x
1822 lappend disable_on_lock \
1823 {.vpane.lower.commarea.buttons.commit conf
-state}
1825 button .vpane.lower.commarea.buttons.push
-text {Push
} \
1826 -command do_push_anywhere
1827 pack .vpane.lower.commarea.buttons.push
-side top
-fill x
1829 # -- Commit Message Buffer
1831 frame .vpane.lower.commarea.buffer
1832 frame .vpane.lower.commarea.buffer.header
1833 set ui_comm .vpane.lower.commarea.buffer.t
1834 set ui_coml .vpane.lower.commarea.buffer.header.l
1835 radiobutton .vpane.lower.commarea.buffer.header.new \
1836 -text {New Commit
} \
1837 -command do_select_commit_type \
1838 -variable selected_commit_type \
1840 lappend disable_on_lock \
1841 [list .vpane.lower.commarea.buffer.header.new conf
-state]
1842 radiobutton .vpane.lower.commarea.buffer.header.amend \
1843 -text {Amend Last Commit
} \
1844 -command do_select_commit_type \
1845 -variable selected_commit_type \
1847 lappend disable_on_lock \
1848 [list .vpane.lower.commarea.buffer.header.amend conf
-state]
1852 proc trace_commit_type
{varname args
} {
1853 global ui_coml commit_type
1854 switch
-glob -- $commit_type {
1855 initial
{set txt
{Initial Commit Message
:}}
1856 amend
{set txt
{Amended Commit Message
:}}
1857 amend-initial
{set txt
{Amended Initial Commit Message
:}}
1858 amend-merge
{set txt
{Amended Merge Commit Message
:}}
1859 merge
{set txt
{Merge Commit Message
:}}
1860 * {set txt
{Commit Message
:}}
1862 $ui_coml conf
-text $txt
1864 trace add variable commit_type
write trace_commit_type
1865 pack
$ui_coml -side left
-fill x
1866 pack .vpane.lower.commarea.buffer.header.amend
-side right
1867 pack .vpane.lower.commarea.buffer.header.new
-side right
1869 text
$ui_comm -background white
-borderwidth 1 \
1872 -autoseparators true \
1874 -width 75 -height 9 -wrap none \
1876 -yscrollcommand {.vpane.lower.commarea.buffer.sby
set}
1877 scrollbar .vpane.lower.commarea.buffer.sby \
1878 -command [list
$ui_comm yview
]
1879 pack .vpane.lower.commarea.buffer.header
-side top
-fill x
1880 pack .vpane.lower.commarea.buffer.sby
-side right
-fill y
1881 pack
$ui_comm -side left
-fill y
1882 pack .vpane.lower.commarea.buffer
-side left
-fill y
1884 # -- Commit Message Buffer Context Menu
1886 set ctxm .vpane.lower.commarea.buffer.ctxm
1887 menu
$ctxm -tearoff 0
1890 -command {tk_textCut
$ui_comm}
1893 -command {tk_textCopy
$ui_comm}
1896 -command {tk_textPaste
$ui_comm}
1899 -command {$ui_comm delete sel.first sel.last
}
1902 -label {Select All
} \
1903 -command {focus
$ui_comm;$ui_comm tag add sel
0.0 end
}
1907 $ui_comm tag add sel
0.0 end
1908 tk_textCopy
$ui_comm
1909 $ui_comm tag remove sel
0.0 end
1915 bind_button3
$ui_comm "tk_popup $ctxm %X %Y"
1919 proc trace_current_diff_path
{varname args
} {
1920 global current_diff_path diff_actions file_states
1921 if {$current_diff_path eq
{}} {
1927 set p
$current_diff_path
1928 set s
[mapdesc
[lindex
$file_states($p) 0] $p]
1930 set p
[escape_path
$p]
1934 .vpane.lower.
diff.header.status configure
-text $s
1935 .vpane.lower.
diff.header.
file configure
-text $f
1936 .vpane.lower.
diff.header.path configure
-text $p
1937 foreach w
$diff_actions {
1941 trace add variable current_diff_path
write trace_current_diff_path
1943 frame .vpane.lower.
diff.header
-background gold
1944 label .vpane.lower.
diff.header.status \
1946 -width $max_status_desc \
1949 label .vpane.lower.
diff.header.
file \
1953 label .vpane.lower.
diff.header.path \
1957 pack .vpane.lower.
diff.header.status
-side left
1958 pack .vpane.lower.
diff.header.
file -side left
1959 pack .vpane.lower.
diff.header.path
-fill x
1960 set ctxm .vpane.lower.
diff.header.ctxm
1961 menu
$ctxm -tearoff 0
1969 -- $current_diff_path
1971 lappend diff_actions
[list
$ctxm entryconf
[$ctxm index last
] -state]
1972 bind_button3 .vpane.lower.
diff.header.path
"tk_popup $ctxm %X %Y"
1976 frame .vpane.lower.
diff.body
1977 set ui_diff .vpane.lower.
diff.body.t
1978 text
$ui_diff -background white
-borderwidth 0 \
1979 -width 80 -height 15 -wrap none \
1981 -xscrollcommand {.vpane.lower.
diff.body.sbx
set} \
1982 -yscrollcommand {.vpane.lower.
diff.body.sby
set} \
1984 scrollbar .vpane.lower.
diff.body.sbx
-orient horizontal \
1985 -command [list
$ui_diff xview
]
1986 scrollbar .vpane.lower.
diff.body.sby
-orient vertical \
1987 -command [list
$ui_diff yview
]
1988 pack .vpane.lower.
diff.body.sbx
-side bottom
-fill x
1989 pack .vpane.lower.
diff.body.sby
-side right
-fill y
1990 pack
$ui_diff -side left
-fill both
-expand 1
1991 pack .vpane.lower.
diff.header
-side top
-fill x
1992 pack .vpane.lower.
diff.body
-side bottom
-fill both
-expand 1
1994 $ui_diff tag conf d_cr
-elide true
1995 $ui_diff tag conf d_@
-foreground blue
-font font_diffbold
1996 $ui_diff tag conf d_
+ -foreground {#00a000}
1997 $ui_diff tag conf d_-
-foreground red
1999 $ui_diff tag conf d_
++ -foreground {#00a000}
2000 $ui_diff tag conf d_--
-foreground red
2001 $ui_diff tag conf d_
+s \
2002 -foreground {#00a000} \
2003 -background {#e2effa}
2004 $ui_diff tag conf d_-s \
2006 -background {#e2effa}
2007 $ui_diff tag conf d_s
+ \
2008 -foreground {#00a000} \
2010 $ui_diff tag conf d_s- \
2014 $ui_diff tag conf d
<<<<<<< \
2015 -foreground orange \
2017 $ui_diff tag conf d
======= \
2018 -foreground orange \
2020 $ui_diff tag conf d
>>>>>>> \
2021 -foreground orange \
2024 $ui_diff tag raise sel
2026 # -- Diff Body Context Menu
2028 set ctxm .vpane.lower.
diff.body.ctxm
2029 menu
$ctxm -tearoff 0
2032 -command reshow_diff
2033 lappend diff_actions
[list
$ctxm entryconf
[$ctxm index last
] -state]
2036 -command {tk_textCopy
$ui_diff}
2037 lappend diff_actions
[list
$ctxm entryconf
[$ctxm index last
] -state]
2039 -label {Select All
} \
2040 -command {focus
$ui_diff;$ui_diff tag add sel
0.0 end
}
2041 lappend diff_actions
[list
$ctxm entryconf
[$ctxm index last
] -state]
2045 $ui_diff tag add sel
0.0 end
2046 tk_textCopy
$ui_diff
2047 $ui_diff tag remove sel
0.0 end
2049 lappend diff_actions
[list
$ctxm entryconf
[$ctxm index last
] -state]
2052 -label {Apply
/Reverse Hunk
} \
2053 -command {apply_hunk
$cursorX $cursorY}
2054 set ui_diff_applyhunk
[$ctxm index last
]
2055 lappend diff_actions
[list
$ctxm entryconf
$ui_diff_applyhunk -state]
2058 -label {Decrease Font Size
} \
2059 -command {incr_font_size font_diff
-1}
2060 lappend diff_actions
[list
$ctxm entryconf
[$ctxm index last
] -state]
2062 -label {Increase Font Size
} \
2063 -command {incr_font_size font_diff
1}
2064 lappend diff_actions
[list
$ctxm entryconf
[$ctxm index last
] -state]
2067 -label {Show Less Context
} \
2068 -command {if {$repo_config(gui.diffcontext
) >= 1} {
2069 incr repo_config
(gui.diffcontext
) -1
2072 lappend diff_actions
[list
$ctxm entryconf
[$ctxm index last
] -state]
2074 -label {Show More Context
} \
2075 -command {if {$repo_config(gui.diffcontext
) < 99} {
2076 incr repo_config
(gui.diffcontext
)
2079 lappend diff_actions
[list
$ctxm entryconf
[$ctxm index last
] -state]
2081 $ctxm add
command -label {Options...
} \
2083 bind_button3
$ui_diff "
2086 if {\$ui_index eq \$current_diff_side} {
2087 $ctxm entryconf $ui_diff_applyhunk -label {Unstage Hunk From Commit}
2089 $ctxm entryconf $ui_diff_applyhunk -label {Stage Hunk For Commit}
2091 tk_popup $ctxm %X %Y
2093 unset ui_diff_applyhunk
2097 label .status
-textvariable ui_status_value \
2102 pack .status
-anchor w
-side bottom
-fill x
2107 set gm
$repo_config(gui.geometry
)
2108 wm geometry .
[lindex
$gm 0]
2109 .vpane sash place
0 \
2110 [lindex
[.vpane sash coord
0] 0] \
2112 .vpane.files sash place
0 \
2114 [lindex
[.vpane.files sash coord
0] 1]
2120 bind $ui_comm <$M1B-Key-Return> {do_commit
;break}
2121 bind $ui_comm <$M1B-Key-i> {do_add_all
;break}
2122 bind $ui_comm <$M1B-Key-I> {do_add_all
;break}
2123 bind $ui_comm <$M1B-Key-x> {tk_textCut
%W
;break}
2124 bind $ui_comm <$M1B-Key-X> {tk_textCut
%W
;break}
2125 bind $ui_comm <$M1B-Key-c> {tk_textCopy
%W
;break}
2126 bind $ui_comm <$M1B-Key-C> {tk_textCopy
%W
;break}
2127 bind $ui_comm <$M1B-Key-v> {tk_textPaste
%W
; %W see insert
; break}
2128 bind $ui_comm <$M1B-Key-V> {tk_textPaste
%W
; %W see insert
; break}
2129 bind $ui_comm <$M1B-Key-a> {%W tag add sel
0.0 end
;break}
2130 bind $ui_comm <$M1B-Key-A> {%W tag add sel
0.0 end
;break}
2132 bind $ui_diff <$M1B-Key-x> {tk_textCopy
%W
;break}
2133 bind $ui_diff <$M1B-Key-X> {tk_textCopy
%W
;break}
2134 bind $ui_diff <$M1B-Key-c> {tk_textCopy
%W
;break}
2135 bind $ui_diff <$M1B-Key-C> {tk_textCopy
%W
;break}
2136 bind $ui_diff <$M1B-Key-v> {break}
2137 bind $ui_diff <$M1B-Key-V> {break}
2138 bind $ui_diff <$M1B-Key-a> {%W tag add sel
0.0 end
;break}
2139 bind $ui_diff <$M1B-Key-A> {%W tag add sel
0.0 end
;break}
2140 bind $ui_diff <Key-Up
> {catch
{%W yview scroll
-1 units
};break}
2141 bind $ui_diff <Key-Down
> {catch
{%W yview scroll
1 units
};break}
2142 bind $ui_diff <Key-Left
> {catch
{%W xview scroll
-1 units
};break}
2143 bind $ui_diff <Key-Right
> {catch
{%W xview scroll
1 units
};break}
2144 bind $ui_diff <Key-k
> {catch
{%W yview scroll
-1 units
};break}
2145 bind $ui_diff <Key-j
> {catch
{%W yview scroll
1 units
};break}
2146 bind $ui_diff <Key-h
> {catch
{%W xview scroll
-1 units
};break}
2147 bind $ui_diff <Key-l
> {catch
{%W xview scroll
1 units
};break}
2148 bind $ui_diff <Control-Key-b
> {catch
{%W yview scroll
-1 pages
};break}
2149 bind $ui_diff <Control-Key-f
> {catch
{%W yview scroll
1 pages
};break}
2150 bind $ui_diff <Button-1
> {focus
%W
}
2152 if {[is_enabled branch
]} {
2153 bind .
<$M1B-Key-n> do_create_branch
2154 bind .
<$M1B-Key-N> do_create_branch
2156 if {[is_enabled transport
]} {
2157 bind .
<$M1B-Key-p> do_push_anywhere
2158 bind .
<$M1B-Key-P> do_push_anywhere
2161 bind .
<Key-F5
> do_rescan
2162 bind .
<$M1B-Key-r> do_rescan
2163 bind .
<$M1B-Key-R> do_rescan
2164 bind .
<$M1B-Key-s> do_signoff
2165 bind .
<$M1B-Key-S> do_signoff
2166 bind .
<$M1B-Key-i> do_add_all
2167 bind .
<$M1B-Key-I> do_add_all
2168 bind .
<$M1B-Key-Return> do_commit
2169 foreach i
[list
$ui_index $ui_workdir] {
2170 bind $i <Button-1
> "toggle_or_diff $i %x %y; break"
2171 bind $i <$M1B-Button-1> "add_one_to_selection $i %x %y; break"
2172 bind $i <Shift-Button-1
> "add_range_to_selection $i %x %y; break"
2176 set file_lists
($ui_index) [list
]
2177 set file_lists
($ui_workdir) [list
]
2179 wm title .
"[appname] ([reponame]) [file normalize [file dirname [gitdir]]]"
2180 focus
-force $ui_comm
2182 # -- Warn the user about environmental problems. Cygwin's Tcl
2183 # does *not* pass its env array onto any processes it spawns.
2184 # This means that git processes get none of our environment.
2189 set msg
"Possible environment issues exist.
2191 The following environment variables are probably
2192 going to be ignored by any Git subprocess run
2196 foreach name
[array names env
] {
2197 switch
-regexp -- $name {
2198 {^GIT_INDEX_FILE$
} -
2199 {^GIT_OBJECT_DIRECTORY$
} -
2200 {^GIT_ALTERNATE_OBJECT_DIRECTORIES$
} -
2202 {^GIT_EXTERNAL_DIFF$
} -
2206 {^GIT_CONFIG_LOCAL$
} -
2207 {^GIT_
(AUTHOR|COMMITTER
)_DATE$
} {
2208 append msg
" - $name\n"
2211 {^GIT_
(AUTHOR|COMMITTER
)_
(NAME|EMAIL
)$
} {
2212 append msg
" - $name\n"
2214 set suggest_user
$name
2218 if {$ignored_env > 0} {
2220 This is due to a known issue with the
2221 Tcl binary distributed by Cygwin."
2223 if {$suggest_user ne
{}} {
2226 A good replacement for $suggest_user
2227 is placing values for the user.name and
2228 user.email settings into your personal
2234 unset ignored_env msg suggest_user name
2237 # -- Only initialize complex UI if we are going to stay running.
2239 if {[is_enabled transport
]} {
2243 populate_branch_menu
2248 # -- Only suggest a gc run if we are going to stay running.
2250 if {[is_enabled multicommit
]} {
2251 set object_limit
2000
2252 if {[is_Windows
]} {set object_limit
200}
2253 regexp
{^
([0-9]+) objects
,} [git count-objects
] _junk objects_current
2254 if {$objects_current >= $object_limit} {
2256 "This repository currently has $objects_current loose objects.
2258 To maintain optimal performance it is strongly recommended that you compress the database when more than $object_limit loose objects exist.
2260 Compress the database now?"] eq
yes} {
2264 unset object_limit _junk objects_current
2267 lock_index begin-read