diff --git a/src/System.Drawing.Common/src/System.Drawing.Common.csproj b/src/System.Drawing.Common/src/System.Drawing.Common.csproj index ecf5b2eb9a5..cdad96909d4 100644 --- a/src/System.Drawing.Common/src/System.Drawing.Common.csproj +++ b/src/System.Drawing.Common/src/System.Drawing.Common.csproj @@ -334,6 +334,10 @@ Since .NET 7, non-Windows platforms are not supported, even with the runtime con + + + + diff --git a/src/System.Drawing.Common/src/System/Drawing/GdiplusNative.cs b/src/System.Drawing.Common/src/System/Drawing/GdiplusNative.cs index ac218b2e94d..dd8df18bf68 100644 --- a/src/System.Drawing.Common/src/System/Drawing/GdiplusNative.cs +++ b/src/System.Drawing.Common/src/System/Drawing/GdiplusNative.cs @@ -4455,6 +4455,25 @@ internal static partial int GdipComment( [LibraryImport(LibraryName)] internal static partial int GdipCreateBitmapFromStreamICM(IntPtr stream, IntPtr* bitmap); + +#if NET8_0_OR_GREATER + [LibraryImport(LibraryName)] + internal static partial int GdipCreateCachedBitmap( + [MarshalUsing(typeof(HandleRefMarshaller))] HandleRef bitmap, + [MarshalUsing(typeof(HandleRefMarshaller))] HandleRef graphics, + out nint cachedBitmap); + + [LibraryImport(LibraryName)] + internal static partial int GdipDeleteCachedBitmap( + nint cachedBitmap); + + [LibraryImport(LibraryName)] + internal static partial int GdipDrawCachedBitmap( + [MarshalUsing(typeof(HandleRefMarshaller))] HandleRef graphics, + [MarshalUsing(typeof(HandleRefMarshaller))] HandleRef cachedBitmap, + int x, + int y); +#endif } } } diff --git a/src/System.Drawing.Common/src/System/Drawing/Graphics.cs b/src/System.Drawing.Common/src/System/Drawing/Graphics.cs index 865ff419cc1..f4cfdc1e7f7 100644 --- a/src/System.Drawing.Common/src/System/Drawing/Graphics.cs +++ b/src/System.Drawing.Common/src/System/Drawing/Graphics.cs @@ -3851,5 +3851,36 @@ private static void CheckErrorStatus(int status) // Legitimate error, throw our status exception. throw Gdip.StatusException(status); } + +#if NET8_0_OR_GREATER + + /// + /// Draws the given . + /// + /// The that contains the image to be drawn. + /// The x-coordinate of the upper-left corner of the drawn image. + /// The y-coordinate of the upper-left corner of the drawn image. + /// is . + /// + /// + /// The is not compatible with the device state. + /// + /// + /// - or - + /// + /// + /// The object has a transform applied other than a translation. + /// + /// + public void DrawCachedBitmap(CachedBitmap cachedBitmap, int x, int y) + { + ArgumentNullException.ThrowIfNull(cachedBitmap); + + Gdip.CheckStatus(Gdip.GdipDrawCachedBitmap( + new(this, NativeGraphics), + new(cachedBitmap, cachedBitmap.Handle), + x, y)); + } +#endif } } diff --git a/src/System.Drawing.Common/src/System/Drawing/Imaging/CachedBitmap.cs b/src/System.Drawing.Common/src/System/Drawing/Imaging/CachedBitmap.cs new file mode 100644 index 00000000000..8421141b474 --- /dev/null +++ b/src/System.Drawing.Common/src/System/Drawing/Imaging/CachedBitmap.cs @@ -0,0 +1,79 @@ +// Licensed to the .NET Foundation under one or more agreements. +// The .NET Foundation licenses this file to you under the MIT license. + +using System.Threading; + +#if NET8_0_OR_GREATER + + +using static System.Drawing.SafeNativeMethods; + +namespace System.Drawing.Imaging; + +/// +/// A device dependent copy of a matching a specified object's current +/// device (display) settings. Avoids reformatting step when rendering, which can significantly improve performance. +/// +/// +/// +/// matches the current bit depth of the 's device. If the device bit +/// depth changes, the will no longer be usable and a new instance will need to be created +/// that matches. If the was created against it +/// will always work. +/// +/// +/// will not work with any transformations other than translation. +/// +/// +/// cannot be used to draw to a printer or metafile. +/// +/// +public sealed class CachedBitmap : IDisposable +{ + private nint _handle; + + /// + /// Create a device dependent copy of the given for the device settings of the given + /// + /// + /// The to convert. + /// The object to use to format the cached copy of the . + /// or is . + public CachedBitmap(Bitmap bitmap, Graphics graphics) + { + ArgumentNullException.ThrowIfNull(bitmap); + ArgumentNullException.ThrowIfNull(graphics); + + Gdip.CheckStatus(Gdip.GdipCreateCachedBitmap( + new(bitmap, bitmap.nativeImage), + new(graphics, graphics.NativeGraphics), + out _handle)); + } + + internal nint Handle => _handle; + + private void Dispose(bool disposing) + { + nint handle = Interlocked.Exchange(ref _handle, 0); + if (handle == 0) + { + return; + } + + int status = Gdip.GdipDeleteCachedBitmap(handle); + if (disposing) + { + // Don't want to throw on the finalizer thread. + Gdip.CheckStatus(status); + } + } + + ~CachedBitmap() => Dispose(disposing: false); + + public void Dispose() + { + Dispose(disposing: true); + GC.SuppressFinalize(this); + } +} +#endif diff --git a/src/System.Drawing.Common/tests/GraphicsTests.cs b/src/System.Drawing.Common/tests/GraphicsTests.cs index 308bfec8488..72518f54af8 100644 --- a/src/System.Drawing.Common/tests/GraphicsTests.cs +++ b/src/System.Drawing.Common/tests/GraphicsTests.cs @@ -3197,5 +3197,85 @@ private static void VerifyGraphics(Graphics graphics, RectangleF expectedVisible Assert.Equal(new Matrix(), graphics.Transform); Assert.Equal(expectedVisibleClipBounds, graphics.VisibleClipBounds); } + +#if NET8_0_OR_GREATER + [Fact] + public void DrawCachedBitmap_ThrowsArgumentNullException() + { + using Bitmap bitmap = new(10, 10); + using Graphics graphics = Graphics.FromImage(bitmap); + + Assert.Throws(() => graphics.DrawCachedBitmap(null!, 0, 0)); + } + + [Fact] + public void DrawCachedBitmap_DisposedBitmap_ThrowsArgumentException() + { + using Bitmap bitmap = new(10, 10); + using Graphics graphics = Graphics.FromImage(bitmap); + CachedBitmap cachedBitmap = new(bitmap, graphics); + cachedBitmap.Dispose(); + + Assert.Throws(() => graphics.DrawCachedBitmap(cachedBitmap, 0, 0)); + } + + [Fact] + public void DrawCachedBitmap_Simple() + { + using Bitmap bitmap = new(10, 10); + using Graphics graphics = Graphics.FromImage(bitmap); + using CachedBitmap cachedBitmap = new(bitmap, graphics); + + graphics.DrawCachedBitmap(cachedBitmap, 0, 0); + } + + [Fact] + public void DrawCachedBitmap_Translate() + { + using Bitmap bitmap = new(10, 10); + using Graphics graphics = Graphics.FromImage(bitmap); + graphics.TranslateTransform(1.0f, 8.0f); + using CachedBitmap cachedBitmap = new(bitmap, graphics); + + graphics.DrawCachedBitmap(cachedBitmap, 0, 0); + } + + [Fact] + public void DrawCachedBitmap_Rotation_ThrowsInvalidOperation() + { + using Bitmap bitmap = new(10, 10); + using Graphics graphics = Graphics.FromImage(bitmap); + graphics.RotateTransform(36.0f); + using CachedBitmap cachedBitmap = new(bitmap, graphics); + + Assert.Throws(() => graphics.DrawCachedBitmap(cachedBitmap, 0, 0)); + } + + [Theory] + [InlineData(PixelFormat.Format16bppRgb555, PixelFormat.Format32bppRgb, false)] + [InlineData(PixelFormat.Format32bppRgb, PixelFormat.Format16bppRgb555, true)] + [InlineData(PixelFormat.Format32bppArgb, PixelFormat.Format16bppRgb555, false)] + public void DrawCachedBitmap_ColorDepthChange_ThrowsInvalidOperation( + PixelFormat sourceFormat, + PixelFormat destinationFormat, + bool shouldSucceed) + { + using Bitmap bitmap = new(10, 10, sourceFormat); + using Graphics graphics = Graphics.FromImage(bitmap); + using CachedBitmap cachedBitmap = new(bitmap, graphics); + + using Bitmap bitmap2 = new(10, 10, destinationFormat); + using Graphics graphics2 = Graphics.FromImage(bitmap2); + + if (shouldSucceed) + { + graphics2.DrawCachedBitmap(cachedBitmap, 0, 0); + } + else + { + Assert.Throws(() => graphics2.DrawCachedBitmap(cachedBitmap, 0, 0)); + } + } +#endif } } diff --git a/src/System.Drawing.Common/tests/Imaging/BitmapDataTests.cs b/src/System.Drawing.Common/tests/Imaging/BitmapDataTests.cs index 2a0ae84e55d..0eacf968005 100644 --- a/src/System.Drawing.Common/tests/Imaging/BitmapDataTests.cs +++ b/src/System.Drawing.Common/tests/Imaging/BitmapDataTests.cs @@ -1,4 +1,4 @@ -// Licensed to the .NET Foundation under one or more agreements. +// Licensed to the .NET Foundation under one or more agreements. // The .NET Foundation licenses this file to you under the MIT license. using System.ComponentModel; diff --git a/src/System.Drawing.Common/tests/Imaging/CachedBitmapTests.cs b/src/System.Drawing.Common/tests/Imaging/CachedBitmapTests.cs new file mode 100644 index 00000000000..474dc8e4efe --- /dev/null +++ b/src/System.Drawing.Common/tests/Imaging/CachedBitmapTests.cs @@ -0,0 +1,21 @@ +// Licensed to the .NET Foundation under one or more agreements. +// The .NET Foundation licenses this file to you under the MIT license. + +using Xunit; + +namespace System.Drawing.Imaging.Tests; + +#if NET8_0_OR_GREATER +public class CachedBitmapTests +{ + [Fact] + public void Ctor_Throws_ArgumentNullException() + { + using var bitmap = new Bitmap(10, 10); + using var graphics = Graphics.FromImage(bitmap); + + Assert.Throws(() => new CachedBitmap(bitmap, null)); + Assert.Throws(() => new CachedBitmap(null, graphics)); + } +} +#endif diff --git a/src/System.Drawing.Common/tests/System.Drawing.Common.Tests.csproj b/src/System.Drawing.Common/tests/System.Drawing.Common.Tests.csproj index 2b377f5c823..f584caaf7d3 100644 --- a/src/System.Drawing.Common/tests/System.Drawing.Common.Tests.csproj +++ b/src/System.Drawing.Common/tests/System.Drawing.Common.Tests.csproj @@ -37,6 +37,7 @@ +