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

ucar.nc2.ft2.coverage.HorizCoordSys Maven / Gradle / Ivy

The newest version!
/*
 * Copyright (c) 1998-2018 John Caron and University Corporation for Atmospheric Research/Unidata
 * See LICENSE for license information.
 */

package ucar.nc2.ft2.coverage;

import com.google.common.collect.ImmutableList;
import com.google.common.collect.Lists;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import ucar.ma2.*;
import ucar.nc2.units.SimpleUnit;
import ucar.nc2.util.Optional;
import ucar.unidata.geoloc.*;
import javax.annotation.concurrent.Immutable;
import java.util.*;

/**
 * Horizontal CoordSys
 * 

* 1) has x,y,proj (1D) isProjection * 2) lat,lon (1D) isLatLon1D * 3) lat,lon (2D) class HorizCoordSys2D * 4) has x,y,proj and lat,lon (2D) LOOK 2D not used ? *

* Must be exactly one in a CoverageDataset. * * @author caron * @since 7/11/2015 */ // TODO: This class is trying to do too much: it's handling projection, latLon1D AND latLon2D CRSs. We should break // some of the code into HorizCoordSysProj, HorizCoordSysLatlon1D, and HorizCoordSysLatLon2D subclasses. HorizCoordSys // might then become an interface or abstract class with common method declarations/definitions. @Immutable public class HorizCoordSys { private static final Logger logger = LoggerFactory.getLogger(HorizCoordSys.class); public static HorizCoordSys factory(CoverageCoordAxis1D xAxis, CoverageCoordAxis1D yAxis, CoverageCoordAxis latAxis, CoverageCoordAxis lonAxis, CoverageTransform transform) { boolean isProjection = (xAxis != null) && (yAxis != null) && (transform != null); boolean hasLatLon = (latAxis != null) && (lonAxis != null); boolean has2DlatLon = latAxis instanceof LatLonAxis2D && lonAxis instanceof LatLonAxis2D; if (!isProjection && !hasLatLon) { throw new IllegalArgumentException("must have horiz coordinates (x,y,projection or lat,lon)"); } if (!isProjection && has2DlatLon) return new HorizCoordSys2D((LatLonAxis2D) latAxis, (LatLonAxis2D) lonAxis); else return new HorizCoordSys(xAxis, yAxis, latAxis, lonAxis, transform); } private final CoverageCoordAxis1D xAxis, yAxis; private final CoverageCoordAxis1D latAxis, lonAxis; // used in HorizCoordSys2D, do we really need to keep these in HorizCoordSys? protected final LatLonAxis2D latAxis2D, lonAxis2D; private final CoverageTransform transform; private final boolean isProjection; private final boolean isLatLon1D; private boolean isLatLon2D; // isProjection and isLatLon2D may both be "true". // scale factor for x, y axis if they have different units than the projection's default units private final double coordinateConversionFactor; protected HorizCoordSys(CoverageCoordAxis1D xAxis, CoverageCoordAxis1D yAxis, CoverageCoordAxis latAxis, CoverageCoordAxis lonAxis, CoverageTransform transform) { this.xAxis = xAxis; this.yAxis = yAxis; this.transform = transform; this.isProjection = (xAxis != null) && (yAxis != null) && (transform != null); this.isLatLon1D = latAxis instanceof CoverageCoordAxis1D && lonAxis instanceof CoverageCoordAxis1D; this.isLatLon2D = latAxis instanceof LatLonAxis2D && lonAxis instanceof LatLonAxis2D; assert isProjection || isLatLon1D || isLatLon2D : "missing horiz coordinates (x,y,projection or lat,lon)"; coordinateConversionFactor = getCoordinateConversionFactor(); if (isProjection && isLatLon2D) { boolean ok = true; if (!latAxis.getDependsOn().equalsIgnoreCase(lonAxis.getDependsOn())) ok = false; if (latAxis.getDependenceType() != CoverageCoordAxis.DependenceType.twoD) ok = false; if (lonAxis.getDependenceType() != CoverageCoordAxis.DependenceType.twoD) ok = false; String dependsOn = latAxis.getDependsOn(); if (!dependsOn.contains(xAxis.getName())) ok = false; if (!dependsOn.contains(yAxis.getName())) ok = false; if (!ok) { isLatLon2D = false; } } if (!isProjection && isLatLon2D && !(this instanceof HorizCoordSys2D)) logger.warn("Should be HorizCoordSys2D"); if (isLatLon1D) { this.latAxis = (CoverageCoordAxis1D) latAxis; this.lonAxis = (CoverageCoordAxis1D) lonAxis; } else { this.latAxis = null; this.lonAxis = null; } if (isLatLon2D) { this.latAxis2D = (LatLonAxis2D) latAxis; this.lonAxis2D = (LatLonAxis2D) lonAxis; } else { this.latAxis2D = null; this.lonAxis2D = null; } } public String getName() { if (isProjection) return xAxis.getName() + " " + yAxis.getName() + " " + transform.getName(); else return latAxis.getName() + " " + lonAxis.getName(); } public boolean isProjection() { return isProjection; } public boolean isLatLon2D() { return false; } public List getCoordAxes() { List result = new ArrayList<>(); if (xAxis != null) result.add(xAxis); if (yAxis != null) result.add(yAxis); if (latAxis != null) result.add(latAxis); if (lonAxis != null) result.add(lonAxis); return result; } public CoverageTransform getTransform() { return transform; } ///////////////////////////////////////////////////////////////////////////////////// public Optional subset(SubsetParams params) { LatLonRect llbb = (LatLonRect) params.get(SubsetParams.latlonBB); ProjectionRect projbb = (ProjectionRect) params.get(SubsetParams.projBB); LatLonPoint latlon = (LatLonPoint) params.get(SubsetParams.latlonPoint); Integer horizStride = (Integer) params.get(SubsetParams.horizStride); if (horizStride == null || horizStride < 1) horizStride = 1; CoverageCoordAxis1D xaxisSubset = null, yaxisSubset = null; CoverageCoordAxis lataxisSubset = null, lonaxisSubset = null; Optional opt; Optional optb; Formatter errMessages = new Formatter(); try { if (latlon != null) { // overrides other horiz subset params if (isProjection) { CoordAxisHelper xhelper = new CoordAxisHelper(xAxis); CoordAxisHelper yhelper = new CoordAxisHelper(yAxis); // we have to transform latlon to projection coordinates ProjectionImpl proj = transform.getProjection(); final ProjectionPoint projectionRectInDefaultUnits = proj.latLonToProj(latlon); optb = xhelper.subsetContaining(convertFromDefaultUnits(projectionRectInDefaultUnits.getX())); if (optb.isPresent()) xaxisSubset = new CoverageCoordAxis1D(optb.get()); else errMessages.format("xaxis: %s;%n", optb.getErrorMessage()); optb = yhelper.subsetContaining(convertFromDefaultUnits(projectionRectInDefaultUnits.getY())); if (optb.isPresent()) yaxisSubset = new CoverageCoordAxis1D(optb.get()); else errMessages.format("yaxis: %s;%n", optb.getErrorMessage()); } else { CoordAxisHelper xhelper = new CoordAxisHelper(lonAxis); CoordAxisHelper yhelper = new CoordAxisHelper(latAxis); double lonNormal = LatLonPoints.lonNormalFrom(latlon.getLongitude(), lonAxis.getStartValue()); optb = xhelper.subsetContaining(lonNormal); if (optb.isPresent()) lonaxisSubset = new CoverageCoordAxis1D(optb.get()); else errMessages.format("lonaxis: %s;%n", optb.getErrorMessage()); optb = yhelper.subsetContaining(latlon.getLatitude()); if (optb.isPresent()) lataxisSubset = new CoverageCoordAxis1D(optb.get()); else errMessages.format("lataxis: %s;%n", optb.getErrorMessage()); } } else if (projbb != null) { if (isProjection) { opt = xAxis.subset(projbb.getMinX(), projbb.getMaxX(), horizStride); if (opt.isPresent()) xaxisSubset = (CoverageCoordAxis1D) opt.get(); else errMessages.format("xaxis: %s;%n", opt.getErrorMessage()); opt = yAxis.subset(projbb.getMinY(), projbb.getMaxY(), horizStride); if (opt.isPresent()) yaxisSubset = (CoverageCoordAxis1D) opt.get(); else errMessages.format("yaxis: %s;%n", opt.getErrorMessage()); } /* * else { // WTF projbb on non Projection ? * ProjectionImpl proj = transform.getProjection(); * LatLonRect llrect = proj.projToLatLonBB(projbb); * opt = lonaxis.subset(llrect.getLonMin(), llrect.getLonMax(), horizStride); * if (opt.isPresent()) lonaxisSubset = opt.get(); * else errMessages.format("lonaxis: %s;%n", opt.getErrorMessage()); * * opt = lataxis.subset(llrect.getLatMin(), llrect.getLatMax(), horizStride); * if (opt.isPresent()) lataxisSubset = opt.get(); * else errMessages.format("lataxis: %s;%n", opt.getErrorMessage()); * } */ } else if (llbb != null) { LatLonRect full = calcLatLonBoundingBox(); assert full != null; if (!full.containedIn(llbb)) { // if request contains entire bb, then no subsetting needed if (isProjection) { // we have to transform latlon to projection coordinates ProjectionImpl proj = transform.getProjection(); final ProjectionRect projectionRectInDefaultUnits = proj.latLonToProjBB(llbb); // allow projection to // override final double xMinInCorrectUnits = convertFromDefaultUnits(projectionRectInDefaultUnits.getMinX()); final double xMaxInCorrectUnits = convertFromDefaultUnits(projectionRectInDefaultUnits.getMaxX()); opt = xAxis.subset(xMinInCorrectUnits, xMaxInCorrectUnits, horizStride); if (opt.isPresent()) xaxisSubset = (CoverageCoordAxis1D) opt.get(); else errMessages.format("xaxis: %s;%n", opt.getErrorMessage()); final double yMinInCorrectUnits = convertFromDefaultUnits(projectionRectInDefaultUnits.getMinY()); final double yMaxInCorrectUnits = convertFromDefaultUnits(projectionRectInDefaultUnits.getMaxY()); opt = yAxis.subset(yMinInCorrectUnits, yMaxInCorrectUnits, horizStride); if (opt.isPresent()) yaxisSubset = (CoverageCoordAxis1D) opt.get(); else errMessages.format("yaxis: %s;%n", opt.getErrorMessage()); } else { opt = subsetLon(llbb, horizStride); if (opt.isPresent()) lonaxisSubset = opt.get(); else errMessages.format("lonaxis: %s;%n", opt.getErrorMessage()); opt = latAxis.subset(llbb.getLatMin(), llbb.getLatMax(), horizStride); if (opt.isPresent()) lataxisSubset = opt.get(); else errMessages.format("lataxis: %s;%n", opt.getErrorMessage()); } } } else if (horizStride > 1) { // no bounding box, just horiz stride if (isProjection) { opt = xAxis.subsetByIndex(xAxis.getRange().copyWithStride(horizStride)); if (opt.isPresent()) xaxisSubset = (CoverageCoordAxis1D) opt.get(); else errMessages.format("xaxis: %s;%n", opt.getErrorMessage()); opt = yAxis.subsetByIndex(yAxis.getRange().copyWithStride(horizStride)); if (opt.isPresent()) yaxisSubset = (CoverageCoordAxis1D) opt.get(); else errMessages.format("yaxis: %s;%n", opt.getErrorMessage()); } else { opt = lonAxis.subsetByIndex(lonAxis.getRange().copyWithStride(horizStride)); if (opt.isPresent()) lonaxisSubset = opt.get(); else errMessages.format("lonaxis: %s;%n", opt.getErrorMessage()); opt = latAxis.subsetByIndex(latAxis.getRange().copyWithStride(horizStride)); if (opt.isPresent()) lataxisSubset = opt.get(); else errMessages.format("lataxis: %s;%n", opt.getErrorMessage()); } } } catch (InvalidRangeException e) { errMessages.format("%s;%n", e.getMessage()); } String errs = errMessages.toString(); if (!errs.isEmpty()) return Optional.empty(errs); // makes a copy of the axis if (xaxisSubset == null && xAxis != null) xaxisSubset = (CoverageCoordAxis1D) xAxis.copy(); if (yaxisSubset == null && yAxis != null) yaxisSubset = (CoverageCoordAxis1D) yAxis.copy(); if (lataxisSubset == null && latAxis != null) lataxisSubset = latAxis.copy(); if (lonaxisSubset == null && lonAxis != null) lonaxisSubset = lonAxis.copy(); return Optional.of(new HorizCoordSys(xaxisSubset, yaxisSubset, lataxisSubset, lonaxisSubset, transform)); } public LatLonPoint getLatLon(int yindex, int xindex) { if (isProjection) { double x = xAxis.getCoordMidpoint(xindex); double y = yAxis.getCoordMidpoint(yindex); ProjectionImpl proj = transform.getProjection(); return proj.projToLatLon(convertToDefaultUnits(x), convertToDefaultUnits(y)); } else { double lat = latAxis.getCoordMidpoint(yindex); double lon = lonAxis.getCoordMidpoint(xindex); return LatLonPoint.create(lat, lon); } } // here's where to deal with crossing seam private Optional subsetLon(LatLonRect latLonBoundingBox, int stride) { double wantMin = LatLonPoints.lonNormalFrom(latLonBoundingBox.getLonMin(), lonAxis.getStartValue()); double wantMax = LatLonPoints.lonNormalFrom(latLonBoundingBox.getLonMax(), lonAxis.getStartValue()); double start = lonAxis.getStartValue(); double end = lonAxis.getEndValue(); // use MAMath.MinMax as a container for two values, min and max List lonIntvs = subsetLonIntervals(wantMin, wantMax, start, end, latLonBoundingBox.containsAllLongitude()); if (lonIntvs.isEmpty()) return Optional.empty( String.format("longitude want [%f,%f] does not intersect lon axis [%f,%f]", wantMin, wantMax, start, end)); if (lonIntvs.size() == 1) { MAMath.MinMax lonIntv = lonIntvs.get(0); return lonAxis.subset(lonIntv.min, lonIntv.max, stride); } // this is the seam crossing case return lonAxis.subsetByIntervals(lonIntvs, stride); } /* * longitude subset, after normalizing to start * draw a circle, representing longitude values from start to start + 360. * all values are on this circle and are > start. * put start at bottom of circle, end > start, data has values from start, counterclockwise to end. * wantMin, wantMax can be anywhere, want goes from wantMin counterclockwise to wantMax. * wantMin may be less than or greater than wantMax. * * cases: * 0. contains all longitude, return all longitude * (could be normalized such that wantMin == wantMax so handle separately) * A. wantMin < wantMax * 1 wantMin, wantMax > end : empty * 2 wantMin < end : [wantMin, min(wantMax,end)] * * B. wantMin > wantMax * 1 wantMin, wantMax > end : all [start, end] * 2 wantMin, wantMax < end : 2 pieces: [wantMin, end] + [start, max] */ private List subsetLonIntervals(double wantMin, double wantMax, double start, double end, boolean containsAllLongitude) { if (containsAllLongitude) { return Lists.newArrayList(new MAMath.MinMax(start, end)); } if (wantMin <= wantMax) { if (wantMin > end) { // none A.1 return ImmutableList.of(); } else { // A.2 return Lists.newArrayList(new MAMath.MinMax(wantMin, Math.min(wantMax, end))); } } else { // wantMin > wantMax if (wantMax > end) { // all B.1 return Lists.newArrayList(new MAMath.MinMax(start, end)); // LOOK is this correct?! } else { return Lists.newArrayList(new MAMath.MinMax(wantMin, end), new MAMath.MinMax(start, wantMax)); } } } // return y, x range public List getRanges() { List result = new ArrayList<>(); result.add(getYAxis().getRange()); RangeIterator lonRange = getXAxis().getRangeIterator(); if (lonRange == null) lonRange = getXAxis().getRange(); // clumsy result.add(lonRange); return result; } public CoverageCoordAxis1D getXAxis() { return (xAxis != null) ? xAxis : lonAxis; } public CoverageCoordAxis1D getYAxis() { return (yAxis != null) ? yAxis : latAxis; } public LatLonAxis2D getLonAxis2D() { return lonAxis2D; } public LatLonAxis2D getLatAxis2D() { return latAxis2D; } public static class CoordReturn { public int x, y; public double xcoord, ycoord; } public Optional findXYindexFromCoord(double x, double y) { CoordReturn result = new CoordReturn(); if (isProjection) { CoordAxisHelper xhelper = new CoordAxisHelper(xAxis); CoordAxisHelper yhelper = new CoordAxisHelper(yAxis); result.x = xhelper.findCoordElement(x, false); result.y = yhelper.findCoordElement(y, false); if (result.x >= 0 && result.x < xAxis.getNcoords() && result.y >= 0 && result.y < yAxis.getNcoords()) { result.xcoord = xAxis.getCoordMidpoint(result.x); result.ycoord = yAxis.getCoordMidpoint(result.y); return Optional.of(result); } else { return Optional.empty("not in grid"); } } else { // 1D lat lon case CoordAxisHelper xhelper = new CoordAxisHelper(lonAxis); CoordAxisHelper yhelper = new CoordAxisHelper(latAxis); double lon = LatLonPoints.lonNormalFrom(x, lonAxis.getStartValue()); result.x = xhelper.findCoordElement(lon, false); result.y = yhelper.findCoordElement(y, false); if (result.x >= 0 && result.x < lonAxis.getNcoords() && result.y >= 0 && result.y < latAxis.getNcoords()) { result.xcoord = lonAxis.getCoordMidpoint(result.x); result.ycoord = latAxis.getCoordMidpoint(result.y); return Optional.of(result); } else { return Optional.empty("not in grid"); } } } ///////////////////////////////////// Boundary calculations ///////////////////////////////////// /** * Calculates the bounding box of this coordinate reference system, in projection coordinates. If this CRS * {@link #isProjection isn't a projection}, than {@code null} is returned. * * @return the bounding box of this CRS, in projection coordinates. */ public ProjectionRect calcProjectionBoundingBox() { if (!isProjection) return null; double minX = Math.min(xAxis.getCoordEdgeFirst(), xAxis.getCoordEdgeLast()); double minY = Math.min(yAxis.getCoordEdgeFirst(), yAxis.getCoordEdgeLast()); double width = Math.abs(xAxis.getCoordEdgeLast() - xAxis.getCoordEdgeFirst()); double height = Math.abs(yAxis.getCoordEdgeLast() - yAxis.getCoordEdgeFirst()); return new ProjectionRect(ProjectionPoint.create(minX, minY), width, height); } /** * Calculates the bounding box of this coordinate reference system, in latitude/longitude. This method properly * handles coverages that straddle the international date line by deriving its bounding box from the * {@link #calcConnectedLatLonBoundaryPoints(int, int) connected latitude/longitude boundary}. *

