Compare commits

...

3 Commits

Author SHA1 Message Date
Jürgen Mummert 478e01c435 style: tweak pin marker layering and image hover 2026-04-01 13:09:13 +02:00
Jürgen Mummert 6269d383e6 feat: render image-only notes without inner padding 2026-04-01 13:03:22 +02:00
Jürgen Mummert bd54d74151 feat: add backend image thumbnail column 2026-04-01 12:57:06 +02:00
4 changed files with 60 additions and 4 deletions
+31 -2
View File
@@ -4,7 +4,9 @@ declare(strict_types=1);
use Contao\DataContainer; use Contao\DataContainer;
use Contao\Database; use Contao\Database;
use Contao\FilesModel;
use Contao\Input; use Contao\Input;
use Contao\System;
$GLOBALS['TL_DCA']['tl_pinnwand'] = [ $GLOBALS['TL_DCA']['tl_pinnwand'] = [
'config' => [ 'config' => [
@@ -26,7 +28,7 @@ $GLOBALS['TL_DCA']['tl_pinnwand'] = [
'panelLayout' => 'filter;sort,search,limit', 'panelLayout' => 'filter;sort,search,limit',
], ],
'label' => [ 'label' => [
'fields' => ['ueberschrift', 'dateAdded', 'dateModified'], 'fields' => ['bild', 'ueberschrift', 'dateAdded', 'dateModified'],
'showColumns' => true, 'showColumns' => true,
], ],
'global_operations' => [ 'global_operations' => [
@@ -132,6 +134,7 @@ $GLOBALS['TL_DCA']['tl_pinnwand'] = [
'filter' => true, 'filter' => true,
'toggle' => true, 'toggle' => true,
'inputType' => 'checkbox', 'inputType' => 'checkbox',
'default' => '1',
'eval' => ['doNotCopy' => true, 'tl_class' => 'w50 m12'], 'eval' => ['doNotCopy' => true, 'tl_class' => 'w50 m12'],
'sql' => "char(1) NOT NULL default ''", 'sql' => "char(1) NOT NULL default ''",
], ],
@@ -203,8 +206,34 @@ $GLOBALS['TL_DCA']['tl_pinnwand']['fields']['bild']['save_callback'][] = static
} }
if ('image' === $contentType && empty($value)) { if ('image' === $contentType && empty($value)) {
throw new \RuntimeException($GLOBALS['TL_LANG']['ERR']['mandatory'] ?? 'Dieses Feld darf nicht leer sein.'); throw new \RuntimeException($GLOBALS['TL_LANG']['tl_pinnwand']['imageRequired'] ?? 'Bitte wählen Sie ein Bild für den Pinnwandeintrag aus.');
} }
return $value; return $value;
}; };
$GLOBALS['TL_DCA']['tl_pinnwand']['list']['label']['label_callback'] = static function (array $row, string $label, DataContainer $dc, array $args): array {
$args[0] = '';
if (empty($row['bild'])) {
return $args;
}
$fileModel = FilesModel::findByUuid($row['bild']);
if (null === $fileModel) {
return $args;
}
$staticUrl = (string) System::getContainer()->get('contao.assets.files_context')->getStaticUrl();
$src = $staticUrl . System::urlEncode($fileModel->path);
$alt = htmlspecialchars((string) ($row['ueberschrift'] ?: 'Pinnwandbild'), \ENT_QUOTES | \ENT_SUBSTITUTE, 'UTF-8');
$args[0] = sprintf(
'<img src="%s" alt="%s" style="max-width:64px;max-height:64px;width:auto;height:auto;display:block;" loading="lazy">',
$src,
$alt
);
return $args;
};
+2 -1
View File
@@ -9,10 +9,11 @@ $GLOBALS['TL_LANG']['tl_pinnwand']['bild'] = ['Bild', 'Optionales Bild für den
$GLOBALS['TL_LANG']['tl_pinnwand']['contentType'] = ['Inhaltstyp', 'Wählen Sie, ob der Eintrag nur aus einem Bild oder aus Text mit optionalem Bild besteht.']; $GLOBALS['TL_LANG']['tl_pinnwand']['contentType'] = ['Inhaltstyp', 'Wählen Sie, ob der Eintrag nur aus einem Bild oder aus Text mit optionalem Bild besteht.'];
$GLOBALS['TL_LANG']['tl_pinnwand']['contentType_options']['image'] = 'Bild'; $GLOBALS['TL_LANG']['tl_pinnwand']['contentType_options']['image'] = 'Bild';
$GLOBALS['TL_LANG']['tl_pinnwand']['contentType_options']['textImage'] = 'Text mit optionalem Bild'; $GLOBALS['TL_LANG']['tl_pinnwand']['contentType_options']['textImage'] = 'Text mit optionalem Bild';
$GLOBALS['TL_LANG']['tl_pinnwand']['imageRequired'] = 'Bitte wählen Sie ein Bild für den Pinnwandeintrag aus.';
$GLOBALS['TL_LANG']['tl_pinnwand']['dateAdded'] = ['Erstellt am', 'Datum/Uhrzeit der Erstellung.']; $GLOBALS['TL_LANG']['tl_pinnwand']['dateAdded'] = ['Erstellt am', 'Datum/Uhrzeit der Erstellung.'];
$GLOBALS['TL_LANG']['tl_pinnwand']['dateModified'] = ['Geändert am', 'Datum/Uhrzeit der letzten Änderung.']; $GLOBALS['TL_LANG']['tl_pinnwand']['dateModified'] = ['Geändert am', 'Datum/Uhrzeit der letzten Änderung.'];
$GLOBALS['TL_LANG']['tl_pinnwand']['published'] = ['Veröffentlicht', 'Nur veröffentlichte Einträge erscheinen im Frontend.']; $GLOBALS['TL_LANG']['tl_pinnwand']['published'] = ['Veröffentlicht', 'Nur veröffentlichte Einträge erscheinen im Frontend.'];
$GLOBALS['TL_LANG']['tl_pinnwand']['hervorgehoben'] = ['Hervorgehoben', 'Hervorgehobene Einträge liegen im Frontend immer oben.']; $GLOBALS['TL_LANG']['tl_pinnwand']['hervorgehoben'] = ['Hervorgehoben', 'Hervorgehobene Einträge hängen an der Pinnwand immer ganz oben.'];
$GLOBALS['TL_LANG']['tl_pinnwand']['title_legend'] = 'Inhalt'; $GLOBALS['TL_LANG']['tl_pinnwand']['title_legend'] = 'Inhalt';
$GLOBALS['TL_LANG']['tl_pinnwand']['meta_legend'] = 'Metadaten'; $GLOBALS['TL_LANG']['tl_pinnwand']['meta_legend'] = 'Metadaten';
+2 -1
View File
@@ -5,8 +5,9 @@
<section class="pinboard" data-pinboard> <section class="pinboard" data-pinboard>
<div class="pinboard__surface" data-pinboard-surface> <div class="pinboard__surface" data-pinboard-surface>
{% for entry in entries %} {% for entry in entries %}
{% set isImageOnly = entry.imageFigure and not entry.headline and not entry.text and not entry.link %}
<article <article
class="pin-note{% if entry.highlighted %} is-highlighted{% endif %}" class="pin-note{% if entry.highlighted %} is-highlighted{% endif %}{% if isImageOnly %} pin-note--image-only{% endif %}"
data-pin-note data-pin-note
data-highlighted="{{ entry.highlighted ? '1' : '0' }}" data-highlighted="{{ entry.highlighted ? '1' : '0' }}"
data-seed="{{ entry.id }}" data-seed="{{ entry.id }}"
+25
View File
@@ -56,6 +56,7 @@
width: 18px; width: 18px;
height: 18px; height: 18px;
transform: translateX(-50%); transform: translateX(-50%);
z-index: 1;
border-radius: 50%; border-radius: 50%;
background: radial-gradient(circle at 35% 35%, #f2f2f2, #979797 68%, #707070 100%); background: radial-gradient(circle at 35% 35%, #f2f2f2, #979797 68%, #707070 100%);
box-shadow: 0 2px 4px rgba(0, 0, 0, 0.35); box-shadow: 0 2px 4px rgba(0, 0, 0, 0.35);
@@ -73,6 +74,26 @@
margin: 0 0 0.7rem; margin: 0 0 0.7rem;
} }
.pin-note--image-only .pin-note__inner {
min-height: 0;
padding: 0;
background: transparent;
box-shadow: none;
}
.pin-note--image-only .pin-note__image-wrap {
margin: 0;
}
.pin-note--image-only .pin-note__image-wrap img,
.pin-note--image-only .pin-note__image {
display: block;
width: 100%;
height: auto;
border-radius: 0.3rem;
box-shadow: 0 10px 22px rgba(20, 10, 5, 0.34);
}
.pin-note__image { .pin-note__image {
display: block; display: block;
width: 100%; width: 100%;
@@ -80,6 +101,10 @@
border-radius: 0.25rem; border-radius: 0.25rem;
} }
figure.pin-note__image-wrap a:hover img {
transform: none;
}
.pin-note__headline { .pin-note__headline {
margin: 0 0 0.5rem; margin: 0 0 0.5rem;
font-size: 1.1rem; font-size: 1.1rem;