113 lines
3.6 KiB
TypeScript
113 lines
3.6 KiB
TypeScript
// This file was cobbled together and is the messiest part of this project
|
|
|
|
import { at } from "../etc";
|
|
import { u8 } from "../num";
|
|
|
|
const HEX_CHARACTERS = ["0", "1", "2", "3", "4", "5", "6", "7", "8", "9", "A", "B", "C", "D", "E", "F"];
|
|
|
|
export class EditorContext {
|
|
private list: Array<HTMLElement>;
|
|
private width: number;
|
|
private height: number;
|
|
private enabled: boolean = false;
|
|
private current_cell_info: { left?: string; right?: string; old?: string };
|
|
private edit_callback: (n: number, value: u8) => void;
|
|
constructor(list: Array<HTMLElement>, width: number, height: number, callback: (n: number, value: u8) => void) {
|
|
this.list = list;
|
|
this.width = width;
|
|
this.height = height;
|
|
this.edit_callback = callback;
|
|
this.current_cell_info = {};
|
|
|
|
for (const [i, cell] of this.list.entries()) {
|
|
cell.setAttribute("spellcheck", "false");
|
|
cell.addEventListener("keydown", (e) => {
|
|
this.keydown(e, i);
|
|
});
|
|
cell.addEventListener("focus", () => {
|
|
if (!this.enabled) return;
|
|
this.current_cell_info.old = cell.textContent ?? "00";
|
|
this.current_cell_info.left = undefined;
|
|
this.current_cell_info.right = undefined;
|
|
cell.classList.add("caret_selected");
|
|
|
|
// Reset cursor position (I know there's an API for this, but this is a simpler, more robust solution)
|
|
cell.textContent = cell.textContent ?? "00";
|
|
});
|
|
|
|
cell.addEventListener("blur", () => {
|
|
const left = this.current_cell_info.left;
|
|
const right = this.current_cell_info.right;
|
|
cell.classList.remove("caret_selected");
|
|
if (left === undefined || right === undefined) {
|
|
cell.textContent = this.current_cell_info.old ?? "00";
|
|
} else if (left !== undefined && right !== undefined) {
|
|
const text = `${left}${right}`;
|
|
cell.textContent = text;
|
|
const val = Number.parseInt(text, 16) as u8;
|
|
this.edit_callback(i, val);
|
|
cell.classList.add("recent_edit");
|
|
}
|
|
});
|
|
}
|
|
}
|
|
|
|
enable(): void {
|
|
this.enabled = true;
|
|
for (const cell of this.list) {
|
|
cell.setAttribute("contenteditable", "true");
|
|
}
|
|
}
|
|
disable(): void {
|
|
this.enabled = false;
|
|
for (const cell of this.list) {
|
|
cell.removeAttribute("contenteditable");
|
|
cell.blur();
|
|
}
|
|
this.current_cell_info = {};
|
|
}
|
|
|
|
private keydown(e: KeyboardEvent, cell_index: number): void {
|
|
if (!this.enabled) return;
|
|
const cell = e.target as HTMLElement;
|
|
|
|
const next: null | HTMLElement = at(this.list, cell_index + 1);
|
|
const prev: null | HTMLElement = at(this.list, cell_index - 1);
|
|
const up: null | HTMLElement = at(this.list, cell_index - this.width);
|
|
const down: null | HTMLElement = at(this.list, cell_index + this.width);
|
|
|
|
const k = e.key;
|
|
if (k === "ArrowUp") {
|
|
(up ?? prev)?.focus();
|
|
cell.blur();
|
|
} else if (k === "ArrowDown") {
|
|
(down ?? next)?.focus();
|
|
cell.blur();
|
|
} else if (k === "ArrowLeft" || k === "Backspace") {
|
|
prev?.focus();
|
|
cell.blur();
|
|
} else if (k === "ArrowRight") {
|
|
next?.focus();
|
|
cell.blur();
|
|
} else if (k === "Enter") {
|
|
cell.blur();
|
|
} else if (k === "Escape") {
|
|
cell.blur();
|
|
return;
|
|
} else if (HEX_CHARACTERS.includes(k.toUpperCase())) {
|
|
if (this.current_cell_info.left === undefined) {
|
|
this.current_cell_info.left = k.toUpperCase();
|
|
cell.innerHTML = `<span class="pending_edit">${this.current_cell_info.left}</span>0`;
|
|
} else if (this.current_cell_info.right === undefined) {
|
|
this.current_cell_info.right = k.toUpperCase();
|
|
cell.textContent = `${this.current_cell_info.left}${this.current_cell_info.right}`;
|
|
next?.focus();
|
|
cell.blur();
|
|
}
|
|
} else if (k === "Tab") {
|
|
return;
|
|
}
|
|
e.preventDefault();
|
|
}
|
|
}
|