trains3.tcl

Description

Richard Suchenwirth 2001-03-27 - This weekend fun project varies the theme of Model railroading with Tcl and takes a windshield perspective. Imagine you're standing at a railroad crossing, red lights are flashing... and then the train runs by - an armour yellow F7A, boxcars, gondola, trailer on flat car.. and finally, the caboose. That's what the following piece shows on a Tk canvas. You can control train speed with left (faster), middle (emergency stop), and right (slower, or back) mouse buttons.

WikiDbImage trains.jpg


Jeff Smith 2019-06-30 : Below is an online demo using CloudTk

Jeff Smith 2025-01-12 : This demo now runs with Tcl9.

Please Note : This demo has a run time of 2 minutes.


In order to cope with the higher data complexity, some more structure and a rr namespace were introduced. The API, so to speak, is simple:

rr::init   $canvas ;# creates and packs a canvas, if not existing
rr::create $type $number [$otherdata] ;# make a vehicle (loco or car)
rr::train  $number $consist ;# vehicles of which a train is made up
rr::run    $trainnumber ;# guess what that does ;-)

See the demo at end for concrete examples.

Changes

PYK 2012-12-09: eliminated update. speed affects after instead of move

Code

package require Tk

namespace eval rr {
    variable data
    set data(curx) 700
    set data(y) 190
    proc init w {
        variable data
        set data(c) $w
        set data(trainspeed) 10
        set data(autospeed)  1
        set data(timespeed) 40 
        if ![winfo exists $w] {
            scrollbar ${w}sc -ori hori -command [list $w xview]
            canvas $w -width 2000 -height 700 -bg lightblue \
            -xscrollcommand [list ${w}sc set]
            grid $w -sticky news
            grid ${w}sc -sticky news
            grid rowconf . 0 -weight 1
            grid columnconf . 0 -weight 1
        }
        $w delete all
        foreach i [after info] {after cancel $i}
        bind .c <Shift-1> [list source [info script]]
        bind .c <1> {incr [namespace current]::data(trainspeed)  1}
        bind .c <2> {set  [namespace current]::data(trainspeed)  0}
        bind .c <3> {incr [namespace current]::data(trainspeed) -1}
        bind .c <Motion> {wm title . [.c canvasx %x],[expr [.c canvasy %y]-190]}
        for {set i 0} {$i<200} {incr i} {
            set x0 [expr -10000+rand()*18000]
            set y0 [expr -50+rand()*50]
            set x1 [expr $x0+100+rand()*260]
            set y1 [expr $y0+15+rand()*25]
            $w create oval $x0 $y0 $x1 $y1 -fill white -outline white \
            -tag cloud
        }
        $w create text 0 5 -text "TclTrains - Richard Suchenwirth 2001"\
            -anchor nw -fill LightSteelBlue2
        $w create poly 1000 1000 1000 390 2000 360 2300 350 2500 300 2500 1000 -fill green4
        waterfall $w 1800 370 12 130
        bridge $w 820 175 1700
        $w create poly 7500 200 8100 50 8200 70 8333 60 8666 100 9000 200\
             -fill gray70
        birds $w 30
        $w create poly -10000 2000 -10000 147 -8500 82 -8100 41 -8000 66 -7920 33\
            -7800 77 -7700 30 -7000 160 42 67 99 130 155 73 199 102 255 83 312 126\
            380 116 433 105 501 75 600 104 1700 450 2100 620 2400 390 2500 180 \
            10000 170 10000 2000  -fill green3 -tag ground
        tower $w -7750 80
        $w create rect -7660 200 -7800 350 -fill tan -outline tan ;# signal base
        $w create poly -8100 1000 -7800 80 -7500 1000 -fill tan -outline tan
        $w create rect -10000 191 10000 200 -fill tan -outline tan;# ballast
        for {set y 1000} {$y>80} {set y [expr {$y*0.96}]} {
            set dx [expr {($y-70)/5}]
            $w create rect [expr {-7800-$dx}] $y [expr {-7800+$dx}] [expr $y*0.982] \
                -fill brown -outline {}
        }
        $w create poly -7920 1000 -7800 80 -7940 1000 -fill grey
        $w create poly -7660 1000 -7800 80 -7680 1000 -fill grey
        signal $w -7680 300
        $w create poly -1100 230 -900 10 -700 20 -500 -20 -300 10 -100 220 -110 270\
             -300 400 -500 278  -fill grey60 -tag {ground fg}
        station 3800 182 200 Springfield
        $w create poly 3200 1000 3600 220 4200 220 4600 1000 \
        -fill yellow4
        $w create rect 3500 185 9000 190 -fill tan ;# ballast
        $w create line 3500 184 9000 184 -width 3 -fill grey50 ;# siding
        $w create poly -500 1000 100 130 700 1000 -fill gray50 ;# road
        $w create poly 80 1000 100 130 120 1000 -outline yellow -fill gray50
        $w create line -10000 190 10000 190 -fill gray75 -width 3 ;# rail
        crossing 210 215
        for {set i 0} {$i<250} {incr i} {
            lappend trees [list [expr round(-9900+rand()*19800)]\
                 [expr round(50+rand()*350)]]
        }
        foreach tree [lsort -integer -index 1 $trees] {
            eval tree $tree
        }
     auto $w 135 800
        $w raise fg
        $w config -scrollregion [$w bbox all]
    }

