当前位置: 首页 > news >正文

复制粘贴——QT实现原理

复制粘贴——QT实现原理

QT 剪贴板相关类

QClipboard

对外通用的剪贴板类,一般通过QGuiApplication::clipboard() 来获取对应的剪贴板实例。

// qtbase/src/gui/kernel/qclipboard.h
class Q_GUI_EXPORT QClipboard : public QObject
{Q_OBJECT
private:explicit QClipboard(QObject *parent);~QClipboard();public:enum Mode { Clipboard, Selection, FindBuffer, LastMode = FindBuffer };void clear(Mode mode = Clipboard);bool supportsSelection() const;bool supportsFindBuffer() const;bool ownsSelection() const;bool ownsClipboard() const;bool ownsFindBuffer() const;QString text(Mode mode = Clipboard) const;QString text(QString& subtype, Mode mode = Clipboard) const;void setText(const QString &, Mode mode = Clipboard);const QMimeData *mimeData(Mode mode = Clipboard ) const;void setMimeData(QMimeData *data, Mode mode = Clipboard);QImage image(Mode mode = Clipboard) const;QPixmap pixmap(Mode mode = Clipboard) const;void setImage(const QImage &, Mode mode  = Clipboard);void setPixmap(const QPixmap &, Mode mode  = Clipboard);Q_SIGNALS:void changed(QClipboard::Mode mode);void selectionChanged();void findBufferChanged();void dataChanged();protected:friend class QApplication;friend class QApplicationPrivate;friend class QGuiApplication;friend class QBaseApplication;friend class QDragManager;friend class QPlatformClipboard;private:Q_DISABLE_COPY(QClipboard)bool supportsMode(Mode mode) const;bool ownsMode(Mode mode) const;void emitChanged(Mode mode);
};

QPlatformClipboard

系统剪切板平台接口类,各种桌面平台(Windows,X11,Wayland等)通过这个类提供统一的剪贴板操作接口。

// qtbase/src/gui/kernel/qplatformclipboard.h
class Q_GUI_EXPORT QPlatformClipboard
{
public:virtual ~QPlatformClipboard();virtual QMimeData *mimeData(QClipboard::Mode mode = QClipboard::Clipboard);virtual void setMimeData(QMimeData *data, QClipboard::Mode mode = QClipboard::Clipboard);virtual bool supportsMode(QClipboard::Mode mode) const;virtual bool ownsMode(QClipboard::Mode mode) const;void emitChanged(QClipboard::Mode mode);
};

QXcbClipboard

X11平台实现的剪贴板接口类,继承自QPlatformClipboard,它主要实现了基类的大部分接口,除了emitChanged 这个接口。

// qtbase/src/plugins/platforms/xcb/qxcbclipboard.h
class QXcbClipboard : public QXcbObject, public QPlatformClipboard
{
public:QXcbClipboard(QXcbConnection *connection);~QXcbClipboard();QMimeData *mimeData(QClipboard::Mode mode) override;void setMimeData(QMimeData *data, QClipboard::Mode mode) override;bool supportsMode(QClipboard::Mode mode) const override;bool ownsMode(QClipboard::Mode mode) const override;
...
};

QWindowsClipboard

Windows平台下的剪贴板接口类,继承自QPlatformClipboard

// qtbase/src/plugins/platforms/windows/qwindowsclipboard.h
class QWindowsClipboard : public QPlatformClipboard
{
public:QWindowsClipboard();~QWindowsClipboard();void registerViewer(); // Call in initialization, when context is up.void cleanup();QMimeData *mimeData(QClipboard::Mode mode = QClipboard::Clipboard) override;void setMimeData(QMimeData *data, QClipboard::Mode mode = QClipboard::Clipboard) override;bool supportsMode(QClipboard::Mode mode) const override;bool ownsMode(QClipboard::Mode mode) const override;
...
}

可以看出,同一目录下还有其他各种平台的实现接口:
在这里插入图片描述

QWaylandClipboard

Wayland平台实现的剪贴板接口.

// qtwayland/src/client/qwaylandclipboard_p.h
class Q_WAYLAND_CLIENT_EXPORT QWaylandClipboard : public QPlatformClipboard
{
public:QWaylandClipboard(QWaylandDisplay *display);~QWaylandClipboard() override;QMimeData *mimeData(QClipboard::Mode mode = QClipboard::Clipboard) override;void setMimeData(QMimeData *data, QClipboard::Mode mode = QClipboard::Clipboard) override;bool supportsMode(QClipboard::Mode mode) const override;bool ownsMode(QClipboard::Mode mode) const override;private:QWaylandDisplay *mDisplay = nullptr;QMimeData m_emptyData;
};

QT 剪贴板相关接口

通过查看QClipboard 类的定义,我们比较关心的接口有:

    const QMimeData *mimeData(Mode mode = Clipboard ) const;void setMimeData(QMimeData *data, Mode mode = Clipboard);
Q_SIGNALS:void changed(QClipboard::Mode mode);void selectionChanged();void findBufferChanged();void dataChanged();

获取剪贴板最基础的应该是mimeData 这个接口:

const QMimeData* QClipboard::mimeData(Mode mode) const
{// 获取一个QPlatformClipboard对象,根据不同平台返回的应该是不同的子类,比如x11下就返回的是QXcbClipboardQPlatformClipboard *clipboard = QGuiApplicationPrivate::platformIntegration()->clipboard();if (!clipboard->supportsMode(mode)) return 0;return clipboard->mimeData(mode);
}

可以看出,最终是通过X11接口拿到的。

另外,我们关系剪贴板变化的信号在什么情况下发出来,从实现可以看出,基本是在emitChanged 里发出来的。

/*!\internalEmits the appropriate changed signal for \a mode.
*/
void QClipboard::emitChanged(Mode mode)
{switch (mode) {case Clipboard:emit dataChanged();break;case Selection:emit selectionChanged();break;case FindBuffer:emit findBufferChanged();break;default:break;}emit changed(mode);
}

还有一个地方会通过emitChanged发出变化的信号:

void QPlatformClipboard::emitChanged(QClipboard::Mode mode)
{if (!QGuiApplicationPrivate::is_app_closing) // QTBUG-39317, prevent emission when closing down.QGuiApplication::clipboard()->emitChanged(mode);
}

可以再往下看下谁会调用emitChanged ,可以发现是QPlatformClipboard 的子类QXcbClipboard

// qtbase/src/plugins/platforms/xcb/qxcbclipboard.cpp
void QXcbClipboard::setMimeData(QMimeData *data, QClipboard::Mode mode)
{if (mode > QClipboard::Selection)return;QXcbClipboardMime *xClipboard = 0;// verify if there is data to be cleared on global X Clipboard.if (!data) {xClipboard = qobject_cast<QXcbClipboardMime *>(mimeData(mode));if (xClipboard) {if (xClipboard->isEmpty())return;}}if (!xClipboard && (m_clientClipboard[mode] == data))return;xcb_atom_t modeAtom = atomForMode(mode);xcb_window_t newOwner = XCB_NONE;if (m_clientClipboard[mode]) {if (m_clientClipboard[QClipboard::Clipboard] != m_clientClipboard[QClipboard::Selection])delete m_clientClipboard[mode];m_clientClipboard[mode] = 0;m_timestamp[mode] = XCB_CURRENT_TIME;}if (connection()->time() == XCB_CURRENT_TIME)connection()->setTime(connection()->getTimestamp());if (data) {newOwner = owner();m_clientClipboard[mode] = data;m_timestamp[mode] = connection()->time();}xcb_set_selection_owner(xcb_connection(), newOwner, modeAtom, connection()->time());if (getSelectionOwner(modeAtom) != newOwner) {qWarning("QXcbClipboard::setMimeData: Cannot set X11 selection owner");}emitChanged(mode);
}void QXcbClipboard::handleXFixesSelectionRequest(xcb_xfixes_selection_notify_event_t *event)
{QClipboard::Mode mode = modeForAtom(event->selection);if (mode > QClipboard::Selection)return;// Note1: Here we care only about the xfixes events that come from other processes.// Note2: If the QClipboard::clear() is issued, event->owner is XCB_NONE,// so we check selection_timestamp to not handle our own QClipboard::clear().if (event->owner != owner() && event->selection_timestamp > m_timestamp[mode]) {if (!m_xClipboard[mode]) {m_xClipboard[mode].reset(new QXcbClipboardMime(mode, this));} else {m_xClipboard[mode]->reset();}emitChanged(mode);} else if (event->subtype == XCB_XFIXES_SELECTION_EVENT_SELECTION_CLIENT_CLOSE ||event->subtype == XCB_XFIXES_SELECTION_EVENT_SELECTION_WINDOW_DESTROY)emitChanged(mode);
}

