Files
qt-creator/src/plugins/todo/lineparser.cpp
Dmitry Savchenko 9fe784115a Allow not only a colon as a keyword-text divider in todos.
This makes possible to scan such things as doxygen \todo, @todo
entries where colons cannot be added after a keyword.

Change-Id: I13c757294c94a1f4758e7dc9634a79ea6e91c68f
Reviewed-by: Eike Ziller <eike.ziller@nokia.com>
2012-05-24 20:38:23 +02:00

188 lines
5.3 KiB
C++

/**************************************************************************
**
** This file is part of Qt Creator
**
** Copyright (c) 2012 Dmitry Savchenko.
** Copyright (c) 2010 Vasiliy Sorokin.
**
** Contact: Nokia Corporation (qt-info@nokia.com)
**
**
** GNU Lesser General Public License Usage
**
** 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.
**
** Other Usage
**
** Alternatively, this file may be used in accordance with the terms and
** conditions contained in a signed written agreement between you and Nokia.
**
** If you have questions regarding the use of this file, please contact
** Nokia at qt-info@nokia.com.
**
**************************************************************************/
#include "lineparser.h"
#include <QMap>
namespace Todo {
namespace Internal {
LineParser::LineParser(const KeywordList &keywordList)
{
setKeywordList(keywordList);
}
void LineParser::setKeywordList(const KeywordList &keywordList)
{
m_keywords = keywordList;
}
QList<TodoItem> LineParser::parse(const QString &line)
{
QMap<int, int> entryCandidates = findKeywordEntryCandidates(line);
QList<KeywordEntry> entries = keywordEntriesFromCandidates(entryCandidates, line);
return todoItemsFromKeywordEntries(entries);
}
bool LineParser::isKeywordSeparator(const QChar &ch)
{
return ch.isSpace()
|| (ch == QLatin1Char(':'))
|| (ch == QLatin1Char('/'))
|| (ch == QLatin1Char('*'));
}
LineParser::KeywordEntryCandidates LineParser::findKeywordEntryCandidates(const QString &line)
{
KeywordEntryCandidates entryCandidates;
for (int i = 0; i < m_keywords.count(); ++i) {
int searchFrom = -1;
forever {
const int index = line.lastIndexOf(m_keywords.at(i).name, searchFrom);
if (index == -1)
break; // 'forever' loop exit condition
searchFrom = index - line.length() - 1;
if (isKeywordAt(index, line, m_keywords.at(i).name))
entryCandidates.insert(index, i);
}
}
return entryCandidates;
}
bool LineParser::isKeywordAt(int index, const QString &line, const QString &keyword)
{
if (!isFirstCharOfTheWord(index, line))
return false;
if (!isLastCharOfTheWord(index + keyword.length() - 1, line))
return false;
return true;
}
bool LineParser::isFirstCharOfTheWord(int index, const QString &line)
{
return (index == 0) || isKeywordSeparator(line.at(index - 1));
}
bool LineParser::isLastCharOfTheWord(int index, const QString &line)
{
return (index == line.length() - 1) || isKeywordSeparator(line.at(index + 1));
}
QList<LineParser::KeywordEntry> LineParser::keywordEntriesFromCandidates(
const QMap<int, int> &candidates, const QString &line)
{
// Ensure something is found
if (candidates.isEmpty())
return QList<KeywordEntry>();
// Convert candidates to entries
QList<KeywordEntry> entries;
QMapIterator<int, int> i(candidates);
i.toBack();
while (i.hasPrevious()) {
i.previous();
KeywordEntry entry;
entry.keywordStart = i.key();
entry.keywordIndex = i.value();
int keywordLength = m_keywords.at(entry.keywordIndex).name.length();
int entryTextLength = -1;
if (!entries.empty())
entryTextLength = entries.last().keywordStart - (entry.keywordStart + keywordLength);
entry.text = line.mid(entry.keywordStart + keywordLength, entryTextLength);
if (trimSeparators(entry.text).isEmpty() && !entries.empty())
// Take the text form the previous entry, consider:
// '<keyword1>: <keyword2>: <some text>'
entry.text = entries.last().text;
entries << entry;
}
return entries;
}
QString LineParser::trimSeparators(const QString &string)
{
QString result = string.trimmed();
while (startsWithSeparator(result))
result = result.right(result.length() - 1);
while (endsWithSeparator(result))
result = result.left(result.length() - 1);
return result;
}
bool LineParser::startsWithSeparator(const QString &string)
{
return !string.isEmpty() && isKeywordSeparator(string.at(0));
}
bool LineParser::endsWithSeparator(const QString &string)
{
return !string.isEmpty() && isKeywordSeparator(string.at(string.length() - 1));
}
QList<TodoItem> LineParser::todoItemsFromKeywordEntries(const QList<KeywordEntry> &entries)
{
QList<TodoItem> todoItems;
foreach (const KeywordEntry &entry, entries) {
TodoItem item;
item.text = m_keywords.at(entry.keywordIndex).name + entry.text;
item.color = m_keywords.at(entry.keywordIndex).color;
item.iconResource = m_keywords.at(entry.keywordIndex).iconResource;
todoItems << item;
}
return todoItems;
}
} // namespace Internal
} // namespace Todo