- Introduction
- Installation
- Usage
- Store methods
- Store Events
- Independent Methods
- Vue Components
- Examples
This library manages the global state of a series of selectors which have reactive dependencies between each other. These dependencies and the selectors' expected behaviour can be easily defined through a configuration file.
Add the dependency for your proyect into package.json
"magical-state": "git+https://gitlab.lbd.org.es/publico/magical-state.git"
To use the library we must create a store. For this we will first need to define a specification and an object consisted of two methods, getValues and defaultValuesGetter, that will retrieve the items that will populate each selector and specify a default value to be set.
To begin with, we must define in a json the selectors we want to create, the way they are related to each other, and their expected behaviour. The store will accept the next parameters in the specification:
name | type | default | description |
---|---|---|---|
id | string | undefined | Identifier of the selector, must be unique. |
label | string | undefined | Label to be placed in the label field of the selector. |
setItemsOnMounted | boolean | false | It can be specified if we want the selector to load data when it is rendered for the first time. |
triggerCallback | boolean | false | If this option is selected the callback defined passed in store creation will be fired whenever the value of the selector changes |
actions | array | [] | List of identifiers of store elements that need to reload their items every time the value of this element changes. |
Example of a specification.json
This spec will generate two selectors: "COUNTRIES" and "CITIES".
COUNTRIES will trigger the getValues method on mount and set as its selector options to the returned elements, setting as its selected value the element retrieved by defaultValuesGetter. Everytime the selected value changes CITIES will update its items and, again, defaultValuesGetter will be invoked.
[
{
"id": "COUNTRIES",
"label": "Countries",
"triggerCallback": true,
"actions": ["SPATIAL_FILTER"]
},
{
"id": "CITIES",
"label": "Cities",
"actions": []
}
]
Be careful with defining cyclic dependencies between selectors, and that the identifiers in the actions are defined in the specification.
We must define how we want to retrieve the necessary information to populate each of our selectors.
To do this the store will call the method getValues passed on store creation. This method is expected to receive the next parameters in the listed order:
- propId: The identifier of the selector that needs its items to be loaded.
- params: A list of key-value elements that contains the current value of the selectors that have dependent selectors (actions list not empty).
- store: A key-value object containing all the selectors' values.
Example of a getValues function
export default async (propId, params, store) => {
switch (propId) {
case "COUNTRIES":
return countriesService.getCountries();
case "CITIES":
return citiesService.getCities(params["COUNTRIES"]);
}
};
The function always must return a Promise, and the retrieved data accepts the next format:
[
{
"label": "any",
"value": "any"
}
]
Field label is only mandatory when using the library's Vue components.
We must define a method that indicates the store the default values to be set on selectors that are dependent of others (are present in any selectors' actions) or those that are meant to be populated on store creation.
To do this the store will call the method defaultValuesGetter passed on store creation. This method is expected to receive the next parameters in the listed order:
- propId: The identifier of the selector that needs its items to be loaded.
- items: A list containing the selector's items.
- store: A key-value object containing all the selectors' values.
The defaultValuesGetter function is expected to retrieve a Promise containing the desired new value.
Example of a defaultValuesGetter function
export default async (propId, items, store) => {
return new Promise((resolve) => {
switch (propId) {
// Will set as value the last item
case "TEMPROAL_FILTER":
resolve(items[items.length - 1].value);
break;
case "DATE_FILTER":
resolve(formatDateInverse(new Date()));
break;
default:
resolve();
break;
}
});
};
createStore (jsonSpec: Array, getValues: Function, state: Object|String, callback: Function) : Store
-
jsonSpec: the specification needed to create the desired selectors.
-
{getValues: ()=>{}, defaultValuesGetter: ()=>{} }: the fetch data and default values getter functions.
-
state: a state to be set when the store is created (url encoded or an object with key/values). When passed the store will ignore the specification for value getting/setting and will set the selectors' items based on the values passed.
-
callback: a Promise to be executed when the store values are updated. The callback function will be triggered everytime a selector that has the triggerCallback property to true suffers a value change. Note that if the selector has any related selectors on its actions, the store will wait for their items and default values to be set before triggering the callback.
-
Returns an instance of the store.
Example of store creation
import jsonSpec from "./specification.json";
import {
createStore
} from "./magical-state/index.js";
import getValues from "./valueGetters";
import defaultValuesGetter from "./defaultValuesGetter";
...
...
...
const state = {
COUNTRIES: 2,
};
this.store = createStore(
jsonSpec,
{
getValues: getValues,
defaultValuesGetter: defaultValuesGetter
},
// will set 2 as default value on selector 'countries'
state,
// the callback receives as a parameter an {id: value} object
// that constains the selectors with values not null
(storeCurrentState) => {
return new Promise(async (resolve) => {
resolve();
});
};
);
change(propId: String, newVal: Any, needsRedraw: Boolean): Promise<>
: Changes the value of the selector with the given id. needsRedraw prop defaults to true and specifies if the change on the selector's value should trigger the callback, in case the selector has the triggerCallback property set to true in the specification.setSelector(selectorId: String, newVal: Any, deep: Boolean, triggerCallbak: Boolean): Promise<>
: Changes the value of the selector with the given id. If deep is true, or the selector doesn't have items, setSelector will call getValues to populate the selector's items.setItems(id: String, values: Array): void
: Sets the items of the specified store element to the parameter values.setHasErrors(selectorId: String, value: Boolean, useSpecConfig: Boolean): Boolean
: Sets the property 'hasErrors' of the specified selector to the value provided. If it's set to true this will prevent the store from firing the callback when an element changes value.
- If a selector has an error (hasError property equaling true) and its value changes to a correct one, it's important to first set this property back to false so that the callback is triggered when calling change function.
getSelector(selectorId: String): Proxy
: Returns the proxy of the selector with the given id).(getter) objFromObservable(): Array<{id: String, value: Any}>
: Returns a copy of the actual state of the store.getUI(): Array<{id: String, label: String, value: Any, type: String, items: []}>
: Returns a list of objects that contains the basic information about every observable element so they can be displayed on the UI.triggerGetValues(id: String): Promise<Any>
: Triggers the getValues method of the selector with the given id. Returns the promise returned by the implementation of the getVlues method passed on store creation.
For importing and exporting the store, you can use the methods:
exportStoreEncodedURL(): String
: Returns an encoded URL with the current state of the store.importStoreEncodedURL(url): void
: Parses the URL and sets the state of the store).(independent from store) parseUrl(url,jsonSpec): Object<key,value>
: Returns the decoded URL, this method can be called without instantiating the store, to parse an URL before store creation.
Example of url import/export
import {
createStore, parseUrl
} from "./magical-state/index";
import getValues from "./getValues";
import defaultValuesGetter from "./defaultValuesGetter";
...
// If you need to access the readable values of an encoded
//URL, you can use the method:
const newState = parseUrl(url, jsonSpec);
// You can create the new store with an url encoded or
// with an object with key/values
const newStateUrl = "MD0yJjI9Mg=="
const newStateObject = {
"SPATIAL_AGGREGATION": "PROVINCE",
"SPATIAL_FILTER": "A CORUÑA",
"TEMPORAL_AGGREGATION": "YEAR",
"TEMPORAL_FILTER": "2018"
};
// Examples creating the store
this.store = createStore(
jsonSpec,
{getValues: getValues, defaultValuesGetter: defaultValuesGetter},
newStateUrl || newStateObject,
(storeContent) => {
console.log("created with new state");
}
);
// Then you can import/export the store
this.store.exportStoreEncodedURL()
// returns the encoded URL (MD0yJjI9Mg==)
// set the state of the store with the encoded URL
this.store.importStoreEncodedURL(newStateUrl);
The store will dispatch the next events:
-
redrawFulfilled: It is dispatched after completing the execution of the callback function.
-
itemsLoaded: It is dispatched after retrieving the items of a selector through the use of the getValues function.
-
change: Dispatched when a selector has changed its value.
Important: To dispatch these events we use CustomEvent so we can append on them iformation about the store. With each of them an object with the next content is passed:
{event:
{detail:
{
id: String,
label: String,
value: Any,
type: String,
items: Array
}
}
}
parseUrl(url,jsonSpec): Object<key,value>
Returns the decoded URL.
The library provides some vue2 components that work against a store. You cand find how to use them at: vue2-components.
Extended examples at: Examples.
In order to automate CHANGELOG update, developers should write commit messages following Angular Commit Message Conventions