// Copyright (c) 2007-2017 ppy Pty Ltd . // Licensed under the MIT Licence - https://raw.githubusercontent.com/ppy/osu/master/LICENCE using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.Text.RegularExpressions; using System.Threading.Tasks; namespace osu.Game.Online.Chat { public static class MessageFormatter { // [[Performance Points]] -> wiki:Performance Points (https://osu.ppy.sh/wiki/Performance_Points) private static Regex wikiRegex = new Regex(@"\[\[([^\]]+)\]\]"); // (test)[https://osu.ppy.sh/b/1234] -> test (https://osu.ppy.sh/b/1234) private static Regex oldLinkRegex = new Regex(@"\(([^\)]*)\)\[([a-z]+://[^ ]+)\]"); // [https://osu.ppy.sh/b/1234 Beatmap [Hard] (poop)] -> Beatmap [hard] (poop) (https://osu.ppy.sh/b/1234) private static Regex newLinkRegex = new Regex(@"\[([a-z]+://[^ ]+) ([^\[\]]*(((?\[)[^\[\]]*)+((?\])[^\[\]]*)+)*(?(open)(?!)))\]"); // advanced, RFC-compatible regular expression that matches any possible URL, *but* allows certain invalid characters that are widely used // This is in the format (, [optional]): // http[s]://.[:port][/path][?query][#fragment] private static Regex advancedLinkRegex = new Regex(@"(?\([^)]*)?" + @"(?https?:\/\/" + @"(?(?:[a-z0-9]\.|[a-z0-9][a-z0-9-]*[a-z0-9]\.)*[a-z][a-z0-9-]*[a-z0-9]" + // domain, TLD @"(?::\d+)?)" + // port @"(?(?:(?:\/+(?:[a-z0-9$_\.\+!\*\',;:\(\)@&~=-]|%[0-9a-f]{2})*)*" + // path @"(?:\?(?:[a-z0-9$_\+!\*\',;:\(\)@&=\/~-]|%[0-9a-f]{2})*)?)?" + // query @"(?:#(?:[a-z0-9$_\+!\*\',;:\(\)@&=\/~-]|%[0-9a-f]{2})*)?)?)", // fragment RegexOptions.IgnoreCase); // 00:00:000 (1,2,3) - test private static Regex timeRegex = new Regex(@"\d\d:\d\d:\d\d\d? [^-]*"); // #osu private static Regex channelRegex = new Regex(@"#[a-zA-Z]+[a-zA-Z0-9]+"); // Unicode emojis private static Regex emojiRegex = new Regex(@"(\uD83D[\uDC00-\uDE4F])"); private static void handleMatches(Regex regex, string display, string link, MessageFormatterResult result, int startIndex = 0) { int captureOffset = 0; foreach (Match m in regex.Matches(result.Text, startIndex)) { var index = m.Index - captureOffset; var displayText = string.Format(display, m.Groups[0], m.Groups.Count > 1 ? m.Groups[1].Value : "", m.Groups.Count > 2 ? m.Groups[2].Value : "").Trim(); var linkText = string.Format(link, m.Groups[0], m.Groups.Count > 1 ? m.Groups[1].Value : "", m.Groups.Count > 2 ? m.Groups[2].Value : "").Trim(); if (displayText.Length == 0 || linkText.Length == 0) continue; // Check for encapsulated links if (result.Links.Find(l => (l.Index <= index && l.Index + l.Length >= index + m.Length) || index <= l.Index && index + m.Length >= l.Index + l.Length) == null) { result.Text = result.Text.Remove(index, m.Length).Insert(index, displayText); //since we just changed the line display text, offset any already processed links. result.Links.ForEach(l => l.Index -= l.Index > index ? m.Length - displayText.Length : 0); result.Links.Add(new Link(linkText, index, displayText.Length)); //adjust the offset for processing the current matches group. captureOffset += (m.Length - displayText.Length); } } } private static void handleAdvanced(Regex regex, MessageFormatterResult result, int startIndex = 0) { foreach (Match m in regex.Matches(result.Text, startIndex)) { var index = m.Index; var prefix = m.Groups["paren"].Value; var link = m.Groups["link"].Value; var indexLength = link.Length; if (!String.IsNullOrEmpty(prefix)) { index += prefix.Length; if (link.EndsWith(")")) { indexLength = indexLength - 1; link = link.Remove(link.Length - 1); } } result.Links.Add(new Link(link, index, indexLength)); } } private static MessageFormatterResult format(string toFormat, int startIndex = 0, int space = 3) { var result = new MessageFormatterResult(toFormat); // handle the [link display] format handleMatches(newLinkRegex, "{2}", "{1}", result, startIndex); // handle the ()[] link format handleMatches(oldLinkRegex, "{1}", "{2}", result, startIndex); // handle wiki links handleMatches(wikiRegex, "wiki:{1}", "https://osu.ppy.sh/wiki/{1}", result, startIndex); // handle bare links handleAdvanced(advancedLinkRegex, result, startIndex); // handle editor times handleMatches(timeRegex, "{0}", "osu://edit/{0}", result, startIndex); // handle channels handleMatches(channelRegex, "{0}", "osu://chan/{0}", result, startIndex); var empty = ""; while (space-- > 0) empty += "\0"; handleMatches(emojiRegex, empty, "{0}", result, startIndex); return result; } public static Message FormatMessage(Message inputMessage) { var result = format(inputMessage.Content); var formatted = inputMessage; formatted.Content = result.Text; // Sometimes, regex matches are not in order result.Links.Sort(); formatted.Links = result.Links; return formatted; } public class MessageFormatterResult { public List Links = new List(); public string Text; public string OriginalText; public MessageFormatterResult(string text) { OriginalText = Text = text; } } public class Link : IComparable { public string Url; public int Index; public int Length; public Link(string url, int startIndex, int length) { Url = url; Index = startIndex; Length = length; } public int CompareTo(Link otherLink) => Index > otherLink.Index ? 1 : -1; } } }