/traits-4.2.0/traits/adapter.py
# · Python · 234 lines · 110 code · 30 blank · 94 comment · 3 complexity · 9caf833b5adbfc10d8920ef55b05044c MD5 · raw file
- #-------------------------------------------------------------------------------
- #
- # Copyright (c) 2007, Enthought, Inc.
- # All rights reserved.
- #
- # This software is provided without warranty under the terms of the BSD
- # license included in /LICENSE.txt and may be redistributed only
- # under the conditions described in the aforementioned license. The license
- # is also available online at http://www.enthought.com/licenses/BSD.txt
- #
- # Thanks for using Enthought open source!
- #
- # Author: Martin Chilvers
- # Date: 07/18/2007
- #
- #-------------------------------------------------------------------------------
- """ An extension to PyProtocols to simplify the declaration of adapters.
- """
- #-------------------------------------------------------------------------------
- # Imports:
- #-------------------------------------------------------------------------------
- from __future__ import absolute_import
- # Standard library imports:
- import weakref
- # Traits imports:
- from .has_traits import HasTraits
- from .trait_types import Any, Bool, Expression
- # PyProtocols imports:
- from .protocols.api import addClassAdvisor, declareAdapter, declareImplementation, Protocol
- #-------------------------------------------------------------------------------
- # 'Adapter' class:
- #-------------------------------------------------------------------------------
- class Adapter ( HasTraits ):
- """ The base class for all traits adapters.
- In Traits, an *adapter* is a special type of class whose role is to
- transform some type of object which does not implement a specific interface,
- or set of interfaces, into one that does.
- This class is provided as a convenience. If you subclass this class, the
- only things you need to add to the subclass definition are:
- * An implements() function call declaring which interfaces the adapter
- class implements on behalf of the object is is adapting.
- * A declaration for the **adaptee** trait, usually as an Instance of
- a particular class.
- * The actual implementations of the interfaces declared in the
- implements() call. Usually the implementation code is written in
- terms of the **adaptee** trait.
- """
- #-- Trait Definitions ------------------------------------------------------
- # The object that is being adapted.
- adaptee = Any
- #-- Constructor ------------------------------------------------------------
- def __init__ ( self, adaptee ):
- """ Constructor.
- We have to declare an explicit constructor because adapters are
- created by PyProtocols itself, which knows nothing about traits.
- """
- super( Adapter, self ).__init__()
- self.adaptee = adaptee
- #-------------------------------------------------------------------------------
- # 'DefaultAdapterFactory' class:
- #-------------------------------------------------------------------------------
- class DefaultAdapterFactory ( HasTraits ):
- """ An adapter factory for producing cached or categorized adapters.
- """
- #-- 'DefaultAdapterFactory' Interface --------------------------------------
- # The adapter class that this factory creates instances of
- klass = Any
- # Does the factory generate cached adapters?
- # If an adapter is cached then the factory will produce at most one
- # adapter per instance.
- cached = Bool( False )
- # An expression that is used to select which instances of a particular
- # type can be adapted by this factory.
- #
- # The expression is evaluated in a namespace that contains a single name
- # 'adaptee', which is bound to the object that this factory is attempting
- # to adapt (e.g. 'adaptee.is_folder').
- when = Expression
- #-- Private Interface ------------------------------------------------------
- # If this is a cached adapter factory, then this mapping will contain
- # the adapters keyed by weak references to the adapted objects.
- _adapters = Any
- #-------------------------------------------------------------------------------
- # 'IAdapterFactory' interface:
- #-------------------------------------------------------------------------------
- def __call__ ( self, object ):
- """ Creates an adapter for the specified object.
- Returns **None** if the factory cannot perform the required
- adaptation.
- """
- namespace = { 'adaptee': object }
- if eval( self.when_, namespace, namespace ):
- if self.cached:
- adapter = self._adapters.get( object )
- if adapter is None:
- self._adapters[ object ] = adapter = self.klass( object )
- return adapter
- return self.klass( object )
- return None
- #---------------------------------------------------------------------------
- # Private interface:
- #---------------------------------------------------------------------------
- def __adapters_default ( self ):
- """ Trait initializer.
- """
- return weakref.WeakKeyDictionary()
- #-------------------------------------------------------------------------------
- # 'adapts' function:
- #-------------------------------------------------------------------------------
- def adapts ( from_, to, extra = None, factory = None, cached = False,
- when = '' ):
- """ A class advisor for declaring adapters.
- Parameters
- ----------
- ``from_`` : type or interface
- What the adapter adapts *from*, or a list of such types or interfaces
- (the '_' suffix is used because 'from' is a Python keyword).
- to : type or interface
- What the adapter adapts *to*, or a list of such types or interfaces.
- factory : callable
- An (optional) factory for actually creating the adapters. This is
- any callable that takes a single argument which is the object to
- be adapted. The factory should return an adapter if it can
- perform the adaptation and **None** if it cannot.
- The following arguments are ignored if *factory* is specified:
- cached : Boolean
- Should the adapters be cached? If an adapter is cached, then the
- factory will produce at most one adapter per instance.
- when : A Python expression
- Selects which instances of a particular type can be adapted by this
- factory. The expression is evaluated in a namespace that contains a
- single name *adaptee*, which is bound to the object to be adapted
- (e.g., 'adaptee.is_folder').
- """
- if extra is not None:
- adapter, from_, to = from_, to, extra
- else:
- adapter = None
- def callback ( klass ):
- """ Called when the class has been created. """
- # What the adapter adapts from:
- if type( from_ ) is not list:
- for_items = [ from_ ]
- else:
- for_items = from_
- # The things we adapt from have to be split into two lists for
- # PyProtocols, one containing Python types (i.e. classes) and one
- # containing protocols (i.e. interfaces):
- for_types = []
- for_protocols = []
- for item in for_items:
- if isinstance( item, Protocol ):
- for_protocols.append( item )
- else:
- for_types.append( item )
- # What the adapter adapts to:
- if type( to ) is not list:
- provides = [ to ]
- else:
- provides = to
- # Tell PyProtocols that the adapter class implements the protocols that
- # it adapts to:
- declareImplementation( klass, instancesProvide = provides )
- # If a factory was specified then use it:
- if factory is not None:
- f = factory
- # If the adapter is cached or has a 'when' expression then create a
- # default factory:
- elif cached or (when != ''):
- f = DefaultAdapterFactory( klass = klass,
- cached = cached,
- when = when or 'True' )
- # Otherwise, just use the adapter class itself:
- else:
- f = klass
- # Tell PyProtocols about the factory:
- declareAdapter( f, provides, forProtocols = for_protocols,
- forTypes = for_types )
- return klass
- if adapter is not None:
- callback( adapter )
- else:
- addClassAdvisor( callback )