2022-06-07 18:33:59 -05:00
using System ;
using System.Collections.Generic ;
using System.Threading ;
using System.Threading.Tasks ;
using System.Timers ;
2024-04-12 14:09:19 +02:00
using ConfusedPolarBear.Plugin.IntroSkipper.Configuration ;
2024-08-31 18:56:48 +02:00
using ConfusedPolarBear.Plugin.IntroSkipper.Data ;
2022-06-09 15:38:30 -05:00
using MediaBrowser.Common.Extensions ;
2022-06-07 18:33:59 -05:00
using MediaBrowser.Controller.Library ;
using MediaBrowser.Controller.Session ;
using MediaBrowser.Model.Entities ;
2024-04-12 14:09:19 +02:00
using MediaBrowser.Model.Plugins ;
2022-06-07 18:33:59 -05:00
using MediaBrowser.Model.Session ;
2024-03-22 23:41:58 +01:00
using Microsoft.Extensions.Hosting ;
2022-06-07 18:33:59 -05:00
using Microsoft.Extensions.Logging ;
2024-04-20 12:58:29 +02:00
using Timer = System . Timers . Timer ;
2022-06-07 18:33:59 -05:00
namespace ConfusedPolarBear.Plugin.IntroSkipper ;
/// <summary>
/// Automatically skip past introduction sequences.
/// Commands clients to seek to the end of the intro as soon as they start playing it.
/// </summary>
2024-03-22 23:41:58 +01:00
public class AutoSkip : IHostedService , IDisposable
2022-06-07 18:33:59 -05:00
{
private readonly object _sentSeekCommandLock = new ( ) ;
private ILogger < AutoSkip > _logger ;
private IUserDataManager _userDataManager ;
private ISessionManager _sessionManager ;
2024-04-20 12:58:29 +02:00
private Timer _playbackTimer = new ( 1000 ) ;
2022-06-07 18:33:59 -05:00
private Dictionary < string , bool > _sentSeekCommand ;
/// <summary>
/// Initializes a new instance of the <see cref="AutoSkip"/> class.
/// </summary>
/// <param name="userDataManager">User data manager.</param>
/// <param name="sessionManager">Session manager.</param>
/// <param name="logger">Logger.</param>
public AutoSkip (
IUserDataManager userDataManager ,
ISessionManager sessionManager ,
ILogger < AutoSkip > logger )
{
_userDataManager = userDataManager ;
_sessionManager = sessionManager ;
_logger = logger ;
_sentSeekCommand = new Dictionary < string , bool > ( ) ;
}
2024-04-12 14:09:19 +02:00
private void AutoSkipChanged ( object? sender , BasePluginConfiguration e )
2022-06-14 14:36:05 -05:00
{
2024-04-12 14:09:19 +02:00
var configuration = ( PluginConfiguration ) e ;
var newState = configuration . AutoSkip ;
2022-06-14 14:36:05 -05:00
_logger . LogDebug ( "Setting playback timer enabled to {NewState}" , newState ) ;
_playbackTimer . Enabled = newState ;
}
2022-06-07 18:33:59 -05:00
private void UserDataManager_UserDataSaved ( object? sender , UserDataSaveEventArgs e )
{
var itemId = e . Item . Id ;
2022-09-02 01:54:49 -05:00
var newState = false ;
var episodeNumber = e . Item . IndexNumber . GetValueOrDefault ( - 1 ) ;
2022-06-07 18:33:59 -05:00
// Ignore all events except playback start & end
if ( e . SaveReason ! = UserDataSaveReason . PlaybackStart & & e . SaveReason ! = UserDataSaveReason . PlaybackFinished )
{
return ;
}
// Lookup the session for this item.
SessionInfo ? session = null ;
try
{
foreach ( var needle in _sessionManager . Sessions )
{
if ( needle . UserId = = e . UserId & & needle . NowPlayingItem . Id = = itemId )
{
session = needle ;
break ;
}
}
if ( session = = null )
{
_logger . LogInformation ( "Unable to find session for {Item}" , itemId ) ;
return ;
}
}
2022-06-09 15:38:30 -05:00
catch ( Exception ex ) when ( ex is NullReferenceException | | ex is ResourceNotFoundException )
2022-06-07 18:33:59 -05:00
{
return ;
}
2022-09-02 01:54:49 -05:00
// If this is the first episode in the season, and SkipFirstEpisode is false, pretend that we've already sent the seek command for this playback session.
2024-06-12 16:29:07 +02:00
if ( Plugin . Instance ! . Configuration . SkipFirstEpisode & & episodeNumber = = 1 )
2022-09-02 01:54:49 -05:00
{
newState = true ;
}
2022-06-07 18:33:59 -05:00
// Reset the seek command state for this device.
lock ( _sentSeekCommandLock )
{
var device = session . DeviceId ;
_logger . LogDebug ( "Resetting seek command state for session {Session}" , device ) ;
2022-09-02 01:54:49 -05:00
_sentSeekCommand [ device ] = newState ;
2022-06-07 18:33:59 -05:00
}
}
private void PlaybackTimer_Elapsed ( object? sender , ElapsedEventArgs e )
{
foreach ( var session in _sessionManager . Sessions )
{
2024-08-31 16:48:31 +00:00
if ( WarningManager . HasFlag ( PluginWarning . UnableToAddSkipButton ) )
{
2024-09-01 16:56:33 +02:00
_logger . LogTrace ( "using autoskip to skip the intro because the injection of the skip button failed" ) ;
2024-08-31 16:48:31 +00:00
}
// only need for official Android TV App and jellyfin-kodi
2024-09-02 13:10:21 +01:00
else if ( session . Client ! = "Android TV" & & session . Client ! = "Kodi" )
2024-08-31 16:48:31 +00:00
{
continue ;
}
2022-06-07 18:33:59 -05:00
var deviceId = session . DeviceId ;
var itemId = session . NowPlayingItem . Id ;
var position = session . PlayState . PositionTicks / TimeSpan . TicksPerSecond ;
// Don't send the seek command more than once in the same session.
lock ( _sentSeekCommandLock )
{
2022-07-03 02:47:48 -05:00
if ( _sentSeekCommand . TryGetValue ( deviceId , out var sent ) & & sent )
2022-06-07 18:33:59 -05:00
{
_logger . LogTrace ( "Already sent seek command for session {Session}" , deviceId ) ;
continue ;
}
}
// Assert that an intro was detected for this item.
if ( ! Plugin . Instance ! . Intros . TryGetValue ( itemId , out var intro ) | | ! intro . Valid )
{
continue ;
}
// Seek is unreliable if called at the very start of an episode.
2024-08-02 13:41:03 +00:00
var adjustedStart = Math . Max ( 5 , intro . IntroStart + Plugin . Instance . Configuration . SecondsOfIntroStartToPlay ) ;
var adjustedEnd = intro . IntroEnd - Plugin . Instance . Configuration . RemainingSecondsOfIntro ;
2022-06-07 18:33:59 -05:00
_logger . LogTrace (
"Playback position is {Position}, intro runs from {Start} to {End}" ,
position ,
adjustedStart ,
2024-08-02 13:41:03 +00:00
adjustedEnd ) ;
2022-06-07 18:33:59 -05:00
2024-08-02 13:41:03 +00:00
if ( position < adjustedStart | | position > adjustedEnd )
2022-06-07 18:33:59 -05:00
{
continue ;
}
2022-06-08 16:15:59 -05:00
// Notify the user that an introduction is being skipped for them.
2024-04-20 12:21:07 +02:00
var notificationText = Plugin . Instance . Configuration . AutoSkipNotificationText ;
2022-11-06 21:20:52 -06:00
if ( ! string . IsNullOrWhiteSpace ( notificationText ) )
{
_sessionManager . SendMessageCommand (
2022-06-08 16:15:59 -05:00
session . Id ,
session . Id ,
2024-04-20 12:58:29 +02:00
new MessageCommand
2022-06-08 16:15:59 -05:00
{
2022-06-18 14:57:46 -05:00
Header = string . Empty , // some clients require header to be a string instead of null
2022-11-06 21:20:52 -06:00
Text = notificationText ,
2022-06-08 16:15:59 -05:00
TimeoutMs = 2000 ,
} ,
CancellationToken . None ) ;
2022-11-06 21:20:52 -06:00
}
2022-06-08 16:15:59 -05:00
2022-06-07 18:33:59 -05:00
_logger . LogDebug ( "Sending seek command to {Session}" , deviceId ) ;
2022-06-08 16:15:59 -05:00
_sessionManager . SendPlaystateCommand (
session . Id ,
session . Id ,
new PlaystateRequest
{
Command = PlaystateCommand . Seek ,
2024-05-13 15:28:15 +02:00
ControllingUserId = session . UserId . ToString ( ) ,
2024-08-02 13:41:03 +00:00
SeekPositionTicks = ( long ) adjustedEnd * TimeSpan . TicksPerSecond ,
2022-06-08 16:15:59 -05:00
} ,
CancellationToken . None ) ;
2022-06-07 18:33:59 -05:00
// Flag that we've sent the seek command so that it's not sent repeatedly
lock ( _sentSeekCommandLock )
{
_logger . LogTrace ( "Setting seek command state for session {Session}" , deviceId ) ;
_sentSeekCommand [ deviceId ] = true ;
}
}
}
/// <summary>
/// Dispose.
/// </summary>
public void Dispose ( )
{
Dispose ( true ) ;
GC . SuppressFinalize ( this ) ;
}
/// <summary>
/// Protected dispose.
/// </summary>
/// <param name="disposing">Dispose.</param>
protected virtual void Dispose ( bool disposing )
{
if ( ! disposing )
{
return ;
}
_playbackTimer . Stop ( ) ;
_playbackTimer . Dispose ( ) ;
}
2024-03-22 23:41:58 +01:00
/// <inheritdoc />
public Task StartAsync ( CancellationToken cancellationToken )
{
_logger . LogDebug ( "Setting up automatic skipping" ) ;
_userDataManager . UserDataSaved + = UserDataManager_UserDataSaved ;
2024-04-12 14:09:19 +02:00
Plugin . Instance ! . ConfigurationChanged + = AutoSkipChanged ;
2024-03-22 23:41:58 +01:00
// Make the timer restart automatically and set enabled to match the configuration value.
_playbackTimer . AutoReset = true ;
_playbackTimer . Elapsed + = PlaybackTimer_Elapsed ;
2024-04-20 12:21:07 +02:00
AutoSkipChanged ( null , Plugin . Instance . Configuration ) ;
2024-03-22 23:41:58 +01:00
return Task . CompletedTask ;
}
/// <inheritdoc />
public Task StopAsync ( CancellationToken cancellationToken )
{
_userDataManager . UserDataSaved - = UserDataManager_UserDataSaved ;
return Task . CompletedTask ;
}
2022-06-07 18:33:59 -05:00
}