Accordion
Lorem ipsum dolor sit amet, consetetur sadipscing elitr, sed diam nonumy eirmod tempor invidunt ut labore et dolore magna aliquyam erat, sed diam voluptua. At vero eos et accusam et justo duo dolores et ea rebum. Stet clita kasd gubergren, no sea takimata sanctus est Lorem ipsum dolor sit amet. Lorem ipsum dolor sit amet, consetetur sadipscing elitr, sed diam nonumy eirmod tempor invidunt ut labore et dolore magna aliquyam erat, sed diam voluptua. At vero eos et accusam et justo duo dolores et ea rebum. Stet clita kasd gubergren, no sea takimata sanctus est Lorem ipsum dolor sit amet.
Lorem ipsum dolor sit amet, consetetur sadipscing elitr, sed diam nonumy eirmod tempor invidunt ut labore et dolore magna aliquyam erat, sed diam voluptua. At vero eos et accusam et justo duo dolores et ea rebum. Stet clita kasd gubergren, no sea takimata sanctus est Lorem ipsum dolor sit amet. Lorem ipsum dolor sit amet, consetetur sadipscing elitr, sed diam nonumy eirmod tempor invidunt ut labore et dolore magna aliquyam erat, sed diam voluptua. At vero eos et accusam et justo duo dolores et ea rebum. Stet clita kasd gubergren, no sea takimata sanctus est Lorem ipsum dolor sit amet.
Developer Reference
Schema
Pass this object inside the blocks option when calling initBridge() to register this block type with the admin UI. See Custom Blocks for the full setup guide.
{
"accordion": {
"blockSchema": {
"properties": {
"panels": {
"title": "Panels",
"widget": "object_list",
"schema": {
"properties": {
"title": {
"title": "Title"
},
"blocks_layout": {
"title": "Content",
"widget": "blocks_layout",
"allowedBlocks": [
"slate",
"image"
],
"defaultBlockType": "slate"
}
}
}
}
}
}
}
}JSON Block Data
Example JSON as stored in the Plone content API. This is the data structure your component will receive in the block prop.
{
"@type": "accordion",
"panels": [
{
"@id": "panel-1",
"title": "Frequently Asked Questions",
"blocks": {
"content-text-1": {
"@type": "slate",
"value": [
{
"type": "p",
"children": [
{
"text": "Here are the answers to common questions."
}
]
}
]
}
},
"blocks_layout": {
"items": [
"content-text-1"
]
}
}
]
}Component
Render component for your frontend framework. Add this to your block renderer's switch/map so it handles this @type.
function AccordionBlock({ block }) {
const panels = block.panels || [];
return (
<div data-block-uid={block['@uid']} className="accordion-block">
{panels.map(panel => {
const panelId = panel['@id'];
return <AccordionPanel key={panelId} panel={panel} panelId={panelId} />;
})}
</div>
);
}
function AccordionPanel({ panel, panelId }) {
const [open, setOpen] = useState(!panel.collapsed);
const contentBlocks = panel.blocks || {};
const contentLayout = panel.blocks_layout?.items || [];
return (
<div data-block-uid={panelId} className="accordion-panel">
<button onClick={() => setOpen(!open)} className="accordion-header">
<span data-edit-text="title">{panel.title}</span>
<span>{open ? '▲' : '▼'}</span>
</button>
{open && (
<div className="accordion-content">
{contentLayout.map(id => (
<BlockRenderer key={id} block={{ ...contentBlocks[id], '@uid': id }} />
))}
</div>
)}
</div>
);
}