至此,我们可以知道QClipboard是如何发出剪贴板内容变化的信号了:

  1. QClipboard设置剪贴板内容(setMimeData),QXcbClipboard设置完剪贴板内容,emitChanged通知内容变化
  2. QXcbClipboard收到X11剪贴板变化的事件,主动emitChanged通知QClipboard剪贴板变化

总结

qt的剪贴板底层是由各个平台的剪贴板接口驱动的,如果是X11平台,那么整个剪贴板就是X11接口驱动的。关于如何使用X11原生剪贴板接口,参考:https://stackoverflow.com/questions/27378318/c-get-string-from-clipboard-on-linux

相关文章:

复制粘贴——QT实现原理

复制粘贴——QT实现原理 QT 剪贴板相关类 QClipboard 对外通用的剪贴板类&#xff0c;一般通过QGuiApplication::clipboard() 来获取对应的剪贴板实例。 // qtbase/src/gui/kernel/qclipboard.h class Q_GUI_EXPORT QClipboard : public QObject {Q_OBJECT private:explici…...

(一)五种最新算法(SWO、COA、LSO、GRO、LO)求解无人机路径规划MATLAB

一、五种算法&#xff08;SWO、COA、LSO、GRO、LO&#xff09;简介 1、蜘蛛蜂优化算法SWO 蜘蛛蜂优化算法&#xff08;Spider wasp optimizer&#xff0c;SWO&#xff09;由Mohamed Abdel-Basset等人于2023年提出&#xff0c;该算法模型雌性蜘蛛蜂的狩猎、筑巢和交配行为&…...

LED透镜粘接UV胶是一种特殊的UV固化胶,用于固定和粘合LED透镜。

LED透镜粘接UV胶是一种特殊的UV固化胶&#xff0c;用于固定和粘合LED透镜。 它具有以下特点&#xff1a; 1. 高透明度&#xff1a;LED透镜粘接UV胶具有高透明度&#xff0c;可以确保光线的透过性&#xff0c;不影响LED的亮度和效果。 2. 快速固化&#xff1a;经过UV紫外线照射…...

C语言 题目

1.写一个函数算一个数的二进制(补码)表示中有几个1 #include<stdio.h>//统计二进制数中有几个1 //如13:1101 //需要考虑负数情况 如-1 结果应该是32// n 1101 //n-1 1100 //n 1100 //n-1 1011 //n 1000 //n-1 0111 //n 0000 //看n的变化 int funca(int c){int co…...

CDN 内容分发网络

CDN常见问题 什么是 CDN &#xff1f; CDN 全称是 Content Delivery Network/Content Distribution Network&#xff0c;翻译过的意思是 内容分发网络 。 我们可以将内容分发网络拆开来看&#xff1a; 内容&#xff1a;指的是静态资源比如图片、视频、文档、JS、CSS、HTML。…...

Android : Xui- RecyclerView+BannerLayout 轮播图简单应用

实例图&#xff1a; 1.引用XUI http://t.csdnimg.cn/Wb4KR 2.创建显示图片布局 banner_item.xml <?xml version"1.0" encoding"utf-8"?> <LinearLayout xmlns:android"http://schemas.android.com/apk/res/android"xmlns:app"…...

Java网络通信-第21章

Java网络通信-第21章 1.网络程序设计基础 网络程序设计基础涵盖了许多方面&#xff0c;包括网络协议、Web开发、数据库连接、安全性等。 1.1局域网与互联网 局域网&#xff08;LAN&#xff09;与互联网&#xff08;Internet&#xff09;是两个不同的概念&#xff0c;它们分…...

Leetcode 345. Reverse Vowels of a String

Problem Given a string s, reverse only all the vowels in the string and return it. The vowels are ‘a’, ‘e’, ‘i’, ‘o’, and ‘u’, and they can appear in both lower and upper cases, more than once. Algorithm Collect all the vowels and reverse the…...

