mirror of
https://github.com/SubtitleEdit/subtitleedit.git
synced 2024-11-24 04:02:36 +01:00
254 lines
11 KiB
C#
254 lines
11 KiB
C#
using Nikse.SubtitleEdit.Core.Enums;
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.Text;
|
|
using System.Text.RegularExpressions;
|
|
|
|
namespace Nikse.SubtitleEdit.Core.SubtitleFormats
|
|
{
|
|
/// <summary>
|
|
/// LRC is a format that synchronizes song lyrics with an audio/video file, [mm:ss.xx] where mm is minutes, ss is seconds and xx is hundredths of a second.
|
|
///
|
|
/// http://wiki.nicksoft.info/specifications:lrc-file
|
|
///
|
|
/// Tags:
|
|
/// [al:''Album where the song is from'']
|
|
/// [ar:''Lyrics artist'']
|
|
/// [by:''Creator of the LRC file'']
|
|
/// [offset:''+/- Overall timestamp adjustment in milliseconds, + shifts time up, - shifts down'']
|
|
/// [re:''The player or editor that creates LRC file'']
|
|
/// [ti:''Lyrics(song) title'']
|
|
/// [ve:''version of program'']
|
|
/// </summary>
|
|
public class Lrc : SubtitleFormat
|
|
{
|
|
private static readonly Regex RegexTimeCodes = new Regex(@"^\[\d+:\d\d\.\d\d\].*$", RegexOptions.Compiled);
|
|
|
|
public override string Extension => ".lrc";
|
|
|
|
public override string Name => "LRC Lyrics";
|
|
|
|
public override bool IsMine(List<string> lines, string fileName)
|
|
{
|
|
var subtitle = new Subtitle();
|
|
LoadSubtitle(subtitle, lines, fileName);
|
|
|
|
if (subtitle.Paragraphs.Count > 4)
|
|
{
|
|
bool allStartWithNumber = true;
|
|
foreach (Paragraph p in subtitle.Paragraphs)
|
|
{
|
|
if (p.Text.Length > 1 && !Utilities.IsInteger(p.Text.Substring(0, 2)))
|
|
{
|
|
allStartWithNumber = false;
|
|
break;
|
|
}
|
|
}
|
|
if (allStartWithNumber)
|
|
return false;
|
|
}
|
|
if (subtitle.Paragraphs.Count > _errorCount)
|
|
{
|
|
if (new UnknownSubtitle33().IsMine(lines, fileName) || new UnknownSubtitle36().IsMine(lines, fileName) || new TMPlayer().IsMine(lines, fileName))
|
|
return false;
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
public override string ToText(Subtitle subtitle, string title)
|
|
{
|
|
var sb = new StringBuilder();
|
|
if (!string.IsNullOrEmpty(subtitle.Header) && (subtitle.Header.Contains("[ar:") || subtitle.Header.Contains("[ti:") || subtitle.Header.Contains("[by:") || subtitle.Header.Contains("[id:")))
|
|
sb.AppendLine(subtitle.Header.Trim());
|
|
else if (!string.IsNullOrEmpty(title))
|
|
sb.AppendLine("[ti:" + title.Replace("[", string.Empty).Replace("]", string.Empty) + "]");
|
|
|
|
const string timeCodeFormat = "[{0:00}:{1:00}.{2:00}]{3}";
|
|
for (int i = 0; i < subtitle.Paragraphs.Count; i++)
|
|
{
|
|
Paragraph p = subtitle.Paragraphs[i];
|
|
Paragraph next = subtitle.GetParagraphOrDefault(i + 1);
|
|
|
|
string text = HtmlUtil.RemoveHtmlTags(p.Text);
|
|
text = text.Replace(Environment.NewLine, " ");
|
|
sb.AppendLine(string.Format(timeCodeFormat, p.StartTime.Hours * 60 + p.StartTime.Minutes, p.StartTime.Seconds, (int)Math.Round(p.StartTime.Milliseconds / 10.0), text));
|
|
|
|
if (next == null || next.StartTime.TotalMilliseconds - p.EndTime.TotalMilliseconds > 100)
|
|
{
|
|
var tc = new TimeCode(p.EndTime.TotalMilliseconds);
|
|
sb.AppendLine(string.Format(timeCodeFormat, tc.Hours * 60 + tc.Minutes, tc.Seconds, (int)Math.Round(tc.Milliseconds / 10.0), string.Empty));
|
|
}
|
|
}
|
|
return sb.ToString().Trim();
|
|
}
|
|
|
|
public override void LoadSubtitle(Subtitle subtitle, List<string> lines, string fileName)
|
|
{ //[01:05.99]I've been walking in the same way as I do
|
|
_errorCount = 0;
|
|
var offsetInMilliseconds = 0.0d;
|
|
var header = new StringBuilder();
|
|
char[] splitChars = { ':', '.' };
|
|
foreach (string line in lines)
|
|
{
|
|
if (line.StartsWith('[') && RegexTimeCodes.Match(line).Success)
|
|
{
|
|
string s = line.Substring(1, 8);
|
|
string[] parts = s.Split(splitChars, StringSplitOptions.RemoveEmptyEntries);
|
|
if (parts.Length == 3)
|
|
{
|
|
try
|
|
{
|
|
int minutes = int.Parse(parts[0]);
|
|
int seconds = int.Parse(parts[1]);
|
|
int milliseconds = int.Parse(parts[2]) * 10;
|
|
string text = line.Remove(0, 9).Trim().TrimStart(']').Trim();
|
|
var start = new TimeCode(0, minutes, seconds, milliseconds);
|
|
var p = new Paragraph(start, new TimeCode(), text);
|
|
subtitle.Paragraphs.Add(p);
|
|
}
|
|
catch
|
|
{
|
|
_errorCount++;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
_errorCount++;
|
|
}
|
|
}
|
|
else if (line.StartsWith("[ar:", StringComparison.Ordinal)) // [ar:Lyrics artist]
|
|
{
|
|
if (subtitle.Paragraphs.Count < 1)
|
|
header.AppendLine(line);
|
|
}
|
|
else if (line.StartsWith("[id:", StringComparison.Ordinal)) // [ar:Lyrics artist]
|
|
{
|
|
if (subtitle.Paragraphs.Count < 1)
|
|
header.AppendLine(line);
|
|
}
|
|
else if (line.StartsWith("[al:", StringComparison.Ordinal)) // [al:Album where the song is from]
|
|
{
|
|
if (subtitle.Paragraphs.Count < 1)
|
|
header.AppendLine(line);
|
|
}
|
|
else if (line.StartsWith("[ti:", StringComparison.Ordinal)) // [ti:Lyrics (song) title]
|
|
{
|
|
if (subtitle.Paragraphs.Count < 1)
|
|
header.AppendLine(line);
|
|
}
|
|
else if (line.StartsWith("[au:", StringComparison.Ordinal)) // [au:Creator of the Songtext]
|
|
{
|
|
if (subtitle.Paragraphs.Count < 1)
|
|
header.AppendLine(line);
|
|
}
|
|
else if (line.StartsWith("[length:", StringComparison.Ordinal)) // [length:How long the song is]
|
|
{
|
|
if (subtitle.Paragraphs.Count < 1)
|
|
header.AppendLine(line);
|
|
}
|
|
else if (line.StartsWith("[offset:", StringComparison.Ordinal)) // [length:How long the song is]
|
|
{
|
|
var temp = line.Replace("[offset:", string.Empty).Replace("]", string.Empty).Replace("'", string.Empty).RemoveChar(' ').TrimEnd();
|
|
double d;
|
|
if (double.TryParse(temp, out d))
|
|
{
|
|
offsetInMilliseconds = d;
|
|
}
|
|
}
|
|
else if (line.StartsWith("[by:", StringComparison.Ordinal)) // [by:Creator of the LRC file]
|
|
{
|
|
if (subtitle.Paragraphs.Count < 1)
|
|
header.AppendLine(line);
|
|
}
|
|
else if (!string.IsNullOrWhiteSpace(line))
|
|
{
|
|
if (subtitle.Paragraphs.Count < 1)
|
|
header.AppendLine(line);
|
|
_errorCount++;
|
|
}
|
|
else if (subtitle.Paragraphs.Count < 1)
|
|
{
|
|
header.AppendLine(line);
|
|
}
|
|
}
|
|
subtitle.Header = header.ToString();
|
|
|
|
int max = subtitle.Paragraphs.Count;
|
|
for (int i = 0; i < max; i++)
|
|
{
|
|
Paragraph p = subtitle.Paragraphs[i];
|
|
while (RegexTimeCodes.Match(p.Text).Success)
|
|
{
|
|
string s = p.Text.Substring(1, 8);
|
|
p.Text = p.Text.Remove(0, 10).Trim();
|
|
string[] parts = s.Split(splitChars, StringSplitOptions.RemoveEmptyEntries);
|
|
try
|
|
{
|
|
int minutes = int.Parse(parts[0]);
|
|
int seconds = int.Parse(parts[1]);
|
|
int milliseconds = int.Parse(parts[2]) * 10;
|
|
string text = GetTextAfterTimeCodes(p.Text);
|
|
var start = new TimeCode(0, minutes, seconds, milliseconds);
|
|
var newParagraph = new Paragraph(start, new TimeCode(), text);
|
|
subtitle.Paragraphs.Add(newParagraph);
|
|
}
|
|
catch
|
|
{
|
|
_errorCount++;
|
|
}
|
|
}
|
|
}
|
|
|
|
subtitle.Sort(SubtitleSortCriteria.StartTime);
|
|
|
|
int index = 0;
|
|
foreach (Paragraph p in subtitle.Paragraphs)
|
|
{
|
|
p.Text = Utilities.AutoBreakLine(p.Text);
|
|
Paragraph next = subtitle.GetParagraphOrDefault(index + 1);
|
|
if (next != null)
|
|
{
|
|
if (string.IsNullOrEmpty(next.Text))
|
|
{
|
|
p.EndTime = new TimeCode(next.StartTime.TotalMilliseconds);
|
|
}
|
|
else
|
|
{
|
|
p.EndTime.TotalMilliseconds = next.StartTime.TotalMilliseconds - Configuration.Settings.General.MinimumMillisecondsBetweenLines;
|
|
}
|
|
if (p.Duration.TotalMilliseconds > Configuration.Settings.General.SubtitleMaximumDisplayMilliseconds)
|
|
{
|
|
double duration = Configuration.Settings.General.SubtitleMaximumDisplayMilliseconds;
|
|
p.EndTime = new TimeCode(p.StartTime.TotalMilliseconds + duration);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
double duration = Utilities.GetOptimalDisplayMilliseconds(p.Text, 16) + 1500;
|
|
p.EndTime = new TimeCode(p.StartTime.TotalMilliseconds + duration);
|
|
}
|
|
index++;
|
|
}
|
|
subtitle.RemoveEmptyLines();
|
|
subtitle.Renumber();
|
|
if (Math.Abs(offsetInMilliseconds) > 0.01)
|
|
{
|
|
foreach (var paragraph in subtitle.Paragraphs)
|
|
{
|
|
paragraph.StartTime.TotalMilliseconds += offsetInMilliseconds;
|
|
paragraph.EndTime.TotalMilliseconds += offsetInMilliseconds;
|
|
}
|
|
}
|
|
}
|
|
|
|
private static string GetTextAfterTimeCodes(string s)
|
|
{
|
|
while (RegexTimeCodes.IsMatch(s))
|
|
s = s.Remove(0, 10).Trim();
|
|
return s;
|
|
}
|
|
|
|
}
|
|
}
|