c#透明悬浮球实现 从零开始用C#写一个桌面应用程序(三)

news/2024/9/14 2:15:26/ 标签: c#, 开发语言

目标:透明悬浮球

记录日期:20240308

要求基础:C#语言基础部分事件与委托,c#桌面程序基础操作     注:可见前文

http://t.csdnimg.cn/9uWK8

 今天开始做一个悬浮球软件。本以为最难的是让悬浮球的具体功能,没想到卡在如何让悬浮球变成一个完整圆形并且实现透明这件事情上了。

创建悬浮球

创建两个c#界面分别对应悬浮球和点击之后打开的菜单。

变成圆形

看看我们现在的这个界面,我们使用什么方法把他变成圆形。

//在类中添加以下代码
int dheight = 136;
int dwidth = 136;
private ball functionForm; private void BallForm_Paint(object sender, PaintEventArgs e){e.Graphics.SmoothingMode = System.Drawing.Drawing2D.SmoothingMode.HighQuality;e.Graphics.FillEllipse(new SolidBrush(Color.Yellow), 0, 0, dwidth, dheight);}//在界面的构造函数中添加下面代码
//用处是为我们的绘制函数添加我们自定义的绘制方法
this.Paint += new PaintEventHandler(this.BallForm_Paint);
this.ClientSize = new Size(136, 136);

也就是我们是通过加写paint函数来绘制圆形界面的。

  • SmoothingMode.HighQuality 设置使得图形平滑,看起来更清晰。
  • FillEllipse 方法用来绘制一个椭圆,这里画的是一个黄色的圆形,其位置是 (0, 0),即左上角,并且宽度和高度分别是 dwidth 和 dheight

请注意这里我们是在绘制事件中添加了绘制方法而不是重写了绘制方法。这对后续添加的控件有重要意义。 

使他透明

我们的圆形界面怎么让他变得透明? 

查询之后,发现实现透明的方法很多,但是各有限制,我选择使用c#提供的一个属性opacity。

 然而,这样我们的界面仍然保留旧的绘制函数。你会发现最后窗口内还有边角是版灰色。

这时候轮到另外一个角色transparentkey了。

this.TransparencyKey = this.BackColor;

 设置大小

为什么我们设置悬浮球这么大?136*136.

为什么不使用70*70呢?按照其他人的解释是由于:

Winforms在设置Form宽度的Form.set_Width中,我们会把试图设置的大小和SystemInformation.MinWindowTrackSize进行比较。Winforms不允许Form的大小比MinWindowTrackSize小。也就是说MinWindowTrackSize规定了Form最小的大小。如果我们设置的值比MinWindowTrackSize要小,Winforms会把Form的大小设置为MinWindowTrackSize。

SystemInformation.MinWindowTrackSize的大小随着Windows的系统设置的改变而改变。

事实上我之前绘制的图形是可以很小的,但是发现这样实际的外围窗口大小还是那么大。 

 也就是最后计算宽度和高度的时候你会发现获取的数据是错误的。宽度被死死限制在136.

我暂时没找到比较好的方法,只能针对这个情况进行修补。如果你想修改悬浮球的大小,可以修改上文中的clientsize和绘制方法中的宽高,最后视觉效果上是与你代码相符的,但是不要忘记了,你悬浮球的实际大小是136的宽度(我是2k屏幕,我猜测根据实际情况可能有所不同)

所以我这里为了让实际窗口大小和展示大小一致,选择了这一个比较奇怪的宽高。

添加拖动逻辑

 

private void AddMouseHandlers(Control control)
{control.MouseDown += new MouseEventHandler(BallForm_MouseDown);control.MouseMove += new MouseEventHandler(BallForm_MouseMove);control.MouseUp += new MouseEventHandler(BallForm_MouseUp);// 递归地为所有子控件添加事件处理器foreach (Control childControl in control.Controls){AddMouseHandlers(childControl);}
}

这里很好理解,鼠标按下,鼠标移动,鼠标松起事件,然后为自控件也添加鼠标事件。

聪明的同学应该想到:后面的点击事件应该也放在类似的递归中来添加给子控件。

实现粘连

我们的悬浮球移动到屏幕边缘需要隐藏一半以免影响我们的使用,这种情况怎么办?快来使用位置改变的事件绑定到我们的悬浮球这里吧!

 private void BallForm_LocationChanged(object sender, EventArgs e){if (isSticking) return; // 如果窗体正在靠边粘连,那么我们不需要再次检查其位置if (this.Left <= StickGap) // 窗体靠近屏幕的左边缘{isSticking = true;this.Left = -this.Width / 2;}else if (this.Right >= Screen.PrimaryScreen.Bounds.Width - StickGap) // 窗体靠近屏幕的右边缘{isSticking = true;this.Left = Screen.PrimaryScreen.Bounds.Width - this.Width / 2;}else if (this.Top <= StickGap) // 窗体靠近屏幕的上边缘{isSticking = true;this.Top = -this.Height / 2;}else if (this.Bottom >= Screen.PrimaryScreen.Bounds.Height - StickGap) // 窗体靠近屏幕的下边缘{isSticking = true;this.Top = Screen.PrimaryScreen.Bounds.Height - this.Height / 2;}}

添加子窗口

点击悬浮球弹出一个界面是我们最终展现功能的地方,子窗体创建很简单,就把他作为我们悬浮球窗体的一个子但是你会发现,这和我们前文的事件有所冲突:

鼠标的点击事件和鼠标的mousedown事件到底谁先触发呢?很现实的问题就是如果先触发点击事件,我们就要避免点击事件影响拖动逻辑,如果先触发鼠标按下事件,就是避免拖动逻辑影响我们的点击事件了。事实上不用担心两个中间会触发一个不触发另外一个。

只是先后次序影响我们处理的逻辑。我们发现是:mousedown-mousemove-mouseup-mouseclick

然后我们就发现,我们拖拽鼠标的时候会导致子界面被打开,这不是我们希望的。于是设立标志:

 private bool dragging = false;private bool isclick = false;private Point dragCursorPoint;private Point dragFormPoint;private const int DragThreshold = 5; // 你可以根据需要调整这个值

在鼠标移动事件处理中添加以下代码: 

Point dif = Point.Subtract(Cursor.Position, new Size(dragCursorPoint));// 检查鼠标是否移动了一定的距离
if (Math.Abs(dif.X) > DragThreshold || Math.Abs(dif.Y) > DragThreshold)
{isclick = false;
}

 我们的逻辑就是:如果拖动超过了一定像素,我们才对其触发mouseclick事件:打开子界面。

 private void BallForm_MouseClick(object sender, MouseEventArgs e){// 如果不是点击操作,那么就不执行点击操作的代码if (isclick==false){return;}// 检查是哪个鼠标按钮被点击if (e.Button == MouseButtons.Left){// 如果 functionForm 窗体当前是显示状态,那么就隐藏它// 如果 functionForm 窗体当前是隐藏状态,那么就显示它if (functionForm.Visible){functionForm.Hide();}else{// 计算 functionForm 窗体的位置int x = this.Location.X + this.Width;int y = this.Location.Y;// 检查 functionForm 窗体是否会被屏幕右边缘遮挡,如果会,那么就调整它的 X 坐标if (x + functionForm.Width > Screen.PrimaryScreen.WorkingArea.Width){x = Screen.PrimaryScreen.WorkingArea.Width - functionForm.Width;}// 检查 functionForm 窗体是否会被屏幕下边缘遮挡,如果会,那么就调整它的 Y 坐标if (y + functionForm.Height > Screen.PrimaryScreen.WorkingArea.Height){y = Screen.PrimaryScreen.WorkingArea.Height - functionForm.Height;}// 设置 functionForm 窗体的位置并显示它functionForm.Location = new Point(x, y);functionForm.Show();}}else if (e.Button == MouseButtons.Right){// 右键被点击// 在这里添加你的代码}}

添加文字

添加文字到悬浮球这一操作应当是比较正常的操作,那么我们怎么做才能实现呢?

首先在控件中拖入一个label,此时他应当在你拖动的位置,但是当我们重新绘制以及重新设置了工作区之后,它可能在我们的悬浮球之外,也就不可见了。

首先要保证它的位置在我们的悬浮球内,修改它的location属性。

文字居中

注意autosize属性改为false,因为我们之后要利用它自带的文字对齐来实现文字居中。

当然了这里提前给出其他解决方式:修改location到悬浮球中间。以及修改margin控制左上外间距。这在我们实现悬浮球大小变化之时会有不同的影响。

 lb_yizhu.Font = new Font(lb_yizhu.Font.FontFamily, 13);lb_yizhu.TextAlign = ContentAlignment.MiddleCenter;lb_yizhu.Width = 136;  // 设置为你需要的宽度lb_yizhu.Height = 136;  // 设置为你需要的高度lb_yizhu.BackColor = BallForm.DefaultBackColor;lb_yizhu.BackColor = this.BackColor;//lb_yizhu.Location = new Point(50,37);lb_yizhu.BackColor = Color.Transparent;//lb_yizhu.Location = new Point(35, 35);

实现效果

这里没有使用transparentkey展示。使用之后实际效果如下图所示:

这就是悬浮窗的全部内容,具体功能在子窗体,我这里是:

 

this.lb_yizhu.MouseClick += new System.Windows.Forms.MouseEventHandler(this.BallForm_MouseClick);this.MouseClick += new System.Windows.Forms.MouseEventHandler(this.BallForm_MouseClick);

这里就回收刚刚的问题了:我们的鼠标点击事件没有递归添加的原因是因为之后click事件最后会作用在文本身上。当然了如果你使用的是其他方法改变的文本居中,并没有改变文本本身大小的话,这里就需要把上述两个代码都写在代码里或者用事件的方式绑定在控件上了。

 

 

 可以我们绑定的事件显示在这里啦:在设计代码中修改才会出现在这里哦。

子功能

悬浮剪贴板

意为在程序运行期间可以持续监测用户的复制与剪切板操作,从而形成列表供用户浏览和再一次使用。功能的话是使用了winapi。具体的实现就不多赘述。 

 此为:ClipboardWatcher.cs

using System;
using System.Runtime.InteropServices;
using System.Windows.Forms;
using System.Collections.Generic;public class ClipboardWatcher : NativeWindow
{private const int WM_CLIPBOARDUPDATE = 0x031D;private List<DataObject> clipboardHistory = new List<DataObject>();[DllImport("user32.dll", SetLastError = true)][return: MarshalAs(UnmanagedType.Bool)]public static extern bool AddClipboardFormatListener(IntPtr hwnd);[DllImport("user32.dll", SetLastError = true)][return: MarshalAs(UnmanagedType.Bool)]public static extern bool RemoveClipboardFormatListener(IntPtr hwnd);public ClipboardWatcher(){CreateHandle(new CreateParams());AddClipboardFormatListener(Handle);}protected override void WndProc(ref Message m){if (m.Msg == WM_CLIPBOARDUPDATE){try{DataObject clipboardData = Clipboard.GetDataObject() as DataObject;if (clipboardData != null){clipboardHistory.Add(clipboardData);}}catch (Exception ex){// 处理错误Console.WriteLine("An error occurred while accessing the clipboard: " + ex.Message);}}base.WndProc(ref m);}public void Stop(){RemoveClipboardFormatListener(Handle);}public List<DataObject> GetClipboardHistory(){return clipboardHistory;}
}

这就不属于我们讲的c#桌面程序设计了,需要对具体的情况,具体的功能实际实现来进行各类的对接和查询。

期待的功能比如实现展示内存利用率,清理内存等,这些都是顺水推舟的事情,同学们可以自行实现。

完整代码后续可能上传github(毕竟实在太简单了!)


http://www.ppmy.cn/news/1516600.html

相关文章

【Qt】常见控件 —— QPushButton | QRadioButton

文章目录 QPushButtonQPushButton 的基本功能介绍QPushButton 添加快捷键通过图片实现 上下左右实现方向键的槽函数设置快捷键连发功能 QRadioButtonQRadioButton 的基本功能介绍通过 QRadioButton 选择性别具有排他效果禁用 选项 槽函数的使用情况基于 QRadioButton 实现一个简…

爬虫使用代理IP:提升数据抓取效率的实践

爬虫使用代理IP的技巧和方法 在进行网络爬虫时&#xff0c;使用代理IP可以帮助你提高数据抓取效率和保护隐私。本文将介绍爬虫使用代理IP的技巧和方法&#xff0c;帮助你更好地进行数据抓取。 为什么爬虫需要使用代理IP 在进行大规模数据抓取时&#xff0c;目标网站可能会检…

数据仓库: 4- 数据质量管理 5- 元数据管理

目录 4- 数据质量管理4.1 数据清洗4.1.1 数据清洗的重要性4.1.2 数据清洗常见的问题4.1.3 数据清洗的步骤4.1.3.1 数据质量评估:4.1.3.2 制定清洗规则:4.1.3.3 执行清洗操作:4.1.3.4 验证清洗结果:4.1.3.5 迭代优化: 4.1.4 数据清洗的常用方法4.1.5 数据清洗的最佳实践4.1.6 总…

外贸管理系统采购销售报关计算机毕业设计VUE/PYTHON/MYSQL

开发一个基于Vue、Python和MySQL的外贸管理系统&#xff0c;用于处理采购、销售以及报关等业务流程。这样的系统通常涉及前端界面展示、后端逻辑处理以及数据库存储等多个部分。下面是一些关键组件的设计建议&#xff1a;1. 技术栈选择 前端: Vue.js 后端: Python (Flask/Djang…

大数据分析与挖掘技术实训室解决方案

一、大数据分析与挖掘技术概述 大数据分析与挖掘技术是指运用算法、工具和技术处理和分析大规模数据集的过程。这些数据集来源于多个渠道&#xff0c;例如传感器数据、社交媒体数据、网络日志和金融交易记录等。其目标是在这些海量数据中发现有价值的信息、模式和趋势&#xf…

【MySQL】黑悟空都掌握的技能,数据库隔离级别全攻略

前言 &#x1f34a;缘由 黑神话悟空玩家必备&#xff0c;数据库隔离级别完全解读 &#x1f423;闪亮主角 大家好&#xff0c;我是JavaDog程序狗 今天借着黑神话悟空的热度&#xff0c;跟大家分享一下数据库隔离级别&#xff0c;也是面试必备的八股文 &#x1f608;你想听的…

使用阿里的EasyExcel导入数据

工作遇到一种情况,在导入excel的时候数量过多,导致占用内存太大最终OOM.为了避免这样的情况再次出现,更换easyPoi为EasyExcel,它是一行一行读,非常节省内存且快速. 首先依赖 <dependency><groupId>com.alibaba</groupId><artifactId>easyexcel-core<…

对想学习人工智能或者大模型技术从业者的建议

“ 技术的价值在于应用&#xff0c;理论与实践相结合才能事半功倍” 写这个关于AI技术的公众号也有差不多五个月的时间了&#xff0c;最近一段时间基本上都在保持日更状态&#xff0c;而且写的大部分都是关于大模型技术理论和技术方面的东西。‍‍‍‍‍‍‍‍‍ 然后最近一段…

Mozilla为本地音频到文本翻译开发Whisperfile引擎

Mozilla Ocho 小组正进行 Mozilla 的"创新和实验"。Llamafile 用于将大型语言模型以单个文件的形式发布&#xff0c;以便在不同的硬件/软件间轻松执行。Whisperfile 是一项将音频轻松转化为文本的新引擎。 正如其名称所暗示的&#xff0c;Whisperfile 是围绕 OpenAI…

RabbitMQ如果有100万消息堆积在MQ,如何解决(消息堆积如何解决)面试版

什么情况下产生消息堆积 消息堆积&#xff1a;当生产者发送消息的速度 超过了 消费者处理消息的速度&#xff0c;就会导致队列中的消息堆积。 消息堆积会产生的问题&#xff1a;直到队列存储的消息达到上限。之后发送的消息就会成为死信&#xff0c;可能会被丢弃。 解决消息…

数据库表的nb3和sql后缀的处理方式

后缀是sql的话就直接运行sql文件 就可以把数据库添加到本地了&#xff08;像这样&#xff09; 右键你选择存放的数据库 -- 运行sql文件 -- 选择后缀是sql的文件 如果同事给你了一个后缀是nb3的话 那么就需要你去还原了 你想把这个表加到哪个库下就选择 右键备份 -- 还原备份从…

数据库查询大量数据避免内存溢出的方法

原理就是分批查询。每次查询一定数量数据之后记录id&#xff0c;进行数据处理之后再继续查询继续处理&#xff0c; allFrameObject mapper.findAllFrameObjectByMaxId(minTime, beginRow, 1000); while (CollectionUtils.isNotEmpty(allFrameObject)) {beginRow allFrameObj…

如何构建小学至大学素质评价档案系统 —— php Vue 实践指南

&#x1f34a;作者&#xff1a;计算机毕设匠心工作室 &#x1f34a;简介&#xff1a;毕业后就一直专业从事计算机软件程序开发&#xff0c;至今也有8年工作经验。擅长Java、Python、微信小程序、安卓、大数据、PHP、.NET|C#、Golang等。 擅长&#xff1a;按照需求定制化开发项目…

GPIO(通用输入/输出)、中断(hal库)

目录 GPIO&#xff08;通用输入/输出)&#xff08;hal库&#xff09; GPIO工作模式 推挽输出&#xff08;Push-Pull Output&#xff09; 开漏输出&#xff08;Open-Drain Output&#xff09; 复用推挽输出&#xff08;Alternate Function Push-Pull Output&#xff09; 复…

金融涉案账户压降行动的实施成效与挑战

2024年上半年我国出台了关于金融行业相关管理办法 1 - 5 号令&#xff0c;不断完善相关法律法规&#xff0c;加强对欺诈行为的打击力度。加强了对互联网企业的监管力度&#xff0c;要求企业加强内部管理&#xff0c;建立健全用户信息保护机制&#xff0c;防止用户信息泄露和被滥…

Lora微调训练参数解读

前言 通过前面两次微调训练欺诈文本分类微调&#xff08;六&#xff09;&#xff1a;Lora单卡和欺诈文本分类微调&#xff08;七&#xff09;—— lora单卡二次调优&#xff0c;我们已经初步理解了微调的整个过程&#xff0c;里面涉及到不少的参数配置&#xff0c;这篇文章就对…

redis面试(二十三)写锁释放

先加了写锁&#xff0c;后面再次加写锁或者读锁 anyLock: { “mode”: “write”, “UUID_01:threadId_01:write”: 2, “UUID_01:threadId_01”: 1 } 写锁的释放lua脚本在这里 RedissonWriteLock.unlockInnerAsync() 比如说现在的参数是这 KEYS[1] anyLock KEYS[2] redi…

卖旧电脑前怎么彻底清除数据?卖旧电脑不留隐患

在科技日新月异的今天&#xff0c;电脑已成为我们日常生活和工作中不可或缺的工具。然而&#xff0c;随着技术的不断进步&#xff0c;我们可能会考虑更换新的电脑设备&#xff0c;而将旧的电脑出售或转让。 在卖旧电脑前&#xff0c;彻底清除电脑中的数据至关重要&#xff0c;…

leetcode46:全排列

全排列 给定一个不含重复数字的数组 nums &#xff0c;返回其 所有可能的全排列 。你可以 按任意顺序 返回答案。 List<List<Integer>> list new ArrayList<>();public List<List<Integer>> permute(int[] nums) {LinkedList<Integer> …

Shader 中的渲染顺序

1、深度测试和深度写入 有了深度测试和深度写入发挥作用让我们不需要关心不透明物体的渲染顺序比如一个物体A 挡住了 物体B&#xff0c;即使底层逻辑中 先渲染A&#xff0c;后渲染B&#xff0c;我们也不用担心 B的颜色会把A覆盖&#xff0c;因为在进行深度测试时&#xff0c;远…