PmWiki kann für Seitenlisten angepasste order=-Werte enthalten, die der Administrator in der /local/config.php vordefiniert. Zuerst müssen wir PmWiki mitteilen, welche Funktion aufgerufen werden soll für unseren selbstdefinierten order=-Parameter.
Als Beispiel nehmen wir eine Daten-Seite, die Seitentext-Variablen enthält, in denen Daten über Bücher gespeichert werden. Die Seite Daten-Gruppe.SeitenName enthält die kommaseparierten Werte $:Year (Jahr der Veröffentlichung, $:Work (Titel des Buches) und $:WAuthor (Autor des Buches). In manchen Fällen sollen die Daten für die Sortierreihenfolge für Gruppe.SeitenName von diesen korrespondierenden Daten-Seiten stammen.
Es gibt zwei Wege, die angepassten Kriterien für die Seitenlistenreihenfolge zu erzeugen.
Methode 1
Wenn die gewünschte Sortierreihenfolge $:Year,$:Work,$:WAuthor ist, benutzen wir 'yearworkwauthor' als unseren eigenen Parameter für die Sortierfunktion, so dass das Seitenlistenkriterium so aussieht:
(:pagelist order=yearworkwauthor:)
Das Array, das die order=-Parameter auf die angepassten die Sortierfunktionen abbildet, die dafür aufgerufen werden sollen, ist $PageListSortCmp:
$PageListSortCmp['yearworkwauthor'] = 'YearWorkWAuthor($x, $y)';
$PageListSortCmp ist ein Array von Seitenlistenfunktionen. Jede gelistete Funktion (nach dem =) erwartet zwei Parameter — jeder enthält den Seitennamen für Seiten, die sortiert werden sollen, dabei werden immer nur zwei Seiten miteinander verglichen. Das heißt, um einen Vergleich zwischen zwei Seiten in der Seitenliste durchzuführen (die durch $x und $y gegeben sind), rufe die Funktion YearWorkWAuthor() auf und übergib die Seitennamen als Argumente.
Die YearWorkWAuthor()-Funktion gibt einen Wert kleiner als Null zurück, wenn $x in der Liste vor $y erscheinen soll, größer als null, wenn $x nach $y erscheinen soll und null, wenn beide Namen 'gleichwertig' im Sinne der Sortierung sind.
Natürlich enthalten in diesem Szenario die Seiten, die durch $x und $y gegeben sind, nicht die Werte, nach denen wir sortieren wollten — jene Werte sind in den korrespondierenden "Daten-*"-Seiten von $x und $y —, sonst müssten wir die Sortierung nicht anpassen. Darum suchen wir nach den korrespondierenden "Daten-"-Seiten und vergleichen die Seitentext-Variablen aus diesen Seiten:
function YearWorkWAuthor($x, $y) { ## zuerst holen wir die "Daten-"-Version der Seitennamen $datax = 'Daten-' . PageVar($x, '$BaseName'); $datay = 'Daten-' . PageVar($y, '$BaseName'); ## vergleiche die $:Year-Werte $c = strcmp(PageVar($datax, '$:Year'), PageVar($datay, '$:Year')); if ($c != 0) return $c; ## vergleiche die $:Work-Werte $c = strcmp(PageVar($datax, '$:Work'), PageVar($datay, '$:Work')); if ($c != 0) return $c; ## vergleiche die $:WAuthor-Werte $c = strcmp(PageVar($datax, '$:WAuthor'), PageVar($datay, '$:WAuthor')); return $c; }
In der obigen Funktion finden die ersten beiden Zeilen den Namen der korrespondierenden "Daten-*"-Seiten heraus und speichern sie in $datax und $datay. Die nächsten zwei Zeilen holen die $:Year-Seitentext-Variablen aus $datax und $datay und geben einen positiven oder negativen Wert zurück, wenn sie verschieden sind. "strcmp" ist eine eingebaute PHP-Funktion ("string compare", Stringvergleich), und die gibt einen numerischen Wert zurück, der aussagt, wie verschieden zwei Datenteile (Text) sind. Wenn sie gleich sind (d. h. $c == 0), fallen wir durch zum Test der $:Work-Seitentext-Variablen mit der gleiche Logik, und wenn auch die gleich sind, testen wir die $:WAuthor-Seitentext-Variablen und geben das Ergebnis zurück.
So wie die Funktion geschrieben ist, gibt es ein kleines bißchen Overhead durch den wiederholten Aufruf von PageVar(), den wir vermeiden könnten, wenn Geschwindigkeit eine Rolle spielt, aber der obige Kode illustriert das Basiskonzept hinter der angepassten Sortierung.
Methode 2
Um den Wikibenutzern mehr Flexibilität zu geben, ist eine andere Annäherung, eine generische DataCompare()-Funktion für den Vergleich von Seitentext-Variablen in "Daten-*"-Seiten zu erstellen, und dann getrennte "year"-, "work"-, und "wauthor"-Optionen für den "order="-Parameter zu definieren, die ein passendes Argument an DataCompare() übergeben.
function DataCompare($x, $y, $var) { ## hole zuerst die "Daten-"-Version der Seitennamen, $datax = 'Daten-' . PageVar($x, '$BaseName'); $datay = 'Daten-' . PageVar($y, '$BaseName'); ## führe dann den geforderten Vergleich durch. $c = strcmp(PageVar($datax, $var), PageVar($datay, $var)); return $c; } $PageListSortCmp['year'] = 'DataCompare($x, $y, "$:Year")'; $PageListSortCmp['work'] = 'DataCompare($x, $y, "$:Work")'; $PageListSortCmp['wauthor'] = 'DataCompare($x, $y, "$:WAuthor")';
Dann kann man jede Menge an Seitenlisten-"order="-Kombinationen wählen wie:
order=year # sortieren nach $:Year aus den "Daten-*"-Seiten order=year,work # sortieren nach $:Year, dann nach $:Work order=year,-wauthor # sortieren nach $:Year, dann rückwärts nach $:WAuthor order=wauthor # sortieren nur nach $:WAuthor
Das ist näher dran an dem, was ein Autor erwarten würde, weil andere Sortierkriterien vom Benutzer geformt und verschachtelt werden können. Wenn Sie Ihre Benutzer in die Lage versetzen wollen, die Sortierreihenfolge anzupassen, ohne dass Sie eine Neuprogrammierung in der config.php wegen neuer Anforderungen durchführen müssen, ist dies möglicherweise die bessere Methode.
Alternativer Weg
Nehmen wir an, dass Seiten-Variablen der generelle PmWiki-Aufhänger sind (oder sein sollten), um angepasste Dinge mit den Attributen und Eigenschaften von Seiten zu machen.
Tatsächlich ist hier ein *weiterer* Weg, das Sortieren/Gruppen/Anzeigen-Problem zu behandeln, indem angepasste Seiten-Variablen definiert werden, die genau das Erwünschte haben, ohne dass irgendein benutzerdefiniertes Sortierfeature für (:pagelist:)
definiert werden müsste.
Lassen Sie uns $Year-, $Work- und $WAuthor-Seiten-Variablen für jede Seite so definieren, dass die Werte von $Year, $Work, und $WAuthor für jede Gruppe.XYZ-Seite immer die $:Year-, $:Work-, und $:WAuthor-Seitentext-Variablen von Gruppe.XYZs korrespondierender "Daten-*"-Seite sind. Mit anderen Worten, {$Year}
für jede Seite wird immer so funktionieren, als hätte man {Daten-{$BaseName}$:Year}
angegeben.
Hier folgen die Definitionen:
$FmtPV['$Year'] = "PageTextVar('Daten-'.MakeBaseName(\$pn), 'Year')"; $FmtPV['$Work'] = "PageTextVar('Daten-'.MakeBaseName(\$pn), 'Work')"; $FmtPV['$WAuthor'] = "PageTextVar('Daten-'.MakeBaseName(\$pn), 'WAuthor')";
Tja, was haben wir jetzt davon? Nun, der Wert von
{$Year}
ist immer der Wert von {$:Year}
der korrespondierenden
"Daten-*"-Seite. Dadurch gibt {Gruppe.Steinbeck$Year}
immer den Wert von
{Daten-Gruppe.Steinbeck$Year}
zurück. Darüber hinaus funktioniert das sogar, wenn
die aktuelle Seite im Daten-Bereich ist — d. h., der Wert von {Daten-Gruppe.Steinbeck$Year}
ist der gleiche wie {Daten-Gruppe.Steinbeck$:Year}
. (Die erste ist eine Seiten-Variable, die zweite ist eine Seitentext-Variable.)
Was wir damit getan haben ist die Verlagerung aller Ergebnisse der den Daten-Seiten entsprechenden Seiten aus den Seitenlistenvorlagen in einige relativ einfache Definitionen von Seiten-Variablen. Damit sehen unsere Seitenlisten-Direktiven so aus:
(:pagelist group=Gruppe order=$Year,$Work,$WAuthor:) (:pagelist group=Daten-Gruppe order=$Year,$Work,$WAuthor:)
Die Angabe von order=$Year,$Work,$WAuthor (Seiten-Variablen) bedeutet, dass wir die Seitenlisten auf der Basis der $:Year-, $:Work-, und $:WAuthor-Seitentext-Variablen der korrespondierenden Seiten in der Gruppe Daten-Gruppe sortieren.
Beachten Sie, dass wir uns auch keine Sorgen darum machen müssen, ob die Seitenliste durch die Gruppe selbst oder die entsprechende "Daten-*"-Gruppe läuft, da unsere angepassten Seiten-Variablen immer die Seitennamen auf die "Daten-*"-Form der Gruppen abbildet.
Das vereinfacht die Seitenlisten-Vorlage, denn wir können nun einfach schreiben:
(:if ! equal {<$Year} {=$Year}:) !! {=$Year} (:if:)
Nochmal, die '$Year'-Seiten-Variable achtet selbst auf die Details
bei der Beschaffung von $:Year aus der korrekten Daten-{$BaseName}
-Seite,
anstatt zu versuchen, die Auswertung durch das Markup vorzunehmen.
Siehe auch
- Cookbook:CustomPagelistSortOrderFunctions - (angepasste Funktionen zum Erstellen von angepassten Sortierreihenfolgen bei Seitenlisten, englisch)
Originalseite auf PmWikiDe.CustomPagelistSortOrder - Rückverweise
Zuletzt geändert:
PmWikiDe.CustomPagelistSortOrder am 10.09.2011