-
Notifications
You must be signed in to change notification settings - Fork 1
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
## π 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
1 parent
b6cf8ae
commit 72816da
Showing
18 changed files
with
1,301 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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 |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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`. |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,5 @@ | ||
{ | ||
"name": "account_tree", | ||
"style": "outlined", | ||
"source": "@material-symbols/svg-300" | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,5 @@ | ||
{ | ||
"name": "door_open", | ||
"style": "outlined", | ||
"source": "@material-symbols/svg-300" | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,5 @@ | ||
{ | ||
"name": "key", | ||
"style": "outlined", | ||
"source": "@material-symbols/svg-300" | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,5 @@ | ||
{ | ||
"name": "shield", | ||
"style": "outlined", | ||
"source": "@material-symbols/svg-300" | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,5 @@ | ||
{ | ||
"name": "view_list", | ||
"style": "outlined", | ||
"source": "@material-symbols/svg-300" | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,5 @@ | ||
{ | ||
"name": "widgets", | ||
"style": "outlined", | ||
"source": "@material-symbols/svg-300" | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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} /> |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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"); | ||
} |
Oops, something went wrong.