* If this CRS {@link #isProjection is a projection}, its lat/lon boundary is computed by converting each point * in its {@link #calcProjectionBoundaryPoints() projection boundary} to latitude/longitude using the * {@link Projection projection}. * * @return the bounding box of this CRS, in latitude/longitude. */ public LatLonRect calcLatLonBoundingBox() { double minLat = Double.MAX_VALUE; double minLon = Double.MAX_VALUE; double maxLat = -Double.MAX_VALUE; double maxLon = -Double.MAX_VALUE; for (LatLonPointNoNormalize boundaryPoint : calcConnectedLatLonBoundaryPoints()) { minLat = Math.min(minLat, boundaryPoint.getLatitude()); minLon = Math.min(minLon, boundaryPoint.getLongitude()); maxLat = Math.max(maxLat, boundaryPoint.getLatitude()); maxLon = Math.max(maxLon, boundaryPoint.getLongitude()); } return new LatLonRect(LatLonPoint.create(minLat, minLon), maxLat - minLat, maxLon - minLon); } /** * Calls {@link #calcConnectedLatLonBoundaryPoints(int, int)} with {@link Integer#MAX_VALUE} as both arguments. * In effect, the boundary will contain ALL of the points along the edges of the CRS. * * @return the connected latitude/longitude boundary of this CRS. */ public List calcConnectedLatLonBoundaryPoints() { return calcConnectedLatLonBoundaryPoints(Integer.MAX_VALUE, Integer.MAX_VALUE); } /** * Calculates the latitude/longitude boundary of this coordinate reference system. The boundary starts at the lower * left corner of the coverage--i.e. {@code (y[0], x[0])}--and consists of the points that lie along the bottom, * right, top, and left edges, in that order. *

* The {@code maxPointsInYEdge} parameter establishes a limit on the number of boundary points that'll be included * from the right and left edges. {@code maxPointsInXEdge} establishes a similar limit for the bottom and top edges. * The size of the returned list will be {@code ≤ 2 * maxPointsInYEdge + 2 * maxPointsInXEdge}. Note that the corners * are always included, regardless of the arguments. If you wish to include ALL of the points along the edges in * the boundary, simply choose values for the parameters that are greater than the lengths of the corresponding * axes in the CRS. {@link Integer#MAX_VALUE} works great. In that case, the size of the returned list will be * {@code 2 * numXcoords + 2 * numYcoords}. *

* If this CRS {@link #isProjection is a projection}, the lat/lon boundary is computed by converting each point * in its {@link #calcProjectionBoundaryPoints() projection boundary} to latitude/longitude using the * {@link Projection projection}. *

* Points in the boundary will be {@link #connectLatLonPoints connected}. This facilitates proper interpretation of * the boundary if it's rendered as a georeferenced polygon, particularly when the boundary crosses the international * date line. * * @param maxPointsInYEdge the maximum number of boundary points to include from the right and left edges. * @param maxPointsInXEdge the maximum number of boundary points to include from the bottom and top edges. * @return the connected latitude/longitude boundary of this CRS. */ public List calcConnectedLatLonBoundaryPoints(int maxPointsInYEdge, int maxPointsInXEdge) { List points; if (isProjection) { points = calcLatLonBoundaryPointsFromProjection(maxPointsInYEdge, maxPointsInXEdge); } else if (isLatLon1D) { points = calcLatLon1DBoundaryPoints(maxPointsInYEdge, maxPointsInXEdge); } else if (isLatLon2D) { points = calcLatLon2DBoundaryPoints(maxPointsInYEdge, maxPointsInXEdge); } else { throw new AssertionError("HorizCoordSys was not a projection, latLon1D, or latLon2D."); } return connectLatLonPoints(points); } /** * Calls {@link #calcProjectionBoundaryPoints(int, int)} with {@link Integer#MAX_VALUE} as both arguments. * In effect, the boundary will contain ALL of the points along the edges of the CRS. * * @return the boundary of this coordinate reference system, in projection coordinates. */ public List calcProjectionBoundaryPoints() { return calcProjectionBoundaryPoints(Integer.MAX_VALUE, Integer.MAX_VALUE); } /** * Calculates the boundary of this coordinate reference system, in projection coordinates. The boundary starts at the * lower left corner of the coverage--i.e. {@code (y[0], x[0])}--and consists of the points that lie along the bottom, * right, top, and left edges, in that order. *

* The {@code maxPointsInYEdge} parameter establishes a limit on the number of boundary points that'll be included * from the right and left edges. {@code maxPointsInXEdge} establishes a similar limit for the bottom and top edges. * The size of the returned list will be {@code ≤ 2 * maxPointsInYEdge + 2 * maxPointsInXEdge}. Note that the corners * are always included, regardless of the arguments. If you wish to include ALL of the points along the edges in * the boundary, simply choose values for the parameters that are greater than the lengths of the corresponding * axes in the CRS. {@link Integer#MAX_VALUE} works great. In that case, the size of the returned list will be * {@code 2 * numXcoords + 2 * numYcoords}. * * @param maxPointsInYEdge the maximum number of boundary points to include from the right and left edges. * @param maxPointsInXEdge the maximum number of boundary points to include from the bottom and top edges. * @return the boundary of this coordinate reference system, in projection coordinates. * @throws UnsupportedOperationException if this CRS is not a projection. */ public List calcProjectionBoundaryPoints(int maxPointsInYEdge, int maxPointsInXEdge) { if (!isProjection) { throw new UnsupportedOperationException("Coordinate system is not a projection."); } checkMaxPointsInEdges(maxPointsInYEdge, maxPointsInXEdge); int numYtotal = yAxis.getNcoords(); int numXtotal = xAxis.getNcoords(); int strideY = calcStride(numYtotal, maxPointsInYEdge); int strideX = calcStride(numXtotal, maxPointsInXEdge); List points = new LinkedList<>(); // Bottom boundary points for (int i = 0; i < numXtotal; i += strideX) { points.add(ProjectionPoint.create(xAxis.getCoordEdge1(i), yAxis.getCoordEdgeFirst())); } // Right boundary points for (int j = 0; j < numYtotal; j += strideY) { points.add(ProjectionPoint.create(xAxis.getCoordEdgeLast(), yAxis.getCoordEdge1(j))); } // Top boundary points for (int i = numXtotal - 1; i >= 0; i -= strideX) { points.add(ProjectionPoint.create(xAxis.getCoordEdge2(i), yAxis.getCoordEdgeLast())); } // Left boundary points for (int j = numYtotal - 1; j >= 0; j -= strideY) { points.add(ProjectionPoint.create(xAxis.getCoordEdgeFirst(), yAxis.getCoordEdge2(j))); } assertNotExceedingMaxBoundaryPoints(points.size(), maxPointsInYEdge, maxPointsInXEdge); return points; } private List calcLatLon1DBoundaryPoints(int maxPointsInYEdge, int maxPointsInXEdge) { if (!isLatLon1D) { throw new UnsupportedOperationException("Coordinate system is not 1D latitude/longitude."); } checkMaxPointsInEdges(maxPointsInYEdge, maxPointsInXEdge); int numYtotal = latAxis.getNcoords(); int numXtotal = lonAxis.getNcoords(); int strideY = calcStride(numYtotal, maxPointsInYEdge); int strideX = calcStride(numXtotal, maxPointsInXEdge); List points = new LinkedList<>(); // Bottom boundary points for (int i = 0; i < numXtotal; i += strideX) { points.add(LatLonPoint.create(latAxis.getCoordEdgeFirst(), lonAxis.getCoordEdge1(i))); } // Right boundary points for (int j = 0; j < numYtotal; j += strideY) { points.add(LatLonPoint.create(latAxis.getCoordEdge1(j), lonAxis.getCoordEdgeLast())); } // Top boundary points for (int i = numXtotal - 1; i >= 0; i -= strideX) { points.add(LatLonPoint.create(latAxis.getCoordEdgeLast(), lonAxis.getCoordEdge2(i))); } // Left boundary points for (int j = numYtotal - 1; j >= 0; j -= strideY) { points.add(LatLonPoint.create(latAxis.getCoordEdge2(j), lonAxis.getCoordEdgeFirst())); } assertNotExceedingMaxBoundaryPoints(points.size(), maxPointsInYEdge, maxPointsInXEdge); return points; } private List calcLatLonBoundaryPointsFromProjection(int maxPointsInYEdge, int maxPointsInXEdge) { List projPoints = calcProjectionBoundaryPoints(maxPointsInYEdge, maxPointsInXEdge); List latLonPoints = new LinkedList<>(); for (ProjectionPoint projPoint : projPoints) { final ProjectionPoint projPointInKm = ProjectionPoint.create(convertToDefaultUnits(projPoint.getX()), convertToDefaultUnits(projPoint.getY())); final LatLonPoint latLonPoint = transform.getProjection().projToLatLon(projPointInKm); if (!Double.isNaN(latLonPoint.getLatitude()) && !Double.isNaN(latLonPoint.getLongitude())) { latLonPoints.add(latLonPoint); } } return latLonPoints; } private double getCoordinateConversionFactor() { if (!isProjection) { return 1.0; } final String defaultUnits = transform.getProjection().getDefaultUnits(); final String unit = xAxis.getUnits(); return SimpleUnit.isCompatible(unit, defaultUnits) ? SimpleUnit.getConversionFactor(unit, defaultUnits) : 1.0; } private double convertToDefaultUnits(double coordinate) { return coordinate * coordinateConversionFactor; } private double convertFromDefaultUnits(double coordinateInDefaultUnit) { return coordinateInDefaultUnit / coordinateConversionFactor; } private List calcLatLon2DBoundaryPoints(int maxPointsInYEdge, int maxPointsInXEdge) { if (!isLatLon2D) { throw new UnsupportedOperationException("Coordinate system is not 2D latitude/longitude."); } checkMaxPointsInEdges(maxPointsInYEdge, maxPointsInXEdge); assert Arrays.equals(latAxis2D.getShape(), lonAxis2D.getShape()) : "2D lat/lon axes ought to have the same shape"; int[] midpointsShape = latAxis2D.getShape().clone(); // Clone because we will modify later. int numYtotal = midpointsShape[0]; int numXtotal = midpointsShape[1]; int strideY = calcStride(numYtotal, maxPointsInYEdge); int strideX = calcStride(numXtotal, maxPointsInXEdge); ArrayDouble.D2 latEdges = (ArrayDouble.D2) latAxis2D.getCoordBoundsAsArray(); ArrayDouble.D2 lonEdges = (ArrayDouble.D2) lonAxis2D.getCoordBoundsAsArray(); assert Arrays.equals(latEdges.getShape(), lonEdges.getShape()) : "2D lat/lon edges ought to have the same shape"; int[] edgesShape = latEdges.getShape(); midpointsShape[0]++; midpointsShape[1]++; assert Arrays.equals(midpointsShape, edgesShape) : "edgesShape should be 1 greater than midpointsShape in each dim"; List points = new LinkedList<>(); // Bottom boundary points: y = 0 for (int i = 0; i < numXtotal; i += strideX) { points.add(LatLonPoint.create(latEdges.get(0, i), lonEdges.get(0, i))); } // Right boundary points: x = numXtotal // numXtotal is not OOB, because edgesShape is 1 bigger than midpointsShape in each dim, and numXtotal // is relative to midpointsShape. for (int j = 0; j < numYtotal; j += strideY) { points.add(LatLonPoint.create(latEdges.get(j, numXtotal), lonEdges.get(j, numXtotal))); } // Top boundary points: y = numYtotal for (int i = numXtotal; i > 0; i -= strideX) { points.add(LatLonPoint.create(latEdges.get(numYtotal, i), lonEdges.get(numYtotal, i))); } // Left boundary points: x = 0 for (int j = numYtotal; j > 0; j -= strideY) { points.add(LatLonPoint.create(latEdges.get(j, 0), lonEdges.get(j, 0))); } assertNotExceedingMaxBoundaryPoints(points.size(), maxPointsInYEdge, maxPointsInXEdge); return points; } private static void checkMaxPointsInEdges(int maxPointsInYEdge, int maxPointsInXEdge) throws IllegalArgumentException { if (maxPointsInYEdge < 1) { throw new IllegalArgumentException(String.format("maxPointsInYEdge (%d) must be > 0", maxPointsInYEdge)); } else if (maxPointsInXEdge < 1) { throw new IllegalArgumentException(String.format("maxPointsInXEdge (%d) must be > 0", maxPointsInXEdge)); } } private static int calcStride(int numTotal, int maxToInclude) { int stride = Math.max(1, (int) Math.ceil(numTotal / (double) maxToInclude)); int numIncluded = (int) Math.ceil(numTotal / (double) stride); assert numIncluded <= maxToInclude : String.format("We're set to include %d points, but we wanted a max of %d.", numIncluded, maxToInclude); return stride; } private static void assertNotExceedingMaxBoundaryPoints(int numBoundaryPoints, int maxPointsInYEdge, int maxPointsInXEdge) { // Widen to long to avoid possible overflow, because last two arguments will often be Integer.MAX_VALUE. long maxBoundaryPoints = 2 * (long) maxPointsInYEdge + 2 * (long) maxPointsInXEdge; assert numBoundaryPoints <= maxBoundaryPoints : String.format( "We should be returning a maximum of %d boundary points, but we're returning %d instead.", maxBoundaryPoints, numBoundaryPoints); } /** * Returns a list of points that is equivalent to the input list, but with longitude values adjusted to ensure that * adjacent elements are "connected". *

* Two points are "connected" if the absolute difference of their {@link LatLonPoints#lonNormal normalized * longitudes} is {@code ≤180}. For example, the longitudes {@code 112} and {@code 124} are connected. So are * {@code 15} and {@code -27}. *

* Two points may be "disconnected" if they lie on opposite sides of the international date line. For example, * the longitudes {@code 175} and {@code -175} are disconnected because their absolute difference is {@code 350}, * which is {@code >180}. To connect the two points, we adjust the second longitude to an equivalent value * in the range {@code [firstLon ± 180]} by adding or subtracting {@code 360}. So, {@code -175} would become * {@code 185}. We perform this adjustment for each pair of adjacent elements in the list. *

* Performing the above adjustment will result in longitudes that lie outside of the normalized range of * ({@code [-180, 180]}). To be precise, if adjustments are necessary, all of the longitudes in the returned list * will be in either {@code [-360, 0]} or {@code [0, 360]}. Consequently, adjusted points cannot be returned as * {@link LatLonPoint}s; they are returned as {@link LatLonPointNoNormalize} objects instead. *

* Longitudes {@code lon1} and {@code lon2} are considered equivalent if {@code lon1 == lon2 + 360 * i}, for some * integer {@code i}. * * @param points a sequence of normalized lat/lon points that potentially crosses the international date line. * @return an equivalent sequence of points that has been adjusted to be "connected". */ public static List connectLatLonPoints(List points) { LinkedList connectedPoints = new LinkedList<>(); for (LatLonPoint point : points) { double curLat = point.getLatitude(); double curLon = point.getLongitude(); if (!connectedPoints.isEmpty()) { double prevLon = connectedPoints.getLast().getLongitude(); curLon = LatLonPoints.lonNormal(curLon, prevLon); } connectedPoints.add(new LatLonPointNoNormalize(curLat, curLon)); } return connectedPoints; } /** * Calls {@link #getLatLonBoundaryAsWKT(int, int)} with {@link Integer#MAX_VALUE} as both arguments. * In effect, the boundary will contain ALL of the points along the edges of the CRS. * * @return the latitude/longitude boundary of this CRS as a polygon in WKT. */ public String getLatLonBoundaryAsWKT() { return getLatLonBoundaryAsWKT(Integer.MAX_VALUE, Integer.MAX_VALUE); } /** * Returns the {@link #calcConnectedLatLonBoundaryPoints(int, int) latitude/longitude boundary} of this coordinate * reference system as a polygon in WKT. It is used in the OpenLayers map in NCSS, as well as the * "datasetBoundaries" endpoint. * * @param maxPointsInYEdge the maximum number of boundary points to include from the right and left edges. * @param maxPointsInXEdge the maximum number of boundary points to include from the bottom and top edges. * @return the latitude/longitude boundary of this CRS as a polygon in WKT. */ public String getLatLonBoundaryAsWKT(int maxPointsInYEdge, int maxPointsInXEdge) { List points = calcConnectedLatLonBoundaryPoints(maxPointsInYEdge, maxPointsInXEdge); StringBuilder sb = new StringBuilder("POLYGON(("); for (LatLonPointNoNormalize point : points) { sb.append(String.format(Locale.ROOT, "%.3f %.3f, ", point.getLongitude(), point.getLatitude())); } sb.delete(sb.length() - 2, sb.length()); // Nuke trailing comma and space. sb.append("))"); return sb.toString(); } /** * Calls {@link #getLatLonBoundaryAsGeoJSON(int, int)} with {@link Integer#MAX_VALUE} as both arguments. * In effect, the boundary will contain ALL of the points along the edges of the CRS. * * @return the latitude/longitude boundary of this CRS as a polygon in GeoJSON. */ public String getLatLonBoundaryAsGeoJSON() { return getLatLonBoundaryAsGeoJSON(Integer.MAX_VALUE, Integer.MAX_VALUE); } /** * Similar to {@link #getLatLonBoundaryAsWKT}, but returns a GeoJSON polygon instead. */ public String getLatLonBoundaryAsGeoJSON(int maxPointsInYEdge, int maxPointsInXEdge) { List points = calcConnectedLatLonBoundaryPoints(maxPointsInYEdge, maxPointsInXEdge); StringBuilder sb = new StringBuilder("{ 'type': 'Polygon', 'coordinates': [ [ "); for (LatLonPointNoNormalize point : points) { sb.append(String.format("[%.3f, %.3f], ", point.getLongitude(), point.getLatitude())); } sb.delete(sb.length() - 2, sb.length()); // Nuke trailing comma and space. sb.append(" ] ] }"); return sb.toString(); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy