import { useContext, useMemo, useEffect, useLayoutEffect } from 'preact/hooks'; import { deepSignal, peek } from 'deepsignal'; import { useSignalEffect } from './utils'; import { directive } from './hooks'; import { prefetch, navigate } from './router'; const isObject = ( item ) => item && typeof item === 'object' && ! Array.isArray( item ); const mergeDeepSignals = ( target, source ) => { for ( const k in source ) { if ( typeof peek( target, k ) === 'undefined' ) { target[ `$${ k }` ] = source[ `$${ k }` ]; } else if ( isObject( peek( target, k ) ) && isObject( peek( source, k ) ) ) { mergeDeepSignals( target[ `$${ k }` ].peek(), source[ `$${ k }` ].peek() ); } } }; export default () => { // data-wc-context directive( 'context', ( { directives: { context: { default: context }, }, props: { children }, context: inherited, } ) => { const { Provider } = inherited; const inheritedValue = useContext( inherited ); const value = useMemo( () => { const localValue = deepSignal( context ); mergeDeepSignals( localValue, inheritedValue ); return localValue; }, [ context, inheritedValue ] ); return { children }; }, { priority: 5 } ); // data-wc-effect--[name] directive( 'effect', ( { directives: { effect }, context, evaluate } ) => { const contextValue = useContext( context ); Object.values( effect ).forEach( ( path ) => { useSignalEffect( () => { return evaluate( path, { context: contextValue } ); } ); } ); } ); // data-wc-layout-init--[name] directive( 'layout-init', ( { directives: { 'layout-init': layoutInit }, context, evaluate, } ) => { const contextValue = useContext( context ); Object.values( layoutInit ).forEach( ( path ) => { useLayoutEffect( () => { return evaluate( path, { context: contextValue } ); }, [] ); } ); } ); // data-wc-on--[event] directive( 'on', ( { directives: { on }, element, evaluate, context } ) => { const contextValue = useContext( context ); Object.entries( on ).forEach( ( [ name, path ] ) => { element.props[ `on${ name }` ] = ( event ) => { evaluate( path, { event, context: contextValue } ); }; } ); } ); // data-wc-class--[classname] directive( 'class', ( { directives: { class: className }, element, evaluate, context, } ) => { const contextValue = useContext( context ); Object.keys( className ) .filter( ( n ) => n !== 'default' ) .forEach( ( name ) => { const result = evaluate( className[ name ], { className: name, context: contextValue, } ); const currentClass = element.props.class || ''; const classFinder = new RegExp( `(^|\\s)${ name }(\\s|$)`, 'g' ); if ( ! result ) element.props.class = currentClass .replace( classFinder, ' ' ) .trim(); else if ( ! classFinder.test( currentClass ) ) element.props.class = currentClass ? `${ currentClass } ${ name }` : name; useEffect( () => { // This seems necessary because Preact doesn't change the class names // on the hydration, so we have to do it manually. It doesn't need // deps because it only needs to do it the first time. if ( ! result ) { element.ref.current.classList.remove( name ); } else { element.ref.current.classList.add( name ); } }, [] ); } ); } ); // data-wc-bind--[attribute] directive( 'bind', ( { directives: { bind }, element, context, evaluate } ) => { const contextValue = useContext( context ); Object.entries( bind ) .filter( ( n ) => n !== 'default' ) .forEach( ( [ attribute, path ] ) => { const result = evaluate( path, { context: contextValue, } ); element.props[ attribute ] = result; // This seems necessary because Preact doesn't change the attributes // on the hydration, so we have to do it manually. It doesn't need // deps because it only needs to do it the first time. useEffect( () => { // aria- and data- attributes have no boolean representation. // A `false` value is different from the attribute not being // present, so we can't remove it. // We follow Preact's logic: https://github.com/preactjs/preact/blob/ea49f7a0f9d1ff2c98c0bdd66aa0cbc583055246/src/diff/props.js#L131C24-L136 if ( result === false && attribute[ 4 ] !== '-' ) { element.ref.current.removeAttribute( attribute ); } else { element.ref.current.setAttribute( attribute, result === true && attribute[ 4 ] !== '-' ? '' : result ); } }, [] ); } ); } ); // data-wc-navigation-link directive( 'navigation-link', ( { directives: { 'navigation-link': { default: link }, }, props: { href }, element, } ) => { useEffect( () => { // Prefetch the page if it is in the directive options. if ( link?.prefetch ) { prefetch( href ); } } ); // Don't do anything if it's falsy. if ( link !== false ) { element.props.onclick = async ( event ) => { event.preventDefault(); // Fetch the page (or return it from cache). await navigate( href ); // Update the scroll, depending on the option. True by default. if ( link?.scroll === 'smooth' ) { window.scrollTo( { top: 0, left: 0, behavior: 'smooth', } ); } else if ( link?.scroll !== false ) { window.scrollTo( 0, 0 ); } }; } } ); // data-wc-show directive( 'show', ( { directives: { show: { default: show }, }, element, evaluate, context, } ) => { const contextValue = useContext( context ); if ( ! evaluate( show, { context: contextValue } ) ) element.props.children = ( ); } ); // data-wc-ignore directive( 'ignore', ( { element: { type: Type, props: { innerHTML, ...rest }, }, } ) => { // Preserve the initial inner HTML. const cached = useMemo( () => innerHTML, [] ); return ( ); } ); // data-wc-text directive( 'text', ( { directives: { text: { default: text }, }, element, evaluate, context, } ) => { const contextValue = useContext( context ); element.props.children = evaluate( text, { context: contextValue, } ); } ); };