Skip to content

Commit

Permalink
Plans: Header price-discounting refactors (#96492)
Browse files Browse the repository at this point in the history
* use wp-data store

* Revert "use wp-data store"

This reverts commit 29abd6f.

* wrap header discounting logic in a hook

* move it to header-price context

* fix tests
  • Loading branch information
chriskmnds authored Nov 25, 2024
1 parent fdaf7c6 commit 5e425d8
Show file tree
Hide file tree
Showing 5 changed files with 101 additions and 55 deletions.
40 changes: 21 additions & 19 deletions packages/plans-grid-next/src/components/comparison-grid/index.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -36,6 +36,7 @@ import PopularBadge from '../popular-badge';
import ActionButton from '../shared/action-button';
import BillingTimeframe from '../shared/billing-timeframe';
import HeaderPrice from '../shared/header-price';
import HeaderPriceContextProvider from '../shared/header-price/header-price-context';
import { PlanStorage } from '../shared/storage';
import { StickyContainer } from '../sticky-container';
import type {
Expand All @@ -53,7 +54,6 @@ import type {
PlanSlug,
FeatureGroupMap,
} from '@automattic/calypso-products';

import './style.scss';

const featureGroupRowTitleCellMaxWidth = 450;
Expand Down Expand Up @@ -500,24 +500,26 @@ const ComparisonGridHeader = forwardRef< HTMLDivElement, ComparisonGridHeaderPro
</PlanTypeSelectorWrapper>
) }
</RowTitleCell>
{ visibleGridPlans.map( ( { planSlug }, index ) => (
<ComparisonGridHeaderCell
planSlug={ planSlug }
key={ planSlug }
isLastInRow={ index === visibleGridPlans.length - 1 }
isFooter={ isFooter }
allVisible={ allVisible }
isInSignup={ isInSignup }
visibleGridPlans={ visibleGridPlans }
onPlanChange={ onPlanChange }
displayedGridPlans={ displayedGridPlans }
currentSitePlanSlug={ currentSitePlanSlug }
planActionOverrides={ planActionOverrides }
selectedPlan={ selectedPlan }
showRefundPeriod={ showRefundPeriod }
isStuck={ isStuck }
/>
) ) }
<HeaderPriceContextProvider>
{ visibleGridPlans.map( ( { planSlug }, index ) => (
<ComparisonGridHeaderCell
planSlug={ planSlug }
key={ planSlug }
isLastInRow={ index === visibleGridPlans.length - 1 }
isFooter={ isFooter }
allVisible={ allVisible }
isInSignup={ isInSignup }
visibleGridPlans={ visibleGridPlans }
onPlanChange={ onPlanChange }
displayedGridPlans={ displayedGridPlans }
currentSitePlanSlug={ currentSitePlanSlug }
planActionOverrides={ planActionOverrides }
selectedPlan={ selectedPlan }
showRefundPeriod={ showRefundPeriod }
isStuck={ isStuck }
/>
) ) }
</HeaderPriceContextProvider>
</PlanRow>
);
}
Expand Down
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
import { GridPlan } from '../../types';
import PlanDivOrTdContainer from '../plan-div-td-container';
import HeaderPrice from '../shared/header-price';
import HeaderPriceContextProvider from '../shared/header-price/header-price-context';

type PlanPricesProps = {
currentSitePlanSlug?: string | null;
Expand All @@ -11,22 +12,26 @@ type PlanPricesProps = {
};

const PlanPrices = ( { currentSitePlanSlug, options, renderedGridPlans }: PlanPricesProps ) => {
return renderedGridPlans.map( ( { planSlug } ) => {
return (
<PlanDivOrTdContainer
scope="col"
key={ planSlug }
className="plan-features-2023-grid__table-item plan-price"
isTableCell={ options?.isTableCell }
>
<HeaderPrice
planSlug={ planSlug }
currentSitePlanSlug={ currentSitePlanSlug }
visibleGridPlans={ renderedGridPlans }
/>
</PlanDivOrTdContainer>
);
} );
return (
<HeaderPriceContextProvider>
{ renderedGridPlans.map( ( { planSlug } ) => {
return (
<PlanDivOrTdContainer
scope="col"
key={ planSlug }
className="plan-features-2023-grid__table-item plan-price"
isTableCell={ options?.isTableCell }
>
<HeaderPrice
planSlug={ planSlug }
currentSitePlanSlug={ currentSitePlanSlug }
visibleGridPlans={ renderedGridPlans }
/>
</PlanDivOrTdContainer>
);
} ) }
</HeaderPriceContextProvider>
);
};

export default PlanPrices;
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
import { createContext, useContext, useState } from '@wordpress/element';

interface HeaderPriceContext {
isAnyPlanPriceDiscounted: boolean;
setIsAnyPlanPriceDiscounted: ( isAnyPlanPriceDiscounted: boolean ) => void;
}

const HeaderPriceContext = createContext< HeaderPriceContext >( {} as HeaderPriceContext );

const HeaderPriceContextProvider = ( { children }: { children: React.ReactNode } ) => {
const [ isAnyPlanPriceDiscounted, setIsAnyPlanPriceDiscounted ] = useState( false );

return (
<HeaderPriceContext.Provider
value={ { isAnyPlanPriceDiscounted, setIsAnyPlanPriceDiscounted } }
>
{ children }
</HeaderPriceContext.Provider>
);
};

export const useHeaderPriceContext = (): HeaderPriceContext => useContext( HeaderPriceContext );

export default HeaderPriceContextProvider;
Original file line number Diff line number Diff line change
Expand Up @@ -7,11 +7,13 @@ import {
} from '@automattic/calypso-products';
import { PlanPrice } from '@automattic/components';
import { AddOns, Plans } from '@automattic/data-stores';
import { useEffect } from '@wordpress/element';
import clsx from 'clsx';
import { useTranslate } from 'i18n-calypso';
import { usePlansGridContext } from '../../../grid-context';
import useIsLargeCurrency from '../../../hooks/use-is-large-currency';
import { usePlanPricingInfoFromGridPlans } from '../../../hooks/use-plan-pricing-info-from-grid-plans';
import { useHeaderPriceContext } from './header-price-context';
import type { GridPlan } from '../../../types';
import './style.scss';

Expand Down Expand Up @@ -44,6 +46,7 @@ const HeaderPrice = ( { planSlug, visibleGridPlans }: HeaderPriceProps ) => {
const translate = useTranslate();
const { gridPlansIndex, enableTermSavingsPriceDisplay, siteId, coupon, helpers } =
usePlansGridContext();
const { isAnyPlanPriceDiscounted, setIsAnyPlanPriceDiscounted } = useHeaderPriceContext();
const {
current,
pricing: { currencyCode, originalPrice, discountedPrice, introOffer, billingPeriod },
Expand All @@ -56,14 +59,7 @@ const HeaderPrice = ( { planSlug, visibleGridPlans }: HeaderPriceProps ) => {
* We currently only support the `One time discount` in some currencies
*/
const isGridPlanOneTimeDiscounted = Number.isFinite( discountedPrice.monthly );
const isAnyVisibleGridPlanOneTimeDiscounted = visibleGridPlans.some( ( { pricing } ) =>
Number.isFinite( pricing.discountedPrice.monthly )
);

const isGridPlanOnIntroOffer = introOffer && ! introOffer.isOfferComplete;
const isAnyVisibleGridPlanOnIntroOffer = visibleGridPlans.some(
( { pricing } ) => pricing.introOffer && ! pricing.introOffer.isOfferComplete
);

const { prices } = usePlanPricingInfoFromGridPlans( { gridPlans: visibleGridPlans } );
const isLargeCurrency = useIsLargeCurrency( {
Expand All @@ -82,6 +78,30 @@ const HeaderPrice = ( { planSlug, visibleGridPlans }: HeaderPriceProps ) => {
useCheckPlanAvailabilityForPurchase: helpers?.useCheckPlanAvailabilityForPurchase,
} )?.[ termVariantPlanSlug ?? '' ];

const termVariantPrice =
termVariantPricing?.discountedPrice.monthly ?? termVariantPricing?.originalPrice.monthly ?? 0;
const planPrice = discountedPrice.monthly ?? originalPrice.monthly ?? 0;
const savings =
termVariantPrice > planPrice
? Math.floor( ( ( termVariantPrice - planPrice ) / termVariantPrice ) * 100 )
: 0;

useEffect( () => {
if (
isGridPlanOneTimeDiscounted ||
isGridPlanOnIntroOffer ||
( enableTermSavingsPriceDisplay && savings )
) {
setIsAnyPlanPriceDiscounted( true );
}
}, [
enableTermSavingsPriceDisplay,
isGridPlanOnIntroOffer,
isGridPlanOneTimeDiscounted,
savings,
setIsAnyPlanPriceDiscounted,
] );

if ( isWpcomEnterpriseGridPlan( planSlug ) || ! isPricedPlan ) {
return null;
}
Expand Down Expand Up @@ -160,14 +180,6 @@ const HeaderPrice = ( { planSlug, visibleGridPlans }: HeaderPriceProps ) => {
);
}

const termVariantPrice =
termVariantPricing?.discountedPrice.monthly ?? termVariantPricing?.originalPrice.monthly ?? 0;
const planPrice = discountedPrice.monthly ?? originalPrice.monthly ?? 0;
const savings =
termVariantPrice > planPrice
? Math.floor( ( ( termVariantPrice - planPrice ) / termVariantPrice ) * 100 )
: 0;

if ( enableTermSavingsPriceDisplay && termVariantPricing && savings ) {
return (
<div className="plans-grid-next-header-price">
Expand Down Expand Up @@ -205,11 +217,7 @@ const HeaderPrice = ( { planSlug, visibleGridPlans }: HeaderPriceProps ) => {
);
}

if (
isAnyVisibleGridPlanOneTimeDiscounted ||
isAnyVisibleGridPlanOnIntroOffer ||
enableTermSavingsPriceDisplay
) {
if ( isAnyPlanPriceDiscounted ) {
return (
<div className="plans-grid-next-header-price">
<div className="plans-grid-next-header-price__badge is-hidden">' '</div>
Expand Down
7 changes: 7 additions & 0 deletions packages/plans-grid-next/src/components/test/header-price.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,13 @@ jest.mock( '@automattic/data-stores', () => ( {
},
} ) );

jest.mock( '../shared/header-price/header-price-context', () => ( {
useHeaderPriceContext: () => ( {
isAnyPlanPriceDiscounted: false,
setIsAnyPlanPriceDiscounted: jest.fn(),
} ),
} ) );

import {
type PlanSlug,
PLAN_ANNUAL_PERIOD,
Expand Down

0 comments on commit 5e425d8

Please sign in to comment.