diff --git a/packages/components/src/color-edit/index.js b/packages/components/src/color-edit/index.js
index b8c38b35bba65..12589220d9ff6 100644
--- a/packages/components/src/color-edit/index.js
+++ b/packages/components/src/color-edit/index.js
@@ -1,326 +1,300 @@
/**
* External dependencies
*/
-import classnames from 'classnames';
-import { isEmpty, kebabCase } from 'lodash';
+import kebabCase from 'lodash';
/**
* WordPress dependencies
*/
-import { __, sprintf } from '@wordpress/i18n';
-import { useEffect, useState } from '@wordpress/element';
-import { edit, close, chevronDown, chevronUp, plus } from '@wordpress/icons';
+import { useState, useRef, useEffect } from '@wordpress/element';
+import { __ } from '@wordpress/i18n';
+import { lineSolid, moreVertical, plus } from '@wordpress/icons';
+import { __experimentalUseFocusOutside as useFocusOutside } from '@wordpress/compose';
/**
* Internal dependencies
*/
-import Dropdown from '../dropdown';
-import CircularOptionPicker from '../circular-option-picker';
-import { ColorPicker } from '../color-picker';
import Button from '../button';
-import TextControl from '../text-control';
-import BaseControl from '../base-control';
+import { ColorPicker } from '../color-picker';
+import { FlexItem } from '../flex';
+import { HStack } from '../h-stack';
+import { ItemGroup } from '../item-group';
+import { MenuGroup } from '../menu-group';
+import { MenuItem } from '../menu-item';
+import { VStack } from '../v-stack';
+import ColorPalette from '../color-palette';
+import DropdownMenu from '../dropdown-menu';
+import Popover from '../popover';
+import {
+ ColorActionsContainer,
+ ColorEditStyles,
+ ColorHeading,
+ ColorHStackHeader,
+ ColorIndicatorStyled,
+ ColorItem,
+ ColorNameContainer,
+ ColorNameInputControl,
+ DoneButton,
+} from './styles';
-function DropdownOpenOnMount( { shouldOpen, isOpen, onToggle } ) {
- useEffect( () => {
- if ( shouldOpen && ! isOpen ) {
- onToggle();
- }
- }, [] );
- return null;
+function ColorNameInput( { value, onChange } ) {
+ return (
+
+ );
}
function ColorOption( {
color,
- name,
- slug,
onChange,
+ isEditing,
+ onStartEditing,
onRemove,
- onConfirm,
- confirmLabel = __( 'OK' ),
- isEditingNameOnMount = false,
- isEditingColorOnMount = false,
- onCancel,
- immutableColorSlugs = [],
+ onStopEditing,
} ) {
- const [ isHover, setIsHover ] = useState( false );
- const [ isFocused, setIsFocused ] = useState( false );
- const [ isEditingName, setIsEditingName ] = useState(
- isEditingNameOnMount
- );
- const [ isShowingAdvancedPanel, setIsShowingAdvancedPanel ] = useState(
- false
- );
-
- const isShowingControls =
- ( isHover || isFocused || isEditingName || isShowingAdvancedPanel ) &&
- ! immutableColorSlugs.includes( slug );
-
+ const focusOutsideProps = useFocusOutside( onStopEditing );
return (
-
setIsHover( true ) }
- onMouseLeave={ () => setIsHover( false ) }
- onFocus={ () => setIsFocused( true ) }
- onBlur={ () => setIsFocused( false ) }
- aria-label={
- name
- ? // translators: %s: The name of the color e.g: "vivid red".
- sprintf( __( 'Color: %s' ), name )
- : // translators: %s: color hex code e.g: "#f00".
- sprintf( __( 'Color code: %s' ), color )
- }
+
-
-
(
- <>
-
-
- >
- ) }
- renderContent={ () => (
-
+
+
+
+
+
+ { isEditing ? (
+
onChange( {
- color: newColor,
- slug,
- name,
+ ...color,
+ name: nextName,
+ slug: kebabCase( nextName ),
} )
}
/>
+ ) : (
+ { color.name }
) }
- />
- { ! isEditingName && (
-
- { name }
-
- ) }
- { isEditingName && (
- <>
-
- onChange( {
- color,
- slug: kebabCase( newColorName ),
- name: newColorName,
- } )
- }
- label={ __( 'Color name' ) }
- placeholder={ __( 'Name' ) }
- value={ name }
- />
-
- >
- ) }
- { ! isEditingName && (
- <>
+
+ { isEditing && (
+
) }
- {
- if ( isShowingAdvancedPanel ) {
- setIsFocused( false );
- }
- setIsShowingAdvancedPanel( ! isShowingAdvancedPanel );
- } }
- aria-expanded={ isShowingAdvancedPanel }
- />
-
- { onCancel && (
-
+ { isEditing && (
+
- { __( 'Cancel' ) }
-
- ) }
- { isShowingAdvancedPanel && (
-
- onChange( {
- color,
- slug: newSlug,
- name,
- } )
- }
- label={ __( 'Slug' ) }
- value={ slug }
- />
+
+ onChange( {
+ ...color,
+ color: newColor,
+ } )
+ }
+ />
+
) }
-
+
);
}
-function ColorInserter( { onInsert, onCancel } ) {
- const [ color, setColor ] = useState( {
- color: '#fff',
- name: '',
- slug: '',
- } );
+function ColorPaletteEditListView( {
+ colors,
+ onChange,
+ editingColor,
+ setEditingColor,
+} ) {
+ // When unmounting the component if there are empty colors (the user did not complete the insertion) clean them.
+ const colorReference = useRef();
+ useEffect( () => {
+ colorReference.current = colors;
+ }, [ colors ] );
+ useEffect( () => {
+ return () => {
+ if ( colorReference.current.some( ( { slug } ) => ! slug ) ) {
+ const newColors = colorReference.current.filter(
+ ( { slug } ) => slug
+ );
+ onChange( newColors.length ? newColors : undefined );
+ }
+ };
+ }, [] );
return (
- onInsert( color ) }
- isEditingNameOnMount
- isEditingColorOnMount
- onCancel={ onCancel }
- />
+
+
+ { colors.map( ( color, index ) => (
+ {
+ if ( editingColor !== index ) {
+ setEditingColor( index );
+ }
+ } }
+ onChange={ ( newColor ) => {
+ onChange(
+ colors.map( ( currentColor, currentIndex ) => {
+ if ( currentIndex === index ) {
+ return newColor;
+ }
+ return currentColor;
+ } )
+ );
+ } }
+ onRemove={ () => {
+ setEditingColor( null );
+ const newColors = colors.filter(
+ ( _currentColor, currentIndex ) => {
+ if ( currentIndex === index ) {
+ return false;
+ }
+ return true;
+ }
+ );
+ onChange(
+ newColors.length ? newColors : undefined
+ );
+ } }
+ isEditing={ index === editingColor }
+ onStopEditing={ () => {
+ if ( index === editingColor ) {
+ setEditingColor( null );
+ }
+ } }
+ />
+ ) ) }
+
+
);
}
-export default function ColorEdit( {
- colors,
- onChange,
- emptyUI,
- immutableColorSlugs,
- canReset = true,
-} ) {
- const [ isInsertingColor, setIsInsertingColor ] = useState( false );
+const EMPTY_ARRAY = [];
+
+export default function ColorEdit( { colors = EMPTY_ARRAY, onChange } ) {
+ const [ isEditing, setIsEditing ] = useState( false );
+ const [ editingColor, setEditingColor ] = useState( null );
+ const isAdding =
+ isEditing &&
+ editingColor &&
+ colors[ editingColor ] &&
+ ! colors[ editingColor ].slug;
+
+ const hasColors = colors.length > 0;
+
return (
-
-
-
+
);
}
diff --git a/packages/components/src/color-edit/style.scss b/packages/components/src/color-edit/style.scss
index ef050b1a6fa4a..6370e796c488e 100644
--- a/packages/components/src/color-edit/style.scss
+++ b/packages/components/src/color-edit/style.scss
@@ -1,47 +1,6 @@
-.components-color-edit__color-option-main-area {
- display: flex;
- align-items: center;
- div.components-circular-option-picker__option-wrapper {
- display: block;
- margin: $grid-unit-10;
+@include break-medium() {
+ .components-color-edit__color-popover.components-popover .components-popover__content.components-popover__content.components-popover__content {
+ margin-right: #{ ($sidebar-width / 2) + $grid-unit-20 };
+ margin-top: #{ -($grid-unit-60 + $border-width) };
}
}
-
-.components-color-edit__color-option.is-hover {
- background: $gray-200;
-}
-
-.components-color-edit__cancel-button {
- float: right;
-}
-
-.components-color-edit__color-option-color-name {
- width: 100%;
-}
-.components-color-edit__label-and-insert-container {
- display: flex;
- align-items: center;
- justify-content: space-between;
-}
-
-.components-color-edit__insert-button {
- margin-top: -$grid-unit-10;
-}
-
-.components-color-edit__hidden-control {
- position: relative;
- left: -9999px;
-}
-
-.components-color-edit__color-option-color-name-input .components-base-control__field {
- margin-bottom: 0;
- margin-right: $grid-unit-10;
-}
-
-.components-color-edit__slug-input {
- margin-left: $grid-unit-10;
-}
-
-.components-color-edit__reset-button {
- float: right;
-}
diff --git a/packages/components/src/color-edit/styles.js b/packages/components/src/color-edit/styles.js
new file mode 100644
index 0000000000000..1a270634853ee
--- /dev/null
+++ b/packages/components/src/color-edit/styles.js
@@ -0,0 +1,89 @@
+/**
+ * External dependencies
+ */
+import styled from '@emotion/styled';
+
+/**
+ * Internal dependencies
+ */
+import Button from '../button';
+import { Heading } from '../heading';
+import { HStack } from '../h-stack';
+import { space } from '../ui/utils/space';
+import { View } from '../view';
+import ColorIndicator from '../color-indicator';
+import InputControl from '../input-control';
+import Item from '../item-group/item';
+import {
+ Container as InputControlContainer,
+ Input,
+ BackdropUI as InputBackdropUI,
+} from '../input-control/styles/input-control-styles';
+
+export const ColorIndicatorStyled = styled( ColorIndicator )`
+ && {
+ display: block;
+ border-radius: 50%;
+ border: 0;
+ height: ${ space( 6 ) };
+ width: ${ space( 6 ) };
+ margin-left: 0;
+ padding: 0;
+ }
+`;
+
+export const ColorNameInputControl = styled( InputControl )`
+ ${ InputControlContainer } {
+ background: #f0f0f0;
+ border-radius: 2px;
+ ${ Input }${ Input }${ Input }${ Input } {
+ width: 160px;
+ height: 32px;
+ }
+ ${ InputBackdropUI }${ InputBackdropUI }${ InputBackdropUI } {
+ border-color: transparent;
+ box-shadow: none;
+ }
+ }
+`;
+
+export const ColorItem = styled( Item )`
+ padding: 3px 12px;
+`;
+
+export const ColorNameContainer = styled.span`
+ line-height: 32px;
+`;
+
+export const ColorHeading = styled( Heading )`
+ text-transform: uppercase;
+ line-height: 24px;
+ font-weight: 500;
+ &&& {
+ font-size: 11px;
+ margin-bottom: 0;
+ }
+`;
+
+export const ColorActionsContainer = styled( View )`
+ height: 24px;
+ display: flex;
+`;
+
+export const ColorHStackHeader = styled( HStack )`
+ margin-bottom: ${ space( 2 ) };
+`;
+
+export const ColorEditStyles = styled( View )`
+ &&& {
+ .components-button.has-icon {
+ min-width: 0;
+ padding: 0 2px;
+ }
+`;
+
+export const DoneButton = styled( Button )`
+ && {
+ color: #3858e9;
+ }
+`;
diff --git a/packages/edit-site/src/components/global-styles/color-palette-panel.js b/packages/edit-site/src/components/global-styles/color-palette-panel.js
index 76a9a951b3595..b069738de3fa0 100644
--- a/packages/edit-site/src/components/global-styles/color-palette-panel.js
+++ b/packages/edit-site/src/components/global-styles/color-palette-panel.js
@@ -2,57 +2,21 @@
* WordPress dependencies
*/
import { __experimentalColorEdit as ColorEdit } from '@wordpress/components';
-import { __ } from '@wordpress/i18n';
-import { useMemo } from '@wordpress/element';
/**
* Internal dependencies
*/
import { useSetting } from './hooks';
-/**
- * Shared reference to an empty array for cases where it is important to avoid
- * returning a new array reference on every invocation, as in a connected or
- * other pure component which performs `shouldComponentUpdate` check on props.
- * This should be used as a last resort, since the normalized data should be
- * maintained by the reducer result in state.
- *
- * @type {Array}
- */
-const EMPTY_ARRAY = [];
-
export default function ColorPalettePanel( { name } ) {
- const [ colors, setColors ] = useSetting( 'color.palette', name );
- const [ userColors ] = useSetting( 'color.palette', name, 'user' );
- const [ baseGlobalPalette ] = useSetting(
- 'color.palette',
- undefined,
- 'base'
- );
- const [ baseContextualPalette ] = useSetting(
+ const [ userColors, setColors ] = useSetting(
'color.palette',
name,
- 'base'
+ 'user'
);
- const immutableColorSlugs = useMemo( () => {
- const basePalette = baseContextualPalette ?? baseGlobalPalette;
- if ( ! basePalette ) {
- return EMPTY_ARRAY;
- }
- return basePalette.map( ( { slug } ) => slug );
- }, [ baseContextualPalette, baseGlobalPalette ] );
-
return (
-
+
);
}