diff --git a/vignettes/metadata.Rmd b/vignettes/metadata.Rmd index 577757af..d32333d8 100644 --- a/vignettes/metadata.Rmd +++ b/vignettes/metadata.Rmd @@ -2,7 +2,7 @@ title: "Metadata driven UI" output: rmarkdown::html_vignette vignette: > - %\VignetteIndexEntry{metadata} + %\VignetteIndexEntry{Metadata driven UI} %\VignetteEngine{knitr::rmarkdown} %\VignetteEncoding{UTF-8} --- @@ -16,38 +16,50 @@ knitr::opts_chunk$set( ## Motivation -Historically HINT had different components for its plots each with its own logic on how to handle different filter selections. For example: +We wanted to create a unified way to add filters and controls for all our plots in the front end. Previously there were different approaches to this for each plot but we wanted to create a unified way to reduce code duplication, make the interface consistent and mean enhancements to one plot can easily be ported over to all others. The drop downs in the side bar are split into two sections, plot controls and filters. + +1. `controls` - dropdowns that influence other dropdowns +1. `filters` - dropdowns that act as filters for slicing the data + +The `controls` can run `effects` to modify the filters such as: + +* Set specific filters as multi-selects. For example, in the barchart users can select an `X Axis` and `Disaggregate by` controls which set whatever is selected into a multi-select dropdown, while keeping the rest as single selects. See barchart screenshot +* Set selected value for a filter e.g. if users select an indicator we might want to select specific sex and age groups which have data for this indicator e.g. ANC indicators should be female and 15-49. +* Set filters e.g. ART data has a column for period, ANC data has a column for year, we want to show the period for filter for ART only and the year filter for ANC only +* Filters can be hidden if we want to simplify the UI to remove clutter for users whilst ensuring we still filter the underlying data +* Set custom effects that the plot impl can pick up on and use to control the plot UI e.g. in the table we have a custom effect which defines for each control what is displayed as a header and as a row in the table view. + +Below are some examples of the plot control and filter UIs that the metadata allow us to build ### Choropleth -![Choropleth state 1](images/choropleth-1.png) + +* Area as a multiselect + +![](images/choropleth-1.png){width="100%"} ### Barchart -![Barchart state 1](images/barchart-1.png) -![Barchart state 2](images/barchart-2.png) -### Table -![Table state 1](images/table-1.png) -![Table state 2](images/table-2.png) +* Indicator control sets filter values +* `X Axis` & `Disaggregate by` set the selected filters as multiselects + +![](images/barchart-1.png){width="100%"} +![](images/barchart-2.png){width="100%"} -There are lots of differences between these filters but the main relevant differences are: +### Table -* Some filters are single selects and some are multi-selects -* The barchart's `X Axis` and `Disaggregate by` filters turn whatever is selected into a multi-select dropdown while keeping the rest single selects -* The table's `Presets` dropdown changes the _filters_ that are shown while also changing whether they are multi-selects or single selects and also changing the value of the filters +* Presets set specific filters as multiselects and what filter to use as the header group and row +* Indicator sets specific filter values to relevant ones for that indicator -The aim is to unify these differences into one framework that can support all these dependencies between dropdowns while also allowing for easy additions and modifications. +![](images/table-1.png){width="100%"} +![](images/table-2.png){width="100%"} ## Implementation overview The backend (this repository) provides metadata JSON endpoints that tell the frontend how to construct the UI and also how to dynamically update it based on the value of the dropdowns. -Note: For each tab in the frontend UI there is a different endpoint and associated JSON schema that specifies the metadata: +For details on where metadata is currently served from see [Endpoints](#endpoints) section. -* [`Review output`](../inst/schema/CalibrateMetadataResponse.schema.json) -* [`Calibrate model`](../inst/schema/CalibratePlotMetadata.schema.json) -* [`Review inputs`](../inst/schema/ReviewInputFilterMetadataResponse.schema.json) - -The structures of these are the same bar the keys to the `plotSettingsControl` which identifies the plot. In the bullet points above there are two main types of dropdowns that we want: +Reminder that there are two types of dropdowns that can be created via metadata: 1. Dropdowns that influence other dropdowns - we will call these `controls` and call their values `settings` 1. Dropdowns that acts as filters for slicing the data - we will call these `filters` and their values `selections` @@ -81,6 +93,7 @@ Here is a useful visualisation of the schema that I'll keep referring to so that │ │ │ │ │ ├── setMultiple │ │ │ │ │ ├── setFilterValues │ │ │ │ │ ├── setHidden +│ │ │ │ │ ├── customPlotEffect ``` This is just a full version of the json schema. @@ -113,12 +126,13 @@ This is just a full version of the json schema. │ │ │ │ │ ├── setMultiple │ │ │ │ │ ├── setFilterValues │ │ │ │ │ ├── setHidden +│ │ │ │ │ ├── customPlotEffect ``` The `filterTypes` key is the simplest. It is an array of all the filters that can be displayed in the UI with their options. Explanations for each number in the visualisation: 1. The `id` is just an id we can assign to the filter to refer to it later 1. The `column_id` this is the column id of the data which we have to filter -1. `use_shape_regions` is a boolean that should largely be ignored, it is a one off trick to reduce the amount of payload we send to the frontend by telling it to derive the options of a filter select with this boolean from the shape file that the user has uploaded +1. `use_shape_regions` is a boolean that should largely be ignored, it is a one off trick for geographical area filters only to reduce the amount of payload we send to the frontend by telling it to derive the options of a filter select with this boolean from the shape file that the user has uploaded 1. The `options` are the options visible to the user in the dropdown. Each option has an `id` by which filter selections can refer to the option, a `label` to display in the dropdown menu and an optional `description` prop for contextual information (this doesn't get displayed to the user at all) ## Plot Settings Control @@ -150,6 +164,7 @@ The `filterTypes` key is the simplest. It is an array of all the filters that ca │ │ │ │ │ ├── setMultiple │ │ │ │ │ ├── setFilterValues │ │ │ │ │ ├── setHidden +│ │ │ │ │ ├── customPlotEffect ``` The plot settings controls is an object with keys of the relevant plot name (e.g. `choropleth`) and value with the structure shown in the visualisation. Explanation of keys: @@ -190,6 +205,7 @@ The plot settings controls is an object with keys of the relevant plot name (e.g │ │ │ │ │ ├── setMultiple │ │ │ │ │ ├── setFilterValues │ │ │ │ │ ├── setHidden +│ │ │ │ │ ├── customPlotEffect ``` The `settings` completely determine the `filters` and their `selections`. The next section will go through each effect and how the frontend resolves it but for now, explanation of keys: @@ -200,6 +216,8 @@ The `settings` completely determine the `filters` and their `selections`. The ne ## Effects +Effects are the actions that can be undertaken when a plot control is selected. They can be used to modify the filters and set custom effects which can be used by the plot. + ```php ├── filterTypes[] │ ├── id # a @@ -227,9 +245,10 @@ The `settings` completely determine the `filters` and their `selections`. The ne │ │ │ │ │ ├── setMultiple │ │ │ │ │ ├── setFilterValues │ │ │ │ │ ├── setHidden +│ │ │ │ │ ├── customPlotEffect ``` -For this section we will need a more detailed view of the effects structure +For this section we will need a more detailed view of the effects structure: ```php ├── setFilters[] # 1 @@ -239,17 +258,28 @@ For this section we will need a more detailed view of the effects structure ├── setMultiple[] # 2 ├── setFilterValues{} # 3 ├── setHidden[] # 4 +├── customPlotEffect{} # 5 ``` Explanation of each effect and how the frontend aggregates this: 1. `setFilters` is how the effect specifies what `filters` the user sees from the `filterTypes`. Only one `setFilters` effect is used, they are not combined. So if two `settings` set filters, the last `setFilters` in the for loop would be used when going through all the `settings`. Keys: - 1. `filterId` is the `id` key in a `filterTypes`element (`a` in the visualisation) + 1. `filterId` is the `id` key in a `filterTypes`element (`a` in the schema above) 1. `label` is the title of the `filter` displayed to the user in the UI 1. `stateFilterId` is the id the state uses to identify this filter. This is distinct to the `filterId` because in the bubble plot we have two indicators filters that are the same filter type (so have the same `options` and the same `column_id`) however they are distinct because one controls the color of the bubbles and one controls the size of the bubbles so while both their `filterId`s are `indicator`, the `stateFilterId`s for both are different, `colorIndicator` and `sizeIndicator`. For most cases this can be set as the same thing as the `filterId`. All other effects will now identify filters by `stateFilterId` as these uniquely correspond to what the user sees -1. `setMultiple` is an effect that converts a `filter` to a multi-select dropdown (they are all single select by default). It is an array of `stateFilterId`s (`1.3` in the visualisation) This is aggregated via concatenation in the frontend so if two settings have `setMultiple` properties `["state_id_1", "state_id_2"]` and `["state_id_3"]` respectively then all three filters with those state ids will be set to multi-selects -1. `setFilterValues` is an effect that sets the selection of the filters in `setFilters`. It is an object with keys equal to `stateFilterId` and value an array of ids of the `selections` (`b` in the visualisation). Note this is still an array of length one if you want to specify a value for a single select filter. This is aggregated by the [spread operator](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Operators/Spread_syntax#spread_in_object_literals) in the frontend +1. `setMultiple` is an effect that converts a `filter` to a multi-select dropdown (they are all single select by default). It is an array of `stateFilterId`s (`1.3` in the schema) This is aggregated via concatenation in the frontend so if two settings have `setMultiple` properties `["state_id_1", "state_id_2"]` and `["state_id_3"]` respectively then all three filters with those state ids will be set to multi-selects +1. `setFilterValues` is an effect that sets the selection of the filters in `setFilters`. It is an object with keys equal to `stateFilterId` and value an array of ids of the `selections` (`b` in the schema). Note this is still an array of length one if you want to specify a value for a single select filter. This is aggregated by the [spread operator](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Operators/Spread_syntax#spread_in_object_literals) in the frontend 1. `setHidden` is an effect that hides `filters` and is usually used when we have a fixed filter that we don't want the user to change but is still required to slice the data. This is aggregated via concatenation in the frontend +1. `customPlotEffect` is an effect that plots can use to build reactive UI from for a specific plot type. At the moment we are only using this in the table, so set the row and header groups. But it could be expanded to general metadata which we want a plot control to update, but does not have an effect on the filters themselves. + +### stateFilterId vs filterId + +To hammer it home for Rob because he can't remember this no matter how many times he reads it. `FilterRef`s have a `filterId` and ` stateFilterId` + +* `filterId` - used to identify the `id` in the full set of `filters` i.e. to get the label and options for this filter +* `stateFilterId` - reference to this filter within the rest of the metadata and the effects i.e. to identify which drop down to update when a plot control changes + +This is because on the bubble plot, there are two drop downs which have `indicator` as their options, but run different effects. ## Typescript definition and derived frontend store @@ -286,7 +316,8 @@ type PlotSettingEffect = { setFilters?: FilterRef[], setMultiple?: StateFilterId[], setFilterValues?: Record, - setHidden?: StateFilterId[] + setHidden?: StateFilterId[], + customPlotEffect?: any } type PlotSettingOption = { @@ -357,3 +388,73 @@ type FrontendDataState = { [plotName: PlotName]: any[] } ``` + +## Endpoints {#endpoints} + +For each data source in the frontend UI there is a different endpoint and associated JSON schema that specifies the metadata. Note this will probably go stale but should give a starting point + +| Step | Plot | Data source | Metadata source | Schema | +|------|------|-------------|-----------------|--------| +| Input review | Time series | `/chart-data/input-time-series/` | `/review-input/metadata` | [Review inputs](../inst/schema/ReviewInputFilterMetadataResponse.schema.json) | +| Input review | Choropleth | `/validate/survey-and-programme` |`/review-input/metadata` | [Review inputs](../inst/schema/ReviewInputFilterMetadataResponse.schema.json) | +| Input review | Comparison table | `/chart-data/input-comparison` | `/chart-data/input-comparison` | [Input comparison](../inst/schema/InputComparisonResponse.schema.json) | +| Input review | Comparison barchart |`/chart-data/input-comparison` | `/chart-data/input-comparison` | [Input comparison](../inst/schema/InputComparisonResponse.schema.json) | +| Input review | Population | `/validate/baseline-individual` only for population data |`/chart-data/input-population` | [Input population](../inst/schema/InputPopulationMetadataResponse.schema.json) | +| Calibrate | Calibration barchart | `/calibrate/plot/` | `/calibrate/plot/` | [Calibrate plot](../inst/schema/CalibratePlotResponse.schema)| +| Output review | Choropleth | `/calibrate/result/path/` (*)| `/calibrate/result/metadata/` | [Review output](../inst/schema/CalibrateMetadataResponse.schema.json)| +| Output review | Barchart | `/calibrate/result/path/` (*)| `/calibrate/result/metadata/` | [Review output](../inst/schema/CalibrateMetadataResponse.schema.json)| +| Output review | Choropleth | `/calibrate/result/path/` (*)| `/calibrate/result/metadata/` | [Review output](../inst/schema/CalibrateMetadataResponse.schema.json)| +| Output review | Table | `/calibrate/result/path/` (*)| `/calibrate/result/metadata/` | [Review output](../inst/schema/CalibrateMetadataResponse.schema.json)| +| Output review | Comparison | `/comparison/plot/` | `/comparison/plot/` | [Output comparison](../inst/schema/CalibrateMetadataResponse.schema.json)| +| Output review | Bubble | `/calibrate/result/path/` (*)| `/calibrate/result/metadata/` | [Review output](../inst/schema/ComparisonPlotResponse.schema)| +| Output review | Cascade | `/calibrate/result/path/` (*)| `/calibrate/result/metadata/` | [Review output](../inst/schema/ComparisonPlotResponse.schema)| + +(*) Kotlin backend gets the path to the duckdb output file using hintr endpoint at `/calibrate/result/path/`. It then slices this using SQL based on the filters the user set in the front end. So we only return the slice of data the user is looking at. + +Note that data and metadata can be in the same endpoint or different ones. Favour returning them from the same endpoint unless splitting them up will significantly speed up the UI e.g. returning metadata for output plots all together saves reading the output and building the filters from this multiple times. In general, I think having the data & metadata returned together is simpler. + +## Front end implementation + +This section gives an overview of how the front end handles the metadata and where any custom logic for a specific plot should be injected. The aim of the front end impl is that the wiring up of state will be as straight forward as possible, and we get as much code use as possible. + +### Overview + +#### Data fetching + +**For output plots** data is fetched on demand when users set filters, therefore there is no big blob of data in state +**For all other plots** data is pre-fetched via a store action. This might be when an input file is uploaded (e.g. population, survey, ART, ANC), or when a plot is opened e.g. input comparison, input time series. Data is fetched via an action and saved in the store state somewhere, this varies from data source to data source but will be in `baseline`, `surveyAndProgramme`, `reviewInput` + +#### Metadata fetching + +Metadata is fetched via a store action which calls to endpoints in the table above. It can be fetched as part of the same endpoint as the data, or separately. When metadata is fetched separately we typically do a couple of modifications before saving it into the store. Firstly, if the data has `use_shape_regions` or `use_shape_area_level` set we fill in these filter options from those already in state. We also commit the initial selections, this runs any default effects and saves them resulting controls and filters into `plotSelections` state, it also runs `getPlotData` which manges running the plot-specific filtering logic and saves this into the `plotData` state. + +#### On mount + +The plot should have a prop which is of type `PlotName`. It should then read the metadata via the `getMetadataFromPlotName` helper and use the data from `plotData` state. For each specific plot, it should then modify the `plotData` into a structure which can be used by plot component. Potentially using metadata if it needs, including any `customPlotEffect`s for the selection. + +#### On plot control/filter update + +This dispatches an action `plotSelections/updateSelections` which runs any effects to get the new state, it then runs `getPlotData` updating the `plotData` in state with the filtered data. + +### Adding new plot + +If you are using an existing data source or metadata source, you will need to + +1. Add metadata into existing endpoint or new endpoint +1. If a new endpoint, add an action which fetched the metadata, runs `filtersAfterUseShapeRegions` and `commitPlotDefaultSelections`. +1. Add the new plot type into `PlotName` +1. Add handling for new plot type into `getMetadataFromPlotName`, it should just map to the location of the metadata +1. Add handling for new plot type into `getPlotData`, if this is using a new data source, write a function to do the filtering for this data source, and run any specific logic for this data source type. +1. Write the component pulling data from `plotData` state and metadata via `getMetadataFromPlotName`. + +### Where do I write custom logic? + +**Modify the metadata before it is ever used**, write it in the action after the data is fetched and before it saved to state. For exampe, setting options for a filter to those from the shape file. + +**Customise the filter logic**, write it in one of the function `getPlotData` calls to. For example can use this to switch between different data sources in one plot (time series), derive some data from selections (population data gets aggregated to selected level) + +**Modify selections when a plot control changes**, hook into the filtering logic, you can do this in `handlePlotControlOverrides`. We have used this to set x-axis for the comparison barchart. Plot selections cannot by default update other plot selections, but we wanted this for the comparison barchart. We wanted the UI to show a "plot type" control but not let users modify the x-axis because they could easily break the display. Instead we have a hidden x-axis and then hook into the filtering logic to set the x-axis control if the "plot control" is changed. This one I think we should be careful with, if we spot common overrides we should think about refactoring this into an effect. + +**Modify the plot itself based on a plot control**, use a `customPlotEffect` to set metadata for this specific plot selection e.g. for the table we specify in `customPlotEffect` what category in the data should be a column group or shown in the rows. + +