cybertools/composer/schema/instance.py
helmutm 90377b06b9 refactoring field processing: class FieldInstance for processing and collecting state information
git-svn-id: svn://svn.cy55.de/Zope3/src/cybertools/trunk@1971 fd906abe-77d9-0310-91a1-e0d9ade77398
2007-08-30 07:53:09 +00:00

128 lines
4.3 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
#
"""
Instance adapter classes for schemas.
$Id$
"""
from BTrees.OOBTree import OOBTree
from zope.component import adapts
from zope.interface import implements
from cybertools.composer.instance import Instance
from cybertools.composer.interfaces import IInstance
from cybertools.composer.schema.interfaces import IClient, IFieldInstance
from cybertools.composer.schema.schema import FormState
class Editor(Instance):
aspect = 'schema.editor.default'
def applyTemplate(self, data={}, *args, **kw):
for c in self.template.components:
# TODO: implement the real stuff
# save data (if available) in context
# build sequence of fields with data from context
# or directly use request...
print c.name, getattr(self.context, c.name, '-')
class ClientInstance(object):
implements(IInstance)
adapts(IClient)
attrsName = '__schema_attributes__'
template = None
baseAspect = 'schema.client.'
@property
def aspect(self):
return self.baseAspect + self.template.name
def __init__(self, context):
self.context = context
def applyTemplate(self, **kw):
""" Return a mapping of field names from self.template (a schema)
to the corresponding values from the context object.
"""
result = {}
mode = kw.get('mode', 'view')
attrs = getattr(self.context, self.attrsName, None)
if attrs is None:
return result
template = self.template
values = attrs.setdefault(self.aspect, {})
if template is not None:
for f in template.fields:
fi = IFieldInstance(f)
name = f.name
value = values.get(name, u'')
value = mode == 'view' and fi.display(value) or fi.marshall(value)
result[name] = value
result['__name__'] = self.context.__name__
return result
class ClientInstanceEditor(ClientInstance):
def applyTemplate(self, data={}, **kw):
""" Store the attributes described by self.template (a schema)
using corresponding values from the data argument.
Return the resulting form state (an object providing IFormState).
"""
template = self.template
if template is None:
return FormState()
formState = self.validate(data)
if formState.severity > 0:
# don't do anything if there is an error
return formState
attrs = getattr(self.context, self.attrsName, None)
if attrs is None:
attrs = OOBTree()
setattr(self.context, self.attrsName, attrs)
values = attrs.setdefault(self.aspect, OOBTree())
for f in template.fields:
#fi = IFieldInstance(f)
name = f.name
fi = formState.fieldInstances[name]
value = fi.unmarshall(data.get(name))
if name in data:
oldValue = values.get(name)
if value != oldValue:
values[name] = value
fi.change = (oldValue, value)
formState.changed = True
return formState
def validate(self, data):
formState = FormState()
for f in self.template.fields:
fi = IFieldInstance(f)
value = fi.unmarshall(data.get(f.name))
fi.validate(value)
formState.fieldInstances.append(fi)
formState.severity = max(formState.severity, fi.severity)
return formState