Skip to content

Commit

Permalink
feat: ForgeLayout (#1545)
Browse files Browse the repository at this point in the history
## πŸ“ Changes

- Adds ForgeLayout

## βœ… Checklist

Easy UI has certain UX standards that must be met. In general,
non-trivial changes should meet the following criteria:

- [x] Visuals match Design Specs in Figma
- [x] Stories accompany any component changes
- [x] Code is in accordance with our style guide
- [x] Design tokens are utilized
- [x] Unit tests accompany any component changes
- [x] TSDoc is written for any API surface area
- [x] Specs are up-to-date
- [x] Console is free from warnings
- [x] No accessibility violations are reported
- [x] Cross-browser check is performed (Chrome, Safari, Firefox)
- [x] Changeset is added

~Strikethrough~ any items that are not applicable to this pull request.
  • Loading branch information
stephenjwatkins authored Dec 19, 2024
1 parent b6cf8ae commit 72816da
Show file tree
Hide file tree
Showing 18 changed files with 1,301 additions and 0 deletions.
6 changes: 6 additions & 0 deletions .changeset/tasty-news-own.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
---
"@easypost/easy-ui-icons": minor
"@easypost/easy-ui": minor
---

feat: ForgeLayout
120 changes: 120 additions & 0 deletions documentation/specs/ForgeLayout.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,120 @@
# `ForgeLayout` Component Specification

## Overview

`ForgeLayout` defines the header, nav, and main content areas of a Forge product page.

### Prior Art

- [Primer `<PageLayout />`](https://primer.style/design/components/page-layout/react)
- [Paste `<SidebarNavigation /`>](https://paste.twilio.design/components/sidebar-navigation)

---

## Design

`ForgeLayout` will be a compound component consisting of `ForgeLayout`, `ForgeLayout.Nav`, `ForgeLayout.Header`, and `ForgeLayout.Content`.

`ForgeLayout` is highly composable. Subcomponents within a `ForgeLayout` can be replaced as needed. Subcomponents are lightweight wrappers with built-in styles and constraints.

`ForgeLayout` is concerned only with presentational structure. It is meant to be wrapped by an app layout that may include app-specific business logic and configuration.

`ForgeLayout` can be in an `expanded` or `collapsed` navigational state by using the `navState` prop. When `expanded`, the navigation is present, along with any relevant header controls. When `collapsed`, the navigation is hidden, and the relevant controls are presented in the header.

`ForgeLayout` is aware of a global `mode` prop. When passed `test`, the shell is decorated with a color to indicate a non-production environment. The `mode` can be changed with the `ModeSwitcher` control.

### API

```tsx
import { ForgeLayout } from "@easypost/easy-ui/ForgeLayout";

function App() {
return (
<ForgeLayout mode="test" navState="expanded">
<ForgeLayout.Nav>
<ForgeLayout.NavLink href="/1" iconSymbol={Icon}>
Item 1
</ForgeLayout.NavLink>
<ForgeLayout.NavSection title={<>Title</>}>
<ForgeLayout.NavLink href="/2" iconSymbol={Icon}>
Item 2
</ForgeLayout.NavLink>
<ForgeLayout.NavLink href="/3" iconSymbol={Icon}>
Item 3
</ForgeLayout.NavLink>
</ForgeLayout.NavSection>
<ForgeLayout.NavSection title={<>Title</>}>
<ForgeLayout.NavLink href="/4" iconSymbol={Icon}>
Item 4
</ForgeLayout.NavLink>
<ForgeLayout.NavLink href="/5" iconSymbol={Icon}>
Item 5
</ForgeLayout.NavLink>
</ForgeLayout.NavSection>
</ForgeLayout.Nav>
<ForgeLayout.Header>
<ForgeLayout.Controls visibleWhenNavStateIs="collapsed">
<ForgeLayout.BreadrumbsNavigation>
<ForgeLayout.BackButton onClick={() => {}}>
Back
</ForgeLayout.BackButton>
<ForgeLayout.Breadrumbs>
<ForgeLayout.Breadrumb>Breadcrumb</ForgeLayout.Breadrumb>
<ForgeLayout.Breadrumb>Breadcrumb</ForgeLayout.Breadrumb>
</ForgeLayout.Breadrumbs>
</ForgeLayout.BreadrumbsNavigation>
</ForgeLayout.Controls>
<ForgeLayout.Controls visibleWhenNavStateIs="expanded">
<ForgeLayout.ModeSwitcher onModeChange={action("Mode changed!")} />
<ForgeLayout.Search value={"search"} onChange={() => {}} />
</ForgeLayout.Controls>
<ForgeLayout.Actions>
<ForgeLayout.MenuAction
accessibilityLabel="Action 1"
iconSymbol={AlarmIcon}
renderBadge={() => <ForgeLayout.ActionBadge />}
>
<Menu.Overlay onAction={action("Menu item clicked!")}>
<Menu.Item>Action 1:1</Menu.Item>
<Menu.Item>Action 1:2</Menu.Item>
</Menu.Overlay>
</ForgeLayout.MenuAction>
<ForgeLayout.MenuAction
accessibilityLabel="Action 2"
iconSymbol={SupportIcon}
>
<Menu.Overlay onAction={action("Menu item clicked!")}>
<Menu.Item>Action 2:1</Menu.Item>
<Menu.Item>Action 2:2</Menu.Item>
</Menu.Overlay>
</ForgeLayout.MenuAction>
<ForgeLayout.LinkAction
href="/4"
accessibilityLabel="Action 3"
iconSymbol={SettingsIcon}
/>
</ForgeLayout.Actions>
</ForgeLayout.Header>
<ForgeLayout.Content>Page Content</ForgeLayout.Content>
</ForgeLayout>
);
}
```

---

## Behavior

### Accessibility

- `ForgeLayout.Header` will render as `header`
- `ForgeLayout.Content` will render as `main`
- `ForgeLayout.Nav` will be rendered as `nav` with associated `aria-label`
- `ForgeLayout.NavLink` will render as `<a>`
- Selected nav links will be decorated as `aria-current="page"`

### Dependencies

- `Text`
- `useLink`
- Will use `EasyUIProvider`'s navigation hooks to support client-side links. See [client side routing](https://react-spectrum.adobe.com/react-aria/routing.html#routerprovider). This was added as part of `NexusLayout`.
5 changes: 5 additions & 0 deletions easy-ui-icons/src/AccountTree.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
{
"name": "account_tree",
"style": "outlined",
"source": "@material-symbols/svg-300"
}
5 changes: 5 additions & 0 deletions easy-ui-icons/src/DoorOpen.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
{
"name": "door_open",
"style": "outlined",
"source": "@material-symbols/svg-300"
}
5 changes: 5 additions & 0 deletions easy-ui-icons/src/Key.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
{
"name": "key",
"style": "outlined",
"source": "@material-symbols/svg-300"
}
5 changes: 5 additions & 0 deletions easy-ui-icons/src/Shield.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
{
"name": "shield",
"style": "outlined",
"source": "@material-symbols/svg-300"
}
5 changes: 5 additions & 0 deletions easy-ui-icons/src/ViewList.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
{
"name": "view_list",
"style": "outlined",
"source": "@material-symbols/svg-300"
}
5 changes: 5 additions & 0 deletions easy-ui-icons/src/Widgets.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
{
"name": "widgets",
"style": "outlined",
"source": "@material-symbols/svg-300"
}
40 changes: 40 additions & 0 deletions easy-ui-react/src/ForgeLayout/ForgeLayout.mdx
Original file line number Diff line number Diff line change
@@ -0,0 +1,40 @@
import React from "react";
import { Canvas, Meta, ArgTypes, Controls } from "@storybook/blocks";
import { ForgeLayout } from "./ForgeLayout";
import * as ForgeLayoutStories from "./ForgeLayout.stories";

<Meta of={ForgeLayoutStories} />

# ForgeLayout

`ForgeLayout` defines the header, main content, and multipage content areas of a Nexus product page.

<Canvas of={ForgeLayoutStories.Default} />

`ForgeLayout` is a compound component consisting of `ForgeLayout`, `ForgeLayout.Nav`, `ForgeLayout.Header`, and `ForgeLayout.Content`.

`ForgeLayout` also provides components for building navigation with `ForgeLayout.Nav`, `ForgeLayout.NavLink`, and `ForgeLayout.NavSection`.

`ForgeLayout` is highly composable. Subcomponents within a `ForgeLayout` can be replaced as needed. Subcomponents are simple lightweight wrappers with built-in styles and constraints.

## Test Mode

`ForgeLayout` supports an obvious visual indicator for test mode with the `mode="test"` prop.

<Canvas of={ForgeLayoutStories.TestMode} />

<Controls of={ForgeLayoutStories.TestMode} include={["mode"]} />

## Collapsed Navigation

`ForgeLayout` supports a distraction-free content mode with the `navState="collapsed"` prop.

<Canvas of={ForgeLayoutStories.Collapsed} />

<Controls of={ForgeLayoutStories.Collapsed} include={["navState"]} />

## Properties

### ForgeLayout

<ArgTypes of={ForgeLayout} />
100 changes: 100 additions & 0 deletions easy-ui-react/src/ForgeLayout/ForgeLayout.module.scss
Original file line number Diff line number Diff line change
@@ -0,0 +1,100 @@
@use "../styles/common" as *;
@use "../styles/unstyled";

.ForgeLayout {
@include component-token("forge-layout", "header-height", 56px);
@include component-token("forge-layout", "shell-gutter", 20px);
@include component-token("forge-layout", "menu-border-color", transparent);
@include component-token(
"forge-layout",
"header-border-color",
design-token("color.neutral.300")
);
@include component-token("forge-layout", "header-border-width", 1px);

display: flex;
flex-direction: row;
align-items: flex-start;
padding-left: component-token("forge-layout", "shell-gutter");
padding-right: component-token("forge-layout", "shell-gutter");
gap: component-token("forge-layout", "shell-gutter");
min-height: 100svh;
background-color: design-token("color.neutral.025");
position: relative;
}

.modeTest {
@include component-token(
"forge-layout",
"menu-border-color",
design-token("color.warning.600")
);
@include component-token(
"forge-layout",
"header-border-color",
design-token("color.warning.600")
);
@include component-token("forge-layout", "header-border-width", 2px);
}

.backgroundDecoration01 {
background-image: url(""),
url("");
background-repeat: no-repeat;
background-position:
-40px -90px,
calc(100% + 24px) calc(100% + 24px);
}

.body {
flex: 1;
display: flex;
flex-direction: column;
}

.header {
display: flex;
flex-wrap: nowrap;
align-items: center;
justify-content: space-between;
min-width: 0;
border-bottom: 2px solid transparent;
z-index: calc(#{design-token("z-index.nav")} + 1);
}

.content {
padding-top: component-token("forge-layout", "shell-gutter");
padding-bottom: component-token("forge-layout", "shell-gutter");
}

.controls {
display: flex;
align-items: center;
gap: design-token("space.2");
}

.fauxContainer {
position: absolute;
top: 0;
left: 0;
width: 100%;
height: 100%;

display: flex;
flex-direction: column;
pointer-events: none;
}

.fauxHeader {
background-color: design-token("color.neutral.025");
border-bottom: component-token("forge-layout", "header-border-width") solid
component-token("forge-layout", "header-border-color");
z-index: design-token("z-index.nav");
}

.fauxHeader,
.header {
position: sticky;
top: 0;
height: component-token("forge-layout", "header-height");
}
Loading

0 comments on commit 72816da

Please sign in to comment.