diff --git a/packages/block-editor/src/components/link-control/README.md b/packages/block-editor/src/components/link-control/README.md index a887f092933a14..d458448175ee9e 100644 --- a/packages/block-editor/src/components/link-control/README.md +++ b/packages/block-editor/src/components/link-control/README.md @@ -58,6 +58,19 @@ through of its function parameter. - Type: `Function` - Required: No +Use this callback to take an action after a user set or updated a link. +The function callback will receive the selected item, or Null. + +```es6 + { + item + ? console.log( `The item selected has the ${ item.id } id.` ) + : console.warn( 'No Item selected.' ); + } +/> +``` + ### onSettingChange - Type: `Function` diff --git a/packages/block-library/src/navigation-menu-item/block.json b/packages/block-library/src/navigation-menu-item/block.json index 20b33eca1f4a05..00e59fc5224f54 100644 --- a/packages/block-library/src/navigation-menu-item/block.json +++ b/packages/block-library/src/navigation-menu-item/block.json @@ -12,9 +12,15 @@ "title": { "type": "string" }, + "type": { + "type": "string" + }, "description": { "type": "string" }, + "linkId": { + "type": "number" + }, "opensInNewTab": { "type": "boolean", "default": false diff --git a/packages/block-library/src/navigation-menu-item/edit.js b/packages/block-library/src/navigation-menu-item/edit.js index b1e6ab69f24df5..9a0273c5de3c57 100644 --- a/packages/block-library/src/navigation-menu-item/edit.js +++ b/packages/block-library/src/navigation-menu-item/edit.js @@ -33,14 +33,41 @@ import { BlockControls, InnerBlocks, InspectorControls, - URLPopover, RichText, + __experimentalLinkControl as LinkControl, } from '@wordpress/block-editor'; -import { - Fragment, - useRef, - useState, -} from '@wordpress/element'; +import { Fragment, useState, useEffect } from '@wordpress/element'; + +/** + * It updates the link attribute when the + * link settings changes. + * + * @param {Function} setter Setter attribute function. + */ +const updateLinkSetting = ( setter ) => ( setting, value ) => { + if ( setting === 'new-tab' ) { + setter( { opensInNewTab: value } ); + } +}; + +/** + * Updates the link attribute when it changes + * through of the `onLinkChange` LinkControl callback. + * + * @param {Function} setter Setter attribute function. + * @param {string} label ItemMenu link label. + */ +const updateLink = ( setter, label ) => ( { title: newTitle = '', url: newURL = '' } = {} ) => { + setter( { + title: newTitle, + url: newURL, + } ); + + // Set the item label as well if it isn't already defined. + if ( ! label ) { + setter( { label: newTitle } ); + } +}; function NavigationMenuItemEdit( { attributes, @@ -50,40 +77,48 @@ function NavigationMenuItemEdit( { setAttributes, insertMenuItemBlock, } ) { - const [ isLinkOpen, setIsLinkOpen ] = useState( false ); - const [ isEditingLink, setIsEditingLink ] = useState( false ); - const [ urlInput, setUrlInput ] = useState( null ); + const { label, opensInNewTab, title, url } = attributes; + const link = title ? { title, url } : null; + const [ isLinkOpen, setIsLinkOpen ] = useState( ! label && isSelected ); - const inputValue = urlInput !== null ? urlInput : url; + let onCloseTimerId = null; - const onKeyDown = ( event ) => { - if ( [ LEFT, DOWN, RIGHT, UP, BACKSPACE, ENTER ].indexOf( event.keyCode ) > -1 ) { - // Stop the key event from propagating up to ObserveTyping.startTypingInTextField. - event.stopPropagation(); + /** + * It's a kind of hack to handle closing the LinkControl popover + * clicking on the ToolbarButton link. + */ + useEffect( () => { + if ( ! isSelected ) { + setIsLinkOpen( false ); } - }; - const closeURLPopover = () => { - setIsEditingLink( false ); - setUrlInput( null ); - setIsLinkOpen( false ); - }; + return () => { + // Clear LinkControl.OnClose timeout. + if ( onCloseTimerId ) { + clearTimeout( onCloseTimerId ); + } + }; + }, [ isSelected ] ); - const autocompleteRef = useRef( null ); + /** + * `onKeyDown` LinkControl handler. + * It takes over to stop the event propagation to make the + * navigation work, avoiding undesired behaviors. + * For instance, it will block to move between menu items + * when the LinkControl is focused. + * + * @param {Event} event + */ + const handleLinkControlOnKeyDown = ( event ) => { + const { keyCode } = event; - const onFocusOutside = ( event ) => { - const autocompleteElement = autocompleteRef.current; - if ( autocompleteElement && autocompleteElement.contains( event.target ) ) { - return; + if ( [ LEFT, DOWN, RIGHT, UP, BACKSPACE, ENTER ].indexOf( keyCode ) > -1 ) { + // Stop the key event from propagating up to ObserveTyping.startTypingInTextField. + event.stopPropagation(); } - closeURLPopover(); - }; - - const stopPropagation = ( event ) => { - event.stopPropagation(); }; - const { label, url } = attributes; + const itemLabelPlaceholder = __( 'Add item…' ); return ( @@ -93,42 +128,20 @@ function NavigationMenuItemEdit( { name="link" icon="admin-links" title={ __( 'Link' ) } - onClick={ () => setIsLinkOpen( ! isLinkOpen ) } + onClick={ () => { + if ( isLinkOpen ) { + return; + } + setIsLinkOpen( ! isLinkOpen ); + } } /> - { } title={ __( 'Add submenu item' ) } onClick={ insertMenuItemBlock } - /> } + /> - { isLinkOpen && - <> - - { ( ! url || isEditingLink ) && - event.preventDefault() } - autocompleteRef={ autocompleteRef } - /> - } - { ( url && ! isEditingLink ) && - - } - - - - } { - setAttributes( { opensInNewTab } ); + onChange={ ( newTab ) => { + setAttributes( { opensInNewTab: newTab } ); } } label={ __( 'Open in new tab' ) } /> @@ -154,8 +167,8 @@ function NavigationMenuItemEdit( { > { - setAttributes( { title } ); + onChange={ ( itemTitle ) => { + setAttributes( { title: itemTitle } ); } } label={ __( 'Title Attribute' ) } help={ __( 'Provide more context about where the link goes.' ) } @@ -186,13 +199,29 @@ function NavigationMenuItemEdit( { 'is-selected': isSelected, } ) } > - setAttributes( { label: labelValue } ) } - placeholder={ __( 'Add item…' ) } - withoutInteractiveFormatting - /> +
+ setAttributes( { label: labelValue } ) } + placeholder={ itemLabelPlaceholder } + withoutInteractiveFormatting + /> + { isLinkOpen && ( + event.stopPropagation() } + currentLink={ link } + onLinkChange={ updateLink( setAttributes, label ) } + onClose={ () => { + onCloseTimerId = setTimeout( () => setIsLinkOpen( false ), 100 ); + } } + currentSettings={ { 'new-tab': opensInNewTab } } + onSettingsChange={ updateLinkSetting( setAttributes ) } + /> + ) } +
{ - return [ 'core/navigation-menu-item', - { label: page.title.rendered, url: page.permalink_template }, - ]; - } ); + + return pages.map( ( { title, type, link: url, id: linkId } ) => ( + [ 'core/navigation-menu-item', { + label: title.rendered, + title: title.raw, + type, + linkId, + url, + opensInNewTab: false, + } ] + ) ); }, [ pages ] ); diff --git a/packages/block-library/src/navigation-menu/index.php b/packages/block-library/src/navigation-menu/index.php index b49a41a06aa11c..02ba29277d7d25 100644 --- a/packages/block-library/src/navigation-menu/index.php +++ b/packages/block-library/src/navigation-menu/index.php @@ -102,17 +102,26 @@ function build_navigation_menu_html( $block, $colors ) { class="wp-block-navigation-menu-item__link ' . $colors['text_css_classes'] . '" ' . $colors['text_inline_styles']; + // Start appending HTML attributes to anchor tag. if ( isset( $menu_item['attrs']['url'] ) ) { $html .= ' href="' . $menu_item['attrs']['url'] . '"'; } if ( isset( $menu_item['attrs']['title'] ) ) { $html .= ' title="' . $menu_item['attrs']['title'] . '"'; } + + if ( isset( $menu_item['attrs']['opensInNewTab'] ) && true === $menu_item['attrs']['opensInNewTab'] ) { + $html .= ' target="_blank" '; + } + // End appending HTML attributes to anchor tag. + + // Start anchor tag content. $html .= '>'; if ( isset( $menu_item['attrs']['label'] ) ) { $html .= $menu_item['attrs']['label']; } $html .= ''; + // End anchor tag content. if ( count( (array) $menu_item['innerBlocks'] ) > 0 ) { $html .= build_navigation_menu_html( $menu_item, $colors );