/* * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed with * this work for additional information regarding copyright ownership. * The ASF licenses this file to You under the Apache License, Version 2.0 * (the "License"); you may not use this file except in compliance with * the License. You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package org.apache.cocoon.portal.layout.renderer.aspect.impl; import java.util.Collections; import java.util.Iterator; import java.util.Map; import java.util.List; import java.util.ArrayList; import org.apache.avalon.framework.parameters.ParameterException; import org.apache.avalon.framework.parameters.Parameters; import org.apache.cocoon.portal.PortalService; import org.apache.cocoon.portal.aspect.impl.DefaultAspectDescription; import org.apache.cocoon.portal.event.impl.ChangeAspectDataEvent; import org.apache.cocoon.portal.layout.CompositeLayout; import org.apache.cocoon.portal.layout.Item; import org.apache.cocoon.portal.layout.Layout; import org.apache.cocoon.portal.layout.NamedItem; import org.apache.cocoon.portal.layout.renderer.aspect.RendererAspectContext; import org.apache.cocoon.xml.AttributesImpl; import org.apache.cocoon.xml.XMLUtils; import org.xml.sax.ContentHandler; import org.xml.sax.SAXException; /** * <h2>Example XML:</h2> * <pre> * <composite> * <named-item name="..." parameter="link-event"/> * <named-item name="..." selected="true"> * <!-- output from processing layout --> * </named-item> * <named-item name="..." parameter="link-event"/> * <named-item name="..." parameter="link-event"/> * </composite> * </pre> * * <h2>Example XML with sub-navigation (child-tag-name enabled):</h2> * <pre> * <composite> * <named-item name="..." parameter="link-event"/> * <named-item name="..." selected="true"> * <!-- output from processing layout --> * </named-item> * <named-item name="..." parameter="link-event"/> * <named-item name="..." parameter="link-event"> * <<i>child-tag-name</i>> * <named-item name="..." parameter="link-event"/> * <named-item name="..." parameter="link-event"/> * </<i>child-tag-name</i>> * </named-item> * </composite> * </pre> * * <h2>Example XML with sub-navigation (show-all-nav without child-tag-name enabled):</h2> * <pre> * <composite> * <named-item name="..." parameter="link-event"/> * <named-item name="..." selected="true"> * <!-- output from processing layout --> * </named-item> * <named-item name="..." parameter="link-event"/> * <named-item name="..." parameter="link-event"> * <named-item name="..." parameter="link-event"/> * <named-item name="..." parameter="link-event"/> * </named-item> * </composite> * </pre> * * * * <h2>Applicable to:</h2> * <ul> * <li>{@link org.apache.cocoon.portal.layout.CompositeLayout}</li> * </ul> * * <h2>Parameters</h2> * <table><tbody> * <tr><th>store</th><td></td><td>req</td><td>String</td><td><code>null</code></td></tr> * <tr><th>aspect-name</th><td>Aspect holding the current tab state.</td><td>req</td><td>String</td><td><code>"tab"</code></td></tr> * <tr><th>tag-name</th><td>Name of the tag enclosing the following output.</td> * <td></td><td>String</td><td><code>"composite"</code></td></tr> * <tr><th>root-tag</th><td>Should a tag enclosing the following output be generated?</td> * <td></td><td>boolean</td><td><code>true</code></td></tr> * <tr><th>child-tag-name</th><td>The name of the tag to enclose named items (i.e. the subnavigation) * of non-selected (default) items. Setting this parameter will enable show-all-nav.</td> * <td></td><td>String</td><td><code>""</code></td></tr> * <tr><th>show-all-nav</th><td>Setting this value to true will output the enclosed named-items</td> * <td></td><td>boolean</td><td><code>false</code></td></tr> * <tr><th>include-selected</th><td>Setting this value to true will output the enclosed named-items of the selected tab too.</td> * <td></td><td>boolean</td><td><code>false</code></td></tr> * </tbody></table> * * @author <a href="mailto:cziegeler@s-und-n.de">Carsten Ziegeler</a> * @author <a href="mailto:volker.schmitt@basf-it-services.com">Volker Schmitt</a> * * @version CVS $Id$ */ public class TabContentAspect extends CompositeContentAspect { /** * Chains that include this aspect must always render * @return true */ public boolean isRequired() { return true; } /* (non-Javadoc) * @see org.apache.cocoon.portal.layout.renderer.RendererAspect#toSAX(org.apache.cocoon.portal.layout.renderer.RendererAspectContext, org.apache.cocoon.portal.layout.Layout, org.apache.cocoon.portal.PortalService, org.xml.sax.ContentHandler) */ public void toSAX(RendererAspectContext context, Layout layout, PortalService service, ContentHandler handler) throws SAXException { if (layout instanceof CompositeLayout) { TabPreparedConfiguration config = (TabPreparedConfiguration)context.getAspectConfiguration(); if ( config.rootTag ) { XMLUtils.startElement(handler, config.tagName); } AttributesImpl attributes = new AttributesImpl(); CompositeLayout tabLayout = (CompositeLayout) layout; // selected tab Integer data = (Integer) layout.getAspectData(config.aspectName); int selected = data.intValue(); // loop over all tabs for (int j = 0; j < tabLayout.getSize(); j++) { Item tab = tabLayout.getItem(j); ChangeAspectDataEvent event = null; // open named-item tag attributes.clear(); if ( tab instanceof NamedItem ) { attributes.addCDATAAttribute("name", String.valueOf(((NamedItem)tab).getName())); } if (j == selected) { attributes.addCDATAAttribute("selected", "true"); } event = new ChangeAspectDataEvent(tabLayout, config.aspectName, new Integer(j)); attributes.addCDATAAttribute("parameter", service.getComponentManager().getLinkService().getLinkURI(event)); // add parameters final Iterator iter = tab.getParameters().entrySet().iterator(); while ( iter.hasNext() ) { final Map.Entry entry = (Map.Entry) iter.next(); attributes.addCDATAAttribute((String)entry.getKey(), (String)entry.getValue()); } XMLUtils.startElement(handler, "named-item", attributes); if (j == selected) { this.processLayout(tab.getLayout(), service, handler); if (config.includeSelected) { List events = new ArrayList(); events.add(event); this.processNav(context, tab.getLayout(), service, handler, events); } } else if (config.showAllNav) { List events = new ArrayList(); events.add(event); this.processNav(context, tab.getLayout(), service, handler, events); } // close named-item tag XMLUtils.endElement(handler, "named-item"); } if ( config.rootTag ) { XMLUtils.endElement(handler, config.tagName); } } else { throw new SAXException("Wrong layout type, TabLayout expected: " + layout.getClass().getName()); } } /** * Return the aspects required for this renderer * @return An iterator for the aspect descriptions or null. */ public Iterator getAspectDescriptions(Object configuration) { TabPreparedConfiguration pc = (TabPreparedConfiguration)configuration; DefaultAspectDescription desc = new DefaultAspectDescription(); desc.setName(pc.aspectName); desc.setClassName("java.lang.Integer"); desc.setPersistence(pc.store); desc.setAutoCreate(true); return Collections.singletonList(desc).iterator(); } /* * Generate the sub navigation for non-selected tabs * @param context * @param layout * @param service * @param handler * @throws SAXException */ private void processNav(RendererAspectContext context, Layout layout, PortalService service, ContentHandler handler, List parentEvents) throws SAXException { if (layout instanceof CompositeLayout) { CompositeLayout tabLayout = (CompositeLayout)layout; if (tabLayout.getSize() == 0) { return; } TabPreparedConfiguration config = (TabPreparedConfiguration) context.getAspectConfiguration(); AttributesImpl attributes = new AttributesImpl(); boolean subNav = false; // loop over all tabs for (int j = 0; j < tabLayout.getSize(); j++) { Item tab = tabLayout.getItem(j); // open named-item tag attributes.clear(); if (tab instanceof NamedItem) { if (!subNav && !config.childTagName.equals("")) { XMLUtils.startElement(handler, config.childTagName); subNav = true; } attributes.addCDATAAttribute("name", String.valueOf(((NamedItem) tab).getName())); ChangeAspectDataEvent event = new ChangeAspectDataEvent(tabLayout, config.aspectName, new Integer(j)); List events = new ArrayList(parentEvents); events.add(event); attributes.addCDATAAttribute("parameter", service.getComponentManager().getLinkService().getLinkURI(events)); // add parameters final Iterator iter = tab.getParameters().entrySet().iterator(); while (iter.hasNext()) { final Map.Entry entry = (Map.Entry) iter.next(); attributes.addCDATAAttribute((String) entry.getKey(), (String) entry.getValue()); } XMLUtils.startElement(handler, "named-item", attributes); this.processNav(context, tab.getLayout(), service, handler, events); // close named-item tag XMLUtils.endElement(handler, "named-item"); } } // close sub-nav tag if (subNav) { XMLUtils.endElement(handler, config.childTagName); } } } protected class TabPreparedConfiguration extends PreparedConfiguration { public String aspectName; public String store; public boolean showAllNav = false; public boolean includeSelected = false; public String childTagName; public void takeValues(TabPreparedConfiguration from) { super.takeValues(from); this.aspectName = from.aspectName; this.store = from.store; this.showAllNav = from.showAllNav; this.includeSelected = from.includeSelected; this.childTagName = from.childTagName; } } /* (non-Javadoc) * @see org.apache.cocoon.portal.layout.renderer.aspect.RendererAspect#prepareConfiguration(org.apache.avalon.framework.parameters.Parameters) */ public Object prepareConfiguration(Parameters configuration) throws ParameterException { TabPreparedConfiguration pc = new TabPreparedConfiguration(); pc.takeValues((PreparedConfiguration)super.prepareConfiguration(configuration)); pc.aspectName = configuration.getParameter("aspect-name", "tab"); pc.store = configuration.getParameter("store"); pc.childTagName = configuration.getParameter("child-tag-name", ""); if (!pc.childTagName.equals("")) { pc.showAllNav = true; } else { pc.showAllNav = configuration.getParameterAsBoolean("show-all-nav", false); } pc.includeSelected = configuration.getParameterAsBoolean("include-selected", false); return pc; } }