From 3edc9b857d6c37cdfde1f1c535a98191c197cfce Mon Sep 17 00:00:00 2001 From: Daan Vanden Bosch Date: Sun, 5 Jan 2020 23:20:53 +0100 Subject: [PATCH] Improved FileButton behavior, opening the same file twice in a row will now work. Also simplified its implementation. --- deploy.ps1 | 6 ++ src/core/files.ts | 35 +++++++++ src/core/gui/FileButton.css | 11 --- src/core/gui/FileButton.ts | 72 +++---------------- src/core/read_file.ts | 15 ---- .../QuestEditorToolBarController.ts | 16 ++--- src/quest_editor/gui/QuestEditorToolBar.ts | 5 +- .../QuestEditorToolBar.test.ts.snap | 16 ++--- src/viewer/controllers/TextureController.ts | 2 +- .../model/CharacterClassOptionsController.ts | 2 +- .../model/ModelToolBarController.ts | 2 +- src/viewer/gui/TextureView.ts | 3 +- .../__snapshots__/TextureView.test.ts.snap | 15 ++-- .../gui/model/CharacterClassOptionsView.ts | 4 +- src/viewer/gui/model/ModelToolBarView.ts | 3 +- .../__snapshots__/ModelView.test.ts.snap | 15 ++-- version.txt | 2 +- 17 files changed, 84 insertions(+), 140 deletions(-) create mode 100644 src/core/files.ts delete mode 100644 src/core/gui/FileButton.css delete mode 100644 src/core/read_file.ts diff --git a/deploy.ps1 b/deploy.ps1 index 76de6f20..12f9b7ab 100644 --- a/deploy.ps1 +++ b/deploy.ps1 @@ -3,9 +3,11 @@ Write-Output "Running tests." yarn test +if ($LastExitCode -ne 0) { throw "Tests failed." } Write-Output "Generating production build." yarn build +if ($LastExitCode -ne 0) { throw "Build failed." } Write-Output "Deleting ./deployment contents." Remove-Item -Recurse ./deployment/* @@ -22,9 +24,13 @@ Write-Output $version > version.txt Write-Output "Committing and pushing to gh-pages." Set-Location ./deployment git pull +if ($LastExitCode -ne 0) { throw "Git pull failed." } git add . +if ($LastExitCode -ne 0) { throw "Git add failed." } git commit -m "Release $version." +if ($LastExitCode -ne 0) { throw "Git commit failed." } git push +if ($LastExitCode -ne 0) { throw "Git push failed." } Set-Location .. diff --git a/src/core/files.ts b/src/core/files.ts new file mode 100644 index 00000000..70f0d786 --- /dev/null +++ b/src/core/files.ts @@ -0,0 +1,35 @@ +import { input } from "./gui/dom"; + +export function open_files(options?: { accept?: string; multiple?: boolean }): Promise { + return new Promise((resolve) => { + const el = input({ type: "file" }); + el.accept = options?.accept ?? ""; + el.multiple = options?.multiple ?? false; + + el.onchange = () => { + if (el.files && el.files.length) { + resolve([...el.files]); + } else { + resolve([]); + } + }; + + el.click(); + }); +} + +export function read_file(file: File): Promise { + return new Promise((resolve, reject) => { + const reader = new FileReader(); + + reader.addEventListener("loadend", () => { + if (reader.result instanceof ArrayBuffer) { + resolve(reader.result); + } else { + reject(new Error("Couldn't read file.")); + } + }); + + reader.readAsArrayBuffer(file); + }); +} diff --git a/src/core/gui/FileButton.css b/src/core/gui/FileButton.css deleted file mode 100644 index f8f72632..00000000 --- a/src/core/gui/FileButton.css +++ /dev/null @@ -1,11 +0,0 @@ -@import "./Button.css"; - -.core_FileButton_input { - overflow: hidden; - clip: rect(0, 0, 0, 0); - position: absolute; - width: 1px; - height: 1px; - border: none; - padding: 0; -} diff --git a/src/core/gui/FileButton.ts b/src/core/gui/FileButton.ts index dfc7293f..75b7c8b8 100644 --- a/src/core/gui/FileButton.ts +++ b/src/core/gui/FileButton.ts @@ -1,82 +1,30 @@ -import { icon, Icon, input, label, span } from "./dom"; -import "./FileButton.css"; import { property } from "../observable"; import { Property } from "../observable/property/Property"; -import { Control, ControlOptions } from "./Control"; import { WritableProperty } from "../observable/property/WritableProperty"; +import { Button, ButtonOptions } from "./Button"; +import { open_files } from "../files"; -export type FileButtonOptions = ControlOptions & { +export type FileButtonOptions = ButtonOptions & { accept?: string; multiple?: boolean; - icon_left?: Icon; }; -export class FileButton extends Control { - readonly element = label({ - className: "core_FileButton core_Button", - }); - - readonly files: Property; - - private input: HTMLInputElement = input({ - className: "core_FileButton_input core_Button_inner", - }); - +export class FileButton extends Button { private readonly _files: WritableProperty = property([]); - constructor(text: string, options?: FileButtonOptions) { + readonly files: Property = this._files; + + constructor(options?: FileButtonOptions) { super(options); - this.files = this._files; - - this.input.type = "file"; - this.input.onchange = () => { - if (this.input.files && this.input.files.length) { - this._files.val = [...this.input.files!]; - } else { - this._files.val = []; - } - }; - - const inner_element = span({ - className: "core_FileButton_inner core_Button_inner", - }); - - if (options) { - if (options.accept != undefined) this.input.accept = options.accept; - - if (options.multiple != undefined) this.input.multiple = options.multiple; - - if (options.icon_left != undefined) { - inner_element.append( - span( - { className: "core_FileButton_left core_Button_left" }, - icon(options.icon_left), - ), - ); - } - } - - inner_element.append(span({ className: "core_Button_center" }, text)); - - this.element.append(inner_element, this.input); + this.element.classList.add("core_FileButton"); this.disposables( - this.enabled.observe(({ value }) => { - this.input.disabled = !value; - - if (value) { - this.element.classList.remove("disabled"); - } else { - this.element.classList.add("disabled"); - } + this.click.observe(async () => { + this._files.val = await open_files(options); }), ); this.finalize_construction(); } - - click(): void { - this.input.click(); - } } diff --git a/src/core/read_file.ts b/src/core/read_file.ts deleted file mode 100644 index 51c48317..00000000 --- a/src/core/read_file.ts +++ /dev/null @@ -1,15 +0,0 @@ -export async function read_file(file: File): Promise { - return new Promise((resolve, reject) => { - const reader = new FileReader(); - - reader.addEventListener("loadend", () => { - if (reader.result instanceof ArrayBuffer) { - resolve(reader.result); - } else { - reject(new Error("Couldn't read file.")); - } - }); - - reader.readAsArrayBuffer(file); - }); -} diff --git a/src/quest_editor/controllers/QuestEditorToolBarController.ts b/src/quest_editor/controllers/QuestEditorToolBarController.ts index f9c4cd8b..2a678e47 100644 --- a/src/quest_editor/controllers/QuestEditorToolBarController.ts +++ b/src/quest_editor/controllers/QuestEditorToolBarController.ts @@ -8,7 +8,7 @@ import { undo_manager } from "../../core/undo/UndoManager"; import { Controller } from "../../core/controllers/Controller"; import { Episode } from "../../core/data_formats/parsing/quest/Episode"; import { create_new_quest } from "../stores/quest_creation"; -import { read_file } from "../../core/read_file"; +import { open_files, read_file } from "../../core/files"; import { parse_bin_dat_to_quest, parse_qst_to_quest, @@ -19,7 +19,6 @@ import { ArrayBufferCursor } from "../../core/data_formats/cursor/ArrayBufferCur import { Endianness } from "../../core/data_formats/Endianness"; import { convert_quest_from_model, convert_quest_to_model } from "../stores/model_conversion"; import { LogManager } from "../../core/Logger"; -import { input } from "../../core/gui/dom"; import { basename } from "../../core/util"; const logger = LogManager.get("quest_editor/controllers/QuestEditorToolBarController"); @@ -102,14 +101,9 @@ export class QuestEditorToolBarController extends Controller { this.quest_filename = undefined; }), - gui_store.on_global_keydown(GuiTool.QuestEditor, "Ctrl-O", () => { - const input_element = input(); - input_element.type = "file"; - input_element.multiple = true; - input_element.onchange = () => { - this.open_files(Array.prototype.slice.apply(input_element.files)); - }; - input_element.click(); + gui_store.on_global_keydown(GuiTool.QuestEditor, "Ctrl-O", async () => { + const files = await open_files({ accept: ".bin, .dat, .qst", multiple: true }); + this.parse_files(files); }), gui_store.on_global_keydown(GuiTool.QuestEditor, "Ctrl-Shift-S", this.save_as), @@ -140,7 +134,7 @@ export class QuestEditorToolBarController extends Controller { this.quest_editor_store.set_current_quest(create_new_quest(this.area_store, episode)); // TODO: notify user of problems. - open_files = async (files: File[]): Promise => { + parse_files = async (files: File[]): Promise => { try { if (files.length === 0) return; diff --git a/src/quest_editor/gui/QuestEditorToolBar.ts b/src/quest_editor/gui/QuestEditorToolBar.ts index e21141a5..bb7b78fd 100644 --- a/src/quest_editor/gui/QuestEditorToolBar.ts +++ b/src/quest_editor/gui/QuestEditorToolBar.ts @@ -19,8 +19,9 @@ export class QuestEditorToolBar extends ToolBar { items: [Episode.I], to_label: episode => `Episode ${Episode[episode]}`, }); - const open_file_button = new FileButton("Open file...", { + const open_file_button = new FileButton({ icon_left: Icon.File, + text: "Open file...", accept: ".bin, .dat, .qst", multiple: true, tooltip: "Open a quest file (Ctrl-O)", @@ -108,7 +109,7 @@ export class QuestEditorToolBar extends ToolBar { this.disposables( new_quest_button.chosen.observe(({ value: episode }) => ctrl.create_new_quest(episode)), - open_file_button.files.observe(({ value: files }) => ctrl.open_files(files)), + open_file_button.files.observe(({ value: files }) => ctrl.parse_files(files)), save_as_button.click.observe(ctrl.save_as), save_as_button.enabled.bind_to(ctrl.can_save), diff --git a/src/quest_editor/gui/__snapshots__/QuestEditorToolBar.test.ts.snap b/src/quest_editor/gui/__snapshots__/QuestEditorToolBar.test.ts.snap index b10efe22..f13059ae 100644 --- a/src/quest_editor/gui/__snapshots__/QuestEditorToolBar.test.ts.snap +++ b/src/quest_editor/gui/__snapshots__/QuestEditorToolBar.test.ts.snap @@ -55,15 +55,15 @@ exports[`Renders correctly. 1`] = ` - +
= this.add( + const section_id_select: Select = this.add( new Select({ class: "viewer_model_CharacterClassOptionsView_section_id", label: "Section ID:", items: SectionIds, selected: ctrl.current_section_id, - to_label: section_id => SectionId[section_id], + to_label: section_id => (section_id == undefined ? "" : SectionId[section_id]), enabled: ctrl.enabled, }), ); diff --git a/src/viewer/gui/model/ModelToolBarView.ts b/src/viewer/gui/model/ModelToolBarView.ts index c8201c5c..3a9ebcb7 100644 --- a/src/viewer/gui/model/ModelToolBarView.ts +++ b/src/viewer/gui/model/ModelToolBarView.ts @@ -22,8 +22,9 @@ export class ModelToolBarView extends View { constructor(ctrl: ModelToolBarController) { super(); - const open_file_button = new FileButton("Open file...", { + const open_file_button = new FileButton({ icon_left: Icon.File, + text: "Open file...", accept: ".afs, .nj, .njm, .xj, .xvm", }); const skeleton_checkbox = new CheckBox(false, { label: "Show skeleton" }); diff --git a/src/viewer/gui/model/__snapshots__/ModelView.test.ts.snap b/src/viewer/gui/model/__snapshots__/ModelView.test.ts.snap index 323e5c70..fd953e8a 100644 --- a/src/viewer/gui/model/__snapshots__/ModelView.test.ts.snap +++ b/src/viewer/gui/model/__snapshots__/ModelView.test.ts.snap @@ -8,14 +8,14 @@ exports[`Renders correctly. 1`] = ` class="core_ToolBar" style="height: 33px;" > - +
diff --git a/version.txt b/version.txt index c739b42c..ea90ee31 100644 --- a/version.txt +++ b/version.txt @@ -1 +1 @@ -44 +45