synxed-sdk 0.1.2 → 0.1.4
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 +97 -50
- package/dist/index.d.mts +154 -15
- package/dist/index.d.ts +154 -15
- package/dist/index.js +9 -2
- package/dist/index.js.map +1 -1
- package/dist/index.mjs +9 -2
- package/dist/index.mjs.map +1 -1
- package/package.json +2 -3
package/dist/index.mjs
CHANGED
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import
|
|
1
|
+
import*as L from'protobufjs';import {Howl}from'howler';var G=`
|
|
2
2
|
syntax = "proto3";
|
|
3
3
|
|
|
4
4
|
enum ContentKind {
|
|
@@ -6,6 +6,7 @@ enum ContentKind {
|
|
|
6
6
|
CONTENT_KIND_SONG = 1;
|
|
7
7
|
CONTENT_KIND_PLAYLIST = 2;
|
|
8
8
|
CONTENT_KIND_CATEGORY = 3;
|
|
9
|
+
CONTENT_KIND_RADIO = 4;
|
|
9
10
|
}
|
|
10
11
|
|
|
11
12
|
enum ErrorCode {
|
|
@@ -79,5 +80,11 @@ message SdkServerEnvelope {
|
|
|
79
80
|
SdkServerError error = 2;
|
|
80
81
|
}
|
|
81
82
|
}
|
|
82
|
-
`,S=E.parse(C).root,f=S.lookupType("SdkClientEnvelope"),k=S.lookupType("SdkServerEnvelope"),T=(i=>(i[i.UNSPECIFIED=0]="UNSPECIFIED",i[i.SONG=1]="SONG",i[i.PLAYLIST=2]="PLAYLIST",i[i.CATEGORY=3]="CATEGORY",i))(T||{}),I=(s=>(s[s.UNSPECIFIED=0]="UNSPECIFIED",s[s.UNAUTHORIZED=1]="UNAUTHORIZED",s[s.VALIDATION_ERROR=2]="VALIDATION_ERROR",s[s.NOT_FOUND=3]="NOT_FOUND",s[s.PROCESSING=4]="PROCESSING",s[s.SERVICE_UNAVAILABLE=5]="SERVICE_UNAVAILABLE",s[s.BAD_PROTOBUF=6]="BAD_PROTOBUF",s))(I||{});var c=class{static encodeClientEnvelope(n){let t=f.create(n);return f.encode(t).finish()}static decodeServerEnvelope(n){let t=k.decode(n);return k.toObject(t,{enums:String,longs:String,bytes:String,defaults:true,oneofs:true})}};var a=class{constructor(){this.listeners=new Map;}on(n,t){this.listeners.has(n)||this.listeners.set(n,new Set),this.listeners.get(n).add(t);}off(n,t){let e=this.listeners.get(n);e&&e.delete(t);}once(n,t){let e=((...i)=>{t(...i),this.off(n,e);});this.on(n,e);}emit(n,...t){let e=this.listeners.get(n);e&&e.forEach(i=>i(...t));}removeAllListeners(){this.listeners.clear();}};var h=class extends a{constructor(){super();this.socket=null;this.apiKey=null;this.serverUrl=null;}connect(t,e){if(this.socket?.connected)return;this.apiKey=t,this.serverUrl=e;let i=e.endsWith("/")?e.slice(0,-1):e,r=i.includes("localhost")||i.includes("127.0.0.1");this.socket=io(`${i}/sdk`,{auth:{apiKey:t},transports:r?["polling","websocket"]:["websocket","polling"],extraHeaders:{"ngrok-skip-browser-warning":"true"}}),this.socket.on("connect",()=>{this.emit("connected");}),this.socket.on("disconnect",o=>{this.emit("disconnected",o);}),this.socket.on("connect_error",o=>{this.emit("error",o);}),this.socket.on("d",o=>{try{let s=o instanceof Uint8Array?o:new Uint8Array(o),d=c.decodeServerEnvelope(s);this.emit("message",d);}catch(s){this.emit("error",new Error(`Failed to decode message: ${s}`));}});}async waitForConnection(){if(console.log("[Synxed] Waiting for connection..."),this.socket?.connected){console.log("[Synxed] Already connected");return}return new Promise((t,e)=>{let i=()=>{console.log("[Synxed] Socket connected event fired"),o(),t();},r=s=>{console.error("[Synxed] Socket connection error:",s),o(),e(s);},o=()=>{this.socket?.off("connect",i),this.socket?.off("connect_error",r);};this.socket?.once("connect",i),this.socket?.once("connect_error",r),this.socket||(console.error("[Synxed] No socket instance found"),e(new Error("Socket not initialized. Call connect() first.")));})}disconnect(){this.socket&&(this.socket.disconnect(),this.socket=null);}sendInit(t){if(!this.socket?.connected)throw new Error("Socket not connected");console.log("[Synxed] Sending init packet:",t);let e={init:t},i=c.encodeClientEnvelope(e);this.socket.emit("d",i);}sendControl(t){if(!this.socket?.connected)return;let e={control:t},i=c.encodeClientEnvelope(e);this.socket.emit("d",i);}sendAnalytics(t){if(!this.socket?.connected)return;let e={analytics:t},i=c.encodeClientEnvelope(e);this.socket.emit("d",i);}get isConnected(){return this.socket?.connected||false}};var p=class extends a{constructor(){super();this.howl=null;this.hls=null;this.audioEl=null;this.updateTimer=null;}async load(t,e){this.stop(),this.emit("loading"),console.log("[Synxed] AudioEngine loading URL:",t,"isHls:",e);let i=/^((?!chrome|android).)*safari/i.test(navigator.userAgent);if(e&&!i)try{let{default:r}=await import('hls.js');if(r.isSupported()){console.log("[Synxed] Using Hls.js for playback");let s=new URL(t).search;this.hls=new r({xhrSetup:d=>{d.setRequestHeader("ngrok-skip-browser-warning","true");}}),this.audioEl=new Audio,this.hls.loadSource(t),this.hls.attachMedia(this.audioEl),this.audioEl.onplay=()=>{this.emit("playing"),this.startTimeUpdateLoop();},this.audioEl.onpause=()=>{this.emit("paused"),this.stopTimeUpdateLoop();},this.audioEl.onended=()=>{this.emit("ended"),this.stopTimeUpdateLoop();},this.audioEl.onerror=d=>{console.error("[Synxed] HLS Audio error:",d),this.emit("error",d);},this.audioEl.onloadedmetadata=()=>{this.emit("loaded");},this.hls.on(r.Events.ERROR,(d,v)=>{v.fatal&&(console.error("[Synxed] Fatal HLS error:",v),this.emit("error",v));});return}}catch{console.warn("[Synxed] hls.js not found or error loading, falling back to native");}this.howl=new Howl({src:[t],html5:true,format:e?["m3u8"]:void 0,onload:()=>{console.log("[Synxed] AudioEngine loaded successfully"),this.emit("loaded");},onloaderror:(r,o)=>{console.error("[Synxed] AudioEngine load error:",o,"ID:",r),this.emit("error",o);},onplay:()=>{this.emit("playing"),this.startTimeUpdateLoop();},onpause:()=>{this.emit("paused"),this.stopTimeUpdateLoop();},onstop:()=>{this.emit("stopped"),this.stopTimeUpdateLoop();},onend:()=>{this.emit("ended"),this.stopTimeUpdateLoop();}});}play(){this.audioEl?this.audioEl.play().catch(t=>console.error("[Synxed] Play failed:",t)):this.howl?.play();}pause(){this.audioEl?this.audioEl.pause():this.howl?.pause();}stop(){this.hls&&(this.hls.destroy(),this.hls=null),this.audioEl&&(this.audioEl.pause(),this.audioEl.src="",this.audioEl=null),this.howl&&(this.howl.stop(),this.howl.unload(),this.howl=null),this.stopTimeUpdateLoop();}seek(t){this.audioEl?this.audioEl.currentTime=t/1e3:this.howl?.seek(t/1e3);}setVolume(t){this.audioEl&&(this.audioEl.volume=t),this.howl&&this.howl.volume(t);}get duration(){return this.audioEl?this.audioEl.duration*1e3:(this.howl?.duration()||0)*1e3}get currentTime(){return this.audioEl?this.audioEl.currentTime*1e3:(this.howl?.seek()||0)*1e3}startTimeUpdateLoop(){this.stopTimeUpdateLoop();let t=()=>{(this.audioEl?!this.audioEl.paused:this.howl?.playing())&&(this.emit("timeupdate",{currentTime:this.currentTime,duration:this.duration}),this.updateTimer=requestAnimationFrame(t));};this.updateTimer=requestAnimationFrame(t);}stopTimeUpdateLoop(){this.updateTimer!==null&&(cancelAnimationFrame(this.updateTimer),this.updateTimer=null);}};var m=class extends a{constructor(){super();this.queue=[];this.currentIndex=-1;}setQueue(t,e){this.queue=t,this.currentIndex=typeof e=="number"&&e>=0&&e<t.length?e:t.length>0?0:-1,this.emit("queueUpdated",this.queue),this.getCurrentTrack()&&this.emit("trackChanged",this.getCurrentTrack());}getCurrentTrack(){return this.currentIndex>=0&&this.currentIndex<this.queue.length?this.queue[this.currentIndex]:null}next(){if(this.currentIndex<this.queue.length-1){this.currentIndex++;let t=this.getCurrentTrack();return this.emit("trackChanged",t),t}return null}previous(){if(this.currentIndex>0){this.currentIndex--;let t=this.getCurrentTrack();return this.emit("trackChanged",t),t}return null}skipTo(t){if(t>=0&&t<this.queue.length){this.currentIndex=t;let e=this.getCurrentTrack();return this.emit("trackChanged",e),e}return null}reset(){this.queue=[],this.currentIndex=-1,this.emit("queueUpdated",[]);}get hasNext(){return this.currentIndex<this.queue.length-1}get hasPrevious(){return this.currentIndex>0}get queueSnapshot(){return [...this.queue]}get currentIndexSnapshot(){return this.currentIndex}};var u=class extends Error{constructor(t,e){super(t);this.code=e;this.name="SynxedError";}},y=class extends u{constructor(n){super(n),this.name="SynxedConnectionError";}},g=class extends u{constructor(n){super(typeof n=="string"?n:"Playback failed"),this.name="SynxedPlaybackError";}},O=class extends u{constructor(n){super(n),this.name="SynxedProtocolError";}};var x=class extends a{constructor(t){super();this.status="idle";this.volume=.8;this.config=t,this.transport=new h,this.audio=new p,this.playlist=new m,this.setupListeners(),t.autoConnect&&this.connect();}get currentTrack(){return this.playlist.getCurrentTrack()}setupListeners(){this.transport.on("connected",()=>this.emit("connected")),this.transport.on("disconnected",t=>this.emit("disconnected",t)),this.transport.on("error",t=>this.emit("error",new y(t.message))),this.transport.on("message",t=>this.handleServerMessage(t)),this.audio.on("playing",()=>this.updateStatus("playing")),this.audio.on("paused",()=>this.updateStatus("paused")),this.audio.on("stopped",()=>this.updateStatus("idle")),this.audio.on("loading",()=>this.updateStatus("loading")),this.audio.on("error",t=>this.emit("error",new g(t))),this.audio.on("timeupdate",t=>this.emit("timeUpdate",t)),this.audio.on("ended",()=>this.handleTrackEnded()),this.playlist.on("trackChanged",t=>this.emit("trackChange",t)),this.playlist.on("queueUpdated",t=>this.emit("queueUpdated",t));}connect(){this.transport.connect(this.config.apiKey,this.config.serverUrl);}async playSong(t){this.transport.isConnected||this.connect(),await this.transport.waitForConnection(),this.transport.sendInit({contentKind:1,catalogTrackId:t.catalogTrackId,internalTrackId:t.internalTrackId,listenerId:t.listenerId});}async playPlaylist(t){this.transport.isConnected||this.connect(),await this.transport.waitForConnection(),this.transport.sendInit({contentKind:2,playlistCode:t.playlistCode,listenerId:t.listenerId});}pause(){this.audio.pause(),this.transport.sendControl({action:2,positionMs:Math.floor(this.audio.currentTime*1e3)});}resume(){this.audio.play(),this.transport.sendControl({action:1,positionMs:Math.floor(this.audio.currentTime*1e3)});}stop(){this.audio.stop(),this.transport.sendControl({action:3,positionMs:Math.floor(this.audio.currentTime*1e3)});}skip(){if(this.playlist.queueSnapshot.length>1){this.transport.sendControl({action:5,positionMs:Math.floor(this.audio.currentTime*1e3)}),this.updateStatus("loading");return}let t=this.playlist.next();t&&this.playSong(t.kind==="internal"?{internalTrackId:t.id}:{catalogTrackId:t.id});}previous(){if(this.playlist.queueSnapshot.length>1){this.transport.sendControl({action:6,positionMs:Math.floor(this.audio.currentTime*1e3)}),this.updateStatus("loading");return}let t=this.playlist.previous();t&&this.playSong(t.kind==="internal"?{internalTrackId:t.id}:{catalogTrackId:t.id});}skipTo(t){let e=this.playlist.skipTo(t);e&&this.playSong(e.kind==="internal"?{internalTrackId:e.id}:{catalogTrackId:e.id});}seek(t){this.audio.seek(t),this.transport.sendControl({action:4,positionMs:t});}setVolume(t){this.volume=Math.max(0,Math.min(1,t)),this.audio.setVolume(this.volume);}emitAnalytics(t,e,i){this.transport.sendAnalytics({eventType:t,positionMs:e??Math.floor(this.audio.currentTime*1e3),extraJson:i?JSON.stringify(i):void 0});}async handleServerMessage(t){if(console.log("[Synxed] Received server message:",t),t.initAck){let e=t.initAck;if(console.log("[Synxed] Loading playback URL:",e.playbackUrl,"isHls:",e.isHls),await this.audio.load(e.playbackUrl,e.isHls),this.audio.play(),this.emitAnalytics("stream_start"),e.contentSummary)try{let i=JSON.parse(e.contentSummary);if(i.tracks&&Array.isArray(i.tracks)){let r=typeof i.currentIndex=="number"?i.currentIndex:0;this.playlist.setQueue(i.tracks,r);}}catch{}}else t.error&&this.emit("error",new u(t.error.message,t.error.code));}handleTrackEnded(){this.playlist.hasNext?this.skip():this.updateStatus("idle");}updateStatus(t){this.status=t,this.emit("stateChange",{status:t,currentTrack:this.playlist.getCurrentTrack(),currentTime:this.audio.currentTime,duration:this.audio.duration,volume:this.volume});}destroy(){this.audio.stop(),this.transport.disconnect(),this.playlist.reset(),this.removeAllListeners();}};export{T as ContentKind,I as ErrorCode,y as SynxedConnectionError,u as SynxedError,g as SynxedPlaybackError,x as SynxedPlayer,O as SynxedProtocolError};//# sourceMappingURL=index.mjs.map
|
|
83
|
+
`,A=L.parse(G).root,E=A.lookupType("SdkClientEnvelope"),T=A.lookupType("SdkServerEnvelope"),v=(s=>(s[s.UNSPECIFIED=0]="UNSPECIFIED",s[s.SONG=1]="SONG",s[s.PLAYLIST=2]="PLAYLIST",s[s.CATEGORY=3]="CATEGORY",s[s.RADIO=4]="RADIO",s))(v||{}),M=(a=>(a[a.UNSPECIFIED=0]="UNSPECIFIED",a[a.UNAUTHORIZED=1]="UNAUTHORIZED",a[a.VALIDATION_ERROR=2]="VALIDATION_ERROR",a[a.NOT_FOUND=3]="NOT_FOUND",a[a.PROCESSING=4]="PROCESSING",a[a.SERVICE_UNAVAILABLE=5]="SERVICE_UNAVAILABLE",a[a.BAD_PROTOBUF=6]="BAD_PROTOBUF",a))(M||{});var u=class{static encodeClientEnvelope(e){let t=E.create(e);return E.encode(t).finish()}static decodeServerEnvelope(e){let t=T.decode(e);return T.toObject(t,{enums:String,longs:String,bytes:String,defaults:true,oneofs:true})}};var c=class{constructor(){this.listeners=new Map;}on(e,t){this.listeners.has(e)||this.listeners.set(e,new Set),this.listeners.get(e).add(t);}off(e,t){let i=this.listeners.get(e);i&&i.delete(t);}once(e,t){let i=((...r)=>{t(...r),this.off(e,i);});this.on(e,i);}emit(e,...t){let i=this.listeners.get(e);i&&i.forEach(r=>r(...t));}removeAllListeners(){this.listeners.clear();}};var _=false;function U(n){_=n;}function p(...n){_&&console.debug("[Synxed]",...n);}function h(...n){console.warn("[Synxed]",...n);}function W(n,e){let i=(n.endsWith("/")?n.slice(0,-1):n).replace(/^http:/,"ws:").replace(/^https:/,"wss:"),r=new URL(`${i}/sdk`);return r.searchParams.set("apiKey",e),r.toString()}var g=class extends c{constructor(){super(...arguments);this.ws=null;this.connectPromise=null;}connect(t,i){if(this.ws&&(this.ws.readyState===WebSocket.OPEN||this.ws.readyState===WebSocket.CONNECTING))return;let r=W(i,t);p("Opening native WebSocket",r.replace(/apiKey=[^&]+/,"apiKey=***")),this.ws=new WebSocket(r),this.ws.binaryType="arraybuffer",this.ws.onopen=()=>{this.connectPromise=null,this.emit("connected");},this.ws.onclose=s=>{let o=s.reason||`code ${s.code}`;this.ws=null,this.connectPromise=null,this.emit("disconnected",o);},this.ws.onerror=()=>{this.emit("error",new Error("WebSocket connection error"));},this.ws.onmessage=s=>{try{let{data:o}=s;if(!(o instanceof ArrayBuffer))return;let a=u.decodeServerEnvelope(new Uint8Array(o));this.emit("message",a);}catch(o){this.emit("error",new Error(`Failed to decode message: ${o}`));}};}async waitForConnection(){if(p("Waiting for connection\u2026"),this.ws?.readyState===WebSocket.OPEN){p("Already connected");return}if(!this.ws)throw new Error("WebSocket not initialized. Call connect() first.");return this.connectPromise?this.connectPromise:(this.connectPromise=new Promise((t,i)=>{let r=this.ws;if(r.readyState===WebSocket.OPEN){t();return}let s=()=>{p("WebSocket connected"),d(),t();},o=()=>{h("WebSocket connection error"),d(),i(new Error("WebSocket connection failed"));},a=()=>{d(),i(new Error("WebSocket closed before connection was established"));},d=()=>{r.removeEventListener("open",s),r.removeEventListener("error",o),r.removeEventListener("close",a);};r.addEventListener("open",s),r.addEventListener("error",o),r.addEventListener("close",a);}),this.connectPromise)}disconnect(){this.ws&&(this.ws.onclose=null,this.ws.close(),this.ws=null,this.connectPromise=null);}sendInit(t){if(!this.isConnected)throw new Error("WebSocket not connected");p("Sending init packet",t),this.sendBytes(u.encodeClientEnvelope({init:t}));}sendControl(t){this.isConnected&&this.sendBytes(u.encodeClientEnvelope({control:t}));}sendAnalytics(t){this.isConnected&&this.sendBytes(u.encodeClientEnvelope({analytics:t}));}get isConnected(){return this.ws?.readyState===WebSocket.OPEN}sendBytes(t){if(!this.ws||this.ws.readyState!==WebSocket.OPEN)throw new Error("WebSocket not connected");this.ws.send(t);}};var x=class extends c{constructor(){super();this.howl=null;this.hls=null;this.audioEl=null;this.updateTimer=null;}async load(t,i){this.stop(),this.emit("loading"),p("AudioEngine loading",{url:t,isHls:i});let r=/^((?!chrome|android).)*safari/i.test(navigator.userAgent);if(i&&!r)try{let{default:s}=await import('hls.js');if(s.isSupported()){p("Using Hls.js for playback");let a=new URL(t).search;this.hls=new s({xhrSetup:d=>{d.setRequestHeader("ngrok-skip-browser-warning","true");}}),this.audioEl=new Audio,this.hls.loadSource(t),this.hls.attachMedia(this.audioEl),this.audioEl.onplay=()=>{this.emit("playing"),this.startTimeUpdateLoop();},this.audioEl.onpause=()=>{this.emit("paused"),this.stopTimeUpdateLoop();},this.audioEl.onended=()=>{this.emit("ended"),this.stopTimeUpdateLoop();},this.audioEl.onerror=d=>{h("HLS audio element error",d),this.emit("error",d);},this.audioEl.onloadedmetadata=()=>{this.emit("loaded");},this.hls.on(s.Events.ERROR,(d,m)=>{m.fatal&&(h("Fatal HLS error",m),this.emit("error",m));});return}}catch{h("hls.js not available, falling back to native playback");}this.howl=new Howl({src:[t],html5:true,format:i?["m3u8"]:void 0,onload:()=>{p("AudioEngine loaded"),this.emit("loaded");},onloaderror:(s,o)=>{h("AudioEngine load error",o,s),this.emit("error",o);},onplay:()=>{this.emit("playing"),this.startTimeUpdateLoop();},onpause:()=>{this.emit("paused"),this.stopTimeUpdateLoop();},onstop:()=>{this.emit("stopped"),this.stopTimeUpdateLoop();},onend:()=>{this.emit("ended"),this.stopTimeUpdateLoop();}});}play(){this.audioEl?this.audioEl.play().catch(t=>h("Play failed",t)):this.howl?.play();}pause(){this.audioEl?this.audioEl.pause():this.howl?.pause();}stop(){this.hls&&(this.hls.destroy(),this.hls=null),this.audioEl&&(this.audioEl.pause(),this.audioEl.src="",this.audioEl=null),this.howl&&(this.howl.stop(),this.howl.unload(),this.howl=null),this.stopTimeUpdateLoop();}seek(t){this.audioEl?this.audioEl.currentTime=t/1e3:this.howl?.seek(t/1e3);}setVolume(t){this.audioEl&&(this.audioEl.volume=t),this.howl&&this.howl.volume(t);}get duration(){return this.audioEl?this.audioEl.duration*1e3:(this.howl?.duration()||0)*1e3}get currentTime(){return this.audioEl?this.audioEl.currentTime*1e3:(this.howl?.seek()||0)*1e3}startTimeUpdateLoop(){this.stopTimeUpdateLoop();let t=()=>{(this.audioEl?!this.audioEl.paused:this.howl?.playing())&&(this.emit("timeupdate",{currentTime:this.currentTime,duration:this.duration}),this.updateTimer=requestAnimationFrame(t));};this.updateTimer=requestAnimationFrame(t);}stopTimeUpdateLoop(){this.updateTimer!==null&&(cancelAnimationFrame(this.updateTimer),this.updateTimer=null);}};var b=class extends c{constructor(){super();this.queue=[];this.currentIndex=-1;}setQueue(t,i){this.queue=t,this.currentIndex=typeof i=="number"&&i>=0&&i<t.length?i:t.length>0?0:-1,this.emit("queueUpdated",this.queue),this.getCurrentTrack()&&this.emit("trackChanged",this.getCurrentTrack());}getCurrentTrack(){return this.currentIndex>=0&&this.currentIndex<this.queue.length?this.queue[this.currentIndex]:null}next(){if(this.currentIndex<this.queue.length-1){this.currentIndex++;let t=this.getCurrentTrack();return this.emit("trackChanged",t),t}return null}previous(){if(this.currentIndex>0){this.currentIndex--;let t=this.getCurrentTrack();return this.emit("trackChanged",t),t}return null}skipTo(t){if(t>=0&&t<this.queue.length){this.currentIndex=t;let i=this.getCurrentTrack();return this.emit("trackChanged",i),i}return null}reset(){this.queue=[],this.currentIndex=-1,this.emit("queueUpdated",[]);}get hasNext(){return this.currentIndex<this.queue.length-1}get hasPrevious(){return this.currentIndex>0}get queueSnapshot(){return [...this.queue]}get currentIndexSnapshot(){return this.currentIndex}};var y=class extends Error{constructor(t,i){super(t);this.code=i;this.name="SynxedError";}},w=class extends y{constructor(e){super(e),this.name="SynxedConnectionError";}},S=class extends y{constructor(e){super(typeof e=="string"?e:"Playback failed"),this.name="SynxedPlaybackError";}},K=class extends y{constructor(e){super(e),this.name="SynxedProtocolError";}};function C(n,...e){for(let t of e){let i=n[t];if(typeof i=="string"&&i.length>0)return i}}function J(n,...e){for(let t of e){let i=n[t];if(typeof i=="number"&&Number.isFinite(i))return i;if(typeof i=="string"&&i!==""&&!Number.isNaN(Number(i)))return Number(i)}}function D(n){let e=n.initAck??n.init_ack;if(!e||typeof e!="object")return null;let t=e,i=C(t,"playbackUrl","playback_url");return i?{sessionId:C(t,"sessionId","session_id")??"",playbackUrl:i,isHls:!!(t.isHls??t.is_hls),heartbeatIntervalMs:J(t,"heartbeatIntervalMs","heartbeat_interval_ms")??0,contentSummary:C(t,"contentSummary","content_summary")}:null}function X(n){if(!n||typeof n!="object")return null;let e=n,t=e.id;if(typeof t!="string"||!t.trim())return null;let i=e.kind==="internal"?"internal":"catalog",r=e.albumArt??e.album_art;return {id:t.trim(),kind:i,title:typeof e.title=="string"?e.title:void 0,artist:typeof e.artist=="string"?e.artist:void 0,duration:typeof e.duration=="number"?e.duration:void 0,albumArt:typeof r=="string"?r:void 0}}function B(n,e){if(!n?.trim())return null;let t;try{t=JSON.parse(n);}catch{return null}if(!t||typeof t!="object")return null;let i=t;if(i.type==="live_radio"||e===4)return {tracks:[{id:"synxed-radio",kind:"catalog",title:typeof i.station=="string"&&i.station.trim()?i.station.trim():"Synxed Radio",artist:""}],currentIndex:0};let s=i.tracks;if(!Array.isArray(s)||s.length===0)return null;let o=[];for(let m of s){let N=X(m);N&&o.push(N);}if(o.length===0)return null;let a=typeof i.currentIndex=="number"?i.currentIndex:typeof i.current_index=="number"?i.current_index:0,d=Math.max(0,Math.min(Math.floor(a),o.length-1));return {tracks:o,currentIndex:d}}var k=class extends c{constructor(t){super();this.status="idle";this.volume=.8;this.activeContentKind=0;this.heartbeatTimer=null;this.config=t,U(!!t.debug),this.transport=new g,this.audio=new x,this.playlist=new b,this.setupListeners(),t.autoConnect&&this.connect();}get currentTrack(){return this.playlist.getCurrentTrack()}get contentKind(){return this.activeContentKind}controlPositionMs(){return this.activeContentKind===4?0:this.positionMsForAnalytics()}setupListeners(){this.transport.on("connected",()=>this.emit("connected")),this.transport.on("disconnected",t=>this.emit("disconnected",t)),this.transport.on("error",t=>this.emit("error",new w(t.message))),this.transport.on("message",t=>this.handleServerMessage(t)),this.audio.on("playing",()=>this.updateStatus("playing")),this.audio.on("paused",()=>this.updateStatus("paused")),this.audio.on("stopped",()=>this.updateStatus("idle")),this.audio.on("loading",()=>this.updateStatus("loading")),this.audio.on("error",t=>this.emit("error",new S(t))),this.audio.on("timeupdate",t=>this.emit("timeUpdate",t)),this.audio.on("ended",()=>this.handleTrackEnded()),this.playlist.on("trackChanged",t=>this.emit("trackChange",t)),this.playlist.on("queueUpdated",t=>this.emit("queueUpdated",t));}connect(){this.transport.connect(this.config.apiKey,this.config.serverUrl);}async playSong(t){this.transport.isConnected||this.connect(),await this.transport.waitForConnection(),this.activeContentKind=1,this.transport.sendInit({contentKind:1,catalogTrackId:t.catalogTrackId,internalTrackId:t.internalTrackId,listenerId:t.listenerId});}async playPlaylist(t){this.transport.isConnected||this.connect(),await this.transport.waitForConnection(),this.activeContentKind=2,this.transport.sendInit({contentKind:2,playlistCode:t.playlistCode,listenerId:t.listenerId});}async playRadio(t={}){this.transport.isConnected||this.connect(),await this.transport.waitForConnection(),this.activeContentKind=4,this.transport.sendInit({contentKind:4,listenerId:t.listenerId});}pause(){this.audio.pause(),this.transport.sendControl({action:2,positionMs:this.controlPositionMs()});}resume(){this.audio.play(),this.transport.sendControl({action:1,positionMs:this.controlPositionMs()});}stop(){this.audio.stop(),this.transport.sendControl({action:3,positionMs:this.controlPositionMs()});}skip(){if(this.activeContentKind===4)return;if(this.playlist.queueSnapshot.length>1){this.transport.sendControl({action:5,positionMs:this.controlPositionMs()}),this.updateStatus("loading");return}let t=this.playlist.next();t&&this.playSong(t.kind==="internal"?{internalTrackId:t.id}:{catalogTrackId:t.id});}previous(){if(this.activeContentKind===4)return;if(this.playlist.queueSnapshot.length>1){this.transport.sendControl({action:6,positionMs:this.controlPositionMs()}),this.updateStatus("loading");return}let t=this.playlist.previous();t&&this.playSong(t.kind==="internal"?{internalTrackId:t.id}:{catalogTrackId:t.id});}skipTo(t){if(this.activeContentKind===4)return;let i=this.playlist.skipTo(t);i&&this.playSong(i.kind==="internal"?{internalTrackId:i.id}:{catalogTrackId:i.id});}seek(t){this.activeContentKind!==4&&(this.audio.seek(t),this.transport.sendControl({action:4,positionMs:t}));}setVolume(t){this.volume=Math.max(0,Math.min(1,t)),this.audio.setVolume(this.volume);}positionMsForAnalytics(t){return typeof t=="number"?Math.max(0,Math.min(4294967295,Math.floor(t))):Math.max(0,Math.min(4294967295,Math.floor(this.audio.currentTime)))}emitAnalytics(t,i,r){this.transport.sendAnalytics({eventType:t,positionMs:this.positionMsForAnalytics(i),extraJson:r?JSON.stringify(r):void 0});}clearHeartbeat(){this.heartbeatTimer!==null&&(clearInterval(this.heartbeatTimer),this.heartbeatTimer=null);}startHeartbeat(t){this.clearHeartbeat();let i=Number(t);!i||i<1e3||(this.heartbeatTimer=setInterval(()=>{let r=this.activeContentKind===4?0:this.positionMsForAnalytics();this.emitAnalytics("heartbeat",r);},i));}applyContentSummary(t){let i=B(t,this.activeContentKind);if(i){this.playlist.setQueue(i.tracks,i.currentIndex);return}t?.trim()&&h("Could not parse contentSummary; queue metadata unavailable.");}async handleServerMessage(t){p("Received server message",t);let i=D(t);if(i){p("Loading playback",{url:i.playbackUrl,isHls:i.isHls}),await this.audio.load(i.playbackUrl,i.isHls),this.audio.play(),this.emitAnalytics("stream_start",this.activeContentKind===4?0:void 0),this.startHeartbeat(i.heartbeatIntervalMs),this.applyContentSummary(i.contentSummary);return}let r=t.error;if(r&&typeof r=="object"){let s=r,o=typeof s.message=="string"?s.message:"Unknown server error",a=typeof s.code=="string"?s.code:void 0;this.emit("error",new y(o,a));}}handleTrackEnded(){this.activeContentKind!==4&&(this.playlist.hasNext?this.skip():this.updateStatus("idle"));}updateStatus(t){this.status=t,this.emit("stateChange",{status:t,currentTrack:this.playlist.getCurrentTrack(),currentTime:this.audio.currentTime,duration:this.audio.duration,volume:this.volume});}destroy(){this.clearHeartbeat(),this.audio.stop(),this.transport.disconnect(),this.playlist.reset(),this.removeAllListeners();}};async function P(n,e){let i=`${n.replace(/\/$/,"")}/radio/now-playing`;try{let r=await fetch(i,{...e,headers:{Accept:"application/json",...e?.headers}});if(!r.ok)return null;let o=(await r.json())?.data;return !o?.title||typeof o.title!="string"?null:{title:o.title,station:typeof o.station=="string"?o.station:void 0,isLive:typeof o.isLive=="boolean"?o.isLive:void 0}}catch{return null}}function l(n,e){let t=document.createElement(n);if(e?.className&&(t.className=e.className),e?.attrs)for(let[i,r]of Object.entries(e.attrs))t.setAttribute(i,r);return e?.style&&Object.assign(t.style,e.style),e?.text&&(t.textContent=e.text),e?.html&&(t.innerHTML=e.html),e?.onClick&&t.addEventListener("click",e.onClick),e?.children&&e.children.forEach(i=>t.appendChild(i)),t}function f(n,e=20){let t=document.createElementNS("http://www.w3.org/2000/svg","svg");t.setAttribute("width",String(e)),t.setAttribute("height",String(e)),t.setAttribute("viewBox","0 0 24 24"),t.setAttribute("fill","currentColor"),t.setAttribute("aria-hidden","true");let i=document.createElementNS("http://www.w3.org/2000/svg","path");return i.setAttribute("d",n),t.appendChild(i),t}var I="M8 5v14l11-7L8 5z",H="M6 5h4v14H6V5zm8 0h4v14h-4V5z",O="M6 18l8.5-6L6 6v12zM16 6v12h2V6h-2z",z="M6 6h2v12H6V6zm11.5 12L9 12l8.5-6v12z";function F(){let n="synxed-web-player-styles";if(document.getElementById(n))return;let e=document.createElement("style");e.id=n,e.textContent=`
|
|
84
|
+
@keyframes synxed-wp-spin { from { transform: rotate(0); } to { transform: rotate(360deg); } }
|
|
85
|
+
@keyframes synxed-wp-vbar {
|
|
86
|
+
0%, 100% { transform: scaleY(0.28); opacity: 0.45; }
|
|
87
|
+
50% { transform: scaleY(1); opacity: 1; }
|
|
88
|
+
}
|
|
89
|
+
`,document.head.appendChild(e);}function V(n){let e=[" \u2014 "," \u2013 "," - "," \u2014"," \u2013"," -"];for(let t of e){let i=n.indexOf(t);if(i>0)return {title:n.slice(0,i).trim(),artist:n.slice(i+t.length).trim()}}return {title:n.trim()}}function $(n="bottom-center",e=16,t=16){let i={position:"fixed",zIndex:"50"};switch(n){case "top-left":return {...i,top:`${t}px`,left:`${e}px`};case "top-right":return {...i,top:`${t}px`,right:`${e}px`};case "bottom-left":return {...i,bottom:`${t}px`,left:`${e}px`};case "bottom-right":return {...i,bottom:`${t}px`,right:`${e}px`};default:return {...i,bottom:`${t}px`,left:"50%",transform:"translateX(-50%)"}}}function j(n={}){return {accent:n.accent??"#22c55e",accentMuted:n.accentMuted??n.accent??"#22c55e",background:n.background??"#051107",backgroundInner:n.backgroundInner??"#0a1f10",border:n.border??"rgba(34, 197, 94, 0.35)",text:n.text??"#ffffff",textMuted:n.textMuted??"rgba(255,255,255,0.35)",stationText:n.stationText??"rgba(34, 197, 94, 0.85)",liveDot:n.liveDot??"#CCFF00",glow:n.glow??"rgba(34, 197, 94, 0.35)"}}function q(n,e){n.style.setProperty("--sw-accent",e.accent),n.style.setProperty("--sw-accent-muted",e.accentMuted),n.style.setProperty("--sw-bg",e.background),n.style.setProperty("--sw-bg-inner",e.backgroundInner),n.style.setProperty("--sw-border",e.border),n.style.setProperty("--sw-text",e.text),n.style.setProperty("--sw-text-muted",e.textMuted),n.style.setProperty("--sw-station",e.stationText),n.style.setProperty("--sw-live",e.liveDot),n.style.setProperty("--sw-glow",e.glow);}var R=class n{constructor(e){this.engine=null;this.pollTimer=null;this.destroyed=false;this.isPlaying=false;this.currentTrack=null;this.nowPlaying=null;this.avatarRing=null;this.titleEl=null;this.artistInlineEl=null;this.artistBlockEl=null;this.footerEl=null;this.playBtn=null;this.options={mode:"wide",attribution:"Synxed",nowPlayingPollMs:12e3,powerByLabel:"Powered by Synxed",...e},this.theme=j(this.options.theme),this.ownsRoot=!e.container,this.root=e.container??document.createElement("div"),this.ownsRoot&&document.body.appendChild(this.root),F(),this.mountShell(),this.initEngine(),this.options.source.type==="radio"&&this.startNowPlayingPoll();}static mount(e){return new n(e)}get player(){return this.engine}get element(){return this.root}destroy(){this.destroyed||(this.destroyed=true,this.stopNowPlayingPoll(),this.engine?.destroy(),this.engine=null,this.ownsRoot&&this.root.remove());}get isRadio(){return this.options.source.type==="radio"}get mode(){return this.options.mode??"wide"}mountShell(){let{position:e={},className:t,style:i}=this.options,r=e.placement??"bottom-center",s=e.offsetX??16,o=e.offsetY??(r==="bottom-center"?24:16);this.root.className=t??"",this.root.dataset.synxedWebPlayer=this.mode,q(this.root,this.theme),Object.assign(this.root.style,$(r,s,o)),i&&Object.assign(this.root.style,i),this.root.replaceChildren(),this.mode==="mini"?this.buildMini():this.mode==="large"?this.buildLarge():this.buildWide(),this.refreshLabels(),this.setPlayingVisual(this.isPlaying);}buildAvatar(e,t=false){let i=l("div",{style:{position:"relative",flexShrink:"0"}});this.avatarRing=l("div",{style:{width:`${e}px`,height:`${e}px`,borderRadius:"50%",border:"3px solid var(--sw-accent)",padding:"3px",boxShadow:"0 0 24px var(--sw-glow), 0 0 48px rgba(0,0,0,0.25)",background:"var(--sw-bg)",boxSizing:"border-box"}});let r=l("div",{style:{width:"100%",height:"100%",borderRadius:"50%",background:"var(--sw-bg-inner)",overflow:"hidden"}});return this.options.avatarUrl?r.appendChild(l("img",{attrs:{src:this.options.avatarUrl,alt:""},style:{width:"100%",height:"100%",objectFit:"cover",objectPosition:"top",transform:this.mode==="wide"?"scale(1.15)":"scale(1.12)"}})):r.appendChild(l("div",{text:"S",style:{width:"100%",height:"100%",display:"flex",alignItems:"center",justifyContent:"center",color:"var(--sw-accent)",fontWeight:"900",fontSize:`${Math.round(e*.28)}px`}})),this.avatarRing.appendChild(r),i.appendChild(this.avatarRing),i.appendChild(l("span",{style:{position:"absolute",...t?{top:"4px",left:"4px"}:{bottom:"6px",left:"6px"},width:t?"10px":"12px",height:t?"10px":"12px",borderRadius:"50%",background:"var(--sw-live)",border:"2px solid var(--sw-bg)",boxShadow:"0 0 8px var(--sw-live)"}})),i}buildMini(){let t=l("button",{attrs:{type:"button","aria-label":this.options.onMiniClick?"Expand player":"Live DJ"},style:{position:"relative",padding:"0",border:"none",background:"transparent",cursor:this.options.onMiniClick?"pointer":"default"},onClick:()=>this.options.onMiniClick?.()});t.appendChild(this.buildAvatar(72)),this.root.appendChild(t);}buildWide(){let t="0 8px 40px var(--sw-glow), 0 0 0 1px var(--sw-border)";this.root.style.width="min(calc(100vw - 20px), 560px)";let i=l("div",{style:{position:"relative",display:"flex",alignItems:"center",width:"100%"}}),r=Math.max(90*.42,40);i.appendChild(l("div",{style:{position:"absolute",top:"8px",bottom:"8px",left:`${r}px`,right:"0",background:"var(--sw-bg)",border:"1px solid var(--sw-border)",borderRadius:"28px 20px 20px 28px",boxShadow:t,overflow:"hidden"},children:[l("div",{style:{position:"absolute",inset:"0",background:"linear-gradient(90deg, color-mix(in srgb, var(--sw-accent) 12%, transparent), transparent)",pointerEvents:"none"}})]}));let s=this.buildAvatar(90);Object.assign(s.style,{position:"relative",zIndex:"1",marginLeft:"-2px"}),i.appendChild(s);let o=l("div",{style:{position:"relative",zIndex:"1",flex:"1",minWidth:"0",marginLeft:"14px",padding:"18px 8px"}}),a=l("div",{style:{display:"flex",alignItems:"baseline",gap:"8px",overflow:"hidden"}});this.titleEl=l("h3",{style:{margin:"0",fontSize:"clamp(15px, 2.8vw, 20px)",fontWeight:"900",color:"var(--sw-text)",overflow:"hidden",textOverflow:"ellipsis",whiteSpace:"nowrap"}}),a.appendChild(this.titleEl),this.artistInlineEl=l("span",{style:{fontSize:"clamp(12px, 2.4vw, 15px)",fontWeight:"300",color:"var(--sw-text-muted)",overflow:"hidden",textOverflow:"ellipsis",whiteSpace:"nowrap",display:"none"}}),a.appendChild(this.artistInlineEl),o.appendChild(a),this.footerEl=l("p",{style:{margin:"8px 0 0",fontSize:"9px",fontWeight:"700",color:"var(--sw-text-muted)",letterSpacing:"0.18em",textTransform:"uppercase",overflow:"hidden",textOverflow:"ellipsis",whiteSpace:"nowrap"}}),o.appendChild(this.footerEl),i.appendChild(o);let d=l("div",{style:{position:"relative",zIndex:"1",flexShrink:"0",display:"flex",alignItems:"center",gap:"14px",paddingRight:"16px"}});this.playBtn=this.createPlayButton(52),d.appendChild(this.playBtn),this.isRadio||d.appendChild(l("button",{attrs:{type:"button","aria-label":"Skip"},style:{border:"none",background:"transparent",color:"var(--sw-text-muted)",display:"flex",cursor:"pointer"},onClick:()=>this.engine?.skip(),children:[f(O,22)]})),d.appendChild(this.decoLines()),i.appendChild(d),this.root.appendChild(i);}buildLarge(){this.root.style.width="min(calc(100vw - 24px), 380px)";let t=l("div",{style:{borderRadius:"24px",border:"1px solid var(--sw-border)",background:"var(--sw-bg)",boxShadow:"0 8px 40px var(--sw-glow), 0 0 0 1px var(--sw-border)",padding:"16px",display:"flex",flexDirection:"column",gap:"14px",fontFamily:"system-ui, -apple-system, Segoe UI, Roboto, sans-serif"}}),i=l("div",{style:{display:"flex",gap:"12px",alignItems:"center",minWidth:"0"}});i.appendChild(this.buildAvatar(96,true));let r=l("div",{style:{minWidth:"0",flex:"1"}});this.titleEl=l("h3",{style:{margin:"0",fontSize:"18px",fontWeight:"900",color:"var(--sw-text)",overflow:"hidden",textOverflow:"ellipsis",whiteSpace:"nowrap"}}),r.appendChild(this.titleEl),this.artistBlockEl=l("p",{style:{margin:"6px 0 0",fontSize:"13px",fontWeight:"700",color:"var(--sw-accent-muted)",letterSpacing:"0.08em",textTransform:"uppercase",overflow:"hidden",textOverflow:"ellipsis",whiteSpace:"nowrap",display:"none"}}),r.appendChild(this.artistBlockEl),this.footerEl=l("p",{style:{margin:"6px 0 0",fontSize:"10px",fontWeight:"700",color:"var(--sw-station)",letterSpacing:"0.18em",textTransform:"uppercase",overflow:"hidden",textOverflow:"ellipsis",whiteSpace:"nowrap"}}),r.appendChild(this.footerEl),i.appendChild(r),t.appendChild(i),t.appendChild(this.buildVisualizer()),t.appendChild(l("p",{text:this.options.powerByLabel??"Powered by Synxed",style:{margin:"0",fontSize:"9px",fontWeight:"600",color:"var(--sw-text-muted)",letterSpacing:"0.12em",textTransform:"uppercase",textAlign:"center"}}));let s=l("div",{style:{display:"flex",alignItems:"center",justifyContent:"center",gap:"18px"}});this.isRadio||s.appendChild(l("button",{attrs:{type:"button","aria-label":"Previous"},style:this.roundControlStyle(),onClick:()=>this.engine?.previous(),children:[f(z,22)]})),this.playBtn=this.createPlayButton(56),s.appendChild(this.playBtn),this.isRadio||s.appendChild(l("button",{attrs:{type:"button","aria-label":"Skip"},style:this.roundControlStyle(),onClick:()=>this.engine?.skip(),children:[f(O,22)]})),t.appendChild(s),this.root.appendChild(t);}roundControlStyle(){return {width:"46px",height:"46px",borderRadius:"50%",border:"1px solid var(--sw-border)",background:"rgba(255,255,255,0.04)",color:"var(--sw-text)",display:"flex",alignItems:"center",justifyContent:"center",cursor:"pointer"}}createPlayButton(e){return l("button",{attrs:{type:"button","aria-label":"Play"},style:{width:`${e}px`,height:`${e}px`,borderRadius:"50%",border:"none",background:"var(--sw-accent)",color:"#0a0a0a",display:"flex",alignItems:"center",justifyContent:"center",cursor:"pointer",boxShadow:"0 0 18px var(--sw-glow)",flexShrink:"0"},onClick:()=>this.togglePlay(),children:[f(I,e>=56?22:20)]})}buildVisualizer(){let e=l("div",{style:{display:"flex",alignItems:"flex-end",gap:"4px",height:"52px",padding:"8px 4px",borderRadius:"14px",border:"1px solid var(--sw-border)",overflow:"hidden"}});for(let t=0;t<24;t++)e.appendChild(l("span",{style:{flex:"1",minWidth:"3px",height:"100%",borderRadius:"4px",background:"linear-gradient(180deg, var(--sw-accent), transparent)",transformOrigin:"center bottom",animation:`synxed-wp-vbar ${1+t%5*.12}s ease-in-out infinite`,animationDelay:`${t*.04}s`,opacity:"0.85"}}));return e}decoLines(){let e=l("div",{style:{display:"flex",flexDirection:"column",gap:"3px",opacity:"0.22"}});return e.appendChild(l("span",{style:{display:"block",width:"16px",height:"2px",borderRadius:"999px",background:"var(--sw-text)",transform:"translateX(3px)"}})),e.appendChild(l("span",{style:{display:"block",width:"16px",height:"2px",borderRadius:"999px",background:"var(--sw-text)"}})),e}initEngine(){let{apiKey:e,serverUrl:t,source:i}=this.options,r=new k({apiKey:e,serverUrl:t,autoConnect:true});this.engine=r;let s=a=>{this.isPlaying=a.status==="playing",a.currentTrack&&(this.currentTrack=a.currentTrack),this.setPlayingVisual(this.isPlaying),this.refreshLabels();};r.on("stateChange",s),r.on("trackChange",a=>{this.currentTrack=a,this.refreshLabels();}),(i.type==="radio"?()=>r.playRadio():()=>r.playPlaylist({playlistCode:i.playlistCode}))().catch(()=>{});}startNowPlayingPoll(){let e=async()=>{let t=await P(this.options.serverUrl);!this.destroyed&&t&&(this.nowPlaying=t,this.refreshLabels());};e(),this.pollTimer=setInterval(e,this.options.nowPlayingPollMs??12e3);}stopNowPlayingPoll(){this.pollTimer!==null&&(clearInterval(this.pollTimer),this.pollTimer=null);}displayLine(){return this.isRadio&&this.nowPlaying?.title?V(this.nowPlaying.title):this.currentTrack?.title?{title:this.currentTrack.title,artist:this.currentTrack.artist}:{title:"Loading\u2026"}}refreshLabels(){let e=this.displayLine();if(this.titleEl&&(this.titleEl.textContent=e.title),this.artistInlineEl){let t=!!e.artist;this.artistInlineEl.style.display=t?"inline":"none",t&&(this.artistInlineEl.textContent=e.artist);}if(this.artistBlockEl){let t=!!e.artist;this.artistBlockEl.style.display=t?"block":"none",t&&(this.artistBlockEl.textContent=e.artist);}if(this.footerEl){let t=this.isRadio&&(this.nowPlaying?.station??this.currentTrack?.title??"Synxed Radio");this.footerEl.textContent=t&&this.isRadio?`${t} \xB7 ${this.options.attribution}`:this.options.attribution??"Synxed";}}setPlayingVisual(e){this.avatarRing&&(this.avatarRing.style.animation=e?"synxed-wp-spin 4s linear infinite":""),this.playBtn&&(this.playBtn.replaceChildren(f(e?H:I,20)),this.playBtn.setAttribute("aria-label",e?"Pause":"Play"));}togglePlay(){this.engine&&(this.isPlaying?this.engine.pause():this.engine.resume());}};export{v as ContentKind,M as ErrorCode,w as SynxedConnectionError,y as SynxedError,S as SynxedPlaybackError,k as SynxedPlayer,K as SynxedProtocolError,R as SynxedWebPlayer,g as TransportManager,W as buildSdkWebSocketUrl,P as fetchRadioNowPlaying};//# sourceMappingURL=index.mjs.map
|
|
83
90
|
//# sourceMappingURL=index.mjs.map
|
package/dist/index.mjs.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/proto/sdk-streaming.ts","../src/transport/ProtocolCodec.ts","../src/core/EventEmitter.ts","../src/transport/TransportManager.ts","../src/audio/AudioEngine.ts","../src/playlist/PlaylistManager.ts","../src/core/errors.ts","../src/core/SynxedPlayer.ts"],"names":["schema","root","SdkClientEnvelope","SdkServerEnvelope","ContentKind","ErrorCode","ProtocolCodec","payload","message","data","decoded","EventEmitter","event","handler","set","onceHandler","args","TransportManager","apiKey","serverUrl","normalizedUrl","isLocal","io","reason","error","uint8Array","err","resolve","reject","onConnect","cleanup","onError","params","bytes","AudioEngine","url","isHls","isSafari","Hls","searchParams","xhr","e","Howl","id","ms","volume","update","PlaylistManager","tracks","currentIndex","track","index","SynxedError","code","SynxedConnectionError","SynxedPlaybackError","SynxedProtocolError","SynxedPlayer","config","time","options","nextTrack","prevTrack","eventType","positionMs","extra","ack","summary","status"],"mappings":"6FAMMA,CAAAA,CAAS;AAAA;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;;AAAA;AAAA;AAAA;AAAA;;AAAA;AAAA;AAAA;AAAA;AAAA;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;;AAAA;AAAA;AAAA;AAAA;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,CAAA,CAmFTC,CAAAA,CAAgB,CAAA,CAAA,KAAA,CAAMD,CAAM,CAAA,CAAE,IAAA,CAEvBE,CAAAA,CAAoBD,CAAAA,CAAK,UAAA,CAAW,mBAAmB,CAAA,CACvDE,CAAAA,CAAoBF,CAAAA,CAAK,UAAA,CAAW,mBAAmB,CAAA,CAExDG,CAAAA,CAAAA,CAAAA,CAAAA,GACVA,CAAAA,CAAAA,CAAAA,CAAA,WAAA,CAAc,CAAA,CAAA,CAAd,aAAA,CACAA,CAAAA,CAAAA,CAAAA,CAAA,IAAA,CAAO,CAAA,CAAA,CAAP,MAAA,CACAA,CAAAA,CAAAA,CAAAA,CAAA,QAAA,CAAW,CAAA,CAAA,CAAX,UAAA,CACAA,CAAAA,CAAAA,CAAAA,CAAA,QAAA,CAAW,CAAA,CAAA,CAAX,UAAA,CAJUA,CAAAA,CAAAA,EAAAA,CAAAA,EAAA,EAAA,CAAA,CAOAC,CAAAA,CAAAA,CAAAA,CAAAA,GACVA,CAAAA,CAAAA,CAAAA,CAAA,WAAA,CAAc,CAAA,CAAA,CAAd,aAAA,CACAA,CAAAA,CAAAA,CAAAA,CAAA,YAAA,CAAe,CAAA,CAAA,CAAf,cAAA,CACAA,CAAAA,CAAAA,CAAAA,CAAA,gBAAA,CAAmB,CAAA,CAAA,CAAnB,kBAAA,CACAA,CAAAA,CAAAA,CAAAA,CAAA,SAAA,CAAY,CAAA,CAAA,CAAZ,WAAA,CACAA,CAAAA,CAAAA,CAAAA,CAAA,UAAA,CAAa,CAAA,CAAA,CAAb,YAAA,CACAA,CAAAA,CAAAA,CAAAA,CAAA,mBAAA,CAAsB,CAAA,CAAA,CAAtB,qBAAA,CACAA,CAAAA,CAAAA,CAAAA,CAAA,aAAe,CAAA,CAAA,CAAf,cAAA,CAPUA,CAAAA,CAAAA,EAAAA,CAAAA,EAAA,EAAA,ECnGL,IAAMC,CAAAA,CAAN,KAAoB,CAIzB,OAAO,oBAAA,CAAqBC,CAAAA,CAA0B,CACpD,IAAMC,CAAAA,CAAUN,CAAAA,CAAkB,MAAA,CAAOK,CAAO,CAAA,CAChD,OAAOL,CAAAA,CAAkB,MAAA,CAAOM,CAAO,CAAA,CAAE,MAAA,EAC3C,CAKA,OAAO,oBAAA,CAAqBC,CAAAA,CAAuB,CACjD,IAAMC,CAAAA,CAAUP,CAAAA,CAAkB,MAAA,CAAOM,CAAI,CAAA,CAC7C,OAAON,CAAAA,CAAkB,QAAA,CAASO,CAAAA,CAAS,CACzC,KAAA,CAAO,MAAA,CACP,KAAA,CAAO,MAAA,CACP,KAAA,CAAO,MAAA,CACP,QAAA,CAAU,IAAA,CACV,MAAA,CAAQ,IACV,CAAC,CACH,CACF,CAAA,CCtBO,IAAMC,CAAAA,CAAN,KAAuD,CAAvD,WAAA,EAAA,CACL,IAAA,CAAQ,SAAA,CAA6C,IAAI,IAAA,CAEzD,GAA2BC,CAAAA,CAAUC,CAAAA,CAA0B,CACxD,IAAA,CAAK,SAAA,CAAU,GAAA,CAAID,CAAK,CAAA,EAC3B,IAAA,CAAK,SAAA,CAAU,GAAA,CAAIA,CAAAA,CAAO,IAAI,GAAK,CAAA,CAErC,IAAA,CAAK,SAAA,CAAU,GAAA,CAAIA,CAAK,CAAA,CAAG,GAAA,CAAIC,CAAO,EACxC,CAEA,GAAA,CAA4BD,CAAAA,CAAUC,CAAAA,CAA0B,CAC9D,IAAMC,CAAAA,CAAM,IAAA,CAAK,SAAA,CAAU,GAAA,CAAIF,CAAK,CAAA,CAChCE,CAAAA,EACFA,CAAAA,CAAI,MAAA,CAAOD,CAAO,EAEtB,CAEA,IAAA,CAA6BD,CAAAA,CAAUC,CAAAA,CAA0B,CAC/D,IAAME,CAAAA,EAAe,CAAA,GAAIC,CAAAA,GAAgB,CACvCH,CAAAA,CAAQ,GAAGG,CAAI,CAAA,CACf,IAAA,CAAK,GAAA,CAAIJ,CAAAA,CAAOG,CAAwB,EAC1C,CAAA,CAAA,CACA,IAAA,CAAK,EAAA,CAAGH,CAAAA,CAAOG,CAAW,EAC5B,CAEU,IAAA,CAA6BH,CAAAA,CAAAA,GAAaI,CAAAA,CAAmC,CACrF,IAAMF,CAAAA,CAAM,IAAA,CAAK,SAAA,CAAU,GAAA,CAAIF,CAAK,CAAA,CAChCE,CAAAA,EACFA,CAAAA,CAAI,OAAA,CAASD,CAAAA,EAAYA,CAAAA,CAAQ,GAAGG,CAAI,CAAC,EAE7C,CAEA,kBAAA,EAA2B,CACzB,IAAA,CAAK,SAAA,CAAU,KAAA,GACjB,CACF,CAAA,CC1BO,IAAMC,CAAAA,CAAN,cAA+BN,CAA8B,CAKlE,WAAA,EAAc,CACZ,KAAA,EAAM,CALR,IAAA,CAAQ,MAAA,CAAwB,IAAA,CAChC,IAAA,CAAQ,MAAA,CAAwB,IAAA,CAChC,IAAA,CAAQ,SAAA,CAA2B,KAInC,CAKA,OAAA,CAAQO,CAAAA,CAAgBC,CAAAA,CAAyB,CAC/C,GAAI,IAAA,CAAK,MAAA,EAAQ,SAAA,CAAW,OAE5B,IAAA,CAAK,MAAA,CAASD,CAAAA,CACd,IAAA,CAAK,SAAA,CAAYC,CAAAA,CAEjB,IAAMC,CAAAA,CAAgBD,CAAAA,CAAU,QAAA,CAAS,GAAG,CAAA,CAAIA,CAAAA,CAAU,KAAA,CAAM,CAAA,CAAG,EAAE,CAAA,CAAIA,CAAAA,CAEnEE,CAAAA,CAAUD,CAAAA,CAAc,QAAA,CAAS,WAAW,CAAA,EAAKA,CAAAA,CAAc,QAAA,CAAS,WAAW,CAAA,CAEzF,IAAA,CAAK,MAAA,CAASE,EAAAA,CAAG,CAAA,EAAGF,CAAa,CAAA,IAAA,CAAA,CAAQ,CACvC,IAAA,CAAM,CAAE,MAAA,CAAAF,CAAO,CAAA,CAEf,UAAA,CAAYG,CAAAA,CAAU,CAAC,SAAA,CAAW,WAAW,CAAA,CAAI,CAAC,WAAA,CAAa,SAAS,CAAA,CACxE,YAAA,CAAc,CACZ,4BAAA,CAA8B,MAChC,CACF,CAAC,CAAA,CAED,IAAA,CAAK,MAAA,CAAO,EAAA,CAAG,SAAA,CAAW,IAAM,CAC9B,IAAA,CAAK,IAAA,CAAK,WAAW,EACvB,CAAC,CAAA,CAED,IAAA,CAAK,MAAA,CAAO,EAAA,CAAG,aAAeE,CAAAA,EAAW,CACvC,IAAA,CAAK,IAAA,CAAK,cAAA,CAAgBA,CAAM,EAClC,CAAC,CAAA,CAED,IAAA,CAAK,MAAA,CAAO,EAAA,CAAG,eAAA,CAAkBC,CAAAA,EAAU,CACzC,IAAA,CAAK,IAAA,CAAK,OAAA,CAASA,CAAK,EAC1B,CAAC,CAAA,CAED,IAAA,CAAK,MAAA,CAAO,EAAA,CAAG,GAAA,CAAMf,CAAAA,EAAmC,CACtD,GAAI,CACF,IAAMgB,CAAAA,CAAahB,CAAAA,YAAgB,UAAA,CAAaA,CAAAA,CAAO,IAAI,UAAA,CAAWA,CAAI,CAAA,CACpEF,CAAAA,CAAUD,CAAAA,CAAc,oBAAA,CAAqBmB,CAAU,CAAA,CAC7D,IAAA,CAAK,IAAA,CAAK,SAAA,CAAWlB,CAAO,EAC9B,CAAA,MAASmB,CAAAA,CAAK,CACZ,IAAA,CAAK,IAAA,CAAK,OAAA,CAAS,IAAI,KAAA,CAAM,CAAA,0BAAA,EAA6BA,CAAG,CAAA,CAAE,CAAC,EAClE,CACF,CAAC,EACH,CAKA,MAAM,iBAAA,EAAmC,CAEvC,GADA,OAAA,CAAQ,GAAA,CAAI,oCAAoC,CAAA,CAC5C,IAAA,CAAK,MAAA,EAAQ,SAAA,CAAW,CAC1B,OAAA,CAAQ,GAAA,CAAI,4BAA4B,CAAA,CACxC,MACF,CAEA,OAAO,IAAI,OAAA,CAAQ,CAACC,CAAAA,CAASC,CAAAA,GAAW,CACtC,IAAMC,CAAAA,CAAY,IAAM,CACtB,OAAA,CAAQ,GAAA,CAAI,uCAAuC,CAAA,CACnDC,CAAAA,EAAQ,CACRH,CAAAA,GACF,CAAA,CAEMI,CAAAA,CAAWL,CAAAA,EAAa,CAC5B,OAAA,CAAQ,KAAA,CAAM,mCAAA,CAAqCA,CAAG,CAAA,CACtDI,CAAAA,EAAQ,CACRF,CAAAA,CAAOF,CAAG,EACZ,CAAA,CAEMI,CAAAA,CAAU,IAAM,CACpB,IAAA,CAAK,MAAA,EAAQ,GAAA,CAAI,SAAA,CAAWD,CAAS,CAAA,CACrC,IAAA,CAAK,MAAA,EAAQ,IAAI,eAAA,CAAiBE,CAAO,EAC3C,CAAA,CAEA,IAAA,CAAK,MAAA,EAAQ,IAAA,CAAK,SAAA,CAAWF,CAAS,CAAA,CACtC,IAAA,CAAK,MAAA,EAAQ,IAAA,CAAK,eAAA,CAAiBE,CAAO,CAAA,CAErC,IAAA,CAAK,MAAA,GACR,OAAA,CAAQ,KAAA,CAAM,mCAAmC,CAAA,CACjDH,CAAAA,CAAO,IAAI,KAAA,CAAM,+CAA+C,CAAC,CAAA,EAErE,CAAC,CACH,CAKA,UAAA,EAAmB,CACb,IAAA,CAAK,MAAA,GACP,IAAA,CAAK,MAAA,CAAO,UAAA,EAAW,CACvB,IAAA,CAAK,MAAA,CAAS,IAAA,EAElB,CAKA,QAAA,CAASI,CAAAA,CAAmB,CAC1B,GAAI,CAAC,IAAA,CAAK,MAAA,EAAQ,SAAA,CAChB,MAAM,IAAI,KAAA,CAAM,sBAAsB,CAAA,CAGxC,OAAA,CAAQ,GAAA,CAAI,+BAAA,CAAiCA,CAAM,CAAA,CACnD,IAAMzB,CAAAA,CAAU,CAAE,KAAMyB,CAAO,CAAA,CACzBC,CAAAA,CAAQ3B,CAAAA,CAAc,oBAAA,CAAqBC,CAAO,CAAA,CACxD,IAAA,CAAK,MAAA,CAAO,IAAA,CAAK,GAAA,CAAK0B,CAAK,EAC7B,CAKA,WAAA,CAAYD,CAAAA,CAAmB,CAC7B,GAAI,CAAC,IAAA,CAAK,MAAA,EAAQ,SAAA,CAAW,OAC7B,IAAMzB,CAAAA,CAAU,CAAE,OAAA,CAASyB,CAAO,CAAA,CAC5BC,CAAAA,CAAQ3B,CAAAA,CAAc,oBAAA,CAAqBC,CAAO,CAAA,CACxD,IAAA,CAAK,MAAA,CAAO,IAAA,CAAK,GAAA,CAAK0B,CAAK,EAC7B,CAKA,aAAA,CAAcD,CAAAA,CAAmB,CAC/B,GAAI,CAAC,IAAA,CAAK,MAAA,EAAQ,SAAA,CAAW,OAC7B,IAAMzB,CAAAA,CAAU,CAAE,SAAA,CAAWyB,CAAO,CAAA,CAC9BC,CAAAA,CAAQ3B,CAAAA,CAAc,oBAAA,CAAqBC,CAAO,CAAA,CACxD,IAAA,CAAK,MAAA,CAAO,IAAA,CAAK,IAAK0B,CAAK,EAC7B,CAKA,IAAI,WAAA,EAAuB,CACzB,OAAO,IAAA,CAAK,MAAA,EAAQ,SAAA,EAAa,KACnC,CACF,CAAA,CC3IO,IAAMC,CAAAA,CAAN,cAA0BvB,CAA0B,CAMzD,WAAA,EAAc,CACZ,KAAA,EAAM,CANR,IAAA,CAAQ,IAAA,CAAoB,IAAA,CAC5B,IAAA,CAAQ,GAAA,CAAW,IAAA,CACnB,IAAA,CAAQ,OAAA,CAAmC,IAAA,CAC3C,IAAA,CAAQ,WAAA,CAA6B,KAIrC,CAMA,MAAM,IAAA,CAAKwB,CAAAA,CAAaC,CAAAA,CAA+B,CACrD,IAAA,CAAK,IAAA,EAAK,CACV,IAAA,CAAK,IAAA,CAAK,SAAS,CAAA,CAEnB,OAAA,CAAQ,GAAA,CAAI,mCAAA,CAAqCD,CAAAA,CAAK,QAAA,CAAUC,CAAK,CAAA,CAGrE,IAAMC,CAAAA,CAAW,gCAAA,CAAiC,IAAA,CAAK,SAAA,CAAU,SAAS,EAE1E,GAAID,CAAAA,EAAS,CAACC,CAAAA,CACZ,GAAI,CACF,GAAM,CAAE,OAAA,CAASC,CAAI,CAAA,CAAI,MAAM,OAAO,QAAQ,CAAA,CAC9C,GAAIA,CAAAA,CAAI,WAAA,EAAY,CAAG,CACrB,OAAA,CAAQ,GAAA,CAAI,oCAAoC,CAAA,CAIhD,IAAMC,CAAAA,CADS,IAAI,GAAA,CAAIJ,CAAG,CAAA,CACE,MAAA,CAE5B,IAAA,CAAK,GAAA,CAAM,IAAIG,CAAAA,CAAI,CACjB,QAAA,CAAWE,CAAAA,EAAQ,CACjBA,CAAAA,CAAI,gBAAA,CAAiB,4BAAA,CAA8B,MAAM,EAC3D,CACF,CAAC,CAAA,CACD,IAAA,CAAK,OAAA,CAAU,IAAI,KAAA,CACnB,IAAA,CAAK,GAAA,CAAI,UAAA,CAAWL,CAAG,CAAA,CACvB,IAAA,CAAK,GAAA,CAAI,WAAA,CAAY,IAAA,CAAK,OAAO,CAAA,CAEjC,IAAA,CAAK,OAAA,CAAQ,OAAS,IAAM,CAC1B,IAAA,CAAK,IAAA,CAAK,SAAS,CAAA,CACnB,IAAA,CAAK,mBAAA,GACP,CAAA,CACA,IAAA,CAAK,OAAA,CAAQ,OAAA,CAAU,IAAM,CAC3B,IAAA,CAAK,IAAA,CAAK,QAAQ,CAAA,CAClB,IAAA,CAAK,kBAAA,GACP,CAAA,CACA,IAAA,CAAK,OAAA,CAAQ,OAAA,CAAU,IAAM,CAC3B,IAAA,CAAK,IAAA,CAAK,OAAO,CAAA,CACjB,IAAA,CAAK,kBAAA,GACP,CAAA,CACA,IAAA,CAAK,OAAA,CAAQ,OAAA,CAAWM,CAAAA,EAAM,CAC5B,OAAA,CAAQ,KAAA,CAAM,2BAAA,CAA6BA,CAAC,CAAA,CAC5C,IAAA,CAAK,IAAA,CAAK,OAAA,CAASA,CAAC,EACtB,CAAA,CACA,IAAA,CAAK,OAAA,CAAQ,gBAAA,CAAmB,IAAM,CACpC,IAAA,CAAK,IAAA,CAAK,QAAQ,EACpB,CAAA,CAEA,IAAA,CAAK,GAAA,CAAI,GAAGH,CAAAA,CAAI,MAAA,CAAO,KAAA,CAAO,CAAC1B,CAAAA,CAAYH,CAAAA,GAAc,CACnDA,CAAAA,CAAK,KAAA,GACP,OAAA,CAAQ,KAAA,CAAM,2BAAA,CAA6BA,CAAI,CAAA,CAC/C,IAAA,CAAK,IAAA,CAAK,OAAA,CAASA,CAAI,CAAA,EAE3B,CAAC,CAAA,CAED,MACF,CACF,CAAA,KAAY,CACV,OAAA,CAAQ,IAAA,CAAK,oEAAoE,EACnF,CAIF,IAAA,CAAK,IAAA,CAAO,IAAIiC,IAAAA,CAAK,CACnB,GAAA,CAAK,CAACP,CAAG,CAAA,CACT,KAAA,CAAO,IAAA,CACP,MAAA,CAAQC,CAAAA,CAAQ,CAAC,MAAM,CAAA,CAAI,MAAA,CAC3B,MAAA,CAAQ,IAAM,CACZ,OAAA,CAAQ,GAAA,CAAI,0CAA0C,CAAA,CACtD,IAAA,CAAK,IAAA,CAAK,QAAQ,EACpB,CAAA,CACA,WAAA,CAAa,CAACO,CAAAA,CAAIjB,CAAAA,GAAQ,CACxB,QAAQ,KAAA,CAAM,kCAAA,CAAoCA,CAAAA,CAAK,KAAA,CAAOiB,CAAE,CAAA,CAChE,IAAA,CAAK,IAAA,CAAK,OAAA,CAASjB,CAAG,EACxB,CAAA,CACA,MAAA,CAAQ,IAAM,CACZ,IAAA,CAAK,IAAA,CAAK,SAAS,CAAA,CACnB,IAAA,CAAK,mBAAA,GACP,CAAA,CACA,OAAA,CAAS,IAAM,CACb,IAAA,CAAK,IAAA,CAAK,QAAQ,CAAA,CAClB,IAAA,CAAK,kBAAA,GACP,CAAA,CACA,MAAA,CAAQ,IAAM,CACZ,IAAA,CAAK,IAAA,CAAK,SAAS,CAAA,CACnB,IAAA,CAAK,kBAAA,GACP,CAAA,CACA,KAAA,CAAO,IAAM,CACX,IAAA,CAAK,IAAA,CAAK,OAAO,CAAA,CACjB,IAAA,CAAK,kBAAA,GACP,CACF,CAAC,EACH,CAEA,IAAA,EAAa,CACP,IAAA,CAAK,OAAA,CACP,IAAA,CAAK,QAAQ,IAAA,EAAK,CAAE,KAAA,CAAMe,CAAAA,EAAK,OAAA,CAAQ,KAAA,CAAM,uBAAA,CAAyBA,CAAC,CAAC,CAAA,CAExE,IAAA,CAAK,IAAA,EAAM,IAAA,GAEf,CAEA,KAAA,EAAc,CACR,IAAA,CAAK,OAAA,CACP,IAAA,CAAK,OAAA,CAAQ,KAAA,EAAM,CAEnB,IAAA,CAAK,IAAA,EAAM,KAAA,GAEf,CAEA,IAAA,EAAa,CACP,IAAA,CAAK,GAAA,GACP,IAAA,CAAK,GAAA,CAAI,OAAA,EAAQ,CACjB,IAAA,CAAK,GAAA,CAAM,IAAA,CAAA,CAET,IAAA,CAAK,OAAA,GACP,IAAA,CAAK,OAAA,CAAQ,KAAA,EAAM,CACnB,IAAA,CAAK,OAAA,CAAQ,GAAA,CAAM,EAAA,CACnB,IAAA,CAAK,OAAA,CAAU,IAAA,CAAA,CAEb,IAAA,CAAK,IAAA,GACP,IAAA,CAAK,IAAA,CAAK,IAAA,EAAK,CACf,IAAA,CAAK,IAAA,CAAK,MAAA,EAAO,CACjB,IAAA,CAAK,IAAA,CAAO,MAEd,IAAA,CAAK,kBAAA,GACP,CAEA,IAAA,CAAKG,CAAAA,CAAkB,CACjB,IAAA,CAAK,OAAA,CACP,IAAA,CAAK,OAAA,CAAQ,WAAA,CAAcA,CAAAA,CAAK,GAAA,CAEhC,IAAA,CAAK,IAAA,EAAM,IAAA,CAAKA,CAAAA,CAAK,GAAI,EAE7B,CAEA,SAAA,CAAUC,CAAAA,CAAsB,CAC1B,IAAA,CAAK,OAAA,GACP,IAAA,CAAK,OAAA,CAAQ,MAAA,CAASA,CAAAA,CAAAA,CAEpB,IAAA,CAAK,IAAA,EACP,IAAA,CAAK,IAAA,CAAK,MAAA,CAAOA,CAAM,EAE3B,CAEA,IAAI,QAAA,EAAmB,CACrB,OAAI,IAAA,CAAK,OAAA,CAAgB,IAAA,CAAK,OAAA,CAAQ,QAAA,CAAW,GAAA,CAAA,CACzC,IAAA,CAAK,IAAA,EAAM,QAAA,EAAS,EAAK,CAAA,EAAK,GACxC,CAEA,IAAI,WAAA,EAAsB,CACxB,OAAI,IAAA,CAAK,OAAA,CAAgB,IAAA,CAAK,OAAA,CAAQ,WAAA,CAAc,KAC5C,IAAA,CAAK,IAAA,EAAM,IAAA,EAAK,EAAe,CAAA,EAAK,GAC9C,CAEQ,mBAAA,EAA4B,CAClC,IAAA,CAAK,kBAAA,EAAmB,CACxB,IAAMC,CAAAA,CAAS,IAAM,CAAA,CACD,IAAA,CAAK,OAAA,CAAU,CAAC,IAAA,CAAK,OAAA,CAAQ,MAAA,CAAS,IAAA,CAAK,IAAA,EAAM,OAAA,EAAQ,IAEzE,IAAA,CAAK,IAAA,CAAK,YAAA,CAAc,CACtB,WAAA,CAAa,IAAA,CAAK,WAAA,CAClB,QAAA,CAAU,IAAA,CAAK,QACjB,CAAC,CAAA,CACD,IAAA,CAAK,WAAA,CAAc,qBAAA,CAAsBA,CAAM,CAAA,EAEnD,CAAA,CACA,IAAA,CAAK,WAAA,CAAc,qBAAA,CAAsBA,CAAM,EACjD,CAEQ,kBAAA,EAA2B,CAC7B,IAAA,CAAK,WAAA,GAAgB,IAAA,GACvB,oBAAA,CAAqB,IAAA,CAAK,WAAW,CAAA,CACrC,IAAA,CAAK,WAAA,CAAc,IAAA,EAEvB,CACF,CAAA,CCpMO,IAAMC,CAAAA,CAAN,cAA8BpC,CAA6B,CAIhE,WAAA,EAAc,CACZ,KAAA,EAAM,CAJR,IAAA,CAAQ,KAAA,CAAqB,EAAC,CAC9B,IAAA,CAAQ,YAAA,CAAuB,GAI/B,CAEA,QAAA,CAASqC,CAAAA,CAAqBC,CAAAA,CAA6B,CACzD,IAAA,CAAK,KAAA,CAAQD,CAAAA,CACb,IAAA,CAAK,YAAA,CACH,OAAOC,CAAAA,EAAiB,QAAA,EACxBA,CAAAA,EAAgB,CAAA,EAChBA,CAAAA,CAAeD,CAAAA,CAAO,MAAA,CAClBC,CAAAA,CACAD,CAAAA,CAAO,MAAA,CAAS,CAAA,CACd,CAAA,CACA,EAAA,CACR,IAAA,CAAK,IAAA,CAAK,cAAA,CAAgB,IAAA,CAAK,KAAK,CAAA,CAChC,IAAA,CAAK,eAAA,EAAgB,EACvB,IAAA,CAAK,IAAA,CAAK,cAAA,CAAgB,IAAA,CAAK,eAAA,EAAkB,EAErD,CAEA,eAAA,EAAoC,CAClC,OAAI,IAAA,CAAK,YAAA,EAAgB,CAAA,EAAK,IAAA,CAAK,YAAA,CAAe,KAAK,KAAA,CAAM,MAAA,CACpD,IAAA,CAAK,KAAA,CAAM,IAAA,CAAK,YAAY,CAAA,CAE9B,IACT,CAEA,IAAA,EAAyB,CACvB,GAAI,IAAA,CAAK,YAAA,CAAe,IAAA,CAAK,KAAA,CAAM,MAAA,CAAS,CAAA,CAAG,CAC7C,IAAA,CAAK,YAAA,EAAA,CACL,IAAME,CAAAA,CAAQ,IAAA,CAAK,eAAA,EAAgB,CACnC,OAAA,IAAA,CAAK,IAAA,CAAK,cAAA,CAAgBA,CAAM,CAAA,CACzBA,CACT,CACA,OAAO,IACT,CAEA,QAAA,EAA6B,CAC3B,GAAI,IAAA,CAAK,YAAA,CAAe,CAAA,CAAG,CACzB,IAAA,CAAK,YAAA,EAAA,CACL,IAAMA,CAAAA,CAAQ,IAAA,CAAK,eAAA,EAAgB,CACnC,OAAA,IAAA,CAAK,IAAA,CAAK,cAAA,CAAgBA,CAAM,CAAA,CACzBA,CACT,CACA,OAAO,IACT,CAEA,MAAA,CAAOC,CAAAA,CAAiC,CACtC,GAAIA,GAAS,CAAA,EAAKA,CAAAA,CAAQ,IAAA,CAAK,KAAA,CAAM,MAAA,CAAQ,CAC3C,IAAA,CAAK,YAAA,CAAeA,CAAAA,CACpB,IAAMD,CAAAA,CAAQ,IAAA,CAAK,eAAA,EAAgB,CACnC,OAAA,IAAA,CAAK,IAAA,CAAK,cAAA,CAAgBA,CAAM,CAAA,CACzBA,CACT,CACA,OAAO,IACT,CAEA,KAAA,EAAc,CACZ,IAAA,CAAK,KAAA,CAAQ,EAAC,CACd,IAAA,CAAK,YAAA,CAAe,EAAA,CACpB,IAAA,CAAK,IAAA,CAAK,cAAA,CAAgB,EAAE,EAC9B,CAEA,IAAI,OAAA,EAAmB,CACrB,OAAO,IAAA,CAAK,YAAA,CAAe,IAAA,CAAK,KAAA,CAAM,MAAA,CAAS,CACjD,CAEA,IAAI,WAAA,EAAuB,CACzB,OAAO,IAAA,CAAK,YAAA,CAAe,CAC7B,CAEA,IAAI,aAAA,EAA6B,CAC/B,OAAO,CAAC,GAAG,IAAA,CAAK,KAAK,CACvB,CAEA,IAAI,oBAAA,EAA+B,CACjC,OAAO,IAAA,CAAK,YACd,CACF,CAAA,CC1FO,IAAME,CAAAA,CAAN,cAA0B,KAAM,CACrC,WAAA,CAAY5C,CAAAA,CAAwB6C,CAAAA,CAAe,CACjD,KAAA,CAAM7C,CAAO,CAAA,CADqB,IAAA,CAAA,IAAA,CAAA6C,CAAAA,CAElC,IAAA,CAAK,IAAA,CAAO,cACd,CACF,CAAA,CAEaC,CAAAA,CAAN,cAAoCF,CAAY,CACrD,WAAA,CAAY5C,CAAAA,CAAiB,CAC3B,KAAA,CAAMA,CAAO,CAAA,CACb,IAAA,CAAK,IAAA,CAAO,wBACd,CACF,CAAA,CAEa+C,CAAAA,CAAN,cAAkCH,CAAY,CACnD,WAAA,CAAY5B,CAAAA,CAAY,CACtB,KAAA,CAAM,OAAOA,CAAAA,EAAU,QAAA,CAAWA,CAAAA,CAAQ,iBAAiB,CAAA,CAC3D,IAAA,CAAK,IAAA,CAAO,sBACd,CACF,EAEagC,CAAAA,CAAN,cAAkCJ,CAAY,CACnD,WAAA,CAAY5C,CAAAA,CAAiB,CAC3B,KAAA,CAAMA,CAAO,CAAA,CACb,IAAA,CAAK,IAAA,CAAO,sBACd,CACF,EClBO,IAAMiD,CAAAA,CAAN,cAA2B9C,CAA2B,CAQ3D,WAAA,CAAY+C,CAAAA,CAAsB,CAChC,KAAA,EAAM,CAJR,IAAA,CAAQ,MAAA,CAAgC,MAAA,CACxC,IAAA,CAAQ,MAAA,CAAS,EAAA,CAIf,IAAA,CAAK,MAAA,CAASA,CAAAA,CACd,IAAA,CAAK,SAAA,CAAY,IAAIzC,CAAAA,CACrB,IAAA,CAAK,KAAA,CAAQ,IAAIiB,CAAAA,CACjB,IAAA,CAAK,QAAA,CAAW,IAAIa,CAAAA,CAEpB,IAAA,CAAK,cAAA,EAAe,CAEhBW,CAAAA,CAAO,WAAA,EACT,IAAA,CAAK,OAAA,GAET,CAEA,IAAI,YAAA,EAAiC,CACnC,OAAO,IAAA,CAAK,QAAA,CAAS,eAAA,EACvB,CAEQ,gBAAuB,CAE7B,IAAA,CAAK,SAAA,CAAU,EAAA,CAAG,WAAA,CAAa,IAAM,IAAA,CAAK,IAAA,CAAK,WAAW,CAAC,CAAA,CAC3D,IAAA,CAAK,SAAA,CAAU,EAAA,CAAG,cAAA,CAAiBnC,CAAAA,EAAW,IAAA,CAAK,IAAA,CAAK,cAAA,CAAgBA,CAAM,CAAC,CAAA,CAC/E,IAAA,CAAK,SAAA,CAAU,EAAA,CAAG,OAAA,CAAUC,CAAAA,EAAU,IAAA,CAAK,IAAA,CAAK,OAAA,CAAS,IAAI8B,CAAAA,CAAsB9B,CAAAA,CAAM,OAAO,CAAC,CAAC,CAAA,CAClG,IAAA,CAAK,SAAA,CAAU,EAAA,CAAG,SAAA,CAAYjB,CAAAA,EAAY,IAAA,CAAK,mBAAA,CAAoBA,CAAO,CAAC,CAAA,CAG3E,IAAA,CAAK,KAAA,CAAM,EAAA,CAAG,SAAA,CAAW,IAAM,IAAA,CAAK,YAAA,CAAa,SAAS,CAAC,CAAA,CAC3D,IAAA,CAAK,KAAA,CAAM,EAAA,CAAG,QAAA,CAAU,IAAM,IAAA,CAAK,YAAA,CAAa,QAAQ,CAAC,CAAA,CACzD,IAAA,CAAK,KAAA,CAAM,EAAA,CAAG,SAAA,CAAW,IAAM,IAAA,CAAK,YAAA,CAAa,MAAM,CAAC,CAAA,CACxD,IAAA,CAAK,KAAA,CAAM,EAAA,CAAG,SAAA,CAAW,IAAM,IAAA,CAAK,YAAA,CAAa,SAAS,CAAC,CAAA,CAC3D,IAAA,CAAK,KAAA,CAAM,EAAA,CAAG,OAAA,CAAUiB,CAAAA,EAAU,IAAA,CAAK,IAAA,CAAK,OAAA,CAAS,IAAI+B,CAAAA,CAAoB/B,CAAK,CAAC,CAAC,CAAA,CACpF,IAAA,CAAK,KAAA,CAAM,EAAA,CAAG,YAAA,CAAemC,CAAAA,EAAS,IAAA,CAAK,IAAA,CAAK,YAAA,CAAcA,CAAI,CAAC,CAAA,CACnE,IAAA,CAAK,KAAA,CAAM,EAAA,CAAG,OAAA,CAAS,IAAM,IAAA,CAAK,gBAAA,EAAkB,CAAA,CAGpD,IAAA,CAAK,QAAA,CAAS,EAAA,CAAG,cAAA,CAAiBT,CAAAA,EAAU,IAAA,CAAK,IAAA,CAAK,aAAA,CAAeA,CAAK,CAAC,CAAA,CAC3E,IAAA,CAAK,QAAA,CAAS,EAAA,CAAG,cAAA,CAAiBF,CAAAA,EAAW,IAAA,CAAK,IAAA,CAAK,cAAA,CAAgBA,CAAM,CAAC,EAChF,CAEQ,OAAA,EAAgB,CACtB,IAAA,CAAK,SAAA,CAAU,OAAA,CAAQ,IAAA,CAAK,MAAA,CAAO,MAAA,CAAQ,IAAA,CAAK,MAAA,CAAO,SAAS,EAClE,CAEA,MAAM,QAAA,CAASY,CAAAA,CAAyC,CACjD,IAAA,CAAK,SAAA,CAAU,WAAA,EAAa,IAAA,CAAK,OAAA,EAAQ,CAC9C,MAAM,IAAA,CAAK,SAAA,CAAU,iBAAA,EAAkB,CAEvC,IAAA,CAAK,SAAA,CAAU,QAAA,CAAS,CACtB,WAAA,CAAA,CAAA,CACA,cAAA,CAAgBA,CAAAA,CAAQ,cAAA,CACxB,eAAA,CAAiBA,CAAAA,CAAQ,eAAA,CACzB,UAAA,CAAYA,CAAAA,CAAQ,UACtB,CAAC,EACH,CAEA,MAAM,YAAA,CAAaA,CAAAA,CAA6C,CACzD,IAAA,CAAK,UAAU,WAAA,EAAa,IAAA,CAAK,OAAA,EAAQ,CAC9C,MAAM,IAAA,CAAK,SAAA,CAAU,iBAAA,EAAkB,CAEvC,IAAA,CAAK,SAAA,CAAU,QAAA,CAAS,CACtB,WAAA,CAAA,CAAA,CACA,YAAA,CAAcA,CAAAA,CAAQ,YAAA,CACtB,UAAA,CAAYA,CAAAA,CAAQ,UACtB,CAAC,EACH,CAEA,KAAA,EAAc,CACZ,IAAA,CAAK,KAAA,CAAM,KAAA,EAAM,CACjB,IAAA,CAAK,SAAA,CAAU,WAAA,CAAY,CACzB,MAAA,CAAA,CAAA,CACA,UAAA,CAAY,IAAA,CAAK,KAAA,CAAM,IAAA,CAAK,KAAA,CAAM,WAAA,CAAc,GAAI,CACtD,CAAC,EACH,CAEA,MAAA,EAAe,CACb,IAAA,CAAK,KAAA,CAAM,IAAA,EAAK,CAChB,IAAA,CAAK,SAAA,CAAU,WAAA,CAAY,CACzB,MAAA,CAAA,CAAA,CACA,UAAA,CAAY,IAAA,CAAK,KAAA,CAAM,IAAA,CAAK,KAAA,CAAM,WAAA,CAAc,GAAI,CACtD,CAAC,EACH,CAEA,IAAA,EAAa,CACX,IAAA,CAAK,KAAA,CAAM,IAAA,EAAK,CAChB,IAAA,CAAK,SAAA,CAAU,WAAA,CAAY,CACzB,MAAA,CAAA,CAAA,CACA,UAAA,CAAY,IAAA,CAAK,KAAA,CAAM,IAAA,CAAK,KAAA,CAAM,WAAA,CAAc,GAAI,CACtD,CAAC,EACH,CAEA,IAAA,EAAa,CAEX,GAAI,IAAA,CAAK,QAAA,CAAS,aAAA,CAAc,MAAA,CAAS,CAAA,CAAG,CAC1C,IAAA,CAAK,SAAA,CAAU,WAAA,CAAY,CACzB,MAAA,CAAA,CAAA,CACA,UAAA,CAAY,IAAA,CAAK,KAAA,CAAM,IAAA,CAAK,KAAA,CAAM,WAAA,CAAc,GAAI,CACtD,CAAC,CAAA,CACD,IAAA,CAAK,YAAA,CAAa,SAAS,CAAA,CAC3B,MACF,CAGA,IAAMC,CAAAA,CAAY,IAAA,CAAK,QAAA,CAAS,IAAA,EAAK,CACjCA,CAAAA,EACF,IAAA,CAAK,QAAA,CACHA,CAAAA,CAAU,OAAS,UAAA,CACf,CAAE,eAAA,CAAiBA,CAAAA,CAAU,EAAG,CAAA,CAChC,CAAE,cAAA,CAAgBA,CAAAA,CAAU,EAAG,CACrC,EAEJ,CAEA,QAAA,EAAiB,CACf,GAAI,IAAA,CAAK,QAAA,CAAS,aAAA,CAAc,MAAA,CAAS,CAAA,CAAG,CAC1C,IAAA,CAAK,SAAA,CAAU,WAAA,CAAY,CACzB,MAAA,CAAA,CAAA,CACA,UAAA,CAAY,IAAA,CAAK,KAAA,CAAM,IAAA,CAAK,KAAA,CAAM,WAAA,CAAc,GAAI,CACtD,CAAC,CAAA,CACD,IAAA,CAAK,YAAA,CAAa,SAAS,CAAA,CAC3B,MACF,CAEA,IAAMC,CAAAA,CAAY,IAAA,CAAK,QAAA,CAAS,QAAA,EAAS,CACrCA,CAAAA,EACF,IAAA,CAAK,QAAA,CACHA,CAAAA,CAAU,IAAA,GAAS,UAAA,CACf,CAAE,eAAA,CAAiBA,CAAAA,CAAU,EAAG,CAAA,CAChC,CAAE,cAAA,CAAgBA,CAAAA,CAAU,EAAG,CACrC,EAEJ,CAEA,MAAA,CAAOX,CAAAA,CAAqB,CAC1B,IAAMD,CAAAA,CAAQ,IAAA,CAAK,QAAA,CAAS,MAAA,CAAOC,CAAK,CAAA,CACpCD,CAAAA,EACF,IAAA,CAAK,QAAA,CACHA,CAAAA,CAAM,IAAA,GAAS,UAAA,CACX,CAAE,eAAA,CAAiBA,CAAAA,CAAM,EAAG,CAAA,CAC5B,CAAE,cAAA,CAAgBA,CAAAA,CAAM,EAAG,CACjC,EAEJ,CAEA,IAAA,CAAKN,CAAAA,CAAkB,CACrB,IAAA,CAAK,KAAA,CAAM,IAAA,CAAKA,CAAE,CAAA,CAClB,IAAA,CAAK,SAAA,CAAU,WAAA,CAAY,CACzB,MAAA,CAAA,CAAA,CACA,UAAA,CAAYA,CACd,CAAC,EACH,CAEA,SAAA,CAAUC,CAAAA,CAAsB,CAC9B,IAAA,CAAK,MAAA,CAAS,IAAA,CAAK,GAAA,CAAI,CAAA,CAAG,IAAA,CAAK,GAAA,CAAI,CAAA,CAAGA,CAAM,CAAC,CAAA,CAC7C,IAAA,CAAK,KAAA,CAAM,UAAU,IAAA,CAAK,MAAM,EAClC,CAEQ,aAAA,CAAckB,CAAAA,CAAmBC,CAAAA,CAAqBC,CAAAA,CAAmB,CAC/E,IAAA,CAAK,SAAA,CAAU,aAAA,CAAc,CAC3B,SAAA,CAAAF,CAAAA,CACA,UAAA,CAAYC,CAAAA,EAAc,IAAA,CAAK,KAAA,CAAM,IAAA,CAAK,KAAA,CAAM,WAAA,CAAc,GAAI,CAAA,CAClE,SAAA,CAAWC,CAAAA,CAAQ,IAAA,CAAK,SAAA,CAAUA,CAAK,CAAA,CAAI,MAC7C,CAAC,EACH,CAEA,MAAc,mBAAA,CAAoB1D,CAAAA,CAA6B,CAE7D,GADA,OAAA,CAAQ,GAAA,CAAI,mCAAA,CAAqCA,CAAO,CAAA,CACpDA,CAAAA,CAAQ,OAAA,CAAS,CACnB,IAAM2D,CAAAA,CAAM3D,CAAAA,CAAQ,OAAA,CAOpB,GANA,OAAA,CAAQ,GAAA,CAAI,gCAAA,CAAkC2D,CAAAA,CAAI,WAAA,CAAa,QAAA,CAAUA,CAAAA,CAAI,KAAK,CAAA,CAClF,MAAM,IAAA,CAAK,KAAA,CAAM,KAAKA,CAAAA,CAAI,WAAA,CAAaA,CAAAA,CAAI,KAAK,CAAA,CAChD,IAAA,CAAK,KAAA,CAAM,IAAA,EAAK,CAChB,IAAA,CAAK,aAAA,CAAc,cAAc,CAAA,CAG7BA,CAAAA,CAAI,cAAA,CACN,GAAI,CACF,IAAMC,CAAAA,CAAU,IAAA,CAAK,KAAA,CAAMD,CAAAA,CAAI,cAAc,CAAA,CAC7C,GAAIC,CAAAA,CAAQ,MAAA,EAAU,KAAA,CAAM,OAAA,CAAQA,CAAAA,CAAQ,MAAM,CAAA,CAAG,CACnD,IAAMlB,CAAAA,CAAe,OAAOkB,CAAAA,CAAQ,YAAA,EAAiB,QAAA,CAAWA,CAAAA,CAAQ,YAAA,CAAe,CAAA,CACvF,IAAA,CAAK,QAAA,CAAS,QAAA,CAASA,CAAAA,CAAQ,MAAA,CAAuBlB,CAAY,EACpE,CACF,CAAA,KAAY,CAEZ,CAEJ,CAAA,KAAW1C,CAAAA,CAAQ,KAAA,EACjB,IAAA,CAAK,IAAA,CAAK,OAAA,CAAS,IAAI6C,CAAAA,CAAY7C,CAAAA,CAAQ,KAAA,CAAM,OAAA,CAASA,EAAQ,KAAA,CAAM,IAAI,CAAC,EAEjF,CAEQ,gBAAA,EAAyB,CAC3B,IAAA,CAAK,QAAA,CAAS,OAAA,CAChB,IAAA,CAAK,IAAA,EAAK,CAEV,IAAA,CAAK,YAAA,CAAa,MAAM,EAE5B,CAEQ,YAAA,CAAa6D,CAAAA,CAAqC,CACxD,IAAA,CAAK,MAAA,CAASA,CAAAA,CACd,IAAA,CAAK,IAAA,CAAK,aAAA,CAAe,CACvB,MAAA,CAAAA,CAAAA,CACA,YAAA,CAAc,IAAA,CAAK,SAAS,eAAA,EAAgB,CAC5C,WAAA,CAAa,IAAA,CAAK,KAAA,CAAM,WAAA,CACxB,QAAA,CAAU,IAAA,CAAK,KAAA,CAAM,QAAA,CACrB,MAAA,CAAQ,IAAA,CAAK,MACf,CAAC,EACH,CAEA,OAAA,EAAgB,CACd,IAAA,CAAK,KAAA,CAAM,IAAA,EAAK,CAChB,IAAA,CAAK,SAAA,CAAU,UAAA,EAAW,CAC1B,IAAA,CAAK,QAAA,CAAS,KAAA,EAAM,CACpB,IAAA,CAAK,kBAAA,GACP,CACF","file":"index.mjs","sourcesContent":["import * as protobuf from 'protobufjs';\n\n/**\n * Protobuf schema definition for the SDK.\n * Matches the backend contract in src/modules/streaming/sdk-streaming.proto\n */\nconst schema = `\nsyntax = \"proto3\";\n\nenum ContentKind {\n CONTENT_KIND_UNSPECIFIED = 0;\n CONTENT_KIND_SONG = 1;\n CONTENT_KIND_PLAYLIST = 2;\n CONTENT_KIND_CATEGORY = 3;\n}\n\nenum ErrorCode {\n ERROR_CODE_UNSPECIFIED = 0;\n UNAUTHORIZED = 1;\n VALIDATION_ERROR = 2;\n NOT_FOUND = 3;\n PROCESSING = 4;\n SERVICE_UNAVAILABLE = 5;\n BAD_PROTOBUF = 6;\n}\n\nenum SdkControlAction {\n CONTROL_UNSPECIFIED = 0;\n CONTROL_PLAY = 1;\n CONTROL_PAUSE = 2;\n CONTROL_STOP = 3;\n CONTROL_SEEK = 4;\n CONTROL_SKIP_NEXT = 5;\n CONTROL_SKIP_PREVIOUS = 6;\n}\n\nmessage SdkClientInit {\n ContentKind contentKind = 1;\n string catalogTrackId = 2;\n string internalTrackId = 3;\n string playlistCode = 4;\n string categoryQuery = 5;\n string listenerId = 6;\n string deviceType = 7;\n string countryCode = 8;\n string region = 9;\n string protocolVersion = 10;\n}\n\nmessage SdkClientControl {\n SdkControlAction action = 1;\n uint32 positionMs = 2;\n}\n\nmessage SdkClientAnalytics {\n string eventType = 1;\n uint32 positionMs = 2;\n string extraJson = 3;\n}\n\nmessage SdkClientEnvelope {\n oneof payload {\n SdkClientInit init = 1;\n SdkClientControl control = 2;\n SdkClientAnalytics analytics = 3;\n }\n}\n\nmessage SdkServerInitAck {\n string sessionId = 1;\n string playbackUrl = 2;\n bool isHls = 3;\n uint32 heartbeatIntervalMs = 4;\n string contentSummary = 5;\n}\n\nmessage SdkServerError {\n ErrorCode code = 1;\n string message = 2;\n}\n\nmessage SdkServerEnvelope {\n oneof payload {\n SdkServerInitAck initAck = 1;\n SdkServerError error = 2;\n }\n}\n`;\n\nconst root = protobuf.parse(schema).root;\n\nexport const SdkClientEnvelope = root.lookupType('SdkClientEnvelope');\nexport const SdkServerEnvelope = root.lookupType('SdkServerEnvelope');\n\nexport enum ContentKind {\n UNSPECIFIED = 0,\n SONG = 1,\n PLAYLIST = 2,\n CATEGORY = 3,\n}\n\nexport enum ErrorCode {\n UNSPECIFIED = 0,\n UNAUTHORIZED = 1,\n VALIDATION_ERROR = 2,\n NOT_FOUND = 3,\n PROCESSING = 4,\n SERVICE_UNAVAILABLE = 5,\n BAD_PROTOBUF = 6,\n}\n\nexport enum SdkControlAction {\n CONTROL_UNSPECIFIED = 0,\n CONTROL_PLAY = 1,\n CONTROL_PAUSE = 2,\n CONTROL_STOP = 3,\n CONTROL_SEEK = 4,\n CONTROL_SKIP_NEXT = 5,\n CONTROL_SKIP_PREVIOUS = 6,\n}\n","import { SdkClientEnvelope, SdkServerEnvelope } from '../proto/sdk-streaming';\n\nexport class ProtocolCodec {\n /**\n * Encodes a client envelope into a binary payload.\n */\n static encodeClientEnvelope(payload: any): Uint8Array {\n const message = SdkClientEnvelope.create(payload);\n return SdkClientEnvelope.encode(message).finish();\n }\n\n /**\n * Decodes a binary payload into a server envelope.\n */\n static decodeServerEnvelope(data: Uint8Array): any {\n const decoded = SdkServerEnvelope.decode(data);\n return SdkServerEnvelope.toObject(decoded, {\n enums: String,\n longs: String,\n bytes: String,\n defaults: true,\n oneofs: true,\n });\n }\n}\n","type Handler = (...args: any[]) => void;\n\nexport class EventEmitter<Events extends Record<string, any>> {\n private listeners: Map<keyof Events, Set<Handler>> = new Map();\n\n on<K extends keyof Events>(event: K, handler: Events[K]): void {\n if (!this.listeners.has(event)) {\n this.listeners.set(event, new Set());\n }\n this.listeners.get(event)!.add(handler);\n }\n\n off<K extends keyof Events>(event: K, handler: Events[K]): void {\n const set = this.listeners.get(event);\n if (set) {\n set.delete(handler);\n }\n }\n\n once<K extends keyof Events>(event: K, handler: Events[K]): void {\n const onceHandler = ((...args: any[]) => {\n handler(...args);\n this.off(event, onceHandler as Events[K]);\n }) as Events[K];\n this.on(event, onceHandler);\n }\n\n protected emit<K extends keyof Events>(event: K, ...args: Parameters<Events[K]>): void {\n const set = this.listeners.get(event);\n if (set) {\n set.forEach((handler) => handler(...args));\n }\n }\n\n removeAllListeners(): void {\n this.listeners.clear();\n }\n}\n","import { io, Socket } from 'socket.io-client';\nimport { ProtocolCodec } from './ProtocolCodec';\nimport { EventEmitter } from '../core/EventEmitter';\n\nexport interface TransportEvents {\n connected: () => void;\n disconnected: (reason: string) => void;\n error: (error: any) => void;\n message: (payload: any) => void;\n}\n\nexport class TransportManager extends EventEmitter<TransportEvents> {\n private socket: Socket | null = null;\n private apiKey: string | null = null;\n private serverUrl: string | null = null;\n\n constructor() {\n super();\n }\n\n /**\n * Connects to the Synxed SDK namespace.\n */\n connect(apiKey: string, serverUrl: string): void {\n if (this.socket?.connected) return;\n\n this.apiKey = apiKey;\n this.serverUrl = serverUrl;\n\n const normalizedUrl = serverUrl.endsWith('/') ? serverUrl.slice(0, -1) : serverUrl;\n\n const isLocal = normalizedUrl.includes('localhost') || normalizedUrl.includes('127.0.0.1');\n\n this.socket = io(`${normalizedUrl}/sdk`, {\n auth: { apiKey },\n // Use websocket first for production to bypass CORS pre-flight blocks\n transports: isLocal ? ['polling', 'websocket'] : ['websocket', 'polling'],\n extraHeaders: {\n \"ngrok-skip-browser-warning\": \"true\"\n }\n });\n\n this.socket.on('connect', () => {\n this.emit('connected');\n });\n\n this.socket.on('disconnect', (reason) => {\n this.emit('disconnected', reason);\n });\n\n this.socket.on('connect_error', (error) => {\n this.emit('error', error);\n });\n\n this.socket.on('d', (data: ArrayBuffer | Uint8Array) => {\n try {\n const uint8Array = data instanceof Uint8Array ? data : new Uint8Array(data);\n const payload = ProtocolCodec.decodeServerEnvelope(uint8Array);\n this.emit('message', payload);\n } catch (err) {\n this.emit('error', new Error(`Failed to decode message: ${err}`));\n }\n });\n }\n\n /**\n * Returns a promise that resolves when the socket is connected.\n */\n async waitForConnection(): Promise<void> {\n console.log('[Synxed] Waiting for connection...');\n if (this.socket?.connected) {\n console.log('[Synxed] Already connected');\n return;\n }\n \n return new Promise((resolve, reject) => {\n const onConnect = () => {\n console.log('[Synxed] Socket connected event fired');\n cleanup();\n resolve();\n };\n \n const onError = (err: any) => {\n console.error('[Synxed] Socket connection error:', err);\n cleanup();\n reject(err);\n };\n \n const cleanup = () => {\n this.socket?.off('connect', onConnect);\n this.socket?.off('connect_error', onError);\n };\n\n this.socket?.once('connect', onConnect);\n this.socket?.once('connect_error', onError);\n \n if (!this.socket) {\n console.error('[Synxed] No socket instance found');\n reject(new Error('Socket not initialized. Call connect() first.'));\n }\n });\n }\n\n /**\n * Disconnects from the server.\n */\n disconnect(): void {\n if (this.socket) {\n this.socket.disconnect();\n this.socket = null;\n }\n }\n\n /**\n * Sends an init message to the backend.\n */\n sendInit(params: any): void {\n if (!this.socket?.connected) {\n throw new Error('Socket not connected');\n }\n\n console.log('[Synxed] Sending init packet:', params);\n const payload = { init: params };\n const bytes = ProtocolCodec.encodeClientEnvelope(payload);\n this.socket.emit('d', bytes);\n }\n\n /**\n * Sends a control message.\n */\n sendControl(params: any): void {\n if (!this.socket?.connected) return;\n const payload = { control: params };\n const bytes = ProtocolCodec.encodeClientEnvelope(payload);\n this.socket.emit('d', bytes);\n }\n\n /**\n * Sends an analytics message.\n */\n sendAnalytics(params: any): void {\n if (!this.socket?.connected) return;\n const payload = { analytics: params };\n const bytes = ProtocolCodec.encodeClientEnvelope(payload);\n this.socket.emit('d', bytes);\n }\n\n /**\n * Checks if the socket is connected.\n */\n get isConnected(): boolean {\n return this.socket?.connected || false;\n }\n}\n","import { Howl } from 'howler';\nimport { EventEmitter } from '../core/EventEmitter';\n\nexport interface AudioEvents {\n playing: () => void;\n paused: () => void;\n stopped: () => void;\n ended: () => void;\n loading: () => void;\n loaded: () => void;\n error: (error: any) => void;\n timeupdate: (data: { currentTime: number; duration: number }) => void;\n}\n\nexport class AudioEngine extends EventEmitter<AudioEvents> {\n private howl: Howl | null = null;\n private hls: any = null;\n private audioEl: HTMLAudioElement | null = null;\n private updateTimer: number | null = null;\n\n constructor() {\n super();\n }\n\n\n /**\n * Loads a track URL.\n */\n async load(url: string, isHls: boolean): Promise<void> {\n this.stop();\n this.emit('loading');\n\n console.log('[Synxed] AudioEngine loading URL:', url, 'isHls:', isHls);\n\n // Safari has native HLS support. Other browsers need hls.js.\n const isSafari = /^((?!chrome|android).)*safari/i.test(navigator.userAgent);\n\n if (isHls && !isSafari) {\n try {\n const { default: Hls } = await import('hls.js');\n if (Hls.isSupported()) {\n console.log('[Synxed] Using Hls.js for playback');\n \n // Extract query parameters from the manifest URL to forward to segments\n const urlObj = new URL(url);\n const searchParams = urlObj.search;\n\n this.hls = new Hls({\n xhrSetup: (xhr) => {\n xhr.setRequestHeader(\"ngrok-skip-browser-warning\", \"true\");\n }\n });\n this.audioEl = new Audio();\n this.hls.loadSource(url);\n this.hls.attachMedia(this.audioEl);\n \n this.audioEl.onplay = () => {\n this.emit('playing');\n this.startTimeUpdateLoop();\n };\n this.audioEl.onpause = () => {\n this.emit('paused');\n this.stopTimeUpdateLoop();\n };\n this.audioEl.onended = () => {\n this.emit('ended');\n this.stopTimeUpdateLoop();\n };\n this.audioEl.onerror = (e) => {\n console.error('[Synxed] HLS Audio error:', e);\n this.emit('error', e);\n };\n this.audioEl.onloadedmetadata = () => {\n this.emit('loaded');\n };\n\n this.hls.on(Hls.Events.ERROR, (event: any, data: any) => {\n if (data.fatal) {\n console.error('[Synxed] Fatal HLS error:', data);\n this.emit('error', data);\n }\n });\n\n return;\n }\n } catch (e) {\n console.warn('[Synxed] hls.js not found or error loading, falling back to native');\n }\n }\n\n // Fallback to Howler\n this.howl = new Howl({\n src: [url],\n html5: true, \n format: isHls ? ['m3u8'] : undefined,\n onload: () => {\n console.log('[Synxed] AudioEngine loaded successfully');\n this.emit('loaded');\n },\n onloaderror: (id, err) => {\n console.error('[Synxed] AudioEngine load error:', err, 'ID:', id);\n this.emit('error', err);\n },\n onplay: () => {\n this.emit('playing');\n this.startTimeUpdateLoop();\n },\n onpause: () => {\n this.emit('paused');\n this.stopTimeUpdateLoop();\n },\n onstop: () => {\n this.emit('stopped');\n this.stopTimeUpdateLoop();\n },\n onend: () => {\n this.emit('ended');\n this.stopTimeUpdateLoop();\n },\n });\n }\n\n play(): void {\n if (this.audioEl) {\n this.audioEl.play().catch(e => console.error('[Synxed] Play failed:', e));\n } else {\n this.howl?.play();\n }\n }\n\n pause(): void {\n if (this.audioEl) {\n this.audioEl.pause();\n } else {\n this.howl?.pause();\n }\n }\n\n stop(): void {\n if (this.hls) {\n this.hls.destroy();\n this.hls = null;\n }\n if (this.audioEl) {\n this.audioEl.pause();\n this.audioEl.src = '';\n this.audioEl = null;\n }\n if (this.howl) {\n this.howl.stop();\n this.howl.unload();\n this.howl = null;\n }\n this.stopTimeUpdateLoop();\n }\n\n seek(ms: number): void {\n if (this.audioEl) {\n this.audioEl.currentTime = ms / 1000;\n } else {\n this.howl?.seek(ms / 1000);\n }\n }\n\n setVolume(volume: number): void {\n if (this.audioEl) {\n this.audioEl.volume = volume;\n }\n if (this.howl) {\n this.howl.volume(volume);\n }\n }\n\n get duration(): number {\n if (this.audioEl) return this.audioEl.duration * 1000;\n return (this.howl?.duration() || 0) * 1000;\n }\n\n get currentTime(): number {\n if (this.audioEl) return this.audioEl.currentTime * 1000;\n return (this.howl?.seek() as number || 0) * 1000;\n }\n\n private startTimeUpdateLoop(): void {\n this.stopTimeUpdateLoop();\n const update = () => {\n const isPlaying = this.audioEl ? !this.audioEl.paused : this.howl?.playing();\n if (isPlaying) {\n this.emit('timeupdate', {\n currentTime: this.currentTime,\n duration: this.duration,\n });\n this.updateTimer = requestAnimationFrame(update);\n }\n };\n this.updateTimer = requestAnimationFrame(update);\n }\n\n private stopTimeUpdateLoop(): void {\n if (this.updateTimer !== null) {\n cancelAnimationFrame(this.updateTimer);\n this.updateTimer = null;\n }\n }\n}\n","import { EventEmitter } from '../core/EventEmitter';\nimport { TrackInfo } from '../types';\n\nexport interface PlaylistEvents {\n trackChanged: (track: TrackInfo) => void;\n queueUpdated: (tracks: TrackInfo[]) => void;\n}\n\nexport class PlaylistManager extends EventEmitter<PlaylistEvents> {\n private queue: TrackInfo[] = [];\n private currentIndex: number = -1;\n\n constructor() {\n super();\n }\n\n setQueue(tracks: TrackInfo[], currentIndex?: number): void {\n this.queue = tracks;\n this.currentIndex =\n typeof currentIndex === 'number' &&\n currentIndex >= 0 &&\n currentIndex < tracks.length\n ? currentIndex\n : tracks.length > 0\n ? 0\n : -1;\n this.emit('queueUpdated', this.queue);\n if (this.getCurrentTrack()) {\n this.emit('trackChanged', this.getCurrentTrack()!);\n }\n }\n\n getCurrentTrack(): TrackInfo | null {\n if (this.currentIndex >= 0 && this.currentIndex < this.queue.length) {\n return this.queue[this.currentIndex];\n }\n return null;\n }\n\n next(): TrackInfo | null {\n if (this.currentIndex < this.queue.length - 1) {\n this.currentIndex++;\n const track = this.getCurrentTrack();\n this.emit('trackChanged', track!);\n return track;\n }\n return null;\n }\n\n previous(): TrackInfo | null {\n if (this.currentIndex > 0) {\n this.currentIndex--;\n const track = this.getCurrentTrack();\n this.emit('trackChanged', track!);\n return track;\n }\n return null;\n }\n\n skipTo(index: number): TrackInfo | null {\n if (index >= 0 && index < this.queue.length) {\n this.currentIndex = index;\n const track = this.getCurrentTrack();\n this.emit('trackChanged', track!);\n return track;\n }\n return null;\n }\n\n reset(): void {\n this.queue = [];\n this.currentIndex = -1;\n this.emit('queueUpdated', []);\n }\n\n get hasNext(): boolean {\n return this.currentIndex < this.queue.length - 1;\n }\n\n get hasPrevious(): boolean {\n return this.currentIndex > 0;\n }\n\n get queueSnapshot(): TrackInfo[] {\n return [...this.queue];\n }\n\n get currentIndexSnapshot(): number {\n return this.currentIndex;\n }\n}\n","export class SynxedError extends Error {\n constructor(message: string, public code?: string) {\n super(message);\n this.name = 'SynxedError';\n }\n}\n\nexport class SynxedConnectionError extends SynxedError {\n constructor(message: string) {\n super(message);\n this.name = 'SynxedConnectionError';\n }\n}\n\nexport class SynxedPlaybackError extends SynxedError {\n constructor(error: any) {\n super(typeof error === 'string' ? error : 'Playback failed');\n this.name = 'SynxedPlaybackError';\n }\n}\n\nexport class SynxedProtocolError extends SynxedError {\n constructor(message: string) {\n super(message);\n this.name = 'SynxedProtocolError';\n }\n}\n","import { TransportManager } from '../transport/TransportManager';\nimport { AudioEngine } from '../audio/AudioEngine';\nimport { PlaylistManager } from '../playlist/PlaylistManager';\nimport { EventEmitter } from './EventEmitter';\nimport { ContentKind, SdkControlAction } from '../proto/sdk-streaming';\nimport { SynxedConfig, PlayerState, PlaySongOptions, PlayPlaylistOptions, SynxedEvents, TrackInfo } from '../types';\nimport { SynxedError, SynxedPlaybackError, SynxedConnectionError } from './errors';\n\nexport class SynxedPlayer extends EventEmitter<SynxedEvents> {\n private transport: TransportManager;\n private audio: AudioEngine;\n private playlist: PlaylistManager;\n private config: SynxedConfig;\n private status: PlayerState['status'] = 'idle';\n private volume = 0.8;\n\n constructor(config: SynxedConfig) {\n super();\n this.config = config;\n this.transport = new TransportManager();\n this.audio = new AudioEngine();\n this.playlist = new PlaylistManager();\n\n this.setupListeners();\n\n if (config.autoConnect) {\n this.connect();\n }\n }\n\n get currentTrack(): TrackInfo | null {\n return this.playlist.getCurrentTrack();\n }\n\n private setupListeners(): void {\n // Transport Listeners\n this.transport.on('connected', () => this.emit('connected'));\n this.transport.on('disconnected', (reason) => this.emit('disconnected', reason));\n this.transport.on('error', (error) => this.emit('error', new SynxedConnectionError(error.message)));\n this.transport.on('message', (payload) => this.handleServerMessage(payload));\n\n // Audio Listeners\n this.audio.on('playing', () => this.updateStatus('playing'));\n this.audio.on('paused', () => this.updateStatus('paused'));\n this.audio.on('stopped', () => this.updateStatus('idle'));\n this.audio.on('loading', () => this.updateStatus('loading'));\n this.audio.on('error', (error) => this.emit('error', new SynxedPlaybackError(error)));\n this.audio.on('timeupdate', (time) => this.emit('timeUpdate', time));\n this.audio.on('ended', () => this.handleTrackEnded());\n\n // Playlist Listeners\n this.playlist.on('trackChanged', (track) => this.emit('trackChange', track));\n this.playlist.on('queueUpdated', (tracks) => this.emit('queueUpdated', tracks));\n }\n\n private connect(): void {\n this.transport.connect(this.config.apiKey, this.config.serverUrl);\n }\n\n async playSong(options: PlaySongOptions): Promise<void> {\n if (!this.transport.isConnected) this.connect();\n await this.transport.waitForConnection();\n \n this.transport.sendInit({\n contentKind: ContentKind.SONG,\n catalogTrackId: options.catalogTrackId,\n internalTrackId: options.internalTrackId,\n listenerId: options.listenerId,\n });\n }\n\n async playPlaylist(options: PlayPlaylistOptions): Promise<void> {\n if (!this.transport.isConnected) this.connect();\n await this.transport.waitForConnection();\n\n this.transport.sendInit({\n contentKind: ContentKind.PLAYLIST,\n playlistCode: options.playlistCode,\n listenerId: options.listenerId,\n });\n }\n\n pause(): void {\n this.audio.pause();\n this.transport.sendControl({\n action: SdkControlAction.CONTROL_PAUSE,\n positionMs: Math.floor(this.audio.currentTime * 1000),\n });\n }\n\n resume(): void {\n this.audio.play();\n this.transport.sendControl({\n action: SdkControlAction.CONTROL_PLAY,\n positionMs: Math.floor(this.audio.currentTime * 1000),\n });\n }\n\n stop(): void {\n this.audio.stop();\n this.transport.sendControl({\n action: SdkControlAction.CONTROL_STOP,\n positionMs: Math.floor(this.audio.currentTime * 1000),\n });\n }\n\n skip(): void {\n // For multi-track playlists, delegate to server-side skip\n if (this.playlist.queueSnapshot.length > 1) {\n this.transport.sendControl({\n action: SdkControlAction.CONTROL_SKIP_NEXT,\n positionMs: Math.floor(this.audio.currentTime * 1000),\n });\n this.updateStatus('loading');\n return;\n }\n\n // Fallback: client-side skip for single-track sessions\n const nextTrack = this.playlist.next();\n if (nextTrack) {\n this.playSong(\n nextTrack.kind === 'internal'\n ? { internalTrackId: nextTrack.id }\n : { catalogTrackId: nextTrack.id },\n );\n }\n }\n\n previous(): void {\n if (this.playlist.queueSnapshot.length > 1) {\n this.transport.sendControl({\n action: SdkControlAction.CONTROL_SKIP_PREVIOUS,\n positionMs: Math.floor(this.audio.currentTime * 1000),\n });\n this.updateStatus('loading');\n return;\n }\n\n const prevTrack = this.playlist.previous();\n if (prevTrack) {\n this.playSong(\n prevTrack.kind === 'internal'\n ? { internalTrackId: prevTrack.id }\n : { catalogTrackId: prevTrack.id },\n );\n }\n }\n\n skipTo(index: number): void {\n const track = this.playlist.skipTo(index);\n if (track) {\n this.playSong(\n track.kind === 'internal'\n ? { internalTrackId: track.id }\n : { catalogTrackId: track.id },\n );\n }\n }\n\n seek(ms: number): void {\n this.audio.seek(ms);\n this.transport.sendControl({\n action: SdkControlAction.CONTROL_SEEK,\n positionMs: ms,\n });\n }\n\n setVolume(volume: number): void {\n this.volume = Math.max(0, Math.min(1, volume));\n this.audio.setVolume(this.volume);\n }\n\n private emitAnalytics(eventType: string, positionMs?: number, extra?: any): void {\n this.transport.sendAnalytics({\n eventType,\n positionMs: positionMs ?? Math.floor(this.audio.currentTime * 1000),\n extraJson: extra ? JSON.stringify(extra) : undefined,\n });\n }\n\n private async handleServerMessage(payload: any): Promise<void> {\n console.log('[Synxed] Received server message:', payload);\n if (payload.initAck) {\n const ack = payload.initAck;\n console.log('[Synxed] Loading playback URL:', ack.playbackUrl, 'isHls:', ack.isHls);\n await this.audio.load(ack.playbackUrl, ack.isHls);\n this.audio.play();\n this.emitAnalytics('stream_start');\n \n // Parse contentSummary to populate playlist queue\n if (ack.contentSummary) {\n try {\n const summary = JSON.parse(ack.contentSummary);\n if (summary.tracks && Array.isArray(summary.tracks)) {\n const currentIndex = typeof summary.currentIndex === 'number' ? summary.currentIndex : 0;\n this.playlist.setQueue(summary.tracks as TrackInfo[], currentIndex);\n }\n } catch (e) {\n // Content summary parse failure is non-fatal\n }\n }\n } else if (payload.error) {\n this.emit('error', new SynxedError(payload.error.message, payload.error.code));\n }\n }\n\n private handleTrackEnded(): void {\n if (this.playlist.hasNext) {\n this.skip();\n } else {\n this.updateStatus('idle');\n }\n }\n\n private updateStatus(status: PlayerState['status']): void {\n this.status = status;\n this.emit('stateChange', {\n status,\n currentTrack: this.playlist.getCurrentTrack(),\n currentTime: this.audio.currentTime,\n duration: this.audio.duration,\n volume: this.volume,\n });\n }\n\n destroy(): void {\n this.audio.stop();\n this.transport.disconnect();\n this.playlist.reset();\n this.removeAllListeners();\n }\n}\n"]}
|
|
1
|
+
{"version":3,"sources":["../src/proto/sdk-streaming.ts","../src/transport/ProtocolCodec.ts","../src/core/EventEmitter.ts","../src/core/debug.ts","../src/transport/TransportManager.ts","../src/audio/AudioEngine.ts","../src/playlist/PlaylistManager.ts","../src/core/errors.ts","../src/core/contentSummary.ts","../src/core/SynxedPlayer.ts","../src/radio/nowPlaying.ts","../src/web/dom.ts","../src/web/SynxedWebPlayer.ts"],"names":["schema","root","SdkClientEnvelope","SdkServerEnvelope","ContentKind","ErrorCode","ProtocolCodec","payload","message","data","decoded","EventEmitter","event","handler","set","onceHandler","args","debugEnabled","setSynxedDebug","enabled","synxedDebug","synxedWarn","buildSdkWebSocketUrl","serverUrl","apiKey","wsBase","url","TransportManager","reason","err","resolve","reject","ws","onOpen","cleanup","onError","onClose","params","bytes","AudioEngine","isHls","isSafari","Hls","searchParams","xhr","e","Howl","id","ms","volume","update","PlaylistManager","tracks","currentIndex","track","index","SynxedError","code","SynxedConnectionError","SynxedPlaybackError","error","SynxedProtocolError","readString","obj","keys","key","v","readNumber","normalizeInitAck","raw","ack","playbackUrl","normalizeTrack","item","t","kind","albumArt","parseContentSummary","contentSummary","contentKind","summary","s","tracksRaw","rawIndex","SynxedPlayer","config","time","options","nextTrack","prevTrack","override","eventType","positionMs","extra","intervalMs","interval","pos","parsed","status","fetchRadioNowPlaying","init","res","d","el","tag","props","node","k","c","svgIcon","path","size","svg","p","ICON_PLAY","ICON_PAUSE","ICON_SKIP","ICON_SKIP_BACK","ensureGlobalStyles","style","parseDisplayTitle","seps","sep","placementStyle","placement","offsetX","offsetY","base","resolveTheme","input","applyCssVars","theme","SynxedWebPlayer","_SynxedWebPlayer","position","className","liveTop","wrap","inner","btn","shellShadow","row","pillInset","avatar","info","titleRow","controls","card","head","meta","transport","box","i","source","onState","state","tick","np","line","show","station","playing"],"mappings":"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","file":"index.mjs","sourcesContent":["import * as protobuf from 'protobufjs';\n\n/**\n * Protobuf schema definition for the SDK.\n * Matches the backend contract in src/modules/streaming/sdk-streaming.proto\n */\nconst schema = `\nsyntax = \"proto3\";\n\nenum ContentKind {\n CONTENT_KIND_UNSPECIFIED = 0;\n CONTENT_KIND_SONG = 1;\n CONTENT_KIND_PLAYLIST = 2;\n CONTENT_KIND_CATEGORY = 3;\n CONTENT_KIND_RADIO = 4;\n}\n\nenum ErrorCode {\n ERROR_CODE_UNSPECIFIED = 0;\n UNAUTHORIZED = 1;\n VALIDATION_ERROR = 2;\n NOT_FOUND = 3;\n PROCESSING = 4;\n SERVICE_UNAVAILABLE = 5;\n BAD_PROTOBUF = 6;\n}\n\nenum SdkControlAction {\n CONTROL_UNSPECIFIED = 0;\n CONTROL_PLAY = 1;\n CONTROL_PAUSE = 2;\n CONTROL_STOP = 3;\n CONTROL_SEEK = 4;\n CONTROL_SKIP_NEXT = 5;\n CONTROL_SKIP_PREVIOUS = 6;\n}\n\nmessage SdkClientInit {\n ContentKind contentKind = 1;\n string catalogTrackId = 2;\n string internalTrackId = 3;\n string playlistCode = 4;\n string categoryQuery = 5;\n string listenerId = 6;\n string deviceType = 7;\n string countryCode = 8;\n string region = 9;\n string protocolVersion = 10;\n}\n\nmessage SdkClientControl {\n SdkControlAction action = 1;\n uint32 positionMs = 2;\n}\n\nmessage SdkClientAnalytics {\n string eventType = 1;\n uint32 positionMs = 2;\n string extraJson = 3;\n}\n\nmessage SdkClientEnvelope {\n oneof payload {\n SdkClientInit init = 1;\n SdkClientControl control = 2;\n SdkClientAnalytics analytics = 3;\n }\n}\n\nmessage SdkServerInitAck {\n string sessionId = 1;\n string playbackUrl = 2;\n bool isHls = 3;\n uint32 heartbeatIntervalMs = 4;\n string contentSummary = 5;\n}\n\nmessage SdkServerError {\n ErrorCode code = 1;\n string message = 2;\n}\n\nmessage SdkServerEnvelope {\n oneof payload {\n SdkServerInitAck initAck = 1;\n SdkServerError error = 2;\n }\n}\n`;\n\nconst root = protobuf.parse(schema).root;\n\nexport const SdkClientEnvelope = root.lookupType('SdkClientEnvelope');\nexport const SdkServerEnvelope = root.lookupType('SdkServerEnvelope');\n\nexport enum ContentKind {\n UNSPECIFIED = 0,\n SONG = 1,\n PLAYLIST = 2,\n CATEGORY = 3,\n RADIO = 4,\n}\n\nexport enum ErrorCode {\n UNSPECIFIED = 0,\n UNAUTHORIZED = 1,\n VALIDATION_ERROR = 2,\n NOT_FOUND = 3,\n PROCESSING = 4,\n SERVICE_UNAVAILABLE = 5,\n BAD_PROTOBUF = 6,\n}\n\nexport enum SdkControlAction {\n CONTROL_UNSPECIFIED = 0,\n CONTROL_PLAY = 1,\n CONTROL_PAUSE = 2,\n CONTROL_STOP = 3,\n CONTROL_SEEK = 4,\n CONTROL_SKIP_NEXT = 5,\n CONTROL_SKIP_PREVIOUS = 6,\n}\n","import { SdkClientEnvelope, SdkServerEnvelope } from '../proto/sdk-streaming';\n\nexport class ProtocolCodec {\n /**\n * Encodes a client envelope into a binary payload.\n */\n static encodeClientEnvelope(payload: any): Uint8Array {\n const message = SdkClientEnvelope.create(payload);\n return SdkClientEnvelope.encode(message).finish();\n }\n\n /**\n * Decodes a binary payload into a server envelope.\n */\n static decodeServerEnvelope(data: Uint8Array): any {\n const decoded = SdkServerEnvelope.decode(data);\n return SdkServerEnvelope.toObject(decoded, {\n enums: String,\n longs: String,\n bytes: String,\n defaults: true,\n oneofs: true,\n });\n }\n}\n","type Handler = (...args: any[]) => void;\n\nexport class EventEmitter<Events extends Record<string, any>> {\n private listeners: Map<keyof Events, Set<Handler>> = new Map();\n\n on<K extends keyof Events>(event: K, handler: Events[K]): void {\n if (!this.listeners.has(event)) {\n this.listeners.set(event, new Set());\n }\n this.listeners.get(event)!.add(handler);\n }\n\n off<K extends keyof Events>(event: K, handler: Events[K]): void {\n const set = this.listeners.get(event);\n if (set) {\n set.delete(handler);\n }\n }\n\n once<K extends keyof Events>(event: K, handler: Events[K]): void {\n const onceHandler = ((...args: any[]) => {\n handler(...args);\n this.off(event, onceHandler as Events[K]);\n }) as Events[K];\n this.on(event, onceHandler);\n }\n\n protected emit<K extends keyof Events>(event: K, ...args: Parameters<Events[K]>): void {\n const set = this.listeners.get(event);\n if (set) {\n set.forEach((handler) => handler(...args));\n }\n }\n\n removeAllListeners(): void {\n this.listeners.clear();\n }\n}\n","let debugEnabled = false;\n\nexport function setSynxedDebug(enabled: boolean): void {\n debugEnabled = enabled;\n}\n\nexport function synxedDebug(...args: unknown[]): void {\n if (debugEnabled) {\n // eslint-disable-next-line no-console\n console.debug('[Synxed]', ...args);\n }\n}\n\nexport function synxedWarn(...args: unknown[]): void {\n // eslint-disable-next-line no-console\n console.warn('[Synxed]', ...args);\n}\n","import { ProtocolCodec } from './ProtocolCodec';\nimport { EventEmitter } from '../core/EventEmitter';\nimport { synxedDebug, synxedWarn } from '../core/debug';\n\nexport interface TransportEvents {\n connected: () => void;\n disconnected: (reason: string) => void;\n error: (error: any) => void;\n message: (payload: any) => void;\n}\n\nexport function buildSdkWebSocketUrl(serverUrl: string, apiKey: string): string {\n const normalized = serverUrl.endsWith('/') ? serverUrl.slice(0, -1) : serverUrl;\n const wsBase = normalized.replace(/^http:/, 'ws:').replace(/^https:/, 'wss:');\n const url = new URL(`${wsBase}/sdk`);\n url.searchParams.set('apiKey', apiKey);\n return url.toString();\n}\n\nexport class TransportManager extends EventEmitter<TransportEvents> {\n private ws: WebSocket | null = null;\n private connectPromise: Promise<void> | null = null;\n\n connect(apiKey: string, serverUrl: string): void {\n if (\n this.ws &&\n (this.ws.readyState === WebSocket.OPEN ||\n this.ws.readyState === WebSocket.CONNECTING)\n ) {\n return;\n }\n\n const url = buildSdkWebSocketUrl(serverUrl, apiKey);\n synxedDebug('Opening native WebSocket', url.replace(/apiKey=[^&]+/, 'apiKey=***'));\n\n this.ws = new WebSocket(url);\n this.ws.binaryType = 'arraybuffer';\n\n this.ws.onopen = () => {\n this.connectPromise = null;\n this.emit('connected');\n };\n\n this.ws.onclose = (event) => {\n const reason = event.reason || `code ${event.code}`;\n this.ws = null;\n this.connectPromise = null;\n this.emit('disconnected', reason);\n };\n\n this.ws.onerror = () => {\n this.emit('error', new Error('WebSocket connection error'));\n };\n\n this.ws.onmessage = (event) => {\n try {\n const { data } = event;\n if (!(data instanceof ArrayBuffer)) return;\n\n const payload = ProtocolCodec.decodeServerEnvelope(new Uint8Array(data));\n this.emit('message', payload);\n } catch (err) {\n this.emit('error', new Error(`Failed to decode message: ${err}`));\n }\n };\n }\n\n async waitForConnection(): Promise<void> {\n synxedDebug('Waiting for connection…');\n if (this.ws?.readyState === WebSocket.OPEN) {\n synxedDebug('Already connected');\n return;\n }\n\n if (!this.ws) {\n throw new Error('WebSocket not initialized. Call connect() first.');\n }\n\n if (this.connectPromise) return this.connectPromise;\n\n this.connectPromise = new Promise((resolve, reject) => {\n const ws = this.ws!;\n\n if (ws.readyState === WebSocket.OPEN) {\n resolve();\n return;\n }\n\n const onOpen = () => {\n synxedDebug('WebSocket connected');\n cleanup();\n resolve();\n };\n\n const onError = () => {\n synxedWarn('WebSocket connection error');\n cleanup();\n reject(new Error('WebSocket connection failed'));\n };\n\n const onClose = () => {\n cleanup();\n reject(new Error('WebSocket closed before connection was established'));\n };\n\n const cleanup = () => {\n ws.removeEventListener('open', onOpen);\n ws.removeEventListener('error', onError);\n ws.removeEventListener('close', onClose);\n };\n\n ws.addEventListener('open', onOpen);\n ws.addEventListener('error', onError);\n ws.addEventListener('close', onClose);\n });\n\n return this.connectPromise;\n }\n\n disconnect(): void {\n if (!this.ws) return;\n\n this.ws.onclose = null;\n this.ws.close();\n this.ws = null;\n this.connectPromise = null;\n }\n\n sendInit(params: any): void {\n if (!this.isConnected) {\n throw new Error('WebSocket not connected');\n }\n\n synxedDebug('Sending init packet', params);\n this.sendBytes(ProtocolCodec.encodeClientEnvelope({ init: params }));\n }\n\n sendControl(params: any): void {\n if (!this.isConnected) return;\n this.sendBytes(ProtocolCodec.encodeClientEnvelope({ control: params }));\n }\n\n sendAnalytics(params: any): void {\n if (!this.isConnected) return;\n this.sendBytes(ProtocolCodec.encodeClientEnvelope({ analytics: params }));\n }\n\n get isConnected(): boolean {\n return this.ws?.readyState === WebSocket.OPEN;\n }\n\n private sendBytes(bytes: Uint8Array): void {\n if (!this.ws || this.ws.readyState !== WebSocket.OPEN) {\n throw new Error('WebSocket not connected');\n }\n this.ws.send(bytes);\n }\n}\n","import { Howl } from 'howler';\nimport { EventEmitter } from '../core/EventEmitter';\nimport { synxedDebug, synxedWarn } from '../core/debug';\n\nexport interface AudioEvents {\n playing: () => void;\n paused: () => void;\n stopped: () => void;\n ended: () => void;\n loading: () => void;\n loaded: () => void;\n error: (error: any) => void;\n timeupdate: (data: { currentTime: number; duration: number }) => void;\n}\n\nexport class AudioEngine extends EventEmitter<AudioEvents> {\n private howl: Howl | null = null;\n private hls: any = null;\n private audioEl: HTMLAudioElement | null = null;\n private updateTimer: number | null = null;\n\n constructor() {\n super();\n }\n\n\n /**\n * Loads a track URL.\n */\n async load(url: string, isHls: boolean): Promise<void> {\n this.stop();\n this.emit('loading');\n\n synxedDebug('AudioEngine loading', { url, isHls });\n\n // Safari has native HLS support. Other browsers need hls.js.\n const isSafari = /^((?!chrome|android).)*safari/i.test(navigator.userAgent);\n\n if (isHls && !isSafari) {\n try {\n const { default: Hls } = await import('hls.js');\n if (Hls.isSupported()) {\n synxedDebug('Using Hls.js for playback');\n \n // Extract query parameters from the manifest URL to forward to segments\n const urlObj = new URL(url);\n const searchParams = urlObj.search;\n\n this.hls = new Hls({\n xhrSetup: (xhr) => {\n xhr.setRequestHeader(\"ngrok-skip-browser-warning\", \"true\");\n }\n });\n this.audioEl = new Audio();\n this.hls.loadSource(url);\n this.hls.attachMedia(this.audioEl);\n \n this.audioEl.onplay = () => {\n this.emit('playing');\n this.startTimeUpdateLoop();\n };\n this.audioEl.onpause = () => {\n this.emit('paused');\n this.stopTimeUpdateLoop();\n };\n this.audioEl.onended = () => {\n this.emit('ended');\n this.stopTimeUpdateLoop();\n };\n this.audioEl.onerror = (e) => {\n synxedWarn('HLS audio element error', e);\n this.emit('error', e);\n };\n this.audioEl.onloadedmetadata = () => {\n this.emit('loaded');\n };\n\n this.hls.on(Hls.Events.ERROR, (event: any, data: any) => {\n if (data.fatal) {\n synxedWarn('Fatal HLS error', data);\n this.emit('error', data);\n }\n });\n\n return;\n }\n } catch (e) {\n synxedWarn('hls.js not available, falling back to native playback');\n }\n }\n\n // Fallback to Howler\n this.howl = new Howl({\n src: [url],\n html5: true, \n format: isHls ? ['m3u8'] : undefined,\n onload: () => {\n synxedDebug('AudioEngine loaded');\n this.emit('loaded');\n },\n onloaderror: (id, err) => {\n synxedWarn('AudioEngine load error', err, id);\n this.emit('error', err);\n },\n onplay: () => {\n this.emit('playing');\n this.startTimeUpdateLoop();\n },\n onpause: () => {\n this.emit('paused');\n this.stopTimeUpdateLoop();\n },\n onstop: () => {\n this.emit('stopped');\n this.stopTimeUpdateLoop();\n },\n onend: () => {\n this.emit('ended');\n this.stopTimeUpdateLoop();\n },\n });\n }\n\n play(): void {\n if (this.audioEl) {\n this.audioEl.play().catch((e) => synxedWarn('Play failed', e));\n } else {\n this.howl?.play();\n }\n }\n\n pause(): void {\n if (this.audioEl) {\n this.audioEl.pause();\n } else {\n this.howl?.pause();\n }\n }\n\n stop(): void {\n if (this.hls) {\n this.hls.destroy();\n this.hls = null;\n }\n if (this.audioEl) {\n this.audioEl.pause();\n this.audioEl.src = '';\n this.audioEl = null;\n }\n if (this.howl) {\n this.howl.stop();\n this.howl.unload();\n this.howl = null;\n }\n this.stopTimeUpdateLoop();\n }\n\n seek(ms: number): void {\n if (this.audioEl) {\n this.audioEl.currentTime = ms / 1000;\n } else {\n this.howl?.seek(ms / 1000);\n }\n }\n\n setVolume(volume: number): void {\n if (this.audioEl) {\n this.audioEl.volume = volume;\n }\n if (this.howl) {\n this.howl.volume(volume);\n }\n }\n\n get duration(): number {\n if (this.audioEl) return this.audioEl.duration * 1000;\n return (this.howl?.duration() || 0) * 1000;\n }\n\n get currentTime(): number {\n if (this.audioEl) return this.audioEl.currentTime * 1000;\n return (this.howl?.seek() as number || 0) * 1000;\n }\n\n private startTimeUpdateLoop(): void {\n this.stopTimeUpdateLoop();\n const update = () => {\n const isPlaying = this.audioEl ? !this.audioEl.paused : this.howl?.playing();\n if (isPlaying) {\n this.emit('timeupdate', {\n currentTime: this.currentTime,\n duration: this.duration,\n });\n this.updateTimer = requestAnimationFrame(update);\n }\n };\n this.updateTimer = requestAnimationFrame(update);\n }\n\n private stopTimeUpdateLoop(): void {\n if (this.updateTimer !== null) {\n cancelAnimationFrame(this.updateTimer);\n this.updateTimer = null;\n }\n }\n}\n","import { EventEmitter } from '../core/EventEmitter';\nimport { TrackInfo } from '../types';\n\nexport interface PlaylistEvents {\n trackChanged: (track: TrackInfo) => void;\n queueUpdated: (tracks: TrackInfo[]) => void;\n}\n\nexport class PlaylistManager extends EventEmitter<PlaylistEvents> {\n private queue: TrackInfo[] = [];\n private currentIndex: number = -1;\n\n constructor() {\n super();\n }\n\n setQueue(tracks: TrackInfo[], currentIndex?: number): void {\n this.queue = tracks;\n this.currentIndex =\n typeof currentIndex === 'number' &&\n currentIndex >= 0 &&\n currentIndex < tracks.length\n ? currentIndex\n : tracks.length > 0\n ? 0\n : -1;\n this.emit('queueUpdated', this.queue);\n if (this.getCurrentTrack()) {\n this.emit('trackChanged', this.getCurrentTrack()!);\n }\n }\n\n getCurrentTrack(): TrackInfo | null {\n if (this.currentIndex >= 0 && this.currentIndex < this.queue.length) {\n return this.queue[this.currentIndex];\n }\n return null;\n }\n\n next(): TrackInfo | null {\n if (this.currentIndex < this.queue.length - 1) {\n this.currentIndex++;\n const track = this.getCurrentTrack();\n this.emit('trackChanged', track!);\n return track;\n }\n return null;\n }\n\n previous(): TrackInfo | null {\n if (this.currentIndex > 0) {\n this.currentIndex--;\n const track = this.getCurrentTrack();\n this.emit('trackChanged', track!);\n return track;\n }\n return null;\n }\n\n skipTo(index: number): TrackInfo | null {\n if (index >= 0 && index < this.queue.length) {\n this.currentIndex = index;\n const track = this.getCurrentTrack();\n this.emit('trackChanged', track!);\n return track;\n }\n return null;\n }\n\n reset(): void {\n this.queue = [];\n this.currentIndex = -1;\n this.emit('queueUpdated', []);\n }\n\n get hasNext(): boolean {\n return this.currentIndex < this.queue.length - 1;\n }\n\n get hasPrevious(): boolean {\n return this.currentIndex > 0;\n }\n\n get queueSnapshot(): TrackInfo[] {\n return [...this.queue];\n }\n\n get currentIndexSnapshot(): number {\n return this.currentIndex;\n }\n}\n","export class SynxedError extends Error {\n constructor(message: string, public code?: string) {\n super(message);\n this.name = 'SynxedError';\n }\n}\n\nexport class SynxedConnectionError extends SynxedError {\n constructor(message: string) {\n super(message);\n this.name = 'SynxedConnectionError';\n }\n}\n\nexport class SynxedPlaybackError extends SynxedError {\n constructor(error: any) {\n super(typeof error === 'string' ? error : 'Playback failed');\n this.name = 'SynxedPlaybackError';\n }\n}\n\nexport class SynxedProtocolError extends SynxedError {\n constructor(message: string) {\n super(message);\n this.name = 'SynxedProtocolError';\n }\n}\n","import { ContentKind } from '../proto/sdk-streaming';\nimport type { TrackInfo } from '../types';\n\nexport interface NormalizedInitAck {\n sessionId: string;\n playbackUrl: string;\n isHls: boolean;\n heartbeatIntervalMs: number;\n contentSummary?: string;\n}\n\nexport interface ParsedContentSummary {\n tracks: TrackInfo[];\n currentIndex: number;\n}\n\nfunction readString(obj: Record<string, unknown>, ...keys: string[]): string | undefined {\n for (const key of keys) {\n const v = obj[key];\n if (typeof v === 'string' && v.length > 0) return v;\n }\n return undefined;\n}\n\nfunction readNumber(obj: Record<string, unknown>, ...keys: string[]): number | undefined {\n for (const key of keys) {\n const v = obj[key];\n if (typeof v === 'number' && Number.isFinite(v)) return v;\n if (typeof v === 'string' && v !== '' && !Number.isNaN(Number(v))) return Number(v);\n }\n return undefined;\n}\n\n/** Normalizes initAck from protobuf `toObject` (camelCase or snake_case). */\nexport function normalizeInitAck(payload: Record<string, unknown>): NormalizedInitAck | null {\n const raw = payload.initAck ?? payload.init_ack;\n if (!raw || typeof raw !== 'object') return null;\n\n const ack = raw as Record<string, unknown>;\n const playbackUrl = readString(ack, 'playbackUrl', 'playback_url');\n if (!playbackUrl) return null;\n\n return {\n sessionId: readString(ack, 'sessionId', 'session_id') ?? '',\n playbackUrl,\n isHls: Boolean(ack.isHls ?? ack.is_hls),\n heartbeatIntervalMs: readNumber(ack, 'heartbeatIntervalMs', 'heartbeat_interval_ms') ?? 0,\n contentSummary: readString(ack, 'contentSummary', 'content_summary'),\n };\n}\n\nfunction normalizeTrack(item: unknown): TrackInfo | null {\n if (!item || typeof item !== 'object') return null;\n const t = item as Record<string, unknown>;\n const id = t.id;\n if (typeof id !== 'string' || !id.trim()) return null;\n\n const kind: TrackInfo['kind'] = t.kind === 'internal' ? 'internal' : 'catalog';\n const albumArt = t.albumArt ?? t.album_art;\n\n return {\n id: id.trim(),\n kind,\n title: typeof t.title === 'string' ? t.title : undefined,\n artist: typeof t.artist === 'string' ? t.artist : undefined,\n duration: typeof t.duration === 'number' ? t.duration : undefined,\n albumArt: typeof albumArt === 'string' ? albumArt : undefined,\n };\n}\n\n/**\n * Parses `contentSummary` JSON from initAck into a playlist queue.\n * Returns `null` when summary is absent or has no usable queue (playback may still continue).\n */\nexport function parseContentSummary(\n contentSummary: string | undefined,\n contentKind: ContentKind,\n): ParsedContentSummary | null {\n if (!contentSummary?.trim()) return null;\n\n let summary: unknown;\n try {\n summary = JSON.parse(contentSummary);\n } catch {\n return null;\n }\n\n if (!summary || typeof summary !== 'object') return null;\n const s = summary as Record<string, unknown>;\n\n const type = s.type;\n if (type === 'live_radio' || contentKind === ContentKind.RADIO) {\n const station =\n typeof s.station === 'string' && s.station.trim() ? s.station.trim() : 'Synxed Radio';\n return {\n tracks: [{ id: 'synxed-radio', kind: 'catalog', title: station, artist: '' }],\n currentIndex: 0,\n };\n }\n\n const tracksRaw = s.tracks;\n if (!Array.isArray(tracksRaw) || tracksRaw.length === 0) return null;\n\n const tracks: TrackInfo[] = [];\n for (const item of tracksRaw) {\n const track = normalizeTrack(item);\n if (track) tracks.push(track);\n }\n if (tracks.length === 0) return null;\n\n const rawIndex =\n typeof s.currentIndex === 'number'\n ? s.currentIndex\n : typeof s.current_index === 'number'\n ? s.current_index\n : 0;\n\n const currentIndex = Math.max(0, Math.min(Math.floor(rawIndex), tracks.length - 1));\n return { tracks, currentIndex };\n}\n","import { TransportManager } from '../transport/TransportManager';\nimport { AudioEngine } from '../audio/AudioEngine';\nimport { PlaylistManager } from '../playlist/PlaylistManager';\nimport { EventEmitter } from './EventEmitter';\nimport { ContentKind, SdkControlAction } from '../proto/sdk-streaming';\nimport {\n SynxedConfig,\n PlayerState,\n PlaySongOptions,\n PlayPlaylistOptions,\n PlayRadioOptions,\n SynxedEvents,\n TrackInfo,\n} from '../types';\nimport { SynxedError, SynxedPlaybackError, SynxedConnectionError } from './errors';\nimport { normalizeInitAck, parseContentSummary } from './contentSummary';\nimport { setSynxedDebug, synxedDebug, synxedWarn } from './debug';\n\nexport class SynxedPlayer extends EventEmitter<SynxedEvents> {\n private transport: TransportManager;\n private audio: AudioEngine;\n private playlist: PlaylistManager;\n private config: SynxedConfig;\n private status: PlayerState['status'] = 'idle';\n private volume = 0.8;\n private activeContentKind: ContentKind = ContentKind.UNSPECIFIED;\n private heartbeatTimer: ReturnType<typeof setInterval> | null = null;\n\n constructor(config: SynxedConfig) {\n super();\n this.config = config;\n setSynxedDebug(Boolean(config.debug));\n this.transport = new TransportManager();\n this.audio = new AudioEngine();\n this.playlist = new PlaylistManager();\n\n this.setupListeners();\n\n if (config.autoConnect) {\n this.connect();\n }\n }\n\n get currentTrack(): TrackInfo | null {\n return this.playlist.getCurrentTrack();\n }\n\n /** Active playback init kind from the last `play*` call (RADIO, PLAYLIST, SONG, …). */\n get contentKind(): ContentKind {\n return this.activeContentKind;\n }\n\n private controlPositionMs(): number {\n if (this.activeContentKind === ContentKind.RADIO) return 0;\n return this.positionMsForAnalytics();\n }\n\n private setupListeners(): void {\n // Transport Listeners\n this.transport.on('connected', () => this.emit('connected'));\n this.transport.on('disconnected', (reason) => this.emit('disconnected', reason));\n this.transport.on('error', (error) => this.emit('error', new SynxedConnectionError(error.message)));\n this.transport.on('message', (payload) => this.handleServerMessage(payload));\n\n // Audio Listeners\n this.audio.on('playing', () => this.updateStatus('playing'));\n this.audio.on('paused', () => this.updateStatus('paused'));\n this.audio.on('stopped', () => this.updateStatus('idle'));\n this.audio.on('loading', () => this.updateStatus('loading'));\n this.audio.on('error', (error) => this.emit('error', new SynxedPlaybackError(error)));\n this.audio.on('timeupdate', (time) => this.emit('timeUpdate', time));\n this.audio.on('ended', () => this.handleTrackEnded());\n\n // Playlist Listeners\n this.playlist.on('trackChanged', (track) => this.emit('trackChange', track));\n this.playlist.on('queueUpdated', (tracks) => this.emit('queueUpdated', tracks));\n }\n\n private connect(): void {\n this.transport.connect(this.config.apiKey, this.config.serverUrl);\n }\n\n async playSong(options: PlaySongOptions): Promise<void> {\n if (!this.transport.isConnected) this.connect();\n await this.transport.waitForConnection();\n\n this.activeContentKind = ContentKind.SONG;\n this.transport.sendInit({\n contentKind: ContentKind.SONG,\n catalogTrackId: options.catalogTrackId,\n internalTrackId: options.internalTrackId,\n listenerId: options.listenerId,\n });\n }\n\n async playPlaylist(options: PlayPlaylistOptions): Promise<void> {\n if (!this.transport.isConnected) this.connect();\n await this.transport.waitForConnection();\n\n this.activeContentKind = ContentKind.PLAYLIST;\n this.transport.sendInit({\n contentKind: ContentKind.PLAYLIST,\n playlistCode: options.playlistCode,\n listenerId: options.listenerId,\n });\n }\n\n async playRadio(options: PlayRadioOptions = {}): Promise<void> {\n if (!this.transport.isConnected) this.connect();\n await this.transport.waitForConnection();\n\n this.activeContentKind = ContentKind.RADIO;\n this.transport.sendInit({\n contentKind: ContentKind.RADIO,\n listenerId: options.listenerId,\n });\n }\n\n pause(): void {\n this.audio.pause();\n this.transport.sendControl({\n action: SdkControlAction.CONTROL_PAUSE,\n positionMs: this.controlPositionMs(),\n });\n }\n\n resume(): void {\n this.audio.play();\n this.transport.sendControl({\n action: SdkControlAction.CONTROL_PLAY,\n positionMs: this.controlPositionMs(),\n });\n }\n\n stop(): void {\n this.audio.stop();\n this.transport.sendControl({\n action: SdkControlAction.CONTROL_STOP,\n positionMs: this.controlPositionMs(),\n });\n }\n\n skip(): void {\n if (this.activeContentKind === ContentKind.RADIO) return;\n // For multi-track playlists, delegate to server-side skip\n if (this.playlist.queueSnapshot.length > 1) {\n this.transport.sendControl({\n action: SdkControlAction.CONTROL_SKIP_NEXT,\n positionMs: this.controlPositionMs(),\n });\n this.updateStatus('loading');\n return;\n }\n\n // Fallback: client-side skip for single-track sessions\n const nextTrack = this.playlist.next();\n if (nextTrack) {\n this.playSong(\n nextTrack.kind === 'internal'\n ? { internalTrackId: nextTrack.id }\n : { catalogTrackId: nextTrack.id },\n );\n }\n }\n\n previous(): void {\n if (this.activeContentKind === ContentKind.RADIO) return;\n if (this.playlist.queueSnapshot.length > 1) {\n this.transport.sendControl({\n action: SdkControlAction.CONTROL_SKIP_PREVIOUS,\n positionMs: this.controlPositionMs(),\n });\n this.updateStatus('loading');\n return;\n }\n\n const prevTrack = this.playlist.previous();\n if (prevTrack) {\n this.playSong(\n prevTrack.kind === 'internal'\n ? { internalTrackId: prevTrack.id }\n : { catalogTrackId: prevTrack.id },\n );\n }\n }\n\n skipTo(index: number): void {\n if (this.activeContentKind === ContentKind.RADIO) return;\n const track = this.playlist.skipTo(index);\n if (track) {\n this.playSong(\n track.kind === 'internal'\n ? { internalTrackId: track.id }\n : { catalogTrackId: track.id },\n );\n }\n }\n\n seek(ms: number): void {\n if (this.activeContentKind === ContentKind.RADIO) return;\n this.audio.seek(ms);\n this.transport.sendControl({\n action: SdkControlAction.CONTROL_SEEK,\n positionMs: ms,\n });\n }\n\n setVolume(volume: number): void {\n this.volume = Math.max(0, Math.min(1, volume));\n this.audio.setVolume(this.volume);\n }\n\n private positionMsForAnalytics(override?: number): number {\n if (typeof override === 'number') {\n return Math.max(0, Math.min(0xffffffff, Math.floor(override)));\n }\n return Math.max(0, Math.min(0xffffffff, Math.floor(this.audio.currentTime)));\n }\n\n private emitAnalytics(eventType: string, positionMs?: number, extra?: any): void {\n this.transport.sendAnalytics({\n eventType,\n positionMs: this.positionMsForAnalytics(positionMs),\n extraJson: extra ? JSON.stringify(extra) : undefined,\n });\n }\n\n private clearHeartbeat(): void {\n if (this.heartbeatTimer !== null) {\n clearInterval(this.heartbeatTimer);\n this.heartbeatTimer = null;\n }\n }\n\n private startHeartbeat(intervalMs: number): void {\n this.clearHeartbeat();\n const interval = Number(intervalMs);\n if (!interval || interval < 1000) return;\n\n this.heartbeatTimer = setInterval(() => {\n const pos =\n this.activeContentKind === ContentKind.RADIO\n ? 0\n : this.positionMsForAnalytics();\n this.emitAnalytics('heartbeat', pos);\n }, interval);\n }\n\n private applyContentSummary(contentSummary: string | undefined): void {\n const parsed = parseContentSummary(contentSummary, this.activeContentKind);\n if (parsed) {\n this.playlist.setQueue(parsed.tracks, parsed.currentIndex);\n return;\n }\n\n if (!contentSummary?.trim()) return;\n\n synxedWarn('Could not parse contentSummary; queue metadata unavailable.');\n }\n\n private async handleServerMessage(payload: Record<string, unknown>): Promise<void> {\n synxedDebug('Received server message', payload);\n\n const ack = normalizeInitAck(payload);\n if (ack) {\n synxedDebug('Loading playback', { url: ack.playbackUrl, isHls: ack.isHls });\n await this.audio.load(ack.playbackUrl, ack.isHls);\n this.audio.play();\n this.emitAnalytics('stream_start', this.activeContentKind === ContentKind.RADIO ? 0 : undefined);\n\n this.startHeartbeat(ack.heartbeatIntervalMs);\n this.applyContentSummary(ack.contentSummary);\n return;\n }\n\n const err = payload.error;\n if (err && typeof err === 'object') {\n const e = err as Record<string, unknown>;\n const message = typeof e.message === 'string' ? e.message : 'Unknown server error';\n const code = typeof e.code === 'string' ? e.code : undefined;\n this.emit('error', new SynxedError(message, code));\n }\n }\n\n private handleTrackEnded(): void {\n if (this.activeContentKind === ContentKind.RADIO) {\n return;\n }\n if (this.playlist.hasNext) {\n this.skip();\n } else {\n this.updateStatus('idle');\n }\n }\n\n private updateStatus(status: PlayerState['status']): void {\n this.status = status;\n this.emit('stateChange', {\n status,\n currentTrack: this.playlist.getCurrentTrack(),\n currentTime: this.audio.currentTime,\n duration: this.audio.duration,\n volume: this.volume,\n });\n }\n\n destroy(): void {\n this.clearHeartbeat();\n this.audio.stop();\n this.transport.disconnect();\n this.playlist.reset();\n this.removeAllListeners();\n }\n}\n","import type { RadioNowPlaying } from '../types';\n\n/**\n * Fetches live radio \"now playing\" metadata from the Synxed backend.\n * Poll every 10–15 seconds to keep UI in sync (see integration docs).\n */\nexport async function fetchRadioNowPlaying(\n serverUrl: string,\n init?: RequestInit,\n): Promise<RadioNowPlaying | null> {\n const base = serverUrl.replace(/\\/$/, '');\n const url = `${base}/radio/now-playing`;\n try {\n const res = await fetch(url, {\n ...init,\n headers: {\n Accept: 'application/json',\n ...(init?.headers as Record<string, string>),\n },\n });\n if (!res.ok) return null;\n const json = await res.json();\n const d = json?.data;\n if (!d?.title || typeof d.title !== 'string') return null;\n return {\n title: d.title,\n station: typeof d.station === 'string' ? d.station : undefined,\n isLive: typeof d.isLive === 'boolean' ? d.isLive : undefined,\n };\n } catch {\n return null;\n }\n}\n","import type { SynxedWebPlayerPlacement } from './types';\n\nexport function el<K extends keyof HTMLElementTagNameMap>(\n tag: K,\n props?: {\n className?: string;\n attrs?: Record<string, string>;\n style?: Partial<CSSStyleDeclaration>;\n text?: string;\n html?: string;\n onClick?: (e: Event) => void;\n children?: Node[];\n },\n): HTMLElementTagNameMap[K] {\n const node = document.createElement(tag);\n if (props?.className) node.className = props.className;\n if (props?.attrs) {\n for (const [k, v] of Object.entries(props.attrs)) node.setAttribute(k, v);\n }\n if (props?.style) Object.assign(node.style, props.style);\n if (props?.text) node.textContent = props.text;\n if (props?.html) node.innerHTML = props.html;\n if (props?.onClick) node.addEventListener('click', props.onClick);\n if (props?.children) props.children.forEach((c) => node.appendChild(c));\n return node;\n}\n\nexport function svgIcon(path: string, size = 20): SVGSVGElement {\n const svg = document.createElementNS('http://www.w3.org/2000/svg', 'svg');\n svg.setAttribute('width', String(size));\n svg.setAttribute('height', String(size));\n svg.setAttribute('viewBox', '0 0 24 24');\n svg.setAttribute('fill', 'currentColor');\n svg.setAttribute('aria-hidden', 'true');\n const p = document.createElementNS('http://www.w3.org/2000/svg', 'path');\n p.setAttribute('d', path);\n svg.appendChild(p);\n return svg;\n}\n\nexport const ICON_PLAY = 'M8 5v14l11-7L8 5z';\nexport const ICON_PAUSE = 'M6 5h4v14H6V5zm8 0h4v14h-4V5z';\nexport const ICON_SKIP = 'M6 18l8.5-6L6 6v12zM16 6v12h2V6h-2z';\nexport const ICON_SKIP_BACK = 'M6 6h2v12H6V6zm11.5 12L9 12l8.5-6v12z';\n\nexport function ensureGlobalStyles(): void {\n const id = 'synxed-web-player-styles';\n if (document.getElementById(id)) return;\n const style = document.createElement('style');\n style.id = id;\n style.textContent = `\n@keyframes synxed-wp-spin { from { transform: rotate(0); } to { transform: rotate(360deg); } }\n@keyframes synxed-wp-vbar {\n 0%, 100% { transform: scaleY(0.28); opacity: 0.45; }\n 50% { transform: scaleY(1); opacity: 1; }\n}\n`;\n document.head.appendChild(style);\n}\n\nexport function parseDisplayTitle(raw: string): { title: string; artist?: string } {\n const seps = [' — ', ' – ', ' - ', ' —', ' –', ' -'];\n for (const sep of seps) {\n const i = raw.indexOf(sep);\n if (i > 0) {\n return { title: raw.slice(0, i).trim(), artist: raw.slice(i + sep.length).trim() };\n }\n }\n return { title: raw.trim() };\n}\n\nexport function placementStyle(\n placement: SynxedWebPlayerPlacement = 'bottom-center',\n offsetX = 16,\n offsetY = 16,\n): Partial<CSSStyleDeclaration> {\n const base: Partial<CSSStyleDeclaration> = { position: 'fixed', zIndex: '50' };\n switch (placement) {\n case 'top-left':\n return { ...base, top: `${offsetY}px`, left: `${offsetX}px` };\n case 'top-right':\n return { ...base, top: `${offsetY}px`, right: `${offsetX}px` };\n case 'bottom-left':\n return { ...base, bottom: `${offsetY}px`, left: `${offsetX}px` };\n case 'bottom-right':\n return { ...base, bottom: `${offsetY}px`, right: `${offsetX}px` };\n case 'bottom-center':\n default:\n return { ...base, bottom: `${offsetY}px`, left: '50%', transform: 'translateX(-50%)' };\n }\n}\n\nexport function resolveTheme(input: import('./types').SynxedWebPlayerTheme = {}) {\n return {\n accent: input.accent ?? '#22c55e',\n accentMuted: input.accentMuted ?? input.accent ?? '#22c55e',\n background: input.background ?? '#051107',\n backgroundInner: input.backgroundInner ?? '#0a1f10',\n border: input.border ?? 'rgba(34, 197, 94, 0.35)',\n text: input.text ?? '#ffffff',\n textMuted: input.textMuted ?? 'rgba(255,255,255,0.35)',\n stationText: input.stationText ?? 'rgba(34, 197, 94, 0.85)',\n liveDot: input.liveDot ?? '#CCFF00',\n glow: input.glow ?? 'rgba(34, 197, 94, 0.35)',\n };\n}\n\nexport function applyCssVars(root: HTMLElement, theme: ReturnType<typeof resolveTheme>): void {\n root.style.setProperty('--sw-accent', theme.accent);\n root.style.setProperty('--sw-accent-muted', theme.accentMuted);\n root.style.setProperty('--sw-bg', theme.background);\n root.style.setProperty('--sw-bg-inner', theme.backgroundInner);\n root.style.setProperty('--sw-border', theme.border);\n root.style.setProperty('--sw-text', theme.text);\n root.style.setProperty('--sw-text-muted', theme.textMuted);\n root.style.setProperty('--sw-station', theme.stationText);\n root.style.setProperty('--sw-live', theme.liveDot);\n root.style.setProperty('--sw-glow', theme.glow);\n}\n","import { SynxedPlayer } from '../core/SynxedPlayer';\nimport { fetchRadioNowPlaying } from '../radio/nowPlaying';\nimport type { PlayerState, RadioNowPlaying, TrackInfo } from '../types';\nimport {\n ICON_PAUSE,\n ICON_PLAY,\n ICON_SKIP,\n ICON_SKIP_BACK,\n applyCssVars,\n el,\n ensureGlobalStyles,\n parseDisplayTitle,\n placementStyle,\n resolveTheme,\n svgIcon,\n} from './dom';\nimport type { SynxedWebPlayerMode, SynxedWebPlayerOptions } from './types';\n\nexport type {\n SynxedWebPlayerMode,\n SynxedWebPlayerOptions,\n SynxedWebPlayerPlacement,\n SynxedWebPlayerPosition,\n SynxedWebPlayerSource,\n SynxedWebPlayerTheme,\n} from './types';\n\n/**\n * Framework-agnostic web overlay player (vanilla DOM).\n * Works in React, Vue, Angular, or plain HTML — no React dependency.\n */\nexport class SynxedWebPlayer {\n private readonly options: SynxedWebPlayerOptions;\n private readonly root: HTMLElement;\n private readonly ownsRoot: boolean;\n private readonly theme: ReturnType<typeof resolveTheme>;\n private engine: SynxedPlayer | null = null;\n private pollTimer: ReturnType<typeof setInterval> | null = null;\n private destroyed = false;\n\n private isPlaying = false;\n private currentTrack: TrackInfo | null = null;\n private nowPlaying: RadioNowPlaying | null = null;\n\n private avatarRing: HTMLElement | null = null;\n private titleEl: HTMLElement | null = null;\n private artistInlineEl: HTMLElement | null = null;\n private artistBlockEl: HTMLElement | null = null;\n private footerEl: HTMLElement | null = null;\n private playBtn: HTMLButtonElement | null = null;\n\n constructor(options: SynxedWebPlayerOptions) {\n this.options = {\n mode: 'wide',\n attribution: 'Synxed',\n nowPlayingPollMs: 12_000,\n powerByLabel: 'Powered by Synxed',\n ...options,\n };\n this.theme = resolveTheme(this.options.theme);\n this.ownsRoot = !options.container;\n this.root = options.container ?? document.createElement('div');\n if (this.ownsRoot) document.body.appendChild(this.root);\n\n ensureGlobalStyles();\n this.mountShell();\n this.initEngine();\n if (this.options.source.type === 'radio') this.startNowPlayingPoll();\n }\n\n /** Convenience factory — same as `new SynxedWebPlayer(options)`. */\n static mount(options: SynxedWebPlayerOptions): SynxedWebPlayer {\n return new SynxedWebPlayer(options);\n }\n\n /** Underlying streaming engine (advanced integrations). */\n get player(): SynxedPlayer | null {\n return this.engine;\n }\n\n get element(): HTMLElement {\n return this.root;\n }\n\n destroy(): void {\n if (this.destroyed) return;\n this.destroyed = true;\n this.stopNowPlayingPoll();\n this.engine?.destroy();\n this.engine = null;\n if (this.ownsRoot) this.root.remove();\n }\n\n private get isRadio(): boolean {\n return this.options.source.type === 'radio';\n }\n\n private get mode(): SynxedWebPlayerMode {\n return this.options.mode ?? 'wide';\n }\n\n private mountShell(): void {\n const { position = {}, className, style } = this.options;\n const placement = position.placement ?? 'bottom-center';\n const offsetX = position.offsetX ?? 16;\n const offsetY = position.offsetY ?? (placement === 'bottom-center' ? 24 : 16);\n\n this.root.className = className ?? '';\n this.root.dataset.synxedWebPlayer = this.mode;\n applyCssVars(this.root, this.theme);\n Object.assign(this.root.style, placementStyle(placement, offsetX, offsetY));\n if (style) Object.assign(this.root.style, style);\n\n this.root.replaceChildren();\n if (this.mode === 'mini') this.buildMini();\n else if (this.mode === 'large') this.buildLarge();\n else this.buildWide();\n\n this.refreshLabels();\n this.setPlayingVisual(this.isPlaying);\n }\n\n private buildAvatar(size: number, liveTop = false): HTMLElement {\n const wrap = el('div', { style: { position: 'relative', flexShrink: '0' } });\n this.avatarRing = el('div', {\n style: {\n width: `${size}px`,\n height: `${size}px`,\n borderRadius: '50%',\n border: '3px solid var(--sw-accent)',\n padding: '3px',\n boxShadow: `0 0 24px var(--sw-glow), 0 0 48px rgba(0,0,0,0.25)`,\n background: 'var(--sw-bg)',\n boxSizing: 'border-box',\n },\n });\n const inner = el('div', {\n style: {\n width: '100%',\n height: '100%',\n borderRadius: '50%',\n background: 'var(--sw-bg-inner)',\n overflow: 'hidden',\n },\n });\n if (this.options.avatarUrl) {\n inner.appendChild(\n el('img', {\n attrs: { src: this.options.avatarUrl, alt: '' },\n style: {\n width: '100%',\n height: '100%',\n objectFit: 'cover',\n objectPosition: 'top',\n transform: this.mode === 'wide' ? 'scale(1.15)' : 'scale(1.12)',\n },\n }),\n );\n } else {\n inner.appendChild(\n el('div', {\n text: 'S',\n style: {\n width: '100%',\n height: '100%',\n display: 'flex',\n alignItems: 'center',\n justifyContent: 'center',\n color: 'var(--sw-accent)',\n fontWeight: '900',\n fontSize: `${Math.round(size * 0.28)}px`,\n },\n }),\n );\n }\n this.avatarRing.appendChild(inner);\n wrap.appendChild(this.avatarRing);\n wrap.appendChild(\n el('span', {\n style: {\n position: 'absolute',\n ...(liveTop ? { top: '4px', left: '4px' } : { bottom: '6px', left: '6px' }),\n width: liveTop ? '10px' : '12px',\n height: liveTop ? '10px' : '12px',\n borderRadius: '50%',\n background: 'var(--sw-live)',\n border: '2px solid var(--sw-bg)',\n boxShadow: '0 0 8px var(--sw-live)',\n },\n }),\n );\n return wrap;\n }\n\n private buildMini(): void {\n const size = 72;\n const btn = el('button', {\n attrs: { type: 'button', 'aria-label': this.options.onMiniClick ? 'Expand player' : 'Live DJ' },\n style: {\n position: 'relative',\n padding: '0',\n border: 'none',\n background: 'transparent',\n cursor: this.options.onMiniClick ? 'pointer' : 'default',\n },\n onClick: () => this.options.onMiniClick?.(),\n });\n btn.appendChild(this.buildAvatar(size));\n this.root.appendChild(btn);\n }\n\n private buildWide(): void {\n const avatarSize = 90;\n const shellShadow = `0 8px 40px var(--sw-glow), 0 0 0 1px var(--sw-border)`;\n this.root.style.width = 'min(calc(100vw - 20px), 560px)';\n\n const row = el('div', {\n style: { position: 'relative', display: 'flex', alignItems: 'center', width: '100%' },\n });\n const pillInset = Math.max(avatarSize * 0.42, 40);\n row.appendChild(\n el('div', {\n style: {\n position: 'absolute',\n top: '8px',\n bottom: '8px',\n left: `${pillInset}px`,\n right: '0',\n background: 'var(--sw-bg)',\n border: '1px solid var(--sw-border)',\n borderRadius: '28px 20px 20px 28px',\n boxShadow: shellShadow,\n overflow: 'hidden',\n },\n children: [\n el('div', {\n style: {\n position: 'absolute',\n inset: '0',\n background:\n 'linear-gradient(90deg, color-mix(in srgb, var(--sw-accent) 12%, transparent), transparent)',\n pointerEvents: 'none',\n },\n }),\n ],\n }),\n );\n\n const avatar = this.buildAvatar(avatarSize);\n Object.assign(avatar.style, { position: 'relative', zIndex: '1', marginLeft: '-2px' });\n row.appendChild(avatar);\n\n const info = el('div', {\n style: {\n position: 'relative',\n zIndex: '1',\n flex: '1',\n minWidth: '0',\n marginLeft: '14px',\n padding: '18px 8px',\n },\n });\n const titleRow = el('div', {\n style: { display: 'flex', alignItems: 'baseline', gap: '8px', overflow: 'hidden' },\n });\n this.titleEl = el('h3', {\n style: {\n margin: '0',\n fontSize: 'clamp(15px, 2.8vw, 20px)',\n fontWeight: '900',\n color: 'var(--sw-text)',\n overflow: 'hidden',\n textOverflow: 'ellipsis',\n whiteSpace: 'nowrap',\n },\n });\n titleRow.appendChild(this.titleEl);\n this.artistInlineEl = el('span', {\n style: {\n fontSize: 'clamp(12px, 2.4vw, 15px)',\n fontWeight: '300',\n color: 'var(--sw-text-muted)',\n overflow: 'hidden',\n textOverflow: 'ellipsis',\n whiteSpace: 'nowrap',\n display: 'none',\n },\n });\n titleRow.appendChild(this.artistInlineEl);\n info.appendChild(titleRow);\n this.footerEl = el('p', {\n style: {\n margin: '8px 0 0',\n fontSize: '9px',\n fontWeight: '700',\n color: 'var(--sw-text-muted)',\n letterSpacing: '0.18em',\n textTransform: 'uppercase',\n overflow: 'hidden',\n textOverflow: 'ellipsis',\n whiteSpace: 'nowrap',\n },\n });\n info.appendChild(this.footerEl);\n row.appendChild(info);\n\n const controls = el('div', {\n style: {\n position: 'relative',\n zIndex: '1',\n flexShrink: '0',\n display: 'flex',\n alignItems: 'center',\n gap: '14px',\n paddingRight: '16px',\n },\n });\n this.playBtn = this.createPlayButton(52);\n controls.appendChild(this.playBtn);\n if (!this.isRadio) {\n controls.appendChild(\n el('button', {\n attrs: { type: 'button', 'aria-label': 'Skip' },\n style: {\n border: 'none',\n background: 'transparent',\n color: 'var(--sw-text-muted)',\n display: 'flex',\n cursor: 'pointer',\n },\n onClick: () => this.engine?.skip(),\n children: [svgIcon(ICON_SKIP, 22)],\n }),\n );\n }\n controls.appendChild(this.decoLines());\n row.appendChild(controls);\n this.root.appendChild(row);\n }\n\n private buildLarge(): void {\n this.root.style.width = 'min(calc(100vw - 24px), 380px)';\n const shellShadow = `0 8px 40px var(--sw-glow), 0 0 0 1px var(--sw-border)`;\n const card = el('div', {\n style: {\n borderRadius: '24px',\n border: '1px solid var(--sw-border)',\n background: 'var(--sw-bg)',\n boxShadow: shellShadow,\n padding: '16px',\n display: 'flex',\n flexDirection: 'column',\n gap: '14px',\n fontFamily: 'system-ui, -apple-system, Segoe UI, Roboto, sans-serif',\n },\n });\n\n const head = el('div', { style: { display: 'flex', gap: '12px', alignItems: 'center', minWidth: '0' } });\n head.appendChild(this.buildAvatar(96, true));\n const meta = el('div', { style: { minWidth: '0', flex: '1' } });\n this.titleEl = el('h3', {\n style: {\n margin: '0',\n fontSize: '18px',\n fontWeight: '900',\n color: 'var(--sw-text)',\n overflow: 'hidden',\n textOverflow: 'ellipsis',\n whiteSpace: 'nowrap',\n },\n });\n meta.appendChild(this.titleEl);\n this.artistBlockEl = el('p', {\n style: {\n margin: '6px 0 0',\n fontSize: '13px',\n fontWeight: '700',\n color: 'var(--sw-accent-muted)',\n letterSpacing: '0.08em',\n textTransform: 'uppercase',\n overflow: 'hidden',\n textOverflow: 'ellipsis',\n whiteSpace: 'nowrap',\n display: 'none',\n },\n });\n meta.appendChild(this.artistBlockEl);\n this.footerEl = el('p', {\n style: {\n margin: '6px 0 0',\n fontSize: '10px',\n fontWeight: '700',\n color: 'var(--sw-station)',\n letterSpacing: '0.18em',\n textTransform: 'uppercase',\n overflow: 'hidden',\n textOverflow: 'ellipsis',\n whiteSpace: 'nowrap',\n },\n });\n meta.appendChild(this.footerEl);\n head.appendChild(meta);\n card.appendChild(head);\n card.appendChild(this.buildVisualizer());\n card.appendChild(\n el('p', {\n text: this.options.powerByLabel ?? 'Powered by Synxed',\n style: {\n margin: '0',\n fontSize: '9px',\n fontWeight: '600',\n color: 'var(--sw-text-muted)',\n letterSpacing: '0.12em',\n textTransform: 'uppercase',\n textAlign: 'center',\n },\n }),\n );\n\n const transport = el('div', {\n style: { display: 'flex', alignItems: 'center', justifyContent: 'center', gap: '18px' },\n });\n if (!this.isRadio) {\n transport.appendChild(\n el('button', {\n attrs: { type: 'button', 'aria-label': 'Previous' },\n style: this.roundControlStyle(),\n onClick: () => this.engine?.previous(),\n children: [svgIcon(ICON_SKIP_BACK, 22)],\n }),\n );\n }\n this.playBtn = this.createPlayButton(56);\n transport.appendChild(this.playBtn);\n if (!this.isRadio) {\n transport.appendChild(\n el('button', {\n attrs: { type: 'button', 'aria-label': 'Skip' },\n style: this.roundControlStyle(),\n onClick: () => this.engine?.skip(),\n children: [svgIcon(ICON_SKIP, 22)],\n }),\n );\n }\n card.appendChild(transport);\n this.root.appendChild(card);\n }\n\n private roundControlStyle(): Partial<CSSStyleDeclaration> {\n return {\n width: '46px',\n height: '46px',\n borderRadius: '50%',\n border: '1px solid var(--sw-border)',\n background: 'rgba(255,255,255,0.04)',\n color: 'var(--sw-text)',\n display: 'flex',\n alignItems: 'center',\n justifyContent: 'center',\n cursor: 'pointer',\n };\n }\n\n private createPlayButton(size: number): HTMLButtonElement {\n const btn = el('button', {\n attrs: { type: 'button', 'aria-label': 'Play' },\n style: {\n width: `${size}px`,\n height: `${size}px`,\n borderRadius: '50%',\n border: 'none',\n background: 'var(--sw-accent)',\n color: '#0a0a0a',\n display: 'flex',\n alignItems: 'center',\n justifyContent: 'center',\n cursor: 'pointer',\n boxShadow: '0 0 18px var(--sw-glow)',\n flexShrink: '0',\n },\n onClick: () => this.togglePlay(),\n children: [svgIcon(ICON_PLAY, size >= 56 ? 22 : 20)],\n });\n return btn;\n }\n\n private buildVisualizer(): HTMLElement {\n const box = el('div', {\n style: {\n display: 'flex',\n alignItems: 'flex-end',\n gap: '4px',\n height: '52px',\n padding: '8px 4px',\n borderRadius: '14px',\n border: '1px solid var(--sw-border)',\n overflow: 'hidden',\n },\n });\n for (let i = 0; i < 24; i++) {\n box.appendChild(\n el('span', {\n style: {\n flex: '1',\n minWidth: '3px',\n height: '100%',\n borderRadius: '4px',\n background: 'linear-gradient(180deg, var(--sw-accent), transparent)',\n transformOrigin: 'center bottom',\n animation: `synxed-wp-vbar ${1 + (i % 5) * 0.12}s ease-in-out infinite`,\n animationDelay: `${i * 0.04}s`,\n opacity: '0.85',\n },\n }),\n );\n }\n return box;\n }\n\n private decoLines(): HTMLElement {\n const wrap = el('div', { style: { display: 'flex', flexDirection: 'column', gap: '3px', opacity: '0.22' } });\n wrap.appendChild(\n el('span', {\n style: {\n display: 'block',\n width: '16px',\n height: '2px',\n borderRadius: '999px',\n background: 'var(--sw-text)',\n transform: 'translateX(3px)',\n },\n }),\n );\n wrap.appendChild(\n el('span', {\n style: {\n display: 'block',\n width: '16px',\n height: '2px',\n borderRadius: '999px',\n background: 'var(--sw-text)',\n },\n }),\n );\n return wrap;\n }\n\n private initEngine(): void {\n const { apiKey, serverUrl, source } = this.options;\n const p = new SynxedPlayer({ apiKey, serverUrl, autoConnect: true });\n this.engine = p;\n\n const onState = (state: PlayerState) => {\n this.isPlaying = state.status === 'playing';\n if (state.currentTrack) this.currentTrack = state.currentTrack;\n this.setPlayingVisual(this.isPlaying);\n this.refreshLabels();\n };\n p.on('stateChange', onState);\n p.on('trackChange', (track) => {\n this.currentTrack = track;\n this.refreshLabels();\n });\n\n const start: () => Promise<void> =\n source.type === 'radio'\n ? () => p.playRadio()\n : () => p.playPlaylist({ playlistCode: source.playlistCode });\n\n start().catch(() => undefined);\n }\n\n private startNowPlayingPoll(): void {\n const tick = async () => {\n const np = await fetchRadioNowPlaying(this.options.serverUrl);\n if (!this.destroyed && np) {\n this.nowPlaying = np;\n this.refreshLabels();\n }\n };\n void tick();\n this.pollTimer = setInterval(tick, this.options.nowPlayingPollMs ?? 12_000);\n }\n\n private stopNowPlayingPoll(): void {\n if (this.pollTimer !== null) {\n clearInterval(this.pollTimer);\n this.pollTimer = null;\n }\n }\n\n private displayLine(): { title: string; artist?: string } {\n if (this.isRadio && this.nowPlaying?.title) {\n return parseDisplayTitle(this.nowPlaying.title);\n }\n if (this.currentTrack?.title) {\n return { title: this.currentTrack.title, artist: this.currentTrack.artist };\n }\n return { title: 'Loading…' };\n }\n\n private refreshLabels(): void {\n const line = this.displayLine();\n if (this.titleEl) this.titleEl.textContent = line.title;\n\n if (this.artistInlineEl) {\n const show = Boolean(line.artist);\n this.artistInlineEl.style.display = show ? 'inline' : 'none';\n if (show) this.artistInlineEl.textContent = line.artist!;\n }\n if (this.artistBlockEl) {\n const show = Boolean(line.artist);\n this.artistBlockEl.style.display = show ? 'block' : 'none';\n if (show) this.artistBlockEl.textContent = line.artist!;\n }\n\n if (this.footerEl) {\n const station =\n this.isRadio && (this.nowPlaying?.station ?? this.currentTrack?.title ?? 'Synxed Radio');\n this.footerEl.textContent =\n station && this.isRadio\n ? `${station} · ${this.options.attribution}`\n : (this.options.attribution ?? 'Synxed');\n }\n }\n\n private setPlayingVisual(playing: boolean): void {\n if (this.avatarRing) {\n this.avatarRing.style.animation = playing ? 'synxed-wp-spin 4s linear infinite' : '';\n }\n if (this.playBtn) {\n this.playBtn.replaceChildren(svgIcon(playing ? ICON_PAUSE : ICON_PLAY, 20));\n this.playBtn.setAttribute('aria-label', playing ? 'Pause' : 'Play');\n }\n }\n\n private togglePlay(): void {\n if (!this.engine) return;\n if (this.isPlaying) this.engine.pause();\n else this.engine.resume();\n }\n}\n"]}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "synxed-sdk",
|
|
3
|
-
"version": "0.1.
|
|
3
|
+
"version": "0.1.4",
|
|
4
4
|
"description": "Synxed music SDK — integrate streaming music playback into any web app",
|
|
5
5
|
"bin": {
|
|
6
6
|
"synxed": "dist/cli.js"
|
|
@@ -28,8 +28,7 @@
|
|
|
28
28
|
},
|
|
29
29
|
"dependencies": {
|
|
30
30
|
"howler": "^2.2.4",
|
|
31
|
-
"protobufjs": "^7.3.0"
|
|
32
|
-
"socket.io-client": "^4.7.5"
|
|
31
|
+
"protobufjs": "^7.3.0"
|
|
33
32
|
},
|
|
34
33
|
"devDependencies": {
|
|
35
34
|
"@types/howler": "^2.2.11",
|