# # Copyright (c) 2017 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 # """ View class for Node objects. """ from logging import getLogger import urllib from urlparse import urlparse, urlunparse from zope import component, interface, schema from zope.cachedescriptors.property import Lazy from zope.annotation.interfaces import IAnnotations from zope.app.catalog.interfaces import ICatalog from zope.app.container.browser.contents import JustContents from zope.app.container.browser.adding import Adding from zope.app.container.traversal import ItemTraverser from zope.app.pagetemplate import ViewPageTemplateFile from zope.app.security.interfaces import IUnauthenticatedPrincipal from zope.dottedname.resolve import resolve from zope.event import notify from zope.lifecycleevent import ObjectCreatedEvent, ObjectModifiedEvent from zope.lifecycleevent import Attributes from zope.formlib.form import Form, FormFields from zope.proxy import removeAllProxies from zope.publisher.defaultview import getDefaultViewName from zope.publisher.interfaces import NotFound from zope.security import canAccess, canWrite, checkPermission from zope.security.proxy import removeSecurityProxy from zope.traversing.api import getParent, getParents, getPath from zope.traversing.browser import absoluteURL from cybertools.ajax import innerHtml from cybertools.browser import configurator from cybertools.browser.action import Action from cybertools.browser.view import GenericView from cybertools.stateful.interfaces import IStateful from cybertools.typology.interfaces import IType, ITypeManager from cybertools.util.jeep import Jeep from cybertools.xedit.browser import ExternalEditorView from loops.browser.action import actions, DialogAction from loops.browser.common import BaseView from loops.browser.concept import ConceptView from loops.common import adapted, AdapterBase, baseObject from loops.i18n.browser import i18n_macros, LanguageInfo from loops.interfaces import IConcept, IResource, IDocument, IMediaAsset, INode from loops.interfaces import IViewConfiguratorSchema from loops.organize.interfaces import IPresence from loops.organize.tracking import access from loops.resource import MediaAsset from loops.security.common import canWriteObject from loops import util from loops.util import _ from loops.versioning.util import getVersion logger = getLogger('loops.browser.node') node_macros = ViewPageTemplateFile('node_macros.pt') info_macros = ViewPageTemplateFile('info.pt') calendar_macros = ViewPageTemplateFile('calendar.pt') class NodeView(BaseView): _itemNum = 0 template = node_macros nextUrl = None setupTarget = True def __init__(self, context, request): super(NodeView, self).__init__(context, request) self.viewAnnotations.setdefault('nodeView', self) self.viewAnnotations.setdefault('node', self.context) self.setSkin(self.viewConfig.get('skinName')) def __call__(self, *args, **kw): if self.nodeType == 'raw': vn = self.context.viewName if vn: self.request.response.setHeader('content-type', vn) return self.context.body tv = self.viewAnnotations.get('targetView') if tv is not None: if tv.isToplevel: return tv(*args, **kw) if self.controller is not None: self.controller.setMainPage() return super(NodeView, self).__call__(*args, **kw) @Lazy def viewConfig(self): return getViewConfiguration(self.context, self.request) @Lazy def viewConfigOptions(self): result = {} for opt in self.viewConfig.get('options') or []: if ':' in opt: k, v = opt.split(':', 1) result[k] = v.split(',') else: result[opt] = True return result @Lazy def copyright(self): cr = self.viewConfigOptions.get('copyright') if cr: return cr[0] cr = self.globalOptions('copyright') return cr and cr[0] or 'cyberconcepts.org team' @Lazy def macro(self): return self.template.macros['content'] @Lazy def subparts(self): def getParts(n): t = n.targetObjectView if t is None: return [] return t.subparts parts = getParts(self) #return parts for n in self.textItems: parts.extend(getParts(n)) return parts def update(self, topLevel=True): if topLevel and self.view != self: return self.view.update(False) result = super(NodeView, self).update() self.recordAccess() return result @Lazy def title(self): return self.context.title or getName(self.context) def breadcrumbs(self): if not self.globalOptions('showBreadcrumbs'): return [] menu = self.menu data = [dict(label=menu.title, url=menu.url)] menuItem = self.getNearestMenuItem(all=True) if menuItem != menu.context: data.append(dict(label=menuItem.title, url=absoluteURL(menuItem, self.request))) for p in getParents(menuItem): if p == menu.context: break data.insert(1, dict(label=p.title, url=absoluteURL(p, self.request))) if self.virtualTarget: data.extend(self.virtualTarget.breadcrumbs()) if data and not '?' in data[-1]['url']: if self.urlParamString: data[-1]['url'] += self.urlParamString return data def viewModes(self): if self.virtualTarget: return self.virtualTarget.viewModes() return Jeep() def recordAccess(self, viewName=''): target = self.virtualTargetObject targetUid = target is not None and util.getUidForObject(target) or '' access.record(self.request, principal=self.principalId, node=self.uniqueId, target=targetUid, view=viewName) def setupController(self): cm = self.controller.macros cm.register('css', identifier='loops.css', resourceName='loops.css', media='all', priority=60) cm.register('js', 'loops.js', resourceName='loops.js', priority=60) cm.register('top_actions', 'top_actions', name='multi_actions', subMacros=[node_macros.macros['page_actions'], i18n_macros.macros['language_switch']]) if self.globalOptions('expert.quicksearch'): from loops.expert.browser.search import search_template cm.register('top_actions', 'top_quicksearch', name='multi_actions', subMacros=[search_template.macros['quicksearch']], priority=20) cm.register('portlet_left', 'navigation', title='Navigation', subMacro=node_macros.macros['menu']) if canWriteObject(self.context) or ( # TODO: is this useful in any case? self.virtualTargetObject is not None and canWriteObject(self.virtualTargetObject)): # check if there are any available actions; # store list of actions in macro object (evaluate only once) actions = [act for act in self.getAllowedActions('portlet', target=self.virtualTarget) if act.condition] if actions: cm.register('portlet_right', 'actions', title=_(u'Actions'), subMacro=node_macros.macros['actions'], priority=100, actions=actions) if self.isAnonymous and self.globalOptions('provideLogin'): cm.register('portlet_right', 'login', title=_(u'Not logged in'), subMacro=node_macros.macros['login'], icon='cybertools.icons/user.png', priority=10) if not self.isAnonymous: mi = self.controller.memberInfo title = mi.title.value or _(u'Personal Informations') url=None obj = mi.get('object') if obj is not None: query = self.conceptManager.get('personal_info') if query is None: #url = self.url + '/personal_info.html' url = self.getUrlForTarget(obj.value) else: url = self.getUrlForTarget(query) cm.register('portlet_right', 'personal', title=title, subMacro=node_macros.macros['personal'], icon='cybertools.icons/user.png', url=url, priority=10) if self.globalOptions('organize.showPresence'): cm.register('portlet_right', 'presence', title=_(u'Presence'), subMacro=node_macros.macros['presence'], icon='cybertools.icons/group.png', priority=11) if self.globalOptions('organize.showCalendar'): cm.register('portlet_left', 'calendar', title=_(u'Calendar'), subMacro=calendar_macros.macros['main'], priority=90) # force early portlet registrations by target by setting up target view if self.virtualTarget is not None: std = self.virtualTarget.typeOptions('portlet_states') if std: from loops.organize.stateful.browser import registerStatesPortlet registerStatesPortlet(self.controller, self.virtualTarget, std) @Lazy def usersPresent(self): presence = component.getUtility(IPresence) presence.update(self.request.principal.id) data = presence.getPresentUsers(self.context) for u in data: if IConcept.providedBy(u): url = self.getUrlForTarget(u) else: url = None yield dict(title=u.title, url=url) @Lazy def view(self): name = self.request.get('loops.viewName', '') or self.context.viewName if name and '?' in name: name, params = name.split('?', 1) ann = self.viewAnnotations ann['params'] = params if name: adapter = component.queryMultiAdapter( (self.context, self.request), name=name) if adapter is not None: return adapter return self @Lazy def item(self): tv = self.viewAnnotations.get('targetView') if tv is not None: return tv viewName = self.request.get('loops.viewName') or '' # was there a .target... element in the URL? #target = self.virtualTargetObject # ignores page even for direktly assignd target target = self.viewAnnotations.get('target') if target is not None: basicView = component.getMultiAdapter((target, self.request), name=viewName) # xxx: obsolete when self.targetObject is virtual target: if hasattr(basicView, 'view'): #basicView.setupController() return basicView.view return self.page @Lazy def targetItem(self): tv = self.viewAnnotations.get('targetView') if tv is not None: return tv viewName = self.request.get('loops.viewName') or '' target = self.virtualTargetObject if target is not None: basicView = component.getMultiAdapter((target, self.request), name=viewName) # xxx: obsolete when self.targetObject is virtual target: if hasattr(basicView, 'view'): #basicView.setupController() return basicView.view return self.page @Lazy def page(self): page = self.context.getPage() return page is not None and NodeView(page, self.request).view or None @Lazy def textItems(self): return [NodeView(child, self.request) for child in self.context.getTextItems()] @Lazy def pageItems(self): return [NodeView(child, self.request) for child in self.context.getPageItems()] @property def itemNum(self): self._itemNum += 1 return self._itemNum @Lazy def nodeType(self): return self.context.nodeType def render(self, text=None): if text is None: text = self.context.body if not text: return u'' if text.startswith('<'): # seems to be HTML return text source = component.createObject(self.context.contentType, text) view = component.getMultiAdapter((removeAllProxies(source), self.request)) return view.render() @Lazy def targetObject(self): # xxx: use virtualTargetObject #return self.virtualTargetObject target = self.context.target if target is not None: target = getVersion(target, self.request) return target @Lazy def targetObjectView(self): obj = self.targetObject if obj is not None: basicView = component.getMultiAdapter((obj, self.request)) basicView._viewName = self.context.viewName if self.context.nodeType != 'text': basicView.setupController() return basicView.view @Lazy def targetUrl(self): t = self.targetObjectView if t is not None: return self.makeTargetUrl(self.url, t.uniqueId, t.title) return '' def renderTarget(self): target = self.targetObjectView return target is not None and target.render() or u'' @Lazy def body(self): return self.render() @Lazy def bodyMacro(self): # TODO: replace by something like: return self.target.macroName target = self.targetObject if (target is None or IDocument.providedBy(target) or (IResource.providedBy(target) and target.contentType.startswith('text/'))): return 'textbody' if IConcept.providedBy(target): return 'conceptbody' if IResource.providedBy(target) and target.contentType.startswith('image/'): return 'imagebody' return 'filebody' @Lazy def editable(self): return canWrite(self.context, 'body') def hasTopPage(self, name): page = self.topMenu.context.get(name) return page is not None # menu stuff @Lazy def menuObject(self): return self.context.getMenu() @Lazy def menu(self): menu = self.menuObject return menu is not None and NodeView(menu, self.request) or None @Lazy def topMenu(self): menu = self.menuObject parentMenu = None while menu is not None: parent = getParent(menu) if INode.providedBy(parent): parentMenu = parent.getMenu() if parentMenu is None or parentMenu is menu: return NodeView(menu, self.request) menu = parentMenu return menu is not None and NodeView(menu, self.request) or None @Lazy def headTitle(self): parts = [] menuObject = self.menuObject if menuObject is not None and (menuObject != self.context or self.virtualTarget): parts.append(super(NodeView, self.menu).headTitle) if self.virtualTarget: ht = self.virtualTarget.headTitle if ht not in parts: parts.append(ht) if len(parts) < 2: ht = super(NodeView, self).headTitle if ht not in parts: parts.append(ht) if self.globalOptions('reverseHeadTitle'): parts.reverse() return ' - ' .join(parts) @Lazy def menuItems(self): items = [NodeView(child, self.request).view for child in self.context.getMenuItems()] return [item for item in items if item.isVisible] @Lazy def parents(self): return getParents(self.context) @Lazy def nearestMenuItem(self): return self.getNearestMenuItem() def getNearestMenuItem(self, all=False): menu = self.menuObject menuItem = None for p in [self.context] + self.parents: if not all and not p.isMenuItem(): menuItem = None elif menuItem is None: menuItem = p if p == menu: return menuItem return None def selected(self, item): return item.context == self.nearestMenuItem def active(self, item): return item.context == self.context or item.context in self.parents @Lazy def logoutUrl(self): nextUrl = urllib.urlencode(dict(nextUrl=self.menu.url)) return '%s/logout.html?%s' % (self.menu.url, nextUrl) @Lazy def authenticationMethod(self): return self.viewAnnotations.get('auth_method') or 'standard' # virtual target support @Lazy def virtualTargetObject(self): target = self.viewAnnotations.get('target') if target is None: target = self.context.target if target is not None: target = getVersion(target, self.request) return target target = virtualTargetObject @Lazy def targetUid(self): if self.virtualTargetObject: return util.getUidForObject(self.virtualTargetObject) else: return None def targetView(self, name='index.html', methodName='show'): if name == 'index.html': # only when called for default view tv = self.viewAnnotations.get('targetView') if tv is not None and callable(tv): return tv() if '?' in name: name, params = name.split('?', 1) target = self.virtualTargetObject if target is not None: if isinstance(target, AdapterBase): target = target.context targetView = component.queryMultiAdapter( (adapted(target), self.request), name=name) if targetView is None: targetView = component.getMultiAdapter( (target, self.request), name=name) if name == 'index.html' and hasattr(targetView, 'show'): return targetView.show() method = getattr(targetView, methodName, None) if method: return method() return targetView() return u'' def targetDefaultView(self): target = self.virtualTargetObject if target is not None: # zope.app.publisher.browser name = getDefaultViewName(target, self.request) return self.targetView(name) return u'' def targetDownload(self): return self.targetView('download.html', 'download') def targetRender(self): return u'