Skip to content

Commit

Permalink
Study View: Always show selected genes on top and alterante selected …
Browse files Browse the repository at this point in the history
…rows color
  • Loading branch information
kalletlak committed Jan 13, 2020
1 parent 0a9c5d3 commit 41500f4
Show file tree
Hide file tree
Showing 4 changed files with 111 additions and 65 deletions.
12 changes: 6 additions & 6 deletions src/pages/studyView/StudyViewPageStore.ts
Original file line number Diff line number Diff line change
Expand Up @@ -1603,16 +1603,16 @@ export class StudyViewPageStore {

}

public getMutatedGenesTableFilters(): string[] {
return _.flatMap(this._mutatedGeneFilter, filter => filter.entrezGeneIds).map(entrezGeneId => getMutationUniqueKey(entrezGeneId, this.geneMapCache[entrezGeneId]));
public getMutatedGenesTableFilters() {
return _.map(this._mutatedGeneFilter, filter => filter.entrezGeneIds.map(entrezGeneId => getMutationUniqueKey(entrezGeneId, this.geneMapCache[entrezGeneId])));
}

public getFusionGenesTableFilters(): string[] {
return _.flatMap(this._fusionGeneFilter, filter => filter.entrezGeneIds).map(entrezGeneId => getMutationUniqueKey(entrezGeneId, this.geneMapCache[entrezGeneId]));
public getFusionGenesTableFilters() {
return _.map(this._fusionGeneFilter, filter => filter.entrezGeneIds.map(entrezGeneId => getMutationUniqueKey(entrezGeneId, this.geneMapCache[entrezGeneId])));
}

public getCNAGenesTableFilters(): string[] {
return _.flatMap(this._cnaGeneFilter, filter => filter.alterations).map(alteration => getCnaUniqueKey(alteration.entrezGeneId, this.geneMapCache[alteration.entrezGeneId], alteration.alteration));
public getCNAGenesTableFilters() {
return _.map(this._cnaGeneFilter, filter => filter.alterations.map(alteration => getCnaUniqueKey(alteration.entrezGeneId, this.geneMapCache[alteration.entrezGeneId], alteration.alteration)));
}

public getClinicalDataFiltersByUniqueKey(uniqueKey: string): string[] {
Expand Down
56 changes: 49 additions & 7 deletions src/pages/studyView/table/FixedHeaderTable.tsx
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
import * as React from "react";
import {Column, LazyMobXTableStore, SortDirection} from "../../../shared/components/lazyMobXTable/LazyMobXTable";
import {Column, LazyMobXTableStore, SortDirection, lazyMobXTableSort} from "../../../shared/components/lazyMobXTable/LazyMobXTable";
import {
Column as RVColumn,
SortDirection as RVSortDirection,
Expand All @@ -17,9 +17,11 @@ import {If} from 'react-if';
import autobind from 'autobind-decorator';
import {inputBoxChangeTimeoutEvent} from "../../../shared/lib/EventUtils";
import {DefaultTooltip} from "cbioportal-frontend-commons";
import { SimpleGetterLazyMobXTableApplicationDataStore } from "shared/lib/ILazyMobXTableApplicationDataStore";

export type IFixedHeaderTableProps<T> = {
columns: Column<T>[],
fixedTopRowsData?: T[];
data: T[];
sortBy?: string;
sortDirection?: SortDirection;
Expand All @@ -38,6 +40,7 @@ export type IFixedHeaderTableProps<T> = {
removeAllDisabled?:boolean;
showSelectableNumber?: boolean;
isSelectedRow?: (data: T) => boolean;
selectedRowClassName?: (data: T) => string | null;
autoFocusSearchAfterRendering?:boolean;
afterSorting?: (sortBy: string, sortDirection: SortDirection) => void;
};
Expand All @@ -47,6 +50,24 @@ const RVSDTtoStrType = {
['asc' as SortDirection]: RVSortDirection.ASC
};

export class FixedHeaderTableDataStore extends SimpleGetterLazyMobXTableApplicationDataStore<any> {

constructor(getData: () => any[], fixedTopRowsData: any[]) {
super(getData);
this.fixedTopRowsData = fixedTopRowsData;
}

@observable private fixedTopRowsData: any[];

@computed get sortedData() {
// if not defined, use default values for sortMetric and sortAscending
const sortMetric = this.sortMetric || (() => 0);
const sortAscending = this.sortAscending !== undefined ? this.sortAscending : true;

return [...this.fixedTopRowsData, ...lazyMobXTableSort(this.allData, sortMetric, sortAscending)];
}
}

@observer
export default class FixedHeaderTable<T> extends React.Component<IFixedHeaderTableProps<T>, {}> {
private _store: LazyMobXTableStore<T>;
Expand Down Expand Up @@ -78,32 +99,53 @@ export default class FixedHeaderTable<T> extends React.Component<IFixedHeaderTab
this.initDataStore();
}

componentWillReceiveProps(nextProps: any) {
componentWillReceiveProps(nextProps: IFixedHeaderTableProps<T>) {
this.updateDataStore(nextProps);
}

updateDataStore(nextProps: any) {
updateDataStore(nextProps: IFixedHeaderTableProps<T>) {
const tableDataStore = new FixedHeaderTableDataStore(
() => {
return this.props.data;
},
this.props.fixedTopRowsData || []
);
this._store.setProps({
columns: nextProps.columns,
data: nextProps.data,
dataStore: tableDataStore,
initialSortColumn: this._sortBy,
initialSortDirection: this._sortDirection
});
}

initDataStore() {
const tableDataStore = new FixedHeaderTableDataStore(
() => {
return this.props.data;
},
this.props.fixedTopRowsData || []
);
this._store = new LazyMobXTableStore<T>({
columns: this.props.columns,
data: this.props.data,
dataStore: tableDataStore,
initialSortColumn: this._sortBy,
initialSortDirection: this._sortDirection
});
}

@autobind
rowClassName({index}: any) {
rowClassName({ index }: any) {
if (index > -1 && this.isSelectedRow(this._store.dataStore.sortedFilteredData[index])) {
return classnames(styles.row, styles.highlightedRow);
const classNames: string[] = [styles.row]
if (this.props.selectedRowClassName) {
const className = this.props.selectedRowClassName(this._store.dataStore.sortedFilteredData[index]);
if (className !== null) {
classNames.push(className);
}
} else {
classNames.push(styles.highlightedRow);
}
return classnames(classNames);
} else if (index < 0) {
return styles.headerRow;
} else {
Expand Down
102 changes: 51 additions & 51 deletions src/pages/studyView/table/GeneTable.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -22,10 +22,11 @@ import {GeneCell} from "pages/studyView/table/GeneCell";
import LabeledCheckbox from "shared/components/labeledCheckbox/LabeledCheckbox";
import styles from "pages/studyView/table/tables.module.scss";
import MobxPromise from "mobxpromise";
import { stringListToIndexSet } from "cbioportal-frontend-commons";
import ifNotDefined from "shared/lib/ifNotDefined";

export type GeneTableUserSelectionWithIndex = {
uniqueKey: string;
rowIndex: number;
};

export type GeneTableRow = OncokbCancerGene & {
Expand Down Expand Up @@ -61,7 +62,7 @@ export type GeneTableProps = & {
promise: MobxPromise<GeneTableRow[]>;
width: number;
height: number;
filters: string[];
filters: string[][];
onUserSelection: (value: string[]) => void;
numOfSelectedSamples: number;
onGeneSelect: (hugoGeneSymbol: string) => void;
Expand Down Expand Up @@ -367,6 +368,36 @@ export class GeneTable extends React.Component<GeneTableProps, {}> {
return this.isFilteredByCancerGeneList ? _.filter(this.props.promise.result, data => data.isCancerGene) : (this.props.promise.result || []);
}

@computed get flatterendFilters() {
return _.flatMap(this.props.filters);
}

@computed get selectableTableData() {
if (this.flatterendFilters.length === 0) {
return this.tableData;
}
return _.filter(this.tableData, data => !this.flatterendFilters.includes(data.uniqueKey));
}

@computed
get selectedRows() {
if (this.flatterendFilters.length === 0) {
return [];
}
const order = stringListToIndexSet(this.flatterendFilters);
return _.chain(this.tableData)
.filter(data => this.flatterendFilters.includes(data.uniqueKey))
.sortBy<GeneTableRow>(data => ifNotDefined(order[data.uniqueKey], Number.POSITIVE_INFINITY))
.value();
}

@computed
get selectedRowsIndexes(): GeneTableUserSelectionWithIndex[] {
return this.selectedRows.map(row => ({
uniqueKey: row.uniqueKey
}));
}

@computed
get tableColumns() {
return this.props.columns.map(column => this.getDefaultColumnDefinition(column.columnKey, this.columnsWidth[column.columnKey], this.cellMargin[column.columnKey]));
Expand Down Expand Up @@ -401,12 +432,12 @@ export class GeneTable extends React.Component<GeneTableProps, {}> {

@autobind
isChecked(uniqueKey: string) {
return rowIsChecked(uniqueKey, this.preSelectedRows, this.selectedRows);
return rowIsChecked(uniqueKey, this.preSelectedRows, this.selectedRowsIndexes);
}

@autobind
isDisabled(uniqueKey: string) {
return rowIsDisabled(uniqueKey, this.selectedRows);
return rowIsDisabled(uniqueKey, this.selectedRowsIndexes);
}

@autobind
Expand All @@ -416,73 +447,40 @@ export class GeneTable extends React.Component<GeneTableProps, {}> {
(row: GeneTableUserSelectionWithIndex) => row.uniqueKey === uniqueKey
);
if (_.isUndefined(record)) {
let dataIndex = -1;
// definitely there is a match
const datum = _.find(
this.tableData,
(row, index: number) => {
const exist = row.uniqueKey! === uniqueKey;
if (exist) {
dataIndex = index;
}
return exist;
}
);

if (!_.isUndefined(datum)) {
this.preSelectedRows.push({
rowIndex: dataIndex,
uniqueKey: datum.uniqueKey!,
});
}
this.preSelectedRows.push({ uniqueKey: uniqueKey});
} else {
this.preSelectedRows = _.xorBy(this.preSelectedRows, [record], "rowIndex");
this.preSelectedRows = _.xorBy(this.preSelectedRows, [record], "uniqueKey");
}
}

@autobind
@action
afterSelectingRows() {
console.log(JSON.stringify(this.preSelectedRows))
this.props.onUserSelection(
this.preSelectedRows.map(row => row.uniqueKey)
);
this.preSelectedRows = [];
}

@computed
get selectedRows() {
if (this.props.filters.length === 0) {
return [];
} else {
return _.reduce(
this.tableData,
(
acc: GeneTableUserSelectionWithIndex[],
row,
index: number
) => {
if (_.includes(this.props.filters, row.uniqueKey)) {
acc.push({
rowIndex: index,
uniqueKey: row.uniqueKey
});
}
return acc;
},
[]
);
}
}

@autobind
isSelectedRow(data: GeneTableRow) {
return !_.isUndefined(
_.find(_.union(this.selectedRows, this.preSelectedRows), function (row) {
_.find(_.union(this.selectedRowsIndexes, this.preSelectedRows), function (row) {
return row.uniqueKey === data.uniqueKey;
})
);
}

@autobind
selectedRowClassName(data: GeneTableRow) {
const index = _.findIndex(this.props.filters, filter => filter.includes(data.uniqueKey));
if (index === -1) {
return null;
}
return index % 2 === 0 ? styles.highlightedEvenRow : styles.highlightedOddRow;
}

@autobind
@action
afterSorting(sortBy: GeneTableColumnKey, sortDirection: SortDirection) {
Expand All @@ -498,14 +496,16 @@ export class GeneTable extends React.Component<GeneTableProps, {}> {
<GeneTableComponent
width={this.props.width}
height={this.props.height}
data={this.tableData}
data={this.selectableTableData}
columns={this.tableColumns}
showSelectSamples={true && this.preSelectedRows.length > 0}
isSelectedRow={this.isSelectedRow}
afterSelectingRows={this.afterSelectingRows}
sortBy={this.sortBy}
sortDirection={this.sortDirection}
afterSorting={this.afterSorting}
fixedTopRowsData={this.selectedRows}
selectedRowClassName={this.selectedRowClassName}
/>
)}
{this.props.genePanelCache ? (
Expand Down
6 changes: 5 additions & 1 deletion src/pages/studyView/table/tables.module.scss
Original file line number Diff line number Diff line change
Expand Up @@ -67,10 +67,14 @@ $study-view-table-icon-size: 10px;
background-color: #ffffff;
}

.highlightedRow {
.highlightedEvenRow, .highlightedRow {
background-color: #d3d3d3;
}

.highlightedOddRow {
background-color: #e0e0e0;
}

.cancerGeneIcon {
margin-right: 5px;
}
Expand Down

0 comments on commit 41500f4

Please sign in to comment.