淡定的跑步鞋 · sql server select ...· 11 月前 · |
有情有义的枇杷 · ThingsBoard教程(六):设备上报 ...· 1 年前 · |
奋斗的木瓜 · react-router-dom ...· 1 年前 · |
踏实的水煮鱼 · Unix : 查找和替换连续的逗号至连续的管道· 1 年前 · |
不羁的槟榔 · Python之tarfile模块的使用 - ...· 1 年前 · |
我想使用 DwmSetIconicThumbnail 函数为我的应用程序的缩略图预览设置一个静态图像。
正如上面的参考链接所指出的,首先需要调用
DwmSetWindowAttribute
来启用
DWMWA_FORCE_ICONIC_REPRESENTATION
和
DWMWA_HAS_ICONIC_BITMAP
属性。
我已经这么做了。我已经从 WindowsAPICodePack 源代码 这里 中提取了所有的定义,并且我遵循了相同的步骤(至少我认为是这样)。
问题是,当我尝试为我的WinForms窗口调整示例时,在下面代码的末尾调用
DwmSetIconicThumbnail
函数时,我得到了一个
DwmSetIconicThumbnail
HRESULT代码,我不确定问题参数是hwnd还是hBitmap。
我做错什么了?
C#:
Bitmap bmp;
IntPtr hBitmap;
IntPtr hwnd;
int hresult;
const int DisplayThumbnailFrame = 0x1;
public enum DwmWindowAttribute : uint
NcRenderingEnabled = 1,
NcRenderingPolicy,
TransitionsForceDisabled,
AllowNcPaint,
CaptionButtonBounds,
NonClientRtlLayout,
ForceIconicRepresentation,
Flip3DPolicy,
ExtendedFrameBounds,
HasIconicBitmap,
DisallowPeek,
ExcludedFromPeek,
Cloak,
Cloaked,
FreezeRepresentation,
[DllImport("dwmapi.dll", PreserveSig = true)]
static internal extern int DwmSetWindowAttribute(IntPtr hwnd,
DwmWindowAttribute dwAttributeToSet,
IntPtr pvAttributeValue,
uint cbAttribute);
[DllImport("Dwmapi.dll")]
public static extern int DwmSetIconicThumbnail(IntPtr hwnd,
IntPtr hBitmap,
int flags);
private void Form1_Shown() {
bmp = (Bitmap)Bitmap.FromFile("C:\\Image.jpg");
hBitmap = bmp.GetHbitmap();
hwnd = Process.GetCurrentProcess.MainWindowHandle;
IntPtr block = Marshal.AllocHGlobal(4);
int value = Math.Abs(Convert.ToInt32(true)); // or 1
Marshal.WriteInt32(block, value);
try {
hresult = DwmSetWindowAttribute(hwnd, DwmWindowAttribute.HasIconicBitmap, block, 4);
if ((hresult != 0)) {
throw Marshal.GetExceptionForHR(hresult);
hresult = DwmSetWindowAttribute(hwnd, DwmWindowAttribute.ForceIconicRepresentation, block, 4);
if ((hresult != 0)) {
throw Marshal.GetExceptionForHR(hresult);
} finally {
Marshal.FreeHGlobal(block);
hresult = DwmSetIconicThumbnail(hwnd, hBitmap, DisplayThumbnailFrame);
if ((hresult != 0)) {
throw Marshal.GetExceptionForHR(hresult);
}
VB.NET:
Dim bmp As Bitmap
Dim hBitmap As IntPtr
Dim hwnd As IntPtr
Dim hresult As Integer
Const DisplayThumbnailFrame As Integer = &H1
Enum DwmWindowAttribute As UInteger
NcRenderingEnabled = 1
NcRenderingPolicy
TransitionsForceDisabled
AllowNcPaint
CaptionButtonBounds
NonClientRtlLayout
ForceIconicRepresentation
Flip3DPolicy
ExtendedFrameBounds
HasIconicBitmap
DisallowPeek
ExcludedFromPeek
Cloak
Cloaked
FreezeRepresentation
End Enum
<DllImport("dwmapi.dll", PreserveSig:=True)>
Friend Shared Function DwmSetWindowAttribute(hwnd As IntPtr,
dwAttributeToSet As DwmWindowAttribute,
pvAttributeValue As IntPtr,
cbAttribute As UInteger
) As Integer
End Function
<DllImport("Dwmapi.dll")>
Public Shared Function DwmSetIconicThumbnail(ByVal hwnd As IntPtr,
ByVal hBitmap As IntPtr,
ByVal flags As Integer
) As Integer
End Function
Private Sub Form1_Shown() Handles MyBase.Shown
bmp = DirectCast(Bitmap.FromFile("C:\Image.jpg"), Bitmap)
hBitmap = bmp.GetHbitmap()
hwnd = Process.GetCurrentProcess.MainWindowHandle
Dim block As IntPtr = Marshal.AllocHGlobal(4)
Dim value As Integer = Math.Abs(CInt(True)) ' or 1
Marshal.WriteInt32(block, value)
hresult = DwmSetWindowAttribute(hwnd, DwmWindowAttribute.HasIconicBitmap, block, 4)
If (hresult <> 0) Then
Throw Marshal.GetExceptionForHR(hresult)
End If
hresult = DwmSetWindowAttribute(hwnd, DwmWindowAttribute.ForceIconicRepresentation, block, 4)
If (hresult <> 0) Then
Throw Marshal.GetExceptionForHR(hresult)
End If
Finally
Marshal.FreeHGlobal(block)
End Try
hresult = DwmSetIconicThumbnail(hwnd, hBitmap, DisplayThumbnailFrame)
If (hresult <> 0) Then
Throw Marshal.GetExceptionForHR(hresult)
End If
End Sub
发布于 2016-05-21 15:14:29
根据 MSDN文档
应用程序通常在收到窗口的DwmSetIconicThumbnail消息后调用WM_DWMSENDICONICTHUMBNAIL函数。缩略图不应超过该消息中指定的最大x坐标和y坐标。缩略图也必须有一个32位的颜色深度.
因此,使用以下32乘32位图,具有32位的颜色深度,它工作了:
例外就消失了。但是,它并没有完全替换应用程序图标缩略图,而是追加了它。
这就是使用ALT+TAB的样子:
当它在上空盘旋时:
请注意,我根本没有修改您的代码并按原样运行它,只是使用了一个合适的
Bitmap
。这些是Windows 10机器的结果。
更新
因为
DwmSetIconicThumbnail
函数返回错误是因为图像超过缩略图的最大大小,也就是说,调整大小不是由Windows本身处理的,所以我们需要做更多的工作。
我不确定是哪个因素决定了我们可以为图像建立的最大缩略图大小,这是推测,但我认为这取决于一个Windows注册表值,该值决定缩略图预览的宽度和高度(对不起,我完全不记得该值的注册表位置,但它可以很容易地搜索到)。
好的,如上所述,我们需要处理
WM_DWMSENDICONICTHUMBNAIL
(
0x323
)消息,因此我们需要重写基本窗口过程(即.一个Win32窗口的
WNDPROC
(一个表单)),然后我们可以从消息参数中检索缩略图创建的最大宽度和高度。
这是一个工作代码示例:
Private Const WM_DWMSENDICONICTHUMBNAIL As Integer = &H323
Protected Overrides Sub WndProc(ByRef m As Windows.Forms.Message)
Select Case m.Msg
Case WM_DWMSENDICONICTHUMBNAIL
Dim hwnd As IntPtr = Process.GetCurrentProcess().MainWindowHandle
Dim dWord As Integer = m.LParam.ToInt32()
Dim maxWidth As Short = BitConverter.ToInt16(BitConverter.GetBytes(dWord), 2)
Dim maxHeight As Short = BitConverter.ToInt16(BitConverter.GetBytes(dWord), 0)
Using img As Image = Bitmap.FromFile("C:\Image.jpg")
Using thumb As Bitmap = CType(img.GetThumbnailImage(maxWidth, maxHeight, Nothing, Nothing), Bitmap)
Dim hBitmap As IntPtr = thumb.GetHbitmap()
Dim hresult As Integer = NativeMethods.DwmSetIconicThumbnail(hwnd, hBitmap, 0)
If (hresult <> 0) Then
' Handle error...
' Throw Marshal.GetExceptionForHR(hresult)
End If
NativeMethods.DeleteObject(hBitmap)
End Using
End Using
End Select
' Return Message to base message handler.
MyBase.WndProc(m)
End Sub
作为最后一条评论,如果将来我需要记住这一点,我将分享我在MSDN上发现的问题,这对于有
WM_DWMSENDICONICTHUMBNAIL
消息问题的人是有帮助的:
发布于 2016-05-21 15:23:26
使用来自 Microsoft参考源 的稍微修改的声明(以提供一个返回值),我能够使它像预期的那样发挥作用。
internal static class NativeMethods
[DllImport("dwmapi.dll")]
public static extern int DwmSetIconicThumbnail(IntPtr hwnd, IntPtr hbmp, DWM_SIT dwSITFlags);
[DllImport("dwmapi.dll")]
public static extern int DwmSetWindowAttribute(IntPtr hwnd, DWMWA dwAttribute, ref int pvAttribute, int cbAttribute);
public enum DWM_SIT
None,
DISPLAYFRAME = 1
public enum DWMWA
NCRENDERING_ENABLED = 1,
NCRENDERING_POLICY,
TRANSITIONS_FORCEDISABLED,
ALLOW_NCPAINT,
CAPTION_BUTTON_BOUNDS,
NONCLIENT_RTL_LAYOUT,
FORCE_ICONIC_REPRESENTATION,
FLIP3D_POLICY,
EXTENDED_FRAME_BOUNDS,
// New to Windows 7:
HAS_ICONIC_BITMAP,
DISALLOW_PEEK,
EXCLUDED_FROM_PEEK
// LAST
public const uint TRUE = 1;
}
然后,我修改了您现有的C#代码以适应这些签名。
private void Form1_Shown(object sender, EventArgs e)
bmp = (Bitmap)Bitmap.FromFile("C:\\Image.jpg");
hBitmap = bmp.GetHbitmap();
hwnd = Process.GetCurrentProcess().MainWindowHandle;
int attributeTrue = (int)NativeMethods.TRUE;
hresult = NativeMethods.DwmSetWindowAttribute(hwnd, NativeMethods.DWMWA.HAS_ICONIC_BITMAP, ref attributeTrue, sizeof(int));
if ((hresult != 0))
throw Marshal.GetExceptionForHR(hresult);
hresult = NativeMethods.DwmSetWindowAttribute(hwnd, NativeMethods.DWMWA.FORCE_ICONIC_REPRESENTATION, ref attributeTrue, sizeof(int));
if ((hresult != 0))
throw Marshal.GetExceptionForHR(hresult);
hresult = NativeMethods.DwmSetIconicThumbnail(hwnd, hBitmap, NativeMethods.DWM_SIT.DISPLAYFRAME);
if ((hresult != 0))
throw Marshal.GetExceptionForHR(hresult);
踏实的水煮鱼 · Unix : 查找和替换连续的逗号至连续的管道 1 年前 |