init
This commit is contained in:
		
						commit
						6825123206
					
				
							
								
								
									
										83
									
								
								app.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										83
									
								
								app.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,83 @@ | ||||
| // @ts-check
 | ||||
| /// <reference path="./types.d.ts"/>
 | ||||
| 
 | ||||
| import { h, html, render } from "https://esm.sh/htm/preact"; | ||||
| import { createContext, Fragment } from 'https://esm.sh/preact'; | ||||
| import { useReducer, useState, useEffect, useLayoutEffect, useMemo, useRef } from 'https://esm.sh/preact/hooks'; | ||||
| 
 | ||||
| 
 | ||||
| /** @type {(props:{channel:string, interval:number})=>any} */ | ||||
| const App = props => | ||||
| { | ||||
|     /** @type {Boxcast.StateBinding<Array<Boxcast.Broadcast>>} */ | ||||
|     const [ListGet, ListSet] = useState([]); | ||||
| 
 | ||||
|     /** @type {Boxcast.StateBinding<Boxcast.Broadcast>} */ | ||||
|     const Selected = useState(false); | ||||
| 
 | ||||
|     useEffect(()=> | ||||
|     { | ||||
|         Ping(); | ||||
|         const timer = setInterval(Ping, props.interval); | ||||
|         return ()=>clearInterval(timer); | ||||
|     } | ||||
|     , []); | ||||
| 
 | ||||
|     /** @type {()=>Promise} */ | ||||
|     const Ping = async () => | ||||
|     { | ||||
|         const response = await fetch(`https://rest.boxcast.com/channels/${props.channel}/broadcasts?s=starts_at&l=1000`); | ||||
|         const json = await response.json(); | ||||
|         ListSet(json); | ||||
|     }; | ||||
| 
 | ||||
|     return html` | ||||
|     <div class="Playlist"> | ||||
|     ${ | ||||
|         ListGet.map(item => html`<${BroadcastItem} item=${item} active=${Selected} />`) | ||||
|     } | ||||
|     </div> | ||||
|     `;
 | ||||
| } | ||||
| 
 | ||||
| /** @type {(props:{item:Boxcast.Broadcast, active:Boxcast.StateBinding<Boxcast.Broadcast>})=>any} */ | ||||
| const BroadcastItem = ({item, active}) => | ||||
| { | ||||
|     const startDate = useMemo(()=>DateParse(item.starts_at), [item.starts_at]); | ||||
| 
 | ||||
|     return html` | ||||
|     <div key=${item.id} onClick=${()=>active[1](item)}> | ||||
|         <span>${startDate.Month} ${startDate.Date}</span> | ||||
|         <strong>${item.name}</strong> | ||||
|         <span>${startDate.Hours}:${startDate.Minutes} ${startDate.M}</span> | ||||
|         ${ | ||||
|             active[0]?.id == item.id ? html`<dd>Active</dd>` : null | ||||
|         } | ||||
|     </div> | ||||
|     `;
 | ||||
| }; | ||||
| 
 | ||||
| 
 | ||||
| /** @type {(inDate:string)=>Boxcast.Date} */ | ||||
| const DateParse = (inDateString) => | ||||
| { | ||||
|     let date = new Date(inDateString); | ||||
|     /** @type {Boxcast.Date} */ | ||||
|     let obj = { | ||||
|         Zone: date.toString().match(/\(([A-Za-z\s].*)\)/), | ||||
|         Day: ["Sunday", "Monday", "Tuesday", "Wednesday", "Thursday", "Friday", "Saturday"][date.getDay()], | ||||
|         Month: ["January", "February", "March", "April", "May", "June", "July", "August", "September", "October", "November", "December"][date.getMonth()], | ||||
|         Date: date.getDate(), | ||||
|         Hours: date.getHours(), | ||||
|         Minutes: date.getMinutes(), | ||||
|     }; | ||||
| 
 | ||||
|     obj.Zone = obj.Zone ? obj.Zone[1] : "local time"; | ||||
|     obj.M = obj.Hours >= 12 ? "PM" : "AM"; | ||||
|     obj.Hours %= 12; | ||||
|     if(obj.Hours == 0){ obj.Hours = 12; } | ||||
|     if(obj.Minutes < 10){ obj.Minutes = "0"+obj.Minutes; } | ||||
|     return obj; | ||||
| }; | ||||
| 
 | ||||
| export default App; | ||||
							
								
								
									
										49
									
								
								index.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										49
									
								
								index.html
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,49 @@ | ||||
| <!DOCTYPE html> | ||||
| <html> | ||||
| <head></head> | ||||
| <body> | ||||
| 
 | ||||
| <div id="boxcast-player"></div> | ||||
| <div id="boxcast-playlist"></div> | ||||
| <script type="module"> | ||||
| 
 | ||||
| import 'https://js.boxcast.com/v3.min.js'; | ||||
| 
 | ||||
| const Channel = { | ||||
|     Basics: "sfz7ja3rlpoous6usu8a", | ||||
|     Sunday: "gzahmhugrzogttfdtbjj", | ||||
|     Dev: "r3os2zfdnhlquhuypgtp" | ||||
| }; | ||||
| 
 | ||||
| /* | ||||
| const Player = boxcast("#boxcast-player"); | ||||
| Player.loadChannel(Channels.Basics, | ||||
| { | ||||
|     showTitle: true, | ||||
|     showDescription: true, | ||||
|     showCountdown: true, | ||||
| 
 | ||||
|     //selectedBroadcastId: "yo8sefnnejvw3cx3vhup", | ||||
|     //showRelated: true, | ||||
|     //relatedBroadcastsQuery: {q: "timeframe:relevant starts_at:[2022-01-01 TO 2022-12-01]"}, | ||||
|     onLoadBroadcast: console.log, | ||||
| 
 | ||||
|     autoplay: true, | ||||
|     defaultVideo: "next" | ||||
| }); | ||||
| */ | ||||
| 
 | ||||
| 
 | ||||
| import { h, html, render } from "https://esm.sh/htm/preact"; | ||||
| import App from "./app.js"; | ||||
| 
 | ||||
| render( | ||||
|     h(App, {channel:Channel.Basics, interval:5000}), | ||||
|     document.querySelector("#boxcast-playlist") | ||||
| ); | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| </script> | ||||
| </body> | ||||
| </html> | ||||
							
								
								
									
										46
									
								
								types.d.ts
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										46
									
								
								types.d.ts
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,46 @@ | ||||
| namespace Boxcast | ||||
| { | ||||
|     type Broadcast = | ||||
|     { | ||||
|         account_id:string | ||||
|         channel_id:string | ||||
|         description:string | ||||
|         id:string | ||||
|         name:string | ||||
|         poster:string | ||||
|         preview:string | ||||
|         starts_at:string | ||||
|         stops_at:string | ||||
|         tags:Array<string> | ||||
|         timeframe: "future" | "preroll" | "current" | "past" | ||||
|         start?: Date | ||||
|         stop?: Date | ||||
|     } | ||||
| 
 | ||||
|     type Date = | ||||
|     { | ||||
|         Zone: string | RegExpMatchArray | ||||
|         Day: "Sunday" | "Monday" | "Tuesday" | "Wednesday" | "Thursday" | "Friday" | "Saturday" | string | ||||
|         Month: "January" | "February" | "March" | "April" | "May" | "June" | "July" | "August" | "September" | "October" | "November" | "December" | string | ||||
|         Date: number | ||||
|         Hours: number | ||||
|         Minutes: string | number | ||||
|         M?: "AM" | "PM" | ||||
|     } | ||||
| 
 | ||||
|     type StateBinding<T> = [T, (item:T)=>void]; | ||||
| } | ||||
| 
 | ||||
| namespace Store | ||||
| { | ||||
|     type State =  | ||||
|     { | ||||
|         Channel:string, | ||||
|         List:Array<Boxcast.Broadcast>, | ||||
|         Active:false | Boxcast.Broadcast | ||||
|     } | ||||
| 
 | ||||
|     type Action = ["ping-out"] | ["ping-back", Array<Boxcast.Broadcast>] | ["select", Boxcast.Broadcast] | ||||
| 
 | ||||
|     type Reducer = (inState:State, inAction:Action) => inState | ||||
| } | ||||
		Loading…
	
		Reference in New Issue
	
	Block a user