14 Commits

Author SHA1 Message Date
71196b5c16 UX: Ziel-Elternseite und Zielverzeichnis als Baumauswahl (1.2.3) 2026-03-15 20:33:46 +01:00
847c3aaf36 UX: Seitenbaum-Filter behaelt Eltern bei Treffer in Unterknoten sichtbar 2026-03-15 20:15:34 +01:00
16627b0433 UX: Seitenbaum mit Parent-Child-Auswahl und indeterminate Status 2026-03-15 20:14:44 +01:00
147b163fd5 UX: Seitenauswahl als hierarchischer Seitenbaum im Backend 2026-03-15 20:13:15 +01:00
976d33cd1b Docs: Changelog fuer 1.1.0 bis 1.1.5 ergaenzt 2026-03-15 20:06:20 +01:00
6f279a2cc2 Fix: Array-Serialisierung kompatibel fuer Contao 4.13/5.x 2026-03-15 19:59:03 +01:00
f244d1f52a Fix: SQL ORDER BY ohne nicht vorhandene sorting-Spalten in News/Event-Kopie 2026-03-15 19:31:28 +01:00
2e09acdc00 Chore: Backend-Icon aktualisiert 2026-03-15 18:52:45 +01:00
461c80d75d Chore: Backend-Icon und README aktualisiert 2026-03-15 18:42:03 +01:00
c361999ea8 Chore: composer metadata links auf GitHub umgestellt 2026-03-12 18:10:00 +01:00
0617f19c28 Fix: News-Referenzen und Reader-Module bei Archiv/Kalender-Kopie korrekt umbiegen 2026-03-12 15:50:00 +01:00
30e5a28ce3 Feature: Newsarchive und Kalender inkl. Eintraege kopieren und Referenzen aktualisieren 2026-03-12 15:36:52 +01:00
42cef07833 Chore: Packagist-Metadaten in composer.json ergaenzt (keywords, homepage, support, authors) 2026-03-11 22:43:53 +01:00
c9b29620ab Fix: robuste Copy-Logik fuer verschachtelte Content-Elemente und Alias-Rewrites 2026-03-11 22:23:41 +01:00
10 changed files with 1023 additions and 62 deletions

42
CHANGELOG.md Normal file
View File

@@ -0,0 +1,42 @@
# Changelog
Alle nennenswerten Aenderungen an `webfarben/contao-dummy-copier` werden in dieser Datei dokumentiert.
## [1.1.5] - 2026-03-15
### Fixed
- Kompatibilitaet fuer Contao 4.13 und 5.x verbessert: Array-Serialisierung nutzt jetzt natives PHP `serialize()` statt `StringUtil::serialize()`.
## [1.1.4] - 2026-03-15
### Fixed
- SQL-Fehler in Umgebungen ohne `sorting`-Spalte in `tl_news` bzw. `tl_calendar_events` behoben.
- Sortierung fuer News/Events auf robuste ORDER-BY-Klauseln ohne `sorting` angepasst.
## [1.1.3] - 2026-03-15
### Changed
- Backend-Modul-Icon (`public/icon.svg`) hinzugefuegt/aktualisiert.
- README auf aktuellen Funktionsumfang und Installationsweg ueber Packagist gebracht.
## [1.1.2] - 2026-03-12
### Changed
- Paket-Metadaten (`homepage`, `support.source`, `support.issues`) auf GitHub als kanonische Quelle umgestellt.
## [1.1.1] - 2026-03-12
### Fixed
- Rewiring fuer interne News-Referenzen (`related`) in kopierten News verbessert.
- Reader-Modul-Referenzen in kopierten Modulen korrigiert (`news_readerModule`, `cal_readerModule`).
## [1.1.0] - 2026-03-12
### Added
- Kopieren von Newsarchiven und Newsbeitraegen (`tl_news_archive`, `tl_news`).
- Kopieren von Kalendern und Events (`tl_calendar`, `tl_calendar_events`).
- Auswahlfelder fuer Newsarchive/Kalender im Backend-Modul.
- Ergebnisdaten um Zaehler und Mapping fuer News/Kalender erweitert.
### Changed
- Referenz-Umschreibung fuer kopierte Module um Archivzuordnungen erweitert (`news_archives`, `cal_calendar`).

View File

