@codecademy/gamut 67.6.1-alpha.1b85c1.0 → 67.6.1-alpha.7ad049.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/BarChart/Bar/elements.d.ts +1392 -0
- package/dist/BarChart/Bar/elements.js +61 -0
- package/dist/BarChart/Bar/index.d.ts +1 -0
- package/dist/BarChart/Bar/index.js +2 -0
- package/dist/BarChart/BarChartProvider.d.ts +33 -0
- package/dist/BarChart/BarChartProvider.js +39 -0
- package/dist/BarChart/BarRow.d.ts +7 -0
- package/dist/BarChart/BarRow.js +213 -0
- package/dist/BarChart/GridLines.d.ts +6 -0
- package/dist/BarChart/GridLines.js +45 -0
- package/dist/BarChart/ScaleChartHeader.d.ts +17 -0
- package/dist/BarChart/ScaleChartHeader.js +58 -0
- package/dist/BarChart/index.d.ts +4 -0
- package/dist/BarChart/index.js +81 -0
- package/dist/BarChart/types.d.ts +56 -0
- package/dist/BarChart/types.js +1 -0
- package/dist/BarChart/utils/index.d.ts +23 -0
- package/dist/BarChart/utils/index.js +56 -0
- package/dist/ConnectedForm/ConnectedFormGroup.d.ts +0 -5
- package/dist/ConnectedForm/ConnectedFormGroup.js +1 -1
- package/dist/Form/elements/FormGroupLabel.js +2 -8
- package/dist/GridForm/GridFormInputGroup/__fixtures__/renderers.d.ts +0 -4
- package/dist/GridForm/types.d.ts +0 -5
- package/dist/Tip/InfoTip/InfoTipButton.js +2 -5
- package/dist/Tip/InfoTip/elements.d.ts +12 -0
- package/dist/Tip/InfoTip/elements.js +9 -0
- package/dist/Tip/InfoTip/index.d.ts +0 -13
- package/dist/Tip/InfoTip/index.js +66 -36
- package/dist/Tip/__tests__/helpers.d.ts +26 -5
- package/dist/Tip/shared/FloatingTip.js +3 -3
- package/dist/Tip/shared/InlineTip.js +1 -4
- package/dist/Tip/shared/types.d.ts +1 -1
- package/dist/index.d.ts +1 -0
- package/dist/index.js +1 -0
- package/package.json +2 -2
|
@@ -0,0 +1,61 @@
|
|
|
1
|
+
import _styled from "@emotion/styled/base";
|
|
2
|
+
import { css } from '@codecademy/gamut-styles';
|
|
3
|
+
import { motion } from 'framer-motion';
|
|
4
|
+
import { Box } from '../../Box';
|
|
5
|
+
export const minBarWidth = 8;
|
|
6
|
+
const baseBarStyles = {
|
|
7
|
+
alignItems: 'center',
|
|
8
|
+
height: '100%',
|
|
9
|
+
display: 'flex',
|
|
10
|
+
position: 'absolute',
|
|
11
|
+
left: 0,
|
|
12
|
+
borderRadius: 'inherit'
|
|
13
|
+
};
|
|
14
|
+
|
|
15
|
+
/**
|
|
16
|
+
* Animated bar element for background/total value display
|
|
17
|
+
*/
|
|
18
|
+
export const BackgroundBar = /*#__PURE__*/_styled(motion.create(Box), {
|
|
19
|
+
target: "e1lnm2ux2",
|
|
20
|
+
label: "BackgroundBar"
|
|
21
|
+
})(css({
|
|
22
|
+
...baseBarStyles,
|
|
23
|
+
border: 1,
|
|
24
|
+
borderColor: 'border-primary'
|
|
25
|
+
}), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9CYXJDaGFydC9CYXIvZWxlbWVudHMudHN4Il0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQW9CNkIiLCJmaWxlIjoiLi4vLi4vLi4vc3JjL0JhckNoYXJ0L0Jhci9lbGVtZW50cy50c3giLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBjc3MgfSBmcm9tICdAY29kZWNhZGVteS9nYW11dC1zdHlsZXMnO1xuaW1wb3J0IHN0eWxlZCBmcm9tICdAZW1vdGlvbi9zdHlsZWQnO1xuaW1wb3J0IHsgbW90aW9uIH0gZnJvbSAnZnJhbWVyLW1vdGlvbic7XG5cbmltcG9ydCB7IEJveCB9IGZyb20gJy4uLy4uL0JveCc7XG5cbmV4cG9ydCBjb25zdCBtaW5CYXJXaWR0aCA9IDg7XG5cbmNvbnN0IGJhc2VCYXJTdHlsZXMgPSB7XG4gIGFsaWduSXRlbXM6ICdjZW50ZXInLFxuICBoZWlnaHQ6ICcxMDAlJyxcbiAgZGlzcGxheTogJ2ZsZXgnLFxuICBwb3NpdGlvbjogJ2Fic29sdXRlJyxcbiAgbGVmdDogMCxcbiAgYm9yZGVyUmFkaXVzOiAnaW5oZXJpdCcsXG59IGFzIGNvbnN0O1xuXG4vKipcbiAqIEFuaW1hdGVkIGJhciBlbGVtZW50IGZvciBiYWNrZ3JvdW5kL3RvdGFsIHZhbHVlIGRpc3BsYXlcbiAqL1xuZXhwb3J0IGNvbnN0IEJhY2tncm91bmRCYXIgPSBzdHlsZWQobW90aW9uLmNyZWF0ZShCb3gpKShcbiAgY3NzKHtcbiAgICAuLi5iYXNlQmFyU3R5bGVzLFxuICAgIGJvcmRlcjogMSxcbiAgICBib3JkZXJDb2xvcjogJ2JvcmRlci1wcmltYXJ5JyxcbiAgfSlcbik7XG5cbi8qKlxuICogRm9yZWdyb3VuZCBiYXIgZm9yIHN0YWNrZWQgZGlzcGxheSAocHJvZ3Jlc3MgdmFsdWUpXG4gKiBDb2xvciBzaG91bGQgYmUgcGFzc2VkIHZpYSBiZyBwcm9wIGZyb20gcGFyZW50XG4gKi9cbmV4cG9ydCBjb25zdCBGb3JlZ3JvdW5kQmFyID0gc3R5bGVkKG1vdGlvbi5jcmVhdGUoQm94KSkoXG4gIGNzcyh7XG4gICAgLi4uYmFzZUJhclN0eWxlcyxcbiAgICBoZWlnaHQ6ICdjYWxjKDEwMCUgLSAycHgpJyxcbiAgICB0b3A6IDEsXG4gICAgekluZGV4OiAxLFxuICB9KVxuKTtcblxuLyoqXG4gKiBDb250YWluZXIgZm9yIGJhcnMgd2l0aCByZXNwb25zaXZlIGhlaWdodFxuICovXG5leHBvcnQgY29uc3QgQmFyV3JhcHBlciA9IHN0eWxlZChCb3gpKFxuICBjc3Moe1xuICAgIGRpc3BsYXk6ICdmbGV4JyxcbiAgICBvdmVyZmxvdzogJ2hpZGRlbicsXG4gICAgcG9zaXRpb246ICdyZWxhdGl2ZScsXG4gICAgYWxpZ25JdGVtczogJ2NlbnRlcicsXG4gICAgaGVpZ2h0OiB7IF86IDEyLCBzbTogMjAgfSxcbiAgICBib3JkZXJSYWRpdXM6IHsgXzogJ21kJywgc206ICd4bCcgfSxcbiAgICB3aWR0aDogJzEwMCUnLFxuICB9KVxuKTtcbiJdfQ== */");
|
|
26
|
+
|
|
27
|
+
/**
|
|
28
|
+
* Foreground bar for stacked display (progress value)
|
|
29
|
+
* Color should be passed via bg prop from parent
|
|
30
|
+
*/
|
|
31
|
+
export const ForegroundBar = /*#__PURE__*/_styled(motion.create(Box), {
|
|
32
|
+
target: "e1lnm2ux1",
|
|
33
|
+
label: "ForegroundBar"
|
|
34
|
+
})(css({
|
|
35
|
+
...baseBarStyles,
|
|
36
|
+
height: 'calc(100% - 2px)',
|
|
37
|
+
top: 1,
|
|
38
|
+
zIndex: 1
|
|
39
|
+
}), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9CYXJDaGFydC9CYXIvZWxlbWVudHMudHN4Il0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQWdDNkIiLCJmaWxlIjoiLi4vLi4vLi4vc3JjL0JhckNoYXJ0L0Jhci9lbGVtZW50cy50c3giLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBjc3MgfSBmcm9tICdAY29kZWNhZGVteS9nYW11dC1zdHlsZXMnO1xuaW1wb3J0IHN0eWxlZCBmcm9tICdAZW1vdGlvbi9zdHlsZWQnO1xuaW1wb3J0IHsgbW90aW9uIH0gZnJvbSAnZnJhbWVyLW1vdGlvbic7XG5cbmltcG9ydCB7IEJveCB9IGZyb20gJy4uLy4uL0JveCc7XG5cbmV4cG9ydCBjb25zdCBtaW5CYXJXaWR0aCA9IDg7XG5cbmNvbnN0IGJhc2VCYXJTdHlsZXMgPSB7XG4gIGFsaWduSXRlbXM6ICdjZW50ZXInLFxuICBoZWlnaHQ6ICcxMDAlJyxcbiAgZGlzcGxheTogJ2ZsZXgnLFxuICBwb3NpdGlvbjogJ2Fic29sdXRlJyxcbiAgbGVmdDogMCxcbiAgYm9yZGVyUmFkaXVzOiAnaW5oZXJpdCcsXG59IGFzIGNvbnN0O1xuXG4vKipcbiAqIEFuaW1hdGVkIGJhciBlbGVtZW50IGZvciBiYWNrZ3JvdW5kL3RvdGFsIHZhbHVlIGRpc3BsYXlcbiAqL1xuZXhwb3J0IGNvbnN0IEJhY2tncm91bmRCYXIgPSBzdHlsZWQobW90aW9uLmNyZWF0ZShCb3gpKShcbiAgY3NzKHtcbiAgICAuLi5iYXNlQmFyU3R5bGVzLFxuICAgIGJvcmRlcjogMSxcbiAgICBib3JkZXJDb2xvcjogJ2JvcmRlci1wcmltYXJ5JyxcbiAgfSlcbik7XG5cbi8qKlxuICogRm9yZWdyb3VuZCBiYXIgZm9yIHN0YWNrZWQgZGlzcGxheSAocHJvZ3Jlc3MgdmFsdWUpXG4gKiBDb2xvciBzaG91bGQgYmUgcGFzc2VkIHZpYSBiZyBwcm9wIGZyb20gcGFyZW50XG4gKi9cbmV4cG9ydCBjb25zdCBGb3JlZ3JvdW5kQmFyID0gc3R5bGVkKG1vdGlvbi5jcmVhdGUoQm94KSkoXG4gIGNzcyh7XG4gICAgLi4uYmFzZUJhclN0eWxlcyxcbiAgICBoZWlnaHQ6ICdjYWxjKDEwMCUgLSAycHgpJyxcbiAgICB0b3A6IDEsXG4gICAgekluZGV4OiAxLFxuICB9KVxuKTtcblxuLyoqXG4gKiBDb250YWluZXIgZm9yIGJhcnMgd2l0aCByZXNwb25zaXZlIGhlaWdodFxuICovXG5leHBvcnQgY29uc3QgQmFyV3JhcHBlciA9IHN0eWxlZChCb3gpKFxuICBjc3Moe1xuICAgIGRpc3BsYXk6ICdmbGV4JyxcbiAgICBvdmVyZmxvdzogJ2hpZGRlbicsXG4gICAgcG9zaXRpb246ICdyZWxhdGl2ZScsXG4gICAgYWxpZ25JdGVtczogJ2NlbnRlcicsXG4gICAgaGVpZ2h0OiB7IF86IDEyLCBzbTogMjAgfSxcbiAgICBib3JkZXJSYWRpdXM6IHsgXzogJ21kJywgc206ICd4bCcgfSxcbiAgICB3aWR0aDogJzEwMCUnLFxuICB9KVxuKTtcbiJdfQ== */");
|
|
40
|
+
|
|
41
|
+
/**
|
|
42
|
+
* Container for bars with responsive height
|
|
43
|
+
*/
|
|
44
|
+
export const BarWrapper = /*#__PURE__*/_styled(Box, {
|
|
45
|
+
target: "e1lnm2ux0",
|
|
46
|
+
label: "BarWrapper"
|
|
47
|
+
})(css({
|
|
48
|
+
display: 'flex',
|
|
49
|
+
overflow: 'hidden',
|
|
50
|
+
position: 'relative',
|
|
51
|
+
alignItems: 'center',
|
|
52
|
+
height: {
|
|
53
|
+
_: 12,
|
|
54
|
+
sm: 20
|
|
55
|
+
},
|
|
56
|
+
borderRadius: {
|
|
57
|
+
_: 'md',
|
|
58
|
+
sm: 'xl'
|
|
59
|
+
},
|
|
60
|
+
width: '100%'
|
|
61
|
+
}), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9CYXJDaGFydC9CYXIvZWxlbWVudHMudHN4Il0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQTRDMEIiLCJmaWxlIjoiLi4vLi4vLi4vc3JjL0JhckNoYXJ0L0Jhci9lbGVtZW50cy50c3giLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBjc3MgfSBmcm9tICdAY29kZWNhZGVteS9nYW11dC1zdHlsZXMnO1xuaW1wb3J0IHN0eWxlZCBmcm9tICdAZW1vdGlvbi9zdHlsZWQnO1xuaW1wb3J0IHsgbW90aW9uIH0gZnJvbSAnZnJhbWVyLW1vdGlvbic7XG5cbmltcG9ydCB7IEJveCB9IGZyb20gJy4uLy4uL0JveCc7XG5cbmV4cG9ydCBjb25zdCBtaW5CYXJXaWR0aCA9IDg7XG5cbmNvbnN0IGJhc2VCYXJTdHlsZXMgPSB7XG4gIGFsaWduSXRlbXM6ICdjZW50ZXInLFxuICBoZWlnaHQ6ICcxMDAlJyxcbiAgZGlzcGxheTogJ2ZsZXgnLFxuICBwb3NpdGlvbjogJ2Fic29sdXRlJyxcbiAgbGVmdDogMCxcbiAgYm9yZGVyUmFkaXVzOiAnaW5oZXJpdCcsXG59IGFzIGNvbnN0O1xuXG4vKipcbiAqIEFuaW1hdGVkIGJhciBlbGVtZW50IGZvciBiYWNrZ3JvdW5kL3RvdGFsIHZhbHVlIGRpc3BsYXlcbiAqL1xuZXhwb3J0IGNvbnN0IEJhY2tncm91bmRCYXIgPSBzdHlsZWQobW90aW9uLmNyZWF0ZShCb3gpKShcbiAgY3NzKHtcbiAgICAuLi5iYXNlQmFyU3R5bGVzLFxuICAgIGJvcmRlcjogMSxcbiAgICBib3JkZXJDb2xvcjogJ2JvcmRlci1wcmltYXJ5JyxcbiAgfSlcbik7XG5cbi8qKlxuICogRm9yZWdyb3VuZCBiYXIgZm9yIHN0YWNrZWQgZGlzcGxheSAocHJvZ3Jlc3MgdmFsdWUpXG4gKiBDb2xvciBzaG91bGQgYmUgcGFzc2VkIHZpYSBiZyBwcm9wIGZyb20gcGFyZW50XG4gKi9cbmV4cG9ydCBjb25zdCBGb3JlZ3JvdW5kQmFyID0gc3R5bGVkKG1vdGlvbi5jcmVhdGUoQm94KSkoXG4gIGNzcyh7XG4gICAgLi4uYmFzZUJhclN0eWxlcyxcbiAgICBoZWlnaHQ6ICdjYWxjKDEwMCUgLSAycHgpJyxcbiAgICB0b3A6IDEsXG4gICAgekluZGV4OiAxLFxuICB9KVxuKTtcblxuLyoqXG4gKiBDb250YWluZXIgZm9yIGJhcnMgd2l0aCByZXNwb25zaXZlIGhlaWdodFxuICovXG5leHBvcnQgY29uc3QgQmFyV3JhcHBlciA9IHN0eWxlZChCb3gpKFxuICBjc3Moe1xuICAgIGRpc3BsYXk6ICdmbGV4JyxcbiAgICBvdmVyZmxvdzogJ2hpZGRlbicsXG4gICAgcG9zaXRpb246ICdyZWxhdGl2ZScsXG4gICAgYWxpZ25JdGVtczogJ2NlbnRlcicsXG4gICAgaGVpZ2h0OiB7IF86IDEyLCBzbTogMjAgfSxcbiAgICBib3JkZXJSYWRpdXM6IHsgXzogJ21kJywgc206ICd4bCcgfSxcbiAgICB3aWR0aDogJzEwMCUnLFxuICB9KVxuKTtcbiJdfQ== */");
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export * from './elements';
|
|
@@ -0,0 +1,33 @@
|
|
|
1
|
+
/// <reference types="react" />
|
|
2
|
+
import { BarChartStyles } from './types';
|
|
3
|
+
export interface BarChartContextProps {
|
|
4
|
+
minRange: number;
|
|
5
|
+
maxRange: number;
|
|
6
|
+
xScale: number;
|
|
7
|
+
unit: string;
|
|
8
|
+
styleConfig: Required<BarChartStyles>;
|
|
9
|
+
animate: boolean;
|
|
10
|
+
}
|
|
11
|
+
export declare const BarChartContext: import("react").Context<BarChartContextProps>;
|
|
12
|
+
export declare const BarChartProvider: import("react").Provider<BarChartContextProps>;
|
|
13
|
+
export declare function useBarChartContext(): BarChartContextProps;
|
|
14
|
+
export interface UseBarChartOptions {
|
|
15
|
+
minRange: number;
|
|
16
|
+
maxRange: number;
|
|
17
|
+
xScale?: number;
|
|
18
|
+
unit?: string;
|
|
19
|
+
styleConfig?: BarChartStyles;
|
|
20
|
+
animate?: boolean;
|
|
21
|
+
}
|
|
22
|
+
export declare function useBarChart({ minRange, maxRange, xScale, unit, styleConfig, animate, }: UseBarChartOptions): {
|
|
23
|
+
minRange: number;
|
|
24
|
+
maxRange: number;
|
|
25
|
+
xScale: number;
|
|
26
|
+
unit: string;
|
|
27
|
+
styleConfig: {
|
|
28
|
+
textColor: "text" | "background" | "primary" | "secondary" | "danger" | "interface" | "text-secondary" | "text-accent" | "text-disabled" | "feedback-error" | "feedback-success" | "feedback-warning" | "background-primary" | "background-disabled" | "background-error" | "background-success" | "background-warning" | "background-contrast" | "background-current" | "background-selected" | "background-hover" | "shadow-primary" | "shadow-secondary" | "primary-hover" | "primary-inverse" | "secondary-hover" | "danger-hover" | "interface-hover" | "border-primary" | "border-secondary" | "border-disabled" | "border-tertiary";
|
|
29
|
+
foregroundBarColor: "text" | "background" | "primary" | "secondary" | "danger" | "interface" | "text-secondary" | "text-accent" | "text-disabled" | "feedback-error" | "feedback-success" | "feedback-warning" | "background-primary" | "background-disabled" | "background-error" | "background-success" | "background-warning" | "background-contrast" | "background-current" | "background-selected" | "background-hover" | "shadow-primary" | "shadow-secondary" | "primary-hover" | "primary-inverse" | "secondary-hover" | "danger-hover" | "interface-hover" | "border-primary" | "border-secondary" | "border-disabled" | "border-tertiary";
|
|
30
|
+
backgroundBarColor: "text" | "background" | "primary" | "secondary" | "danger" | "interface" | "text-secondary" | "text-accent" | "text-disabled" | "feedback-error" | "feedback-success" | "feedback-warning" | "background-primary" | "background-disabled" | "background-error" | "background-success" | "background-warning" | "background-contrast" | "background-current" | "background-selected" | "background-hover" | "shadow-primary" | "shadow-secondary" | "primary-hover" | "primary-inverse" | "secondary-hover" | "danger-hover" | "interface-hover" | "border-primary" | "border-secondary" | "border-disabled" | "border-tertiary";
|
|
31
|
+
};
|
|
32
|
+
animate: boolean;
|
|
33
|
+
};
|
|
@@ -0,0 +1,39 @@
|
|
|
1
|
+
import { createContext, useContext, useMemo } from 'react';
|
|
2
|
+
const defaultStyleConfig = {
|
|
3
|
+
textColor: 'text',
|
|
4
|
+
foregroundBarColor: 'feedback-warning',
|
|
5
|
+
backgroundBarColor: 'background-primary'
|
|
6
|
+
};
|
|
7
|
+
export const BarChartContext = /*#__PURE__*/createContext({
|
|
8
|
+
minRange: 0,
|
|
9
|
+
maxRange: 100,
|
|
10
|
+
xScale: 10,
|
|
11
|
+
unit: '',
|
|
12
|
+
styleConfig: defaultStyleConfig,
|
|
13
|
+
animate: false
|
|
14
|
+
});
|
|
15
|
+
BarChartContext.displayName = 'BarChartContext';
|
|
16
|
+
export const BarChartProvider = BarChartContext.Provider;
|
|
17
|
+
export function useBarChartContext() {
|
|
18
|
+
return useContext(BarChartContext);
|
|
19
|
+
}
|
|
20
|
+
export function useBarChart({
|
|
21
|
+
minRange,
|
|
22
|
+
maxRange,
|
|
23
|
+
xScale,
|
|
24
|
+
unit = '',
|
|
25
|
+
styleConfig,
|
|
26
|
+
animate = false
|
|
27
|
+
}) {
|
|
28
|
+
return useMemo(() => ({
|
|
29
|
+
minRange,
|
|
30
|
+
maxRange,
|
|
31
|
+
xScale: xScale ?? Math.ceil((maxRange - minRange) / 5),
|
|
32
|
+
unit,
|
|
33
|
+
styleConfig: {
|
|
34
|
+
...defaultStyleConfig,
|
|
35
|
+
...styleConfig
|
|
36
|
+
},
|
|
37
|
+
animate
|
|
38
|
+
}), [minRange, maxRange, xScale, unit, styleConfig, animate]);
|
|
39
|
+
}
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
/// <reference types="react" />
|
|
2
|
+
import { BarProps } from './types';
|
|
3
|
+
export interface BarRowProps extends BarProps {
|
|
4
|
+
/** Index for animation staggering */
|
|
5
|
+
index?: number;
|
|
6
|
+
}
|
|
7
|
+
export declare const BarRow: import("react").ForwardRefExoticComponent<BarRowProps & import("react").RefAttributes<HTMLDivElement | HTMLAnchorElement | HTMLButtonElement>>;
|
|
@@ -0,0 +1,213 @@
|
|
|
1
|
+
import _styled from "@emotion/styled/base";
|
|
2
|
+
function _extends() { return _extends = Object.assign ? Object.assign.bind() : function (n) { for (var e = 1; e < arguments.length; e++) { var t = arguments[e]; for (var r in t) ({}).hasOwnProperty.call(t, r) && (n[r] = t[r]); } return n; }, _extends.apply(null, arguments); }
|
|
3
|
+
import { css, styledOptions } from '@codecademy/gamut-styles';
|
|
4
|
+
import { forwardRef } from 'react';
|
|
5
|
+
import { FlexBox } from '../Box';
|
|
6
|
+
import { Text } from '../Typography';
|
|
7
|
+
import { useBarChartContext } from './BarChartProvider';
|
|
8
|
+
import { calculateBarWidth } from './utils';
|
|
9
|
+
import { BackgroundBar, BarWrapper, ForegroundBar, minBarWidth } from './Bar/elements';
|
|
10
|
+
|
|
11
|
+
// Polymorphic row wrapper styles
|
|
12
|
+
import { jsx as _jsx, jsxs as _jsxs, Fragment as _Fragment } from "react/jsx-runtime";
|
|
13
|
+
const rowBaseStyles = css({
|
|
14
|
+
display: 'flex',
|
|
15
|
+
alignItems: 'center',
|
|
16
|
+
width: '100%',
|
|
17
|
+
gap: 16,
|
|
18
|
+
py: 8,
|
|
19
|
+
px: 0,
|
|
20
|
+
bg: 'transparent',
|
|
21
|
+
border: 'none',
|
|
22
|
+
textDecoration: 'none',
|
|
23
|
+
cursor: 'inherit',
|
|
24
|
+
'&:focus': {
|
|
25
|
+
outline: 'none'
|
|
26
|
+
},
|
|
27
|
+
'&:focus-visible': {
|
|
28
|
+
outline: '2px solid',
|
|
29
|
+
outlineColor: 'primary',
|
|
30
|
+
outlineOffset: '2px'
|
|
31
|
+
}
|
|
32
|
+
});
|
|
33
|
+
const interactiveStyles = css({
|
|
34
|
+
cursor: 'pointer',
|
|
35
|
+
'&:hover': {
|
|
36
|
+
bg: 'background-hover'
|
|
37
|
+
}
|
|
38
|
+
});
|
|
39
|
+
const RowWrapper = /*#__PURE__*/_styled('div', _extends({}, {
|
|
40
|
+
target: "eew7ix72",
|
|
41
|
+
label: "RowWrapper"
|
|
42
|
+
}, styledOptions()))(rowBaseStyles, process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/BarChart/BarRow.tsx"],"names":[],"mappings":"AA6CmB","file":"../../src/BarChart/BarRow.tsx","sourcesContent":["import { css, styledOptions } from '@codecademy/gamut-styles';\nimport styled from '@emotion/styled';\nimport { forwardRef, MouseEvent, MutableRefObject } from 'react';\n\nimport { FlexBox } from '../Box';\nimport { Text } from '../Typography';\nimport { useBarChartContext } from './BarChartProvider';\nimport { BarProps } from './types';\nimport { calculateBarWidth } from './utils';\nimport {\n  BackgroundBar,\n  BarWrapper,\n  ForegroundBar,\n  minBarWidth,\n} from './Bar/elements';\n\n// Polymorphic row wrapper styles\nconst rowBaseStyles = css({\n  display: 'flex',\n  alignItems: 'center',\n  width: '100%',\n  gap: 16,\n  py: 8,\n  px: 0,\n  bg: 'transparent',\n  border: 'none',\n  textDecoration: 'none',\n  cursor: 'inherit',\n  '&:focus': {\n    outline: 'none',\n  },\n  '&:focus-visible': {\n    outline: '2px solid',\n    outlineColor: 'primary',\n    outlineOffset: '2px',\n  },\n});\n\nconst interactiveStyles = css({\n  cursor: 'pointer',\n  '&:hover': {\n    bg: 'background-hover',\n  },\n});\n\nconst RowWrapper = styled('div', styledOptions<'div'>())(rowBaseStyles);\nconst RowButton = styled('button', styledOptions<'button'>())(\n  rowBaseStyles,\n  interactiveStyles\n);\nconst RowAnchor = styled('a', styledOptions<'a'>())(\n  rowBaseStyles,\n  interactiveStyles\n);\n\nexport interface BarRowProps extends BarProps {\n  /** Index for animation staggering */\n  index?: number;\n}\n\n/**\n * Generates an accessible summary of the bar values\n */\nfunction getValuesSummary({\n  yLabel,\n  seriesOneValue,\n  seriesTwoValue,\n  unit,\n}: {\n  yLabel: string;\n  seriesOneValue: number;\n  seriesTwoValue?: number;\n  unit: string;\n}): string {\n  if (seriesTwoValue !== undefined) {\n    const gained = seriesOneValue;\n    return `${gained} ${unit} gained - now at ${seriesTwoValue} ${unit} in ${yLabel} category`;\n  }\n  return `${seriesOneValue} ${unit} in ${yLabel} category`;\n}\n\nexport const BarRow = forwardRef<\n  HTMLDivElement | HTMLButtonElement | HTMLAnchorElement,\n  BarRowProps\n>(\n  (\n    {\n      yLabel,\n      seriesOneValue,\n      seriesTwoValue,\n      icon: Icon,\n      onClick,\n      href,\n      index = 0,\n    },\n    ref\n  ) => {\n    const { maxRange, unit, styleConfig, animate } = useBarChartContext();\n\n    const isStacked = seriesTwoValue !== undefined;\n    const displayValue = isStacked ? seriesTwoValue : seriesOneValue;\n\n    // Calculate bar widths as percentages\n    const backgroundBarWidth = calculateBarWidth({\n      value: displayValue,\n      maxRange,\n    });\n\n    const foregroundBarWidth = isStacked\n      ? calculateBarWidth({\n          value: seriesOneValue,\n          maxRange,\n        })\n      : 0;\n\n    const bgWidthStr = `${Math.max(minBarWidth, backgroundBarWidth)}%`;\n    const fgWidthStr = `${Math.max(minBarWidth, foregroundBarWidth)}%`;\n\n    const valuesSummary = getValuesSummary({\n      yLabel,\n      seriesOneValue,\n      seriesTwoValue,\n      unit,\n    });\n\n    // Animation delay for staggered bar entrance\n    const animationDelay = animate ? index * 0.1 : 0;\n\n    const rowContent = (\n      <>\n        {/* Y-axis label with optional icon */}\n        <FlexBox\n          alignItems=\"center\"\n          gap={8}\n          minWidth={{ _: 80, sm: 120 }}\n          flexShrink={0}\n        >\n          {Icon && <Icon size={16} color={styleConfig.textColor} />}\n          <Text\n            variant=\"p-small\"\n            color={styleConfig.textColor}\n            truncate=\"ellipsis\"\n            truncateLines={1}\n          >\n            {yLabel}\n          </Text>\n        </FlexBox>\n\n        {/* Bar container */}\n        <FlexBox flex={1} alignItems=\"center\" position=\"relative\">\n          <BarWrapper>\n            {/* Background bar (total value in stacked, or single value in non-stacked) */}\n            <BackgroundBar\n              bg={styleConfig.backgroundBarColor}\n              initial={animate ? { width: '0%' } : undefined}\n              animate={animate ? { width: bgWidthStr } : undefined}\n              style={!animate ? { width: bgWidthStr } : undefined}\n              transition={{ duration: 0.5, delay: animationDelay }}\n              data-testid=\"background-bar\"\n            />\n\n            {/* Foreground bar (progress value in stacked mode only) */}\n            {isStacked && (\n              <ForegroundBar\n                bg={styleConfig.foregroundBarColor}\n                initial={animate ? { width: '0%' } : undefined}\n                animate={animate ? { width: fgWidthStr } : undefined}\n                style={!animate ? { width: fgWidthStr } : undefined}\n                transition={{ duration: 0.5, delay: animationDelay + 0.25 }}\n                data-testid=\"foreground-bar\"\n              />\n            )}\n          </BarWrapper>\n        </FlexBox>\n\n        {/* Value display */}\n        <FlexBox\n          alignItems=\"center\"\n          justifyContent=\"flex-end\"\n          minWidth={{ _: 40, sm: 60 }}\n          flexShrink={0}\n        >\n          <Text variant=\"p-small\" color={styleConfig.textColor}>\n            {displayValue.toLocaleString()}\n            {unit && ` ${unit}`}\n          </Text>\n        </FlexBox>\n      </>\n    );\n\n    // Polymorphic rendering based on interactivity\n    if (href) {\n      return (\n        <li>\n          <RowAnchor\n            href={href}\n            aria-label={valuesSummary}\n            ref={ref as MutableRefObject<HTMLAnchorElement>}\n          >\n            {rowContent}\n          </RowAnchor>\n        </li>\n      );\n    }\n\n    if (onClick) {\n      return (\n        <li>\n          <RowButton\n            type=\"button\"\n            onClick={onClick}\n            onKeyDown={(e) => {\n              if (e.key === 'Enter') {\n                onClick(e as unknown as MouseEvent<HTMLButtonElement>);\n              }\n            }}\n            aria-label={valuesSummary}\n            ref={ref as MutableRefObject<HTMLButtonElement>}\n          >\n            {rowContent}\n          </RowButton>\n        </li>\n      );\n    }\n\n    return (\n      <li>\n        <RowWrapper\n          aria-label={valuesSummary}\n          ref={ref as MutableRefObject<HTMLDivElement>}\n        >\n          {rowContent}\n        </RowWrapper>\n      </li>\n    );\n  }\n);\n\nBarRow.displayName = 'BarRow';\n"]} */");
|
|
43
|
+
const RowButton = /*#__PURE__*/_styled('button', _extends({}, {
|
|
44
|
+
target: "eew7ix71",
|
|
45
|
+
label: "RowButton"
|
|
46
|
+
}, styledOptions()))(rowBaseStyles, interactiveStyles, process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/BarChart/BarRow.tsx"],"names":[],"mappings":"AA8CkB","file":"../../src/BarChart/BarRow.tsx","sourcesContent":["import { css, styledOptions } from '@codecademy/gamut-styles';\nimport styled from '@emotion/styled';\nimport { forwardRef, MouseEvent, MutableRefObject } from 'react';\n\nimport { FlexBox } from '../Box';\nimport { Text } from '../Typography';\nimport { useBarChartContext } from './BarChartProvider';\nimport { BarProps } from './types';\nimport { calculateBarWidth } from './utils';\nimport {\n  BackgroundBar,\n  BarWrapper,\n  ForegroundBar,\n  minBarWidth,\n} from './Bar/elements';\n\n// Polymorphic row wrapper styles\nconst rowBaseStyles = css({\n  display: 'flex',\n  alignItems: 'center',\n  width: '100%',\n  gap: 16,\n  py: 8,\n  px: 0,\n  bg: 'transparent',\n  border: 'none',\n  textDecoration: 'none',\n  cursor: 'inherit',\n  '&:focus': {\n    outline: 'none',\n  },\n  '&:focus-visible': {\n    outline: '2px solid',\n    outlineColor: 'primary',\n    outlineOffset: '2px',\n  },\n});\n\nconst interactiveStyles = css({\n  cursor: 'pointer',\n  '&:hover': {\n    bg: 'background-hover',\n  },\n});\n\nconst RowWrapper = styled('div', styledOptions<'div'>())(rowBaseStyles);\nconst RowButton = styled('button', styledOptions<'button'>())(\n  rowBaseStyles,\n  interactiveStyles\n);\nconst RowAnchor = styled('a', styledOptions<'a'>())(\n  rowBaseStyles,\n  interactiveStyles\n);\n\nexport interface BarRowProps extends BarProps {\n  /** Index for animation staggering */\n  index?: number;\n}\n\n/**\n * Generates an accessible summary of the bar values\n */\nfunction getValuesSummary({\n  yLabel,\n  seriesOneValue,\n  seriesTwoValue,\n  unit,\n}: {\n  yLabel: string;\n  seriesOneValue: number;\n  seriesTwoValue?: number;\n  unit: string;\n}): string {\n  if (seriesTwoValue !== undefined) {\n    const gained = seriesOneValue;\n    return `${gained} ${unit} gained - now at ${seriesTwoValue} ${unit} in ${yLabel} category`;\n  }\n  return `${seriesOneValue} ${unit} in ${yLabel} category`;\n}\n\nexport const BarRow = forwardRef<\n  HTMLDivElement | HTMLButtonElement | HTMLAnchorElement,\n  BarRowProps\n>(\n  (\n    {\n      yLabel,\n      seriesOneValue,\n      seriesTwoValue,\n      icon: Icon,\n      onClick,\n      href,\n      index = 0,\n    },\n    ref\n  ) => {\n    const { maxRange, unit, styleConfig, animate } = useBarChartContext();\n\n    const isStacked = seriesTwoValue !== undefined;\n    const displayValue = isStacked ? seriesTwoValue : seriesOneValue;\n\n    // Calculate bar widths as percentages\n    const backgroundBarWidth = calculateBarWidth({\n      value: displayValue,\n      maxRange,\n    });\n\n    const foregroundBarWidth = isStacked\n      ? calculateBarWidth({\n          value: seriesOneValue,\n          maxRange,\n        })\n      : 0;\n\n    const bgWidthStr = `${Math.max(minBarWidth, backgroundBarWidth)}%`;\n    const fgWidthStr = `${Math.max(minBarWidth, foregroundBarWidth)}%`;\n\n    const valuesSummary = getValuesSummary({\n      yLabel,\n      seriesOneValue,\n      seriesTwoValue,\n      unit,\n    });\n\n    // Animation delay for staggered bar entrance\n    const animationDelay = animate ? index * 0.1 : 0;\n\n    const rowContent = (\n      <>\n        {/* Y-axis label with optional icon */}\n        <FlexBox\n          alignItems=\"center\"\n          gap={8}\n          minWidth={{ _: 80, sm: 120 }}\n          flexShrink={0}\n        >\n          {Icon && <Icon size={16} color={styleConfig.textColor} />}\n          <Text\n            variant=\"p-small\"\n            color={styleConfig.textColor}\n            truncate=\"ellipsis\"\n            truncateLines={1}\n          >\n            {yLabel}\n          </Text>\n        </FlexBox>\n\n        {/* Bar container */}\n        <FlexBox flex={1} alignItems=\"center\" position=\"relative\">\n          <BarWrapper>\n            {/* Background bar (total value in stacked, or single value in non-stacked) */}\n            <BackgroundBar\n              bg={styleConfig.backgroundBarColor}\n              initial={animate ? { width: '0%' } : undefined}\n              animate={animate ? { width: bgWidthStr } : undefined}\n              style={!animate ? { width: bgWidthStr } : undefined}\n              transition={{ duration: 0.5, delay: animationDelay }}\n              data-testid=\"background-bar\"\n            />\n\n            {/* Foreground bar (progress value in stacked mode only) */}\n            {isStacked && (\n              <ForegroundBar\n                bg={styleConfig.foregroundBarColor}\n                initial={animate ? { width: '0%' } : undefined}\n                animate={animate ? { width: fgWidthStr } : undefined}\n                style={!animate ? { width: fgWidthStr } : undefined}\n                transition={{ duration: 0.5, delay: animationDelay + 0.25 }}\n                data-testid=\"foreground-bar\"\n              />\n            )}\n          </BarWrapper>\n        </FlexBox>\n\n        {/* Value display */}\n        <FlexBox\n          alignItems=\"center\"\n          justifyContent=\"flex-end\"\n          minWidth={{ _: 40, sm: 60 }}\n          flexShrink={0}\n        >\n          <Text variant=\"p-small\" color={styleConfig.textColor}>\n            {displayValue.toLocaleString()}\n            {unit && ` ${unit}`}\n          </Text>\n        </FlexBox>\n      </>\n    );\n\n    // Polymorphic rendering based on interactivity\n    if (href) {\n      return (\n        <li>\n          <RowAnchor\n            href={href}\n            aria-label={valuesSummary}\n            ref={ref as MutableRefObject<HTMLAnchorElement>}\n          >\n            {rowContent}\n          </RowAnchor>\n        </li>\n      );\n    }\n\n    if (onClick) {\n      return (\n        <li>\n          <RowButton\n            type=\"button\"\n            onClick={onClick}\n            onKeyDown={(e) => {\n              if (e.key === 'Enter') {\n                onClick(e as unknown as MouseEvent<HTMLButtonElement>);\n              }\n            }}\n            aria-label={valuesSummary}\n            ref={ref as MutableRefObject<HTMLButtonElement>}\n          >\n            {rowContent}\n          </RowButton>\n        </li>\n      );\n    }\n\n    return (\n      <li>\n        <RowWrapper\n          aria-label={valuesSummary}\n          ref={ref as MutableRefObject<HTMLDivElement>}\n        >\n          {rowContent}\n        </RowWrapper>\n      </li>\n    );\n  }\n);\n\nBarRow.displayName = 'BarRow';\n"]} */");
|
|
47
|
+
const RowAnchor = /*#__PURE__*/_styled('a', _extends({}, {
|
|
48
|
+
target: "eew7ix70",
|
|
49
|
+
label: "RowAnchor"
|
|
50
|
+
}, styledOptions()))(rowBaseStyles, interactiveStyles, process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/BarChart/BarRow.tsx"],"names":[],"mappings":"AAkDkB","file":"../../src/BarChart/BarRow.tsx","sourcesContent":["import { css, styledOptions } from '@codecademy/gamut-styles';\nimport styled from '@emotion/styled';\nimport { forwardRef, MouseEvent, MutableRefObject } from 'react';\n\nimport { FlexBox } from '../Box';\nimport { Text } from '../Typography';\nimport { useBarChartContext } from './BarChartProvider';\nimport { BarProps } from './types';\nimport { calculateBarWidth } from './utils';\nimport {\n  BackgroundBar,\n  BarWrapper,\n  ForegroundBar,\n  minBarWidth,\n} from './Bar/elements';\n\n// Polymorphic row wrapper styles\nconst rowBaseStyles = css({\n  display: 'flex',\n  alignItems: 'center',\n  width: '100%',\n  gap: 16,\n  py: 8,\n  px: 0,\n  bg: 'transparent',\n  border: 'none',\n  textDecoration: 'none',\n  cursor: 'inherit',\n  '&:focus': {\n    outline: 'none',\n  },\n  '&:focus-visible': {\n    outline: '2px solid',\n    outlineColor: 'primary',\n    outlineOffset: '2px',\n  },\n});\n\nconst interactiveStyles = css({\n  cursor: 'pointer',\n  '&:hover': {\n    bg: 'background-hover',\n  },\n});\n\nconst RowWrapper = styled('div', styledOptions<'div'>())(rowBaseStyles);\nconst RowButton = styled('button', styledOptions<'button'>())(\n  rowBaseStyles,\n  interactiveStyles\n);\nconst RowAnchor = styled('a', styledOptions<'a'>())(\n  rowBaseStyles,\n  interactiveStyles\n);\n\nexport interface BarRowProps extends BarProps {\n  /** Index for animation staggering */\n  index?: number;\n}\n\n/**\n * Generates an accessible summary of the bar values\n */\nfunction getValuesSummary({\n  yLabel,\n  seriesOneValue,\n  seriesTwoValue,\n  unit,\n}: {\n  yLabel: string;\n  seriesOneValue: number;\n  seriesTwoValue?: number;\n  unit: string;\n}): string {\n  if (seriesTwoValue !== undefined) {\n    const gained = seriesOneValue;\n    return `${gained} ${unit} gained - now at ${seriesTwoValue} ${unit} in ${yLabel} category`;\n  }\n  return `${seriesOneValue} ${unit} in ${yLabel} category`;\n}\n\nexport const BarRow = forwardRef<\n  HTMLDivElement | HTMLButtonElement | HTMLAnchorElement,\n  BarRowProps\n>(\n  (\n    {\n      yLabel,\n      seriesOneValue,\n      seriesTwoValue,\n      icon: Icon,\n      onClick,\n      href,\n      index = 0,\n    },\n    ref\n  ) => {\n    const { maxRange, unit, styleConfig, animate } = useBarChartContext();\n\n    const isStacked = seriesTwoValue !== undefined;\n    const displayValue = isStacked ? seriesTwoValue : seriesOneValue;\n\n    // Calculate bar widths as percentages\n    const backgroundBarWidth = calculateBarWidth({\n      value: displayValue,\n      maxRange,\n    });\n\n    const foregroundBarWidth = isStacked\n      ? calculateBarWidth({\n          value: seriesOneValue,\n          maxRange,\n        })\n      : 0;\n\n    const bgWidthStr = `${Math.max(minBarWidth, backgroundBarWidth)}%`;\n    const fgWidthStr = `${Math.max(minBarWidth, foregroundBarWidth)}%`;\n\n    const valuesSummary = getValuesSummary({\n      yLabel,\n      seriesOneValue,\n      seriesTwoValue,\n      unit,\n    });\n\n    // Animation delay for staggered bar entrance\n    const animationDelay = animate ? index * 0.1 : 0;\n\n    const rowContent = (\n      <>\n        {/* Y-axis label with optional icon */}\n        <FlexBox\n          alignItems=\"center\"\n          gap={8}\n          minWidth={{ _: 80, sm: 120 }}\n          flexShrink={0}\n        >\n          {Icon && <Icon size={16} color={styleConfig.textColor} />}\n          <Text\n            variant=\"p-small\"\n            color={styleConfig.textColor}\n            truncate=\"ellipsis\"\n            truncateLines={1}\n          >\n            {yLabel}\n          </Text>\n        </FlexBox>\n\n        {/* Bar container */}\n        <FlexBox flex={1} alignItems=\"center\" position=\"relative\">\n          <BarWrapper>\n            {/* Background bar (total value in stacked, or single value in non-stacked) */}\n            <BackgroundBar\n              bg={styleConfig.backgroundBarColor}\n              initial={animate ? { width: '0%' } : undefined}\n              animate={animate ? { width: bgWidthStr } : undefined}\n              style={!animate ? { width: bgWidthStr } : undefined}\n              transition={{ duration: 0.5, delay: animationDelay }}\n              data-testid=\"background-bar\"\n            />\n\n            {/* Foreground bar (progress value in stacked mode only) */}\n            {isStacked && (\n              <ForegroundBar\n                bg={styleConfig.foregroundBarColor}\n                initial={animate ? { width: '0%' } : undefined}\n                animate={animate ? { width: fgWidthStr } : undefined}\n                style={!animate ? { width: fgWidthStr } : undefined}\n                transition={{ duration: 0.5, delay: animationDelay + 0.25 }}\n                data-testid=\"foreground-bar\"\n              />\n            )}\n          </BarWrapper>\n        </FlexBox>\n\n        {/* Value display */}\n        <FlexBox\n          alignItems=\"center\"\n          justifyContent=\"flex-end\"\n          minWidth={{ _: 40, sm: 60 }}\n          flexShrink={0}\n        >\n          <Text variant=\"p-small\" color={styleConfig.textColor}>\n            {displayValue.toLocaleString()}\n            {unit && ` ${unit}`}\n          </Text>\n        </FlexBox>\n      </>\n    );\n\n    // Polymorphic rendering based on interactivity\n    if (href) {\n      return (\n        <li>\n          <RowAnchor\n            href={href}\n            aria-label={valuesSummary}\n            ref={ref as MutableRefObject<HTMLAnchorElement>}\n          >\n            {rowContent}\n          </RowAnchor>\n        </li>\n      );\n    }\n\n    if (onClick) {\n      return (\n        <li>\n          <RowButton\n            type=\"button\"\n            onClick={onClick}\n            onKeyDown={(e) => {\n              if (e.key === 'Enter') {\n                onClick(e as unknown as MouseEvent<HTMLButtonElement>);\n              }\n            }}\n            aria-label={valuesSummary}\n            ref={ref as MutableRefObject<HTMLButtonElement>}\n          >\n            {rowContent}\n          </RowButton>\n        </li>\n      );\n    }\n\n    return (\n      <li>\n        <RowWrapper\n          aria-label={valuesSummary}\n          ref={ref as MutableRefObject<HTMLDivElement>}\n        >\n          {rowContent}\n        </RowWrapper>\n      </li>\n    );\n  }\n);\n\nBarRow.displayName = 'BarRow';\n"]} */");
|
|
51
|
+
/**
|
|
52
|
+
* Generates an accessible summary of the bar values
|
|
53
|
+
*/
|
|
54
|
+
function getValuesSummary({
|
|
55
|
+
yLabel,
|
|
56
|
+
seriesOneValue,
|
|
57
|
+
seriesTwoValue,
|
|
58
|
+
unit
|
|
59
|
+
}) {
|
|
60
|
+
if (seriesTwoValue !== undefined) {
|
|
61
|
+
const gained = seriesOneValue;
|
|
62
|
+
return `${gained} ${unit} gained - now at ${seriesTwoValue} ${unit} in ${yLabel} category`;
|
|
63
|
+
}
|
|
64
|
+
return `${seriesOneValue} ${unit} in ${yLabel} category`;
|
|
65
|
+
}
|
|
66
|
+
export const BarRow = /*#__PURE__*/forwardRef(({
|
|
67
|
+
yLabel,
|
|
68
|
+
seriesOneValue,
|
|
69
|
+
seriesTwoValue,
|
|
70
|
+
icon: Icon,
|
|
71
|
+
onClick,
|
|
72
|
+
href,
|
|
73
|
+
index = 0
|
|
74
|
+
}, ref) => {
|
|
75
|
+
const {
|
|
76
|
+
maxRange,
|
|
77
|
+
unit,
|
|
78
|
+
styleConfig,
|
|
79
|
+
animate
|
|
80
|
+
} = useBarChartContext();
|
|
81
|
+
const isStacked = seriesTwoValue !== undefined;
|
|
82
|
+
const displayValue = isStacked ? seriesTwoValue : seriesOneValue;
|
|
83
|
+
|
|
84
|
+
// Calculate bar widths as percentages
|
|
85
|
+
const backgroundBarWidth = calculateBarWidth({
|
|
86
|
+
value: displayValue,
|
|
87
|
+
maxRange
|
|
88
|
+
});
|
|
89
|
+
const foregroundBarWidth = isStacked ? calculateBarWidth({
|
|
90
|
+
value: seriesOneValue,
|
|
91
|
+
maxRange
|
|
92
|
+
}) : 0;
|
|
93
|
+
const bgWidthStr = `${Math.max(minBarWidth, backgroundBarWidth)}%`;
|
|
94
|
+
const fgWidthStr = `${Math.max(minBarWidth, foregroundBarWidth)}%`;
|
|
95
|
+
const valuesSummary = getValuesSummary({
|
|
96
|
+
yLabel,
|
|
97
|
+
seriesOneValue,
|
|
98
|
+
seriesTwoValue,
|
|
99
|
+
unit
|
|
100
|
+
});
|
|
101
|
+
|
|
102
|
+
// Animation delay for staggered bar entrance
|
|
103
|
+
const animationDelay = animate ? index * 0.1 : 0;
|
|
104
|
+
const rowContent = /*#__PURE__*/_jsxs(_Fragment, {
|
|
105
|
+
children: [/*#__PURE__*/_jsxs(FlexBox, {
|
|
106
|
+
alignItems: "center",
|
|
107
|
+
gap: 8,
|
|
108
|
+
minWidth: {
|
|
109
|
+
_: 80,
|
|
110
|
+
sm: 120
|
|
111
|
+
},
|
|
112
|
+
flexShrink: 0,
|
|
113
|
+
children: [Icon && /*#__PURE__*/_jsx(Icon, {
|
|
114
|
+
size: 16,
|
|
115
|
+
color: styleConfig.textColor
|
|
116
|
+
}), /*#__PURE__*/_jsx(Text, {
|
|
117
|
+
variant: "p-small",
|
|
118
|
+
color: styleConfig.textColor,
|
|
119
|
+
truncate: "ellipsis",
|
|
120
|
+
truncateLines: 1,
|
|
121
|
+
children: yLabel
|
|
122
|
+
})]
|
|
123
|
+
}), /*#__PURE__*/_jsx(FlexBox, {
|
|
124
|
+
flex: 1,
|
|
125
|
+
alignItems: "center",
|
|
126
|
+
position: "relative",
|
|
127
|
+
children: /*#__PURE__*/_jsxs(BarWrapper, {
|
|
128
|
+
children: [/*#__PURE__*/_jsx(BackgroundBar, {
|
|
129
|
+
bg: styleConfig.backgroundBarColor,
|
|
130
|
+
initial: animate ? {
|
|
131
|
+
width: '0%'
|
|
132
|
+
} : undefined,
|
|
133
|
+
animate: animate ? {
|
|
134
|
+
width: bgWidthStr
|
|
135
|
+
} : undefined,
|
|
136
|
+
style: !animate ? {
|
|
137
|
+
width: bgWidthStr
|
|
138
|
+
} : undefined,
|
|
139
|
+
transition: {
|
|
140
|
+
duration: 0.5,
|
|
141
|
+
delay: animationDelay
|
|
142
|
+
},
|
|
143
|
+
"data-testid": "background-bar"
|
|
144
|
+
}), isStacked && /*#__PURE__*/_jsx(ForegroundBar, {
|
|
145
|
+
bg: styleConfig.foregroundBarColor,
|
|
146
|
+
initial: animate ? {
|
|
147
|
+
width: '0%'
|
|
148
|
+
} : undefined,
|
|
149
|
+
animate: animate ? {
|
|
150
|
+
width: fgWidthStr
|
|
151
|
+
} : undefined,
|
|
152
|
+
style: !animate ? {
|
|
153
|
+
width: fgWidthStr
|
|
154
|
+
} : undefined,
|
|
155
|
+
transition: {
|
|
156
|
+
duration: 0.5,
|
|
157
|
+
delay: animationDelay + 0.25
|
|
158
|
+
},
|
|
159
|
+
"data-testid": "foreground-bar"
|
|
160
|
+
})]
|
|
161
|
+
})
|
|
162
|
+
}), /*#__PURE__*/_jsx(FlexBox, {
|
|
163
|
+
alignItems: "center",
|
|
164
|
+
justifyContent: "flex-end",
|
|
165
|
+
minWidth: {
|
|
166
|
+
_: 40,
|
|
167
|
+
sm: 60
|
|
168
|
+
},
|
|
169
|
+
flexShrink: 0,
|
|
170
|
+
children: /*#__PURE__*/_jsxs(Text, {
|
|
171
|
+
variant: "p-small",
|
|
172
|
+
color: styleConfig.textColor,
|
|
173
|
+
children: [displayValue.toLocaleString(), unit && ` ${unit}`]
|
|
174
|
+
})
|
|
175
|
+
})]
|
|
176
|
+
});
|
|
177
|
+
|
|
178
|
+
// Polymorphic rendering based on interactivity
|
|
179
|
+
if (href) {
|
|
180
|
+
return /*#__PURE__*/_jsx("li", {
|
|
181
|
+
children: /*#__PURE__*/_jsx(RowAnchor, {
|
|
182
|
+
href: href,
|
|
183
|
+
"aria-label": valuesSummary,
|
|
184
|
+
ref: ref,
|
|
185
|
+
children: rowContent
|
|
186
|
+
})
|
|
187
|
+
});
|
|
188
|
+
}
|
|
189
|
+
if (onClick) {
|
|
190
|
+
return /*#__PURE__*/_jsx("li", {
|
|
191
|
+
children: /*#__PURE__*/_jsx(RowButton, {
|
|
192
|
+
type: "button",
|
|
193
|
+
onClick: onClick,
|
|
194
|
+
onKeyDown: e => {
|
|
195
|
+
if (e.key === 'Enter') {
|
|
196
|
+
onClick(e);
|
|
197
|
+
}
|
|
198
|
+
},
|
|
199
|
+
"aria-label": valuesSummary,
|
|
200
|
+
ref: ref,
|
|
201
|
+
children: rowContent
|
|
202
|
+
})
|
|
203
|
+
});
|
|
204
|
+
}
|
|
205
|
+
return /*#__PURE__*/_jsx("li", {
|
|
206
|
+
children: /*#__PURE__*/_jsx(RowWrapper, {
|
|
207
|
+
"aria-label": valuesSummary,
|
|
208
|
+
ref: ref,
|
|
209
|
+
children: rowContent
|
|
210
|
+
})
|
|
211
|
+
});
|
|
212
|
+
});
|
|
213
|
+
BarRow.displayName = 'BarRow';
|
|
@@ -0,0 +1,45 @@
|
|
|
1
|
+
import _styled from "@emotion/styled/base";
|
|
2
|
+
import { css } from '@codecademy/gamut-styles';
|
|
3
|
+
import { Box } from '../Box';
|
|
4
|
+
import { jsx as _jsx } from "react/jsx-runtime";
|
|
5
|
+
const GridLineContainer = /*#__PURE__*/_styled(Box, {
|
|
6
|
+
target: "e1y7txrp1",
|
|
7
|
+
label: "GridLineContainer"
|
|
8
|
+
})(css({
|
|
9
|
+
position: 'absolute',
|
|
10
|
+
top: 0,
|
|
11
|
+
left: 0,
|
|
12
|
+
right: 0,
|
|
13
|
+
bottom: 0,
|
|
14
|
+
display: {
|
|
15
|
+
_: 'none',
|
|
16
|
+
sm: 'flex'
|
|
17
|
+
},
|
|
18
|
+
pointerEvents: 'none',
|
|
19
|
+
zIndex: 0
|
|
20
|
+
}), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9CYXJDaGFydC9HcmlkTGluZXMudHN4Il0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUswQiIsImZpbGUiOiIuLi8uLi9zcmMvQmFyQ2hhcnQvR3JpZExpbmVzLnRzeCIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IGNzcyB9IGZyb20gJ0Bjb2RlY2FkZW15L2dhbXV0LXN0eWxlcyc7XG5pbXBvcnQgc3R5bGVkIGZyb20gJ0BlbW90aW9uL3N0eWxlZCc7XG5cbmltcG9ydCB7IEJveCB9IGZyb20gJy4uL0JveCc7XG5cbmNvbnN0IEdyaWRMaW5lQ29udGFpbmVyID0gc3R5bGVkKEJveCkoXG4gIGNzcyh7XG4gICAgcG9zaXRpb246ICdhYnNvbHV0ZScsXG4gICAgdG9wOiAwLFxuICAgIGxlZnQ6IDAsXG4gICAgcmlnaHQ6IDAsXG4gICAgYm90dG9tOiAwLFxuICAgIGRpc3BsYXk6IHsgXzogJ25vbmUnLCBzbTogJ2ZsZXgnIH0sXG4gICAgcG9pbnRlckV2ZW50czogJ25vbmUnLFxuICAgIHpJbmRleDogMCxcbiAgfSlcbik7XG5cbmNvbnN0IEdyaWRMaW5lID0gc3R5bGVkKEJveCkoXG4gIGNzcyh7XG4gICAgYm9yZGVyTGVmdDogMSxcbiAgICBib3JkZXJDb2xvckxlZnQ6ICdib3JkZXItdGVydGlhcnknLFxuICAgIGhlaWdodDogJzEwMCUnLFxuICAgIGZsZXg6IDEsXG4gICAgJyY6Zmlyc3Qtb2YtdHlwZSc6IHtcbiAgICAgIGJvcmRlckxlZnQ6ICdub25lJyxcbiAgICB9LFxuICB9KVxuKTtcblxuZXhwb3J0IGludGVyZmFjZSBHcmlkTGluZXNQcm9wcyB7XG4gIC8qKiBOdW1iZXIgb2YgZ3JpZCBsaW5lcyB0byByZW5kZXIgKi9cbiAgdGlja0NvdW50OiBudW1iZXI7XG59XG5cbmV4cG9ydCBjb25zdCBHcmlkTGluZXM6IFJlYWN0LkZDPEdyaWRMaW5lc1Byb3BzPiA9ICh7IHRpY2tDb3VudCB9KSA9PiB7XG4gIGNvbnN0IGxpbmVzID0gQXJyYXkuZnJvbSh7IGxlbmd0aDogdGlja0NvdW50IH0sIChfLCBpKSA9PiAoXG4gICAgPEdyaWRMaW5lIGtleT17aX0gYXJpYS1oaWRkZW49XCJ0cnVlXCIgLz5cbiAgKSk7XG5cbiAgcmV0dXJuIDxHcmlkTGluZUNvbnRhaW5lciBhcmlhLWhpZGRlbj1cInRydWVcIj57bGluZXN9PC9HcmlkTGluZUNvbnRhaW5lcj47XG59O1xuIl19 */");
|
|
21
|
+
const GridLine = /*#__PURE__*/_styled(Box, {
|
|
22
|
+
target: "e1y7txrp0",
|
|
23
|
+
label: "GridLine"
|
|
24
|
+
})(css({
|
|
25
|
+
borderLeft: 1,
|
|
26
|
+
borderColorLeft: 'border-tertiary',
|
|
27
|
+
height: '100%',
|
|
28
|
+
flex: 1,
|
|
29
|
+
'&:first-of-type': {
|
|
30
|
+
borderLeft: 'none'
|
|
31
|
+
}
|
|
32
|
+
}), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9CYXJDaGFydC9HcmlkTGluZXMudHN4Il0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQWtCaUIiLCJmaWxlIjoiLi4vLi4vc3JjL0JhckNoYXJ0L0dyaWRMaW5lcy50c3giLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBjc3MgfSBmcm9tICdAY29kZWNhZGVteS9nYW11dC1zdHlsZXMnO1xuaW1wb3J0IHN0eWxlZCBmcm9tICdAZW1vdGlvbi9zdHlsZWQnO1xuXG5pbXBvcnQgeyBCb3ggfSBmcm9tICcuLi9Cb3gnO1xuXG5jb25zdCBHcmlkTGluZUNvbnRhaW5lciA9IHN0eWxlZChCb3gpKFxuICBjc3Moe1xuICAgIHBvc2l0aW9uOiAnYWJzb2x1dGUnLFxuICAgIHRvcDogMCxcbiAgICBsZWZ0OiAwLFxuICAgIHJpZ2h0OiAwLFxuICAgIGJvdHRvbTogMCxcbiAgICBkaXNwbGF5OiB7IF86ICdub25lJywgc206ICdmbGV4JyB9LFxuICAgIHBvaW50ZXJFdmVudHM6ICdub25lJyxcbiAgICB6SW5kZXg6IDAsXG4gIH0pXG4pO1xuXG5jb25zdCBHcmlkTGluZSA9IHN0eWxlZChCb3gpKFxuICBjc3Moe1xuICAgIGJvcmRlckxlZnQ6IDEsXG4gICAgYm9yZGVyQ29sb3JMZWZ0OiAnYm9yZGVyLXRlcnRpYXJ5JyxcbiAgICBoZWlnaHQ6ICcxMDAlJyxcbiAgICBmbGV4OiAxLFxuICAgICcmOmZpcnN0LW9mLXR5cGUnOiB7XG4gICAgICBib3JkZXJMZWZ0OiAnbm9uZScsXG4gICAgfSxcbiAgfSlcbik7XG5cbmV4cG9ydCBpbnRlcmZhY2UgR3JpZExpbmVzUHJvcHMge1xuICAvKiogTnVtYmVyIG9mIGdyaWQgbGluZXMgdG8gcmVuZGVyICovXG4gIHRpY2tDb3VudDogbnVtYmVyO1xufVxuXG5leHBvcnQgY29uc3QgR3JpZExpbmVzOiBSZWFjdC5GQzxHcmlkTGluZXNQcm9wcz4gPSAoeyB0aWNrQ291bnQgfSkgPT4ge1xuICBjb25zdCBsaW5lcyA9IEFycmF5LmZyb20oeyBsZW5ndGg6IHRpY2tDb3VudCB9LCAoXywgaSkgPT4gKFxuICAgIDxHcmlkTGluZSBrZXk9e2l9IGFyaWEtaGlkZGVuPVwidHJ1ZVwiIC8+XG4gICkpO1xuXG4gIHJldHVybiA8R3JpZExpbmVDb250YWluZXIgYXJpYS1oaWRkZW49XCJ0cnVlXCI+e2xpbmVzfTwvR3JpZExpbmVDb250YWluZXI+O1xufTtcbiJdfQ== */");
|
|
33
|
+
export const GridLines = ({
|
|
34
|
+
tickCount
|
|
35
|
+
}) => {
|
|
36
|
+
const lines = Array.from({
|
|
37
|
+
length: tickCount
|
|
38
|
+
}, (_, i) => /*#__PURE__*/_jsx(GridLine, {
|
|
39
|
+
"aria-hidden": "true"
|
|
40
|
+
}, i));
|
|
41
|
+
return /*#__PURE__*/_jsx(GridLineContainer, {
|
|
42
|
+
"aria-hidden": "true",
|
|
43
|
+
children: lines
|
|
44
|
+
});
|
|
45
|
+
};
|
|
@@ -0,0 +1,17 @@
|
|
|
1
|
+
/// <reference types="react" />
|
|
2
|
+
export interface ScaleChartHeaderProps {
|
|
3
|
+
/** Minimum value on the scale */
|
|
4
|
+
min: number;
|
|
5
|
+
/** Maximum value on the scale */
|
|
6
|
+
max: number;
|
|
7
|
+
/** Number of labels to display */
|
|
8
|
+
labelCount: number;
|
|
9
|
+
}
|
|
10
|
+
/**
|
|
11
|
+
* Calculates the value for a given label position
|
|
12
|
+
*/
|
|
13
|
+
export declare const getLabel: (labelCount: number, labelIndex: number, max: number) => number;
|
|
14
|
+
/**
|
|
15
|
+
* Header component showing the x-axis scale labels
|
|
16
|
+
*/
|
|
17
|
+
export declare const ScaleChartHeader: React.FC<ScaleChartHeaderProps>;
|
|
@@ -0,0 +1,58 @@
|
|
|
1
|
+
import _styled from "@emotion/styled/base";
|
|
2
|
+
import { css } from '@codecademy/gamut-styles';
|
|
3
|
+
import { FlexBox } from '../Box';
|
|
4
|
+
import { Text } from '../Typography';
|
|
5
|
+
import { formatNumberUSCompact } from './utils';
|
|
6
|
+
import { jsx as _jsx } from "react/jsx-runtime";
|
|
7
|
+
const StyledLabelText = /*#__PURE__*/_styled(Text, {
|
|
8
|
+
target: "egjxop0",
|
|
9
|
+
label: "StyledLabelText"
|
|
10
|
+
})(css({
|
|
11
|
+
flex: 1
|
|
12
|
+
}), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9CYXJDaGFydC9TY2FsZUNoYXJ0SGVhZGVyLnRzeCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFnQndCIiwiZmlsZSI6Ii4uLy4uL3NyYy9CYXJDaGFydC9TY2FsZUNoYXJ0SGVhZGVyLnRzeCIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IGNzcyB9IGZyb20gJ0Bjb2RlY2FkZW15L2dhbXV0LXN0eWxlcyc7XG5pbXBvcnQgc3R5bGVkIGZyb20gJ0BlbW90aW9uL3N0eWxlZCc7XG5cbmltcG9ydCB7IEZsZXhCb3ggfSBmcm9tICcuLi9Cb3gnO1xuaW1wb3J0IHsgVGV4dCB9IGZyb20gJy4uL1R5cG9ncmFwaHknO1xuaW1wb3J0IHsgZm9ybWF0TnVtYmVyVVNDb21wYWN0IH0gZnJvbSAnLi91dGlscyc7XG5cbmV4cG9ydCBpbnRlcmZhY2UgU2NhbGVDaGFydEhlYWRlclByb3BzIHtcbiAgLyoqIE1pbmltdW0gdmFsdWUgb24gdGhlIHNjYWxlICovXG4gIG1pbjogbnVtYmVyO1xuICAvKiogTWF4aW11bSB2YWx1ZSBvbiB0aGUgc2NhbGUgKi9cbiAgbWF4OiBudW1iZXI7XG4gIC8qKiBOdW1iZXIgb2YgbGFiZWxzIHRvIGRpc3BsYXkgKi9cbiAgbGFiZWxDb3VudDogbnVtYmVyO1xufVxuXG5jb25zdCBTdHlsZWRMYWJlbFRleHQgPSBzdHlsZWQoVGV4dCkoXG4gIGNzcyh7XG4gICAgZmxleDogMSxcbiAgfSlcbik7XG5cbi8qKlxuICogQ2FsY3VsYXRlcyB0aGUgdmFsdWUgZm9yIGEgZ2l2ZW4gbGFiZWwgcG9zaXRpb25cbiAqL1xuZXhwb3J0IGNvbnN0IGdldExhYmVsID0gKFxuICBsYWJlbENvdW50OiBudW1iZXIsXG4gIGxhYmVsSW5kZXg6IG51bWJlcixcbiAgbWF4OiBudW1iZXJcbik6IG51bWJlciA9PiB7XG4gIGlmIChsYWJlbENvdW50IDw9IDEpIHJldHVybiBtYXg7XG4gIGNvbnN0IGluY3JlbWVudGFsRGVjaW1hbCA9IGxhYmVsSW5kZXggLyAobGFiZWxDb3VudCAtIDEpO1xuICByZXR1cm4gTWF0aC5mbG9vcihpbmNyZW1lbnRhbERlY2ltYWwgKiBtYXgpO1xufTtcblxuLyoqXG4gKiBIZWFkZXIgY29tcG9uZW50IHNob3dpbmcgdGhlIHgtYXhpcyBzY2FsZSBsYWJlbHNcbiAqL1xuZXhwb3J0IGNvbnN0IFNjYWxlQ2hhcnRIZWFkZXI6IFJlYWN0LkZDPFNjYWxlQ2hhcnRIZWFkZXJQcm9wcz4gPSAoe1xuICBsYWJlbENvdW50LFxuICBtYXgsXG59KSA9PiB7XG4gIGNvbnN0IHNjYWxlTGFiZWxzID0gQXJyYXkuZnJvbSh7IGxlbmd0aDogbGFiZWxDb3VudCB9LCAoXywgaSkgPT4gKFxuICAgIDxTdHlsZWRMYWJlbFRleHRcbiAgICAgIGtleT17aX1cbiAgICAgIHZhcmlhbnQ9XCJwLXNtYWxsXCJcbiAgICAgIHRleHRDb2xvcj1cInRleHQtc2Vjb25kYXJ5XCJcbiAgICAgIGRhdGEtdGVzdGlkPVwiY2hhcnQtaGVhZGVyLWxhYmVsXCJcbiAgICAgIHRleHRBbGlnbj17aSA9PT0gMCA/ICdsZWZ0JyA6IGkgPT09IGxhYmVsQ291bnQgLSAxID8gJ3JpZ2h0JyA6ICdjZW50ZXInfVxuICAgID5cbiAgICAgIHtmb3JtYXROdW1iZXJVU0NvbXBhY3QoZ2V0TGFiZWwobGFiZWxDb3VudCwgaSwgbWF4KSl9XG4gICAgPC9TdHlsZWRMYWJlbFRleHQ+XG4gICkpO1xuXG4gIHJldHVybiAoXG4gICAgPEZsZXhCb3hcbiAgICAgIHdpZHRoPVwiMTAwJVwiXG4gICAgICBqdXN0aWZ5Q29udGVudD1cInNwYWNlLWJldHdlZW5cIlxuICAgICAgbWI9ezh9XG4gICAgICBwbD17eyBfOiA2NCwgc206IDk2IH19XG4gICAgICBwcj17eyBfOiA0MCwgc206IDY0IH19XG4gICAgICBkaXNwbGF5PXt7IF86ICdub25lJywgc206ICdmbGV4JyB9fVxuICAgICAgYXJpYS1oaWRkZW49XCJ0cnVlXCJcbiAgICA+XG4gICAgICB7c2NhbGVMYWJlbHN9XG4gICAgPC9GbGV4Qm94PlxuICApO1xufTtcbiJdfQ== */");
|
|
13
|
+
|
|
14
|
+
/**
|
|
15
|
+
* Calculates the value for a given label position
|
|
16
|
+
*/
|
|
17
|
+
export const getLabel = (labelCount, labelIndex, max) => {
|
|
18
|
+
if (labelCount <= 1) return max;
|
|
19
|
+
const incrementalDecimal = labelIndex / (labelCount - 1);
|
|
20
|
+
return Math.floor(incrementalDecimal * max);
|
|
21
|
+
};
|
|
22
|
+
|
|
23
|
+
/**
|
|
24
|
+
* Header component showing the x-axis scale labels
|
|
25
|
+
*/
|
|
26
|
+
export const ScaleChartHeader = ({
|
|
27
|
+
labelCount,
|
|
28
|
+
max
|
|
29
|
+
}) => {
|
|
30
|
+
const scaleLabels = Array.from({
|
|
31
|
+
length: labelCount
|
|
32
|
+
}, (_, i) => /*#__PURE__*/_jsx(StyledLabelText, {
|
|
33
|
+
variant: "p-small",
|
|
34
|
+
textColor: "text-secondary",
|
|
35
|
+
"data-testid": "chart-header-label",
|
|
36
|
+
textAlign: i === 0 ? 'left' : i === labelCount - 1 ? 'right' : 'center',
|
|
37
|
+
children: formatNumberUSCompact(getLabel(labelCount, i, max))
|
|
38
|
+
}, i));
|
|
39
|
+
return /*#__PURE__*/_jsx(FlexBox, {
|
|
40
|
+
width: "100%",
|
|
41
|
+
justifyContent: "space-between",
|
|
42
|
+
mb: 8,
|
|
43
|
+
pl: {
|
|
44
|
+
_: 64,
|
|
45
|
+
sm: 96
|
|
46
|
+
},
|
|
47
|
+
pr: {
|
|
48
|
+
_: 40,
|
|
49
|
+
sm: 64
|
|
50
|
+
},
|
|
51
|
+
display: {
|
|
52
|
+
_: 'none',
|
|
53
|
+
sm: 'flex'
|
|
54
|
+
},
|
|
55
|
+
"aria-hidden": "true",
|
|
56
|
+
children: scaleLabels
|
|
57
|
+
});
|
|
58
|
+
};
|
|
@@ -0,0 +1,81 @@
|
|
|
1
|
+
import { useMemo } from 'react';
|
|
2
|
+
import { Box } from '../Box';
|
|
3
|
+
import { BarChartProvider, useBarChart } from './BarChartProvider';
|
|
4
|
+
import { BarRow } from './BarRow';
|
|
5
|
+
import { GridLines } from './GridLines';
|
|
6
|
+
import { ScaleChartHeader } from './ScaleChartHeader';
|
|
7
|
+
import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
|
|
8
|
+
/**
|
|
9
|
+
* Sort bars based on sortBy and order configuration
|
|
10
|
+
*/
|
|
11
|
+
function sortBars(bars, sortBy, order) {
|
|
12
|
+
if (sortBy === 'none' || !sortBy) {
|
|
13
|
+
return bars;
|
|
14
|
+
}
|
|
15
|
+
const sorted = [...bars].sort((a, b) => {
|
|
16
|
+
if (sortBy === 'label') {
|
|
17
|
+
return a.yLabel.localeCompare(b.yLabel);
|
|
18
|
+
}
|
|
19
|
+
// sortBy === 'value' - use seriesTwoValue if available, otherwise seriesOneValue
|
|
20
|
+
const aValue = a.seriesTwoValue ?? a.seriesOneValue;
|
|
21
|
+
const bValue = b.seriesTwoValue ?? b.seriesOneValue;
|
|
22
|
+
return aValue - bValue;
|
|
23
|
+
});
|
|
24
|
+
return order === 'descending' ? sorted.reverse() : sorted;
|
|
25
|
+
}
|
|
26
|
+
export const BarChart = ({
|
|
27
|
+
'aria-label': ariaLabel,
|
|
28
|
+
'aria-labelledby': ariaLabelledBy,
|
|
29
|
+
animate = false,
|
|
30
|
+
barValues,
|
|
31
|
+
maxRange,
|
|
32
|
+
minRange,
|
|
33
|
+
order = 'ascending',
|
|
34
|
+
sortBy = 'none',
|
|
35
|
+
unit = '',
|
|
36
|
+
styleConfig,
|
|
37
|
+
xScale
|
|
38
|
+
}) => {
|
|
39
|
+
const contextValue = useBarChart({
|
|
40
|
+
minRange,
|
|
41
|
+
maxRange,
|
|
42
|
+
xScale,
|
|
43
|
+
unit,
|
|
44
|
+
styleConfig,
|
|
45
|
+
animate
|
|
46
|
+
});
|
|
47
|
+
const sortedBars = useMemo(() => sortBars(barValues, sortBy, order), [barValues, sortBy, order]);
|
|
48
|
+
|
|
49
|
+
// Calculate number of ticks for the scale header
|
|
50
|
+
const tickCount = Math.ceil((maxRange - minRange) / contextValue.xScale) + 1;
|
|
51
|
+
return /*#__PURE__*/_jsx(BarChartProvider, {
|
|
52
|
+
value: contextValue,
|
|
53
|
+
children: /*#__PURE__*/_jsxs(Box, {
|
|
54
|
+
width: "100%",
|
|
55
|
+
position: "relative",
|
|
56
|
+
children: [/*#__PURE__*/_jsx(ScaleChartHeader, {
|
|
57
|
+
min: minRange,
|
|
58
|
+
max: maxRange,
|
|
59
|
+
labelCount: tickCount
|
|
60
|
+
}), /*#__PURE__*/_jsxs(Box, {
|
|
61
|
+
position: "relative",
|
|
62
|
+
width: "100%",
|
|
63
|
+
children: [/*#__PURE__*/_jsx(GridLines, {
|
|
64
|
+
tickCount: tickCount
|
|
65
|
+
}), /*#__PURE__*/_jsx(Box, {
|
|
66
|
+
as: "ul",
|
|
67
|
+
"aria-label": ariaLabel,
|
|
68
|
+
"aria-labelledby": ariaLabelledBy,
|
|
69
|
+
p: 0,
|
|
70
|
+
m: 0,
|
|
71
|
+
listStyle: "none",
|
|
72
|
+
children: sortedBars.map((bar, index) => /*#__PURE__*/_jsx(BarRow, {
|
|
73
|
+
index: index,
|
|
74
|
+
...bar
|
|
75
|
+
}, `${bar.yLabel}-${index}`))
|
|
76
|
+
})]
|
|
77
|
+
})]
|
|
78
|
+
})
|
|
79
|
+
});
|
|
80
|
+
};
|
|
81
|
+
export * from './types';
|