ucar.nc2.dataset.CoordinateAxis1D Maven / Gradle / Ivy
/*
* Copyright (c) 1998-2020 John Caron and University Corporation for Atmospheric Research/Unidata
* See LICENSE for license information.
*/
package ucar.nc2.dataset;
import ucar.ma2.Array;
import ucar.ma2.ArrayChar;
import ucar.ma2.DataType;
import ucar.ma2.Index;
import ucar.ma2.IndexIterator;
import ucar.ma2.InvalidRangeException;
import ucar.ma2.MAMath;
import ucar.ma2.Range;
import ucar.ma2.Section;
import ucar.nc2.Group;
import ucar.nc2.constants.AxisType;
import ucar.nc2.constants.CF;
import ucar.nc2.util.NamedObject;
import ucar.unidata.util.Format;
import java.io.IOException;
import java.util.ArrayList;
import java.util.List;
/**
* A 1-dimensional Coordinate Axis. Its values must be monotonic.
*
* If this is char valued, it will have rank 2, otherwise it will have rank 1.
*
* If string or char valued, only getCoordName() can be called.
*
* If the coordinates are regularly spaced, isRegular() is true, and the values are equal to
* getStart() + i * getIncrement().
*
* This will also set "cell bounds" for this axis. By default, the cell bounds are midway between the coordinates
* values,
* and are therefore contiguous, and can be accessed though getCoordEdge(i).
* The only way the bounds can be set is if the coordinate variable has an attribute "bounds" that points to another
* variable
* bounds(ncoords,2). These contain the cell bounds, and must be ascending or descending as the coordinate values are.
* In
* this case isContiguous() is true when bounds1(i+1) == bounds2(i) for all i.
*
* @author john caron
* @see CoordinateAxis#fromVariableDS
*/
public class CoordinateAxis1D extends CoordinateAxis {
private static org.slf4j.Logger log = org.slf4j.LoggerFactory.getLogger(CoordinateAxis1D.class);
/**
* Create a 1D coordinate axis from an existing Variable
*
* @param ncd the containing dataset
* @param vds wrap this VariableDS, which is not changed.
* @deprecated Use CoordinateAxis1D.builder()
*/
@Deprecated
public CoordinateAxis1D(NetcdfDataset ncd, VariableDS vds) {
super(ncd, vds);
vds.setCaching(true);
}
/**
* Copy constructor
*
* @param ncd ok to reparent
* @param org copy from here
* @deprecated Use CoordinateAxis1D.toBuilder()
*/
@Deprecated
CoordinateAxis1D(NetcdfDataset ncd, CoordinateAxis1D org) {
super(ncd, org);
this.orgName = org.orgName;
this.cache.reset(); // decouple cache
org.setCaching(true);
// copy rest of state
this.increment = org.getIncrement();
this.isAscending = org.isAscending;
this.isInterval = org.isInterval();
this.isRegular = org.isRegular();
if (isNumeric()) {
this.coords = org.getCoordValues();
this.edge = org.getCoordEdges();
}
this.names = org.names;
if (isInterval) {
this.bound1 = org.getBound1();
this.bound2 = org.getBound2();
}
this.wasBoundsDone = org.wasBoundsDone;
this.wasCalcRegular = org.wasCalcRegular;
this.wasRead = org.wasRead;
}
/**
* Constructor when theres no underlying variable. You better set the values too!
*
* @param ds the containing dataset.
* @param group the containing group; if null, use rootGroup
* @param shortName axis name.
* @param dataType data type
* @param dims list of dimension names
* @param units units of coordinates, preferably udunit compatible.
* @param desc long name.
* @deprecated Use CoordinateAxis1D.builder()
*/
@Deprecated
public CoordinateAxis1D(NetcdfDataset ds, Group group, String shortName, DataType dataType, String dims, String units,
String desc) {
super(ds, group, shortName, dataType, dims, units, desc);
}
/**
* Create a new CoordinateAxis1D as a section of this CoordinateAxis1D.
*
* @param r the section range
* @return a new CoordinateAxis1D as a section of this CoordinateAxis1D
* @throws InvalidRangeException if IllegalRange
*/
public CoordinateAxis1D section(Range r) throws InvalidRangeException {
Section section = Section.builder().appendRange(r).build();
CoordinateAxis1D result = (CoordinateAxis1D) section(section);
int len = r.length();
// deal with the midpoints, bounds
if (isNumeric()) {
double[] new_mids = new double[len];
for (int idx = 0; idx < len; idx++) {
int old_idx = r.element(idx);
new_mids[idx] = coords[old_idx];
}
result.coords = new_mids;
if (isInterval) {
double[] new_bound1 = new double[len];
double[] new_bound2 = new double[len];
double[] new_edge = new double[len + 1];
for (int idx = 0; idx < len; idx++) {
int old_idx = r.element(idx);
new_bound1[idx] = bound1[old_idx];
new_bound2[idx] = bound2[old_idx];
new_edge[idx] = bound1[old_idx];
new_edge[idx + 1] = bound2[old_idx]; // all but last are overwritten
}
result.bound1 = new_bound1;
result.bound2 = new_bound2;
result.edge = new_edge;
} else {
double[] new_edge = new double[len + 1];
for (int idx = 0; idx < len; idx++) {
int old_idx = r.element(idx);
new_edge[idx] = edge[old_idx];
new_edge[idx + 1] = edge[old_idx + 1]; // all but last are overwritten
}
result.edge = new_edge;
}
}
if (names != null) {
String[] new_names = new String[len];
for (int idx = 0; idx < len; idx++) {
int old_idx = r.element(idx);
new_names[idx] = names[old_idx];
}
result.names = new_names;
}
result.wasCalcRegular = false;
result.calcIsRegular();
return result;
}
// for section and slice
/** @deprecated Use {@link #toBuilder()} */
@Deprecated
@Override
protected CoordinateAxis1D copy() {
return new CoordinateAxis1D(this.ncd, this);
}
@Override
public CoordinateAxis copyNoCache() {
CoordinateAxis1D axis = new CoordinateAxis1D(ncd, getParentGroupOrRoot(), getShortName(), getDataType(),
getDimensionsString(), getUnitsString(), getDescription());
// other state
axis.axisType = this.axisType;
axis.boundaryRef = this.boundaryRef;
axis.isContiguous = this.isContiguous;
axis.positive = this.positive;
axis.cache.reset(); // decouple cache
return axis;
}
/**
* Get the list of names, to be used for user selection.
* The ith one refers to the ith coordinate.
*
* @return List of ucar.nc2.util.NamedObject, or empty list.
* @deprecated will move in ver 6
*/
@Deprecated
public List getNames() {
int n = getDimension(0).getLength();
List names = new ArrayList<>(n);
for (int i = 0; i < n; i++)
names.add(new ucar.nc2.util.NamedAnything(getCoordName(i), getShortName() + " " + getUnitsString()));
return names;
}
/**
* The "name" of the ith coordinate. If nominal, this is all there is to a coordinate.
* If numeric, this will return a String representation of the coordinate.
*
* @param index which one ?
* @return the ith coordinate value as a String
*/
public String getCoordName(int index) {
if (!wasRead)
doRead();
if (isNumeric())
return Format.d(getCoordValue(index), 5, 8);
else
return names[index];
}
/**
* Get the ith coordinate value. This is the value of the coordinate axis at which
* the data value is associated. These must be strictly monotonic.
*
* @param index which coordinate. Between 0 and getNumElements()-1 inclusive.
* @return coordinate value.
* @throws UnsupportedOperationException if !isNumeric()
*/
public double getCoordValue(int index) {
if (!isNumeric())
throw new UnsupportedOperationException("CoordinateAxis1D.getCoordValue() on non-numeric");
if (!wasRead)
doRead();
return coords[index];
}
@Override
public double getMinValue() {
if (!isNumeric())
throw new UnsupportedOperationException("CoordinateAxis1D.getCoordValue() on non-numeric");
if (!wasRead)
doRead();
return Math.min(coords[0], coords[coords.length - 1]);
}
@Override
public double getMaxValue() {
if (!isNumeric())
throw new UnsupportedOperationException("CoordinateAxis1D.getCoordValue() on non-numeric");
if (!wasRead)
doRead();
return Math.max(coords[0], coords[coords.length - 1]);
}
public double getMinEdgeValue() {
if (edge == null)
return getMinValue();
if (!isNumeric())
throw new UnsupportedOperationException("CoordinateAxis1D.getCoordValue() on non-numeric");
if (!wasRead)
doRead();
return Math.min(edge[0], edge[edge.length - 1]);
}
public double getMaxEdgeValue() {
if (edge == null)
return getMaxValue();
if (!isNumeric())
throw new UnsupportedOperationException("CoordinateAxis1D.getCoordValue() on non-numeric");
if (!wasRead)
doRead();
return Math.max(edge[0], edge[edge.length - 1]);
}
/**
* Get the ith coordinate edge. Exact only if isContiguous() is true, otherwise use getBound1() and getBound2().
* This is the value where the underlying grid element switches
* from "belonging to" coordinate value i-1 to "belonging to" coordinate value i.
* In some grids, this may not be well defined, and so should be considered an
* approximation or a visualization hint.
*
*
*
* Coordinate edges must be strictly monotonic:
* coordEdge(0) < coordValue(0) < coordEdge(1) < coordValue(1) ...
* ... coordEdge(i) < coordValue(i) < coordEdge(i+1) < coordValue(i+1) ...
* ... coordEdge(n-1) < coordValue(n-1) < coordEdge(n)
*
*
* @param index which coordinate. Between 0 and getNumElements() inclusive.
* @return coordinate edge.
* @throws UnsupportedOperationException if !isNumeric()
*/
public double getCoordEdge(int index) {
if (!isNumeric())
throw new UnsupportedOperationException("CoordinateAxis1D.getCoordEdge() on non-numeric");
if (!wasBoundsDone)
makeBounds();
return edge[index];
}
/**
* Get the coordinate values as a double array.
*
* @return coordinate value.
* @throws UnsupportedOperationException if !isNumeric()
*/
public double[] getCoordValues() {
if (!isNumeric())
throw new UnsupportedOperationException("CoordinateAxis1D.getCoordValues() on non-numeric");
if (!wasRead)
doRead();
return coords.clone();
}
/**
* Get the coordinate edges as a double array.
* Exact only if isContiguous() is true, otherwise use getBound1() and getBound2().
*
* @return coordinate edges.
* @throws UnsupportedOperationException if !isNumeric()
*/
public double[] getCoordEdges() {
if (!isNumeric())
throw new UnsupportedOperationException("CoordinateAxis1D.getCoordEdges() on non-numeric");
if (!wasBoundsDone)
makeBounds();
return edge.clone();
}
@Override
public boolean isContiguous() {
if (!wasBoundsDone)
makeBounds(); // this sets isContiguous
return isContiguous;
}
///////////////////////////////////////////////
/**
* If this coordinate has interval values.
* If so, then one should use getBound1, getBound2, and not getCoordEdges()
*
* @return true if coordinate has interval values
*/
public boolean isInterval() {
if (!wasBoundsDone)
makeBounds(); // this sets isInterval
return isInterval;
}
/**
* Get the coordinate bound1 as a double array.
* bound1[i] # coordValue[i] # bound2[i], where # is < if increasing (bound1[i] < bound1[i+1])
* else < if decreasing.
*
* @return coordinate bound1.
* @throws UnsupportedOperationException if !isNumeric()
*/
public double[] getBound1() {
if (!isNumeric())
throw new UnsupportedOperationException("CoordinateAxis1D.getBound1() on non-numeric");
if (!wasBoundsDone)
makeBounds();
if (bound1 == null)
makeBoundsFromEdges();
assert bound1 != null;
return bound1.clone();
}
/**
* Get the coordinate bound1 as a double array.
* bound1[i] # coordValue[i] # bound2[i], where # is < if increasing (bound1[i] < bound1[i+1])
* else < if decreasing.
*
* @return coordinate bound2.
* @throws UnsupportedOperationException if !isNumeric()
*/
public double[] getBound2() {
if (!isNumeric())
throw new UnsupportedOperationException("CoordinateAxis1D.getBound2() on non-numeric");
if (!wasBoundsDone)
makeBounds();
if (bound2 == null)
makeBoundsFromEdges();
assert bound2 != null;
return bound2.clone();
}
/**
* Get the coordinate bounds for the ith coordinate.
* Can use this for isContiguous() true or false.
*
* @param i coordinate index
* @return double[2] edges for ith coordinate
*/
public double[] getCoordBounds(int i) {
if (!wasBoundsDone)
makeBounds();
double[] e = new double[2];
if (isContiguous()) {
e[0] = getCoordEdge(i);
e[1] = getCoordEdge(i + 1);
} else {
e[0] = bound1[i];
e[1] = bound2[i];
}
return e;
}
public double getCoordBoundsMidpoint(int i) {
double[] bounds = getCoordBounds(i);
return (bounds[0] + bounds[1]) / 2;
}
/**
* Given a coordinate value, find what grid element contains it.
* This means that
*
*
* edge[i] <= value < edge[i+1] (if values are ascending)
* edge[i] > value >= edge[i+1] (if values are descending)
*
*
* @param coordVal position in this coordinate system
* @return index of grid point containing it, or -1 if outside grid area
*/
public int findCoordElement(double coordVal) {
if (!isNumeric())
throw new UnsupportedOperationException("CoordinateAxis.findCoordElement() on non-numeric");
if (isRegular())
return findCoordElementRegular(coordVal, false);
if (isContiguous())
return findCoordElementIrregular(coordVal, false);
else
return findCoordElementNonContiguous(coordVal, false);
}
/**
* Given a coordinate position, find what grid element contains it, or is closest to it.
*
* @param coordVal position in this coordinate system
* @return index of grid point containing it, or best estimate of closest grid interval.
*/
public int findCoordElementBounded(double coordVal) {
if (!isNumeric())
throw new UnsupportedOperationException("CoordinateAxis.findCoordElementBounded() on non-numeric");
// the scalar or len-1 case:
if (this.getSize() == 1)
return 0;
if (isRegular())
return findCoordElementRegular(coordVal, true);
if (isContiguous())
return findCoordElementIrregular(coordVal, true);
else
return findCoordElementNonContiguous(coordVal, true);
}
/**
* @deprecated use findCoordElement(coordVal)
*/
public int findCoordElement(double coordVal, int lastIndex) {
return findCoordElement(coordVal);
}
//////////////////////////////////////////////////////////////////
// following is from Jon Blower's ncWMS
// faster routines for coordValue -> index search
// significantly modified
/**
* Optimize the regular case
* Gets the index of the given point. Uses index = (value - start) / stride,
* hence this is faster than an exhaustive search.
* from jon blower's ncWMS.
*
* @param coordValue The value along this coordinate axis
* @param bounded if false and not in range, return -1, else nearest index
* @return the index that is nearest to this point, or -1 if the point is
* out of range for the axis
*/
private int findCoordElementRegular(double coordValue, boolean bounded) {
int n = (int) this.getSize();
// the scalar or len-1 case:
if (this.getSize() == 1) {
return 0;
}
/*
* if (axisType == AxisType.Lon) {
* double maxValue = this.start + this.increment * n;
* if (betweenLon(coordValue, this.start, maxValue)) {
* double distance = LatLonPointImpl.getClockwiseDistanceTo(this.start, coordValue);
* double exactNumSteps = distance / this.increment;
* // This axis might wrap, so we make sure that the returned index is within range
* return ((int) Math.round(exactNumSteps)) % (int) this.getSize();
*
* } else if (coordValue < this.start) {
* return bounded ? 0 : -1;
* } else {
* return bounded ? n - 1 : -1;
* }
* }
*/
double distance = coordValue - this.start;
double exactNumSteps = distance / this.increment;
int index = (int) Math.round(exactNumSteps);
if (index < 0)
return bounded ? 0 : -1;
else if (index >= n)
return bounded ? n - 1 : -1;
return index;
}
private boolean betweenLon(double lon, double lonBeg, double lonEnd) {
while (lon < lonBeg)
lon += 360;
return (lon >= lonBeg) && (lon <= lonEnd);
}
/**
* Performs a binary search to find the index of the element of the array
* whose value is contained in the interval, so must be contiguous.
*
* @param target The value to search for
* @param bounded if false, and not in range, return -1, else nearest index
* @return the index of the element in values whose value is closest to target,
* or -1 if the target is out of range
*/
private int findCoordElementIrregular(double target, boolean bounded) {
int n = (int) this.getSize();
int low = 0;
int high = n;
if (isAscending) {
// Check that the point is within range
if (target < this.edge[low])
return bounded ? 0 : -1;
else if (target > this.edge[high])
return bounded ? n - 1 : -1;
// do a binary search to find the nearest index
int mid;
while (high > low + 1) {
mid = (low + high) / 2;
double midVal = this.edge[mid];
if (midVal == target)
return mid;
else if (midVal < target)
low = mid;
else
high = mid;
}
return low;
} else {
// Check that the point is within range
if (target > this.edge[low])
return bounded ? 0 : -1;
else if (target < this.edge[high])
return bounded ? n - 1 : -1;
// do a binary search to find the nearest index
int mid;
while (high > low + 1) {
mid = (low + high) / 2;
double midVal = this.edge[mid];
if (midVal == target)
return mid;
else if (midVal < target)
high = mid;
else
low = mid;
}
return high - 1;
}
}
/**
* Given a coordinate position, find what grid element contains it.
* Only use if isContiguous() == false
* This algorithm does a linear search in the bound1[] amd bound2[] array.
*
* This means that
*
*
* edge[i] <= pos < edge[i+1] (if values are ascending)
* edge[i] > pos >= edge[i+1] (if values are descending)
*
*
* @param target The value to search for
* @param bounded if false, and not in range, return -1, else nearest index
* @return the index of the element in values whose value is closest to target,
* or -1 if the target is out of range
*/
private int findCoordElementNonContiguous(double target, boolean bounded) {
double[] bounds1 = getBound1();
double[] bounds2 = getBound2();
int n = bounds1.length;
if (isAscending) {
// Check that the point is within range
if (target < bounds1[0])
return bounded ? 0 : -1;
else if (target > bounds2[n - 1])
return bounded ? n - 1 : -1;
int[] idx = findSingleHit(bounds1, bounds2, target);
if (idx[0] == 0 && !bounded)
return -1; // no hits
if (idx[0] == 1)
return idx[1]; // one hit
// multiple hits = choose closest to the midpoint i guess
return findClosest(coords, target);
} else {
// Check that the point is within range
if (target > bounds1[0])
return bounded ? 0 : -1;
else if (target < bounds2[n - 1])
return bounded ? n - 1 : -1;
int[] idx = findSingleHit(bounds2, bounds1, target);
if (idx[0] == 0 && !bounded)
return -1; // no hits
if (idx[0] == 1)
return idx[1];
// multiple hits = choose closest to the midpoint i guess
return findClosest(getCoordValues(), target);
}
}
// return index if only one match, else -1
private int[] findSingleHit(double[] low, double[] high, double target) {
int hits = 0;
int idxFound = -1;
int n = low.length;
for (int i = 0; i < n; i++) {
if ((low[i] <= target) && (target <= high[i])) {
hits++;
idxFound = i;
}
}
return new int[] {hits, idxFound};
}
// return index of closest value to target
private int findClosest(double[] values, double target) {
double minDiff = Double.MAX_VALUE;
int idxFound = -1;
int n = values.length;
for (int i = 0; i < n; i++) {
double diff = Math.abs(values[i] - target);
if (diff < minDiff) {
minDiff = diff;
idxFound = i;
}
}
return idxFound;
}
///////////////////////////////////////////////////////////////////////////////
// check if Regular
/**
* Get starting value if isRegular()
*
* @return starting value if isRegular()
*/
public double getStart() {
calcIsRegular();
return start;
}
/**
* Get increment value if isRegular()
*
* @return increment value if isRegular()
*/
public double getIncrement() {
calcIsRegular();
return increment;
}
/**
* If true, then value(i) = getStart() + i * getIncrement().
*
* @return if evenly spaced.
*/
public boolean isRegular() {
calcIsRegular();
return isRegular;
}
private void calcIsRegular() {
if (wasCalcRegular)
return;
if (!wasRead)
doRead();
if (!isNumeric())
isRegular = false;
else if (getSize() < 2)
isRegular = true;
else {
start = getCoordValue(0);
int n = (int) getSize();
increment = (getCoordValue(n - 1) - getCoordValue(0)) / (n - 1);
isRegular = true;
for (int i = 1; i < getSize(); i++)
if (!ucar.nc2.util.Misc.nearlyEquals(getCoordValue(i) - getCoordValue(i - 1), increment, 5.0e-3)) {
isRegular = false;
break;
}
}
wasCalcRegular = true;
}
///////////////////////////////////////////////////////////////////////////////
private void doRead() {
if (isNumeric()) {
readValues();
wasRead = true;
if (getSize() < 2)
isAscending = true;
else
isAscending = getCoordValue(0) < getCoordValue(1);
// calcIsRegular(); */
} else if (getDataType() == DataType.STRING) {
readStringValues();
wasRead = true;
} else {
readCharValues();
wasRead = true;
}
}
// LOOK turns longitude coordinate into monotonic, dealing with possible wrap.
/** @deprecated do not use. */
@Deprecated
public void correctLongitudeWrap() {
// correct non-monotonic longitude coords
if (axisType != AxisType.Lon) {
return;
}
if (!wasRead)
doRead();
if (!wasBoundsDone)
makeBounds();
boolean monotonic = true;
for (int i = 0; i < coords.length - 1; i++)
monotonic &= isAscending ? coords[i] < coords[i + 1] : coords[i] > coords[i + 1];
if (!monotonic) {
boolean cross = false;
if (isAscending) {
for (int i = 0; i < coords.length; i++) {
if (cross)
coords[i] += 360;
if (!cross && (i < coords.length - 1) && (coords[i] > coords[i + 1]))
cross = true;
}
} else {
for (int i = 0; i < coords.length; i++) {
if (cross)
coords[i] -= 360;
if (!cross && (i < coords.length - 1) && (coords[i] < coords[i + 1]))
cross = true;
}
}
// LOOK - need to make sure we get stuff from the cache
Array cachedData = Array.factory(DataType.DOUBLE, getShape(), coords);
if (getDataType() != DataType.DOUBLE)
cachedData = MAMath.convert(cachedData, getDataType());
setCachedData(cachedData);
if (!isInterval) {
makeEdges();
}
}
}
// only used if String
private void readStringValues() {
int count = 0;
Array data;
try {
data = read();
} catch (IOException ioe) {
log.error("Error reading string coordinate values ", ioe);
throw new IllegalStateException(ioe);
}
names = new String[(int) data.getSize()];
IndexIterator ii = data.getIndexIterator();
while (ii.hasNext())
names[count++] = (String) ii.getObjectNext();
}
private void readCharValues() {
int count = 0;
ArrayChar data;
try {
data = (ArrayChar) read();
} catch (IOException ioe) {
log.error("Error reading char coordinate values ", ioe);
throw new IllegalStateException(ioe);
}
ArrayChar.StringIterator iter = data.getStringIterator();
names = new String[iter.getNumElems()];
while (iter.hasNext())
names[count++] = iter.next();
}
protected void readValues() {
Array data;
try {
// setUseNaNs(false); // missing values not allowed LOOK not true for point data !!
data = read();
// if (!hasCachedData()) setCachedData(data, false); //cache data for subsequent reading
} catch (IOException ioe) {
log.error("Error reading coordinate values ", ioe);
throw new IllegalStateException(ioe);
}
coords = (double[]) data.get1DJavaArray(DataType.DOUBLE);
this.wasRead = true;
// IndexIterator iter = data.getIndexIterator();
// while (iter.hasNext())
// coords[count++] = iter.getDoubleNext();
}
/**
* Calculate bounds, set isInterval, isContiguous
*/
private void makeBounds() {
if (!wasRead)
doRead();
if (isNumeric()) {
if (!makeBoundsFromAux()) {
makeEdges();
}
}
wasBoundsDone = true;
}
private boolean makeBoundsFromAux() {
String boundsVarName = attributes().findAttributeString(CF.BOUNDS, null);
if (boundsVarName == null) {
return false;
}
VariableDS boundsVar = (VariableDS) getParentGroupOrRoot().findVariableLocal(boundsVarName);
if (null == boundsVar)
return false;
if (2 != boundsVar.getRank())
return false;
if (getDimension(0) != boundsVar.getDimension(0))
return false;
if (2 != boundsVar.getDimension(1).getLength())
return false;
Array data;
try {
// LOOK this seems bogus
boundsVar.removeEnhancement(NetcdfDataset.Enhance.ConvertMissing); // Don't convert missing values to NaN.
data = boundsVar.read();
} catch (IOException e) {
log.warn("CoordinateAxis1D.hasBounds read failed ", e);
return false;
}
assert (data.getRank() == 2) && (data.getShape()[1] == 2) : "incorrect shape data for variable " + boundsVar;
// extract the bounds
int n = shape[0];
double[] value1 = new double[n];
double[] value2 = new double[n];
Index ima = data.getIndex();
for (int i = 0; i < n; i++) {
ima.set0(i);
value1[i] = data.getDouble(ima.set1(0));
value2[i] = data.getDouble(ima.set1(1));
}
/*
* flip if needed
* boolean firstLower = true; // in the first interval, is lower < upper ?
* for (int i = 0; i < value1.length; i++) {
* if (Misc.nearlyEquals(value1[i], value2[i])) continue; // skip when lower == upper
* firstLower = value1[i] < value2[i];
* break;
* }
* // check first against last : lower, unless all lower equal then upper
* boolean goesUp = (n < 2) || value1[n - 1] > value1[0] || (Misc.nearlyEquals(value1[n - 1], value2[0]) && value2[n
* - 1] > value2[0]);
* if (goesUp != firstLower) {
* double[] temp = value1;
* value1 = value2;
* value2 = temp;
* }
*/
// decide if they are contiguous
boolean contig = true;
for (int i = 0; i < n - 1; i++) {
if (!ucar.nc2.util.Misc.nearlyEquals(value1[i + 1], value2[i]))
contig = false;
}
if (contig) {
edge = new double[n + 1];
edge[0] = value1[0];
for (int i = 1; i < n + 1; i++)
edge[i] = value2[i - 1];
} else { // what does edge mean when not contiguous ??
edge = new double[n + 1];
edge[0] = value1[0];
for (int i = 1; i < n; i++)
edge[i] = (value1[i] + value2[i - 1]) / 2;
edge[n] = value2[n - 1];
isContiguous = false;
}
bound1 = value1;
bound2 = value2;
isInterval = true;
return true;
}
private void makeEdges() {
int size = (int) getSize();
edge = new double[size + 1];
if (size < 1) {
return;
}
if (size == 1) {
// if the axis size is one, edges are the same as the coord
edge[0] = coords[0];
edge[1] = coords[0];
return;
}
for (int i = 1; i < size; i++) {
edge[i] = (coords[i - 1] + coords[i]) / 2;
}
edge[0] = coords[0] - (edge[1] - coords[0]);
edge[size] = coords[size - 1] + (coords[size - 1] - edge[size - 1]);
isContiguous = true;
}
private void makeBoundsFromEdges() {
int size = (int) getSize();
if (size == 0)
return;
bound1 = new double[size];
bound2 = new double[size];
for (int i = 0; i < size; i++) {
bound1[i] = edge[i];
bound2[i] = edge[i + 1];
}
// flip if needed
if (bound1[0] > bound2[0]) {
double[] temp = bound1;
bound1 = bound2;
bound2 = temp;
}
}
////////////////////////////////////////////////////////////////////////////////////////////
// These are all calculated, I think?
protected boolean wasRead; // have the data values been read
private boolean wasBoundsDone; // have we created the bounds arrays if exists ?
private boolean isInterval; // is this an interval coordinates - then should use bounds
private boolean isAscending;
// read in on doRead()
protected double[] coords; // coordinate values, must be between edges
private String[] names; // only set if String or char values
// defer making until asked, use makeBounds()
private double[] edge; // n+1 edges, edge[k] < midpoint[k] < edge[k+1]
private double[] bound1, bound2; // may be contiguous or not
private boolean wasCalcRegular; // have we checked if the data is regularly spaced ?
private boolean isRegular;
private double start, increment;
protected CoordinateAxis1D(Builder> builder, Group parentGroup) {
super(builder, parentGroup);
}
public Builder> toBuilder() {
return addLocalFieldsToBuilder(builder());
}
// Add local fields to the passed - in builder.
protected Builder> addLocalFieldsToBuilder(Builder extends Builder>> b) {
return (Builder>) super.addLocalFieldsToBuilder(b);
}
/**
* Get Builder for this class that allows subclassing.
*
* @see "https://community.oracle.com/blogs/emcmanus/2010/10/24/using-builder-pattern-subclasses"
*/
public static Builder> builder() {
return new Builder2();
}
private static class Builder2 extends Builder {
@Override
protected Builder2 self() {
return this;
}
}
public static abstract class Builder> extends CoordinateAxis.Builder {
private boolean built;
protected abstract T self();
public CoordinateAxis1D build(Group parentGroup) {
if (built)
throw new IllegalStateException("already built");
built = true;
return new CoordinateAxis1D(this, parentGroup);
}
}
}