mirror of
https://github.com/jellyfin/jellyfin.git
synced 2025-07-09 03:04:24 -04:00
Merge pull request #12543 from Shadowghost/upgrade-lrcparser
Upgrade LRCParser to 2024.0728.2
This commit is contained in:
commit
9effdc7df6
@ -22,7 +22,7 @@
|
|||||||
<PackageVersion Include="IDisposableAnalyzers" Version="4.0.8" />
|
<PackageVersion Include="IDisposableAnalyzers" Version="4.0.8" />
|
||||||
<PackageVersion Include="Jellyfin.XmlTv" Version="10.8.0" />
|
<PackageVersion Include="Jellyfin.XmlTv" Version="10.8.0" />
|
||||||
<PackageVersion Include="libse" Version="4.0.7" />
|
<PackageVersion Include="libse" Version="4.0.7" />
|
||||||
<PackageVersion Include="LrcParser" Version="2023.524.0" />
|
<PackageVersion Include="LrcParser" Version="2024.0728.2" />
|
||||||
<PackageVersion Include="MetaBrainz.MusicBrainz" Version="6.1.0" />
|
<PackageVersion Include="MetaBrainz.MusicBrainz" Version="6.1.0" />
|
||||||
<PackageVersion Include="Microsoft.AspNetCore.Authorization" Version="8.0.8" />
|
<PackageVersion Include="Microsoft.AspNetCore.Authorization" Version="8.0.8" />
|
||||||
<PackageVersion Include="Microsoft.AspNetCore.Mvc.Testing" Version="8.0.8" />
|
<PackageVersion Include="Microsoft.AspNetCore.Mvc.Testing" Version="8.0.8" />
|
||||||
|
@ -1,6 +1,5 @@
|
|||||||
using System;
|
using System;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.Globalization;
|
|
||||||
using System.IO;
|
using System.IO;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
using Jellyfin.Extensions;
|
using Jellyfin.Extensions;
|
||||||
@ -20,7 +19,6 @@ public class LrcLyricParser : ILyricParser
|
|||||||
private readonly LyricParser _lrcLyricParser;
|
private readonly LyricParser _lrcLyricParser;
|
||||||
|
|
||||||
private static readonly string[] _supportedMediaTypes = [".lrc", ".elrc"];
|
private static readonly string[] _supportedMediaTypes = [".lrc", ".elrc"];
|
||||||
private static readonly string[] _acceptedTimeFormats = ["HH:mm:ss", "H:mm:ss", "mm:ss", "m:ss"];
|
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Initializes a new instance of the <see cref="LrcLyricParser"/> class.
|
/// Initializes a new instance of the <see cref="LrcLyricParser"/> class.
|
||||||
@ -59,37 +57,7 @@ public class LrcLyricParser : ILyricParser
|
|||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
List<LrcParser.Model.Lyric> sortedLyricData = lyricData.Lyrics.Where(x => x.TimeTags.Count > 0).OrderBy(x => x.TimeTags.First().Value).ToList();
|
List<LrcParser.Model.Lyric> sortedLyricData = lyricData.Lyrics.OrderBy(x => x.StartTime).ToList();
|
||||||
|
|
||||||
// Parse metadata rows
|
|
||||||
var metaDataRows = lyricData.Lyrics
|
|
||||||
.Where(x => x.TimeTags.Count == 0)
|
|
||||||
.Where(x => x.Text.StartsWith('[') && x.Text.EndsWith(']'))
|
|
||||||
.Select(x => x.Text)
|
|
||||||
.ToList();
|
|
||||||
|
|
||||||
var fileMetaData = new Dictionary<string, string>(StringComparer.OrdinalIgnoreCase);
|
|
||||||
foreach (string metaDataRow in metaDataRows)
|
|
||||||
{
|
|
||||||
var index = metaDataRow.IndexOf(':', StringComparison.OrdinalIgnoreCase);
|
|
||||||
if (index == -1)
|
|
||||||
{
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Remove square bracket before field name, and after field value
|
|
||||||
// Example 1: [au: 1hitsong]
|
|
||||||
// Example 2: [ar: Calabrese]
|
|
||||||
var metaDataFieldName = GetMetadataFieldName(metaDataRow, index);
|
|
||||||
var metaDataFieldValue = GetMetadataValue(metaDataRow, index);
|
|
||||||
|
|
||||||
if (string.IsNullOrEmpty(metaDataFieldName) || string.IsNullOrEmpty(metaDataFieldValue))
|
|
||||||
{
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
fileMetaData[metaDataFieldName] = metaDataFieldValue;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (sortedLyricData.Count == 0)
|
if (sortedLyricData.Count == 0)
|
||||||
{
|
{
|
||||||
@ -100,99 +68,10 @@ public class LrcLyricParser : ILyricParser
|
|||||||
|
|
||||||
for (int i = 0; i < sortedLyricData.Count; i++)
|
for (int i = 0; i < sortedLyricData.Count; i++)
|
||||||
{
|
{
|
||||||
var timeData = sortedLyricData[i].TimeTags.First().Value;
|
long ticks = TimeSpan.FromMilliseconds(sortedLyricData[i].StartTime).Ticks;
|
||||||
if (timeData is null)
|
|
||||||
{
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
long ticks = TimeSpan.FromMilliseconds(timeData.Value).Ticks;
|
|
||||||
lyricList.Add(new LyricLine(sortedLyricData[i].Text.Trim(), ticks));
|
lyricList.Add(new LyricLine(sortedLyricData[i].Text.Trim(), ticks));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (fileMetaData.Count != 0)
|
|
||||||
{
|
|
||||||
// Map metaData values from LRC file to LyricMetadata properties
|
|
||||||
LyricMetadata lyricMetadata = MapMetadataValues(fileMetaData);
|
|
||||||
|
|
||||||
return new LyricDto { Metadata = lyricMetadata, Lyrics = lyricList };
|
|
||||||
}
|
|
||||||
|
|
||||||
return new LyricDto { Lyrics = lyricList };
|
return new LyricDto { Lyrics = lyricList };
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Converts metadata from an LRC file to LyricMetadata properties.
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="metaData">The metadata from the LRC file.</param>
|
|
||||||
/// <returns>A lyricMetadata object with mapped property data.</returns>
|
|
||||||
private static LyricMetadata MapMetadataValues(Dictionary<string, string> metaData)
|
|
||||||
{
|
|
||||||
LyricMetadata lyricMetadata = new();
|
|
||||||
|
|
||||||
if (metaData.TryGetValue("ar", out var artist) && !string.IsNullOrEmpty(artist))
|
|
||||||
{
|
|
||||||
lyricMetadata.Artist = artist;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (metaData.TryGetValue("al", out var album) && !string.IsNullOrEmpty(album))
|
|
||||||
{
|
|
||||||
lyricMetadata.Album = album;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (metaData.TryGetValue("ti", out var title) && !string.IsNullOrEmpty(title))
|
|
||||||
{
|
|
||||||
lyricMetadata.Title = title;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (metaData.TryGetValue("au", out var author) && !string.IsNullOrEmpty(author))
|
|
||||||
{
|
|
||||||
lyricMetadata.Author = author;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (metaData.TryGetValue("length", out var length) && !string.IsNullOrEmpty(length))
|
|
||||||
{
|
|
||||||
if (DateTime.TryParseExact(length, _acceptedTimeFormats, null, DateTimeStyles.None, out var value))
|
|
||||||
{
|
|
||||||
lyricMetadata.Length = value.TimeOfDay.Ticks;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (metaData.TryGetValue("by", out var by) && !string.IsNullOrEmpty(by))
|
|
||||||
{
|
|
||||||
lyricMetadata.By = by;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (metaData.TryGetValue("offset", out var offset) && !string.IsNullOrEmpty(offset))
|
|
||||||
{
|
|
||||||
if (int.TryParse(offset, out var value))
|
|
||||||
{
|
|
||||||
lyricMetadata.Offset = TimeSpan.FromMilliseconds(value).Ticks;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (metaData.TryGetValue("re", out var creator) && !string.IsNullOrEmpty(creator))
|
|
||||||
{
|
|
||||||
lyricMetadata.Creator = creator;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (metaData.TryGetValue("ve", out var version) && !string.IsNullOrEmpty(version))
|
|
||||||
{
|
|
||||||
lyricMetadata.Version = version;
|
|
||||||
}
|
|
||||||
|
|
||||||
return lyricMetadata;
|
|
||||||
}
|
|
||||||
|
|
||||||
private static string GetMetadataFieldName(string metaDataRow, int index)
|
|
||||||
{
|
|
||||||
var metadataFieldName = metaDataRow.AsSpan(1, index - 1).Trim();
|
|
||||||
return metadataFieldName.IsEmpty ? string.Empty : metadataFieldName.ToString();
|
|
||||||
}
|
|
||||||
|
|
||||||
private static string GetMetadataValue(string metaDataRow, int index)
|
|
||||||
{
|
|
||||||
var metadataValue = metaDataRow.AsSpan(index + 1, metaDataRow.Length - index - 2).Trim();
|
|
||||||
return metadataValue.IsEmpty ? string.Empty : metadataValue.ToString();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user