Compare commits
10 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
7c12c4ffc1 | ||
|
|
42a816d347 | ||
|
|
ab5e9b30b5 | ||
|
|
c863ab98f0 | ||
|
|
80dd19257c | ||
|
|
980a6d83dd | ||
|
|
4ef0608d64 | ||
|
|
e3f842f2d3 | ||
|
|
8848fdcbc3 | ||
|
|
f8a5d9348e |
@@ -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';
|
||||||
|
|||||||
@@ -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
5
contao/dca/tl_module.php
Normal 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';
|
||||||
@@ -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&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();
|
||||||
|
};
|
||||||
|
|||||||
@@ -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'];
|
||||||
|
|||||||
0
contao/templates/.twig-root
Normal file
0
contao/templates/.twig-root
Normal file
39
contao/templates/frontend/pinnwand.html.twig
Normal file
39
contao/templates/frontend/pinnwand.html.twig
Normal file
@@ -0,0 +1,39 @@
|
|||||||
|
{% 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 }}"
|
||||||
|
>
|
||||||
|
<div class="pin-note__inner" data-pin-note-inner>
|
||||||
|
{% 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 %}
|
||||||
|
</div>
|
||||||
|
</article>
|
||||||
|
{% else %}
|
||||||
|
<p class="pinboard__empty">Keine veröffentlichten Pinnwandeinträge vorhanden.</p>
|
||||||
|
{% endfor %}
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<script src="https://cdn.jsdelivr.net/npm/muuri@0.9.5/dist/muuri.min.js" defer></script>
|
||||||
|
<script src="{{ asset('bundles/contaopinboard/assets/pinboard.js') }}" defer></script>
|
||||||
@@ -2,38 +2,53 @@
|
|||||||
width: 100%;
|
width: 100%;
|
||||||
max-width: 1400px;
|
max-width: 1400px;
|
||||||
margin: 0 auto;
|
margin: 0 auto;
|
||||||
padding: 1.5rem;
|
padding: 2em;
|
||||||
|
box-sizing: border-box;
|
||||||
|
}
|
||||||
|
|
||||||
|
.pinboard {
|
||||||
|
--pin-gap: 1.25rem;
|
||||||
}
|
}
|
||||||
|
|
||||||
.pinboard__surface {
|
.pinboard__surface {
|
||||||
position: relative;
|
position: relative;
|
||||||
min-height: 42rem;
|
min-height: 42rem;
|
||||||
|
padding: 1.2rem;
|
||||||
|
border: 30px solid #cba888;
|
||||||
border-radius: 1.2rem;
|
border-radius: 1.2rem;
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
background:
|
background:
|
||||||
radial-gradient(circle at 20% 10%, rgba(255, 255, 255, 0.25), transparent 42%),
|
radial-gradient(circle at 20% 10%, rgba(255, 255, 255, 0.18), transparent 42%),
|
||||||
radial-gradient(circle at 80% 90%, rgba(50, 30, 10, 0.2), transparent 36%),
|
radial-gradient(circle at 80% 90%, rgba(40, 22, 10, 0.16), 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),
|
url('/bundles/contaopinboard/assets/wood_0011_color_1k.jpg') center/cover no-repeat;
|
||||||
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);
|
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 {
|
.pin-note {
|
||||||
position: absolute;
|
position: absolute;
|
||||||
width: clamp(220px, 28vw, 340px);
|
width: 320px;
|
||||||
|
min-height: 230px;
|
||||||
|
padding: 1.5em;
|
||||||
|
box-sizing: border-box;
|
||||||
|
overflow: visible;
|
||||||
|
}
|
||||||
|
|
||||||
|
.pin-note__inner {
|
||||||
|
position: relative;
|
||||||
min-height: 190px;
|
min-height: 190px;
|
||||||
padding: 1rem 1rem 1.25rem;
|
padding: 1rem 1rem 1.25rem;
|
||||||
border-radius: 0.3rem;
|
border-radius: 0.3rem;
|
||||||
background: linear-gradient(160deg, #fff8a8 0%, #f5eb85 100%);
|
background: linear-gradient(160deg, #fff8a8 0%, #f5eb85 100%);
|
||||||
color: #2a241c;
|
color: #2a241c;
|
||||||
box-shadow: 0 10px 22px rgba(20, 10, 5, 0.34);
|
box-shadow: 0 10px 22px rgba(20, 10, 5, 0.34);
|
||||||
cursor: grab;
|
cursor: pointer;
|
||||||
user-select: none;
|
user-select: none;
|
||||||
touch-action: none;
|
touch-action: auto;
|
||||||
transition: box-shadow 160ms ease, transform 160ms ease;
|
transform: translate(0, 0) rotate(0deg);
|
||||||
|
transition: box-shadow 160ms ease;
|
||||||
}
|
}
|
||||||
|
|
||||||
.pin-note::before {
|
.pin-note__inner::before {
|
||||||
content: '';
|
content: '';
|
||||||
position: absolute;
|
position: absolute;
|
||||||
top: -9px;
|
top: -9px;
|
||||||
@@ -46,12 +61,11 @@
|
|||||||
box-shadow: 0 2px 4px rgba(0, 0, 0, 0.35);
|
box-shadow: 0 2px 4px rgba(0, 0, 0, 0.35);
|
||||||
}
|
}
|
||||||
|
|
||||||
.pin-note.is-highlighted {
|
.pin-note.is-highlighted .pin-note__inner {
|
||||||
background: linear-gradient(160deg, #ffd99b 0%, #f7c46f 100%);
|
background: linear-gradient(160deg, #ffd99b 0%, #f7c46f 100%);
|
||||||
}
|
}
|
||||||
|
|
||||||
.pin-note.is-dragging {
|
.pin-note.is-front .pin-note__inner {
|
||||||
cursor: grabbing;
|
|
||||||
box-shadow: 0 18px 34px rgba(20, 10, 5, 0.46);
|
box-shadow: 0 18px 34px rgba(20, 10, 5, 0.46);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -97,14 +111,10 @@
|
|||||||
|
|
||||||
@media (max-width: 768px) {
|
@media (max-width: 768px) {
|
||||||
.pinboard {
|
.pinboard {
|
||||||
padding: 0.8rem;
|
padding: 2em;
|
||||||
}
|
}
|
||||||
|
|
||||||
.pinboard__surface {
|
.pinboard__surface {
|
||||||
min-height: 36rem;
|
min-height: 36rem;
|
||||||
}
|
}
|
||||||
|
|
||||||
.pin-note {
|
|
||||||
width: min(84vw, 290px);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
87
public/assets/pinboard.js
Normal file
87
public/assets/pinboard.js
Normal file
@@ -0,0 +1,87 @@
|
|||||||
|
(() => {
|
||||||
|
const board = document.querySelector('[data-pinboard-surface]');
|
||||||
|
|
||||||
|
if (!board) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (typeof Muuri === 'undefined') {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const notes = Array.from(board.querySelectorAll('[data-pin-note]'));
|
||||||
|
|
||||||
|
if (!notes.length) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
let zCounter = 500;
|
||||||
|
|
||||||
|
const highlighted = notes.filter((note) => note.dataset.highlighted === '1');
|
||||||
|
const normal = notes.filter((note) => note.dataset.highlighted !== '1');
|
||||||
|
const ordered = [...highlighted, ...normal];
|
||||||
|
|
||||||
|
ordered.forEach((note) => {
|
||||||
|
board.appendChild(note);
|
||||||
|
});
|
||||||
|
|
||||||
|
ordered.forEach((note, index) => {
|
||||||
|
const isHighlighted = note.dataset.highlighted === '1';
|
||||||
|
note.style.zIndex = String(isHighlighted ? 300 + index : 100 + index);
|
||||||
|
});
|
||||||
|
|
||||||
|
const grid = new Muuri(board, {
|
||||||
|
items: '.pin-note',
|
||||||
|
dragEnabled: false,
|
||||||
|
layout: {
|
||||||
|
fillGaps: false,
|
||||||
|
horizontal: false,
|
||||||
|
alignRight: false,
|
||||||
|
alignBottom: false,
|
||||||
|
rounding: false,
|
||||||
|
},
|
||||||
|
layoutDuration: 250,
|
||||||
|
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);
|
||||||
|
|
||||||
|
ordered.forEach((item) => item.classList.remove('is-front'));
|
||||||
|
note.classList.add('is-front');
|
||||||
|
};
|
||||||
|
|
||||||
|
ordered.forEach((note) => {
|
||||||
|
note.addEventListener('pointerdown', () => {
|
||||||
|
bringToFront(note);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
const relayout = () => {
|
||||||
|
updateItemWidths();
|
||||||
|
grid.refreshItems().layout();
|
||||||
|
};
|
||||||
|
|
||||||
|
relayout();
|
||||||
|
window.addEventListener('resize', relayout);
|
||||||
|
})();
|
||||||
BIN
public/assets/wood_0011_color_1k.jpg
Normal file
BIN
public/assets/wood_0011_color_1k.jpg
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 225 KiB |
@@ -5,16 +5,25 @@ 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\InsertTag\InsertTagParser;
|
||||||
|
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,
|
||||||
|
private readonly InsertTagParser $insertTagParser,
|
||||||
|
) {
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function getResponse(FragmentTemplate $template, ModuleModel $model, Request $request): Response
|
||||||
{
|
{
|
||||||
$collection = PinboardModel::findBy(
|
$collection = PinboardModel::findBy(
|
||||||
['published = ?'],
|
['published = ?'],
|
||||||
@@ -26,29 +35,56 @@ 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[] = [
|
||||||
'id' => (int) $entry->id,
|
'id' => (int) $entry->id,
|
||||||
'headline' => (string) $entry->ueberschrift,
|
'headline' => (string) $entry->ueberschrift,
|
||||||
'text' => (string) $entry->text,
|
'text' => (string) $entry->text,
|
||||||
'link' => (string) $entry->link,
|
'link' => $this->resolveLink((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();
|
||||||
|
}
|
||||||
|
|
||||||
|
private function resolveLink(string $link): string
|
||||||
|
{
|
||||||
|
$link = trim($link);
|
||||||
|
|
||||||
|
if ('' === $link) {
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
|
||||||
|
return html_entity_decode($this->insertTagParser->replaceInline($link), \ENT_QUOTES | \ENT_HTML5);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,121 +0,0 @@
|
|||||||
(() => {
|
|
||||||
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) => {
|
|
||||||
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();
|
|
||||||
});
|
|
||||||
})();
|
|
||||||
@@ -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 %}
|
|
||||||
Reference in New Issue
Block a user