UNPKG

zumly

Version:

Javascript library for building zooming user interfaces

7 lines 58.2 kB
/* * zumly * v.0.92.3 * Author Juan Martin Muda - Zumerlab * License MIT */ var ot=class{#t={};constructor(t={}){this.#t=t}#e(t){return typeof t=="function"?"function":t instanceof HTMLElement?"element":typeof t=="object"&&t!==null&&typeof t.render=="function"?"object":typeof t!="string"?"unknown":/^https?:\/\/|^\/|\.html$|\.php$/i.test(t)?"url":t.includes("<")?"html":t.includes("-")?"webcomponent":"unknown"}async resolve(t,e=null){if(typeof t=="string"&&Object.prototype.hasOwnProperty.call(this.#t,t)){let o=this.#t[t];return this.resolve(o,e)}switch(this.#e(t)){case"html":{let o=document.createElement("div");o.innerHTML=t.trim();let n=o.firstElementChild;if(!n)throw new Error("Zumly: view produced no element (html)");return n}case"url":{let o=new AbortController,n=setTimeout(()=>o.abort(),1e4),i;try{i=await fetch(t,{signal:o.signal})}finally{clearTimeout(n)}if(!i.ok)throw new Error(`Zumly: fetch failed for "${t}" (${i.status})`);let a=await i.text(),c=document.createElement("div");c.innerHTML=a.trim();let l=c.firstElementChild;if(!l)throw new Error("Zumly: view produced no element (url)");return l}case"function":{let o=e&&e.target||document.createElement("div"),n=await t({...e,target:o});if(typeof n=="string")return this.resolve(n,e);if(n instanceof HTMLElement)return n;let i=o.firstElementChild||o;return i.classList.contains("z-view")||i.classList.add("z-view"),i}case"object":{let o=await t.render(e);if(typeof o!="string")throw new Error("Zumly: view render() must return string");return this.resolve(o,e)}case"element":return t.cloneNode(!0);case"webcomponent":return await customElements.whenDefined(t),document.createElement(t);default:throw new Error("Zumly: unknown view type for source")}}};async function K(s,t,e,r,o,n){if(!s||!s.classList){let i=document.createElement("div");i.classList.add("z-view"),s&&i.appendChild(s),s=i}else s.classList.contains("z-view")||s.classList.add("z-view");return s.dataset.viewName=t,s.getAttribute("aria-label")||s.setAttribute("aria-label",`View: ${t}`),s.style.transformOrigin="0 0",s.style.position="absolute",r?s.classList.add("is-current-view"):s.classList.add("is-new-current-view","has-no-events","hide"),e.append(s),typeof o[t]=="object"&&typeof o[t].mounted=="function"&&await o[t].mounted(),s}function z(s,t,e){t&&e==="welcome"&&console.info(`%c Zumly %c ${t}`,"background: #424085; color: white; border-radius: 3px;","color: #424085"),t&&s&&(e==="info"||e===void 0)&&console.info(`%c Zumly %c ${t}`,"background: #6679A3; color: #304157; border-radius: 3px;","color: #6679A3"),t&&e==="warn"&&console.warn(`%c Zumly %c ${t}`,"background: #DCBF53; color: #424085; border-radius: 3px;","color: #424085"),t&&e==="error"&&console.error(`%c Zumly %c ${t}`,"background: #BE4747; color: white; border-radius: 3px;","color: #424085")}function It(s,t){if(t.isValid=!1,!s||typeof s!="object"){z(!1,"'options' object has to be provided when instance is defined","error");return}if(typeof s.mount!="string"){z(!1,"'mount' must be a string selector","error");return}if(t.mount=s.mount,typeof s.initialView!="string"){z(!1,"'initialView' must be a string","error");return}if(t.initialView=s.initialView,!s.views||typeof s.views!="object"){z(!1,"'views' must be an object","error");return}t.views=s.views,t.isValid=!0,t.preload=Array.isArray(s.preload)?s.preload:[],t.debug=typeof s.debug=="boolean"?s.debug:!1,t.componentContext=s.componentContext&&typeof s.componentContext=="object"?s.componentContext:new Map;let e=s.transitions&&typeof s.transitions=="object"?s.transitions:null,r=e&&e.cover,o=typeof r=="string"?r.toLowerCase():null,n=o==="width"||o==="height";t.cover=n?o:"width",!n&&e&&r!==void 0&&z(!1,`'transitions.cover' must be either "width" or "height". Falling back to "width".`,"warn"),t.duration=e&&typeof e.duration=="string"?e.duration:"1s",t.ease=e&&typeof e.ease=="string"?e.ease:"ease-in-out",t.deferred=typeof s.deferred=="boolean"?s.deferred:!1;let i=e&&e.hideTrigger;i===!0||i==="fade"?t.hideTrigger=i:(t.hideTrigger=!1,i!=null&&i!==!1&&z(!1,`'transitions.hideTrigger' must be true or "fade". Falling back to false.`,"warn"));let a=e&&e.effects;Array.isArray(a)&&a.length>=2&&typeof a[0]=="string"&&typeof a[1]=="string"?t.effects=[a[0],a[1]]:Array.isArray(a)&&a.length===1&&typeof a[0]=="string"?t.effects=[a[0],a[0]]:(t.effects=["none","none"],a!=null&&z(!1,`'transitions.effects' must be an array of 1-2 CSS filter strings (e.g. ["blur(3px)", "blur(8px) saturate(0)"]). Falling back to no effects.`,"warn"));let c=e&&e.driver,l=typeof c=="string"?c.toLowerCase():null,u=typeof c=="function"?c:null;if(l!==null){let y=["css","waapi","none","anime","gsap","motion"];t.transitionDriver=y.includes(l)?l:"css",y.includes(l)||z(!1,`'transitions.driver' must be "css", "waapi", "none", "anime", "gsap", "motion", or a function. Got "${c}". Falling back to "css".`,"warn")}else u!==null?t.transitionDriver=u:t.transitionDriver="css";let f=e&&e.stagger;typeof f=="number"&&f>0?t.stagger=f:(t.stagger=0,f!=null&&f!==0&&z(!1,"'transitions.stagger' must be a positive number (ms). Falling back to 0 (disabled).","warn")),t.parallax=0,t.threshold=null;let h=e&&e.threshold;h&&typeof h=="object"&&h.enabled&&(t.threshold={duration:typeof h.duration=="number"&&h.duration>0?h.duration:300,commitAt:typeof h.commitAt=="number"&&h.commitAt>0&&h.commitAt<=1?h.commitAt:.5});let d=s.lateralNav;if(d===!1)t.lateralNav=!1;else if(d&&typeof d=="object"){let y=typeof d.mode=="string"?d.mode.toLowerCase():null,g=y==="auto"||y==="always";y&&!g&&z(!1,`'lateralNav.mode' must be "auto" or "always". Falling back to "auto".`,"warn"),t.lateralNav={mode:g?y:"auto",arrows:typeof d.arrows=="boolean"?d.arrows:!0,dots:typeof d.dots=="boolean"?d.dots:!0,keepAlive:d.keepAlive===!0||d.keepAlive==="visible"?d.keepAlive:!1}}else t.lateralNav={mode:"auto",arrows:!0,dots:!0,keepAlive:!1};let m=s.depthNav;if(m===!1)t.depthNav=!1;else if(m&&typeof m=="object"){let y=typeof m.position=="string"?m.position.toLowerCase():null,g=y==="bottom-left"||y==="top-left";y&&!g&&z(!1,`'depthNav.position' must be "bottom-left" or "top-left". Falling back to "bottom-left".`,"warn"),t.depthNav={position:g?y:"bottom-left"}}else t.depthNav={position:"bottom-left"};let p=s.lateralNav&&typeof s.lateralNav=="object"?s.lateralNav.position:void 0;if(t.lateralNav&&typeof t.lateralNav=="object"){let y=p==="bottom-center"||p==="top-center";p&&!y&&z(!1,`'lateralNav.position' must be "bottom-center" or "top-center". Falling back to "bottom-center".`,"warn"),t.lateralNav.position=y?p:"bottom-center"}let v=s.inputs;v&&typeof v=="object"?t.inputs={wheel:typeof v.wheel=="boolean"?v.wheel:!0,keyboard:typeof v.keyboard=="boolean"?v.keyboard:!0,click:typeof v.click=="boolean"?v.click:!0,touch:typeof v.touch=="boolean"?v.touch:!0}:t.inputs={wheel:!0,keyboard:!0,click:!0,touch:!0}}function Zt(s,t,e,r,o){let n=e/s,i=r/t;return o==="height"?{scale:i,scaleInv:1/i}:{scale:n,scaleInv:1/n}}function $t(s,t,e,r){let o=t.left,n=t.top,i=s.x-o+(s.width-e.width*r)/2,a=s.y-n+(s.height-e.height*r)/2;return`translate(${i}px, ${a}px) scale(${r})`}function Nt(s,t,e){let r=t.left,o=t.top,n=s.x-r+(s.width-e.width)/2,i=s.y-o+(s.height-e.height)/2;return`translate(${n}px, ${i}px)`}function Mt(s,t){let e=s.x+s.width/2-t.x,r=s.y+s.height/2-t.y;return`${e}px ${r}px`}function Ft(s,t,e,r,o=0){let n=s.width/2-t.width/2-t.x+e.x,i=s.height/2-t.height/2-t.y+e.y,a=1-o,c=n*a,l=i*a;return{x:n,y:i,transform:`translate(${c}px, ${l}px) scale(${r})`}}function Dt({canvasRect:s,canvasOffset:t,triggerRect:e,previousViewRectAtBaseTransform:r,lastViewZoomedElementRect:o,previousViewRectWithPreviousAtEndTransform:n,scale:i,preScale:a,parallax:c=0}){let l=t.left,u=t.top,f=s.width/2-e.width/2-e.x,h=s.height/2-e.height/2-e.y,d=f+(r.x-o.x)+n.x-l+(n.width-o.width)/2,m=h+(r.y-o.y)+n.y-u+(n.height-o.height)/2,p=1-Math.min(c*2,.9);return`translate(${d*p}px, ${m*p}px) scale(${i*a})`}function Ht(s,t,e,r,o){return`translate(${s-e.left}px, ${t-e.top}px) scale(${r*o})`}function nt(s){let t=s.split(/\s+/);return{x:parseFloat(t[0])||0,y:parseFloat(t[1])||0}}function G(s){let t=s.match(/translate\(\s*([-\d.]+)px[,\s]+([-\d.]+)px\s*\)/),e=s.match(/scale\(\s*([-\d.]+)\s*\)/);return{tx:t?parseFloat(t[1]):0,ty:t?parseFloat(t[2]):0,scale:e?parseFloat(e[1]):1}}function it(s,t,e,r,o,n,i,a,c,l){let u=t.x-e.x*(1-n)-r,f=t.y-e.y*(1-n)-o,h=t.width/n,d=t.height/n,m=u+e.x+(s.x-u-e.x-r)/n,p=f+e.y+(s.y-f-e.y-o)/n,v=s.width/n,y=s.height/n,g=u+i.x+a+l*(m-u-i.x),w=f+i.y+c+l*(p-f-i.y),b=v*l,L=y*l;return{x:g,y:w,width:b,height:L,left:g,top:w,right:g+b,bottom:w+L}}function Q(s,t,e){return{viewName:s,backwardState:t,forwardState:e}}function jt(s){return{detachedNode:s}}function Rt(s,t,e,r,o){let n=[];return t!==null&&n.push(t),e!==null&&n.push(e),r!==null&&n.push(r),o!==null&&n.push(o),{zoomLevel:s,views:n}}function qt(s){let t=s.views?.[3];if(!(!t||!(t.detachedNode instanceof Node)))return t.detachedNode}var at=class{#t=new Map;set(t,e,r=null){this.#t.set(t,{node:e,expires:r?Date.now()+r:null})}get(t){let e=this.#t.get(t);return e?e.expires!==null&&Date.now()>e.expires?(this.#t.delete(t),null):e.node.cloneNode(!0):null}has(t){let e=this.#t.get(t);return e?e.expires!==null&&Date.now()>e.expires?(this.#t.delete(t),!1):!0:!1}invalidate(t){this.#t.delete(t)}clear(){this.#t.clear()}};var Ee=300*1e3,_e=/^https?:\/\/|^\/|\.html$|\.php$/i,lt=class{#t;#e;#r;#s=new Map;constructor(t={}){this.#r=t,this.#t=new ot(t),this.#e=new at}#o(t){return typeof this.#r[t]=="string"}#n(t){return _e.test(t)?Ee:null}async get(t,e=null){if(this.#o(t)){let r=this.#e.get(t);if(r)return r;if(this.#s.has(t))return this.#s.get(t);let o=(async()=>{let n=await this.#t.resolve(t,e),i=this.#r[t],a=this.#n(i);return this.#e.set(t,n,a),this.#s.delete(t),n})();return this.#s.set(t,o),o}return this.#t.resolve(t,e)}async preloadEager(t,e=null){!Array.isArray(t)||t.length===0||await Promise.all(t.map(r=>this.get(r,e)))}prefetch(t,e=null){this.get(t,e).catch(()=>{})}prefetchOnHover(t,e=null){this.prefetch(t,e)}scanAndPrefetch(t,e=null){if(!t||!t.querySelectorAll)return;t.querySelectorAll(".zoom-me[data-to]").forEach(o=>{let n=o.dataset.to;n&&(o.hasAttribute("role")||o.setAttribute("role","button"),o.hasAttribute("aria-label")||o.setAttribute("aria-label",`Zoom to ${n}`),this.prefetch(n,e))})}};function V(s){s&&(s.style.contentVisibility="hidden")}function R(s){s&&(s.style.contentVisibility="visible")}function $(s){if(typeof s=="number"&&!Number.isNaN(s))return Math.max(0,s);let e=String(s).match(/^(\d+(?:\.\d+)?)\s*(ms|s)?$/i);if(!e)return 500;let r=parseFloat(e[1]);return(e[2]||"s").toLowerCase()==="ms"?Math.max(0,r):Math.max(0,r*1e3)}function ct(s){return $(s)/1e3}function S(s,t){if(s.classList.contains("is-new-current-view")){let e=t.views[0].forwardState;s.classList.replace("is-new-current-view","is-current-view"),s.classList.remove("zoom-current-view","has-no-events"),s.style.transformOrigin=e.origin,s.style.transform=e.transform;return}if(s.classList.contains("is-previous-view")){let e=t.views[1].forwardState;s.classList.remove("zoom-previous-view","has-no-events"),s.style.transformOrigin=e.origin,s.style.transform=e.transform;return}if(s.classList.contains("is-last-view")){let e=t.views[2].forwardState;s.classList.remove("zoom-last-view","has-no-events"),s.style.transformOrigin=e.origin,s.style.transform=e.transform}}function k(s,t){s.classList.remove("zoom-previous-view-reverse","has-no-events","has-effect","has-effect-reverse"),s.style.removeProperty("--z-effect-filter"),s.style.transformOrigin="0 0",s.style.transform=t.transform}function C(s,t){s.classList.remove("zoom-last-view-reverse","has-no-events","has-effect","has-effect-reverse"),s.style.removeProperty("--z-effect-filter"),s.style.transformOrigin=t.origin,s.style.transform=t.transform}function T(s,t){try{t&&t.removeChild(s)}catch{try{s.parentElement&&t.removeChild(s.parentElement)}catch{}}}function x(...s){for(let t of s)t&&(t.classList.remove("hide"),R(t))}function M(s,t){let{currentView:e,previousView:r,backView:o,backViewState:n,lastView:i,lastViewState:a,incomingTransformEnd:c,currentStage:l,canvas:u}=s,f=l.views[0];x(e),e.classList.replace("is-new-current-view","is-current-view"),e.classList.remove("zoom-current-view","has-no-events"),e.style.transformOrigin=f.forwardState.origin,e.style.transform=c||f.forwardState.transform,o&&n&&(o.style.transform=n.transformEnd),i&&a&&(i.style.transform=a.transformEnd),s.keepAlive||T(r,u),t()}var F=150;function D(s,t){let e=!1,r=setTimeout(()=>{e||(e=!0,s())},t);return{finish(){e||(e=!0,clearTimeout(r),s())},safetyTimer:r}}function Bt(){return{a:1,b:0,c:0,d:1,e:0,f:0}}function xe(s){if(!s||s==="none")return Bt();let t=String(s).match(/matrix\(([-\d.eE]+),\s*([-\d.eE]+),\s*([-\d.eE]+),\s*([-\d.eE]+),\s*([-\d.eE]+),\s*([-\d.eE]+)\)/);return t?{a:parseFloat(t[1])||1,b:parseFloat(t[2])||0,c:parseFloat(t[3])||0,d:parseFloat(t[4])||1,e:parseFloat(t[5])||0,f:parseFloat(t[6])||0}:Bt()}function Y(s){return`matrix(${s.a}, ${s.b}, ${s.c}, ${s.d}, ${s.e}, ${s.f})`}function X(s,t,e){return s+(t-s)*e}function U(s,t,e){let r=Math.max(0,Math.min(1,e));return{a:X(s.a,t.a,r),b:X(s.b,t.b,r),c:X(s.c,t.c,r),d:X(s.d,t.d,r),e:X(s.e,t.e,r),f:X(s.f,t.f,r)}}function P(s,t,e){s.style.transformOrigin=t,s.style.transform=e;try{s.getBoundingClientRect()}catch{}return xe(getComputedStyle(s).transform)}function Vt(s,t){let{type:e,currentView:r,previousView:o,lastView:n,currentStage:i,duration:a,ease:c,canvas:l}=s;if(!r||!o||!i){t();return}e==="lateral"?Le(s,t):e==="zoomIn"?Te(r,o,n,i,a,c,t):e==="zoomOut"?ze(r,o,n,i,a,c,l,t):t()}function Le(s,t){let{currentView:e,previousView:r,backView:o,backViewState:n,lastView:i,lastViewState:a,incomingTransformStart:c,incomingTransformEnd:l,outgoingTransform:u,outgoingTransformEnd:f,currentStage:h,duration:d,ease:m,canvas:p}=s,v=$(d),y=h.views[0];x(e),e.classList.replace("is-new-current-view","is-current-view"),e.classList.remove("zoom-current-view","has-no-events"),e.style.transformOrigin=y.forwardState.origin,o&&n&&(I(o,d,m,{"--lateral-from":n.transformStart,"--lateral-to":n.transformEnd}),o.classList.add("zoom-lateral-back")),i&&a&&(I(i,d,m,{"--lateral-from":a.transformStart,"--lateral-to":a.transformEnd}),i.classList.add("zoom-lateral-back")),s.keepAlive!=="visible"&&(I(r,d,m,{"--lateral-out-from":u,"--lateral-out-to":f}),r.classList.add("zoom-lateral-out")),I(e,d,m,{"--lateral-in-from":c,"--lateral-in-to":l}),e.classList.add("zoom-lateral-in");let g=s.keepAlive==="visible"?[e]:[r,e];o&&n&&g.push(o),i&&a&&g.push(i);let w=g.length,{finish:b}=D(()=>{vt(g,L),s.keepAlive?(r.classList.remove("zoom-lateral-out"),r.style.opacity=""):T(r,p),o&&(o.classList.remove("zoom-lateral-back"),o.style.transform=n?.transformEnd||o.style.transform),i&&(i.classList.remove("zoom-lateral-back"),i.style.transform=a?.transformEnd||i.style.transform),e.classList.remove("zoom-lateral-in"),e.style.transform=l,t()},v+F);function L(Z){let H=Z?.target;H&&H.removeEventListener("animationend",L),w--,w<=0&&b()}g.forEach(Z=>Z.addEventListener("animationend",L))}function Te(s,t,e,r,o,n,i){x(s,t,e);let a=r.stagger||0;I(s,o,n,{"--current-view-transform-start":r.views[0].backwardState.transform,"--current-view-transform-end":r.views[0].forwardState.transform}),a>0&&s.style.setProperty("animation-delay","0ms"),I(t,o,n,{"--previous-view-transform-start":r.views[1].backwardState.transform,"--previous-view-transform-end":r.views[1].forwardState.transform}),a>0&&t.style.setProperty("animation-delay",`${a}ms`),e&&(I(e,o,n,{"--last-view-transform-start":r.views[2].backwardState.transform,"--last-view-transform-end":r.views[2].forwardState.transform}),a>0&&e.style.setProperty("animation-delay",`${a*2}ms`)),s.classList.add("zoom-current-view"),t.classList.add("zoom-previous-view"),e&&e.classList.add("zoom-last-view");let c=e?[s,t,e]:[s,t],l=c.length,u=$(o),f=e?a*2:a,{finish:h}=D(()=>{vt(c,d),c.forEach(m=>m.style.removeProperty("animation-delay")),i()},u+f+F);function d(m){let p=m?.target;if(p&&p.removeEventListener("animationend",d),p?.isConnected)try{S(p,r)}catch{}l--,l<=0&&h()}c.forEach(m=>m.addEventListener("animationend",d))}function ze(s,t,e,r,o,n,i,a){let c=r.views[0],l=r.views[1],u=e&&r.views[2]?r.views[2]:null,f=r.stagger||0;I(s,o,n,{"--current-view-transform-start":c.backwardState.transform,"--current-view-transform-end":c.forwardState.transform}),f>0&&s.style.setProperty("animation-delay","0ms"),I(t,o,n,{"--previous-view-transform-start":l.backwardState.transform,"--previous-view-transform-end":l.forwardState.transform}),f>0&&t.style.setProperty("animation-delay",`${f}ms`),e&&u&&(I(e,o,n,{"--last-view-transform-start":u.backwardState.transform,"--last-view-transform-end":u.forwardState.transform}),f>0&&e.style.setProperty("animation-delay",`${f*2}ms`)),s.classList.add("zoom-current-view-reverse"),t.classList.add("zoom-previous-view-reverse"),e&&e.classList.add("zoom-last-view-reverse");let h=e?[s,t,e]:[s,t],d=h.length,m=$(o),p=e?f*2:f,{finish:v}=D(()=>{vt(h,y),h.forEach(g=>g.style.removeProperty("animation-delay")),a()},m+p+F);function y(g){let w=g?.target;if(w&&w.removeEventListener("animationend",y),w?.isConnected)try{ke(w,r,i)}catch{}d--,d<=0&&v()}h.forEach(g=>g.addEventListener("animationend",y))}function ke(s,t,e){if(s.classList.contains("zoom-current-view-reverse")){T(s,e);return}if(s.classList.contains("zoom-previous-view-reverse")){k(s,t.views[1].backwardState);return}s.classList.contains("zoom-last-view-reverse")&&C(s,t.views[2].backwardState)}function I(s,t,e,r){s.style.setProperty("--zoom-duration",t),s.style.setProperty("--zoom-ease",e);for(let[o,n]of Object.entries(r))s.style.setProperty(o,n)}function vt(s,t){for(let e of s)e?.removeEventListener&&e.removeEventListener("animationend",t)}function Xt(s,t){let{type:e,currentView:r,previousView:o,lastView:n,currentStage:i,canvas:a}=s;if(!r||!o||!i){t();return}if(e==="lateral"){M(s,t);return}if(e==="zoomIn"){x(r,o,n),S(r,i),S(o,i),n&&S(n,i),t();return}if(e==="zoomOut"){T(r,a),x(o,n),k(o,i.views[1].backwardState),n&&C(n,i.views[2].backwardState),t();return}t()}function Yt(s,t){let{type:e,currentView:r,previousView:o,lastView:n,currentStage:i,duration:a,ease:c,canvas:l}=s;if(!r||!o||!i){t();return}let u=$(a);if(e==="lateral"){Ae(s,u,c,t);return}if(e==="zoomIn"){Ce(r,o,n,i,u,c,t);return}if(e==="zoomOut"){Oe(r,o,n,i,u,c,l,t);return}t()}function Ce(s,t,e,r,o,n,i){x(s,t,e);let a=r.views[0],c=r.views[1],l=e&&r.views[2]?r.views[2]:null;s.style.transformOrigin=a.backwardState.origin,s.style.transform=a.backwardState.transform,t.style.transformOrigin=c.backwardState.origin,t.style.transform=c.backwardState.transform,l&&(e.style.transformOrigin=l.backwardState.origin,e.style.transform=l.backwardState.transform);let u=r.stagger||0,f=[];f.push(s.animate([{transform:a.backwardState.transform},{transform:a.forwardState.transform}],{duration:o,easing:n,fill:"forwards"})),f.push(t.animate([{transform:c.backwardState.transform},{transform:c.forwardState.transform}],{duration:o,delay:u,easing:n,fill:"forwards"})),l&&f.push(e.animate([{transform:l.backwardState.transform},{transform:l.forwardState.transform}],{duration:o,delay:u*2,easing:n,fill:"forwards"}));let h=l?u*2:u,{finish:d}=D(()=>{yt(f);try{S(s,r),S(t,r),e&&S(e,r)}catch{}i()},o+h+F);Promise.all(f.map(m=>m.finished)).then(d).catch(d)}function Oe(s,t,e,r,o,n,i,a){let c=r.views[0],l=r.views[1],u=e&&r.views[2]?r.views[2]:null,f=l.backwardState,h=u?u.backwardState:null;s.style.transformOrigin=c.forwardState.origin,s.style.transform=c.forwardState.transform,requestAnimationFrame(()=>{requestAnimationFrame(()=>{t.style.transformOrigin=l.forwardState.origin,e&&u&&(e.style.transformOrigin=u.forwardState.origin);let d=t.style.transform||getComputedStyle(t).transform||l.forwardState.transform,m=e&&u?e.style.transform||getComputedStyle(e).transform||u.forwardState.transform:null,p=r.stagger||0,v=[];v.push(s.animate([{transform:c.forwardState.transform},{transform:c.backwardState.transform}],{duration:o,easing:n,fill:"forwards"})),v.push(t.animate([{transform:d},{transform:f.transform}],{duration:o,delay:p,easing:n,fill:"both"})),e&&h&&v.push(e.animate([{transform:m},{transform:h.transform}],{duration:o,delay:p*2,easing:n,fill:"both"})),t.style.removeProperty("transform"),e&&e.style.removeProperty("transform");let y=e&&h?p*2:p,{finish:g}=D(()=>{yt(v);try{T(s,i),k(t,f),e&&h&&C(e,h)}catch{}a()},o+y+F);Promise.all(v.map(w=>w.finished)).then(g).catch(g)})})}function Ae(s,t,e,r){let{currentView:o,previousView:n,backView:i,backViewState:a,lastView:c,lastViewState:l,incomingTransformStart:u,incomingTransformEnd:f,outgoingTransform:h,outgoingTransformEnd:d,currentStage:m,canvas:p}=s,v=m.views[0];x(o),o.classList.replace("is-new-current-view","is-current-view"),o.classList.remove("zoom-current-view","has-no-events"),o.style.transformOrigin=v.forwardState.origin;let y=[];y.push(o.animate([{transform:u,opacity:0},{transform:f,opacity:1}],{duration:t,easing:e,fill:"forwards"})),s.keepAlive!=="visible"&&y.push(n.animate([{transform:h,opacity:1},{transform:d,opacity:0}],{duration:t,easing:e,fill:"forwards"})),i&&a&&y.push(i.animate([{transform:a.transformStart},{transform:a.transformEnd}],{duration:t,easing:e,fill:"forwards"})),c&&l&&y.push(c.animate([{transform:l.transformStart},{transform:l.transformEnd}],{duration:t,easing:e,fill:"forwards"}));let{finish:g}=D(()=>{yt(y),o.style.transform=f||v.forwardState.transform,i&&a&&(i.style.transform=a.transformEnd),c&&l&&(c.style.transform=l.transformEnd),s.keepAlive?(n.style.opacity="",n.style.transform=d):T(n,p),r()},t+F);Promise.all(y.map(w=>w.finished)).then(g).catch(g)}function yt(s){for(let t of s)try{t.cancel()}catch{}}function Ut(s,t){let e=typeof globalThis<"u"&&globalThis.anime;if(!e||typeof e!="function"){console.warn('Zumly anime driver: Anime.js not loaded. Add <script src="https://cdnjs.cloudflare.com/ajax/libs/animejs/3.2.2/anime.min.js"><\/script>'),t();return}let{type:r,currentView:o,previousView:n,lastView:i,currentStage:a,duration:c,ease:l,canvas:u}=s;if(!o||!n||!a){t();return}let f=$(c);r==="lateral"?M(s,t):r==="zoomIn"?Pe(e,o,n,i,a,f,l,t):r==="zoomOut"?Ie(e,o,n,i,a,f,l,u,t):t()}function Pe(s,t,e,r,o,n,i,a){x(t,e,r);let c=o.views[0],l=o.views[1],u=r&&o.views[2]?o.views[2]:null,f=Wt([{el:t,backward:c.backwardState,forward:c.forwardState},{el:e,backward:l.backwardState,forward:l.forwardState},...u?[{el:r,backward:u.backwardState,forward:u.forwardState}]:[]],"forward");Kt(f,0);let h=o.stagger||0,d=n+(u?h*2:h),m={value:0};s({targets:m,value:1,duration:d,easing:Qt(i),update:()=>{let p=m.value*d;Gt(f,p,n,h)},complete:()=>{S(t,o),S(e,o),r&&S(r,o),a()}})}function Ie(s,t,e,r,o,n,i,a,c){let l=o.views[0],u=o.views[1],f=r&&o.views[2]?o.views[2]:null,h=u.backwardState,d=f?f.backwardState:null,m=Wt([{el:t,backward:l.backwardState,forward:l.forwardState},{el:e,backward:u.backwardState,forward:u.forwardState},...f?[{el:r,backward:f.backwardState,forward:f.forwardState}]:[]],"backward");Kt(m,0);let p=o.stagger||0,v=n+(f?p*2:p),y={value:0};s({targets:y,value:1,duration:v,easing:Qt(i),update:()=>{let g=y.value*v;Gt(m,g,n,p)},complete:()=>{T(t,a),k(e,h),r&&d&&C(r,d),c()}})}function Wt(s,t){return s.map(({el:e,backward:r,forward:o})=>{if(t==="forward"){let n=P(e,r.origin,r.transform),i=P(e,r.origin,o.transform);return{el:e,from:n,to:i}}else{let n=P(e,o.origin,o.transform),i=P(e,o.origin,r.transform);return{el:e,from:n,to:i}}})}function Kt(s,t){for(let{el:e,from:r,to:o}of s)e.style.transform=Y(U(r,o,t))}function Gt(s,t,e,r){for(let o=0;o<s.length;o++){let{el:n,from:i,to:a}=s[o],c=o*r,l=Math.max(0,t-c),u=e>0?Math.min(1,l/e):1;n.style.transform=Y(U(i,a,u))}}function Qt(s){if(typeof s!="string")return"easeInOutQuad";let t=s.toLowerCase();return t==="linear"?"linear":t.includes("ease-in-out")?"easeInOutQuad":t.includes("ease-in")?"easeInQuad":t.includes("ease-out")?"easeOutQuad":t}function Jt(s,t){let e=typeof globalThis<"u"&&globalThis.gsap;if(!e||typeof e.to!="function"){console.warn('Zumly GSAP driver: GSAP not loaded. Add <script src="https://cdnjs.cloudflare.com/ajax/libs/gsap/3.12.5/gsap.min.js"><\/script>'),t();return}let{type:r,currentView:o,previousView:n,lastView:i,currentStage:a,duration:c,ease:l,canvas:u}=s;if(!o||!n||!a){t();return}let f=ct(c);r==="lateral"?M(s,t):r==="zoomIn"?Ze(e,o,n,i,a,f,l,t):r==="zoomOut"?$e(e,o,n,i,a,f,l,u,t):t()}function Ze(s,t,e,r,o,n,i,a){x(t,e,r);let c=o.views[0],l=o.views[1],u=r&&o.views[2]?o.views[2]:null;ft(t,c.backwardState),ft(e,l.backwardState),u&&ft(r,u.backwardState);let f=s.timeline({onComplete:()=>{S(t,o),S(e,o),r&&S(r,o),a()}}),h=te(i),d=(o.stagger||0)/1e3;f.to(t,{transform:c.forwardState.transform,duration:n,ease:h},0),f.to(e,{transform:l.forwardState.transform,duration:n,ease:h},d),u&&f.to(r,{transform:u.forwardState.transform,duration:n,ease:h},d*2)}function $e(s,t,e,r,o,n,i,a,c){let l=o.views[0],u=o.views[1],f=r&&o.views[2]?o.views[2]:null,h=u.backwardState,d=f?f.backwardState:null;ft(t,l.forwardState);let m=e.style.transform||getComputedStyle(e).transform||u.forwardState.transform,p=r&&f?r.style.transform||getComputedStyle(r).transform||f.forwardState.transform:null;e.style.transformOrigin=u.forwardState.origin,r&&f&&(r.style.transformOrigin=f.forwardState.origin);let v=te(i),y=s.timeline({onComplete:()=>{T(t,a),k(e,h),r&&d&&C(r,d),c()}}),g=(o.stagger||0)/1e3;s.set(e,{transform:m}),r&&p&&s.set(r,{transform:p}),y.fromTo(t,{transform:l.forwardState.transform},{transform:l.backwardState.transform,duration:n,ease:v},0),y.fromTo(e,{transform:m},{transform:h.transform,duration:n,ease:v},g),r&&d&&y.fromTo(r,{transform:p},{transform:d.transform,duration:n,ease:v},g*2)}function ft(s,t){s.style.transformOrigin=t.origin,s.style.transform=t.transform}function te(s){if(typeof s!="string")return"power2.inOut";let t=s.toLowerCase();return t==="linear"?"none":t.includes("ease-in-out")?"power2.inOut":t.includes("ease-in")?"power2.in":t.includes("ease-out")?"power2.out":t}function ee(s,t){let e=Fe();if(!e||typeof e!="function"){console.warn('Zumly Motion driver: Motion not loaded. Add <script src="https://cdn.jsdelivr.net/npm/motion@11/dist/motion.min.js"><\/script>'),t();return}let{type:r,currentView:o,previousView:n,lastView:i,currentStage:a,duration:c,ease:l,canvas:u}=s;if(!o||!n||!a){t();return}let f=ct(c);r==="lateral"?M(s,t):r==="zoomIn"?Ne(e,o,n,i,a,f,l,t):r==="zoomOut"?Me(e,o,n,i,a,f,l,u,t):t()}function Ne(s,t,e,r,o,n,i,a){x(t,e,r);let c=se(t,e,r,o,"forward");re(c,0);let l=o.stagger||0,u=l/1e3,f=n*1e3,h=n+(c.length>2?u*2:u);s(0,1,{duration:h,ease:ne(i),onUpdate:m=>{let p=m*h*1e3;oe(c,p,f,l)}}).then(()=>{S(t,o),S(e,o),r&&S(r,o),a()}).catch(()=>a())}function Me(s,t,e,r,o,n,i,a,c){let l=o.views[1],u=r&&o.views[2]?o.views[2]:null,f=l.backwardState,h=u?u.backwardState:null,d=se(t,e,r,o,"backward");re(d,0);let m=o.stagger||0,p=m/1e3,v=n*1e3,y=n+(d.length>2?p*2:p);s(0,1,{duration:y,ease:ne(i),onUpdate:w=>{let b=w*y*1e3;oe(d,b,v,m)}}).then(()=>{T(t,a),k(e,f),r&&h&&C(r,h),c()}).catch(()=>c())}function se(s,t,e,r,o){let n=r.views[0],i=r.views[1],a=e&&r.views[2]?r.views[2]:null,c=[{el:s,backward:n.backwardState,forward:n.forwardState},{el:t,backward:i.backwardState,forward:i.forwardState}];return a&&c.push({el:e,backward:a.backwardState,forward:a.forwardState}),c.map(({el:l,backward:u,forward:f})=>o==="forward"?{el:l,from:P(l,u.origin,u.transform),to:P(l,u.origin,f.transform)}:{el:l,from:P(l,f.origin,f.transform),to:P(l,f.origin,u.transform)})}function re(s,t){for(let{el:e,from:r,to:o}of s)e.style.transform=Y(U(r,o,t))}function oe(s,t,e,r){for(let o=0;o<s.length;o++){let{el:n,from:i,to:a}=s[o],c=o*r,l=Math.max(0,t-c),u=e>0?Math.min(1,l/e):1;n.style.transform=Y(U(i,a,u))}}function Fe(){let s=typeof globalThis<"u"?globalThis:typeof window<"u"?window:{};return s.motion?.animate||s.Motion?.animate||s.animate}function ne(s){if(typeof s!="string")return"easeInOut";let t=s.toLowerCase();return t==="linear"?"linear":t.includes("ease-in-out")?"easeInOut":t.includes("ease-in")?"easeIn":t.includes("ease-out")?"easeOut":s}var ie={css:Vt,waapi:Yt,none:Xt,anime:Ut,gsap:Jt,motion:ee};function ae(s){if(typeof s=="function")return{runTransition(r,o){s(r,o)}};let t=typeof s=="string"?s.toLowerCase():"css";return{runTransition:ie[t]||ie.css}}function De(s){if(typeof s!="string"||s.trim()==="")return{tx:0,ty:0,rest:""};let t=s.match(/translate\s*\(\s*([-\d.eE]+)px\s*,\s*([-\d.eE]+)px\s*\)/);if(!t)return null;let e=parseFloat(t[1]),r=parseFloat(t[2]),o=s.replace(/translate\s*\([^)]+\)\s*/,"").trim();return{tx:e,ty:r,rest:o}}function wt(s,t,e){let r=De(s);if(r===null)return s;let o=r.tx*t,n=r.ty*e,i=r.rest?` ${r.rest}`:"";return`translate(${o}px, ${n}px)${i}`.trim()}function He(s){if(typeof s!="string"||!s.trim())return null;let t=s.trim().match(/^([-\d.eE]+)px\s+([-\d.eE]+)px$/);return t?{x:parseFloat(t[1]),y:parseFloat(t[2])}:null}function gt(s,t,e){let r=He(s);return r===null?s:`${r.x*t}px ${r.y*e}px`}function bt(s,t,e,r,o){if(!s.storedViews?.length||!s.canvas||t<=0||e<=0)return;let n=r/t,i=o/e;for(let d of s.storedViews){let m=d.views;if(Array.isArray(m))for(let p=0;p<m.length;p++){let v=m[p];!v||v.detachedNode||(v.backwardState&&(v.backwardState.transform=wt(v.backwardState.transform,n,i),v.backwardState.origin!=null&&(v.backwardState.origin=gt(v.backwardState.origin,n,i))),v.forwardState&&(v.forwardState.transform=wt(v.forwardState.transform,n,i),v.forwardState.origin!=null&&(v.forwardState.origin=gt(v.forwardState.origin,n,i))))}}let a=s.canvas,c=a.querySelector(".is-current-view"),l=a.querySelector(".is-previous-view"),u=a.querySelector(".is-last-view"),f=a.querySelector(".is-new-current-view"),h=[c,l,u,f].filter(Boolean);for(let d of h){if(!d.style)continue;let m=d.style.transform;m!=null&&m!==""&&(d.style.transform=wt(m,n,i));let p=d.style.transformOrigin;p!=null&&p!==""&&(d.style.transformOrigin=gt(p,n,i))}}var je=8e3,tt=class{constructor(t){if(this.storedViews=[],this.currentStage=null,this.debug=!1,this.trace=[],this.blockEvents=!1,this._blockEventsSafetyTimer=null,this.touchstartX=0,this.touchstartY=0,this.touchendX=0,this.touchendY=0,this.touching=!1,this.lateralHistory=[],this._lastCanvasWidth=0,this._lastCanvasHeight=0,this._pendingResizeCorrection=!1,this._destroyed=!1,this._hooks={},this._plugins=[],It(t,this),!this.isValid){this.notify("is unable to start: invalid or missing required options (mount, initialView, views).","error");return}if(this.canvas=document.querySelector(this.mount),!this.canvas){this.notify(`mount selector "${this.mount}" did not match any element.`,"error"),this.isValid=!1;return}this.transitionDriver=ae(this.transitionDriver),this._onZoom=this.onZoom.bind(this),this._onTouchStart=this.onTouchStart.bind(this),this._onTouchEnd=this.onTouchEnd.bind(this),this._onKeyUp=this.onKeyUp.bind(this),this._onWheel=this.onWheel.bind(this),this._wheelCooldown=!1,this._onPrefetchTrigger=e=>{e.target.classList.contains("zoom-me")&&e.target.dataset.to&&this.prefetcher.prefetch(e.target.dataset.to,{trigger:e.target,...e.target.dataset})},this._onResize=this._handleResize.bind(this),this._resizeDebounceTimer=null,this._RESIZE_DEBOUNCE_MS=80,this.prefetcher=new lt(this.views),this._bindEvents()}_bindEvents(){let t=this.canvas;t&&(t.setAttribute("tabindex","0"),t.setAttribute("role","application"),t.setAttribute("aria-roledescription","zoomable interface"),t.setAttribute("aria-live","polite"),t.addEventListener("mouseup",this._onZoom,!1),t.addEventListener("touchend",this._onZoom,!1),t.addEventListener("touchstart",this._onTouchStart,{passive:!0}),t.addEventListener("touchend",this._onTouchEnd,!1),t.addEventListener("keyup",this._onKeyUp,!1),t.addEventListener("wheel",this._onWheel,{passive:!1}),t.addEventListener("mouseover",this._onPrefetchTrigger,{passive:!0}),t.addEventListener("focusin",this._onPrefetchTrigger,{passive:!0}),window.addEventListener("resize",this._onResize,{passive:!0}),typeof ResizeObserver<"u"&&(this._resizeObserver=new ResizeObserver(()=>this._handleResize()),this._resizeObserver.observe(t)))}_unbindEvents(){let t=this.canvas;t&&(t.removeEventListener("mouseup",this._onZoom,!1),t.removeEventListener("touchend",this._onZoom,!1),t.removeEventListener("touchstart",this._onTouchStart),t.removeEventListener("touchend",this._onTouchEnd,!1),t.removeEventListener("keyup",this._onKeyUp,!1),t.removeEventListener("wheel",this._onWheel),t.removeEventListener("mouseover",this._onPrefetchTrigger),t.removeEventListener("focusin",this._onPrefetchTrigger)),window.removeEventListener("resize",this._onResize),this._resizeObserver&&(this._resizeObserver.disconnect(),this._resizeObserver=null)}_setBlockEvents(){this.blockEvents=!0,this._clearBlockEventsSafety(),this._blockEventsSafetyTimer=setTimeout(()=>{this.blockEvents&&!this._destroyed&&(this.notify("blockEvents safety timeout: driver did not call onComplete. Force-resetting.","warn"),this.blockEvents=!1,this._onTransitionComplete())},je)}_clearBlockEventsSafety(){this._blockEventsSafetyTimer!==null&&(clearTimeout(this._blockEventsSafetyTimer),this._blockEventsSafetyTimer=null)}storeViews(t){this.tracing("storedViews()"),this.storedViews.push(t),this.debug&&console.debug("Zumly storedViews",t)}tracing(t){if(this.debug)if(t==="ended"){let e=this.trace.map((r,o)=>`${o===0?`Instance ${this.instance}: ${r}`:`${r}`}`).join(" > ");this.notify(e),this.trace=[]}else this.trace.push(t)}notify(t,e){return z(this.debug,t,e)}_recordCanvasSize(){if(!this.canvas)return;let t=this.canvas.getBoundingClientRect();this._lastCanvasWidth=t.width,this._lastCanvasHeight=t.height}_handleResize(){this._destroyed||(this._resizeDebounceTimer&&clearTimeout(this._resizeDebounceTimer),this._resizeDebounceTimer=setTimeout(()=>{if(this._resizeDebounceTimer=null,!this.isValid||!this.canvas||this._lastCanvasWidth===0)return;let t=this.canvas.getBoundingClientRect(),e=t.width,r=t.height;if(!(e===this._lastCanvasWidth&&r===this._lastCanvasHeight)){if(this.blockEvents){this._pendingResizeCorrection=!0;return}bt(this,this._lastCanvasWidth,this._lastCanvasHeight,e,r),this._lastCanvasWidth=e,this._lastCanvasHeight=r}},this._RESIZE_DEBOUNCE_MS))}_onTransitionComplete(){if(this._clearBlockEventsSafety(),this._pendingResizeCorrection&&!this.blockEvents){this._pendingResizeCorrection=!1;let t=this.canvas?.getBoundingClientRect();if(t&&this._lastCanvasWidth>0){let e=t.width,r=t.height;(e!==this._lastCanvasWidth||r!==this._lastCanvasHeight)&&bt(this,this._lastCanvasWidth,this._lastCanvasHeight,e,r),this._lastCanvasWidth=e,this._lastCanvasHeight=r}}else this._recordCanvasSize();this._manageFocus()}_manageFocus(){if(!this.canvas)return;let t=this.canvas.querySelector(".is-current-view");if(!t)return;let e=t.querySelector('a[href], button:not([disabled]), input:not([disabled]), select:not([disabled]), textarea:not([disabled]), [tabindex]:not([tabindex="-1"])');e?e.focus({preventScroll:!0}):(t.setAttribute("tabindex","-1"),t.focus({preventScroll:!0}))}_resolveEffects(t){let e=t?.dataset?.withEffects;if(e&&typeof e=="string"){let r=e.split("|").map(o=>o.trim());return[r[0]||"none",r[1]||r[0]||"none"]}return this.effects}_applyEffects(t,e,r,o,n){!r||r[0]==="none"&&r[1]==="none"||(t&&r[0]!=="none"&&(t.style.setProperty("--z-effect-filter",r[0]),t.style.setProperty("--zoom-duration",o),t.style.setProperty("--zoom-ease",n),t.classList.add("has-effect")),e&&r[1]!=="none"&&(e.style.setProperty("--z-effect-filter",r[1]),e.style.setProperty("--zoom-duration",o),e.style.setProperty("--zoom-ease",n),e.classList.add("has-effect")))}_removeEffect(t){if(!t||!t.classList.contains("has-effect"))return;t.classList.remove("has-effect"),t.classList.add("has-effect-reverse");let e=()=>{t.classList.remove("has-effect-reverse"),t.style.removeProperty("--z-effect-filter"),t.removeEventListener("transitionend",e)};t.addEventListener("transitionend",e,{once:!0});let r=t.style.getPropertyValue("--zoom-duration");setTimeout(e,r?parseFloat(r)*(r.includes("ms")?1:1e3)+100:1100)}_resolveHideTrigger(t){return t?.dataset?.hideTrigger!==void 0?t.dataset.hideTrigger==="fade"?"fade":!0:this.hideTrigger}_applyHideTrigger(t,e,r,o,n){!r||!t||(r==="fade"?(t.style.setProperty("--zoom-duration",o),t.style.setProperty("--zoom-ease",n),t.classList.add("z-trigger-fade"),e.style.setProperty("--zoom-duration",o),e.style.setProperty("--zoom-ease",n),e.classList.add("z-view-fade-in"),e.offsetHeight,e.classList.remove("hide")):t.classList.add("z-trigger-hidden"))}_restoreHideTrigger(t,e,r,o){if(!(!e||!t))if(e==="fade"){t.classList.remove("z-trigger-fade"),t.style.setProperty("--zoom-duration",r),t.style.setProperty("--zoom-ease",o),t.classList.add("z-trigger-fade-reverse");let n=()=>{t.classList.remove("z-trigger-fade-reverse"),t.style.removeProperty("--zoom-duration"),t.style.removeProperty("--zoom-ease"),t.removeEventListener("transitionend",n)};t.addEventListener("transitionend",n,{once:!0}),setTimeout(n,parseFloat(r)*(r.includes("ms")?1:1e3)+100)}else t.classList.remove("z-trigger-hidden")}on(t,e){return typeof e!="function"?this:((this._hooks[t]||=[]).push(e),this)}use(t,e){if(!t)return this;let r={plugin:t,options:e};return this._plugins.push(r),this._initialized&&this._installPlugin(r),this}_installPlugin(t){let{plugin:e,options:r}=t;try{typeof e=="function"?e(this,r):typeof e.install=="function"&&e.install(this,r)}catch(o){this.notify(`plugin install error: ${o.message}`,"error")}}off(t,e){return e?this._hooks[t]&&(this._hooks[t]=this._hooks[t].filter(r=>r!==e)):delete this._hooks[t],this}_emit(t,e={}){let r=this._hooks[t];if(!(!r||r.length===0))for(let o of r)try{o(e)}catch(n){this.debug&&console.error(`Zumly hook "${t}" threw:`,n)}}zoomLevel(){return this.storedViews.length}getZoomLevel(){return this.zoomLevel()}getCurrentViewName(){return!this.storedViews||this.storedViews.length===0?null:this.storedViews[this.storedViews.length-1]?.views?.[0]?.viewName??null}back(){if(!this._destroyed){if(this.lateralHistory&&this.lateralHistory.length>0){let t=this.lateralHistory.pop(),e=t&&typeof t=="object"?t.name:t,r=t&&typeof t=="object"?t.entry:void 0,o=t&&typeof t=="object"?t.node:void 0;return this._doLateral(e,!0,{savedEntry:r,keepAliveNode:o})}this.zoomOut()}}async goTo(t,e={}){return this._destroyed?void 0:(e.mode==="lateral"?"lateral":"depth")==="depth"?this.zoomTo(t,e):this._doLateral(t,!1,e)}async zoomTo(t,e={}){if(this._destroyed)return;if(!this.isValid||!this.canvas){this.notify("zoomTo() cannot run: instance is invalid or canvas not found.","error");return}if(typeof t!="string"||!t){this.notify("zoomTo() requires a non-empty view name.","warn");return}if(!Object.prototype.hasOwnProperty.call(this.views,t)){this.notify(`zoomTo("${t}"): view not found in views. Available: ${Object.keys(this.views).join(", ")}`,"warn");return}if(this.getCurrentViewName()===t)return;let o=this.canvas.getBoundingClientRect(),n=Math.max(40,o.width*.1),i=Math.max(40,o.height*.1),c={rect:{x:o.left+(o.width-n)/2,y:o.top+(o.height-i)/2,width:n,height:i},duration:e.duration??this.duration,ease:e.ease??this.ease,props:e.props??{}};await this._doZoomIn(t,c)}async init(){if(this._destroyed)return;if(!this.isValid||!this.canvas){this.notify("init() cannot run: instance is invalid or canvas element was not found.","error");return}this.tracing("init()"),this.preload&&this.preload.length&&await this.prefetcher.preloadEager(this.preload,null);let t=await this.prefetcher.get(this.initialView,null),e=await K(t,this.initialView,this.canvas,!0,this.views,this.componentContext);this._emit("viewMounted",{viewName:this.initialView,node:e}),this.prefetcher.scanAndPrefetch(e,null),this.storeViews({zoomLevel:this.storedViews.length,scale:1,views:[{viewName:this.initialView,backwardState:{origin:"0 0",transform:""}}]}),this.currentStage=this.storedViews[this.storedViews.length-1],this._recordCanvasSize(),this._initialized=!0;for(let r of this._plugins)this._installPlugin(r)}destroy(){this._destroyed||(this._emit("destroy"),this._destroyed=!0,this._clearBlockEventsSafety(),this._resizeDebounceTimer&&(clearTimeout(this._resizeDebounceTimer),this._resizeDebounceTimer=null),this._unbindEvents(),this._removeNav(),this._cleanupLateralKeepAlive(),this.blockEvents=!1,this.storedViews=[],this.currentStage=null,this.lateralHistory=[],this.trace=[],this._hooks={},this.prefetcher=null,this.transitionDriver=null,this.canvas&&(this.canvas.removeAttribute("tabindex"),this.canvas.removeAttribute("role"),this.canvas.removeAttribute("aria-roledescription"),this.canvas.removeAttribute("aria-live")),this.isValid=!1)}async _doZoomIn(t,e){if(this._destroyed)return;this._cleanupLateralKeepAlive(),this.lateralHistory=[],this._emit("beforeZoomIn",{viewName:t}),this.tracing("zoomIn()");let r=this.canvas,o=e.el,n=r.getBoundingClientRect(),i=n.left,a=n.top,l=(this.storedViews.length>0?this.storedViews[this.storedViews.length-1]:null)?.scale??1;this.tracing("getView()");let u=o?{trigger:o,target:document.createElement("div"),context:this.componentContext,props:Object.assign({},o.dataset)}:{target:document.createElement("div"),context:this.componentContext,props:e.props??{}},f=o?.dataset?.deferred!==void 0?!0:this.deferred,h,d=null;if(f){let E=await this.prefetcher.get(t,u);if(this._destroyed)return;for(d=document.createDocumentFragment();E.firstChild;)d.appendChild(E.firstChild);h=await K(E,t,r,!1,{},this.componentContext)}else{let E=await this.prefetcher.get(t,u);if(this._destroyed)return;this.prefetcher.scanAndPrefetch(E,u),h=await K(E,t,r,!1,this.views,this.componentContext)}if(!h)return;f||this._emit("viewMounted",{viewName:t,node:h}),o&&o.classList.add("zoomed");let m=o?(()=>{let E=o.getBoundingClientRect();return{x:E.x,y:E.y,width:E.width,height:E.height}})():e.rect,p=o?o.dataset.withDuration||this.duration:e.duration??this.duration,v=o?o.dataset.withEase||this.ease:e.ease??this.ease,y=o?o.dataset.withCover||this.cover:e.cover??this.cover,g=o?parseInt(o.dataset.withStagger,10)||this.stagger:e.stagger??this.stagger,w=r.querySelector(".is-current-view"),b=r.querySelector(".is-previous-view"),L=r.querySelector(".is-last-view");V(h),V(w),V(b),L&&(V(L),r.removeChild(L));let Z,H,W,_,O,N,A,fe=w.style.transform,Et=w.style.transformOrigin,ue=b?b.style.transform:null,_t=b?b.style.transformOrigin:null;try{w.classList.replace("is-current-view","is-previous-view"),b&&b.classList.replace("is-previous-view","is-last-view");let E=h.getBoundingClientRect(),j=w.getBoundingClientRect(),dt=null,xt=null,st=null;b&&(dt=b.getBoundingClientRect(),st=b.querySelector(".zoomed"),xt=st&&st.getBoundingClientRect?st.getBoundingClientRect():dt);let rt={left:i,top:a},Lt={width:n.width,height:n.height},Tt={width:E.width,height:E.height},zt=Zt(m.width,m.height,E.width,E.height,y);A=zt.scale;let we=zt.scaleInv;Z=$t(m,rt,Tt,we),W=w.style.transform;let kt=Mt(m,j);_=Ft(Lt,m,j,A,this.parallax);let Ct=nt(Et||"0 0"),q=G(W||""),Ot=nt(kt),B=G(_.transform),At=m;if(o&&(At=it(m,j,Ct,q.tx,q.ty,q.scale,Ot,B.tx,B.ty,B.scale)),H=Nt(At,rt,Tt),b){O=b.style.transform;let ge=it(j,j,Ct,q.tx,q.ty,q.scale,Ot,B.tx,B.ty,B.scale),be=Ht(_.x,_.y,rt,A,l),Pt=nt(_t||"0 0"),mt=G(O||""),pt=G(be),Se=it(xt,dt,Pt,mt.tx,mt.ty,mt.scale,Pt,pt.tx,pt.ty,pt.scale);N=Dt({canvasRect:Lt,canvasOffset:rt,triggerRect:m,previousViewRectAtBaseTransform:j,lastViewZoomedElementRect:Se,previousViewRectWithPreviousAtEndTransform:ge,scale:A,preScale:l,parallax:this.parallax})}h.style.transform=Z,w.style.transformOrigin=kt}catch(E){this.notify(`zoomIn geometry computation failed: ${E.message}. Aborting zoom.`,"error"),this.debug&&console.error("Zumly _doZoomIn error:",E),w.style.transform=fe,w.style.transformOrigin=Et,w.classList.contains("is-previous-view")&&w.classList.replace("is-previous-view","is-current-view"),b&&(b.style.transform=ue,b.style.transformOrigin=_t,b.classList.contains("is-last-view")&&b.classList.replace("is-last-view","is-previous-view"));try{r.removeChild(h)}catch{}R(w),R(b);return}let he=Q(h.dataset.viewName,{origin:h.style.transformOrigin,duration:p,ease:v,transform:Z},{origin:h.style.transformOrigin,duration:p,ease:v,transform:H}),de=Q(w.dataset.viewName,{origin:w.style.transformOrigin,duration:p,ease:v,transform:W},{origin:w.style.transformOrigin,duration:p,ease:v,transform:_.transform}),me=b?Q(b.dataset.viewName,{origin:b.style.transformOrigin,duration:p,ease:v,transform:O},{origin:b.style.transformOrigin,duration:p,ease:v,transform:N}):null,pe=L?jt(L):null,et=Rt(this.storedViews.length,he,de,me,pe);et.scale=A,et.stagger=g||0,this.storeViews(et),this.currentStage=this.storedViews[this.storedViews.length-1],this.tracing("setCSSVariables()");let ve=this._resolveEffects(o);this._applyEffects(w,b,ve,p,v);let ht=this._resolveHideTrigger(o);this._applyHideTrigger(o,h,ht,p,v),ht&&(et.hideTriggerMode=ht),this._setBlockEvents();let ye={type:"zoomIn",currentView:h,previousView:w,lastView:b,currentStage:this.currentStage,duration:p,ease:v};this.transitionDriver.runTransition(ye,async()=>{f&&d&&h&&!this._destroyed&&(h.appendChild(d),this.prefetcher.scanAndPrefetch(h,u),typeof this.views[t]=="object"&&typeof this.views[t].mounted=="function"&&await this.views[t].mounted(),this._emit("viewMounted",{viewName:t,node:h})),this.blockEvents=!1,this._onTransitionComplete(),this._updateNav(),this._emit("afterZoomIn",{viewName:t,zoomLevel:this.zoomLevel()}),this.tracing("ended")})}async zoomIn(t){this._destroyed||t?.dataset?.to&&await this._doZoomIn(t.dataset.to,{el:t})}async _doLateral(t,e=!1,r={}){if(this._destroyed||!this.isValid||!this.canvas)return;if(!Object.prototype.hasOwnProperty.call(this.views,t)){this.notify(`goTo("${t}", { mode: 'lateral' }): view not found in views. Available: ${Object.keys(this.views).join(", ")}`,"warn");return}let o=this.canvas.querySelector(".is-current-view");if(!o)return;let n=o.dataset?.viewName;if(n===t)return;this._emit("beforeLateral",{viewName:t,from:n,isBack:e});let i=this.lateralNav&&this.lateralNav.keepAlive;if(!e){this.lateralHistory=this.lateralHistory||[];let _=this.storedViews[this.storedViews.length-1];this.lateralHistory.push({name:n,entry:_.views[0],node:i?o:null})}this.tracing("lateral()");let a=r.duration??this.duration,c=r.ease??this.ease,l=this.canvas.querySelector(".is-previous-view"),u=this.canvas.querySelector(".is-last-view"),f=0,h=0;if(l){let _=l.querySelector(`.zoom-me[data-to="${n}"]`),O=l.querySelector(`.zoom-me[data-to="${t}"]`);if(_&&O){let N=_.getBoundingClientRect(),A=O.getBoundingClientRect();f=N.left+N.width/2-(A.left+A.width/2),h=N.top+N.height/2-(A.top+A.height/2)}else f=this.canvas.getBoundingClientRect().width*.15}let d=null,m=null;if(i&&(e&&r.keepAliveNode?m=r.keepAliveNode:m=this.canvas.querySelector(`.is-lateral-hidden[data-view-name="${t}"]`)),m){if(m.style.display="",m.style.opacity="",m.classList.remove("is-lateral-hidden","zoom-lateral-out"),m.classList.add("is-new-current-view","has-no-events"),d=m,!e&&this.lateralHistory){let _=this.lateralHistory.findIndex(O=>O.node===m);_!==-1&&this.lateralHistory.splice(_,1)}}else{let _={target:document.createElement("div"),context:this.componentContext,props:r.props??{}},O=await this.prefetcher.get(t,_);if(this._destroyed||(this.prefetcher.scanAndPrefetch(O,_),d=await K(O,t,this.canvas,!1,this.views,this.componentContext),!d))return;this._emit("viewMounted",{viewName:t,node:d})}V(d);let p=o.style.transform||"",v=o.style.transformOrigin||"0 0";d.style.transform=p,d.style.transformOrigin=v;let y=this.storedViews[this.storedViews.length-1],g=e&&r.savedEntry?{...r.savedEntry,viewName:t}:Q(t,{origin:v,duration:a,ease:c,transform:p},{origin:v,duration:a,ease:c,transform:p});y.views[0]=g,this.currentStage=y;let w=l?{transformStart:l.style.transform||"",transformEnd:this._computeLateralBackTransform(l.style.transform||"",f,h)}:null,b=u&&y.views[2]?{transformStart:u.style.transform||"",transformEnd:this._computeLateralBackTransform(u.style.transform||"",f*.7,h*.7)}:null;w&&y.views[1]&&(y.views[1].forwardState={...y.views[1].forwardState,transform:w.transformEnd}),b&&y.views[2]&&(y.views[2].forwardState={...y.views[2].forwardState,transform:b.transformEnd});let L=p,Z=this._computeLateralBackTransform(L,-f,-h),H=this._computeLateralBackTransform(p,f,h);this._setBlockEvents();let W={type:"lateral",currentView:d,previousView:o,lastView:u||null,backView:l||null,backViewState:w,lastViewState:b,incomingTransformStart:Z,incomingTransformEnd:L,outgoingTransform:p,outgoingTransformEnd:H,currentStage:this.currentStage,duration:a,ease:c,canvas:this.canvas,slideDeltaX:f,slideDeltaY:h,keepAlive:i&&!e?i:!1};this.transitionDriver.runTransition(W,()=>{i&&!e&&(o.classList.remove("is-current-view","is-new-current-view","has-no-events"),o.classList.add("is-lateral-hidden"),i!=="visible"?o.style.display="none":(o.style.transform=p,o.style.opacity="")),this.blockEvents=!1,this._onTransitionComplete(),this._updateNav(),this._emit("afterLateral",{viewName:t,from:n,isBack:e}),this.tracing("ended")})}_computeLateralBackTransform(t,e,r){let o=t.match(/translate\s*\(\s*([-\d.eE]+)px\s*,\s*([-\d.eE]+)px\s*\)/);if(o){let n=parseFloat(o[1])+e,i=parseFloat(o[2])+r,a=t.replace(/translate\s*\([^)]+\)\s*/,"");return`translate(${n}px, ${i}px) ${a}`.trim()}return`translate(${e}px, ${r}px) ${t}`.trim()}zoomOut(){if(this._destroyed)return;this._cleanupLateralKeepAlive(),this.lateralHistory=[],this._emit("beforeZoomOut",{zoomLevel:this.zoomLevel()}),this.tracing("zoomOut()");let t=this.canvas,e=t.querySelector(".is-current-view"),r=t.querySelector(".is-previous-view");if(!e||!r){this.notify("zoomOut: current or previous view not found (animation may still be running)","warn");return}this._setBlockEvents(),this.currentStage=this.storedViews[this.storedViews.length-1];let o=t.querySelector(".is-last-view");this.tracing("setCSSVariables()");let n=this.currentStage.views[0]?.forwardState?.duration??this.duration,i=this.currentStage.views[0]?.forwardState?.ease??this.ease,a=r.querySelector(".zoomed");a&&a.classList.remove("zoomed");let c=this.currentStage.hideTriggerMode;c&&a&&this._restoreHideTrigger(a,c,n,i),c==="fade"?(e.classList.remove("z-view-fade-in"),e.style.setProperty("--zoom-duration",n),e.style.setProperty("--zoom-ease"