cybertools/composer/schema/field.py
helmutm a4dfb9b209 extend composer.schema for usage for standard form rendering
git-svn-id: svn://svn.cy55.de/Zope3/src/cybertools/trunk@2060 fd906abe-77d9-0310-91a1-e0d9ade77398
2007-09-19 15:46:03 +00:00

137 lines
3.8 KiB
Python

#
# Copyright (c) 2007 Helmut Merz helmutm@cy55.de
#
# This program is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation; either version 2 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
#
"""
Schema fields and related classes.
$Id$
"""
from zope.interface import implements
from zope.component import adapts
from zope import component
from cybertools.composer.base import Component
from cybertools.composer.schema.interfaces import IField, IFieldInstance
from cybertools.composer.schema.interfaces import fieldTypes
from cybertools.composer.schema.schema import formErrors
from cybertools.util.format import toStr, toUnicode
class Field(Component):
implements(IField)
required = False
standardFieldName = None
vocabulary = None
defaultValue = None
renderFactory = None
def __init__(self, name, title=None, fieldType='textline', **kw):
assert name
self.__name__ = name
title = title or u''
self.fieldType = fieldType
super(Field, self).__init__(title, __name__=name, **kw)
self.title = title
for k, v in kw.items():
setattr(self, k, v)
@property
def name(self):
return self.__name__
def getTitleValue(self):
return self.title or self.name
def getVocabularyItems(self):
voc = (self.vocabulary or '')
if isinstance(voc, basestring):
voc = voc.splitlines()
return [dict(token=t, title=t) for t in voc if t.strip()]
else:
return [dict(token=t.token, title=t.title) for t in voc]
def getFieldTypeInfo(self):
return fieldTypes.getTerm(self.fieldType)
def getFieldInstance(self):
instanceName = self.getFieldTypeInfo().instanceName
return component.getAdapter(self, IFieldInstance, name=instanceName)
class FieldInstance(object):
implements(IFieldInstance)
adapts(IField)
def __init__(self, context):
self.context = context
self.name = self.__name__ = context.name
self.errors = []
self.severity = 0
self.change = None
def marshall(self, value):
return value
#return toStr(value)
def display(self, value):
return value
#return toStr(value)
def unmarshall(self, value):
return toUnicode(value) or u''
def validate(self, value):
if not value and self.context.required:
self.setError('required_missing')
def setError(self, errorName):
error = formErrors[errorName]
self.errors.append(error)
self.severity = max(error.severity, self.severity)
class NumberFieldInstance(FieldInstance):
def marshall(self, value):
if value is None:
return ''
return str(value)
def display(self, value):
if value is None:
return '-'
return str(value)
def unmarshall(self, value):
if not value:
return None
return int(value)
def validate(self, value):
if value in ('', None):
if self.context.required:
self.setError('required_missing')
else:
try:
int(value)
except (TypeError, ValueError):
self.setError('invalid_number')