mirror of
https://bitbucket.org/smil3y/kde-extraapps.git
synced 2025-02-24 19:02:53 +00:00
221 lines
6.9 KiB
C++
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
|