From d9b6c7015aab7a645288f5027e135719b0c64f9f Mon Sep 17 00:00:00 2001 From: Daan Vanden Bosch Date: Sat, 14 Sep 2019 16:11:45 +0200 Subject: [PATCH] The hunt method tables can be sorted again. There's no visual feedback in the table headers though. --- src/core/gui/Table.ts | 62 +++++++- src/core/gui/dom.ts | 79 ++++------ .../property/list/SimpleListProperty.ts | 11 ++ .../property/list/WritableListProperty.ts | 2 + .../gui/MethodsForEpisodeView.ts | 43 ++++- .../hunt_optimizer/ui/MethodsComponent.css | 7 - .../hunt_optimizer/ui/MethodsComponent.tsx | 147 ------------------ 7 files changed, 141 insertions(+), 210 deletions(-) delete mode 100644 src/old/hunt_optimizer/ui/MethodsComponent.css delete mode 100644 src/old/hunt_optimizer/ui/MethodsComponent.tsx diff --git a/src/core/gui/Table.ts b/src/core/gui/Table.ts index e1c61bb6..f4d42399 100644 --- a/src/core/gui/Table.ts +++ b/src/core/gui/Table.ts @@ -12,12 +12,14 @@ import Logger = require("js-logger"); const logger = Logger.get("core/gui/Table"); export type Column = { + key?: string; title: string; fixed?: boolean; width: number; input?: boolean; text_align?: string; tooltip?: (value: T) => string; + sortable?: boolean; render_cell(value: T, disposer: Disposer): string | HTMLElement; footer?: { render_cell(): string; @@ -25,10 +27,15 @@ export type Column = { }; }; +export enum SortDirection { + Asc, + Desc, +} + export type TableOptions = WidgetOptions & { values: ListProperty; columns: Column[]; - sort?(columns: Column): void; + sort?(sort_columns: { column: Column; direction: SortDirection }[]): void; }; export class Table extends Widget { @@ -44,6 +51,8 @@ export class Table extends Widget { this.values = options.values; this.columns = options.columns; + let sort_columns: { column: Column; direction: SortDirection }[] = []; + const thead_element = el.thead(); const header_tr_element = el.tr(); @@ -51,8 +60,11 @@ export class Table extends Widget { let has_footer = false; header_tr_element.append( - ...this.columns.map(column => { - const th = el.th({}, el.span({ text: column.title })); + ...this.columns.map((column, index) => { + const th = el.th( + { data: { index: index.toString() } }, + el.span({ text: column.title }), + ); if (column.fixed) { th.style.position = "sticky"; @@ -70,6 +82,50 @@ export class Table extends Widget { }), ); + const sort = options.sort; + + if (sort) { + header_tr_element.onmousedown = e => { + if (e.target instanceof HTMLElement) { + let element: HTMLElement = e.target; + + for (let i = 0; i < 5; i++) { + if (element.dataset.index) { + break; + } else if (element.parentElement) { + element = element.parentElement; + } else { + return; + } + } + + if (!element.dataset.index) return; + + const index = parseInt(element.dataset.index, 10); + const column = this.columns[index]; + if (!column.sortable) return; + + const existing_index = sort_columns.findIndex(sc => sc.column === column); + + if (existing_index === 0) { + const sc = sort_columns[0]; + sc.direction = + sc.direction === SortDirection.Asc + ? SortDirection.Desc + : SortDirection.Asc; + } else { + if (existing_index !== -1) { + sort_columns.splice(existing_index, 1); + } + + sort_columns.unshift({ column, direction: SortDirection.Asc }); + } + + sort(sort_columns); + } + }; + } + thead_element.append(header_tr_element); this.tbody_element = el.tbody(); this.element.append(thead_element, this.tbody_element); diff --git a/src/core/gui/dom.ts b/src/core/gui/dom.ts index e27a114e..1a4b7aa2 100644 --- a/src/core/gui/dom.ts +++ b/src/core/gui/dom.ts @@ -3,50 +3,30 @@ import { Observable } from "../observable/Observable"; import { is_property } from "../observable/property/Property"; import { SectionId } from "../model"; +type ElementAttributes = { + class?: string; + tab_index?: number; + text?: string; + title?: string; + data?: { [key: string]: string }; +}; + export const el = { - div: ( - attributes?: { - class?: string; - tab_index?: number; - text?: string; - data?: { [key: string]: string }; - }, - ...children: HTMLElement[] - ): HTMLDivElement => create_element("div", attributes, ...children), + div: (attributes?: ElementAttributes, ...children: HTMLElement[]): HTMLDivElement => + create_element("div", attributes, ...children), - span: ( - attributes?: { - class?: string; - tab_index?: number; - text?: string; - data?: { [key: string]: string }; - }, - ...children: HTMLElement[] - ): HTMLSpanElement => create_element("span", attributes, ...children), + span: (attributes?: ElementAttributes, ...children: HTMLElement[]): HTMLSpanElement => + create_element("span", attributes, ...children), - h2: ( - attributes?: { - class?: string; - tab_index?: number; - text?: string; - data?: { [key: string]: string }; - }, - ...children: HTMLElement[] - ): HTMLHeadingElement => create_element("h2", attributes, ...children), + h2: (attributes?: ElementAttributes, ...children: HTMLElement[]): HTMLHeadingElement => + create_element("h2", attributes, ...children), - p: ( - attributes?: { - class?: string; - text?: string; - }, - ...children: HTMLElement[] - ): HTMLParagraphElement => create_element("p", attributes, ...children), + p: (attributes?: ElementAttributes, ...children: HTMLElement[]): HTMLParagraphElement => + create_element("p", attributes, ...children), a: ( - attributes?: { - class?: string; + attributes?: ElementAttributes & { href?: string; - title?: string; }, ...children: HTMLElement[] ): HTMLAnchorElement => { @@ -60,47 +40,42 @@ export const el = { return element; }, - table: (attributes?: {}, ...children: HTMLElement[]): HTMLTableElement => + table: (attributes?: ElementAttributes, ...children: HTMLElement[]): HTMLTableElement => create_element("table", attributes, ...children), - thead: (attributes?: {}, ...children: HTMLElement[]): HTMLTableSectionElement => + thead: (attributes?: ElementAttributes, ...children: HTMLElement[]): HTMLTableSectionElement => create_element("thead", attributes, ...children), - tbody: (attributes?: {}, ...children: HTMLElement[]): HTMLTableSectionElement => + tbody: (attributes?: ElementAttributes, ...children: HTMLElement[]): HTMLTableSectionElement => create_element("tbody", attributes, ...children), - tfoot: (attributes?: {}, ...children: HTMLElement[]): HTMLTableSectionElement => + tfoot: (attributes?: ElementAttributes, ...children: HTMLElement[]): HTMLTableSectionElement => create_element("tfoot", attributes, ...children), - tr: (attributes?: {}, ...children: HTMLElement[]): HTMLTableRowElement => + tr: (attributes?: ElementAttributes, ...children: HTMLElement[]): HTMLTableRowElement => create_element("tr", attributes, ...children), th: ( - attributes?: { class?: string; text?: string; col_span?: number }, + attributes?: ElementAttributes & { col_span?: number }, ...children: HTMLElement[] ): HTMLTableHeaderCellElement => create_element("th", attributes, ...children), td: ( - attributes?: { text?: string; col_span?: number }, + attributes?: ElementAttributes & { col_span?: number }, ...children: HTMLElement[] ): HTMLTableCellElement => create_element("td", attributes, ...children), - button: (attributes?: {}, ...children: HTMLElement[]): HTMLButtonElement => + button: (attributes?: ElementAttributes, ...children: HTMLElement[]): HTMLButtonElement => create_element("button", attributes, ...children), - textarea: (attributes?: {}, ...children: HTMLElement[]): HTMLTextAreaElement => + textarea: (attributes?: ElementAttributes, ...children: HTMLElement[]): HTMLTextAreaElement => create_element("textarea", attributes, ...children), }; export function create_element( tag_name: string, - attributes?: { - class?: string; - tab_index?: number; - text?: string; - title?: string; + attributes?: ElementAttributes & { href?: string; - data?: { [key: string]: string }; col_span?: number; }, ...children: HTMLElement[] diff --git a/src/core/observable/property/list/SimpleListProperty.ts b/src/core/observable/property/list/SimpleListProperty.ts index 8bd4adce..c86e609c 100644 --- a/src/core/observable/property/list/SimpleListProperty.ts +++ b/src/core/observable/property/list/SimpleListProperty.ts @@ -194,6 +194,17 @@ export class SimpleListProperty extends AbstractProperty return removed; } + sort(compare: (a: T, b: T) => number): void { + this.values.sort(compare); + + this.finalize_update({ + type: ListChangeType.ListChange, + index: 0, + removed: this.values, + inserted: this.values, + }); + } + /** * Does the following in the given order: * - Updates value observers diff --git a/src/core/observable/property/list/WritableListProperty.ts b/src/core/observable/property/list/WritableListProperty.ts index f09ceb3f..b76795e1 100644 --- a/src/core/observable/property/list/WritableListProperty.ts +++ b/src/core/observable/property/list/WritableListProperty.ts @@ -14,4 +14,6 @@ export interface WritableListProperty extends ListProperty, WritableProper remove(...values: T[]): void; clear(): void; + + sort(compare: (a: T, b: T) => number): void; } diff --git a/src/hunt_optimizer/gui/MethodsForEpisodeView.ts b/src/hunt_optimizer/gui/MethodsForEpisodeView.ts index 9c90ecbb..8826cfef 100644 --- a/src/hunt_optimizer/gui/MethodsForEpisodeView.ts +++ b/src/hunt_optimizer/gui/MethodsForEpisodeView.ts @@ -12,7 +12,7 @@ import "./MethodsForEpisodeView.css"; import { Disposer } from "../../core/observable/Disposer"; import { DurationInput } from "../../core/gui/DurationInput"; import { Disposable } from "../../core/observable/Disposable"; -import { Table } from "../../core/gui/Table"; +import { SortDirection, Table } from "../../core/gui/Table"; import { list_property } from "../../core/observable"; export class MethodsForEpisodeView extends ResizableWidget { @@ -33,20 +33,59 @@ export class MethodsForEpisodeView extends ResizableWidget { new Table({ class: "hunt_optimizer_MethodsForEpisodeView_table", values: hunt_methods, + sort: sort_columns => { + hunt_methods.sort((a, b) => { + for (const { column, direction } of sort_columns) { + let cmp = 0; + + switch (column.key) { + case "method": + cmp = a.name.localeCompare(b.name); + break; + + case "time": + cmp = a.time.val.as("minutes") - b.time.val.as("minutes"); + break; + + default: + { + const type = (NpcType as any)[column.key!]; + + if (type) { + cmp = + (a.enemy_counts.get(type) || 0) - + (b.enemy_counts.get(type) || 0); + } + } + break; + } + + if (cmp !== 0) { + return direction === SortDirection.Asc ? cmp : -cmp; + } + } + + return 0; + }); + }, columns: [ { + key: "method", title: "Method", fixed: true, width: 250, + sortable: true, render_cell(method: HuntMethodModel) { return method.name; }, }, { + key: "time", title: "Time", fixed: true, width: 60, input: true, + sortable: true, render_cell(method: HuntMethodModel, disposer: Disposer) { const time_input = disposer.add(new DurationInput(method.time.val)); @@ -61,9 +100,11 @@ export class MethodsForEpisodeView extends ResizableWidget { }, ...this.enemy_types.map(enemy_type => { return { + key: NpcType[enemy_type], title: npc_data(enemy_type).simple_name, width: 90, text_align: "right", + sortable: true, render_cell(method: HuntMethodModel) { const count = method.enemy_counts.get(enemy_type); return count == undefined ? "" : count.toString(); diff --git a/src/old/hunt_optimizer/ui/MethodsComponent.css b/src/old/hunt_optimizer/ui/MethodsComponent.css deleted file mode 100644 index 32132074..00000000 --- a/src/old/hunt_optimizer/ui/MethodsComponent.css +++ /dev/null @@ -1,7 +0,0 @@ -.main { - flex: 1; -} - -.timepicker :global(.ant-time-picker-icon) { - display: none; -} diff --git a/src/old/hunt_optimizer/ui/MethodsComponent.tsx b/src/old/hunt_optimizer/ui/MethodsComponent.tsx deleted file mode 100644 index 063dba4e..00000000 --- a/src/old/hunt_optimizer/ui/MethodsComponent.tsx +++ /dev/null @@ -1,147 +0,0 @@ -import { TimePicker } from "antd"; -import { observer } from "mobx-react"; -import moment, { Moment } from "moment"; -import React, { Component, ReactNode } from "react"; -import { AutoSizer, Index, SortDirection } from "react-virtualized"; -import { hunt_method_store } from "../stores/HuntMethodStore"; -import { BigTable, Column, ColumnSort } from "../../core/ui/BigTable"; -import styles from "./MethodsComponent.css"; -import { Episode } from "../../../core/data_formats/parsing/quest/Episode"; -import { - ENEMY_NPC_TYPES, - npc_data, - NpcType, -} from "../../../core/data_formats/parsing/quest/npc_types"; -import { HuntMethod } from "../domain"; - -@observer -export class MethodsComponent extends Component { - static columns: Column[] = (() => { - // Standard columns. - const columns: Column[] = [ - { - key: "name", - name: "Method", - width: 250, - cell_renderer: method => method.name, - sortable: true, - }, - { - key: "episode", - name: "Ep.", - width: 34, - cell_renderer: method => Episode[method.episode], - sortable: true, - }, - { - key: "time", - name: "Time", - width: 50, - cell_renderer: method => , - class_name: "integrated", - sortable: true, - }, - ]; - - // One column per enemy type. - for (const enemy_type of ENEMY_NPC_TYPES) { - columns.push({ - key: NpcType[enemy_type], - name: npc_data(enemy_type).name, - width: 75, - cell_renderer: method => { - const count = method.enemy_counts.get(enemy_type); - return count == null ? "" : count.toString(); - }, - class_name: "number", - sortable: true, - }); - } - - return columns; - })(); - - render(): ReactNode { - const methods = hunt_method_store.methods.current.value; - - return ( -
- - {({ width, height }) => ( - - width={width} - height={height} - row_count={methods.length} - columns={MethodsComponent.columns} - fixed_column_count={3} - record={this.record} - sort={this.sort} - update_trigger={hunt_method_store.methods.current.value} - /> - )} - -
- ); - } - - private record = ({ index }: Index) => { - return hunt_method_store.methods.current.value[index]; - }; - - private sort = (sorts: ColumnSort[]) => { - const methods = hunt_method_store.methods.current.value.slice(); - - methods.sort((a, b) => { - for (const { column, direction } of sorts) { - let cmp = 0; - - if (column.key === "name") { - cmp = a.name.localeCompare(b.name); - } else if (column.key === "episode") { - cmp = a.episode - b.episode; - } else if (column.key === "time") { - cmp = a.time - b.time; - } else if (column.key) { - const type = (NpcType as any)[column.key]; - - if (type) { - cmp = (a.enemy_counts.get(type) || 0) - (b.enemy_counts.get(type) || 0); - } - } - - if (cmp !== 0) { - return direction === SortDirection.ASC ? cmp : -cmp; - } - } - - return 0; - }); - - hunt_method_store.methods.current.value = methods; - }; -} - -@observer -class TimeComponent extends React.Component<{ method: HuntMethod }> { - render(): ReactNode { - const time = this.props.method.time; - const hour = Math.floor(time); - const minute = Math.round(60 * (time - hour)); - - return ( - } - onChange={this.change} - /> - ); - } - - private change = (time: Moment) => { - this.props.method.user_time = time.hour() + time.minute() / 60; - }; -}