You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

297 lines
9.0 KiB
TypeScript

// Copyright (C) 2020 Intel Corporation
//
// SPDX-License-Identifier: MIT
import React from 'react';
import { RouteComponentProps } from 'react-router';
import { withRouter } from 'react-router-dom';
import { Row, Col } from 'antd/lib/grid';
import Alert from 'antd/lib/alert';
import Button from 'antd/lib/button';
import Collapse from 'antd/lib/collapse';
import notification from 'antd/lib/notification';
import Text from 'antd/lib/typography/Text';
import ConnectedFileManager from 'containers/file-manager/file-manager';
import LabelsEditor from 'components/labels-editor/labels-editor';
import { Files } from 'components/file-manager/file-manager';
import BasicConfigurationForm, { BaseConfiguration } from './basic-configuration-form';
import ProjectSearchField from './project-search-field';
import AdvancedConfigurationForm, { AdvancedConfiguration } from './advanced-configuration-form';
export interface CreateTaskData {
projectId: number | null;
basic: BaseConfiguration;
advanced: AdvancedConfiguration;
labels: any[];
files: Files;
}
interface Props {
onCreate: (data: CreateTaskData) => void;
status: string;
taskId: number | null;
projectId: number | null;
installedGit: boolean;
}
type State = CreateTaskData;
const defaultState = {
projectId: null,
basic: {
name: '',
},
advanced: {
lfs: false,
useZipChunks: true,
useCache: true,
},
labels: [],
files: {
local: [],
share: [],
remote: [],
},
};
class CreateTaskContent extends React.PureComponent<Props & RouteComponentProps, State> {
private basicConfigurationComponent: any;
private advancedConfigurationComponent: any;
private fileManagerContainer: any;
public constructor(props: Props & RouteComponentProps) {
super(props);
this.state = { ...defaultState };
}
public componentDidMount(): void {
const { projectId } = this.props;
if (projectId) {
this.handleProjectIdChange(projectId);
}
}
public componentDidUpdate(prevProps: Props): void {
const { status, history, taskId } = this.props;
if (status === 'CREATED' && prevProps.status !== 'CREATED') {
const btn = <Button onClick={() => history.push(`/tasks/${taskId}`)}>Open task</Button>;
notification.info({
message: 'The task has been created',
btn,
});
this.basicConfigurationComponent.resetFields();
if (this.advancedConfigurationComponent) {
this.advancedConfigurationComponent.resetFields();
}
this.fileManagerContainer.reset();
this.setState({
...defaultState,
});
}
}
private validateLabelsOrProject = (): boolean => {
const { projectId, labels } = this.state;
return !!labels.length || !!projectId;
};
private validateFiles = (): boolean => {
const files = this.fileManagerContainer.getFiles();
this.setState({
files,
});
const totalLen = Object.keys(files).reduce((acc, key) => acc + files[key].length, 0);
return !!totalLen;
};
private handleProjectIdChange = (value: null | number): void => {
this.setState({
projectId: value,
});
};
private handleSubmitBasicConfiguration = (values: BaseConfiguration): void => {
this.setState({
basic: { ...values },
});
};
private handleSubmitAdvancedConfiguration = (values: AdvancedConfiguration): void => {
this.setState({
advanced: { ...values },
});
};
private handleSubmitClick = (): void => {
if (!this.validateLabelsOrProject()) {
notification.error({
message: 'Could not create a task',
description: 'A task must contain at least one label or belong to some project',
});
return;
}
if (!this.validateFiles()) {
notification.error({
message: 'Could not create a task',
description: 'A task must contain at least one file',
});
return;
}
this.basicConfigurationComponent
.submit()
.then(() => {
if (this.advancedConfigurationComponent) {
return this.advancedConfigurationComponent.submit();
}
return new Promise((resolve): void => {
resolve();
});
})
.then((): void => {
const { onCreate } = this.props;
onCreate(this.state);
})
.catch((error: Error): void => {
notification.error({
message: 'Could not create a task',
description: error.toString(),
});
});
};
private renderBasicBlock(): JSX.Element {
return (
<Col span={24}>
<BasicConfigurationForm
wrappedComponentRef={(component: any): void => {
this.basicConfigurationComponent = component;
}}
onSubmit={this.handleSubmitBasicConfiguration}
/>
</Col>
);
}
private renderProjectBlock(): JSX.Element {
const { projectId } = this.state;
return (
<>
<Col span={24}>
<Text className='cvat-text-color'>Project:</Text>
</Col>
<Col span={24}>
<ProjectSearchField onSelect={this.handleProjectIdChange} value={projectId} />
</Col>
</>
);
}
private renderLabelsBlock(): JSX.Element {
const { projectId, labels } = this.state;
if (projectId) {
return (
<>
<Col span={24}>
<Text className='cvat-text-color'>Labels:</Text>
</Col>
<Col span={24}>
<Text type='secondary'>Project labels will be used</Text>
</Col>
</>
);
}
return (
<Col span={24}>
<Text type='danger'>* </Text>
<Text className='cvat-text-color'>Labels:</Text>
<LabelsEditor
labels={labels}
onSubmit={(newLabels): void => {
this.setState({
labels: newLabels,
});
}}
/>
</Col>
);
}
private renderFilesBlock(): JSX.Element {
return (
<Col span={24}>
<Text type='danger'>* </Text>
<Text className='cvat-text-color'>Select files:</Text>
<ConnectedFileManager
ref={(container: any): void => {
this.fileManagerContainer = container;
}}
withRemote
/>
</Col>
);
}
private renderAdvancedBlock(): JSX.Element {
const { installedGit } = this.props;
return (
<Col span={24}>
<Collapse>
<Collapse.Panel key='1' header={<Text className='cvat-title'>Advanced configuration</Text>}>
<AdvancedConfigurationForm
installedGit={installedGit}
wrappedComponentRef={(component: any): void => {
this.advancedConfigurationComponent = component;
}}
onSubmit={this.handleSubmitAdvancedConfiguration}
/>
</Collapse.Panel>
</Collapse>
</Col>
);
}
public render(): JSX.Element {
const { status } = this.props;
const loading = !!status && status !== 'CREATED' && status !== 'FAILED';
return (
<Row type='flex' justify='start' align='middle' className='cvat-create-task-content'>
<Col span={24}>
<Text className='cvat-title'>Basic configuration</Text>
</Col>
{this.renderBasicBlock()}
{this.renderProjectBlock()}
{this.renderLabelsBlock()}
{this.renderFilesBlock()}
{this.renderAdvancedBlock()}
<Col span={18}>{loading ? <Alert message={status} /> : null}</Col>
<Col span={6} className='cvat-create-task-submit-section'>
<Button loading={loading} disabled={loading} type='primary' onClick={this.handleSubmitClick}>
Submit
</Button>
</Col>
</Row>
);
}
}
export default withRouter(CreateTaskContent);