Auto stash before merge of "lyric-lrc-file-support" and "origin/lyric-lrc-file-support"
This commit is contained in:
parent
7520a19985
commit
d9be3874ba
|
@ -68,6 +68,7 @@ using MediaBrowser.Controller.Dto;
|
||||||
using MediaBrowser.Controller.Entities;
|
using MediaBrowser.Controller.Entities;
|
||||||
using MediaBrowser.Controller.Library;
|
using MediaBrowser.Controller.Library;
|
||||||
using MediaBrowser.Controller.LiveTv;
|
using MediaBrowser.Controller.LiveTv;
|
||||||
|
using MediaBrowser.Controller.Lyrics;
|
||||||
using MediaBrowser.Controller.MediaEncoding;
|
using MediaBrowser.Controller.MediaEncoding;
|
||||||
using MediaBrowser.Controller.Net;
|
using MediaBrowser.Controller.Net;
|
||||||
using MediaBrowser.Controller.Notifications;
|
using MediaBrowser.Controller.Notifications;
|
||||||
|
@ -95,6 +96,7 @@ using MediaBrowser.Model.Serialization;
|
||||||
using MediaBrowser.Model.System;
|
using MediaBrowser.Model.System;
|
||||||
using MediaBrowser.Model.Tasks;
|
using MediaBrowser.Model.Tasks;
|
||||||
using MediaBrowser.Providers.Chapters;
|
using MediaBrowser.Providers.Chapters;
|
||||||
|
using MediaBrowser.Providers.Lyric;
|
||||||
using MediaBrowser.Providers.Manager;
|
using MediaBrowser.Providers.Manager;
|
||||||
using MediaBrowser.Providers.Plugins.Tmdb;
|
using MediaBrowser.Providers.Plugins.Tmdb;
|
||||||
using MediaBrowser.Providers.Subtitles;
|
using MediaBrowser.Providers.Subtitles;
|
||||||
|
|
|
@ -19,6 +19,7 @@ using MediaBrowser.Controller.Entities;
|
||||||
using MediaBrowser.Controller.Entities.Audio;
|
using MediaBrowser.Controller.Entities.Audio;
|
||||||
using MediaBrowser.Controller.Library;
|
using MediaBrowser.Controller.Library;
|
||||||
using MediaBrowser.Controller.LiveTv;
|
using MediaBrowser.Controller.LiveTv;
|
||||||
|
using MediaBrowser.Controller.Lyrics;
|
||||||
using MediaBrowser.Controller.Persistence;
|
using MediaBrowser.Controller.Persistence;
|
||||||
using MediaBrowser.Controller.Playlists;
|
using MediaBrowser.Controller.Playlists;
|
||||||
using MediaBrowser.Controller.Providers;
|
using MediaBrowser.Controller.Providers;
|
||||||
|
@ -51,6 +52,8 @@ namespace Emby.Server.Implementations.Dto
|
||||||
private readonly IMediaSourceManager _mediaSourceManager;
|
private readonly IMediaSourceManager _mediaSourceManager;
|
||||||
private readonly Lazy<ILiveTvManager> _livetvManagerFactory;
|
private readonly Lazy<ILiveTvManager> _livetvManagerFactory;
|
||||||
|
|
||||||
|
private readonly IEnumerable<ILyricsProvider> _lyricProviders;
|
||||||
|
|
||||||
public DtoService(
|
public DtoService(
|
||||||
ILogger<DtoService> logger,
|
ILogger<DtoService> logger,
|
||||||
ILibraryManager libraryManager,
|
ILibraryManager libraryManager,
|
||||||
|
@ -60,7 +63,8 @@ namespace Emby.Server.Implementations.Dto
|
||||||
IProviderManager providerManager,
|
IProviderManager providerManager,
|
||||||
IApplicationHost appHost,
|
IApplicationHost appHost,
|
||||||
IMediaSourceManager mediaSourceManager,
|
IMediaSourceManager mediaSourceManager,
|
||||||
Lazy<ILiveTvManager> livetvManagerFactory)
|
Lazy<ILiveTvManager> livetvManagerFactory,
|
||||||
|
IEnumerable<ILyricsProvider> lyricProviders)
|
||||||
{
|
{
|
||||||
_logger = logger;
|
_logger = logger;
|
||||||
_libraryManager = libraryManager;
|
_libraryManager = libraryManager;
|
||||||
|
@ -71,6 +75,7 @@ namespace Emby.Server.Implementations.Dto
|
||||||
_appHost = appHost;
|
_appHost = appHost;
|
||||||
_mediaSourceManager = mediaSourceManager;
|
_mediaSourceManager = mediaSourceManager;
|
||||||
_livetvManagerFactory = livetvManagerFactory;
|
_livetvManagerFactory = livetvManagerFactory;
|
||||||
|
_lyricProviders = lyricProviders;
|
||||||
}
|
}
|
||||||
|
|
||||||
private ILiveTvManager LivetvManager => _livetvManagerFactory.Value;
|
private ILiveTvManager LivetvManager => _livetvManagerFactory.Value;
|
||||||
|
@ -142,7 +147,7 @@ namespace Emby.Server.Implementations.Dto
|
||||||
}
|
}
|
||||||
else if (item is Audio)
|
else if (item is Audio)
|
||||||
{
|
{
|
||||||
dto.HasLocalLyricsFile = ItemHelper.HasLyricFile(item.Path);
|
dto.HasLyrics = MediaBrowser.Controller.Lyrics.LyricInfo.HasLyricFile(_lyricProviders, item.Path);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (item is IItemByName itemByName
|
if (item is IItemByName itemByName
|
||||||
|
|
|
@ -13,6 +13,7 @@ using MediaBrowser.Controller.Dto;
|
||||||
using MediaBrowser.Controller.Entities;
|
using MediaBrowser.Controller.Entities;
|
||||||
using MediaBrowser.Controller.Entities.Audio;
|
using MediaBrowser.Controller.Entities.Audio;
|
||||||
using MediaBrowser.Controller.Library;
|
using MediaBrowser.Controller.Library;
|
||||||
|
using MediaBrowser.Controller.Lyrics;
|
||||||
using MediaBrowser.Controller.Providers;
|
using MediaBrowser.Controller.Providers;
|
||||||
using MediaBrowser.Model.Dto;
|
using MediaBrowser.Model.Dto;
|
||||||
using MediaBrowser.Model.Entities;
|
using MediaBrowser.Model.Entities;
|
||||||
|
@ -414,8 +415,8 @@ namespace Jellyfin.Api.Controllers
|
||||||
return NotFound();
|
return NotFound();
|
||||||
}
|
}
|
||||||
|
|
||||||
// Super nieve implementation. I would suggest building a lyric service of some sort and doing this there.
|
var result = MediaBrowser.Controller.Lyrics.LyricInfo.GetLyricData(_lyricProviders, item);
|
||||||
foreach (var provider in _lyricProviders)
|
if (result is not null)
|
||||||
{
|
{
|
||||||
provider.Process(item);
|
provider.Process(item);
|
||||||
if (provider.HasData)
|
if (provider.HasData)
|
||||||
|
|
|
@ -1,106 +0,0 @@
|
||||||
using System;
|
|
||||||
using System.Collections.Generic;
|
|
||||||
using System.Dynamic;
|
|
||||||
using System.Globalization;
|
|
||||||
using System.IO;
|
|
||||||
using System.Linq;
|
|
||||||
using Jellyfin.Api.Models.UserDtos;
|
|
||||||
using LrcParser.Model;
|
|
||||||
using LrcParser.Parser;
|
|
||||||
using MediaBrowser.Controller.Entities;
|
|
||||||
|
|
||||||
namespace Jellyfin.Api.Helpers
|
|
||||||
{
|
|
||||||
/// <summary>
|
|
||||||
/// Item helper.
|
|
||||||
/// </summary>
|
|
||||||
public static class ItemHelper
|
|
||||||
{
|
|
||||||
/// <summary>
|
|
||||||
/// Opens lyrics file, converts to a List of Lyrics, and returns it.
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="item">Requested Item.</param>
|
|
||||||
/// <returns>Collection of Lyrics.</returns>
|
|
||||||
internal static object? GetLyricData(BaseItem item)
|
|
||||||
{
|
|
||||||
// Find all classes that implement ILyricsProvider Interface
|
|
||||||
var foundLyricProviders = System.Reflection.Assembly.GetExecutingAssembly()
|
|
||||||
.GetTypes()
|
|
||||||
.Where(type => typeof(ILyricsProvider).IsAssignableFrom(type) && !type.IsInterface);
|
|
||||||
|
|
||||||
if (!foundLyricProviders.Any())
|
|
||||||
{
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
foreach (var provider in foundLyricProviders)
|
|
||||||
{
|
|
||||||
ILyricsProvider? newProvider = Activator.CreateInstance(provider) as ILyricsProvider;
|
|
||||||
if (newProvider is not null)
|
|
||||||
{
|
|
||||||
newProvider.Process(item);
|
|
||||||
if (newProvider.HasData)
|
|
||||||
{
|
|
||||||
return newProvider.Data;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Checks if requested item has a matching lyric file.
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="itemPath">Path of requested item.</param>
|
|
||||||
/// <returns>True if item has a matching lyrics file.</returns>
|
|
||||||
public static string? GetLyricFilePath(string itemPath)
|
|
||||||
{
|
|
||||||
// Find all classes that implement ILyricsProvider Interface
|
|
||||||
var foundLyricProviders = System.Reflection.Assembly.GetExecutingAssembly()
|
|
||||||
.GetTypes()
|
|
||||||
.Where(type => typeof(ILyricsProvider).IsAssignableFrom(type) && !type.IsInterface);
|
|
||||||
|
|
||||||
if (!foundLyricProviders.Any())
|
|
||||||
{
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Iterate over all found lyric providers
|
|
||||||
foreach (var provider in foundLyricProviders)
|
|
||||||
{
|
|
||||||
ILyricsProvider? foundProvider = Activator.CreateInstance(provider) as ILyricsProvider;
|
|
||||||
if (foundProvider?.FileExtensions is null)
|
|
||||||
{
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (foundProvider.FileExtensions.Any())
|
|
||||||
{
|
|
||||||
foreach (string lyricFileExtension in foundProvider.FileExtensions)
|
|
||||||
{
|
|
||||||
string lyricFilePath = @Path.ChangeExtension(itemPath, lyricFileExtension);
|
|
||||||
if (System.IO.File.Exists(lyricFilePath))
|
|
||||||
{
|
|
||||||
return lyricFilePath;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Checks if requested item has a matching local lyric file.
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="itemPath">Path of requested item.</param>
|
|
||||||
/// <returns>True if item has a matching lyrics file; otherwise false.</returns>
|
|
||||||
public static bool HasLyricFile(string itemPath)
|
|
||||||
{
|
|
||||||
string? lyricFilePath = GetLyricFilePath(itemPath);
|
|
||||||
return !string.IsNullOrEmpty(lyricFilePath);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -17,7 +17,6 @@
|
||||||
</PropertyGroup>
|
</PropertyGroup>
|
||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<PackageReference Include="LrcParser" Version="2022.529.1" />
|
|
||||||
<PackageReference Include="Microsoft.AspNetCore.Authorization" Version="6.0.8" />
|
<PackageReference Include="Microsoft.AspNetCore.Authorization" Version="6.0.8" />
|
||||||
<PackageReference Include="Microsoft.Extensions.Http" Version="6.0.0" />
|
<PackageReference Include="Microsoft.Extensions.Http" Version="6.0.0" />
|
||||||
<PackageReference Include="Swashbuckle.AspNetCore" Version="6.2.3" />
|
<PackageReference Include="Swashbuckle.AspNetCore" Version="6.2.3" />
|
||||||
|
|
|
@ -1,34 +0,0 @@
|
||||||
using System.Collections.ObjectModel;
|
|
||||||
using MediaBrowser.Controller.Entities;
|
|
||||||
|
|
||||||
namespace Jellyfin.Api.Models.UserDtos
|
|
||||||
{
|
|
||||||
/// <summary>
|
|
||||||
/// Interface ILyricsProvider.
|
|
||||||
/// </summary>
|
|
||||||
public interface ILyricsProvider
|
|
||||||
{
|
|
||||||
/// <summary>
|
|
||||||
/// Gets a value indicating the File Extenstions this provider works with.
|
|
||||||
/// </summary>
|
|
||||||
public Collection<string>? FileExtensions { get; }
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Gets a value indicating whether Process() generated data.
|
|
||||||
/// </summary>
|
|
||||||
/// <returns><c>true</c> if data generated; otherwise, <c>false</c>.</returns>
|
|
||||||
bool HasData { get; }
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Gets Data object generated by Process() method.
|
|
||||||
/// </summary>
|
|
||||||
/// <returns><c>Object</c> with data if no error occured; otherwise, <c>null</c>.</returns>
|
|
||||||
object? Data { get; }
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Opens lyric file for [the specified item], and processes it for API return.
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="item">The item to to process.</param>
|
|
||||||
void Process(BaseItem item);
|
|
||||||
}
|
|
||||||
}
|
|
24
MediaBrowser.Controller/Lyrics/ILyricsProvider.cs
Normal file
24
MediaBrowser.Controller/Lyrics/ILyricsProvider.cs
Normal file
|
@ -0,0 +1,24 @@
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using MediaBrowser.Controller.Entities;
|
||||||
|
|
||||||
|
namespace MediaBrowser.Controller.Lyrics
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Interface ILyricsProvider.
|
||||||
|
/// </summary>
|
||||||
|
public interface ILyricsProvider
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Gets the supported media types for this provider.
|
||||||
|
/// </summary>
|
||||||
|
/// <value>The supported media types.</value>
|
||||||
|
IEnumerable<string> SupportedMediaTypes { get; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets the lyrics.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="item">The item to to process.</param>
|
||||||
|
/// <returns>Task{LyricResponse}.</returns>
|
||||||
|
LyricResponse? GetLyrics(BaseItem item);
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,4 +1,4 @@
|
||||||
namespace Jellyfin.Api.Models.UserDtos
|
namespace MediaBrowser.Controller.Lyrics
|
||||||
{
|
{
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Lyric dto.
|
/// Lyric dto.
|
87
MediaBrowser.Controller/Lyrics/LyricInfo.cs
Normal file
87
MediaBrowser.Controller/Lyrics/LyricInfo.cs
Normal file
|
@ -0,0 +1,87 @@
|
||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.IO;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using MediaBrowser.Controller.Entities;
|
||||||
|
using MediaBrowser.Controller.Lyrics;
|
||||||
|
using MediaBrowser.Controller.Net;
|
||||||
|
using Microsoft.AspNetCore.Mvc;
|
||||||
|
|
||||||
|
namespace MediaBrowser.Controller.Lyrics
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Item helper.
|
||||||
|
/// </summary>
|
||||||
|
public class LyricInfo
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Opens lyrics file, converts to a List of Lyrics, and returns it.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="lyricProviders">Collection of all registered <see cref="ILyricsProvider"/> interfaces.</param>
|
||||||
|
/// <param name="item">Requested Item.</param>
|
||||||
|
/// <returns>Collection of Lyrics.</returns>
|
||||||
|
public static LyricResponse? GetLyricData(IEnumerable<ILyricsProvider> lyricProviders, BaseItem item)
|
||||||
|
{
|
||||||
|
|
||||||
|
foreach (var provider in lyricProviders)
|
||||||
|
{
|
||||||
|
var result = provider.GetLyrics(item);
|
||||||
|
if (result is not null)
|
||||||
|
{
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return new LyricResponse
|
||||||
|
{
|
||||||
|
Lyrics = new List<Lyric>
|
||||||
|
{
|
||||||
|
new Lyric { Start = 0, Text = "Test" }
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Checks if requested item has a matching lyric file.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="lyricProvider">The current lyricProvider interface.</param>
|
||||||
|
/// <param name="itemPath">Path of requested item.</param>
|
||||||
|
/// <returns>True if item has a matching lyrics file.</returns>
|
||||||
|
public static string? GetLyricFilePath(ILyricsProvider lyricProvider, string itemPath)
|
||||||
|
{
|
||||||
|
if (lyricProvider.SupportedMediaTypes.Any())
|
||||||
|
{
|
||||||
|
foreach (string lyricFileExtension in lyricProvider.SupportedMediaTypes)
|
||||||
|
{
|
||||||
|
string lyricFilePath = @Path.ChangeExtension(itemPath, lyricFileExtension);
|
||||||
|
if (System.IO.File.Exists(lyricFilePath))
|
||||||
|
{
|
||||||
|
return lyricFilePath;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Checks if requested item has a matching local lyric file.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="lyricProviders">Collection of all registered <see cref="ILyricsProvider"/> interfaces.</param>
|
||||||
|
/// <param name="itemPath">Path of requested item.</param>
|
||||||
|
/// <returns>True if item has a matching lyrics file; otherwise false.</returns>
|
||||||
|
public static bool HasLyricFile(IEnumerable<ILyricsProvider> lyricProviders, string itemPath)
|
||||||
|
{
|
||||||
|
foreach (var provider in lyricProviders)
|
||||||
|
{
|
||||||
|
if (GetLyricFilePath(provider, itemPath) is not null)
|
||||||
|
{
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
15
MediaBrowser.Controller/Lyrics/LyricResponse.cs
Normal file
15
MediaBrowser.Controller/Lyrics/LyricResponse.cs
Normal file
|
@ -0,0 +1,15 @@
|
||||||
|
#nullable disable
|
||||||
|
|
||||||
|
#pragma warning disable CS1591
|
||||||
|
|
||||||
|
using System.Collections.Generic;
|
||||||
|
|
||||||
|
namespace MediaBrowser.Controller.Lyrics
|
||||||
|
{
|
||||||
|
public class LyricResponse
|
||||||
|
{
|
||||||
|
public IDictionary<string, object> MetaData { get; set; }
|
||||||
|
|
||||||
|
public IEnumerable<Lyric> Lyrics { get; set; }
|
||||||
|
}
|
||||||
|
}
|
|
@ -76,7 +76,7 @@ namespace MediaBrowser.Model.Dto
|
||||||
|
|
||||||
public bool? CanDownload { get; set; }
|
public bool? CanDownload { get; set; }
|
||||||
|
|
||||||
public bool? HasLocalLyricsFile { get; set; }
|
public bool? HasLyrics { get; set; }
|
||||||
|
|
||||||
public bool? HasSubtitles { get; set; }
|
public bool? HasSubtitles { get; set; }
|
||||||
|
|
||||||
|
|
|
@ -4,11 +4,14 @@ using System.Collections.ObjectModel;
|
||||||
using System.Dynamic;
|
using System.Dynamic;
|
||||||
using System.Globalization;
|
using System.Globalization;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using Jellyfin.Api.Helpers;
|
||||||
using LrcParser.Model;
|
using LrcParser.Model;
|
||||||
using LrcParser.Parser;
|
using LrcParser.Parser;
|
||||||
using MediaBrowser.Controller.Entities;
|
using MediaBrowser.Controller.Entities;
|
||||||
|
using MediaBrowser.Controller.Lyrics;
|
||||||
|
|
||||||
namespace Jellyfin.Api.Models.UserDtos
|
namespace MediaBrowser.Providers.Lyric
|
||||||
{
|
{
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// LRC File Lyric Provider.
|
/// LRC File Lyric Provider.
|
||||||
|
@ -20,7 +23,7 @@ namespace Jellyfin.Api.Models.UserDtos
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public LrcLyricsProvider()
|
public LrcLyricsProvider()
|
||||||
{
|
{
|
||||||
FileExtensions = new Collection<string>
|
SupportedMediaTypes = new Collection<string>
|
||||||
{
|
{
|
||||||
"lrc"
|
"lrc"
|
||||||
};
|
};
|
||||||
|
@ -29,13 +32,7 @@ namespace Jellyfin.Api.Models.UserDtos
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Gets a value indicating the File Extenstions this provider works with.
|
/// Gets a value indicating the File Extenstions this provider works with.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public Collection<string>? FileExtensions { get; }
|
public IEnumerable<string> SupportedMediaTypes { get; }
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Gets or Sets a value indicating whether Process() generated data.
|
|
||||||
/// </summary>
|
|
||||||
/// <returns><c>true</c> if data generated; otherwise, <c>false</c>.</returns>
|
|
||||||
public bool HasData { get; set; }
|
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Gets or Sets Data object generated by Process() method.
|
/// Gets or Sets Data object generated by Process() method.
|
||||||
|
@ -47,16 +44,17 @@ namespace Jellyfin.Api.Models.UserDtos
|
||||||
/// Opens lyric file for [the specified item], and processes it for API return.
|
/// Opens lyric file for [the specified item], and processes it for API return.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
/// <param name="item">The item to to process.</param>
|
/// <param name="item">The item to to process.</param>
|
||||||
public void Process(BaseItem item)
|
/// <returns><placeholder>A <see cref="Task"/> representing the asynchronous operation.</placeholder></returns>
|
||||||
|
public LyricResponse? GetLyrics(BaseItem item)
|
||||||
{
|
{
|
||||||
string? lyricFilePath = Helpers.ItemHelper.GetLyricFilePath(item.Path);
|
string? lyricFilePath = LyricInfo.GetLyricFilePath(this, item.Path);
|
||||||
|
|
||||||
if (string.IsNullOrEmpty(lyricFilePath))
|
if (string.IsNullOrEmpty(lyricFilePath))
|
||||||
{
|
{
|
||||||
return;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
List<Lyric> lyricsList = new List<Lyric>();
|
List<MediaBrowser.Controller.Lyrics.Lyric> lyricsList = new List<MediaBrowser.Controller.Lyrics.Lyric>();
|
||||||
|
|
||||||
List<LrcParser.Model.Lyric> sortedLyricData = new List<LrcParser.Model.Lyric>();
|
List<LrcParser.Model.Lyric> sortedLyricData = new List<LrcParser.Model.Lyric>();
|
||||||
var metaData = new ExpandoObject() as IDictionary<string, object>;
|
var metaData = new ExpandoObject() as IDictionary<string, object>;
|
||||||
|
@ -88,30 +86,27 @@ namespace Jellyfin.Api.Models.UserDtos
|
||||||
}
|
}
|
||||||
catch
|
catch
|
||||||
{
|
{
|
||||||
return;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!sortedLyricData.Any())
|
if (!sortedLyricData.Any())
|
||||||
{
|
{
|
||||||
return;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
for (int i = 0; i < sortedLyricData.Count; i++)
|
for (int i = 0; i < sortedLyricData.Count; i++)
|
||||||
{
|
{
|
||||||
var timeData = sortedLyricData[i].TimeTags.ToArray()[0].Value;
|
var timeData = sortedLyricData[i].TimeTags.ToArray()[0].Value;
|
||||||
double ticks = Convert.ToDouble(timeData, new NumberFormatInfo()) * 10000;
|
double ticks = Convert.ToDouble(timeData, new NumberFormatInfo()) * 10000;
|
||||||
lyricsList.Add(new Lyric { Start = Math.Ceiling(ticks), Text = sortedLyricData[i].Text });
|
lyricsList.Add(new MediaBrowser.Controller.Lyrics.Lyric { Start = Math.Ceiling(ticks), Text = sortedLyricData[i].Text });
|
||||||
}
|
}
|
||||||
|
|
||||||
this.HasData = true;
|
|
||||||
if (metaData.Any())
|
if (metaData.Any())
|
||||||
{
|
{
|
||||||
this.Data = new { MetaData = metaData, lyrics = lyricsList };
|
return new LyricResponse { MetaData = metaData, Lyrics = lyricsList };
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
this.Data = new { lyrics = lyricsList };
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return new LyricResponse { Lyrics = lyricsList };
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -1,14 +1,13 @@
|
||||||
using System;
|
using System;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.Collections.ObjectModel;
|
using System.Collections.ObjectModel;
|
||||||
using System.Dynamic;
|
|
||||||
using System.Globalization;
|
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
using LrcParser.Model;
|
using System.Threading.Tasks;
|
||||||
using LrcParser.Parser;
|
using Jellyfin.Api.Helpers;
|
||||||
using MediaBrowser.Controller.Entities;
|
using MediaBrowser.Controller.Entities;
|
||||||
|
using MediaBrowser.Controller.Lyrics;
|
||||||
|
|
||||||
namespace Jellyfin.Api.Models.UserDtos
|
namespace MediaBrowser.Providers.Lyric
|
||||||
{
|
{
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// TXT File Lyric Provider.
|
/// TXT File Lyric Provider.
|
||||||
|
@ -20,7 +19,7 @@ namespace Jellyfin.Api.Models.UserDtos
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public TxtLyricsProvider()
|
public TxtLyricsProvider()
|
||||||
{
|
{
|
||||||
FileExtensions = new Collection<string>
|
SupportedMediaTypes = new Collection<string>
|
||||||
{
|
{
|
||||||
"lrc", "txt"
|
"lrc", "txt"
|
||||||
};
|
};
|
||||||
|
@ -29,13 +28,7 @@ namespace Jellyfin.Api.Models.UserDtos
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Gets a value indicating the File Extenstions this provider works with.
|
/// Gets a value indicating the File Extenstions this provider works with.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public Collection<string>? FileExtensions { get; }
|
public IEnumerable<string> SupportedMediaTypes { get; }
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Gets or Sets a value indicating whether Process() generated data.
|
|
||||||
/// </summary>
|
|
||||||
/// <returns><c>true</c> if data generated; otherwise, <c>false</c>.</returns>
|
|
||||||
public bool HasData { get; set; }
|
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Gets or Sets Data object generated by Process() method.
|
/// Gets or Sets Data object generated by Process() method.
|
||||||
|
@ -47,16 +40,17 @@ namespace Jellyfin.Api.Models.UserDtos
|
||||||
/// Opens lyric file for [the specified item], and processes it for API return.
|
/// Opens lyric file for [the specified item], and processes it for API return.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
/// <param name="item">The item to to process.</param>
|
/// <param name="item">The item to to process.</param>
|
||||||
public void Process(BaseItem item)
|
/// <returns><placeholder>A <see cref="Task"/> representing the asynchronous operation.</placeholder></returns>
|
||||||
|
public LyricResponse? GetLyrics(BaseItem item)
|
||||||
{
|
{
|
||||||
string? lyricFilePath = Helpers.ItemHelper.GetLyricFilePath(item.Path);
|
string? lyricFilePath = LyricInfo.GetLyricFilePath(this, item.Path);
|
||||||
|
|
||||||
if (string.IsNullOrEmpty(lyricFilePath))
|
if (string.IsNullOrEmpty(lyricFilePath))
|
||||||
{
|
{
|
||||||
return;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
List<Lyric> lyricsList = new List<Lyric>();
|
List<MediaBrowser.Controller.Lyrics.Lyric> lyricsList = new List<MediaBrowser.Controller.Lyrics.Lyric>();
|
||||||
|
|
||||||
string lyricData = System.IO.File.ReadAllText(lyricFilePath);
|
string lyricData = System.IO.File.ReadAllText(lyricFilePath);
|
||||||
|
|
||||||
|
@ -66,16 +60,15 @@ namespace Jellyfin.Api.Models.UserDtos
|
||||||
|
|
||||||
if (!lyricTextLines.Any())
|
if (!lyricTextLines.Any())
|
||||||
{
|
{
|
||||||
return;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
foreach (string lyricLine in lyricTextLines)
|
foreach (string lyricLine in lyricTextLines)
|
||||||
{
|
{
|
||||||
lyricsList.Add(new Lyric { Text = lyricLine });
|
lyricsList.Add(new MediaBrowser.Controller.Lyrics.Lyric { Text = lyricLine });
|
||||||
}
|
}
|
||||||
|
|
||||||
this.HasData = true;
|
return new LyricResponse { Lyrics = lyricsList };
|
||||||
this.Data = new { lyrics = lyricsList };
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -6,6 +6,7 @@
|
||||||
</PropertyGroup>
|
</PropertyGroup>
|
||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
|
<ProjectReference Include="..\Jellyfin.Api\Jellyfin.Api.csproj" />
|
||||||
<ProjectReference Include="..\MediaBrowser.Controller\MediaBrowser.Controller.csproj" />
|
<ProjectReference Include="..\MediaBrowser.Controller\MediaBrowser.Controller.csproj" />
|
||||||
<ProjectReference Include="..\MediaBrowser.Model\MediaBrowser.Model.csproj" />
|
<ProjectReference Include="..\MediaBrowser.Model\MediaBrowser.Model.csproj" />
|
||||||
<ProjectReference Include="..\DvdLib\DvdLib.csproj" />
|
<ProjectReference Include="..\DvdLib\DvdLib.csproj" />
|
||||||
|
@ -16,6 +17,7 @@
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
|
<PackageReference Include="LrcParser" Version="2022.529.1" />
|
||||||
<PackageReference Include="Microsoft.Extensions.Configuration.Abstractions" Version="6.0.0" />
|
<PackageReference Include="Microsoft.Extensions.Configuration.Abstractions" Version="6.0.0" />
|
||||||
<PackageReference Include="Microsoft.Extensions.Caching.Abstractions" Version="6.0.0" />
|
<PackageReference Include="Microsoft.Extensions.Caching.Abstractions" Version="6.0.0" />
|
||||||
<PackageReference Include="Microsoft.Extensions.Http" Version="6.0.0" />
|
<PackageReference Include="Microsoft.Extensions.Http" Version="6.0.0" />
|
||||||
|
|
Loading…
Reference in New Issue
Block a user