Browse Source

add media card overflow menu (#2180)

* add media card overflow menu

* remove commented code
pull/2181/head
Jason Dove 4 weeks ago committed by GitHub
parent
commit
7a4c832156
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
  1. 2
      CHANGELOG.md
  2. 2
      ErsatzTV.Application/MediaCards/ActorCardViewModel.cs
  3. 3
      ErsatzTV.Application/MediaCards/ArtistCardViewModel.cs
  4. 3
      ErsatzTV.Application/MediaCards/ImageCardViewModel.cs
  5. 3
      ErsatzTV.Application/MediaCards/MediaCardViewModel.cs
  6. 3
      ErsatzTV.Application/MediaCards/MovieCardViewModel.cs
  7. 3
      ErsatzTV.Application/MediaCards/MusicVideoCardViewModel.cs
  8. 3
      ErsatzTV.Application/MediaCards/OtherVideoCardViewModel.cs
  9. 3
      ErsatzTV.Application/MediaCards/RemoteStreamCardViewModel.cs
  10. 3
      ErsatzTV.Application/MediaCards/SongCardViewModel.cs
  11. 3
      ErsatzTV.Application/MediaCards/TelevisionEpisodeCardViewModel.cs
  12. 3
      ErsatzTV.Application/MediaCards/TelevisionSeasonCardViewModel.cs
  13. 3
      ErsatzTV.Application/MediaCards/TelevisionShowCardViewModel.cs
  14. 3
      ErsatzTV.Application/MediaCollections/MediaCollectionViewModel.cs
  15. 16
      ErsatzTV.Application/MediaItems/Queries/GetMediaItemInfoHandler.cs
  16. 84
      ErsatzTV/Shared/MediaCard.razor
  17. 13
      ErsatzTV/wwwroot/css/site.css

2
CHANGELOG.md

@ -128,6 +128,8 @@ The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/). @@ -128,6 +128,8 @@ The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/).
- `Playlist Item Size`: will play all media items from the current playlist item before continuing to the next schedule item
- `Multi-Episode Group Size`: will play all media items from the current multi-part episode group, or one ungrouped media item
- Change watermark width and margins to allow decimals
- Move `Add To Collection` button to overflow menu on all media cards, and add `Show Media Info` to overflow menu
- This allows showing media info for all media kinds
### Fixed
- Fix QSV acceleration in docker with older Intel devices

2
ErsatzTV.Application/MediaCards/ActorCardViewModel.cs

@ -3,4 +3,4 @@ @@ -3,4 +3,4 @@
namespace ErsatzTV.Application.MediaCards;
public record ActorCardViewModel(int Id, string Name, string Role, string Thumb, MediaItemState State) :
MediaCardViewModel(Id, Name, Role, Name, Thumb, State);
MediaCardViewModel(Id, Name, Role, Name, Thumb, State, HasMediaInfo: false);

3
ErsatzTV.Application/MediaCards/ArtistCardViewModel.cs

@ -14,4 +14,5 @@ public record ArtistCardViewModel( @@ -14,4 +14,5 @@ public record ArtistCardViewModel(
Subtitle,
SortTitle,
Poster,
State);
State,
HasMediaInfo: false);

3
ErsatzTV.Application/MediaCards/ImageCardViewModel.cs

@ -14,7 +14,8 @@ public record ImageCardViewModel( @@ -14,7 +14,8 @@ public record ImageCardViewModel(
Subtitle,
SortTitle,
Poster,
State)
State,
HasMediaInfo: true)
{
public int CustomIndex { get; set; }
}

3
ErsatzTV.Application/MediaCards/MediaCardViewModel.cs

@ -8,4 +8,5 @@ public record MediaCardViewModel( @@ -8,4 +8,5 @@ public record MediaCardViewModel(
string Subtitle,
string SortTitle,
string Poster,
MediaItemState State);
MediaItemState State,
bool HasMediaInfo);

3
ErsatzTV.Application/MediaCards/MovieCardViewModel.cs

@ -14,7 +14,8 @@ public record MovieCardViewModel( @@ -14,7 +14,8 @@ public record MovieCardViewModel(
Subtitle,
SortTitle,
Poster,
State)
State,
HasMediaInfo: true)
{
public int CustomIndex { get; set; }
}

3
ErsatzTV.Application/MediaCards/MusicVideoCardViewModel.cs

@ -18,7 +18,8 @@ public record MusicVideoCardViewModel( @@ -18,7 +18,8 @@ public record MusicVideoCardViewModel(
Subtitle,
SortTitle,
Poster,
State)
State,
HasMediaInfo: true)
{
public int CustomIndex { get; set; }
}

3
ErsatzTV.Application/MediaCards/OtherVideoCardViewModel.cs

@ -14,7 +14,8 @@ public record OtherVideoCardViewModel( @@ -14,7 +14,8 @@ public record OtherVideoCardViewModel(
Subtitle,
SortTitle,
Poster,
State)
State,
HasMediaInfo: true)
{
public int CustomIndex { get; set; }
}

3
ErsatzTV.Application/MediaCards/RemoteStreamCardViewModel.cs

@ -14,7 +14,8 @@ public record RemoteStreamCardViewModel( @@ -14,7 +14,8 @@ public record RemoteStreamCardViewModel(
Subtitle,
SortTitle,
Poster,
State)
State,
HasMediaInfo: true)
{
public int CustomIndex { get; set; }
}

3
ErsatzTV.Application/MediaCards/SongCardViewModel.cs

@ -14,7 +14,8 @@ public record SongCardViewModel( @@ -14,7 +14,8 @@ public record SongCardViewModel(
Subtitle,
SortTitle,
Poster,
State)
State,
HasMediaInfo: true)
{
public int CustomIndex { get; set; }
}

3
ErsatzTV.Application/MediaCards/TelevisionEpisodeCardViewModel.cs

@ -24,4 +24,5 @@ public record TelevisionEpisodeCardViewModel( @@ -24,4 +24,5 @@ public record TelevisionEpisodeCardViewModel(
$"Episode {Episode}",
SortTitle,
Poster,
State);
State,
HasMediaInfo: true);

3
ErsatzTV.Application/MediaCards/TelevisionSeasonCardViewModel.cs

@ -17,4 +17,5 @@ public record TelevisionSeasonCardViewModel( @@ -17,4 +17,5 @@ public record TelevisionSeasonCardViewModel(
Subtitle,
SortTitle,
Poster,
State);
State,
HasMediaInfo: false);

3
ErsatzTV.Application/MediaCards/TelevisionShowCardViewModel.cs

@ -14,4 +14,5 @@ public record TelevisionShowCardViewModel( @@ -14,4 +14,5 @@ public record TelevisionShowCardViewModel(
Subtitle,
SortTitle,
Poster,
State);
State,
HasMediaInfo: false);

3
ErsatzTV.Application/MediaCollections/MediaCollectionViewModel.cs

@ -14,4 +14,5 @@ public record MediaCollectionViewModel( @@ -14,4 +14,5 @@ public record MediaCollectionViewModel(
string.Empty,
Name,
string.Empty,
State);
State,
HasMediaInfo: false);

16
ErsatzTV.Application/MediaItems/Queries/GetMediaItemInfoHandler.cs

