Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: Implement page footer with USWDS components #1285

Open
wants to merge 72 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
72 commits
Select commit Hold shift + click to select a range
a014680
initial commit
snmln Dec 3, 2024
4e4fef7
adding outstanding footer imports
snmln Dec 3, 2024
7f300ea
pushing up initial slim work
snmln Dec 4, 2024
545697e
Move uswds footer behind feature flag
AliceR Dec 5, 2024
6f58910
Disable eslint to not warn, explanation in comment
AliceR Dec 5, 2024
f838337
Move Connections component behind the same FF
AliceR Dec 5, 2024
9f059df
Hide new USWDS footer behind feature flag (#1287)
snmln Dec 5, 2024
e856157
usa foot implementation
snmln Dec 5, 2024
a82d132
Merge branch 'main' into 1135-Refactor-Layout-Components-Footer
AliceR Dec 9, 2024
65dacea
adding uswds style infrastructure
snmln Dec 9, 2024
1aafd0c
Merge branch '1135-Refactor-Layout-Components-Footer' into 1135-slim-…
snmln Dec 9, 2024
18f0173
implementing styling and creating interface
snmln Dec 9, 2024
60398ed
wiring up config
snmln Dec 9, 2024
a5b4502
correcting styles
snmln Dec 10, 2024
c47e331
Merge branch '1135-Refactor-Layout-Components-Footer' into 1135-slim-…
snmln Dec 10, 2024
e629882
correct ts and lint errors
snmln Dec 10, 2024
a7a91ec
correctin merge conflict issues.
snmln Dec 10, 2024
d0a4cd4
updating styles
snmln Dec 10, 2024
59dd97e
dialing in css
snmln Dec 10, 2024
4bdf994
refactor: 1135 slim footer (#1308)
snmln Dec 10, 2024
e2afeed
Fix hideFooter prop name
AliceR Dec 11, 2024
9f4a2fd
Add back comments to .env
AliceR Dec 11, 2024
fe545d4
Remove duplicated styles
AliceR Dec 11, 2024
f224268
wiring up menu items
snmln Dec 11, 2024
02e5b61
fix: hideFooter prop name, small cleanup (#1315)
AliceR Dec 12, 2024
89a9421
fix: svg logo to not require css styles
AliceR Dec 11, 2024
0755ef6
Pass svg logo as prop to footer
AliceR Dec 12, 2024
4e3a21a
fix: svg logo (#1316)
AliceR Dec 12, 2024
2d175f1
adding comment in createDynamicNavMenuList
snmln Dec 12, 2024
6bf3149
wirign up dynamic link creation
snmln Dec 12, 2024
bb5dcad
Merge branch '1135-Refactor-Layout-Components-Footer' into 1135-wirin…
snmln Dec 12, 2024
886912e
cleaning up veda.config and styling
snmln Dec 13, 2024
8c3c769
fix: 1135 wiring up veda config (#1320)
AliceR Dec 16, 2024
d830c03
Add comments to .env
AliceR Dec 16, 2024
5089c7e
Revert "Add comments to .env"
AliceR Dec 16, 2024
98fd9ed
Merge remote-tracking branch 'origin' into 1135-Refactor-Layout-Compo…
AliceR Dec 16, 2024
c7846f8
Remove TODO as completed
AliceR Dec 16, 2024
fc7d58e
Add page footer to exported components
AliceR Dec 16, 2024
75f6fd3
createing jest tests
snmln Dec 16, 2024
d72afea
correcting test
snmln Dec 16, 2024
d43dccc
fix:1335 creating footer test suite (#1330)
AliceR Dec 17, 2024
a9fd5ce
Add secondary footer content to footer config
AliceR Dec 17, 2024
e039620
fix: Add secondary footer content to footer config (#1331)
snmln Dec 17, 2024
c13d9c1
Merge branch 'main' into 1135-Refactor-Layout-Components-Footer
AliceR Dec 19, 2024
13c9cb9
Update color mappings, use system token instead of theme token
AliceR Dec 19, 2024
8ff3230
Merge branch 'main' into 1135-Refactor-Layout-Components-Footer
AliceR Dec 19, 2024
b45d108
Replace values with theme vars
AliceR Dec 19, 2024
3584e15
Add role and aria-label to svg logo
AliceR Dec 19, 2024
c149d5b
Merge branch 'main' into 1135-Refactor-Layout-Components-Footer
AliceR Dec 19, 2024
23850fd
corrcting layout, line-height, and coloring issues
snmln Dec 20, 2024
0c78b01
updating documentation
snmln Dec 23, 2024
c4f1b24
updating documentation
snmln Dec 23, 2024
b5fad30
updating tests
snmln Dec 24, 2024
2267afc
Merge remote-tracking branch 'origin/main' into 1135-Refactor-Layout-…
snmln Jan 9, 2025
c0d279a
Update app/scripts/components/common/layout-root/index.tsx
snmln Jan 9, 2025
8cfe1ce
implementing isUswdsFooterEnabled rename
snmln Jan 9, 2025
6846e20
moving return to top button outside of footer
snmln Jan 9, 2025
522dab0
fixing footer uswds import
snmln Jan 9, 2025
e8a7b14
eslint removal
snmln Jan 9, 2025
bda0bde
updating styles
snmln Jan 9, 2025
73eff21
removing link in logo
snmln Jan 9, 2025
9bbd492
implementing link properties
snmln Jan 14, 2025
9877502
updating test names
snmln Jan 14, 2025
dc2d615
returntotop button
snmln Jan 14, 2025
8b758bf
updating default settings and tests
snmln Jan 14, 2025
09babe8
updating types import
snmln Jan 14, 2025
48737c4
updating return to top focus
snmln Jan 15, 2025
eb8ad3e
implementing feedback
snmln Jan 15, 2025
2ac6245
updating css
snmln Jan 15, 2025
6712d33
Merge branch 'main' into 1135-Refactor-Layout-Components-Footer
snmln Jan 15, 2025
46c2f17
type corrections
snmln Jan 15, 2025
1d26ca0
Update app/scripts/components/common/types.ts
snmln Jan 16, 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
4 changes: 3 additions & 1 deletion .env
Original file line number Diff line number Diff line change
Expand Up @@ -15,4 +15,6 @@ GOOGLE_FORM = 'https://docs.google.com/forms/d/e/1FAIpQLSfGcd3FDsM3kQIOVKjzdPn4f
SHOW_CONFIGURABLE_COLOR_MAP = 'TRUE'

# Enables the refactor page header component that uses the USWDS design system
ENABLE_USWDS_PAGE_HEADER = 'TRUE'
ENABLE_USWDS_PAGE_HEADER = 'TRUE'
# Enables the refactor page footer component that uses the USWDS design system
ENABLE_USWDS_PAGE_FOOTER = 'TRUE'
20 changes: 19 additions & 1 deletion app/scripts/components/common/layout-root/index.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -10,13 +10,18 @@ import { useDeepCompareEffect } from 'use-deep-compare';
import styled from 'styled-components';
import { Outlet } from 'react-router';
import { reveal } from '@devseed-ui/animation';
import { NavLink } from 'react-router-dom';

import {
getBannerFromVedaConfig,
getCookieConsentFromVedaConfig,
getSiteAlertFromVedaConfig
} from 'veda';
import MetaTags from '../meta-tags';
import PageFooter from '../page-footer';
import PageFooterLegacy from '../page-footer-legacy';
import NasaLogoColor from '../nasa-logo-color';

const Banner = React.lazy(() => import('../banner'));
const SiteAlert = React.lazy(() => import('../site-alert'));
const CookieConsent = React.lazy(() => import('../cookie-consent'));
Expand All @@ -31,9 +36,11 @@ import {
mainNavItems,
subNavItems
} from '$components/common/page-header/default-config';
import { checkEnvFlag } from '$utils/utils';

const appTitle = process.env.APP_TITLE;
const appDescription = process.env.APP_DESCRIPTION;
const isUswdsFooterEnabled = checkEnvFlag(process.env.ENABLE_USWDS_PAGE_FOOTER);

export const PAGE_BODY_ID = 'pagebody';

Expand Down Expand Up @@ -64,6 +71,7 @@ function LayoutRoot(props: { children?: ReactNode }) {
useEffect(() => {
// When there is no cookie consent form set up
!cookieConsentContent && setGoogleTagManager();
// eslint-disable-next-line react-hooks/exhaustive-deps
}, []);

const { title, thumbnail, description, hideFooter } =
Expand Down Expand Up @@ -105,7 +113,17 @@ function LayoutRoot(props: { children?: ReactNode }) {
/>
)}
</PageBody>
<PageFooter isHidden={hideFooter} />
{isUswdsFooterEnabled ? (
<PageFooter
mainNavItems={mainNavItems}
subNavItems={subNavItems}
hideFooter={hideFooter}
snmln marked this conversation as resolved.
Show resolved Hide resolved
linkProperties={{ LinkElement: NavLink, pathAttributeKeyName: 'to' }}
logoSvg={<NasaLogoColor />}
/>
snmln marked this conversation as resolved.
Show resolved Hide resolved
) : (
<PageFooterLegacy hideFooter={hideFooter} />
)}
</Page>
);
}
Expand Down
452 changes: 245 additions & 207 deletions app/scripts/components/common/nasa-logo-color.js

Large diffs are not rendered by default.

Original file line number Diff line number Diff line change
Expand Up @@ -109,7 +109,7 @@ const InfoList = styled.dl`
}
`;

function PageFooter(props) {
function PageFooterLegacy(props) {
const nowDate = new Date();

return (
Expand Down Expand Up @@ -174,8 +174,8 @@ function PageFooter(props) {
);
}

export default PageFooter;
export default PageFooterLegacy;

PageFooter.propTypes = {
PageFooterLegacy.propTypes = {
isHidden: T.bool
};
16 changes: 16 additions & 0 deletions app/scripts/components/common/page-footer/default-config.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,16 @@
import { getFooterSettingsFromVedaConfig } from 'veda';
const defaultFooterSettings = {
secondarySection: {
division: 'NASA EarthData 2024',
version: 'BETA VERSION',
title: 'NASA Official',
name: 'Manil Maskey',
to: '[email protected]',
type: 'email'
},
returnToTop: true
};
const footerSettings =
getFooterSettingsFromVedaConfig() ?? defaultFooterSettings;

export { footerSettings };
97 changes: 97 additions & 0 deletions app/scripts/components/common/page-footer/footer.test.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,97 @@
import React, { ComponentType } from 'react';
import { render, screen } from '@testing-library/react';
import { navItems } from '../../../../../mock/veda.config.js';
import NasaLogoColor from '../nasa-logo-color';
import { NavItem } from '../page-header/types.js';

import PageFooter from './index';

const mockMainNavItems: NavItem[] = navItems.mainNavItems;
const mockSubNavItems: NavItem[] = navItems.subNavItems;
Copy link
Collaborator

@sandrahoang686 sandrahoang686 Jan 9, 2025

Choose a reason for hiding this comment

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

These mocks are mocking what looks to be used for the Header which includes dropdowns which footer doesn't worry about, we should probably explicity define the config for footer separately. And then test the dynamicness?

// const mockFooterSettings = footerSettings;
const hideFooter = false;
const mockLinkProperties = {
pathAttributeKeyName: 'to',
LinkElement: 'a' as unknown as ComponentType
};
jest.mock('./default-config', () => ({
footerSettings: {
secondarySection: {
division: 'NASA EarthData 2024',
version: 'BETA VERSION',
title: 'NASA Official',
name: 'test',
to: '[email protected]',
type: 'email'
},
returnToTop: true
}
}));

describe('PageFooter', () => {
afterEach(() => {
jest.clearAllMocks();
});
test('renders the PageFooter', () => {
render(
<PageFooter
mainNavItems={mockMainNavItems}
subNavItems={mockSubNavItems}
hideFooter={hideFooter}
logoSvg={<NasaLogoColor />}
linkProperties={mockLinkProperties}
/>
);
const footerElement = document.querySelector('footer');

expect(footerElement).toBeInTheDocument();
expect(footerElement).not.toHaveClass('display-none');
});

test('renders correct buttons and links', () => {
render(
<PageFooter
mainNavItems={mockMainNavItems}
subNavItems={mockSubNavItems}
hideFooter={hideFooter}
logoSvg={<NasaLogoColor />}
linkProperties={mockLinkProperties}
/>
);
expect(screen.getByText('Data Catalog')).toBeInTheDocument();
expect(screen.getByText('Exploration')).toBeInTheDocument();
expect(screen.getByText('Stories')).toBeInTheDocument();

expect(screen.getByText('About')).toBeInTheDocument();
expect(screen.getByText('Return to top')).toBeInTheDocument();
});
});

describe('PageFooter dynamic settings', () => {
test('Hide footer should function correctly', () => {
jest.mock('./default-config', () => ({
footerSettings: {
secondarySection: {
division: 'NASA EarthData 2024',
version: 'BETA VERSION',
title: 'NASA Official',
name: 'test',
to: '[email protected]',
type: 'email'
},
returnToTop: true
}
}));
render(
<PageFooter
linkProperties={mockLinkProperties}
mainNavItems={mockMainNavItems}
subNavItems={mockSubNavItems}
hideFooter={true}
logoSvg={<NasaLogoColor />}
/>
);
const footerElement = document.querySelector('footer');
expect(footerElement).toHaveClass('display-none');
});
});
152 changes: 152 additions & 0 deletions app/scripts/components/common/page-footer/index.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,152 @@
import React, { useMemo } from 'react';
import { Icon } from '@trussworks/react-uswds';
import { DropdownNavLink, NavLinkItem } from '../types';
import { ActionNavItem, NavItemType } from '../page-header/types';
import { NavItemCTA } from '../page-header/nav/nav-item-cta';
import ReturnToTopButton from './return-to-top-button';
import { footerSettings } from './default-config';

import { LinkProperties } from '$types/veda';
import {
USWDSFooter,
USWDSFooterNav,
USWDSAddress
} from '$components/common/uswds';

interface PageFooterProps {
//use of NavItem is causing issues with TS and throwing erros in the .
Copy link
Collaborator

Choose a reason for hiding this comment

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

Suggested change
//use of NavItem is causing issues with TS and throwing erros in the .

still an issue?

mainNavItems: (NavLinkItem | DropdownNavLink | ActionNavItem)[];
subNavItems: (NavLinkItem | DropdownNavLink | ActionNavItem)[];
snmln marked this conversation as resolved.
Show resolved Hide resolved
hideFooter?: boolean;
logoSvg?: SVGElement | JSX.Element;
linkProperties: LinkProperties;
}

//TODO: clean up PageFooterProps, Unexpected any. Specify a different interface.
Copy link
Collaborator

Choose a reason for hiding this comment

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

Suggested change
//TODO: clean up PageFooterProps, Unexpected any. Specify a different interface.


export default function PageFooter({
mainNavItems,
subNavItems,
hideFooter,
logoSvg,
linkProperties
}: PageFooterProps) {
const { returnToTop, secondarySection } = footerSettings;
const FooterNavItemInternalLink = (item) => {
const { item: linkContents, linkClasses, linkProperties } = item;
if (linkProperties.LinkElement) {
const path = {
[linkProperties.pathAttributeKeyName]: linkContents.to
};
const LinkElement = linkProperties.LinkElement;
return (
<LinkElement
key={linkContents.id}
{...path}
className={linkClasses}
id={linkContents.id}
>
<span>{linkContents.title}</span>
</LinkElement>
);
}
// If the link provided is invalid, do not render the element
return null;
};

const createNavElement = (navItems, linkClasses) => {
//removing 'dropdown' items from array
const cleanedNavItems = navItems.filter((a) => {
if (a.type !== 'dropdown') {
return a;
}
});

return cleanedNavItems.map((item) => {
switch (item.type) {
case NavItemType.ACTION:
return <NavItemCTA item={item} customClasses={linkClasses} />;

case NavItemType.EXTERNAL_LINK:
return (
<a className={linkClasses} href={item.to} key={item.id}>
{item.title}
</a>
);
case NavItemType.INTERNAL_LINK:
return (
<FooterNavItemInternalLink
item={item}
linkClasses={linkClasses}
linkProperties={linkProperties}
/>
);
snmln marked this conversation as resolved.
Show resolved Hide resolved

default:
return <></>;
}
});
};

const primaryItems = useMemo(
() => createNavElement(mainNavItems, 'usa-footer__primary-link'),
[mainNavItems]
);
const secondaryItems = useMemo(
() =>
createNavElement(subNavItems, 'usa-link text-base-dark text-underline'),
[mainNavItems]
);

return (
<USWDSFooter
size='slim'
returnToTop={returnToTop && <ReturnToTopButton />}
className={hideFooter && 'display-none'}
primary={
<div className='grid-row usa-footer__primary-container footer_primary_container'>
<div className='mobile-lg:grid-col-8'>
<USWDSFooterNav
aria-label='Footer navigation'
size='slim'
links={primaryItems}
/>
</div>
<div className='tablet:grid-col-4'>
<USWDSAddress
size='slim'
className='flex-justify-end'
items={secondaryItems}
/>
</div>
</div>
}
secondary={
<div id='footer_secondary_container' className='grid-row'>
<div id='logo-container'>
<a id='logo-container-link' href='#'>
{logoSvg as JSX.Element}
<span className='footer-text'>
{secondarySection.division} • {secondarySection.version}
</span>
</a>
</div>
<div className='grid-col-4 footer-text grid-gap-6 flex-justify-end'>
snmln marked this conversation as resolved.
Show resolved Hide resolved
<span>{secondarySection.title}: </span>
<a
key={secondarySection.type}
href={`mailto:${secondarySection.to}`}
className='text-primary-light'
>
<Icon.Mail
className='margin-right-1 width-205 height-auto position-relative'
id='mail_icon'
/>
{secondarySection.name}
</a>
</div>
</div>
}
/>
);
}
Loading
Loading