@auticlabs/bulut 1.3.2 → 1.3.3
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/embed.cjs +1 -1
- package/dist/embed.cjs.map +1 -1
- package/dist/embed.js +25 -35
- package/dist/embed.js.map +1 -1
- package/package.json +1 -4
package/dist/embed.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"embed.js","sources":["../node_modules/preact/dist/preact.module.js","../node_modules/preact/jsx-runtime/dist/jsxRuntime.module.js","../node_modules/preact/hooks/dist/hooks.module.js","../src/styles/constants.ts","../node_modules/@heroicons/react/24/outline/esm/ArrowPathIcon.js","../node_modules/@heroicons/react/24/outline/esm/CommandLineIcon.js","../node_modules/@heroicons/react/24/outline/esm/CursorArrowRaysIcon.js","../node_modules/@heroicons/react/24/outline/esm/FaceSmileIcon.js","../node_modules/@heroicons/react/24/outline/esm/HandRaisedIcon.js","../node_modules/@heroicons/react/24/outline/esm/MapIcon.js","../node_modules/@heroicons/react/24/outline/esm/MicrophoneIcon.js","../node_modules/@heroicons/react/24/outline/esm/QueueListIcon.js","../node_modules/@heroicons/react/24/outline/esm/StopIcon.js","../node_modules/@heroicons/react/24/outline/esm/XMarkIcon.js","../src/components/ChatButton.tsx","../src/agent/context.ts","../src/agent/contextConfig.ts","../src/agent/tools.ts","../src/api/client.ts","../src/audio/sfxManager.ts","../src/assets/sfx/sent.mp3","../src/assets/sfx/thinking.mp3","../src/assets/sfx/tool_call.mp3","../src/assets/sfx/completed.mp3","../src/components/SvgIcon.tsx","../src/components/ChatWindow.tsx","../src/assets/icons/bulut_logo.svg?raw","../src/index.tsx"],"sourcesContent":["var n,l,u,t,i,o,r,e,f,c,s,a,h,p={},v=[],y=/acit|ex(?:s|g|n|p|$)|rph|grid|ows|mnc|ntw|ine[ch]|zoo|^ord|itera/i,d=Array.isArray;function w(n,l){for(var u in l)n[u]=l[u];return n}function g(n){n&&n.parentNode&&n.parentNode.removeChild(n)}function _(l,u,t){var i,o,r,e={};for(r in u)\"key\"==r?i=u[r]:\"ref\"==r?o=u[r]:e[r]=u[r];if(arguments.length>2&&(e.children=arguments.length>3?n.call(arguments,2):t),\"function\"==typeof l&&null!=l.defaultProps)for(r in l.defaultProps)void 0===e[r]&&(e[r]=l.defaultProps[r]);return m(l,e,i,o,null)}function m(n,t,i,o,r){var e={type:n,props:t,key:i,ref:o,__k:null,__:null,__b:0,__e:null,__c:null,constructor:void 0,__v:null==r?++u:r,__i:-1,__u:0};return null==r&&null!=l.vnode&&l.vnode(e),e}function b(){return{current:null}}function k(n){return n.children}function x(n,l){this.props=n,this.context=l}function S(n,l){if(null==l)return n.__?S(n.__,n.__i+1):null;for(var u;l<n.__k.length;l++)if(null!=(u=n.__k[l])&&null!=u.__e)return u.__e;return\"function\"==typeof n.type?S(n):null}function C(n){var l,u;if(null!=(n=n.__)&&null!=n.__c){for(n.__e=n.__c.base=null,l=0;l<n.__k.length;l++)if(null!=(u=n.__k[l])&&null!=u.__e){n.__e=n.__c.base=u.__e;break}return C(n)}}function M(n){(!n.__d&&(n.__d=!0)&&i.push(n)&&!$.__r++||o!=l.debounceRendering)&&((o=l.debounceRendering)||r)($)}function $(){for(var n,u,t,o,r,f,c,s=1;i.length;)i.length>s&&i.sort(e),n=i.shift(),s=i.length,n.__d&&(t=void 0,o=void 0,r=(o=(u=n).__v).__e,f=[],c=[],u.__P&&((t=w({},o)).__v=o.__v+1,l.vnode&&l.vnode(t),O(u.__P,t,o,u.__n,u.__P.namespaceURI,32&o.__u?[r]:null,f,null==r?S(o):r,!!(32&o.__u),c),t.__v=o.__v,t.__.__k[t.__i]=t,N(f,t,c),o.__e=o.__=null,t.__e!=r&&C(t)));$.__r=0}function I(n,l,u,t,i,o,r,e,f,c,s){var a,h,y,d,w,g,_,m=t&&t.__k||v,b=l.length;for(f=P(u,l,m,f,b),a=0;a<b;a++)null!=(y=u.__k[a])&&(h=-1==y.__i?p:m[y.__i]||p,y.__i=a,g=O(n,y,h,i,o,r,e,f,c,s),d=y.__e,y.ref&&h.ref!=y.ref&&(h.ref&&B(h.ref,null,y),s.push(y.ref,y.__c||d,y)),null==w&&null!=d&&(w=d),(_=!!(4&y.__u))||h.__k===y.__k?f=A(y,f,n,_):\"function\"==typeof y.type&&void 0!==g?f=g:d&&(f=d.nextSibling),y.__u&=-7);return u.__e=w,f}function P(n,l,u,t,i){var o,r,e,f,c,s=u.length,a=s,h=0;for(n.__k=new Array(i),o=0;o<i;o++)null!=(r=l[o])&&\"boolean\"!=typeof r&&\"function\"!=typeof r?(\"string\"==typeof r||\"number\"==typeof r||\"bigint\"==typeof r||r.constructor==String?r=n.__k[o]=m(null,r,null,null,null):d(r)?r=n.__k[o]=m(k,{children:r},null,null,null):void 0===r.constructor&&r.__b>0?r=n.__k[o]=m(r.type,r.props,r.key,r.ref?r.ref:null,r.__v):n.__k[o]=r,f=o+h,r.__=n,r.__b=n.__b+1,e=null,-1!=(c=r.__i=L(r,u,f,a))&&(a--,(e=u[c])&&(e.__u|=2)),null==e||null==e.__v?(-1==c&&(i>s?h--:i<s&&h++),\"function\"!=typeof r.type&&(r.__u|=4)):c!=f&&(c==f-1?h--:c==f+1?h++:(c>f?h--:h++,r.__u|=4))):n.__k[o]=null;if(a)for(o=0;o<s;o++)null!=(e=u[o])&&0==(2&e.__u)&&(e.__e==t&&(t=S(e)),D(e,e));return t}function A(n,l,u,t){var i,o;if(\"function\"==typeof n.type){for(i=n.__k,o=0;i&&o<i.length;o++)i[o]&&(i[o].__=n,l=A(i[o],l,u,t));return l}n.__e!=l&&(t&&(l&&n.type&&!l.parentNode&&(l=S(n)),u.insertBefore(n.__e,l||null)),l=n.__e);do{l=l&&l.nextSibling}while(null!=l&&8==l.nodeType);return l}function H(n,l){return l=l||[],null==n||\"boolean\"==typeof n||(d(n)?n.some(function(n){H(n,l)}):l.push(n)),l}function L(n,l,u,t){var i,o,r,e=n.key,f=n.type,c=l[u],s=null!=c&&0==(2&c.__u);if(null===c&&null==e||s&&e==c.key&&f==c.type)return u;if(t>(s?1:0))for(i=u-1,o=u+1;i>=0||o<l.length;)if(null!=(c=l[r=i>=0?i--:o++])&&0==(2&c.__u)&&e==c.key&&f==c.type)return r;return-1}function T(n,l,u){\"-\"==l[0]?n.setProperty(l,null==u?\"\":u):n[l]=null==u?\"\":\"number\"!=typeof u||y.test(l)?u:u+\"px\"}function j(n,l,u,t,i){var o,r;n:if(\"style\"==l)if(\"string\"==typeof u)n.style.cssText=u;else{if(\"string\"==typeof t&&(n.style.cssText=t=\"\"),t)for(l in t)u&&l in u||T(n.style,l,\"\");if(u)for(l in u)t&&u[l]==t[l]||T(n.style,l,u[l])}else if(\"o\"==l[0]&&\"n\"==l[1])o=l!=(l=l.replace(f,\"$1\")),r=l.toLowerCase(),l=r in n||\"onFocusOut\"==l||\"onFocusIn\"==l?r.slice(2):l.slice(2),n.l||(n.l={}),n.l[l+o]=u,u?t?u.u=t.u:(u.u=c,n.addEventListener(l,o?a:s,o)):n.removeEventListener(l,o?a:s,o);else{if(\"http://www.w3.org/2000/svg\"==i)l=l.replace(/xlink(H|:h)/,\"h\").replace(/sName$/,\"s\");else if(\"width\"!=l&&\"height\"!=l&&\"href\"!=l&&\"list\"!=l&&\"form\"!=l&&\"tabIndex\"!=l&&\"download\"!=l&&\"rowSpan\"!=l&&\"colSpan\"!=l&&\"role\"!=l&&\"popover\"!=l&&l in n)try{n[l]=null==u?\"\":u;break n}catch(n){}\"function\"==typeof u||(null==u||!1===u&&\"-\"!=l[4]?n.removeAttribute(l):n.setAttribute(l,\"popover\"==l&&1==u?\"\":u))}}function F(n){return function(u){if(this.l){var t=this.l[u.type+n];if(null==u.t)u.t=c++;else if(u.t<t.u)return;return t(l.event?l.event(u):u)}}}function O(n,u,t,i,o,r,e,f,c,s){var a,h,p,v,y,_,m,b,S,C,M,$,P,A,H,L,T,j=u.type;if(void 0!==u.constructor)return null;128&t.__u&&(c=!!(32&t.__u),r=[f=u.__e=t.__e]),(a=l.__b)&&a(u);n:if(\"function\"==typeof j)try{if(b=u.props,S=\"prototype\"in j&&j.prototype.render,C=(a=j.contextType)&&i[a.__c],M=a?C?C.props.value:a.__:i,t.__c?m=(h=u.__c=t.__c).__=h.__E:(S?u.__c=h=new j(b,M):(u.__c=h=new x(b,M),h.constructor=j,h.render=E),C&&C.sub(h),h.state||(h.state={}),h.__n=i,p=h.__d=!0,h.__h=[],h._sb=[]),S&&null==h.__s&&(h.__s=h.state),S&&null!=j.getDerivedStateFromProps&&(h.__s==h.state&&(h.__s=w({},h.__s)),w(h.__s,j.getDerivedStateFromProps(b,h.__s))),v=h.props,y=h.state,h.__v=u,p)S&&null==j.getDerivedStateFromProps&&null!=h.componentWillMount&&h.componentWillMount(),S&&null!=h.componentDidMount&&h.__h.push(h.componentDidMount);else{if(S&&null==j.getDerivedStateFromProps&&b!==v&&null!=h.componentWillReceiveProps&&h.componentWillReceiveProps(b,M),u.__v==t.__v||!h.__e&&null!=h.shouldComponentUpdate&&!1===h.shouldComponentUpdate(b,h.__s,M)){for(u.__v!=t.__v&&(h.props=b,h.state=h.__s,h.__d=!1),u.__e=t.__e,u.__k=t.__k,u.__k.some(function(n){n&&(n.__=u)}),$=0;$<h._sb.length;$++)h.__h.push(h._sb[$]);h._sb=[],h.__h.length&&e.push(h);break n}null!=h.componentWillUpdate&&h.componentWillUpdate(b,h.__s,M),S&&null!=h.componentDidUpdate&&h.__h.push(function(){h.componentDidUpdate(v,y,_)})}if(h.context=M,h.props=b,h.__P=n,h.__e=!1,P=l.__r,A=0,S){for(h.state=h.__s,h.__d=!1,P&&P(u),a=h.render(h.props,h.state,h.context),H=0;H<h._sb.length;H++)h.__h.push(h._sb[H]);h._sb=[]}else do{h.__d=!1,P&&P(u),a=h.render(h.props,h.state,h.context),h.state=h.__s}while(h.__d&&++A<25);h.state=h.__s,null!=h.getChildContext&&(i=w(w({},i),h.getChildContext())),S&&!p&&null!=h.getSnapshotBeforeUpdate&&(_=h.getSnapshotBeforeUpdate(v,y)),L=a,null!=a&&a.type===k&&null==a.key&&(L=V(a.props.children)),f=I(n,d(L)?L:[L],u,t,i,o,r,e,f,c,s),h.base=u.__e,u.__u&=-161,h.__h.length&&e.push(h),m&&(h.__E=h.__=null)}catch(n){if(u.__v=null,c||null!=r)if(n.then){for(u.__u|=c?160:128;f&&8==f.nodeType&&f.nextSibling;)f=f.nextSibling;r[r.indexOf(f)]=null,u.__e=f}else{for(T=r.length;T--;)g(r[T]);z(u)}else u.__e=t.__e,u.__k=t.__k,n.then||z(u);l.__e(n,u,t)}else null==r&&u.__v==t.__v?(u.__k=t.__k,u.__e=t.__e):f=u.__e=q(t.__e,u,t,i,o,r,e,c,s);return(a=l.diffed)&&a(u),128&u.__u?void 0:f}function z(n){n&&n.__c&&(n.__c.__e=!0),n&&n.__k&&n.__k.forEach(z)}function N(n,u,t){for(var i=0;i<t.length;i++)B(t[i],t[++i],t[++i]);l.__c&&l.__c(u,n),n.some(function(u){try{n=u.__h,u.__h=[],n.some(function(n){n.call(u)})}catch(n){l.__e(n,u.__v)}})}function V(n){return\"object\"!=typeof n||null==n||n.__b&&n.__b>0?n:d(n)?n.map(V):w({},n)}function q(u,t,i,o,r,e,f,c,s){var a,h,v,y,w,_,m,b=i.props||p,k=t.props,x=t.type;if(\"svg\"==x?r=\"http://www.w3.org/2000/svg\":\"math\"==x?r=\"http://www.w3.org/1998/Math/MathML\":r||(r=\"http://www.w3.org/1999/xhtml\"),null!=e)for(a=0;a<e.length;a++)if((w=e[a])&&\"setAttribute\"in w==!!x&&(x?w.localName==x:3==w.nodeType)){u=w,e[a]=null;break}if(null==u){if(null==x)return document.createTextNode(k);u=document.createElementNS(r,x,k.is&&k),c&&(l.__m&&l.__m(t,e),c=!1),e=null}if(null==x)b===k||c&&u.data==k||(u.data=k);else{if(e=e&&n.call(u.childNodes),!c&&null!=e)for(b={},a=0;a<u.attributes.length;a++)b[(w=u.attributes[a]).name]=w.value;for(a in b)if(w=b[a],\"children\"==a);else if(\"dangerouslySetInnerHTML\"==a)v=w;else if(!(a in k)){if(\"value\"==a&&\"defaultValue\"in k||\"checked\"==a&&\"defaultChecked\"in k)continue;j(u,a,null,w,r)}for(a in k)w=k[a],\"children\"==a?y=w:\"dangerouslySetInnerHTML\"==a?h=w:\"value\"==a?_=w:\"checked\"==a?m=w:c&&\"function\"!=typeof w||b[a]===w||j(u,a,w,b[a],r);if(h)c||v&&(h.__html==v.__html||h.__html==u.innerHTML)||(u.innerHTML=h.__html),t.__k=[];else if(v&&(u.innerHTML=\"\"),I(\"template\"==t.type?u.content:u,d(y)?y:[y],t,i,o,\"foreignObject\"==x?\"http://www.w3.org/1999/xhtml\":r,e,f,e?e[0]:i.__k&&S(i,0),c,s),null!=e)for(a=e.length;a--;)g(e[a]);c||(a=\"value\",\"progress\"==x&&null==_?u.removeAttribute(\"value\"):null!=_&&(_!==u[a]||\"progress\"==x&&!_||\"option\"==x&&_!=b[a])&&j(u,a,_,b[a],r),a=\"checked\",null!=m&&m!=u[a]&&j(u,a,m,b[a],r))}return u}function B(n,u,t){try{if(\"function\"==typeof n){var i=\"function\"==typeof n.__u;i&&n.__u(),i&&null==u||(n.__u=n(u))}else n.current=u}catch(n){l.__e(n,t)}}function D(n,u,t){var i,o;if(l.unmount&&l.unmount(n),(i=n.ref)&&(i.current&&i.current!=n.__e||B(i,null,u)),null!=(i=n.__c)){if(i.componentWillUnmount)try{i.componentWillUnmount()}catch(n){l.__e(n,u)}i.base=i.__P=null}if(i=n.__k)for(o=0;o<i.length;o++)i[o]&&D(i[o],u,t||\"function\"!=typeof n.type);t||g(n.__e),n.__c=n.__=n.__e=void 0}function E(n,l,u){return this.constructor(n,u)}function G(u,t,i){var o,r,e,f;t==document&&(t=document.documentElement),l.__&&l.__(u,t),r=(o=\"function\"==typeof i)?null:i&&i.__k||t.__k,e=[],f=[],O(t,u=(!o&&i||t).__k=_(k,null,[u]),r||p,p,t.namespaceURI,!o&&i?[i]:r?null:t.firstChild?n.call(t.childNodes):null,e,!o&&i?i:r?r.__e:t.firstChild,o,f),N(e,u,f)}function J(n,l){G(n,l,J)}function K(l,u,t){var i,o,r,e,f=w({},l.props);for(r in l.type&&l.type.defaultProps&&(e=l.type.defaultProps),u)\"key\"==r?i=u[r]:\"ref\"==r?o=u[r]:f[r]=void 0===u[r]&&null!=e?e[r]:u[r];return arguments.length>2&&(f.children=arguments.length>3?n.call(arguments,2):t),m(l.type,f,i||l.key,o||l.ref,null)}function Q(n){function l(n){var u,t;return this.getChildContext||(u=new Set,(t={})[l.__c]=this,this.getChildContext=function(){return t},this.componentWillUnmount=function(){u=null},this.shouldComponentUpdate=function(n){this.props.value!=n.value&&u.forEach(function(n){n.__e=!0,M(n)})},this.sub=function(n){u.add(n);var l=n.componentWillUnmount;n.componentWillUnmount=function(){u&&u.delete(n),l&&l.call(n)}}),n.children}return l.__c=\"__cC\"+h++,l.__=n,l.Provider=l.__l=(l.Consumer=function(n,l){return n.children(l)}).contextType=l,l}n=v.slice,l={__e:function(n,l,u,t){for(var i,o,r;l=l.__;)if((i=l.__c)&&!i.__)try{if((o=i.constructor)&&null!=o.getDerivedStateFromError&&(i.setState(o.getDerivedStateFromError(n)),r=i.__d),null!=i.componentDidCatch&&(i.componentDidCatch(n,t||{}),r=i.__d),r)return i.__E=i}catch(l){n=l}throw n}},u=0,t=function(n){return null!=n&&void 0===n.constructor},x.prototype.setState=function(n,l){var u;u=null!=this.__s&&this.__s!=this.state?this.__s:this.__s=w({},this.state),\"function\"==typeof n&&(n=n(w({},u),this.props)),n&&w(u,n),null!=n&&this.__v&&(l&&this._sb.push(l),M(this))},x.prototype.forceUpdate=function(n){this.__v&&(this.__e=!0,n&&this.__h.push(n),M(this))},x.prototype.render=k,i=[],r=\"function\"==typeof Promise?Promise.prototype.then.bind(Promise.resolve()):setTimeout,e=function(n,l){return n.__v.__b-l.__v.__b},$.__r=0,f=/(PointerCapture)$|Capture$/i,c=0,s=F(!1),a=F(!0),h=0;export{x as Component,k as Fragment,K as cloneElement,Q as createContext,_ as createElement,b as createRef,_ as h,J as hydrate,t as isValidElement,l as options,G as render,H as toChildArray};\n//# sourceMappingURL=preact.module.js.map\n","import{options as r,Fragment as e}from\"preact\";export{Fragment}from\"preact\";var t=/[\"&<]/;function n(r){if(0===r.length||!1===t.test(r))return r;for(var e=0,n=0,o=\"\",f=\"\";n<r.length;n++){switch(r.charCodeAt(n)){case 34:f=\""\";break;case 38:f=\"&\";break;case 60:f=\"<\";break;default:continue}n!==e&&(o+=r.slice(e,n)),o+=f,e=n+1}return n!==e&&(o+=r.slice(e,n)),o}var o=/acit|ex(?:s|g|n|p|$)|rph|grid|ows|mnc|ntw|ine[ch]|zoo|^ord|itera/i,f=0,i=Array.isArray;function u(e,t,n,o,i,u){t||(t={});var a,c,p=t;if(\"ref\"in p)for(c in p={},t)\"ref\"==c?a=t[c]:p[c]=t[c];var l={type:e,props:p,key:n,ref:a,__k:null,__:null,__b:0,__e:null,__c:null,constructor:void 0,__v:--f,__i:-1,__u:0,__source:i,__self:u};if(\"function\"==typeof e&&(a=e.defaultProps))for(c in a)void 0===p[c]&&(p[c]=a[c]);return r.vnode&&r.vnode(l),l}function a(r){var t=u(e,{tpl:r,exprs:[].slice.call(arguments,1)});return t.key=t.__v,t}var c={},p=/[A-Z]/g;function l(e,t){if(r.attr){var f=r.attr(e,t);if(\"string\"==typeof f)return f}if(t=function(r){return null!==r&&\"object\"==typeof r&&\"function\"==typeof r.valueOf?r.valueOf():r}(t),\"ref\"===e||\"key\"===e)return\"\";if(\"style\"===e&&\"object\"==typeof t){var i=\"\";for(var u in t){var a=t[u];if(null!=a&&\"\"!==a){var l=\"-\"==u[0]?u:c[u]||(c[u]=u.replace(p,\"-$&\").toLowerCase()),s=\";\";\"number\"!=typeof a||l.startsWith(\"--\")||o.test(l)||(s=\"px;\"),i=i+l+\":\"+a+s}}return e+'=\"'+n(i)+'\"'}return null==t||!1===t||\"function\"==typeof t||\"object\"==typeof t?\"\":!0===t?e:e+'=\"'+n(\"\"+t)+'\"'}function s(r){if(null==r||\"boolean\"==typeof r||\"function\"==typeof r)return null;if(\"object\"==typeof r){if(void 0===r.constructor)return r;if(i(r)){for(var e=0;e<r.length;e++)r[e]=s(r[e]);return r}}return n(\"\"+r)}export{u as jsx,l as jsxAttr,u as jsxDEV,s as jsxEscape,a as jsxTemplate,u as jsxs};\n//# sourceMappingURL=jsxRuntime.module.js.map\n","import{options as n}from\"preact\";var t,r,u,i,o=0,f=[],c=n,e=c.__b,a=c.__r,v=c.diffed,l=c.__c,m=c.unmount,s=c.__;function p(n,t){c.__h&&c.__h(r,n,o||t),o=0;var u=r.__H||(r.__H={__:[],__h:[]});return n>=u.__.length&&u.__.push({}),u.__[n]}function d(n){return o=1,h(D,n)}function h(n,u,i){var o=p(t++,2);if(o.t=n,!o.__c&&(o.__=[i?i(u):D(void 0,u),function(n){var t=o.__N?o.__N[0]:o.__[0],r=o.t(t,n);t!==r&&(o.__N=[r,o.__[1]],o.__c.setState({}))}],o.__c=r,!r.__f)){var f=function(n,t,r){if(!o.__c.__H)return!0;var u=o.__c.__H.__.filter(function(n){return!!n.__c});if(u.every(function(n){return!n.__N}))return!c||c.call(this,n,t,r);var i=o.__c.props!==n;return u.forEach(function(n){if(n.__N){var t=n.__[0];n.__=n.__N,n.__N=void 0,t!==n.__[0]&&(i=!0)}}),c&&c.call(this,n,t,r)||i};r.__f=!0;var c=r.shouldComponentUpdate,e=r.componentWillUpdate;r.componentWillUpdate=function(n,t,r){if(this.__e){var u=c;c=void 0,f(n,t,r),c=u}e&&e.call(this,n,t,r)},r.shouldComponentUpdate=f}return o.__N||o.__}function y(n,u){var i=p(t++,3);!c.__s&&C(i.__H,u)&&(i.__=n,i.u=u,r.__H.__h.push(i))}function _(n,u){var i=p(t++,4);!c.__s&&C(i.__H,u)&&(i.__=n,i.u=u,r.__h.push(i))}function A(n){return o=5,T(function(){return{current:n}},[])}function F(n,t,r){o=6,_(function(){if(\"function\"==typeof n){var r=n(t());return function(){n(null),r&&\"function\"==typeof r&&r()}}if(n)return n.current=t(),function(){return n.current=null}},null==r?r:r.concat(n))}function T(n,r){var u=p(t++,7);return C(u.__H,r)&&(u.__=n(),u.__H=r,u.__h=n),u.__}function q(n,t){return o=8,T(function(){return n},t)}function x(n){var u=r.context[n.__c],i=p(t++,9);return i.c=n,u?(null==i.__&&(i.__=!0,u.sub(r)),u.props.value):n.__}function P(n,t){c.useDebugValue&&c.useDebugValue(t?t(n):n)}function b(n){var u=p(t++,10),i=d();return u.__=n,r.componentDidCatch||(r.componentDidCatch=function(n,t){u.__&&u.__(n,t),i[1](n)}),[i[0],function(){i[1](void 0)}]}function g(){var n=p(t++,11);if(!n.__){for(var u=r.__v;null!==u&&!u.__m&&null!==u.__;)u=u.__;var i=u.__m||(u.__m=[0,0]);n.__=\"P\"+i[0]+\"-\"+i[1]++}return n.__}function j(){for(var n;n=f.shift();)if(n.__P&&n.__H)try{n.__H.__h.forEach(z),n.__H.__h.forEach(B),n.__H.__h=[]}catch(t){n.__H.__h=[],c.__e(t,n.__v)}}c.__b=function(n){r=null,e&&e(n)},c.__=function(n,t){n&&t.__k&&t.__k.__m&&(n.__m=t.__k.__m),s&&s(n,t)},c.__r=function(n){a&&a(n),t=0;var i=(r=n.__c).__H;i&&(u===r?(i.__h=[],r.__h=[],i.__.forEach(function(n){n.__N&&(n.__=n.__N),n.u=n.__N=void 0})):(i.__h.forEach(z),i.__h.forEach(B),i.__h=[],t=0)),u=r},c.diffed=function(n){v&&v(n);var t=n.__c;t&&t.__H&&(t.__H.__h.length&&(1!==f.push(t)&&i===c.requestAnimationFrame||((i=c.requestAnimationFrame)||w)(j)),t.__H.__.forEach(function(n){n.u&&(n.__H=n.u),n.u=void 0})),u=r=null},c.__c=function(n,t){t.some(function(n){try{n.__h.forEach(z),n.__h=n.__h.filter(function(n){return!n.__||B(n)})}catch(r){t.some(function(n){n.__h&&(n.__h=[])}),t=[],c.__e(r,n.__v)}}),l&&l(n,t)},c.unmount=function(n){m&&m(n);var t,r=n.__c;r&&r.__H&&(r.__H.__.forEach(function(n){try{z(n)}catch(n){t=n}}),r.__H=void 0,t&&c.__e(t,r.__v))};var k=\"function\"==typeof requestAnimationFrame;function w(n){var t,r=function(){clearTimeout(u),k&&cancelAnimationFrame(t),setTimeout(n)},u=setTimeout(r,35);k&&(t=requestAnimationFrame(r))}function z(n){var t=r,u=n.__c;\"function\"==typeof u&&(n.__c=void 0,u()),r=t}function B(n){var t=r;n.__c=n.__(),r=t}function C(n,t){return!n||n.length!==t.length||t.some(function(t,r){return t!==n[r]})}function D(n,t){return\"function\"==typeof t?t(n):t}export{q as useCallback,x as useContext,P as useDebugValue,y as useEffect,b as useErrorBoundary,g as useId,F as useImperativeHandle,_ as useLayoutEffect,T as useMemo,h as useReducer,A as useRef,d as useState};\n//# sourceMappingURL=hooks.module.js.map\n","// Widget dimensions\r\nexport const BUTTON_SIZE = 60;\r\nexport const WINDOW_WIDTH = 350;\r\nexport const WINDOW_HEIGHT = 500;\r\n\r\n// Positioning\r\nexport const POSITION_BOTTOM = 20;\r\nexport const POSITION_RIGHT = 20;\r\n\r\n// Colors — initial values match the backend ProjectSettings default.\r\n// applyTheme() in index.tsx overrides primary/primaryHover/messageUser\r\n// with the remote config value before the widget renders.\r\nexport const COLORS = {\r\n primary: \"#6C03C1\",\r\n primaryHover: \"#5b02a4\",\r\n background: \"#ffffff\",\r\n text: \"hsla(215, 100%, 5%, 1)\",\r\n textSecondary: \"hsla(215, 100%, 5%, 1)\",\r\n border: \"#e5e7eb\",\r\n messageBot: \"\",\r\n messageUser: \"#6C03C1\",\r\n messageUserText: \"#ffffff\",\r\n};\r\n\r\nconst normalizeHexColor = (hex: string): string => {\r\n const trimmed = hex.trim();\r\n if (!/^#([0-9a-fA-F]{3}|[0-9a-fA-F]{6})$/.test(trimmed)) {\r\n return \"hsla(215, 100%, 5%, 1)\";\r\n }\r\n if (trimmed.length === 4) {\r\n const r = trimmed[1];\r\n const g = trimmed[2];\r\n const b = trimmed[3];\r\n return `#${r}${r}${g}${g}${b}${b}`;\r\n }\r\n return trimmed;\r\n};\r\n\r\nexport const getContrastIconFilter = (backgroundHex: string): string => {\r\n const hex = normalizeHexColor(backgroundHex).slice(1);\r\n const r = parseInt(hex.slice(0, 2), 16);\r\n const g = parseInt(hex.slice(2, 4), 16);\r\n const b = parseInt(hex.slice(4, 6), 16);\r\n\r\n // Relative luminance approximation for quick contrast choice.\r\n const luminance = (0.299 * r + 0.587 * g + 0.114 * b) / 255;\r\n return luminance > 0.62 ? \"brightness(0) invert(0)\" : \"brightness(0) invert(1)\";\r\n};\r\n\r\n// Border radius\r\nexport const BORDER_RADIUS = {\r\n button: '50%',\r\n window: '17px',\r\n message: '10px'\r\n};\r\n\r\n// Shadows\r\nexport const SHADOW = \"0 0 15px hsla(215, 100%, 5%, 0.15)\";\r\n\r\n// Transitions\r\nexport const TRANSITIONS = {\r\n fast: '150ms ease-in-out',\r\n medium: '250ms ease-in-out'\r\n};\r\n","import * as React from \"react\";\nfunction ArrowPathIcon({\n title,\n titleId,\n ...props\n}, svgRef) {\n return /*#__PURE__*/React.createElement(\"svg\", Object.assign({\n xmlns: \"http://www.w3.org/2000/svg\",\n fill: \"none\",\n viewBox: \"0 0 24 24\",\n strokeWidth: 1.5,\n stroke: \"currentColor\",\n \"aria-hidden\": \"true\",\n \"data-slot\": \"icon\",\n ref: svgRef,\n \"aria-labelledby\": titleId\n }, props), title ? /*#__PURE__*/React.createElement(\"title\", {\n id: titleId\n }, title) : null, /*#__PURE__*/React.createElement(\"path\", {\n strokeLinecap: \"round\",\n strokeLinejoin: \"round\",\n d: \"M16.023 9.348h4.992v-.001M2.985 19.644v-4.992m0 0h4.992m-4.993 0 3.181 3.183a8.25 8.25 0 0 0 13.803-3.7M4.031 9.865a8.25 8.25 0 0 1 13.803-3.7l3.181 3.182m0-4.991v4.99\"\n }));\n}\nconst ForwardRef = /*#__PURE__*/ React.forwardRef(ArrowPathIcon);\nexport default ForwardRef;","import * as React from \"react\";\nfunction CommandLineIcon({\n title,\n titleId,\n ...props\n}, svgRef) {\n return /*#__PURE__*/React.createElement(\"svg\", Object.assign({\n xmlns: \"http://www.w3.org/2000/svg\",\n fill: \"none\",\n viewBox: \"0 0 24 24\",\n strokeWidth: 1.5,\n stroke: \"currentColor\",\n \"aria-hidden\": \"true\",\n \"data-slot\": \"icon\",\n ref: svgRef,\n \"aria-labelledby\": titleId\n }, props), title ? /*#__PURE__*/React.createElement(\"title\", {\n id: titleId\n }, title) : null, /*#__PURE__*/React.createElement(\"path\", {\n strokeLinecap: \"round\",\n strokeLinejoin: \"round\",\n d: \"m6.75 7.5 3 2.25-3 2.25m4.5 0h3m-9 8.25h13.5A2.25 2.25 0 0 0 21 18V6a2.25 2.25 0 0 0-2.25-2.25H5.25A2.25 2.25 0 0 0 3 6v12a2.25 2.25 0 0 0 2.25 2.25Z\"\n }));\n}\nconst ForwardRef = /*#__PURE__*/ React.forwardRef(CommandLineIcon);\nexport default ForwardRef;","import * as React from \"react\";\nfunction CursorArrowRaysIcon({\n title,\n titleId,\n ...props\n}, svgRef) {\n return /*#__PURE__*/React.createElement(\"svg\", Object.assign({\n xmlns: \"http://www.w3.org/2000/svg\",\n fill: \"none\",\n viewBox: \"0 0 24 24\",\n strokeWidth: 1.5,\n stroke: \"currentColor\",\n \"aria-hidden\": \"true\",\n \"data-slot\": \"icon\",\n ref: svgRef,\n \"aria-labelledby\": titleId\n }, props), title ? /*#__PURE__*/React.createElement(\"title\", {\n id: titleId\n }, title) : null, /*#__PURE__*/React.createElement(\"path\", {\n strokeLinecap: \"round\",\n strokeLinejoin: \"round\",\n d: \"M15.042 21.672 13.684 16.6m0 0-2.51 2.225.569-9.47 5.227 7.917-3.286-.672ZM12 2.25V4.5m5.834.166-1.591 1.591M20.25 10.5H18M7.757 14.743l-1.59 1.59M6 10.5H3.75m4.007-4.243-1.59-1.59\"\n }));\n}\nconst ForwardRef = /*#__PURE__*/ React.forwardRef(CursorArrowRaysIcon);\nexport default ForwardRef;","import * as React from \"react\";\nfunction FaceSmileIcon({\n title,\n titleId,\n ...props\n}, svgRef) {\n return /*#__PURE__*/React.createElement(\"svg\", Object.assign({\n xmlns: \"http://www.w3.org/2000/svg\",\n fill: \"none\",\n viewBox: \"0 0 24 24\",\n strokeWidth: 1.5,\n stroke: \"currentColor\",\n \"aria-hidden\": \"true\",\n \"data-slot\": \"icon\",\n ref: svgRef,\n \"aria-labelledby\": titleId\n }, props), title ? /*#__PURE__*/React.createElement(\"title\", {\n id: titleId\n }, title) : null, /*#__PURE__*/React.createElement(\"path\", {\n strokeLinecap: \"round\",\n strokeLinejoin: \"round\",\n d: \"M15.182 15.182a4.5 4.5 0 0 1-6.364 0M21 12a9 9 0 1 1-18 0 9 9 0 0 1 18 0ZM9.75 9.75c0 .414-.168.75-.375.75S9 10.164 9 9.75 9.168 9 9.375 9s.375.336.375.75Zm-.375 0h.008v.015h-.008V9.75Zm5.625 0c0 .414-.168.75-.375.75s-.375-.336-.375-.75.168-.75.375-.75.375.336.375.75Zm-.375 0h.008v.015h-.008V9.75Z\"\n }));\n}\nconst ForwardRef = /*#__PURE__*/ React.forwardRef(FaceSmileIcon);\nexport default ForwardRef;","import * as React from \"react\";\nfunction HandRaisedIcon({\n title,\n titleId,\n ...props\n}, svgRef) {\n return /*#__PURE__*/React.createElement(\"svg\", Object.assign({\n xmlns: \"http://www.w3.org/2000/svg\",\n fill: \"none\",\n viewBox: \"0 0 24 24\",\n strokeWidth: 1.5,\n stroke: \"currentColor\",\n \"aria-hidden\": \"true\",\n \"data-slot\": \"icon\",\n ref: svgRef,\n \"aria-labelledby\": titleId\n }, props), title ? /*#__PURE__*/React.createElement(\"title\", {\n id: titleId\n }, title) : null, /*#__PURE__*/React.createElement(\"path\", {\n strokeLinecap: \"round\",\n strokeLinejoin: \"round\",\n d: \"M10.05 4.575a1.575 1.575 0 1 0-3.15 0v3m3.15-3v-1.5a1.575 1.575 0 0 1 3.15 0v1.5m-3.15 0 .075 5.925m3.075.75V4.575m0 0a1.575 1.575 0 0 1 3.15 0V15M6.9 7.575a1.575 1.575 0 1 0-3.15 0v8.175a6.75 6.75 0 0 0 6.75 6.75h2.018a5.25 5.25 0 0 0 3.712-1.538l1.732-1.732a5.25 5.25 0 0 0 1.538-3.712l.003-2.024a.668.668 0 0 1 .198-.471 1.575 1.575 0 1 0-2.228-2.228 3.818 3.818 0 0 0-1.12 2.687M6.9 7.575V12m6.27 4.318A4.49 4.49 0 0 1 16.35 15m.002 0h-.002\"\n }));\n}\nconst ForwardRef = /*#__PURE__*/ React.forwardRef(HandRaisedIcon);\nexport default ForwardRef;","import * as React from \"react\";\nfunction MapIcon({\n title,\n titleId,\n ...props\n}, svgRef) {\n return /*#__PURE__*/React.createElement(\"svg\", Object.assign({\n xmlns: \"http://www.w3.org/2000/svg\",\n fill: \"none\",\n viewBox: \"0 0 24 24\",\n strokeWidth: 1.5,\n stroke: \"currentColor\",\n \"aria-hidden\": \"true\",\n \"data-slot\": \"icon\",\n ref: svgRef,\n \"aria-labelledby\": titleId\n }, props), title ? /*#__PURE__*/React.createElement(\"title\", {\n id: titleId\n }, title) : null, /*#__PURE__*/React.createElement(\"path\", {\n strokeLinecap: \"round\",\n strokeLinejoin: \"round\",\n d: \"M9 6.75V15m6-6v8.25m.503 3.498 4.875-2.437c.381-.19.622-.58.622-1.006V4.82c0-.836-.88-1.38-1.628-1.006l-3.869 1.934c-.317.159-.69.159-1.006 0L9.503 3.252a1.125 1.125 0 0 0-1.006 0L3.622 5.689C3.24 5.88 3 6.27 3 6.695V19.18c0 .836.88 1.38 1.628 1.006l3.869-1.934c.317-.159.69-.159 1.006 0l4.994 2.497c.317.158.69.158 1.006 0Z\"\n }));\n}\nconst ForwardRef = /*#__PURE__*/ React.forwardRef(MapIcon);\nexport default ForwardRef;","import * as React from \"react\";\nfunction MicrophoneIcon({\n title,\n titleId,\n ...props\n}, svgRef) {\n return /*#__PURE__*/React.createElement(\"svg\", Object.assign({\n xmlns: \"http://www.w3.org/2000/svg\",\n fill: \"none\",\n viewBox: \"0 0 24 24\",\n strokeWidth: 1.5,\n stroke: \"currentColor\",\n \"aria-hidden\": \"true\",\n \"data-slot\": \"icon\",\n ref: svgRef,\n \"aria-labelledby\": titleId\n }, props), title ? /*#__PURE__*/React.createElement(\"title\", {\n id: titleId\n }, title) : null, /*#__PURE__*/React.createElement(\"path\", {\n strokeLinecap: \"round\",\n strokeLinejoin: \"round\",\n d: \"M12 18.75a6 6 0 0 0 6-6v-1.5m-6 7.5a6 6 0 0 1-6-6v-1.5m6 7.5v3.75m-3.75 0h7.5M12 15.75a3 3 0 0 1-3-3V4.5a3 3 0 1 1 6 0v8.25a3 3 0 0 1-3 3Z\"\n }));\n}\nconst ForwardRef = /*#__PURE__*/ React.forwardRef(MicrophoneIcon);\nexport default ForwardRef;","import * as React from \"react\";\nfunction QueueListIcon({\n title,\n titleId,\n ...props\n}, svgRef) {\n return /*#__PURE__*/React.createElement(\"svg\", Object.assign({\n xmlns: \"http://www.w3.org/2000/svg\",\n fill: \"none\",\n viewBox: \"0 0 24 24\",\n strokeWidth: 1.5,\n stroke: \"currentColor\",\n \"aria-hidden\": \"true\",\n \"data-slot\": \"icon\",\n ref: svgRef,\n \"aria-labelledby\": titleId\n }, props), title ? /*#__PURE__*/React.createElement(\"title\", {\n id: titleId\n }, title) : null, /*#__PURE__*/React.createElement(\"path\", {\n strokeLinecap: \"round\",\n strokeLinejoin: \"round\",\n d: \"M3.75 12h16.5m-16.5 3.75h16.5M3.75 19.5h16.5M5.625 4.5h12.75a1.875 1.875 0 0 1 0 3.75H5.625a1.875 1.875 0 0 1 0-3.75Z\"\n }));\n}\nconst ForwardRef = /*#__PURE__*/ React.forwardRef(QueueListIcon);\nexport default ForwardRef;","import * as React from \"react\";\nfunction StopIcon({\n title,\n titleId,\n ...props\n}, svgRef) {\n return /*#__PURE__*/React.createElement(\"svg\", Object.assign({\n xmlns: \"http://www.w3.org/2000/svg\",\n fill: \"none\",\n viewBox: \"0 0 24 24\",\n strokeWidth: 1.5,\n stroke: \"currentColor\",\n \"aria-hidden\": \"true\",\n \"data-slot\": \"icon\",\n ref: svgRef,\n \"aria-labelledby\": titleId\n }, props), title ? /*#__PURE__*/React.createElement(\"title\", {\n id: titleId\n }, title) : null, /*#__PURE__*/React.createElement(\"path\", {\n strokeLinecap: \"round\",\n strokeLinejoin: \"round\",\n d: \"M5.25 7.5A2.25 2.25 0 0 1 7.5 5.25h9a2.25 2.25 0 0 1 2.25 2.25v9a2.25 2.25 0 0 1-2.25 2.25h-9a2.25 2.25 0 0 1-2.25-2.25v-9Z\"\n }));\n}\nconst ForwardRef = /*#__PURE__*/ React.forwardRef(StopIcon);\nexport default ForwardRef;","import * as React from \"react\";\nfunction XMarkIcon({\n title,\n titleId,\n ...props\n}, svgRef) {\n return /*#__PURE__*/React.createElement(\"svg\", Object.assign({\n xmlns: \"http://www.w3.org/2000/svg\",\n fill: \"none\",\n viewBox: \"0 0 24 24\",\n strokeWidth: 1.5,\n stroke: \"currentColor\",\n \"aria-hidden\": \"true\",\n \"data-slot\": \"icon\",\n ref: svgRef,\n \"aria-labelledby\": titleId\n }, props), title ? /*#__PURE__*/React.createElement(\"title\", {\n id: titleId\n }, title) : null, /*#__PURE__*/React.createElement(\"path\", {\n strokeLinecap: \"round\",\n strokeLinejoin: \"round\",\n d: \"M6 18 18 6M6 6l12 12\"\n }));\n}\nconst ForwardRef = /*#__PURE__*/ React.forwardRef(XMarkIcon);\nexport default ForwardRef;","import {\r\n BUTTON_SIZE,\r\n POSITION_BOTTOM,\r\n POSITION_RIGHT,\r\n COLORS,\r\n SHADOW,\r\n TRANSITIONS,\r\n BORDER_RADIUS,\r\n} from \"../styles/constants\";\r\nimport { MicrophoneIcon, StopIcon, XMarkIcon } from \"@heroicons/react/24/outline\";\r\n\r\ninterface ChatButtonProps {\r\n onMicClick: () => void;\r\n onCancelRecording: () => void;\r\n onStopTask: () => void;\r\n isRecording: boolean;\r\n isBusy: boolean;\r\n accessibilityMode?: boolean;\r\n showBubble: boolean;\r\n onBubbleClick: () => void;\r\n previewMessage: string | null;\r\n onPreviewClick: () => void;\r\n onPreviewClose: () => void;\r\n}\r\n\r\nexport const ChatButton = ({\r\n onMicClick,\r\n onCancelRecording,\r\n onStopTask,\r\n isRecording,\r\n isBusy,\r\n accessibilityMode = false,\r\n showBubble,\r\n onBubbleClick,\r\n previewMessage,\r\n onPreviewClick,\r\n onPreviewClose,\r\n}: ChatButtonProps) => {\r\n const bgColor = COLORS.primary;\r\n const popupBoxShadow = accessibilityMode\r\n ? `inset 0 0 0 2px ${COLORS.primary}, ${SHADOW}`\r\n : SHADOW;\r\n\r\n const containerStyle: { [key: string]: string } = {\r\n position: \"fixed\",\r\n right: `${POSITION_RIGHT}px`,\r\n bottom: `${POSITION_BOTTOM}px`,\r\n display: \"flex\",\r\n flexDirection: \"column\",\r\n alignItems: \"flex-end\",\r\n gap: \"8px\",\r\n zIndex: \"9999\",\r\n fontFamily: \"\\\"Geist\\\", sans-serif\",\r\n };\r\n\r\n const controlsRowStyle: { [key: string]: string } = {\r\n display: \"flex\",\r\n alignItems: \"center\",\r\n gap: \"10px\",\r\n };\r\n\r\n const buttonStyle: { [key: string]: string } = {\r\n width: `${BUTTON_SIZE}px`,\r\n height: `${BUTTON_SIZE}px`,\r\n minWidth: `${BUTTON_SIZE}px`,\r\n minHeight: `${BUTTON_SIZE}px`,\r\n borderRadius: BORDER_RADIUS.button,\r\n backgroundColor: bgColor,\r\n color: \"#ffffff\",\r\n border: \"none\",\r\n cursor: \"pointer\",\r\n display: \"flex\",\r\n alignItems: \"center\",\r\n justifyContent: \"center\",\r\n transition: `background-color ${TRANSITIONS.fast}, transform ${TRANSITIONS.fast}`,\r\n position: \"relative\",\r\n flexShrink: \"0\",\r\n };\r\n\r\n const iconStyle: { [key: string]: string } = {\r\n width: \"24px\",\r\n height: \"24px\",\r\n display: \"block\",\r\n };\r\n\r\n const showStopButton = isBusy && !isRecording;\r\n const MainIcon = showStopButton ? StopIcon : isRecording ? XMarkIcon : MicrophoneIcon;\r\n\r\n const handleClick = () => {\r\n if (showStopButton) {\r\n onStopTask();\r\n } else if (isRecording) {\r\n onCancelRecording();\r\n } else {\r\n onMicClick();\r\n }\r\n };\r\n\r\n // Shared close button style\r\n const closeBtnStyle: { [key: string]: string } = {\r\n position: \"absolute\",\r\n top: \"6px\",\r\n right: \"6px\",\r\n width: \"20px\",\r\n height: \"20px\",\r\n borderRadius: \"50%\",\r\n border: \"none\",\r\n background: \"transparent\",\r\n cursor: \"pointer\",\r\n display: \"flex\",\r\n alignItems: \"center\",\r\n justifyContent: \"center\",\r\n fontSize: \"12px\",\r\n lineHeight: \"1\",\r\n color: COLORS.text,\r\n padding: \"0\",\r\n };\r\n\r\n const renderPopup = (\r\n content: preact.ComponentChildren,\r\n onClick: () => void,\r\n onClose: (() => void) | null,\r\n extraClass: string,\r\n scrollable: boolean,\r\n ) => (\r\n <div\r\n className={`bulut-popup ${extraClass}`}\r\n style={{ cursor: \"pointer\" }}\r\n onClick={onClick}\r\n >\r\n {/* Close button */}\r\n {onClose && (\r\n <button\r\n type=\"button\"\r\n style={closeBtnStyle}\r\n onClick={(e) => {\r\n e.stopPropagation();\r\n onClose();\r\n }}\r\n aria-label=\"Kapat\"\r\n >\r\n <XMarkIcon aria-hidden=\"true\" width={14} height={14} strokeWidth={3} />\r\n </button>\r\n )}\r\n\r\n {/* Text content */}\r\n <div\r\n style={{\r\n paddingRight: onClose ? \"22px\" : \"0\",\r\n wordBreak: \"break-word\",\r\n ...(scrollable ? { maxHeight: \"96px\", overflowY: \"auto\" } : {}),\r\n }}\r\n >\r\n {content}\r\n </div>\r\n </div>\r\n );\r\n\r\n return (\r\n <>\r\n <style>{`\r\n @import url('https://fonts.googleapis.com/css2?family=Geist:wght@100..900&display=swap');\r\n\r\n .bulut-popup {\r\n background: #ffffff;\r\n color: ${COLORS.text};\r\n padding: 10px 14px;\r\n border-radius: 12px;\r\n font-size: 14px;\r\n line-height: 1.4;\r\n position: relative;\r\n overflow: visible;\r\n box-shadow: ${popupBoxShadow};\r\n }\r\n .bulut-popup-bubble {\r\n animation: bulut-bubbleIn 400ms ease-out;\r\n }\r\n .bulut-popup-preview {\r\n animation: bulut-popIn ${TRANSITIONS.medium};\r\n }\r\n\r\n /* Desktop: limit width, lean to right */\r\n .bulut-popup {\r\n max-width: 320px;\r\n }\r\n\r\n /* Mobile: full-width layout with popup filling remaining space */\r\n @media (max-width: 600px) {\r\n .bulut-fab-container:has(.bulut-popup) {\r\n left: 16px !important;\r\n right: 16px !important;\r\n }\r\n .bulut-fab-container:has(.bulut-popup) .bulut-controls-row {\r\n width: 100%;\r\n }\r\n .bulut-popup {\r\n flex: 1;\r\n min-width: 0;\r\n max-width: none;\r\n }\r\n }\r\n\r\n @keyframes bulut-popIn {\r\n from { opacity: 0; transform: translateX(10px); }\r\n to { opacity: 1; transform: translateX(0); }\r\n }\r\n @keyframes bulut-bubbleIn {\r\n 0% { opacity: 0; transform: translateX(10px) scale(0.95); }\r\n 60% { opacity: 1; transform: translateX(-4px) scale(1.02); }\r\n 100% { opacity: 1; transform: translateX(0) scale(1); }\r\n }\r\n @keyframes bulut-badgeIn {\r\n from { opacity: 0; transform: translateY(-4px); }\r\n to { opacity: 1; transform: translateY(0); }\r\n }\r\n `}</style>\r\n\r\n <div className=\"bulut-fab-container\" style={containerStyle}>\r\n <div className=\"bulut-controls-row\" style={controlsRowStyle}>\r\n {/* Welcome bubble */}\r\n {showBubble &&\r\n !isRecording &&\r\n !previewMessage &&\r\n renderPopup(\r\n \"Destek lazımsa hemen konuşmaya başlayabiliriz!\",\r\n onBubbleClick,\r\n null,\r\n \"bulut-popup-bubble\",\r\n false,\r\n )}\r\n\r\n {/* New-message preview */}\r\n {previewMessage &&\r\n renderPopup(\r\n previewMessage,\r\n onPreviewClick,\r\n onPreviewClose,\r\n \"bulut-popup-preview\",\r\n true,\r\n )}\r\n\r\n {/* Main mic / cancel button */}\r\n <button\r\n style={buttonStyle}\r\n onClick={handleClick}\r\n onMouseEnter={(e) => {\r\n Object.assign(e.currentTarget.style, {\r\n backgroundColor: COLORS.primaryHover,\r\n transform: \"scale(1.05)\",\r\n });\r\n }}\r\n onMouseLeave={(e) => {\r\n Object.assign(e.currentTarget.style, {\r\n transform: \"scale(1)\",\r\n });\r\n }}\r\n aria-label={showStopButton ? \"Görevi durdur\" : isRecording ? \"Kaydı iptal et\" : \"Konuşmaya başla\"}\r\n >\r\n <MainIcon\r\n aria-hidden=\"true\"\r\n style={iconStyle}\r\n strokeWidth={2.25}\r\n />\r\n </button>\r\n </div>\r\n </div>\r\n </>\r\n );\r\n};\r\n","export interface PageContext {\r\n links: string[];\r\n interactables: string[];\r\n summary: string;\r\n}\r\n\r\nexport interface CachedPageContextEntry {\r\n url: string;\r\n summary: string;\r\n links: string[];\r\n interactables: string[];\r\n capturedAt: number;\r\n version: number;\r\n}\r\n\r\nexport interface PageContextSummaryInput {\r\n url: string;\r\n title: string;\r\n lang: string;\r\n headings: string[];\r\n landmarks: string[];\r\n links: string[];\r\n interactables: string[];\r\n interactionSignals: string[];\r\n styleSelectors: string[];\r\n pageBlueprint: string[];\r\n textSnippets: string[];\r\n outerHtmlDigest: string;\r\n}\r\n\r\ninterface InteractableCandidate {\r\n line: string;\r\n score: number;\r\n order: number;\r\n}\r\n\r\ninterface PageSignalSnapshot {\r\n links: string[];\r\n interactables: string[];\r\n interactionSignals: string[];\r\n styleSelectors: string[];\r\n pageBlueprint: string[];\r\n}\r\n\r\nimport {\r\n MAX_LINKS,\r\n MAX_INTERACTABLES,\r\n MAX_HEADINGS,\r\n MAX_TEXT_SNIPPETS,\r\n MAX_CACHED_PAGES,\r\n MAX_PAGE_SCAN_ELEMENTS,\r\n MAX_EVENT_HINTS_PER_ELEMENT,\r\n MAX_BRANCH_SAMPLES,\r\n MAX_BRANCH_DEPTH,\r\n MAX_STYLESHEET_SELECTORS,\r\n MAX_STYLESHEET_RULES,\r\n} from \"./contextConfig\";\r\n\r\nexport const PAGE_CONTEXT_CACHE_VERSION = 3;\r\nexport const PAGE_CONTEXT_CACHE_KEY = \"auticbot_page_context_cache_v3\";\r\n\r\nconst NON_CONTENT_TAGS = new Set([\r\n \"script\",\r\n \"style\",\r\n \"noscript\",\r\n \"template\",\r\n \"link\",\r\n \"meta\",\r\n]);\r\n\r\nconst NATIVE_INTERACTIVE_TAGS = new Set([\r\n \"a\",\r\n \"button\",\r\n \"input\",\r\n \"textarea\",\r\n \"select\",\r\n \"summary\",\r\n \"details\",\r\n \"option\",\r\n]);\r\n\r\nconst INTERACTIVE_ROLES = new Set([\r\n \"button\",\r\n \"link\",\r\n \"tab\",\r\n \"menuitem\",\r\n \"option\",\r\n \"checkbox\",\r\n \"radio\",\r\n \"switch\",\r\n \"combobox\",\r\n \"textbox\",\r\n \"searchbox\",\r\n \"slider\",\r\n \"spinbutton\",\r\n \"treeitem\",\r\n]);\r\n\r\nconst TRACKED_DISPLAY_VALUES = new Set([\r\n \"block\",\r\n \"inline\",\r\n \"inline-block\",\r\n \"flex\",\r\n \"inline-flex\",\r\n \"grid\",\r\n \"inline-grid\",\r\n]);\r\n\r\nconst TRACKED_POSITION_VALUES = new Set([\r\n \"relative\",\r\n \"absolute\",\r\n \"fixed\",\r\n \"sticky\",\r\n]);\r\n\r\nconst EVENT_HINT_NAMES = [\r\n \"click\",\r\n \"dblclick\",\r\n \"mousedown\",\r\n \"mouseup\",\r\n \"pointerdown\",\r\n \"pointerup\",\r\n \"touchstart\",\r\n \"touchend\",\r\n \"keydown\",\r\n \"keyup\",\r\n \"keypress\",\r\n \"input\",\r\n \"change\",\r\n \"submit\",\r\n \"focus\",\r\n \"blur\",\r\n];\r\n\r\nconst ARIA_INTERACTION_ATTRS = [\r\n \"aria-controls\",\r\n \"aria-expanded\",\r\n \"aria-haspopup\",\r\n \"aria-pressed\",\r\n \"aria-selected\",\r\n];\r\n\r\nconst DATA_INTERACTION_PATTERN =\r\n /(action|click|press|toggle|target|trigger|nav|open|close|menu|modal|command|submit)/i;\r\n\r\nconst STYLESHEET_SELECTOR_PATTERN =\r\n /(:hover|:focus|:active|button|a\\b|input|textarea|select|\\[role=|\\[aria-|\\[data-|\\.btn|\\.link)/i;\r\n\r\nconst pageContextCache = new Map<string, CachedPageContextEntry>();\r\nlet cacheHydrated = false;\r\n\r\nconst normalizeWhitespace = (value: string): string =>\r\n value.replace(/\\s+/g, \" \").trim();\r\n\r\nconst canonicalUrl = (rawUrl: string): string => {\r\n try {\r\n return new URL(rawUrl, rawUrl).href;\r\n } catch {\r\n return rawUrl;\r\n }\r\n};\r\n\r\nconst isCacheEntry = (value: unknown): value is CachedPageContextEntry => {\r\n if (typeof value !== \"object\" || value === null) {\r\n return false;\r\n }\r\n\r\n const obj = value as Record<string, unknown>;\r\n return (\r\n typeof obj.url === \"string\" &&\r\n typeof obj.summary === \"string\" &&\r\n Array.isArray(obj.links) &&\r\n Array.isArray(obj.interactables) &&\r\n typeof obj.capturedAt === \"number\" &&\r\n typeof obj.version === \"number\"\r\n );\r\n};\r\n\r\nconst bumpCount = (map: Map<string, number>, key: string): void => {\r\n if (!key) {\r\n return;\r\n }\r\n map.set(key, (map.get(key) ?? 0) + 1);\r\n};\r\n\r\nconst formatTopCounts = (map: Map<string, number>, maxItems: number): string => {\r\n if (map.size === 0) {\r\n return \"none\";\r\n }\r\n\r\n return Array.from(map.entries())\r\n .sort((a, b) => b[1] - a[1] || a[0].localeCompare(b[0]))\r\n .slice(0, maxItems)\r\n .map(([name, count]) => `${name}*${count}`)\r\n .join(\", \");\r\n};\r\n\r\nconst parseTabIndex = (value: string | null): number | null => {\r\n if (value === null) {\r\n return null;\r\n }\r\n\r\n const parsed = Number.parseInt(value, 10);\r\n return Number.isNaN(parsed) ? null : parsed;\r\n};\r\n\r\nconst compactToken = (value: string): string => {\r\n const compact = value.replace(/\\s+/g, \"-\").replace(/[^a-zA-Z0-9_-]/g, \"\");\r\n return compact || \"\";\r\n};\r\n\r\nconst getElementDepth = (element: Element): number => {\r\n let depth = 0;\r\n let cursor: Element | null = element;\r\n while (cursor?.parentElement) {\r\n depth += 1;\r\n cursor = cursor.parentElement;\r\n if (cursor === document.body) {\r\n break;\r\n }\r\n }\r\n return depth;\r\n};\r\n\r\nconst getPrimaryRole = (element: Element): string => {\r\n const rawRole = normalizeWhitespace(element.getAttribute(\"role\") || \"\")\r\n .toLowerCase()\r\n .split(\" \")[0];\r\n return rawRole || \"\";\r\n};\r\n\r\nconst hydrateCacheFromStorage = (): void => {\r\n if (cacheHydrated || typeof sessionStorage === \"undefined\") {\r\n return;\r\n }\r\n\r\n cacheHydrated = true;\r\n\r\n try {\r\n const raw = sessionStorage.getItem(PAGE_CONTEXT_CACHE_KEY);\r\n if (!raw) {\r\n return;\r\n }\r\n\r\n const parsed = JSON.parse(raw);\r\n if (!Array.isArray(parsed)) {\r\n return;\r\n }\r\n\r\n for (const value of parsed) {\r\n if (!isCacheEntry(value)) {\r\n continue;\r\n }\r\n if (value.version !== PAGE_CONTEXT_CACHE_VERSION) {\r\n continue;\r\n }\r\n pageContextCache.set(value.url, value);\r\n }\r\n if (pageContextCache.size > 0) {\r\n console.info(\r\n `[Autic] context cache restored entries=${pageContextCache.size}`,\r\n );\r\n }\r\n } catch (error) {\r\n console.warn(\"[Autic] context cache restore failed\", error);\r\n }\r\n};\r\n\r\nconst persistCacheToStorage = (): void => {\r\n if (typeof sessionStorage === \"undefined\") {\r\n return;\r\n }\r\n\r\n try {\r\n const serialized = JSON.stringify(\r\n Array.from(pageContextCache.values()).sort(\r\n (a, b) => a.capturedAt - b.capturedAt,\r\n ),\r\n );\r\n sessionStorage.setItem(PAGE_CONTEXT_CACHE_KEY, serialized);\r\n } catch (error) {\r\n console.warn(\"[Autic] context cache persist failed\", error);\r\n }\r\n};\r\n\r\nconst pruneOldestCacheEntries = (): void => {\r\n if (pageContextCache.size <= MAX_CACHED_PAGES) {\r\n return;\r\n }\r\n\r\n const sorted = Array.from(pageContextCache.values()).sort(\r\n (a, b) => a.capturedAt - b.capturedAt,\r\n );\r\n const overflow = sorted.length - MAX_CACHED_PAGES;\r\n for (let i = 0; i < overflow; i += 1) {\r\n pageContextCache.delete(sorted[i].url);\r\n }\r\n};\r\n\r\nconst buildSummaryWithHistory = (\r\n current: CachedPageContextEntry,\r\n): string => {\r\n const recentPages = Array.from(pageContextCache.values())\r\n .filter((entry) => entry.url !== current.url)\r\n .sort((a, b) => b.capturedAt - a.capturedAt)\r\n .slice(0, 3);\r\n\r\n if (recentPages.length === 0) {\r\n return current.summary;\r\n }\r\n\r\n const historySection = [\r\n \"Recent Page Memory:\",\r\n ...recentPages.map((entry) => {\r\n const compactSummary = normalizeWhitespace(entry.summary);\r\n return `- ${entry.url} :: ${compactSummary}`;\r\n }),\r\n ].join(\"\\n\");\r\n\r\n return `${current.summary}\\n\\n${historySection}`;\r\n};\r\n\r\nconst isVisible = (element: Element): boolean => {\r\n if (element.getAttribute(\"aria-hidden\") === \"true\") {\r\n return false;\r\n }\r\n\r\n if (element instanceof HTMLElement && element.hidden) {\r\n return false;\r\n }\r\n\r\n const style = window.getComputedStyle(element);\r\n if (style.display === \"none\" || style.visibility === \"hidden\") {\r\n return false;\r\n }\r\n\r\n const rect = element.getBoundingClientRect();\r\n return rect.width > 0 && rect.height > 0;\r\n};\r\n\r\nconst toAbsoluteUrl = (href: string): string => {\r\n try {\r\n return new URL(href, window.location.href).href;\r\n } catch {\r\n return href;\r\n }\r\n};\r\n\r\nconst escapeCssValue = (value: string): string => {\r\n if (typeof CSS !== \"undefined\" && typeof CSS.escape === \"function\") {\r\n return CSS.escape(value);\r\n }\r\n\r\n return value.replace(/([ #;&,.+*~':\"!^$\\[\\]()=>|\\/@])/g, \"\\\\$1\");\r\n};\r\n\r\nconst buildSelectorSegment = (element: Element): string => {\r\n const tag = element.tagName.toLowerCase();\r\n\r\n if (element.id) {\r\n return `#${escapeCssValue(element.id)}`;\r\n }\r\n\r\n const attrCandidates: Array<[name: string, value: string | null]> = [\r\n [\"name\", element.getAttribute(\"name\")],\r\n [\"data-testid\", element.getAttribute(\"data-testid\")],\r\n [\"data-test-id\", element.getAttribute(\"data-test-id\")],\r\n [\"aria-label\", element.getAttribute(\"aria-label\")],\r\n [\"role\", element.getAttribute(\"role\")],\r\n [\"type\", element.getAttribute(\"type\")],\r\n ];\r\n\r\n for (const [attrName, attrValue] of attrCandidates) {\r\n if (attrValue) {\r\n return `${tag}[${attrName}=\"${escapeCssValue(attrValue)}\"]`;\r\n }\r\n }\r\n\r\n const classes = Array.from(element.classList)\r\n .filter(Boolean)\r\n .slice(0, 3)\r\n .map((className) => `.${escapeCssValue(className)}`)\r\n .join(\"\");\r\n if (classes) {\r\n return `${tag}${classes}`;\r\n }\r\n\r\n const parent = element.parentElement;\r\n if (!parent) {\r\n return tag;\r\n }\r\n\r\n const siblingsOfTag = Array.from(parent.children).filter(\r\n (sibling) => sibling.tagName === element.tagName,\r\n );\r\n const index = siblingsOfTag.indexOf(element) + 1;\r\n return `${tag}:nth-of-type(${index})`;\r\n};\r\n\r\nconst buildSelector = (element: Element): string => {\r\n const segments: string[] = [];\r\n let cursor: Element | null = element;\r\n let depth = 0;\r\n\r\n while (cursor && depth < 4) {\r\n const segment = buildSelectorSegment(cursor);\r\n segments.unshift(segment);\r\n if (segment.startsWith(\"#\")) {\r\n break;\r\n }\r\n cursor = cursor.parentElement;\r\n depth += 1;\r\n }\r\n\r\n return segments.join(\" > \");\r\n};\r\n\r\nconst getElementLabel = (element: Element): string => {\r\n const text = normalizeWhitespace(\r\n (element instanceof HTMLElement ? element.innerText : element.textContent) ||\r\n \"\",\r\n );\r\n const ariaLabel = normalizeWhitespace(element.getAttribute(\"aria-label\") || \"\");\r\n const title = normalizeWhitespace(element.getAttribute(\"title\") || \"\");\r\n const placeholder = normalizeWhitespace(\r\n element.getAttribute(\"placeholder\") || \"\",\r\n );\r\n const name = normalizeWhitespace(element.getAttribute(\"name\") || \"\");\r\n const value =\r\n element instanceof HTMLInputElement ||\r\n element instanceof HTMLTextAreaElement ||\r\n element instanceof HTMLButtonElement\r\n ? normalizeWhitespace(element.value || \"\")\r\n : \"\";\r\n\r\n const classHint = Array.from(element.classList)\r\n .map((item) => compactToken(item))\r\n .find(Boolean);\r\n const fallback =\r\n (element.id && `#${element.id}`) ||\r\n (classHint && `.${classHint}`) ||\r\n buildSelector(element);\r\n\r\n const label =\r\n text || ariaLabel || title || placeholder || value || name || fallback;\r\n\r\n if (element.tagName.toLowerCase() === \"input\") {\r\n const inputType = element.getAttribute(\"type\") || \"text\";\r\n return `${inputType} ${label || \"input\"}`;\r\n }\r\n\r\n return label || \"untitled\";\r\n};\r\n\r\nconst getEventHints = (element: Element): string[] => {\r\n const record = element as unknown as Record<string, unknown>;\r\n const eventHints: string[] = [];\r\n\r\n for (const eventName of EVENT_HINT_NAMES) {\r\n const handlerKey = `on${eventName}`;\r\n const hasInlineHandler = Boolean(element.getAttribute(handlerKey));\r\n const hasPropertyHandler = typeof record[handlerKey] === \"function\";\r\n\r\n if (!hasInlineHandler && !hasPropertyHandler) {\r\n continue;\r\n }\r\n\r\n eventHints.push(eventName);\r\n if (eventHints.length >= MAX_EVENT_HINTS_PER_ELEMENT) {\r\n break;\r\n }\r\n }\r\n\r\n return eventHints;\r\n};\r\n\r\nconst getAriaInteractionHints = (element: Element): string[] =>\r\n ARIA_INTERACTION_ATTRS.filter((attrName) => element.hasAttribute(attrName)).map(\r\n (attrName) => attrName.replace(\"aria-\", \"\"),\r\n );\r\n\r\nconst getDataInteractionHints = (element: Element): string[] =>\r\n element\r\n .getAttributeNames()\r\n .filter(\r\n (attrName) =>\r\n attrName.startsWith(\"data-\") && DATA_INTERACTION_PATTERN.test(attrName),\r\n )\r\n .slice(0, 2)\r\n .map((attrName) => attrName.replace(\"data-\", \"\"));\r\n\r\nconst getComputedStyleSignals = (style: CSSStyleDeclaration): string[] => {\r\n const signals: string[] = [];\r\n\r\n if (style.cursor && style.cursor !== \"auto\") {\r\n signals.push(`cursor:${style.cursor}`);\r\n }\r\n if (style.display) {\r\n signals.push(`display:${style.display}`);\r\n }\r\n if (style.position) {\r\n signals.push(`position:${style.position}`);\r\n }\r\n if (style.zIndex && style.zIndex !== \"auto\") {\r\n signals.push(`z-index:${style.zIndex}`);\r\n }\r\n if (style.pointerEvents && style.pointerEvents !== \"auto\") {\r\n signals.push(`pointer-events:${style.pointerEvents}`);\r\n }\r\n if (style.visibility && style.visibility !== \"visible\") {\r\n signals.push(`visibility:${style.visibility}`);\r\n }\r\n if (style.opacity && style.opacity !== \"1\") {\r\n signals.push(`opacity:${style.opacity}`);\r\n }\r\n\r\n return Array.from(new Set(signals));\r\n};\r\n\r\nconst buildBlueprintToken = (element: Element): string => {\r\n const tag = element.tagName.toLowerCase();\r\n const idToken = element.id ? `#${compactToken(element.id)}` : \"\";\r\n const classToken = Array.from(element.classList)\r\n .map((item) => compactToken(item))\r\n .find(Boolean);\r\n\r\n return `${tag}${idToken}${classToken ? `.${classToken}` : \"\"}`;\r\n};\r\n\r\nconst buildBranchDigest = (element: Element, depth: number): string => {\r\n const token = buildBlueprintToken(element);\r\n if (depth <= 0) {\r\n return token;\r\n }\r\n\r\n const children = Array.from(element.children)\r\n .filter((child) => !NON_CONTENT_TAGS.has(child.tagName.toLowerCase()))\r\n .filter((child) => isVisible(child));\r\n if (children.length === 0) {\r\n return token;\r\n }\r\n\r\n const sampled = children\r\n .slice(0, 3)\r\n .map((child) => buildBranchDigest(child, depth - 1));\r\n const overflow =\r\n children.length > sampled.length ? `+${children.length - sampled.length}` : \"\";\r\n\r\n return `${token}>${sampled.join(\"+\")}${overflow}`;\r\n};\r\n\r\nconst collectDomBranchDigest = (): string[] => {\r\n const root = document.body ?? document.documentElement;\r\n const topLevelNodes = Array.from(root.children)\r\n .filter((child) => !NON_CONTENT_TAGS.has(child.tagName.toLowerCase()))\r\n .filter((child) => isVisible(child))\r\n .slice(0, MAX_BRANCH_SAMPLES);\r\n\r\n return topLevelNodes.map((child) =>\r\n buildBranchDigest(child, MAX_BRANCH_DEPTH),\r\n );\r\n};\r\n\r\nconst formatSection = (title: string, lines: string[]): string => {\r\n if (lines.length === 0) {\r\n return `${title}:\\n- none`;\r\n }\r\n\r\n return `${title}:\\n${lines.join(\"\\n\")}`;\r\n};\r\n\r\nconst buildOuterHtmlDigest = (): string => {\r\n const raw = document.body?.outerHTML || document.documentElement.outerHTML;\r\n\r\n const withoutScripts = raw\r\n .replace(/<script[\\s\\S]*?<\\/script>/gi, \"\")\r\n .replace(/<style[\\s\\S]*?<\\/style>/gi, \"\")\r\n .replace(/<noscript[\\s\\S]*?<\\/noscript>/gi, \"\")\r\n .replace(/<!--[\\s\\S]*?-->/g, \"\")\r\n .replace(/\\s+/g, \" \")\r\n .trim();\r\n\r\n const structural = withoutScripts\r\n .replace(/>[^<]*</g, \"><\")\r\n .replace(/\\s+/g, \" \")\r\n .trim();\r\n\r\n return structural;\r\n};\r\n\r\nconst collectTextSnippets = (): string[] => {\r\n const root =\r\n document.querySelector(\"main, article, [role='main']\") ?? document.body;\r\n const snippets: string[] = [];\r\n const seen = new Set<string>();\r\n\r\n const candidates = Array.from(root.querySelectorAll(\"p, li, h1, h2, h3\"));\r\n for (const node of candidates) {\r\n if (!isVisible(node)) {\r\n continue;\r\n }\r\n\r\n const text = normalizeWhitespace(node.textContent || \"\");\r\n if (!text || text.length < 20) {\r\n continue;\r\n }\r\n\r\n if (seen.has(text)) {\r\n continue;\r\n }\r\n\r\n seen.add(text);\r\n snippets.push(`- ${text}`);\r\n if (snippets.length >= MAX_TEXT_SNIPPETS) {\r\n break;\r\n }\r\n }\r\n\r\n return snippets;\r\n};\r\n\r\nconst collectLandmarkSnapshot = (): string[] => {\r\n const probes: Array<{ label: string; selector: string }> = [\r\n { label: \"main\", selector: \"main, [role='main']\" },\r\n { label: \"nav\", selector: \"nav, [role='navigation']\" },\r\n { label: \"section\", selector: \"section\" },\r\n { label: \"article\", selector: \"article\" },\r\n { label: \"form\", selector: \"form\" },\r\n { label: \"a\", selector: \"a\" },\r\n { label: \"button\", selector: \"button\" },\r\n { label: \"input\", selector: \"input\" },\r\n { label: \"role=button/link\", selector: \"[role='button'], [role='link']\" },\r\n { label: \"onclick attrs\", selector: \"[onclick]\" },\r\n {\r\n label: \"other event attrs\",\r\n selector:\r\n \"[onpointerdown], [onpointerup], [onkeydown], [onkeyup], [onchange], [onsubmit]\",\r\n },\r\n { label: \"tabindex\", selector: \"[tabindex]\" },\r\n { label: \"contenteditable\", selector: \"[contenteditable='true']\" },\r\n { label: \"inline cursor styles\", selector: \"[style*='cursor']\" },\r\n ];\r\n\r\n return probes.map(\r\n ({ label, selector }) => `- ${label}: ${document.querySelectorAll(selector).length}`,\r\n );\r\n};\r\n\r\nconst collectSelectorsFromRuleList = (\r\n rules: CSSRuleList,\r\n selectors: Set<string>,\r\n scanned: { count: number },\r\n): void => {\r\n for (const rule of Array.from(rules)) {\r\n if (\r\n scanned.count >= MAX_STYLESHEET_RULES ||\r\n selectors.size >= MAX_STYLESHEET_SELECTORS\r\n ) {\r\n return;\r\n }\r\n\r\n scanned.count += 1;\r\n\r\n if (rule instanceof CSSStyleRule) {\r\n const parts = rule.selectorText\r\n .split(\",\")\r\n .map((selector) => normalizeWhitespace(selector))\r\n .filter(Boolean);\r\n\r\n for (const selector of parts) {\r\n if (!STYLESHEET_SELECTOR_PATTERN.test(selector)) {\r\n continue;\r\n }\r\n selectors.add(selector);\r\n if (selectors.size >= MAX_STYLESHEET_SELECTORS) {\r\n return;\r\n }\r\n }\r\n continue;\r\n }\r\n\r\n if (\"cssRules\" in rule) {\r\n try {\r\n const nestedRules = (rule as CSSMediaRule).cssRules;\r\n collectSelectorsFromRuleList(nestedRules, selectors, scanned);\r\n } catch {\r\n // Ignore inaccessible nested rules.\r\n }\r\n }\r\n }\r\n};\r\n\r\nconst collectStylesheetSelectors = (): string[] => {\r\n const selectors = new Set<string>();\r\n const scanned = { count: 0 };\r\n\r\n for (const stylesheet of Array.from(document.styleSheets)) {\r\n if (\r\n scanned.count >= MAX_STYLESHEET_RULES ||\r\n selectors.size >= MAX_STYLESHEET_SELECTORS\r\n ) {\r\n break;\r\n }\r\n\r\n try {\r\n if (!stylesheet.cssRules) {\r\n continue;\r\n }\r\n collectSelectorsFromRuleList(stylesheet.cssRules, selectors, scanned);\r\n } catch {\r\n // Ignore cross-origin stylesheets.\r\n }\r\n }\r\n\r\n return Array.from(selectors).map((selector) => `- ${selector}`);\r\n};\r\n\r\nconst collectPageSignalSnapshot = (): PageSignalSnapshot => {\r\n const allElements = Array.from(document.querySelectorAll(\"*\"));\r\n const sampledElements = allElements.slice(0, MAX_PAGE_SCAN_ELEMENTS);\r\n\r\n const links: string[] = [];\r\n const linkSet = new Set<string>();\r\n const interactableCandidates = new Map<string, InteractableCandidate>();\r\n const tagCounts = new Map<string, number>();\r\n const roleCounts = new Map<string, number>();\r\n const eventCounts = new Map<string, number>();\r\n const displayCounts = new Map<string, number>();\r\n const positionCounts = new Map<string, number>();\r\n const styleSignalCounts = new Map<string, number>();\r\n\r\n let visibleElements = 0;\r\n let maxDepth = 0;\r\n let semanticInteractables = 0;\r\n let nonSemanticInteractables = 0;\r\n let eventHintElements = 0;\r\n let tabStopElements = 0;\r\n let pointerCursorElements = 0;\r\n let dataHintElements = 0;\r\n let ariaHintElements = 0;\r\n let contentEditableElements = 0;\r\n\r\n for (let order = 0; order < sampledElements.length; order += 1) {\r\n const element = sampledElements[order];\r\n const tag = element.tagName.toLowerCase();\r\n\r\n if (NON_CONTENT_TAGS.has(tag)) {\r\n continue;\r\n }\r\n\r\n if (!isVisible(element)) {\r\n continue;\r\n }\r\n\r\n visibleElements += 1;\r\n bumpCount(tagCounts, tag);\r\n\r\n const role = getPrimaryRole(element);\r\n if (role) {\r\n bumpCount(roleCounts, role);\r\n }\r\n\r\n const depth = getElementDepth(element);\r\n if (depth > maxDepth) {\r\n maxDepth = depth;\r\n }\r\n\r\n const style = window.getComputedStyle(element);\r\n if (TRACKED_DISPLAY_VALUES.has(style.display)) {\r\n bumpCount(displayCounts, style.display);\r\n }\r\n if (TRACKED_POSITION_VALUES.has(style.position)) {\r\n bumpCount(positionCounts, style.position);\r\n }\r\n\r\n const computedStyleSignals = getComputedStyleSignals(style);\r\n for (const styleSignal of computedStyleSignals) {\r\n bumpCount(styleSignalCounts, styleSignal);\r\n }\r\n\r\n const eventHints = getEventHints(element);\r\n if (eventHints.length > 0) {\r\n eventHintElements += 1;\r\n for (const eventName of eventHints) {\r\n bumpCount(eventCounts, eventName);\r\n }\r\n }\r\n\r\n const tabIndex = parseTabIndex(element.getAttribute(\"tabindex\"));\r\n const hasTabStop = tabIndex !== null && tabIndex >= 0;\r\n if (hasTabStop) {\r\n tabStopElements += 1;\r\n }\r\n\r\n const hasPointerCursor = style.cursor === \"pointer\";\r\n if (hasPointerCursor) {\r\n pointerCursorElements += 1;\r\n }\r\n\r\n const dataHints = getDataInteractionHints(element);\r\n if (dataHints.length > 0) {\r\n dataHintElements += 1;\r\n }\r\n\r\n const ariaHints = getAriaInteractionHints(element);\r\n if (ariaHints.length > 0) {\r\n ariaHintElements += 1;\r\n }\r\n\r\n const isContentEditable = element.getAttribute(\"contenteditable\") === \"true\";\r\n if (isContentEditable) {\r\n contentEditableElements += 1;\r\n }\r\n\r\n const href = element.getAttribute(\"href\");\r\n const isNativeInteractive = NATIVE_INTERACTIVE_TAGS.has(tag) && (tag !== \"a\" || Boolean(href));\r\n const isRoleInteractive = INTERACTIVE_ROLES.has(role);\r\n const isDisabled =\r\n element.hasAttribute(\"disabled\") ||\r\n element.getAttribute(\"aria-disabled\") === \"true\";\r\n\r\n if (\r\n tag === \"a\" &&\r\n href &&\r\n !href.startsWith(\"#\") &&\r\n !href.startsWith(\"javascript:\")\r\n ) {\r\n const absoluteHref = toAbsoluteUrl(href);\r\n const label = getElementLabel(element) || absoluteHref;\r\n const line = `- ${label} -> ${absoluteHref}`;\r\n\r\n if (!linkSet.has(line)) {\r\n linkSet.add(line);\r\n links.push(line);\r\n if (links.length >= MAX_LINKS) {\r\n // Keep scanning other elements for page blueprint and interactables.\r\n }\r\n }\r\n }\r\n\r\n const hasInteractionSignals =\r\n isNativeInteractive ||\r\n isRoleInteractive ||\r\n isContentEditable ||\r\n eventHints.length > 0 ||\r\n hasTabStop ||\r\n hasPointerCursor ||\r\n dataHints.length > 0 ||\r\n ariaHints.length > 0;\r\n\r\n if (!hasInteractionSignals || isDisabled) {\r\n continue;\r\n }\r\n\r\n if (isNativeInteractive) {\r\n semanticInteractables += 1;\r\n } else {\r\n nonSemanticInteractables += 1;\r\n }\r\n\r\n const selector = buildSelector(element);\r\n const label = getElementLabel(element);\r\n const styleSignals = computedStyleSignals;\r\n const signalTokens: string[] = [];\r\n\r\n if (eventHints.length > 0) {\r\n signalTokens.push(`evt:${eventHints.join(\"|\")}`);\r\n }\r\n if (isRoleInteractive) {\r\n signalTokens.push(`role:${role}`);\r\n }\r\n if (hasTabStop) {\r\n signalTokens.push(`tab:${tabIndex}`);\r\n }\r\n if (dataHints.length > 0) {\r\n signalTokens.push(`data:${dataHints.join(\"|\")}`);\r\n }\r\n if (ariaHints.length > 0) {\r\n signalTokens.push(`aria:${ariaHints.join(\"|\")}`);\r\n }\r\n if (styleSignals.length > 0) {\r\n signalTokens.push(`css:${styleSignals.join(\"|\")}`);\r\n } else if (hasPointerCursor) {\r\n signalTokens.push(\"css:cursor:pointer\");\r\n }\r\n\r\n const signalBlock =\r\n signalTokens.length > 0 ? ` [${signalTokens.join(\"; \")}]` : \"\";\r\n const line = `- ${tag} ${selector}${signalBlock} (${label})`;\r\n\r\n const score =\r\n eventHints.length * 5 +\r\n (isNativeInteractive ? 5 : 0) +\r\n (isRoleInteractive ? 4 : 0) +\r\n (hasTabStop ? 2 : 0) +\r\n (hasPointerCursor ? 2 : 0) +\r\n (dataHints.length > 0 ? 2 : 0) +\r\n (ariaHints.length > 0 ? 1 : 0) +\r\n (isContentEditable ? 2 : 0);\r\n\r\n const existing = interactableCandidates.get(line);\r\n if (!existing || score > existing.score) {\r\n interactableCandidates.set(line, { line, score, order });\r\n }\r\n }\r\n\r\n const interactables = Array.from(interactableCandidates.values())\r\n .sort((a, b) => b.score - a.score || a.order - b.order)\r\n .slice(0, MAX_INTERACTABLES)\r\n .map((candidate) => candidate.line);\r\n\r\n const interactiveRoleCounts = new Map(\r\n Array.from(roleCounts.entries()).filter(([role]) =>\r\n INTERACTIVE_ROLES.has(role),\r\n ),\r\n );\r\n\r\n const interactionSignals = [\r\n `- coverage: semantic=${semanticInteractables}, non-semantic=${nonSemanticInteractables}, contenteditable=${contentEditableElements}`,\r\n `- listener hints: ${formatTopCounts(eventCounts, 8)}`,\r\n `- interaction cues: tabindex>=0=${tabStopElements}, pointer-cursor=${pointerCursorElements}, data-hints=${dataHintElements}, aria-hints=${ariaHintElements}`,\r\n `- role hints: ${formatTopCounts(interactiveRoleCounts, 8)}`,\r\n `- css footprint: ${formatTopCounts(styleSignalCounts, 10)}`,\r\n \"- listener scope: inline/on* handlers are detected directly; addEventListener handlers are inferred via cues.\",\r\n ];\r\n\r\n const branchDigest = collectDomBranchDigest();\r\n const pageBlueprint = [\r\n `- nodes: total=${allElements.length}, scanned=${sampledElements.length}, visible=${visibleElements}, max-depth=${maxDepth}${allElements.length > sampledElements.length ? \", sampling=on\" : \"\"}`,\r\n `- tag density: ${formatTopCounts(tagCounts, 10)}`,\r\n `- role density: ${formatTopCounts(roleCounts, 8)}`,\r\n `- layout density: display(${formatTopCounts(displayCounts, 6)}), position(${formatTopCounts(positionCounts, 4)})`,\r\n `- branch digest: ${branchDigest.length > 0 ? branchDigest.join(\" || \") : \"none\"}`,\r\n ];\r\n\r\n return {\r\n links: links.slice(0, MAX_LINKS),\r\n interactables,\r\n interactionSignals,\r\n styleSelectors: collectStylesheetSelectors(),\r\n pageBlueprint,\r\n };\r\n};\r\n\r\nexport const buildPageContextSummary = (\r\n input: PageContextSummaryInput,\r\n): string => {\r\n const sections = [\r\n formatSection(\"Meta\", [\r\n `- URL: ${input.url || \"unknown\"}`,\r\n `- Title: ${input.title || \"unknown\"}`,\r\n `- Lang: ${input.lang || \"unknown\"}`,\r\n ]),\r\n formatSection(\"Headings\", input.headings),\r\n formatSection(\"Landmark Snapshot\", input.landmarks),\r\n formatSection(\"Interaction Signals\", input.interactionSignals),\r\n formatSection(\"Stylesheet Selector Snapshot\", input.styleSelectors),\r\n formatSection(\"Compressed Page Blueprint\", input.pageBlueprint),\r\n formatSection(\"Top Links\", input.links),\r\n formatSection(\"Top Interactables\", input.interactables),\r\n formatSection(\"Main Content Snippets\", input.textSnippets),\r\n formatSection(\"OuterHTML Skeleton\", [\r\n `- ${input.outerHtmlDigest || \"unavailable\"}`,\r\n ]),\r\n ];\r\n\r\n return sections.join(\"\\n\\n\");\r\n};\r\n\r\nexport const clearPageContextCache = (): void => {\r\n pageContextCache.clear();\r\n cacheHydrated = false;\r\n if (typeof sessionStorage !== \"undefined\") {\r\n sessionStorage.removeItem(PAGE_CONTEXT_CACHE_KEY);\r\n }\r\n};\r\n\r\nexport const getCachedPageContexts = (): CachedPageContextEntry[] => {\r\n hydrateCacheFromStorage();\r\n return Array.from(pageContextCache.values()).sort(\r\n (a, b) => b.capturedAt - a.capturedAt,\r\n );\r\n};\r\n\r\nexport const getPageContext = (): PageContext => {\r\n if (typeof window === \"undefined\" || typeof document === \"undefined\") {\r\n return {\r\n links: [],\r\n interactables: [],\r\n summary: \"\",\r\n };\r\n }\r\n\r\n hydrateCacheFromStorage();\r\n const url = canonicalUrl(window.location.href);\r\n const cached = pageContextCache.get(url);\r\n if (cached) {\r\n console.info(`[Autic] context cache hit url=${url}`);\r\n return {\r\n links: cached.links,\r\n interactables: cached.interactables,\r\n summary: buildSummaryWithHistory(cached),\r\n };\r\n }\r\n\r\n console.info(`[Autic] context cache miss url=${url}`);\r\n\r\n const snapshot = collectPageSignalSnapshot();\r\n const headings = Array.from(document.querySelectorAll(\"h1, h2, h3\"))\r\n .filter((element) => isVisible(element))\r\n .map((element) =>\r\n `- ${normalizeWhitespace(element.textContent || \"\")}`,\r\n )\r\n .filter((line) => line !== \"- \")\r\n .slice(0, MAX_HEADINGS);\r\n\r\n const summary = buildPageContextSummary({\r\n url,\r\n title: document.title,\r\n lang: document.documentElement.lang,\r\n headings,\r\n landmarks: collectLandmarkSnapshot(),\r\n links: snapshot.links,\r\n interactables: snapshot.interactables,\r\n interactionSignals: snapshot.interactionSignals,\r\n styleSelectors: snapshot.styleSelectors,\r\n pageBlueprint: snapshot.pageBlueprint,\r\n textSnippets: collectTextSnippets(),\r\n outerHtmlDigest: buildOuterHtmlDigest(),\r\n });\r\n\r\n const entry: CachedPageContextEntry = {\r\n url,\r\n summary,\r\n links: snapshot.links,\r\n interactables: snapshot.interactables,\r\n capturedAt: Date.now(),\r\n version: PAGE_CONTEXT_CACHE_VERSION,\r\n };\r\n\r\n pageContextCache.set(url, entry);\r\n pruneOldestCacheEntries();\r\n persistCacheToStorage();\r\n console.info(\r\n `[Autic] context cache stored url=${url} size=${pageContextCache.size}`,\r\n );\r\n\r\n return {\r\n links: entry.links,\r\n interactables: entry.interactables,\r\n summary: buildSummaryWithHistory(entry),\r\n };\r\n};\r\n","/**\r\n * Tunable parameters for page context collection.\r\n *\r\n * Adjust these values to control how much information the agent receives\r\n * about the current page. Lower values reduce token usage (and latency);\r\n * higher values give the LLM more detail to work with.\r\n */\r\n\r\n/** Maximum number of links to include in the context. */\r\nexport const MAX_LINKS = 20;\r\n\r\n/** Maximum number of interactable elements to include. */\r\nexport const MAX_INTERACTABLES = 24;\r\n\r\n/** Maximum number of headings (h1-h3) to include. */\r\nexport const MAX_HEADINGS = 10;\r\n\r\n/** Maximum number of main-content text snippets. */\r\nexport const MAX_TEXT_SNIPPETS = 4;\r\n\r\n/** Maximum number of pages kept in the in-memory context cache. */\r\nexport const MAX_CACHED_PAGES = 20;\r\n\r\n/** Maximum number of DOM elements scanned per page. */\r\nexport const MAX_PAGE_SCAN_ELEMENTS = 2000;\r\n\r\n/** Maximum number of event-handler hints collected per element. */\r\nexport const MAX_EVENT_HINTS_PER_ELEMENT = 4;\r\n\r\n/** Maximum top-level DOM branches sampled for the branch digest. */\r\nexport const MAX_BRANCH_SAMPLES = 4;\r\n\r\n/** Maximum tree depth for each branch digest sample. */\r\nexport const MAX_BRANCH_DEPTH = 2;\r\n\r\n/** Maximum stylesheet selectors to include in summary. */\r\nexport const MAX_STYLESHEET_SELECTORS = 80;\r\n\r\n/** Maximum CSS rules to scan while collecting stylesheet selectors. */\r\nexport const MAX_STYLESHEET_RULES = 700;\r\n","import { getPageContext } from \"./context\";\r\nimport { COLORS } from \"../styles/constants\";\r\n\r\nconst AGENT_CURSOR_ID = \"auticbot-agent-cursor\";\r\nexport const CURSOR_MOVE_DURATION_MS = 900;\r\nexport const SCROLL_DURATION_MS = 900;\r\nconst CURSOR_EASING = \"cubic-bezier(0.4, 0, 0.2, 1)\";\r\nconst CURSOR_HOVER_RADIUS_PX = 14;\r\nconst CURSOR_DIAMETER_PX = 25;\r\n\r\n// ── Pending Agent Resume State (survives page reloads) ──────────────\r\n\r\nconst RESUME_STORAGE_KEY = \"bulut_agent_resume\";\r\nconst RESUME_TTL_MS = 60_000; // 1 minute\r\n\r\nexport interface PendingAgentResume {\r\n sessionId: string;\r\n projectId: string;\r\n model: string;\r\n voice: string;\r\n accessibilityMode: boolean;\r\n pendingToolCalls: Array<{\r\n call_id: string;\r\n tool: string;\r\n args: Record<string, unknown>;\r\n }>;\r\n completedResults: Array<{ call_id: string; result: string }>;\r\n savedAt: number;\r\n}\r\n\r\nexport const savePendingAgentResume = (\r\n state: Omit<PendingAgentResume, \"savedAt\">,\r\n): void => {\r\n if (typeof localStorage === \"undefined\") return;\r\n try {\r\n localStorage.setItem(\r\n RESUME_STORAGE_KEY,\r\n JSON.stringify({ ...state, savedAt: Date.now() }),\r\n );\r\n } catch {\r\n // localStorage may be full or blocked\r\n }\r\n};\r\n\r\nexport const getPendingAgentResume = (): PendingAgentResume | null => {\r\n if (typeof localStorage === \"undefined\") return null;\r\n const raw = localStorage.getItem(RESUME_STORAGE_KEY);\r\n if (!raw) return null;\r\n try {\r\n const parsed = JSON.parse(raw) as PendingAgentResume;\r\n if (Date.now() - parsed.savedAt > RESUME_TTL_MS) {\r\n clearPendingAgentResume();\r\n return null;\r\n }\r\n return parsed;\r\n } catch {\r\n clearPendingAgentResume();\r\n return null;\r\n }\r\n};\r\n\r\nexport const clearPendingAgentResume = (): void => {\r\n if (typeof localStorage === \"undefined\") return;\r\n localStorage.removeItem(RESUME_STORAGE_KEY);\r\n};\r\n\r\ntype InteractAction = \"move\" | \"click\" | \"type\" | \"submit\";\r\n\r\ninterface InteractToolCall {\r\n tool: \"interact\";\r\n action: InteractAction;\r\n selector?: string;\r\n text?: string;\r\n x?: number;\r\n y?: number;\r\n}\r\n\r\ninterface NavigateToolCall {\r\n tool: \"navigate\";\r\n url: string;\r\n}\r\n\r\ninterface GetPageContextToolCall {\r\n tool: \"getPageContext\";\r\n}\r\n\r\ninterface ScrollToolCall {\r\n tool: \"scroll\";\r\n selector: string;\r\n}\r\n\r\nexport type AgentToolCall =\r\n | InteractToolCall\r\n | NavigateToolCall\r\n | GetPageContextToolCall\r\n | ScrollToolCall;\r\n\r\nexport interface ParsedAgentResponse {\r\n reply: string;\r\n toolCalls: AgentToolCall[];\r\n}\r\n\r\ninterface JsonObject {\r\n [key: string]: unknown;\r\n}\r\n\r\ninterface ResolvedTarget {\r\n element?: HTMLElement;\r\n x: number;\r\n y: number;\r\n}\r\n\r\nconst isObject = (value: unknown): value is JsonObject =>\r\n typeof value === \"object\" && value !== null && !Array.isArray(value);\r\n\r\nconst asString = (value: unknown): string | undefined =>\r\n typeof value === \"string\" && value.trim() ? value.trim() : undefined;\r\n\r\nconst asNumber = (value: unknown): number | undefined =>\r\n typeof value === \"number\" && Number.isFinite(value) ? value : undefined;\r\n\r\nconst extractJsonCandidate = (raw: string): string => {\r\n const trimmed = raw.trim();\r\n const fencedMatch = trimmed.match(/```(?:json)?\\s*([\\s\\S]*?)\\s*```/i);\r\n if (fencedMatch?.[1]) {\r\n return fencedMatch[1].trim();\r\n }\r\n return trimmed;\r\n};\r\n\r\nconst extractFirstJsonObject = (input: string): string | null => {\r\n const start = input.indexOf(\"{\");\r\n if (start < 0) {\r\n return null;\r\n }\r\n\r\n let depth = 0;\r\n let inString = false;\r\n let isEscaped = false;\r\n\r\n for (let i = start; i < input.length; i += 1) {\r\n const char = input[i];\r\n\r\n if (inString) {\r\n if (isEscaped) {\r\n isEscaped = false;\r\n } else if (char === \"\\\\\") {\r\n isEscaped = true;\r\n } else if (char === '\"') {\r\n inString = false;\r\n }\r\n continue;\r\n }\r\n\r\n if (char === '\"') {\r\n inString = true;\r\n continue;\r\n }\r\n\r\n if (char === \"{\") {\r\n depth += 1;\r\n continue;\r\n }\r\n\r\n if (char === \"}\") {\r\n depth -= 1;\r\n if (depth === 0) {\r\n return input.slice(start, i + 1);\r\n }\r\n }\r\n }\r\n\r\n return null;\r\n};\r\n\r\nconst parseJsonFromRaw = (raw: string): unknown => {\r\n const candidate = extractJsonCandidate(raw);\r\n try {\r\n return JSON.parse(candidate);\r\n } catch {\r\n const objectCandidate = extractFirstJsonObject(candidate);\r\n if (!objectCandidate) {\r\n return null;\r\n }\r\n try {\r\n return JSON.parse(objectCandidate);\r\n } catch {\r\n return null;\r\n }\r\n }\r\n};\r\n\r\nconst sanitizeToolCalls = (value: unknown): AgentToolCall[] => {\r\n if (!Array.isArray(value)) {\r\n return [];\r\n }\r\n\r\n const toolCalls: AgentToolCall[] = [];\r\n\r\n for (const item of value) {\r\n if (!isObject(item)) {\r\n continue;\r\n }\r\n\r\n if (item.tool === \"interact\") {\r\n const action = asString(item.action) as InteractAction | undefined;\r\n if (!action || ![\"move\", \"click\", \"type\", \"submit\"].includes(action)) {\r\n continue;\r\n }\r\n\r\n toolCalls.push({\r\n tool: \"interact\",\r\n action,\r\n selector: asString(item.selector),\r\n text: typeof item.text === \"string\" ? item.text : undefined,\r\n x: asNumber(item.x),\r\n y: asNumber(item.y),\r\n });\r\n continue;\r\n }\r\n\r\n if (item.tool === \"navigate\") {\r\n const url = asString(item.url);\r\n if (!url) {\r\n continue;\r\n }\r\n\r\n toolCalls.push({\r\n tool: \"navigate\",\r\n url,\r\n });\r\n continue;\r\n }\r\n\r\n if (item.tool === \"getPageContext\") {\r\n toolCalls.push({\r\n tool: \"getPageContext\",\r\n });\r\n continue;\r\n }\r\n\r\n if (item.tool === \"scroll\") {\r\n const selector = asString(item.selector);\r\n if (!selector) {\r\n continue;\r\n }\r\n\r\n toolCalls.push({\r\n tool: \"scroll\",\r\n selector,\r\n });\r\n }\r\n }\r\n\r\n return toolCalls;\r\n};\r\n\r\nexport const parseAgentResponse = (raw: string): ParsedAgentResponse => {\r\n const parsed = parseJsonFromRaw(raw);\r\n if (!isObject(parsed)) {\r\n return {\r\n reply: raw.trim(),\r\n toolCalls: [],\r\n };\r\n }\r\n\r\n const reply = asString(parsed.reply) || \"\";\r\n const toolCalls = sanitizeToolCalls(parsed.tool_calls ?? parsed.toolCalls);\r\n\r\n return {\r\n reply,\r\n toolCalls,\r\n };\r\n};\r\n\r\nexport const clamp = (value: number, min: number, max: number): number =>\r\n Math.min(max, Math.max(min, value));\r\n\r\nexport const easeInOutCubic = (progress: number): number => {\r\n if (progress < 0.5) {\r\n return 4 * progress * progress * progress;\r\n }\r\n return 1 - Math.pow(-2 * progress + 2, 3) / 2;\r\n};\r\n\r\nexport const easeInOutSine = (progress: number): number =>\r\n -(Math.cos(Math.PI * progress) - 1) / 2;\r\n\r\nexport const isRectOutsideViewport = (\r\n rect: Pick<DOMRect, \"top\" | \"bottom\">,\r\n viewportHeight: number,\r\n): boolean => rect.top < 0 || rect.bottom > viewportHeight;\r\n\r\nexport const computeCenteredScrollTop = (\r\n currentScrollY: number,\r\n rectTop: number,\r\n rectHeight: number,\r\n viewportHeight: number,\r\n maxScrollTop: number,\r\n): number => {\r\n const desired =\r\n currentScrollY + rectTop - (viewportHeight / 2 - rectHeight / 2);\r\n return clamp(desired, 0, Math.max(0, maxScrollTop));\r\n};\r\n\r\nexport const animateWindowScrollTo = async (\r\n targetY: number,\r\n durationMs: number = SCROLL_DURATION_MS,\r\n): Promise<void> => {\r\n if (typeof window === \"undefined\") {\r\n return;\r\n }\r\n\r\n const startY = window.scrollY;\r\n const delta = targetY - startY;\r\n if (Math.abs(delta) < 1) {\r\n return;\r\n }\r\n\r\n await new Promise<void>((resolve) => {\r\n const raf =\r\n window.requestAnimationFrame ||\r\n ((callback: FrameRequestCallback) => window.setTimeout(() => callback(performance.now()), 16));\r\n\r\n const startTime = performance.now();\r\n\r\n const step = (now: number) => {\r\n const elapsed = now - startTime;\r\n const progress = clamp(elapsed / durationMs, 0, 1);\r\n const eased = easeInOutSine(progress);\r\n window.scrollTo(0, startY + delta * eased);\r\n\r\n if (progress < 1) {\r\n raf(step);\r\n } else {\r\n resolve();\r\n }\r\n };\r\n\r\n raf(step);\r\n });\r\n};\r\n\r\nconst setCursorPosition = (cursor: HTMLElement, x: number, y: number) => {\r\n cursor.style.left = `${x}px`;\r\n cursor.style.top = `${y}px`;\r\n};\r\n\r\nconst getCursorPosition = (cursor: HTMLElement): { x: number; y: number } => ({\r\n x: Number.parseFloat(cursor.style.left) || 0,\r\n y: Number.parseFloat(cursor.style.top) || 0,\r\n});\r\n\r\nconst setCursorVisibility = (cursor: HTMLElement, visible: boolean) => {\r\n cursor.style.opacity = visible ? \"1\" : \"0\";\r\n};\r\n\r\nconst isVisibleElement = (element: HTMLElement): boolean => {\r\n const style = window.getComputedStyle(element);\r\n return style.display !== \"none\" && style.visibility !== \"hidden\" && style.opacity !== \"0\";\r\n};\r\n\r\nconst getBulutShadowRoots = (): ShadowRoot[] => {\r\n const roots: ShadowRoot[] = [];\r\n\r\n const defaultHost = document.getElementById(\"bulut-container\");\r\n if (defaultHost?.shadowRoot) {\r\n roots.push(defaultHost.shadowRoot);\r\n }\r\n\r\n const allElements = document.querySelectorAll<HTMLElement>(\"*\");\r\n for (const el of allElements) {\r\n if (!el.shadowRoot) {\r\n continue;\r\n }\r\n if (!roots.includes(el.shadowRoot)) {\r\n roots.push(el.shadowRoot);\r\n }\r\n }\r\n\r\n return roots;\r\n};\r\n\r\nconst findAgentUiAnchorElement = (): HTMLElement | null => {\r\n const roots = getBulutShadowRoots();\r\n\r\n for (const root of roots) {\r\n const panel = root.querySelector<HTMLElement>(\".bulut-chat-window\");\r\n if (panel && isVisibleElement(panel)) {\r\n return panel;\r\n }\r\n }\r\n\r\n for (const root of roots) {\r\n const button = root.querySelector<HTMLElement>(\".bulut-fab-container\");\r\n if (button && isVisibleElement(button)) {\r\n return button;\r\n }\r\n }\r\n\r\n return null;\r\n};\r\n\r\nconst getAgentWindowTopLeft = (): { x: number; y: number } => {\r\n const anchor = findAgentUiAnchorElement();\r\n if (!anchor) {\r\n return {\r\n x: CURSOR_DIAMETER_PX / 2,\r\n y: CURSOR_DIAMETER_PX / 2,\r\n };\r\n }\r\n\r\n const rect = anchor.getBoundingClientRect();\r\n return {\r\n x: rect.left + window.scrollX + CURSOR_DIAMETER_PX / 2,\r\n y: rect.top + window.scrollY + CURSOR_DIAMETER_PX / 2,\r\n };\r\n};\r\n\r\nexport const hideAgentCursor = (): void => {\r\n if (typeof document === \"undefined\" || typeof window === \"undefined\") {\r\n return;\r\n }\r\n\r\n const cursor = document.getElementById(AGENT_CURSOR_ID);\r\n if (!(cursor instanceof HTMLElement)) {\r\n return;\r\n }\r\n\r\n setCursorVisibility(cursor, false);\r\n};\r\n\r\nlet cursorHoverTrackingInitialized = false;\r\nconst initializeCursorHoverTracking = () => {\r\n if (cursorHoverTrackingInitialized) {\r\n return;\r\n }\r\n cursorHoverTrackingInitialized = true;\r\n\r\n document.addEventListener(\"mousemove\", (event) => {\r\n const cursor = document.getElementById(AGENT_CURSOR_ID);\r\n if (!(cursor instanceof HTMLElement)) {\r\n return;\r\n }\r\n\r\n if (cursor.style.opacity !== \"1\") {\r\n return;\r\n }\r\n\r\n const { x, y } = getCursorPosition(cursor);\r\n const pointerX = event.pageX;\r\n const pointerY = event.pageY;\r\n const distance = Math.hypot(pointerX - x, pointerY - y);\r\n\r\n if (distance <= CURSOR_HOVER_RADIUS_PX) {\r\n setCursorVisibility(cursor, false);\r\n }\r\n });\r\n};\r\n\r\nconst ensureCursor = (): HTMLElement => {\r\n const existing = document.getElementById(AGENT_CURSOR_ID);\r\n if (existing) {\r\n // Keep cursor color in sync with the current theme\r\n existing.style.background = COLORS.primary;\r\n initializeCursorHoverTracking();\r\n return existing as HTMLElement;\r\n }\r\n\r\n const cursor = document.createElement(\"div\");\r\n cursor.id = AGENT_CURSOR_ID;\r\n cursor.style.position = \"absolute\";\r\n const startPosition = getAgentWindowTopLeft();\r\n cursor.style.left = `${startPosition.x}px`;\r\n cursor.style.top = `${startPosition.y}px`;\r\n cursor.style.opacity = \"0\";\r\n const width = CURSOR_DIAMETER_PX;\r\n cursor.style.width = `${width}px`;\r\n cursor.style.height = `${width}px`;\r\n cursor.style.borderRadius = \"50%\";\r\n const baseColor = COLORS.primary;\r\n cursor.style.background = baseColor;\r\n const border = 25 * 16 / 100;\r\n cursor.style.border = `${border}px solid #ffffff`;\r\n cursor.style.boxShadow = \"0px 0px 10px rgba(0, 11, 26, 0.5)\";\r\n cursor.style.boxSizing = \"border-box\";\r\n cursor.style.zIndex = \"2147483647\";\r\n cursor.style.pointerEvents = \"none\";\r\n cursor.style.transform = \"translate(-50%, -50%)\";\r\n cursor.style.transition = `left ${CURSOR_MOVE_DURATION_MS}ms ${CURSOR_EASING}, top ${CURSOR_MOVE_DURATION_MS}ms ${CURSOR_EASING}, opacity 150ms ease-out`;\r\n document.body.appendChild(cursor);\r\n initializeCursorHoverTracking();\r\n console.info(`[Autic] cursor created color=${baseColor} duration=${CURSOR_MOVE_DURATION_MS}ms`);\r\n return cursor;\r\n};\r\n\r\nconst waitForNextAnimationFrame = async (): Promise<void> => {\r\n const raf =\r\n window.requestAnimationFrame ||\r\n ((callback: FrameRequestCallback) => window.setTimeout(() => callback(performance.now()), 16));\r\n await new Promise<void>((resolve) => {\r\n raf(() => resolve());\r\n });\r\n};\r\n\r\nconst moveCursor = async (x: number, y: number) => {\r\n const cursor = ensureCursor();\r\n if (cursor.dataset.transitionReady !== \"true\") {\r\n cursor.dataset.transitionReady = \"true\";\r\n await waitForNextAnimationFrame();\r\n }\r\n\r\n const startPosition = getAgentWindowTopLeft();\r\n setCursorVisibility(cursor, true);\r\n setCursorPosition(cursor, startPosition.x, startPosition.y);\r\n await new Promise((resolve) => setTimeout(resolve, CURSOR_MOVE_DURATION_MS));\r\n\r\n setCursorPosition(cursor, x, y);\r\n await new Promise((resolve) => setTimeout(resolve, CURSOR_MOVE_DURATION_MS));\r\n};\r\n\r\nconst getElementCenter = (element: HTMLElement): { x: number; y: number } => {\r\n const rect = element.getBoundingClientRect();\r\n return {\r\n x: rect.left + window.scrollX + rect.width / 2,\r\n y: rect.top + window.scrollY + rect.height / 2,\r\n };\r\n};\r\n\r\nconst CONTAINS_SELECTOR_PATTERN = /^(.*?):contains\\((['\"])(.*?)\\2\\)\\s*$/;\r\n\r\nconst findElementBySelector = (selector: string): Element | null => {\r\n try {\r\n return document.querySelector(selector);\r\n } catch (error) {\r\n const containsMatch = selector.match(CONTAINS_SELECTOR_PATTERN);\r\n if (!containsMatch) {\r\n console.warn(`AuticBot selector invalid: ${selector}`, error);\r\n return null;\r\n }\r\n\r\n const baseSelector = containsMatch[1]?.trim() || \"*\";\r\n const expectedText = containsMatch[3]?.trim() || \"\";\r\n if (!expectedText) {\r\n console.warn(`AuticBot selector contains empty text: ${selector}`);\r\n return null;\r\n }\r\n\r\n try {\r\n const candidates = document.querySelectorAll(baseSelector);\r\n for (const candidate of candidates) {\r\n if (candidate.textContent?.includes(expectedText)) {\r\n return candidate;\r\n }\r\n }\r\n return null;\r\n } catch (fallbackError) {\r\n console.warn(`AuticBot selector fallback invalid: ${selector}`, fallbackError);\r\n return null;\r\n }\r\n }\r\n};\r\n\r\nconst resolveTarget = (call: InteractToolCall): ResolvedTarget | null => {\r\n if (call.selector) {\r\n const selected = findElementBySelector(call.selector);\r\n\r\n if (selected instanceof HTMLElement) {\r\n const center = getElementCenter(selected);\r\n return {\r\n element: selected,\r\n x: center.x,\r\n y: center.y,\r\n };\r\n }\r\n console.warn(`AuticBot interact: selector not found: ${call.selector}`);\r\n }\r\n\r\n if (typeof call.x === \"number\" && typeof call.y === \"number\") {\r\n return {\r\n x: call.x,\r\n y: call.y,\r\n };\r\n }\r\n\r\n console.warn(\"AuticBot interact: missing target selector or coordinates.\", call);\r\n return null;\r\n};\r\n\r\nconst dispatchMouseEvent = (\r\n element: HTMLElement,\r\n type: string,\r\n x: number,\r\n y: number,\r\n) => {\r\n element.dispatchEvent(\r\n new MouseEvent(type, {\r\n bubbles: true,\r\n cancelable: true,\r\n view: window,\r\n clientX: x - window.scrollX,\r\n clientY: y - window.scrollY,\r\n }),\r\n );\r\n};\r\n\r\nconst setNativeInputLikeValue = (\r\n element: HTMLInputElement | HTMLTextAreaElement,\r\n text: string,\r\n) => {\r\n const prototype =\r\n element instanceof HTMLTextAreaElement\r\n ? HTMLTextAreaElement.prototype\r\n : HTMLInputElement.prototype;\r\n const descriptor = Object.getOwnPropertyDescriptor(prototype, \"value\");\r\n if (descriptor?.set) {\r\n descriptor.set.call(element, text);\r\n } else {\r\n element.value = text;\r\n }\r\n\r\n // Keep both current and default values aligned so client re-focus\r\n // does not appear to \"erase\" tool-filled form fields.\r\n element.defaultValue = text;\r\n element.setAttribute(\"value\", text);\r\n};\r\n\r\nconst typeIntoElement = (element: HTMLElement, text: string) => {\r\n if (element instanceof HTMLInputElement || element instanceof HTMLTextAreaElement) {\r\n element.focus();\r\n setNativeInputLikeValue(element, text);\r\n element.dispatchEvent(new Event(\"input\", { bubbles: true }));\r\n element.dispatchEvent(new Event(\"change\", { bubbles: true }));\r\n return;\r\n }\r\n\r\n if (element.isContentEditable) {\r\n element.focus();\r\n element.textContent = text;\r\n element.dispatchEvent(new Event(\"input\", { bubbles: true }));\r\n return;\r\n }\r\n\r\n console.warn(\r\n \"AuticBot interact: type action requires input, textarea, or contenteditable target.\",\r\n );\r\n};\r\n\r\nconst submitElement = (element: HTMLElement) => {\r\n if (element.tagName === \"FORM\") {\r\n (element as HTMLFormElement).requestSubmit();\r\n return;\r\n }\r\n\r\n if (element.tagName === \"BUTTON\" && (element as HTMLButtonElement).form) {\r\n (element as HTMLButtonElement).form?.requestSubmit();\r\n return;\r\n }\r\n\r\n const parentForm = element.closest(\"form\");\r\n if (parentForm) {\r\n parentForm.requestSubmit();\r\n return;\r\n }\r\n\r\n console.warn(\"AuticBot interact: submit action requires a form target.\");\r\n};\r\n\r\nconst slowScrollElementIntoView = async (element: HTMLElement): Promise<void> => {\r\n await slowScrollElementIntoViewWithMode(element, false);\r\n};\r\n\r\nconst slowScrollElementIntoViewWithMode = async (\r\n element: HTMLElement,\r\n forceCenter: boolean,\r\n): Promise<void> => {\r\n const rect = element.getBoundingClientRect();\r\n const viewportHeight = window.innerHeight;\r\n\r\n if (!forceCenter && !isRectOutsideViewport(rect, viewportHeight)) {\r\n return;\r\n }\r\n\r\n const maxScrollTop = Math.max(\r\n 0,\r\n Math.max(document.body.scrollHeight, document.documentElement.scrollHeight) - viewportHeight,\r\n );\r\n const targetY = computeCenteredScrollTop(\r\n window.scrollY,\r\n rect.top,\r\n rect.height,\r\n viewportHeight,\r\n maxScrollTop,\r\n );\r\n\r\n await animateWindowScrollTo(targetY, SCROLL_DURATION_MS);\r\n};\r\n\r\nconst executeScroll = async (call: ScrollToolCall) => {\r\n const selected = findElementBySelector(call.selector);\r\n if (!(selected instanceof HTMLElement)) {\r\n console.warn(`AuticBot scroll: selector not found: ${call.selector}`);\r\n return;\r\n }\r\n\r\n await slowScrollElementIntoViewWithMode(selected, true);\r\n const center = getElementCenter(selected);\r\n await moveCursor(center.x, center.y);\r\n};\r\n\r\nconst executeInteract = async (call: InteractToolCall) => {\r\n const target = resolveTarget(call);\r\n if (!target) {\r\n return;\r\n }\r\n\r\n if (call.action === \"click\" && target.element) {\r\n await slowScrollElementIntoView(target.element);\r\n const center = getElementCenter(target.element);\r\n target.x = center.x;\r\n target.y = center.y;\r\n }\r\n\r\n await moveCursor(target.x, target.y);\r\n\r\n if (call.action === \"move\") {\r\n return;\r\n }\r\n\r\n if (!target.element) {\r\n console.warn(\"AuticBot interact: target element not available for action.\", call.action);\r\n return;\r\n }\r\n\r\n if (call.action === \"click\") {\r\n dispatchMouseEvent(target.element, \"pointerdown\", target.x, target.y);\r\n dispatchMouseEvent(target.element, \"mousedown\", target.x, target.y);\r\n dispatchMouseEvent(target.element, \"pointerup\", target.x, target.y);\r\n dispatchMouseEvent(target.element, \"mouseup\", target.x, target.y);\r\n target.element.click();\r\n return;\r\n }\r\n\r\n if (call.action === \"type\") {\r\n typeIntoElement(target.element, call.text ?? \"\");\r\n return;\r\n }\r\n\r\n submitElement(target.element);\r\n};\r\n\r\nconst isSamePageNavigation = (targetUrl: string): boolean => {\r\n try {\r\n const current = new URL(window.location.href);\r\n const target = new URL(targetUrl);\r\n return current.origin === target.origin && current.pathname === target.pathname;\r\n } catch {\r\n return false;\r\n }\r\n};\r\n\r\n/**\r\n * Find the best matching link element for a target URL.\r\n * Supports exact href match, partial path/query/hash match,\r\n * and text-content match for framework <Link> components.\r\n */\r\nconst findMatchingLinkForTarget = (targetUrl: string): HTMLElement | null => {\r\n let parsedTarget: URL | null = null;\r\n try {\r\n parsedTarget = new URL(targetUrl, window.location.href);\r\n } catch {\r\n // will fall through to text-based matching\r\n }\r\n\r\n const allLinks = Array.from(\r\n document.querySelectorAll('a[href], [role=\"link\"][href], [data-href]'),\r\n ) as HTMLElement[];\r\n\r\n // 1. Exact href match\r\n for (const el of allLinks) {\r\n if (el instanceof HTMLAnchorElement && el.href === parsedTarget?.href) {\r\n return el;\r\n }\r\n }\r\n\r\n if (parsedTarget) {\r\n // 2. Match by pathname + search + hash (ignoring origin)\r\n for (const el of allLinks) {\r\n if (!(el instanceof HTMLAnchorElement)) continue;\r\n try {\r\n const elUrl = new URL(el.href, window.location.href);\r\n if (\r\n elUrl.pathname === parsedTarget.pathname &&\r\n elUrl.search === parsedTarget.search &&\r\n elUrl.hash === parsedTarget.hash\r\n ) {\r\n return el;\r\n }\r\n } catch {\r\n continue;\r\n }\r\n }\r\n\r\n // 3. Match by pathname only (query/hash may differ)\r\n for (const el of allLinks) {\r\n if (!(el instanceof HTMLAnchorElement)) continue;\r\n try {\r\n const elUrl = new URL(el.href, window.location.href);\r\n if (elUrl.pathname === parsedTarget.pathname) {\r\n return el;\r\n }\r\n } catch {\r\n continue;\r\n }\r\n }\r\n\r\n // 4. Partial href attribute match (covers relative paths, query strings)\r\n const rawUrl = targetUrl.replace(/^\\//, \"\");\r\n for (const el of allLinks) {\r\n const href = el.getAttribute(\"href\") || el.getAttribute(\"data-href\") || \"\";\r\n if (href && (href === targetUrl || href === rawUrl || href === `/${rawUrl}`)) {\r\n return el;\r\n }\r\n }\r\n }\r\n\r\n // 5. Text-content match (for framework <Link> or <button> navigations)\r\n const urlSegments = targetUrl\r\n .replace(/^https?:\\/\\/[^/]+/, \"\")\r\n .replace(/[?#].*$/, \"\")\r\n .split(\"/\")\r\n .filter(Boolean);\r\n const lastSegment = urlSegments[urlSegments.length - 1] || \"\";\r\n\r\n if (lastSegment) {\r\n // Also search query param values (e.g., ?tab=interact → \"interact\")\r\n let searchTerms = [lastSegment];\r\n if (parsedTarget) {\r\n for (const [, value] of parsedTarget.searchParams.entries()) {\r\n if (value) searchTerms.push(value);\r\n }\r\n if (parsedTarget.hash) {\r\n searchTerms.push(parsedTarget.hash.replace(/^#/, \"\"));\r\n }\r\n }\r\n searchTerms = searchTerms.map((t) => t.toLowerCase());\r\n\r\n // Look across all clickable elements\r\n const clickables = Array.from(\r\n document.querySelectorAll(\r\n 'a, button, [role=\"link\"], [role=\"tab\"], [role=\"button\"], [data-tab], [onclick]',\r\n ),\r\n ) as HTMLElement[];\r\n\r\n for (const el of clickables) {\r\n const text = (el.textContent || \"\").trim().toLowerCase();\r\n const ariaLabel = (el.getAttribute(\"aria-label\") || \"\").toLowerCase();\r\n const dataTab = (el.getAttribute(\"data-tab\") || \"\").toLowerCase();\r\n for (const term of searchTerms) {\r\n if (\r\n text === term ||\r\n ariaLabel === term ||\r\n dataTab === term ||\r\n text.includes(term)\r\n ) {\r\n return el;\r\n }\r\n }\r\n }\r\n }\r\n\r\n return null;\r\n};\r\n\r\nconst executeNavigate = async (call: NavigateToolCall): Promise<boolean> => {\r\n try {\r\n const targetUrl = call.url;\r\n let resolvedUrl: string;\r\n try {\r\n resolvedUrl = new URL(targetUrl, window.location.href).href;\r\n } catch {\r\n resolvedUrl = targetUrl;\r\n }\r\n\r\n const matchingElement = findMatchingLinkForTarget(targetUrl);\r\n\r\n if (matchingElement) {\r\n console.log(\"AuticBot navigate: clicking element\", resolvedUrl, matchingElement.tagName);\r\n await slowScrollElementIntoView(matchingElement);\r\n\r\n const center = getElementCenter(matchingElement);\r\n await moveCursor(center.x, center.y);\r\n\r\n matchingElement.dispatchEvent(new MouseEvent(\"pointerdown\", { bubbles: true, view: window }));\r\n matchingElement.dispatchEvent(new MouseEvent(\"mousedown\", { bubbles: true, view: window }));\r\n matchingElement.dispatchEvent(new MouseEvent(\"pointerup\", { bubbles: true, view: window }));\r\n matchingElement.dispatchEvent(new MouseEvent(\"mouseup\", { bubbles: true, view: window }));\r\n matchingElement.click();\r\n\r\n return !isSamePageNavigation(resolvedUrl);\r\n }\r\n\r\n // 6. Fallback: direct browser navigation (query param, hash, or full URL)\r\n console.log(\"AuticBot navigate: no matching element found, using direct navigation\", resolvedUrl);\r\n\r\n // Hash-only navigation\r\n try {\r\n const parsed = new URL(resolvedUrl);\r\n if (\r\n parsed.origin === window.location.origin &&\r\n parsed.pathname === window.location.pathname &&\r\n parsed.hash\r\n ) {\r\n window.location.hash = parsed.hash;\r\n return false;\r\n }\r\n } catch { /* continue */ }\r\n\r\n // Query-param or same-origin navigation via History API\r\n try {\r\n const parsed = new URL(resolvedUrl);\r\n if (parsed.origin === window.location.origin) {\r\n // Use pushState + popstate to trigger SPA routers\r\n const newPath = parsed.pathname + parsed.search + parsed.hash;\r\n window.history.pushState({}, \"\", newPath);\r\n window.dispatchEvent(new PopStateEvent(\"popstate\", { state: {} }));\r\n return false;\r\n }\r\n } catch { /* continue */ }\r\n\r\n // Cross-origin: full page navigation\r\n window.location.href = resolvedUrl;\r\n return true;\r\n } catch (error) {\r\n console.warn(\"AuticBot navigate: error\", call.url, error);\r\n return false;\r\n }\r\n};\r\n\r\nconst executeGetPageContext = async () => {\r\n const context = getPageContext();\r\n console.info(\r\n `[Autic] getPageContext tool executed links=${context.links.length} interactables=${context.interactables.length} summary_len=${context.summary.length}`,\r\n );\r\n};\r\n\r\nexport const executeToolCalls = async (toolCalls: AgentToolCall[]) => {\r\n for (const toolCall of toolCalls) {\r\n if (toolCall.tool === \"interact\") {\r\n await executeInteract(toolCall);\r\n continue;\r\n }\r\n\r\n if (toolCall.tool === \"scroll\") {\r\n await executeScroll(toolCall);\r\n continue;\r\n }\r\n\r\n if (toolCall.tool === \"getPageContext\") {\r\n await executeGetPageContext();\r\n continue;\r\n }\r\n\r\n if (toolCall.tool === \"navigate\") {\r\n const terminalNavigation = await executeNavigate(toolCall);\r\n if (terminalNavigation) {\r\n break;\r\n }\r\n }\r\n }\r\n};\r\n\r\n// ── Agent-mode tool execution (returns results) ─────────────────────\r\n\r\nexport type ToolCallWithId = AgentToolCall & {\r\n call_id: string;\r\n};\r\n\r\nexport interface ToolCallResult {\r\n call_id: string;\r\n result: string;\r\n}\r\n\r\n/**\r\n * Execute a single tool call and return a result string.\r\n * Used by the agent loop to feed results back into the LLM.\r\n */\r\nexport const executeSingleToolCall = async (\r\n call: ToolCallWithId,\r\n): Promise<ToolCallResult> => {\r\n const callId = call.call_id;\r\n try {\r\n if (call.tool === \"interact\") {\r\n await executeInteract(call);\r\n return {\r\n call_id: callId,\r\n result: `Etkileşim başarılı: ${call.action}`,\r\n };\r\n }\r\n\r\n if (call.tool === \"scroll\") {\r\n await executeScroll(call);\r\n return {\r\n call_id: callId,\r\n result: \"Öğeye kaydırma başarılı.\",\r\n };\r\n }\r\n\r\n if (call.tool === \"getPageContext\") {\r\n const context = getPageContext();\r\n return {\r\n call_id: callId,\r\n result: context.summary,\r\n };\r\n }\r\n\r\n if (call.tool === \"navigate\") {\r\n await executeNavigate(call);\r\n // Wait for navigation / SPA routing to settle\r\n await new Promise((resolve) => setTimeout(resolve, 1500));\r\n const context = getPageContext();\r\n return {\r\n call_id: callId,\r\n result: `Navigasyon tamamlandı. Şu anki sayfa: ${window.location.href}\\nSayfa bağlamı: ${context.summary}`,\r\n };\r\n }\r\n\r\n return { call_id: callId, result: \"Bilinmeyen araç.\" };\r\n } catch (error) {\r\n const msg = error instanceof Error ? error.message : String(error);\r\n console.warn(`[Autic] Tool execution error: ${call.tool}`, error);\r\n return { call_id: callId, result: `Hata: ${msg}` };\r\n }\r\n};\r\n","import {\r\n savePendingAgentResume,\r\n clearPendingAgentResume,\r\n hideAgentCursor,\r\n type PendingAgentResume,\r\n} from \"../agent/tools\";\r\n\r\nexport type ChatRole = \"system\" | \"user\" | \"assistant\";\r\n\r\nexport interface ChatMessage {\r\n role: ChatRole;\r\n content: string;\r\n}\r\n\r\ninterface ApiErrorBody {\r\n detail?: string;\r\n error?: string;\r\n message?: string;\r\n}\r\n\r\ninterface SseEventPayload {\r\n type?: string;\r\n session_id?: string;\r\n user_text?: string;\r\n assistant_text?: string;\r\n delta?: string;\r\n audio?: string;\r\n format?: string;\r\n mime_type?: string;\r\n sample_rate?: number;\r\n error?: string;\r\n}\r\n\r\ninterface TtsWsEventPayload {\r\n type?: string;\r\n request_id?: string;\r\n seq?: number;\r\n audio?: string;\r\n format?: string;\r\n mime_type?: string;\r\n sample_rate?: number;\r\n error?: string;\r\n retryable?: boolean;\r\n last_seq?: number;\r\n}\r\n\r\ninterface SttWsEventPayload {\r\n type?: string;\r\n session_id?: string;\r\n seq?: number;\r\n text?: string;\r\n error?: string;\r\n retryable?: boolean;\r\n}\r\n\r\nexport type AudioStreamState = \"rendering\" | \"playing\" | \"done\" | \"fallback\";\r\nexport const TTS_WS_RETRY_DELAYS_MS = [250, 750, 1500];\r\n\r\nconst BULUT_AUDIO_STOP_EVENT = \"bulut:audio-stop\";\r\nconst activeAudioElements = new Set<HTMLAudioElement>();\r\nlet audioPlaybackGeneration = 0;\r\n\r\nconst normalizeBaseUrl = (baseUrl: string): string => {\r\n const trimmed = baseUrl.trim().replace(/\\/+$/, \"\");\r\n if (/^https?:\\/\\//i.test(trimmed)) {\r\n return trimmed;\r\n }\r\n // Treat host-only values like \"api.bulut.lu\" as HTTPS absolute URLs.\r\n return `https://${trimmed}`;\r\n};\r\nconst toWebSocketUrl = (baseUrl: string, path: string): string => {\r\n const normalized = normalizeBaseUrl(baseUrl);\r\n const url = new URL(normalized);\r\n url.protocol = url.protocol === \"https:\" ? \"wss:\" : \"ws:\";\r\n url.pathname = `${url.pathname.replace(/\\/$/, \"\")}${path}`;\r\n url.search = \"\";\r\n url.hash = \"\";\r\n return url.toString();\r\n};\r\n\r\nconst createRequestId = (): string => {\r\n if (typeof crypto !== \"undefined\" && \"randomUUID\" in crypto) {\r\n return crypto.randomUUID();\r\n }\r\n return `tts-${Date.now()}-${Math.random().toString(16).slice(2)}`;\r\n};\r\n\r\nexport const parseTtsWsEventPayload = (\r\n value: unknown,\r\n): TtsWsEventPayload | null => {\r\n try {\r\n if (typeof value !== \"string\") {\r\n return null;\r\n }\r\n return JSON.parse(value) as TtsWsEventPayload;\r\n } catch {\r\n return null;\r\n }\r\n};\r\n\r\nexport const parseSttWsEventPayload = (\r\n value: unknown,\r\n): SttWsEventPayload | null => {\r\n try {\r\n if (typeof value !== \"string\") {\r\n return null;\r\n }\r\n return JSON.parse(value) as SttWsEventPayload;\r\n } catch {\r\n return null;\r\n }\r\n};\r\n\r\nexport const shouldAcceptAudioSeq = (\r\n incomingSeq: number,\r\n highestSeqSeen: number,\r\n): boolean => incomingSeq > highestSeqSeen;\r\n\r\nexport const shouldFallbackToSse = (error: unknown): boolean => {\r\n if (typeof error === \"object\" && error !== null && \"retryable\" in error) {\r\n return Boolean((error as { retryable?: boolean }).retryable);\r\n }\r\n return true;\r\n};\r\n\r\nconst parseErrorBody = async (response: Response): Promise<string> => {\r\n try {\r\n const data = (await response.json()) as ApiErrorBody;\r\n const detail = data.detail;\r\n if (typeof detail === \"string\") return detail;\r\n if (detail && typeof detail === \"object\") return JSON.stringify(detail);\r\n return data.error || data.message || response.statusText;\r\n } catch {\r\n return response.statusText;\r\n }\r\n};\r\n\r\nconst sleep = (ms: number): Promise<void> =>\r\n new Promise((resolve) => {\r\n setTimeout(resolve, ms);\r\n });\r\n\r\nconst registerActiveAudioElement = (audioElement: HTMLAudioElement): void => {\r\n activeAudioElements.add(audioElement);\r\n};\r\n\r\nconst unregisterActiveAudioElement = (audioElement: HTMLAudioElement): void => {\r\n activeAudioElements.delete(audioElement);\r\n};\r\n\r\nconst wasPlaybackStoppedAfter = (generationAtStart: number): boolean =>\r\n audioPlaybackGeneration !== generationAtStart;\r\n\r\nexport const getAudioPlaybackGeneration = (): number => audioPlaybackGeneration;\r\n\r\nexport const stopActiveAudioPlayback = (): void => {\r\n audioPlaybackGeneration += 1;\r\n const active = Array.from(activeAudioElements);\r\n for (const audioElement of active) {\r\n try {\r\n audioElement.dispatchEvent(new Event(BULUT_AUDIO_STOP_EVENT));\r\n audioElement.pause();\r\n audioElement.removeAttribute(\"src\");\r\n audioElement.load();\r\n } catch {\r\n // Ignore playback stop errors.\r\n }\r\n }\r\n};\r\n\r\nexport const base64ToUint8Array = (base64: string): Uint8Array<ArrayBuffer> => {\r\n // Strip potential data URI prefix if present\r\n const cleanBase64 = base64.replace(/^data:audio\\/\\w+;base64,/, \"\");\r\n const binaryString = atob(cleanBase64);\r\n const bytes = new Uint8Array(binaryString.length) as Uint8Array<ArrayBuffer>;\r\n for (let i = 0; i < binaryString.length; i += 1) {\r\n bytes[i] = binaryString.charCodeAt(i);\r\n }\r\n return bytes;\r\n};\r\n\r\nconst blobToBase64 = async (blob: Blob): Promise<string> => {\r\n const bytes = new Uint8Array(await blob.arrayBuffer());\r\n const chunkSize = 0x8000;\r\n let binary = \"\";\r\n for (let i = 0; i < bytes.length; i += chunkSize) {\r\n binary += String.fromCharCode(...bytes.subarray(i, i + chunkSize));\r\n }\r\n return btoa(binary);\r\n};\r\n\r\nconst createWavHeader = (\r\n length: number,\r\n sampleRate: number = 16000,\r\n): Uint8Array<ArrayBuffer> => {\r\n const buffer = new ArrayBuffer(44);\r\n const view = new DataView(buffer);\r\n const channels = 1;\r\n\r\n // RIFF chunk descriptor\r\n view.setUint32(0, 0x52494646, false); // \"RIFF\"\r\n view.setUint32(4, 36 + length, true); // file length - 8\r\n view.setUint32(8, 0x57415645, false); // \"WAVE\"\r\n\r\n // fmt sub-chunk\r\n view.setUint32(12, 0x666d7420, false); // \"fmt \"\r\n view.setUint32(16, 16, true); // Subchunk1Size (16 for PCM)\r\n view.setUint16(20, 1, true); // AudioFormat (1 for PCM)\r\n view.setUint16(22, channels, true); // NumChannels\r\n view.setUint32(24, sampleRate, true); // SampleRate\r\n view.setUint32(28, sampleRate * channels * 2, true); // ByteRate\r\n view.setUint16(32, channels * 2, true); // BlockAlign\r\n view.setUint16(34, 16, true); // BitsPerSample\r\n\r\n // data sub-chunk\r\n view.setUint32(36, 0x64617461, false); // \"data\"\r\n view.setUint32(40, length, true); // Subchunk2Size\r\n\r\n return new Uint8Array(buffer) as Uint8Array<ArrayBuffer>;\r\n};\r\nconst waitForPlaybackEnd = async (\r\n audioElement: HTMLAudioElement,\r\n): Promise<void> => {\r\n if (audioElement.ended) {\r\n return;\r\n }\r\n\r\n await new Promise<void>((resolve, reject) => {\r\n const watchdog = window.setInterval(() => {\r\n if (!audioElement.ended) {\r\n console.info(\"[Bulut] playback watchdog: still playing...\");\r\n }\r\n }, 30000);\r\n\r\n const onEnded = () => {\r\n cleanup();\r\n resolve();\r\n };\r\n\r\n const onError = () => {\r\n cleanup();\r\n reject(new Error(\"Ses oynatma hatası oluştu.\"));\r\n };\r\n\r\n const onForcedStop = () => {\r\n cleanup();\r\n resolve();\r\n };\r\n\r\n const cleanup = () => {\r\n window.clearInterval(watchdog);\r\n audioElement.removeEventListener(\"ended\", onEnded);\r\n audioElement.removeEventListener(\"error\", onError);\r\n audioElement.removeEventListener(BULUT_AUDIO_STOP_EVENT, onForcedStop);\r\n };\r\n\r\n audioElement.addEventListener(\"ended\", onEnded);\r\n audioElement.addEventListener(\"error\", onError);\r\n audioElement.addEventListener(BULUT_AUDIO_STOP_EVENT, onForcedStop);\r\n });\r\n};\r\n\r\n\r\n\r\nconst playBufferedAudio = async (\r\n chunks: Uint8Array<ArrayBuffer>[],\r\n mimeType: string,\r\n sampleRate: number = 16000,\r\n onAudioStateChange?: (state: AudioStreamState) => void,\r\n): Promise<void> => {\r\n const playbackGeneration = getAudioPlaybackGeneration();\r\n if (chunks.length === 0) {\r\n onAudioStateChange?.(\"done\");\r\n return;\r\n }\r\n\r\n if (wasPlaybackStoppedAfter(playbackGeneration)) {\r\n onAudioStateChange?.(\"done\");\r\n return;\r\n }\r\n\r\n // Debug info\r\n const totalBytes = chunks.reduce((acc, c) => acc + c.byteLength, 0);\r\n console.log(`[Bulut] Playing buffered audio: ${chunks.length} chunks, ${totalBytes} bytes, type=${mimeType}`);\r\n\r\n onAudioStateChange?.(\"fallback\");\r\n\r\n const blobParts: ArrayBuffer[] = chunks.map((chunk) => {\r\n const copied = new Uint8Array(chunk.byteLength) as Uint8Array<ArrayBuffer>;\r\n copied.set(chunk);\r\n return copied.buffer;\r\n });\r\n\r\n // Verify magic numbers and detect MIME type\r\n let detectedMime = mimeType;\r\n if (chunks.length > 0 && chunks[0].length >= 4) {\r\n const header = Array.from(chunks[0].slice(0, 4))\r\n .map(b => b.toString(16).padStart(2, '0').toUpperCase())\r\n .join(' ');\r\n console.log(`[Bulut] Audio header (hex): ${header}`);\r\n\r\n // Magic number detection\r\n if (header.startsWith(\"49 44 33\")) { // ID3\r\n detectedMime = \"audio/mpeg\";\r\n } else if (header.startsWith(\"FF F3\") || header.startsWith(\"FF F2\")) { // MP3 Sync\r\n detectedMime = \"audio/mpeg\";\r\n } else if (header.startsWith(\"52 49 46 46\")) { // RIFF (WAV)\r\n detectedMime = \"audio/wav\";\r\n } else if (header.startsWith(\"1A 45 DF A3\")) { // EBML (WebM)\r\n detectedMime = \"audio/webm\";\r\n }\r\n }\r\n\r\n // Ensure valid MIME type\r\n // Ensure valid MIME type or wrap raw PCM\r\n let safeMimeType = detectedMime && detectedMime.includes(\"/\") ? detectedMime : \"audio/mpeg\";\r\n let finalBlobParts: BlobPart[] = blobParts;\r\n\r\n if (mimeType === \"audio/pcm\") {\r\n // Wrap raw PCM in WAV container\r\n const totalLength = chunks.reduce((acc, c) => acc + c.byteLength, 0);\r\n const header = createWavHeader(totalLength, sampleRate);\r\n finalBlobParts = [header.buffer, ...blobParts];\r\n safeMimeType = \"audio/wav\";\r\n console.log(`[Bulut] Wrapped raw PCM in WAV (rate=${sampleRate})`);\r\n }\r\n\r\n console.log(`[Bulut] Creating blob with type: ${safeMimeType} (original: ${mimeType})`);\r\n const blob = new Blob(finalBlobParts, { type: safeMimeType });\r\n\r\n const audioElement = new Audio();\r\n const objectUrl = URL.createObjectURL(blob);\r\n\r\n try {\r\n registerActiveAudioElement(audioElement);\r\n\r\n audioElement.preload = \"auto\";\r\n audioElement.autoplay = true;\r\n // Some browsers need this\r\n audioElement.setAttribute(\"playsinline\", \"true\");\r\n audioElement.src = objectUrl;\r\n\r\n if (wasPlaybackStoppedAfter(playbackGeneration)) {\r\n onAudioStateChange?.(\"done\");\r\n return;\r\n }\r\n\r\n await audioElement.play();\r\n onAudioStateChange?.(\"playing\");\r\n await waitForPlaybackEnd(audioElement);\r\n onAudioStateChange?.(\"done\");\r\n } catch (err) {\r\n console.error(`[Bulut] Playback failed: ${err}`, { mimeType: safeMimeType, size: blob.size });\r\n onAudioStateChange?.(\"done\"); // Signal done to unblock UI even on error\r\n throw err;\r\n } finally {\r\n unregisterActiveAudioElement(audioElement);\r\n audioElement.pause();\r\n audioElement.removeAttribute(\"src\");\r\n audioElement.load();\r\n URL.revokeObjectURL(objectUrl);\r\n }\r\n};\r\n\r\nexport interface StreamController {\r\n stop: () => void;\r\n done: Promise<void>;\r\n}\r\n\r\nexport const parseSseEventPayload = (eventBlock: string): SseEventPayload | null => {\r\n const dataLines = eventBlock\r\n .split(/\\r?\\n/)\r\n .map((line) => line.trim())\r\n .filter((line) => line.startsWith(\"data:\"))\r\n .map((line) => line.slice(5).trimStart());\r\n\r\n if (dataLines.length === 0) {\r\n return null;\r\n }\r\n\r\n const dataStr = dataLines.join(\"\\n\");\r\n if (dataStr === \"[DONE]\") {\r\n return { type: \"done\" };\r\n }\r\n\r\n try {\r\n return JSON.parse(dataStr) as SseEventPayload;\r\n } catch (error) {\r\n console.warn(\"Error parsing SSE chunk:\", error);\r\n return null;\r\n }\r\n};\r\n\r\nexport const isAudioSsePayload = (\r\n payload: SseEventPayload,\r\n): payload is SseEventPayload & { audio: string } =>\r\n typeof payload.audio === \"string\" &&\r\n (payload.type === undefined || payload.type === \"audio\");\r\n\r\n// ── Separated Endpoint Helpers ──────────────────────────────────────\r\n\r\nexport async function transcribeAudio(\r\n baseUrl: string,\r\n file: File,\r\n projectId: string,\r\n sessionId: string | null,\r\n language: string,\r\n onRequestSent?: () => void,\r\n): Promise<{ text: string; session_id: string }> {\r\n const url = `${normalizeBaseUrl(baseUrl)}/chat/stt`;\r\n const formData = new FormData();\r\n formData.append(\"file\", file);\r\n formData.append(\"project_id\", projectId);\r\n if (sessionId) formData.append(\"session_id\", sessionId);\r\n formData.append(\"language\", language);\r\n\r\n const responsePromise = fetch(url, { method: \"POST\", body: formData });\r\n onRequestSent?.();\r\n const response = await responsePromise;\r\n if (!response.ok) {\r\n throw new Error(await parseErrorBody(response));\r\n }\r\n return response.json();\r\n}\r\n\r\nexport interface SttWsResult {\r\n text: string;\r\n session_id: string;\r\n}\r\n\r\nexport interface SttWsEvents {\r\n onRequestSent?: () => void;\r\n onSessionId?: (sessionId: string) => void;\r\n onPartial?: (payload: { seq: number; text: string }) => void;\r\n}\r\n\r\nexport interface SttWsController {\r\n pushChunk: (chunk: Blob) => Promise<void>;\r\n stop: () => Promise<SttWsResult>;\r\n cancel: () => void;\r\n}\r\n\r\nexport const startSttWebSocketStream = (\r\n baseUrl: string,\r\n config: {\r\n projectId: string;\r\n sessionId: string | null;\r\n language?: string;\r\n mimeType?: string;\r\n },\r\n events: SttWsEvents = {},\r\n): SttWsController => {\r\n const wsUrl = toWebSocketUrl(baseUrl, \"/chat/stt/ws\");\r\n console.info(\"[Bulut] STT WS connecting to\", wsUrl);\r\n const socket = new WebSocket(wsUrl);\r\n let seq = 0;\r\n let finalText = \"\";\r\n let finalSessionId = config.sessionId || \"\";\r\n let stopped = false;\r\n let settled = false;\r\n // All chunk sends and the final stop are chained through sendQueue\r\n // so the \"stop\" message always follows all enqueued chunks.\r\n let sendQueue: Promise<void> = Promise.resolve();\r\n\r\n let resolveStart: (() => void) | null = null;\r\n let rejectStart: ((error: Error & { retryable?: boolean }) => void) | null = null;\r\n const startPromise = new Promise<void>((resolve, reject) => {\r\n resolveStart = resolve;\r\n rejectStart = reject;\r\n });\r\n\r\n let resolveDone: ((result: SttWsResult) => void) | null = null;\r\n let rejectDone: ((error: Error & { retryable?: boolean }) => void) | null = null;\r\n const donePromise = new Promise<SttWsResult>((resolve, reject) => {\r\n resolveDone = resolve;\r\n rejectDone = reject;\r\n });\r\n\r\n const rejectAll = (error: Error & { retryable?: boolean }) => {\r\n if (settled) return;\r\n settled = true;\r\n console.warn(\"[Bulut] STT WS rejected:\", error.message);\r\n rejectStart?.(error);\r\n rejectDone?.(error);\r\n };\r\n\r\n const resolveDoneIfPossible = () => {\r\n if (settled) return;\r\n if (!finalText.trim() || !finalSessionId) return;\r\n settled = true;\r\n resolveStart?.();\r\n resolveDone?.({\r\n text: finalText.trim(),\r\n session_id: finalSessionId,\r\n });\r\n };\r\n\r\n socket.onopen = () => {\r\n console.info(\"[Bulut] STT WS connected, sending start\");\r\n events.onRequestSent?.();\r\n socket.send(\r\n JSON.stringify({\r\n type: \"start\",\r\n project_id: config.projectId,\r\n session_id: config.sessionId || undefined,\r\n language: config.language || \"tr\",\r\n mime_type: config.mimeType || \"audio/webm\",\r\n }),\r\n );\r\n };\r\n\r\n socket.onmessage = (event) => {\r\n const payload = parseSttWsEventPayload(String(event.data));\r\n if (!payload) return;\r\n\r\n if (payload.type === \"start_ack\" && typeof payload.session_id === \"string\") {\r\n console.info(\"[Bulut] STT WS start_ack received, session:\", payload.session_id);\r\n finalSessionId = payload.session_id;\r\n events.onSessionId?.(payload.session_id);\r\n resolveStart?.();\r\n return;\r\n }\r\n\r\n if (payload.type === \"partial\" && typeof payload.text === \"string\") {\r\n events.onPartial?.({\r\n seq: typeof payload.seq === \"number\" ? payload.seq : 0,\r\n text: payload.text,\r\n });\r\n return;\r\n }\r\n\r\n if (payload.type === \"final\" && typeof payload.text === \"string\") {\r\n console.info(\"[Bulut] STT WS final text received:\", payload.text.slice(0, 80));\r\n finalText = payload.text;\r\n if (typeof payload.session_id === \"string\") {\r\n finalSessionId = payload.session_id;\r\n events.onSessionId?.(payload.session_id);\r\n }\r\n return;\r\n }\r\n\r\n if (payload.type === \"done\") {\r\n console.info(\"[Bulut] STT WS done\");\r\n resolveDoneIfPossible();\r\n socket.close();\r\n return;\r\n }\r\n\r\n if (payload.type === \"error\") {\r\n console.error(\"[Bulut] STT WS server error:\", payload.error);\r\n const err = buildError(payload.error || \"stt_ws_error\", payload.retryable !== false);\r\n rejectAll(err);\r\n socket.close();\r\n }\r\n };\r\n\r\n socket.onerror = (ev) => {\r\n console.error(\"[Bulut] STT WS transport error\", ev);\r\n rejectAll(buildError(\"stt_ws_transport_error\", true));\r\n };\r\n\r\n socket.onclose = (ev) => {\r\n console.info(\"[Bulut] STT WS closed code=%d reason=%s\", ev.code, ev.reason);\r\n if (settled) return;\r\n if (finalText && finalSessionId) {\r\n resolveDoneIfPossible();\r\n return;\r\n }\r\n rejectAll(buildError(\"stt_ws_closed_before_done\", true));\r\n };\r\n\r\n return {\r\n pushChunk: (chunk: Blob): Promise<void> => {\r\n if (stopped || chunk.size === 0) return Promise.resolve();\r\n // Chain the entire operation (wait for connection, base64-encode,\r\n // send) into sendQueue so that a later stop() is guaranteed to\r\n // follow all previously-enqueued chunks.\r\n sendQueue = sendQueue.then(async () => {\r\n if (stopped) return;\r\n await startPromise;\r\n if (stopped) return;\r\n const audio = await blobToBase64(chunk);\r\n seq += 1;\r\n if (stopped || socket.readyState !== WebSocket.OPEN) return;\r\n console.debug(\"[Bulut] STT WS sending chunk seq=%d size=%d\", seq, chunk.size);\r\n socket.send(JSON.stringify({ type: \"chunk\", seq, audio }));\r\n });\r\n return sendQueue;\r\n },\r\n stop: (): Promise<SttWsResult> => {\r\n console.info(\"[Bulut] STT WS stop requested, draining %d pending chunks\", seq);\r\n // Chain after all pending pushChunk operations so the server\r\n // always receives every chunk before the stop message.\r\n sendQueue = sendQueue.then(async () => {\r\n await startPromise;\r\n if (stopped) return;\r\n if (socket.readyState === WebSocket.OPEN) {\r\n console.info(\"[Bulut] STT WS sending stop after seq=%d\", seq);\r\n socket.send(JSON.stringify({ type: \"stop\" }));\r\n }\r\n });\r\n return donePromise;\r\n },\r\n cancel: () => {\r\n stopped = true;\r\n try {\r\n socket.close();\r\n } catch {\r\n // no-op\r\n }\r\n },\r\n };\r\n};\r\n\r\ninterface TtsCollectResult {\r\n chunks: Uint8Array<ArrayBuffer>[];\r\n mimeType: string;\r\n sampleRate: number;\r\n}\r\n\r\nconst buildError = (message: string, retryable: boolean = true): Error & { retryable: boolean } => {\r\n const error = new Error(message) as Error & { retryable: boolean };\r\n error.retryable = retryable;\r\n return error;\r\n};\r\n\r\nconst collectTtsViaSse = async (\r\n baseUrl: string,\r\n assistantText: string,\r\n voice: string,\r\n accessibilityMode: boolean,\r\n isStopped: () => boolean,\r\n setReader: (reader: ReadableStreamDefaultReader<Uint8Array> | undefined) => void,\r\n): Promise<TtsCollectResult> => {\r\n const ttsFormData = new FormData();\r\n ttsFormData.append(\"text\", assistantText);\r\n ttsFormData.append(\"voice\", voice);\r\n ttsFormData.append(\"accessibility_mode\", String(accessibilityMode));\r\n\r\n const ttsResponse = await fetch(`${normalizeBaseUrl(baseUrl)}/chat/tts`, {\r\n method: \"POST\",\r\n body: ttsFormData,\r\n });\r\n\r\n if (!ttsResponse.ok) {\r\n throw buildError(await parseErrorBody(ttsResponse), false);\r\n }\r\n\r\n const reader = ttsResponse.body?.getReader();\r\n if (!reader) {\r\n throw buildError(\"TTS response body is not readable\", false);\r\n }\r\n\r\n setReader(reader);\r\n\r\n const chunks: Uint8Array<ArrayBuffer>[] = [];\r\n let mimeType = \"audio/mpeg\";\r\n let sampleRate = 16000;\r\n const decoder = new TextDecoder();\r\n let buffer = \"\";\r\n\r\n while (true) {\r\n if (isStopped()) {\r\n break;\r\n }\r\n\r\n const { done, value } = await reader.read();\r\n if (done) {\r\n break;\r\n }\r\n\r\n buffer += decoder.decode(value, { stream: true });\r\n const blocks = buffer.split(/\\r?\\n\\r?\\n/);\r\n buffer = blocks.pop() || \"\";\r\n\r\n for (const block of blocks) {\r\n const payload = parseSseEventPayload(block);\r\n if (!payload) {\r\n continue;\r\n }\r\n\r\n if (isAudioSsePayload(payload)) {\r\n const format = payload.format || \"mp3\";\r\n mimeType = payload.mime_type || (format === \"webm\" ? \"audio/webm\" : \"audio/mpeg\");\r\n chunks.push(base64ToUint8Array(payload.audio));\r\n if (payload.sample_rate) {\r\n sampleRate = payload.sample_rate;\r\n }\r\n }\r\n }\r\n }\r\n\r\n reader.releaseLock();\r\n setReader(undefined);\r\n\r\n return { chunks, mimeType, sampleRate };\r\n};\r\n\r\nconst collectTtsViaWebSocket = async (\r\n baseUrl: string,\r\n assistantText: string,\r\n voice: string,\r\n accessibilityMode: boolean,\r\n isStopped: () => boolean,\r\n setSocket: (socket: WebSocket | null) => void,\r\n): Promise<TtsCollectResult> => {\r\n const wsUrl = toWebSocketUrl(baseUrl, \"/chat/tts/ws\");\r\n const requestId = createRequestId();\r\n const chunks: Uint8Array<ArrayBuffer>[] = [];\r\n let mimeType = \"audio/mpeg\";\r\n let sampleRate = 16000;\r\n let highestSeqSeen = 0;\r\n\r\n const connectOnce = (): Promise<void> =>\r\n new Promise((resolve, reject) => {\r\n if (isStopped()) {\r\n reject(buildError(\"stream_stopped\", false));\r\n return;\r\n }\r\n\r\n let done = false;\r\n let finalError: (Error & { retryable?: boolean }) | null = null;\r\n const socket = new WebSocket(wsUrl);\r\n setSocket(socket);\r\n\r\n const finalize = (\r\n mode: \"resolve\" | \"reject\",\r\n error?: Error & { retryable?: boolean },\r\n ) => {\r\n socket.onopen = null;\r\n socket.onmessage = null;\r\n socket.onerror = null;\r\n socket.onclose = null;\r\n setSocket(null);\r\n if (mode === \"resolve\") {\r\n resolve();\r\n return;\r\n }\r\n reject(error || buildError(\"tts_ws_closed\", true));\r\n };\r\n\r\n socket.onopen = () => {\r\n console.info(\r\n `[Bulut] TTS WS connected request_id=${requestId} resume_seq=${highestSeqSeen}`,\r\n );\r\n socket.send(\r\n JSON.stringify({\r\n type: \"start\",\r\n request_id: requestId,\r\n text: assistantText,\r\n voice,\r\n accessibility_mode: accessibilityMode,\r\n last_seq: highestSeqSeen,\r\n }),\r\n );\r\n };\r\n\r\n socket.onmessage = (event) => {\r\n const payload = parseTtsWsEventPayload(String(event.data));\r\n if (!payload) {\r\n console.warn(\"[Bulut] TTS WS invalid JSON payload\");\r\n return;\r\n }\r\n\r\n if (payload.type === \"audio\" && typeof payload.audio === \"string\") {\r\n const seq = typeof payload.seq === \"number\" ? payload.seq : 0;\r\n if (shouldAcceptAudioSeq(seq, highestSeqSeen)) {\r\n chunks.push(base64ToUint8Array(payload.audio));\r\n highestSeqSeen = seq;\r\n if (payload.mime_type) {\r\n mimeType = payload.mime_type;\r\n }\r\n if (typeof payload.sample_rate === \"number\") {\r\n sampleRate = payload.sample_rate;\r\n }\r\n } else {\r\n console.info(\r\n `[Bulut] TTS WS duplicate chunk ignored request_id=${requestId} seq=${seq} seen=${highestSeqSeen}`,\r\n );\r\n }\r\n\r\n if (socket.readyState === WebSocket.OPEN) {\r\n socket.send(\r\n JSON.stringify({\r\n type: \"ack\",\r\n request_id: requestId,\r\n last_seq: highestSeqSeen,\r\n }),\r\n );\r\n }\r\n return;\r\n }\r\n\r\n if (payload.type === \"done\") {\r\n const streamLastSeq =\r\n typeof payload.last_seq === \"number\" ? payload.last_seq : highestSeqSeen;\r\n if (streamLastSeq > highestSeqSeen) {\r\n finalError = buildError(\"tts_ws_sequence_gap\", true);\r\n done = false;\r\n socket.close();\r\n return;\r\n }\r\n done = true;\r\n socket.close();\r\n return;\r\n }\r\n\r\n if (payload.type === \"error\") {\r\n finalError = buildError(payload.error || \"tts_ws_error\", payload.retryable !== false);\r\n done = false;\r\n socket.close();\r\n }\r\n };\r\n\r\n socket.onerror = () => {\r\n if (!finalError) {\r\n finalError = buildError(\"tts_ws_transport_error\", true);\r\n }\r\n };\r\n\r\n socket.onclose = () => {\r\n if (isStopped()) {\r\n finalize(\"reject\", buildError(\"stream_stopped\", false));\r\n return;\r\n }\r\n if (done) {\r\n finalize(\"resolve\");\r\n return;\r\n }\r\n finalize(\"reject\", finalError || buildError(\"tts_ws_closed_before_done\", true));\r\n };\r\n });\r\n\r\n for (let attempt = 0; attempt <= TTS_WS_RETRY_DELAYS_MS.length; attempt += 1) {\r\n if (attempt > 0) {\r\n const delay = TTS_WS_RETRY_DELAYS_MS[attempt - 1];\r\n console.warn(\r\n `[Bulut] TTS WS retry attempt=${attempt} delay_ms=${delay} last_seq=${highestSeqSeen}`,\r\n );\r\n await sleep(delay);\r\n }\r\n\r\n try {\r\n await connectOnce();\r\n return { chunks, mimeType, sampleRate };\r\n } catch (error) {\r\n const retryable =\r\n shouldFallbackToSse(error);\r\n const message = error instanceof Error ? error.message : String(error);\r\n console.warn(\r\n `[Bulut] TTS WS attempt failed attempt=${attempt} retryable=${retryable} error=${message}`,\r\n );\r\n if (!retryable || attempt === TTS_WS_RETRY_DELAYS_MS.length) {\r\n throw error;\r\n }\r\n }\r\n }\r\n\r\n throw buildError(\"tts_ws_exhausted\", true);\r\n};\r\n\r\n// ── Agent-mode Types ────────────────────────────────────────────────\r\n\r\nexport interface AgentToolCallInfo {\r\n call_id: string;\r\n tool: string;\r\n args: Record<string, unknown>;\r\n}\r\n\r\nexport interface AgentVoiceChatEvents {\r\n onSttRequestSent?: () => void;\r\n onTranscription?: (data: {\r\n session_id: string;\r\n user_text: string;\r\n }) => void;\r\n onAssistantDelta?: (delta: string) => void;\r\n onAssistantDone?: (assistantText: string, expectsReply?: boolean) => void;\r\n onAudioStateChange?: (state: AudioStreamState) => void;\r\n onError?: (error: string) => void;\r\n /** Called when the agent requests tool execution on the frontend. */\r\n onToolCalls?: (calls: AgentToolCallInfo[]) => void;\r\n /** Called after each tool has been executed with the result. */\r\n onToolResult?: (callId: string, toolName: string, result: string) => void;\r\n /** Called at the start of each agent iteration. */\r\n onIteration?: (iteration: number, maxIterations: number) => void;\r\n /** Called when the backend confirms / creates a session ID. */\r\n onSessionId?: (sessionId: string) => void;\r\n /**\r\n * Called when the agent emits a reply text followed by tool calls.\r\n * This text is spoken aloud before the tools run so the user hears\r\n * every piece of the conversation, not just the final reply.\r\n */\r\n onIntermediateReply?: (text: string) => void;\r\n}\r\n\r\n/**\r\n * Standalone TTS helper: synthesize + play a text snippet.\r\n * Uses WebSocket TTS with SSE fallback, same as the main stream functions.\r\n */\r\nexport const speakText = async (\r\n baseUrl: string,\r\n text: string,\r\n voice: string,\r\n accessibilityMode: boolean,\r\n onAudioStateChange?: (state: AudioStreamState) => void,\r\n): Promise<void> => {\r\n const trimmed = text.trim();\r\n if (!trimmed) return;\r\n const playbackGeneration = getAudioPlaybackGeneration();\r\n\r\n console.info(`[Bulut] speakText start (${trimmed.length} chars)`);\r\n onAudioStateChange?.(\"rendering\");\r\n let ttsResult: TtsCollectResult;\r\n\r\n const neverStopped = () => false;\r\n\r\n try {\r\n ttsResult = await collectTtsViaWebSocket(\r\n baseUrl, trimmed, voice, accessibilityMode,\r\n neverStopped,\r\n () => {},\r\n );\r\n } catch {\r\n ttsResult = await collectTtsViaSse(\r\n baseUrl, trimmed, voice, accessibilityMode,\r\n neverStopped,\r\n () => {},\r\n );\r\n }\r\n\r\n if (wasPlaybackStoppedAfter(playbackGeneration)) {\r\n onAudioStateChange?.(\"done\");\r\n return;\r\n }\r\n\r\n if (ttsResult.chunks.length > 0) {\r\n await playBufferedAudio(\r\n ttsResult.chunks, ttsResult.mimeType, ttsResult.sampleRate,\r\n onAudioStateChange,\r\n );\r\n } else {\r\n onAudioStateChange?.(\"done\");\r\n }\r\n};\r\n\r\n// ── Agent Voice Chat Stream (STT → Agent WS → TTS) ─────────────────\r\n\r\nexport const agentVoiceChatStream = (\r\n baseUrl: string,\r\n audioFile: File,\r\n projectId: string,\r\n sessionId: string | null,\r\n config: {\r\n model: string;\r\n voice: string;\r\n pageContext?: string;\r\n accessibilityMode?: boolean;\r\n },\r\n events: AgentVoiceChatEvents,\r\n executeTool: (call: AgentToolCallInfo) => Promise<{ call_id: string; result: string }>,\r\n): StreamController => {\r\n let isStopped = false;\r\n let activeSocket: WebSocket | null = null;\r\n let activeReader: ReadableStreamDefaultReader<Uint8Array> | undefined;\r\n let errorEmitted = false;\r\n\r\n const donePromise = new Promise<void>(async (resolve, reject) => {\r\n try {\r\n // ── 1. STT ────────────────────────────────────────────────\r\n if (isStopped) return resolve();\r\n const sttResult = await transcribeAudio(\r\n baseUrl,\r\n audioFile,\r\n projectId,\r\n sessionId,\r\n \"tr\",\r\n events.onSttRequestSent,\r\n );\r\n\r\n const currentSessionId = sttResult.session_id;\r\n let effectiveSessionId = currentSessionId;\r\n const userText = sttResult.text;\r\n\r\n events.onTranscription?.({\r\n session_id: currentSessionId,\r\n user_text: userText,\r\n });\r\n\r\n if (isStopped) return resolve();\r\n\r\n // ── 2. Agent loop via WebSocket ───────────────────────────\r\n const assistantText = await new Promise<string>((agentResolve, agentReject) => {\r\n if (isStopped) { agentResolve(\"\"); return; }\r\n\r\n const wsUrl = toWebSocketUrl(baseUrl, \"/chat/agent/ws\");\r\n const socket = new WebSocket(wsUrl);\r\n activeSocket = socket;\r\n\r\n let finalReply = \"\";\r\n let resolved = false;\r\n let accumulatedDelta = \"\";\r\n\r\n const finish = (reply: string) => {\r\n if (resolved) return;\r\n resolved = true;\r\n agentResolve(reply);\r\n };\r\n\r\n const fail = (error: Error) => {\r\n if (resolved) return;\r\n resolved = true;\r\n agentReject(error);\r\n };\r\n\r\n socket.onopen = () => {\r\n console.info(\"[Bulut] Agent WS connected\");\r\n socket.send(JSON.stringify({\r\n type: \"start\",\r\n project_id: projectId,\r\n session_id: currentSessionId,\r\n user_text: userText,\r\n model: config.model,\r\n page_context: config.pageContext,\r\n accessibility_mode: config.accessibilityMode,\r\n }));\r\n };\r\n\r\n socket.onmessage = async (event) => {\r\n let data: Record<string, unknown>;\r\n try {\r\n data = JSON.parse(String(event.data));\r\n } catch {\r\n console.warn(\"[Bulut] Agent WS invalid JSON\");\r\n return;\r\n }\r\n\r\n const msgType = data.type as string;\r\n\r\n if (msgType === \"session\" && typeof data.session_id === \"string\") {\r\n effectiveSessionId = data.session_id as string;\r\n events.onSessionId?.(effectiveSessionId);\r\n return;\r\n }\r\n\r\n if (msgType === \"iteration\") {\r\n events.onIteration?.(\r\n data.iteration as number,\r\n data.max_iterations as number,\r\n );\r\n return;\r\n }\r\n\r\n if (msgType === \"reply_delta\" && typeof data.delta === \"string\") {\r\n accumulatedDelta += data.delta;\r\n events.onAssistantDelta?.(data.delta);\r\n return;\r\n }\r\n\r\n if (msgType === \"tool_calls\" && Array.isArray(data.calls)) {\r\n const calls = data.calls as AgentToolCallInfo[];\r\n\r\n // Speak accumulated text before running tools\r\n if (accumulatedDelta.trim()) {\r\n events.onIntermediateReply?.(accumulatedDelta.trim());\r\n }\r\n accumulatedDelta = \"\";\r\n\r\n events.onToolCalls?.(calls);\r\n\r\n const results: { call_id: string; result: string }[] = [];\r\n for (const call of calls) {\r\n // Save resume state before navigate in case of full-page reload\r\n const isNavigate = call.tool === \"navigate\";\r\n if (isNavigate) {\r\n savePendingAgentResume({\r\n sessionId: effectiveSessionId,\r\n projectId,\r\n model: config.model,\r\n voice: config.voice,\r\n accessibilityMode: Boolean(config.accessibilityMode),\r\n pendingToolCalls: calls.map((c) => ({\r\n call_id: c.call_id,\r\n tool: c.tool,\r\n args: c.args,\r\n })),\r\n completedResults: [...results],\r\n });\r\n }\r\n\r\n const result = await executeTool(call);\r\n\r\n // If we reach here, no full-page reload happened\r\n if (isNavigate) {\r\n clearPendingAgentResume();\r\n }\r\n\r\n events.onToolResult?.(call.call_id, call.tool, result.result);\r\n results.push(result);\r\n }\r\n\r\n if (socket.readyState === WebSocket.OPEN) {\r\n socket.send(JSON.stringify({\r\n type: \"tool_results\",\r\n results,\r\n }));\r\n }\r\n return;\r\n }\r\n\r\n if (msgType === \"agent_done\") {\r\n finalReply = (data.final_reply as string) || \"\";\r\n const replyExpectsReply = data.expects_reply !== false;\r\n events.onAssistantDone?.(finalReply, replyExpectsReply);\r\n if (typeof data.session_id === \"string\") {\r\n events.onSessionId?.(data.session_id as string);\r\n }\r\n finish(finalReply);\r\n return;\r\n }\r\n\r\n if (msgType === \"error\") {\r\n const errMsg = (data.error as string) || \"Agent error\";\r\n errorEmitted = true;\r\n events.onError?.(errMsg);\r\n fail(new Error(errMsg));\r\n return;\r\n }\r\n };\r\n\r\n socket.onerror = () => {\r\n console.error(\"[Bulut] Agent WS error\");\r\n errorEmitted = true;\r\n events.onError?.(\"Agent WebSocket connection error\");\r\n fail(new Error(\"Agent WebSocket connection error\"));\r\n };\r\n\r\n socket.onclose = () => {\r\n console.info(\"[Bulut] Agent WS closed\");\r\n finish(finalReply);\r\n };\r\n });\r\n\r\n activeSocket = null;\r\n\r\n // ── 3. TTS ────────────────────────────────────────────────\r\n if (isStopped || !assistantText) {\r\n if (!isStopped) {\r\n hideAgentCursor();\r\n }\r\n return resolve();\r\n }\r\n\r\n console.info(\r\n `[Bulut] TTS start mode=agent voice=${config.voice}`,\r\n );\r\n\r\n events.onAudioStateChange?.(\"rendering\");\r\n let ttsResult: TtsCollectResult;\r\n\r\n try {\r\n ttsResult = await collectTtsViaWebSocket(\r\n baseUrl,\r\n assistantText,\r\n config.voice,\r\n Boolean(config.accessibilityMode),\r\n () => isStopped,\r\n (socket) => { activeSocket = socket; },\r\n );\r\n } catch (wsError) {\r\n if (isStopped) return resolve();\r\n console.warn(\r\n `[Bulut] TTS WS failed, falling back to SSE: ${wsError instanceof Error ? wsError.message : String(wsError)}`,\r\n );\r\n ttsResult = await collectTtsViaSse(\r\n baseUrl,\r\n assistantText,\r\n config.voice,\r\n Boolean(config.accessibilityMode),\r\n () => isStopped,\r\n (reader) => { activeReader = reader; },\r\n );\r\n }\r\n\r\n if (!isStopped) {\r\n hideAgentCursor();\r\n }\r\n\r\n if (!isStopped && ttsResult.chunks.length > 0) {\r\n await playBufferedAudio(\r\n ttsResult.chunks,\r\n ttsResult.mimeType,\r\n ttsResult.sampleRate,\r\n events.onAudioStateChange,\r\n );\r\n } else {\r\n events.onAudioStateChange?.(\"done\");\r\n }\r\n\r\n resolve();\r\n } catch (err) {\r\n // Only emit onError if it hasn't been emitted already by the WS handler\r\n if (!errorEmitted) {\r\n const msg = err instanceof Error ? err.message : String(err);\r\n events.onError?.(msg);\r\n }\r\n reject(err);\r\n } finally {\r\n activeReader?.cancel().catch(() => { });\r\n if (activeSocket && activeSocket.readyState <= WebSocket.OPEN) {\r\n activeSocket.close();\r\n }\r\n activeSocket = null;\r\n }\r\n });\r\n\r\n return {\r\n stop: () => {\r\n isStopped = true;\r\n stopActiveAudioPlayback();\r\n if (activeReader) {\r\n activeReader.cancel().catch(() => { });\r\n }\r\n if (activeSocket && activeSocket.readyState <= WebSocket.OPEN) {\r\n activeSocket.close();\r\n }\r\n },\r\n done: donePromise,\r\n };\r\n};\r\n\r\n// ── Agent Text Chat Stream (no STT, Agent WS → TTS) ────────────────\r\n\r\nexport const agentTextChatStream = (\r\n baseUrl: string,\r\n userText: string,\r\n projectId: string,\r\n sessionId: string | null,\r\n config: {\r\n model: string;\r\n voice: string;\r\n pageContext?: string;\r\n accessibilityMode?: boolean;\r\n },\r\n events: AgentVoiceChatEvents,\r\n executeTool: (call: AgentToolCallInfo) => Promise<{ call_id: string; result: string }>,\r\n): StreamController => {\r\n let isStopped = false;\r\n let activeSocket: WebSocket | null = null;\r\n let activeReader: ReadableStreamDefaultReader<Uint8Array> | undefined;\r\n let errorEmitted = false;\r\n\r\n const donePromise = new Promise<void>(async (resolve, reject) => {\r\n try {\r\n if (isStopped) return resolve();\r\n\r\n // ── 1. Agent loop via WebSocket ───────────────────────────\r\n const assistantText = await new Promise<string>((agentResolve, agentReject) => {\r\n if (isStopped) { agentResolve(\"\"); return; }\r\n\r\n const wsUrl = toWebSocketUrl(baseUrl, \"/chat/agent/ws\");\r\n const socket = new WebSocket(wsUrl);\r\n activeSocket = socket;\r\n\r\n let finalReply = \"\";\r\n let resolved = false;\r\n let effectiveSessionId = sessionId || \"\";\r\n let accumulatedDelta = \"\";\r\n\r\n const finish = (reply: string) => {\r\n if (resolved) return;\r\n resolved = true;\r\n agentResolve(reply);\r\n };\r\n\r\n const fail = (error: Error) => {\r\n if (resolved) return;\r\n resolved = true;\r\n agentReject(error);\r\n };\r\n\r\n socket.onopen = () => {\r\n socket.send(JSON.stringify({\r\n type: \"start\",\r\n project_id: projectId,\r\n session_id: sessionId,\r\n user_text: userText,\r\n model: config.model,\r\n page_context: config.pageContext,\r\n accessibility_mode: config.accessibilityMode,\r\n }));\r\n };\r\n\r\n socket.onmessage = async (event) => {\r\n let data: Record<string, unknown>;\r\n try {\r\n data = JSON.parse(String(event.data));\r\n } catch { return; }\r\n\r\n const msgType = data.type as string;\r\n\r\n if (msgType === \"session\" && typeof data.session_id === \"string\") {\r\n effectiveSessionId = data.session_id as string;\r\n events.onSessionId?.(effectiveSessionId);\r\n return;\r\n }\r\n\r\n if (msgType === \"iteration\") {\r\n events.onIteration?.(\r\n data.iteration as number,\r\n data.max_iterations as number,\r\n );\r\n return;\r\n }\r\n\r\n if (msgType === \"reply_delta\" && typeof data.delta === \"string\") {\r\n accumulatedDelta += data.delta;\r\n events.onAssistantDelta?.(data.delta);\r\n return;\r\n }\r\n\r\n if (msgType === \"tool_calls\" && Array.isArray(data.calls)) {\r\n const calls = data.calls as AgentToolCallInfo[];\r\n\r\n // Speak accumulated text before running tools\r\n if (accumulatedDelta.trim()) {\r\n events.onIntermediateReply?.(accumulatedDelta.trim());\r\n }\r\n accumulatedDelta = \"\";\r\n\r\n events.onToolCalls?.(calls);\r\n\r\n const results: { call_id: string; result: string }[] = [];\r\n for (const call of calls) {\r\n const isNavigate = call.tool === \"navigate\";\r\n if (isNavigate) {\r\n savePendingAgentResume({\r\n sessionId: effectiveSessionId,\r\n projectId,\r\n model: config.model,\r\n voice: config.voice,\r\n accessibilityMode: Boolean(config.accessibilityMode),\r\n pendingToolCalls: calls.map((c) => ({\r\n call_id: c.call_id,\r\n tool: c.tool,\r\n args: c.args,\r\n })),\r\n completedResults: [...results],\r\n });\r\n }\r\n\r\n const result = await executeTool(call);\r\n\r\n if (isNavigate) {\r\n clearPendingAgentResume();\r\n }\r\n\r\n events.onToolResult?.(call.call_id, call.tool, result.result);\r\n results.push(result);\r\n }\r\n\r\n if (socket.readyState === WebSocket.OPEN) {\r\n socket.send(JSON.stringify({\r\n type: \"tool_results\",\r\n results,\r\n }));\r\n }\r\n return;\r\n }\r\n\r\n if (msgType === \"agent_done\") {\r\n finalReply = (data.final_reply as string) || \"\";\r\n const replyExpectsReply = data.expects_reply !== false;\r\n events.onAssistantDone?.(finalReply, replyExpectsReply);\r\n if (typeof data.session_id === \"string\") {\r\n events.onSessionId?.(data.session_id as string);\r\n }\r\n finish(finalReply);\r\n return;\r\n }\r\n\r\n if (msgType === \"error\") {\r\n const errMsg = (data.error as string) || \"Agent error\";\r\n errorEmitted = true;\r\n events.onError?.(errMsg);\r\n fail(new Error(errMsg));\r\n return;\r\n }\r\n };\r\n\r\n socket.onerror = () => {\r\n errorEmitted = true;\r\n events.onError?.(\"Agent WebSocket error\");\r\n fail(new Error(\"Agent WebSocket error\"));\r\n };\r\n socket.onclose = () => finish(finalReply);\r\n });\r\n\r\n activeSocket = null;\r\n\r\n // ── 2. TTS ────────────────────────────────────────────────\r\n if (isStopped || !assistantText) {\r\n if (!isStopped) {\r\n hideAgentCursor();\r\n }\r\n return resolve();\r\n }\r\n\r\n events.onAudioStateChange?.(\"rendering\");\r\n let ttsResult: TtsCollectResult;\r\n\r\n try {\r\n ttsResult = await collectTtsViaWebSocket(\r\n baseUrl, assistantText, config.voice, Boolean(config.accessibilityMode),\r\n () => isStopped,\r\n (socket) => { activeSocket = socket; },\r\n );\r\n } catch (wsError) {\r\n if (isStopped) return resolve();\r\n ttsResult = await collectTtsViaSse(\r\n baseUrl, assistantText, config.voice, Boolean(config.accessibilityMode),\r\n () => isStopped,\r\n (reader) => { activeReader = reader; },\r\n );\r\n }\r\n\r\n if (!isStopped) {\r\n hideAgentCursor();\r\n }\r\n\r\n if (!isStopped && ttsResult.chunks.length > 0) {\r\n await playBufferedAudio(\r\n ttsResult.chunks, ttsResult.mimeType, ttsResult.sampleRate,\r\n events.onAudioStateChange,\r\n );\r\n } else {\r\n events.onAudioStateChange?.(\"done\");\r\n }\r\n\r\n resolve();\r\n } catch (err) {\r\n if (!errorEmitted) {\r\n const msg = err instanceof Error ? err.message : String(err);\r\n events.onError?.(msg);\r\n }\r\n reject(err);\r\n } finally {\r\n activeReader?.cancel().catch(() => { });\r\n if (activeSocket && activeSocket.readyState <= WebSocket.OPEN) {\r\n activeSocket.close();\r\n }\r\n activeSocket = null;\r\n }\r\n });\r\n\r\n return {\r\n stop: () => {\r\n isStopped = true;\r\n stopActiveAudioPlayback();\r\n if (activeReader) activeReader.cancel().catch(() => { });\r\n if (activeSocket && activeSocket.readyState <= WebSocket.OPEN) {\r\n activeSocket.close();\r\n }\r\n },\r\n done: donePromise,\r\n };\r\n};\r\n\r\n// ── Agent Resume Stream (after page navigation reload) ──────────────\r\n//\r\n// When a navigate tool causes a full-page reload, the agent WS is lost.\r\n// This function opens a new WS with {type: \"resume\"}, sends the\r\n// completed tool results (including the navigate result with the new\r\n// page context), and continues the agent loop from where it left off.\r\n\r\nexport const agentResumeStream = (\r\n baseUrl: string,\r\n resumeState: PendingAgentResume,\r\n pageContext: string,\r\n events: AgentVoiceChatEvents,\r\n executeTool: (call: AgentToolCallInfo) => Promise<{ call_id: string; result: string }>,\r\n): StreamController => {\r\n let isStopped = false;\r\n let activeSocket: WebSocket | null = null;\r\n let activeReader: ReadableStreamDefaultReader<Uint8Array> | undefined;\r\n let errorEmitted = false;\r\n\r\n // Build tool results for the calls that were pending when the page reloaded.\r\n // Navigate results include the new page context; other tools that couldn't\r\n // execute get a descriptive skip message.\r\n const allResults = [...resumeState.completedResults];\r\n for (const tc of resumeState.pendingToolCalls) {\r\n if (allResults.some((r) => r.call_id === tc.call_id)) continue;\r\n if (tc.tool === \"navigate\") {\r\n allResults.push({\r\n call_id: tc.call_id,\r\n result: `Navigasyon tamamlandı. Şu anki sayfa: ${typeof window !== \"undefined\" ? window.location.href : \"\"}\\nSayfa bağlamı: ${pageContext}`,\r\n });\r\n } else {\r\n allResults.push({\r\n call_id: tc.call_id,\r\n result: \"Sayfa yeniden yüklendi, bu araç çalıştırılamadı.\",\r\n });\r\n }\r\n }\r\n\r\n const donePromise = new Promise<void>(async (resolve, reject) => {\r\n try {\r\n if (isStopped) return resolve();\r\n\r\n let effectiveSessionId = resumeState.sessionId;\r\n\r\n const assistantText = await new Promise<string>((agentResolve, agentReject) => {\r\n if (isStopped) { agentResolve(\"\"); return; }\r\n\r\n const wsUrl = toWebSocketUrl(baseUrl, \"/chat/agent/ws\");\r\n const socket = new WebSocket(wsUrl);\r\n activeSocket = socket;\r\n\r\n let finalReply = \"\";\r\n let resolved = false;\r\n let accumulatedDelta = \"\";\r\n\r\n const finish = (reply: string) => {\r\n if (resolved) return;\r\n resolved = true;\r\n agentResolve(reply);\r\n };\r\n\r\n const fail = (error: Error) => {\r\n if (resolved) return;\r\n resolved = true;\r\n agentReject(error);\r\n };\r\n\r\n socket.onopen = () => {\r\n console.info(\"[Bulut] Agent WS resume connected\");\r\n socket.send(JSON.stringify({\r\n type: \"resume\",\r\n project_id: resumeState.projectId,\r\n session_id: resumeState.sessionId,\r\n model: resumeState.model,\r\n page_context: pageContext,\r\n accessibility_mode: resumeState.accessibilityMode,\r\n pending_tool_calls: resumeState.pendingToolCalls,\r\n tool_results: allResults,\r\n }));\r\n };\r\n\r\n socket.onmessage = async (event) => {\r\n let data: Record<string, unknown>;\r\n try {\r\n data = JSON.parse(String(event.data));\r\n } catch { return; }\r\n\r\n const msgType = data.type as string;\r\n\r\n if (msgType === \"session\" && typeof data.session_id === \"string\") {\r\n effectiveSessionId = data.session_id as string;\r\n events.onSessionId?.(effectiveSessionId);\r\n return;\r\n }\r\n\r\n if (msgType === \"iteration\") {\r\n events.onIteration?.(\r\n data.iteration as number,\r\n data.max_iterations as number,\r\n );\r\n return;\r\n }\r\n\r\n if (msgType === \"reply_delta\" && typeof data.delta === \"string\") {\r\n accumulatedDelta += data.delta;\r\n events.onAssistantDelta?.(data.delta);\r\n return;\r\n }\r\n\r\n if (msgType === \"tool_calls\" && Array.isArray(data.calls)) {\r\n const calls = data.calls as AgentToolCallInfo[];\r\n\r\n // Speak accumulated text before running tools\r\n if (accumulatedDelta.trim()) {\r\n events.onIntermediateReply?.(accumulatedDelta.trim());\r\n }\r\n accumulatedDelta = \"\";\r\n\r\n events.onToolCalls?.(calls);\r\n\r\n const results: { call_id: string; result: string }[] = [];\r\n for (const call of calls) {\r\n const isNavigate = call.tool === \"navigate\";\r\n if (isNavigate) {\r\n savePendingAgentResume({\r\n sessionId: effectiveSessionId,\r\n projectId: resumeState.projectId,\r\n model: resumeState.model,\r\n voice: resumeState.voice,\r\n accessibilityMode: resumeState.accessibilityMode,\r\n pendingToolCalls: calls.map((c) => ({\r\n call_id: c.call_id,\r\n tool: c.tool,\r\n args: c.args,\r\n })),\r\n completedResults: [...results],\r\n });\r\n }\r\n\r\n const result = await executeTool(call);\r\n\r\n if (isNavigate) {\r\n clearPendingAgentResume();\r\n }\r\n\r\n events.onToolResult?.(call.call_id, call.tool, result.result);\r\n results.push(result);\r\n }\r\n\r\n if (socket.readyState === WebSocket.OPEN) {\r\n socket.send(JSON.stringify({ type: \"tool_results\", results }));\r\n }\r\n return;\r\n }\r\n\r\n if (msgType === \"agent_done\") {\r\n finalReply = (data.final_reply as string) || \"\";\r\n const replyExpectsReply = data.expects_reply !== false;\r\n events.onAssistantDone?.(finalReply, replyExpectsReply);\r\n if (typeof data.session_id === \"string\") {\r\n events.onSessionId?.(data.session_id as string);\r\n }\r\n finish(finalReply);\r\n return;\r\n }\r\n\r\n if (msgType === \"error\") {\r\n const errMsg = (data.error as string) || \"Agent error\";\r\n errorEmitted = true;\r\n events.onError?.(errMsg);\r\n fail(new Error(errMsg));\r\n return;\r\n }\r\n };\r\n\r\n socket.onerror = () => {\r\n errorEmitted = true;\r\n events.onError?.(\"Agent WebSocket error\");\r\n fail(new Error(\"Agent WebSocket error\"));\r\n };\r\n\r\n socket.onclose = () => finish(finalReply);\r\n });\r\n\r\n activeSocket = null;\r\n\r\n // ── TTS ────────────────────────────────────────────────\r\n if (isStopped || !assistantText) {\r\n if (!isStopped) {\r\n hideAgentCursor();\r\n }\r\n return resolve();\r\n }\r\n\r\n console.info(`[Bulut] TTS start mode=resume voice=${resumeState.voice}`);\r\n events.onAudioStateChange?.(\"rendering\");\r\n let ttsResult: TtsCollectResult;\r\n\r\n try {\r\n ttsResult = await collectTtsViaWebSocket(\r\n baseUrl, assistantText, resumeState.voice, Boolean(resumeState.accessibilityMode),\r\n () => isStopped,\r\n (socket) => { activeSocket = socket; },\r\n );\r\n } catch (wsError) {\r\n if (isStopped) return resolve();\r\n console.warn(\r\n `[Bulut] TTS WS failed, falling back to SSE: ${wsError instanceof Error ? wsError.message : String(wsError)}`,\r\n );\r\n ttsResult = await collectTtsViaSse(\r\n baseUrl, assistantText, resumeState.voice, Boolean(resumeState.accessibilityMode),\r\n () => isStopped,\r\n (reader) => { activeReader = reader; },\r\n );\r\n }\r\n\r\n if (!isStopped) {\r\n hideAgentCursor();\r\n }\r\n\r\n if (!isStopped && ttsResult.chunks.length > 0) {\r\n await playBufferedAudio(\r\n ttsResult.chunks, ttsResult.mimeType, ttsResult.sampleRate,\r\n events.onAudioStateChange,\r\n );\r\n } else {\r\n events.onAudioStateChange?.(\"done\");\r\n }\r\n\r\n resolve();\r\n } catch (err) {\r\n if (!errorEmitted) {\r\n const msg = err instanceof Error ? err.message : String(err);\r\n events.onError?.(msg);\r\n }\r\n reject(err);\r\n } finally {\r\n activeReader?.cancel().catch(() => { });\r\n if (activeSocket && activeSocket.readyState <= WebSocket.OPEN) {\r\n activeSocket.close();\r\n }\r\n activeSocket = null;\r\n }\r\n });\r\n\r\n return {\r\n stop: () => {\r\n isStopped = true;\r\n stopActiveAudioPlayback();\r\n if (activeReader) activeReader.cancel().catch(() => { });\r\n if (activeSocket && activeSocket.readyState <= WebSocket.OPEN) {\r\n activeSocket.close();\r\n }\r\n },\r\n done: donePromise,\r\n };\r\n};\r\n","import {\r\n completedSfxUrl,\r\n sentSfxUrl,\r\n thinkingSfxUrl,\r\n toolCallSfxUrl,\r\n} from \"../assets\";\r\n\r\nexport type SfxName = \"sent\" | \"thinking\" | \"toolCall\" | \"completed\";\r\n\r\nconst SFX_SOURCES: Record<SfxName, string> = {\r\n sent: sentSfxUrl,\r\n thinking: thinkingSfxUrl,\r\n toolCall: toolCallSfxUrl,\r\n completed: completedSfxUrl,\r\n};\r\n\r\nclass SfxManager {\r\n private queue: SfxName[] = [];\r\n private active = false;\r\n\r\n private playNow(name: SfxName): Promise<void> {\r\n return new Promise((resolve) => {\r\n if (typeof window === \"undefined\") {\r\n resolve();\r\n return;\r\n }\r\n\r\n const audio = new Audio(SFX_SOURCES[name]);\r\n audio.preload = \"auto\";\r\n\r\n let settled = false;\r\n const finalize = () => {\r\n if (settled) return;\r\n settled = true;\r\n audio.onended = null;\r\n audio.onerror = null;\r\n resolve();\r\n };\r\n\r\n audio.onended = finalize;\r\n audio.onerror = finalize;\r\n void audio.play().catch(() => finalize());\r\n });\r\n }\r\n\r\n private async drain(): Promise<void> {\r\n if (this.active) return;\r\n this.active = true;\r\n try {\r\n while (this.queue.length > 0) {\r\n const next = this.queue.shift();\r\n if (!next) continue;\r\n await this.playNow(next);\r\n }\r\n } finally {\r\n this.active = false;\r\n }\r\n }\r\n\r\n playCue(name: SfxName): void {\r\n if (typeof window === \"undefined\") return;\r\n this.queue.push(name);\r\n if (!this.active) {\r\n void this.drain();\r\n }\r\n }\r\n}\r\n\r\nconst sfxManager = new SfxManager();\r\n\r\nexport const playCue = (name: SfxName): void => {\r\n sfxManager.playCue(name);\r\n};\r\n\r\n","export default \"data:audio/mpeg;base64,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\"","export default \"data:audio/mpeg;base64,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VVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVQ==\"","export default \"data:audio/mpeg;base64,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qqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqgU7bdtbNa5MIRBHgnj4UR9JwIgVBwG4sGQ6iUOI9A8D4gD+UB2EkchSQR4JZYNzUyKpeTJzhe5G6uhMjIwPOFopGnGihQsgHRJAooSJEExgcJJAooD/+rJs9RAqj/URY7JpgTYKAAANIAAAAQAAAaQAAAAgAAA0gAAABIWQTUkROElmHxaqlm7Oz/9nzZqWcuLUkicacUfCakiJxZZlwtGjSini42af/s7P3dtyppyij0FopETgMDMJoLRJGlFHmWpJE4sukDVJVUxBTUUzLjk5LjVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVU=\"","export default \"data:audio/mpeg;base64,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BpGoUioUhyHQcjkOIhCCIQkg1EYApIxw2dKsgSFBRY7KF9qKJlrzBofwHJWHEHg1GskL2IUxVHoaRPKh2w3AlLxBIIwjQThaxUZGCNA3myVKnEbD2UScIrTc0hRFSim+MkJEMh44vC5KlhkaDxdSdJrFSxxeE4XSqyy8Jw33Jf/6smxwIbyA+HBovGsrZIgAAA0gAAABFwmixmwxKYgAADSAAAAEk0Dbmma1Emo9lEQli6+eTKIhEQfOqTqKypYZOqL1dKlhkZKLwnSaxUZKI0E4WsVKnEbD3SvFUpueyiVUhvySEiKnDi/9pov9VVNvpoqVTEFNRTMuOTkuNVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVf/6smw7EwAP8AAAaQAAAAgAAA0gAAABAAABpAAAACAAADSAAAAETEFNRTMuOTkuNVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVTEFNRTMuOTkuNVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVQ==\"","\r\nimport type { CSSProperties, JSX } from \"preact\";\r\nimport { useMemo } from \"preact/hooks\";\r\n\r\nexport type SvgIconProps = Omit<JSX.IntrinsicElements[\"svg\"], \"style\"> & {\r\n src: string;\r\n title?: string;\r\n /**\r\n * If true, replaces fill and stroke attributes with \"currentColor\" in the inner SVG content.\r\n * Default: true\r\n */\r\n stripColors?: boolean;\r\n style?: CSSProperties;\r\n // Explicitly add common props to ensure destructuring works smoothly\r\n className?: string;\r\n fill?: string;\r\n stroke?: string;\r\n strokeWidth?: string | number;\r\n width?: string | number;\r\n height?: string | number;\r\n};\r\n\r\nexport function SvgIcon({\r\n src,\r\n width,\r\n height,\r\n title,\r\n stripColors = true,\r\n className,\r\n style,\r\n fill,\r\n stroke,\r\n strokeWidth,\r\n ...props\r\n}: SvgIconProps) {\r\n const { viewBox, content } = useMemo(() => {\r\n if (!src) return { viewBox: \"0 0 24 24\", content: \"\" };\r\n\r\n const trimmedSrc = src.trim();\r\n // Extract viewBox\r\n const viewBoxMatch = trimmedSrc.match(/viewBox=\"([^\"]*)\"/);\r\n const viewBox = viewBoxMatch ? viewBoxMatch[1] : \"0 0 24 24\";\r\n\r\n // Extract inner content by removing the outer <svg> wrapper\r\n // We remove the opening <svg ...> tag and the closing </svg> tag only.\r\n let content = trimmedSrc.replace(/^<svg[^>]*>/i, \"\").replace(/<\\/svg>\\s*$/i, \"\");\r\n\r\n return { viewBox, content };\r\n }, [src]);\r\n\r\n const cleanContent = useMemo(() => {\r\n if (!content) return \"\";\r\n let nextContent = content;\r\n\r\n if (stripColors) {\r\n // Force inner nodes to use props instead of hardcoded SVG values.\r\n nextContent = nextContent\r\n .replace(/fill=\"[^\"]*\"/gi, `fill=\"${fill}\"`)\r\n .replace(/stroke=\"[^\"]*\"/gi, `stroke=\"${stroke}\"`);\r\n }\r\n\r\n if (strokeWidth !== undefined) {\r\n const strokeWidthValue = String(strokeWidth);\r\n nextContent = /stroke-width=\"[^\"]*\"/i.test(nextContent)\r\n ? nextContent.replace(/stroke-width=\"[^\"]*\"/gi, `stroke-width=\"${strokeWidthValue}\"`)\r\n : nextContent.replace(\r\n /<(path|circle|ellipse|line|polyline|polygon|rect)\\b/gi,\r\n `<$1 stroke-width=\"${strokeWidthValue}\"`,\r\n );\r\n }\r\n\r\n return nextContent;\r\n }, [content, stripColors, fill, stroke, strokeWidth]);\r\n\r\n return (\r\n <svg\r\n viewBox={viewBox}\r\n width={width}\r\n height={height}\r\n className={className}\r\n style={{\r\n display: \"inline-block\",\r\n verticalAlign: \"middle\",\r\n flexShrink: 0,\r\n ...style,\r\n }}\r\n xmlns=\"http://www.w3.org/2000/svg\"\r\n {...props}\r\n dangerouslySetInnerHTML={{ __html: cleanContent }}\r\n >\r\n {title && <title>{title}</title>}\r\n </svg>\r\n );\r\n}\r\n","import { useEffect, useLayoutEffect, useRef, useState } from \"preact/hooks\";\r\nimport type { JSX } from \"preact\";\r\nimport type { BulutRuntimeConfig } from \"../index\";\r\nimport {\r\n agentTextChatStream,\r\n agentVoiceChatStream,\r\n agentResumeStream,\r\n startSttWebSocketStream,\r\n stopActiveAudioPlayback,\r\n speakText,\r\n type AudioStreamState,\r\n type StreamController,\r\n type AgentToolCallInfo,\r\n type SttWsController,\r\n} from \"../api/client\";\r\nimport {\r\n executeSingleToolCall,\r\n parseAgentResponse,\r\n getPendingAgentResume,\r\n clearPendingAgentResume,\r\n type ToolCallWithId,\r\n} from \"../agent/tools\";\r\nimport { getPageContext } from \"../agent/context\";\r\nimport {\r\n WINDOW_WIDTH,\r\n WINDOW_HEIGHT,\r\n POSITION_BOTTOM,\r\n POSITION_RIGHT,\r\n COLORS,\r\n TRANSITIONS,\r\n BORDER_RADIUS,\r\n SHADOW,\r\n} from \"../styles/constants\";\r\nimport {\r\n logoContent,\r\n} from \"../assets\";\r\nimport { StreamingJsonParser } from \"../utils/streamingJson\";\r\nimport { playCue, type SfxName } from \"../audio/sfxManager\";\r\nimport {\r\n ArrowPathIcon,\r\n CommandLineIcon,\r\n CursorArrowRaysIcon,\r\n FaceSmileIcon,\r\n HandRaisedIcon,\r\n MapIcon,\r\n MicrophoneIcon,\r\n QueueListIcon,\r\n StopIcon,\r\n XMarkIcon,\r\n} from \"@heroicons/react/24/outline\";\r\n\r\nimport { SvgIcon } from \"./SvgIcon\";\r\n\r\nexport interface ChatWindowHandle {\r\n startRecording: () => void;\r\n cancelRecording: () => void;\r\n stopTask: () => void;\r\n}\r\n\r\ninterface ChatWindowProps {\r\n onClose: () => void;\r\n config: BulutRuntimeConfig;\r\n accessibilityMode?: boolean;\r\n onAccessibilityToggle?: () => void;\r\n hidden?: boolean;\r\n actionsRef?: { current: ChatWindowHandle | null };\r\n onRecordingChange?: (recording: boolean) => void;\r\n onBusyChange?: (busy: boolean) => void;\r\n onPreviewChange?: (text: string | null) => void;\r\n}\r\n\r\ninterface Message {\r\n id: number;\r\n text: string;\r\n isUser: boolean;\r\n /** \"message\" (default) | \"tool\" for tool call indicators */\r\n type?: \"message\" | \"tool\";\r\n toolKind?: \"context\" | \"cursor\" | \"scroll\" | \"navigate\" | \"form\" | \"interact\" | \"unknown\";\r\n toolLabel?: string;\r\n toolCount?: number;\r\n}\r\n\r\ntype RecordingMode = \"vad\" | \"press\";\r\n\r\ntype StorageLike = {\r\n removeItem: (key: string) => void;\r\n};\r\n\r\nconst STORAGE_KEY = \"bulut_chat_history\";\r\nconst TIMESTAMP_KEY = \"bulut_chat_timestamp\";\r\nconst SESSION_ID_KEY = \"bulut_session_id\";\r\nconst TTL_MS = 5 * 60 * 1000;\r\nconst VAD_THRESHOLD = 0.06;\r\nconst SILENCE_DURATION_MS = 500;\r\nconst ACCESSIBILITY_MIN_SPEECH_DURATION_MS = 1500;\r\nexport const HOLD_THRESHOLD_MS = 250;\r\n\r\nconst STATUS_LABELS = {\r\n ready: \"Hazır\",\r\n loading: \"Bir saniye\",\r\n micInitializing: \"Mikrofonu hazırlıyorum\",\r\n listening: \"Sizi dinliyorum\",\r\n accessibilityActive: \"Erişilebilirlik Aktif\",\r\n transcribing: \"Düşünüyorum\",\r\n thinking: \"Düşünüyorum\",\r\n playingAudio: \".\",\r\n runningTools: \"Siteyle ilgileniyorum\",\r\n} as const;\r\n\r\nexport const getGreetingText = (agentName: string): string =>\r\n `Merhaba, ben ${agentName}. Bu web sayfasında neler yapalım?`;\r\n\r\nexport interface StatusFlags {\r\n isBusy: boolean;\r\n isRecording: boolean;\r\n isTranscribing: boolean;\r\n isThinking: boolean;\r\n isRenderingAudio: boolean;\r\n isPlayingAudio: boolean;\r\n isRunningTools: boolean;\r\n}\r\n\r\nexport const resolveStatusText = (flags: StatusFlags): string => {\r\n if (flags.isRecording) return STATUS_LABELS.listening;\r\n if (flags.isRunningTools) return STATUS_LABELS.runningTools;\r\n if (flags.isPlayingAudio) return STATUS_LABELS.playingAudio;\r\n if (flags.isThinking) return STATUS_LABELS.thinking;\r\n if (flags.isTranscribing) return STATUS_LABELS.transcribing;\r\n if (flags.isBusy) return STATUS_LABELS.loading;\r\n return STATUS_LABELS.ready;\r\n};\r\n\r\nexport const hasActiveStatus = (\r\n flags: StatusFlags,\r\n statusOverride: string | null,\r\n): boolean =>\r\n Boolean(\r\n statusOverride\r\n || flags.isBusy\r\n || flags.isRecording\r\n || flags.isTranscribing\r\n || flags.isThinking\r\n || flags.isRenderingAudio\r\n || flags.isPlayingAudio\r\n || flags.isRunningTools,\r\n );\r\n\r\nexport const formatDurationMs = (durationMs: number): string => {\r\n const totalSeconds = Math.max(0, Math.floor(durationMs / 1000));\r\n const minutes = Math.floor(totalSeconds / 60)\r\n .toString()\r\n .padStart(2, \"0\");\r\n const seconds = (totalSeconds % 60).toString().padStart(2, \"0\");\r\n return `${minutes}:${seconds}`;\r\n};\r\n\r\nexport const classifyMicGesture = (\r\n durationMs: number,\r\n thresholdMs: number = HOLD_THRESHOLD_MS,\r\n): \"tap\" | \"hold\" => (durationMs >= thresholdMs ? \"hold\" : \"tap\");\r\n\r\nexport const createInitialMessages = (agentName: string): Message[] => [\r\n {\r\n id: 1,\r\n text: getGreetingText(agentName),\r\n isUser: false,\r\n },\r\n];\r\n\r\nexport const clearPersistedChatState = (storage: StorageLike | null): void => {\r\n if (!storage) {\r\n return;\r\n }\r\n\r\n storage.removeItem(STORAGE_KEY);\r\n storage.removeItem(TIMESTAMP_KEY);\r\n storage.removeItem(SESSION_ID_KEY);\r\n};\r\n\r\nexport const scrollElementToBottom = (\r\n element: { scrollTop: number; scrollHeight: number } | null,\r\n): void => {\r\n if (!element) {\r\n return;\r\n }\r\n\r\n element.scrollTop = element.scrollHeight;\r\n};\r\n\r\nconst normalizeError = (error: unknown) => {\r\n if (error instanceof Error) {\r\n return error.message;\r\n }\r\n return \"Bilinmeyen hata\";\r\n};\r\n\r\nconst getNextMessageId = (messages: Message[]): number => {\r\n const maxId = messages.reduce((acc, message) => Math.max(acc, message.id), 0);\r\n return maxId + 1;\r\n};\r\n\r\nexport interface AssistantPayloadResolution {\r\n displayText: string;\r\n toolCalls: ReturnType<typeof parseAgentResponse>[\"toolCalls\"];\r\n}\r\n\r\nexport const resolveAssistantPayload = (\r\n assistantText: string,\r\n): AssistantPayloadResolution => {\r\n const parsed = parseAgentResponse(assistantText);\r\n return {\r\n displayText: parsed.reply || assistantText,\r\n toolCalls: parsed.toolCalls,\r\n };\r\n};\r\n\r\nexport const shouldAutoListenAfterAudio = (\r\n accessibilityMode: boolean,\r\n expectsReply: boolean,\r\n isRecording: boolean,\r\n isBusy: boolean,\r\n): boolean => (accessibilityMode || expectsReply) && !isRecording && !isBusy;\r\n\r\nexport const shouldAcceptVadSpeech = (\r\n speechDurationMs: number,\r\n enforceMinSpeechDuration: boolean,\r\n minSpeechDurationMs: number = ACCESSIBILITY_MIN_SPEECH_DURATION_MS,\r\n): boolean => !enforceMinSpeechDuration || speechDurationMs >= minSpeechDurationMs;\r\n\r\ninterface ToolIndicatorMessage {\r\n text: string;\r\n kind: \"context\" | \"cursor\" | \"scroll\" | \"navigate\" | \"form\" | \"interact\" | \"unknown\";\r\n}\r\n\r\nconst getToolIndicatorMessage = (\r\n call: AgentToolCallInfo,\r\n): ToolIndicatorMessage => {\r\n if (call.tool === \"getPageContext\") {\r\n return { text: \"Algılama\", kind: \"context\" };\r\n }\r\n if (call.tool === \"scroll\") {\r\n return { text: \"Kaydırma\", kind: \"scroll\" };\r\n }\r\n if (call.tool === \"navigate\") {\r\n const url = typeof call.args.url === \"string\" ? call.args.url.trim() : \"\";\r\n return {\r\n text: url ? `Sayfa Geçişi: ${url}` : \"Sayfa Geçişi\",\r\n kind: \"navigate\",\r\n };\r\n }\r\n if (call.tool === \"interact\" && call.args.action === \"move\") {\r\n return { text: \"Serbest İmleç\", kind: \"cursor\" };\r\n }\r\n if (call.tool === \"interact\" && call.args.action === \"type\") {\r\n return { text: \"Form Doldurma\", kind: \"form\" };\r\n }\r\n if (call.tool === \"interact\" && call.args.action === \"submit\") {\r\n return { text: \"Form Gönderme\", kind: \"form\" };\r\n }\r\n if (call.tool === \"interact\" && call.args.action === \"click\") {\r\n return { text: \"Tıklama\", kind: \"interact\" };\r\n }\r\n if (call.tool === \"interact\") {\r\n return { text: \"Etkileşim\", kind: \"interact\" };\r\n }\r\n return {\r\n text: call.tool || \"Araç\",\r\n kind: \"unknown\",\r\n };\r\n};\r\n\r\nexport const ChatWindow = ({\r\n onClose,\r\n config,\r\n accessibilityMode = false,\r\n onAccessibilityToggle,\r\n hidden = false,\r\n actionsRef,\r\n onRecordingChange,\r\n onBusyChange,\r\n onPreviewChange,\r\n}: ChatWindowProps) => {\r\n const [messages, setMessages] = useState<Message[]>(() => {\r\n if (typeof localStorage !== \"undefined\") {\r\n const saved = localStorage.getItem(STORAGE_KEY);\r\n const timestamp = localStorage.getItem(TIMESTAMP_KEY);\r\n\r\n if (saved && timestamp) {\r\n const timePassed = Date.now() - parseInt(timestamp, 10);\r\n if (timePassed < TTL_MS) {\r\n try {\r\n return JSON.parse(saved) as Message[];\r\n } catch {\r\n // Ignore parse error and continue with default.\r\n }\r\n } else {\r\n clearPersistedChatState(localStorage);\r\n }\r\n }\r\n }\r\n\r\n return createInitialMessages(config.agentName);\r\n });\r\n\r\n const [isBusy, setIsBusy] = useState(false);\r\n const [isRecording, setIsRecording] = useState(false);\r\n const [isTranscribing, setIsTranscribing] = useState(false);\r\n const [isThinking, setIsThinking] = useState(false);\r\n const [isRenderingAudio, setIsRenderingAudio] = useState(false);\r\n const [isPlayingAudio, setIsPlayingAudio] = useState(false);\r\n const [isRunningTools, setIsRunningTools] = useState(false);\r\n const [isMicPending, setIsMicPending] = useState(false);\r\n const [recordingDurationMs, setRecordingDurationMs] = useState(0);\r\n const [statusOverride, setStatusOverride] = useState<string | null>(null);\r\n const statusFlags: StatusFlags = {\r\n isBusy,\r\n isRecording,\r\n isTranscribing,\r\n isThinking,\r\n isRenderingAudio,\r\n isPlayingAudio,\r\n isRunningTools,\r\n };\r\n const resolvedStatusText = resolveStatusText(statusFlags);\r\n const showStatus = hasActiveStatus(statusFlags, statusOverride);\r\n const statusText = showStatus ? (statusOverride ?? resolvedStatusText) : STATUS_LABELS.ready;\r\n\r\n const isBusyRef = useRef(isBusy);\r\n const isRecordingRef = useRef(isRecording);\r\n\r\n const nextMessageIdRef = useRef(getNextMessageId(messages));\r\n const recorderRef = useRef<MediaRecorder | null>(null);\r\n const streamRef = useRef<MediaStream | null>(null);\r\n const audioChunksRef = useRef<BlobPart[]>([]);\r\n const activeStreamControllerRef = useRef<StreamController | null>(null);\r\n const sessionIdRef = useRef<string | null>(\r\n typeof localStorage !== \"undefined\"\r\n ? (() => {\r\n const ts = localStorage.getItem(TIMESTAMP_KEY);\r\n if (ts && Date.now() - parseInt(ts, 10) < TTL_MS) {\r\n return localStorage.getItem(SESSION_ID_KEY);\r\n }\r\n return null;\r\n })()\r\n : null,\r\n );\r\n\r\n const silenceStartRef = useRef<number | null>(null);\r\n const vadIntervalRef = useRef<number | null>(null);\r\n const audioContextRef = useRef<AudioContext | null>(null);\r\n const sourceRef = useRef<MediaStreamAudioSourceNode | null>(null);\r\n\r\n const discardNextRecordingRef = useRef(false);\r\n\r\n const micPressStartRef = useRef<number | null>(null);\r\n const micHoldTimeoutRef = useRef<number | null>(null);\r\n const micHoldTriggeredRef = useRef(false);\r\n const recordingModeRef = useRef<RecordingMode | null>(null);\r\n const pendingStopAfterStartRef = useRef(false);\r\n const startRecordingPendingRef = useRef(false);\r\n\r\n const assistantMessageIdRef = useRef<number | null>(null);\r\n const assistantTextBufferRef = useRef(\"\");\r\n const transcriptionReceivedRef = useRef(false);\r\n const assistantDoneReceivedRef = useRef(false);\r\n\r\n const recordingStartedAtRef = useRef<number | null>(null);\r\n const recordingTimerIntervalRef = useRef<number | null>(null);\r\n\r\n const messagesContainerRef = useRef<HTMLDivElement | null>(null);\r\n const messagesContentRef = useRef<HTMLDivElement | null>(null);\r\n\r\n const pendingUserTextRef = useRef<string | null>(null);\r\n const pendingAssistantTextRef = useRef<string>(\"\");\r\n const streamingJsonParserRef = useRef<StreamingJsonParser | null>(null);\r\n const awaitingAssistantResponseRef = useRef(false);\r\n const activeSttWsRef = useRef<SttWsController | null>(null);\r\n const liveTranscriptionMessageIdRef = useRef<number | null>(null);\r\n const liveTranscriptionTextRef = useRef(\"\");\r\n const autoListenSuppressedRef = useRef(false);\r\n const expectsReplyRef = useRef(true);\r\n const requestEpochRef = useRef(0);\r\n const sttSendCuePlayedRef = useRef(false);\r\n\r\n useEffect(() => {\r\n isBusyRef.current = isBusy;\r\n }, [isBusy]);\r\n\r\n useEffect(() => {\r\n isRecordingRef.current = isRecording;\r\n }, [isRecording]);\r\n\r\n // Report state changes to parent\r\n useEffect(() => { onRecordingChange?.(isRecording); }, [isRecording]);\r\n useEffect(() => { onBusyChange?.(isBusy); }, [isBusy]);\r\n\r\n // Derive and report preview text to parent\r\n useEffect(() => {\r\n if (!onPreviewChange) return;\r\n if (isRecording) {\r\n onPreviewChange(statusOverride ?? STATUS_LABELS.listening);\r\n return;\r\n }\r\n // When audio is rendering/playing, show the actual message text\r\n if (isRenderingAudio || isPlayingAudio) {\r\n const lastAssistant = [...messages].reverse().find(m => !m.isUser && m.type !== \"tool\");\r\n onPreviewChange(lastAssistant?.text ?? getGreetingText(config.agentName));\r\n return;\r\n }\r\n if (showStatus) {\r\n const st = statusOverride ?? resolveStatusText({\r\n isBusy,\r\n isRecording,\r\n isTranscribing,\r\n isThinking,\r\n isRenderingAudio,\r\n isPlayingAudio,\r\n isRunningTools,\r\n });\r\n onPreviewChange(st);\r\n return;\r\n }\r\n // Show last assistant message (or greeting)\r\n const lastAssistant = [...messages].reverse().find(m => !m.isUser && m.type !== \"tool\");\r\n onPreviewChange(lastAssistant?.text ?? getGreetingText(config.agentName));\r\n }, [\r\n isRecording,\r\n isBusy,\r\n isTranscribing,\r\n isThinking,\r\n isRunningTools,\r\n isPlayingAudio,\r\n isRenderingAudio,\r\n statusOverride,\r\n showStatus,\r\n messages,\r\n ]);\r\n\r\n const playSfx = (name: SfxName) => {\r\n playCue(name);\r\n };\r\n\r\n const beginRequestEpoch = () => {\r\n requestEpochRef.current += 1;\r\n return requestEpochRef.current;\r\n };\r\n\r\n const invalidateRequestEpoch = () => {\r\n requestEpochRef.current += 1;\r\n };\r\n\r\n const isCurrentRequestEpoch = (epoch: number): boolean =>\r\n requestEpochRef.current === epoch;\r\n\r\n const playSttSentCueOnce = () => {\r\n if (sttSendCuePlayedRef.current) {\r\n return;\r\n }\r\n sttSendCuePlayedRef.current = true;\r\n playSfx(\"sent\");\r\n };\r\n\r\n useEffect(() => {\r\n if (typeof localStorage !== \"undefined\") {\r\n localStorage.setItem(STORAGE_KEY, JSON.stringify(messages));\r\n localStorage.setItem(TIMESTAMP_KEY, Date.now().toString());\r\n }\r\n }, [messages]);\r\n\r\n const scrollMessagesToBottom = () => {\r\n scrollElementToBottom(messagesContainerRef.current);\r\n };\r\n\r\n useLayoutEffect(() => {\r\n scrollMessagesToBottom();\r\n }, [messages, statusText, isBusy, isRecording]);\r\n\r\n useEffect(() => {\r\n const content = messagesContentRef.current;\r\n if (!content || typeof ResizeObserver === \"undefined\") {\r\n return;\r\n }\r\n\r\n const observer = new ResizeObserver(() => {\r\n scrollMessagesToBottom();\r\n });\r\n\r\n observer.observe(content);\r\n return () => observer.disconnect();\r\n }, []);\r\n\r\n const stopRecordingTimer = () => {\r\n if (recordingTimerIntervalRef.current !== null) {\r\n window.clearInterval(recordingTimerIntervalRef.current);\r\n recordingTimerIntervalRef.current = null;\r\n }\r\n recordingStartedAtRef.current = null;\r\n };\r\n\r\n const startRecordingTimer = () => {\r\n stopRecordingTimer();\r\n recordingStartedAtRef.current = Date.now();\r\n setRecordingDurationMs(0);\r\n\r\n recordingTimerIntervalRef.current = window.setInterval(() => {\r\n const startedAt = recordingStartedAtRef.current;\r\n if (startedAt === null) {\r\n setRecordingDurationMs(0);\r\n return;\r\n }\r\n setRecordingDurationMs(Date.now() - startedAt);\r\n }, 200);\r\n };\r\n\r\n const resetProcessingFlags = () => {\r\n setIsTranscribing(false);\r\n setIsThinking(false);\r\n setIsRenderingAudio(false);\r\n setIsPlayingAudio(false);\r\n setIsRunningTools(false);\r\n setStatusOverride(null);\r\n assistantMessageIdRef.current = null;\r\n assistantTextBufferRef.current = \"\";\r\n transcriptionReceivedRef.current = false;\r\n assistantDoneReceivedRef.current = false;\r\n awaitingAssistantResponseRef.current = false;\r\n pendingUserTextRef.current = null;\r\n pendingAssistantTextRef.current = \"\";\r\n };\r\n\r\n const clearMicHoldTimeout = () => {\r\n if (micHoldTimeoutRef.current !== null) {\r\n window.clearTimeout(micHoldTimeoutRef.current);\r\n micHoldTimeoutRef.current = null;\r\n }\r\n };\r\n\r\n const cleanupVAD = () => {\r\n if (vadIntervalRef.current !== null) {\r\n clearInterval(vadIntervalRef.current);\r\n vadIntervalRef.current = null;\r\n }\r\n\r\n if (sourceRef.current) {\r\n sourceRef.current.disconnect();\r\n sourceRef.current = null;\r\n }\r\n\r\n if (audioContextRef.current) {\r\n audioContextRef.current.close().catch(() => {});\r\n audioContextRef.current = null;\r\n }\r\n\r\n silenceStartRef.current = null;\r\n };\r\n\r\n const stopStreamTracks = () => {\r\n if (!streamRef.current) {\r\n return;\r\n }\r\n\r\n streamRef.current.getTracks().forEach((track) => track.stop());\r\n streamRef.current = null;\r\n };\r\n\r\n const stopActiveStream = () => {\r\n if (!activeStreamControllerRef.current) {\r\n return;\r\n }\r\n\r\n activeStreamControllerRef.current.stop();\r\n activeStreamControllerRef.current = null;\r\n };\r\n\r\n const cancelActiveSttWs = () => {\r\n const activeSttWs = activeSttWsRef.current;\r\n activeSttWsRef.current = null;\r\n activeSttWs?.cancel();\r\n liveTranscriptionMessageIdRef.current = null;\r\n liveTranscriptionTextRef.current = \"\";\r\n };\r\n\r\n useEffect(\r\n () => () => {\r\n invalidateRequestEpoch();\r\n clearMicHoldTimeout();\r\n pendingStopAfterStartRef.current = false;\r\n\r\n stopActiveStream();\r\n stopActiveAudioPlayback();\r\n cancelActiveSttWs();\r\n cleanupVAD();\r\n stopStreamTracks();\r\n stopRecordingTimer();\r\n\r\n const recorder = recorderRef.current;\r\n if (recorder) {\r\n recorder.ondataavailable = null;\r\n recorder.onstop = null;\r\n recorder.onerror = null;\r\n if (recorder.state !== \"inactive\") {\r\n recorder.stop();\r\n }\r\n recorderRef.current = null;\r\n }\r\n\r\n cancelActiveSttWs();\r\n },\r\n [],\r\n );\r\n\r\n // ── Resume agent loop after full-page navigation ────────────────\r\n useEffect(() => {\r\n const resumeState = getPendingAgentResume();\r\n if (!resumeState) return;\r\n\r\n clearPendingAgentResume();\r\n console.info(\"[Bulut] Resuming agent after navigation\");\r\n\r\n // Restore session ID from resume state\r\n if (resumeState.sessionId) {\r\n sessionIdRef.current = resumeState.sessionId;\r\n if (typeof localStorage !== \"undefined\") {\r\n localStorage.setItem(SESSION_ID_KEY, resumeState.sessionId);\r\n }\r\n }\r\n\r\n const requestEpoch = beginRequestEpoch();\r\n setIsBusy(true);\r\n isBusyRef.current = true;\r\n setIsRunningTools(true);\r\n setStatusOverride(STATUS_LABELS.thinking);\r\n\r\n const freshPageContext = getPageContext().summary;\r\n\r\n const resumeToolExec = async (\r\n call: AgentToolCallInfo,\r\n ): Promise<{ call_id: string; result: string }> => {\r\n const toolCall: ToolCallWithId = {\r\n tool: call.tool as \"navigate\" | \"getPageContext\" | \"interact\" | \"scroll\",\r\n call_id: call.call_id,\r\n ...call.args,\r\n } as ToolCallWithId;\r\n return executeSingleToolCall(toolCall);\r\n };\r\n\r\n const controller = agentResumeStream(\r\n config.backendBaseUrl,\r\n resumeState,\r\n freshPageContext,\r\n {\r\n onSessionId: (sid) => {\r\n if (!isCurrentRequestEpoch(requestEpoch)) return;\r\n if (sid && sid !== sessionIdRef.current) {\r\n sessionIdRef.current = sid;\r\n if (typeof localStorage !== \"undefined\") {\r\n localStorage.setItem(SESSION_ID_KEY, sid);\r\n }\r\n }\r\n },\r\n onAssistantDelta: (delta) => {\r\n if (!isCurrentRequestEpoch(requestEpoch)) return;\r\n setIsRunningTools(false);\r\n setIsThinking(true);\r\n setStatusOverride(null);\r\n\r\n pendingAssistantTextRef.current += delta;\r\n\r\n if (assistantMessageIdRef.current === null) {\r\n assistantMessageIdRef.current = appendMessage(\r\n pendingAssistantTextRef.current,\r\n false,\r\n );\r\n } else {\r\n updateMessageText(\r\n assistantMessageIdRef.current,\r\n pendingAssistantTextRef.current,\r\n );\r\n }\r\n },\r\n onAssistantDone: (assistantText, expectsReply) => {\r\n if (!isCurrentRequestEpoch(requestEpoch)) return;\r\n playSfx(\"completed\");\r\n expectsReplyRef.current = expectsReply !== false;\r\n setStatusOverride(null);\r\n setIsThinking(false);\r\n setIsRenderingAudio(true);\r\n\r\n const finalDisplayText =\r\n assistantText || pendingAssistantTextRef.current;\r\n pendingAssistantTextRef.current = finalDisplayText;\r\n\r\n if (assistantMessageIdRef.current !== null) {\r\n updateMessageText(\r\n assistantMessageIdRef.current,\r\n finalDisplayText,\r\n );\r\n } else {\r\n assistantMessageIdRef.current = appendMessage(\r\n finalDisplayText,\r\n false,\r\n );\r\n }\r\n },\r\n onIntermediateReply: (text) => {\r\n if (!isCurrentRequestEpoch(requestEpoch)) return;\r\n void speakText(\r\n config.backendBaseUrl, text, config.voice,\r\n accessibilityMode, (state) => handleAudioStateChange(state, requestEpoch),\r\n ).catch((err) => console.warn(\"[Bulut] intermediate TTS failed\", err));\r\n },\r\n onToolCalls: (calls) => {\r\n if (!isCurrentRequestEpoch(requestEpoch)) return;\r\n if (calls.length > 0) {\r\n playSfx(\"toolCall\");\r\n }\r\n setIsRunningTools(true);\r\n setStatusOverride(STATUS_LABELS.runningTools);\r\n appendToolIndicatorMessages(calls);\r\n\r\n assistantMessageIdRef.current = null;\r\n pendingAssistantTextRef.current = \"\";\r\n },\r\n onToolResult: () => {},\r\n onIteration: () => {\r\n if (!isCurrentRequestEpoch(requestEpoch)) return;\r\n playSfx(\"thinking\");\r\n setIsThinking(true);\r\n setStatusOverride(STATUS_LABELS.thinking);\r\n },\r\n onAudioStateChange: (state) => {\r\n handleAudioStateChange(state, requestEpoch);\r\n },\r\n onError: (err) => {\r\n if (!isCurrentRequestEpoch(requestEpoch)) return;\r\n setStatusOverride(null);\r\n appendMessage(`Hata: ${err}`, false);\r\n },\r\n },\r\n resumeToolExec,\r\n );\r\n\r\n activeStreamControllerRef.current = controller;\r\n\r\n controller.done\r\n .catch(() => {})\r\n .finally(() => {\r\n if (!isCurrentRequestEpoch(requestEpoch)) return;\r\n setIsBusy(false);\r\n isBusyRef.current = false;\r\n setIsRunningTools(false);\r\n setIsThinking(false);\r\n setIsRenderingAudio(false);\r\n setIsPlayingAudio(false);\r\n setStatusOverride(null);\r\n pendingAssistantTextRef.current = \"\";\r\n assistantMessageIdRef.current = null;\r\n activeStreamControllerRef.current = null;\r\n\r\n if (\r\n !autoListenSuppressedRef.current &&\r\n shouldAutoListenAfterAudio(\r\n accessibilityMode,\r\n expectsReplyRef.current,\r\n isRecordingRef.current,\r\n false,\r\n )\r\n ) {\r\n void startRecording(\"vad\");\r\n }\r\n // Reset for next turn\r\n expectsReplyRef.current = true;\r\n });\r\n }, []); // eslint-disable-line react-hooks/exhaustive-deps\r\n\r\n const appendMessage = (\r\n text: string,\r\n isUser: boolean,\r\n options?: {\r\n type?: \"message\" | \"tool\";\r\n toolKind?: Message[\"toolKind\"];\r\n toolLabel?: string;\r\n toolCount?: number;\r\n },\r\n ): number => {\r\n const id = nextMessageIdRef.current++;\r\n setMessages((previous) => [\r\n ...previous,\r\n {\r\n id,\r\n text,\r\n isUser,\r\n type: options?.type,\r\n toolKind: options?.toolKind,\r\n toolLabel: options?.toolLabel,\r\n toolCount: options?.toolCount,\r\n },\r\n ]);\r\n return id;\r\n };\r\n\r\n const appendToolIndicatorMessages = (calls: AgentToolCallInfo[]) => {\r\n setMessages((previous) => {\r\n const next = [...previous];\r\n\r\n for (const call of calls) {\r\n const indicator = getToolIndicatorMessage(call);\r\n const last = next[next.length - 1];\r\n const previousToolText = typeof last?.text === \"string\"\r\n ? last.text.replace(/\\s+\\(\\d+\\)$/, \"\")\r\n : \"\";\r\n\r\n if (\r\n last\r\n && !last.isUser\r\n && last.type === \"tool\"\r\n && previousToolText === indicator.text\r\n ) {\r\n const extractedCount = Number.parseInt(\r\n (last.text.match(/\\((\\d+)\\)\\s*$/)?.[1] ?? \"1\"),\r\n 10,\r\n );\r\n const safeCurrentCount = Number.isFinite(extractedCount) ? extractedCount : 1;\r\n const nextCount = safeCurrentCount + 1;\r\n const baseLabel = previousToolText || indicator.text;\r\n next[next.length - 1] = {\r\n ...last,\r\n toolLabel: baseLabel,\r\n toolCount: nextCount,\r\n text: `${baseLabel} (${nextCount})`,\r\n };\r\n continue;\r\n }\r\n\r\n const id = nextMessageIdRef.current++;\r\n next.push({\r\n id,\r\n text: indicator.text,\r\n isUser: false,\r\n type: \"tool\",\r\n toolKind: indicator.kind,\r\n toolLabel: indicator.text,\r\n toolCount: 1,\r\n });\r\n }\r\n\r\n return next;\r\n });\r\n };\r\n\r\n const updateMessageText = (id: number, text: string) => {\r\n setMessages((previous) =>\r\n previous.map((message) =>\r\n message.id === id ? { ...message, text } : message,\r\n ),\r\n );\r\n };\r\n\r\n const upsertLiveUserTranscription = (text: string) => {\r\n const normalized = text.trim();\r\n if (!normalized) {\r\n return;\r\n }\r\n liveTranscriptionTextRef.current = normalized;\r\n if (liveTranscriptionMessageIdRef.current === null) {\r\n liveTranscriptionMessageIdRef.current = appendMessage(normalized, true);\r\n return;\r\n }\r\n updateMessageText(liveTranscriptionMessageIdRef.current, normalized);\r\n };\r\n\r\n const clearLiveUserTranscriptionState = () => {\r\n liveTranscriptionMessageIdRef.current = null;\r\n liveTranscriptionTextRef.current = \"\";\r\n };\r\n\r\n const handleAudioStateChange = (state: AudioStreamState, requestEpoch?: number) => {\r\n if (typeof requestEpoch === \"number\" && !isCurrentRequestEpoch(requestEpoch)) {\r\n return;\r\n }\r\n\r\n if (state === \"rendering\") {\r\n setIsRenderingAudio(true);\r\n setIsPlayingAudio(false);\r\n return;\r\n }\r\n\r\n if (state === \"playing\") {\r\n setIsRenderingAudio(false);\r\n setIsPlayingAudio(true);\r\n return;\r\n }\r\n\r\n if (state === \"fallback\") {\r\n setIsRenderingAudio(true);\r\n setIsPlayingAudio(false);\r\n return;\r\n }\r\n\r\n if (state === \"done\") {\r\n setIsRenderingAudio(false);\r\n setIsPlayingAudio(false);\r\n return;\r\n }\r\n\r\n setIsRenderingAudio(false);\r\n setIsPlayingAudio(false);\r\n };\r\n\r\n const finalizeStreamCycle = (requestEpoch?: number) => {\r\n if (typeof requestEpoch === \"number\" && !isCurrentRequestEpoch(requestEpoch)) {\r\n return;\r\n }\r\n\r\n awaitingAssistantResponseRef.current = false;\r\n setStatusOverride(null);\r\n setIsBusy(false);\r\n isBusyRef.current = false;\r\n setIsTranscribing(false);\r\n setIsThinking(false);\r\n setIsRenderingAudio(false);\r\n setIsPlayingAudio(false);\r\n setIsRunningTools(false);\r\n pendingUserTextRef.current = null;\r\n pendingAssistantTextRef.current = \"\";\r\n assistantMessageIdRef.current = null;\r\n if (activeStreamControllerRef.current) {\r\n activeStreamControllerRef.current = null;\r\n }\r\n if (\r\n !autoListenSuppressedRef.current &&\r\n shouldAutoListenAfterAudio(\r\n accessibilityMode,\r\n expectsReplyRef.current,\r\n isRecordingRef.current,\r\n false,\r\n )\r\n ) {\r\n console.info(\"[Bulut] chat-window auto-listen trigger after stream completion\");\r\n void startRecording(\"vad\");\r\n }\r\n // Reset for next turn\r\n expectsReplyRef.current = true;\r\n };\r\n\r\n const runAgentForUserText = async (userText: string) => {\r\n if (!config.projectId) {\r\n appendMessage(\"Hata: Project ID yapılandırılmamış.\", false);\r\n return;\r\n }\r\n\r\n const normalizedUserText = userText.trim();\r\n if (!normalizedUserText) {\r\n appendMessage(\"Ses kaydı metne dönüştürülemedi. Lütfen tekrar deneyin.\", false);\r\n return;\r\n }\r\n\r\n const requestEpoch = beginRequestEpoch();\r\n setIsBusy(true);\r\n isBusyRef.current = true;\r\n setIsTranscribing(false);\r\n setIsThinking(true);\r\n setIsRenderingAudio(false);\r\n setIsPlayingAudio(false);\r\n setIsRunningTools(false);\r\n setStatusOverride(STATUS_LABELS.thinking);\r\n awaitingAssistantResponseRef.current = true;\r\n\r\n try {\r\n pendingUserTextRef.current = normalizedUserText;\r\n upsertLiveUserTranscription(normalizedUserText);\r\n clearLiveUserTranscriptionState();\r\n\r\n stopActiveStream();\r\n const pageContext = getPageContext().summary;\r\n\r\n const handleToolExecution = async (\r\n call: AgentToolCallInfo,\r\n ): Promise<{ call_id: string; result: string }> => {\r\n const toolCall: ToolCallWithId = {\r\n tool: call.tool as\r\n | \"navigate\"\r\n | \"getPageContext\"\r\n | \"interact\"\r\n | \"scroll\",\r\n call_id: call.call_id,\r\n ...call.args,\r\n } as ToolCallWithId;\r\n return executeSingleToolCall(toolCall);\r\n };\r\n\r\n const controller = agentTextChatStream(\r\n config.backendBaseUrl,\r\n normalizedUserText,\r\n config.projectId,\r\n sessionIdRef.current,\r\n {\r\n model: config.model,\r\n voice: config.voice,\r\n pageContext,\r\n accessibilityMode,\r\n },\r\n {\r\n onSessionId: (sid) => {\r\n if (!isCurrentRequestEpoch(requestEpoch)) return;\r\n if (sid && sid !== sessionIdRef.current) {\r\n sessionIdRef.current = sid;\r\n if (typeof localStorage !== \"undefined\") {\r\n localStorage.setItem(SESSION_ID_KEY, sid);\r\n }\r\n }\r\n },\r\n onAssistantDelta: (delta) => {\r\n if (!isCurrentRequestEpoch(requestEpoch)) return;\r\n setIsTranscribing(false);\r\n setIsThinking(true);\r\n setIsRunningTools(false);\r\n if (awaitingAssistantResponseRef.current) {\r\n awaitingAssistantResponseRef.current = false;\r\n setStatusOverride(null);\r\n }\r\n\r\n pendingAssistantTextRef.current += delta;\r\n\r\n if (assistantMessageIdRef.current === null) {\r\n assistantMessageIdRef.current = appendMessage(\r\n pendingAssistantTextRef.current,\r\n false,\r\n );\r\n } else {\r\n updateMessageText(\r\n assistantMessageIdRef.current,\r\n pendingAssistantTextRef.current,\r\n );\r\n }\r\n },\r\n onAssistantDone: (assistantText, expectsReply) => {\r\n if (!isCurrentRequestEpoch(requestEpoch)) return;\r\n playSfx(\"completed\");\r\n expectsReplyRef.current = expectsReply !== false;\r\n awaitingAssistantResponseRef.current = false;\r\n setStatusOverride(null);\r\n setIsThinking(false);\r\n setIsRenderingAudio(true);\r\n\r\n const finalDisplayText = assistantText || pendingAssistantTextRef.current;\r\n pendingAssistantTextRef.current = finalDisplayText;\r\n\r\n if (assistantMessageIdRef.current !== null) {\r\n updateMessageText(\r\n assistantMessageIdRef.current,\r\n finalDisplayText,\r\n );\r\n } else {\r\n assistantMessageIdRef.current = appendMessage(\r\n finalDisplayText,\r\n false,\r\n );\r\n }\r\n },\r\n onIntermediateReply: (text) => {\r\n if (!isCurrentRequestEpoch(requestEpoch)) return;\r\n void speakText(\r\n config.backendBaseUrl, text, config.voice,\r\n accessibilityMode, (state) => handleAudioStateChange(state, requestEpoch),\r\n ).catch((err) => console.warn(\"[Bulut] intermediate TTS failed\", err));\r\n },\r\n onToolCalls: (calls) => {\r\n if (!isCurrentRequestEpoch(requestEpoch)) return;\r\n if (calls.length > 0) {\r\n playSfx(\"toolCall\");\r\n }\r\n setIsRunningTools(true);\r\n setStatusOverride(STATUS_LABELS.runningTools);\r\n appendToolIndicatorMessages(calls);\r\n\r\n assistantMessageIdRef.current = null;\r\n pendingAssistantTextRef.current = \"\";\r\n },\r\n onToolResult: () => {},\r\n onIteration: () => {\r\n if (!isCurrentRequestEpoch(requestEpoch)) return;\r\n playSfx(\"thinking\");\r\n setIsThinking(true);\r\n setStatusOverride(STATUS_LABELS.thinking);\r\n },\r\n onAudioStateChange: (state) => {\r\n handleAudioStateChange(state, requestEpoch);\r\n },\r\n onError: (err) => {\r\n if (!isCurrentRequestEpoch(requestEpoch)) return;\r\n awaitingAssistantResponseRef.current = false;\r\n setStatusOverride(null);\r\n appendMessage(`Hata: ${err}`, false);\r\n },\r\n },\r\n handleToolExecution,\r\n );\r\n\r\n activeStreamControllerRef.current = controller;\r\n await controller.done;\r\n } catch (error) {\r\n if (!isCurrentRequestEpoch(requestEpoch)) return;\r\n awaitingAssistantResponseRef.current = false;\r\n setStatusOverride(null);\r\n if (error instanceof Error) {\r\n appendMessage(`Hata: ${error.message}`, false);\r\n }\r\n } finally {\r\n finalizeStreamCycle(requestEpoch);\r\n }\r\n };\r\n\r\n const handleAudioBlob = async (blob: Blob) => {\r\n if (!config.projectId) {\r\n appendMessage(\"Hata: Project ID yapılandırılmamış.\", false);\r\n return;\r\n }\r\n\r\n const requestEpoch = beginRequestEpoch();\r\n setIsBusy(true);\r\n isBusyRef.current = true;\r\n setIsTranscribing(true);\r\n setIsThinking(false);\r\n setIsRenderingAudio(false);\r\n setIsPlayingAudio(false);\r\n setIsRunningTools(false);\r\n resetProcessingFlags(); // Start fresh\r\n setStatusOverride(STATUS_LABELS.thinking);\r\n awaitingAssistantResponseRef.current = true;\r\n\r\n try {\r\n const fileType = blob.type || \"audio/webm\";\r\n const extension = fileType.includes(\"ogg\")\r\n ? \"ogg\"\r\n : fileType.includes(\"wav\")\r\n ? \"wav\"\r\n : fileType.includes(\"mpeg\") || fileType.includes(\"mp3\")\r\n ? \"mp3\"\r\n : \"webm\";\r\n const file = new File([blob], `voice-input.${extension}`, {\r\n type: fileType,\r\n });\r\n\r\n stopActiveStream();\r\n\r\n const pageContext = getPageContext().summary;\r\n\r\n // Helper: bridge an AgentToolCallInfo to a ToolCallWithId\r\n const handleToolExecution = async (\r\n call: AgentToolCallInfo,\r\n ): Promise<{ call_id: string; result: string }> => {\r\n const toolCall: ToolCallWithId = {\r\n tool: call.tool as\r\n | \"navigate\"\r\n | \"getPageContext\"\r\n | \"interact\"\r\n | \"scroll\",\r\n call_id: call.call_id,\r\n ...call.args,\r\n } as ToolCallWithId;\r\n return executeSingleToolCall(toolCall);\r\n };\r\n\r\n const controller = agentVoiceChatStream(\r\n config.backendBaseUrl,\r\n file,\r\n config.projectId,\r\n sessionIdRef.current,\r\n {\r\n model: config.model,\r\n voice: config.voice,\r\n pageContext,\r\n accessibilityMode,\r\n },\r\n {\r\n onSttRequestSent: () => {\r\n if (!isCurrentRequestEpoch(requestEpoch)) return;\r\n playSttSentCueOnce();\r\n },\r\n onTranscription: (data) => {\r\n if (!isCurrentRequestEpoch(requestEpoch)) return;\r\n if (data.session_id && data.session_id !== sessionIdRef.current) {\r\n sessionIdRef.current = data.session_id;\r\n if (typeof localStorage !== \"undefined\") {\r\n localStorage.setItem(SESSION_ID_KEY, data.session_id);\r\n }\r\n }\r\n\r\n const normalized = data.user_text.trim();\r\n if (normalized) {\r\n const previousUserText = pendingUserTextRef.current;\r\n pendingUserTextRef.current = normalized;\r\n if (liveTranscriptionMessageIdRef.current !== null) {\r\n updateMessageText(liveTranscriptionMessageIdRef.current, normalized);\r\n clearLiveUserTranscriptionState();\r\n } else if (previousUserText !== normalized) {\r\n appendMessage(normalized, true);\r\n }\r\n }\r\n\r\n setIsTranscribing(false);\r\n setIsThinking(true);\r\n setStatusOverride(STATUS_LABELS.thinking);\r\n },\r\n onSessionId: (sid) => {\r\n if (!isCurrentRequestEpoch(requestEpoch)) return;\r\n if (sid && sid !== sessionIdRef.current) {\r\n sessionIdRef.current = sid;\r\n if (typeof localStorage !== \"undefined\") {\r\n localStorage.setItem(SESSION_ID_KEY, sid);\r\n }\r\n }\r\n },\r\n onAssistantDelta: (delta) => {\r\n if (!isCurrentRequestEpoch(requestEpoch)) return;\r\n setIsTranscribing(false);\r\n setIsThinking(true);\r\n setIsRunningTools(false);\r\n if (awaitingAssistantResponseRef.current) {\r\n awaitingAssistantResponseRef.current = false;\r\n setStatusOverride(null);\r\n }\r\n\r\n // Agent returns plain text (not JSON), stream it directly\r\n pendingAssistantTextRef.current += delta;\r\n\r\n if (assistantMessageIdRef.current === null) {\r\n assistantMessageIdRef.current = appendMessage(\r\n pendingAssistantTextRef.current,\r\n false,\r\n );\r\n } else {\r\n updateMessageText(\r\n assistantMessageIdRef.current,\r\n pendingAssistantTextRef.current,\r\n );\r\n }\r\n },\r\n onAssistantDone: (assistantText, expectsReply) => {\r\n if (!isCurrentRequestEpoch(requestEpoch)) return;\r\n playSfx(\"completed\");\r\n expectsReplyRef.current = expectsReply !== false;\r\n awaitingAssistantResponseRef.current = false;\r\n setStatusOverride(null);\r\n setIsThinking(false);\r\n setIsRenderingAudio(true);\r\n\r\n const finalDisplayText =\r\n assistantText || pendingAssistantTextRef.current;\r\n\r\n streamingJsonParserRef.current = null;\r\n pendingAssistantTextRef.current = finalDisplayText;\r\n\r\n if (assistantMessageIdRef.current !== null) {\r\n updateMessageText(\r\n assistantMessageIdRef.current,\r\n finalDisplayText,\r\n );\r\n } else {\r\n assistantMessageIdRef.current = appendMessage(\r\n finalDisplayText,\r\n false,\r\n );\r\n }\r\n },\r\n onIntermediateReply: (text) => {\r\n if (!isCurrentRequestEpoch(requestEpoch)) return;\r\n void speakText(\r\n config.backendBaseUrl, text, config.voice,\r\n accessibilityMode, (state) => handleAudioStateChange(state, requestEpoch),\r\n ).catch((err) => console.warn(\"[Bulut] intermediate TTS failed\", err));\r\n },\r\n onToolCalls: (calls) => {\r\n if (!isCurrentRequestEpoch(requestEpoch)) return;\r\n if (calls.length > 0) {\r\n playSfx(\"toolCall\");\r\n }\r\n setIsRunningTools(true);\r\n setStatusOverride(STATUS_LABELS.runningTools);\r\n appendToolIndicatorMessages(calls);\r\n assistantMessageIdRef.current = null;\r\n pendingAssistantTextRef.current = \"\";\r\n },\r\n onToolResult: (_callId, _toolName, _result) => {\r\n // Tool result sent back to agent automatically.\r\n // Could display detailed result here if needed.\r\n },\r\n onIteration: (_iteration, _maxIterations) => {\r\n if (!isCurrentRequestEpoch(requestEpoch)) return;\r\n // Agent started a new reasoning iteration\r\n playSfx(\"thinking\");\r\n setIsThinking(true);\r\n setStatusOverride(STATUS_LABELS.thinking);\r\n },\r\n onAudioStateChange: (state) => {\r\n handleAudioStateChange(state, requestEpoch);\r\n },\r\n onError: (err) => {\r\n if (!isCurrentRequestEpoch(requestEpoch)) return;\r\n awaitingAssistantResponseRef.current = false;\r\n setStatusOverride(null);\r\n appendMessage(`Hata: ${err}`, false);\r\n },\r\n },\r\n handleToolExecution,\r\n );\r\n\r\n activeStreamControllerRef.current = controller;\r\n await controller.done;\r\n\r\n // Safety: Ensure messages are flushed if not already\r\n // (e.g. if audio 'done' event didn't fire for some reason or there was no audio)\r\n /* if (pendingUserTextRef.current || pendingAssistantTextRef.current) {\r\n // flushPendingMessages(); // Removed as we stream now\r\n } */\r\n } catch (error) {\r\n if (!isCurrentRequestEpoch(requestEpoch)) return;\r\n // Error already shown via onError callback — don't duplicate\r\n awaitingAssistantResponseRef.current = false;\r\n setStatusOverride(null);\r\n } finally {\r\n finalizeStreamCycle(requestEpoch);\r\n }\r\n };\r\n\r\n const stopRecording = (options?: { discard?: boolean }) => {\r\n const recorder = recorderRef.current;\r\n if (!recorder || recorder.state === \"inactive\") {\r\n return;\r\n }\r\n\r\n if (options?.discard) {\r\n discardNextRecordingRef.current = true;\r\n }\r\n\r\n cleanupVAD();\r\n recorder.stop();\r\n };\r\n\r\n const setupVAD = (stream: MediaStream, recorder: MediaRecorder) => {\r\n const AudioCtx =\r\n window.AudioContext ||\r\n (window as Window & { webkitAudioContext?: typeof AudioContext })\r\n .webkitAudioContext;\r\n\r\n if (!AudioCtx) {\r\n return;\r\n }\r\n\r\n const context = new AudioCtx();\r\n audioContextRef.current = context;\r\n\r\n const analyser = context.createAnalyser();\r\n analyser.fftSize = 256;\r\n\r\n const source = context.createMediaStreamSource(stream);\r\n sourceRef.current = source;\r\n source.connect(analyser);\r\n\r\n const dataArray = new Uint8Array(analyser.frequencyBinCount);\r\n silenceStartRef.current = null;\r\n let speechDetected = false;\r\n let speechStartedAt: number | null = null;\r\n const enforceMinSpeechDuration = accessibilityMode;\r\n\r\n vadIntervalRef.current = window.setInterval(() => {\r\n if (!isRecordingRef.current || recorder.state === \"inactive\") {\r\n cleanupVAD();\r\n return;\r\n }\r\n\r\n analyser.getByteFrequencyData(dataArray);\r\n\r\n let sum = 0;\r\n for (const value of dataArray) {\r\n sum += value;\r\n }\r\n const average = sum / dataArray.length;\r\n const volume = average / 255;\r\n\r\n if (volume < VAD_THRESHOLD) {\r\n if (!speechDetected) {\r\n speechStartedAt = null;\r\n silenceStartRef.current = null;\r\n return;\r\n }\r\n\r\n if (silenceStartRef.current === null) {\r\n silenceStartRef.current = Date.now();\r\n return;\r\n }\r\n\r\n const silenceDuration = Date.now() - silenceStartRef.current;\r\n if (speechDetected && silenceDuration > SILENCE_DURATION_MS) {\r\n stopRecording();\r\n }\r\n return;\r\n }\r\n\r\n silenceStartRef.current = null;\r\n if (speechStartedAt === null) {\r\n speechStartedAt = Date.now();\r\n }\r\n\r\n if (!speechDetected) {\r\n const speechDuration = Date.now() - speechStartedAt;\r\n if (shouldAcceptVadSpeech(speechDuration, enforceMinSpeechDuration)) {\r\n speechDetected = true;\r\n if (enforceMinSpeechDuration) {\r\n setStatusOverride(STATUS_LABELS.listening);\r\n }\r\n }\r\n }\r\n }, 50);\r\n };\r\n\r\n const startRecording = async (mode: RecordingMode) => {\r\n if (\r\n isBusyRef.current ||\r\n isRecordingRef.current ||\r\n startRecordingPendingRef.current\r\n ) {\r\n return;\r\n }\r\n\r\n setStatusOverride(STATUS_LABELS.micInitializing);\r\n setIsMicPending(true);\r\n\r\n if (!navigator.mediaDevices?.getUserMedia) {\r\n setStatusOverride(null);\r\n setIsMicPending(false);\r\n appendMessage(\"Bu tarayıcıda mikrofon kullanılamıyor.\", false);\r\n return;\r\n }\r\n\r\n if (typeof MediaRecorder === \"undefined\") {\r\n setStatusOverride(null);\r\n setIsMicPending(false);\r\n appendMessage(\"Bu tarayıcıda MediaRecorder desteklenmiyor.\", false);\r\n return;\r\n }\r\n\r\n startRecordingPendingRef.current = true;\r\n\r\n try {\r\n const stream = await navigator.mediaDevices.getUserMedia({ audio: true });\r\n streamRef.current = stream;\r\n\r\n // Use low bitrate for speech — opus handles voice well at 16-24 kbps\r\n // and produces ~4-5x smaller payloads, speeding up the upload to fal.\r\n const recorderOptions: MediaRecorderOptions = {\r\n audioBitsPerSecond: 16_000,\r\n };\r\n\r\n // Prefer opus-in-ogg (smaller container) → opus-in-webm → browser default\r\n const preferredMimeTypes = [\r\n \"audio/ogg;codecs=opus\",\r\n \"audio/webm;codecs=opus\",\r\n \"audio/webm\",\r\n ];\r\n for (const mime of preferredMimeTypes) {\r\n if (MediaRecorder.isTypeSupported(mime)) {\r\n recorderOptions.mimeType = mime;\r\n break;\r\n }\r\n }\r\n\r\n const recorder = new MediaRecorder(stream, recorderOptions);\r\n recorderRef.current = recorder;\r\n audioChunksRef.current = [];\r\n clearLiveUserTranscriptionState();\r\n sttSendCuePlayedRef.current = false;\r\n\r\n const sttMimeType = (recorder.mimeType || recorderOptions.mimeType || \"audio/webm\")\r\n .split(\";\")[0]\r\n .trim() || \"audio/webm\";\r\n\r\n const sttWsController = startSttWebSocketStream(\r\n config.backendBaseUrl,\r\n {\r\n projectId: config.projectId,\r\n sessionId: sessionIdRef.current,\r\n language: \"tr\",\r\n mimeType: sttMimeType,\r\n },\r\n {\r\n onSessionId: (sid) => {\r\n if (!sid || sid === sessionIdRef.current) {\r\n return;\r\n }\r\n sessionIdRef.current = sid;\r\n if (typeof localStorage !== \"undefined\") {\r\n localStorage.setItem(SESSION_ID_KEY, sid);\r\n }\r\n },\r\n onPartial: ({ text }) => {\r\n if (!text.trim()) {\r\n return;\r\n }\r\n upsertLiveUserTranscription(text);\r\n },\r\n },\r\n );\r\n activeSttWsRef.current = sttWsController;\r\n\r\n recorder.ondataavailable = (event) => {\r\n if (event.data.size > 0) {\r\n audioChunksRef.current.push(event.data);\r\n if (activeSttWsRef.current) {\r\n void activeSttWsRef.current.pushChunk(event.data).catch((error) => {\r\n console.warn(\r\n `[Bulut] STT WS chunk send failed: ${error instanceof Error ? error.message : String(error)}`,\r\n );\r\n });\r\n }\r\n }\r\n };\r\n\r\n recorder.onerror = () => {\r\n appendMessage(\"Mikrofon kaydı sırasında bir hata oluştu.\", false);\r\n };\r\n\r\n recorder.onstop = async () => {\r\n setIsRecording(false);\r\n isRecordingRef.current = false;\r\n recordingModeRef.current = null;\r\n stopRecordingTimer();\r\n\r\n cleanupVAD();\r\n stopStreamTracks();\r\n\r\n const shouldDiscard = discardNextRecordingRef.current;\r\n discardNextRecordingRef.current = false;\r\n\r\n const blob = new Blob(audioChunksRef.current, {\r\n type: recorder.mimeType || \"audio/webm\",\r\n });\r\n audioChunksRef.current = [];\r\n\r\n const currentSttWs = activeSttWsRef.current;\r\n activeSttWsRef.current = null;\r\n\r\n if (shouldDiscard) {\r\n currentSttWs?.cancel();\r\n clearLiveUserTranscriptionState();\r\n setStatusOverride(null);\r\n return;\r\n }\r\n\r\n if (blob.size === 0) {\r\n currentSttWs?.cancel();\r\n clearLiveUserTranscriptionState();\r\n setStatusOverride(null);\r\n appendMessage(\"Ses kaydı alınamadı. Lütfen tekrar deneyin.\", false);\r\n return;\r\n }\r\n\r\n setIsTranscribing(true);\r\n setStatusOverride(STATUS_LABELS.transcribing);\r\n\r\n try {\r\n if (currentSttWs) {\r\n playSttSentCueOnce();\r\n const sttResult = await currentSttWs.stop();\r\n if (sttResult.session_id && sttResult.session_id !== sessionIdRef.current) {\r\n sessionIdRef.current = sttResult.session_id;\r\n if (typeof localStorage !== \"undefined\") {\r\n localStorage.setItem(SESSION_ID_KEY, sttResult.session_id);\r\n }\r\n }\r\n if (sttResult.text.trim()) {\r\n upsertLiveUserTranscription(sttResult.text);\r\n setStatusOverride(STATUS_LABELS.thinking);\r\n await runAgentForUserText(sttResult.text);\r\n return;\r\n }\r\n }\r\n } catch (error) {\r\n console.warn(\r\n `[Bulut] STT WS finalization failed, falling back to HTTP POST /chat/stt: ${error instanceof Error ? error.message : String(error)}`,\r\n );\r\n } finally {\r\n clearLiveUserTranscriptionState();\r\n }\r\n\r\n console.info(\"[Bulut] Using HTTP POST fallback for STT (streaming WS did not succeed)\");\r\n setStatusOverride(STATUS_LABELS.thinking);\r\n await handleAudioBlob(blob);\r\n };\r\n\r\n if (mode === \"vad\") {\r\n setupVAD(stream, recorder);\r\n }\r\n\r\n recorder.start(200);\r\n recordingModeRef.current = mode;\r\n setIsRecording(true);\r\n isRecordingRef.current = true;\r\n startRecordingTimer();\r\n\r\n setStatusOverride(\r\n accessibilityMode && mode === \"vad\"\r\n ? STATUS_LABELS.accessibilityActive\r\n : STATUS_LABELS.listening,\r\n );\r\n\r\n if (pendingStopAfterStartRef.current) {\r\n pendingStopAfterStartRef.current = false;\r\n stopRecording();\r\n }\r\n } catch (error) {\r\n const errMsg = normalizeError(error);\r\n if (errMsg.toLowerCase().includes(\"permission\") || errMsg.toLowerCase().includes(\"denied\")) {\r\n autoListenSuppressedRef.current = true;\r\n }\r\n cancelActiveSttWs();\r\n setStatusOverride(null);\r\n appendMessage(`Mikrofon hatası: ${errMsg}`, false);\r\n cleanupVAD();\r\n stopStreamTracks();\r\n pendingStopAfterStartRef.current = false;\r\n setIsRecording(false);\r\n isRecordingRef.current = false;\r\n stopRecordingTimer();\r\n } finally {\r\n if (!isRecordingRef.current && !isBusyRef.current) {\r\n setStatusOverride(null);\r\n }\r\n startRecordingPendingRef.current = false;\r\n setIsMicPending(false);\r\n }\r\n };\r\n\r\n const resetMicGesture = () => {\r\n micPressStartRef.current = null;\r\n micHoldTriggeredRef.current = false;\r\n clearMicHoldTimeout();\r\n };\r\n\r\n const handleMicPointerDown = (\r\n event: JSX.TargetedPointerEvent<HTMLButtonElement>,\r\n ) => {\r\n event.preventDefault();\r\n\r\n if (isBusyRef.current) {\r\n return;\r\n }\r\n\r\n if (isRecordingRef.current) {\r\n // In VAD mode, tapping the button cancels; in press mode, it sends\r\n if (recordingModeRef.current === \"vad\") {\r\n stopRecording({ discard: true });\r\n } else {\r\n stopRecording();\r\n }\r\n return;\r\n }\r\n\r\n micPressStartRef.current = Date.now();\r\n micHoldTriggeredRef.current = false;\r\n clearMicHoldTimeout();\r\n\r\n if (event.currentTarget.setPointerCapture) {\r\n try {\r\n event.currentTarget.setPointerCapture(event.pointerId);\r\n } catch {\r\n // No-op.\r\n }\r\n }\r\n\r\n micHoldTimeoutRef.current = window.setTimeout(() => {\r\n if (\r\n micPressStartRef.current === null ||\r\n isBusyRef.current ||\r\n isRecordingRef.current\r\n ) {\r\n return;\r\n }\r\n\r\n micHoldTriggeredRef.current = true;\r\n void startRecording(\"press\");\r\n }, HOLD_THRESHOLD_MS);\r\n };\r\n\r\n const handleMicPointerUp = (\r\n event: JSX.TargetedPointerEvent<HTMLButtonElement>,\r\n ) => {\r\n event.preventDefault();\r\n\r\n if (event.currentTarget.releasePointerCapture) {\r\n try {\r\n event.currentTarget.releasePointerCapture(event.pointerId);\r\n } catch {\r\n // No-op.\r\n }\r\n }\r\n\r\n const startedAt = micPressStartRef.current;\r\n const wasHold = micHoldTriggeredRef.current;\r\n resetMicGesture();\r\n\r\n if (startedAt === null) {\r\n return;\r\n }\r\n\r\n if (wasHold) {\r\n if (isRecordingRef.current) {\r\n stopRecording();\r\n } else if (startRecordingPendingRef.current) {\r\n pendingStopAfterStartRef.current = true;\r\n }\r\n return;\r\n }\r\n\r\n const duration = Date.now() - startedAt;\r\n if (classifyMicGesture(duration) === \"tap\") {\r\n void startRecording(\"vad\");\r\n }\r\n };\r\n\r\n const handleMicPointerCancel = (\r\n event: JSX.TargetedPointerEvent<HTMLButtonElement>,\r\n ) => {\r\n handleMicPointerUp(event);\r\n };\r\n\r\n const handleRestart = () => {\r\n invalidateRequestEpoch();\r\n sttSendCuePlayedRef.current = false;\r\n resetMicGesture();\r\n pendingStopAfterStartRef.current = false;\r\n\r\n stopActiveStream();\r\n stopActiveAudioPlayback();\r\n cancelActiveSttWs();\r\n\r\n if (recorderRef.current && recorderRef.current.state !== \"inactive\") {\r\n stopRecording({ discard: true });\r\n } else {\r\n discardNextRecordingRef.current = false;\r\n cleanupVAD();\r\n stopStreamTracks();\r\n }\r\n\r\n stopRecordingTimer();\r\n setRecordingDurationMs(0);\r\n\r\n clearPersistedChatState(\r\n typeof localStorage !== \"undefined\" ? localStorage : null,\r\n );\r\n\r\n sessionIdRef.current = null;\r\n const initialMessages = createInitialMessages(config.agentName);\r\n nextMessageIdRef.current = getNextMessageId(initialMessages);\r\n setMessages(initialMessages);\r\n\r\n setIsBusy(false);\r\n isBusyRef.current = false;\r\n setIsRecording(false);\r\n isRecordingRef.current = false;\r\n resetProcessingFlags();\r\n };\r\n\r\n // Auto-listen when accessibility mode is activated (initial trigger)\r\n useEffect(() => {\r\n if (!accessibilityMode || autoListenSuppressedRef.current) return;\r\n const timer = window.setTimeout(() => {\r\n if (!isRecordingRef.current && !isBusyRef.current && !startRecordingPendingRef.current && !autoListenSuppressedRef.current) {\r\n void startRecording(\"vad\");\r\n }\r\n }, 500);\r\n return () => window.clearTimeout(timer);\r\n }, [accessibilityMode]); // eslint-disable-line react-hooks/exhaustive-deps\r\n\r\n const stopTask = () => {\r\n invalidateRequestEpoch();\r\n sttSendCuePlayedRef.current = false;\r\n stopActiveStream();\r\n stopActiveAudioPlayback();\r\n cancelActiveSttWs();\r\n stopRecording({ discard: true });\r\n cleanupVAD();\r\n stopStreamTracks();\r\n resetProcessingFlags();\r\n setIsBusy(false);\r\n isBusyRef.current = false;\r\n };\r\n\r\n // Expose recording actions to parent via actionsRef\r\n if (actionsRef) {\r\n actionsRef.current = {\r\n startRecording: () => {\r\n autoListenSuppressedRef.current = false;\r\n void startRecording(\"vad\");\r\n },\r\n cancelRecording: () => {\r\n stopActiveAudioPlayback();\r\n cancelActiveSttWs();\r\n const recorder = recorderRef.current;\r\n if (recorder && recorder.state !== \"inactive\") {\r\n stopRecording({ discard: true });\r\n } else {\r\n cleanupVAD();\r\n stopStreamTracks();\r\n }\r\n },\r\n stopTask,\r\n };\r\n }\r\n\r\n const windowStyle: { [key: string]: string } = {\r\n position: \"fixed\",\r\n bottom: `${POSITION_BOTTOM}px`,\r\n right: `${POSITION_RIGHT}px`,\r\n width: `${WINDOW_WIDTH}px`,\r\n maxHeight: `${WINDOW_HEIGHT}px`,\r\n backgroundColor: \"hsla(0, 0%, 100%, 1)\",\r\n borderRadius: BORDER_RADIUS.window,\r\n display: hidden ? \"none\" : \"flex\",\r\n flexDirection: \"column\",\r\n overflow: \"hidden\",\r\n zIndex: \"10000\",\r\n animation: hidden ? \"none\" : `slideIn ${TRANSITIONS.medium}`,\r\n boxShadow: accessibilityMode\r\n ? `inset 0 0 0 2px ${COLORS.primary}, ${SHADOW}`\r\n : SHADOW,\r\n fontFamily: \"\\\"Geist\\\", sans-serif\",\r\n };\r\n\r\n const headerStyle: { [key: string]: string } = {\r\n padding: \"14px 16px\",\r\n display: \"flex\",\r\n justifyContent: \"space-between\",\r\n alignItems: \"center\",\r\n };\r\n\r\n const headerActionsStyle: { [key: string]: string } = {\r\n display: \"flex\",\r\n alignItems: \"center\",\r\n gap: \"6px\",\r\n };\r\n\r\n const headerButtonStyle: { [key: string]: string } = {\r\n background: \"none\",\r\n border: \"none\",\r\n cursor: \"pointer\",\r\n padding: \"4px\",\r\n borderRadius: \"6px\",\r\n color: COLORS.text,\r\n display: \"flex\",\r\n alignItems: \"center\",\r\n justifyContent: \"center\",\r\n transition: `color ${TRANSITIONS.fast}, background-color ${TRANSITIONS.fast}`,\r\n };\r\n\r\n const messagesContainerStyle: { [key: string]: string } = {\r\n padding: \"0px 16px\",\r\n overflowY: \"auto\",\r\n flex: \"1\",\r\n minHeight: \"0\",\r\n };\r\n\r\n const messagesListStyle: { [key: string]: string } = {\r\n display: \"flex\",\r\n flexDirection: \"column\",\r\n gap: \"16px\",\r\n };\r\n\r\n const messageStyle = (isUser: boolean): JSX.CSSProperties => ({\r\n maxWidth: \"84%\",\r\n padding: isUser ? \"9px 14px\" : \"0px 0px\",\r\n borderRadius: BORDER_RADIUS.message,\r\n fontSize: \"14px\",\r\n lineHeight: \"140%\",\r\n wordWrap: \"break-word\",\r\n whiteSpace: \"pre-wrap\",\r\n alignSelf: isUser ? \"flex-end\" : \"flex-start\",\r\n backgroundColor: isUser ? COLORS.messageUser : \"\",\r\n color: isUser ? COLORS.messageUserText : \"hsla(215, 100%, 5%, 1)\",\r\n });\r\n\r\n const resolveToolIcon = (kind: Message[\"toolKind\"]) => {\r\n if (kind === \"cursor\") {\r\n return CursorArrowRaysIcon;\r\n }\r\n if (kind === \"scroll\") {\r\n return HandRaisedIcon;\r\n }\r\n if (kind === \"navigate\") {\r\n return MapIcon;\r\n }\r\n if (kind === \"form\") {\r\n return QueueListIcon;\r\n }\r\n if (kind === \"interact\") {\r\n return HandRaisedIcon;\r\n }\r\n if (kind === \"unknown\") {\r\n return CommandLineIcon;\r\n }\r\n return FaceSmileIcon;\r\n };\r\n\r\n const footerStyle: { [key: string]: string } = {\r\n padding: \"10px 12px\",\r\n display: \"flex\",\r\n alignItems: \"center\",\r\n justifyContent: \"space-between\",\r\n gap: \"8px\",\r\n };\r\n\r\n const statusPanelStyle: { [key: string]: string } = {\r\n flex: \"1\",\r\n minHeight: \"34px\",\r\n borderRadius: \"10px\",\r\n color: COLORS.text,\r\n fontSize: \"14px\",\r\n display: \"flex\",\r\n alignItems: \"center\",\r\n padding: \"0 10px\",\r\n whiteSpace: \"nowrap\",\r\n overflow: \"hidden\",\r\n textOverflow: \"ellipsis\",\r\n opacity: \"0.7\",\r\n };\r\n\r\n const footerActionsStyle: { [key: string]: string } = {\r\n display: \"flex\",\r\n alignItems: \"center\",\r\n gap: \"8px\",\r\n flexShrink: \"0\",\r\n };\r\n\r\n const recordingTimerStyle: { [key: string]: string } = {\r\n minWidth: \"46px\",\r\n fontSize: \"12px\",\r\n fontWeight: \"700\",\r\n color: COLORS.text,\r\n textAlign: \"right\",\r\n };\r\n\r\n const micFooterButtonStyle: { [key: string]: string } = {\r\n width: \"37px\",\r\n height: \"37px\",\r\n borderRadius: \"999px\",\r\n background: \"transparent\",\r\n display: \"flex\",\r\n alignItems: \"center\",\r\n justifyContent: \"center\",\r\n cursor: \"pointer\",\r\n color: \"#ffffff\",\r\n border: \"1px solid hsla(215, 100%, 5%, 0.5)\",\r\n transition: `transform ${TRANSITIONS.fast}`,\r\n };\r\n\r\n const isVadRecording = isRecording && recordingModeRef.current === \"vad\";\r\n const showStopButton = isBusy && !isRecording;\r\n const hideMicButton = isMicPending && !isRecording;\r\n const disableMicControl = isBusy;\r\n\r\n return (\r\n <div className=\"bulut-chat-window\" style={windowStyle}>\r\n <style>{`\r\n @keyframes slideIn {\r\n from {\r\n opacity: 0;\r\n transform: translateY(10px);\r\n }\r\n to {\r\n opacity: 1;\r\n transform: translateY(0);\r\n }\r\n }\r\n\r\n .bulut-header-btn:hover:not(:disabled) {\r\n color: ${COLORS.text};\r\n }\r\n\r\n .bulut-footer-btn:hover:not(:disabled) {\r\n transform: scale(1.04);\r\n }\r\n\r\n .bulut-header-btn:disabled,\r\n .bulut-footer-btn:disabled {\r\n cursor: not-allowed;\r\n opacity: 0.5;\r\n transform: none;\r\n }\r\n\r\n @keyframes bulut-dots {\r\n 0% { content: '.'; }\r\n 33% { content: '..'; }\r\n 66% { content: '...'; }\r\n }\r\n\r\n .bulut-status-dots::after {\r\n content: '.';\r\n animation: bulut-dots 1.2s steps(1) infinite;\r\n display: inline-block;\r\n min-width: 12px;\r\n text-align: left;\r\n }\r\n\r\n /* Mobile: full-screen chat window */\r\n @media (max-width: 600px) {\r\n .bulut-chat-window {\r\n top: 0 !important;\r\n left: 0 !important;\r\n right: 0 !important;\r\n bottom: 0 !important;\r\n width: 100% !important;\r\n max-height: 100% !important;\r\n height: 100% !important;\r\n border-radius: 0 !important;\r\n }\r\n .bulut-close-btn {\r\n width: 40px !important;\r\n height: 40px !important;\r\n padding: 8px !important;\r\n }\r\n .bulut-close-btn svg {\r\n width: 26px !important;\r\n height: 26px !important;\r\n }\r\n }\r\n `}</style>\r\n\r\n <div style={headerStyle}>\r\n <SvgIcon\r\n src={logoContent}\r\n title=\"Bulut Logo\"\r\n style={{ maxWidth: \"80px\", height: \"auto\" }}\r\n stripColors={false}\r\n />\r\n <div style={headerActionsStyle}>\r\n <button\r\n type=\"button\"\r\n className=\"bulut-header-btn\"\r\n style={headerButtonStyle}\r\n onClick={handleRestart}\r\n aria-label=\"Sohbeti yeniden başlat\"\r\n title=\"Sohbeti yeniden başlat\"\r\n >\r\n <ArrowPathIcon aria-hidden=\"true\" width={22} height={22} />\r\n </button>\r\n\r\n <button\r\n type=\"button\"\r\n className=\"bulut-header-btn bulut-close-btn\"\r\n style={{\r\n ...headerButtonStyle,\r\n display: \"flex\",\r\n alignItems: \"center\",\r\n justifyContent: \"center\",\r\n }}\r\n onClick={onClose}\r\n aria-label=\"Sohbeti kapat\"\r\n title=\"Sohbeti kapat\"\r\n >\r\n <XMarkIcon aria-hidden=\"true\" width={22} height={22} />\r\n </button>\r\n </div>\r\n </div>\r\n\r\n <div style={messagesContainerStyle} ref={messagesContainerRef}>\r\n <div style={messagesListStyle} ref={messagesContentRef}>\r\n {messages.map((message) => {\r\n if (message.type === \"tool\") {\r\n const ToolIcon = resolveToolIcon(message.toolKind);\r\n return (\r\n <div\r\n key={message.id}\r\n style={{\r\n ...messageStyle(false),\r\n opacity: \"0.7\",\r\n display: \"flex\",\r\n alignItems: \"center\",\r\n gap: \"8px\",\r\n }}\r\n >\r\n <ToolIcon\r\n aria-hidden=\"true\"\r\n width={20}\r\n height={20}\r\n style={{ flexShrink: \"0\" }}\r\n />\r\n <span>{message.text}</span>\r\n </div>\r\n );\r\n }\r\n\r\n return (\r\n <div key={message.id} style={messageStyle(message.isUser)}>\r\n {message.text}\r\n </div>\r\n );\r\n })}\r\n </div>\r\n </div>\r\n\r\n <div style={footerStyle}>\r\n <div style={{ ...statusPanelStyle, transition: \"opacity 0.2s ease-out\" }}>\r\n {showStatus ? (\r\n <span className=\"bulut-status-dots\" title={statusText}>\r\n {statusText}\r\n </span>\r\n ) : onAccessibilityToggle ? (\r\n <div\r\n style={{\r\n display: \"flex\",\r\n alignItems: \"center\",\r\n gap: \"8px\",\r\n }}\r\n >\r\n <span\r\n style={{\r\n fontSize: \"12px\",\r\n opacity: \"0.6\",\r\n whiteSpace: \"nowrap\",\r\n }}\r\n >\r\n Erişilebilirlik\r\n </span>\r\n <button\r\n type=\"button\"\r\n onClick={onAccessibilityToggle}\r\n aria-label={\r\n accessibilityMode\r\n ? \"Erişilebilirlik modunu kapat\"\r\n : \"Erişilebilirlik modunu aç\"\r\n }\r\n style={{\r\n width: \"36px\",\r\n height: \"20px\",\r\n borderRadius: \"10px\",\r\n border: \"none\",\r\n cursor: \"pointer\",\r\n padding: \"2px\",\r\n display: \"flex\",\r\n alignItems: \"center\",\r\n backgroundColor: accessibilityMode\r\n ? COLORS.primary\r\n : \"hsla(215, 10%, 75%, 1)\",\r\n transition: `background-color ${TRANSITIONS.fast}`,\r\n flexShrink: \"0\",\r\n }}\r\n >\r\n <span\r\n style={{\r\n width: \"16px\",\r\n height: \"16px\",\r\n borderRadius: \"50%\",\r\n backgroundColor: \"#ffffff\",\r\n display: \"block\",\r\n transition: `transform ${TRANSITIONS.fast}`,\r\n transform: accessibilityMode\r\n ? \"translateX(16px)\"\r\n : \"translateX(0)\",\r\n boxShadow: \"0 1px 3px rgba(0,0,0,0.2)\",\r\n }}\r\n />\r\n </button>\r\n </div>\r\n ) : null}\r\n </div>\r\n\r\n <div style={footerActionsStyle}>\r\n {isRecording ? (\r\n <span style={recordingTimerStyle}>\r\n {formatDurationMs(recordingDurationMs)}\r\n </span>\r\n ) : null}\r\n {showStopButton ? (\r\n <button\r\n type=\"button\"\r\n className=\"bulut-footer-btn\"\r\n style={micFooterButtonStyle}\r\n onClick={stopTask}\r\n aria-label=\"Görevi durdur\"\r\n title=\"Görevi durdur\"\r\n >\r\n <StopIcon\r\n aria-hidden=\"true\"\r\n width={22}\r\n height={22}\r\n style={{ color: \"hsla(215, 100%, 5%, 1)\" }}\r\n />\r\n </button>\r\n ) : hideMicButton ? null : (\r\n <button\r\n type=\"button\"\r\n className=\"bulut-footer-btn\"\r\n style={micFooterButtonStyle}\r\n onPointerDown={handleMicPointerDown}\r\n onPointerUp={handleMicPointerUp}\r\n onPointerCancel={handleMicPointerCancel}\r\n disabled={disableMicControl}\r\n aria-label={isVadRecording ? \"Kaydı iptal et\" : isRecording ? \"Kaydı durdur\" : \"Kaydı başlat\"}\r\n title={\r\n isVadRecording\r\n ? \"Kaydı iptal et\"\r\n : isRecording\r\n ? \"Kaydı durdur\"\r\n : \"Dokun: VAD, Basılı tut: bırakınca gönder\"\r\n }\r\n >\r\n {isVadRecording ? (\r\n <XMarkIcon\r\n aria-hidden=\"true\"\r\n width={22}\r\n height={22}\r\n style={{ color: \"hsla(215, 100%, 5%, 1)\" }}\r\n />\r\n ) : (\r\n <MicrophoneIcon\r\n aria-hidden=\"true\"\r\n width={22}\r\n height={22}\r\n style={{ color: \"hsla(215, 100%, 5%, 1)\" }}\r\n />\r\n )}\r\n </button>\r\n )}\r\n </div>\r\n </div>\r\n </div>\r\n );\r\n};\r\n","export default \"<svg width=\\\"2420\\\" height=\\\"438\\\" viewBox=\\\"0 0 2420 438\\\" fill=\\\"none\\\" xmlns=\\\"http://www.w3.org/2000/svg\\\">\\r\\n<path d=\\\"M251.823 0C313.458 0 366.34 37.767 388.932 91.5918C409.253 75.8883 434.678 66.5567 462.264 66.5566C528.885 66.5567 582.893 120.989 582.893 188.134C582.893 188.786 582.885 189.436 582.875 190.086C648.144 193.552 700 247.572 700 313.704C700 382.074 644.575 437.5 576.204 437.5H123.796C55.4255 437.5 7.42386e-05 382.074 0 313.704C0 250.475 47.4025 198.32 108.608 190.833C104.967 177.844 103.019 164.138 103.019 149.975C103.019 67.1461 169.641 0.000357483 251.823 0ZM234.696 141.93C218.117 134.066 199 146.158 199 164.507C199 174.32 204.744 183.225 213.684 187.271L336.761 242.985C337.971 243.533 338.748 244.738 338.748 246.066C338.748 247.394 337.971 248.6 336.761 249.147L213.684 304.861C204.744 308.908 199 317.812 199 327.625C199 345.974 218.117 358.066 234.696 350.202L364.672 288.554C375.142 283.588 381.815 273.036 381.815 261.448V230.684C381.815 219.096 375.142 208.544 364.672 203.578L234.696 141.93ZM413.877 296.146C402.078 296.146 392.513 305.711 392.513 317.511C392.513 329.31 402.078 338.875 413.877 338.875H526.636C538.435 338.875 548 329.31 548 317.511C548 305.711 538.435 296.146 526.636 296.146H413.877Z\\\" fill=\\\"#000B1A\\\"/>\\r\\n<path d=\\\"M2420 424.954H2343.35C2307.08 424.954 2278.64 416.094 2258.04 398.373C2237.43 380.652 2227.13 352.216 2227.13 313.065V55.2871H2319.86V302.556C2319.86 318.217 2323.15 328.726 2329.75 334.083C2336.75 339.028 2348.29 341.501 2364.36 341.501H2420V424.954ZM2420 191.903H2178.29V115.25H2420V191.903Z\\\" fill=\\\"#000B1A\\\"/>\\r\\n<path d=\\\"M1923.18 431.136C1894.33 431.136 1870.22 425.779 1850.85 415.064C1831.89 403.937 1817.68 389.101 1808.2 370.555C1799.13 352.01 1794.6 331.198 1794.6 308.12V115.25H1887.32V277.211C1887.32 301.938 1893.3 320.071 1905.25 331.61C1917.2 342.738 1937.81 348.301 1967.07 348.301C1997.56 348.301 2018.79 342.325 2030.74 330.374C2043.1 318.011 2049.28 298.641 2049.28 272.266L2061.03 271.648L2067.83 330.374H2050.52C2048.05 347.683 2041.87 363.962 2031.98 379.21C2022.5 394.458 2008.69 407.028 1990.56 416.918C1972.84 426.397 1950.38 431.136 1923.18 431.136ZM2142.01 424.955H2055.47V329.138L2049.28 326.047V115.25H2142.01V424.955Z\\\" fill=\\\"#000B1A\\\"/>\\r\\n<path d=\\\"M1734.32 424.955H1641.59V10.7793H1734.32V424.955Z\\\" fill=\\\"#000B1A\\\"/>\\r\\n<path d=\\\"M1362.98 431.136C1334.13 431.136 1310.03 425.779 1290.66 415.064C1271.7 403.937 1257.48 389.101 1248 370.555C1238.94 352.01 1234.4 331.198 1234.4 308.12V115.25H1327.13V277.211C1327.13 301.938 1333.1 320.071 1345.06 331.61C1357.01 342.738 1377.61 348.301 1406.87 348.301C1437.37 348.301 1458.59 342.325 1470.54 330.374C1482.91 318.011 1489.09 298.641 1489.09 272.266L1500.83 271.648L1507.63 330.374H1490.33C1487.85 347.683 1481.67 363.962 1471.78 379.21C1462.3 394.458 1448.5 407.028 1430.36 416.918C1412.64 426.397 1390.18 431.136 1362.98 431.136ZM1581.82 424.955H1495.27V329.138L1489.09 326.047V115.25H1581.82V424.955Z\\\" fill=\\\"#000B1A\\\"/>\\r\\n<path d=\\\"M1035.12 431.137C999.27 431.137 971.04 423.513 950.434 408.264C930.241 392.604 917.259 369.526 911.49 339.029H894.181L900.981 272.266H912.726C912.726 289.987 916.023 304.411 922.617 315.538C929.623 326.253 939.719 334.084 952.907 339.029C966.507 343.974 983.198 346.447 1002.98 346.447C1023.17 346.447 1039.66 343.974 1052.43 339.029C1065.21 334.084 1074.69 326.047 1080.87 314.92C1087.05 303.793 1090.14 288.957 1090.14 270.412C1090.14 251.042 1087.05 236 1080.87 225.285C1074.69 214.158 1065.21 206.122 1052.43 201.177C1039.66 196.231 1023.38 193.758 1003.6 193.758C973.513 193.758 950.847 199.322 935.598 210.449C920.35 221.576 912.726 240.533 912.726 267.321H900.981V197.467H917.671C923.029 171.504 935.392 150.28 954.762 133.796C974.543 117.311 1002.77 109.069 1039.45 109.069C1070.36 109.069 1096.53 115.663 1117.96 128.85C1139.39 142.038 1155.67 160.789 1166.79 185.104C1178.33 209.007 1184.1 237.443 1184.1 270.412C1184.1 302.969 1178.33 331.405 1166.79 355.72C1155.67 379.622 1138.98 398.168 1116.72 411.355C1094.47 424.543 1067.27 431.137 1035.12 431.137ZM905.926 424.955H820V10.7793H912.726V325.429L905.926 333.465V424.955Z\\\" fill=\\\"#000B1A\\\"/>\\r\\n</svg>\\r\\n\"","import { useState, useEffect, useRef, useCallback } from \"preact/hooks\";\r\nimport \"./globals.css\";\r\nimport { render } from \"preact\";\r\nimport { ChatButton } from \"./components/ChatButton\";\r\nimport {\r\n ChatWindow,\r\n type ChatWindowHandle,\r\n} from \"./components/ChatWindow\";\r\nimport { COLORS } from \"./styles/constants\";\r\n\r\nexport type BulutVoice = \"alloy\" | \"zeynep\" | \"ali\";\r\n\r\nexport interface BulutOptions {\r\n containerId?: string;\r\n backendBaseUrl?: string;\r\n projectId?: string;\r\n}\r\n\r\nexport interface BulutRuntimeConfig {\r\n backendBaseUrl: string;\r\n projectId: string;\r\n model: string;\r\n voice: BulutVoice;\r\n baseColor: string;\r\n agentName: string;\r\n}\r\n\r\n/** Default LLM model — keep in sync with backend config.DEFAULT_LLM_MODEL */\r\nconst DEFAULT_LLM_MODEL = \"x-ai/grok-4.1-fast\";\r\n\r\nconst DEFAULT_AGENT_NAME = \"Bulut\";\r\n\r\nconst DEFAULT_CONFIG: BulutRuntimeConfig = {\r\n backendBaseUrl: \"https://api.bulut.lu\",\r\n projectId: \"\", // Must be provided\r\n model: DEFAULT_LLM_MODEL,\r\n voice: \"alloy\",\r\n baseColor: COLORS.primary,\r\n agentName: DEFAULT_AGENT_NAME,\r\n};\r\n\r\nconst isValidHexColor = (value: string): boolean =>\r\n /^#([0-9a-fA-F]{3}|[0-9a-fA-F]{6})$/.test(value);\r\n\r\nconst normalizeHexColor = (value: string): string => {\r\n const trimmed = value.trim();\r\n if (!isValidHexColor(trimmed)) {\r\n return DEFAULT_CONFIG.baseColor;\r\n }\r\n if (trimmed.length === 4) {\r\n const r = trimmed[1];\r\n const g = trimmed[2];\r\n const b = trimmed[3];\r\n return `#${r}${r}${g}${g}${b}${b}`.toLowerCase();\r\n }\r\n return trimmed.toLowerCase();\r\n};\r\n\r\nconst shadeHexColor = (hexColor: string, amount: number): string => {\r\n const normalized = normalizeHexColor(hexColor);\r\n const raw = normalized.slice(1);\r\n const toChannel = (start: number): number => parseInt(raw.slice(start, start + 2), 16);\r\n const clamp = (value: number): number => Math.max(0, Math.min(255, Math.round(value)));\r\n const adjust = (channel: number): number =>\r\n amount < 0 ? channel * (1 + amount) : channel + (255 - channel) * amount;\r\n const toHex = (channel: number): string => clamp(channel).toString(16).padStart(2, \"0\");\r\n\r\n const r = adjust(toChannel(0));\r\n const g = adjust(toChannel(2));\r\n const b = adjust(toChannel(4));\r\n\r\n return `#${toHex(r)}${toHex(g)}${toHex(b)}`;\r\n};\r\n\r\nconst applyTheme = (baseColor: string): void => {\r\n const normalized = normalizeHexColor(baseColor);\r\n COLORS.primary = normalized;\r\n COLORS.primaryHover = shadeHexColor(normalized, -0.15);\r\n COLORS.messageUser = normalized;\r\n};\r\n\r\ninterface RemoteProjectConfig {\r\n base_color: string;\r\n model: string;\r\n agent_name: string;\r\n voice: string;\r\n}\r\n\r\nconst fetchRemoteConfig = async (\r\n baseUrl: string,\r\n projectId: string,\r\n): Promise<RemoteProjectConfig | null> => {\r\n try {\r\n const url = baseUrl.replace(/\\/+$/, \"\");\r\n const res = await fetch(`${url}/projects/${projectId}/config`);\r\n if (!res.ok) return null;\r\n return (await res.json()) as RemoteProjectConfig;\r\n } catch {\r\n return null;\r\n }\r\n};\r\n\r\nconst resolveRuntimeConfig = (\r\n options: BulutOptions,\r\n): BulutRuntimeConfig => ({\r\n backendBaseUrl: options.backendBaseUrl || DEFAULT_CONFIG.backendBaseUrl,\r\n projectId: options.projectId || DEFAULT_CONFIG.projectId,\r\n model: DEFAULT_CONFIG.model,\r\n voice: DEFAULT_CONFIG.voice,\r\n baseColor: DEFAULT_CONFIG.baseColor,\r\n agentName: DEFAULT_CONFIG.agentName,\r\n});\r\n\r\ninterface BulutWidgetProps {\r\n config: BulutRuntimeConfig;\r\n}\r\n\r\nconst ACCESSIBILITY_MODE_KEY = \"bulut_accessibility_mode_enabled\";\r\nconst GEIST_FONT_FAMILY = \"Geist\";\r\nconst GEIST_STYLESHEET_ID = \"bulut-geist-font-stylesheet\";\r\nconst GEIST_STYLESHEET_URL =\r\n \"https://fonts.googleapis.com/css2?family=Geist:wght@100..900&display=swap\";\r\n\r\nconst ensureGeistStylesheet = (): void => {\r\n if (typeof document === \"undefined\") {\r\n return;\r\n }\r\n if (document.getElementById(GEIST_STYLESHEET_ID)) {\r\n return;\r\n }\r\n const link = document.createElement(\"link\");\r\n link.id = GEIST_STYLESHEET_ID;\r\n link.rel = \"stylesheet\";\r\n link.href = GEIST_STYLESHEET_URL;\r\n document.head.appendChild(link);\r\n};\r\n\r\nconst BulutWidget = ({ config }: BulutWidgetProps) => {\r\n // Live config that merges remote settings over initial config\r\n const [liveConfig, setLiveConfig] = useState<BulutRuntimeConfig>(config);\r\n const [configReady, setConfigReady] = useState(false);\r\n\r\n // Fetch remote project config on mount — widget stays hidden until done\r\n useEffect(() => {\r\n if (!config.projectId) {\r\n setConfigReady(true);\r\n return;\r\n }\r\n let cancelled = false;\r\n\r\n fetchRemoteConfig(config.backendBaseUrl, config.projectId).then((remote) => {\r\n if (cancelled) return;\r\n if (remote) {\r\n const merged: BulutRuntimeConfig = {\r\n ...config,\r\n baseColor: normalizeHexColor(remote.base_color || config.baseColor),\r\n model: remote.model || config.model,\r\n agentName: remote.agent_name || config.agentName,\r\n voice: (\r\n remote.voice === \"alloy\" || remote.voice === \"zeynep\" || remote.voice === \"ali\"\r\n ? remote.voice\r\n : config.voice\r\n ) as BulutVoice,\r\n };\r\n applyTheme(merged.baseColor);\r\n setLiveConfig(merged);\r\n }\r\n setConfigReady(true);\r\n });\r\n\r\n return () => { cancelled = true; };\r\n }, [config]);\r\n\r\n // Check localStorage for persisted state\r\n const [isOpen, setIsOpen] = useState(() => {\r\n if (typeof localStorage !== \"undefined\") {\r\n return localStorage.getItem(\"bulut_panel_open\") === \"true\";\r\n }\r\n return false;\r\n });\r\n\r\n const [showBubble, setShowBubble] = useState(false);\r\n const [isAccessibilityEnabled, setIsAccessibilityEnabled] = useState(() => {\r\n if (typeof localStorage === \"undefined\") {\r\n return false;\r\n }\r\n return localStorage.getItem(ACCESSIBILITY_MODE_KEY) === \"true\";\r\n });\r\n\r\n // State reported by ChatWindow\r\n const [isRecording, setIsRecording] = useState(false);\r\n const [isBusy, setIsBusy] = useState(false);\r\n const [previewMessage, setPreviewMessage] = useState<string | null>(null);\r\n const [previewDismissed, setPreviewDismissed] = useState(false);\r\n\r\n // Ref for delegating recording to ChatWindow\r\n const chatActionsRef = useRef<ChatWindowHandle | null>(null);\r\n\r\n const handlePreviewChange = useCallback((text: string | null) => {\r\n setPreviewMessage(text);\r\n if (text !== null) setPreviewDismissed(false);\r\n }, []);\r\n\r\n // Show welcome bubble once for 5 seconds\r\n useEffect(() => {\r\n if (isAccessibilityEnabled) {\r\n setShowBubble(false);\r\n return;\r\n }\r\n if (isOpen) return;\r\n if (typeof localStorage !== \"undefined\") {\r\n if (localStorage.getItem(\"bulut_bubble_shown\") === \"true\") return;\r\n }\r\n\r\n setShowBubble(true);\r\n const timer = setTimeout(() => {\r\n setShowBubble(false);\r\n if (typeof localStorage !== \"undefined\") {\r\n localStorage.setItem(\"bulut_bubble_shown\", \"true\");\r\n }\r\n }, 5000);\r\n return () => clearTimeout(timer);\r\n }, [isOpen, isAccessibilityEnabled]);\r\n\r\n const toggleWidget = () => {\r\n const newState = !isOpen;\r\n setIsOpen(newState);\r\n setShowBubble(false);\r\n if (typeof localStorage !== \"undefined\") {\r\n localStorage.setItem(\"bulut_panel_open\", String(newState));\r\n }\r\n };\r\n\r\n const toggleAccessibilityMode = () => {\r\n const next = !isAccessibilityEnabled;\r\n setIsAccessibilityEnabled(next);\r\n if (typeof localStorage !== \"undefined\") {\r\n localStorage.setItem(ACCESSIBILITY_MODE_KEY, String(next));\r\n }\r\n console.info(`[Bulut] accessibility mode toggled enabled=${next}`);\r\n };\r\n\r\n const handleClose = () => {\r\n setIsOpen(false);\r\n if (typeof localStorage !== \"undefined\") {\r\n localStorage.setItem(\"bulut_panel_open\", \"false\");\r\n }\r\n console.info(\"Bulut chat window closed.\");\r\n };\r\n\r\n // Close on escape key\r\n useEffect(() => {\r\n const handleEscape = (e: KeyboardEvent) => {\r\n if (e.key === \"Escape\" && isOpen) {\r\n setIsOpen(false);\r\n if (typeof localStorage !== \"undefined\") {\r\n localStorage.setItem(\"bulut_panel_open\", \"false\");\r\n }\r\n }\r\n };\r\n\r\n document.addEventListener(\"keydown\", handleEscape);\r\n return () => document.removeEventListener(\"keydown\", handleEscape);\r\n }, [isOpen]);\r\n\r\n if (!configReady) return null;\r\n\r\n return (\r\n <>\r\n {!isOpen && (\r\n <ChatButton\r\n onMicClick={() => chatActionsRef.current?.startRecording()}\r\n onCancelRecording={() => chatActionsRef.current?.cancelRecording()}\r\n onStopTask={() => chatActionsRef.current?.stopTask()}\r\n isRecording={isRecording}\r\n isBusy={isBusy}\r\n accessibilityMode={isAccessibilityEnabled}\r\n showBubble={showBubble}\r\n onBubbleClick={() => {\r\n setShowBubble(false);\r\n toggleWidget();\r\n }}\r\n previewMessage={previewDismissed ? null : previewMessage}\r\n onPreviewClick={() => toggleWidget()}\r\n onPreviewClose={() => setPreviewDismissed(true)}\r\n />\r\n )}\r\n <ChatWindow\r\n onClose={handleClose}\r\n config={liveConfig}\r\n accessibilityMode={isAccessibilityEnabled}\r\n onAccessibilityToggle={toggleAccessibilityMode}\r\n hidden={!isOpen}\r\n actionsRef={chatActionsRef}\r\n onRecordingChange={setIsRecording}\r\n onBusyChange={setIsBusy}\r\n onPreviewChange={handlePreviewChange}\r\n />\r\n </>\r\n );\r\n};\r\n\r\nconst SHADOW_STYLE = `\r\n :host {\r\n all: initial;\r\n contain: layout style paint;\r\n font-family: \"${GEIST_FONT_FAMILY}\", sans-serif;\r\n color: hsla(215, 100%, 5%, 1);\r\n font-size: 16px;\r\n line-height: 1.4;\r\n -webkit-font-smoothing: antialiased;\r\n text-rendering: optimizeLegibility;\r\n }\r\n\r\n #bulut-shadow-mount {\r\n all: initial;\r\n color: inherit;\r\n font-family: \"${GEIST_FONT_FAMILY}\", sans-serif;\r\n font-size: inherit;\r\n line-height: inherit;\r\n }\r\n\r\n #bulut-shadow-mount * {\r\n font-family: \"${GEIST_FONT_FAMILY}\", sans-serif !important;\r\n color: inherit;\r\n }\r\n\r\n #bulut-shadow-mount *, #bulut-shadow-mount *::before, #bulut-shadow-mount *::after {\r\n box-sizing: border-box;\r\n }\r\n`;\r\n\r\n// Container host for the widget\r\nlet widgetContainer: HTMLElement | null = null;\r\nlet widgetMountNode: HTMLElement | null = null;\r\nlet createdContainer = false;\r\nlet isInitialized = false;\r\n\r\n/**\r\n * Initialize the Bulut chat widget\r\n * @param options - Optional configuration options\r\n */\r\nexport const init = (options: BulutOptions = {}) => {\r\n if (isInitialized) {\r\n console.warn(\"Bulut is already initialized\");\r\n return;\r\n }\r\n\r\n ensureGeistStylesheet();\r\n\r\n const runtimeConfig = resolveRuntimeConfig(options);\r\n applyTheme(runtimeConfig.baseColor);\r\n\r\n // Create or find container\r\n if (options.containerId) {\r\n widgetContainer = document.getElementById(options.containerId);\r\n createdContainer = false;\r\n } else {\r\n widgetContainer = document.createElement(\"div\");\r\n widgetContainer.id = \"bulut-container\";\r\n document.body.appendChild(widgetContainer);\r\n createdContainer = true;\r\n }\r\n\r\n if (!widgetContainer) {\r\n console.error(\"Bulut: Container not found\");\r\n return;\r\n }\r\n\r\n const shadowRoot = widgetContainer.shadowRoot || widgetContainer.attachShadow({ mode: \"open\" });\r\n shadowRoot.replaceChildren();\r\n\r\n const style = document.createElement(\"style\");\r\n style.textContent = SHADOW_STYLE;\r\n\r\n const mountNode = document.createElement(\"div\");\r\n mountNode.id = \"bulut-shadow-mount\";\r\n\r\n shadowRoot.append(style, mountNode);\r\n widgetMountNode = mountNode;\r\n\r\n // Render the widget\r\n render(<BulutWidget config={runtimeConfig} />, mountNode);\r\n isInitialized = true;\r\n\r\n console.log(\"Bulut initialized successfully\");\r\n};\r\n\r\n/**\r\n * Destroy the Bulut widget\r\n */\r\nexport const destroy = () => {\r\n if (!isInitialized) {\r\n return;\r\n }\r\n\r\n if (widgetMountNode) {\r\n render(null, widgetMountNode);\r\n widgetMountNode = null;\r\n }\r\n\r\n if (widgetContainer && createdContainer) {\r\n document.body.removeChild(widgetContainer);\r\n }\r\n\r\n widgetContainer = null;\r\n createdContainer = false;\r\n isInitialized = false;\r\n console.log(\"Bulut destroyed\");\r\n};\r\n\r\n/**\r\n * Check if the widget is initialized\r\n */\r\nexport const isReady = () => isInitialized;\r\n\r\nconst Bulut = {\r\n init,\r\n destroy,\r\n isReady,\r\n};\r\n\r\nif (typeof window !== \"undefined\") {\r\n (window as Window & { Bulut?: typeof Bulut }).Bulut = Bulut;\r\n}\r\n\r\n// Export the main widget component for advanced usage\r\nexport { BulutWidget };\r\n\r\n// Export components for custom implementations\r\nexport { ChatButton, ChatWindow };\r\n\r\nexport default Bulut;\r\n"],"names":["n","l","u","i","o","r","e","f","c","s","a","p","v","y","d","Array","isArray","w","g","parentNode","removeChild","m","t","type","props","key","ref","__k","__","__b","__e","__c","constructor","__v","__i","__u","vnode","k","children","x","this","context","S","length","C","base","M","__d","push","$","__r","debounceRendering","sort","shift","__P","O","__n","namespaceURI","N","I","h","_","b","String","L","D","P","B","A","nextSibling","insertBefore","nodeType","T","setProperty","test","j","style","cssText","replace","toLowerCase","slice","addEventListener","removeEventListener","removeAttribute","setAttribute","F","event","H","prototype","render","contextType","value","__E","E","sub","state","__h","_sb","__s","getDerivedStateFromProps","componentWillMount","componentDidMount","componentWillReceiveProps","shouldComponentUpdate","some","componentWillUpdate","componentDidUpdate","getChildContext","getSnapshotBeforeUpdate","V","then","indexOf","z","localName","document","createTextNode","createElementNS","is","__m","data","call","childNodes","attributes","name","__html","innerHTML","content","q","diffed","forEach","map","current","unmount","componentWillUnmount","G","documentElement","arguments","defaultProps","firstChild","getDerivedStateFromError","setState","componentDidCatch","forceUpdate","Promise","bind","resolve","setTimeout","__source","__self","__H","__N","__f","filter","every","requestAnimationFrame","clearTimeout","cancelAnimationFrame","COLORS","primary","primaryHover","text","messageUser","messageUserText","BORDER_RADIUS","SHADOW","TRANSITIONS","ArrowPathIcon","title","titleId","svgRef","React","createElement","Object","assign","xmlns","fill","viewBox","strokeWidth","stroke","id","strokeLinecap","strokeLinejoin","ForwardRef","forwardRef","CommandLineIcon","CursorArrowRaysIcon","FaceSmileIcon","HandRaisedIcon","MapIcon","MicrophoneIcon","QueueListIcon","StopIcon","XMarkIcon","ChatButton","onMicClick","onCancelRecording","onStopTask","isRecording","isBusy","accessibilityMode","showBubble","onBubbleClick","previewMessage","onPreviewClick","onPreviewClose","bgColor","popupBoxShadow","containerStyle","position","right","bottom","display","flexDirection","alignItems","gap","zIndex","fontFamily","buttonStyle","width","height","minWidth","minHeight","borderRadius","backgroundColor","color","border","cursor","justifyContent","transition","flexShrink","showStopButton","MainIcon","closeBtnStyle","top","background","fontSize","lineHeight","padding","renderPopup","onClick","onClose","extraClass","scrollable","jsxs","className","jsx","stopPropagation","paddingRight","wordBreak","maxHeight","overflowY","Fragment","onMouseEnter","currentTarget","transform","onMouseLeave","PAGE_CONTEXT_CACHE_KEY","NON_CONTENT_TAGS","Set","NATIVE_INTERACTIVE_TAGS","INTERACTIVE_ROLES","TRACKED_DISPLAY_VALUES","TRACKED_POSITION_VALUES","EVENT_HINT_NAMES","ARIA_INTERACTION_ATTRS","DATA_INTERACTION_PATTERN","STYLESHEET_SELECTOR_PATTERN","pageContextCache","Map","cacheHydrated","normalizeWhitespace","trim","isCacheEntry","obj","url","summary","links","interactables","capturedAt","version","bumpCount","set","get","formatTopCounts","maxItems","size","from","entries","localeCompare","count","join","parseTabIndex","parsed","Number","parseInt","isNaN","compactToken","getElementDepth","element","depth","parentElement","body","getPrimaryRole","getAttribute","split","buildSummaryWithHistory","recentPages","values","entry","historySection","compactSummary","isVisible","HTMLElement","hidden","window","getComputedStyle","visibility","rect","getBoundingClientRect","toAbsoluteUrl","href","URL","location","escapeCssValue","CSS","escape","buildSelectorSegment","tag","tagName","attrCandidates","attrName","attrValue","classes","classList","Boolean","parent","sibling","buildSelector","segments","segment","unshift","startsWith","getElementLabel","innerText","textContent","ariaLabel","placeholder","HTMLInputElement","HTMLTextAreaElement","HTMLButtonElement","classHint","item","find","fallback","label","getEventHints","record","eventHints","eventName","handlerKey","hasInlineHandler","hasPropertyHandler","getAriaInteractionHints","hasAttribute","getDataInteractionHints","getAttributeNames","getComputedStyleSignals","signals","pointerEvents","opacity","buildBranchDigest","token","idToken","classToken","buildBlueprintToken","child","has","sampled","overflow","formatSection","lines","collectTextSnippets","root","querySelector","snippets","seen","candidates","querySelectorAll","node","add","collectSelectorsFromRuleList","rules","selectors","scanned","rule","CSSStyleRule","parts","selectorText","selector","nestedRules","cssRules","collectStylesheetSelectors","stylesheet","styleSheets","collectPageSignalSnapshot","allElements","sampledElements","linkSet","interactableCandidates","tagCounts","roleCounts","eventCounts","displayCounts","positionCounts","styleSignalCounts","visibleElements","maxDepth","semanticInteractables","nonSemanticInteractables","tabStopElements","pointerCursorElements","dataHintElements","ariaHintElements","contentEditableElements","order","role","computedStyleSignals","styleSignal","tabIndex","hasTabStop","hasPointerCursor","dataHints","ariaHints","isContentEditable","isNativeInteractive","isRoleInteractive","isDisabled","absoluteHref","line","styleSignals","signalTokens","score","existing","candidate","interactiveRoleCounts","interactionSignals","branchDigest","collectDomBranchDigest","pageBlueprint","styleSelectors","getPageContext","sessionStorage","raw","getItem","JSON","parse","console","info","error","warn","hydrateCacheFromStorage","rawUrl","canonicalUrl","cached","snapshot","headings","input","lang","landmarks","textSnippets","outerHtmlDigest","outerHTML","Date","now","sorted","delete","pruneOldestCacheEntries","serialized","stringify","setItem","persistCacheToStorage","AGENT_CURSOR_ID","CURSOR_MOVE_DURATION_MS","CURSOR_EASING","RESUME_STORAGE_KEY","savePendingAgentResume","localStorage","savedAt","clearPendingAgentResume","removeItem","clamp","min","max","Math","animateWindowScrollTo","async","targetY","durationMs","startY","scrollY","delta","abs","raf","callback","performance","startTime","step","progress","eased","cos","PI","easeInOutSine","scrollTo","setCursorPosition","left","setCursorVisibility","visible","isVisibleElement","findAgentUiAnchorElement","roots","defaultHost","getElementById","shadowRoot","el","includes","getBulutShadowRoots","panel","button","getAgentWindowTopLeft","anchor","CURSOR_DIAMETER_PX","scrollX","hideAgentCursor","cursorHoverTrackingInitialized","initializeCursorHoverTracking","parseFloat","getCursorPosition","pointerX","pageX","pointerY","pageY","hypot","moveCursor","startPosition","baseColor","boxShadow","boxSizing","appendChild","ensureCursor","dataset","transitionReady","waitForNextAnimationFrame","getElementCenter","CONTAINS_SELECTOR_PATTERN","findElementBySelector","containsMatch","match","baseSelector","expectedText","fallbackError","dispatchMouseEvent","dispatchEvent","MouseEvent","bubbles","cancelable","view","clientX","clientY","typeIntoElement","focus","descriptor","getOwnPropertyDescriptor","defaultValue","setNativeInputLikeValue","Event","slowScrollElementIntoView","slowScrollElementIntoViewWithMode","forceCenter","viewportHeight","innerHeight","isRectOutsideViewport","maxScrollTop","scrollHeight","currentScrollY","rectTop","rectHeight","computeCenteredScrollTop","executeInteract","target","selected","center","resolveTarget","action","click","requestSubmit","form","parentForm","closest","submitElement","executeNavigate","targetUrl","resolvedUrl","matchingElement","parsedTarget","allLinks","HTMLAnchorElement","elUrl","pathname","search","hash","urlSegments","lastSegment","searchTerms","searchParams","clickables","dataTab","term","findMatchingLinkForTarget","log","origin","isSamePageNavigation","newPath","history","pushState","PopStateEvent","executeSingleToolCall","callId","call_id","tool","result","executeScroll","msg","Error","message","TTS_WS_RETRY_DELAYS_MS","BULUT_AUDIO_STOP_EVENT","activeAudioElements","audioPlaybackGeneration","normalizeBaseUrl","baseUrl","trimmed","toWebSocketUrl","path","normalized","protocol","toString","shouldFallbackToSse","retryable","parseErrorBody","response","json","detail","statusText","sleep","ms","wasPlaybackStoppedAfter","generationAtStart","getAudioPlaybackGeneration","stopActiveAudioPlayback","active","audioElement","pause","load","base64ToUint8Array","base64","cleanBase64","binaryString","atob","bytes","Uint8Array","charCodeAt","playBufferedAudio","chunks","mimeType","sampleRate","onAudioStateChange","playbackGeneration","totalBytes","reduce","acc","byteLength","blobParts","chunk","copied","buffer","detectedMime","header","padStart","toUpperCase","safeMimeType","finalBlobParts","ArrayBuffer","DataView","setUint32","setUint16","channels","createWavHeader","blob","Blob","Audio","objectUrl","createObjectURL","registerActiveAudioElement","preload","autoplay","src","play","ended","reject","watchdog","setInterval","onEnded","cleanup","onError","onForcedStop","clearInterval","waitForPlaybackEnd","err","unregisterActiveAudioElement","revokeObjectURL","parseSseEventPayload","eventBlock","dataLines","trimStart","dataStr","isAudioSsePayload","payload","audio","startSttWebSocketStream","config","events","wsUrl","socket","WebSocket","seq","finalText","finalSessionId","sessionId","stopped","settled","sendQueue","resolveStart","rejectStart","startPromise","resolveDone","rejectDone","donePromise","rejectAll","resolveDoneIfPossible","session_id","onopen","onRequestSent","send","project_id","projectId","language","mime_type","onmessage","parseSttWsEventPayload","onSessionId","close","buildError","onPartial","onerror","ev","onclose","code","reason","pushChunk","arrayBuffer","binary","fromCharCode","subarray","btoa","blobToBase64","readyState","OPEN","debug","stop","cancel","collectTtsViaSse","assistantText","voice","isStopped","setReader","ttsFormData","FormData","append","ttsResponse","fetch","method","ok","reader","getReader","decoder","TextDecoder","done","read","decode","stream","blocks","pop","block","format","sample_rate","releaseLock","collectTtsViaWebSocket","setSocket","requestId","crypto","randomUUID","random","highestSeqSeen","connectOnce","finalError","finalize","mode","request_id","accessibility_mode","last_seq","parseTtsWsEventPayload","incomingSeq","shouldAcceptAudioSeq","attempt","delay","speakText","ttsResult","neverStopped","agentVoiceChatStream","audioFile","executeTool","activeReader","activeSocket","errorEmitted","sttResult","file","formData","responsePromise","transcribeAudio","onSttRequestSent","currentSessionId","effectiveSessionId","userText","onTranscription","user_text","agentResolve","agentReject","finalReply","resolved","accumulatedDelta","finish","reply","fail","model","page_context","pageContext","msgType","onAssistantDelta","calls","onIntermediateReply","onToolCalls","results","isNavigate","pendingToolCalls","args","completedResults","onToolResult","final_reply","replyExpectsReply","expects_reply","onAssistantDone","errMsg","onIteration","iteration","max_iterations","wsError","catch","SFX_SOURCES","sent","thinking","toolCall","completed","sfxManager","queue","playNow","onended","drain","next","playCue","SvgIcon","stripColors","useMemo","trimmedSrc","viewBoxMatch","verticalAlign","dangerouslySetInnerHTML","nextContent","strokeWidthValue","STORAGE_KEY","TIMESTAMP_KEY","SESSION_ID_KEY","TTL_MS","STATUS_LABELS","getGreetingText","agentName","resolveStatusText","flags","isRunningTools","isPlayingAudio","isThinking","isTranscribing","formatDurationMs","totalSeconds","floor","createInitialMessages","isUser","clearPersistedChatState","storage","getNextMessageId","messages","shouldAutoListenAfterAudio","expectsReply","getToolIndicatorMessage","kind","ChatWindow","onAccessibilityToggle","actionsRef","onRecordingChange","onBusyChange","onPreviewChange","setMessages","useState","saved","timestamp","setIsBusy","setIsRecording","setIsTranscribing","setIsThinking","isRenderingAudio","setIsRenderingAudio","setIsPlayingAudio","setIsRunningTools","isMicPending","setIsMicPending","recordingDurationMs","setRecordingDurationMs","statusOverride","setStatusOverride","statusFlags","resolvedStatusText","showStatus","hasActiveStatus","isBusyRef","useRef","isRecordingRef","nextMessageIdRef","recorderRef","streamRef","audioChunksRef","activeStreamControllerRef","sessionIdRef","ts","silenceStartRef","vadIntervalRef","audioContextRef","sourceRef","discardNextRecordingRef","micPressStartRef","micHoldTimeoutRef","micHoldTriggeredRef","recordingModeRef","pendingStopAfterStartRef","startRecordingPendingRef","assistantMessageIdRef","assistantTextBufferRef","transcriptionReceivedRef","assistantDoneReceivedRef","recordingStartedAtRef","recordingTimerIntervalRef","messagesContainerRef","messagesContentRef","pendingUserTextRef","pendingAssistantTextRef","streamingJsonParserRef","awaitingAssistantResponseRef","activeSttWsRef","liveTranscriptionMessageIdRef","liveTranscriptionTextRef","autoListenSuppressedRef","expectsReplyRef","requestEpochRef","sttSendCuePlayedRef","useEffect","lastAssistant","reverse","st","playSfx","beginRequestEpoch","invalidateRequestEpoch","isCurrentRequestEpoch","epoch","playSttSentCueOnce","scrollMessagesToBottom","scrollTop","ResizeObserver","observer","observe","disconnect","stopRecordingTimer","resetProcessingFlags","clearMicHoldTimeout","cleanupVAD","stopStreamTracks","getTracks","track","stopActiveStream","cancelActiveSttWs","activeSttWs","recorder","ondataavailable","onstop","resumeState","getPendingAgentResume","requestEpoch","freshPageContext","controller","allResults","tc","pending_tool_calls","tool_results","agentResumeStream","backendBaseUrl","sid","appendMessage","updateMessageText","finalDisplayText","handleAudioStateChange","appendToolIndicatorMessages","finally","startRecording","options","previous","toolKind","toolLabel","toolCount","indicator","last","previousToolText","extractedCount","nextCount","isFinite","baseLabel","upsertLiveUserTranscription","clearLiveUserTranscriptionState","finalizeStreamCycle","runAgentForUserText","normalizedUserText","handleToolExecution","agentTextChatStream","stopRecording","discard","navigator","mediaDevices","getUserMedia","MediaRecorder","recorderOptions","audioBitsPerSecond","preferredMimeTypes","mime","isTypeSupported","sttMimeType","sttWsController","shouldDiscard","currentSttWs","fileType","extension","File","previousUserText","_callId","_toolName","_result","_iteration","_maxIterations","handleAudioBlob","AudioCtx","AudioContext","webkitAudioContext","analyser","createAnalyser","fftSize","source","createMediaStreamSource","connect","dataArray","frequencyBinCount","speechDetected","speechStartedAt","enforceMinSpeechDuration","getByteFrequencyData","sum","silenceDuration","speechDurationMs","minSpeechDurationMs","shouldAcceptVadSpeech","setupVAD","start","startedAt","normalizeError","resetMicGesture","handleMicPointerUp","preventDefault","releasePointerCapture","pointerId","wasHold","thresholdMs","classifyMicGesture","timer","stopTask","cancelRecording","windowStyle","animation","headerButtonStyle","messageStyle","maxWidth","wordWrap","whiteSpace","alignSelf","statusPanelStyle","flex","textOverflow","recordingTimerStyle","fontWeight","textAlign","micFooterButtonStyle","isVadRecording","hideMicButton","disableMicControl","initialMessages","ToolIcon","onPointerDown","setPointerCapture","onPointerUp","onPointerCancel","disabled","DEFAULT_CONFIG","normalizeHexColor","isValidHexColor","shadeHexColor","hexColor","amount","toChannel","adjust","channel","toHex","round","applyTheme","ACCESSIBILITY_MODE_KEY","GEIST_FONT_FAMILY","GEIST_STYLESHEET_ID","BulutWidget","liveConfig","setLiveConfig","configReady","setConfigReady","cancelled","res","fetchRemoteConfig","remote","merged","base_color","agent_name","isOpen","setIsOpen","setShowBubble","isAccessibilityEnabled","setIsAccessibilityEnabled","setPreviewMessage","previewDismissed","setPreviewDismissed","chatActionsRef","handlePreviewChange","toggleWidget","newState","handleEscape","SHADOW_STYLE","widgetContainer","widgetMountNode","createdContainer","isInitialized","init","link","rel","head","ensureGeistStylesheet","runtimeConfig","resolveRuntimeConfig","containerId","attachShadow","replaceChildren","mountNode","destroy","isReady","Bulut"],"mappings":"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","x_google_ignoreList":[0,1,2,4,5,6,7,8,9,10,11,12,13]}
|
|
1
|
+
{"version":3,"file":"embed.js","sources":["../node_modules/preact/dist/preact.module.js","../node_modules/preact/jsx-runtime/dist/jsxRuntime.module.js","../node_modules/preact/hooks/dist/hooks.module.js","../src/styles/constants.ts","../src/assets.ts","../src/assets/icons/hand-raised.svg?raw","../src/assets/icons/microphone-outline.svg?raw","../src/assets/icons/stop-outline.svg?raw","../src/assets/icons/x-mark.svg?raw","../src/components/SvgIcon.tsx","../src/components/ChatButton.tsx","../src/agent/context.ts","../src/agent/contextConfig.ts","../src/agent/tools.ts","../src/api/client.ts","../src/audio/sfxManager.ts","../src/assets/sfx/sent.mp3","../src/assets/sfx/thinking.mp3","../src/assets/sfx/tool_call.mp3","../src/assets/sfx/completed.mp3","../src/components/ChatWindow.tsx","../src/assets/icons/bulut_logo.svg?raw","../src/assets/icons/arrow-path.svg?raw","../src/assets/icons/cursor-arrow-rays.svg?raw","../src/assets/icons/map.svg?raw","../src/assets/icons/queue-list.svg?raw","../src/assets/icons/command-line.svg?raw","../src/assets/icons/face-smile.svg?raw","../src/index.tsx"],"sourcesContent":["var n,l,u,t,i,o,r,e,f,c,s,a,h,p={},v=[],y=/acit|ex(?:s|g|n|p|$)|rph|grid|ows|mnc|ntw|ine[ch]|zoo|^ord|itera/i,d=Array.isArray;function w(n,l){for(var u in l)n[u]=l[u];return n}function g(n){n&&n.parentNode&&n.parentNode.removeChild(n)}function _(l,u,t){var i,o,r,e={};for(r in u)\"key\"==r?i=u[r]:\"ref\"==r?o=u[r]:e[r]=u[r];if(arguments.length>2&&(e.children=arguments.length>3?n.call(arguments,2):t),\"function\"==typeof l&&null!=l.defaultProps)for(r in l.defaultProps)void 0===e[r]&&(e[r]=l.defaultProps[r]);return m(l,e,i,o,null)}function m(n,t,i,o,r){var e={type:n,props:t,key:i,ref:o,__k:null,__:null,__b:0,__e:null,__c:null,constructor:void 0,__v:null==r?++u:r,__i:-1,__u:0};return null==r&&null!=l.vnode&&l.vnode(e),e}function b(){return{current:null}}function k(n){return n.children}function x(n,l){this.props=n,this.context=l}function S(n,l){if(null==l)return n.__?S(n.__,n.__i+1):null;for(var u;l<n.__k.length;l++)if(null!=(u=n.__k[l])&&null!=u.__e)return u.__e;return\"function\"==typeof n.type?S(n):null}function C(n){var l,u;if(null!=(n=n.__)&&null!=n.__c){for(n.__e=n.__c.base=null,l=0;l<n.__k.length;l++)if(null!=(u=n.__k[l])&&null!=u.__e){n.__e=n.__c.base=u.__e;break}return C(n)}}function M(n){(!n.__d&&(n.__d=!0)&&i.push(n)&&!$.__r++||o!=l.debounceRendering)&&((o=l.debounceRendering)||r)($)}function $(){for(var n,u,t,o,r,f,c,s=1;i.length;)i.length>s&&i.sort(e),n=i.shift(),s=i.length,n.__d&&(t=void 0,o=void 0,r=(o=(u=n).__v).__e,f=[],c=[],u.__P&&((t=w({},o)).__v=o.__v+1,l.vnode&&l.vnode(t),O(u.__P,t,o,u.__n,u.__P.namespaceURI,32&o.__u?[r]:null,f,null==r?S(o):r,!!(32&o.__u),c),t.__v=o.__v,t.__.__k[t.__i]=t,N(f,t,c),o.__e=o.__=null,t.__e!=r&&C(t)));$.__r=0}function I(n,l,u,t,i,o,r,e,f,c,s){var a,h,y,d,w,g,_,m=t&&t.__k||v,b=l.length;for(f=P(u,l,m,f,b),a=0;a<b;a++)null!=(y=u.__k[a])&&(h=-1==y.__i?p:m[y.__i]||p,y.__i=a,g=O(n,y,h,i,o,r,e,f,c,s),d=y.__e,y.ref&&h.ref!=y.ref&&(h.ref&&B(h.ref,null,y),s.push(y.ref,y.__c||d,y)),null==w&&null!=d&&(w=d),(_=!!(4&y.__u))||h.__k===y.__k?f=A(y,f,n,_):\"function\"==typeof y.type&&void 0!==g?f=g:d&&(f=d.nextSibling),y.__u&=-7);return u.__e=w,f}function P(n,l,u,t,i){var o,r,e,f,c,s=u.length,a=s,h=0;for(n.__k=new Array(i),o=0;o<i;o++)null!=(r=l[o])&&\"boolean\"!=typeof r&&\"function\"!=typeof r?(\"string\"==typeof r||\"number\"==typeof r||\"bigint\"==typeof r||r.constructor==String?r=n.__k[o]=m(null,r,null,null,null):d(r)?r=n.__k[o]=m(k,{children:r},null,null,null):void 0===r.constructor&&r.__b>0?r=n.__k[o]=m(r.type,r.props,r.key,r.ref?r.ref:null,r.__v):n.__k[o]=r,f=o+h,r.__=n,r.__b=n.__b+1,e=null,-1!=(c=r.__i=L(r,u,f,a))&&(a--,(e=u[c])&&(e.__u|=2)),null==e||null==e.__v?(-1==c&&(i>s?h--:i<s&&h++),\"function\"!=typeof r.type&&(r.__u|=4)):c!=f&&(c==f-1?h--:c==f+1?h++:(c>f?h--:h++,r.__u|=4))):n.__k[o]=null;if(a)for(o=0;o<s;o++)null!=(e=u[o])&&0==(2&e.__u)&&(e.__e==t&&(t=S(e)),D(e,e));return t}function A(n,l,u,t){var i,o;if(\"function\"==typeof n.type){for(i=n.__k,o=0;i&&o<i.length;o++)i[o]&&(i[o].__=n,l=A(i[o],l,u,t));return l}n.__e!=l&&(t&&(l&&n.type&&!l.parentNode&&(l=S(n)),u.insertBefore(n.__e,l||null)),l=n.__e);do{l=l&&l.nextSibling}while(null!=l&&8==l.nodeType);return l}function H(n,l){return l=l||[],null==n||\"boolean\"==typeof n||(d(n)?n.some(function(n){H(n,l)}):l.push(n)),l}function L(n,l,u,t){var i,o,r,e=n.key,f=n.type,c=l[u],s=null!=c&&0==(2&c.__u);if(null===c&&null==e||s&&e==c.key&&f==c.type)return u;if(t>(s?1:0))for(i=u-1,o=u+1;i>=0||o<l.length;)if(null!=(c=l[r=i>=0?i--:o++])&&0==(2&c.__u)&&e==c.key&&f==c.type)return r;return-1}function T(n,l,u){\"-\"==l[0]?n.setProperty(l,null==u?\"\":u):n[l]=null==u?\"\":\"number\"!=typeof u||y.test(l)?u:u+\"px\"}function j(n,l,u,t,i){var o,r;n:if(\"style\"==l)if(\"string\"==typeof u)n.style.cssText=u;else{if(\"string\"==typeof t&&(n.style.cssText=t=\"\"),t)for(l in t)u&&l in u||T(n.style,l,\"\");if(u)for(l in u)t&&u[l]==t[l]||T(n.style,l,u[l])}else if(\"o\"==l[0]&&\"n\"==l[1])o=l!=(l=l.replace(f,\"$1\")),r=l.toLowerCase(),l=r in n||\"onFocusOut\"==l||\"onFocusIn\"==l?r.slice(2):l.slice(2),n.l||(n.l={}),n.l[l+o]=u,u?t?u.u=t.u:(u.u=c,n.addEventListener(l,o?a:s,o)):n.removeEventListener(l,o?a:s,o);else{if(\"http://www.w3.org/2000/svg\"==i)l=l.replace(/xlink(H|:h)/,\"h\").replace(/sName$/,\"s\");else if(\"width\"!=l&&\"height\"!=l&&\"href\"!=l&&\"list\"!=l&&\"form\"!=l&&\"tabIndex\"!=l&&\"download\"!=l&&\"rowSpan\"!=l&&\"colSpan\"!=l&&\"role\"!=l&&\"popover\"!=l&&l in n)try{n[l]=null==u?\"\":u;break n}catch(n){}\"function\"==typeof u||(null==u||!1===u&&\"-\"!=l[4]?n.removeAttribute(l):n.setAttribute(l,\"popover\"==l&&1==u?\"\":u))}}function F(n){return function(u){if(this.l){var t=this.l[u.type+n];if(null==u.t)u.t=c++;else if(u.t<t.u)return;return t(l.event?l.event(u):u)}}}function O(n,u,t,i,o,r,e,f,c,s){var a,h,p,v,y,_,m,b,S,C,M,$,P,A,H,L,T,j=u.type;if(void 0!==u.constructor)return null;128&t.__u&&(c=!!(32&t.__u),r=[f=u.__e=t.__e]),(a=l.__b)&&a(u);n:if(\"function\"==typeof j)try{if(b=u.props,S=\"prototype\"in j&&j.prototype.render,C=(a=j.contextType)&&i[a.__c],M=a?C?C.props.value:a.__:i,t.__c?m=(h=u.__c=t.__c).__=h.__E:(S?u.__c=h=new j(b,M):(u.__c=h=new x(b,M),h.constructor=j,h.render=E),C&&C.sub(h),h.state||(h.state={}),h.__n=i,p=h.__d=!0,h.__h=[],h._sb=[]),S&&null==h.__s&&(h.__s=h.state),S&&null!=j.getDerivedStateFromProps&&(h.__s==h.state&&(h.__s=w({},h.__s)),w(h.__s,j.getDerivedStateFromProps(b,h.__s))),v=h.props,y=h.state,h.__v=u,p)S&&null==j.getDerivedStateFromProps&&null!=h.componentWillMount&&h.componentWillMount(),S&&null!=h.componentDidMount&&h.__h.push(h.componentDidMount);else{if(S&&null==j.getDerivedStateFromProps&&b!==v&&null!=h.componentWillReceiveProps&&h.componentWillReceiveProps(b,M),u.__v==t.__v||!h.__e&&null!=h.shouldComponentUpdate&&!1===h.shouldComponentUpdate(b,h.__s,M)){for(u.__v!=t.__v&&(h.props=b,h.state=h.__s,h.__d=!1),u.__e=t.__e,u.__k=t.__k,u.__k.some(function(n){n&&(n.__=u)}),$=0;$<h._sb.length;$++)h.__h.push(h._sb[$]);h._sb=[],h.__h.length&&e.push(h);break n}null!=h.componentWillUpdate&&h.componentWillUpdate(b,h.__s,M),S&&null!=h.componentDidUpdate&&h.__h.push(function(){h.componentDidUpdate(v,y,_)})}if(h.context=M,h.props=b,h.__P=n,h.__e=!1,P=l.__r,A=0,S){for(h.state=h.__s,h.__d=!1,P&&P(u),a=h.render(h.props,h.state,h.context),H=0;H<h._sb.length;H++)h.__h.push(h._sb[H]);h._sb=[]}else do{h.__d=!1,P&&P(u),a=h.render(h.props,h.state,h.context),h.state=h.__s}while(h.__d&&++A<25);h.state=h.__s,null!=h.getChildContext&&(i=w(w({},i),h.getChildContext())),S&&!p&&null!=h.getSnapshotBeforeUpdate&&(_=h.getSnapshotBeforeUpdate(v,y)),L=a,null!=a&&a.type===k&&null==a.key&&(L=V(a.props.children)),f=I(n,d(L)?L:[L],u,t,i,o,r,e,f,c,s),h.base=u.__e,u.__u&=-161,h.__h.length&&e.push(h),m&&(h.__E=h.__=null)}catch(n){if(u.__v=null,c||null!=r)if(n.then){for(u.__u|=c?160:128;f&&8==f.nodeType&&f.nextSibling;)f=f.nextSibling;r[r.indexOf(f)]=null,u.__e=f}else{for(T=r.length;T--;)g(r[T]);z(u)}else u.__e=t.__e,u.__k=t.__k,n.then||z(u);l.__e(n,u,t)}else null==r&&u.__v==t.__v?(u.__k=t.__k,u.__e=t.__e):f=u.__e=q(t.__e,u,t,i,o,r,e,c,s);return(a=l.diffed)&&a(u),128&u.__u?void 0:f}function z(n){n&&n.__c&&(n.__c.__e=!0),n&&n.__k&&n.__k.forEach(z)}function N(n,u,t){for(var i=0;i<t.length;i++)B(t[i],t[++i],t[++i]);l.__c&&l.__c(u,n),n.some(function(u){try{n=u.__h,u.__h=[],n.some(function(n){n.call(u)})}catch(n){l.__e(n,u.__v)}})}function V(n){return\"object\"!=typeof n||null==n||n.__b&&n.__b>0?n:d(n)?n.map(V):w({},n)}function q(u,t,i,o,r,e,f,c,s){var a,h,v,y,w,_,m,b=i.props||p,k=t.props,x=t.type;if(\"svg\"==x?r=\"http://www.w3.org/2000/svg\":\"math\"==x?r=\"http://www.w3.org/1998/Math/MathML\":r||(r=\"http://www.w3.org/1999/xhtml\"),null!=e)for(a=0;a<e.length;a++)if((w=e[a])&&\"setAttribute\"in w==!!x&&(x?w.localName==x:3==w.nodeType)){u=w,e[a]=null;break}if(null==u){if(null==x)return document.createTextNode(k);u=document.createElementNS(r,x,k.is&&k),c&&(l.__m&&l.__m(t,e),c=!1),e=null}if(null==x)b===k||c&&u.data==k||(u.data=k);else{if(e=e&&n.call(u.childNodes),!c&&null!=e)for(b={},a=0;a<u.attributes.length;a++)b[(w=u.attributes[a]).name]=w.value;for(a in b)if(w=b[a],\"children\"==a);else if(\"dangerouslySetInnerHTML\"==a)v=w;else if(!(a in k)){if(\"value\"==a&&\"defaultValue\"in k||\"checked\"==a&&\"defaultChecked\"in k)continue;j(u,a,null,w,r)}for(a in k)w=k[a],\"children\"==a?y=w:\"dangerouslySetInnerHTML\"==a?h=w:\"value\"==a?_=w:\"checked\"==a?m=w:c&&\"function\"!=typeof w||b[a]===w||j(u,a,w,b[a],r);if(h)c||v&&(h.__html==v.__html||h.__html==u.innerHTML)||(u.innerHTML=h.__html),t.__k=[];else if(v&&(u.innerHTML=\"\"),I(\"template\"==t.type?u.content:u,d(y)?y:[y],t,i,o,\"foreignObject\"==x?\"http://www.w3.org/1999/xhtml\":r,e,f,e?e[0]:i.__k&&S(i,0),c,s),null!=e)for(a=e.length;a--;)g(e[a]);c||(a=\"value\",\"progress\"==x&&null==_?u.removeAttribute(\"value\"):null!=_&&(_!==u[a]||\"progress\"==x&&!_||\"option\"==x&&_!=b[a])&&j(u,a,_,b[a],r),a=\"checked\",null!=m&&m!=u[a]&&j(u,a,m,b[a],r))}return u}function B(n,u,t){try{if(\"function\"==typeof n){var i=\"function\"==typeof n.__u;i&&n.__u(),i&&null==u||(n.__u=n(u))}else n.current=u}catch(n){l.__e(n,t)}}function D(n,u,t){var i,o;if(l.unmount&&l.unmount(n),(i=n.ref)&&(i.current&&i.current!=n.__e||B(i,null,u)),null!=(i=n.__c)){if(i.componentWillUnmount)try{i.componentWillUnmount()}catch(n){l.__e(n,u)}i.base=i.__P=null}if(i=n.__k)for(o=0;o<i.length;o++)i[o]&&D(i[o],u,t||\"function\"!=typeof n.type);t||g(n.__e),n.__c=n.__=n.__e=void 0}function E(n,l,u){return this.constructor(n,u)}function G(u,t,i){var o,r,e,f;t==document&&(t=document.documentElement),l.__&&l.__(u,t),r=(o=\"function\"==typeof i)?null:i&&i.__k||t.__k,e=[],f=[],O(t,u=(!o&&i||t).__k=_(k,null,[u]),r||p,p,t.namespaceURI,!o&&i?[i]:r?null:t.firstChild?n.call(t.childNodes):null,e,!o&&i?i:r?r.__e:t.firstChild,o,f),N(e,u,f)}function J(n,l){G(n,l,J)}function K(l,u,t){var i,o,r,e,f=w({},l.props);for(r in l.type&&l.type.defaultProps&&(e=l.type.defaultProps),u)\"key\"==r?i=u[r]:\"ref\"==r?o=u[r]:f[r]=void 0===u[r]&&null!=e?e[r]:u[r];return arguments.length>2&&(f.children=arguments.length>3?n.call(arguments,2):t),m(l.type,f,i||l.key,o||l.ref,null)}function Q(n){function l(n){var u,t;return this.getChildContext||(u=new Set,(t={})[l.__c]=this,this.getChildContext=function(){return t},this.componentWillUnmount=function(){u=null},this.shouldComponentUpdate=function(n){this.props.value!=n.value&&u.forEach(function(n){n.__e=!0,M(n)})},this.sub=function(n){u.add(n);var l=n.componentWillUnmount;n.componentWillUnmount=function(){u&&u.delete(n),l&&l.call(n)}}),n.children}return l.__c=\"__cC\"+h++,l.__=n,l.Provider=l.__l=(l.Consumer=function(n,l){return n.children(l)}).contextType=l,l}n=v.slice,l={__e:function(n,l,u,t){for(var i,o,r;l=l.__;)if((i=l.__c)&&!i.__)try{if((o=i.constructor)&&null!=o.getDerivedStateFromError&&(i.setState(o.getDerivedStateFromError(n)),r=i.__d),null!=i.componentDidCatch&&(i.componentDidCatch(n,t||{}),r=i.__d),r)return i.__E=i}catch(l){n=l}throw n}},u=0,t=function(n){return null!=n&&void 0===n.constructor},x.prototype.setState=function(n,l){var u;u=null!=this.__s&&this.__s!=this.state?this.__s:this.__s=w({},this.state),\"function\"==typeof n&&(n=n(w({},u),this.props)),n&&w(u,n),null!=n&&this.__v&&(l&&this._sb.push(l),M(this))},x.prototype.forceUpdate=function(n){this.__v&&(this.__e=!0,n&&this.__h.push(n),M(this))},x.prototype.render=k,i=[],r=\"function\"==typeof Promise?Promise.prototype.then.bind(Promise.resolve()):setTimeout,e=function(n,l){return n.__v.__b-l.__v.__b},$.__r=0,f=/(PointerCapture)$|Capture$/i,c=0,s=F(!1),a=F(!0),h=0;export{x as Component,k as Fragment,K as cloneElement,Q as createContext,_ as createElement,b as createRef,_ as h,J as hydrate,t as isValidElement,l as options,G as render,H as toChildArray};\n//# sourceMappingURL=preact.module.js.map\n","import{options as r,Fragment as e}from\"preact\";export{Fragment}from\"preact\";var t=/[\"&<]/;function n(r){if(0===r.length||!1===t.test(r))return r;for(var e=0,n=0,o=\"\",f=\"\";n<r.length;n++){switch(r.charCodeAt(n)){case 34:f=\""\";break;case 38:f=\"&\";break;case 60:f=\"<\";break;default:continue}n!==e&&(o+=r.slice(e,n)),o+=f,e=n+1}return n!==e&&(o+=r.slice(e,n)),o}var o=/acit|ex(?:s|g|n|p|$)|rph|grid|ows|mnc|ntw|ine[ch]|zoo|^ord|itera/i,f=0,i=Array.isArray;function u(e,t,n,o,i,u){t||(t={});var a,c,p=t;if(\"ref\"in p)for(c in p={},t)\"ref\"==c?a=t[c]:p[c]=t[c];var l={type:e,props:p,key:n,ref:a,__k:null,__:null,__b:0,__e:null,__c:null,constructor:void 0,__v:--f,__i:-1,__u:0,__source:i,__self:u};if(\"function\"==typeof e&&(a=e.defaultProps))for(c in a)void 0===p[c]&&(p[c]=a[c]);return r.vnode&&r.vnode(l),l}function a(r){var t=u(e,{tpl:r,exprs:[].slice.call(arguments,1)});return t.key=t.__v,t}var c={},p=/[A-Z]/g;function l(e,t){if(r.attr){var f=r.attr(e,t);if(\"string\"==typeof f)return f}if(t=function(r){return null!==r&&\"object\"==typeof r&&\"function\"==typeof r.valueOf?r.valueOf():r}(t),\"ref\"===e||\"key\"===e)return\"\";if(\"style\"===e&&\"object\"==typeof t){var i=\"\";for(var u in t){var a=t[u];if(null!=a&&\"\"!==a){var l=\"-\"==u[0]?u:c[u]||(c[u]=u.replace(p,\"-$&\").toLowerCase()),s=\";\";\"number\"!=typeof a||l.startsWith(\"--\")||o.test(l)||(s=\"px;\"),i=i+l+\":\"+a+s}}return e+'=\"'+n(i)+'\"'}return null==t||!1===t||\"function\"==typeof t||\"object\"==typeof t?\"\":!0===t?e:e+'=\"'+n(\"\"+t)+'\"'}function s(r){if(null==r||\"boolean\"==typeof r||\"function\"==typeof r)return null;if(\"object\"==typeof r){if(void 0===r.constructor)return r;if(i(r)){for(var e=0;e<r.length;e++)r[e]=s(r[e]);return r}}return n(\"\"+r)}export{u as jsx,l as jsxAttr,u as jsxDEV,s as jsxEscape,a as jsxTemplate,u as jsxs};\n//# sourceMappingURL=jsxRuntime.module.js.map\n","import{options as n}from\"preact\";var t,r,u,i,o=0,f=[],c=n,e=c.__b,a=c.__r,v=c.diffed,l=c.__c,m=c.unmount,s=c.__;function p(n,t){c.__h&&c.__h(r,n,o||t),o=0;var u=r.__H||(r.__H={__:[],__h:[]});return n>=u.__.length&&u.__.push({}),u.__[n]}function d(n){return o=1,h(D,n)}function h(n,u,i){var o=p(t++,2);if(o.t=n,!o.__c&&(o.__=[i?i(u):D(void 0,u),function(n){var t=o.__N?o.__N[0]:o.__[0],r=o.t(t,n);t!==r&&(o.__N=[r,o.__[1]],o.__c.setState({}))}],o.__c=r,!r.__f)){var f=function(n,t,r){if(!o.__c.__H)return!0;var u=o.__c.__H.__.filter(function(n){return!!n.__c});if(u.every(function(n){return!n.__N}))return!c||c.call(this,n,t,r);var i=o.__c.props!==n;return u.forEach(function(n){if(n.__N){var t=n.__[0];n.__=n.__N,n.__N=void 0,t!==n.__[0]&&(i=!0)}}),c&&c.call(this,n,t,r)||i};r.__f=!0;var c=r.shouldComponentUpdate,e=r.componentWillUpdate;r.componentWillUpdate=function(n,t,r){if(this.__e){var u=c;c=void 0,f(n,t,r),c=u}e&&e.call(this,n,t,r)},r.shouldComponentUpdate=f}return o.__N||o.__}function y(n,u){var i=p(t++,3);!c.__s&&C(i.__H,u)&&(i.__=n,i.u=u,r.__H.__h.push(i))}function _(n,u){var i=p(t++,4);!c.__s&&C(i.__H,u)&&(i.__=n,i.u=u,r.__h.push(i))}function A(n){return o=5,T(function(){return{current:n}},[])}function F(n,t,r){o=6,_(function(){if(\"function\"==typeof n){var r=n(t());return function(){n(null),r&&\"function\"==typeof r&&r()}}if(n)return n.current=t(),function(){return n.current=null}},null==r?r:r.concat(n))}function T(n,r){var u=p(t++,7);return C(u.__H,r)&&(u.__=n(),u.__H=r,u.__h=n),u.__}function q(n,t){return o=8,T(function(){return n},t)}function x(n){var u=r.context[n.__c],i=p(t++,9);return i.c=n,u?(null==i.__&&(i.__=!0,u.sub(r)),u.props.value):n.__}function P(n,t){c.useDebugValue&&c.useDebugValue(t?t(n):n)}function b(n){var u=p(t++,10),i=d();return u.__=n,r.componentDidCatch||(r.componentDidCatch=function(n,t){u.__&&u.__(n,t),i[1](n)}),[i[0],function(){i[1](void 0)}]}function g(){var n=p(t++,11);if(!n.__){for(var u=r.__v;null!==u&&!u.__m&&null!==u.__;)u=u.__;var i=u.__m||(u.__m=[0,0]);n.__=\"P\"+i[0]+\"-\"+i[1]++}return n.__}function j(){for(var n;n=f.shift();)if(n.__P&&n.__H)try{n.__H.__h.forEach(z),n.__H.__h.forEach(B),n.__H.__h=[]}catch(t){n.__H.__h=[],c.__e(t,n.__v)}}c.__b=function(n){r=null,e&&e(n)},c.__=function(n,t){n&&t.__k&&t.__k.__m&&(n.__m=t.__k.__m),s&&s(n,t)},c.__r=function(n){a&&a(n),t=0;var i=(r=n.__c).__H;i&&(u===r?(i.__h=[],r.__h=[],i.__.forEach(function(n){n.__N&&(n.__=n.__N),n.u=n.__N=void 0})):(i.__h.forEach(z),i.__h.forEach(B),i.__h=[],t=0)),u=r},c.diffed=function(n){v&&v(n);var t=n.__c;t&&t.__H&&(t.__H.__h.length&&(1!==f.push(t)&&i===c.requestAnimationFrame||((i=c.requestAnimationFrame)||w)(j)),t.__H.__.forEach(function(n){n.u&&(n.__H=n.u),n.u=void 0})),u=r=null},c.__c=function(n,t){t.some(function(n){try{n.__h.forEach(z),n.__h=n.__h.filter(function(n){return!n.__||B(n)})}catch(r){t.some(function(n){n.__h&&(n.__h=[])}),t=[],c.__e(r,n.__v)}}),l&&l(n,t)},c.unmount=function(n){m&&m(n);var t,r=n.__c;r&&r.__H&&(r.__H.__.forEach(function(n){try{z(n)}catch(n){t=n}}),r.__H=void 0,t&&c.__e(t,r.__v))};var k=\"function\"==typeof requestAnimationFrame;function w(n){var t,r=function(){clearTimeout(u),k&&cancelAnimationFrame(t),setTimeout(n)},u=setTimeout(r,35);k&&(t=requestAnimationFrame(r))}function z(n){var t=r,u=n.__c;\"function\"==typeof u&&(n.__c=void 0,u()),r=t}function B(n){var t=r;n.__c=n.__(),r=t}function C(n,t){return!n||n.length!==t.length||t.some(function(t,r){return t!==n[r]})}function D(n,t){return\"function\"==typeof t?t(n):t}export{q as useCallback,x as useContext,P as useDebugValue,y as useEffect,b as useErrorBoundary,g as useId,F as useImperativeHandle,_ as useLayoutEffect,T as useMemo,h as useReducer,A as useRef,d as useState};\n//# sourceMappingURL=hooks.module.js.map\n","// Widget dimensions\r\nexport const BUTTON_SIZE = 60;\r\nexport const WINDOW_WIDTH = 350;\r\nexport const WINDOW_HEIGHT = 500;\r\n\r\n// Positioning\r\nexport const POSITION_BOTTOM = 20;\r\nexport const POSITION_RIGHT = 20;\r\n\r\n// Colors — initial values match the backend ProjectSettings default.\r\n// applyTheme() in index.tsx overrides primary/primaryHover/messageUser\r\n// with the remote config value before the widget renders.\r\nexport const COLORS = {\r\n primary: \"#6C03C1\",\r\n primaryHover: \"#5b02a4\",\r\n background: \"#ffffff\",\r\n text: \"hsla(215, 100%, 5%, 1)\",\r\n textSecondary: \"hsla(215, 100%, 5%, 1)\",\r\n border: \"#e5e7eb\",\r\n messageBot: \"\",\r\n messageUser: \"#6C03C1\",\r\n messageUserText: \"#ffffff\",\r\n};\r\n\r\nconst normalizeHexColor = (hex: string): string => {\r\n const trimmed = hex.trim();\r\n if (!/^#([0-9a-fA-F]{3}|[0-9a-fA-F]{6})$/.test(trimmed)) {\r\n return \"hsla(215, 100%, 5%, 1)\";\r\n }\r\n if (trimmed.length === 4) {\r\n const r = trimmed[1];\r\n const g = trimmed[2];\r\n const b = trimmed[3];\r\n return `#${r}${r}${g}${g}${b}${b}`;\r\n }\r\n return trimmed;\r\n};\r\n\r\nexport const getContrastIconFilter = (backgroundHex: string): string => {\r\n const hex = normalizeHexColor(backgroundHex).slice(1);\r\n const r = parseInt(hex.slice(0, 2), 16);\r\n const g = parseInt(hex.slice(2, 4), 16);\r\n const b = parseInt(hex.slice(4, 6), 16);\r\n\r\n // Relative luminance approximation for quick contrast choice.\r\n const luminance = (0.299 * r + 0.587 * g + 0.114 * b) / 255;\r\n return luminance > 0.62 ? \"brightness(0) invert(0)\" : \"brightness(0) invert(1)\";\r\n};\r\n\r\n// Border radius\r\nexport const BORDER_RADIUS = {\r\n button: '50%',\r\n window: '17px',\r\n message: '10px'\r\n};\r\n\r\n// Shadows\r\nexport const SHADOW = \"0 0 15px hsla(215, 100%, 5%, 0.15)\";\r\n\r\n// Transitions\r\nexport const TRANSITIONS = {\r\n fast: '150ms ease-in-out',\r\n medium: '250ms ease-in-out'\r\n};\r\n","import microphoneIcon from \"./assets/icons/microphone.svg?inline\";\r\nimport microphoneIconRaw from \"./assets/icons/microphone.svg?raw\";\r\nimport restartIcon from \"./assets/icons/restart.svg?inline\";\r\nimport restartIconRaw from \"./assets/icons/restart.svg?raw\";\r\nimport closeIcon from \"./assets/icons/close.svg?inline\";\r\nimport closeIconRaw from \"./assets/icons/close.svg?raw\";\r\nimport stopIcon from \"./assets/icons/stop.svg?inline\";\r\nimport stopIconRaw from \"./assets/icons/stop.svg?raw\";\r\nimport arrowPathIconRaw from \"./assets/icons/arrow-path.svg?raw\";\r\nimport commandLineIconRaw from \"./assets/icons/command-line.svg?raw\";\r\nimport cursorArrowRaysIconRaw from \"./assets/icons/cursor-arrow-rays.svg?raw\";\r\nimport faceSmileIconRaw from \"./assets/icons/face-smile.svg?raw\";\r\nimport handRaisedIconRaw from \"./assets/icons/hand-raised.svg?raw\";\r\nimport mapIconRaw from \"./assets/icons/map.svg?raw\";\r\nimport microphoneOutlineIconRaw from \"./assets/icons/microphone-outline.svg?raw\";\r\nimport queueListIconRaw from \"./assets/icons/queue-list.svg?raw\";\r\nimport stopOutlineIconRaw from \"./assets/icons/stop-outline.svg?raw\";\r\nimport xMarkIconRaw from \"./assets/icons/x-mark.svg?raw\";\r\nimport bulutLogo from \"./assets/icons/bulut_logo.svg\";\r\nimport bulutLogoRaw from \"./assets/icons/bulut_logo.svg?raw\";\r\nimport sentSfx from \"./assets/sfx/sent.mp3\";\r\nimport thinkingSfx from \"./assets/sfx/thinking.mp3\";\r\nimport toolCallSfx from \"./assets/sfx/tool_call.mp3\";\r\nimport completedSfx from \"./assets/sfx/completed.mp3\";\r\n\r\nexport const microphoneIconUrl = microphoneIcon;\r\nexport const microphoneIconContent = microphoneIconRaw;\r\nexport const restartIconUrl = restartIcon;\r\nexport const restartIconContent = restartIconRaw;\r\nexport const closeIconUrl = closeIcon;\r\nexport const closeIconContent = closeIconRaw;\r\nexport const stopIconUrl = stopIcon;\r\nexport const stopIconContent = stopIconRaw;\r\nexport const arrowPathIconContent = arrowPathIconRaw;\r\nexport const commandLineIconContent = commandLineIconRaw;\r\nexport const cursorArrowRaysIconContent = cursorArrowRaysIconRaw;\r\nexport const faceSmileIconContent = faceSmileIconRaw;\r\nexport const handRaisedIconContent = handRaisedIconRaw;\r\nexport const mapIconContent = mapIconRaw;\r\nexport const microphoneOutlineIconContent = microphoneOutlineIconRaw;\r\nexport const queueListIconContent = queueListIconRaw;\r\nexport const stopOutlineIconContent = stopOutlineIconRaw;\r\nexport const xMarkIconContent = xMarkIconRaw;\r\nexport const logoUrl = bulutLogo;\r\nexport const logoContent = bulutLogoRaw;\r\nexport const sentSfxUrl = sentSfx;\r\nexport const thinkingSfxUrl = thinkingSfx;\r\nexport const toolCallSfxUrl = toolCallSfx;\r\nexport const completedSfxUrl = completedSfx;\r\n","export default \"<svg xmlns=\\\"http://www.w3.org/2000/svg\\\" fill=\\\"none\\\" viewBox=\\\"0 0 24 24\\\" stroke-width=\\\"1.5\\\" stroke=\\\"currentColor\\\">\\r\\n <path stroke-linecap=\\\"round\\\" stroke-linejoin=\\\"round\\\" d=\\\"M10.05 4.575a1.575 1.575 0 1 0-3.15 0v3m3.15-3v-1.5a1.575 1.575 0 0 1 3.15 0v1.5m-3.15 0 .075 5.925m3.075.75V4.575m0 0a1.575 1.575 0 0 1 3.15 0V15M6.9 7.575a1.575 1.575 0 1 0-3.15 0v8.175a6.75 6.75 0 0 0 6.75 6.75h2.018a5.25 5.25 0 0 0 3.712-1.538l1.732-1.732a5.25 5.25 0 0 0 1.538-3.712l.003-2.024a.668.668 0 0 1 .198-.471 1.575 1.575 0 1 0-2.228-2.228 3.818 3.818 0 0 0-1.12 2.687M6.9 7.575V12m6.27 4.318A4.49 4.49 0 0 1 16.35 15m.002 0h-.002\\\" />\\r\\n</svg>\\r\\n\"","export default \"<svg xmlns=\\\"http://www.w3.org/2000/svg\\\" fill=\\\"none\\\" viewBox=\\\"0 0 24 24\\\" stroke-width=\\\"1.5\\\" stroke=\\\"currentColor\\\">\\r\\n <path stroke-linecap=\\\"round\\\" stroke-linejoin=\\\"round\\\" d=\\\"M12 18.75a6 6 0 0 0 6-6v-1.5m-6 7.5a6 6 0 0 1-6-6v-1.5m6 7.5v3.75m-3.75 0h7.5M12 15.75a3 3 0 0 1-3-3V4.5a3 3 0 1 1 6 0v8.25a3 3 0 0 1-3 3Z\\\" />\\r\\n</svg>\\r\\n\"","export default \"<svg xmlns=\\\"http://www.w3.org/2000/svg\\\" fill=\\\"none\\\" viewBox=\\\"0 0 24 24\\\" stroke-width=\\\"1.5\\\" stroke=\\\"currentColor\\\">\\r\\n <path stroke-linecap=\\\"round\\\" stroke-linejoin=\\\"round\\\" d=\\\"M5.25 7.5A2.25 2.25 0 0 1 7.5 5.25h9a2.25 2.25 0 0 1 2.25 2.25v9a2.25 2.25 0 0 1-2.25 2.25h-9a2.25 2.25 0 0 1-2.25-2.25v-9Z\\\" />\\r\\n</svg>\\r\\n\"","export default \"<svg xmlns=\\\"http://www.w3.org/2000/svg\\\" fill=\\\"none\\\" viewBox=\\\"0 0 24 24\\\" stroke-width=\\\"1.5\\\" stroke=\\\"currentColor\\\">\\r\\n <path stroke-linecap=\\\"round\\\" stroke-linejoin=\\\"round\\\" d=\\\"M6 18 18 6M6 6l12 12\\\" />\\r\\n</svg>\\r\\n\"","\r\nimport type { CSSProperties, JSX } from \"preact\";\r\nimport { useMemo } from \"preact/hooks\";\r\n\r\nexport type SvgIconProps = Omit<JSX.IntrinsicElements[\"svg\"], \"style\"> & {\r\n src: string;\r\n title?: string;\r\n /**\r\n * If true, replaces fill and stroke attributes with \"currentColor\" in the inner SVG content.\r\n * Default: true\r\n */\r\n stripColors?: boolean;\r\n style?: CSSProperties;\r\n // Explicitly add common props to ensure destructuring works smoothly\r\n className?: string;\r\n fill?: string;\r\n stroke?: string;\r\n strokeWidth?: string | number;\r\n width?: string | number;\r\n height?: string | number;\r\n};\r\n\r\nexport function SvgIcon({\r\n src,\r\n width,\r\n height,\r\n title,\r\n stripColors = true,\r\n className,\r\n style,\r\n fill,\r\n stroke,\r\n strokeWidth,\r\n ...props\r\n}: SvgIconProps) {\r\n const { viewBox, content } = useMemo(() => {\r\n if (!src) return { viewBox: \"0 0 24 24\", content: \"\" };\r\n\r\n const trimmedSrc = src.trim();\r\n // Extract viewBox\r\n const viewBoxMatch = trimmedSrc.match(/viewBox=\"([^\"]*)\"/);\r\n const viewBox = viewBoxMatch ? viewBoxMatch[1] : \"0 0 24 24\";\r\n\r\n // Extract inner content by removing the outer <svg> wrapper\r\n // We remove the opening <svg ...> tag and the closing </svg> tag only.\r\n let content = trimmedSrc.replace(/^<svg[^>]*>/i, \"\").replace(/<\\/svg>\\s*$/i, \"\");\r\n\r\n return { viewBox, content };\r\n }, [src]);\r\n\r\n const cleanContent = useMemo(() => {\r\n if (!content) return \"\";\r\n let nextContent = content;\r\n\r\n if (stripColors) {\r\n // Force inner nodes to use props instead of hardcoded SVG values.\r\n const fillValue = fill !== undefined ? fill : \"none\";\r\n const strokeValue = stroke !== undefined ? stroke : \"currentColor\";\r\n nextContent = nextContent\r\n .replace(/fill=\"[^\"]*\"/gi, `fill=\"${fillValue}\"`)\r\n .replace(/stroke=\"[^\"]*\"/gi, `stroke=\"${strokeValue}\"`);\r\n }\r\n\r\n if (strokeWidth !== undefined) {\r\n const strokeWidthValue = String(strokeWidth);\r\n nextContent = /stroke-width=\"[^\"]*\"/i.test(nextContent)\r\n ? nextContent.replace(/stroke-width=\"[^\"]*\"/gi, `stroke-width=\"${strokeWidthValue}\"`)\r\n : nextContent.replace(\r\n /<(path|circle|ellipse|line|polyline|polygon|rect)\\b/gi,\r\n `<$1 stroke-width=\"${strokeWidthValue}\"`,\r\n );\r\n }\r\n\r\n return nextContent;\r\n }, [content, stripColors, fill, stroke, strokeWidth]);\r\n\r\n return (\r\n <svg\r\n viewBox={viewBox}\r\n width={width}\r\n height={height}\r\n className={className}\r\n style={{\r\n display: \"inline-block\",\r\n verticalAlign: \"middle\",\r\n flexShrink: 0,\r\n fill: \"none\",\r\n stroke: \"currentColor\",\r\n ...style,\r\n }}\r\n xmlns=\"http://www.w3.org/2000/svg\"\r\n {...props}\r\n dangerouslySetInnerHTML={{ __html: cleanContent }}\r\n >\r\n {title && <title>{title}</title>}\r\n </svg>\r\n );\r\n}\r\n","import {\r\n BUTTON_SIZE,\r\n POSITION_BOTTOM,\r\n POSITION_RIGHT,\r\n COLORS,\r\n SHADOW,\r\n TRANSITIONS,\r\n BORDER_RADIUS,\r\n} from \"../styles/constants\";\r\nimport {\r\n microphoneOutlineIconContent,\r\n stopOutlineIconContent,\r\n xMarkIconContent,\r\n} from \"../assets\";\r\nimport { SvgIcon } from \"./SvgIcon\";\r\n\r\ninterface ChatButtonProps {\r\n onMicClick: () => void;\r\n onCancelRecording: () => void;\r\n onStopTask: () => void;\r\n isRecording: boolean;\r\n isBusy: boolean;\r\n accessibilityMode?: boolean;\r\n showBubble: boolean;\r\n onBubbleClick: () => void;\r\n previewMessage: string | null;\r\n onPreviewClick: () => void;\r\n onPreviewClose: () => void;\r\n}\r\n\r\nexport const ChatButton = ({\r\n onMicClick,\r\n onCancelRecording,\r\n onStopTask,\r\n isRecording,\r\n isBusy,\r\n accessibilityMode = false,\r\n showBubble,\r\n onBubbleClick,\r\n previewMessage,\r\n onPreviewClick,\r\n onPreviewClose,\r\n}: ChatButtonProps) => {\r\n const bgColor = COLORS.primary;\r\n const popupBoxShadow = accessibilityMode\r\n ? `inset 0 0 0 2px ${COLORS.primary}, ${SHADOW}`\r\n : SHADOW;\r\n\r\n const containerStyle: { [key: string]: string } = {\r\n position: \"fixed\",\r\n right: `${POSITION_RIGHT}px`,\r\n bottom: `${POSITION_BOTTOM}px`,\r\n display: \"flex\",\r\n flexDirection: \"column\",\r\n alignItems: \"flex-end\",\r\n gap: \"8px\",\r\n zIndex: \"9999\",\r\n fontFamily: \"\\\"Geist\\\", sans-serif\",\r\n };\r\n\r\n const controlsRowStyle: { [key: string]: string } = {\r\n display: \"flex\",\r\n alignItems: \"center\",\r\n gap: \"10px\",\r\n };\r\n\r\n const buttonStyle: { [key: string]: string } = {\r\n width: `${BUTTON_SIZE}px`,\r\n height: `${BUTTON_SIZE}px`,\r\n minWidth: `${BUTTON_SIZE}px`,\r\n minHeight: `${BUTTON_SIZE}px`,\r\n borderRadius: BORDER_RADIUS.button,\r\n backgroundColor: bgColor,\r\n color: \"#ffffff\",\r\n border: \"none\",\r\n cursor: \"pointer\",\r\n display: \"flex\",\r\n alignItems: \"center\",\r\n justifyContent: \"center\",\r\n transition: `background-color ${TRANSITIONS.fast}, transform ${TRANSITIONS.fast}`,\r\n position: \"relative\",\r\n flexShrink: \"0\",\r\n };\r\n\r\n const iconStyle: { [key: string]: string } = {\r\n width: \"24px\",\r\n height: \"24px\",\r\n display: \"block\",\r\n };\r\n\r\n const showStopButton = isBusy && !isRecording;\r\n const mainIconSrc = showStopButton ? stopOutlineIconContent : isRecording ? xMarkIconContent : microphoneOutlineIconContent;\r\n\r\n const handleClick = () => {\r\n if (showStopButton) {\r\n onStopTask();\r\n } else if (isRecording) {\r\n onCancelRecording();\r\n } else {\r\n onMicClick();\r\n }\r\n };\r\n\r\n // Shared close button style\r\n const closeBtnStyle: { [key: string]: string } = {\r\n position: \"absolute\",\r\n top: \"6px\",\r\n right: \"6px\",\r\n width: \"20px\",\r\n height: \"20px\",\r\n borderRadius: \"50%\",\r\n border: \"none\",\r\n background: \"transparent\",\r\n cursor: \"pointer\",\r\n display: \"flex\",\r\n alignItems: \"center\",\r\n justifyContent: \"center\",\r\n fontSize: \"12px\",\r\n lineHeight: \"1\",\r\n color: COLORS.text,\r\n padding: \"0\",\r\n };\r\n\r\n const renderPopup = (\r\n content: preact.ComponentChildren,\r\n onClick: () => void,\r\n onClose: (() => void) | null,\r\n extraClass: string,\r\n scrollable: boolean,\r\n ) => (\r\n <div\r\n className={`bulut-popup ${extraClass}`}\r\n style={{ cursor: \"pointer\" }}\r\n onClick={onClick}\r\n >\r\n {/* Close button */}\r\n {onClose && (\r\n <button\r\n type=\"button\"\r\n style={closeBtnStyle}\r\n onClick={(e) => {\r\n e.stopPropagation();\r\n onClose();\r\n }}\r\n aria-label=\"Kapat\"\r\n >\r\n <SvgIcon src={xMarkIconContent} aria-hidden=\"true\" width={16} height={16} />\r\n </button>\r\n )}\r\n\r\n {/* Text content */}\r\n <div\r\n style={{\r\n paddingRight: onClose ? \"22px\" : \"0\",\r\n wordBreak: \"break-word\",\r\n ...(scrollable ? { maxHeight: \"96px\", overflowY: \"auto\" } : {}),\r\n }}\r\n >\r\n {content}\r\n </div>\r\n </div>\r\n );\r\n\r\n return (\r\n <>\r\n <style>{`\r\n @import url('https://fonts.googleapis.com/css2?family=Geist:wght@100..900&display=swap');\r\n\r\n .bulut-popup {\r\n background: #ffffff;\r\n color: ${COLORS.text};\r\n padding: 10px 14px;\r\n border-radius: 12px;\r\n font-size: 14px;\r\n line-height: 1.4;\r\n position: relative;\r\n overflow: visible;\r\n box-shadow: ${popupBoxShadow};\r\n }\r\n .bulut-popup-bubble {\r\n animation: bulut-bubbleIn 400ms ease-out;\r\n }\r\n .bulut-popup-preview {\r\n animation: bulut-popIn ${TRANSITIONS.medium};\r\n }\r\n\r\n /* Desktop: limit width, lean to right */\r\n .bulut-popup {\r\n max-width: 320px;\r\n }\r\n\r\n /* Mobile: full-width layout with popup filling remaining space */\r\n @media (max-width: 600px) {\r\n .bulut-fab-container:has(.bulut-popup) {\r\n left: 16px !important;\r\n right: 16px !important;\r\n }\r\n .bulut-fab-container:has(.bulut-popup) .bulut-controls-row {\r\n width: 100%;\r\n }\r\n .bulut-popup {\r\n flex: 1;\r\n min-width: 0;\r\n max-width: none;\r\n }\r\n }\r\n\r\n @keyframes bulut-popIn {\r\n from { opacity: 0; transform: translateX(10px); }\r\n to { opacity: 1; transform: translateX(0); }\r\n }\r\n @keyframes bulut-bubbleIn {\r\n 0% { opacity: 0; transform: translateX(10px) scale(0.95); }\r\n 60% { opacity: 1; transform: translateX(-4px) scale(1.02); }\r\n 100% { opacity: 1; transform: translateX(0) scale(1); }\r\n }\r\n @keyframes bulut-badgeIn {\r\n from { opacity: 0; transform: translateY(-4px); }\r\n to { opacity: 1; transform: translateY(0); }\r\n }\r\n `}</style>\r\n\r\n <div className=\"bulut-fab-container\" style={containerStyle}>\r\n <div className=\"bulut-controls-row\" style={controlsRowStyle}>\r\n {/* Welcome bubble */}\r\n {showBubble &&\r\n !isRecording &&\r\n !previewMessage &&\r\n renderPopup(\r\n \"Destek lazımsa hemen konuşmaya başlayabiliriz!\",\r\n onBubbleClick,\r\n null,\r\n \"bulut-popup-bubble\",\r\n false,\r\n )}\r\n\r\n {/* New-message preview */}\r\n {previewMessage &&\r\n renderPopup(\r\n previewMessage,\r\n onPreviewClick,\r\n onPreviewClose,\r\n \"bulut-popup-preview\",\r\n true,\r\n )}\r\n\r\n {/* Main mic / cancel button */}\r\n <button\r\n style={buttonStyle}\r\n onClick={handleClick}\r\n onMouseEnter={(e) => {\r\n Object.assign(e.currentTarget.style, {\r\n backgroundColor: COLORS.primaryHover,\r\n transform: \"scale(1.05)\",\r\n });\r\n }}\r\n onMouseLeave={(e) => {\r\n Object.assign(e.currentTarget.style, {\r\n transform: \"scale(1)\",\r\n });\r\n }}\r\n aria-label={showStopButton ? \"Görevi durdur\" : isRecording ? \"Kaydı iptal et\" : \"Konuşmaya başla\"}\r\n >\r\n <SvgIcon\r\n src={mainIconSrc}\r\n aria-hidden=\"true\"\r\n style={iconStyle}\r\n />\r\n </button>\r\n </div>\r\n </div>\r\n </>\r\n );\r\n};\r\n","export interface PageContext {\r\n links: string[];\r\n interactables: string[];\r\n summary: string;\r\n}\r\n\r\nexport interface CachedPageContextEntry {\r\n url: string;\r\n summary: string;\r\n links: string[];\r\n interactables: string[];\r\n capturedAt: number;\r\n version: number;\r\n}\r\n\r\nexport interface PageContextSummaryInput {\r\n url: string;\r\n title: string;\r\n lang: string;\r\n headings: string[];\r\n landmarks: string[];\r\n links: string[];\r\n interactables: string[];\r\n interactionSignals: string[];\r\n styleSelectors: string[];\r\n pageBlueprint: string[];\r\n textSnippets: string[];\r\n outerHtmlDigest: string;\r\n}\r\n\r\ninterface InteractableCandidate {\r\n line: string;\r\n score: number;\r\n order: number;\r\n}\r\n\r\ninterface PageSignalSnapshot {\r\n links: string[];\r\n interactables: string[];\r\n interactionSignals: string[];\r\n styleSelectors: string[];\r\n pageBlueprint: string[];\r\n}\r\n\r\nimport {\r\n MAX_LINKS,\r\n MAX_INTERACTABLES,\r\n MAX_HEADINGS,\r\n MAX_TEXT_SNIPPETS,\r\n MAX_CACHED_PAGES,\r\n MAX_PAGE_SCAN_ELEMENTS,\r\n MAX_EVENT_HINTS_PER_ELEMENT,\r\n MAX_BRANCH_SAMPLES,\r\n MAX_BRANCH_DEPTH,\r\n MAX_STYLESHEET_SELECTORS,\r\n MAX_STYLESHEET_RULES,\r\n} from \"./contextConfig\";\r\n\r\nexport const PAGE_CONTEXT_CACHE_VERSION = 3;\r\nexport const PAGE_CONTEXT_CACHE_KEY = \"auticbot_page_context_cache_v3\";\r\n\r\nconst NON_CONTENT_TAGS = new Set([\r\n \"script\",\r\n \"style\",\r\n \"noscript\",\r\n \"template\",\r\n \"link\",\r\n \"meta\",\r\n]);\r\n\r\nconst NATIVE_INTERACTIVE_TAGS = new Set([\r\n \"a\",\r\n \"button\",\r\n \"input\",\r\n \"textarea\",\r\n \"select\",\r\n \"summary\",\r\n \"details\",\r\n \"option\",\r\n]);\r\n\r\nconst INTERACTIVE_ROLES = new Set([\r\n \"button\",\r\n \"link\",\r\n \"tab\",\r\n \"menuitem\",\r\n \"option\",\r\n \"checkbox\",\r\n \"radio\",\r\n \"switch\",\r\n \"combobox\",\r\n \"textbox\",\r\n \"searchbox\",\r\n \"slider\",\r\n \"spinbutton\",\r\n \"treeitem\",\r\n]);\r\n\r\nconst TRACKED_DISPLAY_VALUES = new Set([\r\n \"block\",\r\n \"inline\",\r\n \"inline-block\",\r\n \"flex\",\r\n \"inline-flex\",\r\n \"grid\",\r\n \"inline-grid\",\r\n]);\r\n\r\nconst TRACKED_POSITION_VALUES = new Set([\r\n \"relative\",\r\n \"absolute\",\r\n \"fixed\",\r\n \"sticky\",\r\n]);\r\n\r\nconst EVENT_HINT_NAMES = [\r\n \"click\",\r\n \"dblclick\",\r\n \"mousedown\",\r\n \"mouseup\",\r\n \"pointerdown\",\r\n \"pointerup\",\r\n \"touchstart\",\r\n \"touchend\",\r\n \"keydown\",\r\n \"keyup\",\r\n \"keypress\",\r\n \"input\",\r\n \"change\",\r\n \"submit\",\r\n \"focus\",\r\n \"blur\",\r\n];\r\n\r\nconst ARIA_INTERACTION_ATTRS = [\r\n \"aria-controls\",\r\n \"aria-expanded\",\r\n \"aria-haspopup\",\r\n \"aria-pressed\",\r\n \"aria-selected\",\r\n];\r\n\r\nconst DATA_INTERACTION_PATTERN =\r\n /(action|click|press|toggle|target|trigger|nav|open|close|menu|modal|command|submit)/i;\r\n\r\nconst STYLESHEET_SELECTOR_PATTERN =\r\n /(:hover|:focus|:active|button|a\\b|input|textarea|select|\\[role=|\\[aria-|\\[data-|\\.btn|\\.link)/i;\r\n\r\nconst pageContextCache = new Map<string, CachedPageContextEntry>();\r\nlet cacheHydrated = false;\r\n\r\nconst normalizeWhitespace = (value: string): string =>\r\n value.replace(/\\s+/g, \" \").trim();\r\n\r\nconst canonicalUrl = (rawUrl: string): string => {\r\n try {\r\n return new URL(rawUrl, rawUrl).href;\r\n } catch {\r\n return rawUrl;\r\n }\r\n};\r\n\r\nconst isCacheEntry = (value: unknown): value is CachedPageContextEntry => {\r\n if (typeof value !== \"object\" || value === null) {\r\n return false;\r\n }\r\n\r\n const obj = value as Record<string, unknown>;\r\n return (\r\n typeof obj.url === \"string\" &&\r\n typeof obj.summary === \"string\" &&\r\n Array.isArray(obj.links) &&\r\n Array.isArray(obj.interactables) &&\r\n typeof obj.capturedAt === \"number\" &&\r\n typeof obj.version === \"number\"\r\n );\r\n};\r\n\r\nconst bumpCount = (map: Map<string, number>, key: string): void => {\r\n if (!key) {\r\n return;\r\n }\r\n map.set(key, (map.get(key) ?? 0) + 1);\r\n};\r\n\r\nconst formatTopCounts = (map: Map<string, number>, maxItems: number): string => {\r\n if (map.size === 0) {\r\n return \"none\";\r\n }\r\n\r\n return Array.from(map.entries())\r\n .sort((a, b) => b[1] - a[1] || a[0].localeCompare(b[0]))\r\n .slice(0, maxItems)\r\n .map(([name, count]) => `${name}*${count}`)\r\n .join(\", \");\r\n};\r\n\r\nconst parseTabIndex = (value: string | null): number | null => {\r\n if (value === null) {\r\n return null;\r\n }\r\n\r\n const parsed = Number.parseInt(value, 10);\r\n return Number.isNaN(parsed) ? null : parsed;\r\n};\r\n\r\nconst compactToken = (value: string): string => {\r\n const compact = value.replace(/\\s+/g, \"-\").replace(/[^a-zA-Z0-9_-]/g, \"\");\r\n return compact || \"\";\r\n};\r\n\r\nconst getElementDepth = (element: Element): number => {\r\n let depth = 0;\r\n let cursor: Element | null = element;\r\n while (cursor?.parentElement) {\r\n depth += 1;\r\n cursor = cursor.parentElement;\r\n if (cursor === document.body) {\r\n break;\r\n }\r\n }\r\n return depth;\r\n};\r\n\r\nconst getPrimaryRole = (element: Element): string => {\r\n const rawRole = normalizeWhitespace(element.getAttribute(\"role\") || \"\")\r\n .toLowerCase()\r\n .split(\" \")[0];\r\n return rawRole || \"\";\r\n};\r\n\r\nconst hydrateCacheFromStorage = (): void => {\r\n if (cacheHydrated || typeof sessionStorage === \"undefined\") {\r\n return;\r\n }\r\n\r\n cacheHydrated = true;\r\n\r\n try {\r\n const raw = sessionStorage.getItem(PAGE_CONTEXT_CACHE_KEY);\r\n if (!raw) {\r\n return;\r\n }\r\n\r\n const parsed = JSON.parse(raw);\r\n if (!Array.isArray(parsed)) {\r\n return;\r\n }\r\n\r\n for (const value of parsed) {\r\n if (!isCacheEntry(value)) {\r\n continue;\r\n }\r\n if (value.version !== PAGE_CONTEXT_CACHE_VERSION) {\r\n continue;\r\n }\r\n pageContextCache.set(value.url, value);\r\n }\r\n if (pageContextCache.size > 0) {\r\n console.info(\r\n `[Autic] context cache restored entries=${pageContextCache.size}`,\r\n );\r\n }\r\n } catch (error) {\r\n console.warn(\"[Autic] context cache restore failed\", error);\r\n }\r\n};\r\n\r\nconst persistCacheToStorage = (): void => {\r\n if (typeof sessionStorage === \"undefined\") {\r\n return;\r\n }\r\n\r\n try {\r\n const serialized = JSON.stringify(\r\n Array.from(pageContextCache.values()).sort(\r\n (a, b) => a.capturedAt - b.capturedAt,\r\n ),\r\n );\r\n sessionStorage.setItem(PAGE_CONTEXT_CACHE_KEY, serialized);\r\n } catch (error) {\r\n console.warn(\"[Autic] context cache persist failed\", error);\r\n }\r\n};\r\n\r\nconst pruneOldestCacheEntries = (): void => {\r\n if (pageContextCache.size <= MAX_CACHED_PAGES) {\r\n return;\r\n }\r\n\r\n const sorted = Array.from(pageContextCache.values()).sort(\r\n (a, b) => a.capturedAt - b.capturedAt,\r\n );\r\n const overflow = sorted.length - MAX_CACHED_PAGES;\r\n for (let i = 0; i < overflow; i += 1) {\r\n pageContextCache.delete(sorted[i].url);\r\n }\r\n};\r\n\r\nconst buildSummaryWithHistory = (\r\n current: CachedPageContextEntry,\r\n): string => {\r\n const recentPages = Array.from(pageContextCache.values())\r\n .filter((entry) => entry.url !== current.url)\r\n .sort((a, b) => b.capturedAt - a.capturedAt)\r\n .slice(0, 3);\r\n\r\n if (recentPages.length === 0) {\r\n return current.summary;\r\n }\r\n\r\n const historySection = [\r\n \"Recent Page Memory:\",\r\n ...recentPages.map((entry) => {\r\n const compactSummary = normalizeWhitespace(entry.summary);\r\n return `- ${entry.url} :: ${compactSummary}`;\r\n }),\r\n ].join(\"\\n\");\r\n\r\n return `${current.summary}\\n\\n${historySection}`;\r\n};\r\n\r\nconst isVisible = (element: Element): boolean => {\r\n if (element.getAttribute(\"aria-hidden\") === \"true\") {\r\n return false;\r\n }\r\n\r\n if (element instanceof HTMLElement && element.hidden) {\r\n return false;\r\n }\r\n\r\n const style = window.getComputedStyle(element);\r\n if (style.display === \"none\" || style.visibility === \"hidden\") {\r\n return false;\r\n }\r\n\r\n const rect = element.getBoundingClientRect();\r\n return rect.width > 0 && rect.height > 0;\r\n};\r\n\r\nconst toAbsoluteUrl = (href: string): string => {\r\n try {\r\n return new URL(href, window.location.href).href;\r\n } catch {\r\n return href;\r\n }\r\n};\r\n\r\nconst escapeCssValue = (value: string): string => {\r\n if (typeof CSS !== \"undefined\" && typeof CSS.escape === \"function\") {\r\n return CSS.escape(value);\r\n }\r\n\r\n return value.replace(/([ #;&,.+*~':\"!^$\\[\\]()=>|\\/@])/g, \"\\\\$1\");\r\n};\r\n\r\nconst buildSelectorSegment = (element: Element): string => {\r\n const tag = element.tagName.toLowerCase();\r\n\r\n if (element.id) {\r\n return `#${escapeCssValue(element.id)}`;\r\n }\r\n\r\n const attrCandidates: Array<[name: string, value: string | null]> = [\r\n [\"name\", element.getAttribute(\"name\")],\r\n [\"data-testid\", element.getAttribute(\"data-testid\")],\r\n [\"data-test-id\", element.getAttribute(\"data-test-id\")],\r\n [\"aria-label\", element.getAttribute(\"aria-label\")],\r\n [\"role\", element.getAttribute(\"role\")],\r\n [\"type\", element.getAttribute(\"type\")],\r\n ];\r\n\r\n for (const [attrName, attrValue] of attrCandidates) {\r\n if (attrValue) {\r\n return `${tag}[${attrName}=\"${escapeCssValue(attrValue)}\"]`;\r\n }\r\n }\r\n\r\n const classes = Array.from(element.classList)\r\n .filter(Boolean)\r\n .slice(0, 3)\r\n .map((className) => `.${escapeCssValue(className)}`)\r\n .join(\"\");\r\n if (classes) {\r\n return `${tag}${classes}`;\r\n }\r\n\r\n const parent = element.parentElement;\r\n if (!parent) {\r\n return tag;\r\n }\r\n\r\n const siblingsOfTag = Array.from(parent.children).filter(\r\n (sibling) => sibling.tagName === element.tagName,\r\n );\r\n const index = siblingsOfTag.indexOf(element) + 1;\r\n return `${tag}:nth-of-type(${index})`;\r\n};\r\n\r\nconst buildSelector = (element: Element): string => {\r\n const segments: string[] = [];\r\n let cursor: Element | null = element;\r\n let depth = 0;\r\n\r\n while (cursor && depth < 4) {\r\n const segment = buildSelectorSegment(cursor);\r\n segments.unshift(segment);\r\n if (segment.startsWith(\"#\")) {\r\n break;\r\n }\r\n cursor = cursor.parentElement;\r\n depth += 1;\r\n }\r\n\r\n return segments.join(\" > \");\r\n};\r\n\r\nconst getElementLabel = (element: Element): string => {\r\n const text = normalizeWhitespace(\r\n (element instanceof HTMLElement ? element.innerText : element.textContent) ||\r\n \"\",\r\n );\r\n const ariaLabel = normalizeWhitespace(element.getAttribute(\"aria-label\") || \"\");\r\n const title = normalizeWhitespace(element.getAttribute(\"title\") || \"\");\r\n const placeholder = normalizeWhitespace(\r\n element.getAttribute(\"placeholder\") || \"\",\r\n );\r\n const name = normalizeWhitespace(element.getAttribute(\"name\") || \"\");\r\n const value =\r\n element instanceof HTMLInputElement ||\r\n element instanceof HTMLTextAreaElement ||\r\n element instanceof HTMLButtonElement\r\n ? normalizeWhitespace(element.value || \"\")\r\n : \"\";\r\n\r\n const classHint = Array.from(element.classList)\r\n .map((item) => compactToken(item))\r\n .find(Boolean);\r\n const fallback =\r\n (element.id && `#${element.id}`) ||\r\n (classHint && `.${classHint}`) ||\r\n buildSelector(element);\r\n\r\n const label =\r\n text || ariaLabel || title || placeholder || value || name || fallback;\r\n\r\n if (element.tagName.toLowerCase() === \"input\") {\r\n const inputType = element.getAttribute(\"type\") || \"text\";\r\n return `${inputType} ${label || \"input\"}`;\r\n }\r\n\r\n return label || \"untitled\";\r\n};\r\n\r\nconst getEventHints = (element: Element): string[] => {\r\n const record = element as unknown as Record<string, unknown>;\r\n const eventHints: string[] = [];\r\n\r\n for (const eventName of EVENT_HINT_NAMES) {\r\n const handlerKey = `on${eventName}`;\r\n const hasInlineHandler = Boolean(element.getAttribute(handlerKey));\r\n const hasPropertyHandler = typeof record[handlerKey] === \"function\";\r\n\r\n if (!hasInlineHandler && !hasPropertyHandler) {\r\n continue;\r\n }\r\n\r\n eventHints.push(eventName);\r\n if (eventHints.length >= MAX_EVENT_HINTS_PER_ELEMENT) {\r\n break;\r\n }\r\n }\r\n\r\n return eventHints;\r\n};\r\n\r\nconst getAriaInteractionHints = (element: Element): string[] =>\r\n ARIA_INTERACTION_ATTRS.filter((attrName) => element.hasAttribute(attrName)).map(\r\n (attrName) => attrName.replace(\"aria-\", \"\"),\r\n );\r\n\r\nconst getDataInteractionHints = (element: Element): string[] =>\r\n element\r\n .getAttributeNames()\r\n .filter(\r\n (attrName) =>\r\n attrName.startsWith(\"data-\") && DATA_INTERACTION_PATTERN.test(attrName),\r\n )\r\n .slice(0, 2)\r\n .map((attrName) => attrName.replace(\"data-\", \"\"));\r\n\r\nconst getComputedStyleSignals = (style: CSSStyleDeclaration): string[] => {\r\n const signals: string[] = [];\r\n\r\n if (style.cursor && style.cursor !== \"auto\") {\r\n signals.push(`cursor:${style.cursor}`);\r\n }\r\n if (style.display) {\r\n signals.push(`display:${style.display}`);\r\n }\r\n if (style.position) {\r\n signals.push(`position:${style.position}`);\r\n }\r\n if (style.zIndex && style.zIndex !== \"auto\") {\r\n signals.push(`z-index:${style.zIndex}`);\r\n }\r\n if (style.pointerEvents && style.pointerEvents !== \"auto\") {\r\n signals.push(`pointer-events:${style.pointerEvents}`);\r\n }\r\n if (style.visibility && style.visibility !== \"visible\") {\r\n signals.push(`visibility:${style.visibility}`);\r\n }\r\n if (style.opacity && style.opacity !== \"1\") {\r\n signals.push(`opacity:${style.opacity}`);\r\n }\r\n\r\n return Array.from(new Set(signals));\r\n};\r\n\r\nconst buildBlueprintToken = (element: Element): string => {\r\n const tag = element.tagName.toLowerCase();\r\n const idToken = element.id ? `#${compactToken(element.id)}` : \"\";\r\n const classToken = Array.from(element.classList)\r\n .map((item) => compactToken(item))\r\n .find(Boolean);\r\n\r\n return `${tag}${idToken}${classToken ? `.${classToken}` : \"\"}`;\r\n};\r\n\r\nconst buildBranchDigest = (element: Element, depth: number): string => {\r\n const token = buildBlueprintToken(element);\r\n if (depth <= 0) {\r\n return token;\r\n }\r\n\r\n const children = Array.from(element.children)\r\n .filter((child) => !NON_CONTENT_TAGS.has(child.tagName.toLowerCase()))\r\n .filter((child) => isVisible(child));\r\n if (children.length === 0) {\r\n return token;\r\n }\r\n\r\n const sampled = children\r\n .slice(0, 3)\r\n .map((child) => buildBranchDigest(child, depth - 1));\r\n const overflow =\r\n children.length > sampled.length ? `+${children.length - sampled.length}` : \"\";\r\n\r\n return `${token}>${sampled.join(\"+\")}${overflow}`;\r\n};\r\n\r\nconst collectDomBranchDigest = (): string[] => {\r\n const root = document.body ?? document.documentElement;\r\n const topLevelNodes = Array.from(root.children)\r\n .filter((child) => !NON_CONTENT_TAGS.has(child.tagName.toLowerCase()))\r\n .filter((child) => isVisible(child))\r\n .slice(0, MAX_BRANCH_SAMPLES);\r\n\r\n return topLevelNodes.map((child) =>\r\n buildBranchDigest(child, MAX_BRANCH_DEPTH),\r\n );\r\n};\r\n\r\nconst formatSection = (title: string, lines: string[]): string => {\r\n if (lines.length === 0) {\r\n return `${title}:\\n- none`;\r\n }\r\n\r\n return `${title}:\\n${lines.join(\"\\n\")}`;\r\n};\r\n\r\nconst buildOuterHtmlDigest = (): string => {\r\n const raw = document.body?.outerHTML || document.documentElement.outerHTML;\r\n\r\n const withoutScripts = raw\r\n .replace(/<script[\\s\\S]*?<\\/script>/gi, \"\")\r\n .replace(/<style[\\s\\S]*?<\\/style>/gi, \"\")\r\n .replace(/<noscript[\\s\\S]*?<\\/noscript>/gi, \"\")\r\n .replace(/<!--[\\s\\S]*?-->/g, \"\")\r\n .replace(/\\s+/g, \" \")\r\n .trim();\r\n\r\n const structural = withoutScripts\r\n .replace(/>[^<]*</g, \"><\")\r\n .replace(/\\s+/g, \" \")\r\n .trim();\r\n\r\n return structural;\r\n};\r\n\r\nconst collectTextSnippets = (): string[] => {\r\n const root =\r\n document.querySelector(\"main, article, [role='main']\") ?? document.body;\r\n const snippets: string[] = [];\r\n const seen = new Set<string>();\r\n\r\n const candidates = Array.from(root.querySelectorAll(\"p, li, h1, h2, h3\"));\r\n for (const node of candidates) {\r\n if (!isVisible(node)) {\r\n continue;\r\n }\r\n\r\n const text = normalizeWhitespace(node.textContent || \"\");\r\n if (!text || text.length < 20) {\r\n continue;\r\n }\r\n\r\n if (seen.has(text)) {\r\n continue;\r\n }\r\n\r\n seen.add(text);\r\n snippets.push(`- ${text}`);\r\n if (snippets.length >= MAX_TEXT_SNIPPETS) {\r\n break;\r\n }\r\n }\r\n\r\n return snippets;\r\n};\r\n\r\nconst collectLandmarkSnapshot = (): string[] => {\r\n const probes: Array<{ label: string; selector: string }> = [\r\n { label: \"main\", selector: \"main, [role='main']\" },\r\n { label: \"nav\", selector: \"nav, [role='navigation']\" },\r\n { label: \"section\", selector: \"section\" },\r\n { label: \"article\", selector: \"article\" },\r\n { label: \"form\", selector: \"form\" },\r\n { label: \"a\", selector: \"a\" },\r\n { label: \"button\", selector: \"button\" },\r\n { label: \"input\", selector: \"input\" },\r\n { label: \"role=button/link\", selector: \"[role='button'], [role='link']\" },\r\n { label: \"onclick attrs\", selector: \"[onclick]\" },\r\n {\r\n label: \"other event attrs\",\r\n selector:\r\n \"[onpointerdown], [onpointerup], [onkeydown], [onkeyup], [onchange], [onsubmit]\",\r\n },\r\n { label: \"tabindex\", selector: \"[tabindex]\" },\r\n { label: \"contenteditable\", selector: \"[contenteditable='true']\" },\r\n { label: \"inline cursor styles\", selector: \"[style*='cursor']\" },\r\n ];\r\n\r\n return probes.map(\r\n ({ label, selector }) => `- ${label}: ${document.querySelectorAll(selector).length}`,\r\n );\r\n};\r\n\r\nconst collectSelectorsFromRuleList = (\r\n rules: CSSRuleList,\r\n selectors: Set<string>,\r\n scanned: { count: number },\r\n): void => {\r\n for (const rule of Array.from(rules)) {\r\n if (\r\n scanned.count >= MAX_STYLESHEET_RULES ||\r\n selectors.size >= MAX_STYLESHEET_SELECTORS\r\n ) {\r\n return;\r\n }\r\n\r\n scanned.count += 1;\r\n\r\n if (rule instanceof CSSStyleRule) {\r\n const parts = rule.selectorText\r\n .split(\",\")\r\n .map((selector) => normalizeWhitespace(selector))\r\n .filter(Boolean);\r\n\r\n for (const selector of parts) {\r\n if (!STYLESHEET_SELECTOR_PATTERN.test(selector)) {\r\n continue;\r\n }\r\n selectors.add(selector);\r\n if (selectors.size >= MAX_STYLESHEET_SELECTORS) {\r\n return;\r\n }\r\n }\r\n continue;\r\n }\r\n\r\n if (\"cssRules\" in rule) {\r\n try {\r\n const nestedRules = (rule as CSSMediaRule).cssRules;\r\n collectSelectorsFromRuleList(nestedRules, selectors, scanned);\r\n } catch {\r\n // Ignore inaccessible nested rules.\r\n }\r\n }\r\n }\r\n};\r\n\r\nconst collectStylesheetSelectors = (): string[] => {\r\n const selectors = new Set<string>();\r\n const scanned = { count: 0 };\r\n\r\n for (const stylesheet of Array.from(document.styleSheets)) {\r\n if (\r\n scanned.count >= MAX_STYLESHEET_RULES ||\r\n selectors.size >= MAX_STYLESHEET_SELECTORS\r\n ) {\r\n break;\r\n }\r\n\r\n try {\r\n if (!stylesheet.cssRules) {\r\n continue;\r\n }\r\n collectSelectorsFromRuleList(stylesheet.cssRules, selectors, scanned);\r\n } catch {\r\n // Ignore cross-origin stylesheets.\r\n }\r\n }\r\n\r\n return Array.from(selectors).map((selector) => `- ${selector}`);\r\n};\r\n\r\nconst collectPageSignalSnapshot = (): PageSignalSnapshot => {\r\n const allElements = Array.from(document.querySelectorAll(\"*\"));\r\n const sampledElements = allElements.slice(0, MAX_PAGE_SCAN_ELEMENTS);\r\n\r\n const links: string[] = [];\r\n const linkSet = new Set<string>();\r\n const interactableCandidates = new Map<string, InteractableCandidate>();\r\n const tagCounts = new Map<string, number>();\r\n const roleCounts = new Map<string, number>();\r\n const eventCounts = new Map<string, number>();\r\n const displayCounts = new Map<string, number>();\r\n const positionCounts = new Map<string, number>();\r\n const styleSignalCounts = new Map<string, number>();\r\n\r\n let visibleElements = 0;\r\n let maxDepth = 0;\r\n let semanticInteractables = 0;\r\n let nonSemanticInteractables = 0;\r\n let eventHintElements = 0;\r\n let tabStopElements = 0;\r\n let pointerCursorElements = 0;\r\n let dataHintElements = 0;\r\n let ariaHintElements = 0;\r\n let contentEditableElements = 0;\r\n\r\n for (let order = 0; order < sampledElements.length; order += 1) {\r\n const element = sampledElements[order];\r\n const tag = element.tagName.toLowerCase();\r\n\r\n if (NON_CONTENT_TAGS.has(tag)) {\r\n continue;\r\n }\r\n\r\n if (!isVisible(element)) {\r\n continue;\r\n }\r\n\r\n visibleElements += 1;\r\n bumpCount(tagCounts, tag);\r\n\r\n const role = getPrimaryRole(element);\r\n if (role) {\r\n bumpCount(roleCounts, role);\r\n }\r\n\r\n const depth = getElementDepth(element);\r\n if (depth > maxDepth) {\r\n maxDepth = depth;\r\n }\r\n\r\n const style = window.getComputedStyle(element);\r\n if (TRACKED_DISPLAY_VALUES.has(style.display)) {\r\n bumpCount(displayCounts, style.display);\r\n }\r\n if (TRACKED_POSITION_VALUES.has(style.position)) {\r\n bumpCount(positionCounts, style.position);\r\n }\r\n\r\n const computedStyleSignals = getComputedStyleSignals(style);\r\n for (const styleSignal of computedStyleSignals) {\r\n bumpCount(styleSignalCounts, styleSignal);\r\n }\r\n\r\n const eventHints = getEventHints(element);\r\n if (eventHints.length > 0) {\r\n eventHintElements += 1;\r\n for (const eventName of eventHints) {\r\n bumpCount(eventCounts, eventName);\r\n }\r\n }\r\n\r\n const tabIndex = parseTabIndex(element.getAttribute(\"tabindex\"));\r\n const hasTabStop = tabIndex !== null && tabIndex >= 0;\r\n if (hasTabStop) {\r\n tabStopElements += 1;\r\n }\r\n\r\n const hasPointerCursor = style.cursor === \"pointer\";\r\n if (hasPointerCursor) {\r\n pointerCursorElements += 1;\r\n }\r\n\r\n const dataHints = getDataInteractionHints(element);\r\n if (dataHints.length > 0) {\r\n dataHintElements += 1;\r\n }\r\n\r\n const ariaHints = getAriaInteractionHints(element);\r\n if (ariaHints.length > 0) {\r\n ariaHintElements += 1;\r\n }\r\n\r\n const isContentEditable = element.getAttribute(\"contenteditable\") === \"true\";\r\n if (isContentEditable) {\r\n contentEditableElements += 1;\r\n }\r\n\r\n const href = element.getAttribute(\"href\");\r\n const isNativeInteractive = NATIVE_INTERACTIVE_TAGS.has(tag) && (tag !== \"a\" || Boolean(href));\r\n const isRoleInteractive = INTERACTIVE_ROLES.has(role);\r\n const isDisabled =\r\n element.hasAttribute(\"disabled\") ||\r\n element.getAttribute(\"aria-disabled\") === \"true\";\r\n\r\n if (\r\n tag === \"a\" &&\r\n href &&\r\n !href.startsWith(\"#\") &&\r\n !href.startsWith(\"javascript:\")\r\n ) {\r\n const absoluteHref = toAbsoluteUrl(href);\r\n const label = getElementLabel(element) || absoluteHref;\r\n const line = `- ${label} -> ${absoluteHref}`;\r\n\r\n if (!linkSet.has(line)) {\r\n linkSet.add(line);\r\n links.push(line);\r\n if (links.length >= MAX_LINKS) {\r\n // Keep scanning other elements for page blueprint and interactables.\r\n }\r\n }\r\n }\r\n\r\n const hasInteractionSignals =\r\n isNativeInteractive ||\r\n isRoleInteractive ||\r\n isContentEditable ||\r\n eventHints.length > 0 ||\r\n hasTabStop ||\r\n hasPointerCursor ||\r\n dataHints.length > 0 ||\r\n ariaHints.length > 0;\r\n\r\n if (!hasInteractionSignals || isDisabled) {\r\n continue;\r\n }\r\n\r\n if (isNativeInteractive) {\r\n semanticInteractables += 1;\r\n } else {\r\n nonSemanticInteractables += 1;\r\n }\r\n\r\n const selector = buildSelector(element);\r\n const label = getElementLabel(element);\r\n const styleSignals = computedStyleSignals;\r\n const signalTokens: string[] = [];\r\n\r\n if (eventHints.length > 0) {\r\n signalTokens.push(`evt:${eventHints.join(\"|\")}`);\r\n }\r\n if (isRoleInteractive) {\r\n signalTokens.push(`role:${role}`);\r\n }\r\n if (hasTabStop) {\r\n signalTokens.push(`tab:${tabIndex}`);\r\n }\r\n if (dataHints.length > 0) {\r\n signalTokens.push(`data:${dataHints.join(\"|\")}`);\r\n }\r\n if (ariaHints.length > 0) {\r\n signalTokens.push(`aria:${ariaHints.join(\"|\")}`);\r\n }\r\n if (styleSignals.length > 0) {\r\n signalTokens.push(`css:${styleSignals.join(\"|\")}`);\r\n } else if (hasPointerCursor) {\r\n signalTokens.push(\"css:cursor:pointer\");\r\n }\r\n\r\n const signalBlock =\r\n signalTokens.length > 0 ? ` [${signalTokens.join(\"; \")}]` : \"\";\r\n const line = `- ${tag} ${selector}${signalBlock} (${label})`;\r\n\r\n const score =\r\n eventHints.length * 5 +\r\n (isNativeInteractive ? 5 : 0) +\r\n (isRoleInteractive ? 4 : 0) +\r\n (hasTabStop ? 2 : 0) +\r\n (hasPointerCursor ? 2 : 0) +\r\n (dataHints.length > 0 ? 2 : 0) +\r\n (ariaHints.length > 0 ? 1 : 0) +\r\n (isContentEditable ? 2 : 0);\r\n\r\n const existing = interactableCandidates.get(line);\r\n if (!existing || score > existing.score) {\r\n interactableCandidates.set(line, { line, score, order });\r\n }\r\n }\r\n\r\n const interactables = Array.from(interactableCandidates.values())\r\n .sort((a, b) => b.score - a.score || a.order - b.order)\r\n .slice(0, MAX_INTERACTABLES)\r\n .map((candidate) => candidate.line);\r\n\r\n const interactiveRoleCounts = new Map(\r\n Array.from(roleCounts.entries()).filter(([role]) =>\r\n INTERACTIVE_ROLES.has(role),\r\n ),\r\n );\r\n\r\n const interactionSignals = [\r\n `- coverage: semantic=${semanticInteractables}, non-semantic=${nonSemanticInteractables}, contenteditable=${contentEditableElements}`,\r\n `- listener hints: ${formatTopCounts(eventCounts, 8)}`,\r\n `- interaction cues: tabindex>=0=${tabStopElements}, pointer-cursor=${pointerCursorElements}, data-hints=${dataHintElements}, aria-hints=${ariaHintElements}`,\r\n `- role hints: ${formatTopCounts(interactiveRoleCounts, 8)}`,\r\n `- css footprint: ${formatTopCounts(styleSignalCounts, 10)}`,\r\n \"- listener scope: inline/on* handlers are detected directly; addEventListener handlers are inferred via cues.\",\r\n ];\r\n\r\n const branchDigest = collectDomBranchDigest();\r\n const pageBlueprint = [\r\n `- nodes: total=${allElements.length}, scanned=${sampledElements.length}, visible=${visibleElements}, max-depth=${maxDepth}${allElements.length > sampledElements.length ? \", sampling=on\" : \"\"}`,\r\n `- tag density: ${formatTopCounts(tagCounts, 10)}`,\r\n `- role density: ${formatTopCounts(roleCounts, 8)}`,\r\n `- layout density: display(${formatTopCounts(displayCounts, 6)}), position(${formatTopCounts(positionCounts, 4)})`,\r\n `- branch digest: ${branchDigest.length > 0 ? branchDigest.join(\" || \") : \"none\"}`,\r\n ];\r\n\r\n return {\r\n links: links.slice(0, MAX_LINKS),\r\n interactables,\r\n interactionSignals,\r\n styleSelectors: collectStylesheetSelectors(),\r\n pageBlueprint,\r\n };\r\n};\r\n\r\nexport const buildPageContextSummary = (\r\n input: PageContextSummaryInput,\r\n): string => {\r\n const sections = [\r\n formatSection(\"Meta\", [\r\n `- URL: ${input.url || \"unknown\"}`,\r\n `- Title: ${input.title || \"unknown\"}`,\r\n `- Lang: ${input.lang || \"unknown\"}`,\r\n ]),\r\n formatSection(\"Headings\", input.headings),\r\n formatSection(\"Landmark Snapshot\", input.landmarks),\r\n formatSection(\"Interaction Signals\", input.interactionSignals),\r\n formatSection(\"Stylesheet Selector Snapshot\", input.styleSelectors),\r\n formatSection(\"Compressed Page Blueprint\", input.pageBlueprint),\r\n formatSection(\"Top Links\", input.links),\r\n formatSection(\"Top Interactables\", input.interactables),\r\n formatSection(\"Main Content Snippets\", input.textSnippets),\r\n formatSection(\"OuterHTML Skeleton\", [\r\n `- ${input.outerHtmlDigest || \"unavailable\"}`,\r\n ]),\r\n ];\r\n\r\n return sections.join(\"\\n\\n\");\r\n};\r\n\r\nexport const clearPageContextCache = (): void => {\r\n pageContextCache.clear();\r\n cacheHydrated = false;\r\n if (typeof sessionStorage !== \"undefined\") {\r\n sessionStorage.removeItem(PAGE_CONTEXT_CACHE_KEY);\r\n }\r\n};\r\n\r\nexport const getCachedPageContexts = (): CachedPageContextEntry[] => {\r\n hydrateCacheFromStorage();\r\n return Array.from(pageContextCache.values()).sort(\r\n (a, b) => b.capturedAt - a.capturedAt,\r\n );\r\n};\r\n\r\nexport const getPageContext = (): PageContext => {\r\n if (typeof window === \"undefined\" || typeof document === \"undefined\") {\r\n return {\r\n links: [],\r\n interactables: [],\r\n summary: \"\",\r\n };\r\n }\r\n\r\n hydrateCacheFromStorage();\r\n const url = canonicalUrl(window.location.href);\r\n const cached = pageContextCache.get(url);\r\n if (cached) {\r\n console.info(`[Autic] context cache hit url=${url}`);\r\n return {\r\n links: cached.links,\r\n interactables: cached.interactables,\r\n summary: buildSummaryWithHistory(cached),\r\n };\r\n }\r\n\r\n console.info(`[Autic] context cache miss url=${url}`);\r\n\r\n const snapshot = collectPageSignalSnapshot();\r\n const headings = Array.from(document.querySelectorAll(\"h1, h2, h3\"))\r\n .filter((element) => isVisible(element))\r\n .map((element) =>\r\n `- ${normalizeWhitespace(element.textContent || \"\")}`,\r\n )\r\n .filter((line) => line !== \"- \")\r\n .slice(0, MAX_HEADINGS);\r\n\r\n const summary = buildPageContextSummary({\r\n url,\r\n title: document.title,\r\n lang: document.documentElement.lang,\r\n headings,\r\n landmarks: collectLandmarkSnapshot(),\r\n links: snapshot.links,\r\n interactables: snapshot.interactables,\r\n interactionSignals: snapshot.interactionSignals,\r\n styleSelectors: snapshot.styleSelectors,\r\n pageBlueprint: snapshot.pageBlueprint,\r\n textSnippets: collectTextSnippets(),\r\n outerHtmlDigest: buildOuterHtmlDigest(),\r\n });\r\n\r\n const entry: CachedPageContextEntry = {\r\n url,\r\n summary,\r\n links: snapshot.links,\r\n interactables: snapshot.interactables,\r\n capturedAt: Date.now(),\r\n version: PAGE_CONTEXT_CACHE_VERSION,\r\n };\r\n\r\n pageContextCache.set(url, entry);\r\n pruneOldestCacheEntries();\r\n persistCacheToStorage();\r\n console.info(\r\n `[Autic] context cache stored url=${url} size=${pageContextCache.size}`,\r\n );\r\n\r\n return {\r\n links: entry.links,\r\n interactables: entry.interactables,\r\n summary: buildSummaryWithHistory(entry),\r\n };\r\n};\r\n","/**\r\n * Tunable parameters for page context collection.\r\n *\r\n * Adjust these values to control how much information the agent receives\r\n * about the current page. Lower values reduce token usage (and latency);\r\n * higher values give the LLM more detail to work with.\r\n */\r\n\r\n/** Maximum number of links to include in the context. */\r\nexport const MAX_LINKS = 20;\r\n\r\n/** Maximum number of interactable elements to include. */\r\nexport const MAX_INTERACTABLES = 24;\r\n\r\n/** Maximum number of headings (h1-h3) to include. */\r\nexport const MAX_HEADINGS = 10;\r\n\r\n/** Maximum number of main-content text snippets. */\r\nexport const MAX_TEXT_SNIPPETS = 4;\r\n\r\n/** Maximum number of pages kept in the in-memory context cache. */\r\nexport const MAX_CACHED_PAGES = 20;\r\n\r\n/** Maximum number of DOM elements scanned per page. */\r\nexport const MAX_PAGE_SCAN_ELEMENTS = 2000;\r\n\r\n/** Maximum number of event-handler hints collected per element. */\r\nexport const MAX_EVENT_HINTS_PER_ELEMENT = 4;\r\n\r\n/** Maximum top-level DOM branches sampled for the branch digest. */\r\nexport const MAX_BRANCH_SAMPLES = 4;\r\n\r\n/** Maximum tree depth for each branch digest sample. */\r\nexport const MAX_BRANCH_DEPTH = 2;\r\n\r\n/** Maximum stylesheet selectors to include in summary. */\r\nexport const MAX_STYLESHEET_SELECTORS = 80;\r\n\r\n/** Maximum CSS rules to scan while collecting stylesheet selectors. */\r\nexport const MAX_STYLESHEET_RULES = 700;\r\n","import { getPageContext } from \"./context\";\r\nimport { COLORS } from \"../styles/constants\";\r\n\r\nconst AGENT_CURSOR_ID = \"auticbot-agent-cursor\";\r\nexport const CURSOR_MOVE_DURATION_MS = 900;\r\nexport const SCROLL_DURATION_MS = 900;\r\nconst CURSOR_EASING = \"cubic-bezier(0.4, 0, 0.2, 1)\";\r\nconst CURSOR_HOVER_RADIUS_PX = 14;\r\nconst CURSOR_DIAMETER_PX = 25;\r\n\r\n// ── Pending Agent Resume State (survives page reloads) ──────────────\r\n\r\nconst RESUME_STORAGE_KEY = \"bulut_agent_resume\";\r\nconst RESUME_TTL_MS = 60_000; // 1 minute\r\n\r\nexport interface PendingAgentResume {\r\n sessionId: string;\r\n projectId: string;\r\n model: string;\r\n voice: string;\r\n accessibilityMode: boolean;\r\n pendingToolCalls: Array<{\r\n call_id: string;\r\n tool: string;\r\n args: Record<string, unknown>;\r\n }>;\r\n completedResults: Array<{ call_id: string; result: string }>;\r\n savedAt: number;\r\n}\r\n\r\nexport const savePendingAgentResume = (\r\n state: Omit<PendingAgentResume, \"savedAt\">,\r\n): void => {\r\n if (typeof localStorage === \"undefined\") return;\r\n try {\r\n localStorage.setItem(\r\n RESUME_STORAGE_KEY,\r\n JSON.stringify({ ...state, savedAt: Date.now() }),\r\n );\r\n } catch {\r\n // localStorage may be full or blocked\r\n }\r\n};\r\n\r\nexport const getPendingAgentResume = (): PendingAgentResume | null => {\r\n if (typeof localStorage === \"undefined\") return null;\r\n const raw = localStorage.getItem(RESUME_STORAGE_KEY);\r\n if (!raw) return null;\r\n try {\r\n const parsed = JSON.parse(raw) as PendingAgentResume;\r\n if (Date.now() - parsed.savedAt > RESUME_TTL_MS) {\r\n clearPendingAgentResume();\r\n return null;\r\n }\r\n return parsed;\r\n } catch {\r\n clearPendingAgentResume();\r\n return null;\r\n }\r\n};\r\n\r\nexport const clearPendingAgentResume = (): void => {\r\n if (typeof localStorage === \"undefined\") return;\r\n localStorage.removeItem(RESUME_STORAGE_KEY);\r\n};\r\n\r\ntype InteractAction = \"move\" | \"click\" | \"type\" | \"submit\";\r\n\r\ninterface InteractToolCall {\r\n tool: \"interact\";\r\n action: InteractAction;\r\n selector?: string;\r\n text?: string;\r\n x?: number;\r\n y?: number;\r\n}\r\n\r\ninterface NavigateToolCall {\r\n tool: \"navigate\";\r\n url: string;\r\n}\r\n\r\ninterface GetPageContextToolCall {\r\n tool: \"getPageContext\";\r\n}\r\n\r\ninterface ScrollToolCall {\r\n tool: \"scroll\";\r\n selector: string;\r\n}\r\n\r\nexport type AgentToolCall =\r\n | InteractToolCall\r\n | NavigateToolCall\r\n | GetPageContextToolCall\r\n | ScrollToolCall;\r\n\r\nexport interface ParsedAgentResponse {\r\n reply: string;\r\n toolCalls: AgentToolCall[];\r\n}\r\n\r\ninterface JsonObject {\r\n [key: string]: unknown;\r\n}\r\n\r\ninterface ResolvedTarget {\r\n element?: HTMLElement;\r\n x: number;\r\n y: number;\r\n}\r\n\r\nconst isObject = (value: unknown): value is JsonObject =>\r\n typeof value === \"object\" && value !== null && !Array.isArray(value);\r\n\r\nconst asString = (value: unknown): string | undefined =>\r\n typeof value === \"string\" && value.trim() ? value.trim() : undefined;\r\n\r\nconst asNumber = (value: unknown): number | undefined =>\r\n typeof value === \"number\" && Number.isFinite(value) ? value : undefined;\r\n\r\nconst extractJsonCandidate = (raw: string): string => {\r\n const trimmed = raw.trim();\r\n const fencedMatch = trimmed.match(/```(?:json)?\\s*([\\s\\S]*?)\\s*```/i);\r\n if (fencedMatch?.[1]) {\r\n return fencedMatch[1].trim();\r\n }\r\n return trimmed;\r\n};\r\n\r\nconst extractFirstJsonObject = (input: string): string | null => {\r\n const start = input.indexOf(\"{\");\r\n if (start < 0) {\r\n return null;\r\n }\r\n\r\n let depth = 0;\r\n let inString = false;\r\n let isEscaped = false;\r\n\r\n for (let i = start; i < input.length; i += 1) {\r\n const char = input[i];\r\n\r\n if (inString) {\r\n if (isEscaped) {\r\n isEscaped = false;\r\n } else if (char === \"\\\\\") {\r\n isEscaped = true;\r\n } else if (char === '\"') {\r\n inString = false;\r\n }\r\n continue;\r\n }\r\n\r\n if (char === '\"') {\r\n inString = true;\r\n continue;\r\n }\r\n\r\n if (char === \"{\") {\r\n depth += 1;\r\n continue;\r\n }\r\n\r\n if (char === \"}\") {\r\n depth -= 1;\r\n if (depth === 0) {\r\n return input.slice(start, i + 1);\r\n }\r\n }\r\n }\r\n\r\n return null;\r\n};\r\n\r\nconst parseJsonFromRaw = (raw: string): unknown => {\r\n const candidate = extractJsonCandidate(raw);\r\n try {\r\n return JSON.parse(candidate);\r\n } catch {\r\n const objectCandidate = extractFirstJsonObject(candidate);\r\n if (!objectCandidate) {\r\n return null;\r\n }\r\n try {\r\n return JSON.parse(objectCandidate);\r\n } catch {\r\n return null;\r\n }\r\n }\r\n};\r\n\r\nconst sanitizeToolCalls = (value: unknown): AgentToolCall[] => {\r\n if (!Array.isArray(value)) {\r\n return [];\r\n }\r\n\r\n const toolCalls: AgentToolCall[] = [];\r\n\r\n for (const item of value) {\r\n if (!isObject(item)) {\r\n continue;\r\n }\r\n\r\n if (item.tool === \"interact\") {\r\n const action = asString(item.action) as InteractAction | undefined;\r\n if (!action || ![\"move\", \"click\", \"type\", \"submit\"].includes(action)) {\r\n continue;\r\n }\r\n\r\n toolCalls.push({\r\n tool: \"interact\",\r\n action,\r\n selector: asString(item.selector),\r\n text: typeof item.text === \"string\" ? item.text : undefined,\r\n x: asNumber(item.x),\r\n y: asNumber(item.y),\r\n });\r\n continue;\r\n }\r\n\r\n if (item.tool === \"navigate\") {\r\n const url = asString(item.url);\r\n if (!url) {\r\n continue;\r\n }\r\n\r\n toolCalls.push({\r\n tool: \"navigate\",\r\n url,\r\n });\r\n continue;\r\n }\r\n\r\n if (item.tool === \"getPageContext\") {\r\n toolCalls.push({\r\n tool: \"getPageContext\",\r\n });\r\n continue;\r\n }\r\n\r\n if (item.tool === \"scroll\") {\r\n const selector = asString(item.selector);\r\n if (!selector) {\r\n continue;\r\n }\r\n\r\n toolCalls.push({\r\n tool: \"scroll\",\r\n selector,\r\n });\r\n }\r\n }\r\n\r\n return toolCalls;\r\n};\r\n\r\nexport const parseAgentResponse = (raw: string): ParsedAgentResponse => {\r\n const parsed = parseJsonFromRaw(raw);\r\n if (!isObject(parsed)) {\r\n return {\r\n reply: raw.trim(),\r\n toolCalls: [],\r\n };\r\n }\r\n\r\n const reply = asString(parsed.reply) || \"\";\r\n const toolCalls = sanitizeToolCalls(parsed.tool_calls ?? parsed.toolCalls);\r\n\r\n return {\r\n reply,\r\n toolCalls,\r\n };\r\n};\r\n\r\nexport const clamp = (value: number, min: number, max: number): number =>\r\n Math.min(max, Math.max(min, value));\r\n\r\nexport const easeInOutCubic = (progress: number): number => {\r\n if (progress < 0.5) {\r\n return 4 * progress * progress * progress;\r\n }\r\n return 1 - Math.pow(-2 * progress + 2, 3) / 2;\r\n};\r\n\r\nexport const easeInOutSine = (progress: number): number =>\r\n -(Math.cos(Math.PI * progress) - 1) / 2;\r\n\r\nexport const isRectOutsideViewport = (\r\n rect: Pick<DOMRect, \"top\" | \"bottom\">,\r\n viewportHeight: number,\r\n): boolean => rect.top < 0 || rect.bottom > viewportHeight;\r\n\r\nexport const computeCenteredScrollTop = (\r\n currentScrollY: number,\r\n rectTop: number,\r\n rectHeight: number,\r\n viewportHeight: number,\r\n maxScrollTop: number,\r\n): number => {\r\n const desired =\r\n currentScrollY + rectTop - (viewportHeight / 2 - rectHeight / 2);\r\n return clamp(desired, 0, Math.max(0, maxScrollTop));\r\n};\r\n\r\nexport const animateWindowScrollTo = async (\r\n targetY: number,\r\n durationMs: number = SCROLL_DURATION_MS,\r\n): Promise<void> => {\r\n if (typeof window === \"undefined\") {\r\n return;\r\n }\r\n\r\n const startY = window.scrollY;\r\n const delta = targetY - startY;\r\n if (Math.abs(delta) < 1) {\r\n return;\r\n }\r\n\r\n await new Promise<void>((resolve) => {\r\n const raf =\r\n window.requestAnimationFrame ||\r\n ((callback: FrameRequestCallback) => window.setTimeout(() => callback(performance.now()), 16));\r\n\r\n const startTime = performance.now();\r\n\r\n const step = (now: number) => {\r\n const elapsed = now - startTime;\r\n const progress = clamp(elapsed / durationMs, 0, 1);\r\n const eased = easeInOutSine(progress);\r\n window.scrollTo(0, startY + delta * eased);\r\n\r\n if (progress < 1) {\r\n raf(step);\r\n } else {\r\n resolve();\r\n }\r\n };\r\n\r\n raf(step);\r\n });\r\n};\r\n\r\nconst setCursorPosition = (cursor: HTMLElement, x: number, y: number) => {\r\n cursor.style.left = `${x}px`;\r\n cursor.style.top = `${y}px`;\r\n};\r\n\r\nconst getCursorPosition = (cursor: HTMLElement): { x: number; y: number } => ({\r\n x: Number.parseFloat(cursor.style.left) || 0,\r\n y: Number.parseFloat(cursor.style.top) || 0,\r\n});\r\n\r\nconst setCursorVisibility = (cursor: HTMLElement, visible: boolean) => {\r\n cursor.style.opacity = visible ? \"1\" : \"0\";\r\n};\r\n\r\nconst isVisibleElement = (element: HTMLElement): boolean => {\r\n const style = window.getComputedStyle(element);\r\n return style.display !== \"none\" && style.visibility !== \"hidden\" && style.opacity !== \"0\";\r\n};\r\n\r\nconst getBulutShadowRoots = (): ShadowRoot[] => {\r\n const roots: ShadowRoot[] = [];\r\n\r\n const defaultHost = document.getElementById(\"bulut-container\");\r\n if (defaultHost?.shadowRoot) {\r\n roots.push(defaultHost.shadowRoot);\r\n }\r\n\r\n const allElements = document.querySelectorAll<HTMLElement>(\"*\");\r\n for (const el of allElements) {\r\n if (!el.shadowRoot) {\r\n continue;\r\n }\r\n if (!roots.includes(el.shadowRoot)) {\r\n roots.push(el.shadowRoot);\r\n }\r\n }\r\n\r\n return roots;\r\n};\r\n\r\nconst findAgentUiAnchorElement = (): HTMLElement | null => {\r\n const roots = getBulutShadowRoots();\r\n\r\n for (const root of roots) {\r\n const panel = root.querySelector<HTMLElement>(\".bulut-chat-window\");\r\n if (panel && isVisibleElement(panel)) {\r\n return panel;\r\n }\r\n }\r\n\r\n for (const root of roots) {\r\n const button = root.querySelector<HTMLElement>(\".bulut-fab-container\");\r\n if (button && isVisibleElement(button)) {\r\n return button;\r\n }\r\n }\r\n\r\n return null;\r\n};\r\n\r\nconst getAgentWindowTopLeft = (): { x: number; y: number } => {\r\n const anchor = findAgentUiAnchorElement();\r\n if (!anchor) {\r\n return {\r\n x: CURSOR_DIAMETER_PX / 2,\r\n y: CURSOR_DIAMETER_PX / 2,\r\n };\r\n }\r\n\r\n const rect = anchor.getBoundingClientRect();\r\n return {\r\n x: rect.left + window.scrollX + CURSOR_DIAMETER_PX / 2,\r\n y: rect.top + window.scrollY + CURSOR_DIAMETER_PX / 2,\r\n };\r\n};\r\n\r\nexport const hideAgentCursor = (): void => {\r\n if (typeof document === \"undefined\" || typeof window === \"undefined\") {\r\n return;\r\n }\r\n\r\n const cursor = document.getElementById(AGENT_CURSOR_ID);\r\n if (!(cursor instanceof HTMLElement)) {\r\n return;\r\n }\r\n\r\n setCursorVisibility(cursor, false);\r\n};\r\n\r\nlet cursorHoverTrackingInitialized = false;\r\nconst initializeCursorHoverTracking = () => {\r\n if (cursorHoverTrackingInitialized) {\r\n return;\r\n }\r\n cursorHoverTrackingInitialized = true;\r\n\r\n document.addEventListener(\"mousemove\", (event) => {\r\n const cursor = document.getElementById(AGENT_CURSOR_ID);\r\n if (!(cursor instanceof HTMLElement)) {\r\n return;\r\n }\r\n\r\n if (cursor.style.opacity !== \"1\") {\r\n return;\r\n }\r\n\r\n const { x, y } = getCursorPosition(cursor);\r\n const pointerX = event.pageX;\r\n const pointerY = event.pageY;\r\n const distance = Math.hypot(pointerX - x, pointerY - y);\r\n\r\n if (distance <= CURSOR_HOVER_RADIUS_PX) {\r\n setCursorVisibility(cursor, false);\r\n }\r\n });\r\n};\r\n\r\nconst ensureCursor = (): HTMLElement => {\r\n const existing = document.getElementById(AGENT_CURSOR_ID);\r\n if (existing) {\r\n // Keep cursor color in sync with the current theme\r\n existing.style.background = COLORS.primary;\r\n initializeCursorHoverTracking();\r\n return existing as HTMLElement;\r\n }\r\n\r\n const cursor = document.createElement(\"div\");\r\n cursor.id = AGENT_CURSOR_ID;\r\n cursor.style.position = \"absolute\";\r\n const startPosition = getAgentWindowTopLeft();\r\n cursor.style.left = `${startPosition.x}px`;\r\n cursor.style.top = `${startPosition.y}px`;\r\n cursor.style.opacity = \"0\";\r\n const width = CURSOR_DIAMETER_PX;\r\n cursor.style.width = `${width}px`;\r\n cursor.style.height = `${width}px`;\r\n cursor.style.borderRadius = \"50%\";\r\n const baseColor = COLORS.primary;\r\n cursor.style.background = baseColor;\r\n const border = 25 * 16 / 100;\r\n cursor.style.border = `${border}px solid #ffffff`;\r\n cursor.style.boxShadow = \"0px 0px 10px rgba(0, 11, 26, 0.5)\";\r\n cursor.style.boxSizing = \"border-box\";\r\n cursor.style.zIndex = \"2147483647\";\r\n cursor.style.pointerEvents = \"none\";\r\n cursor.style.transform = \"translate(-50%, -50%)\";\r\n cursor.style.transition = `left ${CURSOR_MOVE_DURATION_MS}ms ${CURSOR_EASING}, top ${CURSOR_MOVE_DURATION_MS}ms ${CURSOR_EASING}, opacity 150ms ease-out`;\r\n document.body.appendChild(cursor);\r\n initializeCursorHoverTracking();\r\n console.info(`[Autic] cursor created color=${baseColor} duration=${CURSOR_MOVE_DURATION_MS}ms`);\r\n return cursor;\r\n};\r\n\r\nconst waitForNextAnimationFrame = async (): Promise<void> => {\r\n const raf =\r\n window.requestAnimationFrame ||\r\n ((callback: FrameRequestCallback) => window.setTimeout(() => callback(performance.now()), 16));\r\n await new Promise<void>((resolve) => {\r\n raf(() => resolve());\r\n });\r\n};\r\n\r\nconst moveCursor = async (x: number, y: number) => {\r\n const cursor = ensureCursor();\r\n if (cursor.dataset.transitionReady !== \"true\") {\r\n cursor.dataset.transitionReady = \"true\";\r\n await waitForNextAnimationFrame();\r\n }\r\n\r\n const startPosition = getAgentWindowTopLeft();\r\n setCursorVisibility(cursor, true);\r\n setCursorPosition(cursor, startPosition.x, startPosition.y);\r\n await new Promise((resolve) => setTimeout(resolve, CURSOR_MOVE_DURATION_MS));\r\n\r\n setCursorPosition(cursor, x, y);\r\n await new Promise((resolve) => setTimeout(resolve, CURSOR_MOVE_DURATION_MS));\r\n};\r\n\r\nconst getElementCenter = (element: HTMLElement): { x: number; y: number } => {\r\n const rect = element.getBoundingClientRect();\r\n return {\r\n x: rect.left + window.scrollX + rect.width / 2,\r\n y: rect.top + window.scrollY + rect.height / 2,\r\n };\r\n};\r\n\r\nconst CONTAINS_SELECTOR_PATTERN = /^(.*?):contains\\((['\"])(.*?)\\2\\)\\s*$/;\r\n\r\nconst findElementBySelector = (selector: string): Element | null => {\r\n try {\r\n return document.querySelector(selector);\r\n } catch (error) {\r\n const containsMatch = selector.match(CONTAINS_SELECTOR_PATTERN);\r\n if (!containsMatch) {\r\n console.warn(`AuticBot selector invalid: ${selector}`, error);\r\n return null;\r\n }\r\n\r\n const baseSelector = containsMatch[1]?.trim() || \"*\";\r\n const expectedText = containsMatch[3]?.trim() || \"\";\r\n if (!expectedText) {\r\n console.warn(`AuticBot selector contains empty text: ${selector}`);\r\n return null;\r\n }\r\n\r\n try {\r\n const candidates = document.querySelectorAll(baseSelector);\r\n for (const candidate of candidates) {\r\n if (candidate.textContent?.includes(expectedText)) {\r\n return candidate;\r\n }\r\n }\r\n return null;\r\n } catch (fallbackError) {\r\n console.warn(`AuticBot selector fallback invalid: ${selector}`, fallbackError);\r\n return null;\r\n }\r\n }\r\n};\r\n\r\nconst resolveTarget = (call: InteractToolCall): ResolvedTarget | null => {\r\n if (call.selector) {\r\n const selected = findElementBySelector(call.selector);\r\n\r\n if (selected instanceof HTMLElement) {\r\n const center = getElementCenter(selected);\r\n return {\r\n element: selected,\r\n x: center.x,\r\n y: center.y,\r\n };\r\n }\r\n console.warn(`AuticBot interact: selector not found: ${call.selector}`);\r\n }\r\n\r\n if (typeof call.x === \"number\" && typeof call.y === \"number\") {\r\n return {\r\n x: call.x,\r\n y: call.y,\r\n };\r\n }\r\n\r\n console.warn(\"AuticBot interact: missing target selector or coordinates.\", call);\r\n return null;\r\n};\r\n\r\nconst dispatchMouseEvent = (\r\n element: HTMLElement,\r\n type: string,\r\n x: number,\r\n y: number,\r\n) => {\r\n element.dispatchEvent(\r\n new MouseEvent(type, {\r\n bubbles: true,\r\n cancelable: true,\r\n view: window,\r\n clientX: x - window.scrollX,\r\n clientY: y - window.scrollY,\r\n }),\r\n );\r\n};\r\n\r\nconst setNativeInputLikeValue = (\r\n element: HTMLInputElement | HTMLTextAreaElement,\r\n text: string,\r\n) => {\r\n const prototype =\r\n element instanceof HTMLTextAreaElement\r\n ? HTMLTextAreaElement.prototype\r\n : HTMLInputElement.prototype;\r\n const descriptor = Object.getOwnPropertyDescriptor(prototype, \"value\");\r\n if (descriptor?.set) {\r\n descriptor.set.call(element, text);\r\n } else {\r\n element.value = text;\r\n }\r\n\r\n // Keep both current and default values aligned so client re-focus\r\n // does not appear to \"erase\" tool-filled form fields.\r\n element.defaultValue = text;\r\n element.setAttribute(\"value\", text);\r\n};\r\n\r\nconst typeIntoElement = (element: HTMLElement, text: string) => {\r\n if (element instanceof HTMLInputElement || element instanceof HTMLTextAreaElement) {\r\n element.focus();\r\n setNativeInputLikeValue(element, text);\r\n element.dispatchEvent(new Event(\"input\", { bubbles: true }));\r\n element.dispatchEvent(new Event(\"change\", { bubbles: true }));\r\n return;\r\n }\r\n\r\n if (element.isContentEditable) {\r\n element.focus();\r\n element.textContent = text;\r\n element.dispatchEvent(new Event(\"input\", { bubbles: true }));\r\n return;\r\n }\r\n\r\n console.warn(\r\n \"AuticBot interact: type action requires input, textarea, or contenteditable target.\",\r\n );\r\n};\r\n\r\nconst submitElement = (element: HTMLElement) => {\r\n if (element.tagName === \"FORM\") {\r\n (element as HTMLFormElement).requestSubmit();\r\n return;\r\n }\r\n\r\n if (element.tagName === \"BUTTON\" && (element as HTMLButtonElement).form) {\r\n (element as HTMLButtonElement).form?.requestSubmit();\r\n return;\r\n }\r\n\r\n const parentForm = element.closest(\"form\");\r\n if (parentForm) {\r\n parentForm.requestSubmit();\r\n return;\r\n }\r\n\r\n console.warn(\"AuticBot interact: submit action requires a form target.\");\r\n};\r\n\r\nconst slowScrollElementIntoView = async (element: HTMLElement): Promise<void> => {\r\n await slowScrollElementIntoViewWithMode(element, false);\r\n};\r\n\r\nconst slowScrollElementIntoViewWithMode = async (\r\n element: HTMLElement,\r\n forceCenter: boolean,\r\n): Promise<void> => {\r\n const rect = element.getBoundingClientRect();\r\n const viewportHeight = window.innerHeight;\r\n\r\n if (!forceCenter && !isRectOutsideViewport(rect, viewportHeight)) {\r\n return;\r\n }\r\n\r\n const maxScrollTop = Math.max(\r\n 0,\r\n Math.max(document.body.scrollHeight, document.documentElement.scrollHeight) - viewportHeight,\r\n );\r\n const targetY = computeCenteredScrollTop(\r\n window.scrollY,\r\n rect.top,\r\n rect.height,\r\n viewportHeight,\r\n maxScrollTop,\r\n );\r\n\r\n await animateWindowScrollTo(targetY, SCROLL_DURATION_MS);\r\n};\r\n\r\nconst executeScroll = async (call: ScrollToolCall) => {\r\n const selected = findElementBySelector(call.selector);\r\n if (!(selected instanceof HTMLElement)) {\r\n console.warn(`AuticBot scroll: selector not found: ${call.selector}`);\r\n return;\r\n }\r\n\r\n await slowScrollElementIntoViewWithMode(selected, true);\r\n const center = getElementCenter(selected);\r\n await moveCursor(center.x, center.y);\r\n};\r\n\r\nconst executeInteract = async (call: InteractToolCall) => {\r\n const target = resolveTarget(call);\r\n if (!target) {\r\n return;\r\n }\r\n\r\n if (call.action === \"click\" && target.element) {\r\n await slowScrollElementIntoView(target.element);\r\n const center = getElementCenter(target.element);\r\n target.x = center.x;\r\n target.y = center.y;\r\n }\r\n\r\n await moveCursor(target.x, target.y);\r\n\r\n if (call.action === \"move\") {\r\n return;\r\n }\r\n\r\n if (!target.element) {\r\n console.warn(\"AuticBot interact: target element not available for action.\", call.action);\r\n return;\r\n }\r\n\r\n if (call.action === \"click\") {\r\n dispatchMouseEvent(target.element, \"pointerdown\", target.x, target.y);\r\n dispatchMouseEvent(target.element, \"mousedown\", target.x, target.y);\r\n dispatchMouseEvent(target.element, \"pointerup\", target.x, target.y);\r\n dispatchMouseEvent(target.element, \"mouseup\", target.x, target.y);\r\n target.element.click();\r\n return;\r\n }\r\n\r\n if (call.action === \"type\") {\r\n typeIntoElement(target.element, call.text ?? \"\");\r\n return;\r\n }\r\n\r\n submitElement(target.element);\r\n};\r\n\r\nconst isSamePageNavigation = (targetUrl: string): boolean => {\r\n try {\r\n const current = new URL(window.location.href);\r\n const target = new URL(targetUrl);\r\n return current.origin === target.origin && current.pathname === target.pathname;\r\n } catch {\r\n return false;\r\n }\r\n};\r\n\r\n/**\r\n * Find the best matching link element for a target URL.\r\n * Supports exact href match, partial path/query/hash match,\r\n * and text-content match for framework <Link> components.\r\n */\r\nconst findMatchingLinkForTarget = (targetUrl: string): HTMLElement | null => {\r\n let parsedTarget: URL | null = null;\r\n try {\r\n parsedTarget = new URL(targetUrl, window.location.href);\r\n } catch {\r\n // will fall through to text-based matching\r\n }\r\n\r\n const allLinks = Array.from(\r\n document.querySelectorAll('a[href], [role=\"link\"][href], [data-href]'),\r\n ) as HTMLElement[];\r\n\r\n // 1. Exact href match\r\n for (const el of allLinks) {\r\n if (el instanceof HTMLAnchorElement && el.href === parsedTarget?.href) {\r\n return el;\r\n }\r\n }\r\n\r\n if (parsedTarget) {\r\n // 2. Match by pathname + search + hash (ignoring origin)\r\n for (const el of allLinks) {\r\n if (!(el instanceof HTMLAnchorElement)) continue;\r\n try {\r\n const elUrl = new URL(el.href, window.location.href);\r\n if (\r\n elUrl.pathname === parsedTarget.pathname &&\r\n elUrl.search === parsedTarget.search &&\r\n elUrl.hash === parsedTarget.hash\r\n ) {\r\n return el;\r\n }\r\n } catch {\r\n continue;\r\n }\r\n }\r\n\r\n // 3. Match by pathname only (query/hash may differ)\r\n for (const el of allLinks) {\r\n if (!(el instanceof HTMLAnchorElement)) continue;\r\n try {\r\n const elUrl = new URL(el.href, window.location.href);\r\n if (elUrl.pathname === parsedTarget.pathname) {\r\n return el;\r\n }\r\n } catch {\r\n continue;\r\n }\r\n }\r\n\r\n // 4. Partial href attribute match (covers relative paths, query strings)\r\n const rawUrl = targetUrl.replace(/^\\//, \"\");\r\n for (const el of allLinks) {\r\n const href = el.getAttribute(\"href\") || el.getAttribute(\"data-href\") || \"\";\r\n if (href && (href === targetUrl || href === rawUrl || href === `/${rawUrl}`)) {\r\n return el;\r\n }\r\n }\r\n }\r\n\r\n // 5. Text-content match (for framework <Link> or <button> navigations)\r\n const urlSegments = targetUrl\r\n .replace(/^https?:\\/\\/[^/]+/, \"\")\r\n .replace(/[?#].*$/, \"\")\r\n .split(\"/\")\r\n .filter(Boolean);\r\n const lastSegment = urlSegments[urlSegments.length - 1] || \"\";\r\n\r\n if (lastSegment) {\r\n // Also search query param values (e.g., ?tab=interact → \"interact\")\r\n let searchTerms = [lastSegment];\r\n if (parsedTarget) {\r\n for (const [, value] of parsedTarget.searchParams.entries()) {\r\n if (value) searchTerms.push(value);\r\n }\r\n if (parsedTarget.hash) {\r\n searchTerms.push(parsedTarget.hash.replace(/^#/, \"\"));\r\n }\r\n }\r\n searchTerms = searchTerms.map((t) => t.toLowerCase());\r\n\r\n // Look across all clickable elements\r\n const clickables = Array.from(\r\n document.querySelectorAll(\r\n 'a, button, [role=\"link\"], [role=\"tab\"], [role=\"button\"], [data-tab], [onclick]',\r\n ),\r\n ) as HTMLElement[];\r\n\r\n for (const el of clickables) {\r\n const text = (el.textContent || \"\").trim().toLowerCase();\r\n const ariaLabel = (el.getAttribute(\"aria-label\") || \"\").toLowerCase();\r\n const dataTab = (el.getAttribute(\"data-tab\") || \"\").toLowerCase();\r\n for (const term of searchTerms) {\r\n if (\r\n text === term ||\r\n ariaLabel === term ||\r\n dataTab === term ||\r\n text.includes(term)\r\n ) {\r\n return el;\r\n }\r\n }\r\n }\r\n }\r\n\r\n return null;\r\n};\r\n\r\nconst executeNavigate = async (call: NavigateToolCall): Promise<boolean> => {\r\n try {\r\n const targetUrl = call.url;\r\n let resolvedUrl: string;\r\n try {\r\n resolvedUrl = new URL(targetUrl, window.location.href).href;\r\n } catch {\r\n resolvedUrl = targetUrl;\r\n }\r\n\r\n const matchingElement = findMatchingLinkForTarget(targetUrl);\r\n\r\n if (matchingElement) {\r\n console.log(\"AuticBot navigate: clicking element\", resolvedUrl, matchingElement.tagName);\r\n await slowScrollElementIntoView(matchingElement);\r\n\r\n const center = getElementCenter(matchingElement);\r\n await moveCursor(center.x, center.y);\r\n\r\n matchingElement.dispatchEvent(new MouseEvent(\"pointerdown\", { bubbles: true, view: window }));\r\n matchingElement.dispatchEvent(new MouseEvent(\"mousedown\", { bubbles: true, view: window }));\r\n matchingElement.dispatchEvent(new MouseEvent(\"pointerup\", { bubbles: true, view: window }));\r\n matchingElement.dispatchEvent(new MouseEvent(\"mouseup\", { bubbles: true, view: window }));\r\n matchingElement.click();\r\n\r\n return !isSamePageNavigation(resolvedUrl);\r\n }\r\n\r\n // 6. Fallback: direct browser navigation (query param, hash, or full URL)\r\n console.log(\"AuticBot navigate: no matching element found, using direct navigation\", resolvedUrl);\r\n\r\n // Hash-only navigation\r\n try {\r\n const parsed = new URL(resolvedUrl);\r\n if (\r\n parsed.origin === window.location.origin &&\r\n parsed.pathname === window.location.pathname &&\r\n parsed.hash\r\n ) {\r\n window.location.hash = parsed.hash;\r\n return false;\r\n }\r\n } catch { /* continue */ }\r\n\r\n // Query-param or same-origin navigation via History API\r\n try {\r\n const parsed = new URL(resolvedUrl);\r\n if (parsed.origin === window.location.origin) {\r\n // Use pushState + popstate to trigger SPA routers\r\n const newPath = parsed.pathname + parsed.search + parsed.hash;\r\n window.history.pushState({}, \"\", newPath);\r\n window.dispatchEvent(new PopStateEvent(\"popstate\", { state: {} }));\r\n return false;\r\n }\r\n } catch { /* continue */ }\r\n\r\n // Cross-origin: full page navigation\r\n window.location.href = resolvedUrl;\r\n return true;\r\n } catch (error) {\r\n console.warn(\"AuticBot navigate: error\", call.url, error);\r\n return false;\r\n }\r\n};\r\n\r\nconst executeGetPageContext = async () => {\r\n const context = getPageContext();\r\n console.info(\r\n `[Autic] getPageContext tool executed links=${context.links.length} interactables=${context.interactables.length} summary_len=${context.summary.length}`,\r\n );\r\n};\r\n\r\nexport const executeToolCalls = async (toolCalls: AgentToolCall[]) => {\r\n for (const toolCall of toolCalls) {\r\n if (toolCall.tool === \"interact\") {\r\n await executeInteract(toolCall);\r\n continue;\r\n }\r\n\r\n if (toolCall.tool === \"scroll\") {\r\n await executeScroll(toolCall);\r\n continue;\r\n }\r\n\r\n if (toolCall.tool === \"getPageContext\") {\r\n await executeGetPageContext();\r\n continue;\r\n }\r\n\r\n if (toolCall.tool === \"navigate\") {\r\n const terminalNavigation = await executeNavigate(toolCall);\r\n if (terminalNavigation) {\r\n break;\r\n }\r\n }\r\n }\r\n};\r\n\r\n// ── Agent-mode tool execution (returns results) ─────────────────────\r\n\r\nexport type ToolCallWithId = AgentToolCall & {\r\n call_id: string;\r\n};\r\n\r\nexport interface ToolCallResult {\r\n call_id: string;\r\n result: string;\r\n}\r\n\r\n/**\r\n * Execute a single tool call and return a result string.\r\n * Used by the agent loop to feed results back into the LLM.\r\n */\r\nexport const executeSingleToolCall = async (\r\n call: ToolCallWithId,\r\n): Promise<ToolCallResult> => {\r\n const callId = call.call_id;\r\n try {\r\n if (call.tool === \"interact\") {\r\n await executeInteract(call);\r\n return {\r\n call_id: callId,\r\n result: `Etkileşim başarılı: ${call.action}`,\r\n };\r\n }\r\n\r\n if (call.tool === \"scroll\") {\r\n await executeScroll(call);\r\n return {\r\n call_id: callId,\r\n result: \"Öğeye kaydırma başarılı.\",\r\n };\r\n }\r\n\r\n if (call.tool === \"getPageContext\") {\r\n const context = getPageContext();\r\n return {\r\n call_id: callId,\r\n result: context.summary,\r\n };\r\n }\r\n\r\n if (call.tool === \"navigate\") {\r\n await executeNavigate(call);\r\n // Wait for navigation / SPA routing to settle\r\n await new Promise((resolve) => setTimeout(resolve, 1500));\r\n const context = getPageContext();\r\n return {\r\n call_id: callId,\r\n result: `Navigasyon tamamlandı. Şu anki sayfa: ${window.location.href}\\nSayfa bağlamı: ${context.summary}`,\r\n };\r\n }\r\n\r\n return { call_id: callId, result: \"Bilinmeyen araç.\" };\r\n } catch (error) {\r\n const msg = error instanceof Error ? error.message : String(error);\r\n console.warn(`[Autic] Tool execution error: ${call.tool}`, error);\r\n return { call_id: callId, result: `Hata: ${msg}` };\r\n }\r\n};\r\n","import {\r\n savePendingAgentResume,\r\n clearPendingAgentResume,\r\n hideAgentCursor,\r\n type PendingAgentResume,\r\n} from \"../agent/tools\";\r\n\r\nexport type ChatRole = \"system\" | \"user\" | \"assistant\";\r\n\r\nexport interface ChatMessage {\r\n role: ChatRole;\r\n content: string;\r\n}\r\n\r\ninterface ApiErrorBody {\r\n detail?: string;\r\n error?: string;\r\n message?: string;\r\n}\r\n\r\ninterface SseEventPayload {\r\n type?: string;\r\n session_id?: string;\r\n user_text?: string;\r\n assistant_text?: string;\r\n delta?: string;\r\n audio?: string;\r\n format?: string;\r\n mime_type?: string;\r\n sample_rate?: number;\r\n error?: string;\r\n}\r\n\r\ninterface TtsWsEventPayload {\r\n type?: string;\r\n request_id?: string;\r\n seq?: number;\r\n audio?: string;\r\n format?: string;\r\n mime_type?: string;\r\n sample_rate?: number;\r\n error?: string;\r\n retryable?: boolean;\r\n last_seq?: number;\r\n}\r\n\r\ninterface SttWsEventPayload {\r\n type?: string;\r\n session_id?: string;\r\n seq?: number;\r\n text?: string;\r\n error?: string;\r\n retryable?: boolean;\r\n}\r\n\r\nexport type AudioStreamState = \"rendering\" | \"playing\" | \"done\" | \"fallback\";\r\nexport const TTS_WS_RETRY_DELAYS_MS = [250, 750, 1500];\r\n\r\nconst BULUT_AUDIO_STOP_EVENT = \"bulut:audio-stop\";\r\nconst activeAudioElements = new Set<HTMLAudioElement>();\r\nlet audioPlaybackGeneration = 0;\r\n\r\nconst normalizeBaseUrl = (baseUrl: string): string => {\r\n const trimmed = baseUrl.trim().replace(/\\/+$/, \"\");\r\n if (/^https?:\\/\\//i.test(trimmed)) {\r\n return trimmed;\r\n }\r\n // Treat host-only values like \"api.bulut.lu\" as HTTPS absolute URLs.\r\n return `https://${trimmed}`;\r\n};\r\nconst toWebSocketUrl = (baseUrl: string, path: string): string => {\r\n const normalized = normalizeBaseUrl(baseUrl);\r\n const url = new URL(normalized);\r\n url.protocol = url.protocol === \"https:\" ? \"wss:\" : \"ws:\";\r\n url.pathname = `${url.pathname.replace(/\\/$/, \"\")}${path}`;\r\n url.search = \"\";\r\n url.hash = \"\";\r\n return url.toString();\r\n};\r\n\r\nconst createRequestId = (): string => {\r\n if (typeof crypto !== \"undefined\" && \"randomUUID\" in crypto) {\r\n return crypto.randomUUID();\r\n }\r\n return `tts-${Date.now()}-${Math.random().toString(16).slice(2)}`;\r\n};\r\n\r\nexport const parseTtsWsEventPayload = (\r\n value: unknown,\r\n): TtsWsEventPayload | null => {\r\n try {\r\n if (typeof value !== \"string\") {\r\n return null;\r\n }\r\n return JSON.parse(value) as TtsWsEventPayload;\r\n } catch {\r\n return null;\r\n }\r\n};\r\n\r\nexport const parseSttWsEventPayload = (\r\n value: unknown,\r\n): SttWsEventPayload | null => {\r\n try {\r\n if (typeof value !== \"string\") {\r\n return null;\r\n }\r\n return JSON.parse(value) as SttWsEventPayload;\r\n } catch {\r\n return null;\r\n }\r\n};\r\n\r\nexport const shouldAcceptAudioSeq = (\r\n incomingSeq: number,\r\n highestSeqSeen: number,\r\n): boolean => incomingSeq > highestSeqSeen;\r\n\r\nexport const shouldFallbackToSse = (error: unknown): boolean => {\r\n if (typeof error === \"object\" && error !== null && \"retryable\" in error) {\r\n return Boolean((error as { retryable?: boolean }).retryable);\r\n }\r\n return true;\r\n};\r\n\r\nconst parseErrorBody = async (response: Response): Promise<string> => {\r\n try {\r\n const data = (await response.json()) as ApiErrorBody;\r\n const detail = data.detail;\r\n if (typeof detail === \"string\") return detail;\r\n if (detail && typeof detail === \"object\") return JSON.stringify(detail);\r\n return data.error || data.message || response.statusText;\r\n } catch {\r\n return response.statusText;\r\n }\r\n};\r\n\r\nconst sleep = (ms: number): Promise<void> =>\r\n new Promise((resolve) => {\r\n setTimeout(resolve, ms);\r\n });\r\n\r\nconst registerActiveAudioElement = (audioElement: HTMLAudioElement): void => {\r\n activeAudioElements.add(audioElement);\r\n};\r\n\r\nconst unregisterActiveAudioElement = (audioElement: HTMLAudioElement): void => {\r\n activeAudioElements.delete(audioElement);\r\n};\r\n\r\nconst wasPlaybackStoppedAfter = (generationAtStart: number): boolean =>\r\n audioPlaybackGeneration !== generationAtStart;\r\n\r\nexport const getAudioPlaybackGeneration = (): number => audioPlaybackGeneration;\r\n\r\nexport const stopActiveAudioPlayback = (): void => {\r\n audioPlaybackGeneration += 1;\r\n const active = Array.from(activeAudioElements);\r\n for (const audioElement of active) {\r\n try {\r\n audioElement.dispatchEvent(new Event(BULUT_AUDIO_STOP_EVENT));\r\n audioElement.pause();\r\n audioElement.removeAttribute(\"src\");\r\n audioElement.load();\r\n } catch {\r\n // Ignore playback stop errors.\r\n }\r\n }\r\n};\r\n\r\nexport const base64ToUint8Array = (base64: string): Uint8Array<ArrayBuffer> => {\r\n // Strip potential data URI prefix if present\r\n const cleanBase64 = base64.replace(/^data:audio\\/\\w+;base64,/, \"\");\r\n const binaryString = atob(cleanBase64);\r\n const bytes = new Uint8Array(binaryString.length) as Uint8Array<ArrayBuffer>;\r\n for (let i = 0; i < binaryString.length; i += 1) {\r\n bytes[i] = binaryString.charCodeAt(i);\r\n }\r\n return bytes;\r\n};\r\n\r\nconst blobToBase64 = async (blob: Blob): Promise<string> => {\r\n const bytes = new Uint8Array(await blob.arrayBuffer());\r\n const chunkSize = 0x8000;\r\n let binary = \"\";\r\n for (let i = 0; i < bytes.length; i += chunkSize) {\r\n binary += String.fromCharCode(...bytes.subarray(i, i + chunkSize));\r\n }\r\n return btoa(binary);\r\n};\r\n\r\nconst createWavHeader = (\r\n length: number,\r\n sampleRate: number = 16000,\r\n): Uint8Array<ArrayBuffer> => {\r\n const buffer = new ArrayBuffer(44);\r\n const view = new DataView(buffer);\r\n const channels = 1;\r\n\r\n // RIFF chunk descriptor\r\n view.setUint32(0, 0x52494646, false); // \"RIFF\"\r\n view.setUint32(4, 36 + length, true); // file length - 8\r\n view.setUint32(8, 0x57415645, false); // \"WAVE\"\r\n\r\n // fmt sub-chunk\r\n view.setUint32(12, 0x666d7420, false); // \"fmt \"\r\n view.setUint32(16, 16, true); // Subchunk1Size (16 for PCM)\r\n view.setUint16(20, 1, true); // AudioFormat (1 for PCM)\r\n view.setUint16(22, channels, true); // NumChannels\r\n view.setUint32(24, sampleRate, true); // SampleRate\r\n view.setUint32(28, sampleRate * channels * 2, true); // ByteRate\r\n view.setUint16(32, channels * 2, true); // BlockAlign\r\n view.setUint16(34, 16, true); // BitsPerSample\r\n\r\n // data sub-chunk\r\n view.setUint32(36, 0x64617461, false); // \"data\"\r\n view.setUint32(40, length, true); // Subchunk2Size\r\n\r\n return new Uint8Array(buffer) as Uint8Array<ArrayBuffer>;\r\n};\r\nconst waitForPlaybackEnd = async (\r\n audioElement: HTMLAudioElement,\r\n): Promise<void> => {\r\n if (audioElement.ended) {\r\n return;\r\n }\r\n\r\n await new Promise<void>((resolve, reject) => {\r\n const watchdog = window.setInterval(() => {\r\n if (!audioElement.ended) {\r\n console.info(\"[Bulut] playback watchdog: still playing...\");\r\n }\r\n }, 30000);\r\n\r\n const onEnded = () => {\r\n cleanup();\r\n resolve();\r\n };\r\n\r\n const onError = () => {\r\n cleanup();\r\n reject(new Error(\"Ses oynatma hatası oluştu.\"));\r\n };\r\n\r\n const onForcedStop = () => {\r\n cleanup();\r\n resolve();\r\n };\r\n\r\n const cleanup = () => {\r\n window.clearInterval(watchdog);\r\n audioElement.removeEventListener(\"ended\", onEnded);\r\n audioElement.removeEventListener(\"error\", onError);\r\n audioElement.removeEventListener(BULUT_AUDIO_STOP_EVENT, onForcedStop);\r\n };\r\n\r\n audioElement.addEventListener(\"ended\", onEnded);\r\n audioElement.addEventListener(\"error\", onError);\r\n audioElement.addEventListener(BULUT_AUDIO_STOP_EVENT, onForcedStop);\r\n });\r\n};\r\n\r\n\r\n\r\nconst playBufferedAudio = async (\r\n chunks: Uint8Array<ArrayBuffer>[],\r\n mimeType: string,\r\n sampleRate: number = 16000,\r\n onAudioStateChange?: (state: AudioStreamState) => void,\r\n): Promise<void> => {\r\n const playbackGeneration = getAudioPlaybackGeneration();\r\n if (chunks.length === 0) {\r\n onAudioStateChange?.(\"done\");\r\n return;\r\n }\r\n\r\n if (wasPlaybackStoppedAfter(playbackGeneration)) {\r\n onAudioStateChange?.(\"done\");\r\n return;\r\n }\r\n\r\n // Debug info\r\n const totalBytes = chunks.reduce((acc, c) => acc + c.byteLength, 0);\r\n console.log(`[Bulut] Playing buffered audio: ${chunks.length} chunks, ${totalBytes} bytes, type=${mimeType}`);\r\n\r\n onAudioStateChange?.(\"fallback\");\r\n\r\n const blobParts: ArrayBuffer[] = chunks.map((chunk) => {\r\n const copied = new Uint8Array(chunk.byteLength) as Uint8Array<ArrayBuffer>;\r\n copied.set(chunk);\r\n return copied.buffer;\r\n });\r\n\r\n // Verify magic numbers and detect MIME type\r\n let detectedMime = mimeType;\r\n if (chunks.length > 0 && chunks[0].length >= 4) {\r\n const header = Array.from(chunks[0].slice(0, 4))\r\n .map(b => b.toString(16).padStart(2, '0').toUpperCase())\r\n .join(' ');\r\n console.log(`[Bulut] Audio header (hex): ${header}`);\r\n\r\n // Magic number detection\r\n if (header.startsWith(\"49 44 33\")) { // ID3\r\n detectedMime = \"audio/mpeg\";\r\n } else if (header.startsWith(\"FF F3\") || header.startsWith(\"FF F2\")) { // MP3 Sync\r\n detectedMime = \"audio/mpeg\";\r\n } else if (header.startsWith(\"52 49 46 46\")) { // RIFF (WAV)\r\n detectedMime = \"audio/wav\";\r\n } else if (header.startsWith(\"1A 45 DF A3\")) { // EBML (WebM)\r\n detectedMime = \"audio/webm\";\r\n }\r\n }\r\n\r\n // Ensure valid MIME type\r\n // Ensure valid MIME type or wrap raw PCM\r\n let safeMimeType = detectedMime && detectedMime.includes(\"/\") ? detectedMime : \"audio/mpeg\";\r\n let finalBlobParts: BlobPart[] = blobParts;\r\n\r\n if (mimeType === \"audio/pcm\") {\r\n // Wrap raw PCM in WAV container\r\n const totalLength = chunks.reduce((acc, c) => acc + c.byteLength, 0);\r\n const header = createWavHeader(totalLength, sampleRate);\r\n finalBlobParts = [header.buffer, ...blobParts];\r\n safeMimeType = \"audio/wav\";\r\n console.log(`[Bulut] Wrapped raw PCM in WAV (rate=${sampleRate})`);\r\n }\r\n\r\n console.log(`[Bulut] Creating blob with type: ${safeMimeType} (original: ${mimeType})`);\r\n const blob = new Blob(finalBlobParts, { type: safeMimeType });\r\n\r\n const audioElement = new Audio();\r\n const objectUrl = URL.createObjectURL(blob);\r\n\r\n try {\r\n registerActiveAudioElement(audioElement);\r\n\r\n audioElement.preload = \"auto\";\r\n audioElement.autoplay = true;\r\n // Some browsers need this\r\n audioElement.setAttribute(\"playsinline\", \"true\");\r\n audioElement.src = objectUrl;\r\n\r\n if (wasPlaybackStoppedAfter(playbackGeneration)) {\r\n onAudioStateChange?.(\"done\");\r\n return;\r\n }\r\n\r\n await audioElement.play();\r\n onAudioStateChange?.(\"playing\");\r\n await waitForPlaybackEnd(audioElement);\r\n onAudioStateChange?.(\"done\");\r\n } catch (err) {\r\n console.error(`[Bulut] Playback failed: ${err}`, { mimeType: safeMimeType, size: blob.size });\r\n onAudioStateChange?.(\"done\"); // Signal done to unblock UI even on error\r\n throw err;\r\n } finally {\r\n unregisterActiveAudioElement(audioElement);\r\n audioElement.pause();\r\n audioElement.removeAttribute(\"src\");\r\n audioElement.load();\r\n URL.revokeObjectURL(objectUrl);\r\n }\r\n};\r\n\r\nexport interface StreamController {\r\n stop: () => void;\r\n done: Promise<void>;\r\n}\r\n\r\nexport const parseSseEventPayload = (eventBlock: string): SseEventPayload | null => {\r\n const dataLines = eventBlock\r\n .split(/\\r?\\n/)\r\n .map((line) => line.trim())\r\n .filter((line) => line.startsWith(\"data:\"))\r\n .map((line) => line.slice(5).trimStart());\r\n\r\n if (dataLines.length === 0) {\r\n return null;\r\n }\r\n\r\n const dataStr = dataLines.join(\"\\n\");\r\n if (dataStr === \"[DONE]\") {\r\n return { type: \"done\" };\r\n }\r\n\r\n try {\r\n return JSON.parse(dataStr) as SseEventPayload;\r\n } catch (error) {\r\n console.warn(\"Error parsing SSE chunk:\", error);\r\n return null;\r\n }\r\n};\r\n\r\nexport const isAudioSsePayload = (\r\n payload: SseEventPayload,\r\n): payload is SseEventPayload & { audio: string } =>\r\n typeof payload.audio === \"string\" &&\r\n (payload.type === undefined || payload.type === \"audio\");\r\n\r\n// ── Separated Endpoint Helpers ──────────────────────────────────────\r\n\r\nexport async function transcribeAudio(\r\n baseUrl: string,\r\n file: File,\r\n projectId: string,\r\n sessionId: string | null,\r\n language: string,\r\n onRequestSent?: () => void,\r\n): Promise<{ text: string; session_id: string }> {\r\n const url = `${normalizeBaseUrl(baseUrl)}/chat/stt`;\r\n const formData = new FormData();\r\n formData.append(\"file\", file);\r\n formData.append(\"project_id\", projectId);\r\n if (sessionId) formData.append(\"session_id\", sessionId);\r\n formData.append(\"language\", language);\r\n\r\n const responsePromise = fetch(url, { method: \"POST\", body: formData });\r\n onRequestSent?.();\r\n const response = await responsePromise;\r\n if (!response.ok) {\r\n throw new Error(await parseErrorBody(response));\r\n }\r\n return response.json();\r\n}\r\n\r\nexport interface SttWsResult {\r\n text: string;\r\n session_id: string;\r\n}\r\n\r\nexport interface SttWsEvents {\r\n onRequestSent?: () => void;\r\n onSessionId?: (sessionId: string) => void;\r\n onPartial?: (payload: { seq: number; text: string }) => void;\r\n}\r\n\r\nexport interface SttWsController {\r\n pushChunk: (chunk: Blob) => Promise<void>;\r\n stop: () => Promise<SttWsResult>;\r\n cancel: () => void;\r\n}\r\n\r\nexport const startSttWebSocketStream = (\r\n baseUrl: string,\r\n config: {\r\n projectId: string;\r\n sessionId: string | null;\r\n language?: string;\r\n mimeType?: string;\r\n },\r\n events: SttWsEvents = {},\r\n): SttWsController => {\r\n const wsUrl = toWebSocketUrl(baseUrl, \"/chat/stt/ws\");\r\n console.info(\"[Bulut] STT WS connecting to\", wsUrl);\r\n const socket = new WebSocket(wsUrl);\r\n let seq = 0;\r\n let finalText = \"\";\r\n let finalSessionId = config.sessionId || \"\";\r\n let stopped = false;\r\n let settled = false;\r\n // All chunk sends and the final stop are chained through sendQueue\r\n // so the \"stop\" message always follows all enqueued chunks.\r\n let sendQueue: Promise<void> = Promise.resolve();\r\n\r\n let resolveStart: (() => void) | null = null;\r\n let rejectStart: ((error: Error & { retryable?: boolean }) => void) | null = null;\r\n const startPromise = new Promise<void>((resolve, reject) => {\r\n resolveStart = resolve;\r\n rejectStart = reject;\r\n });\r\n\r\n let resolveDone: ((result: SttWsResult) => void) | null = null;\r\n let rejectDone: ((error: Error & { retryable?: boolean }) => void) | null = null;\r\n const donePromise = new Promise<SttWsResult>((resolve, reject) => {\r\n resolveDone = resolve;\r\n rejectDone = reject;\r\n });\r\n\r\n const rejectAll = (error: Error & { retryable?: boolean }) => {\r\n if (settled) return;\r\n settled = true;\r\n console.warn(\"[Bulut] STT WS rejected:\", error.message);\r\n rejectStart?.(error);\r\n rejectDone?.(error);\r\n };\r\n\r\n const resolveDoneIfPossible = () => {\r\n if (settled) return;\r\n if (!finalText.trim() || !finalSessionId) return;\r\n settled = true;\r\n resolveStart?.();\r\n resolveDone?.({\r\n text: finalText.trim(),\r\n session_id: finalSessionId,\r\n });\r\n };\r\n\r\n socket.onopen = () => {\r\n console.info(\"[Bulut] STT WS connected, sending start\");\r\n events.onRequestSent?.();\r\n socket.send(\r\n JSON.stringify({\r\n type: \"start\",\r\n project_id: config.projectId,\r\n session_id: config.sessionId || undefined,\r\n language: config.language || \"tr\",\r\n mime_type: config.mimeType || \"audio/webm\",\r\n }),\r\n );\r\n };\r\n\r\n socket.onmessage = (event) => {\r\n const payload = parseSttWsEventPayload(String(event.data));\r\n if (!payload) return;\r\n\r\n if (payload.type === \"start_ack\" && typeof payload.session_id === \"string\") {\r\n console.info(\"[Bulut] STT WS start_ack received, session:\", payload.session_id);\r\n finalSessionId = payload.session_id;\r\n events.onSessionId?.(payload.session_id);\r\n resolveStart?.();\r\n return;\r\n }\r\n\r\n if (payload.type === \"partial\" && typeof payload.text === \"string\") {\r\n events.onPartial?.({\r\n seq: typeof payload.seq === \"number\" ? payload.seq : 0,\r\n text: payload.text,\r\n });\r\n return;\r\n }\r\n\r\n if (payload.type === \"final\" && typeof payload.text === \"string\") {\r\n console.info(\"[Bulut] STT WS final text received:\", payload.text.slice(0, 80));\r\n finalText = payload.text;\r\n if (typeof payload.session_id === \"string\") {\r\n finalSessionId = payload.session_id;\r\n events.onSessionId?.(payload.session_id);\r\n }\r\n return;\r\n }\r\n\r\n if (payload.type === \"done\") {\r\n console.info(\"[Bulut] STT WS done\");\r\n resolveDoneIfPossible();\r\n socket.close();\r\n return;\r\n }\r\n\r\n if (payload.type === \"error\") {\r\n console.error(\"[Bulut] STT WS server error:\", payload.error);\r\n const err = buildError(payload.error || \"stt_ws_error\", payload.retryable !== false);\r\n rejectAll(err);\r\n socket.close();\r\n }\r\n };\r\n\r\n socket.onerror = (ev) => {\r\n console.error(\"[Bulut] STT WS transport error\", ev);\r\n rejectAll(buildError(\"stt_ws_transport_error\", true));\r\n };\r\n\r\n socket.onclose = (ev) => {\r\n console.info(\"[Bulut] STT WS closed code=%d reason=%s\", ev.code, ev.reason);\r\n if (settled) return;\r\n if (finalText && finalSessionId) {\r\n resolveDoneIfPossible();\r\n return;\r\n }\r\n rejectAll(buildError(\"stt_ws_closed_before_done\", true));\r\n };\r\n\r\n return {\r\n pushChunk: (chunk: Blob): Promise<void> => {\r\n if (stopped || chunk.size === 0) return Promise.resolve();\r\n // Chain the entire operation (wait for connection, base64-encode,\r\n // send) into sendQueue so that a later stop() is guaranteed to\r\n // follow all previously-enqueued chunks.\r\n sendQueue = sendQueue.then(async () => {\r\n if (stopped) return;\r\n await startPromise;\r\n if (stopped) return;\r\n const audio = await blobToBase64(chunk);\r\n seq += 1;\r\n if (stopped || socket.readyState !== WebSocket.OPEN) return;\r\n console.debug(\"[Bulut] STT WS sending chunk seq=%d size=%d\", seq, chunk.size);\r\n socket.send(JSON.stringify({ type: \"chunk\", seq, audio }));\r\n });\r\n return sendQueue;\r\n },\r\n stop: (): Promise<SttWsResult> => {\r\n console.info(\"[Bulut] STT WS stop requested, draining %d pending chunks\", seq);\r\n // Chain after all pending pushChunk operations so the server\r\n // always receives every chunk before the stop message.\r\n sendQueue = sendQueue.then(async () => {\r\n await startPromise;\r\n if (stopped) return;\r\n if (socket.readyState === WebSocket.OPEN) {\r\n console.info(\"[Bulut] STT WS sending stop after seq=%d\", seq);\r\n socket.send(JSON.stringify({ type: \"stop\" }));\r\n }\r\n });\r\n return donePromise;\r\n },\r\n cancel: () => {\r\n stopped = true;\r\n try {\r\n socket.close();\r\n } catch {\r\n // no-op\r\n }\r\n },\r\n };\r\n};\r\n\r\ninterface TtsCollectResult {\r\n chunks: Uint8Array<ArrayBuffer>[];\r\n mimeType: string;\r\n sampleRate: number;\r\n}\r\n\r\nconst buildError = (message: string, retryable: boolean = true): Error & { retryable: boolean } => {\r\n const error = new Error(message) as Error & { retryable: boolean };\r\n error.retryable = retryable;\r\n return error;\r\n};\r\n\r\nconst collectTtsViaSse = async (\r\n baseUrl: string,\r\n assistantText: string,\r\n voice: string,\r\n accessibilityMode: boolean,\r\n isStopped: () => boolean,\r\n setReader: (reader: ReadableStreamDefaultReader<Uint8Array> | undefined) => void,\r\n): Promise<TtsCollectResult> => {\r\n const ttsFormData = new FormData();\r\n ttsFormData.append(\"text\", assistantText);\r\n ttsFormData.append(\"voice\", voice);\r\n ttsFormData.append(\"accessibility_mode\", String(accessibilityMode));\r\n\r\n const ttsResponse = await fetch(`${normalizeBaseUrl(baseUrl)}/chat/tts`, {\r\n method: \"POST\",\r\n body: ttsFormData,\r\n });\r\n\r\n if (!ttsResponse.ok) {\r\n throw buildError(await parseErrorBody(ttsResponse), false);\r\n }\r\n\r\n const reader = ttsResponse.body?.getReader();\r\n if (!reader) {\r\n throw buildError(\"TTS response body is not readable\", false);\r\n }\r\n\r\n setReader(reader);\r\n\r\n const chunks: Uint8Array<ArrayBuffer>[] = [];\r\n let mimeType = \"audio/mpeg\";\r\n let sampleRate = 16000;\r\n const decoder = new TextDecoder();\r\n let buffer = \"\";\r\n\r\n while (true) {\r\n if (isStopped()) {\r\n break;\r\n }\r\n\r\n const { done, value } = await reader.read();\r\n if (done) {\r\n break;\r\n }\r\n\r\n buffer += decoder.decode(value, { stream: true });\r\n const blocks = buffer.split(/\\r?\\n\\r?\\n/);\r\n buffer = blocks.pop() || \"\";\r\n\r\n for (const block of blocks) {\r\n const payload = parseSseEventPayload(block);\r\n if (!payload) {\r\n continue;\r\n }\r\n\r\n if (isAudioSsePayload(payload)) {\r\n const format = payload.format || \"mp3\";\r\n mimeType = payload.mime_type || (format === \"webm\" ? \"audio/webm\" : \"audio/mpeg\");\r\n chunks.push(base64ToUint8Array(payload.audio));\r\n if (payload.sample_rate) {\r\n sampleRate = payload.sample_rate;\r\n }\r\n }\r\n }\r\n }\r\n\r\n reader.releaseLock();\r\n setReader(undefined);\r\n\r\n return { chunks, mimeType, sampleRate };\r\n};\r\n\r\nconst collectTtsViaWebSocket = async (\r\n baseUrl: string,\r\n assistantText: string,\r\n voice: string,\r\n accessibilityMode: boolean,\r\n isStopped: () => boolean,\r\n setSocket: (socket: WebSocket | null) => void,\r\n): Promise<TtsCollectResult> => {\r\n const wsUrl = toWebSocketUrl(baseUrl, \"/chat/tts/ws\");\r\n const requestId = createRequestId();\r\n const chunks: Uint8Array<ArrayBuffer>[] = [];\r\n let mimeType = \"audio/mpeg\";\r\n let sampleRate = 16000;\r\n let highestSeqSeen = 0;\r\n\r\n const connectOnce = (): Promise<void> =>\r\n new Promise((resolve, reject) => {\r\n if (isStopped()) {\r\n reject(buildError(\"stream_stopped\", false));\r\n return;\r\n }\r\n\r\n let done = false;\r\n let finalError: (Error & { retryable?: boolean }) | null = null;\r\n const socket = new WebSocket(wsUrl);\r\n setSocket(socket);\r\n\r\n const finalize = (\r\n mode: \"resolve\" | \"reject\",\r\n error?: Error & { retryable?: boolean },\r\n ) => {\r\n socket.onopen = null;\r\n socket.onmessage = null;\r\n socket.onerror = null;\r\n socket.onclose = null;\r\n setSocket(null);\r\n if (mode === \"resolve\") {\r\n resolve();\r\n return;\r\n }\r\n reject(error || buildError(\"tts_ws_closed\", true));\r\n };\r\n\r\n socket.onopen = () => {\r\n console.info(\r\n `[Bulut] TTS WS connected request_id=${requestId} resume_seq=${highestSeqSeen}`,\r\n );\r\n socket.send(\r\n JSON.stringify({\r\n type: \"start\",\r\n request_id: requestId,\r\n text: assistantText,\r\n voice,\r\n accessibility_mode: accessibilityMode,\r\n last_seq: highestSeqSeen,\r\n }),\r\n );\r\n };\r\n\r\n socket.onmessage = (event) => {\r\n const payload = parseTtsWsEventPayload(String(event.data));\r\n if (!payload) {\r\n console.warn(\"[Bulut] TTS WS invalid JSON payload\");\r\n return;\r\n }\r\n\r\n if (payload.type === \"audio\" && typeof payload.audio === \"string\") {\r\n const seq = typeof payload.seq === \"number\" ? payload.seq : 0;\r\n if (shouldAcceptAudioSeq(seq, highestSeqSeen)) {\r\n chunks.push(base64ToUint8Array(payload.audio));\r\n highestSeqSeen = seq;\r\n if (payload.mime_type) {\r\n mimeType = payload.mime_type;\r\n }\r\n if (typeof payload.sample_rate === \"number\") {\r\n sampleRate = payload.sample_rate;\r\n }\r\n } else {\r\n console.info(\r\n `[Bulut] TTS WS duplicate chunk ignored request_id=${requestId} seq=${seq} seen=${highestSeqSeen}`,\r\n );\r\n }\r\n\r\n if (socket.readyState === WebSocket.OPEN) {\r\n socket.send(\r\n JSON.stringify({\r\n type: \"ack\",\r\n request_id: requestId,\r\n last_seq: highestSeqSeen,\r\n }),\r\n );\r\n }\r\n return;\r\n }\r\n\r\n if (payload.type === \"done\") {\r\n const streamLastSeq =\r\n typeof payload.last_seq === \"number\" ? payload.last_seq : highestSeqSeen;\r\n if (streamLastSeq > highestSeqSeen) {\r\n finalError = buildError(\"tts_ws_sequence_gap\", true);\r\n done = false;\r\n socket.close();\r\n return;\r\n }\r\n done = true;\r\n socket.close();\r\n return;\r\n }\r\n\r\n if (payload.type === \"error\") {\r\n finalError = buildError(payload.error || \"tts_ws_error\", payload.retryable !== false);\r\n done = false;\r\n socket.close();\r\n }\r\n };\r\n\r\n socket.onerror = () => {\r\n if (!finalError) {\r\n finalError = buildError(\"tts_ws_transport_error\", true);\r\n }\r\n };\r\n\r\n socket.onclose = () => {\r\n if (isStopped()) {\r\n finalize(\"reject\", buildError(\"stream_stopped\", false));\r\n return;\r\n }\r\n if (done) {\r\n finalize(\"resolve\");\r\n return;\r\n }\r\n finalize(\"reject\", finalError || buildError(\"tts_ws_closed_before_done\", true));\r\n };\r\n });\r\n\r\n for (let attempt = 0; attempt <= TTS_WS_RETRY_DELAYS_MS.length; attempt += 1) {\r\n if (attempt > 0) {\r\n const delay = TTS_WS_RETRY_DELAYS_MS[attempt - 1];\r\n console.warn(\r\n `[Bulut] TTS WS retry attempt=${attempt} delay_ms=${delay} last_seq=${highestSeqSeen}`,\r\n );\r\n await sleep(delay);\r\n }\r\n\r\n try {\r\n await connectOnce();\r\n return { chunks, mimeType, sampleRate };\r\n } catch (error) {\r\n const retryable =\r\n shouldFallbackToSse(error);\r\n const message = error instanceof Error ? error.message : String(error);\r\n console.warn(\r\n `[Bulut] TTS WS attempt failed attempt=${attempt} retryable=${retryable} error=${message}`,\r\n );\r\n if (!retryable || attempt === TTS_WS_RETRY_DELAYS_MS.length) {\r\n throw error;\r\n }\r\n }\r\n }\r\n\r\n throw buildError(\"tts_ws_exhausted\", true);\r\n};\r\n\r\n// ── Agent-mode Types ────────────────────────────────────────────────\r\n\r\nexport interface AgentToolCallInfo {\r\n call_id: string;\r\n tool: string;\r\n args: Record<string, unknown>;\r\n}\r\n\r\nexport interface AgentVoiceChatEvents {\r\n onSttRequestSent?: () => void;\r\n onTranscription?: (data: {\r\n session_id: string;\r\n user_text: string;\r\n }) => void;\r\n onAssistantDelta?: (delta: string) => void;\r\n onAssistantDone?: (assistantText: string, expectsReply?: boolean) => void;\r\n onAudioStateChange?: (state: AudioStreamState) => void;\r\n onError?: (error: string) => void;\r\n /** Called when the agent requests tool execution on the frontend. */\r\n onToolCalls?: (calls: AgentToolCallInfo[]) => void;\r\n /** Called after each tool has been executed with the result. */\r\n onToolResult?: (callId: string, toolName: string, result: string) => void;\r\n /** Called at the start of each agent iteration. */\r\n onIteration?: (iteration: number, maxIterations: number) => void;\r\n /** Called when the backend confirms / creates a session ID. */\r\n onSessionId?: (sessionId: string) => void;\r\n /**\r\n * Called when the agent emits a reply text followed by tool calls.\r\n * This text is spoken aloud before the tools run so the user hears\r\n * every piece of the conversation, not just the final reply.\r\n */\r\n onIntermediateReply?: (text: string) => void;\r\n}\r\n\r\n/**\r\n * Standalone TTS helper: synthesize + play a text snippet.\r\n * Uses WebSocket TTS with SSE fallback, same as the main stream functions.\r\n */\r\nexport const speakText = async (\r\n baseUrl: string,\r\n text: string,\r\n voice: string,\r\n accessibilityMode: boolean,\r\n onAudioStateChange?: (state: AudioStreamState) => void,\r\n): Promise<void> => {\r\n const trimmed = text.trim();\r\n if (!trimmed) return;\r\n const playbackGeneration = getAudioPlaybackGeneration();\r\n\r\n console.info(`[Bulut] speakText start (${trimmed.length} chars)`);\r\n onAudioStateChange?.(\"rendering\");\r\n let ttsResult: TtsCollectResult;\r\n\r\n const neverStopped = () => false;\r\n\r\n try {\r\n ttsResult = await collectTtsViaWebSocket(\r\n baseUrl, trimmed, voice, accessibilityMode,\r\n neverStopped,\r\n () => {},\r\n );\r\n } catch {\r\n ttsResult = await collectTtsViaSse(\r\n baseUrl, trimmed, voice, accessibilityMode,\r\n neverStopped,\r\n () => {},\r\n );\r\n }\r\n\r\n if (wasPlaybackStoppedAfter(playbackGeneration)) {\r\n onAudioStateChange?.(\"done\");\r\n return;\r\n }\r\n\r\n if (ttsResult.chunks.length > 0) {\r\n await playBufferedAudio(\r\n ttsResult.chunks, ttsResult.mimeType, ttsResult.sampleRate,\r\n onAudioStateChange,\r\n );\r\n } else {\r\n onAudioStateChange?.(\"done\");\r\n }\r\n};\r\n\r\n// ── Agent Voice Chat Stream (STT → Agent WS → TTS) ─────────────────\r\n\r\nexport const agentVoiceChatStream = (\r\n baseUrl: string,\r\n audioFile: File,\r\n projectId: string,\r\n sessionId: string | null,\r\n config: {\r\n model: string;\r\n voice: string;\r\n pageContext?: string;\r\n accessibilityMode?: boolean;\r\n },\r\n events: AgentVoiceChatEvents,\r\n executeTool: (call: AgentToolCallInfo) => Promise<{ call_id: string; result: string }>,\r\n): StreamController => {\r\n let isStopped = false;\r\n let activeSocket: WebSocket | null = null;\r\n let activeReader: ReadableStreamDefaultReader<Uint8Array> | undefined;\r\n let errorEmitted = false;\r\n\r\n const donePromise = new Promise<void>(async (resolve, reject) => {\r\n try {\r\n // ── 1. STT ────────────────────────────────────────────────\r\n if (isStopped) return resolve();\r\n const sttResult = await transcribeAudio(\r\n baseUrl,\r\n audioFile,\r\n projectId,\r\n sessionId,\r\n \"tr\",\r\n events.onSttRequestSent,\r\n );\r\n\r\n const currentSessionId = sttResult.session_id;\r\n let effectiveSessionId = currentSessionId;\r\n const userText = sttResult.text;\r\n\r\n events.onTranscription?.({\r\n session_id: currentSessionId,\r\n user_text: userText,\r\n });\r\n\r\n if (isStopped) return resolve();\r\n\r\n // ── 2. Agent loop via WebSocket ───────────────────────────\r\n const assistantText = await new Promise<string>((agentResolve, agentReject) => {\r\n if (isStopped) { agentResolve(\"\"); return; }\r\n\r\n const wsUrl = toWebSocketUrl(baseUrl, \"/chat/agent/ws\");\r\n const socket = new WebSocket(wsUrl);\r\n activeSocket = socket;\r\n\r\n let finalReply = \"\";\r\n let resolved = false;\r\n let accumulatedDelta = \"\";\r\n\r\n const finish = (reply: string) => {\r\n if (resolved) return;\r\n resolved = true;\r\n agentResolve(reply);\r\n };\r\n\r\n const fail = (error: Error) => {\r\n if (resolved) return;\r\n resolved = true;\r\n agentReject(error);\r\n };\r\n\r\n socket.onopen = () => {\r\n console.info(\"[Bulut] Agent WS connected\");\r\n socket.send(JSON.stringify({\r\n type: \"start\",\r\n project_id: projectId,\r\n session_id: currentSessionId,\r\n user_text: userText,\r\n model: config.model,\r\n page_context: config.pageContext,\r\n accessibility_mode: config.accessibilityMode,\r\n }));\r\n };\r\n\r\n socket.onmessage = async (event) => {\r\n let data: Record<string, unknown>;\r\n try {\r\n data = JSON.parse(String(event.data));\r\n } catch {\r\n console.warn(\"[Bulut] Agent WS invalid JSON\");\r\n return;\r\n }\r\n\r\n const msgType = data.type as string;\r\n\r\n if (msgType === \"session\" && typeof data.session_id === \"string\") {\r\n effectiveSessionId = data.session_id as string;\r\n events.onSessionId?.(effectiveSessionId);\r\n return;\r\n }\r\n\r\n if (msgType === \"iteration\") {\r\n events.onIteration?.(\r\n data.iteration as number,\r\n data.max_iterations as number,\r\n );\r\n return;\r\n }\r\n\r\n if (msgType === \"reply_delta\" && typeof data.delta === \"string\") {\r\n accumulatedDelta += data.delta;\r\n events.onAssistantDelta?.(data.delta);\r\n return;\r\n }\r\n\r\n if (msgType === \"tool_calls\" && Array.isArray(data.calls)) {\r\n const calls = data.calls as AgentToolCallInfo[];\r\n\r\n // Speak accumulated text before running tools\r\n if (accumulatedDelta.trim()) {\r\n events.onIntermediateReply?.(accumulatedDelta.trim());\r\n }\r\n accumulatedDelta = \"\";\r\n\r\n events.onToolCalls?.(calls);\r\n\r\n const results: { call_id: string; result: string }[] = [];\r\n for (const call of calls) {\r\n // Save resume state before navigate in case of full-page reload\r\n const isNavigate = call.tool === \"navigate\";\r\n if (isNavigate) {\r\n savePendingAgentResume({\r\n sessionId: effectiveSessionId,\r\n projectId,\r\n model: config.model,\r\n voice: config.voice,\r\n accessibilityMode: Boolean(config.accessibilityMode),\r\n pendingToolCalls: calls.map((c) => ({\r\n call_id: c.call_id,\r\n tool: c.tool,\r\n args: c.args,\r\n })),\r\n completedResults: [...results],\r\n });\r\n }\r\n\r\n const result = await executeTool(call);\r\n\r\n // If we reach here, no full-page reload happened\r\n if (isNavigate) {\r\n clearPendingAgentResume();\r\n }\r\n\r\n events.onToolResult?.(call.call_id, call.tool, result.result);\r\n results.push(result);\r\n }\r\n\r\n if (socket.readyState === WebSocket.OPEN) {\r\n socket.send(JSON.stringify({\r\n type: \"tool_results\",\r\n results,\r\n }));\r\n }\r\n return;\r\n }\r\n\r\n if (msgType === \"agent_done\") {\r\n finalReply = (data.final_reply as string) || \"\";\r\n const replyExpectsReply = data.expects_reply !== false;\r\n events.onAssistantDone?.(finalReply, replyExpectsReply);\r\n if (typeof data.session_id === \"string\") {\r\n events.onSessionId?.(data.session_id as string);\r\n }\r\n finish(finalReply);\r\n return;\r\n }\r\n\r\n if (msgType === \"error\") {\r\n const errMsg = (data.error as string) || \"Agent error\";\r\n errorEmitted = true;\r\n events.onError?.(errMsg);\r\n fail(new Error(errMsg));\r\n return;\r\n }\r\n };\r\n\r\n socket.onerror = () => {\r\n console.error(\"[Bulut] Agent WS error\");\r\n errorEmitted = true;\r\n events.onError?.(\"Agent WebSocket connection error\");\r\n fail(new Error(\"Agent WebSocket connection error\"));\r\n };\r\n\r\n socket.onclose = () => {\r\n console.info(\"[Bulut] Agent WS closed\");\r\n finish(finalReply);\r\n };\r\n });\r\n\r\n activeSocket = null;\r\n\r\n // ── 3. TTS ────────────────────────────────────────────────\r\n if (isStopped || !assistantText) {\r\n if (!isStopped) {\r\n hideAgentCursor();\r\n }\r\n return resolve();\r\n }\r\n\r\n console.info(\r\n `[Bulut] TTS start mode=agent voice=${config.voice}`,\r\n );\r\n\r\n events.onAudioStateChange?.(\"rendering\");\r\n let ttsResult: TtsCollectResult;\r\n\r\n try {\r\n ttsResult = await collectTtsViaWebSocket(\r\n baseUrl,\r\n assistantText,\r\n config.voice,\r\n Boolean(config.accessibilityMode),\r\n () => isStopped,\r\n (socket) => { activeSocket = socket; },\r\n );\r\n } catch (wsError) {\r\n if (isStopped) return resolve();\r\n console.warn(\r\n `[Bulut] TTS WS failed, falling back to SSE: ${wsError instanceof Error ? wsError.message : String(wsError)}`,\r\n );\r\n ttsResult = await collectTtsViaSse(\r\n baseUrl,\r\n assistantText,\r\n config.voice,\r\n Boolean(config.accessibilityMode),\r\n () => isStopped,\r\n (reader) => { activeReader = reader; },\r\n );\r\n }\r\n\r\n if (!isStopped) {\r\n hideAgentCursor();\r\n }\r\n\r\n if (!isStopped && ttsResult.chunks.length > 0) {\r\n await playBufferedAudio(\r\n ttsResult.chunks,\r\n ttsResult.mimeType,\r\n ttsResult.sampleRate,\r\n events.onAudioStateChange,\r\n );\r\n } else {\r\n events.onAudioStateChange?.(\"done\");\r\n }\r\n\r\n resolve();\r\n } catch (err) {\r\n // Only emit onError if it hasn't been emitted already by the WS handler\r\n if (!errorEmitted) {\r\n const msg = err instanceof Error ? err.message : String(err);\r\n events.onError?.(msg);\r\n }\r\n reject(err);\r\n } finally {\r\n activeReader?.cancel().catch(() => { });\r\n if (activeSocket && activeSocket.readyState <= WebSocket.OPEN) {\r\n activeSocket.close();\r\n }\r\n activeSocket = null;\r\n }\r\n });\r\n\r\n return {\r\n stop: () => {\r\n isStopped = true;\r\n stopActiveAudioPlayback();\r\n if (activeReader) {\r\n activeReader.cancel().catch(() => { });\r\n }\r\n if (activeSocket && activeSocket.readyState <= WebSocket.OPEN) {\r\n activeSocket.close();\r\n }\r\n },\r\n done: donePromise,\r\n };\r\n};\r\n\r\n// ── Agent Text Chat Stream (no STT, Agent WS → TTS) ────────────────\r\n\r\nexport const agentTextChatStream = (\r\n baseUrl: string,\r\n userText: string,\r\n projectId: string,\r\n sessionId: string | null,\r\n config: {\r\n model: string;\r\n voice: string;\r\n pageContext?: string;\r\n accessibilityMode?: boolean;\r\n },\r\n events: AgentVoiceChatEvents,\r\n executeTool: (call: AgentToolCallInfo) => Promise<{ call_id: string; result: string }>,\r\n): StreamController => {\r\n let isStopped = false;\r\n let activeSocket: WebSocket | null = null;\r\n let activeReader: ReadableStreamDefaultReader<Uint8Array> | undefined;\r\n let errorEmitted = false;\r\n\r\n const donePromise = new Promise<void>(async (resolve, reject) => {\r\n try {\r\n if (isStopped) return resolve();\r\n\r\n // ── 1. Agent loop via WebSocket ───────────────────────────\r\n const assistantText = await new Promise<string>((agentResolve, agentReject) => {\r\n if (isStopped) { agentResolve(\"\"); return; }\r\n\r\n const wsUrl = toWebSocketUrl(baseUrl, \"/chat/agent/ws\");\r\n const socket = new WebSocket(wsUrl);\r\n activeSocket = socket;\r\n\r\n let finalReply = \"\";\r\n let resolved = false;\r\n let effectiveSessionId = sessionId || \"\";\r\n let accumulatedDelta = \"\";\r\n\r\n const finish = (reply: string) => {\r\n if (resolved) return;\r\n resolved = true;\r\n agentResolve(reply);\r\n };\r\n\r\n const fail = (error: Error) => {\r\n if (resolved) return;\r\n resolved = true;\r\n agentReject(error);\r\n };\r\n\r\n socket.onopen = () => {\r\n socket.send(JSON.stringify({\r\n type: \"start\",\r\n project_id: projectId,\r\n session_id: sessionId,\r\n user_text: userText,\r\n model: config.model,\r\n page_context: config.pageContext,\r\n accessibility_mode: config.accessibilityMode,\r\n }));\r\n };\r\n\r\n socket.onmessage = async (event) => {\r\n let data: Record<string, unknown>;\r\n try {\r\n data = JSON.parse(String(event.data));\r\n } catch { return; }\r\n\r\n const msgType = data.type as string;\r\n\r\n if (msgType === \"session\" && typeof data.session_id === \"string\") {\r\n effectiveSessionId = data.session_id as string;\r\n events.onSessionId?.(effectiveSessionId);\r\n return;\r\n }\r\n\r\n if (msgType === \"iteration\") {\r\n events.onIteration?.(\r\n data.iteration as number,\r\n data.max_iterations as number,\r\n );\r\n return;\r\n }\r\n\r\n if (msgType === \"reply_delta\" && typeof data.delta === \"string\") {\r\n accumulatedDelta += data.delta;\r\n events.onAssistantDelta?.(data.delta);\r\n return;\r\n }\r\n\r\n if (msgType === \"tool_calls\" && Array.isArray(data.calls)) {\r\n const calls = data.calls as AgentToolCallInfo[];\r\n\r\n // Speak accumulated text before running tools\r\n if (accumulatedDelta.trim()) {\r\n events.onIntermediateReply?.(accumulatedDelta.trim());\r\n }\r\n accumulatedDelta = \"\";\r\n\r\n events.onToolCalls?.(calls);\r\n\r\n const results: { call_id: string; result: string }[] = [];\r\n for (const call of calls) {\r\n const isNavigate = call.tool === \"navigate\";\r\n if (isNavigate) {\r\n savePendingAgentResume({\r\n sessionId: effectiveSessionId,\r\n projectId,\r\n model: config.model,\r\n voice: config.voice,\r\n accessibilityMode: Boolean(config.accessibilityMode),\r\n pendingToolCalls: calls.map((c) => ({\r\n call_id: c.call_id,\r\n tool: c.tool,\r\n args: c.args,\r\n })),\r\n completedResults: [...results],\r\n });\r\n }\r\n\r\n const result = await executeTool(call);\r\n\r\n if (isNavigate) {\r\n clearPendingAgentResume();\r\n }\r\n\r\n events.onToolResult?.(call.call_id, call.tool, result.result);\r\n results.push(result);\r\n }\r\n\r\n if (socket.readyState === WebSocket.OPEN) {\r\n socket.send(JSON.stringify({\r\n type: \"tool_results\",\r\n results,\r\n }));\r\n }\r\n return;\r\n }\r\n\r\n if (msgType === \"agent_done\") {\r\n finalReply = (data.final_reply as string) || \"\";\r\n const replyExpectsReply = data.expects_reply !== false;\r\n events.onAssistantDone?.(finalReply, replyExpectsReply);\r\n if (typeof data.session_id === \"string\") {\r\n events.onSessionId?.(data.session_id as string);\r\n }\r\n finish(finalReply);\r\n return;\r\n }\r\n\r\n if (msgType === \"error\") {\r\n const errMsg = (data.error as string) || \"Agent error\";\r\n errorEmitted = true;\r\n events.onError?.(errMsg);\r\n fail(new Error(errMsg));\r\n return;\r\n }\r\n };\r\n\r\n socket.onerror = () => {\r\n errorEmitted = true;\r\n events.onError?.(\"Agent WebSocket error\");\r\n fail(new Error(\"Agent WebSocket error\"));\r\n };\r\n socket.onclose = () => finish(finalReply);\r\n });\r\n\r\n activeSocket = null;\r\n\r\n // ── 2. TTS ────────────────────────────────────────────────\r\n if (isStopped || !assistantText) {\r\n if (!isStopped) {\r\n hideAgentCursor();\r\n }\r\n return resolve();\r\n }\r\n\r\n events.onAudioStateChange?.(\"rendering\");\r\n let ttsResult: TtsCollectResult;\r\n\r\n try {\r\n ttsResult = await collectTtsViaWebSocket(\r\n baseUrl, assistantText, config.voice, Boolean(config.accessibilityMode),\r\n () => isStopped,\r\n (socket) => { activeSocket = socket; },\r\n );\r\n } catch (wsError) {\r\n if (isStopped) return resolve();\r\n ttsResult = await collectTtsViaSse(\r\n baseUrl, assistantText, config.voice, Boolean(config.accessibilityMode),\r\n () => isStopped,\r\n (reader) => { activeReader = reader; },\r\n );\r\n }\r\n\r\n if (!isStopped) {\r\n hideAgentCursor();\r\n }\r\n\r\n if (!isStopped && ttsResult.chunks.length > 0) {\r\n await playBufferedAudio(\r\n ttsResult.chunks, ttsResult.mimeType, ttsResult.sampleRate,\r\n events.onAudioStateChange,\r\n );\r\n } else {\r\n events.onAudioStateChange?.(\"done\");\r\n }\r\n\r\n resolve();\r\n } catch (err) {\r\n if (!errorEmitted) {\r\n const msg = err instanceof Error ? err.message : String(err);\r\n events.onError?.(msg);\r\n }\r\n reject(err);\r\n } finally {\r\n activeReader?.cancel().catch(() => { });\r\n if (activeSocket && activeSocket.readyState <= WebSocket.OPEN) {\r\n activeSocket.close();\r\n }\r\n activeSocket = null;\r\n }\r\n });\r\n\r\n return {\r\n stop: () => {\r\n isStopped = true;\r\n stopActiveAudioPlayback();\r\n if (activeReader) activeReader.cancel().catch(() => { });\r\n if (activeSocket && activeSocket.readyState <= WebSocket.OPEN) {\r\n activeSocket.close();\r\n }\r\n },\r\n done: donePromise,\r\n };\r\n};\r\n\r\n// ── Agent Resume Stream (after page navigation reload) ──────────────\r\n//\r\n// When a navigate tool causes a full-page reload, the agent WS is lost.\r\n// This function opens a new WS with {type: \"resume\"}, sends the\r\n// completed tool results (including the navigate result with the new\r\n// page context), and continues the agent loop from where it left off.\r\n\r\nexport const agentResumeStream = (\r\n baseUrl: string,\r\n resumeState: PendingAgentResume,\r\n pageContext: string,\r\n events: AgentVoiceChatEvents,\r\n executeTool: (call: AgentToolCallInfo) => Promise<{ call_id: string; result: string }>,\r\n): StreamController => {\r\n let isStopped = false;\r\n let activeSocket: WebSocket | null = null;\r\n let activeReader: ReadableStreamDefaultReader<Uint8Array> | undefined;\r\n let errorEmitted = false;\r\n\r\n // Build tool results for the calls that were pending when the page reloaded.\r\n // Navigate results include the new page context; other tools that couldn't\r\n // execute get a descriptive skip message.\r\n const allResults = [...resumeState.completedResults];\r\n for (const tc of resumeState.pendingToolCalls) {\r\n if (allResults.some((r) => r.call_id === tc.call_id)) continue;\r\n if (tc.tool === \"navigate\") {\r\n allResults.push({\r\n call_id: tc.call_id,\r\n result: `Navigasyon tamamlandı. Şu anki sayfa: ${typeof window !== \"undefined\" ? window.location.href : \"\"}\\nSayfa bağlamı: ${pageContext}`,\r\n });\r\n } else {\r\n allResults.push({\r\n call_id: tc.call_id,\r\n result: \"Sayfa yeniden yüklendi, bu araç çalıştırılamadı.\",\r\n });\r\n }\r\n }\r\n\r\n const donePromise = new Promise<void>(async (resolve, reject) => {\r\n try {\r\n if (isStopped) return resolve();\r\n\r\n let effectiveSessionId = resumeState.sessionId;\r\n\r\n const assistantText = await new Promise<string>((agentResolve, agentReject) => {\r\n if (isStopped) { agentResolve(\"\"); return; }\r\n\r\n const wsUrl = toWebSocketUrl(baseUrl, \"/chat/agent/ws\");\r\n const socket = new WebSocket(wsUrl);\r\n activeSocket = socket;\r\n\r\n let finalReply = \"\";\r\n let resolved = false;\r\n let accumulatedDelta = \"\";\r\n\r\n const finish = (reply: string) => {\r\n if (resolved) return;\r\n resolved = true;\r\n agentResolve(reply);\r\n };\r\n\r\n const fail = (error: Error) => {\r\n if (resolved) return;\r\n resolved = true;\r\n agentReject(error);\r\n };\r\n\r\n socket.onopen = () => {\r\n console.info(\"[Bulut] Agent WS resume connected\");\r\n socket.send(JSON.stringify({\r\n type: \"resume\",\r\n project_id: resumeState.projectId,\r\n session_id: resumeState.sessionId,\r\n model: resumeState.model,\r\n page_context: pageContext,\r\n accessibility_mode: resumeState.accessibilityMode,\r\n pending_tool_calls: resumeState.pendingToolCalls,\r\n tool_results: allResults,\r\n }));\r\n };\r\n\r\n socket.onmessage = async (event) => {\r\n let data: Record<string, unknown>;\r\n try {\r\n data = JSON.parse(String(event.data));\r\n } catch { return; }\r\n\r\n const msgType = data.type as string;\r\n\r\n if (msgType === \"session\" && typeof data.session_id === \"string\") {\r\n effectiveSessionId = data.session_id as string;\r\n events.onSessionId?.(effectiveSessionId);\r\n return;\r\n }\r\n\r\n if (msgType === \"iteration\") {\r\n events.onIteration?.(\r\n data.iteration as number,\r\n data.max_iterations as number,\r\n );\r\n return;\r\n }\r\n\r\n if (msgType === \"reply_delta\" && typeof data.delta === \"string\") {\r\n accumulatedDelta += data.delta;\r\n events.onAssistantDelta?.(data.delta);\r\n return;\r\n }\r\n\r\n if (msgType === \"tool_calls\" && Array.isArray(data.calls)) {\r\n const calls = data.calls as AgentToolCallInfo[];\r\n\r\n // Speak accumulated text before running tools\r\n if (accumulatedDelta.trim()) {\r\n events.onIntermediateReply?.(accumulatedDelta.trim());\r\n }\r\n accumulatedDelta = \"\";\r\n\r\n events.onToolCalls?.(calls);\r\n\r\n const results: { call_id: string; result: string }[] = [];\r\n for (const call of calls) {\r\n const isNavigate = call.tool === \"navigate\";\r\n if (isNavigate) {\r\n savePendingAgentResume({\r\n sessionId: effectiveSessionId,\r\n projectId: resumeState.projectId,\r\n model: resumeState.model,\r\n voice: resumeState.voice,\r\n accessibilityMode: resumeState.accessibilityMode,\r\n pendingToolCalls: calls.map((c) => ({\r\n call_id: c.call_id,\r\n tool: c.tool,\r\n args: c.args,\r\n })),\r\n completedResults: [...results],\r\n });\r\n }\r\n\r\n const result = await executeTool(call);\r\n\r\n if (isNavigate) {\r\n clearPendingAgentResume();\r\n }\r\n\r\n events.onToolResult?.(call.call_id, call.tool, result.result);\r\n results.push(result);\r\n }\r\n\r\n if (socket.readyState === WebSocket.OPEN) {\r\n socket.send(JSON.stringify({ type: \"tool_results\", results }));\r\n }\r\n return;\r\n }\r\n\r\n if (msgType === \"agent_done\") {\r\n finalReply = (data.final_reply as string) || \"\";\r\n const replyExpectsReply = data.expects_reply !== false;\r\n events.onAssistantDone?.(finalReply, replyExpectsReply);\r\n if (typeof data.session_id === \"string\") {\r\n events.onSessionId?.(data.session_id as string);\r\n }\r\n finish(finalReply);\r\n return;\r\n }\r\n\r\n if (msgType === \"error\") {\r\n const errMsg = (data.error as string) || \"Agent error\";\r\n errorEmitted = true;\r\n events.onError?.(errMsg);\r\n fail(new Error(errMsg));\r\n return;\r\n }\r\n };\r\n\r\n socket.onerror = () => {\r\n errorEmitted = true;\r\n events.onError?.(\"Agent WebSocket error\");\r\n fail(new Error(\"Agent WebSocket error\"));\r\n };\r\n\r\n socket.onclose = () => finish(finalReply);\r\n });\r\n\r\n activeSocket = null;\r\n\r\n // ── TTS ────────────────────────────────────────────────\r\n if (isStopped || !assistantText) {\r\n if (!isStopped) {\r\n hideAgentCursor();\r\n }\r\n return resolve();\r\n }\r\n\r\n console.info(`[Bulut] TTS start mode=resume voice=${resumeState.voice}`);\r\n events.onAudioStateChange?.(\"rendering\");\r\n let ttsResult: TtsCollectResult;\r\n\r\n try {\r\n ttsResult = await collectTtsViaWebSocket(\r\n baseUrl, assistantText, resumeState.voice, Boolean(resumeState.accessibilityMode),\r\n () => isStopped,\r\n (socket) => { activeSocket = socket; },\r\n );\r\n } catch (wsError) {\r\n if (isStopped) return resolve();\r\n console.warn(\r\n `[Bulut] TTS WS failed, falling back to SSE: ${wsError instanceof Error ? wsError.message : String(wsError)}`,\r\n );\r\n ttsResult = await collectTtsViaSse(\r\n baseUrl, assistantText, resumeState.voice, Boolean(resumeState.accessibilityMode),\r\n () => isStopped,\r\n (reader) => { activeReader = reader; },\r\n );\r\n }\r\n\r\n if (!isStopped) {\r\n hideAgentCursor();\r\n }\r\n\r\n if (!isStopped && ttsResult.chunks.length > 0) {\r\n await playBufferedAudio(\r\n ttsResult.chunks, ttsResult.mimeType, ttsResult.sampleRate,\r\n events.onAudioStateChange,\r\n );\r\n } else {\r\n events.onAudioStateChange?.(\"done\");\r\n }\r\n\r\n resolve();\r\n } catch (err) {\r\n if (!errorEmitted) {\r\n const msg = err instanceof Error ? err.message : String(err);\r\n events.onError?.(msg);\r\n }\r\n reject(err);\r\n } finally {\r\n activeReader?.cancel().catch(() => { });\r\n if (activeSocket && activeSocket.readyState <= WebSocket.OPEN) {\r\n activeSocket.close();\r\n }\r\n activeSocket = null;\r\n }\r\n });\r\n\r\n return {\r\n stop: () => {\r\n isStopped = true;\r\n stopActiveAudioPlayback();\r\n if (activeReader) activeReader.cancel().catch(() => { });\r\n if (activeSocket && activeSocket.readyState <= WebSocket.OPEN) {\r\n activeSocket.close();\r\n }\r\n },\r\n done: donePromise,\r\n };\r\n};\r\n","import {\r\n completedSfxUrl,\r\n sentSfxUrl,\r\n thinkingSfxUrl,\r\n toolCallSfxUrl,\r\n} from \"../assets\";\r\n\r\nexport type SfxName = \"sent\" | \"thinking\" | \"toolCall\" | \"completed\";\r\n\r\nconst SFX_SOURCES: Record<SfxName, string> = {\r\n sent: sentSfxUrl,\r\n thinking: thinkingSfxUrl,\r\n toolCall: toolCallSfxUrl,\r\n completed: completedSfxUrl,\r\n};\r\n\r\nclass SfxManager {\r\n private queue: SfxName[] = [];\r\n private active = false;\r\n\r\n private playNow(name: SfxName): Promise<void> {\r\n return new Promise((resolve) => {\r\n if (typeof window === \"undefined\") {\r\n resolve();\r\n return;\r\n }\r\n\r\n const audio = new Audio(SFX_SOURCES[name]);\r\n audio.preload = \"auto\";\r\n\r\n let settled = false;\r\n const finalize = () => {\r\n if (settled) return;\r\n settled = true;\r\n audio.onended = null;\r\n audio.onerror = null;\r\n resolve();\r\n };\r\n\r\n audio.onended = finalize;\r\n audio.onerror = finalize;\r\n void audio.play().catch(() => finalize());\r\n });\r\n }\r\n\r\n private async drain(): Promise<void> {\r\n if (this.active) return;\r\n this.active = true;\r\n try {\r\n while (this.queue.length > 0) {\r\n const next = this.queue.shift();\r\n if (!next) continue;\r\n await this.playNow(next);\r\n }\r\n } finally {\r\n this.active = false;\r\n }\r\n }\r\n\r\n playCue(name: SfxName): void {\r\n if (typeof window === \"undefined\") return;\r\n this.queue.push(name);\r\n if (!this.active) {\r\n void this.drain();\r\n }\r\n }\r\n}\r\n\r\nconst sfxManager = new SfxManager();\r\n\r\nexport const playCue = (name: SfxName): void => {\r\n sfxManager.playCue(name);\r\n};\r\n\r\n","export default \"data:audio/mpeg;base64,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\"","export default \"data:audio/mpeg;base64,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VVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVQ==\"","export default \"data:audio/mpeg;base64,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qqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqgU7bdtbNa5MIRBHgnj4UR9JwIgVBwG4sGQ6iUOI9A8D4gD+UB2EkchSQR4JZYNzUyKpeTJzhe5G6uhMjIwPOFopGnGihQsgHRJAooSJEExgcJJAooD/+rJs9RAqj/URY7JpgTYKAAANIAAAAQAAAaQAAAAgAAA0gAAABIWQTUkROElmHxaqlm7Oz/9nzZqWcuLUkicacUfCakiJxZZlwtGjSini42af/s7P3dtyppyij0FopETgMDMJoLRJGlFHmWpJE4sukDVJVUxBTUUzLjk5LjVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVU=\"","export default \"data:audio/mpeg;base64,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BpGoUioUhyHQcjkOIhCCIQkg1EYApIxw2dKsgSFBRY7KF9qKJlrzBofwHJWHEHg1GskL2IUxVHoaRPKh2w3AlLxBIIwjQThaxUZGCNA3myVKnEbD2UScIrTc0hRFSim+MkJEMh44vC5KlhkaDxdSdJrFSxxeE4XSqyy8Jw33Jf/6smxwIbyA+HBovGsrZIgAAA0gAAABFwmixmwxKYgAADSAAAAEk0Dbmma1Emo9lEQli6+eTKIhEQfOqTqKypYZOqL1dKlhkZKLwnSaxUZKI0E4WsVKnEbD3SvFUpueyiVUhvySEiKnDi/9pov9VVNvpoqVTEFNRTMuOTkuNVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVf/6smw7EwAP8AAAaQAAAAgAAA0gAAABAAABpAAAACAAADSAAAAETEFNRTMuOTkuNVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVTEFNRTMuOTkuNVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVQ==\"","import { useEffect, useLayoutEffect, useRef, useState } from \"preact/hooks\";\r\nimport type { JSX } from \"preact\";\r\nimport type { BulutRuntimeConfig } from \"../index\";\r\nimport {\r\n agentTextChatStream,\r\n agentVoiceChatStream,\r\n agentResumeStream,\r\n startSttWebSocketStream,\r\n stopActiveAudioPlayback,\r\n speakText,\r\n type AudioStreamState,\r\n type StreamController,\r\n type AgentToolCallInfo,\r\n type SttWsController,\r\n} from \"../api/client\";\r\nimport {\r\n executeSingleToolCall,\r\n parseAgentResponse,\r\n getPendingAgentResume,\r\n clearPendingAgentResume,\r\n type ToolCallWithId,\r\n} from \"../agent/tools\";\r\nimport { getPageContext } from \"../agent/context\";\r\nimport {\r\n WINDOW_WIDTH,\r\n WINDOW_HEIGHT,\r\n POSITION_BOTTOM,\r\n POSITION_RIGHT,\r\n COLORS,\r\n TRANSITIONS,\r\n BORDER_RADIUS,\r\n SHADOW,\r\n} from \"../styles/constants\";\r\nimport {\r\n logoContent,\r\n arrowPathIconContent,\r\n commandLineIconContent,\r\n cursorArrowRaysIconContent,\r\n faceSmileIconContent,\r\n handRaisedIconContent,\r\n mapIconContent,\r\n microphoneOutlineIconContent,\r\n queueListIconContent,\r\n stopOutlineIconContent,\r\n xMarkIconContent,\r\n} from \"../assets\";\r\nimport { StreamingJsonParser } from \"../utils/streamingJson\";\r\nimport { playCue, type SfxName } from \"../audio/sfxManager\";\r\nimport { SvgIcon } from \"./SvgIcon\";\r\n\r\nexport interface ChatWindowHandle {\r\n startRecording: () => void;\r\n cancelRecording: () => void;\r\n stopTask: () => void;\r\n}\r\n\r\ninterface ChatWindowProps {\r\n onClose: () => void;\r\n config: BulutRuntimeConfig;\r\n accessibilityMode?: boolean;\r\n onAccessibilityToggle?: () => void;\r\n hidden?: boolean;\r\n actionsRef?: { current: ChatWindowHandle | null };\r\n onRecordingChange?: (recording: boolean) => void;\r\n onBusyChange?: (busy: boolean) => void;\r\n onPreviewChange?: (text: string | null) => void;\r\n}\r\n\r\ninterface Message {\r\n id: number;\r\n text: string;\r\n isUser: boolean;\r\n /** \"message\" (default) | \"tool\" for tool call indicators */\r\n type?: \"message\" | \"tool\";\r\n toolKind?: \"context\" | \"cursor\" | \"scroll\" | \"navigate\" | \"form\" | \"interact\" | \"unknown\";\r\n toolLabel?: string;\r\n toolCount?: number;\r\n}\r\n\r\ntype RecordingMode = \"vad\" | \"press\";\r\n\r\ntype StorageLike = {\r\n removeItem: (key: string) => void;\r\n};\r\n\r\nconst STORAGE_KEY = \"bulut_chat_history\";\r\nconst TIMESTAMP_KEY = \"bulut_chat_timestamp\";\r\nconst SESSION_ID_KEY = \"bulut_session_id\";\r\nconst TTL_MS = 5 * 60 * 1000;\r\nconst VAD_THRESHOLD = 0.06;\r\nconst SILENCE_DURATION_MS = 500;\r\nconst ACCESSIBILITY_MIN_SPEECH_DURATION_MS = 1500;\r\nexport const HOLD_THRESHOLD_MS = 250;\r\n\r\nconst STATUS_LABELS = {\r\n ready: \"Hazır\",\r\n loading: \"Bir saniye\",\r\n micInitializing: \"Mikrofonu hazırlıyorum\",\r\n listening: \"Sizi dinliyorum\",\r\n accessibilityActive: \"Erişilebilirlik Aktif\",\r\n transcribing: \"Düşünüyorum\",\r\n thinking: \"Düşünüyorum\",\r\n playingAudio: \".\",\r\n runningTools: \"Siteyle ilgileniyorum\",\r\n} as const;\r\n\r\nexport const getGreetingText = (agentName: string): string =>\r\n `Merhaba, ben ${agentName}. Bu web sayfasında neler yapalım?`;\r\n\r\nexport interface StatusFlags {\r\n isBusy: boolean;\r\n isRecording: boolean;\r\n isTranscribing: boolean;\r\n isThinking: boolean;\r\n isRenderingAudio: boolean;\r\n isPlayingAudio: boolean;\r\n isRunningTools: boolean;\r\n}\r\n\r\nexport const resolveStatusText = (flags: StatusFlags): string => {\r\n if (flags.isRecording) return STATUS_LABELS.listening;\r\n if (flags.isRunningTools) return STATUS_LABELS.runningTools;\r\n if (flags.isPlayingAudio) return STATUS_LABELS.playingAudio;\r\n if (flags.isThinking) return STATUS_LABELS.thinking;\r\n if (flags.isTranscribing) return STATUS_LABELS.transcribing;\r\n if (flags.isBusy) return STATUS_LABELS.loading;\r\n return STATUS_LABELS.ready;\r\n};\r\n\r\nexport const hasActiveStatus = (\r\n flags: StatusFlags,\r\n statusOverride: string | null,\r\n): boolean =>\r\n Boolean(\r\n statusOverride\r\n || flags.isBusy\r\n || flags.isRecording\r\n || flags.isTranscribing\r\n || flags.isThinking\r\n || flags.isRenderingAudio\r\n || flags.isPlayingAudio\r\n || flags.isRunningTools,\r\n );\r\n\r\nexport const formatDurationMs = (durationMs: number): string => {\r\n const totalSeconds = Math.max(0, Math.floor(durationMs / 1000));\r\n const minutes = Math.floor(totalSeconds / 60)\r\n .toString()\r\n .padStart(2, \"0\");\r\n const seconds = (totalSeconds % 60).toString().padStart(2, \"0\");\r\n return `${minutes}:${seconds}`;\r\n};\r\n\r\nexport const classifyMicGesture = (\r\n durationMs: number,\r\n thresholdMs: number = HOLD_THRESHOLD_MS,\r\n): \"tap\" | \"hold\" => (durationMs >= thresholdMs ? \"hold\" : \"tap\");\r\n\r\nexport const createInitialMessages = (agentName: string): Message[] => [\r\n {\r\n id: 1,\r\n text: getGreetingText(agentName),\r\n isUser: false,\r\n },\r\n];\r\n\r\nexport const clearPersistedChatState = (storage: StorageLike | null): void => {\r\n if (!storage) {\r\n return;\r\n }\r\n\r\n storage.removeItem(STORAGE_KEY);\r\n storage.removeItem(TIMESTAMP_KEY);\r\n storage.removeItem(SESSION_ID_KEY);\r\n};\r\n\r\nexport const scrollElementToBottom = (\r\n element: { scrollTop: number; scrollHeight: number } | null,\r\n): void => {\r\n if (!element) {\r\n return;\r\n }\r\n\r\n element.scrollTop = element.scrollHeight;\r\n};\r\n\r\nconst normalizeError = (error: unknown) => {\r\n if (error instanceof Error) {\r\n return error.message;\r\n }\r\n return \"Bilinmeyen hata\";\r\n};\r\n\r\nconst getNextMessageId = (messages: Message[]): number => {\r\n const maxId = messages.reduce((acc, message) => Math.max(acc, message.id), 0);\r\n return maxId + 1;\r\n};\r\n\r\nexport interface AssistantPayloadResolution {\r\n displayText: string;\r\n toolCalls: ReturnType<typeof parseAgentResponse>[\"toolCalls\"];\r\n}\r\n\r\nexport const resolveAssistantPayload = (\r\n assistantText: string,\r\n): AssistantPayloadResolution => {\r\n const parsed = parseAgentResponse(assistantText);\r\n return {\r\n displayText: parsed.reply || assistantText,\r\n toolCalls: parsed.toolCalls,\r\n };\r\n};\r\n\r\nexport const shouldAutoListenAfterAudio = (\r\n accessibilityMode: boolean,\r\n expectsReply: boolean,\r\n isRecording: boolean,\r\n isBusy: boolean,\r\n): boolean => (accessibilityMode || expectsReply) && !isRecording && !isBusy;\r\n\r\nexport const shouldAcceptVadSpeech = (\r\n speechDurationMs: number,\r\n enforceMinSpeechDuration: boolean,\r\n minSpeechDurationMs: number = ACCESSIBILITY_MIN_SPEECH_DURATION_MS,\r\n): boolean => !enforceMinSpeechDuration || speechDurationMs >= minSpeechDurationMs;\r\n\r\ninterface ToolIndicatorMessage {\r\n text: string;\r\n kind: \"context\" | \"cursor\" | \"scroll\" | \"navigate\" | \"form\" | \"interact\" | \"unknown\";\r\n}\r\n\r\nconst getToolIndicatorMessage = (\r\n call: AgentToolCallInfo,\r\n): ToolIndicatorMessage => {\r\n if (call.tool === \"getPageContext\") {\r\n return { text: \"Algılama\", kind: \"context\" };\r\n }\r\n if (call.tool === \"scroll\") {\r\n return { text: \"Kaydırma\", kind: \"scroll\" };\r\n }\r\n if (call.tool === \"navigate\") {\r\n const url = typeof call.args.url === \"string\" ? call.args.url.trim() : \"\";\r\n return {\r\n text: url ? `Sayfa Geçişi: ${url}` : \"Sayfa Geçişi\",\r\n kind: \"navigate\",\r\n };\r\n }\r\n if (call.tool === \"interact\" && call.args.action === \"move\") {\r\n return { text: \"Serbest İmleç\", kind: \"cursor\" };\r\n }\r\n if (call.tool === \"interact\" && call.args.action === \"type\") {\r\n return { text: \"Form Doldurma\", kind: \"form\" };\r\n }\r\n if (call.tool === \"interact\" && call.args.action === \"submit\") {\r\n return { text: \"Form Gönderme\", kind: \"form\" };\r\n }\r\n if (call.tool === \"interact\" && call.args.action === \"click\") {\r\n return { text: \"Tıklama\", kind: \"interact\" };\r\n }\r\n if (call.tool === \"interact\") {\r\n return { text: \"Etkileşim\", kind: \"interact\" };\r\n }\r\n return {\r\n text: call.tool || \"Araç\",\r\n kind: \"unknown\",\r\n };\r\n};\r\n\r\nexport const ChatWindow = ({\r\n onClose,\r\n config,\r\n accessibilityMode = false,\r\n onAccessibilityToggle,\r\n hidden = false,\r\n actionsRef,\r\n onRecordingChange,\r\n onBusyChange,\r\n onPreviewChange,\r\n}: ChatWindowProps) => {\r\n const [messages, setMessages] = useState<Message[]>(() => {\r\n if (typeof localStorage !== \"undefined\") {\r\n const saved = localStorage.getItem(STORAGE_KEY);\r\n const timestamp = localStorage.getItem(TIMESTAMP_KEY);\r\n\r\n if (saved && timestamp) {\r\n const timePassed = Date.now() - parseInt(timestamp, 10);\r\n if (timePassed < TTL_MS) {\r\n try {\r\n return JSON.parse(saved) as Message[];\r\n } catch {\r\n // Ignore parse error and continue with default.\r\n }\r\n } else {\r\n clearPersistedChatState(localStorage);\r\n }\r\n }\r\n }\r\n\r\n return createInitialMessages(config.agentName);\r\n });\r\n\r\n const [isBusy, setIsBusy] = useState(false);\r\n const [isRecording, setIsRecording] = useState(false);\r\n const [isTranscribing, setIsTranscribing] = useState(false);\r\n const [isThinking, setIsThinking] = useState(false);\r\n const [isRenderingAudio, setIsRenderingAudio] = useState(false);\r\n const [isPlayingAudio, setIsPlayingAudio] = useState(false);\r\n const [isRunningTools, setIsRunningTools] = useState(false);\r\n const [isMicPending, setIsMicPending] = useState(false);\r\n const [recordingDurationMs, setRecordingDurationMs] = useState(0);\r\n const [statusOverride, setStatusOverride] = useState<string | null>(null);\r\n const statusFlags: StatusFlags = {\r\n isBusy,\r\n isRecording,\r\n isTranscribing,\r\n isThinking,\r\n isRenderingAudio,\r\n isPlayingAudio,\r\n isRunningTools,\r\n };\r\n const resolvedStatusText = resolveStatusText(statusFlags);\r\n const showStatus = hasActiveStatus(statusFlags, statusOverride);\r\n const statusText = showStatus ? (statusOverride ?? resolvedStatusText) : STATUS_LABELS.ready;\r\n\r\n const isBusyRef = useRef(isBusy);\r\n const isRecordingRef = useRef(isRecording);\r\n\r\n const nextMessageIdRef = useRef(getNextMessageId(messages));\r\n const recorderRef = useRef<MediaRecorder | null>(null);\r\n const streamRef = useRef<MediaStream | null>(null);\r\n const audioChunksRef = useRef<BlobPart[]>([]);\r\n const activeStreamControllerRef = useRef<StreamController | null>(null);\r\n const sessionIdRef = useRef<string | null>(\r\n typeof localStorage !== \"undefined\"\r\n ? (() => {\r\n const ts = localStorage.getItem(TIMESTAMP_KEY);\r\n if (ts && Date.now() - parseInt(ts, 10) < TTL_MS) {\r\n return localStorage.getItem(SESSION_ID_KEY);\r\n }\r\n return null;\r\n })()\r\n : null,\r\n );\r\n\r\n const silenceStartRef = useRef<number | null>(null);\r\n const vadIntervalRef = useRef<number | null>(null);\r\n const audioContextRef = useRef<AudioContext | null>(null);\r\n const sourceRef = useRef<MediaStreamAudioSourceNode | null>(null);\r\n\r\n const discardNextRecordingRef = useRef(false);\r\n\r\n const micPressStartRef = useRef<number | null>(null);\r\n const micHoldTimeoutRef = useRef<number | null>(null);\r\n const micHoldTriggeredRef = useRef(false);\r\n const recordingModeRef = useRef<RecordingMode | null>(null);\r\n const pendingStopAfterStartRef = useRef(false);\r\n const startRecordingPendingRef = useRef(false);\r\n\r\n const assistantMessageIdRef = useRef<number | null>(null);\r\n const assistantTextBufferRef = useRef(\"\");\r\n const transcriptionReceivedRef = useRef(false);\r\n const assistantDoneReceivedRef = useRef(false);\r\n\r\n const recordingStartedAtRef = useRef<number | null>(null);\r\n const recordingTimerIntervalRef = useRef<number | null>(null);\r\n\r\n const messagesContainerRef = useRef<HTMLDivElement | null>(null);\r\n const messagesContentRef = useRef<HTMLDivElement | null>(null);\r\n\r\n const pendingUserTextRef = useRef<string | null>(null);\r\n const pendingAssistantTextRef = useRef<string>(\"\");\r\n const streamingJsonParserRef = useRef<StreamingJsonParser | null>(null);\r\n const awaitingAssistantResponseRef = useRef(false);\r\n const activeSttWsRef = useRef<SttWsController | null>(null);\r\n const liveTranscriptionMessageIdRef = useRef<number | null>(null);\r\n const liveTranscriptionTextRef = useRef(\"\");\r\n const autoListenSuppressedRef = useRef(false);\r\n const expectsReplyRef = useRef(true);\r\n const requestEpochRef = useRef(0);\r\n const sttSendCuePlayedRef = useRef(false);\r\n\r\n useEffect(() => {\r\n isBusyRef.current = isBusy;\r\n }, [isBusy]);\r\n\r\n useEffect(() => {\r\n isRecordingRef.current = isRecording;\r\n }, [isRecording]);\r\n\r\n // Report state changes to parent\r\n useEffect(() => { onRecordingChange?.(isRecording); }, [isRecording]);\r\n useEffect(() => { onBusyChange?.(isBusy); }, [isBusy]);\r\n\r\n // Derive and report preview text to parent\r\n useEffect(() => {\r\n if (!onPreviewChange) return;\r\n if (isRecording) {\r\n onPreviewChange(statusOverride ?? STATUS_LABELS.listening);\r\n return;\r\n }\r\n // When audio is rendering/playing, show the actual message text\r\n if (isRenderingAudio || isPlayingAudio) {\r\n const lastAssistant = [...messages].reverse().find(m => !m.isUser && m.type !== \"tool\");\r\n onPreviewChange(lastAssistant?.text ?? getGreetingText(config.agentName));\r\n return;\r\n }\r\n if (showStatus) {\r\n const st = statusOverride ?? resolveStatusText({\r\n isBusy,\r\n isRecording,\r\n isTranscribing,\r\n isThinking,\r\n isRenderingAudio,\r\n isPlayingAudio,\r\n isRunningTools,\r\n });\r\n onPreviewChange(st);\r\n return;\r\n }\r\n // Show last assistant message (or greeting)\r\n const lastAssistant = [...messages].reverse().find(m => !m.isUser && m.type !== \"tool\");\r\n onPreviewChange(lastAssistant?.text ?? getGreetingText(config.agentName));\r\n }, [\r\n isRecording,\r\n isBusy,\r\n isTranscribing,\r\n isThinking,\r\n isRunningTools,\r\n isPlayingAudio,\r\n isRenderingAudio,\r\n statusOverride,\r\n showStatus,\r\n messages,\r\n ]);\r\n\r\n const playSfx = (name: SfxName) => {\r\n playCue(name);\r\n };\r\n\r\n const beginRequestEpoch = () => {\r\n requestEpochRef.current += 1;\r\n return requestEpochRef.current;\r\n };\r\n\r\n const invalidateRequestEpoch = () => {\r\n requestEpochRef.current += 1;\r\n };\r\n\r\n const isCurrentRequestEpoch = (epoch: number): boolean =>\r\n requestEpochRef.current === epoch;\r\n\r\n const playSttSentCueOnce = () => {\r\n if (sttSendCuePlayedRef.current) {\r\n return;\r\n }\r\n sttSendCuePlayedRef.current = true;\r\n playSfx(\"sent\");\r\n };\r\n\r\n useEffect(() => {\r\n if (typeof localStorage !== \"undefined\") {\r\n localStorage.setItem(STORAGE_KEY, JSON.stringify(messages));\r\n localStorage.setItem(TIMESTAMP_KEY, Date.now().toString());\r\n }\r\n }, [messages]);\r\n\r\n const scrollMessagesToBottom = () => {\r\n scrollElementToBottom(messagesContainerRef.current);\r\n };\r\n\r\n useLayoutEffect(() => {\r\n scrollMessagesToBottom();\r\n }, [messages, statusText, isBusy, isRecording]);\r\n\r\n useEffect(() => {\r\n const content = messagesContentRef.current;\r\n if (!content || typeof ResizeObserver === \"undefined\") {\r\n return;\r\n }\r\n\r\n const observer = new ResizeObserver(() => {\r\n scrollMessagesToBottom();\r\n });\r\n\r\n observer.observe(content);\r\n return () => observer.disconnect();\r\n }, []);\r\n\r\n const stopRecordingTimer = () => {\r\n if (recordingTimerIntervalRef.current !== null) {\r\n window.clearInterval(recordingTimerIntervalRef.current);\r\n recordingTimerIntervalRef.current = null;\r\n }\r\n recordingStartedAtRef.current = null;\r\n };\r\n\r\n const startRecordingTimer = () => {\r\n stopRecordingTimer();\r\n recordingStartedAtRef.current = Date.now();\r\n setRecordingDurationMs(0);\r\n\r\n recordingTimerIntervalRef.current = window.setInterval(() => {\r\n const startedAt = recordingStartedAtRef.current;\r\n if (startedAt === null) {\r\n setRecordingDurationMs(0);\r\n return;\r\n }\r\n setRecordingDurationMs(Date.now() - startedAt);\r\n }, 200);\r\n };\r\n\r\n const resetProcessingFlags = () => {\r\n setIsTranscribing(false);\r\n setIsThinking(false);\r\n setIsRenderingAudio(false);\r\n setIsPlayingAudio(false);\r\n setIsRunningTools(false);\r\n setStatusOverride(null);\r\n assistantMessageIdRef.current = null;\r\n assistantTextBufferRef.current = \"\";\r\n transcriptionReceivedRef.current = false;\r\n assistantDoneReceivedRef.current = false;\r\n awaitingAssistantResponseRef.current = false;\r\n pendingUserTextRef.current = null;\r\n pendingAssistantTextRef.current = \"\";\r\n };\r\n\r\n const clearMicHoldTimeout = () => {\r\n if (micHoldTimeoutRef.current !== null) {\r\n window.clearTimeout(micHoldTimeoutRef.current);\r\n micHoldTimeoutRef.current = null;\r\n }\r\n };\r\n\r\n const cleanupVAD = () => {\r\n if (vadIntervalRef.current !== null) {\r\n clearInterval(vadIntervalRef.current);\r\n vadIntervalRef.current = null;\r\n }\r\n\r\n if (sourceRef.current) {\r\n sourceRef.current.disconnect();\r\n sourceRef.current = null;\r\n }\r\n\r\n if (audioContextRef.current) {\r\n audioContextRef.current.close().catch(() => {});\r\n audioContextRef.current = null;\r\n }\r\n\r\n silenceStartRef.current = null;\r\n };\r\n\r\n const stopStreamTracks = () => {\r\n if (!streamRef.current) {\r\n return;\r\n }\r\n\r\n streamRef.current.getTracks().forEach((track) => track.stop());\r\n streamRef.current = null;\r\n };\r\n\r\n const stopActiveStream = () => {\r\n if (!activeStreamControllerRef.current) {\r\n return;\r\n }\r\n\r\n activeStreamControllerRef.current.stop();\r\n activeStreamControllerRef.current = null;\r\n };\r\n\r\n const cancelActiveSttWs = () => {\r\n const activeSttWs = activeSttWsRef.current;\r\n activeSttWsRef.current = null;\r\n activeSttWs?.cancel();\r\n liveTranscriptionMessageIdRef.current = null;\r\n liveTranscriptionTextRef.current = \"\";\r\n };\r\n\r\n useEffect(\r\n () => () => {\r\n invalidateRequestEpoch();\r\n clearMicHoldTimeout();\r\n pendingStopAfterStartRef.current = false;\r\n\r\n stopActiveStream();\r\n stopActiveAudioPlayback();\r\n cancelActiveSttWs();\r\n cleanupVAD();\r\n stopStreamTracks();\r\n stopRecordingTimer();\r\n\r\n const recorder = recorderRef.current;\r\n if (recorder) {\r\n recorder.ondataavailable = null;\r\n recorder.onstop = null;\r\n recorder.onerror = null;\r\n if (recorder.state !== \"inactive\") {\r\n recorder.stop();\r\n }\r\n recorderRef.current = null;\r\n }\r\n\r\n cancelActiveSttWs();\r\n },\r\n [],\r\n );\r\n\r\n // ── Resume agent loop after full-page navigation ────────────────\r\n useEffect(() => {\r\n const resumeState = getPendingAgentResume();\r\n if (!resumeState) return;\r\n\r\n clearPendingAgentResume();\r\n console.info(\"[Bulut] Resuming agent after navigation\");\r\n\r\n // Restore session ID from resume state\r\n if (resumeState.sessionId) {\r\n sessionIdRef.current = resumeState.sessionId;\r\n if (typeof localStorage !== \"undefined\") {\r\n localStorage.setItem(SESSION_ID_KEY, resumeState.sessionId);\r\n }\r\n }\r\n\r\n const requestEpoch = beginRequestEpoch();\r\n setIsBusy(true);\r\n isBusyRef.current = true;\r\n setIsRunningTools(true);\r\n setStatusOverride(STATUS_LABELS.thinking);\r\n\r\n const freshPageContext = getPageContext().summary;\r\n\r\n const resumeToolExec = async (\r\n call: AgentToolCallInfo,\r\n ): Promise<{ call_id: string; result: string }> => {\r\n const toolCall: ToolCallWithId = {\r\n tool: call.tool as \"navigate\" | \"getPageContext\" | \"interact\" | \"scroll\",\r\n call_id: call.call_id,\r\n ...call.args,\r\n } as ToolCallWithId;\r\n return executeSingleToolCall(toolCall);\r\n };\r\n\r\n const controller = agentResumeStream(\r\n config.backendBaseUrl,\r\n resumeState,\r\n freshPageContext,\r\n {\r\n onSessionId: (sid) => {\r\n if (!isCurrentRequestEpoch(requestEpoch)) return;\r\n if (sid && sid !== sessionIdRef.current) {\r\n sessionIdRef.current = sid;\r\n if (typeof localStorage !== \"undefined\") {\r\n localStorage.setItem(SESSION_ID_KEY, sid);\r\n }\r\n }\r\n },\r\n onAssistantDelta: (delta) => {\r\n if (!isCurrentRequestEpoch(requestEpoch)) return;\r\n setIsRunningTools(false);\r\n setIsThinking(true);\r\n setStatusOverride(null);\r\n\r\n pendingAssistantTextRef.current += delta;\r\n\r\n if (assistantMessageIdRef.current === null) {\r\n assistantMessageIdRef.current = appendMessage(\r\n pendingAssistantTextRef.current,\r\n false,\r\n );\r\n } else {\r\n updateMessageText(\r\n assistantMessageIdRef.current,\r\n pendingAssistantTextRef.current,\r\n );\r\n }\r\n },\r\n onAssistantDone: (assistantText, expectsReply) => {\r\n if (!isCurrentRequestEpoch(requestEpoch)) return;\r\n playSfx(\"completed\");\r\n expectsReplyRef.current = expectsReply !== false;\r\n setStatusOverride(null);\r\n setIsThinking(false);\r\n setIsRenderingAudio(true);\r\n\r\n const finalDisplayText =\r\n assistantText || pendingAssistantTextRef.current;\r\n pendingAssistantTextRef.current = finalDisplayText;\r\n\r\n if (assistantMessageIdRef.current !== null) {\r\n updateMessageText(\r\n assistantMessageIdRef.current,\r\n finalDisplayText,\r\n );\r\n } else {\r\n assistantMessageIdRef.current = appendMessage(\r\n finalDisplayText,\r\n false,\r\n );\r\n }\r\n },\r\n onIntermediateReply: (text) => {\r\n if (!isCurrentRequestEpoch(requestEpoch)) return;\r\n void speakText(\r\n config.backendBaseUrl, text, config.voice,\r\n accessibilityMode, (state) => handleAudioStateChange(state, requestEpoch),\r\n ).catch((err) => console.warn(\"[Bulut] intermediate TTS failed\", err));\r\n },\r\n onToolCalls: (calls) => {\r\n if (!isCurrentRequestEpoch(requestEpoch)) return;\r\n if (calls.length > 0) {\r\n playSfx(\"toolCall\");\r\n }\r\n setIsRunningTools(true);\r\n setStatusOverride(STATUS_LABELS.runningTools);\r\n appendToolIndicatorMessages(calls);\r\n\r\n assistantMessageIdRef.current = null;\r\n pendingAssistantTextRef.current = \"\";\r\n },\r\n onToolResult: () => {},\r\n onIteration: () => {\r\n if (!isCurrentRequestEpoch(requestEpoch)) return;\r\n playSfx(\"thinking\");\r\n setIsThinking(true);\r\n setStatusOverride(STATUS_LABELS.thinking);\r\n },\r\n onAudioStateChange: (state) => {\r\n handleAudioStateChange(state, requestEpoch);\r\n },\r\n onError: (err) => {\r\n if (!isCurrentRequestEpoch(requestEpoch)) return;\r\n setStatusOverride(null);\r\n appendMessage(`Hata: ${err}`, false);\r\n },\r\n },\r\n resumeToolExec,\r\n );\r\n\r\n activeStreamControllerRef.current = controller;\r\n\r\n controller.done\r\n .catch(() => {})\r\n .finally(() => {\r\n if (!isCurrentRequestEpoch(requestEpoch)) return;\r\n setIsBusy(false);\r\n isBusyRef.current = false;\r\n setIsRunningTools(false);\r\n setIsThinking(false);\r\n setIsRenderingAudio(false);\r\n setIsPlayingAudio(false);\r\n setStatusOverride(null);\r\n pendingAssistantTextRef.current = \"\";\r\n assistantMessageIdRef.current = null;\r\n activeStreamControllerRef.current = null;\r\n\r\n if (\r\n !autoListenSuppressedRef.current &&\r\n shouldAutoListenAfterAudio(\r\n accessibilityMode,\r\n expectsReplyRef.current,\r\n isRecordingRef.current,\r\n false,\r\n )\r\n ) {\r\n void startRecording(\"vad\");\r\n }\r\n // Reset for next turn\r\n expectsReplyRef.current = true;\r\n });\r\n }, []); // eslint-disable-line react-hooks/exhaustive-deps\r\n\r\n const appendMessage = (\r\n text: string,\r\n isUser: boolean,\r\n options?: {\r\n type?: \"message\" | \"tool\";\r\n toolKind?: Message[\"toolKind\"];\r\n toolLabel?: string;\r\n toolCount?: number;\r\n },\r\n ): number => {\r\n const id = nextMessageIdRef.current++;\r\n setMessages((previous) => [\r\n ...previous,\r\n {\r\n id,\r\n text,\r\n isUser,\r\n type: options?.type,\r\n toolKind: options?.toolKind,\r\n toolLabel: options?.toolLabel,\r\n toolCount: options?.toolCount,\r\n },\r\n ]);\r\n return id;\r\n };\r\n\r\n const appendToolIndicatorMessages = (calls: AgentToolCallInfo[]) => {\r\n setMessages((previous) => {\r\n const next = [...previous];\r\n\r\n for (const call of calls) {\r\n const indicator = getToolIndicatorMessage(call);\r\n const last = next[next.length - 1];\r\n const previousToolText = typeof last?.text === \"string\"\r\n ? last.text.replace(/\\s+\\(\\d+\\)$/, \"\")\r\n : \"\";\r\n\r\n if (\r\n last\r\n && !last.isUser\r\n && last.type === \"tool\"\r\n && previousToolText === indicator.text\r\n ) {\r\n const extractedCount = Number.parseInt(\r\n (last.text.match(/\\((\\d+)\\)\\s*$/)?.[1] ?? \"1\"),\r\n 10,\r\n );\r\n const safeCurrentCount = Number.isFinite(extractedCount) ? extractedCount : 1;\r\n const nextCount = safeCurrentCount + 1;\r\n const baseLabel = previousToolText || indicator.text;\r\n next[next.length - 1] = {\r\n ...last,\r\n toolLabel: baseLabel,\r\n toolCount: nextCount,\r\n text: `${baseLabel} (${nextCount})`,\r\n };\r\n continue;\r\n }\r\n\r\n const id = nextMessageIdRef.current++;\r\n next.push({\r\n id,\r\n text: indicator.text,\r\n isUser: false,\r\n type: \"tool\",\r\n toolKind: indicator.kind,\r\n toolLabel: indicator.text,\r\n toolCount: 1,\r\n });\r\n }\r\n\r\n return next;\r\n });\r\n };\r\n\r\n const updateMessageText = (id: number, text: string) => {\r\n setMessages((previous) =>\r\n previous.map((message) =>\r\n message.id === id ? { ...message, text } : message,\r\n ),\r\n );\r\n };\r\n\r\n const upsertLiveUserTranscription = (text: string) => {\r\n const normalized = text.trim();\r\n if (!normalized) {\r\n return;\r\n }\r\n liveTranscriptionTextRef.current = normalized;\r\n if (liveTranscriptionMessageIdRef.current === null) {\r\n liveTranscriptionMessageIdRef.current = appendMessage(normalized, true);\r\n return;\r\n }\r\n updateMessageText(liveTranscriptionMessageIdRef.current, normalized);\r\n };\r\n\r\n const clearLiveUserTranscriptionState = () => {\r\n liveTranscriptionMessageIdRef.current = null;\r\n liveTranscriptionTextRef.current = \"\";\r\n };\r\n\r\n const handleAudioStateChange = (state: AudioStreamState, requestEpoch?: number) => {\r\n if (typeof requestEpoch === \"number\" && !isCurrentRequestEpoch(requestEpoch)) {\r\n return;\r\n }\r\n\r\n if (state === \"rendering\") {\r\n setIsRenderingAudio(true);\r\n setIsPlayingAudio(false);\r\n return;\r\n }\r\n\r\n if (state === \"playing\") {\r\n setIsRenderingAudio(false);\r\n setIsPlayingAudio(true);\r\n return;\r\n }\r\n\r\n if (state === \"fallback\") {\r\n setIsRenderingAudio(true);\r\n setIsPlayingAudio(false);\r\n return;\r\n }\r\n\r\n if (state === \"done\") {\r\n setIsRenderingAudio(false);\r\n setIsPlayingAudio(false);\r\n return;\r\n }\r\n\r\n setIsRenderingAudio(false);\r\n setIsPlayingAudio(false);\r\n };\r\n\r\n const finalizeStreamCycle = (requestEpoch?: number) => {\r\n if (typeof requestEpoch === \"number\" && !isCurrentRequestEpoch(requestEpoch)) {\r\n return;\r\n }\r\n\r\n awaitingAssistantResponseRef.current = false;\r\n setStatusOverride(null);\r\n setIsBusy(false);\r\n isBusyRef.current = false;\r\n setIsTranscribing(false);\r\n setIsThinking(false);\r\n setIsRenderingAudio(false);\r\n setIsPlayingAudio(false);\r\n setIsRunningTools(false);\r\n pendingUserTextRef.current = null;\r\n pendingAssistantTextRef.current = \"\";\r\n assistantMessageIdRef.current = null;\r\n if (activeStreamControllerRef.current) {\r\n activeStreamControllerRef.current = null;\r\n }\r\n if (\r\n !autoListenSuppressedRef.current &&\r\n shouldAutoListenAfterAudio(\r\n accessibilityMode,\r\n expectsReplyRef.current,\r\n isRecordingRef.current,\r\n false,\r\n )\r\n ) {\r\n console.info(\"[Bulut] chat-window auto-listen trigger after stream completion\");\r\n void startRecording(\"vad\");\r\n }\r\n // Reset for next turn\r\n expectsReplyRef.current = true;\r\n };\r\n\r\n const runAgentForUserText = async (userText: string) => {\r\n if (!config.projectId) {\r\n appendMessage(\"Hata: Project ID yapılandırılmamış.\", false);\r\n return;\r\n }\r\n\r\n const normalizedUserText = userText.trim();\r\n if (!normalizedUserText) {\r\n appendMessage(\"Ses kaydı metne dönüştürülemedi. Lütfen tekrar deneyin.\", false);\r\n return;\r\n }\r\n\r\n const requestEpoch = beginRequestEpoch();\r\n setIsBusy(true);\r\n isBusyRef.current = true;\r\n setIsTranscribing(false);\r\n setIsThinking(true);\r\n setIsRenderingAudio(false);\r\n setIsPlayingAudio(false);\r\n setIsRunningTools(false);\r\n setStatusOverride(STATUS_LABELS.thinking);\r\n awaitingAssistantResponseRef.current = true;\r\n\r\n try {\r\n pendingUserTextRef.current = normalizedUserText;\r\n upsertLiveUserTranscription(normalizedUserText);\r\n clearLiveUserTranscriptionState();\r\n\r\n stopActiveStream();\r\n const pageContext = getPageContext().summary;\r\n\r\n const handleToolExecution = async (\r\n call: AgentToolCallInfo,\r\n ): Promise<{ call_id: string; result: string }> => {\r\n const toolCall: ToolCallWithId = {\r\n tool: call.tool as\r\n | \"navigate\"\r\n | \"getPageContext\"\r\n | \"interact\"\r\n | \"scroll\",\r\n call_id: call.call_id,\r\n ...call.args,\r\n } as ToolCallWithId;\r\n return executeSingleToolCall(toolCall);\r\n };\r\n\r\n const controller = agentTextChatStream(\r\n config.backendBaseUrl,\r\n normalizedUserText,\r\n config.projectId,\r\n sessionIdRef.current,\r\n {\r\n model: config.model,\r\n voice: config.voice,\r\n pageContext,\r\n accessibilityMode,\r\n },\r\n {\r\n onSessionId: (sid) => {\r\n if (!isCurrentRequestEpoch(requestEpoch)) return;\r\n if (sid && sid !== sessionIdRef.current) {\r\n sessionIdRef.current = sid;\r\n if (typeof localStorage !== \"undefined\") {\r\n localStorage.setItem(SESSION_ID_KEY, sid);\r\n }\r\n }\r\n },\r\n onAssistantDelta: (delta) => {\r\n if (!isCurrentRequestEpoch(requestEpoch)) return;\r\n setIsTranscribing(false);\r\n setIsThinking(true);\r\n setIsRunningTools(false);\r\n if (awaitingAssistantResponseRef.current) {\r\n awaitingAssistantResponseRef.current = false;\r\n setStatusOverride(null);\r\n }\r\n\r\n pendingAssistantTextRef.current += delta;\r\n\r\n if (assistantMessageIdRef.current === null) {\r\n assistantMessageIdRef.current = appendMessage(\r\n pendingAssistantTextRef.current,\r\n false,\r\n );\r\n } else {\r\n updateMessageText(\r\n assistantMessageIdRef.current,\r\n pendingAssistantTextRef.current,\r\n );\r\n }\r\n },\r\n onAssistantDone: (assistantText, expectsReply) => {\r\n if (!isCurrentRequestEpoch(requestEpoch)) return;\r\n playSfx(\"completed\");\r\n expectsReplyRef.current = expectsReply !== false;\r\n awaitingAssistantResponseRef.current = false;\r\n setStatusOverride(null);\r\n setIsThinking(false);\r\n setIsRenderingAudio(true);\r\n\r\n const finalDisplayText = assistantText || pendingAssistantTextRef.current;\r\n pendingAssistantTextRef.current = finalDisplayText;\r\n\r\n if (assistantMessageIdRef.current !== null) {\r\n updateMessageText(\r\n assistantMessageIdRef.current,\r\n finalDisplayText,\r\n );\r\n } else {\r\n assistantMessageIdRef.current = appendMessage(\r\n finalDisplayText,\r\n false,\r\n );\r\n }\r\n },\r\n onIntermediateReply: (text) => {\r\n if (!isCurrentRequestEpoch(requestEpoch)) return;\r\n void speakText(\r\n config.backendBaseUrl, text, config.voice,\r\n accessibilityMode, (state) => handleAudioStateChange(state, requestEpoch),\r\n ).catch((err) => console.warn(\"[Bulut] intermediate TTS failed\", err));\r\n },\r\n onToolCalls: (calls) => {\r\n if (!isCurrentRequestEpoch(requestEpoch)) return;\r\n if (calls.length > 0) {\r\n playSfx(\"toolCall\");\r\n }\r\n setIsRunningTools(true);\r\n setStatusOverride(STATUS_LABELS.runningTools);\r\n appendToolIndicatorMessages(calls);\r\n\r\n assistantMessageIdRef.current = null;\r\n pendingAssistantTextRef.current = \"\";\r\n },\r\n onToolResult: () => {},\r\n onIteration: () => {\r\n if (!isCurrentRequestEpoch(requestEpoch)) return;\r\n playSfx(\"thinking\");\r\n setIsThinking(true);\r\n setStatusOverride(STATUS_LABELS.thinking);\r\n },\r\n onAudioStateChange: (state) => {\r\n handleAudioStateChange(state, requestEpoch);\r\n },\r\n onError: (err) => {\r\n if (!isCurrentRequestEpoch(requestEpoch)) return;\r\n awaitingAssistantResponseRef.current = false;\r\n setStatusOverride(null);\r\n appendMessage(`Hata: ${err}`, false);\r\n },\r\n },\r\n handleToolExecution,\r\n );\r\n\r\n activeStreamControllerRef.current = controller;\r\n await controller.done;\r\n } catch (error) {\r\n if (!isCurrentRequestEpoch(requestEpoch)) return;\r\n awaitingAssistantResponseRef.current = false;\r\n setStatusOverride(null);\r\n if (error instanceof Error) {\r\n appendMessage(`Hata: ${error.message}`, false);\r\n }\r\n } finally {\r\n finalizeStreamCycle(requestEpoch);\r\n }\r\n };\r\n\r\n const handleAudioBlob = async (blob: Blob) => {\r\n if (!config.projectId) {\r\n appendMessage(\"Hata: Project ID yapılandırılmamış.\", false);\r\n return;\r\n }\r\n\r\n const requestEpoch = beginRequestEpoch();\r\n setIsBusy(true);\r\n isBusyRef.current = true;\r\n setIsTranscribing(true);\r\n setIsThinking(false);\r\n setIsRenderingAudio(false);\r\n setIsPlayingAudio(false);\r\n setIsRunningTools(false);\r\n resetProcessingFlags(); // Start fresh\r\n setStatusOverride(STATUS_LABELS.thinking);\r\n awaitingAssistantResponseRef.current = true;\r\n\r\n try {\r\n const fileType = blob.type || \"audio/webm\";\r\n const extension = fileType.includes(\"ogg\")\r\n ? \"ogg\"\r\n : fileType.includes(\"wav\")\r\n ? \"wav\"\r\n : fileType.includes(\"mpeg\") || fileType.includes(\"mp3\")\r\n ? \"mp3\"\r\n : \"webm\";\r\n const file = new File([blob], `voice-input.${extension}`, {\r\n type: fileType,\r\n });\r\n\r\n stopActiveStream();\r\n\r\n const pageContext = getPageContext().summary;\r\n\r\n // Helper: bridge an AgentToolCallInfo to a ToolCallWithId\r\n const handleToolExecution = async (\r\n call: AgentToolCallInfo,\r\n ): Promise<{ call_id: string; result: string }> => {\r\n const toolCall: ToolCallWithId = {\r\n tool: call.tool as\r\n | \"navigate\"\r\n | \"getPageContext\"\r\n | \"interact\"\r\n | \"scroll\",\r\n call_id: call.call_id,\r\n ...call.args,\r\n } as ToolCallWithId;\r\n return executeSingleToolCall(toolCall);\r\n };\r\n\r\n const controller = agentVoiceChatStream(\r\n config.backendBaseUrl,\r\n file,\r\n config.projectId,\r\n sessionIdRef.current,\r\n {\r\n model: config.model,\r\n voice: config.voice,\r\n pageContext,\r\n accessibilityMode,\r\n },\r\n {\r\n onSttRequestSent: () => {\r\n if (!isCurrentRequestEpoch(requestEpoch)) return;\r\n playSttSentCueOnce();\r\n },\r\n onTranscription: (data) => {\r\n if (!isCurrentRequestEpoch(requestEpoch)) return;\r\n if (data.session_id && data.session_id !== sessionIdRef.current) {\r\n sessionIdRef.current = data.session_id;\r\n if (typeof localStorage !== \"undefined\") {\r\n localStorage.setItem(SESSION_ID_KEY, data.session_id);\r\n }\r\n }\r\n\r\n const normalized = data.user_text.trim();\r\n if (normalized) {\r\n const previousUserText = pendingUserTextRef.current;\r\n pendingUserTextRef.current = normalized;\r\n if (liveTranscriptionMessageIdRef.current !== null) {\r\n updateMessageText(liveTranscriptionMessageIdRef.current, normalized);\r\n clearLiveUserTranscriptionState();\r\n } else if (previousUserText !== normalized) {\r\n appendMessage(normalized, true);\r\n }\r\n }\r\n\r\n setIsTranscribing(false);\r\n setIsThinking(true);\r\n setStatusOverride(STATUS_LABELS.thinking);\r\n },\r\n onSessionId: (sid) => {\r\n if (!isCurrentRequestEpoch(requestEpoch)) return;\r\n if (sid && sid !== sessionIdRef.current) {\r\n sessionIdRef.current = sid;\r\n if (typeof localStorage !== \"undefined\") {\r\n localStorage.setItem(SESSION_ID_KEY, sid);\r\n }\r\n }\r\n },\r\n onAssistantDelta: (delta) => {\r\n if (!isCurrentRequestEpoch(requestEpoch)) return;\r\n setIsTranscribing(false);\r\n setIsThinking(true);\r\n setIsRunningTools(false);\r\n if (awaitingAssistantResponseRef.current) {\r\n awaitingAssistantResponseRef.current = false;\r\n setStatusOverride(null);\r\n }\r\n\r\n // Agent returns plain text (not JSON), stream it directly\r\n pendingAssistantTextRef.current += delta;\r\n\r\n if (assistantMessageIdRef.current === null) {\r\n assistantMessageIdRef.current = appendMessage(\r\n pendingAssistantTextRef.current,\r\n false,\r\n );\r\n } else {\r\n updateMessageText(\r\n assistantMessageIdRef.current,\r\n pendingAssistantTextRef.current,\r\n );\r\n }\r\n },\r\n onAssistantDone: (assistantText, expectsReply) => {\r\n if (!isCurrentRequestEpoch(requestEpoch)) return;\r\n playSfx(\"completed\");\r\n expectsReplyRef.current = expectsReply !== false;\r\n awaitingAssistantResponseRef.current = false;\r\n setStatusOverride(null);\r\n setIsThinking(false);\r\n setIsRenderingAudio(true);\r\n\r\n const finalDisplayText =\r\n assistantText || pendingAssistantTextRef.current;\r\n\r\n streamingJsonParserRef.current = null;\r\n pendingAssistantTextRef.current = finalDisplayText;\r\n\r\n if (assistantMessageIdRef.current !== null) {\r\n updateMessageText(\r\n assistantMessageIdRef.current,\r\n finalDisplayText,\r\n );\r\n } else {\r\n assistantMessageIdRef.current = appendMessage(\r\n finalDisplayText,\r\n false,\r\n );\r\n }\r\n },\r\n onIntermediateReply: (text) => {\r\n if (!isCurrentRequestEpoch(requestEpoch)) return;\r\n void speakText(\r\n config.backendBaseUrl, text, config.voice,\r\n accessibilityMode, (state) => handleAudioStateChange(state, requestEpoch),\r\n ).catch((err) => console.warn(\"[Bulut] intermediate TTS failed\", err));\r\n },\r\n onToolCalls: (calls) => {\r\n if (!isCurrentRequestEpoch(requestEpoch)) return;\r\n if (calls.length > 0) {\r\n playSfx(\"toolCall\");\r\n }\r\n setIsRunningTools(true);\r\n setStatusOverride(STATUS_LABELS.runningTools);\r\n appendToolIndicatorMessages(calls);\r\n assistantMessageIdRef.current = null;\r\n pendingAssistantTextRef.current = \"\";\r\n },\r\n onToolResult: (_callId, _toolName, _result) => {\r\n // Tool result sent back to agent automatically.\r\n // Could display detailed result here if needed.\r\n },\r\n onIteration: (_iteration, _maxIterations) => {\r\n if (!isCurrentRequestEpoch(requestEpoch)) return;\r\n // Agent started a new reasoning iteration\r\n playSfx(\"thinking\");\r\n setIsThinking(true);\r\n setStatusOverride(STATUS_LABELS.thinking);\r\n },\r\n onAudioStateChange: (state) => {\r\n handleAudioStateChange(state, requestEpoch);\r\n },\r\n onError: (err) => {\r\n if (!isCurrentRequestEpoch(requestEpoch)) return;\r\n awaitingAssistantResponseRef.current = false;\r\n setStatusOverride(null);\r\n appendMessage(`Hata: ${err}`, false);\r\n },\r\n },\r\n handleToolExecution,\r\n );\r\n\r\n activeStreamControllerRef.current = controller;\r\n await controller.done;\r\n\r\n // Safety: Ensure messages are flushed if not already\r\n // (e.g. if audio 'done' event didn't fire for some reason or there was no audio)\r\n /* if (pendingUserTextRef.current || pendingAssistantTextRef.current) {\r\n // flushPendingMessages(); // Removed as we stream now\r\n } */\r\n } catch (error) {\r\n if (!isCurrentRequestEpoch(requestEpoch)) return;\r\n // Error already shown via onError callback — don't duplicate\r\n awaitingAssistantResponseRef.current = false;\r\n setStatusOverride(null);\r\n } finally {\r\n finalizeStreamCycle(requestEpoch);\r\n }\r\n };\r\n\r\n const stopRecording = (options?: { discard?: boolean }) => {\r\n const recorder = recorderRef.current;\r\n if (!recorder || recorder.state === \"inactive\") {\r\n return;\r\n }\r\n\r\n if (options?.discard) {\r\n discardNextRecordingRef.current = true;\r\n }\r\n\r\n cleanupVAD();\r\n recorder.stop();\r\n };\r\n\r\n const setupVAD = (stream: MediaStream, recorder: MediaRecorder) => {\r\n const AudioCtx =\r\n window.AudioContext ||\r\n (window as Window & { webkitAudioContext?: typeof AudioContext })\r\n .webkitAudioContext;\r\n\r\n if (!AudioCtx) {\r\n return;\r\n }\r\n\r\n const context = new AudioCtx();\r\n audioContextRef.current = context;\r\n\r\n const analyser = context.createAnalyser();\r\n analyser.fftSize = 256;\r\n\r\n const source = context.createMediaStreamSource(stream);\r\n sourceRef.current = source;\r\n source.connect(analyser);\r\n\r\n const dataArray = new Uint8Array(analyser.frequencyBinCount);\r\n silenceStartRef.current = null;\r\n let speechDetected = false;\r\n let speechStartedAt: number | null = null;\r\n const enforceMinSpeechDuration = accessibilityMode;\r\n\r\n vadIntervalRef.current = window.setInterval(() => {\r\n if (!isRecordingRef.current || recorder.state === \"inactive\") {\r\n cleanupVAD();\r\n return;\r\n }\r\n\r\n analyser.getByteFrequencyData(dataArray);\r\n\r\n let sum = 0;\r\n for (const value of dataArray) {\r\n sum += value;\r\n }\r\n const average = sum / dataArray.length;\r\n const volume = average / 255;\r\n\r\n if (volume < VAD_THRESHOLD) {\r\n if (!speechDetected) {\r\n speechStartedAt = null;\r\n silenceStartRef.current = null;\r\n return;\r\n }\r\n\r\n if (silenceStartRef.current === null) {\r\n silenceStartRef.current = Date.now();\r\n return;\r\n }\r\n\r\n const silenceDuration = Date.now() - silenceStartRef.current;\r\n if (speechDetected && silenceDuration > SILENCE_DURATION_MS) {\r\n stopRecording();\r\n }\r\n return;\r\n }\r\n\r\n silenceStartRef.current = null;\r\n if (speechStartedAt === null) {\r\n speechStartedAt = Date.now();\r\n }\r\n\r\n if (!speechDetected) {\r\n const speechDuration = Date.now() - speechStartedAt;\r\n if (shouldAcceptVadSpeech(speechDuration, enforceMinSpeechDuration)) {\r\n speechDetected = true;\r\n if (enforceMinSpeechDuration) {\r\n setStatusOverride(STATUS_LABELS.listening);\r\n }\r\n }\r\n }\r\n }, 50);\r\n };\r\n\r\n const startRecording = async (mode: RecordingMode) => {\r\n if (\r\n isBusyRef.current ||\r\n isRecordingRef.current ||\r\n startRecordingPendingRef.current\r\n ) {\r\n return;\r\n }\r\n\r\n setStatusOverride(STATUS_LABELS.micInitializing);\r\n setIsMicPending(true);\r\n\r\n if (!navigator.mediaDevices?.getUserMedia) {\r\n setStatusOverride(null);\r\n setIsMicPending(false);\r\n appendMessage(\"Bu tarayıcıda mikrofon kullanılamıyor.\", false);\r\n return;\r\n }\r\n\r\n if (typeof MediaRecorder === \"undefined\") {\r\n setStatusOverride(null);\r\n setIsMicPending(false);\r\n appendMessage(\"Bu tarayıcıda MediaRecorder desteklenmiyor.\", false);\r\n return;\r\n }\r\n\r\n startRecordingPendingRef.current = true;\r\n\r\n try {\r\n const stream = await navigator.mediaDevices.getUserMedia({ audio: true });\r\n streamRef.current = stream;\r\n\r\n // Use low bitrate for speech — opus handles voice well at 16-24 kbps\r\n // and produces ~4-5x smaller payloads, speeding up the upload to fal.\r\n const recorderOptions: MediaRecorderOptions = {\r\n audioBitsPerSecond: 16_000,\r\n };\r\n\r\n // Prefer opus-in-ogg (smaller container) → opus-in-webm → browser default\r\n const preferredMimeTypes = [\r\n \"audio/ogg;codecs=opus\",\r\n \"audio/webm;codecs=opus\",\r\n \"audio/webm\",\r\n ];\r\n for (const mime of preferredMimeTypes) {\r\n if (MediaRecorder.isTypeSupported(mime)) {\r\n recorderOptions.mimeType = mime;\r\n break;\r\n }\r\n }\r\n\r\n const recorder = new MediaRecorder(stream, recorderOptions);\r\n recorderRef.current = recorder;\r\n audioChunksRef.current = [];\r\n clearLiveUserTranscriptionState();\r\n sttSendCuePlayedRef.current = false;\r\n\r\n const sttMimeType = (recorder.mimeType || recorderOptions.mimeType || \"audio/webm\")\r\n .split(\";\")[0]\r\n .trim() || \"audio/webm\";\r\n\r\n const sttWsController = startSttWebSocketStream(\r\n config.backendBaseUrl,\r\n {\r\n projectId: config.projectId,\r\n sessionId: sessionIdRef.current,\r\n language: \"tr\",\r\n mimeType: sttMimeType,\r\n },\r\n {\r\n onSessionId: (sid) => {\r\n if (!sid || sid === sessionIdRef.current) {\r\n return;\r\n }\r\n sessionIdRef.current = sid;\r\n if (typeof localStorage !== \"undefined\") {\r\n localStorage.setItem(SESSION_ID_KEY, sid);\r\n }\r\n },\r\n onPartial: ({ text }) => {\r\n if (!text.trim()) {\r\n return;\r\n }\r\n upsertLiveUserTranscription(text);\r\n },\r\n },\r\n );\r\n activeSttWsRef.current = sttWsController;\r\n\r\n recorder.ondataavailable = (event) => {\r\n if (event.data.size > 0) {\r\n audioChunksRef.current.push(event.data);\r\n if (activeSttWsRef.current) {\r\n void activeSttWsRef.current.pushChunk(event.data).catch((error) => {\r\n console.warn(\r\n `[Bulut] STT WS chunk send failed: ${error instanceof Error ? error.message : String(error)}`,\r\n );\r\n });\r\n }\r\n }\r\n };\r\n\r\n recorder.onerror = () => {\r\n appendMessage(\"Mikrofon kaydı sırasında bir hata oluştu.\", false);\r\n };\r\n\r\n recorder.onstop = async () => {\r\n setIsRecording(false);\r\n isRecordingRef.current = false;\r\n recordingModeRef.current = null;\r\n stopRecordingTimer();\r\n\r\n cleanupVAD();\r\n stopStreamTracks();\r\n\r\n const shouldDiscard = discardNextRecordingRef.current;\r\n discardNextRecordingRef.current = false;\r\n\r\n const blob = new Blob(audioChunksRef.current, {\r\n type: recorder.mimeType || \"audio/webm\",\r\n });\r\n audioChunksRef.current = [];\r\n\r\n const currentSttWs = activeSttWsRef.current;\r\n activeSttWsRef.current = null;\r\n\r\n if (shouldDiscard) {\r\n currentSttWs?.cancel();\r\n clearLiveUserTranscriptionState();\r\n setStatusOverride(null);\r\n return;\r\n }\r\n\r\n if (blob.size === 0) {\r\n currentSttWs?.cancel();\r\n clearLiveUserTranscriptionState();\r\n setStatusOverride(null);\r\n appendMessage(\"Ses kaydı alınamadı. Lütfen tekrar deneyin.\", false);\r\n return;\r\n }\r\n\r\n setIsTranscribing(true);\r\n setStatusOverride(STATUS_LABELS.transcribing);\r\n\r\n try {\r\n if (currentSttWs) {\r\n playSttSentCueOnce();\r\n const sttResult = await currentSttWs.stop();\r\n if (sttResult.session_id && sttResult.session_id !== sessionIdRef.current) {\r\n sessionIdRef.current = sttResult.session_id;\r\n if (typeof localStorage !== \"undefined\") {\r\n localStorage.setItem(SESSION_ID_KEY, sttResult.session_id);\r\n }\r\n }\r\n if (sttResult.text.trim()) {\r\n upsertLiveUserTranscription(sttResult.text);\r\n setStatusOverride(STATUS_LABELS.thinking);\r\n await runAgentForUserText(sttResult.text);\r\n return;\r\n }\r\n }\r\n } catch (error) {\r\n console.warn(\r\n `[Bulut] STT WS finalization failed, falling back to HTTP POST /chat/stt: ${error instanceof Error ? error.message : String(error)}`,\r\n );\r\n } finally {\r\n clearLiveUserTranscriptionState();\r\n }\r\n\r\n console.info(\"[Bulut] Using HTTP POST fallback for STT (streaming WS did not succeed)\");\r\n setStatusOverride(STATUS_LABELS.thinking);\r\n await handleAudioBlob(blob);\r\n };\r\n\r\n if (mode === \"vad\") {\r\n setupVAD(stream, recorder);\r\n }\r\n\r\n recorder.start(200);\r\n recordingModeRef.current = mode;\r\n setIsRecording(true);\r\n isRecordingRef.current = true;\r\n startRecordingTimer();\r\n\r\n setStatusOverride(\r\n accessibilityMode && mode === \"vad\"\r\n ? STATUS_LABELS.accessibilityActive\r\n : STATUS_LABELS.listening,\r\n );\r\n\r\n if (pendingStopAfterStartRef.current) {\r\n pendingStopAfterStartRef.current = false;\r\n stopRecording();\r\n }\r\n } catch (error) {\r\n const errMsg = normalizeError(error);\r\n if (errMsg.toLowerCase().includes(\"permission\") || errMsg.toLowerCase().includes(\"denied\")) {\r\n autoListenSuppressedRef.current = true;\r\n }\r\n cancelActiveSttWs();\r\n setStatusOverride(null);\r\n appendMessage(`Mikrofon hatası: ${errMsg}`, false);\r\n cleanupVAD();\r\n stopStreamTracks();\r\n pendingStopAfterStartRef.current = false;\r\n setIsRecording(false);\r\n isRecordingRef.current = false;\r\n stopRecordingTimer();\r\n } finally {\r\n if (!isRecordingRef.current && !isBusyRef.current) {\r\n setStatusOverride(null);\r\n }\r\n startRecordingPendingRef.current = false;\r\n setIsMicPending(false);\r\n }\r\n };\r\n\r\n const resetMicGesture = () => {\r\n micPressStartRef.current = null;\r\n micHoldTriggeredRef.current = false;\r\n clearMicHoldTimeout();\r\n };\r\n\r\n const handleMicPointerDown = (\r\n event: JSX.TargetedPointerEvent<HTMLButtonElement>,\r\n ) => {\r\n event.preventDefault();\r\n\r\n if (isBusyRef.current) {\r\n return;\r\n }\r\n\r\n if (isRecordingRef.current) {\r\n // In VAD mode, tapping the button cancels; in press mode, it sends\r\n if (recordingModeRef.current === \"vad\") {\r\n stopRecording({ discard: true });\r\n } else {\r\n stopRecording();\r\n }\r\n return;\r\n }\r\n\r\n micPressStartRef.current = Date.now();\r\n micHoldTriggeredRef.current = false;\r\n clearMicHoldTimeout();\r\n\r\n if (event.currentTarget.setPointerCapture) {\r\n try {\r\n event.currentTarget.setPointerCapture(event.pointerId);\r\n } catch {\r\n // No-op.\r\n }\r\n }\r\n\r\n micHoldTimeoutRef.current = window.setTimeout(() => {\r\n if (\r\n micPressStartRef.current === null ||\r\n isBusyRef.current ||\r\n isRecordingRef.current\r\n ) {\r\n return;\r\n }\r\n\r\n micHoldTriggeredRef.current = true;\r\n void startRecording(\"press\");\r\n }, HOLD_THRESHOLD_MS);\r\n };\r\n\r\n const handleMicPointerUp = (\r\n event: JSX.TargetedPointerEvent<HTMLButtonElement>,\r\n ) => {\r\n event.preventDefault();\r\n\r\n if (event.currentTarget.releasePointerCapture) {\r\n try {\r\n event.currentTarget.releasePointerCapture(event.pointerId);\r\n } catch {\r\n // No-op.\r\n }\r\n }\r\n\r\n const startedAt = micPressStartRef.current;\r\n const wasHold = micHoldTriggeredRef.current;\r\n resetMicGesture();\r\n\r\n if (startedAt === null) {\r\n return;\r\n }\r\n\r\n if (wasHold) {\r\n if (isRecordingRef.current) {\r\n stopRecording();\r\n } else if (startRecordingPendingRef.current) {\r\n pendingStopAfterStartRef.current = true;\r\n }\r\n return;\r\n }\r\n\r\n const duration = Date.now() - startedAt;\r\n if (classifyMicGesture(duration) === \"tap\") {\r\n void startRecording(\"vad\");\r\n }\r\n };\r\n\r\n const handleMicPointerCancel = (\r\n event: JSX.TargetedPointerEvent<HTMLButtonElement>,\r\n ) => {\r\n handleMicPointerUp(event);\r\n };\r\n\r\n const handleRestart = () => {\r\n invalidateRequestEpoch();\r\n sttSendCuePlayedRef.current = false;\r\n resetMicGesture();\r\n pendingStopAfterStartRef.current = false;\r\n\r\n stopActiveStream();\r\n stopActiveAudioPlayback();\r\n cancelActiveSttWs();\r\n\r\n if (recorderRef.current && recorderRef.current.state !== \"inactive\") {\r\n stopRecording({ discard: true });\r\n } else {\r\n discardNextRecordingRef.current = false;\r\n cleanupVAD();\r\n stopStreamTracks();\r\n }\r\n\r\n stopRecordingTimer();\r\n setRecordingDurationMs(0);\r\n\r\n clearPersistedChatState(\r\n typeof localStorage !== \"undefined\" ? localStorage : null,\r\n );\r\n\r\n sessionIdRef.current = null;\r\n const initialMessages = createInitialMessages(config.agentName);\r\n nextMessageIdRef.current = getNextMessageId(initialMessages);\r\n setMessages(initialMessages);\r\n\r\n setIsBusy(false);\r\n isBusyRef.current = false;\r\n setIsRecording(false);\r\n isRecordingRef.current = false;\r\n resetProcessingFlags();\r\n };\r\n\r\n // Auto-listen when accessibility mode is activated (initial trigger)\r\n useEffect(() => {\r\n if (!accessibilityMode || autoListenSuppressedRef.current) return;\r\n const timer = window.setTimeout(() => {\r\n if (!isRecordingRef.current && !isBusyRef.current && !startRecordingPendingRef.current && !autoListenSuppressedRef.current) {\r\n void startRecording(\"vad\");\r\n }\r\n }, 500);\r\n return () => window.clearTimeout(timer);\r\n }, [accessibilityMode]); // eslint-disable-line react-hooks/exhaustive-deps\r\n\r\n const stopTask = () => {\r\n invalidateRequestEpoch();\r\n sttSendCuePlayedRef.current = false;\r\n stopActiveStream();\r\n stopActiveAudioPlayback();\r\n cancelActiveSttWs();\r\n stopRecording({ discard: true });\r\n cleanupVAD();\r\n stopStreamTracks();\r\n resetProcessingFlags();\r\n setIsBusy(false);\r\n isBusyRef.current = false;\r\n };\r\n\r\n // Expose recording actions to parent via actionsRef\r\n if (actionsRef) {\r\n actionsRef.current = {\r\n startRecording: () => {\r\n autoListenSuppressedRef.current = false;\r\n void startRecording(\"vad\");\r\n },\r\n cancelRecording: () => {\r\n stopActiveAudioPlayback();\r\n cancelActiveSttWs();\r\n const recorder = recorderRef.current;\r\n if (recorder && recorder.state !== \"inactive\") {\r\n stopRecording({ discard: true });\r\n } else {\r\n cleanupVAD();\r\n stopStreamTracks();\r\n }\r\n },\r\n stopTask,\r\n };\r\n }\r\n\r\n const windowStyle: { [key: string]: string } = {\r\n position: \"fixed\",\r\n bottom: `${POSITION_BOTTOM}px`,\r\n right: `${POSITION_RIGHT}px`,\r\n width: `${WINDOW_WIDTH}px`,\r\n maxHeight: `${WINDOW_HEIGHT}px`,\r\n backgroundColor: \"hsla(0, 0%, 100%, 1)\",\r\n borderRadius: BORDER_RADIUS.window,\r\n display: hidden ? \"none\" : \"flex\",\r\n flexDirection: \"column\",\r\n overflow: \"hidden\",\r\n zIndex: \"10000\",\r\n animation: hidden ? \"none\" : `slideIn ${TRANSITIONS.medium}`,\r\n boxShadow: accessibilityMode\r\n ? `inset 0 0 0 2px ${COLORS.primary}, ${SHADOW}`\r\n : SHADOW,\r\n fontFamily: \"\\\"Geist\\\", sans-serif\",\r\n };\r\n\r\n const headerStyle: { [key: string]: string } = {\r\n padding: \"14px 16px\",\r\n display: \"flex\",\r\n justifyContent: \"space-between\",\r\n alignItems: \"center\",\r\n };\r\n\r\n const headerActionsStyle: { [key: string]: string } = {\r\n display: \"flex\",\r\n alignItems: \"center\",\r\n gap: \"6px\",\r\n };\r\n\r\n const headerButtonStyle: { [key: string]: string } = {\r\n background: \"none\",\r\n border: \"none\",\r\n cursor: \"pointer\",\r\n padding: \"4px\",\r\n borderRadius: \"6px\",\r\n color: COLORS.text,\r\n display: \"flex\",\r\n alignItems: \"center\",\r\n justifyContent: \"center\",\r\n transition: `color ${TRANSITIONS.fast}, background-color ${TRANSITIONS.fast}`,\r\n };\r\n\r\n const messagesContainerStyle: { [key: string]: string } = {\r\n padding: \"0px 16px\",\r\n overflowY: \"auto\",\r\n flex: \"1\",\r\n minHeight: \"0\",\r\n };\r\n\r\n const messagesListStyle: { [key: string]: string } = {\r\n display: \"flex\",\r\n flexDirection: \"column\",\r\n gap: \"16px\",\r\n };\r\n\r\n const messageStyle = (isUser: boolean): JSX.CSSProperties => ({\r\n maxWidth: \"84%\",\r\n padding: isUser ? \"9px 14px\" : \"0px 0px\",\r\n borderRadius: BORDER_RADIUS.message,\r\n fontSize: \"14px\",\r\n lineHeight: \"140%\",\r\n wordWrap: \"break-word\",\r\n whiteSpace: \"pre-wrap\",\r\n alignSelf: isUser ? \"flex-end\" : \"flex-start\",\r\n backgroundColor: isUser ? COLORS.messageUser : \"\",\r\n color: isUser ? COLORS.messageUserText : \"hsla(215, 100%, 5%, 1)\",\r\n });\r\n\r\n const resolveToolIconSrc = (kind: Message[\"toolKind\"]): string => {\r\n if (kind === \"cursor\") {\r\n return cursorArrowRaysIconContent;\r\n }\r\n if (kind === \"scroll\") {\r\n return handRaisedIconContent;\r\n }\r\n if (kind === \"navigate\") {\r\n return mapIconContent;\r\n }\r\n if (kind === \"form\") {\r\n return queueListIconContent;\r\n }\r\n if (kind === \"interact\") {\r\n return handRaisedIconContent;\r\n }\r\n if (kind === \"unknown\") {\r\n return commandLineIconContent;\r\n }\r\n return faceSmileIconContent;\r\n };\r\n\r\n const footerStyle: { [key: string]: string } = {\r\n padding: \"10px 12px\",\r\n display: \"flex\",\r\n alignItems: \"center\",\r\n justifyContent: \"space-between\",\r\n gap: \"8px\",\r\n };\r\n\r\n const statusPanelStyle: { [key: string]: string } = {\r\n flex: \"1\",\r\n minHeight: \"34px\",\r\n borderRadius: \"10px\",\r\n color: COLORS.text,\r\n fontSize: \"14px\",\r\n display: \"flex\",\r\n alignItems: \"center\",\r\n padding: \"0 10px\",\r\n whiteSpace: \"nowrap\",\r\n overflow: \"hidden\",\r\n textOverflow: \"ellipsis\",\r\n opacity: \"0.7\",\r\n };\r\n\r\n const footerActionsStyle: { [key: string]: string } = {\r\n display: \"flex\",\r\n alignItems: \"center\",\r\n gap: \"8px\",\r\n flexShrink: \"0\",\r\n };\r\n\r\n const recordingTimerStyle: { [key: string]: string } = {\r\n minWidth: \"46px\",\r\n fontSize: \"12px\",\r\n fontWeight: \"700\",\r\n color: COLORS.text,\r\n textAlign: \"right\",\r\n };\r\n\r\n const micFooterButtonStyle: { [key: string]: string } = {\r\n width: \"37px\",\r\n height: \"37px\",\r\n borderRadius: \"999px\",\r\n background: \"transparent\",\r\n display: \"flex\",\r\n alignItems: \"center\",\r\n justifyContent: \"center\",\r\n cursor: \"pointer\",\r\n color: \"#ffffff\",\r\n border: \"1px solid hsla(215, 100%, 5%, 0.5)\",\r\n transition: `transform ${TRANSITIONS.fast}`,\r\n };\r\n\r\n const isVadRecording = isRecording && recordingModeRef.current === \"vad\";\r\n const showStopButton = isBusy && !isRecording;\r\n const hideMicButton = isMicPending && !isRecording;\r\n const disableMicControl = isBusy;\r\n\r\n return (\r\n <div className=\"bulut-chat-window\" style={windowStyle}>\r\n <style>{`\r\n @keyframes slideIn {\r\n from {\r\n opacity: 0;\r\n transform: translateY(10px);\r\n }\r\n to {\r\n opacity: 1;\r\n transform: translateY(0);\r\n }\r\n }\r\n\r\n .bulut-header-btn:hover:not(:disabled) {\r\n color: ${COLORS.text};\r\n }\r\n\r\n .bulut-footer-btn:hover:not(:disabled) {\r\n transform: scale(1.04);\r\n }\r\n\r\n .bulut-header-btn:disabled,\r\n .bulut-footer-btn:disabled {\r\n cursor: not-allowed;\r\n opacity: 0.5;\r\n transform: none;\r\n }\r\n\r\n @keyframes bulut-dots {\r\n 0% { content: '.'; }\r\n 33% { content: '..'; }\r\n 66% { content: '...'; }\r\n }\r\n\r\n .bulut-status-dots::after {\r\n content: '.';\r\n animation: bulut-dots 1.2s steps(1) infinite;\r\n display: inline-block;\r\n min-width: 12px;\r\n text-align: left;\r\n }\r\n\r\n /* Mobile: full-screen chat window */\r\n @media (max-width: 600px) {\r\n .bulut-chat-window {\r\n top: 0 !important;\r\n left: 0 !important;\r\n right: 0 !important;\r\n bottom: 0 !important;\r\n width: 100% !important;\r\n max-height: 100% !important;\r\n height: 100% !important;\r\n border-radius: 0 !important;\r\n }\r\n .bulut-close-btn {\r\n width: 40px !important;\r\n height: 40px !important;\r\n padding: 8px !important;\r\n }\r\n .bulut-close-btn svg {\r\n width: 26px !important;\r\n height: 26px !important;\r\n }\r\n }\r\n `}</style>\r\n\r\n <div style={headerStyle}>\r\n <SvgIcon\r\n src={logoContent}\r\n title=\"Bulut Logo\"\r\n style={{ maxWidth: \"80px\", height: \"auto\" }}\r\n stripColors={false}\r\n />\r\n <div style={headerActionsStyle}>\r\n <button\r\n type=\"button\"\r\n className=\"bulut-header-btn\"\r\n style={headerButtonStyle}\r\n onClick={handleRestart}\r\n aria-label=\"Sohbeti yeniden başlat\"\r\n title=\"Sohbeti yeniden başlat\"\r\n >\r\n <SvgIcon src={arrowPathIconContent} aria-hidden=\"true\" width={22} height={22} />\r\n </button>\r\n\r\n <button\r\n type=\"button\"\r\n className=\"bulut-header-btn bulut-close-btn\"\r\n style={{\r\n ...headerButtonStyle,\r\n display: \"flex\",\r\n alignItems: \"center\",\r\n justifyContent: \"center\",\r\n }}\r\n onClick={onClose}\r\n aria-label=\"Sohbeti kapat\"\r\n title=\"Sohbeti kapat\"\r\n >\r\n <SvgIcon src={xMarkIconContent} aria-hidden=\"true\" width={22} height={22} />\r\n </button>\r\n </div>\r\n </div>\r\n\r\n <div style={messagesContainerStyle} ref={messagesContainerRef}>\r\n <div style={messagesListStyle} ref={messagesContentRef}>\r\n {messages.map((message) => {\r\n if (message.type === \"tool\") {\r\n const toolIconSrc = resolveToolIconSrc(message.toolKind);\r\n return (\r\n <div\r\n key={message.id}\r\n style={{\r\n ...messageStyle(false),\r\n opacity: \"0.7\",\r\n display: \"flex\",\r\n alignItems: \"center\",\r\n gap: \"8px\",\r\n }}\r\n >\r\n <SvgIcon\r\n src={toolIconSrc}\r\n aria-hidden=\"true\"\r\n width={20}\r\n height={20}\r\n style={{ flexShrink: 0 }}\r\n />\r\n <span>{message.text}</span>\r\n </div>\r\n );\r\n }\r\n\r\n return (\r\n <div key={message.id} style={messageStyle(message.isUser)}>\r\n {message.text}\r\n </div>\r\n );\r\n })}\r\n </div>\r\n </div>\r\n\r\n <div style={footerStyle}>\r\n <div style={{ ...statusPanelStyle, transition: \"opacity 0.2s ease-out\" }}>\r\n {showStatus ? (\r\n <span className=\"bulut-status-dots\" title={statusText}>\r\n {statusText}\r\n </span>\r\n ) : onAccessibilityToggle ? (\r\n <div\r\n style={{\r\n display: \"flex\",\r\n alignItems: \"center\",\r\n gap: \"8px\",\r\n }}\r\n >\r\n <span\r\n style={{\r\n fontSize: \"12px\",\r\n opacity: \"0.6\",\r\n whiteSpace: \"nowrap\",\r\n }}\r\n >\r\n Erişilebilirlik\r\n </span>\r\n <button\r\n type=\"button\"\r\n onClick={onAccessibilityToggle}\r\n aria-label={\r\n accessibilityMode\r\n ? \"Erişilebilirlik modunu kapat\"\r\n : \"Erişilebilirlik modunu aç\"\r\n }\r\n style={{\r\n width: \"36px\",\r\n height: \"20px\",\r\n borderRadius: \"10px\",\r\n border: \"none\",\r\n cursor: \"pointer\",\r\n padding: \"2px\",\r\n display: \"flex\",\r\n alignItems: \"center\",\r\n backgroundColor: accessibilityMode\r\n ? COLORS.primary\r\n : \"hsla(215, 10%, 75%, 1)\",\r\n transition: `background-color ${TRANSITIONS.fast}`,\r\n flexShrink: \"0\",\r\n }}\r\n >\r\n <span\r\n style={{\r\n width: \"16px\",\r\n height: \"16px\",\r\n borderRadius: \"50%\",\r\n backgroundColor: \"#ffffff\",\r\n display: \"block\",\r\n transition: `transform ${TRANSITIONS.fast}`,\r\n transform: accessibilityMode\r\n ? \"translateX(16px)\"\r\n : \"translateX(0)\",\r\n boxShadow: \"0 1px 3px rgba(0,0,0,0.2)\",\r\n }}\r\n />\r\n </button>\r\n </div>\r\n ) : null}\r\n </div>\r\n\r\n <div style={footerActionsStyle}>\r\n {isRecording ? (\r\n <span style={recordingTimerStyle}>\r\n {formatDurationMs(recordingDurationMs)}\r\n </span>\r\n ) : null}\r\n {showStopButton ? (\r\n <button\r\n type=\"button\"\r\n className=\"bulut-footer-btn\"\r\n style={micFooterButtonStyle}\r\n onClick={stopTask}\r\n aria-label=\"Görevi durdur\"\r\n title=\"Görevi durdur\"\r\n >\r\n <SvgIcon\r\n src={stopOutlineIconContent}\r\n aria-hidden=\"true\"\r\n width={22}\r\n height={22}\r\n style={{ color: \"hsla(215, 100%, 5%, 1)\" }}\r\n />\r\n </button>\r\n ) : hideMicButton ? null : (\r\n <button\r\n type=\"button\"\r\n className=\"bulut-footer-btn\"\r\n style={micFooterButtonStyle}\r\n onPointerDown={handleMicPointerDown}\r\n onPointerUp={handleMicPointerUp}\r\n onPointerCancel={handleMicPointerCancel}\r\n disabled={disableMicControl}\r\n aria-label={isVadRecording ? \"Kaydı iptal et\" : isRecording ? \"Kaydı durdur\" : \"Kaydı başlat\"}\r\n title={\r\n isVadRecording\r\n ? \"Kaydı iptal et\"\r\n : isRecording\r\n ? \"Kaydı durdur\"\r\n : \"Dokun: VAD, Basılı tut: bırakınca gönder\"\r\n }\r\n >\r\n {isVadRecording ? (\r\n <SvgIcon\r\n src={xMarkIconContent}\r\n aria-hidden=\"true\"\r\n width={22}\r\n height={22}\r\n style={{ color: \"hsla(215, 100%, 5%, 1)\"}}\r\n />\r\n ) : (\r\n <SvgIcon\r\n src={microphoneOutlineIconContent}\r\n aria-hidden=\"true\"\r\n width={22}\r\n height={22}\r\n style={{ color: \"hsla(215, 100%, 5%, 1)\" }}\r\n />\r\n )}\r\n </button>\r\n )}\r\n </div>\r\n </div>\r\n </div>\r\n );\r\n};\r\n","export default \"<svg width=\\\"2420\\\" height=\\\"438\\\" viewBox=\\\"0 0 2420 438\\\" fill=\\\"none\\\" xmlns=\\\"http://www.w3.org/2000/svg\\\">\\r\\n<path d=\\\"M251.823 0C313.458 0 366.34 37.767 388.932 91.5918C409.253 75.8883 434.678 66.5567 462.264 66.5566C528.885 66.5567 582.893 120.989 582.893 188.134C582.893 188.786 582.885 189.436 582.875 190.086C648.144 193.552 700 247.572 700 313.704C700 382.074 644.575 437.5 576.204 437.5H123.796C55.4255 437.5 7.42386e-05 382.074 0 313.704C0 250.475 47.4025 198.32 108.608 190.833C104.967 177.844 103.019 164.138 103.019 149.975C103.019 67.1461 169.641 0.000357483 251.823 0ZM234.696 141.93C218.117 134.066 199 146.158 199 164.507C199 174.32 204.744 183.225 213.684 187.271L336.761 242.985C337.971 243.533 338.748 244.738 338.748 246.066C338.748 247.394 337.971 248.6 336.761 249.147L213.684 304.861C204.744 308.908 199 317.812 199 327.625C199 345.974 218.117 358.066 234.696 350.202L364.672 288.554C375.142 283.588 381.815 273.036 381.815 261.448V230.684C381.815 219.096 375.142 208.544 364.672 203.578L234.696 141.93ZM413.877 296.146C402.078 296.146 392.513 305.711 392.513 317.511C392.513 329.31 402.078 338.875 413.877 338.875H526.636C538.435 338.875 548 329.31 548 317.511C548 305.711 538.435 296.146 526.636 296.146H413.877Z\\\" fill=\\\"#000B1A\\\"/>\\r\\n<path d=\\\"M2420 424.954H2343.35C2307.08 424.954 2278.64 416.094 2258.04 398.373C2237.43 380.652 2227.13 352.216 2227.13 313.065V55.2871H2319.86V302.556C2319.86 318.217 2323.15 328.726 2329.75 334.083C2336.75 339.028 2348.29 341.501 2364.36 341.501H2420V424.954ZM2420 191.903H2178.29V115.25H2420V191.903Z\\\" fill=\\\"#000B1A\\\"/>\\r\\n<path d=\\\"M1923.18 431.136C1894.33 431.136 1870.22 425.779 1850.85 415.064C1831.89 403.937 1817.68 389.101 1808.2 370.555C1799.13 352.01 1794.6 331.198 1794.6 308.12V115.25H1887.32V277.211C1887.32 301.938 1893.3 320.071 1905.25 331.61C1917.2 342.738 1937.81 348.301 1967.07 348.301C1997.56 348.301 2018.79 342.325 2030.74 330.374C2043.1 318.011 2049.28 298.641 2049.28 272.266L2061.03 271.648L2067.83 330.374H2050.52C2048.05 347.683 2041.87 363.962 2031.98 379.21C2022.5 394.458 2008.69 407.028 1990.56 416.918C1972.84 426.397 1950.38 431.136 1923.18 431.136ZM2142.01 424.955H2055.47V329.138L2049.28 326.047V115.25H2142.01V424.955Z\\\" fill=\\\"#000B1A\\\"/>\\r\\n<path d=\\\"M1734.32 424.955H1641.59V10.7793H1734.32V424.955Z\\\" fill=\\\"#000B1A\\\"/>\\r\\n<path d=\\\"M1362.98 431.136C1334.13 431.136 1310.03 425.779 1290.66 415.064C1271.7 403.937 1257.48 389.101 1248 370.555C1238.94 352.01 1234.4 331.198 1234.4 308.12V115.25H1327.13V277.211C1327.13 301.938 1333.1 320.071 1345.06 331.61C1357.01 342.738 1377.61 348.301 1406.87 348.301C1437.37 348.301 1458.59 342.325 1470.54 330.374C1482.91 318.011 1489.09 298.641 1489.09 272.266L1500.83 271.648L1507.63 330.374H1490.33C1487.85 347.683 1481.67 363.962 1471.78 379.21C1462.3 394.458 1448.5 407.028 1430.36 416.918C1412.64 426.397 1390.18 431.136 1362.98 431.136ZM1581.82 424.955H1495.27V329.138L1489.09 326.047V115.25H1581.82V424.955Z\\\" fill=\\\"#000B1A\\\"/>\\r\\n<path d=\\\"M1035.12 431.137C999.27 431.137 971.04 423.513 950.434 408.264C930.241 392.604 917.259 369.526 911.49 339.029H894.181L900.981 272.266H912.726C912.726 289.987 916.023 304.411 922.617 315.538C929.623 326.253 939.719 334.084 952.907 339.029C966.507 343.974 983.198 346.447 1002.98 346.447C1023.17 346.447 1039.66 343.974 1052.43 339.029C1065.21 334.084 1074.69 326.047 1080.87 314.92C1087.05 303.793 1090.14 288.957 1090.14 270.412C1090.14 251.042 1087.05 236 1080.87 225.285C1074.69 214.158 1065.21 206.122 1052.43 201.177C1039.66 196.231 1023.38 193.758 1003.6 193.758C973.513 193.758 950.847 199.322 935.598 210.449C920.35 221.576 912.726 240.533 912.726 267.321H900.981V197.467H917.671C923.029 171.504 935.392 150.28 954.762 133.796C974.543 117.311 1002.77 109.069 1039.45 109.069C1070.36 109.069 1096.53 115.663 1117.96 128.85C1139.39 142.038 1155.67 160.789 1166.79 185.104C1178.33 209.007 1184.1 237.443 1184.1 270.412C1184.1 302.969 1178.33 331.405 1166.79 355.72C1155.67 379.622 1138.98 398.168 1116.72 411.355C1094.47 424.543 1067.27 431.137 1035.12 431.137ZM905.926 424.955H820V10.7793H912.726V325.429L905.926 333.465V424.955Z\\\" fill=\\\"#000B1A\\\"/>\\r\\n</svg>\\r\\n\"","export default \"<svg xmlns=\\\"http://www.w3.org/2000/svg\\\" fill=\\\"none\\\" viewBox=\\\"0 0 24 24\\\" stroke-width=\\\"1.5\\\" stroke=\\\"currentColor\\\">\\r\\n <path stroke-linecap=\\\"round\\\" stroke-linejoin=\\\"round\\\" d=\\\"M16.023 9.348h4.992v-.001M2.985 19.644v-4.992m0 0h4.992m-4.993 0 3.181 3.183a8.25 8.25 0 0 0 13.803-3.7M4.031 9.865a8.25 8.25 0 0 1 13.803-3.7l3.181 3.182m0-4.991v4.99\\\" />\\r\\n</svg>\\r\\n\"","export default \"<svg xmlns=\\\"http://www.w3.org/2000/svg\\\" fill=\\\"none\\\" viewBox=\\\"0 0 24 24\\\" stroke-width=\\\"1.5\\\" stroke=\\\"currentColor\\\">\\r\\n <path stroke-linecap=\\\"round\\\" stroke-linejoin=\\\"round\\\" d=\\\"M15.042 21.672 13.684 16.6m0 0-2.51 2.225.569-9.47 5.227 7.917-3.286-.672ZM12 2.25V4.5m5.834.166-1.591 1.591M20.25 10.5H18M7.757 14.743l-1.59 1.59M6 10.5H3.75m4.007-4.243-1.59-1.59\\\" />\\r\\n</svg>\\r\\n\"","export default \"<svg xmlns=\\\"http://www.w3.org/2000/svg\\\" fill=\\\"none\\\" viewBox=\\\"0 0 24 24\\\" stroke-width=\\\"1.5\\\" stroke=\\\"currentColor\\\">\\r\\n <path stroke-linecap=\\\"round\\\" stroke-linejoin=\\\"round\\\" d=\\\"M9 6.75V15m6-6v8.25m.503 3.498 4.875-2.437c.381-.19.622-.58.622-1.006V4.82c0-.836-.88-1.38-1.628-1.006l-3.869 1.934c-.317.159-.69.159-1.006 0L9.503 3.252a1.125 1.125 0 0 0-1.006 0L3.622 5.689C3.24 5.88 3 6.27 3 6.695V19.18c0 .836.88 1.38 1.628 1.006l3.869-1.934c.317-.159.69-.159 1.006 0l4.994 2.497c.317.158.69.158 1.006 0Z\\\" />\\r\\n</svg>\\r\\n\"","export default \"<svg xmlns=\\\"http://www.w3.org/2000/svg\\\" fill=\\\"none\\\" viewBox=\\\"0 0 24 24\\\" stroke-width=\\\"1.5\\\" stroke=\\\"currentColor\\\">\\r\\n <path stroke-linecap=\\\"round\\\" stroke-linejoin=\\\"round\\\" d=\\\"M3.75 12h16.5m-16.5 3.75h16.5M3.75 19.5h16.5M5.625 4.5h12.75a1.875 1.875 0 0 1 0 3.75H5.625a1.875 1.875 0 0 1 0-3.75Z\\\" />\\r\\n</svg>\\r\\n\"","export default \"<svg xmlns=\\\"http://www.w3.org/2000/svg\\\" fill=\\\"none\\\" viewBox=\\\"0 0 24 24\\\" stroke-width=\\\"1.5\\\" stroke=\\\"currentColor\\\">\\r\\n <path stroke-linecap=\\\"round\\\" stroke-linejoin=\\\"round\\\" d=\\\"m6.75 7.5 3 2.25-3 2.25m4.5 0h3m-9 8.25h13.5A2.25 2.25 0 0 0 21 18V6a2.25 2.25 0 0 0-2.25-2.25H5.25A2.25 2.25 0 0 0 3 6v12a2.25 2.25 0 0 0 2.25 2.25Z\\\" />\\r\\n</svg>\\r\\n\"","export default \"<svg xmlns=\\\"http://www.w3.org/2000/svg\\\" fill=\\\"none\\\" viewBox=\\\"0 0 24 24\\\" stroke-width=\\\"1.5\\\" stroke=\\\"currentColor\\\">\\r\\n <path stroke-linecap=\\\"round\\\" stroke-linejoin=\\\"round\\\" d=\\\"M15.182 15.182a4.5 4.5 0 0 1-6.364 0M21 12a9 9 0 1 1-18 0 9 9 0 0 1 18 0ZM9.75 9.75c0 .414-.168.75-.375.75S9 10.164 9 9.75 9.168 9 9.375 9s.375.336.375.75Zm-.375 0h.008v.015h-.008V9.75Zm5.625 0c0 .414-.168.75-.375.75s-.375-.336-.375-.75.168-.75.375-.75.375.336.375.75Zm-.375 0h.008v.015h-.008V9.75Z\\\" />\\r\\n</svg>\\r\\n\"","import { useState, useEffect, useRef, useCallback } from \"preact/hooks\";\r\nimport \"./globals.css\";\r\nimport { render } from \"preact\";\r\nimport { ChatButton } from \"./components/ChatButton\";\r\nimport {\r\n ChatWindow,\r\n type ChatWindowHandle,\r\n} from \"./components/ChatWindow\";\r\nimport { COLORS } from \"./styles/constants\";\r\n\r\nexport type BulutVoice = \"alloy\" | \"zeynep\" | \"ali\";\r\n\r\nexport interface BulutOptions {\r\n containerId?: string;\r\n backendBaseUrl?: string;\r\n projectId?: string;\r\n}\r\n\r\nexport interface BulutRuntimeConfig {\r\n backendBaseUrl: string;\r\n projectId: string;\r\n model: string;\r\n voice: BulutVoice;\r\n baseColor: string;\r\n agentName: string;\r\n}\r\n\r\n/** Default LLM model — keep in sync with backend config.DEFAULT_LLM_MODEL */\r\nconst DEFAULT_LLM_MODEL = \"x-ai/grok-4.1-fast\";\r\n\r\nconst DEFAULT_AGENT_NAME = \"Bulut\";\r\n\r\nconst DEFAULT_CONFIG: BulutRuntimeConfig = {\r\n backendBaseUrl: \"https://api.bulut.lu\",\r\n projectId: \"\", // Must be provided\r\n model: DEFAULT_LLM_MODEL,\r\n voice: \"alloy\",\r\n baseColor: COLORS.primary,\r\n agentName: DEFAULT_AGENT_NAME,\r\n};\r\n\r\nconst isValidHexColor = (value: string): boolean =>\r\n /^#([0-9a-fA-F]{3}|[0-9a-fA-F]{6})$/.test(value);\r\n\r\nconst normalizeHexColor = (value: string): string => {\r\n const trimmed = value.trim();\r\n if (!isValidHexColor(trimmed)) {\r\n return DEFAULT_CONFIG.baseColor;\r\n }\r\n if (trimmed.length === 4) {\r\n const r = trimmed[1];\r\n const g = trimmed[2];\r\n const b = trimmed[3];\r\n return `#${r}${r}${g}${g}${b}${b}`.toLowerCase();\r\n }\r\n return trimmed.toLowerCase();\r\n};\r\n\r\nconst shadeHexColor = (hexColor: string, amount: number): string => {\r\n const normalized = normalizeHexColor(hexColor);\r\n const raw = normalized.slice(1);\r\n const toChannel = (start: number): number => parseInt(raw.slice(start, start + 2), 16);\r\n const clamp = (value: number): number => Math.max(0, Math.min(255, Math.round(value)));\r\n const adjust = (channel: number): number =>\r\n amount < 0 ? channel * (1 + amount) : channel + (255 - channel) * amount;\r\n const toHex = (channel: number): string => clamp(channel).toString(16).padStart(2, \"0\");\r\n\r\n const r = adjust(toChannel(0));\r\n const g = adjust(toChannel(2));\r\n const b = adjust(toChannel(4));\r\n\r\n return `#${toHex(r)}${toHex(g)}${toHex(b)}`;\r\n};\r\n\r\nconst applyTheme = (baseColor: string): void => {\r\n const normalized = normalizeHexColor(baseColor);\r\n COLORS.primary = normalized;\r\n COLORS.primaryHover = shadeHexColor(normalized, -0.15);\r\n COLORS.messageUser = normalized;\r\n};\r\n\r\ninterface RemoteProjectConfig {\r\n base_color: string;\r\n model: string;\r\n agent_name: string;\r\n voice: string;\r\n}\r\n\r\nconst fetchRemoteConfig = async (\r\n baseUrl: string,\r\n projectId: string,\r\n): Promise<RemoteProjectConfig | null> => {\r\n try {\r\n const url = baseUrl.replace(/\\/+$/, \"\");\r\n const res = await fetch(`${url}/projects/${projectId}/config`);\r\n if (!res.ok) return null;\r\n return (await res.json()) as RemoteProjectConfig;\r\n } catch {\r\n return null;\r\n }\r\n};\r\n\r\nconst resolveRuntimeConfig = (\r\n options: BulutOptions,\r\n): BulutRuntimeConfig => ({\r\n backendBaseUrl: options.backendBaseUrl || DEFAULT_CONFIG.backendBaseUrl,\r\n projectId: options.projectId || DEFAULT_CONFIG.projectId,\r\n model: DEFAULT_CONFIG.model,\r\n voice: DEFAULT_CONFIG.voice,\r\n baseColor: DEFAULT_CONFIG.baseColor,\r\n agentName: DEFAULT_CONFIG.agentName,\r\n});\r\n\r\ninterface BulutWidgetProps {\r\n config: BulutRuntimeConfig;\r\n}\r\n\r\nconst ACCESSIBILITY_MODE_KEY = \"bulut_accessibility_mode_enabled\";\r\nconst GEIST_FONT_FAMILY = \"Geist\";\r\nconst GEIST_STYLESHEET_ID = \"bulut-geist-font-stylesheet\";\r\nconst GEIST_STYLESHEET_URL =\r\n \"https://fonts.googleapis.com/css2?family=Geist:wght@100..900&display=swap\";\r\n\r\nconst ensureGeistStylesheet = (): void => {\r\n if (typeof document === \"undefined\") {\r\n return;\r\n }\r\n if (document.getElementById(GEIST_STYLESHEET_ID)) {\r\n return;\r\n }\r\n const link = document.createElement(\"link\");\r\n link.id = GEIST_STYLESHEET_ID;\r\n link.rel = \"stylesheet\";\r\n link.href = GEIST_STYLESHEET_URL;\r\n document.head.appendChild(link);\r\n};\r\n\r\nconst BulutWidget = ({ config }: BulutWidgetProps) => {\r\n // Live config that merges remote settings over initial config\r\n const [liveConfig, setLiveConfig] = useState<BulutRuntimeConfig>(config);\r\n const [configReady, setConfigReady] = useState(false);\r\n\r\n // Fetch remote project config on mount — widget stays hidden until done\r\n useEffect(() => {\r\n if (!config.projectId) {\r\n setConfigReady(true);\r\n return;\r\n }\r\n let cancelled = false;\r\n\r\n fetchRemoteConfig(config.backendBaseUrl, config.projectId).then((remote) => {\r\n if (cancelled) return;\r\n if (remote) {\r\n const merged: BulutRuntimeConfig = {\r\n ...config,\r\n baseColor: normalizeHexColor(remote.base_color || config.baseColor),\r\n model: remote.model || config.model,\r\n agentName: remote.agent_name || config.agentName,\r\n voice: (\r\n remote.voice === \"alloy\" || remote.voice === \"zeynep\" || remote.voice === \"ali\"\r\n ? remote.voice\r\n : config.voice\r\n ) as BulutVoice,\r\n };\r\n applyTheme(merged.baseColor);\r\n setLiveConfig(merged);\r\n }\r\n setConfigReady(true);\r\n });\r\n\r\n return () => { cancelled = true; };\r\n }, [config]);\r\n\r\n // Check localStorage for persisted state\r\n const [isOpen, setIsOpen] = useState(() => {\r\n if (typeof localStorage !== \"undefined\") {\r\n return localStorage.getItem(\"bulut_panel_open\") === \"true\";\r\n }\r\n return false;\r\n });\r\n\r\n const [showBubble, setShowBubble] = useState(false);\r\n const [isAccessibilityEnabled, setIsAccessibilityEnabled] = useState(() => {\r\n if (typeof localStorage === \"undefined\") {\r\n return false;\r\n }\r\n return localStorage.getItem(ACCESSIBILITY_MODE_KEY) === \"true\";\r\n });\r\n\r\n // State reported by ChatWindow\r\n const [isRecording, setIsRecording] = useState(false);\r\n const [isBusy, setIsBusy] = useState(false);\r\n const [previewMessage, setPreviewMessage] = useState<string | null>(null);\r\n const [previewDismissed, setPreviewDismissed] = useState(false);\r\n\r\n // Ref for delegating recording to ChatWindow\r\n const chatActionsRef = useRef<ChatWindowHandle | null>(null);\r\n\r\n const handlePreviewChange = useCallback((text: string | null) => {\r\n setPreviewMessage(text);\r\n if (text !== null) setPreviewDismissed(false);\r\n }, []);\r\n\r\n // Show welcome bubble once for 5 seconds\r\n useEffect(() => {\r\n if (isAccessibilityEnabled) {\r\n setShowBubble(false);\r\n return;\r\n }\r\n if (isOpen) return;\r\n if (typeof localStorage !== \"undefined\") {\r\n if (localStorage.getItem(\"bulut_bubble_shown\") === \"true\") return;\r\n }\r\n\r\n setShowBubble(true);\r\n const timer = setTimeout(() => {\r\n setShowBubble(false);\r\n if (typeof localStorage !== \"undefined\") {\r\n localStorage.setItem(\"bulut_bubble_shown\", \"true\");\r\n }\r\n }, 5000);\r\n return () => clearTimeout(timer);\r\n }, [isOpen, isAccessibilityEnabled]);\r\n\r\n const toggleWidget = () => {\r\n const newState = !isOpen;\r\n setIsOpen(newState);\r\n setShowBubble(false);\r\n if (typeof localStorage !== \"undefined\") {\r\n localStorage.setItem(\"bulut_panel_open\", String(newState));\r\n }\r\n };\r\n\r\n const toggleAccessibilityMode = () => {\r\n const next = !isAccessibilityEnabled;\r\n setIsAccessibilityEnabled(next);\r\n if (typeof localStorage !== \"undefined\") {\r\n localStorage.setItem(ACCESSIBILITY_MODE_KEY, String(next));\r\n }\r\n console.info(`[Bulut] accessibility mode toggled enabled=${next}`);\r\n };\r\n\r\n const handleClose = () => {\r\n setIsOpen(false);\r\n if (typeof localStorage !== \"undefined\") {\r\n localStorage.setItem(\"bulut_panel_open\", \"false\");\r\n }\r\n console.info(\"Bulut chat window closed.\");\r\n };\r\n\r\n // Close on escape key\r\n useEffect(() => {\r\n const handleEscape = (e: KeyboardEvent) => {\r\n if (e.key === \"Escape\" && isOpen) {\r\n setIsOpen(false);\r\n if (typeof localStorage !== \"undefined\") {\r\n localStorage.setItem(\"bulut_panel_open\", \"false\");\r\n }\r\n }\r\n };\r\n\r\n document.addEventListener(\"keydown\", handleEscape);\r\n return () => document.removeEventListener(\"keydown\", handleEscape);\r\n }, [isOpen]);\r\n\r\n if (!configReady) return null;\r\n\r\n return (\r\n <>\r\n {!isOpen && (\r\n <ChatButton\r\n onMicClick={() => chatActionsRef.current?.startRecording()}\r\n onCancelRecording={() => chatActionsRef.current?.cancelRecording()}\r\n onStopTask={() => chatActionsRef.current?.stopTask()}\r\n isRecording={isRecording}\r\n isBusy={isBusy}\r\n accessibilityMode={isAccessibilityEnabled}\r\n showBubble={showBubble}\r\n onBubbleClick={() => {\r\n setShowBubble(false);\r\n toggleWidget();\r\n }}\r\n previewMessage={previewDismissed ? null : previewMessage}\r\n onPreviewClick={() => toggleWidget()}\r\n onPreviewClose={() => setPreviewDismissed(true)}\r\n />\r\n )}\r\n <ChatWindow\r\n onClose={handleClose}\r\n config={liveConfig}\r\n accessibilityMode={isAccessibilityEnabled}\r\n onAccessibilityToggle={toggleAccessibilityMode}\r\n hidden={!isOpen}\r\n actionsRef={chatActionsRef}\r\n onRecordingChange={setIsRecording}\r\n onBusyChange={setIsBusy}\r\n onPreviewChange={handlePreviewChange}\r\n />\r\n </>\r\n );\r\n};\r\n\r\nconst SHADOW_STYLE = `\r\n :host {\r\n all: initial;\r\n contain: layout style paint;\r\n font-family: \"${GEIST_FONT_FAMILY}\", sans-serif;\r\n color: hsla(215, 100%, 5%, 1);\r\n font-size: 16px;\r\n line-height: 1.4;\r\n -webkit-font-smoothing: antialiased;\r\n text-rendering: optimizeLegibility;\r\n }\r\n\r\n #bulut-shadow-mount {\r\n all: initial;\r\n color: inherit;\r\n font-family: \"${GEIST_FONT_FAMILY}\", sans-serif;\r\n font-size: inherit;\r\n line-height: inherit;\r\n }\r\n\r\n #bulut-shadow-mount * {\r\n font-family: \"${GEIST_FONT_FAMILY}\", sans-serif !important;\r\n color: inherit;\r\n }\r\n\r\n #bulut-shadow-mount *, #bulut-shadow-mount *::before, #bulut-shadow-mount *::after {\r\n box-sizing: border-box;\r\n }\r\n`;\r\n\r\n// Container host for the widget\r\nlet widgetContainer: HTMLElement | null = null;\r\nlet widgetMountNode: HTMLElement | null = null;\r\nlet createdContainer = false;\r\nlet isInitialized = false;\r\n\r\n/**\r\n * Initialize the Bulut chat widget\r\n * @param options - Optional configuration options\r\n */\r\nexport const init = (options: BulutOptions = {}) => {\r\n if (isInitialized) {\r\n console.warn(\"Bulut is already initialized\");\r\n return;\r\n }\r\n\r\n ensureGeistStylesheet();\r\n\r\n const runtimeConfig = resolveRuntimeConfig(options);\r\n applyTheme(runtimeConfig.baseColor);\r\n\r\n // Create or find container\r\n if (options.containerId) {\r\n widgetContainer = document.getElementById(options.containerId);\r\n createdContainer = false;\r\n } else {\r\n widgetContainer = document.createElement(\"div\");\r\n widgetContainer.id = \"bulut-container\";\r\n document.body.appendChild(widgetContainer);\r\n createdContainer = true;\r\n }\r\n\r\n if (!widgetContainer) {\r\n console.error(\"Bulut: Container not found\");\r\n return;\r\n }\r\n\r\n const shadowRoot = widgetContainer.shadowRoot || widgetContainer.attachShadow({ mode: \"open\" });\r\n shadowRoot.replaceChildren();\r\n\r\n const style = document.createElement(\"style\");\r\n style.textContent = SHADOW_STYLE;\r\n\r\n const mountNode = document.createElement(\"div\");\r\n mountNode.id = \"bulut-shadow-mount\";\r\n\r\n shadowRoot.append(style, mountNode);\r\n widgetMountNode = mountNode;\r\n\r\n // Render the widget\r\n render(<BulutWidget config={runtimeConfig} />, mountNode);\r\n isInitialized = true;\r\n\r\n console.log(\"Bulut initialized successfully\");\r\n};\r\n\r\n/**\r\n * Destroy the Bulut widget\r\n */\r\nexport const destroy = () => {\r\n if (!isInitialized) {\r\n return;\r\n }\r\n\r\n if (widgetMountNode) {\r\n render(null, widgetMountNode);\r\n widgetMountNode = null;\r\n }\r\n\r\n if (widgetContainer && createdContainer) {\r\n document.body.removeChild(widgetContainer);\r\n }\r\n\r\n widgetContainer = null;\r\n createdContainer = false;\r\n isInitialized = false;\r\n console.log(\"Bulut destroyed\");\r\n};\r\n\r\n/**\r\n * Check if the widget is initialized\r\n */\r\nexport const isReady = () => isInitialized;\r\n\r\nconst Bulut = {\r\n init,\r\n destroy,\r\n isReady,\r\n};\r\n\r\nif (typeof window !== \"undefined\") {\r\n (window as Window & { Bulut?: typeof Bulut }).Bulut = Bulut;\r\n}\r\n\r\n// Export the main widget component for advanced usage\r\nexport { BulutWidget };\r\n\r\n// Export components for custom implementations\r\nexport { ChatButton, ChatWindow };\r\n\r\nexport default Bulut;\r\n"],"names":["n","l","u","i","o","r","e","f","c","s","a","p","v","y","d","Array","isArray","w","g","parentNode","removeChild","m","t","type","props","key","ref","__k","__","__b","__e","__c","constructor","__v","__i","__u","vnode","k","children","x","this","context","S","length","C","base","M","__d","push","$","__r","debounceRendering","sort","shift","__P","O","__n","namespaceURI","N","I","h","_","b","String","L","D","P","B","A","nextSibling","insertBefore","nodeType","T","setProperty","test","j","style","cssText","replace","toLowerCase","slice","addEventListener","removeEventListener","removeAttribute","setAttribute","F","event","H","prototype","render","contextType","value","__E","E","sub","state","__h","_sb","__s","getDerivedStateFromProps","componentWillMount","componentDidMount","componentWillReceiveProps","shouldComponentUpdate","some","componentWillUpdate","componentDidUpdate","getChildContext","getSnapshotBeforeUpdate","V","then","indexOf","z","localName","document","createTextNode","createElementNS","is","__m","data","call","childNodes","attributes","name","__html","innerHTML","content","q","diffed","forEach","map","current","unmount","componentWillUnmount","G","documentElement","arguments","defaultProps","firstChild","getDerivedStateFromError","setState","componentDidCatch","forceUpdate","Promise","bind","resolve","setTimeout","__source","__self","__H","__N","__f","filter","every","requestAnimationFrame","clearTimeout","cancelAnimationFrame","COLORS","primary","primaryHover","text","messageUser","messageUserText","BORDER_RADIUS","SHADOW","TRANSITIONS","handRaisedIconContent","microphoneOutlineIconContent","stopOutlineIconContent","xMarkIconContent","SvgIcon","src","width","height","title","stripColors","className","fill","stroke","strokeWidth","viewBox","useMemo","trimmedSrc","trim","viewBoxMatch","match","jsx","display","verticalAlign","flexShrink","xmlns","dangerouslySetInnerHTML","nextContent","fillValue","strokeValue","strokeWidthValue","ChatButton","onMicClick","onCancelRecording","onStopTask","isRecording","isBusy","accessibilityMode","showBubble","onBubbleClick","previewMessage","onPreviewClick","onPreviewClose","bgColor","popupBoxShadow","containerStyle","position","right","bottom","flexDirection","alignItems","gap","zIndex","fontFamily","buttonStyle","minWidth","minHeight","borderRadius","backgroundColor","color","border","cursor","justifyContent","transition","showStopButton","mainIconSrc","closeBtnStyle","top","background","fontSize","lineHeight","padding","renderPopup","onClick","onClose","extraClass","scrollable","jsxs","stopPropagation","paddingRight","wordBreak","maxHeight","overflowY","Fragment","onMouseEnter","Object","assign","currentTarget","transform","onMouseLeave","PAGE_CONTEXT_CACHE_KEY","NON_CONTENT_TAGS","Set","NATIVE_INTERACTIVE_TAGS","INTERACTIVE_ROLES","TRACKED_DISPLAY_VALUES","TRACKED_POSITION_VALUES","EVENT_HINT_NAMES","ARIA_INTERACTION_ATTRS","DATA_INTERACTION_PATTERN","STYLESHEET_SELECTOR_PATTERN","pageContextCache","Map","cacheHydrated","normalizeWhitespace","isCacheEntry","obj","url","summary","links","interactables","capturedAt","version","bumpCount","set","get","formatTopCounts","maxItems","size","from","entries","localeCompare","count","join","parseTabIndex","parsed","Number","parseInt","isNaN","compactToken","getElementDepth","element","depth","parentElement","body","getPrimaryRole","getAttribute","split","buildSummaryWithHistory","recentPages","values","entry","historySection","compactSummary","isVisible","HTMLElement","hidden","window","getComputedStyle","visibility","rect","getBoundingClientRect","toAbsoluteUrl","href","URL","location","escapeCssValue","CSS","escape","buildSelectorSegment","tag","tagName","id","attrCandidates","attrName","attrValue","classes","classList","Boolean","parent","sibling","buildSelector","segments","segment","unshift","startsWith","getElementLabel","innerText","textContent","ariaLabel","placeholder","HTMLInputElement","HTMLTextAreaElement","HTMLButtonElement","classHint","item","find","fallback","label","getEventHints","record","eventHints","eventName","handlerKey","hasInlineHandler","hasPropertyHandler","getAriaInteractionHints","hasAttribute","getDataInteractionHints","getAttributeNames","getComputedStyleSignals","signals","pointerEvents","opacity","buildBranchDigest","token","idToken","classToken","buildBlueprintToken","child","has","sampled","overflow","formatSection","lines","collectTextSnippets","root","querySelector","snippets","seen","candidates","querySelectorAll","node","add","collectSelectorsFromRuleList","rules","selectors","scanned","rule","CSSStyleRule","parts","selectorText","selector","nestedRules","cssRules","collectStylesheetSelectors","stylesheet","styleSheets","collectPageSignalSnapshot","allElements","sampledElements","linkSet","interactableCandidates","tagCounts","roleCounts","eventCounts","displayCounts","positionCounts","styleSignalCounts","visibleElements","maxDepth","semanticInteractables","nonSemanticInteractables","tabStopElements","pointerCursorElements","dataHintElements","ariaHintElements","contentEditableElements","order","role","computedStyleSignals","styleSignal","tabIndex","hasTabStop","hasPointerCursor","dataHints","ariaHints","isContentEditable","isNativeInteractive","isRoleInteractive","isDisabled","absoluteHref","line","styleSignals","signalTokens","score","existing","candidate","interactiveRoleCounts","interactionSignals","branchDigest","collectDomBranchDigest","pageBlueprint","styleSelectors","getPageContext","sessionStorage","raw","getItem","JSON","parse","console","info","error","warn","hydrateCacheFromStorage","rawUrl","canonicalUrl","cached","snapshot","headings","input","lang","landmarks","textSnippets","outerHtmlDigest","outerHTML","Date","now","sorted","delete","pruneOldestCacheEntries","serialized","stringify","setItem","persistCacheToStorage","AGENT_CURSOR_ID","CURSOR_MOVE_DURATION_MS","CURSOR_EASING","RESUME_STORAGE_KEY","savePendingAgentResume","localStorage","savedAt","clearPendingAgentResume","removeItem","clamp","min","max","Math","animateWindowScrollTo","async","targetY","durationMs","startY","scrollY","delta","abs","raf","callback","performance","startTime","step","progress","eased","cos","PI","easeInOutSine","scrollTo","setCursorPosition","left","setCursorVisibility","visible","isVisibleElement","findAgentUiAnchorElement","roots","defaultHost","getElementById","shadowRoot","el","includes","getBulutShadowRoots","panel","button","getAgentWindowTopLeft","anchor","CURSOR_DIAMETER_PX","scrollX","hideAgentCursor","cursorHoverTrackingInitialized","initializeCursorHoverTracking","parseFloat","getCursorPosition","pointerX","pageX","pointerY","pageY","hypot","moveCursor","createElement","startPosition","baseColor","boxShadow","boxSizing","appendChild","ensureCursor","dataset","transitionReady","waitForNextAnimationFrame","getElementCenter","CONTAINS_SELECTOR_PATTERN","findElementBySelector","containsMatch","baseSelector","expectedText","fallbackError","dispatchMouseEvent","dispatchEvent","MouseEvent","bubbles","cancelable","view","clientX","clientY","typeIntoElement","focus","descriptor","getOwnPropertyDescriptor","defaultValue","setNativeInputLikeValue","Event","slowScrollElementIntoView","slowScrollElementIntoViewWithMode","forceCenter","viewportHeight","innerHeight","isRectOutsideViewport","maxScrollTop","scrollHeight","currentScrollY","rectTop","rectHeight","computeCenteredScrollTop","executeInteract","target","selected","center","resolveTarget","action","click","requestSubmit","form","parentForm","closest","submitElement","executeNavigate","targetUrl","resolvedUrl","matchingElement","parsedTarget","allLinks","HTMLAnchorElement","elUrl","pathname","search","hash","urlSegments","lastSegment","searchTerms","searchParams","clickables","dataTab","term","findMatchingLinkForTarget","log","origin","isSamePageNavigation","newPath","history","pushState","PopStateEvent","executeSingleToolCall","callId","call_id","tool","result","executeScroll","msg","Error","message","TTS_WS_RETRY_DELAYS_MS","BULUT_AUDIO_STOP_EVENT","activeAudioElements","audioPlaybackGeneration","normalizeBaseUrl","baseUrl","trimmed","toWebSocketUrl","path","normalized","protocol","toString","shouldFallbackToSse","retryable","parseErrorBody","response","json","detail","statusText","sleep","ms","wasPlaybackStoppedAfter","generationAtStart","getAudioPlaybackGeneration","stopActiveAudioPlayback","active","audioElement","pause","load","base64ToUint8Array","base64","cleanBase64","binaryString","atob","bytes","Uint8Array","charCodeAt","playBufferedAudio","chunks","mimeType","sampleRate","onAudioStateChange","playbackGeneration","totalBytes","reduce","acc","byteLength","blobParts","chunk","copied","buffer","detectedMime","header","padStart","toUpperCase","safeMimeType","finalBlobParts","ArrayBuffer","DataView","setUint32","setUint16","channels","createWavHeader","blob","Blob","Audio","objectUrl","createObjectURL","registerActiveAudioElement","preload","autoplay","play","ended","reject","watchdog","setInterval","onEnded","cleanup","onError","onForcedStop","clearInterval","waitForPlaybackEnd","err","unregisterActiveAudioElement","revokeObjectURL","parseSseEventPayload","eventBlock","dataLines","trimStart","dataStr","isAudioSsePayload","payload","audio","startSttWebSocketStream","config","events","wsUrl","socket","WebSocket","seq","finalText","finalSessionId","sessionId","stopped","settled","sendQueue","resolveStart","rejectStart","startPromise","resolveDone","rejectDone","donePromise","rejectAll","resolveDoneIfPossible","session_id","onopen","onRequestSent","send","project_id","projectId","language","mime_type","onmessage","parseSttWsEventPayload","onSessionId","close","buildError","onPartial","onerror","ev","onclose","code","reason","pushChunk","arrayBuffer","binary","fromCharCode","subarray","btoa","blobToBase64","readyState","OPEN","debug","stop","cancel","collectTtsViaSse","assistantText","voice","isStopped","setReader","ttsFormData","FormData","append","ttsResponse","fetch","method","ok","reader","getReader","decoder","TextDecoder","done","read","decode","stream","blocks","pop","block","format","sample_rate","releaseLock","collectTtsViaWebSocket","setSocket","requestId","crypto","randomUUID","random","highestSeqSeen","connectOnce","finalError","finalize","mode","request_id","accessibility_mode","last_seq","parseTtsWsEventPayload","incomingSeq","shouldAcceptAudioSeq","attempt","delay","speakText","ttsResult","neverStopped","agentVoiceChatStream","audioFile","executeTool","activeReader","activeSocket","errorEmitted","sttResult","file","formData","responsePromise","transcribeAudio","onSttRequestSent","currentSessionId","effectiveSessionId","userText","onTranscription","user_text","agentResolve","agentReject","finalReply","resolved","accumulatedDelta","finish","reply","fail","model","page_context","pageContext","msgType","onAssistantDelta","calls","onIntermediateReply","onToolCalls","results","isNavigate","pendingToolCalls","args","completedResults","onToolResult","final_reply","replyExpectsReply","expects_reply","onAssistantDone","errMsg","onIteration","iteration","max_iterations","wsError","catch","SFX_SOURCES","sent","thinking","toolCall","completed","sfxManager","queue","playNow","onended","drain","next","playCue","STORAGE_KEY","TIMESTAMP_KEY","SESSION_ID_KEY","TTL_MS","STATUS_LABELS","getGreetingText","agentName","resolveStatusText","flags","isRunningTools","isPlayingAudio","isThinking","isTranscribing","formatDurationMs","totalSeconds","floor","createInitialMessages","isUser","clearPersistedChatState","storage","getNextMessageId","messages","shouldAutoListenAfterAudio","expectsReply","getToolIndicatorMessage","kind","ChatWindow","onAccessibilityToggle","actionsRef","onRecordingChange","onBusyChange","onPreviewChange","setMessages","useState","saved","timestamp","setIsBusy","setIsRecording","setIsTranscribing","setIsThinking","isRenderingAudio","setIsRenderingAudio","setIsPlayingAudio","setIsRunningTools","isMicPending","setIsMicPending","recordingDurationMs","setRecordingDurationMs","statusOverride","setStatusOverride","statusFlags","resolvedStatusText","showStatus","hasActiveStatus","isBusyRef","useRef","isRecordingRef","nextMessageIdRef","recorderRef","streamRef","audioChunksRef","activeStreamControllerRef","sessionIdRef","ts","silenceStartRef","vadIntervalRef","audioContextRef","sourceRef","discardNextRecordingRef","micPressStartRef","micHoldTimeoutRef","micHoldTriggeredRef","recordingModeRef","pendingStopAfterStartRef","startRecordingPendingRef","assistantMessageIdRef","assistantTextBufferRef","transcriptionReceivedRef","assistantDoneReceivedRef","recordingStartedAtRef","recordingTimerIntervalRef","messagesContainerRef","messagesContentRef","pendingUserTextRef","pendingAssistantTextRef","streamingJsonParserRef","awaitingAssistantResponseRef","activeSttWsRef","liveTranscriptionMessageIdRef","liveTranscriptionTextRef","autoListenSuppressedRef","expectsReplyRef","requestEpochRef","sttSendCuePlayedRef","useEffect","lastAssistant","reverse","st","playSfx","beginRequestEpoch","invalidateRequestEpoch","isCurrentRequestEpoch","epoch","playSttSentCueOnce","scrollMessagesToBottom","scrollTop","ResizeObserver","observer","observe","disconnect","stopRecordingTimer","resetProcessingFlags","clearMicHoldTimeout","cleanupVAD","stopStreamTracks","getTracks","track","stopActiveStream","cancelActiveSttWs","activeSttWs","recorder","ondataavailable","onstop","resumeState","getPendingAgentResume","requestEpoch","freshPageContext","controller","allResults","tc","pending_tool_calls","tool_results","agentResumeStream","backendBaseUrl","sid","appendMessage","updateMessageText","finalDisplayText","handleAudioStateChange","appendToolIndicatorMessages","finally","startRecording","options","previous","toolKind","toolLabel","toolCount","indicator","last","previousToolText","extractedCount","nextCount","isFinite","baseLabel","upsertLiveUserTranscription","clearLiveUserTranscriptionState","finalizeStreamCycle","runAgentForUserText","normalizedUserText","handleToolExecution","agentTextChatStream","stopRecording","discard","navigator","mediaDevices","getUserMedia","MediaRecorder","recorderOptions","audioBitsPerSecond","preferredMimeTypes","mime","isTypeSupported","sttMimeType","sttWsController","shouldDiscard","currentSttWs","fileType","extension","File","previousUserText","_callId","_toolName","_result","_iteration","_maxIterations","handleAudioBlob","AudioCtx","AudioContext","webkitAudioContext","analyser","createAnalyser","fftSize","source","createMediaStreamSource","connect","dataArray","frequencyBinCount","speechDetected","speechStartedAt","enforceMinSpeechDuration","getByteFrequencyData","sum","silenceDuration","speechDurationMs","minSpeechDurationMs","shouldAcceptVadSpeech","setupVAD","start","startedAt","normalizeError","resetMicGesture","handleMicPointerUp","preventDefault","releasePointerCapture","pointerId","wasHold","thresholdMs","classifyMicGesture","timer","stopTask","cancelRecording","windowStyle","animation","headerButtonStyle","messageStyle","maxWidth","wordWrap","whiteSpace","alignSelf","statusPanelStyle","flex","textOverflow","recordingTimerStyle","fontWeight","textAlign","micFooterButtonStyle","isVadRecording","hideMicButton","disableMicControl","initialMessages","toolIconSrc","onPointerDown","setPointerCapture","onPointerUp","onPointerCancel","disabled","DEFAULT_CONFIG","normalizeHexColor","isValidHexColor","shadeHexColor","hexColor","amount","toChannel","adjust","channel","toHex","round","applyTheme","ACCESSIBILITY_MODE_KEY","GEIST_FONT_FAMILY","GEIST_STYLESHEET_ID","BulutWidget","liveConfig","setLiveConfig","configReady","setConfigReady","cancelled","res","fetchRemoteConfig","remote","merged","base_color","agent_name","isOpen","setIsOpen","setShowBubble","isAccessibilityEnabled","setIsAccessibilityEnabled","setPreviewMessage","previewDismissed","setPreviewDismissed","chatActionsRef","handlePreviewChange","toggleWidget","newState","handleEscape","SHADOW_STYLE","widgetContainer","widgetMountNode","createdContainer","isInitialized","init","link","rel","head","ensureGeistStylesheet","runtimeConfig","resolveRuntimeConfig","containerId","attachShadow","replaceChildren","mountNode","destroy","isReady","Bulut"],"mappings":"AAAG,IAACA,EAAEC,EAAEC,EAAIC,EAAEC,EAAEC,EAAEC,EAAEC,EAAEC,EAAEC,EAAEC,EAAIC,EAAE,CAAA,EAAGC,EAAE,GAAGC,EAAE,oEAAoEC,EAAEC,MAAMC,QAAQ,SAASC,EAAEjB,EAAEC,GAAG,IAAA,IAAQC,KAAKD,EAAED,EAAEE,GAAGD,EAAEC,GAAG,OAAOF,CAAC,CAAC,SAASkB,EAAElB,GAAGA,GAAGA,EAAEmB,YAAYnB,EAAEmB,WAAWC,YAAYpB,EAAE,CAAsS,SAASqB,EAAErB,EAAEsB,EAAEnB,EAAEC,EAAEC,GAAG,IAAIC,EAAE,CAACiB,KAAKvB,EAAEwB,MAAMF,EAAEG,IAAItB,EAAEuB,IAAItB,EAAEuB,IAAI,KAAKC,GAAG,KAAKC,IAAI,EAAEC,IAAI,KAAKC,IAAI,KAAKC,mBAAmBC,IAAI,MAAM5B,IAAIH,EAAEG,EAAE6B,KAAI,EAAGC,IAAI,GAAG,OAAO,MAAM9B,GAAG,MAAMJ,EAAEmC,OAAOnC,EAAEmC,MAAM9B,GAAGA,CAAC,CAAmC,SAAS+B,EAAErC,GAAG,OAAOA,EAAEsC,QAAQ,CAAC,SAASC,EAAEvC,EAAEC,GAAGuC,KAAKhB,MAAMxB,EAAEwC,KAAKC,QAAQxC,CAAC,CAAC,SAASyC,EAAE1C,EAAEC,GAAG,GAAG,MAAMA,EAAE,OAAOD,EAAE4B,GAAGc,EAAE1C,EAAE4B,GAAG5B,EAAEkC,IAAI,GAAG,KAAK,IAAA,IAAQhC,EAAED,EAAED,EAAE2B,IAAIgB,OAAO1C,IAAI,GAAG,OAAOC,EAAEF,EAAE2B,IAAI1B,KAAK,MAAMC,EAAE4B,WAAW5B,EAAE4B,IAAI,MAAM,mBAAmB9B,EAAEuB,KAAKmB,EAAE1C,GAAG,IAAI,CAAC,SAAS4C,EAAE5C,GAAG,IAAIC,EAAEC,EAAE,GAAG,OAAOF,EAAEA,EAAE4B,KAAK,MAAM5B,EAAE+B,IAAI,CAAC,IAAI/B,EAAE8B,IAAI9B,EAAE+B,IAAIc,KAAK,KAAK5C,EAAE,EAAEA,EAAED,EAAE2B,IAAIgB,OAAO1C,IAAI,GAAG,OAAOC,EAAEF,EAAE2B,IAAI1B,KAAK,MAAMC,EAAE4B,IAAI,CAAC9B,EAAE8B,IAAI9B,EAAE+B,IAAIc,KAAK3C,EAAE4B,IAAI,KAAK,CAAC,OAAOc,EAAE5C,EAAE,CAAC,CAAC,SAAS8C,EAAE9C,KAAKA,EAAE+C,MAAM/C,EAAE+C,KAAI,IAAK5C,EAAE6C,KAAKhD,KAAKiD,EAAEC,OAAO9C,GAAGH,EAAEkD,sBAAsB/C,EAAEH,EAAEkD,oBAAoB9C,GAAG4C,EAAE,CAAC,SAASA,IAAI,IAAA,IAAQjD,EAAEE,EAAEoB,EAAElB,EAAEC,EAAEE,EAAEC,EAAEC,EAAE,EAAEN,EAAEwC,QAAQxC,EAAEwC,OAAOlC,GAAGN,EAAEiD,KAAK9C,GAAGN,EAAEG,EAAEkD,QAAQ5C,EAAEN,EAAEwC,OAAO3C,EAAE+C,MAAMzB,SAASlB,OAAE,EAAOC,GAAGD,GAAGF,EAAEF,GAAGiC,KAAKH,IAAIvB,EAAE,GAAGC,EAAE,GAAGN,EAAEoD,OAAOhC,EAAEL,EAAE,GAAGb,IAAI6B,IAAI7B,EAAE6B,IAAI,EAAEhC,EAAEmC,OAAOnC,EAAEmC,MAAMd,GAAGiC,EAAErD,EAAEoD,IAAIhC,EAAElB,EAAEF,EAAEsD,IAAItD,EAAEoD,IAAIG,aAAa,GAAGrD,EAAE+B,IAAI,CAAC9B,GAAG,KAAKE,EAAE,MAAMF,EAAEqC,EAAEtC,GAAGC,KAAK,GAAGD,EAAE+B,KAAK3B,GAAGc,EAAEW,IAAI7B,EAAE6B,IAAIX,EAAEM,GAAGD,IAAIL,EAAEY,KAAKZ,EAAEoC,EAAEnD,EAAEe,EAAEd,GAAGJ,EAAE0B,IAAI1B,EAAEwB,GAAG,KAAKN,EAAEQ,KAAKzB,GAAGuC,EAAEtB,KAAK2B,EAAEC,IAAI,CAAC,CAAC,SAASS,EAAE3D,EAAEC,EAAEC,EAAEoB,EAAEnB,EAAEC,EAAEC,EAAEC,EAAEC,EAAEC,EAAEC,GAAG,IAAIC,EAAEkD,EAAE/C,EAAEC,EAAEG,EAAEC,EAAE2C,EAAExC,EAAEC,GAAGA,EAAEK,KAAKf,EAAEkD,EAAE7D,EAAE0C,OAAO,IAAIpC,EAAyV,SAAWP,EAAEC,EAAEC,EAAEoB,EAAEnB,GAAG,IAAIC,EAAEC,EAAEC,EAAEC,EAAEC,EAAEC,EAAEP,EAAEyC,OAAOjC,EAAED,EAAEmD,EAAE,EAAE,IAAI5D,EAAE2B,IAAI,IAAIZ,MAAMZ,GAAGC,EAAE,EAAEA,EAAED,EAAEC,IAAI,OAAOC,EAAEJ,EAAEG,KAAK,kBAAkBC,GAAG,mBAAmBA,GAAG,iBAAiBA,GAAG,iBAAiBA,GAAG,iBAAiBA,GAAGA,EAAE2B,aAAa+B,OAAO1D,EAAEL,EAAE2B,IAAIvB,GAAGiB,EAAE,KAAKhB,EAAE,KAAK,KAAK,MAAMS,EAAET,GAAGA,EAAEL,EAAE2B,IAAIvB,GAAGiB,EAAEgB,EAAE,CAACC,SAASjC,GAAG,KAAK,KAAK,WAAM,IAASA,EAAE2B,aAAa3B,EAAEwB,IAAI,EAAExB,EAAEL,EAAE2B,IAAIvB,GAAGiB,EAAEhB,EAAEkB,KAAKlB,EAAEmB,MAAMnB,EAAEoB,IAAIpB,EAAEqB,IAAIrB,EAAEqB,IAAI,KAAKrB,EAAE4B,KAAKjC,EAAE2B,IAAIvB,GAAGC,EAAEE,EAAEH,EAAEwD,EAAEvD,EAAEuB,GAAG5B,EAAEK,EAAEwB,IAAI7B,EAAE6B,IAAI,EAAEvB,EAAE,MAAK,IAAKE,EAAEH,EAAE6B,IAAI8B,EAAE3D,EAAEH,EAAEK,EAAEG,MAAMA,KAAKJ,EAAEJ,EAAEM,MAAMF,EAAE6B,KAAK,IAAI,MAAM7B,GAAG,MAAMA,EAAE2B,MAAK,GAAIzB,IAAIL,EAAEM,EAAEmD,IAAIzD,EAAEM,GAAGmD,KAAK,mBAAmBvD,EAAEkB,OAAOlB,EAAE8B,KAAK,IAAI3B,GAAGD,IAAIC,GAAGD,EAAE,EAAEqD,IAAIpD,GAAGD,EAAE,EAAEqD,KAAKpD,EAAED,EAAEqD,IAAIA,IAAIvD,EAAE8B,KAAK,KAAKnC,EAAE2B,IAAIvB,GAAG,KAAK,GAAGM,EAAE,IAAIN,EAAE,EAAEA,EAAEK,EAAEL,IAAI,OAAOE,EAAEJ,EAAEE,OAAS,EAAEE,EAAE6B,OAAO7B,EAAEwB,KAAKR,IAAIA,EAAEoB,EAAEpC,IAAI2D,EAAE3D,EAAEA,IAAI,OAAOgB,CAAC,CAAjkC4C,CAAEhE,EAAED,EAAEoB,EAAEd,EAAEuD,GAAGpD,EAAE,EAAEA,EAAEoD,EAAEpD,IAAI,OAAOG,EAAEX,EAAEyB,IAAIjB,MAAMkD,GAAE,GAAI/C,EAAEqB,IAAIvB,EAAEU,EAAER,EAAEqB,MAAMvB,EAAEE,EAAEqB,IAAIxB,EAAEQ,EAAEqC,EAAEvD,EAAEa,EAAE+C,EAAEzD,EAAEC,EAAEC,EAAEC,EAAEC,EAAEC,EAAEC,GAAGK,EAAED,EAAEiB,IAAIjB,EAAEa,KAAKkC,EAAElC,KAAKb,EAAEa,MAAMkC,EAAElC,KAAKyC,EAAEP,EAAElC,IAAI,KAAKb,GAAGJ,EAAEuC,KAAKnC,EAAEa,IAAIb,EAAEkB,KAAKjB,EAAED,IAAI,MAAMI,GAAG,MAAMH,IAAIG,EAAEH,IAAI+C,KAAK,EAAEhD,EAAEsB,OAAOyB,EAAEjC,MAAMd,EAAEc,IAAIpB,EAAE6D,EAAEvD,EAAEN,EAAEP,EAAE6D,GAAG,mBAAmBhD,EAAEU,WAAM,IAASL,EAAEX,EAAEW,EAAEJ,IAAIP,EAAEO,EAAEuD,aAAaxD,EAAEsB,MAAK,GAAI,OAAOjC,EAAE4B,IAAIb,EAAEV,CAAC,CAA4uB,SAAS6D,EAAEpE,EAAEC,EAAEC,EAAEoB,GAAG,IAAInB,EAAEC,EAAE,GAAG,mBAAmBJ,EAAEuB,KAAK,CAAC,IAAIpB,EAAEH,EAAE2B,IAAIvB,EAAE,EAAED,GAAGC,EAAED,EAAEwC,OAAOvC,IAAID,EAAEC,KAAKD,EAAEC,GAAGwB,GAAG5B,EAAEC,EAAEmE,EAAEjE,EAAEC,GAAGH,EAAEC,EAAEoB,IAAI,OAAOrB,CAAC,CAACD,EAAE8B,KAAK7B,IAAIqB,IAAIrB,GAAGD,EAAEuB,OAAOtB,EAAEkB,aAAalB,EAAEyC,EAAE1C,IAAIE,EAAEoE,aAAatE,EAAE8B,IAAI7B,GAAG,OAAOA,EAAED,EAAE8B,KAAK,GAAG7B,EAAEA,GAAGA,EAAEoE,kBAAkB,MAAMpE,GAAG,GAAGA,EAAEsE,UAAU,OAAOtE,CAAC,CAA6G,SAAS+D,EAAEhE,EAAEC,EAAEC,EAAEoB,GAAG,IAAInB,EAAEC,EAAEC,EAAEC,EAAEN,EAAEyB,IAAIlB,EAAEP,EAAEuB,KAAKf,EAAEP,EAAEC,GAAGO,EAAE,MAAMD,KAAO,EAAEA,EAAE2B,KAAK,GAAG,OAAO3B,GAAG,MAAMF,GAAGG,GAAGH,GAAGE,EAAEiB,KAAKlB,GAAGC,EAAEe,KAAK,OAAOrB,EAAE,GAAGoB,GAAGb,EAAE,EAAE,GAAG,IAAIN,EAAED,EAAE,EAAEE,EAAEF,EAAE,EAAEC,GAAG,GAAGC,EAAEH,EAAE0C,QAAQ,GAAG,OAAOnC,EAAEP,EAAEI,EAAEF,GAAG,EAAEA,IAAIC,SAAW,EAAEI,EAAE2B,MAAM7B,GAAGE,EAAEiB,KAAKlB,GAAGC,EAAEe,KAAK,OAAOlB,EAAE,QAAQ,CAAC,SAASmE,EAAExE,EAAEC,EAAEC,GAAG,KAAKD,EAAE,GAAGD,EAAEyE,YAAYxE,EAAE,MAAMC,EAAE,GAAGA,GAAGF,EAAEC,GAAG,MAAMC,EAAE,GAAG,iBAAiBA,GAAGW,EAAE6D,KAAKzE,GAAGC,EAAEA,EAAE,IAAI,CAAC,SAASyE,EAAE3E,EAAEC,EAAEC,EAAEoB,EAAEnB,GAAG,IAAIC,EAAEC,EAAEL,EAAE,GAAG,SAASC,EAAE,GAAG,iBAAiBC,EAAEF,EAAE4E,MAAMC,QAAQ3E,MAAM,CAAC,GAAG,iBAAiBoB,IAAItB,EAAE4E,MAAMC,QAAQvD,EAAE,IAAIA,EAAE,IAAIrB,KAAKqB,EAAEpB,GAAGD,KAAKC,GAAGsE,EAAExE,EAAE4E,MAAM3E,EAAE,IAAI,GAAGC,EAAE,IAAID,KAAKC,EAAEoB,GAAGpB,EAAED,IAAIqB,EAAErB,IAAIuE,EAAExE,EAAE4E,MAAM3E,EAAEC,EAAED,GAAG,MAAA,GAAS,KAAKA,EAAE,IAAI,KAAKA,EAAE,GAAGG,EAAEH,IAAIA,EAAEA,EAAE6E,QAAQvE,EAAE,OAAOF,EAAEJ,EAAE8E,cAAc9E,EAAEI,KAAKL,GAAG,cAAcC,GAAG,aAAaA,EAAEI,EAAE2E,MAAM,GAAG/E,EAAE+E,MAAM,GAAGhF,EAAEC,IAAID,EAAEC,EAAE,IAAID,EAAEC,EAAEA,EAAEG,GAAGF,EAAEA,EAAEoB,EAAEpB,EAAEA,EAAEoB,EAAEpB,GAAGA,EAAEA,EAAEM,EAAER,EAAEiF,iBAAiBhF,EAAEG,EAAEM,EAAED,EAAEL,IAAIJ,EAAEkF,oBAAoBjF,EAAEG,EAAEM,EAAED,EAAEL,OAAO,CAAC,GAAG,8BAA8BD,EAAEF,EAAEA,EAAE6E,QAAQ,cAAc,KAAKA,QAAQ,SAAS,UAAG,GAAU,SAAS7E,GAAG,UAAUA,GAAG,QAAQA,GAAG,QAAQA,GAAG,QAAQA,GAAG,YAAYA,GAAG,YAAYA,GAAG,WAAWA,GAAG,WAAWA,GAAG,QAAQA,GAAG,WAAWA,GAAGA,KAAKD,EAAE,IAAIA,EAAEC,GAAG,MAAMC,EAAE,GAAGA,EAAE,MAAMF,CAAC,OAAOA,GAAG,CAAC,mBAAmBE,IAAI,MAAMA,IAAG,IAAKA,GAAG,KAAKD,EAAE,GAAGD,EAAEmF,gBAAgBlF,GAAGD,EAAEoF,aAAanF,EAAE,WAAWA,GAAG,GAAGC,EAAE,GAAGA,GAAG,CAAC,CAAC,SAASmF,EAAErF,GAAG,OAAO,SAASE,GAAG,GAAGsC,KAAKvC,EAAE,CAAC,IAAIqB,EAAEkB,KAAKvC,EAAEC,EAAEqB,KAAKvB,GAAG,GAAG,MAAME,EAAEoB,EAAEpB,EAAEoB,EAAEd,SAAAA,GAAYN,EAAEoB,EAAEA,EAAEpB,EAAE,OAAO,OAAOoB,EAAErB,EAAEqF,MAAMrF,EAAEqF,MAAMpF,GAAGA,EAAE,CAAC,CAAC,CAAC,SAASqD,EAAEvD,EAAEE,EAAEoB,EAAEnB,EAAEC,EAAEC,EAAEC,EAAEC,EAAEC,EAAEC,GAAG,IAAIC,EAAEkD,EAAEjD,EAAEC,EAAEC,EAAEgD,EAAExC,EAAEyC,EAAEpB,EAAEE,EAAEE,EAAEG,EAAEiB,EAAEE,EAAEmB,EAAEvB,EAAEQ,EAAEG,EAAEzE,EAAEqB,KAAK,QAAG,IAASrB,EAAE8B,YAAY,OAAO,KAAK,IAAIV,EAAEa,MAAM3B,KAAK,GAAGc,EAAEa,KAAK9B,EAAE,CAACE,EAAEL,EAAE4B,IAAIR,EAAEQ,OAAOpB,EAAET,EAAE4B,MAAMnB,EAAER,GAAGF,EAAE,GAAG,mBAAmB2E,EAAE,IAAI,GAAGb,EAAE5D,EAAEsB,MAAMkB,EAAE,cAAciC,GAAGA,EAAEa,UAAUC,OAAO7C,GAAGlC,EAAEiE,EAAEe,cAAcvF,EAAEO,EAAEqB,KAAKe,EAAEpC,EAAEkC,EAAEA,EAAEpB,MAAMmE,MAAMjF,EAAEkB,GAAGzB,EAAEmB,EAAES,IAAIV,GAAGuC,EAAE1D,EAAE6B,IAAIT,EAAES,KAAKH,GAAGgC,EAAEgC,KAAKlD,EAAExC,EAAE6B,IAAI6B,EAAE,IAAIe,EAAEb,EAAEhB,IAAI5C,EAAE6B,IAAI6B,EAAE,IAAIrB,EAAEuB,EAAEhB,GAAGc,EAAE5B,YAAY2C,EAAEf,EAAE6B,OAAOI,GAAGjD,GAAGA,EAAEkD,IAAIlC,GAAGA,EAAEmC,QAAQnC,EAAEmC,MAAM,CAAA,GAAInC,EAAEJ,IAAIrD,EAAEQ,EAAEiD,EAAEb,KAAI,EAAGa,EAAEoC,IAAI,GAAGpC,EAAEqC,IAAI,IAAIvD,GAAG,MAAMkB,EAAEsC,MAAMtC,EAAEsC,IAAItC,EAAEmC,OAAOrD,GAAG,MAAMiC,EAAEwB,2BAA2BvC,EAAEsC,KAAKtC,EAAEmC,QAAQnC,EAAEsC,IAAIjF,EAAE,CAAA,EAAG2C,EAAEsC,MAAMjF,EAAE2C,EAAEsC,IAAIvB,EAAEwB,yBAAyBrC,EAAEF,EAAEsC,OAAOtF,EAAEgD,EAAEpC,MAAMX,EAAE+C,EAAEmC,MAAMnC,EAAE3B,IAAI/B,EAAES,EAAE+B,GAAG,MAAMiC,EAAEwB,0BAA0B,MAAMvC,EAAEwC,oBAAoBxC,EAAEwC,qBAAqB1D,GAAG,MAAMkB,EAAEyC,mBAAmBzC,EAAEoC,IAAIhD,KAAKY,EAAEyC,uBAAuB,CAAC,GAAG3D,GAAG,MAAMiC,EAAEwB,0BAA0BrC,IAAIlD,GAAG,MAAMgD,EAAE0C,2BAA2B1C,EAAE0C,0BAA0BxC,EAAEhB,GAAG5C,EAAE+B,KAAKX,EAAEW,MAAM2B,EAAE9B,KAAK,MAAM8B,EAAE2C,wBAAuB,IAAK3C,EAAE2C,sBAAsBzC,EAAEF,EAAEsC,IAAIpD,GAAG,CAAC,IAAI5C,EAAE+B,KAAKX,EAAEW,MAAM2B,EAAEpC,MAAMsC,EAAEF,EAAEmC,MAAMnC,EAAEsC,IAAItC,EAAEb,KAAI,GAAI7C,EAAE4B,IAAIR,EAAEQ,IAAI5B,EAAEyB,IAAIL,EAAEK,IAAIzB,EAAEyB,IAAI6E,KAAK,SAASxG,GAAGA,IAAIA,EAAE4B,GAAG1B,EAAE,GAAG+C,EAAE,EAAEA,EAAEW,EAAEqC,IAAItD,OAAOM,IAAIW,EAAEoC,IAAIhD,KAAKY,EAAEqC,IAAIhD,IAAIW,EAAEqC,IAAI,GAAGrC,EAAEoC,IAAIrD,QAAQrC,EAAE0C,KAAKY,GAAG,MAAM5D,CAAC,CAAC,MAAM4D,EAAE6C,qBAAqB7C,EAAE6C,oBAAoB3C,EAAEF,EAAEsC,IAAIpD,GAAGJ,GAAG,MAAMkB,EAAE8C,oBAAoB9C,EAAEoC,IAAIhD,KAAK,WAAWY,EAAE8C,mBAAmB9F,EAAEC,EAAEgD,EAAE,EAAE,CAAC,GAAGD,EAAEnB,QAAQK,EAAEc,EAAEpC,MAAMsC,EAAEF,EAAEN,IAAItD,EAAE4D,EAAE9B,KAAI,EAAGoC,EAAEjE,EAAEiD,IAAIkB,EAAE,EAAE1B,EAAE,CAAC,IAAIkB,EAAEmC,MAAMnC,EAAEsC,IAAItC,EAAEb,KAAI,EAAGmB,GAAGA,EAAEhE,GAAGQ,EAAEkD,EAAE6B,OAAO7B,EAAEpC,MAAMoC,EAAEmC,MAAMnC,EAAEnB,SAAS8C,EAAE,EAAEA,EAAE3B,EAAEqC,IAAItD,OAAO4C,IAAI3B,EAAEoC,IAAIhD,KAAKY,EAAEqC,IAAIV,IAAI3B,EAAEqC,IAAI,EAAE,MAAM,GAAGrC,EAAEb,KAAI,EAAGmB,GAAGA,EAAEhE,GAAGQ,EAAEkD,EAAE6B,OAAO7B,EAAEpC,MAAMoC,EAAEmC,MAAMnC,EAAEnB,SAASmB,EAAEmC,MAAMnC,EAAEsC,UAAUtC,EAAEb,OAAOqB,EAAE,IAAIR,EAAEmC,MAAMnC,EAAEsC,IAAI,MAAMtC,EAAE+C,kBAAkBxG,EAAEc,EAAEA,EAAE,CAAA,EAAGd,GAAGyD,EAAE+C,oBAAoBjE,IAAI/B,GAAG,MAAMiD,EAAEgD,0BAA0B/C,EAAED,EAAEgD,wBAAwBhG,EAAEC,IAAImD,EAAEtD,EAAE,MAAMA,GAAGA,EAAEa,OAAOc,GAAG,MAAM3B,EAAEe,MAAMuC,EAAE6C,EAAEnG,EAAEc,MAAMc,WAAW/B,EAAEoD,EAAE3D,EAAEc,EAAEkD,GAAGA,EAAE,CAACA,GAAG9D,EAAEoB,EAAEnB,EAAEC,EAAEC,EAAEC,EAAEC,EAAEC,EAAEC,GAAGmD,EAAEf,KAAK3C,EAAE4B,IAAI5B,EAAEiC,UAAUyB,EAAEoC,IAAIrD,QAAQrC,EAAE0C,KAAKY,GAAGvC,IAAIuC,EAAEgC,IAAIhC,EAAEhC,GAAG,KAAK,OAAO5B,GAAG,GAAGE,EAAE+B,IAAI,KAAKzB,GAAG,MAAMH,EAAE,GAAGL,EAAE8G,KAAK,CAAC,IAAI5G,EAAEiC,KAAK3B,EAAE,IAAI,IAAID,GAAG,GAAGA,EAAEgE,UAAUhE,EAAE8D,aAAa9D,EAAEA,EAAE8D,YAAYhE,EAAEA,EAAE0G,QAAQxG,IAAI,KAAKL,EAAE4B,IAAIvB,CAAC,KAAK,CAAC,IAAIiE,EAAEnE,EAAEsC,OAAO6B,KAAKtD,EAAEb,EAAEmE,IAAIwC,EAAE9G,EAAE,MAAMA,EAAE4B,IAAIR,EAAEQ,IAAI5B,EAAEyB,IAAIL,EAAEK,IAAI3B,EAAE8G,MAAME,EAAE9G,GAAGD,EAAE6B,IAAI9B,EAAEE,EAAEoB,EAAE,MAAM,MAAMjB,GAAGH,EAAE+B,KAAKX,EAAEW,KAAK/B,EAAEyB,IAAIL,EAAEK,IAAIzB,EAAE4B,IAAIR,EAAEQ,KAAKvB,EAAEL,EAAE4B,IAA0Z,SAAW5B,EAAEoB,EAAEnB,EAAEC,EAAEC,EAAEC,EAAEC,EAAEC,EAAEC,GAAG,IAAIC,EAAEkD,EAAEhD,EAAEC,EAAEI,EAAE4C,EAAExC,EAAEyC,EAAE3D,EAAEqB,OAAOb,EAAE0B,EAAEf,EAAEE,MAAMe,EAAEjB,EAAEC,KAAK,GAAG,OAAOgB,EAAElC,EAAE,6BAA6B,QAAQkC,EAAElC,EAAE,qCAAqCA,IAAIA,EAAE,gCAAgC,MAAMC,EAAE,IAAII,EAAE,EAAEA,EAAEJ,EAAEqC,OAAOjC,IAAI,IAAIO,EAAEX,EAAEI,KAAK,iBAAiBO,KAAKsB,IAAIA,EAAEtB,EAAEgG,WAAW1E,EAAE,GAAGtB,EAAEsD,UAAU,CAACrE,EAAEe,EAAEX,EAAEI,GAAG,KAAK,KAAK,CAAC,GAAG,MAAMR,EAAE,CAAC,GAAG,MAAMqC,EAAE,OAAO2E,SAASC,eAAe9E,GAAGnC,EAAEgH,SAASE,gBAAgB/G,EAAEkC,EAAEF,EAAEgF,IAAIhF,GAAG7B,IAAIP,EAAEqH,KAAKrH,EAAEqH,IAAIhG,EAAEhB,GAAGE,GAAE,GAAIF,EAAE,IAAI,CAAC,GAAG,MAAMiC,EAAEuB,IAAIzB,GAAG7B,GAAGN,EAAEqH,MAAMlF,IAAInC,EAAEqH,KAAKlF,OAAO,CAAC,GAAG/B,EAAEA,GAAGN,EAAEwH,KAAKtH,EAAEuH,aAAajH,GAAG,MAAMF,EAAE,IAAIwD,EAAE,GAAGpD,EAAE,EAAEA,EAAER,EAAEwH,WAAW/E,OAAOjC,IAAIoD,GAAG7C,EAAEf,EAAEwH,WAAWhH,IAAIiH,MAAM1G,EAAE0E,MAAM,IAAIjF,KAAKoD,EAAE,GAAG7C,EAAE6C,EAAEpD,GAAG,YAAYA,QAAE,GAAS,2BAA2BA,EAAEE,EAAEK,OAAA,KAAYP,KAAK2B,GAAG,CAAC,GAAG,SAAS3B,GAAG,iBAAiB2B,GAAG,WAAW3B,GAAG,mBAAmB2B,EAAE,SAASsC,EAAEzE,EAAEQ,EAAE,KAAKO,EAAEZ,EAAE,CAAC,IAAIK,KAAK2B,EAAEpB,EAAEoB,EAAE3B,GAAG,YAAYA,EAAEG,EAAEI,EAAE,2BAA2BP,EAAEkD,EAAE3C,EAAE,SAASP,EAAEmD,EAAE5C,EAAE,WAAWP,EAAEW,EAAEJ,EAAET,GAAG,mBAAmBS,GAAG6C,EAAEpD,KAAKO,GAAG0D,EAAEzE,EAAEQ,EAAEO,EAAE6C,EAAEpD,GAAGL,GAAG,GAAGuD,EAAEpD,GAAGI,IAAIgD,EAAEgE,QAAQhH,EAAEgH,QAAQhE,EAAEgE,QAAQ1H,EAAE2H,aAAa3H,EAAE2H,UAAUjE,EAAEgE,QAAQtG,EAAEK,IAAI,WAAWf,IAAIV,EAAE2H,UAAU,IAAIlE,EAAE,YAAYrC,EAAEC,KAAKrB,EAAE4H,QAAQ5H,EAAEY,EAAED,GAAGA,EAAE,CAACA,GAAGS,EAAEnB,EAAEC,EAAE,iBAAiBmC,EAAE,+BAA+BlC,EAAEC,EAAEC,EAAED,EAAEA,EAAE,GAAGH,EAAEwB,KAAKe,EAAEvC,EAAE,GAAGK,EAAEC,GAAG,MAAMH,EAAE,IAAII,EAAEJ,EAAEqC,OAAOjC,KAAKQ,EAAEZ,EAAEI,IAAIF,IAAIE,EAAE,QAAQ,YAAY6B,GAAG,MAAMsB,EAAE3D,EAAEiF,gBAAgB,SAAS,MAAMtB,IAAIA,IAAI3D,EAAEQ,IAAI,YAAY6B,IAAIsB,GAAG,UAAUtB,GAAGsB,GAAGC,EAAEpD,KAAKiE,EAAEzE,EAAEQ,EAAEmD,EAAEC,EAAEpD,GAAGL,GAAGK,EAAE,UAAU,MAAMW,GAAGA,GAAGnB,EAAEQ,IAAIiE,EAAEzE,EAAEQ,EAAEW,EAAEyC,EAAEpD,GAAGL,GAAG,CAAC,OAAOH,CAAC,CAAn0D6H,CAAEzG,EAAEQ,IAAI5B,EAAEoB,EAAEnB,EAAEC,EAAEC,EAAEC,EAAEE,EAAEC,GAAG,OAAOC,EAAET,EAAE+H,SAAStH,EAAER,GAAG,IAAIA,EAAEiC,SAAI,EAAO5B,CAAC,CAAC,SAASyG,EAAEhH,GAAGA,GAAGA,EAAE+B,MAAM/B,EAAE+B,IAAID,KAAI,GAAI9B,GAAGA,EAAE2B,KAAK3B,EAAE2B,IAAIsG,QAAQjB,EAAE,CAAC,SAAStD,EAAE1D,EAAEE,EAAEoB,GAAG,IAAA,IAAQnB,EAAE,EAAEA,EAAEmB,EAAEqB,OAAOxC,IAAIgE,EAAE7C,EAAEnB,GAAGmB,IAAInB,GAAGmB,IAAInB,IAAIF,EAAE8B,KAAK9B,EAAE8B,IAAI7B,EAAEF,GAAGA,EAAEwG,KAAK,SAAStG,GAAG,IAAIF,EAAEE,EAAE8F,IAAI9F,EAAE8F,IAAI,GAAGhG,EAAEwG,KAAK,SAASxG,GAAGA,EAAEwH,KAAKtH,EAAE,EAAE,OAAOF,GAAGC,EAAE6B,IAAI9B,EAAEE,EAAE+B,IAAI,CAAC,EAAE,CAAC,SAAS4E,EAAE7G,GAAG,MAAM,iBAAiBA,GAAG,MAAMA,GAAGA,EAAE6B,KAAK7B,EAAE6B,IAAI,EAAE7B,EAAEc,EAAEd,GAAGA,EAAEkI,IAAIrB,GAAG5F,EAAE,CAAA,EAAGjB,EAAE,CAA+6C,SAASmE,EAAEnE,EAAEE,EAAEoB,GAAG,IAAI,GAAG,mBAAmBtB,EAAE,CAAC,IAAIG,EAAE,mBAAmBH,EAAEmC,IAAIhC,GAAGH,EAAEmC,MAAMhC,GAAG,MAAMD,IAAIF,EAAEmC,IAAInC,EAAEE,GAAG,MAAMF,EAAEmI,QAAQjI,CAAC,OAAOF,GAAGC,EAAE6B,IAAI9B,EAAEsB,EAAE,CAAC,CAAC,SAAS2C,EAAEjE,EAAEE,EAAEoB,GAAG,IAAInB,EAAEC,EAAE,GAAGH,EAAEmI,SAASnI,EAAEmI,QAAQpI,IAAIG,EAAEH,EAAE0B,OAAOvB,EAAEgI,SAAShI,EAAEgI,SAASnI,EAAE8B,KAAKqC,EAAEhE,EAAE,KAAKD,IAAI,OAAOC,EAAEH,EAAE+B,KAAK,CAAC,GAAG5B,EAAEkI,qBAAqB,IAAIlI,EAAEkI,sBAAsB,OAAOrI,GAAGC,EAAE6B,IAAI9B,EAAEE,EAAE,CAACC,EAAE0C,KAAK1C,EAAEmD,IAAI,IAAI,CAAC,GAAGnD,EAAEH,EAAE2B,IAAI,IAAIvB,EAAE,EAAEA,EAAED,EAAEwC,OAAOvC,IAAID,EAAEC,IAAI6D,EAAE9D,EAAEC,GAAGF,EAAEoB,GAAG,mBAAmBtB,EAAEuB,MAAMD,GAAGJ,EAAElB,EAAE8B,KAAK9B,EAAE+B,IAAI/B,EAAE4B,GAAG5B,EAAE8B,SAAI,CAAM,CAAC,SAAS+D,EAAE7F,EAAEC,EAAEC,GAAG,OAAOsC,KAAKR,YAAYhC,EAAEE,EAAE,CAAC,SAASoI,EAAEpI,EAAEoB,EAAEnB,GAAG,IAAME,EAAEC,EAAEC,EAAEe,GAAG4F,WAAW5F,EAAE4F,SAASqB,iBAAiBtI,EAAE2B,IAAI3B,EAAE2B,GAAG1B,EAAEoB,GAAGjB,GAAK,EAAsB,KAAeiB,EAAEK,IAAIrB,EAAE,GAAGC,EAAE,GAAGgD,EAAEjC,EAAEpB,EAAUoB,EAAGK,IAAjhS,SAAW1B,EAAEC,EAAEoB,GAAG,IAAInB,EAAEC,EAAEC,EAAEC,EAAE,GAAG,IAAID,KAAKH,EAAE,OAAOG,EAAEF,EAAED,EAAEG,GAAG,OAAOA,EAAED,EAAEF,EAAEG,GAAGC,EAAED,GAAGH,EAAEG,GAAG,GAAGmI,UAAU7F,OAAO,IAAIrC,EAAEgC,SAASkG,UAAU7F,OAAO,EAAE3C,EAAEwH,KAAKgB,UAAU,GAAGlH,GAAG,mBAAmBrB,GAAG,MAAMA,EAAEwI,aAAa,IAAIpI,KAAKJ,EAAEwI,kBAAa,IAASnI,EAAED,KAAKC,EAAED,GAAGJ,EAAEwI,aAAapI,IAAI,OAAOgB,EAAEpB,EAAEK,EAAEH,EAAEC,EAAE,KAAK,CAAivRyD,CAAExB,EAAE,KAAK,CAACnC,IAAIG,GAAGM,EAAEA,EAAEW,EAAEmC,aAAuBpD,EAAE,KAAKiB,EAAEoH,WAAW1I,EAAEwH,KAAKlG,EAAEmG,YAAY,KAAKnH,EAAUD,EAAEA,EAAEyB,IAAIR,EAAEoH,WAA1L,MAAuMnI,GAAGmD,EAAEpD,EAAEJ,EAAEK,EAAE,CAAy1BP,EAAEY,EAAEoE,MAAM/E,EAAE,CAAC6B,IAAI,SAAS9B,EAAEC,EAAEC,EAAEoB,GAAG,IAAA,IAAQnB,EAAEC,EAAEC,EAAEJ,EAAEA,EAAE2B,IAAI,IAAIzB,EAAEF,EAAE8B,OAAO5B,EAAEyB,GAAG,IAAI,IAAIxB,EAAED,EAAE6B,cAAc,MAAM5B,EAAEuI,2BAA2BxI,EAAEyI,SAASxI,EAAEuI,yBAAyB3I,IAAIK,EAAEF,EAAE4C,KAAK,MAAM5C,EAAE0I,oBAAoB1I,EAAE0I,kBAAkB7I,EAAEsB,GAAG,CAAA,GAAIjB,EAAEF,EAAE4C,KAAK1C,EAAE,OAAOF,EAAEyF,IAAIzF,CAAC,OAAOF,GAAGD,EAAEC,CAAC,CAAC,MAAMD,CAAC,GAAGE,EAAE,EAAwDqC,EAAEiD,UAAUoD,SAAS,SAAS5I,EAAEC,GAAG,IAAIC,EAAEA,EAAE,MAAMsC,KAAK0D,KAAK1D,KAAK0D,KAAK1D,KAAKuD,MAAMvD,KAAK0D,IAAI1D,KAAK0D,IAAIjF,EAAE,CAAA,EAAGuB,KAAKuD,OAAO,mBAAmB/F,IAAIA,EAAEA,EAAEiB,EAAE,CAAA,EAAGf,GAAGsC,KAAKhB,QAAQxB,GAAGiB,EAAEf,EAAEF,GAAG,MAAMA,GAAGwC,KAAKP,MAAMhC,GAAGuC,KAAKyD,IAAIjD,KAAK/C,GAAG6C,EAAEN,MAAM,EAAED,EAAEiD,UAAUsD,YAAY,SAAS9I,GAAGwC,KAAKP,MAAMO,KAAKV,KAAI,EAAG9B,GAAGwC,KAAKwD,IAAIhD,KAAKhD,GAAG8C,EAAEN,MAAM,EAAED,EAAEiD,UAAUC,OAAOpD,EAAElC,EAAE,GAAGE,EAAE,mBAAmB0I,QAAQA,QAAQvD,UAAUsB,KAAKkC,KAAKD,QAAQE,WAAWC,WAAW5I,EAAE,SAASN,EAAEC,GAAG,OAAOD,EAAEiC,IAAIJ,IAAI5B,EAAEgC,IAAIJ,GAAG,EAAEoB,EAAEC,IAAI,EAAE3C,EAAE,8BAA8BC,EAAE,EAAEC,EAAE4E,GAAE,GAAI3E,EAAE2E,GAAE,GCA1tV,IAAuE9E,EAAE,EAAkB,SAASL,EAAEI,EAAEgB,EAAEtB,EAAEI,EAAED,EAAED,GAAGoB,IAAIA,EAAE,IAAI,IAAIZ,EAAEF,EAAEG,EAAEW,EAAE,GAAG,QAAQX,EAAE,IAAIH,KAAKG,EAAE,CAAA,EAAGW,EAAE,OAAOd,EAAEE,EAAEY,EAAEd,GAAGG,EAAEH,GAAGc,EAAEd,GAAG,IAAIP,EAAE,CAACsB,KAAKjB,EAAEkB,MAAMb,EAAEc,IAAIzB,EAAE0B,IAAIhB,EAAEiB,IAAI,KAAKC,GAAG,KAAKC,IAAI,EAAEC,IAAI,KAAKC,IAAI,KAAKC,iBAAY,EAAOC,MAAM1B,EAAE2B,KAAI,EAAGC,IAAI,EAAEgH,SAAShJ,EAAEiJ,OAAOlJ,GAAG,GAAG,mBAAmBI,IAAII,EAAEJ,EAAEmI,cAAc,IAAIjI,KAAKE,OAAE,IAASC,EAAEH,KAAKG,EAAEH,GAAGE,EAAEF,IAAI,OAAOH,EAAE+B,OAAO/B,EAAE+B,MAAMnC,GAAGA,CAAC,CCA1wB,IAAIqB,EAAEjB,EAAEH,EAAEC,EAAEC,EAAE,EAAEG,EAAE,GAAGC,EAAER,EAAEM,EAAEE,EAAEqB,IAAInB,EAAEF,EAAE0C,IAAItC,EAAEJ,EAAEwH,OAAO/H,EAAEO,EAAEuB,IAAIV,EAAEb,EAAE4H,QAAQ3H,EAAED,EAAEoB,GAAG,SAASjB,EAAEX,EAAEsB,GAAGd,EAAEwF,KAAKxF,EAAEwF,IAAI3F,EAAEL,EAAEI,GAAGkB,GAAGlB,EAAE,EAAE,IAAIF,EAAEG,EAAEgJ,MAAMhJ,EAAEgJ,IAAI,CAACzH,GAAG,GAAGoE,IAAI,KAAK,OAAOhG,GAAGE,EAAE0B,GAAGe,QAAQzC,EAAE0B,GAAGoB,KAAK,CAAA,GAAI9C,EAAE0B,GAAG5B,EAAE,CAAC,SAASc,GAAEd,GAAG,OAAOI,EAAE,EAAS,SAAWJ,EAAEE,GAAK,IAAIE,EAAEO,EAAEW,IAAI,GAAG,GAAGlB,EAAEkB,EAAEtB,GAAGI,EAAE2B,MAAM3B,EAAEwB,GAAG,CAAQqC,QAAE,EAAO/D,GAAG,SAASF,GAAG,IAAIsB,EAAElB,EAAEkJ,IAAIlJ,EAAEkJ,IAAI,GAAGlJ,EAAEwB,GAAG,GAAGvB,EAAED,EAAEkB,EAAEA,EAAEtB,GAAGsB,IAAIjB,IAAID,EAAEkJ,IAAI,CAACjJ,EAAED,EAAEwB,GAAG,IAAIxB,EAAE2B,IAAI6G,SAAS,CAAA,GAAI,GAAGxI,EAAE2B,IAAI1B,GAAGA,EAAEkJ,KAAK,CAAC,IAAIhJ,EAAE,SAASP,EAAEsB,EAAEjB,GAAG,IAAID,EAAE2B,IAAIsH,IAAI,OAAM,EAAG,IAAInJ,EAAEE,EAAE2B,IAAIsH,IAAIzH,GAAG4H,OAAO,SAASxJ,GAAG,QAAQA,EAAE+B,GAAG,GAAG,GAAG7B,EAAEuJ,MAAM,SAASzJ,GAAG,OAAOA,EAAEsJ,GAAG,GAAG,OAAO9I,GAAGA,EAAEgH,KAAKhF,KAAKxC,EAAEsB,EAAEjB,GAAG,IAAIF,EAAEC,EAAE2B,IAAIP,QAAQxB,EAAE,OAAOE,EAAE+H,QAAQ,SAASjI,GAAG,GAAGA,EAAEsJ,IAAI,CAAC,IAAIhI,EAAEtB,EAAE4B,GAAG,GAAG5B,EAAE4B,GAAG5B,EAAEsJ,IAAItJ,EAAEsJ,SAAI,EAAOhI,IAAItB,EAAE4B,GAAG,KAAKzB,GAAE,EAAG,CAAC,GAAGK,GAAGA,EAAEgH,KAAKhF,KAAKxC,EAAEsB,EAAEjB,IAAIF,CAAC,EAAEE,EAAEkJ,KAAI,EAAG,IAAI/I,EAAEH,EAAEkG,sBAAsBjG,EAAED,EAAEoG,oBAAoBpG,EAAEoG,oBAAoB,SAASzG,EAAEsB,EAAEjB,GAAG,GAAGmC,KAAKV,IAAI,CAAC,IAAI5B,EAAEM,EAAEA,OAAE,EAAOD,EAAEP,EAAEsB,EAAEjB,GAAGG,EAAEN,CAAC,CAACI,GAAGA,EAAEkH,KAAKhF,KAAKxC,EAAEsB,EAAEjB,EAAE,EAAEA,EAAEkG,sBAAsBhG,CAAC,CAAC,OAAOH,EAAEkJ,KAAKlJ,EAAEwB,EAAE,CAArtBgC,CAAEK,GAAEjE,EAAE,CAAgtB,SAASa,GAAEb,EAAEE,GAAG,IAAIC,EAAEQ,EAAEW,IAAI,IAAId,EAAE0F,KAAKtD,GAAEzC,EAAEkJ,IAAInJ,KAAKC,EAAEyB,GAAG5B,EAAEG,EAAED,EAAEA,EAAEG,EAAEgJ,IAAIrD,IAAIhD,KAAK7C,GAAG,CAAiF,SAASiE,GAAEpE,GAAG,OAAOI,EAAE,EAAEoE,GAAE,WAAW,MAAM,CAAC2D,QAAQnI,EAAE,EAAE,GAAG,CAAsN,SAASwE,GAAExE,EAAEK,GAAG,IAAIH,EAAES,EAAEW,IAAI,GAAG,OAAOsB,GAAE1C,EAAEmJ,IAAIhJ,KAAKH,EAAE0B,GAAG5B,IAAIE,EAAEmJ,IAAIhJ,EAAEH,EAAE8F,IAAIhG,GAAGE,EAAE0B,EAAE,CAAqiB,SAAS+C,KAAI,IAAA,IAAQ3E,EAAEA,EAAEO,EAAE8C,YAAYrD,EAAEsD,KAAKtD,EAAEqJ,IAAI,IAAIrJ,EAAEqJ,IAAIrD,IAAIiC,QAAQjB,IAAGhH,EAAEqJ,IAAIrD,IAAIiC,QAAQ9D,IAAGnE,EAAEqJ,IAAIrD,IAAI,EAAE,OAAO1E,GAAGtB,EAAEqJ,IAAIrD,IAAI,GAAGxF,EAAEsB,IAAIR,EAAEtB,EAAEiC,IAAI,CAAC,CAACzB,EAAEqB,IAAI,SAAS7B,GAAGK,EAAE,KAAKC,GAAGA,EAAEN,EAAE,EAAEQ,EAAEoB,GAAG,SAAS5B,EAAEsB,GAAGtB,GAAGsB,EAAEK,KAAKL,EAAEK,IAAI2F,MAAMtH,EAAEsH,IAAIhG,EAAEK,IAAI2F,KAAK7G,GAAGA,EAAET,EAAEsB,EAAE,EAAEd,EAAE0C,IAAI,SAASlD,GAAGU,GAAGA,EAAEV,GAAGsB,EAAE,EAAE,IAAInB,GAAGE,EAAEL,EAAE+B,KAAKsH,IAAIlJ,IAAID,IAAIG,GAAGF,EAAE6F,IAAI,GAAG3F,EAAE2F,IAAI,GAAG7F,EAAEyB,GAAGqG,QAAQ,SAASjI,GAAGA,EAAEsJ,MAAMtJ,EAAE4B,GAAG5B,EAAEsJ,KAAKtJ,EAAEE,EAAEF,EAAEsJ,SAAI,CAAM,KAAKnJ,EAAE6F,IAAIiC,QAAQjB,IAAG7G,EAAE6F,IAAIiC,QAAQ9D,IAAGhE,EAAE6F,IAAI,GAAG1E,EAAE,IAAIpB,EAAEG,CAAC,EAAEG,EAAEwH,OAAO,SAAShI,GAAGY,GAAGA,EAAEZ,GAAG,IAAIsB,EAAEtB,EAAE+B,IAAIT,GAAGA,EAAE+H,MAAM/H,EAAE+H,IAAIrD,IAAIrD,SAAS,IAAIpC,EAAEyC,KAAK1B,IAAInB,IAAIK,EAAEkJ,yBAAyBvJ,EAAEK,EAAEkJ,wBAAwBzI,IAAG0D,KAAIrD,EAAE+H,IAAIzH,GAAGqG,QAAQ,SAASjI,GAAGA,EAAEE,IAAIF,EAAEqJ,IAAIrJ,EAAEE,GAAGF,EAAEE,OAAE,CAAM,IAAIA,EAAEG,EAAE,IAAI,EAAEG,EAAEuB,IAAI,SAAS/B,EAAEsB,GAAGA,EAAEkF,KAAK,SAASxG,GAAG,IAAIA,EAAEgG,IAAIiC,QAAQjB,IAAGhH,EAAEgG,IAAIhG,EAAEgG,IAAIwD,OAAO,SAASxJ,GAAG,OAAOA,EAAE4B,IAAIuC,GAAEnE,EAAE,EAAE,OAAOK,GAAGiB,EAAEkF,KAAK,SAASxG,GAAGA,EAAEgG,MAAMhG,EAAEgG,IAAI,GAAG,GAAG1E,EAAE,GAAGd,EAAEsB,IAAIzB,EAAEL,EAAEiC,IAAI,CAAC,GAAGhC,GAAGA,EAAED,EAAEsB,EAAE,EAAEd,EAAE4H,QAAQ,SAASpI,GAAGqB,GAAGA,EAAErB,GAAG,IAAIsB,EAAEjB,EAAEL,EAAE+B,IAAI1B,GAAGA,EAAEgJ,MAAMhJ,EAAEgJ,IAAIzH,GAAGqG,QAAQ,SAASjI,GAAG,IAAIgH,GAAEhH,EAAE,OAAOA,GAAGsB,EAAEtB,CAAC,CAAC,GAAGK,EAAEgJ,SAAI,EAAO/H,GAAGd,EAAEsB,IAAIR,EAAEjB,EAAE4B,KAAK,EAAE,IAAII,GAAE,mBAAmBqH,sBAAsB,SAASzI,GAAEjB,GAAG,IAAIsB,EAAEjB,EAAE,WAAWsJ,aAAazJ,GAAGmC,IAAGuH,qBAAqBtI,GAAG4H,WAAWlJ,EAAE,EAAEE,EAAEgJ,WAAW7I,EAAE,IAAIgC,KAAIf,EAAEoI,sBAAsBrJ,GAAG,CAAC,SAAS2G,GAAEhH,GAAG,IAAIsB,EAAEjB,EAAEH,EAAEF,EAAE+B,IAAI,mBAAmB7B,IAAIF,EAAE+B,SAAI,EAAO7B,KAAKG,EAAEiB,CAAC,CAAC,SAAS6C,GAAEnE,GAAG,IAAIsB,EAAEjB,EAAEL,EAAE+B,IAAI/B,EAAE4B,KAAKvB,EAAEiB,CAAC,CAAC,SAASsB,GAAE5C,EAAEsB,GAAG,OAAOtB,GAAGA,EAAE2C,SAASrB,EAAEqB,QAAQrB,EAAEkF,KAAK,SAASlF,EAAEjB,GAAG,OAAOiB,IAAItB,EAAEK,EAAE,EAAE,CAAC,SAAS4D,GAAEjE,EAAEsB,GAAG,MAAM,mBAAmBA,EAAEA,EAAEtB,GAAGsB,CAAC,CCCt6G,MAWMuI,GAAS,CACpBC,QAAS,UACTC,aAAc,UAEdC,KAAM,yBAINC,YAAa,UACbC,gBAAiB,WA6BNC,GACH,MADGA,GAEH,OAFGA,GAGF,OAIEC,GAAS,qCAGTC,GACL,oBADKA,GAEH,oBCzBGC,GCrCE,goBDuCFC,GEvCE,8UFyCFC,GGzCE,+TH0CFC,GI1CE,wNCsBR,SAASC,IAAQC,IACpBA,EAAAC,MACAA,EAAAC,OACAA,EAAAC,MACAA,EAAAC,YACAA,GAAc,EAAAC,UACdA,EAAApG,MACAA,EAAAqG,KACAA,EAAAC,OACAA,EAAAC,YACAA,KACG3J,IAEH,MAAM4J,QAAEA,EAAAtD,QAASA,GAAYuD,GAAQ,KACjC,IAAKV,EAAK,MAAO,CAAES,QAAS,YAAatD,QAAS,IAElD,MAAMwD,EAAaX,EAAIY,OAEjBC,EAAeF,EAAWG,MAAM,qBAOtC,MAAO,CAAEL,QANOI,EAAeA,EAAa,GAAK,YAM/B1D,QAFJwD,EAAWxG,QAAQ,eAAgB,IAAIA,QAAQ,eAAgB,MAG9E,CAAC6F;AA4BJ,OACIe,EAAC,MAAA,CACGN,UACAR,QACAC,SACAG,YACApG,MAAO,CACH+G,QAAS,eACTC,cAAe,SACfC,WAAY,EACZZ,KAAM,OACNC,OAAQ,kBACLtG,GAEPkH,MAAM,gCACFtK,EACJuK,wBAAyB,CAAEnE,OA1CdyD,GAAQ,KACzB,IAAKvD,EAAS,MAAO,GACrB,IAAIkE,EAAclE,EAElB,GAAIiD,EAAa,CAEb,MAAMkB,OAAqB,IAAThB,EAAqBA,EAAO,OACxCiB,OAAyB,IAAXhB,EAAuBA,EAAS,eACpDc,EAAcA,EACTlH,QAAQ,iBAAkB,SAASmH,MACnCnH,QAAQ,mBAAoB,WAAWoH,KAChD,CAEA,QAAoB,IAAhBf,EAA2B,CAC3B,MAAMgB,EAAmBpI,OAAOoH,GAChCa,EAAc,wBAAwBtH,KAAKsH,GACrCA,EAAYlH,QAAQ,yBAA0B,iBAAiBqH,MAC/DH,EAAYlH,QACV,wDACA,qBAAqBqH,KAEjC,CAEA,OAAOH,GACR,CAAClE,EAASiD,EAAaE,EAAMC,EAAQC,KAoB/B7I,SAAAwI,kBAASY,EAAC,QAAA,CAAOpJ,SAAAwI,KAG9B,CCnEO,MAAMsB,GAAa,EACxBC,aACAC,oBACAC,aACAC,cACAC,SACAC,qBAAoB,EACpBC,aACAC,gBACAC,iBACAC,iBACAC,qBAEA,MAAMC,EAAUnD,GAAOC,QACjBmD,EAAiBP,EACnB,mBAAmB7C,GAAOC,YAAYM,KACtCA,GAEE8C,EAA4C,CAChDC,SAAU,QACVC,MAAO,OACPC,OAAQ,OACR1B,QAAS,OACT2B,cAAe,SACfC,WAAY,WACZC,IAAK,MACLC,OAAQ,OACRC,WAAY,uBASRC,EAAyC,CAC7C/C,MAAO,OACPC,OAAQ,OACR+C,SAAU,OACVC,UAAW,OACXC,aAAc3D,GACd4D,gBAAiBf,EACjBgB,MAAO,UACPC,OAAQ,OACRC,OAAQ,UACRvC,QAAS,OACT4B,WAAY,SACZY,eAAgB,SAChBC,WAAY,oBAAoB/D,iBAA+BA,KAC/D8C,SAAU,WACVtB,WAAY,KASRwC,EAAiB5B,IAAWD,EAC5B8B,EAAcD,EAAiB7D,GAAyBgC,EAAc/B,GAAmBF,GAazFgE,EAA2C,CAC/CpB,SAAU,WACVqB,IAAK,MACLpB,MAAO,MACPxC,MAAO,OACPC,OAAQ,OACRiD,aAAc,MACdG,OAAQ,OACRQ,WAAY,cACZP,OAAQ,UACRvC,QAAS,OACT4B,WAAY,SACZY,eAAgB,SAChBO,SAAU,OACVC,WAAY,IACZX,MAAOnE,GAAOG,KACd4E,QAAS,KAGLC,EAAc,CAClB/G,EACAgH,EACAC,EACAC,EACAC,mBAEAC,EAAC,MAAA,CACClE,UAAW,eAAegE,IAC1BpK,MAAO,CAAEsJ,OAAQ,WACjBY,UAGCxM,SAAA,CAAAyM,kBACCrD,EAAC,SAAA,CACCnK,KAAK,SACLqD,MAAO2J,EACPO,QAAUxO,IACRA,EAAE6O,kBACFJ,KAEF,aAAW,QAEXzM,wBAAAoJ,EAAChB,IAAQC,IAAKF,GAAkB,cAAY,OAAOG,MAAO,GAAIC,OAAQ;eAK1Ea,EAAC,MAAA,CACC9G,MAAO,CACLwK,aAAcL,EAAU,OAAS,IACjCM,UAAW,gBACPJ,EAAa,CAAEK,UAAW,OAAQC,UAAW,QAAW,CAAA,GAG7DjN,SAAAwF;AAKP,OACEoH,EAAAM,EAAA,CACElN,SAAA;eAAAoJ,EAAC,QAAA,CAAOpJ,SAAA,mLAKKuH,GAAOG,qNAOFiD,kLAMW5C;eAuC7BqB,EAAC,MAAA,CAAIV,UAAU,sBAAsBpG,MAAOsI,EAC1C5K,wBAAA4M,EAAC,MAAA,CAAIlE,UAAU,qBAAqBpG,MAnKU,CAClD+G,QAAS,OACT4B,WAAY,SACZC,IAAK,QAkKElL,SAAA,CAAAqK,IACEH,IACAK,GACDgC,EACE,iDACAjC,EACA,KACA,sBACA,GAIHC,GACCgC,EACEhC,EACAC,EACAC,EACA,uBACA;eAIJrB,EAAC,SAAA,CACC9G,MAAO+I,EACPmB,QA5JU,KACdT,EACF9B,IACSC,EACTF,IAEAD,KAuJMoD,aAAenP,IACboP,OAAOC,OAAOrP,EAAEsP,cAAchL,MAAO,CACnCmJ,gBAAiBlE,GAAOE,aACxB8F,UAAW,iBAGfC,aAAexP,IACboP,OAAOC,OAAOrP,EAAEsP,cAAchL,MAAO,CACnCiL,UAAW,cAGf,aAAYxB,EAAiB,gBAAkB7B,EAAc,iBAAmB,kBAEhFlK,wBAAAoJ,EAAChB,GAAA,CACCC,IAAK2D,EACL,cAAY,OACZ1J,MAtLiC,CAC3CgG,MAAO,OACPC,OAAQ,OACRc,QAAS,sBC5BAoE,GAAyB,iCAEhCC,sBAAuBC,IAAI,CAC/B,SACA,QACA,WACA,WACA,OACA,SAGIC,sBAA8BD,IAAI,CACtC,IACA,SACA,QACA,WACA,SACA,UACA,UACA,WAGIE,sBAAwBF,IAAI,CAChC,SACA,OACA,MACA,WACA,SACA,WACA,QACA,SACA,WACA,UACA,YACA,SACA,aACA,aAGIG,sBAA6BH,IAAI,CACrC,QACA,SACA,eACA,OACA,cACA,OACA,gBAGII,sBAA8BJ,IAAI,CACtC,WACA,WACA,QACA,WAGIK,GAAmB,CACvB,QACA,WACA,YACA,UACA,cACA,YACA,aACA,WACA,UACA,QACA,WACA,QACA,SACA,SACA,QACA,QAGIC,GAAyB,CAC7B,gBACA,gBACA,gBACA,eACA,iBAGIC,GACJ,uFAEIC,GACJ,iGAEIC,sBAAuBC,IAC7B,IAAIC,IAAgB,EAEpB,MAAMC,GAAuBlL,GAC3BA,EAAMb,QAAQ,OAAQ,KAAKyG,OAUvBuF,GAAgBnL,IACpB,GAAqB,iBAAVA,GAAgC,OAAVA,EAC/B,OAAO,EAGT,MAAMoL,EAAMpL,EACZ,MACqB,iBAAZoL,EAAIC,KACY,iBAAhBD,EAAIE,SACXlQ,MAAMC,QAAQ+P,EAAIG,QAClBnQ,MAAMC,QAAQ+P,EAAII,gBACQ,iBAAnBJ,EAAIK,YACY,iBAAhBL,EAAIM,SAITC,GAAY,CAACpJ,EAA0BzG,KACtCA,GAGLyG,EAAIqJ,IAAI9P,GAAMyG,EAAIsJ,IAAI/P,IAAQ,GAAK,IAG/BgQ,GAAkB,CAACvJ,EAA0BwJ,IAChC,IAAbxJ,EAAIyJ,KACC,OAGF5Q,MAAM6Q,KAAK1J,EAAI2J,WACnBzO,KAAK,CAAC1C,EAAGoD,IAAMA,EAAE,GAAKpD,EAAE,IAAMA,EAAE,GAAGoR,cAAchO,EAAE,KACnDkB,MAAM,EAAG0M,GACTxJ,IAAI,EAAEP,EAAMoK,KAAW,GAAGpK,KAAQoK,KAClCC,KAAK,MAGJC,GAAiBtM,IACrB,GAAc,OAAVA,EACF,OAAO,KAGT,MAAMuM,EAASC,OAAOC,SAASzM,EAAO,IACtC,OAAOwM,OAAOE,MAAMH,GAAU,KAAOA,GAGjCI,GAAgB3M,GACJA,EAAMb,QAAQ,OAAQ,KAAKA,QAAQ,kBAAmB,KACpD,GAGdyN,GAAmBC,IACvB,IAAIC,EAAQ,EACRvE,EAAyBsE,EAC7B,KAAOtE,GAAQwE,gBACbD,GAAS,EACTvE,EAASA,EAAOwE,cACZxE,IAAWhH,SAASyL,QAI1B,OAAOF,GAGHG,GAAkBJ,GACN3B,GAAoB2B,EAAQK,aAAa,SAAW,IACjE9N,cACA+N,MAAM,KAAK,IACI,GAuEdC,GACJ5K,IAEA,MAAM6K,EAAcjS,MAAM6Q,KAAKlB,GAAiBuC,UAC7CzJ,OAAQ0J,GAAUA,EAAMlC,MAAQ7I,EAAQ6I,KACxC5N,KAAK,CAAC1C,EAAGoD,IAAMA,EAAEsN,WAAa1Q,EAAE0Q,YAChCpM,MAAM,EAAG,GAEZ,GAA2B,IAAvBgO,EAAYrQ,OACd,OAAOwF,EAAQ8I,QAGjB,MAAMkC,EAAiB,CACrB,yBACGH,EAAY9K,IAAKgL,IAClB,MAAME,EAAiBvC,GAAoBqC,EAAMjC,SACjD,MAAO,KAAKiC,EAAMlC,UAAUoC,OAE9BpB,KAAK,MAEP,MAAO,GAAG7J,EAAQ8I,cAAckC,KAG5BE,GAAab,IACjB,GAA4C,SAAxCA,EAAQK,aAAa,eACvB,OAAO,EAGT,GAAIL,aAAmBc,aAAed,EAAQe,OAC5C,OAAO,EAGT,MAAM3O,EAAQ4O,OAAOC,iBAAiBjB,GACtC,GAAsB,SAAlB5N,EAAM+G,SAA2C,WAArB/G,EAAM8O,WACpC,OAAO,EAGT,MAAMC,EAAOnB,EAAQoB,wBACrB,OAAOD,EAAK/I,MAAQ,GAAK+I,EAAK9I,OAAS,GAGnCgJ,GAAiBC,IACrB,IACE,OAAO,IAAIC,IAAID,EAAMN,OAAOQ,SAASF,MAAMA,IAC7C,CAAA,MACE,OAAOA,CACT,GAGIG,GAAkBtO,GACH,oBAARuO,KAA6C,mBAAfA,IAAIC,OACpCD,IAAIC,OAAOxO,GAGbA,EAAMb,QAAQ,mCAAoC,QAGrDsP,GAAwB5B,IAC5B,MAAM6B,EAAM7B,EAAQ8B,QAAQvP,cAE5B,GAAIyN,EAAQ+B,GACV,MAAO,IAAIN,GAAezB,EAAQ+B,MAGpC,MAAMC,EAA8D,CAClE,CAAC,OAAQhC,EAAQK,aAAa,SAC9B,CAAC,cAAeL,EAAQK,aAAa,gBACrC,CAAC,eAAgBL,EAAQK,aAAa,iBACtC,CAAC,aAAcL,EAAQK,aAAa,eACpC,CAAC,OAAQL,EAAQK,aAAa,SAC9B,CAAC,OAAQL,EAAQK,aAAa,UAGhC,IAAA,MAAY4B,EAAUC,KAAcF,EAClC,GAAIE,EACF,MAAO,GAAGL,KAAOI,MAAaR,GAAeS,OAIjD,MAAMC,EAAU5T,MAAM6Q,KAAKY,EAAQoC,WAChCpL,OAAOqL,SACP7P,MAAM,EAAG,GACTkD,IAAK8C,GAAc,IAAIiJ,GAAejJ,MACtCgH,KAAK,IACR,GAAI2C,EACF,MAAO,GAAGN,IAAMM,IAGlB,MAAMG,EAAStC,EAAQE,cACvB,IAAKoC,EACH,OAAOT,EAOT,MAAO,GAAGA,iBAJYtT,MAAM6Q,KAAKkD,EAAOxS,UAAUkH,OAC/CuL,GAAYA,EAAQT,UAAY9B,EAAQ8B,SAEfvN,QAAQyL,GAAW,MAI3CwC,GAAiBxC,IACrB,MAAMyC,EAAqB,GAC3B,IAAI/G,EAAyBsE,EACzBC,EAAQ,EAEZ,KAAOvE,GAAUuE,EAAQ,GAAG,CAC1B,MAAMyC,EAAUd,GAAqBlG,GAErC,GADA+G,EAASE,QAAQD,GACbA,EAAQE,WAAW,KACrB,MAEFlH,EAASA,EAAOwE,cAChBD,GAAS,CACX,CAEA,OAAOwC,EAASjD,KAAK,QAGjBqD,GAAmB7C,IACvB,MAAMxI,EAAO6G,IACV2B,aAAmBc,YAAcd,EAAQ8C,UAAY9C,EAAQ+C,cAC5D,IAEEC,EAAY3E,GAAoB2B,EAAQK,aAAa,eAAiB,IACtE/H,EAAQ+F,GAAoB2B,EAAQK,aAAa,UAAY,IAC7D4C,EAAc5E,GAClB2B,EAAQK,aAAa,gBAAkB,IAEnClL,EAAOkJ,GAAoB2B,EAAQK,aAAa,SAAW,IAC3DlN,EACJ6M,aAAmBkD,kBACnBlD,aAAmBmD,qBACnBnD,aAAmBoD,kBACf/E,GAAoB2B,EAAQ7M,OAAS,IACrC,GAEAkQ,EAAY9U,MAAM6Q,KAAKY,EAAQoC,WAClC1M,IAAK4N,GAASxD,GAAawD,IAC3BC,KAAKlB,SACFmB,EACHxD,EAAQ+B,IAAM,IAAI/B,EAAQ+B,MAC1BsB,GAAa,IAAIA,KAClBb,GAAcxC,GAEVyD,EACJjM,GAAQwL,GAAa1K,GAAS2K,GAAe9P,GAASgC,GAAQqO,EAEhE,GAAsC,UAAlCxD,EAAQ8B,QAAQvP,cAA2B,CAE7C,MAAO,GADWyN,EAAQK,aAAa,SAAW,UAC3BoD,GAAS,SAClC,CAEA,OAAOA,GAAS,YAGZC,GAAiB1D,IACrB,MAAM2D,EAAS3D,EACT4D,EAAuB,GAE7B,IAAA,MAAWC,KAAa/F,GAAkB,CACxC,MAAMgG,EAAa,KAAKD,IAClBE,EAAmB1B,QAAQrC,EAAQK,aAAayD,IAChDE,EAAmD,mBAAvBL,EAAOG,GAEzC,IAAKC,GAAqBC,KAI1BJ,EAAWpT,KAAKqT,GACZD,EAAWzT,QCzbwB,GD0brC,KAEJ,CAEA,OAAOyT,GAGHK,GAA2BjE,GAC/BjC,GAAuB/G,OAAQiL,GAAajC,EAAQkE,aAAajC,IAAWvM,IACzEuM,GAAaA,EAAS3P,QAAQ,QAAS,KAGtC6R,GAA2BnE,GAC/BA,EACGoE,oBACApN,OACEiL,GACCA,EAASW,WAAW,UAAY5E,GAAyB9L,KAAK+P,IAEjEzP,MAAM,EAAG,GACTkD,IAAKuM,GAAaA,EAAS3P,QAAQ,QAAS,KAE3C+R,GAA2BjS,IAC/B,MAAMkS,EAAoB,GAwB1B,OAtBIlS,EAAMsJ,QAA2B,SAAjBtJ,EAAMsJ,QACxB4I,EAAQ9T,KAAK,UAAU4B,EAAMsJ,UAE3BtJ,EAAM+G,SACRmL,EAAQ9T,KAAK,WAAW4B,EAAM+G,WAE5B/G,EAAMuI,UACR2J,EAAQ9T,KAAK,YAAY4B,EAAMuI,YAE7BvI,EAAM6I,QAA2B,SAAjB7I,EAAM6I,QACxBqJ,EAAQ9T,KAAK,WAAW4B,EAAM6I,UAE5B7I,EAAMmS,eAAyC,SAAxBnS,EAAMmS,eAC/BD,EAAQ9T,KAAK,kBAAkB4B,EAAMmS,iBAEnCnS,EAAM8O,YAAmC,YAArB9O,EAAM8O,YAC5BoD,EAAQ9T,KAAK,cAAc4B,EAAM8O,cAE/B9O,EAAMoS,SAA6B,MAAlBpS,EAAMoS,SACzBF,EAAQ9T,KAAK,WAAW4B,EAAMoS,WAGzBjW,MAAM6Q,KAAK,IAAI3B,IAAI6G,KAatBG,GAAoB,CAACzE,EAAkBC,KAC3C,MAAMyE,EAXoB,CAAC1E,IAC3B,MAAM6B,EAAM7B,EAAQ8B,QAAQvP,cACtBoS,EAAU3E,EAAQ+B,GAAK,IAAIjC,GAAaE,EAAQ+B,MAAQ,GACxD6C,EAAarW,MAAM6Q,KAAKY,EAAQoC,WACnC1M,IAAK4N,GAASxD,GAAawD,IAC3BC,KAAKlB,SAER,MAAO,GAAGR,IAAM8C,IAAUC,EAAa,IAAIA,IAAe,MAI5CC,CAAoB7E,GAClC,GAAIC,GAAS,EACX,OAAOyE,EAGT,MAAM5U,EAAWvB,MAAM6Q,KAAKY,EAAQlQ,UACjCkH,OAAQ8N,IAAWtH,GAAiBuH,IAAID,EAAMhD,QAAQvP,gBACtDyE,OAAQ8N,GAAUjE,GAAUiE,IAC/B,GAAwB,IAApBhV,EAASK,OACX,OAAOuU,EAGT,MAAMM,EAAUlV,EACb0C,MAAM,EAAG,GACTkD,IAAKoP,GAAUL,GAAkBK,EAAO7E,EAAQ,IAC7CgF,EACJnV,EAASK,OAAS6U,EAAQ7U,OAAS,KAAIL,EAASK,OAAS6U,EAAQ7U,QAAW,GAE9E,MAAO,GAAGuU,KAASM,EAAQxF,KAAK,OAAOyF,KAenCC,GAAgB,CAAC5M,EAAe6M,IACf,IAAjBA,EAAMhV,OACD,GAAGmI,aAGL,GAAGA,OAAW6M,EAAM3F,KAAK,QAsB5B4F,GAAsB,KAC1B,MAAMC,EACJ3Q,SAAS4Q,cAAc,iCAAmC5Q,SAASyL,KAC/DoF,EAAqB,GACrBC,qBAAW/H,IAEXgI,EAAalX,MAAM6Q,KAAKiG,EAAKK,iBAAiB,sBACpD,IAAA,MAAWC,KAAQF,EAAY,CAC7B,IAAK5E,GAAU8E,GACb,SAGF,MAAMnO,EAAO6G,GAAoBsH,EAAK5C,aAAe,IACrD,GAAKvL,KAAQA,EAAKrH,OAAS,OAIvBqV,EAAKT,IAAIvN,KAIbgO,EAAKI,IAAIpO,GACT+N,EAAS/U,KAAK,KAAKgH,KACf+N,EAASpV,QCnlBgB,IDolB3B,KAEJ,CAEA,OAAOoV,GA8BHM,GAA+B,CACnCC,EACAC,EACAC,KAEA,IAAA,MAAWC,KAAQ1X,MAAM6Q,KAAK0G,GAAQ,CACpC,GACEE,EAAQzG,OCxmBsB,KDymB9BwG,EAAU5G,MC5mBwB,GD8mBlC,OAKF,GAFA6G,EAAQzG,OAAS,EAEb0G,aAAgBC,aAAc,CAChC,MAAMC,EAAQF,EAAKG,aAChB9F,MAAM,KACN5K,IAAK2Q,GAAahI,GAAoBgI,IACtCrP,OAAOqL,SAEV,IAAA,MAAWgE,KAAYF,EACrB,GAAKlI,GAA4B/L,KAAKmU,KAGtCN,EAAUH,IAAIS,GACVN,EAAU5G,MC9nBkB,ID+nB9B,OAGJ,QACF,CAEA,GAAI,aAAc8G,EAChB,IACE,MAAMK,EAAeL,EAAsBM,SAC3CV,GAA6BS,EAAaP,EAAWC,EACvD,CAAA,MAEA,CAEJ,GAGIQ,GAA6B,KACjC,MAAMT,qBAAgBtI,IAChBuI,EAAU,CAAEzG,MAAO,GAEzB,IAAA,MAAWkH,KAAclY,MAAM6Q,KAAK1K,SAASgS,aAAc,CACzD,GACEV,EAAQzG,OCnpBsB,KDopB9BwG,EAAU5G,MCvpBwB,GDypBlC,MAGF,IACE,IAAKsH,EAAWF,SACd,SAEFV,GAA6BY,EAAWF,SAAUR,EAAWC,EAC/D,CAAA,MAEA,CACF,CAEA,OAAOzX,MAAM6Q,KAAK2G,GAAWrQ,IAAK2Q,GAAa,KAAKA,MAGhDM,GAA4B,KAChC,MAAMC,EAAcrY,MAAM6Q,KAAK1K,SAASgR,iBAAiB,MACnDmB,EAAkBD,EAAYpU,MAAM,ECvrBN,KDyrB9BkM,EAAkB,GAClBoI,qBAAcrJ,IACdsJ,qBAA6B5I,IAC7B6I,qBAAgB7I,IAChB8I,qBAAiB9I,IACjB+I,qBAAkB/I,IAClBgJ,qBAAoBhJ,IACpBiJ,qBAAqBjJ,IACrBkJ,qBAAwBlJ,IAE9B,IAAImJ,EAAkB,EAClBC,EAAW,EACXC,EAAwB,EACxBC,EAA2B,EAE3BC,EAAkB,EAClBC,EAAwB,EACxBC,EAAmB,EACnBC,EAAmB,EACnBC,EAA0B,EAE9B,IAAA,IAASC,EAAQ,EAAGA,EAAQlB,EAAgB1W,OAAQ4X,GAAS,EAAG,CAC9D,MAAM/H,EAAU6G,EAAgBkB,GAC1BlG,EAAM7B,EAAQ8B,QAAQvP,cAE5B,GAAIiL,GAAiBuH,IAAIlD,GACvB,SAGF,IAAKhB,GAAUb,GACb,SAGFsH,GAAmB,EACnBxI,GAAUkI,EAAWnF,GAErB,MAAMmG,EAAO5H,GAAeJ,GACxBgI,GACFlJ,GAAUmI,EAAYe,GAGxB,MAAM/H,EAAQF,GAAgBC,GAC1BC,EAAQsH,IACVA,EAAWtH,GAGb,MAAM7N,EAAQ4O,OAAOC,iBAAiBjB,GAClCpC,GAAuBmH,IAAI3S,EAAM+G,UACnC2F,GAAUqI,EAAe/U,EAAM+G,SAE7B0E,GAAwBkH,IAAI3S,EAAMuI,WACpCmE,GAAUsI,EAAgBhV,EAAMuI,UAGlC,MAAMsN,EAAuB5D,GAAwBjS,GACrD,IAAA,MAAW8V,KAAeD,EACxBnJ,GAAUuI,EAAmBa,GAG/B,MAAMtE,EAAaF,GAAc1D,GACjC,GAAI4D,EAAWzT,OAAS,EAEtB,IAAA,MAAW0T,KAAaD,EACtB9E,GAAUoI,EAAarD,GAI3B,MAAMsE,EAAW1I,GAAcO,EAAQK,aAAa,aAC9C+H,EAA0B,OAAbD,GAAqBA,GAAY,EAChDC,IACFV,GAAmB,GAGrB,MAAMW,EAAoC,YAAjBjW,EAAMsJ,OAC3B2M,IACFV,GAAyB,GAG3B,MAAMW,EAAYnE,GAAwBnE,GACtCsI,EAAUnY,OAAS,IACrByX,GAAoB,GAGtB,MAAMW,EAAYtE,GAAwBjE,GACtCuI,EAAUpY,OAAS,IACrB0X,GAAoB,GAGtB,MAAMW,EAAgE,SAA5CxI,EAAQK,aAAa,mBAC3CmI,IACFV,GAA2B,GAG7B,MAAMxG,EAAOtB,EAAQK,aAAa,QAC5BoI,EAAsB/K,GAAwBqH,IAAIlD,KAAiB,MAARA,GAAeQ,QAAQf,IAClFoH,EAAoB/K,GAAkBoH,IAAIiD,GAC1CW,EACJ3I,EAAQkE,aAAa,aACqB,SAA1ClE,EAAQK,aAAa,iBAEvB,GACU,MAARwB,GACAP,IACCA,EAAKsB,WAAW,OAChBtB,EAAKsB,WAAW,eACjB,CACA,MAAMgG,EAAevH,GAAcC,GAE7BuH,EAAO,KADChG,GAAgB7C,IAAY4I,QACZA,IAEzB9B,EAAQ/B,IAAI8D,KACf/B,EAAQlB,IAAIiD,GACZnK,EAAMlO,KAAKqY,GAKf,CAYA,KATEJ,GACAC,GACAF,GACA5E,EAAWzT,OAAS,GACpBiY,GACAC,GACAC,EAAUnY,OAAS,GACnBoY,EAAUpY,OAAS,IAESwY,EAC5B,SAGEF,EACFjB,GAAyB,EAEzBC,GAA4B,EAG9B,MAAMpB,EAAW7D,GAAcxC,GACzByD,EAAQZ,GAAgB7C,GACxB8I,EAAeb,EACfc,EAAyB,GAE3BnF,EAAWzT,OAAS,GACtB4Y,EAAavY,KAAK,OAAOoT,EAAWpE,KAAK,QAEvCkJ,GACFK,EAAavY,KAAK,QAAQwX,KAExBI,GACFW,EAAavY,KAAK,OAAO2X,KAEvBG,EAAUnY,OAAS,GACrB4Y,EAAavY,KAAK,QAAQ8X,EAAU9I,KAAK,QAEvC+I,EAAUpY,OAAS,GACrB4Y,EAAavY,KAAK,QAAQ+X,EAAU/I,KAAK,QAEvCsJ,EAAa3Y,OAAS,EACxB4Y,EAAavY,KAAK,OAAOsY,EAAatJ,KAAK,QAClC6I,GACTU,EAAavY,KAAK,sBAGpB,MAEMqY,EAAO,KAAKhH,KAAOwE,IADvB0C,EAAa5Y,OAAS,EAAI,KAAK4Y,EAAavJ,KAAK,SAAW,OACViE,KAE9CuF,EACgB,EAApBpF,EAAWzT,QACVsY,EAAsB,EAAI,IAC1BC,EAAoB,EAAI,IACxBN,EAAa,EAAI,IACjBC,EAAmB,EAAI,IACvBC,EAAUnY,OAAS,EAAI,EAAI,IAC3BoY,EAAUpY,OAAS,EAAI,EAAI,IAC3BqY,EAAoB,EAAI,GAErBS,EAAWlC,EAAuB/H,IAAI6J,KACvCI,GAAYD,EAAQC,EAASD,QAChCjC,EAAuBhI,IAAI8J,EAAM,CAAEA,OAAMG,QAAOjB,SAEpD,CAEA,MAAMpJ,EAAgBpQ,MAAM6Q,KAAK2H,EAAuBtG,UACrD7P,KAAK,CAAC1C,EAAGoD,IAAMA,EAAE0X,MAAQ9a,EAAE8a,OAAS9a,EAAE6Z,MAAQzW,EAAEyW,OAChDvV,MAAM,ECh4BsB,IDi4B5BkD,IAAKwT,GAAcA,EAAUL,MAE1BM,EAAwB,IAAIhL,IAChC5P,MAAM6Q,KAAK6H,EAAW5H,WAAWrI,OAAO,EAAEgR,KACxCrK,GAAkBoH,IAAIiD,KAIpBoB,EAAqB,CACzB,wBAAwB5B,mBAAuCC,sBAA6CK,IAC5G,qBAAqB7I,GAAgBiI,EAAa,KAClD,mCAAmCQ,qBAAmCC,iBAAqCC,iBAAgCC,IAC3I,iBAAiB5I,GAAgBkK,EAAuB,KACxD,oBAAoBlK,GAAgBoI,EAAmB,MACvD,iHAGIgC,EAvXuB,MAC7B,MAAMhE,EAAO3Q,SAASyL,MAAQzL,SAASqB,gBAMvC,OALsBxH,MAAM6Q,KAAKiG,EAAKvV,UACnCkH,OAAQ8N,IAAWtH,GAAiBuH,IAAID,EAAMhD,QAAQvP,gBACtDyE,OAAQ8N,GAAUjE,GAAUiE,IAC5BtS,MAAM,EC9gBuB,GDghBXkD,IAAKoP,GACxBL,GAAkBK,EC9gBU,KD63BTwE,GACfC,EAAgB,CACpB,kBAAkB3C,EAAYzW,mBAAmB0W,EAAgB1W,mBAAmBmX,gBAA8BC,IAAWX,EAAYzW,OAAS0W,EAAgB1W,OAAS,gBAAkB,KAC7L,kBAAkB8O,GAAgB+H,EAAW,MAC7C,mBAAmB/H,GAAgBgI,EAAY,KAC/C,6BAA6BhI,GAAgBkI,EAAe,iBAAiBlI,GAAgBmI,EAAgB,MAC7G,oBAAoBiC,EAAalZ,OAAS,EAAIkZ,EAAa7J,KAAK,QAAU,UAG5E,MAAO,CACLd,MAAOA,EAAMlM,MAAM,EC/5BE,IDg6BrBmM,gBACAyK,qBACAI,eAAgBhD,KAChB+C,kBA4CSE,GAAiB,KAC5B,GAAsB,oBAAXzI,QAA8C,oBAAbtM,SAC1C,MAAO,CACLgK,MAAO,GACPC,cAAe,GACfF,QAAS,IAtvBiB,MAC9B,IAAIL,IAA2C,oBAAnBsL,eAA5B,CAIAtL,IAAgB,EAEhB,IACE,MAAMuL,EAAMD,eAAeE,QAAQrM,IACnC,IAAKoM,EACH,OAGF,MAAMjK,EAASmK,KAAKC,MAAMH,GAC1B,IAAKpb,MAAMC,QAAQkR,GACjB,OAGF,IAAA,MAAWvM,KAASuM,EACbpB,GAAanL,IAhMkB,IAmMhCA,EAAM0L,SAGVX,GAAiBa,IAAI5L,EAAMqL,IAAKrL,GAE9B+K,GAAiBiB,KAAO,GAC1B4K,QAAQC,KACN,0CAA0C9L,GAAiBiB,OAGjE,OAAS8K,GACPF,QAAQG,KAAK,uCAAwCD,EACvD,CA/BA,GAuvBAE,GACA,MAAM3L,EAx0Ba,CAAC4L,IACpB,IACE,OAAO,IAAI7I,IAAI6I,EAAQA,GAAQ9I,IACjC,CAAA,MACE,OAAO8I,CACT,GAm0BYC,CAAarJ,OAAOQ,SAASF,MACnCgJ,EAASpM,GAAiBc,IAAIR,GACpC,GAAI8L,EAEF,OADAP,QAAQC,KAAK,iCAAiCxL,KACvC,CACLE,MAAO4L,EAAO5L,MACdC,cAAe2L,EAAO3L,cACtBF,QAAS8B,GAAwB+J,IAIrCP,QAAQC,KAAK,kCAAkCxL,KAE/C,MAAM+L,EAAW5D,KACX6D,EAAWjc,MAAM6Q,KAAK1K,SAASgR,iBAAiB,eACnD1O,OAAQgJ,GAAYa,GAAUb,IAC9BtK,IAAKsK,GACJ,KAAK3B,GAAoB2B,EAAQ+C,aAAe,OAEjD/L,OAAQ6R,GAAkB,OAATA,GACjBrW,MAAM,ECv+BiB,IDi6BW,IACrCiY,EAsFA,MAAM/J,EAAgC,CACpClC,MACAC,SAxFFgM,EAuEwC,CACtCjM,MACAlG,MAAO5D,SAAS4D,MAChBoS,KAAMhW,SAASqB,gBAAgB2U,KAC/BF,WACAG,UA/YyD,CACzD,CAAElH,MAAO,OAAQ4C,SAAU,uBAC3B,CAAE5C,MAAO,MAAO4C,SAAU,4BAC1B,CAAE5C,MAAO,UAAW4C,SAAU,WAC9B,CAAE5C,MAAO,UAAW4C,SAAU,WAC9B,CAAE5C,MAAO,OAAQ4C,SAAU,QAC3B,CAAE5C,MAAO,IAAK4C,SAAU,KACxB,CAAE5C,MAAO,SAAU4C,SAAU,UAC7B,CAAE5C,MAAO,QAAS4C,SAAU,SAC5B,CAAE5C,MAAO,mBAAoB4C,SAAU,kCACvC,CAAE5C,MAAO,gBAAiB4C,SAAU,aACpC,CACE5C,MAAO,oBACP4C,SACE,kFAEJ,CAAE5C,MAAO,WAAY4C,SAAU,cAC/B,CAAE5C,MAAO,kBAAmB4C,SAAU,4BACtC,CAAE5C,MAAO,uBAAwB4C,SAAU,sBAG/B3Q,IACZ,EAAG+N,QAAO4C,cAAe,KAAK5C,MAAU/O,SAASgR,iBAAiBW,GAAUlW,UA0X5EuO,MAAO6L,EAAS7L,MAChBC,cAAe4L,EAAS5L,cACxByK,mBAAoBmB,EAASnB,mBAC7BI,eAAgBe,EAASf,eACzBD,cAAegB,EAAShB,cACxBqB,aAAcxF,KACdyF,iBAxcUnW,SAASyL,MAAM2K,WAAapW,SAASqB,gBAAgB+U,WAG9DxY,QAAQ,8BAA+B,IACvCA,QAAQ,4BAA6B,IACrCA,QAAQ,kCAAmC,IAC3CA,QAAQ,mBAAoB,IAC5BA,QAAQ,OAAQ,KAChByG,OAGAzG,QAAQ,WAAY,MACpBA,QAAQ,OAAQ,KAChByG,QA0Wc,CACfmM,GAAc,OAAQ,CACpB,UAAUuF,EAAMjM,KAAO,YACvB,YAAYiM,EAAMnS,OAAS,YAC3B,WAAWmS,EAAMC,MAAQ,cAE3BxF,GAAc,WAAYuF,EAAMD,UAChCtF,GAAc,oBAAqBuF,EAAME,WACzCzF,GAAc,sBAAuBuF,EAAMrB,oBAC3ClE,GAAc,+BAAgCuF,EAAMjB,gBACpDtE,GAAc,4BAA6BuF,EAAMlB,eACjDrE,GAAc,YAAauF,EAAM/L,OACjCwG,GAAc,oBAAqBuF,EAAM9L,eACzCuG,GAAc,wBAAyBuF,EAAMG,cAC7C1F,GAAc,qBAAsB,CAClC,KAAKuF,EAAMI,iBAAmB,mBAIlBrL,KAAK,SAoEnBd,MAAO6L,EAAS7L,MAChBC,cAAe4L,EAAS5L,cACxBC,WAAYmM,KAAKC,MACjBnM,QAn9BsC,GA69BxC,OAPAX,GAAiBa,IAAIP,EAAKkC,GAnvBI,MAC9B,GAAIxC,GAAiBiB,MCzQS,GD0Q5B,OAGF,MAAM8L,EAAS1c,MAAM6Q,KAAKlB,GAAiBuC,UAAU7P,KACnD,CAAC1C,EAAGoD,IAAMpD,EAAE0Q,WAAatN,EAAEsN,YAEvBqG,EAAWgG,EAAO9a,OChRM,GDiR9B,IAAA,IAASxC,EAAI,EAAGA,EAAIsX,EAAUtX,GAAK,EACjCuQ,GAAiBgN,OAAOD,EAAOtd,GAAG6Q,MA0uBpC2M,GArwB4B,MAC5B,GAA8B,oBAAnBzB,eAIX,IACE,MAAM0B,EAAavB,KAAKwB,UACtB9c,MAAM6Q,KAAKlB,GAAiBuC,UAAU7P,KACpC,CAAC1C,EAAGoD,IAAMpD,EAAE0Q,WAAatN,EAAEsN,aAG/B8K,eAAe4B,QAAQ/N,GAAwB6N,EACjD,OAASnB,GACPF,QAAQG,KAAK,uCAAwCD,EACvD,GAwvBAsB,GACAxB,QAAQC,KACN,oCAAoCxL,UAAYN,GAAiBiB,QAG5D,CACLT,MAAOgC,EAAMhC,MACbC,cAAe+B,EAAM/B,cACrBF,QAAS8B,GAAwBG,KEvhC/B8K,GAAkB,wBACXC,GAA0B,IAEjCC,GAAgB,+BAMhBC,GAAqB,qBAkBdC,GACXrY,IAEA,GAA4B,oBAAjBsY,aACX,IACEA,aAAaP,QACXK,GACA9B,KAAKwB,UAAU,IAAK9X,EAAOuY,QAASf,KAAKC,QAE7C,CAAA,MAEA,GAoBWe,GAA0B,KACT,oBAAjBF,cACXA,aAAaG,WAAWL,KAoNbM,GAAQ,CAAC9Y,EAAe+Y,EAAaC,IAChDC,KAAKF,IAAIC,EAAKC,KAAKD,IAAID,EAAK/Y,IA6BjBkZ,GAAwBC,MACnCC,EACAC,EA9SgC,OAgThC,GAAsB,oBAAXxL,OACT,OAGF,MAAMyL,EAASzL,OAAO0L,QAChBC,EAAQJ,EAAUE,EACpBL,KAAKQ,IAAID,GAAS,SAIhB,IAAIpW,QAAeE,IACvB,MAAMoW,EACJ7L,OAAO9J,uBAAA,CACL4V,GAAmC9L,OAAOtK,WAAW,IAAMoW,EAASC,YAAY/B,OAAQ,KAEtFgC,EAAYD,YAAY/B,MAExBiC,EAAQjC,IACZ,MACMkC,EAAWjB,IADDjB,EAAMgC,GACWR,EAAY,EAAG,GAC1CW,EA5CiB,CAACD,KAC1Bd,KAAKgB,IAAIhB,KAAKiB,GAAKH,GAAY,GAAK,EA2CpBI,CAAcJ,GAC5BlM,OAAOuM,SAAS,EAAGd,EAASE,EAAQQ,GAEhCD,EAAW,EACbL,EAAII,GAEJxW,KAIJoW,EAAII,MAIFO,GAAoB,CAAC9R,EAAqB3L,EAAW1B,KACzDqN,EAAOtJ,MAAMqb,KAAO,GAAG1d,MACvB2L,EAAOtJ,MAAM4J,IAAM,GAAG3N,OAQlBqf,GAAsB,CAAChS,EAAqBiS,KAChDjS,EAAOtJ,MAAMoS,QAAUmJ,EAAU,IAAM,KAGnCC,GAAoB5N,IACxB,MAAM5N,EAAQ4O,OAAOC,iBAAiBjB,GACtC,MAAyB,SAAlB5N,EAAM+G,SAA2C,WAArB/G,EAAM8O,YAA6C,MAAlB9O,EAAMoS,SAwBtEqJ,GAA2B,KAC/B,MAAMC,EAtBoB,MAC1B,MAAMA,EAAsB,GAEtBC,EAAcrZ,SAASsZ,eAAe,mBACxCD,GAAaE,YACfH,EAAMtd,KAAKud,EAAYE,YAGzB,MAAMrH,EAAclS,SAASgR,iBAA8B,KAC3D,IAAA,MAAWwI,KAAMtH,EACVsH,EAAGD,aAGHH,EAAMK,SAASD,EAAGD,aACrBH,EAAMtd,KAAK0d,EAAGD,aAIlB,OAAOH,GAIOM,GAEd,IAAA,MAAW/I,KAAQyI,EAAO,CACxB,MAAMO,EAAQhJ,EAAKC,cAA2B,sBAC9C,GAAI+I,GAAST,GAAiBS,GAC5B,OAAOA,CAEX,CAEA,IAAA,MAAWhJ,KAAQyI,EAAO,CACxB,MAAMQ,EAASjJ,EAAKC,cAA2B,wBAC/C,GAAIgJ,GAAUV,GAAiBU,GAC7B,OAAOA,CAEX,CAEA,OAAO,MAGHC,GAAwB,KAC5B,MAAMC,EAASX,KACf,IAAKW,EACH,MAAO,CACLze,EAAG0e,KACHpgB,EAAGogB,MAIP,MAAMtN,EAAOqN,EAAOpN,wBACpB,MAAO,CACLrR,EAAGoR,EAAKsM,KAAOzM,OAAO0N,QAAUD,KAChCpgB,EAAG8S,EAAKnF,IAAMgF,OAAO0L,QAAU+B,OAItBE,GAAkB,KAC7B,GAAwB,oBAAbja,UAA8C,oBAAXsM,OAC5C,OAGF,MAAMtF,EAAShH,SAASsZ,eAAexC,IACjC9P,aAAkBoF,aAIxB4M,GAAoBhS,GAAQ,IAG9B,IAAIkT,IAAiC,EACrC,MAAMC,GAAgC,KAChCD,KAGJA,IAAiC,EAEjCla,SAASjC,iBAAiB,YAAcK,IACtC,MAAM4I,EAAShH,SAASsZ,eAAexC,IACvC,KAAM9P,aAAkBoF,aACtB,OAGF,GAA6B,MAAzBpF,EAAOtJ,MAAMoS,QACf,OAGF,MAAQzU,EAAAA,EAAG1B,EAAAA,GArGW,CAACqN,IAAA,CACzB3L,EAAG4P,OAAOmP,WAAWpT,EAAOtJ,MAAMqb,OAAS,EAC3Cpf,EAAGsR,OAAOmP,WAAWpT,EAAOtJ,MAAM4J,MAAQ,IAmGvB+S,CAAkBrT,GAC7BsT,EAAWlc,EAAMmc,MACjBC,EAAWpc,EAAMqc,MACN/C,KAAKgD,MAAMJ,EAAWjf,EAAGmf,EAAW7gB,IA7b1B,IAgczBqf,GAAoBhS,GAAQ,OAkD5B2T,GAAa/C,MAAOvc,EAAW1B,KACnC,MAAMqN,EA9Ca,MACnB,MAAMuN,EAAWvU,SAASsZ,eAAexC,IACzC,GAAIvC,EAIF,OAFAA,EAAS7W,MAAM6J,WAAa5E,GAAOC,QACnCuX,KACO5F,EAGT,MAAMvN,EAAShH,SAAS4a,cAAc,OACtC5T,EAAOqG,GAAKyJ,GACZ9P,EAAOtJ,MAAMuI,SAAW,WACxB,MAAM4U,EAAgBhB,KACtB7S,EAAOtJ,MAAMqb,KAAO,GAAG8B,EAAcxf,MACrC2L,EAAOtJ,MAAM4J,IAAM,GAAGuT,EAAclhB,MACpCqN,EAAOtJ,MAAMoS,QAAU,IAEvB9I,EAAOtJ,MAAMgG,MAAQ,OACrBsD,EAAOtJ,MAAMiG,OAAS,OACtBqD,EAAOtJ,MAAMkJ,aAAe,MAC5B,MAAMkU,EAAYnY,GAAOC,QAazB,OAZAoE,EAAOtJ,MAAM6J,WAAauT,EAE1B9T,EAAOtJ,MAAMqJ,OAAS,oBACtBC,EAAOtJ,MAAMqd,UAAY,oCACzB/T,EAAOtJ,MAAMsd,UAAY,aACzBhU,EAAOtJ,MAAM6I,OAAS,aACtBS,EAAOtJ,MAAMmS,cAAgB,OAC7B7I,EAAOtJ,MAAMiL,UAAY,wBACzB3B,EAAOtJ,MAAMwJ,WAAa,cAAqC8P,iBAAmDA,6BAClHhX,SAASyL,KAAKwP,YAAYjU,GAC1BmT,KACA9E,QAAQC,KAAK,gCAAgCwF,oBACtC9T,GAaQkU,GACwB,SAAnClU,EAAOmU,QAAQC,kBACjBpU,EAAOmU,QAAQC,gBAAkB,YAZHxD,WAChC,MAAMO,EACJ7L,OAAO9J,uBAAA,CACL4V,GAAmC9L,OAAOtK,WAAW,IAAMoW,EAASC,YAAY/B,OAAQ,WACtF,IAAIzU,QAAeE,IACvBoW,EAAI,IAAMpW,QAQJsZ,IAGR,MAAMR,EAAgBhB,KACtBb,GAAoBhS,GAAQ,GAC5B8R,GAAkB9R,EAAQ6T,EAAcxf,EAAGwf,EAAclhB,SACnD,IAAIkI,QAASE,GAAYC,WAAWD,EAASgV,KAEnD+B,GAAkB9R,EAAQ3L,EAAG1B,SACvB,IAAIkI,QAASE,GAAYC,WAAWD,EAASgV,MAG/CuE,GAAoBhQ,IACxB,MAAMmB,EAAOnB,EAAQoB,wBACrB,MAAO,CACLrR,EAAGoR,EAAKsM,KAAOzM,OAAO0N,QAAUvN,EAAK/I,MAAQ,EAC7C/J,EAAG8S,EAAKnF,IAAMgF,OAAO0L,QAAUvL,EAAK9I,OAAS,IAI3C4X,GAA4B,uCAE5BC,GAAyB7J,IAC7B,IACE,OAAO3R,SAAS4Q,cAAce,EAChC,OAAS4D,GACP,MAAMkG,EAAgB9J,EAASpN,MAAMgX,IACrC,IAAKE,EAEH,OADApG,QAAQG,KAAK,8BAA8B7D,IAAY4D,GAChD,KAGT,MAAMmG,EAAeD,EAAc,IAAIpX,QAAU,IAC3CsX,EAAeF,EAAc,IAAIpX,QAAU,GACjD,IAAKsX,EAEH,OADAtG,QAAQG,KAAK,0CAA0C7D,KAChD,KAGT,IACE,MAAMZ,EAAa/Q,SAASgR,iBAAiB0K,GAC7C,IAAA,MAAWlH,KAAazD,EACtB,GAAIyD,EAAUnG,aAAaoL,SAASkC,GAClC,OAAOnH,EAGX,OAAO,IACT,OAASoH,GAEP,OADAvG,QAAQG,KAAK,uCAAuC7D,IAAYiK,GACzD,IACT,CACF,GA6BIC,GAAqB,CACzBvQ,EACAjR,EACAgB,EACA1B,KAEA2R,EAAQwQ,cACN,IAAIC,WAAW1hB,EAAM,CACnB2hB,SAAS,EACTC,YAAY,EACZC,KAAM5P,OACN6P,QAAS9gB,EAAIiR,OAAO0N,QACpBoC,QAASziB,EAAI2S,OAAO0L,YA0BpBqE,GAAkB,CAAC/Q,EAAsBxI,IACzCwI,aAAmBkD,kBAAoBlD,aAAmBmD,qBAC5DnD,EAAQgR,QAvBoB,EAC9BhR,EACAxI,KAEA,MAAMxE,EACJgN,aAAmBmD,oBACfA,oBAAoBnQ,UACpBkQ,iBAAiBlQ,UACjBie,EAAa/T,OAAOgU,yBAAyBle,EAAW,SAC1Die,GAAYlS,IACdkS,EAAWlS,IAAI/J,KAAKgL,EAASxI,GAE7BwI,EAAQ7M,MAAQqE,EAKlBwI,EAAQmR,aAAe3Z,EACvBwI,EAAQpN,aAAa,QAAS4E,IAM5B4Z,CAAwBpR,EAASxI,GACjCwI,EAAQwQ,cAAc,IAAIa,MAAM,QAAS,CAAEX,SAAS,UACpD1Q,EAAQwQ,cAAc,IAAIa,MAAM,SAAU,CAAEX,SAAS,MAInD1Q,EAAQwI,mBACVxI,EAAQgR,QACRhR,EAAQ+C,YAAcvL,OACtBwI,EAAQwQ,cAAc,IAAIa,MAAM,QAAS,CAAEX,SAAS,WAItD3G,QAAQG,KACN,uFAwBEoH,GAA4BhF,MAAOtM,UACjCuR,GAAkCvR,GAAS,IAG7CuR,GAAoCjF,MACxCtM,EACAwR,KAEA,MAAMrQ,EAAOnB,EAAQoB,wBACfqQ,EAAiBzQ,OAAO0Q,YAE9B,IAAKF,IAvY8B,EACnCrQ,EACAsQ,IACYtQ,EAAKnF,IAAM,GAAKmF,EAAKtG,OAAS4W,EAoYrBE,CAAsBxQ,EAAMsQ,GAC/C,OAGF,MAAMG,EAAexF,KAAKD,IACxB,EACAC,KAAKD,IAAIzX,SAASyL,KAAK0R,aAAcnd,SAASqB,gBAAgB8b,cAAgBJ,GAE1ElF,EA1YgC,EACtCuF,EACAC,EACAC,EACAP,EACAG,IAIO3F,GADL6F,EAAiBC,GAAWN,EAAiB,EAAIO,EAAa,GAC1C,EAAG5F,KAAKD,IAAI,EAAGyF,IAiYrBK,CACdjR,OAAO0L,QACPvL,EAAKnF,IACLmF,EAAK9I,OACLoZ,EACAG,SAGIvF,GAAsBE,EAlrBI,MAisB5B2F,GAAkB5F,MAAOtX,IAC7B,MAAMmd,EApJc,CAACnd,IACrB,GAAIA,EAAKqR,SAAU,CACjB,MAAM+L,EAAWlC,GAAsBlb,EAAKqR,UAE5C,GAAI+L,aAAoBtR,YAAa,CACnC,MAAMuR,EAASrC,GAAiBoC,GAChC,MAAO,CACLpS,QAASoS,EACTriB,EAAGsiB,EAAOtiB,EACV1B,EAAGgkB,EAAOhkB,EAEd,CACA0b,QAAQG,KAAK,0CAA0ClV,EAAKqR,WAC9D,CAEA,MAAsB,iBAAXrR,EAAKjF,GAAoC,iBAAXiF,EAAK3G,EACrC,CACL0B,EAAGiF,EAAKjF,EACR1B,EAAG2G,EAAK3G,IAIZ0b,QAAQG,KAAK,6DAA8DlV,GACpE,OA6HQsd,CAActd,GAC7B,GAAKmd,EAAL,CAIA,GAAoB,UAAhBnd,EAAKud,QAAsBJ,EAAOnS,QAAS,OACvCsR,GAA0Ba,EAAOnS,SACvC,MAAMqS,EAASrC,GAAiBmC,EAAOnS,SACvCmS,EAAOpiB,EAAIsiB,EAAOtiB,EAClBoiB,EAAO9jB,EAAIgkB,EAAOhkB,CACpB,CAIA,SAFMghB,GAAW8C,EAAOpiB,EAAGoiB,EAAO9jB,GAEd,SAAhB2G,EAAKud,OAAT,CAIA,GAAKJ,EAAOnS,QAKZ,MAAoB,UAAhBhL,EAAKud,QACPhC,GAAmB4B,EAAOnS,QAAS,cAAemS,EAAOpiB,EAAGoiB,EAAO9jB,GACnEkiB,GAAmB4B,EAAOnS,QAAS,YAAamS,EAAOpiB,EAAGoiB,EAAO9jB,GACjEkiB,GAAmB4B,EAAOnS,QAAS,YAAamS,EAAOpiB,EAAGoiB,EAAO9jB,GACjEkiB,GAAmB4B,EAAOnS,QAAS,UAAWmS,EAAOpiB,EAAGoiB,EAAO9jB,QAC/D8jB,EAAOnS,QAAQwS,cAIG,SAAhBxd,EAAKud,OA/FW,CAACvS,IACrB,GAAwB,SAApBA,EAAQ8B,QAEV,YADC9B,EAA4ByS,gBAI/B,GAAwB,WAApBzS,EAAQ8B,SAAyB9B,EAA8B0S,KAEjE,YADC1S,EAA8B0S,MAAMD,gBAIvC,MAAME,EAAa3S,EAAQ4S,QAAQ,QAC/BD,EACFA,EAAWF,gBAIb1I,QAAQG,KAAK,6DAmFb2I,CAAcV,EAAOnS,SAJnB+Q,GAAgBoB,EAAOnS,QAAShL,EAAKwC,MAAQ,KAd7CuS,QAAQG,KAAK,8DAA+DlV,EAAKud,OAHnF,CAbA,GAgKIO,GAAkBxG,MAAOtX,IAC7B,IACE,MAAM+d,EAAY/d,EAAKwJ,IACvB,IAAIwU,EACJ,IACEA,EAAc,IAAIzR,IAAIwR,EAAW/R,OAAOQ,SAASF,MAAMA,IACzD,CAAA,MACE0R,EAAcD,CAChB,CAEA,MAAME,EAtHwB,CAACF,IACjC,IAAIG,EAA2B,KAC/B,IACEA,EAAe,IAAI3R,IAAIwR,EAAW/R,OAAOQ,SAASF,KACpD,CAAA,MAEA,CAEA,MAAM6R,EAAW5kB,MAAM6Q,KACrB1K,SAASgR,iBAAiB,8CAI5B,IAAA,MAAWwI,KAAMiF,EACf,GAAIjF,aAAckF,mBAAqBlF,EAAG5M,OAAS4R,GAAc5R,KAC/D,OAAO4M,EAIX,GAAIgF,EAAc,CAEhB,IAAA,MAAWhF,KAAMiF,EACf,GAAMjF,aAAckF,kBACpB,IACE,MAAMC,EAAQ,IAAI9R,IAAI2M,EAAG5M,KAAMN,OAAOQ,SAASF,MAC/C,GACE+R,EAAMC,WAAaJ,EAAaI,UAChCD,EAAME,SAAWL,EAAaK,QAC9BF,EAAMG,OAASN,EAAaM,KAE5B,OAAOtF,CAEX,CAAA,MACE,QACF,CAIF,IAAA,MAAWA,KAAMiF,EACf,GAAMjF,aAAckF,kBACpB,IAEE,GADc,IAAI7R,IAAI2M,EAAG5M,KAAMN,OAAOQ,SAASF,MACrCgS,WAAaJ,EAAaI,SAClC,OAAOpF,CAEX,CAAA,MACE,QACF,CAIF,MAAM9D,EAAS2I,EAAUzgB,QAAQ,MAAO,IACxC,IAAA,MAAW4b,KAAMiF,EAAU,CACzB,MAAM7R,EAAO4M,EAAG7N,aAAa,SAAW6N,EAAG7N,aAAa,cAAgB,GACxE,GAAIiB,IAASA,IAASyR,GAAazR,IAAS8I,GAAU9I,IAAS,IAAI8I,KACjE,OAAO8D,CAEX,CACF,CAGA,MAAMuF,EAAcV,EACjBzgB,QAAQ,oBAAqB,IAC7BA,QAAQ,UAAW,IACnBgO,MAAM,KACNtJ,OAAOqL,SACJqR,EAAcD,EAAYA,EAAYtjB,OAAS,IAAM,GAE3D,GAAIujB,EAAa,CAEf,IAAIC,EAAc,CAACD,GACnB,GAAIR,EAAc,CAChB,IAAA,MAAW,CAAG/f,KAAU+f,EAAaU,aAAavU,UAC5ClM,GAAOwgB,EAAYnjB,KAAK2C,GAE1B+f,EAAaM,MACfG,EAAYnjB,KAAK0iB,EAAaM,KAAKlhB,QAAQ,KAAM,IAErD,CACAqhB,EAAcA,EAAYje,IAAK5G,GAAMA,EAAEyD,eAGvC,MAAMshB,EAAatlB,MAAM6Q,KACvB1K,SAASgR,iBACP,mFAIJ,IAAA,MAAWwI,KAAM2F,EAAY,CAC3B,MAAMrc,GAAQ0W,EAAGnL,aAAe,IAAIhK,OAAOxG,cACrCyQ,GAAakL,EAAG7N,aAAa,eAAiB,IAAI9N,cAClDuhB,GAAW5F,EAAG7N,aAAa,aAAe,IAAI9N,cACpD,IAAA,MAAWwhB,KAAQJ,EACjB,GACEnc,IAASuc,GACT/Q,IAAc+Q,GACdD,IAAYC,GACZvc,EAAK2W,SAAS4F,GAEd,OAAO7F,CAGb,CACF,CAEA,OAAO,MAamB8F,CAA0BjB,GAElD,GAAIE,EAAiB,CACnBlJ,QAAQkK,IAAI,sCAAuCjB,EAAaC,EAAgBnR,eAC1EwP,GAA0B2B,GAEhC,MAAMZ,EAASrC,GAAiBiD,GAShC,aARM5D,GAAWgD,EAAOtiB,EAAGsiB,EAAOhkB,GAElC4kB,EAAgBzC,cAAc,IAAIC,WAAW,cAAe,CAAEC,SAAS,EAAME,KAAM5P,UACnFiS,EAAgBzC,cAAc,IAAIC,WAAW,YAAa,CAAEC,SAAS,EAAME,KAAM5P,UACjFiS,EAAgBzC,cAAc,IAAIC,WAAW,YAAa,CAAEC,SAAS,EAAME,KAAM5P,UACjFiS,EAAgBzC,cAAc,IAAIC,WAAW,UAAW,CAAEC,SAAS,EAAME,KAAM5P,UAC/EiS,EAAgBT,SAlJO,CAACO,IAC5B,IACE,MAAMpd,EAAU,IAAI4L,IAAIP,OAAOQ,SAASF,MAClC6Q,EAAS,IAAI5Q,IAAIwR,GACvB,OAAOpd,EAAQue,SAAW/B,EAAO+B,QAAUve,EAAQ2d,WAAanB,EAAOmB,QACzE,CAAA,MACE,OAAO,CACT,GA6IYa,CAAqBnB,EAC/B,CAGAjJ,QAAQkK,IAAI,wEAAyEjB,GAGrF,IACE,MAAMtT,EAAS,IAAI6B,IAAIyR,GACvB,GACEtT,EAAOwU,SAAWlT,OAAOQ,SAAS0S,QAClCxU,EAAO4T,WAAatS,OAAOQ,SAAS8R,UACpC5T,EAAO8T,KAGP,OADAxS,OAAOQ,SAASgS,KAAO9T,EAAO8T,MACvB,CAEX,CAAA,MAAyB,CAGzB,IACE,MAAM9T,EAAS,IAAI6B,IAAIyR,GACvB,GAAItT,EAAOwU,SAAWlT,OAAOQ,SAAS0S,OAAQ,CAE5C,MAAME,EAAU1U,EAAO4T,SAAW5T,EAAO6T,OAAS7T,EAAO8T,KAGzD,OAFAxS,OAAOqT,QAAQC,UAAU,CAAA,EAAI,GAAIF,GACjCpT,OAAOwP,cAAc,IAAI+D,cAAc,WAAY,CAAEhhB,MAAO,CAAA,MACrD,CACT,CACF,CAAA,MAAyB,CAIzB,OADAyN,OAAOQ,SAASF,KAAO0R,GAChB,CACT,OAAS/I,GAEP,OADAF,QAAQG,KAAK,2BAA4BlV,EAAKwJ,IAAKyL,IAC5C,CACT,GAmDWuK,GAAwBlI,MACnCtX,IAEA,MAAMyf,EAASzf,EAAK0f,QACpB,IACE,GAAkB,aAAd1f,EAAK2f,KAEP,aADMzC,GAAgBld,GACf,CACL0f,QAASD,EACTG,OAAQ,uBAAuB5f,EAAKud,UAIxC,GAAkB,WAAdvd,EAAK2f,KAEP,YAhTgBrI,OAAOtX,IAC3B,MAAMod,EAAWlC,GAAsBlb,EAAKqR,UAC5C,KAAM+L,aAAoBtR,aAExB,YADAiJ,QAAQG,KAAK,wCAAwClV,EAAKqR,kBAItDkL,GAAkCa,GAAU,GAClD,MAAMC,EAASrC,GAAiBoC,SAC1B/C,GAAWgD,EAAOtiB,EAAGsiB,EAAOhkB,IAsSxBwmB,CAAc7f,GACb,CACL0f,QAASD,EACTG,OAAQ,4BAIZ,GAAkB,mBAAd5f,EAAK2f,KAA2B,CAElC,MAAO,CACLD,QAASD,EACTG,OAHcnL,KAGEhL,QAEpB,CAEA,GAAkB,aAAdzJ,EAAK2f,KAAqB,OACtB7B,GAAgB9d,SAEhB,IAAIuB,QAASE,GAAYC,WAAWD,EAAS,OACnD,MAAMxG,EAAUwZ,KAChB,MAAO,CACLiL,QAASD,EACTG,OAAQ,yCAAyC5T,OAAOQ,SAASF,wBAAwBrR,EAAQwO,UAErG,CAEA,MAAO,CAAEiW,QAASD,EAAQG,OAAQ,mBACpC,OAAS3K,GACP,MAAM6K,EAAM7K,aAAiB8K,MAAQ9K,EAAM+K,QAAUzjB,OAAO0Y,GAE5D,OADAF,QAAQG,KAAK,iCAAiClV,EAAK2f,OAAQ1K,GACpD,CAAEyK,QAASD,EAAQG,OAAQ,SAASE,IAC7C,GCh9BWG,GAAyB,CAAC,IAAK,IAAK,MAE3CC,GAAyB,mBACzBC,sBAA0B1X,IAChC,IAAI2X,GAA0B,EAE9B,MAAMC,GAAoBC,IACxB,MAAMC,EAAUD,EAAQvc,OAAOzG,QAAQ,OAAQ,IAC/C,MAAI,gBAAgBJ,KAAKqjB,GAChBA,EAGF,WAAWA,KAEdC,GAAiB,CAACF,EAAiBG,KACvC,MAAMC,EAAaL,GAAiBC,GAC9B9W,EAAM,IAAI+C,IAAImU,GAKpB,OAJAlX,EAAImX,SAA4B,WAAjBnX,EAAImX,SAAwB,OAAS,MACpDnX,EAAI8U,SAAW,GAAG9U,EAAI8U,SAAShhB,QAAQ,MAAO,MAAMmjB,IACpDjX,EAAI+U,OAAS,GACb/U,EAAIgV,KAAO,GACJhV,EAAIoX,YAyCAC,GAAuB5L,GACb,iBAAVA,GAAgC,OAAVA,KAAkB,cAAeA,IACzD5H,QAAS4H,EAAkC6L,WAKhDC,GAAiBzJ,MAAO0J,IAC5B,IACE,MAAMjhB,QAAcihB,EAASC,OACvBC,EAASnhB,EAAKmhB,OACpB,MAAsB,iBAAXA,EAA4BA,EACnCA,GAA4B,iBAAXA,EAA4BrM,KAAKwB,UAAU6K,GACzDnhB,EAAKkV,OAASlV,EAAKigB,SAAWgB,EAASG,UAChD,CAAA,MACE,OAAOH,EAASG,UAClB,GAGIC,GAASC,GACb,IAAI9f,QAASE,IACXC,WAAWD,EAAS4f,KAWlBC,GAA2BC,GAC/BnB,KAA4BmB,EAEjBC,GAA6B,IAAcpB,GAE3CqB,GAA0B,KACrCrB,IAA2B,EAC3B,MAAMsB,EAASnoB,MAAM6Q,KAAK+V,IAC1B,IAAA,MAAWwB,KAAgBD,EACzB,IACEC,EAAanG,cAAc,IAAIa,MAAM6D,KACrCyB,EAAaC,QACbD,EAAahkB,gBAAgB,OAC7BgkB,EAAaE,MACf,CAAA,MAEA,GAISC,GAAsBC,IAEjC,MAAMC,EAAcD,EAAOzkB,QAAQ,2BAA4B,IACzD2kB,EAAeC,KAAKF,GACpBG,EAAQ,IAAIC,WAAWH,EAAa9mB,QAC1C,IAAA,IAASxC,EAAI,EAAGA,EAAIspB,EAAa9mB,OAAQxC,GAAK,EAC5CwpB,EAAMxpB,GAAKspB,EAAaI,WAAW1pB,GAErC,OAAOwpB,GAsFHG,GAAoBhL,MACxBiL,EACAC,EACAC,EAAqB,KACrBC,KAEA,MAAMC,EAAqBnB,KAC3B,GAAsB,IAAlBe,EAAOpnB,OAET,YADAunB,IAAqB,QAIvB,GAAIpB,GAAwBqB,GAE1B,YADAD,IAAqB,QAKvB,MAAME,EAAaL,EAAOM,OAAO,CAACC,EAAK9pB,IAAM8pB,EAAM9pB,EAAE+pB,WAAY,GACjEhO,QAAQkK,IAAI,mCAAmCsD,EAAOpnB,kBAAkBynB,iBAA0BJ,KAElGE,IAAqB,YAErB,MAAMM,EAA2BT,EAAO7hB,IAAKuiB,IAC3C,MAAMC,EAAS,IAAId,WAAWa,EAAMF,YAEpC,OADAG,EAAOnZ,IAAIkZ,GACJC,EAAOC,SAIhB,IAAIC,EAAeZ,EACnB,GAAID,EAAOpnB,OAAS,GAAKonB,EAAO,GAAGpnB,QAAU,EAAG,CAC9C,MAAMkoB,EAAS9pB,MAAM6Q,KAAKmY,EAAO,GAAG/kB,MAAM,EAAG,IAC1CkD,OAASpE,EAAEskB,SAAS,IAAI0C,SAAS,EAAG,KAAKC,eACzC/Y,KAAK,KACRuK,QAAQkK,IAAI,+BAA+BoE,KAGvCA,EAAOzV,WAAW,aAEXyV,EAAOzV,WAAW,UAAYyV,EAAOzV,WAAW,SADzDwV,EAAe,aAGNC,EAAOzV,WAAW,eAC3BwV,EAAe,YACNC,EAAOzV,WAAW,iBAC3BwV,EAAe,aAEnB,CAIA,IAAII,EAAeJ,GAAgBA,EAAajK,SAAS,KAAOiK,EAAe,aAC3EK,EAA6BT,EAEjC,GAAiB,cAAbR,EAA0B,CAE5B,MACMa,EAlIc,EACtBloB,EACAsnB,EAAqB,QAErB,MAAMU,EAAS,IAAIO,YAAY,IACzB9H,EAAO,IAAI+H,SAASR,GAsB1B,OAlBAvH,EAAKgI,UAAU,EAAG,YAAY,GAC9BhI,EAAKgI,UAAU,EAAG,GAAKzoB,GAAQ,GAC/BygB,EAAKgI,UAAU,EAAG,YAAY,GAG9BhI,EAAKgI,UAAU,GAAI,YAAY,GAC/BhI,EAAKgI,UAAU,GAAI,IAAI,GACvBhI,EAAKiI,UAAU,GAAI,GAAG,GACtBjI,EAAKiI,UAAU,GAXE,GAWY,GAC7BjI,EAAKgI,UAAU,GAAInB,GAAY,GAC/B7G,EAAKgI,UAAU,GAbE,EAaEnB,EAAwB,GAAG,GAC9C7G,EAAKiI,UAAU,GAAIC,GAAc,GACjClI,EAAKiI,UAAU,GAAI,IAAI,GAGvBjI,EAAKgI,UAAU,GAAI,YAAY,GAC/BhI,EAAKgI,UAAU,GAAIzoB,GAAQ,GAEpB,IAAIinB,WAAWe,IAuGLY,CADKxB,EAAOM,OAAO,CAACC,EAAK9pB,IAAM8pB,EAAM9pB,EAAE+pB,WAAY,GACtBN,GAC5CgB,EAAiB,CAACJ,EAAOF,UAAWH,GACpCQ,EAAe,YACfzO,QAAQkK,IAAI,wCAAwCwD,KACtD,CAEA1N,QAAQkK,IAAI,oCAAoCuE,gBAA2BhB,MAC3E,MAAMwB,EAAO,IAAIC,KAAKR,EAAgB,CAAE1pB,KAAMypB,IAExC7B,EAAe,IAAIuC,MACnBC,EAAY5X,IAAI6X,gBAAgBJ,GAEtC,IASE,GAxM+B,CAACrC,IAClCxB,GAAoBvP,IAAI+Q,IA+LtB0C,CAA2B1C,GAE3BA,EAAa2C,QAAU,OACvB3C,EAAa4C,UAAW,EAExB5C,EAAa/jB,aAAa,cAAe,QACzC+jB,EAAaxe,IAAMghB,EAEf7C,GAAwBqB,GAE1B,YADAD,IAAqB,cAIjBf,EAAa6C,OACnB9B,IAAqB,gBAhIEpL,OACzBqK,IAEIA,EAAa8C,aAIX,IAAIljB,QAAc,CAACE,EAASijB,KAChC,MAAMC,EAAW3Y,OAAO4Y,YAAY,KAC7BjD,EAAa8C,OAChB1P,QAAQC,KAAK,gDAEd,KAEG6P,EAAU,KACdC,IACArjB,KAGIsjB,EAAU,KACdD,IACAJ,EAAO,IAAI3E,MAAM,gCAGbiF,EAAe,KACnBF,IACArjB,KAGIqjB,EAAU,KACd9Y,OAAOiZ,cAAcN,GACrBhD,EAAajkB,oBAAoB,QAASmnB,GAC1ClD,EAAajkB,oBAAoB,QAASqnB,GAC1CpD,EAAajkB,oBAAoBwiB,GAAwB8E,IAG3DrD,EAAalkB,iBAAiB,QAASonB,GACvClD,EAAalkB,iBAAiB,QAASsnB,GACvCpD,EAAalkB,iBAAiByiB,GAAwB8E,MA2FhDE,CAAmBvD,GACzBe,IAAqB,OACvB,OAASyC,GAGP,MAFApQ,QAAQE,MAAM,4BAA4BkQ,IAAO,CAAE3C,SAAUgB,EAAcrZ,KAAM6Z,EAAK7Z,OACtFuY,IAAqB,QACfyC,CACR,CAAA,QAjNmC,CAACxD,IACpCxB,GAAoBjK,OAAOyL,IAiNzByD,CAA6BzD,GAC7BA,EAAaC,QACbD,EAAahkB,gBAAgB,OAC7BgkB,EAAaE,OACbtV,IAAI8Y,gBAAgBlB,EACtB,GAQWmB,GAAwBC,IACnC,MAAMC,EAAYD,EACfja,MAAM,SACN5K,IAAKmT,GAASA,EAAK9P,QACnB/B,OAAQ6R,GAASA,EAAKjG,WAAW,UACjClN,IAAKmT,GAASA,EAAKrW,MAAM,GAAGioB,aAE/B,GAAyB,IAArBD,EAAUrqB,OACZ,OAAO,KAGT,MAAMuqB,EAAUF,EAAUhb,KAAK,MAC/B,GAAgB,WAAZkb,EACF,MAAO,CAAE3rB,KAAM,QAGjB,IACE,OAAO8a,KAAKC,MAAM4Q,EACpB,OAASzQ,GAEP,OADAF,QAAQG,KAAK,2BAA4BD,GAClC,IACT,GAGW0Q,GACXC,GAEyB,iBAAlBA,EAAQC,aACG,IAAjBD,EAAQ7rB,MAAuC,UAAjB6rB,EAAQ7rB,MA6ClC,MAAM+rB,GAA0B,CACrCxF,EACAyF,EAMAC,EAAsB,CAAA,KAEtB,MAAMC,EAAQzF,GAAeF,EAAS,gBACtCvL,QAAQC,KAAK,+BAAgCiR,GAC7C,MAAMC,EAAS,IAAIC,UAAUF,GAC7B,IAAIG,EAAM,EACNC,EAAY,GACZC,EAAiBP,EAAOQ,WAAa,GACrCC,GAAU,EACVC,GAAU,EAGVC,EAA2BnlB,QAAQE,UAEnCklB,EAAoC,KACpCC,EAAyE,KAC7E,MAAMC,EAAe,IAAItlB,QAAc,CAACE,EAASijB,KAC/CiC,EAAellB,EACfmlB,EAAclC,IAGhB,IAAIoC,EAAsD,KACtDC,EAAwE,KAC5E,MAAMC,EAAc,IAAIzlB,QAAqB,CAACE,EAASijB,KACrDoC,EAAcrlB,EACdslB,EAAarC,IAGTuC,EAAahS,IACbwR,IACJA,GAAU,EACV1R,QAAQG,KAAK,2BAA4BD,EAAM+K,SAC/C4G,IAAc3R,GACd8R,IAAa9R,KAGTiS,EAAwB,KACxBT,GACCJ,EAAUtiB,QAAWuiB,IAC1BG,GAAU,EACVE,MACAG,IAAc,CACZtkB,KAAM6jB,EAAUtiB,OAChBojB,WAAYb,MA8EhB,OA1EAJ,EAAOkB,OAAS,KACdrS,QAAQC,KAAK,2CACbgR,EAAOqB,kBACPnB,EAAOoB,KACLzS,KAAKwB,UAAU,CACbtc,KAAM,QACNwtB,WAAYxB,EAAOyB,UACnBL,WAAYpB,EAAOQ,gBAAa,EAChCkB,SAAU1B,EAAO0B,UAAY,KAC7BC,UAAW3B,EAAOvD,UAAY,iBAKpC0D,EAAOyB,UAAa7pB,IAClB,MAAM8nB,EA5Z4B,CACpCznB,IAEA,IACE,MAAqB,iBAAVA,EACF,KAEF0W,KAAKC,MAAM3W,EACpB,CAAA,MACE,OAAO,IACT,GAkZkBypB,CAAuBrrB,OAAOuB,EAAMiC,OACpD,GAAK6lB,EAAL,CAEA,GAAqB,cAAjBA,EAAQ7rB,MAAsD,iBAAvB6rB,EAAQuB,WAKjD,OAJApS,QAAQC,KAAK,8CAA+C4Q,EAAQuB,YACpEb,EAAiBV,EAAQuB,WACzBnB,EAAO6B,cAAcjC,EAAQuB,iBAC7BR,MAIF,GAAqB,YAAjBf,EAAQ7rB,MAA8C,iBAAjB6rB,EAAQpjB,KAAjD,CAQA,GAAqB,UAAjBojB,EAAQ7rB,MAA4C,iBAAjB6rB,EAAQpjB,KAO7C,OANAuS,QAAQC,KAAK,sCAAuC4Q,EAAQpjB,KAAKhF,MAAM,EAAG,KAC1E6oB,EAAYT,EAAQpjB,UACc,iBAAvBojB,EAAQuB,aACjBb,EAAiBV,EAAQuB,WACzBnB,EAAO6B,cAAcjC,EAAQuB,cAKjC,GAAqB,SAAjBvB,EAAQ7rB,KAIV,OAHAgb,QAAQC,KAAK,uBACbkS,SACAhB,EAAO4B,QAIT,GAAqB,UAAjBlC,EAAQ7rB,KAAkB,CAC5Bgb,QAAQE,MAAM,+BAAgC2Q,EAAQ3Q,OACtD,MAAMkQ,EAAM4C,GAAWnC,EAAQ3Q,OAAS,gBAAsC,IAAtB2Q,EAAQ9E,WAChEmG,EAAU9B,GACVe,EAAO4B,OACT,CAxBA,MALE9B,EAAOgC,YAAY,CACjB5B,IAA4B,iBAAhBR,EAAQQ,IAAmBR,EAAQQ,IAAM,EACrD5jB,KAAMojB,EAAQpjB,MAbJ,GA2ChB0jB,EAAO+B,QAAWC,IAChBnT,QAAQE,MAAM,iCAAkCiT,GAChDjB,EAAUc,GAAW,0BAA0B,KAGjD7B,EAAOiC,QAAWD,IAChBnT,QAAQC,KAAK,0CAA2CkT,EAAGE,KAAMF,EAAGG,QAChE5B,IACAJ,GAAaC,EACfY,IAGFD,EAAUc,GAAW,6BAA6B,MAG7C,CACLO,UAAYrF,GACNuD,GAA0B,IAAfvD,EAAM9Y,KAAmB5I,QAAQE,WAIhDilB,EAAYA,EAAUpnB,KAAKgY,UACzB,GAAIkP,EAAS,OAEb,SADMK,EACFL,EAAS,OACb,MAAMX,OAhZOvO,OAAO0M,IAC1B,MAAM7B,EAAQ,IAAIC,iBAAiB4B,EAAKuE,eAExC,IAAIC,EAAS,GACb,IAAA,IAAS7vB,EAAI,EAAGA,EAAIwpB,EAAMhnB,OAAQxC,GAFhB,MAGhB6vB,GAAUjsB,OAAOksB,gBAAgBtG,EAAMuG,SAAS/vB,EAAGA,EAHnC,QAKlB,OAAOgwB,KAAKH,IAyYcI,CAAa3F,GACjCmD,GAAO,EACHI,GAAWN,EAAO2C,aAAe1C,UAAU2C,OAC/C/T,QAAQgU,MAAM,8CAA+C3C,EAAKnD,EAAM9Y,MACxE+b,EAAOoB,KAAKzS,KAAKwB,UAAU,CAAEtc,KAAM,QAASqsB,MAAKP,cAE5Ca,GAETsC,KAAM,KACJjU,QAAQC,KAAK,4DAA6DoR,GAG1EM,EAAYA,EAAUpnB,KAAKgY,gBACnBuP,EACFL,GACAN,EAAO2C,aAAe1C,UAAU2C,OAClC/T,QAAQC,KAAK,2CAA4CoR,GACzDF,EAAOoB,KAAKzS,KAAKwB,UAAU,CAAEtc,KAAM,aAGhCitB,GAETiC,OAAQ,KACNzC,GAAU,EACV,IACEN,EAAO4B,OACT,CAAA,MAEA,KAWAC,GAAa,CAAC/H,EAAiBc,GAAqB,KACxD,MAAM7L,EAAQ,IAAI8K,MAAMC,GAExB,OADA/K,EAAM6L,UAAYA,EACX7L,GAGHiU,GAAmB5R,MACvBgJ,EACA6I,EACAC,EACAlkB,EACAmkB,EACAC,KAEA,MAAMC,EAAc,IAAIC,SACxBD,EAAYE,OAAO,OAAQN,GAC3BI,EAAYE,OAAO,QAASL,GAC5BG,EAAYE,OAAO,qBAAsBltB,OAAO2I,IAEhD,MAAMwkB,QAAoBC,MAAM,GAAGtJ,GAAiBC,cAAqB,CACvEsJ,OAAQ,OACRze,KAAMoe,IAGR,IAAKG,EAAYG,GACf,MAAM9B,SAAiBhH,GAAe2I,IAAc,GAGtD,MAAMI,EAASJ,EAAYve,MAAM4e,YACjC,IAAKD,EACH,MAAM/B,GAAW,qCAAqC,GAGxDuB,EAAUQ,GAEV,MAAMvH,EAAoC,GAC1C,IAAIC,EAAW,aACXC,EAAa,KACjB,MAAMuH,EAAU,IAAIC,YACpB,IAAI9G,EAAS,GAEb,MACMkG,KADO,CAKX,MAAMa,KAAEA,EAAA/rB,MAAMA,SAAgB2rB,EAAOK,OACrC,GAAID,EACF,MAGF/G,GAAU6G,EAAQI,OAAOjsB,EAAO,CAAEksB,QAAQ,IAC1C,MAAMC,EAASnH,EAAO7X,MAAM,cAC5B6X,EAASmH,EAAOC,OAAS,GAEzB,IAAA,MAAWC,KAASF,EAAQ,CAC1B,MAAM1E,EAAUN,GAAqBkF,GACrC,GAAK5E,GAIDD,GAAkBC,GAAU,CAC9B,MAAM6E,EAAS7E,EAAQ6E,QAAU,MACjCjI,EAAWoD,EAAQ8B,YAAyB,SAAX+C,EAAoB,aAAe,cACpElI,EAAO/mB,KAAKsmB,GAAmB8D,EAAQC,QACnCD,EAAQ8E,cACVjI,EAAamD,EAAQ8E,YAEzB,CACF,CACF,CAKA,OAHAZ,EAAOa,cACPrB,OAAU,GAEH,CAAE/G,SAAQC,WAAUC,eAGvBmI,GAAyBtT,MAC7BgJ,EACA6I,EACAC,EACAlkB,EACAmkB,EACAwB,KAEA,MAAM5E,EAAQzF,GAAeF,EAAS,gBAChCwK,EAlnBgB,oBAAXC,QAA0B,eAAgBA,OAC5CA,OAAOC,aAET,OAAOjV,KAAKC,SAASoB,KAAK6T,SAASrK,SAAS,IAAIpjB,MAAM,KAgnBvD+kB,EAAoC,GAC1C,IAAIC,EAAW,aACXC,EAAa,KACbyI,EAAiB,EAErB,MAAMC,EAAc,IAClB,IAAI5pB,QAAQ,CAACE,EAASijB,KACpB,GAAI2E,IAEF,YADA3E,EAAOqD,GAAW,kBAAkB,IAItC,IAAImC,GAAO,EACPkB,EAAuD,KAC3D,MAAMlF,EAAS,IAAIC,UAAUF,GAC7B4E,EAAU3E,GAEV,MAAMmF,EAAW,CACfC,EACArW,KAEAiR,EAAOkB,OAAS,KAChBlB,EAAOyB,UAAY,KACnBzB,EAAO+B,QAAU,KACjB/B,EAAOiC,QAAU,KACjB0C,EAAU,MACG,YAATS,EAIJ5G,EAAOzP,GAAS8S,GAAW,iBAAiB,IAH1CtmB,KAMJykB,EAAOkB,OAAS,KACdrS,QAAQC,KACN,uCAAuC8V,gBAAwBI,KAEjEhF,EAAOoB,KACLzS,KAAKwB,UAAU,CACbtc,KAAM,QACNwxB,WAAYT,EACZtoB,KAAM2mB,EACNC,QACAoC,mBAAoBtmB,EACpBumB,SAAUP,MAKhBhF,EAAOyB,UAAa7pB,IAClB,MAAM8nB,EA/pBwB,CACpCznB,IAEA,IACE,MAAqB,iBAAVA,EACF,KAEF0W,KAAKC,MAAM3W,EACpB,CAAA,MACE,OAAO,IACT,GAqpBsButB,CAAuBnvB,OAAOuB,EAAMiC,OACpD,GAAK6lB,EAAL,CAKA,GAAqB,UAAjBA,EAAQ7rB,MAA6C,iBAAlB6rB,EAAQC,MAAoB,CACjE,MAAMO,EAA6B,iBAAhBR,EAAQQ,IAAmBR,EAAQQ,IAAM,EAyB5D,MArqB0B,EAClCuF,EACAT,IACYS,EAAcT,EA0oBdU,CAAqBxF,EAAK8E,IAC5B3I,EAAO/mB,KAAKsmB,GAAmB8D,EAAQC,QACvCqF,EAAiB9E,EACbR,EAAQ8B,YACVlF,EAAWoD,EAAQ8B,WAEc,iBAAxB9B,EAAQ8E,cACjBjI,EAAamD,EAAQ8E,cAGvB3V,QAAQC,KACN,qDAAqD8V,SAAiB1E,UAAY8E,UAIlFhF,EAAO2C,aAAe1C,UAAU2C,MAClC5C,EAAOoB,KACLzS,KAAKwB,UAAU,CACbtc,KAAM,MACNwxB,WAAYT,EACZW,SAAUP,KAKlB,CAEA,GAAqB,SAAjBtF,EAAQ7rB,KAAiB,CAG3B,OAD8B,iBAArB6rB,EAAQ6F,SAAwB7F,EAAQ6F,SAAWP,GACxCA,GAClBE,EAAarD,GAAW,uBAAuB,GAC/CmC,GAAO,OACPhE,EAAO4B,UAGToC,GAAO,OACPhE,EAAO4B,QAET,CAEqB,UAAjBlC,EAAQ7rB,OACVqxB,EAAarD,GAAWnC,EAAQ3Q,OAAS,gBAAsC,IAAtB2Q,EAAQ9E,WACjEoJ,GAAO,EACPhE,EAAO4B,QAhDT,MAFE/S,QAAQG,KAAK,wCAsDjBgR,EAAO+B,QAAU,KACVmD,IACHA,EAAarD,GAAW,0BAA0B,KAItD7B,EAAOiC,QAAU,KACXkB,IACFgC,EAAS,SAAUtD,GAAW,kBAAkB,IAG9CmC,EACFmB,EAAS,WAGXA,EAAS,SAAUD,GAAcrD,GAAW,6BAA6B,OAI/E,IAAA,IAAS8D,EAAU,EAAGA,GAAW5L,GAAuB9kB,OAAQ0wB,GAAW,EAAG,CAC5E,GAAIA,EAAU,EAAG,CACf,MAAMC,EAAQ7L,GAAuB4L,EAAU,GAC/C9W,QAAQG,KACN,gCAAgC2W,cAAoBC,cAAkBZ,WAElE9J,GAAM0K,EACd,CAEA,IAEE,aADMX,IACC,CAAE5I,SAAQC,WAAUC,aAC7B,OAASxN,GACP,MAAM6L,EACJD,GAAoB5L,GAChB+K,EAAU/K,aAAiB8K,MAAQ9K,EAAM+K,QAAUzjB,OAAO0Y,GAIhE,GAHAF,QAAQG,KACN,yCAAyC2W,eAAqB/K,WAAmBd,MAE9Ec,GAAa+K,IAAY5L,GAAuB9kB,OACnD,MAAM8Z,CAEV,CACF,CAEA,MAAM8S,GAAW,oBAAoB,IAyC1BgE,GAAYzU,MACvBgJ,EACA9d,EACA4mB,EACAlkB,EACAwd,KAEA,MAAMnC,EAAU/d,EAAKuB,OACrB,IAAKwc,EAAS,OACd,MAAMoC,EAAqBnB,KAI3B,IAAIwK,EAFJjX,QAAQC,KAAK,4BAA4BuL,EAAQplB,iBACjDunB,IAAqB,aAGrB,MAAMuJ,EAAe,KAAM,EAE3B,IACED,QAAkBpB,GAChBtK,EAASC,EAAS6I,EAAOlkB,EACzB+mB,EACA,OAEJ,CAAA,MACED,QAAkB9C,GAChB5I,EAASC,EAAS6I,EAAOlkB,EACzB+mB,EACA,OAEJ,CAEI3K,GAAwBqB,GAC1BD,IAAqB,QAInBsJ,EAAUzJ,OAAOpnB,OAAS,QACtBmnB,GACJ0J,EAAUzJ,OAAQyJ,EAAUxJ,SAAUwJ,EAAUvJ,WAChDC,GAGFA,IAAqB,SAMZwJ,GAAuB,CAClC5L,EACA6L,EACA3E,EACAjB,EACAR,EAMAC,EACAoG,KAEA,IAEIC,EAFAhD,GAAY,EACZiD,EAAiC,KAEjCC,GAAe,EAEnB,MAAMvF,EAAc,IAAIzlB,QAAc+V,MAAO7V,EAASijB,KACpD,IAEE,GAAI2E,SAAkB5nB,IACtB,MAAM+qB,QAzjBZlV,eACEgJ,EACAmM,EACAjF,EACAjB,EACAkB,EACAJ,GAEA,MAAM7d,EAAM,GAAG6W,GAAiBC,cAC1BoM,EAAW,IAAIlD,SACrBkD,EAASjD,OAAO,OAAQgD,GACxBC,EAASjD,OAAO,aAAcjC,GAC1BjB,GAAWmG,EAASjD,OAAO,aAAclD,GAC7CmG,EAASjD,OAAO,WAAYhC,GAE5B,MAAMkF,EAAkBhD,MAAMngB,EAAK,CAAEogB,OAAQ,OAAQze,KAAMuhB,IAC3DrF,MACA,MAAMrG,QAAiB2L,EACvB,IAAK3L,EAAS6I,GACZ,MAAM,IAAI9J,YAAYgB,GAAeC,IAEvC,OAAOA,EAASC,MAClB,CAmiB8B2L,CACtBtM,EACA6L,EACA3E,EACAjB,EACA,KACAP,EAAO6G,kBAGHC,EAAmBN,EAAUrF,WACnC,IAAI4F,EAAqBD,EACzB,MAAME,EAAWR,EAAUhqB,KAO3B,GALAwjB,EAAOiH,kBAAkB,CACvB9F,WAAY2F,EACZI,UAAWF,IAGT3D,SAAkB5nB,IAGtB,MAAM0nB,QAAsB,IAAI5nB,QAAgB,CAAC4rB,EAAcC,KAC7D,GAAI/D,EAA+B,YAAlB8D,EAAa,IAE9B,MAAMlH,EAAQzF,GAAeF,EAAS,kBAChC4F,EAAS,IAAIC,UAAUF,GAC7BqG,EAAepG,EAEf,IAAImH,EAAa,GACbC,GAAW,EACXC,EAAmB,GAEvB,MAAMC,EAAUC,IACVH,IACJA,GAAW,EACXH,EAAaM,KAGTC,EAAQzY,IACRqY,IACJA,GAAW,EACXF,EAAYnY,KAGdiR,EAAOkB,OAAS,KACdrS,QAAQC,KAAK,8BACbkR,EAAOoB,KAAKzS,KAAKwB,UAAU,CACzBtc,KAAM,QACNwtB,WAAYC,EACZL,WAAY2F,EACZI,UAAWF,EACXW,MAAO5H,EAAO4H,MACdC,aAAc7H,EAAO8H,YACrBrC,mBAAoBzF,EAAO7gB,sBAI/BghB,EAAOyB,UAAYrQ,MAAOxZ,IACxB,IAAIiC,EACJ,IACEA,EAAO8U,KAAKC,MAAMvY,OAAOuB,EAAMiC,MACjC,CAAA,MAEE,YADAgV,QAAQG,KAAK,gCAEf,CAEA,MAAM4Y,EAAU/tB,EAAKhG,KAErB,GAAgB,YAAZ+zB,GAAoD,iBAApB/tB,EAAKonB,WAGvC,OAFA4F,EAAqBhtB,EAAKonB,gBAC1BnB,EAAO6B,cAAckF,GAIvB,GAAgB,cAAZe,EAAJ,CAQA,GAAgB,gBAAZA,GAAmD,iBAAf/tB,EAAK4X,MAG3C,OAFA4V,GAAoBxtB,EAAK4X,WACzBqO,EAAO+H,mBAAmBhuB,EAAK4X,OAIjC,GAAgB,eAAZmW,GAA4Bv0B,MAAMC,QAAQuG,EAAKiuB,OAAQ,CACzD,MAAMA,EAAQjuB,EAAKiuB,MAGfT,EAAiBxpB,QACnBiiB,EAAOiI,sBAAsBV,EAAiBxpB,QAEhDwpB,EAAmB,GAEnBvH,EAAOkI,cAAcF,GAErB,MAAMG,EAAiD,GACvD,IAAA,MAAWnuB,KAAQguB,EAAO,CAExB,MAAMI,EAA2B,aAAdpuB,EAAK2f,KACpByO,GACFxX,GAAuB,CACrB2P,UAAWwG,EACXvF,YACAmG,MAAO5H,EAAO4H,MACdvE,MAAOrD,EAAOqD,MACdlkB,kBAAmBmI,QAAQ0Y,EAAO7gB,mBAClCmpB,iBAAkBL,EAAMttB,IAAK1H,IAAAA,CAC3B0mB,QAAS1mB,EAAE0mB,QACXC,KAAM3mB,EAAE2mB,KACR2O,KAAMt1B,EAAEs1B,QAEVC,iBAAkB,IAAIJ,KAI1B,MAAMvO,QAAewM,EAAYpsB,GAG7BouB,GACFrX,KAGFiP,EAAOwI,eAAexuB,EAAK0f,QAAS1f,EAAK2f,KAAMC,EAAOA,QACtDuO,EAAQ3yB,KAAKokB,EACf,CAQA,YANIsG,EAAO2C,aAAe1C,UAAU2C,MAClC5C,EAAOoB,KAAKzS,KAAKwB,UAAU,CACzBtc,KAAM,eACNo0B,aAIN,CAEA,GAAgB,eAAZL,EAA0B,CAC5BT,EAActtB,EAAK0uB,aAA0B,GAC7C,MAAMC,GAA2C,IAAvB3uB,EAAK4uB,cAM/B,OALA3I,EAAO4I,kBAAkBvB,EAAYqB,GACN,iBAApB3uB,EAAKonB,YACdnB,EAAO6B,cAAc9nB,EAAKonB,iBAE5BqG,EAAOH,EAET,CAEA,GAAgB,UAAZS,EAAqB,CACvB,MAAMe,EAAU9uB,EAAKkV,OAAoB,cAIzC,OAHAsX,GAAe,EACfvG,EAAOjB,UAAU8J,QACjBnB,EAAK,IAAI3N,MAAM8O,GAEjB,CA5EA,MALE7I,EAAO8I,cACL/uB,EAAKgvB,UACLhvB,EAAKivB,iBAkFX9I,EAAO+B,QAAU,KACflT,QAAQE,MAAM,0BACdsX,GAAe,EACfvG,EAAOjB,UAAU,oCACjB2I,EAAK,IAAI3N,MAAM,sCAGjBmG,EAAOiC,QAAU,KACfpT,QAAQC,KAAK,2BACbwY,EAAOH,MAOX,GAHAf,EAAe,KAGXjD,IAAcF,EAIhB,OAHKE,GACH1P,KAEKlY,IAQT,IAAIuqB,EALJjX,QAAQC,KACN,sCAAsC+Q,EAAOqD,SAG/CpD,EAAOtD,qBAAqB,aAG5B,IACEsJ,QAAkBpB,GAChBtK,EACA6I,EACApD,EAAOqD,MACP/b,QAAQ0Y,EAAO7gB,mBACf,IAAMmkB,EACLnD,IAAaoG,EAAepG,GAEjC,OAAS+I,GACP,GAAI5F,SAAkB5nB,IACtBsT,QAAQG,KACN,+CAA+C+Z,aAAmBlP,MAAQkP,EAAQjP,QAAUzjB,OAAO0yB,MAErGjD,QAAkB9C,GAChB5I,EACA6I,EACApD,EAAOqD,MACP/b,QAAQ0Y,EAAO7gB,mBACf,IAAMmkB,EACLS,IAAauC,EAAevC,GAEjC,CAEKT,GACH1P,MAGG0P,GAAa2C,EAAUzJ,OAAOpnB,OAAS,QACpCmnB,GACJ0J,EAAUzJ,OACVyJ,EAAUxJ,SACVwJ,EAAUvJ,WACVuD,EAAOtD,oBAGTsD,EAAOtD,qBAAqB,QAG9BjhB,GACF,OAAS0jB,GAEP,IAAKoH,EAAc,CACjB,MAAMzM,EAAMqF,aAAepF,MAAQoF,EAAInF,QAAUzjB,OAAO4oB,GACxDa,EAAOjB,UAAUjF,EACnB,CACA4E,EAAOS,EACT,CAAA,QACEkH,GAAcpD,SAASiG,MAAM,QACzB5C,GAAgBA,EAAazD,YAAc1C,UAAU2C,MACvDwD,EAAaxE,QAEfwE,EAAe,IACjB,IAGF,MAAO,CACLtD,KAAM,KACJK,GAAY,EACZ5H,KACI4K,GACFA,EAAapD,SAASiG,MAAM,QAE1B5C,GAAgBA,EAAazD,YAAc1C,UAAU2C,MACvDwD,EAAaxE,SAGjBoC,KAAMlD,ICjsCJmI,GAAuC,CAC3CC,KCVa,s7gBDWbC,SEXa,8n+BFYbC,SGZa,s7gBHabC,UIba,+n+BJoEf,MAAMC,GAAa,IApDnB,MACUC,MAAmB,GACnB/N,QAAS,EAET,OAAAgO,CAAQvvB,GACd,OAAO,IAAIoB,QAASE,IAClB,GAAsB,oBAAXuK,OAET,YADAvK,IAIF,MAAMokB,EAAQ,IAAI3B,MAAMiL,GAAYhvB,IACpC0lB,EAAMvB,QAAU,OAEhB,IAAImC,GAAU,EACd,MAAM4E,EAAW,KACX5E,IACJA,GAAU,EACVZ,EAAM8J,QAAU,KAChB9J,EAAMoC,QAAU,KAChBxmB,MAGFokB,EAAM8J,QAAUtE,EAChBxF,EAAMoC,QAAUoD,EACXxF,EAAMrB,OAAO0K,MAAM,IAAM7D,MAElC,CAEA,WAAcuE,GACZ,IAAI50B,KAAK0mB,OAAT,CACA1mB,KAAK0mB,QAAS,EACd,IACE,KAAO1mB,KAAKy0B,MAAMt0B,OAAS,GAAG,CAC5B,MAAM00B,EAAO70B,KAAKy0B,MAAM5zB,QACnBg0B,SACC70B,KAAK00B,QAAQG,EACrB,CACF,CAAA,QACE70B,KAAK0mB,QAAS,CAChB,CAViB,CAWnB,CAEA,OAAAoO,CAAQ3vB,GACgB,oBAAX6L,SACXhR,KAAKy0B,MAAMj0B,KAAK2E,GACXnF,KAAK0mB,QACH1mB,KAAK40B,QAEd,GKoBIG,GAAc,qBACdC,GAAgB,uBAChBC,GAAiB,mBACjBC,GAAS,IAMTC,GACG,QADHA,GAEK,aAFLA,GAGa,yBAHbA,GAIO,kBAJPA,GAKiB,wBALjBA,GAMU,cANVA,GAOM,cAPNA,GAQU,IARVA,GASU,wBAGHC,GAAmBC,GAC9B,gBAAgBA,sCAYLC,GAAqBC,GAC5BA,EAAMvrB,YAAoBmrB,GAC1BI,EAAMC,eAAuBL,GAC7BI,EAAME,eAAuBN,GAC7BI,EAAMG,WAAmBP,GACzBI,EAAMI,eAAuBR,GAC7BI,EAAMtrB,OAAekrB,GAClBA,GAkBIS,GAAoBpZ,IAC/B,MAAMqZ,EAAezZ,KAAKD,IAAI,EAAGC,KAAK0Z,MAAMtZ,EAAa,MAKzD,MAAO,GAJSJ,KAAK0Z,MAAMD,EAAe,IACvCjQ,WACA0C,SAAS,EAAG,SACEuN,EAAe,IAAIjQ,WAAW0C,SAAS,EAAG,QAShDyN,GAAyBV,GAAiC,CACrE,CACEtjB,GAAI,EACJvK,KAAM4tB,GAAgBC,GACtBW,QAAQ,IAICC,GAA2BC,IACjCA,IAILA,EAAQla,WAAW+Y,IACnBmB,EAAQla,WAAWgZ,IACnBkB,EAAQla,WAAWiZ,MAoBfkB,GAAoBC,GACVA,EAASvO,OAAO,CAACC,EAAK9C,IAAY5I,KAAKD,IAAI2L,EAAK9C,EAAQjT,IAAK,GAC5D,EAkBJskB,GAA6B,CACxCnsB,EACAosB,EACAtsB,EACAC,KACaC,GAAqBosB,KAAkBtsB,IAAe,EAa/DusB,GACJvxB,IAEA,GAAkB,mBAAdA,EAAK2f,KACP,MAAO,CAAEnd,KAAM,WAAYgvB,KAAM,WAEnC,GAAkB,WAAdxxB,EAAK2f,KACP,MAAO,CAAEnd,KAAM,WAAYgvB,KAAM,UAEnC,GAAkB,aAAdxxB,EAAK2f,KAAqB,CAC5B,MAAMnW,EAA+B,iBAAlBxJ,EAAKsuB,KAAK9kB,IAAmBxJ,EAAKsuB,KAAK9kB,IAAIzF,OAAS,GACvE,MAAO,CACLvB,KAAMgH,EAAM,iBAAiBA,IAAQ,eACrCgoB,KAAM,WAEV,CACA,MAAkB,aAAdxxB,EAAK2f,MAA4C,SAArB3f,EAAKsuB,KAAK/Q,OACjC,CAAE/a,KAAM,gBAAiBgvB,KAAM,UAEtB,aAAdxxB,EAAK2f,MAA4C,SAArB3f,EAAKsuB,KAAK/Q,OACjC,CAAE/a,KAAM,gBAAiBgvB,KAAM,QAEtB,aAAdxxB,EAAK2f,MAA4C,WAArB3f,EAAKsuB,KAAK/Q,OACjC,CAAE/a,KAAM,gBAAiBgvB,KAAM,QAEtB,aAAdxxB,EAAK2f,MAA4C,UAArB3f,EAAKsuB,KAAK/Q,OACjC,CAAE/a,KAAM,UAAWgvB,KAAM,YAEhB,aAAdxxB,EAAK2f,KACA,CAAEnd,KAAM,YAAagvB,KAAM,YAE7B,CACLhvB,KAAMxC,EAAK2f,MAAQ,OACnB6R,KAAM,YAIGC,GAAa,EACxBlqB,UACAwe,SACA7gB,qBAAoB,EACpBwsB,wBACA3lB,UAAS,EACT4lB,aACAC,oBACAC,eACAC,sBAEA,MAAOV,EAAUW,GAAeC,GAAoB,KAClD,GAA4B,oBAAjBnb,aAA8B,CACvC,MAAMob,EAAQpb,aAAajC,QAAQmb,IAC7BmC,EAAYrb,aAAajC,QAAQob,IAEvC,GAAIiC,GAASC,EAAW,CAEtB,GADmBnc,KAAKC,MAAQpL,SAASsnB,EAAW,IACnChC,GACf,IACE,OAAOrb,KAAKC,MAAMmd,EACpB,CAAA,MAEA,MAEAhB,GAAwBpa,aAE5B,CACF,CAEA,OAAOka,GAAsBhL,EAAOsK,cAG/BprB,EAAQktB,GAAaH,IAAS,IAC9BhtB,EAAaotB,GAAkBJ,IAAS,IACxCrB,EAAgB0B,GAAqBL,IAAS,IAC9CtB,EAAY4B,GAAiBN,IAAS,IACtCO,EAAkBC,GAAuBR,IAAS,IAClDvB,EAAgBgC,GAAqBT,IAAS,IAC9CxB,EAAgBkC,GAAqBV,IAAS,IAC9CW,EAAcC,GAAmBZ,IAAS,IAC1Ca,EAAqBC,GAA0Bd,GAAS,IACxDe,EAAgBC,GAAqBhB,GAAwB,MAC9DiB,EAA2B,CAC/BhuB,SACAD,cACA2rB,iBACAD,aACA6B,mBACA9B,iBACAD,kBAEI0C,EAAqB5C,GAAkB2C,GACvCE,EAhMuB,EAC7B5C,EACAwC,IAEA1lB,QACE0lB,GACGxC,EAAMtrB,QACNsrB,EAAMvrB,aACNurB,EAAMI,gBACNJ,EAAMG,YACNH,EAAMgC,kBACNhC,EAAME,gBACNF,EAAMC,gBAoLQ4C,CAAgBH,EAAaF,GAC1C5R,EAAagS,EAAcJ,GAAkBG,EAAsB/C,GAEnEkD,EAAYC,GAAOruB,GACnBsuB,EAAiBD,GAAOtuB,GAExBwuB,EAAmBF,GAAOnC,GAAiBC,IAC3CqC,EAAcH,GAA6B,MAC3CI,EAAYJ,GAA2B,MACvCK,EAAiBL,GAAmB,IACpCM,EAA4BN,GAAgC,MAC5DO,EAAeP,GACK,oBAAjBzc,aAAiB,MAElB,MAAMid,EAAKjd,aAAajC,QAAQob,IAChC,OAAI8D,GAAM/d,KAAKC,MAAQpL,SAASkpB,EAAI,IAAM5D,GACjCrZ,aAAajC,QAAQqb,IAEvB,MANW,GAQpB,MAGA8D,EAAkBT,GAAsB,MACxCU,EAAiBV,GAAsB,MACvCW,EAAkBX,GAA4B,MAC9CY,EAAYZ,GAA0C,MAEtDa,EAA0Bb,IAAO,GAEjCc,EAAmBd,GAAsB,MACzCe,GAAoBf,GAAsB,MAC1CgB,GAAsBhB,IAAO,GAC7BiB,GAAmBjB,GAA6B,MAChDkB,GAA2BlB,IAAO,GAClCmB,GAA2BnB,IAAO,GAElCoB,GAAwBpB,GAAsB,MAC9CqB,GAAyBrB,GAAO,IAChCsB,GAA2BtB,IAAO,GAClCuB,GAA2BvB,IAAO,GAElCwB,GAAwBxB,GAAsB,MAC9CyB,GAA4BzB,GAAsB,MAElD0B,GAAuB1B,GAA8B,MACrD2B,GAAqB3B,GAA8B,MAEnD4B,GAAqB5B,GAAsB,MAC3C6B,GAA0B7B,GAAe,IACzC8B,GAAyB9B,GAAmC,MAC5D+B,GAA+B/B,IAAO,GACtCgC,GAAiBhC,GAA+B,MAChDiC,GAAgCjC,GAAsB,MACtDkC,GAA2BlC,GAAO,IAClCmC,GAA0BnC,IAAO,GACjCoC,GAAkBpC,IAAO,GACzBqC,GAAkBrC,GAAO,GACzBsC,GAAsBtC,IAAO,GAEnCuC,GAAU,KACRxC,EAAU1yB,QAAUsE,GACnB,CAACA,IAEJ4wB,GAAU,KACRtC,EAAe5yB,QAAUqE,GACxB,CAACA,IAGJ6wB,GAAU,KAAQjE,IAAoB5sB,IAAiB,CAACA,IACxD6wB,GAAU,KAAQhE,IAAe5sB,IAAY,CAACA,IAG9C4wB,GAAU,KACR,IAAK/D,EAAiB,OACtB,GAAI9sB,EAEF,YADA8sB,EAAgBiB,GAAkB5C,IAIpC,GAAIoC,GAAoB9B,EAAgB,CACtC,MAAMqF,EAAgB,IAAI1E,GAAU2E,UAAUxnB,KAAK1U,IAAMA,EAAEm3B,QAAqB,SAAXn3B,EAAEE,MAEvE,YADA+3B,EAAgBgE,GAAetzB,MAAQ4tB,GAAgBrK,EAAOsK,WAEhE,CACA,GAAI8C,EAAY,CACd,MAAM6C,EAAKjD,GAAkBzC,GAAkB,CAC7CrrB,SACAD,cACA2rB,iBACAD,aAEAD,iBACAD,mBAGF,YADAsB,EAAgBkE,EAElB,CAEA,MAAMF,EAAgB,IAAI1E,GAAU2E,UAAUxnB,KAAK1U,IAAMA,EAAEm3B,QAAqB,SAAXn3B,EAAEE,MACvE+3B,EAAgBgE,GAAetzB,MAAQ4tB,GAAgBrK,EAAOsK,aAC7D,CACDrrB,EACAC,EACA0rB,EACAD,EACAF,EACAC,EACA8B,EACAQ,EACAI,EACA/B,IAGF,MAAM6E,GAAW91B,IL7WI,CAACA,IACtBqvB,GAAWM,QAAQ3vB,IK6WjB2vB,CAAQ3vB,IAGJ+1B,GAAoB,KACxBP,GAAgBh1B,SAAW,EACpBg1B,GAAgBh1B,SAGnBw1B,GAAyB,KAC7BR,GAAgBh1B,SAAW,GAGvBy1B,GAAyBC,GAC7BV,GAAgBh1B,UAAY01B,EAExBC,GAAqB,KACrBV,GAAoBj1B,UAGxBi1B,GAAoBj1B,SAAU,EAC9Bs1B,GAAQ,UAGVJ,GAAU,KACoB,oBAAjBhf,eACTA,aAAaP,QAAQyZ,GAAalb,KAAKwB,UAAU+a,IACjDva,aAAaP,QAAQ0Z,GAAeja,KAAKC,MAAM4K,cAEhD,CAACwQ,IAEJ,MAAMmF,GAAyB,KAlSI,IACnCvrB,KAkSwBgqB,GAAqBr0B,WA5R7CqK,EAAQwrB,UAAYxrB,EAAQ6R,elBvLihC,IAAWrkB,GAAEE,GAAOC,GAATH,GkBsdxiC,KACd+9B,MlBvdwjC79B,GkBwdvjC,CAAC04B,EAAUjQ,EAAYlc,EAAQD,GlBxd+hCrM,GAAEQ,EAAEW,IAAI,IAAId,EAAE0F,KAAKtD,GAAEzC,GAAEkJ,IAAInJ,MAAKC,GAAEyB,GAAG5B,GAAEG,GAAED,EAAEA,GAAEG,EAAE2F,IAAIhD,KAAK7C,KkB0dznCk9B,GAAU,KACR,MAAMv1B,EAAU20B,GAAmBt0B,QACnC,IAAKL,GAAqC,oBAAnBm2B,eACrB,OAGF,MAAMC,EAAW,IAAID,eAAe,KAClCF,OAIF,OADAG,EAASC,QAAQr2B,GACV,IAAMo2B,EAASE,cACrB,IAEH,MAAMC,GAAqB,KACiB,OAAtC9B,GAA0Bp0B,UAC5BqL,OAAOiZ,cAAc8P,GAA0Bp0B,SAC/Co0B,GAA0Bp0B,QAAU,MAEtCm0B,GAAsBn0B,QAAU,MAkB5Bm2B,GAAuB,KAC3BzE,GAAkB,GAClBC,GAAc,GACdE,GAAoB,GACpBC,GAAkB,GAClBC,GAAkB,GAClBM,EAAkB,MAClB0B,GAAsB/zB,QAAU,KAChCg0B,GAAuBh0B,QAAU,GACjCi0B,GAAyBj0B,SAAU,EACnCk0B,GAAyBl0B,SAAU,EACnC00B,GAA6B10B,SAAU,EACvCu0B,GAAmBv0B,QAAU,KAC7Bw0B,GAAwBx0B,QAAU,IAG9Bo2B,GAAsB,KACQ,OAA9B1C,GAAkB1zB,UACpBqL,OAAO7J,aAAakyB,GAAkB1zB,SACtC0zB,GAAkB1zB,QAAU,OAI1Bq2B,GAAa,KACc,OAA3BhD,EAAerzB,UACjBskB,cAAc+O,EAAerzB,SAC7BqzB,EAAerzB,QAAU,MAGvBuzB,EAAUvzB,UACZuzB,EAAUvzB,QAAQi2B,aAClB1C,EAAUvzB,QAAU,MAGlBszB,EAAgBtzB,UAClBszB,EAAgBtzB,QAAQmnB,QAAQoH,MAAM,QACtC+E,EAAgBtzB,QAAU,MAG5BozB,EAAgBpzB,QAAU,MAGtBs2B,GAAmB,KAClBvD,EAAU/yB,UAIf+yB,EAAU/yB,QAAQu2B,YAAYz2B,QAAS02B,GAAUA,EAAMnO,QACvD0K,EAAU/yB,QAAU,OAGhBy2B,GAAmB,KAClBxD,EAA0BjzB,UAI/BizB,EAA0BjzB,QAAQqoB,OAClC4K,EAA0BjzB,QAAU,OAGhC02B,GAAoB,KACxB,MAAMC,EAAchC,GAAe30B,QACnC20B,GAAe30B,QAAU,KACzB22B,GAAarO,SACbsM,GAA8B50B,QAAU,KACxC60B,GAAyB70B,QAAU,IAGrCk1B,GACE,IAAM,KACJM,KACAY,KACAvC,GAAyB7zB,SAAU,EAEnCy2B,KACA3V,KACA4V,KACAL,KACAC,KACAJ,KAEA,MAAMU,EAAW9D,EAAY9yB,QACzB42B,IACFA,EAASC,gBAAkB,KAC3BD,EAASE,OAAS,KAClBF,EAAStP,QAAU,KACI,aAAnBsP,EAASh5B,OACXg5B,EAASvO,OAEXyK,EAAY9yB,QAAU,MAGxB02B,MAEF,IAIFxB,GAAU,KACR,MAAM6B,EPtjB2B,MACnC,GAA4B,oBAAjB7gB,aAA8B,OAAO,KAChD,MAAMlC,EAAMkC,aAAajC,QAAQ+B,IACjC,IAAKhC,EAAK,OAAO,KACjB,IACE,MAAMjK,EAASmK,KAAKC,MAAMH,GAC1B,OAAIoB,KAAKC,MAAQtL,EAAOoM,QArCN,KAsChBC,KACO,MAEFrM,CACT,CAAA,MAEE,OADAqM,KACO,IACT,GOwiBsB4gB,GACpB,IAAKD,EAAa,OAElB3gB,KACAhC,QAAQC,KAAK,2CAGT0iB,EAAYnR,YACdsN,EAAalzB,QAAU+2B,EAAYnR,UACP,oBAAjB1P,cACTA,aAAaP,QAAQ2Z,GAAgByH,EAAYnR,YAIrD,MAAMqR,EAAe1B,KACrB/D,GAAU,GACVkB,EAAU1yB,SAAU,EACpB+xB,GAAkB,GAClBM,EAAkB7C,IAElB,MAAM0H,EAAmBpjB,KAAiBhL,QAapCquB,EN+zBuB,EAC/BxX,EACAoX,EACA7J,EACA7H,EACAoG,KAEA,IAEIC,EAFAhD,GAAY,EACZiD,EAAiC,KAEjCC,GAAe,EAKnB,MAAMwL,EAAa,IAAIL,EAAYnJ,kBACnC,IAAA,MAAWyJ,KAAMN,EAAYrJ,iBACvB0J,EAAW/4B,KAAMnG,GAAMA,EAAE6mB,UAAYsY,EAAGtY,WAC5B,aAAZsY,EAAGrY,KACLoY,EAAWv8B,KAAK,CACdkkB,QAASsY,EAAGtY,QACZE,OAAQ,yCAA2D,oBAAX5T,OAAyBA,OAAOQ,SAASF,KAAO,sBAAsBuhB,MAGhIkK,EAAWv8B,KAAK,CACdkkB,QAASsY,EAAGtY,QACZE,OAAQ,sDAoNd,MAAO,CACLoJ,KAAM,KACJK,GAAY,EACZ5H,KACI4K,GAAcA,EAAapD,SAASiG,MAAM,QAC1C5C,GAAgBA,EAAazD,YAAc1C,UAAU2C,MACvDwD,EAAaxE,SAGjBoC,KAxNkB,IAAI3oB,QAAc+V,MAAO7V,EAASijB,KACpD,IACE,GAAI2E,SAAkB5nB,IAEtB,IAAIsrB,EAAqB2K,EAAYnR,UAErC,MAAM4C,QAAsB,IAAI5nB,QAAgB,CAAC4rB,EAAcC,KAC7D,GAAI/D,EAA+B,YAAlB8D,EAAa,IAE9B,MAAMlH,EAAQzF,GAAeF,EAAS,kBAChC4F,EAAS,IAAIC,UAAUF,GAC7BqG,EAAepG,EAEf,IAAImH,EAAa,GACbC,GAAW,EACXC,EAAmB,GAEvB,MAAMC,EAAUC,IACVH,IACJA,GAAW,EACXH,EAAaM,KAGTC,EAAQzY,IACRqY,IACJA,GAAW,EACXF,EAAYnY,KAGdiR,EAAOkB,OAAS,KACdrS,QAAQC,KAAK,qCACbkR,EAAOoB,KAAKzS,KAAKwB,UAAU,CACzBtc,KAAM,SACNwtB,WAAYmQ,EAAYlQ,UACxBL,WAAYuQ,EAAYnR,UACxBoH,MAAO+J,EAAY/J,MACnBC,aAAcC,EACdrC,mBAAoBkM,EAAYxyB,kBAChC+yB,mBAAoBP,EAAYrJ,iBAChC6J,aAAcH,MAIlB7R,EAAOyB,UAAYrQ,MAAOxZ,IACxB,IAAIiC,EACJ,IACEA,EAAO8U,KAAKC,MAAMvY,OAAOuB,EAAMiC,MACjC,CAAA,MAAU,MAAQ,CAElB,MAAM+tB,EAAU/tB,EAAKhG,KAErB,GAAgB,YAAZ+zB,GAAoD,iBAApB/tB,EAAKonB,WAGvC,OAFA4F,EAAqBhtB,EAAKonB,gBAC1BnB,EAAO6B,cAAckF,GAIvB,GAAgB,cAAZe,EAAJ,CAQA,GAAgB,gBAAZA,GAAmD,iBAAf/tB,EAAK4X,MAG3C,OAFA4V,GAAoBxtB,EAAK4X,WACzBqO,EAAO+H,mBAAmBhuB,EAAK4X,OAIjC,GAAgB,eAAZmW,GAA4Bv0B,MAAMC,QAAQuG,EAAKiuB,OAAQ,CACzD,MAAMA,EAAQjuB,EAAKiuB,MAGfT,EAAiBxpB,QACnBiiB,EAAOiI,sBAAsBV,EAAiBxpB,QAEhDwpB,EAAmB,GAEnBvH,EAAOkI,cAAcF,GAErB,MAAMG,EAAiD,GACvD,IAAA,MAAWnuB,KAAQguB,EAAO,CACxB,MAAMI,EAA2B,aAAdpuB,EAAK2f,KACpByO,GACFxX,GAAuB,CACrB2P,UAAWwG,EACXvF,UAAWkQ,EAAYlQ,UACvBmG,MAAO+J,EAAY/J,MACnBvE,MAAOsO,EAAYtO,MACnBlkB,kBAAmBwyB,EAAYxyB,kBAC/BmpB,iBAAkBL,EAAMttB,IAAK1H,IAAAA,CAC3B0mB,QAAS1mB,EAAE0mB,QACXC,KAAM3mB,EAAE2mB,KACR2O,KAAMt1B,EAAEs1B,QAEVC,iBAAkB,IAAIJ,KAI1B,MAAMvO,QAAewM,EAAYpsB,GAE7BouB,GACFrX,KAGFiP,EAAOwI,eAAexuB,EAAK0f,QAAS1f,EAAK2f,KAAMC,EAAOA,QACtDuO,EAAQ3yB,KAAKokB,EACf,CAKA,YAHIsG,EAAO2C,aAAe1C,UAAU2C,MAClC5C,EAAOoB,KAAKzS,KAAKwB,UAAU,CAAEtc,KAAM,eAAgBo0B,aAGvD,CAEA,GAAgB,eAAZL,EAA0B,CAC5BT,EAActtB,EAAK0uB,aAA0B,GAC7C,MAAMC,GAA2C,IAAvB3uB,EAAK4uB,cAM/B,OALA3I,EAAO4I,kBAAkBvB,EAAYqB,GACN,iBAApB3uB,EAAKonB,YACdnB,EAAO6B,cAAc9nB,EAAKonB,iBAE5BqG,EAAOH,EAET,CAEA,GAAgB,UAAZS,EAAqB,CACvB,MAAMe,EAAU9uB,EAAKkV,OAAoB,cAIzC,OAHAsX,GAAe,EACfvG,EAAOjB,UAAU8J,QACjBnB,EAAK,IAAI3N,MAAM8O,GAEjB,CAvEA,MALE7I,EAAO8I,cACL/uB,EAAKgvB,UACLhvB,EAAKivB,iBA6EX9I,EAAO+B,QAAU,KACfsE,GAAe,EACfvG,EAAOjB,UAAU,yBACjB2I,EAAK,IAAI3N,MAAM,2BAGjBmG,EAAOiC,QAAU,IAAMqF,EAAOH,KAMhC,GAHAf,EAAe,KAGXjD,IAAcF,EAIhB,OAHKE,GACH1P,KAEKlY,IAKT,IAAIuqB,EAFJjX,QAAQC,KAAK,uCAAuC0iB,EAAYtO,SAChEpD,EAAOtD,qBAAqB,aAG5B,IACEsJ,QAAkBpB,GAChBtK,EAAS6I,EAAeuO,EAAYtO,MAAO/b,QAAQqqB,EAAYxyB,mBAC/D,IAAMmkB,EACLnD,IAAaoG,EAAepG,GAEjC,OAAS+I,GACP,GAAI5F,SAAkB5nB,IACtBsT,QAAQG,KACN,+CAA+C+Z,aAAmBlP,MAAQkP,EAAQjP,QAAUzjB,OAAO0yB,MAErGjD,QAAkB9C,GAChB5I,EAAS6I,EAAeuO,EAAYtO,MAAO/b,QAAQqqB,EAAYxyB,mBAC/D,IAAMmkB,EACLS,IAAauC,EAAevC,GAEjC,CAEKT,GACH1P,MAGG0P,GAAa2C,EAAUzJ,OAAOpnB,OAAS,QACpCmnB,GACJ0J,EAAUzJ,OAAQyJ,EAAUxJ,SAAUwJ,EAAUvJ,WAChDuD,EAAOtD,oBAGTsD,EAAOtD,qBAAqB,QAG9BjhB,GACF,OAAS0jB,GACP,IAAKoH,EAAc,CACjB,MAAMzM,EAAMqF,aAAepF,MAAQoF,EAAInF,QAAUzjB,OAAO4oB,GACxDa,EAAOjB,UAAUjF,EACnB,CACA4E,EAAOS,EACT,CAAA,QACEkH,GAAcpD,SAASiG,MAAM,QACzB5C,GAAgBA,EAAazD,YAAc1C,UAAU2C,MACvDwD,EAAaxE,QAEfwE,EAAe,IACjB,MM1iCmB6L,CACjBpS,EAAOqS,eACPV,EACAG,EACA,CACEhQ,YAAcwQ,IACPjC,GAAsBwB,IACvBS,GAAOA,IAAQxE,EAAalzB,UAC9BkzB,EAAalzB,QAAU03B,EACK,oBAAjBxhB,cACTA,aAAaP,QAAQ2Z,GAAgBoI,KAI3CtK,iBAAmBpW,IACZye,GAAsBwB,KAC3BlF,GAAkB,GAClBJ,GAAc,GACdU,EAAkB,MAElBmC,GAAwBx0B,SAAWgX,EAEG,OAAlC+c,GAAsB/zB,QACxB+zB,GAAsB/zB,QAAU23B,GAC9BnD,GAAwBx0B,SACxB,GAGF43B,GACE7D,GAAsB/zB,QACtBw0B,GAAwBx0B,WAI9BiuB,gBAAiB,CAACzF,EAAemI,KAC/B,IAAK8E,GAAsBwB,GAAe,OAC1C3B,GAAQ,aACRP,GAAgB/0B,SAA2B,IAAjB2wB,EAC1B0B,EAAkB,MAClBV,GAAc,GACdE,GAAoB,GAEpB,MAAMgG,EACJrP,GAAiBgM,GAAwBx0B,QAC3Cw0B,GAAwBx0B,QAAU63B,EAEI,OAAlC9D,GAAsB/zB,QACxB43B,GACE7D,GAAsB/zB,QACtB63B,GAGF9D,GAAsB/zB,QAAU23B,GAC9BE,GACA,IAINvK,oBAAsBzrB,IACf4zB,GAAsBwB,IACtB7L,GACHhG,EAAOqS,eAAgB51B,EAAMujB,EAAOqD,MACpClkB,EAAoB3G,GAAUk6B,GAAuBl6B,EAAOq5B,IAC5D1I,MAAO/J,GAAQpQ,QAAQG,KAAK,kCAAmCiQ,KAEnE+I,YAAcF,IACPoI,GAAsBwB,KACvB5J,EAAM7yB,OAAS,GACjB86B,GAAQ,YAEVvD,GAAkB,GAClBM,EAAkB7C,IAClBuI,GAA4B1K,GAE5B0G,GAAsB/zB,QAAU,KAChCw0B,GAAwBx0B,QAAU,KAEpC6tB,aAAc,OACdM,YAAa,KACNsH,GAAsBwB,KAC3B3B,GAAQ,YACR3D,GAAc,GACdU,EAAkB7C,MAEpBzN,mBAAqBnkB,IACnBk6B,GAAuBl6B,EAAOq5B,IAEhC7S,QAAUI,IACHiR,GAAsBwB,KAC3B5E,EAAkB,MAClBsF,GAAc,SAASnT,KAAO,MArGb7N,MACrBtX,IAEA,MAAMsvB,EAA2B,CAC/B3P,KAAM3f,EAAK2f,KACXD,QAAS1f,EAAK0f,WACX1f,EAAKsuB,MAEV,OAAO9O,GAAsB8P,KAmG/BsE,EAA0BjzB,QAAUm3B,EAEpCA,EAAW5N,KACRgF,MAAM,QACNyJ,QAAQ,KACFvC,GAAsBwB,KAC3BzF,GAAU,GACVkB,EAAU1yB,SAAU,EACpB+xB,GAAkB,GAClBJ,GAAc,GACdE,GAAoB,GACpBC,GAAkB,GAClBO,EAAkB,MAClBmC,GAAwBx0B,QAAU,GAClC+zB,GAAsB/zB,QAAU,KAChCizB,EAA0BjzB,QAAU,MAGjC80B,GAAwB90B,SACzB0wB,GACEnsB,EACAwwB,GAAgB/0B,QAChB4yB,EAAe5yB,UAIZi4B,GAAe,OAGtBlD,GAAgB/0B,SAAU,MAE7B,IAEH,MAAM23B,GAAgB,CACpB91B,EACAwuB,EACA6H,KAOA,MAAM9rB,EAAKymB,EAAiB7yB,UAa5B,OAZAoxB,EAAa+G,GAAa,IACrBA,EACH,CACE/rB,KACAvK,OACAwuB,SACAj3B,KAAM8+B,GAAS9+B,KACfg/B,SAAUF,GAASE,SACnBC,UAAWH,GAASG,UACpBC,UAAWJ,GAASI,aAGjBlsB,GAGH2rB,GAA+B1K,IACnC+D,EAAa+G,IACX,MAAMjJ,EAAO,IAAIiJ,GAEjB,IAAA,MAAW94B,KAAQguB,EAAO,CACxB,MAAMkL,EAAY3H,GAAwBvxB,GACpCm5B,EAAOtJ,EAAKA,EAAK10B,OAAS,GAC1Bi+B,EAAyC,iBAAfD,GAAM32B,KAClC22B,EAAK32B,KAAKlF,QAAQ,cAAe,IACjC,GAEJ,GACE67B,IACIA,EAAKnI,QACQ,SAAdmI,EAAKp/B,MACLq/B,IAAqBF,EAAU12B,KAClC,CACA,MAAM62B,EAAiB1uB,OAAOC,SAC3BuuB,EAAK32B,KAAKyB,MAAM,mBAAmB,IAAM,IAC1C,IAGIq1B,GADmB3uB,OAAO4uB,SAASF,GAAkBA,EAAiB,GACvC,EAC/BG,EAAYJ,GAAoBF,EAAU12B,KAChDqtB,EAAKA,EAAK10B,OAAS,GAAK,IACnBg+B,EACHH,UAAWQ,EACXP,UAAWK,EACX92B,KAAM,GAAGg3B,MAAcF,MAEzB,QACF,CAEA,MAAMvsB,EAAKymB,EAAiB7yB,UAC5BkvB,EAAKr0B,KAAK,CACRuR,KACAvK,KAAM02B,EAAU12B,KAChBwuB,QAAQ,EACRj3B,KAAM,OACNg/B,SAAUG,EAAU1H,KACpBwH,UAAWE,EAAU12B,KACrBy2B,UAAW,GAEf,CAEA,OAAOpJ,KAIL0I,GAAoB,CAACxrB,EAAYvK,KACrCuvB,EAAa+G,GACXA,EAASp4B,IAAKsf,GACZA,EAAQjT,KAAOA,EAAK,IAAKiT,EAASxd,QAASwd,KAK3CyZ,GAA+Bj3B,IACnC,MAAMke,EAAale,EAAKuB,OACnB2c,IAGL8U,GAAyB70B,QAAU+f,EACW,OAA1C6U,GAA8B50B,QAIlC43B,GAAkBhD,GAA8B50B,QAAS+f,GAHvD6U,GAA8B50B,QAAU23B,GAAc5X,GAAY,KAMhEgZ,GAAkC,KACtCnE,GAA8B50B,QAAU,KACxC60B,GAAyB70B,QAAU,IAG/B83B,GAAyB,CAACl6B,EAAyBq5B,KACvD,GAA4B,iBAAjBA,GAA8BxB,GAAsBwB,GAA/D,CAIA,GAAc,cAAVr5B,EAGF,OAFAi0B,GAAoB,QACpBC,GAAkB,GAIpB,GAAc,YAAVl0B,EAGF,OAFAi0B,GAAoB,QACpBC,GAAkB,GAIpB,GAAc,aAAVl0B,EAGF,OAFAi0B,GAAoB,QACpBC,GAAkB,GAIpB,GAAc,SAAVl0B,EAGF,OAFAi0B,GAAoB,QACpBC,GAAkB,GAIpBD,GAAoB,GACpBC,GAAkB,EA3BlB,GA8BIkH,GAAuB/B,KACC,iBAAjBA,GAA8BxB,GAAsBwB,MAI/DvC,GAA6B10B,SAAU,EACvCqyB,EAAkB,MAClBb,GAAU,GACVkB,EAAU1yB,SAAU,EACpB0xB,GAAkB,GAClBC,GAAc,GACdE,GAAoB,GACpBC,GAAkB,GAClBC,GAAkB,GAClBwC,GAAmBv0B,QAAU,KAC7Bw0B,GAAwBx0B,QAAU,GAClC+zB,GAAsB/zB,QAAU,KAC5BizB,EAA0BjzB,UAC5BizB,EAA0BjzB,QAAU,OAGnC80B,GAAwB90B,SACzB0wB,GACEnsB,EACAwwB,GAAgB/0B,QAChB4yB,EAAe5yB,WAIjBoU,QAAQC,KAAK,mEACR4jB,GAAe,QAGtBlD,GAAgB/0B,SAAU,IAGtBi5B,GAAsBtiB,MAAO0V,IACjC,IAAKjH,EAAOyB,UAEV,YADA8Q,GAAc,uCAAuC,GAIvD,MAAMuB,EAAqB7M,EAASjpB,OACpC,IAAK81B,EAEH,YADAvB,GAAc,2DAA2D,GAI3E,MAAMV,EAAe1B,KACrB/D,GAAU,GACVkB,EAAU1yB,SAAU,EACpB0xB,GAAkB,GAClBC,GAAc,GACdE,GAAoB,GACpBC,GAAkB,GAClBC,GAAkB,GAClBM,EAAkB7C,IAClBkF,GAA6B10B,SAAU,EAEvC,IACEu0B,GAAmBv0B,QAAUk5B,EAC7BJ,GAA4BI,GAC5BH,KAEAtC,KACA,MAAMvJ,EAAcpZ,KAAiBhL,QAE/BqwB,EAAsBxiB,MAC1BtX,IAEA,MAAMsvB,EAA2B,CAC/B3P,KAAM3f,EAAK2f,KAKXD,QAAS1f,EAAK0f,WACX1f,EAAKsuB,MAEV,OAAO9O,GAAsB8P,IAGzBwI,ENoPuB,EACjCxX,EACA0M,EACAxF,EACAjB,EACAR,EAMAC,EACAoG,KAEA,IAEIC,EAFAhD,GAAY,EACZiD,EAAiC,KAEjCC,GAAe,EA6MnB,MAAO,CACLvD,KAAM,KACJK,GAAY,EACZ5H,KACI4K,GAAcA,EAAapD,SAASiG,MAAM,QAC1C5C,GAAgBA,EAAazD,YAAc1C,UAAU2C,MACvDwD,EAAaxE,SAGjBoC,KApNkB,IAAI3oB,QAAc+V,MAAO7V,EAASijB,KACpD,IACE,GAAI2E,SAAkB5nB,IAGtB,MAAM0nB,QAAsB,IAAI5nB,QAAgB,CAAC4rB,EAAcC,KAC7D,GAAI/D,EAA+B,YAAlB8D,EAAa,IAE9B,MAAMlH,EAAQzF,GAAeF,EAAS,kBAChC4F,EAAS,IAAIC,UAAUF,GAC7BqG,EAAepG,EAEf,IAAImH,EAAa,GACbC,GAAW,EACXP,EAAqBxG,GAAa,GAClCgH,EAAmB,GAEvB,MAAMC,EAAUC,IACVH,IACJA,GAAW,EACXH,EAAaM,KAGTC,EAAQzY,IACRqY,IACJA,GAAW,EACXF,EAAYnY,KAGdiR,EAAOkB,OAAS,KACdlB,EAAOoB,KAAKzS,KAAKwB,UAAU,CACzBtc,KAAM,QACNwtB,WAAYC,EACZL,WAAYZ,EACZ2G,UAAWF,EACXW,MAAO5H,EAAO4H,MACdC,aAAc7H,EAAO8H,YACrBrC,mBAAoBzF,EAAO7gB,sBAI/BghB,EAAOyB,UAAYrQ,MAAOxZ,IACxB,IAAIiC,EACJ,IACEA,EAAO8U,KAAKC,MAAMvY,OAAOuB,EAAMiC,MACjC,CAAA,MAAU,MAAQ,CAElB,MAAM+tB,EAAU/tB,EAAKhG,KAErB,GAAgB,YAAZ+zB,GAAoD,iBAApB/tB,EAAKonB,WAGvC,OAFA4F,EAAqBhtB,EAAKonB,gBAC1BnB,EAAO6B,cAAckF,GAIvB,GAAgB,cAAZe,EAAJ,CAQA,GAAgB,gBAAZA,GAAmD,iBAAf/tB,EAAK4X,MAG3C,OAFA4V,GAAoBxtB,EAAK4X,WACzBqO,EAAO+H,mBAAmBhuB,EAAK4X,OAIjC,GAAgB,eAAZmW,GAA4Bv0B,MAAMC,QAAQuG,EAAKiuB,OAAQ,CACzD,MAAMA,EAAQjuB,EAAKiuB,MAGfT,EAAiBxpB,QACnBiiB,EAAOiI,sBAAsBV,EAAiBxpB,QAEhDwpB,EAAmB,GAEnBvH,EAAOkI,cAAcF,GAErB,MAAMG,EAAiD,GACvD,IAAA,MAAWnuB,KAAQguB,EAAO,CACxB,MAAMI,EAA2B,aAAdpuB,EAAK2f,KACpByO,GACFxX,GAAuB,CACrB2P,UAAWwG,EACXvF,YACAmG,MAAO5H,EAAO4H,MACdvE,MAAOrD,EAAOqD,MACdlkB,kBAAmBmI,QAAQ0Y,EAAO7gB,mBAClCmpB,iBAAkBL,EAAMttB,IAAK1H,IAAAA,CAC3B0mB,QAAS1mB,EAAE0mB,QACXC,KAAM3mB,EAAE2mB,KACR2O,KAAMt1B,EAAEs1B,QAEVC,iBAAkB,IAAIJ,KAI1B,MAAMvO,QAAewM,EAAYpsB,GAE7BouB,GACFrX,KAGFiP,EAAOwI,eAAexuB,EAAK0f,QAAS1f,EAAK2f,KAAMC,EAAOA,QACtDuO,EAAQ3yB,KAAKokB,EACf,CAQA,YANIsG,EAAO2C,aAAe1C,UAAU2C,MAClC5C,EAAOoB,KAAKzS,KAAKwB,UAAU,CACzBtc,KAAM,eACNo0B,aAIN,CAEA,GAAgB,eAAZL,EAA0B,CAC5BT,EAActtB,EAAK0uB,aAA0B,GAC7C,MAAMC,GAA2C,IAAvB3uB,EAAK4uB,cAM/B,OALA3I,EAAO4I,kBAAkBvB,EAAYqB,GACN,iBAApB3uB,EAAKonB,YACdnB,EAAO6B,cAAc9nB,EAAKonB,iBAE5BqG,EAAOH,EAET,CAEA,GAAgB,UAAZS,EAAqB,CACvB,MAAMe,EAAU9uB,EAAKkV,OAAoB,cAIzC,OAHAsX,GAAe,EACfvG,EAAOjB,UAAU8J,QACjBnB,EAAK,IAAI3N,MAAM8O,GAEjB,CA1EA,MALE7I,EAAO8I,cACL/uB,EAAKgvB,UACLhvB,EAAKivB,iBAgFX9I,EAAO+B,QAAU,KACfsE,GAAe,EACfvG,EAAOjB,UAAU,yBACjB2I,EAAK,IAAI3N,MAAM,2BAEjBmG,EAAOiC,QAAU,IAAMqF,EAAOH,KAMhC,GAHAf,EAAe,KAGXjD,IAAcF,EAIhB,OAHKE,GACH1P,KAEKlY,IAIT,IAAIuqB,EADJhG,EAAOtD,qBAAqB,aAG5B,IACEsJ,QAAkBpB,GAChBtK,EAAS6I,EAAepD,EAAOqD,MAAO/b,QAAQ0Y,EAAO7gB,mBACrD,IAAMmkB,EACLnD,IAAaoG,EAAepG,GAEjC,OAAS+I,GACP,GAAI5F,SAAkB5nB,IACtBuqB,QAAkB9C,GAChB5I,EAAS6I,EAAepD,EAAOqD,MAAO/b,QAAQ0Y,EAAO7gB,mBACrD,IAAMmkB,EACLS,IAAauC,EAAevC,GAEjC,CAEKT,GACH1P,MAGG0P,GAAa2C,EAAUzJ,OAAOpnB,OAAS,QACpCmnB,GACJ0J,EAAUzJ,OAAQyJ,EAAUxJ,SAAUwJ,EAAUvJ,WAChDuD,EAAOtD,oBAGTsD,EAAOtD,qBAAqB,QAG9BjhB,GACF,OAAS0jB,GACP,IAAKoH,EAAc,CACjB,MAAMzM,EAAMqF,aAAepF,MAAQoF,EAAInF,QAAUzjB,OAAO4oB,GACxDa,EAAOjB,UAAUjF,EACnB,CACA4E,EAAOS,EACT,CAAA,QACEkH,GAAcpD,SAASiG,MAAM,QACzB5C,GAAgBA,EAAazD,YAAc1C,UAAU2C,MACvDwD,EAAaxE,QAEfwE,EAAe,IACjB,MM/cqByN,CACjBhU,EAAOqS,eACPyB,EACA9T,EAAOyB,UACPqM,EAAalzB,QACb,CACEgtB,MAAO5H,EAAO4H,MACdvE,MAAOrD,EAAOqD,MACdyE,cACA3oB,qBAEF,CACE2iB,YAAcwQ,IACPjC,GAAsBwB,IACvBS,GAAOA,IAAQxE,EAAalzB,UAC9BkzB,EAAalzB,QAAU03B,EACK,oBAAjBxhB,cACTA,aAAaP,QAAQ2Z,GAAgBoI,KAI3CtK,iBAAmBpW,IACZye,GAAsBwB,KAC3BvF,GAAkB,GAClBC,GAAc,GACdI,GAAkB,GACd2C,GAA6B10B,UAC/B00B,GAA6B10B,SAAU,EACvCqyB,EAAkB,OAGpBmC,GAAwBx0B,SAAWgX,EAEG,OAAlC+c,GAAsB/zB,QACxB+zB,GAAsB/zB,QAAU23B,GAC9BnD,GAAwBx0B,SACxB,GAGF43B,GACE7D,GAAsB/zB,QACtBw0B,GAAwBx0B,WAI9BiuB,gBAAiB,CAACzF,EAAemI,KAC/B,IAAK8E,GAAsBwB,GAAe,OAC1C3B,GAAQ,aACRP,GAAgB/0B,SAA2B,IAAjB2wB,EAC1B+D,GAA6B10B,SAAU,EACvCqyB,EAAkB,MAClBV,GAAc,GACdE,GAAoB,GAEpB,MAAMgG,EAAmBrP,GAAiBgM,GAAwBx0B,QAClEw0B,GAAwBx0B,QAAU63B,EAEI,OAAlC9D,GAAsB/zB,QACxB43B,GACE7D,GAAsB/zB,QACtB63B,GAGF9D,GAAsB/zB,QAAU23B,GAC9BE,GACA,IAINvK,oBAAsBzrB,IACf4zB,GAAsBwB,IACtB7L,GACHhG,EAAOqS,eAAgB51B,EAAMujB,EAAOqD,MACpClkB,EAAoB3G,GAAUk6B,GAAuBl6B,EAAOq5B,IAC5D1I,MAAO/J,GAAQpQ,QAAQG,KAAK,kCAAmCiQ,KAEnE+I,YAAcF,IACPoI,GAAsBwB,KACvB5J,EAAM7yB,OAAS,GACjB86B,GAAQ,YAEVvD,GAAkB,GAClBM,EAAkB7C,IAClBuI,GAA4B1K,GAE5B0G,GAAsB/zB,QAAU,KAChCw0B,GAAwBx0B,QAAU,KAEpC6tB,aAAc,OACdM,YAAa,KACNsH,GAAsBwB,KAC3B3B,GAAQ,YACR3D,GAAc,GACdU,EAAkB7C,MAEpBzN,mBAAqBnkB,IACnBk6B,GAAuBl6B,EAAOq5B,IAEhC7S,QAAUI,IACHiR,GAAsBwB,KAC3BvC,GAA6B10B,SAAU,EACvCqyB,EAAkB,MAClBsF,GAAc,SAASnT,KAAO,MAGlC2U,GAGFlG,EAA0BjzB,QAAUm3B,QAC9BA,EAAW5N,IACnB,OAASjV,GACP,IAAKmhB,GAAsBwB,GAAe,OAC1CvC,GAA6B10B,SAAU,EACvCqyB,EAAkB,MACd/d,aAAiB8K,OACnBuY,GAAc,SAASrjB,EAAM+K,WAAW,EAE5C,CAAA,QACE2Z,GAAoB/B,EACtB,GAwNIoC,GAAiBnB,IACrB,MAAMtB,EAAW9D,EAAY9yB,QACxB42B,GAA+B,aAAnBA,EAASh5B,QAItBs6B,GAASoB,UACX9F,EAAwBxzB,SAAU,GAGpCq2B,KACAO,EAASvO,SAgFL4P,GAAiBthB,MAAOgU,IAC5B,KACE+H,EAAU1yB,SACV4yB,EAAe5yB,SACf8zB,GAAyB9zB,SAH3B,CAWA,GAHAqyB,EAAkB7C,IAClByC,GAAgB,IAEXsH,UAAUC,cAAcC,aAI3B,OAHApH,EAAkB,MAClBJ,GAAgB,QAChB0F,GAAc,0CAA0C,GAI1D,GAA6B,oBAAlB+B,cAIT,OAHArH,EAAkB,MAClBJ,GAAgB,QAChB0F,GAAc,+CAA+C,GAI/D7D,GAAyB9zB,SAAU,EAEnC,IACE,MAAM0pB,QAAe6P,UAAUC,aAAaC,aAAa,CAAEvU,OAAO,IAClE6N,EAAU/yB,QAAU0pB,EAIpB,MAAMiQ,EAAwC,CAC5CC,mBAAoB,MAIhBC,EAAqB,CACzB,wBACA,yBACA,cAEF,IAAA,MAAWC,KAAQD,EACjB,GAAIH,cAAcK,gBAAgBD,GAAO,CACvCH,EAAgB9X,SAAWiY,EAC3B,KACF,CAGF,MAAMlD,EAAW,IAAI8C,cAAchQ,EAAQiQ,GAC3C7G,EAAY9yB,QAAU42B,EACtB5D,EAAehzB,QAAU,GACzB+4B,KACA9D,GAAoBj1B,SAAU,EAE9B,MAAMg6B,GAAepD,EAAS/U,UAAY8X,EAAgB9X,UAAY,cACnElX,MAAM,KAAK,GACXvH,QAAU,aAEP62B,EAAkB9U,GACtBC,EAAOqS,eACP,CACE5Q,UAAWzB,EAAOyB,UAClBjB,UAAWsN,EAAalzB,QACxB8mB,SAAU,KACVjF,SAAUmY,GAEZ,CACE9S,YAAcwQ,IACPA,GAAOA,IAAQxE,EAAalzB,UAGjCkzB,EAAalzB,QAAU03B,EACK,oBAAjBxhB,cACTA,aAAaP,QAAQ2Z,GAAgBoI,KAGzCrQ,UAAW,EAAGxlB,WACPA,EAAKuB,QAGV01B,GAA4Bj3B,MAIlC8yB,GAAe30B,QAAUi6B,EAEzBrD,EAASC,gBAAmB15B,IACtBA,EAAMiC,KAAKoK,KAAO,IACpBwpB,EAAehzB,QAAQnF,KAAKsC,EAAMiC,MAC9Bu1B,GAAe30B,SACZ20B,GAAe30B,QAAQ2nB,UAAUxqB,EAAMiC,MAAMmvB,MAAOja,IACvDF,QAAQG,KACN,qCAAqCD,aAAiB8K,MAAQ9K,EAAM+K,QAAUzjB,OAAO0Y,UAO/FsiB,EAAStP,QAAU,KACjBqQ,GAAc,6CAA6C,IAG7Df,EAASE,OAASngB,UAChB8a,GAAe,GACfmB,EAAe5yB,SAAU,EACzB4zB,GAAiB5zB,QAAU,KAC3Bk2B,KAEAG,KACAC,KAEA,MAAM4D,EAAgB1G,EAAwBxzB,QAC9CwzB,EAAwBxzB,SAAU,EAElC,MAAMqjB,EAAO,IAAIC,KAAK0P,EAAehzB,QAAS,CAC5C5G,KAAMw9B,EAAS/U,UAAY,eAE7BmR,EAAehzB,QAAU,GAEzB,MAAMm6B,EAAexF,GAAe30B,QAGpC,GAFA20B,GAAe30B,QAAU,KAErBk6B,EAIF,OAHAC,GAAc7R,SACdyQ,UACA1G,EAAkB,MAIpB,GAAkB,IAAdhP,EAAK7Z,KAKP,OAJA2wB,GAAc7R,SACdyQ,KACA1G,EAAkB,WAClBsF,GAAc,+CAA+C,GAI/DjG,GAAkB,GAClBW,EAAkB7C,IAElB,IACE,GAAI2K,EAAc,CAChBxE,KACA,MAAM9J,QAAkBsO,EAAa9R,OAOrC,GANIwD,EAAUrF,YAAcqF,EAAUrF,aAAe0M,EAAalzB,UAChEkzB,EAAalzB,QAAU6rB,EAAUrF,WACL,oBAAjBtQ,cACTA,aAAaP,QAAQ2Z,GAAgBzD,EAAUrF,aAG/CqF,EAAUhqB,KAAKuB,OAIjB,OAHA01B,GAA4BjN,EAAUhqB,MACtCwwB,EAAkB7C,eACZyJ,GAAoBpN,EAAUhqB,MAGxC,CACF,OAASyS,GACPF,QAAQG,KACN,4EAA4ED,aAAiB8K,MAAQ9K,EAAM+K,QAAUzjB,OAAO0Y,KAEhI,CAAA,QACEykB,IACF,CAEA3kB,QAAQC,KAAK,2EACbge,EAAkB7C,SA1dA7Y,OAAO0M,IAC7B,IAAK+B,EAAOyB,UAEV,YADA8Q,GAAc,uCAAuC,GAIvD,MAAMV,EAAe1B,KACrB/D,GAAU,GACVkB,EAAU1yB,SAAU,EACpB0xB,GAAkB,GAClBC,GAAc,GACdE,GAAoB,GACpBC,GAAkB,GAClBC,GAAkB,GAClBoE,KACA9D,EAAkB7C,IAClBkF,GAA6B10B,SAAU,EAEvC,IACE,MAAMo6B,EAAW/W,EAAKjqB,MAAQ,aACxBihC,EAAYD,EAAS5hB,SAAS,OAChC,MACA4hB,EAAS5hB,SAAS,OAChB,MACA4hB,EAAS5hB,SAAS,SAAW4hB,EAAS5hB,SAAS,OAC7C,MACA,OACFsT,EAAO,IAAIwO,KAAK,CAACjX,GAAO,eAAegX,IAAa,CACxDjhC,KAAMghC,IAGR3D,KAEA,MAAMvJ,EAAcpZ,KAAiBhL,QAG/BqwB,EAAsBxiB,MAC1BtX,IAEA,MAAMsvB,EAA2B,CAC/B3P,KAAM3f,EAAK2f,KAKXD,QAAS1f,EAAK0f,WACX1f,EAAKsuB,MAEV,OAAO9O,GAAsB8P,IAGzBwI,EAAa5L,GACjBnG,EAAOqS,eACP3L,EACA1G,EAAOyB,UACPqM,EAAalzB,QACb,CACEgtB,MAAO5H,EAAO4H,MACdvE,MAAOrD,EAAOqD,MACdyE,cACA3oB,qBAEF,CACE2nB,iBAAkB,KACXuJ,GAAsBwB,IAC3BtB,MAEFrJ,gBAAkBltB,IAChB,IAAKq2B,GAAsBwB,GAAe,OACtC73B,EAAKonB,YAAcpnB,EAAKonB,aAAe0M,EAAalzB,UACtDkzB,EAAalzB,QAAUZ,EAAKonB,WACA,oBAAjBtQ,cACTA,aAAaP,QAAQ2Z,GAAgBlwB,EAAKonB,aAI9C,MAAMzG,EAAa3gB,EAAKmtB,UAAUnpB,OAClC,GAAI2c,EAAY,CACd,MAAMwa,EAAmBhG,GAAmBv0B,QAC5Cu0B,GAAmBv0B,QAAU+f,EACiB,OAA1C6U,GAA8B50B,SAChC43B,GAAkBhD,GAA8B50B,QAAS+f,GACzDgZ,MACSwB,IAAqBxa,GAC9B4X,GAAc5X,GAAY,EAE9B,CAEA2R,GAAkB,GAClBC,GAAc,GACdU,EAAkB7C,KAEpBtI,YAAcwQ,IACPjC,GAAsBwB,IACvBS,GAAOA,IAAQxE,EAAalzB,UAC9BkzB,EAAalzB,QAAU03B,EACK,oBAAjBxhB,cACTA,aAAaP,QAAQ2Z,GAAgBoI,KAI3CtK,iBAAmBpW,IACZye,GAAsBwB,KAC3BvF,GAAkB,GAClBC,GAAc,GACdI,GAAkB,GACd2C,GAA6B10B,UAC/B00B,GAA6B10B,SAAU,EACvCqyB,EAAkB,OAIpBmC,GAAwBx0B,SAAWgX,EAEG,OAAlC+c,GAAsB/zB,QACxB+zB,GAAsB/zB,QAAU23B,GAC9BnD,GAAwBx0B,SACxB,GAGF43B,GACE7D,GAAsB/zB,QACtBw0B,GAAwBx0B,WAI9BiuB,gBAAiB,CAACzF,EAAemI,KAC/B,IAAK8E,GAAsBwB,GAAe,OAC1C3B,GAAQ,aACRP,GAAgB/0B,SAA2B,IAAjB2wB,EAC1B+D,GAA6B10B,SAAU,EACvCqyB,EAAkB,MAClBV,GAAc,GACdE,GAAoB,GAEpB,MAAMgG,EACJrP,GAAiBgM,GAAwBx0B,QAE3Cy0B,GAAuBz0B,QAAU,KACjCw0B,GAAwBx0B,QAAU63B,EAEI,OAAlC9D,GAAsB/zB,QACxB43B,GACE7D,GAAsB/zB,QACtB63B,GAGF9D,GAAsB/zB,QAAU23B,GAC9BE,GACA,IAINvK,oBAAsBzrB,IACf4zB,GAAsBwB,IACtB7L,GACHhG,EAAOqS,eAAgB51B,EAAMujB,EAAOqD,MACpClkB,EAAoB3G,GAAUk6B,GAAuBl6B,EAAOq5B,IAC5D1I,MAAO/J,GAAQpQ,QAAQG,KAAK,kCAAmCiQ,KAEnE+I,YAAcF,IACPoI,GAAsBwB,KACvB5J,EAAM7yB,OAAS,GACjB86B,GAAQ,YAEVvD,GAAkB,GAClBM,EAAkB7C,IAClBuI,GAA4B1K,GAC5B0G,GAAsB/zB,QAAU,KAChCw0B,GAAwBx0B,QAAU,KAEpC6tB,aAAc,CAAC2M,EAASC,EAAWC,OAInCvM,YAAa,CAACwM,EAAYC,KACnBnF,GAAsBwB,KAE3B3B,GAAQ,YACR3D,GAAc,GACdU,EAAkB7C,MAEpBzN,mBAAqBnkB,IACnBk6B,GAAuBl6B,EAAOq5B,IAEhC7S,QAAUI,IACHiR,GAAsBwB,KAC3BvC,GAA6B10B,SAAU,EACvCqyB,EAAkB,MAClBsF,GAAc,SAASnT,KAAO,MAGlC2U,GAGFlG,EAA0BjzB,QAAUm3B,QAC9BA,EAAW5N,IAOnB,OAASjV,GACP,IAAKmhB,GAAsBwB,GAAe,OAE1CvC,GAA6B10B,SAAU,EACvCqyB,EAAkB,KACpB,CAAA,QACE2G,GAAoB/B,EACtB,GAyQU4D,CAAgBxX,IAGX,QAATsH,GA3PS,EAACjB,EAAqBkN,KACrC,MAAMkE,EACJzvB,OAAO0vB,cACN1vB,OACE2vB,mBAEL,IAAKF,EACH,OAGF,MAAMxgC,EAAU,IAAIwgC,EACpBxH,EAAgBtzB,QAAU1F,EAE1B,MAAM2gC,EAAW3gC,EAAQ4gC,iBACzBD,EAASE,QAAU,IAEnB,MAAMC,EAAS9gC,EAAQ+gC,wBAAwB3R,GAC/C6J,EAAUvzB,QAAUo7B,EACpBA,EAAOE,QAAQL,GAEf,MAAMM,EAAY,IAAI9Z,WAAWwZ,EAASO,mBAC1CpI,EAAgBpzB,QAAU,KAC1B,IAAIy7B,GAAiB,EACjBC,EAAiC,KACrC,MAAMC,EAA2Bp3B,EAEjC8uB,EAAerzB,QAAUqL,OAAO4Y,YAAY,KAC1C,IAAK2O,EAAe5yB,SAA8B,aAAnB42B,EAASh5B,MAEtC,YADAy4B,KAIF4E,EAASW,qBAAqBL,GAE9B,IAAIM,EAAM,EACV,IAAA,MAAWr+B,KAAS+9B,EAClBM,GAAOr+B,EAKT,GAHgBq+B,EAAMN,EAAU/gC,OACP,IAvwCT,IAywCY,CAC1B,IAAKihC,EAGH,OAFAC,EAAkB,UAClBtI,EAAgBpzB,QAAU,MAI5B,GAAgC,OAA5BozB,EAAgBpzB,QAElB,YADAozB,EAAgBpzB,QAAUoV,KAAKC,OAIjC,MAAMymB,EAAkB1mB,KAAKC,MAAQ+d,EAAgBpzB,QAIrD,YAHIy7B,GAAkBK,EArxCF,KAsxClBzC,KAGJ,CAEAjG,EAAgBpzB,QAAU,KACF,OAApB07B,IACFA,EAAkBtmB,KAAKC,OAGpBomB,GA9pC0B,EACnCM,EACAJ,EACAK,EApI2C,QAqI9BL,GAA4BI,GAAoBC,EA4pCnDC,CADmB7mB,KAAKC,MAAQqmB,EACMC,KACxCF,GAAiB,EACbE,GACFtJ,EAAkB7C,MAIvB,KAkLC0M,CAASxS,EAAQkN,GAGnBA,EAASuF,MAAM,KACfvI,GAAiB5zB,QAAU2qB,EAC3B8G,GAAe,GACfmB,EAAe5yB,SAAU,EA1kC3Bk2B,KACA/B,GAAsBn0B,QAAUoV,KAAKC,MACrC8c,EAAuB,GAEvBiC,GAA0Bp0B,QAAUqL,OAAO4Y,YAAY,KACrD,MAAMmY,EAAYjI,GAAsBn0B,QAKxCmyB,EAJkB,OAAdiK,EAImBhnB,KAAKC,MAAQ+mB,EAHX,IAIxB,KAkkCD/J,EACE9tB,GAA8B,QAATomB,EACjB6E,GACAA,IAGFqE,GAAyB7zB,UAC3B6zB,GAAyB7zB,SAAU,EACnCq5B,KAEJ,OAAS/kB,GACP,MAAM4Z,EA/4CW,CAAC5Z,GAClBA,aAAiB8K,MACZ9K,EAAM+K,QAER,kBA24CYgd,CAAe/nB,IAC1B4Z,EAAOtxB,cAAc4b,SAAS,eAAiB0V,EAAOtxB,cAAc4b,SAAS,aAC/Esc,GAAwB90B,SAAU,GAEpC02B,KACArE,EAAkB,MAClBsF,GAAc,oBAAoBzJ,KAAU,GAC5CmI,KACAC,KACAzC,GAAyB7zB,SAAU,EACnCyxB,GAAe,GACfmB,EAAe5yB,SAAU,EACzBk2B,IACF,CAAA,QACOtD,EAAe5yB,SAAY0yB,EAAU1yB,SACxCqyB,EAAkB,MAEpByB,GAAyB9zB,SAAU,EACnCiyB,GAAgB,EAClB,CA/MA,GAkNIqK,GAAkB,KACtB7I,EAAiBzzB,QAAU,KAC3B2zB,GAAoB3zB,SAAU,EAC9Bo2B,MAgDImG,GACJp/B,IAIA,GAFAA,EAAMq/B,iBAEFr/B,EAAMsK,cAAcg1B,sBACtB,IACEt/B,EAAMsK,cAAcg1B,sBAAsBt/B,EAAMu/B,UAClD,CAAA,MAEA,CAGF,MAAMN,EAAY3I,EAAiBzzB,QAC7B28B,EAAUhJ,GAAoB3zB,QAGpC,GAFAs8B,KAEkB,OAAdF,EACF,OAGF,GAAIO,EAMF,YALI/J,EAAe5yB,QACjBq5B,KACSvF,GAAyB9zB,UAClC6zB,GAAyB7zB,SAAU,IAMF,QAxhDP,EAChC6W,EACA+lB,EA/D+B,MAgEX/lB,GAAc+lB,EAAc,OAAS,MAqhDnDC,CADaznB,KAAKC,MAAQ+mB,IAEvBnE,GAAe,QAgDxB/C,GAAU,KACR,IAAK3wB,GAAqBuwB,GAAwB90B,QAAS,OAC3D,MAAM88B,EAAQzxB,OAAOtK,WAAW,KACzB6xB,EAAe5yB,SAAY0yB,EAAU1yB,SAAY8zB,GAAyB9zB,SAAY80B,GAAwB90B,SAC5Gi4B,GAAe,QAErB,KACH,MAAO,IAAM5sB,OAAO7J,aAAas7B,IAChC,CAACv4B,IAEJ,MAAMw4B,GAAW,KACfvH,KACAP,GAAoBj1B,SAAU,EAC9By2B,KACA3V,KACA4V,KACA2C,GAAc,CAAEC,SAAS,IACzBjD,KACAC,KACAH,KACA3E,GAAU,GACVkB,EAAU1yB,SAAU,GAIlBgxB,IACFA,EAAWhxB,QAAU,CACnBi4B,eAAgB,KACdnD,GAAwB90B,SAAU,EAC7Bi4B,GAAe,QAEtB+E,gBAAiB,KACflc,KACA4V,KACA,MAAME,EAAW9D,EAAY9yB,QACzB42B,GAA+B,aAAnBA,EAASh5B,MACvBy7B,GAAc,CAAEC,SAAS,KAEzBjD,KACAC,OAGJyG,cAIJ,MAAME,GAAyC,CAC7Cj4B,SAAU,QACVE,OAAQ,OACRD,MAAO,OACPxC,MAAO,QACP0E,UAAW,QACXvB,gBAAiB,uBACjBD,aAAc3D,GACdwB,QAAS4H,EAAS,OAAS,OAC3BjG,cAAe,SACfmK,SAAU,SACVhK,OAAQ,QACR43B,UAAW9xB,EAAS,OAAS,WAAWlJ,KACxC4X,UAAWvV,EACP,mBAAmB7C,GAAOC,YAAYM,KACtCA,GACJsD,WAAY,uBAgBR43B,GAA+C,CACnD72B,WAAY,OACZR,OAAQ,OACRC,OAAQ,UACRU,QAAS,MACTd,aAAc,MACdE,MAAOnE,GAAOG,KACd2B,QAAS,OACT4B,WAAY,SACZY,eAAgB,SAChBC,WAAY,SAAS/D,wBAAsCA,MAgBvDk7B,GAAgB/M,IAAA,CACpBgN,SAAU,MACV52B,QAAS4pB,EAAS,WAAa,UAC/B1qB,aAAc3D,GACduE,SAAU,OACVC,WAAY,OACZ82B,SAAU,aACVC,WAAY,WACZC,UAAWnN,EAAS,WAAa,aACjCzqB,gBAAiByqB,EAAS3uB,GAAOI,YAAc,GAC/C+D,MAAOwqB,EAAS3uB,GAAOK,gBAAkB,2BAiCrC07B,GAA8C,CAClDC,KAAM,IACNh4B,UAAW,OACXC,aAAc,OACdE,MAAOnE,GAAOG,KACd0E,SAAU,OACV/C,QAAS,OACT4B,WAAY,SACZqB,QAAS,SACT82B,WAAY,SACZjuB,SAAU,SACVquB,aAAc,WACd9uB,QAAS,OAUL+uB,GAAiD,CACrDn4B,SAAU,OACVc,SAAU,OACVs3B,WAAY,MACZh4B,MAAOnE,GAAOG,KACdi8B,UAAW,SAGPC,GAAkD,CACtDt7B,MAAO,OACPC,OAAQ,OACRiD,aAAc,QACdW,WAAY,cACZ9C,QAAS,OACT4B,WAAY,SACZY,eAAgB,SAChBD,OAAQ,UACRF,MAAO,UACPC,OAAQ,qCACRG,WAAY,aAAa/D,MAGrB87B,GAAiB35B,GAA4C,QAA7BuvB,GAAiB5zB,QACjDkG,GAAiB5B,IAAWD,EAC5B45B,GAAgBjM,IAAiB3tB,EACjC65B,GAAoB55B;AAE1B,SACG,MAAA,CAAIzB,UAAU,oBAAoBpG,MAAOwgC,GACxC9iC,SAAA;eAAAoJ,EAAC,QAAA,CAAOpJ,SAAA,kTAaKuH,GAAOG;eAoDpBkF,EAAC,MAAA,CAAItK,MAtMsC,CAC7CgK,QAAS,YACTjD,QAAS,OACTwC,eAAgB,gBAChBZ,WAAY,UAmMRjL,SAAA;eAAAoJ,EAAChB,GAAA,CACCC,IC3+DK,yiID4+DLG,MAAM,aACNlG,MAAO,CAAE4gC,SAAU,OAAQ36B,OAAQ,QACnCE,aAAa;eAEfmE,EAAC,MAAA,CAAItK,MAtM2C,CACpD+G,QAAS,OACT4B,WAAY,SACZC,IAAK,OAoMClL,SAAA;eAAAoJ,EAAC,SAAA,CACCnK,KAAK,SACLyJ,UAAU,mBACVpG,MAAO0gC,GACPx2B,QAzTY,KACpB6uB,KACAP,GAAoBj1B,SAAU,EAC9Bs8B,KACAzI,GAAyB7zB,SAAU,EAEnCy2B,KACA3V,KACA4V,KAEI5D,EAAY9yB,SAAyC,aAA9B8yB,EAAY9yB,QAAQpC,MAC7Cy7B,GAAc,CAAEC,SAAS,KAEzB9F,EAAwBxzB,SAAU,EAClCq2B,KACAC,MAGFJ,KACA/D,EAAuB,GAEvB7B,GAC0B,oBAAjBpa,aAA+BA,aAAe,MAGvDgd,EAAalzB,QAAU,KACvB,MAAMm+B,EAAkB/N,GAAsBhL,EAAOsK,WACrDmD,EAAiB7yB,QAAUwwB,GAAiB2N,GAC5C/M,EAAY+M,GAEZ3M,GAAU,GACVkB,EAAU1yB,SAAU,EACpByxB,GAAe,GACfmB,EAAe5yB,SAAU,EACzBm2B,MAwRQ,aAAW,yBACXxzB,MAAM,yBAENxI,wBAAAoJ,EAAChB,IAAQC,IEz/DN,2WFy/DiC,cAAY,OAAOC,MAAO,GAAIC,OAAQ;eAG5Ea,EAAC,SAAA,CACCnK,KAAK,SACLyJ,UAAU,mCACVpG,MAAO,IACF0gC,GACH35B,QAAS,OACT4B,WAAY,SACZY,eAAgB,UAElBW,QAASC,EACT,aAAW,gBACXjE,MAAM,gBAENxI,wBAAAoJ,EAAChB,IAAQC,IAAKF,GAAkB,cAAY,OAAOG,MAAO,GAAIC,OAAQ;iBAK3E,MAAA,CAAIjG,MAjNiD,CACxDgK,QAAS,WACTW,UAAW,OACXs2B,KAAM,IACNh4B,UAAW,KA6M2BnM,IAAK86B,GACvCl6B,wBAAAoJ,EAAC,MAAA,CAAI9G,MA3M0C,CACnD+G,QAAS,OACT2B,cAAe,SACfE,IAAK,QAwM8B9L,IAAK+6B,GACjCn6B,SAAAs2B,EAAS1wB,IAAKsf,IACb,GAAqB,SAAjBA,EAAQjmB,KAAiB,CAC3B,MAAMglC,EA1LH,YADavN,EA2LuBxR,EAAQ+Y,UGlhE9C,wXH21DE,WAATvH,EACK1uB,GAEI,aAAT0uB,EI91DO,wgBJi2DE,SAATA,EKj2DO,yTLo2DE,aAATA,EACK1uB,GAEI,YAAT0uB,EMv2DO,yVCAA;APmhED,OACE9pB,EAAC,MAAA,CAECtK,MAAO,IACF2gC,IAAa,GAChBvuB,QAAS,MACTrL,QAAS,OACT4B,WAAY,SACZC,IAAK,OAGPlL,SAAA;eAAAoJ,EAAChB,GAAA,CACCC,IAAK47B,EACL,cAAY,OACZ37B,MAAO,GACPC,OAAQ,GACRjG,MAAO,CAAEiH,WAAY;iBAEtB,OAAA,CAAMvJ,SAAAklB,EAAQxd,SAhBVwd,EAAQjT,GAmBnB,CAjNiB,IAACykB;AAmNlB,OACEttB,EAAC,MAAA,CAAqB9G,MAAO2gC,GAAa/d,EAAQgR,QAC/Cl2B,SAAAklB,EAAQxd,MADDwd,EAAQjT;eAQ1BrF,EAAC,MAAA,CAAItK,MAtMsC,CAC7CgK,QAAS,YACTjD,QAAS,OACT4B,WAAY,SACZY,eAAgB,gBAChBX,IAAK,OAkMDlL,SAAA;eAAAoJ,EAAC,OAAI9G,MAAO,IAAKghC,GAAkBx3B,WAAY,yBAC5C9L,SAAAq4B,iBACCjvB,EAAC,QAAKV,UAAU,oBAAoBF,MAAO6d,EACxCrmB,SAAAqmB,IAEDuQ,iBACFhqB,EAAC,MAAA,CACCtK,MAAO,CACL+G,QAAS,OACT4B,WAAY,SACZC,IAAK,OAGPlL,SAAA;eAAAoJ,EAAC,OAAA,CACC9G,MAAO,CACL8J,SAAU,OACVsI,QAAS,MACT0uB,WAAY,UAEfpjC,SAAA;eAGDoJ,EAAC,SAAA,CACCnK,KAAK,SACLuN,QAASoqB,EACT,aACExsB,EACI,+BACA,4BAEN9H,MAAO,CACLgG,MAAO,OACPC,OAAQ,OACRiD,aAAc,OACdG,OAAQ,OACRC,OAAQ,UACRU,QAAS,MACTjD,QAAS,OACT4B,WAAY,SACZQ,gBAAiBrB,EACb7C,GAAOC,QACP,yBACJsE,WAAY,oBAAoB/D,KAChCwB,WAAY,KAGdvJ,wBAAAoJ,EAAC,OAAA,CACC9G,MAAO,CACLgG,MAAO,OACPC,OAAQ,OACRiD,aAAc,MACdC,gBAAiB,UACjBpC,QAAS,QACTyC,WAAY,aAAa/D,KACzBwF,UAAWnD,EACP,mBACA,gBACJuV,UAAW,oCAKjB;eAGN/S,EAAC,MAAA,CAAItK,MAjP2C,CACpD+G,QAAS,OACT4B,WAAY,SACZC,IAAK,MACL3B,WAAY,KA8OLvJ,SAAA,CAAAkK,mBACE,OAAA,CAAK5H,MAAOmhC,GACVzjC,SAAA81B,GAAiBiC,KAElB,KACHhsB,kBACC3C,EAAC,SAAA,CACCnK,KAAK,SACLyJ,UAAU,mBACVpG,MAAOshC,GACPp3B,QAASo2B,GACT,aAAW,gBACXp6B,MAAM,gBAENxI,wBAAAoJ,EAAChB,GAAA,CACCC,IAAKH,GACL,cAAY,OACZI,MAAO,GACPC,OAAQ,GACRjG,MAAO,CAAEoJ,MAAO,8BAGlBo4B,GAAgB,oBAClB16B,EAAC,SAAA,CACCnK,KAAK,SACLyJ,UAAU,mBACVpG,MAAOshC,GACPM,cA3iBVlhC,IAIA,GAFAA,EAAMq/B,kBAEF9J,EAAU1yB,QAId,GAAI4yB,EAAe5yB,QAEgB,QAA7B4zB,GAAiB5zB,QACnBq5B,GAAc,CAAEC,SAAS,IAEzBD,SALJ,CAcA,GAJA5F,EAAiBzzB,QAAUoV,KAAKC,MAChCse,GAAoB3zB,SAAU,EAC9Bo2B,KAEIj5B,EAAMsK,cAAc62B,kBACtB,IACEnhC,EAAMsK,cAAc62B,kBAAkBnhC,EAAMu/B,UAC9C,CAAA,MAEA,CAGFhJ,GAAkB1zB,QAAUqL,OAAOtK,WAAW,KAEb,OAA7B0yB,EAAiBzzB,SACjB0yB,EAAU1yB,SACV4yB,EAAe5yB,UAKjB2zB,GAAoB3zB,SAAU,EACzBi4B,GAAe,WAljDO,IA0hD7B,GA4hBUsG,YAAahC,GACbiC,gBA5dVrhC,IAEAo/B,GAAmBp/B,IA2dTshC,SAAUP,GACV,aAAYF,GAAiB,iBAAmB35B,EAAc,eAAiB,eAC/E1B,MACEq7B,GACI,iBACA35B,EACE,eACA,2CAGPlK,wBACCoJ,EAAChB,GADFy7B,GACE,CACCx7B,IAAKF,GACL,cAAY,OACZG,MAAO,GACPC,OAAQ,GACRjG,MAAO,CAAEoJ,MAAO,2BAGjB,CACCrD,IAAKJ,GACL,cAAY,OACZK,MAAO,GACPC,OAAQ,GACRjG,MAAO,CAAEoJ,MAAO,wCQ5oE5B64B,GAAqC,CACzCjH,eAAgB,uBAChB5Q,UAAW,GACXmG,MAPwB,qBAQxBvE,MAAO,QACP5O,UAAWnY,GAAOC,QAClB+tB,UARyB,SAcrBiP,GAAqBnhC,IACzB,MAAMoiB,EAAUpiB,EAAM4F,OACtB,IALsB,CAAC5F,GACvB,qCAAqCjB,KAAKiB,GAIrCohC,CAAgBhf,GACnB,OAAO8e,GAAe7kB,UAExB,GAAuB,IAAnB+F,EAAQplB,OAAc,CACxB,MAAMtC,EAAI0nB,EAAQ,GACZ7mB,EAAI6mB,EAAQ,GACZjkB,EAAIikB,EAAQ,GAClB,MAAO,IAAI1nB,IAAIA,IAAIa,IAAIA,IAAI4C,IAAIA,IAAIiB,aACrC,CACA,OAAOgjB,EAAQhjB,eAGXiiC,GAAgB,CAACC,EAAkBC,KACvC,MACM/qB,EADa2qB,GAAkBG,GACdjiC,MAAM,GACvBmiC,EAAa7C,GAA0BlyB,SAAS+J,EAAInX,MAAMs/B,EAAOA,EAAQ,GAAI,IAE7E8C,EAAUC,GACDA,GAAW,EAAIH,GACxBI,EAASD,IAA4B5oB,OAH5B9Y,EAGkC0hC,EAHRzoB,KAAKD,IAAI,EAAGC,KAAKF,IAAI,IAAKE,KAAK2oB,MAAM5hC,MAGpByiB,SAAS,IAAI0C,SAAS,EAAG,KAHrE,IAACnlB,GAKTtF,EAAI+mC,EAAOD,EAAU,IACrBjmC,EAAIkmC,EAAOD,EAAU,IACrBrjC,EAAIsjC,EAAOD,EAAU,IAE3B,MAAO,IAAIG,EAAMjnC,KAAKinC,EAAMpmC,KAAKomC,EAAMxjC,MAGnC0jC,GAAcxlB,IAClB,MAAMkG,EAAa4e,GAAkB9kB,GACrCnY,GAAOC,QAAUoe,EACjBre,GAAOE,aAAei9B,GAAc9e,GAAY,KAChDre,GAAOI,YAAcie,GAuCjBuf,GAAyB,mCACzBC,GAAoB,QACpBC,GAAsB,8BAkBtBC,GAAc,EAAGra,aAErB,MAAOsa,EAAYC,GAAiBtO,GAA6BjM,IAC1Dwa,EAAaC,GAAkBxO,IAAS,GAG/C6D,GAAU,KACR,IAAK9P,EAAOyB,UAEV,YADAgZ,GAAe,GAGjB,IAAIC,GAAY,EAsBhB,MAlFsBnpB,OACxBgJ,EACAkH,KAEA,IACE,MAAMhe,EAAM8W,EAAQhjB,QAAQ,OAAQ,IAC9BojC,QAAY/W,MAAM,GAAGngB,cAAgBge,YAC3C,OAAKkZ,EAAI7W,SACK6W,EAAIzf,OADE,IAEtB,CAAA,MACE,OAAO,IACT,GAmDE0f,CAAkB5a,EAAOqS,eAAgBrS,EAAOyB,WAAWloB,KAAMshC,IAC/D,IAAIH,EAAJ,CACA,GAAIG,EAAQ,CACV,MAAMC,EAA6B,IAC9B9a,EACHvL,UAAW8kB,GAAkBsB,EAAOE,YAAc/a,EAAOvL,WACzDmT,MAAOiT,EAAOjT,OAAS5H,EAAO4H,MAC9B0C,UAAWuQ,EAAOG,YAAchb,EAAOsK,UACvCjH,MACmB,UAAjBwX,EAAOxX,OAAsC,WAAjBwX,EAAOxX,OAAuC,QAAjBwX,EAAOxX,MAC5DwX,EAAOxX,MACPrD,EAAOqD,OAGf4W,GAAWa,EAAOrmB,WAClB8lB,EAAcO,EAChB,CACAL,GAAe,EAhBA,IAmBV,KAAQC,GAAY,IAC1B,CAAC1a,IAGJ,MAAOib,EAAQC,GAAajP,GAAS,IACP,oBAAjBnb,cAC2C,SAA7CA,aAAajC,QAAQ,sBAKzBzP,EAAY+7B,GAAiBlP,IAAS,IACtCmP,EAAwBC,GAA6BpP,GAAS,IACvC,oBAAjBnb,cAG6C,SAAjDA,aAAajC,QAAQqrB,MAIvBj7B,EAAaotB,GAAkBJ,IAAS,IACxC/sB,EAAQktB,GAAaH,IAAS,IAC9B3sB,EAAgBg8B,GAAqBrP,GAAwB,OAC7DsP,EAAkBC,GAAuBvP,IAAS,GAGnDwP,EAAiBlO,GAAgC,MAEjDmO,G1BtMs+CjpC,E0BsMn8CgK,IACvC6+B,EAAkB7+B,GACL,OAATA,GAAe++B,GAAoB,I1BxM+8C3oC,EAAE,EAAEoE,GAAE,WAAW,OAAOxE,CAAC,E0ByM9gD,K1BzM89C,IAAWA,E0B4M5+Cq9B,GAAU,KACR,GAAIsL,EAEF,YADAD,GAAc,GAGhB,GAAIF,EAAQ,OACZ,GAA4B,oBAAjBnqB,cAC0C,SAA/CA,aAAajC,QAAQ,sBAAkC,OAG7DssB,GAAc,GACd,MAAMzD,EAAQ/7B,WAAW,KACvBw/B,GAAc,GACc,oBAAjBrqB,cACTA,aAAaP,QAAQ,qBAAsB,SAE5C,KACH,MAAO,IAAMnU,aAAas7B,IACzB,CAACuD,EAAQG,IAEZ,MAAMO,EAAe,KACnB,MAAMC,GAAYX,EAClBC,EAAUU,GACVT,GAAc,GACc,oBAAjBrqB,cACTA,aAAaP,QAAQ,mBAAoB/Z,OAAOolC,KAoCpD,OAdA9L,GAAU,KACR,MAAM+L,EAAgB9oC,IACN,WAAVA,EAAEmB,KAAoB+mC,IACxBC,GAAU,GACkB,oBAAjBpqB,cACTA,aAAaP,QAAQ,mBAAoB,WAM/C,OADA5W,SAASjC,iBAAiB,UAAWmkC,GAC9B,IAAMliC,SAAShC,oBAAoB,UAAWkkC,IACpD,CAACZ,IAECT,iBAGH74B,EAAAM,EAAA,CACGlN,SAAA,EAACkmC,kBACA98B,EAACU,GAAA,CACCC,WAAY,IAAM28B,EAAe7gC,SAASi4B,iBAC1C9zB,kBAAmB,IAAM08B,EAAe7gC,SAASg9B,kBACjD54B,WAAY,IAAMy8B,EAAe7gC,SAAS+8B,WAC1C14B,cACAC,SACAC,kBAAmBi8B,EACnBh8B,aACAC,cAAe,KACb87B,GAAc,GACdQ,KAEFr8B,eAAgBi8B,EAAmB,KAAOj8B,EAC1CC,eAAgB,IAAMo8B,IACtBn8B,eAAgB,IAAMg8B,GAAoB;eAG9Cr9B,EAACutB,GAAA,CACClqB,QA9Cc,KAClB05B,GAAU,GACkB,oBAAjBpqB,cACTA,aAAaP,QAAQ,mBAAoB,SAE3CvB,QAAQC,KAAK,8BA0CT+Q,OAAQsa,EACRn7B,kBAAmBi8B,EACnBzP,sBA1D0B,KAC9B,MAAM7B,GAAQsR,EACdC,EAA0BvR,GACE,oBAAjBhZ,cACTA,aAAaP,QAAQ2pB,GAAwB1jC,OAAOszB,IAEtD9a,QAAQC,KAAK,8CAA8C6a,MAqDvD9jB,QAASi1B,EACTrP,WAAY6P,EACZ5P,kBAAmBQ,EACnBP,aAAcM,EACdL,gBAAiB2P,OA/BE,MAqCrBI,GAAe,uFAID3B,mRAWAA,6HAMAA,sLAUpB,IAAI4B,GAAsC,KACtCC,GAAsC,KACtCC,IAAmB,EACnBC,IAAgB,EAMb,MAAMC,GAAO,CAACrJ,EAAwB,MAC3C,GAAIoJ,GAEF,YADAltB,QAAQG,KAAK,gCA7Na,MAC5B,GAAwB,oBAAbxV,SACT,OAEF,GAAIA,SAASsZ,eAAemnB,IAC1B,OAEF,MAAMgC,EAAOziC,SAAS4a,cAAc,QACpC6nB,EAAKp1B,GAAKozB,GACVgC,EAAKC,IAAM,aACXD,EAAK71B,KAZL,4EAaA5M,SAAS2iC,KAAK1nB,YAAYwnB,IAsN1BG,GAEA,MAAMC,EAxPqB,CAC3B1J,IAAA,CAEAT,eAAgBS,EAAQT,gBAAkBiH,GAAejH,eACzD5Q,UAAWqR,EAAQrR,WAAa6X,GAAe7X,UAC/CmG,MAAO0R,GAAe1R,MACtBvE,MAAOiW,GAAejW,MACtB5O,UAAW6kB,GAAe7kB,UAC1B6V,UAAWgP,GAAehP,YAgPJmS,CAAqB3J,GAc3C,GAbAmH,GAAWuC,EAAc/nB,WAGrBqe,EAAQ4J,aACVX,GAAkBpiC,SAASsZ,eAAe6f,EAAQ4J,aAClDT,IAAmB,IAEnBF,GAAkBpiC,SAAS4a,cAAc,OACzCwnB,GAAgB/0B,GAAK,kBACrBrN,SAASyL,KAAKwP,YAAYmnB,IAC1BE,IAAmB,IAGhBF,GAEH,YADA/sB,QAAQE,MAAM,8BAIhB,MAAMgE,EAAa6oB,GAAgB7oB,YAAc6oB,GAAgBY,aAAa,CAAEpX,KAAM,SACtFrS,EAAW0pB,kBAEX,MAAMvlC,EAAQsC,SAAS4a,cAAc,SACrCld,EAAM2Q,YAAc8zB,GAEpB,MAAMe,EAAYljC,SAAS4a,cAAc,OACzCsoB,EAAU71B,GAAK,qBAEfkM,EAAWwQ,OAAOrsB,EAAOwlC,GACzBb,GAAkBa,EAGlB3kC,mBAAQmiC,GAAA,CAAYra,OAAQwc,IAAmBK,GAC/CX,IAAgB,EAEhBltB,QAAQkK,IAAI,mCAMD4jB,GAAU,KAChBZ,KAIDF,KACF9jC,EAAO,KAAM8jC,IACbA,GAAkB,MAGhBD,IAAmBE,IACrBtiC,SAASyL,KAAKvR,YAAYkoC,IAG5BA,GAAkB,KAClBE,IAAmB,EACnBC,IAAgB,EAChBltB,QAAQkK,IAAI,qBAMD6jB,GAAU,IAAMb,GAEvBc,GAAQ,CACZb,QACAW,WACAC,YAGoB,oBAAX92B,SACRA,OAA6C+2B,MAAQA","x_google_ignoreList":[0,1,2]}
|