1
0
mirror of https://github.com/algora-io/tv.git synced 2024-12-06 01:03:34 +02:00
algora-tv/assets/js/app.ts

433 lines
12 KiB
TypeScript
Raw Normal View History

import "phoenix_html";
import { Socket } from "phoenix";
import { LiveSocket, type ViewHook } from "phoenix_live_view";
import topbar from "../vendor/topbar";
import videojs from "../vendor/video";
import "../vendor/videojs-youtube";
// TODO: add eslint & biome
// TODO: enable strict mode
// TODO: eliminate anys
interface PhxEvent extends Event {
target: Element;
detail: Record<string, any>;
}
type PhxEventKey = `js:${string}` | `phx:${string}`;
declare global {
interface Window {
liveSocket: LiveSocket;
addEventListener<K extends keyof WindowEventMap | PhxEventKey>(
type: K,
listener: (
this: Window,
ev: K extends keyof WindowEventMap ? WindowEventMap[K] : PhxEvent
) => any,
options?: boolean | AddEventListenerOptions | undefined
): void;
}
}
let isVisible = (el) =>
!!(el.offsetWidth || el.offsetHeight || el.getClientRects().length > 0);
let execJS = (selector, attr) => {
document
.querySelectorAll(selector)
.forEach((el) => liveSocket.execJS(el, el.getAttribute(attr)));
};
const Hooks = {
Flash: {
mounted() {
let hide = () =>
liveSocket.execJS(this.el, this.el.getAttribute("phx-click"));
this.timer = setTimeout(() => hide(), 8000);
this.el.addEventListener("phx:hide-start", () =>
clearTimeout(this.timer)
);
this.el.addEventListener("mouseover", () => {
clearTimeout(this.timer);
this.timer = setTimeout(() => hide(), 8000);
});
},
destroyed() {
clearTimeout(this.timer);
},
},
Menu: {
getAttr(name) {
let val = this.el.getAttribute(name);
if (val === null) {
throw new Error(`no ${name} attribute configured for menu`);
}
return val;
},
reset() {
this.enabled = false;
this.activeClass = this.getAttr("data-active-class");
this.deactivate(this.menuItems());
this.activeItem = null;
window.removeEventListener("keydown", this.handleKeyDown);
},
destroyed() {
this.reset();
},
mounted() {
this.menuItemsContainer = document.querySelector(
`[aria-labelledby="${this.el.id}"]`
);
this.reset();
this.handleKeyDown = (e) => this.onKeyDown(e);
this.el.addEventListener("keydown", (e) => {
if (
(e.key === "Enter" || e.key === " ") &&
e.currentTarget.isSameNode(this.el)
) {
this.enabled = true;
}
});
this.el.addEventListener("click", (e) => {
if (!e.currentTarget.isSameNode(this.el)) {
return;
}
window.addEventListener("keydown", this.handleKeyDown);
// disable if button clicked and click was not a keyboard event
if (this.enabled) {
window.requestAnimationFrame(() => this.activate(0));
}
});
this.menuItemsContainer.addEventListener("phx:hide-start", () =>
this.reset()
);
},
activate(index, fallbackIndex) {
let menuItems = this.menuItems();
this.activeItem = menuItems[index] || menuItems[fallbackIndex];
this.activeItem.classList.add(this.activeClass);
this.activeItem.focus();
},
deactivate(items) {
items.forEach((item) => item.classList.remove(this.activeClass));
},
menuItems() {
return Array.from(
this.menuItemsContainer.querySelectorAll("[role=menuitem]")
);
},
onKeyDown(e) {
if (e.key === "Escape") {
document.body.click();
this.el.focus();
this.reset();
} else if (e.key === "Enter" && !this.activeItem) {
this.activate(0);
} else if (e.key === "Enter") {
this.activeItem.click();
}
if (e.key === "ArrowDown") {
e.preventDefault();
let menuItems = this.menuItems();
this.deactivate(menuItems);
this.activate(menuItems.indexOf(this.activeItem) + 1, 0);
} else if (e.key === "ArrowUp") {
e.preventDefault();
let menuItems = this.menuItems();
this.deactivate(menuItems);
this.activate(
menuItems.indexOf(this.activeItem) - 1,
menuItems.length - 1
);
} else if (e.key === "Tab") {
e.preventDefault();
}
},
},
VideoPlayer: {
mounted() {
const backdrop = document.querySelector("#video-backdrop");
this.player = videojs("video-player", {
autoplay: true,
liveui: true,
html5: {
vhs: {
llhls: true,
},
},
});
2024-05-07 01:41:15 +02:00
const playVideo = (opts: {
url: string;
title: string;
player_type: string;
current_time?: number;
channel_name: string;
}) => {
const setMediaSession = () => {
if (!("mediaSession" in navigator)) {
return;
}
navigator.mediaSession.metadata = new MediaMetadata({
title: opts.title,
artist: opts.channel_name,
album: "Algora TV",
artwork: [96, 128, 192, 256, 384, 512, 1024].map((px) => ({
2024-05-07 01:41:15 +02:00
src: `https://console.algora.io/asset/storage/v1/object/public/images/algora-gradient-${px}px.png`,
sizes: `${px}x${px}`,
type: "image/png",
})),
});
};
this.player.options({
2024-05-07 01:41:15 +02:00
techOrder: [
opts.player_type === "video/youtube" ? "youtube" : "html5",
],
...(opts.current_time && opts.player_type === "video/youtube"
? { youtube: { customVars: { start: opts.current_time } } }
2024-05-01 22:34:05 +02:00
: {}),
});
2024-05-07 01:41:15 +02:00
this.player.src({ src: opts.url, type: opts.player_type });
this.player.play();
2024-05-07 01:41:15 +02:00
setMediaSession();
if (opts.current_time && opts.player_type !== "video/youtube") {
this.player.currentTime(opts.current_time);
2024-05-01 22:34:05 +02:00
}
this.player.el().parentElement.classList.remove("hidden");
this.player.el().parentElement.classList.add("flex");
if (backdrop) {
backdrop.classList.remove("opacity-10");
backdrop.classList.add("opacity-20");
}
};
this.handleEvent("play_video", playVideo);
},
},
Chat: {
mounted() {
this.el.scrollTo(0, this.el.scrollHeight);
},
updated() {
const pixelsBelowBottom =
this.el.scrollHeight - this.el.clientHeight - this.el.scrollTop;
if (pixelsBelowBottom < 200) {
this.el.scrollTo(0, this.el.scrollHeight);
}
},
},
2024-05-25 16:12:19 +02:00
TimezoneDetector: {
mounted() {
this.pushEvent("get_timezone", {
tz: Intl.DateTimeFormat().resolvedOptions().timeZone,
});
},
},
NavBar: {
mounted() {
const offset = 16;
this.isOpaque = false;
const onScroll = () => {
if (!this.isOpaque && window.scrollY > offset) {
this.isOpaque = true;
this.el.classList.add("bg-gray-950");
this.el.classList.remove("bg-transparent");
} else if (this.isOpaque && window.scrollY <= offset) {
this.isOpaque = false;
this.el.classList.add("bg-transparent");
this.el.classList.remove("bg-gray-950");
}
};
window.addEventListener("scroll", onScroll, { passive: true });
},
},
} satisfies Record<string, Partial<ViewHook> & Record<string, unknown>>;
// Accessible focus handling
let Focus = {
focusMain() {
let target =
document.querySelector<HTMLElement>("main h1") ||
document.querySelector<HTMLElement>("main");
if (target) {
let origTabIndex = target.tabIndex;
target.tabIndex = -1;
target.focus();
target.tabIndex = origTabIndex;
}
},
// Subject to the W3C Software License at https://www.w3.org/Consortium/Legal/2015/copyright-software-and-document
isFocusable(el) {
if (
el.tabIndex > 0 ||
(el.tabIndex === 0 && el.getAttribute("tabIndex") !== null)
) {
return true;
}
if (el.disabled) {
return false;
}
switch (el.nodeName) {
case "A":
return !!el.href && el.rel !== "ignore";
case "INPUT":
return el.type != "hidden" && el.type !== "file";
case "BUTTON":
case "SELECT":
case "TEXTAREA":
return true;
default:
return false;
}
},
// Subject to the W3C Software License at https://www.w3.org/Consortium/Legal/2015/copyright-software-and-document
attemptFocus(el) {
if (!el) {
return;
}
if (!this.isFocusable(el)) {
return false;
}
try {
el.focus();
} catch (e) {}
return document.activeElement === el;
},
// Subject to the W3C Software License at https://www.w3.org/Consortium/Legal/2015/copyright-software-and-document
focusFirstDescendant(el) {
for (let i = 0; i < el.childNodes.length; i++) {
let child = el.childNodes[i];
if (this.attemptFocus(child) || this.focusFirstDescendant(child)) {
return true;
}
}
return false;
},
// Subject to the W3C Software License at https://www.w3.org/Consortium/Legal/2015/copyright-software-and-document
focusLastDescendant(element) {
for (let i = element.childNodes.length - 1; i >= 0; i--) {
let child = element.childNodes[i];
if (this.attemptFocus(child) || this.focusLastDescendant(child)) {
return true;
}
}
return false;
},
};
let csrfToken = document
.querySelector("meta[name='csrf-token']")!
.getAttribute("content");
let liveSocket = new LiveSocket("/live", Socket, {
hooks: Hooks,
params: { _csrf_token: csrfToken },
dom: {
onNodeAdded(node) {
if (node instanceof HTMLElement && node.autofocus) {
node.focus();
}
return node;
},
},
});
let routeUpdated = () => {
// TODO: uncomment
// Focus.focusMain();
2024-05-01 22:34:05 +02:00
const player = document.querySelector("#video-player")?.parentElement;
if (!player) {
return;
}
2024-05-05 01:27:04 +02:00
const { pathname } = new URL(window.location.href);
if (pathname.endsWith("/embed")) {
return;
}
2024-05-01 22:34:05 +02:00
const pipClasses = [
"fixed",
"bottom-0",
"right-0",
"z-[1000]",
"w-[100vw]",
"sm:w-[30vw]",
];
2024-05-05 01:27:04 +02:00
if (/^\/[^\/]+\/\d+$/.test(pathname)) {
2024-05-01 22:34:05 +02:00
player.classList.add("lg:pr-[24rem]");
player.classList.remove(...pipClasses);
} else {
player.classList.remove("lg:pr-[24rem]");
player.classList.add(...pipClasses);
}
};
// Show progress bar on live navigation and form submits
topbar.config({
2024-05-01 22:34:05 +02:00
barColors: { 0: "rgba(79, 70, 229, 1)" },
shadowColor: "rgba(0, 0, 0, .3)",
});
window.addEventListener("phx:page-loading-start", (info) =>
topbar.delayedShow(200)
);
window.addEventListener("phx:page-loading-stop", (info) => topbar.hide());
// Accessible routing
window.addEventListener("phx:page-loading-stop", routeUpdated);
window.addEventListener("js:exec", (e) =>
e.target[e.detail.call](...e.detail.args)
);
window.addEventListener("js:focus", (e) => {
let parent = document.querySelector(e.detail.parent);
if (parent && isVisible(parent)) {
(e.target as any).focus();
}
});
window.addEventListener("js:focus-closest", (e) => {
let el = e.target;
let sibling = el.nextElementSibling;
while (sibling) {
if (isVisible(sibling) && Focus.attemptFocus(sibling)) {
return;
}
sibling = sibling.nextElementSibling;
}
sibling = el.previousElementSibling;
while (sibling) {
if (isVisible(sibling) && Focus.attemptFocus(sibling)) {
return;
}
sibling = sibling.previousElementSibling;
}
Focus.attemptFocus((el as any).parent) || Focus.focusMain();
});
window.addEventListener("phx:remove-el", (e) =>
document.getElementById(e.detail.id)?.remove()
);
// connect if there are any LiveViews on the page
liveSocket.getSocket().onOpen(() => execJS("#connection-status", "js-hide"));
liveSocket.getSocket().onError(() => execJS("#connection-status", "js-show"));
liveSocket.connect();
// expose liveSocket on window for web console debug logs and latency simulation:
// >> liveSocket.enableDebug()
// >> liveSocket.enableLatencySim(1000) // enabled for duration of browser session
// >> liveSocket.disableLatencySim()
window.liveSocket = liveSocket;