- android - RelativeLayout 背景可绘制重叠内容
- android - 如何链接 cpufeatures lib 以获取 native android 库?
- java - OnItemClickListener 不起作用,但 OnLongItemClickListener 在自定义 ListView 中起作用
- java - Android 文件转字符串
我正在使用 VSTO 构建一个办公插件。在具有多个不同 DPI 设置的显示器的系统上,我的自定义任务 Pane 的内容在具有较高 DPI 设置的显示器上绘制两次:
只有较小的版本才真正响应用户输入。较大的版本似乎只是一个放大的图像。
我试过各种 DPI 相关设置,例如:
AutoScaleMode
在我的用户控件上。我尝试了所有选项,没有任何改变。SetProcessDpiAwareness
将进程设置为 DPI 感知(或不感知)。我尝试了所有选项,没有任何改变。dpiAware
设置为 true
和 false
。没有变化。新的 Web 插件没有这个问题。此外,内部任务 Pane 没有此问题。
这是一个已知问题吗?我该如何解决这个问题?
最佳答案
这似乎是 Office 产品处理 WM_DPICHANGED
的方式中的错误。信息。该应用程序应该枚举其所有子窗口并重新缩放它们以响应消息,但它无法正确处理加载项 Pane 。
解决此错误的方法是禁用 DPI 缩放。你说你试过调用 SetProcessDpiAwareness
,但一旦为应用程序设置了 DPI 感知,该函数就会失败,而你正在使用的应用程序显然已经设置了它,因为它适用于父窗口。然后你应该做的是调用 SetThreadDpiAwarenessContext
,如 this C# wrapper .不幸的是,我自己没有 Win10 multimon 设置来对此进行测试,但这应该在应用程序运行时起作用。尝试 this add-in ,它有一个按钮来设置线程 DPI 感知上下文,看看它是否适合你。
应用程序 Hook 方法
由于 SetThreadDpiAwarenessContext
在您的系统上可能不可用,处理该问题的一种方法是让主窗口忽略 WM_DPICHANGED
消息。这可以通过安装应用程序 Hook 来更改消息或通过子类化窗口来完成。应用程序 Hook 是一种稍微容易一些且陷阱较少的方法。基本上这个想法是拦截主应用程序的 GetMessage
和 change WM_DPICHANGED
to WM_NULL
,这将使应用程序丢弃该消息。缺点是这种方法仅适用于发布的消息,但 WM_DPICHANGED
应该是其中之一。
因此,要安装应用程序 Hook ,您的加载项代码应如下所示:
public partial class ThisAddIn
{
public enum HookType : int
{
WH_JOURNALRECORD = 0,
WH_JOURNALPLAYBACK = 1,
WH_KEYBOARD = 2,
WH_GETMESSAGE = 3,
WH_CALLWNDPROC = 4,
WH_CBT = 5,
WH_SYSMSGFILTER = 6,
WH_MOUSE = 7,
WH_HARDWARE = 8,
WH_DEBUG = 9,
WH_SHELL = 10,
WH_FOREGROUNDIDLE = 11,
WH_CALLWNDPROCRET = 12,
WH_KEYBOARD_LL = 13,
WH_MOUSE_LL = 14
}
delegate IntPtr HookProc(int code, IntPtr wParam, IntPtr lParam);
[DllImport("user32.dll", SetLastError = true)]
static extern IntPtr SetWindowsHookEx(HookType hookType, HookProc lpfn, IntPtr hMod, uint dwThreadId);
[DllImport("user32.dll")]
static extern IntPtr CallNextHookEx(IntPtr hhk, int nCode, IntPtr wParam, IntPtr lParam);
[StructLayout(LayoutKind.Sequential)]
public struct POINT
{
public int X;
public int Y;
}
public struct MSG
{
public IntPtr hwnd;
public uint message;
public IntPtr wParam;
public IntPtr lParam;
public uint time;
public POINT pt;
}
HookProc cbGetMessage = null;
private UserControl1 myUserControl1;
private Microsoft.Office.Tools.CustomTaskPane myCustomTaskPane;
private void ThisAddIn_Startup(object sender, System.EventArgs e)
{
this.cbGetMessage = new HookProc(this.MyGetMessageCb);
SetWindowsHookEx(HookType.WH_GETMESSAGE, this.cbGetMessage, IntPtr.Zero, (uint)AppDomain.GetCurrentThreadId());
myUserControl1 = new UserControl1();
myCustomTaskPane = this.CustomTaskPanes.Add(myUserControl1, "My Task Pane");
myCustomTaskPane.Visible = true;
}
private IntPtr MyGetMessageCb(int code, IntPtr wParam, IntPtr lParam)
{
unsafe
{
MSG* msg = (MSG*)lParam;
if (msg->message == 0x02E0)
msg->message = 0;
}
return CallNextHookEx(IntPtr.Zero, code, wParam, lParam);
}
private void ThisAddIn_Shutdown(object sender, System.EventArgs e)
{
}
#region VSTO generated code
private void InternalStartup()
{
this.Startup += new System.EventHandler(ThisAddIn_Startup);
this.Shutdown += new System.EventHandler(ThisAddIn_Shutdown);
}
#endregion
}
请注意,这大部分是未经测试的代码,如果它可以阻止 WM_DPICHANGED
消息,您可能必须确保在应用程序退出之前通过删除 Hook 进行清理。
子类化方法
如果您要阻止的消息没有发布到窗口,而是发送了,应用程序 Hook 方法将不起作用,主窗口必须是subclassed instead。 .这次我们将代码放在用户控件中,因为在调用 SetWindowLong
之前需要完全初始化主窗口。
因此,为了对 Power Point 窗口进行子类化,我们的用户控件(在插件中)看起来像这样(请注意,我为此使用了 OnPaint,但您可以使用任何东西,只要它保证窗口初始化为SetWindowLong
调用时间:
public partial class UserControl1 : UserControl
{
const int GWLP_WNDPROC = -4;
[DllImport("user32", SetLastError = true)]
extern static IntPtr CallWindowProc(IntPtr lpPrevWndFunc, IntPtr hWnd, uint uMsg, IntPtr wParam, IntPtr lParam);
[DllImport("user32", SetLastError = true)]
extern static IntPtr SetWindowLongPtr(IntPtr hWnd, int nIndex, IntPtr lpNewLong);
[DllImport("user32", SetLastError = true)]
extern static IntPtr SetWindowLong(IntPtr hWnd, int nIndex, IntPtr lpNewLong);
delegate IntPtr WindowProc(IntPtr hwnd, uint uMsg, IntPtr wParam, IntPtr lParam);
private IntPtr origProc = IntPtr.Zero;
private WindowProc wpDelegate = null;
public UserControl1()
{
InitializeComponent();
this.Paint += UserControl1_Paint;
}
void UserControl1_Paint(object sender, PaintEventArgs e)
{
if (origProc == IntPtr.Zero)
{
//Subclassing
this.wpDelegate = new WindowProc(MyWndProc);
Process process = Process.GetCurrentProcess();
IntPtr wpDelegatePtr = Marshal.GetFunctionPointerForDelegate(wpDelegate);
if (IntPtr.Size == 8)
{
origProc = SetWindowLongPtr(process.MainWindowHandle, GWLP_WNDPROC, wpDelegatePtr);
}
else
{
origProc = SetWindowLong(process.MainWindowHandle, GWLP_WNDPROC, wpDelegatePtr);
}
}
}
//Subclassing
private IntPtr MyWndProc(IntPtr hwnd, uint uMsg, IntPtr wParam, IntPtr lParam)
{
if (uMsg == 0x02E0) //WM_DPICHANGED
return IntPtr.Zero;
IntPtr retVal = CallWindowProc(origProc, hwnd, uMsg, wParam, lParam);
return retVal;
}
}
关于c# - 多 DPI 系统上的 VSTO 自定义任务 Pane 显示两次内容,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/50100564/
我正在尝试设置一个具有三个分割 Pane 的用户界面。前两个是垂直 Pane ,位于屏幕的左侧和右侧。每个拆分的一侧都有一个标题 Pane 。用户可以从这些 Pane 中选择项目以包含在中央 Pane
我经常使用 SQL Server Management Studio,并且始终使用以下 Pane 来处理数据:图表、条件和 SQL。 SQL Panes that I always use 目前,我每
可能是一个有点愚蠢的问题。问题很简单,我倾向于有两个 Pane ,一个用于编写代码,另一个用于编译它。我用 windows 终端 (cmd + vim + Mingw) 做到了这一点,但有些事情困扰着
我想出了如何使用 the Working with Map Panes tutorial 在自定义 Pane 中创建多边形和 this Stack Overflow question及其相关的 Fid
如何将输出连接到 paneWithList? PaneWithList 在其 JList 上有一个监听器,以便将所选行输出到控制台。如何将该输出定向到输出上的 JTextPane? PaneWithL
横向和纵向不同布局之间的方向变化似乎很容易,但是我找不到适合我当前情况的解决方案。 我想在横向模式下使用带有两个 fragment 的双 Pane 布局 res\layout-land\dashboa
我在Xcode中使用Swift编程时,当我想使用实用程序 Pane 编辑按钮时,我意识到它变成了“不适用”。属性检查器,连接检查器等全部变为“不适用”,仅快速帮助检查器仍显示。 我已选择 Storyb
我正在 JavaFX 中编写 vector Canvas ( vector 图形, Canvas 上仅显示形状):好吧,这样您就可以四处移动并放大它。 由于翻译时节点左上角的位置发生变化,并且不再响应
我有一个带有中心 Pane 的边框 Pane - VBox。VBox 包含一个 GridPane 和一个 VBox Pane 。 垂直框(垂直框内的垂直框)最初设置为不可见。 我想要做的是,如果将 G
我正在 JavaFX 中编写 vector Canvas ( vector 图形, Canvas 上仅显示形状):好吧,这样您就可以四处移动并放大它。 由于翻译时节点左上角的位置发生变化,并且不再响应
简单地说,我想让 java 做我想做的事情,但我无法理解布局管理中的任何其他自动调整大小到它感觉要做的事情。 我想要的是一个固定高度的“页脚”和顶部“主”区域,以根据窗口的高度自动调整大小。 水平方向
我尝试创建一个在堆栈 Pane 中包含 2 个 Pane 的应用程序。一个 Pane 是主 Pane 并居中,第二个 Pane 较小并停靠在舞台的左下角。 问题是我试过使用“setAlignment”
我已经在我的第一个 android 应用程序中实现了滑动 Pane 布局,但是当我滑动菜单时,操作栏仍然保持静态,我想实现像 Google+ 应用程序那样的东西 在 Stack Overflow 中也
我只是 Java 的初学者。这是我的问题: 我用这段代码创建了一个可滚动的文本字段(1): jZone_Text = new JTextPane(); scrollPane = new JScroll
我在左 Pane 中创建了一个文件夹,其中包含各种图像。它被称为“wikiImages”。如何访问此文件夹或获取此文件夹的路径? 我找到了 Bundle.main.resourcePath!,但这似乎
好吧,我是 JavaFx 的新手,而且我已经很长时间没有使用 Java 了,所以我遇到了很多问题。最大的问题是如何更改该死的 Pane 的背景。 下面是Controller类 //Styling pr
我正在使用 JavaFX 编写一个程序,其中我有 边框 Pane -> 中心 -> VBox -> 滚动 Pane -> 网格 Pane 我希望能够向网格 Pane 添加任意多的行,并让滚动 Pane
我已经尝试使这项工作 2 天了,并且阅读了许多示例和堆栈溢出问题。我是 html 和 css 的新手,这是我的第 3 天。对于我做错了什么的任何提示或见解,以及一般性评论或批评,我们将不胜感激。 我试
我有一个位于另一个 Pane 内的可拖动 Pane 。我想让子 Pane 只能在父 Pane 的边界内拖动,但默认情况下,子 Pane 可以拖动到任何地方。如何解决这个问题。 最佳答案 看看这个dem
我现在正在学习基本的 JavaFX,我不明白我正在阅读的书中的这句话:“不,像文本字段这样的节点只能添加到一个 Pane 中一次。添加一个节点多次或不同的 Pane 将导致运行时错误。”从书上提供的U
我是一名优秀的程序员,十分优秀!