
git-svn-id: svn://svn.cy55.de/Zope3/src/loops/trunk@2219 fd906abe-77d9-0310-91a1-e0d9ade77398
432 lines
13 KiB
Python
432 lines
13 KiB
Python
#
|
|
# Copyright (c) 2006 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
|
|
#
|
|
|
|
"""
|
|
Common base class for loops browser view classes.
|
|
|
|
$Id$
|
|
"""
|
|
|
|
from zope.app import zapi
|
|
from zope import component
|
|
from zope.app.form.browser.interfaces import ITerms
|
|
from zope.app.i18n.interfaces import ITranslationDomain
|
|
from zope.app.security.interfaces import IAuthentication
|
|
from zope.app.pagetemplate import ViewPageTemplateFile
|
|
from zope.cachedescriptors.property import Lazy
|
|
from zope.dottedname.resolve import resolve
|
|
from zope.dublincore.interfaces import IZopeDublinCore
|
|
from zope.formlib import form
|
|
from zope.formlib.form import FormFields
|
|
from zope.formlib.namedtemplate import NamedTemplate
|
|
from zope.interface import Interface, implements
|
|
from zope.publisher.browser import applySkin
|
|
#from zope.publisher.interfaces.browser import ISkin
|
|
from zope.publisher.interfaces.browser import IBrowserSkinType
|
|
from zope import schema
|
|
from zope.schema.vocabulary import SimpleTerm
|
|
from zope.security import canAccess, canWrite, checkPermission
|
|
from zope.security.interfaces import ForbiddenAttribute, Unauthorized
|
|
from zope.security.proxy import removeSecurityProxy
|
|
from zope.traversing.browser import absoluteURL
|
|
from zope.traversing.api import getName
|
|
|
|
from cybertools.browser.view import GenericView
|
|
from cybertools.relation.interfaces import IRelationRegistry
|
|
from cybertools.text import mimetypes
|
|
from cybertools.typology.interfaces import IType, ITypeManager
|
|
from loops.common import adapted
|
|
from loops.interfaces import IView
|
|
from loops.resource import Resource
|
|
from loops.type import ITypeConcept
|
|
from loops import util
|
|
from loops.util import _
|
|
from loops.versioning.interfaces import IVersionable
|
|
|
|
|
|
concept_macros = ViewPageTemplateFile('concept_macros.pt')
|
|
conceptMacrosTemplate = concept_macros # "old" name
|
|
|
|
|
|
class NameField(schema.ASCIILine):
|
|
|
|
def _validate(self, value):
|
|
super(NameField, self)._validate(value)
|
|
|
|
|
|
class IAddForm(Interface):
|
|
|
|
name = NameField(
|
|
title=_(u'Object name'),
|
|
description=_(u'Name of the object - will be used for addressing the '
|
|
'object via a URL; should therefore be unique within '
|
|
'the container and not contain special characters')
|
|
)
|
|
|
|
|
|
class AddForm(form.AddForm):
|
|
|
|
form_fields = FormFields(IAddForm)
|
|
template = NamedTemplate('loops.pageform')
|
|
|
|
|
|
class EditForm(form.EditForm):
|
|
|
|
template = NamedTemplate('loops.pageform')
|
|
|
|
def deleteObjectAction(self):
|
|
return None # better not to show the edit button at the moment
|
|
parent = zapi.getParent(self.context)
|
|
parentUrl = absoluteURL(parent, self.request)
|
|
return parentUrl + '/contents.html'
|
|
|
|
|
|
class BaseView(GenericView):
|
|
|
|
actions = {} # default only, don't update
|
|
|
|
def __init__(self, context, request):
|
|
super(BaseView, self).__init__(context, request)
|
|
# TODO: get rid of removeSecurityProxy() call
|
|
self.context = removeSecurityProxy(context)
|
|
self.setSkin(self.loopsRoot.skinName)
|
|
try:
|
|
if not canAccess(context, 'title'):
|
|
raise Unauthorized
|
|
#request.response.redirect('login.html')
|
|
except ForbiddenAttribute: # ignore when testing
|
|
pass
|
|
|
|
@Lazy
|
|
def target(self):
|
|
# allow for having a separate object the view acts upon
|
|
return self.context
|
|
|
|
def setSkin(self, skinName):
|
|
skin = None
|
|
if skinName and IView.providedBy(self.context):
|
|
skin = component.queryUtility(IBrowserSkinType, skinName)
|
|
if skin:
|
|
applySkin(self.request, skin)
|
|
self.skin = skin
|
|
|
|
@Lazy
|
|
def modified(self):
|
|
""" get date/time of last modification
|
|
"""
|
|
dc = IZopeDublinCore(self.context)
|
|
d = dc.modified or dc.created
|
|
return d and d.strftime('%Y-%m-%d %H:%M') or ''
|
|
|
|
@Lazy
|
|
def creators(self):
|
|
cr = IZopeDublinCore(self.context).creators or []
|
|
pau = component.getUtility(IAuthentication)
|
|
creators = []
|
|
for c in cr:
|
|
principal = pau.getPrincipal(c)
|
|
if principal is not None:
|
|
creators.append(principal.title)
|
|
return ', '.join(creators)
|
|
|
|
@Lazy
|
|
def loopsRoot(self):
|
|
return self.context.getLoopsRoot()
|
|
|
|
@Lazy
|
|
def url(self):
|
|
return absoluteURL(self.context, self.request)
|
|
|
|
@Lazy
|
|
def rootUrl(self):
|
|
return absoluteURL(self.loopsRoot, self.request)
|
|
|
|
@Lazy
|
|
def view(self):
|
|
return self
|
|
|
|
@Lazy
|
|
def token(self):
|
|
return self.loopsRoot.getLoopsUri(self.context)
|
|
|
|
@Lazy
|
|
def title(self):
|
|
return self.context.title or getName(self.context)
|
|
|
|
@Lazy
|
|
def description(self):
|
|
return adapted(self.context).description
|
|
|
|
@Lazy
|
|
def dcTitle(self):
|
|
return IZopeDublinCore(self.context).title or self.title
|
|
|
|
@Lazy
|
|
def headTitle(self):
|
|
return self.dcTitle
|
|
|
|
@Lazy
|
|
def value(self):
|
|
return self.context
|
|
|
|
@Lazy
|
|
def uniqueId(self):
|
|
return util.getUidForObject(self.context)
|
|
|
|
# type stuff
|
|
|
|
@Lazy
|
|
def type(self):
|
|
return IType(self.context)
|
|
|
|
@Lazy
|
|
def typeProvider(self):
|
|
return self.type.typeProvider
|
|
|
|
@Lazy
|
|
def typeInterface(self):
|
|
return self.type.typeInterface
|
|
|
|
@Lazy
|
|
def typeAdapter(self):
|
|
ifc = self.typeInterface
|
|
if ifc is not None:
|
|
return ifc(self.context)
|
|
|
|
@Lazy
|
|
def typeTitle(self):
|
|
return self.type.title
|
|
|
|
@Lazy
|
|
def longTypeTitle(self):
|
|
t = self.typeTitle
|
|
ct = getattr(self.context, 'contentType')
|
|
if ct:
|
|
ext = mimetypes.extensions.get(ct)
|
|
if ext:
|
|
#return '%s (%s)' % (t, ext.upper())
|
|
return ext.upper()
|
|
return t
|
|
|
|
@Lazy
|
|
def typeUrl(self):
|
|
provider = self.typeProvider
|
|
if provider is not None:
|
|
return absoluteURL(provider, self.request)
|
|
return None
|
|
|
|
def viewIterator(self, objs):
|
|
request = self.request
|
|
for o in objs:
|
|
yield BaseView(o, request)
|
|
|
|
# type listings
|
|
|
|
def listTypes(self, include=None, exclude=None, sortOn='title'):
|
|
types = [dict(token=t.token, title=t.title)
|
|
for t in ITypeManager(self.context).listTypes(include, exclude)]
|
|
#if sortOn:
|
|
# types.sort(key=lambda x: x[sortOn])
|
|
return types
|
|
|
|
def getTypesVocabulary(self, include=None):
|
|
return util.KeywordVocabulary(self.listTypes(include, ('hidden',)))
|
|
|
|
def resourceTypes(self):
|
|
return util.KeywordVocabulary(self.listTypes(('resource',), ('hidden',)))
|
|
#if t.factory == Resource]) # ? if necessary -> type.qualifiers
|
|
|
|
def conceptTypes(self):
|
|
return util.KeywordVocabulary(self.listTypes(('concept',), ('hidden',)))
|
|
|
|
def listTypesForSearch(self, include=None, exclude=None, sortOn='title'):
|
|
types = [dict(token=t.tokenForSearch, title=t.title)
|
|
for t in ITypeManager(self.context).listTypes(include, exclude)]
|
|
if sortOn:
|
|
types.sort(key=lambda x: x[sortOn])
|
|
return types
|
|
|
|
def typesForSearch(self):
|
|
general = [('loops:resource:*', 'Any Resource'),
|
|
('loops:concept:*', 'Any Concept'),]
|
|
return util.KeywordVocabulary(general
|
|
+ self.listTypesForSearch(exclude=('system', 'hidden',))
|
|
+ [('loops:*', 'Any')])
|
|
|
|
def conceptTypesForSearch(self):
|
|
general = [('loops:concept:*', 'Any Concept'),]
|
|
return util.KeywordVocabulary(general
|
|
+ self.listTypesForSearch(('concept',), ('system', 'hidden',),))
|
|
|
|
def resourceTypesForSearch(self):
|
|
general = [('loops:resource:*', 'Any Resource'),]
|
|
return util.KeywordVocabulary(general
|
|
+ self.listTypesForSearch(('resource',), ('system', 'hidden'),))
|
|
|
|
# versioning
|
|
|
|
@Lazy
|
|
def useVersioning(self):
|
|
if 'useVersioning' in self.loopsRoot.options:
|
|
return True
|
|
options = getattr(self.controller, 'options', None)
|
|
if options:
|
|
return 'useVersioning' in options.value
|
|
|
|
@Lazy
|
|
def versionId(self):
|
|
#versionable = IVersionable(self.context, None)
|
|
versionable = IVersionable(self.target, None)
|
|
return versionable and versionable.versionId or ''
|
|
|
|
@Lazy
|
|
def currentVersionId(self):
|
|
#versionable = IVersionable(self.context, None)
|
|
versionable = IVersionable(self.target, None)
|
|
return versionable and versionable.currentVersion.versionId or ''
|
|
|
|
@Lazy
|
|
def hasVersions(self):
|
|
#versionable = IVersionable(self.context, None)
|
|
versionable = IVersionable(self.target, None)
|
|
return versionable and len(versionable.versions) > 1 or False
|
|
|
|
@Lazy
|
|
def versionInfo(self):
|
|
if not self.useVersioning:
|
|
return None
|
|
#context = self.context
|
|
#versionable = IVersionable(context, None)
|
|
target = self.target
|
|
versionable = IVersionable(target, None)
|
|
if versionable is None:
|
|
return ''
|
|
versionId = versionable.versionId
|
|
td = component.getUtility(ITranslationDomain, _._domain)
|
|
current = ((versionable.currentVersion == target)
|
|
and td.translate(_(u'current'), context=self.request)
|
|
or u'')
|
|
released = ((versionable.releasedVersion == target)
|
|
and td.translate(_(u'released'), context=self.request)
|
|
or u'')
|
|
if not current and not released:
|
|
return versionId
|
|
addInfo = u', '.join(e for e in (current, released) if e)
|
|
return u'%s (%s)' % (versionId, addInfo)
|
|
|
|
# controlling editing
|
|
|
|
@Lazy
|
|
def editable(self):
|
|
return canWrite(self.context, 'title')
|
|
|
|
def getActions(self, category='object', page=None):
|
|
""" Return a list of actions that provide the view and edit actions
|
|
available for the context object.
|
|
"""
|
|
actions = []
|
|
if category in self.actions:
|
|
actions.extend(self.actions[category](self, page=page))
|
|
return actions
|
|
|
|
def openEditWindow(self, viewName='edit.html'):
|
|
if self.editable:
|
|
if checkPermission('loops.ManageSite', self.context):
|
|
return "openEditWindow('%s/@@%s')" % (self.url, viewName)
|
|
return ''
|
|
|
|
@Lazy
|
|
def xeditable(self):
|
|
ct = getattr(self.context, 'contentType', '')
|
|
if not ct or ct == 'application/pdf':
|
|
return False
|
|
if ct.startswith('text/') and ct != 'text/rtf':
|
|
return checkPermission('loops.ManageSite', self.context)
|
|
return canWrite(self.context, 'title')
|
|
|
|
@Lazy
|
|
def inlineEditingActive(self):
|
|
# this may depend on system and user settings...
|
|
return True
|
|
|
|
@Lazy
|
|
def conceptMapEditorUrl(self):
|
|
return (checkPermission('loops.xmlrpc.ManageConcepts', self.context)
|
|
and self.rootUrl + '/swf.html'
|
|
or None)
|
|
|
|
inlineEditable = False
|
|
|
|
def inlineEdit(self, id):
|
|
self.registerDojo()
|
|
return 'return inlineEdit("%s", "")' % id
|
|
|
|
def registerDojo(self):
|
|
cm = self.controller.macros
|
|
cm.register('js', 'dojo.js', resourceName='ajax.dojo/dojo.js')
|
|
#cm.register('js', 'dojo.js', resourceName='ajax.dojo1/dojo/dojo.js')
|
|
|
|
|
|
# vocabulary stuff
|
|
|
|
class LoopsTerms(object):
|
|
""" Provide the ITerms interface, e.g. for usage in selection
|
|
lists.
|
|
"""
|
|
|
|
implements(ITerms)
|
|
|
|
def __init__(self, source, request):
|
|
# the source parameter is a view or adapter of a real context object:
|
|
self.source = source
|
|
self.context = source.context
|
|
self.request = request
|
|
|
|
@Lazy
|
|
def loopsRoot(self):
|
|
return self.context.getLoopsRoot()
|
|
|
|
def getTerm(self, value):
|
|
#if value is None:
|
|
# return SimpleTerm(None, '', u'not assigned')
|
|
title = value.title or zapi.getName(value)
|
|
token = self.loopsRoot.getLoopsUri(value)
|
|
return SimpleTerm(value, token, title)
|
|
|
|
def getValue(self, token):
|
|
return self.loopsRoot.loopsTraverse(token)
|
|
|
|
|
|
class InterfaceTerms(object):
|
|
""" Provide the ITerms interface for source list of interfaces.
|
|
"""
|
|
|
|
implements(ITerms)
|
|
|
|
def __init__(self, source, request):
|
|
self.source = source
|
|
self.request = request
|
|
|
|
def getTerm(self, value):
|
|
token = '.'.join((value.__module__, value.__name__))
|
|
return SimpleTerm(value, token, token)
|
|
|
|
def getValue(self, token):
|
|
return resolve(token)
|
|
|
|
|