From 1e659a42035433ae41b51465d152887d054871b1 Mon Sep 17 00:00:00 2001 From: mukayevolzhas Date: Thu, 25 Jul 2024 01:04:27 +0300 Subject: [PATCH] feat: add category bar chart --- .../remote/specs/core/studyview.spec.js | 40 ++- src/pages/studyView/StudyViewConfig.ts | 12 + src/pages/studyView/StudyViewPageStore.ts | 90 ++++- src/pages/studyView/StudyViewUtils.tsx | 44 ++- .../studyView/chartHeader/ChartHeader.tsx | 143 +++++--- src/pages/studyView/charts/ChartContainer.tsx | 43 ++- .../studyView/charts/barChart/BarChart.tsx | 23 +- .../charts/barChart/BarChartAxisLabel.tsx | 18 +- .../charts/barChart/BarChartToolTip.tsx | 12 +- .../charts/barChart/CustomBinsModal.tsx | 11 +- .../categoryBarChart/CategoryBarChart.tsx | 328 ++++++++++++++++++ .../CategoryBarChartAxisLabel.tsx | 38 ++ src/pages/studyView/tabs/SummaryTab.tsx | 84 +++++ 13 files changed, 774 insertions(+), 112 deletions(-) create mode 100644 src/pages/studyView/charts/categoryBarChart/CategoryBarChart.tsx create mode 100644 src/pages/studyView/charts/categoryBarChart/CategoryBarChartAxisLabel.tsx diff --git a/end-to-end-test/remote/specs/core/studyview.spec.js b/end-to-end-test/remote/specs/core/studyview.spec.js index ae24a5aed7c..896af9e58e0 100644 --- a/end-to-end-test/remote/specs/core/studyview.spec.js +++ b/end-to-end-test/remote/specs/core/studyview.spec.js @@ -477,7 +477,7 @@ describe('study view lgg_tcga study tests', () => { }); describe('pie chart', () => { describe('chart controls', () => { - it('the table icon should be available', () => { + it('should display the change chart option', () => { $(pieChart).waitForDisplayed({ timeout: WAIT_FOR_VISIBLE_TIMEOUT, }); @@ -486,24 +486,42 @@ describe('study view lgg_tcga study tests', () => { browser.waitUntil(() => { return $(pieChart + ' .controls').isExisting(); }, 10000); - assert($(pieChart + ' .controls .fa-table').isExisting()); + assert($(pieChart + ' .controls .fa-exchange').isExisting()); }); - }); - }); - describe('table', () => { - describe('chart controls', () => { - it('the pie icon should be available', () => { - $(table).waitForDisplayed({ + it('should display the submenu when hovering over change chart', () => { + $(pieChart).waitForDisplayed({ timeout: WAIT_FOR_VISIBLE_TIMEOUT, }); - jsApiHover(table); + jsApiHover(pieChart); + + const changeChartItem = $(pieChart + ' .dropdown-item'); + changeChartItem.moveTo(); browser.waitUntil(() => { - return $(table + ' .controls').isExisting(); + return changeChartItem + .$('.dropdown-menu.show') + .isExisting(); }, 10000); - assert($(table + ' .controls .fa-pie-chart').isExisting()); + assert(changeChartItem.$('.dropdown-menu.show').isExisting()); }); + it('should display available chart types in the submenu', () => { + $(pieChart).waitForDisplayed({ + timeout: WAIT_FOR_VISIBLE_TIMEOUT, + }); + jsApiHover(pieChart); + + const changeChartItem = $(pieChart + ' .dropdown-item'); + changeChartItem.moveTo(); + const submenuItems = changeChartItem.$$( + '.dropdown-menu.show .dropdown-item' + ); + assert(submenuItems.length > 0); + }); + }); + }); + describe('table', () => { + describe('chart controls', () => { it('table should be sorted by Freq in the default setting', () => { // we need to move to the top of the page, otherwise the offset of add chart button is calculated wrong $('body').moveTo({ xOffset: 0, yOffset: 0 }); diff --git a/src/pages/studyView/StudyViewConfig.ts b/src/pages/studyView/StudyViewConfig.ts index 166ef974a1f..afa09668b39 100644 --- a/src/pages/studyView/StudyViewConfig.ts +++ b/src/pages/studyView/StudyViewConfig.ts @@ -23,6 +23,7 @@ export type StudyViewColorTheme = { export type StudyViewThreshold = { pieToTable: number; + pieToBar: number; piePadding: number; barRatio: number; rowsInTableForOneGrid: number; @@ -61,9 +62,14 @@ export type StudyViewFrontEndConfig = { export type StudyViewConfig = StudyView & StudyViewFrontEndConfig; +export type ChangeChartOptionsMap = { + [chartType in ChartTypeEnum]?: ChartType[]; +}; + export enum ChartTypeEnum { PIE_CHART = 'PIE_CHART', BAR_CHART = 'BAR_CHART', + BAR_CATEGORICAL_CHART = 'BAR_CATEGORICAL_CHART', SURVIVAL = 'SURVIVAL', TABLE = 'TABLE', SCATTER = 'SCATTER', @@ -88,6 +94,7 @@ export enum ChartTypeEnum { export enum ChartTypeNameEnum { PIE_CHART = 'pie chart', BAR_CHART = 'bar chart', + BAR_CATEGORICAL_CHART = 'categorical bar chart', SURVIVAL = 'survival plot', TABLE = 'table', SCATTER = 'density plot', @@ -160,6 +167,7 @@ const studyViewFrontEnd = { }, thresholds: { pieToTable: 20, + pieToBar: 5, piePadding: 20, barRatio: 0.8, rowsInTableForOneGrid: 4, @@ -186,6 +194,10 @@ const studyViewFrontEnd = { w: 2, h: 1, }, + [ChartTypeEnum.BAR_CATEGORICAL_CHART]: { + w: 2, + h: 1, + }, [ChartTypeEnum.SCATTER]: { w: 2, h: 2, diff --git a/src/pages/studyView/StudyViewPageStore.ts b/src/pages/studyView/StudyViewPageStore.ts index 98d9da7d788..52369bd46a2 100644 --- a/src/pages/studyView/StudyViewPageStore.ts +++ b/src/pages/studyView/StudyViewPageStore.ts @@ -184,6 +184,7 @@ import { MutationCategorization, getChartMetaSet, getVisibleAttributes, + CategoryDataBin, } from './StudyViewUtils'; import { SingleGeneQuery } from 'shared/lib/oql/oql-parser'; import autobind from 'autobind-decorator'; @@ -592,7 +593,9 @@ export class StudyViewPageStore } }; - public isShowNAToggleVisible(dataBins: DataBin[]): boolean { + public isShowNAToggleVisible( + dataBins: DataBin[] | CategoryDataBin[] + ): boolean { return ( dataBins.length !== 0 && dataBins.some(dataBin => dataBin.specialValue === 'NA') @@ -2155,6 +2158,11 @@ export class StudyViewPageStore ChartDimension >(); + @observable public availableChartTypes = observable.map< + ChartUniqueKey, + ChartType[] + >(); + @observable public chartsType = observable.map(); private newlyAddedCharts = observable.array(); @@ -4532,7 +4540,7 @@ export class StudyViewPageStore data.forEach(item => { const uniqueKey = item.attributeId; if (this.isNewlyAdded(uniqueKey)) { - this.showAsPieChart(uniqueKey, item.counts.length); + this.showAsPieChart(uniqueKey, item.counts); this.newlyAddedCharts.remove(uniqueKey); } }); @@ -4559,7 +4567,7 @@ export class StudyViewPageStore data.forEach(item => { const uniqueKey = item.attributeId; if (this.isNewlyAdded(uniqueKey)) { - this.showAsPieChart(uniqueKey, item.counts.length); + this.showAsPieChart(uniqueKey, item.counts); this.newlyAddedCharts.remove(uniqueKey); } }); @@ -4588,7 +4596,7 @@ export class StudyViewPageStore data.forEach(item => { const uniqueKey = item.attributeId; this.unfilteredClinicalDataCountCache[uniqueKey] = item; - this.showAsPieChart(uniqueKey, item.counts.length); + this.showAsPieChart(uniqueKey, item.counts); this.newlyAddedCharts.remove(uniqueKey); }); }, @@ -5843,7 +5851,10 @@ export class StudyViewPageStore onError: () => {}, onResult: clinicalAttributes => { clinicalAttributes.forEach((obj: ClinicalAttribute) => { - if (obj.datatype === DataType.NUMBER) { + if ( + obj.datatype === DataType.NUMBER || + obj.datatype === DataType.STRING + ) { const uniqueKey = getUniqueKey(obj); let filter = getDefaultClinicalDataBinFilter(obj); @@ -7600,7 +7611,7 @@ export class StudyViewPageStore if (this.queriedPhysicalStudyIds.result.length > 1) { this.showAsPieChart( SpecialChartsUniqueKeyEnum.CANCER_STUDIES, - this.queriedPhysicalStudyIds.result.length + this.queriedPhysicalStudyIds.result ); } } @@ -7643,7 +7654,7 @@ export class StudyViewPageStore this.getTableDimensionByNumberOfRecords(data.result!.length) ); } - this.chartsType.set(attr.uniqueKey, ChartTypeEnum.TABLE); + this.chartsType.set(attr.uniqueKey, newChartType); } else { this.chartsDimension.set( attr.uniqueKey, @@ -7792,7 +7803,7 @@ export class StudyViewPageStore _.each( this.initialVisibleAttributesClinicalDataCountData.result, item => { - this.showAsPieChart(item.attributeId, item.counts.length); + this.showAsPieChart(item.attributeId, item.counts); } ); } @@ -9879,31 +9890,84 @@ export class StudyViewPageStore }); @action - showAsPieChart(uniqueKey: string, dataSize: number): void { + showAsPieChart( + uniqueKey: string, + data: ClinicalDataCount[] | string[] + ): void { + const { totalCount, naCount } = (data as ( + | ClinicalDataCount + | string + )[]).reduce( + (acc: { totalCount: number; naCount: number }, item) => { + if ( + typeof item === 'object' && + 'value' in item && + 'count' in item + ) { + acc.totalCount += item.count; + if (item.value === 'NA') { + acc.naCount += item.count; + } + } + return acc; + }, + { totalCount: 0, naCount: 0 } + ); + + const naProportion = totalCount > 0 ? naCount / totalCount : 0; + if ( shouldShowChart( this.initialFilters, - dataSize, + data.length, this.samples.result.length ) ) { this.changeChartVisibility(uniqueKey, true); if ( - dataSize > STUDY_VIEW_CONFIG.thresholds.pieToTable || - _.includes(STUDY_VIEW_CONFIG.tableAttrs, uniqueKey) + data.length > STUDY_VIEW_CONFIG.thresholds.pieToTable || + STUDY_VIEW_CONFIG.tableAttrs.includes(uniqueKey) ) { this.chartsType.set(uniqueKey, ChartTypeEnum.TABLE); this.chartsDimension.set( uniqueKey, - this.getTableDimensionByNumberOfRecords(dataSize) + this.getTableDimensionByNumberOfRecords(data.length) + ); + this.availableChartTypes.set(uniqueKey, [ + ChartTypeEnum.PIE_CHART, + ChartTypeEnum.TABLE, + ]); + } else if ( + data.length > STUDY_VIEW_CONFIG.thresholds.pieToBar || + naProportion > 0.5 + ) { + this.chartsType.set( + uniqueKey, + ChartTypeEnum.BAR_CATEGORICAL_CHART + ); + this.chartsDimension.set( + uniqueKey, + STUDY_VIEW_CONFIG.layout.dimensions[ + ChartTypeEnum.BAR_CATEGORICAL_CHART + ] ); + this.availableChartTypes.set(uniqueKey, [ + ChartTypeEnum.PIE_CHART, + ChartTypeEnum.BAR_CATEGORICAL_CHART, + ChartTypeEnum.TABLE, + ]); } else { this.chartsType.set(uniqueKey, ChartTypeEnum.PIE_CHART); this.chartsDimension.set( uniqueKey, STUDY_VIEW_CONFIG.layout.dimensions[ChartTypeEnum.PIE_CHART] ); + this.availableChartTypes.set(uniqueKey, [ + ChartTypeEnum.PIE_CHART, + ChartTypeEnum.BAR_CATEGORICAL_CHART, + ChartTypeEnum.TABLE, + ]); } } } diff --git a/src/pages/studyView/StudyViewUtils.tsx b/src/pages/studyView/StudyViewUtils.tsx index e2e8b78bc78..5ee88b20c72 100644 --- a/src/pages/studyView/StudyViewUtils.tsx +++ b/src/pages/studyView/StudyViewUtils.tsx @@ -35,7 +35,6 @@ import { } from 'cbioportal-ts-api-client'; import * as React from 'react'; import { buildCBioPortalPageUrl } from '../../shared/api/urls'; -import { BarDatum } from './charts/barChart/BarChart'; import { BinMethodOption, GenericAssayChart, @@ -226,6 +225,18 @@ export type DataBin = { start: number; }; +export type BarDatum = { + x: number; + y: number; + dataBin: DataBin; +}; + +export type CategoryDataBin = { + id: string; + count: number; + specialValue: string; +}; + export type MutationCategorization = 'MUTATED' | 'MUTATION_TYPE'; export const SPECIAL_CHARTS: ChartMetaWithDimensionAndChartType[] = [ @@ -1246,6 +1257,16 @@ export function filterIntervalBins(numericalBins: DataBin[]) { ); } +export function clinicalDataToDataBin( + data: ClinicalDataCountSummary[] +): CategoryDataBin[] { + return data.map(item => ({ + id: item.value, + count: item.count, + specialValue: `${item.value}`, + })); +} + export function calcIntervalBinValues(intervalBins: DataBin[]) { const values = intervalBins.map(dataBin => dataBin.start); @@ -1332,6 +1353,19 @@ export function generateCategoricalData( })); } +export function generateCategoricalBarData( + categoryBins: CategoryDataBin[], + startIndex: number +): BarDatum[] { + // x is not the actual data value, it is the normalized data for representation + // y is the actual count value + return categoryBins.map((dataBin: DataBin, index: number) => ({ + x: startIndex + index + 1, + y: dataBin.count, + dataBin, + })); +} + export function isLogScaleByValues(values: number[]) { return ( // empty list is not considered log scale @@ -1362,6 +1396,14 @@ export function isEveryBinDistinct(data?: DataBin[]) { ); } +export const onlyContainsNA = (element: BarDatum): boolean => { + return element.dataBin.specialValue === 'NA'; +}; + +export const doesNotContainNA = (element: BarDatum): boolean => { + return !onlyContainsNA(element); +}; + function createRangeForDataBinOrFilter( start?: number, end?: number, diff --git a/src/pages/studyView/chartHeader/ChartHeader.tsx b/src/pages/studyView/chartHeader/ChartHeader.tsx index 83a273aa57d..7e9d9e33fd8 100644 --- a/src/pages/studyView/chartHeader/ChartHeader.tsx +++ b/src/pages/studyView/chartHeader/ChartHeader.tsx @@ -8,7 +8,7 @@ import { import classnames from 'classnames'; import { action, computed, makeObservable, observable } from 'mobx'; import { observer } from 'mobx-react'; -import { ChartTypeEnum } from '../StudyViewConfig'; +import { ChartTypeEnum, ChartTypeNameEnum } from '../StudyViewConfig'; import { ChartMeta, getClinicalAttributeOverlay } from '../StudyViewUtils'; import { DataType, @@ -82,6 +82,7 @@ export interface ChartControls { showSwapAxes?: boolean; showSurvivalPlotLeftTruncationToggle?: boolean; survivalPlotLeftTruncationChecked?: boolean; + showChartChangeOptions?: boolean; } @observer @@ -90,6 +91,7 @@ export class ChartHeader extends React.Component { @observable downloadSubmenuOpen = false; @observable comparisonSubmenuOpen = false; @observable showCustomBinModal: boolean = false; + @observable showChartChangeOptions: boolean = false; private closeMenuTimeout: number | undefined = undefined; constructor(props: IChartHeaderProps) { @@ -414,6 +416,88 @@ export class ChartHeader extends React.Component { ); } + if (this.props.chartControls?.showChartChangeOptions) { + const submenuWidth = 120; + const availableCharts = + this.props.store.availableChartTypes.get( + this.props.chartMeta.uniqueKey + ) || []; + items.push( +
  • +
    + (this.showChartChangeOptions = true) + } + onMouseLeave={() => + (this.showChartChangeOptions = false) + } + > +
    +