mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-04 22:58:29 +08:00
The hunt method tables can be sorted again. There's no visual feedback in the table headers though.
This commit is contained in:
parent
9906ea88a9
commit
d9b6c7015a
@ -12,12 +12,14 @@ import Logger = require("js-logger");
|
||||
const logger = Logger.get("core/gui/Table");
|
||||
|
||||
export type Column<T> = {
|
||||
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<T> = {
|
||||
};
|
||||
};
|
||||
|
||||
export enum SortDirection {
|
||||
Asc,
|
||||
Desc,
|
||||
}
|
||||
|
||||
export type TableOptions<T> = WidgetOptions & {
|
||||
values: ListProperty<T>;
|
||||
columns: Column<T>[];
|
||||
sort?(columns: Column<T>): void;
|
||||
sort?(sort_columns: { column: Column<T>; direction: SortDirection }[]): void;
|
||||
};
|
||||
|
||||
export class Table<T> extends Widget<HTMLTableElement> {
|
||||
@ -44,6 +51,8 @@ export class Table<T> extends Widget<HTMLTableElement> {
|
||||
this.values = options.values;
|
||||
this.columns = options.columns;
|
||||
|
||||
let sort_columns: { column: Column<T>; direction: SortDirection }[] = [];
|
||||
|
||||
const thead_element = el.thead();
|
||||
const header_tr_element = el.tr();
|
||||
|
||||
@ -51,8 +60,11 @@ export class Table<T> extends Widget<HTMLTableElement> {
|
||||
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<T> extends Widget<HTMLTableElement> {
|
||||
}),
|
||||
);
|
||||
|
||||
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);
|
||||
|
@ -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<T extends HTMLElement>(
|
||||
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[]
|
||||
|
@ -194,6 +194,17 @@ export class SimpleListProperty<T> extends AbstractProperty<T[]>
|
||||
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
|
||||
|
@ -14,4 +14,6 @@ export interface WritableListProperty<T> extends ListProperty<T>, WritableProper
|
||||
remove(...values: T[]): void;
|
||||
|
||||
clear(): void;
|
||||
|
||||
sort(compare: (a: T, b: T) => number): void;
|
||||
}
|
||||
|
@ -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();
|
||||
|
@ -1,7 +0,0 @@
|
||||
.main {
|
||||
flex: 1;
|
||||
}
|
||||
|
||||
.timepicker :global(.ant-time-picker-icon) {
|
||||
display: none;
|
||||
}
|
@ -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<HuntMethod>[] = (() => {
|
||||
// Standard columns.
|
||||
const columns: Column<HuntMethod>[] = [
|
||||
{
|
||||
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 => <TimeComponent method={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 (
|
||||
<section className={styles.main}>
|
||||
<AutoSizer>
|
||||
{({ width, height }) => (
|
||||
<BigTable<HuntMethod>
|
||||
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}
|
||||
/>
|
||||
)}
|
||||
</AutoSizer>
|
||||
</section>
|
||||
);
|
||||
}
|
||||
|
||||
private record = ({ index }: Index) => {
|
||||
return hunt_method_store.methods.current.value[index];
|
||||
};
|
||||
|
||||
private sort = (sorts: ColumnSort<HuntMethod>[]) => {
|
||||
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 (
|
||||
<TimePicker
|
||||
className={styles.timepicker}
|
||||
value={moment({ hour, minute })}
|
||||
format="HH:mm"
|
||||
size="small"
|
||||
allowClear={false}
|
||||
suffixIcon={<span />}
|
||||
onChange={this.change}
|
||||
/>
|
||||
);
|
||||
}
|
||||
|
||||
private change = (time: Moment) => {
|
||||
this.props.method.user_time = time.hour() + time.minute() / 60;
|
||||
};
|
||||
}
|
Loading…
Reference in New Issue
Block a user