// Copyright (c) Six Labors and contributors. // Licensed under the Apache License, Version 2.0. using System; using System.IO; using System.Linq; using SixLabors.ImageSharp.Formats; using SixLabors.ImageSharp.IO; using SixLabors.ImageSharp.PixelFormats; using Moq; using Xunit; // ReSharper disable InconsistentNaming namespace SixLabors.ImageSharp.Tests { using System.Runtime.CompilerServices; /// /// Tests the class. /// public class ImageSaveTests : IDisposable { private readonly Image Image; private readonly Mock fileSystem; private readonly Mock encoder; private readonly Mock encoderNotInFormat; private IImageFormatDetector localMimeTypeDetector; private Mock localImageFormat; public ImageSaveTests() { this.localImageFormat = new Mock(); this.localImageFormat.Setup(x => x.FileExtensions).Returns(new[] { "png" }); this.localMimeTypeDetector = new MockImageFormatDetector(this.localImageFormat.Object); this.encoder = new Mock(); this.encoderNotInFormat = new Mock(); this.fileSystem = new Mock(); var config = new Configuration() { FileSystem = this.fileSystem.Object }; config.ImageFormatsManager.AddImageFormatDetector(this.localMimeTypeDetector); config.ImageFormatsManager.SetEncoder(this.localImageFormat.Object, this.encoder.Object); this.Image = new Image(config, 1, 1); } [Theory] [WithTestPatternImages(13, 19, PixelTypes.Rgba32 | PixelTypes.Bgr24)] public void SavePixelData_ToPixelStructArray(TestImageProvider provider) where TPixel : struct, IPixel { using (Image image = provider.GetImage()) { TPixel[] buffer = new TPixel[image.Width * image.Height]; image.SavePixelData(buffer); image.ComparePixelBufferTo(buffer); // TODO: We need a separate test-case somewhere ensuring that image pixels are stored in row-major order! } } [Theory] [WithTestPatternImages(19, 13, PixelTypes.Rgba32 | PixelTypes.Bgr24)] public void SavePixelData_ToByteArray(TestImageProvider provider) where TPixel : struct, IPixel { using (Image image = provider.GetImage()) { byte[] buffer = new byte[image.Width * image.Height * Unsafe.SizeOf()]; image.SavePixelData(buffer); image.ComparePixelBufferTo(buffer.AsSpan().NonPortableCast()); } } [Fact] public void SavePixelData_Rgba32_WhenBufferIsTooSmall_Throws() { using (var img = new Image(2, 2)) { img[0, 0] = Rgba32.White; img[1, 0] = Rgba32.Black; img[0, 1] = Rgba32.Red; img[1, 1] = Rgba32.Blue; byte[] buffer = new byte[2 * 2]; // width * height * bytes per pixel Assert.Throws(() => { img.SavePixelData(buffer); }); } } [Fact] public void SavePath() { Stream stream = new MemoryStream(); this.fileSystem.Setup(x => x.Create("path.png")).Returns(stream); this.Image.Save("path.png"); this.encoder.Verify(x => x.Encode(this.Image, stream)); } [Fact] public void SavePathWithEncoder() { Stream stream = new MemoryStream(); this.fileSystem.Setup(x => x.Create("path.jpg")).Returns(stream); this.Image.Save("path.jpg", this.encoderNotInFormat.Object); this.encoderNotInFormat.Verify(x => x.Encode(this.Image, stream)); } [Fact] public void ToBase64String() { string str = this.Image.ToBase64String(this.localImageFormat.Object); this.encoder.Verify(x => x.Encode(this.Image, It.IsAny())); } [Fact] public void SaveStreamWithMime() { Stream stream = new MemoryStream(); this.Image.Save(stream, this.localImageFormat.Object); this.encoder.Verify(x => x.Encode(this.Image, stream)); } [Fact] public void SaveStreamWithEncoder() { Stream stream = new MemoryStream(); this.Image.Save(stream, this.encoderNotInFormat.Object); this.encoderNotInFormat.Verify(x => x.Encode(this.Image, stream)); } public void Dispose() { this.Image.Dispose(); } } }