mirror of
https://github.com/Ombi-app/Ombi.git
synced 2025-07-11 15:56:05 -07:00
frontend and tv episodes api work for #254
This commit is contained in:
parent
d7c40164cb
commit
33ba1db20b
12 changed files with 433 additions and 210 deletions
|
@ -84,6 +84,7 @@
|
|||
<Compile Include="Sonarr\SonarrProfile.cs" />
|
||||
<Compile Include="Sonarr\SystemStatus.cs" />
|
||||
<Compile Include="Tv\Authentication.cs" />
|
||||
<Compile Include="Tv\TvMazeEpisodes.cs" />
|
||||
<Compile Include="Tv\TvMazeSearch.cs" />
|
||||
<Compile Include="Tv\TvMazeSeasons.cs" />
|
||||
<Compile Include="Tv\TVMazeShow.cs" />
|
||||
|
|
44
PlexRequests.Api.Models/Tv/TvMazeEpisodes.cs
Normal file
44
PlexRequests.Api.Models/Tv/TvMazeEpisodes.cs
Normal file
|
@ -0,0 +1,44 @@
|
|||
#region Copyright
|
||||
// /************************************************************************
|
||||
// Copyright (c) 2016 Jamie Rees
|
||||
// File: TvMazeEpisodes.cs
|
||||
// Created By: Jamie Rees
|
||||
//
|
||||
// Permission is hereby granted, free of charge, to any person obtaining
|
||||
// a copy of this software and associated documentation files (the
|
||||
// "Software"), to deal in the Software without restriction, including
|
||||
// without limitation the rights to use, copy, modify, merge, publish,
|
||||
// distribute, sublicense, and/or sell copies of the Software, and to
|
||||
// permit persons to whom the Software is furnished to do so, subject to
|
||||
// the following conditions:
|
||||
//
|
||||
// The above copyright notice and this permission notice shall be
|
||||
// included in all copies or substantial portions of the Software.
|
||||
//
|
||||
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||||
// EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
||||
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
||||
// NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
||||
// LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
||||
// OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
||||
// WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
||||
// ************************************************************************/
|
||||
#endregion
|
||||
namespace PlexRequests.Api.Models.Tv
|
||||
{
|
||||
public class TvMazeEpisodes
|
||||
{
|
||||
public int id { get; set; }
|
||||
public string url { get; set; }
|
||||
public string name { get; set; }
|
||||
public int season { get; set; }
|
||||
public int number { get; set; }
|
||||
public string airdate { get; set; }
|
||||
public string airtime { get; set; }
|
||||
public string airstamp { get; set; }
|
||||
public int runtime { get; set; }
|
||||
public Image image { get; set; }
|
||||
public string summary { get; set; }
|
||||
public Links _links { get; set; }
|
||||
}
|
||||
}
|
|
@ -1,15 +1,37 @@
|
|||
using System;
|
||||
#region Copyright
|
||||
// /************************************************************************
|
||||
// Copyright (c) 2016 Jamie Rees
|
||||
// File: TvMazeSearch.cs
|
||||
// Created By: Jamie Rees
|
||||
//
|
||||
// Permission is hereby granted, free of charge, to any person obtaining
|
||||
// a copy of this software and associated documentation files (the
|
||||
// "Software"), to deal in the Software without restriction, including
|
||||
// without limitation the rights to use, copy, modify, merge, publish,
|
||||
// distribute, sublicense, and/or sell copies of the Software, and to
|
||||
// permit persons to whom the Software is furnished to do so, subject to
|
||||
// the following conditions:
|
||||
//
|
||||
// The above copyright notice and this permission notice shall be
|
||||
// included in all copies or substantial portions of the Software.
|
||||
//
|
||||
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||||
// EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
||||
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
||||
// NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
||||
// LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
||||
// OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
||||
// WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
||||
// ************************************************************************/
|
||||
#endregion
|
||||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
using System.Text;
|
||||
using System.Threading.Tasks;
|
||||
|
||||
namespace PlexRequests.Api.Models.Tv
|
||||
{
|
||||
public class Schedule
|
||||
{
|
||||
public string time { get; set; }
|
||||
public List<object> days { get; set; }
|
||||
public string time { get; set; }
|
||||
}
|
||||
|
||||
public class Rating
|
||||
|
@ -19,23 +41,23 @@ namespace PlexRequests.Api.Models.Tv
|
|||
|
||||
public class Country
|
||||
{
|
||||
public string name { get; set; }
|
||||
public string code { get; set; }
|
||||
public string name { get; set; }
|
||||
public string timezone { get; set; }
|
||||
}
|
||||
|
||||
public class Network
|
||||
{
|
||||
public Country country { get; set; }
|
||||
public int id { get; set; }
|
||||
public string name { get; set; }
|
||||
public Country country { get; set; }
|
||||
}
|
||||
|
||||
public class Externals
|
||||
{
|
||||
public int? tvrage { get; set; }
|
||||
public int? thetvdb { get; set; }
|
||||
public string imdb { get; set; }
|
||||
public int? thetvdb { get; set; }
|
||||
public int? tvrage { get; set; }
|
||||
}
|
||||
|
||||
public class Image
|
||||
|
@ -61,32 +83,32 @@ namespace PlexRequests.Api.Models.Tv
|
|||
|
||||
public class Links
|
||||
{
|
||||
public Self self { get; set; }
|
||||
public Previousepisode previousepisode { get; set; }
|
||||
public Nextepisode nextepisode { get; set; }
|
||||
public Previousepisode previousepisode { get; set; }
|
||||
public Self self { get; set; }
|
||||
}
|
||||
|
||||
public class Show
|
||||
{
|
||||
public int id { get; set; }
|
||||
public string url { get; set; }
|
||||
public string name { get; set; }
|
||||
public string type { get; set; }
|
||||
public string language { get; set; }
|
||||
public List<object> genres { get; set; }
|
||||
public string status { get; set; }
|
||||
public int? runtime { get; set; }
|
||||
public string premiered { get; set; }
|
||||
public Schedule schedule { get; set; }
|
||||
public Rating rating { get; set; }
|
||||
public int weight { get; set; }
|
||||
public Network network { get; set; }
|
||||
public object webChannel { get; set; }
|
||||
public Externals externals { get; set; }
|
||||
public Image image { get; set; }
|
||||
public string summary { get; set; }
|
||||
public int updated { get; set; }
|
||||
public Links _links { get; set; }
|
||||
public Externals externals { get; set; }
|
||||
public List<object> genres { get; set; }
|
||||
public int id { get; set; }
|
||||
public Image image { get; set; }
|
||||
public string language { get; set; }
|
||||
public string name { get; set; }
|
||||
public Network network { get; set; }
|
||||
public string premiered { get; set; }
|
||||
public Rating rating { get; set; }
|
||||
public int? runtime { get; set; }
|
||||
public Schedule schedule { get; set; }
|
||||
public string status { get; set; }
|
||||
public string summary { get; set; }
|
||||
public string type { get; set; }
|
||||
public int updated { get; set; }
|
||||
public string url { get; set; }
|
||||
public object webChannel { get; set; }
|
||||
public int weight { get; set; }
|
||||
}
|
||||
|
||||
public class TvMazeSearch
|
||||
|
|
|
@ -69,6 +69,19 @@ namespace PlexRequests.Api
|
|||
return Api.Execute<TvMazeShow>(request, new Uri(Uri));
|
||||
}
|
||||
|
||||
public IEnumerable<TvMazeEpisodes> EpisodeLookup(int showId)
|
||||
{
|
||||
var request = new RestRequest
|
||||
{
|
||||
Method = Method.GET,
|
||||
Resource = "shows/{id}/episodes"
|
||||
};
|
||||
request.AddUrlSegment("id", showId.ToString());
|
||||
request.AddHeader("Content-Type", "application/json");
|
||||
|
||||
return Api.Execute<List<TvMazeEpisodes>>(request, new Uri(Uri));
|
||||
}
|
||||
|
||||
public TvMazeShow ShowLookupByTheTvDbId(int theTvDbId)
|
||||
{
|
||||
var request = new RestRequest
|
||||
|
|
4
PlexRequests.UI/Content/base.css
vendored
4
PlexRequests.UI/Content/base.css
vendored
|
@ -329,3 +329,7 @@ label {
|
|||
.landing-title {
|
||||
font-weight: bold; }
|
||||
|
||||
.checkbox-custom {
|
||||
margin-top: 0 !important;
|
||||
margin-bottom: 0 !important; }
|
||||
|
||||
|
|
2
PlexRequests.UI/Content/base.min.css
vendored
2
PlexRequests.UI/Content/base.min.css
vendored
File diff suppressed because one or more lines are too long
|
@ -384,7 +384,7 @@ $border-radius: 10px;
|
|||
|
||||
.bootstrap-datetimepicker-widget table td.active,
|
||||
.bootstrap-datetimepicker-widget table td.active:hover {
|
||||
color: #fff !important;
|
||||
color: #fff $i;
|
||||
}
|
||||
|
||||
.landing-header {
|
||||
|
@ -393,7 +393,7 @@ $border-radius: 10px;
|
|||
}
|
||||
|
||||
.landing-block {
|
||||
background: #2f2f2f !important;
|
||||
background: #2f2f2f $i;
|
||||
padding: 5px;
|
||||
}
|
||||
|
||||
|
@ -415,3 +415,8 @@ $border-radius: 10px;
|
|||
.landing-title {
|
||||
font-weight: bold;
|
||||
}
|
||||
|
||||
.checkbox-custom{
|
||||
margin-top:0 $i;
|
||||
margin-bottom:0 $i;
|
||||
}
|
|
@ -12,9 +12,14 @@ $(function () {
|
|||
var searchSource = $("#search-template").html();
|
||||
var seasonsSource = $("#seasons-template").html();
|
||||
var musicSource = $("#music-template").html();
|
||||
var seasonsNumberSource = $("#seasonNumber-template").html();
|
||||
var episodeSource = $("#episode-template").html();
|
||||
|
||||
var searchTemplate = Handlebars.compile(searchSource);
|
||||
var musicTemplate = Handlebars.compile(musicSource);
|
||||
var seasonsTemplate = Handlebars.compile(seasonsSource);
|
||||
var seasonsNumberTemplate = Handlebars.compile(seasonsNumberSource);
|
||||
var episodesTemplate = Handlebars.compile(episodeSource);
|
||||
|
||||
var base = $('#baseUrl').text();
|
||||
|
||||
|
@ -256,6 +261,7 @@ $(function () {
|
|||
$('#typeModal').val(type);
|
||||
});
|
||||
|
||||
|
||||
function focusSearch($content) {
|
||||
if ($content.length > 0) {
|
||||
$('input[type=text].form-control', $content).first().focus();
|
||||
|
@ -531,4 +537,71 @@ $(function () {
|
|||
|
||||
});
|
||||
|
||||
$('#episodesModal').on('show.bs.modal', function (event) {
|
||||
var button = $(event.relatedTarget); // Button that triggered the modal
|
||||
var id = button.data('identifier'); // Extract info from data-* attributes
|
||||
var url = createBaseUrl(base, '/search/episodes/');
|
||||
var seenSeasons = [];
|
||||
$.ajax({
|
||||
type: "get",
|
||||
url: url,
|
||||
data: { tvId: id },
|
||||
dataType: "json",
|
||||
success: function (results) {
|
||||
var $content = $("#episodesBody");
|
||||
$content.html("");
|
||||
$('#selectedEpisodeId').val(id);
|
||||
results.forEach(function (result) {
|
||||
|
||||
|
||||
var episodes = buildEpisodesView(result);
|
||||
|
||||
if (!seenSeasons.find(x => x === episodes.season)) {
|
||||
// Create the seasons heading
|
||||
seenSeasons.push(episodes.season);
|
||||
var context = buildSeasonsCount(result);
|
||||
$content.append(seasonsNumberTemplate(context));
|
||||
}
|
||||
var episodesResult = episodesTemplate(episodes);
|
||||
$content.append(episodesResult);
|
||||
});
|
||||
},
|
||||
error: function (e) {
|
||||
console.log(e);
|
||||
generateNotify("Something went wrong!", "danger");
|
||||
}
|
||||
});
|
||||
|
||||
function buildSeasonsContext(result) {
|
||||
var context = {
|
||||
id: result
|
||||
};
|
||||
return context;
|
||||
};
|
||||
|
||||
function buildSeasonsCount(result) {
|
||||
return {
|
||||
seasonNumber: result.season
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
function buildEpisodesView(result) {
|
||||
return {
|
||||
id: result.id,
|
||||
url: result.url,
|
||||
name: result.name,
|
||||
season: result.season,
|
||||
number: result.number,
|
||||
airdate: result.airdate,
|
||||
airtime: result.airtime,
|
||||
airstamp: result.airstamp,
|
||||
runtime: result.runtime,
|
||||
image: result.image,
|
||||
summary: result.summary,
|
||||
links : result._links
|
||||
}
|
||||
}
|
||||
});
|
||||
|
||||
});
|
||||
|
|
|
@ -100,6 +100,7 @@ namespace PlexRequests.UI.Modules
|
|||
IssueService = issue;
|
||||
Analytics = a;
|
||||
RequestLimitRepo = rl;
|
||||
TvApi = new TvMazeApi();
|
||||
|
||||
|
||||
Get["SearchIndex","/", true] = async (x, ct) => await RequestLoad();
|
||||
|
@ -120,7 +121,9 @@ namespace PlexRequests.UI.Modules
|
|||
Get["/notifyuser", true] = async (x, ct) => await GetUserNotificationSettings();
|
||||
|
||||
Get["/seasons"] = x => GetSeasons();
|
||||
Get["/episodes"] = x => GetEpisodes();
|
||||
}
|
||||
private TvMazeApi TvApi { get; }
|
||||
private IPlexApi PlexApi { get; }
|
||||
private TheMovieDbApi MovieApi { get; }
|
||||
private INotificationService NotificationService { get; }
|
||||
|
@ -854,14 +857,21 @@ namespace PlexRequests.UI.Modules
|
|||
|
||||
private Response GetSeasons()
|
||||
{
|
||||
var tv = new TvMazeApi();
|
||||
var seriesId = (int)Request.Query.tvId;
|
||||
var show = tv.ShowLookupByTheTvDbId(seriesId);
|
||||
var seasons = tv.GetSeasons(show.id);
|
||||
var show = TvApi.ShowLookupByTheTvDbId(seriesId);
|
||||
var seasons = TvApi.GetSeasons(show.id);
|
||||
var model = seasons.Select(x => x.number);
|
||||
return Response.AsJson(model);
|
||||
}
|
||||
|
||||
private Response GetEpisodes()
|
||||
{
|
||||
var seriesId = (int)Request.Query.tvId;
|
||||
var show = TvApi.ShowLookupByTheTvDbId(seriesId);
|
||||
var seasons = TvApi.EpisodeLookup(show.id);
|
||||
return Response.AsJson(seasons);
|
||||
}
|
||||
|
||||
private async Task<bool> CheckRequestLimit(PlexRequestSettings s, RequestType type)
|
||||
{
|
||||
if (IsAdmin)
|
||||
|
|
9
PlexRequests.UI/Resources/UI.Designer.cs
generated
9
PlexRequests.UI/Resources/UI.Designer.cs
generated
|
@ -888,6 +888,15 @@ namespace PlexRequests.UI.Resources {
|
|||
}
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Looks up a localized string similar to Select Episode.
|
||||
/// </summary>
|
||||
public static string Search_SelectEpisode {
|
||||
get {
|
||||
return ResourceManager.GetString("Search_SelectEpisode", resourceCulture);
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Looks up a localized string similar to Select .
|
||||
/// </summary>
|
||||
|
|
|
@ -431,4 +431,7 @@
|
|||
<data name="Layout_French" xml:space="preserve">
|
||||
<value>French</value>
|
||||
</data>
|
||||
<data name="Search_SelectEpisode" xml:space="preserve">
|
||||
<value>Select Episode</value>
|
||||
</data>
|
||||
</root>
|
|
@ -192,6 +192,7 @@
|
|||
<li><a id="{{id}}" season-select="1" class="dropdownTv" href="#">@UI.Search_FirstSeason</a></li>
|
||||
<li><a id="{{id}}" season-select="2" class="dropdownTv" href="#">@UI.Search_LatestSeason</a></li>
|
||||
<li><a id="SeasonSelect" data-identifier="{{id}}" data-toggle="modal" data-target="#seasonsModal" href="#">@UI.Search_SelectSeason...</a></li>
|
||||
<li><a id="EpisodeSelect" data-identifier="{{id}}" data-toggle="modal" data-target="#episodesModal" href="#">@UI.Search_SelectEpisode...</a></li>
|
||||
</ul>
|
||||
</div>
|
||||
{{/if_eq}}
|
||||
|
@ -291,6 +292,26 @@
|
|||
</div>
|
||||
</div>
|
||||
|
||||
<div class="modal fade" id="episodesModal">
|
||||
<div class="modal-dialog modal-lg">
|
||||
<div class="modal-content">
|
||||
<div class="modal-header">
|
||||
<button type="button" class="close" data-dismiss="modal" aria-hidden="true">×</button>
|
||||
<h4 class="modal-title">@UI.Search_Modal_SeasonsTitle</h4>
|
||||
</div>
|
||||
<div class="modal-body" id="episodesBody">
|
||||
|
||||
</div>
|
||||
|
||||
<div hidden="hidden" id="selectedEpisodeId"></div>
|
||||
<div class="modal-footer">
|
||||
<button type="button" class="btn btn-default" data-dismiss="modal">@UI.Common_Close</button>
|
||||
<button type="button" id="episodesRequest" class="btn btn-primary">@UI.Search_Request</button>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
|
||||
<div class="modal fade" id="issuesModal">
|
||||
<div class="modal-dialog">
|
||||
|
@ -321,6 +342,24 @@
|
|||
<input type="checkbox" class="selectedSeasons" id="{{id}}" name="{{id}}"><label for="{{id}}">@UI.Search_Season {{id}}</label>
|
||||
</div>
|
||||
</div>
|
||||
</script>
|
||||
|
||||
<script id="seasonNumber-template" type="text/x-handlebars-template">
|
||||
<div class="row"></div>
|
||||
<div id="seasonNumber{{seasonNumber}}">
|
||||
<strong>@UI.Search_Season {{seasonNumber}}</strong>
|
||||
</div>
|
||||
|
||||
|
||||
</script>
|
||||
|
||||
|
||||
<script id="episode-template" type="text/x-handlebars-template">
|
||||
<div class="form-group col-md-6">
|
||||
<div class="checkbox" style="margin-bottom:0px; margin-top:0px;">
|
||||
<input type="checkbox" class="selectedEpisodes" id="{{id}}" name="{{id}}"><label for="{{id}}">{{number}}. {{name}}</label>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
</script>
|
||||
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue