diff --git a/packages/polaris-viz/src/components/Grid/Grid.tsx b/packages/polaris-viz/src/components/Grid/Grid.tsx index ab718a1cc..bb7a2e070 100644 --- a/packages/polaris-viz/src/components/Grid/Grid.tsx +++ b/packages/polaris-viz/src/components/Grid/Grid.tsx @@ -223,6 +223,23 @@ export function Grid(props: GridProps) { }; }; + const handleBodyClick = useCallback(() => { + setGroupSelected(null); + handleGroupHover(null); + }, [handleGroupHover]); + + useEffect(() => { + if (groupSelected) { + setTimeout(() => { + document.body.addEventListener('click', handleBodyClick); + }, 0); + + return () => { + document.body.removeEventListener('click', handleBodyClick); + }; + } + }, [groupSelected, handleBodyClick]); + const yTicks = useMemo(() => { return Array.from({length: gridDimensions.rows}, (_, index) => ({ value: gridDimensions.rows - 1 - index, diff --git a/packages/polaris-viz/src/components/Grid/components/Arrows.scss b/packages/polaris-viz/src/components/Grid/components/Arrows.scss index b265793b9..19a990c8a 100644 --- a/packages/polaris-viz/src/components/Grid/components/Arrows.scss +++ b/packages/polaris-viz/src/components/Grid/components/Arrows.scss @@ -2,6 +2,10 @@ @import '../utilities/constants'; +.Arrow { + filter: drop-shadow(0 1px 2px rgba(0, 0, 0, 0.4)); +} + .ArrowShaft { stroke-dasharray: 35; stroke-dashoffset: 35; diff --git a/packages/polaris-viz/src/components/Grid/components/Arrows.tsx b/packages/polaris-viz/src/components/Grid/components/Arrows.tsx index d847bd676..e96f6f740 100644 --- a/packages/polaris-viz/src/components/Grid/components/Arrows.tsx +++ b/packages/polaris-viz/src/components/Grid/components/Arrows.tsx @@ -193,6 +193,7 @@ export function Arrows({ return ( (null); + + useLayoutEffect(() => { + if (group?.actions?.length && actionsContainerRef.current) { + // This is a workaround to determine if the buttons should be displayed in one or two columns depending if the copy fits in one line + const buttonsWrapper = actionsContainerRef.current; + const actionsButtons = buttonsWrapper.querySelectorAll('a span'); + + const heights: Set = new Set(); + let areWrapping = false; + for (const button of actionsButtons) { + const wrapping = + parseFloat(window.getComputedStyle(button).height) / + parseFloat(window.getComputedStyle(button).lineHeight) > + 1; + if (wrapping) { + areWrapping = true; + } + heights.add(button.getBoundingClientRect().height); + } + + const haveSameHeight = heights.size === 1; + setActionsButtonsColumns(haveSameHeight && !areWrapping ? 2 : 1); + } + // eslint-disable-next-line react-hooks/exhaustive-deps + }, [group?.actions, actionsContainerRef.current]); + if (!group) { return null; } @@ -26,6 +55,9 @@ export function Tooltip({x, y, group}: TooltipProps) { transform: `translate(${x}px, ${y}px)`, }} aria-label={group.name} + onClick={(event) => { + event.stopPropagation(); + }} >
{group.name}
@@ -56,6 +88,40 @@ export function Tooltip({x, y, group}: TooltipProps) {

{group.goal}

)} + {group.actions?.length && ( +
+
+ {group.actions?.map((action, index) => { + const numActions = group.actions?.length || 0; + const oddNumActions = numActions % 2 !== 0; + const lastAction = index === numActions - 1; + const spanFullWidth = oddNumActions && lastAction; + return ( + + ); + })} +
+
+ )} , container, diff --git a/packages/polaris-viz/src/components/Grid/stories/data.tsx b/packages/polaris-viz/src/components/Grid/stories/data.tsx index 319b4dc60..8683debcf 100644 --- a/packages/polaris-viz/src/components/Grid/stories/data.tsx +++ b/packages/polaris-viz/src/components/Grid/stories/data.tsx @@ -33,6 +33,18 @@ export const CELL_GROUPS: CellGroup[] = [ 'Customers without recent purchases, but with a very strong history of orders and spend.', goal: 'Goal: move customers to Loyal', metricInformation: '1,424 (10.9% of customer base)', + actions: [ + { + children: 'Preview segment', + url: '#', + target: '_blank', + }, + { + children: 'View report', + url: '#', + target: '_blank', + }, + ], }, { id: 'at_risk', @@ -48,6 +60,18 @@ export const CELL_GROUPS: CellGroup[] = [ description: 'Customers without recent purchases, but with a strong history of orders and spend.', goal: 'Goal: move customers to Loyal or Needs Attention', + actions: [ + { + children: 'Preview segment', + url: '#', + target: '_blank', + }, + { + children: 'View report', + url: '#', + target: '_blank', + }, + ], }, { id: 'dormant', @@ -63,6 +87,18 @@ export const CELL_GROUPS: CellGroup[] = [ 'Customers without recent orders, with infrequent orders, and with low spend.', goal: 'Goal: move customers to Almost lost', metricInformation: '1,424 (10.9% of customer base)', + actions: [ + { + children: 'Preview segment', + url: '#', + target: '_blank', + }, + { + children: 'View report', + url: '#', + target: '_blank', + }, + ], }, { id: 'loyal', @@ -78,6 +114,18 @@ export const CELL_GROUPS: CellGroup[] = [ 'Customers without recent purchases, but with a very strong history of orders and spend.', goal: 'Goal: move customers to Loyal', metricInformation: '1,424 (10.9% of customer base)', + actions: [ + { + children: 'Preview segment', + url: '#', + target: '_blank', + }, + { + children: 'View report', + url: '#', + target: '_blank', + }, + ], }, { id: 'needs_attention', @@ -93,6 +141,18 @@ export const CELL_GROUPS: CellGroup[] = [ 'Customers who buy less recently, order sometimes and spend moderately with your store.', goal: 'Goal: move customers to Loyal or Potential', metricInformation: '1,424 (10.9% of customer base)', + actions: [ + { + children: 'Preview segment', + url: '#', + target: '_blank', + }, + { + children: 'View report long text', + url: '#', + target: '_blank', + }, + ], }, { id: 'almost_lost', @@ -108,6 +168,18 @@ export const CELL_GROUPS: CellGroup[] = [ 'Customers without recent purchases, fewer orders, and with lower spend.', goal: 'Goal: move customers to Active or Promising', metricInformation: '1,424 (10.9% of customer base)', + actions: [ + { + children: 'Preview segment', + url: '#', + target: '_blank', + }, + { + children: 'View report', + url: '#', + target: '_blank', + }, + ], }, { id: 'promising', @@ -122,6 +194,18 @@ export const CELL_GROUPS: CellGroup[] = [ description: 'Customers with recent purchases, few orders, and low spend.', goal: 'Goal: move customers to Active.', metricInformation: '1,424 (10.9% of customer base)', + actions: [ + { + children: 'Preview segment', + url: '#', + target: '_blank', + }, + { + children: 'View report', + url: '#', + target: '_blank', + }, + ], }, { id: 'active', @@ -137,6 +221,18 @@ export const CELL_GROUPS: CellGroup[] = [ 'Customers with recent purchases, some orders, and moderate spend.', goal: 'Goal: move customers to Champions or Loyal', metricInformation: '1,424 (10.9% of customer base)', + actions: [ + { + children: 'Preview segment', + url: '#', + target: '_blank', + }, + { + children: 'View report', + url: '#', + target: '_blank', + }, + ], }, { id: 'new', @@ -152,6 +248,18 @@ export const CELL_GROUPS: CellGroup[] = [ 'Clients ayant effectué des achats très récemment, ayant passé peu de commandes et ayant dépensé peu dargent.', goal: 'Goal: move customers to Active', metricInformation: '1,424 (10.9% of customer base)', + actions: [ + { + children: 'Preview segment', + url: '#', + target: '_blank', + }, + { + children: 'View report', + url: '#', + target: '_blank', + }, + ], }, { id: 'champions', @@ -166,6 +274,18 @@ export const CELL_GROUPS: CellGroup[] = [ 'Customers with very recent purchases, many orders, and the most spend.', goal: null, metricInformation: '1,424 (10.9% of customer base)', + actions: [ + { + children: 'Preview segment', + url: '#', + target: '_blank', + }, + { + children: 'View report', + url: '#', + target: '_blank', + }, + ], }, ]; diff --git a/packages/polaris-viz/src/components/Grid/types.ts b/packages/polaris-viz/src/components/Grid/types.ts index 3713f8d6f..82d6f8b8c 100644 --- a/packages/polaris-viz/src/components/Grid/types.ts +++ b/packages/polaris-viz/src/components/Grid/types.ts @@ -18,6 +18,13 @@ export interface CellGroup { secondaryValue: string; value: string; metricInformation?: string; + actions?: GroupAction[]; +} + +export interface GroupAction { + children: string; + url?: string; + target?: '_blank' | '_self' | '_parent' | '_top'; } export interface TooltipInfo { diff --git a/packages/polaris-viz/src/components/Grid/utilities/constants.ts b/packages/polaris-viz/src/components/Grid/utilities/constants.ts index c4ce984f9..8ff4ba480 100644 --- a/packages/polaris-viz/src/components/Grid/utilities/constants.ts +++ b/packages/polaris-viz/src/components/Grid/utilities/constants.ts @@ -1,5 +1,5 @@ -export const TOOLTIP_WIDTH = 250; -export const TOOLTIP_HEIGHT = 155; +export const TOOLTIP_WIDTH = 280; +export const TOOLTIP_HEIGHT = 180; export const TOOLTIP_HORIZONTAL_OFFSET = 10; export const Y_LABEL_OFFSET = 25;