===============
Ready 2 go Menu
===============

The z3c.menu.ready2go package provides a menu implementation which allows you
to implement menus based on content providers and viewlets.

First let's setup our defualt menu item template:

  >>> import os
  >>> import zope.component
  >>> from zope.publisher.interfaces.browser import IDefaultBrowserLayer
  >>> from zope.publisher.interfaces.browser import IBrowserView
  >>> from z3c.template.interfaces import IContentTemplate
  >>> from z3c.template.template import TemplateFactory
  >>> import z3c.menu.ready2go
  >>> baseDir = os.path.split(z3c.menu.ready2go.__file__)[0]
  >>> itemTemplate = os.path.join(baseDir, 'item.pt')
  >>> itemTemplateFactory = TemplateFactory(itemTemplate, 'text/html')
  >>> zope.component.provideAdapter(itemTemplateFactory,
  ...     (IBrowserView, IDefaultBrowserLayer), IContentTemplate)


Global Menu
-----------

Let's create some menu and register them as viewlet manager:

  >>> from zope.viewlet.interfaces import IViewlet
  >>> from zope.viewlet import manager
  >>> from z3c.menu.ready2go import interfaces
  >>> from z3c.menu.ready2go import IGlobalMenu
  >>> from z3c.menu.ready2go import ISiteMenu
  >>> from z3c.menu.ready2go import IContextMenu
  >>> from z3c.menu.ready2go import IAddMenu
  >>> from z3c.menu.ready2go.manager import MenuManager

And we configure our menu item as viewlet Managers. This is normaly done by the
``viewletManager`` ZCML directive:

  >>> GlobalMenu = manager.ViewletManager('left', IGlobalMenu,
  ...     bases=(MenuManager,))

  >>> SiteMenu = manager.ViewletManager('left', ISiteMenu,
  ...     bases=(MenuManager,))

  >>> ContextMenu = manager.ViewletManager('left', IContextMenu,
  ...     bases=(MenuManager,))

  >>> AddMenu = manager.ViewletManager('left', IAddMenu,
  ...     bases=(MenuManager,))

Our menu managers implement IMenuManager:

  >>> interfaces.IMenuManager.implementedBy(GlobalMenu)
  True

  >>> interfaces.IMenuManager.implementedBy(SiteMenu)
  True

  >>> interfaces.IMenuManager.implementedBy(ContextMenu)
  True

  >>> interfaces.IMenuManager.implementedBy(AddMenu)
  True

We also need our checker adapter which can check if a menu item is available
and/or selected:

  >>> from z3c.menu.ready2go import checker
  >>> zope.component.provideAdapter(checker.GlobalSelectedChecker)
  >>> zope.component.provideAdapter(checker.SiteSelectedChecker)
  >>> zope.component.provideAdapter(checker.ContextSelectedChecker)

Now we have to define a site and a context:

  >>> import zope.interface
  >>> from zope.container import contained, btree
  >>> from zope.container.interfaces import IContained
  >>> from zope.component.interfaces import IPossibleSite
  >>> from zope.site.site import SiteManagerContainer
  >>> from zope.site.site import LocalSiteManager

  >>> @zope.interface.implementer(IPossibleSite)
  ... class Site(btree.BTreeContainer, SiteManagerContainer):
  ...     def __init__(self):
  ...         super(Site, self).__init__()
  ...         self.setSiteManager(LocalSiteManager(self))

  >>> @zope.interface.implementer(IContained)
  ... class Content(contained.Contained):
  ...     pass

  >>> root['site'] = Site()
  >>> site = root['site']

Now we have to set the site object as site. This is normaly done by the
traverser but we do this here with the hooks helper because we do not really
traaverse to the site within the publisher/traverser:

  >>> from zope.site import hooks
  >>> hooks.setSite(site)

  >>> site['content'] = Content()
  >>> content = site['content']

  >>> from zope.publisher.browser import TestRequest
  >>> request = TestRequest()

And we need a view which knows about it's parent:

  >>> @zope.interface.implementer(IBrowserView)
  ... class View(contained.Contained):
  ...
  ...     def __init__(self, context, request):
  ...         self.__parent__ = context
  ...         self.context = context
  ...         self.request = request

  >>> view = View(content, request)

Our menus can adapt the context, request and view. See IViewletManager in
zope.viewlet for more infos about this pattern. If we render them, there is an
empty string returned. This means the menus don't find menu items for rendering:

  >>> globalMenu = GlobalMenu(content, request, view)
  >>> globalMenu.update()
  >>> globalMenu.render()
  u''

  >>> siteMenu = SiteMenu(content, request, view)
  >>> siteMenu.update()
  >>> siteMenu.render()
  u''

  >>> contextMenu = ContextMenu(content, request, view)
  >>> contextMenu.update()
  >>> contextMenu.render()
  u''

  >>> addMenu = AddMenu(content, request, view)
  >>> addMenu.update()
  >>> addMenu.render()
  u''


Global Menu Item
----------------

Now we register a context menu item for our IGlobalMenu:

  >>> from z3c.menu.ready2go.item import GlobalMenuItem
  >>> class MyGlobalMenuItem(GlobalMenuItem):
  ...
  ...     viewName = 'root.html'

