Logo Search packages:      
Sourcecode: qtscriptgenerator version File versions  Download package

setupgenerator.cpp

/****************************************************************************
**
** Copyright (C) 1992-2008 Trolltech ASA. All rights reserved.
**
** This file is part of the Qt Script Generator project on Trolltech Labs.
**
** This file may be used under the terms of the GNU General Public
** License version 2.0 as published by the Free Software Foundation
** and appearing in the file LICENSE.GPL included in the packaging of
** this file.  Please review the following information to ensure GNU
** General Public Licensing requirements will be met:
** http://www.trolltech.com/products/qt/opensource.html
**
** If you are unsure which license is appropriate for your use, please
** review the following information:
** http://www.trolltech.com/products/qt/licensing.html or contact the
** sales department at sales@trolltech.com.
**
** This file is provided AS IS with NO WARRANTY OF ANY KIND, INCLUDING THE
** WARRANTY OF DESIGN, MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.
**
****************************************************************************/

#include "setupgenerator.h"
#include "reporthandler.h"
#include "fileout.h"

//#define Q_SCRIPT_LAZY_GENERATOR

void SetupGenerator::addClass(const AbstractMetaClass *cls) 
{
    packHash[cls->package()].append(cls);
}

void writeQtScriptQtBindingsLicense(QTextStream &stream);

static void writeIncludes(QTextStream &stream) 
{
    stream << "#include <QtScript/QScriptExtensionPlugin>" << endl
           << "#include <QtScript/QScriptValue>" << endl
           << "#include <QtScript/QScriptEngine>" << endl;
#ifdef Q_SCRIPT_LAZY_GENERATOR
    stream << "#include <QtScript/QScriptClass>" << endl
           << "#include <QtScript/QScriptString>" << endl;
#endif
    stream << "#include <QtCore/QDebug>" << endl;
}

void writeInclude(QTextStream &stream, const Include &inc);
void maybeDeclareMetaType(QTextStream &stream, const QString &typeName,
                          QSet<QString> &registeredTypeNames);
bool hasDefaultConstructor(const AbstractMetaClass *meta_class);

void SetupGenerator::generate()
{
    QHashIterator<QString, QList<const AbstractMetaClass*> > pack(packHash);
    while (pack.hasNext()) {
        pack.next();
        QList<const AbstractMetaClass*> list = pack.value();
        if (list.isEmpty())
            continue;

        QString packName = pack.key();
        QStringList components = packName.split(".");
        if ((components.size() > 2) && (components.at(0) == "com")
            && (components.at(1) == "trolltech")) {
            // kill com.trolltech in key
            components.removeAt(0);
            components.removeAt(0);
        }
        packName.replace(".", "_");
        FileOut file(m_out_dir + "/generated_cpp/" + packName + "/main.cpp");
        QTextStream &s = file.stream;

        if (FileOut::license)
            writeQtScriptQtBindingsLicense(s);

        writeIncludes(s);
        s << endl;
        foreach (const AbstractMetaClass *cls, list)
            writeInclude(s, cls->typeEntry()->include());
        s << endl;

        // declare individual class creation functions
        foreach (const AbstractMetaClass *cls, list) {
            s << "QScriptValue qtscript_create_" << cls->name() << "_class(QScriptEngine *engine);" << endl;
        }
        s << endl;

        // write table of class names
        {
            s << "static const char * const qtscript_" << packName << "_class_names[] = {" << endl;
            bool needComma = false;
            foreach (const AbstractMetaClass *cls, list) {
                s << "    ";
                if (needComma)
                    s << ", ";
                s << "\"" << cls->name() << "\"" << endl;
                needComma = true;
            }
            s << "};" << endl << endl;
        }

        // write table of function pointers
        {
            s << "typedef QScriptValue (*QtBindingCreator)(QScriptEngine *engine);" << endl;
            s << "static const QtBindingCreator qtscript_" << packName << "_class_functions[] = {" << endl;
            bool needComma = false;
            foreach (const AbstractMetaClass *cls, list) {
                s << "    ";
                if (needComma)
                    s << ", ";
                s << "qtscript_create_" << cls->name() << "_class" << endl;
                needComma = true;
            }
            s << "};" << endl << endl;
        }

#ifdef Q_SCRIPT_LAZY_GENERATOR
        {
            // declare meta-types
            QSet<QString> registeredTypeNames = m_qmetatype_declared_typenames;
            foreach (const AbstractMetaClass *cls, list) {
                if (cls->isNamespace())
                    continue;
                QString name = cls->qualifiedCppName();
                if (cls->typeEntry()->isValue() && ::hasDefaultConstructor(cls))
                    maybeDeclareMetaType(s, name, registeredTypeNames);
                maybeDeclareMetaType(s, name + "*", registeredTypeNames);
            }
            s << endl;
            // write table of metatype-ids
            s << "static const int qtscript_" << packName << "_metatype_ids[] = {" << endl;
            for (int i = 0; i < list.size(); ++i) {
                const AbstractMetaClass *cls = list.at(i);
                s << "    ";
                if (i > 0)
                    s << ", ";
                if (cls->isNamespace()) {
                    s << "-1, -1";
                } else {
                    QString name = cls->qualifiedCppName();
                    if (cls->typeEntry()->isValue() && ::hasDefaultConstructor(cls))
                        s << "qMetaTypeId<" << name << ">()";
                    else
                        s << "-1";
                    s << ", qMetaTypeId<" << name << "*>()";
                }
                s << endl;
            }
            s << "};" << endl << endl;
        }

        // write the fake prototype class
        {
            s << "class qtscript_" << packName << "_FakePrototype : public QScriptClass" << endl
              << "{" << endl
              << "public:" << endl
              << "    qtscript_" << packName << "_FakePrototype(QScriptEngine *engine)" << endl
              << "        : QScriptClass(engine) {}" << endl << endl

              << "    QueryFlags queryProperty(const QScriptValue &fake," << endl
              << "        const QScriptString &name, QueryFlags flags, uint *)" << endl
              << "    {" << endl
              << "        if (fake.prototype().isValid())" << endl
              << "            return 0;" << endl
              << "        int classIndex = fake.data().toInt32();" << endl
              << "        const char *className = qtscript_" << packName << "_class_names[classIndex];" << endl
//              << "        qDebug() << \"faking\" << className;" << endl
              << "        QScriptValue extensionObject = engine()->globalObject();" << endl
              << "        QScriptValue ctor = extensionObject.property(className);" << endl
              << "        QScriptValue genuine = ctor.property(\"prototype\");" << endl
              << "        Q_ASSERT(genuine.isObject());" << endl
              << "        const_cast<QScriptValue&>(fake).setPrototype(genuine);" << endl
              << "        if (!genuine.property(name).isValid())" << endl
              << "            flags &= ~HandlesReadAccess;" << endl
              << "        return flags & ~HandlesWriteAccess;" << endl
              << "    }" << endl << endl

              << "    QScriptValue property(const QScriptValue &fake, "
              << "const QScriptString &name, uint)" << endl
              << "    {" << endl
              << "        return fake.prototype().property(name, QScriptValue::ResolveLocal);" << endl
              << "    }" << endl
              << "};" << endl << endl;
        }

        // write the lazy class loader
        {
            s << "static QScriptValue qtscript_" << packName << "_getSetClass("
              << "QScriptContext *context, QScriptEngine *engine)" << endl
              << "{" << endl
              << "    QScriptValue target = context->thisObject();" << endl
              << "    int classIndex = context->callee().data().toInt32();" << endl
              << "    const char *className = qtscript_" << packName << "_class_names[classIndex];" << endl
              << "    qDebug() << \"loading\" << className;" << endl
              << "    target.setProperty(className, QScriptValue(), "
              << "QScriptValue::PropertyGetter|QScriptValue::PropertySetter);" << endl
              << "    if (context->argumentCount() == 1) {" << endl
              << "        target.setProperty(className, context->argument(0));" << endl
              << "    } else {" << endl
              << "        target.setProperty(className, qtscript_"
              << packName << "_class_functions[classIndex](engine)," << endl
              << "            QScriptValue::SkipInEnumeration);" << endl
              << "    }" << endl
              << "    return target.property(className);" << endl
              << "}" << endl << endl;
        }
#endif

        // plugin class declaration
        s << "class " << packName << "_ScriptPlugin : public QScriptExtensionPlugin" << endl
          << "{" << endl
          << "public:" << endl
          << "    QStringList keys() const;" << endl
          << "    void initialize(const QString &key, QScriptEngine *engine);" << endl
          << "};" << endl
          << "" << endl;

        // keys()
        s << "QStringList " << packName << "_ScriptPlugin::keys() const" << endl
          << "{" << endl
          << "    QStringList list;" << endl;
        {
            QString key;
            for (int i = 0; i < components.size(); ++i) {
                if (i > 0)
                    key.append(".");
                key.append(components.at(i));
                s << "    list << QLatin1String(\"" << key << "\");" << endl;
            }
        }
        s << "    return list;" << endl
          << "}" << endl;

        // initialize()
        s << endl
          << "void " << packName << "_ScriptPlugin::initialize(const QString &key, QScriptEngine *engine)" << endl
          << "{";
        {
            QString key;
            for (int i = 0; i < components.size(); ++i) {
                s << endl << "    ";
                if (i > 0) {
                    key.append(".");
                    s << "} else ";
                }
                key.append(components.at(i));
                s << "if (key == QLatin1String(\"" << key << "\")) {";
            }
        }
        s << endl << "        QScriptValue extensionObject = ";
        // ### generalize
        if (packName == "com.trolltech.qt.phonon")
            s << "setupPackage(\"phonon\", engine)";
        else
            s << "engine->globalObject()";
        s << ";" << endl;

#ifdef Q_SCRIPT_LAZY_GENERATOR
        s << "        qtscript_" << packName << "_FakePrototype *fakeProtoClass;" << endl
          << "        fakeProtoClass = new qtscript_" << packName << "_FakePrototype(engine);" << endl;
#endif
        s << "        for (int i = 0; i < " << list.size() << "; ++i) {" << endl
#ifndef Q_SCRIPT_LAZY_GENERATOR
          << "            extensionObject.setProperty(qtscript_" << packName << "_class_names[i]," << endl
          << "                qtscript_" << packName << "_class_functions[i](engine)," << endl
          << "                QScriptValue::SkipInEnumeration);" << endl
#else
          << "            QScriptValue classIndex(engine, i);" << endl
          << "            QScriptValue fakeCtor = engine->newFunction(qtscript_" << packName << "_getSetClass);" << endl
          << "            fakeCtor.setData(classIndex);" << endl
          << "            extensionObject.setProperty(qtscript_" << packName << "_class_names[i]," << endl
          << "                fakeCtor, QScriptValue::PropertyGetter|QScriptValue::PropertySetter"
          << "|QScriptValue::SkipInEnumeration);" << endl
          << "            QScriptValue fakeProto = engine->newObject(fakeProtoClass, classIndex);" << endl
          << "            fakeProto.setPrototype(QScriptValue());" << endl
          << "            if (qtscript_" << packName << "_metatype_ids[i*2] != -1)" << endl
          << "                engine->setDefaultPrototype(qtscript_"
          << packName << "_metatype_ids[i*2], fakeProto);" << endl
          << "            if (qtscript_" << packName << "_metatype_ids[i*2+1] != -1)" << endl
          << "                engine->setDefaultPrototype(qtscript_"
          << packName << "_metatype_ids[i*2+1], fakeProto);" << endl
#endif
          << "        }" << endl;

        s << "    } else {" << endl
          << "        Q_ASSERT_X(false, \"" << packName << "::initialize\", qPrintable(key));" << endl
          << "    }" << endl
          << "}" << endl;

        s << "Q_EXPORT_STATIC_PLUGIN(" << packName << "_ScriptPlugin)" << endl
          << "Q_EXPORT_PLUGIN2(qtscript_" << packName.toLower() << ", " << packName << "_ScriptPlugin)" << endl << endl;
    
        if (file.done())
            ++m_num_generated_written;
        ++m_num_generated;
    }
}

Generated by  Doxygen 1.6.0   Back to index