Java tutorial
/** * DnDMasonryLayout.java (Masonry) * * Copyright 2014 Vaadin Ltd, Sami Viitanen <sami.viitanen@vaadin.org> * * Licensed 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.vaadin.alump.masonry; import com.vaadin.event.dd.DragAndDropEvent; import com.vaadin.event.dd.DropHandler; import com.vaadin.event.dd.acceptcriteria.AcceptCriterion; import com.vaadin.event.dd.acceptcriteria.ServerSideCriterion; import com.vaadin.ui.Component; import com.vaadin.ui.DragAndDropWrapper; import java.util.ArrayList; import java.util.List; /** * Adds DnD reordering features to MasonryLayout. MasonryDndWrapper inherits DragAndDropWrapper, * how ever this default implementation does not do any drop handling. If you want to handle dropping * on layout itself, set the DropHandler. If all you need is reordering by dragging, see MasonryDndLayout class. * @see org.vaadin.alump.masonry.MasonryDnDLayout */ public class MasonryDnDWrapper extends DragAndDropWrapper { protected boolean allowReorder = true; protected List<MasonryReorderListener> reorderListeners = new ArrayList<MasonryReorderListener>(); /** * Interface called when user has reordered components by dragging them */ public interface MasonryReorderListener { /** * Called when user reordered components * @param event Event containing the reordering information */ void onUserReorder(MasonryReorderEvent event); } /** * Event containing information of reorder event caused by user */ public static class MasonryReorderEvent { protected MasonryDnDWrapper layout; public MasonryReorderEvent(MasonryDnDWrapper layout) { this.layout = layout; //TODO: more data } /** * Get layout where reordering happened * @return Layout where reordering happened */ public MasonryDnDWrapper getLayout() { return layout; } } private DragAndDropWrapper.DragStartMode componentDragStartMode = DragAndDropWrapper.DragStartMode.WRAPPER; /** * Default drop handler that takes care of reordering calls when child is dragged above other children */ public static class MasonryDropHandler implements DropHandler { protected MasonryDnDWrapper layout; protected DragAndDropWrapper target; public MasonryDropHandler(MasonryDnDWrapper layout) { this(layout, null); } public MasonryDropHandler(MasonryDnDWrapper layout, DragAndDropWrapper childWrapper) { this.layout = layout; this.target = childWrapper; } @Override public void drop(DragAndDropEvent event) { Component dragged = event.getTransferable().getSourceComponent(); // If no target just add to end if (target == null) { layout.getMasonryLayout().addComponent(dragged); } else { WrapperTargetDetails details = (WrapperTargetDetails) event.getTargetDetails(); String wrapperStyleName = layout.getMasonryLayout().getComponentWrapperStyleName(dragged); layout.getMasonryLayout().addComponentBefore(dragged, wrapperStyleName, target); } layout.notifyReorderListeners(); } @Override public AcceptCriterion getAcceptCriterion() { ServerSideCriterion criterion = new ServerSideCriterion() { @Override public boolean accept(DragAndDropEvent dragEvent) { if (!layout.isReorderable()) { return false; } try { Component source = dragEvent.getTransferable().getSourceComponent(); if (target != null) { return target != source.getParent() && source.getParent().getParent() == layout; } else { return source.getParent().getParent() == layout; } } catch (Exception e) { e.printStackTrace(); return false; } } }; return criterion; } } public MasonryDnDWrapper() { super(new MasonryLayout()); addStyleName("dnd-masonry-layout"); // Simplifies the drag hints getMasonryLayout().addStyleName("no-vertical-drag-hints"); getMasonryLayout().addStyleName("no-horizontal-drag-hints"); // Make sure layout takes full width getMasonryLayout().setWidth("100%"); // No drop handler used in this version setDragStartMode(DragStartMode.NONE); //setDropHandler(new DndMasonryDropHandler(this)); } public MasonryDnDWrapper(int columnWidth) { super(new MasonryLayout(columnWidth)); addStyleName("dnd-masonry-layout"); // Simplifies the drag hints getMasonryLayout().addStyleName("no-vertical-drag-hints"); getMasonryLayout().addStyleName("no-horizontal-drag-hints"); // Make sure layout takes full width getMasonryLayout().setWidth("100%"); // No drop handler used in this version setDragStartMode(DragStartMode.NONE); //setDropHandler(new DndMasonryDropHandler(this)); } public void setComponentDragStartMode(DragAndDropWrapper.DragStartMode dragStartMode) { this.componentDragStartMode = dragStartMode; } public DragAndDropWrapper.DragStartMode getComponentDragStartMode() { return componentDragStartMode; } /** * Add component to layout. Will wrap component with DragAndDropWrapper. * @param component Component added */ public void addComponentToLayout(Component component) { addComponentToLayout(component, getMasonryLayout().getComponentCount()); } /** * Add component to layout to given index. Will wrap component with DragAndDropWrapper. * @param component Component added * @param index Index where component is added */ public void addComponentToLayout(Component component, int index) { addComponentToLayout(component, null, index); } /** * Add component to layout with given wrapper style name. Will wrap component with DragAndDropWrapper. * @param component Component added * @param wrapperStyleName Style name added to wrapper (eq. use to define it take double width) */ public void addComponentToLayout(Component component, String wrapperStyleName) { addComponentToLayout(component, wrapperStyleName, getMasonryLayout().getComponentCount()); } /** * Add component to layout with given wrapper style name to given index. Will wrap component with * DragAndDropWrapper. * @param component Component added * @param wrapperStyleName Style name added to wrapper (eq. use to define it take double width) * @param index Index where component is added */ public void addComponentToLayout(Component component, String wrapperStyleName, int index) { getMasonryLayout().addComponent(createComponentDnDWrapper(component, wrapperStyleName), wrapperStyleName, index); } /** * Remove component from layout * @param component Component remvoed */ public void removeComponentInLayout(Component component) { DragAndDropWrapper wrapper = getComponentDnDWrapper(component); if (wrapper != null) { getMasonryLayout().removeComponent(wrapper); } } /** * Replace component with another component * @param oldComponent Old component replaced * @param newComponent New component used as replacement */ public void replaceComponentInLayout(Component oldComponent, Component newComponent) { DragAndDropWrapper oldWrapper = getComponentDnDWrapper(oldComponent); if (oldWrapper == null) { throw new IllegalArgumentException("Given component not found"); } DragAndDropWrapper newWrapper = createComponentDnDWrapper(newComponent, null); getMasonryLayout().replaceComponent(oldWrapper, newWrapper); } /** * Request client side to re-layout. Useful if component sizes have changed. */ public void requestLayout() { getMasonryLayout().requestLayout(); } /** * Create DnD wrapper for component if not yet defined * @param component Component wrapped * @return Wrapper made or found */ protected DragAndDropWrapper createComponentDnDWrapper(Component component, String wrapperStyleName) { DragAndDropWrapper wrapper = getComponentDnDWrapper(component); if (wrapper == null) { wrapper = new DragAndDropWrapper(component); wrapper.addStyleName("masonry-dnd-wrapper"); if (wrapperStyleName != null) { wrapper.addStyleName(wrapperStyleName); } wrapper.setDragStartMode(allowReorder ? getComponentDragStartMode() : DragStartMode.NONE); wrapper.setDropHandler(createDropHandlerForComponents(wrapper)); } return wrapper; } /** * Override to have custom drop handler for your items * @param childWrapper Wrapper of child component that need DropHandler * @return DropHandler made for childWrapper */ protected DropHandler createDropHandlerForComponents(DragAndDropWrapper childWrapper) { return new MasonryDropHandler(this, childWrapper); } /** * Get DragAndDropWrapper for given component * @param component Component added to this DndMasonryLayout * @return Wrapper of component or null if not set yet, or if component is not under this layout */ protected DragAndDropWrapper getComponentDnDWrapper(Component component) { if (!(component.getParent() instanceof DragAndDropWrapper)) { return null; } DragAndDropWrapper wrapper = (DragAndDropWrapper) component.getParent(); if (wrapper.getParent() != this) { return null; } return wrapper; } /** * Get access to Masonry layout inside DnDMasonryLayout. This is protected to prevent messing up wrapper structure * this component is trying to maintain. If you really need to access this, inherit class and call it that way. Just * be warned issues might follow. * @return MasonryLayout wrapped inside this DragAndDropWrapper */ protected MasonryLayout getMasonryLayout() { return (MasonryLayout) super.getCompositionRoot(); } /** * Get component in layout with given index. When you index components with this, use * {@link #getComponentCountInLayout()} as indexing limit. * @param index Index of component searched * @return Component at given index */ public Component getComponentInLayout(int index) { DragAndDropWrapper wrapper = (DragAndDropWrapper) getMasonryLayout().getComponent(index); if (wrapper != null) { return wrapper.iterator().next(); } else { return null; } } /** * Use this number when you want to know how many components have been added to layout. * @return Number of components in layout */ public int getComponentCountInLayout() { return getMasonryLayout().getComponentCount(); } /** * Add component to layout with given wrapper style name to given index. Will wrap component with * DragAndDropWrapper. * @param component Component added * @param wrapperStyleName Style name added to wrapper (eq. use to define it take double width) */ public void addComponentFirst(Component component, String wrapperStyleName) { addComponentToLayout(component, wrapperStyleName, 0); } /** * Set if user is allowed to reorder components by dragging them. This does not affect to server APIs. * @param reorderable true to allow, false to disallow */ public void setReorderable(boolean reorderable) { allowReorder = reorderable; // Update drag start modes for (int i = 0; i < getMasonryLayout().getComponentCount(); ++i) { Component child = getMasonryLayout().getComponent(i); if (child instanceof DragAndDropWrapper) { DragAndDropWrapper wrapper = (DragAndDropWrapper) child; wrapper.setDragStartMode(allowReorder ? getComponentDragStartMode() : DragStartMode.NONE); } } } /** * If user is allowed to reorder components by dragging them * @return true if allowed, false if not */ public boolean isReorderable() { return allowReorder; } /** * Remove all components from layout */ public void removeAllComponentsFromLayout() { getMasonryLayout().removeAllComponents(); } /** * Add reorder listener that will be called when user has reordered components * @param listener Listener added */ public void addMasonryReorderListener(MasonryReorderListener listener) { reorderListeners.add(listener); } /** * Remove reorder listener that will be called when user has reordered components * @param listener Listener removed */ public void removeMasonryReorderListener(MasonryReorderListener listener) { reorderListeners.remove(listener); } /** * Notify all listeners that user has changed the order of components */ protected void notifyReorderListeners() { MasonryReorderEvent event = new MasonryReorderEvent(this); //TODO: add more data to event for (MasonryReorderListener listener : reorderListeners) { listener.onUserReorder(event); } } /** * Get column width used with this masonry layout * @return Column width in pixels */ public int getColumnWidth() { return getMasonryLayout().getColumnWidth(); } /** * Adds style name to internal layout * @param styleName Style name added */ public void addStyleNameToLayout(String styleName) { getMasonryLayout().addStyleName(styleName); } /** * Removes style name from internal layout * @param styleName Style name removed */ public void removeStyleNameFromLayout(String styleName) { getMasonryLayout().removeStyleName(styleName); } /** * Define if client side should automatically relayout when images are loaded. Use this when you images with * undefined heights in your layouts. * @param relayout if true client side will relayout automatically when images loaded */ public void setAutomaticLayoutWhenImagesLoaded(boolean relayout) { getMasonryLayout().setAutomaticLayoutWhenImagesLoaded(relayout); } /** * Check if client side is hooked to relayout when images are loaded. * @return true if client side will automatically relayout when images loaded. */ public boolean isAutomaticLayoutWhenImagesLoaded() { return getMasonryLayout().isAutomaticLayoutWhenImagesLoaded(); } }