mirror of
https://github.com/git/git.git
synced 2024-11-05 08:47:56 +01:00
28bf928cf8
If a variable reference to a field is to an array, and it is the only reference to that field in that method we cannot make it an inlined [set foo] call as the regexp was converting the Tcl code wrong. We were producing "[set foo](x)" for "$foo(x)", and that isn't valid Tcl when foo is an array. So we just punt if the only occurance has a ( after it. Signed-off-by: Shawn O. Pearce <spearce@spearce.org>
154 lines
3.5 KiB
Tcl
154 lines
3.5 KiB
Tcl
# git-gui simple class/object fake-alike
|
|
# Copyright (C) 2007 Shawn Pearce
|
|
|
|
proc class {class body} {
|
|
if {[namespace exists $class]} {
|
|
error "class $class already declared"
|
|
}
|
|
namespace eval $class {
|
|
variable __nextid 0
|
|
variable __sealed 0
|
|
variable __field_list {}
|
|
variable __field_array
|
|
|
|
proc cb {name args} {
|
|
upvar this this
|
|
set args [linsert $args 0 $name $this]
|
|
return [uplevel [list namespace code $args]]
|
|
}
|
|
}
|
|
namespace eval $class $body
|
|
}
|
|
|
|
proc field {name args} {
|
|
set class [uplevel {namespace current}]
|
|
variable ${class}::__sealed
|
|
variable ${class}::__field_array
|
|
|
|
switch [llength $args] {
|
|
0 { set new [list $name] }
|
|
1 { set new [list $name [lindex $args 0]] }
|
|
default { error "wrong # args: field name value?" }
|
|
}
|
|
|
|
if {$__sealed} {
|
|
error "class $class is sealed (cannot add new fields)"
|
|
}
|
|
|
|
if {[catch {set old $__field_array($name)}]} {
|
|
variable ${class}::__field_list
|
|
lappend __field_list $new
|
|
set __field_array($name) 1
|
|
} else {
|
|
error "field $name already declared"
|
|
}
|
|
}
|
|
|
|
proc constructor {name params body} {
|
|
set class [uplevel {namespace current}]
|
|
set ${class}::__sealed 1
|
|
variable ${class}::__field_list
|
|
set mbodyc {}
|
|
|
|
append mbodyc {set this } $class
|
|
append mbodyc {::__o[incr } $class {::__nextid]} \;
|
|
append mbodyc {namespace eval $this {}} \;
|
|
|
|
if {$__field_list ne {}} {
|
|
append mbodyc {upvar #0}
|
|
foreach n $__field_list {
|
|
set n [lindex $n 0]
|
|
append mbodyc { ${this}::} $n { } $n
|
|
regsub -all @$n\\M $body "\${this}::$n" body
|
|
}
|
|
append mbodyc \;
|
|
foreach n $__field_list {
|
|
if {[llength $n] == 2} {
|
|
append mbodyc \
|
|
{set } [lindex $n 0] { } [list [lindex $n 1]] \;
|
|
}
|
|
}
|
|
}
|
|
append mbodyc $body
|
|
namespace eval $class [list proc $name $params $mbodyc]
|
|
}
|
|
|
|
proc method {name params body {deleted {}} {del_body {}}} {
|
|
set class [uplevel {namespace current}]
|
|
set ${class}::__sealed 1
|
|
variable ${class}::__field_list
|
|
set params [linsert $params 0 this]
|
|
set mbodyc {}
|
|
|
|
switch $deleted {
|
|
{} {}
|
|
ifdeleted {
|
|
append mbodyc {if {![namespace exists $this]} }
|
|
append mbodyc \{ $del_body \; return \} \;
|
|
}
|
|
default {
|
|
error "wrong # args: method name args body (ifdeleted body)?"
|
|
}
|
|
}
|
|
|
|
set decl {}
|
|
foreach n $__field_list {
|
|
set n [lindex $n 0]
|
|
if {[regexp -- $n\\M $body]} {
|
|
if { [regexp -all -- $n\\M $body] == 1
|
|
&& [regexp -all -- \\\$$n\\M $body] == 1
|
|
&& [regexp -all -- \\\$$n\\( $body] == 0} {
|
|
regsub -all \\\$$n\\M $body "\[set \${this}::$n\]" body
|
|
} else {
|
|
append decl { ${this}::} $n { } $n
|
|
regsub -all @$n\\M $body "\${this}::$n" body
|
|
}
|
|
}
|
|
}
|
|
if {$decl ne {}} {
|
|
append mbodyc {upvar #0} $decl \;
|
|
}
|
|
append mbodyc $body
|
|
namespace eval $class [list proc $name $params $mbodyc]
|
|
}
|
|
|
|
proc delete_this {{t {}}} {
|
|
if {$t eq {}} {
|
|
upvar this this
|
|
set t $this
|
|
}
|
|
if {[namespace exists $t]} {namespace delete $t}
|
|
}
|
|
|
|
proc make_toplevel {t w} {
|
|
upvar $t top $w pfx
|
|
if {[winfo ismapped .]} {
|
|
upvar this this
|
|
regsub -all {::} $this {__} w
|
|
set top .$w
|
|
set pfx $top
|
|
toplevel $top
|
|
} else {
|
|
set top .
|
|
set pfx {}
|
|
}
|
|
}
|
|
|
|
|
|
## auto_mkindex support for class/constructor/method
|
|
##
|
|
auto_mkindex_parser::command class {name body} {
|
|
variable parser
|
|
variable contextStack
|
|
set contextStack [linsert $contextStack 0 $name]
|
|
$parser eval [list _%@namespace eval $name] $body
|
|
set contextStack [lrange $contextStack 1 end]
|
|
}
|
|
auto_mkindex_parser::command constructor {name args} {
|
|
variable index
|
|
variable scriptFile
|
|
append index [list set auto_index([fullname $name])] \
|
|
[format { [list source [file join $dir %s]]} \
|
|
[file split $scriptFile]] "\n"
|
|
}
|
|
|