QAbstractTableModel получить пользовательский объект по измененным данным

Недавно я снова взял Qt и начал освежать свою память.
Создать собственную модель данных для таблицы было достаточно просто.

Сейчас я пытаюсь получить выбранные данные.
Обратите внимание, что я использую пользовательские объекты данных.

Пример моей пользовательской модели:

platform.h

class Platform
{
public:
Platform();
Platform(QString name);
QString getName();
void setName(QString name);
private:
QString m_name;
};

Очень простая структура данных для целей тестирования.
Затем я реализовал QAbstractTableModel, метод Data () выглядит так:

platformmodel.cpp

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

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

if (role == Qt::DisplayRole) {
Platform platform = m_platforms.at(index.row());
qDebug() << platform.getName();
return platform.getName();
}
return QVariant();
}

Из этого кода я понимаю, что для выбираемых элементов всегда возвращается строка, а не объект платформы.

Для отображения это работает нормально, я вижу реальные объекты в представлении.
Теперь я хочу выбрать реальный объект из модели, а не просто QString.

Таким образом, тело метода будет примерно таким:

void MainWindow::selectionChangedSlot(const QItemSelection &, const QItemSelection &)
{
//get the text of the selected item
const QModelIndex index = ui->lvPlatforms->selectionModel()->currentIndex();
Platform selectedPlatform = index.data();//This returns a QVariant and will fail at compile time, but I want to achieve something along this line.
setWindowTitle(selectedPlatform.getName());
}

Постскриптум Может быть, я пытаюсь найти неправильную вещь, я могу найти примеры, которые используют пользовательские объекты, но никто не говорит о получении выбранного элемента.

Должен быть лучший способ, чем получить строку, затем выполнить цикл по списку платформ и сравнить имя с выбранным элементом. Если у меня большой список, необходимость циклически проходить по каждому элементу и сравнивать строки не очень эффективна. ,

Я надеюсь, что моя проблема достаточно ясна. Если что-то важное не хватает, дайте мне знать, чтобы я мог отредактировать мой пример.

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

Я попробовал Q_DECLARE_METATYPE (Платформа);

И да, это работает, это позволяет хранить его в QVariant,
проблема в том, что для отображения всегда ожидается строка или 9/10 раз в любом случае.
До сих пор кажется невозможным иметь как текстовый дисплей, так и полный объект платформы из модели выбора (я могу сделать оба по отдельности … довольно бесполезно …)

1

Решение

Вы можете создать пользовательский тип, совместимый с QVariant, используя макрос Q_DECLARE_METATYPE.
Если вы объявите свой класс как метатип, вы можете сохранить его в QVariant и извлечь его с помощью приведения.

Вот пример, который показывает, как создать пользовательский делегат, который может отображать данные из пользовательского класса, используя QVariant:

class Data {
private:
QString name;
int value;
public:
Data() : name(""), value(-1){}
Data( QString n, int v ) : name(n), value(v){}
QString text() {
return QString( "Test %1 - %2" ).arg( name ).arg( value );
}
};

Q_DECLARE_METATYPE( Data )

class Delegate : public QStyledItemDelegate {
protected:
void paint(QPainter *painter, const QStyleOptionViewItem &option, const QModelIndex &index) const {
Data d = index.data().value<Data>();
painter->drawText( option.rect, d.text() );
}
};int main( int argc, char **argv) {
QApplication app(argc, argv, true);

QVariant var0, var1, var2;
var0.setValue(Data( "Item A", 0 ));
var1.setValue(Data( "Item B", 1 ));
var2.setValue(Data( "Item C", 2 ));

QListView *view = new QListView();
QStandardItemModel model(3, 1);

model.setData( model.index( 0, 0 ), var0 );
model.setData( model.index( 1, 0 ), var1 );
model.setData( model.index( 2, 0 ), var2 );
view->setModel( &model );
view->show();
view->setItemDelegate( new Delegate() );
return app.exec();
}
3

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

Других решений пока нет …

По вопросам рекламы ammmcru@yandex.ru
Adblock
detector