Gefilterte Liste: Anzahl Sätze anzeigen (x von Y)
YForm bietet die Möglichkeit, links neben der Tabelle ein Suchformular anzuzeigen. Das Ergebnis der Suche ist eine mehr oder weniger lange Liste. Jedoch gibt es keine Information zur Einordnung des Filterergebnisses, also wie viel Sätze herausgefiltert bzw. tatsächlich vorhanden sind. Aus Tabellenkalkulationen wie Excel kennt man beim Filtern die Rückmeldung "x von y Zeilen" - geht das nicht auch für gefilterte YForm-Tabellen? So zum Beispiel:
Der Trick: rex_i18n::addMsg
Nein, es gibt keinen Extension-Point, über den die Überschrift der Datentabelle vor der Ausgabe
gezielt angepasst werden kann. Das geht nur über einen kleinen Trick in der Sprachverwaltung
rex_i18n
und Verwendung eines EP kurz vor der Ausgabe.
Die Überschrift ("Datentabelle") wird über die Sprachdateien bereitgestellt. Die Texte lassen sich nicht nur in den .lang-Dateien setzen, sondern auch progammatisch per PHP:
rex_i18n::addMsg( 'key', 'value' );
Vorhandene Texte werden ersetzt, wenn 'key'
schon existiert. Das machen wir uns zunutze und
ändern kurz vor der Ausgabe der Tabelle den Text des Schlüssels 'yform_tabledata_overview'
in "Datentabelle (x von y)").
rex_i18n::addMsg( 'yform_tabledata_overview', "$Datentabelle ($lines von $total)" );
Basisversion (keine Sprachvarianten)
Der obige Code sowie die Ermittlung der Anzahl Datensätze in der Tabelle und die Anzahl der
Datensätze in der Anzeige werden im Extension-Point YFORM_DATA_LIST
platziert, der als letzter
YForm-EP vor der Tabellenausgabe angezogen wird.
\rex_extension::register('YFORM_DATA_LIST',
static function (\rex_extension_point $ep) {
// Anzahl Zeilen in der Query
/** @var \rex_yform_list $list */
$list = $ep->getSubject();
$lines = $list->getRows();
// Anzahl Zeilen in der Tabelle
$sql = \rex_sql::factory();
$sql->setTable($ep->getParams()['table']['table_name']);
$sql->select('count(*)');
$total = $sql->getValue('count(*)');
// Tabellentitel nur erweitern wenn gefiltert.
if ($lines < $total) {
\rex_i18n::addMsg('yform_tabledata_overview', "Datentabelle ($lines von $total)");
}
},
);
I.d.R. wird das Verfahren so ausreichen, wenn die REDAXO-Instanz mit nur einer Sprachversion arbeitet. Auch die eher seltene Ausgabe mehrerer Tabellen in einer Page bereitet keine Probleme.
Schwieriger wird es, wenn Mehrsprachigkeit gefordert ist.
Mehrsprachige Version
Wie man schon am Term "Datentabelle ($lines von $total)"
sieht, sind zwei Wörter relevant:
-
"Datentabelle": Der Text ist der Originaltext des .lang-Spracheintrags
'yform_tabledata_overview'
. Er kann abgerufen und statt "Datentabelle" in den Text eingebaut werden.$label = \rex_i18n::rawMsg( 'yform_tabledata_overview' ); \rex_i18n::addMsg( 'yform_tabledata_overview', "$label ($lines von $total)" );
-
"von": Für diesen Textteil ("von", "of", ...) gibt es noch keinen nutzbaren Eintrag. Somit bleibt als Lösung, das Wort zu neutralisieren ("x/y", sieht nicht so schön aus) oder ebenfalls via eigenem .lang-Eintrag sprachspezifisch bereitzustellen.
Da 'yform_tabledata_overview'
verändert wird, ergibt sich bei (seltenen) Pages mit mehreren
Tabellen ein zusätzliches Problem: Wenn zwei Listen gefiltert sind, wird
'yform_tabledata_overview'
mehrfach erweitert (z.B. "Datentabelle (x1 von y1) (x2 von y2)").
Plant man ohnehin eigene .lang-Einträge für "von", kann das auch genutzt werden, um den kompletten Titel bereitzustellen und die wiederholte Erweiterung zu vermeiden.
Dazu bedarf es .lang-Dateien im eigenen Projekt z.B. mit diesem Inhalt:
- de_de.lang:
yform_tabledata_overview_xy = Datentabelle ({0} von {1})
- en_gb.lang:
yform_tabledata_overview_xy = Data tables ({0} of {1})
- es_es.lang:
yform_tabledata_overview_xy = Tabla de datos ({0} de {1})
- pt_br.lang:
yform_tabledata_overview_xy = Tabelas de dados ({0} de {1})
- sv_se.lang:
yform_tabledata_overview_xy = Datatabell ({0} av {1})
Die Originaltexte zu yform_tabledata_overview
sind unter redaxo/src/addons/yform/plugins/manager/lang/*.lang
zu finden.
Der Code ändert sich nur geringfügig:
\rex_extension::register('YFORM_DATA_LIST',
static function (\rex_extension_point $ep) {
// Anzahl Zeilen in der Query
/** @var \rex_yform_list $list */
$list = $ep->getSubject();
$lines = $list->getRows();
// Anzahl Zeilen in der Tabelle
$sql = \rex_sql::factory();
$sql->setTable($ep->getParams()['table']['table_name']);
$sql->select('count(*)');
$total = $sql->getValue('count(*)');
// Tabellentitel nur erweitern wenn gefiltert.
if ($lines < $total) {
$yform_tabledata_overview = \rex_i18n::rawMsg('yform_tabledata_overview_xy', (string) $lines, (string) $total);
\rex_i18n::addMsg('yform_tabledata_overview', $yform_tabledata_overview);
}
},
);