Now we need a security checker for our menu item

  >>> from zope.security.checker import NamesChecker, defineChecker
  >>> viewletChecker = NamesChecker(('update', 'render'))
  >>> defineChecker(MyGlobalMenuItem, viewletChecker)

And we configure our menu item for IGlobalMenu. This is normaly done by the
``viewlet`` ZCML directive:

  >>> zope.component.provideAdapter(
  ...     MyGlobalMenuItem,
  ...     (zope.interface.Interface, IDefaultBrowserLayer,
  ...     IBrowserView, IGlobalMenu),
  ...     IViewlet, name='My Global')

Now let's update the menu manager and see that this manager now contains
the menu item:

  >>> globalMenu.update()
  >>> myGlobalMenuItem = globalMenu.viewlets[0]
  >>> myGlobalMenuItem
  <MyGlobalMenuItem u'My Global'>

Now let's render the global menu manager and you can see that the menu item
get rendered:

  >>> print(globalMenu.render())
  <li>
    <a href="http://127.0.0.1/root.html"><span>My Global</span></a>
  </li>


Site Menu Item
--------------

Now we register a context menu item for our ISiteMenu:

  >>> import zope.component
  >>> from zope.publisher.interfaces.browser import IDefaultBrowserLayer

  >>> from z3c.menu.ready2go.item import SiteMenuItem
  >>> class MySiteMenuItem(SiteMenuItem):
  ...
  ...     viewName = 'site.html'

Now we need a security checker for our menu item

  >>> from zope.security.checker import NamesChecker, defineChecker
  >>> viewletChecker = NamesChecker(('update', 'render'))
  >>> defineChecker(MySiteMenuItem, viewletChecker)

And we configure our menu item for ISiteMenu. This is normaly done by the
``viewlet`` ZCML directive:

  >>> zope.component.provideAdapter(
  ...     MySiteMenuItem,
  ...     (zope.interface.Interface, IDefaultBrowserLayer,
  ...     IBrowserView, ISiteMenu),
  ...     IViewlet, name='My Site')

Now let's render the site menu again. You can see that we ve got a menu item
and the url points to our site:

  >>> siteMenu.update()
  >>> print(siteMenu.render())
  <li>
    <a href="http://127.0.0.1/site/site.html"><span>My Site</span></a>
  </li>


Context Menu Item
-----------------

Now we register a context menu item for our IContextMenu:

  >>> import zope.component
  >>> from zope.publisher.interfaces.browser import IDefaultBrowserLayer

  >>> from z3c.menu.ready2go.item import ContextMenuItem
  >>> class MyContextMenuItem(ContextMenuItem):
  ...     viewName = 'context.html'
  ...     weight = 1

Now we need a security checker for our menu item

  >>> from zope.security.checker import NamesChecker, defineChecker
  >>> viewletChecker = NamesChecker(('update', 'render'))
  >>> defineChecker(MyContextMenuItem, viewletChecker)

And we configure our menu item for IContextMenu. This is normaly done by the
``viewlet`` ZCML directive:

  >>> zope.component.provideAdapter(
  ...     MyContextMenuItem,
  ...     (zope.interface.Interface, IDefaultBrowserLayer,
  ...     IBrowserView, IContextMenu),
  ...     IViewlet, name='My Context')

Now let's render the context menu again. You can see that we ve got a menu
item. Another important point here is, that the url of such ContextMemuItem
implementations point to the context of the view:

  >>> contextMenu.update()
  >>> print(contextMenu.render())
  <li>
    <a href="http://127.0.0.1/site/content/context.html"><span>My Context</span></a>
  </li>

Let's set the view  __name__ to ``context.html``. This will reflect that
the view offers the same name that our context menu needs to get rendered as
selected:

  >>> view.__name__ = 'context.html'

Now try again and see if the context menu item get rendered as selected:

  >>> contextMenu.update()
  >>> print(contextMenu.render())
  <li class="selected">
    <a href="http://127.0.0.1/site/content/context.html"><span>My Context</span></a>
  </li>

Also, let's check that menu item is marked selected even if we provided a viewName in
the ``@@context.html`` form:

  >>> MyContextMenuItem.viewName = '@@context.html'
  >>> contextMenu.update()
  >>> print(contextMenu.render())
  <li class="selected">
    <a href="http://127.0.0.1/site/content/@@context.html"><span>My Context</span></a>
  </li>

Okay, change viewName back to ``context.html`` for further tests:

  >>> MyContextMenuItem.viewName = 'context.html'

Now add a second context menu item and check if we can use the cssInActive
argument which is normaly a empty string:

  >>> class InActiveMenuItem(ContextMenuItem):
  ...     viewName = 'inActive.html'
  ...     cssInActive = 'inActive'
  ...     weight = 2

  >>> defineChecker(InActiveMenuItem, viewletChecker)

  >>> zope.component.provideAdapter(
  ...     InActiveMenuItem,
  ...     (zope.interface.Interface, IDefaultBrowserLayer,
  ...     IBrowserView, IContextMenu),
  ...     IViewlet, name='In Active')

