Compare commits

..

5 Commits

Author SHA1 Message Date
Jürgen Mummert b46e4a563c chore: rename package to mummert/pinboard-bundle 2026-04-01 12:48:53 +02:00
Jürgen Mummert 605c637fcb feat: add pinboard content type modes in backend 2026-04-01 12:42:43 +02:00
Jürgen Mummert c995c4f93c Decode entities in pinboard text output 2026-03-04 18:58:37 +01:00
Jürgen Mummert 7c12c4ffc1 Add wood texture asset for pinboard background 2026-03-04 18:53:49 +01:00
Jürgen Mummert 42a816d347 Tune pinboard styling and Muuri-driven dynamic item widths 2026-03-04 18:50:49 +01:00
8 changed files with 69 additions and 14 deletions
+1 -1
View File
@@ -1,5 +1,5 @@
{
"name": "eiswurm/contao-pinboard-bundle",
"name": "mummert/pinboard-bundle",
"description": "Pinboard bundle for Contao 5.7",
"type": "contao-bundle",
"license": "proprietary",
+28 -1
View File
@@ -61,7 +61,12 @@ $GLOBALS['TL_DCA']['tl_pinnwand'] = [
],
],
'palettes' => [
'default' => '{title_legend},ueberschrift,text,link,bild;{meta_legend},dateAdded,dateModified;{publish_legend},published,hervorgehoben',
'__selector__' => ['contentType'],
'default' => '{title_legend},contentType;{meta_legend},dateAdded,dateModified;{publish_legend},published,hervorgehoben',
],
'subpalettes' => [
'contentType_image' => 'bild',
'contentType_textImage' => 'ueberschrift,text,link,bild',
],
'fields' => [
'id' => [
@@ -70,6 +75,14 @@ $GLOBALS['TL_DCA']['tl_pinnwand'] = [
'tstamp' => [
'sql' => 'int(10) unsigned NOT NULL default 0',
],
'contentType' => [
'exclude' => true,
'inputType' => 'radio',
'options' => ['image', 'textImage'],
'reference' => &$GLOBALS['TL_LANG']['tl_pinnwand']['contentType_options'],
'eval' => ['mandatory' => true, 'submitOnChange' => true, 'tl_class' => 'w50 clr'],
'sql' => "varchar(16) NOT NULL default 'textImage'",
],
'ueberschrift' => [
'exclude' => true,
'search' => true,
@@ -181,3 +194,17 @@ $GLOBALS['TL_DCA']['tl_pinnwand']['fields']['dateModified']['load_callback'][] =
return $timestamp > 0 ? $timestamp : time();
};
$GLOBALS['TL_DCA']['tl_pinnwand']['fields']['bild']['save_callback'][] = static function (mixed $value, DataContainer $dataContainer): mixed {
$contentType = (string) Input::post('contentType');
if ('' === $contentType && null !== $dataContainer->activeRecord) {
$contentType = (string) ($dataContainer->activeRecord->contentType ?? 'textImage');
}
if ('image' === $contentType && empty($value)) {
throw new \RuntimeException($GLOBALS['TL_LANG']['ERR']['mandatory'] ?? 'Dieses Feld darf nicht leer sein.');
}
return $value;
};
+3
View File
@@ -6,6 +6,9 @@ $GLOBALS['TL_LANG']['tl_pinnwand']['ueberschrift'] = ['Überschrift', 'Die Über
$GLOBALS['TL_LANG']['tl_pinnwand']['text'] = ['Text', 'Inhalt des Pinnwandeintrags (maximal 3000 Zeichen).'];
$GLOBALS['TL_LANG']['tl_pinnwand']['link'] = ['Link', 'Optionaler Link zum Eintrag.'];
$GLOBALS['TL_LANG']['tl_pinnwand']['bild'] = ['Bild', 'Optionales Bild für den Pinnwandeintrag.'];
$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']['textImage'] = 'Text mit optionalem Bild';
$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']['published'] = ['Veröffentlicht', 'Nur veröffentlichte Einträge erscheinen im Frontend.'];
+7 -2
View File
@@ -19,8 +19,13 @@
</figure>
{% endif %}
<h3 class="pin-note__headline">{{ entry.headline }}</h3>
<div class="pin-note__text">{{ entry.text|e|nl2br }}</div>
{% if entry.headline %}
<h3 class="pin-note__headline">{{ entry.headline }}</h3>
{% endif %}
{% if entry.text %}
<div class="pin-note__text">{{ entry.text|e|nl2br }}</div>
{% endif %}
{% if entry.link %}
<p class="pin-note__link-wrap">
+9 -9
View File
@@ -6,23 +6,27 @@
box-sizing: border-box;
}
.pinboard {
--pin-gap: 1.25rem;
}
.pinboard__surface {
position: relative;
min-height: 42rem;
padding: 1.2rem;
border: 30px solid #cba888;
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%);
radial-gradient(circle at 20% 10%, rgba(255, 255, 255, 0.18), transparent 42%),
radial-gradient(circle at 80% 90%, rgba(40, 22, 10, 0.16), transparent 36%),
url('/bundles/contaopinboard/assets/wood_0011_color_1k.jpg') center/cover no-repeat;
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, 26vw, 320px);
width: 320px;
min-height: 230px;
padding: 1.5em;
box-sizing: border-box;
@@ -113,8 +117,4 @@
.pinboard__surface {
min-height: 36rem;
}
.pin-note {
width: min(84vw, 290px);
}
}
+20
View File
@@ -44,6 +44,25 @@
layoutEasing: 'ease',
});
const toPx = (value) => {
const parsed = Number.parseFloat(value);
return Number.isNaN(parsed) ? 0 : parsed;
};
const updateItemWidths = () => {
const boardStyles = window.getComputedStyle(board);
const boardWidth = board.clientWidth;
const gap = toPx(boardStyles.getPropertyValue('--pin-gap')) || 20;
const minItemWidth = 260;
const columns = Math.max(1, Math.floor((boardWidth + gap) / (minItemWidth + gap)));
const itemWidth = Math.max(220, Math.floor((boardWidth - (columns - 1) * gap) / columns));
ordered.forEach((note) => {
note.style.width = `${itemWidth}px`;
});
};
const bringToFront = (note) => {
zCounter += 1;
note.style.zIndex = String(zCounter);
@@ -59,6 +78,7 @@
});
const relayout = () => {
updateItemWidths();
grid.refreshItems().layout();
};
Binary file not shown.

After

Width:  |  Height:  |  Size: 225 KiB

@@ -61,7 +61,7 @@ final class PinboardController extends AbstractFrontendModuleController
$notes[] = [
'id' => (int) $entry->id,
'headline' => (string) $entry->ueberschrift,
'text' => (string) $entry->text,
'text' => StringUtil::decodeEntities((string) $entry->text),
'link' => $this->resolveLink((string) $entry->link),
'dateAdded' => (int) $entry->dateAdded,
'dateModified' => (int) $entry->dateModified,