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

ucar.nc2.dataset.CoordinateAxis1D Maven / Gradle / Ivy

The newest version!
/*
 * 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> 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); } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy