mirror of
https://github.com/SubtitleEdit/subtitleedit.git
synced 2024-11-25 04:33:04 +01:00
130 lines
4.7 KiB
C#
130 lines
4.7 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.Linq;
|
|
|
|
namespace Nikse.SubtitleEdit.Core
|
|
{
|
|
public class TextSplit
|
|
{
|
|
private readonly List<TextSplitResult> _splits;
|
|
private readonly List<TextSplitResult> _allSplits;
|
|
private readonly int _singleLineMaxLength;
|
|
private const string EndLineChars = ".!?…؟";
|
|
private const string Commas = ",،";
|
|
|
|
public TextSplit(string text, int singleLineMaxLength, string language)
|
|
{
|
|
_singleLineMaxLength = singleLineMaxLength;
|
|
|
|
// create list with all split possibilities
|
|
_splits = new List<TextSplitResult>();
|
|
_allSplits = new List<TextSplitResult>();
|
|
for (int i = 1; i < text.Length - 1; i++)
|
|
{
|
|
if (text[i] == ' ')
|
|
{
|
|
var l1 = text.Substring(0, i).Trim();
|
|
var l2 = text.Substring(i + 1).Trim();
|
|
_allSplits.Add(new TextSplitResult(new List<string> { l1, l2 }));
|
|
if (Utilities.CanBreak(text, i, language))
|
|
{
|
|
_splits.Add(new TextSplitResult(new List<string> { l1, l2 }));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
public string AutoBreak(bool breakEarlyDialog, bool breakEarlyLineEnding, bool breakEarlyComma, bool usePixelWidth)
|
|
{
|
|
if (breakEarlyDialog)
|
|
{
|
|
var s = GetBestDialog();
|
|
if (s != null)
|
|
{
|
|
return s;
|
|
}
|
|
}
|
|
if (breakEarlyLineEnding)
|
|
{
|
|
var s = GetBestEnding(EndLineChars);
|
|
if (s != null)
|
|
{
|
|
return s;
|
|
}
|
|
}
|
|
if (breakEarlyComma)
|
|
{
|
|
var s = GetBestEnding(Commas);
|
|
if (s != null)
|
|
{
|
|
return s;
|
|
}
|
|
}
|
|
if (usePixelWidth)
|
|
{
|
|
var s = GetBestPixelSplit();
|
|
if (s != null)
|
|
{
|
|
return s;
|
|
}
|
|
}
|
|
|
|
return GetBestLengthSplit();
|
|
}
|
|
|
|
private string GetBestDialog()
|
|
{
|
|
var orderedArray = _splits.Where(IsDialog).OrderBy(p => p.DiffFromAveragePixel());
|
|
var best = orderedArray.FirstOrDefault();
|
|
return best != null ? string.Join(Environment.NewLine, best.Lines) : null;
|
|
}
|
|
|
|
private string GetBestEnding(string chars)
|
|
{
|
|
var orderedArray = _splits.Where(p => p.IsLineLengthOkay(_singleLineMaxLength) && EndsWith(p, chars)).OrderBy(p => p.DiffFromAveragePixel());
|
|
var best = orderedArray.FirstOrDefault();
|
|
return best != null ? string.Join(Environment.NewLine, best.Lines) : null;
|
|
}
|
|
|
|
private string GetBestPixelSplit()
|
|
{
|
|
var orderedArray = _splits.Where(p => p.IsLineLengthOkay(_singleLineMaxLength)).OrderBy(p => p.DiffFromAveragePixel());
|
|
var best = orderedArray.FirstOrDefault();
|
|
if (best != null && !best.IsBottomHeavy && Configuration.Settings.Tools.AutoBreakUsePixelWidth && Configuration.Settings.Tools.AutoBreakPreferBottomHeavy)
|
|
{
|
|
orderedArray = _splits.Where(p => p.IsLineLengthOkay(_singleLineMaxLength)).OrderBy(p => p.DiffFromAveragePixelBottomHeavy());
|
|
best = orderedArray.FirstOrDefault() ?? best;
|
|
}
|
|
return best != null ? string.Join(Environment.NewLine, best.Lines) : null;
|
|
}
|
|
|
|
private string GetBestLengthSplit()
|
|
{
|
|
var orderedArray = _splits.OrderBy(p => p.DiffFromAverage());
|
|
var best = orderedArray.FirstOrDefault();
|
|
if (best != null)
|
|
{
|
|
return string.Join(Environment.NewLine, best.Lines);
|
|
}
|
|
|
|
orderedArray = _allSplits.OrderBy(p => p.DiffFromAverage());
|
|
best = orderedArray.FirstOrDefault();
|
|
return best != null ? string.Join(Environment.NewLine, best.Lines) : null;
|
|
}
|
|
|
|
private static bool EndsWith(TextSplitResult textSplitResult, string chars)
|
|
{
|
|
var line1 = textSplitResult.Lines[0].TrimEnd('"', '\'');
|
|
return line1.Length > 0 && chars.Contains(line1[line1.Length - 1]);
|
|
}
|
|
|
|
private static bool IsDialog(TextSplitResult textSplitResult)
|
|
{
|
|
var line1 = textSplitResult.Lines[0].TrimEnd('"', '\'');
|
|
var line2 = textSplitResult.Lines[1].TrimStart('"', '\'');
|
|
return line2.Length > 0 && line1.Length > 0 &&
|
|
line2.StartsWith('-') && EndLineChars.Contains(line1[line1.Length - 1]);
|
|
}
|
|
}
|
|
}
|