Building a custom template for the wagtail StreamField
block I found myself in the situation that I need somehow pass the ID of the current block to the other views.
For instance when the URL is clicked in the particular block, the landing page view
must know exactly in which of the blocks the URL has been clicked. Then the view
can extract other information which is associated with the particular block but not necessarily visually present to the user.
My current strategy is using the snippets
, so I can pass the ID of the snippet
and the view
may obtain related but beforehand hidden data.
This works not so bad, but people have to edit the content in two places and I have to look at their sad faces.
It seems that the value
variable in the block template context is an instance of the wagtail.core.blocks.struct_block.StructValue
, which gives me access to all the fields of the block but it doesn't seem to reveal its footprint in the DB.
Further value
has an interesting attribute: value.block
, which seems like it's an instance of the actual model used to construct the block, but again I can't find anything useful like id
or pk
which would allow to identify that instance in the database.
Is there a way?
The block IDs you see in the database representation of a StreamField are a detail implemented by the enclosing StreamBlock, so that we can keep track of each block's history as it gets added / moved / deleted from the stream. The items within the stream do not know their own ID - this is because they could be any possible data type (for example, a CharBlock
produces a string value, and you can't attach an ID to a string). As a result, the block template doesn't have access to the ID either.
To access the ID, you'll need to make use of the BoundBlock
(or, more precisely, StreamChild
) object that's returned whenever you iterate over the StreamField value (or access it by index, e.g. page.body[0]
or page.body.0
within template code); this object is a wrapper around the block value which knows the block's type and ID. (More background on BoundBlock
in the docs here: http://docs.wagtail.io/en/v2.0/topics/streamfield.html#boundblocks-and-values)
{% for block in page.body %}
{% include_block block with id=block.id %}
{% endfor %}
Here block
is an instance of StreamChild
, which has 'value', 'block_type' and 'id' properties. Normally, the {% include_block %}
tag will just pass on the value
variable to the block template, but here we're passing id
as an additional variable that will now be available within that block template.
StreamField blocks are not 'real' database objects, so to retrieve the value again based on the ID you'll need to scan through the StreamField, using code such as:
value = None
for block in page.body:
if block.id == requested_id:
value = block.value
break
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With