2007-11-05 21:59:42 +08:00
|
|
|
"""
|
2009-12-22 23:18:51 +08:00
|
|
|
Convenience routines for creating non-trivial Field subclasses, as well as
|
|
|
|
backwards compatibility utilities.
|
2007-11-05 21:59:42 +08:00
|
|
|
|
2012-08-15 05:44:46 +08:00
|
|
|
Add SubfieldBase as the metaclass for your Field subclass, implement
|
|
|
|
to_python() and the other necessary methods and everything will work
|
|
|
|
seamlessly.
|
2007-11-05 21:59:42 +08:00
|
|
|
"""
|
|
|
|
|
2014-08-12 20:08:40 +08:00
|
|
|
import warnings
|
|
|
|
|
|
|
|
from django.utils.deprecation import RemovedInDjango20Warning
|
|
|
|
|
2013-07-08 08:39:54 +08:00
|
|
|
|
2011-03-31 01:48:42 +08:00
|
|
|
class SubfieldBase(type):
|
2007-11-05 21:59:42 +08:00
|
|
|
"""
|
|
|
|
A metaclass for custom Field subclasses. This ensures the model's attribute
|
|
|
|
has the descriptor protocol attached to it.
|
|
|
|
"""
|
2010-12-04 13:59:38 +08:00
|
|
|
def __new__(cls, name, bases, attrs):
|
2014-08-12 20:08:40 +08:00
|
|
|
warnings.warn("SubfieldBase has been deprecated. Use Field.from_db_value instead.",
|
|
|
|
RemovedInDjango20Warning)
|
|
|
|
|
2010-12-04 13:59:38 +08:00
|
|
|
new_class = super(SubfieldBase, cls).__new__(cls, name, bases, attrs)
|
2007-11-05 21:59:42 +08:00
|
|
|
new_class.contribute_to_class = make_contrib(
|
2010-11-03 02:40:53 +08:00
|
|
|
new_class, attrs.get('contribute_to_class')
|
|
|
|
)
|
2007-11-05 21:59:42 +08:00
|
|
|
return new_class
|
|
|
|
|
2013-07-08 08:39:54 +08:00
|
|
|
|
2007-11-05 21:59:42 +08:00
|
|
|
class Creator(object):
|
|
|
|
"""
|
|
|
|
A placeholder class that provides a way to set the attribute on the model.
|
|
|
|
"""
|
|
|
|
def __init__(self, field):
|
|
|
|
self.field = field
|
|
|
|
|
|
|
|
def __get__(self, obj, type=None):
|
|
|
|
if obj is None:
|
2013-09-18 14:56:05 +08:00
|
|
|
return self
|
2009-12-22 23:18:51 +08:00
|
|
|
return obj.__dict__[self.field.name]
|
2007-11-05 21:59:42 +08:00
|
|
|
|
|
|
|
def __set__(self, obj, value):
|
2007-11-30 03:30:49 +08:00
|
|
|
obj.__dict__[self.field.name] = self.field.to_python(value)
|
2007-11-05 21:59:42 +08:00
|
|
|
|
2013-07-08 08:39:54 +08:00
|
|
|
|
2010-11-03 02:40:53 +08:00
|
|
|
def make_contrib(superclass, func=None):
|
2007-11-05 21:59:42 +08:00
|
|
|
"""
|
|
|
|
Returns a suitable contribute_to_class() method for the Field subclass.
|
|
|
|
|
|
|
|
If 'func' is passed in, it is the existing contribute_to_class() method on
|
|
|
|
the subclass and it is called before anything else. It is assumed in this
|
|
|
|
case that the existing contribute_to_class() calls all the necessary
|
|
|
|
superclass methods.
|
|
|
|
"""
|
2014-07-23 17:41:06 +08:00
|
|
|
def contribute_to_class(self, cls, name, **kwargs):
|
2007-11-05 21:59:42 +08:00
|
|
|
if func:
|
2014-07-23 17:41:06 +08:00
|
|
|
func(self, cls, name, **kwargs)
|
2007-11-05 21:59:42 +08:00
|
|
|
else:
|
2014-07-23 17:41:06 +08:00
|
|
|
super(superclass, self).contribute_to_class(cls, name, **kwargs)
|
2007-11-05 21:59:42 +08:00
|
|
|
setattr(cls, self.name, Creator(self))
|
|
|
|
|
|
|
|
return contribute_to_class
|