Package org.vfny.geoserver.wms.responses.map.kml

Source Code of org.vfny.geoserver.wms.responses.map.kml.KMLRasterTransformer$KMLRasterTranslator

/* Copyright (c) 2001 - 2007 TOPP - www.openplans.org. All rights reserved.
* This code is licensed under the GPL 2.0 license, availible at the root
* application directory.
*/
package org.vfny.geoserver.wms.responses.map.kml;

import java.util.Iterator;
import java.util.List;
import java.util.logging.Level;

import org.geoserver.wms.util.WMSRequests;
import org.geotools.data.FeatureSource;
import org.geotools.feature.FeatureCollection;
import org.geotools.geometry.jts.ReferencedEnvelope;
import org.geotools.map.MapLayer;
import org.geotools.referencing.CRS;
import org.geotools.referencing.crs.DefaultGeographicCRS;
import org.geotools.styling.FeatureTypeStyle;
import org.geotools.styling.Symbolizer;
import org.geotools.xml.transform.Translator;
import org.opengis.feature.simple.SimpleFeature;
import org.opengis.feature.simple.SimpleFeatureType;
import org.vfny.geoserver.wms.WMSMapContext;
import org.vfny.geoserver.wms.WmsException;
import org.vfny.geoserver.wms.requests.GetMapRequest;
import org.xml.sax.ContentHandler;

import com.vividsolutions.jts.geom.Envelope;
import com.vividsolutions.jts.geom.Geometry;
import com.vividsolutions.jts.geom.GeometryFactory;
import com.vividsolutions.jts.geom.MultiPolygon;
import com.vividsolutions.jts.geom.Polygon;


/**
* Tranforms a feature colleciton to a kml "Document" element which contains a
* "Folder" element consisting of "GroundOverlay" elements.
* <p>
* Usage:
* <pre>
<code>
*  //have a reference to a map context and output stream
*  WMSMapContext context = ...
*  OutputStream output = ...;
*
*  KMLRasterTransformer tx = new KMLRasterTransformer( context );
*  for ( int i = 0; i < context.getLayerCount(); i++ ) {
*    MapLayer mapLayer = mapConext.getMapLayer( i );
*
*    //transform
*    tx.transform( mapLayer, output );
*  }
</code>
* </pre>
* </p>
* <p>
* The inline parameter {@link #setInline(boolean)} controls wether the images
* for the request are refernces "inline" as local images, or remoteley as wms
* requests.
* </p>
*
* @author Justin Deoliveira, The Open Planning Project, jdeolive@openplans.org
*
*/
public class KMLRasterTransformer extends KMLMapTransformer {
    /**
     * Flag controlling wether images are refernces inline or as remote wms calls.
     */
    boolean inline = false;

    public KMLRasterTransformer(WMSMapContext mapContext) {
        super(mapContext, null);

        setNamespaceDeclarationEnabled(false);
    }

    public void setInline(boolean inline) {
        this.inline = inline;
    }

    public Translator createTranslator(ContentHandler handler) {
        return new KMLRasterTranslator(handler);
    }

    class KMLRasterTranslator extends KMLMapTranslatorSupport {
        public KMLRasterTranslator(ContentHandler handler) {
            super(handler);
        }

