mirror of
https://github.com/Unleash/unleash.git
synced 2025-01-06 00:07:44 +01:00
semi working input state store
This commit is contained in:
parent
d12396cf57
commit
ab64d7eb1c
@ -1,27 +1,37 @@
|
|||||||
import React, { PropTypes } from 'react';
|
import React, { Component, PropTypes } from 'react';
|
||||||
import { Input, Switch, Button } from 'react-toolbox';
|
import { Input, Switch, Button } from 'react-toolbox';
|
||||||
import AddFeatureToggleStrategy from './strategies-for-toggle';
|
import SelectStrategies from './strategies-for-toggle';
|
||||||
import SelectedStrategies from './selected-strategies';
|
import SelectedStrategies from './selected-strategies';
|
||||||
|
|
||||||
const AddFeatureToggleComponent = ({
|
class AddFeatureToggleComponent extends Component {
|
||||||
strategies,
|
|
||||||
featureToggle,
|
componentWillMount () {
|
||||||
updateField,
|
// TODO unwind this stuff
|
||||||
|
if (this.props.initCallRequired === true) {
|
||||||
|
this.props.init(this.props.input);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
render () {
|
||||||
|
const {
|
||||||
|
input,
|
||||||
|
setValue,
|
||||||
addStrategy,
|
addStrategy,
|
||||||
removeStrategy,
|
removeStrategy,
|
||||||
onSubmit,
|
onSubmit,
|
||||||
onCancel,
|
onCancel,
|
||||||
editmode,
|
editmode = false,
|
||||||
}) => {
|
} = this.props;
|
||||||
|
|
||||||
const {
|
const {
|
||||||
name, // eslint-disable-line
|
name, // eslint-disable-line
|
||||||
description,
|
description,
|
||||||
enabled,
|
enabled,
|
||||||
} = featureToggle;
|
} = input;
|
||||||
const configuredStrategies = featureToggle.strategies;
|
const configuredStrategies = input.strategies || [];
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<form onSubmit={onSubmit}>
|
<form onSubmit={onSubmit(input)}>
|
||||||
<section>
|
<section>
|
||||||
<Input
|
<Input
|
||||||
type="text"
|
type="text"
|
||||||
@ -30,21 +40,20 @@ const AddFeatureToggleComponent = ({
|
|||||||
disabled={editmode}
|
disabled={editmode}
|
||||||
required
|
required
|
||||||
value={name}
|
value={name}
|
||||||
onChange={updateField.bind(this, 'name')} />
|
onChange={(v) => setValue('name', v)} />
|
||||||
<Input
|
<Input
|
||||||
type="text"
|
type="text"
|
||||||
multiline label="Description"
|
multiline label="Description"
|
||||||
required
|
required
|
||||||
value={description}
|
value={description}
|
||||||
onChange={updateField.bind(this, 'description')} />
|
onChange={(v) => setValue('description', v)} />
|
||||||
|
|
||||||
<br />
|
<br />
|
||||||
|
|
||||||
<Switch
|
<Switch
|
||||||
checked={enabled}
|
checked={enabled}
|
||||||
label="Enabled"
|
label="Enabled"
|
||||||
onChange={updateField.bind(this, 'enabled')} />
|
onChange={(v) => setValue('enabled', v)} />
|
||||||
|
|
||||||
<br />
|
<br />
|
||||||
</section>
|
</section>
|
||||||
|
|
||||||
@ -56,23 +65,24 @@ const AddFeatureToggleComponent = ({
|
|||||||
</section>
|
</section>
|
||||||
|
|
||||||
<section>
|
<section>
|
||||||
<AddFeatureToggleStrategy strategies={strategies} addStrategy={addStrategy} />
|
<SelectStrategies addStrategy={addStrategy} />
|
||||||
</section>
|
</section>
|
||||||
|
|
||||||
<br />
|
<br />
|
||||||
|
|
||||||
|
|
||||||
<Button type="submit" raised primary label={editmode ? 'Update' : 'Create'} />
|
<Button type="submit" raised primary label={editmode ? 'Update' : 'Create'} />
|
||||||
|
|
||||||
<Button type="cancel" raised label="Cancel" onClick={onCancel} />
|
<Button type="cancel" raised label="Cancel" onClick={onCancel} />
|
||||||
</form>
|
</form>
|
||||||
);
|
);
|
||||||
|
}
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
AddFeatureToggleComponent.propTypes = {
|
AddFeatureToggleComponent.propTypes = {
|
||||||
strategies: PropTypes.array.required,
|
strategies: PropTypes.array.required,
|
||||||
featureToggle: PropTypes.object,
|
input: PropTypes.object,
|
||||||
updateField: PropTypes.func.required,
|
setValue: PropTypes.func.required,
|
||||||
addStrategy: PropTypes.func.required,
|
addStrategy: PropTypes.func.required,
|
||||||
removeStrategy: PropTypes.func.required,
|
removeStrategy: PropTypes.func.required,
|
||||||
onSubmit: PropTypes.func.required,
|
onSubmit: PropTypes.func.required,
|
||||||
|
@ -1,80 +1,36 @@
|
|||||||
import React, { PropTypes } from 'react';
|
|
||||||
import { connect } from 'react-redux';
|
import { connect } from 'react-redux';
|
||||||
|
|
||||||
import { createFeatureToggles } from '../../store/feature-actions';
|
import { createFeatureToggles } from '../../store/feature-actions';
|
||||||
import { fetchStrategies } from '../../store/strategy-actions';
|
import AddComponent from './add-component';
|
||||||
import AddFeatureToggleUI from './add-component';
|
import { createMapper, createActions } from '../input-helpers';
|
||||||
|
|
||||||
class AddFeatureToggle extends React.Component {
|
const ID = 'add-feature-toggle';
|
||||||
constructor () {
|
const mapStateToProps = createMapper({ id: ID });
|
||||||
super();
|
const prepare = (methods, dispatch) => {
|
||||||
this.state = {
|
methods.onSubmit = (input) => (
|
||||||
name: '',
|
(e) => {
|
||||||
description: '',
|
e.preventDefault();
|
||||||
enabled: false,
|
// TODO: should add error handling
|
||||||
strategies: [],
|
createFeatureToggles(input)(dispatch)
|
||||||
};
|
.then(() => methods.clear())
|
||||||
|
.then(() => window.history.back());
|
||||||
}
|
}
|
||||||
|
|
||||||
static propTypes () {
|
|
||||||
return {
|
|
||||||
dispatch: PropTypes.func.isRequired,
|
|
||||||
strategies: PropTypes.array,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
static contextTypes = {
|
|
||||||
router: React.PropTypes.object,
|
|
||||||
}
|
|
||||||
|
|
||||||
onSubmit = (evt) => {
|
|
||||||
evt.preventDefault();
|
|
||||||
this.props.dispatch(createFeatureToggles(this.state));
|
|
||||||
this.context.router.push('/features');
|
|
||||||
};
|
|
||||||
|
|
||||||
onCancel = (evt) => {
|
|
||||||
evt.preventDefault();
|
|
||||||
this.context.router.push('/features');
|
|
||||||
};
|
|
||||||
|
|
||||||
updateField = (key, value) => {
|
|
||||||
const change = {};
|
|
||||||
change[key] = value;
|
|
||||||
this.setState(change);
|
|
||||||
};
|
|
||||||
|
|
||||||
addStrategy = (strategy) => {
|
|
||||||
const strategies = this.state.strategies;
|
|
||||||
strategies.push(strategy);
|
|
||||||
this.setState({ strategies });
|
|
||||||
}
|
|
||||||
|
|
||||||
removeStrategy = (strategy) => {
|
|
||||||
const strategies = this.state.strategies.filter(s => s !== strategy);
|
|
||||||
this.setState({ strategies });
|
|
||||||
}
|
|
||||||
|
|
||||||
componentDidMount () {
|
|
||||||
this.props.fetchStrategies();
|
|
||||||
}
|
|
||||||
|
|
||||||
render () {
|
|
||||||
return (
|
|
||||||
<AddFeatureToggleUI
|
|
||||||
strategies={this.props.strategies}
|
|
||||||
featureToggle={this.state}
|
|
||||||
updateField={this.updateField}
|
|
||||||
addStrategy={this.addStrategy}
|
|
||||||
removeStrategy={this.removeStrategy}
|
|
||||||
onSubmit={this.onSubmit}
|
|
||||||
onCancel={this.onCancel}
|
|
||||||
/>
|
|
||||||
);
|
);
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const mapStateToProps = (state) => ({
|
methods.onCancel = () => {
|
||||||
strategies: state.strategies.get('list').toArray(),
|
window.history.back();
|
||||||
});
|
};
|
||||||
|
|
||||||
export default connect(mapStateToProps, { fetchStrategies })(AddFeatureToggle);
|
methods.addStrategy = (v) => {
|
||||||
|
methods.pushToList('strategies', v);
|
||||||
|
};
|
||||||
|
|
||||||
|
methods.removeStrategy = (v) => {
|
||||||
|
methods.removeFromList('strategies', v);
|
||||||
|
};
|
||||||
|
|
||||||
|
return methods;
|
||||||
|
};
|
||||||
|
const actions = createActions({ id: ID, prepare });
|
||||||
|
|
||||||
|
export default connect(mapStateToProps, actions)(AddComponent);
|
||||||
|
@ -1,86 +1,62 @@
|
|||||||
import React, { PropTypes } from 'react';
|
|
||||||
import { connect } from 'react-redux';
|
import { connect } from 'react-redux';
|
||||||
import { editFeatureToggle } from '../../store/feature-actions';
|
|
||||||
import AddFeatureToggleComponent from './add-component';
|
|
||||||
import { fetchStrategies } from '../../store/strategy-actions';
|
|
||||||
|
|
||||||
|
import { requestUpdateFeatureToggle } from '../../store/feature-actions';
|
||||||
|
import AddComponent from './add-component';
|
||||||
|
import { createMapper, createActions } from '../input-helpers';
|
||||||
|
|
||||||
const mapStateToProps = (state, ownProps) => ({
|
const ID = 'edit-feature-toggle';
|
||||||
strategies: state.strategies.get('list').toArray(),
|
function getId (props) {
|
||||||
featureToggle: state.features.toJS().find(toggle => toggle.name === ownProps.featureToggleName) || {},
|
return [ID, props.featureToggleName];
|
||||||
|
}
|
||||||
|
// TODO: need to scope to the active featureToggle
|
||||||
|
// best is to emulate the "input-storage"?
|
||||||
|
const mapStateToProps = createMapper({
|
||||||
|
id: getId,
|
||||||
|
getDefault: (state, ownProps) => {
|
||||||
|
if (ownProps.featureToggleName) {
|
||||||
|
const match = state.features.findEntry((entry) => entry.get('name') === ownProps.featureToggleName);
|
||||||
|
|
||||||
|
if (match && match[1]) {
|
||||||
|
return match[1].toJS();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return {};
|
||||||
|
},
|
||||||
|
prepare: (props) => {
|
||||||
|
props.editmode = true;
|
||||||
|
return props;
|
||||||
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
class EditFeatureToggle extends React.Component {
|
const prepare = (methods, dispatch) => {
|
||||||
constructor (props) {
|
methods.onSubmit = (input) => (
|
||||||
super(props);
|
(e) => {
|
||||||
this.state = {
|
e.preventDefault();
|
||||||
name: props.featureToggle.name || '',
|
// TODO: should add error handling
|
||||||
description: props.featureToggle.description || '',
|
requestUpdateFeatureToggle(input)(dispatch)
|
||||||
enabled: props.featureToggle.enabled || false,
|
.then(() => methods.clear())
|
||||||
strategies: props.featureToggle.strategies || [],
|
.then(() => window.history.back());
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static propTypes () {
|
|
||||||
return {
|
|
||||||
dispatch: PropTypes.func.isRequired,
|
|
||||||
strategies: PropTypes.array,
|
|
||||||
featureToggle: PropTypes.featureToggle.isRequired,
|
|
||||||
fetchFeatureToggles: PropTypes.func.isRequired,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
componentDidMount () {
|
|
||||||
// todo fetch feature if missing? (reload of page does not fetch data from url)
|
|
||||||
this.props.fetchStrategies();
|
|
||||||
}
|
|
||||||
|
|
||||||
static contextTypes = {
|
|
||||||
router: React.PropTypes.object,
|
|
||||||
}
|
|
||||||
|
|
||||||
onSubmit = (evt) => {
|
|
||||||
evt.preventDefault();
|
|
||||||
this.props.dispatch(editFeatureToggle(this.state));
|
|
||||||
this.context.router.push('/features');
|
|
||||||
};
|
|
||||||
|
|
||||||
onCancel = (evt) => {
|
|
||||||
evt.preventDefault();
|
|
||||||
this.context.router.push('/features');
|
|
||||||
};
|
|
||||||
|
|
||||||
updateField = (key, value) => {
|
|
||||||
const change = {};
|
|
||||||
change[key] = value;
|
|
||||||
this.setState(change);
|
|
||||||
};
|
|
||||||
|
|
||||||
addStrategy = (strategy) => {
|
|
||||||
const strategies = this.state.strategies;
|
|
||||||
strategies.push(strategy);
|
|
||||||
this.setState({ strategies });
|
|
||||||
}
|
|
||||||
|
|
||||||
removeStrategy = (strategy) => {
|
|
||||||
const strategies = this.state.strategies.filter(s => s !== strategy);
|
|
||||||
this.setState({ strategies });
|
|
||||||
}
|
|
||||||
|
|
||||||
render () {
|
|
||||||
return (
|
|
||||||
<AddFeatureToggleComponent
|
|
||||||
editmode="true"
|
|
||||||
strategies={this.props.strategies}
|
|
||||||
featureToggle={this.state}
|
|
||||||
updateField={this.updateField}
|
|
||||||
addStrategy={this.addStrategy}
|
|
||||||
removeStrategy={this.removeStrategy}
|
|
||||||
onSubmit={this.onSubmit}
|
|
||||||
onCancel={this.onCancel}
|
|
||||||
/>
|
|
||||||
);
|
);
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
export default connect(mapStateToProps, { fetchStrategies })(EditFeatureToggle);
|
methods.onCancel = () => {
|
||||||
|
window.history.back();
|
||||||
|
};
|
||||||
|
|
||||||
|
methods.addStrategy = (v) => {
|
||||||
|
methods.pushToList('strategies', v);
|
||||||
|
};
|
||||||
|
|
||||||
|
methods.removeStrategy = (v) => {
|
||||||
|
methods.removeFromList('strategies', v);
|
||||||
|
};
|
||||||
|
|
||||||
|
return methods;
|
||||||
|
};
|
||||||
|
|
||||||
|
const actions = createActions({
|
||||||
|
id: getId,
|
||||||
|
prepare,
|
||||||
|
});
|
||||||
|
|
||||||
|
export default connect(mapStateToProps, actions)(AddComponent);
|
||||||
|
@ -0,0 +1,8 @@
|
|||||||
|
import { connect } from 'react-redux';
|
||||||
|
import SelectStrategies from './select-strategies';
|
||||||
|
import { fetchStrategies } from '../../store/strategy-actions';
|
||||||
|
|
||||||
|
|
||||||
|
export default connect((state) => ({
|
||||||
|
strategies: state.strategies.get('list').toArray(),
|
||||||
|
}), { fetchStrategies })(SelectStrategies);
|
@ -18,6 +18,17 @@ class SelectStrategies extends React.Component {
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
componentWillMount () {
|
||||||
|
this.props.fetchStrategies();
|
||||||
|
}
|
||||||
|
|
||||||
|
componentWillReceiveProps (nextProps) {
|
||||||
|
// this will fix async strategies list loading after mounted
|
||||||
|
if (!this.state.selectedStrategy && nextProps.strategies.length > 0) {
|
||||||
|
this.setState({ selectedStrategy: nextProps.strategies[0] });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
handleChange = (evt) => {
|
handleChange = (evt) => {
|
||||||
const strategyName = evt.target.value;
|
const strategyName = evt.target.value;
|
||||||
const selectedStrategy = this.props.strategies.find(s => s.name === strategyName);
|
const selectedStrategy = this.props.strategies.find(s => s.name === strategyName);
|
||||||
@ -63,7 +74,11 @@ class SelectStrategies extends React.Component {
|
|||||||
padding: '10px',
|
padding: '10px',
|
||||||
};
|
};
|
||||||
|
|
||||||
const selectedStrategy = this.state.selectedStrategy;
|
const selectedStrategy = this.state.selectedStrategy || this.props.strategies[0];
|
||||||
|
|
||||||
|
if (!selectedStrategy) {
|
||||||
|
return <div>Strategies loading...</div>;
|
||||||
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div style={style}>
|
<div style={style}>
|
||||||
|
@ -19,7 +19,7 @@ class SelectedStrategies extends React.Component {
|
|||||||
|
|
||||||
render () {
|
render () {
|
||||||
const removeStrategy = this.props.removeStrategy;
|
const removeStrategy = this.props.removeStrategy;
|
||||||
const strategies = this.props.configuredStrategies.map((s, index) => (
|
const configuredStrategies = this.props.configuredStrategies.map((s, index) => (
|
||||||
<Chip
|
<Chip
|
||||||
key={`${index}-${s.name}`}
|
key={`${index}-${s.name}`}
|
||||||
deletable
|
deletable
|
||||||
@ -31,7 +31,7 @@ class SelectedStrategies extends React.Component {
|
|||||||
));
|
));
|
||||||
return (
|
return (
|
||||||
<div>
|
<div>
|
||||||
{strategies.length > 0 ? strategies : <p>No activation strategies added</p>}
|
{configuredStrategies.length > 0 ? configuredStrategies : <p>No activation strategies added</p>}
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
import React, { PropTypes } from 'react';
|
import React, { PropTypes } from 'react';
|
||||||
import SelectStrategies from './select-strategies';
|
import SelectStrategies from './select-strategies-container';
|
||||||
|
|
||||||
class AddStrategiesToToggle extends React.Component {
|
class AddStrategiesToToggle extends React.Component {
|
||||||
constructor () {
|
constructor () {
|
||||||
@ -11,7 +11,6 @@ class AddStrategiesToToggle extends React.Component {
|
|||||||
|
|
||||||
static propTypes () {
|
static propTypes () {
|
||||||
return {
|
return {
|
||||||
strategies: PropTypes.array.isRequired,
|
|
||||||
addStrategy: PropTypes.func.isRequired,
|
addStrategy: PropTypes.func.isRequired,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
@ -39,12 +38,11 @@ class AddStrategiesToToggle extends React.Component {
|
|||||||
}
|
}
|
||||||
|
|
||||||
render () {
|
render () {
|
||||||
return this.state.showConfigure ?
|
return (
|
||||||
<SelectStrategies
|
this.state.showConfigure ?
|
||||||
strategies={this.props.strategies}
|
<SelectStrategies cancelConfig={this.cancelConfig} addStrategy={this.addStrategy} /> :
|
||||||
cancelConfig={this.cancelConfig}
|
this.renderAddLink()
|
||||||
addStrategy={this.addStrategy} /> :
|
);
|
||||||
this.renderAddLink();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,32 +1,63 @@
|
|||||||
import { createInc, createClear, createSet } from '../store/input-actions';
|
import {
|
||||||
|
createInc,
|
||||||
|
createClear,
|
||||||
|
createSet,
|
||||||
|
createPop,
|
||||||
|
createPush,
|
||||||
|
createInit,
|
||||||
|
} from '../store/input-actions';
|
||||||
|
|
||||||
export function createMapper (id, prepare = (v) => v) {
|
function getId (id, ownProps) {
|
||||||
return (state) => {
|
if (typeof id === 'function') {
|
||||||
|
return id(ownProps); // should return array...
|
||||||
|
}
|
||||||
|
return [id];
|
||||||
|
}
|
||||||
|
|
||||||
|
export function createMapper ({ id, getDefault, prepare = (v) => v }) {
|
||||||
|
return (state, ownProps) => {
|
||||||
let input;
|
let input;
|
||||||
if (state.input.has(id)) {
|
let initCallRequired = false;
|
||||||
input = state.input.get(id).toJS();
|
const scope = getId(id, ownProps);
|
||||||
|
if (state.input.hasIn(scope)) {
|
||||||
|
input = state.input.getIn(scope).toJS();
|
||||||
} else {
|
} else {
|
||||||
input = {};
|
initCallRequired = true;
|
||||||
|
input = getDefault ? getDefault(state, ownProps) : {};
|
||||||
}
|
}
|
||||||
|
|
||||||
return prepare({
|
return prepare({
|
||||||
|
initCallRequired,
|
||||||
input,
|
input,
|
||||||
}, state);
|
}, state, ownProps);
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
export function createActions (id, prepare = (v) => v) {
|
export function createActions ({ id, prepare = (v) => v }) {
|
||||||
return (dispatch) => (prepare({
|
return (dispatch, ownProps) => (prepare({
|
||||||
|
|
||||||
clear () {
|
clear () {
|
||||||
dispatch(createClear({ id }));
|
dispatch(createClear({ id: getId(id, ownProps) }));
|
||||||
|
},
|
||||||
|
|
||||||
|
init (value) {
|
||||||
|
dispatch(createInit({ id: getId(id, ownProps), value }));
|
||||||
},
|
},
|
||||||
|
|
||||||
setValue (key, value) {
|
setValue (key, value) {
|
||||||
dispatch(createSet({ id, key, value }));
|
dispatch(createSet({ id: getId(id, ownProps), key, value }));
|
||||||
|
},
|
||||||
|
|
||||||
|
pushToList (key, value) {
|
||||||
|
dispatch(createPush({ id: getId(id, ownProps), key, value }));
|
||||||
|
},
|
||||||
|
|
||||||
|
removeFromList (key, value) {
|
||||||
|
dispatch(createPop({ id: getId(id, ownProps), key, value }));
|
||||||
},
|
},
|
||||||
|
|
||||||
incValue (key) {
|
incValue (key) {
|
||||||
dispatch(createInc({ id, key }));
|
dispatch(createInc({ id: getId(id, ownProps), key }));
|
||||||
},
|
},
|
||||||
}, dispatch));
|
}, dispatch, ownProps));
|
||||||
}
|
}
|
||||||
|
@ -7,7 +7,7 @@ import AddStrategy, { PARAM_PREFIX } from './add-strategy';
|
|||||||
|
|
||||||
const ID = 'add-strategy';
|
const ID = 'add-strategy';
|
||||||
|
|
||||||
const actions = createActions(ID, (methods, dispatch) => {
|
const prepare = (methods, dispatch) => {
|
||||||
methods.onSubmit = (input) => (
|
methods.onSubmit = (input) => (
|
||||||
(e) => {
|
(e) => {
|
||||||
e.preventDefault();
|
e.preventDefault();
|
||||||
@ -35,6 +35,11 @@ const actions = createActions(ID, (methods, dispatch) => {
|
|||||||
|
|
||||||
|
|
||||||
return methods;
|
return methods;
|
||||||
|
};
|
||||||
|
|
||||||
|
const actions = createActions({
|
||||||
|
id: ID,
|
||||||
|
prepare,
|
||||||
});
|
});
|
||||||
|
|
||||||
export default connect(createMapper(ID), actions)(AddStrategy);
|
export default connect(createMapper({ id: ID }), actions)(AddStrategy);
|
||||||
|
@ -1,11 +1,17 @@
|
|||||||
export const actions = {
|
export const actions = {
|
||||||
SET_VALUE: 'SET_VALUE',
|
SET_VALUE: 'SET_VALUE',
|
||||||
INCREMENT_VALUE: 'INCREMENT_VALUE',
|
INCREMENT_VALUE: 'INCREMENT_VALUE',
|
||||||
|
LIST_PUSH: 'LIST_PUSH',
|
||||||
|
LIST_POP: 'LIST_POP',
|
||||||
CLEAR: 'CLEAR',
|
CLEAR: 'CLEAR',
|
||||||
|
INIT: 'INIT',
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export const createInit = ({ id, value }) => ({ type: actions.INIT, id, value });
|
||||||
export const createInc = ({ id, key }) => ({ type: actions.INCREMENT_VALUE, id, key });
|
export const createInc = ({ id, key }) => ({ type: actions.INCREMENT_VALUE, id, key });
|
||||||
export const createSet = ({ id, key, value }) => ({ type: actions.SET_VALUE, id, key, value });
|
export const createSet = ({ id, key, value }) => ({ type: actions.SET_VALUE, id, key, value });
|
||||||
|
export const createPush = ({ id, key, value }) => ({ type: actions.LIST_PUSH, id, key, value });
|
||||||
|
export const createPop = ({ id, key, value }) => ({ type: actions.LIST_POP, id, key, value });
|
||||||
export const createClear = ({ id }) => ({ type: actions.CLEAR, id });
|
export const createClear = ({ id }) => ({ type: actions.CLEAR, id });
|
||||||
|
|
||||||
export default actions;
|
export default actions;
|
||||||
|
@ -1,41 +1,68 @@
|
|||||||
import { Map as $Map } from 'immutable';
|
import { Map as $Map, List, fromJS } from 'immutable';
|
||||||
import actions from './input-actions';
|
import actions from './input-actions';
|
||||||
|
|
||||||
function getInitState () {
|
function getInitState () {
|
||||||
return new $Map();
|
return new $Map();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function init (state, { id, value }) {
|
||||||
|
state = assertId(state, id);
|
||||||
|
return state.setIn(id, fromJS(value));
|
||||||
|
}
|
||||||
|
|
||||||
function assertId (state, id) {
|
function assertId (state, id) {
|
||||||
if (!state.has(id)) {
|
if (!state.hasIn(id)) {
|
||||||
return state.set(id, new $Map({ inputId: id }));
|
return state.setIn(id, new $Map({ inputId: id }));
|
||||||
|
}
|
||||||
|
return state;
|
||||||
|
}
|
||||||
|
|
||||||
|
function assertList (state, id, key) {
|
||||||
|
if (!state.getIn(id).has(key)) {
|
||||||
|
return state.setIn(id.concat([key]), new List());
|
||||||
}
|
}
|
||||||
return state;
|
return state;
|
||||||
}
|
}
|
||||||
|
|
||||||
function setKeyValue (state, { id, key, value }) {
|
function setKeyValue (state, { id, key, value }) {
|
||||||
state = assertId(state, id);
|
state = assertId(state, id);
|
||||||
return state.setIn([id, key], value);
|
return state.setIn(id.concat([key]), value);
|
||||||
}
|
}
|
||||||
|
|
||||||
function increment (state, { id, key }) {
|
function increment (state, { id, key }) {
|
||||||
state = assertId(state, id);
|
state = assertId(state, id);
|
||||||
return state.updateIn([id, key], (value = 0) => value + 1);
|
return state.updateIn(id.concat([key]), (value = 0) => value + 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
function clear (state, { id }) {
|
function clear (state, { id }) {
|
||||||
if (state.has(id)) {
|
if (state.hasIn(id)) {
|
||||||
return state.remove(id);
|
return state.removeIn(id);
|
||||||
}
|
}
|
||||||
return state;
|
return state;
|
||||||
}
|
}
|
||||||
|
|
||||||
const inputState = (state = getInitState(), action) => {
|
function addToList (state, { id, key, value }) {
|
||||||
|
state = assertId(state, id);
|
||||||
|
state = assertList(state, id, key);
|
||||||
|
|
||||||
|
return state.updateIn(id.concat([key]), (list) => list.push(value));
|
||||||
|
}
|
||||||
|
|
||||||
|
function removeFromList (state, { id, key, value }) {
|
||||||
|
state = assertId(state, id);
|
||||||
|
state = assertList(state, id, key);
|
||||||
|
|
||||||
|
return state.updateIn(id.concat([key]), (list) => list.remove(list.indexOf(value)));
|
||||||
|
}
|
||||||
|
|
||||||
|
const inputState = (state = getInitState(), action) => {
|
||||||
if (!action.id) {
|
if (!action.id) {
|
||||||
return state;
|
return state;
|
||||||
}
|
}
|
||||||
|
|
||||||
switch (action.type) {
|
switch (action.type) {
|
||||||
|
case actions.INIT:
|
||||||
|
return init(state, action);
|
||||||
case actions.SET_VALUE:
|
case actions.SET_VALUE:
|
||||||
if (actions.key != null && actions.value != null) {
|
if (actions.key != null && actions.value != null) {
|
||||||
throw new Error('Missing required key / value');
|
throw new Error('Missing required key / value');
|
||||||
@ -43,6 +70,10 @@ const inputState = (state = getInitState(), action) => {
|
|||||||
return setKeyValue(state, action);
|
return setKeyValue(state, action);
|
||||||
case actions.INCREMENT_VALUE:
|
case actions.INCREMENT_VALUE:
|
||||||
return increment(state, action);
|
return increment(state, action);
|
||||||
|
case actions.LIST_PUSH:
|
||||||
|
return addToList(state, action);
|
||||||
|
case actions.LIST_POP:
|
||||||
|
return removeFromList(state, action);
|
||||||
case actions.CLEAR:
|
case actions.CLEAR:
|
||||||
return clear(state, action);
|
return clear(state, action);
|
||||||
default:
|
default:
|
||||||
|
Loading…
Reference in New Issue
Block a user