Compare commits
3 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
80dd19257c | ||
|
|
980a6d83dd | ||
|
|
4ef0608d64 |
@@ -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';
|
||||||
|
|||||||
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';
|
||||||
@@ -102,14 +102,16 @@ $GLOBALS['TL_DCA']['tl_pinnwand'] = [
|
|||||||
'sorting' => true,
|
'sorting' => true,
|
||||||
'flag' => 6,
|
'flag' => 6,
|
||||||
'inputType' => 'text',
|
'inputType' => 'text',
|
||||||
'eval' => ['rgxp' => 'datim', 'datepicker' => true, 'mandatory' => true, 'default' => time(), 'tl_class' => 'w50 wizard'],
|
'default' => time(),
|
||||||
|
'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',
|
||||||
],
|
],
|
||||||
'dateModified' => [
|
'dateModified' => [
|
||||||
'sorting' => true,
|
'sorting' => true,
|
||||||
'flag' => 6,
|
'flag' => 6,
|
||||||
'inputType' => 'text',
|
'inputType' => 'text',
|
||||||
'eval' => ['rgxp' => 'datim', 'readonly' => true, 'default' => time(), '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' => [
|
||||||
@@ -145,14 +147,14 @@ $GLOBALS['TL_DCA']['tl_pinnwand']['fields']['ueberschrift']['save_callback'][] =
|
|||||||
$start = 2;
|
$start = 2;
|
||||||
$base = $baseHeadline;
|
$base = $baseHeadline;
|
||||||
|
|
||||||
if (preg_match('/^(.*)\s-\s(\d+)$/', $baseHeadline, $matches)) {
|
if (preg_match('/^(.*)\s\((\d+)\)$/', $baseHeadline, $matches)) {
|
||||||
$base = trim($matches[1]);
|
$base = trim($matches[1]);
|
||||||
$start = (int) $matches[2] + 1;
|
$start = (int) $matches[2] + 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
$id = (int) ($dataContainer->id ?? 0);
|
$id = (int) ($dataContainer->id ?? 0);
|
||||||
$number = max(2, $start);
|
$number = max(2, $start);
|
||||||
$candidate = sprintf('%s - %d', $base, $number);
|
$candidate = sprintf('%s (%d)', $base, $number);
|
||||||
|
|
||||||
do {
|
do {
|
||||||
$exists = Database::getInstance()
|
$exists = Database::getInstance()
|
||||||
@@ -164,6 +166,18 @@ $GLOBALS['TL_DCA']['tl_pinnwand']['fields']['ueberschrift']['save_callback'][] =
|
|||||||
}
|
}
|
||||||
|
|
||||||
++$number;
|
++$number;
|
||||||
$candidate = sprintf('%s - %d', $base, $number);
|
$candidate = sprintf('%s (%d)', $base, $number);
|
||||||
} while (true);
|
} 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();
|
||||||
|
};
|
||||||
|
|||||||
0
contao/templates/.twig-root
Normal file
0
contao/templates/.twig-root
Normal file
@@ -1,4 +1,6 @@
|
|||||||
<link rel="stylesheet" href="{{ asset('bundles/contaopinboard/css/pinboard.css') }}">
|
{% use '@Contao/component/_figure.html.twig' %}
|
||||||
|
|
||||||
|
<link rel="stylesheet" href="{{ asset('bundles/contaopinboard/assets/pinboard.css') }}">
|
||||||
|
|
||||||
<section class="pinboard" data-pinboard>
|
<section class="pinboard" data-pinboard>
|
||||||
<div class="pinboard__surface" data-pinboard-surface>
|
<div class="pinboard__surface" data-pinboard-surface>
|
||||||
@@ -10,9 +12,9 @@
|
|||||||
data-seed="{{ entry.id }}"
|
data-seed="{{ entry.id }}"
|
||||||
aria-label="{{ entry.headline }}"
|
aria-label="{{ entry.headline }}"
|
||||||
>
|
>
|
||||||
{% if entry.imagePath %}
|
{% if entry.imageFigure %}
|
||||||
<figure class="pin-note__image-wrap">
|
<figure class="pin-note__image-wrap">
|
||||||
<img src="{{ asset(entry.imagePath) }}" alt="" class="pin-note__image">
|
{% with {figure: entry.imageFigure} %}{{ block('figure_component') }}{% endwith %}
|
||||||
</figure>
|
</figure>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
|
|
||||||
@@ -31,4 +33,4 @@
|
|||||||
</div>
|
</div>
|
||||||
</section>
|
</section>
|
||||||
|
|
||||||
<script src="{{ asset('bundles/contaopinboard/js/pinboard.js') }}" defer></script>
|
<script src="{{ asset('bundles/contaopinboard/assets/pinboard.js') }}" defer></script>
|
||||||
|
|||||||
110
src/Resources/public/assets/pinboard.css
Normal file
110
src/Resources/public/assets/pinboard.css
Normal 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);
|
||||||
|
}
|
||||||
|
}
|
||||||
125
src/Resources/public/assets/pinboard.js
Normal file
125
src/Resources/public/assets/pinboard.js
Normal 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();
|
||||||
|
});
|
||||||
|
})();
|
||||||
Reference in New Issue
Block a user