casibase/web/src/WordsetEditPage.js

232 lines
7.9 KiB
JavaScript

import React from "react";
import {Button, Card, Col, Input, InputNumber, Row, Select} from "antd";
import * as WordsetBackend from "./backend/WordsetBackend";
import * as Setting from "./Setting";
import i18next from "i18next";
import VectorTable from "./VectorTable";
import WordsetGraph from "./WordsetGraph";
import * as VectorsetBackend from "./backend/VectorsetBackend";
const {Option} = Select;
class WordsetEditPage extends React.Component {
constructor(props) {
super(props);
this.state = {
classes: props,
wordsetName: props.match.params.wordsetName,
wordset: null,
vectorsets: null,
matchLoading: false,
};
}
UNSAFE_componentWillMount() {
this.getWordset();
this.getVectorsets();
}
getWordset() {
WordsetBackend.getWordset(this.props.account.name, this.state.wordsetName)
.then((wordset) => {
if (wordset.status === "ok") {
this.setState({
wordset: wordset.data,
});
} else {
Setting.showMessage("error", `Failed to get wordset: ${wordset.msg}`);
}
});
}
getVectorsets() {
VectorsetBackend.getVectorsets(this.props.account.name)
.then((res) => {
if (res.status === "ok") {
this.setState({
vectorsets: res.data,
});
} else {
Setting.showMessage("error", `Failed to get vectorsets: ${res.msg}`);
}
});
}
parseWordsetField(key, value) {
if (["score"].includes(key)) {
value = Setting.myParseInt(value);
}
return value;
}
updateWordsetField(key, value) {
value = this.parseWordsetField(key, value);
const wordset = this.state.wordset;
wordset[key] = value;
this.setState({
wordset: wordset,
});
}
renderWordset() {
const allWords = this.state.wordset?.vectors.length;
const validWords = this.state.wordset?.vectors.filter(vector => vector.data.length !== 0).length;
return (
<Card size="small" title={
<div>
{i18next.t("wordset:Edit Wordset")}&nbsp;&nbsp;&nbsp;&nbsp;
<Button type="primary" onClick={this.submitWordsetEdit.bind(this)}>{i18next.t("general:Save")}</Button>
</div>
} style={{marginLeft: "5px"}} type="inner">
<Row style={{marginTop: "10px"}} >
<Col style={{marginTop: "5px"}} span={(Setting.isMobile()) ? 22 : 2}>
{i18next.t("general:Name")}:
</Col>
<Col span={22} >
<Input value={this.state.wordset.name} onChange={e => {
this.updateWordsetField("name", e.target.value);
}} />
</Col>
</Row>
<Row style={{marginTop: "20px"}} >
<Col style={{marginTop: "5px"}} span={(Setting.isMobile()) ? 22 : 2}>
{i18next.t("general:Display name")}:
</Col>
<Col span={22} >
<Input value={this.state.wordset.displayName} onChange={e => {
this.updateWordsetField("displayName", e.target.value);
}} />
</Col>
</Row>
<Row style={{marginTop: "20px"}} >
<Col style={{marginTop: "5px"}} span={(Setting.isMobile()) ? 22 : 2}>
{i18next.t("wordset:Vectorset")}:
</Col>
<Col span={22} >
<Select virtual={false} style={{width: "100%"}} value={this.state.wordset.vectorset} onChange={(value => {this.updateWordsetField("vectorset", value);})}>
{
this.state.vectorsets?.map((vectorset, index) => <Option key={index} value={vectorset.name}>{vectorset.name}</Option>)
}
</Select>
</Col>
</Row>
<Row style={{marginTop: "20px"}} >
<Col style={{marginTop: "5px"}} span={(Setting.isMobile()) ? 22 : 2}>
{i18next.t("wordset:Match")}:
</Col>
<Col span={22} >
<Button loading={this.state.matchLoading} type="primary" onClick={() => {
this.setState({
matchLoading: true,
});
WordsetBackend.getWordsetMatch(this.props.account.name, this.state.wordsetName)
.then((wordset) => {
if (wordset.status === "ok") {
this.setState({
wordset: wordset.data,
matchLoading: false,
});
} else {
Setting.showMessage("error", `Failed to get wordset: ${wordset.msg}`);
}
});
}}>{i18next.t("wordset:Match")}</Button>
</Col>
</Row>
<Row style={{marginTop: "20px"}} >
<Col style={{marginTop: "5px"}} span={(Setting.isMobile()) ? 22 : 2}>
{i18next.t("wordset:Matched")}:
</Col>
<Col span={22} >
<Input value={`${Setting.getPercentage(allWords === 0 ? 0 : validWords / allWords)}% (${validWords} / ${allWords})`} />
</Col>
</Row>
<Row style={{marginTop: "20px"}} >
<Col style={{marginTop: "5px"}} span={(Setting.isMobile()) ? 22 : 2}>
{i18next.t("wordset:Distance limit")}:
</Col>
<Col span={22} >
<InputNumber value={this.state.wordset.distanceLimit} onChange={value => {
this.updateWordsetField("distanceLimit", value);
}} />
</Col>
</Row>
<Row style={{marginTop: "20px"}} >
<Col style={{marginTop: "5px"}} span={(Setting.isMobile()) ? 22 : 2}>
{i18next.t("wordset:Words")}:
</Col>
<Col span={22} >
<VectorTable
title={i18next.t("wordset:Words")}
table={this.state.wordset.vectors}
wordset={this.state.wordset}
onUpdateTable={(value) => {this.updateWordsetField("vectors", value);}}
/>
</Col>
</Row>
<Row style={{marginTop: "20px"}} >
<Col style={{marginTop: "5px"}} span={(Setting.isMobile()) ? 22 : 2}>
{i18next.t("general:Preview")}:
</Col>
<Col span={22} >
<WordsetGraph wordset={this.state.wordset} wordsetName={this.state.wordset.name} />
</Col>
</Row>
</Card>
);
}
submitWordsetEdit() {
const wordset = Setting.deepCopy(this.state.wordset);
WordsetBackend.updateWordset(this.state.wordset.owner, this.state.wordsetName, wordset)
.then((res) => {
if (res.status === "ok") {
if (res.data) {
Setting.showMessage("success", "Successfully saved");
this.setState({
wordsetName: this.state.wordset.name,
});
this.props.history.push(`/wordsets/${this.state.wordset.name}`);
} else {
Setting.showMessage("error", "failed to save: server side failure");
this.updateWordsetField("name", this.state.wordsetName);
}
} else {
Setting.showMessage("error", `failed to save: ${res.msg}`);
}
})
.catch(error => {
Setting.showMessage("error", `failed to save: ${error}`);
});
}
render() {
return (
<div>
<Row style={{width: "100%"}}>
<Col span={1}>
</Col>
<Col span={22}>
{
this.state.wordset !== null ? this.renderWordset() : null
}
</Col>
<Col span={1}>
</Col>
</Row>
<Row style={{margin: 10}}>
<Col span={2}>
</Col>
<Col span={18}>
<Button type="primary" size="large" onClick={this.submitWordsetEdit.bind(this)}>{i18next.t("general:Save")}</Button>
</Col>
</Row>
</div>
);
}
}
export default WordsetEditPage;