From be3336c2280013927ad917692f7f58c12ee74b64 Mon Sep 17 00:00:00 2001 From: ScrubN <72096833+ScrubN@users.noreply.github.com> Date: Thu, 9 Nov 2023 22:32:51 -0500 Subject: [PATCH 1/6] Move FilenameService to Core --- .../Tools}/FilenameService.cs | 5 ++--- TwitchDownloaderWPF/PageChatUpdate.xaml.cs | 1 + TwitchDownloaderWPF/WindowQueueOptions.xaml.cs | 1 + 3 files changed, 4 insertions(+), 3 deletions(-) rename {TwitchDownloaderWPF/Services => TwitchDownloaderCore/Tools}/FilenameService.cs (95%) diff --git a/TwitchDownloaderWPF/Services/FilenameService.cs b/TwitchDownloaderCore/Tools/FilenameService.cs similarity index 95% rename from TwitchDownloaderWPF/Services/FilenameService.cs rename to TwitchDownloaderCore/Tools/FilenameService.cs index 5cb3a678..4523bfb1 100644 --- a/TwitchDownloaderWPF/Services/FilenameService.cs +++ b/TwitchDownloaderCore/Tools/FilenameService.cs @@ -2,9 +2,8 @@ using System.IO; using System.Text; using System.Text.RegularExpressions; -using TwitchDownloaderCore.Tools; -namespace TwitchDownloaderWPF.Services +namespace TwitchDownloaderCore.Tools { public static class FilenameService { @@ -22,7 +21,7 @@ private static string[] GetTemplateSubfolders(ref string fullPath) return returnString; } - internal static string GetFilename(string template, string title, string id, DateTime date, string channel, TimeSpan cropStart, TimeSpan cropEnd, string viewCount, string game) + public static string GetFilename(string template, string title, string id, DateTime date, string channel, TimeSpan cropStart, TimeSpan cropEnd, string viewCount, string game) { var videoLength = cropEnd - cropStart; diff --git a/TwitchDownloaderWPF/PageChatUpdate.xaml.cs b/TwitchDownloaderWPF/PageChatUpdate.xaml.cs index 706ac3f3..38cb7d5d 100644 --- a/TwitchDownloaderWPF/PageChatUpdate.xaml.cs +++ b/TwitchDownloaderWPF/PageChatUpdate.xaml.cs @@ -12,6 +12,7 @@ using TwitchDownloaderCore; using TwitchDownloaderCore.Chat; using TwitchDownloaderCore.Options; +using TwitchDownloaderCore.Tools; using TwitchDownloaderCore.TwitchObjects; using TwitchDownloaderCore.TwitchObjects.Gql; using TwitchDownloaderWPF.Properties; diff --git a/TwitchDownloaderWPF/WindowQueueOptions.xaml.cs b/TwitchDownloaderWPF/WindowQueueOptions.xaml.cs index db63129b..95db422f 100644 --- a/TwitchDownloaderWPF/WindowQueueOptions.xaml.cs +++ b/TwitchDownloaderWPF/WindowQueueOptions.xaml.cs @@ -7,6 +7,7 @@ using System.Windows.Media; using TwitchDownloaderCore.Chat; using TwitchDownloaderCore.Options; +using TwitchDownloaderCore.Tools; using TwitchDownloaderWPF.Properties; using TwitchDownloaderWPF.Services; using TwitchDownloaderWPF.TwitchTasks; From 479b3d35402582dde7f7f4d5b1ea23f972a91f81 Mon Sep 17 00:00:00 2001 From: ScrubN <72096833+ScrubN@users.noreply.github.com> Date: Thu, 9 Nov 2023 22:38:34 -0500 Subject: [PATCH 2/6] Reduce filename generator memory footprint --- .../Extensions/StringExtensions.cs | 43 +++++++++ TwitchDownloaderCore/Tools/FilenameService.cs | 90 +++++-------------- 2 files changed, 63 insertions(+), 70 deletions(-) create mode 100644 TwitchDownloaderCore/Extensions/StringExtensions.cs diff --git a/TwitchDownloaderCore/Extensions/StringExtensions.cs b/TwitchDownloaderCore/Extensions/StringExtensions.cs new file mode 100644 index 00000000..4b5cf129 --- /dev/null +++ b/TwitchDownloaderCore/Extensions/StringExtensions.cs @@ -0,0 +1,43 @@ +using System; + +namespace TwitchDownloaderCore.Extensions +{ + public static class StringExtensions + { + public static string ReplaceAny(this string str, ReadOnlySpan oldChars, char newChar) + { + if (string.IsNullOrWhiteSpace(str)) + { + return str; + } + + var index = str.AsSpan().IndexOfAny(oldChars); + if (index == -1) + { + return str; + } + + const ushort MAX_STACK_SIZE = 512; + var span = str.Length <= MAX_STACK_SIZE + ? stackalloc char[str.Length] + : str.ToCharArray(); + + // Unfortunately this cannot be inlined with the previous statement because a ternary is required for the stackalloc to compile + if (str.Length <= MAX_STACK_SIZE) + str.CopyTo(span); + + var tempSpan = span; + do + { + tempSpan[index] = newChar; + tempSpan = tempSpan[(index + 1)..]; + + index = tempSpan.IndexOfAny(oldChars); + if (index == -1) + break; + } while (true); + + return span.ToString(); + } + } +} \ No newline at end of file diff --git a/TwitchDownloaderCore/Tools/FilenameService.cs b/TwitchDownloaderCore/Tools/FilenameService.cs index 4523bfb1..1e45a9e3 100644 --- a/TwitchDownloaderCore/Tools/FilenameService.cs +++ b/TwitchDownloaderCore/Tools/FilenameService.cs @@ -2,6 +2,7 @@ using System.IO; using System.Text; using System.Text.RegularExpressions; +using TwitchDownloaderCore.Extensions; namespace TwitchDownloaderCore.Tools { @@ -40,81 +41,25 @@ public static string GetFilename(string template, string title, string id, DateT if (template.Contains("{date_custom=")) { var dateRegex = new Regex("{date_custom=\"(.*)\"}"); - var dateDone = false; - while (!dateDone) - { - var dateMatch = dateRegex.Match(stringBuilder.ToString()); - if (dateMatch.Success) - { - var formatString = dateMatch.Groups[1].Value; - stringBuilder.Remove(dateMatch.Groups[0].Index, dateMatch.Groups[0].Length); - stringBuilder.Insert(dateMatch.Groups[0].Index, RemoveInvalidFilenameChars(date.ToString(formatString))); - } - else - { - dateDone = true; - } - } + ReplaceCustomWithFormattable(stringBuilder, dateRegex, date); } if (template.Contains("{crop_start_custom=")) { var cropStartRegex = new Regex("{crop_start_custom=\"(.*)\"}"); - var cropStartDone = false; - while (!cropStartDone) - { - var cropStartMatch = cropStartRegex.Match(stringBuilder.ToString()); - if (cropStartMatch.Success) - { - var formatString = cropStartMatch.Groups[1].Value; - stringBuilder.Remove(cropStartMatch.Groups[0].Index, cropStartMatch.Groups[0].Length); - stringBuilder.Insert(cropStartMatch.Groups[0].Index, RemoveInvalidFilenameChars(cropStart.ToString(formatString))); - } - else - { - cropStartDone = true; - } - } + ReplaceCustomWithFormattable(stringBuilder, cropStartRegex, cropStart); } if (template.Contains("{crop_end_custom=")) { var cropEndRegex = new Regex("{crop_end_custom=\"(.*)\"}"); - var cropEndDone = false; - while (!cropEndDone) - { - var cropEndMatch = cropEndRegex.Match(stringBuilder.ToString()); - if (cropEndMatch.Success) - { - var formatString = cropEndMatch.Groups[1].Value; - stringBuilder.Remove(cropEndMatch.Groups[0].Index, cropEndMatch.Groups[0].Length); - stringBuilder.Insert(cropEndMatch.Groups[0].Index, RemoveInvalidFilenameChars(cropEnd.ToString(formatString))); - } - else - { - cropEndDone = true; - } - } + ReplaceCustomWithFormattable(stringBuilder, cropEndRegex, cropEnd); } if (template.Contains("{length_custom=")) { var lengthRegex = new Regex("{length_custom=\"(.*)\"}"); - var lengthDone = false; - while (!lengthDone) - { - var lengthMatch = lengthRegex.Match(stringBuilder.ToString()); - if (lengthMatch.Success) - { - var formatString = lengthMatch.Groups[1].Value; - stringBuilder.Remove(lengthMatch.Groups[0].Index, lengthMatch.Groups[0].Length); - stringBuilder.Insert(lengthMatch.Groups[0].Index, RemoveInvalidFilenameChars(videoLength.ToString(formatString))); - } - else - { - lengthDone = true; - } - } + ReplaceCustomWithFormattable(stringBuilder, lengthRegex, videoLength); } var fileName = stringBuilder.ToString(); @@ -122,19 +67,24 @@ public static string GetFilename(string template, string title, string id, DateT return Path.Combine(Path.Combine(additionalSubfolders), RemoveInvalidFilenameChars(fileName)); } - private static string RemoveInvalidFilenameChars(string filename) + private static void ReplaceCustomWithFormattable(StringBuilder sb, Regex regex, IFormattable formattable, IFormatProvider formatProvider = null) { - if (string.IsNullOrWhiteSpace(filename)) + do { - return filename; - } - - if (filename.IndexOfAny(Path.GetInvalidFileNameChars()) == -1) - { - return filename; - } + // There's probably a better way to do this that doesn't require calling ToString() + // However we need .NET7+ for span support in the regex matcher. + var match = regex.Match(sb.ToString()); + if (!match.Success) + break; - return string.Join('_', filename.Split(Path.GetInvalidFileNameChars())); + var formatString = match.Groups[1].Value; + sb.Remove(match.Groups[0].Index, match.Groups[0].Length); + sb.Insert(match.Groups[0].Index, RemoveInvalidFilenameChars(formattable.ToString(formatString, formatProvider))); + } while (true); } + + private static readonly char[] FilenameInvalidChars = Path.GetInvalidFileNameChars(); + + private static string RemoveInvalidFilenameChars(string filename) => filename.ReplaceAny(FilenameInvalidChars, '_'); } } \ No newline at end of file From 1c392139322889e94fe42e7ea119e7094d3b3c75 Mon Sep 17 00:00:00 2001 From: ScrubN <72096833+ScrubN@users.noreply.github.com> Date: Thu, 9 Nov 2023 22:39:00 -0500 Subject: [PATCH 3/6] Fix game name not being sanitized --- TwitchDownloaderCore/Tools/FilenameService.cs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/TwitchDownloaderCore/Tools/FilenameService.cs b/TwitchDownloaderCore/Tools/FilenameService.cs index 1e45a9e3..408116a1 100644 --- a/TwitchDownloaderCore/Tools/FilenameService.cs +++ b/TwitchDownloaderCore/Tools/FilenameService.cs @@ -36,7 +36,7 @@ public static string GetFilename(string template, string title, string id, DateT .Replace("{crop_end}", TimeSpanHFormat.ReusableInstance.Format(@"HH\-mm\-ss", cropEnd)) .Replace("{length}", TimeSpanHFormat.ReusableInstance.Format(@"HH\-mm\-ss", videoLength)) .Replace("{views}", viewCount) - .Replace("{game}", game); + .Replace("{game}", RemoveInvalidFilenameChars(game)); if (template.Contains("{date_custom=")) { From 412bde54eb9ecea29443cff096b762eb9c301252 Mon Sep 17 00:00:00 2001 From: ScrubN <72096833+ScrubN@users.noreply.github.com> Date: Sun, 12 Nov 2023 02:52:32 -0500 Subject: [PATCH 4/6] string.IsNullOrWhiteSpace -> string.IsNullOrEmpty --- TwitchDownloaderCore/Extensions/StringExtensions.cs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/TwitchDownloaderCore/Extensions/StringExtensions.cs b/TwitchDownloaderCore/Extensions/StringExtensions.cs index 4b5cf129..cb0fa60d 100644 --- a/TwitchDownloaderCore/Extensions/StringExtensions.cs +++ b/TwitchDownloaderCore/Extensions/StringExtensions.cs @@ -6,7 +6,7 @@ public static class StringExtensions { public static string ReplaceAny(this string str, ReadOnlySpan oldChars, char newChar) { - if (string.IsNullOrWhiteSpace(str)) + if (string.IsNullOrEmpty(str)) { return str; } From da76fccd74993ecbc2faa3749a90a32516ce288e Mon Sep 17 00:00:00 2001 From: ScrubN <72096833+ScrubN@users.noreply.github.com> Date: Sun, 12 Nov 2023 03:15:23 -0500 Subject: [PATCH 5/6] Support escaping with quote marks in both ReadOnlySpanExtensions.TryReplaceNonEscaped and TimeSpanHFormat --- .../Extensions/ReadOnlySpanExtensions.cs | 77 ++++++++++++---- TwitchDownloaderCore/Tools/TimeSpanHFormat.cs | 88 ++++++++++++++----- 2 files changed, 129 insertions(+), 36 deletions(-) diff --git a/TwitchDownloaderCore/Extensions/ReadOnlySpanExtensions.cs b/TwitchDownloaderCore/Extensions/ReadOnlySpanExtensions.cs index 5a3fcaf3..433553bd 100644 --- a/TwitchDownloaderCore/Extensions/ReadOnlySpanExtensions.cs +++ b/TwitchDownloaderCore/Extensions/ReadOnlySpanExtensions.cs @@ -4,44 +4,91 @@ namespace TwitchDownloaderCore.Extensions { public static class ReadOnlySpanExtensions { - /// Replaces all occurrences of not prepended by a backslash with . - public static bool TryReplaceNonEscaped(this ReadOnlySpan str, Span destination, out int charsWritten, char oldChar, char newChar) + /// Replaces all occurrences of not prepended by a backslash or contained within quotation marks with . + public static bool TryReplaceNonEscaped(this ReadOnlySpan str, Span destination, char oldChar, char newChar) { + const string ESCAPE_CHARS = @"\'"""; + if (destination.Length < str.Length) - { - charsWritten = 0; return false; - } str.CopyTo(destination); - charsWritten = str.Length; var firstIndex = destination.IndexOf(oldChar); - if (firstIndex == -1) - { return true; - } - firstIndex = Math.Min(firstIndex, destination.IndexOf('\\')); + var firstEscapeIndex = destination.IndexOfAny(ESCAPE_CHARS); + if (firstEscapeIndex != -1 && firstEscapeIndex < firstIndex) + firstIndex = firstEscapeIndex; - for (var i = firstIndex; i < str.Length; i++) + var lastIndex = destination.LastIndexOf(oldChar); + var lastEscapeIndex = destination.LastIndexOfAny(ESCAPE_CHARS); + if (lastEscapeIndex != -1 && lastEscapeIndex > lastIndex) + lastIndex = lastEscapeIndex; + + lastIndex++; + for (var i = firstIndex; i < lastIndex; i++) { var readChar = destination[i]; - if (readChar == '\\' && i + 1 < str.Length) + switch (readChar) + { + case '\\': + i++; + break; + case '\'': + case '\"': + { + i = FindCloseQuoteMark(destination, i, lastIndex, readChar); + + if (i == -1) + { + destination.Clear(); + return false; + } + + break; + } + default: + { + if (readChar == oldChar) + { + destination[i] = newChar; + } + + break; + } + } + } + + return true; + } + + private static int FindCloseQuoteMark(ReadOnlySpan destination, int openQuoteIndex, int endIndex, char readChar) + { + var i = openQuoteIndex + 1; + var quoteFound = false; + while (i < endIndex) + { + var readCharQuote = destination[i]; + i++; + + if (readCharQuote == '\\') { i++; continue; } - if (readChar == oldChar) + if (readCharQuote == readChar) { - destination[i] = newChar; + i--; + quoteFound = true; + break; } } - return true; + return quoteFound ? i : -1; } } } \ No newline at end of file diff --git a/TwitchDownloaderCore/Tools/TimeSpanHFormat.cs b/TwitchDownloaderCore/Tools/TimeSpanHFormat.cs index f9fbb120..531f04cb 100644 --- a/TwitchDownloaderCore/Tools/TimeSpanHFormat.cs +++ b/TwitchDownloaderCore/Tools/TimeSpanHFormat.cs @@ -49,76 +49,121 @@ public string Format(string format, TimeSpan timeSpan, IFormatProvider formatPro if (timeSpan.Days == 0) { var newFormat = format.Length <= 256 ? stackalloc char[format.Length] : new char[format.Length]; - if (!format.AsSpan().TryReplaceNonEscaped(newFormat, out var charsWritten, 'H', 'h')) + if (!format.AsSpan().TryReplaceNonEscaped(newFormat, 'H', 'h')) { - throw new Exception("Failed to generate ToString() compatible format. This should not have been possible."); + throw new FormatException($"Invalid character escaping in the format string: {format}"); } // If the format contains more than 2 sequential unescaped h's, it will throw a format exception. If so, we can fallback to our parser. if (newFormat.IndexOf("hhh") == -1) { - return HandleOtherFormats(newFormat[..charsWritten].ToString(), timeSpan, formatProvider); + return HandleOtherFormats(newFormat.ToString(), timeSpan, formatProvider); } } - var sb = new StringBuilder(format.Length); + return HandleBigHFormat(format.AsSpan(), timeSpan); + } + + private static string HandleBigHFormat(ReadOnlySpan format, TimeSpan timeSpan) + { + var formatLength = format.Length; + var sb = new StringBuilder(formatLength); var regularFormatCharStart = -1; var bigHStart = -1; - var formatSpan = format.AsSpan(); - for (var i = 0; i < formatSpan.Length; i++) + for (var i = 0; i < formatLength; i++) { - var readChar = formatSpan[i]; + var readChar = format[i]; if (readChar == 'H') { if (bigHStart == -1) - { bigHStart = i; - } if (regularFormatCharStart != -1) { - AppendRegularFormat(sb, timeSpan, format, regularFormatCharStart, i - regularFormatCharStart); + var formatEnd = i - regularFormatCharStart; + AppendRegularFormat(sb, timeSpan, format.Slice(regularFormatCharStart, formatEnd)); regularFormatCharStart = -1; } } else { if (regularFormatCharStart == -1) - { regularFormatCharStart = i; - } if (bigHStart != -1) { - AppendBigHFormat(sb, timeSpan, i - bigHStart); + var bigHCount = i - bigHStart; + AppendBigHFormat(sb, timeSpan, bigHCount); bigHStart = -1; } - // If the current char is an escape we can skip the next char - if (readChar == '\\' && i + 1 < formatSpan.Length) + switch (readChar) { - i++; + // If the current char is an escape we can skip the next char + case '\\' when i + 1 < formatLength: + i++; + continue; + // If the current char is a quote we can skip the next quote, if it exists + case '\'' when i + 1 < formatLength: + case '\"' when i + 1 < formatLength: + { + i = FindCloseQuoteMark(format, i, formatLength, readChar); + + if (i == -1) + { + throw new FormatException($"Invalid character escaping in the format string: {format}"); + } + + continue; + } } } } if (regularFormatCharStart != -1) { - AppendRegularFormat(sb, timeSpan, format, regularFormatCharStart, formatSpan.Length - regularFormatCharStart); + var formatEnd = format.Length - regularFormatCharStart; + AppendRegularFormat(sb, timeSpan, format.Slice(regularFormatCharStart, formatEnd)); } else if (bigHStart != -1) { - AppendBigHFormat(sb, timeSpan, formatSpan.Length - bigHStart); + var bigHCount = format.Length - bigHStart; + AppendBigHFormat(sb, timeSpan, bigHCount); } return sb.ToString(); } - private static void AppendRegularFormat(StringBuilder sb, TimeSpan timeSpan, string formatString, int start, int length) + private static int FindCloseQuoteMark(ReadOnlySpan format, int openQuoteIndex, int endIndex, char readChar) + { + var i = openQuoteIndex + 1; + var quoteFound = false; + while (i < endIndex) + { + var readCharQuote = format[i]; + i++; + + if (readCharQuote == '\\') + { + i++; + continue; + } + + if (readCharQuote == readChar) + { + i--; + quoteFound = true; + break; + } + } + + return quoteFound ? i : -1; + } + + private static void AppendRegularFormat(StringBuilder sb, TimeSpan timeSpan, ReadOnlySpan format) { Span destination = stackalloc char[256]; - var format = formatString.AsSpan(start, length); if (timeSpan.TryFormat(destination, out var charsWritten, format)) { @@ -132,7 +177,8 @@ private static void AppendRegularFormat(StringBuilder sb, TimeSpan timeSpan, str private static void AppendBigHFormat(StringBuilder sb, TimeSpan timeSpan, int count) { - Span destination = stackalloc char[8]; + const int TIMESPAN_MAX_HOURS_LENGTH = 9; // The maximum integer hours a TimeSpan can hold is 256204778. + Span destination = stackalloc char[TIMESPAN_MAX_HOURS_LENGTH]; Span format = stackalloc char[count]; format.Fill('0'); From 0a85a617400d2bcd3a06172cacfd9b9022ee26c5 Mon Sep 17 00:00:00 2001 From: ScrubN <72096833+ScrubN@users.noreply.github.com> Date: Sun, 12 Nov 2023 03:18:01 -0500 Subject: [PATCH 6/6] Rename TimeSpanExtensions to UrlTimeCode --- TwitchDownloaderCore/Chat/ChatJson.cs | 2 +- .../TimeSpanExtensions.cs => Tools/UrlTimeCode.cs} | 6 +++--- TwitchDownloaderWPF/PageChatDownload.xaml.cs | 2 +- TwitchDownloaderWPF/PageVodDownload.xaml.cs | 2 +- 4 files changed, 6 insertions(+), 6 deletions(-) rename TwitchDownloaderCore/{Extensions/TimeSpanExtensions.cs => Tools/UrlTimeCode.cs} (91%) diff --git a/TwitchDownloaderCore/Chat/ChatJson.cs b/TwitchDownloaderCore/Chat/ChatJson.cs index b156a184..bdfe1bc4 100644 --- a/TwitchDownloaderCore/Chat/ChatJson.cs +++ b/TwitchDownloaderCore/Chat/ChatJson.cs @@ -207,7 +207,7 @@ private static async Task UpgradeChatJson(ChatRoot chatRoot) if (chatRoot.video.duration is not null) { - chatRoot.video.length = TimeSpanExtensions.ParseTimeCode(chatRoot.video.duration).TotalSeconds; + chatRoot.video.length = UrlTimeCode.Parse(chatRoot.video.duration).TotalSeconds; chatRoot.video.end = chatRoot.video.length; chatRoot.video.duration = null; } diff --git a/TwitchDownloaderCore/Extensions/TimeSpanExtensions.cs b/TwitchDownloaderCore/Tools/UrlTimeCode.cs similarity index 91% rename from TwitchDownloaderCore/Extensions/TimeSpanExtensions.cs rename to TwitchDownloaderCore/Tools/UrlTimeCode.cs index 508ff4d0..57498951 100644 --- a/TwitchDownloaderCore/Extensions/TimeSpanExtensions.cs +++ b/TwitchDownloaderCore/Tools/UrlTimeCode.cs @@ -1,15 +1,15 @@ using System; -namespace TwitchDownloaderCore.Extensions +namespace TwitchDownloaderCore.Tools { - public static class TimeSpanExtensions + public static class UrlTimeCode { /// /// Converts the span representation of a time interval in the format of '2d21h11m9s' to its equivalent. /// /// A span containing the characters that represent the time interval to convert. /// The equivalent to the time interval contained in the span. - public static TimeSpan ParseTimeCode(ReadOnlySpan input) + public static TimeSpan Parse(ReadOnlySpan input) { var dayIndex = input.IndexOf('d'); var hourIndex = input.IndexOf('h'); diff --git a/TwitchDownloaderWPF/PageChatDownload.xaml.cs b/TwitchDownloaderWPF/PageChatDownload.xaml.cs index 08fa9ac7..4ed52a29 100644 --- a/TwitchDownloaderWPF/PageChatDownload.xaml.cs +++ b/TwitchDownloaderWPF/PageChatDownload.xaml.cs @@ -137,7 +137,7 @@ private async Task GetVideoInfo() var urlTimeCodeMatch = TwitchRegex.UrlTimeCode.Match(textUrl.Text); if (urlTimeCodeMatch.Success) { - var time = TimeSpanExtensions.ParseTimeCode(urlTimeCodeMatch.ValueSpan); + var time = UrlTimeCode.Parse(urlTimeCodeMatch.ValueSpan); checkCropStart.IsChecked = true; numStartHour.Value = time.Hours; numStartMinute.Value = time.Minutes; diff --git a/TwitchDownloaderWPF/PageVodDownload.xaml.cs b/TwitchDownloaderWPF/PageVodDownload.xaml.cs index 01cd107a..98717176 100644 --- a/TwitchDownloaderWPF/PageVodDownload.xaml.cs +++ b/TwitchDownloaderWPF/PageVodDownload.xaml.cs @@ -146,7 +146,7 @@ private async Task GetVideoInfo() var urlTimeCodeMatch = TwitchRegex.UrlTimeCode.Match(textUrl.Text); if (urlTimeCodeMatch.Success) { - var time = TimeSpanExtensions.ParseTimeCode(urlTimeCodeMatch.ValueSpan); + var time = UrlTimeCode.Parse(urlTimeCodeMatch.ValueSpan); checkStart.IsChecked = true; numStartHour.Value = time.Hours; numStartMinute.Value = time.Minutes;