9 Commits

Author SHA1 Message Date
Jürgen Mummert
80dd19257c Align pinboard assets with /assets path convention 2026-03-04 17:50:10 +01:00
Jürgen Mummert
980a6d83dd Include twig root marker and pending pinnwand DCA/template fixes 2026-03-04 17:39:48 +01:00
Jürgen Mummert
4ef0608d64 Ensure pinnwand module palette exposes imgSize reliably 2026-03-04 17:35:08 +01:00
Jürgen Mummert
e3f842f2d3 Fix image lightbox click handling and module image size rendering 2026-03-04 17:26:46 +01:00
Jürgen Mummert
8848fdcbc3 Fix frontend module template location for Contao resolver 2026-03-04 16:50:22 +01:00
Jürgen Mummert
f8a5d9348e Refine pinnwand backend behavior and fix frontend template rendering 2026-03-04 16:43:05 +01:00
Jürgen Mummert
ca7f7d759b Fix bundle path resolution so tl_pinnwand schema is detected 2026-03-04 16:29:26 +01:00
Jürgen Mummert
5c1fda002c Fix service loading and model compatibility for Contao 5.7 2026-03-04 16:17:41 +01:00
Jürgen Mummert
4a183ca344 fix: improve bundle auto-enable compatibility 2026-03-04 16:01:41 +01:00
18 changed files with 432 additions and 77 deletions

View File

@@ -14,6 +14,12 @@ Contao 5.7 Bundle für eine Pinwand mit frei angeordneten Pinnwandeinträgen.
## Installation (Bundle im Projekt einbinden) ## Installation (Bundle im Projekt einbinden)
1. Bundle per Composer einbinden. 1. Bundle per Composer einbinden.
2. Cache leeren. 2. Falls Composer-Plugins im Zielsystem eingeschränkt sind, Bundle manuell in `config/bundles.php` aktivieren:
3. Datenbank aktualisieren.
4. Assets installieren, damit CSS/JS unter `bundles/contaopinboard/` verfügbar sind. ```php
Eiswurm\ContaoPinboardBundle\ContaoPinboardBundle::class => ['all' => true],
```
3. Cache leeren.
4. Datenbank aktualisieren.
5. Assets installieren, damit CSS/JS unter `bundles/contaopinboard/` verfügbar sind.

View File

