a0d6f2d6ec
Former-commit-id: d3ff4118f95cc6907059c6001e5157df8832a115
531 lines
19 KiB
C#
531 lines
19 KiB
C#
//
|
|
// System.Drawing.Imaging.Metafile.cs
|
|
//
|
|
// Authors:
|
|
// Christian Meyer, eMail: Christian.Meyer@cs.tum.edu
|
|
// Dennis Hayes (dennish@raytek.com)
|
|
// Sebastien Pouliot <sebastien@ximian.com>
|
|
//
|
|
// (C) 2002 Ximian, Inc. http://www.ximian.com
|
|
// Copyright (C) 2004,2006-2007 Novell, Inc (http://www.novell.com)
|
|
//
|
|
// Permission is hereby granted, free of charge, to any person obtaining
|
|
// a copy of this software and associated documentation files (the
|
|
// "Software"), to deal in the Software without restriction, including
|
|
// without limitation the rights to use, copy, modify, merge, publish,
|
|
// distribute, sublicense, and/or sell copies of the Software, and to
|
|
// permit persons to whom the Software is furnished to do so, subject to
|
|
// the following conditions:
|
|
//
|
|
// The above copyright notice and this permission notice shall be
|
|
// included in all copies or substantial portions of the Software.
|
|
//
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
// EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
|
// NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
|
// LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
|
// OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
|
// WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
//
|
|
|
|
using System.IO;
|
|
using System.Reflection;
|
|
using System.ComponentModel;
|
|
using System.Runtime.InteropServices;
|
|
|
|
namespace System.Drawing.Imaging {
|
|
|
|
[MonoTODO ("Metafiles, both WMF and EMF formats, are only partially supported.")]
|
|
[Serializable]
|
|
[Editor ("System.Drawing.Design.MetafileEditor, " + Consts.AssemblySystem_Drawing_Design, typeof (System.Drawing.Design.UITypeEditor))]
|
|
public sealed class Metafile : Image {
|
|
|
|
// Non-null if a graphics instance was created using
|
|
// Graphics.FromImage(this) The metadata holder is responsible for
|
|
// freeing the nativeImage if the Metadata instance is disposed before
|
|
// the Graphics instance.
|
|
private MetafileHolder _metafileHolder;
|
|
|
|
// A class responsible for disposing of the native Metafile instance
|
|
// if it needs to outlive the managed Metafile instance.
|
|
//
|
|
// The following are both legal with win32 GDI+:
|
|
// Metafile mf = ...; // get a metafile instance
|
|
// Graphics g = Graphics.FromImage(mf); // get a graphics instance
|
|
// g.Dispose(); mf.Dispose(); // dispose of the graphics instance first
|
|
// OR
|
|
// mf.Dispose(); g.Dispose(); // dispose of the metafile instance first
|
|
//
|
|
// ligbgdiplus has a bug where disposing of the metafile instance first will
|
|
// trigger a use of freed memory when the graphics instance is disposed, which
|
|
// could lead to crashes when the native memory is reused.
|
|
//
|
|
// The metafile holder is designed to take ownership of the native metafile image
|
|
// when the managed Metafile instance is disposed while a Graphics instance is still
|
|
// not disposed (ie the second code pattern above) and to keep the native image alive until the graphics
|
|
// instance is disposed.
|
|
//
|
|
// Note that the following throws, so we only ever need to keep track of one Graphics
|
|
// instance at a time:
|
|
// Metafile mf = ...; // get a metafile instance
|
|
// Graphics g = Graphics.FromImage(mf);
|
|
// Graphics g2 = Graphics.FromImage(mf); // throws OutOfMemoryException on GDI+ on Win32
|
|
internal sealed class MetafileHolder : IDisposable
|
|
{
|
|
private bool _disposed;
|
|
private IntPtr _nativeImage;
|
|
|
|
|
|
internal bool Disposed { get => _disposed; }
|
|
internal MetafileHolder()
|
|
{
|
|
_disposed = false;
|
|
_nativeImage = IntPtr.Zero;
|
|
}
|
|
|
|
~MetafileHolder() => Dispose(false);
|
|
|
|
public void Dispose()
|
|
{
|
|
Dispose(true);
|
|
GC.SuppressFinalize(this);
|
|
}
|
|
|
|
internal void Dispose(bool disposing)
|
|
{
|
|
if (!_disposed)
|
|
{
|
|
IntPtr nativeImage = _nativeImage;
|
|
_nativeImage = IntPtr.Zero;
|
|
_disposed = true;
|
|
if (nativeImage != IntPtr.Zero)
|
|
{
|
|
Status status = GDIPlus.GdipDisposeImage(nativeImage);
|
|
GDIPlus.CheckStatus(status);
|
|
}
|
|
}
|
|
}
|
|
|
|
internal void MetafileDisposed(IntPtr nativeImage)
|
|
{
|
|
_nativeImage = nativeImage;
|
|
}
|
|
|
|
internal void GraphicsDisposed()
|
|
{
|
|
Dispose();
|
|
}
|
|
}
|
|
|
|
internal MetafileHolder AddMetafileHolder()
|
|
{
|
|
// If _metafileHolder is not null and hasn't been disposed yet, there's already a graphics instance associated with
|
|
// this metafile, the native code will return an error status.
|
|
if (_metafileHolder != null && !_metafileHolder.Disposed)
|
|
return null;
|
|
_metafileHolder = new MetafileHolder();
|
|
return _metafileHolder;
|
|
}
|
|
|
|
// constructors
|
|
|
|
internal Metafile (IntPtr ptr)
|
|
{
|
|
nativeObject = ptr;
|
|
}
|
|
|
|
// Usually called when cloning images that need to have
|
|
// not only the handle saved, but also the underlying stream
|
|
// (when using MS GDI+ and IStream we must ensure the stream stays alive for all the life of the Image)
|
|
internal Metafile (IntPtr ptr, Stream stream)
|
|
{
|
|
// under Win32 stream is owned by SD/GDI+ code
|
|
if (GDIPlus.RunningOnWindows ())
|
|
this.stream = stream;
|
|
nativeObject = ptr;
|
|
}
|
|
|
|
public Metafile (Stream stream)
|
|
{
|
|
if (stream == null)
|
|
throw new ArgumentException ("stream");
|
|
|
|
Status status;
|
|
if (GDIPlus.RunningOnUnix ()) {
|
|
// With libgdiplus we use a custom API for this, because there's no easy way
|
|
// to get the Stream down to libgdiplus. So, we wrap the stream with a set of delegates.
|
|
GDIPlus.GdiPlusStreamHelper sh = new GDIPlus.GdiPlusStreamHelper (stream, false);
|
|
status = GDIPlus.GdipCreateMetafileFromDelegate_linux (sh.GetHeaderDelegate, sh.GetBytesDelegate,
|
|
sh.PutBytesDelegate, sh.SeekDelegate, sh.CloseDelegate, sh.SizeDelegate, out nativeObject);
|
|
} else {
|
|
status = GDIPlus.GdipCreateMetafileFromStream (new ComIStreamWrapper (stream), out nativeObject);
|
|
}
|
|
GDIPlus.CheckStatus (status);
|
|
}
|
|
|
|
public Metafile (string filename)
|
|
{
|
|
if (filename == null)
|
|
throw new ArgumentNullException ("filename");
|
|
if (filename.Length == 0)
|
|
throw new ArgumentException ("filename");
|
|
|
|
Status status = GDIPlus.GdipCreateMetafileFromFile (filename, out nativeObject);
|
|
if (status == Status.GenericError)
|
|
throw new ExternalException ("Couldn't load specified file.");
|
|
GDIPlus.CheckStatus (status);
|
|
}
|
|
|
|
public Metafile (IntPtr henhmetafile, bool deleteEmf)
|
|
{
|
|
Status status = GDIPlus.GdipCreateMetafileFromEmf (henhmetafile, deleteEmf, out nativeObject);
|
|
GDIPlus.CheckStatus (status);
|
|
}
|
|
|
|
public Metafile (IntPtr referenceHdc, EmfType emfType) :
|
|
this (referenceHdc, new RectangleF (), MetafileFrameUnit.GdiCompatible, emfType, null)
|
|
{
|
|
}
|
|
|
|
public Metafile (IntPtr referenceHdc, Rectangle frameRect) :
|
|
this (referenceHdc, frameRect, MetafileFrameUnit.GdiCompatible, EmfType.EmfPlusDual, null)
|
|
{
|
|
}
|
|
|
|
public Metafile (IntPtr referenceHdc, RectangleF frameRect) :
|
|
this (referenceHdc, frameRect, MetafileFrameUnit.GdiCompatible, EmfType.EmfPlusDual, null)
|
|
{
|
|
}
|
|
|
|
public Metafile (IntPtr hmetafile, WmfPlaceableFileHeader wmfHeader)
|
|
{
|
|
Status status = GDIPlus.GdipCreateMetafileFromEmf (hmetafile, false, out nativeObject);
|
|
GDIPlus.CheckStatus (status);
|
|
}
|
|
|
|
public Metafile (Stream stream, IntPtr referenceHdc) :
|
|
this (stream, referenceHdc, new RectangleF (), MetafileFrameUnit.GdiCompatible, EmfType.EmfPlusDual, null)
|
|
{
|
|
}
|
|
|
|
public Metafile (string fileName, IntPtr referenceHdc) :
|
|
this (fileName, referenceHdc, new RectangleF (), MetafileFrameUnit.GdiCompatible, EmfType.EmfPlusDual,
|
|
null)
|
|
{
|
|
}
|
|
|
|
public Metafile (IntPtr referenceHdc, EmfType emfType, string description) :
|
|
this (referenceHdc, new RectangleF (), MetafileFrameUnit.GdiCompatible, emfType, description)
|
|
{
|
|
}
|
|
|
|
public Metafile (IntPtr referenceHdc, Rectangle frameRect, MetafileFrameUnit frameUnit) :
|
|
this (referenceHdc, frameRect, frameUnit, EmfType.EmfPlusDual, null)
|
|
{
|
|
}
|
|
|
|
public Metafile (IntPtr referenceHdc, RectangleF frameRect, MetafileFrameUnit frameUnit) :
|
|
this (referenceHdc, frameRect, frameUnit, EmfType.EmfPlusDual, null)
|
|
{
|
|
}
|
|
|
|
public Metafile (IntPtr hmetafile, WmfPlaceableFileHeader wmfHeader, bool deleteWmf)
|
|
{
|
|
Status status = GDIPlus.GdipCreateMetafileFromEmf (hmetafile, deleteWmf, out nativeObject);
|
|
GDIPlus.CheckStatus (status);
|
|
}
|
|
|
|
public Metafile (Stream stream, IntPtr referenceHdc, EmfType type) :
|
|
this (stream, referenceHdc, new RectangleF (), MetafileFrameUnit.GdiCompatible, type, null)
|
|
{
|
|
}
|
|
|
|
public Metafile (Stream stream, IntPtr referenceHdc, Rectangle frameRect) :
|
|
this (stream, referenceHdc, frameRect, MetafileFrameUnit.GdiCompatible, EmfType.EmfPlusDual, null)
|
|
{
|
|
}
|
|
|
|
public Metafile (Stream stream, IntPtr referenceHdc, RectangleF frameRect) :
|
|
this (stream, referenceHdc, frameRect, MetafileFrameUnit.GdiCompatible, EmfType.EmfPlusDual, null)
|
|
{
|
|
}
|
|
|
|
public Metafile (string fileName, IntPtr referenceHdc, EmfType type) :
|
|
this (fileName, referenceHdc, new RectangleF (), MetafileFrameUnit.GdiCompatible, type, null)
|
|
{
|
|
}
|
|
|
|
public Metafile (string fileName, IntPtr referenceHdc, Rectangle frameRect) :
|
|
this (fileName, referenceHdc, frameRect, MetafileFrameUnit.GdiCompatible, EmfType.EmfPlusDual, null)
|
|
{
|
|
}
|
|
|
|
public Metafile (string fileName, IntPtr referenceHdc, RectangleF frameRect) :
|
|
this (fileName, referenceHdc, frameRect, MetafileFrameUnit.GdiCompatible, EmfType.EmfPlusDual, null)
|
|
{
|
|
}
|
|
|
|
public Metafile (IntPtr referenceHdc, Rectangle frameRect, MetafileFrameUnit frameUnit, EmfType type) :
|
|
this (referenceHdc, frameRect, frameUnit, type, null)
|
|
{
|
|
}
|
|
|
|
public Metafile (IntPtr referenceHdc, RectangleF frameRect, MetafileFrameUnit frameUnit, EmfType type) :
|
|
this (referenceHdc, frameRect, frameUnit, type, null)
|
|
{
|
|
}
|
|
|
|
public Metafile (Stream stream, IntPtr referenceHdc, EmfType type, string description) :
|
|
this (stream, referenceHdc, new RectangleF (), MetafileFrameUnit.GdiCompatible, type, description)
|
|
{
|
|
}
|
|
|
|
public Metafile (Stream stream, IntPtr referenceHdc, Rectangle frameRect, MetafileFrameUnit frameUnit) :
|
|
this (stream, referenceHdc, frameRect, frameUnit, EmfType.EmfPlusDual, null)
|
|
{
|
|
}
|
|
|
|
public Metafile (Stream stream, IntPtr referenceHdc, RectangleF frameRect, MetafileFrameUnit frameUnit) :
|
|
this (stream, referenceHdc, frameRect, frameUnit, EmfType.EmfPlusDual, null)
|
|
{
|
|
}
|
|
|
|
public Metafile (string fileName, IntPtr referenceHdc, EmfType type, string description) :
|
|
this (fileName, referenceHdc, new RectangleF (), MetafileFrameUnit.GdiCompatible, type, description)
|
|
{
|
|
}
|
|
|
|
public Metafile (string fileName, IntPtr referenceHdc, Rectangle frameRect, MetafileFrameUnit frameUnit) :
|
|
this (fileName, referenceHdc, frameRect, frameUnit, EmfType.EmfPlusDual, null)
|
|
{
|
|
}
|
|
|
|
public Metafile (string fileName, IntPtr referenceHdc, RectangleF frameRect, MetafileFrameUnit frameUnit) :
|
|
this (fileName, referenceHdc, frameRect, frameUnit, EmfType.EmfPlusDual, null)
|
|
{
|
|
}
|
|
|
|
public Metafile (IntPtr referenceHdc, Rectangle frameRect, MetafileFrameUnit frameUnit, EmfType type,
|
|
string desc)
|
|
{
|
|
Status status = GDIPlus.GdipRecordMetafileI (referenceHdc, type, ref frameRect, frameUnit,
|
|
desc, out nativeObject);
|
|
GDIPlus.CheckStatus (status);
|
|
}
|
|
|
|
public Metafile (IntPtr referenceHdc, RectangleF frameRect, MetafileFrameUnit frameUnit, EmfType type,
|
|
string description)
|
|
{
|
|
Status status = GDIPlus.GdipRecordMetafile (referenceHdc, type, ref frameRect, frameUnit,
|
|
description, out nativeObject);
|
|
GDIPlus.CheckStatus (status);
|
|
}
|
|
|
|
public Metafile (Stream stream, IntPtr referenceHdc, Rectangle frameRect, MetafileFrameUnit frameUnit,
|
|
EmfType type) : this (stream, referenceHdc, frameRect, frameUnit, type, null)
|
|
{
|
|
}
|
|
|
|
public Metafile (Stream stream, IntPtr referenceHdc, RectangleF frameRect, MetafileFrameUnit frameUnit,
|
|
EmfType type) : this (stream, referenceHdc, frameRect, frameUnit, type, null)
|
|
{
|
|
}
|
|
|
|
public Metafile (string fileName, IntPtr referenceHdc, Rectangle frameRect, MetafileFrameUnit frameUnit,
|
|
EmfType type) : this (fileName, referenceHdc, frameRect, frameUnit, type, null)
|
|
{
|
|
}
|
|
|
|
public Metafile (string fileName, IntPtr referenceHdc, Rectangle frameRect, MetafileFrameUnit frameUnit,
|
|
string description) : this (fileName, referenceHdc, frameRect, frameUnit, EmfType.EmfPlusDual, description)
|
|
{
|
|
}
|
|
|
|
public Metafile (string fileName, IntPtr referenceHdc, RectangleF frameRect, MetafileFrameUnit frameUnit,
|
|
EmfType type) : this (fileName, referenceHdc, frameRect, frameUnit, type, null)
|
|
{
|
|
}
|
|
|
|
public Metafile (string fileName, IntPtr referenceHdc, RectangleF frameRect, MetafileFrameUnit frameUnit,
|
|
string desc) : this (fileName, referenceHdc, frameRect, frameUnit, EmfType.EmfPlusDual,
|
|
desc)
|
|
{
|
|
}
|
|
|
|
public Metafile (Stream stream, IntPtr referenceHdc, Rectangle frameRect, MetafileFrameUnit frameUnit,
|
|
EmfType type, string description)
|
|
{
|
|
if (stream == null)
|
|
throw new NullReferenceException ("stream");
|
|
|
|
Status status = Status.NotImplemented;
|
|
if (GDIPlus.RunningOnUnix ()) {
|
|
// With libgdiplus we use a custom API for this, because there's no easy way
|
|
// to get the Stream down to libgdiplus. So, we wrap the stream with a set of delegates.
|
|
GDIPlus.GdiPlusStreamHelper sh = new GDIPlus.GdiPlusStreamHelper (stream, false);
|
|
status = GDIPlus.GdipRecordMetafileFromDelegateI_linux (sh.GetHeaderDelegate, sh.GetBytesDelegate,
|
|
sh.PutBytesDelegate, sh.SeekDelegate, sh.CloseDelegate, sh.SizeDelegate, referenceHdc,
|
|
type, ref frameRect, frameUnit, description, out nativeObject);
|
|
} else {
|
|
status = GDIPlus.GdipRecordMetafileStreamI (new ComIStreamWrapper (stream), referenceHdc,
|
|
type, ref frameRect, frameUnit, description, out nativeObject);
|
|
}
|
|
GDIPlus.CheckStatus (status);
|
|
}
|
|
|
|
public Metafile (Stream stream, IntPtr referenceHdc, RectangleF frameRect, MetafileFrameUnit frameUnit,
|
|
EmfType type, string description)
|
|
{
|
|
if (stream == null)
|
|
throw new NullReferenceException ("stream");
|
|
|
|
Status status = Status.NotImplemented;
|
|
if (GDIPlus.RunningOnUnix ()) {
|
|
// With libgdiplus we use a custom API for this, because there's no easy way
|
|
// to get the Stream down to libgdiplus. So, we wrap the stream with a set of delegates.
|
|
GDIPlus.GdiPlusStreamHelper sh = new GDIPlus.GdiPlusStreamHelper (stream, false);
|
|
status = GDIPlus.GdipRecordMetafileFromDelegate_linux (sh.GetHeaderDelegate, sh.GetBytesDelegate,
|
|
sh.PutBytesDelegate, sh.SeekDelegate, sh.CloseDelegate, sh.SizeDelegate, referenceHdc,
|
|
type, ref frameRect, frameUnit, description, out nativeObject);
|
|
} else {
|
|
status = GDIPlus.GdipRecordMetafileStream (new ComIStreamWrapper (stream), referenceHdc,
|
|
type, ref frameRect, frameUnit, description, out nativeObject);
|
|
}
|
|
GDIPlus.CheckStatus (status);
|
|
}
|
|
|
|
public Metafile (string fileName, IntPtr referenceHdc, Rectangle frameRect, MetafileFrameUnit frameUnit,
|
|
EmfType type, string description)
|
|
{
|
|
Status status = GDIPlus.GdipRecordMetafileFileNameI (fileName, referenceHdc, type, ref frameRect,
|
|
frameUnit, description, out nativeObject);
|
|
GDIPlus.CheckStatus (status);
|
|
}
|
|
|
|
public Metafile (string fileName, IntPtr referenceHdc, RectangleF frameRect, MetafileFrameUnit frameUnit,
|
|
EmfType type, string description)
|
|
{
|
|
Status status = GDIPlus.GdipRecordMetafileFileName (fileName, referenceHdc, type, ref frameRect, frameUnit,
|
|
description, out nativeObject);
|
|
GDIPlus.CheckStatus (status);
|
|
}
|
|
|
|
protected override void Dispose(bool disposing)
|
|
{
|
|
if (_metafileHolder != null && !_metafileHolder.Disposed)
|
|
{
|
|
// There's a graphics instance created from this Metafile,
|
|
// transfer responsibility for disposing the nativeImage to the
|
|
// MetafileHolder
|
|
_metafileHolder.MetafileDisposed(nativeObject);
|
|
_metafileHolder = null;
|
|
nativeObject = IntPtr.Zero;
|
|
}
|
|
|
|
base.Dispose(disposing);
|
|
}
|
|
|
|
// methods
|
|
|
|
public IntPtr GetHenhmetafile ()
|
|
{
|
|
return nativeObject;
|
|
}
|
|
|
|
[MonoLimitation ("Metafiles aren't only partially supported by libgdiplus.")]
|
|
public MetafileHeader GetMetafileHeader ()
|
|
{
|
|
IntPtr header = Marshal.AllocHGlobal (Marshal.SizeOf (typeof (MetafileHeader)));
|
|
try {
|
|
Status status = GDIPlus.GdipGetMetafileHeaderFromMetafile (nativeObject, header);
|
|
GDIPlus.CheckStatus (status);
|
|
return new MetafileHeader (header);
|
|
}
|
|
finally {
|
|
Marshal.FreeHGlobal (header);
|
|
}
|
|
}
|
|
|
|
[MonoLimitation ("Metafiles aren't only partially supported by libgdiplus.")]
|
|
public static MetafileHeader GetMetafileHeader (IntPtr henhmetafile)
|
|
{
|
|
IntPtr header = Marshal.AllocHGlobal (Marshal.SizeOf (typeof (MetafileHeader)));
|
|
try {
|
|
Status status = GDIPlus.GdipGetMetafileHeaderFromEmf (henhmetafile, header);
|
|
GDIPlus.CheckStatus (status);
|
|
return new MetafileHeader (header);
|
|
}
|
|
finally {
|
|
Marshal.FreeHGlobal (header);
|
|
}
|
|
}
|
|
|
|
[MonoLimitation ("Metafiles aren't only partially supported by libgdiplus.")]
|
|
public static MetafileHeader GetMetafileHeader (Stream stream)
|
|
{
|
|
if (stream == null)
|
|
throw new NullReferenceException ("stream");
|
|
|
|
IntPtr header = Marshal.AllocHGlobal (Marshal.SizeOf (typeof (MetafileHeader)));
|
|
try {
|
|
Status status;
|
|
|
|
if (GDIPlus.RunningOnUnix ()) {
|
|
// With libgdiplus we use a custom API for this, because there's no easy way
|
|
// to get the Stream down to libgdiplus. So, we wrap the stream with a set of delegates.
|
|
GDIPlus.GdiPlusStreamHelper sh = new GDIPlus.GdiPlusStreamHelper (stream, false);
|
|
status = GDIPlus.GdipGetMetafileHeaderFromDelegate_linux (sh.GetHeaderDelegate,
|
|
sh.GetBytesDelegate, sh.PutBytesDelegate, sh.SeekDelegate, sh.CloseDelegate,
|
|
sh.SizeDelegate, header);
|
|
} else {
|
|
status = GDIPlus.GdipGetMetafileHeaderFromStream (new ComIStreamWrapper (stream), header);
|
|
}
|
|
GDIPlus.CheckStatus (status);
|
|
return new MetafileHeader (header);
|
|
}
|
|
finally {
|
|
Marshal.FreeHGlobal (header);
|
|
}
|
|
}
|
|
|
|
[MonoLimitation ("Metafiles aren't only partially supported by libgdiplus.")]
|
|
public static MetafileHeader GetMetafileHeader (string fileName)
|
|
{
|
|
if (fileName == null)
|
|
throw new ArgumentNullException ("fileName");
|
|
|
|
IntPtr header = Marshal.AllocHGlobal (Marshal.SizeOf (typeof (MetafileHeader)));
|
|
try {
|
|
Status status = GDIPlus.GdipGetMetafileHeaderFromFile (fileName, header);
|
|
GDIPlus.CheckStatus (status);
|
|
return new MetafileHeader (header);
|
|
}
|
|
finally {
|
|
Marshal.FreeHGlobal (header);
|
|
}
|
|
}
|
|
|
|
[MonoLimitation ("Metafiles aren't only partially supported by libgdiplus.")]
|
|
public static MetafileHeader GetMetafileHeader (IntPtr hmetafile, WmfPlaceableFileHeader wmfHeader)
|
|
{
|
|
IntPtr header = Marshal.AllocHGlobal (Marshal.SizeOf (typeof (MetafileHeader)));
|
|
try {
|
|
Status status = GDIPlus.GdipGetMetafileHeaderFromEmf (hmetafile, header);
|
|
GDIPlus.CheckStatus (status);
|
|
return new MetafileHeader (header);
|
|
}
|
|
finally {
|
|
Marshal.FreeHGlobal (header);
|
|
}
|
|
}
|
|
|
|
[MonoLimitation ("Metafiles aren't only partially supported by libgdiplus.")]
|
|
public void PlayRecord (EmfPlusRecordType recordType, int flags, int dataSize, byte[] data)
|
|
{
|
|
Status status = GDIPlus.GdipPlayMetafileRecord (nativeObject, recordType, flags, dataSize, data);
|
|
GDIPlus.CheckStatus (status);
|
|
}
|
|
}
|
|
}
|