Mercurial > dive4elements > river
view flys-artifacts/src/main/java/de/intevation/flys/artifacts/states/WQSelect.java @ 4837:9e25c7523485
Fixed calculation of effective width in MINFO SQ relation.
* Get all (including empty datasets) from db.
* Filter empty datasets when processing data of the same date.
* Added debug outputs.
author | Raimund Renkert <rrenkert@intevation.de> |
---|---|
date | Wed, 23 Jan 2013 11:14:41 +0100 |
parents | 8e66293c5369 |
children |
line wrap: on
line source
package de.intevation.flys.artifacts.states; import java.text.NumberFormat; import gnu.trove.TDoubleArrayList; import org.apache.log4j.Logger; import org.w3c.dom.Element; import de.intevation.artifacts.Artifact; import de.intevation.artifacts.CallContext; import de.intevation.artifacts.common.utils.XMLUtils; import de.intevation.artifacts.common.utils.XMLUtils.ElementCreator; import de.intevation.artifactdatabase.ProtocolUtils; import de.intevation.artifactdatabase.data.StateData; import de.intevation.flys.model.Gauge; import de.intevation.flys.model.River; import de.intevation.flys.model.Wst; import de.intevation.flys.artifacts.FLYSArtifact; import de.intevation.flys.artifacts.WINFOArtifact; import de.intevation.flys.artifacts.model.WstFactory; import de.intevation.flys.artifacts.model.WstValueTable; import de.intevation.flys.artifacts.resources.Resources; import de.intevation.flys.utils.FLYSUtils; /** * @author <a href="mailto:ingo.weinzierl@intevation.de">Ingo Weinzierl</a> */ public class WQSelect extends DefaultState { /** The logger used in this class. */ private static Logger logger = Logger.getLogger(WQSelect.class); /** The default step width for Qs. */ public static final String DEFAULT_STEP_Q = "50"; /** The default step width for Qs. */ public static final String DEFAULT_STEP_W = "30"; /** The max number of steps for Qs and Ws. */ public static final int MAX_STEPS = 30; /** The name of the 'mode' field. */ public static final String WQ_MODE = "wq_isq"; /** Them name fo the 'free' field. */ public static final String WQ_FREE = "wq_isfree"; /** The name of the 'selection' field. */ public static final String WQ_SELECTION = "wq_isrange"; /** The name of the 'from' field. */ public static final String WQ_FROM = "wq_from"; /** The name of the 'to' field. */ public static final String WQ_TO = "wq_to"; /** The name of the 'step' field. */ public static final String WQ_STEP = "wq_step"; /** The name of the 'single' field. */ public static final String WQ_SINGLE = "wq_single"; /** * The default constructor that initializes an empty State object. */ public WQSelect() { } @Override protected Element createStaticData( FLYSArtifact flys, ElementCreator creator, CallContext cc, String name, String value, String type ) { if (!name.equals(WQ_SINGLE)) { return super.createStaticData(flys, creator, cc, name, value, type); } Boolean isQ = flys.getDataAsBoolean(WQ_MODE); Boolean isFree = flys.getDataAsBoolean(WQ_FREE); WINFOArtifact winfo = (WINFOArtifact) flys; Element dataElement = creator.create("data"); creator.addAttr(dataElement, "name", name, true); creator.addAttr(dataElement, "type", type, true); Element itemElement = creator.create("item"); creator.addAttr(itemElement, "value", value, true); String label; if (!isQ || isFree) { label = getLabel(winfo, cc, value); } else { label = getSpecialLabel(winfo, cc, value); } creator.addAttr(itemElement, "label", label, true); dataElement.appendChild(itemElement); return dataElement; } protected static String getLabel( WINFOArtifact winfo, CallContext cc, String raw ) { String[] values = raw.split(" "); if (values.length < 1) { return null; } StringBuilder label = new StringBuilder(); NumberFormat nf = NumberFormat.getInstance( Resources.getLocale(cc.getMeta())); for (String value: values) { try { double v = Double.parseDouble(value.trim()); String formatted = nf.format(v); if (label.length() > 0) { label.append(';'); } label.append(formatted); } catch (NumberFormatException nfe) { // do nothing here } } return label.toString(); } protected static String getSpecialLabel( WINFOArtifact winfo, CallContext cc, String raw ) { String[] values = raw.split(" "); if (values.length < 1) { return null; } NumberFormat nf = NumberFormat.getInstance( Resources.getLocale(cc.getMeta())); Gauge gauge = winfo.getGauge(); boolean debug = logger.isDebugEnabled(); StringBuilder label = new StringBuilder(); for (String value: values) { try { double v = Double.parseDouble(value.trim()); String tmp = nf.format(v); String mv = FLYSUtils.getNamedMainValue(gauge, v); if (mv != null && mv.length() > 0) { tmp = mv + ": " + tmp; if (debug) { logger.debug("Add main value: '" + mv + "'"); } } if (label.length() > 0) { label.append(';'); } label.append(tmp); } catch (NumberFormatException nfe) { // do nothing here } } return label.toString(); } @Override protected Element createData( XMLUtils.ElementCreator cr, Artifact artifact, StateData data, CallContext context) { Element select = ProtocolUtils.createArtNode( cr, "select", null, null); cr.addAttr(select, "name", data.getName(), true); Element label = ProtocolUtils.createArtNode( cr, "label", null, null); // XXX: DEAD CODE /* Element choices = ProtocolUtils.createArtNode( cr, "choices", null, null); */ label.setTextContent(Resources.getMsg( context.getMeta(), data.getName(), data.getName())); select.appendChild(label); return select; } @Override protected Element[] createItems( XMLUtils.ElementCreator cr, Artifact artifact, String name, CallContext context) { double[] minmaxW = determineMinMaxW(artifact); double[] minmaxWFree = determineMinMaxWFree(artifact); double[] minmaxQ = determineMinMaxQAtGauge(artifact); double[] minmaxQFree = determineMinMaxQ(artifact); if (name.equals("wq_from")) { Element minW = createItem(cr, new String[] { "minW", String.valueOf(minmaxW[0])}); Element minQ = createItem(cr, new String[] { "minQ", String.valueOf(minmaxQ[0])}); Element minQFree = createItem(cr, new String[] { "minQFree", String.valueOf(minmaxQFree[0])}); Element minWFree = createItem(cr, new String[] { "minWFree", String.valueOf(minmaxWFree[0])}); return new Element[] { minW, minQ, minQFree, minWFree }; } else if (name.equals("wq_to")) { Element maxW = createItem(cr, new String[] { "maxW", String.valueOf(minmaxW[1])}); Element maxQ = createItem(cr, new String[] { "maxQ", String.valueOf(minmaxQ[1])}); Element maxQFree = createItem(cr, new String[] { "maxQFree", String.valueOf(minmaxQFree[1])}); Element maxWFree = createItem(cr, new String[] { "maxWFree", String.valueOf(minmaxWFree[1])}); return new Element[] { maxW, maxQ, maxQFree, maxWFree }; } else { Element stepW = createItem( cr, new String[] { "stepW", String.valueOf(getStepsW(minmaxW[0], minmaxW[1]))}); Element stepQ = createItem( cr, new String[] { "stepQ", String.valueOf(getStepsQ(minmaxQ[0], minmaxQ[1]))}); Element stepQFree = createItem( cr, new String[] { "stepQFree", String.valueOf(getStepsQ(minmaxQFree[0], minmaxQFree[1]))}); Element stepWFree = createItem( cr, new String[] { "stepWFree", String.valueOf(getStepsW(minmaxWFree[0], minmaxWFree[1]))}); return new Element[] { stepW, stepQ, stepQFree, stepWFree }; } } protected static double getStepsW(double min, double max) { double diff = min < max ? max - min : min - max; double step = diff / MAX_STEPS; if (step < 10) { return getSteps(step, 1); } else if (step < 100) { return getSteps(step, 10); } else if (step < 1000) { return getSteps(step, 100); } else { return step; } } protected static double getStepsQ(double min, double max) { double diff = min < max ? max - min : min - max; double step = diff / MAX_STEPS; if (step < 10) { return getSteps(step, 1); } else if (step < 100) { return getSteps(step, 10); } else if (step < 1000) { return getSteps(step, 100); } else { return step; } } protected static double getSteps(double steps, double factor) { int fac = (int) (steps / factor); double diff = steps - fac * factor; if (diff == 0) { return steps; } return factor * (fac + 1); } protected Element createItem(XMLUtils.ElementCreator cr, Object obj) { Element item = ProtocolUtils.createArtNode(cr, "item", null, null); Element label = ProtocolUtils.createArtNode(cr, "label", null, null); Element value = ProtocolUtils.createArtNode(cr, "value", null, null); String[] arr = (String[]) obj; label.setTextContent(arr[0]); value.setTextContent(arr[1]); item.appendChild(label); item.appendChild(value); return item; } @Override protected String getUIProvider() { return "wq_panel"; } /** * Determines the min and max W value for the current gauge. If no min and * max values could be determined, this method will return * [Double.MIN_VALUE, Double.MAX_VALUE]. * * @param artifact The FLYSArtifact. * * @return the min and max W values for the current gauge. */ protected double[] determineMinMaxW(Artifact artifact) { logger.debug("WQSelect.determineCurrentGauge"); Gauge gauge = ((WINFOArtifact) artifact).getGauge(); double[] minmaxW = gauge != null ? gauge.determineMinMaxW() : null; double minW = minmaxW != null ? minmaxW[0] : Double.MIN_VALUE; double maxW = minmaxW != null ? minmaxW[1] : Double.MAX_VALUE; return new double[] { minW, maxW }; } /** * Determines the min and max W value. If no min and * max values could be determined, this method will return * [Double.MIN_VALUE, Double.MAX_VALUE]. * * @param artifact The FLYSArtifact. * * @return the min and max W values. */ protected double[] determineMinMaxWFree(Artifact artifact) { logger.debug("WQSelect.determineMinMaxWFree"); WINFOArtifact winfo = (WINFOArtifact) artifact; WstValueTable valueTable = winfo.getWstValueTable(); double[] minmaxW = null; if(valueTable != null) { double[] km = null; if(winfo.isRange()) { km = winfo.getFromToStep(); // Use the start km to determine the min max values. minmaxW = valueTable.getMinMaxW(km[0]); } else { km = winfo.getKms(); minmaxW = valueTable.getMinMaxW(km[0]); } } return minmaxW != null ? minmaxW : new double[] { Double.MIN_VALUE, Double.MAX_VALUE }; } /** * Determines the min and max Q value for the current gauge. If no min and * max values could be determined, this method will return * [Double.MIN_VALUE, Double.MAX_VALUE]. * * @param artifact The FLYSArtifact. * * @return the min and max Q values for the current gauge. */ protected double[] determineMinMaxQAtGauge(Artifact artifact) { logger.debug("WQSelect.determineMinMaxQAtGauge"); WINFOArtifact flysArtifact = (WINFOArtifact) artifact; River river = FLYSUtils.getRiver(flysArtifact); Gauge gauge = flysArtifact.getGauge(); Wst wst = WstFactory.getWst(river); double[] minmaxQ = gauge != null ? wst.determineMinMaxQ(gauge.getRange()) : null; double minQ = minmaxQ != null ? minmaxQ[0] : Double.MIN_VALUE; double maxQ = minmaxQ != null ? minmaxQ[1] : Double.MAX_VALUE; return new double[] { minQ, maxQ }; } /** * Determines the min and max Q value for the current kilometer range. If no * min and max values could be determined, this method will return * * @param artifact The FLYSArtifact. * * @return the min and max Q values for the current kilometer range. */ protected double[] determineMinMaxQ(Artifact artifact) { logger.debug("WQSelect.determineMinMaxQ"); WINFOArtifact winfo = (WINFOArtifact) artifact; WstValueTable valueTable = winfo.getWstValueTable(); double[] minmaxQ = null; if(valueTable != null) { double[] km = null; if(winfo.isRange()) { km = winfo.getFromToStep(); minmaxQ = valueTable.getMinMaxQ(km[0], km[1], km[2]); } else { km = winfo.getKms(); minmaxQ = valueTable.getMinMaxQ(km[0]); for (int i = 1; i < km.length; i++) { double[] tmp = valueTable.getMinMaxQ(km[i]); if(tmp[0] < minmaxQ[0]) { minmaxQ[0] = tmp[0]; } if(tmp[1] > minmaxQ[1]) { minmaxQ[1] = tmp[1]; } } } } return minmaxQ != null ? minmaxQ : new double[] { Double.MIN_VALUE, Double.MAX_VALUE }; } @Override public boolean validate(Artifact artifact) throws IllegalArgumentException { logger.debug("WQSelect.validate"); WINFOArtifact flys = (WINFOArtifact) artifact; StateData data = getData(flys, WQ_SELECTION); boolean isRange = data != null ? Boolean.valueOf((String) data.getValue()) : false; if (!isRange) { return validateSingle(artifact); } else { return validateRange(artifact); } } protected boolean validateBounds( double fromValid, double toValid, double from, double to, double step) throws IllegalArgumentException { logger.debug("RangeState.validateRange"); if (from < fromValid) { logger.error( "Invalid 'from'. " + from + " is smaller than " + fromValid); throw new IllegalArgumentException("error_feed_from_out_of_range"); } else if (to > toValid) { logger.error( "Invalid 'to'. " + to + " is bigger than " + toValid); throw new IllegalArgumentException("error_feed_to_out_of_range"); } return true; } protected boolean validateSingle(Artifact artifact) throws IllegalArgumentException { logger.debug("WQSelect.validateSingle"); WINFOArtifact flys = (WINFOArtifact) artifact; StateData data = getData(flys, WQ_SINGLE); String tmp = data != null ? (String) data.getValue() : null; if (tmp == null || tmp.length() == 0) { throw new IllegalArgumentException("error_empty_state"); } String[] strValues = tmp.split(" "); TDoubleArrayList all = new TDoubleArrayList(); for (String strValue: strValues) { try { all.add(Double.parseDouble(strValue)); } catch (NumberFormatException nfe) { logger.warn(nfe, nfe); } } all.sort(); FLYSUtils.WQ_MODE mode = FLYSUtils.getWQMode(flys); logger.debug("WQ Mode: " + mode); double[] minmax = null; if (mode == FLYSUtils.WQ_MODE.WGAUGE) { minmax = determineMinMaxW(artifact); } else if (mode == FLYSUtils.WQ_MODE.QGAUGE) { minmax = determineMinMaxQAtGauge(artifact); } else if (mode == FLYSUtils.WQ_MODE.QFREE) { minmax = determineMinMaxQ(artifact); } else { minmax = determineMinMaxWFree(artifact); } double min = all.get(0); double max = all.get(all.size()-1); logger.debug("Inserted min value = " + min); logger.debug("Inserted max value = " + max); return validateBounds(minmax[0], minmax[1], min, max, 0d); } protected boolean validateRange(Artifact artifact) throws IllegalArgumentException { logger.debug("WQSelect.validateRange"); WINFOArtifact flys = (WINFOArtifact) artifact; FLYSUtils.WQ_MODE mode = FLYSUtils.getWQMode(flys); if (mode == null) { throw new IllegalArgumentException("error_feed_invalid_wq_mode"); } StateData dFrom = flys.getData(WQ_FROM); StateData dTo = flys.getData(WQ_TO); StateData dStep = flys.getData(WQ_STEP); String fromStr = dFrom != null ? (String) dFrom.getValue() : null; String toStr = dTo != null ? (String) dTo.getValue() : null; String stepStr = dStep != null ? (String) dStep.getValue() : null; if (fromStr == null || toStr == null || stepStr == null) { throw new IllegalArgumentException("error_empty_state"); } try { double from = Double.parseDouble(fromStr); double to = Double.parseDouble(toStr); double step = Double.parseDouble(stepStr); if (mode == FLYSUtils.WQ_MODE.WGAUGE) { return validateGaugeW(artifact, from, to, step); } else if (mode == FLYSUtils.WQ_MODE.QGAUGE) { return validateGaugeQ(artifact, from, to, step); } else if (mode == FLYSUtils.WQ_MODE.QFREE) { return validateFreeQ(artifact, from, to, step); } else if (mode == FLYSUtils.WQ_MODE.WFREE) { return validateFreeW(artifact, from, to, step); } else { throw new IllegalArgumentException( "error_feed_invalid_wq_mode"); } } catch (NumberFormatException nfe) { throw new IllegalArgumentException("error_feed_number_format"); } } /** * Validates the inserted W values. * * @param artifact The owner artifact. * @param from The lower value of the W range. * @param to The upper value of the W range. * @param step The step width. * * @return true, if everything was fine, otherwise an exception is thrown. */ protected boolean validateGaugeW( Artifact artifact, double from, double to, double step) throws IllegalArgumentException { logger.debug("WQSelect.validateGaugeW"); double[] minmaxW = determineMinMaxW(artifact); return validateBounds(minmaxW[0], minmaxW[1], from, to, step); } /** * Validates the inserted Q values based on the Q range for the current * gauge. * * @param artifact The owner artifact. * @param from The lower value of the Q range. * @param to The upper value of the Q range. * @param step The step width. * * @return true, if everything was fine, otherwise an exception is thrown. */ protected boolean validateGaugeQ( Artifact artifact, double from, double to, double step) throws IllegalArgumentException { logger.debug("WQSelect.validateGaugeQ"); double[] minmaxQ = determineMinMaxQAtGauge(artifact); return validateBounds(minmaxQ[0], minmaxQ[1], from, to, step); } /** * Validates the inserted Q values based on the Q range for the current * kilometer range. * * @param artifact The owner artifact. * @param from The lower value of the Q range. * @param to The upper value of the Q range. * @param step The step width. * * @return true, if everything was fine, otherwise an exception is thrown. */ protected boolean validateFreeQ( Artifact artifact, double from, double to, double step) throws IllegalArgumentException { logger.debug("WQSelect.validateFreeQ"); double[] minmaxQ = determineMinMaxQ(artifact); return validateBounds(minmaxQ[0], minmaxQ[1], from, to, step); } /** * Validates the inserted W values based on the W range for the current * kilometer range. * * @param artifact The owner artifact. * @param from The lower value of the W range. * @param to The upper value of the W range. * @param step The step width. * * @return true, if everything was fine, otherwise an exception is thrown. */ protected boolean validateFreeW( Artifact artifact, double from, double to, double step) throws IllegalArgumentException { logger.debug("WQSelect.validateFreeW"); double[] minmaxW = determineMinMaxWFree(artifact); return validateBounds(minmaxW[0], minmaxW[1], from, to, step); } } // vim:set ts=4 sw=4 si et sta sts=4 fenc=utf-8 :