在一次需求中,需要使用到Ctrl+ Ctrl - 组合快捷键来缩放画布。这个效果借助WPF 的命令绑定很容易实现,代码如下:
Window.GetWindow(this.View).InputBindings.AddRange(CreateInputBinding());
internal System.Windows.Input.InputBinding[] CreateInputBinding()
{
List<System.Windows.Input.KeyBinding> commands = new List<System.Windows.Input.KeyBinding>();
commands.Add(new System.Windows.Input.KeyBinding(new DelegateCommand(() =>
{
// ctrl + Home快捷键
}), Key.Home, ModifierKeys.Control));
commands.Add(new System.Windows.Input.KeyBinding(new DelegateCommand(() =>
{
// ctrl + end 快捷键
}), Key.End, ModifierKeys.Control));
commands.Add(new System.Windows.Input.KeyBinding(new DelegateCommand(() =>
{
this.myCanvas.ZoomOut();
}), Key.OemPlus, ModifierKeys.Control));
commands.Add(new System.Windows.Input.KeyBinding(new DelegateCommand(() =>
{
this.myCanvas.ZoomIn();
}), Key.OemMinus, ModifierKeys.Control));
return commands.ToArray();
}
这里的ZoomOut 和ZoomIn 方法就是让myCanvas画布形变缩放,缩放过程中有一个类似Android的 Toast提示缩放百分比;
然而就是这个Toast弹窗却影响到了快捷键的使用,问题的原因就是窗体失去了焦点。
于是乎有了下面的办法
InkMessageBox.Info(((int)(_scaleValue * 100)).ToString() + "%", 500, () =>
{
this.focus();
});
这里就是在Toast 效果的回调中把当前Canvas激活,无论是focus(), bringIntoView()或Focusmanager之类的方法都使用到了,可就是没有效果。 如果把鼠标再次在Canvas对象上单击,快捷键又回来了,穷途末路时都想到了Win32 API借助模拟鼠标点击来解决,比如:
[System.Runtime.InteropServices.DllImport("user32")]
private static extern int mouse_event(int dwFlags, int dx, int dy, int cButtons, int dwExtraInfo);
const int MOUSEEVENTF_LEFTUP = 0x0004; // 模拟鼠标左键抬起
const int MOUSEEVENTF_LEFTDOWN = 0x0002; // 模拟鼠标左键按下
mouse_event(MOUSEEVENTF_LEFTDOWN | MOUSEEVENTF_LEFTUP, 100, 100, 0, 0);
其实是没有搞清楚问题本质:我们注册的快捷键是在主窗体,把当前canvas激活没有用,改进下变成如下:
InkMessageBox.Info(((int)(_scaleValue * 100)).ToString() + "%", 500, () =>
{
Application.Current.MainWindow.Activate();
});
问题解决完毕,心情十分愉悦!