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

com.codename1.ui.SwipeableContainer 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.ui;

import com.codename1.ui.animations.Motion;
import com.codename1.ui.events.ActionEvent;
import com.codename1.ui.events.ActionListener;
import com.codename1.ui.layouts.BorderLayout;
import com.codename1.ui.layouts.LayeredLayout;
import com.codename1.ui.util.EventDispatcher;

/**
 * 

{@code SwipeableContainer} allows us to side swipe a component and expose underlying configuration * within it. This is useful for editing, ranking of elements within a set of components e.g. in the * sample code below we use a ranking widget and swiping to expose the elements: *

* * Swipeable Container * * @author Chen */ public class SwipeableContainer extends Container { private Container bottomLeftWrapper; private Container bottomRightWrapper; private Container topWrapper; private boolean open = false; private boolean openedToRight = false; private boolean openedToLeft = false; private Motion openCloseMotion; private boolean swipeActivated = true; private SwipeListener press, drag, release; private int initialX = -1; private int initialY = -1; private int topX = -1; private boolean waitForRelease; private SwipeableContainer previouslyOpened; private final EventDispatcher dispatcher = new EventDispatcher(); /** * Simple Constructor * * @param bottomLeft the Component below the top, this Component is exposed * when dragging the top to the right * @param top the component on top. */ public SwipeableContainer(Component bottomLeft, Component top) { this(bottomLeft, null, top); } /** * Simple Constructor * * @param bottomLeft the Component below the top, this Component is exposed * when dragging the top to the right * @param bottomRight the Component below the top, this Component is exposed * when dragging the top to the Left * @param top the component on top. */ public SwipeableContainer(Component bottomLeft, Component bottomRight, Component top) { setLayout(new LayeredLayout()); bottomLeftWrapper = new Container(new BorderLayout()); if(bottomLeft != null){ bottomLeftWrapper.addComponent(BorderLayout.WEST, bottomLeft); bottomLeftWrapper.setVisible(false); } bottomRightWrapper = new Container(new BorderLayout()); if(bottomRight != null){ bottomRightWrapper.addComponent(BorderLayout.EAST, bottomRight); bottomRightWrapper.setVisible(false); } topWrapper = new Container(new BorderLayout()); topWrapper.addComponent(BorderLayout.CENTER, top); addComponent(bottomRightWrapper); addComponent(bottomLeftWrapper); addComponent(topWrapper); press = new SwipeListener(SwipeListener.PRESS); drag = new SwipeListener(SwipeListener.DRAG); release = new SwipeListener(SwipeListener.RELEASE); } /** * {@inheritDoc} */ protected void deinitialize() { waitForRelease = false; Form form = this.getComponentForm(); if (form != null) { form.removePointerPressedListener(press); form.removePointerReleasedListener(release); form.removePointerDraggedListener(drag); } super.deinitialize(); } /** * {@inheritDoc} */ protected void initComponent() { super.initComponent(); Form form = this.getComponentForm(); if (form != null && swipeActivated) { form.addPointerPressedListener(press); form.addPointerReleasedListener(release); form.addPointerDraggedListener(drag); } } /** * This method will open the top Component to the right if there is a Component * to expose on the left. */ public void openToRight() { if (open || openedToRight) { return; } if(bottomLeftWrapper.getComponentCount() == 0){ return; } Component bottom = bottomLeftWrapper.getComponentAt(0); if(bottomRightWrapper.getComponentCount() > 0){ bottomRightWrapper.setVisible(false); } bottomLeftWrapper.setVisible(true); int topX = topWrapper.getX(); openCloseMotion = Motion.createSplineMotion(topX, bottom.getWidth(), 300); getComponentForm().registerAnimated(this); openCloseMotion.start(); openedToRight = true; open = true; } /** * This method will open the top Component to the left if there is a Component * to expose on the right. */ public void openToLeft() { if (open || openedToLeft) { return; } if(bottomRightWrapper.getComponentCount() == 0){ return; } Component bottom = bottomRightWrapper.getComponentAt(0); if(bottomLeftWrapper.getComponentCount() > 0){ bottomLeftWrapper.setVisible(false); } bottomRightWrapper.setVisible(true); int topX = topWrapper.getX(); openCloseMotion = Motion.createSplineMotion(-topX, bottom.getWidth(), 300); getComponentForm().registerAnimated(this); openCloseMotion.start(); openedToLeft = true; open = true; } /** * Close the top component if it is currently opened. */ public void close() { if (!open) { return; } Form f = getComponentForm(); if(f != null) { if (openedToRight) { int topX = topWrapper.getX(); openCloseMotion = Motion.createSplineMotion(topX, 0, 300); } else { int topX = topWrapper.getX(); openCloseMotion = Motion.createSplineMotion(-topX, 0, 300); } f.registerAnimated(this); openCloseMotion.start(); } open = false; } @Override public Component getComponentAt(int x, int y) { if (!open) { return topWrapper.getComponentAt(x, y); } else { return super.getComponentAt(x, y); } } @Override public boolean animate() { if (openCloseMotion != null) { int val = openCloseMotion.getValue(); if (openedToRight) { topWrapper.setX(val); } else { topWrapper.setX(-val); } repaint(); boolean finished = openCloseMotion.isFinished(); if (finished) { //getComponentForm().deregisterAnimated(this); openCloseMotion = null; if(!open){ bottomRightWrapper.setVisible(false); bottomLeftWrapper.setVisible(false); openedToLeft = false; openedToRight = false; }else{ dispatcher.fireActionEvent(new ActionEvent(this,ActionEvent.Type.Swipe)); } } return !finished; } return false; } /** * disable/enable dragging of the top Component */ public void setSwipeActivated(boolean swipeActivated) { this.swipeActivated = swipeActivated; } /** * Returns true if swipe is activated */ public boolean isSwipeActivated() { return swipeActivated; } /** * Returns true if the top Component is currently opened */ public boolean isOpen() { return open && (openedToRight || openedToLeft); } /** * Returns true if the top Component is opened to the right side */ public boolean isOpenedToRight() { return openedToRight; } /** * Returns true if the top Component is opened to the left side */ public boolean isOpenedToLeft() { return openedToLeft; } @Override void doLayout() { int x = topWrapper.getX(); super.doLayout(); topWrapper.setX(x); } /** * Adds a listener to the SwipeableContainer which will cause an event to * dispatch once the SwipeableContainer is fully opened * * @param l implementation of the action listener interface */ public void addSwipeOpenListener(ActionListener l){ dispatcher.addListener(l); } /** * Removes the given listener from the SwipeableContainer * * @param l implementation of the action listener interface */ public void removeSwipeOpenListener(ActionListener l){ dispatcher.removeListener(l); } class SwipeListener implements ActionListener { private final static int PRESS = 0; private final static int DRAG = 1; private final static int RELEASE = 2; private final int type; public SwipeListener(int type) { this.type = type; } private void dragInitiatedRecursive(Container cnt) { for(Component c : cnt) { if(c instanceof Container) { dragInitiatedRecursive((Container)c); } c.dragInitiated(); } } public void actionPerformed(ActionEvent evt) { if (getComponentCount() == 0 || !swipeActivated || animate()) { return; } final int x = evt.getX(); final int y = evt.getY(); Form f = getComponentForm(); if (f == null) { return; } Component cmp = f.getComponentAt(x, y); if (!waitForRelease && !contains(cmp)) { return; } if (!waitForRelease && !topWrapper.contains(x, y)) { return; } Component bottomL; int bottomLeftW = 0; if(bottomLeftWrapper.getComponentCount() > 0){ bottomL = bottomLeftWrapper.getComponentAt(0); bottomLeftW = bottomL.getWidth(); } Component bottomR; int bottomRightW = 0; int bottomRightX = topWrapper.getWidth(); if(bottomRightWrapper.getComponentCount() > 0){ bottomR = bottomRightWrapper.getComponentAt(0); bottomRightW = bottomR.getWidth(); bottomRightX = bottomR.getX(); } switch (type) { case PRESS: { if (!topWrapper.contains(x, y)) { return; } topX = topWrapper.getX(); initialX = x; initialY = y; waitForRelease = true; break; } case DRAG: { if (!waitForRelease) { return; } if (Math.abs(y - initialY) > Math.abs(x - initialX)) { return; } if (!topWrapper.contains(x, y)) { return; } Component top = topWrapper.getComponentAt(0); top.dragInitiated(); if(top instanceof Container && top.getLeadComponent() == null) { dragInitiatedRecursive((Container)top); } if (initialX != -1) { if (getPreviouslyOpened() != null && getPreviouslyOpened() != SwipeableContainer.this && getPreviouslyOpened().isOpen()) { getPreviouslyOpened().close(); } int diff = x - initialX; int val = 0; if(!isOpen()){ val = Math.min(diff, bottomLeftW); val = Math.max(val, -bottomRightW); } if(openedToRight){ val = Math.min(diff, 0); val = Math.max(val, -bottomLeftW+1); }else if(openedToLeft){ val = Math.max(diff, 0); val = Math.min(val, bottomRightW-1); } topWrapper.setX(topX + val); if(topWrapper.getX()>0){ bottomRightWrapper.setVisible(false); bottomLeftWrapper.setVisible(true); }else{ bottomRightWrapper.setVisible(true); bottomLeftWrapper.setVisible(false); } repaint(); } break; } case RELEASE: { if (waitForRelease) { initialX = -1; //if (!isOpen()) { int topX = topWrapper.getX(); //it's opened to the right if (topX > 0) { if (Display.getInstance().getDragSpeed(false) < 0) { open = false; openedToRight = false; openToRight(); } else { open = true; close(); } } else if (topX < 0) { //check explicitly if opened to the left if (Display.getInstance().getDragSpeed(false) > 0) { open = false; openedToLeft = false; openToLeft(); } else { open = true; close(); } } //} waitForRelease = false; } break; } } } } /** * returns a previously opened SwipeableContainer that should be * automatically closed when starting to open this one. Called as soon as * this Swipeable starts opening. One approach is to override this method to * return a previously opened SwipeableContainer Can be overridden to return * a SwipeableContainer stored outside this container. * * @return an already open SwipeableContainer that will be closed when * opening this one, or null if none */ public SwipeableContainer getPreviouslyOpened() { return previouslyOpened; } /** * set a previously open SwipeableContainer, it will be closed as soon as * the user starts swiping this one. Be aware that with a long list of * Swipeable containers it may be a better approach to store the previously * opened outside the list and simply override getPreviouslyOpened to return * it * * @param previouslyOpened an already open SwipeableContainer that will be * closed if this one is opened */ public void setPreviouslyOpened(SwipeableContainer previouslyOpened) { this.previouslyOpened = previouslyOpened; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy