const LOCAL_TEST = window.location.href.indexOf('localhost:') >= 0; const URL_PREFIX = LOCAL_TEST ? 'http://localhost:8080' : ''; const URL_STATUS = `${URL_PREFIX}/status`; const URL_CHAT_HISTORY = `${URL_PREFIX}/chat`; const URL_STREAM = `${URL_PREFIX}/hls/stream.m3u8`; const URL_WEBSOCKET = LOCAL_TEST ? 'wss://goth.land/entry' : `${location.protocol === 'https:' ? 'wss' : 'ws'}://${location.host}/entry`; const POSTER_DEFAULT = `${URL_PREFIX}/img/logo.png`; const POSTER_THUMB = `${URL_PREFIX}/thumbnail.jpg`; const URL_CONFIG = `${URL_PREFIX}/config`; const URL_OWNCAST = 'https://github.com/gabek/owncast'; // used in footer // Webscoket setup const SOCKET_MESSAGE_TYPES = { CHAT: 'CHAT', PING: 'PING' } // Video setup const VIDEO_ID = 'video'; const VIDEO_SRC = { src: URL_STREAM, type: 'application/x-mpegURL', }; const VIDEO_OPTIONS = { autoplay: false, liveui: true, // try this liveTracker: { trackingThreshold: 0, }, sources: [VIDEO_SRC], }; // local storage keys for chat const KEY_USERNAME = 'owncast_username'; const KEY_AVATAR = 'owncast_avatar'; const KEY_CHAT_DISPLAYED = 'owncast_chat'; const KEY_CHAT_FIRST_MESSAGE_SENT = 'owncast_first_message_sent'; const TIMER_STATUS_UPDATE = 5000; // ms const TIMER_WEBSOCKET_RECONNECT = 5000; // ms const TIMER_DISABLE_CHAT_AFTER_OFFLINE = 5 * 60 * 1000; // 5 mins const TIMER_STREAM_DURATION_COUNTER = 1000; const TEMP_IMAGE = ''; const MESSAGE_OFFLINE = 'Stream is offline.'; const MESSAGE_ONLINE = 'Stream is online'; const CHAT_INITIAL_PLACEHOLDER_TEXT = 'Type here to chat, no account necessary.'; const CHAT_PLACEHOLDER_TEXT = 'Message'; const CHAT_PLACEHOLDER_OFFLINE = 'Chat is offline.'; function getLocalStorage(key) { try { return localStorage.getItem(key); } catch (e) { } return null; } function setLocalStorage(key, value) { try { if (value !== "" && value !== null) { localStorage.setItem(key, value); } else { localStorage.removeItem(key); } return true; } catch (e) {} return false; } function clearLocalStorage(key) { localStorage.removeItem(key); } // jump down to the max height of a div, with a slight delay function jumpToBottom(element) { if (!element) return; setTimeout(() => { element.scrollTo({ top: element.scrollHeight, left: 0, behavior: 'smooth' }); }, 50, element); } // convert newlines to
s function addNewlines(str) { return str.replace(/(?:\r\n|\r|\n)/g, '
'); } function pluralize(string, count) { if (count === 1) { return string; } else { return string + "s"; } } // Trying to determine if browser is mobile/tablet. // Source: https://developer.mozilla.org/en-US/docs/Web/HTTP/Browser_detection_using_the_user_agent function hasTouchScreen() { var hasTouchScreen = false; if ("maxTouchPoints" in navigator) { hasTouchScreen = navigator.maxTouchPoints > 0; } else if ("msMaxTouchPoints" in navigator) { hasTouchScreen = navigator.msMaxTouchPoints > 0; } else { var mQ = window.matchMedia && matchMedia("(pointer:coarse)"); if (mQ && mQ.media === "(pointer:coarse)") { hasTouchScreen = !!mQ.matches; } else if ('orientation' in window) { hasTouchScreen = true; // deprecated, but good fallback } else { // Only as a last resort, fall back to user agent sniffing var UA = navigator.userAgent; hasTouchScreen = ( /\b(BlackBerry|webOS|iPhone|IEMobile)\b/i.test(UA) || /\b(Android|Windows Phone|iPad|iPod)\b/i.test(UA) ); } } return hasTouchScreen; } // generate random avatar from https://robohash.org function generateAvatar(hash) { const avatarSource = 'https://robohash.org/'; const optionSize = '?size=80x80'; const optionSet = '&set=set3'; const optionBg = ''; // or &bgset=bg1 or bg2 return avatarSource + hash + optionSize + optionSet + optionBg; } function generateUsername() { return `User ${(Math.floor(Math.random() * 42) + 1)}`; } function secondsToHMMSS(seconds = 0) { const finiteSeconds = Number.isFinite(+seconds) ? Math.abs(seconds) : 0; const hours = Math.floor(finiteSeconds / 3600); const hoursString = hours ? `${hours}:` : ''; const mins = Math.floor((finiteSeconds / 60) % 60); const minString = mins < 10 ? `0${mins}:` : `${mins}:`; const secs = Math.floor(finiteSeconds % 60); const secsString = secs < 10 ? `0${secs}` : `${secs}`; return hoursString + minString + secsString; } function setVHvar() { var vh = window.innerHeight * 0.01; // Then we set the value in the --vh custom property to the root of the document document.documentElement.style.setProperty('--vh', `${vh}px`); console.log("== new vh", vh) }