WordPress-Postings in zufälliger Reihenfolge

Wenn ein Blog eine Menge Artikel hat, kann es passieren, dass die älteren Beiträge so tief im Archiv versunken liegen, dass der interessierte Leser die kaum noch zur Kenntnis nehmen kann.

Das ist gerade bei Artikeln, die eher zeitlos sind, schon ein richtiges Problem. Mein Blogprojekt „orga-dich“ hatte genau dieses Problem.

Zufällige Reihenfolge

WordPress ermittelt die Postings, die in der Blogroll, also in der Liste der Beiträge, angezeigt werden, über ein wp_query-Objekt. Davon gibt es erfahrungsgemäß mehrere innerhalb eines Blogs, beispielsweise zum Auslesen aller Menüseiten oder innerhalb bestimmter Seitenelemente, die ebenfalls Artikel oder Seiten enthalten sollen.

Aber es gibt auch eine „Master-Query“, die für die Hauptseite zuständig ist. Und eine Query kann auch Parameter haben, die wir übersteuern können…

Oha, functions.php, ick hör dir trapsen

Innerhalb eines WordPress-Monsters können wir uns mit sogenannten Hooks an verschiedenen Stellen der Verarbeitung „beteiligen“.

Und einer davon ist der „posts_orderby„-Hook: Über den können wir zusätzliche Parameter für die Sortierung an das Query-Objekt mitgeben – also frisch ran ans Werk.

Captain Hook

Einen filter zu einem hook zu schreiben ist banal:

add_filter( 'posts_orderby', 'zufaellige_postings' );

Damit haken wir uns in den „posts_orderby„-Hook ein und geben bekannt, dass die PHP-Function „zufaellige_postings“ ausgeführt werden soll, sobald dieser Hook/diese Action ausgeführt wird.

Diese Funktion sieht für unseren Bedarf auch noch recht übersichtlich aus:

function zufaellige_postings( $orderby ) {
   if( is_front_page() ) {
      $orderby = 'RAND';
      return $orderby;
   }
}

Das „is_front_page()“ sorgt dafür, dass wir nur die Query beackern, die für die Anzeige der Startseite mit der Beitragsliste zuständig ist. Wir wollen ja nicht sämtliche DatenbankPost-Queries zufällig gestalten 🙂

That’s it – oder?

Hm – im Prinzip war es das. Aber wir haben ein Problem. Startseiten enthalten ja nicht grundsätzlich alle Postings des Blogs. Sondern immer nur eine Auswahl, der Rest möchte über eine Seitenwahl ausgewählt werden, die im Fachjargon „Pagination“ heißt:

pagination
pagination

Also dieser Bereich mit den Zahlen (1 2 … 4) und Pfeilen. Damit navigiert der geneigte Leser zwischen den vielen hunderttausend Postings eines Blogs.

Wenn wir unsere Zufallsmethode jetzt live schalten, wird auch alles schön gewürfelt – leider aber auf jeder Seite von Neuem! Das kann ja nicht gewollt sein…

Schöne Lösung aus dem Netz der Netze

Und tatsächlich, Ehre wem Ehre gebührt, ein User mit dem klangvollen Namen „hlashbrooke“ (Hugh Lashbrooke) hat auf GitHub eine passende Lösung gefunden.

Er speichert eine zufällig generierte Seed-Zahl in einer Session-Variable und übergibt sie immer erneut an das Query-Objekt als „rand„-Parameter. Dadurch beginnt der Zufallsgenerator immer wieder beim selben Ausgangswert (= Seed) und generiert dieselben Zufallszahlen.

Wie wir wissen, generieren übliche Algorithmen keine ernsthaft zufälligen Zahlenfolgen – sie brauchen einen Ausgangspunkt (den sog. „Seed“) und wenn dieser gleich ist, wird die Reihenfolge der „zufälligen“ Zahlen auch immer gleich sein.

Auf diese Weise wird bei einem seitenweisen Abruf der Postings die einmal erstellte Reihenfolge beibehalten – und liefert entsprechend nicht für jede Seite neue, zufällige Postings sondern liefert die Postings in einer zufälligen Folge innerhalb der gesamten Query.

So sieht das Spaß dann aus:

session_start();

add_filter( 'posts_orderby', 'randomise_with_pagination' );
function randomise_with_pagination( $orderby ) {

	if( is_front_page() ) {

	  	// Reset seed on load of initial archive page
		if( ! get_query_var( 'paged' ) || get_query_var( 'paged' ) == 0 || get_query_var( 'paged' ) == 1 ) {
			if( isset( $_SESSION['seed'] ) ) {
				unset( $_SESSION['seed'] );
			}
		}
	
		// Get seed from session variable if it exists
		$seed = false;
		if( isset( $_SESSION['seed'] ) ) {
			$seed = $_SESSION['seed'];
		}
	
	    	// Set new seed if none exists
	    	if ( ! $seed ) {
	      		$seed = rand();
	      		$_SESSION['seed'] = $seed;
	    	}
	
	    	// Update ORDER BY clause to use seed
	    	$orderby = 'RAND(' . $seed . ')';
	}

	return $orderby;
}
?>

Leider geil – und funktioniert super! Und ist auch angemessen dokumentiert, danke liebe(r) Hugh Lashbrooke (da sind noch mehr coole Sachen zu finden).

So sieht es dann aus

Im Ergebnis kommt das schon ganz nice.

orga-dich postings
orga-dich postings