1024programmer Asp.Net Explore the underlying implementation of WPF’s ITabletManager.GetTabletCount in Win11 system

Explore the underlying implementation of WPF’s ITabletManager.GetTabletCount in Win11 system

Explore the underlying implementation of WPF’s ITabletManager.GetTabletCount in Win11 system

This article will introduce to you the underlying implementation of the GetTabletCount method of ITabletManager provided for WPF touch module in Windows 11 system.

This article will introduce to you the underlying implementation of the GetTabletCount method of ITabletManager provided specifically for the WPF touch module in Windows 11 system

This article belongs to a series of blogs related to WPF touch, focusing on the bottom layer of the system. For more touch blogs, please see WPF touch related

As we all know, in Windows 7 system, there are dedicated pen and touch services to provide support for touch messages. WPF is a framework that has been around since the Vista era, so it naturally needs to support the XP system. In the XP system, there is no perfect WM_Touch message, and at the same time, performance needs to be taken into consideration. The best solution is RealTimeStylus. There is a set of COM interfaces specifically used for WPF touch modules under Windows. This set of interfaces provides almost the same implementation functions as RealTimeStylus. For details, please see https://learn.microsoft.com/en-us/windows/win32/tablet /com-apis-used-by-windows-presentation-foundation

For more introduction to this COM touch layer, please see the touch COM interface used in WPF

If you are interested in the docking of this COM touch layer in WPF, please refer to how the WPF touch bottom PenImc works

But starting from Win10, there are no dedicated pen and touch services in the system, but touch messages are integrated into the system

This article will talk to you about the touch bottom layer of WPF under Windows 11, that is, where the ITabletManager interface is defined, and how the GetTabletCount method is implemented

Since this can be changed on each system, this article focuses on writing code for debugging and understanding the implementation on Windows 11 22H2 22621 with the help of VisualStudio and IDA

In order to know where the specific implementation DLL of ITabletManager is, you can define the COM interface and learn the corresponding DLL file by getting the virtual function table address of the COM interface

First write the code that defines the ITabletManager interface, the code is as follows

using System;
 using System.Runtime.CompilerServices;
 using System.Runtime.InteropServices;
 using HRESULT = System.Int32;

 [ComImport, Guid("764DE8AA-1867-47C1-8F6A-122445ABD89A"), InterfaceType(ComInterfaceType.InterfaceIsIUnknown)]
 public interface ITabletManager
 {
     int GetDefaultTablet(out ITablet ppTablet);
     int GetTabletCount(out ulong pcTablets);
     int GetTablet(ulong iTablet, out ITablet ppTablet);
 }
 

The above ITablet interface is not the focus of this article. We only need to define the empty interface and do not need to define the methods in it

[ComImport, Guid("1CB2EFC3-ABC7-4172-8FCB-3BC9CB93E29F"), InterfaceType(ComInterfaceType.InterfaceIsIUnknown)]
 public interface ITablet //: IUnknown
 {
 }
 

Then in the code, first create the CLSID_TabletManagerS object and then convert it to the ITabletManager interface

Call CoCreateInstance with a class ID of CLSID_TabletManagerS, and then call QueryInterface to get a pointer to the ITabletManager Interface. The CLSID_TabletManagerS GUID is defined as follows: #define CLSID_TabletManagerS uuid(A5B020FD-E04B-4e67-B65A-E7DEED25B2CF)

The C# code corresponding to the above document is as follows

 var typeFromClsid = Type.GetTypeFromCLSID(new Guid("A5B020FD-E04B-4e67-B65A-E7DEED25B2CF"));
             object comObject = Activator.CreateInstance(typeFromClsid);

             var manager = comObject as ITabletManager;
             manager!.GetTabletCount(out var tabletCount);
 

Enable native debugging, run the code, and set a breakpoint at the last sentence of the above code. After entering the breakpoint, you can expand the native view of comObject and find the of the COM object. __vfptr address. Then according to the address, find the DLL file falling within the address range from the debugging module of VisualStudio. As shown below

