All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.codename1.components.Accordion Maven / Gradle / Ivy

There is a newer version: 7.0.167
Show newest version
/*
 * Copyright (c) 2012, Codename One and/or its affiliates. All rights reserved.
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
 * This code is free software; you can redistribute it and/or modify it
 * under the terms of the GNU General Public License version 2 only, as
 * published by the Free Software Foundation.  Codename One designates this
 * particular file as subject to the "Classpath" exception as provided
 * by Oracle in the LICENSE file that accompanied this code.
 *  
 * This code 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
 * version 2 for more details (a copy is included in the LICENSE file that
 * accompanied this code).
 * 
 * You should have received a copy of the GNU General Public License version
 * 2 along with this work; if not, write to the Free Software Foundation,
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
 * 
 * Please contact Codename One through http://www.codenameone.com/ if you 
 * need additional information or have any questions.
 */
package com.codename1.components;

import com.codename1.ui.Button;
import com.codename1.ui.Component;
import com.codename1.ui.Container;
import com.codename1.ui.FontImage;
import com.codename1.ui.Image;
import com.codename1.ui.Label;
import com.codename1.ui.events.ActionEvent;
import com.codename1.ui.events.ActionListener;
import com.codename1.ui.layouts.BorderLayout;
import com.codename1.ui.layouts.BoxLayout;
import com.codename1.ui.plaf.LookAndFeel;
import com.codename1.ui.plaf.UIManager;
import com.codename1.ui.util.EventDispatcher;

/**
 * 

The {@code Accordion} ui pattern is a vertically stacked list of items. * Each item can be opened/closed to reveal more content similarly to a {@link com.codename1.ui.tree.Tree} * however unlike the {@link com.codename1.ui.tree.Tree} the {@code Accordion} is designed to include * containers or arbitrary components rather than model based data.

*

* This makes the {@code Accordion} more convenient as a tool for folding/collapsing UI elements known in advance * whereas a {@link com.codename1.ui.tree.Tree} makes more sense as a tool to map data e.g. filesystem * structure, XML hierarchy etc. *

*

* Note that the {@code Accordion} like many composite components in Codename One is scrollable by default * which means you should use it within a non-scrollable hierarchy. If you wish to add it into a scrollable * {@link com.codename1.ui.Container} you should disable it's default scrollability using {@code setScrollable(false)}. *

* *

Example Usage

* * * *

Screenshots

*

Accordion Component

* * * @author Chen */ public class Accordion extends Container { private Image closeIcon; private Image openIcon; private boolean autoClose = true; private String uiidBackGroundItem = "AccordionItem"; private String uiidHeader = "AccordionArrow"; private String uiidOpenCloseIcon = "AccordionArrow"; private final EventDispatcher listeners = new EventDispatcher(); /** * Empty Constructor */ public Accordion() { super.setLayout(BoxLayout.y()); this.closeIcon = isRTL() ? FontImage.createMaterial(FontImage.MATERIAL_KEYBOARD_ARROW_LEFT, UIManager.getInstance().getComponentStyle(uiidOpenCloseIcon)) : FontImage.createMaterial(FontImage.MATERIAL_KEYBOARD_ARROW_RIGHT, UIManager.getInstance().getComponentStyle(uiidOpenCloseIcon)); this.openIcon = FontImage.createMaterial(FontImage.MATERIAL_KEYBOARD_ARROW_DOWN, UIManager.getInstance().getComponentStyle(uiidOpenCloseIcon)); setScrollableY(true); } /** * Create an accordion with open and close icons set * @param openIcon the open icon of the accordion * @param closeIcon the close icon of the accordion */ public Accordion(Image openIcon, Image closeIcon) { super.setLayout(BoxLayout.y()); this.closeIcon = openIcon; this.openIcon = closeIcon; setScrollableY(true); } /** * Create an accordion with open and close icons set * @param openIcon the open icon of the accordion * @param closeIcon the close icon of the accordion */ public Accordion(char openIcon, char closeIcon) { super.setLayout(BoxLayout.y()); this.closeIcon = FontImage.createMaterial(openIcon, UIManager.getInstance().getComponentStyle(uiidOpenCloseIcon)); this.openIcon = FontImage.createMaterial(closeIcon, UIManager.getInstance().getComponentStyle(uiidOpenCloseIcon)); setScrollableY(true); } /** * Create an accordion with open and close icons set * @param openIcon the open icon of the accordion * @param closeIcon the close icon of the accordion * @param openCloseUIID the uiid of the open and close icons */ public Accordion(char openIcon, char closeIcon, String openCloseUIID) { super.setLayout(BoxLayout.y()); this.uiidOpenCloseIcon = openCloseUIID; this.closeIcon = FontImage.createMaterial(openIcon, UIManager.getInstance().getComponentStyle(openCloseUIID)); this.openIcon = FontImage.createMaterial(closeIcon, UIManager.getInstance().getComponentStyle(openCloseUIID)); setScrollableY(true); } /** * Add an item to the Accordion Container * * @param header the item title * @param body the item Component to hide/show */ public void addContent(String header, Component body) { addContent(new Label(header, uiidHeader), body); } /** * Replaces the title for content that was already added. Notice that this will fail if the content isn't * in yet. * @param header the new title for the content * @param body the content that was already added with a different header using addContent */ public void setHeader(String header, Component body) { AccordionContent ac = (AccordionContent) body.getParent(); ((Label)ac.header).setText(header); } /** * Replaces the title for content that was already added. Notice that this will fail if the content isn't * in yet. * @param header the new title for the content * @param body the content that was already added with a different header using addContent */ public void setHeader(Component header, Component body) { AccordionContent ac = (AccordionContent) body.getParent(); ac.header.getParent().replace(ac.header, header, null); } /** * Removes the content from the accordion * @param body the body previously added with {@link #addContent(com.codename1.ui.Component, com.codename1.ui.Component)} or */ public void removeContent(Component body) { body.getParent().remove(); body.remove(); } /** * Add an item to the Accordion Container * * @param header the item title Component * @param body the item Component to hide/show */ public void addContent(Component header, Component body) { add(new AccordionContent(header, body)); } /** * Sets the header UIID for the given accordion uiid * * @param body the component within the accordion * @param uiid the uiid for the header */ public void setHeaderUIID(Component body, String uiid) { AccordionContent ac = (AccordionContent) body.getParent(); if(ac == null) { body.putClientProperty("cn1$setHeaderUIID", uiid); } else { if(ac.top != null) { ac.top.setUIID(uiid); } else { ac.topUiid = uiid; } } } /** * Returns the body component of the currently expanded accordion element or null if none is expanded * @return a component */ public Component getCurrentlyExpanded() { for (Component cc : this) { AccordionContent c = (AccordionContent)cc; if(!c.isClosed()){ return c.body; } } return null; } /** * Expands the accordion with the given "body" * @param body the body component of the accordion to expand */ public void expand(Component body) { if(autoClose) { for (Component cc : this) { AccordionContent c = (AccordionContent)cc; c.openClose(!(body == c.body)); } } else { for (Component cc : this) { AccordionContent c = (AccordionContent)cc; if(body == c.body) { c.openClose(false); } } } } /** * Closes the accordion with the given "body" * @param body the body component of the accordion to close */ public void collapse(Component body) { for (Component cc : this) { AccordionContent c = (AccordionContent)cc; if(body == c.body) { c.openClose(true); } } } /** * Sets the closed icon * @param closeIcon the close icon */ public void setCloseIcon(Image closeIcon) { this.closeIcon = closeIcon; } /** * Sets the open icon * @param openIcon the open icon */ public void setOpenIcon(Image openIcon) { this.openIcon = openIcon; } /** * Sets the closed icon using material image * @param closeIcon the close icon (e.g. {@code FontImage.MATERIAL_KEYBOARD_ARROW_DOWN}) */ public void setCloseIcon(char closeIcon) { this.closeIcon = FontImage.createMaterial(closeIcon, UIManager.getInstance().getComponentStyle(uiidOpenCloseIcon)); } /** * Sets the open icon using material image * @param openIcon the open icon (e.g. {@code FontImage.MATERIAL_KEYBOARD_ARROW_RIGHT}) */ public void setOpenIcon(char openIcon) { this.openIcon = FontImage.createMaterial(openIcon, UIManager.getInstance().getComponentStyle(uiidOpenCloseIcon)); } /** * Sets the closed icon using material image * @param closeIcon the close icon (e.g. {@code FontImage.MATERIAL_KEYBOARD_ARROW_DOWN}) * @param size the size in millimeters for the arrow */ public void setCloseIcon(char closeIcon, float size) { this.closeIcon = FontImage.createMaterial(closeIcon, UIManager.getInstance().getComponentStyle(uiidOpenCloseIcon), size); } /** * Sets the open icon using material image * @param openIcon the open icon (e.g. {@code FontImage.MATERIAL_KEYBOARD_ARROW_RIGHT}) * @param size the size in millimeters for the arrow */ public void setOpenIcon(char openIcon, float size) { this.openIcon = FontImage.createMaterial(openIcon, UIManager.getInstance().getComponentStyle(uiidOpenCloseIcon), size); } /** * Sets the closed icon using material image with a custom uiid * @param closeIcon the close icon (e.g. {@code FontImage.MATERIAL_KEYBOARD_ARROW_DOWN}) * @param uiid to custom icon from res file */ public void setCloseIcon(char closeIcon, String uiid) { this.closeIcon = FontImage.createMaterial(closeIcon, UIManager.getInstance().getComponentStyle(uiid)); } /** * Sets the open icon using material image with a custom uiid * @param openIcon the open icon (e.g. {@code FontImage.MATERIAL_KEYBOARD_ARROW_RIGHT}) * @param uiid to custom icon from res file */ public void setOpenIcon(char openIcon, String uiid) { this.openIcon = FontImage.createMaterial(openIcon, UIManager.getInstance().getComponentStyle(uiid)); } /** * Sets the auto close flag, if this flag is true clicking on an item to open * an item will automatically close the previous opened item. * * @param autoClose determines if more then 1 item can be opened on screen */ public void setAutoClose(boolean autoClose) { this.autoClose = autoClose; } class AccordionContent extends Container { private boolean closed = true; private final Button arrow = new Button(); private Component body; private Component header; private Container top; private String topUiid = uiidHeader; public AccordionContent(Component header, final Component body) { setUIID(uiidBackGroundItem); setLayout(new BorderLayout()); this.body = body; this.header = header; header.setSelectedStyle(header.getUnselectedStyle()); header.setPressedStyle(header.getUnselectedStyle()); String t = (String)body.getClientProperty("cn1$setHeaderUIID"); if(t != null) { topUiid = t; } top = new Container(new BorderLayout(), topUiid); top.add(BorderLayout.CENTER, header); arrow.setUIID(uiidOpenCloseIcon); arrow.setIcon(closeIcon); arrow.addActionListener(new ActionListener() { public void actionPerformed(ActionEvent evt) { //toggle the current openClose(!isClosed()); if(autoClose){ for (int i = 0; i < Accordion.this.getComponentCount(); i++) { AccordionContent c = (AccordionContent)Accordion.this.getComponentAt(i); if(c != AccordionContent.this && !c.isClosed()){ c.openClose(true); } } } Accordion.this.animateLayout(250); fireEvent(evt); } }); top.add(BorderLayout.EAST, arrow); top.setLeadComponent(arrow); add(BorderLayout.NORTH, top); body.setHidden(true); add(BorderLayout.CENTER, body); } public boolean isClosed() { return closed; } public void openClose(boolean close) { closed = close; if (closed) { arrow.setIcon(closeIcon); } else { arrow.setIcon(openIcon); } body.setHidden(closed); } } /** * To listen item click in accordion component * @param a ActionListener to implement the method */ public void addOnClickItemListener(ActionListener a) { listeners.addListener(a); } /** * To remove item click in accordion component * @param a ActionListener to implement the method */ public void removeOnClickItemListener(ActionListener a) { listeners.removeListener(a); } private void fireEvent(ActionEvent ev) { listeners.fireActionEvent(ev); } /** * Default UIID for the content item within the accordion * * @param uiidBackGroundItem to custom the background in the accordion component */ public void setBackgroundItemUIID(String uiidBackGroundItem) { this.uiidBackGroundItem = uiidBackGroundItem; } /** * Default UIID for the content item within the accordion * * @return the uiid */ public String getBackgroundItemUIID() { return uiidBackGroundItem; } /** * UIID for the header component * * @param uiidHeader to custom the header in the accordion component */ public void setHeaderUIID(String uiidHeader) { this.uiidHeader = uiidHeader; } /** * UIID for the header component * * @retrun the uiid */ public String getHeaderUIID() { return uiidHeader; } /** * UIID for the arrow icon for expanding/collapsing * * @param uiidOpenCloseIcon to custom the background of the Open/Close icon */ public void setOpenCloseIconUIID(String uiidOpenCloseIcon) { this.uiidOpenCloseIcon = uiidOpenCloseIcon; } /** * UIID for the arrow icon for expanding/collapsing * * @return the UIID */ public String getOpenCloseIconUIID() { return uiidOpenCloseIcon; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy