WPF:通过捕获鼠标事件来使Canvas中的所有控件都动起来

2023-10-19 02:30

本文主要是介绍WPF:通过捕获鼠标事件来使Canvas中的所有控件都动起来,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

原文地址:https://www.mgenware.com/blog/wp-content/uploads/2012/03/201203301315305256.png

曾经写过一篇文章:WPF:实现一个很简单的设计器(移动和缩放控件),讲了使用WPF中的Adorner来修饰Canvas中的元素从而使它们可以被移动或者缩放。今天讲通过原始事件的方法来使Canvas中的元素可以被移动。

首先就是定义一个Canvas,然后定义Canvas中的控件。接着使用MouseMove,MouseDown,MouseUp这三个Bubble类型的路由事件。我们将会在之后填充这三个事件。

XAML定义:

<Canvas Name="canvas"PreviewMouseMove="canvas_MouseMove"PreviewMouseDown="canvas_MouseDown"PreviewMouseUp="canvas_MouseUp"><Ellipse Fill="GreenYellow"Width="50"Height="50" Canvas.Left="10"Canvas.Top="10"/><Rectangle Fill="Orange"Width="100"Height="40"Canvas.Left="100"Canvas.Top="50"/><Button Canvas.Left="100"Canvas.Top="10"Content="A"/><TextBox Canvas.Left="130"Canvas.Top="20"Width="100"Text="Mgen"/>
</Canvas>

接下来,我们使用一种最原始的方法,首先在MouseDown事件中通过Mouse.DirectlyOver属性确定鼠标指向的Canvas中的控件,然后存下来控件和鼠标点击位置的偏移量,最后在MouseMove事件中设置控件更新的位置,这样控件就被移动了。当然,由于使用原始的鼠标点击事件,对于某些控件比如:Button、TextBox……由于它们对鼠标事件处理的特殊性,所以下面的代码无法移动上述控件(文章下面会给出更好的处理方式)

如下代码:

//选中控件的鼠标位置偏移量
Point targetPoint;//选中控件
UIElement targetElement;private void canvas_MouseDown(object sender, MouseButtonEventArgs e)
{//确定选中控件,然后设置选中控件targetElement = Mouse.DirectlyOver as UIElement;if (targetElement != null)targetPoint = e.GetPosition(targetElement);
}private void canvas_MouseUp(object sender, MouseButtonEventArgs e)
{//鼠标松开后,将选中控件设置成nulltargetElement = null;
}private void canvas_MouseMove(object sender, MouseEventArgs e)
{//确定鼠标左键处于按下状态并且有元素被选中if (e.LeftButton == MouseButtonState.Pressed && targetElement != null){var pCanvas = e.GetPosition(canvas);//设置最终位置Canvas.SetLeft(targetElement, pCanvas.X - targetPoint.X);Canvas.SetTop(targetElement, pCanvas.Y - targetPoint.Y);}
}

上面讲过,上述代码对于大多数控件都表现良好,但是对于某些对鼠标事件有依赖的控件会无法移动,所以我们使用另一种优化的方法,通过使用捕获鼠标事件来移动控件。

 

捕获鼠标通过UIElement.CaptureMouse方法,捕获鼠标后,即使鼠标不在控件上,相应的鼠标事件仍会被控件接受。取消捕获鼠标可以使用Mouse.Capture(null)方法,或者UIElement.ReleaseMouseCapture方法。

代码:(XAML不变)

