AbstractGroupState.cs 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216
  1. using System.Threading;
  2. using MediaBrowser.Controller.Session;
  3. using MediaBrowser.Controller.SyncPlay.PlaybackRequests;
  4. using MediaBrowser.Model.SyncPlay;
  5. using Microsoft.Extensions.Logging;
  6. namespace MediaBrowser.Controller.SyncPlay.GroupStates
  7. {
  8. /// <summary>
  9. /// Class AbstractGroupState.
  10. /// </summary>
  11. /// <remarks>
  12. /// Class is not thread-safe, external locking is required when accessing methods.
  13. /// </remarks>
  14. public abstract class AbstractGroupState : IGroupState
  15. {
  16. /// <summary>
  17. /// Initializes a new instance of the <see cref="AbstractGroupState"/> class.
  18. /// </summary>
  19. /// <param name="logger">Instance of the <see cref="ILogger"/> interface.</param>
  20. protected AbstractGroupState(ILogger logger)
  21. {
  22. Logger = logger;
  23. }
  24. /// <inheritdoc />
  25. public abstract GroupStateType Type { get; }
  26. /// <summary>
  27. /// Gets the logger.
  28. /// </summary>
  29. protected ILogger Logger { get; }
  30. /// <inheritdoc />
  31. public abstract void SessionJoined(IGroupStateContext context, GroupStateType prevState, SessionInfo session, CancellationToken cancellationToken);
  32. /// <inheritdoc />
  33. public abstract void SessionLeaving(IGroupStateContext context, GroupStateType prevState, SessionInfo session, CancellationToken cancellationToken);
  34. /// <inheritdoc />
  35. public virtual void HandleRequest(IGroupStateContext context, GroupStateType prevState, IGroupPlaybackRequest request, SessionInfo session, CancellationToken cancellationToken)
  36. {
  37. UnhandledRequest(request);
  38. }
  39. /// <inheritdoc />
  40. public virtual void HandleRequest(IGroupStateContext context, GroupStateType prevState, PlayGroupRequest request, SessionInfo session, CancellationToken cancellationToken)
  41. {
  42. UnhandledRequest(request);
  43. }
  44. /// <inheritdoc />
  45. public virtual void HandleRequest(IGroupStateContext context, GroupStateType prevState, SetPlaylistItemGroupRequest request, SessionInfo session, CancellationToken cancellationToken)
  46. {
  47. var waitingState = new WaitingGroupState(Logger);
  48. context.SetState(waitingState);
  49. waitingState.HandleRequest(context, Type, request, session, cancellationToken);
  50. }
  51. /// <inheritdoc />
  52. public virtual void HandleRequest(IGroupStateContext context, GroupStateType prevState, RemoveFromPlaylistGroupRequest request, SessionInfo session, CancellationToken cancellationToken)
  53. {
  54. var playingItemRemoved = context.RemoveFromPlayQueue(request.PlaylistItemIds);
  55. var playQueueUpdate = context.GetPlayQueueUpdate(PlayQueueUpdateReason.RemoveItems);
  56. var update = context.NewSyncPlayGroupUpdate(GroupUpdateType.PlayQueue, playQueueUpdate);
  57. context.SendGroupUpdate(session, SyncPlayBroadcastType.AllGroup, update, cancellationToken);
  58. if (playingItemRemoved && !context.PlayQueue.IsItemPlaying())
  59. {
  60. Logger.LogDebug("HandleRequest: {RequestType} in group {GroupId}, play queue is empty.", request.Type, context.GroupId.ToString());
  61. IGroupState idleState = new IdleGroupState(Logger);
  62. context.SetState(idleState);
  63. var stopRequest = new StopGroupRequest();
  64. idleState.HandleRequest(context, Type, stopRequest, session, cancellationToken);
  65. }
  66. }
  67. /// <inheritdoc />
  68. public virtual void HandleRequest(IGroupStateContext context, GroupStateType prevState, MovePlaylistItemGroupRequest request, SessionInfo session, CancellationToken cancellationToken)
  69. {
  70. var result = context.MoveItemInPlayQueue(request.PlaylistItemId, request.NewIndex);
  71. if (!result)
  72. {
  73. Logger.LogError("HandleRequest: {RequestType} in group {GroupId}, unable to move item in play queue.", request.Type, context.GroupId.ToString());
  74. return;
  75. }
  76. var playQueueUpdate = context.GetPlayQueueUpdate(PlayQueueUpdateReason.MoveItem);
  77. var update = context.NewSyncPlayGroupUpdate(GroupUpdateType.PlayQueue, playQueueUpdate);
  78. context.SendGroupUpdate(session, SyncPlayBroadcastType.AllGroup, update, cancellationToken);
  79. }
  80. /// <inheritdoc />
  81. public virtual void HandleRequest(IGroupStateContext context, GroupStateType prevState, QueueGroupRequest request, SessionInfo session, CancellationToken cancellationToken)
  82. {
  83. var result = context.AddToPlayQueue(request.ItemIds, request.Mode);
  84. if (!result)
  85. {
  86. Logger.LogError("HandleRequest: {RequestType} in group {GroupId}, unable to add items to play queue.", request.Type, context.GroupId.ToString());
  87. return;
  88. }
  89. var reason = request.Mode switch
  90. {
  91. GroupQueueMode.QueueNext => PlayQueueUpdateReason.QueueNext,
  92. _ => PlayQueueUpdateReason.Queue
  93. };
  94. var playQueueUpdate = context.GetPlayQueueUpdate(reason);
  95. var update = context.NewSyncPlayGroupUpdate(GroupUpdateType.PlayQueue, playQueueUpdate);
  96. context.SendGroupUpdate(session, SyncPlayBroadcastType.AllGroup, update, cancellationToken);
  97. }
  98. /// <inheritdoc />
  99. public virtual void HandleRequest(IGroupStateContext context, GroupStateType prevState, UnpauseGroupRequest request, SessionInfo session, CancellationToken cancellationToken)
  100. {
  101. UnhandledRequest(request);
  102. }
  103. /// <inheritdoc />
  104. public virtual void HandleRequest(IGroupStateContext context, GroupStateType prevState, PauseGroupRequest request, SessionInfo session, CancellationToken cancellationToken)
  105. {
  106. UnhandledRequest(request);
  107. }
  108. /// <inheritdoc />
  109. public virtual void HandleRequest(IGroupStateContext context, GroupStateType prevState, StopGroupRequest request, SessionInfo session, CancellationToken cancellationToken)
  110. {
  111. UnhandledRequest(request);
  112. }
  113. /// <inheritdoc />
  114. public virtual void HandleRequest(IGroupStateContext context, GroupStateType prevState, SeekGroupRequest request, SessionInfo session, CancellationToken cancellationToken)
  115. {
  116. UnhandledRequest(request);
  117. }
  118. /// <inheritdoc />
  119. public virtual void HandleRequest(IGroupStateContext context, GroupStateType prevState, BufferGroupRequest request, SessionInfo session, CancellationToken cancellationToken)
  120. {
  121. UnhandledRequest(request);
  122. }
  123. /// <inheritdoc />
  124. public virtual void HandleRequest(IGroupStateContext context, GroupStateType prevState, ReadyGroupRequest request, SessionInfo session, CancellationToken cancellationToken)
  125. {
  126. UnhandledRequest(request);
  127. }
  128. /// <inheritdoc />
  129. public virtual void HandleRequest(IGroupStateContext context, GroupStateType prevState, NextTrackGroupRequest request, SessionInfo session, CancellationToken cancellationToken)
  130. {
  131. UnhandledRequest(request);
  132. }
  133. /// <inheritdoc />
  134. public virtual void HandleRequest(IGroupStateContext context, GroupStateType prevState, PreviousTrackGroupRequest request, SessionInfo session, CancellationToken cancellationToken)
  135. {
  136. UnhandledRequest(request);
  137. }
  138. /// <inheritdoc />
  139. public virtual void HandleRequest(IGroupStateContext context, GroupStateType prevState, SetRepeatModeGroupRequest request, SessionInfo session, CancellationToken cancellationToken)
  140. {
  141. context.SetRepeatMode(request.Mode);
  142. var playQueueUpdate = context.GetPlayQueueUpdate(PlayQueueUpdateReason.RepeatMode);
  143. var update = context.NewSyncPlayGroupUpdate(GroupUpdateType.PlayQueue, playQueueUpdate);
  144. context.SendGroupUpdate(session, SyncPlayBroadcastType.AllGroup, update, cancellationToken);
  145. }
  146. /// <inheritdoc />
  147. public virtual void HandleRequest(IGroupStateContext context, GroupStateType prevState, SetShuffleModeGroupRequest request, SessionInfo session, CancellationToken cancellationToken)
  148. {
  149. context.SetShuffleMode(request.Mode);
  150. var playQueueUpdate = context.GetPlayQueueUpdate(PlayQueueUpdateReason.ShuffleMode);
  151. var update = context.NewSyncPlayGroupUpdate(GroupUpdateType.PlayQueue, playQueueUpdate);
  152. context.SendGroupUpdate(session, SyncPlayBroadcastType.AllGroup, update, cancellationToken);
  153. }
  154. /// <inheritdoc />
  155. public virtual void HandleRequest(IGroupStateContext context, GroupStateType prevState, PingGroupRequest request, SessionInfo session, CancellationToken cancellationToken)
  156. {
  157. // Collected pings are used to account for network latency when unpausing playback.
  158. context.UpdatePing(session, request.Ping);
  159. }
  160. /// <inheritdoc />
  161. public virtual void HandleRequest(IGroupStateContext context, GroupStateType prevState, IgnoreWaitGroupRequest request, SessionInfo session, CancellationToken cancellationToken)
  162. {
  163. context.SetIgnoreGroupWait(session, request.IgnoreWait);
  164. }
  165. /// <summary>
  166. /// Sends a group state update to all group.
  167. /// </summary>
  168. /// <param name="context">The context of the state.</param>
  169. /// <param name="reason">The reason of the state change.</param>
  170. /// <param name="session">The session.</param>
  171. /// <param name="cancellationToken">The cancellation token.</param>
  172. protected void SendGroupStateUpdate(IGroupStateContext context, IGroupPlaybackRequest reason, SessionInfo session, CancellationToken cancellationToken)
  173. {
  174. // Notify relevant state change event.
  175. var stateUpdate = new GroupStateUpdate(Type, reason.Type);
  176. var update = context.NewSyncPlayGroupUpdate(GroupUpdateType.StateUpdate, stateUpdate);
  177. context.SendGroupUpdate(session, SyncPlayBroadcastType.AllGroup, update, cancellationToken);
  178. }
  179. private void UnhandledRequest(IGroupPlaybackRequest request)
  180. {
  181. Logger.LogWarning("HandleRequest: unhandled {RequestType} request in {StateType} state.", request.Type, Type);
  182. }
  183. }
  184. }