1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137
|
/*
SPDX-FileCopyrightText: 2007 Andreas Pakulat <apaku@gmx.de>
SPDX-License-Identifier: LGPL-2.0-or-later
*/
#include "variablereferenceparser.h"
bool isVarNameChar(QChar* c)
{
bool ret = c->isLetterOrNumber() || c->unicode() == '_';
return ret;
}
VariableInfo::VariableInfo()
{
}
VariableReferenceParser::VariableReferenceParser()
{
}
void VariableReferenceParser::setContent(const QString& content)
{
m_content = content;
}
bool VariableReferenceParser::parse()
{
int size = m_content.size();
int curpos = 0;
if (size == 0 || size < 3) {
return true;
}
QChar* it = m_content.data();
do {
if (it->unicode() == '$' && size > curpos + 2) {
it++;
curpos++;
if (it->unicode() == '$') {
int begin = curpos - 1;
it++;
curpos++;
QString variable;
VariableInfo::VariableType type = VariableInfo::QMakeVariable;
if (it->unicode() == '(') {
do {
it++;
curpos++;
} while (curpos < size && it->unicode() != ')');
type = VariableInfo::ShellVariableResolveQMake;
variable = m_content.mid(begin + 3, curpos - begin - 3);
++curpos;
} else if (it->unicode() == '{') {
do {
it++;
curpos++;
if (it->unicode() == '(') {
type = VariableInfo::FunctionCall;
}
} while (curpos < size && it->unicode() != '}');
variable = m_content.mid(begin + 3, curpos - begin - 3);
++curpos;
} else if (it->unicode() == '[') {
do {
it++;
curpos++;
} while (curpos < size && it->unicode() != ']');
type = VariableInfo::QtConfigVariable;
variable = m_content.mid(begin + 3, curpos - begin - 3);
++curpos;
} else {
do {
it++;
curpos++;
} while (curpos < size && isVarNameChar(it));
variable = m_content.mid(begin + 2, curpos - begin - 2);
if (it->unicode() == '(') {
type = VariableInfo::FunctionCall;
int braceCount = 0;
do {
it++;
curpos++;
if (it->unicode() == ')') {
braceCount--;
} else if (it->unicode() == '(') {
braceCount++;
}
} while (curpos < size && (it->unicode() != ')' || braceCount == 0));
// count the current position one further if we have it
// at the closing brace, this is needed for proper end-calculation
if (curpos < size && it->unicode() == ')') {
it++;
curpos++;
}
}
}
int end = curpos - 1;
appendPosition(variable, begin, end, type);
} else if (it->unicode() == '(') {
int begin = curpos - 1;
do {
it++;
curpos++;
} while (curpos < size && it->unicode() != ')');
int end = curpos - 1;
appendPosition(m_content.mid(begin + 2, end - (begin + 2)), begin, end,
VariableInfo::ShellVariableResolveMake);
}
} else {
curpos++;
}
} while (curpos < size);
return true;
}
QStringList VariableReferenceParser::variableReferences() const
{
return m_variables.keys();
}
VariableInfo VariableReferenceParser::variableInfo(const QString& var) const
{
return m_variables.value(var, VariableInfo());
}
void VariableReferenceParser::appendPosition(const QString& var, int start, int end, VariableInfo::VariableType type)
{
auto variableIt = m_variables.find(var);
if (variableIt == m_variables.end()) {
VariableInfo vi;
vi.type = type;
variableIt = m_variables.insert(var, vi);
}
variableIt->positions << VariableInfo::Position(start, end);
}
|