[linux] 用命令行wget下载google drive的大文件

使用wget命令下载Google drive上的文件_ubuntu上wget下载谷歌云盘文件-CSDN博客 如何用命令行下载Google Drive上的共享文件&#xff1f;-腾讯云开发者社区-腾讯云 举例&#xff1a;https://drive.google.com/drive/folders/1vKj3VvJEKgS_o-uOSmz3I0-GomECpql3 1、在网页上&…...

Docker Network(网络)——8

目录&#xff1a; Docker 为什么需要网络管理Docker 网络架构简介 CNMLibnetwork驱动常见网络类型 bridge 网络host 网络container 网络none 网络overlay 网络docker 网络管理命令 docker network createdocker network inspectdocker network connectdocker network disconne…...

网页设计--第6次课后作业

试用Vue相关指令完成对以下json数据的显示。显示效果如下&#xff1a; 其中&#xff1a;gender1 显示为女&#xff0c;gender2显示为男。价格超过30元&#xff0c;显示“有点小贵”。价格少于等于30元&#xff0c;则显示“价格亲民”。 data: {books: [{"id": "…...

R语言学习

Part1阶段1&#xff1a;入门基础 1安装R和RStudio&#xff1a; 下载并安装R&#xff1a;https://cran.r-project.org/ 下载并安装RStudio&#xff1a;https://www.rstudio.com/products/rstudio/download/ 2Hello World&#xff1a; 学习如何在R中输出"Hello, World!"…...

基于Unity3D 低多边形地形模型纹理贴图

在线工具推荐&#xff1a; 3D数字孪生场景编辑器 - GLTF/GLB材质纹理编辑器 - 3D模型在线转换 - Three.js AI自动纹理开发包 - YOLO 虚幻合成数据生成器 - 三维模型预览图生成器 - 3D模型语义搜索引擎 当谈到游戏角色的3D模型风格时&#xff0c;有几种不同的风格&#xf…...

vue预览pdf,放大缩小拖动,dialog拖动,父页面滚动

公共组件部分代码 main.js import draggable from /directive/drag/index Vue.use(draggable) pdf组件部分代码...

泽攸科技二维材料转移台的应用场景及优势

随着二维材料的广泛研究和各种潜在应用的开发&#xff0c;对于二维材料样品的精密操控与转移的需求日益增加。特别是一些新型二维材料的制备和器件集成制备中&#xff0c;需要在显微镜下对样品进行观察与定位&#xff0c;并能够在微米甚至纳米量级上精确移動和转移样品。 传统…...

JavaScript——基本使用HelloWrold

1. 内部标签 html <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><title>Title</title><!--script标签内&#xff1a;写js代码 也可以写在body标签内--><script>alert(hello,world)&l…...

关于DNS服务器地址总是127.0.0.1且无法解析域名地址

问题 笔者尝试nslookup解释域名时&#xff0c;出现服务器变成本地环回口地址&#xff0c;导致无法解析域名 C:\Users\Zsy>nslookup www.baidu.com 服务器: UnKnown Address: 127.0.0.1*** UnKnown 找不到 www.baidu.com: Server failed排查思路 尝试关闭虚拟网卡&#…...

制作一个RISC-V的操作系统四-嵌入式开发介绍

文章目录 什么是嵌入式开发交叉编译查看一些GCC文件夹 调试器GDB相关语法命令 模拟器QEMUQEMU的安装和使用项目构造工具MakeMakeFile的构成make的运行 练习4-1练习4-2练习4-3 什么是嵌入式开发 程序跑到开发板上&#xff0c;或者说运行到硬件上 交叉编译 简单理解交叉编译来说…...

Python爬虫-实现批量抓取王者荣耀皮肤图片并保存到本地

前言 本文是该专栏的第12篇,后面会持续分享python爬虫案例干货,记得关注。 本文以王者荣耀的英雄皮肤为例,用python实现批量抓取“全部英雄”的皮肤图片,并将图片“批量保存”到本地。具体实现思路和详细逻辑,笔者将在正文结合完整代码进行详细介绍。注意,这里抓取的图片…...

04-详解Eureka注册中心的作用,具体配置,服务注册和服务发现

