Shawn O. Pearce | d41b43e | 2007-07-08 18:40:56 -0400 | [diff] [blame] | 1 | # git-gui commit checkout support |
| 2 | # Copyright (C) 2007 Shawn Pearce |
| 3 | |
| 4 | class checkout_op { |
| 5 | |
| 6 | field w {}; # our window (if we have one) |
| 7 | field w_cons {}; # embedded console window object |
| 8 | |
| 9 | field new_expr ; # expression the user saw/thinks this is |
| 10 | field new_hash ; # commit SHA-1 we are switching to |
| 11 | field new_ref ; # ref we are updating/creating |
Jens Lehmann | c949833 | 2009-03-30 21:46:17 +0200 | [diff] [blame] | 12 | field old_hash ; # commit SHA-1 that was checked out when we started |
Shawn O. Pearce | d41b43e | 2007-07-08 18:40:56 -0400 | [diff] [blame] | 13 | |
| 14 | field parent_w .; # window that started us |
| 15 | field merge_type none; # type of merge to apply to existing branch |
Shawn O. Pearce | 60f7352 | 2007-07-20 02:13:24 -0400 | [diff] [blame] | 16 | field merge_base {}; # merge base if we have another ref involved |
Shawn O. Pearce | d41b43e | 2007-07-08 18:40:56 -0400 | [diff] [blame] | 17 | field fetch_spec {}; # refetch tracking branch if used? |
| 18 | field checkout 1; # actually checkout the branch? |
| 19 | field create 0; # create the branch if it doesn't exist? |
Shawn O. Pearce | fe70225 | 2008-05-08 20:16:43 -0400 | [diff] [blame] | 20 | field remote_source {}; # same as fetch_spec, to setup tracking |
Shawn O. Pearce | d41b43e | 2007-07-08 18:40:56 -0400 | [diff] [blame] | 21 | |
| 22 | field reset_ok 0; # did the user agree to reset? |
| 23 | field fetch_ok 0; # did the fetch succeed? |
| 24 | |
Shawn O. Pearce | b792230 | 2007-07-08 22:48:19 -0400 | [diff] [blame] | 25 | field readtree_d {}; # buffered output from read-tree |
Shawn O. Pearce | d41b43e | 2007-07-08 18:40:56 -0400 | [diff] [blame] | 26 | field update_old {}; # was the update-ref call deferred? |
| 27 | field reflog_msg {}; # log message for the update-ref call |
| 28 | |
| 29 | constructor new {expr hash {ref {}}} { |
| 30 | set new_expr $expr |
| 31 | set new_hash $hash |
| 32 | set new_ref $ref |
| 33 | |
| 34 | return $this |
| 35 | } |
| 36 | |
| 37 | method parent {path} { |
| 38 | set parent_w [winfo toplevel $path] |
| 39 | } |
| 40 | |
| 41 | method enable_merge {type} { |
| 42 | set merge_type $type |
| 43 | } |
| 44 | |
| 45 | method enable_fetch {spec} { |
| 46 | set fetch_spec $spec |
| 47 | } |
| 48 | |
Shawn O. Pearce | fe70225 | 2008-05-08 20:16:43 -0400 | [diff] [blame] | 49 | method remote_source {spec} { |
| 50 | set remote_source $spec |
| 51 | } |
| 52 | |
Shawn O. Pearce | d41b43e | 2007-07-08 18:40:56 -0400 | [diff] [blame] | 53 | method enable_checkout {co} { |
| 54 | set checkout $co |
| 55 | } |
| 56 | |
| 57 | method enable_create {co} { |
| 58 | set create $co |
| 59 | } |
| 60 | |
| 61 | method run {} { |
| 62 | if {$fetch_spec ne {}} { |
| 63 | global M1B |
| 64 | |
| 65 | # We were asked to refresh a single tracking branch |
| 66 | # before we get to work. We should do that before we |
| 67 | # consider any ref updating. |
| 68 | # |
| 69 | set fetch_ok 0 |
| 70 | set l_trck [lindex $fetch_spec 0] |
| 71 | set remote [lindex $fetch_spec 1] |
| 72 | set r_head [lindex $fetch_spec 2] |
| 73 | regsub ^refs/heads/ $r_head {} r_name |
| 74 | |
Shawn O. Pearce | 54febd4 | 2007-07-20 04:10:13 -0400 | [diff] [blame] | 75 | set cmd [list git fetch $remote] |
| 76 | if {$l_trck ne {}} { |
| 77 | lappend cmd +$r_head:$l_trck |
| 78 | } else { |
| 79 | lappend cmd $r_head |
| 80 | } |
| 81 | |
Shawn O. Pearce | d41b43e | 2007-07-08 18:40:56 -0400 | [diff] [blame] | 82 | _toplevel $this {Refreshing Tracking Branch} |
| 83 | set w_cons [::console::embed \ |
| 84 | $w.console \ |
Christian Stimming | 1ac1795 | 2007-07-21 14:21:34 +0200 | [diff] [blame] | 85 | [mc "Fetching %s from %s" $r_name $remote]] |
Shawn O. Pearce | d41b43e | 2007-07-08 18:40:56 -0400 | [diff] [blame] | 86 | pack $w.console -fill both -expand 1 |
Shawn O. Pearce | 54febd4 | 2007-07-20 04:10:13 -0400 | [diff] [blame] | 87 | $w_cons exec $cmd [cb _finish_fetch] |
Shawn O. Pearce | d41b43e | 2007-07-08 18:40:56 -0400 | [diff] [blame] | 88 | |
| 89 | bind $w <$M1B-Key-w> break |
| 90 | bind $w <$M1B-Key-W> break |
| 91 | bind $w <Visibility> " |
| 92 | [list grab $w] |
| 93 | [list focus $w] |
| 94 | " |
| 95 | wm protocol $w WM_DELETE_WINDOW [cb _noop] |
| 96 | tkwait window $w |
| 97 | |
| 98 | if {!$fetch_ok} { |
| 99 | delete_this |
| 100 | return 0 |
| 101 | } |
| 102 | } |
| 103 | |
| 104 | if {$new_ref ne {}} { |
| 105 | # If we have a ref we need to update it before we can |
| 106 | # proceed with a checkout (if one was enabled). |
| 107 | # |
| 108 | if {![_update_ref $this]} { |
| 109 | delete_this |
| 110 | return 0 |
| 111 | } |
| 112 | } |
| 113 | |
| 114 | if {$checkout} { |
| 115 | _checkout $this |
| 116 | return 1 |
| 117 | } |
| 118 | |
| 119 | delete_this |
| 120 | return 1 |
| 121 | } |
| 122 | |
| 123 | method _noop {} {} |
| 124 | |
| 125 | method _finish_fetch {ok} { |
| 126 | if {$ok} { |
| 127 | set l_trck [lindex $fetch_spec 0] |
Shawn O. Pearce | 54febd4 | 2007-07-20 04:10:13 -0400 | [diff] [blame] | 128 | if {$l_trck eq {}} { |
| 129 | set l_trck FETCH_HEAD |
| 130 | } |
Shawn O. Pearce | d41b43e | 2007-07-08 18:40:56 -0400 | [diff] [blame] | 131 | if {[catch {set new_hash [git rev-parse --verify "$l_trck^0"]} err]} { |
| 132 | set ok 0 |
Michele Ballabio | c8c4854 | 2007-09-13 15:19:05 +0200 | [diff] [blame] | 133 | $w_cons insert [mc "fatal: Cannot resolve %s" $l_trck] |
Shawn O. Pearce | d41b43e | 2007-07-08 18:40:56 -0400 | [diff] [blame] | 134 | $w_cons insert $err |
| 135 | } |
| 136 | } |
| 137 | |
| 138 | $w_cons done $ok |
| 139 | set w_cons {} |
| 140 | wm protocol $w WM_DELETE_WINDOW {} |
| 141 | |
| 142 | if {$ok} { |
| 143 | destroy $w |
| 144 | set w {} |
| 145 | } else { |
Christian Stimming | 1ac1795 | 2007-07-21 14:21:34 +0200 | [diff] [blame] | 146 | button $w.close -text [mc Close] -command [list destroy $w] |
Shawn O. Pearce | d41b43e | 2007-07-08 18:40:56 -0400 | [diff] [blame] | 147 | pack $w.close -side bottom -anchor e -padx 10 -pady 10 |
| 148 | } |
| 149 | |
| 150 | set fetch_ok $ok |
| 151 | } |
| 152 | |
| 153 | method _update_ref {} { |
Shawn O. Pearce | fe70225 | 2008-05-08 20:16:43 -0400 | [diff] [blame] | 154 | global null_sha1 current_branch repo_config |
Shawn O. Pearce | d41b43e | 2007-07-08 18:40:56 -0400 | [diff] [blame] | 155 | |
| 156 | set ref $new_ref |
| 157 | set new $new_hash |
| 158 | |
| 159 | set is_current 0 |
| 160 | set rh refs/heads/ |
| 161 | set rn [string length $rh] |
| 162 | if {[string equal -length $rn $rh $ref]} { |
| 163 | set newbranch [string range $ref $rn end] |
| 164 | if {$current_branch eq $newbranch} { |
| 165 | set is_current 1 |
| 166 | } |
| 167 | } else { |
| 168 | set newbranch $ref |
| 169 | } |
| 170 | |
| 171 | if {[catch {set cur [git rev-parse --verify "$ref^0"]}]} { |
| 172 | # Assume it does not exist, and that is what the error was. |
| 173 | # |
| 174 | if {!$create} { |
Christian Stimming | 1ac1795 | 2007-07-21 14:21:34 +0200 | [diff] [blame] | 175 | _error $this [mc "Branch '%s' does not exist." $newbranch] |
Shawn O. Pearce | d41b43e | 2007-07-08 18:40:56 -0400 | [diff] [blame] | 176 | return 0 |
| 177 | } |
| 178 | |
| 179 | set reflog_msg "branch: Created from $new_expr" |
| 180 | set cur $null_sha1 |
Shawn O. Pearce | fe70225 | 2008-05-08 20:16:43 -0400 | [diff] [blame] | 181 | |
| 182 | if {($repo_config(branch.autosetupmerge) eq {true} |
| 183 | || $repo_config(branch.autosetupmerge) eq {always}) |
| 184 | && $remote_source ne {} |
| 185 | && "refs/heads/$newbranch" eq $ref} { |
| 186 | |
| 187 | set c_remote [lindex $remote_source 1] |
| 188 | set c_merge [lindex $remote_source 2] |
| 189 | if {[catch { |
| 190 | git config branch.$newbranch.remote $c_remote |
| 191 | git config branch.$newbranch.merge $c_merge |
| 192 | } err]} { |
| 193 | _error $this [strcat \ |
| 194 | [mc "Failed to configure simplified git-pull for '%s'." $newbranch] \ |
| 195 | "\n\n$err"] |
| 196 | } |
| 197 | } |
Shawn O. Pearce | d41b43e | 2007-07-08 18:40:56 -0400 | [diff] [blame] | 198 | } elseif {$create && $merge_type eq {none}} { |
| 199 | # We were told to create it, but not do a merge. |
| 200 | # Bad. Name shouldn't have existed. |
| 201 | # |
Christian Stimming | 1ac1795 | 2007-07-21 14:21:34 +0200 | [diff] [blame] | 202 | _error $this [mc "Branch '%s' already exists." $newbranch] |
Shawn O. Pearce | d41b43e | 2007-07-08 18:40:56 -0400 | [diff] [blame] | 203 | return 0 |
| 204 | } elseif {!$create && $merge_type eq {none}} { |
| 205 | # We aren't creating, it exists and we don't merge. |
| 206 | # We are probably just a simple branch switch. |
| 207 | # Use whatever value we just read. |
| 208 | # |
| 209 | set new $cur |
| 210 | set new_hash $cur |
| 211 | } elseif {$new eq $cur} { |
| 212 | # No merge would be required, don't compute anything. |
| 213 | # |
| 214 | } else { |
Shawn O. Pearce | 60f7352 | 2007-07-20 02:13:24 -0400 | [diff] [blame] | 215 | catch {set merge_base [git merge-base $new $cur]} |
Shawn O. Pearce | f66b8a6 | 2007-07-20 03:34:56 -0400 | [diff] [blame] | 216 | if {$merge_base eq $cur} { |
| 217 | # The current branch is older. |
| 218 | # |
| 219 | set reflog_msg "merge $new_expr: Fast-forward" |
| 220 | } else { |
| 221 | switch -- $merge_type { |
| 222 | ff { |
| 223 | if {$merge_base eq $new} { |
| 224 | # The current branch is actually newer. |
| 225 | # |
| 226 | set new $cur |
Shawn O. Pearce | 7bd197c | 2007-07-22 04:09:53 -0400 | [diff] [blame] | 227 | set new_hash $cur |
Shawn O. Pearce | f66b8a6 | 2007-07-20 03:34:56 -0400 | [diff] [blame] | 228 | } else { |
Christian Stimming | 1ac1795 | 2007-07-21 14:21:34 +0200 | [diff] [blame] | 229 | _error $this [mc "Branch '%s' already exists.\n\nIt cannot fast-forward to %s.\nA merge is required." $newbranch $new_expr] |
Shawn O. Pearce | f66b8a6 | 2007-07-20 03:34:56 -0400 | [diff] [blame] | 230 | return 0 |
| 231 | } |
Shawn O. Pearce | d41b43e | 2007-07-08 18:40:56 -0400 | [diff] [blame] | 232 | } |
Shawn O. Pearce | f66b8a6 | 2007-07-20 03:34:56 -0400 | [diff] [blame] | 233 | reset { |
Shawn O. Pearce | d41b43e | 2007-07-08 18:40:56 -0400 | [diff] [blame] | 234 | # The current branch will lose things. |
| 235 | # |
| 236 | if {[_confirm_reset $this $cur]} { |
| 237 | set reflog_msg "reset $new_expr" |
| 238 | } else { |
| 239 | return 0 |
| 240 | } |
| 241 | } |
Shawn O. Pearce | f66b8a6 | 2007-07-20 03:34:56 -0400 | [diff] [blame] | 242 | default { |
Christian Stimming | 1ac1795 | 2007-07-21 14:21:34 +0200 | [diff] [blame] | 243 | _error $this [mc "Merge strategy '%s' not supported." $merge_type] |
Shawn O. Pearce | f66b8a6 | 2007-07-20 03:34:56 -0400 | [diff] [blame] | 244 | return 0 |
| 245 | } |
| 246 | } |
Shawn O. Pearce | d41b43e | 2007-07-08 18:40:56 -0400 | [diff] [blame] | 247 | } |
| 248 | } |
| 249 | |
| 250 | if {$new ne $cur} { |
| 251 | if {$is_current} { |
| 252 | # No so fast. We should defer this in case |
| 253 | # we cannot update the working directory. |
| 254 | # |
| 255 | set update_old $cur |
| 256 | return 1 |
| 257 | } |
| 258 | |
| 259 | if {[catch { |
| 260 | git update-ref -m $reflog_msg $ref $new $cur |
| 261 | } err]} { |
Shawn O. Pearce | 31bb1d1 | 2007-09-14 01:50:09 -0400 | [diff] [blame] | 262 | _error $this [strcat [mc "Failed to update '%s'." $newbranch] "\n\n$err"] |
Shawn O. Pearce | d41b43e | 2007-07-08 18:40:56 -0400 | [diff] [blame] | 263 | return 0 |
| 264 | } |
| 265 | } |
| 266 | |
| 267 | return 1 |
| 268 | } |
| 269 | |
| 270 | method _checkout {} { |
| 271 | if {[lock_index checkout_op]} { |
| 272 | after idle [cb _start_checkout] |
| 273 | } else { |
Christian Stimming | 1ac1795 | 2007-07-21 14:21:34 +0200 | [diff] [blame] | 274 | _error $this [mc "Staging area (index) is already locked."] |
Shawn O. Pearce | d41b43e | 2007-07-08 18:40:56 -0400 | [diff] [blame] | 275 | delete_this |
| 276 | } |
| 277 | } |
| 278 | |
| 279 | method _start_checkout {} { |
| 280 | global HEAD commit_type |
| 281 | |
| 282 | # -- Our in memory state should match the repository. |
| 283 | # |
Jens Lehmann | c949833 | 2009-03-30 21:46:17 +0200 | [diff] [blame] | 284 | repository_state curType old_hash curMERGE_HEAD |
Shawn O. Pearce | d41b43e | 2007-07-08 18:40:56 -0400 | [diff] [blame] | 285 | if {[string match amend* $commit_type] |
| 286 | && $curType eq {normal} |
Jens Lehmann | c949833 | 2009-03-30 21:46:17 +0200 | [diff] [blame] | 287 | && $old_hash eq $HEAD} { |
| 288 | } elseif {$commit_type ne $curType || $HEAD ne $old_hash} { |
Christian Stimming | 1ac1795 | 2007-07-21 14:21:34 +0200 | [diff] [blame] | 289 | info_popup [mc "Last scanned state does not match repository state. |
Shawn O. Pearce | d41b43e | 2007-07-08 18:40:56 -0400 | [diff] [blame] | 290 | |
| 291 | Another Git program has modified this repository since the last scan. A rescan must be performed before the current branch can be changed. |
| 292 | |
| 293 | The rescan will be automatically started now. |
Christian Stimming | 1ac1795 | 2007-07-21 14:21:34 +0200 | [diff] [blame] | 294 | "] |
Shawn O. Pearce | d41b43e | 2007-07-08 18:40:56 -0400 | [diff] [blame] | 295 | unlock_index |
| 296 | rescan ui_ready |
| 297 | delete_this |
| 298 | return |
| 299 | } |
| 300 | |
Jens Lehmann | c949833 | 2009-03-30 21:46:17 +0200 | [diff] [blame] | 301 | if {$old_hash eq $new_hash} { |
Shawn O. Pearce | dba0741 | 2007-07-20 03:56:06 -0400 | [diff] [blame] | 302 | _after_readtree $this |
| 303 | } elseif {[is_config_true gui.trustmtime]} { |
Shawn O. Pearce | d41b43e | 2007-07-08 18:40:56 -0400 | [diff] [blame] | 304 | _readtree $this |
| 305 | } else { |
Christian Stimming | 5e6d776 | 2008-02-02 10:20:17 +0100 | [diff] [blame] | 306 | ui_status [mc "Refreshing file status..."] |
Shawn O. Pearce | 0b81261 | 2007-07-09 01:17:09 -0400 | [diff] [blame] | 307 | set fd [git_read update-index \ |
| 308 | -q \ |
| 309 | --unmerged \ |
| 310 | --ignore-missing \ |
| 311 | --refresh \ |
| 312 | ] |
Shawn O. Pearce | d41b43e | 2007-07-08 18:40:56 -0400 | [diff] [blame] | 313 | fconfigure $fd -blocking 0 -translation binary |
| 314 | fileevent $fd readable [cb _refresh_wait $fd] |
| 315 | } |
| 316 | } |
| 317 | |
| 318 | method _refresh_wait {fd} { |
| 319 | read $fd |
| 320 | if {[eof $fd]} { |
| 321 | close $fd |
| 322 | _readtree $this |
| 323 | } |
| 324 | } |
| 325 | |
| 326 | method _name {} { |
| 327 | if {$new_ref eq {}} { |
| 328 | return [string range $new_hash 0 7] |
| 329 | } |
| 330 | |
| 331 | set rh refs/heads/ |
| 332 | set rn [string length $rh] |
| 333 | if {[string equal -length $rn $rh $new_ref]} { |
| 334 | return [string range $new_ref $rn end] |
| 335 | } else { |
| 336 | return $new_ref |
| 337 | } |
| 338 | } |
| 339 | |
| 340 | method _readtree {} { |
| 341 | global HEAD |
| 342 | |
Shawn O. Pearce | b792230 | 2007-07-08 22:48:19 -0400 | [diff] [blame] | 343 | set readtree_d {} |
| 344 | $::main_status start \ |
Michele Ballabio | c8c4854 | 2007-09-13 15:19:05 +0200 | [diff] [blame] | 345 | [mc "Updating working directory to '%s'..." [_name $this]] \ |
Christian Stimming | 5e6d776 | 2008-02-02 10:20:17 +0100 | [diff] [blame] | 346 | [mc "files checked out"] |
Shawn O. Pearce | b792230 | 2007-07-08 22:48:19 -0400 | [diff] [blame] | 347 | |
Shawn O. Pearce | 0b81261 | 2007-07-09 01:17:09 -0400 | [diff] [blame] | 348 | set fd [git_read --stderr read-tree \ |
| 349 | -m \ |
| 350 | -u \ |
| 351 | -v \ |
| 352 | --exclude-per-directory=.gitignore \ |
| 353 | $HEAD \ |
| 354 | $new_hash \ |
| 355 | ] |
Shawn O. Pearce | d41b43e | 2007-07-08 18:40:56 -0400 | [diff] [blame] | 356 | fconfigure $fd -blocking 0 -translation binary |
| 357 | fileevent $fd readable [cb _readtree_wait $fd] |
| 358 | } |
| 359 | |
| 360 | method _readtree_wait {fd} { |
Shawn O. Pearce | b792230 | 2007-07-08 22:48:19 -0400 | [diff] [blame] | 361 | global current_branch |
Shawn O. Pearce | d41b43e | 2007-07-08 18:40:56 -0400 | [diff] [blame] | 362 | |
Shawn O. Pearce | b792230 | 2007-07-08 22:48:19 -0400 | [diff] [blame] | 363 | set buf [read $fd] |
| 364 | $::main_status update_meter $buf |
| 365 | append readtree_d $buf |
| 366 | |
Shawn O. Pearce | d41b43e | 2007-07-08 18:40:56 -0400 | [diff] [blame] | 367 | fconfigure $fd -blocking 1 |
| 368 | if {![eof $fd]} { |
| 369 | fconfigure $fd -blocking 0 |
| 370 | return |
| 371 | } |
| 372 | |
Shawn O. Pearce | b792230 | 2007-07-08 22:48:19 -0400 | [diff] [blame] | 373 | if {[catch {close $fd}]} { |
| 374 | set err $readtree_d |
Shawn O. Pearce | d41b43e | 2007-07-08 18:40:56 -0400 | [diff] [blame] | 375 | regsub {^fatal: } $err {} err |
Christian Stimming | 1ac1795 | 2007-07-21 14:21:34 +0200 | [diff] [blame] | 376 | $::main_status stop [mc "Aborted checkout of '%s' (file level merging is required)." [_name $this]] |
Shawn O. Pearce | 31bb1d1 | 2007-09-14 01:50:09 -0400 | [diff] [blame] | 377 | warn_popup [strcat [mc "File level merge required."] " |
Shawn O. Pearce | d41b43e | 2007-07-08 18:40:56 -0400 | [diff] [blame] | 378 | |
| 379 | $err |
| 380 | |
Christian Stimming | 1ac1795 | 2007-07-21 14:21:34 +0200 | [diff] [blame] | 381 | " [mc "Staying on branch '%s'." $current_branch]] |
Shawn O. Pearce | d41b43e | 2007-07-08 18:40:56 -0400 | [diff] [blame] | 382 | unlock_index |
| 383 | delete_this |
| 384 | return |
| 385 | } |
| 386 | |
Shawn O. Pearce | b792230 | 2007-07-08 22:48:19 -0400 | [diff] [blame] | 387 | $::main_status stop |
| 388 | _after_readtree $this |
| 389 | } |
| 390 | |
| 391 | method _after_readtree {} { |
| 392 | global selected_commit_type commit_type HEAD MERGE_HEAD PARENT |
| 393 | global current_branch is_detached |
| 394 | global ui_comm |
| 395 | |
| 396 | set name [_name $this] |
Shawn O. Pearce | d41b43e | 2007-07-08 18:40:56 -0400 | [diff] [blame] | 397 | set log "checkout: moving" |
| 398 | if {!$is_detached} { |
| 399 | append log " from $current_branch" |
| 400 | } |
| 401 | |
| 402 | # -- Move/create HEAD as a symbolic ref. Core git does not |
| 403 | # even check for failure here, it Just Works(tm). If it |
| 404 | # doesn't we are in some really ugly state that is difficult |
| 405 | # to recover from within git-gui. |
| 406 | # |
| 407 | set rh refs/heads/ |
| 408 | set rn [string length $rh] |
| 409 | if {[string equal -length $rn $rh $new_ref]} { |
| 410 | set new_branch [string range $new_ref $rn end] |
Shawn O. Pearce | 7d5266a | 2007-07-22 04:49:06 -0400 | [diff] [blame] | 411 | if {$is_detached || $current_branch ne $new_branch} { |
| 412 | append log " to $new_branch" |
| 413 | if {[catch { |
| 414 | git symbolic-ref -m $log HEAD $new_ref |
| 415 | } err]} { |
| 416 | _fatal $this $err |
| 417 | } |
| 418 | set current_branch $new_branch |
| 419 | set is_detached 0 |
Shawn O. Pearce | d41b43e | 2007-07-08 18:40:56 -0400 | [diff] [blame] | 420 | } |
Shawn O. Pearce | d41b43e | 2007-07-08 18:40:56 -0400 | [diff] [blame] | 421 | } else { |
Shawn O. Pearce | 881d8f2 | 2007-09-02 15:30:26 -0400 | [diff] [blame] | 422 | if {!$is_detached || $new_hash ne $HEAD} { |
Shawn O. Pearce | 7d5266a | 2007-07-22 04:49:06 -0400 | [diff] [blame] | 423 | append log " to $new_expr" |
| 424 | if {[catch { |
| 425 | _detach_HEAD $log $new_hash |
| 426 | } err]} { |
| 427 | _fatal $this $err |
| 428 | } |
Shawn O. Pearce | d41b43e | 2007-07-08 18:40:56 -0400 | [diff] [blame] | 429 | } |
| 430 | set current_branch HEAD |
| 431 | set is_detached 1 |
| 432 | } |
| 433 | |
| 434 | # -- We had to defer updating the branch itself until we |
| 435 | # knew the working directory would update. So now we |
| 436 | # need to finish that work. If it fails we're in big |
| 437 | # trouble. |
| 438 | # |
| 439 | if {$update_old ne {}} { |
| 440 | if {[catch { |
| 441 | git update-ref \ |
| 442 | -m $reflog_msg \ |
| 443 | $new_ref \ |
| 444 | $new_hash \ |
| 445 | $update_old |
| 446 | } err]} { |
| 447 | _fatal $this $err |
| 448 | } |
| 449 | } |
| 450 | |
| 451 | if {$is_detached} { |
Christian Stimming | 1ac1795 | 2007-07-21 14:21:34 +0200 | [diff] [blame] | 452 | info_popup [mc "You are no longer on a local branch. |
Shawn O. Pearce | d41b43e | 2007-07-08 18:40:56 -0400 | [diff] [blame] | 453 | |
Christian Stimming | 1ac1795 | 2007-07-21 14:21:34 +0200 | [diff] [blame] | 454 | If you wanted to be on a branch, create one now starting from 'This Detached Checkout'."] |
Shawn O. Pearce | d41b43e | 2007-07-08 18:40:56 -0400 | [diff] [blame] | 455 | } |
| 456 | |
Jens Lehmann | c949833 | 2009-03-30 21:46:17 +0200 | [diff] [blame] | 457 | # -- Run the post-checkout hook. |
| 458 | # |
| 459 | set fd_ph [githook_read post-checkout $old_hash $new_hash 1] |
| 460 | if {$fd_ph ne {}} { |
| 461 | global pch_error |
| 462 | set pch_error {} |
| 463 | fconfigure $fd_ph -blocking 0 -translation binary -eofchar {} |
| 464 | fileevent $fd_ph readable [cb _postcheckout_wait $fd_ph] |
| 465 | } else { |
| 466 | _update_repo_state $this |
| 467 | } |
| 468 | } |
| 469 | |
| 470 | method _postcheckout_wait {fd_ph} { |
| 471 | global pch_error |
| 472 | |
| 473 | append pch_error [read $fd_ph] |
| 474 | fconfigure $fd_ph -blocking 1 |
| 475 | if {[eof $fd_ph]} { |
| 476 | if {[catch {close $fd_ph}]} { |
| 477 | hook_failed_popup post-checkout $pch_error 0 |
| 478 | } |
| 479 | unset pch_error |
| 480 | _update_repo_state $this |
| 481 | return |
| 482 | } |
| 483 | fconfigure $fd_ph -blocking 0 |
| 484 | } |
| 485 | |
| 486 | method _update_repo_state {} { |
Shawn O. Pearce | d41b43e | 2007-07-08 18:40:56 -0400 | [diff] [blame] | 487 | # -- Update our repository state. If we were previously in |
| 488 | # amend mode we need to toss the current buffer and do a |
| 489 | # full rescan to update our file lists. If we weren't in |
| 490 | # amend mode our file lists are accurate and we can avoid |
| 491 | # the rescan. |
| 492 | # |
Jens Lehmann | c949833 | 2009-03-30 21:46:17 +0200 | [diff] [blame] | 493 | global selected_commit_type commit_type HEAD MERGE_HEAD PARENT |
| 494 | global ui_comm |
| 495 | |
Shawn O. Pearce | d41b43e | 2007-07-08 18:40:56 -0400 | [diff] [blame] | 496 | unlock_index |
Jens Lehmann | c949833 | 2009-03-30 21:46:17 +0200 | [diff] [blame] | 497 | set name [_name $this] |
Shawn O. Pearce | d41b43e | 2007-07-08 18:40:56 -0400 | [diff] [blame] | 498 | set selected_commit_type new |
| 499 | if {[string match amend* $commit_type]} { |
| 500 | $ui_comm delete 0.0 end |
| 501 | $ui_comm edit reset |
| 502 | $ui_comm edit modified false |
Michele Ballabio | c8c4854 | 2007-09-13 15:19:05 +0200 | [diff] [blame] | 503 | rescan [list ui_status [mc "Checked out '%s'." $name]] |
Shawn O. Pearce | d41b43e | 2007-07-08 18:40:56 -0400 | [diff] [blame] | 504 | } else { |
| 505 | repository_state commit_type HEAD MERGE_HEAD |
| 506 | set PARENT $HEAD |
Christian Stimming | 5e6d776 | 2008-02-02 10:20:17 +0100 | [diff] [blame] | 507 | ui_status [mc "Checked out '%s'." $name] |
Shawn O. Pearce | d41b43e | 2007-07-08 18:40:56 -0400 | [diff] [blame] | 508 | } |
| 509 | delete_this |
| 510 | } |
| 511 | |
| 512 | git-version proc _detach_HEAD {log new} { |
| 513 | >= 1.5.3 { |
| 514 | git update-ref --no-deref -m $log HEAD $new |
| 515 | } |
| 516 | default { |
| 517 | set p [gitdir HEAD] |
| 518 | file delete $p |
| 519 | set fd [open $p w] |
| 520 | fconfigure $fd -translation lf -encoding utf-8 |
| 521 | puts $fd $new |
| 522 | close $fd |
| 523 | } |
| 524 | } |
| 525 | |
| 526 | method _confirm_reset {cur} { |
| 527 | set reset_ok 0 |
| 528 | set name [_name $this] |
| 529 | set gitk [list do_gitk [list $cur ^$new_hash]] |
| 530 | |
| 531 | _toplevel $this {Confirm Branch Reset} |
| 532 | pack [label $w.msg1 \ |
| 533 | -anchor w \ |
| 534 | -justify left \ |
Christian Stimming | 1ac1795 | 2007-07-21 14:21:34 +0200 | [diff] [blame] | 535 | -text [mc "Resetting '%s' to '%s' will lose the following commits:" $name $new_expr]\ |
Shawn O. Pearce | d41b43e | 2007-07-08 18:40:56 -0400 | [diff] [blame] | 536 | ] -anchor w |
| 537 | |
| 538 | set list $w.list.l |
| 539 | frame $w.list |
| 540 | text $list \ |
| 541 | -font font_diff \ |
| 542 | -width 80 \ |
| 543 | -height 10 \ |
| 544 | -wrap none \ |
| 545 | -xscrollcommand [list $w.list.sbx set] \ |
| 546 | -yscrollcommand [list $w.list.sby set] |
| 547 | scrollbar $w.list.sbx -orient h -command [list $list xview] |
| 548 | scrollbar $w.list.sby -orient v -command [list $list yview] |
| 549 | pack $w.list.sbx -fill x -side bottom |
| 550 | pack $w.list.sby -fill y -side right |
| 551 | pack $list -fill both -expand 1 |
| 552 | pack $w.list -fill both -expand 1 -padx 5 -pady 5 |
| 553 | |
| 554 | pack [label $w.msg2 \ |
| 555 | -anchor w \ |
| 556 | -justify left \ |
Christian Stimming | 1ac1795 | 2007-07-21 14:21:34 +0200 | [diff] [blame] | 557 | -text [mc "Recovering lost commits may not be easy."] \ |
Shawn O. Pearce | d41b43e | 2007-07-08 18:40:56 -0400 | [diff] [blame] | 558 | ] |
| 559 | pack [label $w.msg3 \ |
| 560 | -anchor w \ |
| 561 | -justify left \ |
Christian Stimming | 1ac1795 | 2007-07-21 14:21:34 +0200 | [diff] [blame] | 562 | -text [mc "Reset '%s'?" $name] \ |
Shawn O. Pearce | d41b43e | 2007-07-08 18:40:56 -0400 | [diff] [blame] | 563 | ] |
| 564 | |
| 565 | frame $w.buttons |
| 566 | button $w.buttons.visualize \ |
Christian Stimming | 1ac1795 | 2007-07-21 14:21:34 +0200 | [diff] [blame] | 567 | -text [mc Visualize] \ |
Shawn O. Pearce | d41b43e | 2007-07-08 18:40:56 -0400 | [diff] [blame] | 568 | -command $gitk |
| 569 | pack $w.buttons.visualize -side left |
| 570 | button $w.buttons.reset \ |
Christian Stimming | 1ac1795 | 2007-07-21 14:21:34 +0200 | [diff] [blame] | 571 | -text [mc Reset] \ |
Shawn O. Pearce | d41b43e | 2007-07-08 18:40:56 -0400 | [diff] [blame] | 572 | -command " |
| 573 | set @reset_ok 1 |
| 574 | destroy $w |
| 575 | " |
| 576 | pack $w.buttons.reset -side right |
| 577 | button $w.buttons.cancel \ |
| 578 | -default active \ |
Christian Stimming | 1ac1795 | 2007-07-21 14:21:34 +0200 | [diff] [blame] | 579 | -text [mc Cancel] \ |
Shawn O. Pearce | d41b43e | 2007-07-08 18:40:56 -0400 | [diff] [blame] | 580 | -command [list destroy $w] |
| 581 | pack $w.buttons.cancel -side right -padx 5 |
| 582 | pack $w.buttons -side bottom -fill x -pady 10 -padx 10 |
| 583 | |
Shawn O. Pearce | 0b81261 | 2007-07-09 01:17:09 -0400 | [diff] [blame] | 584 | set fd [git_read rev-list --pretty=oneline $cur ^$new_hash] |
Shawn O. Pearce | d41b43e | 2007-07-08 18:40:56 -0400 | [diff] [blame] | 585 | while {[gets $fd line] > 0} { |
| 586 | set abbr [string range $line 0 7] |
| 587 | set subj [string range $line 41 end] |
| 588 | $list insert end "$abbr $subj\n" |
| 589 | } |
| 590 | close $fd |
| 591 | $list configure -state disabled |
| 592 | |
| 593 | bind $w <Key-v> $gitk |
| 594 | bind $w <Visibility> " |
| 595 | grab $w |
| 596 | focus $w.buttons.cancel |
| 597 | " |
| 598 | bind $w <Key-Return> [list destroy $w] |
| 599 | bind $w <Key-Escape> [list destroy $w] |
| 600 | tkwait window $w |
| 601 | return $reset_ok |
| 602 | } |
| 603 | |
| 604 | method _error {msg} { |
| 605 | if {[winfo ismapped $parent_w]} { |
| 606 | set p $parent_w |
| 607 | } else { |
| 608 | set p . |
| 609 | } |
| 610 | |
| 611 | tk_messageBox \ |
| 612 | -icon error \ |
| 613 | -type ok \ |
| 614 | -title [wm title $p] \ |
| 615 | -parent $p \ |
| 616 | -message $msg |
| 617 | } |
| 618 | |
| 619 | method _toplevel {title} { |
| 620 | regsub -all {::} $this {__} w |
| 621 | set w .$w |
| 622 | |
| 623 | if {[winfo ismapped $parent_w]} { |
| 624 | set p $parent_w |
| 625 | } else { |
| 626 | set p . |
| 627 | } |
| 628 | |
| 629 | toplevel $w |
| 630 | wm title $w $title |
| 631 | wm geometry $w "+[winfo rootx $p]+[winfo rooty $p]" |
| 632 | } |
| 633 | |
| 634 | method _fatal {err} { |
Shawn O. Pearce | 31bb1d1 | 2007-09-14 01:50:09 -0400 | [diff] [blame] | 635 | error_popup [strcat [mc "Failed to set current branch. |
Shawn O. Pearce | d41b43e | 2007-07-08 18:40:56 -0400 | [diff] [blame] | 636 | |
| 637 | This working directory is only partially switched. We successfully updated your files, but failed to update an internal Git file. |
| 638 | |
Christian Stimming | 1ac1795 | 2007-07-21 14:21:34 +0200 | [diff] [blame] | 639 | This should not have occurred. %s will now close and give up." [appname]] " |
Shawn O. Pearce | d41b43e | 2007-07-08 18:40:56 -0400 | [diff] [blame] | 640 | |
Christian Stimming | 1ac1795 | 2007-07-21 14:21:34 +0200 | [diff] [blame] | 641 | $err"] |
Shawn O. Pearce | d41b43e | 2007-07-08 18:40:56 -0400 | [diff] [blame] | 642 | exit 1 |
| 643 | } |
| 644 | |
| 645 | } |