loops/browser/form.py
helmutm 46a1638edd use composer.schema for form/dialog handling
git-svn-id: svn://svn.cy55.de/Zope3/src/loops/trunk@2065 fd906abe-77d9-0310-91a1-e0d9ade77398
2007-09-22 14:38:06 +00:00

507 lines
16 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
#
"""
Classes for form presentation and processing.
$Id$
"""
from zope import component, interface, schema
from zope.component import adapts
from zope.event import notify
from zope.lifecycleevent import ObjectCreatedEvent, ObjectModifiedEvent
from zope.app.container.interfaces import INameChooser
from zope.app.container.contained import NameChooser
from zope.app.form.browser.textwidgets import FileWidget, TextAreaWidget
from zope.app.pagetemplate import ViewPageTemplateFile
from zope.cachedescriptors.property import Lazy
from zope.contenttype import guess_content_type
from zope.formlib.form import Form, EditForm, FormFields
from zope.publisher.browser import FileUpload
from zope.publisher.interfaces import BadRequest
from zope.security.proxy import isinstance, removeSecurityProxy
from cybertools.ajax import innerHtml
from cybertools.browser.form import FormController
from cybertools.composer.interfaces import IInstance
from cybertools.composer.schema.browser.common import schema_macros, schema_edit_macros
from cybertools.composer.schema.util import getSchemaFromInterface
from cybertools.typology.interfaces import IType, ITypeManager
from loops.common import adapted
from loops.concept import Concept, ResourceRelation
from loops.interfaces import IConcept, IResourceManager, IDocument
from loops.interfaces import IFile, IExternalFile, INote, ITextDocument
from loops.browser.node import NodeView
from loops.browser.concept import ConceptRelationView
from loops.query import ConceptQuery
from loops.resource import Resource
from loops.type import ITypeConcept
from loops import util
from loops.util import _
from loops.versioning.interfaces import IVersionable
# special widgets
class UploadWidget(FileWidget):
def _toFieldValue(self, input):
# not used at the moment as the context object is updated
# via EditObject.updateFields()
fn = getattr(input, 'filename', '') # zope.publisher.browser.FileUpload
self.request.form['filename'] = fn
if input:
self.request.form['_tempfilename'] = input.headers.get('_tempfilename')
# f = self.context
# f.extfiledata = tempfilename # provide for rename
if fn:
contentType = guess_content_type(fn)
if contentType:
request.form['form.contentType'] = contentType
return super(UploadWidget, self)._toFieldValue(input)
# forms
class ObjectForm(NodeView):
""" Abstract base class for forms.
"""
template = ViewPageTemplateFile('form_macros.pt')
_isSetUp = False
def __init__(self, context, request):
super(ObjectForm, self).__init__(context, request)
# cybertools.composer.schema support
@property
def schemaMacros(self):
return schema_macros.macros
@property
def schemaEditMacros(self):
return schema_edit_macros.macros
@Lazy
def schema(self):
return getSchemaFromInterface(self.typeInterface, manager=self)
@Lazy
def fields(self):
fields = self.schema.fields
fields.data.height = 10
ifc = self.typeInterface
if ifc in widgetControllers:
wc = widgetControllers[ifc](self.context, self.request)
wc.modifySchemaFields(fields)
return fields
@Lazy
def data(self):
instance = self.instance
instance.template = self.schema
data = instance.applyTemplate(mode='edit')
for k, v in data.items():
#overwrite data with values from request.form
if k in self.request.form:
data[k] = form[k]
return data
@Lazy
def instance(self):
return IInstance(adapted(self.context))
# zope.formlib support
@property
def form_fields(self):
ifc = self.typeInterface
ff = FormFields(ifc)
if ifc in widgetControllers:
wc = widgetControllers[ifc](self.context, self.request)
ff = wc.modifyFormFields(ff)
return ff
def setUp(self):
if self._isSetUp:
return
self.setUpWidgets()
desc = self.widgets.get('description')
if desc:
desc.height = 2
if self.typeInterface in widgetControllers:
wc = widgetControllers[self.typeInterface](self.context, self.request)
wc.modifyWidgetSetup(self.widgets)
self._isSetUp = True
# general methods
def __call__(self):
response = self.request.response
response.setHeader('Expires', 'Sat, 1 Jan 2000 00:00:00 GMT')
response.setHeader('Pragma', 'no-cache')
return innerHtml(self)
@Lazy
def typeInterface(self):
return IType(self.context).typeInterface or ITextDocument
@Lazy
def defaultPredicate(self):
return self.loopsRoot.getConceptManager().getDefaultPredicate()
@Lazy
def defaultPredicateUid(self):
return util.getUidForObject(self.defaultPredicate)
@Lazy
def typeManager(self):
return ITypeManager(self.context)
@Lazy
def presetTypesForAssignment(self):
types = list(self.typeManager.listTypes(include=('assign',)))
assigned = [r.context.conceptType for r in self.assignments]
types = [t for t in types if t.typeProvider not in assigned]
return [dict(title=t.title, token=t.tokenForSearch) for t in types]
def conceptsForType(self, token):
noSelection = dict(token='none', title=u'not selected')
result = sorted(ConceptQuery(self).query(type=token), key=lambda x: x.title)
predicateUid = self.defaultPredicateUid
return ([noSelection] +
[dict(title=o.title,
token='%s:%s' % (util.getUidForObject(o), predicateUid))
for o in result])
class WidgetController(object):
def __init__(self, context, request):
self.context = context
self.request = request
def modifySchemaFields(self, fields):
pass
def modifyFormFields(self, formFields):
return formFields
def modifyWidgetSetup(self, widgets):
pass
class NoteWidgetController(WidgetController):
def modifySchemaFields(self, fields):
del fields['description']
fields.data.height = 5
def modifyFormFields(self, formFields):
return formFields.omit('description')
def modifyWidgetSetup(self, widgets):
widgets['data'].height = 5
class FileWidgetController(WidgetController):
def modifySchemaFields(self, fields):
if self.request.principal.id != 'rootadmin':
del fields['contentType']
def modifyFormFields(self, formFields):
if self.request.principal.id == 'rootadmin':
return formFields
return formFields.omit('contentType')
widgetControllers = {
INote: NoteWidgetController,
IFile: FileWidgetController,
IExternalFile: FileWidgetController,
}
class EditObjectForm(ObjectForm, EditForm):
@property
def macro(self):
return self.template.macros['edit']
title = _(u'Edit Resource')
form_action = 'edit_resource'
dialog_name = 'edit'
def __init__(self, context, request):
super(EditObjectForm, self).__init__(context, request)
self.url = self.url # keep virtual target URL (???)
self.context = self.virtualTargetObject
@property
def assignments(self):
for c in self.context.getConceptRelations():
r = ConceptRelationView(c, self.request)
if r.isProtected: continue
yield r
class CreateObjectForm(ObjectForm, Form):
@property
def macro(self): return self.template.macros['create']
title = _(u'Create Resource, Type = ')
form_action = 'create_resource'
dialog_name = 'create'
# cybertools.composer.schema support
@Lazy
def instance(self):
return IInstance(Concept())
# general methods
@Lazy
def typeInterface(self):
typeToken = self.request.get('form.type')
if typeToken:
t = self.loopsRoot.loopsTraverse(typeToken)
return removeSecurityProxy(ITypeConcept(t).typeInterface)
else:
#return INote
return ITextDocument
@property
def assignments(self):
target = self.virtualTargetObject
if (IConcept.providedBy(target) and
target.conceptType !=
self.loopsRoot.getConceptManager().getTypeConcept()):
rv = ConceptRelationView(ResourceRelation(target, None), self.request)
return (rv,)
return ()
class InnerForm(CreateObjectForm):
@property
#def macro(self): return self.template.macros['fields']
def macro(self): return self.schemaMacros['fields']
# processing form input
class EditObject(FormController):
prefix = 'form.'
conceptPrefix = 'assignments.'
old = None
selected = None
@Lazy
def schema(self):
return getSchemaFromInterface(self.typeInterface, manager=self)
@Lazy
def fields(self):
return self.schema.fields
@Lazy
def instance(self):
return component.getAdapter(adapted(self.object), IInstance, name='editor')
#return IInstance(adapted(self.object), name='editor')
@Lazy
def typeInterface(self):
return IType(self.object).typeInterface or ITextDocument
@Lazy
def loopsRoot(self):
return self.view.loopsRoot
def update(self):
# create new version if necessary
target = self.view.virtualTargetObject
obj = self.checkCreateVersion(target)
if obj != target:
# make sure new version is used by the view
self.view.virtualTargetObject = obj
self.request.annotations['loops.view']['target'] = obj
self.object = obj
formState = self.updateFields()
# TODO: error handling
#errors = self.updateFields()
#if errors:
# self.view.setUp()
# for fieldName, message in errors.items():
# self.view.widgets[fieldName].error = message
# return True
self.request.response.redirect(self.view.virtualTargetUrl + '?version=this')
return False
def updateFields(self):
obj = self.object
form = self.request.form
instance = self.instance
instance.template = self.schema
formState = instance.applyTemplate(data=form, fieldHandlers=self.fieldHandlers)
for k in form.keys():
if k.startswith(self.prefix):
fn = k[len(self.prefix):]
value = form[k]
if fn.startswith(self.conceptPrefix) and value:
self.collectConcepts(fn[len(self.conceptPrefix):], value)
if self.old or self.selected:
self.assignConcepts(obj)
notify(ObjectModifiedEvent(obj))
return formState
def handleFileUpload(self, context, value, fieldInstance, formState):
""" Special handler for fileupload fields;
value is a FileUpload instance.
"""
filename = getattr(value, 'filename', '')
if filename: # ignore if no filename present - no file uploaded
value = value.read()
contentType = guess_content_type(filename, value[:100])
if contentType:
ct = contentType[0]
self.request.form['form.contentType'] = ct
context.contentType = ct
setattr(context, fieldInstance.name, value)
context.localFilename = filename
@property
def fieldHandlers(self):
return dict(fileupload=self.handleFileUpload)
def xupdateFields(self): # obsolete
obj = self.object
errors = {}
form = self.request.form
ti = IType(obj).typeInterface
if ti is not None:
adapted = ti(obj)
else:
adapted = obj
for k in form.keys():
# TODO: use self.view.form_fields or better: IInstance(adapted)
if k.startswith(self.prefix):
fn = k[len(self.prefix):]
if fn in ('action', 'type', 'data.used') or fn.endswith('-empty-marker'):
continue
value = form[k]
if fn.startswith(self.conceptPrefix) and value:
self.collectConcepts(fn[len(self.conceptPrefix):], value)
else:
if not value and fn == 'data' and IFile.providedBy(adapted):
# empty file data - don't change
continue
if isinstance(value, FileUpload):
filename = getattr(value, 'filename', '')
value = value.read()
if filename:
#self.request.form['filename'] = filename
contentType = guess_content_type(filename, value[:100])
if contentType:
ct = contentType[0]
self.request.form['form.contentType'] = ct
adapted.contentType = ct
adapted.localFilename = filename
if fn == 'title' and not value:
# TODO: provide general validation mechanism
errors[fn] = 'Field %s must not be empty' % fn
else:
# TODO: provide unmarshalling depending on field type
setattr(adapted, fn, value)
if self.old or self.selected:
self.assignConcepts(obj)
notify(ObjectModifiedEvent(obj))
return errors
def collectConcepts(self, fieldName, value):
if self.old is None: self.old = []
if self.selected is None: self.selected = []
for v in value:
if fieldName == 'old':
self.old.append(v)
elif fieldName == 'selected' and v not in self.selected:
self.selected.append(v)
def assignConcepts(self, obj):
for v in self.old:
if v not in self.selected:
c, p = v.split(':')
concept = util.getObjectForUid(c)
predicate = util.getObjectForUid(p)
obj.deassignConcept(concept, [predicate])
for v in self.selected:
if v != 'none' and v not in self.old:
c, p = v.split(':')
concept = util.getObjectForUid(c)
predicate = util.getObjectForUid(p)
exists = obj.getConceptRelations(predicates=[p], concept=concept)
if not exists:
obj.assignConcept(concept, predicate)
def checkCreateVersion(self, obj):
form = self.request.form
if form.get('version.create'):
versionable = IVersionable(obj)
level = int(form.get('version.level', 1))
version = versionable.createVersion(level)
notify(ObjectCreatedEvent(version))
return version
return obj
class CreateObject(EditObject):
def update(self):
form = self.request.form
container = self.loopsRoot.getResourceManager()
title = form.get('title')
if not title:
raise BadRequest('Title field is empty')
obj = Resource(title)
data = form.get('data')
if data and isinstance(data, FileUpload):
name = getattr(data, 'filename', None)
# strip path from IE uploads:
if '\\' in name:
name = name.rsplit('\\', 1)[-1]
else:
name = None
name = INameChooser(container).chooseName(name, obj)
container[name] = obj
tc = form.get('form.type') or '.loops/concepts/note'
obj.resourceType = self.loopsRoot.loopsTraverse(tc)
notify(ObjectCreatedEvent(obj))
self.object = obj
self.updateFields()
self.request.response.redirect(self.view.virtualTargetUrl)
return False