Skip to content

Commit

Permalink
Tabs: unify vertical tabs styles (WordPress#65387)
Browse files Browse the repository at this point in the history
* Remove inserter pattern overrides

* Make font weights 400 (inactive) and 500 (active)

* Apply styles only when vertical.

* Make vertical indicator theme accent color at 4% opacity.

* Make height 48px.

* Add radius.

* Also use hover styles in focus-visible.

* Fix indicator not visible in inserter > patterns/media.

* Adjust padding.

* Tweak focus ring.

* Wrap long labels.

* Add chevron and fix a few minor details.

* Fix merge issues.

* Fix focus indicator (gets cropped with the new overflow auto setting)

* Fix unwanted chevron.

* Fix unwanted nested scrollbar in inserter > patterns/media vertical tabs.

* Switch to transform for performance.

* Adjust border-radius based on scaling factor.

Co-authored-by: DaniGuardiola <daniguardiola@git.wordpress.org>
Co-authored-by: ciampo <mciampini@git.wordpress.org>
Co-authored-by: stokesman <presstoke@git.wordpress.org>
Co-authored-by: jameskoster <jameskoster@git.wordpress.org>
Co-authored-by: jasmussen <joen@git.wordpress.org>
Co-authored-by: t-hamano <wildworks@git.wordpress.org>
Co-authored-by: ndiego <ndiego@git.wordpress.org>
Co-authored-by: jeryj <jeryj@git.wordpress.org>
* Apply feedback.

* Add changelog entry.

* Switch to `padding-inline`.

* Remove unnecessary styles.

* Fix horizontal tabs height.

* Remove more unnecessary styles (padding).

* Make horizontal padding specific to inline.

* Make flex/whitespace styles more explicit.

* Make scroll margin specific to vertical tabs.

* The "inline" in inline-flex is unnecessary and confusing, removed it.

* Remove unnecessary position: relative

* Make resets more explicit

* Remove unnecessary text-align.

* Improve comment

* Remove unnecessary margin-left

* Clean up TabList styles.

* Adjust text-align.

* Clean up selector

* Fix focus indicator

* Clean up position: relative.

* Fix typo.

* Add position: relative back.

* Improve focus indicator when selectOnMove is enabled.

* Add fade in effect to chevron when selectOnMove is enabled.

* Use [data-focus-visible] consistently.

* Styles clean up.

* Add comment for clarity.

* Move scroll-margin to the right place.

* Use CSS variable for accuracy.

* Fix overflow.

* Skip failing test for Safari :(

* Fix flashing issue.

* Transition chevron only on selected and not on hover or focus-visible.

* Improve chevron opacity transition with suggested value.

* fix changelog
  • Loading branch information
DaniGuardiola authored and karthick-murugan committed Nov 13, 2024
1 parent 69a803e commit 5a59680
Show file tree
Hide file tree
Showing 7 changed files with 131 additions and 97 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -2,14 +2,10 @@
* WordPress dependencies
*/
import { usePrevious, useReducedMotion } from '@wordpress/compose';
import { isRTL } from '@wordpress/i18n';
import {
__experimentalHStack as HStack,
FlexBlock,
privateApis as componentsPrivateApis,
__unstableMotion as motion,
} from '@wordpress/components';
import { Icon, chevronRight, chevronLeft } from '@wordpress/icons';

/**
* Internal dependencies
Expand Down Expand Up @@ -55,18 +51,12 @@ function CategoryTabs( {
<Tabs.Tab
key={ category.name }
tabId={ category.name }
className="block-editor-inserter__category-tab"
aria-label={ category.label }
aria-current={
category === selectedCategory ? 'true' : undefined
}
>
<HStack>
<FlexBlock>{ category.label }</FlexBlock>
<Icon
icon={ isRTL() ? chevronLeft : chevronRight }
/>
</HStack>
{ category.label }
</Tabs.Tab>
) ) }
</Tabs.TabList>
Expand Down
42 changes: 1 addition & 41 deletions packages/block-editor/src/components/inserter/style.scss
Original file line number Diff line number Diff line change
Expand Up @@ -214,55 +214,15 @@ $block-inserter-tabs-height: 44px;

.block-editor-inserter__media-tabs-container,
.block-editor-inserter__block-patterns-tabs-container {
flex-grow: 1;
padding: $grid-unit-20;
height: 100%;
display: flex;
flex-direction: column;
justify-content: space-between;
}

.block-editor-inserter__category-tablist {
display: flex;
flex-direction: column;
border: none;
margin-bottom: $grid-unit-10;
// Push the listitem wrapping the "explore" button to the bottom of the panel.
div[role="listitem"]:last-child {
margin-top: auto;
}

// Temporarily disable the component's indicator animation.
// TODO: remove in favor of using the native component's styles and behavior,
// see https://github.com/WordPress/gutenberg/pull/62879#issuecomment-2219720582
&[aria-orientation="vertical"]::after {
content: none;
}

.block-editor-inserter__category-tab {
// Account for the icon on the right so that it's visually balanced.
padding: $grid-unit-10 $grid-unit-05 $grid-unit-10 $grid-unit-15;
text-align: left;
font-weight: inherit;
display: block;
position: relative;
height: auto;

&[aria-selected="true"] {
color: var(--wp-admin-theme-color);

.components-flex-item {
filter: brightness(0.95);
}

svg {
fill: var(--wp-admin-theme-color);
}
}

&::before {
display: none;
}
}
}

.block-editor-inserter__category-panel {
Expand Down
4 changes: 4 additions & 0 deletions packages/components/CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,10 @@
- `RangeControl`: do not tooltip contents to the DOM when not shown ([#65875](https://github.com/WordPress/gutenberg/pull/65875)).
- `Tabs`: fix skipping indication animation glitch ([#65878](https://github.com/WordPress/gutenberg/pull/65878)).

### Enhancements

- `Tabs`: revamped vertical orientation styles ([#65387](https://github.com/WordPress/gutenberg/pull/65387)).

## 28.9.0 (2024-10-03)

### Bug Fixes
Expand Down
152 changes: 109 additions & 43 deletions packages/components/src/tabs/styles.ts
Original file line number Diff line number Diff line change
Expand Up @@ -9,18 +9,15 @@ import * as Ariakit from '@ariakit/react';
*/
import { COLORS, CONFIG } from '../utils';
import { space } from '../utils/space';
import Icon from '../icon';

export const TabListWrapper = styled.div`
position: relative;
display: flex;
align-items: stretch;
flex-direction: row;
text-align: center;
overflow-x: auto;
&[aria-orientation='vertical'] {
flex-direction: column;
text-align: start;
}
:where( [aria-orientation='horizontal'] ) {
Expand All @@ -40,11 +37,12 @@ export const TabListWrapper = styled.div`
@media not ( prefers-reduced-motion ) {
&[data-indicator-animated]::before {
transition-property: transform;
transition-property: transform, border-radius, border-block;
transition-duration: 0.2s;
transition-timing-function: ease-out;
}
}
position: relative;
&::before {
content: '';
position: absolute;
Expand All @@ -59,7 +57,7 @@ export const TabListWrapper = styled.div`
/* Using a large value to avoid antialiasing rounding issues
when scaling in the transform, see: https://stackoverflow.com/a/52159123 */
--antialiasing-factor: 100;
&:not( [aria-orientation='vertical'] ) {
&[aria-orientation='horizontal'] {
--fade-width: 4rem;
--fade-gradient-base: transparent 0%, black var( --fade-width );
--fade-gradient-composed: var( --fade-gradient-base ), black 60%,
Expand Down Expand Up @@ -104,65 +102,87 @@ export const TabListWrapper = styled.div`
${ COLORS.theme.accent };
}
}
&[aria-orientation='vertical']::before {
top: 0;
left: 0;
width: 100%;
height: calc( var( --antialiasing-factor ) * 1px );
transform: translateY( calc( var( --selected-top, 0 ) * 1px ) )
scaleY(
&[aria-orientation='vertical'] {
&::before {
/* Adjusting the border radius to match the scaling in the y axis. */
border-radius: ${ CONFIG.radiusSmall } /
calc(
var( --selected-height, 0 ) / var( --antialiasing-factor )
)
${ CONFIG.radiusSmall } /
(
var( --selected-height, 0 ) /
var( --antialiasing-factor )
)
);
top: 0;
left: 0;
width: 100%;
height: calc( var( --antialiasing-factor ) * 1px );
transform: translateY( calc( var( --selected-top, 0 ) * 1px ) )
scaleY(
calc(
var( --selected-height, 0 ) /
var( --antialiasing-factor )
)
);
background-color: color-mix(
in srgb,
${ COLORS.theme.accent },
transparent 96%
);
background-color: ${ COLORS.theme.gray[ 100 ] };
}
&[data-select-on-move='true']:has(
:is( :focus-visible, [data-focus-visible] )
)::before {
box-sizing: border-box;
border: var( --wp-admin-border-width-focus ) solid
${ COLORS.theme.accent };
/* Adjusting the border width to match the scaling in the y axis. */
border-block-width: calc(
var( --wp-admin-border-width-focus, 1px ) /
(
var( --selected-height, 0 ) /
var( --antialiasing-factor )
)
);
}
}
`;

export const Tab = styled( Ariakit.Tab )`
& {
scroll-margin: 24px;
flex-grow: 1;
flex-shrink: 0;
display: inline-flex;
align-items: center;
position: relative;
/* Resets */
border-radius: 0;
height: ${ space( 12 ) };
background: transparent;
border: none;
box-shadow: none;
flex: 1 0 auto;
white-space: nowrap;
display: flex;
align-items: center;
cursor: pointer;
line-height: 1.2; // Some languages characters e.g. Japanese may have a native higher line-height.
padding: ${ space( 4 ) };
margin-left: 0;
font-weight: 500;
text-align: inherit;
line-height: 1.2; // Characters in some languages (e.g. Japanese) may have a native higher line-height.
font-weight: 400;
color: ${ COLORS.theme.foreground };
&[aria-disabled='true'] {
cursor: default;
color: ${ COLORS.ui.textDisabled };
}
&:not( [aria-disabled='true'] ):hover {
&:not( [aria-disabled='true'] ):is( :hover, [data-focus-visible] ) {
color: ${ COLORS.theme.accent };
}
&:focus:not( :disabled ) {
position: relative;
box-shadow: none;
outline: none;
}
// Focus.
// Focus indicator.
position: relative;
&::after {
content: '';
position: absolute;
top: ${ space( 3 ) };
right: ${ space( 3 ) };
bottom: ${ space( 3 ) };
left: ${ space( 3 ) };
pointer-events: none;
// Draw the indicator.
Expand All @@ -175,23 +195,69 @@ export const Tab = styled( Ariakit.Tab )`
opacity: 0;
@media not ( prefers-reduced-motion ) {
transition: opacity 0.1s linear;
transition: opacity 0.15s 0.15s linear;
}
}
&:focus-visible::after {
&[data-focus-visible]::after {
opacity: 1;
}
}
[aria-orientation='horizontal'] & {
padding-inline: ${ space( 4 ) };
height: ${ space( 12 ) };
text-align: center;
scroll-margin: 24px;
&::after {
content: '';
inset: ${ space( 3 ) };
}
}
[aria-orientation='vertical'] & {
min-height: ${ space(
10
) }; // Avoid fixed height to allow for long strings that go in multiple lines.
padding: ${ space( 2 ) } ${ space( 3 ) };
min-height: ${ space( 10 ) };
text-align: start;
&[aria-selected='true'] {
color: ${ COLORS.theme.accent };
fill: currentColor;
}
}
[aria-orientation='vertical'][data-select-on-move='false'] &::after {
content: '';
inset: var( --wp-admin-border-width-focus );
}
`;

export const TabChildren = styled.span`
flex-grow: 1;
`;

export const TabChevron = styled( Icon )`
flex-shrink: 0;
margin-inline-end: ${ space( -1 ) };
[aria-orientation='horizontal'] & {
justify-content: center;
display: none;
}
opacity: 0;
[role='tab']:is( [aria-selected='true'], [data-focus-visible], :hover ) & {
opacity: 1;
}
// The chevron is transitioned into existence when selectOnMove is enabled,
// because otherwise it looks jarring, as it shows up outside of the focus
// indicator that's being animated at the same time.
@media not ( prefers-reduced-motion ) {
[data-select-on-move='true']
[role='tab']:is( [aria-selected='true'], )
& {
transition: opacity 0.3s ease-in;
}
}
&:dir( rtl ) {
rotate: 180deg;
}
`;

Expand All @@ -201,7 +267,7 @@ export const TabPanel = styled( Ariakit.TabPanel )`
outline: none;
}
&:focus-visible {
&[data-focus-visible] {
box-shadow: 0 0 0 var( --wp-admin-border-width-focus )
${ COLORS.theme.accent };
// Windows high contrast mode.
Expand Down
10 changes: 8 additions & 2 deletions packages/components/src/tabs/tab.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -10,8 +10,13 @@ import { forwardRef } from '@wordpress/element';
import type { TabProps } from './types';
import warning from '@wordpress/warning';
import { useTabsContext } from './context';
import { Tab as StyledTab } from './styles';
import {
Tab as StyledTab,
TabChildren as StyledTabChildren,
TabChevron as StyledTabChevron,
} from './styles';
import type { WordPressComponentProps } from '../context';
import { chevronRight } from '@wordpress/icons';

export const Tab = forwardRef<
HTMLButtonElement,
Expand All @@ -33,7 +38,8 @@ export const Tab = forwardRef<
render={ render }
{ ...otherProps }
>
{ children }
<StyledTabChildren>{ children }</StyledTabChildren>
<StyledTabChevron icon={ chevronRight } />
</StyledTab>
);
} );
1 change: 1 addition & 0 deletions packages/components/src/tabs/tablist.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -115,6 +115,7 @@ export const TabList = forwardRef<
render={ <TabListWrapper /> }
onBlur={ onBlur }
tabIndex={ -1 }
data-select-on-move={ selectOnMove ? 'true' : 'false' }
{ ...otherProps }
className={ clsx(
overflow.first && 'is-overflowing-first',
Expand Down
7 changes: 7 additions & 0 deletions test/e2e/specs/editor/various/a11y.spec.js
Original file line number Diff line number Diff line change
Expand Up @@ -123,7 +123,14 @@ test.describe( 'a11y (@firefox, @webkit)', () => {
test( 'should make the modal content focusable when it is scrollable', async ( {
page,
pageUtils,
browserName,
} ) => {
// eslint-disable-next-line playwright/no-skipped-test
test.skip(
browserName === 'webkit',
'Known bug with focus order in Safari.'
);

// Note: this test depends on a particular viewport height to determine whether or not
// the modal content is scrollable. If this tests fails and needs to be debugged locally,
// double-check the viewport height when running locally versus in CI. Additionally,
Expand Down

0 comments on commit 5a59680

Please sign in to comment.