/*
* GeoTools - The Open Source Java GIS Toolkit
* http://geotools.org
*
* (C) 2012, 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.data.transform;
import org.geotools.referencing.CRS;
import org.opengis.feature.simple.SimpleFeatureType;
import org.opengis.feature.type.AttributeDescriptor;
import org.opengis.feature.type.GeometryDescriptor;
import org.opengis.filter.expression.Add;
import org.opengis.filter.expression.Divide;
import org.opengis.filter.expression.Expression;
import org.opengis.filter.expression.ExpressionVisitor;
import org.opengis.filter.expression.Function;
import org.opengis.filter.expression.Literal;
import org.opengis.filter.expression.Multiply;
import org.opengis.filter.expression.NilExpression;
import org.opengis.filter.expression.PropertyName;
import org.opengis.filter.expression.Subtract;
import org.opengis.referencing.crs.CoordinateReferenceSystem;
import com.vividsolutions.jts.geom.Geometry;
/**
* Evaluates the CRS of an expression returning geometries by static analysis if possible
*
* @author Andrea Aime - GeoSolutions
*/
class CRSEvaluator implements ExpressionVisitor {
private SimpleFeatureType schema;
public CRSEvaluator(SimpleFeatureType schema) {
this.schema = schema;
}
@Override
public Object visit(Literal expression, Object extraData) {
Object value = expression.getValue();
if (value instanceof Geometry) {
Geometry g = (Geometry) value;
if (g.getUserData() instanceof CoordinateReferenceSystem) {
return g.getUserData();
} else if (g.getSRID() > 0) {
try {
return CRS.decode("EPSG:" + g.getSRID());
} catch (Exception e) {
return null;
}
}
}
return null;
}
@Override
public Object visit(Function expression, Object extraData) {
// geometry manipulation functions that we have today don't mess with the
// crs, but we might have to modify this if we get a function that
// assigns the CRS
for (Expression param : expression.getParameters()) {
Object result = param.accept(this, extraData);
if (result instanceof CoordinateReferenceSystem) {
return result;
}
}
return null;
}
@Override
public Object visit(PropertyName expression, Object extraData) {
AttributeDescriptor ad = expression.evaluate(schema, AttributeDescriptor.class);
if (ad == null) {
throw new IllegalArgumentException(
"Original feature type does not have a property named "
+ expression.getPropertyName());
} else if (ad instanceof GeometryDescriptor) {
return ((GeometryDescriptor) ad).getCoordinateReferenceSystem();
}
return null;
}
@Override
public Object visit(NilExpression expression, Object extraData) {
return null;
}
@Override
public Object visit(Add expression, Object extraData) {
return null;
}
@Override
public Object visit(Divide expression, Object extraData) {
return null;
}
@Override
public Object visit(Multiply expression, Object extraData) {
return null;
}
@Override
public Object visit(Subtract expression, Object extraData) {
return null;
}
}