From fe3859b782d95c3785687b31fe15e7470de31ee1 Mon Sep 17 00:00:00 2001 From: Daan Vanden Bosch Date: Sat, 22 Jun 2019 17:05:42 +0200 Subject: [PATCH] Hunt methods are now sortable. --- src/Loadable.ts | 4 ++ src/stores/HuntMethodStore.ts | 1 + src/ui/BigTable.less | 9 ++++ src/ui/BigTable.tsx | 58 +++++++++++++++++++++- src/ui/hunt-optimizer/MethodsComponent.tsx | 48 ++++++++++++++++-- 5 files changed, 115 insertions(+), 5 deletions(-) diff --git a/src/Loadable.ts b/src/Loadable.ts index e15b75a3..e88e7d61 100644 --- a/src/Loadable.ts +++ b/src/Loadable.ts @@ -58,6 +58,10 @@ export class Loadable { return this._value; } + set value(value: T) { + this._value = value; + } + /** * 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. diff --git a/src/stores/HuntMethodStore.ts b/src/stores/HuntMethodStore.ts index e66a6062..b9da3cee 100644 --- a/src/stores/HuntMethodStore.ts +++ b/src/stores/HuntMethodStore.ts @@ -37,6 +37,7 @@ class HuntMethodStore { } // Filter out some quests. + /* eslint-disable no-fallthrough */ switch (quest.id) { // The following quests are left out because their enemies don't drop anything. case 31: // Black Paper's Dangerous Deal diff --git a/src/ui/BigTable.less b/src/ui/BigTable.less index 93335a38..ec6426b0 100644 --- a/src/ui/BigTable.less +++ b/src/ui/BigTable.less @@ -29,11 +29,20 @@ } .DataTable-header { + user-select: none; background-color: lighten(@component-background, 12%); font-weight: bold; & .DataTable-cell { border-right: solid 1px @border-color-base; + + &.sortable { + cursor: pointer; + } + + & .DataTable-sort-indictator { + fill: currentColor; + } } } diff --git a/src/ui/BigTable.tsx b/src/ui/BigTable.tsx index 0fe55fed..0109a710 100644 --- a/src/ui/BigTable.tsx +++ b/src/ui/BigTable.tsx @@ -1,8 +1,9 @@ import React, { ReactNode } from "react"; -import { GridCellRenderer, Index, MultiGrid } from "react-virtualized"; +import { GridCellRenderer, Index, MultiGrid, SortDirectionType, SortDirection } from "react-virtualized"; import "./BigTable.less"; export type Column = { + key?: string, name: string, width: number, cellRenderer: (record: T) => ReactNode, @@ -13,8 +14,11 @@ export type Column = { * "number" and "integrated" have special meaning. */ className?: string, + sortable?: boolean } +export type ColumnSort = { column: Column, direction: SortDirectionType } + /** * A table with a fixed header. Optionally has fixed columns and a footer. * Uses windowing to support large amounts of rows and columns. @@ -33,8 +37,11 @@ export class BigTable extends React.Component<{ /** * When this changes, the DataTable will re-render. */ - updateTrigger?: any + updateTrigger?: any, + sort?: (sortColumns: Array>) => void }> { + private sortColumns = new Array>(); + render() { return (
extends React.Component<{ private cellRenderer: GridCellRenderer = ({ columnIndex, rowIndex, style }) => { const column = this.props.columns[columnIndex]; let cell: ReactNode; + let sortIndicator: ReactNode; let title: string | undefined; const classes = ['DataTable-cell']; @@ -78,6 +86,30 @@ export class BigTable extends React.Component<{ if (rowIndex === 0) { // Header row 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 = ( + + + + + ); + } else { + sortIndicator = ( + + + + + ); + } + } + } } else { // Record or footer row if (column.className) { @@ -105,17 +137,39 @@ export class BigTable extends React.Component<{ classes.push('custom'); } + const onClick = rowIndex === 0 && column.sortable + ? () => this.headerClicked(column) + : undefined; + return (
{typeof cell === 'string' ? ( {cell} ) : cell} + {sortIndicator}
); } + + private headerClicked = (column: Column) => { + 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); + } + } } diff --git a/src/ui/hunt-optimizer/MethodsComponent.tsx b/src/ui/hunt-optimizer/MethodsComponent.tsx index 45219078..9329ad66 100644 --- a/src/ui/hunt-optimizer/MethodsComponent.tsx +++ b/src/ui/hunt-optimizer/MethodsComponent.tsx @@ -2,11 +2,11 @@ import { TimePicker } from "antd"; import { observer } from "mobx-react"; import moment, { Moment } from "moment"; import React from "react"; -import { AutoSizer, Index } from "react-virtualized"; +import { AutoSizer, Index, SortDirection } from "react-virtualized"; import { Episode, HuntMethod } from "../../domain"; -import { EnemyNpcTypes } from "../../domain/NpcType"; +import { EnemyNpcTypes, NpcType } from "../../domain/NpcType"; import { huntMethodStore } from "../../stores/HuntMethodStore"; -import { BigTable, Column } from "../BigTable"; +import { BigTable, Column, ColumnSort } from "../BigTable"; import "./MethodsComponent.css"; @observer @@ -15,26 +15,33 @@ export class MethodsComponent extends React.Component { // Standard columns. const columns: Column[] = [ { + key: 'name', name: 'Method', width: 250, cellRenderer: (method) => method.name, + sortable: true, }, { + key: 'episode', name: 'Ep.', width: 34, cellRenderer: (method) => Episode[method.episode], + sortable: true, }, { + key: 'time', name: 'Time', width: 50, cellRenderer: (method) => , className: 'integrated', + sortable: true, }, ]; // One column per enemy type. for (const enemy of EnemyNpcTypes) { columns.push({ + key: enemy.code, name: enemy.name, width: 75, cellRenderer: (method) => { @@ -42,6 +49,7 @@ export class MethodsComponent extends React.Component { return count == null ? '' : count.toString(); }, className: 'number', + sortable: true, }); } @@ -62,6 +70,8 @@ export class MethodsComponent extends React.Component { columns={MethodsComponent.columns} fixedColumnCount={3} record={this.record} + sort={this.sort} + updateTrigger={huntMethodStore.methods.current.value} /> )} @@ -72,6 +82,38 @@ export class MethodsComponent extends React.Component { private record = ({ index }: Index) => { return huntMethodStore.methods.current.value[index]; } + + private sort = (sorts: ColumnSort[]) => { + 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