gpt4 book ai didi

c++ - 自定义 QStyledItemDelegate - 将编辑应用于模型

转载 作者:行者123 更新时间:2023-11-28 05:21:34 25 4
gpt4 key购买 nike

在我的项目中,我继承了 QStyledItemDelegate 并从 createEditor 函数返回了一个自定义编辑器。

QWidget* TagEditDelegate::createEditor(QWidget *parent, const QStyleOptionViewItem &option, const QModelIndex &index) const
{
TagEditWidget* tagEditWidget = new TagEditWidget(parent, index.data(Qt::UserRole+4).toInt(), index.data(Qt::UserRole+2).toByteArray(), index.data(Qt::UserRole+3).toByteArray(), index.parent().data(Qt::UserRole+4).toInt() == 9, parent->width());
return tagEditWidget; //tagEditWidget is my custom QWidget
}

编辑完成后,我想将新数据写回模型。所以我覆盖了 setModelData

void TagEditDelegate::setModelData(QWidget * editor, QAbstractItemModel * model, const QModelIndex & index) const
{
TagEditWidget * tagEditWidget = qobject_cast<TagEditWidget*>(editor);
if (!tagEditWidget)
{
QStyledItemDelegate::setModelData(editor, model, index);
return;
}

//Edit model here?
}

这可行,但问题是无论编辑器如何关闭,setModelData 都会被调用。我只想在使用 EndEditHintQAbstractItemDelegate::SubmitModelCache 关闭编辑器时写入新数据。所以我将 closeEditor 信号连接到我制作的名为 editFinished 的插槽。

connect(this, SIGNAL(closeEditor(QWidget*,QAbstractItemDelegate::EndEditHint)), this, SLOT(editFinished(QWidget*,QAbstractItemDelegate::EndEditHint)));

现在我可以通过 EndEditHint 查看编辑器是如何关闭的,以及我是否应该将数据写回模型。但是,setModelDatacloseEditor 信号之前被调用。当最后调用 closeEditor 信号时,如何将数据写回模型?我在这里遗漏了什么吗?

最佳答案

基本答案:

你的概念几乎到最后都还不错。我会专注于 TagEditDelegate::setModelData 方法。

如果您真的不想更新模型中的数据,只需检查它是否没有改变。这意味着当 oldData == newData 时,只需返回; 并跳过模型更新。

附加说明:

看着您创建的编辑器,我的印象是它不包含呈现给用户的单一值。为了更友好地传递参数并更轻松地比较编辑器数据,请考虑为其创建一个单独的 class/struct。所以你可以调用:

new TagEditWidget(parent, editorData, parent->width())

其中 EditorData 是您的类/结构,可以通过单独的函数获取:

EditorData editorData = readEditorData(index);

该函数可以在 setModelData 方法中重用以检查条件:

if (tagEditWidget->getEditorData() == readEditorData(index)) return;

还要避免使用像 Qt::UserRole+2 这样的魔数(Magic Number)。创建您自己的枚举以指定所需的角色。例如:

enum class MyRole
{
Data1 = Qt::UserRole,
Data2,
Data3,
};
根据评论中的讨论

编辑

如果您想要发现用户是否实际上没有以某种方式取消版本,您可以在编辑器或委托(delegate)中覆盖 eventFilter。在构造函数中创建编辑器时调用 installEventFilter。您的 eventFilter 实现可能如下所示:

bool eventFilter(QObject *object, QEvent *event) override
{
if (event->type() == QEvent::KeyPress)
{
const auto key = static_cast<QKeyEvent *>(event)->key();
if (key == Qt::Key_Enter || key == Qt::Key_Return || key == Qt::Key_Tab)
submitted = true;
}
else if (event->type() == QEvent::FocusAboutToChange &&
static_cast<QFocusEvent*>(event)->reason() == Qt::MouseFocusReason)
{
submitted = true;
}
// extetend the conditions (add else if) to include
// events which you might want to treat as submitted

return QLineEdit::eventFilter(object, event);
}

submitted 是在构造函数中初始化为 falsebool 编辑器成员。然后您可以创建一个 getter 方法 isSubmitted() 并准备好在 setModelData 方法中检查状态。

if (tagEditWidget->isSubmitted())
{
// process data or update model here
}

关于c++ - 自定义 QStyledItemDelegate - 将编辑应用于模型,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/41390381/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com