模型中的信号/插槽在QTableView中动态插入和删除行

时间:2017-08-25 07:55:38

标签: c++ qt qtableview qt-signals

我在QPushButton的最后一栏插入QTableview。使用该按钮,我将使用按钮释放信号和插槽handlebutton(int)删除该特定行。

cpp代码:

MainWindow::MainWindow(QWidget *parent) :
    QDialog(parent),
    ui(new Ui::MainWindow)
{
    ui->setupUi(this);
    QSortFilterProxyModel *model = new QSortFilterProxyModel(this);
    model = pCApp->guiClient()->getConnectionManagement()->getProxyModel();
    ui->tableView->setModel(model);
    connect(pCApp, SIGNAL(CloseOpenWindowsRequested()), SLOT(close()));
    connect(ui->tableView->model(), SIGNAL(rowsInserted(QModelIndex,int,int)), this, SLOT(onRowsNumberChanged()));
    connect(ui->tableView->model(), SIGNAL(rowsRemoved(QModelIndex,int,int)), this, SLOT(onRowsNumberChanged()));
    ui->tableView->setSortingEnabled(true);
    QPushButton *button;
    QSignalMapper *mapper = new QSignalMapper(this);
    QObject::connect(mapper, SIGNAL (mapped(int)), this, SLOT (handleButton(int)));
    for (int i = 0; i < model->rowCount(); i++)
    {
        button = new QPushButton;
        button->setText("Disconnect " + QString::number(i));
        button->setStyleSheet("QPushButton { color: #E5E5E5; }");
        ui->tableView->setIndexWidget(model->index(i,2, QModelIndex()), button);
        QObject::connect(button, SIGNAL(released()), mapper, SLOT(map()));
        connect(ui->tableView->model(), SIGNAL(rowsInserted(QModelIndex,int,int)), this, SLOT(onRowsNumberChanged()));
        connect(ui->tableView->model(), SIGNAL(rowsRemoved(QModelIndex,int,int)), this, SLOT(onRowsNumberChanged()));
        mapper->setMapping(button, i);
    }
    setAttribute(Qt::WA_DeleteOnClose);
}

MainWindow::~MainWindow()
{
    delete ui;
}

void MainWindow::handleButton(int row)
{
    this->ui->tableView->model()->removeRow(row);
}

void MainWindow::onRowsNumberChanged()
{
    QSortFilterProxyModel *model = new QSortFilterProxyModel(this);
    model = pCApp->guiClient()->getConnectionManagement()->getProxyModel();
    ui->tableView->setModel(model);
    ui->tableView->setSortingEnabled(true);
    QPushButton *button;
    QSignalMapper *mapper = new QSignalMapper(this);
    QObject::connect(mapper, SIGNAL (mapped(int)), this, SLOT (handleButton(int)));
    for (int i = 0; i < model->rowCount(); i++)
    {
        button = new QPushButton;
        button->setText("Disconnect " + QString::number(i));
        button->setStyleSheet("QPushButton { color: #E5E5E5; }");
        ui->tableView->setIndexWidget(model->index(i,2, QModelIndex()), button);
        QObject::connect(button, SIGNAL(released()), mapper, SLOT(map()));
        mapper->setMapping(button, i);
    }

}

hpp代码:

#ifndef MAINWINDOW_HPP
#define MAINWINDOW_HPP

namespace Ui {
class MainWindow;
}

class MainWindow : public QDialog
{
    Q_OBJECT

public:
    explicit MainWindow(QWidget *parent = 0);
    ~MainWindow();
public slots:
    void onLanguageChanged();
    void handleButton(int row);
    void onRowsNumberChanged();

private:
    Ui::MainWindow *ui;
};

#endif // MAINWINDOW_HPP

正常情况下,代码运行正常。但是,当插入新行和/或删除旧行时,按钮不会根据需要出现在最后一列中。我试图使用信号 -

connect(ui->tvServStat->model(), SIGNAL(rowsInserted(QModelIndex,int,int)), this, SLOT(onRowsNumberChanged()));

connect(ui->tvServStat->model(), SIGNAL(rowsRemoved(QModelIndex,int,int)), this, SLOT(onRowsNumberChanged()));

两者的插槽,我与onRowsNumberChanged()保持一致,我再次尝试在最后一列插入按钮。我的想法是,行数可能会发生变化,所以我重新实现了相同的逻辑。但是,它没有用。

任何人都可以帮助纠正我的逻辑或其他逻辑来实现此功能。提前谢谢!

2 个答案:

答案 0 :(得分:0)

您应该为该列使用delegate。这样,视图自动处理行的创建/删除。见this question

您的代码还存在其他一些问题:

  1. 您多次连接rowsInserted / rowsRemoved
  2. 在您的代码中,您可以创建多个Signal Mapper来映射相同类型的连接。这不是它的意图使用
  3. 重复onRowsNumberChanged循环

答案 1 :(得分:0)

要使用窗口小部件解决您的问题,您可以使用以下方法。它使用Qt 5的新连接语法和C ++ lambdas,因此无需信号映射器:

#include "form.h"
#include <QtWidgets>
#include <QStandardItemModel>

Form::Form(QWidget *parent) :
    QWidget(parent)
{
    QPushButton *b = new QPushButton("add row");
    m_tree = new QTreeView(this);
    QBoxLayout *l = new QVBoxLayout(this);
    l->addWidget(m_tree, 1);
    l->addWidget(b);

    QStandardItemModel *m = new QStandardItemModel(0, 3, this);
    m_tree->setModel(m);

    connect(b, &QPushButton::clicked, this, &Form::addRow);
    connect(m, SIGNAL(rowsInserted(QModelIndex,int,int)), this, SLOT(addItemButtons()));
    connect(m, SIGNAL(rowsRemoved(QModelIndex,int,int)), this, SLOT(addItemButtons()));
}

Form::~Form()
{
}

void Form::addRow()
{
    m_tree->model()->insertRow(m_tree->model()->rowCount());
}

void Form::addItemButtons()
{
    for (int i = 0; i < m_tree->model()->rowCount(); ++i) {
        auto idx = m_tree->model()->index(i, 2);
        QPushButton *b = new QPushButton("X");
        b->setStyleSheet("QPushButton {color: #E5E5E5;}");
        m_tree->setIndexWidget(idx, b);
        connect(b, &QPushButton::clicked, [=](){
            m_tree->model()->removeRow(i);
        });
    }
}