5

QSqlRelationalTableModelで埋められたQTableViewがあります。ボタンを押すと、変更をコミットまたは元に戻す必要があります。一部の行を編集すると、編集が終了するとビューの状態が変更され、submitAll()が呼び出されるとDBに変更が正常にコミットされます。しかし、行を削除しようとすると、表示されたままになります。これが削除ボタンに接続されたスロットです:

def _removeSelectedStatuses(self):
    '''
    Удаляет выбранные строки из таблицы

    pre[self]: self._model is not None
    '''
    model = self.ConservationStatusesTableView.selectionModel()
    l = model.selectedRows()
    if not len(l): return

    rows = set([i.row() for i in l])
    rows = list(rows)
    rows.sort()
    first = rows[0]
    count = len(rows)
    self._model.removeRows(first, count)

私は何が間違っているのですか?

4

4 に答える 4

1

私は、この厄介な動作は仕様によるものであることを調査しました。行はコミット時にモデルから削除され、どの行を描画する必要があり、どの行を描画しないかをビューは認識しません。モデルから行が削除されたときに実行されるのは「!」だけです。header.model()。headerData(index、vert).text()のマーカー。そして、それは嫌です。

私は問題を修正する方法を恥じていますが、ここに私の醜いハックがあります:

from PyQt4 import QtGui
from PyQt4 import QtSql
from PyQt4 import QtCore

class SqlTableView(QtGui.QTableView):
    '''
    Представление, которое не показывает удалённые столбцы, 
    когда коммит ещё не прошёл
    '''


    def __init__(self, parent = None):
        '''
        Конструктор
        '''
        QtGui.QTableView.__init__(self, parent)

    def setModel(self, model):
        '''
        Мы не можем соединиться с моделями, не являющимися QSqlTableModel
        '''
        assert isinstance(model, QtSql.QSqlTableModel)
        QtGui.QTableView.setModel(self, model)

    def paintEvent(self, event):
        '''
        Тут всё и происходит. Осторожно, может стошнить.
        '''
        if self.model() is not None:
            header = self.verticalHeader()
            hm = header.model()
            for i in range(hm.rowCount()):
                if (hm.headerData(i, QtCore.Qt.Vertical).toPyObject() == '!' 
                    and not header.isSectionHidden(i)):
                    header.hideSection(i)
                elif (header.isSectionHidden(i) and 
                    hm.headerData(i, QtCore.Qt.Vertical).toPyObject() != '!'):
                    header.showSection(i)
        PyQt4.QtGui.QTableView.paintEvent(self, event)

また、インターフェイス設計を簡素化するためにQtDesignerに追加しました。

2番目の解決策、それほど厄介ではありません:

class PSqlRelationalTableModel : public QSqlRelationalTableModel
{
    Q_OBJECT

public:
    explicit PSqlRelationalTableModel(QObject *parent = 0, 
        QSqlDatabase db = QSqlDatabase());
    virtual ~PSqlRelationalTableModel();

    bool removeRows(int row, int count, 
        const QModelIndex &parent = QModelIndex());

public slots:
    void revertRow(int row);

signals:
    void rowIsMarkedForDeletion(int index);
    void rowDeletionMarkRemoved(int index);

private:
    QSet<unsigned int> rowsToDelete;
};

//////////////////////////////////////////////////////////////////
void PTableView::setModel(PSqlRelationalTableModel *model)
{
    connect(model, SIGNAL(rowIsMarkedForDeletion(int)), 
        this, SLOT(onRowMarkedForDeletion(int)));
    connect(model, SIGNAL(rowDeletionMarkRemoved(int)), 
        this, SLOT(onRowDeletionMarkRemoved(int)));
    QTableView::setModel(model);
}

void PTableView::onRowMarkedForDeletion(int index)
{
    QHeaderView *hv = verticalHeader();
    hv->hideSection(index);
}

void PTableView::onRowDeletionMarkRemoved(int index)
{
    QHeaderView *hv = verticalHeader();
    hv->showSection(index);
}
于 2010-11-12T15:01:42.167 に答える
1

私は最近同じ問題に直面し、自分自身のために別の解決策を見つけました。このモデルに接続されているQTableViewが1つしかない場合は、QSqlTableModel.deleteRow()の後にQTableView.setRowHidden()メソッドを使用できます。正常に動作します。

(カスタムペイントデリゲートでテキストを削除したいのですが、コミットされていない行を区別するための適切なフラグを見つけることができませんでした。)

于 2020-01-25T08:09:51.180 に答える
0

removeRowsメソッドを実装しましたか?

ここを見てください:

pyqt:QAbstractDataModelとQTreeViewの挿入行を理解しようとしています

欠落しているのは、何かが変更されたことをビューに通知する単純なemitDataChangedだと思います。それがないと、ビューはそれ自体を更新する必要があるかどうかを知ることができません!

お役に立てれば !

于 2010-11-11T12:29:47.807 に答える
0

選択した行をモデルから削除する場合は、次の呼び出しを行う必要があります。model-> removeRow(row); ここで、行は削除する行番号です。これは私にとってはうまくいきます。

于 2013-12-12T09:31:27.603 に答える