    proc speed {speed} {
        variable data
        return [expr {round($data(timespeed) / $speed)}]
    }

    proc auto {c x y} {
        $c create rect [expr $x+5] $y [expr $x+40] [expr $y+30]\
          -fill black -tag auto
        $c create rect [expr $x+255] $y [expr $x+295] [expr $y+30]\
          -fill black -tag auto
        $c create poly $x $y $x [expr $y-105] [expr $x+15] [expr $y-200]\
             [expr $x+280] [expr $y-200] [expr $x+300] [expr $y-105]\
             [expr $x+300] $y \
          -fill [random:select {blue green yellow grey pink orange}]\
          -tag auto
        $c create poly [expr $x+15] [expr $y-100] [expr $x+30] [expr $y-180]\
             [expr $x+270] [expr $y-180] [expr $x+285] [expr $y-100]\
          -fill white -outline black -tag auto
        $c create oval [expr $x+5] [expr $y-25] [expr $x+35] [expr $y-70]\
          -fill red -tag auto
        $c create oval [expr $x+265] [expr $y-25] [expr $x+295] [expr $y-70]\
          -fill red -tag auto
        auto'animate $c $x $y auto
    }

    proc auto'animate {c x y tag} {
        foreach {x0 y0 x1 y1} [$c bbox $tag] break
        if {$y0<130} {
             $c delete $tag
             after [expr {int(100+rand()*50)}] [namespace current]::auto $c $x $y
        } else {
             variable data
             if {!$data(blink) || ($y1<190) || ($y0>230)} {
              $c scale $tag 100 120 0.95 0.95
             }
             after [speed $data(autospeed)] [list after idle [list [namespace current]::auto'animate $c $x $y $tag]]
        }
    }

    proc tower {c x y} {
        $c create rect $x $y [expr $x+60] [expr $y+90] -fill bisque
        $c create rect [expr $x-3] [expr $y-3] [expr $x+62] $y -fill gray
        window $c  [expr $x+5] [expr $y+5] 15 15 2
        door $c [expr $x+5] [expr $y+44] 17 45 15
    }

    proc waterfall {c x y w h} {
        $c create rect $x $y [expr $x+$w] [expr $y+$h] -fill lightblue \
            -outline lightblue
        waterfall'animate $c $x $y $w $h
    }

    proc waterfall'animate {c x y w h} {
        set wx [expr {round($x+rand()*$w)}]
        $c create line $wx $y $wx [expr {$y+12}] -fill white \
            -tag waterfall
        $c move waterfall 0 5
        catch {$c lower waterfall ground} ;# might initially fail
        foreach i [$c find withtag waterfall] {
            if {[lindex [$c bbox $i] 1]>$y+$h} {$c delete $i}
        }
        after [speed 1] [list after idle [list [namespace current]::waterfall'animate $c $x $y $w $h]]
    }

    proc bridge {c x y w} {
        $c create rect $x $y [expr $x+$w] [expr $y+40] -fill grey40 -tag fg
        for {set x0 [expr $x+300]} {$x0<$x+$w} {set x0 [expr $x0+250]} {
            set x1 [expr $x0+70]
            $c create line $x0 [expr $y+40] $x0 1000 -width 2
            $c create line $x1 [expr $y+40] $x1 1000 -width 2
            for {set y0 [expr $y+40]} {$y0<1000} {set y0 $y1} {
                set y1 [expr $y0+50]
                $c create line $x0 $y0 $x1 $y0 -width 2
                $c create line $x0 $y0 $x1 $y1
                $c create line $x0 $y1 $x1 $y0
            }
        }
    }

    proc signal {c x y} {
        variable data
        set tag {}; if {$y>$data(y)} {set tag fg}
        $c create rect $x $y [expr $x+5] [expr $y*0.5] -fill grey25\
            -tag $tag
        $c create rect [expr $x-15] [expr $y*0.62] \
            [expr $x+20] [expr $y*0.38] -fill black\
            -tag $tag
        $c create oval [expr $x-5] [expr $y*0.55-5]\
            [expr $x+10] [expr $y*0.55+10] -fill red -tag $tag
        $c create oval [expr $x-5] [expr $y*0.45-5]\
            [expr $x+10] [expr $y*0.45+10] -fill red -tag $tag
    }

    proc birds {w n} {
        for {set i 0} {$i<$n} {incr i} {
            set x0 [expr -10000+rand()*18000]
            set y0 [expr rand()*80]
            set coords [list $x0 $y0 [expr $x0+4] [expr $y0-1] \
                [expr $x0+6] [expr $y0+1] [expr $x0+8] [expr $y0-1]\
                [expr $x0+12] $y0]
            eval $w create line $coords -tag bird
        }
        birds'animate $w -1
    }

    proc birds'animate {c mode} {
        foreach i [$c find withtag bird] {
            $c move $i [expr {round(-2.5+rand()*4)}] [expr {round(rand()*2-1)}]
            set coords  [$c coords $i]
            foreach {x0 y0 x1 y1 x2 y2 x3 y3 x4 y4} $coords break
            set y0 [expr {$y0-2*$mode}]
            set y2 [expr {$y2+1*$mode}]
            set y4 [expr {$y4-2*$mode}]
            $c coords $i $x0 $y0 $x1 $y1 $x2 $y2 $x3 $y3 $x4 $y4
        }
        after 300 [list after idle [list [namespace current]::birds'animate $c [expr {$mode*-1}]]]
    }

    proc define {name def} {
        variable data
        set data($name) $def
    }

    proc create {type id args} {
        variable data
        set c $data(c)
        set tag $type:$id
        foreach i [split $data($type) \n] {
            set cmd [lindex $i 0]
            switch $cmd {
            bogie {
                set x [lindex $i 1]
                set diameter 21
                $c create oval $x -$diameter [expr $x+$diameter] 0\
                    -fill black -outline white -tag $tag
                set x1 [expr $x+[lindex $i 2]]
                $c create oval $x1 -$diameter [expr $x1+$diameter] 0\
                    -fill black -outline white -tag $tag
                set m [expr {$x+($x1+$diameter-$x)/2}]
                $c create rect [expr $m-10] -10 [expr $m+10] -30 -fill black -tag $tag
                $c create rect [expr $x-5] [expr -$diameter/2-5]\
                     [expr $x1+$diameter+5] [expr -$diameter/2+5] -fill gray20 -tag $tag
            }
            gp38body {
                set t [list gp38body $tag]
                $c create rect 0 -28 430 -21 -fill black -tag $t
                $c create rect 25 -28 70 -80 -fill red -tag $t    ;#nose
                $c create rect 70 -100 130 -28 -fill red -tag $t  ;#cab
                $c create rect 130 -100 410 -28 -fill red2 -tag $t ;#body
                $c create poly 180 -102 264 -102 264 -102 244 -90 200 -90\
                    180 -102 -fill red -outline black -tag $t
                $c create rect 54 -103 170 -93 -fill red -tag $t  ;#cab roof
                $c create line 290 -105 300 -105 -fill red -arrow both \
                    -arrowshape {-6 -5 3} -width 2 -tag $t ;# horns
                $c create rect 190 -22 255 -6 -fill black -tag $t ;#tank
                window $t 86 -90 26 18
                $c create line 100 -90 100 -72 -tag $tag ;# window separator
                $c create line 15 -8 15 -55 -fill white -width 2 -tag $tag
                $c create line 35 -8 35 -60 45 -60 57 -70 72 -70 -tag $t
                $c create line 128 -70 130 -70 140 -60 315 -60 320 -55 \
                    405 -55 405 -8 -tag "$t handrail"
                for {set x 150} {$x<400} {incr x 45} {
                    $c create line $x -60 $x -20 -tag "$t handrail"
                }
                $c create line 425 -8 425 -55 -fill white -width 2 -tag $tag
                for {set y -70} {$y<=-30} {incr y 10} {
                    $c create line 25 $y 30 [expr $y-8] -width 3 -fill white\
                        -tag $t
                }
                $c create text 100 -60 -text $id\
                     -font {Helvetica 11 {bold italic}} -fill white -tag $t
                $c create text 225 -75 -text "CP Rail" \
                     -font {Helvetica 24 {bold italic}} -fill white -tag $t
                $c create oval 350 -90 400 -40 -fill white -outline white -tag $t
                $c create rect 375 -90 400 -40 -fill white -outline white -tag $t
                $c create poly 375 -65 400 -40 400 -90 -fill black -tag $t
                $c raise handrail
            }
            f7abody {
                set t [list f7abody $tag]
                $c create rect 0 -25 430 -22 -fill black -tag $tag
                $c create rect 400 -30 430 -90 -fill black -tag $tag
                $c create poly \
                17 -9 30 -85 35 -88 58 -90 60 -92 67 -106 70 -108 73 -110 \
                    425 -110 425 -15 410 -15 400 -25 295 -25 290 -15 165 -15 \
                    160 -25 45 -25 35 -9 -fill gold -tag $t
                $c create rect 30 -71 51 -59 -fill black -tag $t
                $c create text 31 -71 -text $id -anchor nw -fill white -tag $t
                $c create poly 67 -102 72 -101 76 -97 70 -87 62 -92 \
                    -fill white -outline black -tag $t
                $c create poly 71 -81 80 -94 94 -94 94 -81 -fill white \
                    -outline black -tag $t
                $c create rect 98 -97 114 -52 -outline gold3 -tag $t
                $c create rect 101 -94 111 -81 -fill white -tag $t ;# cab door window
                $c create rect 118 -97 420 -80 -outline gold3 \
                     -tag $t ;# cooler grill
                for {set i 121} {$i<420} {incr i 3} {
                    $c create line $i -97 $i -80 -fill gold3 -tag $t
                }
                $c create rect 140 -110 424 -100 -fill gray75 \
                    -outline gray75 -tag $t;# roof
                $c create line 100 -113 110 -113 -arrow both \
                    -arrowshape {-5 -5 3} -width 2 -tag $t ;# horns
                $c create rect 103 -115 107 -110 -fill black -tag $t
                $c create oval 150 -77 165 -62 -fill gray50 -tag $t
                $c create oval 300 -77 315 -62 -fill gray50 -tag $t
                $c create text 145 -56 -text "U N I O N    P A C I F I C" -fill red \
                -font {Helvetica 13 bold} -anchor nw -tag $t
                $c create text 55 -56 -text $id -fill red -font {Helvetica 13 bold}\
                     -anchor nw -tag $t
                $c create line 55 -37 423 -37 -fill red -width 3 -tag $t
            }
            f7bbody {
                set t [list f7bbody $tag]
                $c create rect 0 -25 430 -22 -fill black -tag $tag
                $c create rect 0 -30 430 -90 -fill black -tag $tag
                $c create poly 17 -110 \
                    425 -110 425 -15 410 -15 400 -25 295 -25 290 -15 165 -15 \
                    160 -25 45 -25 35 -15 17 -15 -fill gold -tag $t
                $c create rect 22 -97 420 -80 -outline gold3 \
                     -tag $t ;# cooler grill
                for {set i 25} {$i<420} {incr i 3} {
                    $c create line $i -97 $i -80 -fill gold3 -tag $t
                }
                $c create rect 18 -110 424 -100 -fill gray75 \
                    -outline gray75 -tag $t;# roof
                $c create oval 140 -77 155 -62 -fill gray50 -tag $t
                $c create oval 220 -77 235 -62 -fill gray50 -tag $t
                $c create oval 300 -77 315 -62 -fill gray50 -tag $t
                $c create text 145 -56 -text "U N I O N    P A C I F I C" -fill red \
                -font {Helvetica 13 bold} -anchor nw -tag $t
                $c create text 55 -56 -text $id -fill red -font {Helvetica 13 bold}\
                     -anchor nw -tag $t
                $c create line 25 -37 423 -37 -fill red -width 3 -tag $t
            }
            boxcarbody {
                $c create rect 0 -25 380 -22 -fill black -tag $tag
                $c create rect 10 -26 370 -105 -fill [lindex $args 1] -tag $tag
                set rgrey grey[expr round(rand()*40+50)]
                $c create rect 10 -100 370 -105 -fill $rgrey -tag $tag
                $c create rect 160 -95 220 -30 -tag $tag
                $c create text 100 -70 -text [lindex $args 0] -fill white -tag $tag
                $c create text 100 -50 -text $id -fill white -tag $tag
            }
            baggagebody {
                $c create rect 0 -25 420 -22 -fill black -tag $tag
                $c create rect 0 -30 420 -92 -fill black -tag $tag
                $c create rect 5 -24 415 -100 -fill [lindex $args 1] -tag $tag
                set rgrey grey[expr round(rand()*20+70)]
                $c create rect 5 -90 415 -100 -fill $rgrey -tag $tag
                $c create line 8 -48 413 -48 -width 5 -fill red -tag $tag
                $c create line 8 -43 413 -43 -width 5 -fill white -tag $tag
                $c create line 8 -38 413 -38 -width 5 -fill blue -tag $tag
                $c create text 205 -43 -text [lindex $args 0]\
                     -fill black -font {Helvetica 8 bold} -tag $tag
                $c create text 71 -43 -text $id  -tag $tag
                door $tag 20 -75 20 50 15
                $c create rect 120 -80 160 -30 -tag $tag
                window $tag 125 -72 12 12 2 4
                $c create rect 260 -80 300 -30 -tag $tag
                window $tag 265 -70 12 12 2 4
            }
            coachbody {
                $c create rect 0 -25 420 -22 -fill black -tag $tag
                $c create rect 0 -30 420 -90 -fill black -tag $tag
                $c create rect 5 -23 415 -100 -fill [lindex $args 1] -tag $tag
                set rgrey grey[expr round(rand()*20+70)]
                $c create rect 5 -90 415 -100 -fill $rgrey -tag $tag
                for {set y -50} {$y<=-30} {incr y 3} {
                    $c create line 11 $y 410 $y -fill white -tag $tag
                }
                $c create text 205 -83 -text [join [split [lindex $args 0] ""] "       "]\
                     -fill red -font {Times 7 bold} -tag $tag
                $c create text 71 -44 -text $id  -tag $tag
                door $tag 10 -75 20 50 15
                window $tag 48 -72 36 16 7 8
                door $tag 390 -75 20 50 16
            }
            ocoachbody {
                $c create rect 0 -25 420 -22 -fill black -tag $tag
                $c create rect 20 -23 400 -90 -fill [lindex $args 1] -tag $tag
                set rgrey grey[expr round(rand()*20+40)]
                $c create rect 5 -90 415 -95 -fill $rgrey -tag $tag
                $c create poly 20 -95 40 -102 380 -102 400 -95 -fill grey\
                    -outline black -width 2 -tag $tag
                for {set x 50} {$x<=370} {incr x 20} {
                    $c create line $x -95 $x -102 -width 2 -tag $tag
                }
                $c create line 5 -5 5 -90 -tag $tag
                $c create line 415 -5 415 -90 -tag $tag
                $c create line 20 -88 400 -88 -fill yellow -tag $tag
                $c create text 205 -83 -text [join [split [lindex $args 0] ""] "       "]\
                     -fill yellow -font {Times 7 bold} -tag $tag
                $c create line 20 -78 400 -78 -fill yellow -tag $tag
                $c create text 40 -40 -text $id -fill yellow -tag $tag
                $c create text 380 -40 -text $id -fill yellow -tag $tag
                window $tag 50 -72 15 22 16 5
                $c create line 50 -62 350 -62 -tag $tag
                $c create line 20 -27 400 -27 -fill yellow -tag $tag
            }
            domebody {
                $c create rect 0 -25 420 -22 -fill black -tag $tag
                $c create rect 0 -30 420 -90 -fill black -tag $tag
                $c create rect 5 -24 415 -100 -fill [lindex $args 1] -tag $tag
                set rgrey grey[expr round(rand()*20+70)]
                $c create rect 5 -90 415 -100 -fill $rgrey -tag $tag
                $c create poly 110 -92 100 -100 130 -120 300 -120 330 -100 320 -92 \
                    -fill lightcyan -outline black -tag $tag ;# dome
                for {set x 130} {$x<=300} {incr x 34} {
                    $c create line $x -120 $x -90 -tag $tag
                }
                $c create line 123 -115 130 -110 300 -110 308 -118 -tag $tag
                $c create text 210 -83 -text [join [split [lindex $args 0] ""] "       "]\
                     -fill red -font {Times 7 bold} -tag $tag
                for {set y -50} {$y<=-30} {incr y 3} {
                    $c create line 11 $y 410 $y -fill white -tag $tag
                }
                $c create text 71 -44 -text $id  -tag $tag
                door $tag 20 -75 20 50 15
                window $tag 110 -60 25 15 6 10
                door $tag 380 -75 20 50 16
            }
            caboosebody {
                $c create rect 0 -24 300 -21 -fill black -tag $tag
                $c create poly 35 -23 35 -100 120 -100 120 -130 190 -130\
                     190 -100 270 -100 270 -23\
                     -fill [lindex $args 1] -tag $tag
                $c create line 10 -10 10 -116 11 -117 12 -119 15 -118 22 -105 -tag $tag
                $c create line 290 -10 290 -116 289 -117 288 -119 284 -118 277 -105 -tag $tag
                set rgrey grey[expr round(rand()*40+10)]
                $c create rect 10 -100 120 -105 -fill $rgrey -tag $tag
                $c create rect 115 -125 195 -130 -fill $rgrey -tag $tag
                $c create rect 190 -100 290 -105 -fill $rgrey -tag $tag
                $c create rect 210 -105 215 -130 -fill black -tag $tag
                window $tag 130 -120 18 15 2 15
                window $tag  50 -75 19 17 2 15
                window $tag 200 -75 19 17 2 15
                $c create text 150 -90 -text [lindex $args 0] -fill white -tag $tag
                $c create text 150 -50 -text $id -fill white -tag $tag
                $c create arc 40 -30 85 -85 -style arc -start 180 \
                    -extent 90 -outline yellow -width 1 -tag $tag
                $c create arc 220 -30 265 -85 -style arc -start 270 \
                    -extent 90 -outline yellow -width 1 -tag $tag
            }
            flatcarbody {
                $c create rect 0 -25 380 -22 -fill black -tag $tag
                $c create rect 10 -26 370 -35 -fill [lindex $args 1] -tag $tag
                $c create text 80 -29 -text [lindex $args 0] -fill white -tag $tag
                $c create text 220 -29 -text $id -fill white -tag $tag
            }
            gondolabody {
                $c create rect 0 -25 380 -22 -fill black -tag $tag
                $c create rect 10 -26 370 -90 -fill [lindex $args 1] -tag $tag
                $c create text 100 -70 -text [lindex $args 0] -fill white -tag $tag
                $c create text 100 -50 -text $id -fill white -tag $tag
            }
            hopperbody {
                $c create rect 0 -25 380 -22 -fill black -tag $tag
                $c create poly 10 -100 10 -50 120 -10 130 -30 180 -10 190 -30\
                    200 -10 250 -30 260 -10 370 -50 370 -100 -fill [lindex $args 1] -tag $tag
                $c create rect 10 -26 370 -100 -width 2 -tag $tag
                for {set x 52} {$x<360} {incr x 93} {
                    $c create line $x -26 $x -100 -width 2 -tag $tag
                }
                $c create text 90 -50 -text "[lindex $args 0] $id" -fill white -tag $tag
                $c create text 190 -70 -text [lindex $args 0] -font {Times 24 bold}\
                    -fill white -tag $tag
           }
            trailer {
                set color [lindex $i 1]
                $c create rect 40 -110 340 -50 -fill $color -tag $tag
                $c create text 190 -80 -text "ROADWAY" \
                    -font {Helvetica 40} -fill green4 -tag $tag
                $c create line 80 -50 80 -35 -width 3 -tag $tag
                $c create oval 240 -50 260 -30 -fill gray50 -tag $tag
                $c create oval 280 -50 300 -30 -fill gray50 -tag $tag
                $c create oval 245 -45 255 -35 -fill $color -tag $tag
                $c create oval 285 -45 295 -35 -fill $color -tag $tag
            }
            "" continue
            default {error "bad definition word $cmd:\n$i"}
            }
        }
    }

    proc train {name rstock} {
        variable data
        set c $data(c)
        set newx 0
        foreach i $rstock {
                $c move $i $data(curx) $data(y)
                set data(curx) [lindex [$c bbox $i] 2]
                $c addtag $name withtag $i
        }
        incr data(curx) 10000
        $c raise $name
    }

    proc crossing {x y} {
        variable data
        set c $data(c)
        $c create line [expr $x-10] [expr $y-40] [expr $x+15] [expr $y-40]\
            -width 3 -tag fg
        $c create rect $x $y [expr $x+5] [expr $y-70] -fill orange -tag fg
        $c create line [expr $x-15] [expr $y-80] [expr $x+20] [expr $y-60]\
             -width 5 -fill white -tag fg
        $c create line [expr $x-15] [expr $y-60] [expr $x+20] [expr $y-80]\
             -width 5 -fill white -tag fg
        $c create oval [expr $x-8] [expr $y-45] [expr $x-18] [expr $y-35]\
            -fill white -tag fg
        $c create oval [expr $x-10] [expr $y-43] [expr $x-16] [expr $y-37]\
            -fill black -tag {fg blink0}
        $c create oval [expr $x+15] [expr $y-45] [expr $x+25] [expr $y-35]\
            -fill white -tag fg
        $c create oval [expr $x+17] [expr $y-43] [expr $x+23] [expr $y-37]\
            -fill black -tag {fg blink1}
        set data(blink) 1
        flashCrossing 0
    }

    proc flashCrossing {which} {
        variable data
        set c $data(c)
        if $data(blink) {$c itemconfig blink$which -fill red -outline red}
        set which [expr 1-$which]
        $c itemconfig blink$which -fill black -outline black
        after 250 [list after idle [list [namespace current]::flashCrossing $which]]
    }

    proc window {t x y w h {n 1} {space 10}} {
        variable data
        set c $data(c)
        for {set i 0} {$i<$n} {incr i} {
            $c create rect $x $y [expr $x+$w] [expr $y+$h] -fill black -tag $t
            $c create rect [expr $x+3] [expr $y+3] [expr $x+$w] [expr $y+$h]\
                 -fill white -tag $t
            set x [expr $x+$w+$space]
        }
    }

    proc door {t x y w h winh} {
        variable data
        set c $data(c)
        $c create rect $x $y [expr $x+$w] [expr $y+$h] -tag $t
        incr w -8
        incr x 4
        incr y 4
        window $t $x $y $w $winh

    }

    proc run {trains} {
        variable data
        set c $data(c)
        foreach i [$c find withtag cloud] {
            $c move $i [expr {rand()/10}] 0
        }
        foreach train $trains {
            $c move $train -5 0
        }
        after idle [list [namespace current]::run2 $trains]
    }

    proc run2 {trains} {
        variable data; set c $data(c)
        foreach train $trains {
            foreach {x0 y0 x1 y1} [$c bbox $train] break
            if {$x1<-10000} {
                $c move $train 25000 0
            } elseif {$x1<0 && $x1>-1000} {
                set data(blink) 0
            } elseif {$x0<1500 && $x1>-1000} {
                set data(blink) 1
            }
        }
        after [speed $data(trainspeed)] [list after idle [list [namespace current]::run $trains]]
        $c raise fg
    }

    proc tree {x y} {
        set color [random:select {DarkGreen ForestGreen}]
        set stemcolor [random:select {brown salmon4 chocolate4 burlywood4}]
        variable data; set c $data(c)
            set where [lindex [$c find overlapping $x $y $x $y] end]
        if {[lsearch [$c itemcget $where -tags] "ground"]>=0} {
            set tag {}
            if {$y>$data(y)} {set tag fg}
            set dx [expr 1+($y*$y/12000.)]
            set x0 [expr $x-$dx*5]
            set x1 [expr $x+$dx+$dx*5]
            set xm [expr $x+$dx/2.]
            set y1 [expr $y-10-$dx*20*rand()]
            set y2 [expr $y1-6*$dx*(2+2*rand())]
            $c create rect $x $y [expr $x+$dx] $y1 -fill $stemcolor -tag $tag
            $c create poly $x0 $y1 $xm $y2 $x1 $y1 -fill $color \
                -tag $tag -outline black
        }
    }

    proc station {x y w name} {
        variable data; set c $data(c)
        $c create rect $x $y [expr $x+$w] [expr $y-100] -fill PeachPuff1
        window $c [expr $x+20] [expr $y-95] 18 22 4 30
        $c create rect [expr $x-5] [expr $y-100] [expr $x+$w+5] [expr $y-160]\
            -fill gray40
        door $c [expr $x+20] [expr $y-58] 20 50 22
        window $c [expr $x+65] [expr $y-55] 18 22 3 30
        set it [$c create text [expr $x+$w/2.] [expr $y-68] \
            -text [string toupper "   $name   "]]
        eval $c create rect [$c bbox $it] -fill white
        $c raise $it
        $c create rect [expr $x-50] [expr $y-60] \
            [expr $x+$w+100] [expr $y-55] -fill PeachPuff3
        for {set i [expr $x-45]} {$i<$x+$w+100} {incr i 60} {
            $c create rect $i [expr $y-55] \
                [expr $i+3] $y -fill PeachPuff3
        }
        $c create rect [expr $x-200] $y [expr $x+$w+500] [expr $y-10]\
            -fill gray30
    }

    proc flip what {
        variable data; set c $data(c)
        foreach {x0 y0 x1 y1} [$c bbox $what] break
        $c scale $what [expr ($x1-$x0)/2.] [expr ($y0-$y0)/2.] -1 1
    }

    define F7A {
        bogie 55 60
        bogie 305 60
        f7abody
    }
    define F7B {
        bogie 55 60
        bogie 305 60
        f7bbody
    }
    define GP38 {
        bogie 60 50
        bogie 310 50
        gp38body
    }
    define boxcar {
        bogie 40 40
        bogie 280 40
        boxcarbody
    }
    define coach {
        bogie 50 40
        bogie 310 40
        coachbody
    }
    define ocoach {
        bogie 50 40
        bogie 310 40
        ocoachbody
    }
    define domecar {
        bogie 50 40
        bogie 310 40
        domebody
    }
    define baggage {
        bogie 50 40
        bogie 310 40
        baggagebody
    }
    define gondola {
        bogie 40 40
        bogie 280 40
        gondolabody
    }
    define hopper {
        bogie 30 40
        bogie 290 40
        hopperbody
    }
    define flatcar {
        bogie 40 40
        bogie 280 40
        trailer gray85
        flatcarbody
    }
    define caboose {
        bogie 40 40
        bogie 190 40
        caboosebody
    }
}

proc random:select {L} {
    lindex $L [expr {int(rand()*[llength $L])}]
}

namespace eval rr {
    # Usage examples, and demo:
    init .c
    create F7A I50I
    create F7B I308
    create GP38 3018
    create GP38 3022
    flip GP38:3022
    create hopper 12988 "N & W" grey30
    create hopper 5603 "UNION PACIFIC" brown
    create boxcar 42135 ATSF brown
    create boxcar 42199 C&NW orange
    create gondola 745219 N.Y.C. salmon4
    create baggage 93152 "Amtrak" grey80
    create coach 4312 "UNION PACIFIC" grey90
    create domecar 7001 "UNION PACIFIC" grey95
    create coach 4319 "UNION PACIFIC" grey90
    create ocoach 4711 "BALTIMORE & OHIO" darkgreen
    create ocoach 5006 "PENNSYLVANIA" firebrick
    create caboose 18832 "C A N A D I A N   P A C I F I C" red2
    create flatcar 88402 "BOSTON & MAINE" black

    for {set i 43127;set parked {}} {$i<43256} {incr i 13} {
        create boxcar $i N.Y.C. \
            [random:select {brown salmon4 firebrick burlywood4}]
        lappend parked boxcar:$i
    }
    train parked $parked
    .c move parked 3700 -5
    train UP1 {
         F7A:I50I F7B:I308 baggage:93152 coach:4319 domecar:7001 coach:4312
         ocoach:4711 ocoach:5006
    }
    train CP123 {
        GP38:3018 GP38:3022 hopper:12988 hopper:5603
        boxcar:42135 flatcar:88402 gondola:745219 boxcar:42199
        caboose:18832
    }
    run {CP123 UP1}
    .c move UP1 -14000 0 ;# quick start
    .c move CP123 -14000 0 ;# quick start
}

Mike Tuxford This is really awesome Richard!! If it didn't use up the cpu cycles so much I'd use it as a screen saver.

haha - great stuff. stevenaaus