@@ -5,7 +5,8 @@
"license": "proprietary", "license": "proprietary",
"require": { "require": {
"php": "^8.4", "php": "^8.4",
"contao/core-bundle": "^5.7" "contao/core-bundle": "^5.7",
"contao/manager-plugin": "^2.0"
}, },
"autoload": { "autoload": {
"psr-4": { "psr-4": {

View File

@@ -9,3 +9,5 @@ $GLOBALS['BE_MOD']['content']['pinnwand'] = [
]; ];
$GLOBALS['TL_MODELS']['tl_pinnwand'] = PinboardModel::class; $GLOBALS['TL_MODELS']['tl_pinnwand'] = PinboardModel::class;
$GLOBALS['TL_DCA']['tl_module']['palettes']['pinnwand'] = '{title_legend},name,headline,type;{image_legend},imgSize;{protected_legend:hide},protected;{expert_legend:hide},guests,cssID';

View File

@@ -15,8 +15,8 @@ services:
name: contao.frontend_module name: contao.frontend_module
category: application category: application
type: pinnwand type: pinnwand
template: frontend_module/pinnwand template: frontend/pinnwand
Eiswurm\ContaoPinboardBundle\EventListener\DataContainer\PinboardTimestampListener: Eiswurm\ContaoPinboardBundle\EventListener\DataContainer\PinboardTimestampListener:
tags: tags:
- { name: contao.callback, table: tl_pinnwand, target: config.onsubmit } - { name: contao.callback, table: tl_pinnwand, target: config.onbeforesubmit, method: onBeforeSubmit }

5
contao/dca/tl_module.php Normal file
View File

@@ -0,0 +1,5 @@
<?php
declare(strict_types=1);
$GLOBALS['TL_DCA']['tl_module']['palettes']['pinnwand'] = '{title_legend},name,headline,type;{image_legend},imgSize;{protected_legend:hide},protected;{expert_legend:hide},guests,cssID';

View File

@@ -2,6 +2,10 @@
declare(strict_types=1); declare(strict_types=1);
use Contao\DataContainer;
use Contao\Database;
use Contao\Input;
$GLOBALS['TL_DCA']['tl_pinnwand'] = [ $GLOBALS['TL_DCA']['tl_pinnwand'] = [
'config' => [ 'config' => [
'dataContainer' => Contao\DC_Table::class, 'dataContainer' => Contao\DC_Table::class,
@@ -22,8 +26,8 @@ $GLOBALS['TL_DCA']['tl_pinnwand'] = [
'panelLayout' => 'filter;sort,search,limit', 'panelLayout' => 'filter;sort,search,limit',
], ],
'label' => [ 'label' => [
'fields' => ['ueberschrift', 'dateAdded'], 'fields' => ['ueberschrift', 'dateAdded', 'dateModified'],
'format' => '%s <span style="color:#999;padding-left:3px">[%s]</span>', 'showColumns' => true,
], ],
'global_operations' => [ 'global_operations' => [
'all' => [ 'all' => [
@@ -46,6 +50,10 @@ $GLOBALS['TL_DCA']['tl_pinnwand'] = [
'icon' => 'delete.svg', 'icon' => 'delete.svg',
'attributes' => 'onclick="if(!confirm(\'' . ($GLOBALS['TL_LANG']['MSC']['deleteConfirm'] ?? 'Möchten Sie den Eintrag wirklich löschen?') . '\'))return false;Backend.getScrollOffset()"', 'attributes' => 'onclick="if(!confirm(\'' . ($GLOBALS['TL_LANG']['MSC']['deleteConfirm'] ?? 'Möchten Sie den Eintrag wirklich löschen?') . '\'))return false;Backend.getScrollOffset()"',
], ],
'toggle' => [
'href' => 'act=toggle&amp;field=published',
'icon' => 'visible.svg',
],
'show' => [ 'show' => [
'href' => 'act=show', 'href' => 'act=show',
'icon' => 'show.svg', 'icon' => 'show.svg',
@@ -74,7 +82,7 @@ $GLOBALS['TL_DCA']['tl_pinnwand'] = [
'exclude' => true, 'exclude' => true,
'search' => true, 'search' => true,
'inputType' => 'textarea', 'inputType' => 'textarea',
'eval' => ['mandatory' => true, 'maxlength' => 3000, 'rte' => 'tinyMCE', 'allowHtml' => true, 'tl_class' => 'clr'], 'eval' => ['mandatory' => true, 'maxlength' => 3000, 'tl_class' => 'clr'],
'sql' => 'text NULL', 'sql' => 'text NULL',
], ],
'link' => [ 'link' => [
@@ -94,6 +102,7 @@ $GLOBALS['TL_DCA']['tl_pinnwand'] = [
'sorting' => true, 'sorting' => true,
'flag' => 6, 'flag' => 6,
'inputType' => 'text', 'inputType' => 'text',
'default' => time(),
'eval' => ['rgxp' => 'datim', 'datepicker' => true, 'mandatory' => true, 'tl_class' => 'w50 wizard'], 'eval' => ['rgxp' => 'datim', 'datepicker' => true, 'mandatory' => true, 'tl_class' => 'w50 wizard'],
'sql' => 'int(10) unsigned NOT NULL default 0', 'sql' => 'int(10) unsigned NOT NULL default 0',
], ],
@@ -101,7 +110,8 @@ $GLOBALS['TL_DCA']['tl_pinnwand'] = [
'sorting' => true, 'sorting' => true,
'flag' => 6, 'flag' => 6,
'inputType' => 'text', 'inputType' => 'text',
'eval' => ['rgxp' => 'datim', 'readonly' => true, 'disabled' => true, 'tl_class' => 'w50'], 'default' => time(),
'eval' => ['rgxp' => 'datim', 'readonly' => true, 'tl_class' => 'w50'],
'sql' => 'int(10) unsigned NOT NULL default 0', 'sql' => 'int(10) unsigned NOT NULL default 0',
], ],
'published' => [ 'published' => [
@@ -122,3 +132,52 @@ $GLOBALS['TL_DCA']['tl_pinnwand'] = [
], ],
], ],
]; ];
$GLOBALS['TL_DCA']['tl_pinnwand']['fields']['ueberschrift']['save_callback'][] = static function (string $value, DataContainer $dataContainer): string {
if ('copy' !== Input::get('act')) {
return $value;
}
$baseHeadline = trim($value);
if ('' === $baseHeadline) {
return $value;
}
$start = 2;
$base = $baseHeadline;
if (preg_match('/^(.*)\s\((\d+)\)$/', $baseHeadline, $matches)) {
$base = trim($matches[1]);
$start = (int) $matches[2] + 1;
}
$id = (int) ($dataContainer->id ?? 0);
$number = max(2, $start);
$candidate = sprintf('%s (%d)', $base, $number);
do {
$exists = Database::getInstance()
->prepare('SELECT id FROM tl_pinnwand WHERE ueberschrift=? AND id!=?')
->execute($candidate, $id);
if ($exists->numRows < 1) {
return $candidate;
}
++$number;
$candidate = sprintf('%s (%d)', $base, $number);
} while (true);
};
$GLOBALS['TL_DCA']['tl_pinnwand']['fields']['dateAdded']['load_callback'][] = static function (mixed $value): int {
$timestamp = (int) $value;
return $timestamp > 0 ? $timestamp : time();
};
$GLOBALS['TL_DCA']['tl_pinnwand']['fields']['dateModified']['load_callback'][] = static function (mixed $value): int {
$timestamp = (int) $value;
return $timestamp > 0 ? $timestamp : time();
};

View File

@@ -2,5 +2,5 @@
declare(strict_types=1); declare(strict_types=1);
$GLOBALS['TL_LANG']['MOD']['pinnwand'] = ['Pinwand', 'Pinnwandeinträge verwalten']; $GLOBALS['TL_LANG']['MOD']['pinnwand'] = ['Pinnwand', 'Pinnwandeinträge verwalten'];
$GLOBALS['TL_LANG']['FMD']['pinnwand'] = ['Pinwand', 'Zeigt veröffentlichte Pinnwandeinträge als frei angeordnete Notizzettel an']; $GLOBALS['TL_LANG']['FMD']['pinnwand'] = ['Pinnwand', 'Zeigt veröffentlichte Pinnwandeinträge als frei angeordnete Notizzettel an'];

View File

View File

@@ -0,0 +1,36 @@
{% use '@Contao/component/_figure.html.twig' %}
<link rel="stylesheet" href="{{ asset('bundles/contaopinboard/assets/pinboard.css') }}">
<section class="pinboard" data-pinboard>
<div class="pinboard__surface" data-pinboard-surface>
{% for entry in entries %}
<article
class="pin-note{% if entry.highlighted %} is-highlighted{% endif %}"
data-pin-note
data-highlighted="{{ entry.highlighted ? '1' : '0' }}"
data-seed="{{ entry.id }}"
aria-label="{{ entry.headline }}"
>
{% if entry.imageFigure %}
<figure class="pin-note__image-wrap">
{% with {figure: entry.imageFigure} %}{{ block('figure_component') }}{% endwith %}
</figure>
{% endif %}
<h3 class="pin-note__headline">{{ entry.headline }}</h3>
<div class="pin-note__text">{{ entry.text|e|nl2br }}</div>
{% if entry.link %}
<p class="pin-note__link-wrap">
<a href="{{ entry.link }}" class="pin-note__link" target="_blank" rel="noopener">Mehr erfahren</a>
</p>
{% endif %}
</article>
{% else %}
<p class="pinboard__empty">Keine veröffentlichten Pinnwandeinträge vorhanden.</p>
{% endfor %}
</div>
</section>
<script src="{{ asset('bundles/contaopinboard/assets/pinboard.js') }}" defer></script>

View File

@@ -8,4 +8,8 @@ use Symfony\Component\HttpKernel\Bundle\Bundle;
final class ContaoPinboardBundle extends Bundle final class ContaoPinboardBundle extends Bundle
{ {
public function getPath(): string
{
return dirname(__DIR__);
}
} }

View File

@@ -5,16 +5,23 @@ declare(strict_types=1);
namespace Eiswurm\ContaoPinboardBundle\Controller\FrontendModule; namespace Eiswurm\ContaoPinboardBundle\Controller\FrontendModule;
use Contao\CoreBundle\Controller\FrontendModule\AbstractFrontendModuleController; use Contao\CoreBundle\Controller\FrontendModule\AbstractFrontendModuleController;
use Contao\CoreBundle\Image\Studio\Studio;
use Contao\CoreBundle\Twig\FragmentTemplate;
use Contao\FilesModel; use Contao\FilesModel;
use Contao\ModuleModel; use Contao\ModuleModel;
use Contao\Template; use Contao\StringUtil;
use Eiswurm\ContaoPinboardBundle\Model\PinboardModel; use Eiswurm\ContaoPinboardBundle\Model\PinboardModel;
use Symfony\Component\HttpFoundation\Request; use Symfony\Component\HttpFoundation\Request;
use Symfony\Component\HttpFoundation\Response; use Symfony\Component\HttpFoundation\Response;
final class PinboardController extends AbstractFrontendModuleController final class PinboardController extends AbstractFrontendModuleController
{ {
protected function getResponse(Template $template, ModuleModel $model, Request $request): Response public function __construct(
private readonly Studio $studio,
) {
}
protected function getResponse(FragmentTemplate $template, ModuleModel $model, Request $request): Response
{ {
$collection = PinboardModel::findBy( $collection = PinboardModel::findBy(
['published = ?'], ['published = ?'],
@@ -26,11 +33,27 @@ final class PinboardController extends AbstractFrontendModuleController
if (null !== $collection) { if (null !== $collection) {
foreach ($collection as $entry) { foreach ($collection as $entry) {
$imagePath = null; $imageFigure = null;
if ($entry->bild) { if ($entry->bild) {
$fileModel = FilesModel::findByUuid($entry->bild); $fileModel = FilesModel::findByUuid($entry->bild);
$imagePath = $fileModel?->path;
if (null !== $fileModel) {
$figureBuilder = $this->studio
->createFigureBuilder()
->fromFilesModel($fileModel)
->enableLightbox()
->setLightboxGroupIdentifier('pinnwand-'.$model->id)
;
$imageSize = StringUtil::deserialize($model->imgSize, true);
if ([] !== $imageSize) {
$figureBuilder->setSize($imageSize);
}
$imageFigure = $figureBuilder->buildIfResourceExists();
}
} }
$notes[] = [ $notes[] = [
@@ -40,15 +63,15 @@ final class PinboardController extends AbstractFrontendModuleController
'link' => (string) $entry->link, 'link' => (string) $entry->link,
'dateAdded' => (int) $entry->dateAdded, 'dateAdded' => (int) $entry->dateAdded,
'dateModified' => (int) $entry->dateModified, 'dateModified' => (int) $entry->dateModified,
'imagePath' => $imagePath, 'imageFigure' => $imageFigure,
'highlighted' => '1' === $entry->hervorgehoben, 'highlighted' => '1' === $entry->hervorgehoben,
]; ];
} }
} }
return $this->render('@Contao/frontend_module/pinnwand.html.twig', [ $template->set('entries', $notes);
'entries' => $notes, $template->set('module', $model);
'module' => $model,
]); return $template->getResponse();
} }
} }

View File

@@ -0,0 +1,19 @@
<?php
declare(strict_types=1);
namespace Eiswurm\ContaoPinboardBundle\DependencyInjection;
use Symfony\Component\Config\FileLocator;
use Symfony\Component\DependencyInjection\ContainerBuilder;
use Symfony\Component\DependencyInjection\Extension\Extension;
use Symfony\Component\DependencyInjection\Loader\YamlFileLoader;
final class ContaoPinboardExtension extends Extension
{
public function load(array $configs, ContainerBuilder $container): void
{
$loader = new YamlFileLoader($container, new FileLocator(__DIR__.'/../../contao/config'));
$loader->load('services.yaml');
}
}

View File

@@ -5,31 +5,30 @@ declare(strict_types=1);
namespace Eiswurm\ContaoPinboardBundle\EventListener\DataContainer; namespace Eiswurm\ContaoPinboardBundle\EventListener\DataContainer;
use Contao\DataContainer; use Contao\DataContainer;
use Contao\Database; use Contao\Input;
final class PinboardTimestampListener final class PinboardTimestampListener
{ {
public function __invoke(DataContainer $dataContainer): void /**
* @param array<string, mixed> $values
*
* @return array<string, mixed>
*/
public function onBeforeSubmit(array $values, DataContainer|null $dataContainer = null): array
{ {
if (null === $dataContainer->id) {
return;
}
$timestamp = time(); $timestamp = time();
$isCopyAction = 'copy' === Input::get('act');
$record = Database::getInstance() if ($isCopyAction || empty($values['dateAdded'])) {
->prepare('SELECT dateAdded FROM tl_pinnwand WHERE id = ?') $values['dateAdded'] = $timestamp;
->limit(1)
->execute($dataContainer->id);
if (!$record->numRows) {
return;
} }
$dateAdded = (int) $record->dateAdded; if ($isCopyAction || empty($values['dateModified'])) {
$values['dateModified'] = $timestamp;
}
Database::getInstance() $values['tstamp'] = $timestamp;
->prepare('UPDATE tl_pinnwand SET dateAdded = ?, dateModified = ?, tstamp = ? WHERE id = ?')
->execute($dateAdded > 0 ? $dateAdded : $timestamp, $timestamp, $timestamp, $dataContainer->id); return $values;
} }
} }

View File

@@ -8,5 +8,5 @@ use Contao\Model;
final class PinboardModel extends Model final class PinboardModel extends Model
{ {
protected static string $strTable = 'tl_pinnwand'; protected static $strTable = 'tl_pinnwand';
} }

View File

@@ -0,0 +1,110 @@
.pinboard {
width: 100%;
max-width: 1400px;
margin: 0 auto;
padding: 1.5rem;
}
.pinboard__surface {
position: relative;
min-height: 42rem;
border-radius: 1.2rem;
overflow: hidden;
background:
radial-gradient(circle at 20% 10%, rgba(255, 255, 255, 0.25), transparent 42%),
radial-gradient(circle at 80% 90%, rgba(50, 30, 10, 0.2), transparent 36%),
repeating-linear-gradient(35deg, rgba(75, 40, 15, 0.18), rgba(75, 40, 15, 0.18) 3px, rgba(93, 52, 23, 0.12) 3px, rgba(93, 52, 23, 0.12) 9px),
linear-gradient(145deg, #b77944 0%, #a66a38 45%, #8d552a 100%);
box-shadow: inset 0 0 0 1px rgba(65, 33, 12, 0.25), inset 0 0 18px rgba(45, 22, 8, 0.35);
}
.pin-note {
position: absolute;
width: clamp(220px, 28vw, 340px);
min-height: 190px;
padding: 1rem 1rem 1.25rem;
border-radius: 0.3rem;
background: linear-gradient(160deg, #fff8a8 0%, #f5eb85 100%);
color: #2a241c;
box-shadow: 0 10px 22px rgba(20, 10, 5, 0.34);
cursor: grab;
user-select: none;
touch-action: none;
transition: box-shadow 160ms ease, transform 160ms ease;
}
.pin-note::before {
content: '';
position: absolute;
top: -9px;
left: 50%;
width: 18px;
height: 18px;
transform: translateX(-50%);
border-radius: 50%;
background: radial-gradient(circle at 35% 35%, #f2f2f2, #979797 68%, #707070 100%);
box-shadow: 0 2px 4px rgba(0, 0, 0, 0.35);
}
.pin-note.is-highlighted {
background: linear-gradient(160deg, #ffd99b 0%, #f7c46f 100%);
}
.pin-note.is-dragging {
cursor: grabbing;
box-shadow: 0 18px 34px rgba(20, 10, 5, 0.46);
}
.pin-note__image-wrap {
margin: 0 0 0.7rem;
}
.pin-note__image {
display: block;
width: 100%;
height: auto;
border-radius: 0.25rem;
}
.pin-note__headline {
margin: 0 0 0.5rem;
font-size: 1.1rem;
line-height: 1.25;
}
.pin-note__text {
font-size: 0.96rem;
line-height: 1.45;
max-height: 14rem;
overflow: hidden;
}
.pin-note__link-wrap {
margin: 0.8rem 0 0;
}
.pin-note__link {
color: inherit;
font-weight: 600;
}
.pinboard__empty {
margin: 0;
padding: 1.5rem;
color: #fff;
font-weight: 600;
}
@media (max-width: 768px) {
.pinboard {
padding: 0.8rem;
}
.pinboard__surface {
min-height: 36rem;
}
.pin-note {
width: min(84vw, 290px);
}
}

View File

@@ -0,0 +1,125 @@
(() => {
const board = document.querySelector('[data-pinboard-surface]');
if (!board) {
return;
}
const notes = Array.from(board.querySelectorAll('[data-pin-note]'));
if (!notes.length) {
return;
}
let zCounter = 200;
const randomBetween = (min, max) => Math.random() * (max - min) + min;
const placeNotes = () => {
const boardRect = board.getBoundingClientRect();
const maxXBase = Math.max(24, boardRect.width - 300);
const maxYBase = Math.max(24, boardRect.height - 220);
notes.forEach((note, index) => {
const maxX = Math.max(16, maxXBase - note.offsetWidth * 0.2);
const maxY = Math.max(16, maxYBase - note.offsetHeight * 0.2);
const x = randomBetween(12, maxX);
const y = randomBetween(12, maxY);
const rotation = randomBetween(-7, 7);
const highlighted = note.dataset.highlighted === '1';
const level = highlighted ? 1000 + index : 100 + index;
note.dataset.baseRotation = String(rotation);
note.dataset.x = String(x);
note.dataset.y = String(y);
note.style.zIndex = String(level);
note.style.transform = `translate(${x}px, ${y}px) rotate(${rotation}deg)`;
});
};
const clampToBoard = (note, x, y) => {
const boardRect = board.getBoundingClientRect();
const noteRect = note.getBoundingClientRect();
const maxX = Math.max(0, boardRect.width - noteRect.width);
const maxY = Math.max(0, boardRect.height - noteRect.height);
return {
x: Math.min(Math.max(0, x), maxX),
y: Math.min(Math.max(0, y), maxY),
};
};
const enableDrag = (note) => {
let dragging = false;
let pointerId = null;
let offsetX = 0;
let offsetY = 0;
note.addEventListener('pointerdown', (event) => {
if (event.target instanceof Element && event.target.closest('a, button, input, select, textarea, label')) {
return;
}
dragging = true;
pointerId = event.pointerId;
note.setPointerCapture(pointerId);
note.classList.add('is-dragging');
note.style.zIndex = String(++zCounter + 2000);
const startX = Number.parseFloat(note.dataset.x ?? '0');
const startY = Number.parseFloat(note.dataset.y ?? '0');
offsetX = event.clientX - startX;
offsetY = event.clientY - startY;
});
note.addEventListener('pointermove', (event) => {
if (!dragging || event.pointerId !== pointerId) {
return;
}
const baseRotation = Number.parseFloat(note.dataset.baseRotation ?? '0');
const nextX = event.clientX - offsetX;
const nextY = event.clientY - offsetY;
const clamped = clampToBoard(note, nextX, nextY);
note.dataset.x = String(clamped.x);
note.dataset.y = String(clamped.y);
note.style.transform = `translate(${clamped.x}px, ${clamped.y}px) rotate(${baseRotation}deg)`;
});
const releaseDrag = (event) => {
if (!dragging || event.pointerId !== pointerId) {
return;
}
dragging = false;
note.classList.remove('is-dragging');
note.releasePointerCapture(pointerId);
pointerId = null;
};
note.addEventListener('pointerup', releaseDrag);
note.addEventListener('pointercancel', releaseDrag);
};
const adjustBoardHeight = () => {
let requiredHeight = 620;
notes.forEach((note) => {
const y = Number.parseFloat(note.dataset.y ?? '0');
requiredHeight = Math.max(requiredHeight, y + note.offsetHeight + 36);
});
board.style.minHeight = `${Math.ceil(requiredHeight)}px`;
};
placeNotes();
notes.forEach(enableDrag);
adjustBoardHeight();
window.addEventListener('resize', () => {
placeNotes();
adjustBoardHeight();
});
})();

View File

@@ -57,6 +57,10 @@
let offsetY = 0; let offsetY = 0;
note.addEventListener('pointerdown', (event) => { note.addEventListener('pointerdown', (event) => {
if (event.target instanceof Element && event.target.closest('a, button, input, select, textarea, label')) {
return;
}
dragging = true; dragging = true;
pointerId = event.pointerId; pointerId = event.pointerId;
note.setPointerCapture(pointerId); note.setPointerCapture(pointerId);

View File

@@ -1,38 +0,0 @@
{% extends '@Contao/frontend_module/_base.html.twig' %}
{% block content %}
<link rel="stylesheet" href="{{ asset('bundles/contaopinboard/css/pinboard.css') }}">
<section class="pinboard" data-pinboard>
<div class="pinboard__surface" data-pinboard-surface>
{% for entry in entries %}
<article
class="pin-note{% if entry.highlighted %} is-highlighted{% endif %}"
data-pin-note
data-highlighted="{{ entry.highlighted ? '1' : '0' }}"
data-seed="{{ entry.id }}"
aria-label="{{ entry.headline }}"
>
{% if entry.imagePath %}
<figure class="pin-note__image-wrap">
<img src="{{ asset(entry.imagePath) }}" alt="" class="pin-note__image">
</figure>
{% endif %}
<h3 class="pin-note__headline">{{ entry.headline }}</h3>
<div class="pin-note__text">{{ entry.text|raw }}</div>
{% if entry.link %}
<p class="pin-note__link-wrap">
<a href="{{ entry.link }}" class="pin-note__link" target="_blank" rel="noopener">Mehr erfahren</a>
</p>
{% endif %}
</article>
{% else %}
<p class="pinboard__empty">Keine veröffentlichten Pinnwandeinträge vorhanden.</p>
{% endfor %}
</div>
</section>
<script src="{{ asset('bundles/contaopinboard/js/pinboard.js') }}" defer></script>
{% endblock %}