import ServerConnections from '../components/ServerConnections'; import toast from '../components/toast/toast'; import loading from '../components/loading/loading'; import { appRouter } from '../components/appRouter'; import baseAlert from '../components/alert'; import baseConfirm from '../components/confirm/confirm'; import globalize from '../scripts/globalize'; import * as webSettings from './settings/webSettings'; export function getCurrentUser() { return window.ApiClient.getCurrentUser(false); } // TODO: investigate url prefix support for serverAddress function export async function serverAddress() { const apiClient = window.ApiClient; if (apiClient) { return Promise.resolve(apiClient.serverAddress()); } const current = await ServerConnections.getAvailableServers().then(servers => { if (servers.length !== 0) { return Promise.resolve(servers[0].ManualAddress); } }); // TODO this makes things faster but it also blocks the wizard in some scenarios // if (current) return Promise.resolve(current); const urls = []; if (window.location.href.indexOf('/web/') !== -1) { // Do we polyfill String.prototype.includes? const split = window.location.href.split('/web/'); for (let i = split.length - 1; i ; i--) { urls.push(split.slice(0, i).join('/web/')); } } urls.push(window.location.origin); urls.push(`https://${window.location.hostname}:8920`); if (window.location.protocol === 'http') { urls.push(`http://${window.location.hostname}:8096`); } urls.push(...await webSettings.getServers()); console.log('URL candidates:', urls); const promises = urls.map(url => { return fetch(`${url}/System/Info/Public`).then(resp => { return { url: url, response: resp }; }).catch(error => { return Promise.resolve(); }); }); return Promise.all(promises).then(responses => { responses = responses.filter(obj => obj && obj.response.ok); return Promise.all(responses.map(obj => { return { url: obj.url, config: obj.response.json() }; })); }).then(configs => { let selection = configs.find(obj => !obj.config.StartupWizardCompleted); if (!selection) selection = configs[0]; return Promise.resolve(selection.url); }).catch(error => { console.log(error); return Promise.resolve(); }); } export function getCurrentUserId() { const apiClient = window.ApiClient; if (apiClient) { return apiClient.getCurrentUserId(); } return null; } export function onServerChanged(userId, accessToken, apiClient) { ServerConnections.setLocalApiClient(apiClient); } export function logout() { ServerConnections.logout().then(function () { webSettings.getMultiServer().then(multi => { multi ? navigate('selectserver.html') : navigate('login.html'); }); }); } export function getPluginUrl(name) { return 'configurationpage?name=' + encodeURIComponent(name); } export function navigate(url, preserveQueryString) { if (!url) { throw new Error('url cannot be null or empty'); } const queryString = getWindowLocationSearch(); if (preserveQueryString && queryString) { url += queryString; } return appRouter.show(url); } export function processPluginConfigurationUpdateResult() { loading.hide(); toast(globalize.translate('SettingsSaved')); } export function processServerConfigurationUpdateResult(result) { loading.hide(); toast(globalize.translate('SettingsSaved')); } export function processErrorResponse(response) { loading.hide(); let status = '' + response.status; if (response.statusText) { status = response.statusText; } alert({ title: status, message: response.headers ? response.headers.get('X-Application-Error-Code') : null }); } export function alert(options) { if (typeof options == 'string') { toast({ text: options }); } else { baseAlert({ title: options.title || globalize.translate('HeaderAlert'), text: options.message }).then(options.callback || function () {}); } } export function capabilities(appHost) { return Object.assign({ PlayableMediaTypes: ['Audio', 'Video'], SupportedCommands: ['MoveUp', 'MoveDown', 'MoveLeft', 'MoveRight', 'PageUp', 'PageDown', 'PreviousLetter', 'NextLetter', 'ToggleOsd', 'ToggleContextMenu', 'Select', 'Back', 'SendKey', 'SendString', 'GoHome', 'GoToSettings', 'VolumeUp', 'VolumeDown', 'Mute', 'Unmute', 'ToggleMute', 'SetVolume', 'SetAudioStreamIndex', 'SetSubtitleStreamIndex', 'DisplayContent', 'GoToSearch', 'DisplayMessage', 'SetRepeatMode', 'SetShuffleQueue', 'ChannelUp', 'ChannelDown', 'PlayMediaSource', 'PlayTrailers'], SupportsPersistentIdentifier: window.appMode === 'cordova' || window.appMode === 'android', SupportsMediaControl: true }, appHost.getPushTokenInfo()); } export function selectServer() { if (window.NativeShell && typeof window.NativeShell.selectServer === 'function') { window.NativeShell.selectServer(); } else { navigate('selectserver.html'); } } export function hideLoadingMsg() { loading.hide(); } export function showLoadingMsg() { loading.show(); } export function confirm(message, title, callback) { baseConfirm(message, title).then(function() { callback(true); }).catch(function() { callback(false); }); } export const pageClassOn = function(eventName, className, fn) { document.addEventListener(eventName, function (event) { const target = event.target; if (target.classList.contains(className)) { fn.call(target, event); } }); }; export const pageIdOn = function(eventName, id, fn) { document.addEventListener(eventName, function (event) { const target = event.target; if (target.id === id) { fn.call(target, event); } }); }; const Dashboard = { alert, capabilities, confirm, getPluginUrl, getCurrentUser, getCurrentUserId, hideLoadingMsg, logout, navigate, onServerChanged, processErrorResponse, processPluginConfigurationUpdateResult, processServerConfigurationUpdateResult, selectServer, serverAddress, showLoadingMsg }; // This is used in plugins and templates, so keep it defined for now. // TODO: Remove once plugins don't need it window.Dashboard = Dashboard; export default Dashboard;