/** * External dependencies */ import type { AddToCartEventDetail } from '@woocommerce/type-defs/events'; const CustomEvent = window.CustomEvent || null; interface DispatchedEventProperties { // Whether the event bubbles. bubbles?: boolean; // Whether the event is cancelable. cancelable?: boolean; // See https://developer.mozilla.org/en-US/docs/Web/API/CustomEvent/detail detail?: unknown; // Element that dispatches the event. By default, the body. element?: HTMLElement; } /** * Wrapper function to dispatch an event. */ export const dispatchEvent = ( name: string, { bubbles = false, cancelable = false, element, detail = {}, }: DispatchedEventProperties ): void => { if ( ! CustomEvent ) { return; } if ( ! element ) { element = document.body; } const event = new CustomEvent( name, { bubbles, cancelable, detail, } ); element.dispatchEvent( event ); }; let fragmentRequestTimeoutId: ReturnType< typeof setTimeout >; // This is a hack to trigger cart updates till we migrate to block based cart // that relies on the store, see // https://github.com/woocommerce/woocommerce-gutenberg-products-block/issues/1247 export const triggerFragmentRefresh = (): void => { if ( fragmentRequestTimeoutId ) { clearTimeout( fragmentRequestTimeoutId ); } fragmentRequestTimeoutId = setTimeout( () => { dispatchEvent( 'wc_fragment_refresh', { bubbles: true, cancelable: true, } ); }, 50 ); }; export const triggerAddingToCartEvent = (): void => { dispatchEvent( 'wc-blocks_adding_to_cart', { bubbles: true, cancelable: true, } ); }; export const triggerAddedToCartEvent = ( { preserveCartData = false, }: AddToCartEventDetail ): void => { dispatchEvent( 'wc-blocks_added_to_cart', { bubbles: true, cancelable: true, detail: { preserveCartData }, } ); }; /** * Function that listens to a jQuery event and dispatches a native JS event. * Useful to convert WC Core events into events that can be read by blocks. * * Returns a function to remove the jQuery event handler. Ideally it should be * used when the component is unmounted. */ export const translateJQueryEventToNative = ( // Name of the jQuery event to listen to. jQueryEventName: string, // Name of the native event to dispatch. nativeEventName: string, // Whether the event bubbles. bubbles = false, // Whether the event is cancelable. cancelable = false ): ( () => void ) => { if ( typeof jQuery !== 'function' ) { return () => void null; } const eventDispatcher = () => { dispatchEvent( nativeEventName, { bubbles, cancelable } ); }; jQuery( document ).on( jQueryEventName, eventDispatcher ); return () => jQuery( document ).off( jQueryEventName, eventDispatcher ); };