summaryrefslogtreecommitdiff
path: root/gi/overrides/__init__.py
blob: a98974f6ed6d8744a2ad542c0cf1245f7bdb10d0 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
import sys
import types

import gobject

registry = None
class _Registry(dict):
    def __setitem__(self, key, value):
        '''We do checks here to make sure only submodules of the override 
        module are added.  Key and value should be the same object and come
        from the gi.override module.  

        We add the override to the dict as "override_module.name".  For instance
        if we were overriding Gtk.Button you would retrive it as such:
        registry['Gtk.Button']
        ''' 
        if not key == value:
            raise KeyError('You have tried to modify the registry.  This should only be done by the override decorator')

        info = getattr(value, '__info__')
        if info == None:
            raise KeyError('Can not override a type %s, which is not in a gobject introspection typelib' % value.__name__)

        if not value.__module__.startswith('gi.overrides'):
            raise KeyError('You have tried to modify the registry outside of the overrides module.  This is not allowed')

        g_type = info.get_g_type()
        assert g_type != gobject.TYPE_NONE
        if g_type != gobject.TYPE_INVALID:
            g_type.pytype = value 

            # strip gi.overrides from module name
            module =  value.__module__[13:]
            key = "%s.%s" % (module, value.__name__)
            super(_Registry, self).__setitem__(key, value)

    def register(self, override_class):
        self[override_class] = override_class


class overridefunc(object):
    '''decorator for overriding a function'''
    def __init__(self, func):
        if not hasattr(func, '__info__'):
            raise TypeError("func must be an gi function")
        from ..importer import modules
        self.module = modules[func.__module__]._introspection_module

    def __call__(self, func):
        def wrapper(*args, **kwargs):
            return func(*args, **kwargs)
        wrapper.__name__ = func.__name__
        setattr(self.module, func.__name__, wrapper)
        return wrapper

registry = _Registry()

def override(type_):
    '''Decorator for registering an override'''
    if type(type_) == types.FunctionType:
        return overridefunc(type_)
    else:
        registry.register(type_)
        return type_