Eureka注册中心的作用 Eureka架构 远程调用的两个问题 服务的ip地址和端口号写死: 生产环境中服务的地址可能会随时发生变化,如果写死每次都需要重新修改代码多实例问题: 在高并发的情况下一个服务可以有多个实例形成一个集群,此时如果采用硬编码的方式只能访问服务的一个实…...

(LeetCode 每日一题) 3442. 奇偶频次间的最大差值 I (哈希、字符串)

题目&#xff1a;3442. 奇偶频次间的最大差值 I 思路 &#xff1a;哈希&#xff0c;时间复杂度0(n)。 用哈希表来记录每个字符串中字符的分布情况&#xff0c;哈希表这里用数组即可实现。 C版本&#xff1a; class Solution { public:int maxDifference(string s) {int a[26]…...

使用分级同态加密防御梯度泄漏

抽象 联邦学习 &#xff08;FL&#xff09; 支持跨分布式客户端进行协作模型训练&#xff0c;而无需共享原始数据&#xff0c;这使其成为在互联和自动驾驶汽车 &#xff08;CAV&#xff09; 等领域保护隐私的机器学习的一种很有前途的方法。然而&#xff0c;最近的研究表明&…...

376. Wiggle Subsequence

376. Wiggle Subsequence 代码 class Solution { public:int wiggleMaxLength(vector<int>& nums) {int n nums.size();int res 1;int prediff 0;int curdiff 0;for(int i 0;i < n-1;i){curdiff nums[i1] - nums[i];if( (prediff > 0 && curdif…...

质量体系的重要

质量体系是为确保产品、服务或过程质量满足规定要求&#xff0c;由相互关联的要素构成的有机整体。其核心内容可归纳为以下五个方面&#xff1a; &#x1f3db;️ 一、组织架构与职责 质量体系明确组织内各部门、岗位的职责与权限&#xff0c;形成层级清晰的管理网络&#xf…...

页面渲染流程与性能优化

页面渲染流程与性能优化详解&#xff08;完整版&#xff09; 一、现代浏览器渲染流程&#xff08;详细说明&#xff09; 1. 构建DOM树 浏览器接收到HTML文档后&#xff0c;会逐步解析并构建DOM&#xff08;Document Object Model&#xff09;树。具体过程如下&#xff1a; (…...

NFT模式:数字资产确权与链游经济系统构建

NFT模式&#xff1a;数字资产确权与链游经济系统构建 ——从技术架构到可持续生态的范式革命 一、确权技术革新&#xff1a;构建可信数字资产基石 1. 区块链底层架构的进化 跨链互操作协议&#xff1a;基于LayerZero协议实现以太坊、Solana等公链资产互通&#xff0c;通过零知…...

在Ubuntu24上采用Wine打开SourceInsight

1. 安装wine sudo apt install wine 2. 安装32位库支持,SourceInsight是32位程序 sudo dpkg --add-architecture i386 sudo apt update sudo apt install wine32:i386 3. 验证安装 wine --version 4. 安装必要的字体和库(解决显示问题) sudo apt install fonts-wqy…...

MFC 抛体运动模拟:常见问题解决与界面美化

在 MFC 中开发抛体运动模拟程序时,我们常遇到 轨迹残留、无效刷新、视觉单调、物理逻辑瑕疵 等问题。本文将针对这些痛点,详细解析原因并提供解决方案,同时兼顾界面美化,让模拟效果更专业、更高效。 问题一:历史轨迹与小球残影残留 现象 小球运动后,历史位置的 “残影”…...

JavaScript 数据类型详解

JavaScript 数据类型详解 JavaScript 数据类型分为 原始类型&#xff08;Primitive&#xff09; 和 对象类型&#xff08;Object&#xff09; 两大类&#xff0c;共 8 种&#xff08;ES11&#xff09;&#xff1a; 一、原始类型&#xff08;7种&#xff09; 1. undefined 定…...

通过MicroSip配置自己的freeswitch服务器进行调试记录

之前用docker安装的freeswitch的&#xff0c;启动是正常的&#xff0c; 但用下面的Microsip连接不上 主要原因有可能一下几个 1、通过下面命令可以看 [rootlocalhost default]# docker exec -it freeswitch fs_cli -x "sofia status profile internal"Name …...