diff --git a/src/Files.App.CsWin32/NativeMethods.json b/src/Files.App.CsWin32/NativeMethods.json
index 40ae5356353c3..1b33355d8286a 100644
--- a/src/Files.App.CsWin32/NativeMethods.json
+++ b/src/Files.App.CsWin32/NativeMethods.json
@@ -1,5 +1,20 @@
{
"$schema": "https://aka.ms/CsWin32.schema.json",
"allowMarshaling": false,
- "public": true
+ "public": true,
+ "comInterop": {
+ "preserveSigMethods": [
+ "IEnumShellItems.Next",
+ "IInitializeWithFile.Initialize",
+ "IInitializeWithItem.Initialize",
+ "IInitializeWithStream.Initialize",
+ "IPreviewHandler.QueryFocus",
+ "IPreviewHandler.SetRect",
+ "IObjectWithSite.SetSite",
+ "IPreviewHandler.SetWindow",
+ "IPreviewHandlerVisuals.SetBackgroundColor",
+ "IPreviewHandlerVisuals.SetFont",
+ "IPreviewHandlerVisuals.SetTextColor",
+ ]
+ }
}
diff --git a/src/Files.App.CsWin32/NativeMethods.txt b/src/Files.App.CsWin32/NativeMethods.txt
index 84d10caa634de..18453c0c6898a 100644
--- a/src/Files.App.CsWin32/NativeMethods.txt
+++ b/src/Files.App.CsWin32/NativeMethods.txt
@@ -135,3 +135,10 @@ GetModuleHandle
RegisterClassEx
CREATESTRUCTW
AssocQueryString
+IPreviewHandlerFrame
+IPreviewHandlerVisuals
+IObjectWithSite
+IInitializeWithStream
+IInitializeWithStreamNative
+IInitializeWithFile
+IInitializeWithItem
diff --git a/src/Files.App/Utils/Shell/PreviewHandler.cs b/src/Files.App/Utils/Shell/PreviewHandler.cs
index 06160b4c2a213..dec86f96bbf4d 100644
--- a/src/Files.App/Utils/Shell/PreviewHandler.cs
+++ b/src/Files.App/Utils/Shell/PreviewHandler.cs
@@ -7,153 +7,52 @@
using Windows.Win32.Foundation;
using Windows.Win32.Graphics.Gdi;
using Windows.Win32.System.Com;
-using Windows.Win32.UI.WindowsAndMessaging;
+using Windows.Win32.System.Ole;
+using Windows.Win32.UI.Shell;
+using Windows.Win32.UI.Shell.PropertiesSystem;
namespace Files.App.Utils.Shell
{
///
/// Credits: https://github.com/GeeLaw/PreviewHost/
///
- public sealed class PreviewHandler : IDisposable
+ public unsafe sealed class PreviewHandler : IDisposable
{
// Fields
- private static readonly Guid IPreviewHandlerIid = Guid.ParseExact("8895b1c6-b41f-4c1c-a562-0d564250836f", "d");
-
- #region IPreviewHandlerFrame support
-
- [StructLayout(LayoutKind.Sequential)]
- public struct PreviewHandlerFrameInfo
- {
- public nint AcceleratorTableHandle;
- public uint AcceleratorEntryCount;
- }
-
- [ComImport, Guid("fec87aaf-35f9-447a-adb7-20234491401a"), InterfaceType(ComInterfaceType.InterfaceIsIUnknown)]
- public interface IPreviewHandlerFrame
- {
- [PreserveSig]
- HRESULT GetWindowContext(out PreviewHandlerFrameInfo pinfo);
- [PreserveSig]
- HRESULT TranslateAccelerator(ref MSG pmsg);
- }
-
- public sealed class PreviewHandlerFrame : IPreviewHandlerFrame, IDisposable
- {
- bool disposed;
- nint hwnd;
-
- public PreviewHandlerFrame(nint frame)
- {
- disposed = true;
- disposed = false;
- hwnd = frame;
- }
-
- public void Dispose()
- {
- disposed = true;
- }
-
- public HRESULT GetWindowContext(out PreviewHandlerFrameInfo pinfo)
- {
- pinfo.AcceleratorTableHandle = nint.Zero;
- pinfo.AcceleratorEntryCount = 0;
- if (disposed)
- return HRESULT.E_FAIL;
- return HRESULT.S_OK;
- }
-
- public HRESULT TranslateAccelerator(ref MSG pmsg)
- {
- if (disposed)
- return HRESULT.E_FAIL;
- return HRESULT.S_FALSE;
- }
- }
-
- #endregion IPreviewHandlerFrame support
-
- #region IPreviewHandler major interfaces
-
- [ComImport, Guid("8895b1c6-b41f-4c1c-a562-0d564250836f"), InterfaceType(ComInterfaceType.InterfaceIsIUnknown)]
- interface IPreviewHandler
- {
- [PreserveSig]
- HRESULT SetWindow(nint hwnd, ref RECT prc);
- [PreserveSig]
- HRESULT SetRect(ref RECT prc);
- [PreserveSig]
- HRESULT DoPreview();
- [PreserveSig]
- HRESULT Unload();
- [PreserveSig]
- HRESULT SetFocus();
- [PreserveSig]
- HRESULT QueryFocus(out nint phwnd);
- // TranslateAccelerator is not used here.
- }
-
- [ComImport, Guid("196bf9a5-b346-4ef0-aa1e-5dcdb76768b1"), InterfaceType(ComInterfaceType.InterfaceIsIUnknown)]
- interface IPreviewHandlerVisuals
- {
- [PreserveSig]
- HRESULT SetBackgroundColor(uint color);
- [PreserveSig]
- HRESULT SetFont(ref LOGFONTW plf);
- [PreserveSig]
- HRESULT SetTextColor(uint color);
- }
-
- static uint ColorRefFromColor(Color color)
- {
- return (((uint)color.B) << 16) | (((uint)color.G) << 8) | ((uint)color.R);
- }
-
- [ComImport, Guid("fc4801a3-2ba9-11cf-a229-00aa003d7352"), InterfaceType(ComInterfaceType.InterfaceIsIUnknown)]
- interface IObjectWithSite
- {
- [PreserveSig]
- HRESULT SetSite([In, MarshalAs(UnmanagedType.IUnknown)] object pUnkSite);
- // GetSite is not used.
- }
-
- #endregion IPreviewHandler major interfaces
-
- bool disposed;
- bool init;
+ IPreviewHandler* _pPreviewHandler;
+ IPreviewHandlerFrame* _pPreviewHandlerFrame;
+ IPreviewHandlerVisuals* _previewHandlerVisuals;
+ HWND _hWnd;
+ bool _disposed;
+ bool _initialized;
bool shown;
- PreviewHandlerFrame comSite;
- nint hwnd;
- IPreviewHandler previewHandler;
- IPreviewHandlerVisuals visuals;
- nint pPreviewHandler;
// Initializer
- public PreviewHandler(Guid clsid, nint frame)
+ public PreviewHandler(Guid clsid, HWND frame)
{
- disposed = true;
- init = false;
+ _disposed = true;
+ _initialized = false;
shown = false;
- comSite = new PreviewHandlerFrame(frame);
- hwnd = frame;
+ _hWnd = frame;
try
{
SetupHandler(clsid);
- disposed = false;
+ _disposed = false;
}
catch
{
- if (previewHandler != null)
- Marshal.ReleaseComObject(previewHandler);
- previewHandler = null;
- if (pPreviewHandler != nint.Zero)
- Marshal.Release(pPreviewHandler);
- pPreviewHandler = nint.Zero;
- comSite.Dispose();
- comSite = null;
+ if (_pPreviewHandler is not null)
+ _pPreviewHandler->Release();
+
+ if (_pPreviewHandlerFrame is not null)
+ _pPreviewHandlerFrame->Release();
+
+ _pPreviewHandler = null;
+ _pPreviewHandlerFrame = null;
+
throw;
}
}
@@ -162,24 +61,11 @@ public PreviewHandler(Guid clsid, nint frame)
unsafe void SetupHandler(Guid clsid)
{
- nint pph;
- var iid = IPreviewHandlerIid;
- var cannotCreate = "Cannot create class " + clsid.ToString() + " as IPreviewHandler.";
- var cannotCast = "Cannot cast class " + clsid.ToString() + " as IObjectWithSite.";
- var cannotSetSite = "Cannot set site to the preview handler object.";
-
- // Important: manually calling CoCreateInstance is necessary.
- // If we use Activator.CreateInstance(Type.GetTypeFromCLSID(...)),
- // CLR will allow in-process server, which defeats isolation and
- // creates strange bugs.
HRESULT hr = PInvoke.CoCreateInstance(
clsid,
null,
CLSCTX.CLSCTX_LOCAL_SERVER,
- ref iid,
- out void* previewHandlerPtr);
-
- object previewHandlerObject = *(IPreviewHandler*)previewHandlerPtr;
+ out _pPreviewHandler);
// See https://blogs.msdn.microsoft.com/adioltean/2005/06/24/when-cocreateinstance-returns-0x80080005-co_e_server_exec_failure/
// CO_E_SERVER_EXEC_FAILURE also tends to happen when debugging in Visual Studio.
@@ -187,67 +73,34 @@ unsafe void SetupHandler(Guid clsid)
// to use another thread with low mandatory label. We keep it simple by creating
// a same-integrity object.
//if (hr == HRESULT.CO_E_SERVER_EXEC_FAILURE)
- // hr = CoCreateInstance(ref clsid, nint.Zero, ClassContext.LocalServer, ref iid, out pph);
- if ((int)hr < 0)
- throw new COMException(cannotCreate, (int)hr);
-
- pPreviewHandler = new(previewHandlerPtr);
- previewHandler = (IPreviewHandler)previewHandlerObject;
- if (previewHandler == null)
- {
- Marshal.ReleaseComObject(previewHandlerObject);
- throw new COMException(cannotCreate);
- }
-
- var objectWithSite = previewHandlerObject as IObjectWithSite;
- if (objectWithSite == null)
- throw new COMException(cannotCast);
-
- hr = objectWithSite.SetSite(comSite);
- if ((int)hr < 0)
- throw new COMException(cannotSetSite, (int)hr);
-
- visuals = previewHandlerObject as IPreviewHandlerVisuals;
- }
+ // hr = CoCreateInstance(clsid, null, CLSCTX.CLSCTX_LOCAL_SERVER, typeof(IPreviewHandler).GUID, out var pph);
+ if (hr.Value < 0)
+ throw new COMException("Cannot create class " + clsid.ToString() + " as IPreviewHandler.", hr.Value);
+ else if (_pPreviewHandler is null)
+ throw new COMException("Cannot create class " + clsid.ToString() + " as IPreviewHandler.");
- #region Initialization interfaces
+ _pPreviewHandler->QueryInterface(typeof(IPreviewHandlerFrame).GUID, out var ppPreviewHandlerFrame);
+ if (ppPreviewHandlerFrame == null)
+ throw new COMException("Cannot cast class " + clsid.ToString() + " as IObjectWithSite.");
- [ComImport, Guid("b824b49d-22ac-4161-ac8a-9916e8fa3f7f"), InterfaceType(ComInterfaceType.InterfaceIsIUnknown)]
- interface IInitializeWithStream
- {
- [PreserveSig]
- HRESULT Initialize(IStream psi, STGM grfMode);
- }
-
- [ComImport, Guid("b824b49d-22ac-4161-ac8a-9916e8fa3f7f"), InterfaceType(ComInterfaceType.InterfaceIsIUnknown)]
- interface IInitializeWithStreamNative
- {
- [PreserveSig]
- HRESULT Initialize(nint psi, STGM grfMode);
- }
+ _pPreviewHandlerFrame = (IPreviewHandlerFrame*)ppPreviewHandlerFrame;
- static readonly Guid IInitializeWithStreamIid = Guid.ParseExact("b824b49d-22ac-4161-ac8a-9916e8fa3f7f", "d");
+ _pPreviewHandler->QueryInterface(typeof(IObjectWithSite).GUID, out var ppObjectWithSite);
+ if (ppObjectWithSite == null)
+ throw new COMException("Cannot cast class " + clsid.ToString() + " as IObjectWithSite.");
- [ComImport, Guid("b7d14566-0509-4cce-a71f-0a554233bd9b"), InterfaceType(ComInterfaceType.InterfaceIsIUnknown)]
- interface IInitializeWithFile
- {
- [PreserveSig]
- HRESULT Initialize([MarshalAs(UnmanagedType.LPWStr)] string pszFilePath, STGM grfMode);
- }
+ var pObjectWithSite = (IObjectWithSite*)ppObjectWithSite;
+ hr = pObjectWithSite->SetSite((IUnknown*)_pPreviewHandlerFrame);
+ if (hr.Value < 0)
+ throw new COMException("Cannot set site to the preview handler object.", hr.Value);
- static readonly Guid IInitializeWithFileIid = Guid.ParseExact("b7d14566-0509-4cce-a71f-0a554233bd9b", "d");
+ _pPreviewHandler->QueryInterface(typeof(IPreviewHandlerVisuals).GUID, out var ppPreviewHandlerVisuals);
+ if (ppObjectWithSite == null)
+ throw new COMException("Cannot cast class " + clsid.ToString() + " as IObjectWithSite.");
- [ComImport, Guid("7f73be3f-fb79-493c-a6c7-7ee14e245841"), InterfaceType(ComInterfaceType.InterfaceIsIUnknown)]
- interface IInitializeWithItem
- {
- [PreserveSig]
- HRESULT Initialize(nint psi, STGM grfMode);
+ _previewHandlerVisuals = (IPreviewHandlerVisuals*)ppPreviewHandlerVisuals;
}
- static readonly Guid IInitializeWithItemIid = Guid.ParseExact("7f73be3f-fb79-493c-a6c7-7ee14e245841", "d");
-
- #endregion
-
///
/// Tries to initialize the preview handler with an IStream.
///
@@ -261,17 +114,20 @@ public bool InitWithStream(IStream stream, STGM mode)
if (mode != STGM.STGM_READ && mode != STGM.STGM_READWRITE)
throw new ArgumentOutOfRangeException("mode", mode, "The argument mode must be Read or ReadWrite.");
- var iws = previewHandler as IInitializeWithStream;
- if (iws == null)
+ IInitializeWithStream* pInitializeWithStream = default;
+ _pPreviewHandlerFrame->QueryInterface(typeof(IInitializeWithStream).GUID, out void* ppInitializeWithStream);
+ if (ppInitializeWithStream is null)
return false;
- var hr = iws.Initialize(stream, mode);
+ pInitializeWithStream = (IInitializeWithStream*)ppInitializeWithStream;
+
+ HRESULT hr = pInitializeWithStream->Initialize(&stream, (uint)mode);
if (hr == HRESULT.E_NOTIMPL)
return false;
- if ((int)hr < 0)
+ else if ((int)hr < 0)
throw new COMException("IInitializeWithStream.Initialize failed.", (int)hr);
- init = true;
+ _initialized = true;
return true;
}
@@ -292,17 +148,20 @@ public bool InitWithStream(nint pStream, STGM mode)
if (mode != STGM.STGM_READ && mode != STGM.STGM_READWRITE)
throw new ArgumentOutOfRangeException("mode", mode, "The argument mode must be Read or ReadWrite.");
- var iws = previewHandler as IInitializeWithStreamNative;
- if (iws == null)
+ IInitializeWithStream* pInitializeWithStream = default;
+ _pPreviewHandlerFrame->QueryInterface(typeof(IInitializeWithStream).GUID, out void* ppInitializeWithStream);
+ if (ppInitializeWithStream is null)
return false;
- var hr = iws.Initialize(pStream, mode);
+ pInitializeWithStream = (IInitializeWithStream*)ppInitializeWithStream;
+
+ HRESULT hr = pInitializeWithStream->Initialize((IStream*)pStream, (uint)mode);
if (hr == HRESULT.E_NOTIMPL)
return false;
- if ((int)hr < 0)
+ else if ((int)hr < 0)
throw new COMException("IInitializeWithStream.Initialize failed.", (int)hr);
- init = true;
+ _initialized = true;
return true;
}
@@ -323,17 +182,20 @@ public bool InitWithItem(nint psi, STGM mode)
if (mode != STGM.STGM_READ && mode != STGM.STGM_READWRITE)
throw new ArgumentOutOfRangeException("mode", mode, "The argument mode must be Read or ReadWrite.");
- var iwi = previewHandler as IInitializeWithItem;
- if (iwi == null)
+ IInitializeWithItem* pInitializeWithStream = default;
+ _pPreviewHandlerFrame->QueryInterface(typeof(IInitializeWithItem).GUID, out void* ppInitializeWithStream);
+ if (ppInitializeWithStream is null)
return false;
- var hr = iwi.Initialize(psi, mode);
+ pInitializeWithStream = (IInitializeWithItem*)ppInitializeWithStream;
+
+ HRESULT hr = pInitializeWithStream->Initialize((IShellItem*)psi, (uint)mode);
if (hr == HRESULT.E_NOTIMPL)
return false;
- if ((int)hr < 0)
- throw new COMException("IInitializeWithItem.Initialize failed.", (int)hr);
+ else if ((int)hr < 0)
+ throw new COMException("IInitializeWithStream.Initialize failed.", (int)hr);
- init = true;
+ _initialized = true;
return true;
}
@@ -354,17 +216,20 @@ public bool InitWithFile(string path, STGM mode)
if (mode != STGM.STGM_READ && mode != STGM.STGM_READWRITE)
throw new ArgumentOutOfRangeException("mode", mode, "The argument mode must be Read or ReadWrite.");
- var iwf = previewHandler as IInitializeWithFile;
- if (iwf == null)
+ IInitializeWithFile* pInitializeWithStream = default;
+ _pPreviewHandlerFrame->QueryInterface(typeof(IInitializeWithFile).GUID, out void* ppInitializeWithStream);
+ if (ppInitializeWithStream is null)
return false;
- var hr = iwf.Initialize(path, mode);
+ pInitializeWithStream = (IInitializeWithFile*)ppInitializeWithStream;
+
+ HRESULT hr = pInitializeWithStream->Initialize(path, (uint)mode);
if (hr == HRESULT.E_NOTIMPL)
return false;
- if ((int)hr < 0)
- throw new COMException("IInitializeWithFile.Initialize failed.", (int)hr);
+ else if ((int)hr < 0)
+ throw new COMException("IInitializeWithStream.Initialize failed.", (int)hr);
- init = true;
+ _initialized = true;
return true;
}
@@ -377,19 +242,16 @@ public bool InitWithFile(string path, STGM mode)
public bool InitWithFileWithEveryWay(string path)
{
var exceptions = new List();
- var pobj = nint.Zero;
-
- // Why should we try IStream first?
- // Because that gives us the best security.
- // If we initialize with string or IShellItem,
- // we have no control over how the preview handler
- // opens the file, which might decide to open the
- // file for read/write exclusively.
+ var pObject = nint.Zero;
+
+ // We try IStream first so that this gives us the best security.
+ // If we initialize with string or IShellItem, we have no control over how the preview handler
+ // opens the file, which might decide to open the file for read/write exclusively.
try
{
- pobj = ItemStreamHelper.IStreamFromPath(path);
- if (pobj != nint.Zero
- && InitWithStream(pobj, STGM.STGM_READ))
+ pObject = ItemStreamHelper.IStreamFromPath(path);
+ if (pObject != nint.Zero
+ && InitWithStream(pObject, STGM.STGM_READ))
return true;
}
catch (Exception ex)
@@ -398,10 +260,10 @@ public bool InitWithFileWithEveryWay(string path)
}
finally
{
- if (pobj != nint.Zero)
- ItemStreamHelper.ReleaseObject(pobj);
+ if (pObject != nint.Zero)
+ ItemStreamHelper.ReleaseObject(pObject);
- pobj = nint.Zero;
+ pObject = nint.Zero;
}
// Next try file because that could save us some P/Invokes.
@@ -417,9 +279,9 @@ public bool InitWithFileWithEveryWay(string path)
try
{
- pobj = ItemStreamHelper.IShellItemFromPath(path);
- if (pobj != nint.Zero
- && InitWithItem(pobj, STGM.STGM_READ))
+ pObject = ItemStreamHelper.IShellItemFromPath(path);
+ if (pObject != nint.Zero
+ && InitWithItem(pObject, STGM.STGM_READ))
return true;
if (exceptions.Count == 0)
@@ -431,10 +293,10 @@ public bool InitWithFileWithEveryWay(string path)
}
finally
{
- if (pobj != nint.Zero)
- ItemStreamHelper.ReleaseObject(pobj);
+ if (pObject != nint.Zero)
+ ItemStreamHelper.ReleaseObject(pObject);
- pobj = nint.Zero;
+ pObject = nint.Zero;
}
throw new AggregateException(exceptions);
@@ -446,14 +308,13 @@ public bool InitWithFileWithEveryWay(string path)
public bool ResetWindow()
{
EnsureNotDisposed();
-
//EnsureInitialized();
- if (!init)
+ if (!_initialized)
return false;
- var hr = previewHandler.SetWindow(hwnd, new());
- return (int)hr >= 0;
+ HRESULT hr = _pPreviewHandler->SetWindow(_hWnd, new RECT());
+ return hr.Value >= 0;
}
///
@@ -465,10 +326,10 @@ public bool ResetBounds(RECT previewerBounds)
//EnsureInitialized();
- if (!init)
+ if (!_initialized)
return false;
- var hr = previewHandler.SetRect(previewerBounds);
+ HRESULT hr = _pPreviewHandler->SetRect(previewerBounds);
return (int)hr >= 0;
}
@@ -479,8 +340,8 @@ public bool ResetBounds(RECT previewerBounds)
/// Whether the call succeeds.
public bool SetBackground(Color color)
{
- var hr = visuals?.SetBackgroundColor(ColorRefFromColor(color));
- return hr.HasValue && (int)hr.Value >= 0;
+ HRESULT hr = _previewHandlerVisuals->SetBackgroundColor(new(ColorRefFromColor(color)));
+ return hr.Value >= 0;
}
///
@@ -490,8 +351,8 @@ public bool SetBackground(Color color)
/// Whether the call succeeds.
public bool SetForeground(Color color)
{
- var hr = visuals?.SetTextColor(ColorRefFromColor(color));
- return hr.HasValue && (int)hr.Value >= 0;
+ HRESULT hr = _previewHandlerVisuals->SetTextColor(new(ColorRefFromColor(color)));
+ return hr.Value >= 0;
}
///
@@ -501,8 +362,8 @@ public bool SetForeground(Color color)
/// Whether the call succeeds.
public bool SetFont(ref LOGFONTW font)
{
- var hr = visuals?.SetFont(ref font);
- return hr.HasValue && (int)hr.Value >= 0;
+ HRESULT hr = _previewHandlerVisuals->SetFont(font);
+ return hr.Value >= 0;
}
///
@@ -511,17 +372,16 @@ public bool SetFont(ref LOGFONTW font)
public void DoPreview()
{
EnsureNotDisposed();
-
//EnsureInitialized();
- if (!init)
+ if (!_initialized)
return;
EnsureNotShown();
ResetWindow();
- previewHandler.DoPreview();
+ _pPreviewHandler->DoPreview();
shown = true;
}
@@ -535,11 +395,12 @@ public void Focus()
//EnsureInitialized();
- if (!init)
+ if (!_initialized)
return;
EnsureShown();
- previewHandler.SetFocus();
+
+ _pPreviewHandler->SetFocus();
}
///
@@ -552,18 +413,18 @@ public nint QueryFocus()
//EnsureInitialized();
- if (!init)
+ if (!_initialized)
return nint.Zero;
EnsureShown();
nint result;
- var hr = previewHandler.QueryFocus(out result);
- if ((int)hr < 0)
+ HRESULT hr = _pPreviewHandler->QueryFocus(out HWND hWnd);
+ if (hr.Value < 0)
return nint.Zero;
- return result;
+ return hWnd.Value;
}
///
@@ -574,62 +435,72 @@ public void UnloadPreview()
Dispose(true);
}
- void EnsureNotDisposed()
+ private void EnsureNotDisposed()
{
- if (disposed)
+ if (_disposed)
throw new ObjectDisposedException("PreviewHandler");
}
- void EnsureInitialized()
+ private void EnsureInitialized()
{
- if (!init)
+ if (!_initialized)
throw new InvalidOperationException("Object must be initialized before calling this method.");
}
- void EnsureNotInitialized()
+ private void EnsureNotInitialized()
{
- if (init)
+ if (_initialized)
throw new InvalidOperationException("Object is already initialized and cannot be initialized again.");
}
- void EnsureShown()
+ private void EnsureShown()
{
if (!shown)
throw new InvalidOperationException("The preview handler must be shown to call this method.");
}
- void EnsureNotShown()
+ private void EnsureNotShown()
{
if (shown)
throw new InvalidOperationException("The preview handler must not be shown to call this method.");
}
- // Dispose
+ private uint ColorRefFromColor(Color color)
+ {
+ return (((uint)color.B) << 16) | (((uint)color.G) << 8) | ((uint)color.R);
+ }
+
+ // Disposers
void Dispose(bool disposing)
{
- if (disposed)
+ if (_disposed)
return;
- disposed = true;
- init = false;
+
+ _disposed = true;
+ _initialized = false;
+
if (disposing)
{
- previewHandler.Unload();
- comSite.Dispose();
- Marshal.ReleaseComObject(previewHandler);
+ _pPreviewHandler->Unload();
+ _pPreviewHandlerFrame->Release();
+ _pPreviewHandler->Release();
}
else
{
// We're in the finalizer.
// Field previewHandler might have been finalized at this point.
// Get a new RCW.
- var phObject = Marshal.GetUniqueObjectForIUnknown(pPreviewHandler);
- var ph = phObject as IPreviewHandler;
- if (ph != null)
- ph.Unload();
- Marshal.ReleaseComObject(phObject);
+
+ //var phObject = Marshal.GetUniqueObjectForIUnknown(_pPreviewHandler);
+ //var ph = phObject as IPreviewHandler;
+ //if (ph != null)
+ // ph.Unload();
+
+ //Marshal.ReleaseComObject(phObject);
}
- Marshal.Release(pPreviewHandler);
+
+ _pPreviewHandler->Release();
}
~PreviewHandler()
diff --git a/src/Files.App/ViewModels/UserControls/Previews/ShellPreviewViewModel.cs b/src/Files.App/ViewModels/UserControls/Previews/ShellPreviewViewModel.cs
index 3062e47e9669e..5b9152879eea6 100644
--- a/src/Files.App/ViewModels/UserControls/Previews/ShellPreviewViewModel.cs
+++ b/src/Files.App/ViewModels/UserControls/Previews/ShellPreviewViewModel.cs
@@ -103,7 +103,7 @@ private unsafe LRESULT WndProc(HWND hwnd, uint msg, WPARAM wParam, LPARAM lParam
try
{
- _currentPreviewHandler = new PreviewHandler(clsid.Value, hwnd.Value);
+ _currentPreviewHandler = new PreviewHandler(clsid.Value, hwnd);
_currentPreviewHandler.InitWithFileWithEveryWay(Item.ItemPath);
_currentPreviewHandler.DoPreview();
}