Skip to content
Merged
Show file tree
Hide file tree
Changes from 39 commits
Commits
Show all changes
43 commits
Select commit Hold shift + click to select a range
aa9c55e
Organised CTA card settings panel
sanne-san Feb 17, 2025
a669fa9
Fixed Colour selection tests
ronaldlangeveld Feb 19, 2025
68cbe9e
Added stopPropagation to Image Upload Form
ronaldlangeveld Feb 20, 2025
3cf4452
Fixed image upload related tests and bugs
ronaldlangeveld Feb 20, 2025
6538fbc
Updated CTA Card tests
ronaldlangeveld Feb 20, 2025
a762a4e
Fixed CTA Node tests
ronaldlangeveld Feb 20, 2025
d691484
Fixed linting
ronaldlangeveld Feb 20, 2025
2eaba1d
Added button color as hex
ronaldlangeveld Feb 24, 2025
cc9490a
Added drag drop to CTA card img button
ronaldlangeveld Feb 24, 2025
32a3ee9
Added img icon to color picker
ronaldlangeveld Feb 25, 2025
efc5d4a
Revert "Added img icon to color picker"
ronaldlangeveld Feb 25, 2025
62adaf2
Fixed swatch test
ronaldlangeveld Feb 25, 2025
29742de
Revert "Revert "Added img icon to color picker""
ronaldlangeveld Feb 25, 2025
51f4a98
Added test for finding img icon
ronaldlangeveld Feb 25, 2025
49b1443
Added image icon on Header
ronaldlangeveld Feb 25, 2025
d5b958d
Updated proptypes
ronaldlangeveld Feb 25, 2025
09bb024
Added image string to headercard
ronaldlangeveld Feb 25, 2025
3cc4b08
Improved button icon test
ronaldlangeveld Feb 25, 2025
91860d2
Fixed console errors when inserting CTA card
kevinansfield Feb 25, 2025
3ea83f2
Cleaned up repetition in `<MediaPlaceholder>`
kevinansfield Feb 25, 2025
c9f3dbe
Renamed call-to-action card test file
kevinansfield Feb 25, 2025
60cb43a
Fixed errors coming from useMoveable
kevinansfield Feb 25, 2025
acc7d53
Fixed background color popover not closing on click outside
kevinansfield Feb 25, 2025
69d31a8
Improved close on click outside for button color picker
kevinansfield Feb 25, 2025
a898422
Fixed display of button accent color in demo
kevinansfield Feb 25, 2025
6f8074d
Fixed `return undefined` from React components in MediaPlaceholder
kevinansfield Feb 25, 2025
9233c48
Merge branch 'main' into ui-updates-reorg
kevinansfield Feb 27, 2025
917f349
Merge branch 'main' into ui-updates-reorg
kevinansfield Feb 27, 2025
bcef9be
Extracted modified UI components to beta copies
kevinansfield Feb 28, 2025
f327cb2
fixed CalloutCard background color setting layout
kevinansfield Mar 3, 2025
a67e9d7
fixed default button text color
kevinansfield Mar 3, 2025
286ff6e
fixed tests
kevinansfield Mar 3, 2025
11cf9a6
Removed tooltip from layout buttons in call to action card
sanne-san Mar 3, 2025
ec790b2
added missing prop types to <MediaUploaderBeta>
kevinansfield Mar 3, 2025
ba2fc95
avoid global Error override in MediaPlaceholderBeta.stories
kevinansfield Mar 3, 2025
5c48803
renamed IconButton export in ButtonGroupBeta to avoid duplicate naming
kevinansfield Mar 3, 2025
44b0feb
added missing prop types to ButtonGroupBeta
kevinansfield Mar 3, 2025
467aee8
added default fn for `onRemoveMedia` prop to avoid errors if omitted
kevinansfield Mar 3, 2025
993d118
Added a divider to call-to-action card settings panel
sanne-san Mar 3, 2025
253f995
added isRequired to required props in ButtonGroupBeta
kevinansfield Mar 3, 2025
c18f8de
fixed incorrect borderStyle conditionals in MediaUploaderBeta
kevinansfield Mar 3, 2025
044dd2b
added missing openImageEditor prop type to MediaUploaderBeta
kevinansfield Mar 3, 2025
44e0a29
added roles and aria-checked to ButtonGroupBeta elements
kevinansfield Mar 3, 2025
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -8,15 +8,15 @@ export class CallToActionNode extends generateDecoratorNode({
properties: [
{name: 'layout', default: 'minimal'},
{name: 'textValue', default: '', wordCount: true},
{name: 'showButton', default: false},
{name: 'buttonText', default: ''},
{name: 'showButton', default: true},
{name: 'buttonText', default: 'Learn more'},
{name: 'buttonUrl', default: ''},
{name: 'buttonColor', default: ''},
{name: 'buttonTextColor', default: ''},
{name: 'buttonColor', default: '#000000'}, // Where colour is customisable, we should use hex values
{name: 'buttonTextColor', default: '#ffffff'},
{name: 'hasSponsorLabel', default: true},
{name: 'sponsorLabel', default: '<p><span style="white-space: pre-wrap;">SPONSORED</span></p>'},
{name: 'backgroundColor', default: 'grey'},
{name: 'imageUrl', default: null},
{name: 'backgroundColor', default: 'grey'}, // Since this is one of a few fixed options, we stick to colour names.
{name: 'imageUrl', default: ''},
{name: 'imageWidth', default: null},
{name: 'imageHeight', default: null}
]
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -33,8 +33,7 @@ function ctaCardTemplate(dataset) {
</div>
` : ''}
${dataset.showButton ? `
<a href="${dataset.buttonUrl}" class="kg-cta-button ${buttonAccent}"
${buttonStyle}>
<a href="${dataset.buttonUrl}" class="kg-cta-button ${buttonAccent}" ${buttonStyle}>
${dataset.buttonText}
</a>
` : ''}
Expand All @@ -46,12 +45,12 @@ function ctaCardTemplate(dataset) {
}

function emailCTATemplate(dataset, options = {}) {
const buttonStyle = dataset.buttonColor === 'accent'
? `color: ${dataset.buttonTextColor};`
const buttonStyle = dataset.buttonColor === 'accent'
? `color: ${dataset.buttonTextColor};`
: `background-color: ${dataset.buttonColor}; color: ${dataset.buttonTextColor};`;

let imageDimensions;

if (dataset.imageUrl && dataset.imageWidth && dataset.imageHeight) {
imageDimensions = {
width: dataset.imageWidth,
Expand Down Expand Up @@ -98,9 +97,10 @@ function emailCTATemplate(dataset, options = {}) {
<table border="0" cellpadding="0" cellspacing="0" class="kg-cta-button-wrapper">
<tr>
<td class="${dataset.buttonColor === 'accent' ? 'kg-style-accent' : ''}" style="${buttonStyle}">
<a href="${dataset.buttonUrl}"
<a href="${dataset.buttonUrl}"
class="kg-cta-button ${dataset.buttonColor === 'accent' ? 'kg-style-accent' : ''}"
style="${buttonStyle}">
style="${buttonStyle}"
>
${dataset.buttonText}
</a>
</td>
Expand Down Expand Up @@ -146,9 +146,10 @@ function emailCTATemplate(dataset, options = {}) {
<table border="0" cellpadding="0" cellspacing="0" width="100%">
<tr>
<td class="kg-cta-button-wrapper ${dataset.buttonColor === 'accent' ? 'kg-style-accent' : ''}" style="${buttonStyle}">
<a href="${dataset.buttonUrl}"
class="kg-cta-button ${dataset.buttonColor === 'accent' ? 'kg-style-accent' : ''}"
style="${buttonStyle}">
<a href="${dataset.buttonUrl}"
class="kg-cta-button ${dataset.buttonColor === 'accent' ? 'kg-style-accent' : ''}"
style="${buttonStyle}"
>
${dataset.buttonText}
</a>
</td>
Expand Down
26 changes: 12 additions & 14 deletions packages/kg-default-nodes/test/nodes/call-to-action.test.js
Original file line number Diff line number Diff line change
Expand Up @@ -81,11 +81,11 @@ describe('CallToActionNode', function () {
callToActionNode.textValue = 'This is a cool advertisement';
callToActionNode.textValue.should.equal('This is a cool advertisement');

callToActionNode.showButton.should.equal(false);
callToActionNode.showButton = true;
callToActionNode.showButton.should.equal(true);
callToActionNode.showButton = false;
callToActionNode.showButton.should.equal(false);

callToActionNode.buttonText.should.equal('');
callToActionNode.buttonText.should.equal('Learn more');
callToActionNode.buttonText = 'click me';
callToActionNode.buttonText.should.equal('click me');

Expand All @@ -97,11 +97,11 @@ describe('CallToActionNode', function () {
callToActionNode.sponsorLabel = 'This post is brought to you by our sponsors';
callToActionNode.sponsorLabel.should.equal('This post is brought to you by our sponsors');

callToActionNode.buttonColor.should.equal('');
callToActionNode.buttonColor = 'red';
callToActionNode.buttonColor.should.equal('red');
callToActionNode.buttonColor.should.equal('#000000');
callToActionNode.buttonColor = '#ffffff';
callToActionNode.buttonColor.should.equal('#ffffff');

callToActionNode.buttonTextColor.should.equal('');
callToActionNode.buttonTextColor.should.equal('#ffffff');
callToActionNode.buttonTextColor = 'black';
callToActionNode.buttonTextColor.should.equal('black');

Expand All @@ -110,10 +110,10 @@ describe('CallToActionNode', function () {
callToActionNode.hasSponsorLabel.should.equal(false);

callToActionNode.backgroundColor.should.equal('grey');
callToActionNode.backgroundColor = '#654321';
callToActionNode.backgroundColor.should.equal('#654321');
callToActionNode.backgroundColor = 'red';
callToActionNode.backgroundColor.should.equal('red');

should(callToActionNode.imageUrl).be.null();
callToActionNode.imageUrl.should.equal('');
callToActionNode.imageUrl = 'http://blog.com/image1.jpg';
callToActionNode.imageUrl.should.equal('http://blog.com/image1.jpg');

Expand Down Expand Up @@ -259,7 +259,6 @@ describe('CallToActionNode', function () {
buttonText: 'Get access now',
buttonTextColor: '#000000',
buttonUrl: 'http://someblog.com/somepost',
hasImage: true,
hasSponsorLabel: true,
sponsorLabel: '<p><span style="white-space: pre-wrap;">SPONSORED</span></p>',
imageUrl: '/content/images/2022/11/koenig-lexical.jpg',
Expand All @@ -276,7 +275,7 @@ describe('CallToActionNode', function () {
html.should.containEql('Get access now');
html.should.containEql('http://someblog.com/somepost');
html.should.containEql('<p><span style="white-space: pre-wrap;">SPONSORED</span></p>'); // because hasSponsorLabel is true
html.should.containEql('/content/images/size/w64h64/2022/11/koenig-lexical.jpg'); // because hasImage is true
html.should.containEql('/content/images/size/w64h64/2022/11/koenig-lexical.jpg');
html.should.containEql('This is a new CTA Card via email.');
}));

Expand All @@ -289,7 +288,6 @@ describe('CallToActionNode', function () {
buttonText: 'Get access now',
buttonTextColor: '#000000',
buttonUrl: 'http://someblog.com/somepost',
hasImage: true,
hasSponsorLabel: true,
sponsorLabel: '<p><span style="white-space: pre-wrap;">SPONSORED</span></p>',
imageUrl: '/content/images/2022/11/koenig-lexical.jpg',
Expand All @@ -301,7 +299,7 @@ describe('CallToActionNode', function () {
const {element} = callToActionNode.exportDOM(exportOptions);

const html = element.outerHTML.toString();
html.should.containEql('/content/images/size/w256h256/2022/11/koenig-lexical.jpg'); // because hasImage is true
html.should.containEql('/content/images/size/w256h256/2022/11/koenig-lexical.jpg');
}));

it('renders email with img width and height when immersive', editorTest(function () {
Expand Down
62 changes: 62 additions & 0 deletions packages/koenig-lexical/src/components/ui/ButtonGroupBeta.jsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,62 @@
import PropTypes from 'prop-types';
import React from 'react';

import {Tooltip} from './Tooltip';
import {usePreviousFocus} from '../../hooks/usePreviousFocus';

export function ButtonGroupBeta({buttons = [], selectedName, onClick, hasTooltip = true}) {
return (
<div className="flex">
<ul className="flex items-center justify-evenly rounded-lg bg-grey-100 font-sans text-md font-normal text-white">
{buttons.map(({label, name, Icon, dataTestId, ariaLabel}) => (
<ButtonGroupIconButton
key={`${name}-${label}`}
ariaLabel={ariaLabel}
dataTestId={dataTestId}
hasTooltip={hasTooltip}
Icon={Icon}
label={label}
name={name}
selectedName={selectedName}
onClick={onClick}
/>
))}
</ul>
</div>
);
}

export function ButtonGroupIconButton({dataTestId, onClick, label, ariaLabel, name, selectedName, Icon, hasTooltip}) {
const isActive = name === selectedName;

const {handleMousedown, handleClick} = usePreviousFocus(onClick, name);

return (
<li className="mb-0">
<button
aria-label={ariaLabel || label}
className={`group relative flex h-7 w-8 cursor-pointer items-center justify-center rounded-lg text-black dark:text-white dark:hover:bg-grey-900 ${isActive ? 'border border-grey-300 bg-white shadow-xs dark:bg-grey-900' : '' } ${Icon ? '' : 'text-[1.3rem] font-bold'}`}
data-testid={dataTestId}
type="button"
onClick={handleClick}
onMouseDown={handleMousedown}
>
{Icon ? <Icon className="size-4 stroke-2" /> : label}
{(Icon && label && hasTooltip) && <Tooltip label={label} />}
</button>
Comment on lines +46 to +48
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

🛠️ Refactor suggestion

Potential accessibility issue with Icon-only buttons.

When a button only has an icon (no visible text), ensure it always has an accessible name via aria-label. The current implementation falls back to the label if ariaLabel isn't provided, but there's a case where both could be missing.

                {Icon ? <Icon className="size-4 stroke-2" /> : label}
                {(Icon && label && hasTooltip) && <Tooltip label={label} />}
+               {(Icon && !label && !ariaLabel) && console.warn('ButtonGroupIconButton with Icon is missing both label and ariaLabel props')}

</li>
);
}

ButtonGroupBeta.propTypes = {
selectedName: PropTypes.oneOf(['regular', 'wide', 'full', 'split', 'center', 'left', 'small', 'medium', 'large', 'grid', 'list', 'minimal', 'immersive']),
hasTooltip: PropTypes.bool,
onClick: PropTypes.func,
buttons: PropTypes.arrayOf(PropTypes.shape({
label: PropTypes.string,
name: PropTypes.string,
Icon: PropTypes.func,
dataTestId: PropTypes.string,
ariaLabel: PropTypes.string
}))
};
Original file line number Diff line number Diff line change
@@ -0,0 +1,48 @@
import ImgFullIcon from '../../assets/icons/kg-img-full.svg?react';
import ImgRegularIcon from '../../assets/icons/kg-img-regular.svg?react';
import ImgWideIcon from '../../assets/icons/kg-img-wide.svg?react';
import React from 'react';
import {ButtonGroupBeta, ButtonGroupIconButton} from './ButtonGroupBeta';

const story = {
title: 'Generic/Button group (beta)',
component: ButtonGroupBeta,
subcomponents: {ButtonGroupIconButton},
parameters: {
status: {
type: 'functional'
}
},
argTypes: {
selectedName: {control: 'select', options: ['regular', 'wide', 'full']}
}
};
export default story;

const Template = (args) => {
return (
<ButtonGroupBeta {...args} />
);
};

export const CardWidth = Template.bind({});
CardWidth.args = {
selectedName: 'regular',
buttons: [
{
label: 'Regular',
name: 'regular',
Icon: ImgRegularIcon
},
{
label: 'Wide',
name: 'wide',
Icon: ImgWideIcon
},
{
label: 'Full',
name: 'full',
Icon: ImgFullIcon
}
]
};
Original file line number Diff line number Diff line change
@@ -0,0 +1,93 @@
import PlusIcon from '../../assets/icons/plus.svg?react';
import React, {useState} from 'react';
import {Tooltip} from './Tooltip';
import {useClickOutside} from '../../hooks/useClickOutside';
import {usePreviousFocus} from '../../hooks/usePreviousFocus';

export function ColorOptionButtonsBeta({buttons = [], selectedName, onClick}) {
const [isOpen, setIsOpen] = useState(false);
const componentRef = React.useRef(null);

const selectedButton = buttons.find(button => button.name === selectedName);

// Close the swatch popover when clicking outside of it
useClickOutside(isOpen, componentRef, () => setIsOpen(false));

return (
<div ref={componentRef} className="relative">
<button
className={`relative size-6 cursor-pointer rounded-full ${selectedName ? 'p-[2px]' : 'border border-grey-200 dark:border-grey-800'}`}
data-testid="color-options-button"
type="button"
onClick={() => setIsOpen(!isOpen)}
>
{selectedName && (
<div className="absolute inset-0 rounded-full bg-clip-content p-[3px]" style={{
background: 'conic-gradient(hsl(360,100%,50%),hsl(315,100%,50%),hsl(270,100%,50%),hsl(225,100%,50%),hsl(180,100%,50%),hsl(135,100%,50%),hsl(90,100%,50%),hsl(45,100%,50%),hsl(0,100%,50%))',
WebkitMask: 'linear-gradient(#fff 0 0) content-box, linear-gradient(#fff 0 0)',
WebkitMaskComposite: 'xor',
mask: 'linear-gradient(#fff 0 0) content-box, linear-gradient(#fff 0 0)',
maskComposite: 'exclude'
}} />
)}
<span
className={`${selectedButton?.color || ''} block size-full rounded-full border-2 border-white`}
></span>
</button>

{/* Color options popover */}
{isOpen && (
<div className="absolute -right-3 bottom-full z-10 mb-2 rounded-lg bg-white px-3 py-2 shadow" data-testid="color-options-popover">
<div className="flex">
<ul className="flex w-full items-center justify-between rounded-md font-sans text-md font-normal text-white">
{buttons.map(({label, name, color}) => (
name !== 'image' ?
<ColorButton
key={`${name}-${label}`}
color={color}
data-testid={`color-options-${name}-button`}
label={label}
name={name}
selectedName={selectedName}
onClick={(title) => {
onClick(title);
setIsOpen(false);
}}
/>
:
<li key='background-image' className={`mb-0 flex size-[3rem] cursor-pointer items-center justify-center rounded-full border-2 ${selectedName === name ? 'border-green' : 'border-transparent'}`} data-testid="background-image-color-button" type="button" onClick={() => onClick(name)}>
<span className="border-1 flex size-6 items-center justify-center rounded-full border border-black/5">
<PlusIcon className="size-3 stroke-grey-700 stroke-2 dark:stroke-grey-500 dark:group-hover:stroke-grey-100" />
</span>
</li>
))}
</ul>
</div>
</div>
)}
</div>
);
}

export function ColorButton({onClick, label, name, color, selectedName}) {
const isActive = name === selectedName;

const {handleMousedown, handleClick} = usePreviousFocus(onClick, name);
return (
<li className="mb-0">
<button
aria-label={label}
className={`group relative flex size-6 cursor-pointer items-center justify-center rounded-full border-2 ${isActive ? 'border-green' : 'border-transparent'}`}
data-test-id={`color-picker-${name}`}
type="button"
onClick={handleClick}
onMouseDown={handleMousedown}
>
<span
className={`${color} size-[1.8rem] rounded-full border`}
></span>
<Tooltip label={label} />
</button>
</li>
);
}
Loading
Loading