@auticlabs/bulut 1.3.4 → 1.4.1
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/README.md +179 -179
- package/dist/complex-ui.css +383 -383
- package/dist/complex-ui.js +31 -31
- package/dist/embed.cjs +1 -1
- package/dist/embed.cjs.map +1 -1
- package/dist/embed.js +16 -14
- package/dist/embed.js.map +1 -1
- package/dist/icons/close.svg +3 -3
- package/dist/icons/microphone.svg +3 -3
- package/dist/icons/restart.svg +3 -3
- package/dist/index.cjs.map +1 -1
- package/dist/index.js.map +1 -1
- package/index.d.ts +22 -22
- package/package.json +75 -75
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","../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\nexport const BUTTON_SIZE = 60;\nexport const WINDOW_WIDTH = 350;\nexport const WINDOW_HEIGHT = 500;\n\n// Positioning\nexport const POSITION_BOTTOM = 20;\nexport const POSITION_RIGHT = 20;\n\n// Colors — initial values match the backend ProjectSettings default.\n// applyTheme() in index.tsx overrides primary/primaryHover/messageUser\n// with the remote config value before the widget renders.\nexport const COLORS = {\n primary: \"#6C03C1\",\n primaryHover: \"#5b02a4\",\n background: \"#ffffff\",\n text: \"hsla(215, 100%, 5%, 1)\",\n textSecondary: \"hsla(215, 100%, 5%, 1)\",\n border: \"#e5e7eb\",\n messageBot: \"\",\n messageUser: \"#6C03C1\",\n messageUserText: \"#ffffff\",\n};\n\nconst normalizeHexColor = (hex: string): string => {\n const trimmed = hex.trim();\n if (!/^#([0-9a-fA-F]{3}|[0-9a-fA-F]{6})$/.test(trimmed)) {\n return \"hsla(215, 100%, 5%, 1)\";\n }\n if (trimmed.length === 4) {\n const r = trimmed[1];\n const g = trimmed[2];\n const b = trimmed[3];\n return `#${r}${r}${g}${g}${b}${b}`;\n }\n return trimmed;\n};\n\nexport const getContrastIconFilter = (backgroundHex: string): string => {\n const hex = normalizeHexColor(backgroundHex).slice(1);\n const r = parseInt(hex.slice(0, 2), 16);\n const g = parseInt(hex.slice(2, 4), 16);\n const b = parseInt(hex.slice(4, 6), 16);\n\n // Relative luminance approximation for quick contrast choice.\n const luminance = (0.299 * r + 0.587 * g + 0.114 * b) / 255;\n return luminance > 0.62 ? \"brightness(0) invert(0)\" : \"brightness(0) invert(1)\";\n};\n\n// Border radius\nexport const BORDER_RADIUS = {\n button: '50%',\n window: '17px',\n message: '10px'\n};\n\n// Shadows\nexport const SHADOW = \"0 0 15px hsla(215, 100%, 5%, 0.15)\";\n\n// Transitions\nexport const TRANSITIONS = {\n fast: '150ms ease-in-out',\n medium: '250ms ease-in-out'\n};\n","import microphoneIcon from \"./assets/icons/microphone.svg?inline\";\nimport microphoneIconRaw from \"./assets/icons/microphone.svg?raw\";\nimport restartIcon from \"./assets/icons/restart.svg?inline\";\nimport restartIconRaw from \"./assets/icons/restart.svg?raw\";\nimport closeIcon from \"./assets/icons/close.svg?inline\";\nimport closeIconRaw from \"./assets/icons/close.svg?raw\";\nimport stopIcon from \"./assets/icons/stop.svg?inline\";\nimport stopIconRaw from \"./assets/icons/stop.svg?raw\";\nimport arrowPathIconRaw from \"./assets/icons/arrow-path.svg?raw\";\nimport commandLineIconRaw from \"./assets/icons/command-line.svg?raw\";\nimport cursorArrowRaysIconRaw from \"./assets/icons/cursor-arrow-rays.svg?raw\";\nimport faceSmileIconRaw from \"./assets/icons/face-smile.svg?raw\";\nimport handRaisedIconRaw from \"./assets/icons/hand-raised.svg?raw\";\nimport mapIconRaw from \"./assets/icons/map.svg?raw\";\nimport microphoneOutlineIconRaw from \"./assets/icons/microphone-outline.svg?raw\";\nimport queueListIconRaw from \"./assets/icons/queue-list.svg?raw\";\nimport stopOutlineIconRaw from \"./assets/icons/stop-outline.svg?raw\";\nimport xMarkIconRaw from \"./assets/icons/x-mark.svg?raw\";\nimport bulutLogo from \"./assets/icons/bulut_logo.svg\";\nimport bulutLogoRaw from \"./assets/icons/bulut_logo.svg?raw\";\nimport sentSfx from \"./assets/sfx/sent.mp3\";\nimport thinkingSfx from \"./assets/sfx/thinking.mp3\";\nimport toolCallSfx from \"./assets/sfx/tool_call.mp3\";\nimport completedSfx from \"./assets/sfx/completed.mp3\";\n\nexport const microphoneIconUrl = microphoneIcon;\nexport const microphoneIconContent = microphoneIconRaw;\nexport const restartIconUrl = restartIcon;\nexport const restartIconContent = restartIconRaw;\nexport const closeIconUrl = closeIcon;\nexport const closeIconContent = closeIconRaw;\nexport const stopIconUrl = stopIcon;\nexport const stopIconContent = stopIconRaw;\nexport const arrowPathIconContent = arrowPathIconRaw;\nexport const commandLineIconContent = commandLineIconRaw;\nexport const cursorArrowRaysIconContent = cursorArrowRaysIconRaw;\nexport const faceSmileIconContent = faceSmileIconRaw;\nexport const handRaisedIconContent = handRaisedIconRaw;\nexport const mapIconContent = mapIconRaw;\nexport const microphoneOutlineIconContent = microphoneOutlineIconRaw;\nexport const queueListIconContent = queueListIconRaw;\nexport const stopOutlineIconContent = stopOutlineIconRaw;\nexport const xMarkIconContent = xMarkIconRaw;\nexport const logoUrl = bulutLogo;\nexport const logoContent = bulutLogoRaw;\nexport const sentSfxUrl = sentSfx;\nexport const thinkingSfxUrl = thinkingSfx;\nexport const toolCallSfxUrl = toolCallSfx;\nexport const completedSfxUrl = completedSfx;\n","export default \"<svg xmlns=\\\"http://www.w3.org/2000/svg\\\" fill=\\\"none\\\" viewBox=\\\"0 0 24 24\\\" stroke-width=\\\"1.5\\\" stroke=\\\"currentColor\\\">\\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\\\" />\\n</svg>\\n\"","export default \"<svg xmlns=\\\"http://www.w3.org/2000/svg\\\" fill=\\\"none\\\" viewBox=\\\"0 0 24 24\\\" stroke-width=\\\"1.5\\\" stroke=\\\"currentColor\\\">\\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\\\" />\\n</svg>\\n\"","export default \"<svg xmlns=\\\"http://www.w3.org/2000/svg\\\" fill=\\\"none\\\" viewBox=\\\"0 0 24 24\\\" stroke-width=\\\"1.5\\\" stroke=\\\"currentColor\\\">\\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\\\" />\\n</svg>\\n\"","export default \"<svg xmlns=\\\"http://www.w3.org/2000/svg\\\" fill=\\\"none\\\" viewBox=\\\"0 0 24 24\\\" stroke-width=\\\"1.5\\\" stroke=\\\"currentColor\\\">\\n <path stroke-linecap=\\\"round\\\" stroke-linejoin=\\\"round\\\" d=\\\"M6 18 18 6M6 6l12 12\\\" />\\n</svg>\\n\"","\nimport type { CSSProperties, JSX } from \"preact\";\nimport { useMemo } from \"preact/hooks\";\n\nexport type SvgIconProps = Omit<JSX.IntrinsicElements[\"svg\"], \"style\"> & {\n src: string;\n title?: string;\n /**\n * If true, replaces fill and stroke attributes with \"currentColor\" in the inner SVG content.\n * Default: true\n */\n stripColors?: boolean;\n style?: CSSProperties;\n // Explicitly add common props to ensure destructuring works smoothly\n className?: string;\n fill?: string;\n stroke?: string;\n strokeWidth?: string | number;\n width?: string | number;\n height?: string | number;\n};\n\nexport function SvgIcon({\n src,\n width,\n height,\n title,\n stripColors = true,\n className,\n style,\n fill,\n stroke,\n strokeWidth,\n ...props\n}: SvgIconProps) {\n const { viewBox, content } = useMemo(() => {\n if (!src) return { viewBox: \"0 0 24 24\", content: \"\" };\n\n const trimmedSrc = src.trim();\n // Extract viewBox\n const viewBoxMatch = trimmedSrc.match(/viewBox=\"([^\"]*)\"/);\n const viewBox = viewBoxMatch ? viewBoxMatch[1] : \"0 0 24 24\";\n\n // Extract inner content by removing the outer <svg> wrapper\n // We remove the opening <svg ...> tag and the closing </svg> tag only.\n let content = trimmedSrc.replace(/^<svg[^>]*>/i, \"\").replace(/<\\/svg>\\s*$/i, \"\");\n\n return { viewBox, content };\n }, [src]);\n\n const cleanContent = useMemo(() => {\n if (!content) return \"\";\n let nextContent = content;\n\n if (stripColors) {\n // Force inner nodes to use props instead of hardcoded SVG values.\n const fillValue = fill !== undefined ? fill : \"none\";\n const strokeValue = stroke !== undefined ? stroke : \"currentColor\";\n nextContent = nextContent\n .replace(/fill=\"[^\"]*\"/gi, `fill=\"${fillValue}\"`)\n .replace(/stroke=\"[^\"]*\"/gi, `stroke=\"${strokeValue}\"`);\n }\n\n if (strokeWidth !== undefined) {\n const strokeWidthValue = String(strokeWidth);\n nextContent = /stroke-width=\"[^\"]*\"/i.test(nextContent)\n ? nextContent.replace(/stroke-width=\"[^\"]*\"/gi, `stroke-width=\"${strokeWidthValue}\"`)\n : nextContent.replace(\n /<(path|circle|ellipse|line|polyline|polygon|rect)\\b/gi,\n `<$1 stroke-width=\"${strokeWidthValue}\"`,\n );\n }\n\n return nextContent;\n }, [content, stripColors, fill, stroke, strokeWidth]);\n\n return (\n <svg\n viewBox={viewBox}\n width={width}\n height={height}\n className={className}\n style={{\n display: \"inline-block\",\n verticalAlign: \"middle\",\n flexShrink: 0,\n fill: \"none\",\n stroke: \"currentColor\",\n ...style,\n }}\n xmlns=\"http://www.w3.org/2000/svg\"\n {...props}\n dangerouslySetInnerHTML={{ __html: cleanContent }}\n >\n {title && <title>{title}</title>}\n </svg>\n );\n}\n","import {\n BUTTON_SIZE,\n POSITION_BOTTOM,\n POSITION_RIGHT,\n COLORS,\n SHADOW,\n TRANSITIONS,\n BORDER_RADIUS,\n} from \"../styles/constants\";\nimport {\n microphoneOutlineIconContent,\n stopOutlineIconContent,\n xMarkIconContent,\n} from \"../assets\";\nimport { SvgIcon } from \"./SvgIcon\";\n\ninterface ChatButtonProps {\n onMicClick: () => void;\n onCancelRecording: () => void;\n onStopTask: () => void;\n isRecording: boolean;\n isBusy: boolean;\n accessibilityMode?: boolean;\n showBubble: boolean;\n onBubbleClick: () => void;\n previewMessage: string | null;\n onPreviewClick: () => void;\n onPreviewClose: () => void;\n}\n\nexport const ChatButton = ({\n onMicClick,\n onCancelRecording,\n onStopTask,\n isRecording,\n isBusy,\n accessibilityMode = false,\n showBubble,\n onBubbleClick,\n previewMessage,\n onPreviewClick,\n onPreviewClose,\n}: ChatButtonProps) => {\n const bgColor = COLORS.primary;\n const popupBoxShadow = accessibilityMode\n ? `inset 0 0 0 2px ${COLORS.primary}, ${SHADOW}`\n : SHADOW;\n\n const containerStyle: { [key: string]: string } = {\n position: \"fixed\",\n right: `${POSITION_RIGHT}px`,\n bottom: `${POSITION_BOTTOM}px`,\n display: \"flex\",\n flexDirection: \"column\",\n alignItems: \"flex-end\",\n gap: \"8px\",\n zIndex: \"9999\",\n fontFamily: \"\\\"Geist\\\", sans-serif\",\n };\n\n const controlsRowStyle: { [key: string]: string } = {\n display: \"flex\",\n alignItems: \"center\",\n gap: \"10px\",\n };\n\n const buttonStyle: { [key: string]: string } = {\n width: `${BUTTON_SIZE}px`,\n height: `${BUTTON_SIZE}px`,\n minWidth: `${BUTTON_SIZE}px`,\n minHeight: `${BUTTON_SIZE}px`,\n borderRadius: BORDER_RADIUS.button,\n backgroundColor: bgColor,\n color: \"#ffffff\",\n border: \"none\",\n cursor: \"pointer\",\n display: \"flex\",\n alignItems: \"center\",\n justifyContent: \"center\",\n transition: `background-color ${TRANSITIONS.fast}, transform ${TRANSITIONS.fast}`,\n position: \"relative\",\n flexShrink: \"0\",\n };\n\n const iconStyle: { [key: string]: string } = {\n width: \"24px\",\n height: \"24px\",\n display: \"block\",\n };\n\n const showStopButton = isBusy && !isRecording;\n const mainIconSrc = showStopButton ? stopOutlineIconContent : isRecording ? xMarkIconContent : microphoneOutlineIconContent;\n\n const handleClick = () => {\n if (showStopButton) {\n onStopTask();\n } else if (isRecording) {\n onCancelRecording();\n } else {\n onMicClick();\n }\n };\n\n // Shared close button style\n const closeBtnStyle: { [key: string]: string } = {\n position: \"absolute\",\n top: \"6px\",\n right: \"6px\",\n width: \"20px\",\n height: \"20px\",\n borderRadius: \"50%\",\n border: \"none\",\n background: \"transparent\",\n cursor: \"pointer\",\n display: \"flex\",\n alignItems: \"center\",\n justifyContent: \"center\",\n fontSize: \"12px\",\n lineHeight: \"1\",\n color: COLORS.text,\n padding: \"0\",\n };\n\n const renderPopup = (\n content: preact.ComponentChildren,\n onClick: () => void,\n onClose: (() => void) | null,\n extraClass: string,\n scrollable: boolean,\n ) => (\n <div\n className={`bulut-popup ${extraClass}`}\n style={{ cursor: \"pointer\" }}\n onClick={onClick}\n >\n {/* Close button */}\n {onClose && (\n <button\n type=\"button\"\n style={closeBtnStyle}\n onClick={(e) => {\n e.stopPropagation();\n onClose();\n }}\n aria-label=\"Kapat\"\n >\n <SvgIcon src={xMarkIconContent} aria-hidden=\"true\" width={16} height={16} />\n </button>\n )}\n\n {/* Text content */}\n <div\n style={{\n paddingRight: onClose ? \"22px\" : \"0\",\n wordBreak: \"break-word\",\n ...(scrollable ? { maxHeight: \"96px\", overflowY: \"auto\" } : {}),\n }}\n >\n {content}\n </div>\n </div>\n );\n\n return (\n <>\n <style>{`\n @import url('https://fonts.googleapis.com/css2?family=Geist:wght@100..900&display=swap');\n\n .bulut-popup {\n background: #ffffff;\n color: ${COLORS.text};\n padding: 10px 14px;\n border-radius: 12px;\n font-size: 14px;\n line-height: 1.4;\n position: relative;\n overflow: visible;\n box-shadow: ${popupBoxShadow};\n }\n .bulut-popup-bubble {\n animation: bulut-bubbleIn 400ms ease-out;\n }\n .bulut-popup-preview {\n animation: bulut-popIn ${TRANSITIONS.medium};\n }\n\n /* Desktop: limit width, lean to right */\n .bulut-popup {\n max-width: 320px;\n }\n\n /* Mobile: full-width layout with popup filling remaining space */\n @media (max-width: 600px) {\n .bulut-fab-container:has(.bulut-popup) {\n left: 16px !important;\n right: 16px !important;\n }\n .bulut-fab-container:has(.bulut-popup) .bulut-controls-row {\n width: 100%;\n }\n .bulut-popup {\n flex: 1;\n min-width: 0;\n max-width: none;\n }\n }\n\n @keyframes bulut-popIn {\n from { opacity: 0; transform: translateX(10px); }\n to { opacity: 1; transform: translateX(0); }\n }\n @keyframes bulut-bubbleIn {\n 0% { opacity: 0; transform: translateX(10px) scale(0.95); }\n 60% { opacity: 1; transform: translateX(-4px) scale(1.02); }\n 100% { opacity: 1; transform: translateX(0) scale(1); }\n }\n @keyframes bulut-badgeIn {\n from { opacity: 0; transform: translateY(-4px); }\n to { opacity: 1; transform: translateY(0); }\n }\n `}</style>\n\n <div className=\"bulut-fab-container\" style={containerStyle}>\n <div className=\"bulut-controls-row\" style={controlsRowStyle}>\n {/* Welcome bubble */}\n {showBubble &&\n !isRecording &&\n !previewMessage &&\n renderPopup(\n \"Destek lazımsa hemen konuşmaya başlayabiliriz!\",\n onBubbleClick,\n null,\n \"bulut-popup-bubble\",\n false,\n )}\n\n {/* New-message preview */}\n {previewMessage &&\n renderPopup(\n previewMessage,\n onPreviewClick,\n onPreviewClose,\n \"bulut-popup-preview\",\n true,\n )}\n\n {/* Main mic / cancel button */}\n <button\n style={buttonStyle}\n onClick={handleClick}\n onMouseEnter={(e) => {\n Object.assign(e.currentTarget.style, {\n backgroundColor: COLORS.primaryHover,\n transform: \"scale(1.05)\",\n });\n }}\n onMouseLeave={(e) => {\n Object.assign(e.currentTarget.style, {\n transform: \"scale(1)\",\n });\n }}\n aria-label={showStopButton ? \"Görevi durdur\" : isRecording ? \"Kaydı iptal et\" : \"Konuşmaya başla\"}\n >\n <SvgIcon\n src={mainIconSrc}\n aria-hidden=\"true\"\n style={iconStyle}\n />\n </button>\n </div>\n </div>\n </>\n );\n};\n","export interface PageContext {\n links: string[];\n interactables: string[];\n summary: string;\n}\n\nexport interface CachedPageContextEntry {\n url: string;\n summary: string;\n links: string[];\n interactables: string[];\n capturedAt: number;\n version: number;\n}\n\nexport interface PageContextSummaryInput {\n url: string;\n title: string;\n lang: string;\n headings: string[];\n landmarks: string[];\n links: string[];\n interactables: string[];\n interactionSignals: string[];\n styleSelectors: string[];\n pageBlueprint: string[];\n textSnippets: string[];\n outerHtmlDigest: string;\n}\n\ninterface InteractableCandidate {\n line: string;\n score: number;\n order: number;\n}\n\ninterface PageSignalSnapshot {\n links: string[];\n interactables: string[];\n interactionSignals: string[];\n styleSelectors: string[];\n pageBlueprint: string[];\n}\n\nimport {\n MAX_LINKS,\n MAX_INTERACTABLES,\n MAX_HEADINGS,\n MAX_TEXT_SNIPPETS,\n MAX_CACHED_PAGES,\n MAX_PAGE_SCAN_ELEMENTS,\n MAX_EVENT_HINTS_PER_ELEMENT,\n MAX_BRANCH_SAMPLES,\n MAX_BRANCH_DEPTH,\n MAX_STYLESHEET_SELECTORS,\n MAX_STYLESHEET_RULES,\n} from \"./contextConfig\";\n\nexport const PAGE_CONTEXT_CACHE_VERSION = 3;\nexport const PAGE_CONTEXT_CACHE_KEY = \"auticbot_page_context_cache_v3\";\n\nconst NON_CONTENT_TAGS = new Set([\n \"script\",\n \"style\",\n \"noscript\",\n \"template\",\n \"link\",\n \"meta\",\n]);\n\nconst NATIVE_INTERACTIVE_TAGS = new Set([\n \"a\",\n \"button\",\n \"input\",\n \"textarea\",\n \"select\",\n \"summary\",\n \"details\",\n \"option\",\n]);\n\nconst INTERACTIVE_ROLES = new Set([\n \"button\",\n \"link\",\n \"tab\",\n \"menuitem\",\n \"option\",\n \"checkbox\",\n \"radio\",\n \"switch\",\n \"combobox\",\n \"textbox\",\n \"searchbox\",\n \"slider\",\n \"spinbutton\",\n \"treeitem\",\n]);\n\nconst TRACKED_DISPLAY_VALUES = new Set([\n \"block\",\n \"inline\",\n \"inline-block\",\n \"flex\",\n \"inline-flex\",\n \"grid\",\n \"inline-grid\",\n]);\n\nconst TRACKED_POSITION_VALUES = new Set([\n \"relative\",\n \"absolute\",\n \"fixed\",\n \"sticky\",\n]);\n\nconst EVENT_HINT_NAMES = [\n \"click\",\n \"dblclick\",\n \"mousedown\",\n \"mouseup\",\n \"pointerdown\",\n \"pointerup\",\n \"touchstart\",\n \"touchend\",\n \"keydown\",\n \"keyup\",\n \"keypress\",\n \"input\",\n \"change\",\n \"submit\",\n \"focus\",\n \"blur\",\n];\n\nconst ARIA_INTERACTION_ATTRS = [\n \"aria-controls\",\n \"aria-expanded\",\n \"aria-haspopup\",\n \"aria-pressed\",\n \"aria-selected\",\n];\n\nconst DATA_INTERACTION_PATTERN =\n /(action|click|press|toggle|target|trigger|nav|open|close|menu|modal|command|submit)/i;\n\nconst STYLESHEET_SELECTOR_PATTERN =\n /(:hover|:focus|:active|button|a\\b|input|textarea|select|\\[role=|\\[aria-|\\[data-|\\.btn|\\.link)/i;\n\nconst pageContextCache = new Map<string, CachedPageContextEntry>();\nlet cacheHydrated = false;\n\nconst normalizeWhitespace = (value: string): string =>\n value.replace(/\\s+/g, \" \").trim();\n\nconst canonicalUrl = (rawUrl: string): string => {\n try {\n return new URL(rawUrl, rawUrl).href;\n } catch {\n return rawUrl;\n }\n};\n\nconst isCacheEntry = (value: unknown): value is CachedPageContextEntry => {\n if (typeof value !== \"object\" || value === null) {\n return false;\n }\n\n const obj = value as Record<string, unknown>;\n return (\n typeof obj.url === \"string\" &&\n typeof obj.summary === \"string\" &&\n Array.isArray(obj.links) &&\n Array.isArray(obj.interactables) &&\n typeof obj.capturedAt === \"number\" &&\n typeof obj.version === \"number\"\n );\n};\n\nconst bumpCount = (map: Map<string, number>, key: string): void => {\n if (!key) {\n return;\n }\n map.set(key, (map.get(key) ?? 0) + 1);\n};\n\nconst formatTopCounts = (map: Map<string, number>, maxItems: number): string => {\n if (map.size === 0) {\n return \"none\";\n }\n\n return Array.from(map.entries())\n .sort((a, b) => b[1] - a[1] || a[0].localeCompare(b[0]))\n .slice(0, maxItems)\n .map(([name, count]) => `${name}*${count}`)\n .join(\", \");\n};\n\nconst parseTabIndex = (value: string | null): number | null => {\n if (value === null) {\n return null;\n }\n\n const parsed = Number.parseInt(value, 10);\n return Number.isNaN(parsed) ? null : parsed;\n};\n\nconst compactToken = (value: string): string => {\n const compact = value.replace(/\\s+/g, \"-\").replace(/[^a-zA-Z0-9_-]/g, \"\");\n return compact || \"\";\n};\n\nconst getElementDepth = (element: Element): number => {\n let depth = 0;\n let cursor: Element | null = element;\n while (cursor?.parentElement) {\n depth += 1;\n cursor = cursor.parentElement;\n if (cursor === document.body) {\n break;\n }\n }\n return depth;\n};\n\nconst getPrimaryRole = (element: Element): string => {\n const rawRole = normalizeWhitespace(element.getAttribute(\"role\") || \"\")\n .toLowerCase()\n .split(\" \")[0];\n return rawRole || \"\";\n};\n\nconst hydrateCacheFromStorage = (): void => {\n if (cacheHydrated || typeof sessionStorage === \"undefined\") {\n return;\n }\n\n cacheHydrated = true;\n\n try {\n const raw = sessionStorage.getItem(PAGE_CONTEXT_CACHE_KEY);\n if (!raw) {\n return;\n }\n\n const parsed = JSON.parse(raw);\n if (!Array.isArray(parsed)) {\n return;\n }\n\n for (const value of parsed) {\n if (!isCacheEntry(value)) {\n continue;\n }\n if (value.version !== PAGE_CONTEXT_CACHE_VERSION) {\n continue;\n }\n pageContextCache.set(value.url, value);\n }\n if (pageContextCache.size > 0) {\n console.info(\n `[Autic] context cache restored entries=${pageContextCache.size}`,\n );\n }\n } catch (error) {\n console.warn(\"[Autic] context cache restore failed\", error);\n }\n};\n\nconst persistCacheToStorage = (): void => {\n if (typeof sessionStorage === \"undefined\") {\n return;\n }\n\n try {\n const serialized = JSON.stringify(\n Array.from(pageContextCache.values()).sort(\n (a, b) => a.capturedAt - b.capturedAt,\n ),\n );\n sessionStorage.setItem(PAGE_CONTEXT_CACHE_KEY, serialized);\n } catch (error) {\n console.warn(\"[Autic] context cache persist failed\", error);\n }\n};\n\nconst pruneOldestCacheEntries = (): void => {\n if (pageContextCache.size <= MAX_CACHED_PAGES) {\n return;\n }\n\n const sorted = Array.from(pageContextCache.values()).sort(\n (a, b) => a.capturedAt - b.capturedAt,\n );\n const overflow = sorted.length - MAX_CACHED_PAGES;\n for (let i = 0; i < overflow; i += 1) {\n pageContextCache.delete(sorted[i].url);\n }\n};\n\nconst buildSummaryWithHistory = (\n current: CachedPageContextEntry,\n): string => {\n const recentPages = Array.from(pageContextCache.values())\n .filter((entry) => entry.url !== current.url)\n .sort((a, b) => b.capturedAt - a.capturedAt)\n .slice(0, 3);\n\n if (recentPages.length === 0) {\n return current.summary;\n }\n\n const historySection = [\n \"Recent Page Memory:\",\n ...recentPages.map((entry) => {\n const compactSummary = normalizeWhitespace(entry.summary);\n return `- ${entry.url} :: ${compactSummary}`;\n }),\n ].join(\"\\n\");\n\n return `${current.summary}\\n\\n${historySection}`;\n};\n\nconst isVisible = (element: Element): boolean => {\n if (element.getAttribute(\"aria-hidden\") === \"true\") {\n return false;\n }\n\n if (element instanceof HTMLElement && element.hidden) {\n return false;\n }\n\n const style = window.getComputedStyle(element);\n if (style.display === \"none\" || style.visibility === \"hidden\") {\n return false;\n }\n\n const rect = element.getBoundingClientRect();\n return rect.width > 0 && rect.height > 0;\n};\n\nconst toAbsoluteUrl = (href: string): string => {\n try {\n return new URL(href, window.location.href).href;\n } catch {\n return href;\n }\n};\n\nconst escapeCssValue = (value: string): string => {\n if (typeof CSS !== \"undefined\" && typeof CSS.escape === \"function\") {\n return CSS.escape(value);\n }\n\n return value.replace(/([ #;&,.+*~':\"!^$\\[\\]()=>|\\/@])/g, \"\\\\$1\");\n};\n\nconst buildSelectorSegment = (element: Element): string => {\n const tag = element.tagName.toLowerCase();\n\n if (element.id) {\n return `#${escapeCssValue(element.id)}`;\n }\n\n const attrCandidates: Array<[name: string, value: string | null]> = [\n [\"name\", element.getAttribute(\"name\")],\n [\"data-testid\", element.getAttribute(\"data-testid\")],\n [\"data-test-id\", element.getAttribute(\"data-test-id\")],\n [\"aria-label\", element.getAttribute(\"aria-label\")],\n [\"role\", element.getAttribute(\"role\")],\n [\"type\", element.getAttribute(\"type\")],\n ];\n\n for (const [attrName, attrValue] of attrCandidates) {\n if (attrValue) {\n return `${tag}[${attrName}=\"${escapeCssValue(attrValue)}\"]`;\n }\n }\n\n const classes = Array.from(element.classList)\n .filter(Boolean)\n .slice(0, 3)\n .map((className) => `.${escapeCssValue(className)}`)\n .join(\"\");\n if (classes) {\n return `${tag}${classes}`;\n }\n\n const parent = element.parentElement;\n if (!parent) {\n return tag;\n }\n\n const siblingsOfTag = Array.from(parent.children).filter(\n (sibling) => sibling.tagName === element.tagName,\n );\n const index = siblingsOfTag.indexOf(element) + 1;\n return `${tag}:nth-of-type(${index})`;\n};\n\nconst buildSelector = (element: Element): string => {\n const segments: string[] = [];\n let cursor: Element | null = element;\n let depth = 0;\n\n while (cursor && depth < 4) {\n const segment = buildSelectorSegment(cursor);\n segments.unshift(segment);\n if (segment.startsWith(\"#\")) {\n break;\n }\n cursor = cursor.parentElement;\n depth += 1;\n }\n\n return segments.join(\" > \");\n};\n\nconst getElementLabel = (element: Element): string => {\n const text = normalizeWhitespace(\n (element instanceof HTMLElement ? element.innerText : element.textContent) ||\n \"\",\n );\n const ariaLabel = normalizeWhitespace(element.getAttribute(\"aria-label\") || \"\");\n const title = normalizeWhitespace(element.getAttribute(\"title\") || \"\");\n const placeholder = normalizeWhitespace(\n element.getAttribute(\"placeholder\") || \"\",\n );\n const name = normalizeWhitespace(element.getAttribute(\"name\") || \"\");\n const value =\n element instanceof HTMLInputElement ||\n element instanceof HTMLTextAreaElement ||\n element instanceof HTMLButtonElement\n ? normalizeWhitespace(element.value || \"\")\n : \"\";\n\n const classHint = Array.from(element.classList)\n .map((item) => compactToken(item))\n .find(Boolean);\n const fallback =\n (element.id && `#${element.id}`) ||\n (classHint && `.${classHint}`) ||\n buildSelector(element);\n\n const label =\n text || ariaLabel || title || placeholder || value || name || fallback;\n\n if (element.tagName.toLowerCase() === \"input\") {\n const inputType = element.getAttribute(\"type\") || \"text\";\n return `${inputType} ${label || \"input\"}`;\n }\n\n return label || \"untitled\";\n};\n\nconst getEventHints = (element: Element): string[] => {\n const record = element as unknown as Record<string, unknown>;\n const eventHints: string[] = [];\n\n for (const eventName of EVENT_HINT_NAMES) {\n const handlerKey = `on${eventName}`;\n const hasInlineHandler = Boolean(element.getAttribute(handlerKey));\n const hasPropertyHandler = typeof record[handlerKey] === \"function\";\n\n if (!hasInlineHandler && !hasPropertyHandler) {\n continue;\n }\n\n eventHints.push(eventName);\n if (eventHints.length >= MAX_EVENT_HINTS_PER_ELEMENT) {\n break;\n }\n }\n\n return eventHints;\n};\n\nconst getAriaInteractionHints = (element: Element): string[] =>\n ARIA_INTERACTION_ATTRS.filter((attrName) => element.hasAttribute(attrName)).map(\n (attrName) => attrName.replace(\"aria-\", \"\"),\n );\n\nconst getDataInteractionHints = (element: Element): string[] =>\n element\n .getAttributeNames()\n .filter(\n (attrName) =>\n attrName.startsWith(\"data-\") && DATA_INTERACTION_PATTERN.test(attrName),\n )\n .slice(0, 2)\n .map((attrName) => attrName.replace(\"data-\", \"\"));\n\nconst getComputedStyleSignals = (style: CSSStyleDeclaration): string[] => {\n const signals: string[] = [];\n\n if (style.cursor && style.cursor !== \"auto\") {\n signals.push(`cursor:${style.cursor}`);\n }\n if (style.display) {\n signals.push(`display:${style.display}`);\n }\n if (style.position) {\n signals.push(`position:${style.position}`);\n }\n if (style.zIndex && style.zIndex !== \"auto\") {\n signals.push(`z-index:${style.zIndex}`);\n }\n if (style.pointerEvents && style.pointerEvents !== \"auto\") {\n signals.push(`pointer-events:${style.pointerEvents}`);\n }\n if (style.visibility && style.visibility !== \"visible\") {\n signals.push(`visibility:${style.visibility}`);\n }\n if (style.opacity && style.opacity !== \"1\") {\n signals.push(`opacity:${style.opacity}`);\n }\n\n return Array.from(new Set(signals));\n};\n\nconst buildBlueprintToken = (element: Element): string => {\n const tag = element.tagName.toLowerCase();\n const idToken = element.id ? `#${compactToken(element.id)}` : \"\";\n const classToken = Array.from(element.classList)\n .map((item) => compactToken(item))\n .find(Boolean);\n\n return `${tag}${idToken}${classToken ? `.${classToken}` : \"\"}`;\n};\n\nconst buildBranchDigest = (element: Element, depth: number): string => {\n const token = buildBlueprintToken(element);\n if (depth <= 0) {\n return token;\n }\n\n const children = Array.from(element.children)\n .filter((child) => !NON_CONTENT_TAGS.has(child.tagName.toLowerCase()))\n .filter((child) => isVisible(child));\n if (children.length === 0) {\n return token;\n }\n\n const sampled = children\n .slice(0, 3)\n .map((child) => buildBranchDigest(child, depth - 1));\n const overflow =\n children.length > sampled.length ? `+${children.length - sampled.length}` : \"\";\n\n return `${token}>${sampled.join(\"+\")}${overflow}`;\n};\n\nconst collectDomBranchDigest = (): string[] => {\n const root = document.body ?? document.documentElement;\n const topLevelNodes = Array.from(root.children)\n .filter((child) => !NON_CONTENT_TAGS.has(child.tagName.toLowerCase()))\n .filter((child) => isVisible(child))\n .slice(0, MAX_BRANCH_SAMPLES);\n\n return topLevelNodes.map((child) =>\n buildBranchDigest(child, MAX_BRANCH_DEPTH),\n );\n};\n\nconst formatSection = (title: string, lines: string[]): string => {\n if (lines.length === 0) {\n return `${title}:\\n- none`;\n }\n\n return `${title}:\\n${lines.join(\"\\n\")}`;\n};\n\nconst buildOuterHtmlDigest = (): string => {\n const raw = document.body?.outerHTML || document.documentElement.outerHTML;\n\n const withoutScripts = raw\n .replace(/<script[\\s\\S]*?<\\/script>/gi, \"\")\n .replace(/<style[\\s\\S]*?<\\/style>/gi, \"\")\n .replace(/<noscript[\\s\\S]*?<\\/noscript>/gi, \"\")\n .replace(/<!--[\\s\\S]*?-->/g, \"\")\n .replace(/\\s+/g, \" \")\n .trim();\n\n const structural = withoutScripts\n .replace(/>[^<]*</g, \"><\")\n .replace(/\\s+/g, \" \")\n .trim();\n\n return structural;\n};\n\nconst collectTextSnippets = (): string[] => {\n const root =\n document.querySelector(\"main, article, [role='main']\") ?? document.body;\n const snippets: string[] = [];\n const seen = new Set<string>();\n\n const candidates = Array.from(root.querySelectorAll(\"p, li, h1, h2, h3\"));\n for (const node of candidates) {\n if (!isVisible(node)) {\n continue;\n }\n\n const text = normalizeWhitespace(node.textContent || \"\");\n if (!text || text.length < 20) {\n continue;\n }\n\n if (seen.has(text)) {\n continue;\n }\n\n seen.add(text);\n snippets.push(`- ${text}`);\n if (snippets.length >= MAX_TEXT_SNIPPETS) {\n break;\n }\n }\n\n return snippets;\n};\n\nconst collectLandmarkSnapshot = (): string[] => {\n const probes: Array<{ label: string; selector: string }> = [\n { label: \"main\", selector: \"main, [role='main']\" },\n { label: \"nav\", selector: \"nav, [role='navigation']\" },\n { label: \"section\", selector: \"section\" },\n { label: \"article\", selector: \"article\" },\n { label: \"form\", selector: \"form\" },\n { label: \"a\", selector: \"a\" },\n { label: \"button\", selector: \"button\" },\n { label: \"input\", selector: \"input\" },\n { label: \"role=button/link\", selector: \"[role='button'], [role='link']\" },\n { label: \"onclick attrs\", selector: \"[onclick]\" },\n {\n label: \"other event attrs\",\n selector:\n \"[onpointerdown], [onpointerup], [onkeydown], [onkeyup], [onchange], [onsubmit]\",\n },\n { label: \"tabindex\", selector: \"[tabindex]\" },\n { label: \"contenteditable\", selector: \"[contenteditable='true']\" },\n { label: \"inline cursor styles\", selector: \"[style*='cursor']\" },\n ];\n\n return probes.map(\n ({ label, selector }) => `- ${label}: ${document.querySelectorAll(selector).length}`,\n );\n};\n\nconst collectSelectorsFromRuleList = (\n rules: CSSRuleList,\n selectors: Set<string>,\n scanned: { count: number },\n): void => {\n for (const rule of Array.from(rules)) {\n if (\n scanned.count >= MAX_STYLESHEET_RULES ||\n selectors.size >= MAX_STYLESHEET_SELECTORS\n ) {\n return;\n }\n\n scanned.count += 1;\n\n if (rule instanceof CSSStyleRule) {\n const parts = rule.selectorText\n .split(\",\")\n .map((selector) => normalizeWhitespace(selector))\n .filter(Boolean);\n\n for (const selector of parts) {\n if (!STYLESHEET_SELECTOR_PATTERN.test(selector)) {\n continue;\n }\n selectors.add(selector);\n if (selectors.size >= MAX_STYLESHEET_SELECTORS) {\n return;\n }\n }\n continue;\n }\n\n if (\"cssRules\" in rule) {\n try {\n const nestedRules = (rule as CSSMediaRule).cssRules;\n collectSelectorsFromRuleList(nestedRules, selectors, scanned);\n } catch {\n // Ignore inaccessible nested rules.\n }\n }\n }\n};\n\nconst collectStylesheetSelectors = (): string[] => {\n const selectors = new Set<string>();\n const scanned = { count: 0 };\n\n for (const stylesheet of Array.from(document.styleSheets)) {\n if (\n scanned.count >= MAX_STYLESHEET_RULES ||\n selectors.size >= MAX_STYLESHEET_SELECTORS\n ) {\n break;\n }\n\n try {\n if (!stylesheet.cssRules) {\n continue;\n }\n collectSelectorsFromRuleList(stylesheet.cssRules, selectors, scanned);\n } catch {\n // Ignore cross-origin stylesheets.\n }\n }\n\n return Array.from(selectors).map((selector) => `- ${selector}`);\n};\n\nconst collectPageSignalSnapshot = (): PageSignalSnapshot => {\n const allElements = Array.from(document.querySelectorAll(\"*\"));\n const sampledElements = allElements.slice(0, MAX_PAGE_SCAN_ELEMENTS);\n\n const links: string[] = [];\n const linkSet = new Set<string>();\n const interactableCandidates = new Map<string, InteractableCandidate>();\n const tagCounts = new Map<string, number>();\n const roleCounts = new Map<string, number>();\n const eventCounts = new Map<string, number>();\n const displayCounts = new Map<string, number>();\n const positionCounts = new Map<string, number>();\n const styleSignalCounts = new Map<string, number>();\n\n let visibleElements = 0;\n let maxDepth = 0;\n let semanticInteractables = 0;\n let nonSemanticInteractables = 0;\n let eventHintElements = 0;\n let tabStopElements = 0;\n let pointerCursorElements = 0;\n let dataHintElements = 0;\n let ariaHintElements = 0;\n let contentEditableElements = 0;\n\n for (let order = 0; order < sampledElements.length; order += 1) {\n const element = sampledElements[order];\n const tag = element.tagName.toLowerCase();\n\n if (NON_CONTENT_TAGS.has(tag)) {\n continue;\n }\n\n if (!isVisible(element)) {\n continue;\n }\n\n visibleElements += 1;\n bumpCount(tagCounts, tag);\n\n const role = getPrimaryRole(element);\n if (role) {\n bumpCount(roleCounts, role);\n }\n\n const depth = getElementDepth(element);\n if (depth > maxDepth) {\n maxDepth = depth;\n }\n\n const style = window.getComputedStyle(element);\n if (TRACKED_DISPLAY_VALUES.has(style.display)) {\n bumpCount(displayCounts, style.display);\n }\n if (TRACKED_POSITION_VALUES.has(style.position)) {\n bumpCount(positionCounts, style.position);\n }\n\n const computedStyleSignals = getComputedStyleSignals(style);\n for (const styleSignal of computedStyleSignals) {\n bumpCount(styleSignalCounts, styleSignal);\n }\n\n const eventHints = getEventHints(element);\n if (eventHints.length > 0) {\n eventHintElements += 1;\n for (const eventName of eventHints) {\n bumpCount(eventCounts, eventName);\n }\n }\n\n const tabIndex = parseTabIndex(element.getAttribute(\"tabindex\"));\n const hasTabStop = tabIndex !== null && tabIndex >= 0;\n if (hasTabStop) {\n tabStopElements += 1;\n }\n\n const hasPointerCursor = style.cursor === \"pointer\";\n if (hasPointerCursor) {\n pointerCursorElements += 1;\n }\n\n const dataHints = getDataInteractionHints(element);\n if (dataHints.length > 0) {\n dataHintElements += 1;\n }\n\n const ariaHints = getAriaInteractionHints(element);\n if (ariaHints.length > 0) {\n ariaHintElements += 1;\n }\n\n const isContentEditable = element.getAttribute(\"contenteditable\") === \"true\";\n if (isContentEditable) {\n contentEditableElements += 1;\n }\n\n const href = element.getAttribute(\"href\");\n const isNativeInteractive = NATIVE_INTERACTIVE_TAGS.has(tag) && (tag !== \"a\" || Boolean(href));\n const isRoleInteractive = INTERACTIVE_ROLES.has(role);\n const isDisabled =\n element.hasAttribute(\"disabled\") ||\n element.getAttribute(\"aria-disabled\") === \"true\";\n\n if (\n tag === \"a\" &&\n href &&\n !href.startsWith(\"#\") &&\n !href.startsWith(\"javascript:\")\n ) {\n const absoluteHref = toAbsoluteUrl(href);\n const label = getElementLabel(element) || absoluteHref;\n const line = `- ${label} -> ${absoluteHref}`;\n\n if (!linkSet.has(line)) {\n linkSet.add(line);\n links.push(line);\n if (links.length >= MAX_LINKS) {\n // Keep scanning other elements for page blueprint and interactables.\n }\n }\n }\n\n const hasInteractionSignals =\n isNativeInteractive ||\n isRoleInteractive ||\n isContentEditable ||\n eventHints.length > 0 ||\n hasTabStop ||\n hasPointerCursor ||\n dataHints.length > 0 ||\n ariaHints.length > 0;\n\n if (!hasInteractionSignals || isDisabled) {\n continue;\n }\n\n if (isNativeInteractive) {\n semanticInteractables += 1;\n } else {\n nonSemanticInteractables += 1;\n }\n\n const selector = buildSelector(element);\n const label = getElementLabel(element);\n const styleSignals = computedStyleSignals;\n const signalTokens: string[] = [];\n\n if (eventHints.length > 0) {\n signalTokens.push(`evt:${eventHints.join(\"|\")}`);\n }\n if (isRoleInteractive) {\n signalTokens.push(`role:${role}`);\n }\n if (hasTabStop) {\n signalTokens.push(`tab:${tabIndex}`);\n }\n if (dataHints.length > 0) {\n signalTokens.push(`data:${dataHints.join(\"|\")}`);\n }\n if (ariaHints.length > 0) {\n signalTokens.push(`aria:${ariaHints.join(\"|\")}`);\n }\n if (styleSignals.length > 0) {\n signalTokens.push(`css:${styleSignals.join(\"|\")}`);\n } else if (hasPointerCursor) {\n signalTokens.push(\"css:cursor:pointer\");\n }\n\n const signalBlock =\n signalTokens.length > 0 ? ` [${signalTokens.join(\"; \")}]` : \"\";\n const line = `- ${tag} ${selector}${signalBlock} (${label})`;\n\n const score =\n eventHints.length * 5 +\n (isNativeInteractive ? 5 : 0) +\n (isRoleInteractive ? 4 : 0) +\n (hasTabStop ? 2 : 0) +\n (hasPointerCursor ? 2 : 0) +\n (dataHints.length > 0 ? 2 : 0) +\n (ariaHints.length > 0 ? 1 : 0) +\n (isContentEditable ? 2 : 0);\n\n const existing = interactableCandidates.get(line);\n if (!existing || score > existing.score) {\n interactableCandidates.set(line, { line, score, order });\n }\n }\n\n const interactables = Array.from(interactableCandidates.values())\n .sort((a, b) => b.score - a.score || a.order - b.order)\n .slice(0, MAX_INTERACTABLES)\n .map((candidate) => candidate.line);\n\n const interactiveRoleCounts = new Map(\n Array.from(roleCounts.entries()).filter(([role]) =>\n INTERACTIVE_ROLES.has(role),\n ),\n );\n\n const interactionSignals = [\n `- coverage: semantic=${semanticInteractables}, non-semantic=${nonSemanticInteractables}, contenteditable=${contentEditableElements}`,\n `- listener hints: ${formatTopCounts(eventCounts, 8)}`,\n `- interaction cues: tabindex>=0=${tabStopElements}, pointer-cursor=${pointerCursorElements}, data-hints=${dataHintElements}, aria-hints=${ariaHintElements}`,\n `- role hints: ${formatTopCounts(interactiveRoleCounts, 8)}`,\n `- css footprint: ${formatTopCounts(styleSignalCounts, 10)}`,\n \"- listener scope: inline/on* handlers are detected directly; addEventListener handlers are inferred via cues.\",\n ];\n\n const branchDigest = collectDomBranchDigest();\n const pageBlueprint = [\n `- nodes: total=${allElements.length}, scanned=${sampledElements.length}, visible=${visibleElements}, max-depth=${maxDepth}${allElements.length > sampledElements.length ? \", sampling=on\" : \"\"}`,\n `- tag density: ${formatTopCounts(tagCounts, 10)}`,\n `- role density: ${formatTopCounts(roleCounts, 8)}`,\n `- layout density: display(${formatTopCounts(displayCounts, 6)}), position(${formatTopCounts(positionCounts, 4)})`,\n `- branch digest: ${branchDigest.length > 0 ? branchDigest.join(\" || \") : \"none\"}`,\n ];\n\n return {\n links: links.slice(0, MAX_LINKS),\n interactables,\n interactionSignals,\n styleSelectors: collectStylesheetSelectors(),\n pageBlueprint,\n };\n};\n\nexport const buildPageContextSummary = (\n input: PageContextSummaryInput,\n): string => {\n const sections = [\n formatSection(\"Meta\", [\n `- URL: ${input.url || \"unknown\"}`,\n `- Title: ${input.title || \"unknown\"}`,\n `- Lang: ${input.lang || \"unknown\"}`,\n ]),\n formatSection(\"Headings\", input.headings),\n formatSection(\"Landmark Snapshot\", input.landmarks),\n formatSection(\"Interaction Signals\", input.interactionSignals),\n formatSection(\"Stylesheet Selector Snapshot\", input.styleSelectors),\n formatSection(\"Compressed Page Blueprint\", input.pageBlueprint),\n formatSection(\"Top Links\", input.links),\n formatSection(\"Top Interactables\", input.interactables),\n formatSection(\"Main Content Snippets\", input.textSnippets),\n formatSection(\"OuterHTML Skeleton\", [\n `- ${input.outerHtmlDigest || \"unavailable\"}`,\n ]),\n ];\n\n return sections.join(\"\\n\\n\");\n};\n\nexport const clearPageContextCache = (): void => {\n pageContextCache.clear();\n cacheHydrated = false;\n if (typeof sessionStorage !== \"undefined\") {\n sessionStorage.removeItem(PAGE_CONTEXT_CACHE_KEY);\n }\n};\n\nexport const getCachedPageContexts = (): CachedPageContextEntry[] => {\n hydrateCacheFromStorage();\n return Array.from(pageContextCache.values()).sort(\n (a, b) => b.capturedAt - a.capturedAt,\n );\n};\n\nexport const getPageContext = (): PageContext => {\n if (typeof window === \"undefined\" || typeof document === \"undefined\") {\n return {\n links: [],\n interactables: [],\n summary: \"\",\n };\n }\n\n hydrateCacheFromStorage();\n const url = canonicalUrl(window.location.href);\n const cached = pageContextCache.get(url);\n if (cached) {\n console.info(`[Autic] context cache hit url=${url}`);\n return {\n links: cached.links,\n interactables: cached.interactables,\n summary: buildSummaryWithHistory(cached),\n };\n }\n\n console.info(`[Autic] context cache miss url=${url}`);\n\n const snapshot = collectPageSignalSnapshot();\n const headings = Array.from(document.querySelectorAll(\"h1, h2, h3\"))\n .filter((element) => isVisible(element))\n .map((element) =>\n `- ${normalizeWhitespace(element.textContent || \"\")}`,\n )\n .filter((line) => line !== \"- \")\n .slice(0, MAX_HEADINGS);\n\n const summary = buildPageContextSummary({\n url,\n title: document.title,\n lang: document.documentElement.lang,\n headings,\n landmarks: collectLandmarkSnapshot(),\n links: snapshot.links,\n interactables: snapshot.interactables,\n interactionSignals: snapshot.interactionSignals,\n styleSelectors: snapshot.styleSelectors,\n pageBlueprint: snapshot.pageBlueprint,\n textSnippets: collectTextSnippets(),\n outerHtmlDigest: buildOuterHtmlDigest(),\n });\n\n const entry: CachedPageContextEntry = {\n url,\n summary,\n links: snapshot.links,\n interactables: snapshot.interactables,\n capturedAt: Date.now(),\n version: PAGE_CONTEXT_CACHE_VERSION,\n };\n\n pageContextCache.set(url, entry);\n pruneOldestCacheEntries();\n persistCacheToStorage();\n console.info(\n `[Autic] context cache stored url=${url} size=${pageContextCache.size}`,\n );\n\n return {\n links: entry.links,\n interactables: entry.interactables,\n summary: buildSummaryWithHistory(entry),\n };\n};\n","/**\n * Tunable parameters for page context collection.\n *\n * Adjust these values to control how much information the agent receives\n * about the current page. Lower values reduce token usage (and latency);\n * higher values give the LLM more detail to work with.\n */\n\n/** Maximum number of links to include in the context. */\nexport const MAX_LINKS = 20;\n\n/** Maximum number of interactable elements to include. */\nexport const MAX_INTERACTABLES = 24;\n\n/** Maximum number of headings (h1-h3) to include. */\nexport const MAX_HEADINGS = 10;\n\n/** Maximum number of main-content text snippets. */\nexport const MAX_TEXT_SNIPPETS = 4;\n\n/** Maximum number of pages kept in the in-memory context cache. */\nexport const MAX_CACHED_PAGES = 20;\n\n/** Maximum number of DOM elements scanned per page. */\nexport const MAX_PAGE_SCAN_ELEMENTS = 2000;\n\n/** Maximum number of event-handler hints collected per element. */\nexport const MAX_EVENT_HINTS_PER_ELEMENT = 4;\n\n/** Maximum top-level DOM branches sampled for the branch digest. */\nexport const MAX_BRANCH_SAMPLES = 4;\n\n/** Maximum tree depth for each branch digest sample. */\nexport const MAX_BRANCH_DEPTH = 2;\n\n/** Maximum stylesheet selectors to include in summary. */\nexport const MAX_STYLESHEET_SELECTORS = 80;\n\n/** Maximum CSS rules to scan while collecting stylesheet selectors. */\nexport const MAX_STYLESHEET_RULES = 700;\n","import { getPageContext } from \"./context\";\nimport { COLORS } from \"../styles/constants\";\n\nconst AGENT_CURSOR_ID = \"auticbot-agent-cursor\";\nexport const CURSOR_MOVE_DURATION_MS = 900;\nexport const SCROLL_DURATION_MS = 900;\nconst CURSOR_EASING = \"cubic-bezier(0.4, 0, 0.2, 1)\";\nconst CURSOR_HOVER_RADIUS_PX = 14;\nconst CURSOR_DIAMETER_PX = 25;\n\n// ── Pending Agent Resume State (survives page reloads) ──────────────\n\nconst RESUME_STORAGE_KEY = \"bulut_agent_resume\";\nconst RESUME_TTL_MS = 60_000; // 1 minute\n\nexport interface PendingAgentResume {\n sessionId: string;\n projectId: string;\n model: string;\n voice: string;\n accessibilityMode: boolean;\n pendingToolCalls: Array<{\n call_id: string;\n tool: string;\n args: Record<string, unknown>;\n }>;\n completedResults: Array<{ call_id: string; result: string }>;\n savedAt: number;\n}\n\nexport const savePendingAgentResume = (\n state: Omit<PendingAgentResume, \"savedAt\">,\n): void => {\n if (typeof localStorage === \"undefined\") return;\n try {\n localStorage.setItem(\n RESUME_STORAGE_KEY,\n JSON.stringify({ ...state, savedAt: Date.now() }),\n );\n } catch {\n // localStorage may be full or blocked\n }\n};\n\nexport const getPendingAgentResume = (): PendingAgentResume | null => {\n if (typeof localStorage === \"undefined\") return null;\n const raw = localStorage.getItem(RESUME_STORAGE_KEY);\n if (!raw) return null;\n try {\n const parsed = JSON.parse(raw) as PendingAgentResume;\n if (Date.now() - parsed.savedAt > RESUME_TTL_MS) {\n clearPendingAgentResume();\n return null;\n }\n return parsed;\n } catch {\n clearPendingAgentResume();\n return null;\n }\n};\n\nexport const clearPendingAgentResume = (): void => {\n if (typeof localStorage === \"undefined\") return;\n localStorage.removeItem(RESUME_STORAGE_KEY);\n};\n\ntype InteractAction = \"move\" | \"click\" | \"type\" | \"submit\";\n\ninterface InteractToolCall {\n tool: \"interact\";\n action: InteractAction;\n selector?: string;\n text?: string;\n x?: number;\n y?: number;\n}\n\ninterface NavigateToolCall {\n tool: \"navigate\";\n url: string;\n}\n\ninterface GetPageContextToolCall {\n tool: \"getPageContext\";\n}\n\ninterface ScrollToolCall {\n tool: \"scroll\";\n selector: string;\n}\n\nexport type AgentToolCall =\n | InteractToolCall\n | NavigateToolCall\n | GetPageContextToolCall\n | ScrollToolCall;\n\nexport interface ParsedAgentResponse {\n reply: string;\n toolCalls: AgentToolCall[];\n}\n\ninterface JsonObject {\n [key: string]: unknown;\n}\n\ninterface ResolvedTarget {\n element?: HTMLElement;\n x: number;\n y: number;\n}\n\nconst isObject = (value: unknown): value is JsonObject =>\n typeof value === \"object\" && value !== null && !Array.isArray(value);\n\nconst asString = (value: unknown): string | undefined =>\n typeof value === \"string\" && value.trim() ? value.trim() : undefined;\n\nconst asNumber = (value: unknown): number | undefined =>\n typeof value === \"number\" && Number.isFinite(value) ? value : undefined;\n\nconst extractJsonCandidate = (raw: string): string => {\n const trimmed = raw.trim();\n const fencedMatch = trimmed.match(/```(?:json)?\\s*([\\s\\S]*?)\\s*```/i);\n if (fencedMatch?.[1]) {\n return fencedMatch[1].trim();\n }\n return trimmed;\n};\n\nconst extractFirstJsonObject = (input: string): string | null => {\n const start = input.indexOf(\"{\");\n if (start < 0) {\n return null;\n }\n\n let depth = 0;\n let inString = false;\n let isEscaped = false;\n\n for (let i = start; i < input.length; i += 1) {\n const char = input[i];\n\n if (inString) {\n if (isEscaped) {\n isEscaped = false;\n } else if (char === \"\\\\\") {\n isEscaped = true;\n } else if (char === '\"') {\n inString = false;\n }\n continue;\n }\n\n if (char === '\"') {\n inString = true;\n continue;\n }\n\n if (char === \"{\") {\n depth += 1;\n continue;\n }\n\n if (char === \"}\") {\n depth -= 1;\n if (depth === 0) {\n return input.slice(start, i + 1);\n }\n }\n }\n\n return null;\n};\n\nconst parseJsonFromRaw = (raw: string): unknown => {\n const candidate = extractJsonCandidate(raw);\n try {\n return JSON.parse(candidate);\n } catch {\n const objectCandidate = extractFirstJsonObject(candidate);\n if (!objectCandidate) {\n return null;\n }\n try {\n return JSON.parse(objectCandidate);\n } catch {\n return null;\n }\n }\n};\n\nconst sanitizeToolCalls = (value: unknown): AgentToolCall[] => {\n if (!Array.isArray(value)) {\n return [];\n }\n\n const toolCalls: AgentToolCall[] = [];\n\n for (const item of value) {\n if (!isObject(item)) {\n continue;\n }\n\n if (item.tool === \"interact\") {\n const action = asString(item.action) as InteractAction | undefined;\n if (!action || ![\"move\", \"click\", \"type\", \"submit\"].includes(action)) {\n continue;\n }\n\n toolCalls.push({\n tool: \"interact\",\n action,\n selector: asString(item.selector),\n text: typeof item.text === \"string\" ? item.text : undefined,\n x: asNumber(item.x),\n y: asNumber(item.y),\n });\n continue;\n }\n\n if (item.tool === \"navigate\") {\n const url = asString(item.url);\n if (!url) {\n continue;\n }\n\n toolCalls.push({\n tool: \"navigate\",\n url,\n });\n continue;\n }\n\n if (item.tool === \"getPageContext\") {\n toolCalls.push({\n tool: \"getPageContext\",\n });\n continue;\n }\n\n if (item.tool === \"scroll\") {\n const selector = asString(item.selector);\n if (!selector) {\n continue;\n }\n\n toolCalls.push({\n tool: \"scroll\",\n selector,\n });\n }\n }\n\n return toolCalls;\n};\n\nexport const parseAgentResponse = (raw: string): ParsedAgentResponse => {\n const parsed = parseJsonFromRaw(raw);\n if (!isObject(parsed)) {\n return {\n reply: raw.trim(),\n toolCalls: [],\n };\n }\n\n const reply = asString(parsed.reply) || \"\";\n const toolCalls = sanitizeToolCalls(parsed.tool_calls ?? parsed.toolCalls);\n\n return {\n reply,\n toolCalls,\n };\n};\n\nexport const clamp = (value: number, min: number, max: number): number =>\n Math.min(max, Math.max(min, value));\n\nexport const easeInOutCubic = (progress: number): number => {\n if (progress < 0.5) {\n return 4 * progress * progress * progress;\n }\n return 1 - Math.pow(-2 * progress + 2, 3) / 2;\n};\n\nexport const easeInOutSine = (progress: number): number =>\n -(Math.cos(Math.PI * progress) - 1) / 2;\n\nexport const isRectOutsideViewport = (\n rect: Pick<DOMRect, \"top\" | \"bottom\">,\n viewportHeight: number,\n): boolean => rect.top < 0 || rect.bottom > viewportHeight;\n\nexport const computeCenteredScrollTop = (\n currentScrollY: number,\n rectTop: number,\n rectHeight: number,\n viewportHeight: number,\n maxScrollTop: number,\n): number => {\n const desired =\n currentScrollY + rectTop - (viewportHeight / 2 - rectHeight / 2);\n return clamp(desired, 0, Math.max(0, maxScrollTop));\n};\n\nexport const animateWindowScrollTo = async (\n targetY: number,\n durationMs: number = SCROLL_DURATION_MS,\n): Promise<void> => {\n if (typeof window === \"undefined\") {\n return;\n }\n\n const startY = window.scrollY;\n const delta = targetY - startY;\n if (Math.abs(delta) < 1) {\n return;\n }\n\n await new Promise<void>((resolve) => {\n const raf =\n window.requestAnimationFrame ||\n ((callback: FrameRequestCallback) => window.setTimeout(() => callback(performance.now()), 16));\n\n const startTime = performance.now();\n\n const step = (now: number) => {\n const elapsed = now - startTime;\n const progress = clamp(elapsed / durationMs, 0, 1);\n const eased = easeInOutSine(progress);\n window.scrollTo(0, startY + delta * eased);\n\n if (progress < 1) {\n raf(step);\n } else {\n resolve();\n }\n };\n\n raf(step);\n });\n};\n\nconst setCursorPosition = (cursor: HTMLElement, x: number, y: number) => {\n cursor.style.left = `${x}px`;\n cursor.style.top = `${y}px`;\n};\n\nconst getCursorPosition = (cursor: HTMLElement): { x: number; y: number } => ({\n x: Number.parseFloat(cursor.style.left) || 0,\n y: Number.parseFloat(cursor.style.top) || 0,\n});\n\nconst setCursorVisibility = (cursor: HTMLElement, visible: boolean) => {\n cursor.style.opacity = visible ? \"1\" : \"0\";\n};\n\nconst isVisibleElement = (element: HTMLElement): boolean => {\n const style = window.getComputedStyle(element);\n return style.display !== \"none\" && style.visibility !== \"hidden\" && style.opacity !== \"0\";\n};\n\nconst getBulutShadowRoots = (): ShadowRoot[] => {\n const roots: ShadowRoot[] = [];\n\n const defaultHost = document.getElementById(\"bulut-container\");\n if (defaultHost?.shadowRoot) {\n roots.push(defaultHost.shadowRoot);\n }\n\n const allElements = document.querySelectorAll<HTMLElement>(\"*\");\n for (const el of allElements) {\n if (!el.shadowRoot) {\n continue;\n }\n if (!roots.includes(el.shadowRoot)) {\n roots.push(el.shadowRoot);\n }\n }\n\n return roots;\n};\n\nconst findAgentUiAnchorElement = (): HTMLElement | null => {\n const roots = getBulutShadowRoots();\n\n for (const root of roots) {\n const panel = root.querySelector<HTMLElement>(\".bulut-chat-window\");\n if (panel && isVisibleElement(panel)) {\n return panel;\n }\n }\n\n for (const root of roots) {\n const button = root.querySelector<HTMLElement>(\".bulut-fab-container\");\n if (button && isVisibleElement(button)) {\n return button;\n }\n }\n\n return null;\n};\n\nconst getAgentWindowTopLeft = (): { x: number; y: number } => {\n const anchor = findAgentUiAnchorElement();\n if (!anchor) {\n return {\n x: CURSOR_DIAMETER_PX / 2,\n y: CURSOR_DIAMETER_PX / 2,\n };\n }\n\n const rect = anchor.getBoundingClientRect();\n return {\n x: rect.left + window.scrollX + CURSOR_DIAMETER_PX / 2,\n y: rect.top + window.scrollY + CURSOR_DIAMETER_PX / 2,\n };\n};\n\nexport const hideAgentCursor = (): void => {\n if (typeof document === \"undefined\" || typeof window === \"undefined\") {\n return;\n }\n\n const cursor = document.getElementById(AGENT_CURSOR_ID);\n if (!(cursor instanceof HTMLElement)) {\n return;\n }\n\n setCursorVisibility(cursor, false);\n};\n\nlet cursorHoverTrackingInitialized = false;\nconst initializeCursorHoverTracking = () => {\n if (cursorHoverTrackingInitialized) {\n return;\n }\n cursorHoverTrackingInitialized = true;\n\n document.addEventListener(\"mousemove\", (event) => {\n const cursor = document.getElementById(AGENT_CURSOR_ID);\n if (!(cursor instanceof HTMLElement)) {\n return;\n }\n\n if (cursor.style.opacity !== \"1\") {\n return;\n }\n\n const { x, y } = getCursorPosition(cursor);\n const pointerX = event.pageX;\n const pointerY = event.pageY;\n const distance = Math.hypot(pointerX - x, pointerY - y);\n\n if (distance <= CURSOR_HOVER_RADIUS_PX) {\n setCursorVisibility(cursor, false);\n }\n });\n};\n\nconst ensureCursor = (): HTMLElement => {\n const existing = document.getElementById(AGENT_CURSOR_ID);\n if (existing) {\n // Keep cursor color in sync with the current theme\n existing.style.background = COLORS.primary;\n initializeCursorHoverTracking();\n return existing as HTMLElement;\n }\n\n const cursor = document.createElement(\"div\");\n cursor.id = AGENT_CURSOR_ID;\n cursor.style.position = \"absolute\";\n const startPosition = getAgentWindowTopLeft();\n cursor.style.left = `${startPosition.x}px`;\n cursor.style.top = `${startPosition.y}px`;\n cursor.style.opacity = \"0\";\n const width = CURSOR_DIAMETER_PX;\n cursor.style.width = `${width}px`;\n cursor.style.height = `${width}px`;\n cursor.style.borderRadius = \"50%\";\n const baseColor = COLORS.primary;\n cursor.style.background = baseColor;\n const border = 25 * 16 / 100;\n cursor.style.border = `${border}px solid #ffffff`;\n cursor.style.boxShadow = \"0px 0px 10px rgba(0, 11, 26, 0.5)\";\n cursor.style.boxSizing = \"border-box\";\n cursor.style.zIndex = \"2147483647\";\n cursor.style.pointerEvents = \"none\";\n cursor.style.transform = \"translate(-50%, -50%)\";\n cursor.style.transition = `left ${CURSOR_MOVE_DURATION_MS}ms ${CURSOR_EASING}, top ${CURSOR_MOVE_DURATION_MS}ms ${CURSOR_EASING}, opacity 150ms ease-out`;\n document.body.appendChild(cursor);\n initializeCursorHoverTracking();\n console.info(`[Autic] cursor created color=${baseColor} duration=${CURSOR_MOVE_DURATION_MS}ms`);\n return cursor;\n};\n\nconst waitForNextAnimationFrame = async (): Promise<void> => {\n const raf =\n window.requestAnimationFrame ||\n ((callback: FrameRequestCallback) => window.setTimeout(() => callback(performance.now()), 16));\n await new Promise<void>((resolve) => {\n raf(() => resolve());\n });\n};\n\nconst moveCursor = async (x: number, y: number) => {\n const cursor = ensureCursor();\n if (cursor.dataset.transitionReady !== \"true\") {\n cursor.dataset.transitionReady = \"true\";\n await waitForNextAnimationFrame();\n }\n\n const startPosition = getAgentWindowTopLeft();\n setCursorVisibility(cursor, true);\n setCursorPosition(cursor, startPosition.x, startPosition.y);\n await new Promise((resolve) => setTimeout(resolve, CURSOR_MOVE_DURATION_MS));\n\n setCursorPosition(cursor, x, y);\n await new Promise((resolve) => setTimeout(resolve, CURSOR_MOVE_DURATION_MS));\n};\n\nconst getElementCenter = (element: HTMLElement): { x: number; y: number } => {\n const rect = element.getBoundingClientRect();\n return {\n x: rect.left + window.scrollX + rect.width / 2,\n y: rect.top + window.scrollY + rect.height / 2,\n };\n};\n\nconst CONTAINS_SELECTOR_PATTERN = /^(.*?):contains\\((['\"])(.*?)\\2\\)\\s*$/;\n\nconst findElementBySelector = (selector: string): Element | null => {\n try {\n return document.querySelector(selector);\n } catch (error) {\n const containsMatch = selector.match(CONTAINS_SELECTOR_PATTERN);\n if (!containsMatch) {\n console.warn(`AuticBot selector invalid: ${selector}`, error);\n return null;\n }\n\n const baseSelector = containsMatch[1]?.trim() || \"*\";\n const expectedText = containsMatch[3]?.trim() || \"\";\n if (!expectedText) {\n console.warn(`AuticBot selector contains empty text: ${selector}`);\n return null;\n }\n\n try {\n const candidates = document.querySelectorAll(baseSelector);\n for (const candidate of candidates) {\n if (candidate.textContent?.includes(expectedText)) {\n return candidate;\n }\n }\n return null;\n } catch (fallbackError) {\n console.warn(`AuticBot selector fallback invalid: ${selector}`, fallbackError);\n return null;\n }\n }\n};\n\nconst resolveTarget = (call: InteractToolCall): ResolvedTarget | null => {\n if (call.selector) {\n const selected = findElementBySelector(call.selector);\n\n if (selected instanceof HTMLElement) {\n const center = getElementCenter(selected);\n return {\n element: selected,\n x: center.x,\n y: center.y,\n };\n }\n console.warn(`AuticBot interact: selector not found: ${call.selector}`);\n }\n\n if (typeof call.x === \"number\" && typeof call.y === \"number\") {\n return {\n x: call.x,\n y: call.y,\n };\n }\n\n console.warn(\"AuticBot interact: missing target selector or coordinates.\", call);\n return null;\n};\n\nconst dispatchMouseEvent = (\n element: HTMLElement,\n type: string,\n x: number,\n y: number,\n) => {\n element.dispatchEvent(\n new MouseEvent(type, {\n bubbles: true,\n cancelable: true,\n view: window,\n clientX: x - window.scrollX,\n clientY: y - window.scrollY,\n }),\n );\n};\n\nconst setNativeInputLikeValue = (\n element: HTMLInputElement | HTMLTextAreaElement,\n text: string,\n) => {\n const prototype =\n element instanceof HTMLTextAreaElement\n ? HTMLTextAreaElement.prototype\n : HTMLInputElement.prototype;\n const descriptor = Object.getOwnPropertyDescriptor(prototype, \"value\");\n if (descriptor?.set) {\n descriptor.set.call(element, text);\n } else {\n element.value = text;\n }\n\n // Keep both current and default values aligned so client re-focus\n // does not appear to \"erase\" tool-filled form fields.\n element.defaultValue = text;\n element.setAttribute(\"value\", text);\n};\n\nconst typeIntoElement = (element: HTMLElement, text: string) => {\n if (element instanceof HTMLInputElement || element instanceof HTMLTextAreaElement) {\n element.focus();\n setNativeInputLikeValue(element, text);\n element.dispatchEvent(new Event(\"input\", { bubbles: true }));\n element.dispatchEvent(new Event(\"change\", { bubbles: true }));\n return;\n }\n\n if (element.isContentEditable) {\n element.focus();\n element.textContent = text;\n element.dispatchEvent(new Event(\"input\", { bubbles: true }));\n return;\n }\n\n console.warn(\n \"AuticBot interact: type action requires input, textarea, or contenteditable target.\",\n );\n};\n\nconst submitElement = (element: HTMLElement) => {\n if (element.tagName === \"FORM\") {\n (element as HTMLFormElement).requestSubmit();\n return;\n }\n\n if (element.tagName === \"BUTTON\" && (element as HTMLButtonElement).form) {\n (element as HTMLButtonElement).form?.requestSubmit();\n return;\n }\n\n const parentForm = element.closest(\"form\");\n if (parentForm) {\n parentForm.requestSubmit();\n return;\n }\n\n console.warn(\"AuticBot interact: submit action requires a form target.\");\n};\n\nconst slowScrollElementIntoView = async (element: HTMLElement): Promise<void> => {\n await slowScrollElementIntoViewWithMode(element, false);\n};\n\nconst slowScrollElementIntoViewWithMode = async (\n element: HTMLElement,\n forceCenter: boolean,\n): Promise<void> => {\n const rect = element.getBoundingClientRect();\n const viewportHeight = window.innerHeight;\n\n if (!forceCenter && !isRectOutsideViewport(rect, viewportHeight)) {\n return;\n }\n\n const maxScrollTop = Math.max(\n 0,\n Math.max(document.body.scrollHeight, document.documentElement.scrollHeight) - viewportHeight,\n );\n const targetY = computeCenteredScrollTop(\n window.scrollY,\n rect.top,\n rect.height,\n viewportHeight,\n maxScrollTop,\n );\n\n await animateWindowScrollTo(targetY, SCROLL_DURATION_MS);\n};\n\nconst executeScroll = async (call: ScrollToolCall) => {\n const selected = findElementBySelector(call.selector);\n if (!(selected instanceof HTMLElement)) {\n console.warn(`AuticBot scroll: selector not found: ${call.selector}`);\n return;\n }\n\n await slowScrollElementIntoViewWithMode(selected, true);\n const center = getElementCenter(selected);\n await moveCursor(center.x, center.y);\n};\n\nconst executeInteract = async (call: InteractToolCall) => {\n const target = resolveTarget(call);\n if (!target) {\n return;\n }\n\n if (call.action === \"click\" && target.element) {\n await slowScrollElementIntoView(target.element);\n const center = getElementCenter(target.element);\n target.x = center.x;\n target.y = center.y;\n }\n\n await moveCursor(target.x, target.y);\n\n if (call.action === \"move\") {\n return;\n }\n\n if (!target.element) {\n console.warn(\"AuticBot interact: target element not available for action.\", call.action);\n return;\n }\n\n if (call.action === \"click\") {\n dispatchMouseEvent(target.element, \"pointerdown\", target.x, target.y);\n dispatchMouseEvent(target.element, \"mousedown\", target.x, target.y);\n dispatchMouseEvent(target.element, \"pointerup\", target.x, target.y);\n dispatchMouseEvent(target.element, \"mouseup\", target.x, target.y);\n target.element.click();\n return;\n }\n\n if (call.action === \"type\") {\n typeIntoElement(target.element, call.text ?? \"\");\n return;\n }\n\n submitElement(target.element);\n};\n\nconst isSamePageNavigation = (targetUrl: string): boolean => {\n try {\n const current = new URL(window.location.href);\n const target = new URL(targetUrl);\n return current.origin === target.origin && current.pathname === target.pathname;\n } catch {\n return false;\n }\n};\n\n/**\n * Find the best matching link element for a target URL.\n * Supports exact href match, partial path/query/hash match,\n * and text-content match for framework <Link> components.\n */\nconst findMatchingLinkForTarget = (targetUrl: string): HTMLElement | null => {\n let parsedTarget: URL | null = null;\n try {\n parsedTarget = new URL(targetUrl, window.location.href);\n } catch {\n // will fall through to text-based matching\n }\n\n const allLinks = Array.from(\n document.querySelectorAll('a[href], [role=\"link\"][href], [data-href]'),\n ) as HTMLElement[];\n\n // 1. Exact href match\n for (const el of allLinks) {\n if (el instanceof HTMLAnchorElement && el.href === parsedTarget?.href) {\n return el;\n }\n }\n\n if (parsedTarget) {\n // 2. Match by pathname + search + hash (ignoring origin)\n for (const el of allLinks) {\n if (!(el instanceof HTMLAnchorElement)) continue;\n try {\n const elUrl = new URL(el.href, window.location.href);\n if (\n elUrl.pathname === parsedTarget.pathname &&\n elUrl.search === parsedTarget.search &&\n elUrl.hash === parsedTarget.hash\n ) {\n return el;\n }\n } catch {\n continue;\n }\n }\n\n // 3. Match by pathname only (query/hash may differ)\n for (const el of allLinks) {\n if (!(el instanceof HTMLAnchorElement)) continue;\n try {\n const elUrl = new URL(el.href, window.location.href);\n if (elUrl.pathname === parsedTarget.pathname) {\n return el;\n }\n } catch {\n continue;\n }\n }\n\n // 4. Partial href attribute match (covers relative paths, query strings)\n const rawUrl = targetUrl.replace(/^\\//, \"\");\n for (const el of allLinks) {\n const href = el.getAttribute(\"href\") || el.getAttribute(\"data-href\") || \"\";\n if (href && (href === targetUrl || href === rawUrl || href === `/${rawUrl}`)) {\n return el;\n }\n }\n }\n\n // 5. Text-content match (for framework <Link> or <button> navigations)\n const urlSegments = targetUrl\n .replace(/^https?:\\/\\/[^/]+/, \"\")\n .replace(/[?#].*$/, \"\")\n .split(\"/\")\n .filter(Boolean);\n const lastSegment = urlSegments[urlSegments.length - 1] || \"\";\n\n if (lastSegment) {\n // Also search query param values (e.g., ?tab=interact → \"interact\")\n let searchTerms = [lastSegment];\n if (parsedTarget) {\n for (const [, value] of parsedTarget.searchParams.entries()) {\n if (value) searchTerms.push(value);\n }\n if (parsedTarget.hash) {\n searchTerms.push(parsedTarget.hash.replace(/^#/, \"\"));\n }\n }\n searchTerms = searchTerms.map((t) => t.toLowerCase());\n\n // Look across all clickable elements\n const clickables = Array.from(\n document.querySelectorAll(\n 'a, button, [role=\"link\"], [role=\"tab\"], [role=\"button\"], [data-tab], [onclick]',\n ),\n ) as HTMLElement[];\n\n for (const el of clickables) {\n const text = (el.textContent || \"\").trim().toLowerCase();\n const ariaLabel = (el.getAttribute(\"aria-label\") || \"\").toLowerCase();\n const dataTab = (el.getAttribute(\"data-tab\") || \"\").toLowerCase();\n for (const term of searchTerms) {\n if (\n text === term ||\n ariaLabel === term ||\n dataTab === term ||\n text.includes(term)\n ) {\n return el;\n }\n }\n }\n }\n\n return null;\n};\n\nconst executeNavigate = async (call: NavigateToolCall): Promise<boolean> => {\n try {\n const targetUrl = call.url;\n let resolvedUrl: string;\n try {\n resolvedUrl = new URL(targetUrl, window.location.href).href;\n } catch {\n resolvedUrl = targetUrl;\n }\n\n const matchingElement = findMatchingLinkForTarget(targetUrl);\n\n if (matchingElement) {\n console.log(\"AuticBot navigate: clicking element\", resolvedUrl, matchingElement.tagName);\n await slowScrollElementIntoView(matchingElement);\n\n const center = getElementCenter(matchingElement);\n await moveCursor(center.x, center.y);\n\n matchingElement.dispatchEvent(new MouseEvent(\"pointerdown\", { bubbles: true, view: window }));\n matchingElement.dispatchEvent(new MouseEvent(\"mousedown\", { bubbles: true, view: window }));\n matchingElement.dispatchEvent(new MouseEvent(\"pointerup\", { bubbles: true, view: window }));\n matchingElement.dispatchEvent(new MouseEvent(\"mouseup\", { bubbles: true, view: window }));\n matchingElement.click();\n\n return !isSamePageNavigation(resolvedUrl);\n }\n\n // 6. Fallback: direct browser navigation (query param, hash, or full URL)\n console.log(\"AuticBot navigate: no matching element found, using direct navigation\", resolvedUrl);\n\n // Hash-only navigation\n try {\n const parsed = new URL(resolvedUrl);\n if (\n parsed.origin === window.location.origin &&\n parsed.pathname === window.location.pathname &&\n parsed.hash\n ) {\n window.location.hash = parsed.hash;\n return false;\n }\n } catch { /* continue */ }\n\n // Query-param or same-origin navigation via History API\n try {\n const parsed = new URL(resolvedUrl);\n if (parsed.origin === window.location.origin) {\n // Use pushState + popstate to trigger SPA routers\n const newPath = parsed.pathname + parsed.search + parsed.hash;\n window.history.pushState({}, \"\", newPath);\n window.dispatchEvent(new PopStateEvent(\"popstate\", { state: {} }));\n return false;\n }\n } catch { /* continue */ }\n\n // Cross-origin: full page navigation\n window.location.href = resolvedUrl;\n return true;\n } catch (error) {\n console.warn(\"AuticBot navigate: error\", call.url, error);\n return false;\n }\n};\n\nconst executeGetPageContext = async () => {\n const context = getPageContext();\n console.info(\n `[Autic] getPageContext tool executed links=${context.links.length} interactables=${context.interactables.length} summary_len=${context.summary.length}`,\n );\n};\n\nexport const executeToolCalls = async (toolCalls: AgentToolCall[]) => {\n for (const toolCall of toolCalls) {\n if (toolCall.tool === \"interact\") {\n await executeInteract(toolCall);\n continue;\n }\n\n if (toolCall.tool === \"scroll\") {\n await executeScroll(toolCall);\n continue;\n }\n\n if (toolCall.tool === \"getPageContext\") {\n await executeGetPageContext();\n continue;\n }\n\n if (toolCall.tool === \"navigate\") {\n const terminalNavigation = await executeNavigate(toolCall);\n if (terminalNavigation) {\n break;\n }\n }\n }\n};\n\n// ── Agent-mode tool execution (returns results) ─────────────────────\n\nexport type ToolCallWithId = AgentToolCall & {\n call_id: string;\n};\n\nexport interface ToolCallResult {\n call_id: string;\n result: string;\n}\n\n/**\n * Execute a single tool call and return a result string.\n * Used by the agent loop to feed results back into the LLM.\n */\nexport const executeSingleToolCall = async (\n call: ToolCallWithId,\n): Promise<ToolCallResult> => {\n const callId = call.call_id;\n try {\n if (call.tool === \"interact\") {\n await executeInteract(call);\n return {\n call_id: callId,\n result: `Etkileşim başarılı: ${call.action}`,\n };\n }\n\n if (call.tool === \"scroll\") {\n await executeScroll(call);\n return {\n call_id: callId,\n result: \"Öğeye kaydırma başarılı.\",\n };\n }\n\n if (call.tool === \"getPageContext\") {\n const context = getPageContext();\n return {\n call_id: callId,\n result: context.summary,\n };\n }\n\n if (call.tool === \"navigate\") {\n await executeNavigate(call);\n // Wait for navigation / SPA routing to settle\n await new Promise((resolve) => setTimeout(resolve, 1500));\n const context = getPageContext();\n return {\n call_id: callId,\n result: `Navigasyon tamamlandı. Şu anki sayfa: ${window.location.href}\\nSayfa bağlamı: ${context.summary}`,\n };\n }\n\n return { call_id: callId, result: \"Bilinmeyen araç.\" };\n } catch (error) {\n const msg = error instanceof Error ? error.message : String(error);\n console.warn(`[Autic] Tool execution error: ${call.tool}`, error);\n return { call_id: callId, result: `Hata: ${msg}` };\n }\n};\n","import {\n savePendingAgentResume,\n clearPendingAgentResume,\n hideAgentCursor,\n type PendingAgentResume,\n} from \"../agent/tools\";\n\nexport type ChatRole = \"system\" | \"user\" | \"assistant\";\n\nexport interface ChatMessage {\n role: ChatRole;\n content: string;\n}\n\ninterface ApiErrorBody {\n detail?: string;\n error?: string;\n message?: string;\n}\n\ninterface SseEventPayload {\n type?: string;\n session_id?: string;\n user_text?: string;\n assistant_text?: string;\n delta?: string;\n audio?: string;\n format?: string;\n mime_type?: string;\n sample_rate?: number;\n error?: string;\n}\n\ninterface TtsWsEventPayload {\n type?: string;\n request_id?: string;\n seq?: number;\n audio?: string;\n format?: string;\n mime_type?: string;\n sample_rate?: number;\n error?: string;\n retryable?: boolean;\n last_seq?: number;\n}\n\ninterface SttWsEventPayload {\n type?: string;\n session_id?: string;\n seq?: number;\n text?: string;\n error?: string;\n retryable?: boolean;\n}\n\nexport type AudioStreamState = \"rendering\" | \"playing\" | \"done\" | \"fallback\";\nexport const TTS_WS_RETRY_DELAYS_MS = [250, 750, 1500];\n\nconst BULUT_AUDIO_STOP_EVENT = \"bulut:audio-stop\";\nconst activeAudioElements = new Set<HTMLAudioElement>();\nlet audioPlaybackGeneration = 0;\n\nconst normalizeBaseUrl = (baseUrl: string): string => {\n const trimmed = baseUrl.trim().replace(/\\/+$/, \"\");\n if (/^https?:\\/\\//i.test(trimmed)) {\n return trimmed;\n }\n // Treat host-only values like \"api.bulut.lu\" as HTTPS absolute URLs.\n return `https://${trimmed}`;\n};\nconst toWebSocketUrl = (baseUrl: string, path: string): string => {\n const normalized = normalizeBaseUrl(baseUrl);\n const url = new URL(normalized);\n url.protocol = url.protocol === \"https:\" ? \"wss:\" : \"ws:\";\n url.pathname = `${url.pathname.replace(/\\/$/, \"\")}${path}`;\n url.search = \"\";\n url.hash = \"\";\n return url.toString();\n};\n\nconst createRequestId = (): string => {\n if (typeof crypto !== \"undefined\" && \"randomUUID\" in crypto) {\n return crypto.randomUUID();\n }\n return `tts-${Date.now()}-${Math.random().toString(16).slice(2)}`;\n};\n\nexport const parseTtsWsEventPayload = (\n value: unknown,\n): TtsWsEventPayload | null => {\n try {\n if (typeof value !== \"string\") {\n return null;\n }\n return JSON.parse(value) as TtsWsEventPayload;\n } catch {\n return null;\n }\n};\n\nexport const parseSttWsEventPayload = (\n value: unknown,\n): SttWsEventPayload | null => {\n try {\n if (typeof value !== \"string\") {\n return null;\n }\n return JSON.parse(value) as SttWsEventPayload;\n } catch {\n return null;\n }\n};\n\nexport const shouldAcceptAudioSeq = (\n incomingSeq: number,\n highestSeqSeen: number,\n): boolean => incomingSeq > highestSeqSeen;\n\nexport const shouldFallbackToSse = (error: unknown): boolean => {\n if (typeof error === \"object\" && error !== null && \"retryable\" in error) {\n return Boolean((error as { retryable?: boolean }).retryable);\n }\n return true;\n};\n\nconst parseErrorBody = async (response: Response): Promise<string> => {\n try {\n const data = (await response.json()) as ApiErrorBody;\n const detail = data.detail;\n if (typeof detail === \"string\") return detail;\n if (detail && typeof detail === \"object\") return JSON.stringify(detail);\n return data.error || data.message || response.statusText;\n } catch {\n return response.statusText;\n }\n};\n\nconst sleep = (ms: number): Promise<void> =>\n new Promise((resolve) => {\n setTimeout(resolve, ms);\n });\n\nconst registerActiveAudioElement = (audioElement: HTMLAudioElement): void => {\n activeAudioElements.add(audioElement);\n};\n\nconst unregisterActiveAudioElement = (audioElement: HTMLAudioElement): void => {\n activeAudioElements.delete(audioElement);\n};\n\nconst wasPlaybackStoppedAfter = (generationAtStart: number): boolean =>\n audioPlaybackGeneration !== generationAtStart;\n\nexport const getAudioPlaybackGeneration = (): number => audioPlaybackGeneration;\n\nexport const stopActiveAudioPlayback = (): void => {\n audioPlaybackGeneration += 1;\n const active = Array.from(activeAudioElements);\n for (const audioElement of active) {\n try {\n audioElement.dispatchEvent(new Event(BULUT_AUDIO_STOP_EVENT));\n audioElement.pause();\n audioElement.removeAttribute(\"src\");\n audioElement.load();\n } catch {\n // Ignore playback stop errors.\n }\n }\n};\n\nexport const base64ToUint8Array = (base64: string): Uint8Array<ArrayBuffer> => {\n // Strip potential data URI prefix if present\n const cleanBase64 = base64.replace(/^data:audio\\/\\w+;base64,/, \"\");\n const binaryString = atob(cleanBase64);\n const bytes = new Uint8Array(binaryString.length) as Uint8Array<ArrayBuffer>;\n for (let i = 0; i < binaryString.length; i += 1) {\n bytes[i] = binaryString.charCodeAt(i);\n }\n return bytes;\n};\n\nconst blobToBase64 = async (blob: Blob): Promise<string> => {\n const bytes = new Uint8Array(await blob.arrayBuffer());\n const chunkSize = 0x8000;\n let binary = \"\";\n for (let i = 0; i < bytes.length; i += chunkSize) {\n binary += String.fromCharCode(...bytes.subarray(i, i + chunkSize));\n }\n return btoa(binary);\n};\n\nconst createWavHeader = (\n length: number,\n sampleRate: number = 16000,\n): Uint8Array<ArrayBuffer> => {\n const buffer = new ArrayBuffer(44);\n const view = new DataView(buffer);\n const channels = 1;\n\n // RIFF chunk descriptor\n view.setUint32(0, 0x52494646, false); // \"RIFF\"\n view.setUint32(4, 36 + length, true); // file length - 8\n view.setUint32(8, 0x57415645, false); // \"WAVE\"\n\n // fmt sub-chunk\n view.setUint32(12, 0x666d7420, false); // \"fmt \"\n view.setUint32(16, 16, true); // Subchunk1Size (16 for PCM)\n view.setUint16(20, 1, true); // AudioFormat (1 for PCM)\n view.setUint16(22, channels, true); // NumChannels\n view.setUint32(24, sampleRate, true); // SampleRate\n view.setUint32(28, sampleRate * channels * 2, true); // ByteRate\n view.setUint16(32, channels * 2, true); // BlockAlign\n view.setUint16(34, 16, true); // BitsPerSample\n\n // data sub-chunk\n view.setUint32(36, 0x64617461, false); // \"data\"\n view.setUint32(40, length, true); // Subchunk2Size\n\n return new Uint8Array(buffer) as Uint8Array<ArrayBuffer>;\n};\nconst waitForPlaybackEnd = async (\n audioElement: HTMLAudioElement,\n): Promise<void> => {\n if (audioElement.ended) {\n return;\n }\n\n await new Promise<void>((resolve, reject) => {\n const watchdog = window.setInterval(() => {\n if (!audioElement.ended) {\n console.info(\"[Bulut] playback watchdog: still playing...\");\n }\n }, 30000);\n\n const onEnded = () => {\n cleanup();\n resolve();\n };\n\n const onError = () => {\n cleanup();\n reject(new Error(\"Ses oynatma hatası oluştu.\"));\n };\n\n const onForcedStop = () => {\n cleanup();\n resolve();\n };\n\n const cleanup = () => {\n window.clearInterval(watchdog);\n audioElement.removeEventListener(\"ended\", onEnded);\n audioElement.removeEventListener(\"error\", onError);\n audioElement.removeEventListener(BULUT_AUDIO_STOP_EVENT, onForcedStop);\n };\n\n audioElement.addEventListener(\"ended\", onEnded);\n audioElement.addEventListener(\"error\", onError);\n audioElement.addEventListener(BULUT_AUDIO_STOP_EVENT, onForcedStop);\n });\n};\n\n\n\nconst playBufferedAudio = async (\n chunks: Uint8Array<ArrayBuffer>[],\n mimeType: string,\n sampleRate: number = 16000,\n onAudioStateChange?: (state: AudioStreamState) => void,\n): Promise<void> => {\n const playbackGeneration = getAudioPlaybackGeneration();\n if (chunks.length === 0) {\n onAudioStateChange?.(\"done\");\n return;\n }\n\n if (wasPlaybackStoppedAfter(playbackGeneration)) {\n onAudioStateChange?.(\"done\");\n return;\n }\n\n // Debug info\n const totalBytes = chunks.reduce((acc, c) => acc + c.byteLength, 0);\n console.log(`[Bulut] Playing buffered audio: ${chunks.length} chunks, ${totalBytes} bytes, type=${mimeType}`);\n\n onAudioStateChange?.(\"fallback\");\n\n const blobParts: ArrayBuffer[] = chunks.map((chunk) => {\n const copied = new Uint8Array(chunk.byteLength) as Uint8Array<ArrayBuffer>;\n copied.set(chunk);\n return copied.buffer;\n });\n\n // Verify magic numbers and detect MIME type\n let detectedMime = mimeType;\n if (chunks.length > 0 && chunks[0].length >= 4) {\n const header = Array.from(chunks[0].slice(0, 4))\n .map(b => b.toString(16).padStart(2, '0').toUpperCase())\n .join(' ');\n console.log(`[Bulut] Audio header (hex): ${header}`);\n\n // Magic number detection\n if (header.startsWith(\"49 44 33\")) { // ID3\n detectedMime = \"audio/mpeg\";\n } else if (header.startsWith(\"FF F3\") || header.startsWith(\"FF F2\")) { // MP3 Sync\n detectedMime = \"audio/mpeg\";\n } else if (header.startsWith(\"52 49 46 46\")) { // RIFF (WAV)\n detectedMime = \"audio/wav\";\n } else if (header.startsWith(\"1A 45 DF A3\")) { // EBML (WebM)\n detectedMime = \"audio/webm\";\n }\n }\n\n // Ensure valid MIME type\n // Ensure valid MIME type or wrap raw PCM\n let safeMimeType = detectedMime && detectedMime.includes(\"/\") ? detectedMime : \"audio/mpeg\";\n let finalBlobParts: BlobPart[] = blobParts;\n\n if (mimeType === \"audio/pcm\") {\n // Wrap raw PCM in WAV container\n const totalLength = chunks.reduce((acc, c) => acc + c.byteLength, 0);\n const header = createWavHeader(totalLength, sampleRate);\n finalBlobParts = [header.buffer, ...blobParts];\n safeMimeType = \"audio/wav\";\n console.log(`[Bulut] Wrapped raw PCM in WAV (rate=${sampleRate})`);\n }\n\n console.log(`[Bulut] Creating blob with type: ${safeMimeType} (original: ${mimeType})`);\n const blob = new Blob(finalBlobParts, { type: safeMimeType });\n\n const audioElement = new Audio();\n const objectUrl = URL.createObjectURL(blob);\n\n try {\n registerActiveAudioElement(audioElement);\n\n audioElement.preload = \"auto\";\n audioElement.autoplay = true;\n // Some browsers need this\n audioElement.setAttribute(\"playsinline\", \"true\");\n audioElement.src = objectUrl;\n\n if (wasPlaybackStoppedAfter(playbackGeneration)) {\n onAudioStateChange?.(\"done\");\n return;\n }\n\n await audioElement.play();\n onAudioStateChange?.(\"playing\");\n await waitForPlaybackEnd(audioElement);\n onAudioStateChange?.(\"done\");\n } catch (err) {\n console.error(`[Bulut] Playback failed: ${err}`, { mimeType: safeMimeType, size: blob.size });\n onAudioStateChange?.(\"done\"); // Signal done to unblock UI even on error\n throw err;\n } finally {\n unregisterActiveAudioElement(audioElement);\n audioElement.pause();\n audioElement.removeAttribute(\"src\");\n audioElement.load();\n URL.revokeObjectURL(objectUrl);\n }\n};\n\nexport interface StreamController {\n stop: () => void;\n done: Promise<void>;\n}\n\nexport const parseSseEventPayload = (eventBlock: string): SseEventPayload | null => {\n const dataLines = eventBlock\n .split(/\\r?\\n/)\n .map((line) => line.trim())\n .filter((line) => line.startsWith(\"data:\"))\n .map((line) => line.slice(5).trimStart());\n\n if (dataLines.length === 0) {\n return null;\n }\n\n const dataStr = dataLines.join(\"\\n\");\n if (dataStr === \"[DONE]\") {\n return { type: \"done\" };\n }\n\n try {\n return JSON.parse(dataStr) as SseEventPayload;\n } catch (error) {\n console.warn(\"Error parsing SSE chunk:\", error);\n return null;\n }\n};\n\nexport const isAudioSsePayload = (\n payload: SseEventPayload,\n): payload is SseEventPayload & { audio: string } =>\n typeof payload.audio === \"string\" &&\n (payload.type === undefined || payload.type === \"audio\");\n\n// ── Separated Endpoint Helpers ──────────────────────────────────────\n\nexport async function transcribeAudio(\n baseUrl: string,\n file: File,\n projectId: string,\n sessionId: string | null,\n language: string,\n onRequestSent?: () => void,\n): Promise<{ text: string; session_id: string }> {\n const url = `${normalizeBaseUrl(baseUrl)}/chat/stt`;\n const formData = new FormData();\n formData.append(\"file\", file);\n formData.append(\"project_id\", projectId);\n if (sessionId) formData.append(\"session_id\", sessionId);\n formData.append(\"language\", language);\n\n const responsePromise = fetch(url, { method: \"POST\", body: formData });\n onRequestSent?.();\n const response = await responsePromise;\n if (!response.ok) {\n throw new Error(await parseErrorBody(response));\n }\n return response.json();\n}\n\nexport interface SttWsResult {\n text: string;\n session_id: string;\n}\n\nexport interface SttWsEvents {\n onRequestSent?: () => void;\n onSessionId?: (sessionId: string) => void;\n onPartial?: (payload: { seq: number; text: string }) => void;\n}\n\nexport interface SttWsController {\n pushChunk: (chunk: Blob) => Promise<void>;\n stop: () => Promise<SttWsResult>;\n cancel: () => void;\n}\n\nexport const startSttWebSocketStream = (\n baseUrl: string,\n config: {\n projectId: string;\n sessionId: string | null;\n language?: string;\n mimeType?: string;\n },\n events: SttWsEvents = {},\n): SttWsController => {\n const wsUrl = toWebSocketUrl(baseUrl, \"/chat/stt/ws\");\n console.info(\"[Bulut] STT WS connecting to\", wsUrl);\n const socket = new WebSocket(wsUrl);\n let seq = 0;\n let finalText = \"\";\n let finalSessionId = config.sessionId || \"\";\n let stopped = false;\n let settled = false;\n // All chunk sends and the final stop are chained through sendQueue\n // so the \"stop\" message always follows all enqueued chunks.\n let sendQueue: Promise<void> = Promise.resolve();\n\n let resolveStart: (() => void) | null = null;\n let rejectStart: ((error: Error & { retryable?: boolean }) => void) | null = null;\n const startPromise = new Promise<void>((resolve, reject) => {\n resolveStart = resolve;\n rejectStart = reject;\n });\n\n let resolveDone: ((result: SttWsResult) => void) | null = null;\n let rejectDone: ((error: Error & { retryable?: boolean }) => void) | null = null;\n const donePromise = new Promise<SttWsResult>((resolve, reject) => {\n resolveDone = resolve;\n rejectDone = reject;\n });\n\n const rejectAll = (error: Error & { retryable?: boolean }) => {\n if (settled) return;\n settled = true;\n console.warn(\"[Bulut] STT WS rejected:\", error.message);\n rejectStart?.(error);\n rejectDone?.(error);\n };\n\n const resolveDoneIfPossible = () => {\n if (settled) return;\n if (!finalText.trim() || !finalSessionId) return;\n settled = true;\n resolveStart?.();\n resolveDone?.({\n text: finalText.trim(),\n session_id: finalSessionId,\n });\n };\n\n socket.onopen = () => {\n console.info(\"[Bulut] STT WS connected, sending start\");\n events.onRequestSent?.();\n socket.send(\n JSON.stringify({\n type: \"start\",\n project_id: config.projectId,\n session_id: config.sessionId || undefined,\n language: config.language || \"tr\",\n mime_type: config.mimeType || \"audio/webm\",\n }),\n );\n };\n\n socket.onmessage = (event) => {\n const payload = parseSttWsEventPayload(String(event.data));\n if (!payload) return;\n\n if (payload.type === \"start_ack\" && typeof payload.session_id === \"string\") {\n console.info(\"[Bulut] STT WS start_ack received, session:\", payload.session_id);\n finalSessionId = payload.session_id;\n events.onSessionId?.(payload.session_id);\n resolveStart?.();\n return;\n }\n\n if (payload.type === \"partial\" && typeof payload.text === \"string\") {\n events.onPartial?.({\n seq: typeof payload.seq === \"number\" ? payload.seq : 0,\n text: payload.text,\n });\n return;\n }\n\n if (payload.type === \"final\" && typeof payload.text === \"string\") {\n console.info(\"[Bulut] STT WS final text received:\", payload.text.slice(0, 80));\n finalText = payload.text;\n if (typeof payload.session_id === \"string\") {\n finalSessionId = payload.session_id;\n events.onSessionId?.(payload.session_id);\n }\n return;\n }\n\n if (payload.type === \"done\") {\n console.info(\"[Bulut] STT WS done\");\n resolveDoneIfPossible();\n socket.close();\n return;\n }\n\n if (payload.type === \"error\") {\n console.error(\"[Bulut] STT WS server error:\", payload.error);\n const err = buildError(payload.error || \"stt_ws_error\", payload.retryable !== false);\n rejectAll(err);\n socket.close();\n }\n };\n\n socket.onerror = (ev) => {\n console.error(\"[Bulut] STT WS transport error\", ev);\n rejectAll(buildError(\"stt_ws_transport_error\", true));\n };\n\n socket.onclose = (ev) => {\n console.info(\"[Bulut] STT WS closed code=%d reason=%s\", ev.code, ev.reason);\n if (settled) return;\n if (finalText && finalSessionId) {\n resolveDoneIfPossible();\n return;\n }\n rejectAll(buildError(\"stt_ws_closed_before_done\", true));\n };\n\n return {\n pushChunk: (chunk: Blob): Promise<void> => {\n if (stopped || chunk.size === 0) return Promise.resolve();\n // Chain the entire operation (wait for connection, base64-encode,\n // send) into sendQueue so that a later stop() is guaranteed to\n // follow all previously-enqueued chunks.\n sendQueue = sendQueue.then(async () => {\n if (stopped) return;\n await startPromise;\n if (stopped) return;\n const audio = await blobToBase64(chunk);\n seq += 1;\n if (stopped || socket.readyState !== WebSocket.OPEN) return;\n console.debug(\"[Bulut] STT WS sending chunk seq=%d size=%d\", seq, chunk.size);\n socket.send(JSON.stringify({ type: \"chunk\", seq, audio }));\n });\n return sendQueue;\n },\n stop: (): Promise<SttWsResult> => {\n console.info(\"[Bulut] STT WS stop requested, draining %d pending chunks\", seq);\n // Chain after all pending pushChunk operations so the server\n // always receives every chunk before the stop message.\n sendQueue = sendQueue.then(async () => {\n await startPromise;\n if (stopped) return;\n if (socket.readyState === WebSocket.OPEN) {\n console.info(\"[Bulut] STT WS sending stop after seq=%d\", seq);\n socket.send(JSON.stringify({ type: \"stop\" }));\n }\n });\n return donePromise;\n },\n cancel: () => {\n stopped = true;\n try {\n socket.close();\n } catch {\n // no-op\n }\n },\n };\n};\n\ninterface TtsCollectResult {\n chunks: Uint8Array<ArrayBuffer>[];\n mimeType: string;\n sampleRate: number;\n}\n\nconst buildError = (message: string, retryable: boolean = true): Error & { retryable: boolean } => {\n const error = new Error(message) as Error & { retryable: boolean };\n error.retryable = retryable;\n return error;\n};\n\nconst collectTtsViaSse = async (\n baseUrl: string,\n assistantText: string,\n voice: string,\n accessibilityMode: boolean,\n isStopped: () => boolean,\n setReader: (reader: ReadableStreamDefaultReader<Uint8Array> | undefined) => void,\n): Promise<TtsCollectResult> => {\n const ttsFormData = new FormData();\n ttsFormData.append(\"text\", assistantText);\n ttsFormData.append(\"voice\", voice);\n ttsFormData.append(\"accessibility_mode\", String(accessibilityMode));\n\n const ttsResponse = await fetch(`${normalizeBaseUrl(baseUrl)}/chat/tts`, {\n method: \"POST\",\n body: ttsFormData,\n });\n\n if (!ttsResponse.ok) {\n throw buildError(await parseErrorBody(ttsResponse), false);\n }\n\n const reader = ttsResponse.body?.getReader();\n if (!reader) {\n throw buildError(\"TTS response body is not readable\", false);\n }\n\n setReader(reader);\n\n const chunks: Uint8Array<ArrayBuffer>[] = [];\n let mimeType = \"audio/mpeg\";\n let sampleRate = 16000;\n const decoder = new TextDecoder();\n let buffer = \"\";\n\n while (true) {\n if (isStopped()) {\n break;\n }\n\n const { done, value } = await reader.read();\n if (done) {\n break;\n }\n\n buffer += decoder.decode(value, { stream: true });\n const blocks = buffer.split(/\\r?\\n\\r?\\n/);\n buffer = blocks.pop() || \"\";\n\n for (const block of blocks) {\n const payload = parseSseEventPayload(block);\n if (!payload) {\n continue;\n }\n\n if (isAudioSsePayload(payload)) {\n const format = payload.format || \"mp3\";\n mimeType = payload.mime_type || (format === \"webm\" ? \"audio/webm\" : \"audio/mpeg\");\n chunks.push(base64ToUint8Array(payload.audio));\n if (payload.sample_rate) {\n sampleRate = payload.sample_rate;\n }\n }\n }\n }\n\n reader.releaseLock();\n setReader(undefined);\n\n return { chunks, mimeType, sampleRate };\n};\n\nconst collectTtsViaWebSocket = async (\n baseUrl: string,\n assistantText: string,\n voice: string,\n accessibilityMode: boolean,\n isStopped: () => boolean,\n setSocket: (socket: WebSocket | null) => void,\n): Promise<TtsCollectResult> => {\n const wsUrl = toWebSocketUrl(baseUrl, \"/chat/tts/ws\");\n const requestId = createRequestId();\n const chunks: Uint8Array<ArrayBuffer>[] = [];\n let mimeType = \"audio/mpeg\";\n let sampleRate = 16000;\n let highestSeqSeen = 0;\n\n const connectOnce = (): Promise<void> =>\n new Promise((resolve, reject) => {\n if (isStopped()) {\n reject(buildError(\"stream_stopped\", false));\n return;\n }\n\n let done = false;\n let finalError: (Error & { retryable?: boolean }) | null = null;\n const socket = new WebSocket(wsUrl);\n setSocket(socket);\n\n const finalize = (\n mode: \"resolve\" | \"reject\",\n error?: Error & { retryable?: boolean },\n ) => {\n socket.onopen = null;\n socket.onmessage = null;\n socket.onerror = null;\n socket.onclose = null;\n setSocket(null);\n if (mode === \"resolve\") {\n resolve();\n return;\n }\n reject(error || buildError(\"tts_ws_closed\", true));\n };\n\n socket.onopen = () => {\n console.info(\n `[Bulut] TTS WS connected request_id=${requestId} resume_seq=${highestSeqSeen}`,\n );\n socket.send(\n JSON.stringify({\n type: \"start\",\n request_id: requestId,\n text: assistantText,\n voice,\n accessibility_mode: accessibilityMode,\n last_seq: highestSeqSeen,\n }),\n );\n };\n\n socket.onmessage = (event) => {\n const payload = parseTtsWsEventPayload(String(event.data));\n if (!payload) {\n console.warn(\"[Bulut] TTS WS invalid JSON payload\");\n return;\n }\n\n if (payload.type === \"audio\" && typeof payload.audio === \"string\") {\n const seq = typeof payload.seq === \"number\" ? payload.seq : 0;\n if (shouldAcceptAudioSeq(seq, highestSeqSeen)) {\n chunks.push(base64ToUint8Array(payload.audio));\n highestSeqSeen = seq;\n if (payload.mime_type) {\n mimeType = payload.mime_type;\n }\n if (typeof payload.sample_rate === \"number\") {\n sampleRate = payload.sample_rate;\n }\n } else {\n console.info(\n `[Bulut] TTS WS duplicate chunk ignored request_id=${requestId} seq=${seq} seen=${highestSeqSeen}`,\n );\n }\n\n if (socket.readyState === WebSocket.OPEN) {\n socket.send(\n JSON.stringify({\n type: \"ack\",\n request_id: requestId,\n last_seq: highestSeqSeen,\n }),\n );\n }\n return;\n }\n\n if (payload.type === \"done\") {\n const streamLastSeq =\n typeof payload.last_seq === \"number\" ? payload.last_seq : highestSeqSeen;\n if (streamLastSeq > highestSeqSeen) {\n finalError = buildError(\"tts_ws_sequence_gap\", true);\n done = false;\n socket.close();\n return;\n }\n done = true;\n socket.close();\n return;\n }\n\n if (payload.type === \"error\") {\n finalError = buildError(payload.error || \"tts_ws_error\", payload.retryable !== false);\n done = false;\n socket.close();\n }\n };\n\n socket.onerror = () => {\n if (!finalError) {\n finalError = buildError(\"tts_ws_transport_error\", true);\n }\n };\n\n socket.onclose = () => {\n if (isStopped()) {\n finalize(\"reject\", buildError(\"stream_stopped\", false));\n return;\n }\n if (done) {\n finalize(\"resolve\");\n return;\n }\n finalize(\"reject\", finalError || buildError(\"tts_ws_closed_before_done\", true));\n };\n });\n\n for (let attempt = 0; attempt <= TTS_WS_RETRY_DELAYS_MS.length; attempt += 1) {\n if (attempt > 0) {\n const delay = TTS_WS_RETRY_DELAYS_MS[attempt - 1];\n console.warn(\n `[Bulut] TTS WS retry attempt=${attempt} delay_ms=${delay} last_seq=${highestSeqSeen}`,\n );\n await sleep(delay);\n }\n\n try {\n await connectOnce();\n return { chunks, mimeType, sampleRate };\n } catch (error) {\n const retryable =\n shouldFallbackToSse(error);\n const message = error instanceof Error ? error.message : String(error);\n console.warn(\n `[Bulut] TTS WS attempt failed attempt=${attempt} retryable=${retryable} error=${message}`,\n );\n if (!retryable || attempt === TTS_WS_RETRY_DELAYS_MS.length) {\n throw error;\n }\n }\n }\n\n throw buildError(\"tts_ws_exhausted\", true);\n};\n\n// ── Agent-mode Types ────────────────────────────────────────────────\n\nexport interface AgentToolCallInfo {\n call_id: string;\n tool: string;\n args: Record<string, unknown>;\n}\n\nexport interface AgentVoiceChatEvents {\n onSttRequestSent?: () => void;\n onTranscription?: (data: {\n session_id: string;\n user_text: string;\n }) => void;\n onAssistantDelta?: (delta: string) => void;\n onAssistantDone?: (assistantText: string, expectsReply?: boolean) => void;\n onAudioStateChange?: (state: AudioStreamState) => void;\n onError?: (error: string) => void;\n /** Called when the agent requests tool execution on the frontend. */\n onToolCalls?: (calls: AgentToolCallInfo[]) => void;\n /** Called after each tool has been executed with the result. */\n onToolResult?: (callId: string, toolName: string, result: string) => void;\n /** Called at the start of each agent iteration. */\n onIteration?: (iteration: number, maxIterations: number) => void;\n /** Called when the backend confirms / creates a session ID. */\n onSessionId?: (sessionId: string) => void;\n /**\n * Called when the agent emits a reply text followed by tool calls.\n * This text is spoken aloud before the tools run so the user hears\n * every piece of the conversation, not just the final reply.\n */\n onIntermediateReply?: (text: string) => void;\n}\n\n/**\n * Standalone TTS helper: synthesize + play a text snippet.\n * Uses WebSocket TTS with SSE fallback, same as the main stream functions.\n */\nexport const speakText = async (\n baseUrl: string,\n text: string,\n voice: string,\n accessibilityMode: boolean,\n onAudioStateChange?: (state: AudioStreamState) => void,\n): Promise<void> => {\n const trimmed = text.trim();\n if (!trimmed) return;\n const playbackGeneration = getAudioPlaybackGeneration();\n\n console.info(`[Bulut] speakText start (${trimmed.length} chars)`);\n onAudioStateChange?.(\"rendering\");\n let ttsResult: TtsCollectResult;\n\n const neverStopped = () => false;\n\n try {\n ttsResult = await collectTtsViaWebSocket(\n baseUrl, trimmed, voice, accessibilityMode,\n neverStopped,\n () => {},\n );\n } catch {\n ttsResult = await collectTtsViaSse(\n baseUrl, trimmed, voice, accessibilityMode,\n neverStopped,\n () => {},\n );\n }\n\n if (wasPlaybackStoppedAfter(playbackGeneration)) {\n onAudioStateChange?.(\"done\");\n return;\n }\n\n if (ttsResult.chunks.length > 0) {\n await playBufferedAudio(\n ttsResult.chunks, ttsResult.mimeType, ttsResult.sampleRate,\n onAudioStateChange,\n );\n } else {\n onAudioStateChange?.(\"done\");\n }\n};\n\n// ── Agent Voice Chat Stream (STT → Agent WS → TTS) ─────────────────\n\nexport const agentVoiceChatStream = (\n baseUrl: string,\n audioFile: File,\n projectId: string,\n sessionId: string | null,\n config: {\n model: string;\n voice: string;\n pageContext?: string;\n accessibilityMode?: boolean;\n },\n events: AgentVoiceChatEvents,\n executeTool: (call: AgentToolCallInfo) => Promise<{ call_id: string; result: string }>,\n): StreamController => {\n let isStopped = false;\n let activeSocket: WebSocket | null = null;\n let activeReader: ReadableStreamDefaultReader<Uint8Array> | undefined;\n let errorEmitted = false;\n\n const donePromise = new Promise<void>(async (resolve, reject) => {\n try {\n // ── 1. STT ────────────────────────────────────────────────\n if (isStopped) return resolve();\n const sttResult = await transcribeAudio(\n baseUrl,\n audioFile,\n projectId,\n sessionId,\n \"tr\",\n events.onSttRequestSent,\n );\n\n const currentSessionId = sttResult.session_id;\n let effectiveSessionId = currentSessionId;\n const userText = sttResult.text;\n\n events.onTranscription?.({\n session_id: currentSessionId,\n user_text: userText,\n });\n\n if (isStopped) return resolve();\n\n // ── 2. Agent loop via WebSocket ───────────────────────────\n const assistantText = await new Promise<string>((agentResolve, agentReject) => {\n if (isStopped) { agentResolve(\"\"); return; }\n\n const wsUrl = toWebSocketUrl(baseUrl, \"/chat/agent/ws\");\n const socket = new WebSocket(wsUrl);\n activeSocket = socket;\n\n let finalReply = \"\";\n let resolved = false;\n let accumulatedDelta = \"\";\n\n const finish = (reply: string) => {\n if (resolved) return;\n resolved = true;\n agentResolve(reply);\n };\n\n const fail = (error: Error) => {\n if (resolved) return;\n resolved = true;\n agentReject(error);\n };\n\n socket.onopen = () => {\n console.info(\"[Bulut] Agent WS connected\");\n socket.send(JSON.stringify({\n type: \"start\",\n project_id: projectId,\n session_id: currentSessionId,\n user_text: userText,\n model: config.model,\n page_context: config.pageContext,\n accessibility_mode: config.accessibilityMode,\n }));\n };\n\n socket.onmessage = async (event) => {\n let data: Record<string, unknown>;\n try {\n data = JSON.parse(String(event.data));\n } catch {\n console.warn(\"[Bulut] Agent WS invalid JSON\");\n return;\n }\n\n const msgType = data.type as string;\n\n if (msgType === \"session\" && typeof data.session_id === \"string\") {\n effectiveSessionId = data.session_id as string;\n events.onSessionId?.(effectiveSessionId);\n return;\n }\n\n if (msgType === \"iteration\") {\n events.onIteration?.(\n data.iteration as number,\n data.max_iterations as number,\n );\n return;\n }\n\n if (msgType === \"reply_delta\" && typeof data.delta === \"string\") {\n accumulatedDelta += data.delta;\n events.onAssistantDelta?.(data.delta);\n return;\n }\n\n if (msgType === \"tool_calls\" && Array.isArray(data.calls)) {\n const calls = data.calls as AgentToolCallInfo[];\n\n // Speak accumulated text before running tools\n if (accumulatedDelta.trim()) {\n events.onIntermediateReply?.(accumulatedDelta.trim());\n }\n accumulatedDelta = \"\";\n\n events.onToolCalls?.(calls);\n\n const results: { call_id: string; result: string }[] = [];\n for (const call of calls) {\n // Save resume state before navigate in case of full-page reload\n const isNavigate = call.tool === \"navigate\";\n if (isNavigate) {\n savePendingAgentResume({\n sessionId: effectiveSessionId,\n projectId,\n model: config.model,\n voice: config.voice,\n accessibilityMode: Boolean(config.accessibilityMode),\n pendingToolCalls: calls.map((c) => ({\n call_id: c.call_id,\n tool: c.tool,\n args: c.args,\n })),\n completedResults: [...results],\n });\n }\n\n const result = await executeTool(call);\n\n // If we reach here, no full-page reload happened\n if (isNavigate) {\n clearPendingAgentResume();\n }\n\n events.onToolResult?.(call.call_id, call.tool, result.result);\n results.push(result);\n }\n\n if (socket.readyState === WebSocket.OPEN) {\n socket.send(JSON.stringify({\n type: \"tool_results\",\n results,\n }));\n }\n return;\n }\n\n if (msgType === \"agent_done\") {\n finalReply = (data.final_reply as string) || \"\";\n const replyExpectsReply = data.expects_reply !== false;\n events.onAssistantDone?.(finalReply, replyExpectsReply);\n if (typeof data.session_id === \"string\") {\n events.onSessionId?.(data.session_id as string);\n }\n finish(finalReply);\n return;\n }\n\n if (msgType === \"error\") {\n const errMsg = (data.error as string) || \"Agent error\";\n errorEmitted = true;\n events.onError?.(errMsg);\n fail(new Error(errMsg));\n return;\n }\n };\n\n socket.onerror = () => {\n console.error(\"[Bulut] Agent WS error\");\n errorEmitted = true;\n events.onError?.(\"Agent WebSocket connection error\");\n fail(new Error(\"Agent WebSocket connection error\"));\n };\n\n socket.onclose = () => {\n console.info(\"[Bulut] Agent WS closed\");\n finish(finalReply);\n };\n });\n\n activeSocket = null;\n\n // ── 3. TTS ────────────────────────────────────────────────\n if (isStopped || !assistantText) {\n if (!isStopped) {\n hideAgentCursor();\n }\n return resolve();\n }\n\n console.info(\n `[Bulut] TTS start mode=agent voice=${config.voice}`,\n );\n\n events.onAudioStateChange?.(\"rendering\");\n let ttsResult: TtsCollectResult;\n\n try {\n ttsResult = await collectTtsViaWebSocket(\n baseUrl,\n assistantText,\n config.voice,\n Boolean(config.accessibilityMode),\n () => isStopped,\n (socket) => { activeSocket = socket; },\n );\n } catch (wsError) {\n if (isStopped) return resolve();\n console.warn(\n `[Bulut] TTS WS failed, falling back to SSE: ${wsError instanceof Error ? wsError.message : String(wsError)}`,\n );\n ttsResult = await collectTtsViaSse(\n baseUrl,\n assistantText,\n config.voice,\n Boolean(config.accessibilityMode),\n () => isStopped,\n (reader) => { activeReader = reader; },\n );\n }\n\n if (!isStopped) {\n hideAgentCursor();\n }\n\n if (!isStopped && ttsResult.chunks.length > 0) {\n await playBufferedAudio(\n ttsResult.chunks,\n ttsResult.mimeType,\n ttsResult.sampleRate,\n events.onAudioStateChange,\n );\n } else {\n events.onAudioStateChange?.(\"done\");\n }\n\n resolve();\n } catch (err) {\n // Only emit onError if it hasn't been emitted already by the WS handler\n if (!errorEmitted) {\n const msg = err instanceof Error ? err.message : String(err);\n events.onError?.(msg);\n }\n reject(err);\n } finally {\n activeReader?.cancel().catch(() => { });\n if (activeSocket && activeSocket.readyState <= WebSocket.OPEN) {\n activeSocket.close();\n }\n activeSocket = null;\n }\n });\n\n return {\n stop: () => {\n isStopped = true;\n stopActiveAudioPlayback();\n if (activeReader) {\n activeReader.cancel().catch(() => { });\n }\n if (activeSocket && activeSocket.readyState <= WebSocket.OPEN) {\n activeSocket.close();\n }\n },\n done: donePromise,\n };\n};\n\n// ── Agent Text Chat Stream (no STT, Agent WS → TTS) ────────────────\n\nexport const agentTextChatStream = (\n baseUrl: string,\n userText: string,\n projectId: string,\n sessionId: string | null,\n config: {\n model: string;\n voice: string;\n pageContext?: string;\n accessibilityMode?: boolean;\n },\n events: AgentVoiceChatEvents,\n executeTool: (call: AgentToolCallInfo) => Promise<{ call_id: string; result: string }>,\n): StreamController => {\n let isStopped = false;\n let activeSocket: WebSocket | null = null;\n let activeReader: ReadableStreamDefaultReader<Uint8Array> | undefined;\n let errorEmitted = false;\n\n const donePromise = new Promise<void>(async (resolve, reject) => {\n try {\n if (isStopped) return resolve();\n\n // ── 1. Agent loop via WebSocket ───────────────────────────\n const assistantText = await new Promise<string>((agentResolve, agentReject) => {\n if (isStopped) { agentResolve(\"\"); return; }\n\n const wsUrl = toWebSocketUrl(baseUrl, \"/chat/agent/ws\");\n const socket = new WebSocket(wsUrl);\n activeSocket = socket;\n\n let finalReply = \"\";\n let resolved = false;\n let effectiveSessionId = sessionId || \"\";\n let accumulatedDelta = \"\";\n\n const finish = (reply: string) => {\n if (resolved) return;\n resolved = true;\n agentResolve(reply);\n };\n\n const fail = (error: Error) => {\n if (resolved) return;\n resolved = true;\n agentReject(error);\n };\n\n socket.onopen = () => {\n socket.send(JSON.stringify({\n type: \"start\",\n project_id: projectId,\n session_id: sessionId,\n user_text: userText,\n model: config.model,\n page_context: config.pageContext,\n accessibility_mode: config.accessibilityMode,\n }));\n };\n\n socket.onmessage = async (event) => {\n let data: Record<string, unknown>;\n try {\n data = JSON.parse(String(event.data));\n } catch { return; }\n\n const msgType = data.type as string;\n\n if (msgType === \"session\" && typeof data.session_id === \"string\") {\n effectiveSessionId = data.session_id as string;\n events.onSessionId?.(effectiveSessionId);\n return;\n }\n\n if (msgType === \"iteration\") {\n events.onIteration?.(\n data.iteration as number,\n data.max_iterations as number,\n );\n return;\n }\n\n if (msgType === \"reply_delta\" && typeof data.delta === \"string\") {\n accumulatedDelta += data.delta;\n events.onAssistantDelta?.(data.delta);\n return;\n }\n\n if (msgType === \"tool_calls\" && Array.isArray(data.calls)) {\n const calls = data.calls as AgentToolCallInfo[];\n\n // Speak accumulated text before running tools\n if (accumulatedDelta.trim()) {\n events.onIntermediateReply?.(accumulatedDelta.trim());\n }\n accumulatedDelta = \"\";\n\n events.onToolCalls?.(calls);\n\n const results: { call_id: string; result: string }[] = [];\n for (const call of calls) {\n const isNavigate = call.tool === \"navigate\";\n if (isNavigate) {\n savePendingAgentResume({\n sessionId: effectiveSessionId,\n projectId,\n model: config.model,\n voice: config.voice,\n accessibilityMode: Boolean(config.accessibilityMode),\n pendingToolCalls: calls.map((c) => ({\n call_id: c.call_id,\n tool: c.tool,\n args: c.args,\n })),\n completedResults: [...results],\n });\n }\n\n const result = await executeTool(call);\n\n if (isNavigate) {\n clearPendingAgentResume();\n }\n\n events.onToolResult?.(call.call_id, call.tool, result.result);\n results.push(result);\n }\n\n if (socket.readyState === WebSocket.OPEN) {\n socket.send(JSON.stringify({\n type: \"tool_results\",\n results,\n }));\n }\n return;\n }\n\n if (msgType === \"agent_done\") {\n finalReply = (data.final_reply as string) || \"\";\n const replyExpectsReply = data.expects_reply !== false;\n events.onAssistantDone?.(finalReply, replyExpectsReply);\n if (typeof data.session_id === \"string\") {\n events.onSessionId?.(data.session_id as string);\n }\n finish(finalReply);\n return;\n }\n\n if (msgType === \"error\") {\n const errMsg = (data.error as string) || \"Agent error\";\n errorEmitted = true;\n events.onError?.(errMsg);\n fail(new Error(errMsg));\n return;\n }\n };\n\n socket.onerror = () => {\n errorEmitted = true;\n events.onError?.(\"Agent WebSocket error\");\n fail(new Error(\"Agent WebSocket error\"));\n };\n socket.onclose = () => finish(finalReply);\n });\n\n activeSocket = null;\n\n // ── 2. TTS ────────────────────────────────────────────────\n if (isStopped || !assistantText) {\n if (!isStopped) {\n hideAgentCursor();\n }\n return resolve();\n }\n\n events.onAudioStateChange?.(\"rendering\");\n let ttsResult: TtsCollectResult;\n\n try {\n ttsResult = await collectTtsViaWebSocket(\n baseUrl, assistantText, config.voice, Boolean(config.accessibilityMode),\n () => isStopped,\n (socket) => { activeSocket = socket; },\n );\n } catch (wsError) {\n if (isStopped) return resolve();\n ttsResult = await collectTtsViaSse(\n baseUrl, assistantText, config.voice, Boolean(config.accessibilityMode),\n () => isStopped,\n (reader) => { activeReader = reader; },\n );\n }\n\n if (!isStopped) {\n hideAgentCursor();\n }\n\n if (!isStopped && ttsResult.chunks.length > 0) {\n await playBufferedAudio(\n ttsResult.chunks, ttsResult.mimeType, ttsResult.sampleRate,\n events.onAudioStateChange,\n );\n } else {\n events.onAudioStateChange?.(\"done\");\n }\n\n resolve();\n } catch (err) {\n if (!errorEmitted) {\n const msg = err instanceof Error ? err.message : String(err);\n events.onError?.(msg);\n }\n reject(err);\n } finally {\n activeReader?.cancel().catch(() => { });\n if (activeSocket && activeSocket.readyState <= WebSocket.OPEN) {\n activeSocket.close();\n }\n activeSocket = null;\n }\n });\n\n return {\n stop: () => {\n isStopped = true;\n stopActiveAudioPlayback();\n if (activeReader) activeReader.cancel().catch(() => { });\n if (activeSocket && activeSocket.readyState <= WebSocket.OPEN) {\n activeSocket.close();\n }\n },\n done: donePromise,\n };\n};\n\n// ── Agent Resume Stream (after page navigation reload) ──────────────\n//\n// When a navigate tool causes a full-page reload, the agent WS is lost.\n// This function opens a new WS with {type: \"resume\"}, sends the\n// completed tool results (including the navigate result with the new\n// page context), and continues the agent loop from where it left off.\n\nexport const agentResumeStream = (\n baseUrl: string,\n resumeState: PendingAgentResume,\n pageContext: string,\n events: AgentVoiceChatEvents,\n executeTool: (call: AgentToolCallInfo) => Promise<{ call_id: string; result: string }>,\n): StreamController => {\n let isStopped = false;\n let activeSocket: WebSocket | null = null;\n let activeReader: ReadableStreamDefaultReader<Uint8Array> | undefined;\n let errorEmitted = false;\n\n // Build tool results for the calls that were pending when the page reloaded.\n // Navigate results include the new page context; other tools that couldn't\n // execute get a descriptive skip message.\n const allResults = [...resumeState.completedResults];\n for (const tc of resumeState.pendingToolCalls) {\n if (allResults.some((r) => r.call_id === tc.call_id)) continue;\n if (tc.tool === \"navigate\") {\n allResults.push({\n call_id: tc.call_id,\n result: `Navigasyon tamamlandı. Şu anki sayfa: ${typeof window !== \"undefined\" ? window.location.href : \"\"}\\nSayfa bağlamı: ${pageContext}`,\n });\n } else {\n allResults.push({\n call_id: tc.call_id,\n result: \"Sayfa yeniden yüklendi, bu araç çalıştırılamadı.\",\n });\n }\n }\n\n const donePromise = new Promise<void>(async (resolve, reject) => {\n try {\n if (isStopped) return resolve();\n\n let effectiveSessionId = resumeState.sessionId;\n\n const assistantText = await new Promise<string>((agentResolve, agentReject) => {\n if (isStopped) { agentResolve(\"\"); return; }\n\n const wsUrl = toWebSocketUrl(baseUrl, \"/chat/agent/ws\");\n const socket = new WebSocket(wsUrl);\n activeSocket = socket;\n\n let finalReply = \"\";\n let resolved = false;\n let accumulatedDelta = \"\";\n\n const finish = (reply: string) => {\n if (resolved) return;\n resolved = true;\n agentResolve(reply);\n };\n\n const fail = (error: Error) => {\n if (resolved) return;\n resolved = true;\n agentReject(error);\n };\n\n socket.onopen = () => {\n console.info(\"[Bulut] Agent WS resume connected\");\n socket.send(JSON.stringify({\n type: \"resume\",\n project_id: resumeState.projectId,\n session_id: resumeState.sessionId,\n model: resumeState.model,\n page_context: pageContext,\n accessibility_mode: resumeState.accessibilityMode,\n pending_tool_calls: resumeState.pendingToolCalls,\n tool_results: allResults,\n }));\n };\n\n socket.onmessage = async (event) => {\n let data: Record<string, unknown>;\n try {\n data = JSON.parse(String(event.data));\n } catch { return; }\n\n const msgType = data.type as string;\n\n if (msgType === \"session\" && typeof data.session_id === \"string\") {\n effectiveSessionId = data.session_id as string;\n events.onSessionId?.(effectiveSessionId);\n return;\n }\n\n if (msgType === \"iteration\") {\n events.onIteration?.(\n data.iteration as number,\n data.max_iterations as number,\n );\n return;\n }\n\n if (msgType === \"reply_delta\" && typeof data.delta === \"string\") {\n accumulatedDelta += data.delta;\n events.onAssistantDelta?.(data.delta);\n return;\n }\n\n if (msgType === \"tool_calls\" && Array.isArray(data.calls)) {\n const calls = data.calls as AgentToolCallInfo[];\n\n // Speak accumulated text before running tools\n if (accumulatedDelta.trim()) {\n events.onIntermediateReply?.(accumulatedDelta.trim());\n }\n accumulatedDelta = \"\";\n\n events.onToolCalls?.(calls);\n\n const results: { call_id: string; result: string }[] = [];\n for (const call of calls) {\n const isNavigate = call.tool === \"navigate\";\n if (isNavigate) {\n savePendingAgentResume({\n sessionId: effectiveSessionId,\n projectId: resumeState.projectId,\n model: resumeState.model,\n voice: resumeState.voice,\n accessibilityMode: resumeState.accessibilityMode,\n pendingToolCalls: calls.map((c) => ({\n call_id: c.call_id,\n tool: c.tool,\n args: c.args,\n })),\n completedResults: [...results],\n });\n }\n\n const result = await executeTool(call);\n\n if (isNavigate) {\n clearPendingAgentResume();\n }\n\n events.onToolResult?.(call.call_id, call.tool, result.result);\n results.push(result);\n }\n\n if (socket.readyState === WebSocket.OPEN) {\n socket.send(JSON.stringify({ type: \"tool_results\", results }));\n }\n return;\n }\n\n if (msgType === \"agent_done\") {\n finalReply = (data.final_reply as string) || \"\";\n const replyExpectsReply = data.expects_reply !== false;\n events.onAssistantDone?.(finalReply, replyExpectsReply);\n if (typeof data.session_id === \"string\") {\n events.onSessionId?.(data.session_id as string);\n }\n finish(finalReply);\n return;\n }\n\n if (msgType === \"error\") {\n const errMsg = (data.error as string) || \"Agent error\";\n errorEmitted = true;\n events.onError?.(errMsg);\n fail(new Error(errMsg));\n return;\n }\n };\n\n socket.onerror = () => {\n errorEmitted = true;\n events.onError?.(\"Agent WebSocket error\");\n fail(new Error(\"Agent WebSocket error\"));\n };\n\n socket.onclose = () => finish(finalReply);\n });\n\n activeSocket = null;\n\n // ── TTS ────────────────────────────────────────────────\n if (isStopped || !assistantText) {\n if (!isStopped) {\n hideAgentCursor();\n }\n return resolve();\n }\n\n console.info(`[Bulut] TTS start mode=resume voice=${resumeState.voice}`);\n events.onAudioStateChange?.(\"rendering\");\n let ttsResult: TtsCollectResult;\n\n try {\n ttsResult = await collectTtsViaWebSocket(\n baseUrl, assistantText, resumeState.voice, Boolean(resumeState.accessibilityMode),\n () => isStopped,\n (socket) => { activeSocket = socket; },\n );\n } catch (wsError) {\n if (isStopped) return resolve();\n console.warn(\n `[Bulut] TTS WS failed, falling back to SSE: ${wsError instanceof Error ? wsError.message : String(wsError)}`,\n );\n ttsResult = await collectTtsViaSse(\n baseUrl, assistantText, resumeState.voice, Boolean(resumeState.accessibilityMode),\n () => isStopped,\n (reader) => { activeReader = reader; },\n );\n }\n\n if (!isStopped) {\n hideAgentCursor();\n }\n\n if (!isStopped && ttsResult.chunks.length > 0) {\n await playBufferedAudio(\n ttsResult.chunks, ttsResult.mimeType, ttsResult.sampleRate,\n events.onAudioStateChange,\n );\n } else {\n events.onAudioStateChange?.(\"done\");\n }\n\n resolve();\n } catch (err) {\n if (!errorEmitted) {\n const msg = err instanceof Error ? err.message : String(err);\n events.onError?.(msg);\n }\n reject(err);\n } finally {\n activeReader?.cancel().catch(() => { });\n if (activeSocket && activeSocket.readyState <= WebSocket.OPEN) {\n activeSocket.close();\n }\n activeSocket = null;\n }\n });\n\n return {\n stop: () => {\n isStopped = true;\n stopActiveAudioPlayback();\n if (activeReader) activeReader.cancel().catch(() => { });\n if (activeSocket && activeSocket.readyState <= WebSocket.OPEN) {\n activeSocket.close();\n }\n },\n done: donePromise,\n };\n};\n","import {\n completedSfxUrl,\n sentSfxUrl,\n thinkingSfxUrl,\n toolCallSfxUrl,\n} from \"../assets\";\n\nexport type SfxName = \"sent\" | \"thinking\" | \"toolCall\" | \"completed\";\n\nconst SFX_SOURCES: Record<SfxName, string> = {\n sent: sentSfxUrl,\n thinking: thinkingSfxUrl,\n toolCall: toolCallSfxUrl,\n completed: completedSfxUrl,\n};\n\nconst SFX_VOLUME = 0.5;\n\nclass SfxManager {\n private queue: SfxName[] = [];\n private active = false;\n\n private playNow(name: SfxName): Promise<void> {\n return new Promise((resolve) => {\n if (typeof window === \"undefined\") {\n resolve();\n return;\n }\n\n const audio = new Audio(SFX_SOURCES[name]);\n audio.preload = \"auto\";\n audio.volume = SFX_VOLUME;\n\n let settled = false;\n const finalize = () => {\n if (settled) return;\n settled = true;\n audio.onended = null;\n audio.onerror = null;\n resolve();\n };\n\n audio.onended = finalize;\n audio.onerror = finalize;\n void audio.play().catch(() => finalize());\n });\n }\n\n private async drain(): Promise<void> {\n if (this.active) return;\n this.active = true;\n try {\n while (this.queue.length > 0) {\n const next = this.queue.shift();\n if (!next) continue;\n await this.playNow(next);\n }\n } finally {\n this.active = false;\n }\n }\n\n playCue(name: SfxName): void {\n if (typeof window === \"undefined\") return;\n this.queue.push(name);\n if (!this.active) {\n void this.drain();\n }\n }\n}\n\nconst sfxManager = new SfxManager();\n\nexport const playCue = (name: SfxName): void => {\n sfxManager.playCue(name);\n};\n","export default \"data:audio/mpeg;base64,SUQzAwAAAAAAMFRZRVIAAAAFAAAAMjAyNlRFTkMAAAAXAAAATEFNRSBpbiBGTCBTdHVkaW8gMjAyNf/6sGz82QAAA/1kxNUZAAIvYBVCoAgAJJnrUTnNAADNAGA3ACAAgAAIBAkyezCCEg4DAek02jngMLMAXCMYCgFhhYueWPRJLFz4FxdiBRK3FKcXF7hOESpFxc+Abi8EB4KVLmJLvcJKJLvwmnL6CiSKGU7u+jvoif/CVLvu7voKGaJUu7+73ohiSKGU7u/DvwifvaEGJzf8u72guL3BAoKJIoZQeHglV////8+X/UcKanBj8TvD/4gccyBR31h8ufBDxAGCjg+UcCEHz/E58H1HPBA4GCBABAAAAAABCXoWBMEh00O5REMTGyHDAGX/C5mNHnI70GQYGi7cLN0MQVA5qJJiAMKFsEOu+MYvNgHZq/8bdsLCgg0NHRADmhGWgdWN0H4JkwyERiL8sGdLKXPJT4NotMIcqxOysERVENJ95FlBhJe6SbsUUOcBS4VRLxU4QNaELCuRZpVqCfoqfWUMRRChIluzlyhkSOOU1L/1BQVDNMicXfup+WHoOrVS6QnPTMKiaY4kntS+tlHc5hh1PWlksu25+pYlldfqISgsCsFa816Hmurlhz6CNfjKd8pvy7b/mVj/z/PXLHwVAE1D6izcX8dWVSORv9DMrgWGv1TY/utlV3zLHHu8/7h/P/ef//////////6ltSXyqvRVblizcv/1Tqp6kQCkQikMiEQCEQhkQgPYfpTSWoDDW0VlOXpT37/u1qR5/3vd9GnCbGT7/7s/QhDRYULpQAE5wgiP8CBAvuYGCJEMC+hhkvGJsuZQJZiwJmBQsYVCxjEWGK6MZbBQkDjAIAQ2tgLhAFMZBSRJsckW0d4b6KWLpwWc//qybDhJIoD2c2fSV3KAAiUgB2DgAAAjXa0lT3HtwICAXIAAAAABIwIiwbkiopKHKFiNVoojKgLERfibRYTVSQ/kFIKXkkyoDbYLChkjasV4QsNIvJPJkvGz0TyjZLzJJ96OohwuYNjEpDiNkjIcoZonnnjInkklTImS6iiiii3ZReJoni8ZI1JfSMi8fRRLpFSKn96KKKKTvb0i8RJJKmYj+Vtect365xHmyRPGoV0fqvsTuf/p9lCuhPf/+5lH0/o9qK7rUb3f6PahK7j6UF1XNIFwACCAAbu7MQMHoCAkmBYAsYowTpVBCMD0OE4+dqjFvEPMPYFgwJAKDA9ApMXADgx35hjJMEgMAUCwOFhhoWmMggQHsxZLAQKSYWMuUqMBAEUOJWSg46qW3C9oVEBm8QCMFr9psck6bPNzEPggMgInCIDrVw5DTq2YzQStsDQ0kwAC16SeifVHlN4SAS5bGMc7xl2+lDuQCeY2XNCSspoyRsoQxn60ZrU9gcwln6YcxO4p/HE9nO4PKuUqpiRg3nsu4zMPiC3TwU7ZPObirobgdyXf1Zd2DfjXb1MZQRtUrtSuTKTBKK6dXTsBOncNSMsIwkC9b09uzX81ZLg+VrGJbSlx1953n0rWduK7eJXB6ChVOt/0cvWeyG3ADdX//yq+zr//uO0Fldv/6Ec8zX//8srt6tvvoSu881K1BAACC/zt0lZh4AIQBBgGExhWB5gOEZhkBh8tvIsXIYHMrQQAoczHtfDagCjCsDQcDAYBa7WmGKgXs1pM5qRhRE4TQF+tGQs8C7YhgQSsWwq4goa1bv1Y6kIvaYo+fptq1r91//qybLSQMYD2NmfNa7lcegAADSAAAAElVecOT+3xCAAANIAAAASXzXcWrzuUetS7H/ZZR77jtRqE5ZdrwEvf9ZY6d61e7hnvPbfsEhzKjEYOW/QKRsl9eXLfEsU5voR3fcIkPdukrKFYQqy85Ex0q6WPogzj7b5d+mwV1PN0R77/viNVSfO2Qp+YMAAMNZA5TBNQKIwBACpMUdBGjA1Av8wy0ExO7FdXjD8QLkxAICWMDEAojASwJ0wy4AdMTHGPzT7AREwEEFQMC2AkzALQFQwBEAaMBFBDjEAwCgwIUARMBCABVNmmmBhBlkEBupMxVWVRYQBppwWl4mdL6CqHEtWN2JbcGSBNkRgReJnssKoG16Qw9IZ2wPx2usTwsMupm2NFDTvfGo4yiaOLK+isZJq01pzDvg6hL+NQqx6wKk7ltuFCSRSSwcZ2PG9X0TRVyQNbbHkTwY0nREe3ngvzXj3q9qaObSxm4/6S+JLEmb857cnIWbxnzG7XV9pLZMzxnxEdvCwPJPBpS8Wrcyuhmw46kY3gJJqkbdYW4vy/tCtbL99h7EiySsd4sHOXNYeQGqLWLKpNAAMDaChDAfAH8wGsJtMFmAVjA0AGQwZYarPUWXGTFbAUMwsgBEMDGAczAjQTAwCwOrMEQUFzRqA20wXkAsMtDsOOIOJo47TlmQMMFkMDDZa6DBkNSGGQADAKvieiKHBBkvs16A47BA0NnuoalbMhCBMA14xKdrIcQcDYrhE53wdbLOwwdOCqfRZaQwLCu26ewy6E1vqSROqFxnzAhl8aGKWFSCh7C7e4U4/HLwKvoBDJJFQ/L8zoU4xYCuPy//qybGh1Xwj5W3rCk/x7YAAADSAAAAEhjcUZr2UzyAAANIAAAAQ0F5psw3LD7S5qj1wxOF2BfV22K2oCviMWYr5DW/yvt5YK2cY7xLM0Z/aExquE6ldMZPSlwwz6XBQuV8bnfv80XLozDTk0xJdxOtKqx2zQ4LjSjKhkkmI1mOC35isrgwIa1RY6tbM2YmRRNTuXQI2t3riYMVoKswCAfjAsBnMTwBQwXAOjC5J+PuUoYxaQ3DC2A1MA4CQMA9MFUB8w8GkzLYDtMFYEwwFwDwuAMXDYaYOALqkcELGEEQDBAA+PsA0woOgVYBMgHFQFEHeicBjXrX8GmMMbwsAESb1qBsD5EUNIcjE27eQGFSTTXZ1BlPDC9arE3vfemSyntvxcmG4M1Z3G3mnKj2xS5cmNO9qixsTEsis3Xh+B3/aFCJZXsWIKisGvvL9xCG/+VIycTI1ULe0o6Z0jRkrBxjGUaWPYlIskKDb3BpuoGJ+ZSL07GSQPC3pNebdyqRi55aNqV4533qrq2azNhgz7uGbWwvdlV5UNTXlOwKp2zZ4AAxerYkkDHtBnMH0JMwMAdDAUClMD8bAwLDWT9sR+MUgJswbAFjAuAUMGECQwFgnzIeYnMScNMSBkNDNTMBgyUNMDGjix4eC4wu6IJaqotNs6XMvIICSYAiq7Vl1qArBYefZ2ljy1/0slbRAEFUGQHOUuWXUcUlcOJBMJWuYSBONKIzA4CERAAoXGghyKBlVEVYClAggMBYlE7Q49TPQNZZi4zskieTzYZMWDDlIUkmSFuNhy1LMijSKaeeZEm04MV7S0e4bVacpyn8xMURdHaX2a//qybJwWaYj40G/EU9t7cgAADSAAAAEgjaEXteyAIAAANIKAAAQKdrJmNiE5q0yVTGip48k9p72KNWkahbjSVfxlhcJH1vZ9WtfBevtfdW1l1/uDjerZexJ62w+fWtivxa0KE+3QKCrVucSCmg27p5ayUYV4TAKCAAIB5gkgPmC+C+YTA7Bo9oImByBkYM4JZgLgXoemCWDsYYQHJjQgVioCpgjgApirEAMZVaYquRnGSNSWsXUzd9FGIDRixUb17tyLbpzvpQNIrKKSBGVhSghIKhOcN13AiDQJLI5XGbVVZkQwjcONabpJ3jiawLqxt/2vy9aKxELXAjtJGVpLboWeMQUEbZiDTHxcOSvgv6IYVptUsZvTDkUkrdia+XuvGmjYW4RCKGmn7OVXks1urT7lti1UpI1lb/est581hyX0+Od3d69+rV/vNWNVq1r5+7Vy1l/N5dv9vcsXcN162We9c7dv//N4bu2A/6F69xG9VeclIFEVIqYmYt/K9TdbbvjNn04kPAIiIHsACDTxQEM0HjZWEzYNAEKEQghHzLw1HEzhUYgZeGmjjaIAG9g4TMvEm1LymehpABnBNRrIsKEpgIepg+qV5iYGSAQQGFQ6MfJASdDiWjzFX2AwsDhB97jY2vGpFYAAjHQ8QiJjCwgUw4aC5CIx5pjLJ226blGJl4sKofpoAgHMGYhoSg5nsvpAaDBAIHAD0KWCQA7dQwopMaERAZGIBZggSYYKFq0PJ+Ls6rT0apn8sRiczsf5MCGPiIcGNxQ3MOAHZcB9aZsy6c4i0G1QSKTT9I/E3F86e3L7jjmKDyQrQiYRRLR+DghX//qybBB0gIAK6HnX/m9gACWgB1DAiAAhHZlDvdyAALIAX2+CMADuqkG3GUp4+mlFojah6VXJZezwldNFOXJZOdwvW2aqHlQASQQSg0QHg1W1EJYqhqqD6tVLmRmIsRlERdFvKr62Ksz/P/////v/////////////Xyvd1jrWV7iP6fo36OX9Bx/p/5o5/B8gc//LkP/hEDkHf//gcgETf////+NA4YAYhAEAAIxAhtAgjBATmHQZgwKjCQGTAYDzDQPTFggThwAjPIqTDAHTCoDSEKA4NjAoJzBIKTEgUzgwfz7wCUmWl3WIr5ABJzYnOAl6iq2u6BpzKmguM6VLd3EWAghNZjYTLDQ1Xyw1lKcqABM6DZPrUqh62aZyOQJEWK4stsylrM9AN7uNLS3AYuGAiAkHAsSd6jiKRqSKlKXMZpu40sZ3S/S7s1nCl9JCtW7WO8b11iL8v7TU2VNlKqb/3VfVUqGIKFa7Fcf/HGUv7S2fwuzspor9LZx7ullLuwneu5ZSqNNyt02VbHcpf10VSuLLcdWu7x53HHLKtKqatayyq48ypsiISCn+CoKhpQnEkgcwzMbHovNsuUiw0Tu0aPo3q6c82tyU/Ta7f9l//+7HUbu9f/7qeidqAAATNMWkMIwKwUDBNADIQWjE9A3MAIEoxMQqTDWIKOsfhAzcRLjC5D7MGoAMwBwKhEBGKBFGBKGOYUCaBkYIOlYQZhbgWGCYASYEoCiAEwBgJQoAoYHwvwCJWMA0CEwJgEwUAM1NpaRDIj7onGCyNalExeQsGMBhN8BLig41D5EA1ni3KLXbUorTLWhgycIKXxMcTdeU//qybFuXSwAJkXJJG9p9UjTgB6gAAAAbCZk7runvqNAAW9gQAABLsIk6SJS0XVn3do3VC4AzbAWFhxFTt/2S2KcRAQcRdVB2UTkar6jVrLsEx7zLo+3rCnRIUcup4Cv1CWmLV3qGq2CVDzc7arD+SgXwNBWQHr5rMZIielYxQps1pCt6138abjrrP/qLQPqZo3WZFuE4ZJ/P5tzONly1Vi4Yl0r3TM8UcefY0nJwfn5rWozr4/t/9fPt9+2MxZ/yoeuAMCIAVLcbWwleWza957Ip94tzrMqp+dZ1U63P2p1zXublV9zetiKFXbEaqfe2SSvvF9Lm6BEAKANF/LbOChRXiYJgYIwvFQuAIAmCwICMZzKTnzFIIDCYGSIEzAwCV9CoQMRMJR4CPCBzUeZPrJ5bNQACYhqVhaguYmzlNsFaKJeXlR0l8ASfl9D1JmkblKi2dNjZkksr/BlH2bfIwwVsgKDXsKbPD5mYXHt32wj9VgtzlI0bywrK6QqfOLe2N9gUma89YclizxfOUKxHw261m0fNliP/pJzQAy16a3lcMrkjmHzZl//+P/rJ1Sb3fGclUlGfeuib0Qp9jemrPzTWffbl4OmC8aivp/Jt6Pz7UyqaRZA/YZF049ulcy001A/pb39PFjWsi4W9j8WSMgUUNUhIi6aFDWMIuoFEYxmhX/oVIAgBBpVAMwnwIwKAWPAqAAAQwygIzAUAUMFEHUwORADUiqkMLYW0wfw2gQAyVgajwJpMDcFgRDB9JXM18l4FqToNhpcLCHzizmmzkAdwFgiyqHUxA6UhzwbjJJyRZV65bQ4jw2ODauAWK7L1PHhP//qybHrkN4QXiGjJ09p7yDcAB3UEYwAetZkib23vYMAAXdQAAADbmnNUkIE/OHcm9aRZiP6xScxMNQ6lCrYAghcWqGslK1O5YUGr9wVLHiGP2TEU3HOuxkxGid+gMdwaqOGIP8rrf+2DOR0Nk2t6SuIBStVdZxvWvXH+7QS0bPjG3eUkVsDer6jyqGlqZkzfcR9Gzqy3F+VLh/Iucb28816Ro/1s1PWNwcgADIDkjE0KQOSEUrQ3MsNN7HMHf8akaj+hSWGWf2PaNS7+PQOSz+tAqaa7+5g8gO/jkjUCqGXzESTAwB2HgEEmRAB8YRYK4IAREYMhhGA4HFKkWZb43xgdAbGAICQYBQFIJACMCIBIwRwJDCGKZNOgHw1kMPMTwMJKasSZStkZhQdcqzQ5DuUpdJN8yocfVB1rLpNRsTbMUDakHSQZA5zDNZbZsdCLA8o80YlUQvzkxK1DILWWmr6T25qjl+NlkY7vCmnhn68lw7qvutPl8WuFSKfzJqAYJqq563HXHo+V9GmJE91zqLf0UfoF3JEk1hGwT4Jq2yxu3ZxT73Z7GumC/w3H+BGszioXL22Hn2zWn+qfd8+JXMCNXF427eFa+IEYCEm3px08MY5MxoADHI3IeYS5F/2rsV9dirF2/rM3pu/Tcj+1dqms/oYMTf+m9LXfYpostgspvatIshAADunsjZMLgGUQAGDQNhgBgQmDCC0YFYEJgaABmEeIqb0y+JkGCFmBoDeYEQBosASWrAAAhgcgTmEQGkZxw450VR1SqRijtJMNLMdpNAVV2JDXU5AkZaYFT8uRmZ0IyJEpkL8v6uGC5REEnrVT//qybJ5SNwDnXlZIU9pj8j3gF4UEAAAgZdMU7z0VyGeAF8gACAzVVvasPA7T+RSdeRnDuRKMSypp9EoZZJqdrNJQVmQQ9GIYLQvvLD/pFi9e3pynlehFoz2Mv9G9cst3uDLV6+YEd77qf5p2HsxDo+2eUd9YKKuHeLKUpVjK1cg+4VVYparlSCMFzCzymNV79x5m3e+y45PHtvv9/yd1nU/sR+t++OTXpFlEh6WjNUYhg1LR5Eelo9DBhIehn///+0ewalo9LBhIehgxAqsiMQwelo+PS0chgwkMQwYlo8iMSBBhQoEAxywRzArBwMCUCULAamHQBSYHQCxg/gqGCQPMcrvghh7geGFaCqYJwMpgOgHGAQBiAQHjAlAzMEM5ozDQ+QwHkwvwPjAuALMAUAJQV3pSUAikwGZbtBWtGYBomaCeMBGLIsqdme1ilIV0QMFlrFThvl4PxRIU3UYoRvAgQ/jqRSi2nlZOgVIcUKFY8nzEux2HkrnzVChF+ULDG1GU0bamSx9fNNe66hLnWsUr9NKdevayxYrxnfuMKPtuYT9qr4r6HpXYTtiotZqg5ConhsOdSUEg/GgtVpEUOrKFhZox1bQ2xVkR8MULQvPeuwqvP5TEqpqqtWtNGpqgzcUmCjYX63f/////+3X///4sz//xUWQaFRRpoCiotSZxAnAkif3fV0w5FBlAKN4SAkgA8wSDswUAswOFsy6uMwnBow8AAwZAgwcCMwRAQYBd6jA0TDO8KjBQBWHOzfpc44CQiAw1KwqehV7HlJylY1BuFOy2bVgQI1FwRUKXTUpSGq6lrYpvZbkb876TSr44U9J6//qybC+MOgD2KF9Ke68degAADSAAAAEdOaUHTz0VSAAANIAAAARlw3IK8TV3jUZw21YSDvFaxo0A5EJ3ejnNhen3DrmLPZrJygt3jQPuE7g23GvSFc3XKL8VizQ/k9q/ZkvrlEPZlwwnQsnQm9rOAl+u9tk0kXIAADyokiATQAA9MG4DkwpwRAqAuYc4Axg4gGGGqEwYi4pRo5eiGUAMyYmIcxgrAJmA+AQEASmAaAmDARjCXF9M6cbwwQAVxIIQoAuQsHgDlqPqYGoBJMFQWtLjTdqtW46w8ol0A3KrDA1niEFE9nKhQtTChRtBukDmpFZ26FBCOnHGzSKfrKuHTDM9tu+4I5SXb39ysLxnu4uCrvHh0B+GU4e6QPCQUtyXoTgqXJvSSZMbfWUEizEUw6BvzzMlUbE90wqsmRV3G5yj6NxBRuZxi1ZLOmm1HSlXSjbOodyfc6J2GE4z6mxXqC0FACAO7f1plGLRshwfDICigTmJQUCQ4GKgSmMQFHyfymiJXGFgwjwGAoBTBMFjAAHwCARgEf5j8CxZcWEFe0ZgaQxsAAmnkhs5cpo4Lh+Sx5IGxabLIIxLo7OpXLGnH1lO5htWjPrHHHfuHH+nIo1pQZdkLgeKKNBgcRqLqodkgsUOutlBcobWS3aVS5AKRo0PbxVRgoCYYl3m6tDPVg8mz4tEyIi8tuF+hmh97pTqqcLzp9+vr+nhgQIA8n0aL/7QUA0A5Nq2UkcshQYZhSAjYbCY7BOYAgaAhGMVw/PxEmMkiPMOgUBQbrob9vU8jBQlTOALR4BggGYlSwZLbDxKWurCKlnO92kr/jGXlZVYn2up//qybM57iQD2BGPD66kfEAAADSAAAAEXKYcLrrB4yAAANIAAAASKTSqmPpAgViUJRbTWOh6bPEoOnuMCVAy849WlJ/+i9XsXw229p92jmuvM1By2RfzVJpG8d0VWZqbBFNnI18wHOExqZPCkU9NS2452nHEIT9mRm8FLsFItMOOXKR3/9uftAZzw923voUxBTUUzLjk5LjVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVQAESUjJIJMTsdIwQgXzCaAPMAICwwewJwECSYFAFhgdBqGwyPYYeoYxgfgdEwCiiLBVKC+Zih+fAkmDhCi7uW5Tu/PIpRzctlVLSxuljlqDJmWQ1DEbhxR6LO/XmYEoWW2X11bhmUyiW1Y5JKWX0FY6zLTPJk8450VBkgs+qO6CLRjq1FCED3MKG4q7hPpHmy8pyfc+4S5ddlp6i6VpKOfDgmC9mdQo90tPIxKyyaJnEukRPzBxU7o7D7w9F0QC1W5S3MWnmAWJxc9JqouIUkiZqT8qYpMFCRks4cHghbgADm4BRMRoEswSANhEBiYb4WBUBDMFoEUwURuTmQajMOwRMwyghTA7AUGgAUmlYC3Zimx1k5CIFjzDZmE0T8QCkTS0zOX+krgTVFWnl/NhiidMWjbiQC01NNjq+GeKMJ5rSh6MQp1nAjTA4yzGAm2hx1nKSqYDdopGqxcWjaECiWA6PVhJKVi0lLxKK5cgsOJKTEtvU5+LjewSwr2S8TB/TnSjSwnRlZKhCSN0i5SVhLTHJsPEN2HjFmRQeMmBejHolmM0VBue+cl2Eb1f//qybHaG1gD243M/09szcgAADSAAAAEjmerwz2mNwAAANIAAAAQ1XnZAYJI9Ekcj6F5GRoHRqQkygrr0yM5LR6XRmWSmlZWnR6apSqqLbShAeHQ4OpLTqYOSX7Y/nR1H761pagQQrSoOi5IQjrlpxCOp16pMQU1FMy45OS41qqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqBLEkcs2zRIRyjuAQ5R+ZEgl9gMTEKQd3dMYDjot616BZJDJKAI1UPrW5nkT+KmV2sr7xUKSxx1KqJC7BsHzdDNDYXNUEoxMBWT0MujpBAfjbrbLONMQXccRKEl6/Xr/NaTCOsZvwWZFMUoi8HVmFlKOaGKtRe82KbCdabSm7rf+cM+UxiHMp/zWSxP0Y1n0XiDVpuayOtp2J599jlkcPnVrD3CC7xU6BtWsJ0ST3bfxpJnwRSY3AgKBYNABhgHsJMjAcKIk0ArSYHBA5BwGf1nURURKgrmCGk0olXbSvN67lNUxyMqFiI8x0KVVWlhVJHPC5kuflzRxBh1I8MIc4BUQstCDifposCFkqMw0xJiFmhAXGS0JpBANVUB6AZ5SVRcbQFNBK3pigkJ0CkySmx+cj8PK43hJhYsdExcblRhhfGPx3EJScfBssWvg1aSMClcYHMv40//qybEteqwD1UGhDa2wz6AAADSAAAAEfLaUFrj2N6AAANIAAAARLbXoSO0EQobgBotEssJxOWXQR7RwoR+SCOU+YdNznzyhTeTnTK2BQhEwpuPNH6CT1eNKfJyGJ5bUvHS8fiwHt7MhSontmLcViB1JSK3VMQU1FMy45OS41VVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVQW25JJv9GSWedJJV/SxUVi03Yx6hT0DWcsc7G+5rG36ma/N8cMDh+JeKAQWmCItqW1iImJGX4C6ZtOSeeeNqmrnhxRsnx9aNfS6SB1pFgbGGUGBq5kBC1wrKyLGSGFBYZVH3LPEjhSe6HaDQ0OqTO0MWckqwjhxmPzr7H5Ghr5w4GMmTYBpw6UuocI2N1gCW0o5LakW2cEEMYaA2BkyRLVEhXGUVM5PRjJyZw8F0dKFt7M5mKo2dRsCQCQqsXNTjnZT/EcJ+BCFoIkLAtguQ0CLAwgnAB+WFWHKIIJmTUAPEnE0HzczAY4XolgVpChXlTFTosJRLoplMoYBomqZyiVa4J6+2p4SRZW9TqVQo7aHMSfSDMro7a8u2F2Q18unJPR2JDlyhTLKYMPR7sMfwVtQPDSblKfyYVpqMs43RwoUYhzuME9jgRzA//qybDHZpID0bmXE60wb+gAADSAAAAEhGaL7rung6AAANIAAAAShNN+2KtMqpwVR+uCoOJUn4cxTIxIrENQqxCmZWJs9jqZjiWzdlqvYU7pOnTCfOaJUrI4vYTCu25c4eynTJGQzUUUNxmrOx9unEpaGioVMQU1FMy45OS41VVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVASbkl2/1slxzgcOZ6neZWI1lh9i5irpI22k8nphgqSoIxRn5CyJQMl1EorhtmSOid5FsVbOMIIrpKdqNuEpKiMYGZ4Y3FFLZm9F5CbYkS4t27EBfP1F2ISlKXCr1z4jH/Q9F3hBtKRb1T8okHLcjrsnWNBretnmieNwx/kQkm25rv9DiWcxEAMEE4KeVnT2UD9WHIazD62qKAb7/r/et523LsBGkEDlrpa41e/YTUdZdDSk3gccQDWiycuCXcUtanTKmVVc8HBVsZegmZwzACkJ2oi4VLoX8usVdsZQErTNkTp+vJCh+KdAjWJEj10okiq280kurS1VDccBoqtF5TqhQ1FKUmw4U1CVSkVD5OqtjU50LhRx2JzL4xOByG6WxgOo8jLanBORz4fKGEyTOKcUS//qybGLtmQHz5mTFa0kbegAADSAAAAEgYaMBreHr4AAANIAAAAS0hJ8n+dBzKBUrKHRoiiuWqBM92dZ2uBYk6cByvW1OpRPpO7kznO5l3L/04XJuMo7Uujy+4VcZMHuu3CRaYYPOFLM5/wl9Jsu2LQe0VGVMQU1FMy45OS41VVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVoAHOdReDiOXY3RkESpX1dntLXQ+THoKgiMlW4mgciqkWIqE4yWiaQajqSTsnm5tUsGcPLg+EpY4PQ8oSViUm5wXVgs9aFFiFC4ocSVTezGGLpqpzjbzR/GXQVumVUJKS5SSI+jQkrGXNJTss5XKFMVVTm2zrTT91oiXbu8cos0zJETPSWaaTvz1YoUImDimMsOXXggQaXIzTWznYyQoSVDBG1JhytpqvPrMcwAYpm3JkduMopM0+0MeAnCflNJ5E5nUf6Bqz7MmU2iA6As5Z9C5dQ1EXdMyt6Huhwl/J4FUDtI2Q5NDjSsyVXkYXtiIMoB4BmiaBAjcOEHuWxnP5XnknJmJyNZCibJ1hVyULxHhwHC7+qdfOZw3RzCkR5sMzxXtaagrpXvGSRsejkNiQPysnn5OafPC8Zox8JqGuDIPSIIBZMn8PgnPB0O38RA1aNxzxk//qybGgcqgj1hGg9k6xK+AAADSAAAAEeHaL3Tb2R6AAANIAAAAR9XF4dGRzbKS8yK6ETSYfLwbPtlg8HgeLqIzCxBJrI4HKmBGS1hIOxKdhSmBcuOmrlp6TPXksQkqpIfmbI/Ly56p7SFv7I7u+Mh+KtQwpMQU1FMy45OS41qqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqAABZNopJwPFhhICyqUNnmoOh4JawHyfVB2OrMCcxushOFcJTPEZ1QOjZvBCDYqIDSHB8CQqmwVJWB8DgKnbqpcs2zNTyVshhJaMnI0MO52sNrSVJEfh0SJM4nJaptUKiGUYFSp5o9LUSFdg6gRNrLvRHVIq4IcfGSEl1CiLKNcobSE1IdRAsjOI0LEsWIViuFpIlR1pVETO1RAniIuemyVVzRCpeKqyTQipDVJguGez+kBWrPoGpMZhaHh1DgdAwliMIDA8FTAYGR2xdoQxWBRCKoASVfkvQxECYDuA4ZjCu5J9tU0nnc1MNvmVFQc+weH2IlzN41x2nccpuGcui6IeoJy2lQiJzrBzkGHOZIuh4HUTg4yDDTeqM+SxF0P1BGGpjtUBmNR3sh+KYlJ2HI5NhbDdNNCS6F4bjcQwtpILVLqcRUc1TsOUwVaehuH+8LebpMk4oIT47D1PBFF3F4rCYIg3TkX2M0lWZj5NG4W1IK5+fjm8NMoT+fqudyPY31lsO8varL+eiRUi5//qybEZ1xgj1jGc/U4xJ+AAADSAAAAElCeTiTuHnyAAANIAAAARMnFAT6tRhMCUqhDsGno/0eNNib2ligGCQRStxdS9psnZRl0TKZKE22ZC2Gw9y/K1dePhjir2ofhSNasgRWKPZnY2CDEXThtn8Bcwo2p5MQU1FMy45OS41qqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqBcctl132124PJy6BIvBkZi0Ow6/qmS6h/HIxVmtXB+JrolNGzkLBbKB8nA6VTyPhPEgniSWC2egJEpWUgdNGhSRKk2gtMLJkJa3PQgZWMkQGlWQ+JWCcCUyEcAyQsKiESH8gMCaSFUPEblmBlhINCoqIiZI6kqq5IDXVFKxK0WTOhElZaMtsdooSAkDSAeW3VA8jFIwDREiNJJKlhUMmIyWSIVooRFMqGSKiZDTwRxJJYTHCZcwfQkXLmJ0WJ2IszLFyVkQkDOABIgkRpEm2t25yUNJwLEdN13QdV22vwKIZuHAGCuB94kBW0CCAAsbmBoEaZaPYMhEUALNxAHdcCBYFYDzoBjg+DgazrV5uKk/1EdaqJwyJMuaWHoLgTAlilLmdZuFsQkl6MNOEfiRVbtQFsXRbEqTtCz8ORZQ8uC7NNiVEzPRgkakIU5O0kQRCTrOhrJ+gy+FgT6HsUdzmTi+q0gcDeQhTl/R6IFwZRXzbIOXuAcjgvoW2//qwbOKZvQD2PWjB60xK+AAADSAAAAEf4aL17jHp4AAANIAAAAQdB+TItpYDoiF3ONMi5q4ehULsubmXw4zsTT9zZNMb9xLeqyeGQe5L3ZyGWui4Jot7mnIt5tt8I/Faaa2hiFtxyIQf5O10XBxVis/sRUxBTUUzLjk5LjVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVAUbcl22t125uoyrU+qiab8A6QEIpEr9v2XvQqjLVCoAdRwH9UuWS77sNxTPXkkYiNAqvE7W2rI3EIWqNNgB5YCEkKMv0nQTwxxoBHUJDOJwiDdYyvJC4H8qSxl0VqiPlSEzONWGkPsvqGoM6YJ0EePWXg31kvhNVyF8Q1pY0Wqj1FgXRkGsTAkpcRZTkO08iYMhfyaHMcbG9bS9E7JwfLGi0NLqBpGKN8lahcieheIUuUvIskBHEkFbMJwdAr4vFM1LseichRyiwQIJ6KNSjGNpTl7HQ2kCXCqJWWJEp07RYoaqimkjRH1YnifronD0sTQVjOnS5LyIJUlTxQsTTBfkSfxGy3HgrYp3Qi4RDKWreZ9z2oscCkm3HrtZZbgocaCPUZo8UORKwqj+UZkopXoSxL6yTSERjFCjWtCWB4eSQB4ulQdiSNRFBIhlAGxfMBGLp2B0vGY6lYvmohFJGJLD5jA2enKw+SuPHLD5k24ZQ0PmWD0rK1J0/EZJ3zFIjPUz6krOKjlx45YjMUNEuQ4njlx46fUn/+rJsQvvLgPixaL7rWHp4AAANIAAAARndowesPYGgAAA0gAAABNL1J0hojFCNlp0rUkpefHKw+WoSpaXlK5KiXHKw+OWz1qFSuSqVyZGtgfPUzZ6tYPmYI1qEjMoXD6Fg+ZSHrTcS6FwyhfMWLnrTZ6dMKkxBTUUzLjk5LjVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVQU7bdtbNa5MIRBHgnj4UR9JwIgVBwG4sGQ6iUOI9A8D4gD+UB2EkchSQR4JZYNzUyKpeTJzhe5G6uhMjIwPOFopGnGihQsgHRJAooSJEExgcJJAooD/+rJs9RAqj/URY7JpgTYKAAANIAAAAQAAAaQAAAAgAAA0gAAABIWQTUkROElmHxaqlm7Oz/9nzZqWcuLUkicacUfCakiJxZZlwtGjSini42af/s7P3dtyppyij0FopETgMDMJoLRJGlFHmWpJE4sukDVJKkxBTUUzLjk5LjWqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqo=\"","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\";\nimport type { JSX } from \"preact\";\nimport type { BulutRuntimeConfig } from \"../index\";\nimport {\n agentTextChatStream,\n agentVoiceChatStream,\n agentResumeStream,\n startSttWebSocketStream,\n stopActiveAudioPlayback,\n speakText,\n type AudioStreamState,\n type StreamController,\n type AgentToolCallInfo,\n type SttWsController,\n} from \"../api/client\";\nimport {\n executeSingleToolCall,\n parseAgentResponse,\n getPendingAgentResume,\n clearPendingAgentResume,\n type ToolCallWithId,\n} from \"../agent/tools\";\nimport { getPageContext } from \"../agent/context\";\nimport {\n WINDOW_WIDTH,\n WINDOW_HEIGHT,\n POSITION_BOTTOM,\n POSITION_RIGHT,\n COLORS,\n TRANSITIONS,\n BORDER_RADIUS,\n SHADOW,\n} from \"../styles/constants\";\nimport {\n logoContent,\n arrowPathIconContent,\n commandLineIconContent,\n cursorArrowRaysIconContent,\n faceSmileIconContent,\n handRaisedIconContent,\n mapIconContent,\n microphoneOutlineIconContent,\n queueListIconContent,\n stopOutlineIconContent,\n xMarkIconContent,\n} from \"../assets\";\nimport { StreamingJsonParser } from \"../utils/streamingJson\";\nimport { playCue, type SfxName } from \"../audio/sfxManager\";\nimport { SvgIcon } from \"./SvgIcon\";\n\nexport interface ChatWindowHandle {\n startRecording: () => void;\n cancelRecording: () => void;\n stopTask: () => void;\n}\n\ninterface ChatWindowProps {\n onClose: () => void;\n config: BulutRuntimeConfig;\n accessibilityMode?: boolean;\n onAccessibilityToggle?: () => void;\n hidden?: boolean;\n actionsRef?: { current: ChatWindowHandle | null };\n onRecordingChange?: (recording: boolean) => void;\n onBusyChange?: (busy: boolean) => void;\n onPreviewChange?: (text: string | null) => void;\n}\n\ninterface Message {\n id: number;\n text: string;\n isUser: boolean;\n /** \"message\" (default) | \"tool\" for tool call indicators */\n type?: \"message\" | \"tool\";\n toolKind?: \"context\" | \"cursor\" | \"scroll\" | \"navigate\" | \"form\" | \"interact\" | \"unknown\";\n toolLabel?: string;\n toolCount?: number;\n}\n\ntype RecordingMode = \"vad\" | \"press\";\n\ntype StorageLike = {\n removeItem: (key: string) => void;\n};\n\nconst STORAGE_KEY = \"bulut_chat_history\";\nconst TIMESTAMP_KEY = \"bulut_chat_timestamp\";\nconst SESSION_ID_KEY = \"bulut_session_id\";\nconst TTL_MS = 5 * 60 * 1000;\nconst VAD_THRESHOLD = 0.06;\nconst SILENCE_DURATION_MS = 500;\nconst ACCESSIBILITY_MIN_SPEECH_DURATION_MS = 1500;\nexport const HOLD_THRESHOLD_MS = 250;\n\nconst STATUS_LABELS = {\n ready: \"Hazır\",\n loading: \"Bir saniye\",\n micInitializing: \"Mikrofonu hazırlıyorum\",\n listening: \"Sizi dinliyorum\",\n accessibilityActive: \"Erişilebilirlik Aktif\",\n transcribing: \"Düşünüyorum\",\n thinking: \"Düşünüyorum\",\n playingAudio: \".\",\n runningTools: \"Siteyle ilgileniyorum\",\n} as const;\n\nexport const getGreetingText = (agentName: string): string =>\n `Merhaba, ben ${agentName}. Bu web sayfasında neler yapalım?`;\n\nexport interface StatusFlags {\n isBusy: boolean;\n isRecording: boolean;\n isTranscribing: boolean;\n isThinking: boolean;\n isRenderingAudio: boolean;\n isPlayingAudio: boolean;\n isRunningTools: boolean;\n}\n\nexport const resolveStatusText = (flags: StatusFlags): string => {\n if (flags.isRecording) return STATUS_LABELS.listening;\n if (flags.isRunningTools) return STATUS_LABELS.runningTools;\n if (flags.isPlayingAudio) return STATUS_LABELS.playingAudio;\n if (flags.isThinking) return STATUS_LABELS.thinking;\n if (flags.isTranscribing) return STATUS_LABELS.transcribing;\n if (flags.isBusy) return STATUS_LABELS.loading;\n return STATUS_LABELS.ready;\n};\n\nexport const hasActiveStatus = (\n flags: StatusFlags,\n statusOverride: string | null,\n): boolean =>\n Boolean(\n statusOverride\n || flags.isBusy\n || flags.isRecording\n || flags.isTranscribing\n || flags.isThinking\n || flags.isRenderingAudio\n || flags.isPlayingAudio\n || flags.isRunningTools,\n );\n\nexport const formatDurationMs = (durationMs: number): string => {\n const totalSeconds = Math.max(0, Math.floor(durationMs / 1000));\n const minutes = Math.floor(totalSeconds / 60)\n .toString()\n .padStart(2, \"0\");\n const seconds = (totalSeconds % 60).toString().padStart(2, \"0\");\n return `${minutes}:${seconds}`;\n};\n\nexport const classifyMicGesture = (\n durationMs: number,\n thresholdMs: number = HOLD_THRESHOLD_MS,\n): \"tap\" | \"hold\" => (durationMs >= thresholdMs ? \"hold\" : \"tap\");\n\nexport const createInitialMessages = (agentName: string): Message[] => [\n {\n id: 1,\n text: getGreetingText(agentName),\n isUser: false,\n },\n];\n\nexport const clearPersistedChatState = (storage: StorageLike | null): void => {\n if (!storage) {\n return;\n }\n\n storage.removeItem(STORAGE_KEY);\n storage.removeItem(TIMESTAMP_KEY);\n storage.removeItem(SESSION_ID_KEY);\n};\n\nexport const scrollElementToBottom = (\n element: { scrollTop: number; scrollHeight: number } | null,\n): void => {\n if (!element) {\n return;\n }\n\n element.scrollTop = element.scrollHeight;\n};\n\nconst normalizeError = (error: unknown) => {\n if (error instanceof Error) {\n return error.message;\n }\n return \"Bilinmeyen hata\";\n};\n\nconst getNextMessageId = (messages: Message[]): number => {\n const maxId = messages.reduce((acc, message) => Math.max(acc, message.id), 0);\n return maxId + 1;\n};\n\nexport interface AssistantPayloadResolution {\n displayText: string;\n toolCalls: ReturnType<typeof parseAgentResponse>[\"toolCalls\"];\n}\n\nexport const resolveAssistantPayload = (\n assistantText: string,\n): AssistantPayloadResolution => {\n const parsed = parseAgentResponse(assistantText);\n return {\n displayText: parsed.reply || assistantText,\n toolCalls: parsed.toolCalls,\n };\n};\n\nexport const shouldAutoListenAfterAudio = (\n accessibilityMode: boolean,\n expectsReply: boolean,\n isRecording: boolean,\n isBusy: boolean,\n): boolean => (accessibilityMode || expectsReply) && !isRecording && !isBusy;\n\nexport const shouldAcceptVadSpeech = (\n speechDurationMs: number,\n enforceMinSpeechDuration: boolean,\n minSpeechDurationMs: number = ACCESSIBILITY_MIN_SPEECH_DURATION_MS,\n): boolean => !enforceMinSpeechDuration || speechDurationMs >= minSpeechDurationMs;\n\ninterface ToolIndicatorMessage {\n text: string;\n kind: \"context\" | \"cursor\" | \"scroll\" | \"navigate\" | \"form\" | \"interact\" | \"unknown\";\n}\n\nconst getToolIndicatorMessage = (\n call: AgentToolCallInfo,\n): ToolIndicatorMessage => {\n if (call.tool === \"getPageContext\") {\n return { text: \"Algılama\", kind: \"context\" };\n }\n if (call.tool === \"scroll\") {\n return { text: \"Kaydırma\", kind: \"scroll\" };\n }\n if (call.tool === \"navigate\") {\n const url = typeof call.args.url === \"string\" ? call.args.url.trim() : \"\";\n return {\n text: url ? `Sayfa Geçişi: ${url}` : \"Sayfa Geçişi\",\n kind: \"navigate\",\n };\n }\n if (call.tool === \"interact\" && call.args.action === \"move\") {\n return { text: \"Serbest İmleç\", kind: \"cursor\" };\n }\n if (call.tool === \"interact\" && call.args.action === \"type\") {\n return { text: \"Form Doldurma\", kind: \"form\" };\n }\n if (call.tool === \"interact\" && call.args.action === \"submit\") {\n return { text: \"Form Gönderme\", kind: \"form\" };\n }\n if (call.tool === \"interact\" && call.args.action === \"click\") {\n return { text: \"Tıklama\", kind: \"interact\" };\n }\n if (call.tool === \"interact\") {\n return { text: \"Etkileşim\", kind: \"interact\" };\n }\n return {\n text: call.tool || \"Araç\",\n kind: \"unknown\",\n };\n};\n\nexport const ChatWindow = ({\n onClose,\n config,\n accessibilityMode = false,\n onAccessibilityToggle,\n hidden = false,\n actionsRef,\n onRecordingChange,\n onBusyChange,\n onPreviewChange,\n}: ChatWindowProps) => {\n const [messages, setMessages] = useState<Message[]>(() => {\n if (typeof localStorage !== \"undefined\") {\n const saved = localStorage.getItem(STORAGE_KEY);\n const timestamp = localStorage.getItem(TIMESTAMP_KEY);\n\n if (saved && timestamp) {\n const timePassed = Date.now() - parseInt(timestamp, 10);\n if (timePassed < TTL_MS) {\n try {\n return JSON.parse(saved) as Message[];\n } catch {\n // Ignore parse error and continue with default.\n }\n } else {\n clearPersistedChatState(localStorage);\n }\n }\n }\n\n return createInitialMessages(config.agentName);\n });\n\n const [isBusy, setIsBusy] = useState(false);\n const [isRecording, setIsRecording] = useState(false);\n const [isTranscribing, setIsTranscribing] = useState(false);\n const [isThinking, setIsThinking] = useState(false);\n const [isRenderingAudio, setIsRenderingAudio] = useState(false);\n const [isPlayingAudio, setIsPlayingAudio] = useState(false);\n const [isRunningTools, setIsRunningTools] = useState(false);\n const [isMicPending, setIsMicPending] = useState(false);\n const [recordingDurationMs, setRecordingDurationMs] = useState(0);\n const [statusOverride, setStatusOverride] = useState<string | null>(null);\n const statusFlags: StatusFlags = {\n isBusy,\n isRecording,\n isTranscribing,\n isThinking,\n isRenderingAudio,\n isPlayingAudio,\n isRunningTools,\n };\n const resolvedStatusText = resolveStatusText(statusFlags);\n const showStatus = hasActiveStatus(statusFlags, statusOverride);\n const statusText = showStatus ? (statusOverride ?? resolvedStatusText) : STATUS_LABELS.ready;\n\n const isBusyRef = useRef(isBusy);\n const isRecordingRef = useRef(isRecording);\n\n const nextMessageIdRef = useRef(getNextMessageId(messages));\n const recorderRef = useRef<MediaRecorder | null>(null);\n const streamRef = useRef<MediaStream | null>(null);\n const audioChunksRef = useRef<BlobPart[]>([]);\n const activeStreamControllerRef = useRef<StreamController | null>(null);\n const sessionIdRef = useRef<string | null>(\n typeof localStorage !== \"undefined\"\n ? (() => {\n const ts = localStorage.getItem(TIMESTAMP_KEY);\n if (ts && Date.now() - parseInt(ts, 10) < TTL_MS) {\n return localStorage.getItem(SESSION_ID_KEY);\n }\n return null;\n })()\n : null,\n );\n\n const silenceStartRef = useRef<number | null>(null);\n const vadIntervalRef = useRef<number | null>(null);\n const audioContextRef = useRef<AudioContext | null>(null);\n const sourceRef = useRef<MediaStreamAudioSourceNode | null>(null);\n\n const discardNextRecordingRef = useRef(false);\n\n const micPressStartRef = useRef<number | null>(null);\n const micHoldTimeoutRef = useRef<number | null>(null);\n const micHoldTriggeredRef = useRef(false);\n const recordingModeRef = useRef<RecordingMode | null>(null);\n const pendingStopAfterStartRef = useRef(false);\n const startRecordingPendingRef = useRef(false);\n\n const assistantMessageIdRef = useRef<number | null>(null);\n const assistantTextBufferRef = useRef(\"\");\n const transcriptionReceivedRef = useRef(false);\n const assistantDoneReceivedRef = useRef(false);\n\n const recordingStartedAtRef = useRef<number | null>(null);\n const recordingTimerIntervalRef = useRef<number | null>(null);\n\n const messagesContainerRef = useRef<HTMLDivElement | null>(null);\n const messagesContentRef = useRef<HTMLDivElement | null>(null);\n\n const pendingUserTextRef = useRef<string | null>(null);\n const pendingAssistantTextRef = useRef<string>(\"\");\n const streamingJsonParserRef = useRef<StreamingJsonParser | null>(null);\n const awaitingAssistantResponseRef = useRef(false);\n const activeSttWsRef = useRef<SttWsController | null>(null);\n const liveTranscriptionMessageIdRef = useRef<number | null>(null);\n const liveTranscriptionTextRef = useRef(\"\");\n const autoListenSuppressedRef = useRef(false);\n const expectsReplyRef = useRef(true);\n const requestEpochRef = useRef(0);\n const sttSendCuePlayedRef = useRef(false);\n\n useEffect(() => {\n isBusyRef.current = isBusy;\n }, [isBusy]);\n\n useEffect(() => {\n isRecordingRef.current = isRecording;\n }, [isRecording]);\n\n // Report state changes to parent\n useEffect(() => { onRecordingChange?.(isRecording); }, [isRecording]);\n useEffect(() => { onBusyChange?.(isBusy); }, [isBusy]);\n\n // Derive and report preview text to parent\n useEffect(() => {\n if (!onPreviewChange) return;\n if (isRecording) {\n onPreviewChange(statusOverride ?? STATUS_LABELS.listening);\n return;\n }\n // When audio is rendering/playing, show the actual message text\n if (isRenderingAudio || isPlayingAudio) {\n const lastAssistant = [...messages].reverse().find(m => !m.isUser && m.type !== \"tool\");\n onPreviewChange(lastAssistant?.text ?? getGreetingText(config.agentName));\n return;\n }\n if (showStatus) {\n const st = statusOverride ?? resolveStatusText({\n isBusy,\n isRecording,\n isTranscribing,\n isThinking,\n isRenderingAudio,\n isPlayingAudio,\n isRunningTools,\n });\n onPreviewChange(st);\n return;\n }\n // Show last assistant message (or greeting)\n const lastAssistant = [...messages].reverse().find(m => !m.isUser && m.type !== \"tool\");\n onPreviewChange(lastAssistant?.text ?? getGreetingText(config.agentName));\n }, [\n isRecording,\n isBusy,\n isTranscribing,\n isThinking,\n isRunningTools,\n isPlayingAudio,\n isRenderingAudio,\n statusOverride,\n showStatus,\n messages,\n ]);\n\n const playSfx = (name: SfxName) => {\n playCue(name);\n };\n\n const beginRequestEpoch = () => {\n requestEpochRef.current += 1;\n return requestEpochRef.current;\n };\n\n const invalidateRequestEpoch = () => {\n requestEpochRef.current += 1;\n };\n\n const isCurrentRequestEpoch = (epoch: number): boolean =>\n requestEpochRef.current === epoch;\n\n const playSttSentCueOnce = () => {\n if (sttSendCuePlayedRef.current) {\n return;\n }\n sttSendCuePlayedRef.current = true;\n playSfx(\"sent\");\n };\n\n useEffect(() => {\n if (typeof localStorage !== \"undefined\") {\n localStorage.setItem(STORAGE_KEY, JSON.stringify(messages));\n localStorage.setItem(TIMESTAMP_KEY, Date.now().toString());\n }\n }, [messages]);\n\n const scrollMessagesToBottom = () => {\n scrollElementToBottom(messagesContainerRef.current);\n };\n\n useLayoutEffect(() => {\n scrollMessagesToBottom();\n }, [messages, statusText, isBusy, isRecording]);\n\n useEffect(() => {\n const content = messagesContentRef.current;\n if (!content || typeof ResizeObserver === \"undefined\") {\n return;\n }\n\n const observer = new ResizeObserver(() => {\n scrollMessagesToBottom();\n });\n\n observer.observe(content);\n return () => observer.disconnect();\n }, []);\n\n const stopRecordingTimer = () => {\n if (recordingTimerIntervalRef.current !== null) {\n window.clearInterval(recordingTimerIntervalRef.current);\n recordingTimerIntervalRef.current = null;\n }\n recordingStartedAtRef.current = null;\n };\n\n const startRecordingTimer = () => {\n stopRecordingTimer();\n recordingStartedAtRef.current = Date.now();\n setRecordingDurationMs(0);\n\n recordingTimerIntervalRef.current = window.setInterval(() => {\n const startedAt = recordingStartedAtRef.current;\n if (startedAt === null) {\n setRecordingDurationMs(0);\n return;\n }\n setRecordingDurationMs(Date.now() - startedAt);\n }, 200);\n };\n\n const resetProcessingFlags = () => {\n setIsTranscribing(false);\n setIsThinking(false);\n setIsRenderingAudio(false);\n setIsPlayingAudio(false);\n setIsRunningTools(false);\n setStatusOverride(null);\n assistantMessageIdRef.current = null;\n assistantTextBufferRef.current = \"\";\n transcriptionReceivedRef.current = false;\n assistantDoneReceivedRef.current = false;\n awaitingAssistantResponseRef.current = false;\n pendingUserTextRef.current = null;\n pendingAssistantTextRef.current = \"\";\n };\n\n const clearMicHoldTimeout = () => {\n if (micHoldTimeoutRef.current !== null) {\n window.clearTimeout(micHoldTimeoutRef.current);\n micHoldTimeoutRef.current = null;\n }\n };\n\n const cleanupVAD = () => {\n if (vadIntervalRef.current !== null) {\n clearInterval(vadIntervalRef.current);\n vadIntervalRef.current = null;\n }\n\n if (sourceRef.current) {\n sourceRef.current.disconnect();\n sourceRef.current = null;\n }\n\n if (audioContextRef.current) {\n audioContextRef.current.close().catch(() => {});\n audioContextRef.current = null;\n }\n\n silenceStartRef.current = null;\n };\n\n const stopStreamTracks = () => {\n if (!streamRef.current) {\n return;\n }\n\n streamRef.current.getTracks().forEach((track) => track.stop());\n streamRef.current = null;\n };\n\n const stopActiveStream = () => {\n if (!activeStreamControllerRef.current) {\n return;\n }\n\n activeStreamControllerRef.current.stop();\n activeStreamControllerRef.current = null;\n };\n\n const cancelActiveSttWs = () => {\n const activeSttWs = activeSttWsRef.current;\n activeSttWsRef.current = null;\n activeSttWs?.cancel();\n liveTranscriptionMessageIdRef.current = null;\n liveTranscriptionTextRef.current = \"\";\n };\n\n useEffect(\n () => () => {\n invalidateRequestEpoch();\n clearMicHoldTimeout();\n pendingStopAfterStartRef.current = false;\n\n stopActiveStream();\n stopActiveAudioPlayback();\n cancelActiveSttWs();\n cleanupVAD();\n stopStreamTracks();\n stopRecordingTimer();\n\n const recorder = recorderRef.current;\n if (recorder) {\n recorder.ondataavailable = null;\n recorder.onstop = null;\n recorder.onerror = null;\n if (recorder.state !== \"inactive\") {\n recorder.stop();\n }\n recorderRef.current = null;\n }\n\n cancelActiveSttWs();\n },\n [],\n );\n\n // ── Resume agent loop after full-page navigation ────────────────\n useEffect(() => {\n const resumeState = getPendingAgentResume();\n if (!resumeState) return;\n\n clearPendingAgentResume();\n console.info(\"[Bulut] Resuming agent after navigation\");\n\n // Restore session ID from resume state\n if (resumeState.sessionId) {\n sessionIdRef.current = resumeState.sessionId;\n if (typeof localStorage !== \"undefined\") {\n localStorage.setItem(SESSION_ID_KEY, resumeState.sessionId);\n }\n }\n\n const requestEpoch = beginRequestEpoch();\n setIsBusy(true);\n isBusyRef.current = true;\n setIsRunningTools(true);\n setStatusOverride(STATUS_LABELS.thinking);\n\n const freshPageContext = getPageContext().summary;\n\n const resumeToolExec = async (\n call: AgentToolCallInfo,\n ): Promise<{ call_id: string; result: string }> => {\n const toolCall: ToolCallWithId = {\n tool: call.tool as \"navigate\" | \"getPageContext\" | \"interact\" | \"scroll\",\n call_id: call.call_id,\n ...call.args,\n } as ToolCallWithId;\n return executeSingleToolCall(toolCall);\n };\n\n const controller = agentResumeStream(\n config.backendBaseUrl,\n resumeState,\n freshPageContext,\n {\n onSessionId: (sid) => {\n if (!isCurrentRequestEpoch(requestEpoch)) return;\n if (sid && sid !== sessionIdRef.current) {\n sessionIdRef.current = sid;\n if (typeof localStorage !== \"undefined\") {\n localStorage.setItem(SESSION_ID_KEY, sid);\n }\n }\n },\n onAssistantDelta: (delta) => {\n if (!isCurrentRequestEpoch(requestEpoch)) return;\n setIsRunningTools(false);\n setIsThinking(true);\n setStatusOverride(null);\n\n pendingAssistantTextRef.current += delta;\n\n if (assistantMessageIdRef.current === null) {\n assistantMessageIdRef.current = appendMessage(\n pendingAssistantTextRef.current,\n false,\n );\n } else {\n updateMessageText(\n assistantMessageIdRef.current,\n pendingAssistantTextRef.current,\n );\n }\n },\n onAssistantDone: (assistantText, expectsReply) => {\n if (!isCurrentRequestEpoch(requestEpoch)) return;\n playSfx(\"completed\");\n expectsReplyRef.current = expectsReply !== false;\n setStatusOverride(null);\n setIsThinking(false);\n setIsRenderingAudio(true);\n\n const finalDisplayText =\n assistantText || pendingAssistantTextRef.current;\n pendingAssistantTextRef.current = finalDisplayText;\n\n if (assistantMessageIdRef.current !== null) {\n updateMessageText(\n assistantMessageIdRef.current,\n finalDisplayText,\n );\n } else {\n assistantMessageIdRef.current = appendMessage(\n finalDisplayText,\n false,\n );\n }\n },\n onIntermediateReply: (text) => {\n if (!isCurrentRequestEpoch(requestEpoch)) return;\n void speakText(\n config.backendBaseUrl, text, config.voice,\n accessibilityMode, (state) => handleAudioStateChange(state, requestEpoch),\n ).catch((err) => console.warn(\"[Bulut] intermediate TTS failed\", err));\n },\n onToolCalls: (calls) => {\n if (!isCurrentRequestEpoch(requestEpoch)) return;\n if (calls.length > 0) {\n playSfx(\"toolCall\");\n }\n setIsRunningTools(true);\n setStatusOverride(STATUS_LABELS.runningTools);\n appendToolIndicatorMessages(calls);\n\n assistantMessageIdRef.current = null;\n pendingAssistantTextRef.current = \"\";\n },\n onToolResult: () => {},\n onIteration: () => {\n if (!isCurrentRequestEpoch(requestEpoch)) return;\n playSfx(\"thinking\");\n setIsThinking(true);\n setStatusOverride(STATUS_LABELS.thinking);\n },\n onAudioStateChange: (state) => {\n handleAudioStateChange(state, requestEpoch);\n },\n onError: (err) => {\n if (!isCurrentRequestEpoch(requestEpoch)) return;\n setStatusOverride(null);\n appendMessage(`Hata: ${err}`, false);\n },\n },\n resumeToolExec,\n );\n\n activeStreamControllerRef.current = controller;\n\n controller.done\n .catch(() => {})\n .finally(() => {\n if (!isCurrentRequestEpoch(requestEpoch)) return;\n setIsBusy(false);\n isBusyRef.current = false;\n setIsRunningTools(false);\n setIsThinking(false);\n setIsRenderingAudio(false);\n setIsPlayingAudio(false);\n setStatusOverride(null);\n pendingAssistantTextRef.current = \"\";\n assistantMessageIdRef.current = null;\n activeStreamControllerRef.current = null;\n\n if (\n !autoListenSuppressedRef.current &&\n shouldAutoListenAfterAudio(\n accessibilityMode,\n expectsReplyRef.current,\n isRecordingRef.current,\n false,\n )\n ) {\n void startRecording(\"vad\");\n }\n // Reset for next turn\n expectsReplyRef.current = true;\n });\n }, []); // eslint-disable-line react-hooks/exhaustive-deps\n\n const appendMessage = (\n text: string,\n isUser: boolean,\n options?: {\n type?: \"message\" | \"tool\";\n toolKind?: Message[\"toolKind\"];\n toolLabel?: string;\n toolCount?: number;\n },\n ): number => {\n const id = nextMessageIdRef.current++;\n setMessages((previous) => [\n ...previous,\n {\n id,\n text,\n isUser,\n type: options?.type,\n toolKind: options?.toolKind,\n toolLabel: options?.toolLabel,\n toolCount: options?.toolCount,\n },\n ]);\n return id;\n };\n\n const appendToolIndicatorMessages = (calls: AgentToolCallInfo[]) => {\n setMessages((previous) => {\n const next = [...previous];\n\n for (const call of calls) {\n const indicator = getToolIndicatorMessage(call);\n const last = next[next.length - 1];\n const previousToolText = typeof last?.text === \"string\"\n ? last.text.replace(/\\s+\\(\\d+\\)$/, \"\")\n : \"\";\n\n if (\n last\n && !last.isUser\n && last.type === \"tool\"\n && previousToolText === indicator.text\n ) {\n const extractedCount = Number.parseInt(\n (last.text.match(/\\((\\d+)\\)\\s*$/)?.[1] ?? \"1\"),\n 10,\n );\n const safeCurrentCount = Number.isFinite(extractedCount) ? extractedCount : 1;\n const nextCount = safeCurrentCount + 1;\n const baseLabel = previousToolText || indicator.text;\n next[next.length - 1] = {\n ...last,\n toolLabel: baseLabel,\n toolCount: nextCount,\n text: `${baseLabel} (${nextCount})`,\n };\n continue;\n }\n\n const id = nextMessageIdRef.current++;\n next.push({\n id,\n text: indicator.text,\n isUser: false,\n type: \"tool\",\n toolKind: indicator.kind,\n toolLabel: indicator.text,\n toolCount: 1,\n });\n }\n\n return next;\n });\n };\n\n const updateMessageText = (id: number, text: string) => {\n setMessages((previous) =>\n previous.map((message) =>\n message.id === id ? { ...message, text } : message,\n ),\n );\n };\n\n const upsertLiveUserTranscription = (text: string) => {\n const normalized = text.trim();\n if (!normalized) {\n return;\n }\n liveTranscriptionTextRef.current = normalized;\n if (liveTranscriptionMessageIdRef.current === null) {\n liveTranscriptionMessageIdRef.current = appendMessage(normalized, true);\n return;\n }\n updateMessageText(liveTranscriptionMessageIdRef.current, normalized);\n };\n\n const clearLiveUserTranscriptionState = () => {\n liveTranscriptionMessageIdRef.current = null;\n liveTranscriptionTextRef.current = \"\";\n };\n\n const handleAudioStateChange = (state: AudioStreamState, requestEpoch?: number) => {\n if (typeof requestEpoch === \"number\" && !isCurrentRequestEpoch(requestEpoch)) {\n return;\n }\n\n if (state === \"rendering\") {\n setIsRenderingAudio(true);\n setIsPlayingAudio(false);\n return;\n }\n\n if (state === \"playing\") {\n setIsRenderingAudio(false);\n setIsPlayingAudio(true);\n return;\n }\n\n if (state === \"fallback\") {\n setIsRenderingAudio(true);\n setIsPlayingAudio(false);\n return;\n }\n\n if (state === \"done\") {\n setIsRenderingAudio(false);\n setIsPlayingAudio(false);\n return;\n }\n\n setIsRenderingAudio(false);\n setIsPlayingAudio(false);\n };\n\n const finalizeStreamCycle = (requestEpoch?: number) => {\n if (typeof requestEpoch === \"number\" && !isCurrentRequestEpoch(requestEpoch)) {\n return;\n }\n\n awaitingAssistantResponseRef.current = false;\n setStatusOverride(null);\n setIsBusy(false);\n isBusyRef.current = false;\n setIsTranscribing(false);\n setIsThinking(false);\n setIsRenderingAudio(false);\n setIsPlayingAudio(false);\n setIsRunningTools(false);\n pendingUserTextRef.current = null;\n pendingAssistantTextRef.current = \"\";\n assistantMessageIdRef.current = null;\n if (activeStreamControllerRef.current) {\n activeStreamControllerRef.current = null;\n }\n if (\n !autoListenSuppressedRef.current &&\n shouldAutoListenAfterAudio(\n accessibilityMode,\n expectsReplyRef.current,\n isRecordingRef.current,\n false,\n )\n ) {\n console.info(\"[Bulut] chat-window auto-listen trigger after stream completion\");\n void startRecording(\"vad\");\n }\n // Reset for next turn\n expectsReplyRef.current = true;\n };\n\n const runAgentForUserText = async (userText: string) => {\n if (!config.projectId) {\n appendMessage(\"Hata: Project ID yapılandırılmamış.\", false);\n return;\n }\n\n const normalizedUserText = userText.trim();\n if (!normalizedUserText) {\n appendMessage(\"Ses kaydı metne dönüştürülemedi. Lütfen tekrar deneyin.\", false);\n return;\n }\n\n const requestEpoch = beginRequestEpoch();\n setIsBusy(true);\n isBusyRef.current = true;\n setIsTranscribing(false);\n setIsThinking(true);\n setIsRenderingAudio(false);\n setIsPlayingAudio(false);\n setIsRunningTools(false);\n setStatusOverride(STATUS_LABELS.thinking);\n awaitingAssistantResponseRef.current = true;\n\n try {\n pendingUserTextRef.current = normalizedUserText;\n upsertLiveUserTranscription(normalizedUserText);\n clearLiveUserTranscriptionState();\n\n stopActiveStream();\n const pageContext = getPageContext().summary;\n\n const handleToolExecution = async (\n call: AgentToolCallInfo,\n ): Promise<{ call_id: string; result: string }> => {\n const toolCall: ToolCallWithId = {\n tool: call.tool as\n | \"navigate\"\n | \"getPageContext\"\n | \"interact\"\n | \"scroll\",\n call_id: call.call_id,\n ...call.args,\n } as ToolCallWithId;\n return executeSingleToolCall(toolCall);\n };\n\n const controller = agentTextChatStream(\n config.backendBaseUrl,\n normalizedUserText,\n config.projectId,\n sessionIdRef.current,\n {\n model: config.model,\n voice: config.voice,\n pageContext,\n accessibilityMode,\n },\n {\n onSessionId: (sid) => {\n if (!isCurrentRequestEpoch(requestEpoch)) return;\n if (sid && sid !== sessionIdRef.current) {\n sessionIdRef.current = sid;\n if (typeof localStorage !== \"undefined\") {\n localStorage.setItem(SESSION_ID_KEY, sid);\n }\n }\n },\n onAssistantDelta: (delta) => {\n if (!isCurrentRequestEpoch(requestEpoch)) return;\n setIsTranscribing(false);\n setIsThinking(true);\n setIsRunningTools(false);\n if (awaitingAssistantResponseRef.current) {\n awaitingAssistantResponseRef.current = false;\n setStatusOverride(null);\n }\n\n pendingAssistantTextRef.current += delta;\n\n if (assistantMessageIdRef.current === null) {\n assistantMessageIdRef.current = appendMessage(\n pendingAssistantTextRef.current,\n false,\n );\n } else {\n updateMessageText(\n assistantMessageIdRef.current,\n pendingAssistantTextRef.current,\n );\n }\n },\n onAssistantDone: (assistantText, expectsReply) => {\n if (!isCurrentRequestEpoch(requestEpoch)) return;\n playSfx(\"completed\");\n expectsReplyRef.current = expectsReply !== false;\n awaitingAssistantResponseRef.current = false;\n setStatusOverride(null);\n setIsThinking(false);\n setIsRenderingAudio(true);\n\n const finalDisplayText = assistantText || pendingAssistantTextRef.current;\n pendingAssistantTextRef.current = finalDisplayText;\n\n if (assistantMessageIdRef.current !== null) {\n updateMessageText(\n assistantMessageIdRef.current,\n finalDisplayText,\n );\n } else {\n assistantMessageIdRef.current = appendMessage(\n finalDisplayText,\n false,\n );\n }\n },\n onIntermediateReply: (text) => {\n if (!isCurrentRequestEpoch(requestEpoch)) return;\n void speakText(\n config.backendBaseUrl, text, config.voice,\n accessibilityMode, (state) => handleAudioStateChange(state, requestEpoch),\n ).catch((err) => console.warn(\"[Bulut] intermediate TTS failed\", err));\n },\n onToolCalls: (calls) => {\n if (!isCurrentRequestEpoch(requestEpoch)) return;\n if (calls.length > 0) {\n playSfx(\"toolCall\");\n }\n setIsRunningTools(true);\n setStatusOverride(STATUS_LABELS.runningTools);\n appendToolIndicatorMessages(calls);\n\n assistantMessageIdRef.current = null;\n pendingAssistantTextRef.current = \"\";\n },\n onToolResult: () => {},\n onIteration: () => {\n if (!isCurrentRequestEpoch(requestEpoch)) return;\n playSfx(\"thinking\");\n setIsThinking(true);\n setStatusOverride(STATUS_LABELS.thinking);\n },\n onAudioStateChange: (state) => {\n handleAudioStateChange(state, requestEpoch);\n },\n onError: (err) => {\n if (!isCurrentRequestEpoch(requestEpoch)) return;\n awaitingAssistantResponseRef.current = false;\n setStatusOverride(null);\n appendMessage(`Hata: ${err}`, false);\n },\n },\n handleToolExecution,\n );\n\n activeStreamControllerRef.current = controller;\n await controller.done;\n } catch (error) {\n if (!isCurrentRequestEpoch(requestEpoch)) return;\n awaitingAssistantResponseRef.current = false;\n setStatusOverride(null);\n if (error instanceof Error) {\n appendMessage(`Hata: ${error.message}`, false);\n }\n } finally {\n finalizeStreamCycle(requestEpoch);\n }\n };\n\n const handleAudioBlob = async (blob: Blob) => {\n if (!config.projectId) {\n appendMessage(\"Hata: Project ID yapılandırılmamış.\", false);\n return;\n }\n\n const requestEpoch = beginRequestEpoch();\n setIsBusy(true);\n isBusyRef.current = true;\n setIsTranscribing(true);\n setIsThinking(false);\n setIsRenderingAudio(false);\n setIsPlayingAudio(false);\n setIsRunningTools(false);\n resetProcessingFlags(); // Start fresh\n setStatusOverride(STATUS_LABELS.thinking);\n awaitingAssistantResponseRef.current = true;\n\n try {\n const fileType = blob.type || \"audio/webm\";\n const extension = fileType.includes(\"ogg\")\n ? \"ogg\"\n : fileType.includes(\"wav\")\n ? \"wav\"\n : fileType.includes(\"mpeg\") || fileType.includes(\"mp3\")\n ? \"mp3\"\n : \"webm\";\n const file = new File([blob], `voice-input.${extension}`, {\n type: fileType,\n });\n\n stopActiveStream();\n\n const pageContext = getPageContext().summary;\n\n // Helper: bridge an AgentToolCallInfo to a ToolCallWithId\n const handleToolExecution = async (\n call: AgentToolCallInfo,\n ): Promise<{ call_id: string; result: string }> => {\n const toolCall: ToolCallWithId = {\n tool: call.tool as\n | \"navigate\"\n | \"getPageContext\"\n | \"interact\"\n | \"scroll\",\n call_id: call.call_id,\n ...call.args,\n } as ToolCallWithId;\n return executeSingleToolCall(toolCall);\n };\n\n const controller = agentVoiceChatStream(\n config.backendBaseUrl,\n file,\n config.projectId,\n sessionIdRef.current,\n {\n model: config.model,\n voice: config.voice,\n pageContext,\n accessibilityMode,\n },\n {\n onSttRequestSent: () => {\n if (!isCurrentRequestEpoch(requestEpoch)) return;\n playSttSentCueOnce();\n },\n onTranscription: (data) => {\n if (!isCurrentRequestEpoch(requestEpoch)) return;\n if (data.session_id && data.session_id !== sessionIdRef.current) {\n sessionIdRef.current = data.session_id;\n if (typeof localStorage !== \"undefined\") {\n localStorage.setItem(SESSION_ID_KEY, data.session_id);\n }\n }\n\n const normalized = data.user_text.trim();\n if (normalized) {\n const previousUserText = pendingUserTextRef.current;\n pendingUserTextRef.current = normalized;\n if (liveTranscriptionMessageIdRef.current !== null) {\n updateMessageText(liveTranscriptionMessageIdRef.current, normalized);\n clearLiveUserTranscriptionState();\n } else if (previousUserText !== normalized) {\n appendMessage(normalized, true);\n }\n }\n\n setIsTranscribing(false);\n setIsThinking(true);\n setStatusOverride(STATUS_LABELS.thinking);\n },\n onSessionId: (sid) => {\n if (!isCurrentRequestEpoch(requestEpoch)) return;\n if (sid && sid !== sessionIdRef.current) {\n sessionIdRef.current = sid;\n if (typeof localStorage !== \"undefined\") {\n localStorage.setItem(SESSION_ID_KEY, sid);\n }\n }\n },\n onAssistantDelta: (delta) => {\n if (!isCurrentRequestEpoch(requestEpoch)) return;\n setIsTranscribing(false);\n setIsThinking(true);\n setIsRunningTools(false);\n if (awaitingAssistantResponseRef.current) {\n awaitingAssistantResponseRef.current = false;\n setStatusOverride(null);\n }\n\n // Agent returns plain text (not JSON), stream it directly\n pendingAssistantTextRef.current += delta;\n\n if (assistantMessageIdRef.current === null) {\n assistantMessageIdRef.current = appendMessage(\n pendingAssistantTextRef.current,\n false,\n );\n } else {\n updateMessageText(\n assistantMessageIdRef.current,\n pendingAssistantTextRef.current,\n );\n }\n },\n onAssistantDone: (assistantText, expectsReply) => {\n if (!isCurrentRequestEpoch(requestEpoch)) return;\n playSfx(\"completed\");\n expectsReplyRef.current = expectsReply !== false;\n awaitingAssistantResponseRef.current = false;\n setStatusOverride(null);\n setIsThinking(false);\n setIsRenderingAudio(true);\n\n const finalDisplayText =\n assistantText || pendingAssistantTextRef.current;\n\n streamingJsonParserRef.current = null;\n pendingAssistantTextRef.current = finalDisplayText;\n\n if (assistantMessageIdRef.current !== null) {\n updateMessageText(\n assistantMessageIdRef.current,\n finalDisplayText,\n );\n } else {\n assistantMessageIdRef.current = appendMessage(\n finalDisplayText,\n false,\n );\n }\n },\n onIntermediateReply: (text) => {\n if (!isCurrentRequestEpoch(requestEpoch)) return;\n void speakText(\n config.backendBaseUrl, text, config.voice,\n accessibilityMode, (state) => handleAudioStateChange(state, requestEpoch),\n ).catch((err) => console.warn(\"[Bulut] intermediate TTS failed\", err));\n },\n onToolCalls: (calls) => {\n if (!isCurrentRequestEpoch(requestEpoch)) return;\n if (calls.length > 0) {\n playSfx(\"toolCall\");\n }\n setIsRunningTools(true);\n setStatusOverride(STATUS_LABELS.runningTools);\n appendToolIndicatorMessages(calls);\n assistantMessageIdRef.current = null;\n pendingAssistantTextRef.current = \"\";\n },\n onToolResult: (_callId, _toolName, _result) => {\n // Tool result sent back to agent automatically.\n // Could display detailed result here if needed.\n },\n onIteration: (_iteration, _maxIterations) => {\n if (!isCurrentRequestEpoch(requestEpoch)) return;\n // Agent started a new reasoning iteration\n playSfx(\"thinking\");\n setIsThinking(true);\n setStatusOverride(STATUS_LABELS.thinking);\n },\n onAudioStateChange: (state) => {\n handleAudioStateChange(state, requestEpoch);\n },\n onError: (err) => {\n if (!isCurrentRequestEpoch(requestEpoch)) return;\n awaitingAssistantResponseRef.current = false;\n setStatusOverride(null);\n appendMessage(`Hata: ${err}`, false);\n },\n },\n handleToolExecution,\n );\n\n activeStreamControllerRef.current = controller;\n await controller.done;\n\n // Safety: Ensure messages are flushed if not already\n // (e.g. if audio 'done' event didn't fire for some reason or there was no audio)\n /* if (pendingUserTextRef.current || pendingAssistantTextRef.current) {\n // flushPendingMessages(); // Removed as we stream now\n } */\n } catch (error) {\n if (!isCurrentRequestEpoch(requestEpoch)) return;\n // Error already shown via onError callback — don't duplicate\n awaitingAssistantResponseRef.current = false;\n setStatusOverride(null);\n } finally {\n finalizeStreamCycle(requestEpoch);\n }\n };\n\n const stopRecording = (options?: { discard?: boolean }) => {\n const recorder = recorderRef.current;\n if (!recorder || recorder.state === \"inactive\") {\n return;\n }\n\n if (options?.discard) {\n discardNextRecordingRef.current = true;\n }\n\n cleanupVAD();\n recorder.stop();\n };\n\n const setupVAD = (stream: MediaStream, recorder: MediaRecorder) => {\n const AudioCtx =\n window.AudioContext ||\n (window as Window & { webkitAudioContext?: typeof AudioContext })\n .webkitAudioContext;\n\n if (!AudioCtx) {\n return;\n }\n\n const context = new AudioCtx();\n audioContextRef.current = context;\n\n const analyser = context.createAnalyser();\n analyser.fftSize = 256;\n\n const source = context.createMediaStreamSource(stream);\n sourceRef.current = source;\n source.connect(analyser);\n\n const dataArray = new Uint8Array(analyser.frequencyBinCount);\n silenceStartRef.current = null;\n let speechDetected = false;\n let speechStartedAt: number | null = null;\n const enforceMinSpeechDuration = accessibilityMode;\n\n vadIntervalRef.current = window.setInterval(() => {\n if (!isRecordingRef.current || recorder.state === \"inactive\") {\n cleanupVAD();\n return;\n }\n\n analyser.getByteFrequencyData(dataArray);\n\n let sum = 0;\n for (const value of dataArray) {\n sum += value;\n }\n const average = sum / dataArray.length;\n const volume = average / 255;\n\n if (volume < VAD_THRESHOLD) {\n if (!speechDetected) {\n speechStartedAt = null;\n silenceStartRef.current = null;\n return;\n }\n\n if (silenceStartRef.current === null) {\n silenceStartRef.current = Date.now();\n return;\n }\n\n const silenceDuration = Date.now() - silenceStartRef.current;\n if (speechDetected && silenceDuration > SILENCE_DURATION_MS) {\n stopRecording();\n }\n return;\n }\n\n silenceStartRef.current = null;\n if (speechStartedAt === null) {\n speechStartedAt = Date.now();\n }\n\n if (!speechDetected) {\n const speechDuration = Date.now() - speechStartedAt;\n if (shouldAcceptVadSpeech(speechDuration, enforceMinSpeechDuration)) {\n speechDetected = true;\n if (enforceMinSpeechDuration) {\n setStatusOverride(STATUS_LABELS.listening);\n }\n }\n }\n }, 50);\n };\n\n const startRecording = async (mode: RecordingMode) => {\n if (\n isBusyRef.current ||\n isRecordingRef.current ||\n startRecordingPendingRef.current\n ) {\n return;\n }\n\n setStatusOverride(STATUS_LABELS.micInitializing);\n setIsMicPending(true);\n\n if (!navigator.mediaDevices?.getUserMedia) {\n setStatusOverride(null);\n setIsMicPending(false);\n appendMessage(\"Bu tarayıcıda mikrofon kullanılamıyor.\", false);\n return;\n }\n\n if (typeof MediaRecorder === \"undefined\") {\n setStatusOverride(null);\n setIsMicPending(false);\n appendMessage(\"Bu tarayıcıda MediaRecorder desteklenmiyor.\", false);\n return;\n }\n\n startRecordingPendingRef.current = true;\n\n try {\n const stream = await navigator.mediaDevices.getUserMedia({ audio: true });\n streamRef.current = stream;\n\n // Use low bitrate for speech — opus handles voice well at 16-24 kbps\n // and produces ~4-5x smaller payloads, speeding up the upload to fal.\n const recorderOptions: MediaRecorderOptions = {\n audioBitsPerSecond: 16_000,\n };\n\n // Prefer opus-in-ogg (smaller container) → opus-in-webm → browser default\n const preferredMimeTypes = [\n \"audio/ogg;codecs=opus\",\n \"audio/webm;codecs=opus\",\n \"audio/webm\",\n ];\n for (const mime of preferredMimeTypes) {\n if (MediaRecorder.isTypeSupported(mime)) {\n recorderOptions.mimeType = mime;\n break;\n }\n }\n\n const recorder = new MediaRecorder(stream, recorderOptions);\n recorderRef.current = recorder;\n audioChunksRef.current = [];\n clearLiveUserTranscriptionState();\n sttSendCuePlayedRef.current = false;\n\n const sttMimeType = (recorder.mimeType || recorderOptions.mimeType || \"audio/webm\")\n .split(\";\")[0]\n .trim() || \"audio/webm\";\n\n const sttWsController = startSttWebSocketStream(\n config.backendBaseUrl,\n {\n projectId: config.projectId,\n sessionId: sessionIdRef.current,\n language: \"tr\",\n mimeType: sttMimeType,\n },\n {\n onSessionId: (sid) => {\n if (!sid || sid === sessionIdRef.current) {\n return;\n }\n sessionIdRef.current = sid;\n if (typeof localStorage !== \"undefined\") {\n localStorage.setItem(SESSION_ID_KEY, sid);\n }\n },\n onPartial: ({ text }) => {\n if (!text.trim()) {\n return;\n }\n upsertLiveUserTranscription(text);\n },\n },\n );\n activeSttWsRef.current = sttWsController;\n\n recorder.ondataavailable = (event) => {\n if (event.data.size > 0) {\n audioChunksRef.current.push(event.data);\n if (activeSttWsRef.current) {\n void activeSttWsRef.current.pushChunk(event.data).catch((error) => {\n console.warn(\n `[Bulut] STT WS chunk send failed: ${error instanceof Error ? error.message : String(error)}`,\n );\n });\n }\n }\n };\n\n recorder.onerror = () => {\n appendMessage(\"Mikrofon kaydı sırasında bir hata oluştu.\", false);\n };\n\n recorder.onstop = async () => {\n setIsRecording(false);\n isRecordingRef.current = false;\n recordingModeRef.current = null;\n stopRecordingTimer();\n\n cleanupVAD();\n stopStreamTracks();\n\n const shouldDiscard = discardNextRecordingRef.current;\n discardNextRecordingRef.current = false;\n\n const blob = new Blob(audioChunksRef.current, {\n type: recorder.mimeType || \"audio/webm\",\n });\n audioChunksRef.current = [];\n\n const currentSttWs = activeSttWsRef.current;\n activeSttWsRef.current = null;\n\n if (shouldDiscard) {\n currentSttWs?.cancel();\n clearLiveUserTranscriptionState();\n setStatusOverride(null);\n return;\n }\n\n if (blob.size === 0) {\n currentSttWs?.cancel();\n clearLiveUserTranscriptionState();\n setStatusOverride(null);\n appendMessage(\"Ses kaydı alınamadı. Lütfen tekrar deneyin.\", false);\n return;\n }\n\n setIsTranscribing(true);\n setStatusOverride(STATUS_LABELS.transcribing);\n\n try {\n if (currentSttWs) {\n playSttSentCueOnce();\n const sttResult = await currentSttWs.stop();\n if (sttResult.session_id && sttResult.session_id !== sessionIdRef.current) {\n sessionIdRef.current = sttResult.session_id;\n if (typeof localStorage !== \"undefined\") {\n localStorage.setItem(SESSION_ID_KEY, sttResult.session_id);\n }\n }\n if (sttResult.text.trim()) {\n upsertLiveUserTranscription(sttResult.text);\n setStatusOverride(STATUS_LABELS.thinking);\n await runAgentForUserText(sttResult.text);\n return;\n }\n }\n } catch (error) {\n console.warn(\n `[Bulut] STT WS finalization failed, falling back to HTTP POST /chat/stt: ${error instanceof Error ? error.message : String(error)}`,\n );\n } finally {\n clearLiveUserTranscriptionState();\n }\n\n console.info(\"[Bulut] Using HTTP POST fallback for STT (streaming WS did not succeed)\");\n setStatusOverride(STATUS_LABELS.thinking);\n await handleAudioBlob(blob);\n };\n\n if (mode === \"vad\") {\n setupVAD(stream, recorder);\n }\n\n recorder.start(200);\n recordingModeRef.current = mode;\n setIsRecording(true);\n isRecordingRef.current = true;\n startRecordingTimer();\n\n setStatusOverride(\n accessibilityMode && mode === \"vad\"\n ? STATUS_LABELS.accessibilityActive\n : STATUS_LABELS.listening,\n );\n\n if (pendingStopAfterStartRef.current) {\n pendingStopAfterStartRef.current = false;\n stopRecording();\n }\n } catch (error) {\n const errMsg = normalizeError(error);\n if (errMsg.toLowerCase().includes(\"permission\") || errMsg.toLowerCase().includes(\"denied\")) {\n autoListenSuppressedRef.current = true;\n }\n cancelActiveSttWs();\n setStatusOverride(null);\n appendMessage(`Mikrofon hatası: ${errMsg}`, false);\n cleanupVAD();\n stopStreamTracks();\n pendingStopAfterStartRef.current = false;\n setIsRecording(false);\n isRecordingRef.current = false;\n stopRecordingTimer();\n } finally {\n if (!isRecordingRef.current && !isBusyRef.current) {\n setStatusOverride(null);\n }\n startRecordingPendingRef.current = false;\n setIsMicPending(false);\n }\n };\n\n const resetMicGesture = () => {\n micPressStartRef.current = null;\n micHoldTriggeredRef.current = false;\n clearMicHoldTimeout();\n };\n\n const handleMicPointerDown = (\n event: JSX.TargetedPointerEvent<HTMLButtonElement>,\n ) => {\n event.preventDefault();\n\n if (isBusyRef.current) {\n return;\n }\n\n if (isRecordingRef.current) {\n // In VAD mode, tapping the button cancels; in press mode, it sends\n if (recordingModeRef.current === \"vad\") {\n stopRecording({ discard: true });\n } else {\n stopRecording();\n }\n return;\n }\n\n micPressStartRef.current = Date.now();\n micHoldTriggeredRef.current = false;\n clearMicHoldTimeout();\n\n if (event.currentTarget.setPointerCapture) {\n try {\n event.currentTarget.setPointerCapture(event.pointerId);\n } catch {\n // No-op.\n }\n }\n\n micHoldTimeoutRef.current = window.setTimeout(() => {\n if (\n micPressStartRef.current === null ||\n isBusyRef.current ||\n isRecordingRef.current\n ) {\n return;\n }\n\n micHoldTriggeredRef.current = true;\n void startRecording(\"press\");\n }, HOLD_THRESHOLD_MS);\n };\n\n const handleMicPointerUp = (\n event: JSX.TargetedPointerEvent<HTMLButtonElement>,\n ) => {\n event.preventDefault();\n\n if (event.currentTarget.releasePointerCapture) {\n try {\n event.currentTarget.releasePointerCapture(event.pointerId);\n } catch {\n // No-op.\n }\n }\n\n const startedAt = micPressStartRef.current;\n const wasHold = micHoldTriggeredRef.current;\n resetMicGesture();\n\n if (startedAt === null) {\n return;\n }\n\n if (wasHold) {\n if (isRecordingRef.current) {\n stopRecording();\n } else if (startRecordingPendingRef.current) {\n pendingStopAfterStartRef.current = true;\n }\n return;\n }\n\n const duration = Date.now() - startedAt;\n if (classifyMicGesture(duration) === \"tap\") {\n void startRecording(\"vad\");\n }\n };\n\n const handleMicPointerCancel = (\n event: JSX.TargetedPointerEvent<HTMLButtonElement>,\n ) => {\n handleMicPointerUp(event);\n };\n\n const handleRestart = () => {\n invalidateRequestEpoch();\n sttSendCuePlayedRef.current = false;\n resetMicGesture();\n pendingStopAfterStartRef.current = false;\n\n stopActiveStream();\n stopActiveAudioPlayback();\n cancelActiveSttWs();\n\n if (recorderRef.current && recorderRef.current.state !== \"inactive\") {\n stopRecording({ discard: true });\n } else {\n discardNextRecordingRef.current = false;\n cleanupVAD();\n stopStreamTracks();\n }\n\n stopRecordingTimer();\n setRecordingDurationMs(0);\n\n clearPersistedChatState(\n typeof localStorage !== \"undefined\" ? localStorage : null,\n );\n\n sessionIdRef.current = null;\n const initialMessages = createInitialMessages(config.agentName);\n nextMessageIdRef.current = getNextMessageId(initialMessages);\n setMessages(initialMessages);\n\n setIsBusy(false);\n isBusyRef.current = false;\n setIsRecording(false);\n isRecordingRef.current = false;\n resetProcessingFlags();\n };\n\n // Auto-listen when accessibility mode is activated (initial trigger)\n useEffect(() => {\n if (!accessibilityMode || autoListenSuppressedRef.current) return;\n const timer = window.setTimeout(() => {\n if (!isRecordingRef.current && !isBusyRef.current && !startRecordingPendingRef.current && !autoListenSuppressedRef.current) {\n void startRecording(\"vad\");\n }\n }, 500);\n return () => window.clearTimeout(timer);\n }, [accessibilityMode]); // eslint-disable-line react-hooks/exhaustive-deps\n\n const stopTask = () => {\n invalidateRequestEpoch();\n sttSendCuePlayedRef.current = false;\n stopActiveStream();\n stopActiveAudioPlayback();\n cancelActiveSttWs();\n stopRecording({ discard: true });\n cleanupVAD();\n stopStreamTracks();\n resetProcessingFlags();\n setIsBusy(false);\n isBusyRef.current = false;\n };\n\n // Expose recording actions to parent via actionsRef\n if (actionsRef) {\n actionsRef.current = {\n startRecording: () => {\n autoListenSuppressedRef.current = false;\n void startRecording(\"vad\");\n },\n cancelRecording: () => {\n stopActiveAudioPlayback();\n cancelActiveSttWs();\n const recorder = recorderRef.current;\n if (recorder && recorder.state !== \"inactive\") {\n stopRecording({ discard: true });\n } else {\n cleanupVAD();\n stopStreamTracks();\n }\n },\n stopTask,\n };\n }\n\n const windowStyle: { [key: string]: string } = {\n position: \"fixed\",\n bottom: `${POSITION_BOTTOM}px`,\n right: `${POSITION_RIGHT}px`,\n width: `${WINDOW_WIDTH}px`,\n maxHeight: `${WINDOW_HEIGHT}px`,\n backgroundColor: \"hsla(0, 0%, 100%, 1)\",\n borderRadius: BORDER_RADIUS.window,\n display: hidden ? \"none\" : \"flex\",\n flexDirection: \"column\",\n overflow: \"hidden\",\n zIndex: \"10000\",\n animation: hidden ? \"none\" : `slideIn ${TRANSITIONS.medium}`,\n boxShadow: accessibilityMode\n ? `inset 0 0 0 2px ${COLORS.primary}, ${SHADOW}`\n : SHADOW,\n fontFamily: \"\\\"Geist\\\", sans-serif\",\n };\n\n const headerStyle: { [key: string]: string } = {\n padding: \"14px 16px\",\n display: \"flex\",\n justifyContent: \"space-between\",\n alignItems: \"center\",\n };\n\n const headerActionsStyle: { [key: string]: string } = {\n display: \"flex\",\n alignItems: \"center\",\n gap: \"6px\",\n };\n\n const headerButtonStyle: { [key: string]: string } = {\n background: \"none\",\n border: \"none\",\n cursor: \"pointer\",\n padding: \"4px\",\n borderRadius: \"6px\",\n color: COLORS.text,\n display: \"flex\",\n alignItems: \"center\",\n justifyContent: \"center\",\n transition: `color ${TRANSITIONS.fast}, background-color ${TRANSITIONS.fast}`,\n };\n\n const messagesContainerStyle: { [key: string]: string } = {\n padding: \"0px 16px\",\n overflowY: \"auto\",\n flex: \"1\",\n minHeight: \"0\",\n };\n\n const messagesListStyle: { [key: string]: string } = {\n display: \"flex\",\n flexDirection: \"column\",\n gap: \"16px\",\n };\n\n const messageStyle = (isUser: boolean): JSX.CSSProperties => ({\n maxWidth: \"84%\",\n padding: isUser ? \"9px 14px\" : \"0px 0px\",\n borderRadius: BORDER_RADIUS.message,\n fontSize: \"14px\",\n lineHeight: \"140%\",\n wordWrap: \"break-word\",\n whiteSpace: \"pre-wrap\",\n alignSelf: isUser ? \"flex-end\" : \"flex-start\",\n backgroundColor: isUser ? COLORS.messageUser : \"\",\n color: isUser ? COLORS.messageUserText : \"hsla(215, 100%, 5%, 1)\",\n });\n\n const resolveToolIconSrc = (kind: Message[\"toolKind\"]): string => {\n if (kind === \"cursor\") {\n return cursorArrowRaysIconContent;\n }\n if (kind === \"scroll\") {\n return handRaisedIconContent;\n }\n if (kind === \"navigate\") {\n return mapIconContent;\n }\n if (kind === \"form\") {\n return queueListIconContent;\n }\n if (kind === \"interact\") {\n return handRaisedIconContent;\n }\n if (kind === \"unknown\") {\n return commandLineIconContent;\n }\n return faceSmileIconContent;\n };\n\n const footerStyle: { [key: string]: string } = {\n padding: \"10px 12px\",\n display: \"flex\",\n alignItems: \"center\",\n justifyContent: \"space-between\",\n gap: \"8px\",\n };\n\n const statusPanelStyle: { [key: string]: string } = {\n flex: \"1\",\n minHeight: \"34px\",\n borderRadius: \"10px\",\n color: COLORS.text,\n fontSize: \"14px\",\n display: \"flex\",\n alignItems: \"center\",\n padding: \"0 10px\",\n whiteSpace: \"nowrap\",\n overflow: \"hidden\",\n textOverflow: \"ellipsis\",\n opacity: \"0.7\",\n };\n\n const footerActionsStyle: { [key: string]: string } = {\n display: \"flex\",\n alignItems: \"center\",\n gap: \"8px\",\n flexShrink: \"0\",\n };\n\n const recordingTimerStyle: { [key: string]: string } = {\n minWidth: \"46px\",\n fontSize: \"12px\",\n fontWeight: \"700\",\n color: COLORS.text,\n textAlign: \"right\",\n };\n\n const micFooterButtonStyle: { [key: string]: string } = {\n width: \"37px\",\n height: \"37px\",\n borderRadius: \"999px\",\n background: \"transparent\",\n display: \"flex\",\n alignItems: \"center\",\n justifyContent: \"center\",\n cursor: \"pointer\",\n color: \"#ffffff\",\n border: \"1px solid hsla(215, 100%, 5%, 0.5)\",\n transition: `transform ${TRANSITIONS.fast}`,\n };\n\n const isVadRecording = isRecording && recordingModeRef.current === \"vad\";\n const showStopButton = isBusy && !isRecording;\n const hideMicButton = isMicPending && !isRecording;\n const disableMicControl = isBusy;\n\n return (\n <div className=\"bulut-chat-window\" style={windowStyle}>\n <style>{`\n @keyframes slideIn {\n from {\n opacity: 0;\n transform: translateY(10px);\n }\n to {\n opacity: 1;\n transform: translateY(0);\n }\n }\n\n .bulut-header-btn:hover:not(:disabled) {\n color: ${COLORS.text};\n }\n\n .bulut-footer-btn:hover:not(:disabled) {\n transform: scale(1.04);\n }\n\n .bulut-header-btn:disabled,\n .bulut-footer-btn:disabled {\n cursor: not-allowed;\n opacity: 0.5;\n transform: none;\n }\n\n @keyframes bulut-dots {\n 0% { content: '.'; }\n 33% { content: '..'; }\n 66% { content: '...'; }\n }\n\n .bulut-status-dots::after {\n content: '.';\n animation: bulut-dots 1.2s steps(1) infinite;\n display: inline-block;\n min-width: 12px;\n text-align: left;\n }\n\n /* Mobile: full-screen chat window */\n @media (max-width: 600px) {\n .bulut-chat-window {\n top: 0 !important;\n left: 0 !important;\n right: 0 !important;\n bottom: 0 !important;\n width: 100% !important;\n max-height: 100% !important;\n height: 100% !important;\n border-radius: 0 !important;\n }\n .bulut-close-btn {\n width: 40px !important;\n height: 40px !important;\n padding: 8px !important;\n }\n .bulut-close-btn svg {\n width: 26px !important;\n height: 26px !important;\n }\n }\n `}</style>\n\n <div style={headerStyle}>\n <SvgIcon\n src={logoContent}\n title=\"Bulut Logo\"\n style={{ maxWidth: \"80px\", height: \"auto\" }}\n stripColors={false}\n />\n <div style={headerActionsStyle}>\n <button\n type=\"button\"\n className=\"bulut-header-btn\"\n style={headerButtonStyle}\n onClick={handleRestart}\n aria-label=\"Sohbeti yeniden başlat\"\n title=\"Sohbeti yeniden başlat\"\n >\n <SvgIcon src={arrowPathIconContent} aria-hidden=\"true\" width={22} height={22} />\n </button>\n\n <button\n type=\"button\"\n className=\"bulut-header-btn bulut-close-btn\"\n style={{\n ...headerButtonStyle,\n display: \"flex\",\n alignItems: \"center\",\n justifyContent: \"center\",\n }}\n onClick={onClose}\n aria-label=\"Sohbeti kapat\"\n title=\"Sohbeti kapat\"\n >\n <SvgIcon src={xMarkIconContent} aria-hidden=\"true\" width={22} height={22} />\n </button>\n </div>\n </div>\n\n <div style={messagesContainerStyle} ref={messagesContainerRef}>\n <div style={messagesListStyle} ref={messagesContentRef}>\n {messages.map((message) => {\n if (message.type === \"tool\") {\n const toolIconSrc = resolveToolIconSrc(message.toolKind);\n return (\n <div\n key={message.id}\n style={{\n ...messageStyle(false),\n opacity: \"0.7\",\n display: \"flex\",\n alignItems: \"center\",\n gap: \"8px\",\n }}\n >\n <SvgIcon\n src={toolIconSrc}\n aria-hidden=\"true\"\n width={20}\n height={20}\n style={{ flexShrink: 0 }}\n />\n <span>{message.text}</span>\n </div>\n );\n }\n\n return (\n <div key={message.id} style={messageStyle(message.isUser)}>\n {message.text}\n </div>\n );\n })}\n </div>\n </div>\n\n <div style={footerStyle}>\n <div style={{ ...statusPanelStyle, transition: \"opacity 0.2s ease-out\" }}>\n {showStatus ? (\n <span className=\"bulut-status-dots\" title={statusText}>\n {statusText}\n </span>\n ) : onAccessibilityToggle ? (\n <div\n style={{\n display: \"flex\",\n alignItems: \"center\",\n gap: \"8px\",\n }}\n >\n <span\n style={{\n fontSize: \"12px\",\n opacity: \"0.6\",\n whiteSpace: \"nowrap\",\n }}\n >\n Erişilebilirlik\n </span>\n <button\n type=\"button\"\n onClick={onAccessibilityToggle}\n aria-label={\n accessibilityMode\n ? \"Erişilebilirlik modunu kapat\"\n : \"Erişilebilirlik modunu aç\"\n }\n style={{\n width: \"36px\",\n height: \"20px\",\n borderRadius: \"10px\",\n border: \"none\",\n cursor: \"pointer\",\n padding: \"2px\",\n display: \"flex\",\n alignItems: \"center\",\n backgroundColor: accessibilityMode\n ? COLORS.primary\n : \"hsla(215, 10%, 75%, 1)\",\n transition: `background-color ${TRANSITIONS.fast}`,\n flexShrink: \"0\",\n }}\n >\n <span\n style={{\n width: \"16px\",\n height: \"16px\",\n borderRadius: \"50%\",\n backgroundColor: \"#ffffff\",\n display: \"block\",\n transition: `transform ${TRANSITIONS.fast}`,\n transform: accessibilityMode\n ? \"translateX(16px)\"\n : \"translateX(0)\",\n boxShadow: \"0 1px 3px rgba(0,0,0,0.2)\",\n }}\n />\n </button>\n </div>\n ) : null}\n </div>\n\n <div style={footerActionsStyle}>\n {isRecording ? (\n <span style={recordingTimerStyle}>\n {formatDurationMs(recordingDurationMs)}\n </span>\n ) : null}\n {showStopButton ? (\n <button\n type=\"button\"\n className=\"bulut-footer-btn\"\n style={micFooterButtonStyle}\n onClick={stopTask}\n aria-label=\"Görevi durdur\"\n title=\"Görevi durdur\"\n >\n <SvgIcon\n src={stopOutlineIconContent}\n aria-hidden=\"true\"\n width={22}\n height={22}\n style={{ color: \"hsla(215, 100%, 5%, 1)\" }}\n />\n </button>\n ) : hideMicButton ? null : (\n <button\n type=\"button\"\n className=\"bulut-footer-btn\"\n style={micFooterButtonStyle}\n onPointerDown={handleMicPointerDown}\n onPointerUp={handleMicPointerUp}\n onPointerCancel={handleMicPointerCancel}\n disabled={disableMicControl}\n aria-label={isVadRecording ? \"Kaydı iptal et\" : isRecording ? \"Kaydı durdur\" : \"Kaydı başlat\"}\n title={\n isVadRecording\n ? \"Kaydı iptal et\"\n : isRecording\n ? \"Kaydı durdur\"\n : \"Dokun: VAD, Basılı tut: bırakınca gönder\"\n }\n >\n {isVadRecording ? (\n <SvgIcon\n src={xMarkIconContent}\n aria-hidden=\"true\"\n width={22}\n height={22}\n style={{ color: \"hsla(215, 100%, 5%, 1)\"}}\n />\n ) : (\n <SvgIcon\n src={microphoneOutlineIconContent}\n aria-hidden=\"true\"\n width={22}\n height={22}\n style={{ color: \"hsla(215, 100%, 5%, 1)\" }}\n />\n )}\n </button>\n )}\n </div>\n </div>\n </div>\n );\n};\n","export default \"<svg width=\\\"2420\\\" height=\\\"438\\\" viewBox=\\\"0 0 2420 438\\\" fill=\\\"none\\\" xmlns=\\\"http://www.w3.org/2000/svg\\\">\\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\\\"/>\\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\\\"/>\\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\\\"/>\\n<path d=\\\"M1734.32 424.955H1641.59V10.7793H1734.32V424.955Z\\\" fill=\\\"#000B1A\\\"/>\\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\\\"/>\\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\\\"/>\\n</svg>\\n\"","export default \"<svg xmlns=\\\"http://www.w3.org/2000/svg\\\" fill=\\\"none\\\" viewBox=\\\"0 0 24 24\\\" stroke-width=\\\"1.5\\\" stroke=\\\"currentColor\\\">\\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\\\" />\\n</svg>\\n\"","export default \"<svg xmlns=\\\"http://www.w3.org/2000/svg\\\" fill=\\\"none\\\" viewBox=\\\"0 0 24 24\\\" stroke-width=\\\"1.5\\\" stroke=\\\"currentColor\\\">\\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\\\" />\\n</svg>\\n\"","export default \"<svg xmlns=\\\"http://www.w3.org/2000/svg\\\" fill=\\\"none\\\" viewBox=\\\"0 0 24 24\\\" stroke-width=\\\"1.5\\\" stroke=\\\"currentColor\\\">\\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\\\" />\\n</svg>\\n\"","export default \"<svg xmlns=\\\"http://www.w3.org/2000/svg\\\" fill=\\\"none\\\" viewBox=\\\"0 0 24 24\\\" stroke-width=\\\"1.5\\\" stroke=\\\"currentColor\\\">\\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\\\" />\\n</svg>\\n\"","export default \"<svg xmlns=\\\"http://www.w3.org/2000/svg\\\" fill=\\\"none\\\" viewBox=\\\"0 0 24 24\\\" stroke-width=\\\"1.5\\\" stroke=\\\"currentColor\\\">\\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\\\" />\\n</svg>\\n\"","export default \"<svg xmlns=\\\"http://www.w3.org/2000/svg\\\" fill=\\\"none\\\" viewBox=\\\"0 0 24 24\\\" stroke-width=\\\"1.5\\\" stroke=\\\"currentColor\\\">\\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\\\" />\\n</svg>\\n\"","import { useState, useEffect, useRef, useCallback } from \"preact/hooks\";\nimport \"./globals.css\";\nimport { render } from \"preact\";\nimport { ChatButton } from \"./components/ChatButton\";\nimport {\n ChatWindow,\n type ChatWindowHandle,\n} from \"./components/ChatWindow\";\nimport { COLORS } from \"./styles/constants\";\n\nexport type BulutVoice = \"alloy\" | \"zeynep\" | \"ali\";\n\nexport interface BulutOptions {\n containerId?: string;\n backendBaseUrl?: string;\n projectId?: string;\n}\n\nexport interface BulutRuntimeConfig {\n backendBaseUrl: string;\n projectId: string;\n model: string;\n voice: BulutVoice;\n baseColor: string;\n agentName: string;\n}\n\n/** Default LLM model — keep in sync with backend config.DEFAULT_LLM_MODEL */\nconst DEFAULT_LLM_MODEL = \"x-ai/grok-4.1-fast\";\n\nconst DEFAULT_AGENT_NAME = \"Bulut\";\n\nconst DEFAULT_CONFIG: BulutRuntimeConfig = {\n backendBaseUrl: \"https://api.bulut.lu\",\n projectId: \"\", // Must be provided\n model: DEFAULT_LLM_MODEL,\n voice: \"alloy\",\n baseColor: COLORS.primary,\n agentName: DEFAULT_AGENT_NAME,\n};\n\nconst isValidHexColor = (value: string): boolean =>\n /^#([0-9a-fA-F]{3}|[0-9a-fA-F]{6})$/.test(value);\n\nconst normalizeHexColor = (value: string): string => {\n const trimmed = value.trim();\n if (!isValidHexColor(trimmed)) {\n return DEFAULT_CONFIG.baseColor;\n }\n if (trimmed.length === 4) {\n const r = trimmed[1];\n const g = trimmed[2];\n const b = trimmed[3];\n return `#${r}${r}${g}${g}${b}${b}`.toLowerCase();\n }\n return trimmed.toLowerCase();\n};\n\nconst shadeHexColor = (hexColor: string, amount: number): string => {\n const normalized = normalizeHexColor(hexColor);\n const raw = normalized.slice(1);\n const toChannel = (start: number): number => parseInt(raw.slice(start, start + 2), 16);\n const clamp = (value: number): number => Math.max(0, Math.min(255, Math.round(value)));\n const adjust = (channel: number): number =>\n amount < 0 ? channel * (1 + amount) : channel + (255 - channel) * amount;\n const toHex = (channel: number): string => clamp(channel).toString(16).padStart(2, \"0\");\n\n const r = adjust(toChannel(0));\n const g = adjust(toChannel(2));\n const b = adjust(toChannel(4));\n\n return `#${toHex(r)}${toHex(g)}${toHex(b)}`;\n};\n\nconst applyTheme = (baseColor: string): void => {\n const normalized = normalizeHexColor(baseColor);\n COLORS.primary = normalized;\n COLORS.primaryHover = shadeHexColor(normalized, -0.15);\n COLORS.messageUser = normalized;\n};\n\ninterface RemoteProjectConfig {\n base_color: string;\n model: string;\n agent_name: string;\n voice: string;\n}\n\nconst fetchRemoteConfig = async (\n baseUrl: string,\n projectId: string,\n): Promise<RemoteProjectConfig | null> => {\n try {\n const url = baseUrl.replace(/\\/+$/, \"\");\n const res = await fetch(`${url}/projects/${projectId}/config`);\n if (!res.ok) return null;\n return (await res.json()) as RemoteProjectConfig;\n } catch {\n return null;\n }\n};\n\nconst resolveRuntimeConfig = (\n options: BulutOptions,\n): BulutRuntimeConfig => ({\n backendBaseUrl: options.backendBaseUrl || DEFAULT_CONFIG.backendBaseUrl,\n projectId: options.projectId || DEFAULT_CONFIG.projectId,\n model: DEFAULT_CONFIG.model,\n voice: DEFAULT_CONFIG.voice,\n baseColor: DEFAULT_CONFIG.baseColor,\n agentName: DEFAULT_CONFIG.agentName,\n});\n\ninterface BulutWidgetProps {\n config: BulutRuntimeConfig;\n}\n\nconst ACCESSIBILITY_MODE_KEY = \"bulut_accessibility_mode_enabled\";\nconst GEIST_FONT_FAMILY = \"Geist\";\nconst GEIST_STYLESHEET_ID = \"bulut-geist-font-stylesheet\";\nconst GEIST_STYLESHEET_URL =\n \"https://fonts.googleapis.com/css2?family=Geist:wght@100..900&display=swap\";\n\nconst ensureGeistStylesheet = (): void => {\n if (typeof document === \"undefined\") {\n return;\n }\n if (document.getElementById(GEIST_STYLESHEET_ID)) {\n return;\n }\n const link = document.createElement(\"link\");\n link.id = GEIST_STYLESHEET_ID;\n link.rel = \"stylesheet\";\n link.href = GEIST_STYLESHEET_URL;\n document.head.appendChild(link);\n};\n\nconst BulutWidget = ({ config }: BulutWidgetProps) => {\n // Live config that merges remote settings over initial config\n const [liveConfig, setLiveConfig] = useState<BulutRuntimeConfig>(config);\n const [configReady, setConfigReady] = useState(false);\n\n // Fetch remote project config on mount — widget stays hidden until done\n useEffect(() => {\n if (!config.projectId) {\n setConfigReady(true);\n return;\n }\n let cancelled = false;\n\n fetchRemoteConfig(config.backendBaseUrl, config.projectId).then((remote) => {\n if (cancelled) return;\n if (remote) {\n const merged: BulutRuntimeConfig = {\n ...config,\n baseColor: normalizeHexColor(remote.base_color || config.baseColor),\n model: remote.model || config.model,\n agentName: remote.agent_name || config.agentName,\n voice: (\n remote.voice === \"alloy\" || remote.voice === \"zeynep\" || remote.voice === \"ali\"\n ? remote.voice\n : config.voice\n ) as BulutVoice,\n };\n applyTheme(merged.baseColor);\n setLiveConfig(merged);\n }\n setConfigReady(true);\n });\n\n return () => { cancelled = true; };\n }, [config]);\n\n // Check localStorage for persisted state\n const [isOpen, setIsOpen] = useState(() => {\n if (typeof localStorage !== \"undefined\") {\n return localStorage.getItem(\"bulut_panel_open\") === \"true\";\n }\n return false;\n });\n\n const [showBubble, setShowBubble] = useState(false);\n const [isAccessibilityEnabled, setIsAccessibilityEnabled] = useState(() => {\n if (typeof localStorage === \"undefined\") {\n return false;\n }\n return localStorage.getItem(ACCESSIBILITY_MODE_KEY) === \"true\";\n });\n\n // State reported by ChatWindow\n const [isRecording, setIsRecording] = useState(false);\n const [isBusy, setIsBusy] = useState(false);\n const [previewMessage, setPreviewMessage] = useState<string | null>(null);\n const [previewDismissed, setPreviewDismissed] = useState(false);\n\n // Ref for delegating recording to ChatWindow\n const chatActionsRef = useRef<ChatWindowHandle | null>(null);\n\n const handlePreviewChange = useCallback((text: string | null) => {\n setPreviewMessage(text);\n if (text !== null) setPreviewDismissed(false);\n }, []);\n\n // Show welcome bubble once for 5 seconds\n useEffect(() => {\n if (isAccessibilityEnabled) {\n setShowBubble(false);\n return;\n }\n if (isOpen) return;\n if (typeof localStorage !== \"undefined\") {\n if (localStorage.getItem(\"bulut_bubble_shown\") === \"true\") return;\n }\n\n setShowBubble(true);\n const timer = setTimeout(() => {\n setShowBubble(false);\n if (typeof localStorage !== \"undefined\") {\n localStorage.setItem(\"bulut_bubble_shown\", \"true\");\n }\n }, 5000);\n return () => clearTimeout(timer);\n }, [isOpen, isAccessibilityEnabled]);\n\n const toggleWidget = () => {\n const newState = !isOpen;\n setIsOpen(newState);\n setShowBubble(false);\n if (typeof localStorage !== \"undefined\") {\n localStorage.setItem(\"bulut_panel_open\", String(newState));\n }\n };\n\n const toggleAccessibilityMode = () => {\n const next = !isAccessibilityEnabled;\n setIsAccessibilityEnabled(next);\n if (typeof localStorage !== \"undefined\") {\n localStorage.setItem(ACCESSIBILITY_MODE_KEY, String(next));\n }\n console.info(`[Bulut] accessibility mode toggled enabled=${next}`);\n };\n\n const handleClose = () => {\n setIsOpen(false);\n if (typeof localStorage !== \"undefined\") {\n localStorage.setItem(\"bulut_panel_open\", \"false\");\n }\n console.info(\"Bulut chat window closed.\");\n };\n\n // Close on escape key\n useEffect(() => {\n const handleEscape = (e: KeyboardEvent) => {\n if (e.key === \"Escape\" && isOpen) {\n setIsOpen(false);\n if (typeof localStorage !== \"undefined\") {\n localStorage.setItem(\"bulut_panel_open\", \"false\");\n }\n }\n };\n\n document.addEventListener(\"keydown\", handleEscape);\n return () => document.removeEventListener(\"keydown\", handleEscape);\n }, [isOpen]);\n\n if (!configReady) return null;\n\n return (\n <>\n {!isOpen && (\n <ChatButton\n onMicClick={() => chatActionsRef.current?.startRecording()}\n onCancelRecording={() => chatActionsRef.current?.cancelRecording()}\n onStopTask={() => chatActionsRef.current?.stopTask()}\n isRecording={isRecording}\n isBusy={isBusy}\n accessibilityMode={isAccessibilityEnabled}\n showBubble={showBubble}\n onBubbleClick={() => {\n setShowBubble(false);\n toggleWidget();\n }}\n previewMessage={previewDismissed ? null : previewMessage}\n onPreviewClick={() => toggleWidget()}\n onPreviewClose={() => setPreviewDismissed(true)}\n />\n )}\n <ChatWindow\n onClose={handleClose}\n config={liveConfig}\n accessibilityMode={isAccessibilityEnabled}\n onAccessibilityToggle={toggleAccessibilityMode}\n hidden={!isOpen}\n actionsRef={chatActionsRef}\n onRecordingChange={setIsRecording}\n onBusyChange={setIsBusy}\n onPreviewChange={handlePreviewChange}\n />\n </>\n );\n};\n\nconst SHADOW_STYLE = `\n :host {\n all: initial;\n contain: layout style paint;\n font-family: \"${GEIST_FONT_FAMILY}\", sans-serif;\n color: hsla(215, 100%, 5%, 1);\n font-size: 16px;\n line-height: 1.4;\n -webkit-font-smoothing: antialiased;\n text-rendering: optimizeLegibility;\n }\n\n #bulut-shadow-mount {\n all: initial;\n color: inherit;\n font-family: \"${GEIST_FONT_FAMILY}\", sans-serif;\n font-size: inherit;\n line-height: inherit;\n }\n\n #bulut-shadow-mount * {\n font-family: \"${GEIST_FONT_FAMILY}\", sans-serif !important;\n color: inherit;\n }\n\n #bulut-shadow-mount *, #bulut-shadow-mount *::before, #bulut-shadow-mount *::after {\n box-sizing: border-box;\n }\n`;\n\n// Container host for the widget\nlet widgetContainer: HTMLElement | null = null;\nlet widgetMountNode: HTMLElement | null = null;\nlet createdContainer = false;\nlet isInitialized = false;\n\n/**\n * Initialize the Bulut chat widget\n * @param options - Optional configuration options\n */\nexport const init = (options: BulutOptions = {}) => {\n if (isInitialized) {\n console.warn(\"Bulut is already initialized\");\n return;\n }\n\n ensureGeistStylesheet();\n\n const runtimeConfig = resolveRuntimeConfig(options);\n applyTheme(runtimeConfig.baseColor);\n\n // Create or find container\n if (options.containerId) {\n widgetContainer = document.getElementById(options.containerId);\n createdContainer = false;\n } else {\n widgetContainer = document.createElement(\"div\");\n widgetContainer.id = \"bulut-container\";\n document.body.appendChild(widgetContainer);\n createdContainer = true;\n }\n\n if (!widgetContainer) {\n console.error(\"Bulut: Container not found\");\n return;\n }\n\n const shadowRoot = widgetContainer.shadowRoot || widgetContainer.attachShadow({ mode: \"open\" });\n shadowRoot.replaceChildren();\n\n const style = document.createElement(\"style\");\n style.textContent = SHADOW_STYLE;\n\n const mountNode = document.createElement(\"div\");\n mountNode.id = \"bulut-shadow-mount\";\n\n shadowRoot.append(style, mountNode);\n widgetMountNode = mountNode;\n\n // Render the widget\n render(<BulutWidget config={runtimeConfig} />, mountNode);\n isInitialized = true;\n\n console.log(\"Bulut initialized successfully\");\n};\n\n/**\n * Destroy the Bulut widget\n */\nexport const destroy = () => {\n if (!isInitialized) {\n return;\n }\n\n if (widgetMountNode) {\n render(null, widgetMountNode);\n widgetMountNode = null;\n }\n\n if (widgetContainer && createdContainer) {\n document.body.removeChild(widgetContainer);\n }\n\n widgetContainer = null;\n createdContainer = false;\n isInitialized = false;\n console.log(\"Bulut destroyed\");\n};\n\n/**\n * Check if the widget is initialized\n */\nexport const isReady = () => isInitialized;\n\nconst Bulut = {\n init,\n destroy,\n isReady,\n};\n\nif (typeof window !== \"undefined\") {\n (window as Window & { Bulut?: typeof Bulut }).Bulut = Bulut;\n}\n\n// Export the main widget component for advanced usage\nexport { BulutWidget };\n\n// Export components for custom implementations\nexport { ChatButton, ChatWindow };\n\nexport default Bulut;\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","volume","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":"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","x_google_ignoreList":[0,1,2]}
|
|
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 elementMap: Map<number, Element>;\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\ninterface InteractableCandidate {\r\n id: number;\r\n line: string;\r\n score: number;\r\n order: number;\r\n element: Element;\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} from \"./contextConfig\";\r\n\r\nexport const PAGE_CONTEXT_CACHE_VERSION = 4;\r\nexport const PAGE_CONTEXT_CACHE_KEY = \"auticbot_page_context_cache_v4\";\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\n/** SVG drawing primitives — never useful as standalone interactables. */\r\nconst SVG_INTERNAL_TAGS = new Set([\r\n \"path\",\r\n \"circle\",\r\n \"line\",\r\n \"rect\",\r\n \"polygon\",\r\n \"polyline\",\r\n \"ellipse\",\r\n \"g\",\r\n \"use\",\r\n \"defs\",\r\n \"clippath\",\r\n \"mask\",\r\n \"symbol\",\r\n \"lineargradient\",\r\n \"radialgradient\",\r\n \"stop\",\r\n \"text\",\r\n \"tspan\",\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 pageContextCache = new Map<string, CachedPageContextEntry>();\r\nlet cacheHydrated = false;\r\n\r\n/**\r\n * Live element map — maps numeric IDs to DOM elements.\r\n * Rebuilt on every page context scan; NOT serialised to cache.\r\n */\r\nlet liveElementMap = new Map<number, Element>();\r\n\r\n/** Look up a DOM element by its semantic-map ID. */\r\nexport const getElementById = (id: number): Element | undefined =>\r\n liveElementMap.get(id);\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 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 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\n/**\r\n * Returns true if the element is nested inside an interactive parent\r\n * (e.g. a `<span>` or `<img>` inside a `<button>` or `<a>`).\r\n * This prevents listing child fragments of an already-listed interactable.\r\n */\r\nconst hasInteractiveAncestor = (element: Element): boolean => {\r\n let parent = element.parentElement;\r\n while (parent && parent !== document.body) {\r\n const parentTag = parent.tagName.toLowerCase();\r\n if (NATIVE_INTERACTIVE_TAGS.has(parentTag)) return true;\r\n const parentRole = getPrimaryRole(parent);\r\n if (INTERACTIVE_ROLES.has(parentRole)) return true;\r\n parent = parent.parentElement;\r\n }\r\n return false;\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 getElementLabel = (element: Element): string => {\r\n const text = normalizeWhitespace(\r\n (element instanceof HTMLElement ? element.innerText : element.textContent) ||\r\n \"\",\r\n ).substring(0, 80);\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 label =\r\n text || ariaLabel || title || placeholder || value || name || \"\";\r\n\r\n const tag = element.tagName.toLowerCase();\r\n\r\n // Images: prefer alt text, fall back to src filename\r\n if (tag === \"img\") {\r\n const alt = normalizeWhitespace(element.getAttribute(\"alt\") || \"\");\r\n if (alt) return alt;\r\n const src = element.getAttribute(\"src\") || \"\";\r\n const filename = src.split(\"/\").pop()?.split(\"?\")[0] || \"\";\r\n return filename ? `img: ${filename}` : compactOuterHtml(element);\r\n }\r\n\r\n if (tag === \"svg\") {\r\n return ariaLabel || title || \"icon\";\r\n }\r\n\r\n if (tag === \"input\") {\r\n const inputType = element.getAttribute(\"type\") || \"text\";\r\n const currentValue = element instanceof HTMLInputElement ? element.value : \"\";\r\n const valueNote = currentValue ? ` val=\"${currentValue.substring(0, 40)}\"` : \"\";\r\n return `${inputType} ${label || \"input\"}${valueNote}`;\r\n }\r\n\r\n if (tag === \"textarea\") {\r\n const currentValue = element instanceof HTMLTextAreaElement ? element.value : \"\";\r\n const valueNote = currentValue ? ` val=\"${currentValue.substring(0, 40)}\"` : \"\";\r\n return `textarea ${label || \"textarea\"}${valueNote}`;\r\n }\r\n\r\n if (tag === \"select\") {\r\n const selectEl = element as HTMLSelectElement;\r\n const selectedText = selectEl.selectedOptions?.[0]?.textContent?.trim() || \"\";\r\n const valueNote = selectedText ? ` val=\"${selectedText}\"` : \"\";\r\n return `select ${label || \"select\"}${valueNote}`;\r\n }\r\n\r\n if (label) return label;\r\n\r\n // Fallback: compact outerHTML snippet so the agent can still identify the element\r\n return compactOuterHtml(element);\r\n};\r\n\r\n/** Return a trimmed, single-line outerHTML (opening tag + short text), max 90 chars. */\r\nconst compactOuterHtml = (element: Element): string => {\r\n const html = element.outerHTML || \"\";\r\n // Take only the opening tag + a little content\r\n const closeIdx = html.indexOf(\">\");\r\n if (closeIdx === -1) return html.substring(0, 90);\r\n const snippet = html.substring(0, Math.min(closeIdx + 30, 90)).replace(/\\s+/g, \" \").trim();\r\n return snippet || \"untitled\";\r\n};\r\n\r\n// ── Semantic element description ────────────────────────────────────\r\n\r\nconst describeElementType = (element: Element): string => {\r\n const tag = element.tagName.toLowerCase();\r\n const role = getPrimaryRole(element);\r\n\r\n // Role-based override\r\n if (role === \"button\" || tag === \"button\") return \"Button\";\r\n if (role === \"link\" || tag === \"a\") return \"Link\";\r\n if (role === \"tab\") return \"Tab\";\r\n if (role === \"menuitem\") return \"MenuItem\";\r\n if (role === \"checkbox\" || (tag === \"input\" && element.getAttribute(\"type\") === \"checkbox\")) return \"Checkbox\";\r\n if (role === \"radio\" || (tag === \"input\" && element.getAttribute(\"type\") === \"radio\")) return \"Radio\";\r\n if (role === \"switch\") return \"Switch\";\r\n if (role === \"combobox\" || tag === \"select\") return \"Select\";\r\n if (role === \"textbox\" || tag === \"textarea\") return \"TextArea\";\r\n if (role === \"searchbox\") return \"SearchBox\";\r\n if (role === \"slider\" || (tag === \"input\" && element.getAttribute(\"type\") === \"range\")) return \"Slider\";\r\n if (role === \"spinbutton\") return \"SpinButton\";\r\n if (role === \"option\" || tag === \"option\") return \"Option\";\r\n if (role === \"treeitem\") return \"TreeItem\";\r\n if (tag === \"input\") {\r\n const t = element.getAttribute(\"type\") || \"text\";\r\n return `Input[${t}]`;\r\n }\r\n if (tag === \"summary\") return \"Summary\";\r\n if (tag === \"details\") return \"Details\";\r\n if (element.getAttribute(\"contenteditable\") === \"true\") return \"Editable\";\r\n\r\n // Generic interactive\r\n return role ? `${role[0].toUpperCase()}${role.slice(1)}` : `<${tag}>`;\r\n};\r\n\r\nconst getElementState = (element: Element): string[] => {\r\n const states: string[] = [];\r\n\r\n const pressed = element.getAttribute(\"aria-pressed\");\r\n if (pressed === \"true\") states.push(\"Pressed\");\r\n else if (pressed === \"false\") states.push(\"Not pressed\");\r\n\r\n const expanded = element.getAttribute(\"aria-expanded\");\r\n if (expanded === \"true\") states.push(\"Expanded\");\r\n else if (expanded === \"false\") states.push(\"Collapsed\");\r\n\r\n const selected = element.getAttribute(\"aria-selected\");\r\n if (selected === \"true\") states.push(\"Selected\");\r\n\r\n if (element instanceof HTMLInputElement) {\r\n if (element.type === \"checkbox\" || element.type === \"radio\") {\r\n states.push(element.checked ? \"Checked\" : \"Unchecked\");\r\n }\r\n }\r\n\r\n if (element.hasAttribute(\"disabled\") || element.getAttribute(\"aria-disabled\") === \"true\") {\r\n states.push(\"Disabled\");\r\n }\r\n\r\n return states;\r\n};\r\n\r\ninterface SemanticScanResult {\r\n links: string[];\r\n interactables: string[];\r\n elementMap: Map<number, Element>;\r\n}\r\n\r\n/**\r\n * Scan the DOM and build a semantic element map.\r\n *\r\n * Every interactive element gets a numeric ID. The LLM uses these IDs\r\n * with `interact(id=N)` instead of fragile CSS selectors.\r\n */\r\nconst collectSemanticElements = (): SemanticScanResult => {\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 candidates: InteractableCandidate[] = [];\r\n const elementMap = new Map<number, Element>();\r\n let idCounter = 1;\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)) continue;\r\n if (SVG_INTERNAL_TAGS.has(tag)) continue;\r\n if (!isVisible(element)) continue;\r\n\r\n const role = getPrimaryRole(element);\r\n const style = window.getComputedStyle(element);\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 tabIndex = parseTabIndex(element.getAttribute(\"tabindex\"));\r\n const hasTabStop = tabIndex !== null && tabIndex >= 0;\r\n const hasPointerCursor = style.cursor === \"pointer\";\r\n const isContentEditable = element.getAttribute(\"contenteditable\") === \"true\";\r\n const isDisabled =\r\n element.hasAttribute(\"disabled\") ||\r\n element.getAttribute(\"aria-disabled\") === \"true\";\r\n\r\n // ── Links ───────────────────────────────────────────────────\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 id = idCounter++;\r\n const line = `- [${id}] ${label} -> ${absoluteHref}`;\r\n\r\n if (!linkSet.has(absoluteHref)) {\r\n linkSet.add(absoluteHref);\r\n links.push(line);\r\n elementMap.set(id, element);\r\n }\r\n }\r\n\r\n // ── Interactables ───────────────────────────────────────────\r\n const hasInteractionSignals =\r\n isNativeInteractive ||\r\n isRoleInteractive ||\r\n isContentEditable ||\r\n hasTabStop ||\r\n hasPointerCursor;\r\n\r\n if (!hasInteractionSignals || isDisabled) continue;\r\n\r\n // Skip children nested inside an already-interactive parent\r\n // (e.g. <span> or <img> inside a <button> or <a>)\r\n if (hasInteractiveAncestor(element)) continue;\r\n\r\n const id = idCounter++;\r\n elementMap.set(id, element);\r\n\r\n const elType = describeElementType(element);\r\n const label = getElementLabel(element);\r\n const stateTokens = getElementState(element);\r\n const statePart = stateTokens.length > 0 ? ` (${stateTokens.join(\", \")})` : \"\";\r\n const line = `- [${id}] ${elType}: \"${label}\"${statePart}`;\r\n\r\n const score =\r\n (isNativeInteractive ? 5 : 0) +\r\n (isRoleInteractive ? 4 : 0) +\r\n (hasTabStop ? 2 : 0) +\r\n (hasPointerCursor ? 2 : 0) +\r\n (isContentEditable ? 2 : 0);\r\n\r\n candidates.push({ id, line, score, order, element });\r\n }\r\n\r\n const interactables = candidates\r\n .sort((a, b) => b.score - a.score || a.order - b.order)\r\n .slice(0, MAX_INTERACTABLES)\r\n .map((c) => c.line);\r\n\r\n return {\r\n links: links.slice(0, MAX_LINKS),\r\n interactables,\r\n elementMap,\r\n };\r\n};\r\n\r\nconst TEXT_CONTENT_SELECTOR = [\r\n \"p\", \"li\", \"h1\", \"h2\", \"h3\", \"h4\", \"h5\", \"h6\",\r\n \"blockquote\", \"figcaption\", \"dd\", \"dt\", \"td\", \"th\",\r\n \"pre\", \"label\", \"caption\",\r\n].join(\", \");\r\n\r\n/**\r\n * Check if an element has meaningful direct text content\r\n * (text nodes that aren't just whitespace).\r\n */\r\nconst hasDirectText = (element: Element): boolean => {\r\n for (const child of Array.from(element.childNodes)) {\r\n if (child.nodeType === Node.TEXT_NODE) {\r\n const trimmed = (child.textContent || \"\").trim();\r\n if (trimmed.length >= 10) return true;\r\n }\r\n }\r\n return false;\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 addSnippet = (raw: string): boolean => {\r\n if (!raw || raw.length < 15) return false;\r\n const text = raw.length > 300 ? raw.substring(0, 300) + \"…\" : raw;\r\n if (seen.has(text)) return false;\r\n seen.add(text);\r\n snippets.push(`- ${text}`);\r\n return snippets.length >= MAX_TEXT_SNIPPETS;\r\n };\r\n\r\n // Pass 1: semantic text elements (p, li, headings, etc.)\r\n const candidates = Array.from(root.querySelectorAll(TEXT_CONTENT_SELECTOR));\r\n for (const node of candidates) {\r\n if (!isVisible(node)) continue;\r\n const raw = normalizeWhitespace(node.textContent || \"\");\r\n if (addSnippet(raw)) return snippets;\r\n }\r\n\r\n // Pass 2: generic containers (div, span, section, etc.) that hold\r\n // direct text not already captured by semantic tags above.\r\n const genericContainers = Array.from(\r\n root.querySelectorAll(\"div, span, section, article, aside, header, footer\"),\r\n );\r\n for (const node of genericContainers) {\r\n if (!isVisible(node)) continue;\r\n // Only include elements whose own direct child text nodes are meaningful,\r\n // to avoid duplicating text already captured from nested semantic tags.\r\n if (!hasDirectText(node)) continue;\r\n const raw = normalizeWhitespace(node.textContent || \"\");\r\n if (addSnippet(raw)) return snippets;\r\n }\r\n\r\n return snippets;\r\n};\r\n\r\nconst formatSection = (title: string, lines: string[]): string => {\r\n if (lines.length === 0) return `${title}:\\n- none`;\r\n return `${title}:\\n${lines.join(\"\\n\")}`;\r\n};\r\n\r\nexport const buildPageContextSummary = (\r\n url: string,\r\n title: string,\r\n lang: string,\r\n headings: string[],\r\n links: string[],\r\n interactables: string[],\r\n textSnippets: string[],\r\n): string => {\r\n const sections = [\r\n formatSection(\"Page\", [\r\n `- URL: ${url || \"unknown\"}`,\r\n `- Title: ${title || \"unknown\"}`,\r\n `- Lang: ${lang || \"unknown\"}`,\r\n ]),\r\n formatSection(\"Headings\", headings),\r\n formatSection(\"Content Snippets\", textSnippets),\r\n formatSection(\"Links\", links),\r\n formatSection(\"Interactive Elements\", interactables),\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 invalidateCurrentPageContext = (): void => {\r\n if (typeof window === \"undefined\") return;\r\n const url = canonicalUrl(window.location.href);\r\n pageContextCache.delete(url);\r\n persistCacheToStorage();\r\n};\r\n\r\nexport const getPageContext = (forceRefresh: boolean = false): PageContext => {\r\n if (typeof window === \"undefined\" || typeof document === \"undefined\") {\r\n return {\r\n links: [],\r\n interactables: [],\r\n summary: \"\",\r\n elementMap: new Map(),\r\n };\r\n }\r\n\r\n hydrateCacheFromStorage();\r\n const url = canonicalUrl(window.location.href);\r\n\r\n // Always rebuild the live element map (it holds DOM references)\r\n const scan = collectSemanticElements();\r\n liveElementMap = scan.elementMap;\r\n\r\n if (!forceRefresh) {\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 elementMap: liveElementMap,\r\n };\r\n }\r\n }\r\n\r\n console.info(`[Autic] context cache miss url=${url}`);\r\n\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 document.title,\r\n document.documentElement.lang,\r\n headings,\r\n scan.links,\r\n scan.interactables,\r\n collectTextSnippets(),\r\n );\r\n\r\n const entry: CachedPageContextEntry = {\r\n url,\r\n summary,\r\n links: scan.links,\r\n interactables: scan.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 elementMap: liveElementMap,\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 = 120;\r\n\r\n/** Maximum number of interactable elements to include. */\r\nexport const MAX_INTERACTABLES = 2000;\r\n\r\n/** Maximum number of headings (h1-h3) to include. */\r\nexport const MAX_HEADINGS = 100;\r\n\r\n/** Maximum number of main-content text snippets. */\r\nexport const MAX_TEXT_SNIPPETS = 30;\r\n\r\n/** Maximum number of pages kept in the in-memory context cache. */\r\nexport const MAX_CACHED_PAGES = 25;\r\n\r\n/** Maximum number of DOM elements scanned per page. */\r\nexport const MAX_PAGE_SCAN_ELEMENTS = 10000;\r\n","import { getPageContext, getElementById, invalidateCurrentPageContext } 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 = 5 * 60_000; // 5 minutes (matches chat history TTL)\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 accumulatedDelta?: 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 id?: number;\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 id?: number;\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 id: asNumber(item.id),\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 id = asNumber(item.id);\r\n const selector = asString(item.selector);\r\n if (!id && !selector) {\r\n continue;\r\n }\r\n\r\n toolCalls.push({\r\n tool: \"scroll\",\r\n id,\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 isAlreadyVisible = cursor.style.opacity === \"1\";\r\n\r\n if (!isAlreadyVisible) {\r\n // First appearance: start from agent window top-left\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\r\n // Animate from current position via CSS transition\r\n setCursorVisibility(cursor, true);\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 // 1. Try resolving by semantic ID from the element map\r\n if (typeof call.id === \"number\") {\r\n const mapped = getElementById(call.id);\r\n if (mapped instanceof HTMLElement) {\r\n const center = getElementCenter(mapped);\r\n return { element: mapped, x: center.x, y: center.y };\r\n }\r\n console.warn(`AuticBot interact: element id=${call.id} not found in map`);\r\n }\r\n\r\n // 2. Fallback to CSS selector\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 id, 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 =\r\n (typeof call.id === \"number\" ? getElementById(call.id) : null) ??\r\n (call.selector ? findElementBySelector(call.selector) : null);\r\n\r\n if (!(selected instanceof HTMLElement)) {\r\n console.warn(`AuticBot scroll: target not found (id=${call.id}, selector=${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 // Invalidate context cache — page state likely changed after click\r\n invalidateCurrentPageContext();\r\n return;\r\n }\r\n\r\n if (call.action === \"type\") {\r\n typeIntoElement(target.element, call.text ?? \"\");\r\n // Invalidate context cache — form state changed\r\n invalidateCurrentPageContext();\r\n return;\r\n }\r\n\r\n submitElement(target.element);\r\n // Invalidate context cache — form submission may change page\r\n invalidateCurrentPageContext();\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(true);\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 onFirstBatchReady?: (batch: TtsCollectResult) => 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 // Streaming: fire first batch of chunks to the caller early\r\n let firstBatchFired = false;\r\n let firstBatchTimer: ReturnType<typeof setTimeout> | null = null;\r\n const FIRST_BATCH_DELAY_MS = 250;\r\n\r\n const fireFirstBatch = () => {\r\n if (firstBatchFired || !onFirstBatchReady || chunks.length === 0) return;\r\n firstBatchFired = true;\r\n if (firstBatchTimer) { clearTimeout(firstBatchTimer); firstBatchTimer = null; }\r\n const batch = chunks.splice(0);\r\n onFirstBatchReady({ chunks: batch, mimeType, sampleRate });\r\n };\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\r\n // Schedule first-batch flush for streaming playback\r\n if (onFirstBatchReady && !firstBatchFired) {\r\n if (firstBatchTimer) clearTimeout(firstBatchTimer);\r\n firstBatchTimer = setTimeout(fireFirstBatch, FIRST_BATCH_DELAY_MS);\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 // Flush first batch if not yet fired (short text = all chunks in one batch)\r\n if (onFirstBatchReady && !firstBatchFired) {\r\n fireFirstBatch();\r\n }\r\n\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/**\r\n * Collect TTS audio via WebSocket (with SSE fallback) and play it.\r\n *\r\n * Uses streaming playback: the first batch of audio chunks starts\r\n * playing immediately while remaining chunks are still being received.\r\n * This dramatically reduces perceived latency, especially for longer\r\n * text where the backend pipelines synthesis sentence-by-sentence.\r\n */\r\nconst collectAndPlayTtsStreamed = async (\r\n baseUrl: string,\r\n text: string,\r\n voice: string,\r\n accessibilityMode: boolean,\r\n isStopped: () => boolean,\r\n setSocket: (socket: WebSocket | null) => void,\r\n setReader: (reader: ReadableStreamDefaultReader<Uint8Array> | undefined) => void,\r\n onAudioStateChange?: (state: AudioStreamState) => void,\r\n): Promise<void> => {\r\n const trimmed = text.trim();\r\n if (!trimmed) { onAudioStateChange?.(\"done\"); return; }\r\n\r\n const playbackGen = getAudioPlaybackGeneration();\r\n if (wasPlaybackStoppedAfter(playbackGen)) {\r\n onAudioStateChange?.(\"done\");\r\n return;\r\n }\r\n\r\n onAudioStateChange?.(\"rendering\");\r\n\r\n let firstBatchPlayPromise: Promise<void> | null = null;\r\n let ttsResult: TtsCollectResult;\r\n\r\n try {\r\n ttsResult = await collectTtsViaWebSocket(\r\n baseUrl, trimmed, voice, accessibilityMode,\r\n isStopped, setSocket,\r\n // Streaming: start playing the first batch immediately\r\n (firstBatch) => {\r\n if (wasPlaybackStoppedAfter(playbackGen) || isStopped()) return;\r\n if (firstBatch.chunks.length === 0) return;\r\n firstBatchPlayPromise = playBufferedAudio(\r\n firstBatch.chunks, firstBatch.mimeType, firstBatch.sampleRate,\r\n (state) => {\r\n // Forward playing state, but suppress \"done\" since more may follow\r\n if (state !== \"done\") onAudioStateChange?.(state);\r\n },\r\n ).catch(err => console.warn(\"[Bulut] first batch playback error\", err));\r\n },\r\n );\r\n } catch (wsErr) {\r\n // If first batch was already playing, accept partial audio\r\n if (firstBatchPlayPromise) {\r\n await firstBatchPlayPromise;\r\n onAudioStateChange?.(\"done\");\r\n return;\r\n }\r\n // Otherwise fall back to SSE (non-streaming, collect-then-play)\r\n if (isStopped()) { onAudioStateChange?.(\"done\"); return; }\r\n console.warn(\r\n `[Bulut] TTS WS failed, falling back to SSE: ${wsErr instanceof Error ? wsErr.message : String(wsErr)}`,\r\n );\r\n ttsResult = await collectTtsViaSse(\r\n baseUrl, trimmed, voice, accessibilityMode,\r\n isStopped, setReader,\r\n );\r\n }\r\n\r\n if (isStopped() || wasPlaybackStoppedAfter(playbackGen)) {\r\n onAudioStateChange?.(\"done\");\r\n return;\r\n }\r\n\r\n // Wait for first batch to finish playing\r\n if (firstBatchPlayPromise) {\r\n await firstBatchPlayPromise;\r\n }\r\n\r\n // Play remaining chunks (if any)\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-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 console.info(`[Bulut] speakText start (${trimmed.length} chars)`);\r\n await collectAndPlayTtsStreamed(\r\n baseUrl, trimmed, voice, accessibilityMode,\r\n () => false,\r\n () => {},\r\n () => {},\r\n onAudioStateChange,\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 let allIntermediateText = \"\";\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 allIntermediateText += accumulatedDelta;\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 accumulatedDelta: allIntermediateText || undefined,\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 hideAgentCursor();\r\n await collectAndPlayTtsStreamed(\r\n baseUrl, assistantText, config.voice, Boolean(config.accessibilityMode),\r\n () => isStopped,\r\n (socket) => { activeSocket = socket; },\r\n (reader) => { activeReader = reader; },\r\n events.onAudioStateChange,\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 let allIntermediateText = \"\";\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 allIntermediateText += accumulatedDelta;\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 accumulatedDelta: allIntermediateText || undefined,\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 hideAgentCursor();\r\n await collectAndPlayTtsStreamed(\r\n baseUrl, assistantText, config.voice, Boolean(config.accessibilityMode),\r\n () => isStopped,\r\n (socket) => { activeSocket = socket; },\r\n (reader) => { activeReader = reader; },\r\n events.onAudioStateChange,\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 let allIntermediateText = \"\";\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 accumulated_delta: resumeState.accumulatedDelta || \"\",\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 allIntermediateText += accumulatedDelta;\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 accumulatedDelta: allIntermediateText || undefined,\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 hideAgentCursor();\r\n await collectAndPlayTtsStreamed(\r\n baseUrl, assistantText, resumeState.voice, Boolean(resumeState.accessibilityMode),\r\n () => isStopped,\r\n (socket) => { activeSocket = socket; },\r\n (reader) => { activeReader = reader; },\r\n events.onAudioStateChange,\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\nconst SFX_VOLUME = 0.5;\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 audio.volume = SFX_VOLUME;\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","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 // Force-persist messages to localStorage immediately so they\r\n // survive a full-page navigate that may happen next frame.\r\n if (typeof localStorage !== \"undefined\") {\r\n try {\r\n localStorage.setItem(STORAGE_KEY, JSON.stringify(next));\r\n localStorage.setItem(TIMESTAMP_KEY, Date.now().toString());\r\n } catch { /* ignore full/blocked storage */ }\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 <div style={{ display: \"flex\", alignItems: \"center\", gap: \"6px\" }}>\r\n <SvgIcon\r\n src={logoContent}\r\n title=\"Bulut Logo\"\r\n style={{ width: \"80px\", minWidth: \"80px\", height: \"auto\", flexShrink: 0 }}\r\n stripColors={false}\r\n />\r\n <span\r\n style={{\r\n marginTop: \"7px\",\r\n fontSize: \"9px\",\r\n color: COLORS.textSecondary,\r\n opacity: 0.45,\r\n fontWeight: 400,\r\n letterSpacing: \"0.02em\",\r\n userSelect: \"none\",\r\n whiteSpace: \"nowrap\",\r\n alignSelf: \"flex-end\",\r\n }}\r\n >\r\n v{typeof __BULUT_VERSION__ !== \"undefined\" ? __BULUT_VERSION__ : \"\"}\r\n </span>\r\n </div>\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","textSecondary","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","SVG_INTERNAL_TAGS","NATIVE_INTERACTIVE_TAGS","INTERACTIVE_ROLES","pageContextCache","Map","cacheHydrated","liveElementMap","getElementById","id","get","normalizeWhitespace","canonicalUrl","rawUrl","URL","href","isCacheEntry","obj","url","summary","links","interactables","capturedAt","version","parseTabIndex","parsed","Number","parseInt","isNaN","getPrimaryRole","element","getAttribute","split","persistCacheToStorage","sessionStorage","serialized","JSON","stringify","from","values","setItem","error","console","warn","buildSummaryWithHistory","recentPages","entry","historySection","compactSummary","join","isVisible","HTMLElement","hidden","window","getComputedStyle","visibility","rect","getBoundingClientRect","hasInteractiveAncestor","parent","parentElement","body","parentTag","tagName","has","parentRole","toAbsoluteUrl","location","getElementLabel","innerText","textContent","substring","ariaLabel","placeholder","HTMLInputElement","HTMLTextAreaElement","HTMLButtonElement","label","tag","alt","filename","pop","compactOuterHtml","inputType","currentValue","selectEl","selectedText","selectedOptions","html","outerHTML","closeIdx","Math","min","describeElementType","role","toUpperCase","getElementState","states","pressed","expanded","checked","hasAttribute","TEXT_CONTENT_SELECTOR","hasDirectText","child","Node","TEXT_NODE","formatSection","lines","invalidateCurrentPageContext","delete","getPageContext","forceRefresh","elementMap","raw","getItem","parse","set","size","info","hydrateCacheFromStorage","scan","sampledElements","querySelectorAll","linkSet","candidates","idCounter","order","isNativeInteractive","Boolean","isRoleInteractive","tabIndex","hasTabStop","hasPointerCursor","isContentEditable","isDisabled","startsWith","absoluteHref","line","add","elType","stateTokens","score","collectSemanticElements","cached","headings","lang","textSnippets","buildPageContextSummary","root","querySelector","snippets","seen","addSnippet","node","genericContainers","collectTextSnippets","Date","now","sorted","overflow","pruneOldestCacheEntries","AGENT_CURSOR_ID","CURSOR_MOVE_DURATION_MS","CURSOR_EASING","RESUME_STORAGE_KEY","savePendingAgentResume","localStorage","savedAt","clearPendingAgentResume","removeItem","clamp","max","animateWindowScrollTo","async","targetY","durationMs","startY","scrollY","delta","abs","raf","callback","performance","startTime","step","progress","eased","cos","PI","easeInOutSine","scrollTo","setCursorPosition","left","setCursorVisibility","visible","opacity","isVisibleElement","findAgentUiAnchorElement","roots","defaultHost","shadowRoot","allElements","el","includes","getBulutShadowRoots","panel","button","getAgentWindowTopLeft","anchor","CURSOR_DIAMETER_PX","scrollX","hideAgentCursor","cursorHoverTrackingInitialized","initializeCursorHoverTracking","parseFloat","getCursorPosition","pointerX","pageX","pointerY","pageY","hypot","moveCursor","existing","createElement","startPosition","baseColor","boxShadow","boxSizing","pointerEvents","appendChild","ensureCursor","dataset","transitionReady","waitForNextAnimationFrame","getElementCenter","CONTAINS_SELECTOR_PATTERN","findElementBySelector","selector","containsMatch","baseSelector","expectedText","candidate","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","mapped","center","selected","resolveTarget","action","click","requestSubmit","form","parentForm","closest","submitElement","executeNavigate","targetUrl","resolvedUrl","matchingElement","parsedTarget","allLinks","HTMLAnchorElement","elUrl","pathname","search","hash","urlSegments","lastSegment","searchTerms","searchParams","entries","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","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","collectTtsViaWebSocket","assistantText","voice","isStopped","setSocket","onFirstBatchReady","requestId","crypto","randomUUID","random","highestSeqSeen","firstBatchFired","firstBatchTimer","fireFirstBatch","batch","splice","connectOnce","done","finalError","finalize","mode","request_id","accessibility_mode","last_seq","parseTtsWsEventPayload","incomingSeq","shouldAcceptAudioSeq","sample_rate","attempt","delay","collectAndPlayTtsStreamed","setReader","playbackGen","ttsResult","firstBatchPlayPromise","firstBatch","catch","wsErr","ttsFormData","FormData","append","ttsResponse","fetch","method","ok","reader","getReader","decoder","TextDecoder","read","decode","stream","blocks","block","format","releaseLock","collectTtsViaSse","speakText","agentVoiceChatStream","audioFile","executeTool","activeReader","activeSocket","errorEmitted","sttResult","file","formData","responsePromise","transcribeAudio","onSttRequestSent","currentSessionId","effectiveSessionId","userText","onTranscription","user_text","agentResolve","agentReject","finalReply","resolved","accumulatedDelta","allIntermediateText","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","SFX_SOURCES","sent","thinking","toolCall","completed","sfxManager","queue","playNow","volume","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","find","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","accumulated_delta","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","marginTop","letterSpacing","userSelect","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":"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","x_google_ignoreList":[0,1,2]}
|