Initial: Webfarben DummyCopier Bundle

This commit is contained in:
2026-03-11 20:01:53 +01:00
commit 80fd71b922
12 changed files with 1071 additions and 0 deletions

View File

@@ -0,0 +1,380 @@
<?php
declare(strict_types=1);
namespace Webfarben\DummyCopier\Backend;
use Webfarben\DummyCopier\Service\DummyCopier;
use Webfarben\DummyCopier\Service\DummyCopyOptions;
use Contao\BackendModule;
use Contao\Environment;
use Contao\FileTree;
use Contao\Input;
use Contao\Message;
use Contao\PageTree;
use Contao\StringUtil;
use Contao\System;
use Contao\Widget;
use Doctrine\DBAL\Connection;
class DummyCopierModule extends BackendModule
{
protected $strTemplate = 'be_dummy_copier';
protected function compile(): void
{
$connection = System::getContainer()->get(Connection::class);
$this->Template->action = Environment::get('request');
$this->Template->requestToken = \defined('REQUEST_TOKEN') ? REQUEST_TOKEN : '';
$this->Template->pageChoices = $this->getPageChoices($connection);
$this->Template->moduleChoices = $this->getModuleChoices($connection);
$this->Template->contentChoices = $this->getContentChoices($connection);
$this->Template->directoryChoices = $this->getDirectoryChoices();
$this->Template->sourcePagesWidget = '';
$this->Template->targetParentPageWidget = '';
$this->Template->sourceDirectoriesWidget = '';
$targetParentPageId = $this->parseSingleIdInput(Input::postRaw('targetParentPage'));
$this->Template->selected = [
'sourcePages' => $this->parseIdInput(Input::postRaw('sourcePages')),
'sourceModules' => $this->parseIdInput(Input::postRaw('sourceModules')),
'sourceContent' => $this->parseIdInput(Input::postRaw('sourceContent')),
'sourceDirectories' => $this->parsePathInput(Input::postRaw('sourceDirectories')),
'targetParentPage' => $targetParentPageId,
'targetArticle' => (int) Input::post('targetArticle'),
'targetDirectory' => trim((string) Input::post('targetDirectory')),
'namePrefix' => trim((string) Input::post('namePrefix')),
];
$this->prepareTreeWidgets();
if (Input::post('FORM_SUBMIT') !== 'tl_dummy_copier') {
return;
}
$options = new DummyCopyOptions(
$this->parseIdInput(Input::postRaw('sourcePages')),
$this->parseIdInput(Input::postRaw('sourceModules')),
$this->parseIdInput(Input::postRaw('sourceContent')),
$this->parsePathInput(Input::postRaw('sourceDirectories')),
$targetParentPageId,
(int) Input::post('targetArticle'),
trim((string) Input::post('targetDirectory')),
trim((string) Input::post('namePrefix')),
(bool) Input::post('includeContent'),
(bool) Input::post('copyModules'),
(bool) Input::post('copyDirectories'),
(bool) Input::post('dryRun')
);
try {
/** @var DummyCopier $copier */
$copier = System::getContainer()->get(DummyCopier::class);
$result = $copier->execute($options);
Message::addConfirmation(sprintf(
'Fertig. Seiten: %d, Module: %d, Content: %d, Verzeichnisse: %d',
$result->copiedPages,
$result->copiedModules,
$result->copiedContent,
$result->copiedDirectories
));
$this->Template->result = $result;
} catch (\Throwable $exception) {
Message::addError($exception->getMessage());
}
}
private function parseIdInput($input): array
{
if (\is_array($input)) {
return array_values(array_filter(array_map('intval', $input), static fn (int $id): bool => $id > 0));
}
$csv = trim((string) $input);
if ($csv === '') {
return [];
}
$deserialized = StringUtil::deserialize($csv, true);
if ($deserialized !== [] && $deserialized !== [$csv]) {
return array_values(array_filter(array_map('intval', $deserialized), static fn (int $id): bool => $id > 0));
}
$parts = array_filter(array_map('trim', explode(',', $csv)), static fn (string $value): bool => $value !== '');
return array_values(array_filter(array_map('intval', $parts), static fn (int $id): bool => $id > 0));
}
private function parsePathInput($input): array
{
if (\is_array($input)) {
return array_values(array_filter(array_map('trim', $input), static fn (string $value): bool => $value !== ''));
}
$csv = trim((string) $input);
if ($csv === '') {
return [];
}
$deserialized = StringUtil::deserialize($csv, true);
if ($deserialized !== [] && $deserialized !== [$csv]) {
return array_values(array_filter(array_map('trim', $deserialized), static fn (string $value): bool => $value !== ''));
}
return array_values(array_filter(array_map('trim', explode(',', $csv)), static fn (string $value): bool => $value !== ''));
}
private function parseSingleIdInput($input): int
{
$ids = $this->parseIdInput($input);
return $ids[0] ?? 0;
}
private function prepareTreeWidgets(): void
{
if (!class_exists(PageTree::class) || !class_exists(FileTree::class) || !class_exists(Widget::class)) {
return;
}
try {
$selectedPages = $this->parseIdInput(Input::postRaw('sourcePages'));
$selectedParent = (int) Input::post('targetParentPage');
$selectedDirectories = $this->parsePathInput(Input::postRaw('sourceDirectories'));
$this->Template->sourcePagesWidget = $this->renderPageTreeWidget(
'sourcePages',
'Quell-Seiten (pageTree)',
$selectedPages,
true
);
$this->Template->targetParentPageWidget = $this->renderPageTreeWidget(
'targetParentPage',
'Ziel-Elternseite (pageTree)',
$selectedParent > 0 ? [$selectedParent] : [],
false
);
$this->Template->sourceDirectoriesWidget = $this->renderFileTreeWidget(
'sourceDirectories',
'Quell-Verzeichnisse (fileTree)',
$selectedDirectories
);
} catch (\Throwable $exception) {
// If widget rendering differs by Contao version, the module falls back to select boxes.
$this->Template->sourcePagesWidget = '';
$this->Template->targetParentPageWidget = '';
$this->Template->sourceDirectoriesWidget = '';
Message::addInfo('Tree-Widgets konnten nicht initialisiert werden, Fallback-Auswahl wird verwendet.');
}
}
private function renderPageTreeWidget(string $name, string $label, array $value, bool $multiple): string
{
$attributes = Widget::getAttributesFromDca([
'inputType' => 'pageTree',
'label' => [$label, ''],
'eval' => [
'fieldType' => $multiple ? 'checkbox' : 'radio',
'multiple' => $multiple,
'tl_class' => 'clr',
],
], $name, $value, $name, 'tl_dummy_copier');
$attributes['id'] = $name;
$attributes['name'] = $name;
$widget = new PageTree($attributes);
return $widget->generate();
}
private function renderFileTreeWidget(string $name, string $label, array $value): string
{
$attributes = Widget::getAttributesFromDca([
'inputType' => 'fileTree',
'label' => [$label, ''],
'eval' => [
'fieldType' => 'checkbox',
'filesOnly' => false,
'files' => false,
'multiple' => true,
'tl_class' => 'clr',
],
], $name, $value, $name, 'tl_dummy_copier');
$attributes['id'] = $name;
$attributes['name'] = $name;
$widget = new FileTree($attributes);
return $widget->generate();
}
/**
* @return array<int,string>
*/
private function getPageChoices(Connection $connection): array
{
$rows = $connection->fetchAllAssociative('SELECT id, pid, title, alias FROM tl_page ORDER BY sorting, id');
$rowsByParent = [];
foreach ($rows as $row) {
$pid = (int) ($row['pid'] ?? 0);
$rowsByParent[$pid][] = $row;
}
$choices = [];
$build = function (int $pid, int $depth) use (&$build, &$choices, $rowsByParent): void {
foreach ($rowsByParent[$pid] ?? [] as $row) {
$id = (int) ($row['id'] ?? 0);
if ($id < 1) {
continue;
}
$title = trim((string) ($row['title'] ?? ''));
$alias = trim((string) ($row['alias'] ?? ''));
$label = $title !== '' ? $title : 'Seite ' . $id;
if ($alias !== '') {
$label .= ' (' . $alias . ')';
}
$indent = str_repeat(' ', max(0, $depth));
$choices[$id] = sprintf('%s%s [ID %d]', $indent, $label, $id);
$build($id, $depth + 1);
}
};
$build(0, 0);
// Fallback for non-rooted records that were not visited from pid=0.
foreach ($rows as $row) {
$id = (int) ($row['id'] ?? 0);
if ($id < 1 || isset($choices[$id])) {
continue;
}
$title = trim((string) ($row['title'] ?? ''));
$alias = trim((string) ($row['alias'] ?? ''));
$label = $title !== '' ? $title : 'Seite ' . $id;
if ($alias !== '') {
$label .= ' (' . $alias . ')';
}
$choices[$id] = sprintf('%s [ID %d]', $label, $id);
}
return $choices;
}
/**
* @return array<int,string>
*/
private function getModuleChoices(Connection $connection): array
{
$rows = $connection->fetchAllAssociative('SELECT id, name, type FROM tl_module ORDER BY id');
$choices = [];
foreach ($rows as $row) {
$id = (int) ($row['id'] ?? 0);
if ($id < 1) {
continue;
}
$name = trim((string) ($row['name'] ?? 'Modul ' . $id));
$type = trim((string) ($row['type'] ?? ''));
$label = $type !== '' ? sprintf('%s (%s)', $name, $type) : $name;
$choices[$id] = sprintf('%s [ID %d]', $label, $id);
}
return $choices;
}
/**
* @return array<int,string>
*/
private function getContentChoices(Connection $connection): array
{
$rows = $connection->fetchAllAssociative('SELECT id, type, pid, headline FROM tl_content ORDER BY id');
$choices = [];
foreach ($rows as $row) {
$id = (int) ($row['id'] ?? 0);
if ($id < 1) {
continue;
}
$type = trim((string) ($row['type'] ?? 'content'));
$pid = (int) ($row['pid'] ?? 0);
$headline = $this->normalizeHeadline($row['headline'] ?? null);
$label = $headline !== '' ? sprintf('%s: %s', $type, $headline) : $type;
$choices[$id] = sprintf('%s [ID %d, Artikel %d]', $label, $id, $pid);
}
return $choices;
}
/**
* @return array<string,string>
*/
private function getDirectoryChoices(): array
{
$projectDir = (string) System::getContainer()->getParameter('kernel.project_dir');
$filesDir = $projectDir . '/files';
if (!is_dir($filesDir)) {
return [];
}
$choices = [];
$iterator = new \RecursiveIteratorIterator(
new \RecursiveDirectoryIterator($filesDir, \FilesystemIterator::SKIP_DOTS),
\RecursiveIteratorIterator::SELF_FIRST
);
foreach ($iterator as $item) {
if (!$item->isDir()) {
continue;
}
$fullPath = $item->getPathname();
$relative = str_replace($projectDir . '/', '', $fullPath);
$trimmed = trim((string) str_replace('files/', '', $relative), '/');
$depth = $trimmed === '' ? 0 : substr_count($trimmed, '/');
$indent = str_repeat(' ', max(0, $depth));
$choices[$relative] = $indent . $relative;
}
ksort($choices);
return $choices;
}
private function normalizeHeadline($headline): string
{
if (\is_string($headline)) {
return trim($headline);
}
if (!\is_array($headline) || !isset($headline['value']) || !\is_string($headline['value'])) {
return '';
}
return trim($headline['value']);
}
}