// Copyright (c) Six Labors. // Licensed under the Six Labors Split License. using SixLabors.ImageSharp.Formats; using SixLabors.ImageSharp.Formats.Cur; using SixLabors.ImageSharp.Formats.Ico; using SixLabors.ImageSharp.PixelFormats; using SixLabors.ImageSharp.Tests.TestUtilities.ImageComparison; using static SixLabors.ImageSharp.Tests.TestImages.Cur; using static SixLabors.ImageSharp.Tests.TestImages.Ico; namespace SixLabors.ImageSharp.Tests.Formats.Icon.Ico; [Trait("Format", "Icon")] public class IcoEncoderTests { private static IcoEncoder Encoder => new(); [Theory] [WithFile(Flutter, PixelTypes.Rgba32)] public void CanRoundTripEncoder(TestImageProvider provider) where TPixel : unmanaged, IPixel { using Image image = provider.GetImage(IcoDecoder.Instance); using MemoryStream memStream = new(); image.DebugSaveMultiFrame(provider); image.Save(memStream, Encoder); memStream.Seek(0, SeekOrigin.Begin); using Image encoded = Image.Load(memStream); encoded.DebugSaveMultiFrame(provider); // Despite preservation of the palette. The process can still be lossy encoded.CompareToOriginalMultiFrame(provider, ImageComparer.TolerantPercentage(.23f), IcoDecoder.Instance); } [Theory] [WithFile(WindowsMouse, PixelTypes.Rgba32)] public void CanConvertFromCur(TestImageProvider provider) where TPixel : unmanaged, IPixel { using Image image = provider.GetImage(CurDecoder.Instance); using MemoryStream memStream = new(); image.Save(memStream, Encoder); memStream.Seek(0, SeekOrigin.Begin); using Image encoded = Image.Load(memStream); encoded.DebugSaveMultiFrame(provider); encoded.CompareToOriginalMultiFrame(provider, ImageComparer.Exact, CurDecoder.Instance); for (int i = 0; i < image.Frames.Count; i++) { CurFrameMetadata curFrame = image.Frames[i].Metadata.GetCurMetadata(); IcoFrameMetadata icoFrame = encoded.Frames[i].Metadata.GetIcoMetadata(); // Compression may differ as we cannot convert that. Assert.Equal(curFrame.BmpBitsPerPixel, icoFrame.BmpBitsPerPixel); Assert.Equal(curFrame.EncodingWidth, icoFrame.EncodingWidth); Assert.Equal(curFrame.EncodingHeight, icoFrame.EncodingHeight); Assert.Equal(curFrame.ColorTable, icoFrame.ColorTable); } } [Fact] public void Encode_WithTransparentColorBehaviorClear_Works() { // arrange using Image image = new(50, 50); IcoEncoder encoder = new() { TransparentColorMode = TransparentColorMode.Clear, }; Rgba32 rgba32 = Color.Blue.ToPixel(); image.ProcessPixelRows(accessor => { for (int y = 0; y < image.Height; y++) { Span rowSpan = accessor.GetRowSpan(y); // Half of the test image should be transparent. if (y > 25) { rgba32.A = 0; } for (int x = 0; x < image.Width; x++) { rowSpan[x] = Rgba32.FromRgba32(rgba32); } } }); // act using MemoryStream memStream = new(); image.Save(memStream, encoder); // assert memStream.Position = 0; using Image actual = Image.Load(memStream); Rgba32 expectedColor = Color.Blue.ToPixel(); actual.ProcessPixelRows(accessor => { Rgba32 transparent = Color.Transparent.ToPixel(); for (int y = 0; y < accessor.Height; y++) { Span rowSpan = accessor.GetRowSpan(y); if (y > 25) { expectedColor = transparent; } for (int x = 0; x < accessor.Width; x++) { Assert.Equal(expectedColor, rowSpan[x]); } } }); } }