从0到1,手把手带你开发截图工具ScreenCap------001实现基本的截图功能

news2024/11/27 16:43:14

ScreenCap---Version:001

说明

image

  • 从0到1,手把手带你开发windows端的截屏软件ScreenCap

  • 当前版本:ScreenCap---001

  • 支持全屏截图

  • 支持鼠标拖动截图区域

  • 支持拖拽截图

  • 支持保存全屏截图

  • 支持另存截图到其他位置

GitHub

  • 仓库master下的ScreenCap项目

  • 若您无法正常访问,每次项目的资源会随文章一同发布,下载压缩包即可,永久免费

  • 压缩包可能较GitHub更新不及时,请谅解

开发环境

  • win10系统

  • 编译器qtcreator4.11.1

  • QT版本:5.14.2

  • C++11

问题解决

需求

  • 提供开始截图的按钮,点击开始截图

  • 在截图界面提供右键菜单选择

  • 菜单实现保存当前的截图

  • 保存全屏截图

  • 截图另存为

  • 全屏截图另存为

  • 退出截图

  • 鼠标可以拖拽截屏区域

  • 图片属性实时计算

结构

image

image

思路

screencapwidget

  • 首先需要创建页面ScreenCapWidget,提供开始截屏,按键设置,默认位置的按钮

  • 首先实现开始截屏的功能,这里不能直接在窗口线程实现,需要单独创建一个screenwidget类实现截屏的主要操作

  • 获取到screenwidget的实例后,应该处理截屏的逻辑了,创建实例的时候直接调用screenwidget父类widget的showFullScreen函数,将screenwidget以全屏的方式显示出来,整个屏幕是当前截屏的操作区域,遮挡其他操作,这里我们重写一下screenwidget的showEvent事件

screenwidget

  • 而这个screenwidget类不应该一直存在,应该是调用开始截屏的时候才开始创建,这里为了保证同一时刻只有一个screenwidget类创建,应该使用单例模式,确保只有一个实例

  • screenwidget创建的时候不需要ui文件,这里我们只需要使用widget里的绘图事件和菜单功能,自己使用代码实现

  • 在头文件里首先维持一个静态的QScopedPointer对象self,用于实现单例模式

  • 定义一个公共的静态接口Instance以实现其他类来生成screenwidget对象

  • 下面来实现类的默认构造函数,提供菜单功能,实现保存当前的截图,保存全屏截图,截图另存为,全屏截图另存为,退出截图的功能

  • 因为screencapwidget调用其fullShowScreen函数,这里重写showEvent函数

  • showEvent函数中,直接获取当前主屏幕的全屏图像保存在fullScreen中,为提示用户截屏开始了,这里获取到全屏对象后,模糊处理全屏,维持一个背景值bgScreen实现背景处理

  • 截屏界面的交互逻辑等会再实现,先处理关键的部分,创建一个myscreen类,实现截屏实现的数据主要逻辑

  • 重写完showEvent后,已经获取到全屏图像了,需要开始处理部分截图了,即处理鼠标事件,首先处理鼠标按下press事件,第一次按下的位置就是起始位置,再根据此时myscreen的STATUS处理对应的事件

  • 处理鼠标移动的事件,如果还在myscreen还在选择状态,那么移动完的位置就是截屏的结束位置,myscreen在移动状态,那么计算偏移量减去移动开始时候的起始位置movPos即可,将偏移量传入myscreen的move函数中,计算move后的截屏区域

  • 主要的鼠标事件处理完了,下面处理release和右键事件

  • 鼠标事件处理完了之后,要截屏的图像的区域我们已经知道了,下面重写paint事件

myscreen

  • 该类主要实现对截屏的数据计算,来给screenwidget重写事件提供详细的数据

  • 这里的类不需要窗口文件,创建纯粹的cpp类即可

  • 需要获得从screenwidget类传入的qsize参数,这里使用带qsize参数的构造函数

  • 首先截屏需要维护屏幕长和宽的值,maxHeight和maxWidth,这里的数据应该是谁调用谁能获取,全部设置为私有属性,还需要设置其getWidth和getHeight方法

  • 还需要维持截屏区域的左上角和右下角的point值leftUpPos和rightDownPos,并设置getLeftUp和getRightDown方法

  • 处理鼠标事件的时候,需要判断当前截图的状态,维持枚举值STATUS,保存选择截屏区域,拖拽截屏,

  • 这里需要实现判断鼠标是否在现有的截屏区域内isInArea和计算移动后的截屏位置的move函数

其他功能

关键代码

注:关键代码只负责解释各部分的逻辑关系,详解看代码注释

  • screencapwidget处理开始截屏的功能,创建screenwidget的唯一实例,并显示全屏窗口

//ScreenWidget全屏显示
    ScreenWidget::Instance()->showFullScreen();
  • 与showFullScreen相关的screenwidget的重写showEvent事件

//重写窗口被显示的事件
void ScreenWidget::showEvent(QShowEvent *)
{
    //设置初始位置
    QPoint point(-1,-1);
    myscreen->setStart(point);
    myscreen->setEnd(point);

    //获取当前屏幕对象
    QScreen* pscreen = QApplication::primaryScreen();
    //调用QScreen的grabwindow进行全屏截图
    *fullScreen = pscreen->grabWindow(0,0,0,myscreen->getWidth(),myscreen->getHeight());

    //设置透明度实现模糊背景
    QPixmap pix(myscreen->getWidth(),myscreen->getHeight());
    pix.fill((QColor(160,160,160,200)));
    bgScreen = new QPixmap(*fullScreen);
    QPainter p(bgScreen);
    p.drawPixmap(0,0,pix);
}
  • screenwidget实现单例模式的主要代码

//定义单例模式,确保截屏的时候只能有一个
ScreenWidget* ScreenWidget::Instance()
{
    //还没有创建实例
    if(self.isNull())
    {
        //加把锁,只能有一个线程访问
        static QMutex mutex;
        //自动加解锁
        QMutexLocker locker(&mutex);
        //再次判断有没有实例,防止等待的时间中有线程获取到实例了
        if(self.isNull())
        {
            self.reset(new ScreenWidget);
        }
    }
    return self.data();

}
  • creenwidget提供的菜单功能

//创建一个菜单文件
    menu = new QMenu(this);
    //添加菜单的功能
    menu->addAction("保存当前的截图",this,SLOT(saveScreen()));
    menu->addAction("保存全屏截图",this,SLOT(saveFullScreen()));
    menu->addAction("截图另存为",this,SLOT(saveScreenOther()));
    menu->addAction("全屏截图另存为",this,SLOT(saveFullOther()));
    menu->addAction("退出截图",this,SLOT(hide()));
  • screenwidget维持myscreen的类,并在screenwidget的构造函数中实例化myscreen类,传入当前屏幕的大小,二者同步生成

myScreen* myscreen;

 //获取屏幕大小
    myscreen = new myScreen(deskGeometry.size());
  • 获取到当前屏幕的qrect对象,调用size函数获取屏幕的size值,使用宏展开式,不单独处理了,需要的时候直接绽开计算

#define deskGeometry qApp->primaryScreen()->geometry()
  • 处理图片移动

void myScreen::move(QPoint p)
{
    //计算move后的四个点坐标
    int lx = leftUpPos.x() + p.x();
    int ly = leftUpPos.y() + p.y();
    int rx = rightDownPos.x() + p.x();
    int ry = rightDownPos.y() + p.y();
    //确保移动后的截屏不会超出屏幕范围
    if(lx < 0)
    {
        lx = 0;
        rx -= p.x();
    }
    if(ly < 0)
    {
        ly = 0;
        ry -= p.y();
    }
    if(rx > maxWidth)
    {
        rx = maxWidth;
        lx -= p.x();
    }
    if(ry > maxHeight)
    {
        ry = maxHeight;
        ly -= p.y();
    }

    //更新移动后的值
    leftUpPos = QPoint(lx,ly);
    rightDownPos = QPoint(rx,ry);
    startPos = leftUpPos;
    endPos = rightDownPos;
}
  • 处理鼠标press

void ScreenWidget::mousePressEvent(QMouseEvent *e)
{
    int status = myscreen->getStatus();
    //选择区域的状态
    if(status == myScreen::SELECT)
    {
        //把鼠标按下的位置设置为开始位置
        myscreen->setStart(e->pos());
    }
    //拖拽截屏
    else if(status == myScreen::MOV)
    {
        //鼠标不在截屏的区域内,是要重新选择截屏区域
        if(myscreen->isInArea(e->pos()) == false)
        {
            //新按下的位置设置为开始位置,并重置状态为选择
            myscreen->setStart(e->pos());
            myscreen->setStatus(myScreen::SELECT);
        }
        //在截屏区域内,是要拖拽截屏
        else
        {
            //开始移动的起始位置就是现在鼠标按下的位置
            movPos = e->pos();
            this->setCursor(Qt::SizeAllCursor);
        }
    }
    this->update();
}
  • 处理鼠标move

void ScreenWidget::mouseMoveEvent(QMouseEvent *e)
{
    //在选择状态
    if(myscreen->getStatus() == myScreen::SELECT)
    {
        myscreen->setEnd(e->pos());
    }
    //在移动状态
    else if(myscreen->getStatus() == myScreen::MOV)
    {
        //计算鼠标偏移量
        QPoint p(e->x() - movPos.x(),e->y() - movPos.y());
        myscreen->move(p);
        movPos = e->pos();//保存上一次鼠标的位置
    }
    //触发窗口的更新,重新绘制屏幕截图和矩形框
    this->update();
}

文章转载自:KanHai1024

原文链接:https://www.cnblogs.com/kanhai1024/p/17883714.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1296034.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

如何排查rpc mount export: RPC: Timed out问题

文章目录 问题描述查看nfs服务是否运行正常如果以上都通过,尝试下面步骤 问题描述 我们将讨论您在 NFS 客户端上看到的 NFS 错误之一的故障排除。在尝试与 NFS 相关的命令时可以看到此错误&#xff0c;如下所示&#xff1a; 通常&#xff0c;当您看到此错误时&#xff0c;您也…

整合消息队列RabbitMQ

为什么使用消息队列MQ&#xff1f; 因为使用消息队列有多个好处&#xff1a;可以实现系统服务的解耦、异步和削峰&#xff1a; 异步通信&#xff1a;消息队列提供了一种异步通信的方式&#xff0c;发送方可以将消息发送到队列中&#xff0c;然后继续执行其他任务&#xff0c;…

vue使用甘特图dhtmlxgantt + gantt.addTaskLayer

效果图&#xff1a; 甘特图 官网地址 gantt安装与使用 vue版---部分功能收费 安装gantt 或 引入文件 npm install dhtmlx-gantt -save或import gantt from "/public/static/dhtmlxgantt/dhtmlxgantt.js"; import "/public/static/dhtmlxgantt/locale/local…

Pytest+Yaml+Excel 接口自动化测试框架的实现示例

一、框架架构 二、项目目录结构 三、框架功能说明 解决痛点&#xff1a; 通过session会话方式&#xff0c;解决了登录之后cookie关联处理框架天然支持接口动态传参、关联灵活处理支持Excel、Yaml文件格式编写接口用例&#xff0c;通过简单配置框架自动读取并执行执行环境一键…

python3.5安装教程及环境配置,python3.7.2安装与配置

大家好&#xff0c;小编来为大家解答以下问题&#xff0c;python3.5安装教程及环境配置&#xff0c;python3.7.2安装与配置&#xff0c;现在让我们一起来看看吧&#xff01; python 从爬虫开始&#xff08;一&#xff09; Python 简介 首先简介一下Python和爬虫的关系与概念&am…

深度学习实战65-人脸检测模型LFFD的搭建,LFFD模型的架构与原理的详细介绍

大家好,我是微学AI,今天给大家介绍一下深度学习实战65-人脸检测模型LFFD的搭建,LFFD模型的架构与原理的详细介绍。LFFD(Light and Fast Face Detector)模型是一种用于人脸检测的深度学习模型,其设计旨在实现轻量级和快速的人脸检测。本文将详细介绍LFFD模型的定义、优点、原…

类人智能体概念、能力与衍生丨AI Agents闭门研讨观点集锦

导读 在智源社区举办的「青源Workshop第27期&#xff1a;AI Agents 闭门研讨会」上&#xff0c;来自英伟达的高级应用科学家王智琳、CAMEL一作李国豪、AutoAgents一作陈光耀&#xff0c;以及相关技术专家们共同参与交流讨论&#xff0c;分享了最新的研究成果&#xff0c;共同探…

人工麝香市场分析:中国市场年需求量超过15吨

人工麝香作为濒危动物药材麝香的替代品&#xff0c;等同天然麝香配方使用。 是国家重大科研成果和保密品种&#xff0c;用人工麝香生产中成药品种近400种&#xff0c;涵盖中成药常用剂型。 是珍稀动物药材代用品研究的重大突破&#xff0c;为其它珍稀动物药材的应用开辟了一条重…

金融量化交易:使用Python实现遗传算法

大家好&#xff0c;遗传算法是一种受自然选择过程启发的进化算法&#xff0c;用于寻找优化和搜索问题的近似解决方案。本文将使用Python来实现一个用于优化简单交易策略的遗传算法。 1.遗传算法简介 遗传算法是一类基于自然选择和遗传学原理的优化算法&#xff0c;其特别适用…

【方法】Excel表格的“限制保护”不想要了,如何取消?

我们知道&#xff0c;Excel表格可以设置“限制保护”&#xff0c;保护文件不被随意更改&#xff0c;那如果后续不需要保护了&#xff0c;如何取消呢&#xff1f; 下面小编来说说Excel表格常用的三种“保护”&#xff0c;是如何取消的。 第一种&#xff0c;Excel表格的工作表或…

第15章:随堂复习与企业真题(File类与IO流)

第15章&#xff1a;随堂复习与企业真题&#xff08;File类与IO流&#xff09; 一、随堂复习 1. File类的使用 File类的一个实例对应着磁盘上的一个文件或文件目录。 ----> “万事万物皆对象”&#xff08;熟悉&#xff09;File的实例化、常用的方法File类中只有新建、删除…

Unity 自定义窗口

放在Editor文件夹下&#xff1b; #if UNITY_EDITORusing System; using UnityEditor; using UnityEngine;namespace EditorCustumTool {/// <summary>/// 自定义窗口/// </summary>public class CustomWindow : EditorWindow{public enum FlagType{Flag1 101,Fl…

Qt内存管理、UI编辑器、客制化组件、弹出对话框、常用部件类

头文件的小技巧 #include <QtWidgets> // 在自动生成的 .h 里面加上此句 适用条件&#xff1a; QT 的内存管理 当父窗体被关闭时&#xff0c;子部件的内存会自动释放。 对象树是一种管理对象生命周期的机制。当一个对象被添加到另一个对象的子对象列表中时&#xff0…

Springboot+AOP+注解实现字段AES+Base64加解密

AOP实现AESBASE64加解密 场景如下&#xff1a; 需要对数据库存储的字段&#xff0c;进行加解密的处理。如果都直接写代码的话&#xff0c;那么代码回冗余很多&#xff0c;所以使用AOP注解去实现。让代码简洁&#xff0c;方便 具体实现如下&#xff1a; 1、依赖 <depende…

C语言搭建项目-学生管理系统(非链表)

、 目录 搭建offer.h文件 搭建offer.c中的main函数 密码登入系统 搭建my_oferr.c中的接口函数 使用帮助菜单接口函数 增加学生信息接口函数 查询学生信息接口函数 删除学生信息接口函数 保存学生信息接口 打开文件fopen 关闭文件fclose 判断是否保存文件fwrite 退出执行文件…

TCP传输层详解(计算机网络复习)

介绍&#xff1a;TCP/IP包含了一系列的协议&#xff0c;也叫TCP/IP协议族&#xff0c;简称TCP/IP。该协议族提供了点对点的连接机制&#xff0c;并将传输数据帧的封装、寻址、传输、路由以及接收方式都予以标准化 TCP/IP的分层模型 在讲TCP/IP协议之前&#xff0c;首先介绍一…

【MATLAB】tvfEMD信号分解+FFT+HHT组合算法

有意向获取代码&#xff0c;请转文末观看代码获取方式~也可转原文链接获取~ 1 基本定义 TVFEMDFFTHHT组合算法是一种结合了总体变分模态分解&#xff08;TVFEMD&#xff09;、傅里叶变换&#xff08;FFT&#xff09;和希尔伯特-黄变换&#xff08;HHT&#xff09;的信号分解方…

配置Smart Link负载分担示例

Smart Link和Monitor Link简介 定义 Smart Link&#xff0c;又叫做备份链路。一个Smart Link由两个接口组成&#xff0c;其中一个接口作为另一个的备份。Smart Link常用于双上行组网&#xff0c;提供可靠高效的备份和快速的切换机制。 Monitor Link是一种接口联动方案&#…

<习题集><LeetCode><链表><2/19/21/23/24>

目录 2. 两数相加 19. 删除链表的倒数第 N 个结点 21. 合并两个有序链表 23. 合并 K 个升序链表 24. 两两交换链表中的节点 2. 两数相加 https://leetcode.cn/problems/add-two-numbers/ public ListNode addTwoNumbers(ListNode l1, ListNode l2) {//head是cur链表头节点…

拼多多选品大作战:通过热词选利润赛道

相信很多人都听过一句话&#xff1a;找对了风口&#xff0c;猪都能飞起来。 我们电商人也应如此&#xff0c;从行业分析到选品都应快速跟上市场节奏。 今天就给大家分享一下如何通过热词来进行一个行业类目的分析与选择。 01 热词是什么 热词通常指的是热搜词和飙升词&#…