/* * Copyright 2000-2014 Vaadin Ltd. * * 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 com.vaadin.data.util; import java.io.Serializable; import java.util.ArrayList; import java.util.Collection; import java.util.Collections; import java.util.HashMap; import java.util.HashSet; import java.util.LinkedHashSet; import java.util.List; import java.util.Map; import java.util.Map.Entry; import java.util.Set; import com.google.gwt.thirdparty.guava.common.collect.Sets; import com.vaadin.data.Container; import com.vaadin.data.Item; import com.vaadin.data.Property; import com.vaadin.data.sort.SortOrder; import com.vaadin.data.util.filter.UnsupportedFilterException; import com.vaadin.shared.data.sort.SortDirection; /** * Container wrapper that adds support for generated properties. This container * only supports adding new generated properties. Adding new normal properties * should be done for the wrapped container. * *

* Removing properties from this container does not remove anything from the * wrapped container but instead only hides them from the results. These * properties can be returned to this container by calling * {@link #addContainerProperty(Object, Class, Object)} with same property id * which was removed. * *

* If wrapped container is Filterable and/or Sortable it should only be handled * through this container as generated properties need to be handled in a * specific way when sorting/filtering. * *

* Items returned by this container do not support adding or removing * properties. Generated properties are always read-only. Trying to make them * editable throws an exception. * * @since 7.4 * @author Vaadin Ltd */ public class GeneratedPropertyContainer extends AbstractContainer implements Container.Indexed, Container.Sortable, Container.Filterable, Container.PropertySetChangeNotifier, Container.ItemSetChangeNotifier { private final Container.Indexed wrappedContainer; private final Map> propertyGenerators; private final Map> activeFilters; private Sortable sortableContainer = null; private Filterable filterableContainer = null; /* Removed properties which are hidden but not actually removed */ private final Set removedProperties = new HashSet(); /** * Property implementation for generated properties */ protected static class GeneratedProperty implements Property { private Item item; private Object itemId; private Object propertyId; private PropertyValueGenerator generator; public GeneratedProperty(Item item, Object propertyId, Object itemId, PropertyValueGenerator generator) { this.item = item; this.itemId = itemId; this.propertyId = propertyId; this.generator = generator; } @Override public T getValue() { return generator.getValue(item, itemId, propertyId); } @Override public void setValue(T newValue) throws ReadOnlyException { throw new ReadOnlyException("Generated properties are read only"); } @Override public Class getType() { return generator.getType(); } @Override public boolean isReadOnly() { return true; } @Override public void setReadOnly(boolean newStatus) { if (newStatus) { // No-op return; } throw new UnsupportedOperationException( "Generated properties are read only"); } } /** * Item implementation for generated properties. */ protected class GeneratedPropertyItem implements Item { private Item wrappedItem; private Object itemId; protected GeneratedPropertyItem(Object itemId, Item item) { this.itemId = itemId; wrappedItem = item; } @Override public Property getItemProperty(Object id) { if (propertyGenerators.containsKey(id)) { return createProperty(wrappedItem, id, itemId, propertyGenerators.get(id)); } return wrappedItem.getItemProperty(id); } @Override public Collection getItemPropertyIds() { Set wrappedProperties = asSet(wrappedItem.getItemPropertyIds()); return Sets.union( Sets.difference(wrappedProperties, removedProperties), propertyGenerators.keySet()); } @Override public boolean addItemProperty(Object id, Property property) throws UnsupportedOperationException { throw new UnsupportedOperationException( "GeneratedPropertyItem does not support adding properties"); } @Override public boolean removeItemProperty(Object id) throws UnsupportedOperationException { throw new UnsupportedOperationException( "GeneratedPropertyItem does not support removing properties"); } /** * Tests if the given object is the same as the this object. Two Items * from the same container with the same ID are equal. * * @param obj * an object to compare with this object * @return true if the given object is the same as this * object, false if not */ @Override public boolean equals(Object obj) { if (this == obj) { return true; } if (obj == null || !obj.getClass().equals(GeneratedPropertyItem.class)) { return false; } final GeneratedPropertyItem li = (GeneratedPropertyItem) obj; return getContainer() == li.getContainer() && itemId.equals(li.itemId); } @Override public int hashCode() { return itemId.hashCode(); } private GeneratedPropertyContainer getContainer() { return GeneratedPropertyContainer.this; } }; /** * Base implementation for item add or remove events. This is used when an * event is fired from wrapped container and needs to be reconstructed to * act like it actually came from this container. */ protected abstract class GeneratedItemAddOrRemoveEvent implements Serializable { private Object firstItemId; private int firstIndex; private int count; protected GeneratedItemAddOrRemoveEvent(Object itemId, int first, int count) { firstItemId = itemId; firstIndex = first; this.count = count; } public Container getContainer() { return GeneratedPropertyContainer.this; } public Object getFirstItemId() { return firstItemId; } public int getFirstIndex() { return firstIndex; } public int getAffectedItemsCount() { return count; } }; protected class GeneratedItemRemoveEvent extends GeneratedItemAddOrRemoveEvent implements ItemRemoveEvent { protected GeneratedItemRemoveEvent(ItemRemoveEvent event) { super(event.getFirstItemId(), event.getFirstIndex(), event .getRemovedItemsCount()); } @Override public int getRemovedItemsCount() { return super.getAffectedItemsCount(); } } protected class GeneratedItemAddEvent extends GeneratedItemAddOrRemoveEvent implements ItemAddEvent { protected GeneratedItemAddEvent(ItemAddEvent event) { super(event.getFirstItemId(), event.getFirstIndex(), event .getAddedItemsCount()); } @Override public int getAddedItemsCount() { return super.getAffectedItemsCount(); } } /** * Constructor for GeneratedPropertyContainer. * * @param container * underlying indexed container */ public GeneratedPropertyContainer(Container.Indexed container) { wrappedContainer = container; propertyGenerators = new HashMap>(); if (wrappedContainer instanceof Sortable) { sortableContainer = (Sortable) wrappedContainer; } if (wrappedContainer instanceof Filterable) { activeFilters = new HashMap>(); filterableContainer = (Filterable) wrappedContainer; } else { activeFilters = null; } // ItemSetChangeEvents if (wrappedContainer instanceof ItemSetChangeNotifier) { ((ItemSetChangeNotifier) wrappedContainer) .addItemSetChangeListener(new ItemSetChangeListener() { @Override public void containerItemSetChange( ItemSetChangeEvent event) { if (event instanceof ItemAddEvent) { final ItemAddEvent addEvent = (ItemAddEvent) event; fireItemSetChange(new GeneratedItemAddEvent( addEvent)); } else if (event instanceof ItemRemoveEvent) { final ItemRemoveEvent removeEvent = (ItemRemoveEvent) event; fireItemSetChange(new GeneratedItemRemoveEvent( removeEvent)); } else { fireItemSetChange(); } } }); } // PropertySetChangeEvents if (wrappedContainer instanceof PropertySetChangeNotifier) { ((PropertySetChangeNotifier) wrappedContainer) .addPropertySetChangeListener(new PropertySetChangeListener() { @Override public void containerPropertySetChange( PropertySetChangeEvent event) { fireContainerPropertySetChange(); } }); } } /* Functions related to generated properties */ /** * Add a new PropertyValueGenerator with given property id. This will * override any existing properties with the same property id. Fires a * PropertySetChangeEvent. * * @param propertyId * property id * @param generator * a property value generator */ public void addGeneratedProperty(Object propertyId, PropertyValueGenerator generator) { propertyGenerators.put(propertyId, generator); fireContainerPropertySetChange(); } /** * Removes any possible PropertyValueGenerator with given property id. Fires * a PropertySetChangeEvent. * * @param propertyId * property id */ public void removeGeneratedProperty(Object propertyId) { if (propertyGenerators.containsKey(propertyId)) { propertyGenerators.remove(propertyId); fireContainerPropertySetChange(); } } private Item createGeneratedPropertyItem(final Object itemId, final Item item) { return new GeneratedPropertyItem(itemId, item); } private Property createProperty(final Item item, final Object propertyId, final Object itemId, final PropertyValueGenerator generator) { return new GeneratedProperty(item, propertyId, itemId, generator); } private static LinkedHashSet asSet(Collection collection) { if (collection instanceof LinkedHashSet) { return (LinkedHashSet) collection; } else { return new LinkedHashSet(collection); } } /* Listener functionality */ @Override public void addItemSetChangeListener(ItemSetChangeListener listener) { super.addItemSetChangeListener(listener); } @Override public void addListener(ItemSetChangeListener listener) { super.addListener(listener); } @Override public void removeItemSetChangeListener(ItemSetChangeListener listener) { super.removeItemSetChangeListener(listener); } @Override public void removeListener(ItemSetChangeListener listener) { super.removeListener(listener); } @Override public void addPropertySetChangeListener(PropertySetChangeListener listener) { super.addPropertySetChangeListener(listener); } @Override public void addListener(PropertySetChangeListener listener) { super.addListener(listener); } @Override public void removePropertySetChangeListener( PropertySetChangeListener listener) { super.removePropertySetChangeListener(listener); } @Override public void removeListener(PropertySetChangeListener listener) { super.removeListener(listener); } /* Filtering functionality */ @Override public void addContainerFilter(Filter filter) throws UnsupportedFilterException { if (filterableContainer == null) { throw new UnsupportedOperationException( "Wrapped container is not filterable"); } List addedFilters = new ArrayList(); for (Entry> entry : propertyGenerators .entrySet()) { Object property = entry.getKey(); if (filter.appliesToProperty(property)) { // Have generated property modify filter to fit the original // data in the container. Filter modifiedFilter = entry.getValue().modifyFilter(filter); filterableContainer.addContainerFilter(modifiedFilter); // Keep track of added filters addedFilters.add(modifiedFilter); } } if (addedFilters.isEmpty()) { // No generated property modified this filter, use it as is addedFilters.add(filter); filterableContainer.addContainerFilter(filter); } // Map filter to actually added filters activeFilters.put(filter, addedFilters); } @Override public void removeContainerFilter(Filter filter) { if (filterableContainer == null) { throw new UnsupportedOperationException( "Wrapped container is not filterable"); } if (activeFilters.containsKey(filter)) { for (Filter f : activeFilters.get(filter)) { filterableContainer.removeContainerFilter(f); } activeFilters.remove(filter); } } @Override public void removeAllContainerFilters() { if (filterableContainer == null) { throw new UnsupportedOperationException( "Wrapped container is not filterable"); } filterableContainer.removeAllContainerFilters(); activeFilters.clear(); } @Override public Collection getContainerFilters() { if (filterableContainer == null) { throw new UnsupportedOperationException( "Wrapped container is not filterable"); } return Collections.unmodifiableSet(activeFilters.keySet()); } /* Sorting functionality */ @Override public void sort(Object[] propertyId, boolean[] ascending) { if (sortableContainer == null) { throw new UnsupportedOperationException( "Wrapped container is not Sortable"); } if (propertyId.length == 0) { sortableContainer.sort(propertyId, ascending); return; } List actualSortProperties = new ArrayList(); List actualSortDirections = new ArrayList(); for (int i = 0; i < propertyId.length; ++i) { Object property = propertyId[i]; SortDirection direction; boolean isAscending = i < ascending.length ? ascending[i] : true; if (isAscending) { direction = SortDirection.ASCENDING; } else { direction = SortDirection.DESCENDING; } if (propertyGenerators.containsKey(property)) { // Sorting by a generated property. Generated property should // modify sort orders to work with original properties in the // container. for (SortOrder s : propertyGenerators.get(property) .getSortProperties(new SortOrder(property, direction))) { actualSortProperties.add(s.getPropertyId()); actualSortDirections .add(s.getDirection() == SortDirection.ASCENDING); } } else { actualSortProperties.add(property); actualSortDirections.add(isAscending); } } boolean[] actualAscending = new boolean[actualSortDirections.size()]; for (int i = 0; i < actualAscending.length; ++i) { actualAscending[i] = actualSortDirections.get(i); } sortableContainer.sort(actualSortProperties.toArray(), actualAscending); } @Override public Collection getSortableContainerPropertyIds() { if (sortableContainer == null) { return Collections.emptySet(); } Set sortablePropertySet = new HashSet( sortableContainer.getSortableContainerPropertyIds()); for (Entry> entry : propertyGenerators .entrySet()) { Object property = entry.getKey(); SortOrder order = new SortOrder(property, SortDirection.ASCENDING); if (entry.getValue().getSortProperties(order).length > 0) { sortablePropertySet.add(property); } else { sortablePropertySet.remove(property); } } return sortablePropertySet; } /* Item related overrides */ @Override public Item addItemAfter(Object previousItemId, Object newItemId) throws UnsupportedOperationException { Item item = wrappedContainer.addItemAfter(previousItemId, newItemId); if (item == null) { return null; } return createGeneratedPropertyItem(newItemId, item); } @Override public Item addItem(Object itemId) throws UnsupportedOperationException { Item item = wrappedContainer.addItem(itemId); if (item == null) { return null; } return createGeneratedPropertyItem(itemId, item); } @Override public Item addItemAt(int index, Object newItemId) throws UnsupportedOperationException { Item item = wrappedContainer.addItemAt(index, newItemId); if (item == null) { return null; } return createGeneratedPropertyItem(newItemId, item); } @Override public Item getItem(Object itemId) { Item item = wrappedContainer.getItem(itemId); if (item == null) { return null; } return createGeneratedPropertyItem(itemId, item); } /* Property related overrides */ @Override public Property getContainerProperty(Object itemId, Object propertyId) { if (propertyGenerators.keySet().contains(propertyId)) { return getItem(itemId).getItemProperty(propertyId); } else if (!removedProperties.contains(propertyId)) { return wrappedContainer.getContainerProperty(itemId, propertyId); } return null; } /** * Returns a list of propety ids available in this container. This * collection will contain properties for generated properties. Removed * properties will not show unless there is a generated property overriding * those. */ @Override public Collection getContainerPropertyIds() { Set wrappedProperties = asSet(wrappedContainer .getContainerPropertyIds()); return Sets.union( Sets.difference(wrappedProperties, removedProperties), propertyGenerators.keySet()); } /** * Adds a previously removed property back to GeneratedPropertyContainer. * Adding a property that is not previously removed causes an * UnsupportedOperationException. */ @Override public boolean addContainerProperty(Object propertyId, Class type, Object defaultValue) throws UnsupportedOperationException { if (!removedProperties.contains(propertyId)) { throw new UnsupportedOperationException( "GeneratedPropertyContainer does not support adding properties."); } removedProperties.remove(propertyId); fireContainerPropertySetChange(); return true; } /** * Marks the given property as hidden. This property from wrapped container * will be removed from {@link #getContainerPropertyIds()} and is no longer * be available in Items retrieved from this container. */ @Override public boolean removeContainerProperty(Object propertyId) throws UnsupportedOperationException { if (wrappedContainer.getContainerPropertyIds().contains(propertyId) && removedProperties.add(propertyId)) { fireContainerPropertySetChange(); return true; } return false; } /* Type related overrides */ @Override public Class getType(Object propertyId) { if (propertyGenerators.containsKey(propertyId)) { return propertyGenerators.get(propertyId).getType(); } else { return wrappedContainer.getType(propertyId); } } /* Unmodified functions */ @Override public Object nextItemId(Object itemId) { return wrappedContainer.nextItemId(itemId); } @Override public Object prevItemId(Object itemId) { return wrappedContainer.prevItemId(itemId); } @Override public Object firstItemId() { return wrappedContainer.firstItemId(); } @Override public Object lastItemId() { return wrappedContainer.lastItemId(); } @Override public boolean isFirstId(Object itemId) { return wrappedContainer.isFirstId(itemId); } @Override public boolean isLastId(Object itemId) { return wrappedContainer.isLastId(itemId); } @Override public Object addItemAfter(Object previousItemId) throws UnsupportedOperationException { return wrappedContainer.addItemAfter(previousItemId); } @Override public Collection getItemIds() { return wrappedContainer.getItemIds(); } @Override public int size() { return wrappedContainer.size(); } @Override public boolean containsId(Object itemId) { return wrappedContainer.containsId(itemId); } @Override public Object addItem() throws UnsupportedOperationException { return wrappedContainer.addItem(); } @Override public boolean removeItem(Object itemId) throws UnsupportedOperationException { return wrappedContainer.removeItem(itemId); } @Override public boolean removeAllItems() throws UnsupportedOperationException { return wrappedContainer.removeAllItems(); } @Override public int indexOfId(Object itemId) { return wrappedContainer.indexOfId(itemId); } @Override public Object getIdByIndex(int index) { return wrappedContainer.getIdByIndex(index); } @Override public List getItemIds(int startIndex, int numberOfItems) { return wrappedContainer.getItemIds(startIndex, numberOfItems); } @Override public Object addItemAt(int index) throws UnsupportedOperationException { return wrappedContainer.addItemAt(index); } /** * Returns the original underlying container. * * @return the original underlying container */ public Container.Indexed getWrappedContainer() { return wrappedContainer; } }