23#include "blockcontext.h"
26#include "rendercontext.h"
30static const char *
const __loadedBlocks =
"__loadedBlocks";
33#define BLOCK_CONTEXT_KEY 0
35BlockNodeFactory::BlockNodeFactory(
QObject *parent)
44 if (expr.size() != 2) {
46 QStringLiteral(
"block tag takes one argument"));
49 const auto blockName = expr.
at(1);
51 auto loadedBlocksVariant = p->
property(__loadedBlocks);
52 QVariantList blockVariantList;
54 if (loadedBlocksVariant.isValid()
55 && loadedBlocksVariant.userType() == qMetaTypeId<QVariantList>()) {
56 blockVariantList = loadedBlocksVariant.value<QVariantList>();
57 for (
auto &item : blockVariantList) {
58 const auto blockNodeName = item.toString();
60 if (blockNodeName == blockName) {
63 QStringLiteral(
"'block' tag with name '%1' appears more than once.")
69 blockVariantList.append(blockName);
70 loadedBlocksVariant =
QVariant(blockVariantList);
72 p->
setProperty(__loadedBlocks, loadedBlocksVariant);
75 const auto list = p->
parse(n, QStringLiteral(
"endblock"));
78 const QStringList acceptableBlocks{QStringLiteral(
"endblock"),
79 QStringLiteral(
"endblock ") + blockName};
80 if (!acceptableBlocks.
contains(endBlock.content)) {
81 p->invalidBlockTag(endBlock, QStringLiteral(
"endblock"), acceptableBlocks);
90 :
Node(parent), m_name(name), m_stream(0)
92 qRegisterMetaType<Cutelee::SafeString>(
"Cutelee::SafeString");
95BlockNode::~BlockNode() {}
97void BlockNode::setNodeList(
const NodeList &list)
const { m_list = list; }
106 if (blockContext.isEmpty()) {
109 c->
insert(QStringLiteral(
"block"),
112 m_list.render(stream, c);
115 auto block =
static_cast<const BlockNode *
>(blockContext.pop(m_name));
121 const auto list = block->m_list;
123 block =
new BlockNode(block->m_name, 0);
124 block->setNodeList(list);
125 block->m_context = c;
126 block->m_stream = stream;
127 c->
insert(QStringLiteral(
"block"),
130 list.render(stream, c);
134 blockContext.push(m_name, push);
143 if (m_context->renderContext()->contains(BLOCK_CONTEXT_KEY)) {
144 QVariant &variant = m_context->renderContext()->
data(BLOCK_CONTEXT_KEY);
146 auto block = blockContext.getBlock(m_name);
150 auto superStream = m_stream->clone(&superTextStream);
151 const_cast<BlockNode *
>(
this)->
render(superStream.get(), m_context);
158NodeList BlockNode::nodeList()
const {
return m_list; }
160QString BlockNode::name()
const {
return m_name; }
Node * getNode(const QString &tagContent, Parser *p) const override
void render(OutputStream *stream, Context *c) const override
SafeString getSuper() const
Base class for all NodeFactories.
The Context class holds the context to render a Template with.
RenderContext * renderContext() const
void insert(const QString &name, QObject *object)
An exception for use when implementing template tags.
A list of Nodes with some convenience API for rendering them.
Base class for all nodes.
The OutputStream class is used to render templates to a QTextStream.
The Parser class processes a string template into a tree of nodes.
NodeList parse(Node *parent, const QStringList &stopAt={})
QVariant & data(const Node *const scopeNode)
A QString wrapper class for containing whether a string is safe or needs to be escaped.
Cutelee::SafeString markSafe(const Cutelee::SafeString &input)
QList< T >::const_reference at(qsizetype i) const const
QVariant property(const char *name) const const
bool setProperty(const char *name, QVariant &&value)
QStringList split(QChar sep, Qt::SplitBehavior behavior, Qt::CaseSensitivity cs) const const
bool contains(QLatin1StringView str, Qt::CaseSensitivity cs) const const
QVariant fromValue(T &&value)
void setValue(QVariant &&value)
Utility functions used throughout Cutelee.