2019-05-29 23:59:47 +08:00
|
|
|
import { Button, Dialog, Intent, Classes, Navbar, NavbarGroup, NavbarHeading, FileInput, HTMLSelect, FormGroup, InputGroup } from '@blueprintjs/core';
|
2019-05-29 23:04:06 +08:00
|
|
|
import { observer } from 'mobx-react';
|
|
|
|
import React, { ChangeEvent, KeyboardEvent } from 'react';
|
|
|
|
import { saveCurrentQuestToFile, setCurrentAreaId } from '../actions/appState';
|
|
|
|
import { loadFile } from '../actions/loadFile';
|
|
|
|
import { appStateStore } from '../stores/AppStateStore';
|
|
|
|
import './ApplicationComponent.css';
|
|
|
|
import { RendererComponent } from './RendererComponent';
|
2019-05-29 00:40:29 +08:00
|
|
|
import { EntityInfoComponent } from './EntityInfoComponent';
|
|
|
|
import { QuestInfoComponent } from './QuestInfoComponent';
|
|
|
|
|
|
|
|
@observer
|
|
|
|
export class ApplicationComponent extends React.Component<{}, {
|
|
|
|
filename?: string,
|
2019-05-29 23:04:06 +08:00
|
|
|
saveDialogOpen: boolean,
|
|
|
|
saveDialogFilename: string
|
2019-05-29 00:40:29 +08:00
|
|
|
}> {
|
|
|
|
state = {
|
|
|
|
filename: undefined,
|
2019-05-29 23:04:06 +08:00
|
|
|
saveDialogOpen: false,
|
|
|
|
saveDialogFilename: 'Untitled',
|
2019-05-29 00:40:29 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
render() {
|
2019-05-29 23:04:06 +08:00
|
|
|
const quest = appStateStore.currentQuest;
|
|
|
|
const model = appStateStore.currentModel;
|
2019-05-29 23:59:47 +08:00
|
|
|
const areas = quest && Array.from(quest.areaVariants).map(a => a.area);
|
2019-05-29 23:04:06 +08:00
|
|
|
const area = appStateStore.currentArea;
|
2019-05-29 23:59:47 +08:00
|
|
|
const areaId = area && String(area.id);
|
2019-05-29 00:40:29 +08:00
|
|
|
|
|
|
|
return (
|
2019-05-29 23:59:47 +08:00
|
|
|
<div className={`ApplicationComponent ${Classes.DARK}`}>
|
|
|
|
<Navbar>
|
|
|
|
<NavbarGroup className="ApplicationComponent-button-bar">
|
|
|
|
<NavbarHeading className="ApplicationComponent-heading">
|
|
|
|
Phantasmal World
|
2019-05-29 00:40:29 +08:00
|
|
|
<sup className="ApplicationComponent-beta">BETA</sup>
|
2019-05-29 23:59:47 +08:00
|
|
|
</NavbarHeading>
|
|
|
|
<FileInput
|
|
|
|
text={this.state.filename || 'Choose file...'}
|
|
|
|
inputProps={{
|
|
|
|
type: 'file',
|
|
|
|
accept: '.nj, .qst, .xj',
|
|
|
|
onChange: this.onFileChange
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
{areas ? (
|
|
|
|
<HTMLSelect
|
|
|
|
onChange={this.onAreaSelectChange}
|
|
|
|
defaultValue={areaId}
|
|
|
|
>
|
|
|
|
{areas.map(area =>
|
|
|
|
<option key={area.id} value={area.id}>{area.name}</option>
|
|
|
|
)}
|
|
|
|
</HTMLSelect>
|
|
|
|
) : null}
|
|
|
|
{quest ? (
|
|
|
|
<Button
|
2019-05-29 00:40:29 +08:00
|
|
|
text="Save as..."
|
|
|
|
icon="floppy-disk"
|
2019-05-29 23:59:47 +08:00
|
|
|
onClick={this.onSaveAsClick}
|
|
|
|
/>
|
|
|
|
) : null}
|
|
|
|
</NavbarGroup>
|
|
|
|
</Navbar>
|
2019-05-29 00:40:29 +08:00
|
|
|
<div className="ApplicationComponent-main">
|
|
|
|
<QuestInfoComponent
|
|
|
|
quest={quest} />
|
2019-05-29 23:04:06 +08:00
|
|
|
<RendererComponent
|
2019-05-29 00:40:29 +08:00
|
|
|
quest={quest}
|
|
|
|
area={area}
|
|
|
|
model={model} />
|
2019-05-29 23:04:06 +08:00
|
|
|
<EntityInfoComponent entity={appStateStore.selectedEntity} />
|
2019-05-29 00:40:29 +08:00
|
|
|
</div>
|
|
|
|
<Dialog
|
|
|
|
title="Save as..."
|
|
|
|
icon="floppy-disk"
|
2019-05-29 23:59:47 +08:00
|
|
|
className={Classes.DARK}
|
2019-05-29 00:40:29 +08:00
|
|
|
style={{ width: 360 }}
|
2019-05-29 23:04:06 +08:00
|
|
|
isOpen={this.state.saveDialogOpen}
|
|
|
|
onClose={this.onSaveDialogClose}>
|
2019-05-29 23:59:47 +08:00
|
|
|
<div className={Classes.DIALOG_BODY}>
|
|
|
|
<FormGroup label="Name:" labelFor="file-name-input">
|
|
|
|
<InputGroup
|
|
|
|
id="file-name-input"
|
2019-05-29 00:40:29 +08:00
|
|
|
autoFocus={true}
|
2019-05-29 23:04:06 +08:00
|
|
|
value={this.state.saveDialogFilename}
|
|
|
|
maxLength={12}
|
|
|
|
onChange={this.onSaveDialogNameChange}
|
|
|
|
onKeyUp={this.onSaveDialogNameKeyUp}
|
|
|
|
/>
|
2019-05-29 23:59:47 +08:00
|
|
|
</FormGroup>
|
2019-05-29 00:40:29 +08:00
|
|
|
</div>
|
2019-05-29 23:59:47 +08:00
|
|
|
<div className={Classes.DIALOG_FOOTER}>
|
|
|
|
<div className={Classes.DIALOG_FOOTER_ACTIONS}>
|
2019-05-29 00:40:29 +08:00
|
|
|
<Button
|
|
|
|
text="Save"
|
|
|
|
style={{ marginLeft: 10 }}
|
2019-05-29 23:04:06 +08:00
|
|
|
onClick={this.onSaveDialogSaveClick}
|
2019-05-29 00:40:29 +08:00
|
|
|
intent={Intent.PRIMARY} />
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</Dialog>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2019-05-29 23:04:06 +08:00
|
|
|
private onFileChange = (e: ChangeEvent<HTMLInputElement>) => {
|
2019-05-29 00:40:29 +08:00
|
|
|
if (e.currentTarget.files) {
|
|
|
|
const file = e.currentTarget.files[0];
|
|
|
|
|
|
|
|
if (file) {
|
|
|
|
this.setState({
|
|
|
|
filename: file.name
|
|
|
|
});
|
2019-05-29 20:46:29 +08:00
|
|
|
loadFile(file);
|
2019-05-29 00:40:29 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-29 23:04:06 +08:00
|
|
|
private onAreaSelectChange = (e: ChangeEvent<HTMLSelectElement>) => {
|
|
|
|
const areaId = parseInt(e.currentTarget.value, 10);
|
|
|
|
setCurrentAreaId(areaId);
|
2019-05-29 00:40:29 +08:00
|
|
|
}
|
|
|
|
|
2019-05-29 23:04:06 +08:00
|
|
|
private onSaveAsClick = () => {
|
2019-05-29 00:40:29 +08:00
|
|
|
let name = this.state.filename || 'Untitled';
|
|
|
|
name = name.endsWith('.qst') ? name.slice(0, -4) : name;
|
|
|
|
|
|
|
|
this.setState({
|
2019-05-29 23:04:06 +08:00
|
|
|
saveDialogOpen: true,
|
|
|
|
saveDialogFilename: name
|
2019-05-29 00:40:29 +08:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2019-05-29 23:04:06 +08:00
|
|
|
private onSaveDialogNameChange = (e: ChangeEvent<HTMLInputElement>) => {
|
|
|
|
this.setState({ saveDialogFilename: e.currentTarget.value });
|
2019-05-29 00:40:29 +08:00
|
|
|
}
|
|
|
|
|
2019-05-29 23:04:06 +08:00
|
|
|
private onSaveDialogNameKeyUp = (e: KeyboardEvent<HTMLInputElement>) => {
|
2019-05-29 00:40:29 +08:00
|
|
|
if (e.key === 'Enter') {
|
2019-05-29 23:04:06 +08:00
|
|
|
this.onSaveDialogSaveClick();
|
2019-05-29 00:40:29 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-29 23:04:06 +08:00
|
|
|
private onSaveDialogSaveClick = () => {
|
|
|
|
saveCurrentQuestToFile(this.state.saveDialogFilename);
|
|
|
|
this.setState({ saveDialogOpen: false });
|
2019-05-29 00:40:29 +08:00
|
|
|
}
|
|
|
|
|
2019-05-29 23:04:06 +08:00
|
|
|
private onSaveDialogClose = () => {
|
|
|
|
this.setState({ saveDialogOpen: false });
|
2019-05-29 00:40:29 +08:00
|
|
|
}
|
|
|
|
}
|