diff --git a/web/client/actions/queryform.js b/web/client/actions/queryform.js index 74541a2992..98ae4d3d08 100644 --- a/web/client/actions/queryform.js +++ b/web/client/actions/queryform.js @@ -16,6 +16,8 @@ const CHANGE_CASCADING_VALUE = 'CHANGE_CASCADING_VALUE'; const EXPAND_ATTRIBUTE_PANEL = 'EXPAND_ATTRIBUTE_PANEL'; const EXPAND_SPATIAL_PANEL = 'EXPAND_SPATIAL_PANEL'; const SELECT_SPATIAL_METHOD = 'SELECT_SPATIAL_METHOD'; +const SELECT_VIEWPORT_SPATIAL_METHOD = 'SELECT_VIEWPORT_SPATIAL_METHOD'; +const UPDATE_GEOMETRY = 'UPDATE_GEOMETRY'; const SELECT_SPATIAL_OPERATION = 'SELECT_SPATIAL_OPERATION'; const CHANGE_SPATIAL_ATTRIBUTE = 'CHANGE_SPATIAL_ATTRIBUTE'; const REMOVE_SPATIAL_SELECT = 'REMOVE_SPATIAL_SELECT'; @@ -126,6 +128,18 @@ function selectSpatialMethod(method, fieldName) { }; } +function selectViewportSM() { + return { + type: SELECT_VIEWPORT_SPATIAL_METHOD + }; +} +function updateGeometrySpatialField(geometry) { + return { + type: UPDATE_GEOMETRY, + geometry + }; +} + function selectSpatialOperation(operation, fieldName) { return { type: SELECT_SPATIAL_OPERATION, @@ -322,6 +336,10 @@ module.exports = { ADD_SIMPLE_FILTER_FIELD, REMOVE_SIMPLE_FILTER_FIELD, REMOVE_ALL_SIMPLE_FILTER_FIELDS, + SELECT_VIEWPORT_SPATIAL_METHOD, + UPDATE_GEOMETRY, + updateGeometrySpatialField, + selectViewportSM, resetZones, zoneChange, // openMenu, diff --git a/web/client/actions/wfsquery.js b/web/client/actions/wfsquery.js index 0c537f4869..ab4eeb08e1 100644 --- a/web/client/actions/wfsquery.js +++ b/web/client/actions/wfsquery.js @@ -1,4 +1,4 @@ -/** +/* * Copyright 2016, GeoSolutions Sas. * All rights reserved. * @@ -103,12 +103,11 @@ function createQuery(searchUrl, filterObj) { }; } -function query(searchUrl, filterObj, retry) { +function query(searchUrl, filterObj) { return { type: QUERY, searchUrl, - filterObj, - retry + filterObj }; } @@ -147,28 +146,18 @@ function closeResponse() { } module.exports = { - FEATURE_TYPE_SELECTED, - FEATURE_TYPE_LOADED, + FEATURE_TYPE_SELECTED, featureTypeSelected, + FEATURE_TYPE_LOADED, featureTypeLoaded, + FEATURE_TYPE_ERROR, featureTypeError, + FEATURE_ERROR, featureError, + FEATURE_CLOSE, featureClose, + QUERY_CREATE, createQuery, + QUERY_RESULT, querySearchResponse, + QUERY_ERROR, queryError, + RESET_QUERY, resetQuery, + QUERY, query, FEATURE_LOADED, - FEATURE_TYPE_ERROR, - FEATURE_ERROR, - FEATURE_CLOSE, - QUERY_CREATE, - QUERY_RESULT, - QUERY_ERROR, - RESET_QUERY, - QUERY, - featureTypeSelected, - featureTypeLoaded, - featureTypeError, - featureError, loadFeature, - createQuery, - query, - featureClose, - resetQuery, toggleQueryPanel, - closeResponse, - queryError, - querySearchResponse + closeResponse }; diff --git a/web/client/components/data/query/QueryBuilder.jsx b/web/client/components/data/query/QueryBuilder.jsx index 33786527ee..48e982710c 100644 --- a/web/client/components/data/query/QueryBuilder.jsx +++ b/web/client/components/data/query/QueryBuilder.jsx @@ -96,6 +96,7 @@ const QueryBuilder = React.createClass({ onChangeDrawingStatus: () => {}, onRemoveSpatialSelection: () => {}, onShowSpatialSelectionDetails: () => {}, + onSelectViewportSM: () => {}, onEndDrawing: () => {}, onChangeDwithinValue: () => {} }, diff --git a/web/client/components/data/query/SpatialFilter.jsx b/web/client/components/data/query/SpatialFilter.jsx index e2e760e5ae..56f3a0d5fd 100644 --- a/web/client/components/data/query/SpatialFilter.jsx +++ b/web/client/components/data/query/SpatialFilter.jsx @@ -38,6 +38,7 @@ const SpatialFilter = React.createClass({ showDetailsPanel: false, withContainer: true, spatialMethodOptions: [ + {id: "Viewport", name: "queryform.spatialfilter.methods.viewport"}, {id: "BBOX", name: "queryform.spatialfilter.methods.box"}, {id: "Circle", name: "queryform.spatialfilter.methods.circle"}, {id: "Polygon", name: "queryform.spatialfilter.methods.poly"} @@ -57,6 +58,7 @@ const SpatialFilter = React.createClass({ onRemoveSpatialSelection: () => {}, onShowSpatialSelectionDetails: () => {}, onEndDrawing: () => {}, + onSelectViewportSM: () => {}, onChangeDwithinValue: () => {}, zoneFilter: () => {}, zoneSearch: () => {}, @@ -314,10 +316,18 @@ const SpatialFilter = React.createClass({ this.props.actions.onSelectSpatialMethod(method, name); - if (method !== "ZONE") { - this.changeDrawingStatus('start', method, "queryform", []); - } else { - this.changeDrawingStatus('clean', null, "queryform", []); + switch (method) { + case "ZONE": { + this.changeDrawingStatus('clean', null, "queryform", []); break; + } + case "Viewport": { + this.changeDrawingStatus('clean', null, "queryform", []); + this.props.actions.onSelectViewportSM(); + break; + } + default: { + this.changeDrawingStatus('start', method, "queryform", []); + } } }, updateSpatialOperation(id, name, value) { diff --git a/web/client/epics/wfsquery.js b/web/client/epics/wfsquery.js index 8c4beda6e5..832b17a4d7 100644 --- a/web/client/epics/wfsquery.js +++ b/web/client/epics/wfsquery.js @@ -8,7 +8,8 @@ const Rx = require('rxjs'); const axios = require('../libs/ajax'); -const {changeSpatialAttribute} = require('../actions/queryform'); +const {changeSpatialAttribute, SELECT_VIEWPORT_SPATIAL_METHOD, updateGeometrySpatialField} = require('../actions/queryform'); +const {CHANGE_MAP_VIEW} = require('../actions/map'); const {FEATURE_TYPE_SELECTED, QUERY, featureTypeLoaded, featureTypeError, createQuery, querySearchResponse, queryError, featureClose} = require('../actions/wfsquery'); const FilterUtils = require('../utils/FilterUtils'); const assign = require('object-assign'); @@ -227,6 +228,33 @@ const closeFeatureEpic = action$ => return action.control && action.control === 'drawer' ? Rx.Observable.of(featureClose()) : Rx.Observable.empty(); }); +function validateExtent(extent) { + if (extent[0] <= -180.0 || extent[2] >= 180.0) { + extent[0] = -180.0; + extent[2] = 180.0; + } + return extent; +} +const viewportSelectedEpic = (action$, store) => + action$.ofType(SELECT_VIEWPORT_SPATIAL_METHOD, CHANGE_MAP_VIEW) + .switchMap((action) => { + // calculate new geometry from map properties + const map = action.type === CHANGE_MAP_VIEW ? action : store.getState().map.present; + const bounds = Object.keys(map.bbox.bounds).reduce((p, c) => { + return assign({}, p, {[c]: parseFloat(map.bbox.bounds[c])}); + }, {}); + const extent = validateExtent([bounds.minx, bounds.miny, bounds.maxx, bounds.maxy]); + const center = [(extent[0] + extent[2]) / 2, (extent[1] + extent[3]) / 2]; + const start = [extent[0], extent[1]]; + const end = [extent[2], extent[3]]; + const coordinates = [[start, [start[0], end[1]], end, [end[0], start[1]], start]]; + let geometry = { + type: "Polygon", radius: 0, projection: "EPSG:4326", + extent, center, coordinates + }; + return Rx.Observable.of(updateGeometrySpatialField(geometry)); + }); + /** * Epics for WFS query requests * @name epics.wfsquery @@ -236,5 +264,6 @@ const closeFeatureEpic = action$ => module.exports = { featureTypeSelectedEpic, wfsQueryEpic, - closeFeatureEpic + closeFeatureEpic, + viewportSelectedEpic }; diff --git a/web/client/plugins/QueryPanel.jsx b/web/client/plugins/QueryPanel.jsx index 458739f38e..0df9d4f191 100644 --- a/web/client/plugins/QueryPanel.jsx +++ b/web/client/plugins/QueryPanel.jsx @@ -24,7 +24,7 @@ const LayersUtils = require('../utils/LayersUtils'); // include application component const QueryBuilder = require('../components/data/query/QueryBuilder'); -const {featureTypeSelectedEpic, wfsQueryEpic, closeFeatureEpic} = require('../epics/wfsquery'); +const {featureTypeSelectedEpic, wfsQueryEpic, closeFeatureEpic, viewportSelectedEpic} = require('../epics/wfsquery'); const {bindActionCreators} = require('redux'); const { @@ -40,6 +40,7 @@ const { expandAttributeFilterPanel, expandSpatialFilterPanel, selectSpatialMethod, + selectViewportSM, selectSpatialOperation, removeSpatialSelection, showSpatialSelectionDetails, @@ -95,6 +96,7 @@ const SmartQueryForm = connect((state) => { spatialFilterActions: bindActionCreators({ onExpandSpatialFilterPanel: expandSpatialFilterPanel, onSelectSpatialMethod: selectSpatialMethod, + onSelectViewportSM: selectViewportSM, onSelectSpatialOperation: selectSpatialOperation, onChangeDrawingStatus: changeDrawingStatus, onRemoveSpatialSelection: removeSpatialSelection, @@ -237,5 +239,5 @@ module.exports = { queryform: require('../reducers/queryform'), query: require('../reducers/query') }, - epics: {featureTypeSelectedEpic, wfsQueryEpic, closeFeatureEpic} + epics: {featureTypeSelectedEpic, wfsQueryEpic, closeFeatureEpic, viewportSelectedEpic} }; diff --git a/web/client/plugins/TOC.jsx b/web/client/plugins/TOC.jsx index f61d991c5f..a0044b60f3 100644 --- a/web/client/plugins/TOC.jsx +++ b/web/client/plugins/TOC.jsx @@ -42,6 +42,7 @@ const { expandAttributeFilterPanel, expandSpatialFilterPanel, selectSpatialMethod, + selectViewportSM, selectSpatialOperation, removeSpatialSelection, showSpatialSelectionDetails, @@ -105,6 +106,7 @@ const SmartQueryForm = connect((state) => { spatialFilterActions: bindActionCreators({ onExpandSpatialFilterPanel: expandSpatialFilterPanel, onSelectSpatialMethod: selectSpatialMethod, + onSelectViewportSM: selectViewportSM, onSelectSpatialOperation: selectSpatialOperation, onChangeDrawingStatus: changeDrawingStatus, onRemoveSpatialSelection: removeSpatialSelection, diff --git a/web/client/reducers/queryform.js b/web/client/reducers/queryform.js index 0ed268517c..0ca7caa072 100644 --- a/web/client/reducers/queryform.js +++ b/web/client/reducers/queryform.js @@ -27,6 +27,7 @@ const { CHANGE_DWITHIN_VALUE, ZONE_FILTER, ZONE_SEARCH, + UPDATE_GEOMETRY, // OPEN_MENU, ZONE_CHANGE, ZONES_RESET, @@ -165,6 +166,9 @@ function queryform(state = initialState, action) { case SELECT_SPATIAL_METHOD: { return assign({}, state, {spatialField: assign({}, state.spatialField, {[action.fieldName]: action.method, geometry: null})}); } + case UPDATE_GEOMETRY: { + return assign({}, state, {spatialField: assign({}, state.spatialField, {geometry: action.geometry})}, {toolbarEnabled: true}); + } case SELECT_SPATIAL_OPERATION: { return assign({}, state, {spatialField: assign({}, state.spatialField, {[action.fieldName]: action.operation})}); } diff --git a/web/client/translations/data.de-DE b/web/client/translations/data.de-DE index 01df880190..548ebd9242 100644 --- a/web/client/translations/data.de-DE +++ b/web/client/translations/data.de-DE @@ -453,6 +453,7 @@ }, "methods": { "zone": "Zone", + "viewport": "Viewport", "box": "BoundingBox", "buffer": "Buffer", "circle": "Kreis", diff --git a/web/client/translations/data.en-US b/web/client/translations/data.en-US index ce11eb37fb..bb2072e454 100644 --- a/web/client/translations/data.en-US +++ b/web/client/translations/data.en-US @@ -453,6 +453,7 @@ }, "methods": { "zone": "Zone", + "viewport": "Viewport", "box": "BoundingBox", "buffer": "Buffer", "circle": "Circle", diff --git a/web/client/translations/data.fr-FR b/web/client/translations/data.fr-FR index 9fb8b1e6c9..5cc7eaf4a3 100644 --- a/web/client/translations/data.fr-FR +++ b/web/client/translations/data.fr-FR @@ -455,6 +455,7 @@ }, "methods": { "zone": "Zone", + "viewport": "Viewport", "box": "Cadre d'objet", "buffer": "Zone tampon", "circle": "Cercle", diff --git a/web/client/translations/data.it-IT b/web/client/translations/data.it-IT index 31afee639c..4f097ebb01 100644 --- a/web/client/translations/data.it-IT +++ b/web/client/translations/data.it-IT @@ -453,6 +453,7 @@ }, "methods": { "zone": "Zona", + "viewport": "Viewport", "box": "BoundingBox", "buffer": "Buffer", "circle": "Cerchio",