@@ -1,43 +1,56 @@
# Contao Dummy Copier (Scaffold) # Contao Dummy Copier
Dieses Bundle stellt ein Backend-Modul `Dummy Copier` bereit, um bestehende Dummyseiten, Inhalte, Module und Verzeichnisse zu kopieren und Referenzen automatisiert umzubiegen. Dieses Bundle stellt ein Backend-Modul `Dummy Copier` bereit, um bestehende Dummydaten in Contao kontrolliert zu vervielfaeltigen und interne Referenzen auf die neuen Zielobjekte umzubiegen.
## Enthaltene Funktionen ## Funktionsumfang
- Rekursives Kopieren von Seitenbaeumen (`tl_page`) - rekursives Kopieren von Seitenbaeumen aus `tl_page`
- Optionales Kopieren von Artikeln und Content (`tl_article`, `tl_content`) - optionales Kopieren von Artikeln und verschachtelten Inhaltselementen aus `tl_article` und `tl_content`
- Optionales Kopieren von Modulen (`tl_module`) - optionales Kopieren von Modulen aus `tl_module`
- Automatisches Umstellen von: - optionales Kopieren von Newsarchiven samt Newsbeitraegen aus `tl_news_archive` und `tl_news`
- Content-Elementen vom Typ `module` auf kopierte Modul-IDs - optionales Kopieren von Kalendern samt Events aus `tl_calendar` und `tl_calendar_events`
- `jumpTo` in kopierten Seiten/Modulen/Content auf kopierte Seiten, falls vorhanden - optionales Spiegeln von Verzeichnissen im Dateisystem
- Optionales Kopieren von Verzeichnissen (Dateisystem-Mirror) - Dry-Run zur Vorschau ohne Schreibzugriffe
- Dry-Run Modus ohne Schreibzugriff
## Automatische Referenzanpassungen
- `jumpTo` in kopierten Seiten, Modulen, Content-Elementen, Newsarchiven, News, Kalendern und Events
- Modulreferenzen in Content-Elementen vom Typ `module`
- Alias-Referenzen in verschachtelten Content-Elementen (`cteAlias`)
- Archiv-Zuordnungen in kopierten Modulen (`news_archives`, `cal_calendar`)
- Reader-Module in kopierten Modulen (`news_readerModule`, `cal_readerModule`)
- verwandte News (`related`), sofern die referenzierten News ebenfalls mitkopiert wurden
## Installation ## Installation
1. Bundle in dein Contao-Projekt legen (oder als VCS-Paket einbinden). Installation ueber Packagist:
2. `composer install` oder `composer update acme/contao-dummy-copier`
3. Cache leeren.
4. Backend-Modul `Dummy Copier` unter `System` oeffnen.
## Bedienung (aktueller Stand) ```bash
composer require webfarben/contao-dummy-copier
```
- Quellobjekte werden ueber Mehrfachauswahlfelder ausgewaehlt (Seiten, Module, Content, Verzeichnisse). Danach wie ueblich:
- Seiten und Verzeichnisse werden in Baumdarstellung (Einrueckung nach Hierarchie) angezeigt.
- Alle Mehrfachauswahlfelder haben Live-Filter sowie `Alle`/`Keine` Buttons.
- Ziel-Elternseite wird per Auswahlfeld gesetzt.
Bei kompatibler Contao-Umgebung nutzt das Modul native `pageTree`/`fileTree` Widgets fuer Seiten und Verzeichnisse. ```bash
Falls die Widget-Initialisierung versionsbedingt fehlschlaegt, wird automatisch auf die Select-Fallbacks gewechselt. php vendor/bin/contao-setup
- Setze optional Zielverzeichnis, Zielartikel-ID und Praefix. php vendor/bin/console contao:migrate
- Aktiviere Optionen nach Bedarf (`inkl. Content`, `Module kopieren`, `Verzeichnisse kopieren`, `Dry-Run`). ```
Hinweis: Das Modul akzeptiert weiterhin CSV-Werte als Fallback, falls du Felder per POST automatisiert befuellst. Das Backend-Modul `Dummy Copier` erscheint anschliessend unter `System`.
## Wichtige Hinweise ## Bedienung
- Nach Verzeichnis-Kopien ggf. `contao:filesync` ausfuehren, damit DBAFS konsistent ist. - Quellobjekte werden ueber Mehrfachauswahlfelder ausgewaehlt.
- Dieses Grundgeruest ist bewusst pragmatisch und kann erweitert werden um: - Seiten, Module, Newsarchive, Kalender und Verzeichnisse koennen separat kombiniert werden.
- PageTree/FileTree Picker statt CSV - Alle Mehrfachauswahlfelder besitzen Live-Filter sowie `Alle`/`Keine` Buttons.
- Feldspezifisches Mapping fuer News/Event/Archive-Felder in `tl_module` - Inhaltselemente von Seiten werden bei aktiver Option automatisch mitkopiert.
- Job-Queue via Messenger bei sehr grossen Kopierlaeufen - Ueber ein Praefix lassen sich Titel, Namen und Aliase der Kopien kenntlich machen.
## Hinweise
- Nach Dateikopien ggf. `php vendor/bin/console contao:filesync` ausfuehren, damit die DBAFS-Daten synchronisiert werden.
- Das Bundle ist fuer pragmatische Redaktions- und Setup-Workflows gedacht. Projektspezifische Sonderfelder oder Referenzen koennen bei Bedarf erweitert werden.
## Changelog
- Siehe `CHANGELOG.md` fuer die dokumentierten Aenderungen ab `1.1.0`.

View File

@@ -3,6 +3,24 @@
"description": "Contao backend module to clone dummy pages, content elements, modules and rewire references.", "description": "Contao backend module to clone dummy pages, content elements, modules and rewire references.",
"type": "contao-bundle", "type": "contao-bundle",
"license": "proprietary", "license": "proprietary",
"keywords": [
"contao",
"contao-bundle",
"backend",
"dummy",
"migration",
"cloner"
],
"homepage": "https://github.com/webfarben/DummyCopier",
"support": {
"source": "https://github.com/webfarben/DummyCopier",
"issues": "https://github.com/webfarben/DummyCopier/issues"
},
"authors": [
{
"name": "Webfarben"
}
],
"require": { "require": {
"php": "^8.1", "php": "^8.1",
"contao/core-bundle": "^4.13 || ^5.0", "contao/core-bundle": "^4.13 || ^5.0",

View File

@@ -2,5 +2,5 @@
$GLOBALS['BE_MOD']['system']['dummy_copier'] = [ $GLOBALS['BE_MOD']['system']['dummy_copier'] = [
'callback' => Webfarben\DummyCopier\Backend\DummyCopierModule::class, 'callback' => Webfarben\DummyCopier\Backend\DummyCopierModule::class,
'icon' => 'bundles/acmedummycopier/icon.svg', 'icon' => 'bundles/dummycopier/icon.svg',
]; ];

View File

@@ -11,6 +11,16 @@
.dc-section { border: 1px solid #ccc; padding: 1rem; margin-bottom: 1.5rem; border-radius: 4px; } .dc-section { border: 1px solid #ccc; padding: 1rem; margin-bottom: 1.5rem; border-radius: 4px; }
.dc-section h3 { margin: 0 0 0.75rem; font-size: 1rem; font-weight: bold; } .dc-section h3 { margin: 0 0 0.75rem; font-size: 1rem; font-weight: bold; }
.dc-hint { color: #666; font-size: 0.85rem; margin: 0.25rem 0 0.75rem; } .dc-hint { color: #666; font-size: 0.85rem; margin: 0.25rem 0 0.75rem; }
.dc-page-tree { border: 1px solid #ddd; border-radius: 4px; background: #fff; max-height: 360px; overflow: auto; padding: 0.5rem; }
.dc-page-tree ul { list-style: none; margin: 0.1rem 0 0.1rem 1.1rem; padding: 0; }
.dc-page-tree > ul { margin-left: 0; }
.dc-page-tree li { margin: 0.1rem 0; }
.dc-page-item { display: flex; align-items: center; gap: 0.45rem; }
.dc-page-id { color: #777; font-size: 0.8rem; }
.dc-tree-option { padding: 0.2rem 0 0.4rem; border-bottom: 1px solid #eee; margin-bottom: 0.3rem; }
.dc-dir-item { cursor: pointer; border-radius: 3px; }
.dc-dir-item:hover { background: #f0f4f8; }
.dc-dir-selected { font-weight: bold; color: #0a5a8c; background: #e8f0fe; }
</style> </style>
<!-- Abschnitt 1: Quell-Seiten --> <!-- Abschnitt 1: Quell-Seiten -->
@@ -19,16 +29,48 @@
<p class="dc-hint">Alle Artikel und Inhaltselemente der gewaehlten Seiten werden automatisch mitkopiert (sofern Option "inkl. Content" aktiv ist).</p> <p class="dc-hint">Alle Artikel und Inhaltselemente der gewaehlten Seiten werden automatisch mitkopiert (sofern Option "inkl. Content" aktiv ist).</p>
<p> <p>
<label>Quell-Seiten (Mehrfachauswahl):<br> <label>Quell-Seiten (Mehrfachauswahl):<br>
<input class="dc-filter" type="text" data-filter-for="sourcePages" placeholder="Seiten filtern..."> <input class="dc-filter" type="text" data-filter-for-tree="sourcePages" placeholder="Seiten im Baum filtern...">
<span class="dc-tools"> <span class="dc-tools">
<button class="dc-button" type="button" data-select-all="sourcePages">Alle</button> <button class="dc-button" type="button" data-check-all="sourcePages">Alle</button>
<button class="dc-button" type="button" data-select-none="sourcePages">Keine</button> <button class="dc-button" type="button" data-check-none="sourcePages">Keine</button>
</span> </span>
<select id="sourcePages" name="sourcePages[]" multiple size="12" style="width:100%;"> <div class="dc-page-tree" id="sourcePages">
<?php foreach (($this->pageChoices ?? []) as $id => $label): ?> <?php
<option value="<?= (int) $id; ?>" <?= in_array((int) $id, ($this->selected['sourcePages'] ?? []), true) ? 'selected' : ''; ?>><?= htmlspecialchars((string) $label, ENT_QUOTES, 'UTF-8'); ?></option> $selectedSourcePages = (array) ($this->selected['sourcePages'] ?? []);
<?php endforeach; ?>
</select> $renderNodes = static function (array $nodes) use (&$renderNodes, $selectedSourcePages): void {
if ($nodes === []) {
return;
}
echo '<ul>';
foreach ($nodes as $node) {
$id = (int) ($node['id'] ?? 0);
$label = (string) ($node['label'] ?? ('Seite ' . $id));
$children = (array) ($node['children'] ?? []);
$checked = in_array($id, $selectedSourcePages, true) ? 'checked' : '';
echo '<li data-tree-item="sourcePages" data-tree-label="' . htmlspecialchars(strtolower($label), ENT_QUOTES, 'UTF-8') . '">';
echo '<label class="dc-page-item">';
echo '<input type="checkbox" name="sourcePages[]" value="' . $id . '" ' . $checked . '>';
echo '<span>' . htmlspecialchars($label, ENT_QUOTES, 'UTF-8') . '</span>';
echo '<span class="dc-page-id">[ID ' . $id . ']</span>';
echo '</label>';
if ($children !== []) {
$renderNodes($children);
}
echo '</li>';
}
echo '</ul>';
};
$renderNodes((array) ($this->pageTreeNodes ?? []));
?>
</div>
</label> </label>
</p> </p>
</div> </div>
@@ -55,7 +97,47 @@
<!-- Abschnitt 3: Quell-Verzeichnisse --> <!-- Abschnitt 3: Quell-Verzeichnisse -->
<div class="dc-section"> <div class="dc-section">
<h3>3. Quell-Verzeichnisse auswaehlen (optional)</h3> <h3>3. Newsarchive auswaehlen (optional)</h3>
<p class="dc-hint">Ausgewaehlte Newsarchive und ihre Newsbeitraege werden kopiert.</p>
<p>
<label>Newsarchive (Mehrfachauswahl):<br>
<input class="dc-filter" type="text" data-filter-for="sourceNewsArchives" placeholder="Newsarchive filtern...">
<span class="dc-tools">
<button class="dc-button" type="button" data-select-all="sourceNewsArchives">Alle</button>
<button class="dc-button" type="button" data-select-none="sourceNewsArchives">Keine</button>
</span>
<select id="sourceNewsArchives" name="sourceNewsArchives[]" multiple size="8" style="width:100%;">
<?php foreach (($this->newsArchiveChoices ?? []) as $id => $label): ?>
<option value="<?= (int) $id; ?>" <?= in_array((int) $id, ($this->selected['sourceNewsArchives'] ?? []), true) ? 'selected' : ''; ?>><?= htmlspecialchars((string) $label, ENT_QUOTES, 'UTF-8'); ?></option>
<?php endforeach; ?>
</select>
</label>
</p>
</div>
<!-- Abschnitt 4: Kalender -->
<div class="dc-section">
<h3>4. Kalender auswaehlen (optional)</h3>
<p class="dc-hint">Ausgewaehlte Kalender und ihre Events werden kopiert.</p>
<p>
<label>Kalender (Mehrfachauswahl):<br>
<input class="dc-filter" type="text" data-filter-for="sourceCalendars" placeholder="Kalender filtern...">
<span class="dc-tools">
<button class="dc-button" type="button" data-select-all="sourceCalendars">Alle</button>
<button class="dc-button" type="button" data-select-none="sourceCalendars">Keine</button>
</span>
<select id="sourceCalendars" name="sourceCalendars[]" multiple size="8" style="width:100%;">
<?php foreach (($this->calendarChoices ?? []) as $id => $label): ?>
<option value="<?= (int) $id; ?>" <?= in_array((int) $id, ($this->selected['sourceCalendars'] ?? []), true) ? 'selected' : ''; ?>><?= htmlspecialchars((string) $label, ENT_QUOTES, 'UTF-8'); ?></option>
<?php endforeach; ?>
</select>
</label>
</p>
</div>
<!-- Abschnitt 5: Quell-Verzeichnisse -->
<div class="dc-section">
<h3>5. Quell-Verzeichnisse auswaehlen (optional)</h3>
<p class="dc-hint">Optionale Dateiverzeichnisse, die gespiegelt werden sollen.</p> <p class="dc-hint">Optionale Dateiverzeichnisse, die gespiegelt werden sollen.</p>
<p> <p>
<label>Verzeichnisse (Mehrfachauswahl):<br> <label>Verzeichnisse (Mehrfachauswahl):<br>
@@ -75,31 +157,109 @@
<!-- Abschnitt 4: Kopieroptionen --> <!-- Abschnitt 4: Kopieroptionen -->
<div class="dc-section"> <div class="dc-section">
<h3>4. Kopieroptionen</h3> <h3>6. Kopieroptionen</h3>
<p><label><input type="checkbox" name="includeContent" value="1" <?= ($this->selected['includeContent'] ?? true) ? 'checked' : ''; ?>> Artikel &amp; Inhaltselemente der Seiten mitkopieren</label></p> <p><label><input type="checkbox" name="includeContent" value="1" <?= ($this->selected['includeContent'] ?? true) ? 'checked' : ''; ?>> Artikel &amp; Inhaltselemente der Seiten mitkopieren</label></p>
<p><label><input type="checkbox" name="copyModules" value="1" <?= ($this->selected['copyModules'] ?? true) ? 'checked' : ''; ?>> Module kopieren und in den kopierten Seiten neu verlinken</label></p> <p><label><input type="checkbox" name="copyModules" value="1" <?= ($this->selected['copyModules'] ?? true) ? 'checked' : ''; ?>> Module kopieren und in den kopierten Seiten neu verlinken</label></p>
<p><label><input type="checkbox" name="copyDirectories" value="1" <?= ($this->selected['copyDirectories'] ?? false) ? 'checked' : ''; ?>> Ausgewaehlte Verzeichnisse in Ziel-Verzeichnis kopieren</label></p> <p><label><input type="checkbox" name="copyDirectories" value="1" <?= ($this->selected['copyDirectories'] ?? false) ? 'checked' : ''; ?>> Ausgewaehlte Verzeichnisse in Ziel-Verzeichnis kopieren</label></p>
<p><label><input type="checkbox" name="dryRun" value="1" <?= ($this->selected['dryRun'] ?? false) ? 'checked' : ''; ?>> Dry-Run (nur Vorschau, keine Schreibzugriffe)</label></p> <p><label><input type="checkbox" name="dryRun" value="1" <?= ($this->selected['dryRun'] ?? false) ? 'checked' : ''; ?>> Dry-Run (nur Vorschau, keine Schreibzugriffe)</label></p>
</div> </div>
<!-- Abschnitt 5: Ziel --> <!-- Abschnitt 7: Ziel & Benennung -->
<div class="dc-section"> <div class="dc-section">
<h3>5. Ziel &amp; Benennung</h3> <h3>7. Ziel &amp; Benennung</h3>
<p> <p>
<label>Ziel-Elternseite (Pflichtfeld):<br> <strong>Ziel-Elternseite:</strong><br>
<select name="targetParentPage" required style="width:100%;"> <span class="dc-hint">Die kopierten Seiten werden als Unterseiten der ausgewaehlten Seite angelegt.</span>
<option value="">Bitte waehlen</option> <input class="dc-filter" type="text" data-filter-for-tree="targetParentPage" placeholder="Seiten im Baum filtern...">
<?php foreach (($this->pageChoices ?? []) as $id => $label): ?> <div class="dc-page-tree" id="targetParentPageTree" style="max-height:280px;">
<option value="<?= (int) $id; ?>" <?= ((int) ($this->selected['targetParentPage'] ?? 0) === (int) $id) ? 'selected' : ''; ?>><?= htmlspecialchars((string) $label, ENT_QUOTES, 'UTF-8'); ?></option> <div class="dc-tree-option">
<?php endforeach; ?> <label class="dc-page-item">
</select> <input type="radio" name="targetParentPage" value="0" <?= ((int) ($this->selected['targetParentPage'] ?? 0) === 0) ? 'checked' : ''; ?>>
</label> <em>&#8212; Auf Root-Ebene einfuegen &#8212;</em>
</label>
</div>
<?php
$selectedTargetPage = (int) ($this->selected['targetParentPage'] ?? 0);
$renderTargetNodes = static function (array $nodes) use (&$renderTargetNodes, $selectedTargetPage): void {
if ($nodes === []) {
return;
}
echo '<ul>';
foreach ($nodes as $node) {
$id = (int) ($node['id'] ?? 0);
$label = (string) ($node['label'] ?? ('Seite ' . $id));
$children = (array) ($node['children'] ?? []);
$checked = ($id > 0 && $id === $selectedTargetPage) ? 'checked' : '';
echo '<li data-tree-item="targetParentPage" data-tree-label="' . htmlspecialchars(strtolower($label), ENT_QUOTES, 'UTF-8') . '">';
echo '<label class="dc-page-item">';
echo '<input type="radio" name="targetParentPage" value="' . $id . '" ' . $checked . '>';
echo '<span>' . htmlspecialchars($label, ENT_QUOTES, 'UTF-8') . '</span>';
echo '<span class="dc-page-id">[ID ' . $id . ']</span>';
echo '</label>';
if ($children !== []) {
$renderTargetNodes($children);
}
echo '</li>';
}
echo '</ul>';
};
$renderTargetNodes((array) ($this->pageTreeNodes ?? []));
?>
</div>
</p> </p>
<p> <p>
<label>Ziel-Verzeichnis fuer Dateien (z. B. files/kunden/kunde-x):<br> <strong>Ziel-Verzeichnis fuer Dateien (optional):</strong><br>
<input type="text" name="targetDirectory" style="width:100%" placeholder="files/kunden/mein-kunde" value="<?= htmlspecialchars((string) ($this->selected['targetDirectory'] ?? ''), ENT_QUOTES, 'UTF-8'); ?>"> <span class="dc-hint">Quellverzeichnisse werden hierhin kopiert. Vorhandenes auswaehlen oder unten manuell eingeben (auch neue Pfade moeglich).</span>
</label> <input class="dc-filter" type="text" data-filter-for-tree="targetDirTree" placeholder="Verzeichnisse filtern...">
<div class="dc-page-tree" id="targetDirTreeContainer" style="max-height:220px;">
<?php
$selectedTargetDir = (string) ($this->selected['targetDirectory'] ?? '');
$renderDirNodes = static function (array $nodes) use (&$renderDirNodes, $selectedTargetDir): void {
if ($nodes === []) {
return;
}
echo '<ul>';
foreach ($nodes as $node) {
$path = (string) ($node['path'] ?? '');
$label = (string) ($node['label'] ?? $path);
$children = (array) ($node['children'] ?? []);
$selClass = ($path !== '' && $path === $selectedTargetDir) ? ' dc-dir-selected' : '';
echo '<li data-tree-item="targetDirTree" data-tree-label="' . htmlspecialchars(strtolower($label), ENT_QUOTES, 'UTF-8') . '" data-dir-path="' . htmlspecialchars($path, ENT_QUOTES, 'UTF-8') . '">';
echo '<span class="dc-page-item dc-dir-item' . $selClass . '" title="' . htmlspecialchars($path, ENT_QUOTES, 'UTF-8') . '">';
echo '<span>' . htmlspecialchars($label, ENT_QUOTES, 'UTF-8') . '</span>';
echo '<span class="dc-page-id">' . htmlspecialchars($path, ENT_QUOTES, 'UTF-8') . '</span>';
echo '</span>';
if ($children !== []) {
$renderDirNodes($children);
}
echo '</li>';
}
echo '</ul>';
};
$renderDirNodes((array) ($this->directoryTreeNodes ?? []));
?>
</div>
<input type="text" id="targetDirectory" name="targetDirectory" style="width:100%; margin-top:0.5rem;" placeholder="z. B. files/kunden/mein-kunde" value="<?= htmlspecialchars((string) ($this->selected['targetDirectory'] ?? ''), ENT_QUOTES, 'UTF-8'); ?>">
</p> </p>
<p> <p>
<label>Praefix fuer Titel / Name / Alias der Kopien:<br> <label>Praefix fuer Titel / Name / Alias der Kopien:<br>
<input type="text" name="namePrefix" style="width:100%" placeholder="z. B. kunde-x-" value="<?= htmlspecialchars((string) ($this->selected['namePrefix'] ?? ''), ENT_QUOTES, 'UTF-8'); ?>"> <input type="text" name="namePrefix" style="width:100%" placeholder="z. B. kunde-x-" value="<?= htmlspecialchars((string) ($this->selected['namePrefix'] ?? ''), ENT_QUOTES, 'UTF-8'); ?>">
@@ -115,9 +275,17 @@
'copiedPages' => $this->result->copiedPages, 'copiedPages' => $this->result->copiedPages,
'copiedModules' => $this->result->copiedModules, 'copiedModules' => $this->result->copiedModules,
'copiedContent' => $this->result->copiedContent, 'copiedContent' => $this->result->copiedContent,
'copiedNewsArchives'=> $this->result->copiedNewsArchives,
'copiedNewsItems' => $this->result->copiedNewsItems,
'copiedCalendars' => $this->result->copiedCalendars,
'copiedEvents' => $this->result->copiedEvents,
'copiedDirectories' => $this->result->copiedDirectories, 'copiedDirectories' => $this->result->copiedDirectories,
'pageMap' => $this->result->pageMap, 'pageMap' => $this->result->pageMap,
'moduleMap' => $this->result->moduleMap, 'moduleMap' => $this->result->moduleMap,
'newsArchiveMap' => $this->result->newsArchiveMap,
'calendarMap' => $this->result->calendarMap,
'newsItemMap' => $this->result->newsItemMap,
'eventMap' => $this->result->eventMap,
'notes' => $this->result->notes, 'notes' => $this->result->notes,
], JSON_PRETTY_PRINT | JSON_UNESCAPED_SLASHES), ENT_QUOTES, 'UTF-8'); ?></pre> ], JSON_PRETTY_PRINT | JSON_UNESCAPED_SLASHES), ENT_QUOTES, 'UTF-8'); ?></pre>
<?php endif; ?> <?php endif; ?>
@@ -126,6 +294,87 @@
(function () { (function () {
function byId(id) { return document.getElementById(id); } function byId(id) { return document.getElementById(id); }
function childCheckboxesOf(li) {
var nested = li.querySelector(':scope > ul');
return nested ? nested.querySelectorAll('input[type="checkbox"]') : [];
}
function parentLiOf(li) {
var parentUl = li.parentElement;
if (!parentUl || parentUl.classList.contains('dc-page-tree')) {
return null;
}
var candidate = parentUl.closest('li[data-tree-item="sourcePages"]');
return candidate || null;
}
function updateParentState(li) {
var children = childCheckboxesOf(li);
if (!children.length) {
return;
}
var own = li.querySelector(':scope > label input[type="checkbox"]');
if (!own) {
return;
}
var checkedCount = 0;
children.forEach(function (cb) {
if (cb.checked) {
checkedCount++;
}
});
if (checkedCount === 0) {
own.checked = false;
own.indeterminate = false;
} else if (checkedCount === children.length) {
own.checked = true;
own.indeterminate = false;
} else {
own.checked = false;
own.indeterminate = true;
}
}
function cascadeToChildren(li, checked) {
childCheckboxesOf(li).forEach(function (cb) {
cb.checked = checked;
cb.indeterminate = false;
});
}
function refreshAllParentStates() {
var nodes = Array.prototype.slice.call(document.querySelectorAll('li[data-tree-item="sourcePages"]'));
nodes.reverse();
nodes.forEach(function (li) {
updateParentState(li);
});
}
document.querySelectorAll('li[data-tree-item="sourcePages"] > label input[type="checkbox"]').forEach(function (checkbox) {
checkbox.addEventListener('change', function () {
var li = checkbox.closest('li[data-tree-item="sourcePages"]');
if (!li) {
return;
}
if (!checkbox.indeterminate) {
cascadeToChildren(li, checkbox.checked);
}
var parent = parentLiOf(li);
while (parent) {
updateParentState(parent);
parent = parentLiOf(parent);
}
});
});
refreshAllParentStates();
document.querySelectorAll('[data-filter-for]').forEach(function (input) { document.querySelectorAll('[data-filter-for]').forEach(function (input) {
input.addEventListener('input', function () { input.addEventListener('input', function () {
var select = byId(input.getAttribute('data-filter-for')); var select = byId(input.getAttribute('data-filter-for'));
@@ -137,6 +386,42 @@
}); });
}); });
document.querySelectorAll('[data-filter-for-tree]').forEach(function (input) {
input.addEventListener('input', function () {
var key = input.getAttribute('data-filter-for-tree');
var query = (input.value || '').toLowerCase();
var items = Array.prototype.slice.call(document.querySelectorAll('[data-tree-item="' + key + '"]'));
if (query === '') {
items.forEach(function (item) {
item.hidden = false;
delete item.dataset.treeMatched;
});
return;
}
items.reverse().forEach(function (item) {
var label = item.getAttribute('data-tree-label') || '';
var selfMatch = label.indexOf(query) !== -1;
var nested = item.querySelector(':scope > ul');
var childMatch = false;
if (nested) {
Array.prototype.forEach.call(nested.children, function (child) {
if (child.matches('[data-tree-item="' + key + '"]') && child.dataset.treeMatched === '1') {
childMatch = true;
}
});
}
var match = selfMatch || childMatch;
item.hidden = !match;
item.dataset.treeMatched = match ? '1' : '0';
});
});
});
document.querySelectorAll('[data-select-all]').forEach(function (button) { document.querySelectorAll('[data-select-all]').forEach(function (button) {
button.addEventListener('click', function () { button.addEventListener('click', function () {
var select = byId(button.getAttribute('data-select-all')); var select = byId(button.getAttribute('data-select-all'));
@@ -156,6 +441,71 @@
}); });
}); });
}); });
document.querySelectorAll('[data-check-all]').forEach(function (button) {
button.addEventListener('click', function () {
var key = button.getAttribute('data-check-all');
document.querySelectorAll('[data-tree-item="' + key + '"] input[type="checkbox"]').forEach(function (checkbox) {
if (!checkbox.closest('li').hidden) {
checkbox.checked = true;
checkbox.indeterminate = false;
}
});
refreshAllParentStates();
});
});
document.querySelectorAll('[data-check-none]').forEach(function (button) {
button.addEventListener('click', function () {
var key = button.getAttribute('data-check-none');
document.querySelectorAll('[data-tree-item="' + key + '"] input[type="checkbox"]').forEach(function (checkbox) {
checkbox.checked = false;
checkbox.indeterminate = false;
});
refreshAllParentStates();
});
});
// Verzeichnisbaum: Klick befuellt Texteingabe
document.querySelectorAll('.dc-dir-item').forEach(function (span) {
span.addEventListener('click', function () {
var li = span.closest('[data-dir-path]');
var path = li ? li.getAttribute('data-dir-path') : '';
var input = document.getElementById('targetDirectory');
if (input) {
input.value = path || '';
}
document.querySelectorAll('.dc-dir-item').forEach(function (s) {
s.classList.remove('dc-dir-selected');
});
span.classList.add('dc-dir-selected');
});
});
// Texteingabe Ziel-Verzeichnis: Auswahl im Baum aufheben wenn manuell getippt
(function () {
var input = document.getElementById('targetDirectory');
if (!input) { return; }
input.addEventListener('input', function () {
var val = input.value.trim();
document.querySelectorAll('.dc-dir-item').forEach(function (s) {
var li = s.closest('[data-dir-path]');
var path = li ? li.getAttribute('data-dir-path') : '';
if (path === val) {
s.classList.add('dc-dir-selected');
} else {
s.classList.remove('dc-dir-selected');
}
});
});
})();
})(); })();
</script> </script>
</form> </form>

70
public/icon.svg Normal file
View File

@@ -0,0 +1,70 @@
<?xml version="1.0" encoding="UTF-8" standalone="no"?>
<svg
viewBox="0 0 32 32"
fill="none"
version="1.1"
id="svg10"
sodipodi:docname="icon.svg"
inkscape:version="1.2.2 (b0a8486541, 2022-12-01)"
xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape"
xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd"
xmlns="http://www.w3.org/2000/svg"
xmlns:svg="http://www.w3.org/2000/svg">
<defs
id="defs14" />
<sodipodi:namedview
id="namedview12"
pagecolor="#ffffff"
bordercolor="#666666"
borderopacity="1.0"
inkscape:showpageshadow="2"
inkscape:pageopacity="0.0"
inkscape:pagecheckerboard="0"
inkscape:deskcolor="#d1d1d1"
showgrid="false"
inkscape:zoom="32"
inkscape:cx="14.015625"
inkscape:cy="16"
inkscape:window-width="2560"
inkscape:window-height="1408"
inkscape:window-x="2560"
inkscape:window-y="0"
inkscape:window-maximized="1"
inkscape:current-layer="svg10" />
<rect
x="4"
y="5"
width="11"
height="9"
rx="2"
stroke="#1f2937"
stroke-width="2"
id="rect2" />
<rect
x="18"
y="5"
width="11"
height="9"
rx="2"
stroke="#1f2937"
stroke-width="2"
opacity="0.55"
id="rect4" />
<rect
x="4"
y="18"
width="11"
height="9"
rx="2"
stroke="#1f2937"
stroke-width="2"
opacity="0.55"
id="rect6" />
<path
d="m 19,22.5 h 9 m -4,-4 4,4 -4,4"
stroke="#0f766e"
stroke-width="2.5"
stroke-linecap="round"
stroke-linejoin="round"
id="path8" />
</svg>

After

Width:  |  Height:  |  Size: 1.6 KiB

View File

@@ -27,8 +27,12 @@ class DummyCopierModule extends BackendModule
$this->Template->action = Environment::get('request'); $this->Template->action = Environment::get('request');
$this->Template->requestToken = $this->getCsrfToken(); $this->Template->requestToken = $this->getCsrfToken();
$this->Template->pageChoices = $this->getPageChoices($connection); $this->Template->pageChoices = $this->getPageChoices($connection);
$this->Template->pageTreeNodes = $this->getPageTreeNodes($connection);
$this->Template->moduleChoices = $this->getModuleChoices($connection); $this->Template->moduleChoices = $this->getModuleChoices($connection);
$this->Template->newsArchiveChoices = $this->getNewsArchiveChoices($connection);
$this->Template->calendarChoices = $this->getCalendarChoices($connection);
$this->Template->directoryChoices = $this->getDirectoryChoices(); $this->Template->directoryChoices = $this->getDirectoryChoices();
$this->Template->directoryTreeNodes = $this->getDirectoryTreeNodes();
$targetParentPageId = $this->parseSingleIdInput(Input::postRaw('targetParentPage')); $targetParentPageId = $this->parseSingleIdInput(Input::postRaw('targetParentPage'));
$isPost = Input::post('FORM_SUBMIT') === 'tl_dummy_copier'; $isPost = Input::post('FORM_SUBMIT') === 'tl_dummy_copier';
@@ -36,6 +40,8 @@ class DummyCopierModule extends BackendModule
$this->Template->selected = [ $this->Template->selected = [
'sourcePages' => $this->parseIdInput(Input::postRaw('sourcePages')), 'sourcePages' => $this->parseIdInput(Input::postRaw('sourcePages')),
'sourceModules' => $this->parseIdInput(Input::postRaw('sourceModules')), 'sourceModules' => $this->parseIdInput(Input::postRaw('sourceModules')),
'sourceNewsArchives' => $this->parseIdInput(Input::postRaw('sourceNewsArchives')),
'sourceCalendars' => $this->parseIdInput(Input::postRaw('sourceCalendars')),
'sourceDirectories' => $this->parsePathInput(Input::postRaw('sourceDirectories')), 'sourceDirectories' => $this->parsePathInput(Input::postRaw('sourceDirectories')),
'targetParentPage' => $targetParentPageId, 'targetParentPage' => $targetParentPageId,
'targetDirectory' => trim((string) Input::post('targetDirectory')), 'targetDirectory' => trim((string) Input::post('targetDirectory')),
@@ -53,6 +59,8 @@ class DummyCopierModule extends BackendModule
$options = new DummyCopyOptions( $options = new DummyCopyOptions(
$this->parseIdInput(Input::postRaw('sourcePages')), $this->parseIdInput(Input::postRaw('sourcePages')),
$this->parseIdInput(Input::postRaw('sourceModules')), $this->parseIdInput(Input::postRaw('sourceModules')),
$this->parseIdInput(Input::postRaw('sourceNewsArchives')),
$this->parseIdInput(Input::postRaw('sourceCalendars')),
[], [],
$this->parsePathInput(Input::postRaw('sourceDirectories')), $this->parsePathInput(Input::postRaw('sourceDirectories')),
$targetParentPageId, $targetParentPageId,
@@ -74,10 +82,14 @@ class DummyCopierModule extends BackendModule
$result = $copier->execute($options); $result = $copier->execute($options);
Message::addConfirmation(sprintf( Message::addConfirmation(sprintf(
'Fertig. Seiten: %d, Module: %d, Content: %d, Verzeichnisse: %d', 'Fertig. Seiten: %d, Module: %d, Content: %d, Newsarchive: %d, Newsbeitraege: %d, Kalender: %d, Events: %d, Verzeichnisse: %d',
$result->copiedPages, $result->copiedPages,
$result->copiedModules, $result->copiedModules,
$result->copiedContent, $result->copiedContent,
$result->copiedNewsArchives,
$result->copiedNewsItems,
$result->copiedCalendars,
$result->copiedEvents,
$result->copiedDirectories $result->copiedDirectories
)); ));
@@ -219,6 +231,50 @@ class DummyCopierModule extends BackendModule
return $choices; return $choices;
} }
/**
* @return array<int,array<string,mixed>>
*/
private function getPageTreeNodes(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;
}
$build = function (int $pid) use (&$build, $rowsByParent): array {
$nodes = [];
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 . ')';
}
$nodes[] = [
'id' => $id,
'label' => $label,
'children' => $build($id),
];
}
return $nodes;
};
return $build(0);
}
/** /**
* @return array<int,string> * @return array<int,string>
*/ */
@@ -285,6 +341,96 @@ class DummyCopierModule extends BackendModule
return $choices; return $choices;
} }
/**
* @return array<int,array<string,mixed>>
*/
private function getDirectoryTreeNodes(): array
{
$projectDir = (string) System::getContainer()->getParameter('kernel.project_dir');
$filesDir = $projectDir . '/files';
if (!is_dir($filesDir)) {
return [];
}
$build = function (string $dir) use (&$build, $projectDir): array {
$nodes = [];
$entries = @scandir($dir);
if (!$entries) {
return [];
}
foreach ($entries as $entry) {
if ($entry === '.' || $entry === '..' || str_starts_with($entry, '.')) {
continue;
}
$full = $dir . '/' . $entry;
if (!is_dir($full)) {
continue;
}
$relative = ltrim(str_replace($projectDir, '', $full), '/');
$nodes[] = [
'path' => $relative,
'label' => $entry,
'children' => $build($full),
];
}
return $nodes;
};
return $build($filesDir);
}
/**
* @return array<int,string>
*/
private function getNewsArchiveChoices(Connection $connection): array
{
$rows = $connection->fetchAllAssociative('SELECT id, title FROM tl_news_archive ORDER BY title');
$choices = [];
foreach ($rows as $row) {
$id = (int) ($row['id'] ?? 0);
if ($id < 1) {
continue;
}
$title = trim((string) ($row['title'] ?? ''));
$choices[$id] = sprintf('%s [ID %d]', $title !== '' ? $title : ('Newsarchiv ' . $id), $id);
}
return $choices;
}
/**
* @return array<int,string>
*/
private function getCalendarChoices(Connection $connection): array
{
$rows = $connection->fetchAllAssociative('SELECT id, title FROM tl_calendar ORDER BY title');
$choices = [];
foreach ($rows as $row) {
$id = (int) ($row['id'] ?? 0);
if ($id < 1) {
continue;
}
$title = trim((string) ($row['title'] ?? ''));
$choices[$id] = sprintf('%s [ID %d]', $title !== '' ? $title : ('Kalender ' . $id), $id);
}
return $choices;
}
private function normalizeHeadline($headline): string private function normalizeHeadline($headline): string
{ {
if (\is_string($headline)) { if (\is_string($headline)) {

View File

@@ -4,6 +4,7 @@ declare(strict_types=1);
namespace Webfarben\DummyCopier\Service; namespace Webfarben\DummyCopier\Service;
use Contao\StringUtil;
use Doctrine\DBAL\Connection; use Doctrine\DBAL\Connection;
use Symfony\Component\Filesystem\Filesystem; use Symfony\Component\Filesystem\Filesystem;
@@ -29,6 +30,10 @@ final class DummyCopier
$result->copiedModules = $options->copyModules ? $this->countRows('tl_module', $options->sourceModuleIds) : 0; $result->copiedModules = $options->copyModules ? $this->countRows('tl_module', $options->sourceModuleIds) : 0;
$result->copiedContent = $this->estimateContentCount($options); $result->copiedContent = $this->estimateContentCount($options);
$result->copiedDirectories = $options->copyDirectories ? \count($options->sourceDirectories) : 0; $result->copiedDirectories = $options->copyDirectories ? \count($options->sourceDirectories) : 0;
$result->copiedNewsArchives = $this->countRows('tl_news_archive', $options->sourceNewsArchiveIds);
$result->copiedNewsItems = $this->countByPid('tl_news', 'pid', $options->sourceNewsArchiveIds);
$result->copiedCalendars = $this->countRows('tl_calendar', $options->sourceCalendarIds);
$result->copiedEvents = $this->countByPid('tl_calendar_events', 'pid', $options->sourceCalendarIds);
$result->addNote('Dry-Run aktiv: Es wurden keine Daten geschrieben.'); $result->addNote('Dry-Run aktiv: Es wurden keine Daten geschrieben.');
return $result; return $result;
@@ -49,6 +54,14 @@ final class DummyCopier
} }
} }
if ($options->sourceNewsArchiveIds !== []) {
$result->newsArchiveMap = $this->copyNewsArchives($options->sourceNewsArchiveIds, $options->namePrefix, $result);
}
if ($options->sourceCalendarIds !== []) {
$result->calendarMap = $this->copyCalendars($options->sourceCalendarIds, $options->namePrefix, $result);
}
if ($options->targetArticleId > 0) { if ($options->targetArticleId > 0) {
foreach ($options->sourceContentIds as $sourceContentId) { foreach ($options->sourceContentIds as $sourceContentId) {
$this->copySingleContent($sourceContentId, $options->targetArticleId, $result->moduleMap, $result); $this->copySingleContent($sourceContentId, $options->targetArticleId, $result->moduleMap, $result);
@@ -57,6 +70,10 @@ final class DummyCopier
$this->rewriteReferences($result); $this->rewriteReferences($result);
if ($options->copyModules) {
$this->rewriteModuleArchiveReferences($result);
}
if ($options->copyDirectories) { if ($options->copyDirectories) {
$this->copyDirectories($options, $result); $this->copyDirectories($options, $result);
} }
@@ -160,13 +177,14 @@ final class DummyCopier
*/ */
private function copyArticleContentTree(int $sourceArticleId, int $targetArticleId, array $moduleMap, DummyCopyResult $result): void private function copyArticleContentTree(int $sourceArticleId, int $targetArticleId, array $moduleMap, DummyCopyResult $result): void
{ {
$visited = [];
$contentIds = $this->connection->fetchFirstColumn( $contentIds = $this->connection->fetchFirstColumn(
'SELECT id FROM tl_content WHERE ptable = ? AND pid = ? ORDER BY sorting', 'SELECT id FROM tl_content WHERE ptable = ? AND pid = ? ORDER BY sorting',
['tl_article', $sourceArticleId] ['tl_article', $sourceArticleId]
); );
foreach ($contentIds as $contentId) { foreach ($contentIds as $contentId) {
$this->copyContentRecursive((int) $contentId, 'tl_article', $targetArticleId, $moduleMap, $result); $this->copyContentRecursive((int) $contentId, 'tl_article', $targetArticleId, $moduleMap, $result, $visited);
} }
} }
@@ -175,7 +193,8 @@ final class DummyCopier
*/ */
private function copySingleContent(int $sourceContentId, int $targetArticleId, array $moduleMap, DummyCopyResult $result): void private function copySingleContent(int $sourceContentId, int $targetArticleId, array $moduleMap, DummyCopyResult $result): void
{ {
$this->copyContentRecursive($sourceContentId, 'tl_article', $targetArticleId, $moduleMap, $result); $visited = [];
$this->copyContentRecursive($sourceContentId, 'tl_article', $targetArticleId, $moduleMap, $result, $visited);
} }
/** /**
@@ -183,8 +202,13 @@ final class DummyCopier
* *
* @param array<int,int> $moduleMap * @param array<int,int> $moduleMap
*/ */
private function copyContentRecursive(int $sourceContentId, string $targetPtable, int $targetPid, array $moduleMap, DummyCopyResult $result): int private function copyContentRecursive(int $sourceContentId, string $targetPtable, int $targetPid, array $moduleMap, DummyCopyResult $result, array &$visited): int
{ {
if (isset($visited[$sourceContentId])) {
return 0;
}
$visited[$sourceContentId] = true;
$contentRow = $this->fetchRow('tl_content', $sourceContentId); $contentRow = $this->fetchRow('tl_content', $sourceContentId);
if ($contentRow === null) { if ($contentRow === null) {
@@ -207,6 +231,7 @@ final class DummyCopier
} }
$newContentId = $this->insertRow('tl_content', $contentRow); $newContentId = $this->insertRow('tl_content', $contentRow);
$result->contentMap[$sourceContentId] = $newContentId;
$result->copiedContent++; $result->copiedContent++;
$result->copiedContentIds[] = $newContentId; $result->copiedContentIds[] = $newContentId;
@@ -220,7 +245,7 @@ final class DummyCopier
); );
foreach ($childIds as $childId) { foreach ($childIds as $childId) {
$this->copyContentRecursive((int) $childId, 'tl_content', $newContentId, $moduleMap, $result); $this->copyContentRecursive((int) $childId, 'tl_content', $newContentId, $moduleMap, $result, $visited);
} }
return $newContentId; return $newContentId;
@@ -260,6 +285,245 @@ final class DummyCopier
$params $params
); );
} }
foreach ($result->contentMap as $oldContentId => $newContentId) {
$sourceAliasTarget = (int) $this->connection->fetchOne(
'SELECT cteAlias FROM tl_content WHERE id = ? AND type = ?',
[$oldContentId, 'alias']
);
if ($sourceAliasTarget < 1 || !isset($result->contentMap[$sourceAliasTarget])) {
continue;
}
$this->connection->update(
'tl_content',
['cteAlias' => $result->contentMap[$sourceAliasTarget]],
['id' => $newContentId, 'type' => 'alias']
);
}
}
private function rewriteModuleArchiveReferences(DummyCopyResult $result): void
{
if ($result->moduleMap === []) {
return;
}
foreach ($result->moduleMap as $newModuleId) {
$row = $this->fetchRow('tl_module', $newModuleId);
if ($row === null) {
continue;
}
$updates = [];
if (array_key_exists('news_archives', $row) && $result->newsArchiveMap !== []) {
$updates['news_archives'] = $this->remapSerializedIds((string) ($row['news_archives'] ?? ''), $result->newsArchiveMap);
}
if (array_key_exists('cal_calendar', $row) && $result->calendarMap !== []) {
$updates['cal_calendar'] = $this->remapSerializedIds((string) ($row['cal_calendar'] ?? ''), $result->calendarMap);
}
if (array_key_exists('news_readerModule', $row)) {
$oldReaderModuleId = (int) ($row['news_readerModule'] ?? 0);
if ($oldReaderModuleId > 0 && isset($result->moduleMap[$oldReaderModuleId])) {
$updates['news_readerModule'] = $result->moduleMap[$oldReaderModuleId];
}
}
if (array_key_exists('cal_readerModule', $row)) {
$oldReaderModuleId = (int) ($row['cal_readerModule'] ?? 0);
if ($oldReaderModuleId > 0 && isset($result->moduleMap[$oldReaderModuleId])) {
$updates['cal_readerModule'] = $result->moduleMap[$oldReaderModuleId];
}
}
if ($updates !== []) {
$updates['tstamp'] = time();
$this->connection->update('tl_module', $updates, ['id' => $newModuleId]);
}
}
$this->rewriteNewsItemReferences($result);
}
private function rewriteNewsItemReferences(DummyCopyResult $result): void
{
if ($result->newsItemMap === []) {
return;
}
foreach ($result->newsItemMap as $oldNewsId => $newNewsId) {
$sourceRow = $this->fetchRow('tl_news', $oldNewsId);
if ($sourceRow === null || !array_key_exists('related', $sourceRow)) {
continue;
}
$related = StringUtil::deserialize((string) ($sourceRow['related'] ?? ''), true);
if ($related === []) {
continue;
}
$mappedRelated = [];
foreach ($related as $relatedIdValue) {
$relatedId = (int) $relatedIdValue;
if ($relatedId < 1) {
continue;
}
$mappedRelated[] = (string) ($result->newsItemMap[$relatedId] ?? $relatedId);
}
$this->connection->update(
'tl_news',
[
'related' => serialize($mappedRelated),
'tstamp' => time(),
],
['id' => $newNewsId]
);
}
}
/**
* @param array<int> $sourceArchiveIds
*
* @return array<int,int>
*/
private function copyNewsArchives(array $sourceArchiveIds, string $prefix, DummyCopyResult $result): array
{
$map = [];
foreach ($sourceArchiveIds as $sourceArchiveId) {
$archiveRow = $this->fetchRow('tl_news_archive', (int) $sourceArchiveId);
if ($archiveRow === null) {
$result->addNote(sprintf('Newsarchiv %d nicht gefunden, wurde uebersprungen.', $sourceArchiveId));
continue;
}
unset($archiveRow['id']);
$archiveRow['tstamp'] = time();
if (isset($archiveRow['title'])) {
$archiveRow['title'] = $this->prefixed((string) $archiveRow['title'], $prefix);
}
if (isset($archiveRow['jumpTo']) && isset($result->pageMap[(int) $archiveRow['jumpTo']])) {
$archiveRow['jumpTo'] = $result->pageMap[(int) $archiveRow['jumpTo']];
}
$newArchiveId = $this->insertRow('tl_news_archive', $archiveRow);
$map[(int) $sourceArchiveId] = $newArchiveId;
$result->copiedNewsArchives++;
$newsIds = $this->connection->fetchFirstColumn('SELECT id FROM tl_news WHERE pid = ? ORDER BY date, id', [(int) $sourceArchiveId]);
foreach ($newsIds as $newsId) {
$newsRow = $this->fetchRow('tl_news', (int) $newsId);
if ($newsRow === null) {
continue;
}
unset($newsRow['id']);
$newsRow['pid'] = $newArchiveId;
$newsRow['tstamp'] = time();
if (isset($newsRow['headline'])) {
$newsRow['headline'] = $this->prefixed((string) $newsRow['headline'], $prefix);
}
if (isset($newsRow['alias']) && trim((string) $newsRow['alias']) !== '') {
$newsRow['alias'] = $this->makeUniqueAliasInTable('tl_news', $this->prefixed((string) $newsRow['alias'], $prefix));
}
if (isset($newsRow['jumpTo']) && isset($result->pageMap[(int) $newsRow['jumpTo']])) {
$newsRow['jumpTo'] = $result->pageMap[(int) $newsRow['jumpTo']];
}
$newNewsId = $this->insertRow('tl_news', $newsRow);
$result->newsItemMap[(int) $newsId] = $newNewsId;
$result->copiedNewsItems++;
}
}
return $map;
}
/**
* @param array<int> $sourceCalendarIds
*
* @return array<int,int>
*/
private function copyCalendars(array $sourceCalendarIds, string $prefix, DummyCopyResult $result): array
{
$map = [];
foreach ($sourceCalendarIds as $sourceCalendarId) {
$calendarRow = $this->fetchRow('tl_calendar', (int) $sourceCalendarId);
if ($calendarRow === null) {
$result->addNote(sprintf('Kalender %d nicht gefunden, wurde uebersprungen.', $sourceCalendarId));
continue;
}
unset($calendarRow['id']);
$calendarRow['tstamp'] = time();
if (isset($calendarRow['title'])) {
$calendarRow['title'] = $this->prefixed((string) $calendarRow['title'], $prefix);
}
if (isset($calendarRow['jumpTo']) && isset($result->pageMap[(int) $calendarRow['jumpTo']])) {
$calendarRow['jumpTo'] = $result->pageMap[(int) $calendarRow['jumpTo']];
}
$newCalendarId = $this->insertRow('tl_calendar', $calendarRow);
$map[(int) $sourceCalendarId] = $newCalendarId;
$result->copiedCalendars++;
$eventIds = $this->connection->fetchFirstColumn('SELECT id FROM tl_calendar_events WHERE pid = ? ORDER BY startTime, id', [(int) $sourceCalendarId]);
foreach ($eventIds as $eventId) {
$eventRow = $this->fetchRow('tl_calendar_events', (int) $eventId);
if ($eventRow === null) {
continue;
}
unset($eventRow['id']);
$eventRow['pid'] = $newCalendarId;
$eventRow['tstamp'] = time();
if (isset($eventRow['title'])) {
$eventRow['title'] = $this->prefixed((string) $eventRow['title'], $prefix);
}
if (isset($eventRow['alias']) && trim((string) $eventRow['alias']) !== '') {
$eventRow['alias'] = $this->makeUniqueAliasInTable('tl_calendar_events', $this->prefixed((string) $eventRow['alias'], $prefix));
}
if (isset($eventRow['jumpTo']) && isset($result->pageMap[(int) $eventRow['jumpTo']])) {
$eventRow['jumpTo'] = $result->pageMap[(int) $eventRow['jumpTo']];
}
$newEventId = $this->insertRow('tl_calendar_events', $eventRow);
$result->eventMap[(int) $eventId] = $newEventId;
$result->copiedEvents++;
}
}
return $map;
} }
private function copyDirectories(DummyCopyOptions $options, DummyCopyResult $result): void private function copyDirectories(DummyCopyOptions $options, DummyCopyResult $result): void
@@ -306,11 +570,16 @@ final class DummyCopier
} }
private function makeUniqueAlias(string $baseAlias): string private function makeUniqueAlias(string $baseAlias): string
{
return $this->makeUniqueAliasInTable('tl_page', $baseAlias);
}
private function makeUniqueAliasInTable(string $table, string $baseAlias): string
{ {
$alias = $this->slugify($baseAlias); $alias = $this->slugify($baseAlias);
$counter = 1; $counter = 1;
while ((int) $this->connection->fetchOne('SELECT COUNT(*) FROM tl_page WHERE alias = ?', [$alias]) > 0) { while ((int) $this->connection->fetchOne(sprintf('SELECT COUNT(*) FROM %s WHERE alias = ?', $table), [$alias]) > 0) {
$alias = $this->slugify($baseAlias) . '-' . $counter; $alias = $this->slugify($baseAlias) . '-' . $counter;
$counter++; $counter++;
} }
@@ -360,9 +629,41 @@ final class DummyCopier
return (int) $this->connection->fetchOne(sprintf('SELECT COUNT(*) FROM %s WHERE id IN (%s)', $table, $placeholders), $ids); return (int) $this->connection->fetchOne(sprintf('SELECT COUNT(*) FROM %s WHERE id IN (%s)', $table, $placeholders), $ids);
} }
private function countByPid(string $table, string $pidField, array $pidValues): int
{
if ($pidValues === []) {
return 0;
}
$placeholders = implode(',', array_fill(0, \count($pidValues), '?'));
return (int) $this->connection->fetchOne(sprintf('SELECT COUNT(*) FROM %s WHERE %s IN (%s)', $table, $pidField, $placeholders), $pidValues);
}
/**
* @param array<int,int> $idMap
*/
private function remapSerializedIds(string $serialized, array $idMap): string
{
$values = StringUtil::deserialize($serialized, true);
$mapped = [];
foreach ($values as $value) {
$id = (int) $value;
if ($id < 1) {
continue;
}
$mapped[] = (string) ($idMap[$id] ?? $id);
}
return serialize($mapped);
}
private function estimateContentCount(DummyCopyOptions $options): int private function estimateContentCount(DummyCopyOptions $options): int
{ {
$count = \count($options->sourceContentIds); $count = $this->countContentTree(array_map('intval', $options->sourceContentIds));
if (!$options->includeContent || $options->sourcePageIds === []) { if (!$options->includeContent || $options->sourcePageIds === []) {
return $count; return $count;

View File

@@ -9,6 +9,8 @@ final class DummyCopyOptions
public function __construct( public function __construct(
public readonly array $sourcePageIds, public readonly array $sourcePageIds,
public readonly array $sourceModuleIds, public readonly array $sourceModuleIds,
public readonly array $sourceNewsArchiveIds,
public readonly array $sourceCalendarIds,
public readonly array $sourceContentIds, public readonly array $sourceContentIds,
public readonly array $sourceDirectories, public readonly array $sourceDirectories,
public readonly int $targetParentPageId, public readonly int $targetParentPageId,

View File

@@ -10,6 +10,10 @@ final class DummyCopyResult
public int $copiedModules = 0; public int $copiedModules = 0;
public int $copiedContent = 0; public int $copiedContent = 0;
public int $copiedDirectories = 0; public int $copiedDirectories = 0;
public int $copiedNewsArchives = 0;
public int $copiedNewsItems = 0;
public int $copiedCalendars = 0;
public int $copiedEvents = 0;
/** @var array<int,int> */ /** @var array<int,int> */
public array $pageMap = []; public array $pageMap = [];
@@ -17,6 +21,21 @@ final class DummyCopyResult
/** @var array<int,int> */ /** @var array<int,int> */
public array $moduleMap = []; public array $moduleMap = [];
/** @var array<int,int> */
public array $contentMap = [];
/** @var array<int,int> */
public array $newsArchiveMap = [];
/** @var array<int,int> */
public array $calendarMap = [];
/** @var array<int,int> */
public array $newsItemMap = [];
/** @var array<int,int> */
public array $eventMap = [];
/** @var array<string> */ /** @var array<string> */
public array $notes = []; public array $notes = [];