24 #include "exception.h" 26 #include "cutelee_version.h" 27 #include "nodebuiltins_p.h" 28 #include "taglibraryinterface.h" 30 #include "template_p.h" 41 : q_ptr(parser), m_tokenList(tokenList)
70 auto ti = qobject_cast<TemplateImpl *>(q->parent());
72 auto cengine = ti->engine();
74 auto engine =
const_cast<Engine *
>(cengine);
77 for (
auto nodeIt = factories.begin(), nodeEnd = factories.end();
78 nodeIt != nodeEnd; ++nodeIt) {
79 nodeIt.value()->setEngine(engine);
80 m_nodeFactories.insert(nodeIt.key(), nodeIt.value());
82 auto filters = library->
filters();
83 for (
auto filterIt = filters.begin(), filterEnd = filters.end();
84 filterIt != filterEnd; ++filterIt) {
85 auto f = std::shared_ptr<Filter>(filterIt.value());
86 m_filters.insert(filterIt.key(), f);
97 auto cengine = ti->engine();
100 auto engine =
const_cast<Engine *
>(cengine);
101 engine->loadDefaultLibraries();
103 for (
const QString &libraryName : libs) {
104 auto library = engine->loadLibrary(libraryName);
107 d->openLibrary(library);
115 qDeleteAll(d_ptr->m_nodeFactories);
119 void Parser::loadLib(
const QString &name)
123 auto cengine = ti->engine();
125 auto engine =
const_cast<Engine *
>(cengine);
126 auto library = engine->loadLibrary(name);
129 d->openLibrary(library);
137 QStringLiteral(
"Node appeared twice in template: %1")
149 if (token.tokenType ==
BlockToken && token.content == tag)
153 UnclosedBlockTagError,
154 QStringLiteral(
"No closing tag found for %1").arg(tag));
160 const auto it = d->m_filters.constFind(name);
161 if (Q_LIKELY(it != d->m_filters.constEnd())) {
165 QStringLiteral(
"Unknown filter: %1").arg(name));
171 return d->parse(
parent, {stopAt});
177 return d->parse(
parent, stopAt);
183 return d->parse(
parent, stopAt);
191 while (q->hasNextToken()) {
192 const auto token = q->takeNextToken();
194 nodeList = extendNodeList(nodeList,
new TextNode(token.content, parent));
196 if (token.content.isEmpty()) {
199 Q_ASSERT(q->hasNextToken());
200 message = QStringLiteral(
"Empty variable before \"%1\", line %2, %3")
201 .
arg(q->takeNextToken().content.left(20))
202 .arg(token.linenumber)
203 .
arg(q->parent()->objectName());
212 QStringLiteral(
"%1, line %2, %3")
214 .arg(token.linenumber)
215 .arg(q->parent()->objectName()));
218 nodeList = extendNodeList(nodeList,
219 new VariableNode(filterExpression, parent));
222 const auto command = token.content.section(
QLatin1Char(
' '), 0, 0);
227 q->prependToken(token);
231 if (command.isEmpty()) {
233 Q_ASSERT(q->hasNextToken());
234 message = QStringLiteral(
"Empty block tag before \"%1\", line %2, %3")
235 .
arg(token.content.left(20))
236 .arg(token.linenumber)
237 .
arg(q->parent()->objectName());
241 auto nodeFactory = m_nodeFactories[command];
245 q->invalidBlockTag(token, command, stopAt);
251 n = nodeFactory->getNode(token.content, q);
254 QStringLiteral(
"%1, line %2, %3")
256 .arg(token.linenumber)
257 .arg(q->parent()->objectName()));
262 QStringLiteral(
"Failed to get node from %1, line %2, %3")
264 .arg(token.linenumber)
265 .arg(q->parent()->objectName()));
270 nodeList = extendNodeList(nodeList, n);
276 = QStringLiteral(
"Unclosed tag in template %1. Expected one of: (%2)")
277 .arg(q->parent()->objectName(),
288 return !d->m_tokenList.isEmpty();
294 return d->m_tokenList.takeFirst();
300 d->m_tokenList.removeFirst();
303 void Parser::invalidBlockTag(
const Token &token,
const QString &command,
306 if (!stopAt.
empty()) {
308 InvalidBlockTagError,
309 QStringLiteral(
"Invalid block tag on line %1: '%2', expected '%3'")
311 .arg(command, stopAt.
join(QStringLiteral(
"', '"))));
314 InvalidBlockTagError,
315 QStringLiteral(
"Invalid block tag on line %1: '%2\''. Did you forget " 316 "to register or load this tag?")
325 d->m_tokenList.prepend(token);
NodeList parse(Node *parent, const QStringList &stopAt={})
bool containsNonText() const
bool contains(QLatin1StringView str, Qt::CaseSensitivity cs) const const
The Cutelee namespace holds all public Cutelee API.
virtual QHash< QString, Filter * > filters(const QString &name={})
void append(Cutelee::Node *node)
virtual QHash< QString, AbstractNodeFactory * > nodeFactories(const QString &name={})
void prependToken(const Token &token)
std::shared_ptr< Filter > getFilter(const QString &name) const
QString join(QChar separator) const const
NodeList parse(QObject *parent, const QStringList &stopAt)
bool isEmpty() const const
Cutelee::Engine is the main entry point for creating Cutelee Templates.
Base class for all nodes.
The Token is a text fragment.
void skipPast(const QString &tag)
QStringList defaultLibraries() const
void setParent(QObject *parent)
The TagLibraryInterface returns available tags and filters from libraries.
int linenumber
The line number this Token starts at.
The Token is a variable node.
The Parser class processes a string template into a tree of nodes.
A list of Nodes with some convenience API for rendering them.
A FilterExpression object represents a filter expression in a template.
The Token is a block, ie, part of a tag.
T qobject_cast(QObject *object)
QObject * parent() const const
QString arg(Args &&... args) const const
Parser(const QList< Token > &tokenList, QObject *parent)
An exception for use when implementing template tags.
bool hasNextToken() const