diff --git a/themes-default/slim/static/js/home/display-show.js b/themes-default/slim/static/js/home/display-show.js deleted file mode 100644 index bef3a32e57..0000000000 --- a/themes-default/slim/static/js/home/display-show.js +++ /dev/null @@ -1,441 +0,0 @@ -MEDUSA.home.displayShow = function() { // eslint-disable-line max-lines - $('.imdbPlot').on('click', function() { - $(this).prev('span').toggle(); - if ($(this).html() === '..show less') { - $(this).html('..show more'); - } else { - $(this).html('..show less'); - } - moveSummaryBackground(); - movecheckboxControlsBackground(); - }); - - // Adjust the summary background position and size on page load and resize - function moveSummaryBackground() { - const height = $('#summary').height() + 10; - const top = $('#summary').offset().top + 5; - $('#summaryBackground').height(height); - $('#summaryBackground').offset({ top, left: 0 }); - $('#summaryBackground').show(); - } - - function movecheckboxControlsBackground() { - const height = $('#checkboxControls').height() + 10; - const top = $('#checkboxControls').offset().top - 3; - $('#checkboxControlsBackground').height(height); - $('#checkboxControlsBackground').offset({ top, left: 0 }); - $('#checkboxControlsBackground').show(); - } - - $(window).resize(() => { - moveSummaryBackground(); - movecheckboxControlsBackground(); - }); - - $(() => { - moveSummaryBackground(); - movecheckboxControlsBackground(); - }); - - $.ajaxEpSearch({ - colorRow: true - }); - - startAjaxEpisodeSubtitles(); // eslint-disable-line no-undef - $.ajaxEpSubtitlesSearch(); - $.ajaxEpRedownloadSubtitle(); - - const setQuality = (quality, seriesSlug, episodes) => { - const patchData = {}; - episodes.forEach(episode => { - patchData[episode] = { quality: parseInt(quality, 10) }; - }); - - api.patch('series/' + seriesSlug + '/episodes', patchData).then(response => { - log.info(response.data); - window.location.reload(); - }).catch(error => { - log.error(error.data); - }); - }; - - $('#seasonJump').on('change', function() { - const id = $('#seasonJump option:selected').val(); - if (id && id !== 'jump') { - const season = $('#seasonJump option:selected').data('season'); - $('html,body').animate({ scrollTop: $('[name="' + id.substring(1) + '"]').offset().top - 100 }, 'slow'); - $('#collapseSeason-' + season).collapse('show'); - location.hash = id; - } - $(this).val('jump'); - }); - - $('#changeStatus').on('click', () => { - const epArr = []; - const status = $('#statusSelect').val(); - const quality = $('#qualitySelect').val(); - const seriesSlug = $('#series-slug').val(); - - $('.epCheck').each(function() { - if (this.checked === true) { - epArr.push($(this).attr('id')); - } - }); - - if (epArr.length === 0) { - return false; - } - - if (quality) { - setQuality(quality, seriesSlug, epArr); - } - - if (status) { - window.location.href = $('base').attr('href') + 'home/setStatus?' + - 'indexername=' + $('#indexer-name').attr('value') + - '&seriesid=' + $('#series-id').attr('value') + - '&eps=' + epArr.join('|') + - '&status=' + status; - } - }); - - $('.seasonCheck').on('click', function() { - const seasCheck = this; - const seasNo = $(seasCheck).attr('id'); - - $('#collapseSeason-' + seasNo).collapse('show'); - const seasonIdentifier = 's' + seasNo; - $('.epCheck:visible').each(function() { - const epParts = $(this).attr('id').split('e'); - if (epParts[0] === seasonIdentifier) { - this.checked = seasCheck.checked; - } - }); - }); - - let lastCheck = null; - $('.epCheck').on('click', function(event) { - if (!lastCheck || !event.shiftKey) { - lastCheck = this; - return; - } - - const check = this; - let found = 0; - - $('.epCheck').each(function() { - if (found === 1) { - this.checked = lastCheck.checked; - } - - if (found === 2) { - return false; - } - - if (this === check || this === lastCheck) { - found++; - } - }); - }); - - // Selects all visible episode checkboxes. - $('.seriesCheck').on('click', () => { - $('.epCheck:visible').each(function() { - this.checked = true; - }); - $('.seasonCheck:visible').each(function() { - this.checked = true; - }); - }); - - // Clears all visible episode checkboxes and the season selectors - $('.clearAll').on('click', () => { - $('.epCheck:visible').each(function() { - this.checked = false; - }); - $('.seasonCheck:visible').each(function() { - this.checked = false; - }); - }); - - // Show/hide different types of rows when the checkboxes are changed - $('#checkboxControls input').on('change', function() { - const whichClass = $(this).attr('id'); - $(this).showHideRows(whichClass); - }); - - // Initially show/hide all the rows according to the checkboxes - $('#checkboxControls input').each(function() { - const status = $(this).prop('checked'); - $('tr.' + $(this).attr('id')).each(function() { - if (status) { - $(this).show(); - } else { - $(this).hide(); - } - }); - }); - - $.fn.showHideRows = function(whichClass) { - const status = $('#checkboxControls > input, #' + whichClass).prop('checked'); - $('tr.' + whichClass).each(function() { - if (status) { - $(this).show(); - } else { - $(this).hide(); - } - }); - - // Hide season headers with no episodes under them - $('tr.seasonheader').each(function() { - let numRows = 0; - const seasonNo = $(this).attr('id'); - $('tr.' + seasonNo + ' :visible').each(() => { - numRows++; - }); - if (numRows === 0) { - $(this).hide(); - $('#' + seasonNo + '-cols').hide(); - } else { - $(this).show(); - $('#' + seasonNo + '-cols').show(); - } - }); - }; - - function setEpisodeSceneNumbering(forSeason, forEpisode, sceneSeason, sceneEpisode) { - const indexerName = $('#indexer-name').val(); - const seriesId = $('#series-id').val(); - - if (sceneSeason === '') { - sceneSeason = null; - } - if (sceneEpisode === '') { - sceneEpisode = null; - } - - $.getJSON('home/setSceneNumbering', { - indexername: indexerName, - seriesid: seriesId, - forSeason, - forEpisode, - sceneSeason, - sceneEpisode - }, data => { - // Set the values we get back - if (data.sceneSeason === null || data.sceneEpisode === null) { - $('#sceneSeasonXEpisode_' + seriesId + '_' + forSeason + '_' + forEpisode).val(''); - } else { - $('#sceneSeasonXEpisode_' + seriesId + '_' + forSeason + '_' + forEpisode).val(data.sceneSeason + 'x' + data.sceneEpisode); - } - if (!data.success) { - if (data.errorMessage) { - alert(data.errorMessage); // eslint-disable-line no-alert - } else { - alert('Update failed.'); // eslint-disable-line no-alert - } - } - }); - } - - function setAbsoluteSceneNumbering(forAbsolute, sceneAbsolute) { - const indexerName = $('#indexer-name').val(); - const seriesId = $('#series-id').val(); - - if (sceneAbsolute === '') { - sceneAbsolute = null; - } - - $.getJSON('home/setSceneNumbering', { - indexername: indexerName, - seriesid: seriesId, - forAbsolute, - sceneAbsolute - }, data => { - // Set the values we get back - if (data.sceneAbsolute === null) { - $('#sceneAbsolute_' + seriesId + '_' + forAbsolute).val(''); - } else { - $('#sceneAbsolute_' + seriesId + '_' + forAbsolute).val(data.sceneAbsolute); - } - - if (!data.success) { - if (data.errorMessage) { - alert(data.errorMessage); // eslint-disable-line no-alert - } else { - alert('Update failed.'); // eslint-disable-line no-alert - } - } - }); - } - - function setInputValidInvalid(valid, el) { - if (valid) { - $(el).css({ - 'background-color': '#90EE90', // Green - 'color': '#FFF', // eslint-disable-line quote-props - 'font-weight': 'bold' - }); - return true; - } - $(el).css({ - 'background-color': '#FF0000', // Red - 'color': '#FFF!important', // eslint-disable-line quote-props - 'font-weight': 'bold' - }); - return false; - } - - $('.sceneSeasonXEpisode').on('change', function() { - // Strip non-numeric characters - const value = $(this).val(); - $(this).val(value.replace(/[^0-9xX]*/g, '')); - const forSeason = $(this).attr('data-for-season'); - const forEpisode = $(this).attr('data-for-episode'); - - // If empty reset the field - if (value === '') { - setEpisodeSceneNumbering(forSeason, forEpisode, null, null); - return; - } - - const m = $(this).val().match(/^(\d+)x(\d+)$/i); - const onlyEpisode = $(this).val().match(/^(\d+)$/i); - let sceneSeason = null; - let sceneEpisode = null; - let isValid = false; - if (m) { - sceneSeason = m[1]; - sceneEpisode = m[2]; - isValid = setInputValidInvalid(true, $(this)); - } else if (onlyEpisode) { - // For example when '5' is filled in instead of '1x5', asume it's the first season - sceneSeason = forSeason; - sceneEpisode = onlyEpisode[1]; - isValid = setInputValidInvalid(true, $(this)); - } else { - isValid = setInputValidInvalid(false, $(this)); - } - - if (isValid) { - setEpisodeSceneNumbering(forSeason, forEpisode, sceneSeason, sceneEpisode); - } - }); - - $('.sceneAbsolute').on('change', function() { - // Strip non-numeric characters - $(this).val($(this).val().replace(/[^0-9xX]*/g, '')); - const forAbsolute = $(this).attr('data-for-absolute'); - - const m = $(this).val().match(/^(\d{1,3})$/i); - let sceneAbsolute = null; - if (m) { - sceneAbsolute = m[1]; - } - setAbsoluteSceneNumbering(forAbsolute, sceneAbsolute); - }); - - $.fn.generateStars = function() { - return this.each((i, e) => { - $(e).html($('').width($(e).text() * 12)); - }); - }; - - $('.imdbstars').generateStars(); - - $('#showTable, #animeTable').tablesorter({ - widgets: ['saveSort', 'stickyHeaders', 'columnSelector'], - widgetOptions: { - columnSelector_saveColumns: true, // eslint-disable-line camelcase - columnSelector_layout: '', // eslint-disable-line camelcase - columnSelector_mediaquery: false, // eslint-disable-line camelcase - columnSelector_cssChecked: 'checked' // eslint-disable-line camelcase - } - }); - - $('#popover').popover({ - placement: 'bottom', - html: true, // Required if content has HTML - content: '
' - }).on('shown.bs.popover', () => { // Bootstrap popover event triggered when the popover opens - $.tablesorter.columnSelector.attachTo($('#showTable, #animeTable'), '#popover-target'); - }); - - // Moved and rewritten this from displayShow. This changes the button when clicked for collapsing/expanding the - // Season to Show Episodes or Hide Episodes. - $(() => { - $('.collapse.toggle').on('hide.bs.collapse', function() { - const reg = /collapseSeason-(\d+)/g; - const result = reg.exec(this.id); - $('#showseason-' + result[1]).text('Show Episodes'); - $('#season-' + result[1] + '-cols').addClass('shadow'); - }); - $('.collapse.toggle').on('show.bs.collapse', function() { - const reg = /collapseSeason-(\d+)/g; - const result = reg.exec(this.id); - $('#showseason-' + result[1]).text('Hide Episodes'); - $('#season-' + result[1] + '-cols').removeClass('shadow'); - }); - }); - - // Set the season exception based on using the get_xem_numbering_for_show() for animes if available in data.xemNumbering, - // or else try to map using just the data.season_exceptions. - function setSeasonSceneException(data) { - $.each(data.seasonExceptions, (season, nameExceptions) => { - let foundInXem = false; - // Check if it is a season name exception, we don't handle the show name exceptions here - if (season >= 0) { - // Loop through the xem mapping, and check if there is a xem_season, that needs to show the season name exception - $.each(data.xemNumbering, (indexerSeason, xemSeason) => { - if (xemSeason === parseInt(season, 10)) { - foundInXem = true; - $('', { - id: 'xem-exception-season-' + xemSeason, - alt: '[xem]', - height: '16', - width: '16', - src: 'images/xem.png', - title: nameExceptions.join(', ') - }).appendTo('[data-season=' + indexerSeason + ']'); - } - }); - - // This is not a xem season exception, let's set the exceptions as a medusa exception - if (!foundInXem) { - $('', { - id: 'xem-exception-season-' + season, - alt: '[medusa]', - height: '16', - width: '16', - src: 'images/ico/favicon-16.png', - title: nameExceptions.join(', ') - }).appendTo('[data-season=' + season + ']'); - } - } - }); - } - - // @TODO: OMG: This is just a basic json, in future it should be based on the CRUD route. - // Get the season exceptions and the xem season mappings. - $.getJSON('home/getSeasonSceneExceptions', { - indexername: $('#indexer-name').val(), - seriesid: $('#series-id').val() // eslint-disable-line camelcase - }, data => { - setSeasonSceneException(data); - }); - - $('.display-specials a').on('click', function() { - api.patch('config/main', { - layout: { - show: { - specials: $(this).text() !== 'Hide' - } - } - }).then(response => { - log.info(response.data); - window.location.reload(); - }).catch(error => { - log.error(error.data); - }); - }); -}; diff --git a/themes-default/slim/static/js/manage/subtitle-missed-post-process.js b/themes-default/slim/static/js/manage/subtitle-missed-post-process.js deleted file mode 100644 index a42df68363..0000000000 --- a/themes-default/slim/static/js/manage/subtitle-missed-post-process.js +++ /dev/null @@ -1,28 +0,0 @@ -MEDUSA.manage.subtitleMissedPP = function() { - startAjaxEpisodeSubtitles(); // eslint-disable-line no-undef - $.ajaxEpSubtitlesSearch(); - - $('#releasesPP:has(tbody tr)').tablesorter({ - sortList: [[3, 1], [0, 0]], - textExtraction: { - 0(node) { return $(node).find('a').text().toLowerCase(); }, // eslint-disable-line brace-style - 1(node) { return $(node).text().toLowerCase(); }, // eslint-disable-line brace-style - 2(node) { return $(node).find('span').text().toLowerCase(); }, // eslint-disable-line brace-style - 3(node) { return $(node).find('span').attr('datetime'); } // eslint-disable-line brace-style - }, - widgets: ['saveSort', 'filter'], - headers: { - 0: { sorter: 'show' }, - 1: { sorter: 'episode' }, - 2: { sorter: 'release' }, - 3: { sorter: 'realISODate' }, - 4: { sorter: false, filter: false } - }, - widgetOptions: { - filter_columnFilters: true, // eslint-disable-line camelcase - filter_hideFilters: true, // eslint-disable-line camelcase - filter_saveFilters: true, // eslint-disable-line camelcase - columnSelector_mediaquery: false // eslint-disable-line camelcase - } - }); -}; diff --git a/themes-default/slim/views/displayShow.mako b/themes-default/slim/views/displayShow.mako index a7fe9ef991..964bb6282b 100644 --- a/themes-default/slim/views/displayShow.mako +++ b/themes-default/slim/views/displayShow.mako @@ -2,13 +2,10 @@ <%! import datetime import urllib - import ntpath from medusa import app, helpers, subtitles, sbdatetime, network_timezones from medusa.common import SKIPPED, WANTED, UNAIRED, ARCHIVED, IGNORED, FAILED, DOWNLOADED, SNATCHED, SNATCHED_PROPER, SNATCHED_BEST - from medusa.common import Quality, qualityPresets, statusStrings, Overview + from medusa.common import Quality, statusStrings, Overview from medusa.helper.common import pretty_file_size - from medusa.indexers.indexer_api import indexerApi - from medusa.indexers.utils import mappings %> <%block name="scripts"> @@ -30,12 +27,451 @@ const startVue = () => { $store.dispatch('getShows'); // Used by show-selector component }, mounted() { - // Adjust the summary background position and size on page load and resize + const { + moveSummaryBackground, + movecheckboxControlsBackground, + setQuality, + setEpisodeSceneNumbering, + setAbsoluteSceneNumbering, + setInputValidInvalid, + setSeasonSceneException, + showHideRows, + } = this; + + $(window).resize(() => { + moveSummaryBackground(); + movecheckboxControlsBackground(); + }); + this.$once('loaded', () => { this.$nextTick(() => { + // Adjust the summary background position and size window.dispatchEvent(new Event('resize')); + + $.ajaxEpSearch({ + colorRow: true + }); + + startAjaxEpisodeSubtitles(); // eslint-disable-line no-undef + $.ajaxEpSubtitlesSearch(); + $.ajaxEpRedownloadSubtitle(); + }); + }); + + $(document.body).on('click', '.imdbPlot', event => { + const $target = $(event.currentTarget); + $target.prev('span').toggle(); + if ($target.html() === '..show less') { + $target.html('..show more'); + } else { + $target.html('..show less'); + } + moveSummaryBackground(); + movecheckboxControlsBackground(); + }); + + $(document.body).on('change', '#seasonJump', (event) => { + const id = $('#seasonJump option:selected').val(); + if (id && id !== 'jump') { + const season = $('#seasonJump option:selected').data('season'); + $('html,body').animate({ scrollTop: $('[name="' + id.substring(1) + '"]').offset().top - 100 }, 'slow'); + $('#collapseSeason-' + season).collapse('show'); + location.hash = id; + } + $(event.currentTarget).val('jump'); + }); + + $(document.body).on('click', '#changeStatus', () => { + const epArr = []; + const status = $('#statusSelect').val(); + const quality = $('#qualitySelect').val(); + const seriesSlug = $('#series-slug').val(); + + $('.epCheck').each((index, element) => { + if (element.checked === true) { + epArr.push($(element).attr('id')); + } + }); + + if (epArr.length === 0) { + return false; + } + + if (quality) { + setQuality(quality, seriesSlug, epArr); + } + + if (status) { + window.location.href = $('base').attr('href') + 'home/setStatus?' + + 'indexername=' + $('#indexer-name').attr('value') + + '&seriesid=' + $('#series-id').attr('value') + + '&eps=' + epArr.join('|') + + '&status=' + status; + } + }); + + $(document.body).on('click', '.seasonCheck', event => { + const seasCheck = event.currentTarget; + const seasNo = $(seasCheck).attr('id'); + + $('#collapseSeason-' + seasNo).collapse('show'); + const seasonIdentifier = 's' + seasNo; + $('.epCheck:visible').each((index, element) => { + const epParts = $(element).attr('id').split('e'); + if (epParts[0] === seasonIdentifier) { + element.checked = seasCheck.checked; + } + }); + }); + + let lastCheck = null; + $(document.body).on('click', '.epCheck', event => { + const target = event.currentTarget; + if (!lastCheck || !event.shiftKey) { + lastCheck = target; + return; + } + + const check = target; + let found = 0; + + $('.epCheck').each((index, element) => { + if (found === 1) { + element.checked = lastCheck.checked; + } + + if (found === 2) { + return false; + } + + if (element === check || element === lastCheck) { + found++; + } + }); + }); + + // Selects all visible episode checkboxes. + $(document.body).on('click', '.seriesCheck', () => { + $('.epCheck:visible').each((index, element) => { + element.checked = true; + }); + $('.seasonCheck:visible').each((index, element) => { + element.checked = true; + }); + }); + + // Clears all visible episode checkboxes and the season selectors + $(document.body).on('click', '.clearAll', () => { + $('.epCheck:visible').each((index, element) => { + element.checked = false; + }); + $('.seasonCheck:visible').each((index, element) => { + element.checked = false; + }); + }); + + // Show/hide different types of rows when the checkboxes are changed + $(document.body).on('change', '#checkboxControls input', event => { + const whichClass = $(event.currentTarget).attr('id'); + showHideRows(whichClass); + }); + + // Initially show/hide all the rows according to the checkboxes + $('#checkboxControls input').each((index, element) => { + const status = $(element).prop('checked'); + $('tr.' + $(element).attr('id')).each((index, tableRow) => { + if (status) { + $(tableRow).show(); + } else { + $(tableRow).hide(); + } }); }); + + $(document.body).on('change', '.sceneSeasonXEpisode', event => { + const target = event.currentTarget; + // Strip non-numeric characters + const value = $(target).val(); + $(target).val(value.replace(/[^0-9xX]*/g, '')); + const forSeason = $(target).attr('data-for-season'); + const forEpisode = $(target).attr('data-for-episode'); + + // If empty reset the field + if (value === '') { + setEpisodeSceneNumbering(forSeason, forEpisode, null, null); + return; + } + + const m = $(target).val().match(/^(\d+)x(\d+)$/i); + const onlyEpisode = $(target).val().match(/^(\d+)$/i); + let sceneSeason = null; + let sceneEpisode = null; + let isValid = false; + if (m) { + sceneSeason = m[1]; + sceneEpisode = m[2]; + isValid = setInputValidInvalid(true, $(target)); + } else if (onlyEpisode) { + // For example when '5' is filled in instead of '1x5', asume it's the first season + sceneSeason = forSeason; + sceneEpisode = onlyEpisode[1]; + isValid = setInputValidInvalid(true, $(target)); + } else { + isValid = setInputValidInvalid(false, $(target)); + } + + if (isValid) { + setEpisodeSceneNumbering(forSeason, forEpisode, sceneSeason, sceneEpisode); + } + }); + + $(document.body).on('change', '.sceneAbsolute', event => { + const target = event.currentTarget; + // Strip non-numeric characters + $(target).val($(target).val().replace(/[^0-9xX]*/g, '')); + const forAbsolute = $(target).attr('data-for-absolute'); + + const m = $(target).val().match(/^(\d{1,3})$/i); + let sceneAbsolute = null; + if (m) { + sceneAbsolute = m[1]; + } + setAbsoluteSceneNumbering(forAbsolute, sceneAbsolute); + }); + + $('#showTable, #animeTable').tablesorter({ + widgets: ['saveSort', 'stickyHeaders', 'columnSelector'], + widgetOptions: { + columnSelector_saveColumns: true, // eslint-disable-line camelcase + columnSelector_layout: '', // eslint-disable-line camelcase + columnSelector_mediaquery: false, // eslint-disable-line camelcase + columnSelector_cssChecked: 'checked' // eslint-disable-line camelcase + } + }); + + $('#popover').popover({ + placement: 'bottom', + html: true, // Required if content has HTML + content: '' + }).on('shown.bs.popover', () => { // Bootstrap popover event triggered when the popover opens + $.tablesorter.columnSelector.attachTo($('#showTable, #animeTable'), '#popover-target'); + }); + + // Moved and rewritten this from displayShow. This changes the button when clicked for collapsing/expanding the + // Season to Show Episodes or Hide Episodes. + $('.collapse.toggle').on('hide.bs.collapse', function() { + const reg = /collapseSeason-(\d+)/g; + const result = reg.exec(this.id); + $('#showseason-' + result[1]).text('Show Episodes'); + $('#season-' + result[1] + '-cols').addClass('shadow'); + }); + $('.collapse.toggle').on('show.bs.collapse', function() { + const reg = /collapseSeason-(\d+)/g; + const result = reg.exec(this.id); + $('#showseason-' + result[1]).text('Hide Episodes'); + $('#season-' + result[1] + '-cols').removeClass('shadow'); + }); + + // Generate IMDB stars + $('.imdbstars').each((index, element) => { + $(element).html($('').width($(element).text() * 12)); + }); + + // @TODO: OMG: This is just a basic json, in future it should be based on the CRUD route. + // Get the season exceptions and the xem season mappings. + $.getJSON('home/getSeasonSceneExceptions', { + indexername: $('#indexer-name').val(), + seriesid: $('#series-id').val() // eslint-disable-line camelcase + }, data => { + setSeasonSceneException(data); + }); + + $(document.body).on('click', '.display-specials a', event => { + api.patch('config/main', { + layout: { + show: { + specials: $(event.currentTarget).text() !== 'Hide' + } + } + }).then(response => { + log.info(response.data); + window.location.reload(); + }).catch(error => { + log.error(error.data); + }); + }); + }, + methods: { + // Adjust the summary background position and size on page load and resize + moveSummaryBackground() { + const height = $('#summary').height() + 10; + const top = $('#summary').offset().top + 5; + $('#summaryBackground').height(height); + $('#summaryBackground').offset({ top, left: 0 }); + $('#summaryBackground').show(); + }, + movecheckboxControlsBackground() { + const height = $('#checkboxControls').height() + 10; + const top = $('#checkboxControls').offset().top - 3; + $('#checkboxControlsBackground').height(height); + $('#checkboxControlsBackground').offset({ top, left: 0 }); + $('#checkboxControlsBackground').show(); + }, + setQuality(quality, seriesSlug, episodes) { + const patchData = {}; + episodes.forEach(episode => { + patchData[episode] = { quality: parseInt(quality, 10) }; + }); + + api.patch('series/' + seriesSlug + '/episodes', patchData).then(response => { + log.info(response.data); + window.location.reload(); + }).catch(error => { + log.error(error.data); + }); + }, + setEpisodeSceneNumbering(forSeason, forEpisode, sceneSeason, sceneEpisode) { + const indexerName = $('#indexer-name').val(); + const seriesId = $('#series-id').val(); + + if (sceneSeason === '') { + sceneSeason = null; + } + if (sceneEpisode === '') { + sceneEpisode = null; + } + + $.getJSON('home/setSceneNumbering', { + indexername: indexerName, + seriesid: seriesId, + forSeason, + forEpisode, + sceneSeason, + sceneEpisode + }, data => { + // Set the values we get back + if (data.sceneSeason === null || data.sceneEpisode === null) { + $('#sceneSeasonXEpisode_' + seriesId + '_' + forSeason + '_' + forEpisode).val(''); + } else { + $('#sceneSeasonXEpisode_' + seriesId + '_' + forSeason + '_' + forEpisode).val(data.sceneSeason + 'x' + data.sceneEpisode); + } + if (!data.success) { + if (data.errorMessage) { + alert(data.errorMessage); // eslint-disable-line no-alert + } else { + alert('Update failed.'); // eslint-disable-line no-alert + } + } + }); + }, + setAbsoluteSceneNumbering(forAbsolute, sceneAbsolute) { + const indexerName = $('#indexer-name').val(); + const seriesId = $('#series-id').val(); + + if (sceneAbsolute === '') { + sceneAbsolute = null; + } + + $.getJSON('home/setSceneNumbering', { + indexername: indexerName, + seriesid: seriesId, + forAbsolute, + sceneAbsolute + }, data => { + // Set the values we get back + if (data.sceneAbsolute === null) { + $('#sceneAbsolute_' + seriesId + '_' + forAbsolute).val(''); + } else { + $('#sceneAbsolute_' + seriesId + '_' + forAbsolute).val(data.sceneAbsolute); + } + + if (!data.success) { + if (data.errorMessage) { + alert(data.errorMessage); // eslint-disable-line no-alert + } else { + alert('Update failed.'); // eslint-disable-line no-alert + } + } + }); + }, + setInputValidInvalid(valid, el) { + if (valid) { + $(el).css({ + 'background-color': '#90EE90', // Green + 'color': '#FFF', // eslint-disable-line quote-props + 'font-weight': 'bold' + }); + return true; + } + $(el).css({ + 'background-color': '#FF0000', // Red + 'color': '#FFF!important', // eslint-disable-line quote-props + 'font-weight': 'bold' + }); + return false; + }, + // Set the season exception based on using the get_xem_numbering_for_show() for animes if available in data.xemNumbering, + // or else try to map using just the data.season_exceptions. + setSeasonSceneException(data) { + $.each(data.seasonExceptions, (season, nameExceptions) => { + let foundInXem = false; + // Check if it is a season name exception, we don't handle the show name exceptions here + if (season >= 0) { + // Loop through the xem mapping, and check if there is a xem_season, that needs to show the season name exception + $.each(data.xemNumbering, (indexerSeason, xemSeason) => { + if (xemSeason === parseInt(season, 10)) { + foundInXem = true; + $('', { + id: 'xem-exception-season-' + xemSeason, + alt: '[xem]', + height: '16', + width: '16', + src: 'images/xem.png', + title: nameExceptions.join(', ') + }).appendTo('[data-season=' + indexerSeason + ']'); + } + }); + + // This is not a xem season exception, let's set the exceptions as a medusa exception + if (!foundInXem) { + $('', { + id: 'xem-exception-season-' + season, + alt: '[medusa]', + height: '16', + width: '16', + src: 'images/ico/favicon-16.png', + title: nameExceptions.join(', ') + }).appendTo('[data-season=' + season + ']'); + } + } + }); + }, + showHideRows(whichClass) { + const status = $('#checkboxControls > input, #' + whichClass).prop('checked'); + $('tr.' + whichClass).each((index, element) => { + if (status) { + $(element).show(); + } else { + $(element).hide(); + } + }); + + // Hide season headers with no episodes under them + $('tr.seasonheader').each((index, element) => { + let numRows = 0; + const seasonNo = $(element).attr('id'); + $('tr.' + seasonNo + ' :visible').each(() => { + numRows++; + }); + if (numRows === 0) { + $(element).hide(); + $('#' + seasonNo + '-cols').hide(); + } else { + $(element).show(); + $('#' + seasonNo + '-cols').show(); + } + }); + } } }); }; diff --git a/themes-default/slim/views/layouts/main.mako b/themes-default/slim/views/layouts/main.mako index d14510ad2e..051e620136 100644 --- a/themes-default/slim/views/layouts/main.mako +++ b/themes-default/slim/views/layouts/main.mako @@ -100,7 +100,6 @@ - @@ -111,7 +110,6 @@ - diff --git a/themes-default/slim/views/manage_subtitleMissedPP.mako b/themes-default/slim/views/manage_subtitleMissedPP.mako index 014b030796..e3fcf2351e 100644 --- a/themes-default/slim/views/manage_subtitleMissedPP.mako +++ b/themes-default/slim/views/manage_subtitleMissedPP.mako @@ -1,11 +1,11 @@ <%inherit file="/layouts/main.mako"/> <%! - from medusa import app import os + + from medusa import app from medusa.helper.common import episode_num %> <%block name="scripts"> - @@ -30,12 +27,451 @@ const startVue = () => { $store.dispatch('getShows'); // Used by show-selector component }, mounted() { - // Adjust the summary background position and size on page load and resize + const { + moveSummaryBackground, + movecheckboxControlsBackground, + setQuality, + setEpisodeSceneNumbering, + setAbsoluteSceneNumbering, + setInputValidInvalid, + setSeasonSceneException, + showHideRows, + } = this; + + $(window).resize(() => { + moveSummaryBackground(); + movecheckboxControlsBackground(); + }); + this.$once('loaded', () => { this.$nextTick(() => { + // Adjust the summary background position and size window.dispatchEvent(new Event('resize')); + + $.ajaxEpSearch({ + colorRow: true + }); + + startAjaxEpisodeSubtitles(); // eslint-disable-line no-undef + $.ajaxEpSubtitlesSearch(); + $.ajaxEpRedownloadSubtitle(); + }); + }); + + $(document.body).on('click', '.imdbPlot', event => { + const $target = $(event.currentTarget); + $target.prev('span').toggle(); + if ($target.html() === '..show less') { + $target.html('..show more'); + } else { + $target.html('..show less'); + } + moveSummaryBackground(); + movecheckboxControlsBackground(); + }); + + $(document.body).on('change', '#seasonJump', (event) => { + const id = $('#seasonJump option:selected').val(); + if (id && id !== 'jump') { + const season = $('#seasonJump option:selected').data('season'); + $('html,body').animate({ scrollTop: $('[name="' + id.substring(1) + '"]').offset().top - 100 }, 'slow'); + $('#collapseSeason-' + season).collapse('show'); + location.hash = id; + } + $(event.currentTarget).val('jump'); + }); + + $(document.body).on('click', '#changeStatus', () => { + const epArr = []; + const status = $('#statusSelect').val(); + const quality = $('#qualitySelect').val(); + const seriesSlug = $('#series-slug').val(); + + $('.epCheck').each((index, element) => { + if (element.checked === true) { + epArr.push($(element).attr('id')); + } + }); + + if (epArr.length === 0) { + return false; + } + + if (quality) { + setQuality(quality, seriesSlug, epArr); + } + + if (status) { + window.location.href = $('base').attr('href') + 'home/setStatus?' + + 'indexername=' + $('#indexer-name').attr('value') + + '&seriesid=' + $('#series-id').attr('value') + + '&eps=' + epArr.join('|') + + '&status=' + status; + } + }); + + $(document.body).on('click', '.seasonCheck', event => { + const seasCheck = event.currentTarget; + const seasNo = $(seasCheck).attr('id'); + + $('#collapseSeason-' + seasNo).collapse('show'); + const seasonIdentifier = 's' + seasNo; + $('.epCheck:visible').each((index, element) => { + const epParts = $(element).attr('id').split('e'); + if (epParts[0] === seasonIdentifier) { + element.checked = seasCheck.checked; + } + }); + }); + + let lastCheck = null; + $(document.body).on('click', '.epCheck', event => { + const target = event.currentTarget; + if (!lastCheck || !event.shiftKey) { + lastCheck = target; + return; + } + + const check = target; + let found = 0; + + $('.epCheck').each((index, element) => { + if (found === 1) { + element.checked = lastCheck.checked; + } + + if (found === 2) { + return false; + } + + if (element === check || element === lastCheck) { + found++; + } + }); + }); + + // Selects all visible episode checkboxes. + $(document.body).on('click', '.seriesCheck', () => { + $('.epCheck:visible').each((index, element) => { + element.checked = true; + }); + $('.seasonCheck:visible').each((index, element) => { + element.checked = true; + }); + }); + + // Clears all visible episode checkboxes and the season selectors + $(document.body).on('click', '.clearAll', () => { + $('.epCheck:visible').each((index, element) => { + element.checked = false; + }); + $('.seasonCheck:visible').each((index, element) => { + element.checked = false; + }); + }); + + // Show/hide different types of rows when the checkboxes are changed + $(document.body).on('change', '#checkboxControls input', event => { + const whichClass = $(event.currentTarget).attr('id'); + showHideRows(whichClass); + }); + + // Initially show/hide all the rows according to the checkboxes + $('#checkboxControls input').each((index, element) => { + const status = $(element).prop('checked'); + $('tr.' + $(element).attr('id')).each((index, tableRow) => { + if (status) { + $(tableRow).show(); + } else { + $(tableRow).hide(); + } }); }); + + $(document.body).on('change', '.sceneSeasonXEpisode', event => { + const target = event.currentTarget; + // Strip non-numeric characters + const value = $(target).val(); + $(target).val(value.replace(/[^0-9xX]*/g, '')); + const forSeason = $(target).attr('data-for-season'); + const forEpisode = $(target).attr('data-for-episode'); + + // If empty reset the field + if (value === '') { + setEpisodeSceneNumbering(forSeason, forEpisode, null, null); + return; + } + + const m = $(target).val().match(/^(\d+)x(\d+)$/i); + const onlyEpisode = $(target).val().match(/^(\d+)$/i); + let sceneSeason = null; + let sceneEpisode = null; + let isValid = false; + if (m) { + sceneSeason = m[1]; + sceneEpisode = m[2]; + isValid = setInputValidInvalid(true, $(target)); + } else if (onlyEpisode) { + // For example when '5' is filled in instead of '1x5', asume it's the first season + sceneSeason = forSeason; + sceneEpisode = onlyEpisode[1]; + isValid = setInputValidInvalid(true, $(target)); + } else { + isValid = setInputValidInvalid(false, $(target)); + } + + if (isValid) { + setEpisodeSceneNumbering(forSeason, forEpisode, sceneSeason, sceneEpisode); + } + }); + + $(document.body).on('change', '.sceneAbsolute', event => { + const target = event.currentTarget; + // Strip non-numeric characters + $(target).val($(target).val().replace(/[^0-9xX]*/g, '')); + const forAbsolute = $(target).attr('data-for-absolute'); + + const m = $(target).val().match(/^(\d{1,3})$/i); + let sceneAbsolute = null; + if (m) { + sceneAbsolute = m[1]; + } + setAbsoluteSceneNumbering(forAbsolute, sceneAbsolute); + }); + + $('#showTable, #animeTable').tablesorter({ + widgets: ['saveSort', 'stickyHeaders', 'columnSelector'], + widgetOptions: { + columnSelector_saveColumns: true, // eslint-disable-line camelcase + columnSelector_layout: '', // eslint-disable-line camelcase + columnSelector_mediaquery: false, // eslint-disable-line camelcase + columnSelector_cssChecked: 'checked' // eslint-disable-line camelcase + } + }); + + $('#popover').popover({ + placement: 'bottom', + html: true, // Required if content has HTML + content: '' + }).on('shown.bs.popover', () => { // Bootstrap popover event triggered when the popover opens + $.tablesorter.columnSelector.attachTo($('#showTable, #animeTable'), '#popover-target'); + }); + + // Moved and rewritten this from displayShow. This changes the button when clicked for collapsing/expanding the + // Season to Show Episodes or Hide Episodes. + $('.collapse.toggle').on('hide.bs.collapse', function() { + const reg = /collapseSeason-(\d+)/g; + const result = reg.exec(this.id); + $('#showseason-' + result[1]).text('Show Episodes'); + $('#season-' + result[1] + '-cols').addClass('shadow'); + }); + $('.collapse.toggle').on('show.bs.collapse', function() { + const reg = /collapseSeason-(\d+)/g; + const result = reg.exec(this.id); + $('#showseason-' + result[1]).text('Hide Episodes'); + $('#season-' + result[1] + '-cols').removeClass('shadow'); + }); + + // Generate IMDB stars + $('.imdbstars').each((index, element) => { + $(element).html($('').width($(element).text() * 12)); + }); + + // @TODO: OMG: This is just a basic json, in future it should be based on the CRUD route. + // Get the season exceptions and the xem season mappings. + $.getJSON('home/getSeasonSceneExceptions', { + indexername: $('#indexer-name').val(), + seriesid: $('#series-id').val() // eslint-disable-line camelcase + }, data => { + setSeasonSceneException(data); + }); + + $(document.body).on('click', '.display-specials a', event => { + api.patch('config/main', { + layout: { + show: { + specials: $(event.currentTarget).text() !== 'Hide' + } + } + }).then(response => { + log.info(response.data); + window.location.reload(); + }).catch(error => { + log.error(error.data); + }); + }); + }, + methods: { + // Adjust the summary background position and size on page load and resize + moveSummaryBackground() { + const height = $('#summary').height() + 10; + const top = $('#summary').offset().top + 5; + $('#summaryBackground').height(height); + $('#summaryBackground').offset({ top, left: 0 }); + $('#summaryBackground').show(); + }, + movecheckboxControlsBackground() { + const height = $('#checkboxControls').height() + 10; + const top = $('#checkboxControls').offset().top - 3; + $('#checkboxControlsBackground').height(height); + $('#checkboxControlsBackground').offset({ top, left: 0 }); + $('#checkboxControlsBackground').show(); + }, + setQuality(quality, seriesSlug, episodes) { + const patchData = {}; + episodes.forEach(episode => { + patchData[episode] = { quality: parseInt(quality, 10) }; + }); + + api.patch('series/' + seriesSlug + '/episodes', patchData).then(response => { + log.info(response.data); + window.location.reload(); + }).catch(error => { + log.error(error.data); + }); + }, + setEpisodeSceneNumbering(forSeason, forEpisode, sceneSeason, sceneEpisode) { + const indexerName = $('#indexer-name').val(); + const seriesId = $('#series-id').val(); + + if (sceneSeason === '') { + sceneSeason = null; + } + if (sceneEpisode === '') { + sceneEpisode = null; + } + + $.getJSON('home/setSceneNumbering', { + indexername: indexerName, + seriesid: seriesId, + forSeason, + forEpisode, + sceneSeason, + sceneEpisode + }, data => { + // Set the values we get back + if (data.sceneSeason === null || data.sceneEpisode === null) { + $('#sceneSeasonXEpisode_' + seriesId + '_' + forSeason + '_' + forEpisode).val(''); + } else { + $('#sceneSeasonXEpisode_' + seriesId + '_' + forSeason + '_' + forEpisode).val(data.sceneSeason + 'x' + data.sceneEpisode); + } + if (!data.success) { + if (data.errorMessage) { + alert(data.errorMessage); // eslint-disable-line no-alert + } else { + alert('Update failed.'); // eslint-disable-line no-alert + } + } + }); + }, + setAbsoluteSceneNumbering(forAbsolute, sceneAbsolute) { + const indexerName = $('#indexer-name').val(); + const seriesId = $('#series-id').val(); + + if (sceneAbsolute === '') { + sceneAbsolute = null; + } + + $.getJSON('home/setSceneNumbering', { + indexername: indexerName, + seriesid: seriesId, + forAbsolute, + sceneAbsolute + }, data => { + // Set the values we get back + if (data.sceneAbsolute === null) { + $('#sceneAbsolute_' + seriesId + '_' + forAbsolute).val(''); + } else { + $('#sceneAbsolute_' + seriesId + '_' + forAbsolute).val(data.sceneAbsolute); + } + + if (!data.success) { + if (data.errorMessage) { + alert(data.errorMessage); // eslint-disable-line no-alert + } else { + alert('Update failed.'); // eslint-disable-line no-alert + } + } + }); + }, + setInputValidInvalid(valid, el) { + if (valid) { + $(el).css({ + 'background-color': '#90EE90', // Green + 'color': '#FFF', // eslint-disable-line quote-props + 'font-weight': 'bold' + }); + return true; + } + $(el).css({ + 'background-color': '#FF0000', // Red + 'color': '#FFF!important', // eslint-disable-line quote-props + 'font-weight': 'bold' + }); + return false; + }, + // Set the season exception based on using the get_xem_numbering_for_show() for animes if available in data.xemNumbering, + // or else try to map using just the data.season_exceptions. + setSeasonSceneException(data) { + $.each(data.seasonExceptions, (season, nameExceptions) => { + let foundInXem = false; + // Check if it is a season name exception, we don't handle the show name exceptions here + if (season >= 0) { + // Loop through the xem mapping, and check if there is a xem_season, that needs to show the season name exception + $.each(data.xemNumbering, (indexerSeason, xemSeason) => { + if (xemSeason === parseInt(season, 10)) { + foundInXem = true; + $('', { + id: 'xem-exception-season-' + xemSeason, + alt: '[xem]', + height: '16', + width: '16', + src: 'images/xem.png', + title: nameExceptions.join(', ') + }).appendTo('[data-season=' + indexerSeason + ']'); + } + }); + + // This is not a xem season exception, let's set the exceptions as a medusa exception + if (!foundInXem) { + $('', { + id: 'xem-exception-season-' + season, + alt: '[medusa]', + height: '16', + width: '16', + src: 'images/ico/favicon-16.png', + title: nameExceptions.join(', ') + }).appendTo('[data-season=' + season + ']'); + } + } + }); + }, + showHideRows(whichClass) { + const status = $('#checkboxControls > input, #' + whichClass).prop('checked'); + $('tr.' + whichClass).each((index, element) => { + if (status) { + $(element).show(); + } else { + $(element).hide(); + } + }); + + // Hide season headers with no episodes under them + $('tr.seasonheader').each((index, element) => { + let numRows = 0; + const seasonNo = $(element).attr('id'); + $('tr.' + seasonNo + ' :visible').each(() => { + numRows++; + }); + if (numRows === 0) { + $(element).hide(); + $('#' + seasonNo + '-cols').hide(); + } else { + $(element).show(); + $('#' + seasonNo + '-cols').show(); + } + }); + } } }); }; diff --git a/themes/dark/templates/layouts/main.mako b/themes/dark/templates/layouts/main.mako index d14510ad2e..051e620136 100644 --- a/themes/dark/templates/layouts/main.mako +++ b/themes/dark/templates/layouts/main.mako @@ -100,7 +100,6 @@ - @@ -111,7 +110,6 @@ - diff --git a/themes/dark/templates/manage_subtitleMissedPP.mako b/themes/dark/templates/manage_subtitleMissedPP.mako index 014b030796..e3fcf2351e 100644 --- a/themes/dark/templates/manage_subtitleMissedPP.mako +++ b/themes/dark/templates/manage_subtitleMissedPP.mako @@ -1,11 +1,11 @@ <%inherit file="/layouts/main.mako"/> <%! - from medusa import app import os + + from medusa import app from medusa.helper.common import episode_num %> <%block name="scripts"> - @@ -30,12 +27,451 @@ const startVue = () => { $store.dispatch('getShows'); // Used by show-selector component }, mounted() { - // Adjust the summary background position and size on page load and resize + const { + moveSummaryBackground, + movecheckboxControlsBackground, + setQuality, + setEpisodeSceneNumbering, + setAbsoluteSceneNumbering, + setInputValidInvalid, + setSeasonSceneException, + showHideRows, + } = this; + + $(window).resize(() => { + moveSummaryBackground(); + movecheckboxControlsBackground(); + }); + this.$once('loaded', () => { this.$nextTick(() => { + // Adjust the summary background position and size window.dispatchEvent(new Event('resize')); + + $.ajaxEpSearch({ + colorRow: true + }); + + startAjaxEpisodeSubtitles(); // eslint-disable-line no-undef + $.ajaxEpSubtitlesSearch(); + $.ajaxEpRedownloadSubtitle(); + }); + }); + + $(document.body).on('click', '.imdbPlot', event => { + const $target = $(event.currentTarget); + $target.prev('span').toggle(); + if ($target.html() === '..show less') { + $target.html('..show more'); + } else { + $target.html('..show less'); + } + moveSummaryBackground(); + movecheckboxControlsBackground(); + }); + + $(document.body).on('change', '#seasonJump', (event) => { + const id = $('#seasonJump option:selected').val(); + if (id && id !== 'jump') { + const season = $('#seasonJump option:selected').data('season'); + $('html,body').animate({ scrollTop: $('[name="' + id.substring(1) + '"]').offset().top - 100 }, 'slow'); + $('#collapseSeason-' + season).collapse('show'); + location.hash = id; + } + $(event.currentTarget).val('jump'); + }); + + $(document.body).on('click', '#changeStatus', () => { + const epArr = []; + const status = $('#statusSelect').val(); + const quality = $('#qualitySelect').val(); + const seriesSlug = $('#series-slug').val(); + + $('.epCheck').each((index, element) => { + if (element.checked === true) { + epArr.push($(element).attr('id')); + } + }); + + if (epArr.length === 0) { + return false; + } + + if (quality) { + setQuality(quality, seriesSlug, epArr); + } + + if (status) { + window.location.href = $('base').attr('href') + 'home/setStatus?' + + 'indexername=' + $('#indexer-name').attr('value') + + '&seriesid=' + $('#series-id').attr('value') + + '&eps=' + epArr.join('|') + + '&status=' + status; + } + }); + + $(document.body).on('click', '.seasonCheck', event => { + const seasCheck = event.currentTarget; + const seasNo = $(seasCheck).attr('id'); + + $('#collapseSeason-' + seasNo).collapse('show'); + const seasonIdentifier = 's' + seasNo; + $('.epCheck:visible').each((index, element) => { + const epParts = $(element).attr('id').split('e'); + if (epParts[0] === seasonIdentifier) { + element.checked = seasCheck.checked; + } + }); + }); + + let lastCheck = null; + $(document.body).on('click', '.epCheck', event => { + const target = event.currentTarget; + if (!lastCheck || !event.shiftKey) { + lastCheck = target; + return; + } + + const check = target; + let found = 0; + + $('.epCheck').each((index, element) => { + if (found === 1) { + element.checked = lastCheck.checked; + } + + if (found === 2) { + return false; + } + + if (element === check || element === lastCheck) { + found++; + } + }); + }); + + // Selects all visible episode checkboxes. + $(document.body).on('click', '.seriesCheck', () => { + $('.epCheck:visible').each((index, element) => { + element.checked = true; + }); + $('.seasonCheck:visible').each((index, element) => { + element.checked = true; + }); + }); + + // Clears all visible episode checkboxes and the season selectors + $(document.body).on('click', '.clearAll', () => { + $('.epCheck:visible').each((index, element) => { + element.checked = false; + }); + $('.seasonCheck:visible').each((index, element) => { + element.checked = false; + }); + }); + + // Show/hide different types of rows when the checkboxes are changed + $(document.body).on('change', '#checkboxControls input', event => { + const whichClass = $(event.currentTarget).attr('id'); + showHideRows(whichClass); + }); + + // Initially show/hide all the rows according to the checkboxes + $('#checkboxControls input').each((index, element) => { + const status = $(element).prop('checked'); + $('tr.' + $(element).attr('id')).each((index, tableRow) => { + if (status) { + $(tableRow).show(); + } else { + $(tableRow).hide(); + } }); }); + + $(document.body).on('change', '.sceneSeasonXEpisode', event => { + const target = event.currentTarget; + // Strip non-numeric characters + const value = $(target).val(); + $(target).val(value.replace(/[^0-9xX]*/g, '')); + const forSeason = $(target).attr('data-for-season'); + const forEpisode = $(target).attr('data-for-episode'); + + // If empty reset the field + if (value === '') { + setEpisodeSceneNumbering(forSeason, forEpisode, null, null); + return; + } + + const m = $(target).val().match(/^(\d+)x(\d+)$/i); + const onlyEpisode = $(target).val().match(/^(\d+)$/i); + let sceneSeason = null; + let sceneEpisode = null; + let isValid = false; + if (m) { + sceneSeason = m[1]; + sceneEpisode = m[2]; + isValid = setInputValidInvalid(true, $(target)); + } else if (onlyEpisode) { + // For example when '5' is filled in instead of '1x5', asume it's the first season + sceneSeason = forSeason; + sceneEpisode = onlyEpisode[1]; + isValid = setInputValidInvalid(true, $(target)); + } else { + isValid = setInputValidInvalid(false, $(target)); + } + + if (isValid) { + setEpisodeSceneNumbering(forSeason, forEpisode, sceneSeason, sceneEpisode); + } + }); + + $(document.body).on('change', '.sceneAbsolute', event => { + const target = event.currentTarget; + // Strip non-numeric characters + $(target).val($(target).val().replace(/[^0-9xX]*/g, '')); + const forAbsolute = $(target).attr('data-for-absolute'); + + const m = $(target).val().match(/^(\d{1,3})$/i); + let sceneAbsolute = null; + if (m) { + sceneAbsolute = m[1]; + } + setAbsoluteSceneNumbering(forAbsolute, sceneAbsolute); + }); + + $('#showTable, #animeTable').tablesorter({ + widgets: ['saveSort', 'stickyHeaders', 'columnSelector'], + widgetOptions: { + columnSelector_saveColumns: true, // eslint-disable-line camelcase + columnSelector_layout: '', // eslint-disable-line camelcase + columnSelector_mediaquery: false, // eslint-disable-line camelcase + columnSelector_cssChecked: 'checked' // eslint-disable-line camelcase + } + }); + + $('#popover').popover({ + placement: 'bottom', + html: true, // Required if content has HTML + content: '' + }).on('shown.bs.popover', () => { // Bootstrap popover event triggered when the popover opens + $.tablesorter.columnSelector.attachTo($('#showTable, #animeTable'), '#popover-target'); + }); + + // Moved and rewritten this from displayShow. This changes the button when clicked for collapsing/expanding the + // Season to Show Episodes or Hide Episodes. + $('.collapse.toggle').on('hide.bs.collapse', function() { + const reg = /collapseSeason-(\d+)/g; + const result = reg.exec(this.id); + $('#showseason-' + result[1]).text('Show Episodes'); + $('#season-' + result[1] + '-cols').addClass('shadow'); + }); + $('.collapse.toggle').on('show.bs.collapse', function() { + const reg = /collapseSeason-(\d+)/g; + const result = reg.exec(this.id); + $('#showseason-' + result[1]).text('Hide Episodes'); + $('#season-' + result[1] + '-cols').removeClass('shadow'); + }); + + // Generate IMDB stars + $('.imdbstars').each((index, element) => { + $(element).html($('').width($(element).text() * 12)); + }); + + // @TODO: OMG: This is just a basic json, in future it should be based on the CRUD route. + // Get the season exceptions and the xem season mappings. + $.getJSON('home/getSeasonSceneExceptions', { + indexername: $('#indexer-name').val(), + seriesid: $('#series-id').val() // eslint-disable-line camelcase + }, data => { + setSeasonSceneException(data); + }); + + $(document.body).on('click', '.display-specials a', event => { + api.patch('config/main', { + layout: { + show: { + specials: $(event.currentTarget).text() !== 'Hide' + } + } + }).then(response => { + log.info(response.data); + window.location.reload(); + }).catch(error => { + log.error(error.data); + }); + }); + }, + methods: { + // Adjust the summary background position and size on page load and resize + moveSummaryBackground() { + const height = $('#summary').height() + 10; + const top = $('#summary').offset().top + 5; + $('#summaryBackground').height(height); + $('#summaryBackground').offset({ top, left: 0 }); + $('#summaryBackground').show(); + }, + movecheckboxControlsBackground() { + const height = $('#checkboxControls').height() + 10; + const top = $('#checkboxControls').offset().top - 3; + $('#checkboxControlsBackground').height(height); + $('#checkboxControlsBackground').offset({ top, left: 0 }); + $('#checkboxControlsBackground').show(); + }, + setQuality(quality, seriesSlug, episodes) { + const patchData = {}; + episodes.forEach(episode => { + patchData[episode] = { quality: parseInt(quality, 10) }; + }); + + api.patch('series/' + seriesSlug + '/episodes', patchData).then(response => { + log.info(response.data); + window.location.reload(); + }).catch(error => { + log.error(error.data); + }); + }, + setEpisodeSceneNumbering(forSeason, forEpisode, sceneSeason, sceneEpisode) { + const indexerName = $('#indexer-name').val(); + const seriesId = $('#series-id').val(); + + if (sceneSeason === '') { + sceneSeason = null; + } + if (sceneEpisode === '') { + sceneEpisode = null; + } + + $.getJSON('home/setSceneNumbering', { + indexername: indexerName, + seriesid: seriesId, + forSeason, + forEpisode, + sceneSeason, + sceneEpisode + }, data => { + // Set the values we get back + if (data.sceneSeason === null || data.sceneEpisode === null) { + $('#sceneSeasonXEpisode_' + seriesId + '_' + forSeason + '_' + forEpisode).val(''); + } else { + $('#sceneSeasonXEpisode_' + seriesId + '_' + forSeason + '_' + forEpisode).val(data.sceneSeason + 'x' + data.sceneEpisode); + } + if (!data.success) { + if (data.errorMessage) { + alert(data.errorMessage); // eslint-disable-line no-alert + } else { + alert('Update failed.'); // eslint-disable-line no-alert + } + } + }); + }, + setAbsoluteSceneNumbering(forAbsolute, sceneAbsolute) { + const indexerName = $('#indexer-name').val(); + const seriesId = $('#series-id').val(); + + if (sceneAbsolute === '') { + sceneAbsolute = null; + } + + $.getJSON('home/setSceneNumbering', { + indexername: indexerName, + seriesid: seriesId, + forAbsolute, + sceneAbsolute + }, data => { + // Set the values we get back + if (data.sceneAbsolute === null) { + $('#sceneAbsolute_' + seriesId + '_' + forAbsolute).val(''); + } else { + $('#sceneAbsolute_' + seriesId + '_' + forAbsolute).val(data.sceneAbsolute); + } + + if (!data.success) { + if (data.errorMessage) { + alert(data.errorMessage); // eslint-disable-line no-alert + } else { + alert('Update failed.'); // eslint-disable-line no-alert + } + } + }); + }, + setInputValidInvalid(valid, el) { + if (valid) { + $(el).css({ + 'background-color': '#90EE90', // Green + 'color': '#FFF', // eslint-disable-line quote-props + 'font-weight': 'bold' + }); + return true; + } + $(el).css({ + 'background-color': '#FF0000', // Red + 'color': '#FFF!important', // eslint-disable-line quote-props + 'font-weight': 'bold' + }); + return false; + }, + // Set the season exception based on using the get_xem_numbering_for_show() for animes if available in data.xemNumbering, + // or else try to map using just the data.season_exceptions. + setSeasonSceneException(data) { + $.each(data.seasonExceptions, (season, nameExceptions) => { + let foundInXem = false; + // Check if it is a season name exception, we don't handle the show name exceptions here + if (season >= 0) { + // Loop through the xem mapping, and check if there is a xem_season, that needs to show the season name exception + $.each(data.xemNumbering, (indexerSeason, xemSeason) => { + if (xemSeason === parseInt(season, 10)) { + foundInXem = true; + $('', { + id: 'xem-exception-season-' + xemSeason, + alt: '[xem]', + height: '16', + width: '16', + src: 'images/xem.png', + title: nameExceptions.join(', ') + }).appendTo('[data-season=' + indexerSeason + ']'); + } + }); + + // This is not a xem season exception, let's set the exceptions as a medusa exception + if (!foundInXem) { + $('', { + id: 'xem-exception-season-' + season, + alt: '[medusa]', + height: '16', + width: '16', + src: 'images/ico/favicon-16.png', + title: nameExceptions.join(', ') + }).appendTo('[data-season=' + season + ']'); + } + } + }); + }, + showHideRows(whichClass) { + const status = $('#checkboxControls > input, #' + whichClass).prop('checked'); + $('tr.' + whichClass).each((index, element) => { + if (status) { + $(element).show(); + } else { + $(element).hide(); + } + }); + + // Hide season headers with no episodes under them + $('tr.seasonheader').each((index, element) => { + let numRows = 0; + const seasonNo = $(element).attr('id'); + $('tr.' + seasonNo + ' :visible').each(() => { + numRows++; + }); + if (numRows === 0) { + $(element).hide(); + $('#' + seasonNo + '-cols').hide(); + } else { + $(element).show(); + $('#' + seasonNo + '-cols').show(); + } + }); + } } }); }; diff --git a/themes/light/templates/layouts/main.mako b/themes/light/templates/layouts/main.mako index d14510ad2e..051e620136 100644 --- a/themes/light/templates/layouts/main.mako +++ b/themes/light/templates/layouts/main.mako @@ -100,7 +100,6 @@ - @@ -111,7 +110,6 @@ - diff --git a/themes/light/templates/manage_subtitleMissedPP.mako b/themes/light/templates/manage_subtitleMissedPP.mako index 014b030796..e3fcf2351e 100644 --- a/themes/light/templates/manage_subtitleMissedPP.mako +++ b/themes/light/templates/manage_subtitleMissedPP.mako @@ -1,11 +1,11 @@ <%inherit file="/layouts/main.mako"/> <%! - from medusa import app import os + + from medusa import app from medusa.helper.common import episode_num %> <%block name="scripts"> -