mirror of
https://github.com/Ombi-app/Ombi.git
synced 2025-07-16 02:02:55 -07:00
#114 first pass at choosing quality profile when approving + focus search input by default and when switching tabs
This commit is contained in:
parent
96fde83488
commit
9ac949a67c
16 changed files with 325 additions and 77 deletions
|
@ -28,6 +28,7 @@
|
||||||
using System;
|
using System;
|
||||||
using Newtonsoft.Json;
|
using Newtonsoft.Json;
|
||||||
using PlexRequests.Helpers;
|
using PlexRequests.Helpers;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
|
||||||
namespace PlexRequests.Core.SettingModels
|
namespace PlexRequests.Core.SettingModels
|
||||||
{
|
{
|
||||||
|
@ -40,6 +41,23 @@ namespace PlexRequests.Core.SettingModels
|
||||||
public string QualityProfile { get; set; }
|
public string QualityProfile { get; set; }
|
||||||
public bool Ssl { get; set; }
|
public bool Ssl { get; set; }
|
||||||
public string SubDir { get; set; }
|
public string SubDir { get; set; }
|
||||||
|
public Dictionary<string, string> Qualities
|
||||||
|
{
|
||||||
|
get
|
||||||
|
{
|
||||||
|
return new Dictionary<string, string>() {
|
||||||
|
{ "default", "Use Deafult" },
|
||||||
|
{ "sdtv", ">SD TV" },
|
||||||
|
{ "sddvd", "SD DVD" },
|
||||||
|
{ "hdtv", "HD TV" },
|
||||||
|
{ "rawhdtv", "Raw HD TV" },
|
||||||
|
{ "hdwebdl", "HD Web DL" },
|
||||||
|
{ "fullhdwebdl", "Full HD Web DL" },
|
||||||
|
{ "hdbluray", "HD Bluray" },
|
||||||
|
{ "fullhdbluray", "Full HD Bluray" }
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
[JsonIgnore]
|
[JsonIgnore]
|
||||||
public Uri FullUri
|
public Uri FullUri
|
||||||
|
|
|
@ -209,3 +209,13 @@ label {
|
||||||
padding: 3px 12px;
|
padding: 3px 12px;
|
||||||
color: rgba(255, 255, 255, 0.7) !important; }
|
color: rgba(255, 255, 255, 0.7) !important; }
|
||||||
|
|
||||||
|
.btn-split .btn {
|
||||||
|
border-radius: 0 !important; }
|
||||||
|
|
||||||
|
.btn-split .btn:not(.dropdown-toggle) {
|
||||||
|
border-radius: 0.25rem 0 0 0.25rem !important; }
|
||||||
|
|
||||||
|
.btn-split .btn.dropdown-toggle {
|
||||||
|
border-radius: 0 0.25rem 0.25rem 0 !important;
|
||||||
|
padding: 12px 8px; }
|
||||||
|
|
||||||
|
|
2
PlexRequests.UI/Content/custom.min.css
vendored
2
PlexRequests.UI/Content/custom.min.css
vendored
|
@ -1 +1 @@
|
||||||
@media(min-width:768px){.row{position:relative;}.bottom-align-text{position:absolute;bottom:0;right:0;}}@media(max-width:48em){.home{padding-top:1rem;}}@media(min-width:48em){.home{padding-top:4rem;}}.btn{border-radius:.25rem !important;}.multiSelect{background-color:#4e5d6c;}.form-control-custom{background-color:#4e5d6c !important;color:#fff !important;border-radius:0;box-shadow:0 0 0 !important;}h1{font-size:3.5rem !important;font-weight:600 !important;}.request-title{margin-top:0 !important;font-size:1.9rem !important;}p{font-size:1.1rem !important;}label{display:inline-block !important;margin-bottom:.5rem !important;font-size:16px !important;}.nav-tabs>li.active>a,.nav-tabs>li.active>a:hover,.nav-tabs>li.active>a:focus{background:#4e5d6c;}.navbar .nav a .fa,.dropdown-menu a .fa{font-size:130%;top:1px;position:relative;display:inline-block;margin-right:5px;}.dropdown-menu a .fa{top:2px;}.btn-danger-outline{color:#d9534f !important;background-color:transparent;background-image:none;border-color:#d9534f !important;}.btn-danger-outline:focus,.btn-danger-outline.focus,.btn-danger-outline:active,.btn-danger-outline.active,.btn-danger-outline:hover,.open>.btn-danger-outline.dropdown-toggle{color:#fff !important;background-color:#d9534f !important;border-color:#d9534f !important;}.btn-primary-outline{color:#ff761b !important;background-color:transparent;background-image:none;border-color:#ff761b !important;}.btn-primary-outline:focus,.btn-primary-outline.focus,.btn-primary-outline:active,.btn-primary-outline.active,.btn-primary-outline:hover,.open>.btn-primary-outline.dropdown-toggle{color:#fff !important;background-color:#df691a !important;border-color:#df691a !important;}.btn-info-outline{color:#5bc0de !important;background-color:transparent;background-image:none;border-color:#5bc0de !important;}.btn-info-outline:focus,.btn-info-outline.focus,.btn-info-outline:active,.btn-info-outline.active,.btn-info-outline:hover,.open>.btn-info-outline.dropdown-toggle{color:#fff !important;background-color:#5bc0de !important;border-color:#5bc0de !important;}.btn-warning-outline{color:#f0ad4e !important;background-color:transparent;background-image:none;border-color:#f0ad4e !important;}.btn-warning-outline:focus,.btn-warning-outline.focus,.btn-warning-outline:active,.btn-warning-outline.active,.btn-warning-outline:hover,.open>.btn-warning-outline.dropdown-toggle{color:#fff !important;background-color:#f0ad4e !important;border-color:#f0ad4e !important;}.btn-success-outline{color:#5cb85c !important;background-color:transparent;background-image:none;border-color:#5cb85c !important;}.btn-success-outline:focus,.btn-success-outline.focus,.btn-success-outline:active,.btn-success-outline.active,.btn-success-outline:hover,.open>.btn-success-outline.dropdown-toggle{color:#fff !important;background-color:#5cb85c !important;border-color:#5cb85c !important;}#movieList .mix{display:none;}#tvList .mix{display:none;}.scroll-top-wrapper{position:fixed;opacity:0;visibility:hidden;overflow:hidden;text-align:center;z-index:99999999;background-color:#4e5d6c;color:#eee;width:50px;height:48px;line-height:48px;right:30px;bottom:30px;padding-top:2px;border-top-left-radius:10px;border-top-right-radius:10px;border-bottom-right-radius:10px;border-bottom-left-radius:10px;-webkit-transition:all .5s ease-in-out;-moz-transition:all .5s ease-in-out;-ms-transition:all .5s ease-in-out;-o-transition:all .5s ease-in-out;transition:all .5s ease-in-out;}.scroll-top-wrapper:hover{background-color:#637689;}.scroll-top-wrapper.show{visibility:visible;cursor:pointer;opacity:1;}.scroll-top-wrapper i.fa{line-height:inherit;}.no-search-results{text-align:center;}.no-search-results .no-search-results-icon{font-size:10em;color:#4e5d6c;}.no-search-results .no-search-results-text{margin:20px 0;color:#ccc;}.form-control-search{padding:25px 105px 25px 16px;}.form-control-withbuttons{padding-right:105px;}.input-group-addon .btn-group{position:absolute;right:45px;z-index:3;top:13px;box-shadow:0 0 0;}.input-group-addon .btn-group .btn{border:1px solid rgba(255,255,255,.7) !important;padding:3px 12px;color:rgba(255,255,255,.7) !important;}
|
@media(min-width:768px){.row{position:relative;}.bottom-align-text{position:absolute;bottom:0;right:0;}}@media(max-width:48em){.home{padding-top:1rem;}}@media(min-width:48em){.home{padding-top:4rem;}}.btn{border-radius:.25rem !important;}.multiSelect{background-color:#4e5d6c;}.form-control-custom{background-color:#4e5d6c !important;color:#fff !important;border-radius:0;box-shadow:0 0 0 !important;}h1{font-size:3.5rem !important;font-weight:600 !important;}.request-title{margin-top:0 !important;font-size:1.9rem !important;}p{font-size:1.1rem !important;}label{display:inline-block !important;margin-bottom:.5rem !important;font-size:16px !important;}.nav-tabs>li.active>a,.nav-tabs>li.active>a:hover,.nav-tabs>li.active>a:focus{background:#4e5d6c;}.navbar .nav a .fa,.dropdown-menu a .fa{font-size:130%;top:1px;position:relative;display:inline-block;margin-right:5px;}.dropdown-menu a .fa{top:2px;}.btn-danger-outline{color:#d9534f !important;background-color:transparent;background-image:none;border-color:#d9534f !important;}.btn-danger-outline:focus,.btn-danger-outline.focus,.btn-danger-outline:active,.btn-danger-outline.active,.btn-danger-outline:hover,.open>.btn-danger-outline.dropdown-toggle{color:#fff !important;background-color:#d9534f !important;border-color:#d9534f !important;}.btn-primary-outline{color:#ff761b !important;background-color:transparent;background-image:none;border-color:#ff761b !important;}.btn-primary-outline:focus,.btn-primary-outline.focus,.btn-primary-outline:active,.btn-primary-outline.active,.btn-primary-outline:hover,.open>.btn-primary-outline.dropdown-toggle{color:#fff !important;background-color:#df691a !important;border-color:#df691a !important;}.btn-info-outline{color:#5bc0de !important;background-color:transparent;background-image:none;border-color:#5bc0de !important;}.btn-info-outline:focus,.btn-info-outline.focus,.btn-info-outline:active,.btn-info-outline.active,.btn-info-outline:hover,.open>.btn-info-outline.dropdown-toggle{color:#fff !important;background-color:#5bc0de !important;border-color:#5bc0de !important;}.btn-warning-outline{color:#f0ad4e !important;background-color:transparent;background-image:none;border-color:#f0ad4e !important;}.btn-warning-outline:focus,.btn-warning-outline.focus,.btn-warning-outline:active,.btn-warning-outline.active,.btn-warning-outline:hover,.open>.btn-warning-outline.dropdown-toggle{color:#fff !important;background-color:#f0ad4e !important;border-color:#f0ad4e !important;}.btn-success-outline{color:#5cb85c !important;background-color:transparent;background-image:none;border-color:#5cb85c !important;}.btn-success-outline:focus,.btn-success-outline.focus,.btn-success-outline:active,.btn-success-outline.active,.btn-success-outline:hover,.open>.btn-success-outline.dropdown-toggle{color:#fff !important;background-color:#5cb85c !important;border-color:#5cb85c !important;}#movieList .mix{display:none;}#tvList .mix{display:none;}.scroll-top-wrapper{position:fixed;opacity:0;visibility:hidden;overflow:hidden;text-align:center;z-index:99999999;background-color:#4e5d6c;color:#eee;width:50px;height:48px;line-height:48px;right:30px;bottom:30px;padding-top:2px;border-top-left-radius:10px;border-top-right-radius:10px;border-bottom-right-radius:10px;border-bottom-left-radius:10px;-webkit-transition:all .5s ease-in-out;-moz-transition:all .5s ease-in-out;-ms-transition:all .5s ease-in-out;-o-transition:all .5s ease-in-out;transition:all .5s ease-in-out;}.scroll-top-wrapper:hover{background-color:#637689;}.scroll-top-wrapper.show{visibility:visible;cursor:pointer;opacity:1;}.scroll-top-wrapper i.fa{line-height:inherit;}.no-search-results{text-align:center;}.no-search-results .no-search-results-icon{font-size:10em;color:#4e5d6c;}.no-search-results .no-search-results-text{margin:20px 0;color:#ccc;}.form-control-search{padding:25px 105px 25px 16px;}.form-control-withbuttons{padding-right:105px;}.input-group-addon .btn-group{position:absolute;right:45px;z-index:3;top:13px;box-shadow:0 0 0;}.input-group-addon .btn-group .btn{border:1px solid rgba(255,255,255,.7) !important;padding:3px 12px;color:rgba(255,255,255,.7) !important;}.btn-split .btn{border-radius:0 !important;}.btn-split .btn:not(.dropdown-toggle){border-radius:.25rem 0 0 .25rem !important;}.btn-split .btn.dropdown-toggle{border-radius:0 .25rem .25rem 0 !important;padding:12px 8px;}
|
|
@ -266,3 +266,16 @@ $border-radius: 10px;
|
||||||
padding: 3px 12px;
|
padding: 3px 12px;
|
||||||
color: rgba(255,255,255,.7) !important;
|
color: rgba(255,255,255,.7) !important;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.btn-split .btn {
|
||||||
|
border-radius: 0 !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn-split .btn:not(.dropdown-toggle) {
|
||||||
|
border-radius: .25rem 0 0 .25rem $i;
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn-split .btn.dropdown-toggle {
|
||||||
|
border-radius: 0 .25rem .25rem 0 $i;
|
||||||
|
padding: 12px 8px;
|
||||||
|
}
|
|
@ -1,8 +1,8 @@
|
||||||
Handlebars.registerHelper('if_eq', function (a, b, opts) {
|
Handlebars.registerHelper('if_eq', function (a, b, opts) {
|
||||||
if (a == b)
|
if (a == b)
|
||||||
return opts.fn(this);
|
return !opts ? null : opts.fn(this);
|
||||||
else
|
else
|
||||||
return opts.inverse(this);
|
return !opts ? null : opts.inverse(this);
|
||||||
});
|
});
|
||||||
|
|
||||||
var searchSource = $("#search-template").html();
|
var searchSource = $("#search-template").html();
|
||||||
|
@ -268,37 +268,44 @@ $(document).on("click", ".delete", function (e) {
|
||||||
// Approve single request
|
// Approve single request
|
||||||
$(document).on("click", ".approve", function (e) {
|
$(document).on("click", ".approve", function (e) {
|
||||||
e.preventDefault();
|
e.preventDefault();
|
||||||
var buttonId = e.target.id;
|
var $this = $(this);
|
||||||
var $form = $('#approve' + buttonId);
|
var $form = $this.parents('form').first();
|
||||||
|
|
||||||
if ($('#' + buttonId).text() === " Loading...") {
|
if ($this.text() === " Loading...") {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
loadingButton(buttonId, "success");
|
loadingButton($this.attr('id'), "success");
|
||||||
|
|
||||||
$.ajax({
|
approveRequest($form, null, function () {
|
||||||
type: $form.prop('method'),
|
$("#" + $this.attr('id') + "notapproved").prop("class", "fa fa-check");
|
||||||
url: $form.prop('action'),
|
|
||||||
data: $form.serialize(),
|
var $group = $this.parent('.btn-split');
|
||||||
dataType: "json",
|
if ($group.length > 0) {
|
||||||
success: function (response) {
|
$group.remove();
|
||||||
|
|
||||||
if (checkJsonResponse(response)) {
|
|
||||||
if (response.message) {
|
|
||||||
generateNotify(response.message, "success");
|
|
||||||
} else {
|
|
||||||
generateNotify("Success! Request Approved.", "success");
|
|
||||||
}
|
|
||||||
|
|
||||||
$("button[custom-button='" + buttonId + "']").remove();
|
|
||||||
$("#" + buttonId + "notapproved").prop("class", "fa fa-check");
|
|
||||||
}
|
|
||||||
},
|
|
||||||
error: function (e) {
|
|
||||||
console.log(e);
|
|
||||||
generateNotify("Something went wrong!", "danger");
|
|
||||||
}
|
}
|
||||||
|
else {
|
||||||
|
$this.remove();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
$(document).on("click", ".approve-with-quality", function (e) {
|
||||||
|
e.preventDefault();
|
||||||
|
var $this = $(this);
|
||||||
|
var $button = $this.parents('.btn-split').children('.approve').first();
|
||||||
|
var qualityId = e.target.id
|
||||||
|
var $form = $this.parents('form').first();
|
||||||
|
|
||||||
|
if ($button.text() === " Loading...") {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
loadingButton($button.attr('id'), "success");
|
||||||
|
|
||||||
|
approveRequest($form, qualityId, function () {
|
||||||
|
$("#" + $button.attr('id') + "notapproved").prop("class", "fa fa-check");
|
||||||
|
$this.parents('.btn-split').remove();
|
||||||
});
|
});
|
||||||
|
|
||||||
});
|
});
|
||||||
|
@ -372,6 +379,37 @@ $(document).on("click", ".change", function (e) {
|
||||||
|
|
||||||
});
|
});
|
||||||
|
|
||||||
|
function approveRequest($form, qualityId, successCallback) {
|
||||||
|
|
||||||
|
var formData = $form.serialize();
|
||||||
|
if (qualityId) formData += ("&qualityId=" + qualityId);
|
||||||
|
|
||||||
|
$.ajax({
|
||||||
|
type: $form.prop('method'),
|
||||||
|
url: $form.prop('action'),
|
||||||
|
data: formData,
|
||||||
|
dataType: "json",
|
||||||
|
success: function (response) {
|
||||||
|
|
||||||
|
if (checkJsonResponse(response)) {
|
||||||
|
if (response.message) {
|
||||||
|
generateNotify(response.message, "success");
|
||||||
|
} else {
|
||||||
|
generateNotify("Success! Request Approved.", "success");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (successCallback) {
|
||||||
|
successCallback();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
error: function (e) {
|
||||||
|
console.log(e);
|
||||||
|
generateNotify("Something went wrong!", "danger");
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
function mixItUpConfig(activeState) {
|
function mixItUpConfig(activeState) {
|
||||||
var conf = mixItUpDefault;
|
var conf = mixItUpDefault;
|
||||||
|
|
||||||
|
@ -483,7 +521,8 @@ function buildRequestContext(result, type) {
|
||||||
imdb: result.imdbId,
|
imdb: result.imdbId,
|
||||||
seriesRequested: result.tvSeriesRequestType,
|
seriesRequested: result.tvSeriesRequestType,
|
||||||
coverArtUrl: result.coverArtUrl,
|
coverArtUrl: result.coverArtUrl,
|
||||||
|
qualities: result.qualities,
|
||||||
|
hasQualities: result.qualities && result.qualities.length > 0
|
||||||
};
|
};
|
||||||
|
|
||||||
return context;
|
return context;
|
||||||
|
|
|
@ -20,6 +20,10 @@ $(function () {
|
||||||
$tabs.first().children('a:first-child').tab('show');
|
$tabs.first().children('a:first-child').tab('show');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
$('a[data-toggle="tab"]').on('shown.bs.tab', function (e) {
|
||||||
|
focusSearch($($(e.target).attr('href')))
|
||||||
|
});
|
||||||
|
focusSearch($('li.active a', '#nav-tabs').first().attr('href'));
|
||||||
|
|
||||||
// Type in movie search
|
// Type in movie search
|
||||||
$("#movieSearchContent").on("input", function () {
|
$("#movieSearchContent").on("input", function () {
|
||||||
|
@ -132,9 +136,14 @@ $(function () {
|
||||||
var data = $form.serialize();
|
var data = $form.serialize();
|
||||||
|
|
||||||
sendRequestAjax(data, type, url, buttonId);
|
sendRequestAjax(data, type, url, buttonId);
|
||||||
|
|
||||||
});
|
});
|
||||||
|
|
||||||
|
function focusSearch($content) {
|
||||||
|
if ($content.length > 0) {
|
||||||
|
$('input[type=text].form-control', $content).first().focus();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
function sendRequestAjax(data, type, url, buttonId) {
|
function sendRequestAjax(data, type, url, buttonId) {
|
||||||
$.ajax({
|
$.ajax({
|
||||||
type: type,
|
type: type,
|
||||||
|
|
|
@ -35,15 +35,25 @@ function checkJsonResponse(response) {
|
||||||
}
|
}
|
||||||
|
|
||||||
function loadingButton(elementId, originalCss) {
|
function loadingButton(elementId, originalCss) {
|
||||||
$('#' + elementId).removeClass("btn-" + originalCss + "-outline");
|
var $element = $('#' + elementId);
|
||||||
$('#' + elementId).addClass("btn-primary-outline");
|
$element.removeClass("btn-" + originalCss + "-outline").addClass("btn-primary-outline").addClass('disabled').html("<i class='fa fa-spinner fa-spin'></i> Loading...");
|
||||||
$('#' + elementId).html("<i class='fa fa-spinner fa-spin'></i> Loading...");
|
|
||||||
|
// handle split-buttons
|
||||||
|
var $dropdown = $element.next('.dropdown-toggle')
|
||||||
|
if ($dropdown.length > 0) {
|
||||||
|
$dropdown.removeClass("btn-" + originalCss + "-outline").addClass("btn-primary-outline").addClass('disabled');
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function finishLoading(elementId, originalCss, html) {
|
function finishLoading(elementId, originalCss, html) {
|
||||||
$('#' + elementId).removeClass("btn-primary-outline");
|
var $element = $('#' + elementId);
|
||||||
$('#' + elementId).addClass("btn-" + originalCss + "-outline");
|
$element.removeClass("btn-primary-outline").removeClass('disabled').addClass("btn-" + originalCss + "-outline").html(html);
|
||||||
$('#' + elementId).html(html);
|
|
||||||
|
// handle split-buttons
|
||||||
|
var $dropdown = $element.next('.dropdown-toggle')
|
||||||
|
if ($dropdown.length > 0) {
|
||||||
|
$dropdown.removeClass("btn-primary-outline").removeClass('disabled').addClass("btn-" + originalCss + "-outline");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
var noResultsHtml = "<div class='no-search-results'>" +
|
var noResultsHtml = "<div class='no-search-results'>" +
|
||||||
|
|
|
@ -31,6 +31,7 @@ using PlexRequests.Api.Models.Sonarr;
|
||||||
using PlexRequests.Core.SettingModels;
|
using PlexRequests.Core.SettingModels;
|
||||||
using PlexRequests.Helpers;
|
using PlexRequests.Helpers;
|
||||||
using PlexRequests.Store;
|
using PlexRequests.Store;
|
||||||
|
using System.Linq;
|
||||||
|
|
||||||
namespace PlexRequests.UI.Helpers
|
namespace PlexRequests.UI.Helpers
|
||||||
{
|
{
|
||||||
|
@ -46,9 +47,19 @@ namespace PlexRequests.UI.Helpers
|
||||||
private static Logger Log = LogManager.GetCurrentClassLogger();
|
private static Logger Log = LogManager.GetCurrentClassLogger();
|
||||||
|
|
||||||
public SonarrAddSeries SendToSonarr(SonarrSettings sonarrSettings, RequestedModel model)
|
public SonarrAddSeries SendToSonarr(SonarrSettings sonarrSettings, RequestedModel model)
|
||||||
|
{
|
||||||
|
return SendToSonarr(sonarrSettings, model, string.Empty);
|
||||||
|
}
|
||||||
|
|
||||||
|
public SonarrAddSeries SendToSonarr(SonarrSettings sonarrSettings, RequestedModel model, string qualityId)
|
||||||
{
|
{
|
||||||
int qualityProfile;
|
int qualityProfile;
|
||||||
int.TryParse(sonarrSettings.QualityProfile, out qualityProfile);
|
|
||||||
|
if (!string.IsNullOrEmpty(qualityId) || !int.TryParse(qualityId, out qualityProfile)) // try to parse the passed in quality, otherwise use the settings default quality
|
||||||
|
{
|
||||||
|
int.TryParse(sonarrSettings.QualityProfile, out qualityProfile);
|
||||||
|
}
|
||||||
|
|
||||||
var result = SonarrApi.AddSeries(model.ProviderId, model.Title, qualityProfile,
|
var result = SonarrApi.AddSeries(model.ProviderId, model.Title, qualityProfile,
|
||||||
sonarrSettings.SeasonFolders, sonarrSettings.RootPath, model.SeasonCount, model.SeasonList, sonarrSettings.ApiKey,
|
sonarrSettings.SeasonFolders, sonarrSettings.RootPath, model.SeasonCount, model.SeasonList, sonarrSettings.ApiKey,
|
||||||
sonarrSettings.FullUri);
|
sonarrSettings.FullUri);
|
||||||
|
@ -61,7 +72,17 @@ namespace PlexRequests.UI.Helpers
|
||||||
|
|
||||||
public SickRageTvAdd SendToSickRage(SickRageSettings sickRageSettings, RequestedModel model)
|
public SickRageTvAdd SendToSickRage(SickRageSettings sickRageSettings, RequestedModel model)
|
||||||
{
|
{
|
||||||
var result = SickrageApi.AddSeries(model.ProviderId, model.SeasonCount, model.SeasonList, sickRageSettings.QualityProfile,
|
return SendToSickRage(sickRageSettings, model, sickRageSettings.QualityProfile);
|
||||||
|
}
|
||||||
|
|
||||||
|
public SickRageTvAdd SendToSickRage(SickRageSettings sickRageSettings, RequestedModel model, string qualityId)
|
||||||
|
{
|
||||||
|
if (!sickRageSettings.Qualities.Any(x => x.Key == qualityId))
|
||||||
|
{
|
||||||
|
qualityId = sickRageSettings.QualityProfile;
|
||||||
|
}
|
||||||
|
|
||||||
|
var result = SickrageApi.AddSeries(model.ProviderId, model.SeasonCount, model.SeasonList, qualityId,
|
||||||
sickRageSettings.ApiKey, sickRageSettings.FullUri);
|
sickRageSettings.ApiKey, sickRageSettings.FullUri);
|
||||||
|
|
||||||
Log.Trace("SickRage Add Result: ");
|
Log.Trace("SickRage Add Result: ");
|
||||||
|
|
8
PlexRequests.UI/Models/QualityModel.cs
Normal file
8
PlexRequests.UI/Models/QualityModel.cs
Normal file
|
@ -0,0 +1,8 @@
|
||||||
|
namespace PlexRequests.UI.Models
|
||||||
|
{
|
||||||
|
public class QualityModel
|
||||||
|
{
|
||||||
|
public string Id { get; set; }
|
||||||
|
public string Name { get; set; }
|
||||||
|
}
|
||||||
|
}
|
|
@ -52,5 +52,6 @@ namespace PlexRequests.UI.Models
|
||||||
public string AdminNotes { get; set; }
|
public string AdminNotes { get; set; }
|
||||||
public string TvSeriesRequestType { get; set; }
|
public string TvSeriesRequestType { get; set; }
|
||||||
public string MusicBrainzId { get; set; }
|
public string MusicBrainzId { get; set; }
|
||||||
|
public QualityModel[] Qualities { get; set; }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -59,7 +59,7 @@ namespace PlexRequests.UI.Modules
|
||||||
SickRageApi = srApi;
|
SickRageApi = srApi;
|
||||||
SickRageSettings = srSettings;
|
SickRageSettings = srSettings;
|
||||||
|
|
||||||
Post["/approve"] = parameters => Approve((int)Request.Form.requestid);
|
Post["/approve"] = parameters => Approve((int)Request.Form.requestid, (string)Request.Form.qualityId);
|
||||||
Post["/approveall"] = x => ApproveAll();
|
Post["/approveall"] = x => ApproveAll();
|
||||||
Post["/approveallmovies"] = x => ApproveAllMovies();
|
Post["/approveallmovies"] = x => ApproveAllMovies();
|
||||||
Post["/approvealltvshows"] = x => ApproveAllTVShows();
|
Post["/approvealltvshows"] = x => ApproveAllTVShows();
|
||||||
|
@ -80,7 +80,7 @@ namespace PlexRequests.UI.Modules
|
||||||
/// </summary>
|
/// </summary>
|
||||||
/// <param name="requestId">The request identifier.</param>
|
/// <param name="requestId">The request identifier.</param>
|
||||||
/// <returns></returns>
|
/// <returns></returns>
|
||||||
private Response Approve(int requestId)
|
private Response Approve(int requestId, string qualityId)
|
||||||
{
|
{
|
||||||
Log.Info("approving request {0}", requestId);
|
Log.Info("approving request {0}", requestId);
|
||||||
if (!Context.CurrentUser.IsAuthenticated())
|
if (!Context.CurrentUser.IsAuthenticated())
|
||||||
|
@ -99,15 +99,15 @@ namespace PlexRequests.UI.Modules
|
||||||
switch (request.Type)
|
switch (request.Type)
|
||||||
{
|
{
|
||||||
case RequestType.Movie:
|
case RequestType.Movie:
|
||||||
return RequestMovieAndUpdateStatus(request);
|
return RequestMovieAndUpdateStatus(request, qualityId);
|
||||||
case RequestType.TvShow:
|
case RequestType.TvShow:
|
||||||
return RequestTvAndUpdateStatus(request);
|
return RequestTvAndUpdateStatus(request, qualityId);
|
||||||
default:
|
default:
|
||||||
throw new ArgumentOutOfRangeException(nameof(request));
|
throw new ArgumentOutOfRangeException(nameof(request));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private Response RequestTvAndUpdateStatus(RequestedModel request)
|
private Response RequestTvAndUpdateStatus(RequestedModel request, string qualityId)
|
||||||
{
|
{
|
||||||
var sender = new TvSender(SonarrApi, SickRageApi);
|
var sender = new TvSender(SonarrApi, SickRageApi);
|
||||||
|
|
||||||
|
@ -115,7 +115,7 @@ namespace PlexRequests.UI.Modules
|
||||||
if (sonarrSettings.Enabled)
|
if (sonarrSettings.Enabled)
|
||||||
{
|
{
|
||||||
Log.Trace("Sending to Sonarr");
|
Log.Trace("Sending to Sonarr");
|
||||||
var result = sender.SendToSonarr(sonarrSettings, request);
|
var result = sender.SendToSonarr(sonarrSettings, request, qualityId);
|
||||||
Log.Trace("Sonarr Result: ");
|
Log.Trace("Sonarr Result: ");
|
||||||
Log.Trace(result.DumpJson());
|
Log.Trace(result.DumpJson());
|
||||||
if (!string.IsNullOrEmpty(result.title))
|
if (!string.IsNullOrEmpty(result.title))
|
||||||
|
@ -141,7 +141,7 @@ namespace PlexRequests.UI.Modules
|
||||||
if (srSettings.Enabled)
|
if (srSettings.Enabled)
|
||||||
{
|
{
|
||||||
Log.Trace("Sending to SickRage");
|
Log.Trace("Sending to SickRage");
|
||||||
var result = sender.SendToSickRage(srSettings, request);
|
var result = sender.SendToSickRage(srSettings, request, qualityId);
|
||||||
Log.Trace("SickRage Result: ");
|
Log.Trace("SickRage Result: ");
|
||||||
Log.Trace(result.DumpJson());
|
Log.Trace(result.DumpJson());
|
||||||
if (result?.result == "success")
|
if (result?.result == "success")
|
||||||
|
@ -169,7 +169,7 @@ namespace PlexRequests.UI.Modules
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
private Response RequestMovieAndUpdateStatus(RequestedModel request)
|
private Response RequestMovieAndUpdateStatus(RequestedModel request, string qualityId)
|
||||||
{
|
{
|
||||||
var cpSettings = CpService.GetSettings();
|
var cpSettings = CpService.GetSettings();
|
||||||
var cp = new CouchPotatoApi();
|
var cp = new CouchPotatoApi();
|
||||||
|
@ -190,7 +190,8 @@ namespace PlexRequests.UI.Modules
|
||||||
Message = "We could not approve this request. Please try again or check the logs."
|
Message = "We could not approve this request. Please try again or check the logs."
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
var result = cp.AddMovie(request.ImdbId, cpSettings.ApiKey, request.Title, cpSettings.FullUri, cpSettings.ProfileId);
|
|
||||||
|
var result = cp.AddMovie(request.ImdbId, cpSettings.ApiKey, request.Title, cpSettings.FullUri, string.IsNullOrEmpty(qualityId) ? cpSettings.ProfileId : qualityId);
|
||||||
Log.Trace("Adding movie to CP result {0}", result);
|
Log.Trace("Adding movie to CP result {0}", result);
|
||||||
if (result)
|
if (result)
|
||||||
{
|
{
|
||||||
|
|
|
@ -41,17 +41,36 @@ using PlexRequests.Services.Notification;
|
||||||
using PlexRequests.Store;
|
using PlexRequests.Store;
|
||||||
using PlexRequests.UI.Models;
|
using PlexRequests.UI.Models;
|
||||||
using PlexRequests.Helpers;
|
using PlexRequests.Helpers;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using PlexRequests.Api.Interfaces;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
|
||||||
namespace PlexRequests.UI.Modules
|
namespace PlexRequests.UI.Modules
|
||||||
{
|
{
|
||||||
public class RequestsModule : BaseModule
|
public class RequestsModule : BaseModule
|
||||||
{
|
{
|
||||||
public RequestsModule(IRequestService service, ISettingsService<PlexRequestSettings> prSettings, ISettingsService<PlexSettings> plex, INotificationService notify) : base("requests")
|
public RequestsModule(
|
||||||
|
IRequestService service,
|
||||||
|
ISettingsService<PlexRequestSettings> prSettings,
|
||||||
|
ISettingsService<PlexSettings> plex,
|
||||||
|
INotificationService notify,
|
||||||
|
ISettingsService<SonarrSettings> sonarrSettings,
|
||||||
|
ISettingsService<SickRageSettings> sickRageSettings,
|
||||||
|
ISettingsService<CouchPotatoSettings> cpSettings,
|
||||||
|
ICouchPotatoApi cpApi,
|
||||||
|
ISonarrApi sonarrApi,
|
||||||
|
ISickRageApi sickRageApi) : base("requests")
|
||||||
{
|
{
|
||||||
Service = service;
|
Service = service;
|
||||||
PrSettings = prSettings;
|
PrSettings = prSettings;
|
||||||
PlexSettings = plex;
|
PlexSettings = plex;
|
||||||
NotificationService = notify;
|
NotificationService = notify;
|
||||||
|
SonarrSettings = sonarrSettings;
|
||||||
|
SickRageSettings = sickRageSettings;
|
||||||
|
CpSettings = cpSettings;
|
||||||
|
SonarrApi = sonarrApi;
|
||||||
|
SickRageApi = sickRageApi;
|
||||||
|
CpApi = cpApi;
|
||||||
|
|
||||||
Get["/"] = _ => LoadRequests();
|
Get["/"] = _ => LoadRequests();
|
||||||
Get["/movies"] = _ => GetMovies();
|
Get["/movies"] = _ => GetMovies();
|
||||||
|
@ -71,6 +90,12 @@ namespace PlexRequests.UI.Modules
|
||||||
private INotificationService NotificationService { get; }
|
private INotificationService NotificationService { get; }
|
||||||
private ISettingsService<PlexRequestSettings> PrSettings { get; }
|
private ISettingsService<PlexRequestSettings> PrSettings { get; }
|
||||||
private ISettingsService<PlexSettings> PlexSettings { get; }
|
private ISettingsService<PlexSettings> PlexSettings { get; }
|
||||||
|
private ISettingsService<SonarrSettings> SonarrSettings { get; }
|
||||||
|
private ISettingsService<SickRageSettings> SickRageSettings { get; }
|
||||||
|
private ISettingsService<CouchPotatoSettings> CpSettings { get; }
|
||||||
|
private ISonarrApi SonarrApi { get; }
|
||||||
|
private ISickRageApi SickRageApi { get; }
|
||||||
|
private ICouchPotatoApi CpApi { get; }
|
||||||
|
|
||||||
private Negotiator LoadRequests()
|
private Negotiator LoadRequests()
|
||||||
{
|
{
|
||||||
|
@ -78,17 +103,51 @@ namespace PlexRequests.UI.Modules
|
||||||
return View["Index", settings];
|
return View["Index", settings];
|
||||||
}
|
}
|
||||||
|
|
||||||
private Response GetMovies()
|
private Response GetMovies() // TODO: async await the API calls
|
||||||
{
|
{
|
||||||
var settings = PrSettings.GetSettings();
|
var settings = PrSettings.GetSettings();
|
||||||
var isAdmin = Context.CurrentUser.IsAuthenticated();
|
var isAdmin = Context.CurrentUser.IsAuthenticated();
|
||||||
var dbMovies = Service.GetAll().Where(x => x.Type == RequestType.Movie);
|
|
||||||
if (settings.UsersCanViewOnlyOwnRequests && !isAdmin)
|
List<Task> taskList = new List<Task>();
|
||||||
|
|
||||||
|
List<RequestedModel> dbMovies = new List<RequestedModel>();
|
||||||
|
taskList.Add(Task.Factory.StartNew(() =>
|
||||||
{
|
{
|
||||||
dbMovies = dbMovies.Where(x => x.UserHasRequested(Username));
|
return Service.GetAll().Where(x => x.Type == RequestType.Movie);
|
||||||
|
|
||||||
|
}).ContinueWith((t) =>
|
||||||
|
{
|
||||||
|
dbMovies = t.Result.ToList();
|
||||||
|
|
||||||
|
if (settings.UsersCanViewOnlyOwnRequests && !isAdmin)
|
||||||
|
{
|
||||||
|
dbMovies = dbMovies.Where(x => x.UserHasRequested(Username)).ToList();
|
||||||
|
}
|
||||||
|
}));
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
List<QualityModel> qualities = new List<QualityModel>();
|
||||||
|
|
||||||
|
if (isAdmin)
|
||||||
|
{
|
||||||
|
var cpSettings = CpSettings.GetSettings();
|
||||||
|
if (cpSettings.Enabled)
|
||||||
|
{
|
||||||
|
taskList.Add(Task.Factory.StartNew(() =>
|
||||||
|
{
|
||||||
|
return CpApi.GetProfiles(cpSettings.FullUri, cpSettings.ApiKey).list.Select(x => new QualityModel() { Id = x._id, Name = x.label }); // TODO: cache this!
|
||||||
|
}).ContinueWith((t) =>
|
||||||
|
{
|
||||||
|
qualities = t.Result.ToList();
|
||||||
|
}));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
var viewModel = dbMovies.Select(movie => {
|
Task.WaitAll(taskList.ToArray());
|
||||||
|
|
||||||
|
var viewModel = dbMovies.Select(movie =>
|
||||||
|
{
|
||||||
return new RequestViewModel
|
return new RequestViewModel
|
||||||
{
|
{
|
||||||
ProviderId = movie.ProviderId,
|
ProviderId = movie.ProviderId,
|
||||||
|
@ -111,23 +170,69 @@ namespace PlexRequests.UI.Modules
|
||||||
Issues = movie.Issues.Humanize(LetterCasing.Title),
|
Issues = movie.Issues.Humanize(LetterCasing.Title),
|
||||||
OtherMessage = movie.OtherMessage,
|
OtherMessage = movie.OtherMessage,
|
||||||
AdminNotes = movie.AdminNote,
|
AdminNotes = movie.AdminNote,
|
||||||
|
Qualities = qualities.ToArray()
|
||||||
};
|
};
|
||||||
}).ToList();
|
}).ToList();
|
||||||
|
|
||||||
return Response.AsJson(viewModel);
|
return Response.AsJson(viewModel);
|
||||||
}
|
}
|
||||||
|
|
||||||
private Response GetTvShows()
|
private Response GetTvShows() // TODO: async await the API calls
|
||||||
{
|
{
|
||||||
var settings = PrSettings.GetSettings();
|
var settings = PrSettings.GetSettings();
|
||||||
var isAdmin = Context.CurrentUser.IsAuthenticated();
|
var isAdmin = Context.CurrentUser.IsAuthenticated();
|
||||||
var dbTv = Service.GetAll().Where(x => x.Type == RequestType.TvShow);
|
|
||||||
if (settings.UsersCanViewOnlyOwnRequests && !isAdmin)
|
List<Task> taskList = new List<Task>();
|
||||||
|
|
||||||
|
List<RequestedModel> dbTv = new List<RequestedModel>();
|
||||||
|
taskList.Add(Task.Factory.StartNew(() =>
|
||||||
{
|
{
|
||||||
dbTv = dbTv.Where(x => x.UserHasRequested(Username));
|
return Service.GetAll().Where(x => x.Type == RequestType.TvShow);
|
||||||
|
|
||||||
|
}).ContinueWith((t) =>
|
||||||
|
{
|
||||||
|
dbTv = t.Result.ToList();
|
||||||
|
|
||||||
|
if (settings.UsersCanViewOnlyOwnRequests && !isAdmin)
|
||||||
|
{
|
||||||
|
dbTv = dbTv.Where(x => x.UserHasRequested(Username)).ToList();
|
||||||
|
}
|
||||||
|
}));
|
||||||
|
|
||||||
|
List<QualityModel> qualities = new List<QualityModel>();
|
||||||
|
if (isAdmin)
|
||||||
|
{
|
||||||
|
var sonarrSettings = SonarrSettings.GetSettings();
|
||||||
|
if (sonarrSettings.Enabled)
|
||||||
|
{
|
||||||
|
taskList.Add(Task.Factory.StartNew(() =>
|
||||||
|
{
|
||||||
|
return SonarrApi.GetProfiles(sonarrSettings.ApiKey, sonarrSettings.FullUri).Select(x => new QualityModel() { Id = x.id.ToString(), Name = x.name }); // TODO: cache this!
|
||||||
|
}).ContinueWith((t) =>
|
||||||
|
{
|
||||||
|
qualities = t.Result.ToList();
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
var sickRageSettings = SickRageSettings.GetSettings();
|
||||||
|
if (sickRageSettings.Enabled)
|
||||||
|
{
|
||||||
|
taskList.Add(Task.Factory.StartNew(() =>
|
||||||
|
{
|
||||||
|
return sickRageSettings.Qualities.Select(x => new QualityModel() { Id = x.Key, Name = x.Value }); // TODO: cache this!
|
||||||
|
}).ContinueWith((t) =>
|
||||||
|
{
|
||||||
|
qualities = t.Result.ToList();
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
var viewModel = dbTv.Select(tv => {
|
Task.WaitAll(taskList.ToArray());
|
||||||
|
|
||||||
|
var viewModel = dbTv.Select(tv =>
|
||||||
|
{
|
||||||
return new RequestViewModel
|
return new RequestViewModel
|
||||||
{
|
{
|
||||||
ProviderId = tv.ProviderId,
|
ProviderId = tv.ProviderId,
|
||||||
|
@ -150,7 +255,8 @@ namespace PlexRequests.UI.Modules
|
||||||
Issues = tv.Issues.Humanize(LetterCasing.Title),
|
Issues = tv.Issues.Humanize(LetterCasing.Title),
|
||||||
OtherMessage = tv.OtherMessage,
|
OtherMessage = tv.OtherMessage,
|
||||||
AdminNotes = tv.AdminNote,
|
AdminNotes = tv.AdminNote,
|
||||||
TvSeriesRequestType = tv.SeasonsRequested
|
TvSeriesRequestType = tv.SeasonsRequested,
|
||||||
|
Qualities = qualities.ToArray()
|
||||||
};
|
};
|
||||||
}).ToList();
|
}).ToList();
|
||||||
|
|
||||||
|
@ -167,7 +273,8 @@ namespace PlexRequests.UI.Modules
|
||||||
dbAlbum = dbAlbum.Where(x => x.UserHasRequested(Username));
|
dbAlbum = dbAlbum.Where(x => x.UserHasRequested(Username));
|
||||||
}
|
}
|
||||||
|
|
||||||
var viewModel = dbAlbum.Select(album => {
|
var viewModel = dbAlbum.Select(album =>
|
||||||
|
{
|
||||||
return new RequestViewModel
|
return new RequestViewModel
|
||||||
{
|
{
|
||||||
ProviderId = album.ProviderId,
|
ProviderId = album.ProviderId,
|
||||||
|
|
|
@ -171,6 +171,7 @@
|
||||||
<Compile Include="Helpers\TvSender.cs" />
|
<Compile Include="Helpers\TvSender.cs" />
|
||||||
<Compile Include="Helpers\ValidationHelper.cs" />
|
<Compile Include="Helpers\ValidationHelper.cs" />
|
||||||
<Compile Include="Models\DatatablesModel.cs" />
|
<Compile Include="Models\DatatablesModel.cs" />
|
||||||
|
<Compile Include="Models\QualityModel.cs" />
|
||||||
<Compile Include="Models\SearchMusicViewModel.cs" />
|
<Compile Include="Models\SearchMusicViewModel.cs" />
|
||||||
<Compile Include="Validators\PushoverSettingsValidator.cs" />
|
<Compile Include="Validators\PushoverSettingsValidator.cs" />
|
||||||
<Compile Include="Validators\PushbulletSettingsValidator.cs" />
|
<Compile Include="Validators\PushbulletSettingsValidator.cs" />
|
||||||
|
|
|
@ -75,15 +75,10 @@
|
||||||
<label for="profiles" class="control-label ">Quality Profiles</label>
|
<label for="profiles" class="control-label ">Quality Profiles</label>
|
||||||
<div id="profiles">
|
<div id="profiles">
|
||||||
<select class="form-control form-control-custom" value="selected">
|
<select class="form-control form-control-custom" value="selected">
|
||||||
<option id="default" value="default">Use Deafult</option>
|
@foreach (var quality in Model.Qualities)
|
||||||
<option id="sdtv" value="sdtv">SD TV</option>
|
{
|
||||||
<option id="sddvd" value="sddvd">SD DVD</option>
|
<option id="@quality.Key" value="@quality.Key">@quality.Value</option>
|
||||||
<option id="hdtv" value="hdtv">HD TV</option>
|
}
|
||||||
<option id="rawhdtv" value="rawhdtv">Raw HD TV</option>
|
|
||||||
<option id="hdwebdl" value="hdwebdl">HD Web DL</option>
|
|
||||||
<option id="fullhdwebdl" value="fullhdwebdl">Full HD Web DL</option>
|
|
||||||
<option id="hdbluray" value="hdbluray">HD Bluray</option>
|
|
||||||
<option id="fullhdbluray" value="fullhdbluray">Full HD Bluray</option>
|
|
||||||
</select>
|
</select>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -177,12 +177,27 @@
|
||||||
{{/if}}
|
{{/if}}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="col-sm-2 col-sm-push-3">
|
<div class="col-sm-3 col-sm-push-3">
|
||||||
{{#if_eq admin true}}
|
{{#if_eq admin true}}
|
||||||
{{#if_eq approved false}}
|
{{#if_eq approved false}}
|
||||||
<form method="POST" action="/approval/approve" id="approve{{requestId}}">
|
<form method="POST" action="/approval/approve" id="approve{{requestId}}">
|
||||||
<input name="requestId" type="text" value="{{requestId}}" hidden="hidden" />
|
<input name="requestId" type="text" value="{{requestId}}" hidden="hidden" />
|
||||||
<button id="{{requestId}}" custom-button="{{requestId}}" style="text-align: right" class="btn btn-sm btn-success-outline approve" type="submit"><i class="fa fa-plus"></i> Approve</button>
|
{{#if_eq hasQualities true}}
|
||||||
|
<div class="btn-group btn-split">
|
||||||
|
<button type="button" class="btn btn-sm btn-success-outline approve" id="{{requestId}}" custom-button="{{requestId}}"><i class="fa fa-plus"></i> Approve</button>
|
||||||
|
<button type="button" class="btn btn-success-outline dropdown-toggle" data-toggle="dropdown" aria-haspopup="true" aria-expanded="false">
|
||||||
|
<span class="caret"></span>
|
||||||
|
<span class="sr-only">Toggle Dropdown</span>
|
||||||
|
</button>
|
||||||
|
<ul class="dropdown-menu">
|
||||||
|
{{#each qualities}}
|
||||||
|
<li><a href="#" class="approve-with-quality" id="{{id}}">{{name}}</a></li>
|
||||||
|
{{/each}}
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
{{else}}
|
||||||
|
<button id="{{requestId}}" custom-button="{{requestId}}" style="text-align: right" class="btn btn-sm btn-success-outline approve" type="submit"><i class="fa fa-plus"></i> Approve</button>
|
||||||
|
{{/if_eq}}
|
||||||
</form>
|
</form>
|
||||||
{{/if_eq}}
|
{{/if_eq}}
|
||||||
<form method="POST" action="/requests/delete" id="delete{{requestId}}">
|
<form method="POST" action="/requests/delete" id="delete{{requestId}}">
|
||||||
|
|
|
@ -167,12 +167,12 @@
|
||||||
<div class="col-sm-5 ">
|
<div class="col-sm-5 ">
|
||||||
<div>
|
<div>
|
||||||
<a href="https://musicbrainz.org/release/{{id}}" target="_blank">
|
<a href="https://musicbrainz.org/release/{{id}}" target="_blank">
|
||||||
<h4>
|
<h4>
|
||||||
{{artist}} - {{title}}
|
{{artist}} - {{title}}
|
||||||
{{#if year}}
|
{{#if year}}
|
||||||
({{year}})
|
({{year}})
|
||||||
{{/if}}
|
{{/if}}
|
||||||
</h4>
|
</h4>
|
||||||
</a>
|
</a>
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue