From c058d180ae8eaf3095c6c4bf757acdac6600c95b Mon Sep 17 00:00:00 2001 From: Toastie <toastie@toastiet0ast.com> Date: Sun, 30 Mar 2025 15:38:40 +1300 Subject: [PATCH] added prices to .nczoom .nc resized to 250x125 --- .../Modules/Games/NCanvas/NCanvasCommands.cs | 107 +++++++++++------- .../Modules/Games/NCanvas/NCanvasService.cs | 33 +++++- .../_common/Services/Impl/FontProvider.cs | 6 +- src/EllieBot/data/commandlist.json | 3 +- src/EllieBot/strings/aliases.yml | 1 + 5 files changed, 101 insertions(+), 49 deletions(-) diff --git a/src/EllieBot/Modules/Games/NCanvas/NCanvasCommands.cs b/src/EllieBot/Modules/Games/NCanvas/NCanvasCommands.cs index bcce32d..bf0e589 100644 --- a/src/EllieBot/Modules/Games/NCanvas/NCanvasCommands.cs +++ b/src/EllieBot/Modules/Games/NCanvas/NCanvasCommands.cs @@ -35,17 +35,22 @@ public partial class Games public async Task NCanvas() { var pixels = await _service.GetCanvas(); - var image = new Image<Rgba32>(_service.GetWidth(), _service.GetHeight()); + var w = _service.GetWidth(); + var h = _service.GetHeight(); + var image = new Image<Rgba32>(w * 2, h * 2); Parallel.For(0, - image.Height, + h * 2, y => { var pixelAccessor = image.DangerousGetPixelRowMemory(y); var row = pixelAccessor.Span; - for (int x = 0; x < image.Width; x++) + for (var x = 0; x < image.Width; x += 2) { - row[x] = new Rgba32(pixels[(y * image.Width) + x]); + var pi = pixels[(y / 2 * w) + x / 2]; + + row[x] = new Rgba32(pi); + row[x + 1] = new Rgba32(pi); } }); @@ -53,15 +58,15 @@ public partial class Games var hint = GetText(strs.nc_hint(prefix, _service.GetWidth(), _service.GetHeight())); await Response() - .File(stream, "ncanvas.png") - .Embed(CreateEmbed() - .WithOkColor() + .File(stream, "ncanvas.png") + .Embed(CreateEmbed() + .WithOkColor() #if GLOBAL_ELLIE .WithDescription("This is not available yet.") #endif - .WithFooter(hint) - .WithImageUrl("attachment://ncanvas.png")) - .SendAsync(); + .WithFooter(hint) + .WithImageUrl("attachment://ncanvas.png")) + .SendAsync(); } [Cmd] @@ -85,8 +90,8 @@ public partial class Games var eb = CreateEmbed() .WithOkColor() .WithImageUrl($"attachment://zoom_{position}.png") - .WithFooter($"`.ncs code color` to set. (.ncs abc green)" ); - + .WithFooter($"`.ncs code color` to set. (.ncs abc green)"); + await Response() .Embed(eb) .File(stream, $"zoom_{position}.png") @@ -106,11 +111,13 @@ public partial class Games const float fontSize = 30; var posFont = _fonts.NotoSans.CreateFont(fontSize, FontStyle.Bold); + var priceFont = _fonts.Symbola.CreateFont(25, FontStyle.Bold); + var size = TextMeasurer.MeasureSize("wwww", new TextOptions(posFont)); var scale = 100f / size.Width; if (scale < 1) posFont = _fonts.NotoSans.CreateFont(fontSize * scale, FontStyle.Bold); - var outlinePen = new SolidPen(SixLabors.ImageSharp.Color.Black, 1f); + var outlinePen = new SolidPen(SixLabors.ImageSharp.Color.Black, 0.5f); Parallel.For(0, pixels.Length, @@ -128,14 +135,26 @@ public partial class Games image.Mutate(x => { x.DrawText(new RichTextOptions(posFont) - { - HorizontalAlignment = HorizontalAlignment.Center, - VerticalAlignment = VerticalAlignment.Center, - Origin = new(startX + 50, startY + 50) - }, + { + HorizontalAlignment = HorizontalAlignment.Center, + VerticalAlignment = VerticalAlignment.Center, + Origin = new(startX + 50, startY + 30) + }, ((kwum)pix.Position).ToString().PadLeft(2, '2'), Brushes.Solid(SixLabors.ImageSharp.Color.White), outlinePen); + + x.DrawText(new RichTextOptions(priceFont) + { + HorizontalAlignment = HorizontalAlignment.Center, + VerticalAlignment = VerticalAlignment.Center, + Origin = new(startX + 50, startY + 80) + }, + // "", Brushes.Solid(SixLabors.ImageSharp.Color.White), outlinePen); + pix.Price + "💵", + // CurrencyHelper.N(pix.Price, Culture, _gcs.Data.Currency.Sign), + Brushes.Solid(SixLabors.ImageSharp.Color.White), + outlinePen); }); }); @@ -165,8 +184,8 @@ public partial class Games _gcs.Data.Currency.Sign)))); if (!await PromptUserConfirmAsync(CreateEmbed() - .WithPendingColor() - .WithDescription(prompt))) + .WithPendingColor() + .WithDescription(prompt))) { return; } @@ -193,12 +212,12 @@ public partial class Games await using var stream = await img.ToStreamAsync(); await Response() - .Embed(CreateEmbed() - .WithOkColor() - .WithDescription(GetText(strs.nc_pixel_set(Format.Code(position.ToString())))) - .WithImageUrl($"attachment://zoom_{position}.png")) - .File(stream, $"zoom_{position}.png") - .SendAsync(); + .Embed(CreateEmbed() + .WithOkColor() + .WithDescription(GetText(strs.nc_pixel_set(Format.Code(position.ToString())))) + .WithImageUrl($"attachment://zoom_{position}.png")) + .File(stream, $"zoom_{position}.png") + .SendAsync(); } [Cmd] @@ -230,18 +249,18 @@ public partial class Games var pos = new kwum(pixel.Position); await Response() - .File(stream, $"{pixel.Position}.png") - .Embed(CreateEmbed() - .WithOkColor() - .WithDescription(string.IsNullOrWhiteSpace(pixel.Text) ? string.Empty : pixel.Text) - .WithTitle(GetText(strs.nc_pixel(pos))) - .AddField(GetText(strs.nc_position), - $"{pixel.Position % _service.GetWidth()} {pixel.Position / _service.GetWidth()}", - true) - .AddField(GetText(strs.price), pixel.Price.ToString(), true) - .AddField(GetText(strs.color), "#" + new Rgba32(pixel.Color).ToHex()) - .WithImageUrl($"attachment://{pixel.Position}.png")) - .SendAsync(); + .File(stream, $"{pixel.Position}.png") + .Embed(CreateEmbed() + .WithOkColor() + .WithDescription(string.IsNullOrWhiteSpace(pixel.Text) ? string.Empty : pixel.Text) + .WithTitle(GetText(strs.nc_pixel(pos))) + .AddField(GetText(strs.nc_position), + $"{pixel.Position % _service.GetWidth()} {pixel.Position / _service.GetWidth()}", + true) + .AddField(GetText(strs.price), pixel.Price.ToString(), true) + .AddField(GetText(strs.color), "#" + new Rgba32(pixel.Color).ToHex()) + .WithImageUrl($"attachment://{pixel.Position}.png")) + .SendAsync(); } [Cmd] @@ -264,9 +283,9 @@ public partial class Games } if (!await PromptUserConfirmAsync(CreateEmbed() - .WithDescription( - "This will reset the canvas to the specified image. All prices, text and colors will be reset.\n\n" - + "Are you sure you want to continue?"))) + .WithDescription( + "This will reset the canvas to the specified image. All prices, text and colors will be reset.\n\n" + + "Are you sure you want to continue?"))) return; using var http = _http.CreateClient(); @@ -294,9 +313,9 @@ public partial class Games await _service.ResetAsync(); if (!await PromptUserConfirmAsync(CreateEmbed() - .WithDescription( - "This will delete all pixels and reset the canvas.\n\n" - + "Are you sure you want to continue?"))) + .WithDescription( + "This will delete all pixels and reset the canvas.\n\n" + + "Are you sure you want to continue?"))) return; await ctx.OkAsync(); diff --git a/src/EllieBot/Modules/Games/NCanvas/NCanvasService.cs b/src/EllieBot/Modules/Games/NCanvas/NCanvasService.cs index f76c7df..fb48ca1 100644 --- a/src/EllieBot/Modules/Games/NCanvas/NCanvasService.cs +++ b/src/EllieBot/Modules/Games/NCanvas/NCanvasService.cs @@ -20,8 +20,8 @@ public sealed class NCanvasService : INCanvasService, IReadyExecutor, IEService private readonly ICurrencyService _cs; private readonly QuestService _quests; - public const int CANVAS_WIDTH = 500; - public const int CANVAS_HEIGHT = 350; + public const int CANVAS_WIDTH = 250; + public const int CANVAS_HEIGHT = 125; public const int INITIAL_PRICE = 3; public NCanvasService( @@ -45,8 +45,35 @@ public sealed class NCanvasService : INCanvasService, IReadyExecutor, IEService await using var uow = _db.GetDbContext(); - if (await uow.GetTable<NCPixel>().CountAsyncLinqToDB() > 0) + var count = await uow.GetTable<NCPixel>().CountAsync(); + if (count == CANVAS_WIDTH * CANVAS_HEIGHT) return; + + var oldWidth = 500; + var oldHeight = 250; + if (count == oldWidth * oldHeight) + { + await uow.GetTable<NCPixel>() + .Where(x => x.Position % (oldWidth * 2) % 2 != 0 // x is odd + || (x.Position / oldWidth) % 2 != 0) // or y is odd + .DeleteAsync(); + + await uow.GetTable<NCPixel>() + .Where(x => x.Position % (oldWidth * 2) % 2 == 0 + && (x.Position / oldWidth) % 2 == 0) + .UpdateAsync(old => new() + { + Position = (old.Position % oldWidth) / 2 + + ((old.Position / oldWidth) / 2) * CANVAS_WIDTH, + Price = INITIAL_PRICE, + Text = old.Text, + OwnerId = old.OwnerId, + Color = old.Color, + }); + + return; + } + await ResetAsync(); } diff --git a/src/EllieBot/_common/Services/Impl/FontProvider.cs b/src/EllieBot/_common/Services/Impl/FontProvider.cs index e5d51b5..e9cdbbb 100644 --- a/src/EllieBot/_common/Services/Impl/FontProvider.cs +++ b/src/EllieBot/_common/Services/Impl/FontProvider.cs @@ -6,6 +6,7 @@ namespace EllieBot.Services; public class FontProvider : IEService { public FontFamily NotoSans { get; } + public FontFamily Symbola { get; } public List<FontFamily> FallBackFonts { get; } private readonly FontCollection _fonts; @@ -14,6 +15,7 @@ public class FontProvider : IEService _fonts = new(); NotoSans = _fonts.Add("data/fonts/NotoSans-Bold.ttf"); + Symbola = _fonts.Add("data/fonts/Symbola-10.24.ttf"); FallBackFonts = new(); @@ -27,7 +29,9 @@ public class FontProvider : IEService FallBackFonts.AddRange(_fonts.AddCollection(Path.Combine(fontsfolder, "msgothic.ttc"))); FallBackFonts.AddRange(_fonts.AddCollection(Path.Combine(fontsfolder, "segoe.ttc"))); } - catch { } + catch + { + } } // any fonts present in data/fonts should be added as fallback fonts diff --git a/src/EllieBot/data/commandlist.json b/src/EllieBot/data/commandlist.json index fdc1d00..8324dad 100644 --- a/src/EllieBot/data/commandlist.json +++ b/src/EllieBot/data/commandlist.json @@ -4315,7 +4315,8 @@ "Aliases": [ ".ncsetpixel", ".ncsp", - ".ncs" + ".ncs", + ".ncset" ], "Description": "Sets a pixel's color and text on the nCanvas.\nYou must specify the position of the pixel to set in alphanumeric format.\nYou can obtain alphanumeric position of the pixel by using `nczoom` or `ncp <x> <y>`", "Usage": [ diff --git a/src/EllieBot/strings/aliases.yml b/src/EllieBot/strings/aliases.yml index 0f5d61d..3fc6067 100644 --- a/src/EllieBot/strings/aliases.yml +++ b/src/EllieBot/strings/aliases.yml @@ -1484,6 +1484,7 @@ ncsetpixel: - ncsetpixel - ncsp - ncs + - ncset nczoom: - nczoom - ncz