loops/view.py
helmutm 24fb20d78a provide external editor usage for files, e.g. word documents
git-svn-id: svn://svn.cy55.de/Zope3/src/loops/trunk@1589 fd906abe-77d9-0310-91a1-e0d9ade77398
2007-02-19 13:00:20 +00:00

216 lines
7.2 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 interfaces import IView, INode
from interfaces import IViewManager, INodeContained
from interfaces import ILoopsContained
from interfaces import ITargetRelation
from interfaces import IConcept
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:
# 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
return self.context
return super(NodeTraverser, self).publishTraverse(request, name)