From 4d10eb425f787f390cb644232ec0f7b395355ef4 Mon Sep 17 00:00:00 2001
From: Randi Mays <19175324+randimays@users.noreply.github.com>
Date: Thu, 14 Nov 2024 17:41:22 -0600
Subject: [PATCH] VACMS-19386 Add SecondaryButtonGroup and RelatedLinks common
components
---
additional.d.ts | 13 ++--
.../common/relatedLinks/index.test.tsx | 60 +++++++++++++++++++
src/templates/common/relatedLinks/index.tsx | 51 ++++++++++++++++
.../relatedLinks/relatedLinks.stories.ts | 29 +++++++++
.../secondaryButtonGroup/index.test.tsx | 51 ++++++++++++++++
.../common/secondaryButtonGroup/index.tsx | 36 +++++++++++
.../secondaryButtonGroup.stories.ts | 30 ++++++++++
src/types/formatted/relatedLinks.ts | 10 ++++
8 files changed, 274 insertions(+), 6 deletions(-)
create mode 100644 src/templates/common/relatedLinks/index.test.tsx
create mode 100644 src/templates/common/relatedLinks/index.tsx
create mode 100644 src/templates/common/relatedLinks/relatedLinks.stories.ts
create mode 100644 src/templates/common/secondaryButtonGroup/index.test.tsx
create mode 100644 src/templates/common/secondaryButtonGroup/index.tsx
create mode 100644 src/templates/common/secondaryButtonGroup/secondaryButtonGroup.stories.ts
create mode 100644 src/types/formatted/relatedLinks.ts
diff --git a/additional.d.ts b/additional.d.ts
index dc6917c89..fb57d2930 100644
--- a/additional.d.ts
+++ b/additional.d.ts
@@ -22,14 +22,15 @@ declare module 'debug'
declare namespace JSX {
interface IntrinsicElements {
- 'va-alert'
- 'va-link'
- 'va-icon'
- 'va-button'
- 'va-breadcrumbs'
'va-accordion'
'va-accordion-item'
- 'va-on-this-page'
+ 'va-alert'
'va-back-to-top'
+ 'va-breadcrumbs'
+ 'va-button'
+ 'va-icon'
+ 'va-link'
+ 'va-link-action'
+ 'va-on-this-page'
}
}
diff --git a/src/templates/common/relatedLinks/index.test.tsx b/src/templates/common/relatedLinks/index.test.tsx
new file mode 100644
index 000000000..9699b4fc1
--- /dev/null
+++ b/src/templates/common/relatedLinks/index.test.tsx
@@ -0,0 +1,60 @@
+import { render } from '@testing-library/react'
+import { RelatedLinks } from '.'
+import { FormattedRelatedLinks } from '@/types/formatted/relatedLinks'
+
+describe('RelatedLinks Component', () => {
+ test('renders the correct number of links when there are multiple', () => {
+ const relatedLinks: FormattedRelatedLinks = {
+ links: [
+ {
+ uri: 'https://va.gov/burials-memorials/eligibility',
+ title: 'Eligibility for burial in a VA national cemetery',
+ summary: null,
+ },
+ {
+ uri: 'https://va.gov/burials-memorials/schedule-a-burial',
+ title: 'Schedule a burial for a Veteran or family member',
+ summary: null,
+ },
+ ],
+ sectionTitle: 'Related information',
+ }
+
+ const { container } = render()
+
+ expect(container.innerHTML).toContain(
+ 'Eligibility for burial in a VA national cemetery'
+ )
+ expect(container.innerHTML).toContain(
+ 'href="https://va.gov/burials-memorials/eligibility"'
+ )
+ expect(container.innerHTML).toContain(
+ 'Schedule a burial for a Veteran or family member'
+ )
+ expect(container.innerHTML).toContain(
+ 'href="https://va.gov/burials-memorials/schedule-a-burial"'
+ )
+ })
+
+ test('renders link correctly when there is only one', () => {
+ const oneLink: FormattedRelatedLinks = {
+ links: [
+ {
+ uri: 'https://va.gov/burials-memorials/schedule-a-burial',
+ title: 'Schedule a burial for a Veteran or family member',
+ summary: null,
+ },
+ ],
+ sectionTitle: 'VA benefits',
+ }
+
+ const { container } = render()
+
+ expect(container.innerHTML).toContain(
+ 'Schedule a burial for a Veteran or family member'
+ )
+ expect(container.innerHTML).toContain(
+ 'href="https://va.gov/burials-memorials/schedule-a-burial"'
+ )
+ })
+})
diff --git a/src/templates/common/relatedLinks/index.tsx b/src/templates/common/relatedLinks/index.tsx
new file mode 100644
index 000000000..8fc92ee8e
--- /dev/null
+++ b/src/templates/common/relatedLinks/index.tsx
@@ -0,0 +1,51 @@
+import { isEmpty } from 'lodash'
+import { FormattedRelatedLinks } from '@/types/formatted/relatedLinks'
+
+// General component used for one or more links with a line of descriptive text underneath
+// Does not map directly to any one Drupal type; it is simply a shared UI component
+export const RelatedLinks = ({
+ links,
+ sectionTitle,
+}: FormattedRelatedLinks): JSX.Element => {
+ if (isEmpty(links)) {
+ return null
+ }
+
+ let link
+ const renderLink = (uri, title, summary) => (
+ <>
+
+
+
+
+
+ {summary}
+ >
+ )
+
+ if (links.length === 1) {
+ link = links[0]
+ }
+
+ return (
+
+ {sectionTitle && (
+
+ {sectionTitle}
+
+ )}
+
+ {links.length > 1 && (
+
+ {links.map((link, index) => (
+ -
+ {renderLink(link.uri, link.title, link.summary)}
+
+ ))}
+
+ )}
+
+ {links.length === 1 && renderLink(link.uri, link.title, link.summary)}
+
+ )
+}
diff --git a/src/templates/common/relatedLinks/relatedLinks.stories.ts b/src/templates/common/relatedLinks/relatedLinks.stories.ts
new file mode 100644
index 000000000..113d63ffe
--- /dev/null
+++ b/src/templates/common/relatedLinks/relatedLinks.stories.ts
@@ -0,0 +1,29 @@
+import { Meta, StoryObj } from '@storybook/react'
+
+import { RelatedLinks } from '.'
+
+const meta: Meta = {
+ title: 'Common/Related Links',
+ component: RelatedLinks,
+}
+export default meta
+
+type Story = StoryObj
+
+export const Example: Story = {
+ args: {
+ sectionTitle: 'VA benefits',
+ links: [
+ {
+ uri: 'https://va.gov/burials-memorials/eligibility',
+ title: 'Eligibility for burial in a VA national cemetery',
+ summary: null,
+ },
+ {
+ uri: 'https://va.gov/burials-memorials/schedule-a-burial',
+ title: 'Schedule a burial for a Veteran or family member',
+ summary: null,
+ },
+ ],
+ },
+}
diff --git a/src/templates/common/secondaryButtonGroup/index.test.tsx b/src/templates/common/secondaryButtonGroup/index.test.tsx
new file mode 100644
index 000000000..b1889270e
--- /dev/null
+++ b/src/templates/common/secondaryButtonGroup/index.test.tsx
@@ -0,0 +1,51 @@
+import { render } from '@testing-library/react'
+import { SecondaryButtonGroup } from './'
+import { Button as FormattedButton } from '@/types/formatted/button'
+import { ParagraphComponent } from '@/types/formatted/paragraph'
+
+describe('SecondaryButtonGroup Component', () => {
+ test('renders action links correctly when there are multiple', () => {
+ const multipleButtons: ParagraphComponent[] = [
+ {
+ id: '1',
+ label: 'Button one',
+ url: 'https://www.va.gov/button-one',
+ },
+ {
+ id: '2',
+ label: 'Button two',
+ url: 'https://www.va.gov/button-two',
+ },
+ ]
+
+ const { container } = render(
+
+ )
+
+ expect(container.innerHTML).toContain('Button one')
+ expect(container.innerHTML).toContain(
+ 'href="https://www.va.gov/button-one"'
+ )
+ expect(container.innerHTML).toContain('Button two')
+ expect(container.innerHTML).toContain(
+ 'href="https://www.va.gov/button-two"'
+ )
+ })
+
+ test('renders action link correctly when there is only one', () => {
+ const oneButton: ParagraphComponent[] = [
+ {
+ id: '1',
+ label: 'Single button',
+ url: 'https://www.va.gov/single-button',
+ },
+ ]
+
+ const { container } = render()
+
+ expect(container.innerHTML).toContain('Single button')
+ expect(container.innerHTML).toContain(
+ 'href="https://www.va.gov/single-button"'
+ )
+ })
+})
diff --git a/src/templates/common/secondaryButtonGroup/index.tsx b/src/templates/common/secondaryButtonGroup/index.tsx
new file mode 100644
index 000000000..bbdb4d537
--- /dev/null
+++ b/src/templates/common/secondaryButtonGroup/index.tsx
@@ -0,0 +1,36 @@
+import { Button as FormattedButton } from '@/types/formatted/button'
+import { ParagraphComponent } from '@/types/formatted/paragraph'
+
+// Used for R&S pages; either a single blue CTA link or multiple
+export const SecondaryButtonGroup = ({
+ buttons,
+}: {
+ buttons: ParagraphComponent[]
+}): JSX.Element => {
+ if (buttons.length > 1) {
+ return (
+
+ {buttons?.map((button, index) => (
+ -
+
+
+ ))}
+
+ )
+ }
+
+ const button = buttons[0]
+
+ if (button) {
+ return (
+
+ )
+ }
+
+ return null
+}
diff --git a/src/templates/common/secondaryButtonGroup/secondaryButtonGroup.stories.ts b/src/templates/common/secondaryButtonGroup/secondaryButtonGroup.stories.ts
new file mode 100644
index 000000000..d9a8a35d8
--- /dev/null
+++ b/src/templates/common/secondaryButtonGroup/secondaryButtonGroup.stories.ts
@@ -0,0 +1,30 @@
+import { Meta, StoryObj } from '@storybook/react'
+
+import { SecondaryButtonGroup } from '.'
+
+const meta: Meta = {
+ title: 'Common/Secondary Button Group',
+ component: SecondaryButtonGroup,
+}
+export default meta
+
+type Story = StoryObj
+
+export const Example: Story = {
+ args: {
+ buttons: [
+ {
+ id: '1',
+ type: 'paragraph--button',
+ url: 'https://www.va.gov/careers-employment/vocational-rehabilitation/eligibility',
+ label: 'See eligibility for VR&E benefits',
+ },
+ {
+ id: '2',
+ type: 'paragraph--button',
+ url: 'https://www.va.gov/careers-employment/vocational-rehabilitation/how-to-apply',
+ label: 'Find out how to apply for VR&E benefits',
+ },
+ ],
+ },
+}
diff --git a/src/types/formatted/relatedLinks.ts b/src/types/formatted/relatedLinks.ts
new file mode 100644
index 000000000..e7e48897d
--- /dev/null
+++ b/src/types/formatted/relatedLinks.ts
@@ -0,0 +1,10 @@
+export type RelatedLink = {
+ title: string
+ uri: string
+ summary: string
+}
+
+export type FormattedRelatedLinks = {
+ sectionTitle: string
+ links: RelatedLink[]
+}