Add Block.is_previewable to avoid showing previews without proper configuration

pull/12700/head
Sage Abdullah 2025-01-09 17:14:59 +00:00
rodzic adf52d85c7
commit c13caa952f
Nie znaleziono w bazie danych klucza dla tego podpisu
ID klucza GPG: EB1A33CC51CC0217
13 zmienionych plików z 91 dodań i 3 usunięć

Wyświetl plik

@ -18,7 +18,11 @@
<body>
<main>
{% block content %}
{% include_block bound_block %}
{% if block_def.is_previewable %}
{% include_block bound_block %}
{% else %}
{% translate 'Preview not available' %}
{% endif %}
{% endblock %}
</main>

Wyświetl plik

@ -82,7 +82,7 @@ class TestStreamFieldBlockPreviewView(WagtailTestUtils, TestCase):
)
self.user.save()
block = blocks.CharBlock()
block = blocks.CharBlock(preview_value="Hello, world!")
response = self.get(block)
self.assertEqual(response.status_code, 200)
@ -93,7 +93,7 @@ class TestStreamFieldBlockPreviewView(WagtailTestUtils, TestCase):
soup = self.get_soup(response.content)
main = soup.select_one("main")
self.assertIsNotNone(main)
self.assertEqual(main.text.strip(), "None")
self.assertEqual(main.text.strip(), "Preview not available")
def test_preview_value_falls_back_to_default(self):
block = blocks.IntegerBlock(default=42)

Wyświetl plik

@ -293,6 +293,25 @@ class Block(metaclass=BaseBlock):
return self.normalize(self.meta.preview_value)
return self.get_default()
@cached_property
def is_previewable(self):
# To prevent showing a broken preview if the block preview has not been
# configured, consider the block to be previewable if either:
# - a preview template, preview value, or default value is defined
# - a preview method is overridden
# which are the intended ways to configure block previews.
#
# If a block is made previewable by other means, the `is_previewable`
# property should be overridden to return `True`.
return (
hasattr(self.meta, "preview_template")
or hasattr(self.meta, "preview_value")
or getattr(self.meta, "default", None) is not None
or self.__class__.get_preview_context is not Block.get_preview_context
or self.__class__.get_preview_template is not Block.get_preview_template
or self.__class__.get_preview_value is not Block.get_preview_value
)
def get_description(self):
return getattr(self.meta, "description", "")

Wyświetl plik

@ -117,6 +117,7 @@ class FieldBlockAdapter(Adapter):
"required": block.required,
"icon": block.meta.icon,
"blockDefId": block.definition_prefix,
"isPreviewable": block.is_previewable,
"classname": " ".join(classname),
"showAddCommentButton": getattr(
block.field.widget, "show_add_comment_button", True

Wyświetl plik

@ -452,6 +452,7 @@ class ListBlockAdapter(Adapter):
"description": block.get_description(),
"icon": block.meta.icon,
"blockDefId": block.definition_prefix,
"isPreviewable": block.is_previewable,
"classname": block.meta.form_classname,
"collapsed": block.meta.collapsed,
"strings": {

Wyświetl plik

@ -60,6 +60,7 @@ class StaticBlockAdapter(Adapter):
"label": block.label,
"description": block.get_description(),
"blockDefId": block.definition_prefix,
"isPreviewable": block.is_previewable,
},
]

Wyświetl plik

@ -830,6 +830,7 @@ class StreamBlockAdapter(Adapter):
"required": block.required,
"icon": block.meta.icon,
"blockDefId": block.definition_prefix,
"isPreviewable": block.is_previewable,
"classname": block.meta.form_classname,
"maxNum": block.meta.max_num,
"minNum": block.meta.min_num,

Wyświetl plik

@ -408,6 +408,7 @@ class StructBlockAdapter(Adapter):
"required": block.required,
"icon": block.meta.icon,
"blockDefId": block.definition_prefix,
"isPreviewable": block.is_previewable,
"classname": block.meta.form_classname,
}

Wyświetl plik

@ -573,6 +573,7 @@ class TestTableBlockForm(WagtailTestUtils, SimpleTestCase):
"required": True,
"icon": "table",
"blockDefId": block.definition_prefix,
"isPreviewable": block.is_previewable,
"classname": "w-field w-field--char_field w-field--table_input",
"showAddCommentButton": True,
"strings": {"ADD_COMMENT": "Add Comment"},

