# OO support for Jim Tcl, with multiple inheritance # Create a new class $classname, with the given # dictionary as class variables. These are the initial # variables which all newly created objects of this class are # initialised with. # # If a list of baseclasses is given, # methods and instance variables are inherited. # The *last* baseclass can be accessed directly with [super] # Later baseclasses take precedence if the same method exists in more than one proc class {classname {baseclasses {}} classvars} { foreach baseclass $baseclasses { # Start by mapping all methods to the parent class foreach method [$baseclass methods] { alias "$classname $method" "$baseclass $method" } # Now import the base class classvars set classvars [dict merge $classvars [$baseclass classvars]] # The last baseclass will win here proc "$classname baseclass" {} baseclass { return $baseclass } } # Make sure that classvars is a dictionary set vars [lsort [dict keys $classvars]] # This is the class dispatcher for $classname # It simply dispatches 'classname cmd' to a procedure named {classname cmd} # with a nice message if the class procedure doesn't exist proc $classname {{cmd new} args} classname { if {![exists -proc "$classname $cmd"]} { return -code error "$classname, unknown command \"$cmd\": should be [join [$classname methods] ", "]" } tailcall "$classname $cmd" {*}$args } # Constructor proc "$classname new" {{instvars {}}} {classname classvars} { set instvars [dict merge $classvars $instvars] # This is the object dispatcher for $classname. # Store the classname in both the ref value and tag, for debugging # ref tag (for debugging) proc [ref $classname $classname "$classname finalize"] {method args} {classname instvars} { if {![exists -proc "$classname $method"]} { return -code error "$classname, unknown method \"$method\": should be [join [$classname methods] ", "]" } "$classname $method" {*}$args } } # Finalizer to invoke destructor during garbage collection proc "$classname finalize" {ref classname} { $ref destroy } # Method creator proc "$classname method" {method arglist body} classname { proc "$classname $method" $arglist {body} { # Make sure this isn't incorrectly called without an object if {![uplevel exists instvars]} { return -code error -level 2 "\"[lindex [info level 0] 0]\" method called with no object" } set self [lindex [info level -1] 0] # Note that we can't use 'dict with' here because # the dict isn't updated until the body completes. foreach _ [$self vars] {upvar 1 instvars($_) $_} unset _ eval $body } } # Other simple class procs proc "$classname vars" {} vars { return $vars } proc "$classname classvars" {} classvars { return $classvars } proc "$classname classname" {} classname { return $classname } proc "$classname methods" {} classname { lsort [lmap p [info procs "$classname *"] { lindex [split $p " "] 1 }] } # Pre-defined some instance methods $classname method destroy {} { rename $self "" } $classname method get {var} { set $var } $classname method eval {{locals {}} code} { foreach var $locals { upvar 2 $var $var } eval $code } return $classname } # From within a method, invokes the given method on the base class. # Note that this will only call the last baseclass given proc super {method args} { upvar self self uplevel 2 [$self baseclass] $method {*}$args }