0

I'm getting confused reading about the differences in newstyle vs oldstyle attributes. In this example:

  1. Is this code using old-style method to modify attributes?

    In Event() at self.__dict__.update(kw)

  2. Do I use getattr, getattr or getattribute ?

    for Dispatcher.dispatch_to_parent()

The code is from http://www.pygame.org/wiki/CommandDispatch?parent=CookBook

class Event(object):
    def __init__(self, name, kw={}):
        self.name = name
        self.__dict__.update(kw)
        self.dict = dict(kw)
        self.method_name = 'EVT_%s' % self.name

    def __repr__(self):
        return "<event>" % (self.name, self.dict)

class Dispatcher(object):
    def add(self, other):
        if isinstance(other, Dispatcher):
            other.parent = self
        else:
            raise ValueError("%s is not a Dispatcher" % other)

    def remove(self, other):
        if isinstance(other, Dispatcher):
            if other.parent is self:
                del other.parent
            else:
                raise ValueError("%s is not my child." % other)
        else:
            raise ValueError("%s is not a Dispatcher" % other)

    def dispatch(self, event):
        method = getattr(self, event.method_name, None)
        if method is not None: 
            return method(event)
        else:
            self.dispatch_to_parent(event)

    def dispatch_to_parent(self, event):
        parent = getattr(self, 'parent', None)
        if parent is not None:
            dispatch = getattr(parent, 'dispatch', None)
            if dispatch is not None: 
                return dispatch(event)

class Root(Dispatcher):
    def __init__(self):
        Dispatcher.__init__(self)
        self.focused_node = self

    def focus(self, node):
        if node is self.focused_node: return
        print 'focusing on', node
        self.focused_node.dispatch(Event('BLUR'))
        self.focused_node = node
        node.dispatch(Event('FOCUS'))

    def blur(self, node):
        self.focused_node = self
        node.dispatch(Event('BLUR'))

    def post(self, event_name, kw):
        event = Event(event_name, kw)
        return self.focused_node.dispatch(event)

if __name__ == "__main__":
    class Button(Dispatcher):
        def EVT_KeyDown(self, event):
            return self
    #create the root node, which is the receiver of all events.        
    r = Root()
    #create a button node
    b = Button()
    #add the button node to the application
    r.add(b)
    #give the button node focus, it will now receive events
    r.focus(b)
    #post a KeyDown event, which the button will receive
    print r.post('KeyDown')
    #post a Quit event, which the button will not receive. 
    #this event will be passed onto the button parent
    print r.post('Quit')
    #the return value of the post method is the node which handled the event
JBernardo
  • 32,262
  • 10
  • 90
  • 115
ninMonkey
  • 7,211
  • 8
  • 37
  • 66

2 Answers2

3

You shouldn't update the instance dictionary directly, you will bypass custom setters. Loop over the keys and do setattr.

MatthewWilkes
  • 1,048
  • 8
  • 15
  • Is it always a problem to bypass custom setters? What are the performance of the two options= – G M Dec 06 '22 at 09:36
0

There's a missing * in def post(self, event_name, kw): line

In due to make you function able to receive a dict as a parameter, you need to add ** before variables name:

def post(self, event_name, **kw):
    event = Event(event_name, kw)
    return self.focused_node.dispatch(event)
Leo Pepe
  • 319
  • 2
  • 5