mirror of
https://github.com/clementine-player/Clementine
synced 2025-01-18 12:32:09 +01:00
1987 lines
64 KiB
C++
1987 lines
64 KiB
C++
/****************************************************************************
|
|
**
|
|
** Copyright (C) 2008-2009 Nokia Corporation and/or its subsidiary(-ies).
|
|
** All rights reserved.
|
|
** Contact: Nokia Corporation (qt-info@nokia.com)
|
|
**
|
|
** This file is part of the Qt Script Generator project on Qt Labs.
|
|
**
|
|
** $QT_BEGIN_LICENSE:LGPL$
|
|
** No Commercial Usage
|
|
** This file contains pre-release code and may not be distributed.
|
|
** You may use this file in accordance with the terms and conditions
|
|
** contained in the Technology Preview License Agreement accompanying
|
|
** this package.
|
|
**
|
|
** GNU Lesser General Public License Usage
|
|
** Alternatively, this file may be used under the terms of the GNU Lesser
|
|
** General Public License version 2.1 as published by the Free Software
|
|
** Foundation and appearing in the file LICENSE.LGPL included in the
|
|
** packaging of this file. Please review the following information to
|
|
** ensure the GNU Lesser General Public License version 2.1 requirements
|
|
** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
|
|
**
|
|
** In addition, as a special exception, Nokia gives you certain additional
|
|
** rights. These rights are described in the Nokia Qt LGPL Exception
|
|
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
|
|
**
|
|
** If you have questions regarding the use of this file, please contact
|
|
** Nokia at qt-info@nokia.com.
|
|
**
|
|
**
|
|
**
|
|
**
|
|
**
|
|
**
|
|
**
|
|
**
|
|
** $QT_END_LICENSE$
|
|
**
|
|
****************************************************************************/
|
|
|
|
#include "abstractmetalang.h"
|
|
#include "reporthandler.h"
|
|
|
|
/*******************************************************************************
|
|
* AbstractMetaType
|
|
*/
|
|
AbstractMetaType *AbstractMetaType::copy() const
|
|
{
|
|
AbstractMetaType *cpy = new AbstractMetaType;
|
|
|
|
cpy->setTypeUsagePattern(typeUsagePattern());
|
|
cpy->setConstant(isConstant());
|
|
cpy->setReference(isReference());
|
|
cpy->setIndirections(indirections());
|
|
cpy->setInstantiations(instantiations());
|
|
cpy->setArrayElementCount(arrayElementCount());
|
|
cpy->setOriginalTypeDescription(originalTypeDescription());
|
|
cpy->setOriginalTemplateType(originalTemplateType() ? originalTemplateType()->copy() : 0);
|
|
|
|
cpy->setArrayElementType(arrayElementType() ? arrayElementType()->copy() : 0);
|
|
|
|
cpy->setTypeEntry(typeEntry());
|
|
|
|
return cpy;
|
|
}
|
|
|
|
QString AbstractMetaType::cppSignature() const
|
|
{
|
|
QString s;
|
|
|
|
if (isConstant())
|
|
s += "const ";
|
|
|
|
s += typeEntry()->qualifiedCppName();
|
|
|
|
if (hasInstantiationInCpp()) {
|
|
QList<AbstractMetaType *> types = instantiations();
|
|
s += "<";
|
|
for (int i=0; i<types.count(); ++i) {
|
|
if (i > 0)
|
|
s += ", ";
|
|
s += types.at(i)->cppSignature();
|
|
}
|
|
s += " >";
|
|
}
|
|
|
|
if (actualIndirections()) {
|
|
s += ' ';
|
|
if (indirections())
|
|
s += QString(indirections(), '*');
|
|
if (isReference())
|
|
s += '&';
|
|
}
|
|
return s;
|
|
}
|
|
|
|
/*******************************************************************************
|
|
* AbstractMetaArgument
|
|
*/
|
|
AbstractMetaArgument *AbstractMetaArgument::copy() const
|
|
{
|
|
AbstractMetaArgument *cpy = new AbstractMetaArgument;
|
|
cpy->setName(AbstractMetaVariable::name());
|
|
cpy->setDefaultValueExpression(defaultValueExpression());
|
|
cpy->setType(type()->copy());
|
|
cpy->setArgumentIndex(argumentIndex());
|
|
|
|
return cpy;
|
|
}
|
|
|
|
|
|
QString AbstractMetaArgument::argumentName() const
|
|
{
|
|
QString n = AbstractMetaVariable::name();
|
|
if (n.isEmpty()) {
|
|
return QString("arg__%2").arg(m_argument_index + 1);
|
|
}
|
|
return n;
|
|
}
|
|
|
|
|
|
QString AbstractMetaArgument::indexedName() const
|
|
{
|
|
QString n = AbstractMetaVariable::name();
|
|
if (n.isEmpty())
|
|
return argumentName();
|
|
return QString("%1%2").arg(n).arg(m_argument_index);
|
|
}
|
|
|
|
QString AbstractMetaArgument::name() const
|
|
{
|
|
Q_ASSERT_X(0, "AbstractMetaArgument::name()", "use argumentName() or indexedName() instead");
|
|
return QString();
|
|
}
|
|
|
|
|
|
/*******************************************************************************
|
|
* AbstractMetaFunction
|
|
*/
|
|
AbstractMetaFunction::~AbstractMetaFunction()
|
|
{
|
|
qDeleteAll(m_arguments);
|
|
delete m_type;
|
|
}
|
|
|
|
/*******************************************************************************
|
|
* Indicates that this function has a modification that removes it
|
|
*/
|
|
bool AbstractMetaFunction::isModifiedRemoved(int types) const
|
|
{
|
|
FunctionModificationList mods = modifications(implementingClass());
|
|
foreach (FunctionModification mod, mods) {
|
|
if (!mod.isRemoveModifier())
|
|
continue;
|
|
|
|
if ((mod.removal & types) == types)
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
bool AbstractMetaFunction::needsCallThrough() const
|
|
{
|
|
if (ownerClass()->isInterface())
|
|
return false;
|
|
if (referenceCounts(implementingClass()).size() > 0)
|
|
return true;
|
|
if (argumentsHaveNativeId() || !isStatic())
|
|
return true;
|
|
|
|
foreach (const AbstractMetaArgument *arg, arguments()) {
|
|
if (arg->type()->isArray() || arg->type()->isTargetLangEnum() || arg->type()->isTargetLangFlags())
|
|
return true;
|
|
}
|
|
|
|
if (type() && (type()->isArray() || type()->isTargetLangEnum() || type()->isTargetLangFlags()))
|
|
return true;
|
|
|
|
for (int i=-1; i<=arguments().size(); ++i) {
|
|
TypeSystem::Ownership owner = this->ownership(implementingClass(), TypeSystem::TargetLangCode, i);
|
|
if (owner != TypeSystem::InvalidOwnership)
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
bool AbstractMetaFunction::needsSuppressUncheckedWarning() const
|
|
{
|
|
for (int i=-1; i<=arguments().size(); ++i) {
|
|
QList<ReferenceCount> referenceCounts = this->referenceCounts(implementingClass(), i);
|
|
foreach (ReferenceCount referenceCount, referenceCounts) {
|
|
if (referenceCount.action != ReferenceCount::Set)
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
QString AbstractMetaFunction::marshalledName() const
|
|
{
|
|
QString returned = "__qt_" + name();
|
|
AbstractMetaArgumentList arguments = this->arguments();
|
|
foreach (const AbstractMetaArgument *arg, arguments) {
|
|
returned += "_";
|
|
if (arg->type()->isNativePointer()) {
|
|
returned += "nativepointer";
|
|
} else if (arg->type()->isIntegerEnum() || arg->type()->isIntegerFlags()) {
|
|
returned += "int";
|
|
} else {
|
|
returned += arg->type()->name().replace("[]", "_3").replace(".", "_");
|
|
}
|
|
}
|
|
return returned;
|
|
}
|
|
|
|
bool AbstractMetaFunction::operator<(const AbstractMetaFunction &other) const
|
|
{
|
|
uint result = compareTo(&other);
|
|
return result & NameLessThan;
|
|
}
|
|
|
|
|
|
/*!
|
|
Returns a mask of CompareResult describing how this function is
|
|
compares to another function
|
|
*/
|
|
uint AbstractMetaFunction::compareTo(const AbstractMetaFunction *other) const
|
|
{
|
|
uint result = 0;
|
|
|
|
// Enclosing class...
|
|
if (ownerClass() == other->ownerClass()) {
|
|
result |= EqualImplementor;
|
|
}
|
|
|
|
// Attributes
|
|
if (attributes() == other->attributes()) {
|
|
result |= EqualAttributes;
|
|
}
|
|
|
|
// Compare types
|
|
AbstractMetaType *t = type();
|
|
AbstractMetaType *ot = other->type();
|
|
if ((!t && !ot) || ((t && ot && t->name() == ot->name()))) {
|
|
result |= EqualReturnType;
|
|
}
|
|
|
|
// Compare names
|
|
int cmp = originalName().compare(other->originalName());
|
|
|
|
if (cmp < 0) {
|
|
result |= NameLessThan;
|
|
} else if (cmp == 0) {
|
|
result |= EqualName;
|
|
}
|
|
|
|
// compare name after modification...
|
|
cmp = modifiedName().compare(other->modifiedName());
|
|
if (cmp == 0)
|
|
result |= EqualModifiedName;
|
|
|
|
// Compare arguments...
|
|
AbstractMetaArgumentList min_arguments;
|
|
AbstractMetaArgumentList max_arguments;
|
|
if (arguments().size() < other->arguments().size()) {
|
|
min_arguments = arguments();
|
|
max_arguments = other->arguments();
|
|
} else {
|
|
min_arguments = other->arguments();
|
|
max_arguments = arguments();
|
|
}
|
|
|
|
int min_count = min_arguments.size();
|
|
int max_count = max_arguments.size();
|
|
bool same = true;
|
|
for (int i=0; i<max_count; ++i) {
|
|
if (i < min_count) {
|
|
const AbstractMetaArgument *min_arg = min_arguments.at(i);
|
|
const AbstractMetaArgument *max_arg = max_arguments.at(i);
|
|
if (min_arg->type()->name() != max_arg->type()->name()
|
|
&& (min_arg->defaultValueExpression().isEmpty() || max_arg->defaultValueExpression().isEmpty())) {
|
|
same = false;
|
|
break;
|
|
}
|
|
} else {
|
|
if (max_arguments.at(i)->defaultValueExpression().isEmpty()) {
|
|
same = false;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (same)
|
|
result |= min_count == max_count ? EqualArguments : EqualDefaultValueOverload;
|
|
|
|
return result;
|
|
}
|
|
|
|
AbstractMetaFunction *AbstractMetaFunction::copy() const
|
|
{
|
|
AbstractMetaFunction *cpy = new AbstractMetaFunction;
|
|
cpy->setName(name());
|
|
cpy->setOriginalName(originalName());
|
|
cpy->setOwnerClass(ownerClass());
|
|
cpy->setImplementingClass(implementingClass());
|
|
cpy->setInterfaceClass(interfaceClass());
|
|
cpy->setFunctionType(functionType());
|
|
cpy->setAttributes(attributes());
|
|
cpy->setDeclaringClass(declaringClass());
|
|
if (type())
|
|
cpy->setType(type()->copy());
|
|
cpy->setConstant(isConstant());
|
|
cpy->setOriginalAttributes(originalAttributes());
|
|
|
|
foreach (AbstractMetaArgument *arg, arguments())
|
|
cpy->addArgument(arg->copy());
|
|
|
|
Q_ASSERT((!type() && !cpy->type())
|
|
|| (type()->instantiations() == cpy->type()->instantiations()));
|
|
|
|
return cpy;
|
|
}
|
|
|
|
QStringList AbstractMetaFunction::introspectionCompatibleSignatures(const QStringList &resolvedArguments) const
|
|
{
|
|
AbstractMetaArgumentList arguments = this->arguments();
|
|
if (arguments.size() == resolvedArguments.size()) {
|
|
return (QStringList() << QMetaObject::normalizedSignature((name() + "(" + resolvedArguments.join(",") + ")").toUtf8().constData()));
|
|
} else {
|
|
QStringList returned;
|
|
|
|
AbstractMetaArgument *argument = arguments.at(resolvedArguments.size());
|
|
QStringList minimalTypeSignature = argument->type()->minimalSignature().split("::");
|
|
for (int i=0; i<minimalTypeSignature.size(); ++i) {
|
|
returned += introspectionCompatibleSignatures(QStringList(resolvedArguments)
|
|
<< QStringList(minimalTypeSignature.mid(minimalTypeSignature.size() - i - 1)).join("::"));
|
|
}
|
|
|
|
return returned;
|
|
}
|
|
}
|
|
|
|
QString AbstractMetaFunction::signature() const
|
|
{
|
|
QString s(m_original_name);
|
|
|
|
s += "(";
|
|
|
|
for (int i=0; i<m_arguments.count(); ++i) {
|
|
if (i > 0)
|
|
s += ", ";
|
|
AbstractMetaArgument *a = m_arguments.at(i);
|
|
s += a->type()->cppSignature();
|
|
|
|
// We need to have the argument names in the qdoc files
|
|
s += " ";
|
|
s += a->argumentName();
|
|
}
|
|
s += ")";
|
|
|
|
if (isConstant())
|
|
s += " const";
|
|
|
|
return s;
|
|
}
|
|
|
|
int AbstractMetaFunction::actualMinimumArgumentCount() const
|
|
{
|
|
AbstractMetaArgumentList arguments = this->arguments();
|
|
|
|
int count = 0;
|
|
for (int i=0; i<arguments.size(); ++i && ++count) {
|
|
if (argumentRemoved(i + 1)) --count;
|
|
else if (!arguments.at(i)->defaultValueExpression().isEmpty()) break;
|
|
}
|
|
|
|
return count;
|
|
}
|
|
|
|
// Returns reference counts for argument at idx, or all arguments if idx == -2
|
|
QList<ReferenceCount> AbstractMetaFunction::referenceCounts(const AbstractMetaClass *cls, int idx) const
|
|
{
|
|
QList<ReferenceCount> returned;
|
|
|
|
FunctionModificationList mods = this->modifications(cls);
|
|
foreach (FunctionModification mod, mods) {
|
|
QList<ArgumentModification> argument_mods = mod.argument_mods;
|
|
foreach (ArgumentModification argument_mod, argument_mods) {
|
|
if (argument_mod.index != idx && idx != -2)
|
|
continue;
|
|
returned += argument_mod.referenceCounts;
|
|
}
|
|
}
|
|
|
|
return returned;
|
|
}
|
|
|
|
QString AbstractMetaFunction::replacedDefaultExpression(const AbstractMetaClass *cls, int key) const
|
|
{
|
|
FunctionModificationList modifications = this->modifications(cls);
|
|
foreach (FunctionModification modification, modifications) {
|
|
QList<ArgumentModification> argument_modifications = modification.argument_mods;
|
|
foreach (ArgumentModification argument_modification, argument_modifications) {
|
|
if (argument_modification.index == key
|
|
&& !argument_modification.replaced_default_expression.isEmpty()) {
|
|
return argument_modification.replaced_default_expression;
|
|
}
|
|
}
|
|
}
|
|
|
|
return QString();
|
|
}
|
|
|
|
bool AbstractMetaFunction::removedDefaultExpression(const AbstractMetaClass *cls, int key) const
|
|
{
|
|
FunctionModificationList modifications = this->modifications(cls);
|
|
foreach (FunctionModification modification, modifications) {
|
|
QList<ArgumentModification> argument_modifications = modification.argument_mods;
|
|
foreach (ArgumentModification argument_modification, argument_modifications) {
|
|
if (argument_modification.index == key
|
|
&& argument_modification.removed_default_expression) {
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
bool AbstractMetaFunction::resetObjectAfterUse(int argument_idx) const
|
|
{
|
|
const AbstractMetaClass *cls = declaringClass();
|
|
FunctionModificationList modifications = this->modifications(cls);
|
|
foreach (FunctionModification modification, modifications) {
|
|
QList<ArgumentModification> argumentModifications = modification.argument_mods;
|
|
foreach (ArgumentModification argumentModification, argumentModifications) {
|
|
if (argumentModification.index == argument_idx && argumentModification.reset_after_use)
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
QString AbstractMetaFunction::nullPointerDefaultValue(const AbstractMetaClass *mainClass, int argument_idx) const
|
|
{
|
|
Q_ASSERT(nullPointersDisabled(mainClass, argument_idx));
|
|
|
|
const AbstractMetaClass *cls = mainClass;
|
|
if (cls == 0)
|
|
cls = implementingClass();
|
|
|
|
do {
|
|
FunctionModificationList modifications = this->modifications(cls);
|
|
foreach (FunctionModification modification, modifications) {
|
|
QList<ArgumentModification> argument_modifications = modification.argument_mods;
|
|
foreach (ArgumentModification argument_modification, argument_modifications) {
|
|
if (argument_modification.index == argument_idx
|
|
&& argument_modification.no_null_pointers) {
|
|
return argument_modification.null_pointer_default_value;
|
|
}
|
|
}
|
|
}
|
|
|
|
cls = cls->baseClass();
|
|
} while (cls != 0 && mainClass == 0); // Once when mainClass != 0, or once for all base classes of implementing class
|
|
|
|
return QString();
|
|
|
|
}
|
|
|
|
bool AbstractMetaFunction::nullPointersDisabled(const AbstractMetaClass *mainClass, int argument_idx) const
|
|
{
|
|
const AbstractMetaClass *cls = mainClass;
|
|
if (cls == 0)
|
|
cls = implementingClass();
|
|
|
|
do {
|
|
FunctionModificationList modifications = this->modifications(cls);
|
|
foreach (FunctionModification modification, modifications) {
|
|
QList<ArgumentModification> argument_modifications = modification.argument_mods;
|
|
foreach (ArgumentModification argument_modification, argument_modifications) {
|
|
if (argument_modification.index == argument_idx
|
|
&& argument_modification.no_null_pointers) {
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
|
|
cls = cls->baseClass();
|
|
} while (cls != 0 && mainClass == 0); // Once when mainClass != 0, or once for all base classes of implementing class
|
|
|
|
return false;
|
|
}
|
|
|
|
QString AbstractMetaFunction::conversionRule(TypeSystem::Language language, int key) const
|
|
{
|
|
FunctionModificationList modifications = this->modifications(declaringClass());
|
|
foreach (FunctionModification modification, modifications) {
|
|
QList<ArgumentModification> argument_modifications = modification.argument_mods;
|
|
foreach (ArgumentModification argument_modification, argument_modifications) {
|
|
if (argument_modification.index != key)
|
|
continue;
|
|
|
|
foreach (CodeSnip snip, argument_modification.conversion_rules) {
|
|
if (snip.language == language && !snip.code().isEmpty())
|
|
return snip.code();
|
|
}
|
|
}
|
|
}
|
|
|
|
return QString();
|
|
}
|
|
|
|
QString AbstractMetaFunction::argumentReplaced(int key) const
|
|
{
|
|
FunctionModificationList modifications = this->modifications(declaringClass());
|
|
foreach (FunctionModification modification, modifications) {
|
|
QList<ArgumentModification> argument_modifications = modification.argument_mods;
|
|
foreach (ArgumentModification argument_modification, argument_modifications) {
|
|
if (argument_modification.index == key && !argument_modification.replace_value.isEmpty()) {
|
|
return argument_modification.replace_value;
|
|
}
|
|
}
|
|
}
|
|
|
|
return "";
|
|
}
|
|
|
|
bool AbstractMetaFunction::argumentRemoved(int key) const
|
|
{
|
|
FunctionModificationList modifications = this->modifications(declaringClass());
|
|
foreach (FunctionModification modification, modifications) {
|
|
QList<ArgumentModification> argument_modifications = modification.argument_mods;
|
|
foreach (ArgumentModification argument_modification, argument_modifications) {
|
|
if (argument_modification.index == key) {
|
|
if (argument_modification.removed) {
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
bool AbstractMetaFunction::isVirtualSlot() const
|
|
{
|
|
FunctionModificationList modifications = this->modifications(declaringClass());
|
|
foreach (FunctionModification modification, modifications) {
|
|
if (modification.isVirtualSlot())
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
bool AbstractMetaFunction::disabledGarbageCollection(const AbstractMetaClass *cls, int key) const
|
|
{
|
|
FunctionModificationList modifications = this->modifications(cls);
|
|
foreach (FunctionModification modification, modifications) {
|
|
QList<ArgumentModification> argument_modifications = modification.argument_mods;
|
|
foreach (ArgumentModification argument_modification, argument_modifications) {
|
|
if (argument_modification.index != key)
|
|
continue;
|
|
|
|
foreach (TypeSystem::Ownership ownership, argument_modification.ownerships.values()) {
|
|
if (ownership == TypeSystem::CppOwnership)
|
|
return true;
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
bool AbstractMetaFunction::isDeprecated() const
|
|
{
|
|
FunctionModificationList modifications = this->modifications(declaringClass());
|
|
foreach (FunctionModification modification, modifications) {
|
|
if (modification.isDeprecated())
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
TypeSystem::Ownership AbstractMetaFunction::ownership(const AbstractMetaClass *cls, TypeSystem::Language language, int key) const
|
|
{
|
|
FunctionModificationList modifications = this->modifications(cls);
|
|
foreach (FunctionModification modification, modifications) {
|
|
QList<ArgumentModification> argument_modifications = modification.argument_mods;
|
|
foreach (ArgumentModification argument_modification, argument_modifications) {
|
|
if (argument_modification.index == key)
|
|
return argument_modification.ownerships.value(language, TypeSystem::InvalidOwnership);
|
|
}
|
|
}
|
|
|
|
return TypeSystem::InvalidOwnership;
|
|
}
|
|
|
|
bool AbstractMetaFunction::isRemovedFromAllLanguages(const AbstractMetaClass *cls) const
|
|
{
|
|
return isRemovedFrom(cls, TypeSystem::All);
|
|
}
|
|
|
|
bool AbstractMetaFunction::isRemovedFrom(const AbstractMetaClass *cls, TypeSystem::Language language) const
|
|
{
|
|
FunctionModificationList modifications = this->modifications(cls);
|
|
foreach (FunctionModification modification, modifications) {
|
|
if ((modification.removal & language) == language)
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
QString AbstractMetaFunction::typeReplaced(int key) const
|
|
{
|
|
FunctionModificationList modifications = this->modifications(declaringClass());
|
|
foreach (FunctionModification modification, modifications) {
|
|
QList<ArgumentModification> argument_modifications = modification.argument_mods;
|
|
foreach (ArgumentModification argument_modification, argument_modifications) {
|
|
if (argument_modification.index == key
|
|
&& !argument_modification.modified_type.isEmpty()) {
|
|
return argument_modification.modified_type;
|
|
}
|
|
}
|
|
}
|
|
|
|
return QString();
|
|
}
|
|
|
|
QString AbstractMetaFunction::minimalSignature() const
|
|
{
|
|
if (!m_cached_minimal_signature.isEmpty())
|
|
return m_cached_minimal_signature;
|
|
|
|
QString minimalSignature = originalName() + "(";
|
|
AbstractMetaArgumentList arguments = this->arguments();
|
|
|
|
for (int i=0; i<arguments.count(); ++i) {
|
|
AbstractMetaType *t = arguments.at(i)->type();
|
|
|
|
if (i > 0)
|
|
minimalSignature += ",";
|
|
|
|
minimalSignature += t->minimalSignature();
|
|
}
|
|
minimalSignature += ")";
|
|
if (isConstant())
|
|
minimalSignature += "const";
|
|
|
|
minimalSignature = QMetaObject::normalizedSignature(minimalSignature.toLocal8Bit().constData());
|
|
m_cached_minimal_signature = minimalSignature;
|
|
|
|
return minimalSignature;
|
|
}
|
|
|
|
FunctionModificationList AbstractMetaFunction::modifications(const AbstractMetaClass *implementor) const
|
|
{
|
|
Q_ASSERT(implementor);
|
|
return implementor->typeEntry()->functionModifications(minimalSignature());
|
|
}
|
|
|
|
bool AbstractMetaFunction::hasModifications(const AbstractMetaClass *implementor) const
|
|
{
|
|
FunctionModificationList mods = modifications(implementor);
|
|
return mods.count() > 0;
|
|
}
|
|
|
|
QString AbstractMetaFunction::modifiedName() const
|
|
{
|
|
if (m_cached_modified_name.isEmpty()) {
|
|
FunctionModificationList mods = modifications(implementingClass());
|
|
foreach (FunctionModification mod, mods) {
|
|
if (mod.isRenameModifier()) {
|
|
m_cached_modified_name = mod.renamedToName;
|
|
break;
|
|
}
|
|
}
|
|
if (m_cached_modified_name.isEmpty())
|
|
m_cached_modified_name = name();
|
|
}
|
|
return m_cached_modified_name;
|
|
}
|
|
|
|
QString AbstractMetaFunction::targetLangSignature(bool minimal) const
|
|
{
|
|
QString s;
|
|
|
|
// Attributes...
|
|
if (!minimal) {
|
|
#if 0 // jambi
|
|
if (isPublic()) s += "public ";
|
|
else if (isProtected()) s += "protected ";
|
|
else if (isPrivate()) s += "private ";
|
|
|
|
// if (isNative()) s += "native ";
|
|
// else
|
|
if (isFinalInTargetLang()) s += "final ";
|
|
else if (isAbstract()) s += "abstract ";
|
|
|
|
if (isStatic()) s += "static ";
|
|
#endif
|
|
// Return type
|
|
if (type())
|
|
s += type()->name() + " ";
|
|
else
|
|
s += "void ";
|
|
}
|
|
|
|
s += modifiedName();
|
|
s += "(";
|
|
|
|
int j = 0;
|
|
for (int i=0; i<m_arguments.size(); ++i) {
|
|
if (argumentRemoved(i+1))
|
|
continue;
|
|
if (j != 0) {
|
|
s += ",";
|
|
if (!minimal)
|
|
s += QLatin1Char(' ');
|
|
}
|
|
s += m_arguments.at(i)->type()->name();
|
|
|
|
if (!minimal) {
|
|
s += " ";
|
|
s += m_arguments.at(i)->argumentName();
|
|
}
|
|
++j;
|
|
}
|
|
|
|
s += ")";
|
|
|
|
return s;
|
|
}
|
|
|
|
|
|
bool function_sorter(AbstractMetaFunction *a, AbstractMetaFunction *b)
|
|
{
|
|
return a->signature() < b->signature();
|
|
}
|
|
|
|
/*******************************************************************************
|
|
* AbstractMetaClass
|
|
*/
|
|
AbstractMetaClass::~AbstractMetaClass()
|
|
{
|
|
qDeleteAll(m_functions);
|
|
qDeleteAll(m_fields);
|
|
}
|
|
|
|
/*AbstractMetaClass *AbstractMetaClass::copy() const
|
|
{
|
|
AbstractMetaClass *cls = new AbstractMetaClass;
|
|
cls->setAttributes(attributes());
|
|
cls->setBaseClass(baseClass());
|
|
cls->setTypeEntry(typeEntry());
|
|
foreach (AbstractMetaFunction *function, functions()) {
|
|
AbstractMetaFunction *copy = function->copy();
|
|
function->setImplementingClass(cls);
|
|
cls->addFunction(copy);
|
|
}
|
|
cls->setEnums(enums());
|
|
foreach (const AbstractMetaField *field, fields()) {
|
|
AbstractMetaField *copy = field->copy();
|
|
copy->setEnclosingClass(cls);
|
|
cls->addField(copy);
|
|
}
|
|
cls->setInterfaces(interfaces());
|
|
|
|
return cls;
|
|
}*/
|
|
|
|
/*******************************************************************************
|
|
* Returns true if this class is a subclass of the given class
|
|
*/
|
|
bool AbstractMetaClass::inheritsFrom(const AbstractMetaClass *cls) const
|
|
{
|
|
Q_ASSERT(cls != 0);
|
|
|
|
const AbstractMetaClass *clazz = this;
|
|
while (clazz != 0) {
|
|
if (clazz == cls)
|
|
return true;
|
|
|
|
clazz = clazz->baseClass();
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
/*******************************************************************************
|
|
* Constructs an interface based on the functions and enums in this
|
|
* class and returns it...
|
|
*/
|
|
AbstractMetaClass *AbstractMetaClass::extractInterface()
|
|
{
|
|
Q_ASSERT(typeEntry()->designatedInterface());
|
|
|
|
if (m_extracted_interface == 0) {
|
|
AbstractMetaClass *iface = new AbstractMetaClass;
|
|
iface->setAttributes(attributes());
|
|
iface->setBaseClass(0);
|
|
iface->setPrimaryInterfaceImplementor(this);
|
|
|
|
iface->setTypeEntry(typeEntry()->designatedInterface());
|
|
|
|
foreach (AbstractMetaFunction *function, functions()) {
|
|
if (!function->isConstructor())
|
|
iface->addFunction(function->copy());
|
|
}
|
|
|
|
// iface->setEnums(enums());
|
|
// setEnums(AbstractMetaEnumList());
|
|
|
|
foreach (const AbstractMetaField *field, fields()) {
|
|
if (field->isPublic()) {
|
|
AbstractMetaField *new_field = field->copy();
|
|
new_field->setEnclosingClass(iface);
|
|
iface->addField(new_field);
|
|
}
|
|
}
|
|
|
|
m_extracted_interface = iface;
|
|
addInterface(iface);
|
|
}
|
|
|
|
return m_extracted_interface;
|
|
}
|
|
|
|
/*******************************************************************************
|
|
* Returns a list of all the functions with a given name
|
|
*/
|
|
AbstractMetaFunctionList AbstractMetaClass::queryFunctionsByName(const QString &name) const
|
|
{
|
|
AbstractMetaFunctionList returned;
|
|
AbstractMetaFunctionList functions = this->functions();
|
|
foreach (AbstractMetaFunction *function, functions) {
|
|
if (function->name() == name)
|
|
returned.append(function);
|
|
}
|
|
|
|
return returned;
|
|
}
|
|
|
|
bool AbstractMetaClass::hasDefaultIsNull() const
|
|
{
|
|
foreach(const AbstractMetaFunction* fun, queryFunctionsByName("isNull")) {
|
|
if (fun->actualMinimumArgumentCount()==0) {
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
/*******************************************************************************
|
|
* Returns all reference count modifications for any function in the class
|
|
*/
|
|
QList<ReferenceCount> AbstractMetaClass::referenceCounts() const
|
|
{
|
|
QList<ReferenceCount> returned;
|
|
|
|
AbstractMetaFunctionList functions = this->functions();
|
|
foreach (AbstractMetaFunction *function, functions) {
|
|
returned += function->referenceCounts(this);
|
|
}
|
|
|
|
return returned;
|
|
}
|
|
|
|
/*******************************************************************************
|
|
* Returns a list of all the functions retrieved during parsing which should
|
|
* be added to the Java API.
|
|
*/
|
|
AbstractMetaFunctionList AbstractMetaClass::functionsInTargetLang() const
|
|
{
|
|
int default_flags = NormalFunctions | Visible | NotRemovedFromTargetLang;
|
|
|
|
// Interfaces don't implement functions
|
|
default_flags |= isInterface() ? 0 : ClassImplements;
|
|
|
|
// Only public functions in final classes
|
|
// default_flags |= isFinal() ? WasPublic : 0;
|
|
int public_flags = isFinal() ? WasPublic : 0;
|
|
|
|
// Constructors
|
|
AbstractMetaFunctionList returned = queryFunctions(Constructors | default_flags | public_flags);
|
|
|
|
// Final functions
|
|
returned += queryFunctions(FinalInTargetLangFunctions | NonStaticFunctions | default_flags | public_flags);
|
|
|
|
// Virtual functions
|
|
returned += queryFunctions(VirtualInTargetLangFunctions | NonStaticFunctions | default_flags | public_flags);
|
|
|
|
// Static functions
|
|
returned += queryFunctions(StaticFunctions | default_flags | public_flags);
|
|
|
|
// Empty, private functions, since they aren't caught by the other ones
|
|
returned += queryFunctions(Empty | Invisible);
|
|
|
|
return returned;
|
|
}
|
|
|
|
AbstractMetaFunctionList AbstractMetaClass::virtualFunctions() const
|
|
{
|
|
AbstractMetaFunctionList list = functionsInShellClass();
|
|
|
|
AbstractMetaFunctionList returned;
|
|
foreach (AbstractMetaFunction *f, list) {
|
|
if (!f->isFinalInCpp() || f->isVirtualSlot())
|
|
returned += f;
|
|
}
|
|
|
|
return returned;
|
|
}
|
|
|
|
AbstractMetaFunctionList AbstractMetaClass::nonVirtualShellFunctions() const
|
|
{
|
|
AbstractMetaFunctionList list = functionsInShellClass();
|
|
AbstractMetaFunctionList returned;
|
|
foreach (AbstractMetaFunction *f, list) {
|
|
if (f->isFinalInCpp() && !f->isVirtualSlot())
|
|
returned += f;
|
|
}
|
|
|
|
return returned;
|
|
}
|
|
|
|
/*******************************************************************************
|
|
* Returns a list of all functions that should be declared and implemented in
|
|
* the shell class which is generated as a wrapper on top of the actual C++ class
|
|
*/
|
|
AbstractMetaFunctionList AbstractMetaClass::functionsInShellClass() const
|
|
{
|
|
// Only functions and only protected and public functions
|
|
int default_flags = NormalFunctions | Visible | WasVisible | NotRemovedFromShell;
|
|
|
|
// All virtual functions
|
|
AbstractMetaFunctionList returned = queryFunctions(VirtualFunctions | default_flags);
|
|
|
|
// All functions explicitly set to be implemented by the shell class
|
|
// (mainly superclass functions that are hidden by other declarations)
|
|
returned += queryFunctions(ForcedShellFunctions | default_flags);
|
|
|
|
// All functions explicitly set to be virtual slots
|
|
returned += queryFunctions(VirtualSlots | default_flags);
|
|
|
|
return returned;
|
|
}
|
|
|
|
/*******************************************************************************
|
|
* Returns a list of all functions that require a public override function to
|
|
* be generated in the shell class. This includes all functions that were originally
|
|
* protected in the superclass.
|
|
*/
|
|
AbstractMetaFunctionList AbstractMetaClass::publicOverrideFunctions() const
|
|
{
|
|
return queryFunctions(NormalFunctions | WasProtected | FinalInCppFunctions | NotRemovedFromTargetLang)
|
|
+ queryFunctions(Signals | WasProtected | FinalInCppFunctions | NotRemovedFromTargetLang);
|
|
}
|
|
|
|
AbstractMetaFunctionList AbstractMetaClass::virtualOverrideFunctions() const
|
|
{
|
|
return queryFunctions(NormalFunctions | NonEmptyFunctions | Visible | VirtualInCppFunctions | NotRemovedFromShell) +
|
|
queryFunctions(Signals | NonEmptyFunctions | Visible | VirtualInCppFunctions | NotRemovedFromShell);
|
|
}
|
|
|
|
void AbstractMetaClass::sortFunctions()
|
|
{
|
|
qSort(m_functions.begin(), m_functions.end(), function_sorter);
|
|
}
|
|
|
|
void AbstractMetaClass::setFunctions(const AbstractMetaFunctionList &functions)
|
|
{
|
|
m_functions = functions;
|
|
|
|
// Functions must be sorted by name before next loop
|
|
sortFunctions();
|
|
|
|
QString currentName;
|
|
bool hasVirtuals = false;
|
|
AbstractMetaFunctionList final_functions;
|
|
foreach (AbstractMetaFunction *f, m_functions) {
|
|
f->setOwnerClass(this);
|
|
|
|
m_has_virtual_slots |= f->isVirtualSlot();
|
|
m_has_virtuals |= !f->isFinal() || f->isVirtualSlot();
|
|
m_has_nonpublic |= !f->isPublic();
|
|
|
|
// If we have non-virtual overloads of a virtual function, we have to implement
|
|
// all the overloads in the shell class to override the hiding rule
|
|
if (currentName == f->name()) {
|
|
hasVirtuals = hasVirtuals || !f->isFinal();
|
|
if (f->isFinal())
|
|
final_functions += f;
|
|
} else {
|
|
if (hasVirtuals && final_functions.size() > 0) {
|
|
foreach (AbstractMetaFunction *final_function, final_functions) {
|
|
*final_function += AbstractMetaAttributes::ForceShellImplementation;
|
|
|
|
QString warn = QString("hiding of function '%1' in class '%2'")
|
|
.arg(final_function->name()).arg(name());
|
|
ReportHandler::warning(warn);
|
|
}
|
|
}
|
|
|
|
hasVirtuals = !f->isFinal();
|
|
final_functions.clear();
|
|
if (f->isFinal())
|
|
final_functions += f;
|
|
currentName = f->name();
|
|
}
|
|
}
|
|
|
|
#ifndef QT_NO_DEBUG
|
|
bool duplicate_function = false;
|
|
for (int j=0; j<m_functions.size(); ++j) {
|
|
FunctionModificationList mods = m_functions.at(j)->modifications(m_functions.at(j)->implementingClass());
|
|
|
|
bool removed = false;
|
|
foreach (const FunctionModification &mod, mods) {
|
|
if (mod.isRemoveModifier()) {
|
|
removed = true;
|
|
break ;
|
|
}
|
|
}
|
|
if (removed)
|
|
continue ;
|
|
|
|
for (int i=0; i<m_functions.size() - 1; ++i) {
|
|
if (j == i)
|
|
continue;
|
|
|
|
mods = m_functions.at(i)->modifications(m_functions.at(i)->implementingClass());
|
|
bool removed = false;
|
|
foreach (const FunctionModification &mod, mods) {
|
|
if (mod.isRemoveModifier()) {
|
|
removed = true;
|
|
break ;
|
|
}
|
|
}
|
|
if (removed)
|
|
continue ;
|
|
|
|
uint cmp = m_functions.at(i)->compareTo(m_functions.at(j));
|
|
if ((cmp & AbstractMetaFunction::EqualName) && (cmp & AbstractMetaFunction::EqualArguments)) {
|
|
printf("%s.%s mostly equal to %s.%s\n",
|
|
qPrintable(m_functions.at(i)->implementingClass()->typeEntry()->qualifiedCppName()),
|
|
qPrintable(m_functions.at(i)->signature()),
|
|
qPrintable(m_functions.at(j)->implementingClass()->typeEntry()->qualifiedCppName()),
|
|
qPrintable(m_functions.at(j)->signature()));
|
|
duplicate_function = true;
|
|
}
|
|
}
|
|
}
|
|
//Q_ASSERT(!duplicate_function);
|
|
#endif
|
|
}
|
|
|
|
bool AbstractMetaClass::hasFieldAccessors() const
|
|
{
|
|
foreach (const AbstractMetaField *field, fields()) {
|
|
if (field->getter() || field->setter())
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
bool AbstractMetaClass::hasDefaultToStringFunction() const
|
|
{
|
|
foreach (AbstractMetaFunction *f, queryFunctionsByName("toString")) {
|
|
if (f->actualMinimumArgumentCount() == 0) {
|
|
return true;
|
|
}
|
|
|
|
}
|
|
return false;
|
|
}
|
|
|
|
void AbstractMetaClass::addFunction(AbstractMetaFunction *function)
|
|
{
|
|
function->setOwnerClass(this);
|
|
|
|
if (!function->isDestructor()) {
|
|
m_functions << function;
|
|
qSort(m_functions.begin(), m_functions.end(), function_sorter);
|
|
}
|
|
|
|
|
|
m_has_virtual_slots |= function->isVirtualSlot();
|
|
m_has_virtuals |= !function->isFinal() || function->isVirtualSlot();
|
|
m_has_nonpublic |= !function->isPublic();
|
|
}
|
|
|
|
bool AbstractMetaClass::hasSignal(const AbstractMetaFunction *other) const
|
|
{
|
|
if (!other->isSignal())
|
|
return false;
|
|
|
|
foreach (const AbstractMetaFunction *f, functions()) {
|
|
if (f->isSignal() && f->compareTo(other) & AbstractMetaFunction::EqualName)
|
|
return other->modifiedName() == f->modifiedName();
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
QString AbstractMetaClass::name() const
|
|
{
|
|
return QString(m_type_entry->targetLangName()).replace("::", "_");
|
|
}
|
|
|
|
bool AbstractMetaClass::hasFunction(const QString &str) const
|
|
{
|
|
foreach (const AbstractMetaFunction *f, functions())
|
|
if (f->name() == str)
|
|
return true;
|
|
return false;
|
|
}
|
|
|
|
/* Returns true if this class has one or more functions that are
|
|
protected. If a class has protected members we need to generate a
|
|
shell class with public accessors to the protected functions, so
|
|
they can be called from the native functions.
|
|
*/
|
|
bool AbstractMetaClass::hasProtectedFunctions() const {
|
|
foreach (AbstractMetaFunction *func, m_functions) {
|
|
if (func->isProtected())
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool AbstractMetaClass::generateShellClass() const
|
|
{
|
|
return m_force_shell_class ||
|
|
(!isFinal()
|
|
&& (hasVirtualFunctions()
|
|
|| hasProtectedFunctions()
|
|
|| hasFieldAccessors()));
|
|
}
|
|
|
|
QPropertySpec *AbstractMetaClass::propertySpecForRead(const QString &name) const
|
|
{
|
|
for (int i=0; i<m_property_specs.size(); ++i)
|
|
if (name == m_property_specs.at(i)->read())
|
|
return m_property_specs.at(i);
|
|
return 0;
|
|
}
|
|
|
|
QPropertySpec *AbstractMetaClass::propertySpecForWrite(const QString &name) const
|
|
{
|
|
for (int i=0; i<m_property_specs.size(); ++i)
|
|
if (name == m_property_specs.at(i)->write())
|
|
return m_property_specs.at(i);
|
|
return 0;
|
|
}
|
|
|
|
QPropertySpec *AbstractMetaClass::propertySpecForReset(const QString &name) const
|
|
{
|
|
for (int i=0; i<m_property_specs.size(); ++i) {
|
|
if (name == m_property_specs.at(i)->reset())
|
|
return m_property_specs.at(i);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
|
|
|
|
static bool functions_contains(const AbstractMetaFunctionList &l, const AbstractMetaFunction *func)
|
|
{
|
|
foreach (const AbstractMetaFunction *f, l) {
|
|
if ((f->compareTo(func) & AbstractMetaFunction::PrettySimilar) == AbstractMetaFunction::PrettySimilar)
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
AbstractMetaField::AbstractMetaField() : m_getter(0), m_setter(0), m_class(0)
|
|
{
|
|
}
|
|
|
|
AbstractMetaField::~AbstractMetaField()
|
|
{
|
|
delete m_setter;
|
|
delete m_getter;
|
|
}
|
|
ushort painters; // refcount
|
|
AbstractMetaField *AbstractMetaField::copy() const
|
|
{
|
|
AbstractMetaField *returned = new AbstractMetaField;
|
|
returned->setEnclosingClass(0);
|
|
returned->setAttributes(attributes());
|
|
returned->setName(name());
|
|
returned->setType(type()->copy());
|
|
returned->setOriginalAttributes(originalAttributes());
|
|
|
|
return returned;
|
|
}
|
|
|
|
static QString upCaseFirst(const QString &str) {
|
|
Q_ASSERT(!str.isEmpty());
|
|
QString s = str;
|
|
s[0] = s.at(0).toUpper();
|
|
return s;
|
|
}
|
|
|
|
static AbstractMetaFunction *createXetter(const AbstractMetaField *g, const QString &name, uint type) {
|
|
AbstractMetaFunction *f = new AbstractMetaFunction;
|
|
|
|
|
|
|
|
f->setName(name);
|
|
f->setOriginalName(name);
|
|
f->setOwnerClass(g->enclosingClass());
|
|
f->setImplementingClass(g->enclosingClass());
|
|
f->setDeclaringClass(g->enclosingClass());
|
|
|
|
uint attr = AbstractMetaAttributes::Native
|
|
| AbstractMetaAttributes::Final
|
|
| type;
|
|
if (g->isStatic())
|
|
attr |= AbstractMetaAttributes::Static;
|
|
if (g->isPublic())
|
|
attr |= AbstractMetaAttributes::Public;
|
|
else if (g->isProtected())
|
|
attr |= AbstractMetaAttributes::Protected;
|
|
else
|
|
attr |= AbstractMetaAttributes::Private;
|
|
f->setAttributes(attr);
|
|
f->setOriginalAttributes(attr);
|
|
|
|
FieldModificationList mods = g->modifications();
|
|
foreach (FieldModification mod, mods) {
|
|
if (mod.isRenameModifier())
|
|
f->setName(mod.renamedTo());
|
|
if (mod.isAccessModifier()) {
|
|
if (mod.isPrivate())
|
|
f->setVisibility(AbstractMetaAttributes::Private);
|
|
else if (mod.isProtected())
|
|
f->setVisibility(AbstractMetaAttributes::Protected);
|
|
else if (mod.isPublic())
|
|
f->setVisibility(AbstractMetaAttributes::Public);
|
|
else if (mod.isFriendly())
|
|
f->setVisibility(AbstractMetaAttributes::Friendly);
|
|
}
|
|
|
|
}
|
|
return f;
|
|
}
|
|
|
|
FieldModificationList AbstractMetaField::modifications() const
|
|
{
|
|
FieldModificationList mods = enclosingClass()->typeEntry()->fieldModifications();
|
|
FieldModificationList returned;
|
|
|
|
foreach (FieldModification mod, mods) {
|
|
if (mod.name == name())
|
|
returned += mod;
|
|
}
|
|
|
|
return returned;
|
|
}
|
|
|
|
const AbstractMetaFunction *AbstractMetaField::setter() const
|
|
{
|
|
if (m_setter == 0) {
|
|
m_setter = createXetter(this,
|
|
name(),
|
|
AbstractMetaAttributes::SetterFunction);
|
|
AbstractMetaArgumentList arguments;
|
|
AbstractMetaArgument *argument = new AbstractMetaArgument;
|
|
argument->setType(type()->copy());
|
|
argument->setName(name());
|
|
arguments.append(argument);
|
|
m_setter->setArguments(arguments);
|
|
}
|
|
return m_setter;
|
|
}
|
|
|
|
const AbstractMetaFunction *AbstractMetaField::getter() const
|
|
{
|
|
if (m_getter == 0) {
|
|
m_getter = createXetter(this,
|
|
name(),
|
|
AbstractMetaAttributes::GetterFunction);
|
|
m_getter->setType(type());
|
|
}
|
|
|
|
return m_getter;
|
|
}
|
|
|
|
|
|
bool AbstractMetaClass::hasConstructors() const
|
|
{
|
|
return queryFunctions(Constructors).size() != 0;
|
|
}
|
|
|
|
void AbstractMetaClass::addDefaultConstructor()
|
|
{
|
|
AbstractMetaFunction *f = new AbstractMetaFunction;
|
|
f->setName(name());
|
|
f->setOwnerClass(this);
|
|
f->setFunctionType(AbstractMetaFunction::ConstructorFunction);
|
|
f->setArguments(AbstractMetaArgumentList());
|
|
f->setDeclaringClass(this);
|
|
|
|
uint attr = AbstractMetaAttributes::Native;
|
|
attr |= AbstractMetaAttributes::Public;
|
|
f->setAttributes(attr);
|
|
f->setImplementingClass(this);
|
|
f->setOriginalAttributes(f->attributes());
|
|
|
|
addFunction(f);
|
|
}
|
|
|
|
bool AbstractMetaClass::hasFunction(const AbstractMetaFunction *f) const
|
|
{
|
|
return functions_contains(m_functions, f);
|
|
}
|
|
|
|
/* Goes through the list of functions and returns a list of all
|
|
functions matching all of the criteria in \a query.
|
|
*/
|
|
|
|
AbstractMetaFunctionList AbstractMetaClass::queryFunctions(uint query) const
|
|
{
|
|
AbstractMetaFunctionList functions;
|
|
|
|
foreach (AbstractMetaFunction *f, m_functions) {
|
|
|
|
if ((query & VirtualSlots) && !f->isVirtualSlot())
|
|
continue;
|
|
|
|
if ((query & NotRemovedFromTargetLang) && f->isRemovedFrom(f->implementingClass(), TypeSystem::TargetLangCode)) {
|
|
continue;
|
|
}
|
|
|
|
if ((query & NotRemovedFromTargetLang) && !f->isFinal() && f->isRemovedFrom(f->declaringClass(), TypeSystem::TargetLangCode)) {
|
|
continue;
|
|
}
|
|
|
|
if ((query & NotRemovedFromShell) && f->isRemovedFrom(f->implementingClass(), TypeSystem::ShellCode)) {
|
|
continue;
|
|
}
|
|
|
|
if ((query & NotRemovedFromShell) && !f->isFinal() && f->isRemovedFrom(f->declaringClass(), TypeSystem::ShellCode)) {
|
|
continue;
|
|
}
|
|
|
|
if ((query & Visible) && f->isPrivate()) {
|
|
continue;
|
|
}
|
|
|
|
if ((query & VirtualInTargetLangFunctions) && f->isFinalInTargetLang()) {
|
|
continue;
|
|
}
|
|
|
|
if ((query & Invisible) && !f->isPrivate()) {
|
|
continue;
|
|
}
|
|
|
|
if ((query & Empty) && !f->isEmptyFunction()) {
|
|
continue;
|
|
}
|
|
|
|
if ((query & WasPublic) && !f->wasPublic()) {
|
|
continue;
|
|
}
|
|
|
|
if ((query & WasVisible) && f->wasPrivate()) {
|
|
continue;
|
|
}
|
|
|
|
if ((query & WasProtected) && !f->wasProtected()) {
|
|
continue;
|
|
}
|
|
|
|
if ((query & ClassImplements) && f->ownerClass() != f->implementingClass()) {
|
|
continue;
|
|
}
|
|
|
|
if ((query & Inconsistent) && (f->isFinalInTargetLang() || !f->isFinalInCpp() || f->isStatic())) {
|
|
continue;
|
|
}
|
|
|
|
if ((query & FinalInTargetLangFunctions) && !f->isFinalInTargetLang()) {
|
|
continue;
|
|
}
|
|
|
|
if ((query & FinalInCppFunctions) && !f->isFinalInCpp()) {
|
|
continue;
|
|
}
|
|
|
|
if ((query & VirtualInCppFunctions) && f->isFinalInCpp()) {
|
|
continue;
|
|
}
|
|
|
|
if ((query & Signals) && (!f->isSignal())) {
|
|
continue;
|
|
}
|
|
|
|
if ((query & ForcedShellFunctions)
|
|
&& (!f->isForcedShellImplementation()
|
|
|| !f->isFinal())) {
|
|
continue;
|
|
}
|
|
|
|
if (((query & Constructors) && (!f->isConstructor()
|
|
|| f->ownerClass() != f->implementingClass()))
|
|
|| (f->isConstructor() && (query & Constructors) == 0)) {
|
|
continue;
|
|
}
|
|
|
|
// Destructors are never included in the functions of a class currently
|
|
/*
|
|
if ((query & Destructors) && (!f->isDestructor()
|
|
|| f->ownerClass() != f->implementingClass())
|
|
|| f->isDestructor() && (query & Destructors) == 0) {
|
|
continue;
|
|
}*/
|
|
|
|
if ((query & VirtualFunctions) && (f->isFinal() || f->isSignal() || f->isStatic())) {
|
|
continue;
|
|
}
|
|
|
|
if ((query & StaticFunctions) && (!f->isStatic() || f->isSignal())) {
|
|
continue;
|
|
}
|
|
|
|
if ((query & NonStaticFunctions) && (f->isStatic())) {
|
|
continue;
|
|
}
|
|
|
|
if ((query & NonEmptyFunctions) && (f->isEmptyFunction())) {
|
|
continue;
|
|
}
|
|
|
|
if ((query & NormalFunctions) && (f->isSignal())) {
|
|
continue;
|
|
}
|
|
|
|
if ((query & AbstractFunctions) && !f->isAbstract()) {
|
|
continue;
|
|
}
|
|
|
|
functions << f;
|
|
}
|
|
|
|
// qDebug() << "queried" << m_type_entry->qualifiedCppName() << "got" << functions.size() << "out of" << m_functions.size();
|
|
|
|
return functions;
|
|
}
|
|
|
|
|
|
bool AbstractMetaClass::hasInconsistentFunctions() const
|
|
{
|
|
return cppInconsistentFunctions().size() > 0;
|
|
}
|
|
|
|
bool AbstractMetaClass::hasSignals() const
|
|
{
|
|
return cppSignalFunctions().size() > 0;
|
|
}
|
|
|
|
|
|
/**
|
|
* Adds the specified interface to this class by adding all the
|
|
* functions in the interface to this class.
|
|
*/
|
|
void AbstractMetaClass::addInterface(AbstractMetaClass *interface)
|
|
{
|
|
Q_ASSERT(!m_interfaces.contains(interface));
|
|
m_interfaces << interface;
|
|
|
|
if (m_extracted_interface != 0 && m_extracted_interface != interface)
|
|
m_extracted_interface->addInterface(interface);
|
|
|
|
foreach (AbstractMetaFunction *function, interface->functions())
|
|
if (!hasFunction(function) && !function->isConstructor()) {
|
|
AbstractMetaFunction *cpy = function->copy();
|
|
// We do not want this in PythonQt:
|
|
//cpy->setImplementingClass(this);
|
|
|
|
// Setup that this function is an interface class.
|
|
cpy->setInterfaceClass(interface);
|
|
*cpy += AbstractMetaAttributes::InterfaceFunction;
|
|
|
|
// Copy the modifications in interface into the implementing classes.
|
|
FunctionModificationList mods = function->modifications(interface);
|
|
foreach (const FunctionModification &mod, mods) {
|
|
m_type_entry->addFunctionModification(mod);
|
|
}
|
|
|
|
// It should be mostly safe to assume that when we implement an interface
|
|
// we don't "pass on" pure virtual functions to our sublcasses...
|
|
// *cpy -= AbstractMetaAttributes::Abstract;
|
|
|
|
addFunction(cpy);
|
|
}
|
|
}
|
|
|
|
|
|
void AbstractMetaClass::setInterfaces(const AbstractMetaClassList &interfaces)
|
|
{
|
|
m_interfaces = interfaces;
|
|
}
|
|
|
|
|
|
AbstractMetaEnum *AbstractMetaClass::findEnum(const QString &enumName)
|
|
{
|
|
foreach (AbstractMetaEnum *e, m_enums) {
|
|
if (e->name() == enumName)
|
|
return e;
|
|
}
|
|
|
|
if (typeEntry()->designatedInterface())
|
|
return extractInterface()->findEnum(enumName);
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
|
|
|
|
/*! Recursivly searches for the enum value named \a enumValueName in
|
|
this class and its superclasses and interfaces. Values belonging to
|
|
\a meta_enum are excluded from the search.
|
|
*/
|
|
AbstractMetaEnumValue *AbstractMetaClass::findEnumValue(const QString &enumValueName, AbstractMetaEnum *meta_enum)
|
|
{
|
|
foreach (AbstractMetaEnum *e, m_enums) {
|
|
if (e == meta_enum)
|
|
continue;
|
|
foreach (AbstractMetaEnumValue *v, e->values()) {
|
|
if (v->name() == enumValueName)
|
|
return v;
|
|
}
|
|
}
|
|
|
|
if (typeEntry()->designatedInterface())
|
|
return extractInterface()->findEnumValue(enumValueName, meta_enum);
|
|
|
|
if (baseClass() != 0)
|
|
return baseClass()->findEnumValue(enumValueName, meta_enum);
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/*!
|
|
* Searches through all of this class' enums for a value matching the
|
|
* name \a enumValueName. The name is excluding the class/namespace
|
|
* prefix. The function recursivly searches interfaces and baseclasses
|
|
* of this class.
|
|
*/
|
|
AbstractMetaEnum *AbstractMetaClass::findEnumForValue(const QString &enumValueName)
|
|
{
|
|
foreach (AbstractMetaEnum *e, m_enums) {
|
|
foreach (AbstractMetaEnumValue *v, e->values()) {
|
|
if (v->name() == enumValueName)
|
|
return e;
|
|
}
|
|
}
|
|
|
|
if (typeEntry()->designatedInterface())
|
|
return extractInterface()->findEnumForValue(enumValueName);
|
|
|
|
if (baseClass() != 0)
|
|
return baseClass()->findEnumForValue(enumValueName);
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
static void add_extra_include_for_type(AbstractMetaClass *meta_class, const AbstractMetaType *type)
|
|
{
|
|
|
|
if (type == 0)
|
|
return;
|
|
|
|
Q_ASSERT(meta_class != 0);
|
|
const TypeEntry *entry = (type ? type->typeEntry() : 0);
|
|
if (entry != 0 && entry->isComplex()) {
|
|
const ComplexTypeEntry *centry = static_cast<const ComplexTypeEntry *>(entry);
|
|
ComplexTypeEntry *class_entry = meta_class->typeEntry();
|
|
if (class_entry != 0 && centry->include().isValid())
|
|
class_entry->addExtraInclude(centry->include());
|
|
}
|
|
|
|
if (type->hasInstantiations()) {
|
|
QList<AbstractMetaType *> instantiations = type->instantiations();
|
|
foreach (AbstractMetaType *instantiation, instantiations)
|
|
add_extra_include_for_type(meta_class, instantiation);
|
|
}
|
|
}
|
|
|
|
static void add_extra_includes_for_function(AbstractMetaClass *meta_class, const AbstractMetaFunction *meta_function)
|
|
{
|
|
Q_ASSERT(meta_class != 0);
|
|
Q_ASSERT(meta_function != 0);
|
|
add_extra_include_for_type(meta_class, meta_function->type());
|
|
|
|
AbstractMetaArgumentList arguments = meta_function->arguments();
|
|
foreach (AbstractMetaArgument *argument, arguments)
|
|
add_extra_include_for_type(meta_class, argument->type());
|
|
}
|
|
|
|
void AbstractMetaClass::fixFunctions()
|
|
{
|
|
if (m_functions_fixed)
|
|
return;
|
|
else
|
|
m_functions_fixed = true;
|
|
|
|
AbstractMetaClass *super_class = baseClass();
|
|
AbstractMetaFunctionList funcs = functions();
|
|
|
|
// printf("fix functions for %s\n", qPrintable(name()));
|
|
|
|
if (super_class != 0)
|
|
super_class->fixFunctions();
|
|
int iface_idx = 0;
|
|
while (super_class || iface_idx < interfaces().size()) {
|
|
// printf(" - base: %s\n", qPrintable(super_class->name()));
|
|
|
|
// Since we always traverse the complete hierarchy we are only
|
|
// interrested in what each super class implements, not what
|
|
// we may have propagated from their base classes again.
|
|
AbstractMetaFunctionList super_funcs;
|
|
if (super_class) {
|
|
|
|
// Super classes can never be final
|
|
if (super_class->isFinalInTargetLang()) {
|
|
ReportHandler::warning("Final class '" + super_class->name() + "' set to non-final, as it is extended by other classes");
|
|
*super_class -= AbstractMetaAttributes::FinalInTargetLang;
|
|
}
|
|
super_funcs = super_class->queryFunctions(AbstractMetaClass::ClassImplements);
|
|
} else {
|
|
super_funcs = interfaces().at(iface_idx)->queryFunctions(AbstractMetaClass::NormalFunctions);
|
|
}
|
|
|
|
QSet<AbstractMetaFunction *> funcs_to_add;
|
|
for (int sfi=0; sfi<super_funcs.size(); ++sfi) {
|
|
AbstractMetaFunction *sf = super_funcs.at(sfi);
|
|
|
|
if (sf->isRemovedFromAllLanguages(sf->implementingClass()))
|
|
continue;
|
|
|
|
// we generally don't care about private functions, but we have to get the ones that are
|
|
// virtual in case they override abstract functions.
|
|
bool add = (sf->isNormal() || sf->isSignal() || sf->isEmptyFunction());
|
|
for (int fi=0; fi<funcs.size(); ++fi) {
|
|
AbstractMetaFunction *f = funcs.at(fi);
|
|
if (f->isRemovedFromAllLanguages(f->implementingClass()))
|
|
continue;
|
|
|
|
uint cmp = f->compareTo(sf);
|
|
|
|
if (cmp & AbstractMetaFunction::EqualModifiedName) {
|
|
// printf(" - %s::%s similar to %s::%s %x vs %x\n",
|
|
// qPrintable(sf->implementingClass()->typeEntry()->qualifiedCppName()),
|
|
// qPrintable(sf->name()),
|
|
// qPrintable(f->implementingClass()->typeEntry()->qualifiedCppName()),
|
|
// qPrintable(f->name()),
|
|
// sf->attributes(),
|
|
// f->attributes());
|
|
|
|
add = false;
|
|
if (cmp & AbstractMetaFunction::EqualArguments) {
|
|
|
|
// if (!(cmp & AbstractMetaFunction::EqualReturnType)) {
|
|
// ReportHandler::warning(QString("%1::%2 and %3::%4 differ in retur type")
|
|
// .arg(sf->implementingClass()->name())
|
|
// .arg(sf->name())
|
|
// .arg(f->implementingClass()->name())
|
|
// .arg(f->name()));
|
|
// }
|
|
|
|
// Same function, propegate virtual...
|
|
if (!(cmp & AbstractMetaFunction::EqualAttributes)) {
|
|
if (!f->isEmptyFunction()) {
|
|
if (!sf->isFinalInCpp() && f->isFinalInCpp()) {
|
|
*f -= AbstractMetaAttributes::FinalInCpp;
|
|
// printf(" --- inherit virtual\n");
|
|
}
|
|
if (!sf->isFinalInTargetLang() && f->isFinalInTargetLang()) {
|
|
*f -= AbstractMetaAttributes::FinalInTargetLang;
|
|
// printf(" --- inherit virtual\n");
|
|
}
|
|
if (!f->isFinalInTargetLang() && f->isPrivate()) {
|
|
f->setFunctionType(AbstractMetaFunction::EmptyFunction);
|
|
f->setVisibility(AbstractMetaAttributes::Protected);
|
|
*f += AbstractMetaAttributes::FinalInTargetLang;
|
|
ReportHandler::warning(QString("private virtual function '%1' in '%2'")
|
|
.arg(f->signature())
|
|
.arg(f->implementingClass()->name()));
|
|
}
|
|
}
|
|
}
|
|
|
|
if (f->visibility() != sf->visibility()) {
|
|
QString warn = QString("visibility of function '%1' modified in class '%2'")
|
|
.arg(f->name()).arg(name());
|
|
ReportHandler::warning(warn);
|
|
|
|
// If new visibility is private, we can't
|
|
// do anything. If it isn't, then we
|
|
// prefer the parent class's visibility
|
|
// setting for the function.
|
|
if (!f->isPrivate() && !sf->isPrivate())
|
|
f->setVisibility(sf->visibility());
|
|
|
|
// Private overrides of abstract functions have to go into the class or
|
|
// the subclasses will not compile as non-abstract classes.
|
|
// But they don't need to be implemented, since they can never be called.
|
|
if (f->isPrivate() && sf->isAbstract()) {
|
|
f->setFunctionType(AbstractMetaFunction::EmptyFunction);
|
|
f->setVisibility(sf->visibility());
|
|
*f += AbstractMetaAttributes::FinalInTargetLang;
|
|
*f += AbstractMetaAttributes::FinalInCpp;
|
|
}
|
|
}
|
|
|
|
// Set the class which first declares this function, afawk
|
|
f->setDeclaringClass(sf->declaringClass());
|
|
|
|
if (sf->isFinalInTargetLang() && !sf->isPrivate() && !f->isPrivate() && !sf->isStatic() && !f->isStatic()) {
|
|
// Shadowed funcion, need to make base class
|
|
// function non-virtual
|
|
if (f->implementingClass() != sf->implementingClass() && f->implementingClass()->inheritsFrom(sf->implementingClass())) {
|
|
|
|
// Check whether the superclass method has been redefined to non-final
|
|
|
|
bool hasNonFinalModifier = false;
|
|
bool isBaseImplPrivate = false;
|
|
FunctionModificationList mods = sf->modifications(sf->implementingClass());
|
|
foreach (FunctionModification mod, mods) {
|
|
if (mod.isNonFinal()) {
|
|
hasNonFinalModifier = true;
|
|
break;
|
|
} else if (mod.isPrivate()) {
|
|
isBaseImplPrivate = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!hasNonFinalModifier && !isBaseImplPrivate) {
|
|
ReportHandler::warning(QString::fromLatin1("Shadowing: %1::%2 and %3::%4; Java code will not compile")
|
|
.arg(sf->implementingClass()->name())
|
|
.arg(sf->signature())
|
|
.arg(f->implementingClass()->name())
|
|
.arg(f->signature()));
|
|
}
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
if (cmp & AbstractMetaFunction::EqualDefaultValueOverload) {
|
|
AbstractMetaArgumentList arguments;
|
|
if (f->arguments().size() < sf->arguments().size())
|
|
arguments = sf->arguments();
|
|
else
|
|
arguments = f->arguments();
|
|
|
|
for (int i=0; i<arguments.size(); ++i)
|
|
arguments[i]->setDefaultValueExpression(QString());
|
|
}
|
|
|
|
|
|
// Otherwise we have function shadowing and we can
|
|
// skip the thing...
|
|
} else if (cmp & AbstractMetaFunction::EqualName && !sf->isSignal()) {
|
|
|
|
// In the case of function shadowing where the function name has been altered to
|
|
// avoid conflict, we don't copy in the original.
|
|
add = false;
|
|
}
|
|
|
|
}
|
|
|
|
if (add)
|
|
funcs_to_add << sf;
|
|
}
|
|
|
|
foreach (AbstractMetaFunction *f, funcs_to_add)
|
|
funcs << f->copy();
|
|
|
|
if (super_class)
|
|
super_class = super_class->baseClass();
|
|
else
|
|
iface_idx++;
|
|
}
|
|
|
|
bool hasPrivateConstructors = false;
|
|
bool hasPublicConstructors = false;
|
|
foreach (AbstractMetaFunction *func, funcs) {
|
|
FunctionModificationList mods = func->modifications(this);
|
|
foreach (const FunctionModification &mod, mods) {
|
|
if (mod.isRenameModifier()) {
|
|
// qDebug() << name() << func->originalName() << func << " from "
|
|
// << func->implementingClass()->name() << "renamed to" << mod.renamedTo();
|
|
func->setName(mod.renamedTo());
|
|
}
|
|
}
|
|
|
|
// Make sure class is abstract if one of the functions is
|
|
if (func->isAbstract()) {
|
|
(*this) += AbstractMetaAttributes::Abstract;
|
|
(*this) -= AbstractMetaAttributes::Final;
|
|
}
|
|
|
|
if (func->isConstructor()) {
|
|
if (func->isPrivate())
|
|
hasPrivateConstructors = true;
|
|
else
|
|
hasPublicConstructors = true;
|
|
}
|
|
|
|
|
|
|
|
// Make sure that we include files for all classes that are in use
|
|
|
|
if (!func->isRemovedFrom(this, TypeSystem::ShellCode))
|
|
add_extra_includes_for_function(this, func);
|
|
}
|
|
|
|
if (hasPrivateConstructors && !hasPublicConstructors) {
|
|
(*this) += AbstractMetaAttributes::Abstract;
|
|
(*this) -= AbstractMetaAttributes::Final;
|
|
}
|
|
|
|
foreach (AbstractMetaFunction *f1, funcs) {
|
|
foreach (AbstractMetaFunction *f2, funcs) {
|
|
if (f1 != f2) {
|
|
uint cmp = f1->compareTo(f2);
|
|
if ((cmp & AbstractMetaFunction::EqualName)
|
|
&& !f1->isFinalInCpp()
|
|
&& f2->isFinalInCpp()) {
|
|
*f2 += AbstractMetaAttributes::FinalOverload;
|
|
// qDebug() << f2 << f2->implementingClass()->name() << "::" << f2->name() << f2->arguments().size() << " vs " << f1 << f1->implementingClass()->name() << "::" << f1->name() << f1->arguments().size();
|
|
// qDebug() << " " << f2;
|
|
// AbstractMetaArgumentList f2Args = f2->arguments();
|
|
// foreach (AbstractMetaArgument *a, f2Args)
|
|
// qDebug() << " " << a->type()->name() << a->name();
|
|
// qDebug() << " " << f1;
|
|
// AbstractMetaArgumentList f1Args = f1->arguments();
|
|
// foreach (AbstractMetaArgument *a, f1Args)
|
|
// qDebug() << " " << a->type()->name() << a->name();
|
|
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
setFunctions(funcs);
|
|
}
|
|
|
|
|
|
QString AbstractMetaType::minimalSignature() const
|
|
{
|
|
QString minimalSignature;
|
|
if (isConstant())
|
|
minimalSignature += "const ";
|
|
minimalSignature += typeEntry()->qualifiedCppName();
|
|
if (hasInstantiations()) {
|
|
QList<AbstractMetaType *> instantiations = this->instantiations();
|
|
minimalSignature += "<";
|
|
for (int i=0;i<instantiations.size();++i) {
|
|
if (i > 0)
|
|
minimalSignature += ",";
|
|
minimalSignature += instantiations.at(i)->minimalSignature();
|
|
}
|
|
minimalSignature += ">";
|
|
}
|
|
|
|
if (isReference())
|
|
minimalSignature += "&";
|
|
for (int j=0; j<indirections(); ++j)
|
|
minimalSignature += "*";
|
|
|
|
return minimalSignature;
|
|
}
|
|
|
|
bool AbstractMetaType::hasNativeId() const
|
|
{
|
|
return (isQObject() || isValue() || isObject()) && typeEntry()->isNativeIdBased();
|
|
}
|
|
|
|
|
|
/*******************************************************************************
|
|
* Other stuff...
|
|
*/
|
|
|
|
|
|
AbstractMetaEnum *AbstractMetaClassList::findEnum(const EnumTypeEntry *entry) const
|
|
{
|
|
Q_ASSERT(entry->isEnum());
|
|
|
|
QString qualified_name = entry->qualifiedCppName();
|
|
int pos = qualified_name.lastIndexOf("::");
|
|
|
|
QString enum_name;
|
|
QString class_name;
|
|
|
|
if (pos > 0) {
|
|
enum_name = qualified_name.mid(pos + 2);
|
|
class_name = qualified_name.mid(0, pos);
|
|
} else {
|
|
enum_name = qualified_name;
|
|
class_name = TypeDatabase::globalNamespaceClassName(entry);
|
|
}
|
|
|
|
AbstractMetaClass *meta_class = findClass(class_name);
|
|
if (!meta_class) {
|
|
ReportHandler::warning(QString("AbstractMeta::findEnum(), unknown class '%1' in '%2'")
|
|
.arg(class_name).arg(entry->qualifiedCppName()));
|
|
return 0;
|
|
}
|
|
|
|
return meta_class->findEnum(enum_name);
|
|
}
|
|
|
|
AbstractMetaEnumValue *AbstractMetaEnumValueList::find(const QString &name) const
|
|
{
|
|
for (int i=0; i<size(); ++i) {
|
|
if (name == at(i)->name())
|
|
return at(i);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
AbstractMetaEnumValue *AbstractMetaClassList::findEnumValue(const QString &name) const
|
|
{
|
|
QStringList lst = name.split(QLatin1String("::"));
|
|
|
|
Q_ASSERT_X(lst.size() == 2, "AbstractMetaClassList::findEnumValue()", "Expected qualified enum");
|
|
|
|
|
|
QString prefixName = lst.at(0);
|
|
QString enumName = lst.at(1);
|
|
|
|
AbstractMetaClass *cl = findClass(prefixName);
|
|
if (cl)
|
|
return cl->findEnumValue(enumName, 0);
|
|
|
|
ReportHandler::warning(QString("no matching enum '%1'").arg(name));
|
|
return 0;
|
|
}
|
|
|
|
/*!
|
|
* Searches the list after a class that mathces \a name; either as
|
|
* C++, Java base name or complete Java package.class name.
|
|
*/
|
|
|
|
AbstractMetaClass *AbstractMetaClassList::findClass(const QString &name) const
|
|
{
|
|
if (name.isEmpty())
|
|
return 0;
|
|
|
|
foreach (AbstractMetaClass *c, *this) {
|
|
if (c->qualifiedCppName() == name)
|
|
return c;
|
|
}
|
|
|
|
foreach (AbstractMetaClass *c, *this) {
|
|
if (c->fullName() == name)
|
|
return c;
|
|
}
|
|
|
|
foreach (AbstractMetaClass *c, *this) {
|
|
if (c->name() == name)
|
|
return c;
|
|
}
|
|
|
|
return 0;
|
|
}
|