mirror of https://github.com/Siphonay/mastodon
Improve compose form performance, upgrade JS dependencies. LightingBox
now allows to cycle through multiple images
This commit is contained in:
parent
3e9d794ea5
commit
2c50687279
|
@ -12,6 +12,9 @@ import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
|
||||||
import Toggle from 'react-toggle';
|
import Toggle from 'react-toggle';
|
||||||
import Collapsable from '../../../components/collapsable';
|
import Collapsable from '../../../components/collapsable';
|
||||||
import UnlistedToggleContainer from '../containers/unlisted_toggle_container';
|
import UnlistedToggleContainer from '../containers/unlisted_toggle_container';
|
||||||
|
import SpoilerToggleContainer from '../containers/spoiler_toggle_container';
|
||||||
|
import PrivateToggleContainer from '../containers/private_toggle_container';
|
||||||
|
import SensitiveToggleContainer from '../containers/sensitive_toggle_container';
|
||||||
|
|
||||||
const messages = defineMessages({
|
const messages = defineMessages({
|
||||||
placeholder: { id: 'compose_form.placeholder', defaultMessage: 'What is on your mind?' },
|
placeholder: { id: 'compose_form.placeholder', defaultMessage: 'What is on your mind?' },
|
||||||
|
@ -26,17 +29,15 @@ const ComposeForm = React.createClass({
|
||||||
text: React.PropTypes.string.isRequired,
|
text: React.PropTypes.string.isRequired,
|
||||||
suggestion_token: React.PropTypes.string,
|
suggestion_token: React.PropTypes.string,
|
||||||
suggestions: ImmutablePropTypes.list,
|
suggestions: ImmutablePropTypes.list,
|
||||||
sensitive: React.PropTypes.bool,
|
|
||||||
spoiler: React.PropTypes.bool,
|
spoiler: React.PropTypes.bool,
|
||||||
spoiler_text: React.PropTypes.string,
|
|
||||||
unlisted: React.PropTypes.bool,
|
|
||||||
private: React.PropTypes.bool,
|
private: React.PropTypes.bool,
|
||||||
|
unlisted: React.PropTypes.bool,
|
||||||
|
spoiler_text: React.PropTypes.string,
|
||||||
fileDropDate: React.PropTypes.instanceOf(Date),
|
fileDropDate: React.PropTypes.instanceOf(Date),
|
||||||
focusDate: React.PropTypes.instanceOf(Date),
|
focusDate: React.PropTypes.instanceOf(Date),
|
||||||
preselectDate: React.PropTypes.instanceOf(Date),
|
preselectDate: React.PropTypes.instanceOf(Date),
|
||||||
is_submitting: React.PropTypes.bool,
|
is_submitting: React.PropTypes.bool,
|
||||||
is_uploading: React.PropTypes.bool,
|
is_uploading: React.PropTypes.bool,
|
||||||
media_count: React.PropTypes.number,
|
|
||||||
me: React.PropTypes.number,
|
me: React.PropTypes.number,
|
||||||
needsPrivacyWarning: React.PropTypes.bool,
|
needsPrivacyWarning: React.PropTypes.bool,
|
||||||
mentionedDomains: React.PropTypes.array.isRequired,
|
mentionedDomains: React.PropTypes.array.isRequired,
|
||||||
|
@ -45,10 +46,7 @@ const ComposeForm = React.createClass({
|
||||||
onClearSuggestions: React.PropTypes.func.isRequired,
|
onClearSuggestions: React.PropTypes.func.isRequired,
|
||||||
onFetchSuggestions: React.PropTypes.func.isRequired,
|
onFetchSuggestions: React.PropTypes.func.isRequired,
|
||||||
onSuggestionSelected: React.PropTypes.func.isRequired,
|
onSuggestionSelected: React.PropTypes.func.isRequired,
|
||||||
onChangeSensitivity: React.PropTypes.func.isRequired,
|
|
||||||
onChangeSpoilerness: React.PropTypes.func.isRequired,
|
|
||||||
onChangeSpoilerText: React.PropTypes.func.isRequired,
|
onChangeSpoilerText: React.PropTypes.func.isRequired,
|
||||||
onChangeVisibility: React.PropTypes.func.isRequired
|
|
||||||
},
|
},
|
||||||
|
|
||||||
mixins: [PureRenderMixin],
|
mixins: [PureRenderMixin],
|
||||||
|
@ -80,23 +78,10 @@ const ComposeForm = React.createClass({
|
||||||
this.props.onSuggestionSelected(tokenStart, token, value);
|
this.props.onSuggestionSelected(tokenStart, token, value);
|
||||||
},
|
},
|
||||||
|
|
||||||
handleChangeSensitivity (e) {
|
|
||||||
this.props.onChangeSensitivity(e.target.checked);
|
|
||||||
},
|
|
||||||
|
|
||||||
handleChangeSpoilerness (e) {
|
|
||||||
this.props.onChangeSpoilerness(e.target.checked);
|
|
||||||
this.props.onChangeSpoilerText('');
|
|
||||||
},
|
|
||||||
|
|
||||||
handleChangeSpoilerText (e) {
|
handleChangeSpoilerText (e) {
|
||||||
this.props.onChangeSpoilerText(e.target.value);
|
this.props.onChangeSpoilerText(e.target.value);
|
||||||
},
|
},
|
||||||
|
|
||||||
handleChangeVisibility (e) {
|
|
||||||
this.props.onChangeVisibility(e.target.checked);
|
|
||||||
},
|
|
||||||
|
|
||||||
componentDidUpdate (prevProps) {
|
componentDidUpdate (prevProps) {
|
||||||
if (this.props.focusDate !== prevProps.focusDate) {
|
if (this.props.focusDate !== prevProps.focusDate) {
|
||||||
// If replying to zero or one users, places the cursor at the end of the textbox.
|
// If replying to zero or one users, places the cursor at the end of the textbox.
|
||||||
|
@ -172,24 +157,10 @@ const ComposeForm = React.createClass({
|
||||||
<UploadButtonContainer style={{ paddingTop: '4px' }} />
|
<UploadButtonContainer style={{ paddingTop: '4px' }} />
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<label className='compose-form__label with-border' style={{ marginTop: '10px' }}>
|
<SpoilerToggleContainer />
|
||||||
<Toggle checked={this.props.spoiler} onChange={this.handleChangeSpoilerness} />
|
<PrivateToggleContainer />
|
||||||
<span className='compose-form__label__text'><FormattedMessage id='compose_form.spoiler' defaultMessage='Hide text behind warning' /></span>
|
|
||||||
</label>
|
|
||||||
|
|
||||||
<label className='compose-form__label with-border'>
|
|
||||||
<Toggle checked={this.props.private} onChange={this.handleChangeVisibility} />
|
|
||||||
<span className='compose-form__label__text'><FormattedMessage id='compose_form.private' defaultMessage='Mark as private' /></span>
|
|
||||||
</label>
|
|
||||||
|
|
||||||
<UnlistedToggleContainer />
|
<UnlistedToggleContainer />
|
||||||
|
<SensitiveToggleContainer />
|
||||||
<Collapsable isVisible={this.props.media_count > 0} fullHeight={39.5}>
|
|
||||||
<label className='compose-form__label'>
|
|
||||||
<Toggle checked={this.props.sensitive} onChange={this.handleChangeSensitivity} />
|
|
||||||
<span className='compose-form__label__text'><FormattedMessage id='compose_form.sensitive' defaultMessage='Mark media as sensitive' /></span>
|
|
||||||
</label>
|
|
||||||
</Collapsable>
|
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,27 @@
|
||||||
|
import PureRenderMixin from 'react-addons-pure-render-mixin';
|
||||||
|
import { FormattedMessage } from 'react-intl';
|
||||||
|
import Toggle from 'react-toggle';
|
||||||
|
|
||||||
|
const PrivateToggle = React.createClass({
|
||||||
|
|
||||||
|
propTypes: {
|
||||||
|
isPrivate: React.PropTypes.bool,
|
||||||
|
onChange: React.PropTypes.func.isRequired
|
||||||
|
},
|
||||||
|
|
||||||
|
mixins: [PureRenderMixin],
|
||||||
|
|
||||||
|
render () {
|
||||||
|
const { isPrivate, onChange } = this.props;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<label className='compose-form__label with-border'>
|
||||||
|
<Toggle checked={isPrivate} onChange={onChange} />
|
||||||
|
<span className='compose-form__label__text'><FormattedMessage id='compose_form.private' defaultMessage='Mark as private' /></span>
|
||||||
|
</label>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
});
|
||||||
|
|
||||||
|
export default PrivateToggle;
|
|
@ -0,0 +1,31 @@
|
||||||
|
import PureRenderMixin from 'react-addons-pure-render-mixin';
|
||||||
|
import { FormattedMessage } from 'react-intl';
|
||||||
|
import Toggle from 'react-toggle';
|
||||||
|
import Collapsable from '../../../components/collapsable';
|
||||||
|
|
||||||
|
const SensitiveToggle = React.createClass({
|
||||||
|
|
||||||
|
propTypes: {
|
||||||
|
hasMedia: React.PropTypes.bool,
|
||||||
|
isSensitive: React.PropTypes.bool,
|
||||||
|
onChange: React.PropTypes.func.isRequired
|
||||||
|
},
|
||||||
|
|
||||||
|
mixins: [PureRenderMixin],
|
||||||
|
|
||||||
|
render () {
|
||||||
|
const { hasMedia, isSensitive, onChange } = this.props;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Collapsable isVisible={hasMedia} fullHeight={39.5}>
|
||||||
|
<label className='compose-form__label'>
|
||||||
|
<Toggle checked={isSensitive} onChange={onChange} />
|
||||||
|
<span className='compose-form__label__text'><FormattedMessage id='compose_form.sensitive' defaultMessage='Mark media as sensitive' /></span>
|
||||||
|
</label>
|
||||||
|
</Collapsable>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
});
|
||||||
|
|
||||||
|
export default SensitiveToggle;
|
|
@ -0,0 +1,27 @@
|
||||||
|
import PureRenderMixin from 'react-addons-pure-render-mixin';
|
||||||
|
import { FormattedMessage } from 'react-intl';
|
||||||
|
import Toggle from 'react-toggle';
|
||||||
|
|
||||||
|
const SpoilerToggle = React.createClass({
|
||||||
|
|
||||||
|
propTypes: {
|
||||||
|
isSpoiler: React.PropTypes.bool,
|
||||||
|
onChange: React.PropTypes.func.isRequired
|
||||||
|
},
|
||||||
|
|
||||||
|
mixins: [PureRenderMixin],
|
||||||
|
|
||||||
|
render () {
|
||||||
|
const { isSpoiler, onChange } = this.props;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<label className='compose-form__label with-border' style={{ marginTop: '10px' }}>
|
||||||
|
<Toggle checked={isSpoiler} onChange={onChange} />
|
||||||
|
<span className='compose-form__label__text'><FormattedMessage id='compose_form.spoiler' defaultMessage='Hide text behind warning' /></span>
|
||||||
|
</label>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
});
|
||||||
|
|
||||||
|
export default SpoilerToggle;
|
|
@ -1,26 +1,29 @@
|
||||||
import { connect } from 'react-redux';
|
import { connect } from 'react-redux';
|
||||||
import ComposeForm from '../components/compose_form';
|
import ComposeForm from '../components/compose_form';
|
||||||
|
import { createSelector } from 'reselect';
|
||||||
import {
|
import {
|
||||||
changeCompose,
|
changeCompose,
|
||||||
submitCompose,
|
submitCompose,
|
||||||
clearComposeSuggestions,
|
clearComposeSuggestions,
|
||||||
fetchComposeSuggestions,
|
fetchComposeSuggestions,
|
||||||
selectComposeSuggestion,
|
selectComposeSuggestion,
|
||||||
changeComposeSensitivity,
|
|
||||||
changeComposeSpoilerness,
|
|
||||||
changeComposeSpoilerText,
|
changeComposeSpoilerText,
|
||||||
changeComposeVisibility,
|
|
||||||
changeComposeListability
|
|
||||||
} from '../../../actions/compose';
|
} from '../../../actions/compose';
|
||||||
|
|
||||||
|
const getMentionedUsernames = createSelector(state => state.getIn(['compose', 'text']), text => text.match(/(?:^|[^\/\w])@([a-z0-9_]+@[a-z0-9\.\-]+)/ig));
|
||||||
|
|
||||||
|
const getMentionedDomains = createSelector(getMentionedUsernames, mentionedUsernamesWithDomains => {
|
||||||
|
return mentionedUsernamesWithDomains !== null ? [...new Set(mentionedUsernamesWithDomains.map(item => item.split('@')[2]))] : [];
|
||||||
|
});
|
||||||
|
|
||||||
const mapStateToProps = (state, props) => {
|
const mapStateToProps = (state, props) => {
|
||||||
const mentionedUsernamesWithDomains = state.getIn(['compose', 'text']).match(/(?:^|[^\/\w])@([a-z0-9_]+@[a-z0-9\.\-]+)/ig);
|
const mentionedUsernames = getMentionedUsernames(state);
|
||||||
|
const mentionedUsernamesWithDomains = getMentionedDomains(state);
|
||||||
|
|
||||||
return {
|
return {
|
||||||
text: state.getIn(['compose', 'text']),
|
text: state.getIn(['compose', 'text']),
|
||||||
suggestion_token: state.getIn(['compose', 'suggestion_token']),
|
suggestion_token: state.getIn(['compose', 'suggestion_token']),
|
||||||
suggestions: state.getIn(['compose', 'suggestions']),
|
suggestions: state.getIn(['compose', 'suggestions']),
|
||||||
sensitive: state.getIn(['compose', 'sensitive']),
|
|
||||||
spoiler: state.getIn(['compose', 'spoiler']),
|
spoiler: state.getIn(['compose', 'spoiler']),
|
||||||
spoiler_text: state.getIn(['compose', 'spoiler_text']),
|
spoiler_text: state.getIn(['compose', 'spoiler_text']),
|
||||||
unlisted: state.getIn(['compose', 'unlisted'], ),
|
unlisted: state.getIn(['compose', 'unlisted'], ),
|
||||||
|
@ -30,10 +33,9 @@ const mapStateToProps = (state, props) => {
|
||||||
preselectDate: state.getIn(['compose', 'preselectDate']),
|
preselectDate: state.getIn(['compose', 'preselectDate']),
|
||||||
is_submitting: state.getIn(['compose', 'is_submitting']),
|
is_submitting: state.getIn(['compose', 'is_submitting']),
|
||||||
is_uploading: state.getIn(['compose', 'is_uploading']),
|
is_uploading: state.getIn(['compose', 'is_uploading']),
|
||||||
media_count: state.getIn(['compose', 'media_attachments']).size,
|
|
||||||
me: state.getIn(['compose', 'me']),
|
me: state.getIn(['compose', 'me']),
|
||||||
needsPrivacyWarning: state.getIn(['compose', 'private']) && mentionedUsernamesWithDomains !== null,
|
needsPrivacyWarning: state.getIn(['compose', 'private']) && mentionedUsernames !== null,
|
||||||
mentionedDomains: mentionedUsernamesWithDomains !== null ? [...new Set(mentionedUsernamesWithDomains.map(item => item.split('@')[2]))] : []
|
mentionedDomains: mentionedUsernamesWithDomains
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -59,22 +61,10 @@ const mapDispatchToProps = (dispatch) => ({
|
||||||
dispatch(selectComposeSuggestion(position, token, accountId));
|
dispatch(selectComposeSuggestion(position, token, accountId));
|
||||||
},
|
},
|
||||||
|
|
||||||
onChangeSensitivity (checked) {
|
|
||||||
dispatch(changeComposeSensitivity(checked));
|
|
||||||
},
|
|
||||||
|
|
||||||
onChangeSpoilerness (checked) {
|
|
||||||
dispatch(changeComposeSpoilerness(checked));
|
|
||||||
},
|
|
||||||
|
|
||||||
onChangeSpoilerText (checked) {
|
onChangeSpoilerText (checked) {
|
||||||
dispatch(changeComposeSpoilerText(checked));
|
dispatch(changeComposeSpoilerText(checked));
|
||||||
},
|
},
|
||||||
|
|
||||||
onChangeVisibility (checked) {
|
|
||||||
dispatch(changeComposeVisibility(checked));
|
|
||||||
},
|
|
||||||
|
|
||||||
});
|
});
|
||||||
|
|
||||||
export default connect(mapStateToProps, mapDispatchToProps)(ComposeForm);
|
export default connect(mapStateToProps, mapDispatchToProps)(ComposeForm);
|
||||||
|
|
|
@ -0,0 +1,17 @@
|
||||||
|
import { connect } from 'react-redux';
|
||||||
|
import PrivateToggle from '../components/private_toggle';
|
||||||
|
import { changeComposeVisibility } from '../../../actions/compose';
|
||||||
|
|
||||||
|
const mapStateToProps = state => ({
|
||||||
|
isPrivate: state.getIn(['compose', 'private'])
|
||||||
|
});
|
||||||
|
|
||||||
|
const mapDispatchToProps = dispatch => ({
|
||||||
|
|
||||||
|
onChange (e) {
|
||||||
|
dispatch(changeComposeVisibility(e.target.checked));
|
||||||
|
}
|
||||||
|
|
||||||
|
});
|
||||||
|
|
||||||
|
export default connect(mapStateToProps, mapDispatchToProps)(PrivateToggle);
|
|
@ -0,0 +1,18 @@
|
||||||
|
import { connect } from 'react-redux';
|
||||||
|
import SensitiveToggle from '../components/sensitive_toggle';
|
||||||
|
import { changeComposeSensitivity } from '../../../actions/compose';
|
||||||
|
|
||||||
|
const mapStateToProps = state => ({
|
||||||
|
hasMedia: state.getIn(['compose', 'media_attachments']).size > 0,
|
||||||
|
isSensitive: state.getIn(['compose', 'sensitive'])
|
||||||
|
});
|
||||||
|
|
||||||
|
const mapDispatchToProps = dispatch => ({
|
||||||
|
|
||||||
|
onChange (e) {
|
||||||
|
dispatch(changeComposeSensitivity(e.target.checked));
|
||||||
|
}
|
||||||
|
|
||||||
|
});
|
||||||
|
|
||||||
|
export default connect(mapStateToProps, mapDispatchToProps)(SensitiveToggle);
|
|
@ -0,0 +1,17 @@
|
||||||
|
import { connect } from 'react-redux';
|
||||||
|
import SpoilerToggle from '../components/spoiler_toggle';
|
||||||
|
import { changeComposeSpoilerness } from '../../../actions/compose';
|
||||||
|
|
||||||
|
const mapStateToProps = state => ({
|
||||||
|
isSpoiler: state.getIn(['compose', 'spoiler'])
|
||||||
|
});
|
||||||
|
|
||||||
|
const mapDispatchToProps = dispatch => ({
|
||||||
|
|
||||||
|
onChange (e) {
|
||||||
|
dispatch(changeComposeSpoilerness(e.target.checked));
|
||||||
|
}
|
||||||
|
|
||||||
|
});
|
||||||
|
|
||||||
|
export default connect(mapStateToProps, mapDispatchToProps)(SpoilerToggle);
|
|
@ -131,19 +131,14 @@ const Modal = React.createClass({
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
const url = media.get(index).get('url');
|
const url = media.get(index).get('url');
|
||||||
const hasLeft = index > 0;
|
|
||||||
const hasRight = index + 1 < media.size;
|
|
||||||
|
|
||||||
let leftNav, rightNav;
|
let leftNav, rightNav;
|
||||||
|
|
||||||
leftNav = rightNav = '';
|
leftNav = rightNav = '';
|
||||||
|
|
||||||
if (hasLeft) {
|
if (media.size > 1) {
|
||||||
leftNav = <div style={leftNavStyle} className='modal-container--nav' onClick={this.handlePrevClick}><i className='fa fa-fw fa-chevron-left' /></div>;
|
leftNav = <div style={leftNavStyle} className='modal-container--nav' onClick={this.handlePrevClick}><i className='fa fa-fw fa-chevron-left' /></div>;
|
||||||
}
|
|
||||||
|
|
||||||
if (hasRight) {
|
|
||||||
rightNav = <div style={rightNavStyle} className='modal-container--nav' onClick={this.handleNextClick}><i className='fa fa-fw fa-chevron-right' /></div>;
|
rightNav = <div style={rightNavStyle} className='modal-container--nav' onClick={this.handleNextClick}><i className='fa fa-fw fa-chevron-right' /></div>;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -116,7 +116,10 @@ export default function compose(state = initialState, action) {
|
||||||
case COMPOSE_SENSITIVITY_CHANGE:
|
case COMPOSE_SENSITIVITY_CHANGE:
|
||||||
return state.set('sensitive', action.checked);
|
return state.set('sensitive', action.checked);
|
||||||
case COMPOSE_SPOILERNESS_CHANGE:
|
case COMPOSE_SPOILERNESS_CHANGE:
|
||||||
return (action.checked ? state : state.set('spoiler_text', '')).set('spoiler', action.checked);
|
return state.withMutations(map => {
|
||||||
|
map.set('spoiler_text', '');
|
||||||
|
map.set('spoiler', action.checked);
|
||||||
|
});
|
||||||
case COMPOSE_SPOILER_TEXT_CHANGE:
|
case COMPOSE_SPOILER_TEXT_CHANGE:
|
||||||
return state.set('spoiler_text', action.text);
|
return state.set('spoiler_text', action.text);
|
||||||
case COMPOSE_VISIBILITY_CHANGE:
|
case COMPOSE_VISIBILITY_CHANGE:
|
||||||
|
|
|
@ -23,9 +23,9 @@ export default function modal(state = initialState, action) {
|
||||||
case MODAL_CLOSE:
|
case MODAL_CLOSE:
|
||||||
return state.set('open', false);
|
return state.set('open', false);
|
||||||
case MODAL_INDEX_DECREASE:
|
case MODAL_INDEX_DECREASE:
|
||||||
return state.update('index', index => Math.max(index - 1, 0));
|
return state.update('index', index => (index - 1) % state.get('media').size);
|
||||||
case MODAL_INDEX_INCREASE:
|
case MODAL_INDEX_INCREASE:
|
||||||
return state.update('index', index => Math.min(index + 1, state.get('media').size - 1));
|
return state.update('index', index => (index + 1) % state.get('media').size);
|
||||||
default:
|
default:
|
||||||
return state;
|
return state;
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
import { createSelector } from 'reselect'
|
import { createSelector } from 'reselect';
|
||||||
import Immutable from 'immutable';
|
import Immutable from 'immutable';
|
||||||
|
|
||||||
const getStatuses = state => state.get('statuses');
|
const getStatuses = state => state.get('statuses');
|
||||||
|
|
54
package.json
54
package.json
|
@ -6,51 +6,51 @@
|
||||||
"start": "babel-node ./streaming/index.js --presets es2015,stage-2"
|
"start": "babel-node ./streaming/index.js --presets es2015,stage-2"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@kadira/storybook": "^2.24.0",
|
"@kadira/storybook": "^2.35.3",
|
||||||
"axios": "^0.14.0",
|
"axios": "^0.15.3",
|
||||||
"babel-cli": "^6.22.2",
|
"babel-cli": "^6.23.0",
|
||||||
"babel-plugin-react-transform": "^2.0.2",
|
"babel-plugin-react-transform": "^2.0.2",
|
||||||
"babel-plugin-transform-decorators-legacy": "^1.3.4",
|
"babel-plugin-transform-decorators-legacy": "^1.3.4",
|
||||||
"babel-plugin-transform-object-rest-spread": "^6.8.0",
|
"babel-plugin-transform-object-rest-spread": "^6.23.0",
|
||||||
"babel-preset-es2015": "^6.22.0",
|
"babel-preset-es2015": "^6.22.0",
|
||||||
"babel-preset-react": "^6.11.1",
|
"babel-preset-react": "^6.11.1",
|
||||||
"babel-preset-stage-2": "^6.22.0",
|
"babel-preset-stage-2": "^6.22.0",
|
||||||
"babelify": "^7.3.0",
|
"babelify": "^7.3.0",
|
||||||
"browserify": "^13.1.0",
|
"browserify": "^14.1.0",
|
||||||
"browserify-incremental": "^3.1.1",
|
"browserify-incremental": "^3.1.1",
|
||||||
"bufferutil": "^2.0.0",
|
"bufferutil": "^2.0.1",
|
||||||
"chai": "^3.5.0",
|
"chai": "^3.5.0",
|
||||||
"chai-enzyme": "^0.5.2",
|
"chai-enzyme": "^0.6.1",
|
||||||
"css-loader": "^0.26.1",
|
"css-loader": "^0.26.2",
|
||||||
"dotenv": "^4.0.0",
|
"dotenv": "^4.0.0",
|
||||||
"emojione": "latest",
|
"emojione": "latest",
|
||||||
"enzyme": "^2.4.1",
|
"enzyme": "^2.7.1",
|
||||||
"es6-promise": "^3.2.1",
|
"es6-promise": "^3.2.1",
|
||||||
"eventsource": "^0.2.1",
|
"eventsource": "^0.2.1",
|
||||||
"express": "^4.14.1",
|
"express": "^4.14.1",
|
||||||
"http-link-header": "^0.5.0",
|
"http-link-header": "^0.5.0",
|
||||||
"immutable": "^3.8.1",
|
"immutable": "^3.8.1",
|
||||||
"intl": "^1.2.5",
|
"intl": "^1.2.5",
|
||||||
"jsdom": "^9.6.0",
|
"jsdom": "^9.11.0",
|
||||||
"mocha": "^3.1.1",
|
"mocha": "^3.2.0",
|
||||||
"node-sass": "^4.0.0",
|
"node-sass": "^4.5.0",
|
||||||
"npmlog": "^4.0.2",
|
"npmlog": "^4.0.2",
|
||||||
"pg": "^6.1.2",
|
"pg": "^6.1.2",
|
||||||
"react": "^15.3.2",
|
"react": "^15.4.2",
|
||||||
"react-addons-perf": "^15.3.2",
|
"react-addons-perf": "^15.4.2",
|
||||||
"react-addons-pure-render-mixin": "^15.3.1",
|
"react-addons-pure-render-mixin": "^15.4.2",
|
||||||
"react-addons-test-utils": "^15.3.2",
|
"react-addons-test-utils": "^15.4.2",
|
||||||
"react-autosuggest": "^7.0.1",
|
"react-autosuggest": "^7.0.1",
|
||||||
"react-decoration": "^1.4.0",
|
"react-decoration": "^1.4.0",
|
||||||
"react-dom": "^15.3.0",
|
"react-dom": "^15.4.2",
|
||||||
"react-imageloader": "^2.1.0",
|
"react-imageloader": "^2.1.0",
|
||||||
"react-immutable-proptypes": "^2.1.0",
|
"react-immutable-proptypes": "^2.1.0",
|
||||||
"react-intl": "^2.1.5",
|
"react-intl": "^2.1.5",
|
||||||
"react-motion": "^0.4.5",
|
"react-motion": "^0.4.5",
|
||||||
"react-notification": "^6.4.0",
|
"react-notification": "^6.6.0",
|
||||||
"react-proxy": "^1.1.8",
|
"react-proxy": "^1.1.8",
|
||||||
"react-redux": "^5.0.1",
|
"react-redux": "^5.0.3",
|
||||||
"react-redux-loading-bar": "^2.4.1",
|
"react-redux-loading-bar": "2.4.1",
|
||||||
"react-router": "^2.8.0",
|
"react-router": "^2.8.0",
|
||||||
"react-router-scroll": "^0.3.2",
|
"react-router-scroll": "^0.3.2",
|
||||||
"react-simple-dropdown": "^1.1.4",
|
"react-simple-dropdown": "^1.1.4",
|
||||||
|
@ -58,17 +58,17 @@
|
||||||
"react-toggle": "^2.1.1",
|
"react-toggle": "^2.1.1",
|
||||||
"redis": "^2.6.5",
|
"redis": "^2.6.5",
|
||||||
"redux": "^3.6.0",
|
"redux": "^3.6.0",
|
||||||
"redux-immutable": "^3.0.8",
|
"redux-immutable": "^3.1.0",
|
||||||
"redux-sounds": "^1.1.1",
|
"redux-sounds": "^1.1.1",
|
||||||
"redux-thunk": "^2.1.0",
|
"redux-thunk": "^2.2.0",
|
||||||
"reselect": "^2.5.4",
|
"reselect": "^2.5.4",
|
||||||
"sass-loader": "^4.0.2",
|
"sass-loader": "^6.0.2",
|
||||||
"sinon": "^1.17.6",
|
"sinon": "^1.17.6",
|
||||||
"style-loader": "^0.13.1",
|
"style-loader": "^0.13.2",
|
||||||
"utf-8-validate": "^3.0.0",
|
"utf-8-validate": "^3.0.1",
|
||||||
"uuid": "^3.0.1",
|
"uuid": "^3.0.1",
|
||||||
"webpack": "^1.14.0",
|
"webpack": "^2.2.1",
|
||||||
"websocket.js": "^0.1.7",
|
"websocket.js": "^0.1.7",
|
||||||
"ws": "^2.0.2"
|
"ws": "^2.1.0"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue