-
Notifications
You must be signed in to change notification settings - Fork 409
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
#10514 - FeatureEditor filter by geometric area #10515
base: master
Are you sure you want to change the base?
Conversation
fdacd45
to
dbb2fd3
Compare
Hi @Gaetanbrl thank you for contributing. We will review asap. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thank you for the PR. In this review I reported all the things that I found that may look unconsistent or problematic.
- see my comments inline
- The original proposal and issue was regarding to restrict editing, but in this implementation, also the view mode encounter this filter. Also this makes the things a little unclear. Maybe the filter should be applied only in edit mode?
- I found a bug that I didn't investigated, so I write about it here, instead of inline. With this tool active, I see an error of "max stack exceed" as well as I try to apply a quick filter.
https://github.com/user-attachments/assets/0193f17b-bfc2-41a2-a152-184e9482c445
I didn't debug so I don't know the exact reason. Anyway usingspatialField
can problematic, because too related to the advanced filter form . I'd suggest to use "filters" instead, as reported in another inline comment.
@@ -392,3 +394,38 @@ export const supportsFeatureEditing = (layer) => includes(supportedEditLayerType | |||
* @returns {boolean} flag | |||
*/ | |||
export const areLayerFeaturesEditable = (layer) => !layer?.disableFeaturesEditing && supportsFeatureEditing(layer); | |||
|
|||
export const isWKT = (wktString) => { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
These utils can not be mapping library specific. So feature grid can work with multiple mapping libraries.
For this reason mapstore provides utilities for WKT parsing.
Please use that utilities.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Please use that utilities.
Thanks. This change will be made.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
is this done?
export const requestRestrictedArea = (action$, store) => | ||
action$.ofType(OPEN_FEATURE_GRID, LOGIN_SUCCESS) | ||
.filter(() => { | ||
return !isAdminUserSelector(store.getState()) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This makes the restriction not be applied for anonymous users. Only for logged, not-admin.
It is a little unconsistent as a restriction.
I think we should at least apply the same restrictedArea to the anonymous users?
Maybe a configuration inside the restrictedAreas
that defines who has the restriction can be more generic.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Indeed, we have think avout authenticated users (to works with geOrchestra restricted org area) first and forget anonymous users.
To apply this restriction, we needs to confirm if restricted area is apply in edit mode only or not as explain in global review comment.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think we should at least apply the same restrictedArea to the anonymous users?
After discussion, we will also apply the restriction to anonymous users to be compliant with MapStore2 behavior.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
is this done?
isRestrictedByArea: ({ restrictedArea }) => { | ||
return (<Button | ||
id="fg-isRestrictedByArea-button" | ||
keyProp="fg-restrictedarea-button" | ||
className="square-button-md" | ||
bsStyle="warning" | ||
tooltipId="featuregrid.toolbar.restrictedByArea" | ||
disabled | ||
style={isEmpty(restrictedArea) ? { | ||
width: 0, | ||
padding: 0, | ||
borderWidth: 0 | ||
} : {}}> | ||
<Glyphicon glyph="1-point-dashed" /> | ||
</Button>); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Here the tooltipId
but the tooltip is not present in translation file.
Moreover the property do not exist for normal bootstrap button. In order to use it you should use TButton (created for feature grid) or Button with tooltip enhancer.
I noticed that bsStyle
property anyway is overridden by the TButton.
So in order to have a tooltip here you should:
- Apply the following changes.
isRestrictedByArea: ({ restrictedArea }) => { | |
return (<Button | |
id="fg-isRestrictedByArea-button" | |
keyProp="fg-restrictedarea-button" | |
className="square-button-md" | |
bsStyle="warning" | |
tooltipId="featuregrid.toolbar.restrictedByArea" | |
disabled | |
style={isEmpty(restrictedArea) ? { | |
width: 0, | |
padding: 0, | |
borderWidth: 0 | |
} : {}}> | |
<Glyphicon glyph="1-point-dashed" /> | |
</Button>); | |
isRestrictedByArea: ({ restrictedArea }) => { | |
return (<TButton | |
id="fg-isRestrictedByArea-button" | |
keyProp="fg-restrictedarea-button" | |
className="square-button-md" | |
bsStyle="warning" | |
visible={!isEmpty(restrictedArea)} | |
tooltipId="featuregrid.toolbar.restrictedByArea" | |
glyph="1-point-dashed" | |
/>); | |
}, |
- insert a localized string for each translation file (at least mandatory ones, as usual)
- Fix
TButton
like this (putting bsStyle after the forced one).
+++ b/web/client/components/data/featuregrid/toolbars/TButton.jsx
@@ -18,7 +18,7 @@ const hideStyle = {
const normalStyle = {};
const getStyle = (visible) => visible ? normalStyle : hideStyle;
export const SimpleTButton = forwardRef(({ disabled, id, visible, onClick, glyph, active, className = "square-button-md", ...props }, ref) => {
- return (<Button ref={ref} {...props} bsStyle={active ? "success" : "primary"} disabled={disabled} id={`fg-${id}`}
+ return (<Button ref={ref} bsStyle={active ? "success" : "primary"} {...props} disabled={disabled} id={`fg-${id}`}
style={getStyle(visible)}
className={className}
onClick={() => !disabled && onClick()}>
Note: I removed "disabled" flag for a better aspect.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thanks. This change will be made.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
is this suggestion applied?
* Create spatialField filters array. | ||
* Contains filters from viewportFilter, restrictedArea, exsting WFS filter | ||
*/ | ||
export const additionnalGridFilters = (state) => { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Using this approach makes the sync with map not working. While it still makes sense with viewport filter (because the map already shows the features in map), here this makes things a little unconsistent, and it could be practical for the user to see only the allowed rows on the map. What do you think?
A more functional approach is to add / remove the filter to the query
object, as well as the other filters, like quick filter, do.
The filters
section recently added to the filter may help you to manage it in a easier way.
Basically you can add to the filter a section filters: that can contain and use cql filter to apply some additional logic.
You can identify these filters by id, adding your own property, to manage them (e.g. adding/removing).
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
it could be practical for the user to see only the allowed rows on the map. What do you think?
Exactly, this is the expected behavior
Using this approach makes the sync with map not working.
Not really sur to understand. additionnalGridFilters methods is not dependent on the screen filter but but also uses it if activate. Maybe my code is wrong with I wanted to do.
Basically you can add to the filter a section filters: that can contain and use cql filter to apply some additional logic.
I will change the code to use filters section as you expect.
Thanks for this proposals @offtherailz .
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
is this done?
* @prop {string} cfg.restrictedArea.url Geometry definition as WKT or GeoJSON loaded from URL or path. | ||
* @prop {string} cfg.restrictedArea.raw Geometry definition as WKT or GeoJSON. | ||
* @prop {string} cfg.restrictedArea.operator Spatial operation to performed between features and the given geometry. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
* @prop {string} cfg.restrictedArea.url Geometry definition as WKT or GeoJSON loaded from URL or path. | |
* @prop {string} cfg.restrictedArea.raw Geometry definition as WKT or GeoJSON. | |
* @prop {string} cfg.restrictedArea.operator Spatial operation to performed between features and the given geometry. | |
* @prop {object} cfg.restrictedArea object containing settings for restricted area. If present, it restricts the editing area to the given geometry. It requires at least `url` or `raw` to be defined. | |
* @prop {string} cfg.restrictedArea.raw Geometry definition as WKT or GeoJSON. This attribute allows to define the geometry directly in the configuration. | |
* @prop {string} cfg.restrictedArea.url Geometry definition as WKT or GeoJSON loaded from URL or path. If present, this wins over the raw geometry configuration. By default, the filter will use `CONTAINS` if not defined. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
is this done?
@offtherailz Thank you for this detailed review. |
restricted area documentation Lint and clean
dbb2fd3
to
3925cf0
Compare
The source branch is now rebase / align with master (8e2d443) |
After read the filters doc it's not easy to identify what filter I need to use and how to use it correctly. This doc say about spatialField (considered as problematic) :
I guess this is a state changes to apply with a specific filter object. But what should I replace the spatialFilter filter with ? logic or mapstore-query-panel or cql ? This doc is a good practice overview but a I can enhance this doc if i correctly understand how to use it with a use case. |
After some discussions, we will make the expected changes (see comments).
About this point, the natif request is to limit view by an area in edit mode and read mode. The natif use case is to limit view and edition according to geOrchestra restricted area (this area is applied to the user's organization via geOrchestra console app). So, we propose to add a configuration to manage 2 options :
Needs investigation. I've seen this error before with CQL filters and geoServer limits (cross layer filter limit ?). |
@offtherailz Thank you so much for previous feedback. But I need another one to continue. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Hi @Gaetanbrl I resumed the review recently and i think you are still working on it right?
if so consider this as a partial review, since changes still have to be submitted yet
all of the following must have unit tests covering changes
- selectors
- reducers
- actions
- epics
please checkout also inline comments
also if you can go through the various conversation and add a comment to say which can be resolved or not.
Thanks
@@ -202,36 +202,87 @@ export const isEditingAllowedSelector = (state) => { | |||
})(state); | |||
return (canEdit || isAllowed) && !isCesium(state); | |||
}; | |||
|
|||
export const restrictedAreaSrcSelector = state => get(state, "featuregrid.restrictedArea"); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
maybe you can reuse it in
restrictedAreaOperatorSelector
restrictedAreaSelector
describeSelector, | ||
isFilterByViewportSupported, | ||
(viewportFilterIsActive, box, projection, spatialField = [], describeLayer, viewportFilterIsSupported) => { | ||
const attribute = findGeometryProperty(describeLayer)?.name; | ||
const existingFilter = spatialField?.operation ? [spatialField] : spatialField; | ||
return viewportFilterIsActive && viewportFilterIsSupported ? { | ||
spatialField: [ | ||
...existingFilter, | ||
// avoid restricted area filter dupplication |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
// avoid restricted area filter dupplication | |
// avoid restricted area filter duplication |
viewportFilter, | ||
projectionSelector, | ||
describeSelector, | ||
state => restrictedAreaOperatorSelector(state), |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
state => restrictedAreaOperatorSelector(state), | |
restrictedAreaOperatorSelector, |
@@ -175,7 +183,9 @@ const EditorPlugin = connect( | |||
virtualScroll: this.props.virtualScroll ?? true, | |||
editingAllowedRoles: this.props.editingAllowedRoles, | |||
editingAllowedGroups: this.props.editingAllowedGroups, | |||
maxStoredPages: this.props.maxStoredPages | |||
maxStoredPages: this.props.maxStoredPages, | |||
restrictedAreaUrl: this.props.restrictedAreaUrl, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
this is not present in config, only restrictedArea with inside three params:
- raw
- operator
- url
@@ -156,7 +157,9 @@ function featuregrid(state = emptyResultsState, action) { | |||
editingAllowedRoles: action.options.editingAllowedRoles || state.editingAllowedRoles || ["ADMIN"], | |||
editingAllowedGroups: action.options.editingAllowedGroups || state.editingAllowedGroups || [], | |||
virtualScroll: !!action.options.virtualScroll, | |||
maxStoredPages: action.options.maxStoredPages || 5 | |||
maxStoredPages: action.options.maxStoredPages || 5, | |||
restrictedAreaUrl: action.options.restrictedAreaUrl || "", |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
this is not needed as url is already inside restrictedArea
* Create spatialField filters array. | ||
* Contains filters from viewportFilter, restrictedArea, exsting WFS filter | ||
*/ | ||
export const additionnalGridFilters = (state) => { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
is this done?
@@ -392,3 +394,38 @@ export const supportsFeatureEditing = (layer) => includes(supportedEditLayerType | |||
* @returns {boolean} flag | |||
*/ | |||
export const areLayerFeaturesEditable = (layer) => !layer?.disableFeaturesEditing && supportsFeatureEditing(layer); | |||
|
|||
export const isWKT = (wktString) => { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
is this done?
export const requestRestrictedArea = (action$, store) => | ||
action$.ofType(OPEN_FEATURE_GRID, LOGIN_SUCCESS) | ||
.filter(() => { | ||
return !isAdminUserSelector(store.getState()) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
is this done?
isRestrictedByArea: ({ restrictedArea }) => { | ||
return (<Button | ||
id="fg-isRestrictedByArea-button" | ||
keyProp="fg-restrictedarea-button" | ||
className="square-button-md" | ||
bsStyle="warning" | ||
tooltipId="featuregrid.toolbar.restrictedByArea" | ||
disabled | ||
style={isEmpty(restrictedArea) ? { | ||
width: 0, | ||
padding: 0, | ||
borderWidth: 0 | ||
} : {}}> | ||
<Glyphicon glyph="1-point-dashed" /> | ||
</Button>); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
is this suggestion applied?
* @prop {string} cfg.restrictedArea.url Geometry definition as WKT or GeoJSON loaded from URL or path. | ||
* @prop {string} cfg.restrictedArea.raw Geometry definition as WKT or GeoJSON. | ||
* @prop {string} cfg.restrictedArea.operator Spatial operation to performed between features and the given geometry. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
is this done?
Description
This improvement will change the featureEditor plugin to allow to limit attributes table features loading by a geometric area (WKT or GeoJSON) for non ADMIN users.
This improvement don't use GeoFence and use a simple geometry.
Please check if the PR fulfills these requirements
What kind of change does this PR introduce? (check one with "x", remove the others)
Issue
What is the current behavior?
User can filter attribute table by custom filter or viewport.
#10514
What is the new behavior?
If configured, attribute table keep natives filters and use an additional geometric filter that can be create from URL (e.g geOrchestra restricted area) or from RAW (GEOJSON or WKT).
Breaking change
Does this PR introduce a breaking change? (check one with "x", remove the other)
Other useful information