Hunt methods are now sortable.

This commit is contained in:
Daan Vanden Bosch 2019-06-22 17:05:42 +02:00
parent 5f7a4d5c1d
commit fe3859b782
5 changed files with 115 additions and 5 deletions

View File

@ -58,6 +58,10 @@ export class Loadable<T> {
return this._value; return this._value;
} }
set value(value: T) {
this._value = value;
}
/** /**
* This property returns valid data as soon as possible. * This property returns valid data as soon as possible.
* If the Loadable is uninitialized a data load will be triggered, otherwise the current value will be returned. * If the Loadable is uninitialized a data load will be triggered, otherwise the current value will be returned.

View File

@ -37,6 +37,7 @@ class HuntMethodStore {
} }
// Filter out some quests. // Filter out some quests.
/* eslint-disable no-fallthrough */
switch (quest.id) { switch (quest.id) {
// The following quests are left out because their enemies don't drop anything. // The following quests are left out because their enemies don't drop anything.
case 31: // Black Paper's Dangerous Deal case 31: // Black Paper's Dangerous Deal

View File

@ -29,11 +29,20 @@
} }
.DataTable-header { .DataTable-header {
user-select: none;
background-color: lighten(@component-background, 12%); background-color: lighten(@component-background, 12%);
font-weight: bold; font-weight: bold;
& .DataTable-cell { & .DataTable-cell {
border-right: solid 1px @border-color-base; border-right: solid 1px @border-color-base;
&.sortable {
cursor: pointer;
}
& .DataTable-sort-indictator {
fill: currentColor;
}
} }
} }

View File

@ -1,8 +1,9 @@
import React, { ReactNode } from "react"; import React, { ReactNode } from "react";
import { GridCellRenderer, Index, MultiGrid } from "react-virtualized"; import { GridCellRenderer, Index, MultiGrid, SortDirectionType, SortDirection } from "react-virtualized";
import "./BigTable.less"; import "./BigTable.less";
export type Column<T> = { export type Column<T> = {
key?: string,
name: string, name: string,
width: number, width: number,
cellRenderer: (record: T) => ReactNode, cellRenderer: (record: T) => ReactNode,
@ -13,8 +14,11 @@ export type Column<T> = {
* "number" and "integrated" have special meaning. * "number" and "integrated" have special meaning.
*/ */
className?: string, className?: string,
sortable?: boolean
} }
export type ColumnSort<T> = { column: Column<T>, direction: SortDirectionType }
/** /**
* A table with a fixed header. Optionally has fixed columns and a footer. * A table with a fixed header. Optionally has fixed columns and a footer.
* Uses windowing to support large amounts of rows and columns. * Uses windowing to support large amounts of rows and columns.
@ -33,8 +37,11 @@ export class BigTable<T> extends React.Component<{
/** /**
* When this changes, the DataTable will re-render. * When this changes, the DataTable will re-render.
*/ */
updateTrigger?: any updateTrigger?: any,
sort?: (sortColumns: Array<ColumnSort<T>>) => void
}> { }> {
private sortColumns = new Array<ColumnSort<T>>();
render() { render() {
return ( return (
<div <div
@ -68,6 +75,7 @@ export class BigTable<T> extends React.Component<{
private cellRenderer: GridCellRenderer = ({ columnIndex, rowIndex, style }) => { private cellRenderer: GridCellRenderer = ({ columnIndex, rowIndex, style }) => {
const column = this.props.columns[columnIndex]; const column = this.props.columns[columnIndex];
let cell: ReactNode; let cell: ReactNode;
let sortIndicator: ReactNode;
let title: string | undefined; let title: string | undefined;
const classes = ['DataTable-cell']; const classes = ['DataTable-cell'];
@ -78,6 +86,30 @@ export class BigTable<T> extends React.Component<{
if (rowIndex === 0) { if (rowIndex === 0) {
// Header row // Header row
cell = title = column.name; cell = title = column.name;
if (column.sortable) {
classes.push('sortable');
const sort = this.sortColumns[0];
if (sort && sort.column === column) {
if (sort.direction === SortDirection.ASC) {
sortIndicator = (
<svg className="DataTable-sort-indictator" width="18" height="18" viewBox="0 0 24 24">
<path d="M7 14l5-5 5 5z"></path>
<path d="M0 0h24v24H0z" fill="none"></path>
</svg>
);
} else {
sortIndicator = (
<svg className="DataTable-sort-indictator" width="18" height="18" viewBox="0 0 24 24">
<path d="M7 10l5 5 5-5z"></path>
<path d="M0 0h24v24H0z" fill="none"></path>
</svg>
);
}
}
}
} else { } else {
// Record or footer row // Record or footer row
if (column.className) { if (column.className) {
@ -105,17 +137,39 @@ export class BigTable<T> extends React.Component<{
classes.push('custom'); classes.push('custom');
} }
const onClick = rowIndex === 0 && column.sortable
? () => this.headerClicked(column)
: undefined;
return ( return (
<div <div
className={classes.join(' ')} className={classes.join(' ')}
key={`${columnIndex}, ${rowIndex}`} key={`${columnIndex}, ${rowIndex}`}
style={style} style={style}
title={title} title={title}
onClick={onClick}
> >
{typeof cell === 'string' ? ( {typeof cell === 'string' ? (
<span className="DataTable-cell-text">{cell}</span> <span className="DataTable-cell-text">{cell}</span>
) : cell} ) : cell}
{sortIndicator}
</div> </div>
); );
} }
private headerClicked = (column: Column<T>) => {
const oldIndex = this.sortColumns.findIndex(sc => sc.column === column);
let old = oldIndex === -1 ? undefined : this.sortColumns.splice(oldIndex, 1)[0];
const direction = oldIndex === 0 && old!.direction === SortDirection.ASC
? SortDirection.DESC
: SortDirection.ASC
this.sortColumns.unshift({ column, direction });
this.sortColumns.splice(10);
if (this.props.sort) {
this.props.sort(this.sortColumns);
}
}
} }

View File

@ -2,11 +2,11 @@ import { TimePicker } from "antd";
import { observer } from "mobx-react"; import { observer } from "mobx-react";
import moment, { Moment } from "moment"; import moment, { Moment } from "moment";
import React from "react"; import React from "react";
import { AutoSizer, Index } from "react-virtualized"; import { AutoSizer, Index, SortDirection } from "react-virtualized";
import { Episode, HuntMethod } from "../../domain"; import { Episode, HuntMethod } from "../../domain";
import { EnemyNpcTypes } from "../../domain/NpcType"; import { EnemyNpcTypes, NpcType } from "../../domain/NpcType";
import { huntMethodStore } from "../../stores/HuntMethodStore"; import { huntMethodStore } from "../../stores/HuntMethodStore";
import { BigTable, Column } from "../BigTable"; import { BigTable, Column, ColumnSort } from "../BigTable";
import "./MethodsComponent.css"; import "./MethodsComponent.css";
@observer @observer
@ -15,26 +15,33 @@ export class MethodsComponent extends React.Component {
// Standard columns. // Standard columns.
const columns: Column<HuntMethod>[] = [ const columns: Column<HuntMethod>[] = [
{ {
key: 'name',
name: 'Method', name: 'Method',
width: 250, width: 250,
cellRenderer: (method) => method.name, cellRenderer: (method) => method.name,
sortable: true,
}, },
{ {
key: 'episode',
name: 'Ep.', name: 'Ep.',
width: 34, width: 34,
cellRenderer: (method) => Episode[method.episode], cellRenderer: (method) => Episode[method.episode],
sortable: true,
}, },
{ {
key: 'time',
name: 'Time', name: 'Time',
width: 50, width: 50,
cellRenderer: (method) => <TimeComponent method={method} />, cellRenderer: (method) => <TimeComponent method={method} />,
className: 'integrated', className: 'integrated',
sortable: true,
}, },
]; ];
// One column per enemy type. // One column per enemy type.
for (const enemy of EnemyNpcTypes) { for (const enemy of EnemyNpcTypes) {
columns.push({ columns.push({
key: enemy.code,
name: enemy.name, name: enemy.name,
width: 75, width: 75,
cellRenderer: (method) => { cellRenderer: (method) => {
@ -42,6 +49,7 @@ export class MethodsComponent extends React.Component {
return count == null ? '' : count.toString(); return count == null ? '' : count.toString();
}, },
className: 'number', className: 'number',
sortable: true,
}); });
} }
@ -62,6 +70,8 @@ export class MethodsComponent extends React.Component {
columns={MethodsComponent.columns} columns={MethodsComponent.columns}
fixedColumnCount={3} fixedColumnCount={3}
record={this.record} record={this.record}
sort={this.sort}
updateTrigger={huntMethodStore.methods.current.value}
/> />
)} )}
</AutoSizer> </AutoSizer>
@ -72,6 +82,38 @@ export class MethodsComponent extends React.Component {
private record = ({ index }: Index) => { private record = ({ index }: Index) => {
return huntMethodStore.methods.current.value[index]; return huntMethodStore.methods.current.value[index];
} }
private sort = (sorts: ColumnSort<HuntMethod>[]) => {
const methods = huntMethodStore.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.byCode(column.key);
if (type) {
cmp = (a.enemyCounts.get(type) || 0) - (b.enemyCounts.get(type) || 0);
}
}
if (cmp !== 0) {
return direction === SortDirection.ASC ? cmp : -cmp;
}
}
return 0;
});
huntMethodStore.methods.current.value = methods;
}
} }
@observer @observer