For coders TYPO3 Tech Corner

FlexForm-Daten sortiert aus einem Select-Feld ausgeben

FlexForm-Daten sortiert aus einem Select-Feld ausgeben

Mit FlexForm kann man leicht Plugins mit Feldern zur Steuerung der Ausgabe erweitern. Ob das jetzt einfache Input-Felder, Checkboxen oder Kind-Elemente mittels IRRE sind, spielt keine Rolle. Alles, was per TCA-Konfiguration geht, geht mit FlexForms.

Wenn man aber eine Select-Box per selectMultipleSideBySide rendert, um dem Plugin eine Selektion von Datensätzen mitzugeben, wird der Wert dieses Feldes in ein String abgespeichert. Dieser String besteht aus kommaseparierten UIDs der ausgewählten Datensätze und zwar in der Reihenfolge, wie sie in dem Select-Feld sortiert wurden (zum Beispiel "4,2,1,3").
Diese Datensätze möchte man nun auch im Frontend genau in dieser Reihenfolge ausgelesen haben und stößt somit auf das Problem, dass keine geeignete Query in Extbase existiert, um dies zu bewerkstelligen.

TYPO3 bringt zum Glück für solche Probleme eine API für Doctrine DBAL mit, die es uns erlaubt mit dem QueryBuilder entsprechende Queries auf die Datenbank loszulassen. Klar könnte man auch den String in ein Array mittels explode teilen und dann für jede enthaltene UID eine einzelne findByUid-Abfrage erstellen, aber darunter würde die Performance leiden.

<?php declare(strict_types=1); namespace Vendor\Package\Domain\Repository; use Vendor\Package\Domain\Model\Item; use TYPO3\CMS\Core\Database\ConnectionPool; use TYPO3\CMS\Core\Database\Query\QueryBuilder; use TYPO3\CMS\Core\Utility\GeneralUtility; use TYPO3\CMS\Extbase\Persistence\Generic\Mapper\DataMapper; /** * ItemRepository */ class ItemRepository { private const TABLE = 'tx_package_domain_model_item'; /** * @var DataMapper */ protected DataMapper $dataMapper; /** * @var QueryBuilder|null */ protected ?QueryBuilder $queryBuilder = null; /** * Constructor */ public function __construct(DataMapper $dataMapper) { $this->dataMapper = $dataMapper; $this->queryBuilder = GeneralUtility::makeInstance(ConnectionPool::class)->getQueryBuilderForTable(self::TABLE); } /** * @param string $uidList * @return array */ public function findByUidList(string $uidList): array { $uids = GeneralUtility::intExplode(',', $uidList, true); if (count($uids) === 0) { return []; } $records = $this->queryBuilder ->select('*') ->from(self::TABLE) ->where($this->queryBuilder->expr()->in('uid', $uids)) ->add('orderBy', 'FIELD(uid,' . implode(',', $uids) . ')') ->execute() ->fetchAll(); return $this->dataMapper->map(Item::class, $records); } }

In der Funktion findByUidList wird eine relativ einfache Query um eine orderBy-Anweisung erweitert, in der die UIDs übergeben werden.

Nicht wundern, dass wir am Anfang der Funktion den String erst per explode auseinandernehmen, ihn dann aber für das orderBy per implode zu einem ähnlichen String zusammenfügen. Mit GeneralUtility::intExplode führen wir nicht nur ein explode aus, sondern casten die dort vorkommenden Werte direkt zu Integer, so dass wir sichergehen können, dass sich in dem String nur UIDs befinden. Mit dem dritten Parameter von intExplode werden sogar noch "leere" Werte direkt entfernt.

Kennst du das: Immer nur schnell schnell?

Wie wäre es einmal mit Zeit und Respekt für Codequalität? Arbeiten im Team? Automatisierte Tests?

Komm zu uns

SQL: Zeige alle Tabellen absteigend nach Größe sortiert

Ich brauche in letzter Zeit häufiger den SQL-Befehl, um herauszufinden, welche Tabellen in der TYPO3-Datenbank am größten sind. Ich habe das Snippet einmal veröffentlicht.

Zum Beitrag

TYPO3 12 mit CKEditor 5: Stile als Einfachauswahl

Wenn man im RTE in TYPO3 einen Link setzt, kann es sein, dass man zwischen verschiedenen Link-Klassen auswählen muss, um beispielsweise Buttons im Frontend zu erzeugen. Neu ist in TYPO3 12 dass man...

Zum Beitrag

Null-Safe Operator im TYPO3-Bereich

Spätestens mit dem Einzug von PHP8 kann es an vielen Stellen zu Problemen mit undefinierten Arrays oder Variablen im Allgemeinen kommen. Hier ein paar Beispiele und einfache Lösung dafür.

Zum Beitrag

Die ersten/letzten Zeilen einer (SQL)-Datei löschen

Zu den nachfolgenden Befehlen gibt es eigentlich nicht viel zu sagen. Manchmal kann es nützlich sein, die ersten (bzw. letzten) X Zeilen aus einer Datei zu löschen. Und wenn die Datei zu groß zum...

Zum Beitrag

b13/container: Kindelemente in der Bearbeitungsansicht hinzufügen und ändern

Anders als in gridelements kann man in der Extension Container von B13 die Kindelemente nicht verwalten, wenn man den Container in der Bearbeitungsansicht öffnet. Wie man das schnell selber einbauen...

Zum Beitrag

Menü-Vergleich: Zahlen, Zahlen, Zahlen

Zum Beitrag