using ChatTwo.Code; using ChatTwo.Util; using Dalamud.Game.Text.SeStringHandling; using Dalamud.Game.Text.SeStringHandling.Payloads; using System.Text.RegularExpressions; using LiteDB; namespace ChatTwo; internal class SortCode { internal ChatType Type { get; } internal ChatSource Source { get; } [BsonCtor] // Used by LegacyMessageImporter public SortCode(ChatType type, ChatSource source) { Type = type; Source = source; } internal SortCode(uint raw) { Type = (ChatType)(raw >> 16); Source = (ChatSource)(raw & 0xFFFF); } internal uint Encode() { return ((uint) Type << 16) | (uint) Source; } private bool Equals(SortCode other) { return Type == other.Type && Source == other.Source; } public override bool Equals(object? obj) { if (ReferenceEquals(null, obj)) { return false; } if (ReferenceEquals(this, obj)) { return true; } return obj.GetType() == GetType() && Equals((SortCode) obj); } public override int GetHashCode() { unchecked { return ((int) Type * 397) ^ (int) Source; } } } internal class Message { internal Guid Id { get; } = Guid.NewGuid(); internal ulong Receiver { get; } internal ulong ContentId { get; set; } internal DateTimeOffset Date { get; } internal ChatCode Code { get; } internal List Sender { get; } internal List Content { get; } internal SeString SenderSource { get; } internal SeString ContentSource { get; } internal SortCode SortCode { get; } internal Guid ExtraChatChannel { get; } // Not stored in the database: internal int Hash { get; } internal Dictionary Height { get; } = new(); internal Dictionary IsVisible { get; } = new(); internal Message(ulong receiver, ChatCode code, List sender, List content, SeString senderSource, SeString contentSource) { Receiver = receiver; Date = DateTimeOffset.UtcNow; Code = code; Sender = sender; Content = ReplaceContentURLs(content); SenderSource = senderSource; ContentSource = contentSource; SortCode = new SortCode(Code.Type, Code.Source); ExtraChatChannel = ExtractExtraChatChannel(); Hash = GenerateHash(); foreach (var chunk in sender.Concat(content)) { chunk.Message = this; } } internal Message(Guid id, ulong receiver, ulong contentId, DateTimeOffset date, ChatCode code, List sender, List content, SeString senderSource, SeString contentSource, SortCode sortCode, Guid extraChatChannel) { Id = id; Receiver = receiver; ContentId = contentId; Date = date; Code = code; Sender = sender; // Don't call ReplaceContentURLs here since we're loading the message // from the database and it should already have parsed URL data. Content = content; SenderSource = senderSource; ContentSource = contentSource; SortCode = sortCode; ExtraChatChannel = extraChatChannel; Hash = GenerateHash(); foreach (var chunk in sender.Concat(content)) { chunk.Message = this; } } private int GenerateHash() { return SortCode.GetHashCode() ^ ExtraChatChannel.GetHashCode() ^ string.Join("", Sender.Select(c => c.StringValue())).GetHashCode() ^ string.Join("", Content.Select(c => c.StringValue())).GetHashCode(); } private Guid ExtractExtraChatChannel() { if (ContentSource.Payloads.Count > 0 && ContentSource.Payloads[0] is RawPayload raw) { // this does an encode and clone every time it's accessed, so cache var data = raw.Data; if (data[1] == 0x27 && data[2] == 18 && data[3] == 0x20) { return new Guid(data[4..^1]); } } return Guid.Empty; } /// /// URLRegex returns a regex object that matches URLs like: /// - https://example.com /// - http://example.com /// - www.example.com /// - https://sub.example.com /// - example.com /// - sub.example.com /// /// It matches URLs with www. or https:// prefix, and also matches URLs /// without a prefix on specific TLDs. /// private static Regex URLRegex = new( @"((https?:\/\/|www\.)[a-z0-9-]+(\.[a-z0-9-]+)*|([a-z0-9-]+(\.[a-z0-9-]+)*\.(com|net|org|co|io|app)))(:[\d]{1,5})?(\/[^\s]+)?", RegexOptions.Compiled | RegexOptions.IgnoreCase ); /// /// Finds all URL strings in all TextChunks, splits the parent TextChunk /// apart and inserts a new TextChunk with a URIPayload. /// private List ReplaceContentURLs(List content) { var newChunks = new List(); void AddChunkWithMessage(Chunk chunk) { chunk.Message = this; newChunks.Add(chunk); } foreach (var chunk in content) { // Use as is if it's not a text chunk or it already has a payload. if (chunk is not TextChunk text || chunk.Link != null) { // No need to call AddChunkWithMessage here since the chunk // already has the Message field set. newChunks.Add(chunk); continue; } // Find all URLs with the regex and insert a new TextChunk with a // URIPayload. var matches = URLRegex.Matches(text.Content); var remainderIndex = 0; foreach (Match match in matches.Cast()) { // Add the text before the URL. if (match.Index > remainderIndex) { AddChunkWithMessage(text.NewWithStyle(chunk.Source, chunk.Link, text.Content[remainderIndex..match.Index])); } // Update the remainder index. remainderIndex = match.Index + match.Length; // Create a new TextChunk with a URIPayload for the URL text. try { var link = UriPayload.ResolveURI(match.Value); AddChunkWithMessage(text.NewWithStyle(chunk.Source, link, match.Value)); } catch (UriFormatException) { Plugin.Log.Debug($"Invalid URL accepted by Regex but failed URI parsing: '{match.Value}'"); // If the URL is invalid, set the remainder index to the // beginning of the match so it'll get included in the next // regular text chunk. remainderIndex = match.Index; } } // Add the text after the last URL. if (remainderIndex < text.Content.Length) AddChunkWithMessage(text.NewWithStyle(chunk.Source, null, text.Content[remainderIndex..])); } return newChunks; } }