- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在使用 QTableWidget
并且需要对编辑进行一些自定义处理,因此我在其上设置了一个 QStyledItemDelegate
。
当用户完成编辑时,信号 closeEditor()
被发出,我连接到它来处理输入的数据。当按下 Enter/Return 时以及用户单击其他地方(在要编辑的 QTableWidgetItem
之外)时都会发出此信号。
我的问题是:是否可以区分用户是按 Return/Enter 还是点击其他地方?我想像按 ESC 键一样处理“外部点击”(没有更改数据并且恢复了 QTableWidgetItem
的原始值)。到目前为止,这两种情况都更改了数据。
QAbstractItemDelegate::EndEditHint
(由 closeEditor()
发出)没有给我该信息。
感谢大家的帮助!
编辑:
要在将数据写回模型之前处理数据,可以实现 setModelData()
,但是,如果通过按 Enter/Return 或通过单击调用此函数,似乎仍然没有办法别的地方……
最佳答案
我最近做了一个样本来(更多)熟悉QStyledItemDelegate
关于单元格的内联编辑:
在这个示例中,我做了底层数据模型重载的更新QStyledItemDelegate::setModelData()
.
读完这个问题后,我刚刚测试了如果通过 ESC 输入完成(或者严格来说:中止)会发生什么。这是我观察到的:
底层数据保持不变。
深入挖掘(即在我重载的 setModelData()
中设置断点),我发现在这种情况下甚至没有调用 setModelData()
。
也许,OP 问题可以通过对他的委托(delegate)进行一些重新设计来轻松解决。
最后,我的派生类(声明)用于实现“全功能”委托(delegate)以内联编辑表格单元格:
class ValueNameDelegate: public QStyledItemDelegate {
// methods:
public:
/// @name Construction & Destruction
//@{
/// constructor.
ValueNameDelegate();
/// destructor.
virtual ~ValueNameDelegate() = default;
// disabled:
ValueNameDelegate(const ValueNameDelegate&) = delete;
ValueNameDelegate& operator=(const ValueNameDelegate&) = delete;
//@}
protected:
/// @name Overloaded Event Handlers
//@{
// inserts editor in table (by setParent(pQParent)) and
// returns the editor widget to edit cell.
virtual QWidget* createEditor(
QWidget *pQParent, const QStyleOptionViewItem &qOption,
const QModelIndex &qMIndex) const override;
// removes editor from table (by setParent(nullptr)).
virtual void destroyEditor(
QWidget *pQEditor, const QModelIndex &qMIndex) const override;
// reads data from table model and updates editor.
virtual void setEditorData(
QWidget *pQEditor, const QModelIndex &qMIndex) const override;
// reads data from editor and updates table model.
virtual void setModelData(
QWidget *pQEditor, QAbstractItemModel *pQModel,
const QModelIndex &qMIndex) const override;
//@}
};
注意:
在我的例子中,只有一个事先创建的编辑器小部件。它被重新用于每个单元格编辑(而不是为每个编辑创建一个新的)。
这与 Qt Spin Box Delegate Example 有点不同但按预期工作。
我无法想象这对 OP 问题没有任何影响。
根据反馈,OP希望处理输入中止之类的失去焦点事件。这与 QLineEdit
中默认的处理方式相反。 .
因此,我的解决方案是提供 QLineEdit
的重载版本.而不是改变 QLineEdit
的行为,它只是在成员 bool _confirmed
中跟踪 Enter 是否被按下。最棘手的部分是确定另一个合适的事件或信号来重置成员。最后,我决定 focusOutEvent()
只是在正确的时间调用,因此适合这项任务。
testQLineEdit-Finished.cc
:
#include <QtWidgets>
class LineEdit: public QLineEdit {
private:
// flag: true ... last finished editing was confirmed
bool _confirmed;
public:
// Construction & Destruction
explicit LineEdit(
const QString &contents = QString(), QWidget *pQParent = nullptr):
QLineEdit(contents, pQParent),
_confirmed(false)
{
QObject::connect(this, &QLineEdit::returnPressed,
[this](){ onSigReturnPressed(); });
}
LineEdit(QWidget *pQParent): LineEdit(QString(), pQParent) { }
virtual ~LineEdit() = default;
LineEdit(const LineEdit&) = delete;
LineEdit& operator=(const LineEdit&) = delete;
public:
// returns whether last finished editing was confirmed.
bool isConfirmed() { return _confirmed; }
protected:
virtual void focusOutEvent(QFocusEvent *pQEvent) override
{
_confirmed = false;
QLineEdit::focusOutEvent(pQEvent);
}
private:
void onSigReturnPressed() { _confirmed = true; }
};
int main(int argc, char **argv)
{
qDebug() << "Qt Version:" << QT_VERSION_STR;
QApplication app(argc, argv);
// setup GUI
LineEdit qEdit(QString::fromUtf8("Hello World"));
qEdit.show();
// install signal handlers
QObject::connect(&qEdit, &LineEdit::editingFinished,
[&]() {
qDebug() << "Edit confirmed:" << qEdit.isConfirmed();
});
// runtime loop
return app.exec();
}
Edit confirmed: true
的输出是通过按 Enter 实现的,false
行是通过单击来实现的。
这很简单,也可以做得更复杂。然而,它显示了用相当少的代码行实现的原理。
关于c++ - QStyledItemDelegate:区分 closeEditor() 或 setModelData() 的原因,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/49207091/
我正在使用Qt 4.7。 我有一个模型,该模型在两列的QTableView中显示,我的目标是在QTableView中提供对该模型的内联编辑。 +-----------------+----------
我有一个代表MyDelegate用于QListWidget .委托(delegate)派生自 QStyledItemDelegate . MyDelegate 的目标之一是在ListWidget 的每
我有一个 QTreeWidget,我想通过使用样式委托(delegate)来完全自定义项目的外观。 我的主要问题是我想在我的项目右侧创建一个自定义按钮,它允许我折叠和展开该项目的子项。经典的“+”按钮
我得到了以下代码,我稍作修改,作为 how to use HTML formatted text in a QListWidget 的答案. main.ui Dialog
我目前正在尝试研究模型 View 方法并编写缩略图查看器应用程序。 在此示例中,我只是尝试绘制 20 个框,但我得到的似乎是随机选择,该选择会随着鼠标移动而更新。滚动使事情变得更糟,并且绘制有时只有框
我是 Python 和 PyQt5 的新手。我正在使用 QStyledItemDelegate 制作仅包含 ComboBox 的 QTableView 列之一。我设法显示了 ComboBox,但我在处
我创建了一个 QStyledItemDelegate 类,我想在其中使一些项目可检查,一些项目带有两个小部件。但它不能正常工作。我错过了什么?这是它的样子: 请看第 1 行,看起来两个小部件在那里,但
在我的一个项目中,我使用 QTableWidget 来显示一些复杂的计算结果。为了提高表格的可读性,我需要在单个表格单元格内显示两个对齐的值。 稍后我想通过使用颜色或箭头等进一步自定义小部件。 为此,
我正在使用 PySide2,但我找不到任何关于如何在 QStyledItemDelegate 子类中使用 paint() 函数的文档。我对类(class)相当陌生,但到目前为止可以理解,但在 PySi
我正在使用双调度创建一个样式化的 QTreeView 来解析数据项的特定委托(delegate),这非常有效。我对 QStyledItemDelegate 的委托(delegate)进行了子类化,以利
我想绘制一个遵循当前样式的自定义项目委托(delegate)。但“WindowsVista/7”风格和“WindowsClassic”风格在文本颜色上存在差异。 我使用以下代码来绘制背景(工作): v
我正在尝试使 QStyledItemDelegate 的行为类似于我编写的自定义 QWidget,因此我可以将我的代码切换到模型/ View 方法。 自定义 QWidget 是一个复杂的按钮,鼠标悬停
在我的项目中,我继承了 QStyledItemDelegate 并从 createEditor 函数返回了一个自定义编辑器。 QWidget* TagEditDelegate::createEdito
我有一个 QTableView,我正在为其设置自定义 QStyledItemDelegate。 除了自定义项绘制之外,我还想为选择/悬停状态设置行的背景色样式。我想要的外观类似于此 KGet 屏幕截图
我有一个 QStyledItemDelegate 派生对象用于 QTableView 派生 View 。我根据模型索引数据类型进一步委托(delegate)绘画和编辑器创建。对于 bool,我想通过复
我想在 QTableWidget 中显示两列,显示两个字符串之间的差异(之前由一些 Levenshtein 距离算法计算)。这些部分存储在每个 QTableWidgetItem 的数据中,作为 QSt
当鼠标悬停在项目委托(delegate)中的文本上时,如何更改鼠标图标?我有这部分,但我找不到任何更改鼠标指针的示例。 我错过了什么? void ListItemDelegate::paint(
我正在尝试使用 Qt,并希望根据模型的值以自定义文本颜色显示模型。以颜色呈现它是一个可选设置,所以我想避免在我的模型中使用 Qt::ForegroundRole,而是在 QStyledItemDele
我应该如何使用委托(delegate)更改我的树项目的背景颜色? 如果不在文本顶部绘画,我一直无法弄清楚如何做到这一点。换句话说,当我使用下面的代码时,颜色绘制在文本之上。文本在背景下... def
我试图使当前代码显示的所有内容都不可编辑。 之前的搜索都建议要么修改模型的flags()函数,要么使用表的setEditTriggers。我在这段代码中同时执行了这两项操作,但它们都不起作用。 查看逐
我是一名优秀的程序员,十分优秀!