From e25aebf26fa63240e7cd2b407ce7d29da2dbc8eb Mon Sep 17 00:00:00 2001 From: Veer Chaurasia <142890355+VeerChaurasia@users.noreply.github.com> Date: Wed, 23 Oct 2024 18:02:17 +0530 Subject: [PATCH] Feature/improve font color assignment in tree map chart component (#3954) * Improve font color assignment in tree map chart component * Update changelog --------- Co-authored-by: Thomas Kaul <4159106+dtslvr@users.noreply.github.com> --- CHANGELOG.md | 1 + .../treemap-chart/interfaces/interfaces.ts | 5 + .../treemap-chart/treemap-chart.component.ts | 177 +++++++++++------- 3 files changed, 120 insertions(+), 63 deletions(-) create mode 100644 libs/ui/src/lib/treemap-chart/interfaces/interfaces.ts diff --git a/CHANGELOG.md b/CHANGELOG.md index c5e2c109d..f5b337f55 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -14,6 +14,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 ### Changed +- Improved the font colors of the chart of the holdings tab on the home page (experimental) - Optimized the dialog sizes for mobile (full screen) - Upgraded `angular` from version `18.1.1` to `18.2.8` - Upgraded `Nx` from version `19.5.6` to `20.0.3` diff --git a/libs/ui/src/lib/treemap-chart/interfaces/interfaces.ts b/libs/ui/src/lib/treemap-chart/interfaces/interfaces.ts new file mode 100644 index 000000000..bb673ed64 --- /dev/null +++ b/libs/ui/src/lib/treemap-chart/interfaces/interfaces.ts @@ -0,0 +1,5 @@ +export interface GetColorParams { + annualizedNetPerformancePercent: number; + negativeNetPerformancePercentsRange: { max: number; min: number }; + positiveNetPerformancePercentsRange: { max: number; min: number }; +} diff --git a/libs/ui/src/lib/treemap-chart/treemap-chart.component.ts b/libs/ui/src/lib/treemap-chart/treemap-chart.component.ts index 388f75052..1acc2c925 100644 --- a/libs/ui/src/lib/treemap-chart/treemap-chart.component.ts +++ b/libs/ui/src/lib/treemap-chart/treemap-chart.component.ts @@ -33,6 +33,8 @@ import { differenceInDays, max } from 'date-fns'; import { orderBy } from 'lodash'; import { NgxSkeletonLoaderModule } from 'ngx-skeleton-loader'; +import { GetColorParams } from './interfaces/interfaces'; + const { gray, green, red } = require('open-color'); @Component({ @@ -63,7 +65,6 @@ export class GfTreemapChartComponent public constructor() { Chart.register(LinearScale, Tooltip, TreemapController, TreemapElement); } - public ngAfterViewInit() { if (this.holdings) { this.initialize(); @@ -80,6 +81,79 @@ export class GfTreemapChartComponent this.chart?.destroy(); } + private getColor({ + annualizedNetPerformancePercent, + negativeNetPerformancePercentsRange, + positiveNetPerformancePercentsRange + }: GetColorParams) { + if (Math.abs(annualizedNetPerformancePercent) === 0) { + return { + backgroundColor: gray[3], + fontColor: gray[9] + }; + } + + if (annualizedNetPerformancePercent > 0) { + let backgroundIndex: number; + const range = + positiveNetPerformancePercentsRange.max - + positiveNetPerformancePercentsRange.min; + + if ( + annualizedNetPerformancePercent >= + positiveNetPerformancePercentsRange.max - range * 0.25 + ) { + backgroundIndex = 9; + } else if ( + annualizedNetPerformancePercent >= + positiveNetPerformancePercentsRange.max - range * 0.5 + ) { + backgroundIndex = 7; + } else if ( + annualizedNetPerformancePercent >= + positiveNetPerformancePercentsRange.max - range * 0.75 + ) { + backgroundIndex = 5; + } else { + backgroundIndex = 3; + } + + return { + backgroundColor: green[backgroundIndex], + fontColor: green[backgroundIndex <= 4 ? 9 : 0] + }; + } else { + let backgroundIndex: number; + const range = + negativeNetPerformancePercentsRange.min - + negativeNetPerformancePercentsRange.max; + + if ( + annualizedNetPerformancePercent <= + negativeNetPerformancePercentsRange.min + range * 0.25 + ) { + backgroundIndex = 9; + } else if ( + annualizedNetPerformancePercent <= + negativeNetPerformancePercentsRange.min + range * 0.5 + ) { + backgroundIndex = 7; + } else if ( + annualizedNetPerformancePercent <= + negativeNetPerformancePercentsRange.min + range * 0.75 + ) { + backgroundIndex = 5; + } else { + backgroundIndex = 3; + } + + return { + backgroundColor: red[backgroundIndex], + fontColor: red[backgroundIndex <= 4 ? 9 : 0] + }; + } + } + private initialize() { this.isLoading = true; @@ -126,8 +200,8 @@ export class GfTreemapChartComponent const data: ChartConfiguration['data'] = { datasets: [ { - backgroundColor(ctx) { - let annualizedNetPerformancePercentWithCurrencyEffect = + backgroundColor: (ctx) => { + let annualizedNetPerformancePercent = getAnnualizedPerformancePercent({ daysInMarket: differenceInDays( endDate, @@ -142,74 +216,51 @@ export class GfTreemapChartComponent }).toNumber(); // Round to 2 decimal places - annualizedNetPerformancePercentWithCurrencyEffect = - Math.round( - annualizedNetPerformancePercentWithCurrencyEffect * 100 - ) / 100; - - if ( - Math.abs(annualizedNetPerformancePercentWithCurrencyEffect) === 0 - ) { - annualizedNetPerformancePercentWithCurrencyEffect = Math.abs( - annualizedNetPerformancePercentWithCurrencyEffect - ); - return gray[3]; - } else if (annualizedNetPerformancePercentWithCurrencyEffect > 0) { - const range = - positiveNetPerformancePercentsRange.max - - positiveNetPerformancePercentsRange.min; - - if ( - annualizedNetPerformancePercentWithCurrencyEffect >= - positiveNetPerformancePercentsRange.max - range * 0.25 - ) { - return green[9]; - } else if ( - annualizedNetPerformancePercentWithCurrencyEffect >= - positiveNetPerformancePercentsRange.max - range * 0.5 - ) { - return green[7]; - } else if ( - annualizedNetPerformancePercentWithCurrencyEffect >= - positiveNetPerformancePercentsRange.max - range * 0.75 - ) { - return green[5]; - } + annualizedNetPerformancePercent = + Math.round(annualizedNetPerformancePercent * 100) / 100; - return green[3]; - } else { - const range = - negativeNetPerformancePercentsRange.min - - negativeNetPerformancePercentsRange.max; - - if ( - annualizedNetPerformancePercentWithCurrencyEffect <= - negativeNetPerformancePercentsRange.min + range * 0.25 - ) { - return red[9]; - } else if ( - annualizedNetPerformancePercentWithCurrencyEffect <= - negativeNetPerformancePercentsRange.min + range * 0.5 - ) { - return red[7]; - } else if ( - annualizedNetPerformancePercentWithCurrencyEffect <= - negativeNetPerformancePercentsRange.min + range * 0.75 - ) { - return red[5]; - } + const { backgroundColor } = this.getColor({ + annualizedNetPerformancePercent, + negativeNetPerformancePercentsRange, + positiveNetPerformancePercentsRange + }); - return red[3]; - } + return backgroundColor; }, borderRadius: 4, key: 'allocationInPercentage', labels: { align: 'left', - color: ['white'], + color: (ctx) => { + let annualizedNetPerformancePercent = + getAnnualizedPerformancePercent({ + daysInMarket: differenceInDays( + endDate, + max([ + ctx.raw._data.dateOfFirstActivity ?? new Date(0), + startDate + ]) + ), + netPerformancePercentage: new Big( + ctx.raw._data.netPerformancePercentWithCurrencyEffect + ) + }).toNumber(); + + // Round to 2 decimal places + annualizedNetPerformancePercent = + Math.round(annualizedNetPerformancePercent * 100) / 100; + + const { fontColor } = this.getColor({ + annualizedNetPerformancePercent, + negativeNetPerformancePercentsRange, + positiveNetPerformancePercentsRange + }); + + return fontColor; + }, display: true, font: [{ size: 16 }, { lineHeight: 1.5, size: 14 }], - formatter(ctx) { + formatter: (ctx) => { const netPerformancePercentWithCurrencyEffect = ctx.raw._data.netPerformancePercentWithCurrencyEffect; @@ -218,7 +269,7 @@ export class GfTreemapChartComponent `${netPerformancePercentWithCurrencyEffect > 0 ? '+' : ''}${(ctx.raw._data.netPerformancePercentWithCurrencyEffect * 100).toFixed(2)}%` ]; }, - hoverColor: 'white', + hoverColor: undefined, position: 'top' }, spacing: 1,