Qt QLineEdit中屏蔽粘贴功能
Qt QLineEdit中屏蔽粘贴功能
(2012-06-05 10:37:21)在Qt4中实现屏蔽粘贴功能有好几种方法,其中最简单的一种就是把控件属性设为叫读 setReadOnly(true),但是这样就不能在控件中输入了。我现在想实现的目标是允许用户手动输入文本,但是不允许用户粘贴文本到控件中。
以QLineEdit为例,默认情况下Qt4提供了3种文本粘贴方式,
1. Ctrl+V
2. 鼠标中键
3. 鼠标右键弹出“上下文菜单”,从中选择粘贴功能
其中屏蔽第3种方式最简单,设置控件的上下文菜单策略为Qt::NoContextMenu即可
但是这个方式有个问题,设置这个选项后就没有上下文菜单了,包括复制、剪切等都没有,但是我只是想屏蔽“粘贴”功能,其它右键菜单的功能还是应该保留的。
其实Qt专门为上下文弹出菜单提供了一个事件处理函数,我们可以继承一个QLineEdit,重写它的contextMenuEvent(QContextMenuEvent *event)函数,来实现屏蔽“粘贴”功能,代码如下:
- {
- QString temp = qApp->clipboard()->text();
- qApp->clipboard()->setText(QString());
- if (QMenu *menu = createStandardContextMenu()) {
- menu->setAttribute(Qt::WA_DeleteOnClose);
- menu->popup(event->globalPos());
- }
- qApp->clipboard()->setText(temp);
- }
这样,在右键弹出的上下文菜单中,先为系统剪切板里的数据作一个备份后清空它,在这个函数的最后再恢复数据。这样,无论系统剪贴板里是否有数据,粘贴功能都是不可用的。其实我们可以模仿默认的createStandardContextMenu()函数来生成自己的右键上下文菜单。
屏蔽Ctrl+V和鼠标中键最好使用事件过滤器来实现,使用事件过滤器的好处是可以为多个控件使用同一个过滤规则,减少编码工作。先直接贴代码吧:
主界面中的事件过滤函数:
- {
- if (target == lineEdit) {
- if (event->type() == QEvent::KeyPress) {
- QKeyEvent *keyEvent = static_cast<QKeyEvent *>(event);
- if (keyEvent->matches(QKeySequence::Paste)) {
- // if (keyEvent->modifiers() == Qt::ControlModifier
- // && keyEvent->key() == Qt::Key_V) {
- qDebug() << "Ctrl + V";
- return true;
- }
- }
- if (event->type() == QEvent::MouseButtonRelease) {
- QMouseEvent *mouseEvent = static_cast<QMouseEvent *>(event);
- if (mouseEvent->button() == Qt::MidButton) {
- qDebug() << "Mouse MidButton Release";
- return true;
- }
- }
- }
- return QWidget::eventFilter(target, event);
- }
屏蔽Ctrl+V有两种方式,使用Qt4内置的QKeySequence和自己判断Qt::ControlModifier和V键按下事件,推荐使用QKeySequence::Paste,因为用户可能会更改系统默认的“粘贴”快捷键。比如在Linux系统下,如果不使用QKeySequence而是自己判断Ctrl+V的话还需要判断Shift+Insert(Qt::ShiftModifier和Qt::Key_Insert)两个键。
经过测试鼠标中健的粘贴功能是在鼠标释放时实现的,所有只要在捕获到鼠标中键的MouseButtonRelease事件直接返回true就可以屏蔽掉默认的粘贴功能了。
在主界面Widget的构造函数中为ui->lineEdit加入事件过滤器:
结尾:相关的QTextEdit等输入控件都可以使用相似的方法来实现屏蔽粘贴功能。
#ifndef QLINEEDITNOPASTE_H #define QLINEEDITNOPASTE_H #include<QLineEdit> class QLineEditNoPaste : public QLineEdit { Q_OBJECT public: //explicit QLineEditNoPaste(); explicit QLineEditNoPaste(QWidget *parent = Q_NULLPTR); protected: /*virtual*/ void contextMenuEvent(QContextMenuEvent *event) override; }; #endif // QLINEEDITNOPASTE_H
#include <QApplication> #include <QClipboard> #include <QMenu> #include <QContextMenuEvent> #include "qlineeditnopaste.h" /* QLineEditNoPaste::QLineEditNoPaste() { } */ QLineEditNoPaste::QLineEditNoPaste(QWidget *parent): QLineEdit(parent) { } void QLineEditNoPaste::contextMenuEvent(QContextMenuEvent *event) { QString temp = qApp->clipboard()->text(); qApp->clipboard()->setText(QString()); if (QMenu *menu = createStandardContextMenu()) { menu->setAttribute(Qt::WA_DeleteOnClose); menu->popup(event->globalPos()); } qApp->clipboard()->setText(temp); }
/******************************************************************************** ** Form generated from reading UI file 'dialog.ui' ** ** Created by: Qt User Interface Compiler version 5.6.0 ** ** WARNING! All changes made in this file will be lost when recompiling UI file! ********************************************************************************/ #ifndef UI_DIALOG_H #define UI_DIALOG_H #include <QtCore/QVariant> #include <QtWidgets/QAction> #include <QtWidgets/QApplication> #include <QtWidgets/QButtonGroup> #include <QtWidgets/QDialog> #include <QtWidgets/QHeaderView> #include "qlineeditnopaste.h" QT_BEGIN_NAMESPACE class Ui_Dialog { public: QLineEditNoPaste *lineEdit; void setupUi(QDialog *Dialog) { if (Dialog->objectName().isEmpty()) Dialog->setObjectName(QStringLiteral("Dialog")); Dialog->resize(400, 300); lineEdit = new QLineEditNoPaste(Dialog); lineEdit->setObjectName(QStringLiteral("lineEdit")); lineEdit->setGeometry(QRect(100, 159, 211, 31)); retranslateUi(Dialog); QMetaObject::connectSlotsByName(Dialog); } // setupUi void retranslateUi(QDialog *Dialog) { Dialog->setWindowTitle(QApplication::translate("Dialog", "Dialog", 0)); } // retranslateUi }; namespace Ui { class Dialog: public Ui_Dialog {}; } // namespace Ui QT_END_NAMESPACE #endif // UI_DIALOG_H
#include "dialog.h" #include "ui_dialog.h" #include "qlineeditnopaste.h" Dialog::Dialog(QWidget *parent) : QDialog(parent), ui(new Ui::Dialog) { ui->setupUi(this); } Dialog::~Dialog() { delete ui; }