423 lines
12 KiB
Python
423 lines
12 KiB
Python
# classes for event-driven programming
|
|
# http://en.wikipedia.org/wiki/Event-driven_programming
|
|
|
|
__all__ = ['StateVar', 'FunctionCall', 'EnterExit', 'Pulse', 'EventPulse',
|
|
'EventArgument', ]
|
|
|
|
from direct.showbase.DirectObject import DirectObject
|
|
|
|
|
|
class PushesStateChanges:
|
|
# base class for objects that broadcast state changes to a set of subscriber objects
|
|
def __init__(self, value):
|
|
self._value = value
|
|
# push state changes to these objects
|
|
self._subscribers = set()
|
|
|
|
def destroy(self):
|
|
if len(self._subscribers) != 0:
|
|
raise Exception('%s object still has subscribers in destroy(): %s' % (
|
|
self.__class__.__name__, self._subscribers))
|
|
del self._subscribers
|
|
del self._value
|
|
|
|
def getState(self):
|
|
return self._value
|
|
|
|
def pushCurrentState(self):
|
|
self._handleStateChange()
|
|
return self
|
|
|
|
def _addSubscription(self, subscriber):
|
|
self._subscribers.add(subscriber)
|
|
subscriber._recvStatePush(self)
|
|
|
|
def _removeSubscription(self, subscriber):
|
|
self._subscribers.remove(subscriber)
|
|
|
|
def _handlePotentialStateChange(self, value):
|
|
oldValue = self._value
|
|
self._value = value
|
|
if oldValue != value:
|
|
self._handleStateChange()
|
|
|
|
def _handleStateChange(self):
|
|
# push this object's state to the subscribing objects
|
|
for subscriber in self._subscribers:
|
|
subscriber._recvStatePush(self)
|
|
|
|
if __debug__:
|
|
psc = PushesStateChanges(0)
|
|
assert psc.getState() == 0
|
|
psc.destroy()
|
|
del psc
|
|
|
|
class ReceivesStateChanges:
|
|
# base class for objects that subscribe to state changes from PushesStateChanges objects
|
|
def __init__(self, source):
|
|
self._source = None
|
|
self._initSource = source
|
|
|
|
def _finishInit(self):
|
|
# initialization is split across two functions to allow objects that derive from this
|
|
# class to set everything up so that they can respond appropriately to the initial
|
|
# state push from the state source
|
|
self._subscribeTo(self._initSource)
|
|
del self._initSource
|
|
|
|
def destroy(self):
|
|
self._unsubscribe()
|
|
del self._source
|
|
|
|
def _subscribeTo(self, source):
|
|
self._unsubscribe()
|
|
self._source = source
|
|
if self._source:
|
|
self._source._addSubscription(self)
|
|
|
|
def _unsubscribe(self):
|
|
if self._source:
|
|
self._source._removeSubscription(self)
|
|
self._source = None
|
|
|
|
def _recvStatePush(self, source):
|
|
pass
|
|
|
|
if __debug__:
|
|
rsc = ReceivesStateChanges(None)
|
|
rsc.destroy()
|
|
del rsc
|
|
|
|
class StateVar(PushesStateChanges):
|
|
# coder-friendly object that allows values to be set on it and pushes those values
|
|
# as state changes
|
|
def set(self, value):
|
|
PushesStateChanges._handlePotentialStateChange(self, value)
|
|
|
|
def get(self):
|
|
return PushesStateChanges.getState(self)
|
|
|
|
if __debug__:
|
|
sv = StateVar(0)
|
|
assert sv.get() == 0
|
|
sv.set(1)
|
|
assert sv.get() == 1
|
|
sv.destroy()
|
|
del sv
|
|
|
|
class StateChangeNode(PushesStateChanges, ReceivesStateChanges):
|
|
# base class that can be used to create a state-change notification chain
|
|
def __init__(self, source):
|
|
ReceivesStateChanges.__init__(self, source)
|
|
PushesStateChanges.__init__(self, source.getState())
|
|
ReceivesStateChanges._finishInit(self)
|
|
|
|
def destroy(self):
|
|
PushesStateChanges.destroy(self)
|
|
ReceivesStateChanges.destroy(self)
|
|
|
|
def _recvStatePush(self, source):
|
|
# got a state push, apply new state to self
|
|
self._handlePotentialStateChange(source._value)
|
|
|
|
if __debug__:
|
|
sv = StateVar(0)
|
|
assert sv.get() == 0
|
|
scn = StateChangeNode(sv)
|
|
assert scn.getState() == 0
|
|
sv.set(1)
|
|
assert sv.get() == 1
|
|
assert scn.getState() == 1
|
|
scn2 = StateChangeNode(scn)
|
|
assert scn2.getState() == 1
|
|
sv.set(2)
|
|
assert scn2.getState() == 2
|
|
scn3 = StateChangeNode(scn)
|
|
assert scn3.getState() == 2
|
|
sv.set(3)
|
|
assert scn2.getState() == 3
|
|
assert scn3.getState() == 3
|
|
scn3.destroy()
|
|
scn2.destroy()
|
|
scn.destroy()
|
|
sv.destroy()
|
|
del scn3
|
|
del scn2
|
|
del scn
|
|
del sv
|
|
|
|
class ReceivesMultipleStateChanges:
|
|
# base class for objects that subscribe to state changes from multiple PushesStateChanges
|
|
# objects
|
|
def __init__(self):
|
|
self._key2source = {}
|
|
self._source2key = {}
|
|
|
|
def destroy(self):
|
|
keys = list(self._key2source.keys())
|
|
for key in keys:
|
|
self._unsubscribe(key)
|
|
del self._key2source
|
|
del self._source2key
|
|
|
|
def _subscribeTo(self, source, key):
|
|
self._unsubscribe(key)
|
|
self._key2source[key] = source
|
|
self._source2key[source] = key
|
|
source._addSubscription(self)
|
|
|
|
def _unsubscribe(self, key):
|
|
if key in self._key2source:
|
|
source = self._key2source[key]
|
|
source._removeSubscription(self)
|
|
del self._key2source[key]
|
|
del self._source2key[source]
|
|
|
|
def _recvStatePush(self, source):
|
|
self._recvMultiStatePush(self._source2key[source], source)
|
|
|
|
def _recvMultiStatePush(self, key, source):
|
|
pass
|
|
|
|
if __debug__:
|
|
rsc = ReceivesMultipleStateChanges()
|
|
sv = StateVar(0)
|
|
sv2 = StateVar('b')
|
|
rsc._subscribeTo(sv, 'a')
|
|
rsc._subscribeTo(sv2, 2)
|
|
rsc._unsubscribe('a')
|
|
rsc.destroy()
|
|
del rsc
|
|
|
|
class FunctionCall(ReceivesMultipleStateChanges, PushesStateChanges):
|
|
# calls func with provided args whenever arguments' state changes
|
|
def __init__(self, func, *args, **kArgs):
|
|
self._initialized = False
|
|
ReceivesMultipleStateChanges.__init__(self)
|
|
PushesStateChanges.__init__(self, None)
|
|
self._func = func
|
|
self._args = args
|
|
self._kArgs = kArgs
|
|
# keep a copy of the arguments ready to go, already filled in with
|
|
# the value of arguments that push state
|
|
self._bakedArgs = []
|
|
self._bakedKargs = {}
|
|
for i, arg in enumerate(self._args):
|
|
key = i
|
|
if isinstance(arg, PushesStateChanges):
|
|
self._bakedArgs.append(arg.getState())
|
|
self._subscribeTo(arg, key)
|
|
else:
|
|
self._bakedArgs.append(self._args[i])
|
|
for key, arg in self._kArgs.items():
|
|
if isinstance(arg, PushesStateChanges):
|
|
self._bakedKargs[key] = arg.getState()
|
|
self._subscribeTo(arg, key)
|
|
else:
|
|
self._bakedKargs[key] = arg
|
|
self._initialized = True
|
|
# call pushCurrentState() instead
|
|
## push the current state to any listeners
|
|
##self._handleStateChange()
|
|
|
|
def destroy(self):
|
|
ReceivesMultipleStateChanges.destroy(self)
|
|
PushesStateChanges.destroy(self)
|
|
del self._func
|
|
del self._args
|
|
del self._kArgs
|
|
del self._bakedArgs
|
|
del self._bakedKargs
|
|
|
|
def getState(self):
|
|
# for any state recievers that are hooked up to us, they get a tuple
|
|
# of (tuple(positional argument values), dict(keyword argument name->value))
|
|
return (tuple(self._bakedArgs), dict(self._bakedKargs))
|
|
|
|
def _recvMultiStatePush(self, key, source):
|
|
# one of the arguments changed
|
|
# pick up the new value
|
|
if isinstance(key, str):
|
|
self._bakedKargs[key] = source.getState()
|
|
else:
|
|
self._bakedArgs[key] = source.getState()
|
|
# and send it out
|
|
self._handlePotentialStateChange(self.getState())
|
|
|
|
def _handleStateChange(self):
|
|
if self._initialized:
|
|
self._func(*self._bakedArgs, **self._bakedKargs)
|
|
PushesStateChanges._handleStateChange(self)
|
|
|
|
if __debug__:
|
|
l = []
|
|
def handler(value, l=l):
|
|
l.append(value)
|
|
assert l == []
|
|
sv = StateVar(0)
|
|
fc = FunctionCall(handler, sv)
|
|
assert l == []
|
|
fc.pushCurrentState()
|
|
assert l == [0,]
|
|
sv.set(1)
|
|
assert l == [0,1,]
|
|
sv.set(2)
|
|
assert l == [0,1,2,]
|
|
fc.destroy()
|
|
sv.destroy()
|
|
del fc
|
|
del sv
|
|
del handler
|
|
del l
|
|
|
|
l = []
|
|
def handler(value, kDummy=None, kValue=None, l=l):
|
|
l.append((value, kValue))
|
|
assert l == []
|
|
sv = StateVar(0)
|
|
ksv = StateVar('a')
|
|
fc = FunctionCall(handler, sv, kValue=ksv)
|
|
assert l == []
|
|
fc.pushCurrentState()
|
|
assert l == [(0,'a',),]
|
|
sv.set(1)
|
|
assert l == [(0,'a'),(1,'a'),]
|
|
ksv.set('b')
|
|
assert l == [(0,'a'),(1,'a'),(1,'b'),]
|
|
fc.destroy()
|
|
sv.destroy()
|
|
del fc
|
|
del sv
|
|
del handler
|
|
del l
|
|
|
|
class EnterExit(StateChangeNode):
|
|
# call enterFunc when our state becomes true, exitFunc when it becomes false
|
|
def __init__(self, source, enterFunc, exitFunc):
|
|
self._enterFunc = enterFunc
|
|
self._exitFunc = exitFunc
|
|
StateChangeNode.__init__(self, source)
|
|
|
|
def destroy(self):
|
|
StateChangeNode.destroy(self)
|
|
del self._exitFunc
|
|
del self._enterFunc
|
|
|
|
def _handlePotentialStateChange(self, value):
|
|
# convert the incoming state as a bool
|
|
StateChangeNode._handlePotentialStateChange(self, bool(value))
|
|
|
|
def _handleStateChange(self):
|
|
if self._value:
|
|
self._enterFunc()
|
|
else:
|
|
self._exitFunc()
|
|
StateChangeNode._handleStateChange(self)
|
|
|
|
if __debug__:
|
|
l = []
|
|
def enter(l=l):
|
|
l.append(1)
|
|
def exit(l=l):
|
|
l.append(0)
|
|
sv = StateVar(0)
|
|
ee = EnterExit(sv, enter, exit)
|
|
sv.set(0)
|
|
assert l == []
|
|
sv.set(1)
|
|
assert l == [1,]
|
|
sv.set(2)
|
|
assert l == [1,]
|
|
sv.set(0)
|
|
assert l == [1,0,]
|
|
sv.set(True)
|
|
assert l == [1,0,1,]
|
|
sv.set(False)
|
|
assert l == [1,0,1,0,]
|
|
ee.destroy()
|
|
sv.destroy()
|
|
del ee
|
|
del sv
|
|
del enter
|
|
del exit
|
|
del l
|
|
|
|
class Pulse(PushesStateChanges):
|
|
# changes state to True then immediately to False whenever sendPulse is called
|
|
def __init__(self):
|
|
PushesStateChanges.__init__(self, False)
|
|
|
|
def sendPulse(self):
|
|
self._handlePotentialStateChange(True)
|
|
self._handlePotentialStateChange(False)
|
|
|
|
if __debug__:
|
|
l = []
|
|
def handler(value, l=l):
|
|
l.append(value)
|
|
p = Pulse()
|
|
fc = FunctionCall(handler, p)
|
|
assert l == []
|
|
fc.pushCurrentState()
|
|
assert l == [False, ]
|
|
p.sendPulse()
|
|
assert l == [False, True, False, ]
|
|
p.sendPulse()
|
|
assert l == [False, True, False, True, False, ]
|
|
fc.destroy()
|
|
p.destroy()
|
|
del fc
|
|
del p
|
|
del l
|
|
del handler
|
|
|
|
class EventPulse(Pulse, DirectObject):
|
|
# sends a True-False "pulse" whenever a specific messenger message is sent
|
|
def __init__(self, event):
|
|
Pulse.__init__(self)
|
|
self.accept(event, self.sendPulse)
|
|
|
|
def destroy(self):
|
|
self.ignoreAll()
|
|
Pulse.destroy(self)
|
|
|
|
class EventArgument(PushesStateChanges, DirectObject):
|
|
# tracks a particular argument to a particular messenger event
|
|
def __init__(self, event, index=0):
|
|
PushesStateChanges.__init__(self, None)
|
|
self._index = index
|
|
self.accept(event, self._handleEvent)
|
|
|
|
def destroy(self):
|
|
self.ignoreAll()
|
|
del self._index
|
|
PushesStateChanges.destroy(self)
|
|
|
|
def _handleEvent(self, *args):
|
|
self._handlePotentialStateChange(args[self._index])
|
|
|
|
class AttrSetter(StateChangeNode):
|
|
def __init__(self, source, object, attrName):
|
|
self._object = object
|
|
self._attrName = attrName
|
|
StateChangeNode.__init__(self, source)
|
|
self._handleStateChange()
|
|
|
|
def _handleStateChange(self):
|
|
setattr(self._object, self._attrName, self._value)
|
|
StateChangeNode._handleStateChange(self)
|
|
|
|
if __debug__:
|
|
o = ScratchPad()
|
|
svar = StateVar(0)
|
|
aset = AttrSetter(svar, o, 'testAttr')
|
|
assert hasattr(o, 'testAttr')
|
|
assert o.testAttr == 0
|
|
svar.set('red')
|
|
assert o.testAttr == 'red'
|
|
aset.destroy()
|
|
svar.destroy()
|
|
o.destroy()
|
|
del aset
|
|
del svar
|
|
del o
|