//选中控件的鼠标位置偏移量
Point targetPoint;private void canvas_MouseDown(object sender, MouseButtonEventArgs e)
{var targetElement = e.Source as IInputElement;if (targetElement != null){targetPoint = e.GetPosition(targetElement);//开始捕获鼠标targetElement.CaptureMouse();}
}private void canvas_MouseUp(object sender, MouseButtonEventArgs e)
{//取消捕获鼠标Mouse.Capture(null);
}private void canvas_MouseMove(object sender, MouseEventArgs e)
{//确定鼠标左键处于按下状态并且有元素被选中var targetElement = Mouse.Captured as UIElement;if (e.LeftButton == MouseButtonState.Pressed && targetElement != null){var pCanvas = e.GetPosition(canvas);//设置最终位置Canvas.SetLeft(targetElement, pCanvas.X - targetPoint.X);Canvas.SetTop(targetElement, pCanvas.Y - targetPoint.Y);}
}

OK,现在所有控件都可以被移动了:

这篇关于WPF:通过捕获鼠标事件来使Canvas中的所有控件都动起来的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/236683

相关文章

使用Navicat工具比对两个数据库所有表结构的差异案例详解

《使用Navicat工具比对两个数据库所有表结构的差异案例详解》:本文主要介绍如何使用Navicat工具对比两个数据库test_old和test_new,并生成相应的DDLSQL语句,以便将te... 目录概要案例一、如图两个数据库test_old和test_new进行比较:二、开始比较总结概要公司存在多

在MyBatis的XML映射文件中<trim>元素所有场景下的完整使用示例代码

《在MyBatis的XML映射文件中<trim>元素所有场景下的完整使用示例代码》在MyBatis的XML映射文件中,trim元素用于动态添加SQL语句的一部分,处理前缀、后缀及多余的逗号或连接符,示... 在MyBATis的XML映射文件中,<trim>元素用于动态地添加SQL语句的一部分,例如SET或W

C#实现获得某个枚举的所有名称

《C#实现获得某个枚举的所有名称》这篇文章主要为大家详细介绍了C#如何实现获得某个枚举的所有名称,文中的示例代码讲解详细,具有一定的借鉴价值,有需要的小伙伴可以参考一下... C#中获得某个枚举的所有名称using System;using System.Collections.Generic;usi

C#实现WinForm控件焦点的获取与失去

《C#实现WinForm控件焦点的获取与失去》在一个数据输入表单中,当用户从一个文本框切换到另一个文本框时,需要准确地判断焦点的转移,以便进行数据验证、提示信息显示等操作,本文将探讨Winform控件... 目录前言获取焦点改变TabIndex属性值调用Focus方法失去焦点总结最后前言在一个数据输入表单

通过C#获取PDF中指定文本或所有文本的字体信息

《通过C#获取PDF中指定文本或所有文本的字体信息》在设计和出版行业中,字体的选择和使用对最终作品的质量有着重要影响,然而,有时我们可能会遇到包含未知字体的PDF文件,这使得我们无法准确地复制或修改文... 目录引言C# 获取PDF中指定文本的字体信息C# 获取PDF文档中用到的所有字体信息引言在设计和出

最好用的WPF加载动画功能

《最好用的WPF加载动画功能》当开发应用程序时,提供良好的用户体验(UX)是至关重要的,加载动画作为一种有效的沟通工具,它不仅能告知用户系统正在工作,还能够通过视觉上的吸引力来增强整体用户体验,本文给... 目录前言需求分析高级用法综合案例总结最后前言当开发应用程序时,提供良好的用户体验(UX)是至关重要

Python中的异步:async 和 await以及操作中的事件循环、回调和异常

《Python中的异步:async和await以及操作中的事件循环、回调和异常》在现代编程中,异步操作在处理I/O密集型任务时,可以显著提高程序的性能和响应速度,Python提供了asyn... 目录引言什么是异步操作?python 中的异步编程基础async 和 await 关键字asyncio 模块理论

禁止平板,iPad长按弹出默认菜单事件

通过监控按下抬起时间差来禁止弹出事件,把以下代码写在要禁止的页面的页面加载事件里面即可     var date;document.addEventListener('touchstart', event => {date = new Date().getTime();});document.addEventListener('touchend', event => {if (new

MySQL数据库宕机,启动不起来,教你一招搞定!

作者介绍:老苏,10余年DBA工作运维经验,擅长Oracle、MySQL、PG、Mongodb数据库运维(如安装迁移,性能优化、故障应急处理等)公众号:老苏畅谈运维欢迎关注本人公众号,更多精彩与您分享。 MySQL数据库宕机,数据页损坏问题,启动不起来,该如何排查和解决,本文将为你说明具体的排查过程。 查看MySQL error日志 查看 MySQL error日志,排查哪个表(表空间

韦季李输入法_输入法和鼠标的深度融合

在数字化输入的新纪元,传统键盘输入方式正悄然进化。以往,面对实体键盘,我们常需目光游离于屏幕与键盘之间,以确认指尖下的精准位置。而屏幕键盘虽直观可见,却常因占据屏幕空间,迫使我们在操作与视野间做出妥协,频繁调整布局以兼顾输入与界面浏览。 幸而,韦季李输入法的横空出世,彻底颠覆了这一现状。它不仅对输入界面进行了革命性的重构,更巧妙地将鼠标这一传统外设融入其中,开创了一种前所未有的交互体验。 想象