
git-svn-id: svn://svn.cy55.de/Zope3/src/loops/trunk@1654 fd906abe-77d9-0310-91a1-e0d9ade77398
220 lines
7.5 KiB
Python
220 lines
7.5 KiB
Python
#
|
|
# Copyright (c) 2005 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
|
|
#
|
|
|
|
"""
|
|
Definition of the View and related classses.
|
|
|
|
$Id$
|
|
"""
|
|
|
|
from zope import component
|
|
from zope.app import zapi
|
|
from zope.app.container.btree import BTreeContainer
|
|
from zope.app.container.contained import Contained
|
|
from zope.app.container.ordered import OrderedContainer
|
|
from zope.app.container.traversal import ContainerTraverser, ItemTraverser
|
|
from zope.app.container.traversal import ContainerTraversable
|
|
from zope.app.intid.interfaces import IIntIds
|
|
from zope.cachedescriptors.property import Lazy, readproperty
|
|
from zope.component import adapts
|
|
from zope.interface import implements
|
|
from zope.interface import alsoProvides, directlyProvides, directlyProvidedBy
|
|
from zope.security.proxy import removeSecurityProxy
|
|
from persistent import Persistent
|
|
from cybertools.relation import DyadicRelation
|
|
from cybertools.relation.registry import getRelations
|
|
from cybertools.relation.interfaces import IRelationRegistry, IRelatable
|
|
|
|
from loops.interfaces import IView, INode
|
|
from loops.interfaces import IViewManager, INodeContained
|
|
from loops.interfaces import ILoopsContained
|
|
from loops.interfaces import ITargetRelation
|
|
from loops.interfaces import IConcept
|
|
from loops.versioning.versioninfo import getVersionInfo
|
|
|
|
|
|
class View(object):
|
|
|
|
implements(IView, INodeContained, IRelatable)
|
|
|
|
_title = u''
|
|
def getTitle(self): return self._title
|
|
def setTitle(self, title): self._title = title
|
|
title = property(getTitle, setTitle)
|
|
|
|
_description = u''
|
|
def getDescription(self): return self._description
|
|
def setDescription(self, description): self._description = description
|
|
description = property(getDescription, setDescription)
|
|
|
|
_viewName = u''
|
|
def getViewName(self): return self._viewName or getattr(self, '_viewer', u'')
|
|
def setViewName(self, viewName): self._viewName = viewName
|
|
viewName = property(getViewName, setViewName)
|
|
|
|
viewer = property(getViewName, setViewName) # BBB
|
|
|
|
def getTarget(self):
|
|
rels = getRelations(first=self, relationships=[TargetRelation])
|
|
if len(rels) == 0:
|
|
return None
|
|
if len(rels) > 1:
|
|
raise ValueError('There may be only one target for a View object: %s - %s'
|
|
% (zapi.getName(self), `[zapi.getName(r.second) for r in rels]`))
|
|
return list(rels)[0].second
|
|
|
|
def setTarget(self, target):
|
|
registry = zapi.getUtility(IRelationRegistry)
|
|
rels = list(registry.query(first=self, relationship=TargetRelation))
|
|
if len(rels) > 0:
|
|
oldRel = rels[0]
|
|
if oldRel.second == target:
|
|
return
|
|
else:
|
|
registry.unregister(oldRel)
|
|
if target:
|
|
targetSchema = target.proxyInterface
|
|
rel = TargetRelation(self, target)
|
|
registry.register(rel)
|
|
alsoProvides(self, targetSchema)
|
|
|
|
target = property(getTarget, setTarget)
|
|
|
|
def __init__(self, title=u'', description=u''):
|
|
self.title = title
|
|
self.description = description
|
|
super(View, self).__init__()
|
|
|
|
def getLoopsRoot(self):
|
|
return zapi.getParent(self).getLoopsRoot()
|
|
|
|
|
|
class Node(View, OrderedContainer):
|
|
|
|
implements(INode)
|
|
|
|
_nodeType = 'info'
|
|
def getNodeType(self): return self._nodeType
|
|
def setNodeType(self, nodeType): self._nodeType = nodeType
|
|
nodeType = property(getNodeType, setNodeType)
|
|
|
|
_body = u''
|
|
def getBody(self): return self._body
|
|
def setBody(self, body): self._body = body
|
|
body = property(getBody, setBody)
|
|
|
|
contentType = u'zope.source.rest'
|
|
|
|
def getParentNode(self, nodeTypes=None):
|
|
parent = zapi.getParent(self)
|
|
while INode.providedBy(parent):
|
|
if not nodeTypes or parent.nodeType in nodeTypes:
|
|
return parent
|
|
parent = zapi.getParent(parent)
|
|
return None
|
|
|
|
def getChildNodes(self, nodeTypes=None):
|
|
for item in self.values():
|
|
if INode.providedBy(item) \
|
|
and (not nodeTypes or item.nodeType in nodeTypes):
|
|
yield item
|
|
else:
|
|
continue
|
|
|
|
def getMenu(self):
|
|
if self.nodeType == 'menu':
|
|
return self
|
|
return self.getParentNode(['menu'])
|
|
|
|
def getPage(self):
|
|
pageTypes = ['page', 'menu', 'info']
|
|
if self.nodeType in pageTypes:
|
|
return self
|
|
return self.getParentNode(pageTypes)
|
|
|
|
def getMenuItems(self):
|
|
return self.getChildNodes(['page', 'menu'])
|
|
|
|
def getPageItems(self):
|
|
return self.getChildNodes(['page', 'menu', 'info'])
|
|
|
|
def getTextItems(self):
|
|
return self.getChildNodes(['text'])
|
|
|
|
def isMenuItem(self):
|
|
return self.nodeType in ('page', 'menu')
|
|
|
|
|
|
class ViewManager(OrderedContainer):
|
|
|
|
implements(IViewManager, ILoopsContained)
|
|
|
|
def getLoopsRoot(self):
|
|
return zapi.getParent(self)
|
|
|
|
def getViewManager(self):
|
|
return self
|
|
|
|
|
|
class TargetRelation(DyadicRelation):
|
|
""" A relation between a view and its target.
|
|
"""
|
|
implements(ITargetRelation)
|
|
|
|
|
|
# adapters
|
|
|
|
class NodeTraverser(ItemTraverser):
|
|
|
|
adapts(INode)
|
|
|
|
def publishTraverse(self, request, name):
|
|
if name == '.loops':
|
|
return self.context.getLoopsRoot()
|
|
if name.startswith('.target'):
|
|
traversalStack = request._traversal_stack
|
|
while traversalStack and traversalStack[0].startswith('.target'):
|
|
# skip obsolete target references in the url
|
|
name = traversalStack.pop(0)
|
|
traversedNames = request._traversed_names
|
|
if traversedNames:
|
|
lastTraversed = traversedNames[-1]
|
|
if lastTraversed.startswith('.target') and lastTraversed != name:
|
|
# let <base .../> tag show the current object
|
|
traversedNames[-1] = name
|
|
if len(name) > len('.target'):
|
|
uid = int(name[len('.target'):])
|
|
target = component.getUtility(IIntIds).getObject(uid)
|
|
else:
|
|
target = self.context.target
|
|
if target is not None:
|
|
# provide versioning info and switch to correct version if appropriate
|
|
target, versionInfo = getVersionInfo(target, request)
|
|
# remember self.context in request
|
|
viewAnnotations = request.annotations.setdefault('loops.view', {})
|
|
viewAnnotations['node'] = self.context
|
|
if request.method == 'PUT':
|
|
# we have to use the target object directly
|
|
return target
|
|
else:
|
|
# we'll use the target object in the node's context
|
|
viewAnnotations['target'] = target
|
|
viewAnnotations['versionInfo'] = versionInfo
|
|
return self.context
|
|
return super(NodeTraverser, self).publishTraverse(request, name)
|
|
|