Now update and render again:

  >>> contextMenu.update()
  >>> print(contextMenu.render())
  <li class="selected">
    <a href="http://127.0.0.1/site/content/context.html"><span>My Context</span></a>
  </li>
  <li class="inActive">
    <a href="http://127.0.0.1/site/content/inActive.html"><span>In Active</span></a>
  </li>

AddMenu
-------

The add menu can be used for offering links to any kind of add forms per
context. This allows us to offer independent add form links doesn't matter which
form framework is used. Let's now define such a simple AddMenuItem pointing
to a add form url. Not; the add form and it's url do not exist in thsi test.
This aslo means there is no guarantee that a form exist if a add menu item
is configured.

  >>> from z3c.menu.ready2go.item import AddMenuItem
  >>> class MyAddMenuItem(AddMenuItem):
  ...
  ...     viewName = 'addSomething.html'

Now we need a security checker for our menu item

  >>> from zope.security.checker import NamesChecker, defineChecker
  >>> viewletChecker = NamesChecker(('update', 'render'))
  >>> defineChecker(MyAddMenuItem, viewletChecker)

And we configure our menu item for IAddMenu. This is normaly done by the
``viewlet`` ZCML directive:

  >>> zope.component.provideAdapter(
  ...     MyAddMenuItem,
  ...     (zope.interface.Interface, IDefaultBrowserLayer,
  ...     IBrowserView, IAddMenu),
  ...     IViewlet, name='My AddMenu')

Now we can update and render our add menu:

  >>> addMenu.update()
  >>> print(addMenu.render())
  <li>
    <a href="http://127.0.0.1/site/content/addSomething.html"><span>My AddMenu</span></a>
  </li>


Menu groups
-----------

The global and the site menu items are grouped menu items. This means such menu
items should get rendered as selected if a context menu item is selected. This
reflects the menu hierarchie. Let's show how we can solve this not so simple
problem. We offer a ISelectedChecker adapter which can decide if a menu get
rendered as selected or not. This is very usefull because normaly a menu get
registered and later we add views and can not change the menu item
implementation. Let's see how such an adapter can handle an existing menu,
context and view setup and change the selected rendering. We register a
selected checker for our site menu item:

  >>> zope.component.provideAdapter(checker.TrueSelectedChecker,
  ...     (IContained, IDefaultBrowserLayer, None, ISiteMenu, MySiteMenuItem),
  ...     interfaces.ISelectedChecker)

Now we can render the site menu again. Note that our context is still the
sample content object.

  >>> siteMenu.update()
  >>> print(siteMenu.render())
  <li class="selected">
    <a href="http://127.0.0.1/site/site.html"><span>My Site</span></a>
  </li>

This reflects that the site menu is a group menu which the context menu item
of the content object is selected too.

  >>> contextMenu.update()
  >>> print(contextMenu.render())
  <li class="selected">
    <a href="http://127.0.0.1/site/content/context.html"><span>My Context</span></a>
  </li>
  <li class="inActive">
    <a href="http://127.0.0.1/site/content/inActive.html"><span>In Active</span></a>
  </li>


EmptyMenuManager
----------------

There is a empty menu manager whihc could be used for override existing
menu managers.

  >>> from z3c.menu.ready2go.manager import EmptyMenuManager
  >>> emptyMenu = EmptyMenuManager(None, None, None)

Our empty menu manager implements ``IMenuManager``:

  >>> interfaces.IMenuManager.providedBy(emptyMenu)
  True

This empty menu manager returns allways an empty string if we render them:

  >>> emptyMenu.update()
  >>> emptyMenu.render()
  u''


Special use case
----------------

We have some special use case because of Zope's internals. One important part
is that our menu heavy depend on context and it's __parent__ chain to the
zope application root. This is not allways supported by Zopes default setup.
One part is the bad integrated application control part which fakes a root
object which doesn't know about the real childs of the real root from the
ZODB e.g. application root. Now we will show you that our menu by default
render no items if we get such a fake root which messes up our menu structure.

Let's define a object which does not know about any __parent__.

  >>> nirvana = Content()
  >>> nirvanaView = View(nirvana, request)

Now we can check what's happen to the menus if we adapt the parent less nirvana
context and update and render the menus. You can see that the global menu does
not contain any menu item. That's because the global menu items tries to find
the root by traversing from the context to the root by the __parent__ chain
and we don't support any parent for our nirvana object:

  >>> globalMenu = GlobalMenu(nirvana, request, nirvanaView)
  >>> globalMenu.update()
  >>> globalMenu.render()
  u''

Also the SiteMenu doesn't contain any menu item because of the parent less
object:

  >>> siteMenu = SiteMenu(nirvana, request, nirvanaView)
  >>> siteMenu.update()
  >>> siteMenu.render()
  u''

  >>> contextMenu = ContextMenu(nirvana, request, nirvanaView)
  >>> contextMenu.update()
  >>> contextMenu.render()
  u''

  >>> addMenu = AddMenu(nirvana, request, nirvanaView)
  >>> addMenu.update()
  >>> addMenu.render()
  u''
