mirror of
https://github.com/TryGhost/Ghost.git
synced 2025-04-01 02:41:39 -05:00
Added more discoverable discount label
This commit is contained in:
parent
aa70a1b7b0
commit
ef38ff1132
2 changed files with 115 additions and 12 deletions
|
@ -69,6 +69,15 @@ export const PlanSectionStyles = `
|
|||
border-bottom-right-radius: 0;
|
||||
}
|
||||
|
||||
.gh-portal-plans-container.has-multiple-products.has-discount {
|
||||
margin-top: 40px;
|
||||
}
|
||||
|
||||
.gh-portal-plans-container.has-multiple-products.has-discount,
|
||||
.gh-portal-plans-container.has-multiple-products.has-discount .gh-portal-plan-section:last-of-type::before {
|
||||
border-top-right-radius: 0;
|
||||
}
|
||||
|
||||
.gh-portal-plans-container.is-change-plan.has-multiple-products .gh-portal-plan-section::before {
|
||||
border-top-left-radius: 0;
|
||||
border-top-right-radius: 0;
|
||||
|
@ -498,16 +507,16 @@ function PlanOptions({plans, selectedPlan, onPlanSelect, changePlan}) {
|
|||
break;
|
||||
}
|
||||
|
||||
const planClass = (isChecked ? 'gh-portal-plan-section checked' : 'gh-portal-plan-section');
|
||||
let planClass = isChecked ? 'gh-portal-plan-section checked' : 'gh-portal-plan-section';
|
||||
const planNameClass = planDetails.feature ? 'gh-portal-plan-name' : 'gh-portal-plan-name no-description';
|
||||
const featureClass = hasMultipleProductsFeature({site}) ? 'gh-portal-plan-featurewrapper hidden' : 'gh-portal-plan-featurewrapper';
|
||||
|
||||
return (
|
||||
<div className={planClass} key={id} onClick={e => onPlanSelect(e, id)}>
|
||||
{(hasMultipleProductsFeature({site}) ? <PlanDiscount discount={description} /> : ``)}
|
||||
<Checkbox name={name} id={id} isChecked={isChecked} onPlanSelect={onPlanSelect} />
|
||||
<h4 className={planNameClass}>{displayName}</h4>
|
||||
<PriceLabel currencySymbol={currencySymbol} price={price} interval={interval} />
|
||||
{(hasMultipleProductsFeature({site}) ? <PlanDiscount discount={description} /> : ``)}
|
||||
<div className={featureClass}>
|
||||
<PlanFeature feature={planDetails.feature} />
|
||||
{(changePlan && selectedPlan === id ? <span className='gh-portal-plan-current'>Current plan</span> : '')}
|
||||
|
@ -605,6 +614,21 @@ function getPlanClassNames({changePlan, cookiesDisabled, plans = [], showVertica
|
|||
}
|
||||
if (hasMultipleProductsFeature({site})) {
|
||||
className += ' has-multiple-products';
|
||||
|
||||
const filteredPlans = plans.filter(d => d.id !== 'free');
|
||||
const monthlyPlan = plans.find((d) => {
|
||||
return d.name === 'Monthly' && !d.description && d.interval === 'month';
|
||||
});
|
||||
const yearlyPlan = plans.find((d) => {
|
||||
return d.name === 'Yearly' && !d.description && d.interval === 'year';
|
||||
});
|
||||
|
||||
if (filteredPlans.length === 2 && monthlyPlan && yearlyPlan) {
|
||||
const discount = calculateDiscount(monthlyPlan.amount, yearlyPlan.amount);
|
||||
if (discount) {
|
||||
className += ' has-discount';
|
||||
}
|
||||
}
|
||||
}
|
||||
return className;
|
||||
}
|
||||
|
|
|
@ -4,6 +4,7 @@ import {ReactComponent as CheckmarkIcon} from '../../images/icons/checkmark.svg'
|
|||
import {getSiteProducts, getCurrencySymbol, getPriceString, getStripeAmount, isCookiesDisabled, getMemberActivePrice, getProductFromPrice} from '../../utils/helpers';
|
||||
import AppContext from '../../AppContext';
|
||||
import ActionButton from './ActionButton';
|
||||
import calculateDiscount from '../../utils/discount';
|
||||
|
||||
export const ProductsSectionStyles = ({site}) => {
|
||||
const products = getSiteProducts({site});
|
||||
|
@ -30,6 +31,11 @@ export const ProductsSectionStyles = ({site}) => {
|
|||
letter-spacing: 0.3px;
|
||||
text-transform: uppercase;
|
||||
margin: 0 6px;
|
||||
min-width: 180px;
|
||||
}
|
||||
|
||||
.gh-portal-priceoption-label.monthly {
|
||||
text-align: right;
|
||||
}
|
||||
|
||||
.gh-portal-products-priceswitch .gh-portal-for-switch label, .gh-portal-for-switch .container {
|
||||
|
@ -59,6 +65,48 @@ export const ProductsSectionStyles = ({site}) => {
|
|||
transform: translateX(19px);
|
||||
}
|
||||
|
||||
.gh-portal-discount-label {
|
||||
position: absolute;
|
||||
top: -19px;
|
||||
left: -1px;
|
||||
right: -1px;
|
||||
color: var(--brandcolor);
|
||||
font-size: 1.1rem;
|
||||
font-weight: 600;
|
||||
letter-spacing: 0.3px;
|
||||
text-transform: uppercase;
|
||||
padding: 0px 4px;
|
||||
text-align: center;
|
||||
}
|
||||
|
||||
.gh-portal-discount-label:before {
|
||||
position: absolute;
|
||||
content: "";
|
||||
top: 0;
|
||||
right: 0;
|
||||
bottom: 0;
|
||||
left: 0;
|
||||
background: var(--brandcolor);
|
||||
opacity: 0.15;
|
||||
border-radius: 2px 2px 0 0;
|
||||
}
|
||||
|
||||
.gh-portal-products-priceswitch .gh-portal-discount-label {
|
||||
position: relative;
|
||||
font-size: 1.3rem;
|
||||
letter-spacing: 0.3px;
|
||||
margin: 0 0 0 4px;
|
||||
padding: 2px 6px;
|
||||
top: unset;
|
||||
left: unset;
|
||||
right: unset;
|
||||
width: unset;
|
||||
}
|
||||
|
||||
.gh-portal-products-priceswitch. gh-portal-discount-label:before {
|
||||
border-radius: 5px;
|
||||
}
|
||||
|
||||
.gh-portal-products-grid {
|
||||
display: grid;
|
||||
grid-template-columns: repeat(${productColumns(noOfProducts)}, minmax(0, ${(productColumns(noOfProducts) <= 3 ? `360px` : `300px`)}));
|
||||
|
@ -85,6 +133,13 @@ export const ProductsSectionStyles = ({site}) => {
|
|||
}
|
||||
}
|
||||
|
||||
@media (max-width: 360px) {
|
||||
div:not(.gh-portal-products-priceswitch) .gh-portal-discount-label {
|
||||
font-size: 0.9rem;
|
||||
white-space: nowrap;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
.gh-portal-product-card {
|
||||
position: relative;
|
||||
|
@ -242,13 +297,6 @@ export const ProductsSectionStyles = ({site}) => {
|
|||
padding: 0;
|
||||
}
|
||||
|
||||
.gh-portal-discount-label {
|
||||
color: var(--brandcolor);
|
||||
font-size: 1.2rem;
|
||||
margin-top: 4px;
|
||||
margin-bottom: -12px;
|
||||
}
|
||||
|
||||
@media (max-width: 480px) {
|
||||
.gh-portal-products {
|
||||
margin: 0 -32px;
|
||||
|
@ -295,6 +343,12 @@ export const ProductsSectionStyles = ({site}) => {
|
|||
.gh-portal-product-description {
|
||||
grid-column: 2 / 3;
|
||||
margin-bottom: 0px;
|
||||
text-align: left;
|
||||
}
|
||||
|
||||
.gh-portal-singleproduct-benefits .gh-portal-product-description {
|
||||
text-align: center;
|
||||
padding-bottom: 12px;
|
||||
}
|
||||
|
||||
.gh-portal-product-price {
|
||||
|
@ -632,20 +686,44 @@ function FreeProductCard() {
|
|||
);
|
||||
}
|
||||
|
||||
function ProductPriceSwitch({selectedInterval, setSelectedInterval}) {
|
||||
function YearlyDiscount({discount}) {
|
||||
if (discount === 0) {
|
||||
return null;
|
||||
}
|
||||
|
||||
return (
|
||||
<>
|
||||
<span className="gh-portal-discount-label">{discount}% discount</span>
|
||||
</>
|
||||
);
|
||||
}
|
||||
|
||||
function ProductPriceSwitch({products, selectedInterval, setSelectedInterval}) {
|
||||
const {site} = useContext(AppContext);
|
||||
const {selectedProduct} = useContext(ProductsContext);
|
||||
const {portal_plans: portalPlans} = site;
|
||||
if (!portalPlans.includes('monthly') || !portalPlans.includes('yearly')) {
|
||||
return null;
|
||||
}
|
||||
|
||||
let yearlyDiscount = 0;
|
||||
|
||||
if (selectedProduct !== 'free') {
|
||||
const product = products.find(prod => prod.id === selectedProduct);
|
||||
yearlyDiscount = calculateDiscount(product.monthlyPrice.amount, product.yearlyPrice.amount);
|
||||
}
|
||||
|
||||
return (
|
||||
<div className="gh-portal-products-priceswitch">
|
||||
<span className="gh-portal-priceoption-label">Monthly</span>
|
||||
<span className="gh-portal-priceoption-label monthly">Monthly</span>
|
||||
<Switch id='product-interval' onToggle={(e) => {
|
||||
const interval = selectedInterval === 'month' ? 'year' : 'month';
|
||||
setSelectedInterval(interval);
|
||||
}} checked={selectedInterval === 'year'} />
|
||||
<span className="gh-portal-priceoption-label">Yearly</span>
|
||||
<span className="gh-portal-priceoption-label">
|
||||
Yearly
|
||||
<YearlyDiscount discount={yearlyDiscount} />
|
||||
</span>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
@ -719,6 +797,7 @@ function ProductsSection({onPlanSelect, products, type = null}) {
|
|||
}}>
|
||||
<section className={className}>
|
||||
<ProductPriceSwitch
|
||||
products={products}
|
||||
selectedInterval={activeInterval}
|
||||
setSelectedInterval={setSelectedInterval}
|
||||
/>
|
||||
|
|
Loading…
Add table
Reference in a new issue