Only when I wrote this did I see that the wisp.dll file has been written in VisualStudio. I don’t need to calculate the address myself, which is convenient

I learned that the current ITabletManager is defined in the C:\Windows\System32\wisp.dll file. You can throw this file into IDA and decompile it, as shown below

You can see that the GetPointerDevices method is used in line 53. I feel that this is the core implementation. This GetPointerDevices is a method to obtain the number of touch devices under the WM_Pointer touch series under Win10

That is to say, the core implementation of GetTabletCount of ITabletManager is in the POINTER mechanism again. This is beyond the scope of this article, but you can know that the underlying GetTabletCount of ITabletManager is also based on the POINTER mechanism.�Enough for me to play with. Because this reflects that Win11 is not retaining the old code, but just API redirection and adding compatible code. In other words, if a bug exists in the Pointer layer, then WPF’s COM touch layer will also exist. But the converse is not true. If there is a bug in the COM touch layer of WPF, it may be caused by Win11’s API calls or compatibility code, and it may not necessarily be a problem with Pointer

For a description of GetPointerDevices, see https://learn.microsoft.com/en-us/windows/win32/api/winuser/nf-winuser-getpointerdevices

Simple GetPointerDevices usage can be called using PInvoke, as shown in the following example

First install the Microsoft.Windows.CsWin32 library, such as dotnet. Use the CsWin32 library to simplify the Win32 function calling logic. The method provided by the blog

Next write code to obtain touch information from GetPointerDevices

 StringBuilder stringBuilder = ...

                 // Get the number of Pointer devices
                 uint deviceCount = 0;
                 PInvoke.GetPointerDevices(ref deviceCount,
                     (Windows.Win32.UI.Controls.POINTER_DEVICE_INFO*)IntPtr.Zero);
                 Windows.Win32.UI.Controls.POINTER_DEVICE_INFO[] pointerDeviceInfo =
                     new Windows.Win32.UI.Controls.POINTER_DEVICE_INFO[deviceCount];
                 fixed (Windows.Win32.UI.Controls.POINTER_DEVICE_INFO* pDeviceInfo = &pointerDeviceInfo[0])
                 {
                     // You need to get it twice here, the first time to get the quantity, the second time to get the information
                     PInvoke.GetPointerDevices(ref deviceCount, pDeviceInfo);
                     stringBuilder.AppendLine($"PointerDeviceCount:{deviceCount} device list:");
                     foreach (var info in pointerDeviceInfo)
                     {
                         stringBuilder.AppendLine($" - {info.productString}");
                     }
                 }
 

GetPointerDevices needs to be called twice, the first to get the quantity and the second to get the information. When the first parameter passed in to GetPointerDevices is 0, it will not fill in the second parameter array information

The above is the underlying implementation of the GetTabletCount method of ITabletManager provided specifically for the WPF touch module in Windows 11 system

The Blog Garden blog only makes backups and will no longer be updated when the blog is published. If you want to see the latest blog, please go to https://blog.lindexi.com/

Creative Commons License
This work Licensed under a Creative Commons Attribution-NonCommercial-ShareAlike 4.0 International License. You are welcome to reprint, use and republish, but be sure to retain the article signature [Lin Dexi] (https://www.cnblogs.com/lindexi) (including link: https://www.cnblogs.com/lindexi) and may not be used for commercial purposes For this purpose, modified works based on this article must be released under the same license. If you have any questions, please contact me [mailto:lindexi_gd@163.com].

This article is from the internet and does not represent1024programmerPosition, please indicate the source when reprinting:https://www.1024programmer.com/explore-the-underlying-implementation-of-wpfs-itabletmanager-gettabletcount-in-win11-system-2/

author: admin

Previous article
Next article

Leave a Reply

Your email address will not be published. Required fields are marked *

Contact Us

Contact us

181-3619-1160

Online consultation: QQ交谈

E-mail: 34331943@QQ.com

Working hours: Monday to Friday, 9:00-17:30, holidays off

Follow wechat
Scan wechat and follow us

Scan wechat and follow us

Follow Weibo
Back to top
首页
微信
电话
搜索