ui: Hoist DC menu into the top navigation bar (#10034)

* Add story for %main-nav-vertical plus additions for hoisting menu items

* Make sure we don't source app.css twice

* Hoist the DC menu
pull/10054/head
John Cowen 2021-04-16 12:13:06 +01:00 committed by GitHub
parent a11ea6254e
commit d008dc3d85
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
9 changed files with 183 additions and 44 deletions

View File

@ -25,6 +25,16 @@
@extend %main-nav-horizontal; @extend %main-nav-horizontal;
margin-left: auto; margin-left: auto;
} }
%main-nav-vertical-hoisted {
top: 11px;
}
%main-nav-vertical-hoisted > .popover-menu > label > button {
@extend %main-nav-horizontal-action;
border: none;
}
%main-nav-vertical-hoisted.is-active > label > * {
@extend %main-nav-horizontal-action-active;
}
%main-nav-sidebar, %main-nav-sidebar,
main { main {
@extend %transition-pushover; @extend %transition-pushover;

View File

@ -18,14 +18,54 @@
<:main-nav> <:main-nav>
{{#if @dc}} {{#if @dc}}
<ul> <ul>
<li
class="dcs"
data-test-datacenter-menu
>
<PopoverMenu
aria-label="Datacenter"
@position="left"
as |components|>
<BlockSlot @name="trigger">
{{@dc.Name}}
</BlockSlot>
<BlockSlot @name="menu">
{{#let components.MenuItem components.MenuSeparator as |MenuItem MenuSeparator|}}
<DataSource
@src="/*/*/datacenters"
@onchange={{action (mut @dcs) value="data"}}
@loading="lazy"
/>
{{#each (sort-by 'Name' @dcs) as |item|}}
<MenuItem
data-test-datacenter-picker
class={{concat (if (eq @dc.Name item.Name) 'is-active') (if item.Local ' is-local') }}
@href={{href-mut (hash dc=item.Name)}}
>
<BlockSlot @name="label">
{{item.Name}}
{{#if item.Local}}
<span>Local</span>
{{/if}}
</BlockSlot>
</MenuItem>
{{/each}}
{{/let}}
</BlockSlot>
</PopoverMenu>
</li>
{{#let (or this.nspaces @nspaces) as |nspaces|}} {{#let (or this.nspaces @nspaces) as |nspaces|}}
{{#if (can "choose nspaces" nspaces=nspaces)}} {{#if (can "choose nspaces" nspaces=nspaces)}}
<li <li
class="nspaces" class="nspaces"
data-test-nspace-menu data-test-nspace-menu
> >
Namespace <PopoverMenu
<PopoverMenu @position="left" as |components api|> aria-label="Namespace"
@position="left"
as |components api|>
<BlockSlot @name="trigger"> <BlockSlot @name="trigger">
{{@nspace.Name}} {{@nspace.Name}}
</BlockSlot> </BlockSlot>
@ -70,41 +110,6 @@
</li> </li>
{{/if}} {{/if}}
{{/let}} {{/let}}
<li
class="dcs"
data-test-datacenter-menu
>
Datacenter
<PopoverMenu @position="left" as |components|>
<BlockSlot @name="trigger">
{{@dc.Name}}
</BlockSlot>
<BlockSlot @name="menu">
{{#let components.MenuItem components.MenuSeparator as |MenuItem MenuSeparator|}}
<DataSource
@src="/*/*/datacenters"
@onchange={{action (mut @dcs) value="data"}}
@loading="lazy"
/>
{{#each (sort-by 'Name' @dcs) as |item|}}
<MenuItem
data-test-datacenter-picker
class={{concat (if (eq @dc.Name item.Name) 'is-active') (if item.Local ' is-local') }}
@href={{href-mut (hash dc=item.Name)}}
>
<BlockSlot @name="label">
{{item.Name}}
{{#if item.Local}}
<span>Local</span>
{{/if}}
</BlockSlot>
</MenuItem>
{{/each}}
{{/let}}
</BlockSlot>
</PopoverMenu>
</li>
{{#if (can "read services")}} {{#if (can "read services")}}
<li data-test-main-nav-services class={{if (is-href 'dc.services' @dc.Name) 'is-active'}}> <li data-test-main-nav-services class={{if (is-href 'dc.services' @dc.Name) 'is-active'}}>
<a href={{href-to 'dc.services' @dc.Name}}>Services</a> <a href={{href-to 'dc.services' @dc.Name}}>Services</a>

View File

@ -1,4 +1,11 @@
.hashicorp-consul { .hashicorp-consul {
[role="banner"] nav .dcs {
@extend %main-nav-vertical-hoisted;
left: 100px;
}
[role="banner"] nav .dcs .popover-menu[aria-label]::before {
display: none;
}
[role="banner"] a svg { [role="banner"] a svg {
fill: var(--brand-600); fill: var(--brand-600);
} }

View File

@ -0,0 +1,71 @@
---
class: css
---
# MainNavVertical
Used for styles of vertically orientated main application menus/navigation.
Menu item active state is applied on `.is-active` `<li>` elements. Additionally you can use the following placeholders for setting certin states manually:
- `%menu-nav-vertical-action-active` The 'active' or currently selected state.
- `%menu-nav-vertical-action-intent` The highlighted state, usually for `:hover`
and `:focus`.
`%menu-nav-vertical-hoisted` will 'hoist' an `<li>` element to the top of the
containing block, the containing block defaults to the current viewport. If
you need to define a different ancestor for a containing block you can use
`transform` (see below).
```hbs preview-template
<div class="wrapper">
<nav class="main-nav-vertical in-viewport">
<ul>
<li role="separator">Title</li>
<li>
<a href="">One</a>
</li>
<li class="is-active">
<a href="">Two (is-active)</a>
</li>
<li class="with-intent">
<a href="">Three (with-intent)</a>
</li>
<li class="hoisted">
<a href="">Four (hoisted)</a>
</li>
<li role="separator">Title</li>
<li class="custom-active">
<a href="">One (custom-active)</a>
</li>
<li>
<a href="">Two</a>
</li>
<li>
<a href="">Three</a>
</li>
</ul>
</nav>
</div>
```
```css preview-template
.main-nav-vertical {
@extend %main-nav-vertical;
}
.main-nav-vertical li.hoisted {
@extend %main-nav-vertical-hoisted;
}
.main-nav-vertical .with-intent > * {
@extend %main-nav-vertical-action-intent;
}
.main-nav-vertical .custom-active > * {
@extend %main-nav-vertical-action-active;
}
.wrapper {
/* a transform is required to mark this element as the containing block */
/* for hoisting, otherwise the viewport is the containing block */
transform: translate(0, 0);
background-color: var(--gray-600);
padding-top: 64px;
}
```

View File

@ -0,0 +1,29 @@
#docfy-demo-preview-main-nav-vertical {
.main-nav-vertical {
@extend %main-nav-vertical;
}
.main-nav-vertical li.hoisted {
@extend %main-nav-vertical-hoisted;
}
.main-nav-vertical .with-intent > * {
@extend %main-nav-vertical-action-intent;
}
.main-nav-vertical .custom-active > * {
@extend %main-nav-vertical-action-active;
}
.wrapper {
/* a transform is required to mark this element as the containing block */
/* for hoisting, otherwise the viewport is the containing block */
transform: translate(0, 0);
background-color: var(--gray-600);
padding-top: 64px;
}
// TODO: Reduce the need for these debug overrides
.main-nav-vertical {
position: static;
height: auto;
}
.main-nav-vertical ul {
margin-bottom: 0 !important;
}
}

View File

@ -40,3 +40,8 @@
%main-nav-vertical-menu-panel { %main-nav-vertical-menu-panel {
min-width: 248px; min-width: 248px;
} }
%main-nav-vertical-hoisted {
visibility: visible;
position: fixed;
z-index: 10;
}

View File

@ -40,6 +40,12 @@
background-color: var(--gray-150); background-color: var(--gray-150);
border-color: var(--gray-999); border-color: var(--gray-999);
} }
%main-nav-vertical .popover-menu[aria-label]::before {
content: attr(aria-label);
display: block;
margin-top: -.5rem;
margin-bottom: .5rem;
}
%main-nav-vertical .is-local span:last-of-type { %main-nav-vertical .is-local span:last-of-type {
@extend %pill-200; @extend %pill-200;
color: var(--gray-000); color: var(--gray-000);

View File

@ -1,6 +1,9 @@
@import './base/reset/index'; @import './app';
@import './base/index';
@import 'prism-coldark-dark'; @import 'prism-coldark-dark';
// temporary component debugging setup
@import 'consul-ui/components/main-nav-vertical/debug';
.docs { .docs {
.tabular-collection, .tabular-collection,
.list-collection { .list-collection {
@ -31,8 +34,12 @@
} }
ol, ol,
ul { ul {
list-style-position: inside; list-style-position: outside;
margin-bottom: 1rem; margin-bottom: 1rem;
margin-left: 2rem;
}
ul {
list-style-type: disc;
} }
} }
.docfy-demo { .docfy-demo {

View File

@ -7,10 +7,9 @@ module.exports = ({ appName, environment, rootURL, config }) => `
<link rel="icon" type="image/png" href="${rootURL}assets/favicon-32x32.png" sizes="32x32"> <link rel="icon" type="image/png" href="${rootURL}assets/favicon-32x32.png" sizes="32x32">
<link rel="icon" type="image/png" href="${rootURL}assets/favicon-16x16.png" sizes="16x16"> <link rel="icon" type="image/png" href="${rootURL}assets/favicon-16x16.png" sizes="16x16">
<link integrity="" rel="stylesheet" href="${rootURL}assets/vendor.css"> <link integrity="" rel="stylesheet" href="${rootURL}assets/vendor.css">
<link integrity="" rel="stylesheet" href="${rootURL}assets/${appName}.css"> <link integrity="" rel="stylesheet" href="${rootURL}assets/${
${ environment === 'development' ? 'debug' : appName
environment === 'development' ? `<link rel="stylesheet" href="${rootURL}assets/debug.css">` : `` }.css">
}
${ ${
environment === 'test' ? `<link rel="stylesheet" href="${rootURL}assets/test-support.css">` : `` environment === 'test' ? `<link rel="stylesheet" href="${rootURL}assets/test-support.css">` : ``
} }