Qt tablView с указателем модели на пользовательский класс

Я создал свой собственный класс (CustomObject), который наследуется от QGraphicsItem. Я использую эти объекты для рисования на сцене (прямоугольники, многоугольники и прочее) и храню указатели на них в QList. Теперь мне нужно отобразить все мои CustomOBjects в tableView, но есть 2 условия:

  1. Когда я выбираю и взаимодействую с объектом в табличном представлении — я должен иметь возможность взаимодействовать с «настоящим» CustomObject, представленным им (пример: я выбрал obj1 в табличном представлении и нажимаю кнопку «удалить» или «редактировать»), и я хочу иметь возможность взаимодействовать с объектом acctaul (удалить или отредактировать его).
  2. Когда я добавляю новый или меняю его — я хочу видеть изменение в tableView.

Я не уверен, смогу ли я добиться этого с помощью jsut table view и soem custom mdoel — или, если я сделаю свой собственный класс QAbstractItemModel, но если я это сделаю — как мне это сделать? Должен ли я сделать класс наследовать от QAbstractItemModel и добавить указатель на мой CustomObject, или просто заставить мои CustomObjects в конкретной конкретной модели?

Маленькие кусочки моего кода:

Вот мой CustomObject.h
// Я удалил некоторый код, который был строго связан с «личными» функциями, связанными с определенными функциями моего приложения

    class CustomObject : public QGraphicsItem
{
public:
CustomObject();
CustomObject(int _x, int _y, int _w, int _h);
virtual QRectF boundingRect() const;

void set_Name(QString name);
QString get_Name();

protected:
void mousePressEvent(QGraphicsSceneMouseEvent *event);
void mouseReleaseEvent(QGraphicsSceneMouseEvent *event);
void mouseMoveEvent(QGraphicsSceneMouseEvent *event);

private:
QString Name;

Я храню их в списке, в моем классе «Overseer»:

    class ObjOverseer
public:
void drawingCustomObject_Do(int x, int y); //This function creates new "CustomObject" and adds it to the list (below)

QList<CustomObject*> ObjectsList_CustomObjects;

В моем главном окне — я просто создаю этот ObjOverseer и сохраняю его указатель.

РЕДАКТИРОВАТЬ 1

Я использовал этот пример:
https://doc.qt.io/archives/4.6/itemviews-addressbook.html
и создал этот класс:

    CustomModelOfCustomObject::CustomModelOfCustomObject()
{
}

CustomModelOfCustomObject::CustomModelOfCustomObject(QObject *parent)
: QAbstractTableModel(parent)
{
}

CustomModelOfCustomObject::CustomModelOfCustomObject(QList<CustomObject*> objects, QObject *parent)
: QAbstractTableModel(parent)
{
ListOfObjects=objects;
}

int CustomModelOfCustomObject::rowCount(const QModelIndex &parent) const
{
Q_UNUSED(parent);
return ListOfObjects.size();
}

int CustomModelOfCustomObject::columnCount(const QModelIndex &parent) const
{
Q_UNUSED(parent);
return 2;//TODO - ZMIENIC ILOSC KOLUMN
}

QVariant CustomModelOfCustomObject::data(const QModelIndex &index, int role) const
{
if (!index.isValid())
return QVariant();

if (index.row() >= ListOfObjects.size() || index.row() < 0)
return QVariant();

if (role == Qt::DisplayRole) {
CustomObject* obj = ListOfObjects.at(index.row());

if (index.column() == 0)
return obj->get_Name();
else if (index.column() == 1)
return obj->get_Address();
}
return QVariant();
}

QVariant CustomModelOfCustomObject::headerData(int section, Qt::Orientation orientation, int role) const
{
if (role != Qt::DisplayRole)
return QVariant();

if (orientation == Qt::Horizontal) {
switch (section) {
case 0:
return tr("Name");

case 1:
return tr("Address");

default:
return QVariant();
}
}
return QVariant();
}

bool CustomModelOfCustomObject::insertRows(int position, int rows, const QModelIndex &index)
{
Q_UNUSED(index);
beginInsertRows(QModelIndex(), position, position+rows-1);

for (int row=0; row < rows; row++) {
CustomObject* obj;
ListOfObjects.insert(position, obj);
}

endInsertRows();
return true;
}

bool CustomModelOfCustomObject::removeRows(int position, int rows, const QModelIndex &index)
{
Q_UNUSED(index);
beginRemoveRows(QModelIndex(), position, position+rows-1);

for (int row=0; row < rows; ++row) {
ListOfObjects.removeAt(position);
}

endRemoveRows();
return true;
}

bool CustomModelOfCustomObject::setData(const QModelIndex &index, const QVariant &value, int role)
{
if (index.isValid() && role == Qt::EditRole) {
int row = index.row();

CustomObject* p = ListOfObjects.value(row);

if (index.column() == 0)
p->set_Name(value.toString());
else if (index.column() == 1)
p->set_Address(value.toString());
else
return false;

ListOfObjects.replace(row, p);
emit(dataChanged(index, index));

return true;
}

return false;
}

Qt::ItemFlags CustomModelOfCustomObject::flags(const QModelIndex &index) const
{
if (!index.isValid())
return Qt::ItemIsEnabled;

return QAbstractTableModel::flags(index) | Qt::ItemIsEditable;
}

QList<CustomObject*> CustomModelOfCustomObject::getList()
{
return ListOfObjects;
}

Но тем не менее, когда я достигаю точки в своей функции, где я должен использовать эту модель — я не знаю, как мне ее добавить, или даже если я смогу использовать ее по назначению.

РЕДАКТИРОВАТЬ 2
Когда я изменил ListOfObject для общественности и попытался:

   MyModel->ListOfObjects.append(newObj);

все разбилось

1

Решение

Первое, что нужно отметить, это то, что список в ObjOverseer и CustomModelOfCustomObject не связаны, вам нужно сохранить указатель списка ObjOverseer вместо того, чтобы скопировать его в CustomModelOfCustomObject. Вот этот :

CustomModelOfCustomObject::CustomModelOfCustomObject(QList<CustomObject*> objects, QObject *parent)
: QAbstractTableModel(parent)
{
ListOfObjects=objects;
}

Вам необходимо добавить функцию в вашу CustomModel, которая будет обрабатывать добавление нового пользовательского объекта:

 bool CustomModelOfCustomObject::insertNewData(CustomObject  *obj, int rowposition = -1)
{
int row = rowposition < 0 ? ListOfObjects.size : row;
beginInserRows(QModelIndex(), row, row);
ListOfObjects.insert(row, obj);
endInsertRow();
}

И когда вы хотите добавить новый объект, просто вызовите эти функции. Если список по модели подключен к ObjOverseer (тип указателя), вам не нужно добавлять новый объект в ObjOverseer.

0

Другие решения

Решил это. Теперь я только добавляю «представление» объекта в модель и всякий раз, когда мне нужно обновить какой-либо объект (например, изменить имя или цвет), я просто делаю это через моего надзирателя, передавая soem-тип guid / identifier / id / что-нибудь, что woudl позволяет мне отличать объекты друг от друга.

0

По вопросам рекламы [email protected]