From 55bdb6e2b990cdf6eafd56ace2fa32d59008a9b0 Mon Sep 17 00:00:00 2001 From: Andrea Date: Fri, 9 Mar 2018 18:13:54 +0100 Subject: [PATCH] Fix #2668 Increase precision of Query Panel Circle/BBox Details (#2720) --- .../components/data/query/GeometryDetails.jsx | 67 +++++++++++-------- .../components/data/query/QueryBuilder.jsx | 6 +- .../components/data/query/SpatialFilter.jsx | 9 ++- .../enhancers/__tests__/debounce-test.jsx | 46 +++++++++++++ .../components/misc/enhancers/debounce.js | 27 ++++++++ web/client/plugins/QueryPanel.jsx | 5 +- web/client/translations/data.de-DE | 2 +- web/client/translations/data.en-US | 2 +- web/client/translations/data.es-ES | 2 +- web/client/translations/data.fr-FR | 2 +- web/client/translations/data.it-IT | 2 +- web/client/translations/data.nl-NL | 2 +- web/client/translations/data.zh-ZH | 2 +- 13 files changed, 131 insertions(+), 43 deletions(-) create mode 100644 web/client/components/misc/enhancers/__tests__/debounce-test.jsx create mode 100644 web/client/components/misc/enhancers/debounce.js diff --git a/web/client/components/data/query/GeometryDetails.jsx b/web/client/components/data/query/GeometryDetails.jsx index 23edd46a85..31037f55c2 100644 --- a/web/client/components/data/query/GeometryDetails.jsx +++ b/web/client/components/data/query/GeometryDetails.jsx @@ -16,6 +16,7 @@ const assign = require('object-assign'); const CoordinatesUtils = require("../../../utils/CoordinatesUtils"); + class GeometryDetails extends React.Component { static propTypes = { useMapProjection: PropTypes.bool, @@ -23,7 +24,8 @@ class GeometryDetails extends React.Component { type: PropTypes.string, onShowPanel: PropTypes.func, onChangeDrawingStatus: PropTypes.func, - onEndDrawing: PropTypes.func + onEndDrawing: PropTypes.func, + zoom: PropTypes.number }; static defaultProps = { @@ -101,7 +103,6 @@ class GeometryDetails extends React.Component { onModifyGeometry = () => { let geometry; - // Update the geometry if (this.props.type === "BBOX") { this.extent = this.tempExtent; @@ -168,28 +169,27 @@ class GeometryDetails extends React.Component { }; onClosePanel = () => { - if (this.props.type === "BBOX") { - this.resetBBOX(); - } else if (this.props.type === "Circle") { - this.resetCircle(); - } - + this.resetGeom(); this.props.onShowPanel(false); }; - + getStep = (zoom = 1) => Math.min(1 / Math.pow(10, Math.ceil(Math.min(zoom, 21) / 3) - 2), 1); + getStepCircle = (zoom, name) => { + const step = this.getStep(zoom); + return name === 'radius' && step * 10000 || step; + }; getBBOXDimensions = (geometry) => { const extent = geometry.projection !== 'EPSG:4326' && !this.props.useMapProjection ? CoordinatesUtils.reprojectBbox(geometry.extent, geometry.projection, 'EPSG:4326') : geometry.extent; return { // minx - west: Math.round(extent[0] * 100) / 100, + west: extent[0], // miny - sud: Math.round(extent[1] * 100) / 100, + sud: extent[1], // maxx - est: Math.round(extent[2] * 100) / 100, + est: extent[2], // maxy - north: Math.round(extent[3] * 100) / 100 + north: extent[3] }; }; getCircleDimensions = (geometry) => { @@ -201,9 +201,9 @@ class GeometryDetails extends React.Component { center = (center.x === undefined) ? {x: center[0], y: center[1]} : center; return { - x: Math.round(center.x * 100) / 100, - y: Math.round(center.y * 100) / 100, - radius: Math.round(geometry.radius * 100) / 100 + x: center.x, + y: center.y, + radius: geometry.radius }; }; renderCoordinateField = (value, name) => { @@ -214,18 +214,19 @@ class GeometryDetails extends React.Component { style={{minWidth: '105px', margin: 'auto'}} type="number" id={"queryform_bbox_" + name} - defaultValue={value} + step={!this.isWGS84() ? 1 : this.getStep(this.props.zoom)} + defaultValue={this.roundValue(value, !this.isWGS84() ? 100 : 1000000)} onChange={(evt) => this.onUpdateBBOX(evt.target.value, name)}/> ); }; - renderCircleField = (value, name) => { return ( this.onUpdateCircle(evt.target.value, name)}/> ); }; @@ -352,7 +353,7 @@ class GeometryDetails extends React.Component { key: 'reset', tooltipId: 'queryform.reset', glyph: 'clear-filter', - onClick: () => this.resetBBOX() + onClick: () => this.resetGeom() }, { key: 'close', glyph: '1-close', @@ -362,29 +363,37 @@ class GeometryDetails extends React.Component { ); } - + isWGS84 = () => (this.props.geometry || {}).projection === 'EPSG:4326' || !this.props.useMapProjection; + roundValue = (val, prec = 1000000) => Math.round(val * prec) / prec; + resetGeom = () => { + if (this.props.type === "BBOX") { + this.resetBBOX(); + } else if (this.props.type === "Circle") { + this.resetCircle(); + } + }; resetBBOX = () => { for (let prop in this.extent) { if (prop) { let coordinateInput = document.getElementById("queryform_bbox_" + prop); - coordinateInput.value = this.extent[prop]; - this.onUpdateBBOX(coordinateInput.value, prop); + coordinateInput.value = this.roundValue(this.extent[prop], !this.isWGS84() ? 100 : 1000000); + this.onUpdateBBOX(this.extent[prop], prop); } } }; resetCircle = () => { let radiusInput = document.getElementById("queryform_circle_radius"); - radiusInput.value = this.circle.radius; - this.onUpdateCircle(radiusInput.value, "radius"); + radiusInput.value = this.roundValue(this.circle.radius, 100); + this.onUpdateCircle(this.circle.radius, "radius"); let coordinateXInput = document.getElementById("queryform_circle_x"); - coordinateXInput.value = this.circle.x; - this.onUpdateCircle(coordinateXInput.value, "x"); + coordinateXInput.value = this.roundValue(this.circle.x, !this.isWGS84() ? 100 : 1000000); + this.onUpdateCircle(this.circle.x, "x"); let coordinateYInput = document.getElementById("queryform_circle_y"); - coordinateYInput.value = this.circle.y; - this.onUpdateCircle(coordinateYInput.value, "y"); + coordinateYInput.value = this.roundValue(this.circle.y, !this.isWGS84() ? 100 : 1000000); + this.onUpdateCircle(this.circle.y, "y"); }; } diff --git a/web/client/components/data/query/QueryBuilder.jsx b/web/client/components/data/query/QueryBuilder.jsx index 7e527d3d67..98ad7fdb63 100644 --- a/web/client/components/data/query/QueryBuilder.jsx +++ b/web/client/components/data/query/QueryBuilder.jsx @@ -62,7 +62,8 @@ class QueryBuilder extends React.Component { allowEmptyFilter: PropTypes.bool, autocompleteEnabled: PropTypes.bool, emptyFilterWarning: PropTypes.bool, - header: PropTypes.node + header: PropTypes.node, + zoom: PropTypes.number }; static defaultProps = { @@ -180,7 +181,8 @@ class QueryBuilder extends React.Component { spatialMethodOptions={this.props.spatialMethodOptions} spatialPanelExpanded={this.props.spatialPanelExpanded} showDetailsPanel={this.props.showDetailsPanel} - actions={this.props.spatialFilterActions}/> + actions={this.props.spatialFilterActions} + zoom={this.props.zoom}/> ) + onEndDrawing={this.props.actions.onEndDrawing} + zoom={this.props.zoom}/>) : ; diff --git a/web/client/components/misc/enhancers/__tests__/debounce-test.jsx b/web/client/components/misc/enhancers/__tests__/debounce-test.jsx new file mode 100644 index 0000000000..d0aa09a0a1 --- /dev/null +++ b/web/client/components/misc/enhancers/__tests__/debounce-test.jsx @@ -0,0 +1,46 @@ +/* + * Copyright 2018, GeoSolutions Sas. + * All rights reserved. + * + * This source code is licensed under the BSD-style license found in the + * LICENSE file in the root directory of this source tree. + */ + +const React = require('react'); +const ReactDOM = require('react-dom'); +const {createSink, compose} = require('recompose'); +const expect = require('expect'); +const debounce = require('../debounce'); + +describe('debounce enhancer', () => { + beforeEach((done) => { + document.body.innerHTML = '
'; + setTimeout(done); + }); + afterEach((done) => { + ReactDOM.unmountComponentAtNode(document.getElementById("container")); + document.body.innerHTML = ''; + setTimeout(done); + }); + it('debounce call only last action', (done) => { + const action = (status, method, owner, features) => { + expect(status).toExist(); + expect(status).toBe("replace"); + expect(method).toNotExist(); + expect(owner).toExist(); + expect(owner).toBe("queryform"); + expect(features).toExist(); + expect(features).toBe("geom2"); + done(); + }; + const Sink = compose(debounce("onChangeDrawingStatus", 800))(createSink( props => { + expect(props).toExist(); + expect(props.onChangeDrawingStatus).toExist(); + props.onChangeDrawingStatus("geom"); + props.onChangeDrawingStatus("geom1"); + props.onChangeDrawingStatus("replace", undefined, "queryform", "geom2"); + })); + ReactDOM.render((), document.getElementById("container")); + }); +}); diff --git a/web/client/components/misc/enhancers/debounce.js b/web/client/components/misc/enhancers/debounce.js new file mode 100644 index 0000000000..e27ab0ebbd --- /dev/null +++ b/web/client/components/misc/enhancers/debounce.js @@ -0,0 +1,27 @@ +/** +* Copyright 2018, GeoSolutions Sas. +* All rights reserved. +* +* This source code is licensed under the BSD-style license found in the +* LICENSE file in the root directory of this source tree. +*/ +const {withHandlers} = require('recompose'); +const {debounce} = require("lodash"); +const emptyFunc = () => {}; +/** + * This enhancer de-bounce a method passed as prop of the given time. + * The action should be present in the props passed to the component + * @memberof components.misc.enhancers + * @function + * @name debounce + * @example + * // example: every props change increment the *count* prop + * compose(debounce("onChangeDrawingStatus", 800)); + * the onChangeDrawingStatus action is debounced by 800 ms + */ +module.exports = (action = "", debounceTime = 1000) => withHandlers((initProp = {}) => { + const debounced = debounce(initProp[action] || emptyFunc, debounceTime); + return { + [action]: () => debounced + }; +}); diff --git a/web/client/plugins/QueryPanel.jsx b/web/client/plugins/QueryPanel.jsx index 016f0d992e..dfed92166a 100644 --- a/web/client/plugins/QueryPanel.jsx +++ b/web/client/plugins/QueryPanel.jsx @@ -21,6 +21,7 @@ const {zoomToExtent} = require('../actions/map'); const {toggleControl} = require('../actions/controls'); const {groupsSelector} = require('../selectors/layers'); +const {mapSelector} = require('../selectors/map'); const { crossLayerFilterSelector, availableCrossLayerFilterLayersSelector @@ -111,11 +112,11 @@ const SmartQueryForm = connect((state) => { showGeneratedFilter: false, allowEmptyFilter: true, emptyFilterWarning: true, - maxHeight: state.map && state.map.present && state.map.present.size && state.map.present.size.height + maxHeight: state.map && state.map.present && state.map.present.size && state.map.present.size.height, + zoom: (mapSelector(state) || {}).zoom }; }, dispatch => { return { - attributeFilterActions: bindActionCreators({ onAddGroupField: addGroupField, onAddFilterField: addFilterField, diff --git a/web/client/translations/data.de-DE b/web/client/translations/data.de-DE index 566a962891..f23c473b35 100644 --- a/web/client/translations/data.de-DE +++ b/web/client/translations/data.de-DE @@ -652,7 +652,7 @@ "reset_bbox": "Zurücksetzen", "save_bbox": "BBOX Änderungen speichern", "save_radius": "Radius/Kreiszentrum Änderungen speichern", - "radius": "Radius" + "radius": "Radius(m)" }, "methods": { "zone": "Zone", diff --git a/web/client/translations/data.en-US b/web/client/translations/data.en-US index 568b4a419b..6f9ecfefef 100644 --- a/web/client/translations/data.en-US +++ b/web/client/translations/data.en-US @@ -653,7 +653,7 @@ "reset_bbox": "Reset", "save_bbox": "Save BBOX modifications", "save_radius": "Save the radius/center modifications", - "radius": "Radius" + "radius": "Radius(m)" }, "methods": { "zone": "Zone", diff --git a/web/client/translations/data.es-ES b/web/client/translations/data.es-ES index 6c47e7339f..9851562c5f 100644 --- a/web/client/translations/data.es-ES +++ b/web/client/translations/data.es-ES @@ -652,7 +652,7 @@ "reset_bbox": "Reset", "save_bbox": "Guardar los cambios", "save_radius": "Guardar los cambios", - "radius": "Radio" + "radius": "Radio(m)" }, "methods": { "zone": "Zona", diff --git a/web/client/translations/data.fr-FR b/web/client/translations/data.fr-FR index d1107349b5..61f240336a 100644 --- a/web/client/translations/data.fr-FR +++ b/web/client/translations/data.fr-FR @@ -653,7 +653,7 @@ "reset_bbox": "Recommencer", "save_bbox": "Sauver les modifications", "save_radius": "Sauver les modifications", - "radius": "Rayon" + "radius": "Rayon(m)" }, "methods": { "zone": "Zone", diff --git a/web/client/translations/data.it-IT b/web/client/translations/data.it-IT index f03a9a82e2..61df49f9e5 100644 --- a/web/client/translations/data.it-IT +++ b/web/client/translations/data.it-IT @@ -652,7 +652,7 @@ "reset_bbox": "Reset", "save_bbox": "Salva le modifiche al BBOX", "save_radius": "Salva le modifiche al raggio e/o al centro", - "radius": "Raggio" + "radius": "Raggio(m)" }, "methods": { "zone": "Zona", diff --git a/web/client/translations/data.nl-NL b/web/client/translations/data.nl-NL index e3c0f8cf3c..6b28063827 100644 --- a/web/client/translations/data.nl-NL +++ b/web/client/translations/data.nl-NL @@ -559,7 +559,7 @@ "reset_bbox": "Herbegin", "save_bbox": "Wijzigingen opslaan", "save_radius": "Wijzigingen opslaan", - "radius": "Straal" + "radius": "Straal(m)" }, "methods": { "zone": "Zone", diff --git a/web/client/translations/data.zh-ZH b/web/client/translations/data.zh-ZH index 2150cd9fed..e2ef8d3355 100644 --- a/web/client/translations/data.zh-ZH +++ b/web/client/translations/data.zh-ZH @@ -562,7 +562,7 @@ "reset_bbox": "Reset", "save_bbox": "Save BBOX modifications", "save_radius": "Save the radius/center modifications", - "radius": "Radius" + "radius": "Radius(m)" }, "methods": { "zone": "Zone",