2022-05-01 00:33:22 -05:00
|
|
|
using System;
|
2022-06-13 01:52:41 -05:00
|
|
|
using System.Collections.Generic;
|
2022-05-01 00:33:22 -05:00
|
|
|
using System.Net.Mime;
|
2022-11-06 21:20:52 -06:00
|
|
|
using ConfusedPolarBear.Plugin.IntroSkipper.Configuration;
|
2024-07-27 21:11:01 +00:00
|
|
|
using ConfusedPolarBear.Plugin.IntroSkipper.Data;
|
2024-10-16 14:47:20 +02:00
|
|
|
using Jellyfin.Data.Enums;
|
2024-05-13 23:50:51 +02:00
|
|
|
using MediaBrowser.Common.Api;
|
2022-07-29 03:34:55 -05:00
|
|
|
using MediaBrowser.Controller.Entities.TV;
|
2022-05-01 00:33:22 -05:00
|
|
|
using Microsoft.AspNetCore.Authorization;
|
|
|
|
using Microsoft.AspNetCore.Mvc;
|
|
|
|
|
|
|
|
namespace ConfusedPolarBear.Plugin.IntroSkipper.Controllers;
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Skip intro controller.
|
|
|
|
/// </summary>
|
|
|
|
[Authorize]
|
|
|
|
[ApiController]
|
|
|
|
[Produces(MediaTypeNames.Application.Json)]
|
|
|
|
public class SkipIntroController : ControllerBase
|
|
|
|
{
|
|
|
|
/// <summary>
|
2022-05-09 22:50:41 -05:00
|
|
|
/// Initializes a new instance of the <see cref="SkipIntroController"/> class.
|
2022-05-01 00:33:22 -05:00
|
|
|
/// </summary>
|
|
|
|
public SkipIntroController()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <summary>
|
2022-06-12 21:28:24 -05:00
|
|
|
/// Returns the timestamps of the introduction in a television episode. Responses are in API version 1 format.
|
2022-05-01 00:33:22 -05:00
|
|
|
/// </summary>
|
2022-05-01 01:24:57 -05:00
|
|
|
/// <param name="id">ID of the episode. Required.</param>
|
2022-11-25 00:40:02 -06:00
|
|
|
/// <param name="mode">Timestamps to return. Optional. Defaults to Introduction for backwards compatibility.</param>
|
2022-05-01 00:33:22 -05:00
|
|
|
/// <response code="200">Episode contains an intro.</response>
|
|
|
|
/// <response code="404">Failed to find an intro in the provided episode.</response>
|
2022-05-09 22:50:41 -05:00
|
|
|
/// <returns>Detected intro.</returns>
|
2022-05-01 00:33:22 -05:00
|
|
|
[HttpGet("Episode/{id}/IntroTimestamps")]
|
2022-06-12 21:28:24 -05:00
|
|
|
[HttpGet("Episode/{id}/IntroTimestamps/v1")]
|
2022-11-25 00:40:02 -06:00
|
|
|
public ActionResult<Intro> GetIntroTimestamps(
|
|
|
|
[FromRoute] Guid id,
|
2024-10-16 14:47:20 +02:00
|
|
|
[FromQuery] MediaSegmentType mode = MediaSegmentType.Intro)
|
2022-05-01 00:33:22 -05:00
|
|
|
{
|
2022-11-25 00:40:02 -06:00
|
|
|
var intro = GetIntro(id, mode);
|
2022-05-01 00:33:22 -05:00
|
|
|
|
2022-06-12 21:28:24 -05:00
|
|
|
if (intro is null || !intro.Valid)
|
2022-05-01 00:33:22 -05:00
|
|
|
{
|
|
|
|
return NotFound();
|
|
|
|
}
|
|
|
|
|
|
|
|
return intro;
|
|
|
|
}
|
2022-06-07 12:18:03 -05:00
|
|
|
|
2024-07-27 21:11:01 +00:00
|
|
|
/// <summary>
|
|
|
|
/// Updates the timestamps for the provided episode.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="id">Episode ID to update timestamps for.</param>
|
|
|
|
/// <param name="timestamps">New timestamps Introduction/Credits start and end times.</param>
|
|
|
|
/// <response code="204">New timestamps saved.</response>
|
2024-07-28 19:37:46 +02:00
|
|
|
/// <response code="404">Given ID is not an Episode.</response>
|
2024-07-27 21:11:01 +00:00
|
|
|
/// <returns>No content.</returns>
|
|
|
|
[Authorize(Policy = Policies.RequiresElevation)]
|
|
|
|
[HttpPost("Episode/{Id}/Timestamps")]
|
|
|
|
public ActionResult UpdateTimestamps([FromRoute] Guid id, [FromBody] TimeStamps timestamps)
|
|
|
|
{
|
2024-07-28 19:37:46 +02:00
|
|
|
// only update existing episodes
|
|
|
|
var rawItem = Plugin.Instance!.GetItem(id);
|
|
|
|
if (rawItem == null || rawItem is not Episode episode)
|
|
|
|
{
|
|
|
|
return NotFound();
|
|
|
|
}
|
|
|
|
|
2024-09-12 08:37:47 +00:00
|
|
|
if (timestamps?.Introduction.End > 0.0)
|
2024-07-27 21:11:01 +00:00
|
|
|
{
|
2024-09-12 08:37:47 +00:00
|
|
|
var tr = new TimeRange(timestamps.Introduction.Start, timestamps.Introduction.End);
|
|
|
|
Plugin.Instance!.Intros[id] = new Segment(id, tr);
|
2024-07-27 21:11:01 +00:00
|
|
|
}
|
|
|
|
|
2024-09-12 08:37:47 +00:00
|
|
|
if (timestamps?.Credits.End > 0.0)
|
2024-07-27 21:11:01 +00:00
|
|
|
{
|
2024-09-12 08:37:47 +00:00
|
|
|
var cr = new TimeRange(timestamps.Credits.Start, timestamps.Credits.End);
|
|
|
|
Plugin.Instance!.Credits[id] = new Segment(id, cr);
|
2024-07-27 21:11:01 +00:00
|
|
|
}
|
|
|
|
|
2024-10-16 14:47:20 +02:00
|
|
|
Plugin.Instance!.SaveTimestamps(MediaSegmentType.Intro);
|
|
|
|
Plugin.Instance!.SaveTimestamps(MediaSegmentType.Outro);
|
2024-07-27 21:11:01 +00:00
|
|
|
|
|
|
|
return NoContent();
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Gets the timestamps for the provided episode.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="id">Episode ID.</param>
|
2024-07-28 19:37:46 +02:00
|
|
|
/// <response code="200">Sucess.</response>
|
|
|
|
/// <response code="404">Given ID is not an Episode.</response>
|
2024-07-27 21:11:01 +00:00
|
|
|
/// <returns>Episode Timestamps.</returns>
|
|
|
|
[HttpGet("Episode/{Id}/Timestamps")]
|
|
|
|
[ActionName("UpdateTimestamps")]
|
|
|
|
public ActionResult<TimeStamps> GetTimestamps([FromRoute] Guid id)
|
|
|
|
{
|
2024-07-28 18:53:23 +02:00
|
|
|
// only get return content for episodes
|
|
|
|
var rawItem = Plugin.Instance!.GetItem(id);
|
|
|
|
if (rawItem == null || rawItem is not Episode episode)
|
|
|
|
{
|
|
|
|
return NotFound();
|
|
|
|
}
|
|
|
|
|
2024-07-27 21:11:01 +00:00
|
|
|
var times = new TimeStamps();
|
|
|
|
if (Plugin.Instance!.Intros.TryGetValue(id, out var introValue))
|
|
|
|
{
|
|
|
|
times.Introduction = introValue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Plugin.Instance!.Credits.TryGetValue(id, out var creditValue))
|
|
|
|
{
|
|
|
|
times.Credits = creditValue;
|
|
|
|
}
|
|
|
|
|
|
|
|
return times;
|
|
|
|
}
|
|
|
|
|
2023-03-04 00:15:26 -06:00
|
|
|
/// <summary>
|
|
|
|
/// Gets a dictionary of all skippable segments.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="id">Media ID.</param>
|
|
|
|
/// <response code="200">Skippable segments dictionary.</response>
|
|
|
|
/// <returns>Dictionary of skippable segments.</returns>
|
|
|
|
[HttpGet("Episode/{id}/IntroSkipperSegments")]
|
2024-10-16 14:47:20 +02:00
|
|
|
public ActionResult<Dictionary<MediaSegmentType, Intro>> GetSkippableSegments([FromRoute] Guid id)
|
2023-03-04 00:15:26 -06:00
|
|
|
{
|
2024-10-16 14:47:20 +02:00
|
|
|
var segments = new Dictionary<MediaSegmentType, Intro>();
|
2023-03-04 00:15:26 -06:00
|
|
|
|
2024-10-16 14:47:20 +02:00
|
|
|
if (GetIntro(id, MediaSegmentType.Intro) is Intro intro)
|
2023-03-04 00:15:26 -06:00
|
|
|
{
|
2024-10-16 14:47:20 +02:00
|
|
|
segments[MediaSegmentType.Intro] = intro;
|
2023-03-04 00:15:26 -06:00
|
|
|
}
|
|
|
|
|
2024-10-16 14:47:20 +02:00
|
|
|
if (GetIntro(id, MediaSegmentType.Outro) is Intro credits)
|
2023-03-04 00:15:26 -06:00
|
|
|
{
|
2024-10-16 14:47:20 +02:00
|
|
|
segments[MediaSegmentType.Outro] = credits;
|
2023-03-04 00:15:26 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
return segments;
|
|
|
|
}
|
|
|
|
|
2022-11-25 00:40:02 -06:00
|
|
|
/// <summary>Lookup and return the skippable timestamps for the provided item.</summary>
|
2022-06-12 21:28:24 -05:00
|
|
|
/// <param name="id">Unique identifier of this episode.</param>
|
2022-11-25 00:40:02 -06:00
|
|
|
/// <param name="mode">Mode.</param>
|
2022-06-12 21:28:24 -05:00
|
|
|
/// <returns>Intro object if the provided item has an intro, null otherwise.</returns>
|
2024-10-16 14:47:20 +02:00
|
|
|
private static Intro? GetIntro(Guid id, MediaSegmentType mode)
|
2022-06-12 21:28:24 -05:00
|
|
|
{
|
2022-11-25 00:40:02 -06:00
|
|
|
try
|
|
|
|
{
|
2024-09-10 18:08:42 +02:00
|
|
|
var timestamp = Plugin.GetIntroByMode(id, mode);
|
2022-11-25 00:40:02 -06:00
|
|
|
|
2023-03-04 00:15:26 -06:00
|
|
|
// Operate on a copy to avoid mutating the original Intro object stored in the dictionary.
|
|
|
|
var segment = new Intro(timestamp);
|
|
|
|
|
2024-09-10 18:08:42 +02:00
|
|
|
var config = Plugin.Instance!.Configuration;
|
2024-08-02 13:41:03 +00:00
|
|
|
segment.IntroEnd -= config.RemainingSecondsOfIntro;
|
2024-03-01 09:19:12 -05:00
|
|
|
if (config.PersistSkipButton)
|
2024-03-01 18:29:37 +01:00
|
|
|
{
|
2024-04-20 12:29:40 +02:00
|
|
|
segment.ShowSkipPromptAt = segment.IntroStart;
|
2024-06-01 14:15:30 +02:00
|
|
|
segment.HideSkipPromptAt = segment.IntroEnd;
|
2024-03-01 18:29:37 +01:00
|
|
|
}
|
|
|
|
else
|
2024-03-01 09:19:12 -05:00
|
|
|
{
|
2024-04-20 12:29:40 +02:00
|
|
|
segment.ShowSkipPromptAt = Math.Max(0, segment.IntroStart - config.ShowPromptAdjustment);
|
|
|
|
segment.HideSkipPromptAt = Math.Min(
|
|
|
|
segment.IntroStart + config.HidePromptAdjustment,
|
2024-06-01 14:15:30 +02:00
|
|
|
segment.IntroEnd);
|
2024-03-01 09:19:12 -05:00
|
|
|
}
|
|
|
|
|
2023-03-04 00:15:26 -06:00
|
|
|
return segment;
|
2022-11-25 00:40:02 -06:00
|
|
|
}
|
|
|
|
catch (KeyNotFoundException)
|
|
|
|
{
|
|
|
|
return null;
|
|
|
|
}
|
2022-06-12 21:28:24 -05:00
|
|
|
}
|
|
|
|
|
2022-06-07 12:18:03 -05:00
|
|
|
/// <summary>
|
|
|
|
/// Erases all previously discovered introduction timestamps.
|
|
|
|
/// </summary>
|
2022-11-29 02:31:24 -06:00
|
|
|
/// <param name="mode">Mode.</param>
|
2024-06-07 17:09:48 +02:00
|
|
|
/// <param name="eraseCache">Erase cache.</param>
|
2022-06-07 12:18:03 -05:00
|
|
|
/// <response code="204">Operation successful.</response>
|
|
|
|
/// <returns>No content.</returns>
|
2024-05-13 23:50:51 +02:00
|
|
|
[Authorize(Policy = Policies.RequiresElevation)]
|
2022-06-07 12:18:03 -05:00
|
|
|
[HttpPost("Intros/EraseTimestamps")]
|
2024-10-16 14:47:20 +02:00
|
|
|
public ActionResult ResetIntroTimestamps([FromQuery] MediaSegmentType mode, [FromQuery] bool eraseCache = false)
|
2022-06-07 12:18:03 -05:00
|
|
|
{
|
2024-10-16 14:47:20 +02:00
|
|
|
if (mode == MediaSegmentType.Intro)
|
2022-11-29 02:31:24 -06:00
|
|
|
{
|
|
|
|
Plugin.Instance!.Intros.Clear();
|
|
|
|
}
|
2024-10-16 14:47:20 +02:00
|
|
|
else if (mode == MediaSegmentType.Outro)
|
2022-11-29 02:31:24 -06:00
|
|
|
{
|
|
|
|
Plugin.Instance!.Credits.Clear();
|
|
|
|
}
|
|
|
|
|
2024-06-07 17:09:48 +02:00
|
|
|
if (eraseCache)
|
|
|
|
{
|
|
|
|
FFmpegWrapper.DeleteCacheFiles(mode);
|
|
|
|
}
|
|
|
|
|
2024-06-15 13:16:47 +02:00
|
|
|
Plugin.Instance!.EpisodeStates.Clear();
|
2024-05-08 16:27:16 +02:00
|
|
|
Plugin.Instance!.SaveTimestamps(mode);
|
2022-06-07 12:18:03 -05:00
|
|
|
return NoContent();
|
|
|
|
}
|
2022-06-13 01:52:41 -05:00
|
|
|
|
2022-11-06 21:20:52 -06:00
|
|
|
/// <summary>
|
|
|
|
/// Gets the user interface configuration.
|
|
|
|
/// </summary>
|
|
|
|
/// <response code="200">UserInterfaceConfiguration returned.</response>
|
|
|
|
/// <returns>UserInterfaceConfiguration.</returns>
|
2024-03-10 19:47:38 +01:00
|
|
|
[HttpGet]
|
2022-11-06 21:20:52 -06:00
|
|
|
[Route("Intros/UserInterfaceConfiguration")]
|
|
|
|
public ActionResult<UserInterfaceConfiguration> GetUserInterfaceConfiguration()
|
|
|
|
{
|
|
|
|
var config = Plugin.Instance!.Configuration;
|
2023-03-04 00:15:26 -06:00
|
|
|
return new UserInterfaceConfiguration(
|
|
|
|
config.SkipButtonVisible,
|
|
|
|
config.SkipButtonIntroText,
|
2024-10-11 10:11:22 -04:00
|
|
|
config.SkipButtonEndCreditsText,
|
|
|
|
config.AutoSkip,
|
|
|
|
config.AutoSkipCredits,
|
|
|
|
config.ClientList);
|
2022-11-06 21:20:52 -06:00
|
|
|
}
|
2022-05-01 00:33:22 -05:00
|
|
|
}
|