kde-extraapps/kdevelop/languages/cpp/parser/commentformatter.cpp

156 lines
5.5 KiB
C++
Raw Normal View History

/*
Copyright (C) 2007 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 "commentformatter.h"
#include "ast.h"
#include <QStringList>
#include "parsesession.h"
#include "lexer.h"
#include "rpp/chartools.h"
#include <language/duchain/stringhelpers.h>
#include <language/editor/simplerange.h>
#include <interfaces/icore.h>
#include <interfaces/ilanguagecontroller.h>
#include <interfaces/icompletionsettings.h>
#include "control.h"
CommentFormatter::CommentFormatter()
{
if(!KDevelop::ICore::self())
return; // May happen in tests i guess
foreach(QString marker, KDevelop::ICore::self()->languageController()->completionSettings()->todoMarkerWords())
{
m_commentMarkers << marker.toUtf8();
m_commentMarkerIndices << KDevelop::IndexedString(marker).index();
}
}
bool CommentFormatter::containsToDo(const uint* start, const uint* end) const
{
const uint* markersStart = m_commentMarkerIndices.data();
const uint* markersEnd = m_commentMarkerIndices.data() + m_commentMarkerIndices.size();
for(const uint* cursor = start; cursor < end; ++cursor)
for(const uint* marker = markersStart; marker < markersEnd; ++marker)
if(*cursor == *marker)
return true;
return false;
}
bool CommentFormatter::containsToDo(const QByteArray& text) const
{
foreach(const QByteArray& marker, m_commentMarkers)
if(text.contains(marker))
return true;
return false;
}
void CommentFormatter::extractToDos( uint token, const ParseSession* session, Control* control ) {
if( !token )
return;
const Token& commentToken( (*session->token_stream)[token] );
if( !containsToDo(session->contents() + commentToken.position, session->contents() + commentToken.position + commentToken.size) )
return; // Most common code path: No todos
QByteArray comment = stringFromContents(session->contentsVector(), commentToken.position, commentToken.size);
QList<QByteArray> lines = comment.split( '\n' );
if ( !lines.isEmpty() ) {
QList<QByteArray>::iterator bit = lines.begin();
QList<QByteArray>::iterator it = bit;
QList<QByteArray>::iterator eit = lines.end();
KDevelop::IndexedString document = session->url();
KDevelop::SimpleCursor comment_start = session->positionAt(commentToken.position).castToSimpleCursor();
for( ; it != eit; ++it ) {
// remove common leading chars from the beginning of line
int stripped_left = 0;
int stripped_right = 0;
stripped_left += KDevelop::strip( "///", *it );
stripped_left += KDevelop::strip( "//", *it );
stripped_left += KDevelop::strip( "**", *it );
stripped_right += KDevelop::rStrip( "/**", *it );
int left_spaces;
for (left_spaces = 0; left_spaces < it->size(); ++left_spaces) {
if (!isSpace(it->at(left_spaces))) {
break;
}
}
int right_spaces;
for (right_spaces = it->size() - 1; right_spaces >= 0; --right_spaces) {
if (!isSpace(it->at(right_spaces))) {
break;
}
}
right_spaces = it->size() - 1 - right_spaces;
stripped_left += left_spaces;
stripped_right += right_spaces;
*it = it->mid(left_spaces, it->size() - left_spaces - right_spaces);
if( containsToDo(*it) ) {
KDevelop::ProblemPointer p(new KDevelop::Problem());
p->setSource(KDevelop::ProblemData::ToDo);
p->setDescription(QString::fromUtf8(*it));
p->setSeverity(KDevelop::ProblemData::Hint);
int start_line = comment_start.line + (it - bit);
int start_column = (it == bit) ? comment_start.column + stripped_left : stripped_left;
p->setFinalLocation(KDevelop::DocumentRange(session->url(), KDevelop::SimpleRange(start_line, start_column, start_line, it->size() + start_column)));
control->reportProblem(p);
}
}
}
}
QByteArray CommentFormatter::formatComment( uint token, const ParseSession* session ) {
if( !token )
return QByteArray();
///@todo Work directly on lists of IndexedString tokens, rather than QBytearray (faster), and only convert to QByteArray in the end.
const Token& commentToken( (*session->token_stream)[token] );
return KDevelop::formatComment( stringFromContents(session->contentsVector(), commentToken.position, commentToken.size ) );
}
QByteArray CommentFormatter::formatComment( const ListNode<uint>* comments, const ParseSession* session ) {
QByteArray ret;
///@todo Work directly on lists of IndexedString tokens, rather than QBytearray (faster), and only convert to QByteArray in the end.
if( comments )
{
const ListNode<uint> *it = comments->toFront(), *end = it;
do {
QByteArray c = CommentFormatter::formatComment(it->element, session);
if( ret.isEmpty() )
ret = c;
else
ret += "\n(" + c + ")";
it = it->next;
}while( it != end );
}
return ret;
}