/* * 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. * * Created on May 11, 2005, 6:21 PM */ package org.geotools.filter.function; import java.io.IOException; import java.util.Collections; import java.util.List; import java.util.logging.Level; import java.util.logging.Logger; import org.geotools.feature.FeatureCollection; import org.geotools.feature.visitor.CalcResult; import org.geotools.feature.visitor.UniqueVisitor; import org.geotools.filter.AttributeExpression; import org.geotools.filter.Expression; import org.geotools.filter.FunctionExpression; import org.geotools.filter.FunctionExpressionImpl; import org.geotools.filter.IllegalFilterException; import org.geotools.filter.visitor.AbstractFilterVisitor; import org.opengis.feature.simple.SimpleFeature; import org.opengis.feature.simple.SimpleFeatureType; /** * Calculates the unique value of an attribute for a given FeatureCollection * and Expression. * * @author Cory Horner * @since 2.2M2 * @source $URL$ */ public class Collection_UniqueFunction extends FunctionExpressionImpl implements FunctionExpression { /** The logger for the filter module. */ private static final Logger LOGGER = org.geotools.util.logging.Logging.getLogger( "org.geotools.filter.function"); FeatureCollection previousFeatureCollection = null; Object unique = null; /** * Creates a new instance of Collection_UniqueFunction */ public Collection_UniqueFunction() { super("Collection_Unique"); } public int getArgCount() { return 1; } /** * Calculate unique (using FeatureCalc) - only one parameter is used. * * @param collection collection to calculate the unique * @param expression Single Expression argument * * @return An object containing the unique value of the attributes * * @throws IllegalFilterException * @throws IOException */ static CalcResult calculateUnique(FeatureCollection collection, Expression expression) throws IllegalFilterException, IOException { UniqueVisitor uniqueVisitor = new UniqueVisitor(expression); collection.accepts(uniqueVisitor, null); return uniqueVisitor.getResult(); } /** * The provided arguments are evaulated with respect to the * FeatureCollection. * *

* For an aggregate function (like unique) please use the WFS mandated XPath * syntax to refer to featureMember content. *

* *

* To refer to all 'X': featureMember/asterisk/X *

* * @param args DOCUMENT ME! * * @throws IllegalArgumentException DOCUMENT ME! */ public void setParameters(List args) { super.setParameters(args); Expression expr = (Expression) getExpression(0); // if we see "featureMembers/*/ATTRIBUTE" change to "ATTRIBUTE" expr.accept(new AbstractFilterVisitor() { public void visit(AttributeExpression expression) { String xpath = expression.getAttributePath(); if (xpath.startsWith("featureMembers/*/")) { xpath = xpath.substring(17); } else if (xpath.startsWith("featureMember/*/")) { xpath = xpath.substring(16); } try { expression.setAttributePath(xpath); } catch (IllegalFilterException e) { // ignore } } }); } public Object evaluate(Object feature) { if (feature == null) { return new Integer(0); // no features were visited in the making of this answer } FeatureCollection featureCollection = (FeatureCollection) feature; Expression expr = (Expression) getExpression(0); synchronized (featureCollection) { if (featureCollection != previousFeatureCollection) { previousFeatureCollection = featureCollection; unique = null; try { CalcResult result = calculateUnique(featureCollection, expr); if (result != null) { unique = result.getValue(); } } catch (IllegalFilterException e) { LOGGER.log(Level.FINER, e.getLocalizedMessage(), e); } catch (IOException e) { LOGGER.log(Level.FINER, e.getLocalizedMessage(), e); } } } return unique; } public void setExpression(Expression e) { setParameters(Collections.singletonList(e)); } }