All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.mapfish.print.attribute.map.MapAttribute Maven / Gradle / Ivy

package org.mapfish.print.attribute.map;

import org.json.JSONArray;
import org.locationtech.jts.geom.Envelope;
import org.mapfish.print.attribute.map.OverviewMapAttribute.OverviewMapAttributeValues;
import org.mapfish.print.attribute.map.ZoomToFeatures.ZoomType;
import org.mapfish.print.config.Template;
import org.mapfish.print.parser.CanSatisfyOneOf;
import org.mapfish.print.parser.HasDefaultValue;
import org.mapfish.print.parser.OneOf;
import org.mapfish.print.parser.Requires;
import org.mapfish.print.processor.map.CreateMapProcessor;
import org.mapfish.print.wrapper.PArray;
import org.mapfish.print.wrapper.json.PJsonArray;
import org.opengis.referencing.FactoryException;
import org.opengis.referencing.crs.CoordinateReferenceSystem;

import java.util.ArrayList;
import java.util.List;
import java.util.function.Function;
import javax.annotation.Nonnull;

/**
 * 

The attributes for {@link CreateMapProcessor} (see * !createMap processor).

* [[examples=verboseExample]] */ public final class MapAttribute extends GenericMapAttribute { private static final double DEFAULT_SNAP_TOLERANCE = 0.05; private static final ZoomLevelSnapStrategy DEFAULT_SNAP_STRATEGY = ZoomLevelSnapStrategy.CLOSEST_LOWER_SCALE_ON_TIE; private static final boolean DEFAULT_SNAP_GEODETIC = false; @SuppressWarnings("unchecked") @Override public Class getValueType() { return MapAttributeValues.class; } @Override public MapAttributeValues createValue(final Template template) { return new MapAttributeValues(template, getWidth(), getHeight()); } /** * The value of {@link MapAttribute}. */ public class MapAttributeValues extends GenericMapAttributeValues { private static final boolean DEFAULT_ADJUST_BOUNDS = false; private static final double DEFAULT_ROTATION = 0.0; /** * An array of 4 doubles, minX, minY, maxX, maxY. The bounding box of the map. *

* Either the bbox or the center + scale must be defined *

*/ @OneOf("MapBounds") public double[] bbox; /** * A GeoJSON geometry that is essentially the area of the area to draw on the map. *

*/ @CanSatisfyOneOf("MapBounds") @HasDefaultValue public AreaOfInterest areaOfInterest; /** * An array of 2 doubles, (x, y). The center of the map. */ @Requires("scale") @OneOf("MapBounds") public double[] center; /** * If center is defined then this is the scale of the map centered at center. */ @HasDefaultValue public Double scale; /** * Zoom the map to the features of a specific layer or all features of the map. */ @OneOf("MapBounds") public ZoomToFeatures zoomToFeatures; /** * The json with all the layer information. This will be parsed in postConstruct into a list of * layers and therefore this field should not normally be accessed. *

* The first layer in the array will be the top layer in the map. The last layer in the array will be * the bottom layer in the map. There for the last layer will be hidden by the first layer (where not * transparent). */ @HasDefaultValue public PArray layers = new PJsonArray(null, new JSONArray(), null); /** * The output dpi of the printed map. */ public double dpi; private MapBounds mapBounds; /** * Constructor. * * @param template the template this map is part of. * @param width the width of the map. * @param height the height of the map. */ public MapAttributeValues(final Template template, final Integer width, final Integer height) { super(template, width, height); } /** * Constructor. * * @param template the template this map is part of. */ public MapAttributeValues(final Template template) { super(template); } @Override public Double getDpi() { return this.dpi; } @Override public final PArray getRawLayers() { return this.layers; } @Override public void setRawLayers(final PArray newLayers) { this.layers = newLayers; } @Override public final void postConstruct() throws FactoryException { super.postConstruct(); if (getDpi() > getMaxDpi()) { throw new IllegalArgumentException( "dpi parameter was " + getDpi() + " must be limited to " + getMaxDpi() + ". The path to the parameter is: " + getDpi()); } if (this.zoomToFeatures != null) { if (this.zoomToFeatures.zoomType == ZoomType.CENTER) { if (this.scale == null && this.zoomToFeatures.minScale == null) { throw new IllegalArgumentException( "When using 'zoomToFeatures.zoom.Type: center' either 'scale' " + "or 'zoomToFeatures.minScale' has to be given."); } } else if (this.zoomToFeatures.zoomType == ZoomType.EXTENT && this.zoomToFeatures.minScale == null) { throw new IllegalArgumentException( "When using 'zoomToFeatures.zoom.Type: extent' 'zoomToFeatures.minScale'" + "has to be given."); } } this.mapBounds = parseBounds(); } private MapBounds parseBounds() { final CoordinateReferenceSystem crs = parseProjection(); if (this.center != null && this.bbox != null) { throw new IllegalArgumentException("Cannot have both center and bbox defined"); } MapBounds bounds; if (this.center != null) { double centerX = this.center[0]; double centerY = this.center[1]; bounds = new CenterScaleMapBounds(crs, centerX, centerY, this.scale); } else if (this.bbox != null) { final int maxYIndex = 3; double minX = this.bbox[0]; double minY = this.bbox[1]; double maxX = this.bbox[2]; double maxY = this.bbox[maxYIndex]; bounds = new BBoxMapBounds(crs, minX, minY, maxX, maxY); } else if (this.areaOfInterest != null) { Envelope area = this.areaOfInterest.getArea().getEnvelopeInternal(); bounds = new BBoxMapBounds(crs, area); } else if (this.zoomToFeatures != null) { bounds = new BBoxMapBounds(crs, 0, 0, 10, 10); } else { throw new IllegalArgumentException("Expected either: center and scale, bbox, or an " + "areaOfInterest defined in order to calculate " + "the map bounds"); } return bounds; } public MapBounds getMapBounds() { return this.mapBounds; } public void setMapBounds(final MapBounds mapBounds) { this.mapBounds = mapBounds; } /** * Recalculate the bounds after center or bounds have changed. */ public void recalculateBounds() { this.mapBounds = parseBounds(); } @Override public String getProjection() { return getValueOr(super.getProjection(), DEFAULT_PROJECTION); } @Override public Double getZoomSnapTolerance() { return getValueOr(super.getZoomSnapTolerance(), DEFAULT_SNAP_TOLERANCE); } @Override public ZoomLevelSnapStrategy getZoomLevelSnapStrategy() { return getValueOr(super.getZoomLevelSnapStrategy(), DEFAULT_SNAP_STRATEGY); } @Override public Boolean getZoomSnapGeodetic() { return getValueOr(super.getZoomSnapGeodetic(), DEFAULT_SNAP_GEODETIC); } @Override public Double getRotation() { return getValueOr(super.getRotation(), DEFAULT_ROTATION); } @Override public Boolean isUseNearestScale() { return (this.useNearestScale == null || this.useNearestScale) && getZoomLevels() != null; } @Override public Boolean isUseAdjustBounds() { return getValueOr(super.isUseAdjustBounds(), DEFAULT_ADJUST_BOUNDS); } /** * Creates an {@link OverriddenMapAttributeValues} * instance with the current object and a given * {@link OverviewMapAttributeValues} * instance. * * @param paramOverrides Attributes set in this instance will override attributes in the * current instance. */ public final OverriddenMapAttributeValues getWithOverrides( final OverviewMapAttributeValues paramOverrides) { return new OverriddenMapAttributeValues(this, paramOverrides, getTemplate()); } /** * Create a copy of this instance. Should be overridden for each subclass. * * @param width the width of the new map attribute values to create * @param height the height of the new map attribute values to create * @param updater a function which will be called after copy is made but before postConstruct * is called in order to do other configuration changes. */ public MapAttributeValues copy( final int width, final int height, @Nonnull final Function updater) { MapAttributeValues copy = new MapAttributeValues(getTemplate(), width, height); copy.areaOfInterest = this.areaOfInterest.copy(); copy.bbox = this.bbox; copy.center = this.center; copy.scale = this.scale; copy.layers = this.layers; copy.dpi = this.getDpi(); copy.projection = this.getProjection(); copy.rotation = this.getRotation(); copy.useNearestScale = this.isUseNearestScale(); copy.useAdjustBounds = this.useAdjustBounds; copy.longitudeFirst = this.longitudeFirst; copy.zoomToFeatures = (this.zoomToFeatures == null) ? null : this.zoomToFeatures.copy(); updater.apply(copy); try { copy.postConstruct(); } catch (FactoryException e) { throw new RuntimeException(e); } return copy; } } /** * A wrapper around a {@link MapAttributeValues} instance and an * {@link OverviewMapAttributeValues} * instance, which is used to render the overview map. *

* If attributes on the * {@link OverviewMapAttributeValues} * instance are set, those attributes will be returned, otherwise the ones on {@link MapAttributeValues}. */ public class OverriddenMapAttributeValues extends MapAttributeValues { private final MapAttributeValues params; private final OverviewMapAttributeValues paramOverrides; private MapBounds zoomedOutBounds = null; private MapLayer mapExtentLayer = null; /** * Constructor. * * @param params The fallback parameters. * @param paramOverrides The parameters explicitly defined for the overview map. * @param template The template this map is part of. */ public OverriddenMapAttributeValues( final MapAttributeValues params, final OverviewMapAttributeValues paramOverrides, final Template template) { super(template, paramOverrides.getWidth(), paramOverrides.getHeight()); this.params = params; this.paramOverrides = paramOverrides; } /** * The bounds used to render the overview-map. */ @Override public final MapBounds getMapBounds() { return this.zoomedOutBounds; } /** * Custom bounds for the overview-map. Overwrites the bounds of the original map. */ public final MapBounds getCustomBounds() { return this.paramOverrides.getMapBounds(); } /** * The bounds of the original map that this overview-map is associated to. */ public final MapBounds getOriginalBounds() { return this.params.getMapBounds(); } public final void setZoomedOutBounds(final MapBounds zoomedOutBounds) { this.zoomedOutBounds = zoomedOutBounds; } @Override public final Double getDpi() { return getValueOr(this.paramOverrides.getDpi(), this.params.getDpi()); } @Override public final Double getZoomSnapTolerance() { return getValueOr(this.paramOverrides.getZoomSnapTolerance(), this.params.getZoomSnapTolerance()); } @Override public final ZoomLevelSnapStrategy getZoomLevelSnapStrategy() { return getValueOr(this.paramOverrides.getZoomLevelSnapStrategy(), this.params.getZoomLevelSnapStrategy()); } @Override public final ZoomLevels getZoomLevels() { return getValueOr(this.paramOverrides.getZoomLevels(), this.params.getZoomLevels()); } @Override public final Double getRotation() { return getValueOr(this.paramOverrides.getRotation(), this.params.getRotation()); } @Override public final Boolean isUseAdjustBounds() { return getValueOr(this.paramOverrides.isUseAdjustBounds(), this.params.isUseAdjustBounds()); } @Override public final Boolean isUseNearestScale() { final Boolean useNearestScale = getValueOr(this.paramOverrides.useNearestScale, this.params.useNearestScale); return (useNearestScale == null || useNearestScale) && getZoomLevels() != null; } public final void setMapExtentLayer(final MapLayer mapExtentLayer) { this.mapExtentLayer = mapExtentLayer; } @Override public final List getLayers() { // return the layers together with a layer for the bbox rectangle of the map List layers = new ArrayList<>(); if (this.mapExtentLayer != null) { layers.add(this.mapExtentLayer); } if (!this.paramOverrides.getLayers().isEmpty()) { layers.addAll(this.paramOverrides.getLayers()); } else { layers.addAll(this.params.getLayers()); } return layers; } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy