kde-extraapps/kdevplatform/language/duchain/classfunctiondeclaration.cpp
2015-07-26 14:23:17 +03:00

221 lines
6.9 KiB
C++

/* This is part of KDevelop
Copyright 2002-2005 Roberto Raggi <roberto@kdevelop.org>
Copyright 2006 Adam Treat <treat@kde.org>
Copyright 2006 Hamish Rodda <rodda@kde.org>
Copyright 2007-2008 David Nolden <david.nolden.kdevelop@art-master.de>
This library is free software; you can redistribute it and/or
modify it under the terms of the GNU Library General Public
License version 2 as published by the Free Software Foundation.
This library is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
Library General Public License for more details.
You should have received a copy of the GNU Library General Public License
along with this library; see the file COPYING.LIB. If not, write to
the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
Boston, MA 02110-1301, USA.
*/
#include "classfunctiondeclaration.h"
#include "ducontext.h"
#include "types/functiontype.h"
#include "duchainregister.h"
namespace KDevelop
{
static Identifier& conversionIdentifier() {
static Identifier conversionIdentifierObject("operator{...cast...}");
return conversionIdentifierObject;
}
REGISTER_DUCHAIN_ITEM(ClassFunctionDeclaration);
ClassFunctionDeclaration::ClassFunctionDeclaration(const ClassFunctionDeclaration& rhs)
: ClassFunctionDeclarationBase(*new ClassFunctionDeclarationData( *rhs.d_func() )) {
}
void ClassFunctionDeclaration::setAbstractType(AbstractType::Ptr type) {
///TODO: write testcase for typealias case which used to trigger this warning:
/// typedef bool (*EventFilter)(void *message, long *result);
/// in e.g. qcoreapplication.h:172
if(type && !dynamic_cast<FunctionType*>(type.unsafeData()) && type->whichType() != AbstractType::TypeAlias) {
kWarning(9505) << "WARNING: Non-function type assigned to function declaration. Type is: "
<< type->toString() << "whichType:" << type->whichType()
<< "Declaration is:" << toString()
<< topContext()->url().str() << range().castToSimpleRange();
}
ClassMemberDeclaration::setAbstractType(type);
}
DEFINE_LIST_MEMBER_HASH(ClassFunctionDeclarationData, m_defaultParameters, IndexedString)
ClassFunctionDeclaration::ClassFunctionDeclaration(ClassFunctionDeclarationData& data) : ClassFunctionDeclarationBase(data)
{
}
ClassFunctionDeclaration::ClassFunctionDeclaration(const RangeInRevision& range, DUContext* context)
: ClassFunctionDeclarationBase(*new ClassFunctionDeclarationData, range)
{
d_func_dynamic()->setClassId(this);
if( context )
setContext( context );
}
ClassFunctionDeclaration::ClassFunctionDeclaration(ClassFunctionDeclarationData& data, const RangeInRevision& range, DUContext* context)
: ClassFunctionDeclarationBase(data, range)
{
if( context )
setContext( context );
}
Declaration* ClassFunctionDeclaration::clonePrivate() const {
return new ClassFunctionDeclaration(*this);
}
ClassFunctionDeclaration::~ClassFunctionDeclaration()
{
}
bool ClassFunctionDeclaration::isFunctionDeclaration() const
{
return true;
}
QString ClassFunctionDeclaration::toString() const {
if( !abstractType() )
return ClassMemberDeclaration::toString();
TypePtr<FunctionType> function = type<FunctionType>();
if(function) {
return QString("%1 %2 %3").arg(function->partToString( FunctionType::SignatureReturn )).arg(identifier().toString()).arg(function->partToString( FunctionType::SignatureArguments ));
} else {
QString type = abstractType() ? abstractType()->toString() : QString("<notype>");
kDebug(9505) << "A function has a bad type attached:" << type;
return QString("invalid member-function %1 type %2").arg(identifier().toString()).arg(type);
}
}
/*bool ClassFunctionDeclaration::isSimilar(KDevelop::CodeItem *other, bool strict ) const
{
if (!CppClassMemberType::isSimilar(other,strict))
return false;
FunctionModelItem func = dynamic_cast<ClassFunctionDeclaration*>(other);
if (isConstant() != func->isConstant())
return false;
if (arguments().count() != func->arguments().count())
return false;
for (int i=0; i<arguments().count(); ++i)
{
ArgumentModelItem arg1 = arguments().at(i);
ArgumentModelItem arg2 = arguments().at(i);
if (arg1->type() != arg2->type())
return false;
}
return true;
}*/
uint setFlag(bool enable, uint flag, uint flags) {
if(enable)
return flags | flag;
else
return flags & (~flag);
}
bool ClassFunctionDeclaration::isAbstract() const
{
return d_func()->m_functionFlags & AbstractFunctionFlag;
}
void ClassFunctionDeclaration::setIsAbstract(bool abstract)
{
d_func_dynamic()->m_functionFlags = (ClassFunctionFlags)setFlag(abstract, AbstractFunctionFlag, d_func()->m_functionFlags);
}
bool ClassFunctionDeclaration::isFinal() const
{
return d_func()->m_functionFlags & FinalFunctionFlag;
}
void ClassFunctionDeclaration::setIsFinal(bool final)
{
d_func_dynamic()->m_functionFlags = (ClassFunctionFlags)setFlag(final, FinalFunctionFlag, d_func()->m_functionFlags);
}
bool ClassFunctionDeclaration::isSignal() const
{
return d_func()->m_functionFlags & FunctionSignalFlag;
}
void ClassFunctionDeclaration::setIsSignal(bool isSignal) {
d_func_dynamic()->m_functionFlags = (ClassFunctionFlags)setFlag(isSignal, FunctionSignalFlag, d_func()->m_functionFlags);
}
bool ClassFunctionDeclaration::isSlot() const
{
return d_func()->m_functionFlags & FunctionSlotFlag;
}
void ClassFunctionDeclaration::setIsSlot(bool isSlot) {
d_func_dynamic()->m_functionFlags = (ClassFunctionFlags)setFlag(isSlot, FunctionSlotFlag, d_func()->m_functionFlags);
}
bool ClassFunctionDeclaration::isConversionFunction() const {
return identifier() == conversionIdentifier();
}
bool ClassFunctionDeclaration::isConstructor() const
{
DUContext* ctx = context();
if (ctx && ctx->type() == DUContext::Class && ctx->localScopeIdentifier().top().nameEquals(identifier()))
return true;
return false;
}
bool ClassFunctionDeclaration::isDestructor() const
{
DUContext* ctx = context();
QString id = identifier().toString();
return ctx && ctx->type() == DUContext::Class && id.startsWith('~') && id.mid(1) == ctx->localScopeIdentifier().top().toString();
}
uint ClassFunctionDeclaration::additionalIdentity() const
{
if(abstractType())
return abstractType()->hash();
else
return 0;
}
const IndexedString* ClassFunctionDeclaration::defaultParameters() const
{
return d_func()->m_defaultParameters();
}
unsigned int ClassFunctionDeclaration::defaultParametersSize() const
{
return d_func()->m_defaultParametersSize();
}
void ClassFunctionDeclaration::addDefaultParameter(const IndexedString& str)
{
d_func_dynamic()->m_defaultParametersList().append(str);
}
void ClassFunctionDeclaration::clearDefaultParameters()
{
d_func_dynamic()->m_defaultParametersList().clear();
}
}
// kate: space-indent on; indent-width 2; tab-width 4; replace-tabs on; auto-insert-doxygen on