sparkle-react 0.0.38 → 0.0.40

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/index.d.mts CHANGED
@@ -50,11 +50,19 @@ type RealtimeValue = string | number | boolean | {
50
50
 
51
51
  declare function useRealtime<T extends RealtimeValue>(key: string, defaultValue?: T): T[];
52
52
 
53
+ type UseAuthResult = {
54
+ /** Identifiant Twitch (`Account.accountId`), pas l’id interne Sparkle. */
55
+ userId: string | null | undefined;
56
+ /** `User.name` Prisma (nom affiché OAuth). */
57
+ username: string | null | undefined;
58
+ /** `User.image` Prisma (URL avatar). */
59
+ image: string | null | undefined;
60
+ };
53
61
  /**
54
- * Identifiant utilisateur Better Auth (`session.user.id`) de l’app principale.
55
- * `undefined` pendant le chargement, `null` si non connecté ou erreur, sinon l’id.
62
+ * Profil session navigateur : `userId` = id Twitch (`Account.accountId` OAuth) ;
63
+ * `username` / `image` depuis `User`. Champs à `undefined` pendant le chargement.
56
64
  */
57
- declare function useAuth(): string | null | undefined;
65
+ declare function useAuth(): UseAuthResult;
58
66
 
59
67
  /** Version allégée des hooks pour le prompt LLM (sans événements absents de EventTypeMap). */
60
68
  type CCL = "DrugsIntoxication" | "Gambling" | "MatureGame" | "ProfanityVulgarity" | "SexualThemes" | "ViolentGraphic";
@@ -347,4 +355,4 @@ declare function useTask(): {
347
355
  invokeUntil: (taskId: string, delaySeconds: number, payload?: unknown) => Promise<boolean>;
348
356
  };
349
357
 
350
- export { type SparkleHookKey, SparkleHookKeys, SparkleProvider, type SparkleProviderProps, useAuth, useHook, useObs, useObsCall, useRealtime, useTask, useTwitch };
358
+ export { type SparkleHookKey, SparkleHookKeys, SparkleProvider, type SparkleProviderProps, type UseAuthResult, useAuth, useHook, useObs, useObsCall, useRealtime, useTask, useTwitch };
package/dist/index.mjs CHANGED
@@ -1,2 +1,2 @@
1
- import h,{useMemo as re,useEffect as ne,useState as oe}from"react";import*as S from"react";import{createContext as z}from"react";var v=z({token:null}),T=({children:t,config:r})=>S.createElement(v.Provider,{value:{token:r.userId}},t);import G,{useCallback as x,useContext as $,useRef as K}from"react";import{createContext as V,useEffect as C,useState as E}from"react";import D from"events";import l from"zod";var _=l.object({type:l.literal("subscribe"),channels:l.array(l.string())}),H=l.object({type:l.literal("hook"),channels:l.array(l.string())}),J=l.object({type:l.literal("unsubscribe"),channels:l.string()}),Ae=l.object({event:l.literal("update"),type:l.enum(["value","hook"]),data:l.object({key:l.string(),value:l.string()})}),We=l.union([_,J,H]),y=class extends D{constructor(e){super();this.socket=null;this.queue=[];this.subscriptions=[];this.hooks=[];this.isAuthenticating=!1;this.jwt=null;this.url=e}send(e){this.socket?.readyState===WebSocket.OPEN?this.socket?.send(JSON.stringify(e)):this.queue.push(e)}authenticate(e){this.isAuthenticating||(this.isAuthenticating=!0,this.jwt=e,this.connect())}connect(){let e=this.url+"?jwt="+this.jwt;this.socket=new WebSocket(e),this.socket.onopen=()=>{this.queue.forEach(n=>this.send(n)),this.queue=[]},this.socket.onmessage=n=>{let o=JSON.parse(n.data);this.emit("update",o)},this.socket.onclose=()=>{this.reconnect()}}reconnect(){setTimeout(()=>{this.queue=[{type:"subscribe",channels:this.subscriptions},{type:"hook",channels:this.hooks}],console.log("try reconnecting"),this.connect()},5e3)}subscribe(e){let n=e.filter(o=>!this.subscriptions.includes(o));n.length&&(this.subscriptions=[...this.subscriptions,...n],console.log("subscribe",e),this.send({type:"subscribe",channels:e}))}hook(e){let n=[e].filter(o=>!this.hooks.includes(o));n.length&&(this.hooks=[...this.hooks,...n],this.send({type:"hook",channels:[e]}))}unsubscribe(e){this.send({type:"unsubscribe",channels:e})}disconnect(){this.socket&&(this.socket.onclose=()=>{},this.socket.close())}};var g=V({subscribe:()=>{},listen:()=>{},values:{},hooks:{}}),P=({children:t,config:r})=>{let[e,n]=E({}),[o,c]=E({}),s=K(null),{token:d}=$(v);C(()=>(s.current=new y(r.wsUrl),s.current.on("update",u=>{let{data:a,type:j}=u;if(j==="hook"){let p=JSON.parse(a.value);c(m=>({...m,[p.hook]:p.payload})),console.log("receive hooks",p)}else{let p=a.value;console.log("value",p);try{let m=parseFloat(a.value);if(!isNaN(m)&&String(m)===String(a.value).trim())p=m;else try{p=JSON.parse(a.value)}catch{p=a.value}}catch{p=a.value}n(m=>({...m,[a.key]:p})),console.log("receive values",e)}}),()=>{s.current?.disconnect()}),[]),C(()=>{!s.current||!d||s.current.authenticate(d)},[d]);let f=x(u=>{s.current?.subscribe(u)},[s]),i=x(u=>{s.current?.hook(u)},[s]);return G.createElement(g.Provider,{value:{subscribe:f,listen:i,values:e,hooks:o}},t)};import Y,{createContext as F,useCallback as Q,useEffect as X,useRef as I,useState as Z}from"react";var k=F({status:"disconnected",call:()=>Promise.reject(new Error("ObsProvider not mounted"))});function R({url:t,children:r}){let[e,n]=Z("disconnected"),o=I(null),c=I(!1);X(()=>{if(!t)return;c.current=!1;let d;return(async()=>{let{default:i}=await import("obs-websocket-js"),u=new i;o.current=u;let a=async()=>{if(!c.current){n("connecting");try{await u.connect(t),c.current||n("connected")}catch{c.current||(n("error"),d=setTimeout(a,3e3))}}};u.on("ConnectionClosed",()=>{c.current||(n("disconnected"),d=setTimeout(a,3e3))}),u.on("ConnectionError",()=>{c.current||n("error")}),await a()})(),()=>{c.current=!0,clearTimeout(d),o.current?.disconnect(),o.current=null,n("disconnected")}},[t]);let s=Q(async(d,f)=>{let i=o.current;if(!i)throw new Error("OBS not connected");return i.call(d,f)},[]);return Y.createElement(k.Provider,{value:{status:e,call:s}},r)}import{createContext as ee,useContext as te}from"react";var w=ee(null);function O(){let t=te(w);if(!t)throw new Error("Sparkle hooks require SparkleProvider");return t}var se=({children:t})=>{let[r,e]=oe(!1);return ne(()=>{e(!0)},[]),r?t:null},ie={userId:"",wsUrl:"wss://sparkle-bot-v2.fly.dev/ws"},rt=({children:t,config:r})=>{let e={...ie,...r},n=re(()=>e.obs?.clientId?`${e.obs.bridgeUrl??"wss://sparkle-bot-v2.fly.dev/obs-bridge"}?role=client&clientId=${encodeURIComponent(e.obs.clientId)}`:"",[e.obs?.clientId,e.obs?.bridgeUrl]),o=h.createElement(w.Provider,{value:e},h.createElement(T,{config:e},h.createElement(P,{config:e},h.createElement(se,null,t))));return n?h.createElement(R,{url:n},o):o};import{useContext as ae,useEffect as ce,useMemo as ue}from"react";function at(t,r){let e=ae(g),n=ue(()=>[t],[t]);if(!e)throw new Error("You must use useRealtime inside a SparkleProvider");return ce(()=>{e.subscribe([t])},[n]),n.map(o=>e.values[o]??r)}import{useEffect as le,useState as de}from"react";var pe="sparkle-widget-session-request",fe="sparkle-widget-session-result";function me(){return typeof crypto<"u"&&"randomUUID"in crypto?crypto.randomUUID():`${Date.now()}-${Math.random().toString(16).slice(2)}`}function ge(){if(typeof window>"u")return!1;try{return window.self!==window.top}catch{return!0}}function he(){if(typeof window>"u")return Promise.resolve(null);let t=window.top??window.parent;return!t||t===window?Promise.resolve(null):new Promise(r=>{let e=me(),n=c=>{let s=c.data;if(!(!s||s.type!==fe||s.requestId!==e)){if(window.removeEventListener("message",n),window.clearTimeout(o),typeof s.userId=="string"&&s.userId.trim()){r(s.userId);return}r(null)}},o=window.setTimeout(()=>{window.removeEventListener("message",n),r(null)},12e3);window.addEventListener("message",n),t.postMessage({type:pe,requestId:e},"*")})}async function ye(){let t=await fetch("/api/sparkle-widget-session",{method:"GET",credentials:"include"});if(!t.ok)return null;let r=await t.json().catch(()=>null);return typeof r?.userId=="string"&&r.userId.trim()?r.userId:null}function lt(){let[t,r]=de(void 0);return le(()=>{let e=!1;return(async()=>{let o=ge()?await he():await ye();e||r(o)})(),()=>{e=!0}},[]),t}import{useContext as ke,useEffect as ve}from"react";var gt=["notification.follow","notification.cheer","notification.tips","notification.subscribe","notification.subscribe.end","notification.subscribe.gift","notification.raid","chat.message","stream.online","stream.offline","stream.update","reward.create","reward.update","reward.remove","reward.redemption.claim","reward.redemption.update","poll.begin","poll.progress","poll.end","prediction.begin","prediction.progress","prediction.lock","prediction.end","moderator.add","moderator.remove","moderator.ban","moderator.unban","moderator.shield_mode.begin"];function ht(t){let r=ke(g);if(!r)throw new Error("You must use useHook inside a SparkleProvider");return ve(()=>{r.listen(t)},[t]),r.hooks[t]}import{ApiClient as xe}from"@twurple/api";import{StaticAuthProvider as Ce}from"@twurple/auth";import{useEffect as Ee,useState as Pe}from"react";import{useCallback as M,useState as we,useRef as be,useLayoutEffect as Se}from"react";var b=()=>{};var B=typeof window<"u";var Te=(t,r,e)=>{if(!B)return[r,b,b];if(!t)throw new Error("useLocalStorage key may not be falsy");let n=e?e.raw?i=>i:e.deserializer:JSON.parse,o=be(i=>{try{let u=e?e.raw?String:e.serializer:JSON.stringify,a=localStorage.getItem(i);return a!==null?n(a):(r&&localStorage.setItem(i,u(r)),r)}catch{return r}}),[c,s]=we(()=>o.current(t));Se(()=>s(o.current(t)),[t]);let d=M(i=>{try{let u=typeof i=="function"?i(c):i;if(typeof u>"u")return;let a;e?e.raw?typeof u=="string"?a=u:a=JSON.stringify(u):e.serializer?a=e.serializer(u):a=JSON.stringify(u):a=JSON.stringify(u),localStorage.setItem(t,a),s(n(a))}catch{}},[t,s]),f=M(()=>{try{localStorage.removeItem(t),s(void 0)}catch{}},[t,s]);return[c,d,f]},U=Te;function It(t){let[r,e]=Pe(null),[n]=U("twitch");return Ee(()=>{console.log("env",process.env.TWITCH_CLIENT_ID),console.log("init");let o=new xe({authProvider:new Ce("u9lt242tz2pn5hl5x444ls2xllnvip",n,["bits:read","channel:manage:moderators","channel:manage:polls","channel:manage:predictions","channel:manage:redemptions","channel:manage:vips","channel:read:polls","channel:read:predictions","channel:read:redemptions","channel:read:subscriptions","channel:read:vips","moderation:read","moderator:manage:announcements","channel:moderate","moderator:read:blocked_terms","moderator:read:chatters","moderator:read:followers","user:read:email"])});return e(o),()=>{e(null)}},[]),{apiClient:r}}import{useCallback as Ie,useContext as N}from"react";function Mt(){let{status:t,call:r}=N(k);return{status:t,connected:t==="connected",call:r}}function Ut(t){let{call:r}=N(k);return Ie(e=>r(t,e),[r,t])}import{useCallback as q,useMemo as Re}from"react";var Oe="sparkle-invoke-task",Be="sparkle-invoke-task-result";function Me(){return typeof crypto<"u"&&"randomUUID"in crypto?crypto.randomUUID():`${Date.now()}-${Math.random().toString(16).slice(2)}`}function L(t,r){if(typeof window>"u")return Promise.reject(new Error("useTask requires a browser environment"));let e=window.top??window.parent;return!e||e===window?Promise.reject(new Error("useTask: aucun parent \u2014 utilisez la preview \xE9diteur ou une page int\xE9gr\xE9e.")):new Promise((n,o)=>{let c=Me(),s=f=>{let i=f.data;!i||i.type!==Be||i.requestId!==c||(window.removeEventListener("message",s),window.clearTimeout(d),i.ok?n(!!i.result):o(new Error(typeof i.error=="string"?i.error:"invoke_failed")))},d=window.setTimeout(()=>{window.removeEventListener("message",s),o(new Error("useTask: d\xE9lai d\xE9pass\xE9"))},45e3);window.addEventListener("message",s),e.postMessage({type:Oe,requestId:c,projectId:t,...r},"*")})}function A(){if(typeof window>"u")return!1;try{return window.self!==window.top}catch{return!0}}async function W(t,r){let e=await fetch(`/api/projects/${t}/invoke-task`,{method:"POST",credentials:"include",headers:{"Content-Type":"application/json"},body:JSON.stringify(r)});if(!e.ok){let n=await e.json().catch(()=>null);throw new Error(n?.error??e.statusText)}return!0}function At(){let r=O().projectId,e=q(async(o,c)=>{if(!r?.trim())throw new Error("useTask: renseignez `config.projectId` sur SparkleProvider.");let s={kind:"invoke",taskId:o,payload:c};return A()?L(r,s):W(r,s)},[r]),n=q(async(o,c,s)=>{if(!r?.trim())throw new Error("useTask: renseignez `config.projectId` sur SparkleProvider.");let d={kind:"invokeUntil",taskId:o,payload:s,delaySeconds:c};return A()?L(r,d):W(r,d)},[r]);return Re(()=>({invoke:e,invokeUntil:n}),[e,n])}export{gt as SparkleHookKeys,rt as SparkleProvider,lt as useAuth,ht as useHook,Mt as useObs,Ut as useObsCall,at as useRealtime,At as useTask,It as useTwitch};
1
+ import h,{useMemo as oe,useEffect as se,useState as ie}from"react";import*as T from"react";import{createContext as _}from"react";var v=_({token:null}),x=({children:n,config:t})=>T.createElement(v.Provider,{value:{token:t.userId}},n);import K,{useCallback as C,useContext as V,useRef as Y}from"react";import{createContext as F,useEffect as E,useState as P}from"react";import H from"events";import l from"zod";var J=l.object({type:l.literal("subscribe"),channels:l.array(l.string())}),G=l.object({type:l.literal("hook"),channels:l.array(l.string())}),$=l.object({type:l.literal("unsubscribe"),channels:l.string()}),je=l.object({event:l.literal("update"),type:l.enum(["value","hook"]),data:l.object({key:l.string(),value:l.string()})}),We=l.union([J,$,G]),y=class extends H{constructor(e){super();this.socket=null;this.queue=[];this.subscriptions=[];this.hooks=[];this.isAuthenticating=!1;this.jwt=null;this.url=e}send(e){this.socket?.readyState===WebSocket.OPEN?this.socket?.send(JSON.stringify(e)):this.queue.push(e)}authenticate(e){this.isAuthenticating||(this.isAuthenticating=!0,this.jwt=e,this.connect())}connect(){let e=this.url+"?jwt="+this.jwt;this.socket=new WebSocket(e),this.socket.onopen=()=>{this.queue.forEach(r=>this.send(r)),this.queue=[]},this.socket.onmessage=r=>{let o=JSON.parse(r.data);this.emit("update",o)},this.socket.onclose=()=>{this.reconnect()}}reconnect(){setTimeout(()=>{this.queue=[{type:"subscribe",channels:this.subscriptions},{type:"hook",channels:this.hooks}],console.log("try reconnecting"),this.connect()},5e3)}subscribe(e){let r=e.filter(o=>!this.subscriptions.includes(o));r.length&&(this.subscriptions=[...this.subscriptions,...r],console.log("subscribe",e),this.send({type:"subscribe",channels:e}))}hook(e){let r=[e].filter(o=>!this.hooks.includes(o));r.length&&(this.hooks=[...this.hooks,...r],this.send({type:"hook",channels:[e]}))}unsubscribe(e){this.send({type:"unsubscribe",channels:e})}disconnect(){this.socket&&(this.socket.onclose=()=>{},this.socket.close())}};var g=F({subscribe:()=>{},listen:()=>{},values:{},hooks:{}}),I=({children:n,config:t})=>{let[e,r]=P({}),[o,i]=P({}),s=Y(null),{token:d}=V(v);E(()=>(s.current=new y(t.wsUrl),s.current.on("update",c=>{let{data:u,type:D}=c;if(D==="hook"){let f=JSON.parse(u.value);i(m=>({...m,[f.hook]:f.payload})),console.log("receive hooks",f)}else{let f=u.value;console.log("value",f);try{let m=parseFloat(u.value);if(!isNaN(m)&&String(m)===String(u.value).trim())f=m;else try{f=JSON.parse(u.value)}catch{f=u.value}}catch{f=u.value}r(m=>({...m,[u.key]:f})),console.log("receive values",e)}}),()=>{s.current?.disconnect()}),[]),E(()=>{!s.current||!d||s.current.authenticate(d)},[d]);let p=C(c=>{s.current?.subscribe(c)},[s]),a=C(c=>{s.current?.hook(c)},[s]);return K.createElement(g.Provider,{value:{subscribe:p,listen:a,values:e,hooks:o}},n)};import Q,{createContext as X,useCallback as Z,useEffect as ee,useRef as R,useState as te}from"react";var k=X({status:"disconnected",call:()=>Promise.reject(new Error("ObsProvider not mounted"))});function O({url:n,children:t}){let[e,r]=te("disconnected"),o=R(null),i=R(!1);ee(()=>{if(!n)return;i.current=!1;let d;return(async()=>{let{default:a}=await import("obs-websocket-js"),c=new a;o.current=c;let u=async()=>{if(!i.current){r("connecting");try{await c.connect(n),i.current||r("connected")}catch{i.current||(r("error"),d=setTimeout(u,3e3))}}};c.on("ConnectionClosed",()=>{i.current||(r("disconnected"),d=setTimeout(u,3e3))}),c.on("ConnectionError",()=>{i.current||r("error")}),await u()})(),()=>{i.current=!0,clearTimeout(d),o.current?.disconnect(),o.current=null,r("disconnected")}},[n]);let s=Z(async(d,p)=>{let a=o.current;if(!a)throw new Error("OBS not connected");return a.call(d,p)},[]);return Q.createElement(k.Provider,{value:{status:e,call:s}},t)}import{createContext as ne,useContext as re}from"react";var w=ne(null);function B(){let n=re(w);if(!n)throw new Error("Sparkle hooks require SparkleProvider");return n}var ae=({children:n})=>{let[t,e]=ie(!1);return se(()=>{e(!0)},[]),t?n:null},ue={userId:"",wsUrl:"wss://sparkle-bot-v2.fly.dev/ws"},rt=({children:n,config:t})=>{let e={...ue,...t},r=oe(()=>e.obs?.clientId?`${e.obs.bridgeUrl??"wss://sparkle-bot-v2.fly.dev/obs-bridge"}?role=client&clientId=${encodeURIComponent(e.obs.clientId)}`:"",[e.obs?.clientId,e.obs?.bridgeUrl]),o=h.createElement(w.Provider,{value:e},h.createElement(x,{config:e},h.createElement(I,{config:e},h.createElement(ae,null,n))));return r?h.createElement(O,{url:r},o):o};import{useContext as ce,useEffect as le,useMemo as de}from"react";function ut(n,t){let e=ce(g),r=de(()=>[n],[n]);if(!e)throw new Error("You must use useRealtime inside a SparkleProvider");return le(()=>{e.subscribe([n])},[r]),r.map(o=>e.values[o]??t)}import{useEffect as pe,useState as b}from"react";var fe="sparkle-widget-session-request",me="sparkle-widget-session-result";function ge(){return typeof crypto<"u"&&"randomUUID"in crypto?crypto.randomUUID():`${Date.now()}-${Math.random().toString(16).slice(2)}`}function he(){if(typeof window>"u")return!1;try{return window.self!==window.top}catch{return!0}}function M(n){if(!n||typeof n!="object")return{userId:null,username:null,image:null};let t=n,e=typeof t.userId=="string"&&t.userId.trim()?t.userId:null,r=typeof t.username=="string"&&t.username.trim()?t.username:null,o=typeof t.image=="string"&&t.image.trim()?t.image:null;return{userId:e,username:r,image:o}}function ye(){if(typeof window>"u")return Promise.resolve({userId:null,username:null,image:null});let n=window.top??window.parent;return!n||n===window?Promise.resolve({userId:null,username:null,image:null}):new Promise(t=>{let e=ge(),r=i=>{let s=i.data;!s||s.type!==me||s.requestId!==e||(window.removeEventListener("message",r),window.clearTimeout(o),t(M(s)))},o=window.setTimeout(()=>{window.removeEventListener("message",r),t({userId:null,username:null,image:null})},12e3);window.addEventListener("message",r),n.postMessage({type:fe,requestId:e},"*")})}async function ke(){let n=await fetch("/api/sparkle-widget-session",{method:"GET",credentials:"include"});if(!n.ok)return{userId:null,username:null,image:null};let t=await n.json().catch(()=>null);return M(t)}function dt(){let[n,t]=b(void 0),[e,r]=b(void 0),[o,i]=b(void 0);return pe(()=>{let s=!1;return(async()=>{let p=he()?await ye():await ke();s||(t(p.userId),r(p.username),i(p.image))})(),()=>{s=!0}},[]),{userId:n,username:e,image:o}}import{useContext as ve,useEffect as we}from"react";var ht=["notification.follow","notification.cheer","notification.tips","notification.subscribe","notification.subscribe.end","notification.subscribe.gift","notification.raid","chat.message","stream.online","stream.offline","stream.update","reward.create","reward.update","reward.remove","reward.redemption.claim","reward.redemption.update","poll.begin","poll.progress","poll.end","prediction.begin","prediction.progress","prediction.lock","prediction.end","moderator.add","moderator.remove","moderator.ban","moderator.unban","moderator.shield_mode.begin"];function yt(n){let t=ve(g);if(!t)throw new Error("You must use useHook inside a SparkleProvider");return we(()=>{t.listen(n)},[n]),t.hooks[n]}import{ApiClient as Ce}from"@twurple/api";import{StaticAuthProvider as Ee}from"@twurple/auth";import{useEffect as Pe,useState as Ie}from"react";import{useCallback as N,useState as be,useRef as Se,useLayoutEffect as Te}from"react";var S=()=>{};var U=typeof window<"u";var xe=(n,t,e)=>{if(!U)return[t,S,S];if(!n)throw new Error("useLocalStorage key may not be falsy");let r=e?e.raw?a=>a:e.deserializer:JSON.parse,o=Se(a=>{try{let c=e?e.raw?String:e.serializer:JSON.stringify,u=localStorage.getItem(a);return u!==null?r(u):(t&&localStorage.setItem(a,c(t)),t)}catch{return t}}),[i,s]=be(()=>o.current(n));Te(()=>s(o.current(n)),[n]);let d=N(a=>{try{let c=typeof a=="function"?a(i):a;if(typeof c>"u")return;let u;e?e.raw?typeof c=="string"?u=c:u=JSON.stringify(c):e.serializer?u=e.serializer(c):u=JSON.stringify(c):u=JSON.stringify(c),localStorage.setItem(n,u),s(r(u))}catch{}},[n,s]),p=N(()=>{try{localStorage.removeItem(n),s(void 0)}catch{}},[n,s]);return[i,d,p]},q=xe;function Rt(n){let[t,e]=Ie(null),[r]=q("twitch");return Pe(()=>{console.log("env",process.env.TWITCH_CLIENT_ID),console.log("init");let o=new Ce({authProvider:new Ee("u9lt242tz2pn5hl5x444ls2xllnvip",r,["bits:read","channel:manage:moderators","channel:manage:polls","channel:manage:predictions","channel:manage:redemptions","channel:manage:vips","channel:read:polls","channel:read:predictions","channel:read:redemptions","channel:read:subscriptions","channel:read:vips","moderation:read","moderator:manage:announcements","channel:moderate","moderator:read:blocked_terms","moderator:read:chatters","moderator:read:followers","user:read:email"])});return e(o),()=>{e(null)}},[]),{apiClient:t}}import{useCallback as Re,useContext as L}from"react";function Ut(){let{status:n,call:t}=L(k);return{status:n,connected:n==="connected",call:t}}function Nt(n){let{call:t}=L(k);return Re(e=>t(n,e),[t,n])}import{useCallback as A,useMemo as Oe}from"react";var Be="sparkle-invoke-task",Me="sparkle-invoke-task-result";function Ue(){return typeof crypto<"u"&&"randomUUID"in crypto?crypto.randomUUID():`${Date.now()}-${Math.random().toString(16).slice(2)}`}function j(n,t){if(typeof window>"u")return Promise.reject(new Error("useTask requires a browser environment"));let e=window.top??window.parent;return!e||e===window?Promise.reject(new Error("useTask: aucun parent \u2014 utilisez la preview \xE9diteur ou une page int\xE9gr\xE9e.")):new Promise((r,o)=>{let i=Ue(),s=p=>{let a=p.data;!a||a.type!==Me||a.requestId!==i||(window.removeEventListener("message",s),window.clearTimeout(d),a.ok?r(!!a.result):o(new Error(typeof a.error=="string"?a.error:"invoke_failed")))},d=window.setTimeout(()=>{window.removeEventListener("message",s),o(new Error("useTask: d\xE9lai d\xE9pass\xE9"))},45e3);window.addEventListener("message",s),e.postMessage({type:Be,requestId:i,projectId:n,...t},"*")})}function W(){if(typeof window>"u")return!1;try{return window.self!==window.top}catch{return!0}}async function z(n,t){let e=await fetch(`/api/projects/${n}/invoke-task`,{method:"POST",credentials:"include",headers:{"Content-Type":"application/json"},body:JSON.stringify(t)});if(!e.ok){let r=await e.json().catch(()=>null);throw new Error(r?.error??e.statusText)}return!0}function jt(){let t=B().projectId,e=A(async(o,i)=>{if(!t?.trim())throw new Error("useTask: renseignez `config.projectId` sur SparkleProvider.");let s={kind:"invoke",taskId:o,payload:i};return W()?j(t,s):z(t,s)},[t]),r=A(async(o,i,s)=>{if(!t?.trim())throw new Error("useTask: renseignez `config.projectId` sur SparkleProvider.");let d={kind:"invokeUntil",taskId:o,payload:s,delaySeconds:i};return W()?j(t,d):z(t,d)},[t]);return Oe(()=>({invoke:e,invokeUntil:r}),[e,r])}export{ht as SparkleHookKeys,rt as SparkleProvider,dt as useAuth,yt as useHook,Ut as useObs,Nt as useObsCall,ut as useRealtime,jt as useTask,Rt as useTwitch};
2
2
  //# sourceMappingURL=index.mjs.map
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/provider.tsx","../src/providers/authProvider.tsx","../src/providers/realtimeProvider.tsx","../src/lib/websocket.ts","../src/providers/obsProvider.tsx","../src/sparkle-config-context.tsx","../src/hooks/useRealtime.ts","../src/hooks/useAuth.ts","../src/hooks/useHook.ts","../src/hooks/useTwitch.ts","../src/utils/hooks/useLocalStorage.ts","../src/utils/utils.ts","../src/hooks/useObs.ts","../src/hooks/useTask.ts"],"sourcesContent":["import React, { useMemo, type ReactNode, useEffect, useState } from \"react\"\nimport { AuthProvider } from \"./providers/authProvider\"\nimport { WebSocketProvider } from \"./providers/realtimeProvider\"\nimport { ObsProvider } from \"./providers/obsProvider\"\nimport { SparkleConfigContext } from \"./sparkle-config-context\"\nimport { SparkleConfig } from \"./utils/config\"\n\nconst ClientGate = ({ children }: { children: ReactNode }) => {\n const [ready, setReady] = useState(false)\n\n useEffect(() => {\n setReady(true)\n }, [])\n\n if (!ready) return null\n return children\n}\n\nexport interface SparkleProviderProps {\n config?: SparkleConfig\n children: ReactNode\n}\n\nconst defaultConfig: SparkleConfig = {\n userId: \"\",\n wsUrl: \"wss://sparkle-bot-v2.fly.dev/ws\",\n}\n\nexport const SparkleProvider = ({ children, config }: SparkleProviderProps) => {\n const mergedConfig = { ...defaultConfig, ...config }\n\n const obsUrl = useMemo(() => {\n if (!mergedConfig.obs?.clientId) return \"\"\n const base =\n mergedConfig.obs.bridgeUrl ?? \"wss://sparkle-bot-v2.fly.dev/obs-bridge\"\n return `${base}?role=client&clientId=${encodeURIComponent(mergedConfig.obs.clientId)}`\n }, [mergedConfig.obs?.clientId, mergedConfig.obs?.bridgeUrl])\n\n const inner = (\n <SparkleConfigContext.Provider value={mergedConfig}>\n <AuthProvider config={mergedConfig}>\n <WebSocketProvider config={mergedConfig}>\n <ClientGate>{children}</ClientGate>\n </WebSocketProvider>\n </AuthProvider>\n </SparkleConfigContext.Provider>\n )\n\n if (obsUrl) {\n return <ObsProvider url={obsUrl}>{inner}</ObsProvider>\n }\n\n return inner\n}\n","import * as React from \"react\"\r\n\r\nimport { type ReactNode, createContext } from \"react\"\r\n\r\nimport { SparkleConfig } from \"../utils/config\"\r\n\r\ninterface AuthContextType {\r\n token: string | undefined | null\r\n}\r\n\r\nexport const SparkleAuthContext = createContext<AuthContextType>({\r\n token: null,\r\n})\r\n\r\ninterface AuthProviderProps {\r\n config: SparkleConfig\r\n children: ReactNode\r\n}\r\n\r\nexport const AuthProvider = ({ children, config }: AuthProviderProps) => {\r\n return (\r\n <SparkleAuthContext.Provider\r\n value={{\r\n token: config.userId,\r\n }}\r\n >\r\n {children}\r\n </SparkleAuthContext.Provider>\r\n )\r\n}\r\n","import React, { ReactNode, useCallback, useContext, useRef } from \"react\"\r\n\r\nimport { createContext, useEffect, useState } from \"react\"\r\nimport { SparkleAuthContext } from \"./authProvider\"\r\nimport { UpdateMessage, WebSocketClient } from \"../lib/websocket\"\r\nimport { SparkleConfig } from \"../utils/config\"\r\n\r\ninterface WebSocketContextType {\r\n subscribe: (keys: string[]) => void\r\n listen: (hook: string) => void\r\n values: Record<string, any>\r\n hooks: Record<string, any>\r\n}\r\n\r\nexport const RealtimeContext = createContext<WebSocketContextType>({\r\n subscribe: () => {},\r\n listen: () => {},\r\n values: {},\r\n hooks: {},\r\n})\r\n\r\nexport type RealtimeValue = string | number | boolean | { [key: string]: any }\r\n\r\ntype State = Record<string, RealtimeValue>\r\n\r\ninterface WebSocketProviderProps {\r\n config: SparkleConfig\r\n children: ReactNode\r\n}\r\n\r\nexport const WebSocketProvider = ({\r\n children,\r\n config,\r\n}: WebSocketProviderProps) => {\r\n const [values, setValues] = useState<State>({})\r\n const [hooks, setHooks] = useState<Record<string, any>>({})\r\n\r\n const socket = useRef<WebSocketClient>(null)\r\n\r\n const { token } = useContext(SparkleAuthContext)\r\n\r\n useEffect(() => {\r\n socket.current = new WebSocketClient(config.wsUrl)\r\n\r\n socket.current.on(\"update\", (message: UpdateMessage) => {\r\n const { data, type } = message\r\n\r\n if (type === \"hook\") {\r\n const payload = JSON.parse(data.value)\r\n\r\n setHooks((prev) => ({\r\n ...prev,\r\n [payload.hook]: payload.payload,\r\n }))\r\n console.log(\"receive hooks\", payload)\r\n } else {\r\n let value: any = data.value\r\n\r\n console.log(\"value\", value)\r\n\r\n try {\r\n const num = parseFloat(data.value)\r\n if (!isNaN(num) && String(num) === String(data.value).trim()) {\r\n value = num\r\n } else {\r\n try {\r\n value = JSON.parse(data.value)\r\n } catch {\r\n value = data.value\r\n }\r\n }\r\n } catch {\r\n value = data.value\r\n }\r\n\r\n setValues((prev) => ({ ...prev, [data.key]: value }))\r\n console.log(\"receive values\", values)\r\n }\r\n })\r\n\r\n return () => {\r\n socket.current?.disconnect()\r\n }\r\n }, [])\r\n\r\n useEffect(() => {\r\n if (!socket.current || !token) return\r\n\r\n socket.current.authenticate(token)\r\n }, [token])\r\n\r\n const subscribe = useCallback(\r\n (keys: string[]) => {\r\n socket.current?.subscribe(keys)\r\n },\r\n [socket]\r\n )\r\n\r\n const listen = useCallback(\r\n (hook: string) => {\r\n socket.current?.hook(hook)\r\n },\r\n [socket]\r\n )\r\n\r\n return (\r\n <RealtimeContext.Provider\r\n value={{\r\n subscribe,\r\n listen,\r\n values,\r\n hooks,\r\n }}\r\n >\r\n {children}\r\n </RealtimeContext.Provider>\r\n )\r\n}\r\n","import EventEmitter from \"events\"\r\nimport z from \"zod\"\r\n\r\nconst subscribeSchema = z.object({\r\n type: z.literal(\"subscribe\"),\r\n channels: z.array(z.string()),\r\n})\r\n\r\nconst hookSchema = z.object({\r\n type: z.literal(\"hook\"),\r\n channels: z.array(z.string()),\r\n})\r\n\r\nconst unsubscribeSchema = z.object({\r\n type: z.literal(\"unsubscribe\"),\r\n channels: z.string(),\r\n})\r\n\r\nconst updateSchema = z.object({\r\n event: z.literal(\"update\"),\r\n type: z.enum([\"value\", \"hook\"]),\r\n data: z.object({\r\n key: z.string(),\r\n value: z.string(),\r\n }),\r\n})\r\n\r\nconst messageSchema = z.union([subscribeSchema, unsubscribeSchema, hookSchema])\r\n\r\nexport type Message = z.infer<typeof messageSchema>\r\n\r\nexport type UpdateMessage = z.infer<typeof updateSchema>\r\nexport type SubscribeMessage = z.infer<typeof subscribeSchema>\r\nexport type UnsubscribeMessage = z.infer<typeof unsubscribeSchema>\r\n\r\nexport class WebSocketClient extends EventEmitter {\r\n private socket: WebSocket | null = null\r\n private queue: Message[] = []\r\n private subscriptions: string[] = []\r\n private hooks: string[] = []\r\n private isAuthenticating = false\r\n\r\n private url: string\r\n private jwt: string | null = null\r\n\r\n constructor(url: string) {\r\n super()\r\n this.url = url\r\n }\r\n\r\n private send(data: Message) {\r\n if (this.socket?.readyState === WebSocket.OPEN) {\r\n this.socket?.send(JSON.stringify(data))\r\n } else {\r\n this.queue.push(data)\r\n }\r\n }\r\n\r\n authenticate(jwt: string) {\r\n if (this.isAuthenticating) return\r\n this.isAuthenticating = true\r\n this.jwt = jwt\r\n this.connect()\r\n }\r\n\r\n private connect() {\r\n const authenticatedUrl = this.url + \"?jwt=\" + this.jwt\r\n this.socket = new WebSocket(authenticatedUrl)\r\n\r\n this.socket.onopen = () => {\r\n this.queue.forEach((data) => this.send(data))\r\n this.queue = []\r\n }\r\n\r\n this.socket.onmessage = (event) => {\r\n const data = JSON.parse(event.data)\r\n this.emit(\"update\", data)\r\n }\r\n\r\n this.socket.onclose = () => {\r\n this.reconnect()\r\n }\r\n }\r\n\r\n reconnect() {\r\n setTimeout(() => {\r\n this.queue = [\r\n {\r\n type: \"subscribe\",\r\n channels: this.subscriptions,\r\n },\r\n {\r\n type: \"hook\",\r\n channels: this.hooks,\r\n },\r\n ]\r\n\r\n console.log(\"try reconnecting\")\r\n\r\n this.connect()\r\n }, 5000)\r\n }\r\n\r\n subscribe(channels: string[]) {\r\n const newChannels = channels.filter(\r\n (channel) => !this.subscriptions.includes(channel),\r\n )\r\n\r\n if (!newChannels.length) return\r\n this.subscriptions = [...this.subscriptions, ...newChannels]\r\n\r\n console.log(\"subscribe\", channels)\r\n\r\n this.send({\r\n type: \"subscribe\",\r\n channels,\r\n })\r\n }\r\n\r\n hook(hook: string) {\r\n const newChannels = [hook].filter(\r\n (channel) => !this.hooks.includes(channel),\r\n )\r\n\r\n if (!newChannels.length) return\r\n this.hooks = [...this.hooks, ...newChannels]\r\n\r\n this.send({\r\n type: \"hook\",\r\n channels: [hook],\r\n })\r\n }\r\n\r\n unsubscribe(channels: string) {\r\n this.send({ type: \"unsubscribe\", channels })\r\n }\r\n\r\n disconnect() {\r\n if (!this.socket) return\r\n\r\n this.socket.onclose = () => {}\r\n this.socket.close()\r\n }\r\n}\r\n","import React, {\r\n createContext,\r\n useCallback,\r\n useEffect,\r\n useRef,\r\n useState,\r\n type ReactNode,\r\n} from \"react\"\r\nimport type OBSWebSocket from \"obs-websocket-js\"\r\nimport type { OBSRequestTypes, OBSResponseTypes } from \"obs-websocket-js\"\r\n\r\ntype ObsStatus = \"disconnected\" | \"connecting\" | \"connected\" | \"error\"\r\n\r\ninterface ObsContextType {\r\n status: ObsStatus\r\n call: <T extends keyof OBSRequestTypes>(\r\n requestType: T,\r\n requestData?: OBSRequestTypes[T]\r\n ) => Promise<OBSResponseTypes[T]>\r\n}\r\n\r\nexport const ObsContext = createContext<ObsContextType>({\r\n status: \"disconnected\",\r\n call: () => Promise.reject(new Error(\"ObsProvider not mounted\")),\r\n})\r\n\r\nexport interface ObsProviderProps {\r\n url: string\r\n children: ReactNode\r\n}\r\n\r\nexport function ObsProvider({ url, children }: ObsProviderProps) {\r\n const [status, setStatus] = useState<ObsStatus>(\"disconnected\")\r\n const obsRef = useRef<OBSWebSocket | null>(null)\r\n const cancelledRef = useRef(false)\r\n\r\n useEffect(() => {\r\n if (!url) return\r\n\r\n cancelledRef.current = false\r\n let reconnectTimer: ReturnType<typeof setTimeout>\r\n\r\n const setup = async () => {\r\n const { default: OBSWebSocket } = await import(\"obs-websocket-js\")\r\n const obs = new OBSWebSocket()\r\n obsRef.current = obs\r\n\r\n const doConnect = async () => {\r\n if (cancelledRef.current) return\r\n setStatus(\"connecting\")\r\n try {\r\n await obs.connect(url)\r\n if (!cancelledRef.current) setStatus(\"connected\")\r\n } catch {\r\n if (!cancelledRef.current) {\r\n setStatus(\"error\")\r\n reconnectTimer = setTimeout(doConnect, 3000)\r\n }\r\n }\r\n }\r\n\r\n obs.on(\"ConnectionClosed\" as any, () => {\r\n if (!cancelledRef.current) {\r\n setStatus(\"disconnected\")\r\n reconnectTimer = setTimeout(doConnect, 3000)\r\n }\r\n })\r\n\r\n obs.on(\"ConnectionError\" as any, () => {\r\n if (!cancelledRef.current) setStatus(\"error\")\r\n })\r\n\r\n await doConnect()\r\n }\r\n\r\n setup()\r\n\r\n return () => {\r\n cancelledRef.current = true\r\n clearTimeout(reconnectTimer!)\r\n obsRef.current?.disconnect()\r\n obsRef.current = null\r\n setStatus(\"disconnected\")\r\n }\r\n }, [url])\r\n\r\n const call = useCallback(\r\n async <T extends keyof OBSRequestTypes>(\r\n requestType: T,\r\n requestData?: OBSRequestTypes[T]\r\n ): Promise<OBSResponseTypes[T]> => {\r\n const obs = obsRef.current\r\n if (!obs) throw new Error(\"OBS not connected\")\r\n return obs.call(requestType, requestData)\r\n },\r\n []\r\n )\r\n\r\n return (\r\n <ObsContext.Provider value={{ status, call }}>\r\n {children}\r\n </ObsContext.Provider>\r\n )\r\n}\r\n","import { createContext, useContext } from \"react\"\r\nimport type { SparkleConfig } from \"./utils/config\"\r\n\r\nexport const SparkleConfigContext = createContext<SparkleConfig | null>(null)\r\n\r\nexport function useSparkleConfig(): SparkleConfig {\r\n const cfg = useContext(SparkleConfigContext)\r\n if (!cfg) {\r\n throw new Error(\"Sparkle hooks require SparkleProvider\")\r\n }\r\n return cfg\r\n}\r\n","import { useContext, useEffect, useMemo } from \"react\"\r\nimport { RealtimeContext, RealtimeValue } from \"../providers/realtimeProvider\"\r\n\r\nexport function useRealtime<T extends RealtimeValue>(\r\n key: string,\r\n defaultValue?: T\r\n): T[] {\r\n const context = useContext(RealtimeContext)\r\n\r\n const keys = useMemo(() => [key], [key])\r\n\r\n if (!context) {\r\n throw new Error(\"You must use useRealtime inside a SparkleProvider\")\r\n }\r\n\r\n useEffect(() => {\r\n context.subscribe([key])\r\n }, [keys])\r\n\r\n return keys.map((k) => (context.values[k] ?? defaultValue) as T)\r\n}\r\n","import { useEffect, useState } from \"react\"\r\n\r\nconst WIDGET_SESSION_REQ = \"sparkle-widget-session-request\"\r\nconst WIDGET_SESSION_RES = \"sparkle-widget-session-result\"\r\n\r\nfunction newRequestId(): string {\r\n if (typeof crypto !== \"undefined\" && \"randomUUID\" in crypto) {\r\n return crypto.randomUUID()\r\n }\r\n return `${Date.now()}-${Math.random().toString(16).slice(2)}`\r\n}\r\n\r\nfunction isEmbedded(): boolean {\r\n if (typeof window === \"undefined\") return false\r\n try {\r\n return window.self !== window.top\r\n } catch {\r\n return true\r\n }\r\n}\r\n\r\nfunction requestSessionViaParent(): Promise<string | null> {\r\n if (typeof window === \"undefined\") {\r\n return Promise.resolve(null)\r\n }\r\n const target = window.top ?? window.parent\r\n if (!target || target === window) {\r\n return Promise.resolve(null)\r\n }\r\n return new Promise((resolve) => {\r\n const requestId = newRequestId()\r\n const onMsg = (ev: MessageEvent) => {\r\n const d = ev.data\r\n if (!d || d.type !== WIDGET_SESSION_RES || d.requestId !== requestId) return\r\n window.removeEventListener(\"message\", onMsg)\r\n window.clearTimeout(timer)\r\n if (typeof d.userId === \"string\" && d.userId.trim()) {\r\n resolve(d.userId)\r\n return\r\n }\r\n resolve(null)\r\n }\r\n const timer = window.setTimeout(() => {\r\n window.removeEventListener(\"message\", onMsg)\r\n resolve(null)\r\n }, 12_000)\r\n window.addEventListener(\"message\", onMsg)\r\n target.postMessage({ type: WIDGET_SESSION_REQ, requestId }, \"*\")\r\n })\r\n}\r\n\r\nasync function fetchSessionUserId(): Promise<string | null> {\r\n const res = await fetch(\"/api/sparkle-widget-session\", {\r\n method: \"GET\",\r\n credentials: \"include\",\r\n })\r\n if (!res.ok) return null\r\n const data = (await res.json().catch(() => null)) as { userId?: unknown } | null\r\n if (typeof data?.userId === \"string\" && data.userId.trim()) return data.userId\r\n return null\r\n}\r\n\r\n/**\r\n * Identifiant utilisateur Better Auth (`session.user.id`) de l’app principale.\r\n * `undefined` pendant le chargement, `null` si non connecté ou erreur, sinon l’id.\r\n */\r\nexport function useAuth(): string | null | undefined {\r\n const [userId, setUserId] = useState<string | null | undefined>(undefined)\r\n\r\n useEffect(() => {\r\n let cancelled = false\r\n const run = async () => {\r\n const uid = isEmbedded()\r\n ? await requestSessionViaParent()\r\n : await fetchSessionUserId()\r\n if (!cancelled) setUserId(uid)\r\n }\r\n void run()\r\n return () => {\r\n cancelled = true\r\n }\r\n }, [])\r\n\r\n return userId\r\n}\r\n","import { useContext, useEffect, useMemo } from \"react\"\r\nimport { RealtimeContext } from \"../providers/realtimeProvider\"\r\nimport { EventTypeMap } from \"./hook\"\r\n\r\nexport const SparkleHookKeys = [\r\n \"notification.follow\",\r\n \"notification.cheer\",\r\n \"notification.tips\",\r\n \"notification.subscribe\",\r\n \"notification.subscribe.end\",\r\n \"notification.subscribe.gift\",\r\n \"notification.raid\",\r\n\r\n \"chat.message\",\r\n\r\n \"stream.online\",\r\n \"stream.offline\",\r\n \"stream.update\",\r\n\r\n \"reward.create\",\r\n \"reward.update\",\r\n \"reward.remove\",\r\n\r\n \"reward.redemption.claim\",\r\n \"reward.redemption.update\",\r\n\r\n \"poll.begin\",\r\n \"poll.progress\",\r\n \"poll.end\",\r\n\r\n \"prediction.begin\",\r\n \"prediction.progress\",\r\n \"prediction.lock\",\r\n \"prediction.end\",\r\n\r\n \"moderator.add\",\r\n \"moderator.remove\",\r\n \"moderator.ban\",\r\n \"moderator.unban\",\r\n \"moderator.shield_mode.begin\",\r\n] as const\r\n\r\nexport type SparkleHookKey = (typeof SparkleHookKeys)[number]\r\n\r\nexport function useHook<T extends keyof EventTypeMap>(\r\n hook: T\r\n): EventTypeMap[T] | undefined {\r\n const context = useContext(RealtimeContext)\r\n\r\n if (!context) {\r\n throw new Error(\"You must use useHook inside a SparkleProvider\")\r\n }\r\n\r\n useEffect(() => {\r\n context.listen(hook)\r\n }, [hook])\r\n\r\n return context.hooks[hook] as EventTypeMap[T]\r\n}\r\n","import { ApiClient } from \"@twurple/api\"\r\nimport { StaticAuthProvider } from \"@twurple/auth\"\r\nimport { useEffect, useState } from \"react\"\r\nimport useLocalStorage from \"../utils/hooks/useLocalStorage\"\r\n\r\nexport function useTwitch(asUser?: string | number) {\r\n const [apiClient, setApiClient] = useState<ApiClient | null>(null)\r\n const [twitch] = useLocalStorage<string>(\"twitch\")\r\n\r\n useEffect(() => {\r\n console.log(\"env\", process.env.TWITCH_CLIENT_ID)\r\n\r\n console.log(\"init\")\r\n\r\n const apiClient = new ApiClient({\r\n authProvider: new StaticAuthProvider(\r\n \"u9lt242tz2pn5hl5x444ls2xllnvip\",\r\n twitch!,\r\n [\r\n \"bits:read\",\r\n \"channel:manage:moderators\",\r\n \"channel:manage:polls\",\r\n \"channel:manage:predictions\",\r\n \"channel:manage:redemptions\",\r\n \"channel:manage:vips\",\r\n\r\n \"channel:read:polls\",\r\n \"channel:read:predictions\",\r\n \"channel:read:redemptions\",\r\n \"channel:read:subscriptions\",\r\n \"channel:read:vips\",\r\n \"moderation:read\",\r\n \"moderator:manage:announcements\",\r\n \"channel:moderate\",\r\n \"moderator:read:blocked_terms\",\r\n \"moderator:read:chatters\",\r\n \"moderator:read:followers\",\r\n \"user:read:email\",\r\n ]\r\n ),\r\n })\r\n\r\n setApiClient(apiClient)\r\n\r\n return () => {\r\n setApiClient(null)\r\n }\r\n }, [])\r\n\r\n return {\r\n apiClient,\r\n }\r\n}\r\n","import {\r\n Dispatch,\r\n SetStateAction,\r\n useCallback,\r\n useState,\r\n useRef,\r\n useLayoutEffect,\r\n} from \"react\"\r\nimport { isBrowser, noop } from \"../utils\"\r\n\r\ntype parserOptions<T> =\r\n | {\r\n raw: true\r\n }\r\n | {\r\n raw: false\r\n serializer: (value: T) => string\r\n deserializer: (value: string) => T\r\n }\r\n\r\nconst useLocalStorage = <T>(\r\n key: string,\r\n initialValue?: T,\r\n options?: parserOptions<T>\r\n): [T | undefined, Dispatch<SetStateAction<T | undefined>>, () => void] => {\r\n if (!isBrowser) {\r\n return [initialValue as T, noop, noop]\r\n }\r\n if (!key) {\r\n throw new Error(\"useLocalStorage key may not be falsy\")\r\n }\r\n\r\n const deserializer = options\r\n ? options.raw\r\n ? (value) => value\r\n : options.deserializer\r\n : JSON.parse\r\n\r\n // eslint-disable-next-line react-hooks/rules-of-hooks\r\n const initializer = useRef((key: string) => {\r\n try {\r\n const serializer = options\r\n ? options.raw\r\n ? String\r\n : options.serializer\r\n : JSON.stringify\r\n\r\n const localStorageValue = localStorage.getItem(key)\r\n if (localStorageValue !== null) {\r\n return deserializer(localStorageValue)\r\n } else {\r\n initialValue && localStorage.setItem(key, serializer(initialValue))\r\n return initialValue\r\n }\r\n } catch {\r\n // If user is in private mode or has storage restriction\r\n // localStorage can throw. JSON.parse and JSON.stringify\r\n // can throw, too.\r\n return initialValue\r\n }\r\n })\r\n\r\n // eslint-disable-next-line react-hooks/rules-of-hooks\r\n const [state, setState] = useState<T | undefined>(() =>\r\n initializer.current(key)\r\n )\r\n\r\n // eslint-disable-next-line react-hooks/rules-of-hooks\r\n useLayoutEffect(() => setState(initializer.current(key)), [key])\r\n\r\n // eslint-disable-next-line react-hooks/rules-of-hooks\r\n const set: Dispatch<SetStateAction<T | undefined>> = useCallback(\r\n (valOrFunc) => {\r\n try {\r\n const newState =\r\n typeof valOrFunc === \"function\"\r\n ? (valOrFunc as Function)(state)\r\n : valOrFunc\r\n if (typeof newState === \"undefined\") return\r\n let value: string\r\n\r\n if (options)\r\n if (options.raw)\r\n if (typeof newState === \"string\") value = newState\r\n else value = JSON.stringify(newState)\r\n else if (options.serializer) value = options.serializer(newState)\r\n else value = JSON.stringify(newState)\r\n else value = JSON.stringify(newState)\r\n\r\n localStorage.setItem(key, value)\r\n setState(deserializer(value))\r\n } catch {\r\n // If user is in private mode or has storage restriction\r\n // localStorage can throw. Also JSON.stringify can throw.\r\n }\r\n },\r\n [key, setState]\r\n )\r\n\r\n // eslint-disable-next-line react-hooks/rules-of-hooks\r\n const remove = useCallback(() => {\r\n try {\r\n localStorage.removeItem(key)\r\n setState(undefined)\r\n } catch {\r\n // If user is in private mode or has storage restriction\r\n // localStorage can throw.\r\n }\r\n }, [key, setState])\r\n\r\n return [state, set, remove]\r\n}\r\n\r\nexport default useLocalStorage\r\n","export const noop = () => {}\r\n\r\nexport function on<T extends Window | Document | HTMLElement | EventTarget>(\r\n obj: T | null,\r\n ...args: Parameters<T[\"addEventListener\"]> | [string, Function | null, ...any]\r\n): void {\r\n if (obj && obj.addEventListener) {\r\n obj.addEventListener(\r\n ...(args as Parameters<HTMLElement[\"addEventListener\"]>)\r\n )\r\n }\r\n}\r\n\r\nexport function off<T extends Window | Document | HTMLElement | EventTarget>(\r\n obj: T | null,\r\n ...args:\r\n | Parameters<T[\"removeEventListener\"]>\r\n | [string, Function | null, ...any]\r\n): void {\r\n if (obj && obj.removeEventListener) {\r\n obj.removeEventListener(\r\n ...(args as Parameters<HTMLElement[\"removeEventListener\"]>)\r\n )\r\n }\r\n}\r\n\r\nexport const isBrowser = typeof window !== \"undefined\"\r\n\r\nexport const isNavigator = typeof navigator !== \"undefined\"\r\n","import { useCallback, useContext } from \"react\"\r\nimport type { OBSRequestTypes, OBSResponseTypes } from \"obs-websocket-js\"\r\nimport { ObsContext } from \"../providers/obsProvider\"\r\n\r\nexport type { OBSRequestTypes, OBSResponseTypes }\r\n\r\nexport function useObs() {\r\n const { status, call } = useContext(ObsContext)\r\n\r\n return {\r\n status,\r\n connected: status === \"connected\",\r\n call,\r\n }\r\n}\r\n\r\nexport function useObsCall<T extends keyof OBSRequestTypes>(requestType: T) {\r\n const { call } = useContext(ObsContext)\r\n\r\n return useCallback(\r\n (requestData?: OBSRequestTypes[T]) => call(requestType, requestData),\r\n [call, requestType]\r\n )\r\n}\r\n","import { useCallback, useMemo } from \"react\"\r\nimport { useSparkleConfig } from \"../sparkle-config-context\"\r\n\r\nconst BRIDGE_TYPE = \"sparkle-invoke-task\"\r\nconst BRIDGE_RESULT = \"sparkle-invoke-task-result\"\r\n\r\ntype InvokeBridgeBody =\r\n | { kind: \"invoke\"; taskId: string; payload?: unknown }\r\n | {\r\n kind: \"invokeUntil\"\r\n taskId: string\r\n payload?: unknown\r\n delaySeconds: number\r\n }\r\n\r\nfunction newRequestId(): string {\r\n if (typeof crypto !== \"undefined\" && \"randomUUID\" in crypto) {\r\n return crypto.randomUUID()\r\n }\r\n return `${Date.now()}-${Math.random().toString(16).slice(2)}`\r\n}\r\n\r\nfunction bridgeInvoke(\r\n projectId: string,\r\n body: InvokeBridgeBody\r\n): Promise<boolean> {\r\n if (typeof window === \"undefined\") {\r\n return Promise.reject(new Error(\"useTask requires a browser environment\"))\r\n }\r\n const target = window.top ?? window.parent\r\n if (!target || target === window) {\r\n return Promise.reject(\r\n new Error(\r\n \"useTask: aucun parent — utilisez la preview éditeur ou une page intégrée.\"\r\n )\r\n )\r\n }\r\n\r\n return new Promise((resolve, reject) => {\r\n const requestId = newRequestId()\r\n const onMsg = (ev: MessageEvent) => {\r\n const d = ev.data\r\n if (!d || d.type !== BRIDGE_RESULT || d.requestId !== requestId) return\r\n window.removeEventListener(\"message\", onMsg)\r\n window.clearTimeout(timer)\r\n if (d.ok) resolve(Boolean(d.result))\r\n else reject(new Error(typeof d.error === \"string\" ? d.error : \"invoke_failed\"))\r\n }\r\n const timer = window.setTimeout(() => {\r\n window.removeEventListener(\"message\", onMsg)\r\n reject(new Error(\"useTask: délai dépassé\"))\r\n }, 45_000)\r\n window.addEventListener(\"message\", onMsg)\r\n target.postMessage({ type: BRIDGE_TYPE, requestId, projectId, ...body }, \"*\")\r\n })\r\n}\r\n\r\n/**\r\n * Dès que le widget tourne dans une iframe, un `fetch` vers `/api/...` n’emporte en général\r\n * pas la session (partitionnement / contexte imbriqué). Le relais `postMessage` → fenêtre\r\n * parente (même onglet, même origine) envoie les cookies comme une navigation classique.\r\n */\r\nfunction shouldUsePostMessageBridge(): boolean {\r\n if (typeof window === \"undefined\") return false\r\n try {\r\n return window.self !== window.top\r\n } catch {\r\n return true\r\n }\r\n}\r\n\r\nasync function fetchInvoke(\r\n projectId: string,\r\n body: InvokeBridgeBody\r\n): Promise<boolean> {\r\n const res = await fetch(`/api/projects/${projectId}/invoke-task`, {\r\n method: \"POST\",\r\n credentials: \"include\",\r\n headers: { \"Content-Type\": \"application/json\" },\r\n body: JSON.stringify(body),\r\n })\r\n if (!res.ok) {\r\n const err = (await res.json().catch(() => null)) as { error?: string } | null\r\n throw new Error(err?.error ?? res.statusText)\r\n }\r\n return true\r\n}\r\n\r\n/**\r\n * Exécute une tâche Sparkle dans la runtime (équivalent à `ctx.invoke` / `ctx.invokeUntil`).\r\n * En preview Sandpack, les appels passent par `postMessage` vers l’éditeur (session).\r\n */\r\nexport function useTask() {\r\n const config = useSparkleConfig()\r\n const projectId = config.projectId\r\n\r\n const invoke = useCallback(\r\n async (taskId: string, payload?: unknown) => {\r\n if (!projectId?.trim()) {\r\n throw new Error(\"useTask: renseignez `config.projectId` sur SparkleProvider.\")\r\n }\r\n const body: InvokeBridgeBody = { kind: \"invoke\", taskId, payload }\r\n if (shouldUsePostMessageBridge()) {\r\n return bridgeInvoke(projectId, body)\r\n }\r\n return fetchInvoke(projectId, body)\r\n },\r\n [projectId]\r\n )\r\n\r\n const invokeUntil = useCallback(\r\n async (taskId: string, delaySeconds: number, payload?: unknown) => {\r\n if (!projectId?.trim()) {\r\n throw new Error(\"useTask: renseignez `config.projectId` sur SparkleProvider.\")\r\n }\r\n const body: InvokeBridgeBody = {\r\n kind: \"invokeUntil\",\r\n taskId,\r\n payload,\r\n delaySeconds,\r\n }\r\n if (shouldUsePostMessageBridge()) {\r\n return bridgeInvoke(projectId, body)\r\n }\r\n return fetchInvoke(projectId, body)\r\n },\r\n [projectId]\r\n )\r\n\r\n return useMemo(() => ({ invoke, invokeUntil }), [invoke, invokeUntil])\r\n}\r\n"],"mappings":"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","names":["React","useMemo","useEffect","useState","React","createContext","SparkleAuthContext","AuthProvider","children","config","React","useCallback","useContext","useRef","createContext","useEffect","useState","EventEmitter","z","subscribeSchema","hookSchema","unsubscribeSchema","updateSchema","messageSchema","WebSocketClient","url","data","jwt","authenticatedUrl","event","channels","newChannels","channel","hook","RealtimeContext","createContext","WebSocketProvider","children","config","values","setValues","useState","hooks","setHooks","socket","useRef","token","useContext","SparkleAuthContext","useEffect","WebSocketClient","message","data","type","payload","prev","value","num","subscribe","useCallback","keys","listen","hook","React","React","createContext","useCallback","useEffect","useRef","useState","ObsContext","ObsProvider","url","children","status","setStatus","obsRef","cancelledRef","reconnectTimer","OBSWebSocket","obs","doConnect","call","requestType","requestData","createContext","useContext","SparkleConfigContext","useSparkleConfig","cfg","ClientGate","children","ready","setReady","useState","useEffect","defaultConfig","SparkleProvider","config","mergedConfig","obsUrl","useMemo","inner","React","SparkleConfigContext","AuthProvider","WebSocketProvider","ObsProvider","useContext","useEffect","useMemo","useRealtime","key","defaultValue","context","useContext","RealtimeContext","keys","useMemo","useEffect","k","useEffect","useState","WIDGET_SESSION_REQ","WIDGET_SESSION_RES","newRequestId","isEmbedded","requestSessionViaParent","target","resolve","requestId","onMsg","ev","d","timer","fetchSessionUserId","res","data","useAuth","userId","setUserId","cancelled","uid","useContext","useEffect","SparkleHookKeys","useHook","hook","context","useContext","RealtimeContext","useEffect","ApiClient","StaticAuthProvider","useEffect","useState","useCallback","useState","useRef","useLayoutEffect","noop","isBrowser","useLocalStorage","key","initialValue","options","isBrowser","noop","deserializer","value","initializer","useRef","serializer","localStorageValue","state","setState","useState","useLayoutEffect","set","useCallback","valOrFunc","newState","remove","useLocalStorage_default","useTwitch","asUser","apiClient","setApiClient","useState","twitch","useLocalStorage_default","useEffect","ApiClient","StaticAuthProvider","useCallback","useContext","useObs","status","call","useContext","ObsContext","useObsCall","requestType","useCallback","requestData","useCallback","useMemo","BRIDGE_TYPE","BRIDGE_RESULT","newRequestId","bridgeInvoke","projectId","body","target","resolve","reject","requestId","onMsg","ev","d","timer","shouldUsePostMessageBridge","fetchInvoke","res","err","useTask","useSparkleConfig","invoke","useCallback","taskId","payload","invokeUntil","delaySeconds","useMemo"]}
1
+ {"version":3,"sources":["../src/provider.tsx","../src/providers/authProvider.tsx","../src/providers/realtimeProvider.tsx","../src/lib/websocket.ts","../src/providers/obsProvider.tsx","../src/sparkle-config-context.tsx","../src/hooks/useRealtime.ts","../src/hooks/useAuth.ts","../src/hooks/useHook.ts","../src/hooks/useTwitch.ts","../src/utils/hooks/useLocalStorage.ts","../src/utils/utils.ts","../src/hooks/useObs.ts","../src/hooks/useTask.ts"],"sourcesContent":["import React, { useMemo, type ReactNode, useEffect, useState } from \"react\"\nimport { AuthProvider } from \"./providers/authProvider\"\nimport { WebSocketProvider } from \"./providers/realtimeProvider\"\nimport { ObsProvider } from \"./providers/obsProvider\"\nimport { SparkleConfigContext } from \"./sparkle-config-context\"\nimport { SparkleConfig } from \"./utils/config\"\n\nconst ClientGate = ({ children }: { children: ReactNode }) => {\n const [ready, setReady] = useState(false)\n\n useEffect(() => {\n setReady(true)\n }, [])\n\n if (!ready) return null\n return children\n}\n\nexport interface SparkleProviderProps {\n config?: SparkleConfig\n children: ReactNode\n}\n\nconst defaultConfig: SparkleConfig = {\n userId: \"\",\n wsUrl: \"wss://sparkle-bot-v2.fly.dev/ws\",\n}\n\nexport const SparkleProvider = ({ children, config }: SparkleProviderProps) => {\n const mergedConfig = { ...defaultConfig, ...config }\n\n const obsUrl = useMemo(() => {\n if (!mergedConfig.obs?.clientId) return \"\"\n const base =\n mergedConfig.obs.bridgeUrl ?? \"wss://sparkle-bot-v2.fly.dev/obs-bridge\"\n return `${base}?role=client&clientId=${encodeURIComponent(mergedConfig.obs.clientId)}`\n }, [mergedConfig.obs?.clientId, mergedConfig.obs?.bridgeUrl])\n\n const inner = (\n <SparkleConfigContext.Provider value={mergedConfig}>\n <AuthProvider config={mergedConfig}>\n <WebSocketProvider config={mergedConfig}>\n <ClientGate>{children}</ClientGate>\n </WebSocketProvider>\n </AuthProvider>\n </SparkleConfigContext.Provider>\n )\n\n if (obsUrl) {\n return <ObsProvider url={obsUrl}>{inner}</ObsProvider>\n }\n\n return inner\n}\n","import * as React from \"react\"\r\n\r\nimport { type ReactNode, createContext } from \"react\"\r\n\r\nimport { SparkleConfig } from \"../utils/config\"\r\n\r\ninterface AuthContextType {\r\n token: string | undefined | null\r\n}\r\n\r\nexport const SparkleAuthContext = createContext<AuthContextType>({\r\n token: null,\r\n})\r\n\r\ninterface AuthProviderProps {\r\n config: SparkleConfig\r\n children: ReactNode\r\n}\r\n\r\nexport const AuthProvider = ({ children, config }: AuthProviderProps) => {\r\n return (\r\n <SparkleAuthContext.Provider\r\n value={{\r\n token: config.userId,\r\n }}\r\n >\r\n {children}\r\n </SparkleAuthContext.Provider>\r\n )\r\n}\r\n","import React, { ReactNode, useCallback, useContext, useRef } from \"react\"\r\n\r\nimport { createContext, useEffect, useState } from \"react\"\r\nimport { SparkleAuthContext } from \"./authProvider\"\r\nimport { UpdateMessage, WebSocketClient } from \"../lib/websocket\"\r\nimport { SparkleConfig } from \"../utils/config\"\r\n\r\ninterface WebSocketContextType {\r\n subscribe: (keys: string[]) => void\r\n listen: (hook: string) => void\r\n values: Record<string, any>\r\n hooks: Record<string, any>\r\n}\r\n\r\nexport const RealtimeContext = createContext<WebSocketContextType>({\r\n subscribe: () => {},\r\n listen: () => {},\r\n values: {},\r\n hooks: {},\r\n})\r\n\r\nexport type RealtimeValue = string | number | boolean | { [key: string]: any }\r\n\r\ntype State = Record<string, RealtimeValue>\r\n\r\ninterface WebSocketProviderProps {\r\n config: SparkleConfig\r\n children: ReactNode\r\n}\r\n\r\nexport const WebSocketProvider = ({\r\n children,\r\n config,\r\n}: WebSocketProviderProps) => {\r\n const [values, setValues] = useState<State>({})\r\n const [hooks, setHooks] = useState<Record<string, any>>({})\r\n\r\n const socket = useRef<WebSocketClient>(null)\r\n\r\n const { token } = useContext(SparkleAuthContext)\r\n\r\n useEffect(() => {\r\n socket.current = new WebSocketClient(config.wsUrl)\r\n\r\n socket.current.on(\"update\", (message: UpdateMessage) => {\r\n const { data, type } = message\r\n\r\n if (type === \"hook\") {\r\n const payload = JSON.parse(data.value)\r\n\r\n setHooks((prev) => ({\r\n ...prev,\r\n [payload.hook]: payload.payload,\r\n }))\r\n console.log(\"receive hooks\", payload)\r\n } else {\r\n let value: any = data.value\r\n\r\n console.log(\"value\", value)\r\n\r\n try {\r\n const num = parseFloat(data.value)\r\n if (!isNaN(num) && String(num) === String(data.value).trim()) {\r\n value = num\r\n } else {\r\n try {\r\n value = JSON.parse(data.value)\r\n } catch {\r\n value = data.value\r\n }\r\n }\r\n } catch {\r\n value = data.value\r\n }\r\n\r\n setValues((prev) => ({ ...prev, [data.key]: value }))\r\n console.log(\"receive values\", values)\r\n }\r\n })\r\n\r\n return () => {\r\n socket.current?.disconnect()\r\n }\r\n }, [])\r\n\r\n useEffect(() => {\r\n if (!socket.current || !token) return\r\n\r\n socket.current.authenticate(token)\r\n }, [token])\r\n\r\n const subscribe = useCallback(\r\n (keys: string[]) => {\r\n socket.current?.subscribe(keys)\r\n },\r\n [socket]\r\n )\r\n\r\n const listen = useCallback(\r\n (hook: string) => {\r\n socket.current?.hook(hook)\r\n },\r\n [socket]\r\n )\r\n\r\n return (\r\n <RealtimeContext.Provider\r\n value={{\r\n subscribe,\r\n listen,\r\n values,\r\n hooks,\r\n }}\r\n >\r\n {children}\r\n </RealtimeContext.Provider>\r\n )\r\n}\r\n","import EventEmitter from \"events\"\r\nimport z from \"zod\"\r\n\r\nconst subscribeSchema = z.object({\r\n type: z.literal(\"subscribe\"),\r\n channels: z.array(z.string()),\r\n})\r\n\r\nconst hookSchema = z.object({\r\n type: z.literal(\"hook\"),\r\n channels: z.array(z.string()),\r\n})\r\n\r\nconst unsubscribeSchema = z.object({\r\n type: z.literal(\"unsubscribe\"),\r\n channels: z.string(),\r\n})\r\n\r\nconst updateSchema = z.object({\r\n event: z.literal(\"update\"),\r\n type: z.enum([\"value\", \"hook\"]),\r\n data: z.object({\r\n key: z.string(),\r\n value: z.string(),\r\n }),\r\n})\r\n\r\nconst messageSchema = z.union([subscribeSchema, unsubscribeSchema, hookSchema])\r\n\r\nexport type Message = z.infer<typeof messageSchema>\r\n\r\nexport type UpdateMessage = z.infer<typeof updateSchema>\r\nexport type SubscribeMessage = z.infer<typeof subscribeSchema>\r\nexport type UnsubscribeMessage = z.infer<typeof unsubscribeSchema>\r\n\r\nexport class WebSocketClient extends EventEmitter {\r\n private socket: WebSocket | null = null\r\n private queue: Message[] = []\r\n private subscriptions: string[] = []\r\n private hooks: string[] = []\r\n private isAuthenticating = false\r\n\r\n private url: string\r\n private jwt: string | null = null\r\n\r\n constructor(url: string) {\r\n super()\r\n this.url = url\r\n }\r\n\r\n private send(data: Message) {\r\n if (this.socket?.readyState === WebSocket.OPEN) {\r\n this.socket?.send(JSON.stringify(data))\r\n } else {\r\n this.queue.push(data)\r\n }\r\n }\r\n\r\n authenticate(jwt: string) {\r\n if (this.isAuthenticating) return\r\n this.isAuthenticating = true\r\n this.jwt = jwt\r\n this.connect()\r\n }\r\n\r\n private connect() {\r\n const authenticatedUrl = this.url + \"?jwt=\" + this.jwt\r\n this.socket = new WebSocket(authenticatedUrl)\r\n\r\n this.socket.onopen = () => {\r\n this.queue.forEach((data) => this.send(data))\r\n this.queue = []\r\n }\r\n\r\n this.socket.onmessage = (event) => {\r\n const data = JSON.parse(event.data)\r\n this.emit(\"update\", data)\r\n }\r\n\r\n this.socket.onclose = () => {\r\n this.reconnect()\r\n }\r\n }\r\n\r\n reconnect() {\r\n setTimeout(() => {\r\n this.queue = [\r\n {\r\n type: \"subscribe\",\r\n channels: this.subscriptions,\r\n },\r\n {\r\n type: \"hook\",\r\n channels: this.hooks,\r\n },\r\n ]\r\n\r\n console.log(\"try reconnecting\")\r\n\r\n this.connect()\r\n }, 5000)\r\n }\r\n\r\n subscribe(channels: string[]) {\r\n const newChannels = channels.filter(\r\n (channel) => !this.subscriptions.includes(channel),\r\n )\r\n\r\n if (!newChannels.length) return\r\n this.subscriptions = [...this.subscriptions, ...newChannels]\r\n\r\n console.log(\"subscribe\", channels)\r\n\r\n this.send({\r\n type: \"subscribe\",\r\n channels,\r\n })\r\n }\r\n\r\n hook(hook: string) {\r\n const newChannels = [hook].filter(\r\n (channel) => !this.hooks.includes(channel),\r\n )\r\n\r\n if (!newChannels.length) return\r\n this.hooks = [...this.hooks, ...newChannels]\r\n\r\n this.send({\r\n type: \"hook\",\r\n channels: [hook],\r\n })\r\n }\r\n\r\n unsubscribe(channels: string) {\r\n this.send({ type: \"unsubscribe\", channels })\r\n }\r\n\r\n disconnect() {\r\n if (!this.socket) return\r\n\r\n this.socket.onclose = () => {}\r\n this.socket.close()\r\n }\r\n}\r\n","import React, {\r\n createContext,\r\n useCallback,\r\n useEffect,\r\n useRef,\r\n useState,\r\n type ReactNode,\r\n} from \"react\"\r\nimport type OBSWebSocket from \"obs-websocket-js\"\r\nimport type { OBSRequestTypes, OBSResponseTypes } from \"obs-websocket-js\"\r\n\r\ntype ObsStatus = \"disconnected\" | \"connecting\" | \"connected\" | \"error\"\r\n\r\ninterface ObsContextType {\r\n status: ObsStatus\r\n call: <T extends keyof OBSRequestTypes>(\r\n requestType: T,\r\n requestData?: OBSRequestTypes[T]\r\n ) => Promise<OBSResponseTypes[T]>\r\n}\r\n\r\nexport const ObsContext = createContext<ObsContextType>({\r\n status: \"disconnected\",\r\n call: () => Promise.reject(new Error(\"ObsProvider not mounted\")),\r\n})\r\n\r\nexport interface ObsProviderProps {\r\n url: string\r\n children: ReactNode\r\n}\r\n\r\nexport function ObsProvider({ url, children }: ObsProviderProps) {\r\n const [status, setStatus] = useState<ObsStatus>(\"disconnected\")\r\n const obsRef = useRef<OBSWebSocket | null>(null)\r\n const cancelledRef = useRef(false)\r\n\r\n useEffect(() => {\r\n if (!url) return\r\n\r\n cancelledRef.current = false\r\n let reconnectTimer: ReturnType<typeof setTimeout>\r\n\r\n const setup = async () => {\r\n const { default: OBSWebSocket } = await import(\"obs-websocket-js\")\r\n const obs = new OBSWebSocket()\r\n obsRef.current = obs\r\n\r\n const doConnect = async () => {\r\n if (cancelledRef.current) return\r\n setStatus(\"connecting\")\r\n try {\r\n await obs.connect(url)\r\n if (!cancelledRef.current) setStatus(\"connected\")\r\n } catch {\r\n if (!cancelledRef.current) {\r\n setStatus(\"error\")\r\n reconnectTimer = setTimeout(doConnect, 3000)\r\n }\r\n }\r\n }\r\n\r\n obs.on(\"ConnectionClosed\" as any, () => {\r\n if (!cancelledRef.current) {\r\n setStatus(\"disconnected\")\r\n reconnectTimer = setTimeout(doConnect, 3000)\r\n }\r\n })\r\n\r\n obs.on(\"ConnectionError\" as any, () => {\r\n if (!cancelledRef.current) setStatus(\"error\")\r\n })\r\n\r\n await doConnect()\r\n }\r\n\r\n setup()\r\n\r\n return () => {\r\n cancelledRef.current = true\r\n clearTimeout(reconnectTimer!)\r\n obsRef.current?.disconnect()\r\n obsRef.current = null\r\n setStatus(\"disconnected\")\r\n }\r\n }, [url])\r\n\r\n const call = useCallback(\r\n async <T extends keyof OBSRequestTypes>(\r\n requestType: T,\r\n requestData?: OBSRequestTypes[T]\r\n ): Promise<OBSResponseTypes[T]> => {\r\n const obs = obsRef.current\r\n if (!obs) throw new Error(\"OBS not connected\")\r\n return obs.call(requestType, requestData)\r\n },\r\n []\r\n )\r\n\r\n return (\r\n <ObsContext.Provider value={{ status, call }}>\r\n {children}\r\n </ObsContext.Provider>\r\n )\r\n}\r\n","import { createContext, useContext } from \"react\"\r\nimport type { SparkleConfig } from \"./utils/config\"\r\n\r\nexport const SparkleConfigContext = createContext<SparkleConfig | null>(null)\r\n\r\nexport function useSparkleConfig(): SparkleConfig {\r\n const cfg = useContext(SparkleConfigContext)\r\n if (!cfg) {\r\n throw new Error(\"Sparkle hooks require SparkleProvider\")\r\n }\r\n return cfg\r\n}\r\n","import { useContext, useEffect, useMemo } from \"react\"\r\nimport { RealtimeContext, RealtimeValue } from \"../providers/realtimeProvider\"\r\n\r\nexport function useRealtime<T extends RealtimeValue>(\r\n key: string,\r\n defaultValue?: T\r\n): T[] {\r\n const context = useContext(RealtimeContext)\r\n\r\n const keys = useMemo(() => [key], [key])\r\n\r\n if (!context) {\r\n throw new Error(\"You must use useRealtime inside a SparkleProvider\")\r\n }\r\n\r\n useEffect(() => {\r\n context.subscribe([key])\r\n }, [keys])\r\n\r\n return keys.map((k) => (context.values[k] ?? defaultValue) as T)\r\n}\r\n","import { useEffect, useState } from \"react\"\r\n\r\nconst WIDGET_SESSION_REQ = \"sparkle-widget-session-request\"\r\nconst WIDGET_SESSION_RES = \"sparkle-widget-session-result\"\r\n\r\nfunction newRequestId(): string {\r\n if (typeof crypto !== \"undefined\" && \"randomUUID\" in crypto) {\r\n return crypto.randomUUID()\r\n }\r\n return `${Date.now()}-${Math.random().toString(16).slice(2)}`\r\n}\r\n\r\nfunction isEmbedded(): boolean {\r\n if (typeof window === \"undefined\") return false\r\n try {\r\n return window.self !== window.top\r\n } catch {\r\n return true\r\n }\r\n}\r\n\r\nfunction parseSessionMessage(d: unknown): {\r\n userId: string | null\r\n username: string | null\r\n image: string | null\r\n} {\r\n if (!d || typeof d !== \"object\") {\r\n return { userId: null, username: null, image: null }\r\n }\r\n const o = d as Record<string, unknown>\r\n const userId =\r\n typeof o.userId === \"string\" && o.userId.trim() ? o.userId : null\r\n const username =\r\n typeof o.username === \"string\" && o.username.trim() ? o.username : null\r\n const image =\r\n typeof o.image === \"string\" && o.image.trim() ? o.image : null\r\n return { userId, username, image }\r\n}\r\n\r\nfunction requestSessionViaParent(): Promise<{\r\n userId: string | null\r\n username: string | null\r\n image: string | null\r\n}> {\r\n if (typeof window === \"undefined\") {\r\n return Promise.resolve({\r\n userId: null,\r\n username: null,\r\n image: null,\r\n })\r\n }\r\n const target = window.top ?? window.parent\r\n if (!target || target === window) {\r\n return Promise.resolve({\r\n userId: null,\r\n username: null,\r\n image: null,\r\n })\r\n }\r\n return new Promise((resolve) => {\r\n const requestId = newRequestId()\r\n const onMsg = (ev: MessageEvent) => {\r\n const d = ev.data\r\n if (!d || d.type !== WIDGET_SESSION_RES || d.requestId !== requestId)\r\n return\r\n window.removeEventListener(\"message\", onMsg)\r\n window.clearTimeout(timer)\r\n resolve(parseSessionMessage(d))\r\n }\r\n const timer = window.setTimeout(() => {\r\n window.removeEventListener(\"message\", onMsg)\r\n resolve({ userId: null, username: null, image: null })\r\n }, 12_000)\r\n window.addEventListener(\"message\", onMsg)\r\n target.postMessage({ type: WIDGET_SESSION_REQ, requestId }, \"*\")\r\n })\r\n}\r\n\r\nasync function fetchSessionPayload(): Promise<{\r\n userId: string | null\r\n username: string | null\r\n image: string | null\r\n}> {\r\n const res = await fetch(\"/api/sparkle-widget-session\", {\r\n method: \"GET\",\r\n credentials: \"include\",\r\n })\r\n if (!res.ok) {\r\n return { userId: null, username: null, image: null }\r\n }\r\n const data = await res.json().catch(() => null)\r\n return parseSessionMessage(data)\r\n}\r\n\r\nexport type UseAuthResult = {\r\n /** Identifiant Twitch (`Account.accountId`), pas l’id interne Sparkle. */\r\n userId: string | null | undefined\r\n /** `User.name` Prisma (nom affiché OAuth). */\r\n username: string | null | undefined\r\n /** `User.image` Prisma (URL avatar). */\r\n image: string | null | undefined\r\n}\r\n\r\n/**\r\n * Profil session navigateur : `userId` = id Twitch (`Account.accountId` OAuth) ;\r\n * `username` / `image` depuis `User`. Champs à `undefined` pendant le chargement.\r\n */\r\nexport function useAuth(): UseAuthResult {\r\n const [userId, setUserId] = useState<string | null | undefined>(undefined)\r\n const [username, setUsername] = useState<string | null | undefined>(undefined)\r\n const [image, setImage] = useState<string | null | undefined>(undefined)\r\n\r\n useEffect(() => {\r\n let cancelled = false\r\n const run = async () => {\r\n const p = isEmbedded()\r\n ? await requestSessionViaParent()\r\n : await fetchSessionPayload()\r\n if (!cancelled) {\r\n setUserId(p.userId)\r\n setUsername(p.username)\r\n setImage(p.image)\r\n }\r\n }\r\n void run()\r\n return () => {\r\n cancelled = true\r\n }\r\n }, [])\r\n\r\n return { userId, username, image }\r\n}\r\n","import { useContext, useEffect, useMemo } from \"react\"\r\nimport { RealtimeContext } from \"../providers/realtimeProvider\"\r\nimport { EventTypeMap } from \"./hook\"\r\n\r\nexport const SparkleHookKeys = [\r\n \"notification.follow\",\r\n \"notification.cheer\",\r\n \"notification.tips\",\r\n \"notification.subscribe\",\r\n \"notification.subscribe.end\",\r\n \"notification.subscribe.gift\",\r\n \"notification.raid\",\r\n\r\n \"chat.message\",\r\n\r\n \"stream.online\",\r\n \"stream.offline\",\r\n \"stream.update\",\r\n\r\n \"reward.create\",\r\n \"reward.update\",\r\n \"reward.remove\",\r\n\r\n \"reward.redemption.claim\",\r\n \"reward.redemption.update\",\r\n\r\n \"poll.begin\",\r\n \"poll.progress\",\r\n \"poll.end\",\r\n\r\n \"prediction.begin\",\r\n \"prediction.progress\",\r\n \"prediction.lock\",\r\n \"prediction.end\",\r\n\r\n \"moderator.add\",\r\n \"moderator.remove\",\r\n \"moderator.ban\",\r\n \"moderator.unban\",\r\n \"moderator.shield_mode.begin\",\r\n] as const\r\n\r\nexport type SparkleHookKey = (typeof SparkleHookKeys)[number]\r\n\r\nexport function useHook<T extends keyof EventTypeMap>(\r\n hook: T\r\n): EventTypeMap[T] | undefined {\r\n const context = useContext(RealtimeContext)\r\n\r\n if (!context) {\r\n throw new Error(\"You must use useHook inside a SparkleProvider\")\r\n }\r\n\r\n useEffect(() => {\r\n context.listen(hook)\r\n }, [hook])\r\n\r\n return context.hooks[hook] as EventTypeMap[T]\r\n}\r\n","import { ApiClient } from \"@twurple/api\"\r\nimport { StaticAuthProvider } from \"@twurple/auth\"\r\nimport { useEffect, useState } from \"react\"\r\nimport useLocalStorage from \"../utils/hooks/useLocalStorage\"\r\n\r\nexport function useTwitch(asUser?: string | number) {\r\n const [apiClient, setApiClient] = useState<ApiClient | null>(null)\r\n const [twitch] = useLocalStorage<string>(\"twitch\")\r\n\r\n useEffect(() => {\r\n console.log(\"env\", process.env.TWITCH_CLIENT_ID)\r\n\r\n console.log(\"init\")\r\n\r\n const apiClient = new ApiClient({\r\n authProvider: new StaticAuthProvider(\r\n \"u9lt242tz2pn5hl5x444ls2xllnvip\",\r\n twitch!,\r\n [\r\n \"bits:read\",\r\n \"channel:manage:moderators\",\r\n \"channel:manage:polls\",\r\n \"channel:manage:predictions\",\r\n \"channel:manage:redemptions\",\r\n \"channel:manage:vips\",\r\n\r\n \"channel:read:polls\",\r\n \"channel:read:predictions\",\r\n \"channel:read:redemptions\",\r\n \"channel:read:subscriptions\",\r\n \"channel:read:vips\",\r\n \"moderation:read\",\r\n \"moderator:manage:announcements\",\r\n \"channel:moderate\",\r\n \"moderator:read:blocked_terms\",\r\n \"moderator:read:chatters\",\r\n \"moderator:read:followers\",\r\n \"user:read:email\",\r\n ]\r\n ),\r\n })\r\n\r\n setApiClient(apiClient)\r\n\r\n return () => {\r\n setApiClient(null)\r\n }\r\n }, [])\r\n\r\n return {\r\n apiClient,\r\n }\r\n}\r\n","import {\r\n Dispatch,\r\n SetStateAction,\r\n useCallback,\r\n useState,\r\n useRef,\r\n useLayoutEffect,\r\n} from \"react\"\r\nimport { isBrowser, noop } from \"../utils\"\r\n\r\ntype parserOptions<T> =\r\n | {\r\n raw: true\r\n }\r\n | {\r\n raw: false\r\n serializer: (value: T) => string\r\n deserializer: (value: string) => T\r\n }\r\n\r\nconst useLocalStorage = <T>(\r\n key: string,\r\n initialValue?: T,\r\n options?: parserOptions<T>\r\n): [T | undefined, Dispatch<SetStateAction<T | undefined>>, () => void] => {\r\n if (!isBrowser) {\r\n return [initialValue as T, noop, noop]\r\n }\r\n if (!key) {\r\n throw new Error(\"useLocalStorage key may not be falsy\")\r\n }\r\n\r\n const deserializer = options\r\n ? options.raw\r\n ? (value) => value\r\n : options.deserializer\r\n : JSON.parse\r\n\r\n // eslint-disable-next-line react-hooks/rules-of-hooks\r\n const initializer = useRef((key: string) => {\r\n try {\r\n const serializer = options\r\n ? options.raw\r\n ? String\r\n : options.serializer\r\n : JSON.stringify\r\n\r\n const localStorageValue = localStorage.getItem(key)\r\n if (localStorageValue !== null) {\r\n return deserializer(localStorageValue)\r\n } else {\r\n initialValue && localStorage.setItem(key, serializer(initialValue))\r\n return initialValue\r\n }\r\n } catch {\r\n // If user is in private mode or has storage restriction\r\n // localStorage can throw. JSON.parse and JSON.stringify\r\n // can throw, too.\r\n return initialValue\r\n }\r\n })\r\n\r\n // eslint-disable-next-line react-hooks/rules-of-hooks\r\n const [state, setState] = useState<T | undefined>(() =>\r\n initializer.current(key)\r\n )\r\n\r\n // eslint-disable-next-line react-hooks/rules-of-hooks\r\n useLayoutEffect(() => setState(initializer.current(key)), [key])\r\n\r\n // eslint-disable-next-line react-hooks/rules-of-hooks\r\n const set: Dispatch<SetStateAction<T | undefined>> = useCallback(\r\n (valOrFunc) => {\r\n try {\r\n const newState =\r\n typeof valOrFunc === \"function\"\r\n ? (valOrFunc as Function)(state)\r\n : valOrFunc\r\n if (typeof newState === \"undefined\") return\r\n let value: string\r\n\r\n if (options)\r\n if (options.raw)\r\n if (typeof newState === \"string\") value = newState\r\n else value = JSON.stringify(newState)\r\n else if (options.serializer) value = options.serializer(newState)\r\n else value = JSON.stringify(newState)\r\n else value = JSON.stringify(newState)\r\n\r\n localStorage.setItem(key, value)\r\n setState(deserializer(value))\r\n } catch {\r\n // If user is in private mode or has storage restriction\r\n // localStorage can throw. Also JSON.stringify can throw.\r\n }\r\n },\r\n [key, setState]\r\n )\r\n\r\n // eslint-disable-next-line react-hooks/rules-of-hooks\r\n const remove = useCallback(() => {\r\n try {\r\n localStorage.removeItem(key)\r\n setState(undefined)\r\n } catch {\r\n // If user is in private mode or has storage restriction\r\n // localStorage can throw.\r\n }\r\n }, [key, setState])\r\n\r\n return [state, set, remove]\r\n}\r\n\r\nexport default useLocalStorage\r\n","export const noop = () => {}\r\n\r\nexport function on<T extends Window | Document | HTMLElement | EventTarget>(\r\n obj: T | null,\r\n ...args: Parameters<T[\"addEventListener\"]> | [string, Function | null, ...any]\r\n): void {\r\n if (obj && obj.addEventListener) {\r\n obj.addEventListener(\r\n ...(args as Parameters<HTMLElement[\"addEventListener\"]>)\r\n )\r\n }\r\n}\r\n\r\nexport function off<T extends Window | Document | HTMLElement | EventTarget>(\r\n obj: T | null,\r\n ...args:\r\n | Parameters<T[\"removeEventListener\"]>\r\n | [string, Function | null, ...any]\r\n): void {\r\n if (obj && obj.removeEventListener) {\r\n obj.removeEventListener(\r\n ...(args as Parameters<HTMLElement[\"removeEventListener\"]>)\r\n )\r\n }\r\n}\r\n\r\nexport const isBrowser = typeof window !== \"undefined\"\r\n\r\nexport const isNavigator = typeof navigator !== \"undefined\"\r\n","import { useCallback, useContext } from \"react\"\r\nimport type { OBSRequestTypes, OBSResponseTypes } from \"obs-websocket-js\"\r\nimport { ObsContext } from \"../providers/obsProvider\"\r\n\r\nexport type { OBSRequestTypes, OBSResponseTypes }\r\n\r\nexport function useObs() {\r\n const { status, call } = useContext(ObsContext)\r\n\r\n return {\r\n status,\r\n connected: status === \"connected\",\r\n call,\r\n }\r\n}\r\n\r\nexport function useObsCall<T extends keyof OBSRequestTypes>(requestType: T) {\r\n const { call } = useContext(ObsContext)\r\n\r\n return useCallback(\r\n (requestData?: OBSRequestTypes[T]) => call(requestType, requestData),\r\n [call, requestType]\r\n )\r\n}\r\n","import { useCallback, useMemo } from \"react\"\r\nimport { useSparkleConfig } from \"../sparkle-config-context\"\r\n\r\nconst BRIDGE_TYPE = \"sparkle-invoke-task\"\r\nconst BRIDGE_RESULT = \"sparkle-invoke-task-result\"\r\n\r\ntype InvokeBridgeBody =\r\n | { kind: \"invoke\"; taskId: string; payload?: unknown }\r\n | {\r\n kind: \"invokeUntil\"\r\n taskId: string\r\n payload?: unknown\r\n delaySeconds: number\r\n }\r\n\r\nfunction newRequestId(): string {\r\n if (typeof crypto !== \"undefined\" && \"randomUUID\" in crypto) {\r\n return crypto.randomUUID()\r\n }\r\n return `${Date.now()}-${Math.random().toString(16).slice(2)}`\r\n}\r\n\r\nfunction bridgeInvoke(\r\n projectId: string,\r\n body: InvokeBridgeBody\r\n): Promise<boolean> {\r\n if (typeof window === \"undefined\") {\r\n return Promise.reject(new Error(\"useTask requires a browser environment\"))\r\n }\r\n const target = window.top ?? window.parent\r\n if (!target || target === window) {\r\n return Promise.reject(\r\n new Error(\r\n \"useTask: aucun parent — utilisez la preview éditeur ou une page intégrée.\"\r\n )\r\n )\r\n }\r\n\r\n return new Promise((resolve, reject) => {\r\n const requestId = newRequestId()\r\n const onMsg = (ev: MessageEvent) => {\r\n const d = ev.data\r\n if (!d || d.type !== BRIDGE_RESULT || d.requestId !== requestId) return\r\n window.removeEventListener(\"message\", onMsg)\r\n window.clearTimeout(timer)\r\n if (d.ok) resolve(Boolean(d.result))\r\n else reject(new Error(typeof d.error === \"string\" ? d.error : \"invoke_failed\"))\r\n }\r\n const timer = window.setTimeout(() => {\r\n window.removeEventListener(\"message\", onMsg)\r\n reject(new Error(\"useTask: délai dépassé\"))\r\n }, 45_000)\r\n window.addEventListener(\"message\", onMsg)\r\n target.postMessage({ type: BRIDGE_TYPE, requestId, projectId, ...body }, \"*\")\r\n })\r\n}\r\n\r\n/**\r\n * Dès que le widget tourne dans une iframe, un `fetch` vers `/api/...` n’emporte en général\r\n * pas la session (partitionnement / contexte imbriqué). Le relais `postMessage` → fenêtre\r\n * parente (même onglet, même origine) envoie les cookies comme une navigation classique.\r\n */\r\nfunction shouldUsePostMessageBridge(): boolean {\r\n if (typeof window === \"undefined\") return false\r\n try {\r\n return window.self !== window.top\r\n } catch {\r\n return true\r\n }\r\n}\r\n\r\nasync function fetchInvoke(\r\n projectId: string,\r\n body: InvokeBridgeBody\r\n): Promise<boolean> {\r\n const res = await fetch(`/api/projects/${projectId}/invoke-task`, {\r\n method: \"POST\",\r\n credentials: \"include\",\r\n headers: { \"Content-Type\": \"application/json\" },\r\n body: JSON.stringify(body),\r\n })\r\n if (!res.ok) {\r\n const err = (await res.json().catch(() => null)) as { error?: string } | null\r\n throw new Error(err?.error ?? res.statusText)\r\n }\r\n return true\r\n}\r\n\r\n/**\r\n * Exécute une tâche Sparkle dans la runtime (équivalent à `ctx.invoke` / `ctx.invokeUntil`).\r\n * En preview Sandpack, les appels passent par `postMessage` vers l’éditeur (session).\r\n */\r\nexport function useTask() {\r\n const config = useSparkleConfig()\r\n const projectId = config.projectId\r\n\r\n const invoke = useCallback(\r\n async (taskId: string, payload?: unknown) => {\r\n if (!projectId?.trim()) {\r\n throw new Error(\"useTask: renseignez `config.projectId` sur SparkleProvider.\")\r\n }\r\n const body: InvokeBridgeBody = { kind: \"invoke\", taskId, payload }\r\n if (shouldUsePostMessageBridge()) {\r\n return bridgeInvoke(projectId, body)\r\n }\r\n return fetchInvoke(projectId, body)\r\n },\r\n [projectId]\r\n )\r\n\r\n const invokeUntil = useCallback(\r\n async (taskId: string, delaySeconds: number, payload?: unknown) => {\r\n if (!projectId?.trim()) {\r\n throw new Error(\"useTask: renseignez `config.projectId` sur SparkleProvider.\")\r\n }\r\n const body: InvokeBridgeBody = {\r\n kind: \"invokeUntil\",\r\n taskId,\r\n payload,\r\n delaySeconds,\r\n }\r\n if (shouldUsePostMessageBridge()) {\r\n return bridgeInvoke(projectId, body)\r\n }\r\n return fetchInvoke(projectId, body)\r\n },\r\n [projectId]\r\n )\r\n\r\n return useMemo(() => ({ invoke, invokeUntil }), [invoke, invokeUntil])\r\n}\r\n"],"mappings":"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","names":["React","useMemo","useEffect","useState","React","createContext","SparkleAuthContext","AuthProvider","children","config","React","useCallback","useContext","useRef","createContext","useEffect","useState","EventEmitter","z","subscribeSchema","hookSchema","unsubscribeSchema","updateSchema","messageSchema","WebSocketClient","url","data","jwt","authenticatedUrl","event","channels","newChannels","channel","hook","RealtimeContext","createContext","WebSocketProvider","children","config","values","setValues","useState","hooks","setHooks","socket","useRef","token","useContext","SparkleAuthContext","useEffect","WebSocketClient","message","data","type","payload","prev","value","num","subscribe","useCallback","keys","listen","hook","React","React","createContext","useCallback","useEffect","useRef","useState","ObsContext","ObsProvider","url","children","status","setStatus","obsRef","cancelledRef","reconnectTimer","OBSWebSocket","obs","doConnect","call","requestType","requestData","createContext","useContext","SparkleConfigContext","useSparkleConfig","cfg","ClientGate","children","ready","setReady","useState","useEffect","defaultConfig","SparkleProvider","config","mergedConfig","obsUrl","useMemo","inner","React","SparkleConfigContext","AuthProvider","WebSocketProvider","ObsProvider","useContext","useEffect","useMemo","useRealtime","key","defaultValue","context","useContext","RealtimeContext","keys","useMemo","useEffect","k","useEffect","useState","WIDGET_SESSION_REQ","WIDGET_SESSION_RES","newRequestId","isEmbedded","parseSessionMessage","d","o","userId","username","image","requestSessionViaParent","target","resolve","requestId","onMsg","ev","timer","fetchSessionPayload","res","data","useAuth","setUserId","setUsername","setImage","cancelled","useContext","useEffect","SparkleHookKeys","useHook","hook","context","useContext","RealtimeContext","useEffect","ApiClient","StaticAuthProvider","useEffect","useState","useCallback","useState","useRef","useLayoutEffect","noop","isBrowser","useLocalStorage","key","initialValue","options","isBrowser","noop","deserializer","value","initializer","useRef","serializer","localStorageValue","state","setState","useState","useLayoutEffect","set","useCallback","valOrFunc","newState","remove","useLocalStorage_default","useTwitch","asUser","apiClient","setApiClient","useState","twitch","useLocalStorage_default","useEffect","ApiClient","StaticAuthProvider","useCallback","useContext","useObs","status","call","useContext","ObsContext","useObsCall","requestType","useCallback","requestData","useCallback","useMemo","BRIDGE_TYPE","BRIDGE_RESULT","newRequestId","bridgeInvoke","projectId","body","target","resolve","reject","requestId","onMsg","ev","d","timer","shouldUsePostMessageBridge","fetchInvoke","res","err","useTask","useSparkleConfig","invoke","useCallback","taskId","payload","invokeUntil","delaySeconds","useMemo"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "sparkle-react",
3
- "version": "0.0.38",
3
+ "version": "0.0.40",
4
4
  "files": [
5
5
  "dist"
6
6
  ],