@ -30,7 +30,7 @@ public class GetMediaItemInfoHandler : IRequestHandler<GetMediaItemInfo, Either< @@ -30,7 +30,7 @@ public class GetMediaItemInfoHandler : IRequestHandler<GetMediaItemInfo, Either<
.ThenInclude(l => l.MediaSource)
// TODO: support all media types here
.Include(i => (i as Movie).MovieMetadata)
.ThenInclude(mv => mv.Subtitles)
.ThenInclude(mm => mm.Subtitles)
.Include(i => (i as Movie).MediaVersions)
.ThenInclude(mv => mv.Chapters)
.Include(i => (i as Movie).MediaVersions)
@ -40,40 +40,42 @@ public class GetMediaItemInfoHandler : IRequestHandler<GetMediaItemInfo, Either< @@ -40,40 +40,42 @@ public class GetMediaItemInfoHandler : IRequestHandler<GetMediaItemInfo, Either<
.Include(i => (i as Episode).MediaVersions)
.ThenInclude(mv => mv.Streams)
.Include(i => (i as Episode).EpisodeMetadata)
.ThenInclude(mv => mv.Subtitles)
.ThenInclude(em => em.Subtitles)
.Include(i => (i as Episode).Season)
.ThenInclude(s => s.Show)
.ThenInclude(s => s.ShowMetadata)
.Include(i => (i as OtherVideo).OtherVideoMetadata)
.ThenInclude(mv => mv.Subtitles)
.ThenInclude(ovm => ovm.Subtitles)
.Include(i => (i as OtherVideo).MediaVersions)
.ThenInclude(mv => mv.Chapters)
.Include(i => (i as OtherVideo).MediaVersions)
.ThenInclude(mv => mv.Streams)
.Include(i => (i as Image).ImageMetadata)
.ThenInclude(mv => mv.Subtitles)
.ThenInclude(im => im.Subtitles)
.Include(i => (i as Image).MediaVersions)
.ThenInclude(mv => mv.Chapters)
.Include(i => (i as Image).MediaVersions)
.ThenInclude(mv => mv.Streams)
.Include(i => (i as RemoteStream).RemoteStreamMetadata)
.ThenInclude(mv => mv.Subtitles)
.ThenInclude(rsm => rsm.Subtitles)
.Include(i => (i as RemoteStream).MediaVersions)
.ThenInclude(mv => mv.Chapters)
.Include(i => (i as RemoteStream).MediaVersions)
.ThenInclude(mv => mv.Streams)
.Include(i => (i as Song).SongMetadata)
.ThenInclude(mv => mv.Subtitles)
.ThenInclude(sm => sm.Subtitles)
.Include(i => (i as Song).MediaVersions)
.ThenInclude(mv => mv.Chapters)
.Include(i => (i as Song).MediaVersions)
.ThenInclude(mv => mv.Streams)
.Include(i => (i as MusicVideo).MusicVideoMetadata)
.ThenInclude(mv => mv.Subtitles)
.ThenInclude(mvm => mvm.Subtitles)
.Include(i => (i as MusicVideo).MediaVersions)
.ThenInclude(mv => mv.Chapters)
.Include(i => (i as MusicVideo).MediaVersions)
.ThenInclude(mv => mv.Streams)
.Include(i => (i as MusicVideo).Artist)
.ThenInclude(a => a.ArtistMetadata)
.SelectOneAsync(i => i.Id, i => i.Id == request.Id)
.MapT(Project);

84
ErsatzTV/Shared/MediaCard.razor

@ -1,6 +1,9 @@ @@ -1,6 +1,9 @@
@using ErsatzTV.Application.MediaCards
@using ErsatzTV.Application.MediaItems
@using static Prelude
@inject NavigationManager NavigationManager
@inject IDialogService Dialog
@inject IMediator Mediator
<div class="@((ContainerClass ?? "media-card-container mr-6") + " pb-3")" id="@($"item_{Data.MediaItemId}")">
@if (SelectClicked.HasDelegate || !string.IsNullOrWhiteSpace(Href))
@ -35,32 +38,53 @@ @@ -35,32 +38,53 @@
<MudIcon Icon="@Icons.Material.Filled.Warning" Color="Color.Warning"/>
</div>
}
<div class="media-card-overlay" style="">
<MudButton Style="height: 100%; width: 100%"
OnClick="OnOverlayClick">
</MudButton>
<MudElement HtmlTag="div" Class="@_overlayClass" Style="">
<MudButton Style="height: 100%; width: 100%" OnClick="@OnOverlayClick" />
@if (SelectClicked.HasDelegate)
{
<MudIconButton Color="@SelectColor"
Icon="@(IsSelected ? Icons.Material.Filled.CheckBox : Icons.Material.Filled.CheckBoxOutlineBlank)"
Style="left: 0; position: absolute; top: 0;"
Class="media-card-select-checkbox"
OnClick="@(e => SelectClicked.InvokeAsync(e))"/>
}
@if (AddToCollectionClicked.HasDelegate && !IsSelectMode)
@if (!IsSelectMode)
{
<MudIconButton Color="Color.Tertiary"
Icon="@Icons.Material.Filled.PlaylistAdd"
Style="bottom: 0; left: 0; position: absolute;"
OnClick="@(() => AddToCollectionClicked.InvokeAsync(Data))"/>
@if (AddToCollectionClicked.HasDelegate || AddToPlaylistClicked.HasDelegate || Data.HasMediaInfo)
{
<MudMenu Icon="@Icons.Material.Filled.MoreVert"
Style="bottom: 0; left: 0; position: absolute;"
OpenChanged="@OnMenuOpenChanged">
@if (AddToCollectionClicked.HasDelegate)
{
<MudMenuItem Icon="@Icons.Material.Filled.Add"
Label="Add To Collection"
OnClick="@(() => AddToCollectionClicked.InvokeAsync(Data))"/>
}
@if (AddToPlaylistClicked.HasDelegate)
{
<MudMenuItem Icon="@Icons.Material.Filled.PlaylistAdd"
Label="Add To Playlist"
OnClick="@(() => AddToPlaylistClicked.InvokeAsync(Data))"/>
}
@if (Data.HasMediaInfo)
{
<MudMenuItem Icon="@Icons.Material.Filled.Info"
Label="Show Media Info"
OnClick="@ShowMediaInfo"/>
}
</MudMenu>
}
@if (DeleteClicked.HasDelegate)
{
<MudIconButton Color="Color.Error"
Icon="@Icons.Material.Filled.Delete"
Style="bottom: 0; position: absolute; right: 0;"
OnClick="@(() => DeleteClicked.InvokeAsync(Data))"/>
}
}
@if (DeleteClicked.HasDelegate && !IsSelectMode)
{
<MudIconButton Color="Color.Error"
Icon="@Icons.Material.Filled.Delete"
Style="bottom: 0; position: absolute; right: 0;"
OnClick="@(() => DeleteClicked.InvokeAsync(Data))"/>
}
</div>
</MudElement>
</div>
}
else
@ -117,6 +141,9 @@ @@ -117,6 +141,9 @@
[Parameter]
public EventCallback<MediaCardViewModel> AddToCollectionClicked { get; set; }
[Parameter]
public EventCallback<MediaCardViewModel> AddToPlaylistClicked { get; set; }
[Parameter]
public EventCallback<MouseEventArgs> SelectClicked { get; set; }
@ -132,6 +159,8 @@ @@ -132,6 +159,8 @@
[Parameter]
public bool IsRemoteArtwork { get; set; }
private string _overlayClass = "media-card-overlay";
private string GetPlaceholder(string sortTitle)
{
if (Placeholder != null)
@ -173,4 +202,25 @@ @@ -173,4 +202,25 @@
}
}
private void OnMenuOpenChanged(bool open)
{
_overlayClass = open
? "media-card-overlay media-card-overlay-menu-open"
: "media-card-overlay";
StateHasChanged();
}
private async Task ShowMediaInfo()
{
Either<BaseError, MediaItemInfo> maybeInfo = await Mediator.Send(new GetMediaItemInfo(Data.MediaItemId));
foreach (MediaItemInfo info in maybeInfo.RightToSeq())
{
var parameters = new DialogParameters { { "MediaItemInfo", info } };
var options = new DialogOptions { CloseButton = true, CloseOnEscapeKey = true, MaxWidth = MaxWidth.Medium, FullWidth = true };
IDialogReference dialog = await Dialog.ShowAsync<MediaItemInfoDialog>(Data.Title, parameters, options);
DialogResult _ = await dialog.Result;
}
}
}

13
ErsatzTV/wwwroot/css/site.css

@ -31,15 +31,6 @@ @@ -31,15 +31,6 @@
.media-card-poster-placeholder { font-weight: bold; }
.media-card-menu {
bottom: 0;
display: none;
position: absolute;
right: 0;
}
.media-card:hover .media-card-menu { display: block; }
.media-card-overlay {
background: rgba(0, 0, 0, 0.4);
border-radius: 4px;
@ -54,6 +45,10 @@ @@ -54,6 +45,10 @@
.media-card-overlay:hover { opacity: 1; }
.media-card-overlay:not(:hover) button.media-card-select-checkbox { opacity: 0; }
.media-card-overlay-menu-open { opacity: 1; }
.search-bar .mud-input { height: 42px; }
.search-bar {

Loading…
Cancel
Save