Skip to content

Commit

Permalink
Add legend with values + trend indicator data
Browse files Browse the repository at this point in the history
  • Loading branch information
michaelnesen committed Dec 5, 2023
1 parent 90d2535 commit 30ba453
Show file tree
Hide file tree
Showing 15 changed files with 271 additions and 19 deletions.
67 changes: 54 additions & 13 deletions packages/polaris-viz/src/components/DonutChart/Chart.tsx
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
import {Fragment, useState} from 'react';
import {Fragment, useMemo, useState} from 'react';
import {pie} from 'd3-shape';
import {
clamp,
Expand All @@ -16,6 +16,7 @@ import type {
Direction,
} from '@shopify/polaris-viz-core';

import {getTrendIndicatorData} from '../../utilities/getTrendIndicatorData';
import {getContainerAlignmentForLegend} from '../../utilities';
import {estimateLegendItemWidth} from '../Legend';
import type {ComparisonMetricProps} from '../ComparisonMetric';
Expand All @@ -27,14 +28,15 @@ import {
} from '../../hooks';
import {Arc} from '../Arc';
import type {
ColorVisionInteractionMethods,
LegendPosition,
RenderInnerValueContent,
RenderLegendContent,
} from '../../types';
import {ChartSkeleton} from '../../components/ChartSkeleton';

import styles from './DonutChart.scss';
import {InnerValue} from './components';
import {InnerValue, LegendValues} from './components';

const ERROR_ANIMATION_PADDING = 40;
const FULL_CIRCLE = Math.PI * 2;
Expand All @@ -46,6 +48,7 @@ export interface ChartProps {
labelFormatter: LabelFormatter;
legendPosition: LegendPosition;
showLegend: boolean;
showLegendValues: boolean;
state: ChartState;
theme: string;
accessibilityLabel?: string;
Expand All @@ -63,6 +66,7 @@ export function Chart({
labelFormatter,
legendPosition = 'right',
showLegend,
showLegendValues,
state,
theme,
accessibilityLabel = '',
Expand Down Expand Up @@ -92,22 +96,34 @@ export function Chart({
? 'vertical'
: 'horizontal';

const longestLegendWidth = data.reduce((previous, current) => {
const estimatedLegendWidth = estimateLegendItemWidth(
current.name ?? '',
characterWidths,
);

if (estimatedLegendWidth > previous) {
return estimatedLegendWidth;
const maxTrendIndicatorWidth = data.reduce((maxWidth, {metadata}) => {
if (!metadata?.trend) {
return maxWidth;
}

return previous;
const {trendIndicatorWidth} = getTrendIndicatorData(metadata.trend);

return Math.max(maxWidth, trendIndicatorWidth);
}, 0);

const longestLegendWidth = useMemo(() => {
return data.reduce((previous, current) => {
const estimatedLegendWidth = estimateLegendItemWidth(
`${current.name ?? ''} ${current.data[0].value} `,
characterWidths,
);

if (estimatedLegendWidth > previous) {
return estimatedLegendWidth;
}

return previous;
}, 0);
}, [characterWidths, data]);

const maxLegendWidth =
legendDirection === 'vertical'
? Math.min(
? Math.max(
longestLegendWidth,
dimensions.width * MAX_LEGEND_WIDTH_PERCENTAGE,
)
Expand Down Expand Up @@ -167,6 +183,27 @@ export function Chart({
const containerAlignmentStyle =
getContainerAlignmentForLegend(legendPosition);

const renderLegendContentWithValues = ({
getColorVisionStyles,
getColorVisionEventAttrs,
}: ColorVisionInteractionMethods) => {
return (
<LegendValues
data={data}
totalValue={totalValue}
maxTrendIndicatorWidth={maxTrendIndicatorWidth}
labelFormatter={labelFormatter}
getColorVisionStyles={getColorVisionStyles}
getColorVisionEventAttrs={getColorVisionEventAttrs}
/>
);
};

const shouldRenderLegendContentWithValues =
showLegendValues &&
!renderLegendContent &&
(legendPosition === 'right' || legendPosition === 'left');

return (
<div className={styles.DonutWrapper} style={containerAlignmentStyle}>
<div className={styles.Donut}>
Expand Down Expand Up @@ -258,7 +295,11 @@ export function Chart({
direction={legendDirection}
position={legendPosition}
maxWidth={maxLegendWidth}
renderLegendContent={renderLegendContent}
renderLegendContent={
shouldRenderLegendContentWithValues
? renderLegendContentWithValues
: renderLegendContent
}
/>
)}
</div>
Expand Down
5 changes: 5 additions & 0 deletions packages/polaris-viz/src/components/DonutChart/DonutChart.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -13,10 +13,13 @@ import type {
} from '../../types';

import {Chart} from './Chart';
import type {DonutChartDataSeries} from './types';

export type DonutChartProps = {
data: DonutChartDataSeries[];
comparisonMetric?: ComparisonMetricProps;
showLegend?: boolean;
showLegendValues?: boolean;
labelFormatter?: LabelFormatter;
legendFullWidth?: boolean;
legendPosition?: LegendPosition;
Expand All @@ -32,6 +35,7 @@ export function DonutChart(props: DonutChartProps) {
theme = defaultTheme,
comparisonMetric,
showLegend = true,
showLegendValues = false,
labelFormatter = (value) => `${value}`,
legendFullWidth,
legendPosition = 'left',
Expand Down Expand Up @@ -61,6 +65,7 @@ export function DonutChart(props: DonutChartProps) {
labelFormatter={labelFormatter}
comparisonMetric={comparisonMetric}
showLegend={showLegend}
showLegendValues={showLegendValues}
legendFullWidth={legendFullWidth}
legendPosition={legendPosition}
renderInnerValueContent={renderInnerValueContent}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -50,7 +50,9 @@ export function InnerValue({
);

const activeValueExists = activeValue !== null && activeValue !== undefined;
const valueToDisplay = activeValueExists ? activeValue : animatedTotalValue;
const valueToDisplay = activeValueExists
? labelFormatter(activeValue)
: animatedTotalValue;

const innerContent = renderInnerValueContent?.({
activeValue,
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
.Table {
min-width: 200px;
width: 100%;
padding-left: 0;
border-collapse: separate;
border-spacing: 0 10px;

.values {
font-size: 12px;
line-height: 16px;
}

.alignLeft {
text-align: left;
}

.alignRight {
text-align: right;
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,84 @@
import type {ColorVisionInteractionMethods, DataSeries} from 'index';
import type {LabelFormatter} from '@shopify/polaris-viz-core';
import {useTheme} from '@shopify/polaris-viz-core';

import {TrendIndicator} from '../../../TrendIndicator';
import {SquareColorPreview} from '../../../../components/SquareColorPreview';

import styles from './LegendValues.scss';

interface LegendContentProps {
data: DataSeries[];
totalValue: number;
maxTrendIndicatorWidth: number;
labelFormatter: LabelFormatter;
getColorVisionStyles: ColorVisionInteractionMethods['getColorVisionStyles'];
getColorVisionEventAttrs: ColorVisionInteractionMethods['getColorVisionEventAttrs'];
}

export function LegendValues({
data,
totalValue,
maxTrendIndicatorWidth,
labelFormatter,
getColorVisionStyles,
getColorVisionEventAttrs,
}: LegendContentProps) {
const selectedTheme = useTheme();

return (
<table className={styles.Table}>
{data.map(({name, data, metadata}, index) => {
const value = data[0].value || 0;

return (
<tr
key={name}
style={{
...getColorVisionStyles(index),
}}
{...getColorVisionEventAttrs(index)}
>
<td>
<SquareColorPreview
color={selectedTheme.seriesColors.upToEight[index]}
/>
</td>

<td style={{paddingLeft: '4px'}}>
<span
style={{
color: selectedTheme.legend.labelColor,
}}
className={styles.Name}
>
{name}
</span>
</td>

<td style={{paddingLeft: '30px'}} />

<td className={styles.alignRight}>
<span
style={{
color: selectedTheme.legend.labelColor,
}}
>
{labelFormatter(value)}
</span>
</td>

<td
className={styles.alignLeft}
style={{minWidth: maxTrendIndicatorWidth}}
>
<span>
{metadata?.trend && <TrendIndicator {...metadata.trend} />}
</span>
</td>
</tr>
);
})}
</table>
);
}
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
export {LegendValues} from './LegendValues';
Original file line number Diff line number Diff line change
@@ -1 +1,2 @@
export {InnerValue} from './InnerValue';
export {LegendValues} from './LegendValues';
Original file line number Diff line number Diff line change
@@ -0,0 +1,59 @@
import type {Story} from '@storybook/react';

export {META as default} from './meta';

import type {DonutChartProps} from '../../DonutChart';

import {DEFAULT_DATA, DEFAULT_PROPS, Template} from './data';

export const WithLegendValues: Story<DonutChartProps> = Template.bind({});

WithLegendValues.args = {
...DEFAULT_PROPS,
showLegend: true,
showLegendValues: true,
labelFormatter: (value) => `$${value}`,
data: [
{
name: 'Shopify Payments',
data: [{key: 'april - march', value: 50000}],
metadata: {
trend: {
value: '5%',
},
},
},
{
name: 'Paypal',
data: [{key: 'april - march', value: 25000}],
metadata: {
trend: {
value: '50%',
direction: 'downward',
trend: 'negative',
},
},
},
{
name: 'Other',
data: [{key: 'april - march', value: 10000}],
metadata: {
trend: {
value: '100%',
direction: 'upward',
trend: 'positive',
},
},
},
{
name: 'Amazon Pay',
data: [{key: 'april - march', value: 5000}],
metadata: {
trend: {
direction: 'upward',
trend: 'positive',
},
},
},
],
};
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,8 @@ import {
LEGEND_FULL_WIDTH_ARGS,
LEGEND_POSITION_ARGS,
RENDER_LEGEND_CONTENT_ARGS,
SHOW_LEGEND_ARGS,
SHOW_LEGEND_VALUES_ARGS,
THEME_CONTROL_ARGS,
} from '../../../storybook/constants';
import type {DonutChartProps} from '../DonutChart';
Expand All @@ -29,6 +31,8 @@ export const META: Meta<DonutChartProps> = {
data: DATA_SERIES_ARGS,
legendFullWidth: LEGEND_FULL_WIDTH_ARGS,
legendPosition: LEGEND_POSITION_ARGS,
showLegend: SHOW_LEGEND_ARGS,
showLegendValues: SHOW_LEGEND_VALUES_ARGS,
renderLegendContent: RENDER_LEGEND_CONTENT_ARGS,
theme: THEME_CONTROL_ARGS,
state: CHART_STATE_CONTROL_ARGS,
Expand Down
13 changes: 13 additions & 0 deletions packages/polaris-viz/src/components/DonutChart/types.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
import type {DataSeries} from '@shopify/polaris-viz-core/src/types';

import type {TrendIndicatorProps} from '../TrendIndicator';

export type MetaDataTrendIndicator = Omit<TrendIndicatorProps, 'theme'>;

export interface MetaData {
trend?: MetaDataTrendIndicator;
}

export interface DonutChartDataSeries extends DataSeries {
metadata?: MetaData;
}
Original file line number Diff line number Diff line change
Expand Up @@ -55,12 +55,13 @@ export function useLegend({
}

const legends = data.map(({series, shape}) => {
return series.map(({name, color, isComparison}) => {
return series.map(({name, color, isComparison, metadata}) => {
return {
name: name ?? '',
color,
shape,
isComparison,
...(metadata?.trend ? {trend: metadata.trend} : {}),
};
});
});
Expand Down
Loading

0 comments on commit 30ba453

Please sign in to comment.