Подкласс Qt и функция отсутствия соответствия для вызова для подключения

Я пытаюсь инкапсулировать поведение табличного представления и начинаю с подключения сигналов заголовка табличного представления к слотам табличного представления, определенным в подклассе. Я могу получить поведение, которое ищу, без создания подклассов, но это противоречит цели. Когда я пытаюсь создать подкласс, я получаю ужасное сообщение «Нет соответствующего вызова функции для подключения. Все компоненты в конечном итоге являются объектами QObject, поэтому я не думаю, что это проблема. (Но, опять же, может быть, это и есть проблема.) Прямо сейчас Я подключаюсь к 'hideColumn()', но в конечном итоге я хотел бы подключиться к своим собственным слотам (таким как 'my_sectionClicked(int)'.

Приведенный ниже код был взят из исходного кода «Программирование GUI на C++ с помощью Qt 4» Жасмин Бланшетт и Марка Саммерфилда, за исключением добавления «MyTableView», которое я сделал.

MyTableView.h

#ifndef MYTABLEVIEW_HPP
#define MYTABLEVIEW_HPP

#include <QTableView>

class MyTableView : public QTableView
{
    Q_OBJECT

public:
    explicit MyTableView(QWidget *parent = 0);

public slots:
    void my_sectionClicked(int logicalIndex);

private slots:

public:

private:
    QHeaderView *m_rowHeader;

};

#endif // MYTABLEVIEW_HPP

MyTableView.cpp

// Qt Includes
#include <QDebug>
#include <QMenu>

// Local Includes
#include "MyTableView.h"

MyTableView::MyTableView(QWidget *parent) : QTableView(parent)
{
    m_rowHeader = horizontalHeader();
    connect(m_rowHeader, SIGNAL(sectionClicked(int)), this, SLOT(hideColumn(int)));
}

void MyTableView::my_sectionClicked(int logicalIndex)
{
    qDebug().nospace() << "Column " << logicalIndex << " selected." << "\n";
}

валютамодель.h

#ifndef CURRENCYMODEL_H
#define CURRENCYMODEL_H

#include <QAbstractTableModel>
#include <QMap>

class CurrencyModel : public QAbstractTableModel
{
public:
    CurrencyModel(QObject *parent = 0);

    void setCurrencyMap(const QMap<QString, double> &map);
    int rowCount(const QModelIndex &parent) const;
    int columnCount(const QModelIndex &parent) const;
    QVariant data(const QModelIndex &index, int role) const;
    QVariant headerData(int section, Qt::Orientation orientation,
        int role) const;

private:
    QString currencyAt(int offset) const;

    QMap<QString, double> currencyMap;
};

#endif

валютамодель.cpp

#include <QtCore>

#include "currencymodel.h"

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

void CurrencyModel::setCurrencyMap(const QMap<QString, double> &map)
{
    currencyMap = map;
    reset(); // Forces views to refresh data 
}

int CurrencyModel::rowCount(const QModelIndex & /* parent */) const
{
    return currencyMap.count();
}

int CurrencyModel::columnCount(const QModelIndex & /* parent */) const
{
    return currencyMap.count();
}

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

    if (role == Qt::TextAlignmentRole)
    {
        return int(Qt::AlignRight | Qt::AlignVCenter);
    }
    else if (role == Qt::DisplayRole)
    {
        QString rowCurrency = currencyAt(index.row());
        QString columnCurrency = currencyAt(index.column());

        if (currencyMap.value(rowCurrency) == 0.0)
            return "####";

        double amount = currencyMap.value(columnCurrency)
                    / currencyMap.value(rowCurrency);

        return QString("%1").arg(amount, 0, 'f', 4);
    }

    return QVariant();
}

QVariant CurrencyModel::headerData(int section,
    Qt::Orientation /* orientation */,
    int role) const
{
    if (role != Qt::DisplayRole)
        return QVariant();
    return currencyAt(section);
}

QString CurrencyModel::currencyAt(int offset) const
{
    return (currencyMap.begin() + offset).key();
}

главное окно.h

#ifndef MAINWINDOW_H
#define MAINWINDOW_H

#include <QMainWindow>
#include <QtGui>

#include "MyTableView.h"
#include "currencymodel.h"

class MainWindow : public QMainWindow
{
    Q_OBJECT

public:
    MainWindow();

protected:

private slots:
    void on_sectionClicked(int locicalIndex);

private:
    QMap<QString, double> currencyMap;
    CurrencyModel currencyModel;
    QTableView tableView;
    QHeaderView *m_rowHeader;
};

#endif

главное окно.cpp

#include <iostream>
#include "mainwindow.h"

MainWindow::MainWindow()
{
    currencyMap.insert("AUD", 1.3259);
    currencyMap.insert("CHF", 1.2970);
    currencyMap.insert("CZK", 24.510);
    currencyMap.insert("DKK", 6.2168);
    currencyMap.insert("EUR", 0.8333);
    currencyMap.insert("GBP", 0.5661);
    currencyMap.insert("HKD", 7.7562);
    currencyMap.insert("JPY", 112.92);
    currencyMap.insert("NOK", 6.5200);
    currencyMap.insert("NZD", 1.4697);
    currencyMap.insert("SEK", 7.8180);
    currencyMap.insert("SGD", 1.6901);
    currencyMap.insert("USD", 1.0000);

    currencyModel.setCurrencyMap(currencyMap);

    /*
    THIS WORKS!!!

    m_rowHeader = tableView.horizontalHeader();
    connect(m_rowHeader, SIGNAL(sectionClicked(int)),
        &(tableView), SLOT(hideColumn(int)));
    */

    tableView.setModel(&currencyModel);
    tableView.setAlternatingRowColors(true);

    tableView.setWindowTitle(QObject::tr("Currencies"));
    tableView.show();
}

void MainWindow::on_sectionClicked(int logicalIndex)
{
    qDebug().nospace() << "Column " << logicalIndex << " selected." << "\n";
}

main.cpp

#include "mainwindow.h"

int main(int argc, char *argv[])
{
    QApplication app(argc, argv);
    MainWindow mainWin;
    return app.exec();
}

Ошибка, которую я получаю:

g++ -c -pipe -O2 -Wall -W -D_REENTRANT -DQT_NO_DEBUG -DQT_GUI_LIB -DQT_CORE_LIB -DQT_SHARED -I/opt/QtSDK/Desktop/Qt/4.8.1/gcc/mkspecs/default -I. -I/opt/QtSDK/Desktop/Qt/4.8.1/gcc/include/QtCore -I/opt/QtSDK/Desktop/Qt/4.8.1/gcc/include/QtGui -I/opt/QtSDK/Desktop/Qt/4.8.1/gcc/include -I. -I. -o MyTableView.o MyTableView.cpp
MyTableView.cpp: In constructor ‘MyTableView::MyTableView(QWidget*)’:
MyTableView.cpp:11: error: no matching function for call to ‘MyTableView::connect(QHeaderView*&, const char [21], MyTableView* const, const char [17])’
/opt/QtSDK/Desktop/Qt/4.8.1/gcc/include/QtCore/qobject.h:204: note: candidates are: static bool QObject::connect(const QObject*, const char*, const QObject*, const char*, Qt::ConnectionType)
/opt/QtSDK/Desktop/Qt/4.8.1/gcc/include/QtCore/qobject.h:217: note:                 static bool QObject::connect(const QObject*, const QMetaMethod&, const QObject*, const QMetaMethod&, Qt::ConnectionType)
/opt/QtSDK/Desktop/Qt/4.8.1/gcc/include/QtCore/qobject.h:337: note:                 bool QObject::connect(const QObject*, const char*, const char*, Qt::ConnectionType) const
make: *** [MyTableView.o] Error 1

person Jason DeMello    schedule 02.04.2013    source источник
comment
Где используется my_sectionClicked?   -  person Vaibhav Desai    schedule 02.04.2013
comment
тл; др. Публикуйте только соответствующий код.   -  person cmannett85    schedule 02.04.2013
comment
@VaibhavDesai В какой-то момент связь с hideColumn(int) была связью с my_sectionClicked(int). Я изменил его на существующий слот Qt в качестве процесса устранения при устранении неполадок.   -  person Jason DeMello    schedule 03.04.2013


Ответы (1)


Такая проблема обычно возникает, когда вы передаете класс, который был предварительно объявлен, но не #include'd

В вашем случае похоже, что вам, вероятно, нужен #include <QHeaderView> в файле MyTableView.cpp.

Проблема в том, что вы можете передавать указатели на объекты, которые были предварительно объявлены, но компилятор не знает, что ваш класс в конечном итоге является производным от QObject, поэтому ему не разрешено использовать указатель вашего подкласса в качестве указателя QObject при вызове функция connect().

person Chris    schedule 02.04.2013
comment
Спасибо! Хотя это симптом, который, казалось бы, противоречит советам Херба Саттера и Андрея Александреску в их стандартах кодирования C++; 101 правила, рекомендации и передовой опыт. Я имею в виду вступительное предложение пункта 22 о минимизации дефиниционных зависимостей и избегании циклических зависимостей. . . Не #include определение, когда достаточно предварительного объявления. - person Jason DeMello; 02.04.2013
comment
В данном случае это не будет ;) - person Chris; 03.04.2013