-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathbase_choices.py
69 lines (53 loc) · 1.73 KB
/
base_choices.py
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
67
68
69
# -*- coding: utf-8 -*-
# @Author: theo-l
# @Date: 2017-09-08 11:55:34
# @Last Modified by: theo-l
# @Last Modified time: 2017-09-08 11:55:49
import six
class ChoiceItem(object):
def __init__(self, value, desc):
self._value = value
self._desc = desc
@property
def value(self):
return self._value
@property
def desc(self):
return self._desc
def getvalue(self):
return (self.value, self.desc)
def __str__(self):
return str(self.getvalue())
class ChoiceMeta(type):
"""
Meta class for all choice class
"""
def __new__(metaclass, cls, bases, attr):
"""
1. parse each ChoiceItem instance attribute into a special attribute 'choices'
2. reset the ChoiceItem attribute value to the 'value' of ChoiceItem item
"""
base_meta = super(ChoiceMeta, metaclass).__new__(metaclass, cls, bases, attr)
if attr:
choices = []
for field, value in attr.items():
if isinstance(value, ChoiceItem):
choices.append(value.getvalue())
setattr(base_meta, field, value.value)
if choices:
choices = tuple(choices)
setattr(base_meta, 'choices', choices)
return base_meta
class BaseChoice(six.with_metaclass(ChoiceMeta)):
"""
Base class of all choices
Usage:
# customize a subclass of choice
class ColorChoice(BaseChoice):
RED = ChoiceItem(1, 'Red')
GREEN = ChoiceItem(2, 'Green')
# in model definition
class DemoModel(models.Model):
color = models.IntegerField(choices=ColorChoice.choices, default=ColorChoice.RED)
"""
pass