6ccf002e51
* Recaps and Previews Support * Add draft UI of preview / recap edit * remove intro/credit tasks * Update configPage.html * rename task * Reorganize settings by relation * More standardized formatting * Some additional formatting * fix a typo * Update configPage.html * Allow missing recap / prview data * More risk to corrupt than benefit * Update TimeStamps.cs * Update PluginConfiguration.cs * Update configPage.html * Update PluginConfiguration.cs * Add chapter regex to settings * Move all UI into UI section * Move ending seconds with similar * Add default * fixes * Update SkipIntroController.cs * Autoskip all segments * Check if adjacent segment * Update AutoSkip.cs * Update AutoSkip.cs * Settings apply to all segment types * Update SegmentProvider * Update configPage.html Whoops * Update Plugin.cs * Update AutoSkip.cs * Let’s call it missing instead * Update BaseItemAnalyzerTask.cs * Update BaseItemAnalyzerTask.cs * Update BaseItemAnalyzerTask.cs * Move "select" all below list * Clarify button wording * Update configPage.html * Nope, long client list will hide it * Simplify wording * Update QueuedEpisode.cs * fix unit test for ffmpeg7 * Add migration * Restore DataContract * update * Update configPage.html * remove analyzed status * Update AutoSkip.cs * Update configPage.html typo * Store analyzed items in seasoninfo * Update VisualizationController.cs * update * Update IntroSkipperDbContext.cs * Add preview / recap delete * This keeps changing itself * Update SkipIntroController.cs * Rather add it to be removed --------- Co-authored-by: rlauu <46294892+rlauu@users.noreply.github.com> Co-authored-by: TwistedUmbrellaX <1173913+AbandonedCart@users.noreply.github.com> Co-authored-by: Kilian von Pflugk <github@jumoog.io>
275 lines
9.7 KiB
C#
275 lines
9.7 KiB
C#
// Copyright (C) 2024 Intro-Skipper contributors <intro-skipper.org>
|
|
// SPDX-License-Identifier: GPL-3.0-only.
|
|
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.Linq;
|
|
using System.Net.Mime;
|
|
using System.Threading;
|
|
using System.Threading.Tasks;
|
|
using IntroSkipper.Configuration;
|
|
using IntroSkipper.Data;
|
|
using IntroSkipper.Db;
|
|
using IntroSkipper.Manager;
|
|
using MediaBrowser.Common.Api;
|
|
using MediaBrowser.Controller.Entities.Movies;
|
|
using MediaBrowser.Controller.Entities.TV;
|
|
using Microsoft.AspNetCore.Authorization;
|
|
using Microsoft.AspNetCore.Mvc;
|
|
using Microsoft.EntityFrameworkCore;
|
|
|
|
namespace IntroSkipper.Controllers;
|
|
|
|
/// <summary>
|
|
/// Skip intro controller.
|
|
/// </summary>
|
|
[Authorize]
|
|
[ApiController]
|
|
[Produces(MediaTypeNames.Application.Json)]
|
|
public class SkipIntroController(MediaSegmentUpdateManager mediaSegmentUpdateManager) : ControllerBase
|
|
{
|
|
private readonly MediaSegmentUpdateManager _mediaSegmentUpdateManager = mediaSegmentUpdateManager;
|
|
|
|
/// <summary>
|
|
/// Returns the timestamps of the introduction in a television episode. Responses are in API version 1 format.
|
|
/// </summary>
|
|
/// <param name="id">ID of the episode. Required.</param>
|
|
/// <param name="mode">Timestamps to return. Optional. Defaults to Introduction for backwards compatibility.</param>
|
|
/// <response code="200">Episode contains an intro.</response>
|
|
/// <response code="404">Failed to find an intro in the provided episode.</response>
|
|
/// <returns>Detected intro.</returns>
|
|
[HttpGet("Episode/{id}/IntroTimestamps")]
|
|
[HttpGet("Episode/{id}/IntroTimestamps/v1")]
|
|
public ActionResult<Intro> GetIntroTimestamps(
|
|
[FromRoute] Guid id,
|
|
[FromQuery] AnalysisMode mode = AnalysisMode.Introduction)
|
|
{
|
|
var intros = GetIntros(id);
|
|
if (!intros.TryGetValue(mode, out var intro))
|
|
{
|
|
return NotFound();
|
|
}
|
|
|
|
return intro;
|
|
}
|
|
|
|
/// <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>
|
|
/// <param name="cancellationToken">Cancellation Token.</param>
|
|
/// <response code="204">New timestamps saved.</response>
|
|
/// <response code="404">Given ID is not an Episode.</response>
|
|
/// <returns>No content.</returns>
|
|
[Authorize(Policy = Policies.RequiresElevation)]
|
|
[HttpPost("Episode/{Id}/Timestamps")]
|
|
public async Task<ActionResult> UpdateTimestampsAsync([FromRoute] Guid id, [FromBody] TimeStamps timestamps, CancellationToken cancellationToken = default)
|
|
{
|
|
// only update existing episodes
|
|
var rawItem = Plugin.Instance!.GetItem(id);
|
|
if (rawItem is not Episode and not Movie)
|
|
{
|
|
return NotFound();
|
|
}
|
|
|
|
if (timestamps == null)
|
|
{
|
|
return NoContent();
|
|
}
|
|
|
|
var segmentTypes = new[]
|
|
{
|
|
(AnalysisMode.Introduction, timestamps.Introduction),
|
|
(AnalysisMode.Credits, timestamps.Credits),
|
|
(AnalysisMode.Recap, timestamps.Recap),
|
|
(AnalysisMode.Preview, timestamps.Preview)
|
|
};
|
|
|
|
foreach (var (mode, segment) in segmentTypes)
|
|
{
|
|
if (segment.Valid)
|
|
{
|
|
await Plugin.Instance!.UpdateTimestampAsync(segment, mode).ConfigureAwait(false);
|
|
}
|
|
}
|
|
|
|
if (Plugin.Instance.Configuration.UpdateMediaSegments)
|
|
{
|
|
var episode = Plugin.Instance!.QueuedMediaItems[rawItem is Episode e ? e.SeasonId : rawItem.Id]
|
|
.FirstOrDefault(q => q.EpisodeId == rawItem.Id);
|
|
|
|
if (episode is not null)
|
|
{
|
|
await _mediaSegmentUpdateManager.UpdateMediaSegmentsAsync([episode], cancellationToken).ConfigureAwait(false);
|
|
}
|
|
}
|
|
|
|
return NoContent();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the timestamps for the provided episode.
|
|
/// </summary>
|
|
/// <param name="id">Episode ID.</param>
|
|
/// <response code="200">Sucess.</response>
|
|
/// <response code="404">Given ID is not an Episode.</response>
|
|
/// <returns>Episode Timestamps.</returns>
|
|
[HttpGet("Episode/{Id}/Timestamps")]
|
|
[ActionName("UpdateTimestamps")]
|
|
public ActionResult<TimeStamps> GetTimestamps([FromRoute] Guid id)
|
|
{
|
|
// only get return content for episodes
|
|
var rawItem = Plugin.Instance!.GetItem(id);
|
|
if (rawItem is not Episode and not Movie)
|
|
{
|
|
return NotFound();
|
|
}
|
|
|
|
var times = new TimeStamps();
|
|
var segments = Plugin.Instance!.GetTimestamps(id);
|
|
|
|
if (segments.TryGetValue(AnalysisMode.Introduction, out var introSegment))
|
|
{
|
|
times.Introduction = introSegment;
|
|
}
|
|
|
|
if (segments.TryGetValue(AnalysisMode.Credits, out var creditSegment))
|
|
{
|
|
times.Credits = creditSegment;
|
|
}
|
|
|
|
if (segments.TryGetValue(AnalysisMode.Recap, out var recapSegment))
|
|
{
|
|
times.Recap = recapSegment;
|
|
}
|
|
|
|
if (segments.TryGetValue(AnalysisMode.Preview, out var previewSegment))
|
|
{
|
|
times.Preview = previewSegment;
|
|
}
|
|
|
|
return times;
|
|
}
|
|
|
|
/// <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")]
|
|
public ActionResult<Dictionary<AnalysisMode, Intro>> GetSkippableSegments([FromRoute] Guid id)
|
|
{
|
|
var segments = GetIntros(id);
|
|
var result = new Dictionary<AnalysisMode, Intro>();
|
|
|
|
if (segments.TryGetValue(AnalysisMode.Introduction, out var introSegment))
|
|
{
|
|
result[AnalysisMode.Introduction] = introSegment;
|
|
}
|
|
|
|
if (segments.TryGetValue(AnalysisMode.Credits, out var creditSegment))
|
|
{
|
|
result[AnalysisMode.Credits] = creditSegment;
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
/// <summary>Lookup and return the skippable timestamps for the provided item.</summary>
|
|
/// <param name="id">Unique identifier of this episode.</param>
|
|
/// <returns>Intro object if the provided item has an intro, null otherwise.</returns>
|
|
internal static Dictionary<AnalysisMode, Intro> GetIntros(Guid id)
|
|
{
|
|
var timestamps = Plugin.Instance!.GetTimestamps(id);
|
|
var intros = new Dictionary<AnalysisMode, Intro>();
|
|
var runTime = TimeSpan.FromTicks(Plugin.Instance!.GetItem(id)?.RunTimeTicks ?? 0).TotalSeconds;
|
|
var config = Plugin.Instance.Configuration;
|
|
|
|
foreach (var (mode, timestamp) in timestamps)
|
|
{
|
|
if (!timestamp.Valid)
|
|
{
|
|
continue;
|
|
}
|
|
|
|
// Create new Intro to avoid mutating the original stored in dictionary
|
|
var segment = new Intro(timestamp);
|
|
|
|
// Calculate intro end time
|
|
segment.IntroEnd = runTime > 0 && runTime < segment.IntroEnd + 1
|
|
? runTime
|
|
: segment.IntroEnd - config.RemainingSecondsOfIntro;
|
|
|
|
// Set skip button prompt visibility times
|
|
const double MIN_REMAINING_TIME = 3.0; // Minimum seconds before end to hide prompt
|
|
if (config.PersistSkipButton)
|
|
{
|
|
segment.ShowSkipPromptAt = segment.IntroStart;
|
|
segment.HideSkipPromptAt = segment.IntroEnd - MIN_REMAINING_TIME;
|
|
}
|
|
else
|
|
{
|
|
segment.ShowSkipPromptAt = Math.Max(0, segment.IntroStart - config.ShowPromptAdjustment);
|
|
segment.HideSkipPromptAt = Math.Min(
|
|
segment.IntroStart + config.HidePromptAdjustment,
|
|
segment.IntroEnd - MIN_REMAINING_TIME);
|
|
}
|
|
|
|
intros[mode] = segment;
|
|
}
|
|
|
|
return intros;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Erases all previously discovered introduction timestamps.
|
|
/// </summary>
|
|
/// <param name="mode">Mode.</param>
|
|
/// <param name="eraseCache">Erase cache.</param>
|
|
/// <response code="204">Operation successful.</response>
|
|
/// <returns>No content.</returns>
|
|
[Authorize(Policy = Policies.RequiresElevation)]
|
|
[HttpPost("Intros/EraseTimestamps")]
|
|
public async Task<ActionResult> ResetIntroTimestamps([FromQuery] AnalysisMode mode, [FromQuery] bool eraseCache = false)
|
|
{
|
|
using var db = new IntroSkipperDbContext(Plugin.Instance!.DbPath);
|
|
var segments = await db.DbSegment
|
|
.Where(s => s.Type == mode)
|
|
.ToListAsync()
|
|
.ConfigureAwait(false);
|
|
|
|
db.DbSegment.RemoveRange(segments);
|
|
await db.SaveChangesAsync().ConfigureAwait(false);
|
|
|
|
if (eraseCache && mode is AnalysisMode.Introduction or AnalysisMode.Credits)
|
|
{
|
|
await Task.Run(() => FFmpegWrapper.DeleteCacheFiles(mode)).ConfigureAwait(false);
|
|
}
|
|
|
|
return NoContent();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the user interface configuration.
|
|
/// </summary>
|
|
/// <response code="200">UserInterfaceConfiguration returned.</response>
|
|
/// <returns>UserInterfaceConfiguration.</returns>
|
|
[HttpGet]
|
|
[Route("Intros/UserInterfaceConfiguration")]
|
|
public ActionResult<UserInterfaceConfiguration> GetUserInterfaceConfiguration()
|
|
{
|
|
var config = Plugin.Instance!.Configuration;
|
|
return new UserInterfaceConfiguration(
|
|
config.SkipButtonEnabled,
|
|
config.SkipButtonIntroText,
|
|
config.SkipButtonEndCreditsText,
|
|
config.AutoSkip,
|
|
config.AutoSkipCredits,
|
|
config.AutoSkipRecap,
|
|
config.AutoSkipPreview,
|
|
config.ClientList);
|
|
}
|
|
}
|