mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-05 15:28:29 +08:00
Hunt methods are now sortable.
This commit is contained in:
parent
5f7a4d5c1d
commit
fe3859b782
@ -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.
|
||||||
|
@ -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
|
||||||
|
@ -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;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -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);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -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
|
||||||
|
Loading…
Reference in New Issue
Block a user