Недавно я снова взял 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 раз в любом случае.
До сих пор кажется невозможным иметь как текстовый дисплей, так и полный объект платформы из модели выбора (я могу сделать оба по отдельности … довольно бесполезно …)
Вы можете создать пользовательский тип, совместимый с 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();
}
Других решений пока нет …