org.dellroad.stuff.vaadin7.AbstractQueryContainer Maven / Gradle / Ivy
Show all versions of dellroad-stuff-vaadin7 Show documentation
/*
* Copyright (C) 2022 Archie L. Cobbs. All rights reserved.
*/
package org.dellroad.stuff.vaadin7;
import com.vaadin.data.Container;
import com.vaadin.data.Item;
import com.vaadin.data.Property;
import com.vaadin.data.util.AbstractContainer;
import java.util.AbstractList;
import java.util.Collection;
import java.util.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Set;
/**
* Support superclass for read-only {@link Container} implementations where each {@link Item} in the container
* is backed by a Java object, and the Java objects are generated via {@linkplain #query a query} that returns
* a {@link QueryList} containing only some portion of the total list of backing objects at any one time.
*
*
* This {@link Container}'s {@link Property}s are defined via {@link PropertyDef}s, and a {@link PropertyExtractor}
* is used to actually extract the property values from each underlying object (alternately, subclasses can override
* {@link #getPropertyValue getPropertyValue()}). However, the easist way to configure the container {@link Property}s
* is to pass a {@link ProvidesProperty @ProvidesProperty}-annotated Java class to the {@link #AbstractQueryContainer(Class)}
* constructor.
*
*
* This {@link Container}'s item ID's are simply the indexes of the corresponding objects in the overall list. The {@link QueryList}
* interface is designed for scalability; at minimum, it is required to provide only the size of the total list
* (which may only be an estimate) and one backing object at a specified index.
*
*
* This class will invoke {@link #query} as needed to (re)generate the {@link QueryList}; the {@link QueryList} is then cached.
* However, if any invocation of {@link QueryList#get} throws an {@link InvalidQueryListException}, then the cached
* list is discarded and {@link #query} is invoked again to regenerate it. In this way, the {@link QueryList} is
* allowed to decide, on demand, when it is invalid or incapable of providing a specific list member. For example,
* when using JPA, a list may be considered invalid if the current EntityManager session has changed.
*
*
* Note that the {@link QueryList} being invalid is an orthogonal concept from the contents of the list having changed.
* Invalid means "this list can no longer be used" while changed means "this list contains out-of-date information".
* Normally, the latter implies the former (but not vice-versa). The list becoming invalid does not in itself not cause
* any notifications to be sent, so no new query will be performed until e.g. the user interface explicitly requests
* more information.
*
*
* Therefore, if the list size or content changes, first invoke {@link #invalidate} to discard the cached {@link QueryList},
* and then {@link #fireItemSetChange} to notify listeners; for convenience, {@link #reload} will perform these two steps for
* you. The new size and content will be provided by the {@link QueryList} returned by the next invocation of {@link #query}.
*
*
* The subclass may forcibly invalidate the current {@link QueryList} via {@link #invalidate}; this merely discards it
* and will force a new invocation of {@link #query} on the next container access. In many situations, however,
* the use of {@link #invalidate} is never required.
*
*
* For scalability reasons the {@link QueryList} may actually only contain a portion of the list, throwing
* {@link InvalidQueryListException}s when other list members are accessed. An "index hint" parameter provided to
* {@link #query} indicates which member of the container is of current interest. The returned {@link QueryList}
* is required to provide exception-free access only to the indicated member, so in the extreme case only a single
* list member could be kept. In practice, normally range of members near to the index hint would be kept;
* see for example {@link WindowQueryList}.
*
* @param the type of the Java objects that back each {@link Item} in the container
* @see QueryList
* @see SimpleQueryList
* @see WindowQueryList
* @see AbstractUnsizedContainer
*/
@SuppressWarnings({ "serial", "deprecation" })
public abstract class AbstractQueryContainer extends AbstractContainer implements PropertyExtractor,
Container.Ordered, Container.Indexed, Container.PropertySetChangeNotifier, Container.ItemSetChangeNotifier, Connectable {
private final HashMap> propertyMap = new HashMap<>();
private PropertyExtractor super T> propertyExtractor;
private final HashMap> itemMap = new HashMap<>();
private QueryList extends T> queryList;
private long totalSize = -1;
// Constructors
/**
* Constructor.
*
*
* After using this constructor, subsequent invocations of {@link #setPropertyExtractor setPropertyExtractor()}
* and {@link #setProperties setProperties()} are required to define the properties of this container
* and how to extract them.
*/
protected AbstractQueryContainer() {
this((PropertyExtractor)null);
}
/**
* Constructor.
*
*
* After using this constructor, a subsequent invocation of {@link #setProperties setProperties()} is required
* to define the properties of this container.
*
* @param propertyExtractor used to extract properties from the underlying Java objects;
* may be null but then container is not usable until one is configured via
* {@link #setPropertyExtractor setPropertyExtractor()}
*/
protected AbstractQueryContainer(PropertyExtractor super T> propertyExtractor) {
this(propertyExtractor, null);
}
/**
* Constructor.
*
*
* After using this constructor, a subsequent invocation of {@link #setPropertyExtractor setPropertyExtractor()}
* is required to define how to extract the properties of this container; alternately, subclasses can override
* {@link #getPropertyValue getPropertyValue()}.
*
* @param propertyDefs container property definitions; null is treated like the empty set
*/
protected AbstractQueryContainer(Collection extends PropertyDef>> propertyDefs) {
this(null, propertyDefs);
}
/**
* Constructor.
*
* @param propertyExtractor used to extract properties from the underlying Java objects;
* may be null but then container is not usable until one is configured via
* {@link #setPropertyExtractor setPropertyExtractor()}
* @param propertyDefs container property definitions; null is treated like the empty set
*/
protected AbstractQueryContainer(PropertyExtractor super T> propertyExtractor,
Collection extends PropertyDef>> propertyDefs) {
this.setPropertyExtractor(propertyExtractor);
this.setProperties(propertyDefs);
}
/**
* Constructor.
*
*
* Properties will be determined by the {@link ProvidesProperty @ProvidesProperty} and
* {@link ProvidesPropertySort @ProvidesPropertySort} annotated methods in the given class.
*
* @param type class to introspect for annotated methods
* @throws IllegalArgumentException if {@code type} is null
* @throws IllegalArgumentException if {@code type} has two {@link ProvidesProperty @ProvidesProperty}
* or {@link ProvidesPropertySort @ProvidesPropertySort} annotated methods for the same property
* @throws IllegalArgumentException if a {@link ProvidesProperty @ProvidesProperty}-annotated method with no
* {@linkplain ProvidesProperty#value property name specified} has a name which cannot be interpreted as a bean
* property "getter" method
* @see ProvidesProperty
* @see ProvidesPropertySort
* @see ProvidesPropertyScanner
*/
@SuppressWarnings({ "unchecked", "rawtypes" })
protected AbstractQueryContainer(Class super T> type) {
// Why the JLS forces this stupid cast:
// http://stackoverflow.com/questions/4902723/why-cant-a-java-type-parameter-have-a-lower-bound
final ProvidesPropertyScanner super T> propertyReader
= (ProvidesPropertyScanner super T>)new ProvidesPropertyScanner(type);
this.setPropertyExtractor(propertyReader.getPropertyExtractor());
this.setProperties(propertyReader.getPropertyDefs());
}
// Public methods
/**
* Get the configured {@link PropertyExtractor} for this container.
*
* @return the configured {@link PropertyExtractor}
*/
public PropertyExtractor super T> getPropertyExtractor() {
return this.propertyExtractor;
}
/**
* Change the configured {@link PropertyExtractor} for this container.
* Invoking this method does not result in any container notifications.
*
* @param propertyExtractor used to extract properties from the underlying Java objects;
* may be null but the container is not usable without one
*/
public void setPropertyExtractor(PropertyExtractor super T> propertyExtractor) {
this.propertyExtractor = propertyExtractor;
}
/**
* Read the value of the property defined by {@code propertyDef} from the given object.
*
*
* The implementation in {@link AbstractQueryContainer} just delegates to the {@linkplain #setPropertyExtractor configured}
* {@link PropertyExtractor}; subclasses may override to customize property extraction.
*
* @param obj Java object
* @param propertyDef definition of which property to read
* @throws NullPointerException if either parameter is null
* @throws IllegalStateException if no {@link PropertyExtractor} is configured for this container
*/
@Override
public V getPropertyValue(T obj, PropertyDef propertyDef) {
if (this.propertyExtractor == null)
throw new IllegalStateException("no PropertyExtractor is configured for this container");
return this.propertyExtractor.getPropertyValue(obj, propertyDef);
}
/**
* Change the configured properties of this container.
*
* @param propertyDefs container property definitions; null is treated like the empty set
* @throws IllegalArgumentException if {@code propertyDefs} contains a property with a duplicate name
*/
public void setProperties(Collection extends PropertyDef>> propertyDefs) {
if (propertyDefs == null)
propertyDefs = Collections.>emptySet();
this.propertyMap.clear();
for (PropertyDef> propertyDef : propertyDefs) {
if (this.propertyMap.put(propertyDef.getName(), propertyDef) != null)
throw new IllegalArgumentException("duplicate property name `" + propertyDef.getName() + "'");
}
this.fireContainerPropertySetChange();
}
/**
* Reload this container.
*
*
* This discards the current cached {@link QueryList} (if any) and fires an item set change event.
*/
public void reload() {
this.invalidate();
this.fireItemSetChange();
}
// Connectable
/**
* Connect this instance to non-Vaadin resources.
*
*
* The implementation in {@link AbstractQueryContainer} does nothing.
*
* @throws IllegalStateException if there is no {@link com.vaadin.server.VaadinSession} associated with the current thread
*/
@Override
public void connect() {
}
/**
* Disconnect this instance from non-Vaadin resources.
*
*
* The implementation in {@link AbstractQueryContainer} does nothing.
*
* @throws IllegalStateException if there is no {@link com.vaadin.server.VaadinSession} associated with the current thread
*/
@Override
public void disconnect() {
}
// Subclass hooks and methods
/**
* Perform a query to (re)generate the list of Java objects backing this container.
*
*
* The particular position in the list we are interested in is given as a hint by the {@code hint} parameter.
* That is, an invocation of {@link QueryList#get}{@code (hint)} is likely immediately after this method
* returns and if so it must complete without throwing an exception, unless {@code hint} is out of range.
*
*
* The {@code hint} can be used to implement a highly scalable query list containing external objects
* (such as from a database) where only a small "window" of objects is actually kept in memory at any one time.
* Of course, implementations are also free to ignore {@code hint}. However, the returned {@link QueryList}
* must at least tolerate one invocation of {@link QueryList#get get}{@code (hint)} without throwing an exception
* when {@code hint} is less that the {@link QueryList#size size()} of the returned {@link QueryList}.
*
* @param hint index of the list element we are interested in
* @return list of Java objects backing this container
*/
protected abstract QueryList extends T> query(long hint);
/**
* Discard the current cached {@link QueryList}, if any.
*/
protected void invalidate() {
this.queryList = null;
this.itemMap.clear();
}
/**
* Get the Java backing object at the given index in the list.
*
* @param index list index
* @return backing object, or null if {@code index} is out of range
*/
protected T getJavaObject(int index) {
Exception exception;
long currentSize;
int attempt = 1;
while (true) {
currentSize = this.ensureList(index);
if (index < 0 || index >= currentSize)
return null;
try {
return this.queryList.get(index);
} catch (InvalidQueryListException e) {
this.invalidate();
if (attempt == 100) { // avoid infinite loops
exception = e;
break;
}
} catch (IndexOutOfBoundsException e) {
exception = e;
break;
}
attempt++;
}
// The QueryList is behaving badly
throw new RuntimeException("query(" + index + ") returned a QueryList with size() = "
+ currentSize + " but QueryList.get(" + index + ") failed (attempt #" + attempt + ")", exception);
}
/**
* Ensure we have a cached query list.
*
* @param hint index of the list element we are interested in, passed to {@link #query} if no query list is cached
* @return the size of the cached query list
*/
protected long ensureList(int hint) {
if (this.queryList == null) {
this.queryList = this.query(hint);
final long oldTotalSize = this.totalSize;
final long newTotalSize = this.queryList.size();
this.totalSize = newTotalSize;
if (oldTotalSize != -1 && newTotalSize != oldTotalSize)
this.handleSizeChange();
}
return this.totalSize;
}
/**
* Invoked when a new {@link QueryList} has returned a changed {@link QueryList#size size()}
* for the underlying list.
*
*
* Normally, this implies an item set change notification will be generated elsewhere and so
* no additional action needs to be taken by this method. However, some implementations may
* lack such a mechanism, for example, when the container's size is only ever calculated when
* {@link #query} is invoked. In such cases, this method may trigger a notification.
*
*
* Note: to avoid re-entrancy problems, this method should not send out any notifications itself;
* instead, it may schedule notifications to be delivered later, e.g., via {@link VaadinUtil#invokeLater}.
*
*
* The implementation in {@link AbstractQueryContainer} does nothing, assuming the notification is handled
* elsewhere. Subclasses may override if needed.
*/
protected void handleSizeChange() {
}
/**
* Create a {@link BackedItem} for the given backing Java object.
*
*
* The implementation in {@link AbstractQueryContainer} returns
* {@code new SimpleItem(object, propertyDefs, propertyExtractor)}.
*
* @param object underlying Java object
* @param propertyDefs property definitions
* @param propertyExtractor extracts the property value from {@code object}
* @return new {@link BackedItem}
* @throws IllegalArgumentException if any parameter is null
*/
protected BackedItem createBackedItem(T object, Collection> propertyDefs,
PropertyExtractor super T> propertyExtractor) {
return new SimpleItem<>(object, propertyDefs, propertyExtractor);
}
// Container
@Override
public BackedItem getItem(Object itemId) {
if (!(itemId instanceof Integer))
return null;
int index = ((Integer)itemId).intValue();
T obj = this.getJavaObject(index);
if (obj == null)
return null;
BackedItem item = this.itemMap.get(index);
if (item == null) {
item = this.createBackedItem(obj, this.propertyMap.values(), this);
this.itemMap.put(index, item);
}
return item;
}
@Override
public Collection getItemIds() {
return new IntList(this.size());
}
@Override
public Set getContainerPropertyIds() {
return Collections.unmodifiableSet(this.propertyMap.keySet());
}
@Override
@SuppressWarnings("rawtypes")
public Property/*>*/ getContainerProperty(Object itemId, Object propertyId) {
final BackedItem item = this.getItem(itemId);
return item != null ? item.getItemProperty(propertyId) : null;
}
@Override
public Class> getType(Object propertyId) {
PropertyDef> propertyDef = this.propertyMap.get(propertyId);
return propertyDef != null ? propertyDef.getType() : null;
}
@Override
public int size() {
return (int)this.ensureList(0);
}
@Override
public boolean containsId(Object itemId) {
if (!(itemId instanceof Integer))
return false;
int index = ((Integer)itemId).intValue();
return index >= 0 && index < this.ensureList(index);
}
/**
* @throws UnsupportedOperationException always
*/
@Override
public Item addItem(Object itemId) {
throw new UnsupportedOperationException();
}
/**
* @throws UnsupportedOperationException always
*/
@Override
public Item addItem() {
throw new UnsupportedOperationException();
}
/**
* @throws UnsupportedOperationException always
*/
@Override
public boolean removeItem(Object itemId) {
throw new UnsupportedOperationException();
}
/**
* @throws UnsupportedOperationException always
*/
@Override
public boolean addContainerProperty(Object propertyId, Class> type, Object defaultValue) {
throw new UnsupportedOperationException();
}
/**
* @throws UnsupportedOperationException always
*/
@Override
public boolean removeContainerProperty(Object propertyId) {
throw new UnsupportedOperationException();
}
/**
* @throws UnsupportedOperationException always
*/
@Override
public boolean removeAllItems() {
throw new UnsupportedOperationException();
}
// Container.Indexed
/**
* @throws UnsupportedOperationException always
*/
@Override
public Object addItemAt(int index) {
throw new UnsupportedOperationException();
}
/**
* @throws UnsupportedOperationException always
*/
@Override
public Item addItemAt(int index, Object newItemId) {
throw new UnsupportedOperationException();
}
@Override
public Integer getIdByIndex(int index) {
if (index < 0)
throw new IndexOutOfBoundsException("index < " + index);
final long size = this.ensureList(index);
if (index >= size)
throw new IndexOutOfBoundsException("index=" + index + " but size=" + size);
return index;
}
@Override
public List getItemIds(int startIndex, int numberOfItems) {
if (numberOfItems < 0)
throw new IllegalArgumentException("numberOfItems < 0");
final long size = this.ensureList(startIndex);
if (startIndex < 0 || startIndex > size)
throw new IndexOutOfBoundsException("startIndex=" + startIndex + " but size=" + size);
if (startIndex + numberOfItems > size)
numberOfItems = (int)(size - startIndex);
return new IntList(startIndex, numberOfItems);
}
@Override
public int indexOfId(Object itemId) {
if (!(itemId instanceof Integer))
return -1;
final int index = ((Integer)itemId).intValue();
if (index < 0 || index >= this.ensureList(index))
return -1;
return index;
}
// Container.Ordered
@Override
public Integer nextItemId(Object itemId) {
if (!(itemId instanceof Integer))
return null;
int index = ((Integer)itemId).intValue();
if (index < 0 || index + 1 >= this.ensureList(index))
return null;
return index + 1;
}
@Override
public Integer prevItemId(Object itemId) {
if (!(itemId instanceof Integer))
return null;
int index = ((Integer)itemId).intValue();
if (index - 1 < 0 || index >= this.ensureList(index))
return null;
return index - 1;
}
@Override
public Integer firstItemId() {
return this.ensureList(0) == 0 ? null : 0;
}
@Override
public Integer lastItemId() {
long size = this.ensureList(0);
return size == 0 ? null : (int)size - 1;
}
@Override
public boolean isFirstId(Object itemId) {
if (!(itemId instanceof Integer))
return false;
int index = ((Integer)itemId).intValue();
long size = this.ensureList(index);
return size > 0 && index == 0;
}
@Override
public boolean isLastId(Object itemId) {
if (!(itemId instanceof Integer))
return false;
int index = ((Integer)itemId).intValue();
long size = this.ensureList(index);
return size > 0 && index == size - 1;
}
/**
* @throws UnsupportedOperationException always
*/
@Override
public Item addItemAfter(Object previousItemId) {
throw new UnsupportedOperationException();
}
/**
* @throws UnsupportedOperationException always
*/
@Override
public Item addItemAfter(Object previousItemId, Object newItemId) {
throw new UnsupportedOperationException();
}
// Container.PropertySetChangeNotifier
@Override
@SuppressWarnings("deprecation")
public void addListener(Container.PropertySetChangeListener listener) {
super.addListener(listener);
}
@Override
public void addPropertySetChangeListener(Container.PropertySetChangeListener listener) {
super.addPropertySetChangeListener(listener);
}
@Override
@SuppressWarnings("deprecation")
public void removeListener(Container.PropertySetChangeListener listener) {
super.removeListener(listener);
}
@Override
public void removePropertySetChangeListener(Container.PropertySetChangeListener listener) {
super.removePropertySetChangeListener(listener);
}
// Container.ItemSetChangeNotifier
@Override
@SuppressWarnings("deprecation")
public void addListener(Container.ItemSetChangeListener listener) {
super.addListener(listener);
}
@Override
public void addItemSetChangeListener(Container.ItemSetChangeListener listener) {
super.addItemSetChangeListener(listener);
}
@Override
@SuppressWarnings("deprecation")
public void removeListener(Container.ItemSetChangeListener listener) {
super.removeListener(listener);
}
@Override
public void removeItemSetChangeListener(Container.ItemSetChangeListener listener) {
super.removeItemSetChangeListener(listener);
}
// IntList
private static class IntList extends AbstractList {
private final int min;
private final int size;
IntList(int size) {
this(0, size);
}
IntList(int min, int size) {
if (size < 0)
throw new IllegalArgumentException("size < 0");
this.min = min;
this.size = size;
}
@Override
public int size() {
return this.size;
}
@Override
public Integer get(int index) {
if (index < 0 || index >= this.size)
throw new IndexOutOfBoundsException();
return this.min + index;
}
}
}