简体   繁体   中英

qml Unknown method return type: ArchiveFile*, even though qmlRegisterUncreatableType was called

Brief description of what I'm trying to do

I'm trying to make an application for opening some very structurally similar, related files. Í have a base class, named GenericFile , and a subclass, named ArchiveFile . There also exists subclasses of ArchiveFile , which contain implementation logic for the specific archive file format, but those are not relevant for QML and therefore not exposed to QML.

I've yet another class named FileManager , which exposes certain methods to QML to create instances of GenericFile and to downcast them to classes like ArchiveFile , based on GenericFile.getFileCategory .

The problem

I've registered both GenericFile and ArchiveFile using qmlRegisterUncreatableType , like this:

int main(int argc, char** argv) {
    QApplication app(argc, argv);
    QQmlApplicationEngine engine;
    qmlRegisterUncreatableType<GenericFile>("me.henkkalkwater", 1, 0, "GenericFile", "Get your copy of GenericFile using FileManager!");
    qmlRegisterUncreatableType<ArchiveFile>("me.henkkalkwater", 1, 0, "ArchiveFile", "Get your copy of ArchiveFile using FileManager!");
    qmlRegisterSingletonType<FileManager>("me.henkkalkwater", 1, 0, "FileManager", [](QQmlEngine* engine, QJSEngine* jsEngine) -> QObject* {
        Q_UNUSED(engine)
        Q_UNUSED(jsEngine)
        return new FileManager();
    });
    const QUrl url(QStringLiteral("qrc:/qml/main.qml"));
    QObject::connect(&engine, &QQmlApplicationEngine::objectCreated,
                     &app, [url](QObject *obj, const QUrl &objUrl) {
        if (!obj && url == objUrl)
            QCoreApplication::exit(-1);
    }, Qt::QueuedConnection);
    engine.load(url);
}

Calling GenericFile* FileManager.getFile(int index) works fine in QML. But calling ArchiveFile* FileManager.getArchive(int index) doesn't work, with the message "Error: Unknown method return type: ArchiveFile*".

Has the problem to do with the fact that ArchiveFile inherits from GenericFile or did I make a silly mistake somewhere else? Or is the design of my classes seriously flawed? Am I taking some Java conventions to C++ which should be done in another way?

My code

genericfile.h:

class GenericFile;
typedef GenericFile* (*CreateFileFunction) (QIODevice* file, QString fileName, QObject* parent);

class GenericFile : public QObject {
    Q_OBJECT
public:
    enum FileCategory {
        UNKNOWN,
        COMPRESSED,
        ARCHIVE,
        LAYOUT
    };
    Q_ENUM(FileCategory)

    GenericFile(QIODevice* source, QString name, QObject* parent = nullptr);
    virtual ~NinFile();

    Q_PROPERTY(QString fileName READ getFileName)
    Q_PROPERTY(FileCategory fileCategory READ getFileCategory)

    /**
     * @brief Register a file usign a certain string of magic bytes.
     * @param magic
     * @param fn
     */
    static void RegisterFile(QLatin1String magic, CreateFileFunction fn);

    /**
     * @brief Return one of the subclasses of the file.
     * @param file The file to determine.
     * @return The class able to parse this file format.
     *
     * Scans the magic bytes and returns a subclass of GenericFile that represents the file.
     * The subclass will take ownership of the QFile and destroy it whenever it is destroyed
     * itself.
     */
    static GenericFile* fromFile(QFile* file, QObject* parent = nullptr);
    static GenericFile* fromIODevice(QIODevice* file, QString filename, QObject* parent = nullptr);

    /**
     * @brief Initialize the default file associations.
     */
    static void init();

    /**
     * @return The filename of this file
     */
    QString getFileName() const {
        return fileName;
    }

    /**
     * @return return the general category the file belongs in
     */
    virtual FileCategory getFileCategory() const {
        return UNKNOWN;
    }

protected:
    QIODevice* file;
    QString fileName;
};

archivefile.h

class ArchiveFile : public GenericFile
{
public:
    ArchiveFile(QIODevice* device, QString fileName, QObject* parent = nullptr);

    Q_INVOKABLE
    virtual QList<QString> listContents() = 0;
    FileCategory getFileCategory() const override { return GenericFile::ARCHIVE; }
};

FileManager.h (with relevant method implementations included)

class FileManager : public QAbstractListModel
{
    Q_OBJECT
public:
    explicit FileManager(QObject *parent = nullptr);
    enum RoleNames {
        INDEX = Qt::UserRole + 1,
        PATH,
        TYPE,
        CATEGORY
    };

    /**
     * @brief Opens a file
     * @param filePath The path to the file
     * @return false if the opening fails, true otherwise
     * Opens a file, and if successful, returns true and adds it to this model. Otherwise, it returns false.
     */
    Q_INVOKABLE
    bool openFile(QUrl filePath);

    Q_INVOKABLE
    GenericFile* getFile(int index) {
        if (index >= 0 && index < files.length()) {
            return files[index];
        }
        return nullptr;
    }

    Q_INVOKABLE
    ArchiveFile* getArchive(int index) {
        qDebug() << "Callled getArchive";
        try {
            return dynamic_cast<ArchiveFile*>(getFile(index));
        } catch (std::bad_cast e){
            return nullptr;
        }
    }

    QVariant data(const QModelIndex& index, int role = Qt::DisplayRole) const override;
    int rowCount(const QModelIndex& parent = QModelIndex()) const override;
    QHash<int, QByteArray> roleNames() const override;

private:
    QList<GenericFile*> files;
};

Edit: it appears that this happens every time I register a base class and a subclass of said baseclass.

Adding the Q_OBJECT macro to the subclass ( ArchiveFile ) seems to resolve the issue. I wrongly assumed that you only had to define the Q_OBJECT macro in the base class ( GenericFile ) somehow.

The technical post webpages of this site follow the CC BY-SA 4.0 protocol. If you need to reprint, please indicate the site URL or the original address.Any question please contact:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM