/** * External dependencies */ import { TourKit, TourKitTypes } from '@woocommerce/components'; import { __ } from '@wordpress/i18n'; import { useDispatch, useSelect } from '@wordpress/data'; import { useLayoutEffect, useEffect, useState, useRef, createPortal, } from '@wordpress/element'; import { OPTIONS_STORE_NAME } from '@woocommerce/data'; import { recordEvent } from '@woocommerce/tracks'; const REVIEWED_DEFAULTS_OPTION = 'woocommerce_admin_reviewed_default_shipping_zones'; const CREATED_DEFAULTS_OPTION = 'woocommerce_admin_created_default_shipping_zones'; const FLOATER_WRAPPER_CLASS = 'woocommerce-settings-shipping-tour-floater-wrapper'; const FLOATER_CLASS = 'woocommerce-settings-smart-defaults-shipping-tour-floater'; const SHIPPING_ZONES_SETTINGS_TABLE_CLASS = 'table.wc-shipping-zones'; const WCS_LINK_SELECTOR = 'a[href*="woocommerce-services-settings"]'; const SHIPPING_RECOMMENDATIONS_SELECTOR = 'div.woocommerce-recommended-shipping-extensions'; const useShowShippingTour = () => { const { hasCreatedDefaultShippingZones, hasReviewedDefaultShippingOptions, isLoading, } = useSelect( ( select ) => { const { hasFinishedResolution, getOption } = select( OPTIONS_STORE_NAME ); return { isLoading: ! hasFinishedResolution( 'getOption', [ CREATED_DEFAULTS_OPTION, ] ) && ! hasFinishedResolution( 'getOption', [ REVIEWED_DEFAULTS_OPTION, ] ), hasCreatedDefaultShippingZones: getOption( CREATED_DEFAULTS_OPTION ) === 'yes', hasReviewedDefaultShippingOptions: getOption( REVIEWED_DEFAULTS_OPTION ) === 'yes', }; } ); return { isLoading, show: window.wcAdminFeatures[ 'shipping-setting-tour' ] && ! isLoading && hasCreatedDefaultShippingZones && ! hasReviewedDefaultShippingOptions, }; }; type NonEmptySelectorArray = readonly [ string, ...string[] ]; const computeDims = ( elementsSelectors: NonEmptySelectorArray ) => { const rects = elementsSelectors.map( ( elementSelector ) => { const rect = document ?.querySelector( elementSelector ) ?.getBoundingClientRect(); if ( ! rect ) { throw new Error( "Shipping tour: Couldn't find element with selector: " + elementSelector ); } return rect; } ); const originCoords = document .querySelector( `.${ FLOATER_WRAPPER_CLASS }` ) ?.getBoundingClientRect() || { top: 0, left: 0 }; const top = Math.min( ...rects.map( ( rect ) => rect.top ) ); const left = Math.min( ...rects.map( ( rect ) => rect.left ) ); const right = Math.max( ...rects.map( ( rect ) => rect.right ) ); const bottom = Math.max( ...rects.map( ( rect ) => rect.bottom ) ); const width = right - left; const height = bottom - top; // offset top and left from origin const topOffset = top - originCoords.top; const leftOffset = left - originCoords.left; return { left: leftOffset, top: topOffset, width, height }; }; const TourFloater = ( { dims }: { dims: Partial< DOMRect > } ) => { return (
); }; // this is defines the elements to be spotlit for each step const spotlitElementsSelectors: Array< NonEmptySelectorArray > = [ [ // just use bottom right element and top left element instead of all rects // top left = table header cell for sort handles 'th.wc-shipping-zone-sort', // bottom right = worldwide region cell 'tr.wc-shipping-zone-worldwide > td.wc-shipping-zone-region', ], [ // selectors for rightmost column (shipping methods) 'th.wc-shipping-zone-methods', 'tr.wc-shipping-zone-worldwide > td.wc-shipping-zone-methods', ], ]; const TourFloaterWrapper = ( { step }: { step: number } ) => { const thisRef = useRef< HTMLDivElement >( null ); useLayoutEffect( () => { // this moves the element to the correct place which is right before the table element if ( thisRef.current?.parentElement ) { thisRef.current.parentElement.insertBefore( thisRef.current, document.querySelector( 'table.wc-shipping-zones' ) ); } }, [] ); const currentStepSelectors = spotlitElementsSelectors[ step ] ?? spotlitElementsSelectors[ spotlitElementsSelectors.length - 1 ]; const [ dims, setDims ] = useState( computeDims( currentStepSelectors ) ); useEffect( () => { setDims( computeDims( currentStepSelectors ) ); const observer = new ResizeObserver( () => { setDims( computeDims( currentStepSelectors ) ); } ); const shippingSettingsTableElement = document.querySelector( SHIPPING_ZONES_SETTINGS_TABLE_CLASS ); if ( ! shippingSettingsTableElement ) { throw new Error( "Shipping tour: Couldn't find shipping settings table element with selector: " + SHIPPING_ZONES_SETTINGS_TABLE_CLASS ); } observer.observe( shippingSettingsTableElement ); return () => { observer.disconnect(); }; }, [ currentStepSelectors ] ); const shippingSettingsTableParentElement = document.querySelector( SHIPPING_ZONES_SETTINGS_TABLE_CLASS )?.parentElement; if ( ! shippingSettingsTableParentElement ) { throw new Error( "Shipping tour: Couldn't find shipping settings table parent element with selector: " + SHIPPING_ZONES_SETTINGS_TABLE_CLASS ); } /** * use ReactDOM.createPortal to inject our element into non-React controlled DOM * unfortunately createPortal uses .appendChild which puts it in the wrong place, * we want it to be before the table */ return createPortal(