forked from Ivasoft/mattermost-mobile
199 lines
5.6 KiB
JavaScript
199 lines
5.6 KiB
JavaScript
// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
|
|
// See LICENSE.txt for license information.
|
|
|
|
import React, {PureComponent} from 'react';
|
|
import PropTypes from 'prop-types';
|
|
import {ScrollView, View} from 'react-native';
|
|
import {Navigation} from 'react-native-navigation';
|
|
|
|
import {checkDialogElementForError, checkIfErrorsMatchElements} from 'mattermost-redux/utils/integration_utils';
|
|
|
|
import {changeOpacity, makeStyleSheetFromTheme} from 'app/utils/theme';
|
|
|
|
import StatusBar from 'app/components/status_bar';
|
|
import FormattedText from 'app/components/formatted_text';
|
|
|
|
import DialogElement from './dialog_element.js';
|
|
|
|
export default class InteractiveDialog extends PureComponent {
|
|
static propTypes = {
|
|
url: PropTypes.string.isRequired,
|
|
callbackId: PropTypes.string,
|
|
elements: PropTypes.arrayOf(PropTypes.object),
|
|
notifyOnCancel: PropTypes.bool,
|
|
state: PropTypes.string,
|
|
theme: PropTypes.object,
|
|
actions: PropTypes.shape({
|
|
submitInteractiveDialog: PropTypes.func.isRequired,
|
|
dismissModal: PropTypes.func.isRequired,
|
|
}).isRequired,
|
|
};
|
|
|
|
static defaultProps = {
|
|
url: '',
|
|
elements: [],
|
|
};
|
|
|
|
constructor(props) {
|
|
super(props);
|
|
|
|
const values = {};
|
|
if (props.elements != null) {
|
|
props.elements.forEach((e) => {
|
|
values[e.name] = e.default || null;
|
|
});
|
|
}
|
|
|
|
this.state = {
|
|
values,
|
|
errors: {},
|
|
submitting: false,
|
|
};
|
|
}
|
|
|
|
componentDidMount() {
|
|
this.navigationEventListener = Navigation.events().bindComponent(this);
|
|
}
|
|
|
|
navigationButtonPressed({buttonId}) {
|
|
switch (buttonId) {
|
|
case 'submit-dialog':
|
|
this.handleSubmit();
|
|
break;
|
|
case 'close-dialog':
|
|
this.notifyOnCancelIfNeeded();
|
|
this.handleHide();
|
|
break;
|
|
}
|
|
}
|
|
|
|
handleSubmit = async () => {
|
|
const {elements} = this.props;
|
|
const values = this.state.values;
|
|
const errors = {};
|
|
|
|
if (elements) {
|
|
elements.forEach((elem) => {
|
|
const error = checkDialogElementForError(elem, values[elem.name]);
|
|
if (error) {
|
|
errors[elem.name] = (
|
|
<FormattedText
|
|
id={error.id}
|
|
defaultMessage={error.defaultMessage}
|
|
values={error.values}
|
|
/>
|
|
);
|
|
}
|
|
});
|
|
}
|
|
|
|
this.setState({errors});
|
|
|
|
if (Object.keys(errors).length !== 0) {
|
|
return;
|
|
}
|
|
|
|
const {url, callbackId, state} = this.props;
|
|
|
|
const dialog = {
|
|
url,
|
|
callback_id: callbackId,
|
|
state,
|
|
submission: values,
|
|
};
|
|
|
|
const {data} = await this.props.actions.submitInteractiveDialog(dialog);
|
|
|
|
this.submitted = true;
|
|
|
|
if (!data || !data.errors || Object.keys(data.errors).length === 0) {
|
|
this.handleHide();
|
|
return;
|
|
}
|
|
|
|
if (checkIfErrorsMatchElements(data.errors, elements)) {
|
|
this.setState({errors: data.errors});
|
|
return;
|
|
}
|
|
|
|
this.handleHide();
|
|
}
|
|
|
|
notifyOnCancelIfNeeded = () => {
|
|
if (this.submitted) {
|
|
return;
|
|
}
|
|
|
|
const {url, callbackId, state, notifyOnCancel} = this.props;
|
|
|
|
if (!notifyOnCancel) {
|
|
return;
|
|
}
|
|
|
|
const dialog = {
|
|
url,
|
|
callback_id: callbackId,
|
|
state,
|
|
cancelled: true,
|
|
};
|
|
|
|
this.props.actions.submitInteractiveDialog(dialog);
|
|
}
|
|
|
|
handleHide = () => {
|
|
this.props.actions.dismissModal();
|
|
}
|
|
|
|
onChange = (name, value) => {
|
|
const values = {...this.state.values, [name]: value};
|
|
this.setState({values});
|
|
}
|
|
|
|
render() {
|
|
const {elements, theme} = this.props;
|
|
const style = getStyleFromTheme(theme);
|
|
|
|
return (
|
|
<View style={style.container}>
|
|
<ScrollView style={style.scrollView}>
|
|
<StatusBar/>
|
|
{elements && elements.map((e) => {
|
|
return (
|
|
<DialogElement
|
|
key={'dialogelement' + e.name}
|
|
displayName={e.display_name}
|
|
name={e.name}
|
|
type={e.type}
|
|
subtype={e.subtype}
|
|
helpText={e.help_text}
|
|
errorText={this.state.errors[e.name]}
|
|
placeholder={e.placeholder}
|
|
minLength={e.min_length}
|
|
maxLength={e.max_length}
|
|
dataSource={e.data_source}
|
|
optional={e.optional}
|
|
options={e.options}
|
|
value={this.state.values[e.name]}
|
|
onChange={this.onChange}
|
|
theme={theme}
|
|
/>
|
|
);
|
|
})}
|
|
</ScrollView>
|
|
</View>
|
|
);
|
|
}
|
|
}
|
|
|
|
const getStyleFromTheme = makeStyleSheetFromTheme((theme) => {
|
|
return {
|
|
container: {
|
|
backgroundColor: changeOpacity(theme.centerChannelColor, 0.03),
|
|
},
|
|
scrollView: {
|
|
marginBottom: 20,
|
|
marginTop: 10,
|
|
},
|
|
};
|
|
});
|