        public void encode(Object o) throws IllegalArgumentException {
            MapLayer mapLayer = (MapLayer) o;
            int mapLayerOrder = mapContext.indexOf(mapLayer);

            if ( isStandAlone() ) {
                start( "kml" );
            }
           
            //start("Document");
            //element("name", mapLayer.getTitle());

            //start the folder naming it 'layer_<mapLayerOrder>', this is
            // necessary for a GroundOverlay
            start("Folder");
            element("name", "layer_" + mapLayerOrder);
            element("description", mapLayer.getTitle());

            start("GroundOverlay");
            //element( "name", feature.getID() );
            element("name", mapLayer.getTitle());
            element("drawOrder", Integer.toString(mapLayerOrder));

            //encode the icon
            start("Icon");

            encodeHref(mapLayer);

            element("viewRefreshMode", "never");
            element("viewBoundScale", "0.75");
            end("Icon");

            //encde the bounding box
            ReferencedEnvelope box = new ReferencedEnvelope(mapContext.getAreaOfInterest());
            boolean reprojectBBox = (box.getCoordinateReferenceSystem() != null)
            && !CRS.equalsIgnoreMetadata(box.getCoordinateReferenceSystem(), DefaultGeographicCRS.WGS84);
            if (reprojectBBox) {
                try {
                    box = box.transform(DefaultGeographicCRS.WGS84, true);
                } catch(Exception e) {
                    throw new WmsException("Could not transform bbox to WGS84", "ReprojectionError", e);
                }
            }
            start("LatLonBox");
            element("north", Double.toString(box.getMaxY()));
            element("south", Double.toString(box.getMinY()));
            element("east", Double.toString(box.getMaxX()));
            element("west", Double.toString(box.getMinX()));
            end("LatLonBox");

            end("GroundOverlay");

            // if the kmplacemark format option is true, add placemarks to the output
            GetMapRequest request = mapContext.getRequest();
            boolean kmplacemark = request.getWMS().getKmlPlacemark();
            if (request.getFormatOptions().get("kmplacemark") != null)
                kmplacemark = (Boolean) request.getFormatOptions().get("kmplacemark");
            if (kmplacemark) {
                FeatureCollection<SimpleFeatureType, SimpleFeature> features = null;
                try {
                    features = KMLUtils
                            .loadFeatureCollection(
                                    (FeatureSource<SimpleFeatureType, SimpleFeature>) mapLayer
                                            .getFeatureSource(), mapLayer,
                                    mapContext);
                } catch (Exception ex) {
                    String msg = "Error getting features.";
                    LOGGER.log(Level.WARNING, msg, ex);
                }

                if (features != null && features.size() > 0) {
                    Geometry geom = null;
                    Geometry centroidGeom = null;

                    // get geometry of the area of interest
                    Envelope aoi = mapContext.getAreaOfInterest();
                    GeometryFactory factory = new GeometryFactory();
                    Geometry displayGeom = factory.toGeometry(new Envelope(aoi
                            .getMinX(), aoi.getMaxX(), aoi.getMinY(), aoi
                            .getMaxY()));

                    // get the styles for this feature
                    SimpleFeatureType featureType = features.getSchema();
                    FeatureTypeStyle[] fts = KMLUtils.filterFeatureTypeStyles(
                            mapLayer.getStyle(), featureType);

                    Iterator<SimpleFeature> iter = features.iterator();
                    while (iter.hasNext()) {
                        SimpleFeature ftr = iter.next();
                        geom = (Geometry) ftr.getDefaultGeometry();

                        List<Symbolizer> symbolizers = filterSymbolizers(ftr, fts);
                        if (symbolizers.size() != 0) encodeStyle(ftr, symbolizers);

                        // if this is a multipolygon, get the largest polygon
                        // that intersects the AOI
                        if (geom instanceof MultiPolygon) {
                            double maxSize = -1;
                            int numGeoms = geom.getNumGeometries();
                            for (int i = 0; i < numGeoms; i++) {
                                Polygon poly = (Polygon) geom.getGeometryN(i);
                                if (poly.getArea() > maxSize) {
                                    if (displayGeom.intersects(poly)) {
                                        geom = poly;
                                        maxSize = poly.getArea();
                                    }
                                }
                            }
                        }
                        Geometry g1 = displayGeom.intersection(geom);
                        // skip if the geometry is not in the AOI
                        if (g1.isEmpty())
                            continue;
                        centroidGeom = g1.getCentroid();
                        encodePlacemark(ftr, symbolizers, centroidGeom);
                    }
                }
            }

            end("Folder");

            //end("Document");
            if (isStandAlone()) {
                end( "kml" );
            }
        }

        protected void encodeHref(MapLayer mapLayer) {
            if (inline) {
                //inline means reference the image "inline" as in kmz
                // use the mapLayerOrder
                int mapLayerOrder = mapContext.indexOf(mapLayer);
                element("href", "layer_" + mapLayerOrder + ".png");
            } else {
                //reference the image as a remote wms call
                element("href"
                        WMSRequests.getGetMapUrl(
                            mapContext.getRequest(),
                            mapLayer,
                            0,
                            mapContext.getAreaOfInterest(),
                            new String[]{
                                "format", "image/png",
                                "transparent", "true"
                            }
                            )
                       );
            }
        }
    }
}
TOP

Related Classes of org.vfny.geoserver.wms.responses.map.kml.KMLRasterTransformer$KMLRasterTranslator

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.