Spaces:
Running
Running
# Simple Chord for Tcl | |
# | |
# A "chord" is a method with more than one entrypoint and only one body, such | |
# that the body runs only once all the entrypoints have been called by | |
# different asynchronous tasks. In this implementation, the chord is defined | |
# dynamically for each invocation. A SimpleChord object is created, supplying | |
# body script to be run when the chord is completed, and then one or more notes | |
# are added to the chord. Each note can be called like a proc, and returns | |
# immediately if the chord isn't yet complete. When the last remaining note is | |
# called, the body runs before the note returns. | |
# | |
# The SimpleChord class has a constructor that takes the body script, and a | |
# method add_note that returns a note object. Since the body script does not | |
# run in the context of the procedure that defined it, a mechanism is provided | |
# for injecting variables into the chord for use by the body script. The | |
# activation of a note is idempotent; multiple calls have the same effect as | |
# a simple call. | |
# | |
# If you are invoking asynchronous operations with chord notes as completion | |
# callbacks, and there is a possibility that earlier operations could complete | |
# before later ones are started, it is a good practice to create a "common" | |
# note on the chord that prevents it from being complete until you're certain | |
# you've added all the notes you need. | |
# | |
# Example: | |
# | |
# # Turn off the UI while running a couple of async operations. | |
# lock_ui | |
# | |
# set chord [SimpleChord::new { | |
# unlock_ui | |
# # Note: $notice here is not referenced in the calling scope | |
# if {$notice} { info_popup $notice } | |
# } | |
# | |
# # Configure a note to keep the chord from completing until | |
# # all operations have been initiated. | |
# set common_note [$chord add_note] | |
# | |
# # Activate notes in 'after' callbacks to other operations | |
# set newnote [$chord add_note] | |
# async_operation $args [list $newnote activate] | |
# | |
# # Communicate with the chord body | |
# if {$condition} { | |
# # This sets $notice in the same context that the chord body runs in. | |
# $chord eval { set notice "Something interesting" } | |
# } | |
# | |
# # Activate the common note, making the chord eligible to complete | |
# $common_note activate | |
# | |
# At this point, the chord will complete at some unknown point in the future. | |
# The common note might have been the first note activated, or the async | |
# operations might have completed synchronously and the common note is the | |
# last one, completing the chord before this code finishes, or anything in | |
# between. The purpose of the chord is to not have to worry about the order. | |
# SimpleChord class: | |
# Represents a procedure that conceptually has multiple entrypoints that must | |
# all be called before the procedure executes. Each entrypoint is called a | |
# "note". The chord is only "completed" when all the notes are "activated". | |
class SimpleChord { | |
field notes | |
field body | |
field is_completed | |
field eval_ns | |
# Constructor: | |
# set chord [SimpleChord::new {body}] | |
# Creates a new chord object with the specified body script. The | |
# body script is evaluated at most once, when a note is activated | |
# and the chord has no other non-activated notes. | |
constructor new {i_body} { | |
set notes [list] | |
set body $i_body | |
set is_completed 0 | |
set eval_ns "[namespace qualifiers $this]::eval" | |
return $this | |
} | |
# Method: | |
# $chord eval {script} | |
# Runs the specified script in the same context (namespace) in which | |
# the chord body will be evaluated. This can be used to set variable | |
# values for the chord body to use. | |
method eval {script} { | |
namespace eval $eval_ns $script | |
} | |
# Method: | |
# set note [$chord add_note] | |
# Adds a new note to the chord, an instance of ChordNote. Raises an | |
# error if the chord is already completed, otherwise the chord is | |
# updated so that the new note must also be activated before the | |
# body is evaluated. | |
method add_note {} { | |
if {$is_completed} { error "Cannot add a note to a completed chord" } | |
set note [ChordNote::new $this] | |
lappend notes $note | |
return $note | |
} | |
# This method is for internal use only and is intentionally undocumented. | |
method notify_note_activation {} { | |
if {!$is_completed} { | |
foreach note $notes { | |
if {![$note is_activated]} { return } | |
} | |
set is_completed 1 | |
namespace eval $eval_ns $body | |
delete_this | |
} | |
} | |
} | |
# ChordNote class: | |
# Represents a note within a chord, providing a way to activate it. When the | |
# final note of the chord is activated (this can be any note in the chord, | |
# with all other notes already previously activated in any order), the chord's | |
# body is evaluated. | |
class ChordNote { | |
field chord | |
field is_activated | |
# Constructor: | |
# Instances of ChordNote are created internally by calling add_note on | |
# SimpleChord objects. | |
constructor new {c} { | |
set chord $c | |
set is_activated 0 | |
return $this | |
} | |
# Method: | |
# [$note is_activated] | |
# Returns true if this note has already been activated. | |
method is_activated {} { | |
return $is_activated | |
} | |
# Method: | |
# $note activate | |
# Activates the note, if it has not already been activated, and | |
# completes the chord if there are no other notes awaiting | |
# activation. Subsequent calls will have no further effect. | |
method activate {} { | |
if {!$is_activated} { | |
set is_activated 1 | |
$chord notify_note_activation | |
} | |
} | |
} | |