2019-06-01 05:20:13 +08:00
|
|
|
import { Button, Form, Icon, Input, Modal, Select, Upload } from "antd";
|
|
|
|
import { UploadChangeParam } from "antd/lib/upload";
|
|
|
|
import { UploadFile } from "antd/lib/upload/interface";
|
2019-05-30 03:43:06 +08:00
|
|
|
import { observer } from "mobx-react";
|
2019-07-03 02:56:33 +08:00
|
|
|
import React, { ChangeEvent, ReactNode, Component } from "react";
|
2019-06-28 00:50:22 +08:00
|
|
|
import { quest_editor_store } from "../../stores/QuestEditorStore";
|
2019-05-30 03:43:06 +08:00
|
|
|
import { EntityInfoComponent } from "./EntityInfoComponent";
|
2019-07-03 00:08:06 +08:00
|
|
|
import "./QuestEditorComponent.css";
|
2019-05-30 03:43:06 +08:00
|
|
|
import { QuestInfoComponent } from "./QuestInfoComponent";
|
2019-07-11 23:30:23 +08:00
|
|
|
import { RendererComponent } from "../RendererComponent";
|
|
|
|
import { get_quest_renderer } from "../../rendering/QuestRenderer";
|
2019-07-18 21:39:23 +08:00
|
|
|
import { application_store } from "../../stores/ApplicationStore";
|
2019-05-30 03:43:06 +08:00
|
|
|
|
|
|
|
@observer
|
2019-07-03 02:56:33 +08:00
|
|
|
export class QuestEditorComponent extends Component<
|
2019-07-03 00:08:06 +08:00
|
|
|
{},
|
|
|
|
{
|
2019-07-20 03:49:59 +08:00
|
|
|
debug: boolean;
|
2019-07-03 00:08:06 +08:00
|
|
|
filename?: string;
|
|
|
|
save_dialog_open: boolean;
|
|
|
|
save_dialog_filename: string;
|
|
|
|
}
|
|
|
|
> {
|
2019-05-30 03:43:06 +08:00
|
|
|
state = {
|
2019-07-20 03:49:59 +08:00
|
|
|
debug: false,
|
2019-06-28 00:50:22 +08:00
|
|
|
save_dialog_open: false,
|
2019-07-03 00:08:06 +08:00
|
|
|
save_dialog_filename: "Untitled",
|
2019-05-30 03:43:06 +08:00
|
|
|
};
|
|
|
|
|
2019-07-18 21:39:23 +08:00
|
|
|
componentDidMount(): void {
|
|
|
|
application_store.on_global_keyup("quest_editor", this.keyup);
|
|
|
|
}
|
|
|
|
|
2019-07-03 02:56:33 +08:00
|
|
|
render(): ReactNode {
|
2019-06-28 00:50:22 +08:00
|
|
|
const quest = quest_editor_store.current_quest;
|
2019-05-30 03:43:06 +08:00
|
|
|
|
|
|
|
return (
|
2019-06-01 05:20:13 +08:00
|
|
|
<div className="qe-QuestEditorComponent">
|
2019-07-18 21:39:23 +08:00
|
|
|
<Toolbar on_save_as_clicked={this.save_as_clicked} />
|
2019-06-01 05:20:13 +08:00
|
|
|
<div className="qe-QuestEditorComponent-main">
|
2019-05-30 03:43:06 +08:00
|
|
|
<QuestInfoComponent quest={quest} />
|
2019-07-20 03:49:59 +08:00
|
|
|
<RendererComponent renderer={get_quest_renderer()} debug={this.state.debug} />
|
2019-06-28 00:50:22 +08:00
|
|
|
<EntityInfoComponent entity={quest_editor_store.selected_entity} />
|
2019-05-30 03:43:06 +08:00
|
|
|
</div>
|
2019-06-01 05:20:13 +08:00
|
|
|
<SaveAsForm
|
2019-06-28 00:50:22 +08:00
|
|
|
is_open={this.state.save_dialog_open}
|
|
|
|
filename={this.state.save_dialog_filename}
|
|
|
|
on_filename_change={this.save_dialog_filename_changed}
|
|
|
|
on_ok={this.save_dialog_affirmed}
|
|
|
|
on_cancel={this.save_dialog_cancelled}
|
2019-06-01 05:20:13 +08:00
|
|
|
/>
|
2019-05-30 03:43:06 +08:00
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2019-06-28 00:50:22 +08:00
|
|
|
private save_as_clicked = (filename?: string) => {
|
2019-06-01 23:57:27 +08:00
|
|
|
const name = filename
|
2019-07-03 00:08:06 +08:00
|
|
|
? filename.endsWith(".qst")
|
|
|
|
? filename.slice(0, -4)
|
|
|
|
: filename
|
2019-06-28 00:50:22 +08:00
|
|
|
: this.state.save_dialog_filename;
|
2019-05-30 03:43:06 +08:00
|
|
|
|
2019-06-01 05:20:13 +08:00
|
|
|
this.setState({
|
2019-06-28 00:50:22 +08:00
|
|
|
save_dialog_open: true,
|
2019-07-03 00:08:06 +08:00
|
|
|
save_dialog_filename: name,
|
2019-06-01 05:20:13 +08:00
|
|
|
});
|
2019-07-03 00:08:06 +08:00
|
|
|
};
|
2019-05-30 03:43:06 +08:00
|
|
|
|
2019-06-28 00:50:22 +08:00
|
|
|
private save_dialog_filename_changed = (filename: string) => {
|
|
|
|
this.setState({ save_dialog_filename: filename });
|
2019-07-03 00:08:06 +08:00
|
|
|
};
|
2019-05-30 03:43:06 +08:00
|
|
|
|
2019-06-28 00:50:22 +08:00
|
|
|
private save_dialog_affirmed = () => {
|
|
|
|
quest_editor_store.save_current_quest_to_file(this.state.save_dialog_filename);
|
|
|
|
this.setState({ save_dialog_open: false });
|
2019-07-03 00:08:06 +08:00
|
|
|
};
|
2019-05-30 03:43:06 +08:00
|
|
|
|
2019-06-28 00:50:22 +08:00
|
|
|
private save_dialog_cancelled = () => {
|
|
|
|
this.setState({ save_dialog_open: false });
|
2019-07-03 00:08:06 +08:00
|
|
|
};
|
2019-07-18 21:39:23 +08:00
|
|
|
|
|
|
|
private keyup = (e: KeyboardEvent) => {
|
|
|
|
if (e.ctrlKey && e.key === "z" && !e.altKey) {
|
|
|
|
quest_editor_store.undo_stack.undo();
|
|
|
|
} else if (e.ctrlKey && e.key === "Z" && !e.altKey) {
|
|
|
|
quest_editor_store.undo_stack.redo();
|
2019-07-20 03:49:59 +08:00
|
|
|
} else if (e.ctrlKey && e.altKey && e.key === "d") {
|
|
|
|
this.setState(state => ({ debug: !state.debug }));
|
2019-07-18 21:39:23 +08:00
|
|
|
}
|
|
|
|
};
|
2019-06-01 05:20:13 +08:00
|
|
|
}
|
2019-05-30 03:43:06 +08:00
|
|
|
|
2019-06-01 05:20:13 +08:00
|
|
|
@observer
|
2019-07-18 21:39:23 +08:00
|
|
|
class Toolbar extends Component<{ on_save_as_clicked: (filename?: string) => void }> {
|
2019-06-01 05:20:13 +08:00
|
|
|
state = {
|
2019-07-03 00:08:06 +08:00
|
|
|
filename: undefined,
|
|
|
|
};
|
2019-05-30 03:43:06 +08:00
|
|
|
|
2019-07-03 02:56:33 +08:00
|
|
|
render(): ReactNode {
|
2019-07-18 21:39:23 +08:00
|
|
|
const undo = quest_editor_store.undo_stack;
|
2019-06-28 00:50:22 +08:00
|
|
|
const quest = quest_editor_store.current_quest;
|
2019-07-18 21:39:23 +08:00
|
|
|
const areas = quest ? Array.from(quest.area_variants).map(a => a.area) : [];
|
2019-06-28 00:50:22 +08:00
|
|
|
const area = quest_editor_store.current_area;
|
|
|
|
const area_id = area && area.id;
|
2019-06-01 05:20:13 +08:00
|
|
|
|
|
|
|
return (
|
|
|
|
<div className="qe-QuestEditorComponent-toolbar">
|
|
|
|
<Upload
|
2019-07-01 01:55:30 +08:00
|
|
|
accept=".qst"
|
2019-06-01 05:20:13 +08:00
|
|
|
showUploadList={false}
|
2019-06-28 00:50:22 +08:00
|
|
|
onChange={this.set_filename}
|
2019-06-22 02:06:55 +08:00
|
|
|
// Make sure it doesn't do a POST:
|
|
|
|
customRequest={() => false}
|
2019-06-01 05:20:13 +08:00
|
|
|
>
|
2019-07-03 00:08:06 +08:00
|
|
|
<Button icon="file">{this.state.filename || "Open file..."}</Button>
|
2019-06-01 05:20:13 +08:00
|
|
|
</Upload>
|
2019-07-18 21:39:23 +08:00
|
|
|
<Select
|
|
|
|
onChange={quest_editor_store.set_current_area_id}
|
|
|
|
value={area_id}
|
|
|
|
style={{ width: 200 }}
|
|
|
|
disabled={!quest}
|
|
|
|
>
|
|
|
|
{areas.map(area => (
|
|
|
|
<Select.Option key={area.id} value={area.id}>
|
|
|
|
{area.name}
|
|
|
|
</Select.Option>
|
|
|
|
))}
|
|
|
|
</Select>
|
|
|
|
<Button icon="save" onClick={this.save_as} disabled={!quest}>
|
|
|
|
Save as...
|
|
|
|
</Button>
|
|
|
|
<Button
|
|
|
|
icon="undo"
|
|
|
|
onClick={this.undo}
|
|
|
|
title={"Undo" + (undo.first_undo ? ` "${undo.first_undo.description}"` : "")}
|
|
|
|
disabled={!undo.can_undo}
|
|
|
|
/>
|
|
|
|
<Button
|
|
|
|
icon="redo"
|
|
|
|
onClick={this.redo}
|
|
|
|
title={"Redo" + (undo.first_redo ? ` "${undo.first_redo.description}"` : "")}
|
|
|
|
disabled={!quest_editor_store.undo_stack.can_redo}
|
|
|
|
/>
|
2019-06-01 05:20:13 +08:00
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2019-06-28 00:50:22 +08:00
|
|
|
private set_filename = (info: UploadChangeParam<UploadFile>) => {
|
2019-06-01 05:20:13 +08:00
|
|
|
if (info.file.originFileObj) {
|
|
|
|
this.setState({ filename: info.file.name });
|
2019-07-11 23:40:23 +08:00
|
|
|
quest_editor_store.load_file(info.file.originFileObj as File);
|
2019-05-30 03:43:06 +08:00
|
|
|
}
|
2019-07-03 00:08:06 +08:00
|
|
|
};
|
2019-05-30 03:43:06 +08:00
|
|
|
|
2019-07-18 21:39:23 +08:00
|
|
|
private save_as = () => {
|
|
|
|
this.props.on_save_as_clicked(this.state.filename);
|
|
|
|
};
|
|
|
|
|
|
|
|
private undo = () => {
|
|
|
|
quest_editor_store.undo_stack.undo();
|
|
|
|
};
|
|
|
|
|
|
|
|
private redo = () => {
|
|
|
|
quest_editor_store.undo_stack.redo();
|
2019-07-03 00:08:06 +08:00
|
|
|
};
|
2019-06-01 05:20:13 +08:00
|
|
|
}
|
2019-05-30 03:43:06 +08:00
|
|
|
|
2019-06-01 05:20:13 +08:00
|
|
|
class SaveAsForm extends React.Component<{
|
2019-07-03 00:08:06 +08:00
|
|
|
is_open: boolean;
|
|
|
|
filename: string;
|
|
|
|
on_filename_change: (name: string) => void;
|
|
|
|
on_ok: () => void;
|
|
|
|
on_cancel: () => void;
|
2019-06-01 05:20:13 +08:00
|
|
|
}> {
|
2019-07-03 02:56:33 +08:00
|
|
|
render(): ReactNode {
|
2019-06-01 05:20:13 +08:00
|
|
|
return (
|
|
|
|
<Modal
|
2019-07-03 00:08:06 +08:00
|
|
|
title={
|
|
|
|
<>
|
|
|
|
<Icon type="save" /> Save as...
|
|
|
|
</>
|
|
|
|
}
|
2019-06-28 00:50:22 +08:00
|
|
|
visible={this.props.is_open}
|
|
|
|
onOk={this.props.on_ok}
|
|
|
|
onCancel={this.props.on_cancel}
|
2019-06-01 05:20:13 +08:00
|
|
|
>
|
|
|
|
<Form layout="vertical">
|
|
|
|
<Form.Item label="Name">
|
|
|
|
<Input
|
|
|
|
autoFocus={true}
|
|
|
|
maxLength={12}
|
|
|
|
value={this.props.filename}
|
2019-06-28 00:50:22 +08:00
|
|
|
onChange={this.name_changed}
|
2019-06-01 05:20:13 +08:00
|
|
|
/>
|
|
|
|
</Form.Item>
|
|
|
|
</Form>
|
|
|
|
</Modal>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2019-06-28 00:50:22 +08:00
|
|
|
private name_changed = (e: ChangeEvent<HTMLInputElement>) => {
|
|
|
|
this.props.on_filename_change(e.currentTarget.value);
|
2019-07-03 00:08:06 +08:00
|
|
|
};
|
2019-06-01 05:20:13 +08:00
|
|
|
}
|