/* * GeoTools - The Open Source Java GIS Toolkit * http://geotools.org * * (C) 2005-2008, Open Source Geospatial Foundation (OSGeo) * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; * version 2.1 of the License. * * This library 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 * Lesser General Public License for more details. */ package org.geotools.feature.visitor; import java.util.ArrayList; import java.util.Collection; import java.util.Collections; import java.util.HashSet; import java.util.List; import java.util.ListIterator; import java.util.Set; import org.geotools.data.simple.SimpleFeatureCollection; import org.geotools.factory.CommonFactoryFinder; import org.geotools.filter.IllegalFilterException; import org.geotools.util.Converters; import org.opengis.feature.Feature; import org.opengis.feature.simple.SimpleFeature; import org.opengis.feature.simple.SimpleFeatureType; import org.opengis.filter.FilterFactory; import org.opengis.filter.expression.Expression; /** * Generates a list of unique values from a collection * * @author Cory Horner, Refractions * * @since 2.2.M2 * * @source $URL$ */ public class UniqueVisitor implements FeatureCalc { private Expression expr; Set set = new HashSet(); public UniqueVisitor(String attributeTypeName) { FilterFactory factory = CommonFactoryFinder.getFilterFactory(null); expr = factory.property(attributeTypeName); } public UniqueVisitor(int attributeTypeIndex, SimpleFeatureType type) throws IllegalFilterException { FilterFactory factory = CommonFactoryFinder.getFilterFactory(null); expr = factory.property(type.getDescriptor(attributeTypeIndex).getLocalName()); } public UniqueVisitor(String attrName, SimpleFeatureType type) throws IllegalFilterException { FilterFactory factory = CommonFactoryFinder.getFilterFactory(null); expr = factory.property(type.getDescriptor(attrName).getLocalName()); } public UniqueVisitor(Expression expr) { this.expr = expr; } public void init(SimpleFeatureCollection collection) { //do nothing } public void visit(SimpleFeature feature) { visit(feature); } public void visit(Feature feature) { //we ignore null attributes Object value = expr.evaluate(feature); if (value != null) { set.add(value); } } public Expression getExpression() { return expr; } public Set getUnique() { /** * Return a list of unique values from the collection */ return set; } public void setValue(Object newSet) { if (newSet instanceof Collection) { //convert to set this.set = new HashSet((Collection) newSet); } else { Collection collection = Converters.convert(newSet, List.class); if(collection != null) { this.set = new HashSet(collection); } else { this.set = new HashSet(Collections.singleton(newSet)); } } } public void reset() { /** * Reset the unique and current minimum for the features in the * collection */ this.set = new HashSet(); } public CalcResult getResult() { if (set.size() < 1) { return CalcResult.NULL_RESULT; } return new UniqueResult(set); } public static class UniqueResult extends AbstractCalcResult { private Set unique; public UniqueResult(Set newSet) { unique = newSet; } public Object getValue() { return new HashSet(unique); } public boolean isCompatible(CalcResult targetResults) { //list each calculation result which can merge with this type of result if (targetResults instanceof UniqueResult || targetResults == CalcResult.NULL_RESULT) return true; return false; } public CalcResult merge(CalcResult resultsToAdd) { if (!isCompatible(resultsToAdd)) { throw new IllegalArgumentException( "Parameter is not a compatible type"); } if(resultsToAdd == CalcResult.NULL_RESULT) { return this; } if (resultsToAdd instanceof UniqueResult) { //add one set to the other (to create one big unique list) Set newSet = new HashSet(unique); newSet.addAll((Set) resultsToAdd.getValue()); return new UniqueResult(newSet); } else { throw new IllegalArgumentException( "The CalcResults claim to be compatible, but the appropriate merge method has not been implemented."); } } } }