mirror of
https://github.com/jellyfin/jellyfin-web
synced 2025-03-30 19:56:21 +00:00
Merge branch 'master' into deinterlace-improvements
This commit is contained in:
commit
1c96222957
160 changed files with 2876 additions and 3252 deletions
|
@ -44,6 +44,7 @@ module.exports = {
|
||||||
'no-unused-vars': ['error', { 'vars': 'all', 'args': 'none', 'ignoreRestSiblings': true }],
|
'no-unused-vars': ['error', { 'vars': 'all', 'args': 'none', 'ignoreRestSiblings': true }],
|
||||||
'one-var': ['error', 'never'],
|
'one-var': ['error', 'never'],
|
||||||
'padded-blocks': ['error', 'never'],
|
'padded-blocks': ['error', 'never'],
|
||||||
|
//'prefer-const': ['error', {'destructuring': 'all'}],
|
||||||
'quotes': ['error', 'single', { 'avoidEscape': true, 'allowTemplateLiterals': false }],
|
'quotes': ['error', 'single', { 'avoidEscape': true, 'allowTemplateLiterals': false }],
|
||||||
'semi': ['error'],
|
'semi': ['error'],
|
||||||
'space-before-blocks': ['error'],
|
'space-before-blocks': ['error'],
|
||||||
|
|
13
package.json
13
package.json
|
@ -95,6 +95,7 @@
|
||||||
"src/components/alert.js",
|
"src/components/alert.js",
|
||||||
"src/components/alphaPicker/alphaPicker.js",
|
"src/components/alphaPicker/alphaPicker.js",
|
||||||
"src/components/appFooter/appFooter.js",
|
"src/components/appFooter/appFooter.js",
|
||||||
|
"src/components/apphost.js",
|
||||||
"src/components/autoFocuser.js",
|
"src/components/autoFocuser.js",
|
||||||
"src/components/backdrop/backdrop.js",
|
"src/components/backdrop/backdrop.js",
|
||||||
"src/components/cardbuilder/cardBuilder.js",
|
"src/components/cardbuilder/cardBuilder.js",
|
||||||
|
@ -110,6 +111,7 @@
|
||||||
"src/components/favoriteitems.js",
|
"src/components/favoriteitems.js",
|
||||||
"src/components/fetchhelper.js",
|
"src/components/fetchhelper.js",
|
||||||
"src/components/filterdialog/filterdialog.js",
|
"src/components/filterdialog/filterdialog.js",
|
||||||
|
"src/components/filtermenu/filtermenu.js",
|
||||||
"src/components/focusManager.js",
|
"src/components/focusManager.js",
|
||||||
"src/components/groupedcards.js",
|
"src/components/groupedcards.js",
|
||||||
"src/components/guide/guide.js",
|
"src/components/guide/guide.js",
|
||||||
|
@ -140,6 +142,7 @@
|
||||||
"src/components/metadataEditor/metadataEditor.js",
|
"src/components/metadataEditor/metadataEditor.js",
|
||||||
"src/components/metadataEditor/personEditor.js",
|
"src/components/metadataEditor/personEditor.js",
|
||||||
"src/components/multiSelect/multiSelect.js",
|
"src/components/multiSelect/multiSelect.js",
|
||||||
|
"src/components/notifications/notifications.js",
|
||||||
"src/components/nowPlayingBar/nowPlayingBar.js",
|
"src/components/nowPlayingBar/nowPlayingBar.js",
|
||||||
"src/components/playback/brightnessosd.js",
|
"src/components/playback/brightnessosd.js",
|
||||||
"src/components/playback/mediasession.js",
|
"src/components/playback/mediasession.js",
|
||||||
|
@ -162,6 +165,7 @@
|
||||||
"src/components/recordingcreator/seriesrecordingeditor.js",
|
"src/components/recordingcreator/seriesrecordingeditor.js",
|
||||||
"src/components/recordingcreator/recordinghelper.js",
|
"src/components/recordingcreator/recordinghelper.js",
|
||||||
"src/components/refreshdialog/refreshdialog.js",
|
"src/components/refreshdialog/refreshdialog.js",
|
||||||
|
"src/components/qualityOptions.js",
|
||||||
"src/components/remotecontrol/remotecontrol.js",
|
"src/components/remotecontrol/remotecontrol.js",
|
||||||
"src/components/sanatizefilename.js",
|
"src/components/sanatizefilename.js",
|
||||||
"src/components/scrollManager.js",
|
"src/components/scrollManager.js",
|
||||||
|
@ -184,13 +188,17 @@
|
||||||
"src/components/syncPlay/playbackPermissionManager.js",
|
"src/components/syncPlay/playbackPermissionManager.js",
|
||||||
"src/components/syncPlay/syncPlayManager.js",
|
"src/components/syncPlay/syncPlayManager.js",
|
||||||
"src/components/syncPlay/timeSyncManager.js",
|
"src/components/syncPlay/timeSyncManager.js",
|
||||||
|
"src/components/themeMediaPlayer.js",
|
||||||
"src/components/tabbedview/tabbedview.js",
|
"src/components/tabbedview/tabbedview.js",
|
||||||
"src/components/viewManager/viewManager.js",
|
"src/components/viewManager/viewManager.js",
|
||||||
"src/components/tvproviders/schedulesdirect.js",
|
"src/components/tvproviders/schedulesdirect.js",
|
||||||
"src/components/tvproviders/xmltv.js",
|
"src/components/tvproviders/xmltv.js",
|
||||||
"src/components/toast/toast.js",
|
"src/components/toast/toast.js",
|
||||||
|
"src/components/tunerPicker.js",
|
||||||
"src/components/upnextdialog/upnextdialog.js",
|
"src/components/upnextdialog/upnextdialog.js",
|
||||||
|
"src/components/userdatabuttons/userdatabuttons.js",
|
||||||
"src/components/viewContainer.js",
|
"src/components/viewContainer.js",
|
||||||
|
"src/components/viewSettings/viewSettings.js",
|
||||||
"src/components/castSenderApi.js",
|
"src/components/castSenderApi.js",
|
||||||
"src/controllers/session/addServer/index.js",
|
"src/controllers/session/addServer/index.js",
|
||||||
"src/controllers/session/forgotPassword/index.js",
|
"src/controllers/session/forgotPassword/index.js",
|
||||||
|
@ -218,8 +226,8 @@
|
||||||
"src/controllers/dashboard/metadataImages.js",
|
"src/controllers/dashboard/metadataImages.js",
|
||||||
"src/controllers/dashboard/metadatanfo.js",
|
"src/controllers/dashboard/metadatanfo.js",
|
||||||
"src/controllers/dashboard/networking.js",
|
"src/controllers/dashboard/networking.js",
|
||||||
"src/controllers/dashboard/notifications/notification.js",
|
"src/controllers/dashboard/notifications/notification/index.js",
|
||||||
"src/controllers/dashboard/notifications/notifications.js",
|
"src/controllers/dashboard/notifications/notifications/index.js",
|
||||||
"src/controllers/dashboard/playback.js",
|
"src/controllers/dashboard/playback.js",
|
||||||
"src/controllers/dashboard/plugins/add/index.js",
|
"src/controllers/dashboard/plugins/add/index.js",
|
||||||
"src/controllers/dashboard/plugins/installed/index.js",
|
"src/controllers/dashboard/plugins/installed/index.js",
|
||||||
|
@ -253,6 +261,7 @@
|
||||||
"src/controllers/searchpage.js",
|
"src/controllers/searchpage.js",
|
||||||
"src/controllers/livetv/livetvguide.js",
|
"src/controllers/livetv/livetvguide.js",
|
||||||
"src/controllers/livetvtuner.js",
|
"src/controllers/livetvtuner.js",
|
||||||
|
"src/controllers/livetv/livetvsuggested.js",
|
||||||
"src/controllers/livetvstatus.js",
|
"src/controllers/livetvstatus.js",
|
||||||
"src/controllers/livetvguideprovider.js",
|
"src/controllers/livetvguideprovider.js",
|
||||||
"src/controllers/livetvsettings.js",
|
"src/controllers/livetvsettings.js",
|
||||||
|
|
|
@ -7,7 +7,6 @@
|
||||||
}
|
}
|
||||||
|
|
||||||
.osdPoster img,
|
.osdPoster img,
|
||||||
.pageContainer,
|
|
||||||
.videoOsdBottom {
|
.videoOsdBottom {
|
||||||
bottom: 0;
|
bottom: 0;
|
||||||
left: 0;
|
left: 0;
|
||||||
|
@ -248,11 +247,6 @@
|
||||||
animation: spin 4s linear infinite;
|
animation: spin 4s linear infinite;
|
||||||
}
|
}
|
||||||
|
|
||||||
.pageContainer {
|
|
||||||
top: 0;
|
|
||||||
position: fixed;
|
|
||||||
}
|
|
||||||
|
|
||||||
@media all and (max-width: 30em) {
|
@media all and (max-width: 30em) {
|
||||||
.btnFastForward,
|
.btnFastForward,
|
||||||
.btnRewind,
|
.btnRewind,
|
||||||
|
|
|
@ -2,159 +2,159 @@
|
||||||
* require.js module definitions bundled by webpack
|
* require.js module definitions bundled by webpack
|
||||||
*/
|
*/
|
||||||
// Use define from require.js not webpack's define
|
// Use define from require.js not webpack's define
|
||||||
var _define = window.define;
|
const _define = window.define;
|
||||||
|
|
||||||
// fetch
|
// fetch
|
||||||
var fetch = require('whatwg-fetch');
|
const fetch = require('whatwg-fetch');
|
||||||
_define('fetch', function() {
|
_define('fetch', function() {
|
||||||
return fetch;
|
return fetch;
|
||||||
});
|
});
|
||||||
|
|
||||||
// Blurhash
|
// Blurhash
|
||||||
var blurhash = require('blurhash');
|
const blurhash = require('blurhash');
|
||||||
_define('blurhash', function() {
|
_define('blurhash', function() {
|
||||||
return blurhash;
|
return blurhash;
|
||||||
});
|
});
|
||||||
|
|
||||||
// query-string
|
// query-string
|
||||||
var query = require('query-string');
|
const query = require('query-string');
|
||||||
_define('queryString', function() {
|
_define('queryString', function() {
|
||||||
return query;
|
return query;
|
||||||
});
|
});
|
||||||
|
|
||||||
// flvjs
|
// flvjs
|
||||||
var flvjs = require('flv.js/dist/flv').default;
|
const flvjs = require('flv.js/dist/flv').default;
|
||||||
_define('flvjs', function() {
|
_define('flvjs', function() {
|
||||||
return flvjs;
|
return flvjs;
|
||||||
});
|
});
|
||||||
|
|
||||||
// jstree
|
// jstree
|
||||||
var jstree = require('jstree');
|
const jstree = require('jstree');
|
||||||
require('jstree/dist/themes/default/style.css');
|
require('jstree/dist/themes/default/style.css');
|
||||||
_define('jstree', function() {
|
_define('jstree', function() {
|
||||||
return jstree;
|
return jstree;
|
||||||
});
|
});
|
||||||
|
|
||||||
// jquery
|
// jquery
|
||||||
var jquery = require('jquery');
|
const jquery = require('jquery');
|
||||||
_define('jQuery', function() {
|
_define('jQuery', function() {
|
||||||
return jquery;
|
return jquery;
|
||||||
});
|
});
|
||||||
|
|
||||||
// hlsjs
|
// hlsjs
|
||||||
var hlsjs = require('hls.js');
|
const hlsjs = require('hls.js');
|
||||||
_define('hlsjs', function() {
|
_define('hlsjs', function() {
|
||||||
return hlsjs;
|
return hlsjs;
|
||||||
});
|
});
|
||||||
|
|
||||||
// howler
|
// howler
|
||||||
var howler = require('howler');
|
const howler = require('howler');
|
||||||
_define('howler', function() {
|
_define('howler', function() {
|
||||||
return howler;
|
return howler;
|
||||||
});
|
});
|
||||||
|
|
||||||
// resize-observer-polyfill
|
// resize-observer-polyfill
|
||||||
var resize = require('resize-observer-polyfill').default;
|
const resize = require('resize-observer-polyfill').default;
|
||||||
_define('resize-observer-polyfill', function() {
|
_define('resize-observer-polyfill', function() {
|
||||||
return resize;
|
return resize;
|
||||||
});
|
});
|
||||||
|
|
||||||
// swiper
|
// swiper
|
||||||
var swiper = require('swiper/js/swiper');
|
const swiper = require('swiper/js/swiper');
|
||||||
require('swiper/css/swiper.min.css');
|
require('swiper/css/swiper.min.css');
|
||||||
_define('swiper', function() {
|
_define('swiper', function() {
|
||||||
return swiper;
|
return swiper;
|
||||||
});
|
});
|
||||||
|
|
||||||
// sortable
|
// sortable
|
||||||
var sortable = require('sortablejs').default;
|
const sortable = require('sortablejs').default;
|
||||||
_define('sortable', function() {
|
_define('sortable', function() {
|
||||||
return sortable;
|
return sortable;
|
||||||
});
|
});
|
||||||
|
|
||||||
// webcomponents
|
// webcomponents
|
||||||
var webcomponents = require('webcomponents.js/webcomponents-lite');
|
const webcomponents = require('webcomponents.js/webcomponents-lite');
|
||||||
_define('webcomponents', function() {
|
_define('webcomponents', function() {
|
||||||
return webcomponents;
|
return webcomponents;
|
||||||
});
|
});
|
||||||
|
|
||||||
// libass-wasm
|
// libass-wasm
|
||||||
var libassWasm = require('libass-wasm');
|
const libassWasm = require('libass-wasm');
|
||||||
_define('JavascriptSubtitlesOctopus', function() {
|
_define('JavascriptSubtitlesOctopus', function() {
|
||||||
return libassWasm;
|
return libassWasm;
|
||||||
});
|
});
|
||||||
|
|
||||||
// material-icons
|
// material-icons
|
||||||
var materialIcons = require('material-design-icons-iconfont/dist/material-design-icons.css');
|
const materialIcons = require('material-design-icons-iconfont/dist/material-design-icons.css');
|
||||||
_define('material-icons', function() {
|
_define('material-icons', function() {
|
||||||
return materialIcons;
|
return materialIcons;
|
||||||
});
|
});
|
||||||
|
|
||||||
// noto font
|
// noto font
|
||||||
var noto = require('jellyfin-noto');
|
const noto = require('jellyfin-noto');
|
||||||
_define('jellyfin-noto', function () {
|
_define('jellyfin-noto', function () {
|
||||||
return noto;
|
return noto;
|
||||||
});
|
});
|
||||||
|
|
||||||
var epubjs = require('epubjs');
|
const epubjs = require('epubjs');
|
||||||
_define('epubjs', function () {
|
_define('epubjs', function () {
|
||||||
return epubjs;
|
return epubjs;
|
||||||
});
|
});
|
||||||
|
|
||||||
// page.js
|
// page.js
|
||||||
var page = require('page');
|
const page = require('page');
|
||||||
_define('page', function() {
|
_define('page', function() {
|
||||||
return page;
|
return page;
|
||||||
});
|
});
|
||||||
|
|
||||||
// core-js
|
// core-js
|
||||||
var polyfill = require('@babel/polyfill/dist/polyfill');
|
const polyfill = require('@babel/polyfill/dist/polyfill');
|
||||||
_define('polyfill', function () {
|
_define('polyfill', function () {
|
||||||
return polyfill;
|
return polyfill;
|
||||||
});
|
});
|
||||||
|
|
||||||
// domtokenlist-shim
|
// domtokenlist-shim
|
||||||
var classlist = require('classlist.js');
|
const classlist = require('classlist.js');
|
||||||
_define('classlist-polyfill', function () {
|
_define('classlist-polyfill', function () {
|
||||||
return classlist;
|
return classlist;
|
||||||
});
|
});
|
||||||
|
|
||||||
// Date-FNS
|
// Date-FNS
|
||||||
var dateFns = require('date-fns');
|
const dateFns = require('date-fns');
|
||||||
_define('date-fns', function () {
|
_define('date-fns', function () {
|
||||||
return dateFns;
|
return dateFns;
|
||||||
});
|
});
|
||||||
|
|
||||||
var dateFnsLocale = require('date-fns/locale');
|
const dateFnsLocale = require('date-fns/locale');
|
||||||
_define('date-fns/locale', function () {
|
_define('date-fns/locale', function () {
|
||||||
return dateFnsLocale;
|
return dateFnsLocale;
|
||||||
});
|
});
|
||||||
|
|
||||||
var fast_text_encoding = require('fast-text-encoding');
|
const fast_text_encoding = require('fast-text-encoding');
|
||||||
_define('fast-text-encoding', function () {
|
_define('fast-text-encoding', function () {
|
||||||
return fast_text_encoding;
|
return fast_text_encoding;
|
||||||
});
|
});
|
||||||
|
|
||||||
// intersection-observer
|
// intersection-observer
|
||||||
var intersection_observer = require('intersection-observer');
|
const intersection_observer = require('intersection-observer');
|
||||||
_define('intersection-observer', function () {
|
_define('intersection-observer', function () {
|
||||||
return intersection_observer;
|
return intersection_observer;
|
||||||
});
|
});
|
||||||
|
|
||||||
// screenfull
|
// screenfull
|
||||||
var screenfull = require('screenfull');
|
const screenfull = require('screenfull');
|
||||||
_define('screenfull', function () {
|
_define('screenfull', function () {
|
||||||
return screenfull;
|
return screenfull;
|
||||||
});
|
});
|
||||||
|
|
||||||
// headroom.js
|
// headroom.js
|
||||||
var headroom = require('headroom.js/dist/headroom');
|
const headroom = require('headroom.js/dist/headroom');
|
||||||
_define('headroom', function () {
|
_define('headroom', function () {
|
||||||
return headroom;
|
return headroom;
|
||||||
});
|
});
|
||||||
|
|
||||||
// apiclient
|
// apiclient
|
||||||
var apiclient = require('jellyfin-apiclient');
|
const apiclient = require('jellyfin-apiclient');
|
||||||
|
|
||||||
_define('apiclient', function () {
|
_define('apiclient', function () {
|
||||||
return apiclient.ApiClient;
|
return apiclient.ApiClient;
|
||||||
|
|
|
@ -33,7 +33,7 @@
|
||||||
|
|
||||||
<div class="formDialogFooter">
|
<div class="formDialogFooter">
|
||||||
<button is="emby-button" type="submit" class="raised button-submit block formDialogFooterItem">
|
<button is="emby-button" type="submit" class="raised button-submit block formDialogFooterItem">
|
||||||
<span>${ButtonAdd}</span>
|
<span>${Add}</span>
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
</form>
|
</form>
|
||||||
|
|
|
@ -9,14 +9,14 @@ import 'scrollStyles';
|
||||||
import 'listViewStyle';
|
import 'listViewStyle';
|
||||||
|
|
||||||
function getOffsets(elems) {
|
function getOffsets(elems) {
|
||||||
let results = [];
|
const results = [];
|
||||||
|
|
||||||
if (!document) {
|
if (!document) {
|
||||||
return results;
|
return results;
|
||||||
}
|
}
|
||||||
|
|
||||||
for (const elem of elems) {
|
for (const elem of elems) {
|
||||||
let box = elem.getBoundingClientRect();
|
const box = elem.getBoundingClientRect();
|
||||||
|
|
||||||
results.push({
|
results.push({
|
||||||
top: box.top,
|
top: box.top,
|
||||||
|
@ -34,7 +34,7 @@ function getPosition(options, dlg) {
|
||||||
const windowHeight = windowSize.innerHeight;
|
const windowHeight = windowSize.innerHeight;
|
||||||
const windowWidth = windowSize.innerWidth;
|
const windowWidth = windowSize.innerWidth;
|
||||||
|
|
||||||
let pos = getOffsets([options.positionTo])[0];
|
const pos = getOffsets([options.positionTo])[0];
|
||||||
|
|
||||||
if (options.positionY !== 'top') {
|
if (options.positionY !== 'top') {
|
||||||
pos.top += (pos.height || 0) / 2;
|
pos.top += (pos.height || 0) / 2;
|
||||||
|
@ -82,7 +82,7 @@ export function show(options) {
|
||||||
// positionTo
|
// positionTo
|
||||||
// showCancel
|
// showCancel
|
||||||
// title
|
// title
|
||||||
let dialogOptions = {
|
const dialogOptions = {
|
||||||
removeOnClose: true,
|
removeOnClose: true,
|
||||||
enableHistory: options.enableHistory,
|
enableHistory: options.enableHistory,
|
||||||
scrollY: false
|
scrollY: false
|
||||||
|
@ -103,7 +103,7 @@ export function show(options) {
|
||||||
dialogOptions.autoFocus = false;
|
dialogOptions.autoFocus = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
let dlg = dialogHelper.createDialog(dialogOptions);
|
const dlg = dialogHelper.createDialog(dialogOptions);
|
||||||
|
|
||||||
if (isFullscreen) {
|
if (isFullscreen) {
|
||||||
dlg.classList.add('actionsheet-fullscreen');
|
dlg.classList.add('actionsheet-fullscreen');
|
||||||
|
@ -129,7 +129,7 @@ export function show(options) {
|
||||||
}
|
}
|
||||||
|
|
||||||
let renderIcon = false;
|
let renderIcon = false;
|
||||||
let icons = [];
|
const icons = [];
|
||||||
let itemIcon;
|
let itemIcon;
|
||||||
for (const item of options.items) {
|
for (const item of options.items) {
|
||||||
itemIcon = item.icon || (item.selected ? 'check' : null);
|
itemIcon = item.icon || (item.selected ? 'check' : null);
|
||||||
|
@ -241,7 +241,7 @@ export function show(options) {
|
||||||
centerFocus(dlg.querySelector('.actionSheetScroller'), false, true);
|
centerFocus(dlg.querySelector('.actionSheetScroller'), false, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
let btnCloseActionSheet = dlg.querySelector('.btnCloseActionSheet');
|
const btnCloseActionSheet = dlg.querySelector('.btnCloseActionSheet');
|
||||||
if (btnCloseActionSheet) {
|
if (btnCloseActionSheet) {
|
||||||
btnCloseActionSheet.addEventListener('click', function () {
|
btnCloseActionSheet.addEventListener('click', function () {
|
||||||
dialogHelper.close(dlg);
|
dialogHelper.close(dlg);
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
define(['loading', 'globalize', 'events', 'viewManager', 'skinManager', 'backdrop', 'browser', 'page', 'appSettings', 'apphost', 'connectionManager'], function (loading, globalize, events, viewManager, skinManager, backdrop, browser, page, appSettings, appHost, connectionManager) {
|
define(['loading', 'globalize', 'events', 'viewManager', 'skinManager', 'backdrop', 'browser', 'page', 'appSettings', 'apphost', 'connectionManager'], function (loading, globalize, events, viewManager, skinManager, backdrop, browser, page, appSettings, appHost, connectionManager) {
|
||||||
'use strict';
|
'use strict';
|
||||||
|
|
||||||
|
appHost = appHost.default || appHost;
|
||||||
viewManager = viewManager.default || viewManager;
|
viewManager = viewManager.default || viewManager;
|
||||||
browser = browser.default || browser;
|
browser = browser.default || browser;
|
||||||
loading = loading.default || loading;
|
loading = loading.default || loading;
|
||||||
|
|
|
@ -1,447 +1,412 @@
|
||||||
define(['appSettings', 'browser', 'events', 'htmlMediaHelper', 'webSettings', 'globalize'], function (appSettings, browser, events, htmlMediaHelper, webSettings, globalize) {
|
import appSettings from 'appSettings';
|
||||||
'use strict';
|
import browser from 'browser';
|
||||||
|
import events from 'events';
|
||||||
|
import * as htmlMediaHelper from 'htmlMediaHelper';
|
||||||
|
import * as webSettings from 'webSettings';
|
||||||
|
import globalize from 'globalize';
|
||||||
|
|
||||||
browser = browser.default || browser;
|
function getBaseProfileOptions(item) {
|
||||||
|
const disableHlsVideoAudioCodecs = [];
|
||||||
|
|
||||||
function getBaseProfileOptions(item) {
|
if (item && htmlMediaHelper.enableHlsJsPlayer(item.RunTimeTicks, item.MediaType)) {
|
||||||
var disableHlsVideoAudioCodecs = [];
|
if (browser.edge) {
|
||||||
|
disableHlsVideoAudioCodecs.push('mp3');
|
||||||
|
}
|
||||||
|
|
||||||
if (item && htmlMediaHelper.enableHlsJsPlayer(item.RunTimeTicks, item.MediaType)) {
|
disableHlsVideoAudioCodecs.push('ac3');
|
||||||
if (browser.edge) {
|
disableHlsVideoAudioCodecs.push('eac3');
|
||||||
disableHlsVideoAudioCodecs.push('mp3');
|
disableHlsVideoAudioCodecs.push('opus');
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
enableMkvProgressive: false,
|
||||||
|
disableHlsVideoAudioCodecs: disableHlsVideoAudioCodecs
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
function getDeviceProfile(item, options = {}) {
|
||||||
|
return new Promise(function (resolve) {
|
||||||
|
import('browserdeviceprofile').then(({default: profileBuilder}) => {
|
||||||
|
let profile;
|
||||||
|
|
||||||
|
if (window.NativeShell) {
|
||||||
|
profile = window.NativeShell.AppHost.getDeviceProfile(profileBuilder);
|
||||||
|
} else {
|
||||||
|
const builderOpts = getBaseProfileOptions(item);
|
||||||
|
builderOpts.enableSsaRender = (item && !options.isRetry && appSettings.get('subtitleburnin') !== 'allcomplexformats');
|
||||||
|
profile = profileBuilder(builderOpts);
|
||||||
}
|
}
|
||||||
|
|
||||||
disableHlsVideoAudioCodecs.push('ac3');
|
resolve(profile);
|
||||||
disableHlsVideoAudioCodecs.push('eac3');
|
|
||||||
disableHlsVideoAudioCodecs.push('opus');
|
|
||||||
}
|
|
||||||
|
|
||||||
return {
|
|
||||||
enableMkvProgressive: false,
|
|
||||||
disableHlsVideoAudioCodecs: disableHlsVideoAudioCodecs
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
function getDeviceProfileForWindowsUwp(item) {
|
|
||||||
return new Promise(function (resolve, reject) {
|
|
||||||
require(['browserdeviceprofile', 'environments/windows-uwp/mediacaps'], function (profileBuilder, uwpMediaCaps) {
|
|
||||||
var profileOptions = getBaseProfileOptions(item);
|
|
||||||
profileOptions.supportsDts = uwpMediaCaps.supportsDTS();
|
|
||||||
profileOptions.supportsTrueHd = uwpMediaCaps.supportsDolby();
|
|
||||||
profileOptions.audioChannels = uwpMediaCaps.getAudioChannels();
|
|
||||||
resolve(profileBuilder(profileOptions));
|
|
||||||
});
|
|
||||||
});
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function escapeRegExp(str) {
|
||||||
|
return str.replace(/([.*+?^=!:${}()|[\]/\\])/g, '\\$1');
|
||||||
|
}
|
||||||
|
|
||||||
|
function replaceAll(originalString, strReplace, strWith) {
|
||||||
|
const strReplace2 = escapeRegExp(strReplace);
|
||||||
|
const reg = new RegExp(strReplace2, 'ig');
|
||||||
|
return originalString.replace(reg, strWith);
|
||||||
|
}
|
||||||
|
|
||||||
|
function generateDeviceId() {
|
||||||
|
const keys = [];
|
||||||
|
|
||||||
|
if (keys.push(navigator.userAgent), keys.push(new Date().getTime()), self.btoa) {
|
||||||
|
const result = replaceAll(btoa(keys.join('|')), '=', '1');
|
||||||
|
return Promise.resolve(result);
|
||||||
}
|
}
|
||||||
|
|
||||||
function getDeviceProfile(item, options) {
|
return Promise.resolve(new Date().getTime());
|
||||||
options = options || {};
|
}
|
||||||
|
|
||||||
if (self.Windows) {
|
function getDeviceId() {
|
||||||
return getDeviceProfileForWindowsUwp(item);
|
const key = '_deviceId2';
|
||||||
}
|
const deviceId = appSettings.get(key);
|
||||||
|
|
||||||
return new Promise(function (resolve) {
|
if (deviceId) {
|
||||||
require(['browserdeviceprofile'], function (profileBuilder) {
|
return Promise.resolve(deviceId);
|
||||||
var profile;
|
|
||||||
|
|
||||||
if (window.NativeShell) {
|
|
||||||
profile = window.NativeShell.AppHost.getDeviceProfile(profileBuilder);
|
|
||||||
} else {
|
|
||||||
var builderOpts = getBaseProfileOptions(item);
|
|
||||||
builderOpts.enableSsaRender = (item && !options.isRetry && appSettings.get('subtitleburnin') !== 'allcomplexformats');
|
|
||||||
profile = profileBuilder(builderOpts);
|
|
||||||
}
|
|
||||||
|
|
||||||
resolve(profile);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function escapeRegExp(str) {
|
return generateDeviceId().then(function (deviceId) {
|
||||||
return str.replace(/([.*+?^=!:${}()|\[\]\/\\])/g, '\\$1');
|
appSettings.set(key, deviceId);
|
||||||
|
return deviceId;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function getDeviceName() {
|
||||||
|
var deviceName;
|
||||||
|
if (browser.tizen) {
|
||||||
|
deviceName = 'Samsung Smart TV';
|
||||||
|
} else if (browser.web0s) {
|
||||||
|
deviceName = 'LG Smart TV';
|
||||||
|
} else if (browser.operaTv) {
|
||||||
|
deviceName = 'Opera TV';
|
||||||
|
} else if (browser.xboxOne) {
|
||||||
|
deviceName = 'Xbox One';
|
||||||
|
} else if (browser.ps4) {
|
||||||
|
deviceName = 'Sony PS4';
|
||||||
|
} else if (browser.chrome) {
|
||||||
|
deviceName = 'Chrome';
|
||||||
|
} else if (browser.edgeChromium) {
|
||||||
|
deviceName = 'Edge Chromium';
|
||||||
|
} else if (browser.edge) {
|
||||||
|
deviceName = 'Edge';
|
||||||
|
} else if (browser.firefox) {
|
||||||
|
deviceName = 'Firefox';
|
||||||
|
} else if (browser.opera) {
|
||||||
|
deviceName = 'Opera';
|
||||||
|
} else if (browser.safari) {
|
||||||
|
deviceName = 'Safari';
|
||||||
|
} else {
|
||||||
|
deviceName = 'Web Browser';
|
||||||
}
|
}
|
||||||
|
|
||||||
function replaceAll(originalString, strReplace, strWith) {
|
if (browser.ipad) {
|
||||||
var strReplace2 = escapeRegExp(strReplace);
|
deviceName += ' iPad';
|
||||||
var reg = new RegExp(strReplace2, 'ig');
|
} else if (browser.iphone) {
|
||||||
return originalString.replace(reg, strWith);
|
deviceName += ' iPhone';
|
||||||
|
} else if (browser.android) {
|
||||||
|
deviceName += ' Android';
|
||||||
}
|
}
|
||||||
|
|
||||||
function generateDeviceId() {
|
return deviceName;
|
||||||
var keys = [];
|
}
|
||||||
|
|
||||||
if (keys.push(navigator.userAgent), keys.push(new Date().getTime()), self.btoa) {
|
function supportsVoiceInput() {
|
||||||
var result = replaceAll(btoa(keys.join('|')), '=', '1');
|
if (!browser.tv) {
|
||||||
return Promise.resolve(result);
|
return window.SpeechRecognition || window.webkitSpeechRecognition || window.mozSpeechRecognition || window.oSpeechRecognition || window.msSpeechRecognition;
|
||||||
}
|
|
||||||
|
|
||||||
return Promise.resolve(new Date().getTime());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function getDeviceId() {
|
return false;
|
||||||
var key = '_deviceId2';
|
}
|
||||||
var deviceId = appSettings.get(key);
|
|
||||||
|
|
||||||
if (deviceId) {
|
|
||||||
return Promise.resolve(deviceId);
|
|
||||||
}
|
|
||||||
|
|
||||||
return generateDeviceId().then(function (deviceId) {
|
|
||||||
appSettings.set(key, deviceId);
|
|
||||||
return deviceId;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
function getDeviceName() {
|
|
||||||
var deviceName;
|
|
||||||
if (browser.tizen) {
|
|
||||||
deviceName = 'Samsung Smart TV';
|
|
||||||
} else if (browser.web0s) {
|
|
||||||
deviceName = 'LG Smart TV';
|
|
||||||
} else if (browser.operaTv) {
|
|
||||||
deviceName = 'Opera TV';
|
|
||||||
} else if (browser.xboxOne) {
|
|
||||||
deviceName = 'Xbox One';
|
|
||||||
} else if (browser.ps4) {
|
|
||||||
deviceName = 'Sony PS4';
|
|
||||||
} else if (browser.chrome) {
|
|
||||||
deviceName = 'Chrome';
|
|
||||||
} else if (browser.edgeChromium) {
|
|
||||||
deviceName = 'Edge Chromium';
|
|
||||||
} else if (browser.edge) {
|
|
||||||
deviceName = 'Edge';
|
|
||||||
} else if (browser.firefox) {
|
|
||||||
deviceName = 'Firefox';
|
|
||||||
} else if (browser.opera) {
|
|
||||||
deviceName = 'Opera';
|
|
||||||
} else if (browser.safari) {
|
|
||||||
deviceName = 'Safari';
|
|
||||||
} else {
|
|
||||||
deviceName = 'Web Browser';
|
|
||||||
}
|
|
||||||
|
|
||||||
if (browser.ipad) {
|
|
||||||
deviceName += ' iPad';
|
|
||||||
} else if (browser.iphone) {
|
|
||||||
deviceName += ' iPhone';
|
|
||||||
} else if (browser.android) {
|
|
||||||
deviceName += ' Android';
|
|
||||||
}
|
|
||||||
|
|
||||||
return deviceName;
|
|
||||||
}
|
|
||||||
|
|
||||||
function supportsVoiceInput() {
|
|
||||||
if (!browser.tv) {
|
|
||||||
return window.SpeechRecognition || window.webkitSpeechRecognition || window.mozSpeechRecognition || window.oSpeechRecognition || window.msSpeechRecognition;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
function supportsFullscreen() {
|
||||||
|
if (browser.tv) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
function supportsFullscreen() {
|
const element = document.documentElement;
|
||||||
if (browser.tv) {
|
return (element.requestFullscreen || element.mozRequestFullScreen || element.webkitRequestFullscreen || element.msRequestFullscreen) || document.createElement('video').webkitEnterFullscreen;
|
||||||
return false;
|
}
|
||||||
}
|
|
||||||
|
|
||||||
var element = document.documentElement;
|
function getDefaultLayout() {
|
||||||
return (element.requestFullscreen || element.mozRequestFullScreen || element.webkitRequestFullscreen || element.msRequestFullscreen) || document.createElement('video').webkitEnterFullscreen;
|
return 'desktop';
|
||||||
}
|
}
|
||||||
|
|
||||||
function getSyncProfile() {
|
|
||||||
return new Promise(function (resolve) {
|
|
||||||
require(['browserdeviceprofile', 'appSettings'], function (profileBuilder, appSettings) {
|
|
||||||
var profile;
|
|
||||||
|
|
||||||
if (window.NativeShell) {
|
|
||||||
profile = window.NativeShell.AppHost.getSyncProfile(profileBuilder, appSettings);
|
|
||||||
} else {
|
|
||||||
profile = profileBuilder();
|
|
||||||
profile.MaxStaticMusicBitrate = appSettings.maxStaticMusicBitrate();
|
|
||||||
}
|
|
||||||
|
|
||||||
resolve(profile);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
function getDefaultLayout() {
|
|
||||||
return 'desktop';
|
|
||||||
}
|
|
||||||
|
|
||||||
function supportsHtmlMediaAutoplay() {
|
|
||||||
if (browser.edgeUwp || browser.tizen || browser.web0s || browser.orsay || browser.operaTv || browser.ps4 || browser.xboxOne) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (browser.mobile) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
function supportsHtmlMediaAutoplay() {
|
||||||
|
if (browser.edgeUwp || browser.tizen || browser.web0s || browser.orsay || browser.operaTv || browser.ps4 || browser.xboxOne) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
function supportsCue() {
|
if (browser.mobile) {
|
||||||
try {
|
return false;
|
||||||
var video = document.createElement('video');
|
|
||||||
var style = document.createElement('style');
|
|
||||||
|
|
||||||
style.textContent = 'video::cue {background: inherit}';
|
|
||||||
document.body.appendChild(style);
|
|
||||||
document.body.appendChild(video);
|
|
||||||
|
|
||||||
var cue = window.getComputedStyle(video, '::cue').background;
|
|
||||||
document.body.removeChild(style);
|
|
||||||
document.body.removeChild(video);
|
|
||||||
|
|
||||||
return !!cue.length;
|
|
||||||
} catch (err) {
|
|
||||||
console.error('error detecting cue support: ' + err);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function onAppVisible() {
|
return true;
|
||||||
if (isHidden) {
|
}
|
||||||
isHidden = false;
|
|
||||||
console.debug('triggering app resume event');
|
function supportsCue() {
|
||||||
events.trigger(appHost, 'resume');
|
try {
|
||||||
}
|
const video = document.createElement('video');
|
||||||
|
const style = document.createElement('style');
|
||||||
|
|
||||||
|
style.textContent = 'video::cue {background: inherit}';
|
||||||
|
document.body.appendChild(style);
|
||||||
|
document.body.appendChild(video);
|
||||||
|
|
||||||
|
const cue = window.getComputedStyle(video, '::cue').background;
|
||||||
|
document.body.removeChild(style);
|
||||||
|
document.body.removeChild(video);
|
||||||
|
|
||||||
|
return !!cue.length;
|
||||||
|
} catch (err) {
|
||||||
|
console.error('error detecting cue support: ' + err);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function onAppVisible() {
|
||||||
|
if (isHidden) {
|
||||||
|
isHidden = false;
|
||||||
|
console.debug('triggering app resume event');
|
||||||
|
events.trigger(appHost, 'resume');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function onAppHidden() {
|
||||||
|
if (!isHidden) {
|
||||||
|
isHidden = true;
|
||||||
|
console.debug('app is hidden');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const supportedFeatures = function () {
|
||||||
|
const features = [];
|
||||||
|
|
||||||
|
if (navigator.share) {
|
||||||
|
features.push('sharing');
|
||||||
}
|
}
|
||||||
|
|
||||||
function onAppHidden() {
|
if (!browser.edgeUwp && !browser.tv && !browser.xboxOne && !browser.ps4) {
|
||||||
if (!isHidden) {
|
features.push('filedownload');
|
||||||
isHidden = true;
|
|
||||||
console.debug('app is hidden');
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
var supportedFeatures = function () {
|
if (browser.operaTv || browser.tizen || browser.orsay || browser.web0s) {
|
||||||
var features = [];
|
features.push('exit');
|
||||||
|
} else {
|
||||||
|
features.push('exitmenu');
|
||||||
|
features.push('plugins');
|
||||||
|
}
|
||||||
|
|
||||||
if (navigator.share) {
|
if (!browser.operaTv && !browser.tizen && !browser.orsay && !browser.web0s && !browser.ps4) {
|
||||||
features.push('sharing');
|
features.push('externallinks');
|
||||||
}
|
features.push('externalpremium');
|
||||||
|
}
|
||||||
|
|
||||||
if (!browser.edgeUwp && !browser.tv && !browser.xboxOne && !browser.ps4) {
|
if (!browser.operaTv) {
|
||||||
features.push('filedownload');
|
features.push('externallinkdisplay');
|
||||||
}
|
}
|
||||||
|
|
||||||
if (browser.operaTv || browser.tizen || browser.orsay || browser.web0s) {
|
if (supportsVoiceInput()) {
|
||||||
features.push('exit');
|
features.push('voiceinput');
|
||||||
|
}
|
||||||
|
|
||||||
|
if (supportsHtmlMediaAutoplay()) {
|
||||||
|
features.push('htmlaudioautoplay');
|
||||||
|
features.push('htmlvideoautoplay');
|
||||||
|
}
|
||||||
|
|
||||||
|
if (browser.edgeUwp) {
|
||||||
|
features.push('sync');
|
||||||
|
}
|
||||||
|
|
||||||
|
if (supportsFullscreen()) {
|
||||||
|
features.push('fullscreenchange');
|
||||||
|
}
|
||||||
|
|
||||||
|
if (browser.tv || browser.xboxOne || browser.ps4 || browser.mobile) {
|
||||||
|
features.push('physicalvolumecontrol');
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!browser.tv && !browser.xboxOne && !browser.ps4) {
|
||||||
|
features.push('remotecontrol');
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!browser.operaTv && !browser.tizen && !browser.orsay && !browser.web0s && !browser.edgeUwp) {
|
||||||
|
features.push('remotevideo');
|
||||||
|
}
|
||||||
|
|
||||||
|
features.push('displaylanguage');
|
||||||
|
features.push('otherapppromotions');
|
||||||
|
features.push('displaymode');
|
||||||
|
features.push('targetblank');
|
||||||
|
features.push('screensaver');
|
||||||
|
|
||||||
|
webSettings.getMultiServer().then(enabled => {
|
||||||
|
if (enabled) features.push('multiserver');
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!browser.orsay && (browser.firefox || browser.ps4 || browser.edge || supportsCue())) {
|
||||||
|
features.push('subtitleappearancesettings');
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!browser.orsay) {
|
||||||
|
features.push('subtitleburnsettings');
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!browser.tv && !browser.ps4 && !browser.xboxOne) {
|
||||||
|
features.push('fileinput');
|
||||||
|
}
|
||||||
|
|
||||||
|
if (browser.chrome || browser.edgeChromium) {
|
||||||
|
features.push('chromecast');
|
||||||
|
}
|
||||||
|
|
||||||
|
return features;
|
||||||
|
}();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Do exit according to platform
|
||||||
|
*/
|
||||||
|
function doExit() {
|
||||||
|
try {
|
||||||
|
if (window.NativeShell) {
|
||||||
|
window.NativeShell.AppHost.exit();
|
||||||
|
} else if (browser.tizen) {
|
||||||
|
tizen.application.getCurrentApplication().exit();
|
||||||
|
} else if (browser.web0s) {
|
||||||
|
webOS.platformBack();
|
||||||
} else {
|
} else {
|
||||||
features.push('exitmenu');
|
window.close();
|
||||||
features.push('plugins');
|
|
||||||
}
|
}
|
||||||
|
} catch (err) {
|
||||||
|
console.error('error closing application: ' + err);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (!browser.operaTv && !browser.tizen && !browser.orsay && !browser.web0s && !browser.ps4) {
|
let exitPromise;
|
||||||
features.push('externallinks');
|
|
||||||
features.push('externalpremium');
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!browser.operaTv) {
|
/**
|
||||||
features.push('externallinkdisplay');
|
* Ask user for exit
|
||||||
}
|
*/
|
||||||
|
function askForExit() {
|
||||||
if (supportsVoiceInput()) {
|
if (exitPromise) {
|
||||||
features.push('voiceinput');
|
return;
|
||||||
}
|
|
||||||
|
|
||||||
if (supportsHtmlMediaAutoplay()) {
|
|
||||||
features.push('htmlaudioautoplay');
|
|
||||||
features.push('htmlvideoautoplay');
|
|
||||||
}
|
|
||||||
|
|
||||||
if (browser.edgeUwp) {
|
|
||||||
features.push('sync');
|
|
||||||
}
|
|
||||||
|
|
||||||
if (supportsFullscreen()) {
|
|
||||||
features.push('fullscreenchange');
|
|
||||||
}
|
|
||||||
|
|
||||||
if (browser.tv || browser.xboxOne || browser.ps4 || browser.mobile) {
|
|
||||||
features.push('physicalvolumecontrol');
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!browser.tv && !browser.xboxOne && !browser.ps4) {
|
|
||||||
features.push('remotecontrol');
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!browser.operaTv && !browser.tizen && !browser.orsay && !browser.web0s && !browser.edgeUwp) {
|
|
||||||
features.push('remotevideo');
|
|
||||||
}
|
|
||||||
|
|
||||||
features.push('displaylanguage');
|
|
||||||
features.push('otherapppromotions');
|
|
||||||
features.push('displaymode');
|
|
||||||
features.push('targetblank');
|
|
||||||
features.push('screensaver');
|
|
||||||
|
|
||||||
webSettings.getMultiServer().then(enabled => {
|
|
||||||
if (enabled) features.push('multiserver');
|
|
||||||
});
|
|
||||||
|
|
||||||
if (!browser.orsay && (browser.firefox || browser.ps4 || browser.edge || supportsCue())) {
|
|
||||||
features.push('subtitleappearancesettings');
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!browser.orsay) {
|
|
||||||
features.push('subtitleburnsettings');
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!browser.tv && !browser.ps4 && !browser.xboxOne) {
|
|
||||||
features.push('fileinput');
|
|
||||||
}
|
|
||||||
|
|
||||||
if (browser.chrome || browser.edgeChromium) {
|
|
||||||
features.push('chromecast');
|
|
||||||
}
|
|
||||||
|
|
||||||
return features;
|
|
||||||
}();
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Do exit according to platform
|
|
||||||
*/
|
|
||||||
function doExit() {
|
|
||||||
try {
|
|
||||||
if (window.NativeShell) {
|
|
||||||
window.NativeShell.AppHost.exit();
|
|
||||||
} else if (browser.tizen) {
|
|
||||||
tizen.application.getCurrentApplication().exit();
|
|
||||||
} else if (browser.web0s) {
|
|
||||||
webOS.platformBack();
|
|
||||||
} else {
|
|
||||||
window.close();
|
|
||||||
}
|
|
||||||
} catch (err) {
|
|
||||||
console.error('error closing application: ' + err);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
var exitPromise;
|
import('actionsheet').then(({default: actionsheet}) => {
|
||||||
|
exitPromise = actionsheet.show({
|
||||||
/**
|
title: globalize.translate('MessageConfirmAppExit'),
|
||||||
* Ask user for exit
|
items: [
|
||||||
*/
|
{id: 'yes', name: globalize.translate('Yes')},
|
||||||
function askForExit() {
|
{id: 'no', name: globalize.translate('No')}
|
||||||
if (exitPromise) {
|
]
|
||||||
return;
|
}).then(function (value) {
|
||||||
}
|
if (value === 'yes') {
|
||||||
|
|
||||||
require(['actionsheet'], function (actionsheet) {
|
|
||||||
exitPromise = actionsheet.show({
|
|
||||||
title: globalize.translate('MessageConfirmAppExit'),
|
|
||||||
items: [
|
|
||||||
{id: 'yes', name: globalize.translate('Yes')},
|
|
||||||
{id: 'no', name: globalize.translate('No')}
|
|
||||||
]
|
|
||||||
}).then(function (value) {
|
|
||||||
if (value === 'yes') {
|
|
||||||
doExit();
|
|
||||||
}
|
|
||||||
}).finally(function () {
|
|
||||||
exitPromise = null;
|
|
||||||
});
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
var deviceId;
|
|
||||||
var deviceName;
|
|
||||||
var appName = 'Jellyfin Web';
|
|
||||||
var appVersion = '10.7.0';
|
|
||||||
|
|
||||||
var appHost = {
|
|
||||||
getWindowState: function () {
|
|
||||||
return document.windowState || 'Normal';
|
|
||||||
},
|
|
||||||
setWindowState: function (state) {
|
|
||||||
alert('setWindowState is not supported and should not be called');
|
|
||||||
},
|
|
||||||
exit: function () {
|
|
||||||
if (!!window.appMode && browser.tizen) {
|
|
||||||
askForExit();
|
|
||||||
} else {
|
|
||||||
doExit();
|
doExit();
|
||||||
}
|
}
|
||||||
},
|
}).finally(function () {
|
||||||
supports: function (command) {
|
exitPromise = null;
|
||||||
if (window.NativeShell) {
|
});
|
||||||
return window.NativeShell.AppHost.supports(command);
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
return supportedFeatures.indexOf(command.toLowerCase()) !== -1;
|
let deviceId;
|
||||||
},
|
let deviceName;
|
||||||
preferVisualCards: browser.android || browser.chrome,
|
const appName = 'Jellyfin Web';
|
||||||
getSyncProfile: getSyncProfile,
|
const appVersion = '10.7.0';
|
||||||
getDefaultLayout: function () {
|
|
||||||
if (window.NativeShell) {
|
|
||||||
return window.NativeShell.AppHost.getDefaultLayout();
|
|
||||||
}
|
|
||||||
|
|
||||||
return getDefaultLayout();
|
const appHost = {
|
||||||
},
|
getWindowState: function () {
|
||||||
getDeviceProfile: getDeviceProfile,
|
return document.windowState || 'Normal';
|
||||||
init: function () {
|
},
|
||||||
if (window.NativeShell) {
|
setWindowState: function () {
|
||||||
return window.NativeShell.AppHost.init();
|
alert('setWindowState is not supported and should not be called');
|
||||||
}
|
},
|
||||||
|
exit: function () {
|
||||||
deviceName = getDeviceName();
|
if (!!window.appMode && browser.tizen) {
|
||||||
getDeviceId().then(function (id) {
|
askForExit();
|
||||||
deviceId = id;
|
|
||||||
});
|
|
||||||
},
|
|
||||||
deviceName: function () {
|
|
||||||
return window.NativeShell ? window.NativeShell.AppHost.deviceName() : deviceName;
|
|
||||||
},
|
|
||||||
deviceId: function () {
|
|
||||||
return window.NativeShell ? window.NativeShell.AppHost.deviceId() : deviceId;
|
|
||||||
},
|
|
||||||
appName: function () {
|
|
||||||
return window.NativeShell ? window.NativeShell.AppHost.appName() : appName;
|
|
||||||
},
|
|
||||||
appVersion: function () {
|
|
||||||
return window.NativeShell ? window.NativeShell.AppHost.appVersion() : appVersion;
|
|
||||||
},
|
|
||||||
getPushTokenInfo: function () {
|
|
||||||
return {};
|
|
||||||
},
|
|
||||||
setUserScalable: function (scalable) {
|
|
||||||
if (!browser.tv) {
|
|
||||||
var att = scalable ? 'width=device-width, initial-scale=1, minimum-scale=1, user-scalable=yes' : 'width=device-width, initial-scale=1, minimum-scale=1, maximum-scale=1, user-scalable=no';
|
|
||||||
document.querySelector('meta[name=viewport]').setAttribute('content', att);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
var isHidden = false;
|
|
||||||
var hidden;
|
|
||||||
var visibilityChange;
|
|
||||||
|
|
||||||
if (typeof document.hidden !== 'undefined') { /* eslint-disable-line compat/compat */
|
|
||||||
hidden = 'hidden';
|
|
||||||
visibilityChange = 'visibilitychange';
|
|
||||||
} else if (typeof document.webkitHidden !== 'undefined') {
|
|
||||||
hidden = 'webkitHidden';
|
|
||||||
visibilityChange = 'webkitvisibilitychange';
|
|
||||||
}
|
|
||||||
|
|
||||||
document.addEventListener(visibilityChange, function () {
|
|
||||||
/* eslint-disable-next-line compat/compat */
|
|
||||||
if (document[hidden]) {
|
|
||||||
onAppHidden();
|
|
||||||
} else {
|
} else {
|
||||||
onAppVisible();
|
doExit();
|
||||||
|
}
|
||||||
|
},
|
||||||
|
supports: function (command) {
|
||||||
|
if (window.NativeShell) {
|
||||||
|
return window.NativeShell.AppHost.supports(command);
|
||||||
}
|
}
|
||||||
}, false);
|
|
||||||
|
|
||||||
if (self.addEventListener) {
|
return supportedFeatures.indexOf(command.toLowerCase()) !== -1;
|
||||||
self.addEventListener('focus', onAppVisible);
|
},
|
||||||
self.addEventListener('blur', onAppHidden);
|
preferVisualCards: browser.android || browser.chrome,
|
||||||
|
getDefaultLayout: function () {
|
||||||
|
if (window.NativeShell) {
|
||||||
|
return window.NativeShell.AppHost.getDefaultLayout();
|
||||||
|
}
|
||||||
|
|
||||||
|
return getDefaultLayout();
|
||||||
|
},
|
||||||
|
getDeviceProfile: getDeviceProfile,
|
||||||
|
init: function () {
|
||||||
|
if (window.NativeShell) {
|
||||||
|
return window.NativeShell.AppHost.init();
|
||||||
|
}
|
||||||
|
|
||||||
|
deviceName = getDeviceName();
|
||||||
|
getDeviceId().then(function (id) {
|
||||||
|
deviceId = id;
|
||||||
|
});
|
||||||
|
},
|
||||||
|
deviceName: function () {
|
||||||
|
return window.NativeShell ? window.NativeShell.AppHost.deviceName() : deviceName;
|
||||||
|
},
|
||||||
|
deviceId: function () {
|
||||||
|
return window.NativeShell ? window.NativeShell.AppHost.deviceId() : deviceId;
|
||||||
|
},
|
||||||
|
appName: function () {
|
||||||
|
return window.NativeShell ? window.NativeShell.AppHost.appName() : appName;
|
||||||
|
},
|
||||||
|
appVersion: function () {
|
||||||
|
return window.NativeShell ? window.NativeShell.AppHost.appVersion() : appVersion;
|
||||||
|
},
|
||||||
|
getPushTokenInfo: function () {
|
||||||
|
return {};
|
||||||
|
},
|
||||||
|
setUserScalable: function (scalable) {
|
||||||
|
if (!browser.tv) {
|
||||||
|
const att = scalable ? 'width=device-width, initial-scale=1, minimum-scale=1, user-scalable=yes' : 'width=device-width, initial-scale=1, minimum-scale=1, maximum-scale=1, user-scalable=no';
|
||||||
|
document.querySelector('meta[name=viewport]').setAttribute('content', att);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
};
|
||||||
|
|
||||||
return appHost;
|
let isHidden = false;
|
||||||
});
|
let hidden;
|
||||||
|
let visibilityChange;
|
||||||
|
|
||||||
|
if (typeof document.hidden !== 'undefined') { /* eslint-disable-line compat/compat */
|
||||||
|
hidden = 'hidden';
|
||||||
|
visibilityChange = 'visibilitychange';
|
||||||
|
} else if (typeof document.webkitHidden !== 'undefined') {
|
||||||
|
hidden = 'webkitHidden';
|
||||||
|
visibilityChange = 'webkitvisibilitychange';
|
||||||
|
}
|
||||||
|
|
||||||
|
document.addEventListener(visibilityChange, function () {
|
||||||
|
/* eslint-disable-next-line compat/compat */
|
||||||
|
if (document[hidden]) {
|
||||||
|
onAppHidden();
|
||||||
|
} else {
|
||||||
|
onAppVisible();
|
||||||
|
}
|
||||||
|
}, false);
|
||||||
|
|
||||||
|
if (self.addEventListener) {
|
||||||
|
self.addEventListener('focus', onAppVisible);
|
||||||
|
self.addEventListener('blur', onAppHidden);
|
||||||
|
}
|
||||||
|
|
||||||
|
export default appHost;
|
||||||
|
|
|
@ -362,12 +362,12 @@ import 'programStyles';
|
||||||
let hasOpenRow;
|
let hasOpenRow;
|
||||||
let hasOpenSection;
|
let hasOpenSection;
|
||||||
|
|
||||||
let sectionTitleTagName = options.sectionTitleTagName || 'div';
|
const sectionTitleTagName = options.sectionTitleTagName || 'div';
|
||||||
let apiClient;
|
let apiClient;
|
||||||
let lastServerId;
|
let lastServerId;
|
||||||
|
|
||||||
for (const [i, item] of items.entries()) {
|
for (const [i, item] of items.entries()) {
|
||||||
let serverId = item.ServerId || options.serverId;
|
const serverId = item.ServerId || options.serverId;
|
||||||
|
|
||||||
if (serverId !== lastServerId) {
|
if (serverId !== lastServerId) {
|
||||||
lastServerId = serverId;
|
lastServerId = serverId;
|
||||||
|
@ -621,7 +621,7 @@ import 'programStyles';
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
let blurHashes = options.imageBlurhashes || item.ImageBlurHashes || {};
|
const blurHashes = options.imageBlurhashes || item.ImageBlurHashes || {};
|
||||||
|
|
||||||
return {
|
return {
|
||||||
imgUrl: imgUrl,
|
imgUrl: imgUrl,
|
||||||
|
@ -656,7 +656,7 @@ import 'programStyles';
|
||||||
for (let i = 0; i < character.length; i++) {
|
for (let i = 0; i < character.length; i++) {
|
||||||
sum += parseInt(character.charAt(i));
|
sum += parseInt(character.charAt(i));
|
||||||
}
|
}
|
||||||
let index = String(sum).substr(-1);
|
const index = String(sum).substr(-1);
|
||||||
|
|
||||||
return (index % numRandomColors) + 1;
|
return (index % numRandomColors) + 1;
|
||||||
} else {
|
} else {
|
||||||
|
@ -682,7 +682,7 @@ import 'programStyles';
|
||||||
|
|
||||||
for (let i = 0; i < lines.length; i++) {
|
for (let i = 0; i < lines.length; i++) {
|
||||||
let currentCssClass = cssClass;
|
let currentCssClass = cssClass;
|
||||||
let text = lines[i];
|
const text = lines[i];
|
||||||
|
|
||||||
if (valid > 0 && isOuterFooter) {
|
if (valid > 0 && isOuterFooter) {
|
||||||
currentCssClass += ' cardText-secondary';
|
currentCssClass += ' cardText-secondary';
|
||||||
|
@ -707,7 +707,7 @@ import 'programStyles';
|
||||||
}
|
}
|
||||||
|
|
||||||
if (forceLines) {
|
if (forceLines) {
|
||||||
let linesLength = maxLines || Math.min(lines.length, maxLines || lines.length);
|
const linesLength = maxLines || Math.min(lines.length, maxLines || lines.length);
|
||||||
|
|
||||||
while (valid < linesLength) {
|
while (valid < linesLength) {
|
||||||
html += "<div class='" + cssClass + "'> </div>";
|
html += "<div class='" + cssClass + "'> </div>";
|
||||||
|
@ -1036,7 +1036,7 @@ import 'programStyles';
|
||||||
* @returns {string} HTML markup for the item count indicator.
|
* @returns {string} HTML markup for the item count indicator.
|
||||||
*/
|
*/
|
||||||
function getItemCountsHtml(options, item) {
|
function getItemCountsHtml(options, item) {
|
||||||
let counts = [];
|
const counts = [];
|
||||||
let childText;
|
let childText;
|
||||||
|
|
||||||
if (item.Type === 'Playlist') {
|
if (item.Type === 'Playlist') {
|
||||||
|
@ -1318,7 +1318,7 @@ import 'programStyles';
|
||||||
let cardBoxClose = '';
|
let cardBoxClose = '';
|
||||||
let cardScalableClose = '';
|
let cardScalableClose = '';
|
||||||
|
|
||||||
let cardContentClass = 'cardContent';
|
const cardContentClass = 'cardContent';
|
||||||
|
|
||||||
let blurhashAttrib = '';
|
let blurhashAttrib = '';
|
||||||
if (blurhash && blurhash.length > 0) {
|
if (blurhash && blurhash.length > 0) {
|
||||||
|
@ -1337,7 +1337,7 @@ import 'programStyles';
|
||||||
cardImageContainerClose = '</button>';
|
cardImageContainerClose = '</button>';
|
||||||
}
|
}
|
||||||
|
|
||||||
let cardScalableClass = 'cardScalable';
|
const cardScalableClass = 'cardScalable';
|
||||||
|
|
||||||
cardImageContainerOpen = '<div class="' + cardBoxClass + '"><div class="' + cardScalableClass + '"><div class="cardPadder cardPadder-' + shape + '"></div>' + cardImageContainerOpen;
|
cardImageContainerOpen = '<div class="' + cardBoxClass + '"><div class="' + cardScalableClass + '"><div class="cardPadder cardPadder-' + shape + '"></div>' + cardImageContainerOpen;
|
||||||
cardBoxClose = '</div>';
|
cardBoxClose = '</div>';
|
||||||
|
@ -1681,7 +1681,7 @@ import 'programStyles';
|
||||||
const cells = itemsContainer.querySelectorAll('.card[data-id="' + programId + '"]');
|
const cells = itemsContainer.querySelectorAll('.card[data-id="' + programId + '"]');
|
||||||
|
|
||||||
for (let i = 0, length = cells.length; i < length; i++) {
|
for (let i = 0, length = cells.length; i < length; i++) {
|
||||||
let cell = cells[i];
|
const cell = cells[i];
|
||||||
const icon = cell.querySelector('.timerIndicator');
|
const icon = cell.querySelector('.timerIndicator');
|
||||||
if (!icon) {
|
if (!icon) {
|
||||||
const indicatorsElem = ensureIndicators(cell);
|
const indicatorsElem = ensureIndicators(cell);
|
||||||
|
@ -1700,8 +1700,8 @@ import 'programStyles';
|
||||||
const cells = itemsContainer.querySelectorAll('.card[data-timerid="' + timerId + '"]');
|
const cells = itemsContainer.querySelectorAll('.card[data-timerid="' + timerId + '"]');
|
||||||
|
|
||||||
for (let i = 0; i < cells.length; i++) {
|
for (let i = 0; i < cells.length; i++) {
|
||||||
let cell = cells[i];
|
const cell = cells[i];
|
||||||
let icon = cell.querySelector('.timerIndicator');
|
const icon = cell.querySelector('.timerIndicator');
|
||||||
if (icon) {
|
if (icon) {
|
||||||
icon.parentNode.removeChild(icon);
|
icon.parentNode.removeChild(icon);
|
||||||
}
|
}
|
||||||
|
@ -1718,8 +1718,8 @@ import 'programStyles';
|
||||||
const cells = itemsContainer.querySelectorAll('.card[data-seriestimerid="' + cancelledTimerId + '"]');
|
const cells = itemsContainer.querySelectorAll('.card[data-seriestimerid="' + cancelledTimerId + '"]');
|
||||||
|
|
||||||
for (let i = 0; i < cells.length; i++) {
|
for (let i = 0; i < cells.length; i++) {
|
||||||
let cell = cells[i];
|
const cell = cells[i];
|
||||||
let icon = cell.querySelector('.timerIndicator');
|
const icon = cell.querySelector('.timerIndicator');
|
||||||
if (icon) {
|
if (icon) {
|
||||||
icon.parentNode.removeChild(icon);
|
icon.parentNode.removeChild(icon);
|
||||||
}
|
}
|
||||||
|
|
|
@ -391,11 +391,8 @@ import 'scrollStyles';
|
||||||
dlg.setAttribute('data-autofocus', 'true');
|
dlg.setAttribute('data-autofocus', 'true');
|
||||||
}
|
}
|
||||||
|
|
||||||
let defaultEntryAnimation;
|
const defaultEntryAnimation = 'scaleup';
|
||||||
let defaultExitAnimation;
|
const defaultExitAnimation = 'scaledown';
|
||||||
|
|
||||||
defaultEntryAnimation = 'scaleup';
|
|
||||||
defaultExitAnimation = 'scaledown';
|
|
||||||
const entryAnimation = options.entryAnimation || defaultEntryAnimation;
|
const entryAnimation = options.entryAnimation || defaultEntryAnimation;
|
||||||
const exitAnimation = options.exitAnimation || defaultExitAnimation;
|
const exitAnimation = options.exitAnimation || defaultExitAnimation;
|
||||||
|
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
<div style="margin: 0;padding:1.5em 2em;" class="filterDialogContent">
|
<div style="margin: 0;padding:1.5em 2em;" class="filterDialogContent">
|
||||||
|
|
||||||
<div is="emby-collapse" title="${HeaderFilters}">
|
<div is="emby-collapse" title="${Filters}">
|
||||||
<div class="collapseContent">
|
<div class="collapseContent">
|
||||||
<div class="checkboxList">
|
<div class="checkboxList">
|
||||||
<label>
|
<label>
|
||||||
|
@ -63,7 +63,7 @@
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div is="emby-collapse" title="${HeaderFeatures}" class="features hide">
|
<div is="emby-collapse" title="${Features}" class="features hide">
|
||||||
<div class="collapseContent">
|
<div class="collapseContent">
|
||||||
<div class="checkboxList">
|
<div class="checkboxList">
|
||||||
<label>
|
<label>
|
||||||
|
@ -100,7 +100,7 @@
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div is="emby-collapse" title="${HeaderTags}" class="tagFilters hide">
|
<div is="emby-collapse" title="${Tags}" class="tagFilters hide">
|
||||||
<div class="collapseContent filterOptions">
|
<div class="collapseContent filterOptions">
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -1,225 +1,220 @@
|
||||||
define(['require', 'dom', 'focusManager', 'dialogHelper', 'loading', 'apphost', 'inputManager', 'layoutManager', 'connectionManager', 'appRouter', 'globalize', 'userSettings', 'emby-checkbox', 'emby-input', 'paper-icon-button-light', 'emby-select', 'material-icons', 'css!./../formdialog', 'emby-button', 'flexStyles'], function (require, dom, focusManager, dialogHelper, loading, appHost, inputManager, layoutManager, connectionManager, appRouter, globalize, userSettings) {
|
import dom from 'dom';
|
||||||
'use strict';
|
import focusManager from 'focusManager';
|
||||||
focusManager = focusManager.default || focusManager;
|
import dialogHelper from 'dialogHelper';
|
||||||
|
import inputManager from 'inputManager';
|
||||||
|
import layoutManager from 'layoutManager';
|
||||||
|
import connectionManager from 'connectionManager';
|
||||||
|
import globalize from 'globalize';
|
||||||
|
import * as userSettings from 'userSettings';
|
||||||
|
import 'emby-checkbox';
|
||||||
|
import 'emby-input';
|
||||||
|
import 'paper-icon-button-light';
|
||||||
|
import 'emby-select';
|
||||||
|
import 'material-icons';
|
||||||
|
import 'css!./../formdialog';
|
||||||
|
import 'emby-button';
|
||||||
|
import 'flexStyles';
|
||||||
|
|
||||||
layoutManager = layoutManager.default || layoutManager;
|
function onSubmit(e) {
|
||||||
|
e.preventDefault();
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
function renderOptions(context, selector, cssClass, items, isCheckedFn) {
|
||||||
|
var elem = context.querySelector(selector);
|
||||||
|
|
||||||
function onSubmit(e) {
|
if (items.length) {
|
||||||
e.preventDefault();
|
elem.classList.remove('hide');
|
||||||
return false;
|
} else {
|
||||||
|
elem.classList.add('hide');
|
||||||
}
|
}
|
||||||
|
|
||||||
function renderOptions(context, selector, cssClass, items, isCheckedFn) {
|
var html = '';
|
||||||
var elem = context.querySelector(selector);
|
|
||||||
|
|
||||||
if (items.length) {
|
html += items.map(function (filter) {
|
||||||
elem.classList.remove('hide');
|
var itemHtml = '';
|
||||||
|
|
||||||
|
var checkedHtml = isCheckedFn(filter) ? ' checked' : '';
|
||||||
|
itemHtml += '<label>';
|
||||||
|
itemHtml += '<input is="emby-checkbox" type="checkbox"' + checkedHtml + ' data-filter="' + filter.Id + '" class="' + cssClass + '"/>';
|
||||||
|
itemHtml += '<span>' + filter.Name + '</span>';
|
||||||
|
itemHtml += '</label>';
|
||||||
|
|
||||||
|
return itemHtml;
|
||||||
|
}).join('');
|
||||||
|
|
||||||
|
elem.querySelector('.filterOptions').innerHTML = html;
|
||||||
|
}
|
||||||
|
|
||||||
|
function renderDynamicFilters(context, result, options) {
|
||||||
|
renderOptions(context, '.genreFilters', 'chkGenreFilter', result.Genres, function (i) {
|
||||||
|
// Switching from | to ,
|
||||||
|
var delimeter = (options.settings.GenreIds || '').indexOf('|') === -1 ? ',' : '|';
|
||||||
|
return (delimeter + (options.settings.GenreIds || '') + delimeter).indexOf(delimeter + i.Id + delimeter) !== -1;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function setBasicFilter(context, key, elem) {
|
||||||
|
var value = elem.checked;
|
||||||
|
value = value ? value : null;
|
||||||
|
userSettings.setFilter(key, value);
|
||||||
|
}
|
||||||
|
function moveCheckboxFocus(elem, offset) {
|
||||||
|
var parent = dom.parentWithClass(elem, 'checkboxList-verticalwrap');
|
||||||
|
var elems = focusManager.getFocusableElements(parent);
|
||||||
|
|
||||||
|
var index = -1;
|
||||||
|
for (let i = 0, length = elems.length; i < length; i++) {
|
||||||
|
if (elems[i] === elem) {
|
||||||
|
index = i;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
index += offset;
|
||||||
|
|
||||||
|
index = Math.min(elems.length - 1, index);
|
||||||
|
index = Math.max(0, index);
|
||||||
|
|
||||||
|
var newElem = elems[index];
|
||||||
|
if (newElem) {
|
||||||
|
focusManager.focus(newElem);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
function centerFocus(elem, horiz, on) {
|
||||||
|
import('scrollHelper').then(({ default: scrollHelper }) => {
|
||||||
|
var fn = on ? 'on' : 'off';
|
||||||
|
scrollHelper.centerFocus[fn](elem, horiz);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
function onInputCommand(e) {
|
||||||
|
switch (e.detail.command) {
|
||||||
|
case 'left':
|
||||||
|
moveCheckboxFocus(e.target, -1);
|
||||||
|
e.preventDefault();
|
||||||
|
break;
|
||||||
|
case 'right':
|
||||||
|
moveCheckboxFocus(e.target, 1);
|
||||||
|
e.preventDefault();
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
function saveValues(context, settings, settingsKey) {
|
||||||
|
var elems = context.querySelectorAll('.simpleFilter');
|
||||||
|
for (let i = 0, length = elems.length; i < length; i++) {
|
||||||
|
if (elems[i].tagName === 'INPUT') {
|
||||||
|
setBasicFilter(context, settingsKey + '-filter-' + elems[i].getAttribute('data-settingname'), elems[i]);
|
||||||
} else {
|
} else {
|
||||||
elem.classList.add('hide');
|
setBasicFilter(context, settingsKey + '-filter-' + elems[i].getAttribute('data-settingname'), elems[i].querySelector('input'));
|
||||||
}
|
}
|
||||||
|
|
||||||
var html = '';
|
|
||||||
|
|
||||||
html += items.map(function (filter) {
|
|
||||||
var itemHtml = '';
|
|
||||||
|
|
||||||
var checkedHtml = isCheckedFn(filter) ? ' checked' : '';
|
|
||||||
itemHtml += '<label>';
|
|
||||||
itemHtml += '<input is="emby-checkbox" type="checkbox"' + checkedHtml + ' data-filter="' + filter.Id + '" class="' + cssClass + '"/>';
|
|
||||||
itemHtml += '<span>' + filter.Name + '</span>';
|
|
||||||
itemHtml += '</label>';
|
|
||||||
|
|
||||||
return itemHtml;
|
|
||||||
}).join('');
|
|
||||||
|
|
||||||
elem.querySelector('.filterOptions').innerHTML = html;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function renderDynamicFilters(context, result, options) {
|
// Video type
|
||||||
renderOptions(context, '.genreFilters', 'chkGenreFilter', result.Genres, function (i) {
|
var videoTypes = [];
|
||||||
// Switching from | to ,
|
elems = context.querySelectorAll('.chkVideoTypeFilter');
|
||||||
var delimeter = (options.settings.GenreIds || '').indexOf('|') === -1 ? ',' : '|';
|
|
||||||
return (delimeter + (options.settings.GenreIds || '') + delimeter).indexOf(delimeter + i.Id + delimeter) !== -1;
|
for (let i = 0, length = elems.length; i < length; i++) {
|
||||||
});
|
if (elems[i].checked) {
|
||||||
|
videoTypes.push(elems[i].getAttribute('data-filter'));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
userSettings.setFilter(settingsKey + '-filter-VideoTypes', videoTypes.join(','));
|
||||||
|
|
||||||
|
// Series status
|
||||||
|
var seriesStatuses = [];
|
||||||
|
elems = context.querySelectorAll('.chkSeriesStatus');
|
||||||
|
|
||||||
|
for (let i = 0, length = elems.length; i < length; i++) {
|
||||||
|
if (elems[i].checked) {
|
||||||
|
seriesStatuses.push(elems[i].getAttribute('data-filter'));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function loadDynamicFilters(context, options) {
|
// Genres
|
||||||
var apiClient = connectionManager.getApiClient(options.serverId);
|
var genres = [];
|
||||||
|
elems = context.querySelectorAll('.chkGenreFilter');
|
||||||
|
|
||||||
var filterMenuOptions = Object.assign(options.filterMenuOptions, {
|
for (let i = 0, length = elems.length; i < length; i++) {
|
||||||
|
if (elems[i].checked) {
|
||||||
UserId: apiClient.getCurrentUserId(),
|
genres.push(elems[i].getAttribute('data-filter'));
|
||||||
ParentId: options.parentId,
|
}
|
||||||
IncludeItemTypes: options.itemTypes.join(',')
|
|
||||||
});
|
|
||||||
|
|
||||||
apiClient.getFilters(filterMenuOptions).then(function (result) {
|
|
||||||
renderDynamicFilters(context, result, options);
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
userSettings.setFilter(settingsKey + '-filter-GenreIds', genres.join(','));
|
||||||
function initEditor(context, settings) {
|
}
|
||||||
context.querySelector('form').addEventListener('submit', onSubmit);
|
function bindCheckboxInput(context, on) {
|
||||||
|
var elems = context.querySelectorAll('.checkboxList-verticalwrap');
|
||||||
var elems = context.querySelectorAll('.simpleFilter');
|
for (let i = 0, length = elems.length; i < length; i++) {
|
||||||
var i;
|
if (on) {
|
||||||
var length;
|
inputManager.on(elems[i], onInputCommand);
|
||||||
|
|
||||||
for (i = 0, length = elems.length; i < length; i++) {
|
|
||||||
if (elems[i].tagName === 'INPUT') {
|
|
||||||
elems[i].checked = settings[elems[i].getAttribute('data-settingname')] || false;
|
|
||||||
} else {
|
|
||||||
elems[i].querySelector('input').checked = settings[elems[i].getAttribute('data-settingname')] || false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
var videoTypes = settings.VideoTypes ? settings.VideoTypes.split(',') : [];
|
|
||||||
elems = context.querySelectorAll('.chkVideoTypeFilter');
|
|
||||||
|
|
||||||
for (i = 0, length = elems.length; i < length; i++) {
|
|
||||||
elems[i].checked = videoTypes.indexOf(elems[i].getAttribute('data-filter')) !== -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
var seriesStatuses = settings.SeriesStatus ? settings.SeriesStatus.split(',') : [];
|
|
||||||
elems = context.querySelectorAll('.chkSeriesStatus');
|
|
||||||
|
|
||||||
for (i = 0, length = elems.length; i < length; i++) {
|
|
||||||
elems[i].checked = seriesStatuses.indexOf(elems[i].getAttribute('data-filter')) !== -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (context.querySelector('.basicFilterSection .viewSetting:not(.hide)')) {
|
|
||||||
context.querySelector('.basicFilterSection').classList.remove('hide');
|
|
||||||
} else {
|
} else {
|
||||||
context.querySelector('.basicFilterSection').classList.add('hide');
|
inputManager.off(elems[i], onInputCommand);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
function initEditor(context, settings) {
|
||||||
|
context.querySelector('form').addEventListener('submit', onSubmit);
|
||||||
|
|
||||||
if (context.querySelector('.featureSection .viewSetting:not(.hide)')) {
|
var elems = context.querySelectorAll('.simpleFilter');
|
||||||
context.querySelector('.featureSection').classList.remove('hide');
|
var i;
|
||||||
|
var length;
|
||||||
|
|
||||||
|
for (i = 0, length = elems.length; i < length; i++) {
|
||||||
|
if (elems[i].tagName === 'INPUT') {
|
||||||
|
elems[i].checked = settings[elems[i].getAttribute('data-settingname')] || false;
|
||||||
} else {
|
} else {
|
||||||
context.querySelector('.featureSection').classList.add('hide');
|
elems[i].querySelector('input').checked = settings[elems[i].getAttribute('data-settingname')] || false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function saveValues(context, settings, settingsKey) {
|
var videoTypes = settings.VideoTypes ? settings.VideoTypes.split(',') : [];
|
||||||
var elems = context.querySelectorAll('.simpleFilter');
|
elems = context.querySelectorAll('.chkVideoTypeFilter');
|
||||||
var i;
|
|
||||||
var length;
|
|
||||||
for (i = 0, length = elems.length; i < length; i++) {
|
|
||||||
if (elems[i].tagName === 'INPUT') {
|
|
||||||
setBasicFilter(context, settingsKey + '-filter-' + elems[i].getAttribute('data-settingname'), elems[i]);
|
|
||||||
} else {
|
|
||||||
setBasicFilter(context, settingsKey + '-filter-' + elems[i].getAttribute('data-settingname'), elems[i].querySelector('input'));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Video type
|
for (i = 0, length = elems.length; i < length; i++) {
|
||||||
var videoTypes = [];
|
elems[i].checked = videoTypes.indexOf(elems[i].getAttribute('data-filter')) !== -1;
|
||||||
elems = context.querySelectorAll('.chkVideoTypeFilter');
|
|
||||||
|
|
||||||
for (i = 0, length = elems.length; i < length; i++) {
|
|
||||||
if (elems[i].checked) {
|
|
||||||
videoTypes.push(elems[i].getAttribute('data-filter'));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
userSettings.setFilter(settingsKey + '-filter-VideoTypes', videoTypes.join(','));
|
|
||||||
|
|
||||||
// Series status
|
|
||||||
var seriesStatuses = [];
|
|
||||||
elems = context.querySelectorAll('.chkSeriesStatus');
|
|
||||||
|
|
||||||
for (i = 0, length = elems.length; i < length; i++) {
|
|
||||||
if (elems[i].checked) {
|
|
||||||
seriesStatuses.push(elems[i].getAttribute('data-filter'));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Genres
|
|
||||||
var genres = [];
|
|
||||||
elems = context.querySelectorAll('.chkGenreFilter');
|
|
||||||
|
|
||||||
for (i = 0, length = elems.length; i < length; i++) {
|
|
||||||
if (elems[i].checked) {
|
|
||||||
genres.push(elems[i].getAttribute('data-filter'));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
userSettings.setFilter(settingsKey + '-filter-GenreIds', genres.join(','));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function setBasicFilter(context, key, elem) {
|
var seriesStatuses = settings.SeriesStatus ? settings.SeriesStatus.split(',') : [];
|
||||||
var value = elem.checked;
|
elems = context.querySelectorAll('.chkSeriesStatus');
|
||||||
value = value ? value : null;
|
|
||||||
userSettings.setFilter(key, value);
|
for (i = 0, length = elems.length; i < length; i++) {
|
||||||
|
elems[i].checked = seriesStatuses.indexOf(elems[i].getAttribute('data-filter')) !== -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
function centerFocus(elem, horiz, on) {
|
if (context.querySelector('.basicFilterSection .viewSetting:not(.hide)')) {
|
||||||
require(['scrollHelper'], function (scrollHelper) {
|
context.querySelector('.basicFilterSection').classList.remove('hide');
|
||||||
scrollHelper = scrollHelper.default || scrollHelper;
|
} else {
|
||||||
var fn = on ? 'on' : 'off';
|
context.querySelector('.basicFilterSection').classList.add('hide');
|
||||||
scrollHelper.centerFocus[fn](elem, horiz);
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function moveCheckboxFocus(elem, offset) {
|
if (context.querySelector('.featureSection .viewSetting:not(.hide)')) {
|
||||||
var parent = dom.parentWithClass(elem, 'checkboxList-verticalwrap');
|
context.querySelector('.featureSection').classList.remove('hide');
|
||||||
var elems = focusManager.getFocusableElements(parent);
|
} else {
|
||||||
|
context.querySelector('.featureSection').classList.add('hide');
|
||||||
var index = -1;
|
|
||||||
for (var i = 0, length = elems.length; i < length; i++) {
|
|
||||||
if (elems[i] === elem) {
|
|
||||||
index = i;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
index += offset;
|
|
||||||
|
|
||||||
index = Math.min(elems.length - 1, index);
|
|
||||||
index = Math.max(0, index);
|
|
||||||
|
|
||||||
var newElem = elems[index];
|
|
||||||
if (newElem) {
|
|
||||||
focusManager.focus(newElem);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
function loadDynamicFilters(context, options) {
|
||||||
|
var apiClient = connectionManager.getApiClient(options.serverId);
|
||||||
|
|
||||||
function onInputCommand(e) {
|
var filterMenuOptions = Object.assign(options.filterMenuOptions, {
|
||||||
switch (e.detail.command) {
|
|
||||||
case 'left':
|
|
||||||
moveCheckboxFocus(e.target, -1);
|
|
||||||
e.preventDefault();
|
|
||||||
break;
|
|
||||||
case 'right':
|
|
||||||
moveCheckboxFocus(e.target, 1);
|
|
||||||
e.preventDefault();
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function FilterMenu() {
|
UserId: apiClient.getCurrentUserId(),
|
||||||
|
ParentId: options.parentId,
|
||||||
|
IncludeItemTypes: options.itemTypes.join(',')
|
||||||
|
});
|
||||||
|
|
||||||
}
|
apiClient.getFilters(filterMenuOptions).then((result) => {
|
||||||
|
renderDynamicFilters(context, result, options);
|
||||||
function bindCheckboxInput(context, on) {
|
});
|
||||||
var elems = context.querySelectorAll('.checkboxList-verticalwrap');
|
}
|
||||||
for (var i = 0, length = elems.length; i < length; i++) {
|
class FilterMenu {
|
||||||
if (on) {
|
show(options) {
|
||||||
inputManager.on(elems[i], onInputCommand);
|
return new Promise( (resolve, reject) => {
|
||||||
} else {
|
import('text!./filtermenu.template.html').then(({ default: template }) => {
|
||||||
inputManager.off(elems[i], onInputCommand);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
FilterMenu.prototype.show = function (options) {
|
|
||||||
return new Promise(function (resolve, reject) {
|
|
||||||
require(['text!./filtermenu.template.html'], function (template) {
|
|
||||||
var dialogOptions = {
|
var dialogOptions = {
|
||||||
removeOnClose: true,
|
removeOnClose: true,
|
||||||
scrollY: false
|
scrollY: false
|
||||||
};
|
};
|
||||||
|
|
||||||
if (layoutManager.tv) {
|
if (layoutManager.tv) {
|
||||||
dialogOptions.size = 'fullscreen';
|
dialogOptions.size = 'fullscreen';
|
||||||
} else {
|
} else {
|
||||||
|
@ -243,7 +238,7 @@ define(['require', 'dom', 'focusManager', 'dialogHelper', 'loading', 'apphost',
|
||||||
dlg.innerHTML = globalize.translateHtml(html, 'core');
|
dlg.innerHTML = globalize.translateHtml(html, 'core');
|
||||||
|
|
||||||
var settingElements = dlg.querySelectorAll('.viewSetting');
|
var settingElements = dlg.querySelectorAll('.viewSetting');
|
||||||
for (var i = 0, length = settingElements.length; i < length; i++) {
|
for (let i = 0, length = settingElements.length; i < length; i++) {
|
||||||
if (options.visibleSettings.indexOf(settingElements[i].getAttribute('data-settingname')) === -1) {
|
if (options.visibleSettings.indexOf(settingElements[i].getAttribute('data-settingname')) === -1) {
|
||||||
settingElements[i].classList.add('hide');
|
settingElements[i].classList.add('hide');
|
||||||
} else {
|
} else {
|
||||||
|
@ -255,7 +250,6 @@ define(['require', 'dom', 'focusManager', 'dialogHelper', 'loading', 'apphost',
|
||||||
loadDynamicFilters(dlg, options);
|
loadDynamicFilters(dlg, options);
|
||||||
|
|
||||||
bindCheckboxInput(dlg, true);
|
bindCheckboxInput(dlg, true);
|
||||||
|
|
||||||
dlg.querySelector('.btnCancel').addEventListener('click', function () {
|
dlg.querySelector('.btnCancel').addEventListener('click', function () {
|
||||||
dialogHelper.close(dlg);
|
dialogHelper.close(dlg);
|
||||||
});
|
});
|
||||||
|
@ -270,7 +264,7 @@ define(['require', 'dom', 'focusManager', 'dialogHelper', 'loading', 'apphost',
|
||||||
submitted = true;
|
submitted = true;
|
||||||
}, true);
|
}, true);
|
||||||
|
|
||||||
dialogHelper.open(dlg).then(function () {
|
dialogHelper.open(dlg).then( function() {
|
||||||
bindCheckboxInput(dlg, false);
|
bindCheckboxInput(dlg, false);
|
||||||
|
|
||||||
if (layoutManager.tv) {
|
if (layoutManager.tv) {
|
||||||
|
@ -280,16 +274,14 @@ define(['require', 'dom', 'focusManager', 'dialogHelper', 'loading', 'apphost',
|
||||||
if (submitted) {
|
if (submitted) {
|
||||||
//if (!options.onChange) {
|
//if (!options.onChange) {
|
||||||
saveValues(dlg, options.settings, options.settingsKey);
|
saveValues(dlg, options.settings, options.settingsKey);
|
||||||
resolve();
|
return resolve();
|
||||||
//}
|
//}
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
|
return resolve();
|
||||||
reject();
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
};
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return FilterMenu;
|
export default FilterMenu;
|
||||||
});
|
|
||||||
|
|
|
@ -17,8 +17,8 @@ import 'css!./style';
|
||||||
// Although the default values recommended by Blurhash developers is 32x32, a size of 18x18 seems to be the sweet spot for us,
|
// Although the default values recommended by Blurhash developers is 32x32, a size of 18x18 seems to be the sweet spot for us,
|
||||||
// improving the performance and reducing the memory usage, while retaining almost full blur quality.
|
// improving the performance and reducing the memory usage, while retaining almost full blur quality.
|
||||||
// Lower values had more visible pixelation
|
// Lower values had more visible pixelation
|
||||||
let width = 18;
|
const width = 18;
|
||||||
let height = 18;
|
const height = 18;
|
||||||
let pixels;
|
let pixels;
|
||||||
try {
|
try {
|
||||||
pixels = blurhash.decode(blurhashstr, width, height);
|
pixels = blurhash.decode(blurhashstr, width, height);
|
||||||
|
@ -27,11 +27,11 @@ import 'css!./style';
|
||||||
target.classList.add('non-blurhashable');
|
target.classList.add('non-blurhashable');
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
let canvas = document.createElement('canvas');
|
const canvas = document.createElement('canvas');
|
||||||
canvas.width = width;
|
canvas.width = width;
|
||||||
canvas.height = height;
|
canvas.height = height;
|
||||||
let ctx = canvas.getContext('2d');
|
const ctx = canvas.getContext('2d');
|
||||||
let imgData = ctx.createImageData(width, height);
|
const imgData = ctx.createImageData(width, height);
|
||||||
|
|
||||||
imgData.data.set(pixels);
|
imgData.data.set(pixels);
|
||||||
ctx.putImageData(imgData, 0, 0);
|
ctx.putImageData(imgData, 0, 0);
|
||||||
|
@ -55,7 +55,7 @@ import 'css!./style';
|
||||||
if (!entry) {
|
if (!entry) {
|
||||||
throw new Error('entry cannot be null');
|
throw new Error('entry cannot be null');
|
||||||
}
|
}
|
||||||
let target = entry.target;
|
const target = entry.target;
|
||||||
var source = undefined;
|
var source = undefined;
|
||||||
|
|
||||||
if (target) {
|
if (target) {
|
||||||
|
@ -78,7 +78,7 @@ import 'css!./style';
|
||||||
throw new TypeError('url cannot be undefined');
|
throw new TypeError('url cannot be undefined');
|
||||||
}
|
}
|
||||||
|
|
||||||
let preloaderImg = new Image();
|
const preloaderImg = new Image();
|
||||||
preloaderImg.src = url;
|
preloaderImg.src = url;
|
||||||
|
|
||||||
elem.classList.add('lazy-hidden');
|
elem.classList.add('lazy-hidden');
|
||||||
|
|
|
@ -82,7 +82,7 @@ export function enablePlayedIndicator(item) {
|
||||||
|
|
||||||
export function getPlayedIndicatorHtml(item) {
|
export function getPlayedIndicatorHtml(item) {
|
||||||
if (enablePlayedIndicator(item)) {
|
if (enablePlayedIndicator(item)) {
|
||||||
let userData = item.UserData || {};
|
const userData = item.UserData || {};
|
||||||
if (userData.UnplayedItemCount) {
|
if (userData.UnplayedItemCount) {
|
||||||
return '<div class="countIndicator indicator">' + userData.UnplayedItemCount + '</div>';
|
return '<div class="countIndicator indicator">' + userData.UnplayedItemCount + '</div>';
|
||||||
}
|
}
|
||||||
|
|
|
@ -16,7 +16,7 @@ import actionsheet from 'actionsheet';
|
||||||
const canPlay = playbackManager.canPlay(item);
|
const canPlay = playbackManager.canPlay(item);
|
||||||
const restrictOptions = (browser.operaTv || browser.web0s) && !user.Policy.IsAdministrator;
|
const restrictOptions = (browser.operaTv || browser.web0s) && !user.Policy.IsAdministrator;
|
||||||
|
|
||||||
let commands = [];
|
const commands = [];
|
||||||
|
|
||||||
if (canPlay && item.MediaType !== 'Photo') {
|
if (canPlay && item.MediaType !== 'Photo') {
|
||||||
if (options.play !== false) {
|
if (options.play !== false) {
|
||||||
|
@ -367,7 +367,7 @@ import actionsheet from 'actionsheet';
|
||||||
case 'copy-stream': {
|
case 'copy-stream': {
|
||||||
const downloadHref = apiClient.getItemDownloadUrl(itemId);
|
const downloadHref = apiClient.getItemDownloadUrl(itemId);
|
||||||
const textAreaCopy = function () {
|
const textAreaCopy = function () {
|
||||||
let textArea = document.createElement('textarea');
|
const textArea = document.createElement('textarea');
|
||||||
textArea.value = downloadHref;
|
textArea.value = downloadHref;
|
||||||
document.body.appendChild(textArea);
|
document.body.appendChild(textArea);
|
||||||
textArea.focus();
|
textArea.focus();
|
||||||
|
|
|
@ -25,7 +25,7 @@
|
||||||
<div class="folders">
|
<div class="folders">
|
||||||
<div style="display: flex; align-items: center;">
|
<div style="display: flex; align-items: center;">
|
||||||
<h1 style="margin: .5em 0;">${HeadersFolders}</h1>
|
<h1 style="margin: .5em 0;">${HeadersFolders}</h1>
|
||||||
<button is="emby-button" type="button" class="fab btnAddFolder submit" style="margin-left:1em;" title="${ButtonAdd}">
|
<button is="emby-button" type="button" class="fab btnAddFolder submit" style="margin-left:1em;" title="${Add}">
|
||||||
<span class="material-icons add"></span>
|
<span class="material-icons add"></span>
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -19,7 +19,7 @@
|
||||||
<div class="folders hide">
|
<div class="folders hide">
|
||||||
<div style="display: flex; align-items: center;">
|
<div style="display: flex; align-items: center;">
|
||||||
<h1 style="margin: .5em 0;">${HeadersFolders}</h1>
|
<h1 style="margin: .5em 0;">${HeadersFolders}</h1>
|
||||||
<button is="emby-button" type="button" class="fab btnAddFolder submit" style="margin-left:1em;" title="${ButtonAdd}">
|
<button is="emby-button" type="button" class="fab btnAddFolder submit" style="margin-left:1em;" title="${Add}">
|
||||||
<span class="material-icons add"></span>
|
<span class="material-icons add"></span>
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -906,7 +906,7 @@ import 'flexStyles';
|
||||||
}
|
}
|
||||||
|
|
||||||
function populatePeople(context, people) {
|
function populatePeople(context, people) {
|
||||||
let lastType = '';
|
const lastType = '';
|
||||||
let html = '';
|
let html = '';
|
||||||
|
|
||||||
const elem = context.querySelector('#peopleList');
|
const elem = context.querySelector('#peopleList');
|
||||||
|
|
|
@ -1,182 +1,181 @@
|
||||||
define(['serverNotifications', 'playbackManager', 'events', 'globalize', 'require'], function (serverNotifications, playbackManager, events, globalize, require) {
|
import serverNotifications from 'serverNotifications';
|
||||||
'use strict';
|
import playbackManager from 'playbackManager';
|
||||||
|
import events from 'events';
|
||||||
|
import globalize from 'globalize';
|
||||||
|
|
||||||
playbackManager = playbackManager.default || playbackManager;
|
function onOneDocumentClick() {
|
||||||
serverNotifications = serverNotifications.default || serverNotifications;
|
document.removeEventListener('click', onOneDocumentClick);
|
||||||
|
document.removeEventListener('keydown', onOneDocumentClick);
|
||||||
|
|
||||||
function onOneDocumentClick() {
|
// don't request notification permissions if they're already granted or denied
|
||||||
document.removeEventListener('click', onOneDocumentClick);
|
if (window.Notification && window.Notification.permission === 'default') {
|
||||||
document.removeEventListener('keydown', onOneDocumentClick);
|
|
||||||
|
|
||||||
// don't request notification permissions if they're already granted or denied
|
|
||||||
if (window.Notification && window.Notification.permission === 'default') {
|
|
||||||
/* eslint-disable-next-line compat/compat */
|
|
||||||
Notification.requestPermission();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
document.addEventListener('click', onOneDocumentClick);
|
|
||||||
document.addEventListener('keydown', onOneDocumentClick);
|
|
||||||
|
|
||||||
var serviceWorkerRegistration;
|
|
||||||
|
|
||||||
function closeAfter(notification, timeoutMs) {
|
|
||||||
setTimeout(function () {
|
|
||||||
if (notification.close) {
|
|
||||||
notification.close();
|
|
||||||
} else if (notification.cancel) {
|
|
||||||
notification.cancel();
|
|
||||||
}
|
|
||||||
}, timeoutMs);
|
|
||||||
}
|
|
||||||
|
|
||||||
function resetRegistration() {
|
|
||||||
/* eslint-disable-next-line compat/compat */
|
/* eslint-disable-next-line compat/compat */
|
||||||
var serviceWorker = navigator.serviceWorker;
|
Notification.requestPermission();
|
||||||
if (serviceWorker) {
|
}
|
||||||
serviceWorker.ready.then(function (registration) {
|
}
|
||||||
serviceWorkerRegistration = registration;
|
|
||||||
});
|
document.addEventListener('click', onOneDocumentClick);
|
||||||
|
document.addEventListener('keydown', onOneDocumentClick);
|
||||||
|
|
||||||
|
let serviceWorkerRegistration;
|
||||||
|
|
||||||
|
function closeAfter(notification, timeoutMs) {
|
||||||
|
setTimeout(function () {
|
||||||
|
if (notification.close) {
|
||||||
|
notification.close();
|
||||||
|
} else if (notification.cancel) {
|
||||||
|
notification.cancel();
|
||||||
|
}
|
||||||
|
}, timeoutMs);
|
||||||
|
}
|
||||||
|
|
||||||
|
function resetRegistration() {
|
||||||
|
/* eslint-disable-next-line compat/compat */
|
||||||
|
let serviceWorker = navigator.serviceWorker;
|
||||||
|
if (serviceWorker) {
|
||||||
|
serviceWorker.ready.then(function (registration) {
|
||||||
|
serviceWorkerRegistration = registration;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
resetRegistration();
|
||||||
|
|
||||||
|
function showPersistentNotification(title, options, timeoutMs) {
|
||||||
|
serviceWorkerRegistration.showNotification(title, options);
|
||||||
|
}
|
||||||
|
|
||||||
|
function showNonPersistentNotification(title, options, timeoutMs) {
|
||||||
|
try {
|
||||||
|
let notif = new Notification(title, options); /* eslint-disable-line compat/compat */
|
||||||
|
|
||||||
|
if (notif.show) {
|
||||||
|
notif.show();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (timeoutMs) {
|
||||||
|
closeAfter(notif, timeoutMs);
|
||||||
|
}
|
||||||
|
} catch (err) {
|
||||||
|
if (options.actions) {
|
||||||
|
options.actions = [];
|
||||||
|
showNonPersistentNotification(title, options, timeoutMs);
|
||||||
|
} else {
|
||||||
|
throw err;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function showNotification(options, timeoutMs, apiClient) {
|
||||||
|
let title = options.title;
|
||||||
|
|
||||||
|
options.data = options.data || {};
|
||||||
|
options.data.serverId = apiClient.serverInfo().Id;
|
||||||
|
options.icon = options.icon || getIconUrl();
|
||||||
|
options.badge = options.badge || getIconUrl('badge.png');
|
||||||
|
|
||||||
resetRegistration();
|
resetRegistration();
|
||||||
|
|
||||||
function showPersistentNotification(title, options, timeoutMs) {
|
if (serviceWorkerRegistration) {
|
||||||
serviceWorkerRegistration.showNotification(title, options);
|
showPersistentNotification(title, options, timeoutMs);
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
function showNonPersistentNotification(title, options, timeoutMs) {
|
showNonPersistentNotification(title, options, timeoutMs);
|
||||||
try {
|
}
|
||||||
var notif = new Notification(title, options); /* eslint-disable-line compat/compat */
|
|
||||||
|
|
||||||
if (notif.show) {
|
function showNewItemNotification(item, apiClient) {
|
||||||
notif.show();
|
if (playbackManager.isPlayingLocally(['Video'])) {
|
||||||
}
|
return;
|
||||||
|
|
||||||
if (timeoutMs) {
|
|
||||||
closeAfter(notif, timeoutMs);
|
|
||||||
}
|
|
||||||
} catch (err) {
|
|
||||||
if (options.actions) {
|
|
||||||
options.actions = [];
|
|
||||||
showNonPersistentNotification(title, options, timeoutMs);
|
|
||||||
} else {
|
|
||||||
throw err;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function showNotification(options, timeoutMs, apiClient) {
|
let body = item.Name;
|
||||||
var title = options.title;
|
|
||||||
|
|
||||||
options.data = options.data || {};
|
if (item.SeriesName) {
|
||||||
options.data.serverId = apiClient.serverInfo().Id;
|
body = item.SeriesName + ' - ' + body;
|
||||||
options.icon = options.icon || getIconUrl();
|
|
||||||
options.badge = options.badge || getIconUrl('badge.png');
|
|
||||||
|
|
||||||
resetRegistration();
|
|
||||||
|
|
||||||
if (serviceWorkerRegistration) {
|
|
||||||
showPersistentNotification(title, options, timeoutMs);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
showNonPersistentNotification(title, options, timeoutMs);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function showNewItemNotification(item, apiClient) {
|
let notification = {
|
||||||
if (playbackManager.isPlayingLocally(['Video'])) {
|
title: 'New ' + item.Type,
|
||||||
return;
|
body: body,
|
||||||
}
|
vibrate: true,
|
||||||
|
tag: 'newItem' + item.Id,
|
||||||
|
data: {}
|
||||||
|
};
|
||||||
|
|
||||||
var body = item.Name;
|
let imageTags = item.ImageTags || {};
|
||||||
|
|
||||||
if (item.SeriesName) {
|
if (imageTags.Primary) {
|
||||||
body = item.SeriesName + ' - ' + body;
|
notification.icon = apiClient.getScaledImageUrl(item.Id, {
|
||||||
}
|
width: 80,
|
||||||
|
tag: imageTags.Primary,
|
||||||
var notification = {
|
type: 'Primary'
|
||||||
title: 'New ' + item.Type,
|
|
||||||
body: body,
|
|
||||||
vibrate: true,
|
|
||||||
tag: 'newItem' + item.Id,
|
|
||||||
data: {}
|
|
||||||
};
|
|
||||||
|
|
||||||
var imageTags = item.ImageTags || {};
|
|
||||||
|
|
||||||
if (imageTags.Primary) {
|
|
||||||
notification.icon = apiClient.getScaledImageUrl(item.Id, {
|
|
||||||
width: 80,
|
|
||||||
tag: imageTags.Primary,
|
|
||||||
type: 'Primary'
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
showNotification(notification, 15000, apiClient);
|
|
||||||
}
|
|
||||||
|
|
||||||
function onLibraryChanged(data, apiClient) {
|
|
||||||
var newItems = data.ItemsAdded;
|
|
||||||
|
|
||||||
if (!newItems.length) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Don't put a massive number of Id's onto the query string
|
|
||||||
if (newItems.length > 12) {
|
|
||||||
newItems.length = 12;
|
|
||||||
}
|
|
||||||
|
|
||||||
apiClient.getItems(apiClient.getCurrentUserId(), {
|
|
||||||
|
|
||||||
Recursive: true,
|
|
||||||
Limit: 3,
|
|
||||||
Filters: 'IsNotFolder',
|
|
||||||
SortBy: 'DateCreated',
|
|
||||||
SortOrder: 'Descending',
|
|
||||||
Ids: newItems.join(','),
|
|
||||||
MediaTypes: 'Audio,Video',
|
|
||||||
EnableTotalRecordCount: false
|
|
||||||
|
|
||||||
}).then(function (result) {
|
|
||||||
var items = result.Items;
|
|
||||||
|
|
||||||
for (var i = 0, length = items.length ; i < length; i++) {
|
|
||||||
showNewItemNotification(items[i], apiClient);
|
|
||||||
}
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
function getIconUrl(name) {
|
showNotification(notification, 15000, apiClient);
|
||||||
name = name || 'notificationicon.png';
|
}
|
||||||
return require.toUrl('.').split('?')[0] + '/' + name;
|
|
||||||
|
function onLibraryChanged(data, apiClient) {
|
||||||
|
let newItems = data.ItemsAdded;
|
||||||
|
|
||||||
|
if (!newItems.length) {
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
function showPackageInstallNotification(apiClient, installation, status) {
|
// Don't put a massive number of Id's onto the query string
|
||||||
apiClient.getCurrentUser().then(function (user) {
|
if (newItems.length > 12) {
|
||||||
if (!user.Policy.IsAdministrator) {
|
newItems.length = 12;
|
||||||
return;
|
}
|
||||||
}
|
|
||||||
|
|
||||||
var notification = {
|
apiClient.getItems(apiClient.getCurrentUserId(), {
|
||||||
tag: 'install' + installation.Id,
|
|
||||||
data: {}
|
|
||||||
};
|
|
||||||
|
|
||||||
if (status === 'completed') {
|
Recursive: true,
|
||||||
notification.title = globalize.translate('PackageInstallCompleted', installation.Name, installation.Version);
|
Limit: 3,
|
||||||
notification.vibrate = true;
|
Filters: 'IsNotFolder',
|
||||||
} else if (status === 'cancelled') {
|
SortBy: 'DateCreated',
|
||||||
notification.title = globalize.translate('PackageInstallCancelled', installation.Name, installation.Version);
|
SortOrder: 'Descending',
|
||||||
} else if (status === 'failed') {
|
Ids: newItems.join(','),
|
||||||
notification.title = globalize.translate('PackageInstallFailed', installation.Name, installation.Version);
|
MediaTypes: 'Audio,Video',
|
||||||
notification.vibrate = true;
|
EnableTotalRecordCount: false
|
||||||
} else if (status === 'progress') {
|
|
||||||
notification.title = globalize.translate('InstallingPackage', installation.Name, installation.Version);
|
|
||||||
|
|
||||||
notification.actions =
|
}).then(function (result) {
|
||||||
|
let items = result.Items;
|
||||||
|
|
||||||
|
for (const item of items) {
|
||||||
|
showNewItemNotification(item, apiClient);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function getIconUrl(name) {
|
||||||
|
name = name || 'notificationicon.png';
|
||||||
|
return './components/notifications/' + name;
|
||||||
|
}
|
||||||
|
|
||||||
|
function showPackageInstallNotification(apiClient, installation, status) {
|
||||||
|
apiClient.getCurrentUser().then(function (user) {
|
||||||
|
if (!user.Policy.IsAdministrator) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
let notification = {
|
||||||
|
tag: 'install' + installation.Id,
|
||||||
|
data: {}
|
||||||
|
};
|
||||||
|
|
||||||
|
if (status === 'completed') {
|
||||||
|
notification.title = globalize.translate('PackageInstallCompleted', installation.Name, installation.Version);
|
||||||
|
notification.vibrate = true;
|
||||||
|
} else if (status === 'cancelled') {
|
||||||
|
notification.title = globalize.translate('PackageInstallCancelled', installation.Name, installation.Version);
|
||||||
|
} else if (status === 'failed') {
|
||||||
|
notification.title = globalize.translate('PackageInstallFailed', installation.Name, installation.Version);
|
||||||
|
notification.vibrate = true;
|
||||||
|
} else if (status === 'progress') {
|
||||||
|
notification.title = globalize.translate('InstallingPackage', installation.Name, installation.Version);
|
||||||
|
|
||||||
|
notification.actions =
|
||||||
[
|
[
|
||||||
{
|
{
|
||||||
action: 'cancel-install',
|
action: 'cancel-install',
|
||||||
|
@ -185,67 +184,67 @@ define(['serverNotifications', 'playbackManager', 'events', 'globalize', 'requir
|
||||||
}
|
}
|
||||||
];
|
];
|
||||||
|
|
||||||
notification.data.id = installation.id;
|
notification.data.id = installation.id;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (status === 'progress') {
|
if (status === 'progress') {
|
||||||
var percentComplete = Math.round(installation.PercentComplete || 0);
|
let percentComplete = Math.round(installation.PercentComplete || 0);
|
||||||
|
|
||||||
notification.body = percentComplete + '% complete.';
|
notification.body = percentComplete + '% complete.';
|
||||||
}
|
}
|
||||||
|
|
||||||
var timeout = status === 'cancelled' ? 5000 : 0;
|
let timeout = status === 'cancelled' ? 5000 : 0;
|
||||||
|
|
||||||
showNotification(notification, timeout, apiClient);
|
showNotification(notification, timeout, apiClient);
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
events.on(serverNotifications, 'LibraryChanged', function (e, apiClient, data) {
|
|
||||||
onLibraryChanged(data, apiClient);
|
|
||||||
});
|
});
|
||||||
|
}
|
||||||
|
|
||||||
events.on(serverNotifications, 'PackageInstallationCompleted', function (e, apiClient, data) {
|
events.on(serverNotifications, 'LibraryChanged', function (e, apiClient, data) {
|
||||||
showPackageInstallNotification(apiClient, data, 'completed');
|
onLibraryChanged(data, apiClient);
|
||||||
});
|
});
|
||||||
|
|
||||||
events.on(serverNotifications, 'PackageInstallationFailed', function (e, apiClient, data) {
|
events.on(serverNotifications, 'PackageInstallationCompleted', function (e, apiClient, data) {
|
||||||
showPackageInstallNotification(apiClient, data, 'failed');
|
showPackageInstallNotification(apiClient, data, 'completed');
|
||||||
});
|
});
|
||||||
|
|
||||||
events.on(serverNotifications, 'PackageInstallationCancelled', function (e, apiClient, data) {
|
events.on(serverNotifications, 'PackageInstallationFailed', function (e, apiClient, data) {
|
||||||
showPackageInstallNotification(apiClient, data, 'cancelled');
|
showPackageInstallNotification(apiClient, data, 'failed');
|
||||||
});
|
});
|
||||||
|
|
||||||
events.on(serverNotifications, 'PackageInstalling', function (e, apiClient, data) {
|
events.on(serverNotifications, 'PackageInstallationCancelled', function (e, apiClient, data) {
|
||||||
showPackageInstallNotification(apiClient, data, 'progress');
|
showPackageInstallNotification(apiClient, data, 'cancelled');
|
||||||
});
|
});
|
||||||
|
|
||||||
events.on(serverNotifications, 'ServerShuttingDown', function (e, apiClient, data) {
|
events.on(serverNotifications, 'PackageInstalling', function (e, apiClient, data) {
|
||||||
var serverId = apiClient.serverInfo().Id;
|
showPackageInstallNotification(apiClient, data, 'progress');
|
||||||
var notification = {
|
});
|
||||||
tag: 'restart' + serverId,
|
|
||||||
title: globalize.translate('ServerNameIsShuttingDown', apiClient.serverInfo().Name)
|
|
||||||
};
|
|
||||||
showNotification(notification, 0, apiClient);
|
|
||||||
});
|
|
||||||
|
|
||||||
events.on(serverNotifications, 'ServerRestarting', function (e, apiClient, data) {
|
events.on(serverNotifications, 'ServerShuttingDown', function (e, apiClient, data) {
|
||||||
var serverId = apiClient.serverInfo().Id;
|
let serverId = apiClient.serverInfo().Id;
|
||||||
var notification = {
|
let notification = {
|
||||||
tag: 'restart' + serverId,
|
tag: 'restart' + serverId,
|
||||||
title: globalize.translate('ServerNameIsRestarting', apiClient.serverInfo().Name)
|
title: globalize.translate('ServerNameIsShuttingDown', apiClient.serverInfo().Name)
|
||||||
};
|
};
|
||||||
showNotification(notification, 0, apiClient);
|
showNotification(notification, 0, apiClient);
|
||||||
});
|
});
|
||||||
|
|
||||||
events.on(serverNotifications, 'RestartRequired', function (e, apiClient) {
|
events.on(serverNotifications, 'ServerRestarting', function (e, apiClient, data) {
|
||||||
var serverId = apiClient.serverInfo().Id;
|
let serverId = apiClient.serverInfo().Id;
|
||||||
var notification = {
|
let notification = {
|
||||||
tag: 'restart' + serverId,
|
tag: 'restart' + serverId,
|
||||||
title: globalize.translate('PleaseRestartServerName', apiClient.serverInfo().Name)
|
title: globalize.translate('ServerNameIsRestarting', apiClient.serverInfo().Name)
|
||||||
};
|
};
|
||||||
|
showNotification(notification, 0, apiClient);
|
||||||
|
});
|
||||||
|
|
||||||
notification.actions =
|
events.on(serverNotifications, 'RestartRequired', function (e, apiClient) {
|
||||||
|
let serverId = apiClient.serverInfo().Id;
|
||||||
|
let notification = {
|
||||||
|
tag: 'restart' + serverId,
|
||||||
|
title: globalize.translate('PleaseRestartServerName', apiClient.serverInfo().Name)
|
||||||
|
};
|
||||||
|
|
||||||
|
notification.actions =
|
||||||
[
|
[
|
||||||
{
|
{
|
||||||
action: 'restart',
|
action: 'restart',
|
||||||
|
@ -254,6 +253,6 @@ define(['serverNotifications', 'playbackManager', 'events', 'globalize', 'requir
|
||||||
}
|
}
|
||||||
];
|
];
|
||||||
|
|
||||||
showNotification(notification, 0, apiClient);
|
showNotification(notification, 0, apiClient);
|
||||||
});
|
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
|
@ -500,20 +500,20 @@ import 'emby-ratingbutton';
|
||||||
const textLines = nowPlayingItem ? nowPlayingHelper.getNowPlayingNames(nowPlayingItem) : [];
|
const textLines = nowPlayingItem ? nowPlayingHelper.getNowPlayingNames(nowPlayingItem) : [];
|
||||||
nowPlayingTextElement.innerHTML = '';
|
nowPlayingTextElement.innerHTML = '';
|
||||||
if (textLines) {
|
if (textLines) {
|
||||||
let itemText = document.createElement('div');
|
const itemText = document.createElement('div');
|
||||||
let secondaryText = document.createElement('div');
|
const secondaryText = document.createElement('div');
|
||||||
secondaryText.classList.add('nowPlayingBarSecondaryText');
|
secondaryText.classList.add('nowPlayingBarSecondaryText');
|
||||||
if (textLines.length > 1) {
|
if (textLines.length > 1) {
|
||||||
textLines[1].secondary = true;
|
textLines[1].secondary = true;
|
||||||
if (textLines[1].text) {
|
if (textLines[1].text) {
|
||||||
let text = document.createElement('a');
|
const text = document.createElement('a');
|
||||||
text.innerHTML = textLines[1].text;
|
text.innerHTML = textLines[1].text;
|
||||||
secondaryText.appendChild(text);
|
secondaryText.appendChild(text);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (textLines[0].text) {
|
if (textLines[0].text) {
|
||||||
let text = document.createElement('a');
|
const text = document.createElement('a');
|
||||||
text.innerHTML = textLines[0].text;
|
text.innerHTML = textLines[0].text;
|
||||||
itemText.appendChild(text);
|
itemText.appendChild(text);
|
||||||
}
|
}
|
||||||
|
@ -555,10 +555,10 @@ import 'emby-ratingbutton';
|
||||||
if (!layoutManager.mobile) {
|
if (!layoutManager.mobile) {
|
||||||
let contextButton = nowPlayingBarElement.querySelector('.btnToggleContextMenu');
|
let contextButton = nowPlayingBarElement.querySelector('.btnToggleContextMenu');
|
||||||
// We remove the previous event listener by replacing the item in each update event
|
// We remove the previous event listener by replacing the item in each update event
|
||||||
let contextButtonClone = contextButton.cloneNode(true);
|
const contextButtonClone = contextButton.cloneNode(true);
|
||||||
contextButton.parentNode.replaceChild(contextButtonClone, contextButton);
|
contextButton.parentNode.replaceChild(contextButtonClone, contextButton);
|
||||||
contextButton = nowPlayingBarElement.querySelector('.btnToggleContextMenu');
|
contextButton = nowPlayingBarElement.querySelector('.btnToggleContextMenu');
|
||||||
let options = {
|
const options = {
|
||||||
play: false,
|
play: false,
|
||||||
queue: false,
|
queue: false,
|
||||||
clearQueue: true,
|
clearQueue: true,
|
||||||
|
@ -600,10 +600,10 @@ import 'emby-ratingbutton';
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
let shuffleMode = playbackManager.getQueueShuffleMode();
|
const shuffleMode = playbackManager.getQueueShuffleMode();
|
||||||
let context = nowPlayingBarElement;
|
const context = nowPlayingBarElement;
|
||||||
const cssClass = 'buttonActive';
|
const cssClass = 'buttonActive';
|
||||||
let toggleShuffleButton = context.querySelector('.btnShuffleQueue');
|
const toggleShuffleButton = context.querySelector('.btnShuffleQueue');
|
||||||
switch (shuffleMode) {
|
switch (shuffleMode) {
|
||||||
case 'Shuffle':
|
case 'Shuffle':
|
||||||
toggleShuffleButton.classList.add(cssClass);
|
toggleShuffleButton.classList.add(cssClass);
|
||||||
|
|
|
@ -127,7 +127,7 @@ import connectionManager from 'connectionManager';
|
||||||
artwork: getImageUrls(item)
|
artwork: getImageUrls(item)
|
||||||
});
|
});
|
||||||
} else {
|
} else {
|
||||||
let itemImageUrl = seriesImageUrl(item, { maxHeight: 3000 }) || imageUrl(item, { maxHeight: 3000 });
|
const itemImageUrl = seriesImageUrl(item, { maxHeight: 3000 }) || imageUrl(item, { maxHeight: 3000 });
|
||||||
|
|
||||||
window.NativeShell.updateMediaSession({
|
window.NativeShell.updateMediaSession({
|
||||||
action: eventName,
|
action: eventName,
|
||||||
|
@ -244,10 +244,10 @@ import connectionManager from 'connectionManager';
|
||||||
|
|
||||||
/* eslint-disable-next-line compat/compat */
|
/* eslint-disable-next-line compat/compat */
|
||||||
navigator.mediaSession.setActionHandler('seekto', function (object) {
|
navigator.mediaSession.setActionHandler('seekto', function (object) {
|
||||||
let item = playbackManager.getPlayerState(currentPlayer).NowPlayingItem;
|
const item = playbackManager.getPlayerState(currentPlayer).NowPlayingItem;
|
||||||
// Convert to ms
|
// Convert to ms
|
||||||
let duration = parseInt(item.RunTimeTicks ? (item.RunTimeTicks / 10000) : 0);
|
const duration = parseInt(item.RunTimeTicks ? (item.RunTimeTicks / 10000) : 0);
|
||||||
let wantedTime = object.seekTime * 1000;
|
const wantedTime = object.seekTime * 1000;
|
||||||
playbackManager.seekPercent(wantedTime / duration * 100, currentPlayer);
|
playbackManager.seekPercent(wantedTime / duration * 100, currentPlayer);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
|
@ -8,7 +8,7 @@ import * as userSettings from 'userSettings';
|
||||||
import globalize from 'globalize';
|
import globalize from 'globalize';
|
||||||
import connectionManager from 'connectionManager';
|
import connectionManager from 'connectionManager';
|
||||||
import loading from 'loading';
|
import loading from 'loading';
|
||||||
import apphost from 'apphost';
|
import appHost from 'apphost';
|
||||||
import screenfull from 'screenfull';
|
import screenfull from 'screenfull';
|
||||||
|
|
||||||
function enableLocalPlaylistManagement(player) {
|
function enableLocalPlaylistManagement(player) {
|
||||||
|
@ -322,7 +322,7 @@ function getAudioStreamUrl(item, transcodingProfile, directPlayContainers, maxBi
|
||||||
PlaySessionId: startingPlaySession,
|
PlaySessionId: startingPlaySession,
|
||||||
StartTimeTicks: startPosition || 0,
|
StartTimeTicks: startPosition || 0,
|
||||||
EnableRedirection: true,
|
EnableRedirection: true,
|
||||||
EnableRemoteMedia: apphost.supports('remoteaudio')
|
EnableRemoteMedia: appHost.supports('remoteaudio')
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -606,7 +606,7 @@ function supportsDirectPlay(apiClient, item, mediaSource) {
|
||||||
const isFolderRip = mediaSource.VideoType === 'BluRay' || mediaSource.VideoType === 'Dvd' || mediaSource.VideoType === 'HdDvd';
|
const isFolderRip = mediaSource.VideoType === 'BluRay' || mediaSource.VideoType === 'Dvd' || mediaSource.VideoType === 'HdDvd';
|
||||||
|
|
||||||
if (mediaSource.SupportsDirectPlay || isFolderRip) {
|
if (mediaSource.SupportsDirectPlay || isFolderRip) {
|
||||||
if (mediaSource.IsRemote && !apphost.supports('remotevideo')) {
|
if (mediaSource.IsRemote && !appHost.supports('remotevideo')) {
|
||||||
return Promise.resolve(false);
|
return Promise.resolve(false);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -3156,7 +3156,7 @@ class PlaybackManager {
|
||||||
return streamInfo ? streamInfo.playbackStartTimeTicks : null;
|
return streamInfo ? streamInfo.playbackStartTimeTicks : null;
|
||||||
};
|
};
|
||||||
|
|
||||||
if (apphost.supports('remotecontrol')) {
|
if (appHost.supports('remotecontrol')) {
|
||||||
import('serverNotifications').then(({ default: serverNotifications }) => {
|
import('serverNotifications').then(({ default: serverNotifications }) => {
|
||||||
events.on(serverNotifications, 'ServerShuttingDown', self.setDefaultPlayerActive.bind(self));
|
events.on(serverNotifications, 'ServerShuttingDown', self.setDefaultPlayerActive.bind(self));
|
||||||
events.on(serverNotifications, 'ServerRestarting', self.setDefaultPlayerActive.bind(self));
|
events.on(serverNotifications, 'ServerRestarting', self.setDefaultPlayerActive.bind(self));
|
||||||
|
@ -3520,7 +3520,7 @@ class PlaybackManager {
|
||||||
'PlayTrailers'
|
'PlayTrailers'
|
||||||
];
|
];
|
||||||
|
|
||||||
if (apphost.supports('fullscreenchange')) {
|
if (appHost.supports('fullscreenchange')) {
|
||||||
list.push('ToggleFullscreen');
|
list.push('ToggleFullscreen');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -99,6 +99,6 @@
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<button is="emby-button" type="submit" class="raised button-submit block btnSave hide">
|
<button is="emby-button" type="submit" class="raised button-submit block btnSave hide">
|
||||||
<span>${ButtonSave}</span>
|
<span>${Save}</span>
|
||||||
</button>
|
</button>
|
||||||
</form>
|
</form>
|
||||||
|
|
|
@ -1,160 +1,158 @@
|
||||||
define(['globalize'], function (globalize) {
|
import globalize from 'globalize';
|
||||||
'use strict';
|
|
||||||
|
|
||||||
function getVideoQualityOptions(options) {
|
export function getVideoQualityOptions(options) {
|
||||||
var maxStreamingBitrate = options.currentMaxBitrate;
|
var maxStreamingBitrate = options.currentMaxBitrate;
|
||||||
var videoWidth = options.videoWidth;
|
var videoWidth = options.videoWidth;
|
||||||
var videoHeight = options.videoHeight;
|
var videoHeight = options.videoHeight;
|
||||||
|
|
||||||
// If the aspect ratio is less than 16/9 (1.77), set the width as if it were pillarboxed.
|
// If the aspect ratio is less than 16/9 (1.77), set the width as if it were pillarboxed.
|
||||||
// 4:3 1440x1080 -> 1920x1080
|
// 4:3 1440x1080 -> 1920x1080
|
||||||
if (videoWidth / videoHeight < 16 / 9) {
|
if (videoWidth / videoHeight < 16 / 9) {
|
||||||
videoWidth = videoHeight * (16 / 9);
|
videoWidth = videoHeight * (16 / 9);
|
||||||
}
|
|
||||||
|
|
||||||
var maxAllowedWidth = videoWidth || 4096;
|
|
||||||
|
|
||||||
var qualityOptions = [];
|
|
||||||
|
|
||||||
if (maxAllowedWidth >= 3800) {
|
|
||||||
qualityOptions.push({ name: '4K - 120 Mbps', maxHeight: 2160, bitrate: 120000000 });
|
|
||||||
qualityOptions.push({ name: '4K - 100 Mbps', maxHeight: 2160, bitrate: 100000000 });
|
|
||||||
qualityOptions.push({ name: '4K - 80 Mbps', maxHeight: 2160, bitrate: 80000000 });
|
|
||||||
}
|
|
||||||
|
|
||||||
// Some 1080- videos are reported as 1912?
|
|
||||||
if (maxAllowedWidth >= 1900) {
|
|
||||||
qualityOptions.push({ name: '1080p - 60 Mbps', maxHeight: 1080, bitrate: 60000000 });
|
|
||||||
qualityOptions.push({ name: '1080p - 50 Mbps', maxHeight: 1080, bitrate: 50000000 });
|
|
||||||
qualityOptions.push({ name: '1080p - 40 Mbps', maxHeight: 1080, bitrate: 40000000 });
|
|
||||||
qualityOptions.push({ name: '1080p - 30 Mbps', maxHeight: 1080, bitrate: 30000000 });
|
|
||||||
qualityOptions.push({ name: '1080p - 25 Mbps', maxHeight: 1080, bitrate: 25000000 });
|
|
||||||
qualityOptions.push({ name: '1080p - 20 Mbps', maxHeight: 1080, bitrate: 20000000 });
|
|
||||||
qualityOptions.push({ name: '1080p - 15 Mbps', maxHeight: 1080, bitrate: 15000000 });
|
|
||||||
qualityOptions.push({ name: '1080p - 10 Mbps', maxHeight: 1080, bitrate: 10000001 });
|
|
||||||
qualityOptions.push({ name: '1080p - 8 Mbps', maxHeight: 1080, bitrate: 8000001 });
|
|
||||||
qualityOptions.push({ name: '1080p - 6 Mbps', maxHeight: 1080, bitrate: 6000001 });
|
|
||||||
qualityOptions.push({ name: '1080p - 5 Mbps', maxHeight: 1080, bitrate: 5000001 });
|
|
||||||
qualityOptions.push({ name: '1080p - 4 Mbps', maxHeight: 1080, bitrate: 4000002 });
|
|
||||||
} else if (maxAllowedWidth >= 1260) {
|
|
||||||
qualityOptions.push({ name: '720p - 10 Mbps', maxHeight: 720, bitrate: 10000000 });
|
|
||||||
qualityOptions.push({ name: '720p - 8 Mbps', maxHeight: 720, bitrate: 8000000 });
|
|
||||||
qualityOptions.push({ name: '720p - 6 Mbps', maxHeight: 720, bitrate: 6000000 });
|
|
||||||
qualityOptions.push({ name: '720p - 5 Mbps', maxHeight: 720, bitrate: 5000000 });
|
|
||||||
} else if (maxAllowedWidth >= 620) {
|
|
||||||
qualityOptions.push({ name: '480p - 4 Mbps', maxHeight: 480, bitrate: 4000001 });
|
|
||||||
qualityOptions.push({ name: '480p - 3 Mbps', maxHeight: 480, bitrate: 3000001 });
|
|
||||||
qualityOptions.push({ name: '480p - 2.5 Mbps', maxHeight: 480, bitrate: 2500000 });
|
|
||||||
qualityOptions.push({ name: '480p - 2 Mbps', maxHeight: 480, bitrate: 2000001 });
|
|
||||||
qualityOptions.push({ name: '480p - 1.5 Mbps', maxHeight: 480, bitrate: 1500001 });
|
|
||||||
}
|
|
||||||
|
|
||||||
if (maxAllowedWidth >= 1260) {
|
|
||||||
qualityOptions.push({ name: '720p - 4 Mbps', maxHeight: 720, bitrate: 4000000 });
|
|
||||||
qualityOptions.push({ name: '720p - 3 Mbps', maxHeight: 720, bitrate: 3000000 });
|
|
||||||
qualityOptions.push({ name: '720p - 2 Mbps', maxHeight: 720, bitrate: 2000000 });
|
|
||||||
|
|
||||||
// The extra 1 is because they're keyed off the bitrate value
|
|
||||||
qualityOptions.push({ name: '720p - 1.5 Mbps', maxHeight: 720, bitrate: 1500000 });
|
|
||||||
qualityOptions.push({ name: '720p - 1 Mbps', maxHeight: 720, bitrate: 1000001 });
|
|
||||||
}
|
|
||||||
|
|
||||||
qualityOptions.push({ name: '480p - 1 Mbps', maxHeight: 480, bitrate: 1000000 });
|
|
||||||
qualityOptions.push({ name: '480p - 720 kbps', maxHeight: 480, bitrate: 720000 });
|
|
||||||
qualityOptions.push({ name: '480p - 420 kbps', maxHeight: 480, bitrate: 420000 });
|
|
||||||
qualityOptions.push({ name: '360p', maxHeight: 360, bitrate: 400000 });
|
|
||||||
qualityOptions.push({ name: '240p', maxHeight: 240, bitrate: 320000 });
|
|
||||||
qualityOptions.push({ name: '144p', maxHeight: 144, bitrate: 192000 });
|
|
||||||
|
|
||||||
var autoQualityOption = {
|
|
||||||
name: globalize.translate('Auto'),
|
|
||||||
bitrate: 0,
|
|
||||||
selected: options.isAutomaticBitrateEnabled
|
|
||||||
};
|
|
||||||
|
|
||||||
if (options.enableAuto) {
|
|
||||||
qualityOptions.push(autoQualityOption);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (maxStreamingBitrate) {
|
|
||||||
var selectedIndex = -1;
|
|
||||||
for (var i = 0, length = qualityOptions.length; i < length; i++) {
|
|
||||||
var option = qualityOptions[i];
|
|
||||||
|
|
||||||
if (selectedIndex === -1 && option.bitrate <= maxStreamingBitrate) {
|
|
||||||
selectedIndex = i;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (selectedIndex === -1) {
|
|
||||||
selectedIndex = qualityOptions.length - 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
var currentQualityOption = qualityOptions[selectedIndex];
|
|
||||||
|
|
||||||
if (!options.isAutomaticBitrateEnabled) {
|
|
||||||
currentQualityOption.selected = true;
|
|
||||||
} else {
|
|
||||||
autoQualityOption.autoText = currentQualityOption.name;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return qualityOptions;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function getAudioQualityOptions(options) {
|
var maxAllowedWidth = videoWidth || 4096;
|
||||||
var maxStreamingBitrate = options.currentMaxBitrate;
|
|
||||||
|
|
||||||
var qualityOptions = [];
|
var qualityOptions = [];
|
||||||
|
|
||||||
qualityOptions.push({ name: '2 Mbps', bitrate: 2000000 });
|
if (maxAllowedWidth >= 3800) {
|
||||||
qualityOptions.push({ name: '1.5 Mbps', bitrate: 1500000 });
|
qualityOptions.push({ name: '4K - 120 Mbps', maxHeight: 2160, bitrate: 120000000 });
|
||||||
qualityOptions.push({ name: '1 Mbps', bitrate: 1000000 });
|
qualityOptions.push({ name: '4K - 100 Mbps', maxHeight: 2160, bitrate: 100000000 });
|
||||||
qualityOptions.push({ name: '320 kbps', bitrate: 320000 });
|
qualityOptions.push({ name: '4K - 80 Mbps', maxHeight: 2160, bitrate: 80000000 });
|
||||||
qualityOptions.push({ name: '256 kbps', bitrate: 256000 });
|
|
||||||
qualityOptions.push({ name: '192 kbps', bitrate: 192000 });
|
|
||||||
qualityOptions.push({ name: '128 kbps', bitrate: 128000 });
|
|
||||||
qualityOptions.push({ name: '96 kbps', bitrate: 96000 });
|
|
||||||
qualityOptions.push({ name: '64 kbps', bitrate: 64000 });
|
|
||||||
|
|
||||||
var autoQualityOption = {
|
|
||||||
name: globalize.translate('Auto'),
|
|
||||||
bitrate: 0,
|
|
||||||
selected: options.isAutomaticBitrateEnabled
|
|
||||||
};
|
|
||||||
|
|
||||||
if (options.enableAuto) {
|
|
||||||
qualityOptions.push(autoQualityOption);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (maxStreamingBitrate) {
|
|
||||||
var selectedIndex = -1;
|
|
||||||
for (var i = 0, length = qualityOptions.length; i < length; i++) {
|
|
||||||
var option = qualityOptions[i];
|
|
||||||
|
|
||||||
if (selectedIndex === -1 && option.bitrate <= maxStreamingBitrate) {
|
|
||||||
selectedIndex = i;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (selectedIndex === -1) {
|
|
||||||
selectedIndex = qualityOptions.length - 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
var currentQualityOption = qualityOptions[selectedIndex];
|
|
||||||
|
|
||||||
if (!options.isAutomaticBitrateEnabled) {
|
|
||||||
currentQualityOption.selected = true;
|
|
||||||
} else {
|
|
||||||
autoQualityOption.autoText = currentQualityOption.name;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return qualityOptions;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return {
|
// Some 1080- videos are reported as 1912?
|
||||||
getVideoQualityOptions: getVideoQualityOptions,
|
if (maxAllowedWidth >= 1900) {
|
||||||
getAudioQualityOptions: getAudioQualityOptions
|
qualityOptions.push({ name: '1080p - 60 Mbps', maxHeight: 1080, bitrate: 60000000 });
|
||||||
|
qualityOptions.push({ name: '1080p - 50 Mbps', maxHeight: 1080, bitrate: 50000000 });
|
||||||
|
qualityOptions.push({ name: '1080p - 40 Mbps', maxHeight: 1080, bitrate: 40000000 });
|
||||||
|
qualityOptions.push({ name: '1080p - 30 Mbps', maxHeight: 1080, bitrate: 30000000 });
|
||||||
|
qualityOptions.push({ name: '1080p - 25 Mbps', maxHeight: 1080, bitrate: 25000000 });
|
||||||
|
qualityOptions.push({ name: '1080p - 20 Mbps', maxHeight: 1080, bitrate: 20000000 });
|
||||||
|
qualityOptions.push({ name: '1080p - 15 Mbps', maxHeight: 1080, bitrate: 15000000 });
|
||||||
|
qualityOptions.push({ name: '1080p - 10 Mbps', maxHeight: 1080, bitrate: 10000001 });
|
||||||
|
qualityOptions.push({ name: '1080p - 8 Mbps', maxHeight: 1080, bitrate: 8000001 });
|
||||||
|
qualityOptions.push({ name: '1080p - 6 Mbps', maxHeight: 1080, bitrate: 6000001 });
|
||||||
|
qualityOptions.push({ name: '1080p - 5 Mbps', maxHeight: 1080, bitrate: 5000001 });
|
||||||
|
qualityOptions.push({ name: '1080p - 4 Mbps', maxHeight: 1080, bitrate: 4000002 });
|
||||||
|
} else if (maxAllowedWidth >= 1260) {
|
||||||
|
qualityOptions.push({ name: '720p - 10 Mbps', maxHeight: 720, bitrate: 10000000 });
|
||||||
|
qualityOptions.push({ name: '720p - 8 Mbps', maxHeight: 720, bitrate: 8000000 });
|
||||||
|
qualityOptions.push({ name: '720p - 6 Mbps', maxHeight: 720, bitrate: 6000000 });
|
||||||
|
qualityOptions.push({ name: '720p - 5 Mbps', maxHeight: 720, bitrate: 5000000 });
|
||||||
|
} else if (maxAllowedWidth >= 620) {
|
||||||
|
qualityOptions.push({ name: '480p - 4 Mbps', maxHeight: 480, bitrate: 4000001 });
|
||||||
|
qualityOptions.push({ name: '480p - 3 Mbps', maxHeight: 480, bitrate: 3000001 });
|
||||||
|
qualityOptions.push({ name: '480p - 2.5 Mbps', maxHeight: 480, bitrate: 2500000 });
|
||||||
|
qualityOptions.push({ name: '480p - 2 Mbps', maxHeight: 480, bitrate: 2000001 });
|
||||||
|
qualityOptions.push({ name: '480p - 1.5 Mbps', maxHeight: 480, bitrate: 1500001 });
|
||||||
|
}
|
||||||
|
|
||||||
|
if (maxAllowedWidth >= 1260) {
|
||||||
|
qualityOptions.push({ name: '720p - 4 Mbps', maxHeight: 720, bitrate: 4000000 });
|
||||||
|
qualityOptions.push({ name: '720p - 3 Mbps', maxHeight: 720, bitrate: 3000000 });
|
||||||
|
qualityOptions.push({ name: '720p - 2 Mbps', maxHeight: 720, bitrate: 2000000 });
|
||||||
|
|
||||||
|
// The extra 1 is because they're keyed off the bitrate value
|
||||||
|
qualityOptions.push({ name: '720p - 1.5 Mbps', maxHeight: 720, bitrate: 1500000 });
|
||||||
|
qualityOptions.push({ name: '720p - 1 Mbps', maxHeight: 720, bitrate: 1000001 });
|
||||||
|
}
|
||||||
|
|
||||||
|
qualityOptions.push({ name: '480p - 1 Mbps', maxHeight: 480, bitrate: 1000000 });
|
||||||
|
qualityOptions.push({ name: '480p - 720 kbps', maxHeight: 480, bitrate: 720000 });
|
||||||
|
qualityOptions.push({ name: '480p - 420 kbps', maxHeight: 480, bitrate: 420000 });
|
||||||
|
qualityOptions.push({ name: '360p', maxHeight: 360, bitrate: 400000 });
|
||||||
|
qualityOptions.push({ name: '240p', maxHeight: 240, bitrate: 320000 });
|
||||||
|
qualityOptions.push({ name: '144p', maxHeight: 144, bitrate: 192000 });
|
||||||
|
|
||||||
|
var autoQualityOption = {
|
||||||
|
name: globalize.translate('Auto'),
|
||||||
|
bitrate: 0,
|
||||||
|
selected: options.isAutomaticBitrateEnabled
|
||||||
};
|
};
|
||||||
});
|
|
||||||
|
if (options.enableAuto) {
|
||||||
|
qualityOptions.push(autoQualityOption);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (maxStreamingBitrate) {
|
||||||
|
var selectedIndex = -1;
|
||||||
|
for (var i = 0, length = qualityOptions.length; i < length; i++) {
|
||||||
|
var option = qualityOptions[i];
|
||||||
|
|
||||||
|
if (selectedIndex === -1 && option.bitrate <= maxStreamingBitrate) {
|
||||||
|
selectedIndex = i;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (selectedIndex === -1) {
|
||||||
|
selectedIndex = qualityOptions.length - 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
var currentQualityOption = qualityOptions[selectedIndex];
|
||||||
|
|
||||||
|
if (!options.isAutomaticBitrateEnabled) {
|
||||||
|
currentQualityOption.selected = true;
|
||||||
|
} else {
|
||||||
|
autoQualityOption.autoText = currentQualityOption.name;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return qualityOptions;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function getAudioQualityOptions(options) {
|
||||||
|
var maxStreamingBitrate = options.currentMaxBitrate;
|
||||||
|
|
||||||
|
var qualityOptions = [];
|
||||||
|
|
||||||
|
qualityOptions.push({ name: '2 Mbps', bitrate: 2000000 });
|
||||||
|
qualityOptions.push({ name: '1.5 Mbps', bitrate: 1500000 });
|
||||||
|
qualityOptions.push({ name: '1 Mbps', bitrate: 1000000 });
|
||||||
|
qualityOptions.push({ name: '320 kbps', bitrate: 320000 });
|
||||||
|
qualityOptions.push({ name: '256 kbps', bitrate: 256000 });
|
||||||
|
qualityOptions.push({ name: '192 kbps', bitrate: 192000 });
|
||||||
|
qualityOptions.push({ name: '128 kbps', bitrate: 128000 });
|
||||||
|
qualityOptions.push({ name: '96 kbps', bitrate: 96000 });
|
||||||
|
qualityOptions.push({ name: '64 kbps', bitrate: 64000 });
|
||||||
|
|
||||||
|
var autoQualityOption = {
|
||||||
|
name: globalize.translate('Auto'),
|
||||||
|
bitrate: 0,
|
||||||
|
selected: options.isAutomaticBitrateEnabled
|
||||||
|
};
|
||||||
|
|
||||||
|
if (options.enableAuto) {
|
||||||
|
qualityOptions.push(autoQualityOption);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (maxStreamingBitrate) {
|
||||||
|
var selectedIndex = -1;
|
||||||
|
for (var i = 0, length = qualityOptions.length; i < length; i++) {
|
||||||
|
var option = qualityOptions[i];
|
||||||
|
|
||||||
|
if (selectedIndex === -1 && option.bitrate <= maxStreamingBitrate) {
|
||||||
|
selectedIndex = i;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (selectedIndex === -1) {
|
||||||
|
selectedIndex = qualityOptions.length - 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
var currentQualityOption = qualityOptions[selectedIndex];
|
||||||
|
|
||||||
|
if (!options.isAutomaticBitrateEnabled) {
|
||||||
|
currentQualityOption.selected = true;
|
||||||
|
} else {
|
||||||
|
autoQualityOption.autoText = currentQualityOption.name;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return qualityOptions;
|
||||||
|
}
|
||||||
|
|
||||||
|
export default {
|
||||||
|
getVideoQualityOptions,
|
||||||
|
getAudioQualityOptions
|
||||||
|
};
|
||||||
|
|
|
@ -251,7 +251,7 @@ import layoutManager from 'layoutManager';
|
||||||
* @return {ScrollerData} Scroller data.
|
* @return {ScrollerData} Scroller data.
|
||||||
*/
|
*/
|
||||||
function getScrollerData(scroller, vertical) {
|
function getScrollerData(scroller, vertical) {
|
||||||
let data = {};
|
const data = {};
|
||||||
|
|
||||||
if (!vertical) {
|
if (!vertical) {
|
||||||
data.scrollPos = scroller.scrollLeft;
|
data.scrollPos = scroller.scrollLeft;
|
||||||
|
|
|
@ -4,7 +4,7 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
function getTextStyles(settings, preview) {
|
function getTextStyles(settings, preview) {
|
||||||
let list = [];
|
const list = [];
|
||||||
|
|
||||||
switch (settings.textSize || '') {
|
switch (settings.textSize || '') {
|
||||||
case 'smaller':
|
case 'smaller':
|
||||||
|
@ -130,14 +130,14 @@ export function getStyles(settings, preview) {
|
||||||
|
|
||||||
function applyStyleList(styles, elem) {
|
function applyStyleList(styles, elem) {
|
||||||
for (let i = 0, length = styles.length; i < length; i++) {
|
for (let i = 0, length = styles.length; i < length; i++) {
|
||||||
let style = styles[i];
|
const style = styles[i];
|
||||||
|
|
||||||
elem.style[style.name] = style.value;
|
elem.style[style.name] = style.value;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export function applyStyles(elements, appearanceSettings) {
|
export function applyStyles(elements, appearanceSettings) {
|
||||||
let styles = getStyles(appearanceSettings, !!elements.preview);
|
const styles = getStyles(appearanceSettings, !!elements.preview);
|
||||||
|
|
||||||
if (elements.text) {
|
if (elements.text) {
|
||||||
applyStyleList(styles.text, elements.text);
|
applyStyleList(styles.text, elements.text);
|
||||||
|
|
|
@ -23,7 +23,7 @@ import 'css!./subtitlesettings';
|
||||||
*/
|
*/
|
||||||
|
|
||||||
function getSubtitleAppearanceObject(context) {
|
function getSubtitleAppearanceObject(context) {
|
||||||
let appearanceSettings = {};
|
const appearanceSettings = {};
|
||||||
|
|
||||||
appearanceSettings.textSize = context.querySelector('#selectTextSize').value;
|
appearanceSettings.textSize = context.querySelector('#selectTextSize').value;
|
||||||
appearanceSettings.dropShadow = context.querySelector('#selectDropShadow').value;
|
appearanceSettings.dropShadow = context.querySelector('#selectDropShadow').value;
|
||||||
|
@ -41,7 +41,7 @@ function loadForm(context, user, userSettings, appearanceSettings, apiClient) {
|
||||||
context.querySelector('.fldBurnIn').classList.remove('hide');
|
context.querySelector('.fldBurnIn').classList.remove('hide');
|
||||||
}
|
}
|
||||||
|
|
||||||
let selectSubtitleLanguage = context.querySelector('#selectSubtitleLanguage');
|
const selectSubtitleLanguage = context.querySelector('#selectSubtitleLanguage');
|
||||||
|
|
||||||
settingsHelper.populateLanguages(selectSubtitleLanguage, allCultures);
|
settingsHelper.populateLanguages(selectSubtitleLanguage, allCultures);
|
||||||
|
|
||||||
|
@ -101,9 +101,9 @@ function save(instance, context, userId, userSettings, apiClient, enableSaveConf
|
||||||
}
|
}
|
||||||
|
|
||||||
function onSubtitleModeChange(e) {
|
function onSubtitleModeChange(e) {
|
||||||
let view = dom.parentWithClass(e.target, 'subtitlesettings');
|
const view = dom.parentWithClass(e.target, 'subtitlesettings');
|
||||||
|
|
||||||
let subtitlesHelp = view.querySelectorAll('.subtitlesHelp');
|
const subtitlesHelp = view.querySelectorAll('.subtitlesHelp');
|
||||||
for (let i = 0, length = subtitlesHelp.length; i < length; i++) {
|
for (let i = 0, length = subtitlesHelp.length; i < length; i++) {
|
||||||
subtitlesHelp[i].classList.add('hide');
|
subtitlesHelp[i].classList.add('hide');
|
||||||
}
|
}
|
||||||
|
@ -111,11 +111,11 @@ function onSubtitleModeChange(e) {
|
||||||
}
|
}
|
||||||
|
|
||||||
function onAppearanceFieldChange(e) {
|
function onAppearanceFieldChange(e) {
|
||||||
let view = dom.parentWithClass(e.target, 'subtitlesettings');
|
const view = dom.parentWithClass(e.target, 'subtitlesettings');
|
||||||
|
|
||||||
let appearanceSettings = getSubtitleAppearanceObject(view);
|
const appearanceSettings = getSubtitleAppearanceObject(view);
|
||||||
|
|
||||||
let elements = {
|
const elements = {
|
||||||
window: view.querySelector('.subtitleappearance-preview-window'),
|
window: view.querySelector('.subtitleappearance-preview-window'),
|
||||||
text: view.querySelector('.subtitleappearance-preview-text'),
|
text: view.querySelector('.subtitleappearance-preview-text'),
|
||||||
preview: true
|
preview: true
|
||||||
|
@ -226,20 +226,20 @@ export class SubtitleSettings {
|
||||||
}
|
}
|
||||||
|
|
||||||
loadData() {
|
loadData() {
|
||||||
let self = this;
|
const self = this;
|
||||||
let context = self.options.element;
|
const context = self.options.element;
|
||||||
|
|
||||||
loading.show();
|
loading.show();
|
||||||
|
|
||||||
let userId = self.options.userId;
|
const userId = self.options.userId;
|
||||||
let apiClient = connectionManager.getApiClient(self.options.serverId);
|
const apiClient = connectionManager.getApiClient(self.options.serverId);
|
||||||
let userSettings = self.options.userSettings;
|
const userSettings = self.options.userSettings;
|
||||||
|
|
||||||
apiClient.getUser(userId).then(function (user) {
|
apiClient.getUser(userId).then(function (user) {
|
||||||
userSettings.setUserInfo(userId, apiClient).then(function () {
|
userSettings.setUserInfo(userId, apiClient).then(function () {
|
||||||
self.dataLoaded = true;
|
self.dataLoaded = true;
|
||||||
|
|
||||||
let appearanceSettings = userSettings.getSubtitleAppearanceSettings(self.options.appearanceKey);
|
const appearanceSettings = userSettings.getSubtitleAppearanceSettings(self.options.appearanceKey);
|
||||||
|
|
||||||
loadForm(context, user, userSettings, appearanceSettings, apiClient);
|
loadForm(context, user, userSettings, appearanceSettings, apiClient);
|
||||||
});
|
});
|
||||||
|
@ -256,12 +256,12 @@ export class SubtitleSettings {
|
||||||
|
|
||||||
onSubmit(e) {
|
onSubmit(e) {
|
||||||
const self = this;
|
const self = this;
|
||||||
let apiClient = connectionManager.getApiClient(self.options.serverId);
|
const apiClient = connectionManager.getApiClient(self.options.serverId);
|
||||||
let userId = self.options.userId;
|
const userId = self.options.userId;
|
||||||
let userSettings = self.options.userSettings;
|
const userSettings = self.options.userSettings;
|
||||||
|
|
||||||
userSettings.setUserInfo(userId, apiClient).then(function () {
|
userSettings.setUserInfo(userId, apiClient).then(function () {
|
||||||
let enableSaveConfirmation = self.options.enableSaveConfirmation;
|
const enableSaveConfirmation = self.options.enableSaveConfirmation;
|
||||||
save(self, self.options.element, userId, userSettings, apiClient, enableSaveConfirmation);
|
save(self, self.options.element, userId, userSettings, apiClient, enableSaveConfirmation);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
|
@ -1,103 +1,101 @@
|
||||||
define(['playbackManager', 'userSettings', 'connectionManager'], function (playbackManager, userSettings, connectionManager) {
|
import playbackManager from 'playbackManager';
|
||||||
'use strict';
|
import * as userSettings from 'userSettings';
|
||||||
|
import connectionManager from 'connectionManager';
|
||||||
|
|
||||||
playbackManager = playbackManager.default || playbackManager;
|
let currentOwnerId;
|
||||||
|
let currentThemeIds = [];
|
||||||
|
|
||||||
var currentOwnerId;
|
function playThemeMedia(items, ownerId) {
|
||||||
var currentThemeIds = [];
|
const currentThemeItems = items.filter(function (i) {
|
||||||
|
return enabled(i.MediaType);
|
||||||
|
});
|
||||||
|
|
||||||
function playThemeMedia(items, ownerId) {
|
if (currentThemeItems.length) {
|
||||||
var currentThemeItems = items.filter(function (i) {
|
// Stop if a theme song from another ownerId
|
||||||
return enabled(i.MediaType);
|
// Leave it alone if anything else (e.g user playing a movie)
|
||||||
|
if (!currentOwnerId && playbackManager.isPlaying()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
currentThemeIds = currentThemeItems.map(function (i) {
|
||||||
|
return i.Id;
|
||||||
});
|
});
|
||||||
|
|
||||||
if (currentThemeItems.length) {
|
playbackManager.play({
|
||||||
// Stop if a theme song from another ownerId
|
items: currentThemeItems,
|
||||||
// Leave it alone if anything else (e.g user playing a movie)
|
fullscreen: false,
|
||||||
if (!currentOwnerId && playbackManager.isPlaying()) {
|
enableRemotePlayers: false
|
||||||
return;
|
}).then(function () {
|
||||||
}
|
currentOwnerId = ownerId;
|
||||||
|
|
||||||
currentThemeIds = currentThemeItems.map(function (i) {
|
|
||||||
return i.Id;
|
|
||||||
});
|
|
||||||
|
|
||||||
playbackManager.play({
|
|
||||||
items: currentThemeItems,
|
|
||||||
fullscreen: false,
|
|
||||||
enableRemotePlayers: false
|
|
||||||
}).then(function () {
|
|
||||||
currentOwnerId = ownerId;
|
|
||||||
});
|
|
||||||
} else {
|
|
||||||
stopIfPlaying();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function stopIfPlaying() {
|
|
||||||
if (currentOwnerId) {
|
|
||||||
playbackManager.stop();
|
|
||||||
}
|
|
||||||
|
|
||||||
currentOwnerId = null;
|
|
||||||
}
|
|
||||||
|
|
||||||
function enabled(mediaType) {
|
|
||||||
if (mediaType === 'Video') {
|
|
||||||
return userSettings.enableThemeVideos();
|
|
||||||
}
|
|
||||||
|
|
||||||
return userSettings.enableThemeSongs();
|
|
||||||
}
|
|
||||||
|
|
||||||
var excludeTypes = ['CollectionFolder', 'UserView', 'Program', 'SeriesTimer', 'Person', 'TvChannel', 'Channel'];
|
|
||||||
|
|
||||||
function loadThemeMedia(item) {
|
|
||||||
if (item.CollectionType) {
|
|
||||||
stopIfPlaying();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (excludeTypes.indexOf(item.Type) !== -1) {
|
|
||||||
stopIfPlaying();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
var apiClient = connectionManager.getApiClient(item.ServerId);
|
|
||||||
apiClient.getThemeMedia(apiClient.getCurrentUserId(), item.Id, true).then(function (themeMediaResult) {
|
|
||||||
var ownerId = themeMediaResult.ThemeVideosResult.Items.length ? themeMediaResult.ThemeVideosResult.OwnerId : themeMediaResult.ThemeSongsResult.OwnerId;
|
|
||||||
|
|
||||||
if (ownerId !== currentOwnerId) {
|
|
||||||
var items = themeMediaResult.ThemeVideosResult.Items.length ? themeMediaResult.ThemeVideosResult.Items : themeMediaResult.ThemeSongsResult.Items;
|
|
||||||
|
|
||||||
playThemeMedia(items, ownerId);
|
|
||||||
}
|
|
||||||
});
|
});
|
||||||
|
} else {
|
||||||
|
stopIfPlaying();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function stopIfPlaying() {
|
||||||
|
if (currentOwnerId) {
|
||||||
|
playbackManager.stop();
|
||||||
}
|
}
|
||||||
|
|
||||||
document.addEventListener('viewshow', function (e) {
|
currentOwnerId = null;
|
||||||
var state = e.detail.state || {};
|
}
|
||||||
var item = state.item;
|
|
||||||
|
|
||||||
if (item && item.ServerId) {
|
function enabled(mediaType) {
|
||||||
loadThemeMedia(item);
|
if (mediaType === 'Video') {
|
||||||
return;
|
return userSettings.enableThemeVideos();
|
||||||
}
|
}
|
||||||
|
|
||||||
var viewOptions = e.detail.options || {};
|
return userSettings.enableThemeSongs();
|
||||||
|
}
|
||||||
|
|
||||||
if (viewOptions.supportsThemeMedia) {
|
const excludeTypes = ['CollectionFolder', 'UserView', 'Program', 'SeriesTimer', 'Person', 'TvChannel', 'Channel'];
|
||||||
// Do nothing here, allow it to keep playing
|
|
||||||
} else {
|
|
||||||
playThemeMedia([], null);
|
|
||||||
}
|
|
||||||
}, true);
|
|
||||||
|
|
||||||
Events.on(playbackManager, 'playbackstart', function (e, player) {
|
function loadThemeMedia(item) {
|
||||||
var item = playbackManager.currentItem(player);
|
if (item.CollectionType) {
|
||||||
// User played something manually
|
stopIfPlaying();
|
||||||
if (currentThemeIds.indexOf(item.Id) == -1) {
|
return;
|
||||||
currentOwnerId = null;
|
}
|
||||||
|
|
||||||
|
if (excludeTypes.indexOf(item.Type) !== -1) {
|
||||||
|
stopIfPlaying();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const apiClient = connectionManager.getApiClient(item.ServerId);
|
||||||
|
apiClient.getThemeMedia(apiClient.getCurrentUserId(), item.Id, true).then(function (themeMediaResult) {
|
||||||
|
const ownerId = themeMediaResult.ThemeVideosResult.Items.length ? themeMediaResult.ThemeVideosResult.OwnerId : themeMediaResult.ThemeSongsResult.OwnerId;
|
||||||
|
|
||||||
|
if (ownerId !== currentOwnerId) {
|
||||||
|
const items = themeMediaResult.ThemeVideosResult.Items.length ? themeMediaResult.ThemeVideosResult.Items : themeMediaResult.ThemeSongsResult.Items;
|
||||||
|
|
||||||
|
playThemeMedia(items, ownerId);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
document.addEventListener('viewshow', function (e) {
|
||||||
|
const state = e.detail.state || {};
|
||||||
|
const item = state.item;
|
||||||
|
|
||||||
|
if (item && item.ServerId) {
|
||||||
|
loadThemeMedia(item);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const viewOptions = e.detail.options || {};
|
||||||
|
|
||||||
|
if (viewOptions.supportsThemeMedia) {
|
||||||
|
// Do nothing here, allow it to keep playing
|
||||||
|
} else {
|
||||||
|
playThemeMedia([], null);
|
||||||
|
}
|
||||||
|
}, true);
|
||||||
|
|
||||||
|
Events.on(playbackManager, 'playbackstart', function (e, player) {
|
||||||
|
const item = playbackManager.currentItem(player);
|
||||||
|
// User played something manually
|
||||||
|
if (currentThemeIds.indexOf(item.Id) == -1) {
|
||||||
|
currentOwnerId = null;
|
||||||
|
}
|
||||||
});
|
});
|
||||||
|
|
|
@ -1,183 +1,185 @@
|
||||||
define(['dialogHelper', 'dom', 'layoutManager', 'connectionManager', 'globalize', 'loading', 'browser', 'focusManager', 'scrollHelper', 'material-icons', 'formDialogStyle', 'emby-button', 'emby-itemscontainer', 'cardStyle'], function (dialogHelper, dom, layoutManager, connectionManager, globalize, loading, browser, focusManager, scrollHelper) {
|
import dialogHelper from 'dialogHelper';
|
||||||
'use strict';
|
import dom from 'dom';
|
||||||
|
import layoutManager from 'layoutManager';
|
||||||
|
import connectionManager from 'connectionManager';
|
||||||
|
import globalize from 'globalize';
|
||||||
|
import loading from 'loading';
|
||||||
|
import browser from 'browser';
|
||||||
|
import focusManager from 'focusManager';
|
||||||
|
import scrollHelper from 'scrollHelper';
|
||||||
|
import 'material-icons';
|
||||||
|
import 'formDialogStyle';
|
||||||
|
import 'emby-button';
|
||||||
|
import 'emby-itemscontainer';
|
||||||
|
import 'cardStyle';
|
||||||
|
|
||||||
browser = browser.default || browser;
|
const enableFocusTransform = !browser.slow && !browser.edge;
|
||||||
loading = loading.default || loading;
|
|
||||||
layoutManager = layoutManager.default || layoutManager;
|
|
||||||
focusManager = focusManager.default || focusManager;
|
|
||||||
scrollHelper = scrollHelper.default || scrollHelper;
|
|
||||||
|
|
||||||
var enableFocusTransform = !browser.slow && !browser.edge;
|
function getEditorHtml() {
|
||||||
|
let html = '';
|
||||||
|
html += '<div class="formDialogContent scrollY">';
|
||||||
|
html += '<div class="dialogContentInner dialog-content-centered">';
|
||||||
|
html += '<div class="loadingContent hide">';
|
||||||
|
html += '<h1>' + globalize.translate('DetectingDevices') + '...</h1>';
|
||||||
|
html += '<p>' + globalize.translate('MessagePleaseWait') + '</p>';
|
||||||
|
html += '</div>';
|
||||||
|
html += '<h1 style="margin-bottom:.25em;" class="devicesHeader hide">' + globalize.translate('HeaderNewDevices') + '</h1>';
|
||||||
|
html += '<div is="emby-itemscontainer" class="results vertical-wrap">';
|
||||||
|
html += '</div>';
|
||||||
|
html += '</div>';
|
||||||
|
return html += '</div>';
|
||||||
|
}
|
||||||
|
|
||||||
function getEditorHtml() {
|
function getDeviceHtml(device) {
|
||||||
var html = '';
|
let html = '';
|
||||||
html += '<div class="formDialogContent scrollY">';
|
let cssClass = 'card scalableCard backdropCard backdropCard-scalable';
|
||||||
html += '<div class="dialogContentInner dialog-content-centered">';
|
const cardBoxCssClass = 'cardBox visualCardBox';
|
||||||
html += '<div class="loadingContent hide">';
|
const padderClass = 'cardPadder-backdrop';
|
||||||
html += '<h1>' + globalize.translate('DetectingDevices') + '...</h1>';
|
|
||||||
html += '<p>' + globalize.translate('MessagePleaseWait') + '</p>';
|
|
||||||
html += '</div>';
|
|
||||||
html += '<h1 style="margin-bottom:.25em;" class="devicesHeader hide">' + globalize.translate('HeaderNewDevices') + '</h1>';
|
|
||||||
html += '<div is="emby-itemscontainer" class="results vertical-wrap">';
|
|
||||||
html += '</div>';
|
|
||||||
html += '</div>';
|
|
||||||
return html += '</div>';
|
|
||||||
}
|
|
||||||
|
|
||||||
function getDeviceHtml(device) {
|
// TODO move card creation code to Card component
|
||||||
var padderClass;
|
|
||||||
var html = '';
|
|
||||||
var cssClass = 'card scalableCard';
|
|
||||||
var cardBoxCssClass = 'cardBox visualCardBox';
|
|
||||||
cssClass += ' backdropCard backdropCard-scalable';
|
|
||||||
padderClass = 'cardPadder-backdrop';
|
|
||||||
|
|
||||||
// TODO move card creation code to Card component
|
if (layoutManager.tv) {
|
||||||
|
cssClass += ' show-focus';
|
||||||
|
|
||||||
if (layoutManager.tv) {
|
if (enableFocusTransform) {
|
||||||
cssClass += ' show-focus';
|
cssClass += ' show-animation';
|
||||||
|
|
||||||
if (enableFocusTransform) {
|
|
||||||
cssClass += ' show-animation';
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
html += '<button type="button" class="' + cssClass + '" data-id="' + device.DeviceId + '" style="min-width:33.3333%;">';
|
|
||||||
html += '<div class="' + cardBoxCssClass + '">';
|
|
||||||
html += '<div class="cardScalable visualCardBox-cardScalable">';
|
|
||||||
html += '<div class="' + padderClass + '"></div>';
|
|
||||||
html += '<div class="cardContent searchImage">';
|
|
||||||
html += '<div class="cardImageContainer coveredImage"><span class="cardImageIcon material-icons dvr"></span></div>';
|
|
||||||
html += '</div>';
|
|
||||||
html += '</div>';
|
|
||||||
html += '<div class="cardFooter visualCardBox-cardFooter">';
|
|
||||||
html += '<div class="cardText cardTextCentered">' + getTunerName(device.Type) + '</div>';
|
|
||||||
html += '<div class="cardText cardTextCentered cardText-secondary">' + device.FriendlyName + '</div>';
|
|
||||||
html += '<div class="cardText cardText-secondary cardTextCentered">';
|
|
||||||
html += device.Url || ' ';
|
|
||||||
html += '</div>';
|
|
||||||
html += '</div>';
|
|
||||||
html += '</div>';
|
|
||||||
return html += '</button>';
|
|
||||||
}
|
|
||||||
|
|
||||||
function getTunerName(providerId) {
|
|
||||||
switch (providerId = providerId.toLowerCase()) {
|
|
||||||
case 'm3u':
|
|
||||||
return 'M3U';
|
|
||||||
|
|
||||||
case 'hdhomerun':
|
|
||||||
return 'HDHomerun';
|
|
||||||
|
|
||||||
case 'hauppauge':
|
|
||||||
return 'Hauppauge';
|
|
||||||
|
|
||||||
case 'satip':
|
|
||||||
return 'DVB';
|
|
||||||
|
|
||||||
default:
|
|
||||||
return 'Unknown';
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function renderDevices(view, devices) {
|
html += '<button type="button" class="' + cssClass + '" data-id="' + device.DeviceId + '" style="min-width:33.3333%;">';
|
||||||
var i;
|
html += '<div class="' + cardBoxCssClass + '">';
|
||||||
var length;
|
html += '<div class="cardScalable visualCardBox-cardScalable">';
|
||||||
var html = '';
|
html += '<div class="' + padderClass + '"></div>';
|
||||||
|
html += '<div class="cardContent searchImage">';
|
||||||
|
html += '<div class="cardImageContainer coveredImage"><span class="cardImageIcon material-icons dvr"></span></div>';
|
||||||
|
html += '</div>';
|
||||||
|
html += '</div>';
|
||||||
|
html += '<div class="cardFooter visualCardBox-cardFooter">';
|
||||||
|
html += '<div class="cardText cardTextCentered">' + getTunerName(device.Type) + '</div>';
|
||||||
|
html += '<div class="cardText cardTextCentered cardText-secondary">' + device.FriendlyName + '</div>';
|
||||||
|
html += '<div class="cardText cardText-secondary cardTextCentered">';
|
||||||
|
html += device.Url || ' ';
|
||||||
|
html += '</div>';
|
||||||
|
html += '</div>';
|
||||||
|
html += '</div>';
|
||||||
|
return html += '</button>';
|
||||||
|
}
|
||||||
|
|
||||||
for (i = 0, length = devices.length; i < length; i++) {
|
function getTunerName(providerId) {
|
||||||
html += getDeviceHtml(devices[i]);
|
switch (providerId = providerId.toLowerCase()) {
|
||||||
}
|
case 'm3u':
|
||||||
|
return 'M3U';
|
||||||
|
|
||||||
if (devices.length) {
|
case 'hdhomerun':
|
||||||
view.querySelector('.devicesHeader').classList.remove('hide');
|
return 'HDHomerun';
|
||||||
} else {
|
|
||||||
html = '<p><br/>' + globalize.translate('NoNewDevicesFound') + '</p>';
|
|
||||||
view.querySelector('.devicesHeader').classList.add('hide');
|
|
||||||
}
|
|
||||||
|
|
||||||
var elem = view.querySelector('.results');
|
case 'hauppauge':
|
||||||
elem.innerHTML = html;
|
return 'Hauppauge';
|
||||||
|
|
||||||
if (layoutManager.tv) {
|
case 'satip':
|
||||||
focusManager.autoFocus(elem);
|
return 'DVB';
|
||||||
}
|
|
||||||
|
default:
|
||||||
|
return 'Unknown';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function renderDevices(view, devices) {
|
||||||
|
let html = '';
|
||||||
|
|
||||||
|
for (let i = 0, length = devices.length; i < length; i++) {
|
||||||
|
html += getDeviceHtml(devices[i]);
|
||||||
}
|
}
|
||||||
|
|
||||||
function discoverDevices(view, apiClient) {
|
if (devices.length) {
|
||||||
loading.show();
|
view.querySelector('.devicesHeader').classList.remove('hide');
|
||||||
view.querySelector('.loadingContent').classList.remove('hide');
|
} else {
|
||||||
return ApiClient.getJSON(ApiClient.getUrl('LiveTv/Tuners/Discvover', {
|
html = '<p><br/>' + globalize.translate('NoNewDevicesFound') + '</p>';
|
||||||
NewDevicesOnly: true
|
view.querySelector('.devicesHeader').classList.add('hide');
|
||||||
})).then(function (devices) {
|
|
||||||
currentDevices = devices;
|
|
||||||
renderDevices(view, devices);
|
|
||||||
view.querySelector('.loadingContent').classList.add('hide');
|
|
||||||
loading.hide();
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function tunerPicker() {
|
const elem = view.querySelector('.results');
|
||||||
this.show = function (options) {
|
elem.innerHTML = html;
|
||||||
var dialogOptions = {
|
|
||||||
removeOnClose: true,
|
|
||||||
scrollY: false
|
|
||||||
};
|
|
||||||
|
|
||||||
if (layoutManager.tv) {
|
if (layoutManager.tv) {
|
||||||
dialogOptions.size = 'fullscreen';
|
focusManager.autoFocus(elem);
|
||||||
} else {
|
}
|
||||||
dialogOptions.size = 'small';
|
}
|
||||||
}
|
|
||||||
|
|
||||||
var dlg = dialogHelper.createDialog(dialogOptions);
|
function discoverDevices(view, apiClient) {
|
||||||
dlg.classList.add('formDialog');
|
loading.show();
|
||||||
var html = '';
|
view.querySelector('.loadingContent').classList.remove('hide');
|
||||||
html += '<div class="formDialogHeader">';
|
return ApiClient.getJSON(ApiClient.getUrl('LiveTv/Tuners/Discvover', {
|
||||||
html += '<button is="paper-icon-button-light" class="btnCancel autoSize" tabindex="-1"><span class="material-icons arrow_back"></span></button>';
|
NewDevicesOnly: true
|
||||||
html += '<h3 class="formDialogHeaderTitle">';
|
})).then(function (devices) {
|
||||||
html += globalize.translate('HeaderLiveTvTunerSetup');
|
currentDevices = devices;
|
||||||
html += '</h3>';
|
renderDevices(view, devices);
|
||||||
html += '</div>';
|
view.querySelector('.loadingContent').classList.add('hide');
|
||||||
html += getEditorHtml();
|
loading.hide();
|
||||||
dlg.innerHTML = html;
|
});
|
||||||
dlg.querySelector('.btnCancel').addEventListener('click', function () {
|
}
|
||||||
dialogHelper.close(dlg);
|
|
||||||
});
|
|
||||||
var deviceResult;
|
|
||||||
dlg.querySelector('.results').addEventListener('click', function (e) {
|
|
||||||
var tunerCard = dom.parentWithClass(e.target, 'card');
|
|
||||||
|
|
||||||
if (tunerCard) {
|
function tunerPicker() {
|
||||||
var deviceId = tunerCard.getAttribute('data-id');
|
this.show = function (options) {
|
||||||
deviceResult = currentDevices.filter(function (d) {
|
const dialogOptions = {
|
||||||
return d.DeviceId === deviceId;
|
removeOnClose: true,
|
||||||
})[0];
|
scrollY: false
|
||||||
dialogHelper.close(dlg);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
if (layoutManager.tv) {
|
|
||||||
scrollHelper.centerFocus.on(dlg.querySelector('.formDialogContent'), false);
|
|
||||||
}
|
|
||||||
|
|
||||||
var apiClient = connectionManager.getApiClient(options.serverId);
|
|
||||||
discoverDevices(dlg, apiClient);
|
|
||||||
|
|
||||||
if (layoutManager.tv) {
|
|
||||||
scrollHelper.centerFocus.off(dlg.querySelector('.formDialogContent'), false);
|
|
||||||
}
|
|
||||||
|
|
||||||
return dialogHelper.open(dlg).then(function () {
|
|
||||||
if (deviceResult) {
|
|
||||||
return Promise.resolve(deviceResult);
|
|
||||||
}
|
|
||||||
|
|
||||||
return Promise.reject();
|
|
||||||
});
|
|
||||||
};
|
};
|
||||||
}
|
|
||||||
|
|
||||||
var currentDevices = [];
|
if (layoutManager.tv) {
|
||||||
return tunerPicker;
|
dialogOptions.size = 'fullscreen';
|
||||||
});
|
} else {
|
||||||
|
dialogOptions.size = 'small';
|
||||||
|
}
|
||||||
|
|
||||||
|
const dlg = dialogHelper.createDialog(dialogOptions);
|
||||||
|
dlg.classList.add('formDialog');
|
||||||
|
let html = '';
|
||||||
|
html += '<div class="formDialogHeader">';
|
||||||
|
html += '<button is="paper-icon-button-light" class="btnCancel autoSize" tabindex="-1"><span class="material-icons arrow_back"></span></button>';
|
||||||
|
html += '<h3 class="formDialogHeaderTitle">';
|
||||||
|
html += globalize.translate('HeaderLiveTvTunerSetup');
|
||||||
|
html += '</h3>';
|
||||||
|
html += '</div>';
|
||||||
|
html += getEditorHtml();
|
||||||
|
dlg.innerHTML = html;
|
||||||
|
dlg.querySelector('.btnCancel').addEventListener('click', function () {
|
||||||
|
dialogHelper.close(dlg);
|
||||||
|
});
|
||||||
|
let deviceResult;
|
||||||
|
dlg.querySelector('.results').addEventListener('click', function (e) {
|
||||||
|
const tunerCard = dom.parentWithClass(e.target, 'card');
|
||||||
|
|
||||||
|
if (tunerCard) {
|
||||||
|
const deviceId = tunerCard.getAttribute('data-id');
|
||||||
|
deviceResult = currentDevices.filter(function (d) {
|
||||||
|
return d.DeviceId === deviceId;
|
||||||
|
})[0];
|
||||||
|
dialogHelper.close(dlg);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
if (layoutManager.tv) {
|
||||||
|
scrollHelper.centerFocus.on(dlg.querySelector('.formDialogContent'), false);
|
||||||
|
}
|
||||||
|
|
||||||
|
const apiClient = connectionManager.getApiClient(options.serverId);
|
||||||
|
discoverDevices(dlg, apiClient);
|
||||||
|
|
||||||
|
if (layoutManager.tv) {
|
||||||
|
scrollHelper.centerFocus.off(dlg.querySelector('.formDialogContent'), false);
|
||||||
|
}
|
||||||
|
|
||||||
|
return dialogHelper.open(dlg).then(function () {
|
||||||
|
if (deviceResult) {
|
||||||
|
return Promise.resolve(deviceResult);
|
||||||
|
}
|
||||||
|
|
||||||
|
return Promise.reject();
|
||||||
|
});
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
let currentDevices = [];
|
||||||
|
|
||||||
|
export default tunerPicker;
|
||||||
|
|
|
@ -27,7 +27,7 @@
|
||||||
<input is="emby-input" class="txtPass" label="${LabelPassword}" required="required" autocomplete="off" type="password" />
|
<input is="emby-input" class="txtPass" label="${LabelPassword}" required="required" autocomplete="off" type="password" />
|
||||||
</div>
|
</div>
|
||||||
<div>
|
<div>
|
||||||
<button is="emby-button" type="submit" class="raised button-submit block"><span>${ButtonSave}</span></button>
|
<button is="emby-button" type="submit" class="raised button-submit block"><span>${Save}</span></button>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</form>
|
</form>
|
||||||
|
@ -65,7 +65,7 @@
|
||||||
</div>
|
</div>
|
||||||
<br />
|
<br />
|
||||||
<div>
|
<div>
|
||||||
<button is="emby-button" type="submit" class="raised button-submit block btnSubmitListingsContainer btnSubmitListings hide"><span>${ButtonSave}</span></button>
|
<button is="emby-button" type="submit" class="raised button-submit block btnSubmitListingsContainer btnSubmitListings hide"><span>${Save}</span></button>
|
||||||
<button is="emby-button" type="button" class="raised button-cancel block btnCancel hide" onclick="history.back();"><span>${ButtonCancel}</span></button>
|
<button is="emby-button" type="button" class="raised button-cancel block btnCancel hide" onclick="history.back();"><span>${ButtonCancel}</span></button>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -56,7 +56,7 @@
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div>
|
<div>
|
||||||
<button is="emby-button" type="submit" class="raised button-submit block btnSubmitListings hide"><span>${ButtonSave}</span></button>
|
<button is="emby-button" type="submit" class="raised button-submit block btnSubmitListings hide"><span>${Save}</span></button>
|
||||||
<button is="emby-button" type="button" class="raised button-cancel block btnCancel hide" onclick="history.back();"><span>${ButtonCancel}</span></button>
|
<button is="emby-button" type="button" class="raised button-cancel block btnCancel hide" onclick="history.back();"><span>${ButtonCancel}</span></button>
|
||||||
</div>
|
</div>
|
||||||
</form>
|
</form>
|
||||||
|
|
|
@ -1,214 +1,219 @@
|
||||||
define(['connectionManager', 'globalize', 'dom', 'itemHelper', 'paper-icon-button-light', 'material-icons', 'emby-button', 'css!./userdatabuttons'], function (connectionManager, globalize, dom, itemHelper) {
|
import connectionManager from 'connectionManager';
|
||||||
'use strict';
|
import globalize from 'globalize';
|
||||||
|
import dom from 'dom';
|
||||||
|
import itemHelper from 'itemHelper';
|
||||||
|
import 'paper-icon-button-light';
|
||||||
|
import 'material-icons';
|
||||||
|
import 'emby-button';
|
||||||
|
import 'css!./userdatabuttons';
|
||||||
|
|
||||||
var userDataMethods = {
|
const userDataMethods = {
|
||||||
markPlayed: markPlayed,
|
markPlayed: markPlayed,
|
||||||
markDislike: markDislike,
|
markDislike: markDislike,
|
||||||
markLike: markLike,
|
markLike: markLike,
|
||||||
markFavorite: markFavorite
|
markFavorite: markFavorite
|
||||||
};
|
};
|
||||||
|
|
||||||
function getUserDataButtonHtml(method, itemId, serverId, buttonCssClass, iconCssClass, icon, tooltip, style) {
|
function getUserDataButtonHtml(method, itemId, serverId, buttonCssClass, iconCssClass, icon, tooltip, style) {
|
||||||
if (style === 'fab-mini') {
|
if (style === 'fab-mini') {
|
||||||
style = 'fab';
|
style = 'fab';
|
||||||
buttonCssClass = buttonCssClass ? (buttonCssClass + ' mini') : 'mini';
|
buttonCssClass = buttonCssClass ? (buttonCssClass + ' mini') : 'mini';
|
||||||
}
|
|
||||||
|
|
||||||
var is = style === 'fab' ? 'emby-button' : 'paper-icon-button-light';
|
|
||||||
var className = style === 'fab' ? 'autoSize fab' : 'autoSize';
|
|
||||||
|
|
||||||
if (buttonCssClass) {
|
|
||||||
className += ' ' + buttonCssClass;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (iconCssClass) {
|
|
||||||
iconCssClass += ' ';
|
|
||||||
} else {
|
|
||||||
iconCssClass = '';
|
|
||||||
}
|
|
||||||
|
|
||||||
iconCssClass += 'material-icons';
|
|
||||||
|
|
||||||
return '<button title="' + tooltip + '" data-itemid="' + itemId + '" data-serverid="' + serverId + '" is="' + is + '" data-method="' + method + '" class="' + className + '"><span class="' + iconCssClass + ' ' + icon + '"></span></button>';
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function onContainerClick(e) {
|
const is = style === 'fab' ? 'emby-button' : 'paper-icon-button-light';
|
||||||
var btnUserData = dom.parentWithClass(e.target, 'btnUserData');
|
let className = style === 'fab' ? 'autoSize fab' : 'autoSize';
|
||||||
|
|
||||||
if (!btnUserData) {
|
if (buttonCssClass) {
|
||||||
return;
|
className += ' ' + buttonCssClass;
|
||||||
}
|
|
||||||
|
|
||||||
var method = btnUserData.getAttribute('data-method');
|
|
||||||
userDataMethods[method](btnUserData);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function fill(options) {
|
if (iconCssClass) {
|
||||||
var html = getIconsHtml(options);
|
iconCssClass += ' ';
|
||||||
|
} else {
|
||||||
if (options.fillMode === 'insertAdjacent') {
|
iconCssClass = '';
|
||||||
options.element.insertAdjacentHTML(options.insertLocation || 'beforeend', html);
|
|
||||||
} else {
|
|
||||||
options.element.innerHTML = html;
|
|
||||||
}
|
|
||||||
|
|
||||||
dom.removeEventListener(options.element, 'click', onContainerClick, {
|
|
||||||
passive: true
|
|
||||||
});
|
|
||||||
|
|
||||||
dom.addEventListener(options.element, 'click', onContainerClick, {
|
|
||||||
passive: true
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function destroy(options) {
|
iconCssClass += 'material-icons';
|
||||||
options.element.innerHTML = '';
|
|
||||||
|
|
||||||
dom.removeEventListener(options.element, 'click', onContainerClick, {
|
return '<button title="' + tooltip + '" data-itemid="' + itemId + '" data-serverid="' + serverId + '" is="' + is + '" data-method="' + method + '" class="' + className + '"><span class="' + iconCssClass + ' ' + icon + '"></span></button>';
|
||||||
passive: true
|
}
|
||||||
});
|
|
||||||
|
function onContainerClick(e) {
|
||||||
|
const btnUserData = dom.parentWithClass(e.target, 'btnUserData');
|
||||||
|
|
||||||
|
if (!btnUserData) {
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
function getIconsHtml(options) {
|
const method = btnUserData.getAttribute('data-method');
|
||||||
var item = options.item;
|
userDataMethods[method](btnUserData);
|
||||||
var includePlayed = options.includePlayed;
|
}
|
||||||
var cssClass = options.cssClass;
|
|
||||||
var style = options.style;
|
|
||||||
|
|
||||||
var html = '';
|
function fill(options) {
|
||||||
|
const html = getIconsHtml(options);
|
||||||
|
|
||||||
var userData = item.UserData || {};
|
if (options.fillMode === 'insertAdjacent') {
|
||||||
|
options.element.insertAdjacentHTML(options.insertLocation || 'beforeend', html);
|
||||||
|
} else {
|
||||||
|
options.element.innerHTML = html;
|
||||||
|
}
|
||||||
|
|
||||||
var itemId = item.Id;
|
dom.removeEventListener(options.element, 'click', onContainerClick, {
|
||||||
|
passive: true
|
||||||
|
});
|
||||||
|
|
||||||
if (itemHelper.isLocalItem(item)) {
|
dom.addEventListener(options.element, 'click', onContainerClick, {
|
||||||
return html;
|
passive: true
|
||||||
}
|
});
|
||||||
|
}
|
||||||
|
|
||||||
var btnCssClass = 'btnUserData';
|
function destroy(options) {
|
||||||
|
options.element.innerHTML = '';
|
||||||
|
|
||||||
if (cssClass) {
|
dom.removeEventListener(options.element, 'click', onContainerClick, {
|
||||||
btnCssClass += ' ' + cssClass;
|
passive: true
|
||||||
}
|
});
|
||||||
|
}
|
||||||
|
|
||||||
var iconCssClass = options.iconCssClass;
|
function getIconsHtml(options) {
|
||||||
|
const item = options.item;
|
||||||
|
const includePlayed = options.includePlayed;
|
||||||
|
const cssClass = options.cssClass;
|
||||||
|
const style = options.style;
|
||||||
|
|
||||||
var serverId = item.ServerId;
|
let html = '';
|
||||||
|
|
||||||
if (includePlayed !== false) {
|
const userData = item.UserData || {};
|
||||||
var tooltipPlayed = globalize.translate('MarkPlayed');
|
|
||||||
|
|
||||||
if (itemHelper.canMarkPlayed(item)) {
|
const itemId = item.Id;
|
||||||
if (userData.Played) {
|
|
||||||
html += getUserDataButtonHtml('markPlayed', itemId, serverId, btnCssClass + ' btnUserDataOn', iconCssClass, 'check', tooltipPlayed, style);
|
|
||||||
} else {
|
|
||||||
html += getUserDataButtonHtml('markPlayed', itemId, serverId, btnCssClass, iconCssClass, 'check', tooltipPlayed, style);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
var tooltipFavorite = globalize.translate('Favorite');
|
|
||||||
if (userData.IsFavorite) {
|
|
||||||
html += getUserDataButtonHtml('markFavorite', itemId, serverId, btnCssClass + ' btnUserData btnUserDataOn', iconCssClass, 'favorite', tooltipFavorite, style);
|
|
||||||
} else {
|
|
||||||
html += getUserDataButtonHtml('markFavorite', itemId, serverId, btnCssClass + ' btnUserData', iconCssClass, 'favorite', tooltipFavorite, style);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
if (itemHelper.isLocalItem(item)) {
|
||||||
return html;
|
return html;
|
||||||
}
|
}
|
||||||
|
|
||||||
function markFavorite(link) {
|
let btnCssClass = 'btnUserData';
|
||||||
var id = link.getAttribute('data-itemid');
|
|
||||||
var serverId = link.getAttribute('data-serverid');
|
|
||||||
|
|
||||||
var markAsFavorite = !link.classList.contains('btnUserDataOn');
|
if (cssClass) {
|
||||||
|
btnCssClass += ' ' + cssClass;
|
||||||
|
}
|
||||||
|
|
||||||
favorite(id, serverId, markAsFavorite);
|
const iconCssClass = options.iconCssClass;
|
||||||
|
|
||||||
if (markAsFavorite) {
|
const serverId = item.ServerId;
|
||||||
link.classList.add('btnUserDataOn');
|
|
||||||
} else {
|
if (includePlayed !== false) {
|
||||||
link.classList.remove('btnUserDataOn');
|
const tooltipPlayed = globalize.translate('MarkPlayed');
|
||||||
|
|
||||||
|
if (itemHelper.canMarkPlayed(item)) {
|
||||||
|
if (userData.Played) {
|
||||||
|
html += getUserDataButtonHtml('markPlayed', itemId, serverId, btnCssClass + ' btnUserDataOn', iconCssClass, 'check', tooltipPlayed, style);
|
||||||
|
} else {
|
||||||
|
html += getUserDataButtonHtml('markPlayed', itemId, serverId, btnCssClass, iconCssClass, 'check', tooltipPlayed, style);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function markLike(link) {
|
const tooltipFavorite = globalize.translate('Favorite');
|
||||||
var id = link.getAttribute('data-itemid');
|
if (userData.IsFavorite) {
|
||||||
var serverId = link.getAttribute('data-serverid');
|
html += getUserDataButtonHtml('markFavorite', itemId, serverId, btnCssClass + ' btnUserData btnUserDataOn', iconCssClass, 'favorite', tooltipFavorite, style);
|
||||||
|
} else {
|
||||||
if (!link.classList.contains('btnUserDataOn')) {
|
html += getUserDataButtonHtml('markFavorite', itemId, serverId, btnCssClass + ' btnUserData', iconCssClass, 'favorite', tooltipFavorite, style);
|
||||||
likes(id, serverId, true);
|
|
||||||
|
|
||||||
link.classList.add('btnUserDataOn');
|
|
||||||
} else {
|
|
||||||
clearLike(id, serverId);
|
|
||||||
|
|
||||||
link.classList.remove('btnUserDataOn');
|
|
||||||
}
|
|
||||||
|
|
||||||
link.parentNode.querySelector('.btnDislike').classList.remove('btnUserDataOn');
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function markDislike(link) {
|
return html;
|
||||||
var id = link.getAttribute('data-itemid');
|
}
|
||||||
var serverId = link.getAttribute('data-serverid');
|
|
||||||
|
|
||||||
if (!link.classList.contains('btnUserDataOn')) {
|
function markFavorite(link) {
|
||||||
likes(id, serverId, false);
|
const id = link.getAttribute('data-itemid');
|
||||||
|
const serverId = link.getAttribute('data-serverid');
|
||||||
|
|
||||||
link.classList.add('btnUserDataOn');
|
const markAsFavorite = !link.classList.contains('btnUserDataOn');
|
||||||
} else {
|
|
||||||
clearLike(id, serverId);
|
|
||||||
|
|
||||||
link.classList.remove('btnUserDataOn');
|
favorite(id, serverId, markAsFavorite);
|
||||||
}
|
|
||||||
|
|
||||||
link.parentNode.querySelector('.btnLike').classList.remove('btnUserDataOn');
|
if (markAsFavorite) {
|
||||||
|
link.classList.add('btnUserDataOn');
|
||||||
|
} else {
|
||||||
|
link.classList.remove('btnUserDataOn');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function markLike(link) {
|
||||||
|
const id = link.getAttribute('data-itemid');
|
||||||
|
const serverId = link.getAttribute('data-serverid');
|
||||||
|
|
||||||
|
if (!link.classList.contains('btnUserDataOn')) {
|
||||||
|
likes(id, serverId, true);
|
||||||
|
|
||||||
|
link.classList.add('btnUserDataOn');
|
||||||
|
} else {
|
||||||
|
clearLike(id, serverId);
|
||||||
|
|
||||||
|
link.classList.remove('btnUserDataOn');
|
||||||
}
|
}
|
||||||
|
|
||||||
function markPlayed(link) {
|
link.parentNode.querySelector('.btnDislike').classList.remove('btnUserDataOn');
|
||||||
var id = link.getAttribute('data-itemid');
|
}
|
||||||
var serverId = link.getAttribute('data-serverid');
|
|
||||||
|
|
||||||
if (!link.classList.contains('btnUserDataOn')) {
|
function markDislike(link) {
|
||||||
played(id, serverId, true);
|
const id = link.getAttribute('data-itemid');
|
||||||
|
const serverId = link.getAttribute('data-serverid');
|
||||||
|
|
||||||
link.classList.add('btnUserDataOn');
|
if (!link.classList.contains('btnUserDataOn')) {
|
||||||
} else {
|
likes(id, serverId, false);
|
||||||
played(id, serverId, false);
|
|
||||||
|
|
||||||
link.classList.remove('btnUserDataOn');
|
link.classList.add('btnUserDataOn');
|
||||||
}
|
} else {
|
||||||
|
clearLike(id, serverId);
|
||||||
|
|
||||||
|
link.classList.remove('btnUserDataOn');
|
||||||
}
|
}
|
||||||
|
|
||||||
function likes(id, serverId, isLiked) {
|
link.parentNode.querySelector('.btnLike').classList.remove('btnUserDataOn');
|
||||||
var apiClient = connectionManager.getApiClient(serverId);
|
}
|
||||||
return apiClient.updateUserItemRating(apiClient.getCurrentUserId(), id, isLiked);
|
|
||||||
|
function markPlayed(link) {
|
||||||
|
const id = link.getAttribute('data-itemid');
|
||||||
|
const serverId = link.getAttribute('data-serverid');
|
||||||
|
|
||||||
|
if (!link.classList.contains('btnUserDataOn')) {
|
||||||
|
played(id, serverId, true);
|
||||||
|
|
||||||
|
link.classList.add('btnUserDataOn');
|
||||||
|
} else {
|
||||||
|
played(id, serverId, false);
|
||||||
|
|
||||||
|
link.classList.remove('btnUserDataOn');
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
function played(id, serverId, isPlayed) {
|
function likes(id, serverId, isLiked) {
|
||||||
var apiClient = connectionManager.getApiClient(serverId);
|
const apiClient = connectionManager.getApiClient(serverId);
|
||||||
|
return apiClient.updateUserItemRating(apiClient.getCurrentUserId(), id, isLiked);
|
||||||
|
}
|
||||||
|
|
||||||
var method = isPlayed ? 'markPlayed' : 'markUnplayed';
|
function played(id, serverId, isPlayed) {
|
||||||
|
const apiClient = connectionManager.getApiClient(serverId);
|
||||||
|
|
||||||
return apiClient[method](apiClient.getCurrentUserId(), id, new Date());
|
const method = isPlayed ? 'markPlayed' : 'markUnplayed';
|
||||||
}
|
|
||||||
|
|
||||||
function favorite(id, serverId, isFavorite) {
|
return apiClient[method](apiClient.getCurrentUserId(), id, new Date());
|
||||||
var apiClient = connectionManager.getApiClient(serverId);
|
}
|
||||||
|
|
||||||
return apiClient.updateFavoriteStatus(apiClient.getCurrentUserId(), id, isFavorite);
|
function favorite(id, serverId, isFavorite) {
|
||||||
}
|
const apiClient = connectionManager.getApiClient(serverId);
|
||||||
|
|
||||||
function clearLike(id, serverId) {
|
return apiClient.updateFavoriteStatus(apiClient.getCurrentUserId(), id, isFavorite);
|
||||||
var apiClient = connectionManager.getApiClient(serverId);
|
}
|
||||||
|
|
||||||
return apiClient.clearUserItemRating(apiClient.getCurrentUserId(), id);
|
function clearLike(id, serverId) {
|
||||||
}
|
const apiClient = connectionManager.getApiClient(serverId);
|
||||||
|
|
||||||
return {
|
return apiClient.clearUserItemRating(apiClient.getCurrentUserId(), id);
|
||||||
fill: fill,
|
}
|
||||||
destroy: destroy,
|
|
||||||
getIconsHtml: getIconsHtml
|
export default {
|
||||||
};
|
fill: fill,
|
||||||
});
|
destroy: destroy,
|
||||||
|
getIconsHtml: getIconsHtml
|
||||||
|
};
|
||||||
|
|
|
@ -1,60 +1,66 @@
|
||||||
define(['require', 'dialogHelper', 'loading', 'apphost', 'layoutManager', 'connectionManager', 'appRouter', 'globalize', 'userSettings', 'emby-checkbox', 'emby-input', 'paper-icon-button-light', 'emby-select', 'material-icons', 'css!./../formdialog', 'emby-button', 'flexStyles'], function (require, dialogHelper, loading, appHost, layoutManager, connectionManager, appRouter, globalize, userSettings) {
|
import dialogHelper from 'dialogHelper';
|
||||||
'use strict';
|
import layoutManager from 'layoutManager';
|
||||||
|
import globalize from 'globalize';
|
||||||
|
import * as userSettings from 'userSettings';
|
||||||
|
import 'emby-checkbox';
|
||||||
|
import 'emby-input';
|
||||||
|
import 'paper-icon-button-light';
|
||||||
|
import 'emby-select';
|
||||||
|
import 'material-icons';
|
||||||
|
import 'css!./../formdialog';
|
||||||
|
import 'emby-button';
|
||||||
|
import 'flexStyles';
|
||||||
|
|
||||||
layoutManager = layoutManager.default || layoutManager;
|
function onSubmit(e) {
|
||||||
|
e.preventDefault();
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
function onSubmit(e) {
|
function initEditor(context, settings) {
|
||||||
e.preventDefault();
|
context.querySelector('form').addEventListener('submit', onSubmit);
|
||||||
return false;
|
|
||||||
|
const elems = context.querySelectorAll('.viewSetting-checkboxContainer');
|
||||||
|
|
||||||
|
for (const elem of elems) {
|
||||||
|
elem.querySelector('input').checked = settings[elem.getAttribute('data-settingname')] || false;
|
||||||
}
|
}
|
||||||
|
|
||||||
function initEditor(context, settings) {
|
context.querySelector('.selectImageType').value = settings.imageType || 'primary';
|
||||||
context.querySelector('form').addEventListener('submit', onSubmit);
|
}
|
||||||
|
|
||||||
var elems = context.querySelectorAll('.viewSetting-checkboxContainer');
|
function saveValues(context, settings, settingsKey) {
|
||||||
|
const elems = context.querySelectorAll('.viewSetting-checkboxContainer');
|
||||||
for (var i = 0, length = elems.length; i < length; i++) {
|
for (const elem of elems) {
|
||||||
elems[i].querySelector('input').checked = settings[elems[i].getAttribute('data-settingname')] || false;
|
userSettings.set(settingsKey + '-' + elem.getAttribute('data-settingname'), elem.querySelector('input').checked);
|
||||||
}
|
|
||||||
|
|
||||||
context.querySelector('.selectImageType').value = settings.imageType || 'primary';
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function saveValues(context, settings, settingsKey) {
|
userSettings.set(settingsKey + '-imageType', context.querySelector('.selectImageType').value);
|
||||||
var elems = context.querySelectorAll('.viewSetting-checkboxContainer');
|
}
|
||||||
for (var i = 0, length = elems.length; i < length; i++) {
|
|
||||||
userSettings.set(settingsKey + '-' + elems[i].getAttribute('data-settingname'), elems[i].querySelector('input').checked);
|
|
||||||
}
|
|
||||||
|
|
||||||
userSettings.set(settingsKey + '-imageType', context.querySelector('.selectImageType').value);
|
function centerFocus(elem, horiz, on) {
|
||||||
|
import('scrollHelper').then(({default: scrollHelper}) => {
|
||||||
|
const fn = on ? 'on' : 'off';
|
||||||
|
scrollHelper.centerFocus[fn](elem, horiz);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function showIfAllowed(context, selector, visible) {
|
||||||
|
const elem = context.querySelector(selector);
|
||||||
|
|
||||||
|
if (visible && !elem.classList.contains('hiddenFromViewSettings')) {
|
||||||
|
elem.classList.remove('hide');
|
||||||
|
} else {
|
||||||
|
elem.classList.add('hide');
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
function centerFocus(elem, horiz, on) {
|
class ViewSettings {
|
||||||
require(['scrollHelper'], function (scrollHelper) {
|
constructor() {
|
||||||
scrollHelper = scrollHelper.default || scrollHelper;
|
|
||||||
var fn = on ? 'on' : 'off';
|
|
||||||
scrollHelper.centerFocus[fn](elem, horiz);
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
show(options) {
|
||||||
function showIfAllowed(context, selector, visible) {
|
|
||||||
var elem = context.querySelector(selector);
|
|
||||||
|
|
||||||
if (visible && !elem.classList.contains('hiddenFromViewSettings')) {
|
|
||||||
elem.classList.remove('hide');
|
|
||||||
} else {
|
|
||||||
elem.classList.add('hide');
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function ViewSettings() {
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
ViewSettings.prototype.show = function (options) {
|
|
||||||
return new Promise(function (resolve, reject) {
|
return new Promise(function (resolve, reject) {
|
||||||
require(['text!./viewSettings.template.html'], function (template) {
|
import('text!./viewSettings.template.html').then(({default: template}) => {
|
||||||
var dialogOptions = {
|
const dialogOptions = {
|
||||||
removeOnClose: true,
|
removeOnClose: true,
|
||||||
scrollY: false
|
scrollY: false
|
||||||
};
|
};
|
||||||
|
@ -65,11 +71,11 @@ define(['require', 'dialogHelper', 'loading', 'apphost', 'layoutManager', 'conne
|
||||||
dialogOptions.size = 'small';
|
dialogOptions.size = 'small';
|
||||||
}
|
}
|
||||||
|
|
||||||
var dlg = dialogHelper.createDialog(dialogOptions);
|
const dlg = dialogHelper.createDialog(dialogOptions);
|
||||||
|
|
||||||
dlg.classList.add('formDialog');
|
dlg.classList.add('formDialog');
|
||||||
|
|
||||||
var html = '';
|
let html = '';
|
||||||
|
|
||||||
html += '<div class="formDialogHeader">';
|
html += '<div class="formDialogHeader">';
|
||||||
html += '<button is="paper-icon-button-light" class="btnCancel hide-mouse-idle-tv" tabindex="-1"><span class="material-icons arrow_back"></span></button>';
|
html += '<button is="paper-icon-button-light" class="btnCancel hide-mouse-idle-tv" tabindex="-1"><span class="material-icons arrow_back"></span></button>';
|
||||||
|
@ -81,14 +87,14 @@ define(['require', 'dialogHelper', 'loading', 'apphost', 'layoutManager', 'conne
|
||||||
|
|
||||||
dlg.innerHTML = globalize.translateHtml(html, 'core');
|
dlg.innerHTML = globalize.translateHtml(html, 'core');
|
||||||
|
|
||||||
var settingElements = dlg.querySelectorAll('.viewSetting');
|
const settingElements = dlg.querySelectorAll('.viewSetting');
|
||||||
for (var i = 0, length = settingElements.length; i < length; i++) {
|
for (const settingElement of settingElements) {
|
||||||
if (options.visibleSettings.indexOf(settingElements[i].getAttribute('data-settingname')) === -1) {
|
if (options.visibleSettings.indexOf(settingElement.getAttribute('data-settingname')) === -1) {
|
||||||
settingElements[i].classList.add('hide');
|
settingElement.classList.add('hide');
|
||||||
settingElements[i].classList.add('hiddenFromViewSettings');
|
settingElement.classList.add('hiddenFromViewSettings');
|
||||||
} else {
|
} else {
|
||||||
settingElements[i].classList.remove('hide');
|
settingElement.classList.remove('hide');
|
||||||
settingElements[i].classList.remove('hiddenFromViewSettings');
|
settingElement.classList.remove('hiddenFromViewSettings');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -107,7 +113,7 @@ define(['require', 'dialogHelper', 'loading', 'apphost', 'layoutManager', 'conne
|
||||||
centerFocus(dlg.querySelector('.formDialogContent'), false, true);
|
centerFocus(dlg.querySelector('.formDialogContent'), false, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
var submitted;
|
let submitted;
|
||||||
|
|
||||||
dlg.querySelector('.selectImageType').dispatchEvent(new CustomEvent('change', {}));
|
dlg.querySelector('.selectImageType').dispatchEvent(new CustomEvent('change', {}));
|
||||||
|
|
||||||
|
@ -130,7 +136,7 @@ define(['require', 'dialogHelper', 'loading', 'apphost', 'layoutManager', 'conne
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
};
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return ViewSettings;
|
export default ViewSettings;
|
||||||
});
|
|
||||||
|
|
|
@ -3,7 +3,7 @@
|
||||||
<div class="content-primary">
|
<div class="content-primary">
|
||||||
<div class="detailSectionHeader">
|
<div class="detailSectionHeader">
|
||||||
<h2 style="margin:.6em 0;vertical-align:middle;display:inline-block;">${HeaderApiKeys}</h2>
|
<h2 style="margin:.6em 0;vertical-align:middle;display:inline-block;">${HeaderApiKeys}</h2>
|
||||||
<button is="emby-button" type="button" class="fab btnNewKey submit" style="margin-left:1em;" title="${ButtonAdd}">
|
<button is="emby-button" type="button" class="fab btnNewKey submit" style="margin-left:1em;" title="${Add}">
|
||||||
<span class="material-icons add" aria-hidden="true"></span>
|
<span class="material-icons add" aria-hidden="true"></span>
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -24,7 +24,7 @@ import 'emby-itemscontainer';
|
||||||
function showPlaybackInfo(btn, session) {
|
function showPlaybackInfo(btn, session) {
|
||||||
import('alert').then(({default: alert}) => {
|
import('alert').then(({default: alert}) => {
|
||||||
let title;
|
let title;
|
||||||
let text = [];
|
const text = [];
|
||||||
const displayPlayMethod = playMethodHelper.getDisplayPlayMethod(session);
|
const displayPlayMethod = playMethodHelper.getDisplayPlayMethod(session);
|
||||||
|
|
||||||
if (displayPlayMethod === 'DirectStream') {
|
if (displayPlayMethod === 'DirectStream') {
|
||||||
|
|
|
@ -15,7 +15,7 @@
|
||||||
</div>
|
</div>
|
||||||
<div>
|
<div>
|
||||||
<button is="emby-button" type="submit" class="raised button-submit block">
|
<button is="emby-button" type="submit" class="raised button-submit block">
|
||||||
<span>${ButtonSave}</span>
|
<span>${Save}</span>
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
</form>
|
</form>
|
||||||
|
|
|
@ -38,7 +38,7 @@ import 'cardStyle';
|
||||||
}
|
}
|
||||||
|
|
||||||
function showDeviceMenu(view, btn, deviceId) {
|
function showDeviceMenu(view, btn, deviceId) {
|
||||||
let menuItems = [];
|
const menuItems = [];
|
||||||
|
|
||||||
if (canEdit) {
|
if (canEdit) {
|
||||||
menuItems.push({
|
menuItems.push({
|
||||||
|
|
|
@ -10,7 +10,7 @@
|
||||||
<div data-role="controlgroup" data-type="horizontal" data-mini="true">
|
<div data-role="controlgroup" data-type="horizontal" data-mini="true">
|
||||||
<a href="#" is="emby-linkbutton" data-role="button" class="radioTabButton" id="radioInfo" data-value="tabInfo">${TabInfo}</a>
|
<a href="#" is="emby-linkbutton" data-role="button" class="radioTabButton" id="radioInfo" data-value="tabInfo">${TabInfo}</a>
|
||||||
<a href="#" is="emby-linkbutton" data-role="button" class="radioTabButton" id="radioDirectPlay" data-value="tabDirectPlayProfiles">${TabDirectPlay}</a>
|
<a href="#" is="emby-linkbutton" data-role="button" class="radioTabButton" id="radioDirectPlay" data-value="tabDirectPlayProfiles">${TabDirectPlay}</a>
|
||||||
<a href="#" is="emby-linkbutton" data-role="button" class="radioTabButton" id="radioTranscoding" data-value="tabTranscodingProfiles">${TabTranscoding}</a>
|
<a href="#" is="emby-linkbutton" data-role="button" class="radioTabButton" id="radioTranscoding" data-value="tabTranscodingProfiles">${Transcoding}</a>
|
||||||
<a href="#" is="emby-linkbutton" data-role="button" class="radioTabButton" id="radioContainers" data-value="tabContainerProfiles">${TabContainers}</a>
|
<a href="#" is="emby-linkbutton" data-role="button" class="radioTabButton" id="radioContainers" data-value="tabContainerProfiles">${TabContainers}</a>
|
||||||
<a href="#" is="emby-linkbutton" data-role="button" class="radioTabButton" id="radioCodecs" data-value="tabCodecProfiles">${TabCodecs}</a>
|
<a href="#" is="emby-linkbutton" data-role="button" class="radioTabButton" id="radioCodecs" data-value="tabCodecProfiles">${TabCodecs}</a>
|
||||||
<a href="#" is="emby-linkbutton" data-role="button" class="radioTabButton" id="radioMediaProfiles" data-value="tabMediaProfiles">${TabResponses}</a>
|
<a href="#" is="emby-linkbutton" data-role="button" class="radioTabButton" id="radioMediaProfiles" data-value="tabMediaProfiles">${TabResponses}</a>
|
||||||
|
@ -96,14 +96,14 @@
|
||||||
</div>
|
</div>
|
||||||
<div>
|
<div>
|
||||||
<h2 style="vertical-align:middle;display:inline-block;">${HeaderHttpHeaders}</h2>
|
<h2 style="vertical-align:middle;display:inline-block;">${HeaderHttpHeaders}</h2>
|
||||||
<button is="emby-button" type="button" class="fab btnAddIdentificationHttpHeader submit sectionTitleButton" style="margin-left:1em;" title="${ButtonAdd}">
|
<button is="emby-button" type="button" class="fab btnAddIdentificationHttpHeader submit sectionTitleButton" style="margin-left:1em;" title="${Add}">
|
||||||
<span class="material-icons add"></span>
|
<span class="material-icons add"></span>
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
<div class="httpHeaderIdentificationList"></div>
|
<div class="httpHeaderIdentificationList"></div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div is="emby-collapse" title="${HeaderDisplay}">
|
<div is="emby-collapse" title="${Display}">
|
||||||
<div class="collapseContent">
|
<div class="collapseContent">
|
||||||
<br />
|
<br />
|
||||||
<div class="checkboxContainer checkboxContainer-withDescription">
|
<div class="checkboxContainer checkboxContainer-withDescription">
|
||||||
|
@ -209,7 +209,7 @@
|
||||||
<div class="collapseContent">
|
<div class="collapseContent">
|
||||||
<p>${HeaderSubtitleProfilesHelp}</p>
|
<p>${HeaderSubtitleProfilesHelp}</p>
|
||||||
<button is="emby-button" type="button" class="raised submit block btnAddSubtitleProfile">
|
<button is="emby-button" type="button" class="raised submit block btnAddSubtitleProfile">
|
||||||
<span>${ButtonAdd}</span>
|
<span>${Add}</span>
|
||||||
</button>
|
</button>
|
||||||
<div class="subtitleProfileList"></div>
|
<div class="subtitleProfileList"></div>
|
||||||
<br />
|
<br />
|
||||||
|
@ -219,7 +219,7 @@
|
||||||
<div class="collapseContent">
|
<div class="collapseContent">
|
||||||
<div>
|
<div>
|
||||||
<h2 style="vertical-align:middle;display:inline-block;">${HeaderXmlDocumentAttributes}</h2>
|
<h2 style="vertical-align:middle;display:inline-block;">${HeaderXmlDocumentAttributes}</h2>
|
||||||
<button is="emby-button" type="button" class="fab btnAddXmlDocumentAttribute submit sectionTitleButton" style="margin-left:1em;" title="${ButtonAdd}">
|
<button is="emby-button" type="button" class="fab btnAddXmlDocumentAttribute submit sectionTitleButton" style="margin-left:1em;" title="${Add}">
|
||||||
<span class="material-icons add"></span>
|
<span class="material-icons add"></span>
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
|
@ -262,7 +262,7 @@
|
||||||
<br />
|
<br />
|
||||||
<div>
|
<div>
|
||||||
<button is="emby-button" type="submit" class="raised button-submit block">
|
<button is="emby-button" type="submit" class="raised button-submit block">
|
||||||
<span>${ButtonSave}</span>
|
<span>${Save}</span>
|
||||||
</button>
|
</button>
|
||||||
<button is="emby-button" type="button" class="button-cancel raised block" onclick="Dashboard.navigate('dlnaprofiles.html');">
|
<button is="emby-button" type="button" class="button-cancel raised block" onclick="Dashboard.navigate('dlnaprofiles.html');">
|
||||||
<span>${ButtonCancel}</span>
|
<span>${ButtonCancel}</span>
|
||||||
|
|
|
@ -315,7 +315,7 @@ import 'listViewStyle';
|
||||||
let currentType;
|
let currentType;
|
||||||
|
|
||||||
for (let i = 0, length = profiles.length; i < length; i++) {
|
for (let i = 0, length = profiles.length; i < length; i++) {
|
||||||
let profile = profiles[i];
|
const profile = profiles[i];
|
||||||
|
|
||||||
if (profile.Type !== currentType) {
|
if (profile.Type !== currentType) {
|
||||||
html += '<li data-role="list-divider">' + profile.Type + '</li>';
|
html += '<li data-role="list-divider">' + profile.Type + '</li>';
|
||||||
|
@ -401,7 +401,7 @@ import 'listViewStyle';
|
||||||
let currentType;
|
let currentType;
|
||||||
|
|
||||||
for (let i = 0, length = profiles.length; i < length; i++) {
|
for (let i = 0, length = profiles.length; i < length; i++) {
|
||||||
let profile = profiles[i];
|
const profile = profiles[i];
|
||||||
|
|
||||||
if (profile.Type !== currentType) {
|
if (profile.Type !== currentType) {
|
||||||
html += '<li data-role="list-divider">' + profile.Type + '</li>';
|
html += '<li data-role="list-divider">' + profile.Type + '</li>';
|
||||||
|
@ -472,7 +472,7 @@ import 'listViewStyle';
|
||||||
let currentType;
|
let currentType;
|
||||||
|
|
||||||
for (let i = 0, length = profiles.length; i < length; i++) {
|
for (let i = 0, length = profiles.length; i < length; i++) {
|
||||||
let profile = profiles[i];
|
const profile = profiles[i];
|
||||||
const type = profile.Type.replace('VideoAudio', 'Video Audio');
|
const type = profile.Type.replace('VideoAudio', 'Video Audio');
|
||||||
|
|
||||||
if (type !== currentType) {
|
if (type !== currentType) {
|
||||||
|
@ -696,7 +696,7 @@ import 'listViewStyle';
|
||||||
let currentProfile;
|
let currentProfile;
|
||||||
let currentSubProfile;
|
let currentSubProfile;
|
||||||
let isSubProfileNew;
|
let isSubProfileNew;
|
||||||
const allText = globalize.translate('LabelAll');
|
const allText = globalize.translate('All');
|
||||||
|
|
||||||
$(document).on('pageinit', '#dlnaProfilePage', function () {
|
$(document).on('pageinit', '#dlnaProfilePage', function () {
|
||||||
const page = this;
|
const page = this;
|
||||||
|
|
|
@ -36,7 +36,7 @@ import 'emby-button';
|
||||||
}
|
}
|
||||||
|
|
||||||
for (let i = 0, length = profiles.length; i < length; i++) {
|
for (let i = 0, length = profiles.length; i < length; i++) {
|
||||||
let profile = profiles[i];
|
const profile = profiles[i];
|
||||||
html += '<div class="listItem listItem-border">';
|
html += '<div class="listItem listItem-border">';
|
||||||
html += '<span class="listItemIcon material-icons live_tv"></span>';
|
html += '<span class="listItemIcon material-icons live_tv"></span>';
|
||||||
html += '<div class="listItemBody two-line">';
|
html += '<div class="listItemBody two-line">';
|
||||||
|
|
|
@ -59,7 +59,7 @@
|
||||||
</div>
|
</div>
|
||||||
<div>
|
<div>
|
||||||
<button is="emby-button" type="submit" class="raised button-submit block">
|
<button is="emby-button" type="submit" class="raised button-submit block">
|
||||||
<span>${ButtonSave}</span>
|
<span>${Save}</span>
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
</form>
|
</form>
|
||||||
|
|
|
@ -4,7 +4,7 @@
|
||||||
<form class="encodingSettingsForm">
|
<form class="encodingSettingsForm">
|
||||||
<div class="verticalSection">
|
<div class="verticalSection">
|
||||||
<div class="sectionTitleContainer flex align-items-center">
|
<div class="sectionTitleContainer flex align-items-center">
|
||||||
<h2 class="sectionTitle">${TabTranscoding}</h2>
|
<h2 class="sectionTitle">${Transcoding}</h2>
|
||||||
<a is="emby-linkbutton" rel="noopener noreferrer" class="raised button-alt headerHelpButton" target="_blank" href="https://docs.jellyfin.org/general/server/transcoding.html">${Help}</a>
|
<a is="emby-linkbutton" rel="noopener noreferrer" class="raised button-alt headerHelpButton" target="_blank" href="https://docs.jellyfin.org/general/server/transcoding.html">${Help}</a>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
@ -184,7 +184,7 @@
|
||||||
|
|
||||||
<div>
|
<div>
|
||||||
<button is="emby-button" type="submit" class="raised button-submit block">
|
<button is="emby-button" type="submit" class="raised button-submit block">
|
||||||
<span>${ButtonSave}</span>
|
<span>${Save}</span>
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
</form>
|
</form>
|
||||||
|
|
|
@ -63,7 +63,7 @@
|
||||||
<br />
|
<br />
|
||||||
<div>
|
<div>
|
||||||
<button is="emby-button" type="submit" class="raised button-submit block">
|
<button is="emby-button" type="submit" class="raised button-submit block">
|
||||||
<span>${ButtonSave}</span>
|
<span>${Save}</span>
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
</form>
|
</form>
|
||||||
|
|
|
@ -363,10 +363,10 @@ import 'emby-itemrefreshindicator';
|
||||||
name: globalize.translate('HeaderLibraries')
|
name: globalize.translate('HeaderLibraries')
|
||||||
}, {
|
}, {
|
||||||
href: 'librarydisplay.html',
|
href: 'librarydisplay.html',
|
||||||
name: globalize.translate('TabDisplay')
|
name: globalize.translate('Display')
|
||||||
}, {
|
}, {
|
||||||
href: 'metadataimages.html',
|
href: 'metadataimages.html',
|
||||||
name: globalize.translate('TabMetadata')
|
name: globalize.translate('Metadata')
|
||||||
}, {
|
}, {
|
||||||
href: 'metadatanfo.html',
|
href: 'metadatanfo.html',
|
||||||
name: globalize.translate('TabNfoSettings')
|
name: globalize.translate('TabNfoSettings')
|
||||||
|
|
|
@ -49,7 +49,7 @@
|
||||||
|
|
||||||
<br/>
|
<br/>
|
||||||
<button is="emby-button" type="submit" class="raised button-submit block">
|
<button is="emby-button" type="submit" class="raised button-submit block">
|
||||||
<span>${ButtonSave}</span>
|
<span>${Save}</span>
|
||||||
</button>
|
</button>
|
||||||
</form>
|
</form>
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -12,10 +12,10 @@ import 'emby-button';
|
||||||
name: globalize.translate('HeaderLibraries')
|
name: globalize.translate('HeaderLibraries')
|
||||||
}, {
|
}, {
|
||||||
href: 'librarydisplay.html',
|
href: 'librarydisplay.html',
|
||||||
name: globalize.translate('TabDisplay')
|
name: globalize.translate('Display')
|
||||||
}, {
|
}, {
|
||||||
href: 'metadataimages.html',
|
href: 'metadataimages.html',
|
||||||
name: globalize.translate('TabMetadata')
|
name: globalize.translate('Metadata')
|
||||||
}, {
|
}, {
|
||||||
href: 'metadatanfo.html',
|
href: 'metadatanfo.html',
|
||||||
name: globalize.translate('TabNfoSettings')
|
name: globalize.translate('TabNfoSettings')
|
||||||
|
|
|
@ -55,10 +55,10 @@ import 'listViewStyle';
|
||||||
name: globalize.translate('HeaderLibraries')
|
name: globalize.translate('HeaderLibraries')
|
||||||
}, {
|
}, {
|
||||||
href: 'librarydisplay.html',
|
href: 'librarydisplay.html',
|
||||||
name: globalize.translate('TabDisplay')
|
name: globalize.translate('Display')
|
||||||
}, {
|
}, {
|
||||||
href: 'metadataimages.html',
|
href: 'metadataimages.html',
|
||||||
name: globalize.translate('TabMetadata')
|
name: globalize.translate('Metadata')
|
||||||
}, {
|
}, {
|
||||||
href: 'metadatanfo.html',
|
href: 'metadatanfo.html',
|
||||||
name: globalize.translate('TabNfoSettings')
|
name: globalize.translate('TabNfoSettings')
|
||||||
|
|
|
@ -17,7 +17,7 @@
|
||||||
<select is="emby-select" id="selectCountry" required="required" label="${LabelCountry}"></select>
|
<select is="emby-select" id="selectCountry" required="required" label="${LabelCountry}"></select>
|
||||||
</div>
|
</div>
|
||||||
<br />
|
<br />
|
||||||
<div><button is="emby-button" type="submit" class="raised button-submit block"><span>${ButtonSave}</span></button></div>
|
<div><button is="emby-button" type="submit" class="raised button-submit block"><span>${Save}</span></button></div>
|
||||||
</form>
|
</form>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
|
|
@ -41,7 +41,7 @@
|
||||||
</label>
|
</label>
|
||||||
<div class="fieldDescription checkboxFieldDescription">${LabelKodiMetadataEnableExtraThumbsHelp}</div>
|
<div class="fieldDescription checkboxFieldDescription">${LabelKodiMetadataEnableExtraThumbsHelp}</div>
|
||||||
</div>
|
</div>
|
||||||
<div><button is="emby-button" type="submit" class="raised button-submit block"><span>${ButtonSave}</span></button></div>
|
<div><button is="emby-button" type="submit" class="raised button-submit block"><span>${Save}</span></button></div>
|
||||||
</form>
|
</form>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
|
|
@ -52,10 +52,10 @@ import globalize from 'globalize';
|
||||||
name: globalize.translate('HeaderLibraries')
|
name: globalize.translate('HeaderLibraries')
|
||||||
}, {
|
}, {
|
||||||
href: 'librarydisplay.html',
|
href: 'librarydisplay.html',
|
||||||
name: globalize.translate('TabDisplay')
|
name: globalize.translate('Display')
|
||||||
}, {
|
}, {
|
||||||
href: 'metadataimages.html',
|
href: 'metadataimages.html',
|
||||||
name: globalize.translate('TabMetadata')
|
name: globalize.translate('Metadata')
|
||||||
}, {
|
}, {
|
||||||
href: 'metadatanfo.html',
|
href: 'metadatanfo.html',
|
||||||
name: globalize.translate('TabNfoSettings')
|
name: globalize.translate('TabNfoSettings')
|
||||||
|
|
|
@ -112,7 +112,7 @@
|
||||||
</div>
|
</div>
|
||||||
<div>
|
<div>
|
||||||
<button is="emby-button" type="submit" class="raised button-submit block">
|
<button is="emby-button" type="submit" class="raised button-submit block">
|
||||||
<span>${ButtonSave}</span>
|
<span>${Save}</span>
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
</form>
|
</form>
|
||||||
|
|
|
@ -54,7 +54,7 @@
|
||||||
<div>
|
<div>
|
||||||
<br />
|
<br />
|
||||||
<button is="emby-button" type="submit" class="raised button-submit block">
|
<button is="emby-button" type="submit" class="raised button-submit block">
|
||||||
<span>${ButtonSave}</span>
|
<span>${Save}</span>
|
||||||
</button>
|
</button>
|
||||||
|
|
||||||
<button is="emby-button" type="button" class="raised button-cancel block btnCancel" onclick="history.back();">
|
<button is="emby-button" type="button" class="raised button-cancel block btnCancel" onclick="history.back();">
|
||||||
|
|
|
@ -23,7 +23,7 @@
|
||||||
${LabelMinResumeDurationHelp}
|
${LabelMinResumeDurationHelp}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div><button is="emby-button" type="submit" class="raised button-submit block"><span>${ButtonSave}</span></button></div>
|
<div><button is="emby-button" type="submit" class="raised button-submit block"><span>${Save}</span></button></div>
|
||||||
</form>
|
</form>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -8,7 +8,7 @@ function populateHistory(packageInfo, page) {
|
||||||
const length = Math.min(packageInfo.versions.length, 10);
|
const length = Math.min(packageInfo.versions.length, 10);
|
||||||
|
|
||||||
for (let i = 0; i < length; i++) {
|
for (let i = 0; i < length; i++) {
|
||||||
let version = packageInfo.versions[i];
|
const version = packageInfo.versions[i];
|
||||||
html += '<h2 style="margin:.5em 0;">' + version.version + '</h2>';
|
html += '<h2 style="margin:.5em 0;">' + version.version + '</h2>';
|
||||||
html += '<div style="margin-bottom:1.5em;">' + version.changelog + '</div>';
|
html += '<div style="margin-bottom:1.5em;">' + version.changelog + '</div>';
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,7 +3,7 @@
|
||||||
<div class="content-primary">
|
<div class="content-primary">
|
||||||
<div class="sectionTitleContainer flex align-items-center">
|
<div class="sectionTitleContainer flex align-items-center">
|
||||||
<h2 class="sectionTitle">${TabRepositories}</h2>
|
<h2 class="sectionTitle">${TabRepositories}</h2>
|
||||||
<button is="emby-button" type="button" class="fab btnNewRepository submit" style="margin-left:1em;" title="${ButtonAdd}">
|
<button is="emby-button" type="button" class="fab btnNewRepository submit" style="margin-left:1em;" title="${Add}">
|
||||||
<span class="material-icons add" aria-hidden="true"></span>
|
<span class="material-icons add" aria-hidden="true"></span>
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -105,7 +105,7 @@ export default function(view, params) {
|
||||||
});
|
});
|
||||||
|
|
||||||
view.querySelector('.btnNewRepository').addEventListener('click', () => {
|
view.querySelector('.btnNewRepository').addEventListener('click', () => {
|
||||||
let dialog = dialogHelper.createDialog({
|
const dialog = dialogHelper.createDialog({
|
||||||
scrollY: false,
|
scrollY: false,
|
||||||
size: 'large',
|
size: 'large',
|
||||||
modal: false,
|
modal: false,
|
||||||
|
@ -127,7 +127,7 @@ export default function(view, params) {
|
||||||
html += `<input is="emby-input" id="txtRepositoryUrl" label="${globalize.translate('LabelRepositoryUrl')}" type="url" required />`;
|
html += `<input is="emby-input" id="txtRepositoryUrl" label="${globalize.translate('LabelRepositoryUrl')}" type="url" required />`;
|
||||||
html += `<div class="fieldDescription">${globalize.translate('LabelRepositoryUrlHelp')}</div>`;
|
html += `<div class="fieldDescription">${globalize.translate('LabelRepositoryUrlHelp')}</div>`;
|
||||||
html += '</div>';
|
html += '</div>';
|
||||||
html += `<button is="emby-button" type="submit" class="raised button-submit block"><span>${globalize.translate('ButtonSave')}</span></button>`;
|
html += `<button is="emby-button" type="submit" class="raised button-submit block"><span>${globalize.translate('Save')}</span></button>`;
|
||||||
html += '</div>';
|
html += '</div>';
|
||||||
html += '</form>';
|
html += '</form>';
|
||||||
|
|
||||||
|
|
|
@ -73,7 +73,7 @@
|
||||||
</div>
|
</div>
|
||||||
<div>
|
<div>
|
||||||
<button is="emby-button" type="submit" class="raised button-submit block" data-icon="check">
|
<button is="emby-button" type="submit" class="raised button-submit block" data-icon="check">
|
||||||
<span>${ButtonAdd}</span>
|
<span>${Add}</span>
|
||||||
</button>
|
</button>
|
||||||
<button is="emby-button" type="button" class="raised button-cancel block" data-icon="delete" onclick="$(this).parents('.dialog').addClass('hide');">
|
<button is="emby-button" type="button" class="raised button-cancel block" data-icon="delete" onclick="$(this).parents('.dialog').addClass('hide');">
|
||||||
<span>${ButtonCancel}</span>
|
<span>${ButtonCancel}</span>
|
||||||
|
|
|
@ -33,7 +33,7 @@ import 'emby-select';
|
||||||
const ScheduledTaskPage = {
|
const ScheduledTaskPage = {
|
||||||
refreshScheduledTask: function (view) {
|
refreshScheduledTask: function (view) {
|
||||||
loading.show();
|
loading.show();
|
||||||
let id = getParameterByName('id');
|
const id = getParameterByName('id');
|
||||||
ApiClient.getScheduledTask(id).then(function (task) {
|
ApiClient.getScheduledTask(id).then(function (task) {
|
||||||
ScheduledTaskPage.loadScheduledTask(view, task);
|
ScheduledTaskPage.loadScheduledTask(view, task);
|
||||||
});
|
});
|
||||||
|
@ -143,7 +143,7 @@ import 'emby-select';
|
||||||
},
|
},
|
||||||
deleteTrigger: function (view, index) {
|
deleteTrigger: function (view, index) {
|
||||||
loading.show();
|
loading.show();
|
||||||
let id = getParameterByName('id');
|
const id = getParameterByName('id');
|
||||||
ApiClient.getScheduledTask(id).then(function (task) {
|
ApiClient.getScheduledTask(id).then(function (task) {
|
||||||
task.Triggers.remove(index);
|
task.Triggers.remove(index);
|
||||||
ApiClient.updateScheduledTaskTriggers(task.Id, task.Triggers).then(function () {
|
ApiClient.updateScheduledTaskTriggers(task.Id, task.Triggers).then(function () {
|
||||||
|
@ -211,7 +211,7 @@ import 'emby-select';
|
||||||
export default function (view, params) {
|
export default function (view, params) {
|
||||||
function onSubmit(e) {
|
function onSubmit(e) {
|
||||||
loading.show();
|
loading.show();
|
||||||
let id = getParameterByName('id');
|
const id = getParameterByName('id');
|
||||||
ApiClient.getScheduledTask(id).then(function (task) {
|
ApiClient.getScheduledTask(id).then(function (task) {
|
||||||
task.Triggers.push(ScheduledTaskPage.getTriggerToAdd(view));
|
task.Triggers.push(ScheduledTaskPage.getTriggerToAdd(view));
|
||||||
ApiClient.updateScheduledTaskTriggers(task.Id, task.Triggers).then(function () {
|
ApiClient.updateScheduledTaskTriggers(task.Id, task.Triggers).then(function () {
|
||||||
|
|
|
@ -103,7 +103,7 @@ import 'emby-button';
|
||||||
}
|
}
|
||||||
|
|
||||||
function setTaskButtonIcon(button, icon) {
|
function setTaskButtonIcon(button, icon) {
|
||||||
let inner = button.querySelector('.material-icons');
|
const inner = button.querySelector('.material-icons');
|
||||||
inner.classList.remove('stop', 'play_arrow');
|
inner.classList.remove('stop', 'play_arrow');
|
||||||
inner.classList.add(icon);
|
inner.classList.add(icon);
|
||||||
}
|
}
|
||||||
|
@ -160,7 +160,7 @@ import 'emby-button';
|
||||||
|
|
||||||
$('.divScheduledTasks', view).on('click', '.btnStartTask', function() {
|
$('.divScheduledTasks', view).on('click', '.btnStartTask', function() {
|
||||||
const button = this;
|
const button = this;
|
||||||
let id = button.getAttribute('data-taskid');
|
const id = button.getAttribute('data-taskid');
|
||||||
ApiClient.startScheduledTask(id).then(function() {
|
ApiClient.startScheduledTask(id).then(function() {
|
||||||
updateTaskButton(button, 'Running');
|
updateTaskButton(button, 'Running');
|
||||||
reloadList(view);
|
reloadList(view);
|
||||||
|
@ -169,7 +169,7 @@ import 'emby-button';
|
||||||
|
|
||||||
$('.divScheduledTasks', view).on('click', '.btnStopTask', function() {
|
$('.divScheduledTasks', view).on('click', '.btnStopTask', function() {
|
||||||
const button = this;
|
const button = this;
|
||||||
let id = button.getAttribute('data-taskid');
|
const id = button.getAttribute('data-taskid');
|
||||||
ApiClient.stopScheduledTask(id).then(function() {
|
ApiClient.stopScheduledTask(id).then(function() {
|
||||||
updateTaskButton(button, '');
|
updateTaskButton(button, '');
|
||||||
reloadList(view);
|
reloadList(view);
|
||||||
|
|
|
@ -12,7 +12,7 @@
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<button is="emby-button" type="submit" class="raised button-submit block">
|
<button is="emby-button" type="submit" class="raised button-submit block">
|
||||||
<span>${ButtonSave}</span>
|
<span>${Save}</span>
|
||||||
</button>
|
</button>
|
||||||
</form>
|
</form>
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -192,7 +192,7 @@
|
||||||
<br />
|
<br />
|
||||||
<div>
|
<div>
|
||||||
<button is="emby-button" type="submit" class="raised button-submit block">
|
<button is="emby-button" type="submit" class="raised button-submit block">
|
||||||
<span>${ButtonSave}</span>
|
<span>${Save}</span>
|
||||||
</button>
|
</button>
|
||||||
|
|
||||||
<button is="emby-button" type="button" class="raised button-cancel block btnCancel" onclick="history.back();">
|
<button is="emby-button" type="button" class="raised button-cancel block btnCancel" onclick="history.back();">
|
||||||
|
|
|
@ -59,7 +59,7 @@
|
||||||
<br />
|
<br />
|
||||||
<div>
|
<div>
|
||||||
<button is="emby-button" type="submit" class="raised button-submit block">
|
<button is="emby-button" type="submit" class="raised button-submit block">
|
||||||
<span>${ButtonSave}</span>
|
<span>${Save}</span>
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
</form>
|
</form>
|
||||||
|
|
|
@ -49,7 +49,7 @@
|
||||||
|
|
||||||
<div>
|
<div>
|
||||||
<button is="emby-button" type="submit" class="raised button-submit block">
|
<button is="emby-button" type="submit" class="raised button-submit block">
|
||||||
<span>${ButtonSave}</span>
|
<span>${Save}</span>
|
||||||
</button>
|
</button>
|
||||||
|
|
||||||
<button is="emby-button" type="button" class="raised button-cancel block btnCancel" onclick="history.back();">
|
<button is="emby-button" type="button" class="raised button-cancel block btnCancel" onclick="history.back();">
|
||||||
|
|
|
@ -30,7 +30,7 @@
|
||||||
<div class="verticalSection" style="margin-bottom:2em;">
|
<div class="verticalSection" style="margin-bottom:2em;">
|
||||||
<div class="detailSectionHeader sectionTitleContainer">
|
<div class="detailSectionHeader sectionTitleContainer">
|
||||||
<h2 class="sectionTitle">${LabelBlockContentWithTags}</h2>
|
<h2 class="sectionTitle">${LabelBlockContentWithTags}</h2>
|
||||||
<button is="emby-button" type="button" class="fab btnAddBlockedTag submit" style="margin-left:1em;" title="${ButtonAdd}">
|
<button is="emby-button" type="button" class="fab btnAddBlockedTag submit" style="margin-left:1em;" title="${Add}">
|
||||||
<span class="material-icons add"></span>
|
<span class="material-icons add"></span>
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
|
@ -40,7 +40,7 @@
|
||||||
<div class="accessScheduleSection verticalSection" style="margin-bottom:2em;">
|
<div class="accessScheduleSection verticalSection" style="margin-bottom:2em;">
|
||||||
<div class="sectionTitleContainer">
|
<div class="sectionTitleContainer">
|
||||||
<h2 class="sectionTitle">${HeaderAccessSchedule}</h2>
|
<h2 class="sectionTitle">${HeaderAccessSchedule}</h2>
|
||||||
<button is="emby-button" type="button" class="fab btnAddSchedule submit" style="margin-left:1em;" title="${ButtonAdd}">
|
<button is="emby-button" type="button" class="fab btnAddSchedule submit" style="margin-left:1em;" title="${Add}">
|
||||||
<span class="material-icons add"></span>
|
<span class="material-icons add"></span>
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
|
@ -51,7 +51,7 @@
|
||||||
|
|
||||||
<div>
|
<div>
|
||||||
<button is="emby-button" type="submit" class="raised button-submit block">
|
<button is="emby-button" type="submit" class="raised button-submit block">
|
||||||
<span>${ButtonSave}</span>
|
<span>${Save}</span>
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
</form>
|
</form>
|
||||||
|
|
|
@ -29,7 +29,7 @@
|
||||||
</div>
|
</div>
|
||||||
<br />
|
<br />
|
||||||
<div>
|
<div>
|
||||||
<button is="emby-button" type="submit" class="raised button-submit block"><span>${ButtonSave}</span></button>
|
<button is="emby-button" type="submit" class="raised button-submit block"><span>${Save}</span></button>
|
||||||
<button is="emby-button" type="button" id="btnResetPassword" class="raised button-cancel block hide">
|
<button is="emby-button" type="button" id="btnResetPassword" class="raised button-cancel block hide">
|
||||||
<span>${ButtonResetPassword}</span>
|
<span>${ButtonResetPassword}</span>
|
||||||
</button>
|
</button>
|
||||||
|
@ -58,7 +58,7 @@
|
||||||
</div>
|
</div>
|
||||||
<div>
|
<div>
|
||||||
<button is="emby-button" type="submit" class="raised button-submit block">
|
<button is="emby-button" type="submit" class="raised button-submit block">
|
||||||
<span>${ButtonSave}</span>
|
<span>${Save}</span>
|
||||||
</button>
|
</button>
|
||||||
<button is="emby-button" type="button" id="btnResetEasyPassword" class="raised button-cancel block hide">
|
<button is="emby-button" type="button" id="btnResetEasyPassword" class="raised button-cancel block hide">
|
||||||
<span>${ButtonResetEasyPassword}</span>
|
<span>${ButtonResetEasyPassword}</span>
|
||||||
|
|
|
@ -374,7 +374,7 @@ import 'emby-select';
|
||||||
}
|
}
|
||||||
|
|
||||||
function getArtistLinksHtml(artists, serverId, context) {
|
function getArtistLinksHtml(artists, serverId, context) {
|
||||||
let html = [];
|
const html = [];
|
||||||
|
|
||||||
for (const artist of artists) {
|
for (const artist of artists) {
|
||||||
const href = appRouter.getRouteUrl(artist, {
|
const href = appRouter.getRouteUrl(artist, {
|
||||||
|
|
|
@ -1,392 +1,397 @@
|
||||||
define(['layoutManager', 'userSettings', 'inputManager', 'loading', 'globalize', 'libraryBrowser', 'mainTabsManager', 'cardBuilder', 'apphost', 'imageLoader', 'scrollStyles', 'emby-itemscontainer', 'emby-tabs', 'emby-button'], function (layoutManager, userSettings, inputManager, loading, globalize, libraryBrowser, mainTabsManager, cardBuilder, appHost, imageLoader) {
|
import layoutManager from 'layoutManager';
|
||||||
'use strict';
|
import * as userSettings from 'userSettings';
|
||||||
|
import inputManager from 'inputManager';
|
||||||
|
import loading from 'loading';
|
||||||
|
import globalize from 'globalize';
|
||||||
|
import * as mainTabsManager from 'mainTabsManager';
|
||||||
|
import cardBuilder from 'cardBuilder';
|
||||||
|
import imageLoader from 'imageLoader';
|
||||||
|
import 'scrollStyles';
|
||||||
|
import 'emby-itemscontainer';
|
||||||
|
import 'emby-tabs';
|
||||||
|
import 'emby-button';
|
||||||
|
|
||||||
loading = loading.default || loading;
|
function enableScrollX() {
|
||||||
layoutManager = layoutManager.default || layoutManager;
|
return !layoutManager.desktop;
|
||||||
|
}
|
||||||
|
|
||||||
function enableScrollX() {
|
function getBackdropShape() {
|
||||||
return !layoutManager.desktop;
|
if (enableScrollX()) {
|
||||||
|
return 'overflowBackdrop';
|
||||||
|
}
|
||||||
|
return 'backdrop';
|
||||||
|
}
|
||||||
|
|
||||||
|
function getPortraitShape() {
|
||||||
|
if (enableScrollX()) {
|
||||||
|
return 'overflowPortrait';
|
||||||
|
}
|
||||||
|
return 'portrait';
|
||||||
|
}
|
||||||
|
|
||||||
|
function getLimit() {
|
||||||
|
if (enableScrollX()) {
|
||||||
|
return 12;
|
||||||
}
|
}
|
||||||
|
|
||||||
function getBackdropShape() {
|
return 9;
|
||||||
if (enableScrollX()) {
|
}
|
||||||
return 'overflowBackdrop';
|
|
||||||
}
|
function loadRecommendedPrograms(page) {
|
||||||
return 'backdrop';
|
loading.show();
|
||||||
|
let limit = getLimit();
|
||||||
|
|
||||||
|
if (enableScrollX()) {
|
||||||
|
limit *= 2;
|
||||||
}
|
}
|
||||||
|
|
||||||
function getPortraitShape() {
|
ApiClient.getLiveTvRecommendedPrograms({
|
||||||
if (enableScrollX()) {
|
userId: Dashboard.getCurrentUserId(),
|
||||||
return 'overflowPortrait';
|
IsAiring: true,
|
||||||
}
|
limit: limit,
|
||||||
return 'portrait';
|
ImageTypeLimit: 1,
|
||||||
}
|
EnableImageTypes: 'Primary,Thumb,Backdrop',
|
||||||
|
EnableTotalRecordCount: false,
|
||||||
|
Fields: 'ChannelInfo,PrimaryImageAspectRatio'
|
||||||
|
}).then(function (result) {
|
||||||
|
renderItems(page, result.Items, 'activeProgramItems', 'play', {
|
||||||
|
showAirDateTime: false,
|
||||||
|
showAirEndTime: true
|
||||||
|
});
|
||||||
|
loading.hide();
|
||||||
|
|
||||||
function getLimit() {
|
import('autoFocuser').then(({default: autoFocuser}) => {
|
||||||
if (enableScrollX()) {
|
autoFocuser.autoFocus(page);
|
||||||
return 12;
|
});
|
||||||
}
|
});
|
||||||
|
}
|
||||||
|
|
||||||
return 9;
|
function reload(page, enableFullRender) {
|
||||||
}
|
if (enableFullRender) {
|
||||||
|
loadRecommendedPrograms(page);
|
||||||
function loadRecommendedPrograms(page) {
|
ApiClient.getLiveTvPrograms({
|
||||||
loading.show();
|
|
||||||
var limit = getLimit();
|
|
||||||
|
|
||||||
if (enableScrollX()) {
|
|
||||||
limit *= 2;
|
|
||||||
}
|
|
||||||
|
|
||||||
ApiClient.getLiveTvRecommendedPrograms({
|
|
||||||
userId: Dashboard.getCurrentUserId(),
|
userId: Dashboard.getCurrentUserId(),
|
||||||
IsAiring: true,
|
HasAired: false,
|
||||||
limit: limit,
|
limit: getLimit(),
|
||||||
ImageTypeLimit: 1,
|
IsMovie: false,
|
||||||
EnableImageTypes: 'Primary,Thumb,Backdrop',
|
IsSports: false,
|
||||||
|
IsKids: false,
|
||||||
|
IsNews: false,
|
||||||
|
IsSeries: true,
|
||||||
EnableTotalRecordCount: false,
|
EnableTotalRecordCount: false,
|
||||||
Fields: 'ChannelInfo,PrimaryImageAspectRatio'
|
Fields: 'ChannelInfo,PrimaryImageAspectRatio',
|
||||||
|
EnableImageTypes: 'Primary,Thumb'
|
||||||
}).then(function (result) {
|
}).then(function (result) {
|
||||||
renderItems(page, result.Items, 'activeProgramItems', 'play', {
|
renderItems(page, result.Items, 'upcomingEpisodeItems');
|
||||||
showAirDateTime: false,
|
});
|
||||||
showAirEndTime: true
|
ApiClient.getLiveTvPrograms({
|
||||||
|
userId: Dashboard.getCurrentUserId(),
|
||||||
|
HasAired: false,
|
||||||
|
limit: getLimit(),
|
||||||
|
IsMovie: true,
|
||||||
|
EnableTotalRecordCount: false,
|
||||||
|
Fields: 'ChannelInfo',
|
||||||
|
EnableImageTypes: 'Primary,Thumb'
|
||||||
|
}).then(function (result) {
|
||||||
|
renderItems(page, result.Items, 'upcomingTvMovieItems', null, {
|
||||||
|
shape: getPortraitShape(),
|
||||||
|
preferThumb: null,
|
||||||
|
showParentTitle: false
|
||||||
});
|
});
|
||||||
loading.hide();
|
});
|
||||||
|
ApiClient.getLiveTvPrograms({
|
||||||
require(['autoFocuser'], function (autoFocuser) {
|
userId: Dashboard.getCurrentUserId(),
|
||||||
autoFocuser.autoFocus(page);
|
HasAired: false,
|
||||||
|
limit: getLimit(),
|
||||||
|
IsSports: true,
|
||||||
|
EnableTotalRecordCount: false,
|
||||||
|
Fields: 'ChannelInfo,PrimaryImageAspectRatio',
|
||||||
|
EnableImageTypes: 'Primary,Thumb'
|
||||||
|
}).then(function (result) {
|
||||||
|
renderItems(page, result.Items, 'upcomingSportsItems');
|
||||||
|
});
|
||||||
|
ApiClient.getLiveTvPrograms({
|
||||||
|
userId: Dashboard.getCurrentUserId(),
|
||||||
|
HasAired: false,
|
||||||
|
limit: getLimit(),
|
||||||
|
IsKids: true,
|
||||||
|
EnableTotalRecordCount: false,
|
||||||
|
Fields: 'ChannelInfo,PrimaryImageAspectRatio',
|
||||||
|
EnableImageTypes: 'Primary,Thumb'
|
||||||
|
}).then(function (result) {
|
||||||
|
renderItems(page, result.Items, 'upcomingKidsItems');
|
||||||
|
});
|
||||||
|
ApiClient.getLiveTvPrograms({
|
||||||
|
userId: Dashboard.getCurrentUserId(),
|
||||||
|
HasAired: false,
|
||||||
|
limit: getLimit(),
|
||||||
|
IsNews: true,
|
||||||
|
EnableTotalRecordCount: false,
|
||||||
|
Fields: 'ChannelInfo,PrimaryImageAspectRatio',
|
||||||
|
EnableImageTypes: 'Primary,Thumb'
|
||||||
|
}).then(function (result) {
|
||||||
|
renderItems(page, result.Items, 'upcomingNewsItems', null, {
|
||||||
|
showParentTitleOrTitle: true,
|
||||||
|
showTitle: false,
|
||||||
|
showParentTitle: false
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
function reload(page, enableFullRender) {
|
function renderItems(page, items, sectionClass, overlayButton, cardOptions) {
|
||||||
if (enableFullRender) {
|
const html = cardBuilder.getCardsHtml(Object.assign({
|
||||||
loadRecommendedPrograms(page);
|
items: items,
|
||||||
ApiClient.getLiveTvPrograms({
|
preferThumb: 'auto',
|
||||||
userId: Dashboard.getCurrentUserId(),
|
inheritThumb: false,
|
||||||
HasAired: false,
|
shape: enableScrollX() ? 'autooverflow' : 'auto',
|
||||||
limit: getLimit(),
|
defaultShape: getBackdropShape(),
|
||||||
IsMovie: false,
|
showParentTitle: true,
|
||||||
IsSports: false,
|
showTitle: true,
|
||||||
IsKids: false,
|
centerText: true,
|
||||||
IsNews: false,
|
coverImage: true,
|
||||||
IsSeries: true,
|
overlayText: false,
|
||||||
EnableTotalRecordCount: false,
|
lazy: true,
|
||||||
Fields: 'ChannelInfo,PrimaryImageAspectRatio',
|
overlayPlayButton: overlayButton === 'play',
|
||||||
EnableImageTypes: 'Primary,Thumb'
|
overlayMoreButton: overlayButton === 'more',
|
||||||
}).then(function (result) {
|
overlayInfoButton: overlayButton === 'info',
|
||||||
renderItems(page, result.Items, 'upcomingEpisodeItems');
|
allowBottomPadding: !enableScrollX(),
|
||||||
});
|
showAirTime: true,
|
||||||
ApiClient.getLiveTvPrograms({
|
showAirDateTime: true
|
||||||
userId: Dashboard.getCurrentUserId(),
|
}, cardOptions || {}));
|
||||||
HasAired: false,
|
const elem = page.querySelector('.' + sectionClass);
|
||||||
limit: getLimit(),
|
elem.innerHTML = html;
|
||||||
IsMovie: true,
|
imageLoader.lazyChildren(elem);
|
||||||
EnableTotalRecordCount: false,
|
}
|
||||||
Fields: 'ChannelInfo',
|
|
||||||
EnableImageTypes: 'Primary,Thumb'
|
function getTabs() {
|
||||||
}).then(function (result) {
|
return [{
|
||||||
renderItems(page, result.Items, 'upcomingTvMovieItems', null, {
|
name: globalize.translate('Programs')
|
||||||
shape: getPortraitShape(),
|
}, {
|
||||||
preferThumb: null,
|
name: globalize.translate('TabGuide')
|
||||||
showParentTitle: false
|
}, {
|
||||||
});
|
name: globalize.translate('TabChannels')
|
||||||
});
|
}, {
|
||||||
ApiClient.getLiveTvPrograms({
|
name: globalize.translate('TabRecordings')
|
||||||
userId: Dashboard.getCurrentUserId(),
|
}, {
|
||||||
HasAired: false,
|
name: globalize.translate('HeaderSchedule')
|
||||||
limit: getLimit(),
|
}, {
|
||||||
IsSports: true,
|
name: globalize.translate('TabSeries')
|
||||||
EnableTotalRecordCount: false,
|
}];
|
||||||
Fields: 'ChannelInfo,PrimaryImageAspectRatio',
|
}
|
||||||
EnableImageTypes: 'Primary,Thumb'
|
|
||||||
}).then(function (result) {
|
function setScrollClasses(elem, scrollX) {
|
||||||
renderItems(page, result.Items, 'upcomingSportsItems');
|
if (scrollX) {
|
||||||
});
|
elem.classList.add('hiddenScrollX');
|
||||||
ApiClient.getLiveTvPrograms({
|
|
||||||
userId: Dashboard.getCurrentUserId(),
|
if (layoutManager.tv) {
|
||||||
HasAired: false,
|
elem.classList.add('smoothScrollX');
|
||||||
limit: getLimit(),
|
|
||||||
IsKids: true,
|
|
||||||
EnableTotalRecordCount: false,
|
|
||||||
Fields: 'ChannelInfo,PrimaryImageAspectRatio',
|
|
||||||
EnableImageTypes: 'Primary,Thumb'
|
|
||||||
}).then(function (result) {
|
|
||||||
renderItems(page, result.Items, 'upcomingKidsItems');
|
|
||||||
});
|
|
||||||
ApiClient.getLiveTvPrograms({
|
|
||||||
userId: Dashboard.getCurrentUserId(),
|
|
||||||
HasAired: false,
|
|
||||||
limit: getLimit(),
|
|
||||||
IsNews: true,
|
|
||||||
EnableTotalRecordCount: false,
|
|
||||||
Fields: 'ChannelInfo,PrimaryImageAspectRatio',
|
|
||||||
EnableImageTypes: 'Primary,Thumb'
|
|
||||||
}).then(function (result) {
|
|
||||||
renderItems(page, result.Items, 'upcomingNewsItems', null, {
|
|
||||||
showParentTitleOrTitle: true,
|
|
||||||
showTitle: false,
|
|
||||||
showParentTitle: false
|
|
||||||
});
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
elem.classList.add('scrollX');
|
||||||
|
elem.classList.remove('vertical-wrap');
|
||||||
|
} else {
|
||||||
|
elem.classList.remove('hiddenScrollX');
|
||||||
|
elem.classList.remove('smoothScrollX');
|
||||||
|
elem.classList.remove('scrollX');
|
||||||
|
elem.classList.add('vertical-wrap');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function getDefaultTabIndex(folderId) {
|
||||||
|
if (userSettings.get('landing-' + folderId) === 'guide') {
|
||||||
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
function renderItems(page, items, sectionClass, overlayButton, cardOptions) {
|
return 0;
|
||||||
var html = cardBuilder.getCardsHtml(Object.assign({
|
}
|
||||||
items: items,
|
|
||||||
preferThumb: 'auto',
|
export default function (view, params) {
|
||||||
inheritThumb: false,
|
function enableFullRender() {
|
||||||
shape: enableScrollX() ? 'autooverflow' : 'auto',
|
return new Date().getTime() - lastFullRender > 3e5;
|
||||||
defaultShape: getBackdropShape(),
|
|
||||||
showParentTitle: true,
|
|
||||||
showTitle: true,
|
|
||||||
centerText: true,
|
|
||||||
coverImage: true,
|
|
||||||
overlayText: false,
|
|
||||||
lazy: true,
|
|
||||||
overlayPlayButton: overlayButton === 'play',
|
|
||||||
overlayMoreButton: overlayButton === 'more',
|
|
||||||
overlayInfoButton: overlayButton === 'info',
|
|
||||||
allowBottomPadding: !enableScrollX(),
|
|
||||||
showAirTime: true,
|
|
||||||
showAirDateTime: true
|
|
||||||
}, cardOptions || {}));
|
|
||||||
var elem = page.querySelector('.' + sectionClass);
|
|
||||||
elem.innerHTML = html;
|
|
||||||
imageLoader.lazyChildren(elem);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function getTabs() {
|
function onBeforeTabChange(evt) {
|
||||||
return [{
|
preLoadTab(view, parseInt(evt.detail.selectedTabIndex));
|
||||||
name: globalize.translate('Programs')
|
|
||||||
}, {
|
|
||||||
name: globalize.translate('TabGuide')
|
|
||||||
}, {
|
|
||||||
name: globalize.translate('TabChannels')
|
|
||||||
}, {
|
|
||||||
name: globalize.translate('TabRecordings')
|
|
||||||
}, {
|
|
||||||
name: globalize.translate('HeaderSchedule')
|
|
||||||
}, {
|
|
||||||
name: globalize.translate('TabSeries')
|
|
||||||
}];
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function setScrollClasses(elem, scrollX) {
|
function onTabChange(evt) {
|
||||||
if (scrollX) {
|
const previousTabController = tabControllers[parseInt(evt.detail.previousIndex)];
|
||||||
elem.classList.add('hiddenScrollX');
|
|
||||||
|
|
||||||
if (layoutManager.tv) {
|
if (previousTabController && previousTabController.onHide) {
|
||||||
elem.classList.add('smoothScrollX');
|
previousTabController.onHide();
|
||||||
|
}
|
||||||
|
|
||||||
|
loadTab(view, parseInt(evt.detail.selectedTabIndex));
|
||||||
|
}
|
||||||
|
|
||||||
|
function getTabContainers() {
|
||||||
|
return view.querySelectorAll('.pageTabContent');
|
||||||
|
}
|
||||||
|
|
||||||
|
function initTabs() {
|
||||||
|
mainTabsManager.setTabs(view, currentTabIndex, getTabs, getTabContainers, onBeforeTabChange, onTabChange);
|
||||||
|
}
|
||||||
|
|
||||||
|
function getTabController(page, index, callback) {
|
||||||
|
let depends;
|
||||||
|
|
||||||
|
// TODO int is a little hard to read
|
||||||
|
switch (index) {
|
||||||
|
case 0:
|
||||||
|
depends = 'controllers/livetv/livetvsuggested';
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 1:
|
||||||
|
depends = 'controllers/livetv/livetvguide';
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 2:
|
||||||
|
depends = 'controllers/livetv/livetvchannels';
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 3:
|
||||||
|
depends = 'controllers/livetv/livetvrecordings';
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 4:
|
||||||
|
depends = 'controllers/livetv/livetvschedule';
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 5:
|
||||||
|
depends = 'controllers/livetv/livetvseriestimers';
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
import(depends).then(({default: controllerFactory}) => {
|
||||||
|
let tabContent;
|
||||||
|
|
||||||
|
if (index === 0) {
|
||||||
|
tabContent = view.querySelector(`.pageTabContent[data-index="${index}"]`);
|
||||||
|
self.tabContent = tabContent;
|
||||||
}
|
}
|
||||||
|
|
||||||
elem.classList.add('scrollX');
|
let controller = tabControllers[index];
|
||||||
elem.classList.remove('vertical-wrap');
|
|
||||||
} else {
|
|
||||||
elem.classList.remove('hiddenScrollX');
|
|
||||||
elem.classList.remove('smoothScrollX');
|
|
||||||
elem.classList.remove('scrollX');
|
|
||||||
elem.classList.add('vertical-wrap');
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function getDefaultTabIndex(folderId) {
|
if (!controller) {
|
||||||
if (userSettings.get('landing-' + folderId) === 'guide') {
|
tabContent = view.querySelector(`.pageTabContent[data-index="${index}"]`);
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
return 0;
|
if (index === 0) {
|
||||||
}
|
controller = self;
|
||||||
|
} else if (index === 6) {
|
||||||
return function (view, params) {
|
controller = new controllerFactory(view, tabContent, {
|
||||||
function enableFullRender() {
|
collectionType: 'livetv'
|
||||||
return new Date().getTime() - lastFullRender > 3e5;
|
});
|
||||||
}
|
} else {
|
||||||
|
controller = new controllerFactory(view, params, tabContent);
|
||||||
function onBeforeTabChange(evt) {
|
|
||||||
preLoadTab(view, parseInt(evt.detail.selectedTabIndex));
|
|
||||||
}
|
|
||||||
|
|
||||||
function onTabChange(evt) {
|
|
||||||
var previousTabController = tabControllers[parseInt(evt.detail.previousIndex)];
|
|
||||||
|
|
||||||
if (previousTabController && previousTabController.onHide) {
|
|
||||||
previousTabController.onHide();
|
|
||||||
}
|
|
||||||
|
|
||||||
loadTab(view, parseInt(evt.detail.selectedTabIndex));
|
|
||||||
}
|
|
||||||
|
|
||||||
function getTabContainers() {
|
|
||||||
return view.querySelectorAll('.pageTabContent');
|
|
||||||
}
|
|
||||||
|
|
||||||
function initTabs() {
|
|
||||||
mainTabsManager.setTabs(view, currentTabIndex, getTabs, getTabContainers, onBeforeTabChange, onTabChange);
|
|
||||||
}
|
|
||||||
|
|
||||||
function getTabController(page, index, callback) {
|
|
||||||
var depends = [];
|
|
||||||
|
|
||||||
// TODO int is a little hard to read
|
|
||||||
switch (index) {
|
|
||||||
case 0:
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 1:
|
|
||||||
depends.push('controllers/livetv/livetvguide');
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 2:
|
|
||||||
depends.push('controllers/livetv/livetvchannels');
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 3:
|
|
||||||
depends.push('controllers/livetv/livetvrecordings');
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 4:
|
|
||||||
depends.push('controllers/livetv/livetvschedule');
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 5:
|
|
||||||
depends.push('controllers/livetv/livetvseriestimers');
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
require(depends, function (controllerFactory) {
|
|
||||||
controllerFactory = controllerFactory.default || controllerFactory;
|
|
||||||
|
|
||||||
var tabContent;
|
|
||||||
|
|
||||||
if (index == 0) {
|
|
||||||
tabContent = view.querySelector(".pageTabContent[data-index='" + index + "']");
|
|
||||||
self.tabContent = tabContent;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
var controller = tabControllers[index];
|
tabControllers[index] = controller;
|
||||||
|
|
||||||
if (!controller) {
|
if (controller.initTab) {
|
||||||
tabContent = view.querySelector(".pageTabContent[data-index='" + index + "']");
|
controller.initTab();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (index === 0) {
|
callback(controller);
|
||||||
controller = self;
|
});
|
||||||
} else if (index === 6) {
|
}
|
||||||
controller = new controllerFactory(view, tabContent, {
|
|
||||||
collectionType: 'livetv'
|
|
||||||
});
|
|
||||||
} else {
|
|
||||||
controller = new controllerFactory(view, params, tabContent);
|
|
||||||
}
|
|
||||||
|
|
||||||
tabControllers[index] = controller;
|
function preLoadTab(page, index) {
|
||||||
|
getTabController(page, index, function (controller) {
|
||||||
|
if (renderedTabs.indexOf(index) === -1 && controller.preRender) {
|
||||||
|
controller.preRender();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
if (controller.initTab) {
|
function loadTab(page, index) {
|
||||||
controller.initTab();
|
currentTabIndex = index;
|
||||||
}
|
getTabController(page, index, function (controller) {
|
||||||
|
initialTabIndex = null;
|
||||||
|
|
||||||
|
if (renderedTabs.indexOf(index) === -1) {
|
||||||
|
if (index === 1) {
|
||||||
|
renderedTabs.push(index);
|
||||||
}
|
}
|
||||||
|
|
||||||
callback(controller);
|
controller.renderTab();
|
||||||
});
|
} else if (controller.onShow) {
|
||||||
|
controller.onShow();
|
||||||
|
}
|
||||||
|
|
||||||
|
currentTabController = controller;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function onInputCommand(evt) {
|
||||||
|
if (evt.detail.command === 'search') {
|
||||||
|
evt.preventDefault();
|
||||||
|
Dashboard.navigate('search.html?collectionType=livetv');
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
function preLoadTab(page, index) {
|
let isViewRestored;
|
||||||
getTabController(page, index, function (controller) {
|
const self = this;
|
||||||
if (renderedTabs.indexOf(index) === -1 && controller.preRender) {
|
let currentTabIndex = parseInt(params.tab || getDefaultTabIndex('livetv'));
|
||||||
controller.preRender();
|
let initialTabIndex = currentTabIndex;
|
||||||
}
|
let lastFullRender = 0;
|
||||||
});
|
[].forEach.call(view.querySelectorAll('.sectionTitleTextButton-programs'), function (link) {
|
||||||
|
const href = link.href;
|
||||||
|
|
||||||
|
if (href) {
|
||||||
|
link.href = href + '&serverId=' + ApiClient.serverId();
|
||||||
}
|
}
|
||||||
|
});
|
||||||
|
|
||||||
function loadTab(page, index) {
|
self.initTab = function () {
|
||||||
currentTabIndex = index;
|
const tabContent = view.querySelector('.pageTabContent[data-index="0"]');
|
||||||
getTabController(page, index, function (controller) {
|
const containers = tabContent.querySelectorAll('.itemsContainer');
|
||||||
initialTabIndex = null;
|
|
||||||
|
|
||||||
if (renderedTabs.indexOf(index) == -1) {
|
for (let i = 0, length = containers.length; i < length; i++) {
|
||||||
if (index === 1) {
|
setScrollClasses(containers[i], enableScrollX());
|
||||||
renderedTabs.push(index);
|
|
||||||
}
|
|
||||||
|
|
||||||
controller.renderTab();
|
|
||||||
} else if (controller.onShow) {
|
|
||||||
controller.onShow();
|
|
||||||
}
|
|
||||||
|
|
||||||
currentTabController = controller;
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function onInputCommand(evt) {
|
|
||||||
if (evt.detail.command === 'search') {
|
|
||||||
evt.preventDefault();
|
|
||||||
Dashboard.navigate('search.html?collectionType=livetv');
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
var isViewRestored;
|
|
||||||
var self = this;
|
|
||||||
var currentTabIndex = parseInt(params.tab || getDefaultTabIndex('livetv'));
|
|
||||||
var initialTabIndex = currentTabIndex;
|
|
||||||
var lastFullRender = 0;
|
|
||||||
[].forEach.call(view.querySelectorAll('.sectionTitleTextButton-programs'), function (link) {
|
|
||||||
var href = link.href;
|
|
||||||
|
|
||||||
if (href) {
|
|
||||||
link.href = href + '&serverId=' + ApiClient.serverId();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
self.initTab = function () {
|
|
||||||
var tabContent = view.querySelector(".pageTabContent[data-index='0']");
|
|
||||||
var containers = tabContent.querySelectorAll('.itemsContainer');
|
|
||||||
|
|
||||||
for (var i = 0, length = containers.length; i < length; i++) {
|
|
||||||
setScrollClasses(containers[i], enableScrollX());
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
self.renderTab = function () {
|
|
||||||
var tabContent = view.querySelector(".pageTabContent[data-index='0']");
|
|
||||||
|
|
||||||
if (enableFullRender()) {
|
|
||||||
reload(tabContent, true);
|
|
||||||
lastFullRender = new Date().getTime();
|
|
||||||
} else {
|
|
||||||
reload(tabContent);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
var currentTabController;
|
|
||||||
var tabControllers = [];
|
|
||||||
var renderedTabs = [];
|
|
||||||
view.addEventListener('viewbeforeshow', function (evt) {
|
|
||||||
isViewRestored = evt.detail.isRestored;
|
|
||||||
initTabs();
|
|
||||||
});
|
|
||||||
view.addEventListener('viewshow', function (evt) {
|
|
||||||
isViewRestored = evt.detail.isRestored;
|
|
||||||
|
|
||||||
if (!isViewRestored) {
|
|
||||||
mainTabsManager.selectedTabIndex(initialTabIndex);
|
|
||||||
}
|
|
||||||
|
|
||||||
inputManager.on(window, onInputCommand);
|
|
||||||
});
|
|
||||||
view.addEventListener('viewbeforehide', function (e) {
|
|
||||||
if (currentTabController && currentTabController.onHide) {
|
|
||||||
currentTabController.onHide();
|
|
||||||
}
|
|
||||||
|
|
||||||
inputManager.off(window, onInputCommand);
|
|
||||||
});
|
|
||||||
view.addEventListener('viewdestroy', function (evt) {
|
|
||||||
tabControllers.forEach(function (tabController) {
|
|
||||||
if (tabController.destroy) {
|
|
||||||
tabController.destroy();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
});
|
|
||||||
};
|
};
|
||||||
});
|
|
||||||
|
self.renderTab = function () {
|
||||||
|
const tabContent = view.querySelector('.pageTabContent[data-index="0"]');
|
||||||
|
|
||||||
|
if (enableFullRender()) {
|
||||||
|
reload(tabContent, true);
|
||||||
|
lastFullRender = new Date().getTime();
|
||||||
|
} else {
|
||||||
|
reload(tabContent);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
let currentTabController;
|
||||||
|
const tabControllers = [];
|
||||||
|
const renderedTabs = [];
|
||||||
|
view.addEventListener('viewbeforeshow', function (evt) {
|
||||||
|
isViewRestored = evt.detail.isRestored;
|
||||||
|
initTabs();
|
||||||
|
});
|
||||||
|
view.addEventListener('viewshow', function (evt) {
|
||||||
|
isViewRestored = evt.detail.isRestored;
|
||||||
|
|
||||||
|
if (!isViewRestored) {
|
||||||
|
mainTabsManager.selectedTabIndex(initialTabIndex);
|
||||||
|
}
|
||||||
|
|
||||||
|
inputManager.on(window, onInputCommand);
|
||||||
|
});
|
||||||
|
view.addEventListener('viewbeforehide', function () {
|
||||||
|
if (currentTabController && currentTabController.onHide) {
|
||||||
|
currentTabController.onHide();
|
||||||
|
}
|
||||||
|
|
||||||
|
inputManager.off(window, onInputCommand);
|
||||||
|
});
|
||||||
|
view.addEventListener('viewdestroy', function () {
|
||||||
|
tabControllers.forEach(function (tabController) {
|
||||||
|
if (tabController.destroy) {
|
||||||
|
tabController.destroy();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
|
@ -94,7 +94,7 @@
|
||||||
</div>
|
</div>
|
||||||
<br />
|
<br />
|
||||||
<div>
|
<div>
|
||||||
<button is="emby-button" type="submit" class="raised button-submit block"><span>${ButtonSave}</span></button>
|
<button is="emby-button" type="submit" class="raised button-submit block"><span>${Save}</span></button>
|
||||||
</div>
|
</div>
|
||||||
</form>
|
</form>
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -7,7 +7,7 @@
|
||||||
<h2 class="sectionTitle sectionTitle-cards">
|
<h2 class="sectionTitle sectionTitle-cards">
|
||||||
<span>${HeaderTunerDevices}</span>
|
<span>${HeaderTunerDevices}</span>
|
||||||
</h2>
|
</h2>
|
||||||
<button is="emby-button" type="button" class="fab btnAddDevice submit sectionTitleButton" style="margin-left:1em;" title="${ButtonAdd}">
|
<button is="emby-button" type="button" class="fab btnAddDevice submit sectionTitleButton" style="margin-left:1em;" title="${Add}">
|
||||||
<span class="material-icons add"></span>
|
<span class="material-icons add"></span>
|
||||||
</button>
|
</button>
|
||||||
<a is="emby-linkbutton" rel="noopener noreferrer" style="margin-left:2em!important;" class="raised button-alt headerHelpButton" target="_blank" href="https://docs.jellyfin.org/general/server/live-tv/index.html">${Help}</a>
|
<a is="emby-linkbutton" rel="noopener noreferrer" style="margin-left:2em!important;" class="raised button-alt headerHelpButton" target="_blank" href="https://docs.jellyfin.org/general/server/live-tv/index.html">${Help}</a>
|
||||||
|
@ -20,7 +20,7 @@
|
||||||
<div class="verticalSection">
|
<div class="verticalSection">
|
||||||
<div class="sectionTitleContainer">
|
<div class="sectionTitleContainer">
|
||||||
<h2 class="sectionTitle">${HeaderGuideProviders}</h2>
|
<h2 class="sectionTitle">${HeaderGuideProviders}</h2>
|
||||||
<button is="emby-button" type="button" class="fab btnAddProvider submit" style="margin-left:1em;" title="${ButtonAdd}">
|
<button is="emby-button" type="button" class="fab btnAddProvider submit" style="margin-left:1em;" title="${Add}">
|
||||||
<span class="material-icons add"></span>
|
<span class="material-icons add"></span>
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -15,12 +15,10 @@ import 'emby-button';
|
||||||
const enableFocusTransform = !browser.slow && !browser.edge;
|
const enableFocusTransform = !browser.slow && !browser.edge;
|
||||||
|
|
||||||
function getDeviceHtml(device) {
|
function getDeviceHtml(device) {
|
||||||
let padderClass;
|
const padderClass = 'cardPadder-backdrop';
|
||||||
|
let cssClass = 'card scalableCard backdropCard backdropCard-scalable';
|
||||||
|
const cardBoxCssClass = 'cardBox visualCardBox';
|
||||||
let html = '';
|
let html = '';
|
||||||
let cssClass = 'card scalableCard';
|
|
||||||
let cardBoxCssClass = 'cardBox visualCardBox';
|
|
||||||
cssClass += ' backdropCard backdropCard-scalable';
|
|
||||||
padderClass = 'cardPadder-backdrop';
|
|
||||||
|
|
||||||
// TODO move card creation code to Card component
|
// TODO move card creation code to Card component
|
||||||
|
|
||||||
|
|
|
@ -67,7 +67,7 @@
|
||||||
<input type="hidden" class="fldDeviceId" />
|
<input type="hidden" class="fldDeviceId" />
|
||||||
<div>
|
<div>
|
||||||
<button is="emby-button" type="submit" class="raised button-submit block">
|
<button is="emby-button" type="submit" class="raised button-submit block">
|
||||||
<span>${ButtonSave}</span>
|
<span>${Save}</span>
|
||||||
</button>
|
</button>
|
||||||
<button is="emby-button" type="button" class="raised button-cancel block btnCancel" onclick="history.back();">
|
<button is="emby-button" type="button" class="raised button-cancel block btnCancel" onclick="history.back();">
|
||||||
<span>${ButtonCancel}</span>
|
<span>${ButtonCancel}</span>
|
||||||
|
|
|
@ -227,7 +227,7 @@ import 'emby-button';
|
||||||
return [{
|
return [{
|
||||||
name: globalize.translate('Movies')
|
name: globalize.translate('Movies')
|
||||||
}, {
|
}, {
|
||||||
name: globalize.translate('TabSuggestions')
|
name: globalize.translate('Suggestions')
|
||||||
}, {
|
}, {
|
||||||
name: globalize.translate('TabTrailers')
|
name: globalize.translate('TabTrailers')
|
||||||
}, {
|
}, {
|
||||||
|
|
|
@ -104,7 +104,6 @@ import 'flexStyles';
|
||||||
}
|
}
|
||||||
|
|
||||||
var itemsContainer = elem.querySelector('.itemsContainer');
|
var itemsContainer = elem.querySelector('.itemsContainer');
|
||||||
|
|
||||||
itemsContainer.innerHTML = cardBuilder.getCardsHtml({
|
itemsContainer.innerHTML = cardBuilder.getCardsHtml({
|
||||||
items: result.Items,
|
items: result.Items,
|
||||||
showUnplayedIndicator: false,
|
showUnplayedIndicator: false,
|
||||||
|
@ -178,13 +177,13 @@ import 'flexStyles';
|
||||||
|
|
||||||
function getTabs() {
|
function getTabs() {
|
||||||
return [{
|
return [{
|
||||||
name: globalize.translate('TabSuggestions')
|
name: globalize.translate('Suggestions')
|
||||||
}, {
|
}, {
|
||||||
name: globalize.translate('TabAlbums')
|
name: globalize.translate('TabAlbums')
|
||||||
}, {
|
}, {
|
||||||
name: globalize.translate('TabAlbumArtists')
|
name: globalize.translate('TabAlbumArtists')
|
||||||
}, {
|
}, {
|
||||||
name: globalize.translate('TabArtists')
|
name: globalize.translate('Artists')
|
||||||
}, {
|
}, {
|
||||||
name: globalize.translate('TabPlaylists')
|
name: globalize.translate('TabPlaylists')
|
||||||
}, {
|
}, {
|
||||||
|
|
|
@ -30,7 +30,7 @@
|
||||||
|
|
||||||
<div class="nowPlayingInfoButtons">
|
<div class="nowPlayingInfoButtons">
|
||||||
|
|
||||||
<button is="paper-icon-button-light" class="btnCommand btnRepeat repeatToggleButton autoSize" title="${ButtonRepeat}"
|
<button is="paper-icon-button-light" class="btnCommand btnRepeat repeatToggleButton autoSize" title="${Repeat}"
|
||||||
data-command="SetRepeatMode">
|
data-command="SetRepeatMode">
|
||||||
<span class="material-icons repeat"></span>
|
<span class="material-icons repeat"></span>
|
||||||
</button>
|
</button>
|
||||||
|
@ -85,7 +85,7 @@
|
||||||
<span class="material-icons shuffle"></span>
|
<span class="material-icons shuffle"></span>
|
||||||
</button>
|
</button>
|
||||||
|
|
||||||
<button is="paper-icon-button-light" class="btnCommand btnRepeat repeatToggleButton autoSize" title="${ButtonRepeat}"
|
<button is="paper-icon-button-light" class="btnCommand btnRepeat repeatToggleButton autoSize" title="${Repeat}"
|
||||||
data-command="SetRepeatMode">
|
data-command="SetRepeatMode">
|
||||||
<span class="material-icons repeat"></span>
|
<span class="material-icons repeat"></span>
|
||||||
</button>
|
</button>
|
||||||
|
@ -132,7 +132,7 @@
|
||||||
<button is="paper-icon-button-light" class="btnGoHome btnCommand autoSize" title="${ButtonHome}" data-command="GoHome">
|
<button is="paper-icon-button-light" class="btnGoHome btnCommand autoSize" title="${ButtonHome}" data-command="GoHome">
|
||||||
<span class="material-icons home"></span>
|
<span class="material-icons home"></span>
|
||||||
</button>
|
</button>
|
||||||
<button is="paper-icon-button-light" class="btnShowSearch btnCommand autoSize" title="${ButtonSearch}" data-command="GoToSearch">
|
<button is="paper-icon-button-light" class="btnShowSearch btnCommand autoSize" title="${Search}" data-command="GoToSearch">
|
||||||
<span class="material-icons search"></span>
|
<span class="material-icons search"></span>
|
||||||
</button>
|
</button>
|
||||||
<button is="paper-icon-button-light" class="bthShowSettings btnCommand autoSize" title="${ButtonSettings}" data-command="GoToSettings">
|
<button is="paper-icon-button-light" class="bthShowSettings btnCommand autoSize" title="${ButtonSettings}" data-command="GoToSettings">
|
||||||
|
@ -180,10 +180,10 @@
|
||||||
<button id="togglePlaylist" is="paper-icon-button-light" class="btnTogglePlaylist hide" title="${ButtonTogglePlaylist}">
|
<button id="togglePlaylist" is="paper-icon-button-light" class="btnTogglePlaylist hide" title="${ButtonTogglePlaylist}">
|
||||||
<span class="material-icons queue_music"></span>
|
<span class="material-icons queue_music"></span>
|
||||||
</button>
|
</button>
|
||||||
<button is="paper-icon-button-light" class="btnSavePlaylist hide" title="${ButtonSave}">
|
<button is="paper-icon-button-light" class="btnSavePlaylist hide" title="${Save}">
|
||||||
<span class="material-icons save"></span>
|
<span class="material-icons save"></span>
|
||||||
</button>
|
</button>
|
||||||
<button id="toggleContextMenu" is="paper-icon-button-light" class="btnToggleContextMenu" title="${ButtonToggleContextMenu}">
|
<button id="toggleContextMenu" is="paper-icon-button-light" class="btnToggleContextMenu" title="${ButtonMore}">
|
||||||
<span class="material-icons more_vert"></span>
|
<span class="material-icons more_vert"></span>
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -1,5 +1,4 @@
|
||||||
<div id="videoOsdPage" data-role="page" class="page libraryPage" data-backbutton="true">
|
<div id="videoOsdPage" data-role="page" class="page libraryPage" data-backbutton="true">
|
||||||
<div class="pageContainer flex"></div>
|
|
||||||
<div class="upNextContainer hide"></div>
|
<div class="upNextContainer hide"></div>
|
||||||
<div class="videoOsdBottom videoOsdBottom-maincontrols">
|
<div class="videoOsdBottom videoOsdBottom-maincontrols">
|
||||||
<div class="osdPoster"></div>
|
<div class="osdPoster"></div>
|
||||||
|
|
|
@ -172,9 +172,7 @@ import 'css!assets/css/videoosd';
|
||||||
}
|
}
|
||||||
|
|
||||||
setTitle(displayItem, parentName);
|
setTitle(displayItem, parentName);
|
||||||
let titleElement;
|
const titleElement = view.querySelector('.osdTitle');
|
||||||
const osdTitle = view.querySelector('.osdTitle');
|
|
||||||
titleElement = osdTitle;
|
|
||||||
let displayName = itemHelper.getDisplayName(displayItem, {
|
let displayName = itemHelper.getDisplayName(displayItem, {
|
||||||
includeParentInfo: displayItem.Type !== 'Program',
|
includeParentInfo: displayItem.Type !== 'Program',
|
||||||
includeIndexNumber: displayItem.Type !== 'Program'
|
includeIndexNumber: displayItem.Type !== 'Program'
|
||||||
|
@ -1619,7 +1617,7 @@ import 'css!assets/css/videoosd';
|
||||||
const item = currentItem;
|
const item = currentItem;
|
||||||
|
|
||||||
if (item && item.Chapters && item.Chapters.length && item.Chapters[0].ImageTag) {
|
if (item && item.Chapters && item.Chapters.length && item.Chapters[0].ImageTag) {
|
||||||
let html = getChapterBubbleHtml(connectionManager.getApiClient(item.ServerId), item, item.Chapters, ticks);
|
const html = getChapterBubbleHtml(connectionManager.getApiClient(item.ServerId), item, item.Chapters, ticks);
|
||||||
|
|
||||||
if (html) {
|
if (html) {
|
||||||
return html;
|
return html;
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
<div id="searchPage" data-role="page" class="page libraryPage allLibraryPage noSecondaryNavPage" data-title="${ButtonSearch}" data-backbutton="true">
|
<div id="searchPage" data-role="page" class="page libraryPage allLibraryPage noSecondaryNavPage" data-title="${Search}" data-backbutton="true">
|
||||||
<div class="padded-left padded-right searchFields"></div>
|
<div class="padded-left padded-right searchFields"></div>
|
||||||
<div class="searchResults padded-bottom-page padded-top"></div>
|
<div class="searchResults padded-bottom-page padded-top"></div>
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -19,7 +19,7 @@ import 'emby-button';
|
||||||
return [{
|
return [{
|
||||||
name: globalize.translate('TabShows')
|
name: globalize.translate('TabShows')
|
||||||
}, {
|
}, {
|
||||||
name: globalize.translate('TabSuggestions')
|
name: globalize.translate('Suggestions')
|
||||||
}, {
|
}, {
|
||||||
name: globalize.translate('TabLatest')
|
name: globalize.translate('TabLatest')
|
||||||
}, {
|
}, {
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
<div id="displayPreferencesPage" data-role="page" class="page libraryPage userPreferencesPage noSecondaryNavPage" data-title="${HeaderDisplay}" data-backbutton="true">
|
<div id="displayPreferencesPage" data-role="page" class="page libraryPage userPreferencesPage noSecondaryNavPage" data-title="${Display}" data-backbutton="true">
|
||||||
<div class="settingsContainer padded-left padded-right padded-bottom-page">
|
<div class="settingsContainer padded-left padded-right padded-bottom-page">
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -16,7 +16,7 @@
|
||||||
<div class="listItem">
|
<div class="listItem">
|
||||||
<span class="material-icons listItemIcon listItemIcon-transparent tv"></span>
|
<span class="material-icons listItemIcon listItemIcon-transparent tv"></span>
|
||||||
<div class="listItemBody">
|
<div class="listItemBody">
|
||||||
<div class="listItemBodyText">${HeaderDisplay}</div>
|
<div class="listItemBodyText">${Display}</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</a>
|
</a>
|
||||||
|
|
|
@ -32,7 +32,7 @@
|
||||||
</div>
|
</div>
|
||||||
<div>
|
<div>
|
||||||
<button is="emby-button" type="submit" class="raised button-submit block">
|
<button is="emby-button" type="submit" class="raised button-submit block">
|
||||||
<span>${ButtonSave}</span>
|
<span>${Save}</span>
|
||||||
</button>
|
</button>
|
||||||
<button is="emby-button" type="button" id="btnResetPassword" class="raised cancel block hide">
|
<button is="emby-button" type="button" id="btnResetPassword" class="raised cancel block hide">
|
||||||
<span>${ButtonResetPassword}</span>
|
<span>${ButtonResetPassword}</span>
|
||||||
|
@ -57,7 +57,7 @@
|
||||||
</div>
|
</div>
|
||||||
<div>
|
<div>
|
||||||
<button is="emby-button" type="submit" class="raised button-submit block">
|
<button is="emby-button" type="submit" class="raised button-submit block">
|
||||||
<span>${ButtonSave}</span>
|
<span>${Save}</span>
|
||||||
</button>
|
</button>
|
||||||
<button is="emby-button" type="button" id="btnResetEasyPassword" class="raised cancel block hide">
|
<button is="emby-button" type="button" id="btnResetEasyPassword" class="raised cancel block hide">
|
||||||
<span>${ButtonResetEasyPassword}</span>
|
<span>${ButtonResetEasyPassword}</span>
|
||||||
|
|
|
@ -3,7 +3,7 @@
|
||||||
<div class="ui-corner-all ui-shadow wizardContent">
|
<div class="ui-corner-all ui-shadow wizardContent">
|
||||||
<div>
|
<div>
|
||||||
<h1 style="display:inline-block;">${HeaderSetupLibrary}</h1>
|
<h1 style="display:inline-block;">${HeaderSetupLibrary}</h1>
|
||||||
<a href="https://docs.jellyfin.org/general/server/libraries.html" rel="noopener noreferrer" target="_blank" class="clearLink" style="margin-top:-10px;display:inline-block;vertical-align:middle;margin-left:2em;"><button is="emby-button" type="button" class="raised"><span class="material-icons info"></span><span>${ButtonHelp}</span></button></a>
|
<a href="https://docs.jellyfin.org/general/server/libraries.html" rel="noopener noreferrer" target="_blank" class="clearLink" style="margin-top:-10px;display:inline-block;vertical-align:middle;margin-left:2em;"><button is="emby-button" type="button" class="raised"><span class="material-icons info"></span><span>${Help}</span></button></a>
|
||||||
</div>
|
</div>
|
||||||
<br />
|
<br />
|
||||||
<div id="divVirtualFolders"></div>
|
<div id="divVirtualFolders"></div>
|
||||||
|
|
|
@ -5,7 +5,7 @@ import 'webcomponents';
|
||||||
|
|
||||||
/* eslint-disable indent */
|
/* eslint-disable indent */
|
||||||
|
|
||||||
let EmbyCheckboxPrototype = Object.create(HTMLInputElement.prototype);
|
const EmbyCheckboxPrototype = Object.create(HTMLInputElement.prototype);
|
||||||
|
|
||||||
function onKeyDown(e) {
|
function onKeyDown(e) {
|
||||||
// Don't submit form on enter
|
// Don't submit form on enter
|
||||||
|
@ -26,7 +26,7 @@ import 'webcomponents';
|
||||||
const enableRefreshHack = browser.tizen || browser.orsay || browser.operaTv || browser.web0s ? true : false;
|
const enableRefreshHack = browser.tizen || browser.orsay || browser.operaTv || browser.web0s ? true : false;
|
||||||
|
|
||||||
function forceRefresh(loading) {
|
function forceRefresh(loading) {
|
||||||
let elem = this.parentNode;
|
const elem = this.parentNode;
|
||||||
|
|
||||||
elem.style.webkitAnimationName = 'repaintChrome';
|
elem.style.webkitAnimationName = 'repaintChrome';
|
||||||
elem.style.webkitAnimationDelay = (loading === true ? '500ms' : '');
|
elem.style.webkitAnimationDelay = (loading === true ? '500ms' : '');
|
||||||
|
|
|
@ -40,7 +40,7 @@ import 'webcomponents';
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
let EmbyItemRefreshIndicatorPrototype = Object.create(EmbyProgressRing);
|
const EmbyItemRefreshIndicatorPrototype = Object.create(EmbyProgressRing);
|
||||||
|
|
||||||
EmbyItemRefreshIndicatorPrototype.createdCallback = function () {
|
EmbyItemRefreshIndicatorPrototype.createdCallback = function () {
|
||||||
// base method
|
// base method
|
||||||
|
|
|
@ -18,7 +18,7 @@ import 'webcomponents';
|
||||||
|
|
||||||
function onClick(e) {
|
function onClick(e) {
|
||||||
const itemsContainer = this;
|
const itemsContainer = this;
|
||||||
let multiSelect = itemsContainer.multiSelect;
|
const multiSelect = itemsContainer.multiSelect;
|
||||||
|
|
||||||
if (multiSelect) {
|
if (multiSelect) {
|
||||||
if (multiSelect.onContainerClick.call(itemsContainer, e) === false) {
|
if (multiSelect.onContainerClick.call(itemsContainer, e) === false) {
|
||||||
|
@ -164,7 +164,7 @@ import 'webcomponents';
|
||||||
}
|
}
|
||||||
|
|
||||||
function getEventsToMonitor(itemsContainer) {
|
function getEventsToMonitor(itemsContainer) {
|
||||||
let monitor = itemsContainer.getAttribute('data-monitor');
|
const monitor = itemsContainer.getAttribute('data-monitor');
|
||||||
if (monitor) {
|
if (monitor) {
|
||||||
return monitor.split(',');
|
return monitor.split(',');
|
||||||
}
|
}
|
||||||
|
@ -356,7 +356,7 @@ import 'webcomponents';
|
||||||
ItemsContainerPrototype.resume = function (options) {
|
ItemsContainerPrototype.resume = function (options) {
|
||||||
this.paused = false;
|
this.paused = false;
|
||||||
|
|
||||||
let refreshIntervalEndTime = this.refreshIntervalEndTime;
|
const refreshIntervalEndTime = this.refreshIntervalEndTime;
|
||||||
if (refreshIntervalEndTime) {
|
if (refreshIntervalEndTime) {
|
||||||
const remainingMs = refreshIntervalEndTime - new Date().getTime();
|
const remainingMs = refreshIntervalEndTime - new Date().getTime();
|
||||||
if (remainingMs > 0 && !this.needsRefresh) {
|
if (remainingMs > 0 && !this.needsRefresh) {
|
||||||
|
@ -395,7 +395,7 @@ import 'webcomponents';
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
let timeout = this.refreshTimeout;
|
const timeout = this.refreshTimeout;
|
||||||
if (timeout) {
|
if (timeout) {
|
||||||
clearTimeout(timeout);
|
clearTimeout(timeout);
|
||||||
}
|
}
|
||||||
|
@ -434,7 +434,7 @@ import 'webcomponents';
|
||||||
function onDataFetched(result) {
|
function onDataFetched(result) {
|
||||||
const items = result.Items || result;
|
const items = result.Items || result;
|
||||||
|
|
||||||
let parentContainer = this.parentContainer;
|
const parentContainer = this.parentContainer;
|
||||||
if (parentContainer) {
|
if (parentContainer) {
|
||||||
if (items.length) {
|
if (items.length) {
|
||||||
parentContainer.classList.remove('hide');
|
parentContainer.classList.remove('hide');
|
||||||
|
|
|
@ -75,7 +75,7 @@ import EmbyButtonPrototype from 'emby-button';
|
||||||
button.title = globalize.translate('Played');
|
button.title = globalize.translate('Played');
|
||||||
}
|
}
|
||||||
|
|
||||||
let text = button.querySelector('.button-text');
|
const text = button.querySelector('.button-text');
|
||||||
if (text) {
|
if (text) {
|
||||||
text.innerHTML = button.title;
|
text.innerHTML = button.title;
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
/* eslint-disable indent */
|
/* eslint-disable indent */
|
||||||
|
|
||||||
let ProgressBarPrototype = Object.create(HTMLDivElement.prototype);
|
const ProgressBarPrototype = Object.create(HTMLDivElement.prototype);
|
||||||
|
|
||||||
function onAutoTimeProgress() {
|
function onAutoTimeProgress() {
|
||||||
const start = parseInt(this.getAttribute('data-starttime'));
|
const start = parseInt(this.getAttribute('data-starttime'));
|
||||||
|
|
|
@ -3,7 +3,7 @@ import 'webcomponents';
|
||||||
|
|
||||||
/* eslint-disable indent */
|
/* eslint-disable indent */
|
||||||
|
|
||||||
let EmbyProgressRing = Object.create(HTMLDivElement.prototype);
|
const EmbyProgressRing = Object.create(HTMLDivElement.prototype);
|
||||||
|
|
||||||
EmbyProgressRing.createdCallback = function () {
|
EmbyProgressRing.createdCallback = function () {
|
||||||
this.classList.add('progressring');
|
this.classList.add('progressring');
|
||||||
|
@ -79,7 +79,7 @@ import 'webcomponents';
|
||||||
};
|
};
|
||||||
|
|
||||||
EmbyProgressRing.detachedCallback = function () {
|
EmbyProgressRing.detachedCallback = function () {
|
||||||
let observer = this.observer;
|
const observer = this.observer;
|
||||||
|
|
||||||
if (observer) {
|
if (observer) {
|
||||||
// later, you can stop observing
|
// later, you can stop observing
|
||||||
|
|
|
@ -4,7 +4,7 @@ import 'webcomponents';
|
||||||
|
|
||||||
/* eslint-disable indent */
|
/* eslint-disable indent */
|
||||||
|
|
||||||
let EmbyRadioPrototype = Object.create(HTMLInputElement.prototype);
|
const EmbyRadioPrototype = Object.create(HTMLInputElement.prototype);
|
||||||
|
|
||||||
function onKeyDown(e) {
|
function onKeyDown(e) {
|
||||||
// Don't submit form on enter
|
// Don't submit form on enter
|
||||||
|
@ -35,7 +35,7 @@ import 'webcomponents';
|
||||||
|
|
||||||
this.classList.add('mdl-radio__button');
|
this.classList.add('mdl-radio__button');
|
||||||
|
|
||||||
let labelElement = this.parentNode;
|
const labelElement = this.parentNode;
|
||||||
labelElement.classList.add('mdl-radio');
|
labelElement.classList.add('mdl-radio');
|
||||||
labelElement.classList.add('mdl-js-radio');
|
labelElement.classList.add('mdl-js-radio');
|
||||||
labelElement.classList.add('mdl-js-ripple-effect');
|
labelElement.classList.add('mdl-js-ripple-effect');
|
||||||
|
@ -43,7 +43,7 @@ import 'webcomponents';
|
||||||
labelElement.classList.add('show-focus');
|
labelElement.classList.add('show-focus');
|
||||||
}
|
}
|
||||||
|
|
||||||
let labelTextElement = labelElement.querySelector('span');
|
const labelTextElement = labelElement.querySelector('span');
|
||||||
|
|
||||||
labelTextElement.classList.add('radioButtonLabel');
|
labelTextElement.classList.add('radioButtonLabel');
|
||||||
labelTextElement.classList.add('mdl-radio__label');
|
labelTextElement.classList.add('mdl-radio__label');
|
||||||
|
|
|
@ -118,7 +118,7 @@ const EmbyScrollButtonsPrototype = Object.create(HTMLDivElement.prototype);
|
||||||
}
|
}
|
||||||
|
|
||||||
function onScrollButtonClick(e) {
|
function onScrollButtonClick(e) {
|
||||||
let scroller = this.parentNode.nextSibling;
|
const scroller = this.parentNode.nextSibling;
|
||||||
|
|
||||||
const direction = this.getAttribute('data-direction');
|
const direction = this.getAttribute('data-direction');
|
||||||
const scrollSize = getScrollSize(scroller);
|
const scrollSize = getScrollSize(scroller);
|
||||||
|
@ -161,7 +161,7 @@ const EmbyScrollButtonsPrototype = Object.create(HTMLDivElement.prototype);
|
||||||
const parent = this.scroller;
|
const parent = this.scroller;
|
||||||
this.scroller = null;
|
this.scroller = null;
|
||||||
|
|
||||||
let scrollHandler = this.scrollHandler;
|
const scrollHandler = this.scrollHandler;
|
||||||
if (parent && scrollHandler) {
|
if (parent && scrollHandler) {
|
||||||
parent.removeScrollEventListener(scrollHandler, {
|
parent.removeScrollEventListener(scrollHandler, {
|
||||||
capture: false,
|
capture: false,
|
||||||
|
|
|
@ -9,7 +9,7 @@ import 'css!./emby-scroller';
|
||||||
|
|
||||||
/* eslint-disable indent */
|
/* eslint-disable indent */
|
||||||
|
|
||||||
let ScrollerPrototype = Object.create(HTMLDivElement.prototype);
|
const ScrollerPrototype = Object.create(HTMLDivElement.prototype);
|
||||||
|
|
||||||
ScrollerPrototype.createdCallback = function () {
|
ScrollerPrototype.createdCallback = function () {
|
||||||
this.classList.add('emby-scroller');
|
this.classList.add('emby-scroller');
|
||||||
|
|
|
@ -8,7 +8,7 @@ import 'emby-input';
|
||||||
|
|
||||||
/* eslint-disable indent */
|
/* eslint-disable indent */
|
||||||
|
|
||||||
let EmbySliderPrototype = Object.create(HTMLInputElement.prototype);
|
const EmbySliderPrototype = Object.create(HTMLInputElement.prototype);
|
||||||
|
|
||||||
let supportsValueSetOverride = false;
|
let supportsValueSetOverride = false;
|
||||||
|
|
||||||
|
@ -94,7 +94,7 @@ import 'emby-input';
|
||||||
// Keep only one per slider frame request
|
// Keep only one per slider frame request
|
||||||
cancelAnimationFrame(range.updateValuesFrame);
|
cancelAnimationFrame(range.updateValuesFrame);
|
||||||
range.updateValuesFrame = requestAnimationFrame(function () {
|
range.updateValuesFrame = requestAnimationFrame(function () {
|
||||||
let backgroundLower = range.backgroundLower;
|
const backgroundLower = range.backgroundLower;
|
||||||
|
|
||||||
if (backgroundLower) {
|
if (backgroundLower) {
|
||||||
let fraction = (value - range.min) / (range.max - range.min);
|
let fraction = (value - range.min) / (range.max - range.min);
|
||||||
|
|
|
@ -8,7 +8,7 @@ import 'scrollStyles';
|
||||||
|
|
||||||
/* eslint-disable indent */
|
/* eslint-disable indent */
|
||||||
|
|
||||||
let EmbyTabs = Object.create(HTMLDivElement.prototype);
|
const EmbyTabs = Object.create(HTMLDivElement.prototype);
|
||||||
const buttonClass = 'emby-tab-button';
|
const buttonClass = 'emby-tab-button';
|
||||||
const activeButtonClass = buttonClass + '-active';
|
const activeButtonClass = buttonClass + '-active';
|
||||||
|
|
||||||
|
@ -21,7 +21,7 @@ import 'scrollStyles';
|
||||||
}
|
}
|
||||||
|
|
||||||
function removeActivePanelClass(tabs, index) {
|
function removeActivePanelClass(tabs, index) {
|
||||||
let tabPanel = getTabPanel(tabs, index);
|
const tabPanel = getTabPanel(tabs, index);
|
||||||
if (tabPanel) {
|
if (tabPanel) {
|
||||||
tabPanel.classList.remove('is-active');
|
tabPanel.classList.remove('is-active');
|
||||||
}
|
}
|
||||||
|
@ -52,7 +52,7 @@ import 'scrollStyles';
|
||||||
removeActivePanelClass(tabs, previousIndex);
|
removeActivePanelClass(tabs, previousIndex);
|
||||||
}
|
}
|
||||||
|
|
||||||
let newPanel = getTabPanel(tabs, index);
|
const newPanel = getTabPanel(tabs, index);
|
||||||
|
|
||||||
if (newPanel) {
|
if (newPanel) {
|
||||||
// animate new panel ?
|
// animate new panel ?
|
||||||
|
@ -225,7 +225,7 @@ import 'scrollStyles';
|
||||||
}
|
}
|
||||||
}));
|
}));
|
||||||
|
|
||||||
let currentTabButton = tabButtons[current];
|
const currentTabButton = tabButtons[current];
|
||||||
setActiveTabButton(tabs, tabButtons[selected], currentTabButton, false);
|
setActiveTabButton(tabs, tabButtons[selected], currentTabButton, false);
|
||||||
|
|
||||||
if (current !== selected && currentTabButton) {
|
if (current !== selected && currentTabButton) {
|
||||||
|
|
|
@ -27,16 +27,16 @@ export class BookPlayer {
|
||||||
this._loaded = false;
|
this._loaded = false;
|
||||||
|
|
||||||
loading.show();
|
loading.show();
|
||||||
let elem = this.createMediaElement();
|
const elem = this.createMediaElement();
|
||||||
return this.setCurrentSrc(elem, options);
|
return this.setCurrentSrc(elem, options);
|
||||||
}
|
}
|
||||||
|
|
||||||
stop() {
|
stop() {
|
||||||
this.unbindEvents();
|
this.unbindEvents();
|
||||||
|
|
||||||
let elem = this._mediaElement;
|
const elem = this._mediaElement;
|
||||||
let tocElement = this._tocElement;
|
const tocElement = this._tocElement;
|
||||||
let rendition = this._rendition;
|
const rendition = this._rendition;
|
||||||
|
|
||||||
if (elem) {
|
if (elem) {
|
||||||
dialogHelper.close(elem);
|
dialogHelper.close(elem);
|
||||||
|
@ -93,11 +93,11 @@ export class BookPlayer {
|
||||||
}
|
}
|
||||||
|
|
||||||
onWindowKeyUp(e) {
|
onWindowKeyUp(e) {
|
||||||
let key = keyboardnavigation.getKeyName(e);
|
const key = keyboardnavigation.getKeyName(e);
|
||||||
|
|
||||||
// TODO: depending on the event this can be the document or the rendition itself
|
// TODO: depending on the event this can be the document or the rendition itself
|
||||||
let rendition = this._rendition || this;
|
const rendition = this._rendition || this;
|
||||||
let book = rendition.book;
|
const book = rendition.book;
|
||||||
|
|
||||||
if (this._loaded === false) return;
|
if (this._loaded === false) return;
|
||||||
switch (key) {
|
switch (key) {
|
||||||
|
@ -125,8 +125,8 @@ export class BookPlayer {
|
||||||
|
|
||||||
onTouchStart(e) {
|
onTouchStart(e) {
|
||||||
// TODO: depending on the event this can be the document or the rendition itself
|
// TODO: depending on the event this can be the document or the rendition itself
|
||||||
let rendition = this._rendition || this;
|
const rendition = this._rendition || this;
|
||||||
let book = rendition.book;
|
const book = rendition.book;
|
||||||
|
|
||||||
// check that the event is from the book or the document
|
// check that the event is from the book or the document
|
||||||
if (!book || this._loaded === false) return;
|
if (!book || this._loaded === false) return;
|
||||||
|
@ -134,7 +134,8 @@ export class BookPlayer {
|
||||||
// epubjs stores pages off the screen or something for preloading
|
// epubjs stores pages off the screen or something for preloading
|
||||||
// get the modulus of the touch event to account for the increased width
|
// get the modulus of the touch event to account for the increased width
|
||||||
if (!e.touches || e.touches.length === 0) return;
|
if (!e.touches || e.touches.length === 0) return;
|
||||||
let touch = e.touches[0].clientX % dom.getWindowSize().innerWidth;
|
|
||||||
|
const touch = e.touches[0].clientX % dom.getWindowSize().innerWidth;
|
||||||
if (touch < dom.getWindowSize().innerWidth / 2) {
|
if (touch < dom.getWindowSize().innerWidth / 2) {
|
||||||
book.package.metadata.direction === 'rtl' ? rendition.next() : rendition.prev();
|
book.package.metadata.direction === 'rtl' ? rendition.next() : rendition.prev();
|
||||||
} else {
|
} else {
|
||||||
|
@ -147,7 +148,7 @@ export class BookPlayer {
|
||||||
}
|
}
|
||||||
|
|
||||||
bindMediaElementEvents() {
|
bindMediaElementEvents() {
|
||||||
let elem = this._mediaElement;
|
const elem = this._mediaElement;
|
||||||
|
|
||||||
elem.addEventListener('close', this.onDialogClosed, {once: true});
|
elem.addEventListener('close', this.onDialogClosed, {once: true});
|
||||||
elem.querySelector('.btnBookplayerExit').addEventListener('click', this.onDialogClosed, {once: true});
|
elem.querySelector('.btnBookplayerExit').addEventListener('click', this.onDialogClosed, {once: true});
|
||||||
|
@ -166,7 +167,7 @@ export class BookPlayer {
|
||||||
}
|
}
|
||||||
|
|
||||||
unbindMediaElementEvents() {
|
unbindMediaElementEvents() {
|
||||||
let elem = this._mediaElement;
|
const elem = this._mediaElement;
|
||||||
|
|
||||||
elem.removeEventListener('close', this.onDialogClosed);
|
elem.removeEventListener('close', this.onDialogClosed);
|
||||||
elem.querySelector('.btnBookplayerExit').removeEventListener('click', this.onDialogClosed);
|
elem.querySelector('.btnBookplayerExit').removeEventListener('click', this.onDialogClosed);
|
||||||
|
@ -231,7 +232,7 @@ export class BookPlayer {
|
||||||
}
|
}
|
||||||
|
|
||||||
setCurrentSrc(elem, options) {
|
setCurrentSrc(elem, options) {
|
||||||
let item = options.items[0];
|
const item = options.items[0];
|
||||||
this._currentItem = item;
|
this._currentItem = item;
|
||||||
this.streamInfo = {
|
this.streamInfo = {
|
||||||
started: true,
|
started: true,
|
||||||
|
@ -241,25 +242,25 @@ export class BookPlayer {
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
let serverId = item.ServerId;
|
const serverId = item.ServerId;
|
||||||
let apiClient = connectionManager.getApiClient(serverId);
|
const apiClient = connectionManager.getApiClient(serverId);
|
||||||
|
|
||||||
return new Promise((resolve, reject) => {
|
return new Promise((resolve, reject) => {
|
||||||
import('epubjs').then(({default: epubjs}) => {
|
import('epubjs').then(({default: epubjs}) => {
|
||||||
let downloadHref = apiClient.getItemDownloadUrl(item.Id);
|
const downloadHref = apiClient.getItemDownloadUrl(item.Id);
|
||||||
let book = epubjs(downloadHref, {openAs: 'epub'});
|
const book = epubjs(downloadHref, {openAs: 'epub'});
|
||||||
let rendition = book.renderTo(elem, {width: '100%', height: '97%'});
|
const rendition = book.renderTo(elem, {width: '100%', height: '97%'});
|
||||||
|
|
||||||
this._currentSrc = downloadHref;
|
this._currentSrc = downloadHref;
|
||||||
this._rendition = rendition;
|
this._rendition = rendition;
|
||||||
let cancellationToken = {
|
const cancellationToken = {
|
||||||
shouldCancel: false
|
shouldCancel: false
|
||||||
};
|
};
|
||||||
|
|
||||||
this._cancellationToken = cancellationToken;
|
this._cancellationToken = cancellationToken;
|
||||||
|
|
||||||
return rendition.display().then(() => {
|
return rendition.display().then(() => {
|
||||||
let epubElem = document.querySelector('.epub-container');
|
const epubElem = document.querySelector('.epub-container');
|
||||||
epubElem.style.display = 'none';
|
epubElem.style.display = 'none';
|
||||||
|
|
||||||
this.bindEvents();
|
this.bindEvents();
|
||||||
|
|
|
@ -11,7 +11,7 @@ export default class TableOfContents {
|
||||||
}
|
}
|
||||||
|
|
||||||
destroy() {
|
destroy() {
|
||||||
let elem = this._elem;
|
const elem = this._elem;
|
||||||
if (elem) {
|
if (elem) {
|
||||||
this.unbindEvents();
|
this.unbindEvents();
|
||||||
dialogHelper.close(elem);
|
dialogHelper.close(elem);
|
||||||
|
@ -21,14 +21,14 @@ export default class TableOfContents {
|
||||||
}
|
}
|
||||||
|
|
||||||
bindEvents() {
|
bindEvents() {
|
||||||
let elem = this._elem;
|
const elem = this._elem;
|
||||||
|
|
||||||
elem.addEventListener('close', this.onDialogClosed, {once: true});
|
elem.addEventListener('close', this.onDialogClosed, {once: true});
|
||||||
elem.querySelector('.btnBookplayerTocClose').addEventListener('click', this.onDialogClosed, {once: true});
|
elem.querySelector('.btnBookplayerTocClose').addEventListener('click', this.onDialogClosed, {once: true});
|
||||||
}
|
}
|
||||||
|
|
||||||
unbindEvents() {
|
unbindEvents() {
|
||||||
let elem = this._elem;
|
const elem = this._elem;
|
||||||
|
|
||||||
elem.removeEventListener('close', this.onDialogClosed);
|
elem.removeEventListener('close', this.onDialogClosed);
|
||||||
elem.querySelector('.btnBookplayerTocClose').removeEventListener('click', this.onDialogClosed);
|
elem.querySelector('.btnBookplayerTocClose').removeEventListener('click', this.onDialogClosed);
|
||||||
|
@ -39,10 +39,10 @@ export default class TableOfContents {
|
||||||
}
|
}
|
||||||
|
|
||||||
replaceLinks(contents, f) {
|
replaceLinks(contents, f) {
|
||||||
let links = contents.querySelectorAll('a[href]');
|
const links = contents.querySelectorAll('a[href]');
|
||||||
|
|
||||||
links.forEach((link) => {
|
links.forEach((link) => {
|
||||||
let href = link.getAttribute('href');
|
const href = link.getAttribute('href');
|
||||||
|
|
||||||
link.onclick = () => {
|
link.onclick = () => {
|
||||||
f(href);
|
f(href);
|
||||||
|
@ -52,9 +52,9 @@ export default class TableOfContents {
|
||||||
}
|
}
|
||||||
|
|
||||||
createMediaElement() {
|
createMediaElement() {
|
||||||
let rendition = this._rendition;
|
const rendition = this._rendition;
|
||||||
|
|
||||||
let elem = dialogHelper.createDialog({
|
const elem = dialogHelper.createDialog({
|
||||||
size: 'small',
|
size: 'small',
|
||||||
autoFocus: false,
|
autoFocus: false,
|
||||||
removeOnClose: true
|
removeOnClose: true
|
||||||
|
@ -69,7 +69,7 @@ export default class TableOfContents {
|
||||||
rendition.book.navigation.forEach((chapter) => {
|
rendition.book.navigation.forEach((chapter) => {
|
||||||
tocHtml += '<li>';
|
tocHtml += '<li>';
|
||||||
// Remove '../' from href
|
// Remove '../' from href
|
||||||
let link = chapter.href.startsWith('../') ? chapter.href.substr(3) : chapter.href;
|
const link = chapter.href.startsWith('../') ? chapter.href.substr(3) : chapter.href;
|
||||||
tocHtml += `<a href="${rendition.book.path.directory + link}">${chapter.label}</a>`;
|
tocHtml += `<a href="${rendition.book.path.directory + link}">${chapter.label}</a>`;
|
||||||
tocHtml += '</li>';
|
tocHtml += '</li>';
|
||||||
});
|
});
|
||||||
|
@ -78,7 +78,7 @@ export default class TableOfContents {
|
||||||
elem.innerHTML = tocHtml;
|
elem.innerHTML = tocHtml;
|
||||||
|
|
||||||
this.replaceLinks(elem, (href) => {
|
this.replaceLinks(elem, (href) => {
|
||||||
let relative = rendition.book.path.relative(href);
|
const relative = rendition.book.path.relative(href);
|
||||||
rendition.display(relative);
|
rendition.display(relative);
|
||||||
this.destroy();
|
this.destroy();
|
||||||
});
|
});
|
||||||
|
|
|
@ -1,6 +1,8 @@
|
||||||
define(['connectionManager', 'globalize', 'userSettings', 'apphost'], function (connectionManager, globalize, userSettings, appHost) {
|
define(['connectionManager', 'globalize', 'userSettings', 'apphost'], function (connectionManager, globalize, userSettings, appHost) {
|
||||||
'use strict';
|
'use strict';
|
||||||
|
|
||||||
|
appHost = appHost.default || appHost;
|
||||||
|
|
||||||
// TODO: Replace with date-fns
|
// TODO: Replace with date-fns
|
||||||
// https://stackoverflow.com/questions/6117814/get-week-of-year-in-javascript-like-in-php
|
// https://stackoverflow.com/questions/6117814/get-week-of-year-in-javascript-like-in-php
|
||||||
function getWeek(date) {
|
function getWeek(date) {
|
||||||
|
|
|
@ -105,7 +105,7 @@ function tryRemoveElement(elem) {
|
||||||
}
|
}
|
||||||
|
|
||||||
function hidePrePlaybackPage() {
|
function hidePrePlaybackPage() {
|
||||||
let animatedPage = document.querySelector('.page:not(.hide)');
|
const animatedPage = document.querySelector('.page:not(.hide)');
|
||||||
animatedPage.classList.add('hide');
|
animatedPage.classList.add('hide');
|
||||||
// At this point, we must hide the scrollbar placeholder, so it's not being displayed while the item is being loaded
|
// At this point, we must hide the scrollbar placeholder, so it's not being displayed while the item is being loaded
|
||||||
document.body.classList.remove('force-scroll');
|
document.body.classList.remove('force-scroll');
|
||||||
|
@ -1299,7 +1299,7 @@ function tryRemoveElement(elem) {
|
||||||
}
|
}
|
||||||
|
|
||||||
let html = '';
|
let html = '';
|
||||||
let cssClass = 'htmlvideoplayer';
|
const cssClass = 'htmlvideoplayer';
|
||||||
|
|
||||||
// Can't autoplay in these browsers so we need to use the full controls, at least until playback starts
|
// Can't autoplay in these browsers so we need to use the full controls, at least until playback starts
|
||||||
if (!appHost.supports('htmlvideoautoplay')) {
|
if (!appHost.supports('htmlvideoautoplay')) {
|
||||||
|
|
|
@ -1,12 +1,10 @@
|
||||||
(function() {
|
(function() {
|
||||||
'use strict';
|
|
||||||
|
|
||||||
function injectScriptElement(src, onload) {
|
function injectScriptElement(src, onload) {
|
||||||
if (!src) {
|
if (!src) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
var script = document.createElement('script');
|
const script = document.createElement('script');
|
||||||
if (self.dashboardVersion) {
|
if (self.dashboardVersion) {
|
||||||
src += `?v=${self.dashboardVersion}`;
|
src += `?v=${self.dashboardVersion}`;
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,6 +3,10 @@ import skinManager from 'skinManager';
|
||||||
import connectionManager from 'connectionManager';
|
import connectionManager from 'connectionManager';
|
||||||
import events from 'events';
|
import events from 'events';
|
||||||
|
|
||||||
|
// Set the default theme when loading
|
||||||
|
skinManager.setTheme(userSettings.theme());
|
||||||
|
|
||||||
|
// Set the user's prefered theme when signing in
|
||||||
events.on(connectionManager, 'localusersignedin', function (e, user) {
|
events.on(connectionManager, 'localusersignedin', function (e, user) {
|
||||||
skinManager.setTheme(userSettings.theme());
|
skinManager.setTheme(userSettings.theme());
|
||||||
});
|
});
|
||||||
|
|
|
@ -24,7 +24,7 @@ import globalize from 'globalize';
|
||||||
|
|
||||||
// parse strings, leading zeros into proper ints
|
// parse strings, leading zeros into proper ints
|
||||||
const a = [1, 2, 3, 4, 5, 6, 10, 11];
|
const a = [1, 2, 3, 4, 5, 6, 10, 11];
|
||||||
for (let i in a) {
|
for (const i in a) {
|
||||||
d[a[i]] = parseInt(d[a[i]], 10);
|
d[a[i]] = parseInt(d[a[i]], 10);
|
||||||
}
|
}
|
||||||
d[7] = parseFloat(d[7]);
|
d[7] = parseFloat(d[7]);
|
||||||
|
|
Some files were not shown because too many files have changed in this diff Show more
Loading…
Add table
Add a link
Reference in a new issue