Wyświetl plik

@ -336,6 +336,7 @@ class TypedTableBlockAdapter(Adapter):
"required": block.required,
"icon": block.meta.icon,
"blockDefId": block.definition_prefix,
"isPreviewable": block.is_previewable,
"strings": {
"CAPTION": _("Caption"),
"CAPTION_HELP_TEXT": _(

Wyświetl plik

@ -239,6 +239,7 @@ class TestTableBlock(TestCase):
"required": False,
"icon": "table",
"blockDefId": block.definition_prefix,
"isPreviewable": block.is_previewable,
"strings": {
"CAPTION": "Caption",
"CAPTION_HELP_TEXT": (

Wyświetl plik

@ -5544,6 +5544,7 @@ class TestSnippetChooserBlock(TestCase):
"required": True,
"icon": "snippet",
"blockDefId": block.definition_prefix,
"isPreviewable": block.is_previewable,
"helpText": "pick an advert, any advert",
"classname": "w-field w-field--model_choice_field w-field--admin_snippet_chooser",
"showAddCommentButton": True,
@ -5856,6 +5857,7 @@ class TestSnippetChooserBlockWithCustomPrimaryKey(TestCase):
"required": True,
"icon": "snippet",
"blockDefId": block.definition_prefix,
"isPreviewable": block.is_previewable,
"helpText": "pick an advert, any advert",
"classname": "w-field w-field--model_choice_field w-field--admin_snippet_chooser",
"showAddCommentButton": True,

Wyświetl plik

@ -62,6 +62,32 @@ class TestBlock(SimpleTestCase):
block = blocks.Block()
self.assertIs(blocks.Block.definition_registry[block.definition_prefix], block)
def test_block_is_previewable(self):
class CustomContextBlock(blocks.Block):
def get_preview_context(self, value, parent_context=None):
return {"value": value, "foo": "bar"}
class CustomTemplateBlock(blocks.Block):
def get_preview_template(self, value=None, context=None):
return "foo.html"
class CustomValueBlock(blocks.Block):
def get_preview_value(self):
return "foo"
cases = [
(blocks.Block(), False),
(blocks.Block(preview_template="foo.html"), True),
(blocks.Block(preview_value="foo"), True),
(blocks.Block(default="bar"), True),
(CustomContextBlock(), True),
(CustomTemplateBlock(), True),
(CustomValueBlock(), True),
]
for block, is_previewable in cases:
with self.subTest(block=block):
self.assertEqual(block.is_previewable, is_previewable)
class TestFieldBlock(WagtailTestUtils, SimpleTestCase):
def test_charfield_render(self):
@ -100,6 +126,7 @@ class TestFieldBlock(WagtailTestUtils, SimpleTestCase):
"required": True,
"icon": "placeholder",
"blockDefId": block.definition_prefix,
"isPreviewable": block.is_previewable,
"classname": "w-field w-field--char_field w-field--text_input",
"showAddCommentButton": True,
"strings": {"ADD_COMMENT": "Add Comment"},
@ -208,6 +235,7 @@ class TestFieldBlock(WagtailTestUtils, SimpleTestCase):
"required": True,
"icon": "placeholder",
"blockDefId": block.definition_prefix,
"isPreviewable": block.is_previewable,
"classname": "w-field w-field--choice_field w-field--select",
"showAddCommentButton": True,
"strings": {"ADD_COMMENT": "Add Comment"},
@ -619,6 +647,7 @@ class TestRichTextBlock(TestCase):
"label": "Test richtextblock",
"description": "",
"blockDefId": block.definition_prefix,
"isPreviewable": block.is_previewable,
"required": True,
"showAddCommentButton": True,
"strings": {"ADD_COMMENT": "Add Comment"},
@ -643,6 +672,7 @@ class TestRichTextBlock(TestCase):
"required": True,
"icon": "pilcrow",
"blockDefId": block.definition_prefix,
"isPreviewable": block.is_previewable,
"classname": "w-field w-field--char_field w-field--draftail_rich_text_area",
"showAddCommentButton": False, # Draftail manages its own comments
"strings": {"ADD_COMMENT": "Add Comment"},
@ -667,6 +697,7 @@ class TestRichTextBlock(TestCase):
"required": True,
"icon": "pilcrow",
"blockDefId": block.definition_prefix,
"isPreviewable": block.is_previewable,
"classname": "w-field w-field--char_field w-field--draftail_rich_text_area",
"showAddCommentButton": False, # Draftail manages its own comments
"strings": {"ADD_COMMENT": "Add Comment"},
@ -784,6 +815,7 @@ class TestChoiceBlock(WagtailTestUtils, SimpleTestCase):
"required": True,
"icon": "placeholder",
"blockDefId": block.definition_prefix,
"isPreviewable": block.is_previewable,
"classname": "w-field w-field--choice_field w-field--select",
"showAddCommentButton": True,
"strings": {"ADD_COMMENT": "Add Comment"},
@ -1187,6 +1219,7 @@ class TestMultipleChoiceBlock(WagtailTestUtils, SimpleTestCase):
"required": True,
"icon": "placeholder",
"blockDefId": block.definition_prefix,
"isPreviewable": block.is_previewable,
"classname": "w-field w-field--multiple_choice_field w-field--select_multiple",
"showAddCommentButton": True,
"strings": {"ADD_COMMENT": "Add Comment"},
@ -1625,6 +1658,7 @@ class TestRawHTMLBlock(unittest.TestCase):
"required": True,
"icon": "code",
"blockDefId": block.definition_prefix,
"isPreviewable": block.is_previewable,
"classname": "w-field w-field--char_field w-field--textarea",
"showAddCommentButton": True,
"strings": {"ADD_COMMENT": "Add Comment"},
@ -1969,6 +2003,7 @@ class TestStructBlock(SimpleTestCase):
"required": False,
"icon": "placeholder",
"blockDefId": block.definition_prefix,
"isPreviewable": block.is_previewable,
"classname": "struct-block",
},
)
@ -2000,6 +2035,7 @@ class TestStructBlock(SimpleTestCase):
"required": False,
"icon": "placeholder",
"blockDefId": block.definition_prefix,
"isPreviewable": block.is_previewable,
"classname": "struct-block",
"formTemplate": "<div>Hello</div>",
},
@ -2026,6 +2062,7 @@ class TestStructBlock(SimpleTestCase):
"required": False,
"icon": "placeholder",
"blockDefId": block.definition_prefix,
"isPreviewable": block.is_previewable,
"classname": "struct-block",
"formTemplate": "<div>Hello</div>",
},
@ -2061,6 +2098,7 @@ class TestStructBlock(SimpleTestCase):
"required": False,
"icon": "placeholder",
"blockDefId": block.definition_prefix,
"isPreviewable": block.is_previewable,
"classname": "struct-block",
"helpIcon": (
'<svg class="icon icon-help default" aria-hidden="true">'
@ -2088,6 +2126,7 @@ class TestStructBlock(SimpleTestCase):
"required": False,
"icon": "placeholder",
"blockDefId": block.definition_prefix,
"isPreviewable": block.is_previewable,
"classname": "struct-block",
"helpIcon": (
'<svg class="icon icon-help default" aria-hidden="true">'
@ -2718,6 +2757,7 @@ class TestListBlock(WagtailTestUtils, SimpleTestCase):
"description": "",
"icon": "placeholder",
"blockDefId": block.definition_prefix,
"isPreviewable": block.is_previewable,
"classname": None,
"collapsed": False,
"strings": {
@ -2750,6 +2790,7 @@ class TestListBlock(WagtailTestUtils, SimpleTestCase):
"description": "",
"icon": "placeholder",
"blockDefId": block.definition_prefix,
"isPreviewable": block.is_previewable,
"classname": None,
"collapsed": False,
"minNum": 2,
@ -2926,6 +2967,7 @@ class TestListBlock(WagtailTestUtils, SimpleTestCase):
"description": "",
"icon": "placeholder",
"blockDefId": block.definition_prefix,
"isPreviewable": block.is_previewable,
"classname": "special-list-class",
"collapsed": False,
"strings": {
@ -2961,6 +3003,7 @@ class TestListBlock(WagtailTestUtils, SimpleTestCase):
"description": "",
"icon": "placeholder",
"blockDefId": block.definition_prefix,
"isPreviewable": block.is_previewable,
"classname": "custom-list-class",
"collapsed": False,
"strings": {
@ -3600,6 +3643,7 @@ class TestStreamBlock(WagtailTestUtils, SimpleTestCase):
"description": "",
"icon": "placeholder",
"blockDefId": block.definition_prefix,
"isPreviewable": block.is_previewable,
"classname": None,
"collapsed": False,
"maxNum": None,
@ -4328,6 +4372,7 @@ class TestStreamBlock(WagtailTestUtils, SimpleTestCase):
"description": "",
"icon": "placeholder",
"blockDefId": block.definition_prefix,
"isPreviewable": block.is_previewable,
"minNum": None,
"maxNum": None,
"blockCounts": {},
@ -4432,6 +4477,7 @@ class TestStreamBlock(WagtailTestUtils, SimpleTestCase):
"description": "",
"icon": "placeholder",
"blockDefId": block.definition_prefix,
"isPreviewable": block.is_previewable,
"minNum": None,
"maxNum": None,
"blockCounts": {},
@ -4781,6 +4827,7 @@ class TestPageChooserBlock(TestCase):
"required": True,
"icon": "doc-empty-inverse",
"blockDefId": block.definition_prefix,
"isPreviewable": block.is_previewable,
"helpText": "pick a page, any page",
"classname": "w-field w-field--model_choice_field w-field--admin_page_chooser",
"showAddCommentButton": True,
@ -4985,6 +5032,7 @@ class TestStaticBlock(unittest.TestCase):
"text": "Latest posts - This block doesn't need to be configured, it will be displayed automatically",
"icon": "placeholder",
"blockDefId": block.definition_prefix,
"isPreviewable": block.is_previewable,
"label": "Posts static block",
"description": "",
},
@ -5008,6 +5056,7 @@ class TestStaticBlock(unittest.TestCase):
"text": "Latest posts - This block doesn't need to be configured, it will be displayed automatically",
"icon": "placeholder",
"blockDefId": block.definition_prefix,
"isPreviewable": block.is_previewable,
"label": "Posts static block",
"description": "",
},
@ -5030,6 +5079,7 @@ class TestStaticBlock(unittest.TestCase):
"text": "Latest posts: this block has no options.",
"icon": "placeholder",
"blockDefId": block.definition_prefix,
"isPreviewable": block.is_previewable,
"label": "Latest posts",
"description": "",
},
@ -5053,6 +5103,7 @@ class TestStaticBlock(unittest.TestCase):
"text": "Posts static block: this block has no options.",
"icon": "placeholder",
"blockDefId": block.definition_prefix,
"isPreviewable": block.is_previewable,
"label": "Posts static block",
"description": "",
},
@ -5076,6 +5127,7 @@ class TestStaticBlock(unittest.TestCase):
"html": "<b>Latest posts</b> - This block doesn't need to be configured, it will be displayed automatically",
"icon": "placeholder",
"blockDefId": block.definition_prefix,
"isPreviewable": block.is_previewable,
"label": "Posts static block",
"description": "",
},
@ -5133,6 +5185,7 @@ class TestDateBlock(TestCase):
"required": True,
"icon": "date",
"blockDefId": block.definition_prefix,
"isPreviewable": block.is_previewable,
"classname": "w-field w-field--date_field w-field--admin_date_input",
"showAddCommentButton": True,
"strings": {"ADD_COMMENT": "Add Comment"},
@ -5168,6 +5221,7 @@ class TestTimeBlock(TestCase):
"required": True,
"icon": "time",
"blockDefId": block.definition_prefix,
"isPreviewable": block.is_previewable,
"classname": "w-field w-field--time_field w-field--admin_time_input",
"showAddCommentButton": True,
"strings": {"ADD_COMMENT": "Add Comment"},
@ -5203,6 +5257,7 @@ class TestDateTimeBlock(TestCase):
"required": True,
"icon": "date",
"blockDefId": block.definition_prefix,
"isPreviewable": block.is_previewable,
"classname": "w-field w-field--date_time_field w-field--admin_date_time_input",
"showAddCommentButton": True,
"strings": {"ADD_COMMENT": "Add Comment"},